Skip to content
This repository was archived by the owner on Feb 7, 2022. It is now read-only.

Added support for Any and Dict types. #15

Open
wants to merge 1 commit into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
25 changes: 25 additions & 0 deletions tests/test_decoding.py
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
#!/usr/bin/env python3

from typing import Any
from typing import Dict
from typing import Generic
from typing import List
from typing import Optional
Expand Down Expand Up @@ -118,6 +119,30 @@ def test_can_decode_heterogeneous_list() -> None:
assert typedjson.decode(List[Union[str, int]], json) == json


def test_can_decode_any_list() -> None:
json = [1, "foo"]
assert typedjson.decode(List[Any], json) == json


def test_can_decode_string_any_dict() -> None:
json = {"foo": 1, "bar": "baz"}
assert typedjson.decode(Dict[str, Any], json) == json


def test_cannot_decode_dict_with_bad_key() -> None:
json = {"foo": 1, 5: "baz"}

expectation = DecodingError(TypeMismatch(("5",)))
assert typedjson.decode(Dict[str, Any], json) == expectation


def test_cannot_decode_dict_with_bad_value() -> None:
json = {"foo": 1, "baz": "baz"}

expectation = DecodingError(TypeMismatch(("foo",)))
assert typedjson.decode(Dict[str, str], json) == expectation


def test_can_decode_dataclass() -> None:
json = {"id": "test-user", "age": 28, "name": {"first": "Tomoya", "last": "Kose"}}

Expand Down
5 changes: 2 additions & 3 deletions typedjson/annotation.py
Original file line number Diff line number Diff line change
Expand Up @@ -20,9 +20,8 @@ def hints_of(type_: Type) -> Optional[Dict[str, Type]]:
type__ = type_ if origin is None else origin
mapping = dict(zip(parameters_of(type_), args))

# if hasattr(type__, '__annotations__'):
if hasattr(type__, "__init__"):
annotations = get_type_hints(type__.__init__)
if hasattr(type__, "__annotations__"):
annotations = get_type_hints(type__)
if len(mapping) > 0:
annotations_: Dict[str, Type] = {}
for n, t in annotations.items():
Expand Down
40 changes: 40 additions & 0 deletions typedjson/decoding.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@

from typing import Any
from typing import Iterable
from typing import Dict
from typing import Iterator
from typing import List
from typing import Optional
Expand All @@ -10,6 +11,7 @@
from typing import TypeVar
from typing import Union


Decoded = TypeVar("Decoded")
Value = TypeVar("Value")

Expand Down Expand Up @@ -82,6 +84,8 @@ def decode(
decode_as_list,
decode_as_primitive,
decode_as_class,
decode_as_dict,
decode_as_any,
)

result_final: Union[Decoded, DecodingError] = DecodingError(
Expand Down Expand Up @@ -123,6 +127,42 @@ def decode_as_primitive(
return DecodingError(UnsupportedDecoding(path))


def decode_as_dict(
type_: Type[Decoded], json: Any, path: Path
) -> Union[Decoded, DecodingError]:
from typedjson.annotation import args_of
from typedjson.annotation import origin_of

if isinstance(json, dict) and origin_of(type_) is dict:
KeyElement, ValueElement = args_of(type_)
dict_decoded: Dict[Any, Any] = {}

for key, element in json.items():
decoded_value = decode(ValueElement, element, path + (str(key),))
if isinstance(decoded_value, DecodingError):
return decoded_value

decoded_key = decode(KeyElement, key, path + (str(key),))
if isinstance(decoded_key, DecodingError):
return decoded_key

dict_decoded[decoded_key] = decoded_value

return dict_decoded # type: ignore
else:
return DecodingError(UnsupportedDecoding(path))


def decode_as_any(
type_: Type[Decoded], json: Any, path: Path
) -> Union[Decoded, DecodingError]:

if type_ == Any:
return json # type: ignore
else:
return DecodingError(UnsupportedDecoding(path))


def decode_as_class(
type_: Type[Decoded], json: Any, path: Path
) -> Union[Decoded, DecodingError]:
Expand Down