Keyword can no longer be overloaded.
This commit is contained in:
parent
b7e864bb78
commit
fc10df7d63
@ -19,6 +19,7 @@ Breaking changes
|
|||||||
* `private` functions in an `AST` can now be called with `call_fn` etc.
|
* `private` functions in an `AST` can now be called with `call_fn` etc.
|
||||||
* `NativeCallContext::call_fn_dynamic_raw` no longer has the `pub_only` parameter.
|
* `NativeCallContext::call_fn_dynamic_raw` no longer has the `pub_only` parameter.
|
||||||
* `Module::update_fn_metadata` input parameter is changed.
|
* `Module::update_fn_metadata` input parameter is changed.
|
||||||
|
* Function keywords (e.g. `type_of`, `eval`, `Fn`) can no longer be overloaded. It is more trouble than worth. To disable these keywords, use `Engine::disable_symbol`.
|
||||||
|
|
||||||
Enhancements
|
Enhancements
|
||||||
------------
|
------------
|
||||||
|
112
src/fn_call.rs
112
src/fn_call.rs
@ -192,9 +192,9 @@ impl Engine {
|
|||||||
#[inline]
|
#[inline]
|
||||||
fn resolve_function<'s>(
|
fn resolve_function<'s>(
|
||||||
&self,
|
&self,
|
||||||
|
mods: &Imports,
|
||||||
state: &'s mut State,
|
state: &'s mut State,
|
||||||
lib: &[&Module],
|
lib: &[&Module],
|
||||||
mods: &Imports,
|
|
||||||
fn_name: &str,
|
fn_name: &str,
|
||||||
mut hash: NonZeroU64,
|
mut hash: NonZeroU64,
|
||||||
args: &mut FnCallArgs,
|
args: &mut FnCallArgs,
|
||||||
@ -203,8 +203,8 @@ impl Engine {
|
|||||||
fn find_function(
|
fn find_function(
|
||||||
engine: &Engine,
|
engine: &Engine,
|
||||||
hash: NonZeroU64,
|
hash: NonZeroU64,
|
||||||
lib: &[&Module],
|
|
||||||
mods: &Imports,
|
mods: &Imports,
|
||||||
|
lib: &[&Module],
|
||||||
) -> Option<(CallableFunction, Option<ImmutableString>)> {
|
) -> Option<(CallableFunction, Option<ImmutableString>)> {
|
||||||
lib.iter()
|
lib.iter()
|
||||||
.find_map(|m| {
|
.find_map(|m| {
|
||||||
@ -249,7 +249,7 @@ impl Engine {
|
|||||||
let mut bitmask = 1usize; // Bitmask of which parameter to replace with `Dynamic`
|
let mut bitmask = 1usize; // Bitmask of which parameter to replace with `Dynamic`
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
match find_function(self, hash, lib, mods) {
|
match find_function(self, hash, mods, lib) {
|
||||||
// Specific version found
|
// Specific version found
|
||||||
Some(f) => return Some(f),
|
Some(f) => return Some(f),
|
||||||
|
|
||||||
@ -304,7 +304,7 @@ impl Engine {
|
|||||||
let source = state.source.clone();
|
let source = state.source.clone();
|
||||||
|
|
||||||
// Check if function access already in the cache
|
// Check if function access already in the cache
|
||||||
let func = self.resolve_function(state, lib, mods, fn_name, hash_fn, args, true);
|
let func = self.resolve_function(mods, state, lib, fn_name, hash_fn, args, true);
|
||||||
|
|
||||||
if let Some((func, src)) = func {
|
if let Some((func, src)) = func {
|
||||||
assert!(func.is_native());
|
assert!(func.is_native());
|
||||||
@ -578,16 +578,18 @@ impl Engine {
|
|||||||
pub(crate) fn has_override_by_name_and_arguments(
|
pub(crate) fn has_override_by_name_and_arguments(
|
||||||
&self,
|
&self,
|
||||||
mods: Option<&Imports>,
|
mods: Option<&Imports>,
|
||||||
state: Option<&mut State>,
|
|
||||||
lib: &[&Module],
|
lib: &[&Module],
|
||||||
fn_name: &str,
|
fn_name: &str,
|
||||||
arg_types: &[TypeId],
|
arg_types: &[TypeId],
|
||||||
) -> bool {
|
) -> bool {
|
||||||
let arg_types = arg_types.as_ref();
|
let arg_types = arg_types.as_ref();
|
||||||
let hash_fn = calc_native_fn_hash(empty(), fn_name, arg_types.iter().cloned());
|
|
||||||
let hash_script = calc_script_fn_hash(empty(), fn_name, arg_types.len());
|
|
||||||
|
|
||||||
self.has_override(mods, state, lib, hash_fn, hash_script)
|
self.has_override(
|
||||||
|
mods,
|
||||||
|
lib,
|
||||||
|
calc_native_fn_hash(empty(), fn_name, arg_types.iter().cloned()),
|
||||||
|
calc_script_fn_hash(empty(), fn_name, arg_types.len()),
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Has a system function an override?
|
// Has a system function an override?
|
||||||
@ -595,29 +597,12 @@ impl Engine {
|
|||||||
pub(crate) fn has_override(
|
pub(crate) fn has_override(
|
||||||
&self,
|
&self,
|
||||||
mods: Option<&Imports>,
|
mods: Option<&Imports>,
|
||||||
mut state: Option<&mut State>,
|
|
||||||
lib: &[&Module],
|
lib: &[&Module],
|
||||||
hash_fn: Option<NonZeroU64>,
|
hash_fn: Option<NonZeroU64>,
|
||||||
hash_script: Option<NonZeroU64>,
|
hash_script: Option<NonZeroU64>,
|
||||||
) -> bool {
|
) -> bool {
|
||||||
// Check if it is already in the cache
|
|
||||||
if let Some(state) = state.as_mut() {
|
|
||||||
if let Some(hash) = hash_script {
|
|
||||||
match state.fn_resolution_cache().map_or(None, |c| c.get(&hash)) {
|
|
||||||
Some(v) => return v.is_some(),
|
|
||||||
None => (),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if let Some(hash) = hash_fn {
|
|
||||||
match state.fn_resolution_cache().map_or(None, |c| c.get(&hash)) {
|
|
||||||
Some(v) => return v.is_some(),
|
|
||||||
None => (),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// First check script-defined functions
|
// First check script-defined functions
|
||||||
let r = hash_script.map_or(false, |hash| lib.iter().any(|&m| m.contains_fn(hash, false)))
|
hash_script.map_or(false, |hash| lib.iter().any(|&m| m.contains_fn(hash, false)))
|
||||||
//|| hash_fn.map_or(false, |hash| lib.iter().any(|&m| m.contains_fn(hash, false)))
|
//|| hash_fn.map_or(false, |hash| lib.iter().any(|&m| m.contains_fn(hash, false)))
|
||||||
// Then check registered functions
|
// Then check registered functions
|
||||||
|| hash_script.map_or(false, |hash| self.global_namespace.contains_fn(hash, false))
|
|| hash_script.map_or(false, |hash| self.global_namespace.contains_fn(hash, false))
|
||||||
@ -630,21 +615,7 @@ impl Engine {
|
|||||||
|| hash_fn.map_or(false, |hash| mods.map_or(false, |m| m.contains_fn(hash)))
|
|| hash_fn.map_or(false, |hash| mods.map_or(false, |m| m.contains_fn(hash)))
|
||||||
// Then check sub-modules
|
// Then check sub-modules
|
||||||
|| hash_script.map_or(false, |hash| self.global_sub_modules.values().any(|m| m.contains_qualified_fn(hash)))
|
|| hash_script.map_or(false, |hash| self.global_sub_modules.values().any(|m| m.contains_qualified_fn(hash)))
|
||||||
|| hash_fn.map_or(false, |hash| self.global_sub_modules.values().any(|m| m.contains_qualified_fn(hash)));
|
|| hash_fn.map_or(false, |hash| self.global_sub_modules.values().any(|m| m.contains_qualified_fn(hash)))
|
||||||
|
|
||||||
// If there is no override, put that information into the cache
|
|
||||||
if !r {
|
|
||||||
if let Some(state) = state.as_mut() {
|
|
||||||
if let Some(hash) = hash_script {
|
|
||||||
state.fn_resolution_cache_mut().insert(hash, None);
|
|
||||||
}
|
|
||||||
if let Some(hash) = hash_fn {
|
|
||||||
state.fn_resolution_cache_mut().insert(hash, None);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
r
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Perform an actual function call, native Rust or scripted, taking care of special functions.
|
/// Perform an actual function call, native Rust or scripted, taking care of special functions.
|
||||||
@ -678,35 +649,14 @@ impl Engine {
|
|||||||
|
|
||||||
match fn_name {
|
match fn_name {
|
||||||
// type_of
|
// type_of
|
||||||
KEYWORD_TYPE_OF
|
KEYWORD_TYPE_OF if args.len() == 1 => Ok((
|
||||||
if args.len() == 1
|
|
||||||
&& !self.has_override(
|
|
||||||
Some(mods),
|
|
||||||
Some(state),
|
|
||||||
lib,
|
|
||||||
Some(hash_fn),
|
|
||||||
hash_script,
|
|
||||||
) =>
|
|
||||||
{
|
|
||||||
Ok((
|
|
||||||
self.map_type_name(args[0].type_name()).to_string().into(),
|
self.map_type_name(args[0].type_name()).to_string().into(),
|
||||||
false,
|
false,
|
||||||
))
|
)),
|
||||||
}
|
|
||||||
|
|
||||||
// Fn/eval - reaching this point it must be a method-style call, mostly like redirected
|
// Fn/eval - reaching this point it must be a method-style call, mostly like redirected
|
||||||
// by a function pointer so it isn't caught at parse time.
|
// by a function pointer so it isn't caught at parse time.
|
||||||
KEYWORD_FN_PTR | KEYWORD_EVAL
|
KEYWORD_FN_PTR | KEYWORD_EVAL if args.len() == 1 => EvalAltResult::ErrorRuntime(
|
||||||
if args.len() == 1
|
|
||||||
&& !self.has_override(
|
|
||||||
Some(mods),
|
|
||||||
Some(state),
|
|
||||||
lib,
|
|
||||||
Some(hash_fn),
|
|
||||||
hash_script,
|
|
||||||
) =>
|
|
||||||
{
|
|
||||||
EvalAltResult::ErrorRuntime(
|
|
||||||
format!(
|
format!(
|
||||||
"'{}' should not be called in method style. Try {}(...);",
|
"'{}' should not be called in method style. Try {}(...);",
|
||||||
fn_name, fn_name
|
fn_name, fn_name
|
||||||
@ -714,13 +664,12 @@ impl Engine {
|
|||||||
.into(),
|
.into(),
|
||||||
pos,
|
pos,
|
||||||
)
|
)
|
||||||
.into()
|
.into(),
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(not(feature = "no_function"))]
|
#[cfg(not(feature = "no_function"))]
|
||||||
_ if hash_script.is_some() => {
|
_ if hash_script.is_some() => {
|
||||||
let (func, source) = self
|
let (func, source) = self
|
||||||
.resolve_function(state, lib, mods, fn_name, hash_script.unwrap(), args, false)
|
.resolve_function(mods, state, lib, fn_name, hash_script.unwrap(), args, false)
|
||||||
.as_ref()
|
.as_ref()
|
||||||
.map(|(f, s)| (Some(f.clone()), s.clone()))
|
.map(|(f, s)| (Some(f.clone()), s.clone()))
|
||||||
.unwrap_or((None, None));
|
.unwrap_or((None, None));
|
||||||
@ -1037,10 +986,7 @@ impl Engine {
|
|||||||
let mut curry = StaticVec::new();
|
let mut curry = StaticVec::new();
|
||||||
let mut name = fn_name;
|
let mut name = fn_name;
|
||||||
|
|
||||||
if name == KEYWORD_FN_PTR_CALL
|
if name == KEYWORD_FN_PTR_CALL && args_expr.len() >= 1 {
|
||||||
&& args_expr.len() >= 1
|
|
||||||
&& !self.has_override(Some(mods), Some(state), lib, None, hash_script)
|
|
||||||
{
|
|
||||||
let fn_ptr = self.eval_expr(scope, mods, state, lib, this_ptr, &args_expr[0], level)?;
|
let fn_ptr = self.eval_expr(scope, mods, state, lib, this_ptr, &args_expr[0], level)?;
|
||||||
|
|
||||||
if !fn_ptr.is::<FnPtr>() {
|
if !fn_ptr.is::<FnPtr>() {
|
||||||
@ -1067,9 +1013,6 @@ impl Engine {
|
|||||||
|
|
||||||
// Handle Fn()
|
// Handle Fn()
|
||||||
if name == KEYWORD_FN_PTR && args_expr.len() == 1 {
|
if name == KEYWORD_FN_PTR && args_expr.len() == 1 {
|
||||||
let hash_fn = calc_native_fn_hash(empty(), name, once(TypeId::of::<ImmutableString>()));
|
|
||||||
|
|
||||||
if !self.has_override(Some(mods), Some(state), lib, hash_fn, hash_script) {
|
|
||||||
// Fn - only in function call style
|
// Fn - only in function call style
|
||||||
return self
|
return self
|
||||||
.eval_expr(scope, mods, state, lib, this_ptr, &args_expr[0], level)?
|
.eval_expr(scope, mods, state, lib, this_ptr, &args_expr[0], level)?
|
||||||
@ -1081,7 +1024,6 @@ impl Engine {
|
|||||||
.map(Into::<Dynamic>::into)
|
.map(Into::<Dynamic>::into)
|
||||||
.map_err(|err| err.fill_position(args_expr[0].position()));
|
.map_err(|err| err.fill_position(args_expr[0].position()));
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
// Handle curry()
|
// Handle curry()
|
||||||
if name == KEYWORD_FN_PTR_CURRY && args_expr.len() > 1 {
|
if name == KEYWORD_FN_PTR_CURRY && args_expr.len() > 1 {
|
||||||
@ -1119,9 +1061,6 @@ impl Engine {
|
|||||||
|
|
||||||
// Handle is_def_var()
|
// Handle is_def_var()
|
||||||
if name == KEYWORD_IS_DEF_VAR && args_expr.len() == 1 {
|
if name == KEYWORD_IS_DEF_VAR && args_expr.len() == 1 {
|
||||||
let hash_fn = calc_native_fn_hash(empty(), name, once(TypeId::of::<ImmutableString>()));
|
|
||||||
|
|
||||||
if !self.has_override(Some(mods), Some(state), lib, hash_fn, hash_script) {
|
|
||||||
let var_name =
|
let var_name =
|
||||||
self.eval_expr(scope, mods, state, lib, this_ptr, &args_expr[0], level)?;
|
self.eval_expr(scope, mods, state, lib, this_ptr, &args_expr[0], level)?;
|
||||||
let var_name = var_name.as_str().map_err(|err| {
|
let var_name = var_name.as_str().map_err(|err| {
|
||||||
@ -1129,24 +1068,18 @@ impl Engine {
|
|||||||
})?;
|
})?;
|
||||||
return Ok(scope.contains(var_name).into());
|
return Ok(scope.contains(var_name).into());
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
// Handle eval()
|
// Handle eval()
|
||||||
if name == KEYWORD_EVAL && args_expr.len() == 1 {
|
if name == KEYWORD_EVAL && args_expr.len() == 1 {
|
||||||
let hash_fn = calc_native_fn_hash(empty(), name, once(TypeId::of::<ImmutableString>()));
|
|
||||||
|
|
||||||
let script_expr = &args_expr[0];
|
let script_expr = &args_expr[0];
|
||||||
|
|
||||||
if !self.has_override(Some(mods), Some(state), lib, hash_fn, hash_script) {
|
|
||||||
let script_pos = script_expr.position();
|
let script_pos = script_expr.position();
|
||||||
|
|
||||||
// eval - only in function call style
|
// eval - only in function call style
|
||||||
let prev_len = scope.len();
|
let prev_len = scope.len();
|
||||||
let script =
|
let script = self.eval_expr(scope, mods, state, lib, this_ptr, script_expr, level)?;
|
||||||
self.eval_expr(scope, mods, state, lib, this_ptr, script_expr, level)?;
|
let script = script
|
||||||
let script = script.as_str().map_err(|typ| {
|
.as_str()
|
||||||
self.make_type_mismatch_err::<ImmutableString>(typ, script_pos)
|
.map_err(|typ| self.make_type_mismatch_err::<ImmutableString>(typ, script_pos))?;
|
||||||
})?;
|
|
||||||
let result = self.eval_script_expr_in_place(
|
let result = self.eval_script_expr_in_place(
|
||||||
scope,
|
scope,
|
||||||
mods,
|
mods,
|
||||||
@ -1176,7 +1109,6 @@ impl Engine {
|
|||||||
))
|
))
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
// Normal function call - except for Fn, curry, call and eval (handled above)
|
// Normal function call - except for Fn, curry, call and eval (handled above)
|
||||||
let mut arg_values: StaticVec<_>;
|
let mut arg_values: StaticVec<_>;
|
||||||
|
@ -673,7 +673,7 @@ fn optimize_expr(expr: &mut Expr, state: &mut State) {
|
|||||||
let arg_types: StaticVec<_> = arg_values.iter().map(Dynamic::type_id).collect();
|
let arg_types: StaticVec<_> = arg_values.iter().map(Dynamic::type_id).collect();
|
||||||
|
|
||||||
// Search for overloaded operators (can override built-in).
|
// Search for overloaded operators (can override built-in).
|
||||||
if !state.engine.has_override_by_name_and_arguments(Some(&state.mods), None, state.lib, x.name.as_ref(), arg_types.as_ref()) {
|
if !state.engine.has_override_by_name_and_arguments(Some(&state.mods), state.lib, x.name.as_ref(), arg_types.as_ref()) {
|
||||||
if let Some(result) = run_builtin_binary_op(x.name.as_ref(), &arg_values[0], &arg_values[1])
|
if let Some(result) = run_builtin_binary_op(x.name.as_ref(), &arg_values[0], &arg_values[1])
|
||||||
.ok().flatten()
|
.ok().flatten()
|
||||||
.and_then(|result| map_dynamic_to_expr(result, *pos))
|
.and_then(|result| map_dynamic_to_expr(result, *pos))
|
||||||
|
@ -33,7 +33,7 @@ mod fn_ptr_functions {
|
|||||||
let hash_script = calc_script_fn_hash(empty(), fn_name, num_params as usize);
|
let hash_script = calc_script_fn_hash(empty(), fn_name, num_params as usize);
|
||||||
|
|
||||||
ctx.engine()
|
ctx.engine()
|
||||||
.has_override(ctx.mods, None, ctx.lib, None, hash_script)
|
.has_override(ctx.mods, ctx.lib, None, hash_script)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
11
src/token.rs
11
src/token.rs
@ -759,7 +759,6 @@ impl Token {
|
|||||||
#[cfg(not(feature = "no_function"))]
|
#[cfg(not(feature = "no_function"))]
|
||||||
pub(crate) fn into_function_name_for_override(self) -> Result<String, Self> {
|
pub(crate) fn into_function_name_for_override(self) -> Result<String, Self> {
|
||||||
match self {
|
match self {
|
||||||
Self::Reserved(s) if can_override_keyword(&s) => Ok(s),
|
|
||||||
Self::Custom(s) | Self::Identifier(s) if is_valid_identifier(s.chars()) => Ok(s),
|
Self::Custom(s) | Self::Identifier(s) if is_valid_identifier(s.chars()) => Ok(s),
|
||||||
_ => Err(self),
|
_ => Err(self),
|
||||||
}
|
}
|
||||||
@ -1630,16 +1629,6 @@ pub fn is_keyword_function(name: &str) -> bool {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Can this keyword be overridden as a function?
|
|
||||||
#[cfg(not(feature = "no_function"))]
|
|
||||||
#[inline(always)]
|
|
||||||
pub fn can_override_keyword(name: &str) -> bool {
|
|
||||||
match name {
|
|
||||||
KEYWORD_PRINT | KEYWORD_DEBUG | KEYWORD_TYPE_OF | KEYWORD_EVAL | KEYWORD_FN_PTR => true,
|
|
||||||
_ => false,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Is a text string a valid identifier?
|
/// Is a text string a valid identifier?
|
||||||
pub fn is_valid_identifier(name: impl Iterator<Item = char>) -> bool {
|
pub fn is_valid_identifier(name: impl Iterator<Item = char>) -> bool {
|
||||||
let mut first_alphabetic = false;
|
let mut first_alphabetic = false;
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
use rhai::{Engine, EvalAltResult, LexError, ParseErrorType, RegisterFn, Scope, INT};
|
use rhai::{Engine, EvalAltResult, LexError, ParseErrorType, Scope, INT};
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_eval() -> Result<(), Box<EvalAltResult>> {
|
fn test_eval() -> Result<(), Box<EvalAltResult>> {
|
||||||
@ -80,32 +80,6 @@ fn test_eval_function() -> Result<(), Box<EvalAltResult>> {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[cfg(not(feature = "no_function"))]
|
|
||||||
fn test_eval_override() -> Result<(), Box<EvalAltResult>> {
|
|
||||||
let engine = Engine::new();
|
|
||||||
|
|
||||||
assert_eq!(
|
|
||||||
engine.eval::<String>(
|
|
||||||
r#"
|
|
||||||
fn eval(x) { x } // reflect the script back
|
|
||||||
|
|
||||||
eval("40 + 2")
|
|
||||||
"#
|
|
||||||
)?,
|
|
||||||
"40 + 2"
|
|
||||||
);
|
|
||||||
|
|
||||||
let mut engine = Engine::new();
|
|
||||||
|
|
||||||
// Reflect the script back
|
|
||||||
engine.register_fn("eval", |script: &str| script.to_string());
|
|
||||||
|
|
||||||
assert_eq!(engine.eval::<String>(r#"eval("40 + 2")"#)?, "40 + 2");
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_eval_disabled() -> Result<(), Box<EvalAltResult>> {
|
fn test_eval_disabled() -> Result<(), Box<EvalAltResult>> {
|
||||||
let mut engine = Engine::new();
|
let mut engine = Engine::new();
|
||||||
|
Loading…
Reference in New Issue
Block a user