#|hide
#default_exp test
from nbdev.showdoc import show_doc
#|export
from nbdev.imports import *
from nbdev.sync import *
from nbdev.export import *
from nbdev.export import _mk_flag_re
from nbdev.export2html import _re_notebook2script
from fastcore.script import *
from fastcore.parallel import *
from nbconvert.preprocessors import ExecutePreprocessor
import nbformat
The functions that grab the cells containing tests (filtering with potential flags) and execute them
Everything that is not an exported cell is considered a test, so you should make sure your notebooks can all run smoothly (and fast) if you want to use this functionality as the CLI. You can mark some cells with special flags (like slow) to make sure they are only executed when you authorize it. Those flags should be configured in your settings.ini
(separated by a |
if you have several of them). You can also apply flags to one entire notebook by using the all
option, e.g. #all_slow
, in code cells.
If tst_flags=slow|fastai
in settings.ini
, you can:
#slow
flag#fastai
flag.The following functions detect the cells that should be excluded from the tests (unless their special flag is passed).
#|export
class _ReTstFlags():
"Test flag matching regular expressions"
def __init__(self, all_flag):
"match flags applied to all cells?"
self.all_flag = all_flag
def _deferred_init(self):
"Compile at first use but not before since patterns need `get_config().tst_flags`"
if hasattr(self, '_re'): return
tst_flags = get_config().get('tst_flags', '')
tst_flags += f'|skip' if tst_flags else 'skip'
_re_all = 'all_' if self.all_flag else ''
self._re = _mk_flag_re(f"{_re_all}({tst_flags})", 0, "Any line with a test flag")
def findall(self, source):
self._deferred_init()
return self._re.findall(source)
def search(self, source):
self._deferred_init()
return self._re.search(source)
#|export
_re_all_flag = _ReTstFlags(True)
#|export
def get_all_flags(cells):
"Check for all test flags in `cells`"
result = []
for cell in cells:
if cell['cell_type'] == 'code': result.extend(_re_all_flag.findall(cell['source']))
return set(result)
nb = read_nb("04_test.ipynb")
assert get_all_flags(nb['cells']) == set()
#|hide
tst_flags_bck=get_config().get('tst_flags')
try:
get_config()['tst_flags'] = 'fastai|vslow'
if hasattr(_re_all_flag, '_re'): del _re_all_flag._re
cells = [{'cell_type': cell_type, 'source': source} for cell_type, source in [
('code', '# export\nfrom local.core import *'),
('markdown', '# title of some kind'),
('code', '# all_vslow \n# all_fastai'),
('code', '#all_vslow\n# all_fastai'),
('code', '#all_vslow\n#all_skip'),
('code', '# all_fastai'),
('code', '#all_fastai\n')]]
for i in range(3):
test_eq(set(['vslow','fastai','skip']), get_all_flags(cells))
cells.pop(2)
for i in range(2):
test_eq(set(['fastai']), get_all_flags(cells))
cells.pop(2)
test_eq(set(), get_all_flags(cells))
finally:
get_config()['tst_flags'] = tst_flags_bck
del _re_all_flag._re
#|export
_re_flags = _ReTstFlags(False)
#|export
def get_cell_flags(cell):
"Check for any special test flag in `cell`"
if cell['cell_type'] != 'code' or len(get_config().get('tst_flags',''))==0: return []
return _re_flags.findall(cell['source'])
test_eq(get_cell_flags({'cell_type': 'code', 'source': "#hide\n"}), [])
#|hide
from nbformat.v4 import new_code_cell
for expected, flag in [(['fastai'], 'fastai'), ([], 'vslow')]:
test_eq(expected, get_cell_flags(new_code_cell(f"#hide\n# {flag}\n")))
test_eq(expected, get_cell_flags(new_code_cell(f"# {flag}\n#hide\n")))
test_eq(expected, get_cell_flags(new_code_cell(f"#{flag}\n#hide\n")))
test_eq([], get_cell_flags(new_code_cell("#hide\n")))
test_eq([], get_cell_flags(new_code_cell(f"# all_{flag}")))
test_eq([], get_cell_flags(new_code_cell(f"#all_{flag}")))
tst_flags_bck=get_config().get('tst_flags')
try:
get_config()['tst_flags'] = 'fastai|vslow'
del _re_flags._re
test_eq(['vslow'], get_cell_flags(new_code_cell(f"#hide\n# vslow\n")))
test_eq(['vslow'], get_cell_flags(new_code_cell(f"#hide\n#vslow\n")))
test_eq(['vslow', 'fastai'], get_cell_flags(new_code_cell(f"#hide\n# vslow\n# fastai")))
test_eq(['fastai', 'vslow'], get_cell_flags(new_code_cell(f"#fastai\n#vslow")))
finally:
get_config()['tst_flags'] = tst_flags_bck
del _re_flags._re
#|export
class NoExportPreprocessor(ExecutePreprocessor):
"An `ExecutePreprocessor` that executes cells that don't have a flag in `flags`"
def __init__(self, flags, **kwargs):
self.flags = flags
super().__init__(**kwargs)
def preprocess_cell(self, cell, resources, index):
if 'source' not in cell or cell['cell_type'] != "code": return cell, resources
for f in get_cell_flags(cell):
if f not in self.flags: return cell, resources
if check_re(cell, _re_notebook2script): return cell, resources
return super().preprocess_cell(cell, resources, index)
#|export
def test_nb(fn, flags=None):
"Execute tests in notebook in `fn` with `flags`"
os.environ["IN_TEST"] = '1'
if flags is None: flags = []
try:
nb = read_nb(fn)
for f in get_all_flags(nb['cells']):
if f not in flags: return
ep = NoExportPreprocessor(flags, timeout=600, kernel_name='python3')
pnb = nbformat.from_dict(nb)
ep.preprocess(pnb, {})
finally: os.environ.pop("IN_TEST")
test_nb('index.ipynb')
#|export
def _test_one(fname, flags=None, verbose=True):
print(f"testing {fname}")
start = time.time()
try:
test_nb(fname, flags=flags)
return True,time.time()-start
except Exception as e:
if "ZMQError" in str(e): _test_one(item, flags=flags, verbose=verbose)
if verbose: print(f'Error in {fname}:\n{e}')
return False,time.time()-start
#|export
@call_parse
def nbdev_test_nbs(
fname:str=None, # A notebook name or glob to convert
flags:str=None, # Space separated list of flags
n_workers:int=None, # Number of workers to use
verbose:bool_arg=True, # Print errors along the way
timing:bool=False, # Timing each notebook to see the ones are slow
pause:float=0.5 # Pause time (in secs) between notebooks to avoid race conditions
):
"Test in parallel the notebooks matching `fname`, passing along `flags`"
if flags is not None: flags = flags.split(' ')
files = nbglob(fname)
files = [Path(f).absolute() for f in sorted(files)]
assert len(files) > 0, "No files to test found."
if n_workers is None: n_workers = 0 if len(files)==1 else min(num_cpus(), 8)
# make sure we are inside the notebook folder of the project
os.chdir(get_config().path("nbs_path"))
results = parallel(_test_one, files, flags=flags, verbose=verbose, n_workers=n_workers, pause=pause)
passed,times = [r[0] for r in results],[r[1] for r in results]
if all(passed): print("All tests are passing!")
else:
msg = "The following notebooks failed:\n"
raise Exception(msg + '\n'.join([f.name for p,f in zip(passed,files) if not p]))
if timing:
for i,t in sorted(enumerate(times), key=lambda o:o[1], reverse=True):
print(f"Notebook {files[i].name} took {int(t)} seconds")
#|export
@call_parse
def nbdev_read_nbs(
fname:str=None # A notebook name or glob to convert
):
"Check all notebooks matching `fname` can be opened"
files = nbglob(fname)
for nb in files:
try: _ = read_nb(nb)
except Exception as e:
print(f"{nb} is corrupted and can't be opened.")
raise e
#|hide
from nbdev.export import *
notebook2script()
Converted 00_export.ipynb. Converted 01_sync.ipynb. Converted 02_showdoc.ipynb. Converted 03_export2html.ipynb. Converted 04_test.ipynb. Converted 05_merge.ipynb. Converted 06_cli.ipynb. Converted 07_clean.ipynb. Converted 99_search.ipynb. Converted example.ipynb. Converted index.ipynb. Converted nbdev_comments.ipynb. Converted tutorial.ipynb. Converted tutorial_colab.ipynb.