mirror of
https://git.asonix.dog/asonix/pict-rs.git
synced 2025-01-04 00:28:43 +00:00
Finish queue implementation update for sled
This commit is contained in:
parent
c48b7bcdfe
commit
5d4486fdf4
13 changed files with 154 additions and 156 deletions
|
@ -1,2 +1,2 @@
|
||||||
[build]
|
[build]
|
||||||
rustflags = ["--cfg", "tokio_unstable"]
|
rustflags = ["--cfg", "tokio_unstable", "--cfg", "uuid_unstable"]
|
||||||
|
|
7
Cargo.lock
generated
7
Cargo.lock
generated
|
@ -320,6 +320,12 @@ dependencies = [
|
||||||
"syn 2.0.28",
|
"syn 2.0.28",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "atomic"
|
||||||
|
version = "0.5.3"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "c59bdb34bc650a32731b31bd8f0829cc15d24a708ee31559e0bb34f2bc320cba"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "autocfg"
|
name = "autocfg"
|
||||||
version = "1.1.0"
|
version = "1.1.0"
|
||||||
|
@ -2969,6 +2975,7 @@ version = "1.4.1"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "79daa5ed5740825c40b389c5e50312b9c86df53fccd33f281df655642b43869d"
|
checksum = "79daa5ed5740825c40b389c5e50312b9c86df53fccd33f281df655642b43869d"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
|
"atomic",
|
||||||
"getrandom",
|
"getrandom",
|
||||||
"serde",
|
"serde",
|
||||||
]
|
]
|
||||||
|
|
|
@ -74,7 +74,7 @@ tracing-subscriber = { version = "0.3.0", features = [
|
||||||
"tracing-log",
|
"tracing-log",
|
||||||
] }
|
] }
|
||||||
url = { version = "2.2", features = ["serde"] }
|
url = { version = "2.2", features = ["serde"] }
|
||||||
uuid = { version = "1", features = ["v4", "serde"] }
|
uuid = { version = "1", features = ["serde", "std", "v4", "v7"] }
|
||||||
|
|
||||||
[dependencies.tracing-actix-web]
|
[dependencies.tracing-actix-web]
|
||||||
version = "0.7.5"
|
version = "0.7.5"
|
||||||
|
|
|
@ -1,6 +1,5 @@
|
||||||
[server]
|
[server]
|
||||||
address = "0.0.0.0:8080"
|
address = "0.0.0.0:8080"
|
||||||
worker_id = "pict-rs-1"
|
|
||||||
read_only = false
|
read_only = false
|
||||||
max_file_count = 1
|
max_file_count = 1
|
||||||
|
|
||||||
|
|
1
dev.toml
1
dev.toml
|
@ -1,6 +1,5 @@
|
||||||
[server]
|
[server]
|
||||||
address = '0.0.0.0:8080'
|
address = '0.0.0.0:8080'
|
||||||
worker_id = 'pict-rs-1'
|
|
||||||
api_key = 'api-key'
|
api_key = 'api-key'
|
||||||
max_file_count = 10
|
max_file_count = 10
|
||||||
|
|
||||||
|
|
|
@ -5,14 +5,6 @@
|
||||||
# default: 0.0.0.0:8080
|
# default: 0.0.0.0:8080
|
||||||
address = '0.0.0.0:8080'
|
address = '0.0.0.0:8080'
|
||||||
|
|
||||||
## Optional: pict-rs worker id
|
|
||||||
# environment variable PICTRS__SERVER__WORKER_ID
|
|
||||||
# default: pict-rs-1
|
|
||||||
#
|
|
||||||
# This is used for the internal job queue. It will have more meaning once a shared metadata
|
|
||||||
# repository (like postgres) can be defined.
|
|
||||||
worker_id = 'pict-rs-1'
|
|
||||||
|
|
||||||
## Optional: shared secret for internal endpoints
|
## Optional: shared secret for internal endpoints
|
||||||
# environment variable: PICTRS__SERVER__API_KEY
|
# environment variable: PICTRS__SERVER__API_KEY
|
||||||
# default: empty
|
# default: empty
|
||||||
|
|
|
@ -47,7 +47,6 @@ impl Args {
|
||||||
Command::Run(Run {
|
Command::Run(Run {
|
||||||
address,
|
address,
|
||||||
api_key,
|
api_key,
|
||||||
worker_id,
|
|
||||||
client_pool_size,
|
client_pool_size,
|
||||||
client_timeout,
|
client_timeout,
|
||||||
metrics_prometheus_address,
|
metrics_prometheus_address,
|
||||||
|
@ -100,7 +99,6 @@ impl Args {
|
||||||
let server = Server {
|
let server = Server {
|
||||||
address,
|
address,
|
||||||
api_key,
|
api_key,
|
||||||
worker_id,
|
|
||||||
read_only,
|
read_only,
|
||||||
max_file_count,
|
max_file_count,
|
||||||
};
|
};
|
||||||
|
@ -386,8 +384,6 @@ struct Server {
|
||||||
#[serde(skip_serializing_if = "Option::is_none")]
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
address: Option<SocketAddr>,
|
address: Option<SocketAddr>,
|
||||||
#[serde(skip_serializing_if = "Option::is_none")]
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
worker_id: Option<String>,
|
|
||||||
#[serde(skip_serializing_if = "Option::is_none")]
|
|
||||||
api_key: Option<String>,
|
api_key: Option<String>,
|
||||||
#[serde(skip_serializing_if = "std::ops::Not::not")]
|
#[serde(skip_serializing_if = "std::ops::Not::not")]
|
||||||
read_only: bool,
|
read_only: bool,
|
||||||
|
@ -763,10 +759,6 @@ struct Run {
|
||||||
#[arg(long)]
|
#[arg(long)]
|
||||||
api_key: Option<String>,
|
api_key: Option<String>,
|
||||||
|
|
||||||
/// ID of this pict-rs node. Doesn't do much yet
|
|
||||||
#[arg(long)]
|
|
||||||
worker_id: Option<String>,
|
|
||||||
|
|
||||||
/// Number of connections the internel HTTP client should maintain in its pool
|
/// Number of connections the internel HTTP client should maintain in its pool
|
||||||
///
|
///
|
||||||
/// This number defaults to 100, and the total number is multiplied by the number of cores
|
/// This number defaults to 100, and the total number is multiplied by the number of cores
|
||||||
|
|
|
@ -21,7 +21,6 @@ pub(crate) struct Defaults {
|
||||||
#[serde(rename_all = "snake_case")]
|
#[serde(rename_all = "snake_case")]
|
||||||
struct ServerDefaults {
|
struct ServerDefaults {
|
||||||
address: SocketAddr,
|
address: SocketAddr,
|
||||||
worker_id: String,
|
|
||||||
read_only: bool,
|
read_only: bool,
|
||||||
max_file_count: u32,
|
max_file_count: u32,
|
||||||
}
|
}
|
||||||
|
@ -183,7 +182,6 @@ impl Default for ServerDefaults {
|
||||||
fn default() -> Self {
|
fn default() -> Self {
|
||||||
ServerDefaults {
|
ServerDefaults {
|
||||||
address: "0.0.0.0:8080".parse().expect("Valid address string"),
|
address: "0.0.0.0:8080".parse().expect("Valid address string"),
|
||||||
worker_id: String::from("pict-rs-1"),
|
|
||||||
read_only: false,
|
read_only: false,
|
||||||
max_file_count: 1,
|
max_file_count: 1,
|
||||||
}
|
}
|
||||||
|
|
|
@ -95,8 +95,6 @@ pub(crate) enum Repo {
|
||||||
pub(crate) struct Server {
|
pub(crate) struct Server {
|
||||||
pub(crate) address: SocketAddr,
|
pub(crate) address: SocketAddr,
|
||||||
|
|
||||||
pub(crate) worker_id: String,
|
|
||||||
|
|
||||||
#[serde(skip_serializing_if = "Option::is_none")]
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
pub(crate) api_key: Option<String>,
|
pub(crate) api_key: Option<String>,
|
||||||
|
|
||||||
|
|
24
src/lib.rs
24
src/lib.rs
|
@ -48,7 +48,6 @@ use std::{
|
||||||
future::ready,
|
future::ready,
|
||||||
path::Path,
|
path::Path,
|
||||||
path::PathBuf,
|
path::PathBuf,
|
||||||
sync::atomic::{AtomicU64, Ordering},
|
|
||||||
time::{Duration, SystemTime},
|
time::{Duration, SystemTime},
|
||||||
};
|
};
|
||||||
use tokio::sync::Semaphore;
|
use tokio::sync::Semaphore;
|
||||||
|
@ -1495,14 +1494,6 @@ fn build_client(config: &Configuration) -> Result<ClientWithMiddleware, Error> {
|
||||||
.build())
|
.build())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn next_worker_id(config: &Configuration) -> String {
|
|
||||||
static WORKER_ID: AtomicU64 = AtomicU64::new(0);
|
|
||||||
|
|
||||||
let next_id = WORKER_ID.fetch_add(1, Ordering::Relaxed);
|
|
||||||
|
|
||||||
format!("{}-{}", config.server.worker_id, next_id)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn configure_endpoints<
|
fn configure_endpoints<
|
||||||
R: FullRepo + 'static,
|
R: FullRepo + 'static,
|
||||||
S: Store + 'static,
|
S: Store + 'static,
|
||||||
|
@ -1638,26 +1629,15 @@ where
|
||||||
R: FullRepo + 'static,
|
R: FullRepo + 'static,
|
||||||
S: Store + 'static,
|
S: Store + 'static,
|
||||||
{
|
{
|
||||||
let worker_id_1 = next_worker_id(&config);
|
|
||||||
let worker_id_2 = next_worker_id(&config);
|
|
||||||
|
|
||||||
tracing::trace_span!(parent: None, "Spawn task").in_scope(|| {
|
tracing::trace_span!(parent: None, "Spawn task").in_scope(|| {
|
||||||
actix_rt::spawn(queue::process_cleanup(
|
actix_rt::spawn(queue::process_cleanup(
|
||||||
repo.clone(),
|
repo.clone(),
|
||||||
store.clone(),
|
store.clone(),
|
||||||
config.clone(),
|
config.clone(),
|
||||||
worker_id_1,
|
|
||||||
))
|
|
||||||
});
|
|
||||||
tracing::trace_span!(parent: None, "Spawn task").in_scope(|| {
|
|
||||||
actix_rt::spawn(queue::process_images(
|
|
||||||
repo,
|
|
||||||
store,
|
|
||||||
process_map,
|
|
||||||
config,
|
|
||||||
worker_id_2,
|
|
||||||
))
|
))
|
||||||
});
|
});
|
||||||
|
tracing::trace_span!(parent: None, "Spawn task")
|
||||||
|
.in_scope(|| actix_rt::spawn(queue::process_images(repo, store, process_map, config)));
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn launch_file_store<R: FullRepo + 'static, F: Fn(&mut web::ServiceConfig) + Send + Clone>(
|
async fn launch_file_store<R: FullRepo + 'static, F: Fn(&mut web::ServiceConfig) + Send + Clone>(
|
||||||
|
|
79
src/queue.rs
79
src/queue.rs
|
@ -187,17 +187,8 @@ pub(crate) async fn process_cleanup<R: FullRepo, S: Store>(
|
||||||
repo: R,
|
repo: R,
|
||||||
store: S,
|
store: S,
|
||||||
config: Configuration,
|
config: Configuration,
|
||||||
worker_id: String,
|
|
||||||
) {
|
) {
|
||||||
process_jobs(
|
process_jobs(&repo, &store, &config, CLEANUP_QUEUE, cleanup::perform).await
|
||||||
&repo,
|
|
||||||
&store,
|
|
||||||
&config,
|
|
||||||
worker_id,
|
|
||||||
CLEANUP_QUEUE,
|
|
||||||
cleanup::perform,
|
|
||||||
)
|
|
||||||
.await
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) async fn process_images<R: FullRepo + 'static, S: Store + 'static>(
|
pub(crate) async fn process_images<R: FullRepo + 'static, S: Store + 'static>(
|
||||||
|
@ -205,14 +196,12 @@ pub(crate) async fn process_images<R: FullRepo + 'static, S: Store + 'static>(
|
||||||
store: S,
|
store: S,
|
||||||
process_map: ProcessMap,
|
process_map: ProcessMap,
|
||||||
config: Configuration,
|
config: Configuration,
|
||||||
worker_id: String,
|
|
||||||
) {
|
) {
|
||||||
process_image_jobs(
|
process_image_jobs(
|
||||||
&repo,
|
&repo,
|
||||||
&store,
|
&store,
|
||||||
&process_map,
|
&process_map,
|
||||||
&config,
|
&config,
|
||||||
worker_id,
|
|
||||||
PROCESS_QUEUE,
|
PROCESS_QUEUE,
|
||||||
process::perform,
|
process::perform,
|
||||||
)
|
)
|
||||||
|
@ -225,7 +214,6 @@ async fn process_jobs<R, S, F>(
|
||||||
repo: &R,
|
repo: &R,
|
||||||
store: &S,
|
store: &S,
|
||||||
config: &Configuration,
|
config: &Configuration,
|
||||||
worker_id: String,
|
|
||||||
queue: &'static str,
|
queue: &'static str,
|
||||||
callback: F,
|
callback: F,
|
||||||
) where
|
) where
|
||||||
|
@ -235,8 +223,10 @@ async fn process_jobs<R, S, F>(
|
||||||
for<'a> F: Fn(&'a R, &'a S, &'a Configuration, &'a [u8]) -> LocalBoxFuture<'a, Result<(), Error>>
|
for<'a> F: Fn(&'a R, &'a S, &'a Configuration, &'a [u8]) -> LocalBoxFuture<'a, Result<(), Error>>
|
||||||
+ Copy,
|
+ Copy,
|
||||||
{
|
{
|
||||||
|
let worker_id = uuid::Uuid::new_v4();
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
let res = job_loop(repo, store, config, worker_id.clone(), queue, callback).await;
|
let res = job_loop(repo, store, config, worker_id, queue, callback).await;
|
||||||
|
|
||||||
if let Err(e) = res {
|
if let Err(e) = res {
|
||||||
tracing::warn!("Error processing jobs: {}", format!("{e}"));
|
tracing::warn!("Error processing jobs: {}", format!("{e}"));
|
||||||
|
@ -249,15 +239,15 @@ async fn process_jobs<R, S, F>(
|
||||||
}
|
}
|
||||||
|
|
||||||
struct MetricsGuard {
|
struct MetricsGuard {
|
||||||
worker_id: String,
|
worker_id: uuid::Uuid,
|
||||||
queue: &'static str,
|
queue: &'static str,
|
||||||
start: Instant,
|
start: Instant,
|
||||||
armed: bool,
|
armed: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl MetricsGuard {
|
impl MetricsGuard {
|
||||||
fn guard(worker_id: String, queue: &'static str) -> Self {
|
fn guard(worker_id: uuid::Uuid, queue: &'static str) -> Self {
|
||||||
metrics::increment_counter!("pict-rs.job.start", "queue" => queue, "worker-id" => worker_id.clone());
|
metrics::increment_counter!("pict-rs.job.start", "queue" => queue, "worker-id" => worker_id.to_string());
|
||||||
|
|
||||||
Self {
|
Self {
|
||||||
worker_id,
|
worker_id,
|
||||||
|
@ -274,8 +264,8 @@ impl MetricsGuard {
|
||||||
|
|
||||||
impl Drop for MetricsGuard {
|
impl Drop for MetricsGuard {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
metrics::histogram!("pict-rs.job.duration", self.start.elapsed().as_secs_f64(), "queue" => self.queue, "worker-id" => self.worker_id.clone(), "completed" => (!self.armed).to_string());
|
metrics::histogram!("pict-rs.job.duration", self.start.elapsed().as_secs_f64(), "queue" => self.queue, "worker-id" => self.worker_id.to_string(), "completed" => (!self.armed).to_string());
|
||||||
metrics::increment_counter!("pict-rs.job.end", "queue" => self.queue, "worker-id" => self.worker_id.clone(), "completed" => (!self.armed).to_string());
|
metrics::increment_counter!("pict-rs.job.end", "queue" => self.queue, "worker-id" => self.worker_id.to_string(), "completed" => (!self.armed).to_string());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -283,7 +273,7 @@ async fn job_loop<R, S, F>(
|
||||||
repo: &R,
|
repo: &R,
|
||||||
store: &S,
|
store: &S,
|
||||||
config: &Configuration,
|
config: &Configuration,
|
||||||
worker_id: String,
|
worker_id: uuid::Uuid,
|
||||||
queue: &'static str,
|
queue: &'static str,
|
||||||
callback: F,
|
callback: F,
|
||||||
) -> Result<(), Error>
|
) -> Result<(), Error>
|
||||||
|
@ -295,21 +285,27 @@ where
|
||||||
+ Copy,
|
+ Copy,
|
||||||
{
|
{
|
||||||
loop {
|
loop {
|
||||||
let (job_id, bytes) = repo.pop(queue, worker_id.as_bytes().to_vec()).await?;
|
let (job_id, bytes) = repo.pop(queue).await?;
|
||||||
|
|
||||||
let span = tracing::info_span!("Running Job", worker_id = ?worker_id);
|
let span = tracing::info_span!("Running Job", worker_id = ?worker_id);
|
||||||
|
|
||||||
let guard = MetricsGuard::guard(worker_id.clone(), queue);
|
let guard = MetricsGuard::guard(worker_id, queue);
|
||||||
|
|
||||||
span.in_scope(|| {
|
let res = span
|
||||||
|
.in_scope(|| {
|
||||||
heartbeat(
|
heartbeat(
|
||||||
repo,
|
repo,
|
||||||
|
queue,
|
||||||
job_id,
|
job_id,
|
||||||
(callback)(repo, store, config, bytes.as_ref()),
|
(callback)(repo, store, config, bytes.as_ref()),
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
.instrument(span)
|
.instrument(span)
|
||||||
.await?;
|
.await;
|
||||||
|
|
||||||
|
repo.complete_job(queue, job_id).await?;
|
||||||
|
|
||||||
|
res?;
|
||||||
|
|
||||||
guard.disarm();
|
guard.disarm();
|
||||||
}
|
}
|
||||||
|
@ -320,7 +316,6 @@ async fn process_image_jobs<R, S, F>(
|
||||||
store: &S,
|
store: &S,
|
||||||
process_map: &ProcessMap,
|
process_map: &ProcessMap,
|
||||||
config: &Configuration,
|
config: &Configuration,
|
||||||
worker_id: String,
|
|
||||||
queue: &'static str,
|
queue: &'static str,
|
||||||
callback: F,
|
callback: F,
|
||||||
) where
|
) where
|
||||||
|
@ -336,17 +331,11 @@ async fn process_image_jobs<R, S, F>(
|
||||||
) -> LocalBoxFuture<'a, Result<(), Error>>
|
) -> LocalBoxFuture<'a, Result<(), Error>>
|
||||||
+ Copy,
|
+ Copy,
|
||||||
{
|
{
|
||||||
|
let worker_id = uuid::Uuid::new_v4();
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
let res = image_job_loop(
|
let res =
|
||||||
repo,
|
image_job_loop(repo, store, process_map, config, worker_id, queue, callback).await;
|
||||||
store,
|
|
||||||
process_map,
|
|
||||||
config,
|
|
||||||
worker_id.clone(),
|
|
||||||
queue,
|
|
||||||
callback,
|
|
||||||
)
|
|
||||||
.await;
|
|
||||||
|
|
||||||
if let Err(e) = res {
|
if let Err(e) = res {
|
||||||
tracing::warn!("Error processing jobs: {}", format!("{e}"));
|
tracing::warn!("Error processing jobs: {}", format!("{e}"));
|
||||||
|
@ -363,7 +352,7 @@ async fn image_job_loop<R, S, F>(
|
||||||
store: &S,
|
store: &S,
|
||||||
process_map: &ProcessMap,
|
process_map: &ProcessMap,
|
||||||
config: &Configuration,
|
config: &Configuration,
|
||||||
worker_id: String,
|
worker_id: uuid::Uuid,
|
||||||
queue: &'static str,
|
queue: &'static str,
|
||||||
callback: F,
|
callback: F,
|
||||||
) -> Result<(), Error>
|
) -> Result<(), Error>
|
||||||
|
@ -381,27 +370,33 @@ where
|
||||||
+ Copy,
|
+ Copy,
|
||||||
{
|
{
|
||||||
loop {
|
loop {
|
||||||
let (job_id, bytes) = repo.pop(queue, worker_id.as_bytes().to_vec()).await?;
|
let (job_id, bytes) = repo.pop(queue).await?;
|
||||||
|
|
||||||
let span = tracing::info_span!("Running Job", worker_id = ?worker_id);
|
let span = tracing::info_span!("Running Job", worker_id = ?worker_id);
|
||||||
|
|
||||||
let guard = MetricsGuard::guard(worker_id.clone(), queue);
|
let guard = MetricsGuard::guard(worker_id, queue);
|
||||||
|
|
||||||
span.in_scope(|| {
|
let res = span
|
||||||
|
.in_scope(|| {
|
||||||
heartbeat(
|
heartbeat(
|
||||||
repo,
|
repo,
|
||||||
|
queue,
|
||||||
job_id,
|
job_id,
|
||||||
(callback)(repo, store, process_map, config, bytes.as_ref()),
|
(callback)(repo, store, process_map, config, bytes.as_ref()),
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
.instrument(span)
|
.instrument(span)
|
||||||
.await?;
|
.await;
|
||||||
|
|
||||||
|
repo.complete_job(queue, job_id).await?;
|
||||||
|
|
||||||
|
res?;
|
||||||
|
|
||||||
guard.disarm();
|
guard.disarm();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn heartbeat<R, Fut>(repo: &R, job_id: JobId, fut: Fut) -> Fut::Output
|
async fn heartbeat<R, Fut>(repo: &R, queue: &'static str, job_id: JobId, fut: Fut) -> Fut::Output
|
||||||
where
|
where
|
||||||
R: QueueRepo,
|
R: QueueRepo,
|
||||||
Fut: std::future::Future,
|
Fut: std::future::Future,
|
||||||
|
@ -419,7 +414,7 @@ where
|
||||||
}
|
}
|
||||||
_ = interval.tick() => {
|
_ = interval.tick() => {
|
||||||
if hb.is_none() {
|
if hb.is_none() {
|
||||||
hb = Some(repo.heartbeat(job_id));
|
hb = Some(repo.heartbeat(queue, job_id));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
opt = poll_opt(hb.as_mut()), if hb.is_some() => {
|
opt = poll_opt(hb.as_mut()), if hb.is_some() => {
|
||||||
|
|
34
src/repo.rs
34
src/repo.rs
|
@ -289,7 +289,7 @@ pub(crate) struct JobId(Uuid);
|
||||||
|
|
||||||
impl JobId {
|
impl JobId {
|
||||||
pub(crate) fn gen() -> Self {
|
pub(crate) fn gen() -> Self {
|
||||||
Self(Uuid::new_v4())
|
Self(Uuid::now_v7())
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) const fn as_bytes(&self) -> &[u8; 16] {
|
pub(crate) const fn as_bytes(&self) -> &[u8; 16] {
|
||||||
|
@ -303,19 +303,13 @@ impl JobId {
|
||||||
|
|
||||||
#[async_trait::async_trait(?Send)]
|
#[async_trait::async_trait(?Send)]
|
||||||
pub(crate) trait QueueRepo: BaseRepo {
|
pub(crate) trait QueueRepo: BaseRepo {
|
||||||
async fn requeue_timed_out(&self, worker_prefix: Vec<u8>) -> Result<(), RepoError>;
|
|
||||||
|
|
||||||
async fn push(&self, queue: &'static str, job: Self::Bytes) -> Result<(), RepoError>;
|
async fn push(&self, queue: &'static str, job: Self::Bytes) -> Result<(), RepoError>;
|
||||||
|
|
||||||
async fn pop(
|
async fn pop(&self, queue: &'static str) -> Result<(JobId, Self::Bytes), RepoError>;
|
||||||
&self,
|
|
||||||
queue: &'static str,
|
|
||||||
worker_id: Vec<u8>,
|
|
||||||
) -> Result<(JobId, Self::Bytes), RepoError>;
|
|
||||||
|
|
||||||
async fn heartbeat(&self, job_id: JobId) -> Result<(), RepoError>;
|
async fn heartbeat(&self, queue: &'static str, job_id: JobId) -> Result<(), RepoError>;
|
||||||
|
|
||||||
async fn complete_job(&self, job_id: JobId) -> Result<(), RepoError>;
|
async fn complete_job(&self, queue: &'static str, job_id: JobId) -> Result<(), RepoError>;
|
||||||
}
|
}
|
||||||
|
|
||||||
#[async_trait::async_trait(?Send)]
|
#[async_trait::async_trait(?Send)]
|
||||||
|
@ -323,28 +317,20 @@ impl<T> QueueRepo for actix_web::web::Data<T>
|
||||||
where
|
where
|
||||||
T: QueueRepo,
|
T: QueueRepo,
|
||||||
{
|
{
|
||||||
async fn requeue_timed_out(&self, worker_prefix: Vec<u8>) -> Result<(), RepoError> {
|
|
||||||
T::requeue_timed_out(self, worker_prefix).await
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn push(&self, queue: &'static str, job: Self::Bytes) -> Result<(), RepoError> {
|
async fn push(&self, queue: &'static str, job: Self::Bytes) -> Result<(), RepoError> {
|
||||||
T::push(self, queue, job).await
|
T::push(self, queue, job).await
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn pop(
|
async fn pop(&self, queue: &'static str) -> Result<(JobId, Self::Bytes), RepoError> {
|
||||||
&self,
|
T::pop(self, queue).await
|
||||||
queue: &'static str,
|
|
||||||
worker_id: Vec<u8>,
|
|
||||||
) -> Result<(JobId, Self::Bytes), RepoError> {
|
|
||||||
T::pop(self, queue, worker_id).await
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn heartbeat(&self, job_id: JobId) -> Result<(), RepoError> {
|
async fn heartbeat(&self, queue: &'static str, job_id: JobId) -> Result<(), RepoError> {
|
||||||
T::heartbeat(self, job_id).await
|
T::heartbeat(self, queue, job_id).await
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn complete_job(&self, job_id: JobId) -> Result<(), RepoError> {
|
async fn complete_job(&self, queue: &'static str, job_id: JobId) -> Result<(), RepoError> {
|
||||||
T::complete_job(self, job_id).await
|
T::complete_job(self, queue, job_id).await
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
110
src/repo/sled.rs
110
src/repo/sled.rs
|
@ -57,6 +57,9 @@ pub(crate) enum SledError {
|
||||||
|
|
||||||
#[error("Operation panicked")]
|
#[error("Operation panicked")]
|
||||||
Panic,
|
Panic,
|
||||||
|
|
||||||
|
#[error("Another process updated this value before us")]
|
||||||
|
Conflict,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
|
@ -129,7 +132,7 @@ impl SledRepo {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn open(mut path: PathBuf, cache_capacity: u64) -> Result<Db, SledError> {
|
fn open(mut path: PathBuf, cache_capacity: u64) -> Result<Db, SledError> {
|
||||||
path.push("v0.4.0-alpha.1");
|
path.push("v0.5.0");
|
||||||
|
|
||||||
let db = ::sled::Config::new()
|
let db = ::sled::Config::new()
|
||||||
.cache_capacity(cache_capacity)
|
.cache_capacity(cache_capacity)
|
||||||
|
@ -626,7 +629,7 @@ impl UploadRepo for SledRepo {
|
||||||
|
|
||||||
enum JobState {
|
enum JobState {
|
||||||
Pending,
|
Pending,
|
||||||
Running(Vec<u8>),
|
Running([u8; 8]),
|
||||||
}
|
}
|
||||||
|
|
||||||
impl JobState {
|
impl JobState {
|
||||||
|
@ -637,35 +640,34 @@ impl JobState {
|
||||||
fn running() -> Self {
|
fn running() -> Self {
|
||||||
Self::Running(
|
Self::Running(
|
||||||
time::OffsetDateTime::now_utc()
|
time::OffsetDateTime::now_utc()
|
||||||
.format(&time::format_description::well_known::Rfc3339)
|
.unix_timestamp()
|
||||||
.expect("Can format")
|
.to_be_bytes(),
|
||||||
.into_bytes(),
|
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn as_bytes(&self) -> &[u8] {
|
fn as_bytes(&self) -> &[u8] {
|
||||||
match self {
|
match self {
|
||||||
Self::Pending => b"pending",
|
Self::Pending => b"pend",
|
||||||
Self::Running(ref bytes) => bytes,
|
Self::Running(ref bytes) => bytes,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn job_key(queue: &'static str, job_id: JobId) -> Arc<[u8]> {
|
||||||
|
let mut key = queue.as_bytes().to_vec();
|
||||||
|
key.extend(job_id.as_bytes());
|
||||||
|
|
||||||
|
Arc::from(key)
|
||||||
|
}
|
||||||
|
|
||||||
#[async_trait::async_trait(?Send)]
|
#[async_trait::async_trait(?Send)]
|
||||||
impl QueueRepo for SledRepo {
|
impl QueueRepo for SledRepo {
|
||||||
#[tracing::instrument(skip_all, fields(worker_id = %String::from_utf8_lossy(&worker_prefix)))]
|
|
||||||
async fn requeue_timed_out(&self, worker_prefix: Vec<u8>) -> Result<(), RepoError> {
|
|
||||||
todo!()
|
|
||||||
}
|
|
||||||
|
|
||||||
#[tracing::instrument(skip(self, job), fields(job = %String::from_utf8_lossy(&job)))]
|
#[tracing::instrument(skip(self, job), fields(job = %String::from_utf8_lossy(&job)))]
|
||||||
async fn push(&self, queue_name: &'static str, job: Self::Bytes) -> Result<(), RepoError> {
|
async fn push(&self, queue_name: &'static str, job: Self::Bytes) -> Result<(), RepoError> {
|
||||||
let metrics_guard = PushMetricsGuard::guard(queue_name);
|
let metrics_guard = PushMetricsGuard::guard(queue_name);
|
||||||
|
|
||||||
let id = JobId::gen();
|
let id = JobId::gen();
|
||||||
let mut key = queue_name.as_bytes().to_vec();
|
let key = job_key(queue_name, id);
|
||||||
key.push(0);
|
|
||||||
key.extend(id.as_bytes());
|
|
||||||
|
|
||||||
let queue = self.queue.clone();
|
let queue = self.queue.clone();
|
||||||
let job_state = self.job_state.clone();
|
let job_state = self.job_state.clone();
|
||||||
|
@ -674,8 +676,8 @@ impl QueueRepo for SledRepo {
|
||||||
(&queue, &job_state).transaction(|(queue, job_state)| {
|
(&queue, &job_state).transaction(|(queue, job_state)| {
|
||||||
let state = JobState::pending();
|
let state = JobState::pending();
|
||||||
|
|
||||||
queue.insert(key.as_slice(), &job)?;
|
queue.insert(&key[..], &job)?;
|
||||||
job_state.insert(key.as_slice(), state.as_bytes())?;
|
job_state.insert(&key[..], state.as_bytes())?;
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
})
|
})
|
||||||
|
@ -705,26 +707,36 @@ impl QueueRepo for SledRepo {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tracing::instrument(skip(self, worker_id), fields(worker_id = %String::from_utf8_lossy(&worker_id)))]
|
#[tracing::instrument(skip(self))]
|
||||||
async fn pop(
|
async fn pop(&self, queue_name: &'static str) -> Result<(JobId, Self::Bytes), RepoError> {
|
||||||
&self,
|
|
||||||
queue_name: &'static str,
|
|
||||||
worker_id: Vec<u8>,
|
|
||||||
) -> Result<(JobId, Self::Bytes), RepoError> {
|
|
||||||
let metrics_guard = PopMetricsGuard::guard(queue_name);
|
let metrics_guard = PopMetricsGuard::guard(queue_name);
|
||||||
|
|
||||||
|
let now = time::OffsetDateTime::now_utc();
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
let queue = self.queue.clone();
|
let queue = self.queue.clone();
|
||||||
let job_state = self.job_state.clone();
|
let job_state = self.job_state.clone();
|
||||||
|
|
||||||
let opt = actix_rt::task::spawn_blocking(move || {
|
let opt = actix_rt::task::spawn_blocking(move || {
|
||||||
|
// Job IDs are generated with Uuid version 7 - defining their first bits as a
|
||||||
|
// timestamp. Scanning a prefix should give us jobs in the order they were queued.
|
||||||
for res in job_state.scan_prefix(queue_name) {
|
for res in job_state.scan_prefix(queue_name) {
|
||||||
let (key, value) = res?;
|
let (key, value) = res?;
|
||||||
|
|
||||||
if value != "pending" {
|
if value.len() == 8 {
|
||||||
// TODO: requeue dead jobs
|
let unix_timestamp =
|
||||||
|
i64::from_be_bytes(value[0..8].try_into().expect("Verified length"));
|
||||||
|
|
||||||
|
let timestamp = time::OffsetDateTime::from_unix_timestamp(unix_timestamp)
|
||||||
|
.expect("Valid timestamp");
|
||||||
|
|
||||||
|
// heartbeats should update every 5 seconds, so 30 seconds without an
|
||||||
|
// update is 6 missed beats
|
||||||
|
if timestamp.saturating_add(time::Duration::seconds(30)) > now {
|
||||||
|
// job hasn't expired
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
let state = JobState::running();
|
let state = JobState::running();
|
||||||
|
|
||||||
|
@ -738,7 +750,7 @@ impl QueueRepo for SledRepo {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let id_bytes = &key[queue_name.len() + 1..];
|
let id_bytes = &key[queue_name.len()..];
|
||||||
|
|
||||||
let id_bytes: [u8; 16] = id_bytes.try_into().expect("Key length");
|
let id_bytes: [u8; 16] = id_bytes.try_into().expect("Key length");
|
||||||
|
|
||||||
|
@ -780,12 +792,52 @@ impl QueueRepo for SledRepo {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn heartbeat(&self, job_id: JobId) -> Result<(), RepoError> {
|
#[tracing::instrument(skip(self))]
|
||||||
todo!()
|
async fn heartbeat(&self, queue_name: &'static str, job_id: JobId) -> Result<(), RepoError> {
|
||||||
|
let key = job_key(queue_name, job_id);
|
||||||
|
|
||||||
|
let job_state = self.job_state.clone();
|
||||||
|
|
||||||
|
actix_rt::task::spawn_blocking(move || {
|
||||||
|
if let Some(state) = job_state.get(&key)? {
|
||||||
|
let new_state = JobState::running();
|
||||||
|
|
||||||
|
match job_state.compare_and_swap(&key, Some(state), Some(new_state.as_bytes()))? {
|
||||||
|
Ok(_) => Ok(()),
|
||||||
|
Err(_) => Err(SledError::Conflict),
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.await
|
||||||
|
.map_err(|_| RepoError::Canceled)??;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn complete_job(&self, job_id: JobId) -> Result<(), RepoError> {
|
#[tracing::instrument(skip(self))]
|
||||||
todo!()
|
async fn complete_job(&self, queue_name: &'static str, job_id: JobId) -> Result<(), RepoError> {
|
||||||
|
let key = job_key(queue_name, job_id);
|
||||||
|
|
||||||
|
let queue = self.queue.clone();
|
||||||
|
let job_state = self.job_state.clone();
|
||||||
|
|
||||||
|
let res = actix_rt::task::spawn_blocking(move || {
|
||||||
|
(&queue, &job_state).transaction(|(queue, job_state)| {
|
||||||
|
queue.remove(&key[..])?;
|
||||||
|
job_state.remove(&key[..])?;
|
||||||
|
Ok(())
|
||||||
|
})
|
||||||
|
})
|
||||||
|
.await
|
||||||
|
.map_err(|_| RepoError::Canceled)?;
|
||||||
|
|
||||||
|
if let Err(TransactionError::Abort(e) | TransactionError::Storage(e)) = res {
|
||||||
|
return Err(RepoError::from(SledError::from(e)));
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue