From 5e69a1616e2fbaab0008256d9c4e5613a63b2612 Mon Sep 17 00:00:00 2001 From: arthur-adjedj <arthur.adjedj@gmail.com> Date: Sun, 13 Nov 2022 09:48:25 +0100 Subject: [PATCH] chore: add coverage --- kernel/src/environment.rs | 3 +- kernel/src/universe.rs | 154 ++++++++++++++++++++++++++------------ parser/src/parser.rs | 6 +- 3 files changed, 110 insertions(+), 53 deletions(-) diff --git a/kernel/src/environment.rs b/kernel/src/environment.rs index cd6e264e..dfb6573c 100644 --- a/kernel/src/environment.rs +++ b/kernel/src/environment.rs @@ -15,7 +15,7 @@ impl Environment { Self::default() } - pub fn insert(&mut self, s: String, decl : Declaration) -> Result<&Self, KernelError> { + pub fn insert(&mut self, s: String, decl: Declaration) -> Result<&Self, KernelError> { if let hash_map::Entry::Vacant(e) = self.0.entry(s.clone()) { e.insert(decl); Ok(self) @@ -33,7 +33,6 @@ impl Environment { pub fn insert_axiom(&mut self, s: String, ty: Term) -> Result<&Self, KernelError> { self.insert(s, Declaration::make(None, ty)) } - /// Returns the term linked to a definition in a given environment. pub fn get_term(&self, s: &String, vec: &[UniverseLevel]) -> Option<Term> { diff --git a/kernel/src/universe.rs b/kernel/src/universe.rs index 51f4345e..631a6d49 100644 --- a/kernel/src/universe.rs +++ b/kernel/src/universe.rs @@ -87,9 +87,9 @@ impl UniverseLevel { None => match self { Zero => unreachable!("Zero is a numeral"), Succ(_) => { - let (u, n) = self.plus(); - format!("{} + {}", u.pretty_print(), n) - }, + let (u, n) = self.plus(); + format!("{} + {}", u.pretty_print(), n) + } Max(box n, box m) => format!("max ({}) ({})", n.pretty_print(), m.pretty_print()), IMax(box n, box m) => format!("imax ({}) ({})", n.pretty_print(), m.pretty_print()), Var(n) => format!("u{}", n), @@ -107,17 +107,28 @@ impl UniverseLevel { } } - fn substitute_single(self, var : usize, u : UniverseLevel) -> UniverseLevel { + fn substitute_single(self, var: usize, u: UniverseLevel) -> UniverseLevel { match self { Zero => Zero, - Succ(n) => Succ(box n.substitute_single(var, u.clone())), - Max(n, m) => Max(box n.substitute_single(var, u.clone()), box m.substitute_single(var, u)), - IMax(n, m) => IMax(box n.substitute_single(var, u.clone()), box m.substitute_single(var, u)), - Var(n) => if n == var { u } else { Var(n) }, + Succ(n) => Succ(box n.substitute_single(var, u)), + Max(n, m) => Max( + box n.substitute_single(var, u.clone()), + box m.substitute_single(var, u), + ), + IMax(n, m) => IMax( + box n.substitute_single(var, u.clone()), + box m.substitute_single(var, u), + ), + Var(n) => { + if n == var { + u + } else { + Var(n) + } + } } } - pub fn substitute(self, univs: &[UniverseLevel]) -> Self { match self { Zero => Zero, @@ -134,27 +145,26 @@ impl UniverseLevel { /// imax(u, S(v)) = max(u, S(v)) /// imax(u, imax(v, w)) = max(imax(u, w), imax(v, w)) /// imax(u, max(v, w)) = max(imax(u, v), imax(u, w)) - /// - /// Here, the imax normalization pushes imaxes to all have a Var(i) as the second argument. To solve this last case, one needs to substitute + /// + /// Here, the imax normalization pushes imaxes to all have a Var(i) as the second argument. To solve this last case, one needs to substitute /// Var(i) with 0 and S(Var(i)). This gives us a consistent way to unstuck the geq-checking. fn normalize(&self) -> Self { match self { IMax(box Zero, box u) => u.clone(), - IMax(_ , box Zero) => Zero, + IMax(_, box Zero) => Zero, IMax(box u, box Succ(v)) => Max(box u.normalize(), box Succ(v.clone())).normalize(), - IMax(box u, box IMax(box v, box w)) => { - Max(box IMax(box u.clone(), box w.clone()).normalize(), - box IMax(box v.clone(), box w.clone()).normalize()) - } - IMax(box u, box Max(box v, box w)) => { - Max(box IMax(box u.clone(), box v.clone()).normalize(), - box IMax(box u.clone(), box w.clone()).normalize()) - } + IMax(box u, box IMax(box v, box w)) => Max( + box IMax(box u.clone(), box w.clone()).normalize(), + box IMax(box v.clone(), box w.clone()).normalize(), + ), + IMax(box u, box Max(box v, box w)) => Max( + box IMax(box u.clone(), box v.clone()).normalize(), + box IMax(box u.clone(), box w.clone()).normalize(), + ), _ => self.clone(), } } - // checks whether u1 <= u2 + n // In a case where comparison is stuck because of a variable Var(i), it checks whether the test is correct when Var(i) is substituted for // 0 and S(Var(i)). @@ -162,26 +172,40 @@ impl UniverseLevel { // - (true,0) if u1 <= u2 + n, // - (false,0) if !(u1 <= u2 + n), // - (false,i+1) if Var(i) needs to be substituted to unstuck the comparison. - fn geq_no_subst(&self, u2: &UniverseLevel, n: i64) -> (bool,usize) { + fn geq_no_subst(&self, u2: &UniverseLevel, n: i64) -> (bool, usize) { match (self.normalize(), u2.normalize()) { - (Zero, _) if n >= 0 => (true,0), - (_, _) if *self == *u2 && n >= 0 => (true,0), - (Succ(l), _) if l.geq_no_subst(u2, n - 1).0 => (true,0), - (_, Succ(box l)) if self.geq_no_subst(&l, n + 1).0 => (true,0), - (_, Max(box l1, box l2)) if self.geq_no_subst(&l1, n).0 || self.geq_no_subst(&l2, n).0 => (true,0), - (Max(box l1, box l2), _) if l1.geq_no_subst(u2, n).0 && l2.geq_no_subst(u2, n).0 => (true,0), - (_,IMax(_, box Var(i))) | (IMax(_, box Var(i)),_) => (false,i+1), - _ => (false,0), + (Zero, _) if n >= 0 => (true, 0), + (_, _) if *self == *u2 && n >= 0 => (true, 0), + (Succ(l), _) if l.geq_no_subst(u2, n - 1).0 => (true, 0), + (_, Succ(box l)) if self.geq_no_subst(&l, n + 1).0 => (true, 0), + (_, Max(box l1, box l2)) + if self.geq_no_subst(&l1, n).0 || self.geq_no_subst(&l2, n).0 => + { + (true, 0) + } + (Max(box l1, box l2), _) if l1.geq_no_subst(u2, n).0 && l2.geq_no_subst(u2, n).0 => { + (true, 0) + } + (_, IMax(_, box Var(i))) | (IMax(_, box Var(i)), _) => (false, i + 1), + _ => (false, 0), } } fn geq(&self, u2: &UniverseLevel, n: i64) -> bool { match self.geq_no_subst(u2, n) { - (true,_) => true, - (false,0) => false, - (false,i) => { - self.clone().substitute_single(i-1,Zero).geq(&u2.clone().substitute_single(i-1,Zero), n) && - self.clone().substitute_single(i-1,Succ(box Var(i-1))).geq(&u2.clone().substitute_single(i-1,Succ(box Var(i-1))), n) + (true, _) => true, + (false, 0) => false, + (false, i) => { + self.clone() + .substitute_single(i - 1, Zero) + .geq(&u2.clone().substitute_single(i - 1, Zero), n) + && self + .clone() + .substitute_single(i - 1, Succ(box Var(i - 1))) + .geq( + &u2.clone().substitute_single(i - 1, Succ(box Var(i - 1))), + n, + ) } } } @@ -195,26 +219,35 @@ impl UniverseLevel { mod tests { use crate::universe::UniverseLevel::*; - mod pretty_printing { use crate::universe::UniverseLevel::*; #[test] fn to_num() { - assert_eq!(Max(box Succ(box Zero), box Zero).to_numeral(),Some(1)); - assert_eq!(Max(box Succ(box Zero), box Succ(box Var(0))).to_numeral(),None); - assert_eq!(IMax(box Var(0),box Zero).to_numeral(),Some(0)); - assert_eq!(IMax(box Zero,box Succ(box Zero)).to_numeral(),Some(1)) + assert_eq!(Max(box Succ(box Zero), box Zero).to_numeral(), Some(1)); + assert_eq!( + Max(box Succ(box Zero), box Succ(box Var(0))).to_numeral(), + None + ); + assert_eq!(IMax(box Var(0), box Zero).to_numeral(), Some(0)); + assert_eq!(IMax(box Zero, box Succ(box Zero)).to_numeral(), Some(1)) } #[test] fn to_plus() { - assert_eq!(Succ(box Zero).plus(),(Zero,1)); + assert_eq!(Succ(box Zero).plus(), (Zero, 1)); } #[test] fn to_pretty_print() { - assert_eq!(Max(box Succ(box Zero), box IMax(box Max(box Zero, box Var(0)), box Succ(box Var(0)))).pretty_print(), "max (1) (imax (max (0) (u0)) (u0 + 1))"); + assert_eq!( + Max( + box Succ(box Zero), + box IMax(box Max(box Zero, box Var(0)), box Succ(box Var(0))) + ) + .pretty_print(), + "max (1) (imax (max (0) (u0)) (u0 + 1))" + ); } } @@ -229,26 +262,53 @@ mod tests { assert!(!&Max(box Var(1), box Var(1)).is_eq(&Max(box Var(0), box Var(1)))); assert!(&Succ(box Max(box Var(1), box Var(0))) .is_eq(&Max(box Succ(box Var(0)), box Succ(box Var(1))))); - assert!(&Max(box Zero, box IMax(box Zero, box Max(box Succ(box Zero), box Zero))).is_eq(&IMax(box Succ(box Zero), box IMax(box Succ(box Zero), box Succ(box Zero))))); + assert!(&Max( + box Zero, + box IMax(box Zero, box Max(box Succ(box Zero), box Zero)) + ) + .is_eq(&IMax( + box Succ(box Zero), + box IMax(box Succ(box Zero), box Succ(box Zero)) + ))); assert!(&Var(0).is_eq(&IMax(box Var(0), box Var(0)))); + assert!(&IMax(box Succ(box Zero), box Max(box Zero, box Zero)).is_eq(&Zero)); + assert!(!&IMax(box Var(0), box Var(1)).is_eq(&IMax(box Var(1), box Var(0)))) } - #[test] fn univ_vars_count() { assert_eq!( - IMax(box Zero,box Max(box Succ(box Zero), box Max(box Var(0), box Var(1)))).univ_vars(), + IMax( + box Zero, + box Max(box Succ(box Zero), box Max(box Var(0), box Var(1))) + ) + .univ_vars(), 2 ) } #[test] fn subst() { - let lvl = IMax(box Zero, box Max(box Succ(box Zero), box Max(box Var(0), box Var(1)))); + let lvl = IMax( + box Zero, + box Max(box Succ(box Zero), box Max(box Var(0), box Var(1))), + ); let subst = vec![Succ(box Zero), Zero]; assert_eq!( lvl.substitute(&subst), - IMax(box Zero, box Max(box Succ(box Zero), box Max(box Succ(box Zero), box Zero))) + IMax( + box Zero, + box Max(box Succ(box Zero), box Max(box Succ(box Zero), box Zero)) + ) + ) + } + + #[test] + fn single_subst() { + let lvl = IMax(box Max(box Succ(box Zero), box Var(0)), box Var(0)); + assert_eq!( + lvl.substitute_single(0, Zero), + IMax(box Max(box Succ(box Zero), box Zero), box Zero) ) } } diff --git a/parser/src/parser.rs b/parser/src/parser.rs index 39f825e8..90273ffc 100644 --- a/parser/src/parser.rs +++ b/parser/src/parser.rs @@ -58,7 +58,7 @@ fn convert_span(span: Span) -> Loc { fn build_univ_map_from_expr(pair: Pair<Rule>) -> HashMap<String, usize> { let iter = pair.into_inner(); let mut map = HashMap::new(); - for (i,pair) in iter.enumerate() { + for (i, pair) in iter.enumerate() { let str = pair.as_str(); if map.insert(str.to_string(), i).is_some() { panic!("Duplicate universe variable {}", str); @@ -209,9 +209,7 @@ fn build_command_from_expr(pair: Pair<Rule>) -> Command { next.clone().map(|x| x.as_rule()), None | Some(Rule::univ_decl) ) { - let univs = next - .map(build_univ_map_from_expr ) - .unwrap_or_default(); + let univs = next.map(build_univ_map_from_expr).unwrap_or_default(); build_term_from_expr(iter.next().unwrap(), &mut VecDeque::new(), &univs) } else { build_term_from_expr(next.unwrap(), &mut VecDeque::new(), &HashMap::new()) -- GitLab