Allow scripted functions in packages.
This commit is contained in:
parent
a75aba8a85
commit
6c69a40083
@ -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
|
||||||
|
@ -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,
|
||||||
|
@ -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.
|
||||||
|
@ -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(())
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user