-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathtest_match_nx.py
238 lines (194 loc) · 8.16 KB
/
test_match_nx.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
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
"""Test that `graphblas.nxapi` structure matches that of networkx.
This currently checks the locations and names of all networkx-dispatchable functions
that are implemented by `graphblas_algorithms`. It ignores names that begin with `_`.
The `test_dispatched_funcs_in_nxap` test below will say what to add and delete under `nxapi`.
We should consider removing any test here that becomes too much of a nuisance.
For now, though, let's try to match and stay up-to-date with NetworkX!
"""
import sys
from collections import namedtuple
from pathlib import Path
import pytest
try:
import networkx as nx # noqa: F401
except ImportError:
pytest.skip(
"Matching networkx namespace requires networkx to be installed", allow_module_level=True
)
else:
try:
from networkx.utils import backends
IS_NX_30_OR_31 = False
except ImportError: # pragma: no cover (import)
# This is the location in nx 3.1
from networkx.classes import backends # noqa: F401
IS_NX_30_OR_31 = True
def isdispatched(func):
"""Can this NetworkX function dispatch to other backends?"""
if IS_NX_30_OR_31:
return (
callable(func)
and hasattr(func, "dispatchname")
and func.__module__.startswith("networkx")
)
return (
callable(func)
and hasattr(func, "preserve_edge_attrs")
and func.__module__.startswith("networkx")
)
def dispatchname(func):
"""The dispatched name of the dispatchable NetworkX function."""
# Haha, there should be a better way to get this
if not isdispatched(func):
raise ValueError(f"Function is not dispatched in NetworkX: {func.__name__}")
if IS_NX_30_OR_31:
return func.dispatchname
return func.name
def fullname(func):
return f"{func.__module__}.{func.__name__}"
NameInfo = namedtuple("NameInfo", ["dispatchname", "fullname", "curpath"])
@pytest.fixture(scope="module")
def nx_info():
rv = {} # {modulepath: {dispatchname: NameInfo}}
for modname, module in sys.modules.items():
cur = {}
if not modname.startswith("networkx.") and modname != "networkx" or "tests" in modname:
continue
for key, val in vars(module).items():
if not key.startswith("_") and isdispatched(val):
dname = dispatchname(val)
cur[dname] = NameInfo(dname, fullname(val), f"{modname}.{key}")
if cur:
rv[modname] = cur
return rv
@pytest.fixture(scope="module")
def gb_info():
rv = {} # {modulepath: {dispatchname: NameInfo}}
from graphblas_algorithms import nxapi # noqa: F401
from graphblas_algorithms.interface import Dispatcher
ga_map = {
fullname(val): key
for key, val in vars(Dispatcher).items()
if callable(val) and fullname(val).startswith("graphblas_algorithms.nxapi.")
}
for modname, module in sys.modules.items():
cur = {}
if not modname.startswith("graphblas_algorithms.nxapi") or "tests" in modname:
continue
for key, val in vars(module).items():
try:
fname = fullname(val)
except Exception:
continue
if key.startswith("_") or fname not in ga_map:
continue
dname = ga_map[fname]
cur[dname] = NameInfo(dname, fname, f"{modname}.{key}")
if cur:
rv[modname] = cur
return rv
@pytest.fixture(scope="module")
def nx_names_to_info(nx_info):
rv = {} # {dispatchname: {NameInfo}}
for names in nx_info.values():
for name, info in names.items():
if name not in rv:
rv[name] = set()
rv[name].add(info)
return rv
@pytest.fixture(scope="module")
def gb_names_to_info(gb_info):
rv = {} # {dispatchname: {NameInfo}}
for names in gb_info.values():
for name, info in names.items():
if name not in rv:
rv[name] = set()
rv[name].add(info)
return rv
@pytest.mark.checkstructure
def test_nonempty(nx_info, gb_info, nx_names_to_info, gb_names_to_info):
assert len(nx_info) > 15
assert len(gb_info) > 15
assert len(nx_names_to_info) > 30
assert len(gb_names_to_info) > 30
def nx_to_gb_info(info):
gb = "graphblas_algorithms.nxapi"
return NameInfo(
info[0],
info[1].replace("networkx.algorithms", gb).replace("networkx", gb),
info[2].replace("networkx.algorithms", gb).replace("networkx", gb),
)
def module_exists(info):
return info[2].rsplit(".", 1)[0] in sys.modules
@pytest.mark.checkstructure
def test_dispatched_funcs_in_nxapi(nx_names_to_info, gb_names_to_info):
"""Are graphblas_algorithms functions in the correct locations in nxapi?"""
failing = False
for name in nx_names_to_info.keys() & gb_names_to_info.keys():
nx_paths = {
gbinfo
for info in nx_names_to_info[name]
if module_exists(gbinfo := nx_to_gb_info(info))
}
gb_paths = gb_names_to_info[name]
if nx_paths != gb_paths: # pragma: no cover
failing = True
if missing := (nx_paths - gb_paths):
from_ = ":".join(next(iter(missing))[1].rsplit(".", 1))
print(f"Add `{name}` from `{from_}` here:")
for _, _, path in sorted(missing):
print(" ", ":".join(path.rsplit(".", 1)))
if extra := (gb_paths - nx_paths):
from_ = ":".join(next(iter(extra))[1].rsplit(".", 1))
print(f"Remove `{name}` from `{from_}` here:")
for _, _, path in sorted(extra):
print(" ", ":".join(path.rsplit(".", 1)))
if failing: # pragma: no cover
raise AssertionError
def get_fullname(info):
fullname = info.fullname
if not fullname.endswith(f".{info.dispatchname}"):
fullname += f" ({info.dispatchname})"
return fullname
def test_print_dispatched_not_implemented(nx_names_to_info, gb_names_to_info):
"""It may be informative to see the results from this to identify functions to implement.
$ pytest -s -k test_print_dispatched_not_implemented
"""
not_implemented = nx_names_to_info.keys() - gb_names_to_info.keys()
fullnames = {get_fullname(next(iter(nx_names_to_info[name]))) for name in not_implemented}
print()
print("=================================================================================")
print("Functions dispatched in NetworkX that ARE NOT implemented in graphblas-algorithms")
print("---------------------------------------------------------------------------------")
for i, name in enumerate(sorted(fullnames)):
print(i, name)
print("=================================================================================")
def test_print_dispatched_implemented(nx_names_to_info, gb_names_to_info):
"""It may be informative to see the results from this to identify implemented functions.
$ pytest -s -k test_print_dispatched_implemented
"""
implemented = nx_names_to_info.keys() & gb_names_to_info.keys()
fullnames = {get_fullname(next(iter(nx_names_to_info[name]))) for name in implemented}
print()
print("=============================================================================")
print("Functions dispatched in NetworkX that ARE implemented in graphblas-algorithms")
print("-----------------------------------------------------------------------------")
for i, name in enumerate(sorted(fullnames)):
print(i, name)
print("=============================================================================")
def test_algorithms_in_readme(nx_names_to_info, gb_names_to_info):
"""Ensure all algorithms are mentioned in README.md."""
implemented = nx_names_to_info.keys() & gb_names_to_info.keys()
path = Path(__file__).parent.parent.parent / "README.md"
if not path.exists():
return
with path.open("r") as f:
text = f.read()
missing = set()
for name in sorted(implemented):
if name not in text:
missing.add(name)
if missing:
msg = f"Algorithms missing in README.md: {', '.join(sorted(missing))}"
print(msg)
raise AssertionError(msg)