FEATURE: Add experimental tracking of 'real browser' pageviews (#26647)
Our 'page_view_crawler' / 'page_view_anon' metrics are based purely on the User Agent sent by clients. This means that 'badly behaved' bots which are imitating real user agents are counted towards 'anon' page views. This commit introduces a new method of tracking visitors. When an initial HTML request is made, we assume it is a 'non-browser' request (i.e. a bot). Then, once the JS application has booted, we notify the server to count it as a 'browser' request. This reliance on a JavaScript-capable browser matches up more closely to dedicated analytics systems like Google Analytics. Existing data collection and graphs are unchanged. Data collected via the new technique is available in a new 'experimental' report.
This commit is contained in:
parent
52e8d57293
commit
2f2da72747
|
@ -7,6 +7,12 @@ import getURL from "discourse-common/lib/get-url";
|
||||||
|
|
||||||
const LONG_POLL_AFTER_UNSEEN_TIME = 1200000; // 20 minutes
|
const LONG_POLL_AFTER_UNSEEN_TIME = 1200000; // 20 minutes
|
||||||
|
|
||||||
|
let _sendDeferredPageview = false;
|
||||||
|
|
||||||
|
export function sendDeferredPageview() {
|
||||||
|
_sendDeferredPageview = true;
|
||||||
|
}
|
||||||
|
|
||||||
function mbAjax(messageBus, opts) {
|
function mbAjax(messageBus, opts) {
|
||||||
opts.headers ||= {};
|
opts.headers ||= {};
|
||||||
|
|
||||||
|
@ -22,6 +28,11 @@ function mbAjax(messageBus, opts) {
|
||||||
opts.headers["Discourse-Present"] = "true";
|
opts.headers["Discourse-Present"] = "true";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (_sendDeferredPageview) {
|
||||||
|
opts.headers["Discourse-Deferred-Track-View"] = "true";
|
||||||
|
_sendDeferredPageview = false;
|
||||||
|
}
|
||||||
|
|
||||||
const oldComplete = opts.complete;
|
const oldComplete = opts.complete;
|
||||||
opts.complete = function (xhr, stat) {
|
opts.complete = function (xhr, stat) {
|
||||||
handleLogoff(xhr);
|
handleLogoff(xhr);
|
||||||
|
|
|
@ -4,11 +4,20 @@ import {
|
||||||
resetPageTracking,
|
resetPageTracking,
|
||||||
startPageTracking,
|
startPageTracking,
|
||||||
} from "discourse/lib/page-tracker";
|
} from "discourse/lib/page-tracker";
|
||||||
|
import { sendDeferredPageview } from "./message-bus";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
after: "inject-objects",
|
after: "inject-objects",
|
||||||
|
before: "message-bus",
|
||||||
|
|
||||||
initialize(owner) {
|
initialize(owner) {
|
||||||
|
const isErrorPage =
|
||||||
|
document.querySelector("meta#discourse-error")?.dataset.discourseError ===
|
||||||
|
"true";
|
||||||
|
if (!isErrorPage) {
|
||||||
|
sendDeferredPageview();
|
||||||
|
}
|
||||||
|
|
||||||
// Tell our AJAX system to track a page transition
|
// Tell our AJAX system to track a page transition
|
||||||
// eslint-disable-next-line ember/no-private-routing-service
|
// eslint-disable-next-line ember/no-private-routing-service
|
||||||
const router = owner.lookup("router:main");
|
const router = owner.lookup("router:main");
|
||||||
|
|
|
@ -0,0 +1,17 @@
|
||||||
|
document.addEventListener("DOMContentLoaded", function () {
|
||||||
|
const isErrorPage =
|
||||||
|
document.querySelector("meta#discourse-error")?.dataset.discourseError ===
|
||||||
|
"true";
|
||||||
|
|
||||||
|
if (!isErrorPage) {
|
||||||
|
const root =
|
||||||
|
document.querySelector("meta[name=discourse-base-uri]")?.content || "";
|
||||||
|
|
||||||
|
fetch(`${root}/pageview`, {
|
||||||
|
method: "POST",
|
||||||
|
headers: {
|
||||||
|
"Discourse-Deferred-Track-View": "true",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
|
@ -0,0 +1,13 @@
|
||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
class PageviewController < ApplicationController
|
||||||
|
skip_before_action :check_xhr,
|
||||||
|
:redirect_to_login_if_required,
|
||||||
|
:preload_json,
|
||||||
|
:verify_authenticity_token
|
||||||
|
|
||||||
|
def index
|
||||||
|
# pageview tracking is handled by middleware
|
||||||
|
render plain: "ok"
|
||||||
|
end
|
||||||
|
end
|
|
@ -14,6 +14,10 @@ class ApplicationRequest < ActiveRecord::Base
|
||||||
page_view_anon_mobile
|
page_view_anon_mobile
|
||||||
api
|
api
|
||||||
user_api
|
user_api
|
||||||
|
page_view_anon_browser
|
||||||
|
page_view_anon_browser_mobile
|
||||||
|
page_view_logged_in_browser
|
||||||
|
page_view_logged_in_browser_mobile
|
||||||
]
|
]
|
||||||
|
|
||||||
include CachedCounting
|
include CachedCounting
|
||||||
|
|
|
@ -0,0 +1,75 @@
|
||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
module Reports::ConsolidatedPageViewsBrowserDetection
|
||||||
|
extend ActiveSupport::Concern
|
||||||
|
|
||||||
|
class_methods do
|
||||||
|
def report_consolidated_page_views_browser_detection(report)
|
||||||
|
report.modes = [:stacked_chart]
|
||||||
|
|
||||||
|
data =
|
||||||
|
DB.query(
|
||||||
|
<<~SQL,
|
||||||
|
SELECT
|
||||||
|
date,
|
||||||
|
SUM(CASE WHEN req_type = :page_view_logged_in_browser THEN count ELSE 0 END) AS page_view_logged_in_browser,
|
||||||
|
SUM(CASE WHEN req_type = :page_view_anon_browser THEN count ELSE 0 END) AS page_view_anon_browser,
|
||||||
|
SUM(CASE WHEN req_type = :page_view_crawler THEN count ELSE 0 END) AS page_view_crawler,
|
||||||
|
SUM(
|
||||||
|
CASE WHEN req_type = :page_view_anon THEN count
|
||||||
|
WHEN req_type = :page_view_logged_in THEN count
|
||||||
|
WHEN req_type = :page_view_anon_browser THEN -count
|
||||||
|
WHEN req_type = :page_view_logged_in_browser THEN -count
|
||||||
|
ELSE 0
|
||||||
|
END
|
||||||
|
) AS page_view_other
|
||||||
|
FROM application_requests
|
||||||
|
WHERE date >= :start_date AND date <= :end_date
|
||||||
|
GROUP BY date
|
||||||
|
ORDER BY date ASC
|
||||||
|
SQL
|
||||||
|
start_date: report.start_date,
|
||||||
|
end_date: report.end_date,
|
||||||
|
page_view_anon: ApplicationRequest.req_types[:page_view_anon],
|
||||||
|
page_view_crawler: ApplicationRequest.req_types[:page_view_crawler],
|
||||||
|
page_view_logged_in: ApplicationRequest.req_types[:page_view_logged_in],
|
||||||
|
page_view_anon_browser: ApplicationRequest.req_types[:page_view_anon_browser],
|
||||||
|
page_view_logged_in_browser: ApplicationRequest.req_types[:page_view_logged_in_browser],
|
||||||
|
)
|
||||||
|
|
||||||
|
report.data = [
|
||||||
|
{
|
||||||
|
req: "page_view_logged_in_browser",
|
||||||
|
label:
|
||||||
|
I18n.t(
|
||||||
|
"reports.consolidated_page_views_browser_detection.xaxis.page_view_logged_in_browser",
|
||||||
|
),
|
||||||
|
color: report.colors[0],
|
||||||
|
data: data.map { |row| { x: row.date, y: row.page_view_logged_in_browser } },
|
||||||
|
},
|
||||||
|
{
|
||||||
|
req: "page_view_anon_browser",
|
||||||
|
label:
|
||||||
|
I18n.t(
|
||||||
|
"reports.consolidated_page_views_browser_detection.xaxis.page_view_anon_browser",
|
||||||
|
),
|
||||||
|
color: report.colors[1],
|
||||||
|
data: data.map { |row| { x: row.date, y: row.page_view_anon_browser } },
|
||||||
|
},
|
||||||
|
{
|
||||||
|
req: "page_view_crawler",
|
||||||
|
label:
|
||||||
|
I18n.t("reports.consolidated_page_views_browser_detection.xaxis.page_view_crawler"),
|
||||||
|
color: report.colors[2],
|
||||||
|
data: data.map { |row| { x: row.date, y: row.page_view_crawler } },
|
||||||
|
},
|
||||||
|
{
|
||||||
|
req: "page_view_other",
|
||||||
|
label: I18n.t("reports.consolidated_page_views_browser_detection.xaxis.page_view_other"),
|
||||||
|
color: report.colors[2],
|
||||||
|
data: data.map { |row| { x: row.date, y: row.page_view_other } },
|
||||||
|
},
|
||||||
|
]
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
|
@ -49,6 +49,7 @@ class Report
|
||||||
include Reports::UserFlaggingRatio
|
include Reports::UserFlaggingRatio
|
||||||
include Reports::TrustLevelGrowth
|
include Reports::TrustLevelGrowth
|
||||||
include Reports::ConsolidatedPageViews
|
include Reports::ConsolidatedPageViews
|
||||||
|
include Reports::ConsolidatedPageViewsBrowserDetection
|
||||||
include Reports::ConsolidatedApiRequests
|
include Reports::ConsolidatedApiRequests
|
||||||
include Reports::Visits
|
include Reports::Visits
|
||||||
include Reports::TimeToFirstResponse
|
include Reports::TimeToFirstResponse
|
||||||
|
|
|
@ -1,5 +1,13 @@
|
||||||
<% content_for :title do %><%= @page_title %> - <%= SiteSetting.title %><% end %>
|
<% content_for :title do %><%= @page_title %> - <%= SiteSetting.title %><% end %>
|
||||||
|
|
||||||
|
<% content_for :head do %>
|
||||||
|
<meta id="discourse-error" data-discourse-error="true">
|
||||||
|
<% end %>
|
||||||
|
|
||||||
|
<% content_for :no_ember_head do %>
|
||||||
|
<meta id="discourse-error" data-discourse-error="true">
|
||||||
|
<% end %>
|
||||||
|
|
||||||
<div class="page-not-found">
|
<div class="page-not-found">
|
||||||
<h1 class="title"><%= @title %></h1>
|
<h1 class="title"><%= @title %></h1>
|
||||||
|
|
||||||
|
|
|
@ -17,6 +17,8 @@
|
||||||
<%- if allow_plugins? %>
|
<%- if allow_plugins? %>
|
||||||
<%= build_plugin_html 'server:before-head-close' %>
|
<%= build_plugin_html 'server:before-head-close' %>
|
||||||
<%- end -%>
|
<%- end -%>
|
||||||
|
|
||||||
|
<%= preload_script "pageview" %>
|
||||||
</head>
|
</head>
|
||||||
<body class="no-ember <%= @custom_body_class %>">
|
<body class="no-ember <%= @custom_body_class %>">
|
||||||
<%- if allow_plugins? %>
|
<%- if allow_plugins? %>
|
||||||
|
|
|
@ -8,6 +8,7 @@
|
||||||
<%= render_google_tag_manager_head_code %>
|
<%= render_google_tag_manager_head_code %>
|
||||||
<%= render_google_universal_analytics_code %>
|
<%= render_google_universal_analytics_code %>
|
||||||
<%= preload_script 'publish' %>
|
<%= preload_script 'publish' %>
|
||||||
|
<%= preload_script 'pageview' %>
|
||||||
</head>
|
</head>
|
||||||
<body class="<%= @body_classes.to_a.join(' ') %>">
|
<body class="<%= @body_classes.to_a.join(' ') %>">
|
||||||
<%= theme_lookup("header") %>
|
<%= theme_lookup("header") %>
|
||||||
|
|
|
@ -31,7 +31,7 @@ def setup_message_bus_env(env)
|
||||||
"Access-Control-Allow-Origin" => Discourse.base_url_no_prefix,
|
"Access-Control-Allow-Origin" => Discourse.base_url_no_prefix,
|
||||||
"Access-Control-Allow-Methods" => "GET, POST",
|
"Access-Control-Allow-Methods" => "GET, POST",
|
||||||
"Access-Control-Allow-Headers" =>
|
"Access-Control-Allow-Headers" =>
|
||||||
"X-SILENCE-LOGGER, X-Shared-Session-Key, Dont-Chunk, Discourse-Present",
|
"X-SILENCE-LOGGER, X-Shared-Session-Key, Dont-Chunk, Discourse-Present, Discourse-Deferred-Track-View",
|
||||||
"Access-Control-Max-Age" => "7200",
|
"Access-Control-Max-Age" => "7200",
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1338,6 +1338,15 @@ en:
|
||||||
user_api: "User API"
|
user_api: "User API"
|
||||||
yaxis: "Day"
|
yaxis: "Day"
|
||||||
description: "API requests for regular API keys and user API keys."
|
description: "API requests for regular API keys and user API keys."
|
||||||
|
consolidated_page_views_browser_detection:
|
||||||
|
title: "Consolidated Pageviews with Browser Detection (Experimental)"
|
||||||
|
xaxis:
|
||||||
|
page_view_anon_browser: "Anonymous Browser"
|
||||||
|
page_view_logged_in_browser: "Logged In Browser"
|
||||||
|
page_view_crawler: "Known Crawler"
|
||||||
|
page_view_other: "Other pageviews"
|
||||||
|
yaxis: "Day"
|
||||||
|
description: "Pageviews for logged in users, anonymous users, known crawlers and other. This experimental report ensures logged-in/anon requests are coming from real browsers before counting them."
|
||||||
dau_by_mau:
|
dau_by_mau:
|
||||||
title: "DAU/MAU"
|
title: "DAU/MAU"
|
||||||
xaxis: "Day"
|
xaxis: "Day"
|
||||||
|
@ -1497,6 +1506,16 @@ en:
|
||||||
xaxis: "Day"
|
xaxis: "Day"
|
||||||
yaxis: "Mobile Anon Pageviews"
|
yaxis: "Mobile Anon Pageviews"
|
||||||
description: "Number of new pageviews from visitors on a mobile device who are not logged in."
|
description: "Number of new pageviews from visitors on a mobile device who are not logged in."
|
||||||
|
page_view_anon_browser_reqs:
|
||||||
|
title: "Anonymous Browser Pageviews"
|
||||||
|
xaxis: "Day"
|
||||||
|
yaxis: "Anonymous Browser Pageviews"
|
||||||
|
description: "Number of pageviews by anonymous visitors using real browsers."
|
||||||
|
page_view_logged_in_browser_reqs:
|
||||||
|
title: "Logged In Browser Pageviews"
|
||||||
|
xaxis: "Day"
|
||||||
|
yaxis: "Logged In Browser Pageviews"
|
||||||
|
description: "Number of pageviews by logged-in visitors using real browsers."
|
||||||
http_background_reqs:
|
http_background_reqs:
|
||||||
title: "Background"
|
title: "Background"
|
||||||
xaxis: "Day"
|
xaxis: "Day"
|
||||||
|
|
|
@ -1629,6 +1629,8 @@ Discourse::Application.routes.draw do
|
||||||
resources :sidebar_sections, only: %i[index create update destroy]
|
resources :sidebar_sections, only: %i[index create update destroy]
|
||||||
put "/sidebar_sections/reset/:id" => "sidebar_sections#reset"
|
put "/sidebar_sections/reset/:id" => "sidebar_sections#reset"
|
||||||
|
|
||||||
|
post "/pageview" => "pageview#index"
|
||||||
|
|
||||||
get "*url", to: "permalinks#show", constraints: PermalinkConstraint.new
|
get "*url", to: "permalinks#show", constraints: PermalinkConstraint.new
|
||||||
|
|
||||||
get "/form-templates/:id" => "form_templates#show"
|
get "/form-templates/:id" => "form_templates#show"
|
||||||
|
|
|
@ -74,9 +74,31 @@ class Middleware::RequestTracker
|
||||||
elsif data[:has_auth_cookie]
|
elsif data[:has_auth_cookie]
|
||||||
ApplicationRequest.increment!(:page_view_logged_in)
|
ApplicationRequest.increment!(:page_view_logged_in)
|
||||||
ApplicationRequest.increment!(:page_view_logged_in_mobile) if data[:is_mobile]
|
ApplicationRequest.increment!(:page_view_logged_in_mobile) if data[:is_mobile]
|
||||||
|
if data[:explicit_track_view]
|
||||||
|
# Must be a browser if it had this header from our ajax implementation
|
||||||
|
ApplicationRequest.increment!(:page_view_logged_in_browser)
|
||||||
|
ApplicationRequest.increment!(:page_view_logged_in_browser_mobile) if data[:is_mobile]
|
||||||
|
end
|
||||||
elsif !SiteSetting.login_required
|
elsif !SiteSetting.login_required
|
||||||
ApplicationRequest.increment!(:page_view_anon)
|
ApplicationRequest.increment!(:page_view_anon)
|
||||||
ApplicationRequest.increment!(:page_view_anon_mobile) if data[:is_mobile]
|
ApplicationRequest.increment!(:page_view_anon_mobile) if data[:is_mobile]
|
||||||
|
if data[:explicit_track_view]
|
||||||
|
# Must be a browser if it had this header from our ajax implementation
|
||||||
|
ApplicationRequest.increment!(:page_view_anon_browser)
|
||||||
|
ApplicationRequest.increment!(:page_view_anon_browser_mobile) if data[:is_mobile]
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
# Message-bus requests may include this 'deferred track' header which we use to detect
|
||||||
|
# 'real browser' views.
|
||||||
|
if data[:deferred_track] && !data[:is_crawler]
|
||||||
|
if data[:has_auth_cookie]
|
||||||
|
ApplicationRequest.increment!(:page_view_logged_in_browser)
|
||||||
|
ApplicationRequest.increment!(:page_view_logged_in_browser_mobile) if data[:is_mobile]
|
||||||
|
elsif !SiteSetting.login_required
|
||||||
|
ApplicationRequest.increment!(:page_view_anon_browser)
|
||||||
|
ApplicationRequest.increment!(:page_view_anon_browser_mobile) if data[:is_mobile]
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -103,12 +125,20 @@ class Middleware::RequestTracker
|
||||||
request ||= Rack::Request.new(env)
|
request ||= Rack::Request.new(env)
|
||||||
helper = Middleware::AnonymousCache::Helper.new(env, request)
|
helper = Middleware::AnonymousCache::Helper.new(env, request)
|
||||||
|
|
||||||
|
# Value of the discourse-track-view request header
|
||||||
env_track_view = env["HTTP_DISCOURSE_TRACK_VIEW"]
|
env_track_view = env["HTTP_DISCOURSE_TRACK_VIEW"]
|
||||||
track_view = status == 200
|
|
||||||
track_view &&= env_track_view != "0" && env_track_view != "false"
|
# Was the discourse-track-view request header set to true? Likely
|
||||||
track_view &&=
|
# set by our ajax library to indicate a page view.
|
||||||
env_track_view || (request.get? && !request.xhr? && headers["Content-Type"] =~ %r{text/html})
|
explicit_track_view = status == 200 && %w[1 true].include?(env_track_view)
|
||||||
track_view = !!track_view
|
|
||||||
|
# An HTML response to a GET request is tracked implicitly
|
||||||
|
implicit_track_view =
|
||||||
|
status == 200 && !%w[0 false].include?(env_track_view) && request.get? && !request.xhr? &&
|
||||||
|
headers["Content-Type"] =~ %r{text/html}
|
||||||
|
|
||||||
|
track_view = !!(explicit_track_view || implicit_track_view)
|
||||||
|
|
||||||
has_auth_cookie = Auth::DefaultCurrentUserProvider.find_v0_auth_cookie(request).present?
|
has_auth_cookie = Auth::DefaultCurrentUserProvider.find_v0_auth_cookie(request).present?
|
||||||
has_auth_cookie ||= Auth::DefaultCurrentUserProvider.find_v1_auth_cookie(env).present?
|
has_auth_cookie ||= Auth::DefaultCurrentUserProvider.find_v1_auth_cookie(env).present?
|
||||||
|
|
||||||
|
@ -118,6 +148,10 @@ class Middleware::RequestTracker
|
||||||
is_message_bus = request.path.start_with?("#{Discourse.base_path}/message-bus/")
|
is_message_bus = request.path.start_with?("#{Discourse.base_path}/message-bus/")
|
||||||
is_topic_timings = request.path.start_with?("#{Discourse.base_path}/topics/timings")
|
is_topic_timings = request.path.start_with?("#{Discourse.base_path}/topics/timings")
|
||||||
|
|
||||||
|
# This header is sent on a follow-up request after a real browser loads up a page
|
||||||
|
# see `scripts/pageview.js` and `instance-initializers/page-tracking.js`
|
||||||
|
has_deferred_track_header = %w[1 true].include?(env["HTTP_DISCOURSE_DEFERRED_TRACK_VIEW"])
|
||||||
|
|
||||||
h = {
|
h = {
|
||||||
status: status,
|
status: status,
|
||||||
is_crawler: helper.is_crawler?,
|
is_crawler: helper.is_crawler?,
|
||||||
|
@ -129,6 +163,8 @@ class Middleware::RequestTracker
|
||||||
track_view: track_view,
|
track_view: track_view,
|
||||||
timing: timing,
|
timing: timing,
|
||||||
queue_seconds: env["REQUEST_QUEUE_SECONDS"],
|
queue_seconds: env["REQUEST_QUEUE_SECONDS"],
|
||||||
|
explicit_track_view: explicit_track_view,
|
||||||
|
deferred_track: has_deferred_track_header,
|
||||||
}
|
}
|
||||||
|
|
||||||
if h[:is_background]
|
if h[:is_background]
|
||||||
|
@ -166,7 +202,8 @@ class Middleware::RequestTracker
|
||||||
data =
|
data =
|
||||||
begin
|
begin
|
||||||
self.class.get_data(env, result, info, request)
|
self.class.get_data(env, result, info, request)
|
||||||
rescue StandardError
|
rescue StandardError => e
|
||||||
|
Rails.logger.warn("RequestTracker.get_data failed: #{e}")
|
||||||
nil
|
nil
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
|
@ -66,6 +66,7 @@ RSpec.describe Middleware::RequestTracker do
|
||||||
CachedCounting.flush
|
CachedCounting.flush
|
||||||
|
|
||||||
expect(ApplicationRequest.page_view_anon.first.count).to eq(2)
|
expect(ApplicationRequest.page_view_anon.first.count).to eq(2)
|
||||||
|
expect(ApplicationRequest.page_view_anon_browser.first.count).to eq(2)
|
||||||
end
|
end
|
||||||
|
|
||||||
it "can log requests correctly" do
|
it "can log requests correctly" do
|
||||||
|
@ -119,6 +120,23 @@ RSpec.describe Middleware::RequestTracker do
|
||||||
expect(ApplicationRequest.page_view_anon_mobile.first.count).to eq(1)
|
expect(ApplicationRequest.page_view_anon_mobile.first.count).to eq(1)
|
||||||
|
|
||||||
expect(ApplicationRequest.page_view_crawler.first.count).to eq(1)
|
expect(ApplicationRequest.page_view_crawler.first.count).to eq(1)
|
||||||
|
|
||||||
|
expect(ApplicationRequest.page_view_anon_browser.first.count).to eq(1)
|
||||||
|
end
|
||||||
|
|
||||||
|
it "logs deferred pageviews correctly" do
|
||||||
|
data =
|
||||||
|
Middleware::RequestTracker.get_data(
|
||||||
|
env(:path => "/message-bus/abcde/poll", "HTTP_DISCOURSE_DEFERRED_TRACK_VIEW" => "1"),
|
||||||
|
["200", { "Content-Type" => "text/html" }],
|
||||||
|
0.1,
|
||||||
|
)
|
||||||
|
Middleware::RequestTracker.log_request(data)
|
||||||
|
|
||||||
|
expect(data[:deferred_track]).to eq(true)
|
||||||
|
CachedCounting.flush
|
||||||
|
|
||||||
|
expect(ApplicationRequest.page_view_anon_browser.first.count).to eq(1)
|
||||||
end
|
end
|
||||||
|
|
||||||
it "logs API requests correctly" do
|
it "logs API requests correctly" do
|
||||||
|
|
|
@ -0,0 +1,188 @@
|
||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
describe "request tracking", type: :system do
|
||||||
|
before do
|
||||||
|
ApplicationRequest.enable
|
||||||
|
CachedCounting.reset
|
||||||
|
CachedCounting.enable
|
||||||
|
end
|
||||||
|
|
||||||
|
after do
|
||||||
|
CachedCounting.reset
|
||||||
|
ApplicationRequest.disable
|
||||||
|
CachedCounting.disable
|
||||||
|
end
|
||||||
|
|
||||||
|
it "tracks an anonymous visit correctly" do
|
||||||
|
visit "/"
|
||||||
|
|
||||||
|
try_until_success do
|
||||||
|
CachedCounting.flush
|
||||||
|
expect(ApplicationRequest.stats).to include(
|
||||||
|
"page_view_anon_total" => 1,
|
||||||
|
"page_view_anon_browser_total" => 1,
|
||||||
|
"page_view_logged_in_total" => 0,
|
||||||
|
"page_view_crawler_total" => 0,
|
||||||
|
)
|
||||||
|
end
|
||||||
|
|
||||||
|
find(".nav-item_categories a").click
|
||||||
|
|
||||||
|
try_until_success do
|
||||||
|
CachedCounting.flush
|
||||||
|
expect(ApplicationRequest.stats).to include(
|
||||||
|
"page_view_anon_total" => 2,
|
||||||
|
"page_view_anon_browser_total" => 2,
|
||||||
|
"page_view_logged_in_total" => 0,
|
||||||
|
"page_view_crawler_total" => 0,
|
||||||
|
)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
it "tracks a crawler visit correctly" do
|
||||||
|
# Can't change Selenium's user agent... so change site settings to make Discourse detect chrome as a crawler
|
||||||
|
SiteSetting.crawler_user_agents += "|chrome"
|
||||||
|
|
||||||
|
visit "/"
|
||||||
|
|
||||||
|
try_until_success do
|
||||||
|
CachedCounting.flush
|
||||||
|
expect(ApplicationRequest.stats).to include(
|
||||||
|
"page_view_anon_total" => 0,
|
||||||
|
"page_view_anon_browser_total" => 0,
|
||||||
|
"page_view_logged_in_total" => 0,
|
||||||
|
"page_view_crawler_total" => 1,
|
||||||
|
)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
it "tracks a logged-in session correctly" do
|
||||||
|
sign_in Fabricate(:user)
|
||||||
|
|
||||||
|
visit "/"
|
||||||
|
|
||||||
|
try_until_success do
|
||||||
|
CachedCounting.flush
|
||||||
|
expect(ApplicationRequest.stats).to include(
|
||||||
|
"page_view_anon_total" => 0,
|
||||||
|
"page_view_anon_browser_total" => 0,
|
||||||
|
"page_view_logged_in_total" => 1,
|
||||||
|
"page_view_crawler_total" => 0,
|
||||||
|
"page_view_logged_in_browser_total" => 1,
|
||||||
|
)
|
||||||
|
end
|
||||||
|
|
||||||
|
find(".nav-item_categories a").click
|
||||||
|
|
||||||
|
try_until_success do
|
||||||
|
CachedCounting.flush
|
||||||
|
expect(ApplicationRequest.stats).to include(
|
||||||
|
"page_view_anon_total" => 0,
|
||||||
|
"page_view_anon_browser_total" => 0,
|
||||||
|
"page_view_logged_in_total" => 2,
|
||||||
|
"page_view_crawler_total" => 0,
|
||||||
|
"page_view_logged_in_browser_total" => 2,
|
||||||
|
)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
it "tracks normal error pages correctly" do
|
||||||
|
SiteSetting.bootstrap_error_pages = false
|
||||||
|
|
||||||
|
visit "/foobar"
|
||||||
|
|
||||||
|
try_until_success do
|
||||||
|
CachedCounting.flush
|
||||||
|
|
||||||
|
# Does not count error as a pageview
|
||||||
|
expect(ApplicationRequest.stats).to include(
|
||||||
|
"http_4xx_total" => 1,
|
||||||
|
"page_view_anon_total" => 0,
|
||||||
|
"page_view_anon_browser_total" => 0,
|
||||||
|
"page_view_logged_in_total" => 0,
|
||||||
|
"page_view_crawler_total" => 0,
|
||||||
|
)
|
||||||
|
end
|
||||||
|
|
||||||
|
find("#site-logo").click
|
||||||
|
|
||||||
|
try_until_success do
|
||||||
|
CachedCounting.flush
|
||||||
|
expect(ApplicationRequest.stats).to include(
|
||||||
|
"http_4xx_total" => 1,
|
||||||
|
"page_view_anon_total" => 1,
|
||||||
|
"page_view_anon_browser_total" => 1,
|
||||||
|
"page_view_logged_in_total" => 0,
|
||||||
|
"page_view_crawler_total" => 0,
|
||||||
|
)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
it "tracks non-ember pages correctly" do
|
||||||
|
visit "/safe-mode"
|
||||||
|
|
||||||
|
try_until_success do
|
||||||
|
CachedCounting.flush
|
||||||
|
|
||||||
|
# Does not count error as a pageview
|
||||||
|
expect(ApplicationRequest.stats).to include(
|
||||||
|
"page_view_anon_total" => 1,
|
||||||
|
"page_view_anon_browser_total" => 1,
|
||||||
|
"page_view_logged_in_total" => 0,
|
||||||
|
"page_view_crawler_total" => 0,
|
||||||
|
)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
it "tracks bootstrapped error pages correctly" do
|
||||||
|
SiteSetting.bootstrap_error_pages = true
|
||||||
|
|
||||||
|
visit "/foobar"
|
||||||
|
|
||||||
|
try_until_success do
|
||||||
|
CachedCounting.flush
|
||||||
|
|
||||||
|
# Does not count error as a pageview
|
||||||
|
expect(ApplicationRequest.stats).to include(
|
||||||
|
"http_4xx_total" => 1,
|
||||||
|
"page_view_anon_total" => 0,
|
||||||
|
"page_view_anon_browser_total" => 0,
|
||||||
|
"page_view_logged_in_total" => 0,
|
||||||
|
"page_view_crawler_total" => 0,
|
||||||
|
)
|
||||||
|
end
|
||||||
|
|
||||||
|
find("#site-logo").click
|
||||||
|
|
||||||
|
try_until_success do
|
||||||
|
CachedCounting.flush
|
||||||
|
expect(ApplicationRequest.stats).to include(
|
||||||
|
"http_4xx_total" => 1,
|
||||||
|
"page_view_anon_total" => 1,
|
||||||
|
"page_view_anon_browser_total" => 1,
|
||||||
|
"page_view_logged_in_total" => 0,
|
||||||
|
"page_view_crawler_total" => 0,
|
||||||
|
)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
it "tracks published pages correctly" do
|
||||||
|
SiteSetting.enable_page_publishing = true
|
||||||
|
page =
|
||||||
|
Fabricate(:published_page, public: true, slug: "some-page", topic: Fabricate(:post).topic)
|
||||||
|
|
||||||
|
visit "/pub/some-page"
|
||||||
|
|
||||||
|
try_until_success do
|
||||||
|
CachedCounting.flush
|
||||||
|
|
||||||
|
# Does not count error as a pageview
|
||||||
|
expect(ApplicationRequest.stats).to include(
|
||||||
|
"page_view_anon_total" => 1,
|
||||||
|
"page_view_anon_browser_total" => 1,
|
||||||
|
"page_view_logged_in_total" => 0,
|
||||||
|
"page_view_crawler_total" => 0,
|
||||||
|
)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
Loading…
Reference in New Issue