23
23
24
24
from .validation_failure import (
25
25
InvalidNumpyDTypeValidationFailure ,
26
+ SubtypeValidationFailure ,
26
27
TypeVarBoundValidationFailure ,
27
28
ValidationFailureAtIdx ,
28
29
ValidationFailureAtKey ,
@@ -199,6 +200,29 @@ def validation_aliases(**aliases: Any) -> collections.abc.Iterator[None]:
199
200
)
200
201
201
202
203
+ class UnsupportedTypeError (ValueError ):
204
+ """
205
+ Class for errors raised when attempting to validate an unsupported type.
206
+
207
+ .. warning::
208
+
209
+ Currently extends :obj:`ValueError` for backwards compatibility.
210
+ This will be changed to :obj:`NotImplementedError` in v1.3.0.
211
+ """
212
+
213
+
214
+ def _unsupported_type_error (
215
+ t : Any , explanation : Union [str , None ] = None
216
+ ) -> UnsupportedTypeError :
217
+ """
218
+ Error for unsupported types, with optional explanation.
219
+ """
220
+ msg = "Unsupported validation for type {t!r}."
221
+ if explanation is not None :
222
+ msg += " " + explanation
223
+ return UnsupportedTypeError (msg )
224
+
225
+
202
226
def _type_error (
203
227
val : Any , t : Any , * errors : TypeError , is_union : bool = False
204
228
) -> TypeError :
@@ -272,6 +296,13 @@ def _missing_keys_type_error(val: Any, t: Any, *missing_keys: Any) -> TypeError:
272
296
return error
273
297
274
298
299
+ def _subtype_error (s : Any , t : Any ) -> TypeError :
300
+ validation_failure = SubtypeValidationFailure (s , t )
301
+ error = TypeError (str (validation_failure ))
302
+ setattr (error , "validation_failure" , validation_failure )
303
+ return error
304
+
305
+
275
306
def _type_alias_error (t_alias : str , cause : TypeError ) -> TypeError :
276
307
"""
277
308
Repackages a validation error as a type alias error.
@@ -505,21 +536,88 @@ def _validate_typed_dict(val: Any, t: type) -> None:
505
536
except TypeError as e :
506
537
raise _key_type_error (val , t , e , key = k ) from None
507
538
508
-
509
539
def _validate_user_class (val : Any , t : Any ) -> None :
510
540
assert hasattr (t , "__args__" ), _missing_args_msg (t )
511
541
assert isinstance (
512
542
t .__args__ , tuple
513
543
), f"For type { repr (t )} , expected '__args__' to be a tuple."
514
544
if isinstance (val , TypeInspector ):
545
+ if t .__origin__ is type :
546
+ if len (t .__args__ ) != 1 or not _can_validate_subtype_of (
547
+ t .__args__ [0 ]
548
+ ):
549
+ val ._record_unsupported_type (t )
550
+ return
515
551
val ._record_pending_type_generic (t .__origin__ )
516
552
val ._record_user_class (* t .__args__ )
517
553
for arg in t .__args__ :
518
554
validate (val , arg )
519
555
return
520
556
_validate_type (val , t .__origin__ )
521
- # Generic type arguments cannot be validated
557
+ if t .__origin__ is type :
558
+ if len (t .__args__ ) != 1 :
559
+ raise _unsupported_type_error (t )
560
+ _validate_subtype_of (val , t .__args__ [0 ])
561
+ return
562
+ # TODO: Generic type arguments cannot be validated in general,
563
+ # but in a future release it will be possible for classes to define
564
+ # a dunder classmethod which can be used to validate type arguments.
565
+
566
+ def __extract_member_types (u : Any ) -> tuple [Any , ...]| None :
567
+ q = collections .deque ([u ])
568
+ member_types : list [Any ] = []
569
+ while q :
570
+ t = q .popleft ()
571
+ if t is Any :
572
+ return None
573
+ elif UnionType is not None and isinstance (t , UnionType ):
574
+ q .extend (t .__args__ )
575
+ elif hasattr (t , "__origin__" ) and t .__origin__ is Union :
576
+ q .extend (t .__args__ )
577
+ else :
578
+ member_types .append (t )
579
+ return tuple (member_types )
580
+
581
+ def __check_can_validate_subtypes (* subtypes : Any ) -> None :
582
+ for s in subtypes :
583
+ if not isinstance (s , type ):
584
+ raise ValueError (
585
+ "validate(s, Type[t]) is only supported when 's' is "
586
+ "an instance of 'type' or a union of instances of 'type'.\n "
587
+ f"Found s = { '|' .join (str (s ) for s in subtypes )} "
588
+ )
589
+
590
+ def __check_can_validate_supertypes (* supertypes : Any ) -> None :
591
+ for t in supertypes :
592
+ if not isinstance (t , type ):
593
+ raise ValueError (
594
+ "validate(s, Type[t]) is only supported when 't' is "
595
+ "an instance of 'type' or a union of instances of 'type'.\n "
596
+ f"Found t = { '|' .join (str (t ) for t in supertypes )} "
597
+ )
598
+
599
+ def _can_validate_subtype_of (t : Any ) -> bool :
600
+ try :
601
+ # This is the validation part of _validate_subtype:
602
+ t_member_types = __extract_member_types (t )
603
+ if t_member_types is not None :
604
+ __check_can_validate_supertypes (* t_member_types )
605
+ return True
606
+ except ValueError :
607
+ return False
522
608
609
+ def _validate_subtype_of (s : Any , t : Any ) -> None :
610
+ # 1. Validation:
611
+ __check_can_validate_subtypes (s )
612
+ t_member_types = __extract_member_types (t )
613
+ if t_member_types is None :
614
+ # An Any was found amongst the member types, all good.
615
+ return
616
+ __check_can_validate_supertypes (* t_member_types )
617
+ # 2. Subtype check:
618
+ if not issubclass (s , t_member_types ):
619
+ raise _subtype_error (s , t )
620
+ # TODO: improve support for subtype checks.
523
621
524
622
def _extract_dtypes (t : Any ) -> typing .Sequence [Any ]:
525
623
if t is Any :
@@ -575,8 +673,8 @@ def _validate_numpy_array(val: Any, t: Any) -> None:
575
673
if isinstance (val , TypeInspector ):
576
674
val ._record_unsupported_type (t )
577
675
return
578
- raise ValueError (
579
- f"Unsupported validation for NumPy dtype { repr ( dtype_t ) } . "
676
+ raise _unsupported_type_error (
677
+ t , f"Unsupported NumPy dtype { dtype_t !r } "
580
678
) from None
581
679
if isinstance (val , TypeInspector ):
582
680
val ._record_pending_type_generic (t .__origin__ )
@@ -669,21 +767,24 @@ def validate(val: Any, t: Any) -> Literal[True]:
669
767
:param t: the type to type-check against
670
768
:type t: :obj:`~typing.Any`
671
769
:raises TypeError: if ``val`` is not of type ``t``
672
- :raises ValueError : if validation for type ``t`` is not supported
770
+ :raises UnsupportedTypeError : if validation for type ``t`` is not supported
673
771
:raises AssertionError: if things go unexpectedly wrong with ``__args__`` for parametric types
674
772
675
773
"""
676
774
# pylint: disable = too-many-return-statements, too-many-branches, too-many-statements
677
- unsupported_type_error : Optional [ValueError ] = None
775
+ unsupported_type_error : Optional [UnsupportedTypeError ] = None
678
776
if not isinstance (t , Hashable ):
679
777
if isinstance (val , TypeInspector ):
680
778
val ._record_unsupported_type (t )
681
779
return True
682
780
if unsupported_type_error is None :
683
- unsupported_type_error = ValueError (
684
- f"Unsupported validation for type { repr ( t ) } . Type is not hashable."
781
+ unsupported_type_error = _unsupported_type_error (
782
+ t , " Type is not hashable."
685
783
) # pragma: nocover
686
784
raise unsupported_type_error
785
+ if t is typing .Type :
786
+ # Replace non-generic 'Type' with non-generic 'type':
787
+ t = type
687
788
if t in _basic_types :
688
789
# speed things up for the likely most common case
689
790
_validate_type (val , typing .cast (type , t ))
@@ -765,8 +866,8 @@ def validate(val: Any, t: Any) -> Literal[True]:
765
866
if isinstance (val , TypeInspector ):
766
867
val ._record_unsupported_type (t )
767
868
return True
768
- unsupported_type_error = ValueError (
769
- f"Unsupported validation for Protocol { repr ( t ) } , because it is not runtime-checkable."
869
+ unsupported_type_error = _unsupported_type_error (
870
+ t , "Protocol class is not runtime-checkable."
770
871
) # pragma: nocover
771
872
elif _is_typed_dict (t ):
772
873
_validate_typed_dict (val , t )
@@ -788,8 +889,8 @@ def validate(val: Any, t: Any) -> Literal[True]:
788
889
hint = f"Perhaps set it with validation_aliases({ t_alias } =...)?"
789
890
else :
790
891
hint = f"Perhaps set it with validation_aliases(**{{'{ t_alias } ': ...}})?"
791
- unsupported_type_error = ValueError (
792
- f"Type alias ' { t_alias } ' is not known. { hint } "
892
+ unsupported_type_error = _unsupported_type_error (
893
+ t_alias , f"Type alias is not known. { hint } "
793
894
) # pragma: nocover
794
895
else :
795
896
_validate_alias (val , t_alias )
@@ -798,15 +899,13 @@ def validate(val: Any, t: Any) -> Literal[True]:
798
899
val ._record_unsupported_type (t )
799
900
return True
800
901
if unsupported_type_error is None :
801
- unsupported_type_error = ValueError (
802
- f"Unsupported validation for type { repr (t )} ."
803
- ) # pragma: nocover
902
+ unsupported_type_error = _unsupported_type_error (t ) # pragma: nocover
804
903
raise unsupported_type_error
805
904
806
905
807
906
def can_validate (t : Any ) -> TypeInspector :
808
907
r"""
809
- Checks whether validation is supported for the given type ``t``: if not, :func:`validate` will raise :obj:`ValueError `.
908
+ Checks whether validation is supported for the given type ``t``: if not, :func:`validate` will raise :obj:`UnsupportedTypeError `.
810
909
811
910
The returned :class:`TypeInspector` instance can be used wherever a boolean is expected, and will indicate whether the type is supported or not:
812
911
0 commit comments