1 # Pleroma: A lightweight social networking server
2 # Copyright © 2017-2020 Pleroma Authors <https://pleroma.social/>
3 # SPDX-License-Identifier: AGPL-3.0-only
5 defmodule Pleroma.Web do
7 A module that keeps using definitions for controllers,
10 This can be used in your application as:
12 use Pleroma.Web, :controller
13 use Pleroma.Web, :view
15 The definitions below will be executed for every view,
16 controller, etc, so keep them short and clean, focused
17 on imports, uses and aliases.
19 Do NOT define functions inside the quoted expressions
23 alias Pleroma.Web.Plugs.EnsureAuthenticatedPlug
24 alias Pleroma.Web.Plugs.EnsurePublicOrAuthenticatedPlug
25 alias Pleroma.Web.Plugs.ExpectAuthenticatedCheckPlug
26 alias Pleroma.Web.Plugs.ExpectPublicOrAuthenticatedCheckPlug
27 alias Pleroma.Web.Plugs.OAuthScopesPlug
28 alias Pleroma.Web.Plugs.PlugHelper
32 use Phoenix.Controller, namespace: Pleroma.Web
36 import Pleroma.Web.Gettext
37 import Pleroma.Web.Router.Helpers
38 import Pleroma.Web.TranslationHelpers
42 defp set_put_layout(conn, _) do
43 put_layout(conn, Pleroma.Config.get(:app_layout, "app.html"))
46 # Marks plugs intentionally skipped and blocks their execution if present in plugs chain
47 defp skip_plug(conn, plug_modules) do
52 fn plug_module, conn ->
54 plug_module.skip_plug(conn)
56 UndefinedFunctionError ->
57 raise "`#{plug_module}` is not skippable. Append `use Pleroma.Web, :plug` to its code."
63 # Executed just before actual controller action, invokes before-action hooks (callbacks)
64 defp action(conn, params) do
65 with %{halted: false} = conn <- maybe_drop_authentication_if_oauth_check_ignored(conn),
66 %{halted: false} = conn <- maybe_perform_public_or_authenticated_check(conn),
67 %{halted: false} = conn <- maybe_perform_authenticated_check(conn),
68 %{halted: false} = conn <- maybe_halt_on_missing_oauth_scopes_check(conn) do
73 # For non-authenticated API actions, drops auth info if OAuth scopes check was ignored
74 # (neither performed nor explicitly skipped)
75 defp maybe_drop_authentication_if_oauth_check_ignored(conn) do
76 if PlugHelper.plug_called?(conn, ExpectPublicOrAuthenticatedCheckPlug) and
77 not PlugHelper.plug_called_or_skipped?(conn, OAuthScopesPlug) do
78 OAuthScopesPlug.drop_auth_info(conn)
84 # Ensures instance is public -or- user is authenticated if such check was scheduled
85 defp maybe_perform_public_or_authenticated_check(conn) do
86 if PlugHelper.plug_called?(conn, ExpectPublicOrAuthenticatedCheckPlug) do
87 EnsurePublicOrAuthenticatedPlug.call(conn, %{})
93 # Ensures user is authenticated if such check was scheduled
94 # Note: runs prior to action even if it was already executed earlier in plug chain
95 # (since OAuthScopesPlug has option of proceeding unauthenticated)
96 defp maybe_perform_authenticated_check(conn) do
97 if PlugHelper.plug_called?(conn, ExpectAuthenticatedCheckPlug) do
98 EnsureAuthenticatedPlug.call(conn, %{})
104 # Halts if authenticated API action neither performs nor explicitly skips OAuth scopes check
105 defp maybe_halt_on_missing_oauth_scopes_check(conn) do
106 if PlugHelper.plug_called?(conn, ExpectAuthenticatedCheckPlug) and
107 not PlugHelper.plug_called_or_skipped?(conn, OAuthScopesPlug) do
111 "Security violation: OAuth scopes check was neither handled nor explicitly skipped."
124 root: "lib/pleroma/web/templates",
125 namespace: Pleroma.Web
127 # Import convenience functions from controllers
128 import Phoenix.Controller, only: [get_csrf_token: 0, get_flash: 2, view_module: 1]
130 import Pleroma.Web.ErrorHelpers
131 import Pleroma.Web.Gettext
132 import Pleroma.Web.Router.Helpers
136 @doc "Same as `render/3` but wrapped in a rescue block"
137 def safe_render(view, template, assigns \\ %{}) do
138 Phoenix.View.render(view, template, assigns)
142 "#{__MODULE__} failed to render #{inspect({view, template})}\n" <>
143 Exception.format(:error, error, __STACKTRACE__)
150 Same as `render_many/4` but wrapped in rescue block.
152 def safe_render_many(collection, view, template, assigns \\ %{}) do
153 Enum.map(collection, fn resource ->
154 as = Map.get(assigns, :as) || view.__resource__
155 assigns = Map.put(assigns, as, resource)
156 safe_render(view, template, assigns)
166 # credo:disable-for-next-line Credo.Check.Consistency.MultiAliasImportRequireUse
168 import Phoenix.Controller
174 # credo:disable-for-next-line Credo.Check.Consistency.MultiAliasImportRequireUse
176 import Pleroma.Web.Gettext
182 @behaviour Pleroma.Web.Plug
186 Marks a plug intentionally skipped and blocks its execution if it's present in plugs chain.
188 def skip_plug(conn) do
189 PlugHelper.append_to_private_list(
191 PlugHelper.skipped_plugs_list_id(),
198 Before-plug hook that
199 * ensures the plug is not skipped
200 * processes `:if_func` / `:unless_func` functional pre-run conditions
201 * adds plug to the list of called plugs and calls `perform/2` if checks are passed
203 Note: multiple invocations of the same plug (with different or same options) are allowed.
205 def call(%Plug.Conn{} = conn, options) do
206 if PlugHelper.plug_skipped?(conn, __MODULE__) ||
207 (options[:if_func] && !options[:if_func].(conn)) ||
208 (options[:unless_func] && options[:unless_func].(conn)) do
212 PlugHelper.append_to_private_list(
214 PlugHelper.called_plugs_list_id(),
218 apply(__MODULE__, :perform, [conn, options])
225 When used, dispatch to the appropriate controller/view/etc.
227 defmacro __using__(which) when is_atom(which) do
228 apply(__MODULE__, which, [])
232 Pleroma.Web.Endpoint.url()