akkoma/lib/pleroma/web/pleroma_api/controllers/emoji_api_controller.ex

305 lines
8.5 KiB
Elixir
Raw Normal View History

defmodule Pleroma.Web.PleromaAPI.EmojiAPIController do
use Pleroma.Web, :controller
2020-03-28 10:34:32 +00:00
alias Pleroma.Emoji.Pack
2019-08-12 10:13:01 +00:00
plug(Pleroma.Web.ApiSpec.CastAndValidate)
plug(
2020-03-28 10:34:32 +00:00
Pleroma.Plugs.OAuthScopesPlug,
%{scopes: ["write"], admin: true}
when action in [
2020-04-30 14:50:57 +00:00
:import_from_filesystem,
:remote,
:download,
:create,
:update,
:delete,
:add_file,
:update_file,
:delete_file
]
)
@skip_plugs [Pleroma.Plugs.OAuthScopesPlug, Pleroma.Plugs.ExpectPublicOrAuthenticatedCheckPlug]
plug(:skip_plug, @skip_plugs when action in [:archive, :show, :list])
defdelegate open_api_operation(action), to: Pleroma.Web.ApiSpec.PleromaEmojiOperation
def remote(conn, %{url: url}) do
with {:ok, packs} <- Pack.list_remote(url) do
2020-03-28 10:34:32 +00:00
json(conn, packs)
else
2020-05-18 15:43:23 +00:00
{:error, :not_shareable} ->
2020-03-28 10:34:32 +00:00
conn
|> put_status(:internal_server_error)
|> json(%{error: "The requested instance does not support sharing emoji packs"})
end
end
def index(conn, _params) do
2020-03-28 10:34:32 +00:00
emoji_path =
[:instance, :static_dir]
|> Pleroma.Config.get!()
|> Path.join("emoji")
2020-03-28 10:34:32 +00:00
with {:ok, packs} <- Pack.list_local() do
2020-03-28 10:34:32 +00:00
json(conn, packs)
else
2020-05-18 15:43:23 +00:00
{:error, :create_dir, e} ->
conn
|> put_status(:internal_server_error)
2020-03-28 10:34:32 +00:00
|> json(%{error: "Failed to create the emoji pack directory at #{emoji_path}: #{e}"})
2020-05-18 15:43:23 +00:00
{:error, :ls, e} ->
conn
|> put_status(:internal_server_error)
|> json(%{
2020-03-28 10:34:32 +00:00
error: "Failed to get the contents of the emoji pack directory at #{emoji_path}: #{e}"
})
2019-09-11 15:48:51 +00:00
end
end
def show(conn, %{name: name}) do
2020-03-28 10:34:32 +00:00
name = String.trim(name)
2019-09-11 15:48:51 +00:00
2020-03-28 10:34:32 +00:00
with {:ok, pack} <- Pack.show(name) do
json(conn, pack)
2019-09-11 15:48:51 +00:00
else
2020-05-18 15:43:23 +00:00
{:error, :not_found} ->
2020-03-28 10:34:32 +00:00
conn
|> put_status(:not_found)
|> json(%{error: "Pack #{name} does not exist"})
2019-08-12 10:13:01 +00:00
2020-03-28 10:34:32 +00:00
{:error, :empty_values} ->
conn
|> put_status(:bad_request)
|> json(%{error: "pack name cannot be empty"})
end
end
def archive(conn, %{name: name}) do
with {:ok, archive} <- Pack.get_archive(name) do
2020-03-28 10:34:32 +00:00
send_download(conn, {:binary, archive}, filename: "#{name}.zip")
else
2020-05-18 15:43:23 +00:00
{:error, :cant_download} ->
conn
|> put_status(:forbidden)
2019-09-11 16:39:47 +00:00
|> json(%{
2020-03-28 10:34:32 +00:00
error:
"Pack #{name} cannot be downloaded from this instance, either pack sharing was disabled for this pack or some files are missing"
2019-09-11 16:39:47 +00:00
})
2020-05-18 15:43:23 +00:00
{:error, :not_found} ->
conn
|> put_status(:not_found)
2019-09-11 16:39:47 +00:00
|> json(%{error: "Pack #{name} does not exist"})
end
end
def download(%{body_params: %{url: url, name: name} = params} = conn, _) do
2020-05-18 15:43:23 +00:00
with {:ok, _pack} <- Pack.download(name, url, params[:as]) do
2020-03-28 10:34:32 +00:00
json(conn, "ok")
2019-09-11 15:59:31 +00:00
else
2020-05-18 15:43:23 +00:00
{:error, :not_shareable} ->
2020-03-28 10:34:32 +00:00
conn
|> put_status(:internal_server_error)
|> json(%{error: "The requested instance does not support sharing emoji packs"})
2020-05-18 15:43:23 +00:00
{:error, :imvalid_checksum} ->
2020-03-28 10:34:32 +00:00
conn
|> put_status(:internal_server_error)
|> json(%{error: "SHA256 for the pack doesn't match the one sent by the server"})
{:error, e} ->
conn
|> put_status(:internal_server_error)
|> json(%{error: e})
end
end
def create(conn, %{name: name}) do
2020-03-28 10:34:32 +00:00
name = String.trim(name)
2020-05-18 15:43:23 +00:00
with {:ok, _pack} <- Pack.create(name) do
2020-03-28 10:34:32 +00:00
json(conn, "ok")
else
{:error, :eexist} ->
conn
|> put_status(:conflict)
|> json(%{error: "A pack named \"#{name}\" already exists"})
2020-03-28 10:34:32 +00:00
{:error, :empty_values} ->
conn
|> put_status(:bad_request)
|> json(%{error: "pack name cannot be empty"})
2020-03-28 10:34:32 +00:00
{:error, _} ->
render_error(
conn,
:internal_server_error,
"Unexpected error occurred while creating pack."
)
end
end
def delete(conn, %{name: name}) do
2020-03-28 10:34:32 +00:00
name = String.trim(name)
with {:ok, deleted} when deleted != [] <- Pack.delete(name) do
json(conn, "ok")
else
{:ok, []} ->
conn
|> put_status(:not_found)
|> json(%{error: "Pack #{name} does not exist"})
2020-03-28 10:34:32 +00:00
{:error, :empty_values} ->
conn
|> put_status(:bad_request)
|> json(%{error: "pack name cannot be empty"})
2020-02-25 14:34:56 +00:00
{:error, _, _} ->
2019-09-11 16:39:47 +00:00
conn
|> put_status(:internal_server_error)
|> json(%{error: "Couldn't delete the pack #{name}"})
end
end
def update(%{body_params: %{metadata: metadata}} = conn, %{name: name}) do
with {:ok, pack} <- Pack.update_metadata(name, metadata) do
2020-03-28 10:34:32 +00:00
json(conn, pack.pack)
else
2020-05-18 15:43:23 +00:00
{:error, :incomplete} ->
conn
|> put_status(:bad_request)
2019-09-11 16:39:47 +00:00
|> json(%{error: "The fallback archive does not have all files specified in pack.json"})
2020-02-06 15:01:12 +00:00
2020-03-28 10:34:32 +00:00
{:error, _} ->
render_error(
conn,
:internal_server_error,
"Unexpected error occurred while updating pack metadata."
)
2019-09-11 16:39:47 +00:00
end
end
def add_file(%{body_params: params} = conn, %{name: name}) do
filename = params[:filename] || get_filename(params[:file])
shortcode = params[:shortcode] || Path.basename(filename, Path.extname(filename))
2020-03-28 10:34:32 +00:00
with {:ok, pack} <- Pack.add_file(name, shortcode, filename, params[:file]) do
2020-03-28 10:34:32 +00:00
json(conn, pack.files)
2019-09-11 16:39:47 +00:00
else
2020-05-18 15:43:23 +00:00
{:error, :already_exists} ->
2019-09-11 16:39:47 +00:00
conn
|> put_status(:conflict)
|> json(%{error: "An emoji with the \"#{shortcode}\" shortcode already exists"})
2020-05-18 15:43:23 +00:00
{:error, :not_found} ->
2020-03-28 10:34:32 +00:00
conn
|> put_status(:bad_request)
|> json(%{error: "pack \"#{name}\" is not found"})
2020-03-28 10:34:32 +00:00
{:error, :empty_values} ->
2019-09-11 16:39:47 +00:00
conn
|> put_status(:bad_request)
2020-03-28 10:34:32 +00:00
|> json(%{error: "pack name, shortcode or filename cannot be empty"})
{:error, _} ->
render_error(
conn,
:internal_server_error,
"Unexpected error occurred while adding file to pack."
)
2019-09-11 16:39:47 +00:00
end
end
def update_file(%{body_params: %{shortcode: shortcode} = params} = conn, %{name: name}) do
new_shortcode = params[:new_shortcode]
new_filename = params[:new_filename]
force = params[:force]
with {:ok, pack} <- Pack.update_file(name, shortcode, new_shortcode, new_filename, force) do
2020-03-28 10:34:32 +00:00
json(conn, pack.files)
else
2020-05-18 15:43:23 +00:00
{:error, :doesnt_exist} ->
2020-03-28 10:34:32 +00:00
conn
|> put_status(:bad_request)
|> json(%{error: "Emoji \"#{shortcode}\" does not exist"})
2019-09-11 16:39:47 +00:00
2020-05-18 15:43:23 +00:00
{:error, :already_exists} ->
conn
|> put_status(:conflict)
|> json(%{
error:
"New shortcode \"#{new_shortcode}\" is already used. If you want to override emoji use 'force' option"
})
2020-05-18 15:43:23 +00:00
{:error, :not_found} ->
2020-03-28 10:34:32 +00:00
conn
|> put_status(:bad_request)
|> json(%{error: "pack \"#{name}\" is not found"})
2019-09-11 16:39:47 +00:00
2020-03-28 10:34:32 +00:00
{:error, :empty_values} ->
conn
|> put_status(:bad_request)
|> json(%{error: "new_shortcode or new_filename cannot be empty"})
2019-09-11 16:39:47 +00:00
2020-03-28 10:34:32 +00:00
{:error, _} ->
render_error(
conn,
:internal_server_error,
"Unexpected error occurred while updating file in pack."
2020-03-28 10:34:32 +00:00
)
2019-09-11 16:39:47 +00:00
end
end
def delete_file(conn, %{name: name, shortcode: shortcode}) do
with {:ok, pack} <- Pack.delete_file(name, shortcode) do
2020-03-28 10:34:32 +00:00
json(conn, pack.files)
2019-09-11 16:39:47 +00:00
else
2020-05-18 15:43:23 +00:00
{:error, :doesnt_exist} ->
2019-09-11 16:39:47 +00:00
conn
|> put_status(:bad_request)
|> json(%{error: "Emoji \"#{shortcode}\" does not exist"})
2020-05-18 15:43:23 +00:00
{:error, :not_found} ->
2019-09-11 16:39:47 +00:00
conn
|> put_status(:bad_request)
2020-03-28 10:34:32 +00:00
|> json(%{error: "pack \"#{name}\" is not found"})
2019-09-11 16:39:47 +00:00
2020-03-28 10:34:32 +00:00
{:error, :empty_values} ->
2019-09-11 16:39:47 +00:00
conn
|> put_status(:bad_request)
|> json(%{error: "pack name or shortcode cannot be empty"})
2020-03-28 10:34:32 +00:00
{:error, _} ->
render_error(
conn,
:internal_server_error,
"Unexpected error occurred while removing file from pack."
2020-03-28 10:34:32 +00:00
)
end
end
def import_from_filesystem(conn, _params) do
2020-03-28 10:34:32 +00:00
with {:ok, names} <- Pack.import_from_filesystem() do
json(conn, names)
else
2020-03-30 06:09:27 +00:00
{:error, :no_read_write} ->
2020-01-29 10:51:17 +00:00
conn
|> put_status(:internal_server_error)
2020-01-30 14:09:41 +00:00
|> json(%{error: "Error: emoji pack directory must be writable"})
2020-01-29 10:51:17 +00:00
{:error, _} ->
conn
|> put_status(:internal_server_error)
2019-09-11 16:39:47 +00:00
|> json(%{error: "Error accessing emoji pack directory"})
end
end
2020-03-28 10:34:32 +00:00
defp get_filename(%Plug.Upload{filename: filename}), do: filename
defp get_filename(url) when is_binary(url), do: Path.basename(url)
end