comm.py
169 lines
| 5.8 KiB
| text/x-python
|
PythonLexer
MinRK
|
r13195 | """Base class for a Comm""" | ||
MinRK
|
r16623 | # Copyright (c) IPython Development Team. | ||
# Distributed under the terms of the Modified BSD License. | ||||
MinRK
|
r13195 | |||
Min RK
|
r20365 | import threading | ||
MinRK
|
r13195 | import uuid | ||
Min RK
|
r20365 | from zmq.eventloop.ioloop import IOLoop | ||
MinRK
|
r13195 | from IPython.config import LoggingConfigurable | ||
Thomas Kluyver
|
r17987 | from IPython.kernel.zmq.kernelbase import Kernel | ||
MinRK
|
r13195 | |||
MinRK
|
r16623 | from IPython.utils.jsonutil import json_clean | ||
MinRK
|
r13195 | from IPython.utils.traitlets import Instance, Unicode, Bytes, Bool, Dict, Any | ||
class Comm(LoggingConfigurable): | ||||
Min RK
|
r20365 | """Class for communicating between a Frontend and a Kernel""" | ||
Thomas Kluyver
|
r17964 | # If this is instantiated by a non-IPython kernel, shell will be None | ||
shell = Instance('IPython.core.interactiveshell.InteractiveShellABC', | ||||
allow_none=True) | ||||
kernel = Instance('IPython.kernel.zmq.kernelbase.Kernel') | ||||
Thomas Kluyver
|
r17987 | def _kernel_default(self): | ||
if Kernel.initialized(): | ||||
return Kernel.instance() | ||||
MinRK
|
r13199 | |||
MinRK
|
r13195 | iopub_socket = Any() | ||
def _iopub_socket_default(self): | ||||
Thomas Kluyver
|
r17964 | return self.kernel.iopub_socket | ||
MinRK
|
r13195 | session = Instance('IPython.kernel.zmq.session.Session') | ||
def _session_default(self): | ||||
Thomas Kluyver
|
r18081 | if self.kernel is not None: | ||
return self.kernel.session | ||||
MinRK
|
r13195 | |||
MinRK
|
r13204 | target_name = Unicode('comm') | ||
Thomas Kluyver
|
r18467 | target_module = Unicode(None, allow_none=True, help="""requirejs module from | ||
which to load comm target.""") | ||||
MinRK
|
r13195 | |||
topic = Bytes() | ||||
def _topic_default(self): | ||||
return ('comm-%s' % self.comm_id).encode('ascii') | ||||
MinRK
|
r13205 | _open_data = Dict(help="data dict, if any, to be included in comm_open") | ||
MinRK
|
r13195 | _close_data = Dict(help="data dict, if any, to be included in comm_close") | ||
_msg_callback = Any() | ||||
_close_callback = Any() | ||||
Jason Grout
|
r18454 | _closed = Bool(True) | ||
MinRK
|
r13195 | comm_id = Unicode() | ||
def _comm_id_default(self): | ||||
return uuid.uuid4().hex | ||||
primary = Bool(True, help="Am I the primary or secondary Comm?") | ||||
MinRK
|
r13228 | def __init__(self, target_name='', data=None, **kwargs): | ||
if target_name: | ||||
kwargs['target_name'] = target_name | ||||
MinRK
|
r13195 | super(Comm, self).__init__(**kwargs) | ||
if self.primary: | ||||
MinRK
|
r13205 | # I am primary, open my peer. | ||
MinRK
|
r13197 | self.open(data) | ||
Jason Grout
|
r18454 | else: | ||
self._closed = False | ||||
MinRK
|
r13195 | |||
MinRK
|
r18329 | def _publish_msg(self, msg_type, data=None, metadata=None, buffers=None, **keys): | ||
MinRK
|
r13195 | """Helper for sending a comm message on IOPub""" | ||
Min RK
|
r20365 | if threading.current_thread().name != 'MainThread' and IOLoop.initialized(): | ||
# make sure we never send on a zmq socket outside the main IOLoop thread | ||||
IOLoop.instance().add_callback(lambda : self._publish_msg(msg_type, data, metadata, buffers, **keys)) | ||||
return | ||||
Thomas Kluyver
|
r17987 | data = {} if data is None else data | ||
metadata = {} if metadata is None else metadata | ||||
content = json_clean(dict(data=data, comm_id=self.comm_id, **keys)) | ||||
self.session.send(self.iopub_socket, msg_type, | ||||
content, | ||||
metadata=json_clean(metadata), | ||||
parent=self.kernel._parent_header, | ||||
ident=self.topic, | ||||
MinRK
|
r18329 | buffers=buffers, | ||
Thomas Kluyver
|
r17987 | ) | ||
MinRK
|
r13195 | |||
def __del__(self): | ||||
"""trigger close on gc""" | ||||
self.close() | ||||
# publishing messages | ||||
MinRK
|
r18329 | def open(self, data=None, metadata=None, buffers=None): | ||
MinRK
|
r13195 | """Open the frontend-side version of this comm""" | ||
if data is None: | ||||
data = self._open_data | ||||
Thomas Kluyver
|
r17987 | comm_manager = getattr(self.kernel, 'comm_manager', None) | ||
if comm_manager is None: | ||||
raise RuntimeError("Comms cannot be opened without a kernel " | ||||
"and a comm_manager attached to that kernel.") | ||||
comm_manager.register_comm(self) | ||||
Jason Grout
|
r18456 | try: | ||
self._publish_msg('comm_open', | ||||
data=data, metadata=metadata, buffers=buffers, | ||||
Thomas Kluyver
|
r18467 | target_name=self.target_name, | ||
target_module=self.target_module, | ||||
) | ||||
Jason Grout
|
r18456 | self._closed = False | ||
except: | ||||
comm_manager.unregister_comm(self) | ||||
raise | ||||
MinRK
|
r13195 | |||
MinRK
|
r18329 | def close(self, data=None, metadata=None, buffers=None): | ||
MinRK
|
r13195 | """Close the frontend-side version of this comm""" | ||
if self._closed: | ||||
# only close once | ||||
return | ||||
Jason Grout
|
r18454 | self._closed = True | ||
MinRK
|
r13195 | if data is None: | ||
data = self._close_data | ||||
MinRK
|
r18329 | self._publish_msg('comm_close', | ||
data=data, metadata=metadata, buffers=buffers, | ||||
) | ||||
Thomas Kluyver
|
r17987 | self.kernel.comm_manager.unregister_comm(self) | ||
MinRK
|
r13195 | |||
MinRK
|
r18329 | def send(self, data=None, metadata=None, buffers=None): | ||
MinRK
|
r13197 | """Send a message to the frontend-side version of this comm""" | ||
MinRK
|
r18329 | self._publish_msg('comm_msg', | ||
data=data, metadata=metadata, buffers=buffers, | ||||
) | ||||
MinRK
|
r13195 | |||
# registering callbacks | ||||
def on_close(self, callback): | ||||
"""Register a callback for comm_close | ||||
Will be called with the `data` of the close message. | ||||
Call `on_close(None)` to disable an existing callback. | ||||
""" | ||||
self._close_callback = callback | ||||
def on_msg(self, callback): | ||||
"""Register a callback for comm_msg | ||||
Will be called with the `data` of any comm_msg messages. | ||||
Call `on_msg(None)` to disable an existing callback. | ||||
""" | ||||
self._msg_callback = callback | ||||
# handling of incoming messages | ||||
MinRK
|
r13197 | def handle_close(self, msg): | ||
MinRK
|
r13195 | """Handle a comm_close message""" | ||
MinRK
|
r13197 | self.log.debug("handle_close[%s](%s)", self.comm_id, msg) | ||
MinRK
|
r13195 | if self._close_callback: | ||
MinRK
|
r13197 | self._close_callback(msg) | ||
MinRK
|
r13195 | |||
MinRK
|
r13197 | def handle_msg(self, msg): | ||
MinRK
|
r13195 | """Handle a comm_msg message""" | ||
MinRK
|
r13197 | self.log.debug("handle_msg[%s](%s)", self.comm_id, msg) | ||
MinRK
|
r13195 | if self._msg_callback: | ||
Thomas Kluyver
|
r17964 | if self.shell: | ||
self.shell.events.trigger('pre_execute') | ||||
MinRK
|
r13197 | self._msg_callback(msg) | ||
Thomas Kluyver
|
r17964 | if self.shell: | ||
self.shell.events.trigger('post_execute') | ||||
MinRK
|
r13195 | |||
__all__ = ['Comm'] | ||||