use std::error::Error;
use async_trait::async_trait;
use serde_json::Value;
use crate::director::Director;
#[derive(Debug)]
pub enum JobResult {
Accept,
Defer(String),
Reject(String),
Fail(Box<dyn Error + Send + 'static>),
Restart,
Done,
}
impl JobResult {
pub fn accept() -> Self {
Self::Accept
}
pub fn defer<M>(msg: M) -> Self
where
M: Into<String>,
{
Self::Defer(msg.into())
}
pub fn reject<M>(msg: M) -> Self
where
M: Into<String>,
{
Self::Reject(msg.into())
}
pub fn fail<E>(err: E) -> Self
where
E: Into<Box<dyn Error + Send + Sync + 'static>>,
{
Self::Fail(err.into())
}
pub fn restart() -> Self {
Self::Restart
}
pub fn done() -> Self {
Self::Done
}
pub fn combine(self, other: Self) -> Self {
match (self, other) {
(Self::Accept, next) | (next, Self::Accept) => next,
(Self::Done, _) | (_, Self::Done) => Self::Done,
(Self::Restart, _) | (_, Self::Restart) => Self::Restart,
(Self::Defer(left), Self::Defer(right)) => Self::Defer(format!("{}\n{}", left, right)),
(defer @ Self::Defer(_), _) | (_, defer @ Self::Defer(_)) => defer,
(fail @ Self::Fail(_), _) | (_, fail @ Self::Fail(_)) => fail,
(Self::Reject(left), Self::Reject(right)) => {
Self::Reject(format!("{}\n{}", left, right))
},
}
}
}
pub type JobError = Box<dyn Error + Send + Sync>;
pub trait HandlerCore {
fn add_to_director<'a>(&'a self, director: &mut Director<'a>) -> Result<(), JobError>;
}
#[async_trait]
pub trait Handler: HandlerCore {
async fn handle(
&self,
kind: &str,
object: &Value,
retry_count: usize,
) -> Result<JobResult, JobError>;
}
#[cfg(test)]
mod tests {
use crate::JobResult;
#[test]
fn test_job_result_ctors() {
assert!(matches!(JobResult::accept(), JobResult::Accept));
if let JobResult::Defer(d) = JobResult::defer("hi") {
assert_eq!(d, "hi");
} else {
panic!("`JobResult::defer` doesn't return a `JobResult::Defer` variant");
}
if let JobResult::Reject(d) = JobResult::reject("hi") {
assert_eq!(d, "hi");
} else {
panic!("`JobResult::reject` doesn't return a `JobResult::Reject` variant");
}
if let JobResult::Fail(d) = JobResult::fail("hi") {
assert_eq!(format!("{}", d), "hi");
} else {
panic!("`JobResult::fail` doesn't return a `JobResult::Fail` variant");
}
assert!(matches!(JobResult::restart(), JobResult::Restart));
assert!(matches!(JobResult::done(), JobResult::Done));
}
#[test]
fn test_job_result_combine() {
let accept = || JobResult::accept();
let defer = |d| JobResult::defer(d);
let reject = |d| JobResult::reject(d);
let fail = |d| JobResult::fail(d);
let restart = || JobResult::restart();
let done = || JobResult::done();
assert!(matches!(accept().combine(accept()), JobResult::Accept));
if let JobResult::Defer(d) = JobResult::accept().combine(JobResult::defer("defer2")) {
assert_eq!(d, "defer2");
} else {
panic!("`JobResult::Accept` combined with `::Defer` should return a `::Defer`");
}
if let JobResult::Reject(d) = accept().combine(reject("reject2")) {
assert_eq!(d, "reject2");
} else {
panic!("`JobResult::Accept` combined with `::Reject` should return a `::Reject`");
}
if let JobResult::Fail(d) = accept().combine(fail("fail2")) {
assert_eq!(format!("{}", d), "fail2");
} else {
panic!("`JobResult::Accept` combined with `::Fail` should return a `::Fail`");
}
assert!(matches!(accept().combine(restart()), JobResult::Restart));
assert!(matches!(accept().combine(done()), JobResult::Done));
if let JobResult::Defer(d) = defer("defer1").combine(accept()) {
assert_eq!(d, "defer1");
} else {
panic!("`JobResult::Defer` combined with `::Accept` should return a `::Defer`");
}
if let JobResult::Defer(d) = defer("defer1").combine(defer("defer2")) {
assert_eq!(d, "defer1\ndefer2");
} else {
panic!("`JobResult::Defer` combined with `::Defer` should return a `::Defer`");
}
if let JobResult::Defer(d) = defer("defer1").combine(reject("reject2")) {
assert_eq!(d, "defer1");
} else {
panic!("`JobResult::Defer` combined with `::Reject` should return a `::Defer`");
}
if let JobResult::Defer(d) = defer("defer1").combine(fail("fail2")) {
assert_eq!(d, "defer1");
} else {
panic!("`JobResult::Defer` combined with `::Fail` should return a `::Defer`");
}
assert!(matches!(
defer("defer1").combine(restart()),
JobResult::Restart,
));
assert!(matches!(defer("defer1").combine(done()), JobResult::Done));
if let JobResult::Reject(d) = reject("reject1").combine(accept()) {
assert_eq!(d, "reject1");
} else {
panic!("`JobResult::Reject` combined with `::Accept` should return a `::Reject`");
}
if let JobResult::Defer(d) = reject("reject1").combine(defer("defer2")) {
assert_eq!(d, "defer2");
} else {
panic!("`JobResult::Reject` combined with `::Defer` should return a `::Defer`");
}
if let JobResult::Reject(d) = reject("reject1").combine(reject("reject2")) {
assert_eq!(d, "reject1\nreject2");
} else {
panic!("`JobResult::Reject` combined with `::Reject` should return a `::Reject`");
}
if let JobResult::Fail(d) = reject("reject1").combine(fail("fail2")) {
assert_eq!(format!("{}", d), "fail2");
} else {
panic!("`JobResult::Reject` combined with `::Fail` should return a `::Fail`");
}
assert!(matches!(
reject("reject1").combine(restart()),
JobResult::Restart,
));
assert!(matches!(reject("reject1").combine(done()), JobResult::Done));
if let JobResult::Fail(d) = fail("fail1").combine(accept()) {
assert_eq!(format!("{}", d), "fail1");
} else {
panic!("`JobResult::Fail` combined with `::Accept` should return a `::Fail`");
}
if let JobResult::Defer(d) = fail("fail1").combine(defer("defer2")) {
assert_eq!(format!("{}", d), "defer2");
} else {
panic!("`JobResult::Fail` combined with `::Defer` should return a `::Defer`");
}
if let JobResult::Fail(d) = fail("fail1").combine(reject("reject2")) {
assert_eq!(format!("{}", d), "fail1");
} else {
panic!("`JobResult::Fail` combined with `::Reject` should return a `::Fail`");
}
if let JobResult::Fail(d) = fail("fail1").combine(fail("fail2")) {
assert_eq!(format!("{}", d), "fail1");
} else {
panic!("`JobResult::Fail` combined with `::Fail` should return a `::Fail`");
}
assert!(matches!(
fail("fail1").combine(restart()),
JobResult::Restart,
));
assert!(matches!(fail("fail1").combine(done()), JobResult::Done));
assert!(matches!(restart().combine(accept()), JobResult::Restart));
assert!(matches!(
restart().combine(defer("defer2")),
JobResult::Restart,
));
assert!(matches!(
restart().combine(reject("reject2")),
JobResult::Restart,
));
assert!(matches!(
restart().combine(fail("fail2")),
JobResult::Restart,
));
assert!(matches!(restart().combine(restart()), JobResult::Restart));
assert!(matches!(restart().combine(done()), JobResult::Done));
assert!(matches!(done().combine(accept()), JobResult::Done));
assert!(matches!(done().combine(defer("defer2")), JobResult::Done));
assert!(matches!(done().combine(reject("reject2")), JobResult::Done));
assert!(matches!(done().combine(fail("fail2")), JobResult::Done));
assert!(matches!(done().combine(restart()), JobResult::Done));
assert!(matches!(done().combine(done()), JobResult::Done));
}
}