Compare commits
	
		
			10 commits
		
	
	
		
			1498ce6c15
			...
			02339e1842
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 02339e1842 | ||
|  | 4e6657a2e1 | ||
|  | 56b48a9219 | ||
|  | 4ab7c2c840 | ||
|  | b98050dc7a | ||
|  | beb4fd40c5 | ||
|  | 87f1fa67cc | ||
|  | 82ec2c819b | ||
|  | 134b34a65f | ||
|  | a6f06db844 | 
					 20 changed files with 2093 additions and 17 deletions
				
			
		
							
								
								
									
										1505
									
								
								Cargo.lock
									
										
									
										generated
									
									
									
								
							
							
						
						
									
										1505
									
								
								Cargo.lock
									
										
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							|  | @ -4,7 +4,10 @@ version = "0.1.0" | ||||||
| edition = "2021" | edition = "2021" | ||||||
| 
 | 
 | ||||||
| [dependencies] | [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 = { 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 = { version = "0.4.39", features = ["rkyv-64", "serde"] } | ||||||
| chrono-tz = { version = "0.10.1", features = ["arbitrary", "case-insensitive", "filter-by-regex", "serde"] } | chrono-tz = { version = "0.10.1", features = ["arbitrary", "case-insensitive", "filter-by-regex", "serde"] } | ||||||
| clokwerk = "0.4.0" | clokwerk = "0.4.0" | ||||||
|  | @ -13,7 +16,12 @@ cron-job = "0.2.0" | ||||||
| cron_tab = { version = "0.2.8", features = ["all"] } | cron_tab = { version = "0.2.8", features = ["all"] } | ||||||
| croner = "2.1.0" | croner = "2.1.0" | ||||||
| delay_timer = { version = "0.11.6", features = ["full"] } | delay_timer = { version = "0.11.6", features = ["full"] } | ||||||
|  | email_address = "0.2.9" | ||||||
| english-to-cron = "0.1.2" | 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 = { 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"] } | 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"] } | tracing = { version = "0.1.41", features = ["async-await", "log", "valuable", "max_level_debug", "release_max_level_debug"] } | ||||||
|  |  | ||||||
							
								
								
									
										143
									
								
								README.md
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										143
									
								
								README.md
									
										
									
									
									
										Normal 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
									
								
							
							
						
						
									
										116
									
								
								src/apalis_email_service.rs
									
										
									
									
									
										Normal 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:?}") | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										66
									
								
								src/bin/using-crate-apalis-postgres.rs
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								src/bin/using-crate-apalis-postgres.rs
									
										
									
									
									
										Normal 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(()) | ||||||
|  | } | ||||||
							
								
								
									
										88
									
								
								src/bin/using-crate-apalis-sqlite.rs
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										88
									
								
								src/bin/using-crate-apalis-sqlite.rs
									
										
									
									
									
										Normal 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(¬ification_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(()) | ||||||
|  | } | ||||||
							
								
								
									
										40
									
								
								src/bin/using-crate-clokwerk.rs
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								src/bin/using-crate-clokwerk.rs
									
										
									
									
									
										Normal 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(); | ||||||
|  | } | ||||||
|  | @ -1,3 +1,5 @@ | ||||||
|  | //! Based on code from <https://github.com/nambrosini/cron-job/blob/e51067cb2395994cb8643204152a1e7dfc161aa5/README.md>
 | ||||||
|  | 
 | ||||||
| use cron_job::{CronJob, Job}; | use cron_job::{CronJob, Job}; | ||||||
| 
 | 
 | ||||||
| fn main() { | fn main() { | ||||||
|  | @ -12,7 +14,7 @@ fn main() { | ||||||
|     // Say hello every second
 |     // Say hello every second
 | ||||||
|     cron.new_job("* * * * * *", hello_job); |     cron.new_job("* * * * * *", hello_job); | ||||||
|     // Start jobs
 |     // Start jobs
 | ||||||
|     cron.start().expect("Failed start jobs."); |     cron.start().expect("Failed to start jobs"); | ||||||
| } | } | ||||||
| // The function to be executed every second.
 | // The function to be executed every second.
 | ||||||
| fn run_every_second() { | fn run_every_second() { | ||||||
|  | @ -1,3 +1,5 @@ | ||||||
|  | //! Code from <https://github.com/zslayton/cron/blob/956beaf3cfe32091dc7a0b371340b59ae5e1a860/README.md>
 | ||||||
|  | 
 | ||||||
| use chrono::Utc; | use chrono::Utc; | ||||||
| use cron::Schedule; | use cron::Schedule; | ||||||
| use std::str::FromStr; | use std::str::FromStr; | ||||||
|  |  | ||||||
|  | @ -1,3 +1,5 @@ | ||||||
|  | //! Based on code from <https://github.com/tuyentv96/rust-crontab/blob/01a266e8e1c7f6ee86b3d1010b8818ffc4db6518/README.md>
 | ||||||
|  | 
 | ||||||
| use std::sync::Arc; | use std::sync::Arc; | ||||||
| 
 | 
 | ||||||
| use chrono::{FixedOffset, Local, TimeZone}; | use chrono::{FixedOffset, Local, TimeZone}; | ||||||
|  |  | ||||||
|  | @ -1,3 +1,5 @@ | ||||||
|  | //! Based on code from <https://github.com/tuyentv96/rust-crontab/blob/01a266e8e1c7f6ee86b3d1010b8818ffc4db6518/README.md>
 | ||||||
|  | 
 | ||||||
| use chrono::{FixedOffset, Local, TimeZone}; | use chrono::{FixedOffset, Local, TimeZone}; | ||||||
| use cron_tab; | use cron_tab; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -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; | use croner::Cron; | ||||||
| 
 | 
 | ||||||
| fn main() { | fn main() { | ||||||
|     // Parse a cron expression to find the next occurrence at 00:00 on Friday
 |     // Parse cron expression for Fridays in December
 | ||||||
|     let cron = Cron::new("0 0 * * FRI") |     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() |         .parse() | ||||||
|         .expect("Successful parsing"); |         .expect("Couldn't parse cron string"); | ||||||
| 
 | 
 | ||||||
|     // Get the next occurrence from the current time, excluding the current time
 |     let time = Local::now(); | ||||||
|     let next = cron.find_next_occurrence(&Utc::now(), false).unwrap(); |  | ||||||
| 
 | 
 | ||||||
|     println!( |     println!("Finding the next 5 New Year's Eves on a Friday:"); | ||||||
|         "Pattern \"{}\" will match next at {}", |     for time in cron.iter_from(time).take(5) { | ||||||
|         cron.pattern.to_string(), |         println!("{}", time); | ||||||
|         next |     } | ||||||
|     ); |  | ||||||
| } | } | ||||||
|  |  | ||||||
							
								
								
									
										50
									
								
								src/bin/using-crate-delay_timer-in-async-context.rs
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								src/bin/using-crate-delay_timer-in-async-context.rs
									
										
									
									
									
										Normal 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) | ||||||
|  | } | ||||||
							
								
								
									
										47
									
								
								src/bin/using-crate-delay_timer-internal.rs
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										47
									
								
								src/bin/using-crate-delay_timer-internal.rs
									
										
									
									
									
										Normal 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) | ||||||
|  | } | ||||||
|  | @ -1,3 +1,5 @@ | ||||||
|  | //! Code from <https://github.com/kaplanelad/english-to-cron/blob/90fe66631e51e9ad7fb631cd55434b7ab8f990f7/examples/run.rs>
 | ||||||
|  | 
 | ||||||
| fn main() { | fn main() { | ||||||
|     let texts = vec![ |     let texts = vec![ | ||||||
|         "every 15 seconds", |         "every 15 seconds", | ||||||
|  |  | ||||||
|  | @ -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 jobs::tcs_helpers::{run_example, stop_example}; | ||||||
| use tokio_cron_scheduler::{ | use tokio_cron_scheduler::{ | ||||||
|     JobScheduler, PostgresMetadataStore, PostgresNotificationStore, SimpleJobCode, |     JobScheduler, PostgresMetadataStore, PostgresNotificationStore, SimpleJobCode, | ||||||
|  |  | ||||||
|  | @ -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 jobs::tcs_helpers::{run_example, stop_example}; | ||||||
| use tokio_cron_scheduler::JobScheduler; | use tokio_cron_scheduler::JobScheduler; | ||||||
| use tracing::Level; | use tracing::Level; | ||||||
|  |  | ||||||
|  | @ -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 jobs::tcs_helpers::{run_example, stop_example}; | ||||||
| use std::error::Error; | use std::error::Error; | ||||||
| use tokio_cron_scheduler::JobScheduler; | use tokio_cron_scheduler::JobScheduler; | ||||||
|  |  | ||||||
|  | @ -1 +1,2 @@ | ||||||
|  | pub mod apalis_email_service; | ||||||
| pub mod tcs_helpers; | pub mod tcs_helpers; | ||||||
|  |  | ||||||
|  | @ -1,3 +1,5 @@ | ||||||
|  | //! Code from <https://github.com/mvniekerk/tokio-cron-scheduler/blob/6c568541022317cc07905ffd25305f0e6e2cfc74/examples/lib.rs>
 | ||||||
|  | 
 | ||||||
| use chrono::Utc; | use chrono::Utc; | ||||||
| use std::time::Duration; | use std::time::Duration; | ||||||
| use tokio_cron_scheduler::{Job, JobBuilder, JobScheduler, JobSchedulerError}; | use tokio_cron_scheduler::{Job, JobBuilder, JobScheduler, JobSchedulerError}; | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		
		Reference in a new issue