-
-
Notifications
You must be signed in to change notification settings - Fork 12
/
circular_imports_test.py
94 lines (76 loc) · 2.9 KB
/
circular_imports_test.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
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
"""Tests for circular imports in all local packages and modules.
This ensures all internal packages can be imported right away without
any need to import some other module before doing so.
This module is based on an idea that pytest uses for self-testing:
* https://github.com/pytest-dev/pytest/blob/d18c75b/testing/test_meta.py
* https://twitter.com/codewithanthony/status/1229445110510735361
"""
from itertools import chain
from pathlib import Path
import os
import pkgutil
import subprocess
import sys
import pytest
import octomachinery
def _find_all_importables(pkg):
"""Find all importables in the project.
Return them in order.
"""
return sorted(
set(
chain.from_iterable(
_discover_path_importables(Path(p), pkg.__name__)
for p in pkg.__path__
),
),
)
def _discover_path_importables(pkg_pth, pkg_name):
"""Yield all importables under a given path and package."""
for dir_path, _d, file_names in os.walk(pkg_pth):
pkg_dir_path = Path(dir_path)
if pkg_dir_path.parts[-1] == '__pycache__':
continue
if all(Path(_).suffix != '.py' for _ in file_names):
continue
rel_pt = pkg_dir_path.relative_to(pkg_pth)
pkg_pref = '.'.join((pkg_name, ) + rel_pt.parts)
yield from (
pkg_path
for _, pkg_path, _ in pkgutil.walk_packages(
(str(pkg_dir_path), ), prefix=f'{pkg_pref}.',
)
)
@pytest.mark.parametrize(
'import_path',
_find_all_importables(octomachinery),
)
def test_no_warnings(import_path):
"""Verify that exploding importables doesn't explode.
This is seeking for any import errors including ones caused
by circular imports.
"""
imp_cmd = (
sys.executable,
'-W', 'error',
# NOTE: These are necessary for `tox -e old-deps`:
'-W', "ignore:Using or importing the ABCs from 'collections' instead "
"of from 'collections.abc' is deprecated since "
'Python 3.3, and in 3.10 it will stop working:'
'DeprecationWarning:jwt.api_jwt',
'-W', "ignore:Using or importing the ABCs from 'collections' instead "
"of from 'collections.abc' is deprecated since "
'Python 3.3, and in 3.9 it will stop working:'
'DeprecationWarning:jwt.api_jwt',
# NOTE: This looks like the line above but has a typo
# NOTE: (a whitespace is missing):
'-W', "ignore:Using or importing the ABCs from 'collections' instead "
"of from 'collections.abc' is deprecated since "
'Python 3.3,and in 3.9 it will stop working:'
'DeprecationWarning:jwt.api_jwt',
'-W', 'ignore:"@coroutine" decorator is deprecated '
'since Python 3.8, use "async def" instead:'
'DeprecationWarning:aiohttp.helpers',
'-c', f'import {import_path!s}',
)
subprocess.check_call(imp_cmd)