diff options
| author | marcin mikołajczak <me@mkljczk.pl> | 2023-11-12 13:38:08 +0000 | 
|---|---|---|
| committer | lain <lain@soykaf.club> | 2023-11-12 13:38:08 +0000 | 
| commit | 9a063deacc75d3545dcd7a7eb33263ecbf0361ae (patch) | |
| tree | 9c7815f25d27cfbb57bf8f58e25bab421e6c5a8c /lib | |
| parent | 4c5b45ed73e93e6e8bcfeb527b9b398ec64c0caf (diff) | |
| download | pleroma-9a063deacc75d3545dcd7a7eb33263ecbf0361ae.tar.gz pleroma-9a063deacc75d3545dcd7a7eb33263ecbf0361ae.zip | |
Count and display post quotes
Diffstat (limited to 'lib')
| -rw-r--r-- | lib/pleroma/object.ex | 46 | ||||
| -rw-r--r-- | lib/pleroma/web/activity_pub/activity_pub.ex | 21 | ||||
| -rw-r--r-- | lib/pleroma/web/activity_pub/object_validators/common_fields.ex | 1 | ||||
| -rw-r--r-- | lib/pleroma/web/activity_pub/side_effects.ex | 8 | ||||
| -rw-r--r-- | lib/pleroma/web/api_spec/operations/pleroma_status_operation.ex | 45 | ||||
| -rw-r--r-- | lib/pleroma/web/api_spec/schemas/status.ex | 7 | ||||
| -rw-r--r-- | lib/pleroma/web/mastodon_api/views/status_view.ex | 3 | ||||
| -rw-r--r-- | lib/pleroma/web/pleroma_api/controllers/status_controller.ex | 66 | ||||
| -rw-r--r-- | lib/pleroma/web/router.ex | 2 | 
9 files changed, 197 insertions, 2 deletions
| diff --git a/lib/pleroma/object.ex b/lib/pleroma/object.ex index aa137d250..fa5baf1a4 100644 --- a/lib/pleroma/object.ex +++ b/lib/pleroma/object.ex @@ -328,6 +328,52 @@ defmodule Pleroma.Object do      end    end +  def increase_quotes_count(ap_id) do +    Object +    |> where([o], fragment("?->>'id' = ?::text", o.data, ^to_string(ap_id))) +    |> update([o], +      set: [ +        data: +          fragment( +            """ +            safe_jsonb_set(?, '{quotesCount}', +              (coalesce((?->>'quotesCount')::int, 0) + 1)::varchar::jsonb, true) +            """, +            o.data, +            o.data +          ) +      ] +    ) +    |> Repo.update_all([]) +    |> case do +      {1, [object]} -> set_cache(object) +      _ -> {:error, "Not found"} +    end +  end + +  def decrease_quotes_count(ap_id) do +    Object +    |> where([o], fragment("?->>'id' = ?::text", o.data, ^to_string(ap_id))) +    |> update([o], +      set: [ +        data: +          fragment( +            """ +            safe_jsonb_set(?, '{quotesCount}', +              (greatest(0, (?->>'quotesCount')::int - 1))::varchar::jsonb, true) +            """, +            o.data, +            o.data +          ) +      ] +    ) +    |> Repo.update_all([]) +    |> case do +      {1, [object]} -> set_cache(object) +      _ -> {:error, "Not found"} +    end +  end +    def increase_vote_count(ap_id, name, actor) do      with %Object{} = object <- Object.normalize(ap_id, fetch: false),           "Question" <- object.data["type"] do diff --git a/lib/pleroma/web/activity_pub/activity_pub.ex b/lib/pleroma/web/activity_pub/activity_pub.ex index 3979d418e..a81d33fb6 100644 --- a/lib/pleroma/web/activity_pub/activity_pub.ex +++ b/lib/pleroma/web/activity_pub/activity_pub.ex @@ -96,6 +96,17 @@ defmodule Pleroma.Web.ActivityPub.ActivityPub do    defp increase_replies_count_if_reply(_create_data), do: :noop +  defp increase_quotes_count_if_quote(%{ +         "object" => %{"quoteUrl" => quote_ap_id} = object, +         "type" => "Create" +       }) do +    if is_public?(object) do +      Object.increase_quotes_count(quote_ap_id) +    end +  end + +  defp increase_quotes_count_if_quote(_create_data), do: :noop +    @object_types ~w[ChatMessage Question Answer Audio Video Image Event Article Note Page]    @impl true    def persist(%{"type" => type} = object, meta) when type in @object_types do @@ -299,6 +310,7 @@ defmodule Pleroma.Web.ActivityPub.ActivityPub do      with {:ok, activity} <- insert(create_data, local, fake),           {:fake, false, activity} <- {:fake, fake, activity},           _ <- increase_replies_count_if_reply(create_data), +         _ <- increase_quotes_count_if_quote(create_data),           {:quick_insert, false, activity} <- {:quick_insert, quick_insert?, activity},           {:ok, _actor} <- increase_note_count_if_public(actor, activity),           {:ok, _actor} <- update_last_status_at_if_public(actor, activity), @@ -1237,6 +1249,14 @@ defmodule Pleroma.Web.ActivityPub.ActivityPub do    defp restrict_unauthenticated(query, _), do: query +  defp restrict_quote_url(query, %{quote_url: quote_url}) do +    from([_activity, object] in query, +      where: fragment("(?)->'quoteUrl' = ?", object.data, ^quote_url) +    ) +  end + +  defp restrict_quote_url(query, _), do: query +    defp exclude_poll_votes(query, %{include_poll_votes: true}), do: query    defp exclude_poll_votes(query, _) do @@ -1399,6 +1419,7 @@ defmodule Pleroma.Web.ActivityPub.ActivityPub do        |> restrict_instance(opts)        |> restrict_announce_object_actor(opts)        |> restrict_filtered(opts) +      |> restrict_quote_url(opts)        |> maybe_restrict_deactivated_users(opts)        |> exclude_poll_votes(opts)        |> exclude_chat_messages(opts) diff --git a/lib/pleroma/web/activity_pub/object_validators/common_fields.ex b/lib/pleroma/web/activity_pub/object_validators/common_fields.ex index 835ed97b7..1a5d02601 100644 --- a/lib/pleroma/web/activity_pub/object_validators/common_fields.ex +++ b/lib/pleroma/web/activity_pub/object_validators/common_fields.ex @@ -57,6 +57,7 @@ defmodule Pleroma.Web.ActivityPub.ObjectValidators.CommonFields do        field(:replies_count, :integer, default: 0)        field(:like_count, :integer, default: 0)        field(:announcement_count, :integer, default: 0) +      field(:quotes_count, :integer, default: 0)        field(:inReplyTo, ObjectValidators.ObjectID)        field(:quoteUrl, ObjectValidators.ObjectID)        field(:url, ObjectValidators.BareUri) diff --git a/lib/pleroma/web/activity_pub/side_effects.ex b/lib/pleroma/web/activity_pub/side_effects.ex index 098c177c7..6a7ac2806 100644 --- a/lib/pleroma/web/activity_pub/side_effects.ex +++ b/lib/pleroma/web/activity_pub/side_effects.ex @@ -209,6 +209,10 @@ defmodule Pleroma.Web.ActivityPub.SideEffects do          Object.increase_replies_count(in_reply_to)        end +      if quote_url = object.data["quoteUrl"] do +        Object.increase_quotes_count(quote_url) +      end +        reply_depth = (meta[:depth] || 0) + 1        # FIXME: Force inReplyTo to replies @@ -305,6 +309,10 @@ defmodule Pleroma.Web.ActivityPub.SideEffects do                Object.decrease_replies_count(in_reply_to)              end +            if quote_url = deleted_object.data["quoteUrl"] do +              Object.decrease_quotes_count(quote_url) +            end +              MessageReference.delete_for_object(deleted_object)              ap_streamer().stream_out(object) diff --git a/lib/pleroma/web/api_spec/operations/pleroma_status_operation.ex b/lib/pleroma/web/api_spec/operations/pleroma_status_operation.ex new file mode 100644 index 000000000..6e69c5269 --- /dev/null +++ b/lib/pleroma/web/api_spec/operations/pleroma_status_operation.ex @@ -0,0 +1,45 @@ +# Pleroma: A lightweight social networking server +# Copyright © 2017-2022 Pleroma Authors <https://pleroma.social/> +# SPDX-License-Identifier: AGPL-3.0-only + +defmodule Pleroma.Web.ApiSpec.PleromaStatusOperation do +  alias OpenApiSpex.Operation +  alias Pleroma.Web.ApiSpec.Schemas.ApiError +  alias Pleroma.Web.ApiSpec.Schemas.FlakeID +  alias Pleroma.Web.ApiSpec.StatusOperation + +  import Pleroma.Web.ApiSpec.Helpers + +  def open_api_operation(action) do +    operation = String.to_existing_atom("#{action}_operation") +    apply(__MODULE__, operation, []) +  end + +  def quotes_operation do +    %Operation{ +      tags: ["Retrieve status information"], +      summary: "Quoted by", +      description: "View quotes for a given status", +      operationId: "PleromaAPI.StatusController.quotes", +      parameters: [id_param() | pagination_params()], +      security: [%{"oAuth" => ["read:statuses"]}], +      responses: %{ +        200 => +          Operation.response( +            "Array of Status", +            "application/json", +            StatusOperation.array_of_statuses() +          ), +        403 => Operation.response("Forbidden", "application/json", ApiError), +        404 => Operation.response("Not Found", "application/json", ApiError) +      } +    } +  end + +  def id_param do +    Operation.parameter(:id, :path, FlakeID, "Status ID", +      example: "9umDrYheeY451cQnEe", +      required: true +    ) +  end +end diff --git a/lib/pleroma/web/api_spec/schemas/status.ex b/lib/pleroma/web/api_spec/schemas/status.ex index 07f03134a..a4052803b 100644 --- a/lib/pleroma/web/api_spec/schemas/status.ex +++ b/lib/pleroma/web/api_spec/schemas/status.ex @@ -213,6 +213,10 @@ defmodule Pleroma.Web.ApiSpec.Schemas.Status do              type: :boolean,              description: "`true` if the quoted post is visible to the user"            }, +          quotes_count: %Schema{ +            type: :integer, +            description: "How many statuses quoted this status" +          },            local: %Schema{              type: :boolean,              description: "`true` if the post was made on the local instance" @@ -367,7 +371,8 @@ defmodule Pleroma.Web.ApiSpec.Schemas.Status do          "in_reply_to_account_acct" => nil,          "local" => true,          "spoiler_text" => %{"text/plain" => ""}, -        "thread_muted" => false +        "thread_muted" => false, +        "quotes_count" => 0        },        "poll" => nil,        "reblog" => nil, diff --git a/lib/pleroma/web/mastodon_api/views/status_view.ex b/lib/pleroma/web/mastodon_api/views/status_view.ex index d070262cc..e3b5760fa 100644 --- a/lib/pleroma/web/mastodon_api/views/status_view.ex +++ b/lib/pleroma/web/mastodon_api/views/status_view.ex @@ -447,7 +447,8 @@ defmodule Pleroma.Web.MastodonAPI.StatusView do          thread_muted: thread_muted?,          emoji_reactions: emoji_reactions,          parent_visible: visible_for_user?(reply_to, opts[:for]), -        pinned_at: pinned_at +        pinned_at: pinned_at, +        quotes_count: object.data["quotesCount"] || 0        }      }    end diff --git a/lib/pleroma/web/pleroma_api/controllers/status_controller.ex b/lib/pleroma/web/pleroma_api/controllers/status_controller.ex new file mode 100644 index 000000000..482662fdd --- /dev/null +++ b/lib/pleroma/web/pleroma_api/controllers/status_controller.ex @@ -0,0 +1,66 @@ +# Pleroma: A lightweight social networking server +# Copyright © 2017-2022 Pleroma Authors <https://pleroma.social/> +# SPDX-License-Identifier: AGPL-3.0-only + +defmodule Pleroma.Web.PleromaAPI.StatusController do +  use Pleroma.Web, :controller + +  import Pleroma.Web.ControllerHelper, only: [add_link_headers: 2] + +  require Ecto.Query +  require Pleroma.Constants + +  alias Pleroma.Activity +  alias Pleroma.User +  alias Pleroma.Web.ActivityPub.ActivityPub +  alias Pleroma.Web.ActivityPub.Visibility +  alias Pleroma.Web.MastodonAPI.StatusView +  alias Pleroma.Web.Plugs.OAuthScopesPlug + +  plug(Pleroma.Web.ApiSpec.CastAndValidate) + +  action_fallback(Pleroma.Web.MastodonAPI.FallbackController) + +  plug( +    OAuthScopesPlug, +    %{scopes: ["read:statuses"], fallback: :proceed_unauthenticated} when action == :quotes +  ) + +  defdelegate open_api_operation(action), to: Pleroma.Web.ApiSpec.PleromaStatusOperation + +  @doc "GET /api/v1/pleroma/statuses/:id/quotes" +  def quotes(%{assigns: %{user: user}} = conn, %{id: id} = params) do +    with %Activity{object: object} = activity <- Activity.get_by_id_with_object(id), +         true <- Visibility.visible_for_user?(activity, user) do +      params = +        params +        |> Map.put(:type, "Create") +        |> Map.put(:blocking_user, user) +        |> Map.put(:quote_url, object.data["id"]) + +      recipients = +        if user do +          [Pleroma.Constants.as_public()] ++ [user.ap_id | User.following(user)] +        else +          [Pleroma.Constants.as_public()] +        end + +      activities = +        recipients +        |> ActivityPub.fetch_activities(params) +        |> Enum.reverse() + +      conn +      |> add_link_headers(activities) +      |> put_view(StatusView) +      |> render("index.json", +        activities: activities, +        for: user, +        as: :activity +      ) +    else +      nil -> {:error, :not_found} +      false -> {:error, :not_found} +    end +  end +end diff --git a/lib/pleroma/web/router.ex b/lib/pleroma/web/router.ex index 6b9e158a3..d40af499e 100644 --- a/lib/pleroma/web/router.ex +++ b/lib/pleroma/web/router.ex @@ -578,6 +578,8 @@ defmodule Pleroma.Web.Router do        pipe_through(:api)        get("/accounts/:id/favourites", AccountController, :favourites)        get("/accounts/:id/endorsements", AccountController, :endorsements) + +      get("/statuses/:id/quotes", StatusController, :quotes)      end      scope [] do | 
