interface work
This commit is contained in:
		| @@ -1,188 +1,490 @@ | ||||
| """The IPython kernel spec for Jupyter""" | ||||
| """An Application for launching a kernel""" | ||||
|  | ||||
| # Copyright (c) IPython Development Team. | ||||
| # Copyright (c) YAP Development Team. | ||||
| # Distributed under the terms of the Modified BSD License. | ||||
|  | ||||
| from __future__ import print_function | ||||
|  | ||||
| import errno | ||||
| import json | ||||
| import atexit | ||||
| import os | ||||
| import shutil | ||||
| import sys | ||||
| import tempfile | ||||
| import signal | ||||
| import traceback | ||||
| import logging | ||||
|  | ||||
| from jupyter_client.kernelspec import KernelSpecManager | ||||
| from tornado import ioloop | ||||
| import zmq | ||||
| from zmq.eventloop import ioloop as zmq_ioloop | ||||
| from zmq.eventloop.zmqstream import ZMQStream | ||||
|  | ||||
| pjoin = os.path.join | ||||
| from IPython.core.application import ( | ||||
|     BaseIPythonApplication, base_flags, base_aliases, catch_config_error | ||||
| ) | ||||
|  | ||||
| KERNEL_NAME = 'YAP%i' % sys.version_info[0] | ||||
| from IPython.core.profiledir import ProfileDir | ||||
| from IPython.core.shellapp import ( | ||||
|     InteractiveShellApp, shell_flags, shell_aliases | ||||
| ) | ||||
|  | ||||
| # path to kernelspec resources | ||||
| RESOURCES = pjoin(os.path.dirname(__file__), 'resources') | ||||
| from IPython.utils import io | ||||
| from ipython_genutils.path import filefind, ensure_dir_exists | ||||
| from traitlets import ( | ||||
|     Any, Instance, Dict, Unicode, Integer, Bool, DottedObjectName, Type, default | ||||
| ) | ||||
| from ipython_genutils.importstring import import_item | ||||
| from jupyter_core.paths import jupyter_runtime_dir | ||||
| from jupyter_client import write_connection_file | ||||
| from jupyter_client.connect import ConnectionFileMixin | ||||
|  | ||||
| # local imports | ||||
| from ipykernel.iostream import IOPubThread | ||||
| from ipykernel.heartbeat import Heartbeat | ||||
| from .yap_kernel import YAPKernel | ||||
| from ipykernel.parentpoller import ParentPollerUnix, ParentPollerWindows | ||||
| from jupyter_client.session import ( | ||||
|     Session, session_flags, session_aliases, | ||||
| ) | ||||
| from ipykernel.zmqshell import ZMQInteractiveShell | ||||
|  | ||||
| def make_ipkernel_cmd(mod='ipykernel', executable=None, extra_arguments=None, **kw): | ||||
|     """Build Popen command list for launching an IPython kernel. | ||||
| #----------------------------------------------------------------------------- | ||||
| # Flags and Aliases | ||||
| #----------------------------------------------------------------------------- | ||||
|  | ||||
|     Parameters | ||||
|     ---------- | ||||
|     mod : str, optional (default 'ipykernel') | ||||
|         A string of an IPython module whose __main__ starts an IPython kernel | ||||
| kernel_aliases = dict(base_aliases) | ||||
| kernel_aliases.update({ | ||||
|     'ip' : 'YAPKernelApp.ip', | ||||
|     'hb' : 'YAPKernelApp.hb_port', | ||||
|     'shell' : 'YAPKernelApp.shell_port', | ||||
|     'iopub' : 'YAPKernelApp.iopub_port', | ||||
|     'stdin' : 'YAPKernelApp.stdin_port', | ||||
|     'control' : 'YAPKernelApp.control_port', | ||||
|     'f' : 'YAPKernelApp.connection_file', | ||||
|     'transport': 'YAPKernelApp.transport', | ||||
| }) | ||||
|  | ||||
|     executable : str, optional (default sys.executable) | ||||
|         The Python executable to use for the kernel process. | ||||
| kernel_flags = dict(base_flags) | ||||
| kernel_flags.update({ | ||||
|     'no-stdout' : ( | ||||
|             {'YAPKernelApp' : {'no_stdout' : True}}, | ||||
|             "redirect stdout to the null device"), | ||||
|     'no-stderr' : ( | ||||
|             {'YAPKernelApp' : {'no_stderr' : True}}, | ||||
|             "redirect stderr to the null device"), | ||||
|     'pylab' : ( | ||||
|         {'YAPKernelApp' : {'pylab' : 'auto'}}, | ||||
|         """Pre-load matplotlib and numpy for interactive use with | ||||
|         the default matplotlib backend."""), | ||||
| }) | ||||
|  | ||||
|     extra_arguments : list, optional | ||||
|         A list of extra arguments to pass when executing the launch code. | ||||
| # inherit flags&aliases for any IPython shell apps | ||||
| kernel_aliases.update(shell_aliases) | ||||
| kernel_flags.update(shell_flags) | ||||
|  | ||||
|     Returns | ||||
|     ------- | ||||
| # inherit flags&aliases for Sessions | ||||
| kernel_aliases.update(session_aliases) | ||||
| kernel_flags.update(session_flags) | ||||
|  | ||||
|     A Popen command list | ||||
|     """ | ||||
|     if executable is None: | ||||
|         executable = sys.executable | ||||
|     extra_arguments = extra_arguments or [] | ||||
|     arguments = [executable, '-m', mod, '-f', '{connection_file}'] | ||||
|     arguments.extend(extra_arguments) | ||||
| _ctrl_c_message = """\ | ||||
| NOTE: When using the `ipython kernel` entry point, Ctrl-C will not work. | ||||
|  | ||||
|     return arguments | ||||
| To exit, you will have to explicitly quit this process, by either sending | ||||
| "quit" from a client, or using Ctrl-\\ in UNIX-like environments. | ||||
|  | ||||
| To read more about this, see https://github.com/ipython/ipython/issues/2049 | ||||
|  | ||||
| def get_kernel_dict(extra_arguments=None): | ||||
|     """Construct dict for kernel.json""" | ||||
|     return { | ||||
|         'argv': make_ipkernel_cmd(extra_arguments=extra_arguments), | ||||
|         'display_name': 'Python %i' % sys.version_info[0], | ||||
|         'language': 'python', | ||||
| """ | ||||
|  | ||||
| #----------------------------------------------------------------------------- | ||||
| # Application class for starting an YAP Kernel | ||||
| #----------------------------------------------------------------------------- | ||||
|  | ||||
| class YAPKernelApp(BaseIPythonApplication, InteractiveShellApp, | ||||
|         ConnectionFileMixin): | ||||
|     name='YAP-kernel' | ||||
|     aliases = Dict(kernel_aliases) | ||||
|     flags = Dict(kernel_flags) | ||||
|     classes = [YAPKernel, ZMQInteractiveShell, ProfileDir, Session] | ||||
|     # the kernel class, as an importstring | ||||
|     kernel_class = Type('yap_kernel.yap_kernel.YAPKernel', | ||||
|                         klass='ipykernel.kernelbase.Kernel', | ||||
|     help="""The Kernel subclass to be used. | ||||
|  | ||||
|     This should allow easy re-use of the YAPKernelApp entry point | ||||
|     to configure and launch kernels other than YAP's own. | ||||
|     """).tag(config=True) | ||||
|     kernel = Any() | ||||
|     poller = Any() # don't restrict this even though current pollers are all Threads | ||||
|     heartbeat = Instance(Heartbeat, allow_none=True) | ||||
|     ports = Dict() | ||||
|  | ||||
|     subcommands = { | ||||
|         'install': ( | ||||
|             'yap_kernel.kernelspec.InstallYAPKernelSpecApp', | ||||
|             'Install the YAP kernel' | ||||
|         ), | ||||
|     } | ||||
|  | ||||
|     # connection info: | ||||
|     connection_dir = Unicode() | ||||
|  | ||||
| def write_kernel_spec(path=None, overrides=None, extra_arguments=None): | ||||
|     """Write a kernel spec directory to `path` | ||||
|     @default('connection_dir') | ||||
|     def _default_connection_dir(self): | ||||
|         return jupyter_runtime_dir() | ||||
|  | ||||
|     If `path` is not specified, a temporary directory is created. | ||||
|     If `overrides` is given, the kernelspec JSON is updated before writing. | ||||
|     @property | ||||
|     def abs_connection_file(self): | ||||
|         if os.path.basename(self.connection_file) == self.connection_file: | ||||
|             return os.path.join(self.connection_dir, self.connection_file) | ||||
|         else: | ||||
|             return self.connection_file | ||||
|  | ||||
|     The path to the kernelspec is always returned. | ||||
|     """ | ||||
|     if path is None: | ||||
|         path = os.path.join(tempfile.mkdtemp(suffix='_kernels'), KERNEL_NAME) | ||||
|     # streams, etc. | ||||
|     no_stdout = Bool(False, help="redirect stdout to the null device").tag(config=True) | ||||
|     no_stderr = Bool(False, help="redirect stderr to the null device").tag(config=True) | ||||
|     outstream_class = DottedObjectName('ipykernel.iostream.OutStream', | ||||
|         help="The importstring for the OutStream factory").tag(config=True) | ||||
|     displayhook_class = DottedObjectName('ipykernel.displayhook.ZMQDisplayHook', | ||||
|         help="The importstring for the DisplayHook factory").tag(config=True) | ||||
|  | ||||
|     # stage resources | ||||
|     shutil.copytree(RESOURCES, path) | ||||
|     # write kernel.json | ||||
|     kernel_dict = get_kernel_dict(extra_arguments) | ||||
|     # polling | ||||
|     parent_handle = Integer(int(os.environ.get('JPY_PARENT_PID') or 0), | ||||
|         help="""kill this process if its parent dies.  On Windows, the argument | ||||
|         specifies the HANDLE of the parent process, otherwise it is simply boolean. | ||||
|         """).tag(config=True) | ||||
|     interrupt = Integer(int(os.environ.get('JPY_INTERRUPT_EVENT') or 0), | ||||
|         help="""ONLY USED ON WINDOWS | ||||
|         Interrupt this process when the parent is signaled. | ||||
|         """).tag(config=True) | ||||
|  | ||||
|     if overrides: | ||||
|         kernel_dict.update(overrides) | ||||
|     with open(pjoin(path, 'kernel.json'), 'w') as f: | ||||
|         json.dump(kernel_dict, f, indent=1) | ||||
|     def init_crash_handler(self): | ||||
|         sys.excepthook = self.excepthook | ||||
|  | ||||
|     return path | ||||
|     def excepthook(self, etype, evalue, tb): | ||||
|         # write uncaught traceback to 'real' stderr, not zmq-forwarder | ||||
|         traceback.print_exception(etype, evalue, tb, file=sys.__stderr__) | ||||
|  | ||||
|     def init_poller(self): | ||||
|         if sys.platform == 'win32': | ||||
|             if self.interrupt or self.parent_handle: | ||||
|                 self.poller = ParentPollerWindows(self.interrupt, self.parent_handle) | ||||
|         elif self.parent_handle: | ||||
|             self.poller = ParentPollerUnix() | ||||
|  | ||||
| def install(kernel_spec_manager=None, user=False, kernel_name=KERNEL_NAME, display_name=None, | ||||
|             prefix=None, profile=None): | ||||
|     """Install the IPython kernelspec for Jupyter | ||||
|     def _bind_socket(self, s, port): | ||||
|         iface = '%s://%s' % (self.transport, self.ip) | ||||
|         if self.transport == 'tcp': | ||||
|             if port <= 0: | ||||
|                 port = s.bind_to_random_port(iface) | ||||
|             else: | ||||
|                 s.bind("tcp://%s:%i" % (self.ip, port)) | ||||
|         elif self.transport == 'ipc': | ||||
|             if port <= 0: | ||||
|                 port = 1 | ||||
|                 path = "%s-%i" % (self.ip, port) | ||||
|                 while os.path.exists(path): | ||||
|                     port = port + 1 | ||||
|                     path = "%s-%i" % (self.ip, port) | ||||
|             else: | ||||
|                 path = "%s-%i" % (self.ip, port) | ||||
|             s.bind("ipc://%s" % path) | ||||
|         return port | ||||
|  | ||||
|     Parameters | ||||
|     ---------- | ||||
|     def write_connection_file(self): | ||||
|         """write connection info to JSON file""" | ||||
|         cf = self.abs_connection_file | ||||
|         self.log.debug("Writing connection file: %s", cf) | ||||
|         write_connection_file(cf, ip=self.ip, key=self.session.key, transport=self.transport, | ||||
|         shell_port=self.shell_port, stdin_port=self.stdin_port, hb_port=self.hb_port, | ||||
|         iopub_port=self.iopub_port, control_port=self.control_port) | ||||
|  | ||||
|     kernel_spec_manager: KernelSpecManager [optional] | ||||
|         A KernelSpecManager to use for installation. | ||||
|         If none provided, a default instance will be created. | ||||
|     user: bool [default: False] | ||||
|         Whether to do a user-only install, or system-wide. | ||||
|     kernel_name: str, optional | ||||
|         Specify a name for the kernelspec. | ||||
|         This is needed for having multiple IPython kernels for different environments. | ||||
|     display_name: str, optional | ||||
|         Specify the display name for the kernelspec | ||||
|     profile: str, optional | ||||
|         Specify a custom profile to be loaded by the kernel. | ||||
|     prefix: str, optional | ||||
|         Specify an install prefix for the kernelspec. | ||||
|         This is needed to install into a non-default location, such as a conda/virtual-env. | ||||
|     def cleanup_connection_file(self): | ||||
|         cf = self.abs_connection_file | ||||
|         self.log.debug("Cleaning up connection file: %s", cf) | ||||
|         try: | ||||
|             os.remove(cf) | ||||
|         except (IOError, OSError): | ||||
|             pass | ||||
|  | ||||
|     Returns | ||||
|     ------- | ||||
|         self.cleanup_ipc_files() | ||||
|  | ||||
|     The path where the kernelspec was installed. | ||||
|     """ | ||||
|     if kernel_spec_manager is None: | ||||
|         kernel_spec_manager = KernelSpecManager() | ||||
|     def init_connection_file(self): | ||||
|         if not self.connection_file: | ||||
|             self.connection_file = "kernel-%s.json"%os.getpid() | ||||
|         try: | ||||
|             self.connection_file = filefind(self.connection_file, ['.', self.connection_dir]) | ||||
|         except IOError: | ||||
|             self.log.debug("Connection file not found: %s", self.connection_file) | ||||
|             # This means I own it, and I'll create it in this directory: | ||||
|             ensure_dir_exists(os.path.dirname(self.abs_connection_file), 0o700) | ||||
|             # Also, I will clean it up: | ||||
|             atexit.register(self.cleanup_connection_file) | ||||
|             return | ||||
|         try: | ||||
|             self.load_connection_file() | ||||
|         except Exception: | ||||
|             self.log.error("Failed to load connection file: %r", self.connection_file, exc_info=True) | ||||
|             self.exit(1) | ||||
|  | ||||
|     if (kernel_name != KERNEL_NAME) and (display_name is None): | ||||
|         # kernel_name is specified and display_name is not | ||||
|         # default display_name to kernel_name | ||||
|         display_name = kernel_name | ||||
|     overrides = {} | ||||
|     if display_name: | ||||
|         overrides["display_name"] = display_name | ||||
|     if profile: | ||||
|         extra_arguments = ["--profile", profile] | ||||
|         if not display_name: | ||||
|             # add the profile to the default display name | ||||
|             overrides["display_name"] = 'Python %i [profile=%s]' % (sys.version_info[0], profile) | ||||
|     else: | ||||
|         extra_arguments = None | ||||
|     path = write_kernel_spec(overrides=overrides, extra_arguments=extra_arguments) | ||||
|     dest = kernel_spec_manager.install_kernel_spec( | ||||
|         path, kernel_name=kernel_name, user=user, prefix=prefix) | ||||
|     # cleanup afterward | ||||
|     shutil.rmtree(path) | ||||
|     return dest | ||||
|     def init_sockets(self): | ||||
|         # Create a context, a session, and the kernel sockets. | ||||
|         self.log.info("Starting the kernel at pid: %i", os.getpid()) | ||||
|         context = zmq.Context.instance() | ||||
|         # Uncomment this to try closing the context. | ||||
|         # atexit.register(context.term) | ||||
|  | ||||
| # Entrypoint | ||||
|         self.shell_socket = context.socket(zmq.ROUTER) | ||||
|         self.shell_socket.linger = 1000 | ||||
|         self.shell_port = self._bind_socket(self.shell_socket, self.shell_port) | ||||
|         self.log.debug("shell ROUTER Channel on port: %i" % self.shell_port) | ||||
|  | ||||
| from traitlets.config import Application | ||||
|         self.stdin_socket = context.socket(zmq.ROUTER) | ||||
|         self.stdin_socket.linger = 1000 | ||||
|         self.stdin_port = self._bind_socket(self.stdin_socket, self.stdin_port) | ||||
|         self.log.debug("stdin ROUTER Channel on port: %i" % self.stdin_port) | ||||
|  | ||||
|         self.control_socket = context.socket(zmq.ROUTER) | ||||
|         self.control_socket.linger = 1000 | ||||
|         self.control_port = self._bind_socket(self.control_socket, self.control_port) | ||||
|         self.log.debug("control ROUTER Channel on port: %i" % self.control_port) | ||||
|  | ||||
| class InstallYAPKernelSpecApp(Application): | ||||
|     """Dummy app wrapping argparse""" | ||||
|     name = 'ipython-kernel-install' | ||||
|         self.init_iopub(context) | ||||
|  | ||||
|     def init_iopub(self, context): | ||||
|         self.iopub_socket = context.socket(zmq.PUB) | ||||
|         self.iopub_socket.linger = 1000 | ||||
|         self.iopub_port = self._bind_socket(self.iopub_socket, self.iopub_port) | ||||
|         self.log.debug("iopub PUB Channel on port: %i" % self.iopub_port) | ||||
|         self.configure_tornado_logger() | ||||
|         self.iopub_thread = IOPubThread(self.iopub_socket, pipe=True) | ||||
|         self.iopub_thread.start() | ||||
|         # backward-compat: wrap iopub socket API in background thread | ||||
|         self.iopub_socket = self.iopub_thread.background_socket | ||||
|  | ||||
|     def init_heartbeat(self): | ||||
|         """start the heart beating""" | ||||
|         # heartbeat doesn't share context, because it mustn't be blocked | ||||
|         # by the GIL, which is accessed by libzmq when freeing zero-copy messages | ||||
|         hb_ctx = zmq.Context() | ||||
|         self.heartbeat = Heartbeat(hb_ctx, (self.transport, self.ip, self.hb_port)) | ||||
|         self.hb_port = self.heartbeat.port | ||||
|         self.log.debug("Heartbeat REP Channel on port: %i" % self.hb_port) | ||||
|         self.heartbeat.start() | ||||
|  | ||||
|     def log_connection_info(self): | ||||
|         """display connection info, and store ports""" | ||||
|         basename = os.path.basename(self.connection_file) | ||||
|         if basename == self.connection_file or \ | ||||
|             os.path.dirname(self.connection_file) == self.connection_dir: | ||||
|             # use shortname | ||||
|             tail = basename | ||||
|         else: | ||||
|             tail = self.connection_file | ||||
|         lines = [ | ||||
|             "To connect another client to this kernel, use:", | ||||
|             "    --existing %s" % tail, | ||||
|         ] | ||||
|         # log connection info | ||||
|         # info-level, so often not shown. | ||||
|         # frontends should use the %connect_info magic | ||||
|         # to see the connection info | ||||
|         for line in lines: | ||||
|             self.log.info(line) | ||||
|         # also raw print to the terminal if no parent_handle (`ipython kernel`) | ||||
|         # unless log-level is CRITICAL (--quiet) | ||||
|         if not self.parent_handle and self.log_level < logging.CRITICAL: | ||||
|             io.rprint(_ctrl_c_message) | ||||
|             for line in lines: | ||||
|                 io.rprint(line) | ||||
|  | ||||
|         self.ports = dict(shell=self.shell_port, iopub=self.iopub_port, | ||||
|                                 stdin=self.stdin_port, hb=self.hb_port, | ||||
|                                 control=self.control_port) | ||||
|  | ||||
|     def init_blackhole(self): | ||||
|         """redirects stdout/stderr to devnull if necessary""" | ||||
|         if self.no_stdout or self.no_stderr: | ||||
|             blackhole = open(os.devnull, 'w') | ||||
|             if self.no_stdout: | ||||
|                 sys.stdout = sys.__stdout__ = blackhole | ||||
|             if self.no_stderr: | ||||
|                 sys.stderr = sys.__stderr__ = blackhole | ||||
|  | ||||
|     def init_io(self): | ||||
|         """Redirect input streams and set a display hook.""" | ||||
|         if self.outstream_class: | ||||
|             outstream_factory = import_item(str(self.outstream_class)) | ||||
|             sys.stdout = outstream_factory(self.session, self.iopub_thread, u'stdout') | ||||
|             sys.stderr = outstream_factory(self.session, self.iopub_thread, u'stderr') | ||||
|         if self.displayhook_class: | ||||
|             displayhook_factory = import_item(str(self.displayhook_class)) | ||||
|             self.displayhook = displayhook_factory(self.session, self.iopub_socket) | ||||
|             sys.displayhook = self.displayhook | ||||
|  | ||||
|         self.patch_io() | ||||
|  | ||||
|     def patch_io(self): | ||||
|         """Patch important libraries that can't handle sys.stdout forwarding""" | ||||
|         try: | ||||
|             import faulthandler | ||||
|         except ImportError: | ||||
|             pass | ||||
|         else: | ||||
|             # Warning: this is a monkeypatch of `faulthandler.enable`, watch for possible | ||||
|             # updates to the upstream API and update accordingly (up-to-date as of Python 3.5): | ||||
|             # https://docs.python.org/3/library/faulthandler.html#faulthandler.enable | ||||
|  | ||||
|             # change default file to __stderr__ from forwarded stderr | ||||
|             faulthandler_enable = faulthandler.enable | ||||
|             def enable(file=sys.__stderr__, all_threads=True, **kwargs): | ||||
|                 return faulthandler_enable(file=file, all_threads=all_threads, **kwargs) | ||||
|  | ||||
|             faulthandler.enable = enable | ||||
|  | ||||
|             if hasattr(faulthandler, 'register'): | ||||
|                 faulthandler_register = faulthandler.register | ||||
|                 def register(signum, file=sys.__stderr__, all_threads=True, chain=False, **kwargs): | ||||
|                     return faulthandler_register(signum, file=file, all_threads=all_threads, | ||||
|                                                  chain=chain, **kwargs) | ||||
|                 faulthandler.register = register | ||||
|  | ||||
|     def init_signal(self): | ||||
|         signal.signal(signal.SIGINT, signal.SIG_IGN) | ||||
|  | ||||
|     def init_kernel(self): | ||||
|         """Create the Kernel object itself""" | ||||
|         shell_stream = ZMQStream(self.shell_socket) | ||||
|         control_stream = ZMQStream(self.control_socket) | ||||
|  | ||||
|         kernel_factory = self.kernel_class.instance | ||||
|  | ||||
|         kernel = kernel_factory(parent=self, session=self.session, | ||||
|                                 shell_streams=[shell_stream, control_stream], | ||||
|                                 iopub_thread=self.iopub_thread, | ||||
|                                 iopub_socket=self.iopub_socket, | ||||
|                                 stdin_socket=self.stdin_socket, | ||||
|                                 log=self.log, | ||||
|                                 profile_dir=self.profile_dir, | ||||
|                                 user_ns=self.user_ns, | ||||
|         ) | ||||
|         kernel.record_ports({ | ||||
|             name + '_port': port for name, port in self.ports.items() | ||||
|         }) | ||||
|         self.kernel = kernel | ||||
|  | ||||
|         # Allow the displayhook to get the execution count | ||||
|         self.displayhook.get_execution_count = lambda: kernel.execution_count | ||||
|  | ||||
|     def init_gui_pylab(self): | ||||
|         """Enable GUI event loop integration, taking pylab into account.""" | ||||
|  | ||||
|         # Register inline backend as default | ||||
|         # this is higher priority than matplotlibrc, | ||||
|         # but lower priority than anything else (mpl.use() for instance). | ||||
|         # This only affects matplotlib >= 1.5 | ||||
|         if not os.environ.get('MPLBACKEND'): | ||||
|             os.environ['MPLBACKEND'] = 'module://ipykernel.pylab.backend_inline' | ||||
|  | ||||
|         # Provide a wrapper for :meth:`InteractiveShellApp.init_gui_pylab` | ||||
|         # to ensure that any exception is printed straight to stderr. | ||||
|         # Normally _showtraceback associates the reply with an execution, | ||||
|         # which means frontends will never draw it, as this exception | ||||
|         # is not associated with any execute request. | ||||
|  | ||||
|         shell = self.shell | ||||
|         _showtraceback = shell._showtraceback | ||||
|         try: | ||||
|             # replace error-sending traceback with stderr | ||||
|             def print_tb(etype, evalue, stb): | ||||
|                 print ("GUI event loop or pylab initialization failed", | ||||
|                        file=sys.stderr) | ||||
|                 print (shell.InteractiveTB.stb2text(stb), file=sys.stderr) | ||||
|             shell._showtraceback = print_tb | ||||
|             InteractiveShellApp.init_gui_pylab(self) | ||||
|         finally: | ||||
|             shell._showtraceback = _showtraceback | ||||
|  | ||||
|     def init_shell(self): | ||||
|         self.shell = getattr(self.kernel, 'shell', None) | ||||
|         if self.shell: | ||||
|             self.shell.configurables.append(self) | ||||
|  | ||||
|     def init_extensions(self): | ||||
|         super(YAPKernelApp, self).init_extensions() | ||||
|         # BEGIN HARDCODED WIDGETS HACK | ||||
|         # Ensure ipywidgets extension is loaded if available | ||||
|         extension_man = self.shell.extension_manager | ||||
|         if 'ipywidgets' not in extension_man.loaded: | ||||
|             try: | ||||
|                 extension_man.load_extension('ipywidgets') | ||||
|             except ImportError as e: | ||||
|                 self.log.debug('ipywidgets package not installed.  Widgets will not be available.') | ||||
|         # END HARDCODED WIDGETS HACK | ||||
|  | ||||
|     def configure_tornado_logger(self): | ||||
|         """ Configure the tornado logging.Logger. | ||||
|  | ||||
|             Must set up the tornado logger or else tornado will call | ||||
|             basicConfig for the root logger which makes the root logger | ||||
|             go to the real sys.stderr instead of the capture streams. | ||||
|             This function mimics the setup of logging.basicConfig. | ||||
|         """ | ||||
|         logger = logging.getLogger('tornado') | ||||
|         handler = logging.StreamHandler() | ||||
|         formatter = logging.Formatter(logging.BASIC_FORMAT) | ||||
|         handler.setFormatter(formatter) | ||||
|         logger.addHandler(handler) | ||||
|  | ||||
|     @catch_config_error | ||||
|     def initialize(self, argv=None): | ||||
|         if argv is None: | ||||
|             argv = sys.argv[1:] | ||||
|         self.argv = argv | ||||
|         super(YAPKernelApp, self).initialize(argv) | ||||
|         if self.subapp is not None: | ||||
|             return | ||||
|         # register zmq IOLoop with tornado | ||||
|         zmq_ioloop.install() | ||||
|         self.init_blackhole() | ||||
|         self.init_connection_file() | ||||
|         self.init_poller() | ||||
|         self.init_sockets() | ||||
|         self.init_heartbeat() | ||||
|         # writing/displaying connection info must be *after* init_sockets/heartbeat | ||||
|         self.write_connection_file() | ||||
|         # Log connection info after writing connection file, so that the connection | ||||
|         # file is definitely available at the time someone reads the log. | ||||
|         self.log_connection_info() | ||||
|         self.init_io() | ||||
|         self.init_signal() | ||||
|         self.init_kernel() | ||||
|         # shell init steps | ||||
|         self.init_path() | ||||
|         self.init_shell() | ||||
|         if self.shell: | ||||
|             self.init_gui_pylab() | ||||
|             self.init_extensions() | ||||
|             self.init_code() | ||||
|         # flush stdout/stderr, so that anything written to these streams during | ||||
|         # initialization do not get associated with the first execution request | ||||
|         sys.stdout.flush() | ||||
|         sys.stderr.flush() | ||||
|  | ||||
|     def start(self): | ||||
|         import argparse | ||||
|         parser = argparse.ArgumentParser(prog=self.name, | ||||
|             description="Install the YAP kernel spec.") | ||||
|         parser.add_argument('--user', action='store_true', | ||||
|             help="Install for the current user instead of system-wide") | ||||
|         parser.add_argument('--name', type=str, default=KERNEL_NAME, | ||||
|             help="Specify a name for the kernelspec." | ||||
|             " This is needed to have multiple IPython kernels at the same time.") | ||||
|         parser.add_argument('--display-name', type=str, | ||||
|             help="Specify the display name for the kernelspec." | ||||
|             " This is helpful when you have multiple IPython kernels.") | ||||
|         parser.add_argument('--profile', type=str, | ||||
|             help="Specify an IPython profile to load. " | ||||
|             "This can be used to create custom versions of the kernel.") | ||||
|         parser.add_argument('--prefix', type=str, | ||||
|             help="Specify an install prefix for the kernelspec." | ||||
|             " This is needed to install into a non-default location, such as a conda/virtual-env.") | ||||
|         parser.add_argument('--sys-prefix', action='store_const', const=sys.prefix, dest='prefix', | ||||
|             help="Install to Python's sys.prefix." | ||||
|             " Shorthand for --prefix='%s'. For use in conda/virtual-envs." % sys.prefix) | ||||
|         opts = parser.parse_args(self.argv) | ||||
|         if self.subapp is not None: | ||||
|             return self.subapp.start() | ||||
|         if self.poller is not None: | ||||
|             self.poller.start() | ||||
|         self.kernel.start() | ||||
|         try: | ||||
|             dest = install(user=opts.user, kernel_name=opts.name, profile=opts.profile, | ||||
|                            prefix=opts.prefix, display_name=opts.display_name) | ||||
|         except OSError as e: | ||||
|             if e.errno == errno.EACCES: | ||||
|                 print(e, file=sys.stderr) | ||||
|                 if opts.user: | ||||
|                     print("Perhaps you want `sudo` or `--user`?", file=sys.stderr) | ||||
|                 self.exit(1) | ||||
|             raise | ||||
|         print("Installed kernelspec %s in %s" % (opts.name, dest)) | ||||
|             ioloop.IOLoop.instance().start() | ||||
|         except KeyboardInterrupt: | ||||
|             pass | ||||
|  | ||||
| launch_new_instance = YAPKernelApp.launch_instance | ||||
|  | ||||
| def main(): | ||||
|     """Run an IPKernel as an application""" | ||||
|     app = YAPKernelApp.instance() | ||||
|     app.initialize() | ||||
|     app.start() | ||||
|  | ||||
|  | ||||
| if __name__ == '__main__': | ||||
|     InstallYAPKernelSpecApp.launch_instance() | ||||
|     main() | ||||
|   | ||||
		Reference in New Issue
	
	Block a user