difftreelog
refactor rewrite builtins to new type system
in: master
8 files changed
Cargo.lockdiffbeforeafterboth61 packageslockfile v1
Might be heavy and slow!
annotate-snippets
0.9.0crates.io↘ 2↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum5c96c3d1062ea7101741480185a6a1275eab01cbe8b20e378d1311bc056d2e08depends onused byatty
0.2.14crates.io↘ 3↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksumd9b39be18770d11421cdb1b9947a45dd3f37e93092cbf377614828a319d5fee8depends onused byautocfg
1.0.0crates.io↘ 0↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksumf8aac770f1885fd7e387acedd76065302551364496e46b3dd00860b2f8359b9dused bybase64
0.13.0crates.io↘ 0↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum904dfeac50f3cdaba28fc6f57fdcddb75f49ed61346676a78c4ffe55877802fdused bybincode
1.3.1crates.io↘ 2↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksumf30d3a39baa26f9651f17b375061f3233dde33424a8b72b0dbe93a68a0bc896ddepends onused bybitflags
1.2.1crates.io↘ 0↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksumcf1de2fe8c75bc145a2f577add951f8134889b4795d47466a54a5c846d691693used bybyteorder
1.3.4crates.io↘ 0↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum08c48aae112d48ed9f069b33538ea9e3e90aa263cfa3d1c24309612b1f7472deused bycc
1.0.55crates.io↘ 0↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksumb1be3409f94d7bdceeb5f5fac551039d9b3f00e25da7a74fc4d33400a0d96368used byclap
3.0.0-beta.2github.com/clap-rs/clap↘ 11↖ 2sourcegit+https://github.com/clap-rs/clap?rev=50fb4ca63dc7e8fb8aa2914de5c8a5b4413f2c62#50fb4ca63dc7e8fb8aa2914de5c8a5b4413f2c62depends onclap_derive
3.0.0-beta.2github.com/clap-rs/clap↘ 5↖ 1sourcegit+https://github.com/clap-rs/clap?rev=50fb4ca63dc7e8fb8aa2914de5c8a5b4413f2c62#50fb4ca63dc7e8fb8aa2914de5c8a5b4413f2c62used byclosure
0.3.0crates.io↘ 0↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksumd6173fd61b610d15a7566dd7b7620775627441c4ab9dac8906e17cb93a24b782used byhashbrown
0.9.0crates.io↘ 0↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum00d63df3d41950fb462ed38308eea019113ad1508da725bbedcd0fa5a85ef5f7used byheck
0.3.1crates.io↘ 1↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum20564e78d53d2bb135c343b3f47714a56af2061f1c928fdb541dc7b9fdd94205depends onused byhermit-abi
0.1.14crates.io↘ 1↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksumb9586eedd4ce6b3c498bc3b4dd92fc9f11166aa908a914071953768066c67909depends onused byindexmap
1.6.0crates.io↘ 2↖ 2sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum55e2e4c765aa53a0424761bf9f41aa7a6ac1efa87238f59560640e27fca028f2depends onitoa
0.4.6crates.io↘ 0↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksumdc6f3ad7b9d11a0c00842ff8de1b60ee58661048eb8049ed33c73594f359d7e6used byjrsonnet
0.3.3workspace↘ 6↖ 0jrsonnet-cli
0.3.3workspace↘ 3↖ 1jrsonnet-evaluator
0.3.3workspace↘ 14↖ 3jrsonnet-parser
0.3.3workspace↘ 6↖ 4jrsonnet-stdlib
0.3.3workspace↘ 0↖ 2jsonnet
0.3.3workspace↘ 2↖ 0lazy_static
1.4.0crates.io↘ 0↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksume2abad23fbc42b3700f2f279844dc832adb2b2eb069b2df918f455c4e18cc646used bylibc
0.2.71crates.io↘ 0↖ 3sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum9457b06509d27052635f90d6466700c65095fdf75409b3fbdd903e988b886f49md5
0.7.0crates.io↘ 0↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum490cc448043f947bae3cbee9c203358d62dbee0db12107a74be5c30ccfd09771used bymimalloc-sys
0.1.6crates.io↘ 2↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum4aa3cefb626f6ae3d0b2f71c5378c89d2b1d4d7bc246b0ca9a7ee61a4daad291depends onused bymimallocator
0.1.3crates.io↘ 1↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum2d44fe4ebf6b538fcf39d9975c2b90bb3232d1ba8e8bffeacd004f27b20c577adepends onused byos_str_bytes
2.4.0crates.io↘ 0↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksumafb2e1c3ee07430c2cf76151675e583e0f19985fa6efae47d6848a3e2c824f85used bypathdiff
0.2.0crates.io↘ 0↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum877630b3de15c0b64cc52f659345724fbf6bdad9bd9566699fc53688f3c34a34used bypeg
0.6.3crates.io↘ 2↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum9f76678828272f177ac33b7e2ac2e3e73cc6c1cd1e3e387928aa69562fa51367depends onused bypeg-macros
0.6.3crates.io↘ 3↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum636d60acf97633e48d266d7415a9355d4389cea327a193f87df395d88cd2b14dused bypeg-runtime
0.6.3crates.io↘ 0↖ 2sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum9555b1514d2d99d78150d3c799d4c357a3e2c2a8062cd108e93a06d9057629c5used byproc-macro-error
1.0.3crates.io↘ 5↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksumfc175e9777c3116627248584e8f8b3e2987405cabe1c0adf7d1dd28f09dc7880used byproc-macro-error-attr
1.0.3crates.io↘ 5↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum3cc9795ca17eb581285ec44936da7fc2335a3f34f2ddd13118b6f4d515435c50used byproc-macro2
1.0.24crates.io↘ 1↖ 10sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum1e0704ee1a7e00d7bb417d0770ea303c1bccbabf0ef1667dae92b5967f5f8a71depends onquote
1.0.7crates.io↘ 1↖ 9sourceregistry+https://github.com/rust-lang/crates.io-indexchecksumaa563d17ecb180e500da1cfd2b028310ac758de548efdd203e18f283af693f37depends onrustc-hash
1.1.0crates.io↘ 0↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum08d43f7aa6b08d49f382cde6a7982047c3426db949b1424bc4b7ec9ae12c6ce2used byryu
1.0.5crates.io↘ 0↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum71d301d4193d031abdd79ff7e3dd721168a9572ef3fe51a1517aba235bd8f86eused byserde
1.0.117crates.io↘ 1↖ 4sourceregistry+https://github.com/rust-lang/crates.io-indexchecksumb88fa983de7720629c9387e9f517353ed404164b1e482c970a90c1a4aaf7dc1adepends onserde_derive
1.0.117crates.io↘ 3↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksumcbd1ae72adb44aab48f325a02444a5fc079349a8d804c1fc922aed3f7454c74edepends onused byserde_json
1.0.59crates.io↘ 3↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksumdcac07dbffa1c65e7f816ab9eba78eb142c6d44410f4eeba1e26e4f5dfa56b95depends onused bystrsim
0.10.0crates.io↘ 0↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum73473c0e59e6d5812c5dfe2a064a6444949f089e20eec9a2e5506596494e4623used bystructdump
0.1.2crates.io↘ 0↖ 2sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum2e16ec33a0342fdb67d13913b4ffae6527ebccfa04b5d7da174bdc7a31db29b8structdump-derive
0.1.2crates.io↘ 3↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum06c337fdc077e02ccbfcc62af0090564a4af342975c3b7be09705efab90c1888depends onused bysyn
1.0.48crates.io↘ 3↖ 7sourceregistry+https://github.com/rust-lang/crates.io-indexchecksumcc371affeffc477f42a221a1e4297aedcea33d47d19b61455588bd9d8f6b19acsyn-mid
0.5.0crates.io↘ 3↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum7be3539f6c128a931cf19dcee741c1af532c7fd387baa739c03dd2e96479338adepends onused bytermcolor
1.1.0crates.io↘ 1↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksumbb6bfa289a4d7c5766392812c0a1f4c1ba45afa1ad47803c11e1f407d846d75fdepends onused bytextwrap
0.13.2crates.io↘ 1↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum3cdcf6b66102d38821c33eea2bf1e8b7bd738072171cbf8a0683fbb46fcb8b0bdepends onused bythiserror
1.0.22crates.io↘ 1↖ 2sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum0e9ae34b84616eedaaf1e9dd6026dbe00dcafa92aa0c8077cb69df1fcfe5e53edepends onthiserror-impl
1.0.22crates.io↘ 3↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum9ba20f23e85b10754cd195504aebf6a27e2e6cbe28c17778a0c930724628dd56depends onused byunescape
0.1.0crates.io↘ 0↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksumccb97dac3243214f8d8507998906ca3e2e0b900bf9bf4870477f125b82e68f6eused byunicode-segmentation
1.6.0crates.io↘ 0↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksume83e153d1053cbb5a118eeff7fd5be06ed99153f00dbcd8ae310c5fb2b22edc0used byunicode-width
0.1.7crates.io↘ 0↖ 3sourceregistry+https://github.com/rust-lang/crates.io-indexchecksumcaaa9d531767d1ff2150b9332433f32a24622147e5ebb1f26409d5da67afd479unicode-xid
0.2.0crates.io↘ 0↖ 2sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum826e7639553986605ec5979c7dd957c7895e93eabed50ab2ffa7f6128a75097cused byvec_map
0.8.2crates.io↘ 0↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksumf1bddf1187be692e79c5ffeab891132dfb0f236ed36a43c7ed39f1165ee20191used byversion_check
0.9.2crates.io↘ 0↖ 2sourceregistry+https://github.com/rust-lang/crates.io-indexchecksumb5a972e5669d67ba988ce3dc826706fb0a8b01471c088cb0b6110b805cc36aedwinapi
0.3.9crates.io↘ 2↖ 3sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum5c839a674fcd7a98952e593242ea400abe93992746761e38641405d28b00f419winapi-i686-pc-windows-gnu
0.4.0crates.io↘ 0↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksumac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6used bywinapi-util
0.1.5crates.io↘ 1↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum70ec6ce85bb158151cae5e5c87f95a8e97d2c0c4b001223f33a334e3ce5de178depends onused bywinapi-x86_64-pc-windows-gnu
0.4.0crates.io↘ 0↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksum712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183fused byyansi-term
0.1.2crates.io↘ 1↖ 1sourceregistry+https://github.com/rust-lang/crates.io-indexchecksumfe5c30ade05e61656247b2e334a031dfd0cc466fadef865bdcdea8d537951bf1depends onused by
Cargo.tomldiffbeforeafterboth--- a/Cargo.toml
+++ b/Cargo.toml
@@ -4,8 +4,9 @@
"crates/jrsonnet-evaluator",
"crates/jrsonnet-stdlib",
"crates/jrsonnet-cli",
+ "crates/jrsonnet-types",
"bindings/jsonnet",
- "cmds/jrsonnet"
+ "cmds/jrsonnet",
]
[profile.test]
crates/jrsonnet-evaluator/Cargo.tomldiffbeforeafterboth--- a/crates/jrsonnet-evaluator/Cargo.toml
+++ b/crates/jrsonnet-evaluator/Cargo.toml
@@ -26,6 +26,7 @@
[dependencies]
jrsonnet-parser = { path = "../jrsonnet-parser", version = "0.3.3" }
jrsonnet-stdlib = { path = "../jrsonnet-stdlib", version = "0.3.3" }
+jrsonnet-types = { path = "../jrsonnet-types", version = "0.3.3" }
pathdiff = "0.2.0"
closure = "0.3.0"
crates/jrsonnet-evaluator/src/builtin/mod.rsdiffbeforeafterboth--- a/crates/jrsonnet-evaluator/src/builtin/mod.rs
+++ b/crates/jrsonnet-evaluator/src/builtin/mod.rs
@@ -1,17 +1,20 @@
use crate::{
equals,
error::{Error::*, Result},
- evaluate, parse_args, primitive_equals, push, throw, with_state, Context, FuncVal, Val,
- ValType,
+ evaluate, parse_args, primitive_equals, push, throw,
+ typed::CheckType,
+ with_state, ArrValue, Context, FuncVal, LazyVal, Val,
};
use format::{format_arr, format_obj};
-use jrsonnet_parser::{ArgsDesc, ExprLocation};
-use manifest::{escape_string_json, manifest_json_ex, ManifestJsonOptions, ManifestType};
-use std::{path::PathBuf, rc::Rc};
+use jrsonnet_parser::{ArgsDesc, BinaryOpType, ExprLocation};
+use jrsonnet_types::{ty, ComplexValType, ValType};
+use std::{collections::HashMap, path::PathBuf, rc::Rc};
pub mod stdlib;
pub use stdlib::*;
+use self::manifest::{escape_string_json, manifest_json_ex, ManifestJsonOptions, ManifestType};
+
pub mod format;
pub mod manifest;
pub mod sort;
@@ -22,7 +25,7 @@
|| format!("std.format of {}", str),
|| {
Ok(match vals {
- Val::Arr(vals) => Val::Str(format_arr(&str, &vals)?.into()),
+ Val::Arr(vals) => Val::Str(format_arr(&str, &vals.evaluated()?)?.into()),
Val::Obj(obj) => Val::Str(format_obj(&str, &obj)?.into()),
o => Val::Str(format_arr(&str, &[o])?.into()),
})
@@ -30,6 +33,32 @@
)
}
+thread_local! {
+ pub static INTRINSICS: HashMap<&'static str, fn(Context, &Option<ExprLocation>, &ArgsDesc) -> Result<Val>> = {
+ let mut out: HashMap<&'static str, _> = HashMap::new();
+ out.insert("length", intrinsic_length);
+ out
+ };
+}
+
+fn intrinsic_length(context: Context, _loc: &Option<ExprLocation>, args: &ArgsDesc) -> Result<Val> {
+ Ok(parse_args!(context, "length", args, 1, [
+ 0, x: ty!((str | obj | [any]));
+ ], {
+ Ok(match x {
+ Val::Str(n) => Val::Num(n.chars().count() as f64),
+ Val::Arr(a) => Val::Num(a.len() as f64),
+ Val::Obj(o) => Val::Num(
+ o.fields_visibility()
+ .into_iter()
+ .filter(|(_k, v)| *v)
+ .count() as f64,
+ ),
+ _ => unreachable!(),
+ })
+ })?)
+}
+
#[allow(clippy::cognitive_complexity)]
pub fn call_builtin(
context: Context,
@@ -38,13 +67,12 @@
args: &ArgsDesc,
) -> Result<Val> {
Ok(match name as &str {
- // arr/string/function
- "length" => parse_args!(context, "std.length", args, 1, [
- 0, x: [Val::Str|Val::Arr|Val::Obj], vec![ValType::Str, ValType::Arr, ValType::Obj];
+ "length" => parse_args!(context, "length", args, 1, [
+ 0, x: ty!((str | obj | [any]));
], {
Ok(match x {
Val::Str(n) => Val::Num(n.chars().count() as f64),
- Val::Arr(i) => Val::Num(i.len() as f64),
+ Val::Arr(a) => Val::Num(a.len() as f64),
Val::Obj(o) => Val::Num(
o.fields_visibility()
.into_iter()
@@ -54,43 +82,32 @@
_ => unreachable!(),
})
})?,
- // any
- "type" => parse_args!(context, "std.type", args, 1, [
- 0, x, vec![];
+ "type" => parse_args!(context, "type", args, 1, [
+ 0, x: ty!(any);
], {
- Ok(Val::Str(x.value_type()?.name().into()))
+ Ok(Val::Str(x.value_type().name().into()))
})?,
- // length, idx=>any
- "makeArray" => parse_args!(context, "std.makeArray", args, 2, [
- 0, sz: [Val::Num]!!Val::Num, vec![ValType::Num];
- 1, func: [Val::Func]!!Val::Func, vec![ValType::Func];
+ "makeArray" => parse_args!(context, "makeArray", args, 2, [
+ 0, sz: ty!(number((Some(0.0))..(None))) => Val::Num;
+ 1, func: ty!(fn.any) => Val::Func;
], {
- if sz < 0.0 {
- throw!(RuntimeError(format!("makeArray requires size >= 0, got {}", sz).into()));
- }
let mut out = Vec::with_capacity(sz as usize);
for i in 0..sz as usize {
- out.push(func.evaluate_values(
+ out.push(LazyVal::new_resolved(func.evaluate_values(
Context::new(),
&[Val::Num(i as f64)]
- )?)
+ )?))
}
- Ok(Val::Arr(Rc::new(out)))
+ Ok(Val::Arr(out.into()))
})?,
- // string
- "codepoint" => parse_args!(context, "std.codepoint", args, 1, [
- 0, str: [Val::Str]!!Val::Str, vec![ValType::Str];
+ "codepoint" => parse_args!(context, "codepoint", args, 1, [
+ 0, str: ty!(char) => Val::Str;
], {
- assert!(
- str.chars().count() == 1,
- "std.codepoint should receive single char string"
- );
Ok(Val::Num(str.chars().take(1).next().unwrap() as u32 as f64))
})?,
- // object, includeHidden
- "objectFieldsEx" => parse_args!(context, "std.objectFieldsEx",args, 2, [
- 0, obj: [Val::Obj]!!Val::Obj, vec![ValType::Obj];
- 1, inc_hidden: [Val::Bool]!!Val::Bool, vec![ValType::Bool];
+ "objectFieldsEx" => parse_args!(context, "objectFieldsEx", args, 2, [
+ 0, obj: ty!(obj) => Val::Obj;
+ 1, inc_hidden: ty!(bool) => Val::Bool;
], {
let mut out = obj.fields_visibility()
.into_iter()
@@ -98,13 +115,12 @@
.map(|(k, _v)|k)
.collect::<Vec<_>>();
out.sort();
- Ok(Val::Arr(Rc::new(out.into_iter().map(Val::Str).collect())))
+ Ok(Val::Arr(out.into_iter().map(Val::Str).collect::<Vec<_>>().into()))
})?,
- // object, field, includeHidden
- "objectHasEx" => parse_args!(context, "std.objectHasEx", args, 3, [
- 0, obj: [Val::Obj]!!Val::Obj, vec![ValType::Obj];
- 1, f: [Val::Str]!!Val::Str, vec![ValType::Str];
- 2, inc_hidden: [Val::Bool]!!Val::Bool, vec![ValType::Bool];
+ "objectHasEx" => parse_args!(context, "objectHasEx", args, 3, [
+ 0, obj: ty!(obj) => Val::Obj;
+ 1, f: ty!(str) => Val::Str;
+ 2, inc_hidden: ty!(bool) => Val::Bool;
], {
Ok(Val::Bool(
obj.fields_visibility()
@@ -113,13 +129,12 @@
.any(|(k, _v)| *k == *f),
))
})?,
-
// faster
"slice" => parse_args!(context, "slice", args, 4, [
- 0, indexable: [Val::Str | Val::Arr], vec![ValType::Str, ValType::Arr];
- 1, index, vec![ValType::Num, ValType::Null];
- 2, end, vec![ValType::Num, ValType::Null];
- 3, step, vec![ValType::Num, ValType::Null];
+ 0, indexable: ty!((str | [any]));
+ 1, index: ty!((num | null));
+ 2, end: ty!((num | null));
+ 3, step: ty!((num | null));
], {
let index = match index {
Val::Num(v) => v as usize,
@@ -145,53 +160,53 @@
Ok(Val::Str((s.chars().skip(index).take(end-index).step_by(step).collect::<String>()).into()))
}
Val::Arr(arr) => {
- Ok(Val::Arr((arr.iter().skip(index).take(end-index).step_by(step).cloned().collect::<Vec<Val>>()).into()))
+ Ok(Val::Arr((arr.iter().skip(index).take(end-index).step_by(step).collect::<Result<Vec<Val>>>()?).into()))
}
_ => unreachable!()
}
})?,
- "primitiveEquals" => parse_args!(context, "std.primitiveEquals", args, 2, [
- 0, a, vec![];
- 1, b, vec![];
+ "primitiveEquals" => parse_args!(context, "primitiveEquals", args, 2, [
+ 0, a: ty!(any);
+ 1, b: ty!(any);
], {
Ok(Val::Bool(primitive_equals(&a, &b)?))
})?,
// faster
- "equals" => parse_args!(context, "std.equals", args, 2, [
- 0, a, vec![];
- 1, b, vec![];
+ "equals" => parse_args!(context, "equals", args, 2, [
+ 0, a: ty!(any);
+ 1, b: ty!(any);
], {
Ok(Val::Bool(equals(&a, &b)?))
})?,
- "mod" => parse_args!(context, "std.mod", args, 2, [
- 0, a: [Val::Num | Val::Str], vec![ValType::Num, ValType::Str];
- 1, b, vec![];
+ "modulo" => parse_args!(context, "modulo", args, 2, [
+ 0, a: ty!(num) => Val::Num;
+ 1, b: ty!(num) => Val::Num;
+ ], {
+ Ok(Val::Num(a % b))
+ })?,
+ "mod" => parse_args!(context, "mod", args, 2, [
+ 0, a: ty!((num | str));
+ 1, b: ty!(any);
], {
match (a, b) {
(Val::Num(a), Val::Num(b)) => Ok(Val::Num(a % b)),
(Val::Str(str), vals) => std_format(str, vals),
- (a, b) => throw!(BinaryOperatorDoesNotOperateOnValues(jrsonnet_parser::BinaryOpType::Mod, a.value_type()?, b.value_type()?))
+ (a, b) => throw!(BinaryOperatorDoesNotOperateOnValues(BinaryOpType::Mod, a.value_type(), b.value_type()))
}
})?,
- "modulo" => parse_args!(context, "std.modulo", args, 2, [
- 0, a: [Val::Num]!!Val::Num, vec![ValType::Num];
- 1, b: [Val::Num]!!Val::Num, vec![ValType::Num];
- ], {
- Ok(Val::Num(a % b))
- })?,
- "floor" => parse_args!(context, "std.floor", args, 1, [
- 0, x: [Val::Num]!!Val::Num, vec![ValType::Num];
+ "floor" => parse_args!(context, "floor", args, 1, [
+ 0, x: ty!(num) => Val::Num;
], {
Ok(Val::Num(x.floor()))
})?,
- "log" => parse_args!(context, "std.log", args, 2, [
- 0, n: [Val::Num]!!Val::Num, vec![ValType::Num];
+ "log" => parse_args!(context, "log", args, 1, [
+ 0, n: ty!(num) => Val::Num;
], {
Ok(Val::Num(n.ln()))
})?,
- "trace" => parse_args!(context, "std.trace", args, 2, [
- 0, str: [Val::Str]!!Val::Str, vec![ValType::Str];
- 1, rest, vec![];
+ "trace" => parse_args!(context, "trace", args, 2, [
+ 0, str: ty!(str) => Val::Str;
+ 1, rest: ty!(any);
], {
eprint!("TRACE:");
if let Some(loc) = loc {
@@ -203,94 +218,88 @@
eprintln!(" {}", str);
Ok(rest)
})?,
- "pow" => parse_args!(context, "std.modulo", args, 2, [
- 0, x: [Val::Num]!!Val::Num, vec![ValType::Num];
- 1, n: [Val::Num]!!Val::Num, vec![ValType::Num];
+ "pow" => parse_args!(context, "pow", args, 2, [
+ 0, x: ty!(num) => Val::Num;
+ 1, n: ty!(num) => Val::Num;
], {
Ok(Val::Num(x.powf(n)))
})?,
- "extVar" => parse_args!(context, "std.extVar", args, 1, [
- 0, x: [Val::Str]!!Val::Str, vec![ValType::Str];
+ "extVar" => parse_args!(context, "extVar", args, 1, [
+ 0, x: ty!(str) => Val::Str;
], {
Ok(with_state(|s| s.settings().ext_vars.get(&x).cloned()).ok_or(UndefinedExternalVariable(x))?)
})?,
- "native" => parse_args!(context, "std.native", args, 1, [
- 0, x: [Val::Str]!!Val::Str, vec![ValType::Str];
+ "native" => parse_args!(context, "native", args, 1, [
+ 0, x: ty!(str) => Val::Str;
], {
Ok(with_state(|s| s.settings().ext_natives.get(&x).cloned()).map(|v| Val::Func(Rc::new(FuncVal::NativeExt(x.clone(), v)))).ok_or(UndefinedExternalFunction(x))?)
})?,
- "filter" => parse_args!(context, "std.filter", args, 2, [
- 0, func: [Val::Func]!!Val::Func, vec![ValType::Func];
- 1, arr: [Val::Arr]!!Val::Arr, vec![ValType::Arr];
+ "filter" => parse_args!(context, "filter", args, 2, [
+ 0, func: ty!(fn.any) => Val::Func;
+ 1, arr: ty!([any]) => Val::Arr;
], {
- Ok(Val::Arr(Rc::new(
- arr.iter()
- .cloned()
- .filter(|e| {
- func
- .evaluate_values(context.clone(), &[e.clone()])
- .unwrap()
- .try_cast_bool("filter predicate")
- .unwrap()
- })
- .collect(),
- )))
+ let mut out = Vec::new();
+ for item in arr.iter() {
+ let item = item?;
+ if func
+ .evaluate_values(context.clone(), &[item.clone()])?
+ .try_cast_bool("filter predicate")? {
+ out.push(item);
+ }
+ }
+ Ok(Val::Arr(out.into()))
})?,
- // faster
- "foldl" => parse_args!(context, "std.foldl", args, 3, [
- 0, func: [Val::Func]!!Val::Func, vec![ValType::Func];
- 1, arr: [Val::Arr]!!Val::Arr, vec![ValType::Arr];
- 2, init, vec![];
+ "foldl" => parse_args!(context, "foldl", args, 3, [
+ 0, func: ty!(fn.any) => Val::Func;
+ 1, arr: ty!([any]) => Val::Arr;
+ 2, init: ty!(any);
], {
let mut acc = init;
- for i in arr.iter().cloned() {
- acc = func.evaluate_values(context.clone(), &[acc, i])?;
+ for i in arr.iter() {
+ acc = func.evaluate_values(context.clone(), &[acc, i?])?;
}
Ok(acc)
})?,
- // faster
- "foldr" => parse_args!(context, "std.foldr", args, 3, [
- 0, func: [Val::Func]!!Val::Func, vec![ValType::Func];
- 1, arr: [Val::Arr]!!Val::Arr, vec![ValType::Arr];
- 2, init, vec![];
+ "foldr" => parse_args!(context, "foldr", args, 3, [
+ 0, func: ty!(fn.any) => Val::Func;
+ 1, arr: ty!([any]) => Val::Arr;
+ 2, init: ty!(any);
], {
let mut acc = init;
- for i in arr.iter().rev().cloned() {
- acc = func.evaluate_values(context.clone(), &[acc, i])?;
+ for i in arr.iter().rev() {
+ acc = func.evaluate_values(context.clone(), &[acc, i?])?;
}
Ok(acc)
})?,
- // faster
#[allow(non_snake_case)]
- "sortImpl" => parse_args!(context, "std.sort", args, 2, [
- 0, arr: [Val::Arr]!!Val::Arr, vec![ValType::Arr];
- 1, keyF: [Val::Func]!!Val::Func, vec![ValType::Func];
+ "sortImpl" => parse_args!(context, "sort", args, 2, [
+ 0, arr: ty!([any]) => Val::Arr;
+ 1, keyF: ty!(fn.any) => Val::Func;
], {
if arr.len() <= 1 {
return Ok(Val::Arr(arr))
}
- Ok(Val::Arr(sort::sort(context, arr, &keyF)?))
+ Ok(Val::Arr(ArrValue::Eager(sort::sort(context, arr.evaluated()?, &keyF)?)))
})?,
// faster
- "format" => parse_args!(context, "std.format", args, 2, [
- 0, str: [Val::Str]!!Val::Str, vec![ValType::Str];
- 1, vals, vec![]
+ "format" => parse_args!(context, "format", args, 2, [
+ 0, str: ty!(str) => Val::Str;
+ 1, vals: ty!(any)
], {
std_format(str, vals)
})?,
- // faster
- "range" => parse_args!(context, "std.range", args, 2, [
- 0, from: [Val::Num]!!Val::Num, vec![ValType::Num];
- 1, to: [Val::Num]!!Val::Num, vec![ValType::Num];
+ "range" => parse_args!(context, "range", args, 2, [
+ 0, from: ty!(num) => Val::Num;
+ 1, to: ty!(num) => Val::Num;
], {
let mut out = Vec::with_capacity((1+to as usize-from as usize).max(0));
for i in from as usize..=to as usize {
out.push(Val::Num(i as f64));
}
- Ok(Val::Arr(Rc::new(out)))
+ Ok(Val::Arr(out.into()))
})?,
- "char" => parse_args!(context, "std.char", args, 1, [
- 0, n: [Val::Num]!!Val::Num, vec![ValType::Num];
+ "char" => parse_args!(context, "char", args, 1, [
+ 0, n: ty!(num) => Val::Num;
], {
let mut out = String::new();
out.push(std::char::from_u32(n as u32).ok_or_else(||
@@ -298,19 +307,18 @@
)?);
Ok(Val::Str(out.into()))
})?,
- "encodeUTF8" => parse_args!(context, "std.encodeUtf8", args, 1, [
- 0, str: [Val::Str]!!Val::Str, vec![ValType::Str];
+ "encodeUTF8" => parse_args!(context, "encodeUTF8", args, 1, [
+ 0, str: ty!(str) => Val::Str;
], {
- Ok(Val::Arr(Rc::new(str.bytes().map(|b| Val::Num(b as f64)).collect())))
+ Ok(Val::Arr((str.bytes().map(|b| Val::Num(b as f64)).collect::<Vec<Val>>()).into()))
})?,
- "md5" => parse_args!(context, "std.md5", args, 1, [
- 0, str: [Val::Str]!!Val::Str, vec![ValType::Str];
+ "md5" => parse_args!(context, "md5", args, 1, [
+ 0, str: ty!(str) => Val::Str;
], {
Ok(Val::Str(format!("{:x}", md5::compute(&str.as_bytes())).into()))
})?,
- // faster
- "base64" => parse_args!(context, "std.base64", args, 1, [
- 0, input: [Val::Str | Val::Arr], vec![ValType::Arr, ValType::Str];
+ "base64" => parse_args!(context, "base64", args, 1, [
+ 0, input: ty!((str | [num]));
], {
Ok(Val::Str(match input {
Val::Str(s) => {
@@ -318,44 +326,51 @@
},
Val::Arr(a) => {
base64::encode(a.iter().map(|v| {
- Ok(v.clone().try_cast_num("base64 array")? as u8)
+ Ok(v?.clone().unwrap_num()? as u8)
}).collect::<Result<Vec<_>>>()?).into()
},
_ => unreachable!()
}))
})?,
- // faster
- "join" => parse_args!(context, "std.join", args, 2, [
- 0, sep: [Val::Str|Val::Arr], vec![ValType::Str, ValType::Arr];
- 1, arr: [Val::Arr]!!Val::Arr, vec![ValType::Arr];
+ "join" => parse_args!(context, "join", args, 2, [
+ 0, sep: ty!((str | [any]));
+ 1, arr: ty!([any]) => Val::Arr;
], {
Ok(match sep {
Val::Arr(joiner_items) => {
let mut out = Vec::new();
let mut first = true;
- for item in arr.iter().cloned() {
- if let Val::Arr(items) = item.unwrap_if_lazy()? {
+ for item in arr.iter() {
+ let item = item?.clone();
+ if let Val::Arr(items) = item {
if !first {
out.reserve(joiner_items.len());
- out.extend(joiner_items.iter().cloned());
+ // TODO: extend
+ for item in joiner_items.iter() {
+ out.push(item?);
+ }
}
first = false;
out.reserve(items.len());
- out.extend(items.iter().cloned());
+ // TODO: extend
+ for item in items.iter() {
+ out.push(item?);
+ }
} else {
throw!(RuntimeError("in std.join all items should be arrays".into()));
}
}
- Val::Arr(Rc::new(out))
+ Val::Arr(out.into())
},
Val::Str(sep) => {
let mut out = String::new();
let mut first = true;
- for item in arr.iter().cloned() {
- if let Val::Str(item) = item.unwrap_if_lazy()? {
+ for item in arr.iter() {
+ let item = item?.clone();
+ if let Val::Str(item) = item {
if !first {
out += &sep;
}
@@ -371,32 +386,30 @@
_ => unreachable!()
})
})?,
- // Faster
- "escapeStringJson" => parse_args!(context, "std.escapeStringJson", args, 1, [
- 0, str_: [Val::Str]!!Val::Str, vec![ValType::Str];
+ // faster
+ "escapeStringJson" => parse_args!(context, "escapeStringJson", args, 1, [
+ 0, str_: ty!(str) => Val::Str;
], {
Ok(Val::Str(escape_string_json(&str_).into()))
})?,
- // Faster
- "manifestJsonEx" => parse_args!(context, "std.manifestJsonEx", args, 2, [
- 0, value, vec![];
- 1, indent: [Val::Str]!!Val::Str, vec![ValType::Str];
+ // faster
+ "manifestJsonEx" => parse_args!(context, "manifestJsonEx", args, 2, [
+ 0, value: ty!(any);
+ 1, indent: ty!(str) => Val::Str;
], {
Ok(Val::Str(manifest_json_ex(&value, &ManifestJsonOptions {
padding: &indent,
mtype: ManifestType::Std,
})?.into()))
})?,
- // Faster
- "reverse" => parse_args!(context, "std.reverse", args, 1, [
- 0, arr: [Val::Arr]!!Val::Arr, vec![ValType::Arr];
+ // faster
+ "reverse" => parse_args!(context, "reverse", args, 1, [
+ 0, value: ty!([any]) => Val::Arr;
], {
- let mut marr = arr;
- Rc::make_mut(&mut marr).reverse();
- Ok(Val::Arr(marr))
+ Ok(Val::Arr(value.reversed()))
})?,
- "id" => parse_args!(context, "std.id", args, 1, [
- 0, v, vec![];
+ "id" => parse_args!(context, "id", args, 1, [
+ 0, v: ty!(any);
], {
Ok(v)
})?,
crates/jrsonnet-evaluator/src/error.rsdiffbeforeafterboth--- a/crates/jrsonnet-evaluator/src/error.rs
+++ b/crates/jrsonnet-evaluator/src/error.rs
@@ -1,8 +1,9 @@
use crate::{
builtin::{format::FormatError, sort::SortError},
- ValType,
+ typed::TypeLocError,
};
use jrsonnet_parser::{BinaryOpType, ExprLocation, UnaryOpType};
+use jrsonnet_types::ValType;
use std::{path::PathBuf, rc::Rc};
use thiserror::Error;
@@ -117,6 +118,8 @@
#[error("format error: {0}")]
Format(#[from] FormatError),
+ #[error("type error: {0}")]
+ TypeError(TypeLocError),
#[error("sort error: {0}")]
Sort(#[from] SortError),
}
@@ -144,6 +147,9 @@
pub const fn error(&self) -> &Error {
&(self.0).0
}
+ pub fn error_mut(&mut self) -> &mut Error {
+ &mut (self.0).0
+ }
pub const fn trace(&self) -> &StackTrace {
&(self.0).1
}
crates/jrsonnet-evaluator/src/function.rsdiffbeforeafterboth--- a/crates/jrsonnet-evaluator/src/function.rs
+++ b/crates/jrsonnet-evaluator/src/function.rs
@@ -143,9 +143,8 @@
#[macro_export]
macro_rules! parse_args {
($ctx: expr, $fn_name: expr, $args: expr, $total_args: expr, [
- $($id: expr, $name: ident $(: [$($p: path)|+] $(!! $a: path)?)?, $nt: expr);+ $(;)?
+ $($id: expr, $name: ident: $ty: expr $(=>$match: path)?);+ $(;)?
], $handler:block) => {{
- use crate::{throw, error::Error::*};
let args = $args;
if args.len() > $total_args {
throw!(TooManyArgsFunctionHas($total_args));
@@ -160,47 +159,19 @@
throw!(IntrinsicArgumentReorderingIsNotSupportedYet);
}
}
- let $name = evaluate($ctx.clone(), &$name.1)?;
+ let $name = push(&None, || format!("evaluating argument"), || {
+ let value = evaluate($ctx.clone(), &$name.1)?;
+ $ty.check(&value)?;
+ Ok(value)
+ })?;
$(
- match $name {
- $($p(_))|+ => {},
- _ => throw!(TypeMismatch(
- concat!($fn_name, " ", stringify!($id), "nd (", stringify!($name), ") argument"),
- $nt, $name.value_type()?
- )),
+ let $name = if let $match(v) = $name {
+ v
+ } else {
+ unreachable!();
};
- $(
- let $name = match $name {
- $a(v) => v,
- _ =>throw!(TypeMismatch(concat!($fn_name, " ", stringify!($id), "nd (", stringify!($name), ") argument"), $nt, $name.value_type()?)),
- };
- )*
- )*
+ )?
)+
($handler as crate::Result<_>)
}};
-}
-
-#[test]
-fn test() -> Result<()> {
- use crate::val::ValType;
- use jrsonnet_parser::*;
- let state = crate::EvaluationState::default();
- let evaluator = state.with_stdlib();
- let ctx = evaluator.create_default_context()?;
- evaluator.run_in_state(|| {
- parse_args!(ctx, "test", ArgsDesc(vec![
- Arg(None, el!(Expr::Num(2.0))),
- Arg(Some("b".into()), el!(Expr::Num(1.0))),
- ]), 2, [
- 0, a: [Val::Num]!!Val::Num, vec![ValType::Num];
- 1, b: [Val::Num]!!Val::Num, vec![ValType::Num];
- ], {
- assert!((a - 2.0).abs() <= f64::EPSILON);
- assert!((b - 1.0).abs() <= f64::EPSILON);
- Ok(())
- })
- .unwrap();
- Ok(())
- })
}
crates/jrsonnet-evaluator/src/lib.rsdiffbeforeafterboth--- a/crates/jrsonnet-evaluator/src/lib.rs
+++ b/crates/jrsonnet-evaluator/src/lib.rs
@@ -14,6 +14,7 @@
pub mod native;
mod obj;
pub mod trace;
+mod typed;
mod val;
pub use ctx::*;
crates/jrsonnet-evaluator/src/trace/mod.rsdiffbeforeafterboth--- a/crates/jrsonnet-evaluator/src/trace/mod.rs
+++ b/crates/jrsonnet-evaluator/src/trace/mod.rs
@@ -65,7 +65,7 @@
out,
"{}:{}-{}:{}",
start.line,
- end.column - 1,
+ end.column.saturating_sub(1),
start.line,
end.column
)?;