3
3
from __future__ import annotations
4
4
5
5
import base64
6
- from base64 import b64decode
7
6
import binascii
7
+ import logging
8
+ import platform
9
+ import socket
10
+ from base64 import b64decode
8
11
from dataclasses import MISSING , asdict , fields , is_dataclass
9
12
from datetime import datetime
10
13
from enum import Enum
11
14
from functools import cache
12
- from importlib .metadata import PackageNotFoundError , version as pkg_version
13
- import logging
14
- import platform
15
- import socket
15
+ from importlib .metadata import PackageNotFoundError
16
+ from importlib .metadata import version as pkg_version
16
17
from types import NoneType , UnionType
17
18
from typing import (
18
19
TYPE_CHECKING ,
@@ -108,8 +109,13 @@ def parse_value(
108
109
value_type : Any ,
109
110
default : Any = MISSING ,
110
111
allow_none : bool = True ,
112
+ allow_sdk_types : bool = False ,
111
113
) -> Any :
112
- """Try to parse a value from raw (json) data and type annotations."""
114
+ """
115
+ Try to parse a value from raw (json) data and type annotations.
116
+
117
+ If allow_sdk_types is False, any SDK specific custom data types will be converted.
118
+ """
113
119
# pylint: disable=too-many-return-statements,too-many-branches
114
120
115
121
if isinstance (value_type , str ):
@@ -131,9 +137,9 @@ def parse_value(
131
137
return default
132
138
if value is None and value_type is NoneType :
133
139
return None
134
- if value is None and allow_none :
135
- return None
136
140
if value is None and value_type is Nullable :
141
+ return Nullable () if allow_sdk_types else None
142
+ if value is None and allow_none :
137
143
return None
138
144
if is_dataclass (value_type ) and isinstance (value , dict ):
139
145
return dataclass_from_dict (value_type , value )
@@ -220,12 +226,12 @@ def parse_value(
220
226
if value_type is uint and (
221
227
isinstance (value , int ) or (isinstance (value , str ) and value .isnumeric ())
222
228
):
223
- return uint (value )
229
+ return uint (value ) if allow_sdk_types else int ( value )
224
230
if value_type is float32 and (
225
231
isinstance (value , (float , int ))
226
232
or (isinstance (value , str ) and value .isnumeric ())
227
233
):
228
- return float32 (value )
234
+ return float32 (value ) if allow_sdk_types else float ( value )
229
235
230
236
# If we reach this point, we could not match the value with the type and we raise
231
237
if not isinstance (value , value_type ):
@@ -236,7 +242,9 @@ def parse_value(
236
242
return value
237
243
238
244
239
- def dataclass_from_dict (cls : type [_T ], dict_obj : dict , strict : bool = False ) -> _T :
245
+ def dataclass_from_dict (
246
+ cls : type [_T ], dict_obj : dict , strict : bool = False , allow_sdk_types : bool = False
247
+ ) -> _T :
240
248
"""
241
249
Create (instance of) a dataclass by providing a dict with values.
242
250
@@ -259,6 +267,7 @@ def dataclass_from_dict(cls: type[_T], dict_obj: dict, strict: bool = False) ->
259
267
type_hints [field .name ],
260
268
field .default ,
261
269
allow_none = not strict ,
270
+ allow_sdk_types = allow_sdk_types ,
262
271
)
263
272
for field in dc_fields
264
273
if field .init
0 commit comments