#![cfg(not(feature = "no_function"))] use rhai::{Engine, EvalAltResult, Func, ParseError, ParseErrorType, Scope, INT}; #[test] fn test_fn() -> Result<(), Box> { let engine = Engine::new(); // Expect duplicated parameters error assert!(matches!( engine .compile("fn hello(x, x) { x }") .expect_err("should be error"), ParseError(x, _) if *x == 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 = 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] 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(()) }