16 unstable releases (3 breaking)
0.3.3 | Dec 12, 2024 |
---|---|
0.3.1 | Nov 26, 2024 |
0.1.5 | Jun 26, 2024 |
0.1.0 | Nov 14, 2023 |
#94 in Parser implementations
13,165 downloads per month
340KB
8K
SLoC
jsonata-rs
An (incomplete) implementation of JSONata in Rust.
Alpha version. All internal and external interfaces are considered unstable and subject to change without notice.
What is JSONata?
From the JSONata website:
- Lightweight query and transformation language for JSON data
- Inspired by the location path semantics of XPath 3.1
- Sophisticated query expressions with minimal syntax
- Built-in operators and functions for manipulating and combining data
- Create user-defined functions
- Format query results into any JSON output structure
Read the complete documentation, and try it out in Stedi's JSONata Playground.
Getting started
The API is not ergonomic (yet), as you must provide a bumpalo
arena.
First, add the following to your Cargo.toml
:
[dependencies]
jsonata-rs = "0"
bumpalo = "3.9.1"
Then you can evaluate an expression with JSON input like this:
use bumpalo::Bump;
use jsonata_rs::JsonAta;
// Create an arena for allocating values
let arena = Bump::new();
// Provide some JSON input. This could be read from a file or come from the network.
let input = r#"{ "name": "world" }";
// The JSONata expression to evaluate
let expr = r#""Hello, " & name & "!"";
// Parse the expression
let jsonata = JsonAta::new(expr, &arena).unwrap();
// Evaluate the expression against the input. The second parameter should
// contain any binding variables you want to be available during evaluations.
let result = jsonata.evaluate(Some(input), None).unwrap();
// Serialize the result into JSON
println!("{}", result.serialize(false));
There's also a basic CLI tool:
# cargo install jsonata-rs
# jsonata "1 + 1"
2
# jsonata '"Hello, " & name & "!"' '{ "name": "world" }'
"Hello, world!"
The expression and input can be specified on the command line, which requires manual escaping. Alternatively, they can be provided from files. Here's the --help
output:
# jsonata --help
jsonata-rs
A command line JSON processor using JSONata
USAGE:
jsonata [FLAGS] [OPTIONS] [ARGS]
FLAGS:
-a, --ast Parse the given expression, print the AST and exit
-h, --help Prints help information
-V, --version Prints version information
OPTIONS:
-e, --expr-file <expr-file> File containing the JSONata expression to evaluate (overrides expr on command line)
-i, --input-file <input-file> Input JSON file (if not specified, STDIN)
ARGS:
<expr> JSONata expression to evaluate
<input> JSON input
Missing (but planned) features
There are several JSONata features which are not yet implemented:
- Many built-in functions are missing
- Parent operator
- Regular expressions
- Partial function application
Differences from reference JSONata
Function signatures are not supported
Function signatures have problems as described here, and are not supported by this implementation.
Most of the JSONata functions, however, support being passed the context as the first argument as dictated by their signature, e.g:
["Hello", "world"].$substring(1, 2)
/* Output: ["el", "or"] */
This is implemented in each built-in function itself. For example, if $string
sees that it is called without arguments, it will use the current context.
In addition, for all the built-in functions, type checking of arguments is also implemented directly in the functions themselves so that you get equivalent runtime errors for passing the wrong things to these functions as you would in reference JSONata.
Tests
Reference JSONata contains an extensive test suite with over 1,000 tests. Currently, this implementation passes almost 800 of these. You can run them like this:
cargo test testsuite
In tests/testsuite/groups
are the tests groups that are passing, while tests/testsuite/skip
contains the groups that still require feature implementation. There may be tests in the remaining groups that do pass, but I don't want to split them up - only when a test group fully passes is it moved.
Development status and goals
Status
There are several issues to be resolved:
- There are obviously still a bunch of missing features. We're aiming for feature parity with the reference implementation wherever feasible.
- The API has not had any real thought put into it yet.
- This implementation attempts structural sharing of the input and output values with minimal heap allocations. This was a lot of effort working out the lifetimes that may not be worth it. We may consider removing Bumpalo in the future.
- We have made a couple of optimization passes, but there are still lots of opportunities for improvement.
- The code is spaghetti in some places and could be more Rust-idiomatic.
Goals
- Feature-parity with the reference implementation (within reason)
- Clean API and idiomatic code (i.e. make the easy things easy and the complex possible)
- Well-documented for users and easy to onboard for contributors
- Efficient and optimized with minimal low-hanging fruit.
Contribution
We welcome community contributions and pull requests.
License
This project is licensed under the Apache-2.0 License. Any code you submit will be released under that license.
Dependencies
~8.5MB
~156K SLoC