Compare commits

...

10 commits

Author SHA1 Message Date
Erik Nordstrøm
02339e1842 Reword intro sentence slightly. 2025-02-23 01:43:16 +01:00
Erik Nordstrøm
4e6657a2e1 Attribute code samples. Mention in README whether there are additional examples of use available in crates repos or not. 2025-02-23 01:35:54 +01:00
Erik Nordstrøm
56b48a9219 Add additional deps. 2025-02-23 00:06:46 +01:00
Erik Nordstrøm
4ab7c2c840 Add README. 2025-02-23 00:06:43 +01:00
Erik Nordstrøm
b98050dc7a Add two bins using crate apalis-sql. 2025-02-23 00:05:11 +01:00
Erik Nordstrøm
beb4fd40c5 Add another couple of deps. 2025-02-22 23:17:01 +01:00
Erik Nordstrøm
87f1fa67cc Add another couple of deps. 2025-02-22 23:13:02 +01:00
Erik Nordstrøm
82ec2c819b Add two bins using crate delay_timer. 2025-02-22 23:12:53 +01:00
Erik Nordstrøm
134b34a65f Add bin using crate clokwerk. 2025-02-22 22:48:27 +01:00
Erik Nordstrøm
a6f06db844 Recreate Cargo.lock 2025-02-22 22:25:21 +01:00
20 changed files with 2093 additions and 17 deletions

1505
Cargo.lock generated

File diff suppressed because it is too large Load diff

View file

@ -4,7 +4,10 @@ version = "0.1.0"
edition = "2021"
[dependencies]
anyhow = { version = "1.0.96", features = ["backtrace"] }
apalis = { version = "0.6.4", features = ["catch-panic", "docsrs", "document-features", "filter", "layers", "limit", "prometheus", "retry", "sentry", "timeout", "ulid", "uuid"] }
apalis-amqp = "0.4.1"
apalis-sql = { version = "0.6.4", features = ["async-std", "postgres", "sqlite"] }
chrono = { version = "0.4.39", features = ["rkyv-64", "serde"] }
chrono-tz = { version = "0.10.1", features = ["arbitrary", "case-insensitive", "filter-by-regex", "serde"] }
clokwerk = "0.4.0"
@ -13,7 +16,12 @@ cron-job = "0.2.0"
cron_tab = { version = "0.2.8", features = ["all"] }
croner = "2.1.0"
delay_timer = { version = "0.11.6", features = ["full"] }
email_address = "0.2.9"
english-to-cron = "0.1.2"
log = { version = "0.4.26", features = ["kv", "kv_serde", "max_level_debug", "release_max_level_debug", "serde", "std", "sval", "value-bag"] }
serde = { version = "1.0.218", features = ["alloc", "derive", "rc", "serde_derive"] }
smol = "2.0.2"
sqlx = { version = "0.8.3", features = ["runtime-async-std"] }
tokio = { version = "1.43.0", features = ["full", "mio", "test-util", "tracing"] }
tokio-cron-scheduler = { version = "0.13.0", features = ["english", "log", "postgres_storage", "prost", "signal", "tokio-postgres", "tracing-subscriber"] }
tracing = { version = "0.1.41", features = ["async-await", "log", "valuable", "max_level_debug", "release_max_level_debug"] }

143
README.md Normal file
View file

@ -0,0 +1,143 @@
# jobs
Trying out different job scheduling and job parsing crates.
## Apalis
In addition to the below, a lot of other `apalis` examples can be found at
<https://github.com/geofmureithi/apalis/tree/main/examples>.
### PostgreSQL
Create database.
```zsh
createdb foobar
```
Run sample program.
```zsh
DATABASE_URL="postgres://${USER}:passwordifany@localhost/foobar" cargo run --release --bin using-crate-apalis-postgres
```
### SQLite
Run sample program.
```zsh
cargo run --release --bin using-crate-apalis-sqlite
```
## Clokwerk
Run sample program.
```zsh
cargo run --release --bin using-crate-clokwerk
```
The `clokwerk` repo contains only this single example
in its readme and does *not* contain any other
direct examples of use at the time of writing this.
## Cron
Run sample program.
```zsh
cargo run --release --bin using-crate-cron
```
The `cron` repo contains only this single example
in its readme and does *not* contain any other
direct examples of use at the time of writing this.
## Cron-job
Run sample program.
```zsh
cargo run --release --bin using-crate-cron-job
```
The `cron-job` repo contains a couple of other
direct examples of use in the readme, but nothing
beyond that at the time of writing this.
## Cron\_tab
Run sample programs.
```zsh
cargo run --release --bin using-crate-cron_tab-sync
```
```zsh
cargo run --release --bin using-crate-cron_tab-async
```
The `cron_tab` repo contains only these two examples
in its readme, and a copy of the two same sample programs
at <https://github.com/tuyentv96/rust-crontab/tree/master/examples>,
and does *not* contain any other direct examples of use
at the time of writing this.
## Croner
Run sample program.
```zsh
cargo run --release --bin using-crate-croner
```
Additional examples at <https://github.com/Hexagon/croner-rust/tree/main/examples>.
## Delay\_timer
Run sample programs.
```zsh
cargo run --release --bin using-crate-delay_timer-internal
```
```zsh
cargo run --release --bin using-crate-delay_timer-in-async-context
```
Additional examples at: <https://github.com/BinChengZhao/delay-timer/tree/master/examples>
## English-to-cron
Run sample program.
```zsh
cargo run --release --bin using-crate-english-to-cron
```
The `english-to-cron` repo contains only this example
in its readme, and a copy of the same sample program
at <https://github.com/kaplanelad/english-to-cron/tree/main/examples>,
and does *not* contain any other direct examples of use
at the time of writing this.
## Tokio-cron-scheduler
Run sample programs.
```zsh
cargo run --release --bin using-crate-tokio-cron-scheduler-simple_job_tokio_in_a_thread
```
```zsh
cargo run --release --bin using-crate-tokio-cron-scheduler-simple_job
```
```zsh
cargo run --release --bin using-crate-tokio-cron-scheduler-postgres_job
```
One additional example at <https://github.com/mvniekerk/tokio-cron-scheduler/tree/main/examples>
although that one (`nats_job.rs`) is not buildable for me I assume, as enabling the
nats related features did not work for me. Enabling nats features in `Cargo.toml`
for me makes Rust unable to build anything.

116
src/apalis_email_service.rs Normal file
View file

@ -0,0 +1,116 @@
//! Code from <https://github.com/geofmureithi/apalis/blob/060a7d260cc66714afe9ddc20012a569b00103a2/examples/email-service/src/lib.rs>
use std::{str::FromStr, sync::Arc};
use apalis::prelude::*;
use email_address::EmailAddress;
use serde::{Deserialize, Serialize};
#[derive(Debug, Deserialize, Serialize, Clone)]
pub struct Email {
pub to: String,
pub subject: String,
pub text: String,
}
pub async fn send_email(job: Email) -> Result<(), Error> {
let validation = EmailAddress::from_str(&job.to);
match validation {
Ok(email) => {
log::info!("Attempting to send email to {}", email.as_str());
Ok(())
}
Err(email_address::Error::InvalidCharacter) => {
log::error!("Killed send email job. Invalid character {}", job.to);
Err(Error::Abort(Arc::new(Box::new(
email_address::Error::InvalidCharacter,
))))
}
Err(e) => Err(Error::Failed(Arc::new(Box::new(e)))),
}
}
pub fn example_good_email() -> Email {
Email {
subject: "Test Subject".to_string(),
to: "example@gmail.com".to_string(),
text: "Some Text".to_string(),
}
}
pub fn example_killed_email() -> Email {
Email {
subject: "Test Subject".to_string(),
to: "example@©.com".to_string(), // killed because it has © which is invalid
text: "Some Text".to_string(),
}
}
pub fn example_retry_able_email() -> Email {
Email {
subject: "Test Subject".to_string(),
to: "example".to_string(),
text: "Some Text".to_string(),
}
}
pub const FORM_HTML: &str = r#"
<!doctype html>
<html>
<head>
<link href="https://unpkg.com/tailwindcss@1.2.0/dist/tailwind.min.css" rel="stylesheet">
<meta credits="https://tailwindcomponents.com/component/basic-contact-form" />
</head>
<body>
<form style="margin: 0 auto;" class="w-full max-w-lg pt-20" action="/" method="post">
<div class="flex flex-wrap -mx-3 mb-6">
<div class="w-full md:w-2/3 px-3 mb-6 md:mb-0">
<label class="block uppercase tracking-wide text-gray-700 text-xs font-bold mb-2" for="to">
To
</label>
<input class="appearance-none block w-full bg-gray-200 text-gray-700 border border-red-500 rounded py-3 px-4 mb-3 leading-tight focus:outline-none focus:bg-white" id="to" type="email" name="to" placeholder="test@example.com">
<p class="text-red-500 text-xs italic">Please fill out this field.</p>
</div>
</div>
<div class="flex flex-wrap -mx-3 mb-6">
<div class="w-full px-3">
<label class="block uppercase tracking-wide text-gray-700 text-xs font-bold mb-2" for="subject">
Subject
</label>
<input class="appearance-none block w-full bg-gray-200 text-gray-700 border border-gray-200 rounded py-3 px-4 mb-3 leading-tight focus:outline-none focus:bg-white focus:border-gray-500" id="subject" type="text" name="subject">
<p class="text-gray-600 text-xs italic">Some tips - as long as needed</p>
</div>
</div>
<div class="flex flex-wrap -mx-3 mb-6">
<div class="w-full px-3">
<label class="block uppercase tracking-wide text-gray-700 text-xs font-bold mb-2" for="text">
Message
</label>
<textarea class=" no-resize appearance-none block w-full bg-gray-200 text-gray-700 border border-gray-200 rounded py-3 px-4 mb-3 leading-tight focus:outline-none focus:bg-white focus:border-gray-500 h-48 resize-none" id="text" name="text" ></textarea>
</div>
</div>
<div class="md:flex md:items-center">
<div class="md:w-1/3">
<button class="shadow bg-teal-400 hover:bg-teal-400 focus:shadow-outline focus:outline-none text-white font-bold py-2 px-4 rounded" type="submit">
Send
</button>
</div>
<div class="md:w-2/3"></div>
</div>
</form>
</body>
</html>
"#;
#[derive(Debug)]
pub enum EmailError {
NoStorage,
SomeError(&'static str),
}
impl std::fmt::Display for EmailError {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "{self:?}")
}
}

View file

@ -0,0 +1,66 @@
//! Code from <https://github.com/geofmureithi/apalis/blob/060a7d260cc66714afe9ddc20012a569b00103a2/examples/postgres/src/main.rs>
use anyhow::Result;
use apalis::layers::retry::RetryPolicy;
use apalis::prelude::*;
use apalis_sql::{
postgres::{PgListen, PgPool, PostgresStorage},
Config,
};
use jobs::apalis_email_service::{send_email, Email};
use tracing::{debug, info};
async fn produce_jobs(storage: &mut PostgresStorage<Email>) -> Result<()> {
for index in 0..10 {
storage
.push(Email {
to: format!("test{}@example.com", index),
text: "Test background job from apalis".to_string(),
subject: "Background email job".to_string(),
})
.await?;
}
// The sql way
tracing::info!("You can also add jobs via sql query, run this: \n Select apalis.push_job('apalis::Email', json_build_object('subject', 'Test apalis', 'to', 'test1@example.com', 'text', 'Lorem Ipsum'));");
Ok(())
}
#[tokio::main]
async fn main() -> Result<()> {
std::env::set_var("RUST_LOG", "debug,sqlx::query=error");
tracing_subscriber::fmt::init();
let database_url = std::env::var("DATABASE_URL").expect("Must specify path to db");
let pool = PgPool::connect(&database_url).await?;
PostgresStorage::setup(&pool)
.await
.expect("unable to run migrations for postgres");
let mut pg = PostgresStorage::new_with_config(pool.clone(), Config::new("apalis::Email"));
produce_jobs(&mut pg).await?;
let mut listener = PgListen::new(pool).await?;
listener.subscribe_with(&mut pg);
tokio::spawn(async move {
listener.listen().await.unwrap();
});
Monitor::new()
.register({
WorkerBuilder::new("tasty-orange")
.retry(RetryPolicy::retries(5))
.enable_tracing()
.backend(pg)
.build_fn(send_email)
})
.on_event(|e| debug!("{e}"))
.run_with_signal(async {
tokio::signal::ctrl_c().await?;
info!("Shutting down the system");
Ok(())
})
.await?;
Ok(())
}

View file

@ -0,0 +1,88 @@
//! Code from:
//! - <https://github.com/geofmureithi/apalis/blob/060a7d260cc66714afe9ddc20012a569b00103a2/examples/sqlite/src/job.rs>
//! - <https://github.com/geofmureithi/apalis/blob/060a7d260cc66714afe9ddc20012a569b00103a2/examples/sqlite/src/main.rs>
use anyhow::Result;
use apalis::prelude::*;
use apalis_sql::sqlite::SqliteStorage;
use chrono::Utc;
use jobs::apalis_email_service::{send_email, Email};
use serde::{Deserialize, Serialize};
use sqlx::SqlitePool;
#[derive(Debug, Deserialize, Serialize)]
pub struct Notification {
pub to: String,
pub text: String,
}
pub async fn notify(job: Notification) {
tracing::info!("Attempting to send notification to {}", job.to);
}
async fn produce_emails(storage: &SqliteStorage<Email>) -> Result<()> {
let mut storage = storage.clone();
for i in 0..1 {
storage
.schedule(
Email {
to: format!("test{i}@example.com"),
text: "Test background job from apalis".to_string(),
subject: "Background email job".to_string(),
},
(Utc::now() + chrono::Duration::seconds(4)).timestamp(),
)
.await?;
}
Ok(())
}
async fn produce_notifications(storage: &SqliteStorage<Notification>) -> Result<()> {
let mut storage = storage.clone();
for i in 0..20 {
storage
.push(Notification {
to: format!("notify:{i}@example.com"),
text: "Test background job from apalis".to_string(),
})
.await?;
}
Ok(())
}
#[tokio::main]
async fn main() -> Result<()> {
std::env::set_var("RUST_LOG", "debug,sqlx::query=info");
tracing_subscriber::fmt::init();
let pool = SqlitePool::connect("sqlite::memory:").await?;
// Do migrations: Mainly for "sqlite::memory:"
SqliteStorage::setup(&pool)
.await
.expect("unable to run migrations for sqlite");
let email_storage: SqliteStorage<Email> = SqliteStorage::new(pool.clone());
produce_emails(&email_storage).await?;
let notification_storage: SqliteStorage<Notification> = SqliteStorage::new(pool);
produce_notifications(&notification_storage).await?;
Monitor::new()
.register({
WorkerBuilder::new("tasty-banana")
.enable_tracing()
.backend(email_storage)
.build_fn(send_email)
})
.register({
WorkerBuilder::new("tasty-mango")
// .enable_tracing()
.backend(notification_storage)
.build_fn(notify)
})
.run()
.await?;
Ok(())
}

View file

@ -0,0 +1,40 @@
//! Code from <https://github.com/mdsherry/clokwerk/blob/f8180dfe64a98d39a5cded998998a3df8809b92c/README.md>
// Scheduler, and trait for .seconds(), .minutes(), etc.
use clokwerk::{Job, Scheduler, TimeUnits};
// Import week days and WeekDay
use clokwerk::Interval::*;
use std::thread;
use std::time::Duration;
fn main() {
// Create a new scheduler
//let mut scheduler = Scheduler::new();
// or a scheduler with a given timezone
let mut scheduler = Scheduler::with_tz(chrono::Utc);
// Add some tasks to it
scheduler
.every(10.minutes())
.plus(30.seconds())
.run(|| println!("Periodic task"));
scheduler
.every(1.day())
.at("3:20 pm")
.run(|| println!("Daily task"));
scheduler
.every(Tuesday)
.at("14:20:17")
.and_every(Thursday)
.at("15:00")
.run(|| println!("Biweekly task"));
// Manually run the scheduler in an event loop
for _ in 1..10 {
scheduler.run_pending();
thread::sleep(Duration::from_millis(10));
}
// Or run it in a background thread
let thread_handle = scheduler.watch_thread(Duration::from_millis(100));
// The scheduler stops when `thread_handle` is dropped, or `stop` is called
thread_handle.stop();
}

View file

@ -1,3 +1,5 @@
//! Based on code from <https://github.com/nambrosini/cron-job/blob/e51067cb2395994cb8643204152a1e7dfc161aa5/README.md>
use cron_job::{CronJob, Job};
fn main() {
@ -12,7 +14,7 @@ fn main() {
// Say hello every second
cron.new_job("* * * * * *", hello_job);
// Start jobs
cron.start().expect("Failed start jobs.");
cron.start().expect("Failed to start jobs");
}
// The function to be executed every second.
fn run_every_second() {

View file

@ -1,3 +1,5 @@
//! Code from <https://github.com/zslayton/cron/blob/956beaf3cfe32091dc7a0b371340b59ae5e1a860/README.md>
use chrono::Utc;
use cron::Schedule;
use std::str::FromStr;

View file

@ -1,3 +1,5 @@
//! Based on code from <https://github.com/tuyentv96/rust-crontab/blob/01a266e8e1c7f6ee86b3d1010b8818ffc4db6518/README.md>
use std::sync::Arc;
use chrono::{FixedOffset, Local, TimeZone};

View file

@ -1,3 +1,5 @@
//! Based on code from <https://github.com/tuyentv96/rust-crontab/blob/01a266e8e1c7f6ee86b3d1010b8818ffc4db6518/README.md>
use chrono::{FixedOffset, Local, TimeZone};
use cron_tab;

View file

@ -1,18 +1,22 @@
use chrono::Utc;
//! Code from <https://github.com/Hexagon/croner-rust/blob/4da136d3363ff7d99f9bdb211f7c8c852449de6d/README.md>
use chrono::Local;
use croner::Cron;
fn main() {
// Parse a cron expression to find the next occurrence at 00:00 on Friday
let cron = Cron::new("0 0 * * FRI")
// Parse cron expression for Fridays in December
let cron = Cron::new("0 0 0 31 12 FRI")
// Include seconds in pattern
.with_seconds_optional()
// Ensure both day of month and day of week conditions are met
.with_dom_and_dow()
.parse()
.expect("Successful parsing");
.expect("Couldn't parse cron string");
// Get the next occurrence from the current time, excluding the current time
let next = cron.find_next_occurrence(&Utc::now(), false).unwrap();
let time = Local::now();
println!(
"Pattern \"{}\" will match next at {}",
cron.pattern.to_string(),
next
);
println!("Finding the next 5 New Year's Eves on a Friday:");
for time in cron.iter_from(time).take(5) {
println!("{}", time);
}
}

View file

@ -0,0 +1,50 @@
//! Based on code from <https://github.com/BinChengZhao/delay-timer/blob/560c92f9b12c56a6729bc46cdaab52c9557a84e7/README.md>
use delay_timer::prelude::*;
use anyhow::Result;
use smol::Timer;
use std::time::Duration;
#[tokio::main]
async fn main() -> Result<()> {
// In addition to the mixed (smol & tokio) runtime
// You can also share a tokio runtime with delayTimer, please see api `DelayTimerBuilder::tokio_runtime` for details.
// Build an DelayTimer that uses the default configuration of the Smol runtime internally.
let delay_timer = DelayTimerBuilder::default().build();
// Develop a print job that runs in an asynchronous cycle.
let task_instance_chain = delay_timer.insert_task(build_task_async_print()?)?;
// Get the running instance of task 1.
let task_instance = task_instance_chain.next_with_async_wait().await?;
// Cancel running task instances.
task_instance.cancel_with_async_wait().await?;
// Remove task which id is 1.
delay_timer.remove_task(1)?;
// No new tasks are accepted; running tasks are not affected.
Ok(delay_timer.stop_delay_timer()?)
}
fn build_task_async_print() -> Result<Task, TaskError> {
let mut task_builder = TaskBuilder::default();
let body = || async {
println!("create_async_fn_body!");
Timer::after(Duration::from_secs(3)).await;
println!("create_async_fn_body:i'success");
};
task_builder
.set_task_id(1)
.set_frequency_repeated_by_cron_str("@secondly")
.set_maximum_parallel_runnable_num(2)
.spawn_async_routine(body)
}

View file

@ -0,0 +1,47 @@
//! Code from <https://github.com/BinChengZhao/delay-timer/blob/560c92f9b12c56a6729bc46cdaab52c9557a84e7/README.md>
use anyhow::Result;
use delay_timer::prelude::*;
use smol::Timer;
use std::time::Duration;
fn main() -> Result<()> {
// Build an DelayTimer that uses the default configuration of the Smol runtime internally.
let delay_timer = DelayTimerBuilder::default().build();
// Develop a print job that runs in an asynchronous cycle.
// A chain of task instances.
let task_instance_chain = delay_timer.insert_task(build_task_async_print()?)?;
// Get the running instance of task 1.
let task_instance = task_instance_chain.next_with_wait()?;
// Cancel running task instances.
task_instance.cancel_with_wait()?;
// Remove task which id is 1.
delay_timer.remove_task(1)?;
// No new tasks are accepted; running tasks are not affected.
delay_timer.stop_delay_timer()?;
Ok(())
}
fn build_task_async_print() -> Result<Task, TaskError> {
let mut task_builder = TaskBuilder::default();
let body = || async {
println!("create_async_fn_body!");
Timer::after(Duration::from_secs(3)).await;
println!("create_async_fn_body:i'success");
};
task_builder
.set_task_id(1)
.set_frequency_repeated_by_cron_str("@secondly")
.set_maximum_parallel_runnable_num(2)
.spawn_async_routine(body)
}

View file

@ -1,3 +1,5 @@
//! Code from <https://github.com/kaplanelad/english-to-cron/blob/90fe66631e51e9ad7fb631cd55434b7ab8f990f7/examples/run.rs>
fn main() {
let texts = vec![
"every 15 seconds",

View file

@ -1,3 +1,5 @@
//! Code based on <https://github.com/mvniekerk/tokio-cron-scheduler/blob/6c568541022317cc07905ffd25305f0e6e2cfc74/examples/postgres_job.rs>
use jobs::tcs_helpers::{run_example, stop_example};
use tokio_cron_scheduler::{
JobScheduler, PostgresMetadataStore, PostgresNotificationStore, SimpleJobCode,

View file

@ -1,3 +1,5 @@
//! Code based on <https://github.com/mvniekerk/tokio-cron-scheduler/blob/6c568541022317cc07905ffd25305f0e6e2cfc74/examples/simple_job.rs>
use jobs::tcs_helpers::{run_example, stop_example};
use tokio_cron_scheduler::JobScheduler;
use tracing::Level;

View file

@ -1,3 +1,5 @@
//! Code based on <https://github.com/mvniekerk/tokio-cron-scheduler/blob/6c568541022317cc07905ffd25305f0e6e2cfc74/examples/simple_job_tokio_in_a_thread.rs>
use jobs::tcs_helpers::{run_example, stop_example};
use std::error::Error;
use tokio_cron_scheduler::JobScheduler;

View file

@ -1 +1,2 @@
pub mod apalis_email_service;
pub mod tcs_helpers;

View file

@ -1,3 +1,5 @@
//! Code from <https://github.com/mvniekerk/tokio-cron-scheduler/blob/6c568541022317cc07905ffd25305f0e6e2cfc74/examples/lib.rs>
use chrono::Utc;
use std::time::Duration;
use tokio_cron_scheduler::{Job, JobBuilder, JobScheduler, JobSchedulerError};