Replaced binary tuple with struct

This commit is contained in:
Martin Berg Alstad 2024-06-07 16:27:52 +02:00
parent 4fe388aca3
commit b16a4eefe4
7 changed files with 129 additions and 81 deletions

View File

@ -1,39 +1,27 @@
import "@typespec/openapi3"; import "@typespec/http";
using TypeSpec.OpenAPI; import "./models.tsp";
using TypeSpec.Http;
using Models;
enum BinaryOperator { @service({
AND, title: "Simplify Truth Expressions",
OR, })
IMPLICATION namespace Simplify {
} model SimplifyResponse {
model ExpressionNot {
not: Expression;
}
// TODO tuple type, possible in OpenAPI 3.1, but unsupported in typespec
model ExpressionBinary {
operator: BinaryOperator;
left: Expression;
right: Expression;
}
model ExpressionAtomic {
atomic: string;
}
@oneOf
union Expression {
ExpressionNot;
ExpressionBinary;
ExpressionAtomic;
}
model SimplifyResponse {
before: string; before: string;
after: string; after: string;
orderOfOperations: string[]; orderOfOperations?: string[] = [];
expression: Expression; expression: Expression;
} }
op simplify(): SimplifyResponse; model SimplifyOptions {
lang: "en" | "nb" = "en";
simplify: boolean = true;
caseSensitive: boolean = false;
}
op simplify(
@path exp: string,
@query query?: SimplifyOptions
): SimplifyResponse;
}

31
spec/models.tsp Normal file
View File

@ -0,0 +1,31 @@
import "@typespec/openapi3";
using TypeSpec.OpenAPI;
namespace Models;
enum BinaryOperator {
AND,
OR,
IMPLICATION
}
model ExpressionNot {
not: Expression;
}
model ExpressionBinary {
left: Expression;
operator: BinaryOperator;
right: Expression;
}
model ExpressionAtomic {
atomic: string;
}
@oneOf
union Expression {
ExpressionNot;
ExpressionBinary;
ExpressionAtomic;
}

View File

@ -1,13 +1,23 @@
openapi: 3.0.0 openapi: 3.0.0
info: info:
title: (title) title: Simplify Truth Expressions
version: 0.0.0 version: 0.0.0
tags: [] tags: []
paths: paths:
/: /{exp}:
get: get:
operationId: simplify operationId: simplify
parameters: [] parameters:
- name: exp
in: path
required: true
schema:
type: string
- name: query
in: query
required: false
schema:
$ref: '#/components/schemas/SimplifyOptions'
responses: responses:
'200': '200':
description: The request has succeeded. description: The request has succeeded.
@ -17,50 +27,68 @@ paths:
$ref: '#/components/schemas/SimplifyResponse' $ref: '#/components/schemas/SimplifyResponse'
components: components:
schemas: schemas:
BinaryOperator: Expression:
oneOf:
- $ref: '#/components/schemas/Models.ExpressionNot'
- $ref: '#/components/schemas/Models.ExpressionBinary'
- $ref: '#/components/schemas/Models.ExpressionAtomic'
Models.BinaryOperator:
type: string type: string
enum: enum:
- AND - AND
- OR - OR
- IMPLICATION - IMPLICATION
Expression: Models.ExpressionAtomic:
oneOf:
- $ref: '#/components/schemas/ExpressionNot'
- $ref: '#/components/schemas/ExpressionBinary'
- $ref: '#/components/schemas/ExpressionAtomic'
ExpressionAtomic:
type: object type: object
required: required:
- atomic - atomic
properties: properties:
atomic: atomic:
type: string type: string
ExpressionBinary: Models.ExpressionBinary:
type: object type: object
required: required:
- operator
- left - left
- operator
- right - right
properties: properties:
operator:
$ref: '#/components/schemas/BinaryOperator'
left: left:
$ref: '#/components/schemas/Expression' $ref: '#/components/schemas/Expression'
operator:
$ref: '#/components/schemas/Models.BinaryOperator'
right: right:
$ref: '#/components/schemas/Expression' $ref: '#/components/schemas/Expression'
ExpressionNot: Models.ExpressionNot:
type: object type: object
required: required:
- not - not
properties: properties:
not: not:
$ref: '#/components/schemas/Expression' $ref: '#/components/schemas/Expression'
SimplifyOptions:
type: object
required:
- lang
- simplify
- caseSensitive
properties:
lang:
type: string
enum:
- en
- nb
default: en
simplify:
type: boolean
default: true
caseSensitive:
type: boolean
default: false
SimplifyResponse: SimplifyResponse:
type: object type: object
required: required:
- before - before
- after - after
- orderOfOperations
- expression - expression
properties: properties:
before: before:
@ -71,5 +99,6 @@ components:
type: array type: array
items: items:
type: string type: string
default: []
expression: expression:
$ref: '#/components/schemas/Expression' $ref: '#/components/schemas/Expression'

View File

@ -12,7 +12,7 @@ pub trait OppositeEq {
#[serde(rename_all = "camelCase")] #[serde(rename_all = "camelCase")]
pub enum Expression { pub enum Expression {
Not(Box<Expression>), Not(Box<Expression>),
Binary(Box<Expression>, BinaryOperator, Box<Expression>), Binary { left: Box<Expression>, operator: BinaryOperator, right: Box<Expression> },
Atomic(String), Atomic(String),
} }
@ -20,7 +20,7 @@ impl Expression {
pub fn is_atomic(&self) -> bool { pub fn is_atomic(&self) -> bool {
match self { match self {
Expression::Not(expr) => expr.is_atomic(), Expression::Not(expr) => expr.is_atomic(),
Expression::Binary(_, _, _) => false, Expression::Binary { .. } => false,
Expression::Atomic(_) => true Expression::Atomic(_) => true
} }
} }
@ -32,7 +32,7 @@ impl Expression {
pub fn exists(&self, atomic_value: &str) -> bool { pub fn exists(&self, atomic_value: &str) -> bool {
match self { match self {
Expression::Not(expr) => expr.exists(atomic_value), Expression::Not(expr) => expr.exists(atomic_value),
Expression::Binary(left, _, right) => left.exists(atomic_value) || right.exists(atomic_value), Expression::Binary { left, right, .. } => left.exists(atomic_value) || right.exists(atomic_value),
Expression::Atomic(value) => value == atomic_value, Expression::Atomic(value) => value == atomic_value,
} }
} }
@ -69,14 +69,14 @@ impl Display for Expression {
match self { match self {
Expression::Not(expr) if expr.is_atomic() => write!(f, "¬{expr}"), Expression::Not(expr) if expr.is_atomic() => write!(f, "¬{expr}"),
Expression::Not(expr) => write!(f, "¬({expr})"), Expression::Not(expr) => write!(f, "¬({expr})"),
Expression::Binary(left, BinaryOperator::And, right) => { Expression::Binary { left, operator: BinaryOperator::And, right } => {
write!(f, "{left} ⋀ {right}") write!(f, "{left} ⋀ {right}")
} }
// TODO do not use parentheses on root level or if several operators are on the same level // TODO do not use parentheses on root level or if several operators are on the same level
Expression::Binary(left, BinaryOperator::Or, right) => { Expression::Binary { left, operator: BinaryOperator::Or, right } => {
write!(f, "({left} {right})") write!(f, "({left} {right})")
} }
Expression::Binary(left, BinaryOperator::Implication, right) => { Expression::Binary { left, operator: BinaryOperator::Implication, right } => {
write!(f, "{left} ➔ {right}") write!(f, "{left} ➔ {right}")
} }
Expression::Atomic(value) => write!(f, "{value}"), Expression::Atomic(value) => write!(f, "{value}"),

View File

@ -21,8 +21,8 @@ macro_rules! implies {
#[macro_export] #[macro_export]
macro_rules! binary { macro_rules! binary {
($left:expr, $op:expr, $right:expr) => { ($left:expr, $operator:expr, $right:expr) => {
$crate::expressions::expression::Expression::Binary(Box::new($left), $op, Box::new($right)) $crate::expressions::expression::Expression::Binary { left: Box::new($left), operator: $operator, right: Box::new($right) }
}; };
} }
@ -78,16 +78,16 @@ mod tests {
#[test] #[test]
fn eval_and() { fn eval_and() {
assert_eq!(eval!("a" && "b"), Binary(Box::new(Atomic("a".to_string())), And, Box::new(Atomic("b".to_string())))); assert_eq!(eval!("a" && "b"), Binary { left: Box::new(Atomic("a".to_string())), operator: And, right: Box::new(Atomic("b".to_string())) });
} }
#[test] #[test]
fn eval_or() { fn eval_or() {
assert_eq!(eval!("a" || "b"), Binary(Box::new(Atomic("a".to_string())), Or, Box::new(Atomic("b".to_string())))); assert_eq!(eval!("a" || "b"), Binary { left: Box::new(Atomic("a".to_string())), operator: Or, right: Box::new(Atomic("b".to_string())) });
} }
#[test] #[test]
fn eval_implies() { fn eval_implies() {
assert_eq!(eval!("a" => "b"), Binary(Box::new(Atomic("a".to_string())), Implication, Box::new(Atomic("b".to_string())))); assert_eq!(eval!("a" => "b"), Binary { left: Box::new(Atomic("a".to_string())), operator: Implication, right: Box::new(Atomic("b".to_string())) });
} }
} }

View File

@ -28,12 +28,12 @@ impl Simplify for Expression {
fn elimination_of_implication(&self) -> Self { fn elimination_of_implication(&self) -> Self {
match self { match self {
Expression::Not(expr) => not!(expr.elimination_of_implication()), Expression::Not(expr) => not!(expr.elimination_of_implication()),
Expression::Binary(left, BinaryOperator::Implication, right) => { Expression::Binary { left, operator: BinaryOperator::Implication, right } => {
let left = left.elimination_of_implication(); let left = left.elimination_of_implication();
let right = right.elimination_of_implication(); let right = right.elimination_of_implication();
or!(not!(left), right) or!(not!(left), right)
} }
Expression::Binary(left, operator, right) => { Expression::Binary { left, operator, right } => {
let left = left.elimination_of_implication(); let left = left.elimination_of_implication();
let right = right.elimination_of_implication(); let right = right.elimination_of_implication();
binary!(left, *operator, right) binary!(left, *operator, right)
@ -54,7 +54,7 @@ impl Simplify for Expression {
not!(expr.double_negation_elimination()) not!(expr.double_negation_elimination())
} }
} }
Expression::Binary(left, operator, right) => { Expression::Binary { left, operator, right } => {
let left = left.double_negation_elimination(); let left = left.double_negation_elimination();
let right = right.double_negation_elimination(); let right = right.double_negation_elimination();
binary!(left, *operator, right) binary!(left, *operator, right)
@ -67,13 +67,13 @@ impl Simplify for Expression {
match self { match self {
Expression::Not(expr) => { Expression::Not(expr) => {
match *expr.clone() { match *expr.clone() {
Expression::Binary(left, BinaryOperator::And, right) => { Expression::Binary { left, operator: BinaryOperator::And, right } => {
// TODO unnecessary cloning calls to de_morgans_laws? // TODO unnecessary cloning calls to de_morgans_laws?
let left = not!(left.de_morgans_laws()); let left = not!(left.de_morgans_laws());
let right = not!(right.de_morgans_laws()); let right = not!(right.de_morgans_laws());
or!(left, right).de_morgans_laws() or!(left, right).de_morgans_laws()
} }
Expression::Binary(left, BinaryOperator::Or, right) => { Expression::Binary { left, operator: BinaryOperator::Or, right } => {
let left = not!(left.de_morgans_laws()); let left = not!(left.de_morgans_laws());
let right = not!(right.de_morgans_laws()); let right = not!(right.de_morgans_laws());
and!(left, right).de_morgans_laws() and!(left, right).de_morgans_laws()
@ -81,7 +81,7 @@ impl Simplify for Expression {
_ => not!(expr.de_morgans_laws()), _ => not!(expr.de_morgans_laws()),
} }
} }
Expression::Binary(left, operator, right) => { Expression::Binary { left, operator, right } => {
let left = left.de_morgans_laws(); let left = left.de_morgans_laws();
let right = right.de_morgans_laws(); let right = right.de_morgans_laws();
binary!(left, *operator, right) binary!(left, *operator, right)
@ -93,10 +93,10 @@ impl Simplify for Expression {
// TODO deduplicate code // TODO deduplicate code
fn absorption_law(&self) -> Self { fn absorption_law(&self) -> Self {
match self { match self {
Expression::Binary(left, BinaryOperator::And, right) => { Expression::Binary { left, operator: BinaryOperator::And, right } => {
let (left_ref, right_ref) = (left.as_ref(), right.as_ref()); let (left_ref, right_ref) = (left.as_ref(), right.as_ref());
match (left_ref, right_ref) { match (left_ref, right_ref) {
(_, Expression::Binary(right_left, BinaryOperator::Or, right_right)) => { (_, Expression::Binary { left: right_left, operator: BinaryOperator::Or, right: right_right }) => {
if left_ref == right_left.as_ref() || left_ref == right_right.as_ref() { if left_ref == right_left.as_ref() || left_ref == right_right.as_ref() {
return left.absorption_law(); return left.absorption_law();
} else if right_left.is_atomic() && right_right.is_atomic() && left.opposite_eq(right_left) { } else if right_left.is_atomic() && right_right.is_atomic() && left.opposite_eq(right_left) {
@ -108,7 +108,7 @@ impl Simplify for Expression {
} }
and!(left.absorption_law(), right.absorption_law()) and!(left.absorption_law(), right.absorption_law())
} }
(Expression::Binary(left_left, BinaryOperator::Or, left_right), _) => { (Expression::Binary { left: left_left, operator: BinaryOperator::Or, right: left_right }, _) => {
if right_ref == left_left.as_ref() || right_ref == left_right.as_ref() { if right_ref == left_left.as_ref() || right_ref == left_right.as_ref() {
return right.absorption_law(); return right.absorption_law();
} else if left_left.is_atomic() && left_right.is_atomic() && right.opposite_eq(left_left) { } else if left_left.is_atomic() && left_right.is_atomic() && right.opposite_eq(left_left) {
@ -123,10 +123,10 @@ impl Simplify for Expression {
(left, right) => and!(left.absorption_law(), right.absorption_law()) (left, right) => and!(left.absorption_law(), right.absorption_law())
} }
} }
Expression::Binary(left, BinaryOperator::Or, right) => { Expression::Binary { left, operator: BinaryOperator::Or, right } => {
let (left_ref, right_ref) = (left.as_ref(), right.as_ref()); let (left_ref, right_ref) = (left.as_ref(), right.as_ref());
match (left_ref, right_ref) { match (left_ref, right_ref) {
(_, Expression::Binary(right_left, BinaryOperator::And, right_right)) => { (_, Expression::Binary { left: right_left, operator: BinaryOperator::And, right: right_right }) => {
if left_ref == right_left.as_ref() || left_ref == right_right.as_ref() { if left_ref == right_left.as_ref() || left_ref == right_right.as_ref() {
return left.absorption_law(); return left.absorption_law();
} else if right_left.is_atomic() && right_right.is_atomic() && left.opposite_eq(right_left) { } else if right_left.is_atomic() && right_right.is_atomic() && left.opposite_eq(right_left) {
@ -138,7 +138,7 @@ impl Simplify for Expression {
} }
or!(left.absorption_law(), right.absorption_law()) or!(left.absorption_law(), right.absorption_law())
} }
(Expression::Binary(left_left, BinaryOperator::And, left_right), _) => { (Expression::Binary { left: left_left, operator: BinaryOperator::And, right: left_right }, _) => {
if right_ref == left_left.as_ref() || right_ref == left_right.as_ref() { if right_ref == left_left.as_ref() || right_ref == left_right.as_ref() {
return right.absorption_law(); return right.absorption_law();
} else if left_left.is_atomic() && left_right.is_atomic() && right.opposite_eq(left_left) { } else if left_left.is_atomic() && left_right.is_atomic() && right.opposite_eq(left_left) {
@ -153,7 +153,7 @@ impl Simplify for Expression {
(left, right) => or!(left.absorption_law(), right.absorption_law()) (left, right) => or!(left.absorption_law(), right.absorption_law())
} }
} }
Expression::Binary(left, operator, right) => { Expression::Binary { left, operator, right } => {
let left = left.absorption_law(); let left = left.absorption_law();
let right = right.absorption_law(); let right = right.absorption_law();
binary!(left, *operator, right) binary!(left, *operator, right)
@ -170,14 +170,14 @@ impl Simplify for Expression {
// TODO deduplicate code // TODO deduplicate code
fn distribution_law(&self) -> Self { fn distribution_law(&self) -> Self {
match self { match self {
Expression::Binary(left, BinaryOperator::And, right) => { Expression::Binary { left, operator: BinaryOperator::And, right } => {
match (left.as_ref(), right.as_ref()) { match (left.as_ref(), right.as_ref()) {
(Expression::Atomic(_), Expression::Binary(right_left, BinaryOperator::Or, right_right)) => { (Expression::Atomic(_), Expression::Binary { left: right_left, operator: BinaryOperator::Or, right: right_right }) => {
let right_left = right_left.distribution_law(); let right_left = right_left.distribution_law();
let right_right = right_right.distribution_law(); let right_right = right_right.distribution_law();
or!(and!(*left.clone(), right_left), and!(*left.clone(), right_right)) or!(and!(*left.clone(), right_left), and!(*left.clone(), right_right))
} }
(Expression::Binary(left_left, BinaryOperator::Or, left_right), Expression::Atomic(_)) => { (Expression::Binary { left: left_left, operator: BinaryOperator::Or, right: left_right }, Expression::Atomic(_)) => {
let left_left = left_left.distribution_law(); let left_left = left_left.distribution_law();
let left_right = left_right.distribution_law(); let left_right = left_right.distribution_law();
or!(and!(left_left, *right.clone()), and!(left_right, *right.clone())) or!(and!(left_left, *right.clone()), and!(left_right, *right.clone()))
@ -185,14 +185,14 @@ impl Simplify for Expression {
(left, right) => and!(left.distribution_law(), right.distribution_law()) (left, right) => and!(left.distribution_law(), right.distribution_law())
} }
} }
Expression::Binary(left, BinaryOperator::Or, right) => { Expression::Binary { left, operator: BinaryOperator::Or, right } => {
match (left.as_ref(), right.as_ref()) { match (left.as_ref(), right.as_ref()) {
(Expression::Atomic(_), Expression::Binary(right_left, BinaryOperator::And, right_right)) => { (Expression::Atomic(_), Expression::Binary { left: right_left, operator: BinaryOperator::And, right: right_right }) => {
let right_left = right_left.distribution_law(); let right_left = right_left.distribution_law();
let right_right = right_right.distribution_law(); let right_right = right_right.distribution_law();
and!(or!(*left.clone(), right_left), or!(*left.clone(), right_right)) and!(or!(*left.clone(), right_left), or!(*left.clone(), right_right))
} }
(Expression::Binary(left_left, BinaryOperator::And, left_right), Expression::Atomic(_)) => { (Expression::Binary { left: left_left, operator: BinaryOperator::And, right: left_right }, Expression::Atomic(_)) => {
let left_left = left_left.distribution_law(); let left_left = left_left.distribution_law();
let left_right = left_right.distribution_law(); let left_right = left_right.distribution_law();
and!(or!(left_left, *right.clone()), or!(left_right, *right.clone())) and!(or!(left_left, *right.clone()), or!(left_right, *right.clone()))
@ -200,7 +200,7 @@ impl Simplify for Expression {
(left, right) => or!(left.distribution_law(), right.distribution_law()) (left, right) => or!(left.distribution_law(), right.distribution_law())
} }
} }
Expression::Binary(left, operator, right) => { Expression::Binary { left, operator, right } => {
let left = left.distribution_law(); let left = left.distribution_law();
let right = right.distribution_law(); let right = right.distribution_law();
binary!(left, *operator, right) binary!(left, *operator, right)

View File

@ -1,6 +1,6 @@
use std::net::SocketAddr; use std::net::SocketAddr;
use tokio::net::{TcpListener, ToSocketAddrs}; use tokio::net::TcpListener;
use crate::routing::{simplify, table}; use crate::routing::{simplify, table};