test_kernelspec.py 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199
  1. # coding: utf-8
  2. """Tests for the KernelSpecManager"""
  3. # Copyright (c) Jupyter Development Team.
  4. # Distributed under the terms of the Modified BSD License.
  5. import copy
  6. import io
  7. import json
  8. from logging import StreamHandler
  9. import os
  10. from os.path import join as pjoin
  11. from subprocess import Popen, PIPE, STDOUT
  12. import sys
  13. import tempfile
  14. import unittest
  15. import pytest
  16. if str is bytes: # py2
  17. StringIO = io.BytesIO
  18. else:
  19. StringIO = io.StringIO
  20. from ipython_genutils.tempdir import TemporaryDirectory
  21. from jupyter_client import kernelspec
  22. from jupyter_core import paths
  23. from .utils import test_env
  24. sample_kernel_json = {'argv':['cat', '{connection_file}'],
  25. 'display_name':'Test kernel',
  26. }
  27. class KernelSpecTests(unittest.TestCase):
  28. def _install_sample_kernel(self, kernels_dir):
  29. """install a sample kernel in a kernels directory"""
  30. sample_kernel_dir = pjoin(kernels_dir, 'sample')
  31. os.makedirs(sample_kernel_dir)
  32. json_file = pjoin(sample_kernel_dir, 'kernel.json')
  33. with open(json_file, 'w') as f:
  34. json.dump(sample_kernel_json, f)
  35. return sample_kernel_dir
  36. def setUp(self):
  37. self.env_patch = test_env()
  38. self.env_patch.start()
  39. self.sample_kernel_dir = self._install_sample_kernel(
  40. pjoin(paths.jupyter_data_dir(), 'kernels'))
  41. self.ksm = kernelspec.KernelSpecManager()
  42. td2 = TemporaryDirectory()
  43. self.addCleanup(td2.cleanup)
  44. self.installable_kernel = td2.name
  45. with open(pjoin(self.installable_kernel, 'kernel.json'), 'w') as f:
  46. json.dump(sample_kernel_json, f)
  47. def tearDown(self):
  48. self.env_patch.stop()
  49. def test_find_kernel_specs(self):
  50. kernels = self.ksm.find_kernel_specs()
  51. self.assertEqual(kernels['sample'], self.sample_kernel_dir)
  52. def test_get_kernel_spec(self):
  53. ks = self.ksm.get_kernel_spec('SAMPLE') # Case insensitive
  54. self.assertEqual(ks.resource_dir, self.sample_kernel_dir)
  55. self.assertEqual(ks.argv, sample_kernel_json['argv'])
  56. self.assertEqual(ks.display_name, sample_kernel_json['display_name'])
  57. self.assertEqual(ks.env, {})
  58. self.assertEqual(ks.metadata, {})
  59. def test_find_all_specs(self):
  60. kernels = self.ksm.get_all_specs()
  61. self.assertEqual(kernels['sample']['resource_dir'], self.sample_kernel_dir)
  62. self.assertIsNotNone(kernels['sample']['spec'])
  63. def test_kernel_spec_priority(self):
  64. td = TemporaryDirectory()
  65. self.addCleanup(td.cleanup)
  66. sample_kernel = self._install_sample_kernel(td.name)
  67. self.ksm.kernel_dirs.append(td.name)
  68. kernels = self.ksm.find_kernel_specs()
  69. self.assertEqual(kernels['sample'], self.sample_kernel_dir)
  70. self.ksm.kernel_dirs.insert(0, td.name)
  71. kernels = self.ksm.find_kernel_specs()
  72. self.assertEqual(kernels['sample'], sample_kernel)
  73. def test_install_kernel_spec(self):
  74. self.ksm.install_kernel_spec(self.installable_kernel,
  75. kernel_name='tstinstalled',
  76. user=True)
  77. self.assertIn('tstinstalled', self.ksm.find_kernel_specs())
  78. # install again works
  79. self.ksm.install_kernel_spec(self.installable_kernel,
  80. kernel_name='tstinstalled',
  81. user=True)
  82. def test_install_kernel_spec_prefix(self):
  83. td = TemporaryDirectory()
  84. self.addCleanup(td.cleanup)
  85. capture = StringIO()
  86. handler = StreamHandler(capture)
  87. self.ksm.log.addHandler(handler)
  88. self.ksm.install_kernel_spec(self.installable_kernel,
  89. kernel_name='tstinstalled',
  90. prefix=td.name)
  91. captured = capture.getvalue()
  92. self.ksm.log.removeHandler(handler)
  93. self.assertIn("may not be found", captured)
  94. self.assertNotIn('tstinstalled', self.ksm.find_kernel_specs())
  95. # add prefix to path, so we find the spec
  96. self.ksm.kernel_dirs.append(pjoin(td.name, 'share', 'jupyter', 'kernels'))
  97. self.assertIn('tstinstalled', self.ksm.find_kernel_specs())
  98. # Run it again, no warning this time because we've added it to the path
  99. capture = StringIO()
  100. handler = StreamHandler(capture)
  101. self.ksm.log.addHandler(handler)
  102. self.ksm.install_kernel_spec(self.installable_kernel,
  103. kernel_name='tstinstalled',
  104. prefix=td.name)
  105. captured = capture.getvalue()
  106. self.ksm.log.removeHandler(handler)
  107. self.assertNotIn("may not be found", captured)
  108. @pytest.mark.skipif(
  109. not (os.name != 'nt' and not os.access('/usr/local/share', os.W_OK)),
  110. reason="needs Unix system without root privileges")
  111. def test_cant_install_kernel_spec(self):
  112. with self.assertRaises(OSError):
  113. self.ksm.install_kernel_spec(self.installable_kernel,
  114. kernel_name='tstinstalled',
  115. user=False)
  116. def test_remove_kernel_spec(self):
  117. path = self.ksm.remove_kernel_spec('sample')
  118. self.assertEqual(path, self.sample_kernel_dir)
  119. def test_remove_kernel_spec_app(self):
  120. p = Popen(
  121. [sys.executable, '-m', 'jupyter_client.kernelspecapp', 'remove', 'sample', '-f'],
  122. stdout=PIPE, stderr=STDOUT,
  123. env=os.environ,
  124. )
  125. out, _ = p.communicate()
  126. self.assertEqual(p.returncode, 0, out.decode('utf8', 'replace'))
  127. def test_validate_kernel_name(self):
  128. for good in [
  129. 'julia-0.4',
  130. 'ipython',
  131. 'R',
  132. 'python_3',
  133. 'Haskell-1-2-3',
  134. ]:
  135. assert kernelspec._is_valid_kernel_name(good)
  136. for bad in [
  137. 'has space',
  138. u'ünicode',
  139. '%percent',
  140. 'question?',
  141. ]:
  142. assert not kernelspec._is_valid_kernel_name(bad)
  143. def test_subclass(self):
  144. """Test get_all_specs in subclasses that override find_kernel_specs"""
  145. ksm = self.ksm
  146. resource_dir = tempfile.gettempdir()
  147. native_name = kernelspec.NATIVE_KERNEL_NAME
  148. native_kernel = ksm.get_kernel_spec(native_name)
  149. class MyKSM(kernelspec.KernelSpecManager):
  150. def get_kernel_spec(self, name):
  151. spec = copy.copy(native_kernel)
  152. if name == 'fake':
  153. spec.name = name
  154. spec.resource_dir = resource_dir
  155. elif name == native_name:
  156. pass
  157. else:
  158. raise KeyError(name)
  159. return spec
  160. def find_kernel_specs(self):
  161. return {
  162. 'fake': resource_dir,
  163. native_name: native_kernel.resource_dir,
  164. }
  165. # ensure that get_all_specs doesn't raise if only
  166. # find_kernel_specs and get_kernel_spec are defined
  167. myksm = MyKSM()
  168. specs = myksm.get_all_specs()
  169. assert sorted(specs) == ['fake', native_name]