#[cfg(feature = "regex_support")] use regex::Regex; use crate::{ value::{FloatType, IntType}, EvalexprError, Function, Value, ValueType, }; use std::ops::{BitAnd, BitOr, BitXor, Not, Shl, Shr}; macro_rules! simple_math { ($func:ident) => { Some(Function::new(|argument| { let num = argument.as_number()?; Ok(Value::Float(num.$func())) })) }; ($func:ident, 2) => { Some(Function::new(|argument| { let tuple = argument.as_fixed_len_tuple(2)?; let (a, b) = (tuple[0].as_number()?, tuple[1].as_number()?); Ok(Value::Float(a.$func(b))) })) }; } macro_rules! int_function { ($func:ident) => { Some(Function::new(|argument| { let int = argument.as_int()?; Ok(Value::Int(int.$func())) })) }; ($func:ident, 2) => { Some(Function::new(|argument| { let tuple = argument.as_fixed_len_tuple(2)?; let (a, b) = (tuple[0].as_int()?, tuple[1].as_int()?); Ok(Value::Int(a.$func(b))) })) }; } pub fn builtin_function(identifier: &str) -> Option { match identifier { // Log "math::ln" => simple_math!(ln), "math::log" => simple_math!(log, 2), "math::log2" => simple_math!(log2), "math::log10" => simple_math!(log10), // Exp "math::exp" => simple_math!(exp), "math::exp2" => simple_math!(exp2), // Pow "math::pow" => simple_math!(powf, 2), // Cos "math::cos" => simple_math!(cos), "math::acos" => simple_math!(acos), "math::cosh" => simple_math!(cosh), "math::acosh" => simple_math!(acosh), // Sin "math::sin" => simple_math!(sin), "math::asin" => simple_math!(asin), "math::sinh" => simple_math!(sinh), "math::asinh" => simple_math!(asinh), // Tan "math::tan" => simple_math!(tan), "math::atan" => simple_math!(atan), "math::tanh" => simple_math!(tanh), "math::atanh" => simple_math!(atanh), "math::atan2" => simple_math!(atan2, 2), // Root "math::sqrt" => simple_math!(sqrt), "math::cbrt" => simple_math!(cbrt), // Hypotenuse "math::hypot" => simple_math!(hypot, 2), // Rounding "floor" => simple_math!(floor), "round" => simple_math!(round), "ceil" => simple_math!(ceil), // Other "min" => Some(Function::new(|argument| { let arguments = argument.as_tuple()?; let mut min_int = IntType::max_value(); let mut min_float = 1.0f64 / 0.0f64; debug_assert!(min_float.is_infinite()); for argument in arguments { if let Value::Float(float) = argument { min_float = min_float.min(float); } else if let Value::Int(int) = argument { min_int = min_int.min(int); } else { return Err(EvalexprError::expected_number(argument)); } } if (min_int as FloatType) < min_float { Ok(Value::Int(min_int)) } else { Ok(Value::Float(min_float)) } })), "max" => Some(Function::new(|argument| { let arguments = argument.as_tuple()?; let mut max_int = IntType::min_value(); let mut max_float = -1.0f64 / 0.0f64; debug_assert!(max_float.is_infinite()); for argument in arguments { if let Value::Float(float) = argument { max_float = max_float.max(float); } else if let Value::Int(int) = argument { max_int = max_int.max(int); } else { return Err(EvalexprError::expected_number(argument)); } } if (max_int as FloatType) > max_float { Ok(Value::Int(max_int)) } else { Ok(Value::Float(max_float)) } })), "if" => Some(Function::new(|argument| { if let [condition, if_true, if_false] = &argument.as_fixed_len_tuple(3)?[..] { return Ok(if condition.as_boolean()? { if_true } else { if_false } .clone()); } Err(EvalexprError::type_error( argument.clone(), vec![ValueType::Boolean, ValueType::Empty, ValueType::Empty], )) })), "len" => Some(Function::new(|argument| { if let Ok(subject) = argument.as_string() { Ok(Value::from(subject.len() as i64)) } else if let Ok(subject) = argument.as_tuple() { Ok(Value::from(subject.len() as i64)) } else { Err(EvalexprError::type_error( argument.clone(), vec![ValueType::String, ValueType::Tuple], )) } })), // String functions #[cfg(feature = "regex_support")] "str::regex_matches" => Some(Function::new(|argument| { let arguments = argument.as_tuple()?; let subject = arguments[0].as_string()?; let re_str = arguments[1].as_string()?; match Regex::new(&re_str) { Ok(re) => Ok(Value::Boolean(re.is_match(&subject))), Err(err) => Err(EvalexprError::invalid_regex( re_str.to_string(), format!("{}", err), )), } })), #[cfg(feature = "regex_support")] "str::regex_replace" => Some(Function::new(|argument| { let arguments = argument.as_tuple()?; let subject = arguments[0].as_string()?; let re_str = arguments[1].as_string()?; let repl = arguments[2].as_string()?; match Regex::new(&re_str) { Ok(re) => Ok(Value::String( re.replace_all(&subject, repl.as_str()).to_string(), )), Err(err) => Err(EvalexprError::invalid_regex( re_str.to_string(), format!("{}", err), )), } })), "str::to_lowercase" => Some(Function::new(|argument| { let subject = argument.as_string()?; Ok(Value::from(subject.to_lowercase())) })), "str::to_uppercase" => Some(Function::new(|argument| { let subject = argument.as_string()?; Ok(Value::from(subject.to_uppercase())) })), "str::trim" => Some(Function::new(|argument| { let subject = argument.as_string()?; Ok(Value::from(subject.trim())) })), "str::from" => Some(Function::new(|argument| { Ok(Value::String(argument.to_string())) })), // Bitwise operators "bitand" => int_function!(bitand, 2), "bitor" => int_function!(bitor, 2), "bitxor" => int_function!(bitxor, 2), "bitnot" => int_function!(not), "shl" => int_function!(shl, 2), "shr" => int_function!(shr, 2), _ => None, } }