clean(parser::*): organize parser in modules
This commit is contained in:
parent
7c13270089
commit
17dbc79624
794
src/lib.rs
794
src/lib.rs
|
@ -1,32 +1,9 @@
|
||||||
use std::error::Error;
|
use std::error::Error;
|
||||||
|
use crate::parser::tokenizer::tokenize;
|
||||||
|
use crate::parser::sanitizer::sanitize_tokens;
|
||||||
|
use crate::parser::ast_builder::build_ast;
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Clone)]
|
pub mod parser;
|
||||||
enum Token {
|
|
||||||
Number(String),
|
|
||||||
Variable(String),
|
|
||||||
ImaginaryUnit(),
|
|
||||||
Addition(),
|
|
||||||
Multiplication(),
|
|
||||||
Substraction(),
|
|
||||||
Division(),
|
|
||||||
Modulo(),
|
|
||||||
Exponentiation(),
|
|
||||||
Equal(),
|
|
||||||
OpenParenthesis(),
|
|
||||||
CloseParenthesis(),
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq)]
|
|
||||||
enum TokenType {
|
|
||||||
Number(),
|
|
||||||
Variable(),
|
|
||||||
ImaginaryUnit(),
|
|
||||||
Operator(),
|
|
||||||
Substraction(),
|
|
||||||
OpenParenthesis(),
|
|
||||||
CloseParenthesis(),
|
|
||||||
Equal(),
|
|
||||||
}
|
|
||||||
|
|
||||||
struct _Rational {
|
struct _Rational {
|
||||||
numerator: i128,
|
numerator: i128,
|
||||||
|
@ -38,232 +15,13 @@ struct _GaussianRational {
|
||||||
imaginary: _Rational,
|
imaginary: _Rational,
|
||||||
}
|
}
|
||||||
|
|
||||||
fn check_number(my_string: String, i: usize) -> Result<Token, String> {
|
|
||||||
println!("Checking number at index {i}");
|
|
||||||
if my_string.as_str() == "." {
|
|
||||||
return Err(format!("unexpected token: `.` at position {i}"));
|
|
||||||
}
|
|
||||||
let find = my_string.find(".");
|
|
||||||
if let Some(j) = find {
|
|
||||||
let find = my_string[j+1..].find(".");
|
|
||||||
if let Some(_) = find {
|
|
||||||
return Err(format!("unexpected token: `{my_string}` at position {i}"))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Ok(Token::Number(my_string))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn tokenize(query: &str) -> Result<Vec<Token>, String> {
|
|
||||||
let mut tokens: Vec<Token> = vec![];
|
|
||||||
let mut my_string = String::new();
|
|
||||||
let mut is_last_number = false;
|
|
||||||
let mut is_last_variable = false;
|
|
||||||
|
|
||||||
for (i, token) in query.chars().enumerate() {
|
|
||||||
let mut is_still_number = false;
|
|
||||||
let mut is_still_variable = false;
|
|
||||||
let mut is_pop_needed = true;
|
|
||||||
match token {
|
|
||||||
'+' => tokens.push(Token::Addition()),
|
|
||||||
'-' => tokens.push(Token::Substraction()),
|
|
||||||
'*' => tokens.push(Token::Multiplication()),
|
|
||||||
'/' => tokens.push(Token::Division()),
|
|
||||||
'%' => tokens.push(Token::Modulo()),
|
|
||||||
'^' => tokens.push(Token::Exponentiation()),
|
|
||||||
'=' => tokens.push(Token::Equal()),
|
|
||||||
'(' => tokens.push(Token::OpenParenthesis()),
|
|
||||||
')' => tokens.push(Token::CloseParenthesis()),
|
|
||||||
'a'..='z' | 'A'..='Z' => {
|
|
||||||
if is_last_number == true {
|
|
||||||
let mut tmp_token = None;
|
|
||||||
if is_pop_needed {
|
|
||||||
tmp_token = tokens.pop();
|
|
||||||
}
|
|
||||||
tokens.push(check_number(my_string, i)?);
|
|
||||||
if let Some(tok) = tmp_token {
|
|
||||||
tokens.push(tok);
|
|
||||||
}
|
|
||||||
is_last_number = false;
|
|
||||||
my_string = String::new();
|
|
||||||
}
|
|
||||||
if is_last_variable == false {
|
|
||||||
is_last_variable = true;
|
|
||||||
}
|
|
||||||
is_still_variable = true;
|
|
||||||
my_string += &token.to_string();
|
|
||||||
},
|
|
||||||
'0'..='9' | '.' => {
|
|
||||||
if is_last_variable == true {
|
|
||||||
let mut tmp_token = None;
|
|
||||||
if is_pop_needed {
|
|
||||||
tmp_token = tokens.pop();
|
|
||||||
}
|
|
||||||
match my_string.as_str() {
|
|
||||||
"i" => tokens.push(Token::ImaginaryUnit()),
|
|
||||||
_ => tokens.push(Token::Variable(my_string)),
|
|
||||||
}
|
|
||||||
if let Some(tok) = tmp_token {
|
|
||||||
tokens.push(tok);
|
|
||||||
}
|
|
||||||
is_last_variable = false;
|
|
||||||
my_string = String::new();
|
|
||||||
}
|
|
||||||
if is_last_number == false {
|
|
||||||
is_last_number = true;
|
|
||||||
}
|
|
||||||
is_still_number = true;
|
|
||||||
my_string += &token.to_string();
|
|
||||||
},
|
|
||||||
' ' => is_pop_needed = false,
|
|
||||||
_ => return Err(format!("unexpected token: `{token}` at position {i}")),
|
|
||||||
};
|
|
||||||
if is_last_variable && !is_still_variable {
|
|
||||||
let mut tmp_token = None;
|
|
||||||
if is_pop_needed {
|
|
||||||
tmp_token = tokens.pop();
|
|
||||||
}
|
|
||||||
match my_string.as_str() {
|
|
||||||
"i" => tokens.push(Token::ImaginaryUnit()),
|
|
||||||
_ => tokens.push(Token::Variable(my_string)),
|
|
||||||
}
|
|
||||||
if let Some(tok) = tmp_token {
|
|
||||||
tokens.push(tok);
|
|
||||||
}
|
|
||||||
is_last_variable = false;
|
|
||||||
my_string = String::new();
|
|
||||||
}
|
|
||||||
else if is_last_number && !is_still_number {
|
|
||||||
let mut tmp_token = None;
|
|
||||||
if is_pop_needed {
|
|
||||||
tmp_token = tokens.pop();
|
|
||||||
}
|
|
||||||
tokens.push(check_number(my_string, i)?);
|
|
||||||
if let Some(tok) = tmp_token {
|
|
||||||
tokens.push(tok);
|
|
||||||
}
|
|
||||||
is_last_number = false;
|
|
||||||
my_string = String::new();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if is_last_variable {
|
|
||||||
match my_string.as_str() {
|
|
||||||
"i" => tokens.push(Token::ImaginaryUnit()),
|
|
||||||
_ => tokens.push(Token::Variable(my_string)),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if is_last_number {
|
|
||||||
tokens.push(check_number(my_string, query.len())?);
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(tokens)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn get_token_type(token: &Token) -> TokenType {
|
|
||||||
match token {
|
|
||||||
Token::Number(_) => TokenType::Number(),
|
|
||||||
Token::Variable(_) => TokenType::Variable(),
|
|
||||||
Token::ImaginaryUnit() => TokenType::ImaginaryUnit(),
|
|
||||||
Token::OpenParenthesis() => TokenType::OpenParenthesis(),
|
|
||||||
Token::CloseParenthesis() => TokenType::CloseParenthesis(),
|
|
||||||
Token::Substraction() => TokenType::Substraction(),
|
|
||||||
Token::Equal() => TokenType::Equal(),
|
|
||||||
_ => TokenType::Operator(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn sanitize_tokens(tokens: Vec<Token>) -> Result<Vec<Token>, String> {
|
|
||||||
let mut sanitized_tokens: Vec<Token> = vec![];
|
|
||||||
let mut last_token_type = None;
|
|
||||||
let mut open_close_equal_count = (0, 0, 0);
|
|
||||||
|
|
||||||
for token in tokens {
|
|
||||||
let token_type = get_token_type(&token);
|
|
||||||
match token_type {
|
|
||||||
TokenType::OpenParenthesis() => open_close_equal_count.0 += 1,
|
|
||||||
TokenType::CloseParenthesis() => {
|
|
||||||
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 {
|
|
||||||
match (&last_token_type, &token_type) {
|
|
||||||
(TokenType::Number(), TokenType::Variable()) => sanitized_tokens.push(Token::Multiplication()),
|
|
||||||
(TokenType::Number(), TokenType::ImaginaryUnit()) => sanitized_tokens.push(Token::Multiplication()),
|
|
||||||
(TokenType::Number(), TokenType::OpenParenthesis()) => sanitized_tokens.push(Token::Multiplication()),
|
|
||||||
(TokenType::Variable(), TokenType::Number()) => sanitized_tokens.push(Token::Multiplication()),
|
|
||||||
(TokenType::Variable(), TokenType::ImaginaryUnit()) => sanitized_tokens.push(Token::Multiplication()),
|
|
||||||
(TokenType::Variable(), TokenType::OpenParenthesis()) => sanitized_tokens.push(Token::Multiplication()),
|
|
||||||
(TokenType::ImaginaryUnit(), TokenType::Number()) => sanitized_tokens.push(Token::Multiplication()),
|
|
||||||
(TokenType::ImaginaryUnit(), TokenType::Variable()) => sanitized_tokens.push(Token::Multiplication()),
|
|
||||||
(TokenType::ImaginaryUnit(), TokenType::OpenParenthesis()) => sanitized_tokens.push(Token::Multiplication()),
|
|
||||||
(TokenType::CloseParenthesis(), TokenType::OpenParenthesis()) => sanitized_tokens.push(Token::Multiplication()),
|
|
||||||
(TokenType::CloseParenthesis(), TokenType::Number()) => sanitized_tokens.push(Token::Multiplication()),
|
|
||||||
(TokenType::CloseParenthesis(), TokenType::Variable()) => sanitized_tokens.push(Token::Multiplication()),
|
|
||||||
(TokenType::CloseParenthesis(), TokenType::ImaginaryUnit()) => sanitized_tokens.push(Token::Multiplication()),
|
|
||||||
(TokenType::OpenParenthesis(), TokenType::Operator()) => return Err(format!("Error at token {:?}, operator forbidden directly after opening 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::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::CloseParenthesis(), TokenType::CloseParenthesis()) => (),
|
|
||||||
_ => {
|
|
||||||
if token_type == last_token_type {
|
|
||||||
return Err(format!("Error at token {:?}, two incompatible tokens type in a row", token));
|
|
||||||
}
|
|
||||||
},
|
|
||||||
}
|
|
||||||
} else if token_type == TokenType::Operator() || token_type == TokenType::Equal() {
|
|
||||||
return Err(format!("Error at token {:?}, query can't start with this token", token));
|
|
||||||
}
|
|
||||||
sanitized_tokens.push(token);
|
|
||||||
|
|
||||||
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)
|
|
||||||
}
|
|
||||||
|
|
||||||
struct Node {
|
|
||||||
_lhs: Option<Box<Node>>,
|
|
||||||
_rhs: Option<Box<Node>>,
|
|
||||||
}
|
|
||||||
|
|
||||||
fn build_ast(_tokens: Vec<Token>) -> Node {
|
|
||||||
Node { _lhs: None, _rhs: None }
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn parse(query: &str) -> Result<Vec<f64>, Box<dyn Error>> {
|
pub fn parse(query: &str) -> Result<Vec<f64>, Box<dyn Error>> {
|
||||||
let tokens = tokenize(query)?;
|
let tokens = tokenize(query)?;
|
||||||
|
println!("{:?}", tokens);
|
||||||
let sanitized_tokens = sanitize_tokens(tokens)?;
|
let sanitized_tokens = sanitize_tokens(tokens)?;
|
||||||
|
println!("{:?}", sanitized_tokens);
|
||||||
let _ast = build_ast(sanitized_tokens);
|
let _ast = build_ast(sanitized_tokens);
|
||||||
|
|
||||||
println!("{:?}", tokenize(query));
|
|
||||||
|
|
||||||
Ok(vec![])
|
Ok(vec![])
|
||||||
}
|
}
|
||||||
|
@ -356,539 +114,7 @@ pub fn solve(equation: Vec<f64>) {
|
||||||
mod tests {
|
mod tests {
|
||||||
use super::*;
|
use super::*;
|
||||||
|
|
||||||
mod tokenize {
|
#[test]
|
||||||
use super::*;
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn addition() {
|
|
||||||
let query = "+";
|
|
||||||
let result: Vec<Token> = vec![Token::Addition()];
|
|
||||||
|
|
||||||
assert_eq!(tokenize(query).unwrap(), result);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn substraction() {
|
|
||||||
let query = "-";
|
|
||||||
let result: Vec<Token> = vec![Token::Substraction()];
|
|
||||||
|
|
||||||
assert_eq!(tokenize(query).unwrap(), result);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn multiplication() {
|
|
||||||
let query = "*";
|
|
||||||
let result: Vec<Token> = vec![Token::Multiplication()];
|
|
||||||
|
|
||||||
assert_eq!(tokenize(query).unwrap(), result);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn division() {
|
|
||||||
let query = "/";
|
|
||||||
let result: Vec<Token> = vec![Token::Division()];
|
|
||||||
|
|
||||||
assert_eq!(tokenize(query).unwrap(), result);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn modulo() {
|
|
||||||
let query = "%";
|
|
||||||
let result: Vec<Token> = vec![Token::Modulo()];
|
|
||||||
|
|
||||||
assert_eq!(tokenize(query).unwrap(), result);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn exponentiation() {
|
|
||||||
let query = "^";
|
|
||||||
let result: Vec<Token> = vec![Token::Exponentiation()];
|
|
||||||
|
|
||||||
assert_eq!(tokenize(query).unwrap(), result);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn equal() {
|
|
||||||
let query = "=";
|
|
||||||
let result: Vec<Token> = vec![Token::Equal()];
|
|
||||||
|
|
||||||
assert_eq!(tokenize(query).unwrap(), result);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn open_parenthesis() {
|
|
||||||
let query = "(";
|
|
||||||
let result: Vec<Token> = vec![Token::OpenParenthesis()];
|
|
||||||
|
|
||||||
assert_eq!(tokenize(query).unwrap(), result);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn close_parenthesis() {
|
|
||||||
let query = ")";
|
|
||||||
let result: Vec<Token> = vec![Token::CloseParenthesis()];
|
|
||||||
|
|
||||||
assert_eq!(tokenize(query).unwrap(), result);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn imaginary_unit() {
|
|
||||||
let query = "i";
|
|
||||||
let result: Vec<Token> = vec![Token::ImaginaryUnit()];
|
|
||||||
|
|
||||||
assert_eq!(tokenize(query).unwrap(), result);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn variable() {
|
|
||||||
let query = "variable";
|
|
||||||
let result: Vec<Token> = vec![Token::Variable(String::from("variable"))];
|
|
||||||
|
|
||||||
assert_eq!(tokenize(query).unwrap(), result);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn variable_double_i() {
|
|
||||||
let query = "ii";
|
|
||||||
let result: Vec<Token> = vec![Token::Variable(String::from("ii"))];
|
|
||||||
|
|
||||||
assert_eq!(tokenize(query).unwrap(), result);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn number_natural() {
|
|
||||||
let query = "123456";
|
|
||||||
let result: Vec<Token> = vec![Token::Number(String::from("123456"))];
|
|
||||||
|
|
||||||
assert_eq!(tokenize(query).unwrap(), result);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn number_rational() {
|
|
||||||
let query = "123.456";
|
|
||||||
let result: Vec<Token> = vec![Token::Number(String::from("123.456"))];
|
|
||||||
|
|
||||||
assert_eq!(tokenize(query).unwrap(), result);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn number_point_something() {
|
|
||||||
let query = ".123456";
|
|
||||||
let result: Vec<Token> = vec![Token::Number(String::from(".123456"))];
|
|
||||||
|
|
||||||
assert_eq!(tokenize(query).unwrap(), result);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn number_trailing_point() {
|
|
||||||
let query = "123456.";
|
|
||||||
let result: Vec<Token> = vec![Token::Number(String::from("123456."))];
|
|
||||||
|
|
||||||
assert_eq!(tokenize(query).unwrap(), result);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn number_variable() {
|
|
||||||
let query = "23x";
|
|
||||||
let result: Vec<Token> = vec![Token::Number(String::from("23")), Token::Variable(String::from("x"))];
|
|
||||||
assert_eq!(tokenize(query).unwrap(), result);
|
|
||||||
|
|
||||||
let query = "23i";
|
|
||||||
let result: Vec<Token> = vec![Token::Number(String::from("23")), Token::ImaginaryUnit()];
|
|
||||||
assert_eq!(tokenize(query).unwrap(), result);
|
|
||||||
|
|
||||||
let query = "x23";
|
|
||||||
let result: Vec<Token> = vec![Token::Variable(String::from("x")), Token::Number(String::from("23"))];
|
|
||||||
assert_eq!(tokenize(query).unwrap(), result);
|
|
||||||
|
|
||||||
let query = "i23";
|
|
||||||
let result: Vec<Token> = vec![Token::ImaginaryUnit(), Token::Number(String::from("23"))];
|
|
||||||
assert_eq!(tokenize(query).unwrap(), result);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[should_panic]
|
|
||||||
fn number_double_point() {
|
|
||||||
let query = "12.34.56";
|
|
||||||
tokenize(query).unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[should_panic]
|
|
||||||
fn number_double_point_in_a_row() {
|
|
||||||
let query = "123..456";
|
|
||||||
tokenize(query).unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[should_panic]
|
|
||||||
fn number_only_point() {
|
|
||||||
let query = ".";
|
|
||||||
tokenize(query).unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[should_panic]
|
|
||||||
fn invalid_token() {
|
|
||||||
let query = "324*43224+243_+234=234";
|
|
||||||
tokenize(query).unwrap();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
mod sanitize_tokens {
|
|
||||||
use super::*;
|
|
||||||
|
|
||||||
fn plus() -> Token {
|
|
||||||
Token::Addition()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn minus() -> Token {
|
|
||||||
Token::Substraction()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn times() -> Token {
|
|
||||||
Token::Multiplication()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn divided_by() -> Token {
|
|
||||||
Token::Division()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn modulo() -> Token {
|
|
||||||
Token::Modulo()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn equals() -> Token {
|
|
||||||
Token::Equal()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn x() -> Token {
|
|
||||||
Token::Variable(String::from("x"))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn i() -> Token {
|
|
||||||
Token::ImaginaryUnit()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn number() -> Token {
|
|
||||||
Token::Number(String::from("123"))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn open() -> Token {
|
|
||||||
Token::OpenParenthesis()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn close() -> Token {
|
|
||||||
Token::CloseParenthesis()
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[should_panic]
|
|
||||||
fn double_number() {
|
|
||||||
let tokens = vec![number(), number()];
|
|
||||||
sanitize_tokens(tokens).unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[should_panic]
|
|
||||||
fn double_operator() {
|
|
||||||
let tokens = vec![number(), plus(), divided_by()];
|
|
||||||
sanitize_tokens(tokens).unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[should_panic]
|
|
||||||
fn double_variable() {
|
|
||||||
let tokens = vec![x(), x()];
|
|
||||||
sanitize_tokens(tokens).unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[should_panic]
|
|
||||||
fn double_imaginary_unit() {
|
|
||||||
let tokens = vec![i(), i()];
|
|
||||||
sanitize_tokens(tokens).unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[should_panic]
|
|
||||||
fn sanitize_tokens_starts_with_operator() {
|
|
||||||
let tokens = vec![plus(), number(), divided_by(), x()];
|
|
||||||
sanitize_tokens(tokens).unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn starts_with_negative_number() {
|
|
||||||
let tokens = vec![minus(), number(), divided_by(), x()];
|
|
||||||
let results = tokens.clone();
|
|
||||||
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn number_operator_variable() {
|
|
||||||
let tokens = vec![number(), divided_by(), x()];
|
|
||||||
let results = tokens.clone();
|
|
||||||
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn insert_multiplication() {
|
|
||||||
let tokens = vec![number(), x()];
|
|
||||||
let results = vec![number(), times(), x()];
|
|
||||||
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 results = vec![number(), times(), i()];
|
|
||||||
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]
|
|
||||||
fn insert_multiplication_parenthesis() {
|
|
||||||
let tokens = vec![open(), number(), modulo(), number(), close(), open(), i(), times(), x(), close()];
|
|
||||||
let results = vec![open(), number(), modulo(), number(), close(), times(), open(), i(), times(), x(), close()];
|
|
||||||
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
|
||||||
|
|
||||||
let tokens = vec![number(), open(), number(), modulo(), number(), close()];
|
|
||||||
let results= vec![number(), times(), open(), number(), modulo(), number(), close()];
|
|
||||||
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
|
||||||
|
|
||||||
let tokens = vec![i(), open(), number(), modulo(), number(), close()];
|
|
||||||
let results= vec![i(), times(), open(), number(), modulo(), number(), close()];
|
|
||||||
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
|
||||||
|
|
||||||
let tokens = vec![x(), open(), number(), modulo(), number(), close()];
|
|
||||||
let results= vec![x(), times(), open(), number(), modulo(), number(), close()];
|
|
||||||
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
|
||||||
|
|
||||||
let tokens = vec![open(), number(), modulo(), number(), close(), number()];
|
|
||||||
let results= vec![open(), number(), modulo(), number(), close(), times(), number()];
|
|
||||||
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
|
||||||
|
|
||||||
let tokens = vec![open(), number(), modulo(), number(), close(), i()];
|
|
||||||
let results= vec![open(), number(), modulo(), number(), close(), times(), i()];
|
|
||||||
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
|
||||||
|
|
||||||
let tokens = vec![open(), number(), modulo(), number(), close(), x()];
|
|
||||||
let results= vec![open(), number(), modulo(), number(), close(), times(), x()];
|
|
||||||
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
|
||||||
|
|
||||||
let tokens = vec![number(), open(), number(), modulo(), number(), close(), x()];
|
|
||||||
let results= vec![number(), times(), open(), number(), modulo(), number(), close(), times(), x()];
|
|
||||||
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[should_panic]
|
|
||||||
fn parenthesis_operator() {
|
|
||||||
let tokens = vec!(open(), plus(), number(), close());
|
|
||||||
sanitize_tokens(tokens).unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[should_panic]
|
|
||||||
fn operator_parenthesis() {
|
|
||||||
let tokens = vec!(open(), number(), modulo(), close());
|
|
||||||
sanitize_tokens(tokens).unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[should_panic]
|
|
||||||
fn minus_parenthesis() {
|
|
||||||
let tokens = vec!(open(), number(), minus(), close());
|
|
||||||
sanitize_tokens(tokens).unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[should_panic]
|
|
||||||
fn empty_parentheses() {
|
|
||||||
let tokens = vec!(open(), close());
|
|
||||||
sanitize_tokens(tokens).unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn parenthesis_minus() {
|
|
||||||
let tokens = vec!(open(), minus(), number(), close());
|
|
||||||
let results = tokens.clone();
|
|
||||||
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn double_parentheses() {
|
|
||||||
let tokens = vec!(open(), open(), number(), plus(), number(), close(), plus(), number(), close());
|
|
||||||
let results = tokens.clone();
|
|
||||||
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
|
||||||
|
|
||||||
let tokens = vec!(open(), number(), plus(), open(), number(), plus(), number(), close(), close());
|
|
||||||
let results = tokens.clone();
|
|
||||||
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
|
||||||
|
|
||||||
let tokens = vec!(open(), open(), number(), plus(), number(), close(), number(), close());
|
|
||||||
let results = vec!(open(), open(), number(), plus(), number(), close(), times(), number(), close());
|
|
||||||
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
|
||||||
|
|
||||||
let tokens = vec!(open(), number(), open(), number(), plus(), number(), close(), close());
|
|
||||||
let results= vec!(open(), number(), times(), open(), number(), plus(), number(), close(), close());
|
|
||||||
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[should_panic]
|
|
||||||
fn wrong_parentheses_count() {
|
|
||||||
let tokens = vec![open(), number(), close(), close()];
|
|
||||||
sanitize_tokens(tokens).unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[should_panic]
|
|
||||||
fn wrong_parentheses_count_2() {
|
|
||||||
let tokens = vec![open(), open(), number(), close()];
|
|
||||||
sanitize_tokens(tokens).unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[should_panic]
|
|
||||||
fn wrong_parentheses_count_hard() {
|
|
||||||
let tokens = vec![open(), open(), number(), close(), equals(), x(), close()];
|
|
||||||
sanitize_tokens(tokens).unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[should_panic]
|
|
||||||
fn wrong_equal_count() {
|
|
||||||
let tokens = vec![number(), equals(), number(), equals(), number()];
|
|
||||||
sanitize_tokens(tokens).unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[should_panic]
|
|
||||||
fn starts_with_equal() {
|
|
||||||
let tokens = vec![equals(), number(), plus(), number()];
|
|
||||||
sanitize_tokens(tokens).unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[should_panic]
|
|
||||||
fn ends_with_equal() {
|
|
||||||
let tokens = vec![number(), plus(), number(), equals()];
|
|
||||||
sanitize_tokens(tokens).unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[should_panic]
|
|
||||||
fn ends_with_minus() {
|
|
||||||
let tokens = vec![number(), plus(), number(), minus()];
|
|
||||||
sanitize_tokens(tokens).unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
#[should_panic]
|
|
||||||
fn ends_with_operator() {
|
|
||||||
let tokens = vec![number(), plus(), number(), divided_by()];
|
|
||||||
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());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn parse_degree_0() {
|
fn parse_degree_0() {
|
||||||
let query = "5 * X^0 = 3 * X^0";
|
let query = "5 * X^0 = 3 * X^0";
|
||||||
let result: Vec<f64> = vec![2.];
|
let result: Vec<f64> = vec![2.];
|
||||||
|
@ -896,7 +122,7 @@ mod tests {
|
||||||
assert_eq!(parse(query).unwrap(), result);
|
assert_eq!(parse(query).unwrap(), result);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn parse_degree_1() {
|
fn parse_degree_1() {
|
||||||
let query = "5 * X^0 + 3 * X^1 = 3 * X^0";
|
let query = "5 * X^0 + 3 * X^1 = 3 * X^0";
|
||||||
let result: Vec<f64> = vec![2., 3.];
|
let result: Vec<f64> = vec![2., 3.];
|
||||||
|
@ -904,7 +130,7 @@ mod tests {
|
||||||
assert_eq!(parse(query).unwrap(), result);
|
assert_eq!(parse(query).unwrap(), result);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn parse_degree_2() {
|
fn parse_degree_2() {
|
||||||
let query = "5 * X^0 + 6 * X^1 + 8 * X^2 = 3 * X^0 - 2 * X^2";
|
let query = "5 * X^0 + 6 * X^1 + 8 * X^2 = 3 * X^0 - 2 * X^2";
|
||||||
let result = vec![2., 6., 10.];
|
let result = vec![2., 6., 10.];
|
||||||
|
@ -912,7 +138,7 @@ mod tests {
|
||||||
assert_eq!(parse(query).unwrap(), result);
|
assert_eq!(parse(query).unwrap(), result);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn parse_random_order() {
|
fn parse_random_order() {
|
||||||
let query = "9.3 * X^3 + 4.3 * X^0 + 3.4 * X^2 - 1.5 * X^3 - 13.12 * X^1 = 1.4 * X^2 - 5.1 * X^3 + 1.4 * X^1 -6.3 * X^0";
|
let query = "9.3 * X^3 + 4.3 * X^0 + 3.4 * X^2 - 1.5 * X^3 - 13.12 * X^1 = 1.4 * X^2 - 5.1 * X^3 + 1.4 * X^1 -6.3 * X^0";
|
||||||
let result = vec![10.6, -14.52, 2., 12.9];
|
let result = vec![10.6, -14.52, 2., 12.9];
|
||||||
|
|
|
@ -0,0 +1,19 @@
|
||||||
|
pub mod tokenizer;
|
||||||
|
pub mod sanitizer;
|
||||||
|
pub mod ast_builder;
|
||||||
|
|
||||||
|
#[derive(Debug, PartialEq, Clone)]
|
||||||
|
pub enum Token {
|
||||||
|
Number(String),
|
||||||
|
Variable(String),
|
||||||
|
ImaginaryUnit(),
|
||||||
|
Addition(),
|
||||||
|
Multiplication(),
|
||||||
|
Substraction(),
|
||||||
|
Division(),
|
||||||
|
Modulo(),
|
||||||
|
Exponentiation(),
|
||||||
|
Equal(),
|
||||||
|
OpenParenthesis(),
|
||||||
|
CloseParenthesis(),
|
||||||
|
}
|
|
@ -0,0 +1,10 @@
|
||||||
|
use super::Token;
|
||||||
|
|
||||||
|
pub struct Node {
|
||||||
|
_lhs: Option<Box<Node>>,
|
||||||
|
_rhs: Option<Box<Node>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn build_ast(_tokens: Vec<Token>) -> Node {
|
||||||
|
Node { _lhs: None, _rhs: None }
|
||||||
|
}
|
|
@ -0,0 +1,457 @@
|
||||||
|
use super::Token;
|
||||||
|
|
||||||
|
#[derive(Debug, PartialEq)]
|
||||||
|
enum TokenType {
|
||||||
|
Number(),
|
||||||
|
Variable(),
|
||||||
|
ImaginaryUnit(),
|
||||||
|
Operator(),
|
||||||
|
Substraction(),
|
||||||
|
OpenParenthesis(),
|
||||||
|
CloseParenthesis(),
|
||||||
|
Equal(),
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_token_type(token: &Token) -> TokenType {
|
||||||
|
match token {
|
||||||
|
Token::Number(_) => TokenType::Number(),
|
||||||
|
Token::Variable(_) => TokenType::Variable(),
|
||||||
|
Token::ImaginaryUnit() => TokenType::ImaginaryUnit(),
|
||||||
|
Token::OpenParenthesis() => TokenType::OpenParenthesis(),
|
||||||
|
Token::CloseParenthesis() => TokenType::CloseParenthesis(),
|
||||||
|
Token::Substraction() => TokenType::Substraction(),
|
||||||
|
Token::Equal() => TokenType::Equal(),
|
||||||
|
_ => TokenType::Operator(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn sanitize_tokens(tokens: Vec<Token>) -> Result<Vec<Token>, String> {
|
||||||
|
let mut sanitized_tokens: Vec<Token> = vec![];
|
||||||
|
let mut last_token_type = None;
|
||||||
|
let mut open_close_equal_count = (0, 0, 0);
|
||||||
|
|
||||||
|
for token in tokens {
|
||||||
|
let token_type = get_token_type(&token);
|
||||||
|
match token_type {
|
||||||
|
TokenType::OpenParenthesis() => open_close_equal_count.0 += 1,
|
||||||
|
TokenType::CloseParenthesis() => {
|
||||||
|
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 {
|
||||||
|
match (&last_token_type, &token_type) {
|
||||||
|
(TokenType::Number(), TokenType::Variable()) => sanitized_tokens.push(Token::Multiplication()),
|
||||||
|
(TokenType::Number(), TokenType::ImaginaryUnit()) => sanitized_tokens.push(Token::Multiplication()),
|
||||||
|
(TokenType::Number(), TokenType::OpenParenthesis()) => sanitized_tokens.push(Token::Multiplication()),
|
||||||
|
(TokenType::Variable(), TokenType::Number()) => sanitized_tokens.push(Token::Multiplication()),
|
||||||
|
(TokenType::Variable(), TokenType::ImaginaryUnit()) => sanitized_tokens.push(Token::Multiplication()),
|
||||||
|
(TokenType::Variable(), TokenType::OpenParenthesis()) => sanitized_tokens.push(Token::Multiplication()),
|
||||||
|
(TokenType::ImaginaryUnit(), TokenType::Number()) => sanitized_tokens.push(Token::Multiplication()),
|
||||||
|
(TokenType::ImaginaryUnit(), TokenType::Variable()) => sanitized_tokens.push(Token::Multiplication()),
|
||||||
|
(TokenType::ImaginaryUnit(), TokenType::OpenParenthesis()) => sanitized_tokens.push(Token::Multiplication()),
|
||||||
|
(TokenType::CloseParenthesis(), TokenType::OpenParenthesis()) => sanitized_tokens.push(Token::Multiplication()),
|
||||||
|
(TokenType::CloseParenthesis(), TokenType::Number()) => sanitized_tokens.push(Token::Multiplication()),
|
||||||
|
(TokenType::CloseParenthesis(), TokenType::Variable()) => sanitized_tokens.push(Token::Multiplication()),
|
||||||
|
(TokenType::CloseParenthesis(), TokenType::ImaginaryUnit()) => sanitized_tokens.push(Token::Multiplication()),
|
||||||
|
(TokenType::OpenParenthesis(), TokenType::Operator()) => return Err(format!("Error at token {:?}, operator forbidden directly after opening 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::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::CloseParenthesis(), TokenType::CloseParenthesis()) => (),
|
||||||
|
_ => {
|
||||||
|
if token_type == last_token_type {
|
||||||
|
return Err(format!("Error at token {:?}, two incompatible tokens type in a row", token));
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
} else if token_type == TokenType::Operator() || token_type == TokenType::Equal() {
|
||||||
|
return Err(format!("Error at token {:?}, query can't start with this token", token));
|
||||||
|
}
|
||||||
|
sanitized_tokens.push(token);
|
||||||
|
|
||||||
|
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)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use super::*;
|
||||||
|
|
||||||
|
fn plus() -> Token {
|
||||||
|
Token::Addition()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn minus() -> Token {
|
||||||
|
Token::Substraction()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn times() -> Token {
|
||||||
|
Token::Multiplication()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn divided_by() -> Token {
|
||||||
|
Token::Division()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn modulo() -> Token {
|
||||||
|
Token::Modulo()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn equals() -> Token {
|
||||||
|
Token::Equal()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn x() -> Token {
|
||||||
|
Token::Variable(String::from("x"))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn i() -> Token {
|
||||||
|
Token::ImaginaryUnit()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn number() -> Token {
|
||||||
|
Token::Number(String::from("123"))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn open() -> Token {
|
||||||
|
Token::OpenParenthesis()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn close() -> Token {
|
||||||
|
Token::CloseParenthesis()
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[should_panic]
|
||||||
|
fn double_number() {
|
||||||
|
let tokens = vec![number(), number()];
|
||||||
|
sanitize_tokens(tokens).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[should_panic]
|
||||||
|
fn double_operator() {
|
||||||
|
let tokens = vec![number(), plus(), divided_by()];
|
||||||
|
sanitize_tokens(tokens).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[should_panic]
|
||||||
|
fn double_variable() {
|
||||||
|
let tokens = vec![x(), x()];
|
||||||
|
sanitize_tokens(tokens).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[should_panic]
|
||||||
|
fn double_imaginary_unit() {
|
||||||
|
let tokens = vec![i(), i()];
|
||||||
|
sanitize_tokens(tokens).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[should_panic]
|
||||||
|
fn sanitize_tokens_starts_with_operator() {
|
||||||
|
let tokens = vec![plus(), number(), divided_by(), x()];
|
||||||
|
sanitize_tokens(tokens).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn starts_with_negative_number() {
|
||||||
|
let tokens = vec![minus(), number(), divided_by(), x()];
|
||||||
|
let results = tokens.clone();
|
||||||
|
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn number_operator_variable() {
|
||||||
|
let tokens = vec![number(), divided_by(), x()];
|
||||||
|
let results = tokens.clone();
|
||||||
|
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn insert_multiplication() {
|
||||||
|
let tokens = vec![number(), x()];
|
||||||
|
let results = vec![number(), times(), x()];
|
||||||
|
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 results = vec![number(), times(), i()];
|
||||||
|
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]
|
||||||
|
fn insert_multiplication_parenthesis() {
|
||||||
|
let tokens = vec![open(), number(), modulo(), number(), close(), open(), i(), times(), x(), close()];
|
||||||
|
let results = vec![open(), number(), modulo(), number(), close(), times(), open(), i(), times(), x(), close()];
|
||||||
|
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
||||||
|
|
||||||
|
let tokens = vec![number(), open(), number(), modulo(), number(), close()];
|
||||||
|
let results= vec![number(), times(), open(), number(), modulo(), number(), close()];
|
||||||
|
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
||||||
|
|
||||||
|
let tokens = vec![i(), open(), number(), modulo(), number(), close()];
|
||||||
|
let results= vec![i(), times(), open(), number(), modulo(), number(), close()];
|
||||||
|
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
||||||
|
|
||||||
|
let tokens = vec![x(), open(), number(), modulo(), number(), close()];
|
||||||
|
let results= vec![x(), times(), open(), number(), modulo(), number(), close()];
|
||||||
|
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
||||||
|
|
||||||
|
let tokens = vec![open(), number(), modulo(), number(), close(), number()];
|
||||||
|
let results= vec![open(), number(), modulo(), number(), close(), times(), number()];
|
||||||
|
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
||||||
|
|
||||||
|
let tokens = vec![open(), number(), modulo(), number(), close(), i()];
|
||||||
|
let results= vec![open(), number(), modulo(), number(), close(), times(), i()];
|
||||||
|
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
||||||
|
|
||||||
|
let tokens = vec![open(), number(), modulo(), number(), close(), x()];
|
||||||
|
let results= vec![open(), number(), modulo(), number(), close(), times(), x()];
|
||||||
|
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
||||||
|
|
||||||
|
let tokens = vec![number(), open(), number(), modulo(), number(), close(), x()];
|
||||||
|
let results= vec![number(), times(), open(), number(), modulo(), number(), close(), times(), x()];
|
||||||
|
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[should_panic]
|
||||||
|
fn parenthesis_operator() {
|
||||||
|
let tokens = vec!(open(), plus(), number(), close());
|
||||||
|
sanitize_tokens(tokens).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[should_panic]
|
||||||
|
fn operator_parenthesis() {
|
||||||
|
let tokens = vec!(open(), number(), modulo(), close());
|
||||||
|
sanitize_tokens(tokens).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[should_panic]
|
||||||
|
fn minus_parenthesis() {
|
||||||
|
let tokens = vec!(open(), number(), minus(), close());
|
||||||
|
sanitize_tokens(tokens).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[should_panic]
|
||||||
|
fn empty_parentheses() {
|
||||||
|
let tokens = vec!(open(), close());
|
||||||
|
sanitize_tokens(tokens).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn parenthesis_minus() {
|
||||||
|
let tokens = vec!(open(), minus(), number(), close());
|
||||||
|
let results = tokens.clone();
|
||||||
|
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn double_parentheses() {
|
||||||
|
let tokens = vec!(open(), open(), number(), plus(), number(), close(), plus(), number(), close());
|
||||||
|
let results = tokens.clone();
|
||||||
|
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
||||||
|
|
||||||
|
let tokens = vec!(open(), number(), plus(), open(), number(), plus(), number(), close(), close());
|
||||||
|
let results = tokens.clone();
|
||||||
|
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
||||||
|
|
||||||
|
let tokens = vec!(open(), open(), number(), plus(), number(), close(), number(), close());
|
||||||
|
let results = vec!(open(), open(), number(), plus(), number(), close(), times(), number(), close());
|
||||||
|
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
||||||
|
|
||||||
|
let tokens = vec!(open(), number(), open(), number(), plus(), number(), close(), close());
|
||||||
|
let results= vec!(open(), number(), times(), open(), number(), plus(), number(), close(), close());
|
||||||
|
assert_eq!(sanitize_tokens(tokens).unwrap(), results);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[should_panic]
|
||||||
|
fn wrong_parentheses_count() {
|
||||||
|
let tokens = vec![open(), number(), close(), close()];
|
||||||
|
sanitize_tokens(tokens).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[should_panic]
|
||||||
|
fn wrong_parentheses_count_2() {
|
||||||
|
let tokens = vec![open(), open(), number(), close()];
|
||||||
|
sanitize_tokens(tokens).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[should_panic]
|
||||||
|
fn wrong_parentheses_count_hard() {
|
||||||
|
let tokens = vec![open(), open(), number(), close(), equals(), x(), close()];
|
||||||
|
sanitize_tokens(tokens).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[should_panic]
|
||||||
|
fn wrong_equal_count() {
|
||||||
|
let tokens = vec![number(), equals(), number(), equals(), number()];
|
||||||
|
sanitize_tokens(tokens).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[should_panic]
|
||||||
|
fn starts_with_equal() {
|
||||||
|
let tokens = vec![equals(), number(), plus(), number()];
|
||||||
|
sanitize_tokens(tokens).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[should_panic]
|
||||||
|
fn ends_with_equal() {
|
||||||
|
let tokens = vec![number(), plus(), number(), equals()];
|
||||||
|
sanitize_tokens(tokens).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[should_panic]
|
||||||
|
fn ends_with_minus() {
|
||||||
|
let tokens = vec![number(), plus(), number(), minus()];
|
||||||
|
sanitize_tokens(tokens).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[should_panic]
|
||||||
|
fn ends_with_operator() {
|
||||||
|
let tokens = vec![number(), plus(), number(), divided_by()];
|
||||||
|
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());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,301 @@
|
||||||
|
use super::Token;
|
||||||
|
|
||||||
|
fn check_number(my_string: String, i: usize) -> Result<Token, String> {
|
||||||
|
println!("Checking number at index {i}");
|
||||||
|
if my_string.as_str() == "." {
|
||||||
|
return Err(format!("unexpected token: `.` at position {i}"));
|
||||||
|
}
|
||||||
|
let find = my_string.find(".");
|
||||||
|
if let Some(j) = find {
|
||||||
|
let find = my_string[j+1..].find(".");
|
||||||
|
if let Some(_) = find {
|
||||||
|
return Err(format!("unexpected token: `{my_string}` at position {i}"))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(Token::Number(my_string))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn tokenize(query: &str) -> Result<Vec<Token>, String> {
|
||||||
|
let mut tokens: Vec<Token> = vec![];
|
||||||
|
let mut my_string = String::new();
|
||||||
|
let mut is_last_number = false;
|
||||||
|
let mut is_last_variable = false;
|
||||||
|
|
||||||
|
for (i, token) in query.chars().enumerate() {
|
||||||
|
let mut is_still_number = false;
|
||||||
|
let mut is_still_variable = false;
|
||||||
|
let mut is_pop_needed = true;
|
||||||
|
match token {
|
||||||
|
'+' => tokens.push(Token::Addition()),
|
||||||
|
'-' => tokens.push(Token::Substraction()),
|
||||||
|
'*' => tokens.push(Token::Multiplication()),
|
||||||
|
'/' => tokens.push(Token::Division()),
|
||||||
|
'%' => tokens.push(Token::Modulo()),
|
||||||
|
'^' => tokens.push(Token::Exponentiation()),
|
||||||
|
'=' => tokens.push(Token::Equal()),
|
||||||
|
'(' => tokens.push(Token::OpenParenthesis()),
|
||||||
|
')' => tokens.push(Token::CloseParenthesis()),
|
||||||
|
'a'..='z' | 'A'..='Z' => {
|
||||||
|
if is_last_number == true {
|
||||||
|
let mut tmp_token = None;
|
||||||
|
if is_pop_needed {
|
||||||
|
tmp_token = tokens.pop();
|
||||||
|
}
|
||||||
|
tokens.push(check_number(my_string, i)?);
|
||||||
|
if let Some(tok) = tmp_token {
|
||||||
|
tokens.push(tok);
|
||||||
|
}
|
||||||
|
is_last_number = false;
|
||||||
|
my_string = String::new();
|
||||||
|
}
|
||||||
|
if is_last_variable == false {
|
||||||
|
is_last_variable = true;
|
||||||
|
}
|
||||||
|
is_still_variable = true;
|
||||||
|
my_string += &token.to_string();
|
||||||
|
},
|
||||||
|
'0'..='9' | '.' => {
|
||||||
|
if is_last_variable == true {
|
||||||
|
let mut tmp_token = None;
|
||||||
|
if is_pop_needed {
|
||||||
|
tmp_token = tokens.pop();
|
||||||
|
}
|
||||||
|
match my_string.as_str() {
|
||||||
|
"i" => tokens.push(Token::ImaginaryUnit()),
|
||||||
|
_ => tokens.push(Token::Variable(my_string)),
|
||||||
|
}
|
||||||
|
if let Some(tok) = tmp_token {
|
||||||
|
tokens.push(tok);
|
||||||
|
}
|
||||||
|
is_last_variable = false;
|
||||||
|
my_string = String::new();
|
||||||
|
}
|
||||||
|
if is_last_number == false {
|
||||||
|
is_last_number = true;
|
||||||
|
}
|
||||||
|
is_still_number = true;
|
||||||
|
my_string += &token.to_string();
|
||||||
|
},
|
||||||
|
' ' => is_pop_needed = false,
|
||||||
|
_ => return Err(format!("unexpected token: `{token}` at position {i}")),
|
||||||
|
};
|
||||||
|
if is_last_variable && !is_still_variable {
|
||||||
|
let mut tmp_token = None;
|
||||||
|
if is_pop_needed {
|
||||||
|
tmp_token = tokens.pop();
|
||||||
|
}
|
||||||
|
match my_string.as_str() {
|
||||||
|
"i" => tokens.push(Token::ImaginaryUnit()),
|
||||||
|
_ => tokens.push(Token::Variable(my_string)),
|
||||||
|
}
|
||||||
|
if let Some(tok) = tmp_token {
|
||||||
|
tokens.push(tok);
|
||||||
|
}
|
||||||
|
is_last_variable = false;
|
||||||
|
my_string = String::new();
|
||||||
|
}
|
||||||
|
else if is_last_number && !is_still_number {
|
||||||
|
let mut tmp_token = None;
|
||||||
|
if is_pop_needed {
|
||||||
|
tmp_token = tokens.pop();
|
||||||
|
}
|
||||||
|
tokens.push(check_number(my_string, i)?);
|
||||||
|
if let Some(tok) = tmp_token {
|
||||||
|
tokens.push(tok);
|
||||||
|
}
|
||||||
|
is_last_number = false;
|
||||||
|
my_string = String::new();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if is_last_variable {
|
||||||
|
match my_string.as_str() {
|
||||||
|
"i" => tokens.push(Token::ImaginaryUnit()),
|
||||||
|
_ => tokens.push(Token::Variable(my_string)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if is_last_number {
|
||||||
|
tokens.push(check_number(my_string, query.len())?);
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(tokens)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use super::*;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn addition() {
|
||||||
|
let query = "+";
|
||||||
|
let result: Vec<Token> = vec![Token::Addition()];
|
||||||
|
|
||||||
|
assert_eq!(tokenize(query).unwrap(), result);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn substraction() {
|
||||||
|
let query = "-";
|
||||||
|
let result: Vec<Token> = vec![Token::Substraction()];
|
||||||
|
|
||||||
|
assert_eq!(tokenize(query).unwrap(), result);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn multiplication() {
|
||||||
|
let query = "*";
|
||||||
|
let result: Vec<Token> = vec![Token::Multiplication()];
|
||||||
|
|
||||||
|
assert_eq!(tokenize(query).unwrap(), result);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn division() {
|
||||||
|
let query = "/";
|
||||||
|
let result: Vec<Token> = vec![Token::Division()];
|
||||||
|
|
||||||
|
assert_eq!(tokenize(query).unwrap(), result);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn modulo() {
|
||||||
|
let query = "%";
|
||||||
|
let result: Vec<Token> = vec![Token::Modulo()];
|
||||||
|
|
||||||
|
assert_eq!(tokenize(query).unwrap(), result);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn exponentiation() {
|
||||||
|
let query = "^";
|
||||||
|
let result: Vec<Token> = vec![Token::Exponentiation()];
|
||||||
|
|
||||||
|
assert_eq!(tokenize(query).unwrap(), result);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn equal() {
|
||||||
|
let query = "=";
|
||||||
|
let result: Vec<Token> = vec![Token::Equal()];
|
||||||
|
|
||||||
|
assert_eq!(tokenize(query).unwrap(), result);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn open_parenthesis() {
|
||||||
|
let query = "(";
|
||||||
|
let result: Vec<Token> = vec![Token::OpenParenthesis()];
|
||||||
|
|
||||||
|
assert_eq!(tokenize(query).unwrap(), result);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn close_parenthesis() {
|
||||||
|
let query = ")";
|
||||||
|
let result: Vec<Token> = vec![Token::CloseParenthesis()];
|
||||||
|
|
||||||
|
assert_eq!(tokenize(query).unwrap(), result);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn imaginary_unit() {
|
||||||
|
let query = "i";
|
||||||
|
let result: Vec<Token> = vec![Token::ImaginaryUnit()];
|
||||||
|
|
||||||
|
assert_eq!(tokenize(query).unwrap(), result);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn variable() {
|
||||||
|
let query = "variable";
|
||||||
|
let result: Vec<Token> = vec![Token::Variable(String::from("variable"))];
|
||||||
|
|
||||||
|
assert_eq!(tokenize(query).unwrap(), result);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn variable_double_i() {
|
||||||
|
let query = "ii";
|
||||||
|
let result: Vec<Token> = vec![Token::Variable(String::from("ii"))];
|
||||||
|
|
||||||
|
assert_eq!(tokenize(query).unwrap(), result);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn number_natural() {
|
||||||
|
let query = "123456";
|
||||||
|
let result: Vec<Token> = vec![Token::Number(String::from("123456"))];
|
||||||
|
|
||||||
|
assert_eq!(tokenize(query).unwrap(), result);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn number_rational() {
|
||||||
|
let query = "123.456";
|
||||||
|
let result: Vec<Token> = vec![Token::Number(String::from("123.456"))];
|
||||||
|
|
||||||
|
assert_eq!(tokenize(query).unwrap(), result);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn number_point_something() {
|
||||||
|
let query = ".123456";
|
||||||
|
let result: Vec<Token> = vec![Token::Number(String::from(".123456"))];
|
||||||
|
|
||||||
|
assert_eq!(tokenize(query).unwrap(), result);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn number_trailing_point() {
|
||||||
|
let query = "123456.";
|
||||||
|
let result: Vec<Token> = vec![Token::Number(String::from("123456."))];
|
||||||
|
|
||||||
|
assert_eq!(tokenize(query).unwrap(), result);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn number_variable() {
|
||||||
|
let query = "23x";
|
||||||
|
let result: Vec<Token> = vec![Token::Number(String::from("23")), Token::Variable(String::from("x"))];
|
||||||
|
assert_eq!(tokenize(query).unwrap(), result);
|
||||||
|
|
||||||
|
let query = "23i";
|
||||||
|
let result: Vec<Token> = vec![Token::Number(String::from("23")), Token::ImaginaryUnit()];
|
||||||
|
assert_eq!(tokenize(query).unwrap(), result);
|
||||||
|
|
||||||
|
let query = "x23";
|
||||||
|
let result: Vec<Token> = vec![Token::Variable(String::from("x")), Token::Number(String::from("23"))];
|
||||||
|
assert_eq!(tokenize(query).unwrap(), result);
|
||||||
|
|
||||||
|
let query = "i23";
|
||||||
|
let result: Vec<Token> = vec![Token::ImaginaryUnit(), Token::Number(String::from("23"))];
|
||||||
|
assert_eq!(tokenize(query).unwrap(), result);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[should_panic]
|
||||||
|
fn number_double_point() {
|
||||||
|
let query = "12.34.56";
|
||||||
|
tokenize(query).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[should_panic]
|
||||||
|
fn number_double_point_in_a_row() {
|
||||||
|
let query = "123..456";
|
||||||
|
tokenize(query).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[should_panic]
|
||||||
|
fn number_only_point() {
|
||||||
|
let query = ".";
|
||||||
|
tokenize(query).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[should_panic]
|
||||||
|
fn invalid_token() {
|
||||||
|
let query = "324*43224+243_+234=234";
|
||||||
|
tokenize(query).unwrap();
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue