diff options
| -rw-r--r-- | docs/Differences-in-MastodonAPI-Responses.md | 4 | ||||
| -rw-r--r-- | lib/pleroma/user.ex | 1 | ||||
| -rw-r--r-- | lib/pleroma/web/activity_pub/activity_pub.ex | 2 | ||||
| -rw-r--r-- | lib/pleroma/web/mastodon_api/views/status_view.ex | 2 | ||||
| -rw-r--r-- | lib/pleroma/web/twitter_api/representers/activity_representer.ex | 247 | ||||
| -rw-r--r-- | lib/pleroma/web/twitter_api/views/activity_view.ex | 4 | ||||
| -rw-r--r-- | test/web/activity_pub/activity_pub_test.exs | 7 | ||||
| -rw-r--r-- | test/web/mastodon_api/status_view_test.exs | 16 | ||||
| -rw-r--r-- | test/web/twitter_api/representers/activity_representer_test.exs | 43 | ||||
| -rw-r--r-- | test/web/twitter_api/views/activity_view_test.exs | 19 | 
10 files changed, 58 insertions, 287 deletions
diff --git a/docs/Differences-in-MastodonAPI-Responses.md b/docs/Differences-in-MastodonAPI-Responses.md index f6a5b6461..3026e1173 100644 --- a/docs/Differences-in-MastodonAPI-Responses.md +++ b/docs/Differences-in-MastodonAPI-Responses.md @@ -9,3 +9,7 @@ Pleroma uses 128-bit ids as opposed to Mastodon's 64 bits. However just like Mas  ## Attachment cap  Some apps operate under the assumption that no more than 4 attachments can be returned or uploaded. Pleroma however does not enforce any limits on attachment count neither when returning the status object nor when posting. + +## Timelines + +Adding the parameter `with_muted=true` to the timeline queries will also return activities by muted (not by blocked!) users. diff --git a/lib/pleroma/user.ex b/lib/pleroma/user.ex index c5085fa82..d58274508 100644 --- a/lib/pleroma/user.ex +++ b/lib/pleroma/user.ex @@ -961,6 +961,7 @@ defmodule Pleroma.User do      update_and_set_cache(cng)    end +  def mutes?(nil, _), do: false    def mutes?(user, %{ap_id: ap_id}), do: Enum.member?(user.info.mutes, ap_id)    def blocks?(user, %{ap_id: ap_id}) do diff --git a/lib/pleroma/web/activity_pub/activity_pub.ex b/lib/pleroma/web/activity_pub/activity_pub.ex index cc255cc9e..52404c7e5 100644 --- a/lib/pleroma/web/activity_pub/activity_pub.ex +++ b/lib/pleroma/web/activity_pub/activity_pub.ex @@ -602,6 +602,8 @@ defmodule Pleroma.Web.ActivityPub.ActivityPub do    defp restrict_reblogs(query, _), do: query +  defp restrict_muted(query, %{"with_muted" => val}) when val in [true, "true", "1"], do: query +    defp restrict_muted(query, %{"muting_user" => %User{info: info}}) do      mutes = info.mutes diff --git a/lib/pleroma/web/mastodon_api/views/status_view.ex b/lib/pleroma/web/mastodon_api/views/status_view.ex index b90e4252a..3468c0e1c 100644 --- a/lib/pleroma/web/mastodon_api/views/status_view.ex +++ b/lib/pleroma/web/mastodon_api/views/status_view.ex @@ -168,7 +168,7 @@ defmodule Pleroma.Web.MastodonAPI.StatusView do        reblogged: present?(repeated),        favourited: present?(favorited),        bookmarked: present?(bookmarked), -      muted: CommonAPI.thread_muted?(user, activity), +      muted: CommonAPI.thread_muted?(user, activity) || User.mutes?(opts[:for], user),        pinned: pinned?(activity, user),        sensitive: sensitive,        spoiler_text: object["summary"] || "", diff --git a/lib/pleroma/web/twitter_api/representers/activity_representer.ex b/lib/pleroma/web/twitter_api/representers/activity_representer.ex index 192ab7334..55c612ddd 100644 --- a/lib/pleroma/web/twitter_api/representers/activity_representer.ex +++ b/lib/pleroma/web/twitter_api/representers/activity_representer.ex @@ -6,247 +6,10 @@  # THIS MODULE IS DEPRECATED! DON'T USE IT!  # USE THE Pleroma.Web.TwitterAPI.Views.ActivityView MODULE!  defmodule Pleroma.Web.TwitterAPI.Representers.ActivityRepresenter do -  use Pleroma.Web.TwitterAPI.Representers.BaseRepresenter -  alias Pleroma.Web.TwitterAPI.Representers.ObjectRepresenter -  alias Pleroma.Activity -  alias Pleroma.Formatter -  alias Pleroma.HTML -  alias Pleroma.User -  alias Pleroma.Web.TwitterAPI.ActivityView -  alias Pleroma.Web.TwitterAPI.TwitterAPI -  alias Pleroma.Web.TwitterAPI.UserView -  alias Pleroma.Web.CommonAPI.Utils -  alias Pleroma.Web.MastodonAPI.StatusView - -  defp user_by_ap_id(user_list, ap_id) do -    Enum.find(user_list, fn %{ap_id: user_id} -> ap_id == user_id end) -  end - -  def to_map( -        %Activity{data: %{"type" => "Announce", "actor" => actor, "published" => created_at}} = -          activity, -        %{users: users, announced_activity: announced_activity} = opts -      ) do -    user = user_by_ap_id(users, actor) -    created_at = created_at |> Utils.date_to_asctime() - -    text = "#{user.nickname} retweeted a status." - -    announced_user = user_by_ap_id(users, announced_activity.data["actor"]) -    retweeted_status = to_map(announced_activity, Map.merge(%{user: announced_user}, opts)) - -    %{ -      "id" => activity.id, -      "user" => UserView.render("show.json", %{user: user, for: opts[:for]}), -      "statusnet_html" => text, -      "text" => text, -      "is_local" => activity.local, -      "is_post_verb" => false, -      "uri" => "tag:#{activity.data["id"]}:objectType=note", -      "created_at" => created_at, -      "retweeted_status" => retweeted_status, -      "statusnet_conversation_id" => conversation_id(announced_activity), -      "external_url" => activity.data["id"], -      "activity_type" => "repeat" -    } -  end - -  def to_map( -        %Activity{data: %{"type" => "Like", "published" => created_at}} = activity, -        %{user: user, liked_activity: liked_activity} = opts -      ) do -    created_at = created_at |> Utils.date_to_asctime() - -    text = "#{user.nickname} favorited a status." - -    %{ -      "id" => activity.id, -      "user" => UserView.render("show.json", %{user: user, for: opts[:for]}), -      "statusnet_html" => text, -      "text" => text, -      "is_local" => activity.local, -      "is_post_verb" => false, -      "uri" => "tag:#{activity.data["id"]}:objectType=Favourite", -      "created_at" => created_at, -      "in_reply_to_status_id" => liked_activity.id, -      "external_url" => activity.data["id"], -      "activity_type" => "like" -    } -  end - -  def to_map( -        %Activity{data: %{"type" => "Follow", "object" => followed_id}} = activity, -        %{user: user} = opts -      ) do -    created_at = activity.data["published"] || DateTime.to_iso8601(activity.inserted_at) -    created_at = created_at |> Utils.date_to_asctime() - -    followed = User.get_cached_by_ap_id(followed_id) -    text = "#{user.nickname} started following #{followed.nickname}" - -    %{ -      "id" => activity.id, -      "user" => UserView.render("show.json", %{user: user, for: opts[:for]}), -      "attentions" => [], -      "statusnet_html" => text, -      "text" => text, -      "is_local" => activity.local, -      "is_post_verb" => false, -      "created_at" => created_at, -      "in_reply_to_status_id" => nil, -      "external_url" => activity.data["id"], -      "activity_type" => "follow" -    } -  end - -  # TODO: -  # Make this more proper. Just a placeholder to not break the frontend. -  def to_map( -        %Activity{ -          data: %{"type" => "Undo", "published" => created_at, "object" => undid_activity} -        } = activity, -        %{user: user} = opts -      ) do -    created_at = created_at |> Utils.date_to_asctime() - -    text = "#{user.nickname} undid the action at #{undid_activity["id"]}" - -    %{ -      "id" => activity.id, -      "user" => UserView.render("show.json", %{user: user, for: opts[:for]}), -      "attentions" => [], -      "statusnet_html" => text, -      "text" => text, -      "is_local" => activity.local, -      "is_post_verb" => false, -      "created_at" => created_at, -      "in_reply_to_status_id" => nil, -      "external_url" => activity.data["id"], -      "activity_type" => "undo" -    } -  end - -  def to_map( -        %Activity{data: %{"type" => "Delete", "published" => created_at, "object" => _}} = -          activity, -        %{user: user} = opts -      ) do -    created_at = created_at |> Utils.date_to_asctime() - -    %{ -      "id" => activity.id, -      "uri" => activity.data["object"], -      "user" => UserView.render("show.json", %{user: user, for: opts[:for]}), -      "attentions" => [], -      "statusnet_html" => "deleted notice {{tag", -      "text" => "deleted notice {{tag", -      "is_local" => activity.local, -      "is_post_verb" => false, -      "created_at" => created_at, -      "in_reply_to_status_id" => nil, -      "external_url" => activity.data["id"], -      "activity_type" => "delete" -    } -  end - -  def to_map( -        %Activity{data: %{"object" => %{"content" => _content} = object}} = activity, -        %{user: user} = opts -      ) do -    created_at = object["published"] |> Utils.date_to_asctime() -    like_count = object["like_count"] || 0 -    announcement_count = object["announcement_count"] || 0 -    favorited = opts[:for] && opts[:for].ap_id in (object["likes"] || []) -    repeated = opts[:for] && opts[:for].ap_id in (object["announcements"] || []) -    pinned = activity.id in user.info.pinned_activities - -    mentions = opts[:mentioned] || [] - -    attentions = -      [] -      |> Utils.maybe_notify_to_recipients(activity) -      |> Utils.maybe_notify_mentioned_recipients(activity) -      |> Enum.map(fn ap_id -> Enum.find(mentions, fn user -> ap_id == user.ap_id end) end) -      |> Enum.filter(& &1) -      |> Enum.map(fn user -> UserView.render("show.json", %{user: user, for: opts[:for]}) end) - -    conversation_id = conversation_id(activity) - -    tags = activity.data["object"]["tag"] || [] -    possibly_sensitive = activity.data["object"]["sensitive"] || Enum.member?(tags, "nsfw") - -    tags = if possibly_sensitive, do: Enum.uniq(["nsfw" | tags]), else: tags - -    {_summary, content} = ActivityView.render_content(object) - -    html = -      HTML.filter_tags(content, User.html_filter_policy(opts[:for])) -      |> Formatter.emojify(object["emoji"]) - -    attachments = object["attachment"] || [] - -    reply_parent = Activity.get_in_reply_to_activity(activity) - -    reply_user = reply_parent && User.get_cached_by_ap_id(reply_parent.actor) - -    summary = HTML.strip_tags(object["summary"]) - -    card = -      StatusView.render( -        "card.json", -        Pleroma.Web.RichMedia.Helpers.fetch_data_for_activity(activity) -      ) - -    %{ -      "id" => activity.id, -      "uri" => activity.data["object"]["id"], -      "user" => UserView.render("show.json", %{user: user, for: opts[:for]}), -      "statusnet_html" => html, -      "text" => HTML.strip_tags(content), -      "is_local" => activity.local, -      "is_post_verb" => true, -      "created_at" => created_at, -      "in_reply_to_status_id" => object["inReplyToStatusId"], -      "in_reply_to_screen_name" => reply_user && reply_user.nickname, -      "in_reply_to_profileurl" => User.profile_url(reply_user), -      "in_reply_to_ostatus_uri" => reply_user && reply_user.ap_id, -      "in_reply_to_user_id" => reply_user && reply_user.id, -      "statusnet_conversation_id" => conversation_id, -      "attachments" => attachments |> ObjectRepresenter.enum_to_list(opts), -      "attentions" => attentions, -      "fave_num" => like_count, -      "repeat_num" => announcement_count, -      "favorited" => to_boolean(favorited), -      "repeated" => to_boolean(repeated), -      "pinned" => pinned, -      "external_url" => object["external_url"] || object["id"], -      "tags" => tags, -      "activity_type" => "post", -      "possibly_sensitive" => possibly_sensitive, -      "visibility" => Pleroma.Web.MastodonAPI.StatusView.get_visibility(object), -      "summary" => summary, -      "summary_html" => summary |> Formatter.emojify(object["emoji"]), -      "card" => card -    } -  end - -  def conversation_id(activity) do -    with context when not is_nil(context) <- activity.data["context"] do -      TwitterAPI.context_to_conversation_id(context) -    else -      _e -> nil -    end -  end - -  defp to_boolean(false) do -    false -  end - -  defp to_boolean(nil) do -    false -  end - -  defp to_boolean(_) do -    true +  def to_map(activity, opts) do +    Pleroma.Web.TwitterAPI.ActivityView.render( +      "activity.json", +      Map.put(opts, :activity, activity) +    )    end  end diff --git a/lib/pleroma/web/twitter_api/views/activity_view.ex b/lib/pleroma/web/twitter_api/views/activity_view.ex index 661022afa..02ca4ee42 100644 --- a/lib/pleroma/web/twitter_api/views/activity_view.ex +++ b/lib/pleroma/web/twitter_api/views/activity_view.ex @@ -10,6 +10,7 @@ defmodule Pleroma.Web.TwitterAPI.ActivityView do    alias Pleroma.Object    alias Pleroma.Repo    alias Pleroma.User +  alias Pleroma.Web.CommonAPI    alias Pleroma.Web.CommonAPI.Utils    alias Pleroma.Web.MastodonAPI.StatusView    alias Pleroma.Web.TwitterAPI.ActivityView @@ -309,7 +310,8 @@ defmodule Pleroma.Web.TwitterAPI.ActivityView do        "visibility" => StatusView.get_visibility(object),        "summary" => summary,        "summary_html" => summary |> Formatter.emojify(object["emoji"]), -      "card" => card +      "card" => card, +      "muted" => CommonAPI.thread_muted?(user, activity) || User.mutes?(opts[:for], user)      }    end diff --git a/test/web/activity_pub/activity_pub_test.exs b/test/web/activity_pub/activity_pub_test.exs index 11262c523..ac3a565de 100644 --- a/test/web/activity_pub/activity_pub_test.exs +++ b/test/web/activity_pub/activity_pub_test.exs @@ -291,6 +291,13 @@ defmodule Pleroma.Web.ActivityPub.ActivityPubTest do      assert Enum.member?(activities, activity_three)      refute Enum.member?(activities, activity_one) +    # Calling with 'with_muted' will deliver muted activities, too. +    activities = ActivityPub.fetch_activities([], %{"muting_user" => user, "with_muted" => true}) + +    assert Enum.member?(activities, activity_two) +    assert Enum.member?(activities, activity_three) +    assert Enum.member?(activities, activity_one) +      {:ok, user} = User.unmute(user, %User{ap_id: activity_one.data["actor"]})      activities = ActivityPub.fetch_activities([], %{"muting_user" => user}) diff --git a/test/web/mastodon_api/status_view_test.exs b/test/web/mastodon_api/status_view_test.exs index 3412a6be2..351dbf673 100644 --- a/test/web/mastodon_api/status_view_test.exs +++ b/test/web/mastodon_api/status_view_test.exs @@ -126,6 +126,22 @@ defmodule Pleroma.Web.MastodonAPI.StatusViewTest do      assert status == expected    end +  test "tells if the message is muted for some reason" do +    user = insert(:user) +    other_user = insert(:user) + +    {:ok, user} = User.mute(user, other_user) + +    {:ok, activity} = CommonAPI.post(other_user, %{"status" => "test"}) +    status = StatusView.render("status.json", %{activity: activity}) + +    assert status.muted == false + +    status = StatusView.render("status.json", %{activity: activity, for: user}) + +    assert status.muted == true +  end +    test "a reply" do      note = insert(:note_activity)      user = insert(:user) diff --git a/test/web/twitter_api/representers/activity_representer_test.exs b/test/web/twitter_api/representers/activity_representer_test.exs index 365c7f659..0e554623c 100644 --- a/test/web/twitter_api/representers/activity_representer_test.exs +++ b/test/web/twitter_api/representers/activity_representer_test.exs @@ -13,36 +13,6 @@ defmodule Pleroma.Web.TwitterAPI.Representers.ActivityRepresenterTest do    alias Pleroma.Web.TwitterAPI.UserView    import Pleroma.Factory -  test "an announce activity" do -    user = insert(:user) -    note_activity = insert(:note_activity) -    activity_actor = Repo.get_by(User, ap_id: note_activity.data["actor"]) -    object = Object.get_by_ap_id(note_activity.data["object"]["id"]) - -    {:ok, announce_activity, _object} = ActivityPub.announce(user, object) -    note_activity = Activity.get_by_ap_id(note_activity.data["id"]) - -    status = -      ActivityRepresenter.to_map(announce_activity, %{ -        users: [user, activity_actor], -        announced_activity: note_activity, -        for: user -      }) - -    assert status["id"] == announce_activity.id -    assert status["user"] == UserView.render("show.json", %{user: user, for: user}) - -    retweeted_status = -      ActivityRepresenter.to_map(note_activity, %{user: activity_actor, for: user}) - -    assert retweeted_status["repeated"] == true -    assert retweeted_status["id"] == note_activity.id -    assert status["statusnet_conversation_id"] == retweeted_status["statusnet_conversation_id"] - -    assert status["retweeted_status"] == retweeted_status -    assert status["activity_type"] == "repeat" -  end -    test "a like activity" do      user = insert(:user)      note_activity = insert(:note_activity) @@ -168,6 +138,7 @@ defmodule Pleroma.Web.TwitterAPI.Representers.ActivityRepresenterTest do        "uri" => activity.data["object"]["id"],        "visibility" => "direct",        "card" => nil, +      "muted" => false,        "summary" => "2hu :2hu:",        "summary_html" =>          "2hu <img height=\"32px\" width=\"32px\" alt=\"2hu\" title=\"2hu\" src=\"corndog.png\" />" @@ -180,18 +151,6 @@ defmodule Pleroma.Web.TwitterAPI.Representers.ActivityRepresenterTest do             }) == expected_status    end -  test "an undo for a follow" do -    follower = insert(:user) -    followed = insert(:user) - -    {:ok, _follow} = ActivityPub.follow(follower, followed) -    {:ok, unfollow} = ActivityPub.unfollow(follower, followed) - -    map = ActivityRepresenter.to_map(unfollow, %{user: follower}) -    assert map["is_post_verb"] == false -    assert map["activity_type"] == "undo" -  end -    test "a delete activity" do      object = insert(:note)      user = User.get_by_ap_id(object.data["actor"]) diff --git a/test/web/twitter_api/views/activity_view_test.exs b/test/web/twitter_api/views/activity_view_test.exs index 4f854ecaa..0a5384f34 100644 --- a/test/web/twitter_api/views/activity_view_test.exs +++ b/test/web/twitter_api/views/activity_view_test.exs @@ -56,6 +56,22 @@ defmodule Pleroma.Web.TwitterAPI.ActivityViewTest do      assert result["user"]["id"] == user.id    end +  test "tells if the message is muted for some reason" do +    user = insert(:user) +    other_user = insert(:user) + +    {:ok, user} = User.mute(user, other_user) + +    {:ok, activity} = CommonAPI.post(other_user, %{"status" => "test"}) +    status = ActivityView.render("activity.json", %{activity: activity}) + +    assert status["muted"] == false + +    status = ActivityView.render("activity.json", %{activity: activity, for: user}) + +    assert status["muted"] == true +  end +    test "a create activity with a html status" do      text = """      #Bike log - Commute Tuesday\nhttps://pla.bike/posts/20181211/\n#cycling #CHScycling #commute\nMVIMG_20181211_054020.jpg @@ -149,7 +165,8 @@ defmodule Pleroma.Web.TwitterAPI.ActivityViewTest do        "uri" => activity.data["object"]["id"],        "user" => UserView.render("show.json", %{user: user}),        "visibility" => "direct", -      "card" => nil +      "card" => nil, +      "muted" => false      }      assert result == expected  | 
