Fixed truth table generation

This commit is contained in:
Martin Berg Alstad 2024-06-13 11:34:57 +02:00
parent 5c8f602d1d
commit 3ad1ad53fc
7 changed files with 197 additions and 45 deletions

View File

@ -27,3 +27,11 @@ GET {{url}}/simplify/!A
expression("A => B") expression("A => B")
%} %}
GET {{url}}/simplify/{{expression}} GET {{url}}/simplify/{{expression}}
### GET with table
< {%
import {expression} from "./common";
expression("A & B | C")
%}
GET {{url}}/simplify/table/{{expression}}

View File

@ -1,3 +1,4 @@
use std::collections::HashSet;
use std::fmt::Display; use std::fmt::Display;
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
use crate::expressions::iterator::ExpressionIterator; use crate::expressions::iterator::ExpressionIterator;
@ -9,7 +10,7 @@ pub trait OppositeEq {
fn opposite_eq(&self, other: &Self) -> bool; fn opposite_eq(&self, other: &Self) -> bool;
} }
#[derive(Debug, Clone, PartialEq, Deserialize, Serialize)] #[derive(Debug, Clone, PartialEq, Eq, Hash, Deserialize, Serialize)]
#[serde(rename_all = "camelCase")] #[serde(rename_all = "camelCase")]
pub enum Expression { pub enum Expression {
Not(Box<Expression>), Not(Box<Expression>),
@ -34,6 +35,18 @@ impl Expression {
} }
} }
pub fn get_atomic_values(&self) -> HashSet<String> {
match self {
Expression::Not(expr) => expr.get_atomic_values(),
Expression::Binary { left, right, .. } => {
let mut values = left.get_atomic_values();
values.extend(right.get_atomic_values());
values
}
Expression::Atomic(value) => HashSet::from([value.clone()])
}
}
pub fn count_distinct(&self) -> usize { pub fn count_distinct(&self) -> usize {
self.count_distinct_with_visited(vec![].as_mut()) self.count_distinct_with_visited(vec![].as_mut())
} }

View File

@ -3,5 +3,5 @@ pub mod operator;
#[macro_use] #[macro_use]
pub mod helpers; pub mod helpers;
pub mod simplify; pub mod simplify;
mod truth_table; pub mod truth_table;
mod iterator; mod iterator;

View File

@ -1,9 +1,19 @@
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
#[derive(Debug, Copy, Clone, PartialEq, Deserialize, Serialize)] #[derive(Debug, Copy, Clone, PartialEq, Eq, Hash, Deserialize, Serialize)]
#[serde(rename_all = "SCREAMING_SNAKE_CASE")] #[serde(rename_all = "SCREAMING_SNAKE_CASE")]
pub enum BinaryOperator { pub enum BinaryOperator {
Implication, Implication,
Or, Or,
And, And,
} }
impl BinaryOperator {
pub fn eval(&self, left: bool, right: bool) -> bool {
match self {
BinaryOperator::And => left && right,
BinaryOperator::Or => left || right,
BinaryOperator::Implication => !left || right,
}
}
}

View File

@ -1,9 +1,9 @@
use std::slice::Iter; use std::collections::HashMap;
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
use crate::expressions::expression::Expression; use crate::expressions::expression::Expression;
use crate::expressions::operator::BinaryOperator; use crate::map;
use crate::utils::array::Distinct; use crate::utils::array::Distinct;
type TruthMatrix = Vec<Vec<bool>>; type TruthMatrix = Vec<Vec<bool>>;
@ -40,9 +40,10 @@ pub struct TruthTableOptions {
} }
impl TruthTable { impl TruthTable {
// TODO options
pub fn new(expression: &Expression, options: TruthTableOptions) -> Self { pub fn new(expression: &Expression, options: TruthTableOptions) -> Self {
let header = Self::extract_header(expression); let header = Self::extract_header(expression);
let truth_matrix = Self::generate_truth_matrix(expression); let truth_matrix = Self::generate_truth_matrix(expression, &header);
Self { header, truth_matrix } Self { header, truth_matrix }
} }
@ -78,15 +79,20 @@ impl TruthTable {
} }
} }
fn generate_truth_matrix(expression: &Expression) -> TruthMatrix { fn generate_truth_matrix(expression: &Expression, header: &[String]) -> TruthMatrix {
let count = expression.count_distinct(); let mut atomics = expression.get_atomic_values()
if count == 0 { .into_iter().collect::<Vec<String>>();
if atomics.is_empty() {
return vec![]; return vec![];
} }
Self::truth_combinations(count) atomics.sort();
.iter().map(|combo| { Self::truth_combinations(atomics.len()).iter()
Self::resolve_expression(expression, &mut combo.iter()) .map(|combo| {
}).collect() Self::resolve_expression(expression, &atomics.iter()
.enumerate()
.map(|(index, value)| (value.clone(), combo[index]))
.collect(), header)
}).collect()
} }
fn truth_combinations(count: usize) -> TruthMatrix { fn truth_combinations(count: usize) -> TruthMatrix {
@ -97,31 +103,41 @@ impl TruthTable {
).collect() ).collect()
} }
fn resolve_expression(expression: &Expression, booleans: &mut Iter<bool>) -> Vec<bool> { fn resolve_expression(expression: &Expression, booleans: &HashMap<String, bool>, header: &[String]) -> Vec<bool> {
let expression_map = Self::_resolve_expression(expression, booleans);
let string_map = expression_map.iter()
.map(|(key, value)| (key.to_string(), *value))
.collect::<HashMap<String, bool>>();
header.iter()
.map(|s_expr| string_map.get(s_expr).copied().expect("Expression not found in map"))
.collect()
}
fn _resolve_expression<'a>(expression: &'a Expression, booleans: &HashMap<String, bool>) -> HashMap<&'a Expression, bool> {
match expression { match expression {
Expression::Not(expr) => { not @ Expression::Not(expr) => {
Self::resolve_expression(expr, booleans) let mut map = Self::_resolve_expression(expr, booleans);
.iter().map(|value| !value).collect() if let Some(value) = map.get(expr.as_ref()) {
map.insert(not, !value);
}
map
} }
Expression::Binary { left, right, .. } => { binary @ Expression::Binary { left, right, operator } => {
let left_values = Self::resolve_expression(left, booleans); let left_map = Self::_resolve_expression(left, booleans);
let right_values = Self::resolve_expression(right, booleans); let right_map = Self::_resolve_expression(right, booleans);
left_values.iter() let mut map = left_map;
.zip(right_values.iter()) map.extend(right_map);
.flat_map(|(left_value, right_value)| { if let (Some(left_value), Some(right_value)) = (map.get(left.as_ref()), map.get(right.as_ref())) {
[*left_value, *right_value, match expression { map.insert(binary, operator.eval(*left_value, *right_value));
Expression::Binary { operator: BinaryOperator::And, .. } => *left_value && *right_value, }
Expression::Binary { operator: BinaryOperator::Or, .. } => *left_value || *right_value, map
Expression::Binary { operator: BinaryOperator::Implication, .. } => !*left_value || *right_value,
_ => false,
}]
}).collect()
} }
Expression::Atomic(_) => { atomic @ Expression::Atomic(value) => {
if let Some(value) = booleans.next() { if let Some(value) = booleans.get(value) {
vec![*value] map!(atomic => *value)
} else { } else {
vec![] unreachable!("Atomic value not found in booleans")
} }
} }
} }
@ -131,13 +147,21 @@ impl TruthTable {
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use crate::matrix; use crate::matrix;
use super::*; use super::*;
// TODO fails sometimes...
#[test] #[test]
fn test_new_truth_table() { fn test_new_truth_table() {
let expression = and!(atomic!("A"), atomic!("B")); let expression = and!(atomic!("A"), atomic!("B"));
let truth_table = TruthTable::new(&expression, Default::default()); let truth_table = TruthTable::new(&expression, Default::default());
assert_eq!(truth_table.header, vec!["A", "B", "A ⋀ B"]); assert_eq!(truth_table.header, vec!["A", "B", "A ⋀ B"]);
assert_ne!(truth_table.truth_matrix, matrix![
true, true, true;
false, true, false;
true, false, false;
false, false, false
]);
assert_eq!(truth_table.truth_matrix, matrix![ assert_eq!(truth_table.truth_matrix, matrix![
true, true, true; true, true, true;
true, false, false; true, false, false;
@ -147,7 +171,37 @@ mod tests {
} }
#[test] #[test]
fn test_truth_combinations() { fn test_new_truth_table_a_and_b_or_c() {
let expression = and!(or!(atomic!("A"), atomic!("C")), or!(atomic!("B"), atomic!("C")));
let truth_table = TruthTable::new(&expression, Default::default());
let atomics = 3;
assert_eq!(truth_table.header, vec!["A", "C", "(A C)", "B", "(B C)", "(A C) ⋀ (B C)"]);
assert_eq!(truth_table.truth_matrix.len(), 2usize.pow(atomics as u32));
assert_eq!(truth_table.truth_matrix[0].len(), 6);
assert_eq!(truth_table.truth_matrix[0], vec![true, true, true, true, true, true]);
assert_eq!(truth_table.truth_matrix[1], vec![true, false, true, true, true, true]);
assert_eq!(truth_table.truth_matrix[2], vec![true, true, true, false, true, true]);
assert_eq!(truth_table.truth_matrix[3], vec![true, false, true, false, false, false]);
assert_eq!(truth_table.truth_matrix[4], vec![false, true, true, true, true, true]);
assert_eq!(truth_table.truth_matrix[5], vec![false, false, false, true, true, false]);
assert_eq!(truth_table.truth_matrix[6], vec![false, true, true, false, true, true]);
assert_eq!(truth_table.truth_matrix[7], vec![false, false, false, false, false, false]);
}
#[test]
fn test_truth_combinations_2() {
let combinations = TruthTable::truth_combinations(2);
assert_eq!(combinations, matrix![
true, true;
true, false;
false, true;
false, false
]);
}
#[test]
fn test_truth_combinations_3() {
let combinations = TruthTable::truth_combinations(3); let combinations = TruthTable::truth_combinations(3);
assert_eq!(combinations, matrix![ assert_eq!(combinations, matrix![
true, true, true; true, true, true;
@ -164,27 +218,62 @@ mod tests {
#[test] #[test]
fn test_resolve_expression_and_all_true() { fn test_resolve_expression_and_all_true() {
let expression = and!(atomic!("A"), atomic!("B")); let expression = and!(atomic!("A"), atomic!("B"));
let booleans = [true, true]; let booleans = map!["A".into() => true, "B".into() => true];
let values = TruthTable::resolve_expression(&expression, &mut booleans.iter()); let header = vec!["A".into(), "B".into(), "A ⋀ B".into()];
let values = TruthTable::resolve_expression(&expression, &booleans, &header);
assert_eq!(values, vec![true, true, true]); assert_eq!(values, vec![true, true, true]);
} }
#[test] #[test]
fn test_resolve_expression_and_1_true_1_false() { fn test_resolve_expression_and_1_true_1_false() {
let expression = and!(atomic!("A"), atomic!("B")); let expression = and!(atomic!("A"), atomic!("B"));
let booleans = [true, false]; let booleans = map!["A".into() => true, "B".into() => false];
let values = TruthTable::resolve_expression(&expression, &mut booleans.iter()); let header = vec!["A".into(), "B".into(), "A ⋀ B".into()];
let values = TruthTable::resolve_expression(&expression, &booleans, &header);
assert_eq!(values, vec![true, false, false]); assert_eq!(values, vec![true, false, false]);
} }
#[test] #[test]
fn test_resolve_expression_or_1_true_1_false() { fn test_resolve_expression_or_1_true_1_false() {
let expression = or!(atomic!("A"), atomic!("B")); let expression = or!(atomic!("A"), atomic!("B"));
let booleans = [true, false]; let booleans = map!["A".into() => true, "B".into() => false];
let values = TruthTable::resolve_expression(&expression, &mut booleans.iter()); let header = vec!["A".into(), "B".into(), "(A B)".into()];
let values = TruthTable::resolve_expression(&expression, &booleans, &header);
assert_eq!(values, vec![true, false, true]); assert_eq!(values, vec![true, false, true]);
} }
#[test]
fn test_resolve_expression_duplicate_atomic() {
let expression = and!(atomic!("A"), atomic!("A"));
let booleans = map!["A".into() => true];
let header = vec!["A".into(), "A ⋀ A".into()];
let values = TruthTable::resolve_expression(&expression, &booleans, &header);
assert_eq!(values, vec![true, true]);
}
#[test]
fn test_resolve_expression_even_more_duplicates() {
let expression = and!(atomic!("A"), and!(atomic!("A"), and!(atomic!("A"), atomic!("A"))));
let booleans = HashMap::from([("A".into(), true)]);
let header = vec!["A".into(), "A ⋀ A".into(), "A ⋀ A ⋀ A".into(), "A ⋀ A ⋀ A ⋀ A".into()];
let values = TruthTable::resolve_expression(&expression, &booleans, &header);
assert_eq!(values, vec![true, true, true, true]);
}
#[test]
fn _test_resolve_expression_even_more_duplicates() {
let expression = and!(atomic!("A"), and!(atomic!("A"), and!(atomic!("A"), atomic!("A"))));
let booleans = HashMap::from([("A".into(), true)]);
let values = TruthTable::_resolve_expression(&expression, &booleans);
assert_eq!(values, HashMap::from([
(&atomic!("A"), true),
(&and!(atomic!("A"), atomic!("A")), true),
(&and!(atomic!("A"), and!(atomic!("A"), atomic!("A"))), true),
(&and!(atomic!("A"), and!(atomic!("A"), and!(atomic!("A"), atomic!("A")))), true),
]));
}
#[test] #[test]
fn test_atomic_expression() { fn test_atomic_expression() {
let expression = atomic!("A"); let expression = atomic!("A");

View File

@ -6,6 +6,7 @@ use serde::{Deserialize, Serialize};
use crate::expressions::expression::Expression; use crate::expressions::expression::Expression;
use crate::expressions::simplify::Simplify; use crate::expressions::simplify::Simplify;
use crate::expressions::truth_table::{TruthTable, TruthTableOptions};
use crate::language::{AcceptLanguage, Language}; use crate::language::{AcceptLanguage, Language};
pub fn router() -> Router<()> { pub fn router() -> Router<()> {
@ -38,6 +39,8 @@ struct SimplifyResponse {
after: String, after: String,
order_of_operations: Vec<String>, order_of_operations: Vec<String>,
expression: Expression, expression: Expression,
#[serde(skip_serializing_if = "Option::is_none")]
truth_table: Option<TruthTable>,
} }
// TODO // TODO
@ -52,12 +55,29 @@ async fn simplify(Path(path): Path<String>, query: Query<QueryOptions>, accept_l
after: expression.to_string(), after: expression.to_string(),
order_of_operations: vec![], // TODO order_of_operations: vec![], // TODO
expression, expression,
truth_table: None,
}).into_response() }).into_response()
} else { } else {
(StatusCode::BAD_REQUEST, "Invalid expression").into_response() (StatusCode::BAD_REQUEST, "Invalid expression").into_response()
} }
} }
async fn simplify_and_table() { async fn simplify_and_table(Path(path): Path<String>, query: Query<QueryOptions>, accept_language: Option<AcceptLanguage>) -> Response {
unimplemented!("Not yet implemented") if let Ok(mut expression) = Expression::try_from(path.as_str()) {
let before = expression.to_string();
if query.simplify {
expression = expression.simplify();
}
// TODO options
let truth_table = TruthTable::new(&expression, TruthTableOptions::default());
Json(SimplifyResponse {
before,
after: expression.to_string(),
order_of_operations: vec![], // TODO
expression,
truth_table: Some(truth_table),
}).into_response()
} else {
(StatusCode::BAD_REQUEST, "Invalid expression").into_response()
}
} }

View File

@ -1,5 +1,3 @@
use std::cmp::max;
#[macro_export] #[macro_export]
macro_rules! set { macro_rules! set {
() => { std::collections::HashSet::new() }; () => { std::collections::HashSet::new() };
@ -31,6 +29,20 @@ macro_rules! matrix {
}; };
} }
#[macro_export]
macro_rules! map {
() => { std::collections::HashMap::new() };
($($k:expr => $v:expr),*) => {
{
let mut temp_map = std::collections::HashMap::new();
$(
temp_map.insert($k, $v);
)*
temp_map
}
};
}
pub trait Distinct { pub trait Distinct {
fn distinct(&mut self); fn distinct(&mut self);