mirror of
https://git.asonix.dog/asonix/pict-rs
synced 2024-12-31 23:11:26 +00:00
Add metrics & tracing to dropped uploads, turn pop into a single query
This commit is contained in:
parent
d3c663ccd0
commit
858899b943
3 changed files with 69 additions and 39 deletions
|
@ -1,4 +1,5 @@
|
||||||
use reqwest_middleware::ClientWithMiddleware;
|
use reqwest_middleware::ClientWithMiddleware;
|
||||||
|
use time::Instant;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
concurrent_processor::ProcessMap,
|
concurrent_processor::ProcessMap,
|
||||||
|
@ -73,6 +74,40 @@ where
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct UploadGuard {
|
||||||
|
armed: bool,
|
||||||
|
start: Instant,
|
||||||
|
upload_id: UploadId,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl UploadGuard {
|
||||||
|
fn guard(upload_id: UploadId) -> Self {
|
||||||
|
Self {
|
||||||
|
armed: true,
|
||||||
|
start: Instant::now(),
|
||||||
|
upload_id,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn disarm(mut self) {
|
||||||
|
self.armed = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Drop for UploadGuard {
|
||||||
|
fn drop(&mut self) {
|
||||||
|
metrics::increment_counter!("pict-rs.background.upload.ingest", "completed" => (!self.armed).to_string());
|
||||||
|
metrics::histogram!("pict-rs.background.upload.ingest.duration", self.start.elapsed().as_seconds_f64(), "completed" => (!self.armed).to_string());
|
||||||
|
|
||||||
|
if self.armed {
|
||||||
|
tracing::warn!(
|
||||||
|
"Upload future for {} dropped before completion! This can cause clients to wait forever",
|
||||||
|
self.upload_id,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[tracing::instrument(skip(repo, store, client, media))]
|
#[tracing::instrument(skip(repo, store, client, media))]
|
||||||
async fn process_ingest<S>(
|
async fn process_ingest<S>(
|
||||||
repo: &ArcRepo,
|
repo: &ArcRepo,
|
||||||
|
@ -86,6 +121,8 @@ async fn process_ingest<S>(
|
||||||
where
|
where
|
||||||
S: Store + 'static,
|
S: Store + 'static,
|
||||||
{
|
{
|
||||||
|
let guard = UploadGuard::guard(upload_id);
|
||||||
|
|
||||||
let fut = async {
|
let fut = async {
|
||||||
let ident = unprocessed_identifier.clone();
|
let ident = unprocessed_identifier.clone();
|
||||||
let store2 = store.clone();
|
let store2 = store.clone();
|
||||||
|
@ -130,6 +167,8 @@ where
|
||||||
|
|
||||||
repo.complete_upload(upload_id, result).await?;
|
repo.complete_upload(upload_id, result).await?;
|
||||||
|
|
||||||
|
guard.disarm();
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -862,7 +862,7 @@ where
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Repo {
|
impl Repo {
|
||||||
#[tracing::instrument]
|
#[tracing::instrument(skip(config))]
|
||||||
pub(crate) async fn open(config: config::Repo) -> color_eyre::Result<Self> {
|
pub(crate) async fn open(config: config::Repo) -> color_eyre::Result<Self> {
|
||||||
match config {
|
match config {
|
||||||
config::Repo::Sled(config::Sled {
|
config::Repo::Sled(config::Sled {
|
||||||
|
|
|
@ -1169,46 +1169,37 @@ impl QueueRepo for PostgresRepo {
|
||||||
tracing::info!("Reset {count} jobs");
|
tracing::info!("Reset {count} jobs");
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: combine into 1 query
|
let queue_alias = diesel::alias!(schema::job_queue as queue_alias);
|
||||||
let opt = loop {
|
|
||||||
let id_opt = job_queue
|
|
||||||
.select(id)
|
|
||||||
.filter(status.eq(JobStatus::New).and(queue.eq(queue_name)))
|
|
||||||
.order(queue_time)
|
|
||||||
.limit(1)
|
|
||||||
.get_result::<Uuid>(&mut conn)
|
|
||||||
.with_metrics("pict-rs.postgres.queue.select")
|
|
||||||
.with_timeout(Duration::from_secs(5))
|
|
||||||
.await
|
|
||||||
.map_err(|_| PostgresError::DbTimeout)?
|
|
||||||
.optional()
|
|
||||||
.map_err(PostgresError::Diesel)?;
|
|
||||||
|
|
||||||
let Some(id_val) = id_opt else {
|
let id_query = queue_alias
|
||||||
break None;
|
.select(queue_alias.field(id))
|
||||||
};
|
.filter(
|
||||||
|
queue_alias
|
||||||
|
.field(status)
|
||||||
|
.eq(JobStatus::New)
|
||||||
|
.and(queue_alias.field(queue).eq(queue_name)),
|
||||||
|
)
|
||||||
|
.order(queue_alias.field(queue_time))
|
||||||
|
.for_update()
|
||||||
|
.skip_locked()
|
||||||
|
.single_value();
|
||||||
|
|
||||||
let opt = diesel::update(job_queue)
|
let opt = diesel::update(job_queue)
|
||||||
.filter(id.eq(id_val))
|
.filter(id.nullable().eq(id_query))
|
||||||
.filter(status.eq(JobStatus::New))
|
.filter(status.eq(JobStatus::New))
|
||||||
.set((
|
.set((
|
||||||
heartbeat.eq(timestamp),
|
heartbeat.eq(timestamp),
|
||||||
status.eq(JobStatus::Running),
|
status.eq(JobStatus::Running),
|
||||||
worker.eq(worker_id),
|
worker.eq(worker_id),
|
||||||
))
|
))
|
||||||
.returning((id, job))
|
.returning((id, job))
|
||||||
.get_result(&mut conn)
|
.get_result(&mut conn)
|
||||||
.with_metrics("pict-rs.postgres.queue.claim")
|
.with_metrics("pict-rs.postgres.queue.claim")
|
||||||
.with_timeout(Duration::from_secs(5))
|
.with_timeout(Duration::from_secs(5))
|
||||||
.await
|
.await
|
||||||
.map_err(|_| PostgresError::DbTimeout)?
|
.map_err(|_| PostgresError::DbTimeout)?
|
||||||
.optional()
|
.optional()
|
||||||
.map_err(PostgresError::Diesel)?;
|
.map_err(PostgresError::Diesel)?;
|
||||||
|
|
||||||
if let Some(tup) = opt {
|
|
||||||
break Some(tup);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
if let Some((job_id, job_json)) = opt {
|
if let Some((job_id, job_json)) = opt {
|
||||||
guard.disarm();
|
guard.disarm();
|
||||||
|
|
Loading…
Reference in a new issue