1#![allow(clippy::box_default)]23#[cfg(feature = "interop")]4pub mod interop;56pub mod import;7pub mod native;8pub mod val_extract;9pub mod val_make;10pub mod val_modify;11pub mod vars_tlas;1213use std::{14 alloc::Layout,15 borrow::Cow,16 ffi::{CStr, CString, OsStr},17 os::raw::{c_char, c_double, c_int, c_uint},18 path::Path,19};2021use jrsonnet_evaluator::{22 apply_tla, bail,23 function::TlaArg,24 gc::GcHashMap,25 manifest::{JsonFormat, ManifestFormat, ToStringFormat},26 stack::set_stack_depth_limit,27 tb,28 trace::{CompactFormat, PathResolver, TraceFormat},29 FileImportResolver, IStr, Result, State, Val,30};313233#[cfg(target_arch = "wasm32")]34#[no_mangle]35pub extern "C" fn _start() {}363738394041#[no_mangle]42pub extern "C" fn jsonnet_version() -> &'static [u8; 8] {43 b"v0.19.1\0"44}4546unsafe fn parse_path(input: &CStr) -> Cow<Path> {47 #[cfg(target_family = "unix")]48 {49 use std::os::unix::ffi::OsStrExt;50 let str = OsStr::from_bytes(input.to_bytes());51 Cow::Borrowed(Path::new(str))52 }53 #[cfg(not(target_family = "unix"))]54 {55 let string = input.to_str().expect("bad utf-8");56 Cow::Borrowed(string.as_ref())57 }58}5960unsafe fn unparse_path(input: &Path) -> Cow<CStr> {61 #[cfg(target_family = "unix")]62 {63 use std::os::unix::ffi::OsStrExt;64 let str = CString::new(input.as_os_str().as_bytes()).expect("input has zero byte in it");65 Cow::Owned(str)66 }67 #[cfg(not(target_family = "unix"))]68 {69 let str = input.as_os_str().to_str().expect("bad utf-8");70 let cstr = CString::new(str).expect("input has NUL inside");71 Cow::Owned(cstr)72 }73}7475pub struct VM {76 state: State,77 manifest_format: Box<dyn ManifestFormat>,78 trace_format: Box<dyn TraceFormat>,79 tla_args: GcHashMap<IStr, TlaArg>,80}818283#[no_mangle]84#[allow(clippy::box_default)]85pub extern "C" fn jsonnet_make() -> *mut VM {86 let state = State::default();87 state.settings_mut().import_resolver = tb!(FileImportResolver::default());88 state.settings_mut().context_initializer = tb!(jrsonnet_stdlib::ContextInitializer::new(89 state.clone(),90 PathResolver::new_cwd_fallback(),91 ));92 Box::into_raw(Box::new(VM {93 state,94 manifest_format: Box::new(JsonFormat::default()),95 trace_format: Box::new(CompactFormat::default()),96 tla_args: GcHashMap::new(),97 }))98}99100101#[no_mangle]102#[allow(clippy::boxed_local)]103pub extern "C" fn jsonnet_destroy(vm: Box<VM>) {104 drop(vm);105}106107108#[no_mangle]109pub extern "C" fn jsonnet_max_stack(_vm: &VM, v: c_uint) {110 set_stack_depth_limit(v as usize)111}112113114115116#[no_mangle]117pub extern "C" fn jsonnet_gc_min_objects(_vm: &VM, _v: c_uint) {}118119120121122#[no_mangle]123pub extern "C" fn jsonnet_gc_growth_trigger(_vm: &VM, _v: c_double) {}124125126#[no_mangle]127pub extern "C" fn jsonnet_string_output(vm: &mut VM, v: c_int) {128 vm.manifest_format = match v {129 0 => Box::new(JsonFormat::default()),130 1 => Box::new(ToStringFormat),131 _ => panic!("incorrect output format"),132 };133}134135136137138139140141142143#[no_mangle]144pub unsafe extern "C" fn jsonnet_realloc(_vm: &VM, buf: *mut u8, sz: usize) -> *mut u8 {145 if buf.is_null() {146 if sz == 0 {147 return std::ptr::null_mut();148 }149 return unsafe {150 std::alloc::alloc(Layout::from_size_align(sz, std::mem::align_of::<u8>()).unwrap())151 };152 }153 154 155 156 157 let old_layout = Layout::from_size_align(16, std::mem::align_of::<u8>()).unwrap();158 if sz == 0 {159 unsafe { std::alloc::dealloc(buf, old_layout) };160 return std::ptr::null_mut();161 }162 unsafe { std::alloc::realloc(buf, old_layout, sz) }163}164165166167168#[no_mangle]169#[allow(clippy::boxed_local)]170pub extern "C" fn jsonnet_json_destroy(_vm: &VM, v: Box<Val>) {171 drop(v);172}173174175#[no_mangle]176pub extern "C" fn jsonnet_max_trace(vm: &mut VM, v: c_uint) {177 if let Some(format) = vm.trace_format.as_any_mut().downcast_mut::<CompactFormat>() {178 format.max_trace = v as usize179 } else {180 panic!("max_trace is not supported by current tracing format")181 }182}183184185186187188189190191#[no_mangle]192pub unsafe extern "C" fn jsonnet_evaluate_file(193 vm: &VM,194 filename: *const c_char,195 error: &mut c_int,196) -> *const c_char {197 let filename = unsafe { parse_path(CStr::from_ptr(filename)) };198 match vm199 .state200 .import(filename)201 .and_then(|val| apply_tla(vm.state.clone(), &vm.tla_args, val))202 .and_then(|val| val.manifest(&vm.manifest_format))203 {204 Ok(v) => {205 *error = 0;206 CString::new(&*v as &str).unwrap().into_raw()207 }208 Err(e) => {209 *error = 1;210 let mut out = String::new();211 vm.trace_format.write_trace(&mut out, &e).unwrap();212 CString::new(&out as &str).unwrap().into_raw()213 }214 }215}216217218219220221222223224#[no_mangle]225pub unsafe extern "C" fn jsonnet_evaluate_snippet(226 vm: &VM,227 filename: *const c_char,228 snippet: *const c_char,229 error: &mut c_int,230) -> *const c_char {231 let filename = unsafe { CStr::from_ptr(filename) };232 let snippet = unsafe { CStr::from_ptr(snippet) };233 match vm234 .state235 .evaluate_snippet(filename.to_str().unwrap(), snippet.to_str().unwrap())236 .and_then(|val| apply_tla(vm.state.clone(), &vm.tla_args, val))237 .and_then(|val| val.manifest(&vm.manifest_format))238 {239 Ok(v) => {240 *error = 0;241 CString::new(&*v as &str).unwrap().into_raw()242 }243 Err(e) => {244 *error = 1;245 let mut out = String::new();246 vm.trace_format.write_trace(&mut out, &e).unwrap();247 CString::new(&out as &str).unwrap().into_raw()248 }249 }250}251252fn val_to_multi(val: Val, format: &dyn ManifestFormat) -> Result<Vec<(IStr, IStr)>> {253 let Val::Obj(val) = val else {254 bail!("expected object as multi output")255 };256 let mut out = Vec::new();257 for (k, v) in val.iter(258 #[cfg(feature = "exp-preserve-order")]259 false,260 ) {261 out.push((k, v?.manifest(format)?.into()));262 }263 Ok(out)264}265266fn multi_to_raw(multi: Vec<(IStr, IStr)>) -> *const c_char {267 let mut out = Vec::new();268 for (i, (k, v)) in multi.iter().enumerate() {269 if i != 0 {270 out.push(0);271 }272 out.extend_from_slice(k.as_bytes());273 out.push(0);274 out.extend_from_slice(v.as_bytes());275 }276 out.push(0);277 out.push(0);278 let v = out.as_ptr();279 std::mem::forget(out);280 v.cast::<c_char>()281}282283284#[no_mangle]285pub unsafe extern "C" fn jsonnet_evaluate_file_multi(286 vm: &VM,287 filename: *const c_char,288 error: &mut c_int,289) -> *const c_char {290 let filename = unsafe { parse_path(CStr::from_ptr(filename)) };291 match vm292 .state293 .import(filename)294 .and_then(|val| apply_tla(vm.state.clone(), &vm.tla_args, val))295 .and_then(|val| val_to_multi(val, &vm.manifest_format))296 {297 Ok(v) => {298 *error = 0;299 multi_to_raw(v)300 }301 Err(e) => {302 *error = 1;303 let mut out = String::new();304 vm.trace_format.write_trace(&mut out, &e).unwrap();305 CString::new(&out as &str).unwrap().into_raw()306 }307 }308}309310311#[no_mangle]312pub unsafe extern "C" fn jsonnet_evaluate_snippet_multi(313 vm: &VM,314 filename: *const c_char,315 snippet: *const c_char,316 error: &mut c_int,317) -> *const c_char {318 let filename = unsafe { CStr::from_ptr(filename) };319 let snippet = unsafe { CStr::from_ptr(snippet) };320 match vm321 .state322 .evaluate_snippet(filename.to_str().unwrap(), snippet.to_str().unwrap())323 .and_then(|val| apply_tla(vm.state.clone(), &vm.tla_args, val))324 .and_then(|val| val_to_multi(val, &vm.manifest_format))325 {326 Ok(v) => {327 *error = 0;328 multi_to_raw(v)329 }330 Err(e) => {331 *error = 1;332 let mut out = String::new();333 vm.trace_format.write_trace(&mut out, &e).unwrap();334 CString::new(&out as &str).unwrap().into_raw()335 }336 }337}338339fn val_to_stream(val: Val, format: &dyn ManifestFormat) -> Result<Vec<IStr>> {340 let Val::Arr(val) = val else {341 bail!("expected array as stream output")342 };343 let mut out = Vec::new();344 for item in val.iter() {345 out.push(item?.manifest(format)?.into());346 }347 Ok(out)348}349350fn stream_to_raw(multi: Vec<IStr>) -> *const c_char {351 let mut out = Vec::new();352 for (i, v) in multi.iter().enumerate() {353 if i != 0 {354 out.push(0);355 }356 out.extend_from_slice(v.as_bytes());357 }358 out.push(0);359 out.push(0);360 let v = out.as_ptr();361 std::mem::forget(out);362 v as *const c_char363}364365366#[no_mangle]367pub unsafe extern "C" fn jsonnet_evaluate_file_stream(368 vm: &VM,369 filename: *const c_char,370 error: &mut c_int,371) -> *const c_char {372 let filename = unsafe { parse_path(CStr::from_ptr(filename)) };373 match vm374 .state375 .import(filename)376 .and_then(|val| apply_tla(vm.state.clone(), &vm.tla_args, val))377 .and_then(|val| val_to_stream(val, &vm.manifest_format))378 {379 Ok(v) => {380 *error = 0;381 stream_to_raw(v)382 }383 Err(e) => {384 *error = 1;385 let mut out = String::new();386 vm.trace_format.write_trace(&mut out, &e).unwrap();387 CString::new(&out as &str).unwrap().into_raw()388 }389 }390}391392393#[no_mangle]394pub unsafe extern "C" fn jsonnet_evaluate_snippet_stream(395 vm: &VM,396 filename: *const c_char,397 snippet: *const c_char,398 error: &mut c_int,399) -> *const c_char {400 let filename = unsafe { CStr::from_ptr(filename) };401 let snippet = unsafe { CStr::from_ptr(snippet) };402 match vm403 .state404 .evaluate_snippet(filename.to_str().unwrap(), snippet.to_str().unwrap())405 .and_then(|val| apply_tla(vm.state.clone(), &vm.tla_args, val))406 .and_then(|val| val_to_stream(val, &vm.manifest_format))407 {408 Ok(v) => {409 *error = 0;410 stream_to_raw(v)411 }412 Err(e) => {413 *error = 1;414 let mut out = String::new();415 vm.trace_format.write_trace(&mut out, &e).unwrap();416 CString::new(&out as &str).unwrap().into_raw()417 }418 }419}