Change emoji api responses to JSON

This commit is contained in:
Ekaterina Vaartis 2019-09-11 19:39:47 +03:00
parent f24731788e
commit 7c784128fd
2 changed files with 186 additions and 166 deletions

View file

@ -6,7 +6,7 @@ defmodule Pleroma.Web.EmojiAPI.EmojiAPIController do
def reload(conn, _params) do def reload(conn, _params) do
Pleroma.Emoji.reload() Pleroma.Emoji.reload()
conn |> text("ok") conn |> json("ok")
end end
@emoji_dir_path Path.join( @emoji_dir_path Path.join(
@ -133,13 +133,15 @@ keeping it in cache for #{div(cache_ms, 1000)}s")
{:can_download?, _} -> {:can_download?, _} ->
conn conn
|> put_status(:forbidden) |> put_status(:forbidden)
|> text("Pack #{name} cannot be downloaded from this instance, either pack sharing\ |> json(%{
was disabled for this pack or some files are missing") error: "Pack #{name} cannot be downloaded from this instance, either pack sharing\
was disabled for this pack or some files are missing"
})
{:exists?, _} -> {:exists?, _} ->
conn conn
|> put_status(:not_found) |> put_status(:not_found)
|> text("Pack #{name} does not exist") |> json(%{error: "Pack #{name} does not exist"})
end end
end end
@ -200,15 +202,15 @@ keeping it in cache for #{div(cache_ms, 1000)}s")
File.write!(pack_file_path, Jason.encode!(full_pack, pretty: true)) File.write!(pack_file_path, Jason.encode!(full_pack, pretty: true))
end end
text(conn, "ok") json(conn, "ok")
else else
{:error, e} -> {:error, e} ->
conn |> put_status(:internal_server_error) |> text(e) conn |> put_status(:internal_server_error) |> json(%{error: e})
{:sha, _} -> {:sha, _} ->
conn conn
|> put_status(:internal_server_error) |> put_status(:internal_server_error)
|> text("SHA256 for the pack doesn't match the one sent by the server") |> json(%{error: "SHA256 for the pack doesn't match the one sent by the server"})
end end
end end
@ -228,11 +230,11 @@ keeping it in cache for #{div(cache_ms, 1000)}s")
Jason.encode!(%{pack: %{}, files: %{}}) Jason.encode!(%{pack: %{}, files: %{}})
) )
conn |> text("ok") conn |> json("ok")
else else
conn conn
|> put_status(:conflict) |> put_status(:conflict)
|> text("A pack named \"#{name}\" already exists") |> json(%{error: "A pack named \"#{name}\" already exists"})
end end
end end
@ -244,10 +246,12 @@ keeping it in cache for #{div(cache_ms, 1000)}s")
case File.rm_rf(pack_dir) do case File.rm_rf(pack_dir) do
{:ok, _} -> {:ok, _} ->
conn |> text("ok") conn |> json("ok")
{:error, _} -> {:error, _} ->
conn |> put_status(:internal_server_error) |> text("Couldn't delete the pack #{name}") conn
|> put_status(:internal_server_error)
|> json(%{error: "Couldn't delete the pack #{name}"})
end end
end end
@ -281,7 +285,7 @@ keeping it in cache for #{div(cache_ms, 1000)}s")
{:has_all_files?, _} -> {:has_all_files?, _} ->
conn conn
|> put_status(:bad_request) |> put_status(:bad_request)
|> text("The fallback archive does not have all files specified in pack.json") |> json(%{error: "The fallback archive does not have all files specified in pack.json"})
end end
end end
@ -302,6 +306,25 @@ keeping it in cache for #{div(cache_ms, 1000)}s")
json(conn, new_data) json(conn, new_data)
end end
defp get_filename(%{"filename" => filename}), do: filename
defp get_filename(%{"file" => file}) do
case file do
%Plug.Upload{filename: filename} -> filename
url when is_binary(url) -> Path.basename(url)
end
end
defp empty?(str), do: String.trim(str) == ""
defp update_file_and_send(conn, updated_full_pack, pack_file_p) do
# Write the emoji pack file
File.write!(pack_file_p, Jason.encode!(updated_full_pack, pretty: true))
# Return the modified file list
json(conn, updated_full_pack["files"])
end
@doc """ @doc """
Updates a file in a pack. Updates a file in a pack.
@ -316,157 +339,154 @@ keeping it in cache for #{div(cache_ms, 1000)}s")
(from the current filename to `new_filename`) (from the current filename to `new_filename`)
- `remove` removes the emoji named `shortcode` and it's associated file - `remove` removes the emoji named `shortcode` and it's associated file
""" """
# Add
def update_file( def update_file(
conn, conn,
%{"pack_name" => pack_name, "action" => action, "shortcode" => shortcode} = params %{"pack_name" => pack_name, "action" => "add", "shortcode" => shortcode} = params
) do ) do
pack_dir = Path.join(@emoji_dir_path, pack_name) pack_dir = Path.join(@emoji_dir_path, pack_name)
pack_file_p = Path.join(pack_dir, "pack.json") pack_file_p = Path.join(pack_dir, "pack.json")
full_pack = Jason.decode!(File.read!(pack_file_p)) full_pack = Jason.decode!(File.read!(pack_file_p))
res = with {_, false} <- {:has_shortcode, Map.has_key?(full_pack["files"], shortcode)},
case action do filename <- get_filename(params),
"add" -> false <- empty?(shortcode),
unless Map.has_key?(full_pack["files"], shortcode) do false <- empty?(filename) do
filename = file_path = Path.join(pack_dir, filename)
if Map.has_key?(params, "filename") do
params["filename"]
else
case params["file"] do
%Plug.Upload{filename: filename} -> filename
url when is_binary(url) -> Path.basename(url)
end
end
unless String.trim(shortcode) |> String.length() == 0 or # If the name contains directories, create them
String.trim(filename) |> String.length() == 0 do if String.contains?(file_path, "/") do
file_path = Path.join(pack_dir, filename) File.mkdir_p!(Path.dirname(file_path))
# If the name contains directories, create them
if String.contains?(file_path, "/") do
File.mkdir_p!(Path.dirname(file_path))
end
case params["file"] do
%Plug.Upload{path: upload_path} ->
# Copy the uploaded file from the temporary directory
File.copy!(upload_path, file_path)
url when is_binary(url) ->
# Download and write the file
file_contents = Tesla.get!(url).body
File.write!(file_path, file_contents)
end
updated_full_pack = put_in(full_pack, ["files", shortcode], filename)
{:ok, updated_full_pack}
else
{:error,
conn
|> put_status(:bad_request)
|> text("shortcode or filename cannot be empty")}
end
else
{:error,
conn
|> put_status(:conflict)
|> text("An emoji with the \"#{shortcode}\" shortcode already exists")}
end
"remove" ->
if Map.has_key?(full_pack["files"], shortcode) do
{emoji_file_path, updated_full_pack} = pop_in(full_pack, ["files", shortcode])
emoji_file_path = Path.join(pack_dir, emoji_file_path)
# Delete the emoji file
File.rm!(emoji_file_path)
# If the old directory has no more files, remove it
if String.contains?(emoji_file_path, "/") do
dir = Path.dirname(emoji_file_path)
if Enum.empty?(File.ls!(dir)) do
File.rmdir!(dir)
end
end
{:ok, updated_full_pack}
else
{:error,
conn |> put_status(:bad_request) |> text("Emoji \"#{shortcode}\" does not exist")}
end
"update" ->
if Map.has_key?(full_pack["files"], shortcode) do
with %{"new_shortcode" => new_shortcode, "new_filename" => new_filename} <- params do
unless String.trim(new_shortcode) |> String.length() == 0 or
String.trim(new_filename) |> String.length() == 0 do
# First, remove the old shortcode, saving the old path
{old_emoji_file_path, updated_full_pack} = pop_in(full_pack, ["files", shortcode])
old_emoji_file_path = Path.join(pack_dir, old_emoji_file_path)
new_emoji_file_path = Path.join(pack_dir, new_filename)
# If the name contains directories, create them
if String.contains?(new_emoji_file_path, "/") do
File.mkdir_p!(Path.dirname(new_emoji_file_path))
end
# Move/Rename the old filename to a new filename
# These are probably on the same filesystem, so just rename should work
:ok = File.rename(old_emoji_file_path, new_emoji_file_path)
# If the old directory has no more files, remove it
if String.contains?(old_emoji_file_path, "/") do
dir = Path.dirname(old_emoji_file_path)
if Enum.empty?(File.ls!(dir)) do
File.rmdir!(dir)
end
end
# Then, put in the new shortcode with the new path
updated_full_pack =
put_in(updated_full_pack, ["files", new_shortcode], new_filename)
{:ok, updated_full_pack}
else
{:error,
conn
|> put_status(:bad_request)
|> text("new_shortcode or new_filename cannot be empty")}
end
else
_ ->
{:error,
conn
|> put_status(:bad_request)
|> text("new_shortcode or new_file were not specified")}
end
else
{:error,
conn |> put_status(:bad_request) |> text("Emoji \"#{shortcode}\" does not exist")}
end
_ ->
{:error, conn |> put_status(:bad_request) |> text("Unknown action: #{action}")}
end end
case res do case params["file"] do
{:ok, updated_full_pack} -> %Plug.Upload{path: upload_path} ->
# Write the emoji pack file # Copy the uploaded file from the temporary directory
File.write!(pack_file_p, Jason.encode!(updated_full_pack, pretty: true)) File.copy!(upload_path, file_path)
# Return the modified file list url when is_binary(url) ->
conn |> json(updated_full_pack["files"]) # Download and write the file
file_contents = Tesla.get!(url).body
File.write!(file_path, file_contents)
end
{:error, e} -> updated_full_pack = put_in(full_pack, ["files", shortcode], filename)
e update_file_and_send(conn, updated_full_pack, pack_file_p)
else
{:has_shortcode, _} ->
conn
|> put_status(:conflict)
|> json(%{error: "An emoji with the \"#{shortcode}\" shortcode already exists"})
true ->
conn
|> put_status(:bad_request)
|> json(%{error: "shortcode or filename cannot be empty"})
end end
end end
# Remove
def update_file(conn, %{
"pack_name" => pack_name,
"action" => "remove",
"shortcode" => shortcode
}) do
pack_dir = Path.join(@emoji_dir_path, pack_name)
pack_file_p = Path.join(pack_dir, "pack.json")
full_pack = Jason.decode!(File.read!(pack_file_p))
if Map.has_key?(full_pack["files"], shortcode) do
{emoji_file_path, updated_full_pack} = pop_in(full_pack, ["files", shortcode])
emoji_file_path = Path.join(pack_dir, emoji_file_path)
# Delete the emoji file
File.rm!(emoji_file_path)
# If the old directory has no more files, remove it
if String.contains?(emoji_file_path, "/") do
dir = Path.dirname(emoji_file_path)
if Enum.empty?(File.ls!(dir)) do
File.rmdir!(dir)
end
end
update_file_and_send(conn, updated_full_pack, pack_file_p)
else
conn
|> put_status(:bad_request)
|> json(%{error: "Emoji \"#{shortcode}\" does not exist"})
end
end
# Update
def update_file(
conn,
%{"pack_name" => pack_name, "action" => "update", "shortcode" => shortcode} = params
) do
pack_dir = Path.join(@emoji_dir_path, pack_name)
pack_file_p = Path.join(pack_dir, "pack.json")
full_pack = Jason.decode!(File.read!(pack_file_p))
with {_, true} <- {:has_shortcode, Map.has_key?(full_pack["files"], shortcode)},
%{"new_shortcode" => new_shortcode, "new_filename" => new_filename} <- params,
false <- empty?(new_shortcode),
false <- empty?(new_filename) do
# First, remove the old shortcode, saving the old path
{old_emoji_file_path, updated_full_pack} = pop_in(full_pack, ["files", shortcode])
old_emoji_file_path = Path.join(pack_dir, old_emoji_file_path)
new_emoji_file_path = Path.join(pack_dir, new_filename)
# If the name contains directories, create them
if String.contains?(new_emoji_file_path, "/") do
File.mkdir_p!(Path.dirname(new_emoji_file_path))
end
# Move/Rename the old filename to a new filename
# These are probably on the same filesystem, so just rename should work
:ok = File.rename(old_emoji_file_path, new_emoji_file_path)
# If the old directory has no more files, remove it
if String.contains?(old_emoji_file_path, "/") do
dir = Path.dirname(old_emoji_file_path)
if Enum.empty?(File.ls!(dir)) do
File.rmdir!(dir)
end
end
# Then, put in the new shortcode with the new path
updated_full_pack = put_in(updated_full_pack, ["files", new_shortcode], new_filename)
update_file_and_send(conn, updated_full_pack, pack_file_p)
else
{:has_shortcode, _} ->
conn
|> put_status(:bad_request)
|> json(%{error: "Emoji \"#{shortcode}\" does not exist"})
true ->
conn
|> put_status(:bad_request)
|> json(%{error: "new_shortcode or new_filename cannot be empty"})
_ ->
conn
|> put_status(:bad_request)
|> json(%{error: "new_shortcode or new_file were not specified"})
end
end
def update_file(conn, %{"action" => action}) do
conn
|> put_status(:bad_request)
|> json(%{error: "Unknown action: #{action}"})
end
@doc """ @doc """
Imports emoji from the filesystem. Imports emoji from the filesystem.
@ -493,7 +513,7 @@ keeping it in cache for #{div(cache_ms, 1000)}s")
{:error, _} -> {:error, _} ->
conn conn
|> put_status(:internal_server_error) |> put_status(:internal_server_error)
|> text("Error accessing emoji pack directory") |> json(%{error: "Error accessing emoji pack directory"})
end end
end end

View file

@ -101,14 +101,14 @@ defmodule Pleroma.Web.EmojiAPI.EmojiAPIControllerTest do
} }
|> Jason.encode!() |> Jason.encode!()
) )
|> text_response(200) == "ok" |> json_response(200) == "ok"
assert File.exists?("#{@emoji_dir_path}/test_pack2/pack.json") assert File.exists?("#{@emoji_dir_path}/test_pack2/pack.json")
assert File.exists?("#{@emoji_dir_path}/test_pack2/blank.png") assert File.exists?("#{@emoji_dir_path}/test_pack2/blank.png")
assert conn assert conn
|> delete(emoji_api_path(conn, :delete, "test_pack2")) |> delete(emoji_api_path(conn, :delete, "test_pack2"))
|> response(200) == "ok" |> json_response(200) == "ok"
refute File.exists?("#{@emoji_dir_path}/test_pack2") refute File.exists?("#{@emoji_dir_path}/test_pack2")
@ -130,14 +130,14 @@ defmodule Pleroma.Web.EmojiAPI.EmojiAPIControllerTest do
} }
|> Jason.encode!() |> Jason.encode!()
) )
|> text_response(200) == "ok" |> json_response(200) == "ok"
assert File.exists?("#{@emoji_dir_path}/test_pack_nonshared2/pack.json") assert File.exists?("#{@emoji_dir_path}/test_pack_nonshared2/pack.json")
assert File.exists?("#{@emoji_dir_path}/test_pack_nonshared2/blank.png") assert File.exists?("#{@emoji_dir_path}/test_pack_nonshared2/blank.png")
assert conn assert conn
|> delete(emoji_api_path(conn, :delete, "test_pack_nonshared2")) |> delete(emoji_api_path(conn, :delete, "test_pack_nonshared2"))
|> response(200) == "ok" |> json_response(200) == "ok"
refute File.exists?("#{@emoji_dir_path}/test_pack_nonshared2") refute File.exists?("#{@emoji_dir_path}/test_pack_nonshared2")
end end
@ -225,15 +225,15 @@ defmodule Pleroma.Web.EmojiAPI.EmojiAPIControllerTest do
conn = build_conn() conn = build_conn()
assert conn assert (conn
|> assign(:user, ctx[:admin]) |> assign(:user, ctx[:admin])
|> post( |> post(
emoji_api_path(conn, :update_metadata, "test_pack"), emoji_api_path(conn, :update_metadata, "test_pack"),
%{ %{
"new_data" => new_data "new_data" => new_data
} }
) )
|> text_response(:bad_request) =~ "does not have all" |> json_response(:bad_request))["error"] =~ "does not have all"
end end
end end
@ -267,9 +267,9 @@ defmodule Pleroma.Web.EmojiAPI.EmojiAPIControllerTest do
conn = conn |> assign(:user, admin) conn = conn |> assign(:user, admin)
assert conn assert (conn
|> post(emoji_api_path(conn, :update_file, "test_pack"), same_name) |> post(emoji_api_path(conn, :update_file, "test_pack"), same_name)
|> text_response(:conflict) =~ "already exists" |> json_response(:conflict))["error"] =~ "already exists"
assert conn assert conn
|> post(emoji_api_path(conn, :update_file, "test_pack"), different_name) |> post(emoji_api_path(conn, :update_file, "test_pack"), different_name)
@ -350,7 +350,7 @@ defmodule Pleroma.Web.EmojiAPI.EmojiAPIControllerTest do
"test_created" "test_created"
) )
) )
|> text_response(200) == "ok" |> json_response(200) == "ok"
assert File.exists?("#{@emoji_dir_path}/test_created/pack.json") assert File.exists?("#{@emoji_dir_path}/test_created/pack.json")
@ -361,7 +361,7 @@ defmodule Pleroma.Web.EmojiAPI.EmojiAPIControllerTest do
assert conn assert conn
|> delete(emoji_api_path(conn, :delete, "test_created")) |> delete(emoji_api_path(conn, :delete, "test_created"))
|> response(200) == "ok" |> json_response(200) == "ok"
refute File.exists?("#{@emoji_dir_path}/test_created/pack.json") refute File.exists?("#{@emoji_dir_path}/test_created/pack.json")
end end