analytics/test/plausible_web/controllers/api/external_controller_test.exs
Adrian Gruntkowski 51c1138d0d
Implement better user password validation (#3344)
* Add zxcvbn dependency

* Change password length range requirement from 6-64 to 12-128

* Reimplement register form in LV

* Implement server-side check for password strength

* Add rudimentary strength meter

* Make password input with strength a separate component and improve it

* Fix existing tests to provide strong enough password

* Apply formatting

* Replace existing registration form with new one

* Hide built-in label in `.input` component when none provided

* Crop password to first 32 chars for analysis by zxcvbn

* Add tests for new form components

* Integrate hCaptcha into LV

* Fix existing AuthController tests

* Add tests for Live.RegisterForm

* Hide strength meter when password input is empty

* Randomize client IP in headers during tests to avoid hitting rate limit

* Apply auxilliary formatting fixes to AuthController

* Integrate registration from invitation into LV registration logic

* Fix existing password set and reset forms

* Make `password_length_hint` component more customizable

* Optimize `Auth.User.set_password/2`

* Remove unnecessary attribute from registration form

* Move password set and reset forms to LV

* Add tests for SetPasswordForm LV component

* Add tests for password checks in `Auth.User`

* Document code a bit

* Implement simpler approach to hCaptcha integration

* Update CHANGELOG.md

* Improve consistency of color scheme

* Introduce debounce across all text inputs in registration and password forms

* Fix email input background in register form

* Ensure only single error is rendered for empty password confirmation case

* Remove `/password` form entirely in favor of preferred password reset

* Remove unnecessary `router` option from `live_render` calls

* Make expensive assigns in LV with `assign_new` (h/t @aerosol)

* Accept passwords longer than 32 bytes uniformly as very strong

* Avoid displaying blank error side by side with weak password error

* Make register actions handle errors gracefully

* Render only a single piece of feedback to reduce noise

* Make register and password reset forms pw manager friendly (h/t @cnkk)

* Move registration forms to live routes

* Delete no longer used deadviews

* Adjust registration form in accordance to changes in #3290

* Reintroduce dogfood page path for invitation form from #3290

* Use alternative approach to submitting plausible metrics from LV form

* Rename metrics events and extend tests to account for them
2023-09-25 10:27:29 +02:00

1340 lines
35 KiB
Elixir

defmodule PlausibleWeb.Api.ExternalControllerTest do
use PlausibleWeb.ConnCase
use Plausible.ClickhouseRepo
@user_agent "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_13_4) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/70.0.3538.110 Safari/537.36"
@user_agent_mobile "Mozilla/5.0 (Linux; Android 6.0; U007 Pro Build/MRA58K; wv) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Chrome/44.0.2403.119 Mobile Safari/537.36"
@user_agent_tablet "Mozilla/5.0 (Linux; U; Android 4.2.2; it-it; Surfing TAB B 9.7 3G Build/JDQ39) AppleWebKit/534.30 (KHTML, like Gecko) Version/4.0 Mobile Safari/534.30"
describe "POST /api/event" do
setup do
site = insert(:site)
{:ok, site: site}
end
test "records the event", %{conn: conn, site: site} do
params = %{
domain: site.domain,
name: "pageview",
url: "http://gigride.live/",
referrer: "http://m.facebook.com/"
}
conn =
conn
|> put_req_header("user-agent", @user_agent)
|> post("/api/event", params)
pageview = get_event(site)
assert response(conn, 202) == "ok"
assert pageview.hostname == "gigride.live"
assert pageview.site_id == site.id
assert pageview.pathname == "/"
end
test "works with Content-Type: text/plain", %{conn: conn, site: site} do
params = %{
domain: site.domain,
name: "pageview",
url: "http://gigride.live/"
}
conn =
conn
|> put_req_header("content-type", "text/plain")
|> post("/api/event", Jason.encode!(params))
pageview = get_event(site)
assert response(conn, 202) == "ok"
assert pageview.hostname == "gigride.live"
assert pageview.site_id == site.id
assert pageview.pathname == "/"
end
test "returns error if JSON cannot be parsed", %{conn: conn} do
conn =
conn
|> put_req_header("content-type", "text/plain")
|> post("/api/event", "")
assert conn.status == 400
end
test "can send to multiple dashboards by listing multiple domains - same timestamp", %{
conn: conn,
site: site1
} do
site2 = insert(:site)
params = %{
name: "pageview",
url: "http://gigride.live/",
referrer: "http://m.facebook.com/",
domain: "#{site1.domain},#{site2.domain}"
}
conn =
conn
|> put_req_header("user-agent", @user_agent)
|> post("/api/event", params)
assert response(conn, 202) == "ok"
assert e1 = get_event(site1)
assert e2 = get_event(site2)
assert NaiveDateTime.compare(e1.timestamp, e2.timestamp) == :eq
end
test "timestamps differ when two events sent in a row", %{conn: conn, site: site} do
params = %{
domain: site.domain,
name: "pageview",
url: "http://gigride.live/",
referrer: "http://m.facebook.com/"
}
t1 = System.convert_time_unit(System.monotonic_time(), :native, :millisecond)
conn
|> put_req_header("user-agent", @user_agent)
|> post("/api/event", params)
t2 = System.convert_time_unit(System.monotonic_time(), :native, :millisecond)
# timestamps are in second precision, so we need to wait for it to flip
Process.sleep(1000 - (t2 - t1))
conn
|> put_req_header("user-agent", @user_agent)
|> post("/api/event", params)
[e1, e2] = get_events(site)
assert NaiveDateTime.compare(e1.timestamp, e2.timestamp) == :gt
end
test "www. is stripped from domain", %{conn: conn, site: site} do
params = %{
name: "custom event",
url: "http://gigride.live/",
domain: site.domain
}
conn
|> post("/api/event", params)
pageview = get_event(site)
assert pageview.site_id == site.id
end
test "www. is stripped from hostname", %{conn: conn, site: site} do
params = %{
name: "pageview",
url: "http://www.example.com/",
domain: site.domain
}
conn
|> post("/api/event", params)
pageview = get_event(site)
assert pageview.hostname == "example.com"
end
test "empty path defaults to /", %{conn: conn, site: site} do
params = %{
name: "pageview",
url: "http://www.example.com",
domain: site.domain
}
conn
|> post("/api/event", params)
pageview = get_event(site)
assert pageview.pathname == "/"
end
test "trailing whitespace is removed", %{conn: conn, site: site} do
params = %{
name: "pageview",
url: "http://www.example.com/path ",
domain: site.domain
}
conn
|> post("/api/event", params)
pageview = get_event(site)
assert pageview.pathname == "/path"
end
test "bots and crawlers are ignored", %{conn: conn, site: site} do
params = %{
name: "pageview",
url: "http://www.example.com/",
domain: site.domain
}
conn
|> put_req_header("user-agent", "generic crawler")
|> post("/api/event", params)
assert get_event(site) == nil
end
test "Headless Chrome is ignored", %{conn: conn, site: site} do
params = %{
name: "pageview",
url: "http://www.example.com/",
domain: site.domain
}
conn
|> put_req_header(
"user-agent",
"Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) HeadlessChrome/85.0.4183.83 Safari/537.36"
)
|> post("/api/event", params)
assert get_event(site) == nil
end
test "parses user_agent", %{conn: conn, site: site} do
params = %{
name: "pageview",
url: "http://gigride.live/",
domain: site.domain
}
conn =
conn
|> put_req_header("user-agent", @user_agent)
|> post("/api/event", params)
pageview = get_event(site)
assert response(conn, 202) == "ok"
assert pageview.operating_system == "Mac"
assert pageview.operating_system_version == "10.13"
assert pageview.browser == "Chrome"
assert pageview.browser_version == "70.0"
end
test "parses referrer", %{conn: conn, site: site} do
params = %{
name: "pageview",
url: "http://gigride.live/",
referrer: "https://facebook.com",
domain: site.domain
}
conn =
conn
|> put_req_header("user-agent", @user_agent)
|> post("/api/event", params)
pageview = get_event(site)
assert response(conn, 202) == "ok"
assert pageview.referrer_source == "Facebook"
end
test "strips trailing slash from referrer", %{conn: conn, site: site} do
params = %{
name: "pageview",
url: "http://gigride.live/",
referrer: "https://facebook.com/page/",
domain: site.domain
}
conn =
conn
|> put_req_header("user-agent", @user_agent)
|> post("/api/event", params)
pageview = get_event(site)
assert response(conn, 202) == "ok"
assert pageview.referrer == "facebook.com/page"
assert pageview.referrer_source == "Facebook"
end
test "ignores event when referrer is a spammer", %{conn: conn, site: site} do
params = %{
domain: site.domain,
name: "pageview",
url: "http://gigride.live/",
referrer: "https://www.1-best-seo.com"
}
conn =
conn
|> put_req_header("user-agent", @user_agent)
|> post("/api/event", params)
assert response(conn, 202) == "ok"
assert !get_event(site)
end
test "blocks traffic from a domain when it's blocked", %{
conn: conn
} do
site = insert(:site, ingest_rate_limit_threshold: 0)
params = %{
domain: site.domain,
name: "pageview",
url: "https://feature-flag-test.com"
}
conn =
conn
|> put_req_header("user-agent", @user_agent)
|> post("/api/event", params)
assert response(conn, 202) == "ok"
refute get_event(site)
end
test "ignores when referrer is internal", %{conn: conn, site: site} do
params = %{
name: "pageview",
url: "http://gigride.live/",
referrer: "https://gigride.live",
domain: site.domain
}
conn =
conn
|> put_req_header("user-agent", @user_agent)
|> post("/api/event", params)
pageview = get_event(site)
assert response(conn, 202) == "ok"
assert pageview.referrer_source == ""
end
test "ignores localhost referrer", %{conn: conn, site: site} do
params = %{
name: "pageview",
url: "http://gigride.live/",
referrer: "http://localhost:4000/",
domain: site.domain
}
conn =
conn
|> put_req_header("user-agent", @user_agent)
|> post("/api/event", params)
pageview = get_event(site)
assert response(conn, 202) == "ok"
assert pageview.referrer_source == ""
end
test "parses subdomain referrer", %{conn: conn, site: site} do
params = %{
name: "pageview",
url: "http://gigride.live/",
referrer: "https://blog.gigride.live",
domain: site.domain
}
conn =
conn
|> put_req_header("user-agent", @user_agent)
|> post("/api/event", params)
pageview = get_event(site)
assert response(conn, 202) == "ok"
assert pageview.referrer_source == "blog.gigride.live"
end
test "referrer is cleaned", %{conn: conn, site: site} do
params = %{
name: "pageview",
url: "http://www.example.com/",
referrer: "https://www.indiehackers.com/page?query=param#hash",
domain: site.domain
}
conn
|> post("/api/event", params)
pageview = get_event(site)
assert pageview.referrer == "indiehackers.com/page"
end
test "utm_source overrides referrer source", %{conn: conn, site: site} do
params = %{
name: "pageview",
url: "http://www.example.com/?utm_source=betalist",
referrer: "https://betalist.com/my-produxct",
domain: site.domain
}
conn
|> post("/api/event", params)
pageview = get_event(site)
assert pageview.referrer_source == "betalist"
end
test "utm tags are stored", %{conn: conn, site: site} do
params = %{
name: "pageview",
url:
"http://www.example.com/?utm_medium=ads&utm_source=instagram&utm_campaign=video_story",
domain: site.domain
}
conn
|> post("/api/event", params)
pageview = get_event(site)
assert pageview.utm_medium == "ads"
assert pageview.utm_source == "instagram"
assert pageview.utm_campaign == "video_story"
end
test "if it's an :unknown referrer, just the domain is used", %{conn: conn, site: site} do
params = %{
name: "pageview",
url: "http://gigride.live/",
referrer: "https://www.indiehackers.com/landing-page-feedback",
domain: site.domain
}
conn =
conn
|> put_req_header("user-agent", @user_agent)
|> post("/api/event", params)
pageview = get_event(site)
assert response(conn, 202) == "ok"
assert pageview.referrer_source == "indiehackers.com"
end
test "if the referrer is not http or https, it is ignored", %{conn: conn, site: site} do
params = %{
name: "pageview",
url: "http://gigride.live/",
referrer: "android-app://com.google.android.gm",
domain: site.domain
}
conn =
conn
|> put_req_header("user-agent", @user_agent)
|> post("/api/event", params)
pageview = get_event(site)
assert response(conn, 202) == "ok"
assert pageview.referrer_source == ""
end
test "screen size is calculated from user agent", %{conn: conn, site: site} do
params = %{
name: "pageview",
url: "http://gigride.live/",
domain: site.domain
}
conn =
conn
|> put_req_header("user-agent", @user_agent_mobile)
|> post("/api/event", params)
pageview = get_event(site)
assert response(conn, 202) == "ok"
assert pageview.screen_size == "Mobile"
end
test "screen size is nil if user agent is unknown", %{conn: conn, site: site} do
params = %{
name: "pageview",
url: "http://gigride.live/",
domain: site.domain
}
conn =
conn
|> put_req_header("user-agent", "unknown UA")
|> post("/api/event", params)
pageview = get_event(site)
assert response(conn, 202) == "ok"
assert pageview.screen_size == ""
end
test "screen size is calculated from user_agent when is tablet", %{conn: conn, site: site} do
params = %{
name: "pageview",
url: "http://gigride.live/",
domain: site.domain
}
conn =
conn
|> put_req_header("user-agent", @user_agent_tablet)
|> post("/api/event", params)
pageview = get_event(site)
assert response(conn, 202) == "ok"
assert pageview.screen_size == "Tablet"
end
test "screen size is calculated from user_agent when is desktop", %{
conn: conn,
site: site
} do
params = %{
name: "pageview",
url: "http://gigride.live/",
domain: site.domain
}
conn =
conn
|> put_req_header("user-agent", @user_agent)
|> post("/api/event", params)
pageview = get_event(site)
assert response(conn, 202) == "ok"
assert pageview.screen_size == "Desktop"
end
test "can trigger a custom event", %{conn: conn, site: site} do
params = %{
name: "custom event",
url: "http://gigride.live/",
domain: site.domain
}
conn =
conn
|> put_req_header("user-agent", @user_agent)
|> post("/api/event", params)
event = get_event(site)
assert response(conn, 202) == "ok"
assert event.name == "custom event"
end
test "casts custom props to string", %{conn: conn, site: site} do
params = %{
name: "Signup",
url: "http://gigride.live/",
domain: site.domain,
props: %{
bool_test: true,
number_test: 12
}
}
conn
|> post("/api/event", params)
event = get_event(site)
assert Map.get(event, :"meta.key") == ["bool_test", "number_test"]
assert Map.get(event, :"meta.value") == ["true", "12"]
end
test "ignores malformed custom props", %{conn: conn, site: site} do
params = %{
name: "Signup",
url: "http://gigride.live/",
domain: site.domain,
props: "\"show-more:button\""
}
conn
|> post("/api/event", params)
event = get_event(site)
assert Map.get(event, :"meta.key") == []
assert Map.get(event, :"meta.value") == []
end
test "can send props stringified", %{conn: conn, site: site} do
params = %{
name: "Signup",
url: "http://gigride.live/",
domain: site.domain,
props: Jason.encode!(%{number_test: 12})
}
conn
|> post("/api/event", params)
event = get_event(site)
assert Map.get(event, :"meta.key") == ["number_test"]
assert Map.get(event, :"meta.value") == ["12"]
end
test "ignores custom prop with array value", %{conn: conn, site: site} do
params = %{
name: "Signup",
url: "http://gigride.live/",
domain: site.domain,
props: Jason.encode!(%{wat: ["some-thing"], other: "key"})
}
conn = post(conn, "/api/event", params)
assert conn.status == 202
event = get_event(site)
assert Map.get(event, :"meta.key") == ["other"]
assert Map.get(event, :"meta.value") == ["key"]
end
test "ignores custom prop with map value", %{conn: conn, site: site} do
params = %{
name: "Signup",
url: "http://gigride.live/",
domain: site.domain,
props: Jason.encode!(%{foo: %{bar: "baz"}, other_key: 1})
}
conn = post(conn, "/api/event", params)
assert conn.status == 202
event = get_event(site)
assert Map.get(event, :"meta.key") == ["other_key"]
assert Map.get(event, :"meta.value") == ["1"]
end
test "ignores custom prop with empty string value", %{conn: conn, site: site} do
params = %{
name: "Signup",
url: "http://gigride.live/",
domain: site.domain,
props: Jason.encode!(%{foo: "", other_key: true})
}
conn = post(conn, "/api/event", params)
assert conn.status == 202
event = get_event(site)
assert Map.get(event, :"meta.key") == ["other_key"]
assert Map.get(event, :"meta.value") == ["true"]
end
test "ignores custom prop with nil value", %{conn: conn, site: site} do
params = %{
name: "Signup",
url: "http://gigride.live/",
domain: site.domain,
props: Jason.encode!(%{foo: nil, other_key: true})
}
conn = post(conn, "/api/event", params)
assert conn.status == 202
event = get_event(site)
assert Map.get(event, :"meta.key") == ["other_key"]
assert Map.get(event, :"meta.value") == ["true"]
end
test "converts revenue values into the goal currency", %{conn: conn, site: site} do
params = %{
name: "Payment",
url: "http://gigride.live/",
domain: site.domain,
revenue: %{amount: 10.2, currency: "USD"}
}
insert(:goal, event_name: "Payment", currency: "BRL", site: site)
assert %{status: 202} = post(conn, "/api/event", params)
assert %{revenue_reporting_amount: amount} = get_event(site)
assert Decimal.equal?(Decimal.new("7.14"), amount)
end
test "revenue values can be sent with minified keys", %{conn: conn, site: site} do
params = %{
"n" => "Payment",
"u" => "http://gigride.live/",
"d" => site.domain,
"$" => Jason.encode!(%{amount: 10.2, currency: "USD"})
}
insert(:goal, event_name: "Payment", currency: "BRL", site: site)
assert %{status: 202} = post(conn, "/api/event", params)
assert %{revenue_reporting_amount: amount} = get_event(site)
assert Decimal.equal?(Decimal.new("7.14"), amount)
end
test "saves the exact same amount when goal currency is the same as the event", %{
conn: conn,
site: site
} do
params = %{
name: "Payment",
url: "http://gigride.live/",
domain: site.domain,
revenue: %{amount: 10, currency: "BRL"}
}
insert(:goal, event_name: "Payment", currency: "BRL", site: site)
assert %{status: 202} = post(conn, "/api/event", params)
assert %{revenue_reporting_amount: amount} = get_event(site)
assert Decimal.equal?(Decimal.new("10.0"), amount)
end
test "does not fail when revenue value is invalid", %{conn: conn, site: site} do
params = %{
name: "Payment",
url: "http://gigride.live/",
domain: site.domain,
revenue: %{amount: "1831d", currency: "ADSIE"}
}
insert(:goal, event_name: "Payment", currency: "BRL", site: site)
assert %{status: 202} = post(conn, "/api/event", params)
assert %Plausible.ClickhouseEventV2{} = get_event(site)
end
test "does not fail when sending revenue without a matching goal", %{conn: conn, site: site} do
params = %{
name: "Add to Cart",
url: "http://gigride.live/",
domain: site.domain,
revenue: %{amount: 10.2, currency: "USD"}
}
insert(:goal, event_name: "Checkout", currency: "BRL", site: site)
insert(:goal, event_name: "Payment", currency: "USD", site: site)
assert %{status: 202} = post(conn, "/api/event", params)
assert %Plausible.ClickhouseEventV2{revenue_reporting_amount: nil} = get_event(site)
end
test "ignores a malformed referrer URL", %{conn: conn, site: site} do
params = %{
name: "pageview",
url: "http://gigride.live/",
referrer: "https:://twitter.com",
domain: site.domain
}
conn =
conn
|> put_req_header("user-agent", @user_agent)
|> post("/api/event", params)
event = get_event(site)
assert response(conn, 202) == "ok"
assert event.referrer == ""
end
# Fake geo is loaded from test/priv/GeoLite2-City-Test.mmdb
test "looks up location data from the ip address", %{conn: conn, site: site} do
params = %{
name: "pageview",
domain: site.domain,
url: "http://gigride.live/"
}
conn
|> put_req_header("x-forwarded-for", "2.125.160.216")
|> post("/api/event", params)
pageview = get_event(site)
assert pageview.country_code == "GB"
assert pageview.subdivision1_code == "GB-ENG"
assert pageview.subdivision2_code == "GB-WBK"
assert pageview.city_geoname_id == 2_655_045
end
test "ignores unknown country code ZZ", %{conn: conn, site: site} do
params = %{
name: "pageview",
domain: site.domain,
url: "http://gigride.live/"
}
conn
|> put_req_header("x-forwarded-for", "0.0.0.0")
|> post("/api/event", params)
pageview = get_event(site)
assert pageview.country_code == <<0, 0>>
assert pageview.subdivision1_code == ""
assert pageview.subdivision2_code == ""
assert pageview.city_geoname_id == 0
end
test "ignores disputed territory code XX", %{conn: conn, site: site} do
params = %{
name: "pageview",
domain: site.domain,
url: "http://gigride.live/"
}
conn
|> put_req_header("x-forwarded-for", "0.0.0.1")
|> post("/api/event", params)
pageview = get_event(site)
assert pageview.country_code == <<0, 0>>
assert pageview.subdivision1_code == ""
assert pageview.subdivision2_code == ""
assert pageview.city_geoname_id == 0
end
test "ignores TOR exit node country code T1", %{conn: conn, site: site} do
params = %{
name: "pageview",
domain: site.domain,
url: "http://gigride.live/"
}
conn
|> put_req_header("x-forwarded-for", "0.0.0.2")
|> post("/api/event", params)
pageview = get_event(site)
assert pageview.country_code == <<0, 0>>
assert pageview.subdivision1_code == ""
assert pageview.subdivision2_code == ""
assert pageview.city_geoname_id == 0
end
test "scrubs port from x-forwarded-for", %{conn: conn, site: site} do
params = %{
name: "pageview",
domain: site.domain,
url: "http://gigride.live/"
}
conn
|> put_req_header("x-forwarded-for", "216.160.83.56:123")
|> post("/api/event", params)
pageview = get_event(site)
assert pageview.country_code == "US"
end
test "works with ipv6 without port in x-forwarded-for", %{conn: conn, site: site} do
params = %{
name: "pageview",
domain: site.domain,
url: "http://gigride.live/"
}
conn
|> put_req_header("x-forwarded-for", "2001:218:1:1:1:1:1:1")
|> post("/api/event", params)
pageview = get_event(site)
assert pageview.country_code == "JP"
end
test "works with ipv6 with a port number in x-forwarded-for", %{conn: conn, site: site} do
params = %{
name: "pageview",
domain: site.domain,
url: "http://gigride.live/"
}
conn
|> put_req_header("x-forwarded-for", "[2001:218:1:1:1:1:1:1]:123")
|> post("/api/event", params)
pageview = get_event(site)
assert pageview.country_code == "JP"
end
test "uses cloudflare's special header for client IP address if present", %{
conn: conn,
site: site
} do
params = %{
name: "pageview",
domain: site.domain,
url: "http://gigride.live/"
}
conn
|> put_req_header("x-forwarded-for", "0.0.0.0")
|> put_req_header("cf-connecting-ip", "216.160.83.56")
|> post("/api/event", params)
pageview = get_event(site)
assert pageview.country_code == "US"
end
test "uses BunnyCDN's custom header for client IP address if present", %{
conn: conn,
site: site
} do
params = %{
name: "pageview",
domain: site.domain,
url: "http://gigride.live/"
}
conn
|> put_req_header("x-forwarded-for", "0.0.0.0")
|> put_req_header("b-forwarded-for", "216.160.83.56,9.9.9.9")
|> post("/api/event", params)
pageview = get_event(site)
assert pageview.country_code == "US"
end
test "Uses the Forwarded header when cf-connecting-ip and x-forwarded-for are missing", %{
site: site
} do
params = %{
name: "pageview",
domain: site.domain,
url: "http://gigride.live/"
}
build_conn()
|> put_req_header("forwarded", "by=0.0.0.0;for=216.160.83.56;host=somehost.com;proto=https")
|> post("/api/event", params)
pageview = get_event(site)
assert pageview.country_code == "US"
end
test "Forwarded header can parse ipv6", %{site: site} do
params = %{
name: "pageview",
domain: site.domain,
url: "http://gigride.live/"
}
build_conn()
|> put_req_header(
"forwarded",
"by=0.0.0.0;for=\"[2001:218:1:1:1:1:1:1]\",for=0.0.0.0;host=somehost.com;proto=https"
)
|> post("/api/event", params)
pageview = get_event(site)
assert pageview.country_code == "JP"
end
test "URL is decoded", %{conn: conn, site: site} do
params = %{
name: "pageview",
url:
"http://www.example.com/opportunity/category/%D8%AC%D9%88%D8%A7%D8%A6%D8%B2-%D9%88%D9%85%D8%B3%D8%A7%D8%A8%D9%82%D8%A7%D8%AA",
domain: site.domain
}
conn
|> post("/api/event", params)
pageview = get_event(site)
assert pageview.pathname == "/opportunity/category/جوائز-ومسابقات"
end
test "accepts shorthand map keys", %{conn: conn, site: site} do
params = %{
n: "pageview",
u: "http://www.example.com/opportunity",
d: site.domain,
r: "https://facebook.com/page"
}
conn
|> post("/api/event", params)
pageview = get_event(site)
assert pageview.pathname == "/opportunity"
assert pageview.referrer_source == "Facebook"
assert pageview.referrer == "facebook.com/page"
end
test "records hash when in hash mode", %{conn: conn, site: site} do
params = %{
n: "pageview",
u: "http://www.example.com/#page-a",
d: site.domain,
h: 1
}
conn
|> post("/api/event", params)
pageview = get_event(site)
assert pageview.pathname == "/#page-a"
end
test "does not record hash when hash mode is 0", %{conn: conn, site: site} do
params = %{
n: "pageview",
u: "http://www.example.com/#page-a",
d: site.domain,
h: 0
}
conn
|> post("/api/event", params)
pageview = get_event(site)
assert pageview.pathname == "/"
end
test "decodes URL pathname, fragment and search", %{conn: conn, site: site} do
params = %{
n: "pageview",
u:
"https://test.com/%EF%BA%9D%EF%BB%AD%EF%BA%8E%EF%BA%8B%EF%BA%AF-%EF%BB%AE%EF%BB%A4%EF%BA%B3%EF%BA%8E%EF%BA%92%EF%BB%97%EF%BA%8E%EF%BA%97?utm_source=%25balle%25",
d: site.domain,
h: 1
}
conn
|> post("/api/event", params)
pageview = get_event(site)
assert pageview.hostname == "test.com"
assert pageview.pathname == "/ﺝﻭﺎﺋﺯ-ﻮﻤﺳﺎﺒﻗﺎﺗ"
assert pageview.utm_source == "%balle%"
end
test "can use double quotes in query params", %{conn: conn, site: site} do
q = URI.encode_query(%{"utm_source" => "Something \"quoted\""})
params = %{
n: "pageview",
u: "https://test.com/?" <> q,
d: site.domain,
h: 1
}
conn
|> post("/api/event", params)
pageview = get_event(site)
assert pageview.utm_source == "Something \"quoted\""
end
test "responds 400 when required fields are missing", %{conn: conn, site: site} do
params = %{
domain: site.domain,
name: "pageview"
}
conn =
conn
|> put_req_header("user-agent", @user_agent)
|> post("/api/event", params)
assert json_response(conn, 400) == %{
"errors" => %{
"url" => ["is required"]
}
}
end
test "responds 400 with errors when domain is missing", %{conn: conn} do
params = %{
domain: nil,
url: "about:config",
name: "pageview"
}
conn =
conn
|> put_req_header("user-agent", @user_agent)
|> post("/api/event", params)
assert json_response(conn, 400) == %{
"errors" => %{
"domain" => ["can't be blank"]
}
}
end
end
describe "user_id generation" do
setup do
site = insert(:site)
{:ok, site: site}
end
test "with same IP address and user agent, the same user ID is generated", %{
conn: conn,
site: site
} do
params = %{
url: "https://user-id-test-domain.com/",
domain: site.domain,
name: "pageview"
}
conn
|> put_req_header("user-agent", @user_agent)
|> put_req_header("x-forwarded-for", "127.0.0.1")
|> post("/api/event", params)
conn
|> put_req_header("user-agent", @user_agent)
|> put_req_header("x-forwarded-for", "127.0.0.1")
|> post("/api/event", params)
[one, two] = get_events(site)
assert one.user_id == two.user_id
end
test "different IP address results in different user ID", %{conn: conn, site: site} do
params = %{
url: "https://user-id-test-domain.com/",
domain: site.domain,
name: "pageview"
}
conn
|> put_req_header("user-agent", @user_agent)
|> put_req_header("x-forwarded-for", "127.0.0.1")
|> post("/api/event", params)
conn
|> put_req_header("user-agent", @user_agent)
|> put_req_header("x-forwarded-for", "82.32.12.1")
|> post("/api/event", params)
[one, two] = get_events(site)
assert one.user_id != two.user_id
end
test "different user agent results in different user ID", %{conn: conn, site: site} do
params = %{
url: "https://user-id-test-domain.com/",
domain: site.domain,
name: "pageview"
}
conn
|> put_req_header("user-agent", @user_agent)
|> put_req_header("x-forwarded-for", "127.0.0.1")
|> post("/api/event", params)
conn
|> put_req_header("user-agent", @user_agent <> "!!")
|> put_req_header("x-forwarded-for", "127.0.0.1")
|> post("/api/event", params)
[one, two] = get_events(site)
assert one.user_id != two.user_id
end
test "different domain value results in different user ID", %{conn: conn, site: site1} do
site2 = insert(:site)
params = %{
url: "https://user-id-test-domain.com/",
domain: site1.domain,
name: "pageview"
}
conn
|> put_req_header("user-agent", @user_agent)
|> put_req_header("x-forwarded-for", "127.0.0.1")
|> post("/api/event", params)
conn
|> put_req_header("user-agent", @user_agent)
|> put_req_header("x-forwarded-for", "127.0.0.1")
|> post("/api/event", Map.put(params, :domain, site2.domain))
one = get_event(site1)
two = get_event(site2)
assert one.user_id != two.user_id
end
test "different hostname results in different user ID", %{conn: conn, site: site} do
params = %{
url: "https://user-id-test-domain.com/",
domain: site.domain,
name: "pageview"
}
conn
|> put_req_header("user-agent", @user_agent)
|> put_req_header("x-forwarded-for", "127.0.0.1")
|> post("/api/event", params)
conn
|> put_req_header("user-agent", @user_agent)
|> put_req_header("x-forwarded-for", "127.0.0.1")
|> post("/api/event", Map.put(params, :url, "https://other-domain.com/"))
[one, two] = get_events(site)
assert one.user_id != two.user_id
end
test "different hostname results in the same user ID when the root domain in the same", %{
conn: conn,
site: site
} do
params = %{
url: "https://user-id-test-domain.com/",
domain: site.domain,
name: "pageview"
}
conn
|> put_req_header("user-agent", @user_agent)
|> put_req_header("x-forwarded-for", "127.0.0.1")
|> post("/api/event", params)
conn
|> put_req_header("user-agent", @user_agent)
|> put_req_header("x-forwarded-for", "127.0.0.1")
|> post("/api/event", Map.put(params, :url, "https://app.user-id-test-domain.com/"))
[one, two] = get_events(site)
assert one.user_id == two.user_id
end
end
describe "remaining" do
setup do
site = insert(:site)
{:ok, site: site}
end
test "defaults hostname to (none) when missing", %{conn: conn, site: site} do
params = %{
domain: site.domain,
name: "pageview",
url: "file:///android_asset/www/index.html"
}
conn =
conn
|> put_req_header("content-type", "text/plain")
|> post("/api/event", Jason.encode!(params))
pageview = get_event(site)
assert response(conn, 202) == "ok"
assert pageview.hostname == "(none)"
end
test "accepts chrome extension URLs", %{conn: conn, site: site} do
params = %{
domain: site.domain,
name: "pageview",
url: "chrome-extension://liipgellkffekalgefpjolodblggkmjg/popup.html"
}
conn =
conn
|> put_req_header("content-type", "text/plain")
|> post("/api/event", Jason.encode!(params))
pageview = get_event(site)
assert response(conn, 202) == "ok"
assert pageview.hostname == "liipgellkffekalgefpjolodblggkmjg"
end
end
describe "GET /api/health" do
test "returns 200 OK", %{conn: conn} do
conn = get(conn, "/api/health")
assert payload = json_response(conn, 200)
assert payload["postgres"] == "ok"
assert payload["clickhouse"] == "ok"
assert payload["sites_cache"] == "ok"
end
end
defp get_event(site) do
Plausible.Event.WriteBuffer.flush()
ClickhouseRepo.one(
from(e in Plausible.ClickhouseEventV2,
where: e.site_id == ^site.id,
order_by: [desc: e.timestamp]
)
)
end
defp get_events(site) do
Plausible.Event.WriteBuffer.flush()
ClickhouseRepo.all(
from(e in Plausible.ClickhouseEventV2,
where: e.site_id == ^site.id,
order_by: [desc: e.timestamp]
)
)
end
end