1#![allow(clippy::box_default)]23pub mod interop;45pub mod import;6pub mod native;7pub mod val_extract;8pub mod val_make;9pub mod val_modify;10pub mod vars_tlas;1112use std::{13 alloc::Layout,14 any::Any,15 borrow::Cow,16 cell::RefCell,17 ffi::{CStr, CString, OsStr},18 os::raw::{c_char, c_double, c_int, c_uint},19 path::{Path, PathBuf},20};2122use jrsonnet_evaluator::{23 apply_tla, bail,24 function::TlaArg,25 gc::{GcHashMap, TraceBox},26 manifest::{JsonFormat, ManifestFormat, ToStringFormat},27 stack::set_stack_depth_limit,28 tb,29 trace::{CompactFormat, PathResolver, TraceFormat},30 FileImportResolver, IStr, ImportResolver, Result, State, Val,31};32use jrsonnet_gcmodule::Trace;33use jrsonnet_parser::SourcePath;34use jrsonnet_stdlib::ContextInitializer;353637#[cfg(target_arch = "wasm32")]38#[no_mangle]39pub extern "C" fn _start() {}404142434445#[no_mangle]46pub extern "C" fn jsonnet_version() -> &'static [u8; 8] {47 b"v0.20.0\0"48}4950unsafe fn parse_path(input: &CStr) -> Cow<Path> {51 #[cfg(target_family = "unix")]52 {53 use std::os::unix::ffi::OsStrExt;54 let str = OsStr::from_bytes(input.to_bytes());55 Cow::Borrowed(Path::new(str))56 }57 #[cfg(not(target_family = "unix"))]58 {59 let string = input.to_str().expect("bad utf-8");60 Cow::Borrowed(string.as_ref())61 }62}6364unsafe fn unparse_path(input: &Path) -> Cow<CStr> {65 #[cfg(target_family = "unix")]66 {67 use std::os::unix::ffi::OsStrExt;68 let str = CString::new(input.as_os_str().as_bytes()).expect("input has zero byte in it");69 Cow::Owned(str)70 }71 #[cfg(not(target_family = "unix"))]72 {73 let str = input.as_os_str().to_str().expect("bad utf-8");74 let cstr = CString::new(str).expect("input has NUL inside");75 Cow::Owned(cstr)76 }77}7879#[derive(Trace)]80struct VMImportResolver {81 #[trace(tracking(force))]82 inner: RefCell<TraceBox<dyn ImportResolver>>,83}84impl VMImportResolver {85 fn new(value: impl ImportResolver) -> Self {86 Self {87 inner: RefCell::new(tb!(value)),88 }89 }90}91impl ImportResolver for VMImportResolver {92 fn load_file_contents(&self, resolved: &SourcePath) -> Result<Vec<u8>> {93 self.inner.borrow().load_file_contents(resolved)94 }9596 fn resolve_from(&self, from: &SourcePath, path: &str) -> Result<SourcePath> {97 self.inner.borrow().resolve_from(from, path)98 }99100 fn resolve_from_default(&self, path: &str) -> Result<SourcePath> {101 self.inner.borrow().resolve_from_default(path)102 }103104 fn resolve(&self, path: &Path) -> Result<SourcePath> {105 self.inner.borrow().resolve(path)106 }107108 fn as_any(&self) -> &dyn Any {109 self110 }111 fn as_any_mut(&mut self) -> &mut dyn Any {112 self113 }114}115116pub struct VM {117 state: State,118 manifest_format: Box<dyn ManifestFormat>,119 trace_format: Box<dyn TraceFormat>,120 tla_args: GcHashMap<IStr, TlaArg>,121}122impl VM {123 fn replace_import_resolver(&self, resolver: impl ImportResolver) {124 *self125 .state126 .import_resolver()127 .as_any()128 .downcast_ref::<VMImportResolver>()129 .expect("valid resolver ty")130 .inner131 .borrow_mut() = tb!(resolver);132 }133 fn add_jpath(&self, path: PathBuf) {134 self.state135 .import_resolver()136 .as_any()137 .downcast_ref::<VMImportResolver>()138 .expect("valid resolver ty")139 .inner140 .borrow_mut()141 .as_any_mut()142 .downcast_mut::<FileImportResolver>()143 .expect("jpaths are not compatible with callback imports!")144 .add_jpath(path);145 }146}147148149#[no_mangle]150#[allow(clippy::box_default)]151pub extern "C" fn jsonnet_make() -> *mut VM {152 let mut state = State::builder();153 state154 .import_resolver(VMImportResolver::new(FileImportResolver::default()))155 .context_initializer(ContextInitializer::new(PathResolver::new_cwd_fallback()));156 let state = state.build();157 Box::into_raw(Box::new(VM {158 state,159 manifest_format: Box::new(JsonFormat::default()),160 trace_format: Box::new(CompactFormat::default()),161 tla_args: GcHashMap::new(),162 }))163}164165166#[no_mangle]167#[allow(clippy::boxed_local)]168pub extern "C" fn jsonnet_destroy(vm: Box<VM>) {169 drop(vm);170}171172173#[no_mangle]174pub extern "C" fn jsonnet_max_stack(_vm: &VM, v: c_uint) {175 set_stack_depth_limit(v as usize);176}177178179180181#[no_mangle]182pub extern "C" fn jsonnet_gc_min_objects(_vm: &VM, _v: c_uint) {}183184185186187#[no_mangle]188pub extern "C" fn jsonnet_gc_growth_trigger(_vm: &VM, _v: c_double) {}189190191#[no_mangle]192pub extern "C" fn jsonnet_string_output(vm: &mut VM, v: c_int) {193 vm.manifest_format = match v {194 0 => Box::new(JsonFormat::default()),195 1 => Box::new(ToStringFormat),196 _ => panic!("incorrect output format"),197 };198}199200201202203204205206207208#[no_mangle]209pub unsafe extern "C" fn jsonnet_realloc(_vm: &VM, buf: *mut u8, sz: usize) -> *mut u8 {210 if buf.is_null() {211 if sz == 0 {212 return std::ptr::null_mut();213 }214 return unsafe {215 std::alloc::alloc(Layout::from_size_align(sz, std::mem::align_of::<u8>()).unwrap())216 };217 }218 219 220 221 222 let old_layout = Layout::from_size_align(16, std::mem::align_of::<u8>()).unwrap();223 if sz == 0 {224 unsafe { std::alloc::dealloc(buf, old_layout) };225 return std::ptr::null_mut();226 }227 unsafe { std::alloc::realloc(buf, old_layout, sz) }228}229230231232233#[no_mangle]234#[allow(clippy::boxed_local)]235pub extern "C" fn jsonnet_json_destroy(_vm: &VM, v: Box<Val>) {236 drop(v);237}238239240#[no_mangle]241pub extern "C" fn jsonnet_max_trace(vm: &mut VM, v: c_uint) {242 if let Some(format) = vm.trace_format.as_any_mut().downcast_mut::<CompactFormat>() {243 format.max_trace = v as usize;244 } else {245 panic!("max_trace is not supported by current tracing format")246 }247}248249250251252253254255256#[no_mangle]257pub unsafe extern "C" fn jsonnet_evaluate_file(258 vm: &VM,259 filename: *const c_char,260 error: &mut c_int,261) -> *const c_char {262 let filename = unsafe { parse_path(CStr::from_ptr(filename)) };263 match vm264 .state265 .import(filename)266 .and_then(|val| apply_tla(vm.state.clone(), &vm.tla_args, val))267 .and_then(|val| val.manifest(&vm.manifest_format))268 {269 Ok(v) => {270 *error = 0;271 CString::new(&*v as &str).unwrap().into_raw()272 }273 Err(e) => {274 *error = 1;275 let mut out = String::new();276 vm.trace_format.write_trace(&mut out, &e).unwrap();277 CString::new(&out as &str).unwrap().into_raw()278 }279 }280}281282283284285286287288289#[no_mangle]290pub unsafe extern "C" fn jsonnet_evaluate_snippet(291 vm: &VM,292 filename: *const c_char,293 snippet: *const c_char,294 error: &mut c_int,295) -> *const c_char {296 let filename = unsafe { CStr::from_ptr(filename) };297 let snippet = unsafe { CStr::from_ptr(snippet) };298 match vm299 .state300 .evaluate_snippet(filename.to_str().unwrap(), snippet.to_str().unwrap())301 .and_then(|val| apply_tla(vm.state.clone(), &vm.tla_args, val))302 .and_then(|val| val.manifest(&vm.manifest_format))303 {304 Ok(v) => {305 *error = 0;306 CString::new(&*v as &str).unwrap().into_raw()307 }308 Err(e) => {309 *error = 1;310 let mut out = String::new();311 vm.trace_format.write_trace(&mut out, &e).unwrap();312 CString::new(&out as &str).unwrap().into_raw()313 }314 }315}316317fn val_to_multi(val: Val, format: &dyn ManifestFormat) -> Result<Vec<(IStr, IStr)>> {318 let Val::Obj(val) = val else {319 bail!("expected object as multi output")320 };321 let mut out = Vec::new();322 for (k, v) in val.iter(323 #[cfg(feature = "exp-preserve-order")]324 false,325 ) {326 out.push((k, v?.manifest(format)?.into()));327 }328 Ok(out)329}330331fn multi_to_raw(multi: Vec<(IStr, IStr)>) -> *const c_char {332 let mut out = Vec::new();333 for (i, (k, v)) in multi.iter().enumerate() {334 if i != 0 {335 out.push(0);336 }337 out.extend_from_slice(k.as_bytes());338 out.push(0);339 out.extend_from_slice(v.as_bytes());340 }341 out.push(0);342 out.push(0);343 let v = out.as_ptr();344 std::mem::forget(out);345 v.cast::<c_char>()346}347348349#[no_mangle]350pub unsafe extern "C" fn jsonnet_evaluate_file_multi(351 vm: &VM,352 filename: *const c_char,353 error: &mut c_int,354) -> *const c_char {355 let filename = unsafe { parse_path(CStr::from_ptr(filename)) };356 match vm357 .state358 .import(filename)359 .and_then(|val| apply_tla(vm.state.clone(), &vm.tla_args, val))360 .and_then(|val| val_to_multi(val, &vm.manifest_format))361 {362 Ok(v) => {363 *error = 0;364 multi_to_raw(v)365 }366 Err(e) => {367 *error = 1;368 let mut out = String::new();369 vm.trace_format.write_trace(&mut out, &e).unwrap();370 CString::new(&out as &str).unwrap().into_raw()371 }372 }373}374375376#[no_mangle]377pub unsafe extern "C" fn jsonnet_evaluate_snippet_multi(378 vm: &VM,379 filename: *const c_char,380 snippet: *const c_char,381 error: &mut c_int,382) -> *const c_char {383 let filename = unsafe { CStr::from_ptr(filename) };384 let snippet = unsafe { CStr::from_ptr(snippet) };385 match vm386 .state387 .evaluate_snippet(filename.to_str().unwrap(), snippet.to_str().unwrap())388 .and_then(|val| apply_tla(vm.state.clone(), &vm.tla_args, val))389 .and_then(|val| val_to_multi(val, &vm.manifest_format))390 {391 Ok(v) => {392 *error = 0;393 multi_to_raw(v)394 }395 Err(e) => {396 *error = 1;397 let mut out = String::new();398 vm.trace_format.write_trace(&mut out, &e).unwrap();399 CString::new(&out as &str).unwrap().into_raw()400 }401 }402}403404fn val_to_stream(val: Val, format: &dyn ManifestFormat) -> Result<Vec<IStr>> {405 let Val::Arr(val) = val else {406 bail!("expected array as stream output")407 };408 let mut out = Vec::new();409 for item in val.iter() {410 out.push(item?.manifest(format)?.into());411 }412 Ok(out)413}414415fn stream_to_raw(multi: Vec<IStr>) -> *const c_char {416 let mut out = Vec::new();417 for (i, v) in multi.iter().enumerate() {418 if i != 0 {419 out.push(0);420 }421 out.extend_from_slice(v.as_bytes());422 }423 out.push(0);424 out.push(0);425 let v = out.as_ptr();426 std::mem::forget(out);427 v.cast::<c_char>()428}429430431#[no_mangle]432pub unsafe extern "C" fn jsonnet_evaluate_file_stream(433 vm: &VM,434 filename: *const c_char,435 error: &mut c_int,436) -> *const c_char {437 let filename = unsafe { parse_path(CStr::from_ptr(filename)) };438 match vm439 .state440 .import(filename)441 .and_then(|val| apply_tla(vm.state.clone(), &vm.tla_args, val))442 .and_then(|val| val_to_stream(val, &vm.manifest_format))443 {444 Ok(v) => {445 *error = 0;446 stream_to_raw(v)447 }448 Err(e) => {449 *error = 1;450 let mut out = String::new();451 vm.trace_format.write_trace(&mut out, &e).unwrap();452 CString::new(&out as &str).unwrap().into_raw()453 }454 }455}456457458#[no_mangle]459pub unsafe extern "C" fn jsonnet_evaluate_snippet_stream(460 vm: &VM,461 filename: *const c_char,462 snippet: *const c_char,463 error: &mut c_int,464) -> *const c_char {465 let filename = unsafe { CStr::from_ptr(filename) };466 let snippet = unsafe { CStr::from_ptr(snippet) };467 match vm468 .state469 .evaluate_snippet(filename.to_str().unwrap(), snippet.to_str().unwrap())470 .and_then(|val| apply_tla(vm.state.clone(), &vm.tla_args, val))471 .and_then(|val| val_to_stream(val, &vm.manifest_format))472 {473 Ok(v) => {474 *error = 0;475 stream_to_raw(v)476 }477 Err(e) => {478 *error = 1;479 let mut out = String::new();480 vm.trace_format.write_trace(&mut out, &e).unwrap();481 CString::new(&out as &str).unwrap().into_raw()482 }483 }484}