ksl 0.1.30

KSL core library and interpreter
Documentation
//! # ksl::builtin::anl
//!
//! Built-in function `Any` and `All`.

use crate::{Environment, FALSE_SYMBOL, TRUE_SYMBOL, builtin::eval_builtin, eval::apply::eval_apply, value::Value};

pub(crate) fn builtin(args: &[Value], is_any: bool, env: Environment) -> Result<Value, std::sync::Arc<str>> {
    if let [list, predict] = args {
        match eval_apply(list, env.clone())? {
            Value::List(elements) => {
                if elements.is_empty() {
                    Ok(if is_any {
                        FALSE_SYMBOL.clone()
                    } else {
                        TRUE_SYMBOL.clone()
                    })
                } else {
                    let fpredict = std::sync::Arc::new(eval_apply(predict, env.clone())?);
                    for element in elements.iter() {
                        let pval = match fpredict.as_ref() {
                            Value::Builtin(name) => eval_builtin(name, std::slice::from_ref(element), env.clone()),
                            Value::Lambda(_, _, _) | Value::Plugin(_, _) => eval_apply(
                                &Value::Apply(vec![element.clone()], fpredict.clone()),
                                env.clone(),
                            ),
                            e => {
                                return Err(std::sync::Arc::from(format!(
                                    concat!(
                                        "Error[ksl::builtin::{}]: ",
                                        "Expected a function, but got: `{}`."
                                    ),
                                    if is_any { "Any" } else { "All" },
                                    e
                                )));
                            }
                        }?;

                        match pval {
                            p @ Value::Atom(_) => {
                                if p == *TRUE_SYMBOL {
                                    if is_any {
                                        return Ok(p);
                                    }
                                } else if p == *FALSE_SYMBOL {
                                    if !is_any {
                                        return Ok(p);
                                    }
                                } else {
                                    return Err(std::sync::Arc::from(format!(
                                        "Error[ksl::builtin::{}]: `{}` is not a boolean",
                                        if is_any { "Any" } else { "All" },
                                        p
                                    )));
                                }
                            }
                            e => {
                                return Err(std::sync::Arc::from(format!(
                                    concat!(
                                        "Error[ksl::builtin::{}]: ",
                                        "Expected a boolean, but got: `{}`."
                                    ),
                                    if is_any { "Any" } else { "All" },
                                    e
                                )));
                            }
                        }
                    }
                    Ok(if is_any {
                        FALSE_SYMBOL.clone()
                    } else {
                        TRUE_SYMBOL.clone()
                    })
                }
            }
            e => Err(std::sync::Arc::from(format!(
                concat!(
                    "Error[ksl::builtin::{}]: ",
                    "Expected a list, but got: `{}`."
                ),
                if is_any { "Any" } else { "All" },
                e
            ))),
        }
    } else {
        Err(std::sync::Arc::from(format!(
            concat!(
                "Error[ksl::builtin::{}]: ",
                "Expected 2 parameters, but {} were passed."
            ),
            if is_any { "Any" } else { "All" },
            args.len()
        )))
    }
}