Files
2017-10-24 13:56:12 +01:00

346 lines
14 KiB
Python

# -*- coding: utf-8 -*-
"""Tests for the main app."""
from __future__ import (
absolute_import, division, print_function, unicode_literals,
)
import io
import itertools
import json
import logging
import os
import subprocess
from unittest import TestCase
import jupyter_core.paths
import nose.tools as nt
import yaml
from jupyter_contrib_core.notebook_compat import nbextensions
from jupyter_contrib_core.testing_utils import (
get_logger, patch_traitlets_app_logs,
)
from jupyter_contrib_core.testing_utils.jupyter_env import patch_jupyter_dirs
from jupyter_nbextensions_configurator import _process_nbextension_spec
from nose.plugins.skip import SkipTest
from traitlets.config import Config
from traitlets.tests.utils import check_help_all_output, check_help_output
import jupyter_contrib_nbextensions
from jupyter_contrib_nbextensions.application import main as main_app
from jupyter_contrib_nbextensions.application import (
BaseContribNbextensionsApp, BaseContribNbextensionsInstallApp,
ContribNbextensionsApp, InstallContribNbextensionsApp,
UninstallContribNbextensionsApp,
)
from jupyter_contrib_nbextensions.install import toggle_install
# attempt to use LibYaml if available
try:
from yaml import CSafeLoader as SafeLoader
except ImportError:
from yaml import SafeLoader
app_classes = (
BaseContribNbextensionsApp, BaseContribNbextensionsInstallApp,
ContribNbextensionsApp,
InstallContribNbextensionsApp, UninstallContribNbextensionsApp,
)
def _get_files_in_tree(tree_root_dir):
installed_files = []
for root, subdirs, files in os.walk(tree_root_dir, followlinks=True):
installed_files.extend([os.path.join(root, f) for f in files])
return installed_files
class AppTest(TestCase):
"""Tests for the main app."""
@classmethod
def setup_class(cls):
cls.log = cls.log = get_logger(cls.__name__)
cls.log.handlers = []
cls.log.propagate = True
def setUp(self):
"""Set up test fixtures for each test."""
(jupyter_patches, self.jupyter_dirs,
remove_jupyter_dirs) = patch_jupyter_dirs()
for ptch in jupyter_patches:
ptch.start()
self.addCleanup(ptch.stop)
self.addCleanup(remove_jupyter_dirs)
for klass in app_classes:
patch_traitlets_app_logs(klass)
klass.log_level.default_value = logging.DEBUG
def _check_install(self, dirs):
expected_write_dirs = dirs.values()
all_dirs = [
path
for dset in self.jupyter_dirs.values() if isinstance(dset, dict)
for name, path in dset.items() if name != 'root']
msgs = []
installed_files = []
for tree_dir in expected_write_dirs:
in_this_tree = _get_files_in_tree(tree_dir)
installed_files.extend(in_this_tree)
if not in_this_tree:
msgs.append(
'Expected files created in {} but found none.'.format(
tree_dir))
for tree_dir in [d for d in all_dirs if d not in expected_write_dirs]:
in_this_tree = _get_files_in_tree(tree_dir)
if in_this_tree:
msgs.append(
'Expected no files created in {} but found:\n\t{}'.format(
tree_dir, '\n\t'.join(in_this_tree)))
installed_files.extend(in_this_tree)
# check that dependency-provided nbexts got installed
if 'data' in dirs:
expected_require_paths = [
p.replace('/', os.path.sep) + '.js' for p in [
'highlight_selected_word/main',
'latex_envs/latex_envs',
]]
for req_part in expected_require_paths:
if not any([
p for p in installed_files if p.endswith(req_part)]):
msgs.append(
'Expected a file ending in {} but found none'.format(
req_part))
nt.assert_false(bool(msgs), '\n'.join(msgs))
return installed_files
def _check_uninstall(self, dirs, installed_files):
# check that nothing remains in the data directory
data_installed = [
path for path in installed_files
if path.startswith(dirs['data']) and os.path.exists(path)]
nt.assert_false(
data_installed,
'Uninstall should remove all data files from {}'.format(
dirs['data']))
# check the config directory
conf_installed = [
path for path in installed_files
if path.startswith(dirs['conf']) and os.path.exists(path)]
for path in conf_installed:
with open(path, 'r') as f:
conf = Config(json.load(f))
confstrip = {}
confstrip.update(conf)
# strip out config values we are ok to have remain
confstrip.pop('NotebookApp', None)
confstrip.pop('version', None)
conf_exts = confstrip.get('load_extensions', {})
conf_exts.pop('nbextensions_configurator/config_menu/main', None)
conf_exts.pop('nbextensions_configurator/tree_tab/main', None)
if not conf_exts:
confstrip.pop('load_extensions', None)
nt.assert_false(confstrip, 'disable should leave config empty.')
def _get_default_check_kwargs(self, argv=None, dirs=None):
if argv is None:
argv = []
if dirs is None:
dirs = {
'conf': jupyter_core.paths.SYSTEM_CONFIG_PATH[0],
'data': jupyter_core.paths.SYSTEM_JUPYTER_PATH[0],
}
return argv, dirs
def _call_main_app(self, argv):
main_app(argv=argv)
# a bit of a hack to allow initializing a new app instance
for klass in app_classes:
klass.clear_instance()
def _check_subproc(self, args):
proc = subprocess.Popen(
args, stdout=subprocess.PIPE, stderr=subprocess.STDOUT)
try:
output, unused_err = proc.communicate()
except subprocess.CalledProcessError:
proc.kill()
proc.wait()
raise
print(output.decode()) # this gets it captured by nose
retcode = proc.poll()
nt.assert_equal(retcode, 0, 'command should exit with code 0')
def check_app_install(self, argv=None, dirs=None, dirs_install=None):
"""Check files were installed in the correct place."""
argv, dirs = self._get_default_check_kwargs(argv, dirs)
if dirs_install is None:
dirs_install = dirs
self._call_main_app(argv=['install'] + argv)
installed_files = self._check_install(dirs_install)
self._call_main_app(argv=['uninstall'] + argv)
self._check_uninstall(dirs, installed_files)
def check_cli_install(self, argv=None, dirs=None, dirs_install=None,
app_name='jupyter contrib nbextension'):
argv, dirs = self._get_default_check_kwargs(argv, dirs)
if dirs_install is None:
dirs_install = dirs
args = app_name.split(' ') + ['install'] + argv
self._check_subproc(args)
installed_files = self._check_install(dirs_install)
args = app_name.split(' ') + ['uninstall'] + argv
self._check_subproc(args)
self._check_uninstall(dirs, installed_files)
def test_00_extra_args(self):
"""Check that app complains about extra args."""
for subcom in ('install', 'uninstall'):
# sys.exit should be called if extra args specified
with nt.assert_raises(SystemExit):
main_app([subcom, 'arbitrary_extension_name'])
for klass in app_classes:
klass.clear_instance()
def test_01_help_output(self):
"""Check that app help works."""
app_module = 'jupyter_contrib_nbextensions.application'
for subcommand in (None, ['install'], ['uninstall']):
check_help_output(app_module, subcommand=subcommand)
check_help_all_output(app_module, subcommand=subcommand)
# sys.exit should be called if empty argv specified
with nt.assert_raises(SystemExit):
main_app([])
for klass in app_classes:
klass.clear_instance()
def test_02_argument_conflict(self):
"""Check that install objects to multiple flags."""
conflicting_flags = ('--user', '--system', '--sys-prefix')
conflicting_flagsets = []
for nn in range(2, len(conflicting_flags) + 1):
conflicting_flagsets.extend(
itertools.combinations(conflicting_flags, nn))
for subcommand in ('install', 'uninstall'):
for flagset in conflicting_flagsets:
self.log.info('testing conflicting flagset {}'.format(flagset))
nt.assert_raises(nbextensions.ArgumentConflict,
main_app, [subcommand] + list(flagset))
for klass in app_classes:
klass.clear_instance()
conflicting_kwargs = dict(
user=True, sys_prefix=True, prefix='/tmp/blah',
nbextensions_dir='/tmp/nbext/blah')
for nn in range(2, len(conflicting_kwargs) + 1):
for kset in itertools.combinations(conflicting_kwargs.keys(), nn):
kwargs = {k: conflicting_kwargs[k] for k in kset}
self.log.info('testing conflicting kwargs {}'.format(kwargs))
with nt.assert_raises(nbextensions.ArgumentConflict):
toggle_install(True, **kwargs)
def test_03_app_install_defaults(self):
"""Check that app install works correctly using defaults."""
self.check_app_install()
# don't test cli install with defaults, as we can't patch system
# directories in the subprocess
def test_05_app_install_user(self):
"""Check that app install works correctly using --user flag."""
self.check_app_install(
argv=['--user'], dirs=self.jupyter_dirs['env_vars'])
def test_06_cli_install_user(self):
"""Check that cli install works correctly using --user flag."""
self.check_cli_install(
argv=['--user'], dirs=self.jupyter_dirs['env_vars'])
def test_07_app_install_sys_prefix(self):
"""Check that app install works correctly using --sys-prefix flag."""
self.check_app_install(
dirs=self.jupyter_dirs['sys_prefix'], argv=['--sys-prefix'])
# don't test cli install with --sys-prefix flag, as we can't patch
# directories in the subprocess
def test_08_app_install_system(self):
"""Check that app install works correctly using --system flag."""
self.check_app_install(
dirs=self.jupyter_dirs['system'], argv=['--system'])
# don't test cli install with --system flag, as we can't patch
# directories in the subprocess
def test_09_app_install_symlink(self):
"""Check that app install works correctly using --symlink flag."""
if os.name in ('nt', 'dos'):
raise SkipTest('symlinks are not supported on Windows.')
self.check_app_install(argv=['--symlink'])
def test_10_cli_install_symlink(self):
"""Check that cli install works correctly using --symlink flag."""
if os.name in ('nt', 'dos'):
raise SkipTest('symlinks are not supported on Windows.')
self.check_cli_install(
argv=['--user', '--symlink'], dirs=self.jupyter_dirs['env_vars'])
def test_11_app_install_nbextensions_dir(self):
"""Check that app install works correctly using --nbextensions arg."""
dirs = self._get_default_check_kwargs()[1]
dirs['data'] = self.jupyter_dirs['custom']['data']
nbext_dir = os.path.join(dirs['data'], 'nbextensions')
self.check_app_install(dirs=dirs, argv=['--nbextensions=' + nbext_dir])
# We can't test cli install using nbextensions_dir, since it edits system
# config, and we can't patch directories in the subprocess
def test_12_app_plural_alias(self):
"""Check that app works correctly when using 'nbextensions' plural."""
self.check_cli_install(
argv=['--user'], dirs=self.jupyter_dirs['env_vars'],
app_name='jupyter contrib nbextensions')
def test_13_app_install_prefix(self):
"""Check that app install works correctly using --prefix arg."""
dirs = self._get_default_check_kwargs()[1]
dirs['data'] = self.jupyter_dirs['custom']['data']
self.check_app_install(dirs=dirs, argv=['--prefix=' + dirs['data']])
def test_14_app_install_only_files(self):
"""Check that install works correctly using --only-files flag."""
argv, dirs = self._get_default_check_kwargs()
self.check_app_install(
argv=argv + ['--only-files'], dirs=dirs,
dirs_install={'data': dirs['data']})
def test_15_app_install_only_config(self):
"""Check that install works correctly using --only-config flag."""
argv, dirs = self._get_default_check_kwargs()
self.check_app_install(
argv=argv + ['--only-config'], dirs=dirs,
dirs_install={'conf': dirs['conf']})
def test_16_yaml_files(self):
"""Check that the configurator accepts all nbextension yaml files."""
errmsg = ''
root_nbext_dir = os.path.join(os.path.dirname(
jupyter_contrib_nbextensions.__file__), 'nbextensions')
for direct, dirs, files in os.walk(root_nbext_dir, followlinks=True):
for fname in files:
if os.path.splitext(fname)[1] not in ('.yml', '.yaml'):
continue
yaml_path = os.path.join(direct, fname)
with io.open(yaml_path, 'r', encoding='utf-8') as stream:
try:
spec = yaml.load(stream, Loader=SafeLoader)
except yaml.YAMLError:
errmsg += '\nFailed to load yaml: {}'.format(yaml_path)
continue
spec = _process_nbextension_spec(spec)
if not isinstance(spec, dict):
errmsg += '\n{}: {}'.format(spec, os.path.reyaml_path)
nt.assert_false(len(errmsg), 'Error loading yaml file(s):' + errmsg)