diff --git a/changelog.d/heif-upload-filter.add b/changelog.d/heif-upload-filter.add new file mode 100644 index 000000000..a5d75b24f --- /dev/null +++ b/changelog.d/heif-upload-filter.add @@ -0,0 +1 @@ +Pleroma.Upload.Filter.HeifToJpeg to convert Apple HEIF/HEIC files to JPEG diff --git a/docs/configuration/cheatsheet.md b/docs/configuration/cheatsheet.md index 6e2fddcb6..879a7f66a 100644 --- a/docs/configuration/cheatsheet.md +++ b/docs/configuration/cheatsheet.md @@ -712,6 +712,12 @@ No specific configuration. * `args`: List of actions for the `mogrify` command like `"strip"` or `["strip", "auto-orient", {"implode", "1"}]`. +#### Pleroma.Upload.Filter.HeifToJpeg + +This filter converts HEIC/HEIF files commonly found on MacOS/iOS to JPEGs. + +No specific configuration. + ## Email ### Pleroma.Emails.Mailer diff --git a/lib/pleroma/application_requirements.ex b/lib/pleroma/application_requirements.ex index 87ecb7e2d..9fd003b43 100644 --- a/lib/pleroma/application_requirements.ex +++ b/lib/pleroma/application_requirements.ex @@ -172,7 +172,8 @@ defmodule Pleroma.ApplicationRequirements do check_filter(Pleroma.Upload.Filter.Exiftool.ReadDescription, "exiftool"), check_filter(Pleroma.Upload.Filter.Mogrify, "mogrify"), check_filter(Pleroma.Upload.Filter.Mogrifun, "mogrify"), - check_filter(Pleroma.Upload.Filter.AnalyzeMetadata, "ffprobe") + check_filter(Pleroma.Upload.Filter.AnalyzeMetadata, "ffprobe"), + check_filter(Pleroma.Upload.Filter.HeifToJpeg) ] preview_proxy_commands_status = @@ -254,6 +255,12 @@ defmodule Pleroma.ApplicationRequirements do defp check_repo_pool_size!(result), do: result + defp check_filter(Pleroma.Upload.Filter.HeifToJpeg) do + {:ok, supported} = Vix.Vips.Image.supported_saver_suffixes() + + ".heif" in supported + end + defp check_filter(filter, command_required) do filters = Config.get([Pleroma.Upload, :filters]) diff --git a/lib/pleroma/upload/filter/heif_to_jpeg.ex b/lib/pleroma/upload/filter/heif_to_jpeg.ex new file mode 100644 index 000000000..4fc7c2f04 --- /dev/null +++ b/lib/pleroma/upload/filter/heif_to_jpeg.ex @@ -0,0 +1,42 @@ +# Pleroma: A lightweight social networking server +# Copyright © 2017-2021 Pleroma Authors +# SPDX-License-Identifier: AGPL-3.0-only + +defmodule Pleroma.Upload.Filter.HeifToJpeg do + @behaviour Pleroma.Upload.Filter + alias Pleroma.Upload + alias Vix.Vips.Image + + @spec filter(Pleroma.Upload.t()) :: {:ok, :atom} | {:error, String.t()} + def filter( + %Pleroma.Upload{name: name, path: path, tempfile: tempfile, content_type: "image/heic"} = + upload + ) do + try do + name = name |> String.replace_suffix(".heic", ".jpg") + path = path |> String.replace_suffix(".heic", ".jpg") + convert(tempfile) + + {:ok, :filtered, %Upload{upload | name: name, path: path, content_type: "image/jpeg"}} + rescue + e in ErlangError -> + {:error, "#{__MODULE__}: #{inspect(e)}"} + end + end + + def filter(_), do: {:ok, :noop} + + defp convert(tempfile) do + new_file = "#{tempfile}.heic" + + with {:ok, image} <- Image.new_from_file(tempfile) do + Image.write_to_stream(image, ".jpg") + |> Stream.into(File.stream!(new_file)) + |> Stream.run() + + File.rename(new_file, tempfile) + else + _ -> raise("Error converting to JPEG with Vix") + end + end +end diff --git a/test/fixtures/heictmp b/test/fixtures/heictmp new file mode 100644 index 000000000..e69de29bb diff --git a/test/fixtures/image.heic b/test/fixtures/image.heic new file mode 100644 index 000000000..efd119a0e Binary files /dev/null and b/test/fixtures/image.heic differ diff --git a/test/pleroma/upload/filter/heif_to_jpeg_test.exs b/test/pleroma/upload/filter/heif_to_jpeg_test.exs new file mode 100644 index 000000000..2173ecc67 --- /dev/null +++ b/test/pleroma/upload/filter/heif_to_jpeg_test.exs @@ -0,0 +1,39 @@ +# Pleroma: A lightweight social networking server +# Copyright © 2017-2021 Pleroma Authors +# SPDX-License-Identifier: AGPL-3.0-only + +defmodule Pleroma.Upload.Filter.HeifToJpegTest do + use Pleroma.DataCase, async: true + alias Pleroma.Upload.Filter + + @tag :tmp_dir + test "apply HeicToJpeg filter", %{tmp_dir: tmp_dir} do + tmp_file_path = Path.join([tmp_dir, "image.heic"]) + + File.cp!( + "test/fixtures/image.heic", + tmp_file_path + ) + + upload = %Pleroma.Upload{ + name: "image.heic", + content_type: "image/heic", + path: Path.absname("test/fixtures/image.heic"), + tempfile: tmp_file_path + } + + {:ok, :filtered, result} = Filter.HeifToJpeg.filter(upload) + + assert result.content_type == "image/jpeg" + assert result.name == "image.jpg" + assert String.ends_with?(result.path, "jpg") + + assert {:ok, + %Majic.Result{ + content: + "JPEG image data, JFIF standard 1.02, resolution (DPI), density 96x96, segment length 16, progressive, precision 8, 1024x768, components 3", + encoding: "binary", + mime_type: "image/jpeg" + }} == Majic.perform(result.path, pool: Pleroma.MajicPool) + end +end