Merge branch 'fix/credo-issues' into 'develop'
[akkoma] / lib / pleroma / web / nodeinfo / nodeinfo_controller.ex
1 # Pleroma: A lightweight social networking server
2 # Copyright © 2017-2019 Pleroma Authors <https://pleroma.social/>
3 # SPDX-License-Identifier: AGPL-3.0-only
4
5 defmodule Pleroma.Web.Nodeinfo.NodeinfoController do
6 use Pleroma.Web, :controller
7
8 alias Pleroma.Config
9 alias Pleroma.Repo
10 alias Pleroma.Stats
11 alias Pleroma.User
12 alias Pleroma.Web
13 alias Pleroma.Web.ActivityPub.MRF
14
15 plug(Pleroma.Web.FederatingPlug)
16
17 def schemas(conn, _params) do
18 response = %{
19 links: [
20 %{
21 rel: "http://nodeinfo.diaspora.software/ns/schema/2.0",
22 href: Web.base_url() <> "/nodeinfo/2.0.json"
23 },
24 %{
25 rel: "http://nodeinfo.diaspora.software/ns/schema/2.1",
26 href: Web.base_url() <> "/nodeinfo/2.1.json"
27 }
28 ]
29 }
30
31 json(conn, response)
32 end
33
34 # returns a nodeinfo 2.0 map, since 2.1 just adds a repository field
35 # under software.
36 def raw_nodeinfo do
37 instance = Application.get_env(:pleroma, :instance)
38 media_proxy = Application.get_env(:pleroma, :media_proxy)
39 suggestions = Application.get_env(:pleroma, :suggestions)
40 chat = Application.get_env(:pleroma, :chat)
41 gopher = Application.get_env(:pleroma, :gopher)
42 stats = Stats.get_stats()
43
44 mrf_simple =
45 Application.get_env(:pleroma, :mrf_simple)
46 |> Enum.into(%{})
47
48 # This horror is needed to convert regex sigils to strings
49 mrf_keyword =
50 Application.get_env(:pleroma, :mrf_keyword, [])
51 |> Enum.map(fn {key, value} ->
52 {key,
53 Enum.map(value, fn
54 {pattern, replacement} ->
55 %{
56 "pattern" =>
57 if not is_binary(pattern) do
58 inspect(pattern)
59 else
60 pattern
61 end,
62 "replacement" => replacement
63 }
64
65 pattern ->
66 if not is_binary(pattern) do
67 inspect(pattern)
68 else
69 pattern
70 end
71 end)}
72 end)
73 |> Enum.into(%{})
74
75 mrf_policies =
76 MRF.get_policies()
77 |> Enum.map(fn policy -> to_string(policy) |> String.split(".") |> List.last() end)
78
79 quarantined = Keyword.get(instance, :quarantined_instances)
80
81 quarantined =
82 if is_list(quarantined) do
83 quarantined
84 else
85 []
86 end
87
88 staff_accounts =
89 User.moderator_user_query()
90 |> Repo.all()
91 |> Enum.map(fn u -> u.ap_id end)
92
93 mrf_user_allowlist =
94 Config.get([:mrf_user_allowlist], [])
95 |> Enum.into(%{}, fn {k, v} -> {k, length(v)} end)
96
97 federation_response =
98 if Keyword.get(instance, :mrf_transparency) do
99 %{
100 mrf_policies: mrf_policies,
101 mrf_simple: mrf_simple,
102 mrf_keyword: mrf_keyword,
103 mrf_user_allowlist: mrf_user_allowlist,
104 quarantined_instances: quarantined
105 }
106 else
107 %{}
108 end
109
110 features =
111 [
112 "pleroma_api",
113 "mastodon_api",
114 "mastodon_api_streaming",
115 if Keyword.get(media_proxy, :enabled) do
116 "media_proxy"
117 end,
118 if Keyword.get(gopher, :enabled) do
119 "gopher"
120 end,
121 if Keyword.get(chat, :enabled) do
122 "chat"
123 end,
124 if Keyword.get(suggestions, :enabled) do
125 "suggestions"
126 end,
127 if Keyword.get(instance, :allow_relay) do
128 "relay"
129 end
130 ]
131 |> Enum.filter(& &1)
132
133 %{
134 version: "2.0",
135 software: %{
136 name: Pleroma.Application.name() |> String.downcase(),
137 version: Pleroma.Application.version()
138 },
139 protocols: ["ostatus", "activitypub"],
140 services: %{
141 inbound: [],
142 outbound: []
143 },
144 openRegistrations: Keyword.get(instance, :registrations_open),
145 usage: %{
146 users: %{
147 total: stats.user_count || 0
148 },
149 localPosts: stats.status_count || 0
150 },
151 metadata: %{
152 nodeName: Keyword.get(instance, :name),
153 nodeDescription: Keyword.get(instance, :description),
154 private: !Keyword.get(instance, :public, true),
155 suggestions: %{
156 enabled: Keyword.get(suggestions, :enabled, false),
157 thirdPartyEngine: Keyword.get(suggestions, :third_party_engine, ""),
158 timeout: Keyword.get(suggestions, :timeout, 5000),
159 limit: Keyword.get(suggestions, :limit, 23),
160 web: Keyword.get(suggestions, :web, "")
161 },
162 staffAccounts: staff_accounts,
163 federation: federation_response,
164 postFormats: Keyword.get(instance, :allowed_post_formats),
165 uploadLimits: %{
166 general: Keyword.get(instance, :upload_limit),
167 avatar: Keyword.get(instance, :avatar_upload_limit),
168 banner: Keyword.get(instance, :banner_upload_limit),
169 background: Keyword.get(instance, :background_upload_limit)
170 },
171 accountActivationRequired: Keyword.get(instance, :account_activation_required, false),
172 invitesEnabled: Keyword.get(instance, :invites_enabled, false),
173 features: features,
174 restrictedNicknames: Pleroma.Config.get([Pleroma.User, :restricted_nicknames])
175 }
176 }
177 end
178
179 # Schema definition: https://github.com/jhass/nodeinfo/blob/master/schemas/2.0/schema.json
180 # and https://github.com/jhass/nodeinfo/blob/master/schemas/2.1/schema.json
181 def nodeinfo(conn, %{"version" => "2.0"}) do
182 conn
183 |> put_resp_header(
184 "content-type",
185 "application/json; profile=http://nodeinfo.diaspora.software/ns/schema/2.0#; charset=utf-8"
186 )
187 |> json(raw_nodeinfo())
188 end
189
190 def nodeinfo(conn, %{"version" => "2.1"}) do
191 raw_response = raw_nodeinfo()
192
193 updated_software =
194 raw_response
195 |> Map.get(:software)
196 |> Map.put(:repository, Pleroma.Application.repository())
197
198 response =
199 raw_response
200 |> Map.put(:software, updated_software)
201 |> Map.put(:version, "2.1")
202
203 conn
204 |> put_resp_header(
205 "content-type",
206 "application/json; profile=http://nodeinfo.diaspora.software/ns/schema/2.1#; charset=utf-8"
207 )
208 |> json(response)
209 end
210
211 def nodeinfo(conn, _) do
212 conn
213 |> put_status(404)
214 |> json(%{error: "Nodeinfo schema version not handled"})
215 end
216 end