mirror of
https://github.com/plausible/analytics.git
synced 2024-12-20 16:11:49 +03:00
b4b7532f07
* first commit with test and compile job Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * adding 'prepare' stage Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * updated ci script to include "test" compile phase Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * adding environment variables for connecting to postgresql Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * updated ci config for postgres Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * using non-alpine version of elixir Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * re-using the 'compile' artifacts and added explict env variables for testing Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * removing redundant deps fetching from common code Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * formatting using mix.format -- beware no-code changes! Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * added release config Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * adding consistent env variable for Database Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * more cleaning up of environment variables Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * Adding releases config for enabling releases Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * cleaning up env configs Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * Cleaned up config and prepared config for releases Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * updated CI script with new config for test Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * Added Dockerfile for creating production docker image Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * Adding "docker" build job yay! Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * using non-slim version of debian and installing webpack Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * Adding overlays for migrations on releases Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * restricting the docker built to master branch only Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * typo fix Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * adding "Hosting.md" to explain hosting instructions Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * removed the default comments Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * Added documentation related to env variables * updated documentation and fixed typo Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * updated documentation * Bumping up elixir version as `overlays` are only supported in latest version read release notes: https://github.com/elixir-lang/elixir/releases/tag/v1.10.0 Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * Adding tarball assembly during release Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * updated HOSTING.md Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * Added support for db migration Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * minor corrections Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * initializing admin user Admin user has been added in the "migration" phase. A default user is automatically created in the process. One can provide the related env variables, else a new one will be automatically created for you. Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * Initial base domain update - phase#1 These changes are only meant for correct operating it under self-hosting. There are many other cosmetic changes, that require updates to email, site and other places where the original website and author is used. Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * Using dedicated config variable `base_domain` instead Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * adding base_domain to releases config Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * removing the dedicated config "base_domain", relying on endpoint host Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * Removed the usage of "Mix" in code! It is bad practice to use "mix" module inside the code as in actual release this module is unavailable. Replacing this with a config environment variable Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * Added support for SMTP via Bamboo Smtp Adapter Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * Capturing SMTP errors via Sentry Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * Minor updates Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * Adding junit formatter -- useful for generating test reports Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * adding documentation for default user * Resolve "Gitlab Adoption: Add supported services in "Security & Compliance"" * bumping up the debian version to fix issues fixing some vulnerabilities identified by the scanning tools * More updates for self-hosting Changes in most of the places to suit self-hosting. Although, there are some which have been left-off. Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * quick-dirty-fix! * bumping up the db connect timeout Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * bumping up the db connect timeout Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * bumping up the db connect timeout Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * bumping up timeout - skipping MRs :-/ * removing restrictions on watching for changes this stuff isn't working * Update HOSTING.md * renamed the module name * reverting formatting-whitespace changes Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * reverting the name to release Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * adding docker-compose.yml and related instructions Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * using `plausible_url` instead of assuming `https` this is because, it is much to test in local dev machines and in most cases there's already a layer above which is capable for `https` termination and http -> https upgrade Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * WIP: merging changes from upstream Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * wip: more changes * Pushing in changes from upstream Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * changes to ci for testing Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * cleaning up and finishing clickhouse integration Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * updating readme with hosting details * removing deleted files from upstream * minor config adjustments Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me> * formatting changes Signed-off-by: Chandra Tungathurthi <tckb@tgrthi.me>
413 lines
13 KiB
Elixir
413 lines
13 KiB
Elixir
defmodule PlausibleWeb.SiteController do
|
|
use PlausibleWeb, :controller
|
|
use Plausible.Repo
|
|
alias Plausible.{Sites, Goals}
|
|
|
|
plug PlausibleWeb.RequireAccountPlug
|
|
|
|
def index(conn, _params) do
|
|
user = conn.assigns[:current_user] |> Repo.preload(:sites)
|
|
render(conn, "index.html", sites: user.sites)
|
|
end
|
|
|
|
def new(conn, _params) do
|
|
changeset = Plausible.Site.changeset(%Plausible.Site{})
|
|
|
|
render(conn, "new.html", changeset: changeset, layout: {PlausibleWeb.LayoutView, "focus.html"})
|
|
end
|
|
|
|
def create_site(conn, %{"site" => site_params}) do
|
|
user = conn.assigns[:current_user]
|
|
|
|
case insert_site(user.id, site_params) do
|
|
{:ok, %{site: site}} ->
|
|
Plausible.Slack.notify("#{user.name} created #{site.domain}")
|
|
|
|
conn
|
|
|> put_session(site.domain <> "_offer_email_report", true)
|
|
|> redirect(to: "/#{URI.encode_www_form(site.domain)}/snippet")
|
|
|
|
{:error, :site, changeset, _} ->
|
|
render(conn, "new.html",
|
|
changeset: changeset,
|
|
layout: {PlausibleWeb.LayoutView, "focus.html"}
|
|
)
|
|
end
|
|
end
|
|
|
|
def add_snippet(conn, %{"website" => website}) do
|
|
site =
|
|
Sites.get_for_user!(conn.assigns[:current_user].id, website)
|
|
|> Repo.preload(:custom_domain)
|
|
|
|
conn
|
|
|> assign(:skip_plausible_tracking, true)
|
|
|> render("snippet.html", site: site, layout: {PlausibleWeb.LayoutView, "focus.html"})
|
|
end
|
|
|
|
def new_goal(conn, %{"website" => website}) do
|
|
site = Sites.get_for_user!(conn.assigns[:current_user].id, website)
|
|
changeset = Plausible.Goal.changeset(%Plausible.Goal{})
|
|
|
|
conn
|
|
|> assign(:skip_plausible_tracking, true)
|
|
|> render("new_goal.html",
|
|
site: site,
|
|
changeset: changeset,
|
|
layout: {PlausibleWeb.LayoutView, "focus.html"}
|
|
)
|
|
end
|
|
|
|
def create_goal(conn, %{"website" => website, "goal" => goal}) do
|
|
site = Sites.get_for_user!(conn.assigns[:current_user].id, website)
|
|
|
|
case Plausible.Goals.create(site, goal) do
|
|
{:ok, _} ->
|
|
conn
|
|
|> put_flash(:success, "Goal created succesfully")
|
|
|> redirect(to: "/#{URI.encode_www_form(site.domain)}/settings")
|
|
|
|
{:error, :goal, changeset, _} ->
|
|
conn
|
|
|> assign(:skip_plausible_tracking, true)
|
|
|> render("new_goal.html",
|
|
site: site,
|
|
changeset: changeset,
|
|
layout: {PlausibleWeb.LayoutView, "focus.html"}
|
|
)
|
|
end
|
|
end
|
|
|
|
def delete_goal(conn, %{"website" => website, "id" => goal_id}) do
|
|
Plausible.Goals.delete(goal_id)
|
|
|
|
conn
|
|
|> put_flash(:success, "Goal deleted succesfully")
|
|
|> redirect(to: "/#{URI.encode_www_form(website)}/settings")
|
|
end
|
|
|
|
def settings(conn, %{"website" => website}) do
|
|
site =
|
|
Sites.get_for_user!(conn.assigns[:current_user].id, website)
|
|
|> Repo.preload(:google_auth)
|
|
|> Repo.preload(:custom_domain)
|
|
|
|
search_console_domains =
|
|
if site.google_auth do
|
|
Plausible.Google.Api.fetch_verified_properties(site.google_auth)
|
|
end
|
|
|
|
weekly_report = Repo.get_by(Plausible.Site.WeeklyReport, site_id: site.id)
|
|
monthly_report = Repo.get_by(Plausible.Site.MonthlyReport, site_id: site.id)
|
|
goals = Goals.for_site(site.domain)
|
|
shared_links = Repo.all(from l in Plausible.Site.SharedLink, where: l.site_id == ^site.id)
|
|
|
|
conn
|
|
|> assign(:skip_plausible_tracking, true)
|
|
|> render("settings.html",
|
|
site: site,
|
|
weekly_report: weekly_report,
|
|
monthly_report: monthly_report,
|
|
search_console_domains: search_console_domains,
|
|
goals: goals,
|
|
shared_links: shared_links,
|
|
changeset: Plausible.Site.changeset(site, %{})
|
|
)
|
|
end
|
|
|
|
def update_google_auth(conn, %{"website" => website, "google_auth" => attrs}) do
|
|
site =
|
|
Sites.get_for_user!(conn.assigns[:current_user].id, website)
|
|
|> Repo.preload(:google_auth)
|
|
|
|
Plausible.Site.GoogleAuth.set_property(site.google_auth, attrs)
|
|
|> Repo.update!()
|
|
|
|
conn
|
|
|> put_flash(:success, "Google integration saved succesfully")
|
|
|> redirect(to: "/#{URI.encode_www_form(site.domain)}/settings#google-auth")
|
|
end
|
|
|
|
def update_settings(conn, %{"website" => website, "site" => site_params}) do
|
|
site = Sites.get_for_user!(conn.assigns[:current_user].id, website)
|
|
changeset = site |> Plausible.Site.changeset(site_params)
|
|
res = changeset |> Repo.update()
|
|
|
|
case res do
|
|
{:ok, site} ->
|
|
site_session_key = "authorized_site__" <> site.domain
|
|
|
|
conn
|
|
|> put_session(site_session_key, nil)
|
|
|> put_flash(:success, "Site settings saved succesfully")
|
|
|> redirect(to: "/#{URI.encode_www_form(site.domain)}/settings")
|
|
|
|
{:error, changeset} ->
|
|
render("settings.html", site: site, changeset: changeset)
|
|
end
|
|
end
|
|
|
|
def delete_site(conn, %{"website" => website}) do
|
|
site =
|
|
Sites.get_for_user!(conn.assigns[:current_user].id, website)
|
|
|> Repo.preload(:google_auth)
|
|
|
|
Repo.delete_all(from sm in "site_memberships", where: sm.site_id == ^site.id)
|
|
|
|
if site.google_auth do
|
|
Repo.delete!(site.google_auth)
|
|
end
|
|
|
|
Repo.delete!(site)
|
|
|
|
conn
|
|
|> put_flash(:success, "Site deleted succesfully along with all pageviews")
|
|
|> redirect(to: "/")
|
|
end
|
|
|
|
def make_public(conn, %{"website" => website}) do
|
|
site =
|
|
Sites.get_for_user!(conn.assigns[:current_user].id, website)
|
|
|> Plausible.Site.make_public()
|
|
|> Repo.update!()
|
|
|
|
conn
|
|
|> put_flash(:success, "Congrats! Stats for #{site.domain} are now public.")
|
|
|> redirect(to: "/#{URI.encode_www_form(site.domain)}/settings")
|
|
end
|
|
|
|
def make_private(conn, %{"website" => website}) do
|
|
site =
|
|
Sites.get_for_user!(conn.assigns[:current_user].id, website)
|
|
|> Plausible.Site.make_private()
|
|
|> Repo.update!()
|
|
|
|
conn
|
|
|> put_flash(:success, "Stats for #{site.domain} are now private.")
|
|
|> redirect(to: "/#{URI.encode_www_form(site.domain)}/settings")
|
|
end
|
|
|
|
def enable_weekly_report(conn, %{"website" => website}) do
|
|
site = Sites.get_for_user!(conn.assigns[:current_user].id, website)
|
|
|
|
Plausible.Site.WeeklyReport.changeset(%Plausible.Site.WeeklyReport{}, %{
|
|
site_id: site.id,
|
|
recipients: [conn.assigns[:current_user].email]
|
|
})
|
|
|> Repo.insert!()
|
|
|
|
conn
|
|
|> put_flash(:success, "Success! You will receive an email report every Monday going forward")
|
|
|> redirect(to: "/#{URI.encode_www_form(site.domain)}/settings#email-reports")
|
|
end
|
|
|
|
def disable_weekly_report(conn, %{"website" => website}) do
|
|
site = Sites.get_for_user!(conn.assigns[:current_user].id, website)
|
|
Repo.delete_all(from wr in Plausible.Site.WeeklyReport, where: wr.site_id == ^site.id)
|
|
|
|
conn
|
|
|> put_flash(:success, "Success! You will not receive weekly email reports going forward")
|
|
|> redirect(to: "/#{URI.encode_www_form(site.domain)}/settings#email-reports")
|
|
end
|
|
|
|
def add_weekly_report_recipient(conn, %{"website" => website, "recipient" => recipient}) do
|
|
site = Sites.get_for_user!(conn.assigns[:current_user].id, website)
|
|
|
|
Repo.get_by(Plausible.Site.WeeklyReport, site_id: site.id)
|
|
|> Plausible.Site.WeeklyReport.add_recipient(recipient)
|
|
|> Repo.update!()
|
|
|
|
conn
|
|
|> put_flash(:success, "Succesfully added #{recipient} as a recipient for the weekly report")
|
|
|> redirect(to: "/#{URI.encode_www_form(site.domain)}/settings#email-reports")
|
|
end
|
|
|
|
def remove_weekly_report_recipient(conn, %{"website" => website, "recipient" => recipient}) do
|
|
site = Sites.get_for_user!(conn.assigns[:current_user].id, website)
|
|
|
|
Repo.get_by(Plausible.Site.WeeklyReport, site_id: site.id)
|
|
|> Plausible.Site.WeeklyReport.remove_recipient(recipient)
|
|
|> Repo.update!()
|
|
|
|
conn
|
|
|> put_flash(
|
|
:success,
|
|
"Succesfully removed #{recipient} as a recipient for the weekly report"
|
|
)
|
|
|> redirect(to: "/#{URI.encode_www_form(site.domain)}/settings#email-reports")
|
|
end
|
|
|
|
def enable_monthly_report(conn, %{"website" => website}) do
|
|
site = Sites.get_for_user!(conn.assigns[:current_user].id, website)
|
|
|
|
Plausible.Site.MonthlyReport.changeset(%Plausible.Site.MonthlyReport{}, %{
|
|
site_id: site.id,
|
|
recipients: [conn.assigns[:current_user].email]
|
|
})
|
|
|> Repo.insert!()
|
|
|
|
conn
|
|
|> put_flash(:success, "Success! You will receive an email report every month going forward")
|
|
|> redirect(to: "/#{URI.encode_www_form(site.domain)}/settings#email-reports")
|
|
end
|
|
|
|
def disable_monthly_report(conn, %{"website" => website}) do
|
|
site = Sites.get_for_user!(conn.assigns[:current_user].id, website)
|
|
Repo.delete_all(from mr in Plausible.Site.MonthlyReport, where: mr.site_id == ^site.id)
|
|
|
|
conn
|
|
|> put_flash(:success, "Success! You will not receive monthly email reports going forward")
|
|
|> redirect(to: "/#{URI.encode_www_form(site.domain)}/settings#email-reports")
|
|
end
|
|
|
|
def add_monthly_report_recipient(conn, %{"website" => website, "recipient" => recipient}) do
|
|
site = Sites.get_for_user!(conn.assigns[:current_user].id, website)
|
|
|
|
Repo.get_by(Plausible.Site.MonthlyReport, site_id: site.id)
|
|
|> Plausible.Site.MonthlyReport.add_recipient(recipient)
|
|
|> Repo.update!()
|
|
|
|
conn
|
|
|> put_flash(:success, "Succesfully added #{recipient} as a recipient for the monthly report")
|
|
|> redirect(to: "/#{URI.encode_www_form(site.domain)}/settings#email-reports")
|
|
end
|
|
|
|
def remove_monthly_report_recipient(conn, %{"website" => website, "recipient" => recipient}) do
|
|
site = Sites.get_for_user!(conn.assigns[:current_user].id, website)
|
|
|
|
Repo.get_by(Plausible.Site.MonthlyReport, site_id: site.id)
|
|
|> Plausible.Site.MonthlyReport.remove_recipient(recipient)
|
|
|> Repo.update!()
|
|
|
|
conn
|
|
|> put_flash(
|
|
:success,
|
|
"Succesfully removed #{recipient} as a recipient for the monthly report"
|
|
)
|
|
|> redirect(to: "/#{URI.encode_www_form(site.domain)}/settings#email-reports")
|
|
end
|
|
|
|
def new_shared_link(conn, %{"website" => website}) do
|
|
site = Sites.get_for_user!(conn.assigns[:current_user].id, website)
|
|
changeset = Plausible.Site.SharedLink.changeset(%Plausible.Site.SharedLink{}, %{})
|
|
|
|
conn
|
|
|> assign(:skip_plausible_tracking, true)
|
|
|> render("new_shared_link.html",
|
|
site: site,
|
|
changeset: changeset,
|
|
layout: {PlausibleWeb.LayoutView, "focus.html"}
|
|
)
|
|
end
|
|
|
|
def create_shared_link(conn, %{"website" => website, "shared_link" => link}) do
|
|
site = Sites.get_for_user!(conn.assigns[:current_user].id, website)
|
|
|
|
changes =
|
|
Plausible.Site.SharedLink.changeset(
|
|
%Plausible.Site.SharedLink{
|
|
site_id: site.id,
|
|
slug: Nanoid.generate()
|
|
},
|
|
link
|
|
)
|
|
|
|
case Repo.insert(changes) do
|
|
{:ok, _created} ->
|
|
redirect(conn, to: "/#{URI.encode_www_form(site.domain)}/settings#visibility")
|
|
|
|
{:error, changeset} ->
|
|
conn
|
|
|> assign(:skip_plausible_tracking, true)
|
|
|> render("new_shared_link.html",
|
|
site: site,
|
|
changeset: changeset,
|
|
layout: {PlausibleWeb.LayoutView, "focus.html"}
|
|
)
|
|
end
|
|
end
|
|
|
|
def delete_shared_link(conn, %{"website" => website, "slug" => slug}) do
|
|
site = Sites.get_for_user!(conn.assigns[:current_user].id, website)
|
|
|
|
Repo.get_by(Plausible.Site.SharedLink, slug: slug)
|
|
|> Repo.delete!()
|
|
|
|
redirect(conn, to: "/#{URI.encode_www_form(site.domain)}/settings#visibility")
|
|
end
|
|
|
|
def new_custom_domain(conn, %{"website" => website}) do
|
|
site = Sites.get_for_user!(conn.assigns[:current_user].id, website)
|
|
changeset = Plausible.Site.CustomDomain.changeset(%Plausible.Site.CustomDomain{}, %{})
|
|
|
|
conn
|
|
|> assign(:skip_plausible_tracking, true)
|
|
|> render("new_custom_domain.html",
|
|
site: site,
|
|
changeset: changeset,
|
|
layout: {PlausibleWeb.LayoutView, "focus.html"}
|
|
)
|
|
end
|
|
|
|
def custom_domain_dns_setup(conn, %{"website" => website}) do
|
|
site =
|
|
Sites.get_for_user!(conn.assigns[:current_user].id, website)
|
|
|> Repo.preload(:custom_domain)
|
|
|
|
conn
|
|
|> assign(:skip_plausible_tracking, true)
|
|
|> render("custom_domain_dns_setup.html",
|
|
site: site,
|
|
layout: {PlausibleWeb.LayoutView, "focus.html"}
|
|
)
|
|
end
|
|
|
|
def custom_domain_snippet(conn, %{"website" => website}) do
|
|
site =
|
|
Sites.get_for_user!(conn.assigns[:current_user].id, website)
|
|
|> Repo.preload(:custom_domain)
|
|
|
|
conn
|
|
|> assign(:skip_plausible_tracking, true)
|
|
|> render("custom_domain_snippet.html",
|
|
site: site,
|
|
layout: {PlausibleWeb.LayoutView, "focus.html"}
|
|
)
|
|
end
|
|
|
|
def add_custom_domain(conn, %{"website" => website, "custom_domain" => domain}) do
|
|
site = Sites.get_for_user!(conn.assigns[:current_user].id, website)
|
|
|
|
case Sites.add_custom_domain(site, domain["domain"]) do
|
|
{:ok, _custom_domain} ->
|
|
redirect(conn, to: "/sites/#{URI.encode_www_form(site.domain)}/custom-domains/dns-setup")
|
|
|
|
{:error, changeset} ->
|
|
conn
|
|
|> assign(:skip_plausible_tracking, true)
|
|
|> render("new_custom_domain.html",
|
|
site: site,
|
|
changeset: changeset,
|
|
layout: {PlausibleWeb.LayoutView, "focus.html"}
|
|
)
|
|
end
|
|
end
|
|
|
|
defp insert_site(user_id, params) do
|
|
site_changeset = Plausible.Site.changeset(%Plausible.Site{}, params)
|
|
|
|
Ecto.Multi.new()
|
|
|> Ecto.Multi.insert(:site, site_changeset)
|
|
|> Ecto.Multi.run(:site_membership, fn repo, %{site: site} ->
|
|
membership_changeset =
|
|
Plausible.Site.Membership.changeset(%Plausible.Site.Membership{}, %{
|
|
site_id: site.id,
|
|
user_id: user_id
|
|
})
|
|
|
|
repo.insert(membership_changeset)
|
|
end)
|
|
|> Repo.transaction()
|
|
end
|
|
end
|