mod lookup;
mod signatures;
mod types;
use std::sync::OnceLock;
pub use lookup::{
builtin_return_type, is_builtin, is_untyped_boundary_source, iter_builtin_metadata,
iter_builtin_names, lookup,
};
pub use types::{
BuiltinMetadata, BuiltinSignature, Param, ShapeFieldDescriptor, Ty, TY_ANY, TY_BOOL, TY_BYTES,
TY_BYTES_OR_NIL, TY_CLOSURE, TY_DICT, TY_DICT_OR_NIL, TY_DURATION, TY_FLOAT, TY_INT,
TY_INT_OR_NIL, TY_LIST, TY_NEVER, TY_NIL, TY_NUMBER, TY_STRING, TY_STRING_OR_NIL,
};
pub fn all_signatures() -> &'static [BuiltinSignature] {
static ALL_SIGNATURES: OnceLock<Vec<BuiltinSignature>> = OnceLock::new();
ALL_SIGNATURES
.get_or_init(|| {
let groups = signatures::groups();
let mut signatures =
Vec::with_capacity(groups.iter().map(|group| group.len()).sum::<usize>());
for group in groups {
signatures.extend_from_slice(group);
}
signatures.sort_by_key(|sig| sig.name);
signatures
})
.as_slice()
}
#[cfg(test)]
mod tests {
use super::*;
use crate::ast::TypeExpr;
#[test]
fn builtin_signatures_sorted() {
let mut prev = "";
for sig in all_signatures() {
assert!(
sig.name > prev,
"BUILTIN_SIGNATURES not sorted: `{prev}` must come before `{}`",
sig.name
);
prev = sig.name;
}
}
#[test]
fn lookup_hits_and_misses() {
assert!(is_builtin("snake_to_camel"));
assert!(is_builtin("log"));
assert!(is_builtin("await"));
assert!(!is_builtin("definitely_not_a_builtin"));
assert!(!is_builtin(""));
}
#[test]
fn return_type_named_variant() {
assert_eq!(
builtin_return_type("snake_to_camel"),
Some(TypeExpr::Named("string".into()))
);
assert_eq!(
builtin_return_type("log"),
Some(TypeExpr::Named("nil".into()))
);
assert_eq!(
builtin_return_type("file_exists"),
Some(TypeExpr::Named("bool".into()))
);
}
#[test]
fn return_type_union_variant() {
assert_eq!(
builtin_return_type("env"),
Some(TypeExpr::Union(vec![
TypeExpr::Named("string".into()),
TypeExpr::Named("nil".into()),
]))
);
}
#[test]
fn return_type_unknown_for_dynamic_builtins() {
assert!(is_builtin("json_parse"));
assert_eq!(builtin_return_type("json_parse"), None);
}
#[test]
fn return_type_none_for_unknown_names() {
assert_eq!(builtin_return_type("not_a_real_thing"), None);
}
}