//! Built-in implementations for common operators. use crate::fn_native::{FnCallArgs, NativeCallContext}; use crate::stdlib::{any::TypeId, format, string::ToString}; use crate::{Dynamic, ImmutableString, RhaiResult, INT}; #[cfg(not(feature = "no_float"))] use crate::FLOAT; #[cfg(feature = "decimal")] use rust_decimal::Decimal; #[cfg(feature = "no_std")] #[cfg(not(feature = "no_float"))] use num_traits::float::Float; /// Build in common binary operator implementations to avoid the cost of calling a registered function. pub fn get_builtin_binary_op_fn( op: &str, x: &Dynamic, y: &Dynamic, ) -> Option RhaiResult> { let type1 = x.type_id(); let type2 = y.type_id(); if x.is_variant() || y.is_variant() { // One of the operands is a custom type, so it is never built-in return match op { "!=" if type1 != type2 => Some(|_, _| Ok(Dynamic::TRUE)), "==" | ">" | ">=" | "<" | "<=" if type1 != type2 => Some(|_, _| Ok(Dynamic::FALSE)), _ => None, }; } let types_pair = (type1, type2); #[cfg(not(feature = "no_float"))] if types_pair == (TypeId::of::(), TypeId::of::()) || types_pair == (TypeId::of::(), TypeId::of::()) || types_pair == (TypeId::of::(), TypeId::of::()) { #[inline(always)] fn get_xy(args: &FnCallArgs) -> (FLOAT, FLOAT) { let type1 = args[0].type_id(); let type2 = args[1].type_id(); let types_pair = (type1, type2); if types_pair == (TypeId::of::(), TypeId::of::()) { // FLOAT op FLOAT ( args[0].clone().cast::(), args[1].clone().cast::(), ) } else if types_pair == (TypeId::of::(), TypeId::of::()) { // FLOAT op INT ( args[0].clone().cast::(), args[1].clone().cast::() as FLOAT, ) } else if types_pair == (TypeId::of::(), TypeId::of::()) { // INT op FLOAT ( args[0].clone().cast::() as FLOAT, args[1].clone().cast::(), ) } else { unreachable!() } } match op { "+" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x + y).into()) }) } "-" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x - y).into()) }) } "*" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x * y).into()) }) } "/" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x / y).into()) }) } "%" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x % y).into()) }) } "**" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok(x.powf(y).into()) }) } "==" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x == y).into()) }) } "!=" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x != y).into()) }) } ">" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x > y).into()) }) } ">=" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x >= y).into()) }) } "<" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x < y).into()) }) } "<=" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x <= y).into()) }) } _ => return None, } } #[cfg(feature = "decimal")] if types_pair == (TypeId::of::(), TypeId::of::()) || types_pair == (TypeId::of::(), TypeId::of::()) || types_pair == (TypeId::of::(), TypeId::of::()) { #[inline(always)] fn get_xy(args: &FnCallArgs) -> (Decimal, Decimal) { let type1 = args[0].type_id(); let type2 = args[1].type_id(); let types_pair = (type1, type2); if types_pair == (TypeId::of::(), TypeId::of::()) { // Decimal op Decimal ( args[0].clone().cast::(), args[1].clone().cast::(), ) } else if types_pair == (TypeId::of::(), TypeId::of::()) { // Decimal op INT ( args[0].clone().cast::(), Decimal::from(args[1].clone().cast::()), ) } else if types_pair == (TypeId::of::(), TypeId::of::()) { // INT op Decimal ( Decimal::from(args[0].clone().cast::()), args[1].clone().cast::(), ) } else { unreachable!() } } if cfg!(not(feature = "unchecked")) { use crate::packages::arithmetic::decimal_functions::*; match op { "+" => { return Some(|_, args| { let (x, y) = get_xy(args); add(x, y) }) } "-" => { return Some(|_, args| { let (x, y) = get_xy(args); subtract(x, y) }) } "*" => { return Some(|_, args| { let (x, y) = get_xy(args); multiply(x, y) }) } "/" => { return Some(|_, args| { let (x, y) = get_xy(args); divide(x, y) }) } "%" => { return Some(|_, args| { let (x, y) = get_xy(args); modulo(x, y) }) } _ => (), } } else { match op { "+" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x + y).into()) }) } "-" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x - y).into()) }) } "*" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x * y).into()) }) } "/" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x / y).into()) }) } "%" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x % y).into()) }) } _ => (), } } match op { "==" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x == y).into()) }) } "!=" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x != y).into()) }) } ">" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x > y).into()) }) } ">=" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x >= y).into()) }) } "<" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x < y).into()) }) } "<=" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x <= y).into()) }) } _ => return None, } } // char op string if types_pair == (TypeId::of::(), TypeId::of::()) { match op { "+" => { return Some(|_, args| { let x = args[0].clone().cast::(); let y = &*args[1].read_lock::().unwrap(); Ok(format!("{}{}", x, y).into()) }) } "==" | "!=" | ">" | ">=" | "<" | "<=" => { #[inline(always)] fn get_s1s2(args: &FnCallArgs) -> ([char; 2], [char; 2]) { let x = args[0].clone().cast::(); let y = &*args[1].read_lock::().unwrap(); let s1 = [x, '\0']; let mut y = y.chars(); let s2 = [y.next().unwrap_or('\0'), y.next().unwrap_or('\0')]; (s1, s2) } match op { "==" => { return Some(|_, args| { let (s1, s2) = get_s1s2(args); Ok((s1 == s2).into()) }) } "!=" => { return Some(|_, args| { let (s1, s2) = get_s1s2(args); Ok((s1 != s2).into()) }) } ">" => { return Some(|_, args| { let (s1, s2) = get_s1s2(args); Ok((s1 > s2).into()) }) } ">=" => { return Some(|_, args| { let (s1, s2) = get_s1s2(args); Ok((s1 >= s2).into()) }) } "<" => { return Some(|_, args| { let (s1, s2) = get_s1s2(args); Ok((s1 < s2).into()) }) } "<=" => { return Some(|_, args| { let (s1, s2) = get_s1s2(args); Ok((s1 <= s2).into()) }) } _ => unreachable!(), } } _ => return None, } } // string op char if types_pair == (TypeId::of::(), TypeId::of::()) { match op { "+" => { return Some(|_, args| { let x = &*args[0].read_lock::().unwrap(); let y = args[1].clone().cast::(); Ok((x + y).into()) }) } "-" => { return Some(|_, args| { let x = &*args[0].read_lock::().unwrap(); let y = args[1].clone().cast::(); Ok((x - y).into()) }) } "==" | "!=" | ">" | ">=" | "<" | "<=" => { #[inline(always)] fn get_s1s2(args: &FnCallArgs) -> ([char; 2], [char; 2]) { let x = &*args[0].read_lock::().unwrap(); let y = args[1].clone().cast::(); let mut x = x.chars(); let s1 = [x.next().unwrap_or('\0'), x.next().unwrap_or('\0')]; let s2 = [y, '\0']; (s1, s2) } match op { "==" => { return Some(|_, args| { let (s1, s2) = get_s1s2(args); Ok((s1 == s2).into()) }) } "!=" => { return Some(|_, args| { let (s1, s2) = get_s1s2(args); Ok((s1 != s2).into()) }) } ">" => { return Some(|_, args| { let (s1, s2) = get_s1s2(args); Ok((s1 > s2).into()) }) } ">=" => { return Some(|_, args| { let (s1, s2) = get_s1s2(args); Ok((s1 >= s2).into()) }) } "<" => { return Some(|_, args| { let (s1, s2) = get_s1s2(args); Ok((s1 < s2).into()) }) } "<=" => { return Some(|_, args| { let (s1, s2) = get_s1s2(args); Ok((s1 <= s2).into()) }) } _ => unreachable!(), } } _ => return None, } } // Default comparison operators for different types if type2 != type1 { return match op { "!=" => Some(|_, _| Ok(Dynamic::TRUE)), "==" | ">" | ">=" | "<" | "<=" => Some(|_, _| Ok(Dynamic::FALSE)), _ => None, }; } // Beyond here, type1 == type2 if type1 == TypeId::of::() { #[inline(always)] fn get_xy(args: &FnCallArgs) -> (INT, INT) { let x = args[0].clone().cast::(); let y = args[1].clone().cast::(); (x, y) } if cfg!(not(feature = "unchecked")) { use crate::packages::arithmetic::arith_basic::INT::functions::*; match op { "+" => { return Some(|_, args| { let (x, y) = get_xy(args); add(x, y) }) } "-" => { return Some(|_, args| { let (x, y) = get_xy(args); subtract(x, y) }) } "*" => { return Some(|_, args| { let (x, y) = get_xy(args); multiply(x, y) }) } "/" => { return Some(|_, args| { let (x, y) = get_xy(args); divide(x, y) }) } "%" => { return Some(|_, args| { let (x, y) = get_xy(args); modulo(x, y) }) } "**" => { return Some(|_, args| { let (x, y) = get_xy(args); power(x, y) }) } ">>" => { return Some(|_, args| { let (x, y) = get_xy(args); shift_right(x, y) }) } "<<" => { return Some(|_, args| { let (x, y) = get_xy(args); shift_left(x, y) }) } _ => (), } } else { match op { "+" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x + y).into()) }) } "-" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x - y).into()) }) } "*" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x * y).into()) }) } "/" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x / y).into()) }) } "%" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x % y).into()) }) } "**" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok(x.pow(y as u32).into()) }) } ">>" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x >> y).into()) }) } "<<" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x << y).into()) }) } _ => (), } } match op { "==" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x == y).into()) }) } "!=" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x != y).into()) }) } ">" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x > y).into()) }) } ">=" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x >= y).into()) }) } "<" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x < y).into()) }) } "<=" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x <= y).into()) }) } "&" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x & y).into()) }) } "|" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x | y).into()) }) } "^" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x ^ y).into()) }) } _ => return None, } } if type1 == TypeId::of::() { #[inline(always)] fn get_xy(args: &FnCallArgs) -> (bool, bool) { let x = args[0].clone().cast::(); let y = args[1].clone().cast::(); (x, y) } match op { "&" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x && y).into()) }) } "|" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x || y).into()) }) } "^" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x ^ y).into()) }) } "==" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x == y).into()) }) } "!=" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x != y).into()) }) } _ => return None, } } if type1 == TypeId::of::() { match op { "+" => { return Some(|_, args| { let x = &*args[0].read_lock::().unwrap(); let y = &*args[1].read_lock::().unwrap(); Ok((x + y).into()) }) } "-" => { return Some(|_, args| { let x = &*args[0].read_lock::().unwrap(); let y = &*args[1].read_lock::().unwrap(); Ok((x - y).into()) }) } "==" => { return Some(|_, args| { let x = &*args[0].read_lock::().unwrap(); let y = &*args[1].read_lock::().unwrap(); Ok((x == y).into()) }) } "!=" => { return Some(|_, args| { let x = &*args[0].read_lock::().unwrap(); let y = &*args[1].read_lock::().unwrap(); Ok((x != y).into()) }) } ">" => { return Some(|_, args| { let x = &*args[0].read_lock::().unwrap(); let y = &*args[1].read_lock::().unwrap(); Ok((x > y).into()) }) } ">=" => { return Some(|_, args| { let x = &*args[0].read_lock::().unwrap(); let y = &*args[1].read_lock::().unwrap(); Ok((x >= y).into()) }) } "<" => { return Some(|_, args| { let x = &*args[0].read_lock::().unwrap(); let y = &*args[1].read_lock::().unwrap(); Ok((x < y).into()) }) } "<=" => { return Some(|_, args| { let x = &*args[0].read_lock::().unwrap(); let y = &*args[1].read_lock::().unwrap(); Ok((x <= y).into()) }) } _ => return None, } } if type1 == TypeId::of::() { #[inline(always)] fn get_xy(args: &FnCallArgs) -> (char, char) { let x = args[0].clone().cast::(); let y = args[1].clone().cast::(); (x, y) } match op { "+" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok(format!("{}{}", x, y).into()) }) } "==" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x == y).into()) }) } "!=" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x != y).into()) }) } ">" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x > y).into()) }) } ">=" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x >= y).into()) }) } "<" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x < y).into()) }) } "<=" => { return Some(|_, args| { let (x, y) = get_xy(args); Ok((x <= y).into()) }) } _ => return None, } } if type1 == TypeId::of::<()>() { match op { "==" => return Some(|_, _| Ok(Dynamic::TRUE)), "!=" | ">" | ">=" | "<" | "<=" => return Some(|_, _| Ok(Dynamic::FALSE)), _ => return None, } } None } /// Build in common operator assignment implementations to avoid the cost of calling a registered function. pub fn get_builtin_op_assignment_fn( op: &str, x: &Dynamic, y: &Dynamic, ) -> Option RhaiResult> { let type1 = x.type_id(); let type2 = y.type_id(); let types_pair = (type1, type2); #[cfg(not(feature = "no_float"))] if types_pair == (TypeId::of::(), TypeId::of::()) || types_pair == (TypeId::of::(), TypeId::of::()) { fn get_y(args: &FnCallArgs) -> FLOAT { let type2 = args[1].type_id(); if type2 == TypeId::of::() { // FLOAT op= FLOAT args[1].clone().cast::() } else if type2 == TypeId::of::() { // FLOAT op= INT args[1].clone().cast::() as FLOAT } else { unreachable!(); } } match op { "+=" => { return Some(|_, args| { let y = get_y(args); Ok((*args[0].write_lock::().unwrap() += y).into()) }) } "-=" => { return Some(|_, args| { let y = get_y(args); Ok((*args[0].write_lock::().unwrap() -= y).into()) }) } "*=" => { return Some(|_, args| { let y = get_y(args); Ok((*args[0].write_lock::().unwrap() *= y).into()) }) } "/=" => { return Some(|_, args| { let y = get_y(args); Ok((*args[0].write_lock::().unwrap() /= y).into()) }) } "%=" => { return Some(|_, args| { let y = get_y(args); Ok((*args[0].write_lock::().unwrap() %= y).into()) }) } "**=" => { return Some(|_, args| { let y = get_y(args); let mut x = args[0].write_lock::().unwrap(); Ok((*x = x.powf(y)).into()) }) } _ => return None, } } #[cfg(feature = "decimal")] if types_pair == (TypeId::of::(), TypeId::of::()) || types_pair == (TypeId::of::(), TypeId::of::()) { fn get_y(args: &FnCallArgs) -> Decimal { let type2 = args[1].type_id(); if type2 == TypeId::of::() { // Decimal op= Decimal args[1].clone().cast::() } else if type2 == TypeId::of::() { // Decimal op= INT Decimal::from(args[1].clone().cast::()) } else { unreachable!(); } } if cfg!(not(feature = "unchecked")) { use crate::packages::arithmetic::decimal_functions::*; match op { "+=" => { return Some(|_, args| { let y = args[1].clone().cast::(); let mut x = args[0].write_lock::().unwrap(); Ok((*x = add(*x, y)?.as_int().unwrap().into()).into()) }) } "-=" => { return Some(|_, args| { let y = args[1].clone().cast::(); let mut x = args[0].write_lock::().unwrap(); Ok((*x = subtract(*x, y)?.as_int().unwrap().into()).into()) }) } "*=" => { return Some(|_, args| { let y = args[1].clone().cast::(); let mut x = args[0].write_lock::().unwrap(); Ok((*x = multiply(*x, y)?.as_int().unwrap().into()).into()) }) } "/=" => { return Some(|_, args| { let y = args[1].clone().cast::(); let mut x = args[0].write_lock::().unwrap(); Ok((*x = divide(*x, y)?.as_int().unwrap().into()).into()) }) } "%=" => { return Some(|_, args| { let y = args[1].clone().cast::(); let mut x = args[0].write_lock::().unwrap(); Ok((*x = modulo(*x, y)?.as_int().unwrap().into()).into()) }) } _ => (), } } else { match op { "+=" => { return Some(|_, args| { let y = get_y(args); Ok((*args[0].write_lock::().unwrap() += y).into()) }) } "-=" => { return Some(|_, args| { let y = get_y(args); Ok((*args[0].write_lock::().unwrap() -= y).into()) }) } "*=" => { return Some(|_, args| { let y = get_y(args); Ok((*args[0].write_lock::().unwrap() *= y).into()) }) } "/=" => { return Some(|_, args| { let y = get_y(args); Ok((*args[0].write_lock::().unwrap() /= y).into()) }) } "%=" => { return Some(|_, args| { let y = get_y(args); Ok((*args[0].write_lock::().unwrap() %= y).into()) }) } _ => return None, } } } // string op= char if types_pair == (TypeId::of::(), TypeId::of::()) { match op { "+=" => { return Some(|_, args| { let y = args[1].clone().cast::(); Ok((*args[0].write_lock::().unwrap() += y).into()) }) } "-=" => { return Some(|_, args| { let y = args[1].clone().cast::(); Ok((*args[0].write_lock::().unwrap() -= y).into()) }) } _ => return None, } } // char op= string if types_pair == (TypeId::of::(), TypeId::of::()) { match op { "+=" => { return Some(|_, args| { let mut ch = args[0].read_lock::().unwrap().to_string(); ch.push_str(args[1].read_lock::().unwrap().as_str()); let mut x = args[0].write_lock::().unwrap(); Ok((*x = ch.into()).into()) }) } _ => return None, } } // No built-in op-assignments for different types. if type2 != type1 { return None; } // Beyond here, type1 == type2 if type1 == TypeId::of::() { if cfg!(not(feature = "unchecked")) { use crate::packages::arithmetic::arith_basic::INT::functions::*; match op { "+=" => { return Some(|_, args| { let y = args[1].clone().cast::(); let mut x = args[0].write_lock::().unwrap(); Ok((*x = add(*x, y)?.as_int().unwrap().into()).into()) }) } "-=" => { return Some(|_, args| { let y = args[1].clone().cast::(); let mut x = args[0].write_lock::().unwrap(); Ok((*x = subtract(*x, y)?.as_int().unwrap().into()).into()) }) } "*=" => { return Some(|_, args| { let y = args[1].clone().cast::(); let mut x = args[0].write_lock::().unwrap(); Ok((*x = multiply(*x, y)?.as_int().unwrap().into()).into()) }) } "/=" => { return Some(|_, args| { let y = args[1].clone().cast::(); let mut x = args[0].write_lock::().unwrap(); Ok((*x = divide(*x, y)?.as_int().unwrap().into()).into()) }) } "%=" => { return Some(|_, args| { let y = args[1].clone().cast::(); let mut x = args[0].write_lock::().unwrap(); Ok((*x = modulo(*x, y)?.as_int().unwrap().into()).into()) }) } "**=" => { return Some(|_, args| { let y = args[1].clone().cast::(); let mut x = args[0].write_lock::().unwrap(); Ok((*x = power(*x, y)?.as_int().unwrap().into()).into()) }) } ">>=" => { return Some(|_, args| { let y = args[1].clone().cast::(); let mut x = args[0].write_lock::().unwrap(); Ok((*x = shift_right(*x, y)?.as_int().unwrap().into()).into()) }) } "<<=" => { return Some(|_, args| { let y = args[1].clone().cast::(); let mut x = args[0].write_lock::().unwrap(); Ok((*x = shift_left(*x, y)?.as_int().unwrap().into()).into()) }) } _ => (), } } else { match op { "+=" => { return Some(|_, args| { let y = args[1].clone().cast::(); Ok((*args[0].write_lock::().unwrap() += y).into()) }) } "-=" => { return Some(|_, args| { let y = args[1].clone().cast::(); Ok((*args[0].write_lock::().unwrap() -= y).into()) }) } "*=" => { return Some(|_, args| { let y = args[1].clone().cast::(); Ok((*args[0].write_lock::().unwrap() *= y).into()) }) } "/=" => { return Some(|_, args| { let y = args[1].clone().cast::(); Ok((*args[0].write_lock::().unwrap() /= y).into()) }) } "%=" => { return Some(|_, args| { let y = args[1].clone().cast::(); Ok((*args[0].write_lock::().unwrap() %= y).into()) }) } "**=" => { return Some(|_, args| { let y = args[1].clone().cast::(); let mut x = args[0].write_lock::().unwrap(); Ok((*x = x.pow(y as u32)).into()) }) } ">>=" => { return Some(|_, args| { let y = args[1].clone().cast::(); Ok((*args[0].write_lock::().unwrap() >>= y).into()) }) } "<<=" => { return Some(|_, args| { let y = args[1].clone().cast::(); Ok((*args[0].write_lock::().unwrap() <<= y).into()) }) } _ => (), } } match op { "&=" => { return Some(|_, args| { let y = args[1].clone().cast::(); Ok((*args[0].write_lock::().unwrap() &= y).into()) }) } "|=" => { return Some(|_, args| { let y = args[1].clone().cast::(); Ok((*args[0].write_lock::().unwrap() |= y).into()) }) } "^=" => { return Some(|_, args| { let y = args[1].clone().cast::(); Ok((*args[0].write_lock::().unwrap() ^= y).into()) }) } _ => return None, } } if type1 == TypeId::of::() { match op { "&=" => { return Some(|_, args| { let y = args[1].clone().cast::(); let mut x = args[0].write_lock::().unwrap(); Ok((*x = *x && y).into()) }) } "|=" => { return Some(|_, args| { let y = args[1].clone().cast::(); let mut x = args[0].write_lock::().unwrap(); Ok((*x = *x || y).into()) }) } _ => return None, } } if type1 == TypeId::of::() { match op { "+=" => { return Some(|_, args| { let y = args[1].clone().cast::(); let mut x = args[0].write_lock::().unwrap(); Ok((*x = format!("{}{}", *x, y).into()).into()) }) } _ => return None, } } if type1 == TypeId::of::() { match op { "+=" => { return Some(|_, args| { let (first, second) = args.split_first_mut().unwrap(); let mut x = first.write_lock::().unwrap(); let y = &*second[0].read_lock::().unwrap(); Ok((*x += y).into()) }) } "-=" => { return Some(|_, args| { let (first, second) = args.split_first_mut().unwrap(); let mut x = first.write_lock::().unwrap(); let y = &*second[0].read_lock::().unwrap(); Ok((*x -= y).into()) }) } _ => return None, } } None }