2018-12-23 20:04:54 +00:00
|
|
|
# Pleroma: A lightweight social networking server
|
2018-12-31 15:41:47 +00:00
|
|
|
# Copyright © 2017-2019 Pleroma Authors <https://pleroma.social/>
|
2018-12-23 20:04:54 +00:00
|
|
|
# SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2018-12-06 12:29:04 +00:00
|
|
|
defmodule Pleroma.Web.Push do
|
|
|
|
use GenServer
|
|
|
|
|
2019-02-09 15:16:26 +00:00
|
|
|
alias Pleroma.Repo
|
|
|
|
alias Pleroma.User
|
2019-03-04 17:47:34 +00:00
|
|
|
alias Pleroma.Activity
|
|
|
|
alias Pleroma.Object
|
2018-12-06 12:29:04 +00:00
|
|
|
alias Pleroma.Web.Push.Subscription
|
2019-03-04 17:47:34 +00:00
|
|
|
alias Pleroma.Web.Metadata.Utils
|
2018-12-06 12:29:04 +00:00
|
|
|
|
|
|
|
require Logger
|
|
|
|
import Ecto.Query
|
|
|
|
|
|
|
|
@types ["Create", "Follow", "Announce", "Like"]
|
|
|
|
|
|
|
|
def start_link() do
|
|
|
|
GenServer.start_link(__MODULE__, :ok, name: __MODULE__)
|
|
|
|
end
|
|
|
|
|
2018-12-09 12:45:21 +00:00
|
|
|
def vapid_config() do
|
|
|
|
Application.get_env(:web_push_encryption, :vapid_details, [])
|
|
|
|
end
|
2018-12-06 12:29:04 +00:00
|
|
|
|
2018-12-09 12:45:21 +00:00
|
|
|
def enabled() do
|
|
|
|
case vapid_config() do
|
|
|
|
[] -> false
|
|
|
|
list when is_list(list) -> true
|
|
|
|
_ -> false
|
2018-12-06 12:29:04 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def send(notification) do
|
2018-12-09 12:45:21 +00:00
|
|
|
if enabled() do
|
2018-12-06 12:56:56 +00:00
|
|
|
GenServer.cast(Pleroma.Web.Push, {:send, notification})
|
|
|
|
end
|
2018-12-06 12:29:04 +00:00
|
|
|
end
|
|
|
|
|
2018-12-09 12:45:21 +00:00
|
|
|
def init(:ok) do
|
2018-12-14 15:17:27 +00:00
|
|
|
if !enabled() do
|
2018-12-09 12:45:21 +00:00
|
|
|
Logger.warn("""
|
|
|
|
VAPID key pair is not found. If you wish to enabled web push, please run
|
|
|
|
|
|
|
|
mix web_push.gen.keypair
|
|
|
|
|
|
|
|
and add the resulting output to your configuration file.
|
|
|
|
""")
|
|
|
|
|
|
|
|
:ignore
|
|
|
|
else
|
|
|
|
{:ok, nil}
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-12-06 12:29:04 +00:00
|
|
|
def handle_cast(
|
|
|
|
{:send, %{activity: %{data: %{"type" => type}}, user_id: user_id} = notification},
|
|
|
|
state
|
|
|
|
)
|
|
|
|
when type in @types do
|
|
|
|
actor = User.get_cached_by_ap_id(notification.activity.data["actor"])
|
|
|
|
|
2018-12-10 14:50:10 +00:00
|
|
|
type = Pleroma.Activity.mastodon_notification_type(notification.activity)
|
2018-12-08 17:07:10 +00:00
|
|
|
|
2018-12-06 12:29:04 +00:00
|
|
|
Subscription
|
|
|
|
|> where(user_id: ^user_id)
|
2018-12-07 14:08:42 +00:00
|
|
|
|> preload(:token)
|
2018-12-06 12:29:04 +00:00
|
|
|
|> Repo.all()
|
2018-12-08 17:07:10 +00:00
|
|
|
|> Enum.filter(fn subscription ->
|
|
|
|
get_in(subscription.data, ["alerts", type]) || false
|
|
|
|
end)
|
|
|
|
|> Enum.each(fn subscription ->
|
|
|
|
sub = %{
|
2018-12-06 12:29:04 +00:00
|
|
|
keys: %{
|
2018-12-08 17:07:10 +00:00
|
|
|
p256dh: subscription.key_p256dh,
|
|
|
|
auth: subscription.key_auth
|
2018-12-06 12:29:04 +00:00
|
|
|
},
|
2018-12-08 17:07:10 +00:00
|
|
|
endpoint: subscription.endpoint
|
2018-12-06 12:29:04 +00:00
|
|
|
}
|
|
|
|
|
2018-12-07 14:08:42 +00:00
|
|
|
body =
|
|
|
|
Jason.encode!(%{
|
|
|
|
title: format_title(notification),
|
2018-12-08 17:07:10 +00:00
|
|
|
access_token: subscription.token.token,
|
2018-12-07 14:08:42 +00:00
|
|
|
body: format_body(notification, actor),
|
|
|
|
notification_id: notification.id,
|
2018-12-08 17:07:10 +00:00
|
|
|
notification_type: type,
|
2018-12-07 14:08:42 +00:00
|
|
|
icon: User.avatar_url(actor),
|
2018-12-08 16:34:13 +00:00
|
|
|
preferred_locale: "en"
|
2018-12-07 14:08:42 +00:00
|
|
|
})
|
|
|
|
|
2018-12-09 12:45:21 +00:00
|
|
|
case WebPushEncryption.send_web_push(
|
|
|
|
body,
|
|
|
|
sub,
|
|
|
|
Application.get_env(:web_push_encryption, :gcm_api_key)
|
|
|
|
) do
|
2018-12-06 12:29:04 +00:00
|
|
|
{:ok, %{status_code: code}} when 400 <= code and code < 500 ->
|
|
|
|
Logger.debug("Removing subscription record")
|
2018-12-08 17:07:10 +00:00
|
|
|
Repo.delete!(subscription)
|
2018-12-06 12:29:04 +00:00
|
|
|
:ok
|
|
|
|
|
|
|
|
{:ok, %{status_code: code}} when 200 <= code and code < 300 ->
|
|
|
|
:ok
|
|
|
|
|
|
|
|
{:ok, %{status_code: code}} ->
|
2018-12-08 17:07:10 +00:00
|
|
|
Logger.error("Web Push Notification failed with code: #{code}")
|
2018-12-06 12:29:04 +00:00
|
|
|
:error
|
|
|
|
|
2018-12-06 13:55:46 +00:00
|
|
|
_ ->
|
2018-12-08 17:07:10 +00:00
|
|
|
Logger.error("Web Push Notification failed with unknown error")
|
2018-12-06 12:29:04 +00:00
|
|
|
:error
|
|
|
|
end
|
|
|
|
end)
|
|
|
|
|
|
|
|
{:noreply, state}
|
|
|
|
end
|
|
|
|
|
|
|
|
def handle_cast({:send, _}, state) do
|
|
|
|
Logger.warn("Unknown notification type")
|
|
|
|
{:noreply, state}
|
|
|
|
end
|
|
|
|
|
2019-03-04 17:47:34 +00:00
|
|
|
def format_body(
|
|
|
|
%{activity: %{data: %{"type" => "Create", "object" => %{"content" => content}}}},
|
|
|
|
actor
|
|
|
|
) do
|
|
|
|
"@#{actor.nickname}: #{Utils.scrub_html_and_truncate(content, 80)}"
|
|
|
|
end
|
|
|
|
|
|
|
|
def format_body(
|
|
|
|
%{activity: %{data: %{"type" => "Announce", "object" => activity_id}}},
|
|
|
|
actor
|
|
|
|
) do
|
|
|
|
%Activity{data: %{"object" => %{"id" => object_id}}} = Activity.get_by_ap_id(activity_id)
|
|
|
|
%Object{data: %{"content" => content}} = Object.get_by_ap_id(object_id)
|
|
|
|
|
|
|
|
"@#{actor.nickname} repeated: #{Utils.scrub_html_and_truncate(content, 80)}"
|
|
|
|
end
|
|
|
|
|
|
|
|
def format_body(
|
|
|
|
%{activity: %{data: %{"type" => type}}},
|
|
|
|
actor
|
|
|
|
)
|
|
|
|
when type in ["Follow", "Like"] do
|
|
|
|
case type do
|
|
|
|
"Follow" -> "@#{actor.nickname} has followed you"
|
|
|
|
"Like" -> "@#{actor.nickname} has favorited your post"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-12-07 14:08:42 +00:00
|
|
|
defp format_title(%{activity: %{data: %{"type" => type}}}) do
|
|
|
|
case type do
|
|
|
|
"Create" -> "New Mention"
|
|
|
|
"Follow" -> "New Follower"
|
|
|
|
"Announce" -> "New Repeat"
|
|
|
|
"Like" -> "New Favorite"
|
|
|
|
end
|
2018-12-06 12:29:04 +00:00
|
|
|
end
|
|
|
|
end
|