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