mirror of
https://github.com/plausible/analytics.git
synced 2024-12-25 02:24:55 +03:00
18ddf1e7a0
* Revert "Remove 0-sized queue" This reverts commit6d229b3f70
. * Revert "Temporarily remove notification cron worker (#3732)" This reverts commit447e3929cb
. * Revert "Temporarily disable accept_traffic_until actions (#3730)" This reverts commit58afe3376f
. * Send accept_traffic_until notifications at 8 UTC rather than 5 * Add option for dry-run * Expose dry_run in a dedicated function
109 lines
3.1 KiB
Elixir
109 lines
3.1 KiB
Elixir
defmodule Plausible.Workers.AcceptTrafficUntil do
|
|
@moduledoc """
|
|
A worker meant to be run once a day that sends out e-mail notifications to site
|
|
owners assuming:
|
|
- their sites still receive traffic (i.e. have stats for yesterday)
|
|
- `site.accept_traffic_until` is approaching either tomorrow or exactly in 7 days
|
|
|
|
Users having no sites or sites that receive no traffic, won't be notified.
|
|
We make a tiny effort here to make sure we send the same notification at most once a day.
|
|
"""
|
|
use Oban.Worker, queue: :check_accept_traffic_until
|
|
import Ecto.Query
|
|
|
|
alias Plausible.Auth.User
|
|
alias Plausible.Site
|
|
alias Plausible.Repo
|
|
alias Plausible.ClickhouseRepo
|
|
|
|
def dry_run(date) do
|
|
perform(nil, date, true)
|
|
end
|
|
|
|
@impl Oban.Worker
|
|
def perform(_job, today \\ Date.utc_today(), dry_run? \\ false) do
|
|
tomorrow = today |> Date.add(+1)
|
|
next_week = today |> Date.add(+7)
|
|
|
|
# send at most one notification per user, per day
|
|
sent_today_query =
|
|
from s in "sent_accept_traffic_until_notifications",
|
|
where: s.user_id == parent_as(:user).id and s.sent_on == ^today,
|
|
select: true
|
|
|
|
notifications =
|
|
Repo.all(
|
|
from u in User,
|
|
as: :user,
|
|
join: sm in Site.Membership,
|
|
on: sm.user_id == u.id,
|
|
where: sm.role == :owner,
|
|
where: u.accept_traffic_until == ^tomorrow or u.accept_traffic_until == ^next_week,
|
|
where: not exists(sent_today_query),
|
|
select: %{
|
|
id: u.id,
|
|
email: u.email,
|
|
deadline: u.accept_traffic_until,
|
|
site_ids: fragment("array_agg(?.site_id)", sm),
|
|
name: u.name
|
|
},
|
|
group_by: u.id
|
|
)
|
|
|
|
for notification <- notifications do
|
|
case {has_stats?(notification.site_ids, today), notification.deadline} do
|
|
{true, ^tomorrow} ->
|
|
if dry_run? do
|
|
IO.puts("Will send final notification to #{notification.email}")
|
|
else
|
|
notification
|
|
|> store_sent(today)
|
|
|> PlausibleWeb.Email.approaching_accept_traffic_until_tomorrow()
|
|
|> Plausible.Mailer.send()
|
|
end
|
|
|
|
{true, ^next_week} ->
|
|
if dry_run? do
|
|
IO.puts("Will send weekly notification to #{notification.email}")
|
|
else
|
|
notification
|
|
|> store_sent(today)
|
|
|> PlausibleWeb.Email.approaching_accept_traffic_until()
|
|
|> Plausible.Mailer.send()
|
|
end
|
|
|
|
_ ->
|
|
nil
|
|
end
|
|
end
|
|
|
|
{:ok, Enum.count(notifications)}
|
|
end
|
|
|
|
defp has_stats?(site_ids, today) do
|
|
ago_2d = Date.add(today, -2)
|
|
|
|
ClickhouseRepo.exists?(
|
|
from e in "events_v2",
|
|
where: fragment("toDate(?) >= ?", e.timestamp, ^ago_2d),
|
|
where: e.site_id in ^site_ids
|
|
)
|
|
end
|
|
|
|
defp store_sent(notification, today) do
|
|
Repo.insert_all(
|
|
"sent_accept_traffic_until_notifications",
|
|
[
|
|
%{
|
|
user_id: notification.id,
|
|
sent_on: today
|
|
}
|
|
],
|
|
on_conflict: :nothing,
|
|
conflict_target: [:user_id, :sent_on]
|
|
)
|
|
|
|
notification
|
|
end
|
|
end
|