2018-12-23 20:04:54 +00:00
|
|
|
# Pleroma: A lightweight social networking server
|
2023-01-02 20:38:50 +00:00
|
|
|
# Copyright © 2017-2022 Pleroma Authors <https://pleroma.social/>
|
2018-12-23 20:04:54 +00:00
|
|
|
# SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2018-08-06 06:15:22 +00:00
|
|
|
defmodule Pleroma.Web.ActivityPub.Relay do
|
2019-02-09 15:16:26 +00:00
|
|
|
alias Pleroma.Activity
|
2019-03-05 02:52:23 +00:00
|
|
|
alias Pleroma.User
|
2018-08-06 07:14:16 +00:00
|
|
|
alias Pleroma.Web.ActivityPub.ActivityPub
|
2020-05-21 11:16:21 +00:00
|
|
|
alias Pleroma.Web.ActivityPub.Visibility
|
|
|
|
alias Pleroma.Web.CommonAPI
|
2018-08-06 07:14:16 +00:00
|
|
|
require Logger
|
2018-08-06 06:15:22 +00:00
|
|
|
|
2020-08-18 15:21:34 +00:00
|
|
|
@nickname "relay"
|
2020-01-08 13:40:38 +00:00
|
|
|
|
2020-08-18 15:21:34 +00:00
|
|
|
@spec ap_id() :: String.t()
|
|
|
|
def ap_id, do: "#{Pleroma.Web.Endpoint.url()}/#{@nickname}"
|
2019-10-05 20:06:31 +00:00
|
|
|
|
2020-08-18 15:21:34 +00:00
|
|
|
@spec get_actor() :: User.t() | nil
|
|
|
|
def get_actor, do: User.get_or_create_service_actor_by_ap_id(ap_id(), @nickname)
|
2019-10-30 23:26:02 +00:00
|
|
|
|
2019-08-13 21:12:59 +00:00
|
|
|
@spec follow(String.t()) :: {:ok, Activity.t()} | {:error, any()}
|
2018-08-06 07:14:16 +00:00
|
|
|
def follow(target_instance) do
|
|
|
|
with %User{} = local_user <- get_actor(),
|
2019-03-18 13:56:59 +00:00
|
|
|
{:ok, %User{} = target_user} <- User.get_or_fetch_by_ap_id(target_instance),
|
2020-07-08 15:07:24 +00:00
|
|
|
{:ok, _, _, activity} <- CommonAPI.follow(local_user, target_user) do
|
2018-08-06 07:14:16 +00:00
|
|
|
Logger.info("relay: followed instance: #{target_instance}; id=#{activity.data["id"]}")
|
2018-11-10 14:31:37 +00:00
|
|
|
{:ok, activity}
|
2018-08-06 07:14:16 +00:00
|
|
|
else
|
2019-08-24 14:41:53 +00:00
|
|
|
error -> format_error(error)
|
2018-08-06 07:14:16 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-09-28 06:16:42 +00:00
|
|
|
@spec unfollow(String.t(), map()) :: {:ok, Activity.t()} | {:error, any()}
|
|
|
|
def unfollow(target_instance, opts \\ %{}) do
|
2018-08-06 07:14:16 +00:00
|
|
|
with %User{} = local_user <- get_actor(),
|
2020-09-28 06:16:42 +00:00
|
|
|
{:ok, target_user} <- fetch_target_user(target_instance, opts),
|
2019-08-22 18:32:40 +00:00
|
|
|
{:ok, activity} <- ActivityPub.unfollow(local_user, target_user) do
|
2020-09-28 06:16:42 +00:00
|
|
|
case target_user.id do
|
|
|
|
nil -> User.update_following_count(local_user)
|
|
|
|
_ -> User.unfollow(local_user, target_user)
|
|
|
|
end
|
|
|
|
|
2018-08-06 07:14:16 +00:00
|
|
|
Logger.info("relay: unfollowed instance: #{target_instance}: id=#{activity.data["id"]}")
|
2018-11-10 14:31:37 +00:00
|
|
|
{:ok, activity}
|
2018-08-06 07:14:16 +00:00
|
|
|
else
|
2019-08-24 14:41:53 +00:00
|
|
|
error -> format_error(error)
|
2018-08-06 07:14:16 +00:00
|
|
|
end
|
|
|
|
end
|
2018-08-06 07:43:37 +00:00
|
|
|
|
2020-09-28 06:16:42 +00:00
|
|
|
defp fetch_target_user(ap_id, opts) do
|
|
|
|
case {opts[:force], User.get_or_fetch_by_ap_id(ap_id)} do
|
|
|
|
{_, {:ok, %User{} = user}} -> {:ok, user}
|
|
|
|
{true, _} -> {:ok, %User{ap_id: ap_id}}
|
|
|
|
{_, error} -> error
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-05-21 11:16:21 +00:00
|
|
|
@spec publish(any()) :: {:ok, Activity.t()} | {:error, any()}
|
2018-08-06 08:03:10 +00:00
|
|
|
def publish(%Activity{data: %{"type" => "Create"}} = activity) do
|
2018-08-06 07:43:37 +00:00
|
|
|
with %User{} = user <- get_actor(),
|
2020-05-21 11:16:21 +00:00
|
|
|
true <- Visibility.is_public?(activity) do
|
|
|
|
CommonAPI.repeat(activity.id, user)
|
2018-08-06 07:43:37 +00:00
|
|
|
else
|
2019-08-24 14:41:53 +00:00
|
|
|
error -> format_error(error)
|
2018-08-06 07:43:37 +00:00
|
|
|
end
|
|
|
|
end
|
2018-08-06 08:03:10 +00:00
|
|
|
|
2019-08-13 21:12:59 +00:00
|
|
|
def publish(_), do: {:error, "Not implemented"}
|
2019-08-24 14:41:53 +00:00
|
|
|
|
2020-08-18 15:21:34 +00:00
|
|
|
@spec list() :: {:ok, [%{actor: String.t(), followed_back: boolean()}]} | {:error, any()}
|
|
|
|
def list do
|
2019-10-21 07:19:31 +00:00
|
|
|
with %User{} = user <- get_actor() do
|
2020-02-25 13:21:48 +00:00
|
|
|
accepted =
|
2019-10-21 07:19:31 +00:00
|
|
|
user
|
2020-08-18 15:21:34 +00:00
|
|
|
|> following()
|
|
|
|
|> Enum.map(fn actor -> %{actor: actor, followed_back: true} end)
|
2020-02-25 13:21:48 +00:00
|
|
|
|
2020-08-18 15:21:34 +00:00
|
|
|
without_accept =
|
|
|
|
user
|
|
|
|
|> Pleroma.Activity.following_requests_for_actor()
|
|
|
|
|> Enum.map(fn activity -> %{actor: activity.data["object"], followed_back: false} end)
|
|
|
|
|> Enum.uniq()
|
2020-02-25 13:21:48 +00:00
|
|
|
|
2020-08-18 15:21:34 +00:00
|
|
|
{:ok, accepted ++ without_accept}
|
2019-10-11 16:12:29 +00:00
|
|
|
else
|
|
|
|
error -> format_error(error)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-08-18 15:21:34 +00:00
|
|
|
@spec following() :: [String.t()]
|
|
|
|
def following do
|
|
|
|
get_actor()
|
|
|
|
|> following()
|
|
|
|
end
|
|
|
|
|
|
|
|
defp following(user) do
|
|
|
|
user
|
|
|
|
|> User.following_ap_ids()
|
|
|
|
|> Enum.uniq()
|
|
|
|
end
|
|
|
|
|
2019-08-24 14:41:53 +00:00
|
|
|
defp format_error({:error, error}), do: format_error(error)
|
|
|
|
|
|
|
|
defp format_error(error) do
|
|
|
|
Logger.error("error: #{inspect(error)}")
|
|
|
|
{:error, error}
|
|
|
|
end
|
2018-08-06 06:15:22 +00:00
|
|
|
end
|