JSON to JSON transformation where the "specification" for the transform is itself a JSON document.
Port of Java Jolt library written in Rust.
Add fluvio-jolt crate to your Cargo.toml file:
toml
[dependencies]
fluvio-jolt = { version = "0.1"}
Then, for example, if you want to repack your JSON record, you can do the following: ```rust use serdejson::{json, Value}; use fluviojolt::{transform, TransformSpec};
let input: Value = serdejson::fromstr(r#" { "id": 1, "name": "John Smith", "account": { "id": 1000, "type": "Checking" } } "#).unwrap();
let spec: TransformSpec = serdejson::fromstr(r#"[ { "operation": "shift", "spec": { "name": "data.name", "account": "data.account" } } ]"#).unwrap();
let output = transform(input, &spec);
assert_eq!(output, json!({ "data" : { "name": "John Smith", "account": { "id": 1000, "type": "Checking" } } })); ```
shift: copy data from the input tree and put it the output treedefault: apply default values to the treeremove: remove data from the treeComposes a list of operation specifications. Each operation has its own DSL (Domain Specific Language) in order to facilitate its narrow job.
``` use fluvio_jolt::TransformSpec;
let spec: TransformSpec = serdejson::fromstr(r#"[ { "operation": "shift", "spec": { "name": "data.name", "account": "data.account" } } ]"#).unwrap(); ```
Shift operationSpecifies where the data from the input JSON should be placed in the output JSON, or in other words, how the input JSON/data should be shifted around to make the output JSON/data.
At a base level, a single shift operation is a mapping from an input path to an output path,
similar to the mv command in Unix, mv /var/data /var/backup/data.
The input path is a JSON tree structure, and the output path is flattened "dot notation" path notation.
For example, given this simple input JSON:
{
"id": 1,
"name": "John Smith",
"account": {
"id": 1000,
"type": "Checking"
}
}
A simple spec could be constructed by copying that input, and modifying it to supply an output
path for each piece of data:
{
"id": "data.id",
"name": "data.name",
"account": "data.account"
}
would produce the following output JSON:
{
"data" : {
"id": 1,
"name": "John Smith",
"account": {
"id": 1000,
"type": "Checking"
}
}
}
The shift specification on the keys level supports wildcards and conditions:
1. * - match everything
2. name1|name2|nameN - match any of the specified names
& lookup allows referencing the values captured by the * or |.
It allows for specs to be more compact. For example, for this input:
{
"id": 1,
"name": "John Smith",
"account": {
"id": 1000,
"type": "Checking"
}
}
to get the output:
{
"data" : {
"id": 1,
"name": "John Smith",
"account": {
"id": 1000,
"type": "Checking"
}
}
}
the spec with wildcards would be:
{
"*": "data.&0"
}
If you want only id and name in the output, the spec is:
{
"id|name": "data.&0"
}
& wildcard also allows to dereference any level of the path of given node:
{
"foo": {
"bar" : {
"baz": "new_location.&0.&1.&2" // &0 = baz, &1 = bar, &2 = foo
}
}
}
}
for the input:
{
"foo": {
"bar": {
"baz": "value"
}
}
}
will produce:
{
"new_location": {
"baz": {
"bar": {
"foo": "value"
}
}
}
}
Default operationApplies default values if the value is not present in the input JSON.
For example, given this simple input JSON:
{
"phones": {
"mobile": 01234567,
"country": "US"
}
}
with the following specification for default operation:
{
"phones": {
"mobile": 0000000,
"code": "+1"
}
}
the output JSON will be:
{
"phones": {
"mobile": 01234567,
"country": "US",
"code": "+1"
}
}
As you can see, the field mobile remains not affected while the code has a default '+1' value.
Remove operationRemoves content from the input JSON. The spec structure matches the input JSON structure. The value of fields is ignored.
For example, given this simple input JSON:
{
"phones": {
"mobile": 01234567,
"country": "US"
}
}
you can remove the country by the following specification for remove operation:
{
"phones": {
"country": ""
}
}
the output JSON will be:
{
"phones": {
"mobile": 01234567
}
}
If you'd like to contribute to the project, please read our Contributing guide.
This project is licensed under the Apache license.