2018-09-18 11:56:46 +02:00
|
|
|
defmodule Pleroma.Web.Push.Subscription do
|
|
|
|
use Ecto.Schema
|
2018-12-06 13:29:04 +01:00
|
|
|
import Ecto.Changeset
|
2018-09-18 11:56:46 +02:00
|
|
|
alias Pleroma.{Repo, User}
|
|
|
|
alias Pleroma.Web.OAuth.Token
|
|
|
|
alias Pleroma.Web.Push.Subscription
|
|
|
|
|
|
|
|
schema "push_subscriptions" do
|
|
|
|
belongs_to(:user, User)
|
|
|
|
belongs_to(:token, Token)
|
|
|
|
field(:endpoint, :string)
|
|
|
|
field(:key_p256dh, :string)
|
|
|
|
field(:key_auth, :string)
|
|
|
|
field(:data, :map, default: %{})
|
|
|
|
|
|
|
|
timestamps()
|
|
|
|
end
|
|
|
|
|
|
|
|
@supported_alert_types ~w[follow favourite mention reblog]
|
|
|
|
|
|
|
|
defp alerts(%{"data" => %{"alerts" => alerts}}) do
|
|
|
|
alerts = Map.take(alerts, @supported_alert_types)
|
|
|
|
%{"alerts" => alerts}
|
|
|
|
end
|
|
|
|
|
|
|
|
def create(
|
|
|
|
%User{} = user,
|
|
|
|
%Token{} = token,
|
|
|
|
%{
|
|
|
|
"subscription" => %{
|
|
|
|
"endpoint" => endpoint,
|
|
|
|
"keys" => %{"auth" => key_auth, "p256dh" => key_p256dh}
|
|
|
|
}
|
|
|
|
} = params
|
|
|
|
) do
|
|
|
|
Repo.insert(%Subscription{
|
|
|
|
user_id: user.id,
|
|
|
|
token_id: token.id,
|
|
|
|
endpoint: endpoint,
|
2018-12-08 17:32:58 +01:00
|
|
|
key_auth: ensure_base64_urlsafe(key_auth),
|
|
|
|
key_p256dh: ensure_base64_urlsafe(key_p256dh),
|
2018-09-18 11:56:46 +02:00
|
|
|
data: alerts(params)
|
|
|
|
})
|
|
|
|
end
|
|
|
|
|
|
|
|
def get(%User{id: user_id}, %Token{id: token_id}) do
|
|
|
|
Repo.get_by(Subscription, user_id: user_id, token_id: token_id)
|
|
|
|
end
|
|
|
|
|
|
|
|
def update(user, token, params) do
|
|
|
|
get(user, token)
|
|
|
|
|> change(data: alerts(params))
|
|
|
|
|> Repo.update()
|
|
|
|
end
|
|
|
|
|
|
|
|
def delete(user, token) do
|
|
|
|
Repo.delete(get(user, token))
|
|
|
|
end
|
|
|
|
|
|
|
|
def delete_if_exists(user, token) do
|
|
|
|
case get(user, token) do
|
|
|
|
nil -> {:ok, nil}
|
|
|
|
sub -> Repo.delete(sub)
|
|
|
|
end
|
|
|
|
end
|
2018-12-08 17:32:58 +01:00
|
|
|
|
|
|
|
# Some webpush clients (e.g. iOS Toot!) use an non urlsafe base64 as an encoding for the key.
|
|
|
|
# However, the web push rfs specify to use base64 urlsafe, and the `web_push_encryption` library we use
|
|
|
|
# requires the key to be properly encoded. So we just convert base64 to urlsafe base64.
|
|
|
|
defp ensure_base64_urlsafe(string) do
|
|
|
|
string
|
|
|
|
|> String.replace("+", "-")
|
|
|
|
|> String.replace("/", "_")
|
|
|
|
|> String.replace("=", "")
|
|
|
|
end
|
2018-09-18 11:56:46 +02:00
|
|
|
end
|