mirror of
https://git.asonix.dog/asonix/pict-rs
synced 2025-01-04 16:51:25 +00:00
Avoid shelling out to imagemagick for some blurhash operations
This commit is contained in:
parent
31e17b4d62
commit
7ba67cff22
9 changed files with 244 additions and 26 deletions
112
Cargo.lock
generated
112
Cargo.lock
generated
|
@ -521,12 +521,24 @@ version = "3.16.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "79296716171880943b8470b5f8d03aa55eb2e645a4874bdbb28adb49162e012c"
|
checksum = "79296716171880943b8470b5f8d03aa55eb2e645a4874bdbb28adb49162e012c"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "bytemuck"
|
||||||
|
version = "1.20.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "8b37c88a63ffd85d15b406896cc343916d7cf57838a847b3a6f2ca5d39a5695a"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "byteorder"
|
name = "byteorder"
|
||||||
version = "1.5.0"
|
version = "1.5.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "1fd0f2584146f6f2ef48085050886acf353beff7305ebd1ae69500e27c67f64b"
|
checksum = "1fd0f2584146f6f2ef48085050886acf353beff7305ebd1ae69500e27c67f64b"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "byteorder-lite"
|
||||||
|
version = "0.1.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "8f1fe948ff07f4bd06c30984e69f5b4899c516a3ef74f34df92a2df2ab535495"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "bytes"
|
name = "bytes"
|
||||||
version = "1.9.0"
|
version = "1.9.0"
|
||||||
|
@ -643,6 +655,12 @@ dependencies = [
|
||||||
"tracing-error",
|
"tracing-error",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "color_quant"
|
||||||
|
version = "1.1.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "3d7b894f5411737b7867f4827955924d7c254fc9f4d91a6aad6b097804b1018b"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "colorchoice"
|
name = "colorchoice"
|
||||||
version = "1.0.3"
|
version = "1.0.3"
|
||||||
|
@ -1016,6 +1034,15 @@ version = "0.2.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "4443176a9f2c162692bd3d352d745ef9413eec5782a80d8fd6f8a1ac692a07f7"
|
checksum = "4443176a9f2c162692bd3d352d745ef9413eec5782a80d8fd6f8a1ac692a07f7"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "fdeflate"
|
||||||
|
version = "0.3.7"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "1e6853b52649d4ac5c0bd02320cddc5ba956bdb407c4b75a2c6b75bf51500f8c"
|
||||||
|
dependencies = [
|
||||||
|
"simd-adler32",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "flagset"
|
name = "flagset"
|
||||||
version = "0.4.6"
|
version = "0.4.6"
|
||||||
|
@ -1184,6 +1211,16 @@ dependencies = [
|
||||||
"wasm-bindgen",
|
"wasm-bindgen",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "gif"
|
||||||
|
version = "0.13.1"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "3fb2d69b19215e18bb912fa30f7ce15846e301408695e44e0ef719f1da9e19f2"
|
||||||
|
dependencies = [
|
||||||
|
"color_quant",
|
||||||
|
"weezl",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "gimli"
|
name = "gimli"
|
||||||
version = "0.28.1"
|
version = "0.28.1"
|
||||||
|
@ -1610,6 +1647,33 @@ dependencies = [
|
||||||
"icu_properties",
|
"icu_properties",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "image"
|
||||||
|
version = "0.25.5"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "cd6f44aed642f18953a158afeb30206f4d50da59fbc66ecb53c66488de73563b"
|
||||||
|
dependencies = [
|
||||||
|
"bytemuck",
|
||||||
|
"byteorder-lite",
|
||||||
|
"color_quant",
|
||||||
|
"gif",
|
||||||
|
"image-webp",
|
||||||
|
"num-traits",
|
||||||
|
"png",
|
||||||
|
"zune-core",
|
||||||
|
"zune-jpeg",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "image-webp"
|
||||||
|
version = "0.2.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "e031e8e3d94711a9ccb5d6ea357439ef3dcbed361798bd4071dc4d9793fbe22f"
|
||||||
|
dependencies = [
|
||||||
|
"byteorder-lite",
|
||||||
|
"quick-error",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "impl-more"
|
name = "impl-more"
|
||||||
version = "0.1.8"
|
version = "0.1.8"
|
||||||
|
@ -1864,6 +1928,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "e2d80299ef12ff69b16a84bb182e3b9df68b5a91574d3d4fa6e41b65deec4df1"
|
checksum = "e2d80299ef12ff69b16a84bb182e3b9df68b5a91574d3d4fa6e41b65deec4df1"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"adler2",
|
"adler2",
|
||||||
|
"simd-adler32",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -2168,6 +2233,7 @@ dependencies = [
|
||||||
"diesel-derive-enum",
|
"diesel-derive-enum",
|
||||||
"futures-core",
|
"futures-core",
|
||||||
"hex",
|
"hex",
|
||||||
|
"image",
|
||||||
"md-5",
|
"md-5",
|
||||||
"metrics",
|
"metrics",
|
||||||
"metrics-exporter-prometheus",
|
"metrics-exporter-prometheus",
|
||||||
|
@ -2244,6 +2310,19 @@ version = "0.1.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "8b870d8c151b6f2fb93e84a13146138f05d02ed11c7e7c54f8826aaaf7c9f184"
|
checksum = "8b870d8c151b6f2fb93e84a13146138f05d02ed11c7e7c54f8826aaaf7c9f184"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "png"
|
||||||
|
version = "0.17.15"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "b67582bd5b65bdff614270e2ea89a1cf15bef71245cc1e5f7ea126977144211d"
|
||||||
|
dependencies = [
|
||||||
|
"bitflags 1.3.2",
|
||||||
|
"crc32fast",
|
||||||
|
"fdeflate",
|
||||||
|
"flate2",
|
||||||
|
"miniz_oxide 0.8.0",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "portable-atomic"
|
name = "portable-atomic"
|
||||||
version = "1.10.0"
|
version = "1.10.0"
|
||||||
|
@ -2368,6 +2447,12 @@ dependencies = [
|
||||||
"winapi",
|
"winapi",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "quick-error"
|
||||||
|
version = "2.0.1"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "a993555f31e5a609f617c12db6250dedcac1b0a85076912c436e6fc9b2c8e6a3"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "quick-xml"
|
name = "quick-xml"
|
||||||
version = "0.36.2"
|
version = "0.36.2"
|
||||||
|
@ -2982,6 +3067,12 @@ dependencies = [
|
||||||
"libc",
|
"libc",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "simd-adler32"
|
||||||
|
version = "0.3.7"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "d66dc143e6b11c1eddc06d5c423cfc97062865baf299914ab64caa38182078fe"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "siphasher"
|
name = "siphasher"
|
||||||
version = "0.3.11"
|
version = "0.3.11"
|
||||||
|
@ -3876,6 +3967,12 @@ dependencies = [
|
||||||
"rustls-pki-types",
|
"rustls-pki-types",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "weezl"
|
||||||
|
version = "0.1.8"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "53a85b86a771b1c87058196170769dd264f66c0782acf1ae6cc51bfd64b39082"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "whoami"
|
name = "whoami"
|
||||||
version = "1.5.2"
|
version = "1.5.2"
|
||||||
|
@ -4190,3 +4287,18 @@ dependencies = [
|
||||||
"quote",
|
"quote",
|
||||||
"syn",
|
"syn",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "zune-core"
|
||||||
|
version = "0.4.12"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "3f423a2c17029964870cfaabb1f13dfab7d092a62a29a89264f4d36990ca414a"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "zune-jpeg"
|
||||||
|
version = "0.4.13"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "16099418600b4d8f028622f73ff6e3deaabdff330fb9a2a131dea781ee8b0768"
|
||||||
|
dependencies = [
|
||||||
|
"zune-core",
|
||||||
|
]
|
||||||
|
|
|
@ -96,6 +96,7 @@ url = { version = "2.5.2", features = ["serde"] }
|
||||||
uuid = { version = "1.10.0", features = ["serde", "std", "v4", "v7"] }
|
uuid = { version = "1.10.0", features = ["serde", "std", "v4", "v7"] }
|
||||||
# pinned to rustls
|
# pinned to rustls
|
||||||
webpki-roots = "0.26.3"
|
webpki-roots = "0.26.3"
|
||||||
|
image = { version = "0.25.5", default-features = false, features = ["gif", "jpeg", "png", "webp"] }
|
||||||
|
|
||||||
[dependencies.tracing-actix-web]
|
[dependencies.tracing-actix-web]
|
||||||
version = "0.7.15"
|
version = "0.7.15"
|
||||||
|
|
|
@ -1,11 +1,16 @@
|
||||||
use std::ffi::{OsStr, OsString};
|
use std::{
|
||||||
|
ffi::{OsStr, OsString},
|
||||||
|
time::Duration,
|
||||||
|
};
|
||||||
|
|
||||||
use tokio::io::AsyncReadExt;
|
use tokio::io::AsyncReadExt;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
|
bytes_stream::BytesStream,
|
||||||
details::Details,
|
details::Details,
|
||||||
error::{Error, UploadError},
|
error::{Error, UploadError},
|
||||||
formats::ProcessableFormat,
|
formats::ProcessableFormat,
|
||||||
|
future::{WithMetrics, WithTimeout},
|
||||||
magick::{MagickError, MAGICK_CONFIGURE_PATH, MAGICK_TEMPORARY_PATH},
|
magick::{MagickError, MAGICK_CONFIGURE_PATH, MAGICK_TEMPORARY_PATH},
|
||||||
process::Process,
|
process::Process,
|
||||||
repo::Hash,
|
repo::Hash,
|
||||||
|
@ -18,6 +23,27 @@ pub(crate) async fn generate<S>(
|
||||||
hash: Hash,
|
hash: Hash,
|
||||||
original_details: &Details,
|
original_details: &Details,
|
||||||
) -> Result<String, Error>
|
) -> Result<String, Error>
|
||||||
|
where
|
||||||
|
S: Store + 'static,
|
||||||
|
{
|
||||||
|
let permit = crate::process_semaphore().acquire().await?;
|
||||||
|
|
||||||
|
let blurhash = do_generate(state, hash, original_details)
|
||||||
|
.with_timeout(Duration::from_secs(state.config.media.process_timeout * 4))
|
||||||
|
.with_metrics(crate::init_metrics::GENERATE_BLURHASH)
|
||||||
|
.await
|
||||||
|
.map_err(|_| UploadError::ProcessTimeout)??;
|
||||||
|
|
||||||
|
drop(permit);
|
||||||
|
|
||||||
|
Ok(blurhash)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn do_generate<S>(
|
||||||
|
state: &State<S>,
|
||||||
|
hash: Hash,
|
||||||
|
original_details: &Details,
|
||||||
|
) -> Result<String, Error>
|
||||||
where
|
where
|
||||||
S: Store + 'static,
|
S: Store + 'static,
|
||||||
{
|
{
|
||||||
|
@ -35,6 +61,44 @@ where
|
||||||
|
|
||||||
let stream = state.store.to_stream(&identifier, None, None).await?;
|
let stream = state.store.to_stream(&identifier, None, None).await?;
|
||||||
|
|
||||||
|
match input_details.internal_format().image_rs_format() {
|
||||||
|
// supported pure-rust image decoders
|
||||||
|
Some(
|
||||||
|
format @ image::ImageFormat::Gif
|
||||||
|
| format @ image::ImageFormat::Jpeg
|
||||||
|
| format @ image::ImageFormat::Png
|
||||||
|
| format @ image::ImageFormat::WebP,
|
||||||
|
) => {
|
||||||
|
let bytes_stream = BytesStream::try_from_stream(stream).await?;
|
||||||
|
|
||||||
|
let blurhash = crate::sync::spawn_blocking("image-blurhash", move || {
|
||||||
|
let mut vec = Vec::with_capacity(bytes_stream.len());
|
||||||
|
|
||||||
|
for bytes in bytes_stream {
|
||||||
|
vec.extend(bytes);
|
||||||
|
}
|
||||||
|
|
||||||
|
let raw_image = image::ImageReader::with_format(std::io::Cursor::new(vec), format)
|
||||||
|
.decode()
|
||||||
|
.map_err(UploadError::Decode)?
|
||||||
|
.into_rgba8();
|
||||||
|
|
||||||
|
let blurhash = blurhash_update::auto_encode(
|
||||||
|
blurhash_update::ImageBounds {
|
||||||
|
width: raw_image.width(),
|
||||||
|
height: raw_image.height(),
|
||||||
|
},
|
||||||
|
raw_image.as_raw(),
|
||||||
|
);
|
||||||
|
|
||||||
|
Ok(blurhash) as Result<_, UploadError>
|
||||||
|
})
|
||||||
|
.await
|
||||||
|
.map_err(|_| UploadError::Canceled)??;
|
||||||
|
|
||||||
|
Ok(blurhash)
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
let blurhash = read_rgba_command(
|
let blurhash = read_rgba_command(
|
||||||
state,
|
state,
|
||||||
input_details
|
input_details
|
||||||
|
@ -67,6 +131,8 @@ where
|
||||||
.await??;
|
.await??;
|
||||||
|
|
||||||
Ok(blurhash)
|
Ok(blurhash)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn read_rgba_command<S>(
|
async fn read_rgba_command<S>(
|
||||||
|
|
|
@ -174,6 +174,9 @@ pub(crate) enum UploadError {
|
||||||
#[error("Failed external validation")]
|
#[error("Failed external validation")]
|
||||||
FailedExternalValidation,
|
FailedExternalValidation,
|
||||||
|
|
||||||
|
#[error("Failed to decode image")]
|
||||||
|
Decode(#[source] image::error::ImageError),
|
||||||
|
|
||||||
#[cfg(feature = "random-errors")]
|
#[cfg(feature = "random-errors")]
|
||||||
#[error("Randomly generated error for testing purposes")]
|
#[error("Randomly generated error for testing purposes")]
|
||||||
RandomError,
|
RandomError,
|
||||||
|
@ -217,6 +220,7 @@ impl UploadError {
|
||||||
Self::InvalidJob(_, _) => ErrorCode::INVALID_JOB,
|
Self::InvalidJob(_, _) => ErrorCode::INVALID_JOB,
|
||||||
Self::InvalidQuery(_) => ErrorCode::INVALID_QUERY,
|
Self::InvalidQuery(_) => ErrorCode::INVALID_QUERY,
|
||||||
Self::InvalidJson(_) => ErrorCode::INVALID_JSON,
|
Self::InvalidJson(_) => ErrorCode::INVALID_JSON,
|
||||||
|
Self::Decode(_) => ErrorCode::DECODE_IMAGE,
|
||||||
#[cfg(feature = "random-errors")]
|
#[cfg(feature = "random-errors")]
|
||||||
Self::RandomError => ErrorCode::RANDOM_ERROR,
|
Self::RandomError => ErrorCode::RANDOM_ERROR,
|
||||||
}
|
}
|
||||||
|
|
|
@ -153,6 +153,9 @@ impl ErrorCode {
|
||||||
pub(crate) const INVALID_JSON: ErrorCode = ErrorCode {
|
pub(crate) const INVALID_JSON: ErrorCode = ErrorCode {
|
||||||
code: "invalid-json",
|
code: "invalid-json",
|
||||||
};
|
};
|
||||||
|
pub(crate) const DECODE_IMAGE: ErrorCode = ErrorCode {
|
||||||
|
code: "decode-image",
|
||||||
|
};
|
||||||
#[cfg(feature = "random-errors")]
|
#[cfg(feature = "random-errors")]
|
||||||
pub(crate) const RANDOM_ERROR: ErrorCode = ErrorCode {
|
pub(crate) const RANDOM_ERROR: ErrorCode = ErrorCode {
|
||||||
code: "random-error",
|
code: "random-error",
|
||||||
|
|
|
@ -138,6 +138,14 @@ impl InternalFormat {
|
||||||
Self::Video(_) => None,
|
Self::Video(_) => None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub(crate) const fn image_rs_format(self) -> Option<::image::ImageFormat> {
|
||||||
|
match self {
|
||||||
|
Self::Image(format) => format.image_rs_format(),
|
||||||
|
Self::Animation(format) => Some(format.image_rs_format()),
|
||||||
|
Self::Video(_) => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ProcessableFormat {
|
impl ProcessableFormat {
|
||||||
|
|
|
@ -81,4 +81,13 @@ impl AnimationFormat {
|
||||||
Self::Webp => super::mimes::image_webp(),
|
Self::Webp => super::mimes::image_webp(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub(super) const fn image_rs_format(self) -> image::ImageFormat {
|
||||||
|
match self {
|
||||||
|
Self::Apng => image::ImageFormat::Png,
|
||||||
|
Self::Avif => image::ImageFormat::Avif,
|
||||||
|
Self::Gif => image::ImageFormat::Gif,
|
||||||
|
Self::Webp => image::ImageFormat::WebP,
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -99,4 +99,14 @@ impl ImageFormat {
|
||||||
Self::Webp => super::mimes::image_webp(),
|
Self::Webp => super::mimes::image_webp(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub(super) const fn image_rs_format(self) -> Option<image::ImageFormat> {
|
||||||
|
match self {
|
||||||
|
Self::Avif => Some(image::ImageFormat::Avif),
|
||||||
|
Self::Jpeg => Some(image::ImageFormat::Jpeg),
|
||||||
|
Self::Jxl => None,
|
||||||
|
Self::Png => Some(image::ImageFormat::Png),
|
||||||
|
Self::Webp => Some(image::ImageFormat::WebP),
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -558,12 +558,17 @@ fn describe_generate() {
|
||||||
GENERATE_PROCESS,
|
GENERATE_PROCESS,
|
||||||
"Timings for processing media or waiting for media to be processed"
|
"Timings for processing media or waiting for media to be processed"
|
||||||
);
|
);
|
||||||
|
metrics::describe_histogram!(
|
||||||
|
GENERATE_BLURHASH,
|
||||||
|
"Timings for computing blurhashes for media"
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) const GENERATE_START: &str = "pict-rs.generate.start";
|
pub(crate) const GENERATE_START: &str = "pict-rs.generate.start";
|
||||||
pub(crate) const GENERATE_DURATION: &str = "pict-rs.generate.duration";
|
pub(crate) const GENERATE_DURATION: &str = "pict-rs.generate.duration";
|
||||||
pub(crate) const GENERATE_END: &str = "pict-rs.generate.end";
|
pub(crate) const GENERATE_END: &str = "pict-rs.generate.end";
|
||||||
pub(crate) const GENERATE_PROCESS: &str = "pict-rs.generate.process";
|
pub(crate) const GENERATE_PROCESS: &str = "pict-rs.generate.process";
|
||||||
|
pub(crate) const GENERATE_BLURHASH: &str = "pict-rs.generate.blurhash";
|
||||||
|
|
||||||
fn describe_object_storage() {
|
fn describe_object_storage() {
|
||||||
metrics::describe_histogram!(
|
metrics::describe_histogram!(
|
||||||
|
|
Loading…
Reference in a new issue