You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
86 lines
3.3 KiB
86 lines
3.3 KiB
2 years ago
|
r"""Command-line tool to validate and pretty-print JSON
|
||
|
|
||
|
Usage::
|
||
|
|
||
|
$ echo '{"json":"obj"}' | python -m json.tool
|
||
|
{
|
||
|
"json": "obj"
|
||
|
}
|
||
|
$ echo '{ 1.2:3.4}' | python -m json.tool
|
||
|
Expecting property name enclosed in double quotes: line 1 column 3 (char 2)
|
||
|
|
||
|
"""
|
||
|
import argparse
|
||
|
import json
|
||
|
import sys
|
||
|
from pathlib import Path
|
||
|
|
||
|
|
||
|
def main():
|
||
|
prog = 'python -m json.tool'
|
||
|
description = ('A simple command line interface for json module '
|
||
|
'to validate and pretty-print JSON objects.')
|
||
|
parser = argparse.ArgumentParser(prog=prog, description=description)
|
||
|
parser.add_argument('infile', nargs='?',
|
||
|
type=argparse.FileType(encoding="utf-8"),
|
||
|
help='a JSON file to be validated or pretty-printed',
|
||
|
default=sys.stdin)
|
||
|
parser.add_argument('outfile', nargs='?',
|
||
|
type=Path,
|
||
|
help='write the output of infile to outfile',
|
||
|
default=None)
|
||
|
parser.add_argument('--sort-keys', action='store_true', default=False,
|
||
|
help='sort the output of dictionaries alphabetically by key')
|
||
|
parser.add_argument('--no-ensure-ascii', dest='ensure_ascii', action='store_false',
|
||
|
help='disable escaping of non-ASCII characters')
|
||
|
parser.add_argument('--json-lines', action='store_true', default=False,
|
||
|
help='parse input using the JSON Lines format. '
|
||
|
'Use with --no-indent or --compact to produce valid JSON Lines output.')
|
||
|
group = parser.add_mutually_exclusive_group()
|
||
|
group.add_argument('--indent', default=4, type=int,
|
||
|
help='separate items with newlines and use this number '
|
||
|
'of spaces for indentation')
|
||
|
group.add_argument('--tab', action='store_const', dest='indent',
|
||
|
const='\t', help='separate items with newlines and use '
|
||
|
'tabs for indentation')
|
||
|
group.add_argument('--no-indent', action='store_const', dest='indent',
|
||
|
const=None,
|
||
|
help='separate items with spaces rather than newlines')
|
||
|
group.add_argument('--compact', action='store_true',
|
||
|
help='suppress all whitespace separation (most compact)')
|
||
|
options = parser.parse_args()
|
||
|
|
||
|
dump_args = {
|
||
|
'sort_keys': options.sort_keys,
|
||
|
'indent': options.indent,
|
||
|
'ensure_ascii': options.ensure_ascii,
|
||
|
}
|
||
|
if options.compact:
|
||
|
dump_args['indent'] = None
|
||
|
dump_args['separators'] = ',', ':'
|
||
|
|
||
|
with options.infile as infile:
|
||
|
try:
|
||
|
if options.json_lines:
|
||
|
objs = (json.loads(line) for line in infile)
|
||
|
else:
|
||
|
objs = (json.load(infile),)
|
||
|
|
||
|
if options.outfile is None:
|
||
|
out = sys.stdout
|
||
|
else:
|
||
|
out = options.outfile.open('w', encoding='utf-8')
|
||
|
with out as outfile:
|
||
|
for obj in objs:
|
||
|
json.dump(obj, outfile, **dump_args)
|
||
|
outfile.write('\n')
|
||
|
except ValueError as e:
|
||
|
raise SystemExit(e)
|
||
|
|
||
|
|
||
|
if __name__ == '__main__':
|
||
|
try:
|
||
|
main()
|
||
|
except BrokenPipeError as exc:
|
||
|
sys.exit(exc.errno)
|