summaryrefslogtreecommitdiff
path: root/lib/pleroma/web/pleroma_api/controllers/settings_controller.ex
blob: 1136575b6265888a30c1265bca60e817f8dcb9a3 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
# 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.SettingsController do
  use Pleroma.Web, :controller

  alias Pleroma.Web.Plugs.OAuthScopesPlug

  plug(Pleroma.Web.ApiSpec.CastAndValidate)

  plug(
    OAuthScopesPlug,
    %{scopes: ["write:accounts"]} when action in [:update]
  )

  plug(
    OAuthScopesPlug,
    %{scopes: ["read:accounts"]} when action in [:show]
  )

  defdelegate open_api_operation(action), to: Pleroma.Web.ApiSpec.PleromaSettingsOperation

  @doc "GET /api/v1/pleroma/settings/:app"
  def show(%{assigns: %{user: user}} = conn, %{app: app} = _params) do
    conn
    |> json(get_settings(user, app))
  end

  @doc "PATCH /api/v1/pleroma/settings/:app"
  def update(%{assigns: %{user: user}, body_params: body_params} = conn, %{app: app} = _params) do
    settings =
      get_settings(user, app)
      |> merge_recursively(body_params)

    with changeset <-
           Pleroma.User.update_changeset(
             user,
             %{pleroma_settings_store: %{app => settings}}
           ),
         {:ok, _} <- Pleroma.Repo.update(changeset) do
      conn
      |> json(settings)
    end
  end

  defp merge_recursively(old, %{} = new) do
    old = ensure_object(old)

    Enum.reduce(
      new,
      old,
      fn
        {k, nil}, acc ->
          Map.drop(acc, [k])

        {k, %{} = new_child}, acc ->
          Map.put(acc, k, merge_recursively(acc[k], new_child))

        {k, v}, acc ->
          Map.put(acc, k, v)
      end
    )
  end

  defp get_settings(user, app) do
    user.pleroma_settings_store
    |> Map.get(app, %{})
    |> ensure_object()
  end

  defp ensure_object(%{} = object) do
    object
  end

  defp ensure_object(_) do
    %{}
  end
end