Why Sponsor Oils? | source | all docs for version 0.21.0 | all versions | oilshell.org
JSON is used by both web services and command line tools, so a modern Unix shell needs to support it.
Oils has a json
builtin which you can use from bot OSH and YSH.
It also has a parallel json8
builtin with the same uage. See J8
Notation for details on the encoding.
The json
builtin has read
and write
subcommands, which convert
between serialized data languages and in-memory data structures.
YSH data structures are like those in Python and JavaScript, so this correspondence is natural.
json read
parses from stdin
Usage:
json read (PLACE?)
json8 read (PLACE?)
Examples:
$ cat stats.json
{"count": 42}
# Read from a file. By default, the variable _reply is written.
$ json read < stats.json
# Use = to pretty print an expression
$ = _reply
(Dict) {'count': 42}
Specify a place to put the data:
$ json read (&myvar) < stats.json
$ = myvar
(Dict) {'count': 42}
Use it in a pipeline:
# 'json read' is valid at the end of a pipeline (because YSH implements
# shopt -s lastpipe)
$ echo '{"count": 42}' | json read (&myvar)
Failure with invalid input data:
$ echo '[ "incomplete"' | json read (&myvar) < invalid.json
[ "incomplete"
^
json read: premature EOF
$ echo $?
1
json write
prints to stdout
Usage:
json write FLAGS* (EXPR)
EXPR is an expression that evaluates to a serializable object.
Flags:
--indent=2 Indentation size
--pretty=true Whether to add newlines for readability
Examples:
# Create a Dict. As in JavaScript, keys don't require quotes.
$ var d = {name: "bob", age: 42}
# Print the Dict as JSON. By default, newlines are added for readability,
# with 2 space indentation.
$ json write (d)
{
"name": "bob",
"count": 42
}
$ json write --indent 4 (d)
{
"name": "bob",
"count": 42
}
$ json write --pretty=F (d)
{"name": "bob", "count": 42}
Notes:
--indent
is ignored if --pretty
is false.write
builtinTODO
write --j8 hello there
write --json hello there # unicode replacement char
Once your data is deserialized, you can use YSH expression to operate on it.
$ echo '{"counts": [42, 99]}' | json read (&d)
$ = d['counts']
(List) [42, 99]
$ = d['counts'][1]
(Int) 99
# d->counts is a synonym for d["counts"]
$ json write (d->counts)
[
42,
99
]
Note: It may more efficient to filter large data structures with tools like
jq
first.
YSH arrays and shell arrays both serialize to a list of strings:
$ declare sharray=( foo.txt *.py )
$ json write (sharray)
[
"foo.txt",
"one.py",
"two.py"
]
$ var oilarray = :| foo.txt *.py |
$ json write (oilarray)
[
"foo.txt",
"one.py",
"two.py"
]
Bash-style associative arrays are printed like Dict[Str, Str]
:
$ declare -A assoc=(["key"]=value)
$ json write (assoc)
{
"key": "value"
}