-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathtest_notebooks.py
137 lines (110 loc) · 4.44 KB
/
test_notebooks.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
#!/usr/bin/env python
# coding: utf-8
import os
import sys
import unittest
from pathlib import Path
import nbformat
import climada
NOTEBOOK_DIR = Path(__file__).parent.joinpath('doc', 'tutorial')
'''The path to the notebook directories.'''
BOUND_TO_FAIL = '# Note: execution of this cell will fail'
'''Cells containing this line will not be executed in the test'''
class NotebookTest(unittest.TestCase):
'''Generic TestCase for testing the executability of notebooks
Attributes
----------
wd : str
Absolute Path to the working directory, i.e., the directory of the notebook.
notebook : str
File name of the notebook.
'''
def __init__(self, methodName, wd=None, notebook=None):
super(NotebookTest, self).__init__(methodName)
self.wd = wd
self.notebook = notebook
def test_notebook(self):
'''Extracts code cells from the notebook and executes them one by one, using `exec`.
Magic lines and help/? calls are eliminated.
Cells containing `BOUND_TO_FAIL` are elided.
Cells doing multiprocessing are elided.'''
cwd = Path.cwd()
try:
# cd to the notebook directory
os.chdir(self.wd)
print(f'start testing {self.notebook}')
# read the notebook into a string
with open(self.notebook, encoding='utf8') as nb:
content = nb.read()
# parse the string with nbformat.reads
cells = nbformat.reads(content, 4)['cells']
namespace = dict()
for i, c in enumerate(cells):
# skip markdown cells
if c['cell_type'] != 'code': continue
# skip deliberately failing cells
if BOUND_TO_FAIL in c['source']: continue
# skip multiprocessing cells
if any([ tabu in c['source'].split() for tabu in [
'pathos.pools',
'mulitprocessing',
]]):
print('\n'.join([
f'\nskip multiprocessing cell {i} in {self.notebook}',
'+'+'-'*68+'+',
c['source']
]))
continue
# remove non python lines and help calls which require user input
python_code = "\n".join([ln for ln in c['source'].split("\n")
if not ln.startswith('%')
and not ln.startswith('help(')
and not ln.startswith('ask_ok(')
and not ln.strip().endswith('?')
])
# execute the python code
try:
exec(python_code, namespace)
# report failures
except Exception as e:
failure = "\n".join([
f"notebook {self.notebook} cell {i} failed with {e.__class__}",
f"{e}",
'+'+'-'*68+'+',
c['source']
])
print(f'failed {self.notebook}')
print(failure)
self.fail(failure)
print(f'succeeded {self.notebook}')
finally:
os.chdir(cwd)
def main():
# list notebooks in the NOTEBOOK_DIR
notebooks = [f.absolute()
for f in sorted(NOTEBOOK_DIR.iterdir())
if os.path.splitext(f)[1] == ('.ipynb')]
# build a test suite with a test for each notebook
suite = unittest.TestSuite()
for notebook in notebooks:
class NBTest(NotebookTest): pass
test_name = "_".join(notebook.stem.split())
setattr(NBTest, test_name, NBTest.test_notebook)
suite.addTest(NBTest(test_name, notebook.parent, notebook.name))
# run the tests depending on the first input argument: None or 'report'.
# write xml reports for 'report'
if sys.argv[1:]:
arg = sys.argv[1]
if arg == 'report':
import xmlrunner
outdirstr = str(Path(__file__).parent.joinpath('tests_xml'))
xmlrunner.XMLTestRunner(output=outdirstr).run(suite)
else:
jd, nb = os.path.split(arg)
unittest.TextTestRunner(verbosity=2).run(NotebookTest('test_notebook', jd, nb))
# with no argument just run the test
else:
unittest.TextTestRunner(verbosity=2).run(suite)
if __name__ == '__main__':
sys.path.append(str(Path.cwd()))
main()