2020-04-21 00:30:56 +00:00
|
|
|
use crate::{Backoff, JobError, MaxRetries, NewJobInfo};
|
2020-03-21 02:31:03 +00:00
|
|
|
use anyhow::Error;
|
|
|
|
use serde::{de::DeserializeOwned, ser::Serialize};
|
2020-04-21 00:30:56 +00:00
|
|
|
use serde_json::Value;
|
2022-01-17 23:45:24 +00:00
|
|
|
use std::{future::Future, pin::Pin, time::SystemTime};
|
2024-01-08 22:37:32 +00:00
|
|
|
use tracing::{Instrument, Span};
|
2018-11-18 01:39:04 +00:00
|
|
|
|
|
|
|
/// The Job trait defines parameters pertaining to an instance of background job
|
2020-04-21 00:30:56 +00:00
|
|
|
///
|
|
|
|
/// Jobs are defnitions of work to be executed.
|
|
|
|
///
|
|
|
|
/// The simplest implementation defines the job's State and Future types, NAME contant, and
|
|
|
|
/// run method.
|
|
|
|
///
|
|
|
|
/// ### Example
|
|
|
|
///
|
|
|
|
/// ```rust
|
|
|
|
/// use anyhow::Error;
|
|
|
|
/// use background_jobs_core::{Job, new_job};
|
2021-09-16 22:50:32 +00:00
|
|
|
/// use tracing::info;
|
2021-02-04 18:40:28 +00:00
|
|
|
/// use std::future::{ready, Ready};
|
2020-04-21 00:30:56 +00:00
|
|
|
///
|
|
|
|
/// #[derive(serde::Deserialize, serde::Serialize)]
|
|
|
|
/// struct MyJob {
|
|
|
|
/// count: i64,
|
|
|
|
/// }
|
|
|
|
///
|
|
|
|
/// impl Job for MyJob {
|
|
|
|
/// type State = ();
|
|
|
|
/// type Future = Ready<Result<(), Error>>;
|
|
|
|
///
|
|
|
|
/// const NAME: &'static str = "MyJob";
|
|
|
|
///
|
|
|
|
/// fn run(self, _: Self::State) -> Self::Future {
|
|
|
|
/// info!("Processing {}", self.count);
|
|
|
|
///
|
2021-02-04 18:40:28 +00:00
|
|
|
/// ready(Ok(()))
|
2020-04-21 00:30:56 +00:00
|
|
|
/// }
|
|
|
|
/// }
|
|
|
|
///
|
|
|
|
/// fn main() -> Result<(), Error> {
|
|
|
|
/// let job = new_job(MyJob { count: 1234 })?;
|
|
|
|
///
|
|
|
|
/// Ok(())
|
|
|
|
/// }
|
|
|
|
/// ```
|
2019-05-28 00:01:21 +00:00
|
|
|
pub trait Job: Serialize + DeserializeOwned + 'static {
|
|
|
|
/// The application state provided to this job at runtime.
|
|
|
|
type State: Clone + 'static;
|
|
|
|
|
2020-03-21 14:44:38 +00:00
|
|
|
/// The future returned by this job
|
|
|
|
type Future: Future<Output = Result<(), Error>> + Send;
|
|
|
|
|
2020-04-21 00:30:56 +00:00
|
|
|
/// The name of the job
|
|
|
|
///
|
|
|
|
/// This name must be unique!!!
|
|
|
|
const NAME: &'static str;
|
|
|
|
|
|
|
|
/// The name of the default queue for this job
|
|
|
|
///
|
|
|
|
/// This can be overridden on an individual-job level, but if a non-existant queue is supplied,
|
|
|
|
/// the job will never be processed.
|
|
|
|
const QUEUE: &'static str = "default";
|
|
|
|
|
|
|
|
/// Define the default number of retries for this job
|
|
|
|
///
|
|
|
|
/// Defaults to Count(5)
|
|
|
|
/// Jobs can override
|
|
|
|
const MAX_RETRIES: MaxRetries = MaxRetries::Count(5);
|
|
|
|
|
|
|
|
/// Define the default backoff strategy for this job
|
|
|
|
///
|
|
|
|
/// Defaults to Exponential(2)
|
|
|
|
/// Jobs can override
|
|
|
|
const BACKOFF: Backoff = Backoff::Exponential(2);
|
|
|
|
|
2024-01-10 21:06:36 +00:00
|
|
|
/// Define how often a job should update its heartbeat timestamp
|
2020-04-21 00:30:56 +00:00
|
|
|
///
|
|
|
|
/// This is important for allowing the job server to reap processes that were started but never
|
|
|
|
/// completed.
|
|
|
|
///
|
2024-01-10 21:06:36 +00:00
|
|
|
/// Defaults to 5 seconds
|
2020-04-21 00:30:56 +00:00
|
|
|
/// Jobs can override
|
2024-01-10 21:06:36 +00:00
|
|
|
const HEARTBEAT_INTERVAL: u64 = 5_000;
|
2020-04-21 00:30:56 +00:00
|
|
|
|
2018-11-18 01:39:04 +00:00
|
|
|
/// Users of this library must define what it means to run a job.
|
|
|
|
///
|
|
|
|
/// This should contain all the logic needed to complete a job. If that means queuing more
|
|
|
|
/// jobs, sending an email, shelling out (don't shell out), or doing otherwise lengthy
|
|
|
|
/// processes, that logic should all be called from inside this method.
|
2018-11-18 21:05:03 +00:00
|
|
|
///
|
|
|
|
/// The state passed into this job is initialized at the start of the application. The state
|
|
|
|
/// argument could be useful for containing a hook into something like r2d2, or the address of
|
|
|
|
/// an actor in an actix-based system.
|
2020-03-21 14:44:38 +00:00
|
|
|
fn run(self, state: Self::State) -> Self::Future;
|
2018-11-18 01:39:04 +00:00
|
|
|
|
2021-09-21 15:32:48 +00:00
|
|
|
/// Generate a Span that the job will be processed within
|
|
|
|
fn span(&self) -> Option<Span> {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
|
2020-04-21 00:30:56 +00:00
|
|
|
/// If this job should not use it's default queue, this can be overridden in
|
2018-11-18 01:39:04 +00:00
|
|
|
/// user-code.
|
2020-04-21 00:30:56 +00:00
|
|
|
fn queue(&self) -> &str {
|
|
|
|
Self::QUEUE
|
2018-11-18 01:39:04 +00:00
|
|
|
}
|
|
|
|
|
2020-04-21 00:30:56 +00:00
|
|
|
/// If this job should not use it's default maximum retry count, this can be
|
2018-11-18 01:39:04 +00:00
|
|
|
/// overridden in user-code.
|
2020-04-21 00:30:56 +00:00
|
|
|
fn max_retries(&self) -> MaxRetries {
|
|
|
|
Self::MAX_RETRIES
|
2018-11-18 01:39:04 +00:00
|
|
|
}
|
|
|
|
|
2020-04-21 00:30:56 +00:00
|
|
|
/// If this job should not use it's default backoff strategy, this can be
|
2018-11-18 01:39:04 +00:00
|
|
|
/// overridden in user-code.
|
2020-04-21 00:30:56 +00:00
|
|
|
fn backoff_strategy(&self) -> Backoff {
|
|
|
|
Self::BACKOFF
|
2018-11-18 01:39:04 +00:00
|
|
|
}
|
2020-03-21 03:04:23 +00:00
|
|
|
|
2024-01-10 21:06:36 +00:00
|
|
|
/// Define how often a job should update its heartbeat timestamp
|
2020-03-21 03:04:23 +00:00
|
|
|
///
|
|
|
|
/// This is important for allowing the job server to reap processes that were started but never
|
|
|
|
/// completed.
|
2024-01-10 21:06:36 +00:00
|
|
|
fn heartbeat_interval(&self) -> u64 {
|
|
|
|
Self::HEARTBEAT_INTERVAL
|
2020-03-21 03:04:23 +00:00
|
|
|
}
|
2018-11-18 01:39:04 +00:00
|
|
|
}
|
2020-04-21 00:30:56 +00:00
|
|
|
|
|
|
|
/// A provided method to create a new JobInfo from provided arguments
|
|
|
|
pub fn new_job<J>(job: J) -> Result<NewJobInfo, Error>
|
|
|
|
where
|
|
|
|
J: Job,
|
|
|
|
{
|
|
|
|
let job = NewJobInfo::new(
|
|
|
|
J::NAME.to_owned(),
|
|
|
|
job.queue().to_owned(),
|
|
|
|
job.max_retries(),
|
|
|
|
job.backoff_strategy(),
|
2024-01-10 21:06:36 +00:00
|
|
|
job.heartbeat_interval(),
|
2020-04-21 00:30:56 +00:00
|
|
|
serde_json::to_value(job).map_err(|_| ToJson)?,
|
|
|
|
);
|
|
|
|
|
|
|
|
Ok(job)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Create a NewJobInfo to schedule a job to be performed after a certain time
|
2022-01-17 23:45:24 +00:00
|
|
|
pub fn new_scheduled_job<J>(job: J, after: SystemTime) -> Result<NewJobInfo, Error>
|
2020-04-21 00:30:56 +00:00
|
|
|
where
|
|
|
|
J: Job,
|
|
|
|
{
|
|
|
|
let mut job = new_job(job)?;
|
|
|
|
job.schedule(after);
|
|
|
|
|
|
|
|
Ok(job)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// A provided method to coerce arguments into the expected type and run the job
|
|
|
|
pub fn process<J>(
|
|
|
|
args: Value,
|
|
|
|
state: J::State,
|
|
|
|
) -> Pin<Box<dyn Future<Output = Result<(), JobError>> + Send>>
|
|
|
|
where
|
|
|
|
J: Job,
|
|
|
|
{
|
2021-09-21 15:32:48 +00:00
|
|
|
let res = serde_json::from_value::<J>(args).map(move |job| {
|
|
|
|
if let Some(span) = job.span() {
|
|
|
|
let fut = span.in_scope(move || job.run(state));
|
|
|
|
|
|
|
|
(fut, Some(span))
|
|
|
|
} else {
|
|
|
|
(job.run(state), None)
|
|
|
|
}
|
|
|
|
});
|
2020-04-21 00:30:56 +00:00
|
|
|
|
|
|
|
Box::pin(async move {
|
2021-09-21 15:32:48 +00:00
|
|
|
let (fut, span) = res?;
|
|
|
|
|
|
|
|
if let Some(span) = span {
|
|
|
|
fut.instrument(span).await?;
|
|
|
|
} else {
|
|
|
|
fut.await?;
|
|
|
|
}
|
2020-04-21 00:30:56 +00:00
|
|
|
|
|
|
|
Ok(())
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone, Debug, thiserror::Error)]
|
|
|
|
#[error("Failed to to turn job into value")]
|
|
|
|
pub struct ToJson;
|