use crate::{
hash::hash_f64,
imag,
instruction::MemoryReference,
parser::{lex, parse_expression, ParseError},
program::{disallow_leftover, ParseProgramError},
quil::Quil,
real,
};
use lexical::{format, to_string_with_options, WriteFloatOptions};
use nom_locate::LocatedSpan;
use num_complex::Complex64;
use once_cell::sync::Lazy;
use std::{
collections::HashMap,
f64::consts::PI,
fmt,
hash::{Hash, Hasher},
num::NonZeroI32,
ops::{Add, AddAssign, BitXor, BitXorAssign, Div, DivAssign, Mul, MulAssign, Sub, SubAssign},
str::FromStr,
};
#[cfg(test)]
use proptest_derive::Arbitrary;
mod simplification;
#[derive(Clone, Debug, PartialEq, Eq, thiserror::Error)]
pub enum EvaluationError {
#[error("There wasn't enough information to completely evaluate the expression.")]
Incomplete,
#[error("The operation expected a real number but received a complex one.")]
NumberNotReal,
#[error("The operation expected a number but received a different type of expression.")]
NotANumber,
}
#[derive(Clone, Debug)]
pub enum Expression {
Address(MemoryReference),
FunctionCall(FunctionCallExpression),
Infix(InfixExpression),
Number(Complex64),
PiConstant,
Prefix(PrefixExpression),
Variable(String),
}
#[derive(Clone, Debug, PartialEq, Eq, Hash)]
pub struct FunctionCallExpression {
pub function: ExpressionFunction,
pub expression: Box<Expression>,
}
impl FunctionCallExpression {
pub fn new(function: ExpressionFunction, expression: Box<Expression>) -> Self {
Self {
function,
expression,
}
}
}
#[derive(Clone, Debug, PartialEq, Eq, Hash)]
pub struct InfixExpression {
pub left: Box<Expression>,
pub operator: InfixOperator,
pub right: Box<Expression>,
}
impl InfixExpression {
pub fn new(left: Box<Expression>, operator: InfixOperator, right: Box<Expression>) -> Self {
Self {
left,
operator,
right,
}
}
}
#[derive(Clone, Debug, PartialEq, Eq, Hash)]
pub struct PrefixExpression {
pub operator: PrefixOperator,
pub expression: Box<Expression>,
}
impl PrefixExpression {
pub fn new(operator: PrefixOperator, expression: Box<Expression>) -> Self {
Self {
operator,
expression,
}
}
}
impl PartialEq for Expression {
fn eq(&self, other: &Self) -> bool {
match (self, other) {
(Self::Address(left), Self::Address(right)) => left == right,
(Self::Infix(left), Self::Infix(right)) => left == right,
(Self::Number(left), Self::Number(right)) => {
(left.re == right.re || left.re.is_nan() && right.re.is_nan())
&& (left.im == right.im || left.im.is_nan() && right.im.is_nan())
}
(Self::Prefix(left), Self::Prefix(right)) => left == right,
(Self::FunctionCall(left), Self::FunctionCall(right)) => left == right,
(Self::Variable(left), Self::Variable(right)) => left == right,
(Self::PiConstant, Self::PiConstant) => true,
_ => false,
}
}
}
impl Eq for Expression {}
impl Hash for Expression {
fn hash<H: Hasher>(&self, state: &mut H) {
match self {
Self::Address(m) => {
"Address".hash(state);
m.hash(state);
}
Self::FunctionCall(FunctionCallExpression {
function,
expression,
}) => {
"FunctionCall".hash(state);
function.hash(state);
expression.hash(state);
}
Self::Infix(InfixExpression {
left,
operator,
right,
}) => {
"Infix".hash(state);
operator.hash(state);
left.hash(state);
right.hash(state);
}
Self::Number(n) => {
"Number".hash(state);
if n.re.abs() > 0f64 {
hash_f64(n.re, state)
}
if n.im.abs() > 0f64 {
hash_f64(n.im, state)
}
}
Self::PiConstant => {
"PiConstant".hash(state);
}
Self::Prefix(p) => {
"Prefix".hash(state);
p.operator.hash(state);
p.expression.hash(state);
}
Self::Variable(v) => {
"Variable".hash(state);
v.hash(state);
}
}
}
}
macro_rules! impl_expr_op {
($name:ident, $name_assign:ident, $function:ident, $function_assign:ident, $operator:ident) => {
impl $name for Expression {
type Output = Self;
fn $function(self, other: Self) -> Self {
Expression::Infix(InfixExpression {
left: Box::new(self),
operator: InfixOperator::$operator,
right: Box::new(other),
})
}
}
impl $name_assign for Expression {
fn $function_assign(&mut self, other: Self) {
let temp = ::std::mem::replace(self, Self::PiConstant);
*self = temp.$function(other);
}
}
};
}
impl_expr_op!(BitXor, BitXorAssign, bitxor, bitxor_assign, Caret);
impl_expr_op!(Add, AddAssign, add, add_assign, Plus);
impl_expr_op!(Sub, SubAssign, sub, sub_assign, Minus);
impl_expr_op!(Mul, MulAssign, mul, mul_assign, Star);
impl_expr_op!(Div, DivAssign, div, div_assign, Slash);
fn calculate_infix(left: &Complex64, operator: &InfixOperator, right: &Complex64) -> Complex64 {
use InfixOperator::*;
match operator {
Caret => left.powc(*right),
Plus => left + right,
Minus => left - right,
Slash => left / right,
Star => left * right,
}
}
fn calculate_function(function: &ExpressionFunction, argument: &Complex64) -> Complex64 {
use ExpressionFunction::*;
match function {
Sine => argument.sin(),
Cis => argument.cos() + imag!(1f64) * argument.sin(),
Cosine => argument.cos(),
Exponent => argument.exp(),
SquareRoot => argument.sqrt(),
}
}
#[inline(always)]
fn is_small(x: f64) -> bool {
x.abs() < 1e-16
}
impl Expression {
pub fn simplify(&mut self) {
match self {
Expression::Address(_) | Expression::Number(_) | Expression::Variable(_) => {}
Expression::PiConstant => {
*self = Expression::Number(Complex64::from(PI));
}
_ => *self = simplification::run(self),
}
}
pub fn into_simplified(mut self) -> Self {
self.simplify();
self
}
pub fn evaluate(
&self,
variables: &HashMap<String, Complex64>,
memory_references: &HashMap<&str, Vec<f64>>,
) -> Result<Complex64, EvaluationError> {
use Expression::*;
match self {
FunctionCall(FunctionCallExpression {
function,
expression,
}) => {
let evaluated = expression.evaluate(variables, memory_references)?;
Ok(calculate_function(function, &evaluated))
}
Infix(InfixExpression {
left,
operator,
right,
}) => {
let left_evaluated = left.evaluate(variables, memory_references)?;
let right_evaluated = right.evaluate(variables, memory_references)?;
Ok(calculate_infix(&left_evaluated, operator, &right_evaluated))
}
Prefix(PrefixExpression {
operator,
expression,
}) => {
use PrefixOperator::*;
let value = expression.evaluate(variables, memory_references)?;
if matches!(operator, Minus) {
Ok(-value)
} else {
Ok(value)
}
}
Variable(identifier) => match variables.get(identifier.as_str()) {
Some(value) => Ok(*value),
None => Err(EvaluationError::Incomplete),
},
Address(memory_reference) => memory_references
.get(memory_reference.name.as_str())
.and_then(|values| {
let value = values.get(memory_reference.index as usize)?;
Some(real!(*value))
})
.ok_or(EvaluationError::Incomplete),
PiConstant => Ok(real!(PI)),
Number(number) => Ok(*number),
}
}
pub fn substitute_variables(self, variable_values: &HashMap<String, Expression>) -> Self {
use Expression::*;
match self {
FunctionCall(FunctionCallExpression {
function,
expression,
}) => Expression::FunctionCall(FunctionCallExpression {
function,
expression: expression.substitute_variables(variable_values).into(),
}),
Infix(InfixExpression {
left,
operator,
right,
}) => {
let left = left.substitute_variables(variable_values).into();
let right = right.substitute_variables(variable_values).into();
Infix(InfixExpression {
left,
operator,
right,
})
}
Prefix(PrefixExpression {
operator,
expression,
}) => Prefix(PrefixExpression {
operator,
expression: expression.substitute_variables(variable_values).into(),
}),
Variable(identifier) => match variable_values.get(identifier.as_str()) {
Some(value) => value.clone(),
None => Variable(identifier),
},
other => other,
}
}
pub fn to_real(&self) -> Result<f64, EvaluationError> {
match self {
Expression::PiConstant => Ok(PI),
Expression::Number(x) if is_small(x.im) => Ok(x.re),
Expression::Number(_) => Err(EvaluationError::NumberNotReal),
_ => Err(EvaluationError::NotANumber),
}
}
}
impl FromStr for Expression {
type Err = ParseProgramError<Self>;
fn from_str(s: &str) -> Result<Self, Self::Err> {
let input = LocatedSpan::new(s);
let tokens = lex(input)?;
disallow_leftover(parse_expression(&tokens).map_err(ParseError::from_nom_internal_err))
}
}
static FORMAT_REAL_OPTIONS: Lazy<WriteFloatOptions> = Lazy::new(|| {
WriteFloatOptions::builder()
.negative_exponent_break(NonZeroI32::new(-5))
.positive_exponent_break(NonZeroI32::new(15))
.trim_floats(true)
.build()
.expect("options are valid")
});
static FORMAT_IMAGINARY_OPTIONS: Lazy<WriteFloatOptions> = Lazy::new(|| {
WriteFloatOptions::builder()
.negative_exponent_break(NonZeroI32::new(-5))
.positive_exponent_break(NonZeroI32::new(15))
.trim_floats(false) .build()
.expect("options are valid")
});
#[inline(always)]
pub(crate) fn format_complex(value: &Complex64) -> String {
const FORMAT: u128 = format::STANDARD;
if value.re == 0f64 && value.im == 0f64 {
"0".to_owned()
} else if value.im == 0f64 {
to_string_with_options::<_, FORMAT>(value.re, &FORMAT_REAL_OPTIONS)
} else if value.re == 0f64 {
to_string_with_options::<_, FORMAT>(value.im, &FORMAT_IMAGINARY_OPTIONS) + "i"
} else {
let mut out = to_string_with_options::<_, FORMAT>(value.re, &FORMAT_REAL_OPTIONS);
if value.im > 0f64 {
out.push('+')
}
out.push_str(&to_string_with_options::<_, FORMAT>(
value.im,
&FORMAT_IMAGINARY_OPTIONS,
));
out.push('i');
out
}
}
impl Quil for Expression {
fn write(
&self,
f: &mut impl std::fmt::Write,
fall_back_to_debug: bool,
) -> Result<(), crate::quil::ToQuilError> {
use Expression::*;
match self {
Address(memory_reference) => memory_reference.write(f, fall_back_to_debug),
FunctionCall(FunctionCallExpression {
function,
expression,
}) => {
write!(f, "{function}(")?;
expression.write(f, fall_back_to_debug)?;
write!(f, ")")?;
Ok(())
}
Infix(InfixExpression {
left,
operator,
right,
}) => {
format_inner_expression(f, fall_back_to_debug, left)?;
write!(f, "{}", operator)?;
format_inner_expression(f, fall_back_to_debug, right)
}
Number(value) => write!(f, "{}", format_complex(value)).map_err(Into::into),
PiConstant => write!(f, "pi").map_err(Into::into),
Prefix(PrefixExpression {
operator,
expression,
}) => {
write!(f, "{}", operator)?;
format_inner_expression(f, fall_back_to_debug, expression)
}
Variable(identifier) => write!(f, "%{}", identifier).map_err(Into::into),
}
}
}
fn format_inner_expression(
f: &mut impl std::fmt::Write,
fall_back_to_debug: bool,
expression: &Expression,
) -> crate::quil::ToQuilResult<()> {
match expression {
Expression::Infix(InfixExpression {
left,
operator,
right,
}) => {
write!(f, "(")?;
format_inner_expression(f, fall_back_to_debug, left)?;
write!(f, "{operator}")?;
format_inner_expression(f, fall_back_to_debug, right)?;
write!(f, ")")?;
Ok(())
}
_ => expression.write(f, fall_back_to_debug),
}
}
#[cfg(test)]
mod test {
use crate::{
expression::{
Expression, InfixExpression, InfixOperator, PrefixExpression, PrefixOperator,
},
quil::Quil,
real,
};
#[test]
fn formats_nested_expression() {
let expression = Expression::Infix(InfixExpression {
left: Box::new(Expression::Prefix(PrefixExpression {
operator: PrefixOperator::Minus,
expression: Box::new(Expression::Number(real!(3f64))),
})),
operator: InfixOperator::Star,
right: Box::new(Expression::Infix(InfixExpression {
left: Box::new(Expression::PiConstant),
operator: InfixOperator::Slash,
right: Box::new(Expression::Number(real!(2f64))),
})),
});
assert_eq!(expression.to_quil_or_debug(), "-3*(pi/2)");
}
}
#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash)]
#[cfg_attr(test, derive(Arbitrary))]
pub enum ExpressionFunction {
Cis,
Cosine,
Exponent,
Sine,
SquareRoot,
}
impl fmt::Display for ExpressionFunction {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
use ExpressionFunction::*;
write!(
f,
"{}",
match self {
Cis => "cis",
Cosine => "cos",
Exponent => "exp",
Sine => "sin",
SquareRoot => "sqrt",
}
)
}
}
#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash)]
#[cfg_attr(test, derive(Arbitrary))]
pub enum PrefixOperator {
Plus,
Minus,
}
impl fmt::Display for PrefixOperator {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
use PrefixOperator::*;
write!(
f,
"{}",
match self {
Plus => "",
Minus => "-",
}
)
}
}
#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash)]
#[cfg_attr(test, derive(Arbitrary))]
pub enum InfixOperator {
Caret,
Plus,
Minus,
Slash,
Star,
}
impl fmt::Display for InfixOperator {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
use InfixOperator::*;
write!(
f,
"{}",
match self {
Caret => "^",
Plus => "+",
Minus => " - ",
Slash => "/",
Star => "*",
}
)
}
}
#[cfg(test)]
#[allow(clippy::arc_with_non_send_sync)]
mod tests {
use super::*;
use crate::reserved::ReservedToken;
use proptest::prelude::*;
use std::collections::hash_map::DefaultHasher;
use std::collections::HashSet;
#[inline]
fn hash_to_u64<T: Hash>(t: &T) -> u64 {
let mut s = DefaultHasher::new();
t.hash(&mut s);
s.finish()
}
#[test]
fn simplify_and_evaluate() {
use Expression::*;
let one = real!(1.0);
let empty_variables = HashMap::new();
let mut variables = HashMap::new();
variables.insert("foo".to_owned(), real!(10f64));
variables.insert("bar".to_owned(), real!(100f64));
let empty_memory = HashMap::new();
let mut memory_references = HashMap::new();
memory_references.insert("theta", vec![1.0, 2.0]);
memory_references.insert("beta", vec![3.0, 4.0]);
struct TestCase<'a> {
expression: Expression,
variables: &'a HashMap<String, Complex64>,
memory_references: &'a HashMap<&'a str, Vec<f64>>,
simplified: Expression,
evaluated: Result<Complex64, EvaluationError>,
}
let cases: Vec<TestCase> = vec![
TestCase {
expression: Number(one),
variables: &empty_variables,
memory_references: &empty_memory,
simplified: Number(one),
evaluated: Ok(one),
},
TestCase {
expression: Expression::Prefix(PrefixExpression {
operator: PrefixOperator::Minus,
expression: Box::new(Number(real!(1f64))),
}),
variables: &empty_variables,
memory_references: &empty_memory,
simplified: Number(real!(-1f64)),
evaluated: Ok(real!(-1f64)),
},
TestCase {
expression: Expression::Variable("foo".to_owned()),
variables: &variables,
memory_references: &empty_memory,
simplified: Expression::Variable("foo".to_owned()),
evaluated: Ok(real!(10f64)),
},
TestCase {
expression: Expression::from_str("%foo + %bar").unwrap(),
variables: &variables,
memory_references: &empty_memory,
simplified: Expression::from_str("%foo + %bar").unwrap(),
evaluated: Ok(real!(110f64)),
},
TestCase {
expression: Expression::FunctionCall(FunctionCallExpression {
function: ExpressionFunction::Sine,
expression: Box::new(Expression::Number(real!(PI / 2f64))),
}),
variables: &variables,
memory_references: &empty_memory,
simplified: Number(real!(1f64)),
evaluated: Ok(real!(1f64)),
},
TestCase {
expression: Expression::from_str("theta[1] * beta[0]").unwrap(),
variables: &empty_variables,
memory_references: &memory_references,
simplified: Expression::from_str("theta[1] * beta[0]").unwrap(),
evaluated: Ok(real!(6.0)),
},
];
for mut case in cases {
let evaluated = case
.expression
.evaluate(case.variables, case.memory_references);
assert_eq!(evaluated, case.evaluated);
case.expression.simplify();
assert_eq!(case.expression, case.simplified);
}
}
fn parenthesized(expression: &Expression) -> String {
use Expression::*;
match expression {
Address(memory_reference) => memory_reference.to_quil_or_debug(),
FunctionCall(FunctionCallExpression {
function,
expression,
}) => format!("({function}({}))", parenthesized(expression)),
Infix(InfixExpression {
left,
operator,
right,
}) => format!(
"({}{}{})",
parenthesized(left),
operator,
parenthesized(right)
),
Number(value) => format!("({})", format_complex(value)),
PiConstant => "pi".to_string(),
Prefix(PrefixExpression {
operator,
expression,
}) => format!("({}{})", operator, parenthesized(expression)),
Variable(identifier) => format!("(%{identifier})"),
}
}
fn arb_name() -> impl Strategy<Value = String> {
r"[a-z][a-zA-Z0-9]{1,10}".prop_filter("Exclude reserved tokens", |t| {
ReservedToken::from_str(t).is_err() && !t.to_lowercase().starts_with("nan")
})
}
fn arb_memory_reference() -> impl Strategy<Value = MemoryReference> {
(arb_name(), (u64::MIN..u32::MAX as u64))
.prop_map(|(name, index)| MemoryReference { name, index })
}
fn arb_complex64() -> impl Strategy<Value = Complex64> {
let tau = std::f64::consts::TAU;
((-tau..tau), (-tau..tau)).prop_map(|(re, im)| Complex64 { re, im })
}
fn nonzero(strat: impl Strategy<Value = Expression>) -> impl Strategy<Value = Expression> {
strat.prop_filter("Exclude constantly-zero expressions", |expr| {
expr.clone().into_simplified() != Expression::Number(Complex64::new(0.0, 0.0))
})
}
fn arb_expr() -> impl Strategy<Value = Expression> {
use Expression::*;
let leaf = prop_oneof![
arb_memory_reference().prop_map(Address),
arb_complex64().prop_map(Number),
Just(PiConstant),
arb_name().prop_map(Variable),
];
leaf.prop_recursive(
4, 64, 16, |expr| {
let inner = expr.clone();
prop_oneof![
(any::<ExpressionFunction>(), expr.clone()).prop_map(|(function, e)| {
Expression::FunctionCall(FunctionCallExpression {
function,
expression: Box::new(e),
})
}),
(expr.clone(), any::<InfixOperator>())
.prop_flat_map(move |(left, operator)| (
Just(left),
Just(operator),
if let InfixOperator::Slash = operator {
nonzero(inner.clone()).boxed()
} else {
inner.clone().boxed()
}
))
.prop_map(|(l, operator, r)| Infix(InfixExpression {
left: Box::new(l),
operator,
right: Box::new(r)
})),
expr.prop_map(|e| Prefix(PrefixExpression {
operator: PrefixOperator::Minus,
expression: Box::new(e)
}))
]
},
)
}
proptest! {
#[test]
fn eq(a in any::<f64>(), b in any::<f64>()) {
let first = Expression::Infix (InfixExpression {
left: Box::new(Expression::Number(real!(a))),
operator: InfixOperator::Plus,
right: Box::new(Expression::Number(real!(b))),
} );
let differing = Expression::Number(real!(a + b));
prop_assert_eq!(&first, &first);
prop_assert_ne!(&first, &differing);
}
#[test]
fn hash(a in any::<f64>(), b in any::<f64>()) {
let first = Expression::Infix (InfixExpression {
left: Box::new(Expression::Number(real!(a))),
operator: InfixOperator::Plus,
right: Box::new(Expression::Number(real!(b))),
});
let matching = first.clone();
let differing = Expression::Number(real!(a + b));
let mut set = HashSet::new();
set.insert(first);
assert!(set.contains(&matching));
assert!(!set.contains(&differing))
}
#[test]
fn eq_iff_hash_eq(x in arb_expr(), y in arb_expr()) {
prop_assert_eq!(x == y, hash_to_u64(&x) == hash_to_u64(&y));
}
#[test]
fn reals_are_real(x in any::<f64>()) {
prop_assert_eq!(Expression::Number(real!(x)).to_real(), Ok(x))
}
#[test]
fn some_nums_are_real(re in any::<f64>(), im in any::<f64>()) {
let result = Expression::Number(Complex64{re, im}).to_real();
if is_small(im) {
prop_assert_eq!(result, Ok(re))
} else {
prop_assert_eq!(result, Err(EvaluationError::NumberNotReal))
}
}
#[test]
fn no_other_exps_are_real(expr in arb_expr().prop_filter("Not numbers", |e| !matches!(e, Expression::Number(_) | Expression::PiConstant))) {
prop_assert_eq!(expr.to_real(), Err(EvaluationError::NotANumber))
}
#[test]
fn complexes_are_parseable_as_expressions(value in arb_complex64()) {
let parsed = Expression::from_str(&format_complex(&value));
assert!(parsed.is_ok());
let simple = parsed.unwrap().into_simplified();
assert_eq!(Expression::Number(value), simple);
}
#[test]
fn exponentiation_works_as_expected(left in arb_expr(), right in arb_expr()) {
let expected = Expression::Infix (InfixExpression { left: Box::new(left.clone()), operator: InfixOperator::Caret, right: Box::new(right.clone()) } );
prop_assert_eq!(left ^ right, expected);
}
#[test]
fn in_place_exponentiation_works_as_expected(left in arb_expr(), right in arb_expr()) {
let expected = Expression::Infix (InfixExpression { left: Box::new(left.clone()), operator: InfixOperator::Caret, right: Box::new(right.clone()) } );
let mut x = left;
x ^= right;
prop_assert_eq!(x, expected);
}
#[test]
fn addition_works_as_expected(left in arb_expr(), right in arb_expr()) {
let expected = Expression::Infix (InfixExpression { left: Box::new(left.clone()), operator: InfixOperator::Plus, right: Box::new(right.clone()) } );
prop_assert_eq!(left + right, expected);
}
#[test]
fn in_place_addition_works_as_expected(left in arb_expr(), right in arb_expr()) {
let expected = Expression::Infix (InfixExpression { left: Box::new(left.clone()), operator: InfixOperator::Plus, right: Box::new(right.clone()) } );
let mut x = left;
x += right;
prop_assert_eq!(x, expected);
}
#[test]
fn subtraction_works_as_expected(left in arb_expr(), right in arb_expr()) {
let expected = Expression::Infix (InfixExpression { left: Box::new(left.clone()), operator: InfixOperator::Minus, right: Box::new(right.clone()) } );
prop_assert_eq!(left - right, expected);
}
#[test]
fn in_place_subtraction_works_as_expected(left in arb_expr(), right in arb_expr()) {
let expected = Expression::Infix (InfixExpression { left: Box::new(left.clone()), operator: InfixOperator::Minus, right: Box::new(right.clone()) } );
let mut x = left;
x -= right;
prop_assert_eq!(x, expected);
}
#[test]
fn multiplication_works_as_expected(left in arb_expr(), right in arb_expr()) {
let expected = Expression::Infix (InfixExpression { left: Box::new(left.clone()), operator: InfixOperator::Star, right: Box::new(right.clone()) } );
prop_assert_eq!(left * right, expected);
}
#[test]
fn in_place_multiplication_works_as_expected(left in arb_expr(), right in arb_expr()) {
let expected = Expression::Infix (InfixExpression { left: Box::new(left.clone()), operator: InfixOperator::Star, right: Box::new(right.clone()) } );
let mut x = left;
x *= right;
prop_assert_eq!(x, expected);
}
#[test]
fn division_works_as_expected(left in arb_expr(), right in nonzero(arb_expr())) {
let expected = Expression::Infix (InfixExpression { left: Box::new(left.clone()), operator: InfixOperator::Slash, right: Box::new(right.clone()) } );
prop_assert_eq!(left / right, expected);
}
#[test]
fn in_place_division_works_as_expected(left in arb_expr(), right in nonzero(arb_expr())) {
let expected = Expression::Infix (InfixExpression { left: Box::new(left.clone()), operator: InfixOperator::Slash, right: Box::new(right.clone()) } );
let mut x = left;
x /= right;
prop_assert_eq!(x, expected);
}
#[allow(clippy::redundant_clone)]
#[test]
fn round_trip(e in arb_expr()) {
let simple_e = e.clone().into_simplified();
let s = parenthesized(&e);
let p = Expression::from_str(&s);
prop_assert!(p.is_ok());
let p = p.unwrap();
let simple_p = p.clone().into_simplified();
prop_assert_eq!(
simple_p.clone(),
simple_e.clone(),
"Simplified expressions should be equal:\nparenthesized {p} ({p:?}) extracted from {s} simplified to {simple_p}\nvs original {e} ({e:?}) simplified to {simple_e}",
p=p.to_quil_or_debug(),
s=s,
e=e.to_quil_or_debug(),
simple_p=simple_p.to_quil_or_debug(),
simple_e=simple_e.to_quil_or_debug()
);
}
}
#[test]
fn specific_round_trip_tests() {
for input in &[
"-1*(phases[0]+phases[1])",
"(-1*(phases[0]+phases[1]))+(-1*(phases[0]+phases[1]))",
] {
let parsed = Expression::from_str(input);
let parsed = parsed.unwrap();
let restring = parsed.to_quil_or_debug();
assert_eq!(input, &restring);
}
}
#[test]
fn test_nan_is_equal() {
let left = Expression::Number(f64::NAN.into());
let right = left.clone();
assert_eq!(left, right);
}
#[test]
fn specific_simplification_tests() {
for (input, expected) in [
("pi", Expression::Number(PI.into())),
("pi/2", Expression::Number((PI / 2.0).into())),
("pi * pi", Expression::Number((PI.powi(2)).into())),
("1.0/(1.0-1.0)", Expression::Number(f64::NAN.into())),
(
"(a[0]*2*pi)/6.283185307179586",
Expression::Address(MemoryReference {
name: String::from("a"),
index: 0,
}),
),
] {
assert_eq!(
Expression::from_str(input).unwrap().into_simplified(),
expected
)
}
}
#[test]
fn specific_to_real_tests() {
for (input, expected) in [
(Expression::PiConstant, Ok(PI)),
(Expression::Number(Complex64 { re: 1.0, im: 0.0 }), Ok(1.0)),
(
Expression::Number(Complex64 { re: 1.0, im: 1.0 }),
Err(EvaluationError::NumberNotReal),
),
(
Expression::Variable("Not a number".into()),
Err(EvaluationError::NotANumber),
),
] {
assert_eq!(input.to_real(), expected)
}
}
#[test]
fn specific_format_complex_tests() {
for (x, s) in &[
(Complex64::new(0.0, 0.0), "0"),
(Complex64::new(-0.0, 0.0), "0"),
(Complex64::new(-0.0, -0.0), "0"),
(Complex64::new(0.0, 1.0), "1.0i"),
(Complex64::new(1.0, -1.0), "1-1.0i"),
(Complex64::new(1.234, 0.0), "1.234"),
(Complex64::new(0.0, 1.234), "1.234i"),
(Complex64::new(-1.234, 0.0), "-1.234"),
(Complex64::new(0.0, -1.234), "-1.234i"),
(Complex64::new(1.234, 5.678), "1.234+5.678i"),
(Complex64::new(-1.234, 5.678), "-1.234+5.678i"),
(Complex64::new(1.234, -5.678), "1.234-5.678i"),
(Complex64::new(-1.234, -5.678), "-1.234-5.678i"),
(Complex64::new(1e100, 2e-100), "1e100+2.0e-100i"),
] {
assert_eq!(format_complex(x), *s);
}
}
}