|
10 | 10 | from keyword import iskeyword
|
11 | 11 | import sys
|
12 | 12 | import typing
|
13 |
| -from typing import Any, ForwardRef, Hashable, Optional, Union, get_type_hints |
| 13 | +from typing import Any, ForwardRef, Hashable, Optional, TypeVar, Union, get_type_hints |
14 | 14 | import typing_extensions
|
15 | 15 |
|
16 | 16 | from .validation_failure import (
|
17 | 17 | InvalidNumpyDTypeValidationFailure,
|
| 18 | + TypeVarBoundValidationFailure, |
18 | 19 | ValidationFailureAtIdx,
|
19 | 20 | ValidationFailureAtKey,
|
20 | 21 | MissingKeysValidationFailure,
|
@@ -219,6 +220,14 @@ def _type_error(
|
219 | 220 | setattr(error, "validation_failure", validation_failure)
|
220 | 221 | return error
|
221 | 222 |
|
| 223 | +def _typevar_error(val: Any, t: Any, bound_error: TypeError) -> TypeError: |
| 224 | + assert hasattr(bound_error, "validation_failure"), bound_error |
| 225 | + cause = getattr(bound_error, "validation_failure") |
| 226 | + assert isinstance(cause, ValidationFailure), cause |
| 227 | + validation_failure = TypeVarBoundValidationFailure(val, t, cause) |
| 228 | + error = TypeError(str(validation_failure)) |
| 229 | + setattr(error, "validation_failure", validation_failure) |
| 230 | + return error |
222 | 231 |
|
223 | 232 | def _idx_type_error(
|
224 | 233 | val: Any, t: Any, idx_error: TypeError, *, idx: int, ordered: bool
|
@@ -533,7 +542,6 @@ def _extract_dtypes(t: Any) -> typing.Sequence[Any]:
|
533 | 542 | return [t]
|
534 | 543 | raise TypeError()
|
535 | 544 |
|
536 |
| - |
537 | 545 | def _validate_numpy_array(val: Any, t: Any) -> None:
|
538 | 546 | import numpy as np # pylint: disable = import-outside-toplevel
|
539 | 547 | if not isinstance(val, TypeInspector):
|
@@ -569,6 +577,17 @@ def _validate_numpy_array(val: Any, t: Any) -> None:
|
569 | 577 | raise _numpy_dtype_error(val, t)
|
570 | 578 |
|
571 | 579 |
|
| 580 | +def _validate_typevar(val: Any, t: TypeVar) -> None: |
| 581 | + if isinstance(val, TypeInspector): |
| 582 | + val._record_typevar(t) |
| 583 | + pass |
| 584 | + bound = t.__bound__ |
| 585 | + if bound is not None: |
| 586 | + try: |
| 587 | + validate(val, bound) |
| 588 | + except TypeError as e: |
| 589 | + raise _typevar_error(val, t, e) from None |
| 590 | + |
572 | 591 | # def _validate_callable(val: Any, t: Any) -> None:
|
573 | 592 | # """
|
574 | 593 | # Callable validation
|
@@ -618,16 +637,16 @@ def validate(val: Any, t: Any) -> Literal[True]:
|
618 | 637 |
|
619 | 638 | For structured types, the error message keeps track of the chain of validation failures, e.g.
|
620 | 639 |
|
621 |
| - >>> from typing import * |
622 |
| - >>> from typing_validation import validate |
623 |
| - >>> validate([[0, 1, 2], {"hi": 0}], list[Union[Collection[int], dict[str, str]]]) |
624 |
| - TypeError: Runtime validation error raised by validate(val, t), details below. |
625 |
| - For type list[typing.Union[typing.Collection[int], dict[str, str]]], invalid value at idx: 1 |
626 |
| - For union type typing.Union[typing.Collection[int], dict[str, str]], invalid value: {'hi': 0} |
627 |
| - For member type typing.Collection[int], invalid value at idx: 0 |
628 |
| - For type <class 'int'>, invalid value: 'hi' |
629 |
| - For member type dict[str, str], invalid value at key: 'hi' |
630 |
| - For type <class 'str'>, invalid value: 0 |
| 640 | + >>> from typing import * |
| 641 | + >>> from typing_validation import validate |
| 642 | + >>> validate([[0, 1, 2], {"hi": 0}], list[Union[Collection[int], dict[str, str]]]) |
| 643 | + TypeError: Runtime validation error raised by validate(val, t), details below. |
| 644 | + For type list[typing.Union[typing.Collection[int], dict[str, str]]], invalid value at idx: 1 |
| 645 | + For union type typing.Union[typing.Collection[int], dict[str, str]], invalid value: {'hi': 0} |
| 646 | + For member type typing.Collection[int], invalid value at idx: 0 |
| 647 | + For type <class 'int'>, invalid value: 'hi' |
| 648 | + For member type dict[str, str], invalid value at key: 'hi' |
| 649 | + For type <class 'str'>, invalid value: 0 |
631 | 650 |
|
632 | 651 | **Note.** For Python 3.7 and 3.8, use :obj:`~typing.Dict` and :obj:`~typing.List` instead of :obj:`dict` and :obj:`list` for the above examples.
|
633 | 652 |
|
@@ -670,6 +689,9 @@ def validate(val: Any, t: Any) -> Literal[True]:
|
670 | 689 | val._record_any()
|
671 | 690 | return True
|
672 | 691 | return True
|
| 692 | + if isinstance(t, TypeVar): |
| 693 | + _validate_typevar(val, t) |
| 694 | + return True |
673 | 695 | if UnionType is not None and isinstance(t, UnionType):
|
674 | 696 | _validate_union(val, t, union_type=True)
|
675 | 697 | return True
|
|
0 commit comments