An incredibly flexible and performant JSON parser, generator and formatter in pure Erlang.
Euneus is built on the top of the new OTP json module.
Both encoder and decoder fully conform to RFC 8259 and ECMA 404 standards and are tested using JSONTestSuite.
Detailed examples and further explanation can be found at hexdocs.
OTP >= 24.
The new OTP json
module is incredible and blazing fast!
Unfortunately, it is only available for OTP >= 27. Euneus is available from OTP >= 24.
Also, Euneus simplifies a lot of overheads with the new OTP json
module without
losing any option provided by the json module and keeping its performance.
A simple example comparing the OTP json
module with Euneus decoding object keys:
> json:decode(<<"{\"foo\":\"bar\"}">>, [], #{object_push => fun(K, V, Acc) -> [{binary_to_atom(K), V} | Acc] end}).
{#{foo => <<"bar">>},[],<<>>}
> euneus:decode(<<"{\"foo\":\"bar\"}">>, #{object_keys => atom}).
#{foo => <<"bar">>}
Some reasons to use Euneus for JSON encoding:
-
Possibility to skip values
-
Encoding proplists (proplists are not encoded by the OTP json module)
-
Sort object keys
-
Simple custom encoding via codecs:
-type codec() :: timestamp | datetime | ipv4 | ipv6 | {records, #{Name :: atom() := {Fields :: [atom()], Size :: pos_integer()}}} | codec_fun() | custom_codec().
> euneus:encode({0, 0, 0}, #{codecs => [timestamp]}).
<<"\"1970-01-01T00:00:00.000Z\"">>
> euneus:encode({{1970, 01, 01}, {00, 00, 00}}, #{codecs => [datetime]}).
<<"\"1970-01-01T00:00:00Z\"">>
> euneus:encode({0, 0, 0, 0}, #{codecs => [ipv4]}).
<<"\"0.0.0.0\"">>
> euneus:encode({16#fe80, 0, 0, 0, 16#204, 16#acff, 16#fe17, 16#bf38}, #{codecs => [ipv6]}).
<<"\"fe80::204:acff:fe17:bf38\"">>
% -record(foo, {foo, bar}).
> euneus:encode(#foo{foo = 1, bar = 2}, #{
codecs => [
{records, #{
foo => {record_info(fields, foo), record_info(size, foo)}
}}
]
}).
<<"{\"foo\":1,\"bar\":2}">>
Some reasons to use Euneus for JSON decoding:
-
Faster decoding than the OTP
json
module via some options:#{ array_finish => reversed, object_finish => reversed_proplist % or proplist }
-
The overhead of transforming binary keys to, e.g., atoms
-
Simple custom decoding via codecs:
-type codec() :: copy | timestamp | datetime | ipv4 | ipv6 | pid | port | reference | codec_callback().
> euneus:decode(<<"{\"foo\":\"bar\"}">>).
#{<<"foo">> => <<"bar">>}
Just do a binary copy of the key.
> euneus:decode(<<"{\"foo\":\"bar\"}">>, #{object_keys => copy}).
#{<<"foo">> => <<"bar">>}
> euneus:decode(<<"{\"foo\":\"bar\"}">>, #{object_keys => atom}).
#{foo => <<"bar">>}
> euneus:decode(<<"{\"foo\":\"bar\"}">>, #{object_keys => existing_atom}).
#{foo => <<"bar">>}
Just do a binary copy of the value.
> euneus:decode(<<"\"foo\"">>, #{codecs => [copy]}).
<<"foo">>
> euneus:decode(<<"\"1970-01-01T00:00:00.000Z\"">>, #{codecs => [timestamp]}).
{0,0,0}
> euneus:decode(<<"\"1970-01-01T00:00:00Z\"">>, #{codecs => [datetime]}).
{{1970,1,1},{0,0,0}}
> euneus:decode(<<"\"0.0.0.0\"">>, #{codecs => [ipv4]}).
{0,0,0,0}
> euneus:decode(<<"\"::\"">>, #{codecs => [ipv6]}).
{0,0,0,0,0,0,0,0}
> euneus:decode(<<"\"::1\"">>, #{codecs => [ipv6]}).
{0,0,0,0,0,0,0,1}
> euneus:decode(<<"\"::192.168.42.2\"">>, #{codecs => [ipv6]}).
{0,0,0,0,0,0,49320,10754}
> euneus:decode(<<"\"fe80::204:acff:fe17:bf38\"">>, #{codecs => [ipv6]}).
{65152,0,0,0,516,44287,65047,48952}
> euneus:decode(<<"\"<0.92.0>\"">>, #{codecs => [pid]}).
<0.92.0>
> euneus:decode(<<"\"#Port<0.1>\"">>, #{codecs => [port]}).
#Port<0.1>
> euneus:decode(<<"\"#Ref<0.314572725.1088159747.110918>\"">>, #{codecs => [reference]}).
#Ref<0.314572725.1088159747.110918>
% rebar.config
{deps, [
{json_polyfill, "~> 0.2"}, % Required only for OTP < 27
{euneus, "~> 2.4"}
]}.
# mix.exs
defp deps do
[
{:json_polyfill, "~> 0.2"}, # Required only for OTP < 27
{:euneus, "~> 2.4"}
]
end
1> euneus:encode(#{age => 68, name => <<"Joe Armstrong">>, nationality => <<"British">>}).
<<"{\"name\":\"Joe Armstrong\",\"age\":68,\"nationality\":\"British\"}">>
2> euneus:decode(v(1)).
#{<<"age">> => 68,<<"name">> => <<"Joe Armstrong">>,<<"nationality">> => <<"British">>}
The functions euneus:encode/1
euneus:encode/2
encodes an Erlang term into a binary JSON.
The second argument of euneus:encode/2
are options.
Please see the m:euneus_encoder
documentation
for more examples and detailed explanation.
The data mapping and error reasons can be found in the OTP json encode function documentation.
The functions euneus:decode/1
and euneus:decode/2
decodes a binary JSON into an Erlang term.
The second argument of euneus:decode/2
are options.
Please see the m:euneus_decoder
documentation
for more examples and detailed explanation.
The data mapping and error reasons can be found in the OTP json decode function documentation.
Three functions provide JSON decode streaming:
euneus:decode_stream_start/1
- Equivalent toeuneus:decode_stream_start(JSON, #{})
;euneus:decode_stream_start/2
- Begin parsing a stream of bytes of a JSON value;euneus:decode_stream_continue/2
- Continue parsing a stream of bytes of a JSON value.
Please see the m:euneus_decoder
documentation
for more examples and detailed explanation.
Two functions provide JSON formatting:
euneus:minify/1
- Removes any extra spaces and new line characters;euneus:format/2
- Formats the JSON by passing custom options.
Please see the m:euneus_formatter
documentation
for more examples and detailed explanation.
The benchmarks are implemented very simply, but they are a good start foroptimizing
Euneus since no optimizations have been made. You will find the benchmark commands
in euneus_benchmarker
, and data and tests under the test folder.
Important
For the first benchmark run, bootstrapping erlperf
is required:
$ rebar3 as benchmark shell
1> euneus_benchmarker:bootstrap().
===> Verifying dependencies...
===> Analyzing applications...
===> Compiling erlperf
===> Building escript for erlperf...
ok
Setup:
- OS : Linux
- CPU: 12th Gen Intel(R) Core(TM) i9-12900HX
- VM : Erlang/OTP 27 [erts-15.0.1] [source] [64-bit] [smp:24:24] [ds:24:24:10] [async-threads:1] [jit:ns]
$ rebar3 as benchmark shell
1> euneus_benchmarker:encode_benchmark().
Code || Samples Avg StdDev Median P99 Iteration Rel
jiffy 1 3 26 36.69% 25 36 38474 us 100%
euneus 1 3 20 38.20% 18 29 49197 us 78%
thoas 1 3 10 36.06% 9 14 100 ms 38%
2> euneus_benchmarker:decode_benchmark().
Code || Samples Avg StdDev Median P99 Iteration Rel
euneus 1 3 24 2.44% 24 24 42268 us 100%
jiffy 1 3 19 3.09% 19 19 53589 us 79%
thoas 1 3 14 0.00% 14 14 71452 us 59%
If you like this tool, please consider sponsoring me. I'm thankful for your never-ending support ❤️
I also accept coffees ☕
Copyright (c) 2024 William Fank Thomé
Euneus is 100% open-source and community-driven. All components are available under the Apache 2 License on GitHub.
See LICENSE.md for more information.