1 defmodule Pleroma.Web.PleromaAPI.EmojiAPIController do
2 use Pleroma.Web, :controller
6 @emoji_dir_path Path.join(
7 Pleroma.Config.get!([:instance, :static_dir]),
11 @cache_seconds_per_file Pleroma.Config.get!([:emoji, :shared_pack_cache_seconds_per_file])
14 Lists the packs available on the instance as JSON.
16 The information is public and does not require authentification. The format is
17 a map of "pack directory name" to pack.json contents.
19 def list_packs(conn, _params) do
20 with {:ok, results} <- File.ls(@emoji_dir_path) do
23 |> Enum.filter(&has_pack_json?/1)
24 |> Enum.map(&load_pack/1)
25 # Check if all the files are in place and can be sent
26 |> Enum.map(&validate_pack/1)
27 # Transform into a map of pack-name => pack-data
30 json(conn, pack_infos)
34 defp has_pack_json?(file) do
35 dir_path = Path.join(@emoji_dir_path, file)
36 # Filter to only use the pack.json packs
37 File.dir?(dir_path) and File.exists?(Path.join(dir_path, "pack.json"))
40 defp load_pack(pack_name) do
41 pack_path = Path.join(@emoji_dir_path, pack_name)
42 pack_file = Path.join(pack_path, "pack.json")
44 {pack_name, Jason.decode!(File.read!(pack_file))}
47 defp validate_pack({name, pack}) do
48 pack_path = Path.join(@emoji_dir_path, name)
50 if can_download?(pack, pack_path) do
51 archive_for_sha = make_archive(name, pack, pack_path)
52 archive_sha = :crypto.hash(:sha256, archive_for_sha) |> Base.encode16()
56 |> put_in(["pack", "can-download"], true)
57 |> put_in(["pack", "download-sha256"], archive_sha)
61 {name, put_in(pack, ["pack", "can-download"], false)}
65 defp can_download?(pack, pack_path) do
66 # If the pack is set as shared, check if it can be downloaded
67 # That means that when asked, the pack can be packed and sent to the remote
68 # Otherwise, they'd have to download it from external-src
69 pack["pack"]["share-files"] &&
70 Enum.all?(pack["files"], fn {_, path} ->
71 File.exists?(Path.join(pack_path, path))
75 defp create_archive_and_cache(name, pack, pack_dir, md5) do
78 (pack["files"] |> Enum.map(fn {_, path} -> to_charlist(path) end))
80 {:ok, {_, zip_result}} = :zip.zip('#{name}.zip', files, [:memory, cwd: to_charlist(pack_dir)])
82 cache_ms = :timer.seconds(@cache_seconds_per_file * Enum.count(files))
87 # if pack.json MD5 changes, the cache is not valid anymore
88 %{pack_json_md5: md5, pack_data: zip_result},
89 # Add a minute to cache time for every file in the pack
93 Logger.debug("Created an archive for the '#{name}' emoji pack, \
94 keeping it in cache for #{div(cache_ms, 1000)}s")
99 defp make_archive(name, pack, pack_dir) do
100 # Having a different pack.json md5 invalidates cache
101 pack_file_md5 = :crypto.hash(:md5, File.read!(Path.join(pack_dir, "pack.json")))
103 case Cachex.get!(:emoji_packs_cache, name) do
104 %{pack_file_md5: ^pack_file_md5, pack_data: zip_result} ->
105 Logger.debug("Using cache for the '#{name}' shared emoji pack")
109 create_archive_and_cache(name, pack, pack_dir, pack_file_md5)
114 An endpoint for other instances (via admin UI) or users (via browser)
115 to download packs that the instance shares.
117 def download_shared(conn, %{"name" => name}) do
118 pack_dir = Path.join(@emoji_dir_path, name)
119 pack_file = Path.join(pack_dir, "pack.json")
121 with {_, true} <- {:exists?, File.exists?(pack_file)},
122 pack = Jason.decode!(File.read!(pack_file)),
123 {_, true} <- {:can_download?, can_download?(pack, pack_dir)} do
124 zip_result = make_archive(name, pack, pack_dir)
125 send_download(conn, {:binary, zip_result}, filename: "#{name}.zip")
127 {:can_download?, _} ->
129 |> put_status(:forbidden)
131 error: "Pack #{name} cannot be downloaded from this instance, either pack sharing\
132 was disabled for this pack or some files are missing"
137 |> put_status(:not_found)
138 |> json(%{error: "Pack #{name} does not exist"})
143 An admin endpoint to request downloading a pack named `pack_name` from the instance
146 If the requested instance's admin chose to share the pack, it will be downloaded
147 from that instance, otherwise it will be downloaded from the fallback source, if there is one.
149 def download_from(conn, %{"instance_address" => address, "pack_name" => name} = data) do
150 shareable_packs_available =
151 "#{address}/.well-known/nodeinfo"
157 # Get the actual nodeinfo address and fetch it
161 |> get_in(["metadata", "features"])
162 |> Enum.member?("shareable_emoji_packs")
164 if shareable_packs_available do
166 "#{address}/api/pleroma/emoji/packs/list"
173 case full_pack["pack"] do
174 %{"share-files" => true, "can-download" => true, "download-sha256" => sha} ->
178 uri: "#{address}/api/pleroma/emoji/packs/download_shared/#{name}"
181 %{"fallback-src" => src, "fallback-src-sha256" => sha} when is_binary(src) ->
191 "The pack was not set as shared and there is no fallback src to download from"}
194 with {:ok, %{sha: sha, uri: uri} = pinfo} <- pack_info_res,
195 %{body: emoji_archive} <- Tesla.get!(uri),
196 {_, true} <- {:checksum, Base.decode16!(sha) == :crypto.hash(:sha256, emoji_archive)} do
197 local_name = data["as"] || name
198 pack_dir = Path.join(@emoji_dir_path, local_name)
199 File.mkdir_p!(pack_dir)
201 files = Enum.map(full_pack["files"], fn {_, path} -> to_charlist(path) end)
202 # Fallback cannot contain a pack.json file
203 files = if pinfo[:fallback], do: files, else: ['pack.json'] ++ files
205 {:ok, _} = :zip.unzip(emoji_archive, cwd: to_charlist(pack_dir), file_list: files)
207 # Fallback can't contain a pack.json file, since that would cause the fallback-src-sha256
208 # in it to depend on itself
209 if pinfo[:fallback] do
210 pack_file_path = Path.join(pack_dir, "pack.json")
212 File.write!(pack_file_path, Jason.encode!(full_pack, pretty: true))
218 conn |> put_status(:internal_server_error) |> json(%{error: e})
222 |> put_status(:internal_server_error)
223 |> json(%{error: "SHA256 for the pack doesn't match the one sent by the server"})
227 |> put_status(:internal_server_error)
228 |> json(%{error: "The requested instance does not support sharing emoji packs"})
233 Creates an empty pack named `name` which then can be updated via the admin UI.
235 def create(conn, %{"name" => name}) do
236 pack_dir = Path.join(@emoji_dir_path, name)
238 if not File.exists?(pack_dir) do
239 File.mkdir_p!(pack_dir)
241 pack_file_p = Path.join(pack_dir, "pack.json")
245 Jason.encode!(%{pack: %{}, files: %{}})
251 |> put_status(:conflict)
252 |> json(%{error: "A pack named \"#{name}\" already exists"})
257 Deletes the pack `name` and all it's files.
259 def delete(conn, %{"name" => name}) do
260 pack_dir = Path.join(@emoji_dir_path, name)
262 case File.rm_rf(pack_dir) do
268 |> put_status(:internal_server_error)
269 |> json(%{error: "Couldn't delete the pack #{name}"})
274 An endpoint to update `pack_names`'s metadata.
276 `new_data` is the new metadata for the pack, that will replace the old metadata.
278 def update_metadata(conn, %{"pack_name" => name, "new_data" => new_data}) do
279 pack_file_p = Path.join([@emoji_dir_path, name, "pack.json"])
281 full_pack = Jason.decode!(File.read!(pack_file_p))
283 # The new fallback-src is in the new data and it's not the same as it was in the old data
284 should_update_fb_sha =
285 not is_nil(new_data["fallback-src"]) and
286 new_data["fallback-src"] != full_pack["pack"]["fallback-src"]
288 with {_, true} <- {:should_update?, should_update_fb_sha},
289 %{body: pack_arch} <- Tesla.get!(new_data["fallback-src"]),
290 {:ok, flist} <- :zip.unzip(pack_arch, [:memory]),
291 {_, true} <- {:has_all_files?, has_all_files?(full_pack, flist)} do
292 fallback_sha = :crypto.hash(:sha256, pack_arch) |> Base.encode16()
294 new_data = Map.put(new_data, "fallback-src-sha256", fallback_sha)
295 update_metadata_and_send(conn, full_pack, new_data, pack_file_p)
297 {:should_update?, _} ->
298 update_metadata_and_send(conn, full_pack, new_data, pack_file_p)
300 {:has_all_files?, _} ->
302 |> put_status(:bad_request)
303 |> json(%{error: "The fallback archive does not have all files specified in pack.json"})
307 # Check if all files from the pack.json are in the archive
308 defp has_all_files?(%{"files" => files}, flist) do
309 Enum.all?(files, fn {_, from_manifest} ->
310 Enum.find(flist, fn {from_archive, _} ->
311 to_string(from_archive) == from_manifest
316 defp update_metadata_and_send(conn, full_pack, new_data, pack_file_p) do
317 full_pack = Map.put(full_pack, "pack", new_data)
318 File.write!(pack_file_p, Jason.encode!(full_pack, pretty: true))
320 # Send new data back with fallback sha filled
324 defp get_filename(%{"filename" => filename}), do: filename
326 defp get_filename(%{"file" => file}) do
328 %Plug.Upload{filename: filename} -> filename
329 url when is_binary(url) -> Path.basename(url)
333 defp empty?(str), do: String.trim(str) == ""
335 defp update_file_and_send(conn, updated_full_pack, pack_file_p) do
336 # Write the emoji pack file
337 File.write!(pack_file_p, Jason.encode!(updated_full_pack, pretty: true))
339 # Return the modified file list
340 json(conn, updated_full_pack["files"])
344 Updates a file in a pack.
346 Updating can mean three things:
348 - `add` adds an emoji named `shortcode` to the pack `pack_name`,
349 that means that the emoji file needs to be uploaded with the request
350 (thus requiring it to be a multipart request) and be named `file`.
351 There can also be an optional `filename` that will be the new emoji file name
352 (if it's not there, the name will be taken from the uploaded file).
353 - `update` changes emoji shortcode (from `shortcode` to `new_shortcode` or moves the file
354 (from the current filename to `new_filename`)
355 - `remove` removes the emoji named `shortcode` and it's associated file
361 %{"pack_name" => pack_name, "action" => "add", "shortcode" => shortcode} = params
363 pack_dir = Path.join(@emoji_dir_path, pack_name)
364 pack_file_p = Path.join(pack_dir, "pack.json")
366 full_pack = Jason.decode!(File.read!(pack_file_p))
368 with {_, false} <- {:has_shortcode, Map.has_key?(full_pack["files"], shortcode)},
369 filename <- get_filename(params),
370 false <- empty?(shortcode),
371 false <- empty?(filename) do
372 file_path = Path.join(pack_dir, filename)
374 # If the name contains directories, create them
375 if String.contains?(file_path, "/") do
376 File.mkdir_p!(Path.dirname(file_path))
379 case params["file"] do
380 %Plug.Upload{path: upload_path} ->
381 # Copy the uploaded file from the temporary directory
382 File.copy!(upload_path, file_path)
384 url when is_binary(url) ->
385 # Download and write the file
386 file_contents = Tesla.get!(url).body
387 File.write!(file_path, file_contents)
390 updated_full_pack = put_in(full_pack, ["files", shortcode], filename)
391 update_file_and_send(conn, updated_full_pack, pack_file_p)
393 {:has_shortcode, _} ->
395 |> put_status(:conflict)
396 |> json(%{error: "An emoji with the \"#{shortcode}\" shortcode already exists"})
400 |> put_status(:bad_request)
401 |> json(%{error: "shortcode or filename cannot be empty"})
406 def update_file(conn, %{
407 "pack_name" => pack_name,
408 "action" => "remove",
409 "shortcode" => shortcode
411 pack_dir = Path.join(@emoji_dir_path, pack_name)
412 pack_file_p = Path.join(pack_dir, "pack.json")
414 full_pack = Jason.decode!(File.read!(pack_file_p))
416 if Map.has_key?(full_pack["files"], shortcode) do
417 {emoji_file_path, updated_full_pack} = pop_in(full_pack, ["files", shortcode])
419 emoji_file_path = Path.join(pack_dir, emoji_file_path)
421 # Delete the emoji file
422 File.rm!(emoji_file_path)
424 # If the old directory has no more files, remove it
425 if String.contains?(emoji_file_path, "/") do
426 dir = Path.dirname(emoji_file_path)
428 if Enum.empty?(File.ls!(dir)) do
433 update_file_and_send(conn, updated_full_pack, pack_file_p)
436 |> put_status(:bad_request)
437 |> json(%{error: "Emoji \"#{shortcode}\" does not exist"})
444 %{"pack_name" => pack_name, "action" => "update", "shortcode" => shortcode} = params
446 pack_dir = Path.join(@emoji_dir_path, pack_name)
447 pack_file_p = Path.join(pack_dir, "pack.json")
449 full_pack = Jason.decode!(File.read!(pack_file_p))
451 with {_, true} <- {:has_shortcode, Map.has_key?(full_pack["files"], shortcode)},
452 %{"new_shortcode" => new_shortcode, "new_filename" => new_filename} <- params,
453 false <- empty?(new_shortcode),
454 false <- empty?(new_filename) do
455 # First, remove the old shortcode, saving the old path
456 {old_emoji_file_path, updated_full_pack} = pop_in(full_pack, ["files", shortcode])
457 old_emoji_file_path = Path.join(pack_dir, old_emoji_file_path)
458 new_emoji_file_path = Path.join(pack_dir, new_filename)
460 # If the name contains directories, create them
461 if String.contains?(new_emoji_file_path, "/") do
462 File.mkdir_p!(Path.dirname(new_emoji_file_path))
465 # Move/Rename the old filename to a new filename
466 # These are probably on the same filesystem, so just rename should work
467 :ok = File.rename(old_emoji_file_path, new_emoji_file_path)
469 # If the old directory has no more files, remove it
470 if String.contains?(old_emoji_file_path, "/") do
471 dir = Path.dirname(old_emoji_file_path)
473 if Enum.empty?(File.ls!(dir)) do
478 # Then, put in the new shortcode with the new path
479 updated_full_pack = put_in(updated_full_pack, ["files", new_shortcode], new_filename)
480 update_file_and_send(conn, updated_full_pack, pack_file_p)
482 {:has_shortcode, _} ->
484 |> put_status(:bad_request)
485 |> json(%{error: "Emoji \"#{shortcode}\" does not exist"})
489 |> put_status(:bad_request)
490 |> json(%{error: "new_shortcode or new_filename cannot be empty"})
494 |> put_status(:bad_request)
495 |> json(%{error: "new_shortcode or new_file were not specified"})
499 def update_file(conn, %{"action" => action}) do
501 |> put_status(:bad_request)
502 |> json(%{error: "Unknown action: #{action}"})
506 Imports emoji from the filesystem.
508 Importing means checking all the directories in the
509 `$instance_static/emoji/` for directories which do not have
510 `pack.json`. If one has an emoji.txt file, that file will be used
511 to create a `pack.json` file with it's contents. If the directory has
512 neither, all the files with specific configured extenstions will be
513 assumed to be emojis and stored in the new `pack.json` file.
515 def import_from_fs(conn, _params) do
516 with {:ok, results} <- File.ls(@emoji_dir_path) do
517 imported_pack_names =
519 |> Enum.filter(fn file ->
520 dir_path = Path.join(@emoji_dir_path, file)
521 # Find the directories that do NOT have pack.json
522 File.dir?(dir_path) and not File.exists?(Path.join(dir_path, "pack.json"))
524 |> Enum.map(&write_pack_json_contents/1)
526 json(conn, imported_pack_names)
530 |> put_status(:internal_server_error)
531 |> json(%{error: "Error accessing emoji pack directory"})
535 defp write_pack_json_contents(dir) do
536 dir_path = Path.join(@emoji_dir_path, dir)
537 emoji_txt_path = Path.join(dir_path, "emoji.txt")
539 files_for_pack = files_for_pack(emoji_txt_path, dir_path)
540 pack_json_contents = Jason.encode!(%{pack: %{}, files: files_for_pack})
542 File.write!(Path.join(dir_path, "pack.json"), pack_json_contents)
547 defp files_for_pack(emoji_txt_path, dir_path) do
548 if File.exists?(emoji_txt_path) do
549 # There's an emoji.txt file, it's likely from a pack installed by the pack manager.
550 # Make a pack.json file from the contents of that emoji.txt fileh
552 # FIXME: Copy-pasted from Pleroma.Emoji/load_from_file_stream/2
554 # Create a map of shortcodes to filenames from emoji.txt
555 File.read!(emoji_txt_path)
556 |> String.split("\n")
557 |> Enum.map(&String.trim/1)
558 |> Enum.map(fn line ->
559 case String.split(line, ~r/,\s*/) do
560 # This matches both strings with and without tags
561 # and we don't care about tags here
562 [name, file | _] -> {name, file}
566 |> Enum.filter(fn x -> not is_nil(x) end)
569 # If there's no emoji.txt, assume all files
570 # that are of certain extensions from the config are emojis and import them all
571 pack_extensions = Pleroma.Config.get!([:emoji, :pack_extensions])
572 Pleroma.Emoji.make_shortcode_to_file_map(dir_path, pack_extensions)