diff options
Diffstat (limited to 'dhall_core/src')
-rw-r--r-- | dhall_core/src/core.rs | 66 |
1 files changed, 46 insertions, 20 deletions
diff --git a/dhall_core/src/core.rs b/dhall_core/src/core.rs index 1c41acc..d2ecf53 100644 --- a/dhall_core/src/core.rs +++ b/dhall_core/src/core.rs @@ -324,29 +324,18 @@ impl<S, A> Expr<S, A> { } } -impl<S: Clone, A: Clone> Expr<S, Expr<S, A>> { - pub fn squash_embed(&self) -> Expr<S, A> { - match self { - ExprF::Embed(e) => e.clone(), - e => e.map_shallow( - <Expr<S, Expr<S, A>>>::squash_embed, - S::clone, - |_| unreachable!(), - Label::clone, - ), - } - } -} - -impl<S: Clone, A: Clone> Expr<S, SubExpr<S, A>> { - pub fn squash_embed(&self) -> SubExpr<S, A> { +impl<N: Clone, E> Expr<N, E> { + pub fn squash_embed<E2: Clone>( + &self, + f: &impl Fn(&E) -> SubExpr<N, E2>, + ) -> SubExpr<N, E2> { match self.as_ref() { - ExprF::Embed(e) => e.clone(), + ExprF::Embed(e) => f(e), e => e .map( - |e| e.as_ref().squash_embed(), - |_, e| e.as_ref().squash_embed(), - S::clone, + |e| e.as_ref().squash_embed(f), + |_, e| e.as_ref().squash_embed(f), + N::clone, |_| unreachable!(), Label::clone, ) @@ -675,6 +664,43 @@ impl<N, E> SubExpr<N, E> { } } +impl<N: Clone> SubExpr<N, X> { + pub fn absurd<T>(&self) -> SubExpr<N, T> { + rc(self.as_ref().absurd_rec()) + } +} + +impl<N: Clone> Expr<N, X> { + // This is all very sad and I hope this can be avoided sometime + pub fn absurd_rec<T>(&self) -> Expr<N, T> { + self.map_ref( + |e| e.absurd(), + |_, e| e.absurd(), + |_| unreachable!(), + |_| unreachable!(), + Label::clone, + ) + } +} + +impl<SE, L, N, E> ExprF<SE, L, N, E> +where + SE: Clone, + L: Clone + Ord, + N: Clone, +{ + // When we know there is no Embed + pub fn absurd<T>(&self) -> ExprF<SE, L, N, T> { + self.map_ref( + |e| e.clone(), + |_, e| e.clone(), + N::clone, + |_| unreachable!(), + L::clone, + ) + } +} + impl<N, E> Clone for SubExpr<N, E> { fn clone(&self) -> Self { SubExpr(Rc::clone(&self.0)) |