Add two bins using crate apalis-sql.
This commit is contained in:
		
							parent
							
								
									beb4fd40c5
								
							
						
					
					
						commit
						b98050dc7a
					
				
					 4 changed files with 263 additions and 0 deletions
				
			
		
							
								
								
									
										114
									
								
								src/apalis_email_service.rs
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										114
									
								
								src/apalis_email_service.rs
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,114 @@ | |||
| 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:?}") | ||||
|     } | ||||
| } | ||||
							
								
								
									
										64
									
								
								src/bin/using-crate-apalis-postgres.rs
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								src/bin/using-crate-apalis-postgres.rs
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,64 @@ | |||
| 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(()) | ||||
| } | ||||
							
								
								
									
										84
									
								
								src/bin/using-crate-apalis-sqlite.rs
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										84
									
								
								src/bin/using-crate-apalis-sqlite.rs
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,84 @@ | |||
| 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(()) | ||||
| } | ||||
|  | @ -1 +1,2 @@ | |||
| pub mod apalis_email_service; | ||||
| pub mod tcs_helpers; | ||||
|  |  | |||
		Loading…
	
	Add table
		
		Reference in a new issue