From dd4c4140450087e7bec971fb946d1a05a2ba4a69 Mon Sep 17 00:00:00 2001 From: Matthew Hoffman Date: Fri, 25 Aug 2023 12:40:17 -0700 Subject: [PATCH] Improve dataclass_json and _process_class type annotations (#475) Use typing.overload to differentiate the return type of dataclass_json based on whether _cls parameter is None or not --- dataclasses_json/api.py | 43 ++++++++++++++++++++++++++--------------- 1 file changed, 27 insertions(+), 16 deletions(-) diff --git a/dataclasses_json/api.py b/dataclasses_json/api.py index 03421c6c..9d05a977 100644 --- a/dataclasses_json/api.py +++ b/dataclasses_json/api.py @@ -1,9 +1,8 @@ import abc import json -from typing import (Any, Callable, Dict, List, Optional, Tuple, Type, TypeVar, - Union) +from typing import Any, Callable, Dict, List, Optional, Tuple, Type, TypeVar, Union, overload -from dataclasses_json.cfg import config, LetterCase # noqa: F401 +from dataclasses_json.cfg import config, LetterCase from dataclasses_json.core import (Json, _ExtendedEncoder, _asdict, _decode_dataclass) from dataclasses_json.mm import (JsonData, SchemaType, build_schema) @@ -12,6 +11,7 @@ _undefined_parameter_action_safe) A = TypeVar('A', bound="DataClassJsonMixin") +T = TypeVar('T') Fields = List[Tuple[str, Any]] @@ -102,8 +102,18 @@ def schema(cls: Type[A], unknown=unknown) -def dataclass_json(_cls=None, *, letter_case=None, - undefined: Optional[Union[str, Undefined]] = None): +@overload +def dataclass_json(_cls: None = ..., *, letter_case: Optional[LetterCase] = ..., + undefined: Optional[Union[str, Undefined]] = ...) -> Callable[[Type[T]], Type[T]]: ... + + +@overload +def dataclass_json(_cls: Type[T], *, letter_case: Optional[LetterCase] = ..., + undefined: Optional[Union[str, Undefined]] = ...) -> Type[T]: ... + + +def dataclass_json(_cls: Optional[Type[T]] = None, *, letter_case: Optional[LetterCase] = None, + undefined: Optional[Union[str, Undefined]] = None) -> Union[Callable[[Type[T]], Type[T]], Type[T]]: """ Based on the code in the `dataclasses` module to handle optional-parens decorators. See example below: @@ -114,7 +124,7 @@ class Example: ... """ - def wrap(cls): + def wrap(cls: Type[T]) -> Type[T]: return _process_class(cls, letter_case, undefined) if _cls is None: @@ -122,21 +132,22 @@ def wrap(cls): return wrap(_cls) -def _process_class(cls, letter_case, undefined): +def _process_class(cls: Type[T], letter_case: Optional[LetterCase], + undefined: Optional[Union[str, Undefined]]) -> Type[T]: if letter_case is not None or undefined is not None: - cls.dataclass_json_config = config(letter_case=letter_case, - undefined=undefined)[ - 'dataclasses_json'] + cls.dataclass_json_config = config(letter_case=letter_case, # type: ignore[attr-defined] + undefined=undefined)['dataclasses_json'] - cls.to_json = DataClassJsonMixin.to_json + cls.to_json = DataClassJsonMixin.to_json # type: ignore[attr-defined] # unwrap and rewrap classmethod to tag it to cls rather than the literal # DataClassJsonMixin ABC - cls.from_json = classmethod(DataClassJsonMixin.from_json.__func__) # type: ignore - cls.to_dict = DataClassJsonMixin.to_dict - cls.from_dict = classmethod(DataClassJsonMixin.from_dict.__func__) # type: ignore - cls.schema = classmethod(DataClassJsonMixin.schema.__func__) # type: ignore + cls.from_json = classmethod(DataClassJsonMixin.from_json.__func__) # type: ignore[attr-defined] + cls.to_dict = DataClassJsonMixin.to_dict # type: ignore[attr-defined] + cls.from_dict = classmethod(DataClassJsonMixin.from_dict.__func__) # type: ignore[attr-defined] + cls.schema = classmethod(DataClassJsonMixin.schema.__func__) # type: ignore[attr-defined] - cls.__init__ = _handle_undefined_parameters_safe(cls, kvs=(), usage="init") + cls.__init__ = _handle_undefined_parameters_safe(cls, kvs=(), # type: ignore[attr-defined,method-assign] + usage="init") # register cls as a virtual subclass of DataClassJsonMixin DataClassJsonMixin.register(cls) return cls