2
0
Fork 0
mirror of https://git.asonix.dog/asonix/pict-rs synced 2024-11-01 10:09:57 +00:00
pict-rs/src/generate.rs

270 lines
7.4 KiB
Rust
Raw Normal View History

mod ffmpeg;
mod magick;
use crate::{
2023-07-22 16:15:30 +00:00
concurrent_processor::ProcessMap,
details::Details,
error::{Error, UploadError},
formats::{ImageFormat, InputProcessableFormat, InternalVideoFormat, ProcessableFormat},
future::{WithMetrics, WithTimeout},
repo::{ArcRepo, Hash, VariantAlreadyExists},
store::Store,
2023-10-07 16:32:36 +00:00
tmp_file::TmpDir,
};
use actix_web::web::Bytes;
use std::{
path::PathBuf,
sync::Arc,
time::{Duration, Instant},
};
2023-12-23 02:54:02 +00:00
2022-04-07 17:56:40 +00:00
use tracing::Instrument;
2023-07-22 21:47:59 +00:00
struct MetricsGuard {
start: Instant,
armed: bool,
}
impl MetricsGuard {
fn guard() -> Self {
metrics::increment_counter!("pict-rs.generate.start");
Self {
start: Instant::now(),
armed: true,
}
}
fn disarm(mut self) {
self.armed = false;
}
}
impl Drop for MetricsGuard {
fn drop(&mut self) {
metrics::histogram!("pict-rs.generate.duration", self.start.elapsed().as_secs_f64(), "completed" => (!self.armed).to_string());
2023-07-23 02:11:28 +00:00
metrics::increment_counter!("pict-rs.generate.end", "completed" => (!self.armed).to_string());
2023-07-22 21:47:59 +00:00
}
}
2022-10-01 01:02:46 +00:00
#[allow(clippy::too_many_arguments)]
#[tracing::instrument(skip(tmp_dir, repo, store, hash, process_map, config))]
pub(crate) async fn generate<S: Store + 'static>(
2023-10-07 16:32:36 +00:00
tmp_dir: &TmpDir,
repo: &ArcRepo,
store: &S,
2023-07-22 16:15:30 +00:00
process_map: &ProcessMap,
format: InputProcessableFormat,
thumbnail_path: PathBuf,
thumbnail_args: Vec<String>,
original_details: &Details,
2023-11-11 20:22:12 +00:00
config: &crate::config::Configuration,
2023-08-14 19:25:19 +00:00
hash: Hash,
) -> Result<(Details, Bytes), Error> {
2023-11-11 20:22:12 +00:00
if config.server.danger_dummy_mode {
let identifier = repo
.identifier(hash)
.await?
.ok_or(UploadError::MissingIdentifier)?;
2023-11-11 20:22:12 +00:00
let bytes = store.to_bytes(&identifier, None, None).await?.into_bytes();
2023-11-11 20:22:12 +00:00
Ok((original_details.clone(), bytes))
} else {
let process_fut = process(
tmp_dir,
repo,
store,
format,
thumbnail_path.clone(),
thumbnail_args,
original_details,
config,
hash.clone(),
);
let (details, bytes) = process_map
.process(hash, thumbnail_path, process_fut)
.with_timeout(Duration::from_secs(config.media.process_timeout * 4))
.with_metrics("pict-rs.generate.process")
.await
.map_err(|_| UploadError::ProcessTimeout)??;
2023-11-11 20:22:12 +00:00
Ok((details, bytes))
}
}
2022-10-01 01:02:46 +00:00
#[allow(clippy::too_many_arguments)]
2023-12-22 19:30:17 +00:00
#[tracing::instrument(skip(tmp_dir, repo, store, hash, config))]
async fn process<S: Store + 'static>(
2023-10-07 16:32:36 +00:00
tmp_dir: &TmpDir,
repo: &ArcRepo,
store: &S,
output_format: InputProcessableFormat,
thumbnail_path: PathBuf,
thumbnail_args: Vec<String>,
original_details: &Details,
2023-11-11 20:22:12 +00:00
config: &crate::config::Configuration,
2023-08-14 19:25:19 +00:00
hash: Hash,
) -> Result<(Details, Bytes), Error> {
2023-07-22 21:47:59 +00:00
let guard = MetricsGuard::guard();
2023-09-24 20:32:00 +00:00
let permit = crate::process_semaphore().acquire().await?;
let identifier = input_identifier(
2023-10-07 16:32:36 +00:00
tmp_dir,
repo,
store,
output_format,
hash.clone(),
original_details,
2023-11-11 20:22:12 +00:00
&config.media,
)
.await?;
2023-11-11 20:22:12 +00:00
let input_details =
crate::ensure_details_identifier(tmp_dir, repo, store, config, &identifier).await?;
let input_format = input_details
.internal_format()
2023-08-16 17:36:18 +00:00
.processable_format()
.expect("Already verified format is processable");
let format = input_format.process_to(output_format);
let quality = match format {
2023-11-11 20:22:12 +00:00
ProcessableFormat::Image(format) => config.media.image.quality_for(format),
ProcessableFormat::Animation(format) => config.media.animation.quality_for(format),
};
let vec = crate::magick::process_image_store_read(
2023-10-07 16:32:36 +00:00
tmp_dir,
store,
&identifier,
thumbnail_args,
input_format,
format,
quality,
2023-11-11 20:22:12 +00:00
config.media.process_timeout,
)
.await?
.to_vec()
.instrument(tracing::info_span!("Reading processed image to vec"))
.await?;
let bytes = Bytes::from(vec);
drop(permit);
2023-11-11 20:22:12 +00:00
let details = Details::from_bytes(tmp_dir, config.media.process_timeout, bytes.clone()).await?;
let identifier = store
.save_bytes(bytes.clone(), details.media_type())
.await?;
2023-08-16 20:12:16 +00:00
if let Err(VariantAlreadyExists) = repo
.relate_variant_identifier(
hash,
thumbnail_path.to_string_lossy().to_string(),
&identifier,
)
.await?
{
store.remove(&identifier).await?;
}
repo.relate_details(&identifier, &details).await?;
2023-07-22 21:47:59 +00:00
guard.disarm();
Ok((details, bytes)) as Result<(Details, Bytes), Error>
}
#[tracing::instrument(skip_all)]
async fn input_identifier<S>(
2023-10-07 16:32:36 +00:00
tmp_dir: &TmpDir,
repo: &ArcRepo,
store: &S,
output_format: InputProcessableFormat,
hash: Hash,
original_details: &Details,
media: &crate::config::Media,
) -> Result<Arc<str>, Error>
where
S: Store + 'static,
{
let should_thumbnail =
if let Some(input_format) = original_details.internal_format().processable_format() {
let output_format = input_format.process_to(output_format);
input_format.should_thumbnail(output_format)
} else {
// video case
true
};
if should_thumbnail {
if let Some(identifier) = repo.motion_identifier(hash.clone()).await? {
return Ok(identifier);
};
let identifier = repo
.identifier(hash.clone())
.await?
.ok_or(UploadError::MissingIdentifier)?;
let (reader, media_type) = if let Some(processable_format) =
original_details.internal_format().processable_format()
{
let thumbnail_format = media.image.format.unwrap_or(ImageFormat::Webp);
let reader = magick::thumbnail(
2023-10-07 16:32:36 +00:00
tmp_dir,
store,
&identifier,
processable_format,
ProcessableFormat::Image(thumbnail_format),
media.image.quality_for(thumbnail_format),
media.process_timeout,
)
.await?;
(reader, thumbnail_format.media_type())
} else {
let thumbnail_format = match media.image.format {
Some(ImageFormat::Webp | ImageFormat::Avif | ImageFormat::Jxl) => {
ffmpeg::ThumbnailFormat::Webp
}
Some(ImageFormat::Png) => ffmpeg::ThumbnailFormat::Png,
Some(ImageFormat::Jpeg) | None => ffmpeg::ThumbnailFormat::Jpeg,
};
let reader = ffmpeg::thumbnail(
2023-10-07 16:32:36 +00:00
tmp_dir,
store.clone(),
identifier,
original_details
.video_format()
.unwrap_or(InternalVideoFormat::Mp4),
thumbnail_format,
media.process_timeout,
)
.await?;
(reader, thumbnail_format.media_type())
};
let motion_identifier = reader
2023-12-23 02:54:02 +00:00
.with_stdout(|stdout| async { store.save_async_read(stdout, media_type).await })
.await??;
repo.relate_motion_identifier(hash, &motion_identifier)
.await?;
return Ok(motion_identifier);
}
repo.identifier(hash)
.await?
.ok_or(UploadError::MissingIdentifier)
.map_err(From::from)
}