| 
									
										
										
										
											2024-08-13 09:09:38 -07:00
										 |  |  | """Command-line tool to validate and pretty-print JSON
 | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-13 09:09:38 -07:00
										 |  |  | See `json.__main__` for a usage example (invocation as | 
					
						
							|  |  |  | `python -m json.tool` is supported for backwards compatibility). | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | """
 | 
					
						
							| 
									
										
										
										
											2014-03-21 23:17:29 -05:00
										 |  |  | import argparse | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | import json | 
					
						
							| 
									
										
										
										
											2025-04-19 20:11:21 +02:00
										 |  |  | import re | 
					
						
							| 
									
										
										
										
											2014-03-21 23:17:29 -05:00
										 |  |  | import sys | 
					
						
							| 
									
										
										
										
											2025-05-05 23:45:25 +02:00
										 |  |  | from _colorize import get_theme, can_colorize | 
					
						
							| 
									
										
										
										
											2025-04-19 20:11:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # The string we are colorizing is valid JSON, | 
					
						
							|  |  |  | # so we can use a looser but simpler regex to match | 
					
						
							|  |  |  | # the various parts, most notably strings and numbers, | 
					
						
							|  |  |  | # where the regex given by the spec is much more complex. | 
					
						
							|  |  |  | _color_pattern = re.compile(r'''
 | 
					
						
							|  |  |  |     (?P<key>"(\\.|[^"\\])*")(?=:)           | | 
					
						
							|  |  |  |     (?P<string>"(\\.|[^"\\])*")             | | 
					
						
							| 
									
										
										
										
											2025-05-05 23:45:25 +02:00
										 |  |  |     (?P<number>NaN|-?Infinity|[0-9\-+.Ee]+) | | 
					
						
							| 
									
										
										
										
											2025-04-19 20:11:21 +02:00
										 |  |  |     (?P<boolean>true|false)                 | | 
					
						
							|  |  |  |     (?P<null>null) | 
					
						
							|  |  |  | ''', re.VERBOSE)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-05 23:45:25 +02:00
										 |  |  | _group_to_theme_color = { | 
					
						
							|  |  |  |     "key": "definition", | 
					
						
							|  |  |  |     "string": "string", | 
					
						
							|  |  |  |     "number": "number", | 
					
						
							|  |  |  |     "boolean": "keyword", | 
					
						
							|  |  |  |     "null": "keyword", | 
					
						
							| 
									
										
										
										
											2025-04-19 20:11:21 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-05 23:45:25 +02:00
										 |  |  | def _colorize_json(json_str, theme): | 
					
						
							|  |  |  |     def _replace_match_callback(match): | 
					
						
							|  |  |  |         for group, color in _group_to_theme_color.items(): | 
					
						
							|  |  |  |             if m := match.group(group): | 
					
						
							|  |  |  |                 return f"{theme[color]}{m}{theme.reset}" | 
					
						
							|  |  |  |         return match.group() | 
					
						
							| 
									
										
										
										
											2025-04-19 20:11:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return re.sub(_color_pattern, _replace_match_callback, json_str) | 
					
						
							| 
									
										
										
										
											2014-03-21 23:17:29 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | def main(): | 
					
						
							| 
									
										
										
										
											2014-03-21 23:17:29 -05:00
										 |  |  |     description = ('A simple command line interface for json module ' | 
					
						
							|  |  |  |                    'to validate and pretty-print JSON objects.') | 
					
						
							| 
									
										
										
										
											2025-05-05 20:46:46 +03:00
										 |  |  |     parser = argparse.ArgumentParser(description=description, color=True) | 
					
						
							| 
									
										
										
										
											2019-12-04 18:39:31 +09:00
										 |  |  |     parser.add_argument('infile', nargs='?', | 
					
						
							| 
									
										
										
										
											2025-08-22 16:44:25 +08:00
										 |  |  |                         help='a JSON file to be validated or pretty-printed; ' | 
					
						
							|  |  |  |                              'defaults to stdin', | 
					
						
							| 
									
										
										
										
											2024-01-10 15:07:19 +02:00
										 |  |  |                         default='-') | 
					
						
							| 
									
										
										
										
											2019-12-04 18:39:31 +09:00
										 |  |  |     parser.add_argument('outfile', nargs='?', | 
					
						
							| 
									
										
										
										
											2019-05-14 18:52:42 +02:00
										 |  |  |                         help='write the output of infile to outfile', | 
					
						
							| 
									
										
										
										
											2021-11-06 18:11:35 +01:00
										 |  |  |                         default=None) | 
					
						
							| 
									
										
										
										
											2014-11-10 09:56:54 +02:00
										 |  |  |     parser.add_argument('--sort-keys', action='store_true', default=False, | 
					
						
							|  |  |  |                         help='sort the output of dictionaries alphabetically by key') | 
					
						
							| 
									
										
										
										
											2019-12-06 00:44:01 -06:00
										 |  |  |     parser.add_argument('--no-ensure-ascii', dest='ensure_ascii', action='store_false', | 
					
						
							|  |  |  |                         help='disable escaping of non-ASCII characters') | 
					
						
							| 
									
										
										
										
											2018-11-07 18:09:32 +08:00
										 |  |  |     parser.add_argument('--json-lines', action='store_true', default=False, | 
					
						
							| 
									
										
										
										
											2019-12-07 07:14:40 -07:00
										 |  |  |                         help='parse input using the JSON Lines format. ' | 
					
						
							|  |  |  |                         'Use with --no-indent or --compact to produce valid JSON Lines output.') | 
					
						
							| 
									
										
										
										
											2019-12-04 01:15:19 -05:00
										 |  |  |     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)') | 
					
						
							| 
									
										
										
										
											2014-03-21 23:17:29 -05:00
										 |  |  |     options = parser.parse_args() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-04 01:15:19 -05:00
										 |  |  |     dump_args = { | 
					
						
							|  |  |  |         'sort_keys': options.sort_keys, | 
					
						
							|  |  |  |         'indent': options.indent, | 
					
						
							| 
									
										
										
										
											2019-12-06 00:44:01 -06:00
										 |  |  |         'ensure_ascii': options.ensure_ascii, | 
					
						
							| 
									
										
										
										
											2019-12-04 01:15:19 -05:00
										 |  |  |     } | 
					
						
							|  |  |  |     if options.compact: | 
					
						
							|  |  |  |         dump_args['indent'] = None | 
					
						
							|  |  |  |         dump_args['separators'] = ',', ':' | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-10 15:07:19 +02:00
										 |  |  |     try: | 
					
						
							|  |  |  |         if options.infile == '-': | 
					
						
							|  |  |  |             infile = sys.stdin | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             infile = open(options.infile, encoding='utf-8') | 
					
						
							| 
									
										
										
										
											2012-11-29 02:15:18 +02:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2019-12-04 01:15:19 -05:00
										 |  |  |             if options.json_lines: | 
					
						
							| 
									
										
										
										
											2018-11-07 18:09:32 +08:00
										 |  |  |                 objs = (json.loads(line) for line in infile) | 
					
						
							|  |  |  |             else: | 
					
						
							| 
									
										
										
										
											2021-11-06 18:11:35 +01:00
										 |  |  |                 objs = (json.load(infile),) | 
					
						
							| 
									
										
										
										
											2024-01-10 15:07:19 +02:00
										 |  |  |         finally: | 
					
						
							|  |  |  |             if infile is not sys.stdin: | 
					
						
							|  |  |  |                 infile.close() | 
					
						
							| 
									
										
										
										
											2021-11-06 18:11:35 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-10 15:07:19 +02:00
										 |  |  |         if options.outfile is None: | 
					
						
							|  |  |  |             outfile = sys.stdout | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             outfile = open(options.outfile, 'w', encoding='utf-8') | 
					
						
							|  |  |  |         with outfile: | 
					
						
							| 
									
										
										
										
											2025-05-05 23:45:25 +02:00
										 |  |  |             if can_colorize(file=outfile): | 
					
						
							|  |  |  |                 t = get_theme(tty_file=outfile).syntax | 
					
						
							|  |  |  |                 for obj in objs: | 
					
						
							| 
									
										
										
										
											2025-04-19 20:11:21 +02:00
										 |  |  |                     json_str = json.dumps(obj, **dump_args) | 
					
						
							| 
									
										
										
										
											2025-05-05 23:45:25 +02:00
										 |  |  |                     outfile.write(_colorize_json(json_str, t)) | 
					
						
							|  |  |  |                     outfile.write('\n') | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 for obj in objs: | 
					
						
							| 
									
										
										
										
											2025-04-19 20:11:21 +02:00
										 |  |  |                     json.dump(obj, outfile, **dump_args) | 
					
						
							| 
									
										
										
										
											2025-05-05 23:45:25 +02:00
										 |  |  |                     outfile.write('\n') | 
					
						
							| 
									
										
										
										
											2024-01-10 15:07:19 +02:00
										 |  |  |     except ValueError as e: | 
					
						
							|  |  |  |         raise SystemExit(e) | 
					
						
							| 
									
										
										
										
											2008-05-08 14:29:10 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if __name__ == '__main__': | 
					
						
							| 
									
										
										
										
											2020-03-10 16:41:44 +09:00
										 |  |  |     try: | 
					
						
							|  |  |  |         main() | 
					
						
							|  |  |  |     except BrokenPipeError as exc: | 
					
						
							| 
									
										
										
										
											2024-08-13 09:09:38 -07:00
										 |  |  |         raise SystemExit(exc.errno) |