159 lines
4.1 KiB
Elixir
159 lines
4.1 KiB
Elixir
# Pleroma: A lightweight social networking server
|
|
# Copyright © 2017-2022 Pleroma Authors <https://pleroma.social/>
|
|
# SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
defmodule Pleroma.Web.OAuth.App do
|
|
use Ecto.Schema
|
|
import Ecto.Changeset
|
|
import Ecto.Query
|
|
alias Pleroma.Repo
|
|
alias Pleroma.User
|
|
|
|
@type t :: %__MODULE__{}
|
|
|
|
schema "apps" do
|
|
field(:client_name, :string)
|
|
field(:redirect_uris, :string)
|
|
field(:scopes, {:array, :string}, default: [])
|
|
field(:website, :string)
|
|
field(:client_id, :string)
|
|
field(:client_secret, :string)
|
|
field(:trusted, :boolean, default: false)
|
|
|
|
belongs_to(:user, User, type: FlakeId.Ecto.CompatType)
|
|
|
|
has_many(:oauth_authorizations, Pleroma.Web.OAuth.Authorization, on_delete: :delete_all)
|
|
has_many(:oauth_tokens, Pleroma.Web.OAuth.Token, on_delete: :delete_all)
|
|
|
|
timestamps()
|
|
end
|
|
|
|
@spec changeset(t(), map()) :: Ecto.Changeset.t()
|
|
def changeset(struct, params) do
|
|
cast(struct, params, [:client_name, :redirect_uris, :scopes, :website, :trusted, :user_id])
|
|
end
|
|
|
|
@spec register_changeset(t(), map()) :: Ecto.Changeset.t()
|
|
def register_changeset(struct, params \\ %{}) do
|
|
changeset =
|
|
struct
|
|
|> changeset(params)
|
|
|> validate_required([:client_name, :redirect_uris, :scopes])
|
|
|
|
if changeset.valid? do
|
|
changeset
|
|
|> put_change(
|
|
:client_id,
|
|
:crypto.strong_rand_bytes(32) |> Base.url_encode64(padding: false)
|
|
)
|
|
|> put_change(
|
|
:client_secret,
|
|
:crypto.strong_rand_bytes(32) |> Base.url_encode64(padding: false)
|
|
)
|
|
else
|
|
changeset
|
|
end
|
|
end
|
|
|
|
@spec create(map()) :: {:ok, t()} | {:error, Ecto.Changeset.t()}
|
|
def create(params) do
|
|
%__MODULE__{}
|
|
|> register_changeset(params)
|
|
|> Repo.insert()
|
|
end
|
|
|
|
@spec update(pos_integer(), map()) :: {:ok, t()} | {:error, Ecto.Changeset.t()}
|
|
def update(id, params) do
|
|
with %__MODULE__{} = app <- Repo.get(__MODULE__, id) do
|
|
app
|
|
|> changeset(params)
|
|
|> Repo.update()
|
|
end
|
|
end
|
|
|
|
@doc """
|
|
Gets app by attrs or create new with attrs.
|
|
And updates the scopes if need.
|
|
"""
|
|
@spec get_or_make(map(), list(String.t())) :: {:ok, t()} | {:error, Ecto.Changeset.t()}
|
|
def get_or_make(attrs, scopes) do
|
|
with %__MODULE__{} = app <- Repo.get_by(__MODULE__, attrs) do
|
|
update_scopes(app, scopes)
|
|
else
|
|
_e ->
|
|
%__MODULE__{}
|
|
|> register_changeset(Map.put(attrs, :scopes, scopes))
|
|
|> Repo.insert()
|
|
end
|
|
end
|
|
|
|
defp update_scopes(%__MODULE__{} = app, []), do: {:ok, app}
|
|
defp update_scopes(%__MODULE__{scopes: scopes} = app, scopes), do: {:ok, app}
|
|
|
|
defp update_scopes(%__MODULE__{} = app, scopes) do
|
|
app
|
|
|> change(%{scopes: scopes})
|
|
|> Repo.update()
|
|
end
|
|
|
|
@spec search(map()) :: {:ok, [t()], non_neg_integer()}
|
|
def search(params) do
|
|
query = from(a in __MODULE__)
|
|
|
|
query =
|
|
if params[:client_name] do
|
|
from(a in query, where: a.client_name == ^params[:client_name])
|
|
else
|
|
query
|
|
end
|
|
|
|
query =
|
|
if params[:client_id] do
|
|
from(a in query, where: a.client_id == ^params[:client_id])
|
|
else
|
|
query
|
|
end
|
|
|
|
query =
|
|
if Map.has_key?(params, :trusted) do
|
|
from(a in query, where: a.trusted == ^params[:trusted])
|
|
else
|
|
query
|
|
end
|
|
|
|
query =
|
|
from(u in query,
|
|
limit: ^params[:page_size],
|
|
offset: ^((params[:page] - 1) * params[:page_size])
|
|
)
|
|
|
|
count = Repo.aggregate(__MODULE__, :count, :id)
|
|
|
|
{:ok, Repo.all(query), count}
|
|
end
|
|
|
|
@spec get_user_apps(User.t()) :: {:ok, [t()], non_neg_integer()}
|
|
def get_user_apps(%User{id: user_id}) do
|
|
from(a in __MODULE__, where: a.user_id == ^user_id)
|
|
|> Repo.all()
|
|
end
|
|
|
|
@spec destroy(pos_integer()) :: {:ok, t()} | {:error, Ecto.Changeset.t()}
|
|
def destroy(id) do
|
|
with %__MODULE__{} = app <- Repo.get(__MODULE__, id) do
|
|
Repo.delete(app)
|
|
end
|
|
end
|
|
|
|
@spec errors(Ecto.Changeset.t()) :: map()
|
|
def errors(changeset) do
|
|
Enum.reduce(changeset.errors, %{}, fn
|
|
{:client_name, {error, _}}, acc ->
|
|
Map.put(acc, :name, error)
|
|
|
|
{key, {error, _}}, acc ->
|
|
Map.put(acc, key, error)
|
|
end)
|
|
end
|
|
end
|