-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathconftest.py
66 lines (55 loc) · 3.06 KB
/
conftest.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
"""Global fixtures for pytest."""
from typing import Callable
import pytest
@pytest.fixture
def function_tester() -> Callable:
"""Create a reusable fixture to run a basic test case against a function."""
def function_tester(test: dict, func: Callable, compare: Callable | None = None) -> None:
"""Run a basic test case against a function.
Test case guidelines:
- If testing function output, use: args and/or kwargs + returns
- If testing function exceptions, use: args and/or kwargs + raises
- If testing class creation, use class' __call__ in the test, and use: args and/or kwargs + attributes
Args:
test: Configuration parameters for testing a callable.
Optional keys: "args" and/or "kwargs"
Mandatory keys: "returns" or "raises" or "attributes"
func: A callable to pass args and kwargs to, and capture results/errors from.
compare: A function to use for comparing the actual result and expected result.
Defaults to a "==" comparison.
"""
result_types_found = [name in test for name in ("returns", "raises", "attributes")]
if not result_types_found.count(True):
raise ValueError("Test must declare one of: returns, raises, attributes")
if result_types_found.count(True) > 1:
raise ValueError("Test must declare only one of: returns, raises, attributes")
args = test.get("args", [])
kwargs = test.get("kwargs", {})
raises = test.get("raises")
if raises:
with pytest.raises(raises):
func(*args, **kwargs)
else:
expected = test.get("returns")
attributes = test.get("attributes")
if attributes is not None and not attributes:
raise ValueError("Test attributes result type must have values")
result = func(*args, **kwargs)
if attributes:
attribute_results = {name: getattr(result, name) for name in attributes.keys()}
equals = compare(attribute_results, attributes) if compare else attribute_results == attributes
assert equals, f"\nResult:\n\t{attribute_results}\nExpected:\n\t{attributes}"
else:
equals = compare(result, expected) if compare else expected == result
assert equals, f"\nResult:\n\t{result}\nExpected:\n\t{expected}"
return function_tester
def pytest_generate_tests(metafunc: pytest.Metafunc) -> None:
"""Process custom paramtrize options to streamline sets of tests with easily identifiable ids."""
mark = metafunc.definition.get_closest_marker("parametrize_test_case")
if mark:
args = list(mark.args)
test_case = args[1]
args[1] = [value for value in (list(test_case.values()) if isinstance(test_case, dict) else test_case)]
kwargs = mark.kwargs
kwargs["ids"] = [str(value) for value in (list(test_case.keys()) if isinstance(test_case, dict) else test_case)]
metafunc.parametrize(*args, **kwargs)