logo

pleroma

My custom branche(s) on git.pleroma.social/pleroma/pleroma

app.ex (3957B)


  1. # Pleroma: A lightweight social networking server
  2. # Copyright © 2017-2020 Pleroma Authors <https://pleroma.social/>
  3. # SPDX-License-Identifier: AGPL-3.0-only
  4. defmodule Pleroma.Web.OAuth.App do
  5. use Ecto.Schema
  6. import Ecto.Changeset
  7. import Ecto.Query
  8. alias Pleroma.Repo
  9. @type t :: %__MODULE__{}
  10. schema "apps" do
  11. field(:client_name, :string)
  12. field(:redirect_uris, :string)
  13. field(:scopes, {:array, :string}, default: [])
  14. field(:website, :string)
  15. field(:client_id, :string)
  16. field(:client_secret, :string)
  17. field(:trusted, :boolean, default: false)
  18. has_many(:oauth_authorizations, Pleroma.Web.OAuth.Authorization, on_delete: :delete_all)
  19. has_many(:oauth_tokens, Pleroma.Web.OAuth.Token, on_delete: :delete_all)
  20. timestamps()
  21. end
  22. @spec changeset(t(), map()) :: Ecto.Changeset.t()
  23. def changeset(struct, params) do
  24. cast(struct, params, [:client_name, :redirect_uris, :scopes, :website, :trusted])
  25. end
  26. @spec register_changeset(t(), map()) :: Ecto.Changeset.t()
  27. def register_changeset(struct, params \\ %{}) do
  28. changeset =
  29. struct
  30. |> changeset(params)
  31. |> validate_required([:client_name, :redirect_uris, :scopes])
  32. if changeset.valid? do
  33. changeset
  34. |> put_change(
  35. :client_id,
  36. :crypto.strong_rand_bytes(32) |> Base.url_encode64(padding: false)
  37. )
  38. |> put_change(
  39. :client_secret,
  40. :crypto.strong_rand_bytes(32) |> Base.url_encode64(padding: false)
  41. )
  42. else
  43. changeset
  44. end
  45. end
  46. @spec create(map()) :: {:ok, t()} | {:error, Ecto.Changeset.t()}
  47. def create(params) do
  48. %__MODULE__{}
  49. |> register_changeset(params)
  50. |> Repo.insert()
  51. end
  52. @spec update(pos_integer(), map()) :: {:ok, t()} | {:error, Ecto.Changeset.t()}
  53. def update(id, params) do
  54. with %__MODULE__{} = app <- Repo.get(__MODULE__, id) do
  55. app
  56. |> changeset(params)
  57. |> Repo.update()
  58. end
  59. end
  60. @doc """
  61. Gets app by attrs or create new with attrs.
  62. And updates the scopes if need.
  63. """
  64. @spec get_or_make(map(), list(String.t())) :: {:ok, t()} | {:error, Ecto.Changeset.t()}
  65. def get_or_make(attrs, scopes) do
  66. with %__MODULE__{} = app <- Repo.get_by(__MODULE__, attrs) do
  67. update_scopes(app, scopes)
  68. else
  69. _e ->
  70. %__MODULE__{}
  71. |> register_changeset(Map.put(attrs, :scopes, scopes))
  72. |> Repo.insert()
  73. end
  74. end
  75. defp update_scopes(%__MODULE__{} = app, []), do: {:ok, app}
  76. defp update_scopes(%__MODULE__{scopes: scopes} = app, scopes), do: {:ok, app}
  77. defp update_scopes(%__MODULE__{} = app, scopes) do
  78. app
  79. |> change(%{scopes: scopes})
  80. |> Repo.update()
  81. end
  82. @spec search(map()) :: {:ok, [t()], non_neg_integer()}
  83. def search(params) do
  84. query = from(a in __MODULE__)
  85. query =
  86. if params[:client_name] do
  87. from(a in query, where: a.client_name == ^params[:client_name])
  88. else
  89. query
  90. end
  91. query =
  92. if params[:client_id] do
  93. from(a in query, where: a.client_id == ^params[:client_id])
  94. else
  95. query
  96. end
  97. query =
  98. if Map.has_key?(params, :trusted) do
  99. from(a in query, where: a.trusted == ^params[:trusted])
  100. else
  101. query
  102. end
  103. query =
  104. from(u in query,
  105. limit: ^params[:page_size],
  106. offset: ^((params[:page] - 1) * params[:page_size])
  107. )
  108. count = Repo.aggregate(__MODULE__, :count, :id)
  109. {:ok, Repo.all(query), count}
  110. end
  111. @spec destroy(pos_integer()) :: {:ok, t()} | {:error, Ecto.Changeset.t()}
  112. def destroy(id) do
  113. with %__MODULE__{} = app <- Repo.get(__MODULE__, id) do
  114. Repo.delete(app)
  115. end
  116. end
  117. @spec errors(Ecto.Changeset.t()) :: map()
  118. def errors(changeset) do
  119. Enum.reduce(changeset.errors, %{}, fn
  120. {:client_name, {error, _}}, acc ->
  121. Map.put(acc, :name, error)
  122. {key, {error, _}}, acc ->
  123. Map.put(acc, key, error)
  124. end)
  125. end
  126. end