forked from Dokucraft/Dungeons-Level-Format
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpretty_compact_json.py
85 lines (65 loc) · 3.22 KB
/
pretty_compact_json.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
# The MIT License (MIT)
# Copyright (c) 2014, 2016, 2017, 2019 Simon Lydell
# Permission is hereby granted, free of charge, to any person obtaining a copy
# of this software and associated documentation files (the "Software"), to deal
# in the Software without restriction, including without limitation the rights
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
# copies of the Software, and to permit persons to whom the Software is
# furnished to do so, subject to the following conditions:
# The above copyright notice and this permission notice shall be included in
# all copies or substantial portions of the Software.
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
# THE SOFTWARE.
# Based on https://github.com/lydell/json-stringify-pretty-compact/blob/master/index.js
# Ported to Python with some modifications by CCCode
import re
import json
start_or_end = r'([{\[])|([}\]])'
def stringify(passedObj, indent=2, max_length=80):
# The original code supports a replacer function/array, but for now it's
# not implemented here.
indent_str = ' ' * indent
if indent == 0:
max_length = float('inf')
def _stringify(obj, current_indent, reserved, has_key):
# The original code checks if obj has a toJSON function here and uses that
# to convert obj to JSON before continuing. It's not really necessary for
# what I use this function for, so I decided to just remove it.
string = json.dumps(obj, separators=(', ', ': '))
length = max_length - len(current_indent) - reserved
if len(string) <= length:
prettified = re.sub(start_or_end, r'\1 \2', string)
if len(prettified) <= length:
return prettified
if isinstance(obj, dict) or isinstance(obj, list):
next_indent = current_indent + indent_str
items = []
if isinstance(obj, dict):
start = '{'
end = '}'
keys = list(obj.keys())
length = len(keys)
for index in range(length):
key = keys[index]
key_part = json.dumps(key, separators=(',', ':')) + ': '
value = _stringify(obj[key], next_indent, len(key_part) + (0 if index == length - 1 else 1), True)
if not value is None:
items.append(key_part + value)
else: # obj is list
start = '['
end = ']'
length = len(obj)
for index in range(length):
items.append(_stringify(obj[index], next_indent, 0 if index == length - 1 else 1, False) or 'null')
if len(items) > 0:
if has_key or len(items) == 1:
return f'\n{current_indent}'.join([start, indent_str + f',\n{next_indent}'.join(items), end])
else:
return f'\n{current_indent}'.join([start + indent_str[1:] + items[0] + ',', indent_str + f',\n{next_indent}'.join(items[1:]), end])
return string
return _stringify(passedObj, '', 0, False)