Add a test for custom runtime modules
[akkoma] / lib / pleroma / application.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.Application do
6 import Cachex.Spec
7 use Application
8
9 @name Mix.Project.config()[:name]
10 @version Mix.Project.config()[:version]
11 @repository Mix.Project.config()[:source_url]
12 @env Mix.env()
13
14 def name, do: @name
15 def version, do: @version
16 def named_version, do: @name <> " " <> @version
17 def repository, do: @repository
18
19 def user_agent do
20 case Pleroma.Config.get([:http, :user_agent], :default) do
21 :default ->
22 info = "#{Pleroma.Web.base_url()} <#{Pleroma.Config.get([:instance, :email], "")}>"
23 named_version() <> "; " <> info
24
25 custom ->
26 custom
27 end
28 end
29
30 # See http://elixir-lang.org/docs/stable/elixir/Application.html
31 # for more information on OTP Applications
32 def start(_type, _args) do
33 Pleroma.Config.DeprecationWarnings.warn()
34 setup_instrumenters()
35 load_custom_modules()
36
37 # Define workers and child supervisors to be supervised
38 children =
39 [
40 Pleroma.Repo,
41 Pleroma.Scheduler,
42 Pleroma.Config.TransferTask,
43 Pleroma.Emoji,
44 Pleroma.Captcha,
45 Pleroma.Daemons.ScheduledActivityDaemon,
46 Pleroma.Daemons.ActivityExpirationDaemon,
47 Pleroma.Plugs.RateLimiter.Supervisor
48 ] ++
49 cachex_children() ++
50 hackney_pool_children() ++
51 [
52 Pleroma.Stats,
53 Pleroma.JobQueueMonitor,
54 {Oban, Pleroma.Config.get(Oban)}
55 ] ++
56 task_children(@env) ++
57 oauth_cleanup_child(oauth_cleanup_enabled?()) ++
58 streamer_child(@env) ++
59 chat_child(@env, chat_enabled?()) ++
60 [
61 Pleroma.Web.Endpoint,
62 Pleroma.Gopher.Server
63 ]
64
65 # See http://elixir-lang.org/docs/stable/elixir/Supervisor.html
66 # for other strategies and supported options
67 opts = [strategy: :one_for_one, name: Pleroma.Supervisor]
68 Supervisor.start_link(children, opts)
69 end
70
71 def load_custom_modules do
72 dir = Pleroma.Config.get([:modules, :runtime_dir])
73
74 if dir && File.exists?(dir) do
75 dir
76 |> File.ls!()
77 |> Enum.map(&Path.join(dir, &1))
78 |> Kernel.ParallelCompiler.compile()
79 |> case do
80 {:error, _errors, _warnings} ->
81 raise "Invalid custom modules"
82
83 {:ok, modules, _warnings} ->
84 if @env != :test do
85 Enum.each(modules, fn mod ->
86 IO.puts("Custom module loaded: #{inspect(mod)}")
87 end)
88 end
89
90 :ok
91 end
92 end
93 end
94
95 defp setup_instrumenters do
96 require Prometheus.Registry
97
98 if Application.get_env(:prometheus, Pleroma.Repo.Instrumenter) do
99 :ok =
100 :telemetry.attach(
101 "prometheus-ecto",
102 [:pleroma, :repo, :query],
103 &Pleroma.Repo.Instrumenter.handle_event/4,
104 %{}
105 )
106
107 Pleroma.Repo.Instrumenter.setup()
108 end
109
110 Pleroma.Web.Endpoint.MetricsExporter.setup()
111 Pleroma.Web.Endpoint.PipelineInstrumenter.setup()
112 Pleroma.Web.Endpoint.Instrumenter.setup()
113 end
114
115 def enabled_hackney_pools do
116 [:media] ++
117 if Application.get_env(:tesla, :adapter) == Tesla.Adapter.Hackney do
118 [:federation]
119 else
120 []
121 end ++
122 if Pleroma.Config.get([Pleroma.Upload, :proxy_remote]) do
123 [:upload]
124 else
125 []
126 end
127 end
128
129 defp cachex_children do
130 [
131 build_cachex("used_captcha", ttl_interval: seconds_valid_interval()),
132 build_cachex("user", default_ttl: 25_000, ttl_interval: 1000, limit: 2500),
133 build_cachex("object", default_ttl: 25_000, ttl_interval: 1000, limit: 2500),
134 build_cachex("rich_media", default_ttl: :timer.minutes(120), limit: 5000),
135 build_cachex("scrubber", limit: 2500),
136 build_cachex("idempotency", expiration: idempotency_expiration(), limit: 2500),
137 build_cachex("web_resp", limit: 2500),
138 build_cachex("emoji_packs", expiration: emoji_packs_expiration(), limit: 10),
139 build_cachex("failed_proxy_url", limit: 2500)
140 ]
141 end
142
143 defp emoji_packs_expiration,
144 do: expiration(default: :timer.seconds(5 * 60), interval: :timer.seconds(60))
145
146 defp idempotency_expiration,
147 do: expiration(default: :timer.seconds(6 * 60 * 60), interval: :timer.seconds(60))
148
149 defp seconds_valid_interval,
150 do: :timer.seconds(Pleroma.Config.get!([Pleroma.Captcha, :seconds_valid]))
151
152 defp build_cachex(type, opts),
153 do: %{
154 id: String.to_atom("cachex_" <> type),
155 start: {Cachex, :start_link, [String.to_atom(type <> "_cache"), opts]},
156 type: :worker
157 }
158
159 defp chat_enabled?, do: Pleroma.Config.get([:chat, :enabled])
160
161 defp oauth_cleanup_enabled?,
162 do: Pleroma.Config.get([:oauth2, :clean_expired_tokens], false)
163
164 defp streamer_child(:test), do: []
165
166 defp streamer_child(_) do
167 [Pleroma.Web.Streamer.supervisor()]
168 end
169
170 defp oauth_cleanup_child(true),
171 do: [Pleroma.Web.OAuth.Token.CleanWorker]
172
173 defp oauth_cleanup_child(_), do: []
174
175 defp chat_child(:test, _), do: []
176
177 defp chat_child(_env, true) do
178 [Pleroma.Web.ChatChannel.ChatChannelState]
179 end
180
181 defp chat_child(_, _), do: []
182
183 defp hackney_pool_children do
184 for pool <- enabled_hackney_pools() do
185 options = Pleroma.Config.get([:hackney_pools, pool])
186 :hackney_pool.child_spec(pool, options)
187 end
188 end
189
190 defp task_children(:test) do
191 [
192 %{
193 id: :web_push_init,
194 start: {Task, :start_link, [&Pleroma.Web.Push.init/0]},
195 restart: :temporary
196 }
197 ]
198 end
199
200 defp task_children(_) do
201 [
202 %{
203 id: :web_push_init,
204 start: {Task, :start_link, [&Pleroma.Web.Push.init/0]},
205 restart: :temporary
206 },
207 %{
208 id: :internal_fetch_init,
209 start: {Task, :start_link, [&Pleroma.Web.ActivityPub.InternalFetchActor.init/0]},
210 restart: :temporary
211 }
212 ]
213 end
214 end