// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license. mod interface; pub mod local; use std::borrow::Cow; use std::cell::RefCell; use std::rc::Rc; use deno_core::error::get_custom_error_class; use deno_core::error::type_error; use deno_core::error::AnyError; use deno_core::op2; use deno_core::OpState; use deno_core::Resource; use deno_core::ResourceId; pub use crate::interface::*; pub const UNSTABLE_FEATURE_NAME: &str = "cron"; deno_core::extension!(deno_cron, deps = [ deno_console ], parameters = [ C: CronHandler ], ops = [ op_cron_create, op_cron_next, ], esm = [ "01_cron.ts" ], options = { cron_handler: C, }, state = |state, options| { state.put(Rc::new(options.cron_handler)); } ); struct CronResource { handle: Rc, } impl Resource for CronResource { fn name(&self) -> Cow { "cron".into() } fn close(self: Rc) { self.handle.close(); } } #[op2] #[smi] fn op_cron_create( state: Rc>, #[string] name: String, #[string] cron_schedule: String, #[serde] backoff_schedule: Option>, ) -> Result where C: CronHandler + 'static, { let cron_handler = { let state = state.borrow(); // TODO(bartlomieju): replace with `state.feature_checker.check_or_exit` // once we phase out `check_or_exit_with_legacy_fallback` state .feature_checker .check_or_exit_with_legacy_fallback(UNSTABLE_FEATURE_NAME, "Deno.cron"); state.borrow::>().clone() }; validate_cron_name(&name)?; let handle = cron_handler.create(CronSpec { name, cron_schedule, backoff_schedule, })?; let handle_rid = { let mut state = state.borrow_mut(); state.resource_table.add(CronResource { handle: Rc::new(handle), }) }; Ok(handle_rid) } #[op2(async)] async fn op_cron_next( state: Rc>, #[smi] rid: ResourceId, prev_success: bool, ) -> Result where C: CronHandler + 'static, { let cron_handler = { let state = state.borrow(); let resource = match state.resource_table.get::>(rid) { Ok(resource) => resource, Err(err) => { if get_custom_error_class(&err) == Some("BadResource") { return Ok(false); } else { return Err(err); } } }; resource.handle.clone() }; cron_handler.next(prev_success).await } fn validate_cron_name(name: &str) -> Result<(), AnyError> { if name.len() > 64 { return Err(type_error(format!( "Cron name cannot exceed 64 characters: current length {}", name.len() ))); } if !name.chars().all(|c| { c.is_ascii_whitespace() || c.is_ascii_alphanumeric() || c == '_' || c == '-' }) { return Err(type_error("Invalid cron name: only alphanumeric characters, whitespace, hyphens, and underscores are allowed")); } Ok(()) }