2020-01-26 21:36:50 +01:00
|
|
|
defmodule Mobilizon.Web.Email.User do
|
2018-07-04 17:16:02 +02:00
|
|
|
@moduledoc """
|
2019-09-07 23:58:53 +02:00
|
|
|
Handles emails sent to users.
|
2018-07-04 17:16:02 +02:00
|
|
|
"""
|
2018-07-04 14:29:17 +02:00
|
|
|
|
2020-01-26 21:36:50 +01:00
|
|
|
use Bamboo.Phoenix, view: Mobilizon.Web.EmailView
|
2019-09-07 23:58:53 +02:00
|
|
|
|
2019-09-23 19:33:58 +02:00
|
|
|
import Bamboo.Phoenix
|
2019-09-07 23:58:53 +02:00
|
|
|
|
2021-08-02 19:31:16 +02:00
|
|
|
import Mobilizon.Web.Gettext, only: [gettext: 1, gettext: 2]
|
2018-07-04 14:29:17 +02:00
|
|
|
|
2020-01-23 00:55:07 +01:00
|
|
|
alias Mobilizon.{Config, Crypto, Users}
|
|
|
|
alias Mobilizon.Storage.Repo
|
2019-09-07 23:58:53 +02:00
|
|
|
alias Mobilizon.Users.User
|
|
|
|
|
2021-07-27 19:47:54 +02:00
|
|
|
alias Mobilizon.Web.Email
|
2019-09-17 02:45:32 +02:00
|
|
|
|
2020-01-23 00:55:07 +01:00
|
|
|
require Logger
|
|
|
|
|
2019-09-07 23:58:53 +02:00
|
|
|
@spec confirmation_email(User.t(), String.t()) :: Bamboo.Email.t()
|
|
|
|
def confirmation_email(
|
|
|
|
%User{email: email, confirmation_token: confirmation_token},
|
|
|
|
locale \\ "en"
|
|
|
|
) do
|
2020-01-28 19:18:33 +01:00
|
|
|
Gettext.put_locale(locale)
|
2019-09-07 23:58:53 +02:00
|
|
|
|
|
|
|
subject =
|
|
|
|
gettext(
|
2019-09-23 19:33:58 +02:00
|
|
|
"Instructions to confirm your Mobilizon account on %{instance}",
|
|
|
|
instance: Config.instance_name()
|
2019-09-07 23:58:53 +02:00
|
|
|
)
|
|
|
|
|
2019-09-23 19:33:58 +02:00
|
|
|
Email.base_email(to: email, subject: subject)
|
|
|
|
|> assign(:locale, locale)
|
2019-09-07 23:58:53 +02:00
|
|
|
|> assign(:token, confirmation_token)
|
2019-09-23 19:33:58 +02:00
|
|
|
|> assign(:subject, subject)
|
2019-09-24 12:09:43 +02:00
|
|
|
|> render(:registration_confirmation)
|
2018-07-04 14:29:17 +02:00
|
|
|
end
|
|
|
|
|
2019-09-07 23:58:53 +02:00
|
|
|
@spec reset_password_email(User.t(), String.t()) :: Bamboo.Email.t()
|
|
|
|
def reset_password_email(
|
|
|
|
%User{email: email, reset_password_token: reset_password_token},
|
|
|
|
locale \\ "en"
|
|
|
|
) do
|
2020-01-28 19:18:33 +01:00
|
|
|
Gettext.put_locale(locale)
|
2018-07-27 10:45:35 +02:00
|
|
|
|
2019-09-07 23:58:53 +02:00
|
|
|
subject =
|
2018-07-27 10:45:35 +02:00
|
|
|
gettext(
|
2019-09-23 19:33:58 +02:00
|
|
|
"Instructions to reset your password on %{instance}",
|
|
|
|
instance: Config.instance_name()
|
2018-07-27 10:45:35 +02:00
|
|
|
)
|
2019-09-07 23:58:53 +02:00
|
|
|
|
2019-09-23 19:33:58 +02:00
|
|
|
Email.base_email(to: email, subject: subject)
|
|
|
|
|> assign(:locale, locale)
|
2019-09-07 23:58:53 +02:00
|
|
|
|> assign(:token, reset_password_token)
|
2019-09-23 19:33:58 +02:00
|
|
|
|> assign(:subject, subject)
|
2019-09-24 12:09:43 +02:00
|
|
|
|> render(:password_reset)
|
2018-07-04 14:29:17 +02:00
|
|
|
end
|
2020-01-23 00:55:07 +01:00
|
|
|
|
2021-09-10 11:27:59 +02:00
|
|
|
@spec check_confirmation_token(String.t()) :: {:ok, User.t()} | {:error, :invalid_token}
|
2020-01-23 00:55:07 +01:00
|
|
|
def check_confirmation_token(token) when is_binary(token) do
|
|
|
|
with %User{} = user <- Users.get_user_by_activation_token(token),
|
|
|
|
{:ok, %User{} = user} <-
|
|
|
|
Users.update_user(user, %{
|
2020-02-13 15:48:12 +01:00
|
|
|
confirmed_at: DateTime.utc_now() |> DateTime.truncate(:second),
|
|
|
|
confirmation_sent_at: nil,
|
|
|
|
confirmation_token: nil,
|
|
|
|
email: user.unconfirmed_email || user.email
|
2020-01-23 00:55:07 +01:00
|
|
|
}) do
|
|
|
|
Logger.info("User #{user.email} has been confirmed")
|
|
|
|
{:ok, user}
|
|
|
|
else
|
|
|
|
_err ->
|
|
|
|
{:error, :invalid_token}
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def resend_confirmation_email(%User{} = user, locale \\ "en") do
|
|
|
|
with :ok <- we_can_send_email(user, :confirmation_sent_at),
|
|
|
|
{:ok, user} <-
|
|
|
|
Users.update_user(user, %{
|
|
|
|
"confirmation_sent_at" => DateTime.utc_now() |> DateTime.truncate(:second)
|
2021-03-09 11:56:09 +01:00
|
|
|
}),
|
2021-04-20 15:27:33 +02:00
|
|
|
%Bamboo.Email{} <- send_confirmation_email(user, locale) do
|
2020-01-23 00:55:07 +01:00
|
|
|
Logger.info("Sent confirmation email again to #{user.email}")
|
|
|
|
{:ok, user.email}
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-09-10 11:27:59 +02:00
|
|
|
@spec send_confirmation_email(User.t(), String.t()) :: Bamboo.Email.t()
|
2020-01-23 00:55:07 +01:00
|
|
|
def send_confirmation_email(%User{} = user, locale \\ "en") do
|
|
|
|
user
|
|
|
|
|> Email.User.confirmation_email(locale)
|
2021-04-20 15:02:24 +02:00
|
|
|
|> Email.Mailer.send_email_later()
|
2020-01-23 00:55:07 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
@doc """
|
|
|
|
Check that the provided token is correct and update provided password
|
|
|
|
"""
|
2021-09-10 11:27:59 +02:00
|
|
|
@spec check_reset_password_token(String.t(), String.t()) ::
|
|
|
|
{:ok, User.t()} | {:error, String.t()}
|
2020-01-23 00:55:07 +01:00
|
|
|
def check_reset_password_token(password, token) do
|
|
|
|
with %User{} = user <- Users.get_user_by_reset_password_token(token),
|
|
|
|
{:ok, %User{} = user} <-
|
|
|
|
Repo.update(
|
|
|
|
User.password_reset_changeset(user, %{
|
|
|
|
"password" => password,
|
|
|
|
"reset_password_sent_at" => nil,
|
|
|
|
"reset_password_token" => nil
|
|
|
|
})
|
|
|
|
) do
|
|
|
|
{:ok, user}
|
|
|
|
else
|
|
|
|
{:error, %Ecto.Changeset{errors: [password: {"registration.error.password_too_short", _}]}} ->
|
|
|
|
{:error,
|
2021-08-02 19:31:16 +02:00
|
|
|
gettext(
|
|
|
|
"The password you have choosen is too short. Please make sure your password contains at least 6 charaters."
|
|
|
|
)}
|
2020-01-23 00:55:07 +01:00
|
|
|
|
|
|
|
_err ->
|
|
|
|
{:error,
|
2021-08-02 19:31:16 +02:00
|
|
|
gettext(
|
|
|
|
"The token you provided is invalid. Make sure that the URL is exactly the one provided inside the email you got."
|
|
|
|
)}
|
2020-01-23 00:55:07 +01:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
@doc """
|
|
|
|
Send the email reset password, if it's not too soon since the last send
|
|
|
|
"""
|
|
|
|
@spec send_password_reset_email(User.t(), String.t()) :: tuple
|
|
|
|
def send_password_reset_email(%User{} = user, locale \\ "en") do
|
|
|
|
with :ok <- we_can_send_email(user, :reset_password_sent_at),
|
|
|
|
{:ok, %User{} = user_updated} <-
|
|
|
|
Repo.update(
|
|
|
|
User.send_password_reset_changeset(user, %{
|
|
|
|
"reset_password_token" => Crypto.random_string(30),
|
|
|
|
"reset_password_sent_at" => DateTime.utc_now() |> DateTime.truncate(:second)
|
|
|
|
})
|
2021-03-09 11:56:09 +01:00
|
|
|
),
|
2021-04-20 15:27:33 +02:00
|
|
|
%Bamboo.Email{} = mail <-
|
2021-03-09 11:56:09 +01:00
|
|
|
user_updated
|
|
|
|
|> Email.User.reset_password_email(locale)
|
2021-04-20 15:02:24 +02:00
|
|
|
|> Email.Mailer.send_email_later() do
|
2020-01-23 00:55:07 +01:00
|
|
|
{:ok, mail}
|
|
|
|
else
|
|
|
|
{:error, reason} -> {:error, reason}
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-02-13 15:48:12 +01:00
|
|
|
def send_email_reset_old_email(
|
|
|
|
%User{locale: user_locale, email: email, unconfirmed_email: unconfirmed_email} = _user,
|
|
|
|
_locale \\ "en"
|
|
|
|
) do
|
|
|
|
Gettext.put_locale(user_locale)
|
|
|
|
|
|
|
|
subject =
|
|
|
|
gettext(
|
|
|
|
"Mobilizon on %{instance}: email changed",
|
|
|
|
instance: Config.instance_name()
|
|
|
|
)
|
|
|
|
|
|
|
|
Email.base_email(to: email, subject: subject)
|
|
|
|
|> assign(:locale, user_locale)
|
|
|
|
|> assign(:subject, subject)
|
|
|
|
|> assign(:new_email, unconfirmed_email)
|
|
|
|
|> render(:email_changed_old)
|
|
|
|
end
|
|
|
|
|
|
|
|
def send_email_reset_new_email(
|
|
|
|
%User{
|
|
|
|
locale: user_locale,
|
|
|
|
unconfirmed_email: unconfirmed_email,
|
|
|
|
confirmation_token: confirmation_token
|
|
|
|
} = _user,
|
|
|
|
_locale \\ "en"
|
|
|
|
) do
|
|
|
|
Gettext.put_locale(user_locale)
|
2020-08-10 18:30:15 +02:00
|
|
|
instance_name = Config.instance_name()
|
2020-02-13 15:48:12 +01:00
|
|
|
|
|
|
|
subject =
|
|
|
|
gettext(
|
|
|
|
"Mobilizon on %{instance}: confirm your email address",
|
2020-08-10 18:30:15 +02:00
|
|
|
instance: instance_name
|
2020-02-13 15:48:12 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
Email.base_email(to: unconfirmed_email, subject: subject)
|
|
|
|
|> assign(:locale, user_locale)
|
|
|
|
|> assign(:subject, subject)
|
|
|
|
|> assign(:token, confirmation_token)
|
2020-08-10 18:30:15 +02:00
|
|
|
|> assign(:instance_name, instance_name)
|
2020-02-13 15:48:12 +01:00
|
|
|
|> render(:email_changed_new)
|
|
|
|
end
|
|
|
|
|
2020-01-23 00:55:07 +01:00
|
|
|
@spec we_can_send_email(User.t(), atom) :: :ok | {:error, :email_too_soon}
|
|
|
|
defp we_can_send_email(%User{} = user, key) do
|
|
|
|
case Map.get(user, key) do
|
|
|
|
nil ->
|
|
|
|
:ok
|
|
|
|
|
|
|
|
_ ->
|
|
|
|
case Timex.before?(
|
2020-01-23 21:59:50 +01:00
|
|
|
Timex.shift(Map.get(user, key), hours: 1),
|
|
|
|
DateTime.utc_now() |> DateTime.truncate(:second)
|
|
|
|
) do
|
2020-01-23 00:55:07 +01:00
|
|
|
true ->
|
|
|
|
:ok
|
|
|
|
|
|
|
|
false ->
|
|
|
|
{:error, :email_too_soon}
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2018-07-04 14:29:17 +02:00
|
|
|
end
|