2019-03-28 14:39:50 +00:00
|
|
|
use std::fmt;
|
|
|
|
|
2019-08-29 07:02:05 +00:00
|
|
|
use crate::{error::EvalexprResult, value::Value};
|
2019-03-15 18:26:25 +00:00
|
|
|
|
2019-03-20 09:32:49 +00:00
|
|
|
pub(crate) mod builtin;
|
2019-03-19 18:08:23 +00:00
|
|
|
|
2021-08-15 17:42:20 +00:00
|
|
|
/// A helper trait to enable cloning through `Fn` trait objects.
|
|
|
|
trait ClonableFn
|
|
|
|
where
|
|
|
|
Self: Fn(&Value) -> EvalexprResult<Value>,
|
|
|
|
Self: Send + Sync + 'static,
|
|
|
|
{
|
|
|
|
fn dyn_clone(&self) -> Box<dyn ClonableFn>;
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<F> ClonableFn for F
|
|
|
|
where
|
|
|
|
F: Fn(&Value) -> EvalexprResult<Value>,
|
|
|
|
F: Send + Sync + 'static,
|
|
|
|
F: Clone,
|
|
|
|
{
|
|
|
|
fn dyn_clone(&self) -> Box<dyn ClonableFn> {
|
|
|
|
Box::new(self.clone()) as _
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-20 09:32:49 +00:00
|
|
|
/// A user-defined function.
|
2019-03-27 17:09:20 +00:00
|
|
|
/// Functions can be used in expressions by storing them in a `Context`.
|
2019-03-20 09:32:49 +00:00
|
|
|
///
|
|
|
|
/// # Examples
|
|
|
|
///
|
|
|
|
/// ```rust
|
|
|
|
/// use evalexpr::*;
|
|
|
|
///
|
2019-03-27 17:09:20 +00:00
|
|
|
/// let mut context = HashMapContext::new();
|
2021-05-28 11:12:26 +00:00
|
|
|
/// context.set_function("id".into(), Function::new(|argument| {
|
2019-05-04 11:43:29 +00:00
|
|
|
/// Ok(argument.clone())
|
2021-05-28 11:12:26 +00:00
|
|
|
/// })).unwrap(); // Do proper error handling here
|
2019-03-27 17:09:20 +00:00
|
|
|
/// assert_eq!(eval_with_context("id(4)", &context), Ok(Value::from(4)));
|
2019-03-20 09:32:49 +00:00
|
|
|
/// ```
|
2019-03-15 18:26:25 +00:00
|
|
|
pub struct Function {
|
2021-08-15 17:42:20 +00:00
|
|
|
function: Box<dyn ClonableFn>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Clone for Function {
|
|
|
|
fn clone(&self) -> Self {
|
|
|
|
Self {
|
|
|
|
function: self.function.dyn_clone(),
|
|
|
|
}
|
|
|
|
}
|
2019-03-18 17:21:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Function {
|
2019-03-20 09:32:49 +00:00
|
|
|
/// Creates a user-defined function.
|
|
|
|
///
|
2021-08-15 17:42:20 +00:00
|
|
|
/// The `function` is boxed for storage.
|
|
|
|
pub fn new<F>(function: F) -> Self
|
|
|
|
where
|
|
|
|
F: Fn(&Value) -> EvalexprResult<Value>,
|
|
|
|
F: Send + Sync + 'static,
|
|
|
|
F: Clone,
|
|
|
|
{
|
|
|
|
Self {
|
|
|
|
function: Box::new(function) as _,
|
|
|
|
}
|
2019-03-18 17:21:07 +00:00
|
|
|
}
|
|
|
|
|
2019-05-04 11:43:29 +00:00
|
|
|
pub(crate) fn call(&self, argument: &Value) -> EvalexprResult<Value> {
|
|
|
|
(self.function)(argument)
|
2019-03-18 17:21:07 +00:00
|
|
|
}
|
2019-03-18 16:02:45 +00:00
|
|
|
}
|
2019-03-28 14:39:50 +00:00
|
|
|
|
|
|
|
impl fmt::Debug for Function {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error> {
|
2019-05-04 11:54:19 +00:00
|
|
|
write!(f, "Function {{ [...] }}")
|
2019-03-28 14:39:50 +00:00
|
|
|
}
|
|
|
|
}
|
2021-05-28 11:27:41 +00:00
|
|
|
|
|
|
|
/// A trait to ensure a type is `Send` and `Sync`.
|
|
|
|
/// If implemented for a type, the crate will not compile if the type is not `Send` and `Sync`.
|
|
|
|
trait IsSendAndSync: Send + Sync {}
|
|
|
|
|
|
|
|
impl IsSendAndSync for Function {}
|