jaq-interpret 1.5.0

Interpreter for the jaq language
Documentation
use alloc::{string::String, string::ToString, vec::Vec};
use core::fmt::{self, Display};

enum Exn<V> {
    Err(Error<V>),
    TailCall(crate::filter::TailCall<V>),
    Break,
    UpdateIndexError,
}

impl<V> Exn<V> {
    pub fn get_err(self) -> Result<Error<V>, Self> {
        match self {
            Self::Err(e) => Ok(e),
            _ => Err(self),
        }
    }
}

impl<V> From<Error<V>> for Exn<V> {
    fn from(e: Error<V>) -> Self {
        Exn::Err(e)
    }
}

pub struct Error<V>(Part<V, Vec<Part<V>>>);

pub enum Part<V, S = &'static str> {
    Val(V),
    Str(S),
}

impl<V> Error<V> {
    pub fn new(v: V) -> Self {
        Self(Part::Val(v))
    }
}

impl<V> FromIterator<Part<V>> for Error<V> {
    fn from_iter<T: IntoIterator<Item = Part<V>>>(iter: T) -> Self {
        Self(Part::Str(iter.into_iter().collect()))
    }
}

impl<V: From<String> + Display> Error<V> {
    /// Convert the error into a value to be used by `catch` filters.
    pub fn into_val(self) -> V {
        if let Part::Val(v) = self.0 {
            v
        } else {
            V::from(self.to_string())
        }
    }
}

impl<V: Display> Display for Error<V> {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        match &self.0 {
            Part::Val(v) => v.fmt(f),
            Part::Str(parts) => parts.iter().try_for_each(|part| match part {
                Part::Val(v) => v.fmt(f),
                Part::Str(s) => s.fmt(f),
            }),
        }
    }
}