##// END OF EJS Templates
Fix bug with info requests that were not json-safe....
Fix bug with info requests that were not json-safe. This would instantly crash the kernel.

File last commit:

r2934:ff8ca1fe
r2948:f0e4ac07
Show More
kernelmanager.py
785 lines | 27.8 KiB | text/x-python | PythonLexer
Brian Granger
Minor work on kernelmanager....
r2742 """Base classes to manage the interaction with a running kernel.
Brian Granger
Work on the kernel manager.
r2606
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 Todo
====
Brian Granger
Work on the kernel manager.
r2606
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 * Create logger to handle debugging and console messages.
Brian Granger
Work on the kernel manager.
r2606 """
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 #-----------------------------------------------------------------------------
# Copyright (C) 2008-2010 The IPython Development Team
#
# Distributed under the terms of the BSD License. The full license is in
# the file COPYING, distributed as part of this software.
#-----------------------------------------------------------------------------
#-----------------------------------------------------------------------------
# Imports
#-----------------------------------------------------------------------------
epatters
* Refactored KernelManager to use Traitlets and to have its channels as attributes...
r2611 # Standard library imports.
Brian Granger
Work on the kernel manager.
r2606 from Queue import Queue, Empty
epatters
* Implemented a proper main() function for kernel.py that reads command line input....
r2667 from subprocess import Popen
Brian Granger
Work on the kernel manager.
r2606 from threading import Thread
epatters
Added a flush method to the SubSocketChannel. The Qt console frontend now uses this method to ensure that output has been processed before it writes a new prompt.
r2614 import time
Brian Granger
Work on the kernel manager.
r2606
epatters
* Refactored KernelManager to use Traitlets and to have its channels as attributes...
r2611 # System library imports.
Brian Granger
Work on the kernel manager.
r2606 import zmq
from zmq import POLLIN, POLLOUT, POLLERR
from zmq.eventloop import ioloop
epatters
* Refactored KernelManager to use Traitlets and to have its channels as attributes...
r2611
# Local imports.
Fernando Perez
Rework messaging to better conform to our spec....
r2926 from IPython.utils import io
Brian Granger
Minor work on kernelmanager....
r2742 from IPython.utils.traitlets import HasTraits, Any, Instance, Type, TCPAddress
Brian Granger
Work on the kernel manager.
r2606 from session import Session
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 #-----------------------------------------------------------------------------
# Constants and exceptions
#-----------------------------------------------------------------------------
epatters
* Implemented a proper main() function for kernel.py that reads command line input....
r2667 LOCALHOST = '127.0.0.1'
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 class InvalidPortNumber(Exception):
pass
Brian Granger
Work on the kernel manager.
r2606
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 #-----------------------------------------------------------------------------
Fernando Perez
Rework messaging to better conform to our spec....
r2926 # Utility functions
#-----------------------------------------------------------------------------
# some utilities to validate message structure, these might get moved elsewhere
# if they prove to have more generic utility
def validate_string_list(lst):
"""Validate that the input is a list of strings.
Raises ValueError if not."""
if not isinstance(lst, list):
raise ValueError('input %r must be a list' % lst)
for x in lst:
if not isinstance(x, basestring):
raise ValueError('element %r in list must be a string' % x)
def validate_string_dict(dct):
"""Validate that the input is a dict with string keys and values.
Raises ValueError if not."""
for k,v in dct.iteritems():
if not isinstance(k, basestring):
raise ValueError('key %r in dict must be a string' % k)
if not isinstance(v, basestring):
raise ValueError('value %r in dict must be a string' % v)
#-----------------------------------------------------------------------------
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 # ZMQ Socket Channel classes
#-----------------------------------------------------------------------------
Brian Granger
Work on the kernel manager.
r2606
class ZmqSocketChannel(Thread):
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 """The base class for the channels that use ZMQ sockets.
epatters
Cleaned up KernelManager interface and clarified documentation.
r2631 """
Brian Granger
Fixed high CPU usage of XREQ channel....
r2695 context = None
session = None
socket = None
ioloop = None
iostate = None
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 _address = None
def __init__(self, context, session, address):
"""Create a channel
Brian Granger
Fixed high CPU usage of XREQ channel....
r2695
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 Parameters
----------
Brian Granger
Minor work on kernelmanager....
r2742 context : :class:`zmq.Context`
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 The ZMQ context to use.
Brian Granger
Minor work on kernelmanager....
r2742 session : :class:`session.Session`
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 The session to use.
address : tuple
Standard (ip, port) tuple that the kernel is listening on.
"""
epatters
* Added 'stop' methods to the ZmqSocketChannels...
r2632 super(ZmqSocketChannel, self).__init__()
self.daemon = True
Brian Granger
Work on the kernel manager.
r2606 self.context = context
self.session = session
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 if address[1] == 0:
epatters
* Added 'req_port' option to 'launch_kernel' and the kernel entry point....
r2702 message = 'The port number for a channel cannot be 0.'
raise InvalidPortNumber(message)
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 self._address = address
epatters
Cleaned up KernelManager interface and clarified documentation.
r2631
epatters
* Added 'stop' methods to the ZmqSocketChannels...
r2632 def stop(self):
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 """Stop the channel's activity.
Brian Granger
Channels can no longer be restarted.
r2691
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 This calls :method:`Thread.join` and returns when the thread
terminates. :class:`RuntimeError` will be raised if
:method:`self.start` is called again.
epatters
* Added 'stop' methods to the ZmqSocketChannels...
r2632 """
epatters
Fixed kernelmanager threads not being restartable.
r2642 self.join()
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 @property
def address(self):
"""Get the channel's address as an (ip, port) tuple.
By the default, the address is (localhost, 0), where 0 means a random
port.
epatters
* Added 'stop' methods to the ZmqSocketChannels...
r2632 """
return self._address
Brian Granger
Fixed high CPU usage of XREQ channel....
r2695 def add_io_state(self, state):
"""Add IO state to the eventloop.
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 Parameters
----------
state : zmq.POLLIN|zmq.POLLOUT|zmq.POLLERR
The IO state flag to set.
Brian Granger
Fixed high CPU usage of XREQ channel....
r2695 This is thread safe as it uses the thread safe IOLoop.add_callback.
"""
def add_io_state_callback():
if not self.iostate & state:
self.iostate = self.iostate | state
self.ioloop.update_handler(self.socket, self.iostate)
self.ioloop.add_callback(add_io_state_callback)
def drop_io_state(self, state):
"""Drop IO state from the eventloop.
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 Parameters
----------
state : zmq.POLLIN|zmq.POLLOUT|zmq.POLLERR
The IO state flag to set.
Brian Granger
Fixed high CPU usage of XREQ channel....
r2695 This is thread safe as it uses the thread safe IOLoop.add_callback.
"""
def drop_io_state_callback():
if self.iostate & state:
self.iostate = self.iostate & (~state)
self.ioloop.update_handler(self.socket, self.iostate)
self.ioloop.add_callback(drop_io_state_callback)
Brian Granger
Work on the kernel manager.
r2606
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 class XReqSocketChannel(ZmqSocketChannel):
"""The XREQ channel for issues request/replies to the kernel.
"""
Brian Granger
Work on the kernel manager.
r2606
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 command_queue = None
def __init__(self, context, session, address):
self.command_queue = Queue()
super(XReqSocketChannel, self).__init__(context, session, address)
Brian Granger
Work on the kernel manager.
r2606
def run(self):
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 """The thread's main activity. Call start() instead."""
self.socket = self.context.socket(zmq.XREQ)
Brian Granger
Work on the kernel manager.
r2606 self.socket.setsockopt(zmq.IDENTITY, self.session.session)
epatters
* Added 'stop' methods to the ZmqSocketChannels...
r2632 self.socket.connect('tcp://%s:%i' % self.address)
Brian Granger
Work on the kernel manager.
r2606 self.ioloop = ioloop.IOLoop()
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 self.iostate = POLLERR|POLLIN
Brian Granger
Work on the kernel manager.
r2606 self.ioloop.add_handler(self.socket, self._handle_events,
Brian Granger
Fixed high CPU usage of XREQ channel....
r2695 self.iostate)
Brian Granger
Work on the kernel manager.
r2606 self.ioloop.start()
epatters
* Added 'stop' methods to the ZmqSocketChannels...
r2632 def stop(self):
self.ioloop.stop()
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 super(XReqSocketChannel, self).stop()
Brian Granger
Work on the kernel manager.
r2606
def call_handlers(self, msg):
Brian Granger
General cleanup of kernelmanager.py....
r2692 """This method is called in the ioloop thread when a message arrives.
Brian Granger
Work on the kernel manager.
r2606
Brian Granger
General cleanup of kernelmanager.py....
r2692 Subclasses should override this method to handle incoming messages.
It is important to remember that this method is called in the thread
so that some logic must be done to ensure that the application leve
handlers are called in the application thread.
"""
raise NotImplementedError('call_handlers must be defined in a subclass.')
Brian Granger
Work on the kernel manager.
r2606
Fernando Perez
Rework messaging to better conform to our spec....
r2926 def execute(self, code, silent=False,
user_variables=None, user_expressions=None):
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 """Execute code in the kernel.
epatters
Made KernelManager's flush() method more robust and added a timeout parameter for safety.
r2672
Parameters
----------
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 code : str
A string of Python code.
Fernando Perez
Rework messaging to better conform to our spec....
r2926
epatters
* Added support for prompt and history requests to the kernel manager and Qt console frontend....
r2844 silent : bool, optional (default False)
If set, the kernel will execute the code as quietly possible.
epatters
Added a flush method to the SubSocketChannel. The Qt console frontend now uses this method to ensure that output has been processed before it writes a new prompt.
r2614
Fernando Perez
Rework messaging to better conform to our spec....
r2926 user_variables : list, optional
A list of variable names to pull from the user's namespace. They
will come back as a dict with these names as keys and their
:func:`repr` as values.
user_expressions : dict, optional
A dict with string keys and to pull from the user's
namespace. They will come back as a dict with these names as keys
and their :func:`repr` as values.
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 Returns
-------
The msg_id of the message sent.
"""
Fernando Perez
Rework messaging to better conform to our spec....
r2926 if user_variables is None:
user_variables = []
if user_expressions is None:
user_expressions = {}
# Don't waste network traffic if inputs are invalid
if not isinstance(code, basestring):
raise ValueError('code %r must be a string' % code)
validate_string_list(user_variables)
validate_string_dict(user_expressions)
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 # Create class for content/msg creation. Related to, but possibly
# not in Session.
Fernando Perez
Rework messaging to better conform to our spec....
r2926 content = dict(code=code, silent=silent,
user_variables=user_variables,
user_expressions=user_expressions)
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 msg = self.session.msg('execute_request', content)
self._queue_request(msg)
return msg['header']['msg_id']
Brian Granger
Work on the kernel manager.
r2606
Fernando Perez
Multiple improvements to tab completion....
r2839 def complete(self, text, line, cursor_pos, block=None):
epatters
* Tab completion now uses the correct cursor position....
r2841 """Tab complete text in the kernel's namespace.
Brian Granger
Work on the kernel manager.
r2606
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 Parameters
----------
text : str
The text to complete.
line : str
The full line of text that is the surrounding context for the
text to complete.
epatters
* Tab completion now uses the correct cursor position....
r2841 cursor_pos : int
The position of the cursor in the line where the completion was
requested.
block : str, optional
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 The full block of code in which the completion is being requested.
Returns
-------
The msg_id of the message sent.
"""
Fernando Perez
Multiple improvements to tab completion....
r2839 content = dict(text=text, line=line, block=block, cursor_pos=cursor_pos)
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 msg = self.session.msg('complete_request', content)
self._queue_request(msg)
return msg['header']['msg_id']
Brian Granger
Work on the kernel manager.
r2606
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 def object_info(self, oname):
"""Get metadata information about an object.
Brian Granger
Work on the kernel manager.
r2606
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 Parameters
----------
oname : str
A string specifying the object name.
Returns
-------
The msg_id of the message sent.
"""
content = dict(oname=oname)
msg = self.session.msg('object_info_request', content)
self._queue_request(msg)
return msg['header']['msg_id']
epatters
* Added 'stop' methods to the ZmqSocketChannels...
r2632
epatters
* Added support for prompt and history requests to the kernel manager and Qt console frontend....
r2844 def history(self, index=None, raw=False, output=True):
"""Get the history list.
Parameters
----------
index : n or (n1, n2) or None
If n, then the last entries. If a tuple, then all in
range(n1, n2). If None, then all entries. Raises IndexError if
the format of index is incorrect.
raw : bool
If True, return the raw input.
output : bool
If True, then return the output as well.
Returns
-------
The msg_id of the message sent.
"""
content = dict(index=index, raw=raw, output=output)
msg = self.session.msg('history_request', content)
self._queue_request(msg)
return msg['header']['msg_id']
Brian Granger
Work on the kernel manager.
r2606 def _handle_events(self, socket, events):
if events & POLLERR:
self._handle_err()
if events & POLLOUT:
self._handle_send()
if events & POLLIN:
self._handle_recv()
def _handle_recv(self):
msg = self.socket.recv_json()
epatters
Initial checkin of Qt kernel manager. Began refactor of FrontendWidget.
r2609 self.call_handlers(msg)
Brian Granger
Work on the kernel manager.
r2606
def _handle_send(self):
try:
msg = self.command_queue.get(False)
except Empty:
pass
else:
self.socket.send_json(msg)
Brian Granger
Fixed high CPU usage of XREQ channel....
r2695 if self.command_queue.empty():
self.drop_io_state(POLLOUT)
Brian Granger
Work on the kernel manager.
r2606
def _handle_err(self):
Brian Granger
General cleanup of kernelmanager.py....
r2692 # We don't want to let this go silently, so eventually we should log.
Brian Granger
SUB channel _handle_recv is now greedy....
r2694 raise zmq.ZMQError()
Brian Granger
Work on the kernel manager.
r2606
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 def _queue_request(self, msg):
Brian Granger
Work on the kernel manager.
r2606 self.command_queue.put(msg)
Brian Granger
Fixed high CPU usage of XREQ channel....
r2695 self.add_io_state(POLLOUT)
Brian Granger
Work on the kernel manager.
r2606
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699
class SubSocketChannel(ZmqSocketChannel):
"""The SUB channel which listens for messages that the kernel publishes.
"""
def __init__(self, context, session, address):
super(SubSocketChannel, self).__init__(context, session, address)
def run(self):
"""The thread's main activity. Call start() instead."""
self.socket = self.context.socket(zmq.SUB)
self.socket.setsockopt(zmq.SUBSCRIBE,'')
self.socket.setsockopt(zmq.IDENTITY, self.session.session)
self.socket.connect('tcp://%s:%i' % self.address)
self.ioloop = ioloop.IOLoop()
self.iostate = POLLIN|POLLERR
self.ioloop.add_handler(self.socket, self._handle_events,
self.iostate)
self.ioloop.start()
def stop(self):
self.ioloop.stop()
super(SubSocketChannel, self).stop()
Brian Granger
Cleanup of the XREQ channel....
r2697 def call_handlers(self, msg):
"""This method is called in the ioloop thread when a message arrives.
Subclasses should override this method to handle incoming messages.
It is important to remember that this method is called in the thread
so that some logic must be done to ensure that the application leve
handlers are called in the application thread.
"""
raise NotImplementedError('call_handlers must be defined in a subclass.')
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 def flush(self, timeout=1.0):
"""Immediately processes all pending messages on the SUB channel.
Brian Granger
Work on the kernel manager.
r2606
Brian Granger
Minor work on kernelmanager....
r2742 Callers should use this method to ensure that :method:`call_handlers`
has been called for all messages that have been received on the
0MQ SUB socket of this channel.
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 This method is thread safe.
Brian Granger
Work on the kernel manager.
r2606
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 Parameters
----------
timeout : float, optional
The maximum amount of time to spend flushing, in seconds. The
default is one second.
"""
# We do the IOLoop callback process twice to ensure that the IOLoop
# gets to perform at least one full poll.
stop_time = time.time() + timeout
for i in xrange(2):
self._flushed = False
self.ioloop.add_callback(self._flush)
while not self._flushed and time.time() < stop_time:
time.sleep(0.01)
def _handle_events(self, socket, events):
# Turn on and off POLLOUT depending on if we have made a request
if events & POLLERR:
self._handle_err()
if events & POLLIN:
self._handle_recv()
def _handle_err(self):
# We don't want to let this go silently, so eventually we should log.
raise zmq.ZMQError()
def _handle_recv(self):
# Get all of the messages we can
while True:
try:
msg = self.socket.recv_json(zmq.NOBLOCK)
except zmq.ZMQError:
# Check the errno?
Brian Granger
Minor work on kernelmanager....
r2742 # Will this trigger POLLERR?
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 break
else:
self.call_handlers(msg)
def _flush(self):
"""Callback for :method:`self.flush`."""
self._flushed = True
Brian Granger
Work on the kernel manager.
r2606
class RepSocketChannel(ZmqSocketChannel):
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 """A reply channel to handle raw_input requests that the kernel makes."""
Brian Granger
Work on the kernel manager.
r2606
epatters
Basic raw_input implementation is now working.
r2707 msg_queue = None
def __init__(self, context, session, address):
self.msg_queue = Queue()
super(RepSocketChannel, self).__init__(context, session, address)
epatters
Merge branch 'kernelmanager' of git://github.com/ellisonbg/ipython into qtfrontend. Fixed breakage and conflicts from merge....
r2701 def run(self):
"""The thread's main activity. Call start() instead."""
epatters
Basic raw_input implementation is now working.
r2707 self.socket = self.context.socket(zmq.XREQ)
self.socket.setsockopt(zmq.IDENTITY, self.session.session)
self.socket.connect('tcp://%s:%i' % self.address)
epatters
Merge branch 'kernelmanager' of git://github.com/ellisonbg/ipython into qtfrontend. Fixed breakage and conflicts from merge....
r2701 self.ioloop = ioloop.IOLoop()
epatters
Basic raw_input implementation is now working.
r2707 self.iostate = POLLERR|POLLIN
self.ioloop.add_handler(self.socket, self._handle_events,
self.iostate)
epatters
Merge branch 'kernelmanager' of git://github.com/ellisonbg/ipython into qtfrontend. Fixed breakage and conflicts from merge....
r2701 self.ioloop.start()
def stop(self):
self.ioloop.stop()
epatters
* Added 'req_port' option to 'launch_kernel' and the kernel entry point....
r2702 super(RepSocketChannel, self).stop()
Brian Granger
Work on the kernel manager.
r2606
epatters
Basic raw_input implementation is now working.
r2707 def call_handlers(self, msg):
"""This method is called in the ioloop thread when a message arrives.
Subclasses should override this method to handle incoming messages.
It is important to remember that this method is called in the thread
so that some logic must be done to ensure that the application leve
handlers are called in the application thread.
"""
raise NotImplementedError('call_handlers must be defined in a subclass.')
epatters
* Change input mechanism: replace raw_input instead of stdin....
r2730 def input(self, string):
"""Send a string of raw input to the kernel."""
content = dict(value=string)
msg = self.session.msg('input_reply', content)
epatters
Basic raw_input implementation is now working.
r2707 self._queue_reply(msg)
def _handle_events(self, socket, events):
if events & POLLERR:
self._handle_err()
if events & POLLOUT:
self._handle_send()
if events & POLLIN:
self._handle_recv()
def _handle_recv(self):
msg = self.socket.recv_json()
self.call_handlers(msg)
def _handle_send(self):
try:
msg = self.msg_queue.get(False)
except Empty:
pass
else:
self.socket.send_json(msg)
if self.msg_queue.empty():
self.drop_io_state(POLLOUT)
def _handle_err(self):
# We don't want to let this go silently, so eventually we should log.
raise zmq.ZMQError()
def _queue_reply(self, msg):
self.msg_queue.put(msg)
self.add_io_state(POLLOUT)
epatters
* Refactored KernelManager to use Traitlets and to have its channels as attributes...
r2611
Brian Granger
Added heartbeat support.
r2910 class HBSocketChannel(ZmqSocketChannel):
epatters
* Fixed heartbeat thread not stopping cleanly....
r2915 """The heartbeat channel which monitors the kernel heartbeat."""
Brian Granger
Added heartbeat support.
r2910
Brian Granger
Fixing logic in heartbeat monitor.
r2925 time_to_dead = 3.0
Brian Granger
Added heartbeat support.
r2910 socket = None
poller = None
def __init__(self, context, session, address):
super(HBSocketChannel, self).__init__(context, session, address)
epatters
* Fixed heartbeat thread not stopping cleanly....
r2915 self._running = False
Brian Granger
Added heartbeat support.
r2910
def _create_socket(self):
self.socket = self.context.socket(zmq.REQ)
self.socket.setsockopt(zmq.IDENTITY, self.session.session)
self.socket.connect('tcp://%s:%i' % self.address)
self.poller = zmq.Poller()
self.poller.register(self.socket, zmq.POLLIN)
def run(self):
"""The thread's main activity. Call start() instead."""
self._create_socket()
epatters
* Fixed heartbeat thread not stopping cleanly....
r2915 self._running = True
Brian Granger
Fixing logic in heartbeat monitor.
r2925 # Wait 2 seconds for the kernel to come up and the sockets to auto
# connect. If we don't we will see the kernel as dead. Also, before
# the sockets are connected, the poller.poll line below is returning
# too fast. This avoids that because the polling doesn't start until
# after the sockets are connected.
time.sleep(2.0)
epatters
* Fixed heartbeat thread not stopping cleanly....
r2915 while self._running:
Brian Granger
Added heartbeat support.
r2910 since_last_heartbeat = 0.0
request_time = time.time()
try:
Fernando Perez
Rework messaging to better conform to our spec....
r2926 #io.rprint('Ping from HB channel') # dbg
Brian Granger
Added heartbeat support.
r2910 self.socket.send_json('ping')
except zmq.ZMQError, e:
Fernando Perez
Rework messaging to better conform to our spec....
r2926 #io.rprint('*** HB Error:', e) # dbg
Brian Granger
Added heartbeat support.
r2910 if e.errno == zmq.EFSM:
Fernando Perez
Rework messaging to better conform to our spec....
r2926 #io.rprint('sleep...', self.time_to_dead) # dbg
Brian Granger
Added heartbeat support.
r2910 time.sleep(self.time_to_dead)
self._create_socket()
else:
raise
else:
while True:
try:
Fernando Perez
Rework messaging to better conform to our spec....
r2926 self.socket.recv_json(zmq.NOBLOCK)
Brian Granger
Added heartbeat support.
r2910 except zmq.ZMQError, e:
Fernando Perez
Rework messaging to better conform to our spec....
r2926 #io.rprint('*** HB Error 2:', e) # dbg
Brian Granger
Added heartbeat support.
r2910 if e.errno == zmq.EAGAIN:
Fernando Perez
Rework messaging to better conform to our spec....
r2926 before_poll = time.time()
until_dead = self.time_to_dead - (before_poll -
epatters
* Fixed heartbeat thread not stopping cleanly....
r2915 request_time)
Fernando Perez
Rework messaging to better conform to our spec....
r2926
# When the return value of poll() is an empty list,
# that is when things have gone wrong (zeromq bug).
# As long as it is not an empty list, poll is
# working correctly even if it returns quickly.
# Note: poll timeout is in milliseconds.
self.poller.poll(1000*until_dead)
Brian Granger
Added heartbeat support.
r2910 since_last_heartbeat = time.time() - request_time
if since_last_heartbeat > self.time_to_dead:
self.call_handlers(since_last_heartbeat)
break
else:
epatters
* Fixed heartbeat thread not stopping cleanly....
r2915 # FIXME: We should probably log this instead.
Brian Granger
Added heartbeat support.
r2910 raise
else:
epatters
* Fixed heartbeat thread not stopping cleanly....
r2915 until_dead = self.time_to_dead - (time.time() -
request_time)
Brian Granger
Added heartbeat support.
r2910 if until_dead > 0.0:
Fernando Perez
Rework messaging to better conform to our spec....
r2926 #io.rprint('sleep...', self.time_to_dead) # dbg
Brian Granger
Added heartbeat support.
r2910 time.sleep(until_dead)
break
epatters
* Fixed heartbeat thread not stopping cleanly....
r2915 def stop(self):
self._running = False
super(HBSocketChannel, self).stop()
Brian Granger
Added heartbeat support.
r2910 def call_handlers(self, since_last_heartbeat):
"""This method is called in the ioloop thread when a message arrives.
Subclasses should override this method to handle incoming messages.
It is important to remember that this method is called in the thread
so that some logic must be done to ensure that the application leve
handlers are called in the application thread.
"""
raise NotImplementedError('call_handlers must be defined in a subclass.')
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 #-----------------------------------------------------------------------------
# Main kernel manager class
#-----------------------------------------------------------------------------
epatters
* Refactored KernelManager to use Traitlets and to have its channels as attributes...
r2611 class KernelManager(HasTraits):
epatters
Cleaned up KernelManager interface and clarified documentation.
r2631 """ Manages a kernel for a frontend.
epatters
* Refactored KernelManager to use Traitlets and to have its channels as attributes...
r2611
epatters
Cleaned up KernelManager interface and clarified documentation.
r2631 The SUB channel is for the frontend to receive messages published by the
kernel.
The REQ channel is for the frontend to make requests of the kernel.
The REP channel is for the kernel to request stdin (raw_input) from the
frontend.
"""
epatters
* Refactored KernelManager to use Traitlets and to have its channels as attributes...
r2611 # The PyZMQ Context to use for communication with the kernel.
Brian Granger
Cleaned up KernelManager.__init__ for better traits usage.
r2753 context = Instance(zmq.Context,(),{})
epatters
* Refactored KernelManager to use Traitlets and to have its channels as attributes...
r2611
# The Session to use for communication with the kernel.
Brian Granger
Cleaned up KernelManager.__init__ for better traits usage.
r2753 session = Instance(Session,(),{})
epatters
* Refactored KernelManager to use Traitlets and to have its channels as attributes...
r2611
epatters
* Change input mechanism: replace raw_input instead of stdin....
r2730 # The kernel process with which the KernelManager is communicating.
kernel = Instance(Popen)
epatters
* The SVG payload matplotlib backend now works....
r2758 # The addresses for the communication channels.
xreq_address = TCPAddress((LOCALHOST, 0))
sub_address = TCPAddress((LOCALHOST, 0))
rep_address = TCPAddress((LOCALHOST, 0))
Brian Granger
Added heartbeat support.
r2910 hb_address = TCPAddress((LOCALHOST, 0))
epatters
* The SVG payload matplotlib backend now works....
r2758
epatters
* Refactored KernelManager to use Traitlets and to have its channels as attributes...
r2611 # The classes to use for the various channels.
xreq_channel_class = Type(XReqSocketChannel)
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 sub_channel_class = Type(SubSocketChannel)
epatters
* Refactored KernelManager to use Traitlets and to have its channels as attributes...
r2611 rep_channel_class = Type(RepSocketChannel)
Brian Granger
Added heartbeat support.
r2910 hb_channel_class = Type(HBSocketChannel)
epatters
Cleaned up KernelManager interface and clarified documentation.
r2631
epatters
* Refactored KernelManager to use Traitlets and to have its channels as attributes...
r2611 # Protected traits.
epatters
First cut at allowing the kernel to be restarted from the frontend.
r2851 _launch_args = Any
epatters
* Refactored KernelManager to use Traitlets and to have its channels as attributes...
r2611 _xreq_channel = Any
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 _sub_channel = Any
epatters
* Refactored KernelManager to use Traitlets and to have its channels as attributes...
r2611 _rep_channel = Any
Brian Granger
Added heartbeat support.
r2910 _hb_channel = Any
epatters
* Refactored KernelManager to use Traitlets and to have its channels as attributes...
r2611
epatters
* The SVG payload matplotlib backend now works....
r2758 #--------------------------------------------------------------------------
epatters
* Implemented KernelManager's 'signal_kernel' method....
r2686 # Channel management methods:
#--------------------------------------------------------------------------
epatters
* Refactored KernelManager to use Traitlets and to have its channels as attributes...
r2611
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 def start_channels(self):
"""Starts the channels for this kernel.
This will create the channels if they do not exist and then start
them. If port numbers of 0 are being used (random ports) then you
must first call :method:`start_kernel`. If the channels have been
stopped and you call this, :class:`RuntimeError` will be raised.
epatters
Added 'start_listening' and 'stop_listening' methods to the kernel manager.
r2639 """
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 self.xreq_channel.start()
self.sub_channel.start()
self.rep_channel.start()
Brian Granger
Added heartbeat support.
r2910 self.hb_channel.start()
epatters
Added 'start_listening' and 'stop_listening' methods to the kernel manager.
r2639
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 def stop_channels(self):
"""Stops the channels for this kernel.
This stops the channels by joining their threads. If the channels
were not started, :class:`RuntimeError` will be raised.
"""
self.xreq_channel.stop()
self.sub_channel.stop()
self.rep_channel.stop()
Brian Granger
Added heartbeat support.
r2910 self.hb_channel.stop()
epatters
* Implemented KernelManager's 'signal_kernel' method....
r2686
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 @property
def channels_running(self):
"""Are all of the channels created and running?"""
return self.xreq_channel.is_alive() \
and self.sub_channel.is_alive() \
Brian Granger
Added heartbeat support.
r2910 and self.rep_channel.is_alive() \
and self.hb_channel.is_alive()
epatters
Added 'start_listening' and 'stop_listening' methods to the kernel manager.
r2639
epatters
* Implemented KernelManager's 'signal_kernel' method....
r2686 #--------------------------------------------------------------------------
# Kernel process management methods:
#--------------------------------------------------------------------------
epatters
First cut at allowing the kernel to be restarted from the frontend.
r2851 def start_kernel(self, **kw):
epatters
* Implemented KernelManager's 'signal_kernel' method....
r2686 """Starts a kernel process and configures the manager to use it.
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 If random ports (port=0) are being used, this method must be called
before the channels are created.
epatters
* The SVG payload matplotlib backend now works....
r2758
Parameters:
-----------
epatters
* Restored functionality after major merge....
r2778 ipython : bool, optional (default True)
Whether to use an IPython kernel instead of a plain Python kernel.
epatters
* Refactored KernelManager to use Traitlets and to have its channels as attributes...
r2611 """
Brian Granger
Added heartbeat support.
r2910 xreq, sub, rep, hb = self.xreq_address, self.sub_address, \
self.rep_address, self.hb_address
epatters
* Fixed heartbeat thread not stopping cleanly....
r2915 if xreq[0] != LOCALHOST or sub[0] != LOCALHOST or \
rep[0] != LOCALHOST or hb[0] != LOCALHOST:
epatters
* Implemented a proper main() function for kernel.py that reads command line input....
r2667 raise RuntimeError("Can only launch a kernel on localhost."
"Make sure that the '*_address' attributes are "
"configured properly.")
epatters
First cut at allowing the kernel to be restarted from the frontend.
r2851 self._launch_args = kw.copy()
if kw.pop('ipython', True):
epatters
* Restored functionality after major merge....
r2778 from ipkernel import launch_kernel as launch
else:
from pykernel import launch_kernel as launch
Brian Granger
Added heartbeat support.
r2910 self.kernel, xrep, pub, req, hb = launch(
epatters
* Fixed heartbeat thread not stopping cleanly....
r2915 xrep_port=xreq[1], pub_port=sub[1],
req_port=rep[1], hb_port=hb[1], **kw)
Brian Granger
Cleaned up KernelManager.__init__ for better traits usage.
r2753 self.xreq_address = (LOCALHOST, xrep)
self.sub_address = (LOCALHOST, pub)
self.rep_address = (LOCALHOST, req)
Brian Granger
Added heartbeat support.
r2910 self.hb_address = (LOCALHOST, hb)
epatters
* Implemented KernelManager's 'signal_kernel' method....
r2686
epatters
First cut at allowing the kernel to be restarted from the frontend.
r2851 def restart_kernel(self):
"""Restarts a kernel with the same arguments that were used to launch
it. If the old kernel was launched with random ports, the same ports
will be used for the new kernel.
"""
if self._launch_args is None:
raise RuntimeError("Cannot restart the kernel. "
"No previous call to 'start_kernel'.")
else:
if self.has_kernel:
self.kill_kernel()
epatters
* Fixed heartbeat thread not stopping cleanly....
r2915 self.start_kernel(**self._launch_args)
epatters
First cut at allowing the kernel to be restarted from the frontend.
r2851
epatters
* Implemented KernelManager's 'signal_kernel' method....
r2686 @property
def has_kernel(self):
"""Returns whether a kernel process has been specified for the kernel
manager.
epatters
* Added 'stop' methods to the ZmqSocketChannels...
r2632 """
epatters
* Change input mechanism: replace raw_input instead of stdin....
r2730 return self.kernel is not None
epatters
* Implemented KernelManager's 'signal_kernel' method....
r2686
def kill_kernel(self):
""" Kill the running kernel. """
epatters
* Change input mechanism: replace raw_input instead of stdin....
r2730 if self.kernel is not None:
self.kernel.kill()
self.kernel = None
epatters
Added 'start_listening' and 'stop_listening' methods to the kernel manager.
r2639 else:
epatters
* Implemented KernelManager's 'signal_kernel' method....
r2686 raise RuntimeError("Cannot kill kernel. No kernel is running!")
epatters
* Refactored KernelManager to use Traitlets and to have its channels as attributes...
r2611
def signal_kernel(self, signum):
epatters
* Implemented KernelManager's 'signal_kernel' method....
r2686 """ Sends a signal to the kernel. """
epatters
* Change input mechanism: replace raw_input instead of stdin....
r2730 if self.kernel is not None:
self.kernel.send_signal(signum)
epatters
* Implemented KernelManager's 'signal_kernel' method....
r2686 else:
raise RuntimeError("Cannot signal kernel. No kernel is running!")
epatters
* Refactored KernelManager to use Traitlets and to have its channels as attributes...
r2611
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 @property
def is_alive(self):
"""Is the kernel process still running?"""
epatters
* Change input mechanism: replace raw_input instead of stdin....
r2730 if self.kernel is not None:
if self.kernel.poll() is None:
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 return True
else:
return False
else:
# We didn't start the kernel with this KernelManager so we don't
# know if it is running. We should use a heartbeat for this case.
return True
epatters
* Added 'stop' methods to the ZmqSocketChannels...
r2632 #--------------------------------------------------------------------------
# Channels used for communication with the kernel:
#--------------------------------------------------------------------------
epatters
* Refactored KernelManager to use Traitlets and to have its channels as attributes...
r2611 @property
def xreq_channel(self):
"""Get the REQ socket channel object to make requests of the kernel."""
if self._xreq_channel is None:
epatters
Cleaned up KernelManager interface and clarified documentation.
r2631 self._xreq_channel = self.xreq_channel_class(self.context,
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 self.session,
self.xreq_address)
epatters
* Refactored KernelManager to use Traitlets and to have its channels as attributes...
r2611 return self._xreq_channel
@property
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 def sub_channel(self):
"""Get the SUB socket channel object."""
if self._sub_channel is None:
self._sub_channel = self.sub_channel_class(self.context,
self.session,
self.sub_address)
return self._sub_channel
@property
epatters
* Refactored KernelManager to use Traitlets and to have its channels as attributes...
r2611 def rep_channel(self):
"""Get the REP socket channel object to handle stdin (raw_input)."""
if self._rep_channel is None:
epatters
Cleaned up KernelManager interface and clarified documentation.
r2631 self._rep_channel = self.rep_channel_class(self.context,
Brian Granger
Kernel manager is cleaned up and simplified. Still bugs though.
r2699 self.session,
self.rep_address)
epatters
* Refactored KernelManager to use Traitlets and to have its channels as attributes...
r2611 return self._rep_channel
Brian Granger
Added heartbeat support.
r2910
@property
def hb_channel(self):
"""Get the REP socket channel object to handle stdin (raw_input)."""
if self._hb_channel is None:
self._hb_channel = self.hb_channel_class(self.context,
self.session,
self.hb_address)
return self._hb_channel