mech-interpreter 0.3.2

The Mech language runtime.
Documentation
use crate::*;
#[cfg(feature = "convert")]
use crate::stdlib::convert::ConvertKind;

// Literals
// ----------------------------------------------------------------------------

pub fn literal(ltrl: &Literal, p: &Interpreter) -> MResult<Value> {
  match &ltrl {
    Literal::Empty(_) => Ok(empty()),
    #[cfg(feature = "bool")]
    Literal::Boolean(bln) => Ok(boolean(bln)),
    Literal::Number(num) => number(num, p),
    #[cfg(feature = "string")]
    Literal::String(strng) => Ok(string(strng)),
    #[cfg(feature = "atom")]
    Literal::Atom(atm) => Ok(atom(atm, p)),
    #[cfg(feature = "kind_annotation")]
    Literal::Kind(knd) => kind_value(knd, p),
    #[cfg(feature = "convert")]
    Literal::TypedLiteral((ltrl,kind)) => typed_literal(ltrl,kind,p),
    _ => Err(MechError::new(
      FeatureNotEnabledError, None
      ).with_compiler_loc())
  }
}

#[cfg(feature = "kind_annotation")]
pub fn kind_value(knd: &NodeKind, p: &Interpreter) -> MResult<Value> {
  let kind = kind_annotation(knd, p)?;
  Ok(Value::Kind(kind.to_value_kind(&p.state.borrow().kinds)?))
}

pub fn kind_annotation(knd: &NodeKind, p: &Interpreter) -> MResult<Kind> {
  match knd {
    NodeKind::Kind(knd) => {
      let knda = kind_annotation(knd, p)?;
      Ok(Kind::Kind(Box::new(knda)))
    }
    NodeKind::Any => Ok(Kind::Any),
    NodeKind::Atom(atm_identifier) => {
      let id = atm_identifier.hash();
      let name = atm_identifier.to_string();
      Ok(Kind::Atom(id, name))
    }
    NodeKind::Empty => Ok(Kind::Empty),
    NodeKind::Record(elements) => {
      let mut knds = vec![];
      for (id, knd) in elements {
        let knda = kind_annotation(knd, p)?;
        knds.push((id.to_string().clone(), knda));
      }
      Ok(Kind::Record(knds))
    }
    NodeKind::Tuple(elements) => {
      let mut knds = vec![];
      for knd in elements {
        let knda = kind_annotation(knd, p)?;
        knds.push(knda);
      }
      Ok(Kind::Tuple(knds))
    }
    NodeKind::Map(keys, vals) => {
      let key_knd = kind_annotation(keys, p)?;
      let val_knd = kind_annotation(vals, p)?;
      Ok(Kind::Map(Box::new(key_knd), Box::new(val_knd)))
    }
    NodeKind::Scalar(id) => {
      let kind_id = id.hash();
      Ok(Kind::Scalar(kind_id))
    }
    NodeKind::Matrix((knd, size)) => {
      let knda = kind_annotation(knd, p)?;
      let mut dims = vec![];
      for dim in size {
        let dim_val = literal(dim, p)?;
        match dim_val {
          Value::Empty => { dims.push(0); }
          _ => {
            match dim_val.as_usize() {
              Ok(size_val) => dims.push(size_val.clone()),
              Err(_) => { return Err(MechError::new(
                ExpectedNumericForKindSizeError, None
                ).with_compiler_loc())
              }
            } 
          }
        }
      }
      Ok(Kind::Matrix(Box::new(knda.clone()),dims))
    }
    NodeKind::Option(knd) => {
      let knda = kind_annotation(knd, p)?;
      Ok(Kind::Option(Box::new(knda)))
    }
    NodeKind::Table((elements, size)) => {
      let mut knds = vec![];
      for (id, knd) in elements {
        let knda = kind_annotation(knd, p)?;
        knds.push((id.to_string().clone(), knda));
      }
      let size_val = literal(size, p)?;
      let size_val = match size_val {
        Value::Empty => 0,
        _ => {
          match size_val.as_usize() {
            Ok(size_val) => size_val,
            Err(_) => { return Err(MechError::new(
              ExpectedNumericForKindSizeError, None
              ).with_compiler_loc())
            }
          }
        }
      };
      Ok(Kind::Table(knds, size_val))
    }
    NodeKind::Set(knd, size) => {
      let knda = kind_annotation(knd, p)?;
      let size_val = match size {
        Some(size) => literal(size, p)?,
        None => Value::Empty,
      };
      match size_val.as_usize() {
        Ok(size_val) => Ok(Kind::Set(Box::new(knda.clone()), Some(size_val))),
        Err(_) => Ok(Kind::Set(Box::new(knda.clone()), None)),
      }
    }
  }
}

#[cfg(feature = "convert")]
pub fn typed_literal(ltrl: &Literal, knd_attn: &KindAnnotation, p: &Interpreter) -> MResult<Value> {
  let value = literal(ltrl,p)?;
  let kind = kind_annotation(&knd_attn.kind, p)?;
  let args = vec![value, kind.to_value(&p.state.borrow().kinds)?];
  let convert_fxn = ConvertKind{}.compile(&args)?;
  convert_fxn.solve();
  let converted_result = convert_fxn.out();
  p.state.borrow_mut().add_plan_step(convert_fxn);
  Ok(converted_result)
}

#[cfg(feature = "atom")]
pub fn atom(atm: &Atom, p: &Interpreter) -> Value {
  let id = atm.name.hash();
  let state = p.state.borrow();
  let dictionary = state.dictionary.clone();
  {
    let mut dictionary_brrw = dictionary.borrow_mut();
    dictionary_brrw.insert(id, atm.name.to_string());
  }
  Value::Atom(Ref::new(MechAtom((id, dictionary))))
}

pub fn number(num: &Number, p: &Interpreter) -> MResult<Value> {
  match num {
    Number::Real(num) => real(num, p),
    #[cfg(feature = "complex")]
    Number::Complex(num) => complex(num, p),
    _ => panic!("Number type not supported."),
  }
}

#[cfg(feature = "complex")]
fn complex(num: &C64Node, p: &Interpreter) -> MResult<Value> {
  let im: f64 = match real(&num.imaginary.number, p)?.as_f64() {
    Ok(val) => *val.borrow(),
    Err(_) => 0.0,
  };
  let result = match &num.real {
    Some(real_val) => {
      let re: f64 = match real(&real_val, p)?.as_f64() {
        Ok(val) => *val.borrow(),
        Err(_) => 0.0,
      };      
      Value::C64(Ref::new(C64::new(re, im)))
    },
    None => Value::C64(Ref::new(C64::new(0.0, im))),
  };
  Ok(result)
}

pub fn real(rl: &RealNumber, p: &Interpreter) -> MResult<Value> {
  let result = match rl {
    #[cfg(feature = "math_neg")]
    RealNumber::Negated(num) => negated(num, p)?,
    #[cfg(feature = "f64")]
    RealNumber::Integer(num) => integer(num),
    #[cfg(feature = "floats")]
    RealNumber::Float(num) => float(num),
    #[cfg(feature = "i64")]
    RealNumber::Decimal(num) => dec(num),
    #[cfg(feature = "i64")]
    RealNumber::Hexadecimal(num) => hex(num),
    #[cfg(feature = "i64")]
    RealNumber::Octal(num) => oct(num),
    #[cfg(feature = "i64")]
    RealNumber::Binary(num) => binary(num),
    #[cfg(feature = "floats")]
    RealNumber::Scientific(num) => scientific(num),
    #[cfg(feature = "rational")]
    RealNumber::Rational(num) => rational(num),
    #[cfg(feature = "convert")]
    RealNumber::TypedInteger((num_tkn, kind)) => {
      let num: Literal = Literal::Number(Number::Real(RealNumber::Integer(num_tkn.clone())));
      typed_literal(&num, kind, p)?
    },
    _ => panic!("Number type not supported."),
  };
  Ok(result)
}

#[cfg(feature = "math_neg")]
pub fn negated(num: &RealNumber, p: &Interpreter) -> MResult<Value> {
  let num_val = real(&num, p)?;
  let result = match num_val {
    #[cfg(feature = "i8")]
    Value::I8(val) => Value::I8(Ref::new(-*val.borrow())),
    #[cfg(feature = "i16")]
    Value::I16(val) => Value::I16(Ref::new(-*val.borrow())),
    #[cfg(feature = "i32")]
    Value::I32(val) => Value::I32(Ref::new(-*val.borrow())),
    #[cfg(feature = "i64")]
    Value::I64(val) => Value::I64(Ref::new(-*val.borrow())),
    #[cfg(feature = "i128")]
    Value::I128(val) => Value::I128(Ref::new(-*val.borrow())),
    #[cfg(feature = "f64")]
    Value::F64(val) => Value::F64(Ref::new(-(*val.borrow()))),
    #[cfg(feature = "f32")]
    Value::F32(val) => Value::F32(Ref::new(-(*val.borrow()))),
    x => panic!("Negation is only supported for integer and float types, got {:?}", x),
  };
  Ok(result)
}

#[cfg(feature = "rational")]
pub fn rational(rat: &(Token,Token)) -> Value {
  let (num, denom) = rat;
  let num = num.chars.iter().collect::<String>().parse::<i64>().unwrap();
  let denom = denom.chars.iter().collect::<String>().parse::<i64>().unwrap();
  if denom == 0 {
    panic!("Denominator cannot be zero in a rational number");
  }
  let rat_num = R64::new(num, denom);
  Value::R64(Ref::new(rat_num))
}

#[cfg(feature = "i64")]
pub fn dec(bnry: &Token) -> Value {
  let binary_str: String = bnry.chars.iter().collect();
  let num = i64::from_str_radix(&binary_str, 10).unwrap();
  Value::I64(Ref::new(num))
}

#[cfg(feature = "i64")]
pub fn binary(bnry: &Token) -> Value {
  let binary_str: String = bnry.chars.iter().collect();
  let num = i64::from_str_radix(&binary_str, 2).unwrap();
  Value::I64(Ref::new(num))
}

#[cfg(feature = "i64")]
pub fn oct(octl: &Token) -> Value {
  let hex_str: String = octl.chars.iter().collect();
  let num = i64::from_str_radix(&hex_str, 8).unwrap();
  Value::I64(Ref::new(num))
}

#[cfg(feature = "i64")]
pub fn hex(hxdcml: &Token) -> Value {
  let hex_str: String = hxdcml.chars.iter().collect();
  let num = i64::from_str_radix(&hex_str, 16).unwrap();
  Value::I64(Ref::new(num))
}

#[cfg(feature = "f64")]
pub fn scientific(sci: &(Base,Exponent)) -> Value {
  let (base,exp): &(Base,Exponent) = sci;
  let (whole,part): &(Whole,Part) = base;
  let (sign,exp_whole, exp_part): &(Sign, Whole, Part) = exp;

  let a = whole.chars.iter().collect::<String>();
  let b = part.chars.iter().collect::<String>();
  let c = exp_whole.chars.iter().collect::<String>();
  let d = exp_part.chars.iter().collect::<String>();
  let num_f64: f64 = format!("{}.{}",a,b).parse::<f64>().unwrap();
  let mut exp_f64: f64 = format!("{}.{}",c,d).parse::<f64>().unwrap();
  if *sign {
    exp_f64 = -exp_f64;
  }
  let num = num_f64 * 10f64.powf(exp_f64);
  Value::F64(Ref::new(num))
}

#[cfg(feature = "floats")]
pub fn float(flt: &(Token,Token)) -> Value {
  let a = flt.0.chars.iter().collect::<String>();
  let b = flt.1.chars.iter().collect::<String>();
  let num: f64 = format!("{}.{}",a,b).parse::<f64>().unwrap();
  Value::F64(Ref::new(num))
}

#[cfg(feature = "f64")]
pub fn integer(int: &Token) -> Value {
  let num: f64 = int.chars.iter().collect::<String>().parse::<f64>().unwrap();
  Value::F64(Ref::new(num))
}

#[cfg(feature = "string")]
pub fn string(tkn: &MechString) -> Value {
  let strng: String = tkn.text.chars.iter().collect::<String>();
  Value::String(Ref::new(strng))
}

pub fn empty() -> Value {
  Value::Empty
}

#[cfg(feature = "bool")]
pub fn boolean(tkn: &Token) -> Value {
  let val = match tkn.kind {
    TokenKind::True => true,
    TokenKind::False => false,
    _ => unreachable!(),
  };
  Value::Bool(Ref::new(val))
}

//            Err(_) => { return Err(MechError{file: file!().to_string(), tokens: knd.tokens(), msg: "".to_string(), id: line!(), kind: MechErrorKind::ExpectedNumericForSize});} 
#[derive(Debug, Clone)]
pub struct ExpectedNumericForKindSizeError;
impl MechErrorKind for ExpectedNumericForKindSizeError {
  fn name(&self) -> &str {
    "ExpectedNumericForKindSize"
  }
  fn message(&self) -> String {
    "Expected a numeric value for kind size, but received a non-numeric value.".to_string()
  }
}