2020-05-05 09:00:10 +02:00
|
|
|
#![cfg(not(feature = "no_module"))]
|
2020-06-12 12:04:16 +02:00
|
|
|
use rhai::{
|
2021-01-08 17:24:55 +01:00
|
|
|
module_resolvers::{DummyModuleResolver, StaticModuleResolver},
|
2022-08-26 10:20:23 +02:00
|
|
|
Dynamic, Engine, EvalAltResult, FnNamespace, FnPtr, ImmutableString, Module, NativeCallContext,
|
|
|
|
ParseError, ParseErrorType, Scope, Shared, INT,
|
2020-06-12 12:04:16 +02:00
|
|
|
};
|
2020-05-05 09:00:10 +02:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_module() {
|
|
|
|
let mut module = Module::new();
|
2020-05-05 11:51:40 +02:00
|
|
|
module.set_var("answer", 42 as INT);
|
2020-05-05 09:00:10 +02:00
|
|
|
|
2020-05-05 11:51:40 +02:00
|
|
|
assert!(module.contains_var("answer"));
|
|
|
|
assert_eq!(module.get_var_value::<INT>("answer").unwrap(), 42);
|
2020-05-05 09:00:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2020-05-05 17:57:25 +02:00
|
|
|
fn test_module_sub_module() -> Result<(), Box<EvalAltResult>> {
|
2020-05-05 09:00:10 +02:00
|
|
|
let mut module = Module::new();
|
|
|
|
|
|
|
|
let mut sub_module = Module::new();
|
|
|
|
|
|
|
|
let mut sub_module2 = Module::new();
|
2020-05-05 11:51:40 +02:00
|
|
|
sub_module2.set_var("answer", 41 as INT);
|
2020-06-15 15:49:02 +02:00
|
|
|
|
2021-03-15 05:39:06 +01:00
|
|
|
let hash_inc = sub_module2.set_native_fn("inc", |x: &mut INT| Ok(*x + 1));
|
2021-03-05 13:07:35 +01:00
|
|
|
sub_module2.build_index();
|
|
|
|
assert!(!sub_module2.contains_indexed_global_functions());
|
2021-03-03 04:40:27 +01:00
|
|
|
|
2021-03-15 05:39:06 +01:00
|
|
|
let super_hash = sub_module2.set_native_fn("super_inc", |x: &mut INT| Ok(*x + 1));
|
|
|
|
sub_module2.update_fn_namespace(super_hash, FnNamespace::Global);
|
2021-03-05 13:07:35 +01:00
|
|
|
sub_module2.build_index();
|
|
|
|
assert!(sub_module2.contains_indexed_global_functions());
|
2020-11-17 08:11:58 +01:00
|
|
|
|
|
|
|
#[cfg(not(feature = "no_object"))]
|
|
|
|
sub_module2.set_getter_fn("doubled", |x: &mut INT| Ok(*x * 2));
|
2020-05-05 09:00:10 +02:00
|
|
|
|
2020-05-05 11:51:40 +02:00
|
|
|
sub_module.set_sub_module("universe", sub_module2);
|
|
|
|
module.set_sub_module("life", sub_module);
|
2020-07-17 08:50:23 +02:00
|
|
|
module.set_var("MYSTIC_NUMBER", Dynamic::from(42 as INT));
|
2021-03-05 13:07:35 +01:00
|
|
|
module.build_index();
|
2020-05-05 09:00:10 +02:00
|
|
|
|
2021-03-05 13:07:35 +01:00
|
|
|
assert!(module.contains_indexed_global_functions());
|
2021-03-03 04:40:27 +01:00
|
|
|
|
2020-05-05 11:51:40 +02:00
|
|
|
assert!(module.contains_sub_module("life"));
|
|
|
|
let m = module.get_sub_module("life").unwrap();
|
2020-05-05 09:00:10 +02:00
|
|
|
|
2020-05-05 11:51:40 +02:00
|
|
|
assert!(m.contains_sub_module("universe"));
|
|
|
|
let m2 = m.get_sub_module("universe").unwrap();
|
2020-05-05 09:00:10 +02:00
|
|
|
|
2020-05-05 11:51:40 +02:00
|
|
|
assert!(m2.contains_var("answer"));
|
2021-03-17 02:58:08 +01:00
|
|
|
assert!(m2.contains_fn(hash_inc));
|
2020-05-05 11:51:40 +02:00
|
|
|
|
|
|
|
assert_eq!(m2.get_var_value::<INT>("answer").unwrap(), 41);
|
|
|
|
|
2022-03-19 02:43:18 +01:00
|
|
|
module.set_custom_type::<()>("Don't Panic");
|
|
|
|
|
2020-06-27 17:56:24 +02:00
|
|
|
let mut engine = Engine::new();
|
2020-12-23 03:08:43 +01:00
|
|
|
engine.register_static_module("question", module.into());
|
2020-05-05 11:51:40 +02:00
|
|
|
|
2022-03-19 02:43:18 +01:00
|
|
|
assert_eq!(engine.eval::<String>("type_of(())")?, "Don't Panic");
|
|
|
|
|
2020-11-16 07:07:48 +01:00
|
|
|
assert_eq!(engine.eval::<INT>("question::MYSTIC_NUMBER")?, 42);
|
|
|
|
assert!(engine.eval::<INT>("MYSTIC_NUMBER").is_err());
|
2020-12-21 10:39:37 +01:00
|
|
|
assert_eq!(engine.eval::<INT>("question::life::universe::answer")?, 41);
|
2020-05-05 11:51:40 +02:00
|
|
|
assert_eq!(
|
2020-11-16 07:07:48 +01:00
|
|
|
engine.eval::<INT>("question::life::universe::answer + 1")?,
|
2020-07-17 08:50:23 +02:00
|
|
|
42
|
|
|
|
);
|
|
|
|
assert_eq!(
|
2020-11-16 07:07:48 +01:00
|
|
|
engine.eval::<INT>("question::life::universe::inc(question::life::universe::answer)")?,
|
2020-05-05 11:51:40 +02:00
|
|
|
42
|
|
|
|
);
|
2020-11-16 07:07:48 +01:00
|
|
|
assert!(engine
|
|
|
|
.eval::<INT>("inc(question::life::universe::answer)")
|
|
|
|
.is_err());
|
2020-11-17 08:11:58 +01:00
|
|
|
#[cfg(not(feature = "no_object"))]
|
2020-12-21 10:39:37 +01:00
|
|
|
assert_eq!(engine.eval::<INT>("question::MYSTIC_NUMBER.doubled")?, 84);
|
|
|
|
#[cfg(not(feature = "no_object"))]
|
2020-11-17 08:11:58 +01:00
|
|
|
assert_eq!(
|
|
|
|
engine.eval::<INT>("question::life::universe::answer.doubled")?,
|
|
|
|
82
|
|
|
|
);
|
2020-05-05 11:51:40 +02:00
|
|
|
assert_eq!(
|
2020-11-16 07:07:48 +01:00
|
|
|
engine.eval::<INT>("super_inc(question::life::universe::answer)")?,
|
2020-05-05 11:51:40 +02:00
|
|
|
42
|
|
|
|
);
|
|
|
|
|
|
|
|
Ok(())
|
2020-05-05 09:00:10 +02:00
|
|
|
}
|
2020-05-05 17:57:25 +02:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_module_resolver() -> Result<(), Box<EvalAltResult>> {
|
2020-06-27 17:56:24 +02:00
|
|
|
let mut resolver = StaticModuleResolver::new();
|
2020-05-05 17:57:25 +02:00
|
|
|
|
|
|
|
let mut module = Module::new();
|
2020-07-12 05:46:53 +02:00
|
|
|
|
2020-05-05 17:57:25 +02:00
|
|
|
module.set_var("answer", 42 as INT);
|
2021-03-15 05:39:06 +01:00
|
|
|
module.set_native_fn("sum", |x: INT, y: INT, z: INT, w: INT| Ok(x + y + z + w));
|
|
|
|
let double_hash = module.set_native_fn("double", |x: &mut INT| {
|
2020-07-08 07:06:00 +02:00
|
|
|
*x *= 2;
|
|
|
|
Ok(())
|
|
|
|
});
|
2021-03-15 05:39:06 +01:00
|
|
|
module.update_fn_namespace(double_hash, FnNamespace::Global);
|
2020-05-05 17:57:25 +02:00
|
|
|
|
2020-09-15 05:03:46 +02:00
|
|
|
#[cfg(not(feature = "no_float"))]
|
2021-03-15 05:39:06 +01:00
|
|
|
module.set_native_fn(
|
2020-11-17 08:11:58 +01:00
|
|
|
"sum_of_three_args",
|
2020-11-01 09:02:10 +01:00
|
|
|
|target: &mut INT, a: INT, b: INT, c: rhai::FLOAT| {
|
2020-09-15 05:03:46 +02:00
|
|
|
*target = a + b + c as INT;
|
|
|
|
Ok(())
|
2020-09-19 12:12:23 +02:00
|
|
|
},
|
2020-09-15 05:03:46 +02:00
|
|
|
);
|
|
|
|
|
2020-06-27 17:56:24 +02:00
|
|
|
resolver.insert("hello", module);
|
2020-05-05 17:57:25 +02:00
|
|
|
|
|
|
|
let mut engine = Engine::new();
|
2020-12-26 06:05:57 +01:00
|
|
|
engine.set_module_resolver(resolver);
|
2020-05-05 17:57:25 +02:00
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
engine.eval::<INT>(
|
|
|
|
r#"
|
2020-05-15 15:40:54 +02:00
|
|
|
import "hello" as h1;
|
|
|
|
import "hello" as h2;
|
2020-05-28 08:08:21 +02:00
|
|
|
h1::sum(h2::answer, -10, 3, 7)
|
2020-06-27 15:19:53 +02:00
|
|
|
"#
|
2020-05-05 17:57:25 +02:00
|
|
|
)?,
|
|
|
|
42
|
|
|
|
);
|
|
|
|
|
2020-11-17 08:11:58 +01:00
|
|
|
assert!(engine
|
|
|
|
.eval::<INT>(
|
|
|
|
r#"
|
|
|
|
import "hello" as h;
|
|
|
|
sum(h::answer, -10, 3, 7)
|
|
|
|
"#
|
|
|
|
)
|
|
|
|
.is_err());
|
|
|
|
|
2020-08-20 10:26:10 +02:00
|
|
|
assert_eq!(
|
|
|
|
engine.eval::<INT>(
|
|
|
|
r#"
|
|
|
|
import "hello" as h1;
|
|
|
|
import "hello" as h2;
|
|
|
|
let x = 42;
|
|
|
|
h1::sum(x, -10, 3, 7)
|
|
|
|
"#
|
|
|
|
)?,
|
|
|
|
42
|
|
|
|
);
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
engine.eval::<INT>(
|
|
|
|
r#"
|
|
|
|
import "hello" as h1;
|
|
|
|
import "hello" as h2;
|
|
|
|
let x = 42;
|
|
|
|
h1::sum(x, 0, 0, 0);
|
|
|
|
x
|
|
|
|
"#
|
|
|
|
)?,
|
|
|
|
42
|
|
|
|
);
|
|
|
|
|
2020-07-08 07:06:00 +02:00
|
|
|
assert_eq!(
|
|
|
|
engine.eval::<INT>(
|
|
|
|
r#"
|
|
|
|
import "hello" as h;
|
|
|
|
let x = 21;
|
|
|
|
h::double(x);
|
|
|
|
x
|
|
|
|
"#
|
|
|
|
)?,
|
|
|
|
42
|
|
|
|
);
|
2020-11-17 08:11:58 +01:00
|
|
|
assert_eq!(
|
|
|
|
engine.eval::<INT>(
|
|
|
|
r#"
|
|
|
|
import "hello" as h;
|
|
|
|
let x = 21;
|
|
|
|
double(x);
|
|
|
|
x
|
|
|
|
"#
|
|
|
|
)?,
|
|
|
|
42
|
|
|
|
);
|
2020-09-15 05:03:46 +02:00
|
|
|
#[cfg(not(feature = "no_float"))]
|
|
|
|
{
|
|
|
|
assert_eq!(
|
|
|
|
engine.eval::<INT>(
|
|
|
|
r#"
|
2021-01-09 09:52:22 +01:00
|
|
|
import "hello" as h;
|
|
|
|
let x = 21;
|
|
|
|
h::sum_of_three_args(x, 14, 26, 2.0);
|
|
|
|
x
|
|
|
|
"#
|
2020-09-15 05:03:46 +02:00
|
|
|
)?,
|
|
|
|
42
|
|
|
|
);
|
|
|
|
}
|
2020-07-08 07:06:00 +02:00
|
|
|
|
2020-05-23 18:29:06 +02:00
|
|
|
#[cfg(not(feature = "unchecked"))]
|
|
|
|
{
|
|
|
|
engine.set_max_modules(5);
|
|
|
|
|
|
|
|
assert!(matches!(
|
|
|
|
*engine
|
|
|
|
.eval::<INT>(
|
|
|
|
r#"
|
|
|
|
let sum = 0;
|
|
|
|
|
2021-12-15 05:06:17 +01:00
|
|
|
for x in 0..10 {
|
2020-05-23 18:29:06 +02:00
|
|
|
import "hello" as h;
|
|
|
|
sum += h::answer;
|
|
|
|
}
|
|
|
|
|
|
|
|
sum
|
2020-06-27 15:19:53 +02:00
|
|
|
"#
|
2020-05-23 18:29:06 +02:00
|
|
|
)
|
|
|
|
.expect_err("should error"),
|
2022-02-08 02:46:14 +01:00
|
|
|
EvalAltResult::ErrorTooManyModules(..)
|
2020-05-23 18:29:06 +02:00
|
|
|
));
|
|
|
|
|
|
|
|
#[cfg(not(feature = "no_function"))]
|
|
|
|
assert!(matches!(
|
|
|
|
*engine
|
|
|
|
.eval::<INT>(
|
|
|
|
r#"
|
|
|
|
let sum = 0;
|
|
|
|
|
|
|
|
fn foo() {
|
|
|
|
import "hello" as h;
|
|
|
|
sum += h::answer;
|
|
|
|
}
|
|
|
|
|
2021-12-15 05:06:17 +01:00
|
|
|
for x in 0..10 {
|
2020-05-23 18:29:06 +02:00
|
|
|
foo();
|
|
|
|
}
|
|
|
|
|
|
|
|
sum
|
2020-06-27 15:19:53 +02:00
|
|
|
"#
|
2020-05-23 18:29:06 +02:00
|
|
|
)
|
|
|
|
.expect_err("should error"),
|
2022-02-08 02:02:15 +01:00
|
|
|
EvalAltResult::ErrorInFunctionCall(fn_name, _, ..) if fn_name == "foo"
|
2020-05-23 18:29:06 +02:00
|
|
|
));
|
2020-05-15 15:40:54 +02:00
|
|
|
|
2020-06-14 08:25:47 +02:00
|
|
|
engine.set_max_modules(1000);
|
2020-05-17 16:19:49 +02:00
|
|
|
|
2020-05-23 18:29:06 +02:00
|
|
|
#[cfg(not(feature = "no_function"))]
|
2022-01-10 06:26:33 +01:00
|
|
|
engine.run(
|
2020-05-23 18:29:06 +02:00
|
|
|
r#"
|
|
|
|
fn foo() {
|
|
|
|
import "hello" as h;
|
|
|
|
}
|
2020-05-17 16:19:49 +02:00
|
|
|
|
2021-12-15 05:06:17 +01:00
|
|
|
for x in 0..10 {
|
2020-05-23 18:29:06 +02:00
|
|
|
foo();
|
|
|
|
}
|
2020-06-27 15:19:53 +02:00
|
|
|
"#,
|
2020-05-23 18:29:06 +02:00
|
|
|
)?;
|
|
|
|
}
|
2020-05-15 15:40:54 +02:00
|
|
|
|
2021-01-09 08:20:07 +01:00
|
|
|
#[cfg(not(feature = "no_function"))]
|
|
|
|
{
|
|
|
|
let script = r#"
|
2022-05-05 15:34:15 +02:00
|
|
|
fn foo(x) {
|
2021-01-09 08:20:07 +01:00
|
|
|
import "hello" as h;
|
2022-05-05 15:34:15 +02:00
|
|
|
h::answer * x
|
2021-01-09 08:20:07 +01:00
|
|
|
}
|
2022-05-05 15:34:15 +02:00
|
|
|
foo(1) + { import "hello" as h; h::answer }
|
2021-01-09 08:20:07 +01:00
|
|
|
"#;
|
|
|
|
let mut scope = Scope::new();
|
2021-01-08 17:24:55 +01:00
|
|
|
|
2021-01-09 08:20:07 +01:00
|
|
|
let ast = engine.compile_into_self_contained(&mut scope, script)?;
|
2021-01-08 17:24:55 +01:00
|
|
|
|
2021-01-09 08:20:07 +01:00
|
|
|
engine.set_module_resolver(DummyModuleResolver::new());
|
2021-01-08 17:24:55 +01:00
|
|
|
|
2021-01-09 08:20:07 +01:00
|
|
|
assert_eq!(engine.eval_ast::<INT>(&ast)?, 84);
|
2021-01-08 17:24:55 +01:00
|
|
|
|
2021-01-09 08:20:07 +01:00
|
|
|
assert!(engine.eval::<INT>(script).is_err());
|
2022-05-05 15:34:15 +02:00
|
|
|
|
2022-08-12 16:48:15 +02:00
|
|
|
let result = engine.call_fn::<INT>(&mut Scope::new(), &ast, "foo", (2 as INT,))?;
|
2022-05-05 15:34:15 +02:00
|
|
|
|
|
|
|
assert_eq!(result, 84);
|
2022-05-21 05:31:15 +02:00
|
|
|
|
|
|
|
let mut ast2 = engine.compile("fn foo(x) { 42 }")?;
|
|
|
|
|
2022-05-21 05:57:23 +02:00
|
|
|
#[cfg(feature = "internals")]
|
2022-05-21 05:31:15 +02:00
|
|
|
let len = ast.resolver().unwrap().len();
|
|
|
|
|
|
|
|
ast2 += ast;
|
|
|
|
|
2022-05-21 05:57:23 +02:00
|
|
|
#[cfg(feature = "internals")]
|
|
|
|
{
|
|
|
|
assert!(ast2.resolver().is_some());
|
|
|
|
assert_eq!(ast2.resolver().unwrap().len(), len);
|
|
|
|
}
|
2022-05-21 05:31:15 +02:00
|
|
|
|
2022-08-12 16:48:15 +02:00
|
|
|
let result = engine.call_fn::<INT>(&mut Scope::new(), &ast2, "foo", (2 as INT,))?;
|
2022-05-21 05:31:15 +02:00
|
|
|
|
|
|
|
assert_eq!(result, 84);
|
2021-01-09 08:20:07 +01:00
|
|
|
}
|
2021-01-08 17:24:55 +01:00
|
|
|
|
2020-05-05 17:57:25 +02:00
|
|
|
Ok(())
|
|
|
|
}
|
2020-05-06 17:00:26 +02:00
|
|
|
|
|
|
|
#[test]
|
2020-05-12 10:32:22 +02:00
|
|
|
#[cfg(not(feature = "no_function"))]
|
2020-05-06 17:00:26 +02:00
|
|
|
fn test_module_from_ast() -> Result<(), Box<EvalAltResult>> {
|
|
|
|
let mut engine = Engine::new();
|
|
|
|
|
2020-06-27 17:56:24 +02:00
|
|
|
let mut resolver1 = StaticModuleResolver::new();
|
2020-05-06 17:00:26 +02:00
|
|
|
let mut sub_module = Module::new();
|
|
|
|
sub_module.set_var("foo", true);
|
2020-06-27 17:56:24 +02:00
|
|
|
resolver1.insert("another module", sub_module);
|
2020-05-06 17:00:26 +02:00
|
|
|
|
|
|
|
let ast = engine.compile(
|
|
|
|
r#"
|
2020-06-27 15:19:53 +02:00
|
|
|
// Functions become module functions
|
|
|
|
fn calc(x) {
|
|
|
|
x + 1
|
|
|
|
}
|
|
|
|
fn add_len(x, y) {
|
|
|
|
x + len(y)
|
|
|
|
}
|
2020-09-30 03:57:21 +02:00
|
|
|
fn cross_call(x) {
|
|
|
|
calc(x)
|
|
|
|
}
|
2020-06-27 15:19:53 +02:00
|
|
|
private fn hidden() {
|
|
|
|
throw "you shouldn't see me!";
|
|
|
|
}
|
|
|
|
|
|
|
|
// Imported modules become sub-modules
|
|
|
|
import "another module" as extra;
|
|
|
|
|
|
|
|
// Variables defined at global level become module variables
|
2020-11-09 07:38:33 +01:00
|
|
|
export const x = 123;
|
2020-06-27 15:19:53 +02:00
|
|
|
let foo = 41;
|
|
|
|
let hello;
|
|
|
|
|
|
|
|
// Final variable values become constant module variable values
|
|
|
|
foo = calc(foo);
|
2021-04-04 17:22:45 +02:00
|
|
|
hello = `hello, ${foo} worlds!`;
|
2020-06-27 15:19:53 +02:00
|
|
|
|
2022-01-28 03:11:40 +01:00
|
|
|
export x as abc;
|
|
|
|
export x as xxx;
|
|
|
|
export foo;
|
|
|
|
export hello;
|
2020-06-27 15:19:53 +02:00
|
|
|
"#,
|
2020-05-06 17:00:26 +02:00
|
|
|
)?;
|
|
|
|
|
2020-12-26 06:05:57 +01:00
|
|
|
engine.set_module_resolver(resolver1);
|
2020-06-27 17:56:24 +02:00
|
|
|
|
2020-10-03 05:42:54 +02:00
|
|
|
let module = Module::eval_ast_as_new(Scope::new(), &ast, &engine)?;
|
2020-05-06 17:00:26 +02:00
|
|
|
|
2020-06-27 17:56:24 +02:00
|
|
|
let mut resolver2 = StaticModuleResolver::new();
|
|
|
|
resolver2.insert("testing", module);
|
2020-12-26 06:05:57 +01:00
|
|
|
engine.set_module_resolver(resolver2);
|
2020-06-01 04:58:14 +02:00
|
|
|
|
2020-05-06 17:00:26 +02:00
|
|
|
assert_eq!(
|
2020-07-17 08:50:23 +02:00
|
|
|
engine.eval::<INT>(r#"import "testing" as ttt; ttt::abc"#)?,
|
2020-05-06 17:00:26 +02:00
|
|
|
123
|
|
|
|
);
|
2020-11-09 07:38:33 +01:00
|
|
|
assert_eq!(
|
|
|
|
engine.eval::<INT>(r#"import "testing" as ttt; ttt::x"#)?,
|
|
|
|
123
|
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
engine.eval::<INT>(r#"import "testing" as ttt; ttt::xxx"#)?,
|
|
|
|
123
|
|
|
|
);
|
2020-05-06 17:00:26 +02:00
|
|
|
assert_eq!(
|
2020-07-17 08:50:23 +02:00
|
|
|
engine.eval::<INT>(r#"import "testing" as ttt; ttt::foo"#)?,
|
2020-05-06 17:00:26 +02:00
|
|
|
42
|
|
|
|
);
|
2020-07-17 08:50:23 +02:00
|
|
|
assert!(engine.eval::<bool>(r#"import "testing" as ttt; ttt::extra::foo"#)?);
|
2020-05-06 17:00:26 +02:00
|
|
|
assert_eq!(
|
2020-07-17 08:50:23 +02:00
|
|
|
engine.eval::<String>(r#"import "testing" as ttt; ttt::hello"#)?,
|
2020-05-06 17:00:26 +02:00
|
|
|
"hello, 42 worlds!"
|
|
|
|
);
|
|
|
|
assert_eq!(
|
2020-07-17 08:50:23 +02:00
|
|
|
engine.eval::<INT>(r#"import "testing" as ttt; ttt::calc(999)"#)?,
|
2020-05-06 17:00:26 +02:00
|
|
|
1000
|
|
|
|
);
|
2020-09-30 03:57:21 +02:00
|
|
|
assert_eq!(
|
|
|
|
engine.eval::<INT>(r#"import "testing" as ttt; ttt::cross_call(999)"#)?,
|
|
|
|
1000
|
|
|
|
);
|
2020-05-06 17:00:26 +02:00
|
|
|
assert_eq!(
|
2020-07-17 08:50:23 +02:00
|
|
|
engine.eval::<INT>(r#"import "testing" as ttt; ttt::add_len(ttt::foo, ttt::hello)"#)?,
|
2020-05-06 17:00:26 +02:00
|
|
|
59
|
|
|
|
);
|
2020-05-09 05:29:30 +02:00
|
|
|
assert!(matches!(
|
|
|
|
*engine
|
2021-08-06 08:46:27 +02:00
|
|
|
.run(r#"import "testing" as ttt; ttt::hidden()"#)
|
2020-05-09 05:29:30 +02:00
|
|
|
.expect_err("should error"),
|
2022-02-08 02:02:15 +01:00
|
|
|
EvalAltResult::ErrorFunctionNotFound(fn_name, ..) if fn_name == "ttt::hidden ()"
|
2020-05-09 05:29:30 +02:00
|
|
|
));
|
2020-05-06 17:00:26 +02:00
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
2020-06-12 12:04:16 +02:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_module_export() -> Result<(), Box<EvalAltResult>> {
|
|
|
|
let engine = Engine::new();
|
|
|
|
|
|
|
|
assert!(matches!(
|
2021-04-20 06:01:35 +02:00
|
|
|
engine.compile("let x = 10; { export x; }").expect_err("should error"),
|
2022-02-08 02:02:15 +01:00
|
|
|
ParseError(x, ..) if *x == ParseErrorType::WrongExport
|
2020-06-12 12:04:16 +02:00
|
|
|
));
|
|
|
|
|
2020-06-30 12:34:58 +02:00
|
|
|
#[cfg(not(feature = "no_function"))]
|
2020-06-12 12:04:16 +02:00
|
|
|
assert!(matches!(
|
2021-04-20 06:01:35 +02:00
|
|
|
engine.compile("fn abc(x) { export x; }").expect_err("should error"),
|
2022-02-08 02:02:15 +01:00
|
|
|
ParseError(x, ..) if *x == ParseErrorType::WrongExport
|
2020-06-12 12:04:16 +02:00
|
|
|
));
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
2020-09-19 12:12:23 +02:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_module_str() -> Result<(), Box<EvalAltResult>> {
|
2021-03-06 14:25:49 +01:00
|
|
|
fn test_fn(input: ImmutableString) -> Result<INT, Box<EvalAltResult>> {
|
|
|
|
Ok(input.len() as INT)
|
2020-09-19 12:12:23 +02:00
|
|
|
}
|
2021-03-06 14:25:49 +01:00
|
|
|
fn test_fn2(input: &str) -> Result<INT, Box<EvalAltResult>> {
|
|
|
|
Ok(input.len() as INT)
|
2020-09-19 12:12:23 +02:00
|
|
|
}
|
2021-03-06 14:25:49 +01:00
|
|
|
fn test_fn3(input: String) -> Result<INT, Box<EvalAltResult>> {
|
|
|
|
Ok(input.len() as INT)
|
2020-09-19 12:12:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
let mut engine = rhai::Engine::new();
|
|
|
|
let mut module = Module::new();
|
2021-03-15 05:39:06 +01:00
|
|
|
module.set_native_fn("test", test_fn);
|
|
|
|
module.set_native_fn("test2", test_fn2);
|
|
|
|
module.set_native_fn("test3", test_fn3);
|
2020-09-19 12:12:23 +02:00
|
|
|
|
|
|
|
let mut static_modules = rhai::module_resolvers::StaticModuleResolver::new();
|
|
|
|
static_modules.insert("test", module);
|
2020-12-26 06:05:57 +01:00
|
|
|
engine.set_module_resolver(static_modules);
|
2020-09-19 12:12:23 +02:00
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
engine.eval::<INT>(r#"import "test" as test; test::test("test");"#)?,
|
2021-03-06 14:25:49 +01:00
|
|
|
4
|
2020-09-19 12:12:23 +02:00
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
engine.eval::<INT>(r#"import "test" as test; test::test2("test");"#)?,
|
2021-03-06 14:25:49 +01:00
|
|
|
4
|
2020-09-19 12:12:23 +02:00
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
engine.eval::<INT>(r#"import "test" as test; test::test3("test");"#)?,
|
2021-03-06 14:25:49 +01:00
|
|
|
4
|
2020-09-19 12:12:23 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
2020-10-02 08:55:02 +02:00
|
|
|
|
2020-10-02 09:31:59 +02:00
|
|
|
#[cfg(not(feature = "no_function"))]
|
2020-10-02 08:55:02 +02:00
|
|
|
#[test]
|
|
|
|
fn test_module_ast_namespace() -> Result<(), Box<EvalAltResult>> {
|
2022-08-26 10:20:23 +02:00
|
|
|
let script = "
|
2020-10-02 08:55:02 +02:00
|
|
|
fn foo(x) { x + 1 }
|
|
|
|
fn bar(x) { foo(x) }
|
2022-08-26 10:20:23 +02:00
|
|
|
";
|
2020-10-02 08:55:02 +02:00
|
|
|
|
|
|
|
let mut engine = Engine::new();
|
|
|
|
|
|
|
|
let ast = engine.compile(script)?;
|
|
|
|
|
2021-11-07 11:12:37 +01:00
|
|
|
let module = Module::eval_ast_as_new(Scope::new(), &ast, &engine)?;
|
2020-10-02 08:55:02 +02:00
|
|
|
|
|
|
|
let mut resolver = StaticModuleResolver::new();
|
|
|
|
resolver.insert("testing", module);
|
2020-12-26 06:05:57 +01:00
|
|
|
engine.set_module_resolver(resolver);
|
2020-10-02 08:55:02 +02:00
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
engine.eval::<INT>(r#"import "testing" as t; t::foo(41)"#)?,
|
|
|
|
42
|
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
engine.eval::<INT>(r#"import "testing" as t; t::bar(41)"#)?,
|
|
|
|
42
|
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
engine.eval::<INT>(r#"fn foo(x) { x - 1 } import "testing" as t; t::foo(41)"#)?,
|
|
|
|
42
|
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
engine.eval::<INT>(r#"fn foo(x) { x - 1 } import "testing" as t; t::bar(41)"#)?,
|
|
|
|
42
|
|
|
|
);
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
2020-10-16 15:16:06 +02:00
|
|
|
|
|
|
|
#[cfg(not(feature = "no_function"))]
|
|
|
|
#[test]
|
|
|
|
fn test_module_ast_namespace2() -> Result<(), Box<EvalAltResult>> {
|
|
|
|
use rhai::{Engine, Module, Scope};
|
|
|
|
|
2022-08-26 10:20:23 +02:00
|
|
|
const MODULE_TEXT: &str = "
|
2020-10-16 15:16:06 +02:00
|
|
|
fn run_function(function) {
|
2020-10-16 15:26:38 +02:00
|
|
|
call(function)
|
2020-10-16 15:16:06 +02:00
|
|
|
}
|
2022-08-26 10:20:23 +02:00
|
|
|
";
|
2020-10-16 15:16:06 +02:00
|
|
|
|
|
|
|
const SCRIPT: &str = r#"
|
|
|
|
import "test_module" as test;
|
|
|
|
|
|
|
|
fn foo() {
|
|
|
|
print("foo");
|
|
|
|
}
|
|
|
|
|
|
|
|
test::run_function(Fn("foo"));
|
|
|
|
"#;
|
|
|
|
|
|
|
|
let mut engine = Engine::new();
|
|
|
|
let module_ast = engine.compile(MODULE_TEXT)?;
|
|
|
|
let module = Module::eval_ast_as_new(Scope::new(), &module_ast, &engine)?;
|
|
|
|
let mut static_modules = rhai::module_resolvers::StaticModuleResolver::new();
|
|
|
|
static_modules.insert("test_module", module);
|
2020-12-26 06:05:57 +01:00
|
|
|
engine.set_module_resolver(static_modules);
|
2020-10-16 15:16:06 +02:00
|
|
|
|
2021-08-06 08:46:27 +02:00
|
|
|
engine.run(SCRIPT)?;
|
2020-10-16 15:16:06 +02:00
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
2020-11-08 16:00:03 +01:00
|
|
|
|
2022-08-26 10:20:23 +02:00
|
|
|
#[cfg(not(feature = "no_function"))]
|
|
|
|
#[cfg(feature = "internals")]
|
|
|
|
#[test]
|
|
|
|
fn test_module_context() -> Result<(), Box<EvalAltResult>> {
|
|
|
|
let script = "fn bar() { calc(|x| x + 1) }";
|
|
|
|
|
|
|
|
let mut engine = Engine::new();
|
|
|
|
|
|
|
|
let ast = engine.compile(script)?;
|
|
|
|
|
|
|
|
let module = Module::eval_ast_as_new(Scope::new(), &ast, &engine)?;
|
|
|
|
|
|
|
|
let mut resolver = StaticModuleResolver::new();
|
|
|
|
resolver.insert("testing", module);
|
|
|
|
engine.set_module_resolver(resolver);
|
|
|
|
|
|
|
|
engine.register_fn(
|
|
|
|
"calc",
|
|
|
|
|context: NativeCallContext, fp: FnPtr| -> Result<INT, Box<EvalAltResult>> {
|
|
|
|
// Store fields for later use
|
|
|
|
let engine = context.engine();
|
|
|
|
let fn_name = context.fn_name().to_string();
|
|
|
|
let source = context.source().map(|s| s.to_string());
|
|
|
|
let global = context.global_runtime_state().unwrap().clone();
|
|
|
|
let pos = context.position();
|
|
|
|
let call_level = context.call_level();
|
|
|
|
|
|
|
|
// Store the paths of the stack of call modules up to this point
|
|
|
|
let modules_list: Vec<String> = context
|
|
|
|
.iter_namespaces()
|
|
|
|
.map(|m| m.id().unwrap_or("testing"))
|
|
|
|
.filter(|id| !id.is_empty())
|
|
|
|
.map(|id| id.to_string())
|
|
|
|
.collect();
|
|
|
|
|
|
|
|
// Recreate the 'NativeCallContext' - requires the 'internals' feature
|
|
|
|
let mut libraries = Vec::<Shared<Module>>::new();
|
|
|
|
|
|
|
|
for path in modules_list {
|
|
|
|
// Recreate the stack of call modules by resolving each path with
|
|
|
|
// the module resolver.
|
|
|
|
let module = engine.module_resolver().resolve(engine, None, &path, pos)?;
|
|
|
|
|
|
|
|
libraries.push(module);
|
|
|
|
}
|
|
|
|
|
|
|
|
let lib: Vec<&Module> = libraries.iter().map(|m| m.as_ref()).collect();
|
|
|
|
|
|
|
|
let new_context = NativeCallContext::new_with_all_fields(
|
|
|
|
engine,
|
|
|
|
&fn_name,
|
2022-10-29 08:12:18 +02:00
|
|
|
source.as_ref().map(String::as_str),
|
2022-08-26 10:20:23 +02:00
|
|
|
&global,
|
|
|
|
&lib,
|
|
|
|
pos,
|
|
|
|
call_level,
|
|
|
|
);
|
|
|
|
|
|
|
|
fp.call_within_context(&new_context, (41 as INT,))
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
engine.eval::<INT>(r#"import "testing" as t; t::bar()"#)?,
|
|
|
|
42
|
|
|
|
);
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2020-11-08 16:00:03 +01:00
|
|
|
#[test]
|
|
|
|
fn test_module_file() -> Result<(), Box<EvalAltResult>> {
|
|
|
|
let engine = Engine::new();
|
|
|
|
let ast = engine.compile(
|
|
|
|
r#"
|
|
|
|
import "scripts/module";
|
|
|
|
print("top");
|
|
|
|
"#,
|
|
|
|
)?;
|
2021-11-07 11:12:37 +01:00
|
|
|
Module::eval_ast_as_new(Scope::new(), &ast, &engine)?;
|
2020-11-08 16:00:03 +01:00
|
|
|
Ok(())
|
|
|
|
}
|
2022-01-30 10:27:13 +01:00
|
|
|
|
|
|
|
#[cfg(not(feature = "no_function"))]
|
|
|
|
#[test]
|
|
|
|
fn test_module_environ() -> Result<(), Box<EvalAltResult>> {
|
|
|
|
let mut engine = Engine::new();
|
|
|
|
|
|
|
|
let ast = engine.compile(
|
|
|
|
r#"
|
|
|
|
const SECRET = 42;
|
|
|
|
|
|
|
|
fn foo(x) {
|
|
|
|
print(global::SECRET);
|
|
|
|
global::SECRET + x
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
)?;
|
|
|
|
|
|
|
|
let mut m = Module::eval_ast_as_new(Scope::new(), &ast, &engine)?;
|
|
|
|
|
|
|
|
m.set_id("test");
|
|
|
|
m.build_index();
|
|
|
|
|
|
|
|
engine.register_static_module("test", m.into());
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
engine.eval::<String>(
|
|
|
|
r#"
|
|
|
|
const SECRET = "hello";
|
|
|
|
|
|
|
|
fn foo(x) {
|
|
|
|
print(global::SECRET);
|
|
|
|
global::SECRET + x
|
|
|
|
}
|
|
|
|
|
|
|
|
let t = test::foo(0);
|
|
|
|
|
|
|
|
foo(t)
|
|
|
|
"#
|
|
|
|
)?,
|
|
|
|
"hello42"
|
|
|
|
);
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
2022-04-08 05:27:52 +02:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_module_dynamic() -> Result<(), Box<EvalAltResult>> {
|
|
|
|
fn test_fn(input: Dynamic, x: INT) -> Result<INT, Box<EvalAltResult>> {
|
|
|
|
let s = input.into_string().unwrap();
|
|
|
|
Ok(s.len() as INT + x)
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut engine = rhai::Engine::new();
|
|
|
|
let mut module = Module::new();
|
|
|
|
module.set_native_fn("test", test_fn);
|
|
|
|
|
|
|
|
let mut static_modules = rhai::module_resolvers::StaticModuleResolver::new();
|
|
|
|
static_modules.insert("test", module);
|
|
|
|
engine.set_module_resolver(static_modules);
|
2022-08-22 16:16:26 +02:00
|
|
|
engine.register_fn("test2", test_fn);
|
2022-04-08 05:27:52 +02:00
|
|
|
|
|
|
|
assert_eq!(engine.eval::<INT>(r#"test2("test", 38);"#)?, 42);
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
engine.eval::<INT>(r#"import "test" as test; test::test("test", 38);"#)?,
|
|
|
|
42
|
|
|
|
);
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|