diff --git a/python_typing/strict.py b/python_typing/strict.py index bf3f95b..22c3262 100644 --- a/python_typing/strict.py +++ b/python_typing/strict.py @@ -102,7 +102,11 @@ def _assert_structure(self, structure, expected_structure): ) if structure is not expected_structure_type: arg_type = getattr(self._arg_type, "__name__", self._arg_type) - msg = f'Expected type {arg_type} in "{self._func_name}({self._arg_name}={arg_type})" got {structure.__name__}' + msg = ( + f"Expected type {arg_type} in " + f'"{self._func_name}({self._arg_name}={arg_type})" ' + f"got {structure.__name__}" + ) raise TypeError(msg) def _is_of_type_list(self, values: list, type: type): @@ -140,6 +144,10 @@ def _is_of_type(self, value, type): def strict(func): + """ + Raises a TypeError if any specified argument or return type mismatches its value. + """ + def inner(*arg_values, **kwargs): typing = Typing() typing.check_arg_types(func, arg_values, kwargs) diff --git a/python_typing_tests/test_custom_types.py b/python_typing_tests/test_custom_types.py new file mode 100644 index 0000000..6c10a76 --- /dev/null +++ b/python_typing_tests/test_custom_types.py @@ -0,0 +1,54 @@ +import heapq + +import pytest + +from python_typing.strict import strict + + +def test_custom_arg(): + class CustomType: + pass + + @strict + def _func(_: CustomType): + pass + + custom_obj = CustomType() + _func(custom_obj) + + +def test_custom_arg_fails(): + class CustomType: + pass + + @strict + def _func(_: CustomType): + pass + + with pytest.raises(TypeError) as err: + _func(1) + assert ( + str(err.value) == 'Value (1) in "_func(_=CustomType)" is not of type CustomType' + ) + + +def test_heap(): + @strict + def _func(_: list): + pass + + q = [] + heapq.heappush(q, (1, "test")) + _func(q) + + +def test_heap_expect_dict(): + @strict + def _func(_: dict): + pass + + q = [] + heapq.heappush(q, (1, "test")) + with pytest.raises(TypeError) as err: + _func(q) + assert str(err.value) == 'Expected type dict in "_func(_=dict)" got list' diff --git a/python_typing_tests/test_defaults.py b/python_typing_tests/test_defaults.py new file mode 100644 index 0000000..210f921 --- /dev/null +++ b/python_typing_tests/test_defaults.py @@ -0,0 +1,122 @@ +import heapq +from typing import Optional + +import pytest + +from python_typing.strict import strict + + +def test_default(): + @strict + def _func(_: str = "one"): + pass + + _func() + + +def test_single_default(): + @strict + def _func(one: int, two: str = "one"): + pass + + _func(1) + + +def test_single_default_both_specified(): + @strict + def _func(one: int, two: str = "one"): + pass + + _func(1, "two") + + +def test_single_default_specified(): + @strict + def _func(one: int, two: str = "one"): + pass + + _func(1, two="two") + + +def test_single_specified_with_default(): + @strict + def _func(one: int, two: int, three: str = "three"): + pass + + _func(1, two=2) + + +def test_default_wrong_type(): + @strict + def _func(_: str = 1): + pass + + with pytest.raises(TypeError) as err: + _func() + assert str(err.value) == 'Value (1) in "_func(_=str)" is not of type str' + + +def test_one_arg_wrong(): + @strict + def _func(one: int, two: int, three: str = "three"): + pass + + with pytest.raises(TypeError) as err: + _func(1, two="oops!") + assert str(err.value) == 'Value (oops!) in "_func(two=int)" is not of type int' + + +def test_some_specified_some_not(): + @strict + def _func(one: str, two, three: int): + pass + + _func("arg", 2, 3) + + +def test_arg_optional(): + @strict + def _func(_: Optional[str]): + pass + + _func("arg") + + +def test_arg_optional_none_specified(): + @strict + def _func(_: Optional[str]): + # Note the arg isn't actually optional here + pass + + with pytest.raises(TypeError) as err: + _func() + assert str(err.value) == "_func() missing 1 required positional argument: '_'" + + +def test_arg_optional_give_none(): + @strict + def _func(_: Optional[str] = None): + pass + + _func() + + +def test_arg_optional_give_correct_default(): + @strict + def _func(_: Optional[str] = "string"): + pass + + _func() + + +def test_arg_optional_give_wrong_default(): + @strict + def _func(_: Optional[str] = 1): + pass + + with pytest.raises(TypeError) as err: + _func() + assert ( + str(err.value) + == 'Value (1) in "_func(_=typing.Optional[str])" is not any of the valid types (str, NoneType)' + ) diff --git a/python_typing_tests/test_dict.py b/python_typing_tests/test_dict.py new file mode 100644 index 0000000..3d032d6 --- /dev/null +++ b/python_typing_tests/test_dict.py @@ -0,0 +1,136 @@ +from typing import Dict, List, Union + +import pytest + +from python_typing.strict import strict + + +def test_dict_arg(): + @strict + def _func(_: Dict[str, int]): + pass + + _func({"arg": 1}) + + +def test_dict_non_typing_arg(): + @strict + def _func(_: dict[str, int]): + pass + + _func({"arg": 1}) + + +def test_dict_arg_as_list(): + @strict + def _func(_: Dict[str, int]): + pass + + with pytest.raises(TypeError) as err: + _func(["arg", 1]) + assert ( + str(err.value) + == 'Expected type typing.Dict[str, int] in "_func(_=typing.Dict[str, int])" got list' + ) + + +def test_dict_non_typing_arg_as_list(): + @strict + def _func(_: dict[str, int]): + pass + + with pytest.raises(TypeError) as err: + _func(["arg", 1]) + assert str(err.value) == 'Expected type dict in "_func(_=dict)" got list' + + +def test_dict_arg_invalid(): + @strict + def _func(_: Dict[str, int]): + pass + + with pytest.raises(TypeError) as err: + _func({1: 3.14}) + assert ( + str(err.value) + == 'Value (1) in "_func(_=typing.Dict[str, int])" is not of type str' + ) + + +def test_dict_arg_multiple_key_values(): + @strict + def _func(_: Dict[str, int]): + pass + + _func( + { + "arg1": 1, + "arg2": 2, + "arg3": 3, + } + ) + + +def test_dict_arg_malformed_multiple_key_values(): + @strict + def _func(_: Dict[str, int]): + pass + + with pytest.raises(TypeError) as err: + _func( + { + "arg1": 1, + 2: "arg2", + "arg3": 3, + } + ) + assert ( + str(err.value) + == 'Value (2) in "_func(_=typing.Dict[str, int])" is not of type str' + ) + + +def test_dict_arg_multiple_valid_types(): + @strict + def _func(_: Dict[Union[str, float], int]): + pass + + _func({"arg": 1}) + _func({3.14: 1}) + + +def test_dict_of_dict_arg(): + @strict + def _func(_: Dict[str, Dict[str, float]]): + pass + + _func({"arg": {"val": 3.14}}) + + +def test_dict_of_dict_arg_give_list(): + @strict + def _func(_: Dict[str, Dict[str, float]]): + pass + + with pytest.raises(TypeError) as err: + _func({"arg": [1, 2, 3]}) + assert ( + str(err.value) + == 'Expected type typing.Dict[str, typing.Dict[str, float]] in "_func(_=typing.Dict[str, typing.Dict[str, float]])" got list' + ) + + +def test_dict_of_dict_of_list_arg(): + @strict + def _func(_: Dict[str, Dict[str, List[int]]]): + pass + + _func({"arg": {"val": [1, 2, 3]}}) + + +def test_return_dict(): + @strict + def _func() -> Dict[str, int]: + return {"one": 1, "two": 2} + + _func() diff --git a/python_typing_tests/test_full_stack.py b/python_typing_tests/test_full_stack.py deleted file mode 100644 index 0a7aef0..0000000 --- a/python_typing_tests/test_full_stack.py +++ /dev/null @@ -1,613 +0,0 @@ -import heapq -from typing import Dict, List, Optional, Set, Tuple, Union - -import pytest - -from python_typing.strict import strict - - -def test_empty(): - @strict - def _func(): - pass - - _func() - - -def test_empty_pass_arg(): - @strict - def _func(): - pass - - with pytest.raises(TypeError) as err: - _func("arg") - assert str(err.value) == "_func() takes 0 positional arguments but 1 was given" - - -def test_arg_pass_nothing(): - @strict - def _func(_): - pass - - with pytest.raises(TypeError) as err: - _func() - assert str(err.value) == "_func() missing 1 required positional argument: '_'" - - -def test_arg_no_type(): - @strict - def _func(_): - pass - - _func("arg") - - -def test_string_arg(): - @strict - def _func(_: str): - pass - - _func("arg") - - -def test_string_arg_gets_shortened(): - @strict - def _func(_: int): - pass - - with pytest.raises(TypeError) as err: - _func("loooooooooooooooong") - assert str(err.value) == 'Value (looooooooo..) in "_func(_=int)" is not of type int' - - -def test_int_arg(): - @strict - def _func(_: int): - pass - - _func(1) - - -def test_custom_arg(): - class CustomType: - pass - - @strict - def _func(_: CustomType): - pass - - custom_obj = CustomType() - _func(custom_obj) - - -def test_string_arg_fails(): - @strict - def _func(_: str): - pass - - with pytest.raises(TypeError) as err: - _func(1) - assert str(err.value) == 'Value (1) in "_func(_=str)" is not of type str' - - -def test_custom_arg_fails(): - class CustomType: - pass - - @strict - def _func(_: CustomType): - pass - - with pytest.raises(TypeError) as err: - _func(1) - assert ( - str(err.value) == 'Value (1) in "_func(_=CustomType)" is not of type CustomType' - ) - - -def test_list_arg(): - @strict - def _func(_: List[int]): - pass - - _func([1, 2, 3]) - - -def test_list_non_typing(): - @strict - def _func(_: list[int]): - pass - - _func([1, 2, 3]) - - -def test_list_non_typing_wrong_type(): - @strict - def _func(_: list[int]): - pass - - with pytest.raises(TypeError) as err: - _func(["a", "b", "c"]) - assert str(err.value) == 'Value (a) in "_func(_=list)" is not of type int' - - -def test_string_arg_expect_list(): - @strict - def _func(_: List[str]): - pass - - with pytest.raises(TypeError) as err: - _func("arg") - assert ( - str(err.value) - == 'Value (arg) in "_func(_=typing.List[str])" is not of type list' - ) - - -def test_list_arg_as_dict(): - @strict - def _func(_: List[int]): - pass - - with pytest.raises(TypeError) as err: - _func({"arg": 1}) - assert ( - str(err.value) - == 'Expected type typing.List[int] in "_func(_=typing.List[int])" got dict' - ) - - -def test_malformed_list_arg(): - @strict - def _func(_: List[int]): - pass - - with pytest.raises(TypeError) as err: - _func([1, 2, "whoops!", 3]) - assert ( - str(err.value) - == 'Value (whoops!) in "_func(_=typing.List[int])" is not of type int' - ) - - -def test_union_arg_first(): - @strict - def _func(_: Union[int, str]): - pass - - _func(1) - - -def test_union_arg_second(): - @strict - def _func(_: Union[int, str]): - pass - - _func("arg") - - -def test_union_arg_many(): - @strict - def _func(_: Union[int, str, bool, float]): - pass - - _func(3.14) - - -def test_union_arg_invalid(): - @strict - def _func(_: Union[int, str]): - pass - - with pytest.raises(TypeError) as err: - _func(3.14) - assert ( - str(err.value) - == 'Value (3.14) in "_func(_=typing.Union[int, str])" is not any of the valid types (int, str)' - ) - - -def test_malformed_list_arg_multiple_valid_types(): - @strict - def _func(_: List[Union[int, str]]): - pass - - _func([1, 2, "whoops!", 3]) - - -def test_dict_arg(): - @strict - def _func(_: Dict[str, int]): - pass - - _func({"arg": 1}) - - -def test_dict_non_typing_arg(): - @strict - def _func(_: dict[str, int]): - pass - - _func({"arg": 1}) - - -def test_dict_arg_as_list(): - @strict - def _func(_: Dict[str, int]): - pass - - with pytest.raises(TypeError) as err: - _func(["arg", 1]) - assert ( - str(err.value) - == 'Expected type typing.Dict[str, int] in "_func(_=typing.Dict[str, int])" got list' - ) - - -def test_dict_non_typing_arg_as_list(): - @strict - def _func(_: dict[str, int]): - pass - - with pytest.raises(TypeError) as err: - _func(["arg", 1]) - assert str(err.value) == 'Expected type dict in "_func(_=dict)" got list' - - -def test_dict_arg_invalid(): - @strict - def _func(_: Dict[str, int]): - pass - - with pytest.raises(TypeError) as err: - _func({1: 3.14}) - assert ( - str(err.value) - == 'Value (1) in "_func(_=typing.Dict[str, int])" is not of type str' - ) - - -def test_dict_arg_multiple_key_values(): - @strict - def _func(_: Dict[str, int]): - pass - - _func( - { - "arg1": 1, - "arg2": 2, - "arg3": 3, - } - ) - - -def test_dict_arg_malformed_multiple_key_values(): - @strict - def _func(_: Dict[str, int]): - pass - - with pytest.raises(TypeError) as err: - _func( - { - "arg1": 1, - 2: "arg2", - "arg3": 3, - } - ) - assert ( - str(err.value) - == 'Value (2) in "_func(_=typing.Dict[str, int])" is not of type str' - ) - - -def test_dict_arg_multiple_valid_types(): - @strict - def _func(_: Dict[Union[str, float], int]): - pass - - _func({"arg": 1}) - _func({3.14: 1}) - - -def test_dict_of_dict_arg(): - @strict - def _func(_: Dict[str, Dict[str, float]]): - pass - - _func({"arg": {"val": 3.14}}) - - -def test_dict_of_dict_arg_give_list(): - @strict - def _func(_: Dict[str, Dict[str, float]]): - pass - - with pytest.raises(TypeError) as err: - _func({"arg": [1, 2, 3]}) - assert ( - str(err.value) - == 'Expected type typing.Dict[str, typing.Dict[str, float]] in "_func(_=typing.Dict[str, typing.Dict[str, float]])" got list' - ) - - -def test_dict_of_dict_of_list_arg(): - @strict - def _func(_: Dict[str, Dict[str, List[int]]]): - pass - - _func({"arg": {"val": [1, 2, 3]}}) - - -def test_set_arg(): - @strict - def _func(_: Set[str]): - pass - - _func({"one", "two"}) - - -def test_set_arg_wrong_structure(): - @strict - def _func(_: Set[str]): - pass - - with pytest.raises(TypeError) as err: - _func(["one", "two"]) - assert ( - str(err.value) - == 'Expected type typing.Set[str] in "_func(_=typing.Set[str])" got list' - ) - - -def test_set_arg_with_union(): - @strict - def _func(_: Set[Union[str, int]]): - pass - - _func({"one", 2}) - - -def test_tuple_arg(): - @strict - def _func(_: Tuple[str]): - pass - - _func(("one", "two")) - - -def test_tuple_arg_wrong_structure(): - @strict - def _func(_: Tuple[str]): - pass - - with pytest.raises(TypeError) as err: - _func(["one", "two"]) - assert ( - str(err.value) - == 'Expected type typing.Tuple[str] in "_func(_=typing.Tuple[str])" got list' - ) - - -def test_return_str(): - @strict - def _func() -> str: - return "val" - - _func() - - -def test_return_int(): - @strict - def _func() -> int: - return 1 - - _func() - - -def test_return_str_expect_list(): - @strict - def _func() -> List[int]: - return 1 - - with pytest.raises(TypeError) as err: - _func() - assert ( - str(err.value) - == 'Return value (1) in "_func() -> typing.List[int]" is not of type list' - ) - - -def test_return_str_expect_int(): - @strict - def _func() -> int: - return "one" - - with pytest.raises(TypeError) as err: - _func() - assert str(err.value) == 'Return value (one) in "_func() -> int" is not of type int' - - -def test_return_dict(): - @strict - def _func() -> Dict[str, int]: - return {"one": 1, "two": 2} - - _func() - - -def test_return_list(): - @strict - def _func() -> List[int]: - return [1, 2, 3] - - _func() - - -def test_multiple_args(): - @strict - def _func(_1: int, _2: str): - pass - - _func(1, "two") - - -def test_vargs_is_ignored(): - tuple_ = (1, 2, 3) - - @strict - def _func(*args: int): - pass - - _func(*tuple_) - - -def test_kwargs(): - @strict - def _func(one: str = "one"): - pass - - _func(one="two") - - -# Testing defaults - - -def test_default(): - @strict - def _func(_: str = "one"): - pass - - _func() - - -def test_single_default(): - @strict - def _func(one: int, two: str = "one"): - pass - - _func(1) - - -def test_single_default_both_specified(): - @strict - def _func(one: int, two: str = "one"): - pass - - _func(1, "two") - - -def test_single_default_specified(): - @strict - def _func(one: int, two: str = "one"): - pass - - _func(1, two="two") - - -def test_single_specified_with_default(): - @strict - def _func(one: int, two: int, three: str = "three"): - pass - - _func(1, two=2) - - -def test_default_wrong_type(): - @strict - def _func(_: str = 1): - pass - - with pytest.raises(TypeError) as err: - _func() - assert str(err.value) == 'Value (1) in "_func(_=str)" is not of type str' - - -def test_one_arg_wrong(): - @strict - def _func(one: int, two: int, three: str = "three"): - pass - - with pytest.raises(TypeError) as err: - _func(1, two="oops!") - assert str(err.value) == 'Value (oops!) in "_func(two=int)" is not of type int' - - -def test_some_specified_some_not(): - @strict - def _func(one: str, two, three: int): - pass - - _func("arg", 2, 3) - - -# Optional[...] tests - - -def test_arg_optional(): - @strict - def _func(_: Optional[str]): - pass - - _func("arg") - - -def test_arg_optional_none_specified(): - @strict - def _func(_: Optional[str]): - # Note the arg isn't actually optional here - pass - - with pytest.raises(TypeError) as err: - _func() - assert str(err.value) == "_func() missing 1 required positional argument: '_'" - - -def test_arg_optional_give_none(): - @strict - def _func(_: Optional[str] = None): - pass - - _func() - - -def test_arg_optional_give_correct_default(): - @strict - def _func(_: Optional[str] = "string"): - pass - - _func() - - -def test_arg_optional_give_wrong_default(): - @strict - def _func(_: Optional[str] = 1): - pass - - with pytest.raises(TypeError) as err: - _func() - assert ( - str(err.value) - == 'Value (1) in "_func(_=typing.Optional[str])" is not any of the valid types (str, NoneType)' - ) - - -def test_heap(): - @strict - def _func(_: list): - pass - - q = [] - heapq.heappush(q, (1, "test")) - _func(q) - - -def test_heap_expect_dict(): - @strict - def _func(_: dict): - pass - - q = [] - heapq.heappush(q, (1, "test")) - with pytest.raises(TypeError) as err: - _func(q) - assert str(err.value) == 'Expected type dict in "_func(_=dict)" got list' diff --git a/python_typing_tests/test_list.py b/python_typing_tests/test_list.py new file mode 100644 index 0000000..b5715db --- /dev/null +++ b/python_typing_tests/test_list.py @@ -0,0 +1,99 @@ +from typing import List, Union + +import pytest + +from python_typing.strict import strict + + +def test_list_arg(): + @strict + def _func(_: List[int]): + pass + + _func([1, 2, 3]) + + +def test_list_non_typing(): + @strict + def _func(_: list[int]): + pass + + _func([1, 2, 3]) + + +def test_list_non_typing_wrong_type(): + @strict + def _func(_: list[int]): + pass + + with pytest.raises(TypeError) as err: + _func(["a", "b", "c"]) + assert str(err.value) == 'Value (a) in "_func(_=list)" is not of type int' + + +def test_string_arg_expect_list(): + @strict + def _func(_: List[str]): + pass + + with pytest.raises(TypeError) as err: + _func("arg") + assert ( + str(err.value) + == 'Value (arg) in "_func(_=typing.List[str])" is not of type list' + ) + + +def test_list_arg_as_dict(): + @strict + def _func(_: List[int]): + pass + + with pytest.raises(TypeError) as err: + _func({"arg": 1}) + assert ( + str(err.value) + == 'Expected type typing.List[int] in "_func(_=typing.List[int])" got dict' + ) + + +def test_malformed_list_arg(): + @strict + def _func(_: List[int]): + pass + + with pytest.raises(TypeError) as err: + _func([1, 2, "whoops!", 3]) + assert ( + str(err.value) + == 'Value (whoops!) in "_func(_=typing.List[int])" is not of type int' + ) + + +def test_malformed_list_arg_multiple_valid_types(): + @strict + def _func(_: List[Union[int, str]]): + pass + + _func([1, 2, "whoops!", 3]) + + +def test_return_str_expect_list(): + @strict + def _func() -> List[int]: + return 1 + + with pytest.raises(TypeError) as err: + _func() + assert ( + str(err.value) + == 'Return value (1) in "_func() -> typing.List[int]" is not of type list' + ) + + +def test_return_list(): + @strict + def _func() -> List[int]: + return [1, 2, 3] + + _func() diff --git a/python_typing_tests/test_set.py b/python_typing_tests/test_set.py new file mode 100644 index 0000000..fc0083a --- /dev/null +++ b/python_typing_tests/test_set.py @@ -0,0 +1,34 @@ +from typing import Set, Union + +import pytest + +from python_typing.strict import strict + + +def test_set_arg(): + @strict + def _func(_: Set[str]): + pass + + _func({"one", "two"}) + + +def test_set_arg_wrong_structure(): + @strict + def _func(_: Set[str]): + pass + + with pytest.raises(TypeError) as err: + _func(["one", "two"]) + assert ( + str(err.value) + == 'Expected type typing.Set[str] in "_func(_=typing.Set[str])" got list' + ) + + +def test_set_arg_with_union(): + @strict + def _func(_: Set[Union[str, int]]): + pass + + _func({"one", 2}) diff --git a/python_typing_tests/test_strict.py b/python_typing_tests/test_strict.py new file mode 100644 index 0000000..b8593aa --- /dev/null +++ b/python_typing_tests/test_strict.py @@ -0,0 +1,187 @@ +from typing import Callable, Union + +import pytest + +from python_typing.strict import strict + + +def test_empty(): + @strict + def _func(): + pass + + _func() + + +def test_empty_pass_arg(): + @strict + def _func(): + pass + + with pytest.raises(TypeError) as err: + _func("arg") + assert str(err.value) == "_func() takes 0 positional arguments but 1 was given" + + +def test_arg_pass_nothing(): + @strict + def _func(_): + pass + + with pytest.raises(TypeError) as err: + _func() + assert str(err.value) == "_func() missing 1 required positional argument: '_'" + + +def test_arg_no_type(): + @strict + def _func(_): + pass + + _func("arg") + + +def test_string_arg(): + @strict + def _func(_: str): + pass + + _func("arg") + + +def test_string_arg_gets_shortened(): + @strict + def _func(_: int): + pass + + with pytest.raises(TypeError) as err: + _func("loooooooooooooooong") + assert str(err.value) == 'Value (looooooooo..) in "_func(_=int)" is not of type int' + + +def test_int_arg(): + @strict + def _func(_: int): + pass + + _func(1) + + +def test_string_arg_fails(): + @strict + def _func(_: str): + pass + + with pytest.raises(TypeError) as err: + _func(1) + assert str(err.value) == 'Value (1) in "_func(_=str)" is not of type str' + + +def test_union_arg_first(): + @strict + def _func(_: Union[int, str]): + pass + + _func(1) + + +def test_union_arg_second(): + @strict + def _func(_: Union[int, str]): + pass + + _func("arg") + + +def test_union_arg_many(): + @strict + def _func(_: Union[int, str, bool, float]): + pass + + _func(3.14) + + +def test_union_arg_invalid(): + @strict + def _func(_: Union[int, str]): + pass + + with pytest.raises(TypeError) as err: + _func(3.14) + assert ( + str(err.value) + == 'Value (3.14) in "_func(_=typing.Union[int, str])" is not any of the valid types (int, str)' + ) + + +def test_return_str(): + @strict + def _func() -> str: + return "val" + + _func() + + +def test_return_int(): + @strict + def _func() -> int: + return 1 + + _func() + + +def test_return_str_expect_int(): + @strict + def _func() -> int: + return "one" + + with pytest.raises(TypeError) as err: + _func() + assert str(err.value) == 'Return value (one) in "_func() -> int" is not of type int' + + +def test_multiple_args(): + @strict + def _func(_1: int, _2: str): + pass + + _func(1, "two") + + +def test_vargs_is_ignored(): + tuple_ = (1, 2, 3) + + @strict + def _func(*args: int): + pass + + _func(*tuple_) + + +def test_kwargs(): + @strict + def _func(one: str = "one"): + pass + + _func(one="two") + + +def test_callable_arg(): + def _some_callable(): + pass + + @strict + def _func(_: Callable): + pass + + _func(_some_callable) + + +def test_callable_arg_not_callable(): + @strict + def _func(_: Callable): + pass + + with pytest.raises(TypeError) as err: + _func(1) + assert str(err.value) == 'Value (1) in "_func(_=typing.Callable)" is not of type Callable' diff --git a/python_typing_tests/test_tuple.py b/python_typing_tests/test_tuple.py new file mode 100644 index 0000000..4ceacba --- /dev/null +++ b/python_typing_tests/test_tuple.py @@ -0,0 +1,26 @@ +from typing import Tuple + +import pytest + +from python_typing.strict import strict + + +def test_tuple_arg(): + @strict + def _func(_: Tuple[str]): + pass + + _func(("one", "two")) + + +def test_tuple_arg_wrong_structure(): + @strict + def _func(_: Tuple[str]): + pass + + with pytest.raises(TypeError) as err: + _func(["one", "two"]) + assert ( + str(err.value) + == 'Expected type typing.Tuple[str] in "_func(_=typing.Tuple[str])" got list' + )