#![cfg(not(feature = "no_function"))] use rhai::{ Dynamic, Engine, EvalAltResult, FnPtr, Func, Module, ParseErrorType, RegisterFn, Scope, INT, }; use std::any::TypeId; #[test] fn test_fn() -> Result<(), Box> { let engine = Engine::new(); // Expect duplicated parameters error assert_eq!( *engine .compile("fn hello(x, x) { x }") .expect_err("should be error") .0, ParseErrorType::FnDuplicatedParam("hello".to_string(), "x".to_string()) ); Ok(()) } #[test] fn test_call_fn() -> Result<(), Box> { let engine = Engine::new(); let mut scope = Scope::new(); scope.push("foo", 42 as INT); let ast = engine.compile( r" fn hello(x, y) { x + y } fn hello(x) { x *= foo; foo = 1; x } fn hello() { 41 + foo } ", )?; let r: INT = engine.call_fn(&mut scope, &ast, "hello", (42 as INT, 123 as INT))?; assert_eq!(r, 165); let r: INT = engine.call_fn(&mut scope, &ast, "hello", (123 as INT,))?; assert_eq!(r, 5166); let r: INT = engine.call_fn(&mut scope, &ast, "hello", ())?; assert_eq!(r, 42); assert_eq!( scope .get_value::("foo") .expect("variable foo should exist"), 1 ); Ok(()) } #[test] fn test_call_fn_private() -> Result<(), Box> { let engine = Engine::new(); let mut scope = Scope::new(); let ast = engine.compile("fn add(x, n) { x + n }")?; let r: INT = engine.call_fn(&mut scope, &ast, "add", (40 as INT, 2 as INT))?; assert_eq!(r, 42); let ast = engine.compile("private fn add(x, n, ) { x + n }")?; assert!(matches!( *engine.call_fn::<_, INT>(&mut scope, &ast, "add", (40 as INT, 2 as INT)) .expect_err("should error"), EvalAltResult::ErrorFunctionNotFound(fn_name, _) if fn_name == "add" )); Ok(()) } #[test] #[cfg(not(feature = "no_object"))] fn test_fn_ptr_raw() -> Result<(), Box> { let mut engine = Engine::new(); #[allow(deprecated)] engine .register_fn("mul", |x: &mut INT, y: INT| *x *= y) .register_raw_fn( "bar", &[ TypeId::of::(), TypeId::of::(), TypeId::of::(), ], move |context, args| { let fp = std::mem::take(args[1]).cast::(); let value = args[2].clone(); let this_ptr = args.get_mut(0).unwrap(); fp.call_dynamic(context, Some(this_ptr), [value]) }, ); assert_eq!( engine.eval::( r#" fn foo(x) { this += x; } let x = 41; x.bar(Fn("foo"), 1); x "# )?, 42 ); assert_eq!( engine.eval::( r#" fn foo(x, y) { this += x + y; } let x = 40; let v = 1; x.bar(Fn("foo").curry(v), 1); x "# )?, 42 ); assert!(matches!( *engine.eval::( r#" private fn foo(x) { this += x; } let x = 41; x.bar(Fn("foo"), 1); x "# ).expect_err("should error"), EvalAltResult::ErrorFunctionNotFound(x, _) if x.starts_with("foo (") )); assert_eq!( engine.eval::( r#" let x = 21; x.bar(Fn("mul"), 2); x "# )?, 42 ); Ok(()) } #[test] fn test_anonymous_fn() -> Result<(), Box> { let calc_func = Func::<(INT, INT, INT), INT>::create_from_script( Engine::new(), "fn calc(x, y, z,) { (x + y) * z }", "calc", )?; assert_eq!(calc_func(42, 123, 9)?, 1485); let calc_func = Func::<(INT, String, INT), INT>::create_from_script( Engine::new(), "fn calc(x, y, z) { (x + len(y)) * z }", "calc", )?; assert_eq!(calc_func(42, "hello".to_string(), 9)?, 423); let calc_func = Func::<(INT, INT, INT), INT>::create_from_script( Engine::new(), "private fn calc(x, y, z) { (x + y) * z }", "calc", )?; assert!(matches!( *calc_func(42, 123, 9).expect_err("should error"), EvalAltResult::ErrorFunctionNotFound(fn_name, _) if fn_name == "calc" )); Ok(()) }