json-repair 0.50.1__tar.gz → 0.51.0__tar.gz
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- {json_repair-0.50.1/src/json_repair.egg-info → json_repair-0.51.0}/PKG-INFO +2 -1
- {json_repair-0.50.1 → json_repair-0.51.0}/README.md +1 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/pyproject.toml +7 -2
- {json_repair-0.50.1 → json_repair-0.51.0}/src/json_repair/parse_object.py +22 -1
- {json_repair-0.50.1 → json_repair-0.51.0/src/json_repair.egg-info}/PKG-INFO +2 -1
- {json_repair-0.50.1 → json_repair-0.51.0}/tests/test_parse_object.py +14 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/tests/test_performance.py +11 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/LICENSE +0 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/setup.cfg +0 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/src/json_repair/__init__.py +0 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/src/json_repair/__main__.py +0 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/src/json_repair/constants.py +0 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/src/json_repair/json_context.py +0 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/src/json_repair/json_parser.py +0 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/src/json_repair/json_repair.py +0 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/src/json_repair/object_comparer.py +0 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/src/json_repair/parse_array.py +0 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/src/json_repair/parse_boolean_or_null.py +0 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/src/json_repair/parse_comment.py +0 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/src/json_repair/parse_number.py +0 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/src/json_repair/parse_string.py +0 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/src/json_repair/py.typed +0 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/src/json_repair/string_file_wrapper.py +0 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/src/json_repair.egg-info/SOURCES.txt +0 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/src/json_repair.egg-info/dependency_links.txt +0 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/src/json_repair.egg-info/entry_points.txt +0 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/src/json_repair.egg-info/top_level.txt +0 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/tests/test_json_repair.py +0 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/tests/test_parse_array.py +0 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/tests/test_parse_boolean_or_null.py +0 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/tests/test_parse_comment.py +0 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/tests/test_parse_number.py +0 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/tests/test_parse_string.py +0 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/tests/test_repair_json_cli.py +0 -0
- {json_repair-0.50.1 → json_repair-0.51.0}/tests/test_repair_json_from_file.py +0 -0
@@ -1,6 +1,6 @@
|
|
1
1
|
Metadata-Version: 2.4
|
2
2
|
Name: json_repair
|
3
|
-
Version: 0.
|
3
|
+
Version: 0.51.0
|
4
4
|
Summary: A package to repair broken json strings
|
5
5
|
Author-email: Stefano Baccianella <4247706+mangiucugna@users.noreply.github.com>
|
6
6
|
License-Expression: MIT
|
@@ -269,6 +269,7 @@ You will need owner access to this repository
|
|
269
269
|
- Ruby: https://github.com/sashazykov/json-repair-rb
|
270
270
|
- Rust: https://github.com/oramasearch/llm_json
|
271
271
|
- R: https://github.com/cgxjdzz/jsonRepair
|
272
|
+
- Java: https://github.com/du00cs/json-repairj
|
272
273
|
---
|
273
274
|
## Star History
|
274
275
|
|
@@ -252,6 +252,7 @@ You will need owner access to this repository
|
|
252
252
|
- Ruby: https://github.com/sashazykov/json-repair-rb
|
253
253
|
- Rust: https://github.com/oramasearch/llm_json
|
254
254
|
- R: https://github.com/cgxjdzz/jsonRepair
|
255
|
+
- Java: https://github.com/du00cs/json-repairj
|
255
256
|
---
|
256
257
|
## Star History
|
257
258
|
|
@@ -3,7 +3,7 @@ requires = ["setuptools>=61.0"]
|
|
3
3
|
build-backend = "setuptools.build_meta"
|
4
4
|
[project]
|
5
5
|
name = "json_repair"
|
6
|
-
version = "0.
|
6
|
+
version = "0.51.0"
|
7
7
|
license = "MIT"
|
8
8
|
license-files = ["LICENSE"]
|
9
9
|
authors = [
|
@@ -26,12 +26,17 @@ pythonpath = [
|
|
26
26
|
"."
|
27
27
|
]
|
28
28
|
[tool.coverage.run]
|
29
|
+
source = ["src"]
|
29
30
|
omit = [
|
30
31
|
"*/.cursor/extensions/*",
|
31
32
|
"*/pythonFiles/lib/python/*",
|
32
33
|
"*/site-packages/*",
|
34
|
+
"src/json_repair/__main__.py",
|
33
35
|
]
|
34
36
|
[tool.coverage.report]
|
37
|
+
include = [
|
38
|
+
"src/json_repair/*",
|
39
|
+
]
|
35
40
|
exclude_also = [
|
36
41
|
'def __repr__',
|
37
42
|
'if self.debug:',
|
@@ -111,4 +116,4 @@ line-ending = "auto"
|
|
111
116
|
|
112
117
|
[tool.ruff.lint.per-file-ignores]
|
113
118
|
# Explicit re-exports is fine in __init__.py, still a code smell elsewhere.
|
114
|
-
"__init__.py" = ["PLC0414"]
|
119
|
+
"__init__.py" = ["PLC0414"]
|
@@ -1,6 +1,6 @@
|
|
1
1
|
from typing import TYPE_CHECKING
|
2
2
|
|
3
|
-
from .constants import JSONReturnType
|
3
|
+
from .constants import STRING_DELIMITERS, JSONReturnType
|
4
4
|
from .json_context import ContextValues
|
5
5
|
|
6
6
|
if TYPE_CHECKING:
|
@@ -112,4 +112,25 @@ def parse_object(self: "JSONParser") -> dict[str, JSONReturnType]:
|
|
112
112
|
self.skip_whitespaces_at()
|
113
113
|
|
114
114
|
self.index += 1
|
115
|
+
|
116
|
+
# Check if there are more key-value pairs after the closing brace
|
117
|
+
# This handles cases like '{"key": "value"}, "key2": "value2"}'
|
118
|
+
# But only if we're not in a nested context
|
119
|
+
if not self.context.empty:
|
120
|
+
return obj
|
121
|
+
|
122
|
+
self.skip_whitespaces_at()
|
123
|
+
if (self.get_char_at() or "") != ",":
|
124
|
+
return obj
|
125
|
+
self.index += 1
|
126
|
+
self.skip_whitespaces_at()
|
127
|
+
if (self.get_char_at() or "") not in STRING_DELIMITERS:
|
128
|
+
return obj
|
129
|
+
self.log(
|
130
|
+
"Found a comma and string delimiter after object closing brace, checking for additional key-value pairs",
|
131
|
+
)
|
132
|
+
additional_obj = self.parse_object()
|
133
|
+
if isinstance(additional_obj, dict):
|
134
|
+
obj.update(additional_obj)
|
135
|
+
|
115
136
|
return obj
|
@@ -1,6 +1,6 @@
|
|
1
1
|
Metadata-Version: 2.4
|
2
2
|
Name: json_repair
|
3
|
-
Version: 0.
|
3
|
+
Version: 0.51.0
|
4
4
|
Summary: A package to repair broken json strings
|
5
5
|
Author-email: Stefano Baccianella <4247706+mangiucugna@users.noreply.github.com>
|
6
6
|
License-Expression: MIT
|
@@ -269,6 +269,7 @@ You will need owner access to this repository
|
|
269
269
|
- Ruby: https://github.com/sashazykov/json-repair-rb
|
270
270
|
- Rust: https://github.com/oramasearch/llm_json
|
271
271
|
- R: https://github.com/cgxjdzz/jsonRepair
|
272
|
+
- Java: https://github.com/du00cs/json-repairj
|
272
273
|
---
|
273
274
|
## Star History
|
274
275
|
|
@@ -83,3 +83,17 @@ def test_parse_object_edge_cases():
|
|
83
83
|
== '{"key": "{\\"key\\":[\\"value\\"],\\"key2\\":\\"value2\\"}"}'
|
84
84
|
)
|
85
85
|
assert repair_json('{"key": , "key2": "value2"}') == '{"key": "", "key2": "value2"}'
|
86
|
+
|
87
|
+
|
88
|
+
def test_parse_object_merge_at_the_end():
|
89
|
+
assert repair_json('{"key": "value"}, "key2": "value2"}') == '{"key": "value", "key2": "value2"}'
|
90
|
+
assert repair_json('{"key": "value"}, "key2": }') == '{"key": "value", "key2": ""}'
|
91
|
+
assert repair_json('{"key": "value"}, []') == '[{"key": "value"}, []]'
|
92
|
+
assert repair_json('{"key": "value"}, ["abc"]') == '[{"key": "value"}, ["abc"]]'
|
93
|
+
assert repair_json('{"key": "value"}, {}') == '[{"key": "value"}, {}]'
|
94
|
+
assert repair_json('{"key": "value"}, "" : "value2"}') == '{"key": "value", "": "value2"}'
|
95
|
+
assert repair_json('{"key": "value"}, "key2" "value2"}') == '{"key": "value", "key2": "value2"}'
|
96
|
+
assert (
|
97
|
+
repair_json('{"key1": "value1"}, "key2": "value2", "key3": "value3"}')
|
98
|
+
== '{"key1": "value1", "key2": "value2", "key3": "value3"}'
|
99
|
+
)
|
@@ -1,9 +1,12 @@
|
|
1
1
|
import os.path
|
2
2
|
import pathlib
|
3
3
|
|
4
|
+
import pytest
|
5
|
+
|
4
6
|
from src.json_repair import repair_json
|
5
7
|
|
6
8
|
path = pathlib.Path(__file__).parent.resolve()
|
9
|
+
CI = os.getenv("CI") is not None
|
7
10
|
|
8
11
|
with open(os.path.join(path, "valid.json")) as fd:
|
9
12
|
correct_json = fd.read()
|
@@ -12,6 +15,7 @@ with open(os.path.join(path, "invalid.json")) as fd:
|
|
12
15
|
incorrect_json = fd.read()
|
13
16
|
|
14
17
|
|
18
|
+
@pytest.mark.skipif(CI, reason="Performance tests are skipped in CI")
|
15
19
|
def test_true_true_correct(benchmark):
|
16
20
|
benchmark(repair_json, correct_json, return_objects=True, skip_json_loads=True)
|
17
21
|
|
@@ -25,6 +29,7 @@ def test_true_true_correct(benchmark):
|
|
25
29
|
assert mean_time < max_time, f"Benchmark exceeded threshold: {mean_time:.3f}s > {max_time:.3f}s"
|
26
30
|
|
27
31
|
|
32
|
+
@pytest.mark.skipif(CI, reason="Performance tests are skipped in CI")
|
28
33
|
def test_true_true_incorrect(benchmark):
|
29
34
|
benchmark(repair_json, incorrect_json, return_objects=True, skip_json_loads=True)
|
30
35
|
|
@@ -38,6 +43,7 @@ def test_true_true_incorrect(benchmark):
|
|
38
43
|
assert mean_time < max_time, f"Benchmark exceeded threshold: {mean_time:.3f}s > {max_time:.3f}s"
|
39
44
|
|
40
45
|
|
46
|
+
@pytest.mark.skipif(CI, reason="Performance tests are skipped in CI")
|
41
47
|
def test_true_false_correct(benchmark):
|
42
48
|
benchmark(repair_json, correct_json, return_objects=True, skip_json_loads=False)
|
43
49
|
# Retrieve the median execution time
|
@@ -50,6 +56,7 @@ def test_true_false_correct(benchmark):
|
|
50
56
|
assert mean_time < max_time, f"Benchmark exceeded threshold: {mean_time:.3f}s > {max_time:.3f}s"
|
51
57
|
|
52
58
|
|
59
|
+
@pytest.mark.skipif(CI, reason="Performance tests are skipped in CI")
|
53
60
|
def test_true_false_incorrect(benchmark):
|
54
61
|
benchmark(repair_json, incorrect_json, return_objects=True, skip_json_loads=False)
|
55
62
|
# Retrieve the median execution time
|
@@ -62,6 +69,7 @@ def test_true_false_incorrect(benchmark):
|
|
62
69
|
assert mean_time < max_time, f"Benchmark exceeded threshold: {mean_time:.3f}s > {max_time:.3f}s"
|
63
70
|
|
64
71
|
|
72
|
+
@pytest.mark.skipif(CI, reason="Performance tests are skipped in CI")
|
65
73
|
def test_false_true_correct(benchmark):
|
66
74
|
benchmark(repair_json, correct_json, return_objects=False, skip_json_loads=True)
|
67
75
|
# Retrieve the median execution time
|
@@ -74,6 +82,7 @@ def test_false_true_correct(benchmark):
|
|
74
82
|
assert mean_time < max_time, f"Benchmark exceeded threshold: {mean_time:.3f}s > {max_time:.3f}s"
|
75
83
|
|
76
84
|
|
85
|
+
@pytest.mark.skipif(CI, reason="Performance tests are skipped in CI")
|
77
86
|
def test_false_true_incorrect(benchmark):
|
78
87
|
benchmark(repair_json, incorrect_json, return_objects=False, skip_json_loads=True)
|
79
88
|
# Retrieve the median execution time
|
@@ -86,6 +95,7 @@ def test_false_true_incorrect(benchmark):
|
|
86
95
|
assert mean_time < max_time, f"Benchmark exceeded threshold: {mean_time:.3f}s > {max_time:.3f}s"
|
87
96
|
|
88
97
|
|
98
|
+
@pytest.mark.skipif(CI, reason="Performance tests are skipped in CI")
|
89
99
|
def test_false_false_correct(benchmark):
|
90
100
|
benchmark(repair_json, correct_json, return_objects=False, skip_json_loads=False)
|
91
101
|
# Retrieve the median execution time
|
@@ -98,6 +108,7 @@ def test_false_false_correct(benchmark):
|
|
98
108
|
assert mean_time < max_time, f"Benchmark exceeded threshold: {mean_time:.3f}s > {max_time:.3f}s"
|
99
109
|
|
100
110
|
|
111
|
+
@pytest.mark.skipif(CI, reason="Performance tests are skipped in CI")
|
101
112
|
def test_false_false_incorrect(benchmark):
|
102
113
|
benchmark(repair_json, incorrect_json, return_objects=False, skip_json_loads=False)
|
103
114
|
# Retrieve the median execution time
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|