diff options
20 files changed, 1126 insertions, 894 deletions
| diff --git a/.gitlab-ci.yml b/.gitlab-ci.yml index fd0c5c8d4..9a754ed78 100644 --- a/.gitlab-ci.yml +++ b/.gitlab-ci.yml @@ -229,7 +229,7 @@ arm:    only: *release-only    tags:      - arm32 -  image: elixir:1.10.3 +  image: arm32v7/elixir:1.10.3    cache: *release-cache    variables: *release-variables    before_script: *before-release @@ -241,7 +241,7 @@ arm-musl:    only: *release-only    tags:      - arm32 -  image: elixir:1.10.3-alpine +  image: arm32v7/elixir:1.10.3-alpine    cache: *release-cache    variables: *release-variables    before_script: *before-release-musl @@ -253,7 +253,7 @@ arm64:    only: *release-only    tags:      - arm -  image: elixir:1.10.3 +  image: arm64v8/elixir:1.10.3    cache: *release-cache    variables: *release-variables    before_script: *before-release @@ -265,8 +265,7 @@ arm64-musl:    only: *release-only    tags:      - arm -  # TODO: Replace with upstream image when 1.9.0 comes out -  image: elixir:1.10.3-alpine +  image: arm64v8/elixir:1.10.3-alpine    cache: *release-cache    variables: *release-variables    before_script: *before-release-musl diff --git a/CHANGELOG.md b/CHANGELOG.md index f927729b1..fe1114c02 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -15,6 +15,7 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/).  - Reports now generate notifications for admins and mods.  - Experimental websocket-based federation between Pleroma instances. +- Support for local-only statuses  - Support pagination of blocks and mutes.  - Account backup.  - Configuration: Add `:instance, autofollowing_nicknames` setting to provide a way to make accounts automatically follow new users that register on the local Pleroma instance. diff --git a/docs/API/differences_in_mastoapi_responses.md b/docs/API/differences_in_mastoapi_responses.md index ba48a2ca1..17999da55 100644 --- a/docs/API/differences_in_mastoapi_responses.md +++ b/docs/API/differences_in_mastoapi_responses.md @@ -18,7 +18,7 @@ Adding the parameter `instance=lain.com` to the public timeline will show only s  ## Statuses -- `visibility`: has an additional possible value `list` +- `visibility`: has additional possible values `list` and `local` (for local-only statuses)  Has these additional fields under the `pleroma` object: @@ -173,7 +173,7 @@ Additional parameters can be added to the JSON body/Form data:  - `preview`: boolean, if set to `true` the post won't be actually posted, but the status entitiy would still be rendered back. This could be useful for previewing rich text/custom emoji, for example.  - `content_type`: string, contain the MIME type of the status, it is transformed into HTML by the backend. You can get the list of the supported MIME types with the nodeinfo endpoint.  - `to`: A list of nicknames (like `lain@soykaf.club` or `lain` on the local server) that will be used to determine who is going to be addressed by this post. Using this will disable the implicit addressing by mentioned names in the `status` body, only the people in the `to` list will be addressed. The normal rules for for post visibility are not affected by this and will still apply. -- `visibility`: string, besides standard MastoAPI values (`direct`, `private`, `unlisted` or `public`) it can be used to address a List by setting it to `list:LIST_ID`. +- `visibility`: string, besides standard MastoAPI values (`direct`, `private`, `unlisted`, `local` or `public`) it can be used to address a List by setting it to `list:LIST_ID`.  - `expires_in`: The number of seconds the posted activity should expire in. When a posted activity expires it will be deleted from the server, and a delete request for it will be federated. This needs to be longer than an hour.  - `in_reply_to_conversation_id`: Will reply to a given conversation, addressing only the people who are part of the recipient set of that conversation. Sets the visibility to `direct`. diff --git a/lib/pleroma/constants.ex b/lib/pleroma/constants.ex index 13eeaa96b..cf8182d55 100644 --- a/lib/pleroma/constants.ex +++ b/lib/pleroma/constants.ex @@ -26,4 +26,6 @@ defmodule Pleroma.Constants do      do:        ~w(index.html robots.txt static static-fe finmoji emoji packs sounds images instance sw.js sw-pleroma.js favicon.png schemas doc embed.js embed.css)    ) + +  def as_local_public, do: Pleroma.Web.base_url() <> "/#Public"  end diff --git a/lib/pleroma/web/activity_pub/activity_pub_controller.ex b/lib/pleroma/web/activity_pub/activity_pub_controller.ex index 31df80adb..7e5647f8f 100644 --- a/lib/pleroma/web/activity_pub/activity_pub_controller.ex +++ b/lib/pleroma/web/activity_pub/activity_pub_controller.ex @@ -82,7 +82,8 @@ defmodule Pleroma.Web.ActivityPub.ActivityPubController do    def object(conn, _) do      with ap_id <- Endpoint.url() <> conn.request_path,           %Object{} = object <- Object.get_cached_by_ap_id(ap_id), -         {_, true} <- {:public?, Visibility.is_public?(object)} do +         {_, true} <- {:public?, Visibility.is_public?(object)}, +         {_, false} <- {:local?, Visibility.is_local_public?(object)} do        conn        |> assign(:tracking_fun_data, object.id)        |> set_cache_ttl_for(object) @@ -92,6 +93,9 @@ defmodule Pleroma.Web.ActivityPub.ActivityPubController do      else        {:public?, false} ->          {:error, :not_found} + +      {:local?, true} -> +        {:error, :not_found}      end    end @@ -108,7 +112,8 @@ defmodule Pleroma.Web.ActivityPub.ActivityPubController do    def activity(conn, _params) do      with ap_id <- Endpoint.url() <> conn.request_path,           %Activity{} = activity <- Activity.normalize(ap_id), -         {_, true} <- {:public?, Visibility.is_public?(activity)} do +         {_, true} <- {:public?, Visibility.is_public?(activity)}, +         {_, false} <- {:local?, Visibility.is_local_public?(activity)} do        conn        |> maybe_set_tracking_data(activity)        |> set_cache_ttl_for(activity) @@ -117,6 +122,7 @@ defmodule Pleroma.Web.ActivityPub.ActivityPubController do        |> render("object.json", object: activity)      else        {:public?, false} -> {:error, :not_found} +      {:local?, true} -> {:error, :not_found}        nil -> {:error, :not_found}      end    end diff --git a/lib/pleroma/web/activity_pub/builder.ex b/lib/pleroma/web/activity_pub/builder.ex index 298aff6b7..e99f6fd83 100644 --- a/lib/pleroma/web/activity_pub/builder.ex +++ b/lib/pleroma/web/activity_pub/builder.ex @@ -222,6 +222,9 @@ defmodule Pleroma.Web.ActivityPub.Builder do          actor.ap_id == Relay.ap_id() ->            [actor.follower_address] +        public? and Visibility.is_local_public?(object) -> +          [actor.follower_address, object.data["actor"], Pleroma.Constants.as_local_public()] +          public? ->            [actor.follower_address, object.data["actor"], Pleroma.Constants.as_public()] diff --git a/lib/pleroma/web/activity_pub/object_validators/announce_validator.ex b/lib/pleroma/web/activity_pub/object_validators/announce_validator.ex index 6f757f49c..338957db8 100644 --- a/lib/pleroma/web/activity_pub/object_validators/announce_validator.ex +++ b/lib/pleroma/web/activity_pub/object_validators/announce_validator.ex @@ -67,7 +67,12 @@ defmodule Pleroma.Web.ActivityPub.ObjectValidators.AnnounceValidator do           %Object{} = object <- Object.get_cached_by_ap_id(object),           false <- Visibility.is_public?(object) do        same_actor = object.data["actor"] == actor.ap_id -      is_public = Pleroma.Constants.as_public() in (get_field(cng, :to) ++ get_field(cng, :cc)) +      recipients = get_field(cng, :to) ++ get_field(cng, :cc) +      local_public = Pleroma.Constants.as_local_public() + +      is_public = +        Enum.member?(recipients, Pleroma.Constants.as_public()) or +          Enum.member?(recipients, local_public)        cond do          same_actor && is_public -> diff --git a/lib/pleroma/web/activity_pub/pipeline.ex b/lib/pleroma/web/activity_pub/pipeline.ex index 2db86f116..98c32a42b 100644 --- a/lib/pleroma/web/activity_pub/pipeline.ex +++ b/lib/pleroma/web/activity_pub/pipeline.ex @@ -11,6 +11,7 @@ defmodule Pleroma.Web.ActivityPub.Pipeline do    alias Pleroma.Web.ActivityPub.MRF    alias Pleroma.Web.ActivityPub.ObjectValidator    alias Pleroma.Web.ActivityPub.SideEffects +  alias Pleroma.Web.ActivityPub.Visibility    alias Pleroma.Web.Federator    @spec common_pipeline(map(), keyword()) :: @@ -55,7 +56,7 @@ defmodule Pleroma.Web.ActivityPub.Pipeline do      with {:ok, local} <- Keyword.fetch(meta, :local) do        do_not_federate = meta[:do_not_federate] || !Config.get([:instance, :federating]) -      if !do_not_federate && local do +      if !do_not_federate and local and not Visibility.is_local_public?(activity) do          activity =            if object = Keyword.get(meta, :object_data) do              %{activity | data: Map.put(activity.data, "object", object)} diff --git a/lib/pleroma/web/activity_pub/utils.ex b/lib/pleroma/web/activity_pub/utils.ex index 713b0ca1f..46002bec2 100644 --- a/lib/pleroma/web/activity_pub/utils.ex +++ b/lib/pleroma/web/activity_pub/utils.ex @@ -175,7 +175,8 @@ defmodule Pleroma.Web.ActivityPub.Utils do      outgoing_blocks = Config.get([:activitypub, :outgoing_blocks])      with true <- Config.get!([:instance, :federating]), -         true <- type != "Block" || outgoing_blocks do +         true <- type != "Block" || outgoing_blocks, +         false <- Visibility.is_local_public?(activity) do        Pleroma.Web.Federator.publish(activity)      end diff --git a/lib/pleroma/web/activity_pub/visibility.ex b/lib/pleroma/web/activity_pub/visibility.ex index 76bd54a42..2cb5a2bd0 100644 --- a/lib/pleroma/web/activity_pub/visibility.ex +++ b/lib/pleroma/web/activity_pub/visibility.ex @@ -17,7 +17,19 @@ defmodule Pleroma.Web.ActivityPub.Visibility do    def is_public?(%Activity{data: %{"type" => "Move"}}), do: true    def is_public?(%Activity{data: data}), do: is_public?(data)    def is_public?(%{"directMessage" => true}), do: false -  def is_public?(data), do: Utils.label_in_message?(Pleroma.Constants.as_public(), data) + +  def is_public?(data) do +    Utils.label_in_message?(Pleroma.Constants.as_public(), data) or +      Utils.label_in_message?(Pleroma.Constants.as_local_public(), data) +  end + +  def is_local_public?(%Object{data: data}), do: is_local_public?(data) +  def is_local_public?(%Activity{data: data}), do: is_local_public?(data) + +  def is_local_public?(data) do +    Utils.label_in_message?(Pleroma.Constants.as_local_public(), data) and +      not Utils.label_in_message?(Pleroma.Constants.as_public(), data) +  end    def is_private?(activity) do      with false <- is_public?(activity), @@ -114,6 +126,9 @@ defmodule Pleroma.Web.ActivityPub.Visibility do        Pleroma.Constants.as_public() in cc ->          "unlisted" +      Pleroma.Constants.as_local_public() in to -> +        "local" +        # this should use the sql for the object's activity        Enum.any?(to, &String.contains?(&1, "/followers")) ->          "private" diff --git a/lib/pleroma/web/api_spec/schemas/visibility_scope.ex b/lib/pleroma/web/api_spec/schemas/visibility_scope.ex index 831734e27..633269a92 100644 --- a/lib/pleroma/web/api_spec/schemas/visibility_scope.ex +++ b/lib/pleroma/web/api_spec/schemas/visibility_scope.ex @@ -9,6 +9,6 @@ defmodule Pleroma.Web.ApiSpec.Schemas.VisibilityScope do      title: "VisibilityScope",      description: "Status visibility",      type: :string, -    enum: ["public", "unlisted", "private", "direct", "list"] +    enum: ["public", "unlisted", "local", "private", "direct", "list"]    })  end diff --git a/lib/pleroma/web/common_api.ex b/lib/pleroma/web/common_api.ex index 0ab1b115d..e59254791 100644 --- a/lib/pleroma/web/common_api.ex +++ b/lib/pleroma/web/common_api.ex @@ -15,6 +15,7 @@ defmodule Pleroma.Web.CommonAPI do    alias Pleroma.Web.ActivityPub.Pipeline    alias Pleroma.Web.ActivityPub.Utils    alias Pleroma.Web.ActivityPub.Visibility +  alias Pleroma.Web.CommonAPI.ActivityDraft    import Pleroma.Web.Gettext    import Pleroma.Web.CommonAPI.Utils @@ -358,7 +359,7 @@ defmodule Pleroma.Web.CommonAPI do    def get_visibility(_, _, %Participation{}), do: {"direct", "direct"}    def get_visibility(%{visibility: visibility}, in_reply_to, _) -      when visibility in ~w{public unlisted private direct}, +      when visibility in ~w{public local unlisted private direct},        do: {visibility, get_replied_to_visibility(in_reply_to)}    def get_visibility(%{visibility: "list:" <> list_id}, in_reply_to, _) do @@ -399,31 +400,13 @@ defmodule Pleroma.Web.CommonAPI do    end    def listen(user, data) do -    visibility = Map.get(data, :visibility, "public") - -    with {to, cc} <- get_to_and_cc(user, [], nil, visibility, nil), -         listen_data <- -           data -           |> Map.take([:album, :artist, :title, :length]) -           |> Map.new(fn {key, value} -> {to_string(key), value} end) -           |> Map.put("type", "Audio") -           |> Map.put("to", to) -           |> Map.put("cc", cc) -           |> Map.put("actor", user.ap_id), -         {:ok, activity} <- -           ActivityPub.listen(%{ -             actor: user, -             to: to, -             object: listen_data, -             context: Utils.generate_context_id(), -             additional: %{"cc" => cc} -           }) do -      {:ok, activity} +    with {:ok, draft} <- ActivityDraft.listen(user, data) do +      ActivityPub.listen(draft.changes)      end    end    def post(user, %{status: _} = data) do -    with {:ok, draft} <- Pleroma.Web.CommonAPI.ActivityDraft.create(user, data) do +    with {:ok, draft} <- ActivityDraft.create(user, data) do        ActivityPub.create(draft.changes, draft.preview?)      end    end diff --git a/lib/pleroma/web/common_api/activity_draft.ex b/lib/pleroma/web/common_api/activity_draft.ex index 548f76609..aa2616d9e 100644 --- a/lib/pleroma/web/common_api/activity_draft.ex +++ b/lib/pleroma/web/common_api/activity_draft.ex @@ -22,7 +22,7 @@ defmodule Pleroma.Web.CommonAPI.ActivityDraft do              in_reply_to_conversation: nil,              visibility: nil,              expires_at: nil, -            poll: nil, +            extra: nil,              emoji: %{},              content_html: nil,              mentions: [], @@ -35,9 +35,14 @@ defmodule Pleroma.Web.CommonAPI.ActivityDraft do              preview?: false,              changes: %{} -  def create(user, params) do +  def new(user, params) do      %__MODULE__{user: user}      |> put_params(params) +  end + +  def create(user, params) do +    user +    |> new(params)      |> status()      |> summary()      |> with_valid(&attachments/1) @@ -57,6 +62,30 @@ defmodule Pleroma.Web.CommonAPI.ActivityDraft do      |> validate()    end +  def listen(user, params) do +    user +    |> new(params) +    |> visibility() +    |> to_and_cc() +    |> context() +    |> listen_object() +    |> with_valid(&changes/1) +    |> validate() +  end + +  defp listen_object(draft) do +    object = +      draft.params +      |> Map.take([:album, :artist, :title, :length]) +      |> Map.new(fn {key, value} -> {to_string(key), value} end) +      |> Map.put("type", "Audio") +      |> Map.put("to", draft.to) +      |> Map.put("cc", draft.cc) +      |> Map.put("actor", draft.user.ap_id) + +    %__MODULE__{draft | object: object} +  end +    defp put_params(draft, params) do      params = Map.put_new(params, :in_reply_to_status_id, params[:in_reply_to_id])      %__MODULE__{draft | params: params} @@ -121,7 +150,7 @@ defmodule Pleroma.Web.CommonAPI.ActivityDraft do    defp poll(draft) do      case Utils.make_poll_data(draft.params) do        {:ok, {poll, poll_emoji}} -> -        %__MODULE__{draft | poll: poll, emoji: Map.merge(draft.emoji, poll_emoji)} +        %__MODULE__{draft | extra: poll, emoji: Map.merge(draft.emoji, poll_emoji)}        {:error, message} ->          add_error(draft, message) @@ -129,32 +158,18 @@ defmodule Pleroma.Web.CommonAPI.ActivityDraft do    end    defp content(draft) do -    {content_html, mentions, tags} = -      Utils.make_content_html( -        draft.status, -        draft.attachments, -        draft.params, -        draft.visibility -      ) - -    %__MODULE__{draft | content_html: content_html, mentions: mentions, tags: tags} -  end +    {content_html, mentioned_users, tags} = Utils.make_content_html(draft) -  defp to_and_cc(draft) do -    addressed_users = -      draft.mentions +    mentions = +      mentioned_users        |> Enum.map(fn {_, mentioned_user} -> mentioned_user.ap_id end)        |> Utils.get_addressed_users(draft.params[:to]) -    {to, cc} = -      Utils.get_to_and_cc( -        draft.user, -        addressed_users, -        draft.in_reply_to, -        draft.visibility, -        draft.in_reply_to_conversation -      ) +    %__MODULE__{draft | content_html: content_html, mentions: mentions, tags: tags} +  end +  defp to_and_cc(draft) do +    {to, cc} = Utils.get_to_and_cc(draft)      %__MODULE__{draft | to: to, cc: cc}    end @@ -172,19 +187,7 @@ defmodule Pleroma.Web.CommonAPI.ActivityDraft do      emoji = Map.merge(Pleroma.Emoji.Formatter.get_emoji_map(draft.full_payload), draft.emoji)      object = -      Utils.make_note_data( -        draft.user.ap_id, -        draft.to, -        draft.context, -        draft.content_html, -        draft.attachments, -        draft.in_reply_to, -        draft.tags, -        draft.summary, -        draft.cc, -        draft.sensitive, -        draft.poll -      ) +      Utils.make_note_data(draft)        |> Map.put("emoji", emoji)        |> Map.put("source", draft.status) diff --git a/lib/pleroma/web/common_api/utils.ex b/lib/pleroma/web/common_api/utils.ex index 3b71adf0e..1c74ea787 100644 --- a/lib/pleroma/web/common_api/utils.ex +++ b/lib/pleroma/web/common_api/utils.ex @@ -16,6 +16,7 @@ defmodule Pleroma.Web.CommonAPI.Utils do    alias Pleroma.User    alias Pleroma.Web.ActivityPub.Utils    alias Pleroma.Web.ActivityPub.Visibility +  alias Pleroma.Web.CommonAPI.ActivityDraft    alias Pleroma.Web.MediaProxy    alias Pleroma.Web.Plugs.AuthenticationPlug @@ -50,67 +51,62 @@ defmodule Pleroma.Web.CommonAPI.Utils do      {_, descs} = Jason.decode(descs_str)      Enum.map(ids, fn media_id -> -      case Repo.get(Object, media_id) do -        %Object{data: data} -> -          Map.put(data, "name", descs[media_id]) - -        _ -> -          nil +      with %Object{data: data} <- Repo.get(Object, media_id) do +        Map.put(data, "name", descs[media_id])        end      end)      |> Enum.reject(&is_nil/1)    end -  @spec get_to_and_cc( -          User.t(), -          list(String.t()), -          Activity.t() | nil, -          String.t(), -          Participation.t() | nil -        ) :: {list(String.t()), list(String.t())} +  @spec get_to_and_cc(ActivityDraft.t()) :: {list(String.t()), list(String.t())} -  def get_to_and_cc(_, _, _, _, %Participation{} = participation) do +  def get_to_and_cc(%{in_reply_to_conversation: %Participation{} = participation}) do      participation = Repo.preload(participation, :recipients)      {Enum.map(participation.recipients, & &1.ap_id), []}    end -  def get_to_and_cc(user, mentioned_users, inReplyTo, "public", _) do -    to = [Pleroma.Constants.as_public() | mentioned_users] -    cc = [user.follower_address] +  def get_to_and_cc(%{visibility: visibility} = draft) when visibility in ["public", "local"] do +    to = +      case visibility do +        "public" -> [Pleroma.Constants.as_public() | draft.mentions] +        "local" -> [Pleroma.Constants.as_local_public() | draft.mentions] +      end + +    cc = [draft.user.follower_address] -    if inReplyTo do -      {Enum.uniq([inReplyTo.data["actor"] | to]), cc} +    if draft.in_reply_to do +      {Enum.uniq([draft.in_reply_to.data["actor"] | to]), cc}      else        {to, cc}      end    end -  def get_to_and_cc(user, mentioned_users, inReplyTo, "unlisted", _) do -    to = [user.follower_address | mentioned_users] +  def get_to_and_cc(%{visibility: "unlisted"} = draft) do +    to = [draft.user.follower_address | draft.mentions]      cc = [Pleroma.Constants.as_public()] -    if inReplyTo do -      {Enum.uniq([inReplyTo.data["actor"] | to]), cc} +    if draft.in_reply_to do +      {Enum.uniq([draft.in_reply_to.data["actor"] | to]), cc}      else        {to, cc}      end    end -  def get_to_and_cc(user, mentioned_users, inReplyTo, "private", _) do -    {to, cc} = get_to_and_cc(user, mentioned_users, inReplyTo, "direct", nil) -    {[user.follower_address | to], cc} +  def get_to_and_cc(%{visibility: "private"} = draft) do +    {to, cc} = get_to_and_cc(struct(draft, visibility: "direct")) +    {[draft.user.follower_address | to], cc}    end -  def get_to_and_cc(_user, mentioned_users, inReplyTo, "direct", _) do +  def get_to_and_cc(%{visibility: "direct"} = draft) do      # If the OP is a DM already, add the implicit actor. -    if inReplyTo && Visibility.is_direct?(inReplyTo) do -      {Enum.uniq([inReplyTo.data["actor"] | mentioned_users]), []} +    if draft.in_reply_to && Visibility.is_direct?(draft.in_reply_to) do +      {Enum.uniq([draft.in_reply_to.data["actor"] | draft.mentions]), []}      else -      {mentioned_users, []} +      {draft.mentions, []}      end    end -  def get_to_and_cc(_user, mentions, _inReplyTo, {:list, _}, _), do: {mentions, []} +  def get_to_and_cc(%{visibility: {:list, _}, mentions: mentions}), do: {mentions, []}    def get_addressed_users(_, to) when is_list(to) do      User.get_ap_ids_by_nicknames(to) @@ -203,30 +199,25 @@ defmodule Pleroma.Web.CommonAPI.Utils do      end    end -  def make_content_html( -        status, -        attachments, -        data, -        visibility -      ) do +  def make_content_html(%ActivityDraft{} = draft) do      attachment_links = -      data +      draft.params        |> Map.get("attachment_links", Config.get([:instance, :attachment_links]))        |> truthy_param?() -    content_type = get_content_type(data[:content_type]) +    content_type = get_content_type(draft.params[:content_type])      options = -      if visibility == "direct" && Config.get([:instance, :safe_dm_mentions]) do +      if draft.visibility == "direct" && Config.get([:instance, :safe_dm_mentions]) do          [safe_mention: true]        else          []        end -    status +    draft.status      |> format_input(content_type, options) -    |> maybe_add_attachments(attachments, attachment_links) -    |> maybe_add_nsfw_tag(data) +    |> maybe_add_attachments(draft.attachments, attachment_links) +    |> maybe_add_nsfw_tag(draft.params)    end    defp get_content_type(content_type) do @@ -308,33 +299,21 @@ defmodule Pleroma.Web.CommonAPI.Utils do      |> Formatter.html_escape("text/html")    end -  def make_note_data( -        actor, -        to, -        context, -        content_html, -        attachments, -        in_reply_to, -        tags, -        summary \\ nil, -        cc \\ [], -        sensitive \\ false, -        extra_params \\ %{} -      ) do +  def make_note_data(%ActivityDraft{} = draft) do      %{        "type" => "Note", -      "to" => to, -      "cc" => cc, -      "content" => content_html, -      "summary" => summary, -      "sensitive" => truthy_param?(sensitive), -      "context" => context, -      "attachment" => attachments, -      "actor" => actor, -      "tag" => Keyword.values(tags) |> Enum.uniq() +      "to" => draft.to, +      "cc" => draft.cc, +      "content" => draft.content_html, +      "summary" => draft.summary, +      "sensitive" => draft.sensitive, +      "context" => draft.context, +      "attachment" => draft.attachments, +      "actor" => draft.user.ap_id, +      "tag" => Keyword.values(draft.tags) |> Enum.uniq()      } -    |> add_in_reply_to(in_reply_to) -    |> Map.merge(extra_params) +    |> add_in_reply_to(draft.in_reply_to) +    |> Map.merge(draft.extra)    end    defp add_in_reply_to(object, nil), do: object diff --git a/test/pleroma/web/activity_pub/activity_pub_controller_test.exs b/test/pleroma/web/activity_pub/activity_pub_controller_test.exs index b696a24f4..31e48f87f 100644 --- a/test/pleroma/web/activity_pub/activity_pub_controller_test.exs +++ b/test/pleroma/web/activity_pub/activity_pub_controller_test.exs @@ -213,6 +213,23 @@ defmodule Pleroma.Web.ActivityPub.ActivityPubControllerTest do    end    describe "/objects/:uuid" do +    test "it doesn't return a local-only object", %{conn: conn} do +      user = insert(:user) +      {:ok, post} = CommonAPI.post(user, %{status: "test", visibility: "local"}) + +      assert Pleroma.Web.ActivityPub.Visibility.is_local_public?(post) + +      object = Object.normalize(post, false) +      uuid = String.split(object.data["id"], "/") |> List.last() + +      conn = +        conn +        |> put_req_header("accept", "application/json") +        |> get("/objects/#{uuid}") + +      assert json_response(conn, 404) +    end +      test "it returns a json representation of the object with accept application/json", %{        conn: conn      } do @@ -326,6 +343,22 @@ defmodule Pleroma.Web.ActivityPub.ActivityPubControllerTest do    end    describe "/activities/:uuid" do +    test "it doesn't return a local-only activity", %{conn: conn} do +      user = insert(:user) +      {:ok, post} = CommonAPI.post(user, %{status: "test", visibility: "local"}) + +      assert Pleroma.Web.ActivityPub.Visibility.is_local_public?(post) + +      uuid = String.split(post.data["id"], "/") |> List.last() + +      conn = +        conn +        |> put_req_header("accept", "application/json") +        |> get("/activities/#{uuid}") + +      assert json_response(conn, 404) +    end +      test "it returns a json representation of the activity", %{conn: conn} do        activity = insert(:note_activity)        uuid = String.split(activity.data["id"], "/") |> List.last() diff --git a/test/pleroma/web/activity_pub/transmogrifier/note_handling_test.exs b/test/pleroma/web/activity_pub/transmogrifier/note_handling_test.exs new file mode 100644 index 000000000..b4a006aec --- /dev/null +++ b/test/pleroma/web/activity_pub/transmogrifier/note_handling_test.exs @@ -0,0 +1,749 @@ +# Pleroma: A lightweight social networking server +# Copyright © 2017-2020 Pleroma Authors <https://pleroma.social/> +# SPDX-License-Identifier: AGPL-3.0-only + +defmodule Pleroma.Web.ActivityPub.Transmogrifier.NoteHandlingTest do +  use Oban.Testing, repo: Pleroma.Repo +  use Pleroma.DataCase + +  alias Pleroma.Activity +  alias Pleroma.Object +  alias Pleroma.User +  alias Pleroma.Web.ActivityPub.Transmogrifier +  alias Pleroma.Web.CommonAPI + +  import Mock +  import Pleroma.Factory +  import ExUnit.CaptureLog + +  setup_all do +    Tesla.Mock.mock_global(fn env -> apply(HttpRequestMock, :request, [env]) end) +    :ok +  end + +  setup do: clear_config([:instance, :max_remote_account_fields]) + +  describe "handle_incoming" do +    test "it works for incoming notices with tag not being an array (kroeg)" do +      data = File.read!("test/fixtures/kroeg-array-less-emoji.json") |> Jason.decode!() + +      {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data) +      object = Object.normalize(data["object"]) + +      assert object.data["emoji"] == %{ +               "icon_e_smile" => "https://puckipedia.com/forum/images/smilies/icon_e_smile.png" +             } + +      data = File.read!("test/fixtures/kroeg-array-less-hashtag.json") |> Jason.decode!() + +      {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data) +      object = Object.normalize(data["object"]) + +      assert "test" in object.data["tag"] +    end + +    test "it cleans up incoming notices which are not really DMs" do +      user = insert(:user) +      other_user = insert(:user) + +      to = [user.ap_id, other_user.ap_id] + +      data = +        File.read!("test/fixtures/mastodon-post-activity.json") +        |> Jason.decode!() +        |> Map.put("to", to) +        |> Map.put("cc", []) + +      object = +        data["object"] +        |> Map.put("to", to) +        |> Map.put("cc", []) + +      data = Map.put(data, "object", object) + +      {:ok, %Activity{data: data, local: false} = activity} = Transmogrifier.handle_incoming(data) + +      assert data["to"] == [] +      assert data["cc"] == to + +      object_data = Object.normalize(activity).data + +      assert object_data["to"] == [] +      assert object_data["cc"] == to +    end + +    test "it ignores an incoming notice if we already have it" do +      activity = insert(:note_activity) + +      data = +        File.read!("test/fixtures/mastodon-post-activity.json") +        |> Jason.decode!() +        |> Map.put("object", Object.normalize(activity).data) + +      {:ok, returned_activity} = Transmogrifier.handle_incoming(data) + +      assert activity == returned_activity +    end + +    @tag capture_log: true +    test "it fetches reply-to activities if we don't have them" do +      data = +        File.read!("test/fixtures/mastodon-post-activity.json") +        |> Jason.decode!() + +      object = +        data["object"] +        |> Map.put("inReplyTo", "https://mstdn.io/users/mayuutann/statuses/99568293732299394") + +      data = Map.put(data, "object", object) +      {:ok, returned_activity} = Transmogrifier.handle_incoming(data) +      returned_object = Object.normalize(returned_activity, false) + +      assert %Activity{} = +               Activity.get_create_by_object_ap_id( +                 "https://mstdn.io/users/mayuutann/statuses/99568293732299394" +               ) + +      assert returned_object.data["inReplyTo"] == +               "https://mstdn.io/users/mayuutann/statuses/99568293732299394" +    end + +    test "it does not fetch reply-to activities beyond max replies depth limit" do +      data = +        File.read!("test/fixtures/mastodon-post-activity.json") +        |> Jason.decode!() + +      object = +        data["object"] +        |> Map.put("inReplyTo", "https://shitposter.club/notice/2827873") + +      data = Map.put(data, "object", object) + +      with_mock Pleroma.Web.Federator, +        allowed_thread_distance?: fn _ -> false end do +        {:ok, returned_activity} = Transmogrifier.handle_incoming(data) + +        returned_object = Object.normalize(returned_activity, false) + +        refute Activity.get_create_by_object_ap_id( +                 "tag:shitposter.club,2017-05-05:noticeId=2827873:objectType=comment" +               ) + +        assert returned_object.data["inReplyTo"] == "https://shitposter.club/notice/2827873" +      end +    end + +    test "it does not crash if the object in inReplyTo can't be fetched" do +      data = +        File.read!("test/fixtures/mastodon-post-activity.json") +        |> Jason.decode!() + +      object = +        data["object"] +        |> Map.put("inReplyTo", "https://404.site/whatever") + +      data = +        data +        |> Map.put("object", object) + +      assert capture_log(fn -> +               {:ok, _returned_activity} = Transmogrifier.handle_incoming(data) +             end) =~ "[warn] Couldn't fetch \"https://404.site/whatever\", error: nil" +    end + +    test "it does not work for deactivated users" do +      data = File.read!("test/fixtures/mastodon-post-activity.json") |> Jason.decode!() + +      insert(:user, ap_id: data["actor"], deactivated: true) + +      assert {:error, _} = Transmogrifier.handle_incoming(data) +    end + +    test "it works for incoming notices" do +      data = File.read!("test/fixtures/mastodon-post-activity.json") |> Jason.decode!() + +      {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data) + +      assert data["id"] == +               "http://mastodon.example.org/users/admin/statuses/99512778738411822/activity" + +      assert data["context"] == +               "tag:mastodon.example.org,2018-02-12:objectId=20:objectType=Conversation" + +      assert data["to"] == ["https://www.w3.org/ns/activitystreams#Public"] + +      assert data["cc"] == [ +               "http://mastodon.example.org/users/admin/followers", +               "http://localtesting.pleroma.lol/users/lain" +             ] + +      assert data["actor"] == "http://mastodon.example.org/users/admin" + +      object_data = Object.normalize(data["object"]).data + +      assert object_data["id"] == +               "http://mastodon.example.org/users/admin/statuses/99512778738411822" + +      assert object_data["to"] == ["https://www.w3.org/ns/activitystreams#Public"] + +      assert object_data["cc"] == [ +               "http://mastodon.example.org/users/admin/followers", +               "http://localtesting.pleroma.lol/users/lain" +             ] + +      assert object_data["actor"] == "http://mastodon.example.org/users/admin" +      assert object_data["attributedTo"] == "http://mastodon.example.org/users/admin" + +      assert object_data["context"] == +               "tag:mastodon.example.org,2018-02-12:objectId=20:objectType=Conversation" + +      assert object_data["sensitive"] == true + +      user = User.get_cached_by_ap_id(object_data["actor"]) + +      assert user.note_count == 1 +    end + +    test "it works for incoming notices without the sensitive property but an nsfw hashtag" do +      data = File.read!("test/fixtures/mastodon-post-activity-nsfw.json") |> Jason.decode!() + +      {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data) + +      object_data = Object.normalize(data["object"], false).data + +      assert object_data["sensitive"] == true +    end + +    test "it works for incoming notices with hashtags" do +      data = File.read!("test/fixtures/mastodon-post-activity-hashtag.json") |> Jason.decode!() + +      {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data) +      object = Object.normalize(data["object"]) + +      assert Enum.at(object.data["tag"], 2) == "moo" +    end + +    test "it works for incoming notices with contentMap" do +      data = File.read!("test/fixtures/mastodon-post-activity-contentmap.json") |> Jason.decode!() + +      {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data) +      object = Object.normalize(data["object"]) + +      assert object.data["content"] == +               "<p><span class=\"h-card\"><a href=\"http://localtesting.pleroma.lol/users/lain\" class=\"u-url mention\">@<span>lain</span></a></span></p>" +    end + +    test "it works for incoming notices with to/cc not being an array (kroeg)" do +      data = File.read!("test/fixtures/kroeg-post-activity.json") |> Jason.decode!() + +      {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data) +      object = Object.normalize(data["object"]) + +      assert object.data["content"] == +               "<p>henlo from my Psion netBook</p><p>message sent from my Psion netBook</p>" +    end + +    test "it ensures that as:Public activities make it to their followers collection" do +      user = insert(:user) + +      data = +        File.read!("test/fixtures/mastodon-post-activity.json") +        |> Jason.decode!() +        |> Map.put("actor", user.ap_id) +        |> Map.put("to", ["https://www.w3.org/ns/activitystreams#Public"]) +        |> Map.put("cc", []) + +      object = +        data["object"] +        |> Map.put("attributedTo", user.ap_id) +        |> Map.put("to", ["https://www.w3.org/ns/activitystreams#Public"]) +        |> Map.put("cc", []) +        |> Map.put("id", user.ap_id <> "/activities/12345678") + +      data = Map.put(data, "object", object) + +      {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data) + +      assert data["cc"] == [User.ap_followers(user)] +    end + +    test "it ensures that address fields become lists" do +      user = insert(:user) + +      data = +        File.read!("test/fixtures/mastodon-post-activity.json") +        |> Jason.decode!() +        |> Map.put("actor", user.ap_id) +        |> Map.put("to", nil) +        |> Map.put("cc", nil) + +      object = +        data["object"] +        |> Map.put("attributedTo", user.ap_id) +        |> Map.put("to", nil) +        |> Map.put("cc", nil) +        |> Map.put("id", user.ap_id <> "/activities/12345678") + +      data = Map.put(data, "object", object) + +      {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data) + +      assert !is_nil(data["to"]) +      assert !is_nil(data["cc"]) +    end + +    test "it strips internal likes" do +      data = +        File.read!("test/fixtures/mastodon-post-activity.json") +        |> Jason.decode!() + +      likes = %{ +        "first" => +          "http://mastodon.example.org/objects/dbdbc507-52c8-490d-9b7c-1e1d52e5c132/likes?page=1", +        "id" => "http://mastodon.example.org/objects/dbdbc507-52c8-490d-9b7c-1e1d52e5c132/likes", +        "totalItems" => 3, +        "type" => "OrderedCollection" +      } + +      object = Map.put(data["object"], "likes", likes) +      data = Map.put(data, "object", object) + +      {:ok, %Activity{object: object}} = Transmogrifier.handle_incoming(data) + +      refute Map.has_key?(object.data, "likes") +    end + +    test "it strips internal reactions" do +      user = insert(:user) +      {:ok, activity} = CommonAPI.post(user, %{status: "#cofe"}) +      {:ok, _} = CommonAPI.react_with_emoji(activity.id, user, "📢") + +      %{object: object} = Activity.get_by_id_with_object(activity.id) +      assert Map.has_key?(object.data, "reactions") +      assert Map.has_key?(object.data, "reaction_count") + +      object_data = Transmogrifier.strip_internal_fields(object.data) +      refute Map.has_key?(object_data, "reactions") +      refute Map.has_key?(object_data, "reaction_count") +    end + +    test "it correctly processes messages with non-array to field" do +      user = insert(:user) + +      message = %{ +        "@context" => "https://www.w3.org/ns/activitystreams", +        "to" => "https://www.w3.org/ns/activitystreams#Public", +        "type" => "Create", +        "object" => %{ +          "content" => "blah blah blah", +          "type" => "Note", +          "attributedTo" => user.ap_id, +          "inReplyTo" => nil +        }, +        "actor" => user.ap_id +      } + +      assert {:ok, activity} = Transmogrifier.handle_incoming(message) + +      assert ["https://www.w3.org/ns/activitystreams#Public"] == activity.data["to"] +    end + +    test "it correctly processes messages with non-array cc field" do +      user = insert(:user) + +      message = %{ +        "@context" => "https://www.w3.org/ns/activitystreams", +        "to" => user.follower_address, +        "cc" => "https://www.w3.org/ns/activitystreams#Public", +        "type" => "Create", +        "object" => %{ +          "content" => "blah blah blah", +          "type" => "Note", +          "attributedTo" => user.ap_id, +          "inReplyTo" => nil +        }, +        "actor" => user.ap_id +      } + +      assert {:ok, activity} = Transmogrifier.handle_incoming(message) + +      assert ["https://www.w3.org/ns/activitystreams#Public"] == activity.data["cc"] +      assert [user.follower_address] == activity.data["to"] +    end + +    test "it correctly processes messages with weirdness in address fields" do +      user = insert(:user) + +      message = %{ +        "@context" => "https://www.w3.org/ns/activitystreams", +        "to" => [nil, user.follower_address], +        "cc" => ["https://www.w3.org/ns/activitystreams#Public", ["¿"]], +        "type" => "Create", +        "object" => %{ +          "content" => "…", +          "type" => "Note", +          "attributedTo" => user.ap_id, +          "inReplyTo" => nil +        }, +        "actor" => user.ap_id +      } + +      assert {:ok, activity} = Transmogrifier.handle_incoming(message) + +      assert ["https://www.w3.org/ns/activitystreams#Public"] == activity.data["cc"] +      assert [user.follower_address] == activity.data["to"] +    end +  end + +  describe "`handle_incoming/2`, Mastodon format `replies` handling" do +    setup do: clear_config([:activitypub, :note_replies_output_limit], 5) +    setup do: clear_config([:instance, :federation_incoming_replies_max_depth]) + +    setup do +      data = +        "test/fixtures/mastodon-post-activity.json" +        |> File.read!() +        |> Jason.decode!() + +      items = get_in(data, ["object", "replies", "first", "items"]) +      assert length(items) > 0 + +      %{data: data, items: items} +    end + +    test "schedules background fetching of `replies` items if max thread depth limit allows", %{ +      data: data, +      items: items +    } do +      Pleroma.Config.put([:instance, :federation_incoming_replies_max_depth], 10) + +      {:ok, _activity} = Transmogrifier.handle_incoming(data) + +      for id <- items do +        job_args = %{"op" => "fetch_remote", "id" => id, "depth" => 1} +        assert_enqueued(worker: Pleroma.Workers.RemoteFetcherWorker, args: job_args) +      end +    end + +    test "does NOT schedule background fetching of `replies` beyond max thread depth limit allows", +         %{data: data} do +      Pleroma.Config.put([:instance, :federation_incoming_replies_max_depth], 0) + +      {:ok, _activity} = Transmogrifier.handle_incoming(data) + +      assert all_enqueued(worker: Pleroma.Workers.RemoteFetcherWorker) == [] +    end +  end + +  describe "`handle_incoming/2`, Pleroma format `replies` handling" do +    setup do: clear_config([:activitypub, :note_replies_output_limit], 5) +    setup do: clear_config([:instance, :federation_incoming_replies_max_depth]) + +    setup do +      user = insert(:user) + +      {:ok, activity} = CommonAPI.post(user, %{status: "post1"}) + +      {:ok, reply1} = +        CommonAPI.post(user, %{status: "reply1", in_reply_to_status_id: activity.id}) + +      {:ok, reply2} = +        CommonAPI.post(user, %{status: "reply2", in_reply_to_status_id: activity.id}) + +      replies_uris = Enum.map([reply1, reply2], fn a -> a.object.data["id"] end) + +      {:ok, federation_output} = Transmogrifier.prepare_outgoing(activity.data) + +      Repo.delete(activity.object) +      Repo.delete(activity) + +      %{federation_output: federation_output, replies_uris: replies_uris} +    end + +    test "schedules background fetching of `replies` items if max thread depth limit allows", %{ +      federation_output: federation_output, +      replies_uris: replies_uris +    } do +      Pleroma.Config.put([:instance, :federation_incoming_replies_max_depth], 1) + +      {:ok, _activity} = Transmogrifier.handle_incoming(federation_output) + +      for id <- replies_uris do +        job_args = %{"op" => "fetch_remote", "id" => id, "depth" => 1} +        assert_enqueued(worker: Pleroma.Workers.RemoteFetcherWorker, args: job_args) +      end +    end + +    test "does NOT schedule background fetching of `replies` beyond max thread depth limit allows", +         %{federation_output: federation_output} do +      Pleroma.Config.put([:instance, :federation_incoming_replies_max_depth], 0) + +      {:ok, _activity} = Transmogrifier.handle_incoming(federation_output) + +      assert all_enqueued(worker: Pleroma.Workers.RemoteFetcherWorker) == [] +    end +  end + +  describe "reserialization" do +    test "successfully reserializes a message with inReplyTo == nil" do +      user = insert(:user) + +      message = %{ +        "@context" => "https://www.w3.org/ns/activitystreams", +        "to" => ["https://www.w3.org/ns/activitystreams#Public"], +        "cc" => [], +        "type" => "Create", +        "object" => %{ +          "to" => ["https://www.w3.org/ns/activitystreams#Public"], +          "cc" => [], +          "type" => "Note", +          "content" => "Hi", +          "inReplyTo" => nil, +          "attributedTo" => user.ap_id +        }, +        "actor" => user.ap_id +      } + +      {:ok, activity} = Transmogrifier.handle_incoming(message) + +      {:ok, _} = Transmogrifier.prepare_outgoing(activity.data) +    end + +    test "successfully reserializes a message with AS2 objects in IR" do +      user = insert(:user) + +      message = %{ +        "@context" => "https://www.w3.org/ns/activitystreams", +        "to" => ["https://www.w3.org/ns/activitystreams#Public"], +        "cc" => [], +        "type" => "Create", +        "object" => %{ +          "to" => ["https://www.w3.org/ns/activitystreams#Public"], +          "cc" => [], +          "type" => "Note", +          "content" => "Hi", +          "inReplyTo" => nil, +          "attributedTo" => user.ap_id, +          "tag" => [ +            %{"name" => "#2hu", "href" => "http://example.com/2hu", "type" => "Hashtag"}, +            %{"name" => "Bob", "href" => "http://example.com/bob", "type" => "Mention"} +          ] +        }, +        "actor" => user.ap_id +      } + +      {:ok, activity} = Transmogrifier.handle_incoming(message) + +      {:ok, _} = Transmogrifier.prepare_outgoing(activity.data) +    end +  end + +  describe "fix_in_reply_to/2" do +    setup do: clear_config([:instance, :federation_incoming_replies_max_depth]) + +    setup do +      data = Jason.decode!(File.read!("test/fixtures/mastodon-post-activity.json")) +      [data: data] +    end + +    test "returns not modified object when hasn't containts inReplyTo field", %{data: data} do +      assert Transmogrifier.fix_in_reply_to(data) == data +    end + +    test "returns object with inReplyTo when denied incoming reply", %{data: data} do +      Pleroma.Config.put([:instance, :federation_incoming_replies_max_depth], 0) + +      object_with_reply = +        Map.put(data["object"], "inReplyTo", "https://shitposter.club/notice/2827873") + +      modified_object = Transmogrifier.fix_in_reply_to(object_with_reply) +      assert modified_object["inReplyTo"] == "https://shitposter.club/notice/2827873" + +      object_with_reply = +        Map.put(data["object"], "inReplyTo", %{"id" => "https://shitposter.club/notice/2827873"}) + +      modified_object = Transmogrifier.fix_in_reply_to(object_with_reply) +      assert modified_object["inReplyTo"] == %{"id" => "https://shitposter.club/notice/2827873"} + +      object_with_reply = +        Map.put(data["object"], "inReplyTo", ["https://shitposter.club/notice/2827873"]) + +      modified_object = Transmogrifier.fix_in_reply_to(object_with_reply) +      assert modified_object["inReplyTo"] == ["https://shitposter.club/notice/2827873"] + +      object_with_reply = Map.put(data["object"], "inReplyTo", []) +      modified_object = Transmogrifier.fix_in_reply_to(object_with_reply) +      assert modified_object["inReplyTo"] == [] +    end + +    @tag capture_log: true +    test "returns modified object when allowed incoming reply", %{data: data} do +      object_with_reply = +        Map.put( +          data["object"], +          "inReplyTo", +          "https://mstdn.io/users/mayuutann/statuses/99568293732299394" +        ) + +      Pleroma.Config.put([:instance, :federation_incoming_replies_max_depth], 5) +      modified_object = Transmogrifier.fix_in_reply_to(object_with_reply) + +      assert modified_object["inReplyTo"] == +               "https://mstdn.io/users/mayuutann/statuses/99568293732299394" + +      assert modified_object["context"] == +               "tag:shitposter.club,2018-02-22:objectType=thread:nonce=e5a7c72d60a9c0e4" +    end +  end + +  describe "fix_attachments/1" do +    test "returns not modified object" do +      data = Jason.decode!(File.read!("test/fixtures/mastodon-post-activity.json")) +      assert Transmogrifier.fix_attachments(data) == data +    end + +    test "returns modified object when attachment is map" do +      assert Transmogrifier.fix_attachments(%{ +               "attachment" => %{ +                 "mediaType" => "video/mp4", +                 "url" => "https://peertube.moe/stat-480.mp4" +               } +             }) == %{ +               "attachment" => [ +                 %{ +                   "mediaType" => "video/mp4", +                   "type" => "Document", +                   "url" => [ +                     %{ +                       "href" => "https://peertube.moe/stat-480.mp4", +                       "mediaType" => "video/mp4", +                       "type" => "Link" +                     } +                   ] +                 } +               ] +             } +    end + +    test "returns modified object when attachment is list" do +      assert Transmogrifier.fix_attachments(%{ +               "attachment" => [ +                 %{"mediaType" => "video/mp4", "url" => "https://pe.er/stat-480.mp4"}, +                 %{"mimeType" => "video/mp4", "href" => "https://pe.er/stat-480.mp4"} +               ] +             }) == %{ +               "attachment" => [ +                 %{ +                   "mediaType" => "video/mp4", +                   "type" => "Document", +                   "url" => [ +                     %{ +                       "href" => "https://pe.er/stat-480.mp4", +                       "mediaType" => "video/mp4", +                       "type" => "Link" +                     } +                   ] +                 }, +                 %{ +                   "mediaType" => "video/mp4", +                   "type" => "Document", +                   "url" => [ +                     %{ +                       "href" => "https://pe.er/stat-480.mp4", +                       "mediaType" => "video/mp4", +                       "type" => "Link" +                     } +                   ] +                 } +               ] +             } +    end +  end + +  describe "fix_emoji/1" do +    test "returns not modified object when object not contains tags" do +      data = Jason.decode!(File.read!("test/fixtures/mastodon-post-activity.json")) +      assert Transmogrifier.fix_emoji(data) == data +    end + +    test "returns object with emoji when object contains list tags" do +      assert Transmogrifier.fix_emoji(%{ +               "tag" => [ +                 %{"type" => "Emoji", "name" => ":bib:", "icon" => %{"url" => "/test"}}, +                 %{"type" => "Hashtag"} +               ] +             }) == %{ +               "emoji" => %{"bib" => "/test"}, +               "tag" => [ +                 %{"icon" => %{"url" => "/test"}, "name" => ":bib:", "type" => "Emoji"}, +                 %{"type" => "Hashtag"} +               ] +             } +    end + +    test "returns object with emoji when object contains map tag" do +      assert Transmogrifier.fix_emoji(%{ +               "tag" => %{"type" => "Emoji", "name" => ":bib:", "icon" => %{"url" => "/test"}} +             }) == %{ +               "emoji" => %{"bib" => "/test"}, +               "tag" => %{"icon" => %{"url" => "/test"}, "name" => ":bib:", "type" => "Emoji"} +             } +    end +  end + +  describe "set_replies/1" do +    setup do: clear_config([:activitypub, :note_replies_output_limit], 2) + +    test "returns unmodified object if activity doesn't have self-replies" do +      data = Jason.decode!(File.read!("test/fixtures/mastodon-post-activity.json")) +      assert Transmogrifier.set_replies(data) == data +    end + +    test "sets `replies` collection with a limited number of self-replies" do +      [user, another_user] = insert_list(2, :user) + +      {:ok, %{id: id1} = activity} = CommonAPI.post(user, %{status: "1"}) + +      {:ok, %{id: id2} = self_reply1} = +        CommonAPI.post(user, %{status: "self-reply 1", in_reply_to_status_id: id1}) + +      {:ok, self_reply2} = +        CommonAPI.post(user, %{status: "self-reply 2", in_reply_to_status_id: id1}) + +      # Assuming to _not_ be present in `replies` due to :note_replies_output_limit is set to 2 +      {:ok, _} = CommonAPI.post(user, %{status: "self-reply 3", in_reply_to_status_id: id1}) + +      {:ok, _} = +        CommonAPI.post(user, %{ +          status: "self-reply to self-reply", +          in_reply_to_status_id: id2 +        }) + +      {:ok, _} = +        CommonAPI.post(another_user, %{ +          status: "another user's reply", +          in_reply_to_status_id: id1 +        }) + +      object = Object.normalize(activity) +      replies_uris = Enum.map([self_reply1, self_reply2], fn a -> a.object.data["id"] end) + +      assert %{"type" => "Collection", "items" => ^replies_uris} = +               Transmogrifier.set_replies(object.data)["replies"] +    end +  end + +  test "take_emoji_tags/1" do +    user = insert(:user, %{emoji: %{"firefox" => "https://example.org/firefox.png"}}) + +    assert Transmogrifier.take_emoji_tags(user) == [ +             %{ +               "icon" => %{"type" => "Image", "url" => "https://example.org/firefox.png"}, +               "id" => "https://example.org/firefox.png", +               "name" => ":firefox:", +               "type" => "Emoji", +               "updated" => "1970-01-01T00:00:00Z" +             } +           ] +  end +end diff --git a/test/pleroma/web/activity_pub/transmogrifier_test.exs b/test/pleroma/web/activity_pub/transmogrifier_test.exs index e39af1dfc..333bb4f9b 100644 --- a/test/pleroma/web/activity_pub/transmogrifier_test.exs +++ b/test/pleroma/web/activity_pub/transmogrifier_test.exs @@ -26,310 +26,6 @@ defmodule Pleroma.Web.ActivityPub.TransmogrifierTest do    setup do: clear_config([:instance, :max_remote_account_fields])    describe "handle_incoming" do -    test "it works for incoming notices with tag not being an array (kroeg)" do -      data = File.read!("test/fixtures/kroeg-array-less-emoji.json") |> Poison.decode!() - -      {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data) -      object = Object.normalize(data["object"]) - -      assert object.data["emoji"] == %{ -               "icon_e_smile" => "https://puckipedia.com/forum/images/smilies/icon_e_smile.png" -             } - -      data = File.read!("test/fixtures/kroeg-array-less-hashtag.json") |> Poison.decode!() - -      {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data) -      object = Object.normalize(data["object"]) - -      assert "test" in object.data["tag"] -    end - -    test "it cleans up incoming notices which are not really DMs" do -      user = insert(:user) -      other_user = insert(:user) - -      to = [user.ap_id, other_user.ap_id] - -      data = -        File.read!("test/fixtures/mastodon-post-activity.json") -        |> Poison.decode!() -        |> Map.put("to", to) -        |> Map.put("cc", []) - -      object = -        data["object"] -        |> Map.put("to", to) -        |> Map.put("cc", []) - -      data = Map.put(data, "object", object) - -      {:ok, %Activity{data: data, local: false} = activity} = Transmogrifier.handle_incoming(data) - -      assert data["to"] == [] -      assert data["cc"] == to - -      object_data = Object.normalize(activity).data - -      assert object_data["to"] == [] -      assert object_data["cc"] == to -    end - -    test "it ignores an incoming notice if we already have it" do -      activity = insert(:note_activity) - -      data = -        File.read!("test/fixtures/mastodon-post-activity.json") -        |> Poison.decode!() -        |> Map.put("object", Object.normalize(activity).data) - -      {:ok, returned_activity} = Transmogrifier.handle_incoming(data) - -      assert activity == returned_activity -    end - -    @tag capture_log: true -    test "it fetches reply-to activities if we don't have them" do -      data = -        File.read!("test/fixtures/mastodon-post-activity.json") -        |> Poison.decode!() - -      object = -        data["object"] -        |> Map.put("inReplyTo", "https://mstdn.io/users/mayuutann/statuses/99568293732299394") - -      data = Map.put(data, "object", object) -      {:ok, returned_activity} = Transmogrifier.handle_incoming(data) -      returned_object = Object.normalize(returned_activity, false) - -      assert %Activity{} = -               Activity.get_create_by_object_ap_id( -                 "https://mstdn.io/users/mayuutann/statuses/99568293732299394" -               ) - -      assert returned_object.data["inReplyTo"] == -               "https://mstdn.io/users/mayuutann/statuses/99568293732299394" -    end - -    test "it does not fetch reply-to activities beyond max replies depth limit" do -      data = -        File.read!("test/fixtures/mastodon-post-activity.json") -        |> Poison.decode!() - -      object = -        data["object"] -        |> Map.put("inReplyTo", "https://shitposter.club/notice/2827873") - -      data = Map.put(data, "object", object) - -      with_mock Pleroma.Web.Federator, -        allowed_thread_distance?: fn _ -> false end do -        {:ok, returned_activity} = Transmogrifier.handle_incoming(data) - -        returned_object = Object.normalize(returned_activity, false) - -        refute Activity.get_create_by_object_ap_id( -                 "tag:shitposter.club,2017-05-05:noticeId=2827873:objectType=comment" -               ) - -        assert returned_object.data["inReplyTo"] == "https://shitposter.club/notice/2827873" -      end -    end - -    test "it does not crash if the object in inReplyTo can't be fetched" do -      data = -        File.read!("test/fixtures/mastodon-post-activity.json") -        |> Poison.decode!() - -      object = -        data["object"] -        |> Map.put("inReplyTo", "https://404.site/whatever") - -      data = -        data -        |> Map.put("object", object) - -      assert capture_log(fn -> -               {:ok, _returned_activity} = Transmogrifier.handle_incoming(data) -             end) =~ "[warn] Couldn't fetch \"https://404.site/whatever\", error: nil" -    end - -    test "it does not work for deactivated users" do -      data = File.read!("test/fixtures/mastodon-post-activity.json") |> Poison.decode!() - -      insert(:user, ap_id: data["actor"], deactivated: true) - -      assert {:error, _} = Transmogrifier.handle_incoming(data) -    end - -    test "it works for incoming notices" do -      data = File.read!("test/fixtures/mastodon-post-activity.json") |> Poison.decode!() - -      {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data) - -      assert data["id"] == -               "http://mastodon.example.org/users/admin/statuses/99512778738411822/activity" - -      assert data["context"] == -               "tag:mastodon.example.org,2018-02-12:objectId=20:objectType=Conversation" - -      assert data["to"] == ["https://www.w3.org/ns/activitystreams#Public"] - -      assert data["cc"] == [ -               "http://mastodon.example.org/users/admin/followers", -               "http://localtesting.pleroma.lol/users/lain" -             ] - -      assert data["actor"] == "http://mastodon.example.org/users/admin" - -      object_data = Object.normalize(data["object"]).data - -      assert object_data["id"] == -               "http://mastodon.example.org/users/admin/statuses/99512778738411822" - -      assert object_data["to"] == ["https://www.w3.org/ns/activitystreams#Public"] - -      assert object_data["cc"] == [ -               "http://mastodon.example.org/users/admin/followers", -               "http://localtesting.pleroma.lol/users/lain" -             ] - -      assert object_data["actor"] == "http://mastodon.example.org/users/admin" -      assert object_data["attributedTo"] == "http://mastodon.example.org/users/admin" - -      assert object_data["context"] == -               "tag:mastodon.example.org,2018-02-12:objectId=20:objectType=Conversation" - -      assert object_data["sensitive"] == true - -      user = User.get_cached_by_ap_id(object_data["actor"]) - -      assert user.note_count == 1 -    end - -    test "it works for incoming notices without the sensitive property but an nsfw hashtag" do -      data = File.read!("test/fixtures/mastodon-post-activity-nsfw.json") |> Poison.decode!() - -      {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data) - -      object_data = Object.normalize(data["object"], false).data - -      assert object_data["sensitive"] == true -    end - -    test "it works for incoming notices with hashtags" do -      data = File.read!("test/fixtures/mastodon-post-activity-hashtag.json") |> Poison.decode!() - -      {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data) -      object = Object.normalize(data["object"]) - -      assert Enum.at(object.data["tag"], 2) == "moo" -    end - -    test "it works for incoming notices with contentMap" do -      data = -        File.read!("test/fixtures/mastodon-post-activity-contentmap.json") |> Poison.decode!() - -      {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data) -      object = Object.normalize(data["object"]) - -      assert object.data["content"] == -               "<p><span class=\"h-card\"><a href=\"http://localtesting.pleroma.lol/users/lain\" class=\"u-url mention\">@<span>lain</span></a></span></p>" -    end - -    test "it works for incoming notices with to/cc not being an array (kroeg)" do -      data = File.read!("test/fixtures/kroeg-post-activity.json") |> Poison.decode!() - -      {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data) -      object = Object.normalize(data["object"]) - -      assert object.data["content"] == -               "<p>henlo from my Psion netBook</p><p>message sent from my Psion netBook</p>" -    end - -    test "it ensures that as:Public activities make it to their followers collection" do -      user = insert(:user) - -      data = -        File.read!("test/fixtures/mastodon-post-activity.json") -        |> Poison.decode!() -        |> Map.put("actor", user.ap_id) -        |> Map.put("to", ["https://www.w3.org/ns/activitystreams#Public"]) -        |> Map.put("cc", []) - -      object = -        data["object"] -        |> Map.put("attributedTo", user.ap_id) -        |> Map.put("to", ["https://www.w3.org/ns/activitystreams#Public"]) -        |> Map.put("cc", []) -        |> Map.put("id", user.ap_id <> "/activities/12345678") - -      data = Map.put(data, "object", object) - -      {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data) - -      assert data["cc"] == [User.ap_followers(user)] -    end - -    test "it ensures that address fields become lists" do -      user = insert(:user) - -      data = -        File.read!("test/fixtures/mastodon-post-activity.json") -        |> Poison.decode!() -        |> Map.put("actor", user.ap_id) -        |> Map.put("to", nil) -        |> Map.put("cc", nil) - -      object = -        data["object"] -        |> Map.put("attributedTo", user.ap_id) -        |> Map.put("to", nil) -        |> Map.put("cc", nil) -        |> Map.put("id", user.ap_id <> "/activities/12345678") - -      data = Map.put(data, "object", object) - -      {:ok, %Activity{data: data, local: false}} = Transmogrifier.handle_incoming(data) - -      assert !is_nil(data["to"]) -      assert !is_nil(data["cc"]) -    end - -    test "it strips internal likes" do -      data = -        File.read!("test/fixtures/mastodon-post-activity.json") -        |> Poison.decode!() - -      likes = %{ -        "first" => -          "http://mastodon.example.org/objects/dbdbc507-52c8-490d-9b7c-1e1d52e5c132/likes?page=1", -        "id" => "http://mastodon.example.org/objects/dbdbc507-52c8-490d-9b7c-1e1d52e5c132/likes", -        "totalItems" => 3, -        "type" => "OrderedCollection" -      } - -      object = Map.put(data["object"], "likes", likes) -      data = Map.put(data, "object", object) - -      {:ok, %Activity{object: object}} = Transmogrifier.handle_incoming(data) - -      refute Map.has_key?(object.data, "likes") -    end - -    test "it strips internal reactions" do -      user = insert(:user) -      {:ok, activity} = CommonAPI.post(user, %{status: "#cofe"}) -      {:ok, _} = CommonAPI.react_with_emoji(activity.id, user, "📢") - -      %{object: object} = Activity.get_by_id_with_object(activity.id) -      assert Map.has_key?(object.data, "reactions") -      assert Map.has_key?(object.data, "reaction_count") - -      object_data = Transmogrifier.strip_internal_fields(object.data) -      refute Map.has_key?(object_data, "reactions") -      refute Map.has_key?(object_data, "reaction_count") -    end -      test "it works for incoming unfollows with an existing follow" do        user = insert(:user) @@ -387,73 +83,6 @@ defmodule Pleroma.Web.ActivityPub.TransmogrifierTest do        assert activity.data["cc"] == [user.ap_id]      end -    test "it correctly processes messages with non-array to field" do -      user = insert(:user) - -      message = %{ -        "@context" => "https://www.w3.org/ns/activitystreams", -        "to" => "https://www.w3.org/ns/activitystreams#Public", -        "type" => "Create", -        "object" => %{ -          "content" => "blah blah blah", -          "type" => "Note", -          "attributedTo" => user.ap_id, -          "inReplyTo" => nil -        }, -        "actor" => user.ap_id -      } - -      assert {:ok, activity} = Transmogrifier.handle_incoming(message) - -      assert ["https://www.w3.org/ns/activitystreams#Public"] == activity.data["to"] -    end - -    test "it correctly processes messages with non-array cc field" do -      user = insert(:user) - -      message = %{ -        "@context" => "https://www.w3.org/ns/activitystreams", -        "to" => user.follower_address, -        "cc" => "https://www.w3.org/ns/activitystreams#Public", -        "type" => "Create", -        "object" => %{ -          "content" => "blah blah blah", -          "type" => "Note", -          "attributedTo" => user.ap_id, -          "inReplyTo" => nil -        }, -        "actor" => user.ap_id -      } - -      assert {:ok, activity} = Transmogrifier.handle_incoming(message) - -      assert ["https://www.w3.org/ns/activitystreams#Public"] == activity.data["cc"] -      assert [user.follower_address] == activity.data["to"] -    end - -    test "it correctly processes messages with weirdness in address fields" do -      user = insert(:user) - -      message = %{ -        "@context" => "https://www.w3.org/ns/activitystreams", -        "to" => [nil, user.follower_address], -        "cc" => ["https://www.w3.org/ns/activitystreams#Public", ["¿"]], -        "type" => "Create", -        "object" => %{ -          "content" => "…", -          "type" => "Note", -          "attributedTo" => user.ap_id, -          "inReplyTo" => nil -        }, -        "actor" => user.ap_id -      } - -      assert {:ok, activity} = Transmogrifier.handle_incoming(message) - -      assert ["https://www.w3.org/ns/activitystreams#Public"] == activity.data["cc"] -      assert [user.follower_address] == activity.data["to"] -    end -      test "it accepts Move activities" do        old_user = insert(:user)        new_user = insert(:user) @@ -479,95 +108,6 @@ defmodule Pleroma.Web.ActivityPub.TransmogrifierTest do      end    end -  describe "`handle_incoming/2`, Mastodon format `replies` handling" do -    setup do: clear_config([:activitypub, :note_replies_output_limit], 5) -    setup do: clear_config([:instance, :federation_incoming_replies_max_depth]) - -    setup do -      data = -        "test/fixtures/mastodon-post-activity.json" -        |> File.read!() -        |> Poison.decode!() - -      items = get_in(data, ["object", "replies", "first", "items"]) -      assert length(items) > 0 - -      %{data: data, items: items} -    end - -    test "schedules background fetching of `replies` items if max thread depth limit allows", %{ -      data: data, -      items: items -    } do -      Pleroma.Config.put([:instance, :federation_incoming_replies_max_depth], 10) - -      {:ok, _activity} = Transmogrifier.handle_incoming(data) - -      for id <- items do -        job_args = %{"op" => "fetch_remote", "id" => id, "depth" => 1} -        assert_enqueued(worker: Pleroma.Workers.RemoteFetcherWorker, args: job_args) -      end -    end - -    test "does NOT schedule background fetching of `replies` beyond max thread depth limit allows", -         %{data: data} do -      Pleroma.Config.put([:instance, :federation_incoming_replies_max_depth], 0) - -      {:ok, _activity} = Transmogrifier.handle_incoming(data) - -      assert all_enqueued(worker: Pleroma.Workers.RemoteFetcherWorker) == [] -    end -  end - -  describe "`handle_incoming/2`, Pleroma format `replies` handling" do -    setup do: clear_config([:activitypub, :note_replies_output_limit], 5) -    setup do: clear_config([:instance, :federation_incoming_replies_max_depth]) - -    setup do -      user = insert(:user) - -      {:ok, activity} = CommonAPI.post(user, %{status: "post1"}) - -      {:ok, reply1} = -        CommonAPI.post(user, %{status: "reply1", in_reply_to_status_id: activity.id}) - -      {:ok, reply2} = -        CommonAPI.post(user, %{status: "reply2", in_reply_to_status_id: activity.id}) - -      replies_uris = Enum.map([reply1, reply2], fn a -> a.object.data["id"] end) - -      {:ok, federation_output} = Transmogrifier.prepare_outgoing(activity.data) - -      Repo.delete(activity.object) -      Repo.delete(activity) - -      %{federation_output: federation_output, replies_uris: replies_uris} -    end - -    test "schedules background fetching of `replies` items if max thread depth limit allows", %{ -      federation_output: federation_output, -      replies_uris: replies_uris -    } do -      Pleroma.Config.put([:instance, :federation_incoming_replies_max_depth], 1) - -      {:ok, _activity} = Transmogrifier.handle_incoming(federation_output) - -      for id <- replies_uris do -        job_args = %{"op" => "fetch_remote", "id" => id, "depth" => 1} -        assert_enqueued(worker: Pleroma.Workers.RemoteFetcherWorker, args: job_args) -      end -    end - -    test "does NOT schedule background fetching of `replies` beyond max thread depth limit allows", -         %{federation_output: federation_output} do -      Pleroma.Config.put([:instance, :federation_incoming_replies_max_depth], 0) - -      {:ok, _activity} = Transmogrifier.handle_incoming(federation_output) - -      assert all_enqueued(worker: Pleroma.Workers.RemoteFetcherWorker) == [] -    end -  end -    describe "prepare outgoing" do      test "it inlines private announced objects" do        user = insert(:user) @@ -864,60 +404,6 @@ defmodule Pleroma.Web.ActivityPub.TransmogrifierTest do      end    end -  describe "reserialization" do -    test "successfully reserializes a message with inReplyTo == nil" do -      user = insert(:user) - -      message = %{ -        "@context" => "https://www.w3.org/ns/activitystreams", -        "to" => ["https://www.w3.org/ns/activitystreams#Public"], -        "cc" => [], -        "type" => "Create", -        "object" => %{ -          "to" => ["https://www.w3.org/ns/activitystreams#Public"], -          "cc" => [], -          "type" => "Note", -          "content" => "Hi", -          "inReplyTo" => nil, -          "attributedTo" => user.ap_id -        }, -        "actor" => user.ap_id -      } - -      {:ok, activity} = Transmogrifier.handle_incoming(message) - -      {:ok, _} = Transmogrifier.prepare_outgoing(activity.data) -    end - -    test "successfully reserializes a message with AS2 objects in IR" do -      user = insert(:user) - -      message = %{ -        "@context" => "https://www.w3.org/ns/activitystreams", -        "to" => ["https://www.w3.org/ns/activitystreams#Public"], -        "cc" => [], -        "type" => "Create", -        "object" => %{ -          "to" => ["https://www.w3.org/ns/activitystreams#Public"], -          "cc" => [], -          "type" => "Note", -          "content" => "Hi", -          "inReplyTo" => nil, -          "attributedTo" => user.ap_id, -          "tag" => [ -            %{"name" => "#2hu", "href" => "http://example.com/2hu", "type" => "Hashtag"}, -            %{"name" => "Bob", "href" => "http://example.com/bob", "type" => "Mention"} -          ] -        }, -        "actor" => user.ap_id -      } - -      {:ok, activity} = Transmogrifier.handle_incoming(message) - -      {:ok, _} = Transmogrifier.prepare_outgoing(activity.data) -    end -  end -    describe "fix_explicit_addressing" do      setup do        user = insert(:user) @@ -983,64 +469,6 @@ defmodule Pleroma.Web.ActivityPub.TransmogrifierTest do      end    end -  describe "fix_in_reply_to/2" do -    setup do: clear_config([:instance, :federation_incoming_replies_max_depth]) - -    setup do -      data = Poison.decode!(File.read!("test/fixtures/mastodon-post-activity.json")) -      [data: data] -    end - -    test "returns not modified object when hasn't containts inReplyTo field", %{data: data} do -      assert Transmogrifier.fix_in_reply_to(data) == data -    end - -    test "returns object with inReplyTo when denied incoming reply", %{data: data} do -      Pleroma.Config.put([:instance, :federation_incoming_replies_max_depth], 0) - -      object_with_reply = -        Map.put(data["object"], "inReplyTo", "https://shitposter.club/notice/2827873") - -      modified_object = Transmogrifier.fix_in_reply_to(object_with_reply) -      assert modified_object["inReplyTo"] == "https://shitposter.club/notice/2827873" - -      object_with_reply = -        Map.put(data["object"], "inReplyTo", %{"id" => "https://shitposter.club/notice/2827873"}) - -      modified_object = Transmogrifier.fix_in_reply_to(object_with_reply) -      assert modified_object["inReplyTo"] == %{"id" => "https://shitposter.club/notice/2827873"} - -      object_with_reply = -        Map.put(data["object"], "inReplyTo", ["https://shitposter.club/notice/2827873"]) - -      modified_object = Transmogrifier.fix_in_reply_to(object_with_reply) -      assert modified_object["inReplyTo"] == ["https://shitposter.club/notice/2827873"] - -      object_with_reply = Map.put(data["object"], "inReplyTo", []) -      modified_object = Transmogrifier.fix_in_reply_to(object_with_reply) -      assert modified_object["inReplyTo"] == [] -    end - -    @tag capture_log: true -    test "returns modified object when allowed incoming reply", %{data: data} do -      object_with_reply = -        Map.put( -          data["object"], -          "inReplyTo", -          "https://mstdn.io/users/mayuutann/statuses/99568293732299394" -        ) - -      Pleroma.Config.put([:instance, :federation_incoming_replies_max_depth], 5) -      modified_object = Transmogrifier.fix_in_reply_to(object_with_reply) - -      assert modified_object["inReplyTo"] == -               "https://mstdn.io/users/mayuutann/statuses/99568293732299394" - -      assert modified_object["context"] == -               "tag:shitposter.club,2018-02-22:objectType=thread:nonce=e5a7c72d60a9c0e4" -    end -  end -    describe "fix_url/1" do      test "fixes data for object when url is map" do        object = %{ @@ -1076,155 +504,4 @@ defmodule Pleroma.Web.ActivityPub.TransmogrifierTest do                 )      end    end - -  describe "fix_attachments/1" do -    test "returns not modified object" do -      data = Poison.decode!(File.read!("test/fixtures/mastodon-post-activity.json")) -      assert Transmogrifier.fix_attachments(data) == data -    end - -    test "returns modified object when attachment is map" do -      assert Transmogrifier.fix_attachments(%{ -               "attachment" => %{ -                 "mediaType" => "video/mp4", -                 "url" => "https://peertube.moe/stat-480.mp4" -               } -             }) == %{ -               "attachment" => [ -                 %{ -                   "mediaType" => "video/mp4", -                   "type" => "Document", -                   "url" => [ -                     %{ -                       "href" => "https://peertube.moe/stat-480.mp4", -                       "mediaType" => "video/mp4", -                       "type" => "Link" -                     } -                   ] -                 } -               ] -             } -    end - -    test "returns modified object when attachment is list" do -      assert Transmogrifier.fix_attachments(%{ -               "attachment" => [ -                 %{"mediaType" => "video/mp4", "url" => "https://pe.er/stat-480.mp4"}, -                 %{"mimeType" => "video/mp4", "href" => "https://pe.er/stat-480.mp4"} -               ] -             }) == %{ -               "attachment" => [ -                 %{ -                   "mediaType" => "video/mp4", -                   "type" => "Document", -                   "url" => [ -                     %{ -                       "href" => "https://pe.er/stat-480.mp4", -                       "mediaType" => "video/mp4", -                       "type" => "Link" -                     } -                   ] -                 }, -                 %{ -                   "mediaType" => "video/mp4", -                   "type" => "Document", -                   "url" => [ -                     %{ -                       "href" => "https://pe.er/stat-480.mp4", -                       "mediaType" => "video/mp4", -                       "type" => "Link" -                     } -                   ] -                 } -               ] -             } -    end -  end - -  describe "fix_emoji/1" do -    test "returns not modified object when object not contains tags" do -      data = Poison.decode!(File.read!("test/fixtures/mastodon-post-activity.json")) -      assert Transmogrifier.fix_emoji(data) == data -    end - -    test "returns object with emoji when object contains list tags" do -      assert Transmogrifier.fix_emoji(%{ -               "tag" => [ -                 %{"type" => "Emoji", "name" => ":bib:", "icon" => %{"url" => "/test"}}, -                 %{"type" => "Hashtag"} -               ] -             }) == %{ -               "emoji" => %{"bib" => "/test"}, -               "tag" => [ -                 %{"icon" => %{"url" => "/test"}, "name" => ":bib:", "type" => "Emoji"}, -                 %{"type" => "Hashtag"} -               ] -             } -    end - -    test "returns object with emoji when object contains map tag" do -      assert Transmogrifier.fix_emoji(%{ -               "tag" => %{"type" => "Emoji", "name" => ":bib:", "icon" => %{"url" => "/test"}} -             }) == %{ -               "emoji" => %{"bib" => "/test"}, -               "tag" => %{"icon" => %{"url" => "/test"}, "name" => ":bib:", "type" => "Emoji"} -             } -    end -  end - -  describe "set_replies/1" do -    setup do: clear_config([:activitypub, :note_replies_output_limit], 2) - -    test "returns unmodified object if activity doesn't have self-replies" do -      data = Poison.decode!(File.read!("test/fixtures/mastodon-post-activity.json")) -      assert Transmogrifier.set_replies(data) == data -    end - -    test "sets `replies` collection with a limited number of self-replies" do -      [user, another_user] = insert_list(2, :user) - -      {:ok, %{id: id1} = activity} = CommonAPI.post(user, %{status: "1"}) - -      {:ok, %{id: id2} = self_reply1} = -        CommonAPI.post(user, %{status: "self-reply 1", in_reply_to_status_id: id1}) - -      {:ok, self_reply2} = -        CommonAPI.post(user, %{status: "self-reply 2", in_reply_to_status_id: id1}) - -      # Assuming to _not_ be present in `replies` due to :note_replies_output_limit is set to 2 -      {:ok, _} = CommonAPI.post(user, %{status: "self-reply 3", in_reply_to_status_id: id1}) - -      {:ok, _} = -        CommonAPI.post(user, %{ -          status: "self-reply to self-reply", -          in_reply_to_status_id: id2 -        }) - -      {:ok, _} = -        CommonAPI.post(another_user, %{ -          status: "another user's reply", -          in_reply_to_status_id: id1 -        }) - -      object = Object.normalize(activity) -      replies_uris = Enum.map([self_reply1, self_reply2], fn a -> a.object.data["id"] end) - -      assert %{"type" => "Collection", "items" => ^replies_uris} = -               Transmogrifier.set_replies(object.data)["replies"] -    end -  end - -  test "take_emoji_tags/1" do -    user = insert(:user, %{emoji: %{"firefox" => "https://example.org/firefox.png"}}) - -    assert Transmogrifier.take_emoji_tags(user) == [ -             %{ -               "icon" => %{"type" => "Image", "url" => "https://example.org/firefox.png"}, -               "id" => "https://example.org/firefox.png", -               "name" => ":firefox:", -               "type" => "Emoji", -               "updated" => "1970-01-01T00:00:00Z" -             } -           ] -  end  end diff --git a/test/pleroma/web/common_api/utils_test.exs b/test/pleroma/web/common_api/utils_test.exs index e67c10b93..4d6c9ea26 100644 --- a/test/pleroma/web/common_api/utils_test.exs +++ b/test/pleroma/web/common_api/utils_test.exs @@ -6,6 +6,7 @@ defmodule Pleroma.Web.CommonAPI.UtilsTest do    alias Pleroma.Builders.UserBuilder    alias Pleroma.Object    alias Pleroma.Web.CommonAPI +  alias Pleroma.Web.CommonAPI.ActivityDraft    alias Pleroma.Web.CommonAPI.Utils    use Pleroma.DataCase @@ -235,9 +236,9 @@ defmodule Pleroma.Web.CommonAPI.UtilsTest do      test "for public posts, not a reply" do        user = insert(:user)        mentioned_user = insert(:user) -      mentions = [mentioned_user.ap_id] +      draft = %ActivityDraft{user: user, mentions: [mentioned_user.ap_id], visibility: "public"} -      {to, cc} = Utils.get_to_and_cc(user, mentions, nil, "public", nil) +      {to, cc} = Utils.get_to_and_cc(draft)        assert length(to) == 2        assert length(cc) == 1 @@ -252,9 +253,15 @@ defmodule Pleroma.Web.CommonAPI.UtilsTest do        mentioned_user = insert(:user)        third_user = insert(:user)        {:ok, activity} = CommonAPI.post(third_user, %{status: "uguu"}) -      mentions = [mentioned_user.ap_id] -      {to, cc} = Utils.get_to_and_cc(user, mentions, activity, "public", nil) +      draft = %ActivityDraft{ +        user: user, +        mentions: [mentioned_user.ap_id], +        visibility: "public", +        in_reply_to: activity +      } + +      {to, cc} = Utils.get_to_and_cc(draft)        assert length(to) == 3        assert length(cc) == 1 @@ -268,9 +275,9 @@ defmodule Pleroma.Web.CommonAPI.UtilsTest do      test "for unlisted posts, not a reply" do        user = insert(:user)        mentioned_user = insert(:user) -      mentions = [mentioned_user.ap_id] +      draft = %ActivityDraft{user: user, mentions: [mentioned_user.ap_id], visibility: "unlisted"} -      {to, cc} = Utils.get_to_and_cc(user, mentions, nil, "unlisted", nil) +      {to, cc} = Utils.get_to_and_cc(draft)        assert length(to) == 2        assert length(cc) == 1 @@ -285,9 +292,15 @@ defmodule Pleroma.Web.CommonAPI.UtilsTest do        mentioned_user = insert(:user)        third_user = insert(:user)        {:ok, activity} = CommonAPI.post(third_user, %{status: "uguu"}) -      mentions = [mentioned_user.ap_id] -      {to, cc} = Utils.get_to_and_cc(user, mentions, activity, "unlisted", nil) +      draft = %ActivityDraft{ +        user: user, +        mentions: [mentioned_user.ap_id], +        visibility: "unlisted", +        in_reply_to: activity +      } + +      {to, cc} = Utils.get_to_and_cc(draft)        assert length(to) == 3        assert length(cc) == 1 @@ -301,9 +314,9 @@ defmodule Pleroma.Web.CommonAPI.UtilsTest do      test "for private posts, not a reply" do        user = insert(:user)        mentioned_user = insert(:user) -      mentions = [mentioned_user.ap_id] +      draft = %ActivityDraft{user: user, mentions: [mentioned_user.ap_id], visibility: "private"} -      {to, cc} = Utils.get_to_and_cc(user, mentions, nil, "private", nil) +      {to, cc} = Utils.get_to_and_cc(draft)        assert length(to) == 2        assert Enum.empty?(cc) @@ -316,9 +329,15 @@ defmodule Pleroma.Web.CommonAPI.UtilsTest do        mentioned_user = insert(:user)        third_user = insert(:user)        {:ok, activity} = CommonAPI.post(third_user, %{status: "uguu"}) -      mentions = [mentioned_user.ap_id] -      {to, cc} = Utils.get_to_and_cc(user, mentions, activity, "private", nil) +      draft = %ActivityDraft{ +        user: user, +        mentions: [mentioned_user.ap_id], +        visibility: "private", +        in_reply_to: activity +      } + +      {to, cc} = Utils.get_to_and_cc(draft)        assert length(to) == 2        assert Enum.empty?(cc) @@ -330,9 +349,9 @@ defmodule Pleroma.Web.CommonAPI.UtilsTest do      test "for direct posts, not a reply" do        user = insert(:user)        mentioned_user = insert(:user) -      mentions = [mentioned_user.ap_id] +      draft = %ActivityDraft{user: user, mentions: [mentioned_user.ap_id], visibility: "direct"} -      {to, cc} = Utils.get_to_and_cc(user, mentions, nil, "direct", nil) +      {to, cc} = Utils.get_to_and_cc(draft)        assert length(to) == 1        assert Enum.empty?(cc) @@ -345,9 +364,15 @@ defmodule Pleroma.Web.CommonAPI.UtilsTest do        mentioned_user = insert(:user)        third_user = insert(:user)        {:ok, activity} = CommonAPI.post(third_user, %{status: "uguu"}) -      mentions = [mentioned_user.ap_id] -      {to, cc} = Utils.get_to_and_cc(user, mentions, activity, "direct", nil) +      draft = %ActivityDraft{ +        user: user, +        mentions: [mentioned_user.ap_id], +        visibility: "direct", +        in_reply_to: activity +      } + +      {to, cc} = Utils.get_to_and_cc(draft)        assert length(to) == 1        assert Enum.empty?(cc) @@ -356,7 +381,14 @@ defmodule Pleroma.Web.CommonAPI.UtilsTest do        {:ok, direct_activity} = CommonAPI.post(third_user, %{status: "uguu", visibility: "direct"}) -      {to, cc} = Utils.get_to_and_cc(user, mentions, direct_activity, "direct", nil) +      draft = %ActivityDraft{ +        user: user, +        mentions: [mentioned_user.ap_id], +        visibility: "direct", +        in_reply_to: direct_activity +      } + +      {to, cc} = Utils.get_to_and_cc(draft)        assert length(to) == 2        assert Enum.empty?(cc) @@ -532,26 +564,26 @@ defmodule Pleroma.Web.CommonAPI.UtilsTest do      end    end -  describe "make_note_data/11" do +  describe "make_note_data/1" do      test "returns note data" do        user = insert(:user)        note = insert(:note)        user2 = insert(:user)        user3 = insert(:user) -      assert Utils.make_note_data( -               user.ap_id, -               [user2.ap_id], -               "2hu", -               "<h1>This is :moominmamma: note</h1>", -               [], -               note.id, -               [name: "jimm"], -               "test summary", -               [user3.ap_id], -               false, -               %{"custom_tag" => "test"} -             ) == %{ +      draft = %ActivityDraft{ +        user: user, +        to: [user2.ap_id], +        context: "2hu", +        content_html: "<h1>This is :moominmamma: note</h1>", +        in_reply_to: note.id, +        tags: [name: "jimm"], +        summary: "test summary", +        cc: [user3.ap_id], +        extra: %{"custom_tag" => "test"} +      } + +      assert Utils.make_note_data(draft) == %{                 "actor" => user.ap_id,                 "attachment" => [],                 "cc" => [user3.ap_id], diff --git a/test/pleroma/web/common_api_test.exs b/test/pleroma/web/common_api_test.exs index 8e87e69fe..585b2c174 100644 --- a/test/pleroma/web/common_api_test.exs +++ b/test/pleroma/web/common_api_test.exs @@ -1277,4 +1277,128 @@ defmodule Pleroma.Web.CommonAPITest do               } = CommonAPI.get_user("")      end    end + +  describe "with `local` visibility" do +    setup do: clear_config([:instance, :federating], true) + +    test "post" do +      user = insert(:user) + +      with_mock Pleroma.Web.Federator, publish: fn _ -> :ok end do +        {:ok, activity} = CommonAPI.post(user, %{status: "#2hu #2HU", visibility: "local"}) + +        assert Visibility.is_local_public?(activity) +        assert_not_called(Pleroma.Web.Federator.publish(activity)) +      end +    end + +    test "delete" do +      user = insert(:user) + +      {:ok, %Activity{id: activity_id}} = +        CommonAPI.post(user, %{status: "#2hu #2HU", visibility: "local"}) + +      with_mock Pleroma.Web.Federator, publish: fn _ -> :ok end do +        assert {:ok, %Activity{data: %{"deleted_activity_id" => ^activity_id}} = activity} = +                 CommonAPI.delete(activity_id, user) + +        assert Visibility.is_local_public?(activity) +        assert_not_called(Pleroma.Web.Federator.publish(activity)) +      end +    end + +    test "repeat" do +      user = insert(:user) +      other_user = insert(:user) + +      {:ok, %Activity{id: activity_id}} = +        CommonAPI.post(other_user, %{status: "cofe", visibility: "local"}) + +      with_mock Pleroma.Web.Federator, publish: fn _ -> :ok end do +        assert {:ok, %Activity{data: %{"type" => "Announce"}} = activity} = +                 CommonAPI.repeat(activity_id, user) + +        assert Visibility.is_local_public?(activity) +        refute called(Pleroma.Web.Federator.publish(activity)) +      end +    end + +    test "unrepeat" do +      user = insert(:user) +      other_user = insert(:user) + +      {:ok, %Activity{id: activity_id}} = +        CommonAPI.post(other_user, %{status: "cofe", visibility: "local"}) + +      assert {:ok, _} = CommonAPI.repeat(activity_id, user) + +      with_mock Pleroma.Web.Federator, publish: fn _ -> :ok end do +        assert {:ok, %Activity{data: %{"type" => "Undo"}} = activity} = +                 CommonAPI.unrepeat(activity_id, user) + +        assert Visibility.is_local_public?(activity) +        refute called(Pleroma.Web.Federator.publish(activity)) +      end +    end + +    test "favorite" do +      user = insert(:user) +      other_user = insert(:user) + +      {:ok, activity} = CommonAPI.post(other_user, %{status: "cofe", visibility: "local"}) + +      with_mock Pleroma.Web.Federator, publish: fn _ -> :ok end do +        assert {:ok, %Activity{data: %{"type" => "Like"}} = activity} = +                 CommonAPI.favorite(user, activity.id) + +        assert Visibility.is_local_public?(activity) +        refute called(Pleroma.Web.Federator.publish(activity)) +      end +    end + +    test "unfavorite" do +      user = insert(:user) +      other_user = insert(:user) + +      {:ok, activity} = CommonAPI.post(other_user, %{status: "cofe", visibility: "local"}) + +      {:ok, %Activity{}} = CommonAPI.favorite(user, activity.id) + +      with_mock Pleroma.Web.Federator, publish: fn _ -> :ok end do +        assert {:ok, activity} = CommonAPI.unfavorite(activity.id, user) +        assert Visibility.is_local_public?(activity) +        refute called(Pleroma.Web.Federator.publish(activity)) +      end +    end + +    test "react_with_emoji" do +      user = insert(:user) +      other_user = insert(:user) +      {:ok, activity} = CommonAPI.post(other_user, %{status: "cofe", visibility: "local"}) + +      with_mock Pleroma.Web.Federator, publish: fn _ -> :ok end do +        assert {:ok, %Activity{data: %{"type" => "EmojiReact"}} = activity} = +                 CommonAPI.react_with_emoji(activity.id, user, "👍") + +        assert Visibility.is_local_public?(activity) +        refute called(Pleroma.Web.Federator.publish(activity)) +      end +    end + +    test "unreact_with_emoji" do +      user = insert(:user) +      other_user = insert(:user) +      {:ok, activity} = CommonAPI.post(other_user, %{status: "cofe", visibility: "local"}) + +      {:ok, _reaction} = CommonAPI.react_with_emoji(activity.id, user, "👍") + +      with_mock Pleroma.Web.Federator, publish: fn _ -> :ok end do +        assert {:ok, %Activity{data: %{"type" => "Undo"}} = activity} = +                 CommonAPI.unreact_with_emoji(activity.id, user, "👍") + +        assert Visibility.is_local_public?(activity) +        refute called(Pleroma.Web.Federator.publish(activity)) +      end +    end +  end  end diff --git a/test/pleroma/web/mastodon_api/controllers/status_controller_test.exs b/test/pleroma/web/mastodon_api/controllers/status_controller_test.exs index 436608e51..d95200f99 100644 --- a/test/pleroma/web/mastodon_api/controllers/status_controller_test.exs +++ b/test/pleroma/web/mastodon_api/controllers/status_controller_test.exs @@ -1740,4 +1740,23 @@ defmodule Pleroma.Web.MastodonAPI.StatusControllerTest do               |> get("/api/v1/statuses/#{activity.id}")               |> json_response_and_validate_schema(:ok)    end + +  test "posting a local only status" do +    %{user: _user, conn: conn} = oauth_access(["write:statuses"]) + +    conn_one = +      conn +      |> put_req_header("content-type", "application/json") +      |> post("/api/v1/statuses", %{ +        "status" => "cofe", +        "visibility" => "local" +      }) + +    local = Pleroma.Constants.as_local_public() + +    assert %{"content" => "cofe", "id" => id, "visibility" => "local"} = +             json_response(conn_one, 200) + +    assert %Activity{id: ^id, data: %{"to" => [^local]}} = Activity.get_by_id(id) +  end  end | 
