# ljson
[![Build Status][gh-actions-badge]][gh-actions]
[![LFE Versions][lfe-badge]][lfe]
[![Erlang Versions][erlang-badge]][versions]
[![Tags][github-tags-badge]][github-tags]
[![Downloads][hex-downloads]][hex-package]
*A wrapper for `json` in the Erlang stdlib for 27+ and for `jsx` in Erlang 26 and below*
[![Project Logo][logo]][logo-large]
## About
ljson provides a very thin wrapper around the `encode/1`, `encode/2` functions of the jsx library when running in OTP versions 26 and below; it provides a similar wrapper for `encode/1`, `encode/3` when running in OTP 27 and above.
In both cases, binary is returned. Since Erlang 27's `json` returns an iolist, this does add a step for Erlang 27+ and breaks a bit of that compatibility in the interest of preserving more backwards compatibility for projects that have used jsx.
How long will this library be around? Until there's a better library easily usable from LFE projects, or until all supported Erlang versions have `json` and provide a backwards-compatible feature set, this library will be around.
## Core Usage
The following usage examples are from LFE, but the same applies to Erlang (though hyphens in the LFE function name can be replaced with underscores in the same Erlang function).
Encode simple LFE data to JSON:
```cl
lfe> (ljson:encode 'a)
#"\"a\""
lfe> (ljson:encode "a")
#"[97]"
lfe> (ljson:encode 1)
#"1"
lfe> (ljson:encode 3.14)
#"3.14"
lfe> (ljson:encode '(a b c 42))
#"[\"a\",\"b\",\"c\",42]"
lfe> (ljson:encode #m(a b))
#"{\"a\":\"b\"}"
lfe> (ljson:encode #m(a b c d))
#"{\"c\":\"d\",\"a\":\"b\"}"
```
Decode simple JSON:
```cl
lfe> (ljson:decode #"\"a\""))
#"a"
lfe> (ljson:decode #b("[97]"))
"a"
lfe> (ljson:decode #b("1"))
1
lfe> (ljson:decode #b("3.14"))
3.14
lfe> (ljson:decode #b("[\"a\",\"b\",\"c\",42]"))
(#"a" #"b" #"c" 42)
lfe> (ljson:decode #"{\"a\": \"b\"}")
#M(#"a" #"b")
lfe> (ljson:decode #"{\"a\":\"b\",\"c\":\"d\"}")
#M(#"a" #"b" #"c" #"d")
lfe> (ljson:decode
lfe> #B(123 34 97 34 58 34 98 34 44 34 99 34 58 34 100 34 125))
#M(#"a" #"b" #"c" #"d")
```
Decode a JSON data structure (note that, for formatting purposes, the data
below has been presented separated with newlines; this won't work in the
LFE REPL -- you'll need to put it all on one line):
```cl
lfe> (set json-data #"{
\"First Name\": \"Jón\",
\"Last Name\": \"Þórson\",
\"Is Alive?\": true,
\"Age\": 25,
\"Height_cm\": 167.6,
\"Address\": {
\"Street Address\": \"í Gongini 5 Postsmoga 108\",
\"City\": \"Tórshavn\",
\"Country\": \"Faroe Islands\",
\"Postal Code\": \"100\"
},
\"Phone Numbers\": [
{
\"Type\": \"home\",
\"Number\": \"20 60 30\"
},
{
\"Type\": \"office\",
\"Number\": \"+298 20 60 20\"
}
],
\"Children\": [],
\"Spouse\": null}")
lfe> (set data (ljson:decode json-data))
#M(#"Address"
#M(#"City" #"Tórshavn" #"Country" #"Faroe Islands"
#"Postal Code" #"100"
#"Street Address" #"í Gongini 5 Postsmoga 108")
#"Age" 25 #"Children" () #"First Name" #"Jón" #"Height_cm" 167.6
#"Is Alive?" true #"Last Name" #"Þórson"
#"Phone Numbers"
(#M(#"Number" #"20 60 30" #"Type" #"home")
#M(#"Number" #"+298 20 60 20" #"Type" #"office"))
#"Spouse" null)
```
Now let's take it full circle by encoding it again:
```cl
lfe> (ljson:encode data)
#"{\"Address\":{\"City\":\"Tórshavn\",\"Country\":\"Faroe Islands\",\"Postal Code\":\"100\",\"Street Address\":\"í Gongini 5 Postsmoga 108\"},\"Age\":25,\"Children\":[],\"First Name\":\"Jón\",\"Height_cm\":167.6,\"Is Alive?\":true,\"Last Name\":\"Þórson\",\"Phone Numbers\":[{\"Number\":\"20 60 30\",\"Type\":\"home\"},{\"Number\":\"+298 20 60 20\",\"Type\":\"office\"}],\"Spouse\":null}"
```
Let's do the same, but this time from LFE data:
```cl
lfe> (set lfe-data
'#m(#"First Name" #"Jón"
#"Last Name" #"Þórson"
#"Is Alive?" true
#"Age" 25
#"Height_cm" 167.6
#"Address" #m(#"Street Address" #"í Gongini 5 Postsmoga 108"
#"City" #"Tórshavn"
#"Country" #"Faroe Islands"
#"Postal Code" #"100")
#"Phone Numbers" (#m(#"Type" #"home" #"Number" #"20 60 30")
#m(#"Type" #"office" #"Number" #"+298 20 60 20"))
#"Children" ()
#"Spouse" null))
(#(#B(...)))
lfe> (ljson:encode lfe-data)
#"{\"Address\":{\"City\":\"Tórshavn\",\"Country\":\"Faroe Islands\",\"Postal Code\":\"100\",\"Street Address\":\"í Gongini 5 Postsmoga 108\"},\"Age\":25,\"Children\":[],\"First Name\":\"Jón\",\"Height_cm\":167.6,\"Is Alive?\":true,\"Last Name\":\"Þórson\",\"Phone Numbers\":[{\"Number\":\"20 60 30\",\"Type\":\"home\"},{\"Number\":\"+298 20 60 20\",\"Type\":\"office\"}],\"Spouse\":null}"
```
Note that the additional function arguments (`encode/2`, `decode/2`, and `decode/3`) are specific to the underlying JSON library being used. The args will have the same meaning as the parent library used, with no attempt made by ljson to unify these for consistency.
## Working with Files
Write Erlang terms as serialised JSON data to a file:
``` cl
lfe> (ljson-file:write "/tmp/test-data.json" #m(a 1 b 2 c 3))
"/tmp/test-data.json"
```
Read a JSON file, deserialising the data as Erlang terms:
``` cl
lfe> (ljson-file:read "/tmp/test-data.json")
#M(#"a" 1 #"b" 2 #"c" 3)
```
### Helpers for Standard System Directories
The Erlang/LFE project [dirs](https://github.com/lfex/dirs) (a port of the same Rust library)) defines standard/convential directories by in Linux, macos, and Windows operating systems. The ljson project utilises this library to support convenient reading and writing of application data files, etc.
Write a JSON file:
``` cl
lfe> (ljson-file:write 'data "myapp/conponent/state.json" #m(a 1 b 2 c 3))
"/Users/oubiwann/Library/Application Support/myapp/conponent/state.json"
```
Read a JSON file:
``` cl
lfe> (ljson-file:read 'data "myapp/conponent/state.json")
#M(#"a" 1 #"b" 2 #"c" 3)
```
## License
Apache Version 2 License
Copyright © 2014-2015, Dreki Þórgísl
Copyright © 2015, arpunk
Copyright © 2015-2025, Duncan McGreggor <oubiwann@gmail.com>
[//]: ---Named-Links---
[logo]: priv/images/jason-argonauts-small.png
[logo-large]: http://dropr.com/coenhamelink/15218/jason_and_the_argonauts/+?p=97582
[org]: https://github.com/lfex
[github]: https://github.com/lfex/ljson
[gitlab]: https://gitlab.com/lfex/ljson
[gh-actions-badge]: https://github.com/lfex/ljson/workflows/ci%2Fcd/badge.svg
[gh-actions]: https://github.com/lfex/ljson/actions
[lfe]: https://github.com/rvirding/lfe
[lfe-badge]: https://img.shields.io/badge/lfe-2.1+-blue.svg
[erlang-badge]: https://img.shields.io/badge/erlang-21%20to%2027-blue.svg
[versions]: https://github.com/lfex/ljson/blob/master/.travis.yml
[github-tags]: https://github.com/lfex/ljson/tags
[github-tags-badge]: https://img.shields.io/github/tag/lfex/ljson.svg
[hex-badge]: https://img.shields.io/hexpm/v/ljson.svg?maxAge=2592000
[hex-package]: https://hex.pm/packages/ljson
[hex-downloads]: https://img.shields.io/hexpm/dt/ljson.svg