2018-12-23 13:04:54 -07:00
|
|
|
# Pleroma: A lightweight social networking server
|
2022-02-25 23:11:42 -07:00
|
|
|
# Copyright © 2017-2022 Pleroma Authors <https://pleroma.social/>
|
2018-12-23 13:04:54 -07:00
|
|
|
# SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2019-04-09 22:14:37 -06:00
|
|
|
defmodule Pleroma.Emails.UserEmail do
|
2018-12-11 10:17:49 -07:00
|
|
|
@moduledoc "User emails"
|
|
|
|
|
2019-08-16 05:32:25 -06:00
|
|
|
alias Pleroma.Config
|
|
|
|
alias Pleroma.User
|
2019-02-09 08:16:26 -07:00
|
|
|
alias Pleroma.Web.Endpoint
|
|
|
|
alias Pleroma.Web.Router
|
2018-12-11 10:17:49 -07:00
|
|
|
|
2021-05-22 13:48:13 -06:00
|
|
|
import Swoosh.Email
|
|
|
|
import Phoenix.Swoosh, except: [render_body: 3]
|
2020-07-21 00:25:53 -06:00
|
|
|
import Pleroma.Config.Helpers, only: [instance_name: 0, sender: 0]
|
2018-12-11 10:17:49 -07:00
|
|
|
|
2021-05-22 13:48:13 -06:00
|
|
|
def render_body(email, template, assigns \\ %{}) do
|
|
|
|
email
|
|
|
|
|> put_new_layout({Pleroma.Web.LayoutView, :email})
|
|
|
|
|> put_new_view(Pleroma.Web.EmailView)
|
|
|
|
|> Phoenix.Swoosh.render_body(template, assigns)
|
|
|
|
end
|
|
|
|
|
2018-12-13 04:30:48 -07:00
|
|
|
defp recipient(email, nil), do: email
|
|
|
|
defp recipient(email, name), do: {name, email}
|
2019-08-16 05:32:25 -06:00
|
|
|
defp recipient(%User{} = user), do: recipient(user.email, user.name)
|
2018-12-13 04:30:48 -07:00
|
|
|
|
2020-07-21 00:25:53 -06:00
|
|
|
@spec welcome(User.t(), map()) :: Swoosh.Email.t()
|
|
|
|
def welcome(user, opts \\ %{}) do
|
|
|
|
new()
|
|
|
|
|> to(recipient(user))
|
|
|
|
|> from(Map.get(opts, :sender, sender()))
|
|
|
|
|> subject(Map.get(opts, :subject, "Welcome to #{instance_name()}!"))
|
|
|
|
|> html_body(Map.get(opts, :html, "Welcome to #{instance_name()}!"))
|
|
|
|
|> text_body(Map.get(opts, :text, "Welcome to #{instance_name()}!"))
|
|
|
|
end
|
|
|
|
|
2019-06-24 13:01:56 -06:00
|
|
|
def password_reset_email(user, token) when is_binary(token) do
|
|
|
|
password_reset_url = Router.Helpers.reset_password_url(Endpoint, :reset, token)
|
2018-12-11 10:17:49 -07:00
|
|
|
|
|
|
|
html_body = """
|
|
|
|
<h3>Reset your password at #{instance_name()}</h3>
|
|
|
|
<p>Someone has requested password change for your account at #{instance_name()}.</p>
|
|
|
|
<p>If it was you, visit the following link to proceed: <a href="#{password_reset_url}">reset password</a>.</p>
|
|
|
|
<p>If it was someone else, nothing to worry about: your data is secure and your password has not been changed.</p>
|
|
|
|
"""
|
|
|
|
|
|
|
|
new()
|
2018-12-17 07:28:58 -07:00
|
|
|
|> to(recipient(user))
|
2018-12-13 04:30:48 -07:00
|
|
|
|> from(sender())
|
2018-12-11 10:17:49 -07:00
|
|
|
|> subject("Password reset")
|
|
|
|
|> html_body(html_body)
|
|
|
|
end
|
2018-12-13 06:30:10 -07:00
|
|
|
|
2018-12-14 03:52:04 -07:00
|
|
|
def user_invitation_email(
|
|
|
|
user,
|
|
|
|
%Pleroma.UserInviteToken{} = user_invite_token,
|
|
|
|
to_email,
|
|
|
|
to_name \\ nil
|
|
|
|
) do
|
2018-12-13 06:30:10 -07:00
|
|
|
registration_url =
|
|
|
|
Router.Helpers.redirect_url(
|
|
|
|
Endpoint,
|
|
|
|
:registration_page,
|
2018-12-13 07:58:40 -07:00
|
|
|
user_invite_token.token
|
2018-12-13 06:30:10 -07:00
|
|
|
)
|
|
|
|
|
|
|
|
html_body = """
|
|
|
|
<h3>You are invited to #{instance_name()}</h3>
|
|
|
|
<p>#{user.name} invites you to join #{instance_name()}, an instance of Pleroma federated social networking platform.</p>
|
|
|
|
<p>Click the following link to register: <a href="#{registration_url}">accept invitation</a>.</p>
|
|
|
|
"""
|
|
|
|
|
|
|
|
new()
|
|
|
|
|> to(recipient(to_email, to_name))
|
|
|
|
|> from(sender())
|
|
|
|
|> subject("Invitation to #{instance_name()}")
|
|
|
|
|> html_body(html_body)
|
|
|
|
end
|
2018-12-17 07:28:58 -07:00
|
|
|
|
|
|
|
def account_confirmation_email(user) do
|
|
|
|
confirmation_url =
|
|
|
|
Router.Helpers.confirm_email_url(
|
|
|
|
Endpoint,
|
|
|
|
:confirm_email,
|
2018-12-20 03:41:30 -07:00
|
|
|
user.id,
|
2019-10-16 12:59:21 -06:00
|
|
|
to_string(user.confirmation_token)
|
2018-12-17 07:28:58 -07:00
|
|
|
)
|
|
|
|
|
|
|
|
html_body = """
|
2021-02-04 14:10:43 -07:00
|
|
|
<h3>Thank you for registering on #{instance_name()}</h3>
|
2018-12-17 07:28:58 -07:00
|
|
|
<p>Email confirmation is required to activate the account.</p>
|
2021-02-04 14:10:43 -07:00
|
|
|
<p>Please click the following link to <a href="#{confirmation_url}">activate your account</a>.</p>
|
2018-12-17 07:28:58 -07:00
|
|
|
"""
|
|
|
|
|
|
|
|
new()
|
|
|
|
|> to(recipient(user))
|
|
|
|
|> from(sender())
|
|
|
|
|> subject("#{instance_name()} account confirmation")
|
|
|
|
|> html_body(html_body)
|
|
|
|
end
|
2019-04-20 06:42:19 -06:00
|
|
|
|
2020-10-11 19:50:09 -06:00
|
|
|
def approval_pending_email(user) do
|
|
|
|
html_body = """
|
|
|
|
<h3>Awaiting Approval</h3>
|
|
|
|
<p>Your account at #{instance_name()} is being reviewed by staff. You will receive another email once your account is approved.</p>
|
|
|
|
"""
|
|
|
|
|
|
|
|
new()
|
|
|
|
|> to(recipient(user))
|
|
|
|
|> from(sender())
|
|
|
|
|> subject("Your account is awaiting approval")
|
|
|
|
|> html_body(html_body)
|
|
|
|
end
|
|
|
|
|
2021-02-04 14:16:50 -07:00
|
|
|
def successful_registration_email(user) do
|
|
|
|
html_body = """
|
2021-02-05 08:13:53 -07:00
|
|
|
<h3>Hello @#{user.nickname},</h3>
|
2021-02-04 14:16:50 -07:00
|
|
|
<p>Your account at #{instance_name()} has been registered successfully.</p>
|
|
|
|
<p>No further action is required to activate your account.</p>
|
|
|
|
"""
|
|
|
|
|
|
|
|
new()
|
|
|
|
|> to(recipient(user))
|
|
|
|
|> from(sender())
|
|
|
|
|> subject("Account registered on #{instance_name()}")
|
|
|
|
|> html_body(html_body)
|
|
|
|
end
|
|
|
|
|
2019-04-20 06:42:19 -06:00
|
|
|
@doc """
|
|
|
|
Email used in digest email notifications
|
|
|
|
Includes Mentions and New Followers data
|
|
|
|
If there are no mentions (even when new followers exist), the function will return nil
|
|
|
|
"""
|
2019-08-16 05:32:25 -06:00
|
|
|
@spec digest_email(User.t()) :: Swoosh.Email.t() | nil
|
2019-04-20 06:42:19 -06:00
|
|
|
def digest_email(user) do
|
2019-08-16 05:32:25 -06:00
|
|
|
notifications = Pleroma.Notification.for_user_since(user, user.last_digest_emailed_at)
|
|
|
|
|
|
|
|
mentions =
|
|
|
|
notifications
|
|
|
|
|> Enum.filter(&(&1.activity.data["type"] == "Create"))
|
|
|
|
|> Enum.map(fn notification ->
|
2021-01-04 05:38:31 -07:00
|
|
|
object = Pleroma.Object.normalize(notification.activity, fetch: false)
|
2019-08-16 05:32:25 -06:00
|
|
|
|
2020-08-19 02:06:03 -06:00
|
|
|
if not is_nil(object) do
|
|
|
|
object = update_in(object.data["content"], &format_links/1)
|
|
|
|
|
|
|
|
%{
|
|
|
|
data: notification,
|
|
|
|
object: object,
|
|
|
|
from: User.get_by_ap_id(notification.activity.actor)
|
|
|
|
}
|
|
|
|
end
|
2019-08-16 05:32:25 -06:00
|
|
|
end)
|
2020-08-19 02:06:03 -06:00
|
|
|
|> Enum.filter(& &1)
|
2019-08-16 05:32:25 -06:00
|
|
|
|
|
|
|
followers =
|
|
|
|
notifications
|
|
|
|
|> Enum.filter(&(&1.activity.data["type"] == "Follow"))
|
|
|
|
|> Enum.map(fn notification ->
|
2020-08-19 02:06:03 -06:00
|
|
|
from = User.get_by_ap_id(notification.activity.actor)
|
|
|
|
|
|
|
|
if not is_nil(from) do
|
|
|
|
%{
|
|
|
|
data: notification,
|
2021-01-04 05:38:31 -07:00
|
|
|
object: Pleroma.Object.normalize(notification.activity, fetch: false),
|
2020-08-19 02:06:03 -06:00
|
|
|
from: User.get_by_ap_id(notification.activity.actor)
|
|
|
|
}
|
|
|
|
end
|
2019-04-20 06:42:19 -06:00
|
|
|
end)
|
2020-08-19 02:06:03 -06:00
|
|
|
|> Enum.filter(& &1)
|
2019-04-20 06:42:19 -06:00
|
|
|
|
2019-08-16 05:32:25 -06:00
|
|
|
unless Enum.empty?(mentions) do
|
|
|
|
styling = Config.get([__MODULE__, :styling])
|
|
|
|
logo = Config.get([__MODULE__, :logo])
|
2019-08-14 07:46:05 -06:00
|
|
|
|
2019-04-20 06:42:19 -06:00
|
|
|
html_data = %{
|
|
|
|
instance: instance_name(),
|
|
|
|
user: user,
|
|
|
|
mentions: mentions,
|
2019-08-16 05:32:25 -06:00
|
|
|
followers: followers,
|
|
|
|
unsubscribe_link: unsubscribe_url(user, "digest"),
|
|
|
|
styling: styling
|
2019-04-20 06:42:19 -06:00
|
|
|
}
|
|
|
|
|
2019-08-16 05:32:25 -06:00
|
|
|
logo_path =
|
|
|
|
if is_nil(logo) do
|
2020-12-12 07:30:08 -07:00
|
|
|
Path.join(:code.priv_dir(:pleroma), "static/static/logo.svg")
|
2019-08-16 05:32:25 -06:00
|
|
|
else
|
|
|
|
Path.join(Config.get([:instance, :static_dir]), logo)
|
|
|
|
end
|
2019-08-14 07:46:05 -06:00
|
|
|
|
2019-04-20 06:42:19 -06:00
|
|
|
new()
|
|
|
|
|> to(recipient(user))
|
|
|
|
|> from(sender())
|
|
|
|
|> subject("Your digest from #{instance_name()}")
|
2019-08-14 07:46:05 -06:00
|
|
|
|> put_layout(false)
|
2019-04-20 06:42:19 -06:00
|
|
|
|> render_body("digest.html", html_data)
|
2020-12-12 07:30:08 -07:00
|
|
|
|> attachment(Swoosh.Attachment.new(logo_path, filename: "logo.svg", type: :inline))
|
2019-04-20 06:42:19 -06:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-08-14 07:46:05 -06:00
|
|
|
defp format_links(str) do
|
|
|
|
re = ~r/<a.+href=['"].*>/iU
|
2019-08-16 05:32:25 -06:00
|
|
|
%{link_color: color} = Config.get([__MODULE__, :styling])
|
2019-08-14 07:46:05 -06:00
|
|
|
|
2019-08-14 08:03:25 -06:00
|
|
|
Regex.replace(re, str, fn link ->
|
2019-08-16 05:32:25 -06:00
|
|
|
String.replace(link, "<a", "<a style=\"color: #{color};text-decoration: none;\"")
|
2019-08-14 07:46:05 -06:00
|
|
|
end)
|
|
|
|
end
|
|
|
|
|
2019-04-20 06:42:19 -06:00
|
|
|
@doc """
|
|
|
|
Generate unsubscribe link for given user and notifications type.
|
|
|
|
The link contains JWT token with the data, and subscription can be modified without
|
|
|
|
authorization.
|
|
|
|
"""
|
2019-08-16 05:32:25 -06:00
|
|
|
@spec unsubscribe_url(User.t(), String.t()) :: String.t()
|
2019-04-20 06:42:19 -06:00
|
|
|
def unsubscribe_url(user, notifications_type) do
|
|
|
|
token =
|
|
|
|
%{"sub" => user.id, "act" => %{"unsubscribe" => notifications_type}, "exp" => false}
|
|
|
|
|> Pleroma.JWT.generate_and_sign!()
|
|
|
|
|> Base.encode64()
|
|
|
|
|
2019-08-16 05:32:25 -06:00
|
|
|
Router.Helpers.subscription_url(Endpoint, :unsubscribe, token)
|
2019-04-20 06:42:19 -06:00
|
|
|
end
|
2020-09-17 08:42:24 -06:00
|
|
|
|
2020-09-18 12:18:34 -06:00
|
|
|
def backup_is_ready_email(backup, admin_user_id \\ nil) do
|
2020-09-17 08:42:24 -06:00
|
|
|
%{user: user} = Pleroma.Repo.preload(backup, :user)
|
|
|
|
download_url = Pleroma.Web.PleromaAPI.BackupView.download_url(backup)
|
|
|
|
|
2020-09-18 12:18:34 -06:00
|
|
|
html_body =
|
|
|
|
if is_nil(admin_user_id) do
|
|
|
|
"""
|
|
|
|
<p>You requested a full backup of your Pleroma account. It's ready for download:</p>
|
2020-09-21 11:47:36 -06:00
|
|
|
<p><a href="#{download_url}">#{download_url}</a></p>
|
2020-09-18 12:18:34 -06:00
|
|
|
"""
|
|
|
|
else
|
|
|
|
admin = Pleroma.Repo.get(User, admin_user_id)
|
|
|
|
|
|
|
|
"""
|
|
|
|
<p>Admin @#{admin.nickname} requested a full backup of your Pleroma account. It's ready for download:</p>
|
2020-09-21 11:47:36 -06:00
|
|
|
<p><a href="#{download_url}">#{download_url}</a></p>
|
2020-09-18 12:18:34 -06:00
|
|
|
"""
|
|
|
|
end
|
2020-09-17 08:42:24 -06:00
|
|
|
|
|
|
|
new()
|
|
|
|
|> to(recipient(user))
|
|
|
|
|> from(sender())
|
|
|
|
|> subject("Your account archive is ready")
|
|
|
|
|> html_body(html_body)
|
|
|
|
end
|
2018-12-11 10:17:49 -07:00
|
|
|
end
|