feat(parser): sanitizer passes first test round

This commit is contained in:
gbrochar 2023-08-01 18:08:50 +02:00
parent 7d0528264d
commit d2b73945a7
1 changed files with 278 additions and 142 deletions

View File

@ -199,14 +199,30 @@ fn sanitize_tokens(tokens: Vec<Token>) -> Result<Vec<Token>, String> {
let token_type = get_token_type(&token); let token_type = get_token_type(&token);
match token_type { match token_type {
TokenType::OpenParenthesis() => open_close_equal_count.0 += 1, TokenType::OpenParenthesis() => open_close_equal_count.0 += 1,
TokenType::CloseParenthesis() => open_close_equal_count.1 += 1, TokenType::CloseParenthesis() => {
TokenType::Equal() => open_close_equal_count.2 += 1, open_close_equal_count.1 += 1;
if open_close_equal_count.1 > open_close_equal_count.0 {
return Err(format!("Error at token {:?}, closing parenthesis that was never opened", token));
}
},
TokenType::Equal() => {
open_close_equal_count.2 += 1;
if open_close_equal_count.0 != open_close_equal_count.1 {
return Err(format!("Error, query has a different number of opening and closing parentheses"));
}
open_close_equal_count.0 = 0;
open_close_equal_count.1 = 0;
},
_ => (), _ => (),
} }
if let Some(last_token_type) = last_token_type { if let Some(last_token_type) = last_token_type {
match (&last_token_type, &token_type) { match (&last_token_type, &token_type) {
(TokenType::Number(), TokenType::Variable()) => sanitized_tokens.push(Token::Multiplication()), (TokenType::Number(), TokenType::Variable()) => sanitized_tokens.push(Token::Multiplication()),
(TokenType::Variable(), TokenType::Number()) => sanitized_tokens.push(Token::Multiplication()),
(TokenType::Number(), TokenType::ImaginaryUnit()) => sanitized_tokens.push(Token::Multiplication()), (TokenType::Number(), TokenType::ImaginaryUnit()) => sanitized_tokens.push(Token::Multiplication()),
(TokenType::ImaginaryUnit(), TokenType::Number()) => sanitized_tokens.push(Token::Multiplication()),
(TokenType::ImaginaryUnit(), TokenType::Variable()) => sanitized_tokens.push(Token::Multiplication()),
(TokenType::Variable(), TokenType::ImaginaryUnit()) => sanitized_tokens.push(Token::Multiplication()),
(TokenType::Number(), TokenType::OpenParenthesis()) => sanitized_tokens.push(Token::Multiplication()), (TokenType::Number(), TokenType::OpenParenthesis()) => sanitized_tokens.push(Token::Multiplication()),
(TokenType::Variable(), TokenType::OpenParenthesis()) => sanitized_tokens.push(Token::Multiplication()), (TokenType::Variable(), TokenType::OpenParenthesis()) => sanitized_tokens.push(Token::Multiplication()),
(TokenType::ImaginaryUnit(), TokenType::OpenParenthesis()) => sanitized_tokens.push(Token::Multiplication()), (TokenType::ImaginaryUnit(), TokenType::OpenParenthesis()) => sanitized_tokens.push(Token::Multiplication()),
@ -218,6 +234,11 @@ fn sanitize_tokens(tokens: Vec<Token>) -> Result<Vec<Token>, String> {
(TokenType::Operator(), TokenType::CloseParenthesis()) => return Err(format!("Error at token {:?}, all operators are forbidden directly before closing parenthesis", token)), (TokenType::Operator(), TokenType::CloseParenthesis()) => return Err(format!("Error at token {:?}, all operators are forbidden directly before closing parenthesis", token)),
(TokenType::Substraction(), TokenType::CloseParenthesis()) => return Err(format!("Error at token {:?}, all operators are forbidden directly before closing parenthesis", token)), (TokenType::Substraction(), TokenType::CloseParenthesis()) => return Err(format!("Error at token {:?}, all operators are forbidden directly before closing parenthesis", token)),
(TokenType::OpenParenthesis(), TokenType::CloseParenthesis()) => return Err(format!("Error at token {:?}, empty parentheses", token)), (TokenType::OpenParenthesis(), TokenType::CloseParenthesis()) => return Err(format!("Error at token {:?}, empty parentheses", token)),
(TokenType::Substraction(), TokenType::Equal()) => return Err(format!("Error at token {:?}, two incompatible tokens type in a row", token)),
(TokenType::Substraction(), TokenType::Operator()) => return Err(format!("Error at token {:?}, two incompatible tokens type in a row", token)),
(TokenType::Operator(), TokenType::Equal()) => return Err(format!("Error at token {:?}, two incompatible tokens type in a row", token)),
(TokenType::Operator(), TokenType::Substraction()) => return Err(format!("Error at token {:?}, two incompatible tokens type in a row", token)),
(TokenType::Equal(), TokenType::Operator()) => return Err(format!("Error at token {:?}, two incompatible tokens type in a row", token)),
(TokenType::OpenParenthesis(), TokenType::OpenParenthesis()) => (), (TokenType::OpenParenthesis(), TokenType::OpenParenthesis()) => (),
(TokenType::CloseParenthesis(), TokenType::CloseParenthesis()) => (), (TokenType::CloseParenthesis(), TokenType::CloseParenthesis()) => (),
_ => { _ => {
@ -226,13 +247,24 @@ fn sanitize_tokens(tokens: Vec<Token>) -> Result<Vec<Token>, String> {
} }
}, },
} }
} else if token_type == TokenType::Operator() { } else if token_type == TokenType::Operator() || token_type == TokenType::Equal() {
return Err(format!("Error at token {:?}, query can't start with an operator", token)); return Err(format!("Error at token {:?}, query can't start with this token", token));
} }
sanitized_tokens.push(token); sanitized_tokens.push(token);
last_token_type = Some(token_type); last_token_type = Some(token_type);
} }
if open_close_equal_count.0 != open_close_equal_count.1 {
return Err(format!("Error, query has a different number of opening and closing parentheses"));
}
if open_close_equal_count.2 > 1 {
return Err(format!("Error, query shouldn't have more than 1 equals sign"));
}
let last_token = sanitized_tokens.last().unwrap();
match get_token_type(&last_token) {
TokenType::Equal() | TokenType::Operator() | TokenType::Substraction() => return Err(format!("Error at token {:?}, query can't end with this token", last_token)),
_ => (),
}
Ok(sanitized_tokens) Ok(sanitized_tokens)
} }
@ -373,8 +405,11 @@ pub fn solve(equation: Vec<f64>) {
mod tests { mod tests {
use super::*; use super::*;
mod tokenize {
use super::*;
#[test] #[test]
fn tokenize_addition() { fn addition() {
let query = "+"; let query = "+";
let result: Vec<Token> = vec![Token::Addition()]; let result: Vec<Token> = vec![Token::Addition()];
@ -382,7 +417,7 @@ mod tests {
} }
#[test] #[test]
fn tokenize_substraction() { fn substraction() {
let query = "-"; let query = "-";
let result: Vec<Token> = vec![Token::Substraction()]; let result: Vec<Token> = vec![Token::Substraction()];
@ -390,7 +425,7 @@ mod tests {
} }
#[test] #[test]
fn tokenize_multiplication() { fn multiplication() {
let query = "*"; let query = "*";
let result: Vec<Token> = vec![Token::Multiplication()]; let result: Vec<Token> = vec![Token::Multiplication()];
@ -398,7 +433,7 @@ mod tests {
} }
#[test] #[test]
fn tokenize_division() { fn division() {
let query = "/"; let query = "/";
let result: Vec<Token> = vec![Token::Division()]; let result: Vec<Token> = vec![Token::Division()];
@ -406,7 +441,7 @@ mod tests {
} }
#[test] #[test]
fn tokenize_modulo() { fn modulo() {
let query = "%"; let query = "%";
let result: Vec<Token> = vec![Token::Modulo()]; let result: Vec<Token> = vec![Token::Modulo()];
@ -414,7 +449,7 @@ mod tests {
} }
#[test] #[test]
fn tokenize_exponentiation() { fn exponentiation() {
let query = "^"; let query = "^";
let result: Vec<Token> = vec![Token::Exponentiation()]; let result: Vec<Token> = vec![Token::Exponentiation()];
@ -422,7 +457,7 @@ mod tests {
} }
#[test] #[test]
fn tokenize_equal() { fn equal() {
let query = "="; let query = "=";
let result: Vec<Token> = vec![Token::Equal()]; let result: Vec<Token> = vec![Token::Equal()];
@ -430,7 +465,7 @@ mod tests {
} }
#[test] #[test]
fn tokenize_open_parenthesis() { fn open_parenthesis() {
let query = "("; let query = "(";
let result: Vec<Token> = vec![Token::OpenParenthesis()]; let result: Vec<Token> = vec![Token::OpenParenthesis()];
@ -438,7 +473,7 @@ mod tests {
} }
#[test] #[test]
fn tokenize_close_parenthesis() { fn close_parenthesis() {
let query = ")"; let query = ")";
let result: Vec<Token> = vec![Token::CloseParenthesis()]; let result: Vec<Token> = vec![Token::CloseParenthesis()];
@ -446,7 +481,7 @@ mod tests {
} }
#[test] #[test]
fn tokenize_imaginary_unit() { fn imaginary_unit() {
let query = "i"; let query = "i";
let result: Vec<Token> = vec![Token::ImaginaryUnit()]; let result: Vec<Token> = vec![Token::ImaginaryUnit()];
@ -454,7 +489,7 @@ mod tests {
} }
#[test] #[test]
fn tokenize_variable() { fn variable() {
let query = "variable"; let query = "variable";
let result: Vec<Token> = vec![Token::Variable(String::from("variable"))]; let result: Vec<Token> = vec![Token::Variable(String::from("variable"))];
@ -462,7 +497,7 @@ mod tests {
} }
#[test] #[test]
fn tokenize_variable_double_i() { fn variable_double_i() {
let query = "ii"; let query = "ii";
let result: Vec<Token> = vec![Token::Variable(String::from("ii"))]; let result: Vec<Token> = vec![Token::Variable(String::from("ii"))];
@ -470,7 +505,7 @@ mod tests {
} }
#[test] #[test]
fn tokenize_number_natural() { fn number_natural() {
let query = "123456"; let query = "123456";
let result: Vec<Token> = vec![Token::Number(String::from("123456"))]; let result: Vec<Token> = vec![Token::Number(String::from("123456"))];
@ -478,7 +513,7 @@ mod tests {
} }
#[test] #[test]
fn tokenize_number_rational() { fn number_rational() {
let query = "123.456"; let query = "123.456";
let result: Vec<Token> = vec![Token::Number(String::from("123.456"))]; let result: Vec<Token> = vec![Token::Number(String::from("123.456"))];
@ -486,7 +521,7 @@ mod tests {
} }
#[test] #[test]
fn tokenize_number_point_something() { fn number_point_something() {
let query = ".123456"; let query = ".123456";
let result: Vec<Token> = vec![Token::Number(String::from(".123456"))]; let result: Vec<Token> = vec![Token::Number(String::from(".123456"))];
@ -494,7 +529,7 @@ mod tests {
} }
#[test] #[test]
fn tokenize_number_trailing_point() { fn number_trailing_point() {
let query = "123456."; let query = "123456.";
let result: Vec<Token> = vec![Token::Number(String::from("123456."))]; let result: Vec<Token> = vec![Token::Number(String::from("123456."))];
@ -502,7 +537,7 @@ mod tests {
} }
#[test] #[test]
fn tokenize_number_variable() { fn number_variable() {
let query = "23x"; let query = "23x";
let result: Vec<Token> = vec![Token::Number(String::from("23")), Token::Variable(String::from("x"))]; let result: Vec<Token> = vec![Token::Number(String::from("23")), Token::Variable(String::from("x"))];
assert_eq!(tokenize(query).unwrap(), result); assert_eq!(tokenize(query).unwrap(), result);
@ -522,31 +557,32 @@ mod tests {
#[test] #[test]
#[should_panic] #[should_panic]
fn tokenize_number_double_point() { fn number_double_point() {
let query = "12.34.56"; let query = "12.34.56";
tokenize(query).unwrap(); tokenize(query).unwrap();
} }
#[test] #[test]
#[should_panic] #[should_panic]
fn tokenize_number_double_point_in_a_row() { fn number_double_point_in_a_row() {
let query = "123..456"; let query = "123..456";
tokenize(query).unwrap(); tokenize(query).unwrap();
} }
#[test] #[test]
#[should_panic] #[should_panic]
fn tokenize_number_only_point() { fn number_only_point() {
let query = "."; let query = ".";
tokenize(query).unwrap(); tokenize(query).unwrap();
} }
#[test] #[test]
#[should_panic] #[should_panic]
fn tokenize_invalid_token() { fn invalid_token() {
let query = "324*43224+243_+234=234"; let query = "324*43224+243_+234=234";
tokenize(query).unwrap(); tokenize(query).unwrap();
} }
}
mod sanitize_tokens { mod sanitize_tokens {
use super::*; use super::*;
@ -650,9 +686,25 @@ mod tests {
let results = vec![number(), times(), x()]; let results = vec![number(), times(), x()];
assert_eq!(sanitize_tokens(tokens).unwrap(), results); assert_eq!(sanitize_tokens(tokens).unwrap(), results);
let tokens = vec![x(), number()];
let results = vec![x(), times(), number()];
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
let tokens = vec![number(), i()]; let tokens = vec![number(), i()];
let results = vec![number(), times(), i()]; let results = vec![number(), times(), i()];
assert_eq!(sanitize_tokens(tokens).unwrap(), results); assert_eq!(sanitize_tokens(tokens).unwrap(), results);
let tokens = vec![i(), number()];
let results = vec![i(), times(), number()];
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
let tokens = vec![i(), x()];
let results = vec![i(), times(), x()];
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
let tokens = vec![x(), i()];
let results = vec![x(), times(), i()];
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
} }
#[test] #[test]
@ -799,6 +851,90 @@ mod tests {
let tokens = vec![number(), plus(), number(), divided_by()]; let tokens = vec![number(), plus(), number(), divided_by()];
sanitize_tokens(tokens).unwrap(); sanitize_tokens(tokens).unwrap();
} }
#[test]
#[should_panic]
fn ends_with_open_brackets() {
let tokens = vec![number(), plus(), number(), open(), number(), close(), open()];
sanitize_tokens(tokens).unwrap();
}
#[test]
#[should_panic]
fn starts_with_close_brackets() {
let tokens = vec![close(), number(), plus(), number(), open(), number()];
sanitize_tokens(tokens).unwrap();
}
#[test]
#[should_panic]
fn wrong_brackets_order() {
let tokens = vec![number(), close(), number(), plus(), number(), open(), number()];
sanitize_tokens(tokens).unwrap();
}
#[test]
#[should_panic]
fn minus_operator() {
let tokens = vec![number(), minus(), plus(), i()];
sanitize_tokens(tokens).unwrap();
}
#[test]
#[should_panic]
fn operator_minus() {
let tokens = vec![number(), plus(), minus(), i()];
sanitize_tokens(tokens).unwrap();
}
#[test]
#[should_panic]
fn minus_equals() {
let tokens = vec![number(), minus(), equals(), i()];
sanitize_tokens(tokens).unwrap();
}
#[test]
#[should_panic]
fn operator_equals() {
let tokens = vec![number(), modulo(), equals(), i()];
sanitize_tokens(tokens).unwrap();
}
#[test]
#[should_panic]
fn equals_operator() {
let tokens = vec![number(), equals(), times(), i()];
sanitize_tokens(tokens).unwrap();
}
#[test]
fn equals_minus() {
let tokens = vec![number(), equals(), minus(), x()];
let results = tokens.clone();
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
}
mod get_token_type {
use super::*;
#[test]
fn exhaustive() {
assert_eq!(get_token_type(&Token::Number(String::from("123"))), TokenType::Number());
assert_eq!(get_token_type(&Token::Variable(String::from("var"))), TokenType::Variable());
assert_eq!(get_token_type(&Token::ImaginaryUnit()), TokenType::ImaginaryUnit());
assert_eq!(get_token_type(&Token::OpenParenthesis()), TokenType::OpenParenthesis());
assert_eq!(get_token_type(&Token::CloseParenthesis()), TokenType::CloseParenthesis());
assert_eq!(get_token_type(&Token::Substraction()), TokenType::Substraction());
assert_eq!(get_token_type(&Token::Equal()), TokenType::Equal());
assert_eq!(get_token_type(&Token::Addition()), TokenType::Operator());
assert_eq!(get_token_type(&Token::Multiplication()), TokenType::Operator());
assert_eq!(get_token_type(&Token::Division()), TokenType::Operator());
assert_eq!(get_token_type(&Token::Modulo()), TokenType::Operator());
assert_eq!(get_token_type(&Token::Exponentiation()), TokenType::Operator());
}
}
} }
/* /*