//! # A term that can be the input for calculation //! //! Short description //! //! Details //! //! ## Section 1 //! //! ## Section 2 //// ATTRIBUTES //////////////////////////////////////////////////////////////////////////////////// // we want docs #![warn(missing_docs)] #![warn(rustdoc::missing_crate_level_docs)] // we want Debug everywhere. #![warn(missing_debug_implementations)] // enable clippy's extra lints, the pedantic version #![warn(clippy::pedantic)] use std::collections::VecDeque; //// IMPORTS /////////////////////////////////////////////////////////////////////////////////////// pub use super::{Error, Result, Value, base::{self, *}}; use crate::logger::*; //// TYPES ///////////////////////////////////////////////////////////////////////////////////////// //// CONSTANTS ///////////////////////////////////////////////////////////////////////////////////// //// STATICS /////////////////////////////////////////////////////////////////////////////////////// //// MACROS //////////////////////////////////////////////////////////////////////////////////////// //// ENUMS ///////////////////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////////////////////////////// /// ## Parsed value to be calculated /// /// This enum represents anything that goes to the output queue of [`Term::prepare()`] and will /// then be used to actually calculate something in [`Term::process()`]. #[derive(Debug)] enum Token { /// Some kind of operator Operator(Operator), /// A concrete value that we can calculate something with. May be a constant, integer, float, /// etc. Value(super::base::Value), /// A variable of some kind that will be present in the result Variable(char), } //// STRUCTS /////////////////////////////////////////////////////////////////////////////////////// /// ## Term that can be calculated /// /// Represents a signular term, that can be calculated. Terms will be evaluated by the [`Term::prepare`] /// function, afterwards calculated (as much as possible) in the [`Term::process`] function. /// #[derive(Debug)] pub struct Term { /// the original expression to calculate pub original: String, /// the filtered text of the expression, only with relevant information pub text: String, /// the calculated result, may be of diffrent types, see [`crate::math::calculator::result`]. pub result: Option>, ///////////////////////////////////// ///// internal values following ///// ///////////////////////////////////// operator_stack: Vec, output_queue: VecDeque } //// IMPLEMENTATION //////////////////////////////////////////////////////////////////////////////// impl Term { /// Build a new term from an expression /// /// Invalid terms will result in an [`Err`]. pub fn new(orig: String) -> Result { return Ok( Term { original: orig, text: String::new(), // will be initialized in `prepare()` result: None, operator_stack: Vec::new(), output_queue: VecDeque::new() } ) } /// Prepare the term for the processing. pub fn prepare(&mut self) -> Result<()> { trace!("preparing term: {:#?}", self); self.text = Self::filter(&self.original)?; // Storage for unfinished tokens let mut unfinished_chars: Vec = Vec::new(); for (index, c) in self.original.chars().enumerate() { // this will be a mess, but it has to be before i can sort the mess. match c { // TODO: make function to check if character is an operator, use it _ => { let reason = format!("The meaning of '{c}' could not be identified."); warn!(reason); return Err(Error::SyntaxError(reason)); } } } Ok(()) } /// Process a prepared term, calculating it's result pub fn process(&mut self) -> Result<()> { debug!("processing term: {:#?}", self); debug!("queue: {:#?}", self.output_queue); // TODO: process RPN and set result self.result = Some(Ok(19.into())); Ok(()) } /// Convert a character into a token /// /// Returns: A tuple with a [`Token`] and a [`bool`]. If the bool is false, the [`Token`] is /// marked as "incomplete", meaning that the character cannot be used yet. fn to_tok(s: Vec) -> Result { Ok(19.into()) } /// only leave relevant chars for calculation // TODO: make function to check if character is an operator, use it fn filter(s: &str) -> Result { // pre checks // NOTE: Apperently, "alphanumeric" in Rust is a pretty broad term. // Even CJK characters or Japanese Kana are allowed: // - 'さ' alphanumeric // - '数' alphanumeric // - '学' alphanumeric // - '+' not alphanumeric for c in s.chars() { #[cfg(debug_assertions)] { debug!("filter checks for '{c}': alphanumeric: {} allowed special: {} EXCEPT IF ascii control: {} ", !c.is_alphanumeric(), !Self::is_allowed_special_c(&c), c.is_ascii_control(), ) } if ( !c.is_alphanumeric() || !Self::is_allowed_special_c(&c) ) && ( c.is_ascii_control() ) { // TODO: allow any unicode char to be a variable let reason = format!("'{c}' is not a valid character, only alphanumeric, punctuation, operators are allowed."); warn!(reason); return Err(Error::SyntaxError(reason)); } } // filter out single chars let mut filtered = String::new(); for c in s.chars() { if !Self::is_ignore(&c) { filtered.push(c); } } return Ok(filtered) } /// check if we should ignore this character fn is_ignore(c: &char) -> bool { match *c { ' ' => true, _ => false } } /// allowed special chars fn is_allowed_special_c(c: &char) -> bool { match *c { '+' | '-' | '*' | '/' | '%' => true, _ => false } } } //////////////////////////////////////////////////////////////////////////////////////////////////// /// Helper methods for Tokens impl Token { } //////////////////////////////////////////////////////////////////////////////////////////////////// impl From for Token where T: Into, T: PrimInt, u128: TryFrom { fn from(value: T) -> Self { Token::Value(base::Value::from(value)) } } //// PUBLIC FUNCTIONS ////////////////////////////////////////////////////////////////////////////// //// PRIVATE FUNCTIONS ///////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////////////////////////