From 8842da7523ba45dce32f6dd2bfc2d066c2ecb641 Mon Sep 17 00:00:00 2001 From: Luca Versari Date: Wed, 1 May 2024 10:37:58 +0200 Subject: [PATCH] Add JPEG XL support to image processing. As discussed in #2421. --- Cargo.lock | 115 ++++++++++++++++++ components/imageproc/src/format.rs | 10 +- components/imageproc/src/meta.rs | 31 ++++- components/imageproc/src/processor.rs | 53 +++++++- components/imageproc/tests/resize_image.rs | 10 ++ components/imageproc/tests/test_imgs/jxl.jxl | Bin 0 -> 21972 bytes components/libs/Cargo.toml | 1 + components/libs/src/lib.rs | 1 + .../content/image-processing/index.md | 1 + 9 files changed, 210 insertions(+), 12 deletions(-) create mode 100644 components/imageproc/tests/test_imgs/jxl.jxl diff --git a/Cargo.lock b/Cargo.lock index 293d58133a..4c5e73796a 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -636,6 +636,15 @@ version = "0.7.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "98cc8fbded0c607b7ba9dd60cd98df59af97e84d24e49c8557331cfc26d301ce" +[[package]] +name = "cmake" +version = "0.1.50" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a31c789563b815f77f4250caee12365734369f942439b7defd71e18a48197130" +dependencies = [ + "cc", +] + [[package]] name = "codemap" version = "0.1.3" @@ -876,6 +885,41 @@ dependencies = [ "windows-sys 0.52.0", ] +[[package]] +name = "darling" +version = "0.20.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "54e36fcd13ed84ffdfda6f5be89b31287cbb80c439841fe69e04841435464391" +dependencies = [ + "darling_core", + "darling_macro", +] + +[[package]] +name = "darling_core" +version = "0.20.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9c2cf1c23a687a1feeb728783b993c4e1ad83d99f351801977dd809b48d0a70f" +dependencies = [ + "fnv", + "ident_case", + "proc-macro2", + "quote", + "strsim 0.10.0", + "syn 2.0.61", +] + +[[package]] +name = "darling_macro" +version = "0.20.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a668eda54683121533a393014d8692171709ff57a7d61f187b6e782719f8933f" +dependencies = [ + "darling_core", + "quote", + "syn 2.0.61", +] + [[package]] name = "dashmap" version = "5.5.3" @@ -913,6 +957,37 @@ dependencies = [ "powerfmt", ] +[[package]] +name = "derive_builder" +version = "0.20.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0350b5cb0331628a5916d6c5c0b72e97393b8b6b03b47a9284f4e7f5a405ffd7" +dependencies = [ + "derive_builder_macro", +] + +[[package]] +name = "derive_builder_core" +version = "0.20.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d48cda787f839151732d396ac69e3473923d54312c070ee21e9effcaa8ca0b1d" +dependencies = [ + "darling", + "proc-macro2", + "quote", + "syn 2.0.61", +] + +[[package]] +name = "derive_builder_macro" +version = "0.20.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "206868b8242f27cecce124c19fd88157fbd0dd334df2587f36417bafbc85097b" +dependencies = [ + "derive_builder_core", + "syn 2.0.61", +] + [[package]] name = "deunicode" version = "1.4.4" @@ -1654,6 +1729,12 @@ dependencies = [ "cc", ] +[[package]] +name = "ident_case" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b9e0384b61958566e926dc50660321d12159025e767c18e043daf26b70104c39" + [[package]] name = "idna" version = "0.5.0" @@ -1889,6 +1970,39 @@ version = "0.3.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "f5d4a7da358eff58addd2877a45865158f0d78c911d43a5784ceb7bbf52833b0" +[[package]] +name = "jpegxl-rs" +version = "0.10.3+libjxl-0.10.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e697c7532d4a244d91306a19c63763769366fbc61e75d5a08cc602923e478a7a" +dependencies = [ + "byteorder", + "derive_builder", + "half", + "image", + "jpegxl-sys", + "thiserror", +] + +[[package]] +name = "jpegxl-src" +version = "0.10.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7808dc25b79f6ff27137f91b1b7659eb27f570d401ff464d03fd336041fc5c68" +dependencies = [ + "cmake", +] + +[[package]] +name = "jpegxl-sys" +version = "0.10.3+libjxl-0.10.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c8262d19ed55829441dc2a2b1315c8f15e040139443fc1dbc48551890020a5c2" +dependencies = [ + "jpegxl-src", + "pkg-config", +] + [[package]] name = "js-sys" version = "0.3.69" @@ -1992,6 +2106,7 @@ dependencies = [ "globset", "grass", "image", + "jpegxl-rs", "lexical-sort", "minify-html", "nom-bibtex", diff --git a/components/imageproc/src/format.rs b/components/imageproc/src/format.rs index d7c992eab8..ea49d73756 100644 --- a/components/imageproc/src/format.rs +++ b/components/imageproc/src/format.rs @@ -6,12 +6,14 @@ const DEFAULT_Q_JPG: u8 = 75; /// Thumbnail image format #[derive(Debug, Clone, Copy, PartialEq, Eq)] pub enum Format { - /// JPEG, The `u8` argument is JPEG quality (in percent). + /// JPEG, The `u8` argument is JPEG quality (1..100). Jpeg(u8), /// PNG Png, - /// WebP, The `u8` argument is WebP quality (in percent), None meaning lossless. + /// WebP, The `u8` argument is WebP quality (1..100), None meaning lossless. WebP(Option), + /// JPEG XL, The `u8` argument is quality (1..100), None meaning lossless. + JXL(Option), } impl Format { @@ -32,6 +34,7 @@ impl Format { "jpeg" | "jpg" => Ok(Jpeg(jpg_quality)), "png" => Ok(Png), "webp" => Ok(WebP(quality)), + "jxl" => Ok(JXL(quality)), _ => Err(anyhow!("Invalid image format: {}", format)), } } @@ -44,6 +47,7 @@ impl Format { Png => "png", Jpeg(_) => "jpg", WebP(_) => "webp", + JXL(_) => "jxl", } } } @@ -58,6 +62,8 @@ impl Hash for Format { Jpeg(q) => 1001 + q as u16, WebP(None) => 2000, WebP(Some(q)) => 2001 + q as u16, + JXL(None) => 3000, + JXL(Some(q)) => 3001 + q as u16, }; hasher.write_u16(q); diff --git a/components/imageproc/src/meta.rs b/components/imageproc/src/meta.rs index 8ee83c335f..1ba38897c1 100644 --- a/components/imageproc/src/meta.rs +++ b/components/imageproc/src/meta.rs @@ -1,6 +1,7 @@ use errors::{anyhow, Context, Result}; use libs::image::io::Reader as ImgReader; -use libs::image::{ImageFormat, ImageResult}; +use libs::image::ImageFormat; +use libs::jpegxl_rs::decoder_builder; use libs::svg_metadata::Metadata as SvgMetadata; use serde::Serialize; use std::ffi::OsStr; @@ -15,12 +16,23 @@ pub struct ImageMeta { } impl ImageMeta { - pub fn read(path: &Path) -> ImageResult { - let reader = ImgReader::open(path).and_then(ImgReader::with_guessed_format)?; - let format = reader.format(); - let size = reader.into_dimensions()?; + pub fn read(path: &Path) -> Result { + if path.extension().is_some_and(|ext| ext == "jxl") { + Self::read_jxl(path) + } else { + let reader = ImgReader::open(path).and_then(ImgReader::with_guessed_format)?; + let format = reader.format(); + let size = reader.into_dimensions()?; - Ok(Self { size, format }) + Ok(Self { size, format }) + } + } + + fn read_jxl(path: &Path) -> Result { + let input = std::fs::read(path)?; + let decoder = decoder_builder().build()?; + let (meta, _) = decoder.decode(&input)?; + Ok(ImageMeta { size: (meta.width, meta.height), format: None }) } pub fn is_lossy(&self) -> bool { @@ -44,6 +56,9 @@ impl ImageMetaResponse { pub fn new_svg(width: u32, height: u32) -> Self { Self { width, height, format: Some("svg"), mime: Some("text/svg+xml") } } + pub fn new_jxl(width: u32, height: u32) -> Self { + Self { width, height, format: Some("jxl"), mime: Some("image/jxl") } + } } impl From for ImageMetaResponse { @@ -75,6 +90,10 @@ pub fn read_image_metadata>(path: P) -> Result // this is not a typo, this returns the correct values for width and height. .map(|(h, w)| ImageMetaResponse::new_svg(w as u32, h as u32)) } + "jxl" => { + let meta = ImageMeta::read(path)?; + Ok(ImageMetaResponse::new_jxl(meta.size.0, meta.size.1)) + } _ => ImageMeta::read(path).map(ImageMetaResponse::from).with_context(err_context), } } diff --git a/components/imageproc/src/processor.rs b/components/imageproc/src/processor.rs index 577aeb1f3e..72f8a42615 100644 --- a/components/imageproc/src/processor.rs +++ b/components/imageproc/src/processor.rs @@ -4,13 +4,16 @@ use std::io::{BufWriter, Write}; use std::path::{Path, PathBuf}; use config::Config; -use errors::{anyhow, Context, Result}; +use errors::{anyhow, bail, Context, Result}; use libs::ahash::{HashMap, HashSet}; use libs::image::codecs::jpeg::JpegEncoder; use libs::image::imageops::FilterType; -use libs::image::{EncodableLayout, ImageFormat}; +use libs::image::{ColorType, EncodableLayout, ImageFormat}; +use libs::jpegxl_rs::decoder_builder; +use libs::jpegxl_rs::encode::EncoderFrame; +use libs::jpegxl_rs::image::ToDynamic; use libs::rayon::prelude::*; -use libs::{image, webp}; +use libs::{image, jpegxl_rs, webp}; use serde::{Deserialize, Serialize}; use utils::fs as ufs; @@ -39,7 +42,15 @@ impl ImageOp { return Ok(()); } - let img = image::open(&self.input_path)?; + let img = if self.input_path.extension().is_some_and(|ext| ext == "jxl") { + let input = std::fs::read(&self.input_path)?; + let decoder = decoder_builder().build()?; + decoder + .decode_to_image(&input)? + .context("jxl image could not be represented in an Image")? + } else { + image::open(&self.input_path)? + }; let mut img = fix_orientation(&img, &self.input_path).unwrap_or(img); let img = match self.instr.crop_instruction { @@ -71,6 +82,40 @@ impl ImageOp { }; buffered_f.write_all(memory.as_bytes())?; } + Format::JXL(q) => { + let mut encoder = jpegxl_rs::encoder_builder(); + if let Some(q) = q { + if q == 100 { + encoder.uses_original_profile(true); + encoder.lossless(true); + } else { + encoder.set_jpeg_quality(q as f32); + } + } else { + encoder.uses_original_profile(true); + encoder.lossless(true); + } + let frame = EncoderFrame::new(img.as_bytes()); + let frame = match img.color() { + ColorType::L8 => frame.num_channels(1), + ColorType::La8 => { + encoder.has_alpha(true); + frame.num_channels(2) + } + ColorType::Rgb8 => frame.num_channels(3), + ColorType::Rgba8 => { + encoder.has_alpha(true); + frame.num_channels(4) + } + _ => { + bail!("Unsupported pixel type {:?}", img.color()); + } + }; + let mut encoder = encoder.build()?; + buffered_f.write_all( + &encoder.encode_frame::(&frame, img.width(), img.height())?.data, + )?; + } } Ok(()) diff --git a/components/imageproc/tests/resize_image.rs b/components/imageproc/tests/resize_image.rs index 41114abf50..40e32732e1 100644 --- a/components/imageproc/tests/resize_image.rs +++ b/components/imageproc/tests/resize_image.rs @@ -132,6 +132,16 @@ fn resize_image_webp_jpg() { image_op_test("webp.webp", "scale", Some(150), Some(150), "auto", "jpg", 150, 150, 300, 380); } +#[test] +fn resize_image_png_jxl() { + image_op_test("png.png", "scale", Some(150), Some(150), "jxl", "jxl", 150, 150, 300, 380); +} + +#[test] +fn resize_image_jxl_png() { + image_op_test("jxl.jxl", "scale", Some(150), Some(150), "png", "png", 150, 150, 300, 380); +} + #[test] fn read_image_metadata_jpg() { assert_eq!( diff --git a/components/imageproc/tests/test_imgs/jxl.jxl b/components/imageproc/tests/test_imgs/jxl.jxl new file mode 100644 index 0000000000000000000000000000000000000000..a5e3db1de35224a54f7f1cd3ffe582a217bb6e03 GIT binary patch literal 21972 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