@@ -29,9 +29,10 @@ def issoftkeyword(s: str) -> bool:
29
29
return s == "_"
30
30
31
31
if sys .version_info [1 ] >= 10 :
32
- from types import NoneType
32
+ from types import NoneType , UnionType
33
33
else :
34
34
NoneType = type (None )
35
+ UnionType = None
35
36
36
37
_validation_aliases : typing .Dict [str , Any ] = {}
37
38
r"""
@@ -292,7 +293,7 @@ def _validate_tuple(val: Any, t: Any) -> None:
292
293
except TypeError as e :
293
294
raise _idx_type_error (val , t , e , idx = idx , ordered = True ) from None
294
295
295
- def _validate_union (val : Any , t : Any ) -> None :
296
+ def _validate_union (val : Any , t : Any , * , union_type : bool = False ) -> None :
296
297
"""
297
298
Union type validation. Each type ``u`` listed in the union type ``t`` is checked:
298
299
@@ -387,6 +388,18 @@ def _validate_typed_dict(val: Any, t: type) -> None:
387
388
except TypeError as e :
388
389
raise _key_type_error (val , t , e , key = k ) from None
389
390
391
+ def _validate_user_class (val : Any , t : Any ) -> None :
392
+ assert hasattr (t , "__args__" ), _missing_args_msg (t )
393
+ assert isinstance (t .__args__ , tuple ), f"For type { repr (t )} , expected '__args__' to be a tuple."
394
+ if isinstance (val , TypeInspector ):
395
+ val ._record_pending_type_generic (t .__origin__ )
396
+ val ._record_user_class (* t .__args__ )
397
+ for arg in t .__args__ :
398
+ validate (val , arg )
399
+ return
400
+ _validate_type (val , t .__origin__ )
401
+ # Generic type arguments cannot be validated
402
+
390
403
# def _validate_callable(val: Any, t: Any) -> None:
391
404
# """
392
405
# Callable validation
@@ -471,6 +484,9 @@ def validate(val: Any, t: Any) -> None:
471
484
val ._record_any ()
472
485
return
473
486
return
487
+ if UnionType is not None and isinstance (t , UnionType ):
488
+ _validate_union (val , t , union_type = True )
489
+ return
474
490
if hasattr (t , "__origin__" ): # parametric types
475
491
if t .__origin__ is Union :
476
492
_validate_union (val , t )
@@ -483,22 +499,26 @@ def validate(val: Any, t: Any) -> None:
483
499
val ._record_pending_type_generic (t .__origin__ )
484
500
else :
485
501
_validate_type (val , t .__origin__ )
486
- if t .__origin__ in _collection_origins :
487
- ordered = t .__origin__ in _ordered_collection_origins
488
- _validate_collection (val , t , ordered )
489
- return
490
- if t .__origin__ in _mapping_origins :
491
- _validate_mapping (val , t )
492
- return
493
- if t .__origin__ == tuple :
494
- _validate_tuple (val , t )
495
- return
496
- if t .__origin__ in _iterator_origins :
497
- if isinstance (val , TypeInspector ):
502
+ if t .__origin__ in _collection_origins :
503
+ ordered = t .__origin__ in _ordered_collection_origins
504
+ _validate_collection (val , t , ordered )
505
+ return
506
+ if t .__origin__ in _mapping_origins :
507
+ _validate_mapping (val , t )
508
+ return
509
+ if t .__origin__ == tuple :
510
+ _validate_tuple (val , t )
511
+ return
512
+ if t .__origin__ in _iterator_origins :
513
+ if isinstance (val , TypeInspector ):
514
+ _validate_collection (val , t , ordered = False )
515
+ # Item type cannot be validated for iterators (use validated_iter)
516
+ return
517
+ if t .__origin__ in _maybe_collection_origins and isinstance (val , typing .Collection ):
498
518
_validate_collection (val , t , ordered = False )
499
- return
500
- if t .__origin__ in _maybe_collection_origins and isinstance ( val , typing . Collection ):
501
- _validate_collection (val , t , ordered = False )
519
+ return
520
+ elif isinstance ( t .__origin__ , type ):
521
+ _validate_user_class (val , t )
502
522
return
503
523
elif isinstance (t , type ):
504
524
# The `isinstance(t, type)` case goes after the `hasattr(t, "__origin__")` case:
0 commit comments