2020-03-11 06:28:12 +01:00
|
|
|
#![cfg(not(feature = "no_function"))]
|
2021-01-28 08:29:55 +01:00
|
|
|
use rhai::{Dynamic, Engine, EvalAltResult, FnPtr, Func, FuncArgs, RegisterFn, Scope, INT};
|
|
|
|
use std::{any::TypeId, iter::once};
|
2020-03-04 15:00:01 +01:00
|
|
|
|
|
|
|
#[test]
|
2020-04-21 17:25:12 +02:00
|
|
|
fn test_call_fn() -> Result<(), Box<EvalAltResult>> {
|
2020-04-07 07:23:06 +02:00
|
|
|
let engine = Engine::new();
|
2020-04-05 06:17:31 +02:00
|
|
|
let mut scope = Scope::new();
|
|
|
|
|
|
|
|
scope.push("foo", 42 as INT);
|
2020-03-04 15:00:01 +01:00
|
|
|
|
2020-04-04 16:00:44 +02:00
|
|
|
let ast = engine.compile(
|
2020-03-10 07:09:05 +01:00
|
|
|
r"
|
|
|
|
fn hello(x, y) {
|
2020-03-15 15:39:58 +01:00
|
|
|
x + y
|
2020-03-12 07:54:14 +01:00
|
|
|
}
|
|
|
|
fn hello(x) {
|
2020-07-06 10:20:03 +02:00
|
|
|
x *= foo;
|
2020-04-05 06:57:20 +02:00
|
|
|
foo = 1;
|
|
|
|
x
|
2020-03-10 07:09:05 +01:00
|
|
|
}
|
2020-04-04 16:00:44 +02:00
|
|
|
fn hello() {
|
2020-04-05 06:57:20 +02:00
|
|
|
41 + foo
|
2020-04-04 16:00:44 +02:00
|
|
|
}
|
2020-11-06 06:41:04 +01:00
|
|
|
fn define_var() {
|
|
|
|
let bar = 21;
|
|
|
|
bar * 2
|
|
|
|
}
|
2020-04-04 16:00:44 +02:00
|
|
|
",
|
2020-03-10 07:09:05 +01:00
|
|
|
)?;
|
2020-03-04 15:00:01 +01:00
|
|
|
|
2020-05-13 07:49:01 +02:00
|
|
|
let r: INT = engine.call_fn(&mut scope, &ast, "hello", (42 as INT, 123 as INT))?;
|
2020-03-15 15:39:58 +01:00
|
|
|
assert_eq!(r, 165);
|
2020-03-04 15:00:01 +01:00
|
|
|
|
2020-05-13 07:49:01 +02:00
|
|
|
let r: INT = engine.call_fn(&mut scope, &ast, "hello", (123 as INT,))?;
|
2020-04-05 06:17:31 +02:00
|
|
|
assert_eq!(r, 5166);
|
2020-03-04 15:00:01 +01:00
|
|
|
|
2020-05-13 07:49:01 +02:00
|
|
|
let r: INT = engine.call_fn(&mut scope, &ast, "hello", ())?;
|
2020-04-05 06:57:20 +02:00
|
|
|
assert_eq!(r, 42);
|
|
|
|
|
2020-11-06 06:41:04 +01:00
|
|
|
let r: INT = engine.call_fn(&mut scope, &ast, "define_var", ())?;
|
|
|
|
assert_eq!(r, 42);
|
|
|
|
|
|
|
|
assert!(!scope.contains("bar"));
|
|
|
|
|
2020-04-05 06:57:20 +02:00
|
|
|
assert_eq!(
|
|
|
|
scope
|
|
|
|
.get_value::<INT>("foo")
|
|
|
|
.expect("variable foo should exist"),
|
|
|
|
1
|
|
|
|
);
|
2020-04-04 16:00:44 +02:00
|
|
|
|
2020-03-04 15:00:01 +01:00
|
|
|
Ok(())
|
|
|
|
}
|
2020-04-08 17:01:48 +02:00
|
|
|
|
2021-01-28 08:29:55 +01:00
|
|
|
struct Options {
|
|
|
|
pub foo: bool,
|
|
|
|
pub bar: String,
|
|
|
|
pub baz: INT,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl FuncArgs for Options {
|
|
|
|
fn parse<C: Extend<Dynamic>>(self, container: &mut C) {
|
|
|
|
container.extend(once(self.foo.into()));
|
|
|
|
container.extend(once(self.bar.into()));
|
|
|
|
container.extend(once(self.baz.into()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_call_fn_args() -> Result<(), Box<EvalAltResult>> {
|
|
|
|
let options = Options {
|
|
|
|
foo: false,
|
|
|
|
bar: "world".to_string(),
|
|
|
|
baz: 42,
|
|
|
|
};
|
|
|
|
|
|
|
|
let engine = Engine::new();
|
|
|
|
let mut scope = Scope::new();
|
|
|
|
|
|
|
|
let ast = engine.compile(
|
|
|
|
r#"
|
|
|
|
fn hello(x, y, z) {
|
|
|
|
if x { "hello " + y } else { y + z }
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
)?;
|
|
|
|
|
|
|
|
let result: String = engine.call_fn(&mut scope, &ast, "hello", options)?;
|
|
|
|
|
|
|
|
assert_eq!(result, "world42");
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2020-05-13 07:49:01 +02:00
|
|
|
#[test]
|
|
|
|
fn test_call_fn_private() -> Result<(), Box<EvalAltResult>> {
|
|
|
|
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);
|
|
|
|
|
2020-06-16 16:14:46 +02:00
|
|
|
let ast = engine.compile("private fn add(x, n, ) { x + n }")?;
|
2020-05-13 07:49:01 +02:00
|
|
|
|
|
|
|
assert!(matches!(
|
2020-12-26 06:05:57 +01:00
|
|
|
*(engine.call_fn(&mut scope, &ast, "add", (40 as INT, 2 as INT)) as Result<INT, Box<EvalAltResult>>)
|
2020-05-13 07:49:01 +02:00
|
|
|
.expect_err("should error"),
|
|
|
|
EvalAltResult::ErrorFunctionNotFound(fn_name, _) if fn_name == "add"
|
|
|
|
));
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2020-07-05 17:08:44 +02:00
|
|
|
#[test]
|
2020-07-13 13:38:50 +02:00
|
|
|
#[cfg(not(feature = "no_object"))]
|
2020-07-17 04:18:07 +02:00
|
|
|
fn test_fn_ptr_raw() -> Result<(), Box<EvalAltResult>> {
|
2020-07-05 17:08:44 +02:00
|
|
|
let mut engine = Engine::new();
|
|
|
|
|
2020-08-03 04:07:52 +02:00
|
|
|
#[allow(deprecated)]
|
2020-07-27 06:52:32 +02:00
|
|
|
engine
|
|
|
|
.register_fn("mul", |x: &mut INT, y: INT| *x *= y)
|
|
|
|
.register_raw_fn(
|
|
|
|
"bar",
|
|
|
|
&[
|
|
|
|
TypeId::of::<INT>(),
|
|
|
|
TypeId::of::<FnPtr>(),
|
|
|
|
TypeId::of::<INT>(),
|
|
|
|
],
|
2020-10-18 11:02:17 +02:00
|
|
|
move |context, args| {
|
2020-07-27 06:52:32 +02:00
|
|
|
let fp = std::mem::take(args[1]).cast::<FnPtr>();
|
|
|
|
let value = args[2].clone();
|
|
|
|
let this_ptr = args.get_mut(0).unwrap();
|
|
|
|
|
2020-10-18 11:02:17 +02:00
|
|
|
fp.call_dynamic(context, Some(this_ptr), [value])
|
2020-07-27 06:52:32 +02:00
|
|
|
},
|
|
|
|
);
|
2020-07-05 17:08:44 +02:00
|
|
|
|
2020-07-06 06:04:02 +02:00
|
|
|
assert_eq!(
|
|
|
|
engine.eval::<INT>(
|
|
|
|
r#"
|
|
|
|
fn foo(x) { this += x; }
|
|
|
|
|
|
|
|
let x = 41;
|
2020-07-07 16:59:23 +02:00
|
|
|
x.bar(Fn("foo"), 1);
|
2020-07-06 06:04:02 +02:00
|
|
|
x
|
|
|
|
"#
|
|
|
|
)?,
|
|
|
|
42
|
|
|
|
);
|
2020-07-05 17:08:44 +02:00
|
|
|
|
2020-08-22 16:44:24 +02:00
|
|
|
assert_eq!(
|
|
|
|
engine.eval::<INT>(
|
|
|
|
r#"
|
|
|
|
fn foo(x, y) { this += x + y; }
|
|
|
|
|
|
|
|
let x = 40;
|
|
|
|
let v = 1;
|
|
|
|
x.bar(Fn("foo").curry(v), 1);
|
|
|
|
x
|
|
|
|
"#
|
|
|
|
)?,
|
|
|
|
42
|
|
|
|
);
|
|
|
|
|
2020-07-27 06:52:32 +02:00
|
|
|
assert!(matches!(
|
|
|
|
*engine.eval::<INT>(
|
|
|
|
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::<INT>(
|
|
|
|
r#"
|
|
|
|
let x = 21;
|
|
|
|
x.bar(Fn("mul"), 2);
|
|
|
|
x
|
|
|
|
"#
|
|
|
|
)?,
|
|
|
|
42
|
|
|
|
);
|
|
|
|
|
2020-07-05 17:08:44 +02:00
|
|
|
Ok(())
|
|
|
|
}
|
2020-07-22 23:53:40 +02:00
|
|
|
|
|
|
|
#[test]
|
2020-07-29 16:43:57 +02:00
|
|
|
fn test_anonymous_fn() -> Result<(), Box<EvalAltResult>> {
|
|
|
|
let calc_func = Func::<(INT, INT, INT), INT>::create_from_script(
|
|
|
|
Engine::new(),
|
|
|
|
"fn calc(x, y, z,) { (x + y) * z }",
|
|
|
|
"calc",
|
|
|
|
)?;
|
2020-07-26 04:18:18 +02:00
|
|
|
|
2020-07-29 16:43:57 +02:00
|
|
|
assert_eq!(calc_func(42, 123, 9)?, 1485);
|
2020-07-26 04:18:18 +02:00
|
|
|
|
2020-07-29 16:43:57 +02:00
|
|
|
let calc_func = Func::<(INT, String, INT), INT>::create_from_script(
|
|
|
|
Engine::new(),
|
|
|
|
"fn calc(x, y, z) { (x + len(y)) * z }",
|
|
|
|
"calc",
|
|
|
|
)?;
|
2020-07-26 04:18:18 +02:00
|
|
|
|
2020-07-29 16:43:57 +02:00
|
|
|
assert_eq!(calc_func(42, "hello".to_string(), 9)?, 423);
|
2020-07-26 04:18:18 +02:00
|
|
|
|
2020-07-29 16:43:57 +02:00
|
|
|
let calc_func = Func::<(INT, INT, INT), INT>::create_from_script(
|
|
|
|
Engine::new(),
|
|
|
|
"private fn calc(x, y, z) { (x + y) * z }",
|
|
|
|
"calc",
|
|
|
|
)?;
|
2020-07-26 04:18:18 +02:00
|
|
|
|
2020-07-29 16:43:57 +02:00
|
|
|
assert!(matches!(
|
|
|
|
*calc_func(42, 123, 9).expect_err("should error"),
|
|
|
|
EvalAltResult::ErrorFunctionNotFound(fn_name, _) if fn_name == "calc"
|
|
|
|
));
|
2020-07-26 04:18:18 +02:00
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|