1 # Pleroma: A lightweight social networking server
2 # Copyright © 2017-2019 Pleroma Authors <https://pleroma.social/>
3 # SPDX-License-Identifier: AGPL-3.0-only
5 defmodule Pleroma.Web.PleromaAPI.PleromaAPIController do
6 use Pleroma.Web, :controller
8 import Pleroma.Web.ControllerHelper, only: [add_link_headers: 2]
10 alias Pleroma.Activity
11 alias Pleroma.Conversation.Participation
12 alias Pleroma.Notification
14 alias Pleroma.Plugs.OAuthScopesPlug
16 alias Pleroma.Web.ActivityPub.ActivityPub
17 alias Pleroma.Web.CommonAPI
18 alias Pleroma.Web.MastodonAPI.AccountView
19 alias Pleroma.Web.MastodonAPI.ConversationView
20 alias Pleroma.Web.MastodonAPI.NotificationView
21 alias Pleroma.Web.MastodonAPI.StatusView
25 %{scopes: ["read:statuses"]}
26 when action in [:conversation, :conversation_statuses]
31 %{scopes: ["write:statuses"]}
32 when action in [:react_with_emoji, :unreact_with_emoji]
37 %{scopes: ["write:conversations"]} when action == :update_conversation
40 plug(OAuthScopesPlug, %{scopes: ["write:notifications"]} when action == :read_notification)
42 plug(Pleroma.Plugs.EnsurePublicOrAuthenticatedPlug)
44 def emoji_reactions_by(%{assigns: %{user: user}} = conn, %{"id" => activity_id}) do
45 with %Activity{} = activity <- Activity.get_by_id_with_object(activity_id),
46 %Object{data: %{"reactions" => emoji_reactions}} when is_list(emoji_reactions) <-
47 Object.normalize(activity) do
50 |> Enum.map(fn [emoji, users] ->
51 users = Enum.map(users, &User.get_cached_by_ap_id/1)
56 accounts: AccountView.render("index.json", %{users: users, for: user, as: :user})
69 def react_with_emoji(%{assigns: %{user: user}} = conn, %{"id" => activity_id, "emoji" => emoji}) do
70 with {:ok, _activity, _object} <- CommonAPI.react_with_emoji(activity_id, user, emoji),
71 activity <- Activity.get_by_id(activity_id) do
73 |> put_view(StatusView)
74 |> render("show.json", %{activity: activity, for: user, as: :activity})
78 def unreact_with_emoji(%{assigns: %{user: user}} = conn, %{
82 with {:ok, _activity, _object} <- CommonAPI.unreact_with_emoji(activity_id, user, emoji),
83 activity <- Activity.get_by_id(activity_id) do
85 |> put_view(StatusView)
86 |> render("show.json", %{activity: activity, for: user, as: :activity})
90 def conversation(%{assigns: %{user: user}} = conn, %{"id" => participation_id}) do
91 with %Participation{} = participation <- Participation.get(participation_id),
92 true <- user.id == participation.user_id do
94 |> put_view(ConversationView)
95 |> render("participation.json", %{participation: participation, for: user})
99 def conversation_statuses(
100 %{assigns: %{user: user}} = conn,
101 %{"id" => participation_id} = params
103 participation = Participation.get(participation_id, preload: [:conversation])
105 if user.id == participation.user_id do
108 |> Map.put("blocking_user", user)
109 |> Map.put("muting_user", user)
110 |> Map.put("user", user)
113 participation.conversation.ap_id
114 |> ActivityPub.fetch_activities_for_context(params)
118 |> add_link_headers(activities)
119 |> put_view(StatusView)
120 |> render("index.json", %{activities: activities, for: user, as: :activity})
124 def update_conversation(
125 %{assigns: %{user: user}} = conn,
126 %{"id" => participation_id, "recipients" => recipients}
130 |> Participation.get()
132 with true <- user.id == participation.user_id,
133 {:ok, participation} <- Participation.set_recipients(participation, recipients) do
135 |> put_view(ConversationView)
136 |> render("participation.json", %{participation: participation, for: user})
140 def read_conversations(%{assigns: %{user: user}} = conn, _params) do
141 with {:ok, _, participations} <- Participation.mark_all_as_read(user) do
143 |> add_link_headers(participations)
144 |> put_view(ConversationView)
145 |> render("participations.json", participations: participations, for: user)
149 def read_notification(%{assigns: %{user: user}} = conn, %{"id" => notification_id}) do
150 with {:ok, notification} <- Notification.read_one(user, notification_id) do
152 |> put_view(NotificationView)
153 |> render("show.json", %{notification: notification, for: user})
157 |> put_status(:bad_request)
158 |> json(%{"error" => message})
162 def read_notification(%{assigns: %{user: user}} = conn, %{"max_id" => max_id}) do
163 with notifications <- Notification.set_read_up_to(user, max_id) do
164 notifications = Enum.take(notifications, 80)
167 |> put_view(NotificationView)
168 |> render("index.json", %{notifications: notifications, for: user})