##// END OF EJS Templates
Backport PR #8271: Allow to set same mode as existing if load fails....
Backport PR #8271: Allow to set same mode as existing if load fails. Setting the mode many times in a row, would fail to actually set the mode if it was failing to load the mode the first time. The prevent the above behavior by falling back **explicitly** to 'null' if mode does not exist yet, or can't be loaded. ...

File last commit:

r19294:c928f4bc
r21463:d66c4462
Show More
launcher.py
226 lines | 7.6 KiB | text/x-python | PythonLexer
MinRK
test IPython.kernel
r9356 """Utilities for launching kernels
"""
Thomas Kluyver
Add mechanism to specify environment variables in kernel spec
r16349 # Copyright (c) IPython Development Team.
# Distributed under the terms of the Modified BSD License.
MinRK
test IPython.kernel
r9356
import os
import sys
from subprocess import Popen, PIPE
MinRK
Popen doesn't support unicode args on Windows Python 2
r14696 from IPython.utils.encoding import getdefaultencoding
MinRK
cast kernel cwd to bytes on Python 2 on Windows...
r9980 from IPython.utils.py3compat import cast_bytes_py2
MinRK
test IPython.kernel
r9356
def swallow_argv(argv, aliases=None, flags=None):
"""strip frontend-specific aliases and flags from an argument list
For use primarily in frontend apps that want to pass a subset of command-line
arguments through to a subprocess, where frontend-specific flags and aliases
should be removed from the list.
Parameters
----------
argv : list(str)
The starting argv, to be filtered
aliases : container of aliases (dict, list, set, etc.)
The frontend-specific aliases to be removed
flags : container of flags (dict, list, set, etc.)
The frontend-specific flags to be removed
Returns
-------
argv : list(str)
The argv list, excluding flags and aliases that have been stripped
"""
if aliases is None:
aliases = set()
if flags is None:
flags = set()
stripped = list(argv) # copy
swallow_next = False
was_flag = False
for a in argv:
MinRK
check prefixes for swallowing kernel args...
r10426 if a == '--':
break
MinRK
test IPython.kernel
r9356 if swallow_next:
swallow_next = False
# last arg was an alias, remove the next one
# *unless* the last alias has a no-arg flag version, in which
# case, don't swallow the next arg if it's also a flag:
if not (was_flag and a.startswith('-')):
stripped.remove(a)
continue
if a.startswith('-'):
split = a.lstrip('-').split('=')
MinRK
check prefixes for swallowing kernel args...
r10426 name = split[0]
MinRK
add comment about argparse abbreviations
r10442 # we use startswith because argparse accepts any arg to be specified
# by any leading section, as long as it is unique,
# so `--no-br` means `--no-browser` in the notebook, etc.
MinRK
check prefixes for swallowing kernel args...
r10426 if any(alias.startswith(name) for alias in aliases):
MinRK
test IPython.kernel
r9356 stripped.remove(a)
if len(split) == 1:
# alias passed with arg via space
swallow_next = True
# could have been a flag that matches an alias, e.g. `existing`
# in which case, we might not swallow the next arg
MinRK
check prefixes for swallowing kernel args...
r10426 was_flag = name in flags
elif len(split) == 1 and any(flag.startswith(name) for flag in flags):
MinRK
test IPython.kernel
r9356 # strip flag, but don't swallow next, as flags don't take args
stripped.remove(a)
# return shortened list
return stripped
MinRK
use `python -m IPython.kernel` in IPython kernelspec...
r18434 def make_ipkernel_cmd(mod='IPython.kernel', executable=None, extra_arguments=[], **kw):
MinRK
test IPython.kernel
r9356 """Build Popen command list for launching an IPython kernel.
Parameters
----------
MinRK
use `python -m IPython.kernel` in IPython kernelspec...
r18434 mod : str, optional (default 'IPython.kernel')
A string of an IPython module whose __main__ starts an IPython kernel
MinRK
test IPython.kernel
r9356
executable : str, optional (default sys.executable)
The Python executable to use for the kernel process.
extra_arguments : list, optional
A list of extra arguments to pass when executing the launch code.
Returns
-------
A Popen command list
"""
if executable is None:
executable = sys.executable
MinRK
use `python -m IPython.kernel` in IPython kernelspec...
r18434 arguments = [ executable, '-m', mod, '-f', '{connection_file}' ]
MinRK
test IPython.kernel
r9356 arguments.extend(extra_arguments)
Min RK
remove special handling of ipython_kernel in launcher...
r19188
MinRK
test IPython.kernel
r9356 return arguments
Thomas Kluyver
Add mechanism to specify environment variables in kernel spec
r16349 def launch_kernel(cmd, stdin=None, stdout=None, stderr=None, env=None,
MinRK
test IPython.kernel
r9356 independent=False,
Min RK
remove special handling of ipython_kernel in launcher...
r19188 cwd=None,
MinRK
test IPython.kernel
r9356 **kw
):
""" Launches a localhost kernel, binding to the specified ports.
Parameters
----------
cmd : Popen list,
A string of Python code that imports and executes a kernel entry point.
stdin, stdout, stderr : optional (default None)
Standards streams, as defined in subprocess.Popen.
independent : bool, optional (default False)
If set, the kernel process is guaranteed to survive if this process
dies. If not set, an effort is made to ensure that the kernel is killed
when this process dies. Note that in this case it is still good practice
to kill kernels manually before exiting.
cwd : path, optional
The working dir of the kernel process (default: cwd of this process).
Returns
-------
Popen instance for the kernel subprocess
"""
# Popen will fail (sometimes with a deadlock) if stdin, stdout, and stderr
# are invalid. Unfortunately, there is in general no way to detect whether
# they are valid. The following two blocks redirect them to (temporary)
# pipes in certain important cases.
# If this process has been backgrounded, our stdin is invalid. Since there
# is no compelling reason for the kernel to inherit our stdin anyway, we'll
# place this one safe and always redirect.
redirect_in = True
_stdin = PIPE if stdin is None else stdin
# If this process in running on pythonw, we know that stdin, stdout, and
# stderr are all invalid.
redirect_out = sys.executable.endswith('pythonw.exe')
if redirect_out:
Min RK
remove special handling of ipython_kernel in launcher...
r19188 blackhole = open(os.devnull, 'w')
_stdout = blackhole if stdout is None else stdout
_stderr = blackhole if stderr is None else stderr
MinRK
test IPython.kernel
r9356 else:
_stdout, _stderr = stdout, stderr
MinRK
Popen doesn't support unicode args on Windows Python 2
r14696
Thomas Kluyver
Pass Windows interrupt event to kernels as an environment variable...
r16717 env = env if (env is not None) else os.environ.copy()
MinRK
Popen doesn't support unicode args on Windows Python 2
r14696 encoding = getdefaultencoding(prefer_stream=False)
Min RK
remove special handling of ipython_kernel in launcher...
r19188 kwargs = dict(
stdin=_stdin,
stdout=_stdout,
stderr=_stderr,
cwd=cwd,
env=env,
)
MinRK
Popen doesn't support unicode args on Windows Python 2
r14696
MinRK
test IPython.kernel
r9356 # Spawn a kernel.
if sys.platform == 'win32':
MinRK
Popen doesn't support unicode args on Windows Python 2
r14696 # Popen on Python 2 on Windows cannot handle unicode args or cwd
cmd = [ cast_bytes_py2(c, encoding) for c in cmd ]
MinRK
cast kernel cwd to bytes on Python 2 on Windows...
r9980 if cwd:
cwd = cast_bytes_py2(cwd, sys.getfilesystemencoding() or 'ascii')
Min RK
remove special handling of ipython_kernel in launcher...
r19188 kwargs['cwd'] = cwd
MinRK
cast kernel cwd to bytes on Python 2 on Windows...
r9980
MinRK
mv IPython.zmq to IPython.kernel.zmq
r9372 from IPython.kernel.zmq.parentpoller import ParentPollerWindows
Min RK
remove special handling of ipython_kernel in launcher...
r19188 # Create a Win32 event for interrupting the kernel
# and store it in an environment variable.
MinRK
test IPython.kernel
r9356 interrupt_event = ParentPollerWindows.create_interrupt_event()
Min RK
remove special handling of ipython_kernel in launcher...
r19188 env["JPY_INTERRUPT_EVENT"] = str(interrupt_event)
Min RK
deprecate IPY_INTERRUPT_EVENT...
r19255 # deprecated old env name:
env["IPY_INTERRUPT_EVENT"] = env["JPY_INTERRUPT_EVENT"]
Min RK
remove special handling of ipython_kernel in launcher...
r19188
try:
from _winapi import DuplicateHandle, GetCurrentProcess, \
DUPLICATE_SAME_ACCESS, CREATE_NEW_PROCESS_GROUP
except:
from _subprocess import DuplicateHandle, GetCurrentProcess, \
DUPLICATE_SAME_ACCESS, CREATE_NEW_PROCESS_GROUP
# Launch the kernel process
MinRK
test IPython.kernel
r9356 if independent:
Min RK
remove special handling of ipython_kernel in launcher...
r19188 kwargs['creationflags'] = CREATE_NEW_PROCESS_GROUP
MinRK
test IPython.kernel
r9356 else:
Min RK
remove special handling of ipython_kernel in launcher...
r19188 pid = GetCurrentProcess()
handle = DuplicateHandle(pid, pid, pid, 0,
True, # Inheritable by new processes.
DUPLICATE_SAME_ACCESS)
Thomas Kluyver
Convert parent handle to an int before setting environment variable...
r19294 env['JPY_PARENT_PID'] = str(int(handle))
Min RK
remove special handling of ipython_kernel in launcher...
r19188
proc = Popen(cmd, **kwargs)
MinRK
test IPython.kernel
r9356
# Attach the interrupt event to the Popen objet so it can be used later.
proc.win32_interrupt_event = interrupt_event
else:
if independent:
Min RK
remove special handling of ipython_kernel in launcher...
r19188 kwargs['preexec_fn'] = lambda: os.setsid()
MinRK
test IPython.kernel
r9356 else:
Min RK
remove special handling of ipython_kernel in launcher...
r19188 env['JPY_PARENT_PID'] = str(os.getpid())
proc = Popen(cmd, **kwargs)
MinRK
test IPython.kernel
r9356
# Clean up pipes created to work around Popen bug.
if redirect_in:
if stdin is None:
proc.stdin.close()
return proc
MinRK
define and test IPython.kernel public API
r9376 __all__ = [
'swallow_argv',
'make_ipkernel_cmd',
'launch_kernel',
Rob Young
Pass host environment on to kernel...
r9433 ]