Allow scripted functions in packages.

This commit is contained in:
Stephen Chung 2020-09-11 22:32:59 +08:00
parent a75aba8a85
commit 6c69a40083
4 changed files with 68 additions and 23 deletions

View File

@ -8,6 +8,7 @@ New features
------------ ------------
* Plugins support via procedural macros. * Plugins support via procedural macros.
* Scripted functions are allowed in packages.
Version 0.18.3 Version 0.18.3

View File

@ -444,7 +444,7 @@ impl Engine {
//|| self.global_module.contains_fn(hash_script, pub_only) //|| self.global_module.contains_fn(hash_script, pub_only)
|| self.global_module.contains_fn(hash_fn, pub_only) || self.global_module.contains_fn(hash_fn, pub_only)
// Then check packages // Then check packages
//|| self.packages.contains_fn(hash_script, pub_only) || self.packages.contains_fn(hash_script, pub_only)
|| self.packages.contains_fn(hash_fn, pub_only) || self.packages.contains_fn(hash_fn, pub_only)
} }
@ -477,7 +477,17 @@ impl Engine {
// Qualifiers (none) + function name + number of arguments + argument `TypeId`'s. // Qualifiers (none) + function name + number of arguments + argument `TypeId`'s.
let arg_types = args.iter().map(|a| a.type_id()); let arg_types = args.iter().map(|a| a.type_id());
let hash_fn = calc_fn_hash(empty(), fn_name, args.len(), arg_types); let hash_fn = calc_fn_hash(
empty(),
fn_name,
if args.is_empty() {
// Distinguish between a script function and a native function with no parameters
usize::MAX
} else {
args.len()
},
arg_types,
);
match fn_name { match fn_name {
// type_of // type_of
@ -514,9 +524,15 @@ impl Engine {
// Normal script function call // Normal script function call
#[cfg(not(feature = "no_function"))] #[cfg(not(feature = "no_function"))]
_ if hash_script > 0 && lib.contains_fn(hash_script, pub_only) => { _ if lib.contains_fn(hash_script, pub_only)
|| self.packages.contains_fn(hash_script, pub_only) =>
{
// Get scripted function // Get scripted function
let func = lib.get_fn(hash_script, pub_only).unwrap().get_fn_def(); let func = lib
.get_fn(hash_script, pub_only)
.or_else(|| self.packages.get_fn(hash_script, pub_only))
.unwrap()
.get_fn_def();
let scope = &mut Scope::new(); let scope = &mut Scope::new();
let mods = &mut Imports::new(); let mods = &mut Imports::new();
@ -559,6 +575,7 @@ impl Engine {
Ok((result, false)) Ok((result, false))
} }
// Normal native function call // Normal native function call
_ => self.call_native_fn( _ => self.call_native_fn(
state, lib, fn_name, hash_fn, args, is_ref, pub_only, def_val, state, lib, fn_name, hash_fn, args, is_ref, pub_only, def_val,

View File

@ -247,10 +247,14 @@ impl Module {
&mut self, &mut self,
hash_var: u64, hash_var: u64,
) -> Result<&mut Dynamic, Box<EvalAltResult>> { ) -> Result<&mut Dynamic, Box<EvalAltResult>> {
if hash_var == 0 {
Err(EvalAltResult::ErrorVariableNotFound(String::new(), Position::none()).into())
} else {
self.all_variables.get_mut(&hash_var).ok_or_else(|| { self.all_variables.get_mut(&hash_var).ok_or_else(|| {
EvalAltResult::ErrorVariableNotFound(String::new(), Position::none()).into() EvalAltResult::ErrorVariableNotFound(String::new(), Position::none()).into()
}) })
} }
}
/// Set a script-defined function into the module. /// Set a script-defined function into the module.
/// ///
@ -354,7 +358,9 @@ impl Module {
/// assert!(module.contains_fn(hash, true)); /// assert!(module.contains_fn(hash, true));
/// ``` /// ```
pub fn contains_fn(&self, hash_fn: u64, public_only: bool) -> bool { pub fn contains_fn(&self, hash_fn: u64, public_only: bool) -> bool {
if public_only { if hash_fn == 0 {
false
} else if public_only {
self.functions self.functions
.get(&hash_fn) .get(&hash_fn)
.map(|(_, access, _, _)| match access { .map(|(_, access, _, _)| match access {
@ -383,7 +389,14 @@ impl Module {
) -> u64 { ) -> u64 {
let name = name.into(); let name = name.into();
let hash_fn = calc_fn_hash(empty(), &name, arg_types.len(), arg_types.iter().cloned()); let args_len = if arg_types.is_empty() {
// Distinguish between a script function and a function with no parameters
usize::MAX
} else {
arg_types.len()
};
let hash_fn = calc_fn_hash(empty(), &name, args_len, arg_types.iter().cloned());
let params = arg_types.into_iter().cloned().collect(); let params = arg_types.into_iter().cloned().collect();
@ -910,6 +923,9 @@ impl Module {
/// The `u64` hash is calculated by the function `crate::calc_fn_hash`. /// The `u64` hash is calculated by the function `crate::calc_fn_hash`.
/// It is also returned by the `set_fn_XXX` calls. /// It is also returned by the `set_fn_XXX` calls.
pub(crate) fn get_fn(&self, hash_fn: u64, public_only: bool) -> Option<&Func> { pub(crate) fn get_fn(&self, hash_fn: u64, public_only: bool) -> Option<&Func> {
if hash_fn == 0 {
None
} else {
self.functions self.functions
.get(&hash_fn) .get(&hash_fn)
.and_then(|(_, access, _, f)| match access { .and_then(|(_, access, _, f)| match access {
@ -918,6 +934,7 @@ impl Module {
FnAccess::Private => None, FnAccess::Private => None,
}) })
} }
}
/// Get a modules-qualified function. /// Get a modules-qualified function.
/// Name and Position in `EvalAltResult` are None and must be set afterwards. /// Name and Position in `EvalAltResult` are None and must be set afterwards.

View File

@ -1,10 +1,10 @@
use rhai::{Engine, EvalAltResult, INT, Scope};
use rhai::packages::{Package, StandardPackage}; use rhai::packages::{Package, StandardPackage};
use rhai::{Engine, EvalAltResult, Module, Scope, INT};
#[test] #[test]
fn test_packages() -> Result<(), Box<EvalAltResult>> { fn test_packages() -> Result<(), Box<EvalAltResult>> {
let e = Engine::new(); let engine = Engine::new();
let ast = e.compile("x")?; let ast = engine.compile("x")?;
let std_pkg = StandardPackage::new(); let std_pkg = StandardPackage::new();
let make_call = |x: INT| -> Result<INT, Box<EvalAltResult>> { let make_call = |x: INT| -> Result<INT, Box<EvalAltResult>> {
@ -24,10 +24,20 @@ fn test_packages() -> Result<(), Box<EvalAltResult>> {
engine.eval_ast_with_scope::<INT>(&mut scope, &ast) engine.eval_ast_with_scope::<INT>(&mut scope, &ast)
}; };
// The following loop creates 10,000 Engine instances! assert_eq!(make_call(42)?, 42);
for x in 0..10_000 {
assert_eq!(make_call(x)?, x); Ok(())
} }
#[test]
fn test_packages_with_script() -> Result<(), Box<EvalAltResult>> {
let mut engine = Engine::new();
let ast = engine.compile("fn foo(x) { x + 1 }")?;
let module = Module::eval_ast_as_new(Scope::new(), &ast, &engine)?;
engine.load_package(module);
assert_eq!(engine.eval::<INT>("foo(41)")?, 42);
Ok(()) Ok(())
} }