use rhai::{Dynamic, Engine, EvalAltResult, LexError, ParseError, ParseErrorType, Position, INT}; #[test] fn test_custom_syntax() -> Result<(), Box> { let mut engine = Engine::new(); engine.consume("while false {}")?; // Disable 'while' and make sure it still works with custom syntax engine.disable_symbol("while"); assert!(matches!( *engine.compile("while false {}").expect_err("should error").0, ParseErrorType::Reserved(err) if err == "while" )); assert!(matches!( *engine.compile("let while = 0").expect_err("should error").0, ParseErrorType::Reserved(err) if err == "while" )); engine.register_custom_syntax( &[ "exec", "[", "$ident$", ";", "$int$", "]", "->", "$block$", "while", "$expr$", ], true, |context, inputs| { let var_name = inputs[0].get_variable_name().unwrap().to_string(); let max = inputs[1].get_literal_value().unwrap().as_int().unwrap(); let stmt = inputs.get(2).unwrap(); let condition = inputs.get(3).unwrap(); context.scope_mut().push(var_name.clone(), 0 as INT); let mut count: INT = 0; loop { if count >= max { break; } context.eval_expression_tree(stmt)?; count += 1; context .scope_mut() .push(format!("{}{}", var_name, count), count); let stop = !context .eval_expression_tree(condition)? .as_bool() .map_err(|err| { Box::new(EvalAltResult::ErrorMismatchDataType( "bool".to_string(), err.to_string(), condition.position(), )) })?; if stop { break; } } Ok(count.into()) }, )?; assert_eq!( engine.eval::( " let x = 0; let foo = (exec [x;15] -> { x += 2 } while x < 42) * 10; foo " )?, 150 ); assert_eq!( engine.eval::( " let x = 0; exec [x;100] -> { x += 1 } while x < 42; x " )?, 42 ); assert_eq!( engine.eval::( " exec [x;100] -> { x += 1 } while x < 42; x " )?, 42 ); assert_eq!( engine.eval::( " let foo = 123; exec [x;15] -> { x += 1 } while x < 42; foo + x + x1 + x2 + x3 " )?, 144 ); // The first symbol must be an identifier assert_eq!( *engine .register_custom_syntax(&["!"], false, |_, _| Ok(Dynamic::UNIT)) .expect_err("should error") .0, ParseErrorType::BadInput(LexError::ImproperSymbol( "!".to_string(), "Improper symbol for custom syntax at position #1: '!'".to_string() )) ); Ok(()) } #[test] fn test_custom_syntax_raw() -> Result<(), Box> { let mut engine = Engine::new(); engine.register_custom_syntax_raw( "hello", |stream, _| match stream.len() { 0 => unreachable!(), 1 => Ok(Some("$ident$".into())), 2 => match stream[1].as_str() { "world" | "kitty" => Ok(None), s => Err(ParseError( Box::new(ParseErrorType::BadInput(LexError::ImproperSymbol( s.to_string(), Default::default(), ))), Position::NONE, )), }, _ => unreachable!(), }, true, |context, inputs| { context.scope_mut().push("foo", 999 as INT); Ok(match inputs[0].get_variable_name().unwrap() { "world" => 123 as INT, "kitty" => 42 as INT, _ => unreachable!(), } .into()) }, ); assert_eq!(engine.eval::("hello world")?, 123); assert_eq!(engine.eval::("hello kitty")?, 42); assert_eq!( engine.eval::("let foo = 0; (hello kitty) + foo")?, 1041 ); assert_eq!(engine.eval::("(hello kitty) + foo")?, 1041); assert_eq!( *engine.compile("hello hey").expect_err("should error").0, ParseErrorType::BadInput(LexError::ImproperSymbol("hey".to_string(), "".to_string())) ); Ok(()) }