mirror of
https://github.com/plausible/analytics.git
synced 2024-11-26 11:44:03 +03:00
4b21b4e6d0
* Remove Firewall plug; redundant at infra level * Update changelog
557 lines
16 KiB
Elixir
557 lines
16 KiB
Elixir
import Config
|
||
import Plausible.ConfigHelpers
|
||
require Logger
|
||
|
||
if config_env() in [:dev, :test] do
|
||
Envy.load(["config/.env.#{config_env()}"])
|
||
end
|
||
|
||
config_dir = System.get_env("CONFIG_DIR", "/run/secrets")
|
||
|
||
# Listen IP supports IPv4 and IPv6 addresses.
|
||
listen_ip =
|
||
(
|
||
str = get_var_from_path_or_env(config_dir, "LISTEN_IP") || "127.0.0.1"
|
||
|
||
case :inet.parse_address(String.to_charlist(str)) do
|
||
{:ok, ip_addr} ->
|
||
ip_addr
|
||
|
||
{:error, reason} ->
|
||
raise "Invalid LISTEN_IP '#{str}' error: #{inspect(reason)}"
|
||
end
|
||
)
|
||
|
||
# System.get_env does not accept a non string default
|
||
port = get_var_from_path_or_env(config_dir, "PORT") || 8000
|
||
|
||
base_url = get_var_from_path_or_env(config_dir, "BASE_URL")
|
||
|
||
if !base_url do
|
||
raise "BASE_URL configuration option is required. See https://plausible.io/docs/self-hosting-configuration#server"
|
||
end
|
||
|
||
base_url = URI.parse(base_url)
|
||
|
||
if base_url.scheme not in ["http", "https"] do
|
||
raise "BASE_URL must start with `http` or `https`. Currently configured as `#{System.get_env("BASE_URL")}`"
|
||
end
|
||
|
||
secret_key_base = get_var_from_path_or_env(config_dir, "SECRET_KEY_BASE", nil)
|
||
|
||
case secret_key_base do
|
||
nil ->
|
||
raise "SECRET_KEY_BASE configuration option is required. See https://plausible.io/docs/self-hosting-configuration#server"
|
||
|
||
key when byte_size(key) < 32 ->
|
||
raise "SECRET_KEY_BASE must be at least 32 bytes long. See https://plausible.io/docs/self-hosting-configuration#server"
|
||
|
||
_ ->
|
||
nil
|
||
end
|
||
|
||
db_url =
|
||
get_var_from_path_or_env(
|
||
config_dir,
|
||
"DATABASE_URL",
|
||
"postgres://postgres:postgres@plausible_db:5432/plausible_db"
|
||
)
|
||
|
||
db_socket_dir = get_var_from_path_or_env(config_dir, "DATABASE_SOCKET_DIR")
|
||
|
||
super_admin_user_ids =
|
||
get_var_from_path_or_env(config_dir, "ADMIN_USER_IDS", "")
|
||
|> String.split(",")
|
||
|> Enum.map(fn id -> Integer.parse(id) end)
|
||
|> Enum.map(fn
|
||
{int, ""} -> int
|
||
_ -> nil
|
||
end)
|
||
|> Enum.filter(& &1)
|
||
|
||
env = get_var_from_path_or_env(config_dir, "ENVIRONMENT", "prod")
|
||
mailer_adapter = get_var_from_path_or_env(config_dir, "MAILER_ADAPTER", "Bamboo.SMTPAdapter")
|
||
mailer_email = get_var_from_path_or_env(config_dir, "MAILER_EMAIL", "hello@plausible.local")
|
||
app_version = get_var_from_path_or_env(config_dir, "APP_VERSION", "0.0.1")
|
||
|
||
ch_db_url =
|
||
get_var_from_path_or_env(
|
||
config_dir,
|
||
"CLICKHOUSE_DATABASE_URL",
|
||
"http://plausible_events_db:8123/plausible_events_db"
|
||
)
|
||
|
||
{ingest_pool_size, ""} =
|
||
get_var_from_path_or_env(
|
||
config_dir,
|
||
"CLICKHOUSE_INGEST_POOL_SIZE",
|
||
"5"
|
||
)
|
||
|> Integer.parse()
|
||
|
||
{ch_flush_interval_ms, ""} =
|
||
config_dir
|
||
|> get_var_from_path_or_env("CLICKHOUSE_FLUSH_INTERVAL_MS", "5000")
|
||
|> Integer.parse()
|
||
|
||
{ch_max_buffer_size, ""} =
|
||
config_dir
|
||
|> get_var_from_path_or_env("CLICKHOUSE_MAX_BUFFER_SIZE", "10000")
|
||
|> Integer.parse()
|
||
|
||
### Mandatory params End
|
||
|
||
build_metadata_raw = get_var_from_path_or_env(config_dir, "BUILD_METADATA", "{}")
|
||
|
||
build_metadata =
|
||
case Jason.decode(build_metadata_raw) do
|
||
{:ok, build_metadata} ->
|
||
build_metadata
|
||
|
||
{:error, error} ->
|
||
error = Exception.format(:error, error)
|
||
|
||
Logger.warn("""
|
||
failed to parse $BUILD_METADATA: #{error}
|
||
|
||
$BUILD_METADATA is set to #{build_metadata_raw}\
|
||
""")
|
||
|
||
Logger.warn("falling back to empty build metadata, as if $BUILD_METADATA was set to {}")
|
||
|
||
_fallback = %{}
|
||
end
|
||
|
||
runtime_metadata = [
|
||
version: get_in(build_metadata, ["labels", "org.opencontainers.image.version"]),
|
||
commit: get_in(build_metadata, ["labels", "org.opencontainers.image.revision"]),
|
||
created: get_in(build_metadata, ["labels", "org.opencontainers.image.created"]),
|
||
tags: get_in(build_metadata, ["tags"])
|
||
]
|
||
|
||
config :plausible, :runtime_metadata, runtime_metadata
|
||
|
||
sentry_dsn = get_var_from_path_or_env(config_dir, "SENTRY_DSN")
|
||
honeycomb_api_key = get_var_from_path_or_env(config_dir, "HONEYCOMB_API_KEY")
|
||
honeycomb_dataset = get_var_from_path_or_env(config_dir, "HONEYCOMB_DATASET")
|
||
paddle_auth_code = get_var_from_path_or_env(config_dir, "PADDLE_VENDOR_AUTH_CODE")
|
||
paddle_vendor_id = get_var_from_path_or_env(config_dir, "PADDLE_VENDOR_ID")
|
||
google_cid = get_var_from_path_or_env(config_dir, "GOOGLE_CLIENT_ID")
|
||
google_secret = get_var_from_path_or_env(config_dir, "GOOGLE_CLIENT_SECRET")
|
||
postmark_api_key = get_var_from_path_or_env(config_dir, "POSTMARK_API_KEY")
|
||
|
||
cron_enabled =
|
||
config_dir
|
||
|> get_var_from_path_or_env("CRON_ENABLED", "false")
|
||
|> String.to_existing_atom()
|
||
|
||
geolite2_country_db =
|
||
get_var_from_path_or_env(
|
||
config_dir,
|
||
"GEOLITE2_COUNTRY_DB",
|
||
Application.app_dir(:plausible, "/priv/geodb/dbip-country.mmdb.gz")
|
||
)
|
||
|
||
ip_geolocation_db = get_var_from_path_or_env(config_dir, "IP_GEOLOCATION_DB", geolite2_country_db)
|
||
geonames_source_file = get_var_from_path_or_env(config_dir, "GEONAMES_SOURCE_FILE")
|
||
maxmind_license_key = get_var_from_path_or_env(config_dir, "MAXMIND_LICENSE_KEY")
|
||
maxmind_edition = get_var_from_path_or_env(config_dir, "MAXMIND_EDITION", "GeoLite2-City")
|
||
|
||
if System.get_env("DISABLE_AUTH") do
|
||
require Logger
|
||
Logger.warn("DISABLE_AUTH env var is no longer supported")
|
||
end
|
||
|
||
enable_email_verification =
|
||
config_dir
|
||
|> get_var_from_path_or_env("ENABLE_EMAIL_VERIFICATION", "false")
|
||
|> String.to_existing_atom()
|
||
|
||
disable_registration =
|
||
config_dir
|
||
|> get_var_from_path_or_env("DISABLE_REGISTRATION", "false")
|
||
|> String.to_existing_atom()
|
||
|
||
if disable_registration not in [true, false, :invite_only] do
|
||
raise "DISABLE_REGISTRATION must be one of `true`, `false`, or `invite_only`. See https://plausible.io/docs/self-hosting-configuration#server"
|
||
end
|
||
|
||
hcaptcha_sitekey = get_var_from_path_or_env(config_dir, "HCAPTCHA_SITEKEY")
|
||
hcaptcha_secret = get_var_from_path_or_env(config_dir, "HCAPTCHA_SECRET")
|
||
|
||
log_level =
|
||
config_dir
|
||
|> get_var_from_path_or_env("LOG_LEVEL", "warn")
|
||
|> String.to_existing_atom()
|
||
|
||
is_selfhost =
|
||
config_dir
|
||
|> get_var_from_path_or_env("SELFHOST", "true")
|
||
|> String.to_existing_atom()
|
||
|
||
custom_script_name =
|
||
config_dir
|
||
|> get_var_from_path_or_env("CUSTOM_SCRIPT_NAME", "script")
|
||
|
||
{site_limit, ""} =
|
||
config_dir
|
||
|> get_var_from_path_or_env("SITE_LIMIT", "50")
|
||
|> Integer.parse()
|
||
|
||
site_limit_exempt =
|
||
config_dir
|
||
|> get_var_from_path_or_env("SITE_LIMIT_EXEMPT", "")
|
||
|> String.split(",")
|
||
|> Enum.map(&String.trim/1)
|
||
|
||
disable_cron =
|
||
config_dir
|
||
|> get_var_from_path_or_env("DISABLE_CRON", "false")
|
||
|> String.to_existing_atom()
|
||
|
||
config :plausible,
|
||
environment: env,
|
||
mailer_email: mailer_email,
|
||
super_admin_user_ids: super_admin_user_ids,
|
||
site_limit: site_limit,
|
||
site_limit_exempt: site_limit_exempt,
|
||
is_selfhost: is_selfhost,
|
||
custom_script_name: custom_script_name
|
||
|
||
config :plausible, :selfhost,
|
||
enable_email_verification: enable_email_verification,
|
||
disable_registration: disable_registration
|
||
|
||
config :plausible, PlausibleWeb.Endpoint,
|
||
url: [scheme: base_url.scheme, host: base_url.host, path: base_url.path, port: base_url.port],
|
||
http: [
|
||
port: port,
|
||
ip: listen_ip,
|
||
transport_options: [max_connections: :infinity],
|
||
protocol_options: [max_request_line_length: 8192, max_header_value_length: 8192]
|
||
],
|
||
secret_key_base: secret_key_base
|
||
|
||
maybe_ipv6 = if System.get_env("ECTO_IPV6"), do: [:inet6], else: []
|
||
|
||
if is_nil(db_socket_dir) do
|
||
config :plausible, Plausible.Repo,
|
||
url: db_url,
|
||
socket_options: maybe_ipv6
|
||
else
|
||
config :plausible, Plausible.Repo,
|
||
socket_dir: db_socket_dir,
|
||
database: get_var_from_path_or_env(config_dir, "DATABASE_NAME", "plausible")
|
||
end
|
||
|
||
included_environments = if sentry_dsn, do: ["prod", "staging", "dev"], else: []
|
||
sentry_app_version = runtime_metadata[:version] || app_version
|
||
|
||
config :sentry,
|
||
dsn: sentry_dsn,
|
||
environment_name: env,
|
||
included_environments: included_environments,
|
||
release: sentry_app_version,
|
||
tags: %{app_version: sentry_app_version},
|
||
enable_source_code_context: true,
|
||
root_source_code_path: [File.cwd!()],
|
||
client: Plausible.Sentry.Client,
|
||
send_max_attempts: 1,
|
||
filter: Plausible.SentryFilter,
|
||
before_send_event: {Plausible.SentryFilter, :before_send}
|
||
|
||
config :logger, Sentry.LoggerBackend,
|
||
capture_log_messages: true,
|
||
level: :error
|
||
|
||
config :plausible, :paddle,
|
||
vendor_auth_code: paddle_auth_code,
|
||
vendor_id: paddle_vendor_id
|
||
|
||
config :plausible, :google,
|
||
client_id: google_cid,
|
||
client_secret: google_secret,
|
||
api_url: "https://www.googleapis.com",
|
||
reporting_api_url: "https://analyticsreporting.googleapis.com",
|
||
max_buffer_size: get_int_from_path_or_env(config_dir, "GOOGLE_MAX_BUFFER_SIZE", 10_000)
|
||
|
||
config :plausible, Plausible.ClickhouseRepo,
|
||
loggers: [Ecto.LogEntry],
|
||
queue_target: 500,
|
||
queue_interval: 2000,
|
||
url: ch_db_url
|
||
|
||
config :plausible, Plausible.IngestRepo,
|
||
loggers: [Ecto.LogEntry],
|
||
queue_target: 500,
|
||
queue_interval: 2000,
|
||
url: ch_db_url,
|
||
flush_interval_ms: ch_flush_interval_ms,
|
||
max_buffer_size: ch_max_buffer_size,
|
||
pool_size: ingest_pool_size
|
||
|
||
config :plausible, Plausible.AsyncInsertRepo,
|
||
loggers: [Ecto.LogEntry],
|
||
queue_target: 500,
|
||
queue_interval: 2000,
|
||
url: ch_db_url,
|
||
pool_size: 1,
|
||
clickhouse_settings: [
|
||
async_insert: 1,
|
||
wait_for_async_insert: 0
|
||
]
|
||
|
||
case mailer_adapter do
|
||
"Bamboo.PostmarkAdapter" ->
|
||
config :plausible, Plausible.Mailer,
|
||
adapter: Bamboo.PostmarkAdapter,
|
||
request_options: [recv_timeout: 10_000],
|
||
api_key: get_var_from_path_or_env(config_dir, "POSTMARK_API_KEY")
|
||
|
||
"Bamboo.MailgunAdapter" ->
|
||
config :plausible, Plausible.Mailer,
|
||
adapter: Bamboo.MailgunAdapter,
|
||
hackney_opts: [recv_timeout: :timer.seconds(10)],
|
||
api_key: get_var_from_path_or_env(config_dir, "MAILGUN_API_KEY"),
|
||
domain: get_var_from_path_or_env(config_dir, "MAILGUN_DOMAIN")
|
||
|
||
"Bamboo.MandrillAdapter" ->
|
||
config :plausible, Plausible.Mailer,
|
||
adapter: Bamboo.MandrillAdapter,
|
||
hackney_opts: [recv_timeout: :timer.seconds(10)],
|
||
api_key: get_var_from_path_or_env(config_dir, "MANDRILL_API_KEY")
|
||
|
||
"Bamboo.SendGridAdapter" ->
|
||
config :plausible, Plausible.Mailer,
|
||
adapter: Bamboo.SendGridAdapter,
|
||
hackney_opts: [recv_timeout: :timer.seconds(10)],
|
||
api_key: get_var_from_path_or_env(config_dir, "SENDGRID_API_KEY")
|
||
|
||
"Bamboo.SMTPAdapter" ->
|
||
config :plausible, Plausible.Mailer,
|
||
adapter: Bamboo.SMTPAdapter,
|
||
server: get_var_from_path_or_env(config_dir, "SMTP_HOST_ADDR", "mail"),
|
||
hostname: base_url.host,
|
||
port: get_var_from_path_or_env(config_dir, "SMTP_HOST_PORT", "25"),
|
||
username: get_var_from_path_or_env(config_dir, "SMTP_USER_NAME"),
|
||
password: get_var_from_path_or_env(config_dir, "SMTP_USER_PWD"),
|
||
tls: :if_available,
|
||
allowed_tls_versions: [:tlsv1, :"tlsv1.1", :"tlsv1.2"],
|
||
ssl: get_var_from_path_or_env(config_dir, "SMTP_HOST_SSL_ENABLED") || false,
|
||
retries: get_var_from_path_or_env(config_dir, "SMTP_RETRIES") || 2,
|
||
no_mx_lookups: get_var_from_path_or_env(config_dir, "SMTP_MX_LOOKUPS_ENABLED") || true
|
||
|
||
"Bamboo.LocalAdapter" ->
|
||
config :plausible, Plausible.Mailer, adapter: Bamboo.LocalAdapter
|
||
|
||
"Bamboo.TestAdapter" ->
|
||
config :plausible, Plausible.Mailer, adapter: Bamboo.TestAdapter
|
||
|
||
_ ->
|
||
raise ArgumentError, """
|
||
Unknown mailer_adapter: #{inspect(mailer_adapter)}
|
||
|
||
Please see https://hexdocs.pm/bamboo/readme.html#available-adapters
|
||
for the list of available adapters that ship with Bamboo
|
||
"""
|
||
end
|
||
|
||
base_cron = [
|
||
# Daily at midnight
|
||
{"0 0 * * *", Plausible.Workers.RotateSalts},
|
||
# hourly
|
||
{"0 * * * *", Plausible.Workers.ScheduleEmailReports},
|
||
# hourly
|
||
{"0 * * * *", Plausible.Workers.SendSiteSetupEmails},
|
||
# Daily at midday
|
||
{"0 12 * * *", Plausible.Workers.SendCheckStatsEmails},
|
||
# Every 15 minutes
|
||
{"*/15 * * * *", Plausible.Workers.SpikeNotifier},
|
||
# Every day at midnight
|
||
{"0 0 * * *", Plausible.Workers.CleanEmailVerificationCodes},
|
||
# Every day at 1am
|
||
{"0 1 * * *", Plausible.Workers.CleanInvitations}
|
||
]
|
||
|
||
cloud_cron = [
|
||
# Daily at midday
|
||
{"0 12 * * *", Plausible.Workers.SendTrialNotifications},
|
||
# Daily at 14
|
||
{"0 14 * * *", Plausible.Workers.CheckUsage},
|
||
# Daily at 15
|
||
{"0 15 * * *", Plausible.Workers.NotifyAnnualRenewal},
|
||
# Every midnight
|
||
{"0 0 * * *", Plausible.Workers.LockSites}
|
||
]
|
||
|
||
crontab = if(is_selfhost, do: base_cron, else: base_cron ++ cloud_cron)
|
||
|
||
base_queues = [
|
||
rotate_salts: 1,
|
||
schedule_email_reports: 1,
|
||
send_email_reports: 1,
|
||
spike_notifications: 1,
|
||
check_stats_emails: 1,
|
||
site_setup_emails: 1,
|
||
clean_email_verification_codes: 1,
|
||
clean_invitations: 1,
|
||
google_analytics_imports: 1
|
||
]
|
||
|
||
cloud_queues = [
|
||
trial_notification_emails: 1,
|
||
check_usage: 1,
|
||
notify_annual_renewal: 1,
|
||
lock_sites: 1
|
||
]
|
||
|
||
queues = if(is_selfhost, do: base_queues, else: base_queues ++ cloud_queues)
|
||
cron_enabled = !disable_cron
|
||
|
||
thirty_days_in_seconds = 60 * 60 * 24 * 30
|
||
|
||
cond do
|
||
config_env() == :prod ->
|
||
config :plausible, Oban,
|
||
repo: Plausible.Repo,
|
||
plugins: [
|
||
# Keep 30 days history
|
||
{Oban.Plugins.Pruner, max_age: thirty_days_in_seconds},
|
||
{Oban.Plugins.Cron, crontab: if(cron_enabled, do: crontab, else: [])},
|
||
# Rescue orphaned jobs after 2 hours
|
||
{Oban.Plugins.Lifeline, rescue_after: :timer.minutes(120)},
|
||
{Oban.Plugins.Stager, interval: :timer.seconds(5)}
|
||
],
|
||
queues: if(cron_enabled, do: queues, else: []),
|
||
peer: if(cron_enabled, do: Oban.Peers.Postgres, else: false)
|
||
|
||
true ->
|
||
config :plausible, Oban,
|
||
repo: Plausible.Repo,
|
||
queues: queues,
|
||
plugins: false
|
||
end
|
||
|
||
config :plausible, :hcaptcha,
|
||
sitekey: hcaptcha_sitekey,
|
||
secret: hcaptcha_secret
|
||
|
||
config :plausible, Plausible.Sentry.Client,
|
||
finch_request_opts: [
|
||
pool_timeout: get_int_from_path_or_env(config_dir, "SENTRY_FINCH_POOL_TIMEOUT", 5000),
|
||
receive_timeout: get_int_from_path_or_env(config_dir, "SENTRY_FINCH_RECEIVE_TIMEOUT", 15000)
|
||
]
|
||
|
||
config :ref_inspector,
|
||
init: {Plausible.Release, :configure_ref_inspector}
|
||
|
||
config :ua_inspector,
|
||
init: {Plausible.Release, :configure_ua_inspector}
|
||
|
||
config :hammer,
|
||
backend: {Hammer.Backend.ETS, [expiry_ms: 60_000 * 60 * 4, cleanup_interval_ms: 60_000 * 10]}
|
||
|
||
config :kaffy,
|
||
otp_app: :plausible,
|
||
ecto_repo: Plausible.Repo,
|
||
router: PlausibleWeb.Router,
|
||
admin_title: "Plausible Admin",
|
||
resources: [
|
||
auth: [
|
||
resources: [
|
||
user: [schema: Plausible.Auth.User, admin: Plausible.Auth.UserAdmin],
|
||
api_key: [schema: Plausible.Auth.ApiKey, admin: Plausible.Auth.ApiKeyAdmin]
|
||
]
|
||
],
|
||
sites: [
|
||
resources: [
|
||
site: [schema: Plausible.Site, admin: Plausible.SiteAdmin]
|
||
]
|
||
],
|
||
billing: [
|
||
resources: [
|
||
enterprise_plan: [
|
||
schema: Plausible.Billing.EnterprisePlan,
|
||
admin: Plausible.Billing.EnterprisePlanAdmin
|
||
]
|
||
]
|
||
]
|
||
]
|
||
|
||
geo_opts =
|
||
cond do
|
||
maxmind_license_key ->
|
||
[
|
||
license_key: maxmind_license_key,
|
||
edition: maxmind_edition,
|
||
async: true
|
||
]
|
||
|
||
ip_geolocation_db ->
|
||
[path: ip_geolocation_db]
|
||
|
||
true ->
|
||
raise """
|
||
Missing geolocation database configuration.
|
||
|
||
Please set the IP_GEOLOCATION_DB environment value to the location of
|
||
your IP geolocation .mmdb file:
|
||
|
||
IP_GEOLOCATION_DB=/etc/plausible/dbip-city.mmdb
|
||
|
||
Or authenticate with MaxMind by
|
||
configuring MAXMIND_LICENSE_KEY and (optionally) MAXMIND_EDITION environment
|
||
variables:
|
||
|
||
MAXMIND_LICENSE_KEY=LNpsJCCKPis6XvBP
|
||
MAXMIND_EDITION=GeoLite2-City # this is the default edition
|
||
|
||
"""
|
||
end
|
||
|
||
config :plausible, Plausible.Geo, geo_opts
|
||
|
||
if geonames_source_file do
|
||
config :location, :geonames_source_file, geonames_source_file
|
||
end
|
||
|
||
config :logger,
|
||
level: log_level,
|
||
backends: [:console]
|
||
|
||
if honeycomb_api_key && honeycomb_dataset do
|
||
config :opentelemetry,
|
||
resource: Plausible.OpenTelemetry.resource_attributes(runtime_metadata),
|
||
sampler: {Plausible.OpenTelemetry.Sampler, nil},
|
||
span_processor: :batch,
|
||
traces_exporter: :otlp
|
||
|
||
config :opentelemetry_exporter,
|
||
otlp_protocol: :grpc,
|
||
otlp_endpoint: "https://api.honeycomb.io:443",
|
||
otlp_headers: [
|
||
{"x-honeycomb-team", honeycomb_api_key},
|
||
{"x-honeycomb-dataset", honeycomb_dataset}
|
||
]
|
||
else
|
||
config :opentelemetry,
|
||
sampler: :always_off,
|
||
traces_exporter: :none
|
||
end
|
||
|
||
config :tzdata,
|
||
:data_dir,
|
||
get_var_from_path_or_env(config_dir, "STORAGE_DIR", Application.app_dir(:tzdata, "priv"))
|
||
|
||
promex_disabled? =
|
||
config_dir
|
||
|> get_var_from_path_or_env("PROMEX_DISABLED", "true")
|
||
|> String.to_existing_atom()
|
||
|
||
config :plausible, Plausible.PromEx,
|
||
disabled: promex_disabled?,
|
||
manual_metrics_start_delay: :no_delay,
|
||
drop_metrics_groups: [],
|
||
grafana: :disabled,
|
||
metrics_server: :disabled
|