prune-lang 0.2.1

Prune is a constraint logic programming language with branching heuristic.
Documentation
use itertools::Itertools;

use super::ident::{Ident, IdentCtx};
use super::lit::{LitType, LitVal};

use std::collections::HashMap;
use std::convert::Infallible;
use std::fmt;

#[derive(Debug, Clone, PartialEq)]
pub enum Term<V, L, C> {
    Var(V),
    Lit(L),
    Cons(C, Vec<Term<V, L, C>>),
}

impl<V: fmt::Display, L: fmt::Display, C: fmt::Display> fmt::Display for Term<V, L, C> {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        match self {
            Term::Var(var) => fmt::Display::fmt(&var, f),
            Term::Lit(lit) => fmt::Display::fmt(&lit, f),
            Term::Cons(cons, flds) => {
                if flds.is_empty() && !format!("{}", cons).is_empty() {
                    fmt::Display::fmt(&cons, f)
                } else {
                    let flds = flds.iter().format(", ");
                    write!(f, "{cons}({flds})")
                }
            }
        }
    }
}

#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum OptCons<T> {
    Some(T), // constructors
    None,    // placeholder for tuples (without constructor)
}

impl<T: fmt::Display> fmt::Display for OptCons<T> {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        match self {
            OptCons::Some(cons) => fmt::Display::fmt(cons, f),
            OptCons::None => Ok(()), // tuples' placeholder won't be printed.
        }
    }
}

pub type TermVal<V = Ident> = Term<V, LitVal, OptCons<Ident>>;
pub type AtomVal<V = Ident> = Term<V, LitVal, Infallible>;
pub type TermType<V = Ident> = Term<V, LitType, OptCons<Ident>>;

impl<V, L, C> Term<V, L, C> {
    pub fn is_var(&self) -> bool {
        matches!(self, Term::Var(_))
    }

    pub fn is_lit(&self) -> bool {
        matches!(self, Term::Lit(_))
    }

    pub fn is_cons(&self) -> bool {
        matches!(self, Term::Cons(_, _))
    }

    pub fn height(&self) -> usize {
        match self {
            Term::Var(_) => 1,
            Term::Lit(_) => 1,
            Term::Cons(_cons, flds) => {
                let max_fld = flds.iter().map(|fld| fld.height()).max().unwrap_or(0);
                max_fld + 1
            }
        }
    }

    pub fn size(&self) -> usize {
        match self {
            Term::Var(_) => 1,
            Term::Lit(_) => 1,
            Term::Cons(_cons, flds) => {
                let sum_fld: usize = flds.iter().map(|fld| fld.size()).sum();
                sum_fld + 1
            }
        }
    }
}

impl<L: Copy, C: Copy> Term<Ident, L, C> {
    pub fn tag_ctx(&self, ctx: usize) -> Term<IdentCtx, L, C> {
        match self {
            Term::Var(var) => Term::Var(var.tag_ctx(ctx)),
            Term::Lit(lit) => Term::Lit(*lit),
            Term::Cons(cons, flds) => {
                let flds = flds.iter().map(|fld| fld.tag_ctx(ctx)).collect();
                Term::Cons(*cons, flds)
            }
        }
    }
}

impl<V: Copy, L: Copy, C: Copy> Term<V, L, C> {
    pub fn to_atom(&self) -> Option<Term<V, L, Infallible>> {
        match self {
            Term::Var(var) => Some(Term::Var(*var)),
            Term::Lit(lit) => Some(Term::Lit(*lit)),
            Term::Cons(_cons, _flds) => None,
        }
    }
}

impl<V: Copy, L: Copy> Term<V, L, Infallible> {
    pub fn to_term<C>(&self) -> Term<V, L, C> {
        match self {
            Term::Var(var) => Term::Var(*var),
            Term::Lit(lit) => Term::Lit(*lit),
            Term::Cons(_cons, _flds) => unreachable!(),
        }
    }
}

impl<V: Copy + Eq, L, C> Term<V, L, C> {
    pub fn occurs(&self, x: &V) -> bool {
        match self {
            Term::Var(y) => x == y,
            Term::Lit(_) => false,
            Term::Cons(_cons, flds) => flds.iter().any(|fld| fld.occurs(x)),
        }
    }

    pub fn free_vars(&self) -> Vec<V> {
        let mut vec = Vec::new();
        self.free_vars_help(&mut vec);
        vec
    }

    fn free_vars_help(&self, vec: &mut Vec<V>) {
        match self {
            Term::Var(var) => {
                if !vec.contains(var) {
                    vec.push(*var);
                }
            }
            Term::Lit(_lit) => {}
            Term::Cons(_cons, flds) => {
                flds.iter().for_each(|fld| fld.free_vars_help(vec));
            }
        }
    }
}

impl<V: Copy + Eq + std::hash::Hash, L: Copy, C: Copy> Term<V, L, C> {
    pub fn substitute(&self, map: &HashMap<V, Term<V, L, C>>) -> Term<V, L, C> {
        match self {
            Term::Var(var) => {
                if let Some(term) = map.get(var) {
                    term.clone()
                } else {
                    Term::Var(*var)
                }
            }
            Term::Lit(lit) => Term::Lit(*lit),
            Term::Cons(cons, flds) => {
                let flds = flds.iter().map(|fld| fld.substitute(map)).collect();
                Term::Cons(*cons, flds)
            }
        }
    }
}