kernelspec.py
111 lines
| 3.8 KiB
| text/x-python
|
PythonLexer
Thomas Kluyver
|
r16260 | import io | |
import json | |||
import os | |||
Thomas Kluyver
|
r16261 | import sys | |
Thomas Kluyver
|
r16260 | ||
pjoin = os.path.join | |||
from IPython.utils.path import get_ipython_dir | |||
from IPython.utils.py3compat import PY3 | |||
from IPython.utils.traitlets import HasTraits, List, Unicode | |||
USER_KERNEL_DIR = pjoin(get_ipython_dir(), 'kernels') | |||
if os.name == 'nt': | |||
programdata = os.environ.get('PROGRAMDATA', None) | |||
if programdata: | |||
SYSTEM_KERNEL_DIR = pjoin(programdata, 'ipython', 'kernels') | |||
else: # PROGRAMDATA is not defined by default on XP. | |||
SYSTEM_KERNEL_DIR = None | |||
else: | |||
SYSTEM_KERNEL_DIR = "/usr/share/ipython/kernels" | |||
Thomas Kluyver
|
r16274 | ||
# List of kernel directories to search. Later ones take priority over earlier. | |||
kernel_dirs = [ | |||
SYSTEM_KERNEL_DIR, | |||
USER_KERNEL_DIR, | |||
] | |||
Thomas Kluyver
|
r16260 | ||
NATIVE_KERNEL_NAME = 'python3' if PY3 else 'python2' | |||
class KernelSpec(HasTraits): | |||
argv = List() | |||
display_name = Unicode() | |||
language = Unicode() | |||
Thomas Kluyver
|
r16271 | codemirror_mode = None | |
Thomas Kluyver
|
r16260 | ||
resource_dir = Unicode() | |||
def __init__(self, resource_dir, argv, display_name, language, | |||
codemirror_mode=None): | |||
super(KernelSpec, self).__init__(resource_dir=resource_dir, argv=argv, | |||
display_name=display_name, language=language, | |||
codemirror_mode=codemirror_mode) | |||
if not self.codemirror_mode: | |||
self.codemirror_mode = self.language | |||
@classmethod | |||
def from_resource_dir(cls, resource_dir): | |||
Thomas Kluyver
|
r16262 | """Create a KernelSpec object by reading kernel.json | |
Pass the path to the *directory* containing kernel.json. | |||
""" | |||
kernel_file = pjoin(resource_dir, 'kernel.json') | |||
Thomas Kluyver
|
r16260 | with io.open(kernel_file, 'r', encoding='utf-8') as f: | |
kernel_dict = json.load(f) | |||
return cls(resource_dir=resource_dir, **kernel_dict) | |||
def _is_kernel_dir(path): | |||
Thomas Kluyver
|
r16262 | """Is ``path`` a kernel directory?""" | |
Thomas Kluyver
|
r16260 | return os.path.isdir(path) and os.path.isfile(pjoin(path, 'kernel.json')) | |
def _list_kernels_in(dir): | |||
Thomas Kluyver
|
r16262 | """Ensure dir exists, and return a mapping of kernel names to resource | |
directories from it. | |||
""" | |||
Thomas Kluyver
|
r16271 | if dir is None or not os.path.isdir(dir): | |
Thomas Kluyver
|
r16260 | return {} | |
return {f.lower(): pjoin(dir, f) for f in os.listdir(dir) | |||
if _is_kernel_dir(pjoin(dir, f))} | |||
Thomas Kluyver
|
r16261 | def _make_native_kernel_dir(): | |
Thomas Kluyver
|
r16262 | """Makes a kernel directory for the native kernel. | |
The native kernel is the kernel using the same Python runtime as this | |||
process. This will put its informatino in the user kernels directory. | |||
""" | |||
Thomas Kluyver
|
r16261 | path = pjoin(USER_KERNEL_DIR, NATIVE_KERNEL_NAME) | |
Thomas Kluyver
|
r16271 | os.makedirs(path, mode=0o755) | |
Thomas Kluyver
|
r16261 | with io.open(pjoin(path, 'kernel.json'), 'w', encoding='utf-8') as f: | |
json.dump({'argv':[NATIVE_KERNEL_NAME, '-c', | |||
'from IPython.kernel.zmq.kernelapp import main; main()', | |||
'-f', '{connection_file}'], | |||
'display_name': 'Python 3' if PY3 else 'Python 2', | |||
'language': 'python', | |||
'codemirror_mode': {'name': 'python', | |||
'version': sys.version_info[0]}, | |||
}, | |||
Thomas Kluyver
|
r16273 | f, indent=1) | |
Thomas Kluyver
|
r16261 | # TODO: Copy icons into directory | |
return path | |||
Thomas Kluyver
|
r16260 | def list_kernel_specs(): | |
Thomas Kluyver
|
r16262 | """Returns a dict mapping kernel names to resource directories.""" | |
Thomas Kluyver
|
r16274 | d = {} | |
for kernel_dir in kernel_dirs: | |||
d.update(_list_kernels_in(kernel_dir)) | |||
Thomas Kluyver
|
r16260 | ||
if NATIVE_KERNEL_NAME not in d: | |||
Thomas Kluyver
|
r16261 | d[NATIVE_KERNEL_NAME] = _make_native_kernel_dir() | |
Thomas Kluyver
|
r16260 | return d | |
Thomas Kluyver
|
r16262 | # TODO: Caching? | |
Thomas Kluyver
|
r16260 | ||
def get_kernel_spec(kernel_name): | |||
Thomas Kluyver
|
r16262 | """Returns a :class:`KernelSpec` instance for the given kernel_name. | |
Thomas Kluyver
|
r16260 | ||
Raises KeyError if the given kernel name is not found. | |||
""" | |||
Thomas Kluyver
|
r16263 | if kernel_name == 'native': | |
kernel_name = NATIVE_KERNEL_NAME | |||
Thomas Kluyver
|
r16260 | d = list_kernel_specs() | |
resource_dir = d[kernel_name.lower()] | |||
return KernelSpec.from_resource_dir(resource_dir) |