Jsonata.Net.Native 2.7.0

dotnet add package Jsonata.Net.Native --version 2.7.0                
NuGet\Install-Package Jsonata.Net.Native -Version 2.7.0                
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="Jsonata.Net.Native" Version="2.7.0" />                
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add Jsonata.Net.Native --version 2.7.0                
#r "nuget: Jsonata.Net.Native, 2.7.0"                
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
// Install Jsonata.Net.Native as a Cake Addin
#addin nuget:?package=Jsonata.Net.Native&version=2.7.0

// Install Jsonata.Net.Native as a Cake Tool
#tool nuget:?package=Jsonata.Net.Native&version=2.7.0                

About

.Net native implementation of JSONata query and transformation language.

  • Jsonata.Net.Native NuGet
  • Jsonata.Net.Native.JsonNet NuGet
  • Jsonata.Net.Native.SystemTextJson NuGet

This implementation is based on original jsonata-js source and also borrows some ideas from go port.

Performance

This implementation is about 100 times faster than straightforward wrapping of original jsonata.js with Jint JS Engine for C# (the wrapping is published as jsonata.net.js package).

For measurements code see src/BenchmarkApp in this repo.

Usage

Basic

  • simple case
using Jsonata.Net.Native;
...
JsonataQuery query = new JsonataQuery("$.a");
...
string result = query.Eval("{\"a\": \"b\"}");
Debug.Assert(result == "\"b\"");

Since version 2.0.0 this package does not depend on JSON.Net, instead it uses a custom implementation of JSON DOM and parser (see Jsonata.Net.Native.Json namespace). This change gave us the following benefits:

  • Things got faster (see here).
  • More Jsonata features wee implemented and are possible to implement in future.
  • No external dependencies for the core Jsonata.Net.Native package (for those who don't use Json.Net in their projects).

Still this custom implementation is modelled on Json.Net, so the following code should look familliar

using Jsonata.Net.Native;
using Jsonata.Net.Native.Json;
...
JToken data = JToken.Parse("{\"a\": \"b\"}");
...
JToken result = query.Eval(data);
Debug.Assert(result.ToFlatString() == "\"b\"");

JSON.Net

In case you work with JSON.Net you may use a separate binding package Jsonata.Net.Native.JsonNet and its single class JsonataExtensions to:

  • convert token hierarchy to and from Json.Net (ToNewtonsoft() and FromNewtonsoft(), note the overload version with formatting overrides)
  • evaluate Jsonata queries via various EvalNewtonsoft() overloads
  • bind values to EvaluationEnvironment (BindValue())

System.Text.Json

Same goes for when you use System.Text.Json. Separate binding package Jsonata.Net.Native.SystemTextJson provides similar JsonataExtensions class with similar wrappers for both JsonDocument/JsonElement (static DOM) and JsonNode (dynamic DOM).

Querying C# objects

It is also possible to create a JToken tree representation of existing C# object via Jsonata.Net.Native.Json.JToken.FromObject() and then query it with JSONata as a regular JSON. This may come in handy when you want to give your user some way to get and combine data from your program object model.

In case you want to go deeper and get more control over JSON representation of your objects, you may want to use Jsonata.Net.Native.JsonNet.JsonataExtensions.FromObjectViaNewtonsoft() and get all fancy stuff that Json.Net has to offer for this.

Since v2.2.0 there's also a (limited) support for converting JTokens back to C# objects via JToken.ToObject().

C# Features

  • JsonataQuery objects are immutable and therefore reusable and thread-safe.
  • It is possible to provide additional variable bindings via bindings arg of Eval() call.
  • It is possible to provide additional functional bindings via Eval(JToken data, EvaluationEnvironment environment) call. See example
  • Error codes are mostly in sync with the JS implementation, but some checkup is to be done later (TODO).

We also provide an Exerciser app with same functionality as in original JSONata Exerciser: Exerciser

Parsing JSON with Jsonata.Net.Native.Json

As mentioned above, modern versions of Jsonata.Net.Native use custom implementation of JSON DOM and parsing. While re-implementation of JSON object model (JToken hierarchy) has been justified by performance and functionality reasons, writing just another JSON parser from scratch in 2022 looked a bit like re-inventing the wheel. On the other hand, forcing some specific external dependency just for the sake of parsing JSON looked even worse. So here you get just another JSON parser available via JToken.Parse() method.

This parser is being checked over the following test sets:

  • JSONTestSuite — most prominent collection of corner case checks for JSON Parsers. Out implementation results are:
    • From "accepted" (y_) section: 95 out of 95 tests are passing (100%).
    • From "rejected" (n_) section: 178 out of 188 tests are passing. 2 tests are causing stackoverflow (those are ones contating 10 000 open square braces). And remaining 14 tests are considered "okay" to fail — which is to parse things that are not being expected to be parsed by strict JSON parsers (eg. numbers like -.123).
    • From "ambigous" (i_) section: all 35 tests are not causing the parser to crush miserably (and expected parsing results are not specified for those tests).
  • JSON_checker — an official but small json.org's parser tests:
    • From "pass" section: 3 out of 3 tests are passing.
    • From "fail" section: 28 out of 33 tests are passing, and remaining 5 are consiered "okay" for same reasons as above.

We have implemented a number of relaxations to "strict" parser spec used in test, like allowing trailing commas, or single-quoted strings. These options are configurable wia ParseSettings class. All relaxations are enabled by default.

When facing an invalid JSON, the parser would throw a JsonParseException

We have put some effort to this parser, but still the main purpose of the package is not parsing JSON by itself, so in case you need more sophisticated parsing features, like comments (or parsing 10 000 open braces) please use some mature parser package like Json.Net or System.Text.Json and convert results to Jsonata.Net.Native.Json.JToken via routines in a binding package.

Query DOM introspection and query creation via DOM

Since v2.7.0 we provide access to query DOM via Node classes in Jsonata.Net.Native.Dom namespace.

Currently it is possible to:

  • acquire (readonly) DOM representation of an existing query (via JsonataQuery.GetDom() call), and inspect it
  • construct new DOM hierarchy and then create a query from it (via JsonataQuery(Node) ctor)
  • check two DOM (sub)trees for equality (via Node.Equals(Node?))

Please note that right now DOM API is experimental and subject to change in backwards-incompatible way without changing major release version of a library.

Some examples may be found here.

Rationale behind this API is here and here.

JSONata language features support

The goal of the project is to implement 100% of latest JSONata version (1.8.5 at the moment of writing these words), but it's still work in progress. Here's is a list of features in accordance to manual:

  • ✔️ Simple Queries with support to arrays and sequence flattening.
  • ✔️ Predicate Queries, singleton arrays and wildcards.
  • ✔️ Functions and Expressions.
  • ✔️ Result Structures.
  • ✔️ Query Composition.
  • ✔️ Sorting, Grouping and Aggregation.
  • Processing Model - Index (seq#$var) and Join (seq@$var) operators are not yet implemented (TODO).
  • Functional Programming - Conditional operator, variables and bindings are implemented, as well as defining custom functions. Function signatures are parsed but not checked yet (TODO). Recursive functions are supported, but additional checks are needed here (TODO). Tail call optimization is not supported. Higher order functions are supported. 'Functions are closures', 'Partial function application' and 'Function chaining' features are supported.
  • Regular Expressions - all is implemented, except for the unusual handling for excessive group indices in $replace() (If N is greater than the number of captured groups, then it is replaced by the empty string) which is not supported by .Net Regex.Replace(). Also match object does not yet have next property (TODO).
  • ✔️ Date/Time Processing - All functions are implemented.
Operators
  • Path Operators:
    • ✔️ . (Map)
    • ✔️ [ ... ] (Filter)
    • ✔️ ^( ... ) (Order-by)
    • ✔️ { ... } (Reduce)
    • ✔️ * (Wildcard)
    • ✔️ ** (Descendants)
    • ✔️ % (Parent) - done, expect for this issue
    • # (Positional variable binding) - (TODO)
    • @ (Context variable binding) - (TODO)
  • ✔️ Numeric Operators - all, including .. (Range) operator.
  • ✔️ Comparison Operators - all, including in (Inclusion) operator.
  • ✔️ Boolean Operators.
  • ✔️ Other Operators:
    • ✔️ & (Concatenation)
    • ✔️ ? : (Conditional)
    • ✔️ := (Variable binding)
    • ✔️ ~> (Chain)
    • ✔️ ... ~> | ... | ... | (Transform)
Function Library
  • ✔️ String Functions:
    • ✔️ Implemented: $string(), $length(), $substring(), $substringBefore(), $substringAfter(), $uppercase(), $lowercase(), $trim(), $pad(), $contains(), $split(), $join(), $match(), $replace(), $eval(), $base64encode(), $base64decode(), $encodeUrlComponent(), $encodeUrl(), $decodeUrlComponent(), $decodeUrl()
    • ✅ There's a discrepancy when handling UTF-16 surrogate pairs. For example $length("\uD834\uDD1E") would return 2, while in original Jsonata-JS it would return 1.
  • ✔️ Numeric Functions:
    • ✔️ Implemented: $number(), $abs(), $floor(), $ceil(), $round(), $power(), sqrt(), $random(), $formatNumber(), $formatBase(), $formatInteger(), $parseInteger()
    • ✅ Using C# custom and standard format strings for picture argument of $formatNumber(), $formatInteger() and $parseInteger() instead of XPath format used in JSonataJS.
  • ✔️ Aggregation Functions:
    • ✔️ Implemented: $sum(), $max(), $min(), $average()
  • ✔️ Boolean Functions:
    • ✔️ Implemented: $boolean(), $not(), $exists()
  • ✔️ Array Functions:
    • ✔️ Implemented: $count(), $append(), $sort(), $reverse(), $shuffle(), $distinct(), $zip()
  • ✔️ Object Functions:
    • ✔️ Implemented: $keys(), $lookup(), $spread(), $merge(), $sift(), $each(), $error(), $assert(), $type()
  • ✔️ Date/Time functions:
    • ✔️ Implemented: $now(), $millis(), $fromMillis(), $toMillis()
    • ✅ Using C# custom and standard format strings for picture argument instead of XPath format used in JSonataJS.
  • ✔️ Higher Order Functions:
    • ✔️ Implemented: $map(), $filter(), $single(), $reduce(), $sift()

Also, need to check all TODO: markers in the code (TODO).

Detailed results for the reference test suite

We use the test suite from original JSONata JS implementation to check consistency and completeness of the port. Current test results for the latest test run are:

  • _all

Full and brief test reports are also in the repo. Below are current states of each test group in the suite:

  • array-constructor
  • blocks
  • boolean-expresssions
  • closures
  • comments
  • comparison-operators
  • conditionals
  • context
  • descendent-operator
  • encoding
  • errors
  • fields
  • flattening
  • function-abs
  • function-append
  • function-applications
  • function-assert
  • function-average
  • function-boolean
  • function-ceil
  • function-contains
  • function-count
  • function-decodeUrl
  • function-decodeUrlComponent
  • function-distinct
  • function-each
  • function-encodeUrl
  • function-encodeUrlComponent
  • function-error
  • function-eval
  • function-exists
  • function-floor
  • function-formatBase
  • function-formatInteger
  • function-formatNumber
  • function-fromMillis
  • function-join
  • function-keys
  • function-length
  • function-lookup
  • function-lowercase
  • function-max
  • function-merge
  • function-number
  • function-pad
  • function-parseInteger
  • function-power
  • function-replace
  • function-reverse
  • function-round
  • function-shuffle
  • function-sift
  • function-signatures
  • function-sort
  • function-split
  • function-spread
  • function-sqrt
  • function-string
  • function-substring
  • function-substringAfter
  • function-substringBefore
  • function-sum
  • function-tomillis
  • function-trim
  • function-typeOf
  • function-uppercase
  • function-zip
  • higher-order-functions
  • hof-filter
  • hof-map
  • hof-reduce
  • hof-single
  • hof-zip-map
  • inclusion-operator
  • joins
  • lambdas
  • literals
  • matchers
  • missing-paths
  • multiple-array-selectors
  • null
  • numeric-operators
  • object-constructor
  • parentheses
  • parent-operator
  • partial-application
  • predicates
  • quoted-selectors
  • range-operator
  • regex
  • simple-array-selectors
  • sorting
  • string-concat
  • tail-recursion
  • token-conversion
  • transform
  • transforms
  • variables
  • wildcards
Product Compatible and additional computed target framework versions.
.NET net5.0 was computed.  net5.0-windows was computed.  net6.0 is compatible.  net6.0-android was computed.  net6.0-ios was computed.  net6.0-maccatalyst was computed.  net6.0-macos was computed.  net6.0-tvos was computed.  net6.0-windows was computed.  net7.0 was computed.  net7.0-android was computed.  net7.0-ios was computed.  net7.0-maccatalyst was computed.  net7.0-macos was computed.  net7.0-tvos was computed.  net7.0-windows was computed.  net8.0 was computed.  net8.0-android was computed.  net8.0-browser was computed.  net8.0-ios was computed.  net8.0-maccatalyst was computed.  net8.0-macos was computed.  net8.0-tvos was computed.  net8.0-windows was computed. 
.NET Core netcoreapp2.0 was computed.  netcoreapp2.1 was computed.  netcoreapp2.2 was computed.  netcoreapp3.0 was computed.  netcoreapp3.1 was computed. 
.NET Standard netstandard2.0 is compatible.  netstandard2.1 was computed. 
.NET Framework net461 was computed.  net462 was computed.  net463 was computed.  net47 is compatible.  net471 was computed.  net472 was computed.  net48 was computed.  net481 was computed. 
MonoAndroid monoandroid was computed. 
MonoMac monomac was computed. 
MonoTouch monotouch was computed. 
Tizen tizen40 was computed.  tizen60 was computed. 
Xamarin.iOS xamarinios was computed. 
Xamarin.Mac xamarinmac was computed. 
Xamarin.TVOS xamarintvos was computed. 
Xamarin.WatchOS xamarinwatchos was computed. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.
  • .NETFramework 4.7

    • No dependencies.
  • .NETStandard 2.0

    • No dependencies.
  • net6.0

    • No dependencies.

NuGet packages (3)

Showing the top 3 NuGet packages that depend on Jsonata.Net.Native:

Package Downloads
Jsonata.Net.Native.JsonNet

Json.Net (Newtonsoft.Json) bindings for Jsonata.Net.Native

Jsonata.Net.Native.SystemTextJson

System.Text.Json bindings for Jsonata.Net.Native

Confluent.SchemaRegistry.Rules

Provides schema rules support.

GitHub repositories

This package is not used by any popular GitHub repositories.

Version Downloads Last updated
2.7.0 12,979 8/4/2024
2.6.1 29,503 2/22/2024
2.6.0 2,647 2/20/2024
2.5.0 397 2/19/2024
2.4.2 37,895 12/19/2023
2.4.1 17,131 10/15/2023
2.4.0 16,348 9/26/2023
2.3.0 2,495 9/10/2023
2.2.1 18,256 7/12/2023
2.2.0 448 7/11/2023
2.1.0 4,019 6/29/2023
2.0.2 2,105 6/20/2023
2.0.1 50,815 11/29/2022
2.0.0 12,374 8/13/2022
1.2.0 389,962 1/20/2022
1.1.0 432 1/18/2022
1.0.0 316 1/7/2022
1.0.0-pre2 202 11/30/2021
1.0.0-pre1 215 10/25/2021