62dd1b85b7
Just cancel the job Closes #1180 Signed-off-by: Thomas Citharel <tcit@tcit.fr>
149 lines
4.7 KiB
Elixir
149 lines
4.7 KiB
Elixir
defmodule Mobilizon.Service.Workers.Notification do
|
|
@moduledoc """
|
|
Worker to send notifications
|
|
"""
|
|
|
|
alias Mobilizon.{Actors, Events, Users}
|
|
alias Mobilizon.Actors.Actor
|
|
alias Mobilizon.Events.{Event, Participant}
|
|
alias Mobilizon.Storage.Page
|
|
alias Mobilizon.Users.{Setting, User}
|
|
alias Mobilizon.Web.Email.{Mailer, Notification}
|
|
require Logger
|
|
|
|
import Mobilizon.Service.DateTime,
|
|
only: [
|
|
datetime_tz_convert: 2
|
|
]
|
|
|
|
use Mobilizon.Service.Workers.Helper, queue: "mailers"
|
|
|
|
@impl Oban.Worker
|
|
def perform(%Job{
|
|
args: %{"op" => "before_event_notification", "participant_id" => participant_id}
|
|
}) do
|
|
with %Participant{} = participant <- Events.get_participant(participant_id),
|
|
%Event{status: :confirmed} = event <-
|
|
Events.get_event_with_preload!(participant.event_id),
|
|
%Actor{user_id: user_id} = actor when not is_nil(user_id) <-
|
|
Actors.get_actor_with_preload!(participant.actor_id) do
|
|
%User{email: email, locale: locale, settings: %Setting{notification_before_event: true}} =
|
|
Users.get_user_with_settings!(user_id)
|
|
|
|
email
|
|
|> Notification.before_event_notification(
|
|
%Participant{participant | event: event, actor: actor},
|
|
locale
|
|
)
|
|
|> Mailer.send_email()
|
|
|
|
:ok
|
|
end
|
|
end
|
|
|
|
def perform(%Job{
|
|
args: %{"op" => "on_day_notification", "user_id" => user_id}
|
|
}) do
|
|
with %User{locale: locale, settings: %Setting{timezone: timezone, notification_on_day: true}} =
|
|
user <- Users.get_user_with_settings!(user_id),
|
|
{start, tomorrow} <- calculate_start_end(1, timezone || "Etc/UTC"),
|
|
%Page{
|
|
elements: participations,
|
|
total: total
|
|
}
|
|
when total > 0 <-
|
|
Events.list_participations_for_user(user_id, start, tomorrow, 1, 5),
|
|
participations <-
|
|
Enum.filter(participations, fn participation ->
|
|
participation.event.status == :confirmed
|
|
end),
|
|
true <- length(participations) > 0,
|
|
participations <-
|
|
Enum.map(participations, fn participation ->
|
|
%Event{} = event = Events.get_event_with_preload!(participation.event_id)
|
|
%Participant{participation | event: event}
|
|
end) do
|
|
user
|
|
|> Notification.on_day_notification(participations, total, locale)
|
|
|> Mailer.send_email()
|
|
|
|
:ok
|
|
else
|
|
_ -> :ok
|
|
end
|
|
end
|
|
|
|
def perform(%Job{
|
|
args: %{"op" => "weekly_notification", "user_id" => user_id}
|
|
}) do
|
|
with %User{
|
|
locale: locale,
|
|
settings: %Setting{timezone: timezone, notification_each_week: true}
|
|
} = user <- Users.get_user_with_settings!(user_id),
|
|
{start, end_week} <- calculate_start_end(7, timezone || "Etc/UTC"),
|
|
%Page{
|
|
elements: participations,
|
|
total: total
|
|
}
|
|
when total > 0 <-
|
|
Events.list_participations_for_user(user_id, start, end_week, 1, 5),
|
|
participations <-
|
|
Enum.filter(participations, fn participation ->
|
|
participation.event.status == :confirmed
|
|
end),
|
|
true <- length(participations) > 0,
|
|
participations <-
|
|
Enum.map(participations, fn participation ->
|
|
%Event{} = event = Events.get_event_with_preload!(participation.event_id)
|
|
%Participant{participation | event: event}
|
|
end) do
|
|
user
|
|
|> Notification.weekly_notification(participations, total, locale)
|
|
|> Mailer.send_email()
|
|
|
|
:ok
|
|
else
|
|
_err ->
|
|
:ok
|
|
end
|
|
end
|
|
|
|
def perform(%Job{
|
|
args: %{
|
|
"op" => "pending_participation_notification",
|
|
"user_id" => user_id,
|
|
"event_id" => event_id
|
|
}
|
|
}) do
|
|
with %User{} = user <- Users.get_user_with_settings!(user_id),
|
|
{:ok, %Event{} = event} <- Events.get_event_with_preload(event_id),
|
|
%Page{total: total} when total > 0 <-
|
|
Events.list_participants_for_event(event_id, [:not_approved]) do
|
|
user
|
|
|> Notification.pending_participation_notification(event, total)
|
|
|> Mailer.send_email()
|
|
|
|
:ok
|
|
else
|
|
{:error, :event_not_found} ->
|
|
{:cancel, :event_participation_not_found}
|
|
|
|
err ->
|
|
Logger.debug(inspect(err))
|
|
err
|
|
end
|
|
end
|
|
|
|
defp calculate_start_end(days, timezone) do
|
|
now = DateTime.utc_now()
|
|
%DateTime{} = now_shifted = datetime_tz_convert(now, timezone)
|
|
start = %{now_shifted | hour: 8, minute: 0, second: 0, microsecond: {0, 0}}
|
|
|
|
{:ok, %NaiveDateTime{} = tomorrow} =
|
|
Date.utc_today() |> Date.add(days) |> NaiveDateTime.new(~T[08:00:00])
|
|
|
|
{:ok, %DateTime{} = tomorrow} = DateTime.from_naive(tomorrow, timezone)
|
|
{start, tomorrow}
|
|
end
|
|
end
|