1use std::{any::Any, borrow::Cow};23use jrsonnet_gcmodule::Trace;45use super::{arglike::ArgsLike, parse::parse_builtin_call, CallLocation};6use crate::{error::Result, gc::TraceBox, tb, Context, Val};78pub type BuiltinParamName = Cow<'static, str>;910#[derive(Clone, Trace)]11pub struct BuiltinParam {12 13 pub name: Option<BuiltinParamName>,14 15 pub has_default: bool,16}1718192021pub trait Builtin: Trace {22 23 fn name(&self) -> &str;24 25 fn params(&self) -> &[BuiltinParam];26 27 fn call(&self, ctx: Context, loc: CallLocation<'_>, args: &dyn ArgsLike) -> Result<Val>;2829 fn as_any(&self) -> &dyn Any;30}3132pub trait StaticBuiltin: Builtin + Send + Sync33where34 Self: 'static,35{36 37 38}3940#[derive(Trace)]41pub struct NativeCallback {42 pub(crate) params: Vec<BuiltinParam>,43 handler: TraceBox<dyn NativeCallbackHandler>,44}45impl NativeCallback {46 #[deprecated = "prefer using builtins directly, use this interface only for bindings"]47 pub fn new(params: Vec<Cow<'static, str>>, handler: impl NativeCallbackHandler) -> Self {48 Self {49 params: params50 .into_iter()51 .map(|n| BuiltinParam {52 name: Some(n),53 has_default: false,54 })55 .collect(),56 handler: tb!(handler),57 }58 }59}6061impl Builtin for NativeCallback {62 fn name(&self) -> &str {63 64 65 "<native>"66 }6768 fn params(&self) -> &[BuiltinParam] {69 &self.params70 }7172 fn call(&self, ctx: Context, _loc: CallLocation<'_>, args: &dyn ArgsLike) -> Result<Val> {73 let args = parse_builtin_call(ctx, &self.params, args, true)?;74 let args = args75 .into_iter()76 .map(|a| a.expect("legacy natives have no default params"))77 .map(|a| a.evaluate())78 .collect::<Result<Vec<Val>>>()?;79 self.handler.call(&args)80 }8182 fn as_any(&self) -> &dyn Any {83 self84 }85}8687pub trait NativeCallbackHandler: Trace {88 fn call(&self, args: &[Val]) -> Result<Val>;89}