Skip to content

Commit

Permalink
Compatibility with json.dumps on bool, None and invalid keys
Browse files Browse the repository at this point in the history
  • Loading branch information
Wiktor Latanowicz committed Oct 9, 2022
1 parent c3873db commit 9a3fb19
Show file tree
Hide file tree
Showing 3 changed files with 49 additions and 3 deletions.
15 changes: 14 additions & 1 deletion json_stream_generator/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,13 @@
__author__ = "Wiktor Latanowicz"


KEY_CONVERTION_LUT = {
None: "null",
True: "true",
False: "false",
}


def json_generator(obj: Any, depth: int = 1) -> Generator[str, None, None]:
if depth < 1:
yield json.dumps(obj)
Expand All @@ -17,7 +24,13 @@ def json_generator(obj: Any, depth: int = 1) -> Generator[str, None, None]:
for i, (key, value) in enumerate(obj.items()):
if i > 0:
yield ", "
yield json.dumps(str(key)) + ": "

if not isinstance(key, (str, int, float, bool)) and key is not None:
raise TypeError(
f"keys must be str, int, float, bool or None, not {key.__class__.__name__}"
)

yield json.dumps(str(KEY_CONVERTION_LUT.get(key, key))) + ": "
yield from json_generator(value, depth=depth - 1)
yield "}"
elif isinstance(obj, (list, tuple)) or inspect.isgenerator(obj):
Expand Down
3 changes: 1 addition & 2 deletions json_stream_generator/rest_framework/mixins.py
Original file line number Diff line number Diff line change
@@ -1,8 +1,7 @@
import rest_framework
from django.db.models.query import QuerySet
from django.http import StreamingHttpResponse

import rest_framework

from .. import json_generator


Expand Down
34 changes: 34 additions & 0 deletions tests/test_json_generator.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import json
from datetime import datetime

import pytest

Expand Down Expand Up @@ -77,3 +78,36 @@ def test_simple_values():
result = "".join(json_generator(obj))

assert expected == result


def test_different_key_types():
obj = {
123: "A",
123.456: "B",
True: "C",
False: "D",
None: "E",
"abc": "F",
}
expected = json.dumps(obj)
result = "".join(json_generator(obj))

assert expected == result


# fmt: off
@pytest.mark.parametrize(
"obj",
[
{(1, 2): "A"},
[{(1, 2): "A"}],
{datetime.now(): "A"},
[{datetime.now(): "A"}],
({datetime.now(): "A"} for _ in range(5)),
({(1, 2): "A"} for _ in range(5)),
]
)
# fmt: on
def test_invalid_keys(obj):
with pytest.raises(TypeError):
"".join(json_generator(obj))

0 comments on commit 9a3fb19

Please sign in to comment.