2022-04-01 21:51:12 +00:00
|
|
|
use crate::{
|
|
|
|
config::ImageFormat,
|
|
|
|
error::Error,
|
2022-04-02 21:44:03 +00:00
|
|
|
ingest::Session,
|
2022-04-01 21:51:12 +00:00
|
|
|
queue::{LocalBoxFuture, Process},
|
2022-04-02 21:44:03 +00:00
|
|
|
repo::{Alias, DeleteToken, FullRepo, UploadId, UploadResult},
|
2022-04-01 21:51:12 +00:00
|
|
|
serde_str::Serde,
|
2022-04-02 21:44:03 +00:00
|
|
|
store::{Identifier, Store},
|
2022-04-01 21:51:12 +00:00
|
|
|
};
|
2022-04-02 21:44:03 +00:00
|
|
|
use futures_util::TryStreamExt;
|
2022-04-01 21:51:12 +00:00
|
|
|
use std::path::PathBuf;
|
|
|
|
|
|
|
|
pub(super) fn perform<'a, R, S>(
|
|
|
|
repo: &'a R,
|
|
|
|
store: &'a S,
|
|
|
|
job: &'a [u8],
|
|
|
|
) -> LocalBoxFuture<'a, Result<(), Error>>
|
|
|
|
where
|
2022-04-02 21:44:03 +00:00
|
|
|
R: FullRepo + 'static,
|
2022-04-02 22:40:04 +00:00
|
|
|
S: Store + 'static,
|
2022-04-01 21:51:12 +00:00
|
|
|
{
|
|
|
|
Box::pin(async move {
|
|
|
|
match serde_json::from_slice(job) {
|
|
|
|
Ok(job) => match job {
|
|
|
|
Process::Ingest {
|
|
|
|
identifier,
|
|
|
|
upload_id,
|
|
|
|
declared_alias,
|
|
|
|
should_validate,
|
2022-04-08 17:05:14 +00:00
|
|
|
is_cached,
|
2022-04-01 21:51:12 +00:00
|
|
|
} => {
|
2022-04-02 21:44:03 +00:00
|
|
|
process_ingest(
|
2022-04-01 21:51:12 +00:00
|
|
|
repo,
|
|
|
|
store,
|
|
|
|
identifier,
|
2022-04-03 01:56:29 +00:00
|
|
|
Serde::into_inner(upload_id),
|
2022-04-01 21:51:12 +00:00
|
|
|
declared_alias.map(Serde::into_inner),
|
|
|
|
should_validate,
|
2022-04-08 17:05:14 +00:00
|
|
|
is_cached,
|
2022-04-01 21:51:12 +00:00
|
|
|
)
|
|
|
|
.await?
|
|
|
|
}
|
|
|
|
Process::Generate {
|
|
|
|
target_format,
|
|
|
|
source,
|
|
|
|
process_path,
|
|
|
|
process_args,
|
|
|
|
} => {
|
|
|
|
generate(
|
|
|
|
repo,
|
|
|
|
store,
|
|
|
|
target_format,
|
|
|
|
Serde::into_inner(source),
|
|
|
|
process_path,
|
|
|
|
process_args,
|
|
|
|
)
|
|
|
|
.await?
|
|
|
|
}
|
|
|
|
},
|
|
|
|
Err(e) => {
|
2022-09-25 23:46:26 +00:00
|
|
|
tracing::warn!("Invalid job: {}", format!("{}", e));
|
2022-04-01 21:51:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2022-04-02 21:44:03 +00:00
|
|
|
#[tracing::instrument(skip(repo, store))]
|
|
|
|
async fn process_ingest<R, S>(
|
2022-04-01 21:51:12 +00:00
|
|
|
repo: &R,
|
|
|
|
store: &S,
|
2022-04-02 21:44:03 +00:00
|
|
|
unprocessed_identifier: Vec<u8>,
|
|
|
|
upload_id: UploadId,
|
2022-04-01 21:51:12 +00:00
|
|
|
declared_alias: Option<Alias>,
|
|
|
|
should_validate: bool,
|
2022-04-08 17:05:14 +00:00
|
|
|
is_cached: bool,
|
2022-04-02 21:44:03 +00:00
|
|
|
) -> Result<(), Error>
|
|
|
|
where
|
|
|
|
R: FullRepo + 'static,
|
|
|
|
S: Store,
|
|
|
|
{
|
|
|
|
let fut = async {
|
|
|
|
let unprocessed_identifier = S::Identifier::from_bytes(unprocessed_identifier)?;
|
|
|
|
|
|
|
|
let stream = store
|
|
|
|
.to_stream(&unprocessed_identifier, None, None)
|
|
|
|
.await?
|
|
|
|
.map_err(Error::from);
|
|
|
|
|
2022-04-08 17:05:14 +00:00
|
|
|
let session = crate::ingest::ingest(
|
|
|
|
repo,
|
|
|
|
store,
|
|
|
|
stream,
|
|
|
|
declared_alias,
|
|
|
|
should_validate,
|
|
|
|
is_cached,
|
|
|
|
)
|
|
|
|
.await?;
|
2022-04-02 21:44:03 +00:00
|
|
|
|
|
|
|
let token = session.delete_token().await?;
|
|
|
|
|
2022-04-03 02:15:39 +00:00
|
|
|
store.remove(&unprocessed_identifier).await?;
|
|
|
|
|
2022-04-02 21:44:03 +00:00
|
|
|
Ok((session, token)) as Result<(Session<R, S>, DeleteToken), Error>
|
|
|
|
};
|
|
|
|
|
|
|
|
let result = match fut.await {
|
|
|
|
Ok((mut session, token)) => {
|
|
|
|
let alias = session.alias().take().expect("Alias should exist").clone();
|
|
|
|
let result = UploadResult::Success { alias, token };
|
|
|
|
session.disarm();
|
|
|
|
result
|
|
|
|
}
|
|
|
|
Err(e) => {
|
2022-09-25 23:46:26 +00:00
|
|
|
tracing::warn!("Failed to ingest {}, {}", format!("{}", e), format!("{:?}", e));
|
2022-04-02 21:44:03 +00:00
|
|
|
|
|
|
|
UploadResult::Failure {
|
|
|
|
message: e.to_string(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
repo.complete(upload_id, result).await?;
|
|
|
|
|
|
|
|
Ok(())
|
2022-04-01 21:51:12 +00:00
|
|
|
}
|
|
|
|
|
2022-04-02 22:40:04 +00:00
|
|
|
async fn generate<R: FullRepo, S: Store + 'static>(
|
2022-04-01 21:51:12 +00:00
|
|
|
repo: &R,
|
|
|
|
store: &S,
|
|
|
|
target_format: ImageFormat,
|
|
|
|
source: Alias,
|
|
|
|
process_path: PathBuf,
|
|
|
|
process_args: Vec<String>,
|
|
|
|
) -> Result<(), Error> {
|
2022-04-02 22:40:04 +00:00
|
|
|
let hash = repo.hash(&source).await?;
|
|
|
|
|
|
|
|
let path_string = process_path.to_string_lossy().to_string();
|
|
|
|
let identifier_opt = repo
|
|
|
|
.variant_identifier::<S::Identifier>(hash.clone(), path_string)
|
|
|
|
.await?;
|
|
|
|
|
|
|
|
if identifier_opt.is_some() {
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
|
|
|
|
crate::generate::generate(
|
|
|
|
repo,
|
|
|
|
store,
|
|
|
|
target_format,
|
|
|
|
source,
|
|
|
|
process_path,
|
|
|
|
process_args,
|
|
|
|
hash,
|
|
|
|
)
|
|
|
|
.await?;
|
|
|
|
|
|
|
|
Ok(())
|
2022-04-01 21:51:12 +00:00
|
|
|
}
|