use rhai::{Dynamic, Engine, EvalAltResult, ImmutableString, NativeCallContext, INT}; use std::any::TypeId; #[cfg(not(feature = "no_module"))] #[cfg(not(feature = "unchecked"))] #[test] fn test_native_context() -> Result<(), Box> { let mut engine = Engine::new(); engine.set_max_modules(40); engine.register_fn("test", |context: NativeCallContext, x: INT| { context.engine().max_modules() as INT + x }); assert_eq!(engine.eval::("test(2)")?, 42); Ok(()) } #[test] fn test_native_context_fn_name() -> Result<(), Box> { fn add_double( context: NativeCallContext, args: &mut [&mut Dynamic], ) -> Result> { let x = args[0].as_int().unwrap(); let y = args[1].as_int().unwrap(); Ok(format!("{}_{}", context.fn_name(), x + 2 * y).into()) } let mut engine = Engine::new(); engine .register_raw_fn( "add_double", &[TypeId::of::(), TypeId::of::()], add_double, ) .register_raw_fn( "append_x2", &[TypeId::of::(), TypeId::of::()], add_double, ); assert_eq!(engine.eval::("add_double(40, 1)")?, "add_double_42"); assert_eq!(engine.eval::("append_x2(40, 1)")?, "append_x2_42"); Ok(()) } #[test] fn test_native_overload() -> Result<(), Box> { let mut engine = Engine::new(); assert_eq!( engine.eval::(r#"let x = "hello, "; let y = "world"; x + y"#)?, "hello, world" ); assert_eq!( engine.eval::(r#"let x = "hello"; let y = (); x + y"#)?, "hello" ); // Overload the `+` operator for strings engine .register_fn( "+", |s1: ImmutableString, s2: ImmutableString| -> ImmutableString { format!("{s1}***{s2}").into() }, ) .register_fn("+", |s1: ImmutableString, _: ()| -> ImmutableString { format!("{s1} Foo!").into() }); assert_eq!( engine.eval::(r#"let x = "hello"; let y = "world"; x + y"#)?, "hello***world" ); assert_eq!( engine.eval::(r#"let x = "hello"; let y = (); x + y"#)?, "hello Foo!" ); engine.set_fast_operators(true); assert_eq!( engine.eval::(r#"let x = "hello"; let y = "world"; x + y"#)?, "helloworld" ); assert_eq!( engine.eval::(r#"let x = "hello"; let y = (); x + y"#)?, "hello" ); Ok(()) }