Show More
enginefc.py
548 lines
| 19.3 KiB
| text/x-python
|
PythonLexer
Brian E Granger
|
r1234 | # encoding: utf-8 | ||
# -*- test-case-name: IPython.kernel.test.test_enginepb -*- | ||||
""" | ||||
Expose the IPython EngineService using the Foolscap network protocol. | ||||
Foolscap is a high-performance and secure network protocol. | ||||
""" | ||||
__docformat__ = "restructuredtext en" | ||||
#------------------------------------------------------------------------------- | ||||
# Copyright (C) 2008 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 | ||||
#------------------------------------------------------------------------------- | ||||
import os, time | ||||
import cPickle as pickle | ||||
from twisted.python import components, log, failure | ||||
from twisted.python.failure import Failure | ||||
from twisted.internet import defer, reactor, threads | ||||
from twisted.internet.interfaces import IProtocolFactory | ||||
from zope.interface import Interface, implements, Attribute | ||||
from twisted.internet.base import DelayedCall | ||||
DelayedCall.debug = True | ||||
from foolscap import Referenceable, DeadReferenceError | ||||
from foolscap.referenceable import RemoteReference | ||||
from IPython.kernel.pbutil import packageFailure, unpackageFailure | ||||
from IPython.kernel.util import printer | ||||
from IPython.kernel.twistedutil import gatherBoth | ||||
from IPython.kernel import newserialized | ||||
from IPython.kernel.error import ProtocolError | ||||
from IPython.kernel import controllerservice | ||||
from IPython.kernel.controllerservice import IControllerBase | ||||
from IPython.kernel.engineservice import \ | ||||
IEngineBase, \ | ||||
IEngineQueued, \ | ||||
EngineService, \ | ||||
StrictDict | ||||
from IPython.kernel.pickleutil import \ | ||||
can, \ | ||||
canDict, \ | ||||
canSequence, \ | ||||
uncan, \ | ||||
uncanDict, \ | ||||
uncanSequence | ||||
#------------------------------------------------------------------------------- | ||||
# The client (Engine) side of things | ||||
#------------------------------------------------------------------------------- | ||||
# Expose a FC interface to the EngineService | ||||
class IFCEngine(Interface): | ||||
"""An interface that exposes an EngineService over Foolscap. | ||||
The methods in this interface are similar to those from IEngine, | ||||
but their arguments and return values slightly different to reflect | ||||
that FC cannot send arbitrary objects. We handle this by pickling/ | ||||
unpickling that the two endpoints. | ||||
If a remote or local exception is raised, the appropriate Failure | ||||
will be returned instead. | ||||
""" | ||||
pass | ||||
class FCEngineReferenceFromService(Referenceable, object): | ||||
"""Adapt an `IEngineBase` to an `IFCEngine` implementer. | ||||
This exposes an `IEngineBase` to foolscap by adapting it to a | ||||
`foolscap.Referenceable`. | ||||
See the documentation of the `IEngineBase` methods for more details. | ||||
""" | ||||
implements(IFCEngine) | ||||
def __init__(self, service): | ||||
assert IEngineBase.providedBy(service), \ | ||||
"IEngineBase is not provided by" + repr(service) | ||||
self.service = service | ||||
self.collectors = {} | ||||
def remote_get_id(self): | ||||
return self.service.id | ||||
def remote_set_id(self, id): | ||||
self.service.id = id | ||||
def _checkProperties(self, result): | ||||
dosync = self.service.properties.modified | ||||
self.service.properties.modified = False | ||||
return (dosync and pickle.dumps(self.service.properties, 2)), result | ||||
def remote_execute(self, lines): | ||||
d = self.service.execute(lines) | ||||
d.addErrback(packageFailure) | ||||
d.addCallback(self._checkProperties) | ||||
d.addErrback(packageFailure) | ||||
#d.addCallback(lambda r: log.msg("Got result: " + str(r))) | ||||
return d | ||||
#--------------------------------------------------------------------------- | ||||
# Old version of push | ||||
#--------------------------------------------------------------------------- | ||||
def remote_push(self, pNamespace): | ||||
try: | ||||
namespace = pickle.loads(pNamespace) | ||||
except: | ||||
return defer.fail(failure.Failure()).addErrback(packageFailure) | ||||
else: | ||||
return self.service.push(namespace).addErrback(packageFailure) | ||||
#--------------------------------------------------------------------------- | ||||
# pull | ||||
#--------------------------------------------------------------------------- | ||||
def remote_pull(self, keys): | ||||
d = self.service.pull(keys) | ||||
d.addCallback(pickle.dumps, 2) | ||||
d.addErrback(packageFailure) | ||||
return d | ||||
#--------------------------------------------------------------------------- | ||||
# push/pullFuction | ||||
#--------------------------------------------------------------------------- | ||||
def remote_push_function(self, pNamespace): | ||||
try: | ||||
namespace = pickle.loads(pNamespace) | ||||
except: | ||||
return defer.fail(failure.Failure()).addErrback(packageFailure) | ||||
else: | ||||
# The usage of globals() here is an attempt to bind any pickled functions | ||||
# to the globals of this module. What we really want is to have it bound | ||||
# to the globals of the callers module. This will require walking the | ||||
# stack. BG 10/3/07. | ||||
namespace = uncanDict(namespace, globals()) | ||||
return self.service.push_function(namespace).addErrback(packageFailure) | ||||
def remote_pull_function(self, keys): | ||||
d = self.service.pull_function(keys) | ||||
if len(keys)>1: | ||||
d.addCallback(canSequence) | ||||
elif len(keys)==1: | ||||
d.addCallback(can) | ||||
d.addCallback(pickle.dumps, 2) | ||||
d.addErrback(packageFailure) | ||||
return d | ||||
#--------------------------------------------------------------------------- | ||||
# Other methods | ||||
#--------------------------------------------------------------------------- | ||||
def remote_get_result(self, i=None): | ||||
return self.service.get_result(i).addErrback(packageFailure) | ||||
def remote_reset(self): | ||||
return self.service.reset().addErrback(packageFailure) | ||||
def remote_kill(self): | ||||
return self.service.kill().addErrback(packageFailure) | ||||
def remote_keys(self): | ||||
return self.service.keys().addErrback(packageFailure) | ||||
#--------------------------------------------------------------------------- | ||||
# push/pull_serialized | ||||
#--------------------------------------------------------------------------- | ||||
def remote_push_serialized(self, pNamespace): | ||||
try: | ||||
namespace = pickle.loads(pNamespace) | ||||
except: | ||||
return defer.fail(failure.Failure()).addErrback(packageFailure) | ||||
else: | ||||
d = self.service.push_serialized(namespace) | ||||
return d.addErrback(packageFailure) | ||||
def remote_pull_serialized(self, keys): | ||||
d = self.service.pull_serialized(keys) | ||||
d.addCallback(pickle.dumps, 2) | ||||
d.addErrback(packageFailure) | ||||
return d | ||||
#--------------------------------------------------------------------------- | ||||
# Properties interface | ||||
#--------------------------------------------------------------------------- | ||||
def remote_set_properties(self, pNamespace): | ||||
try: | ||||
namespace = pickle.loads(pNamespace) | ||||
except: | ||||
return defer.fail(failure.Failure()).addErrback(packageFailure) | ||||
else: | ||||
return self.service.set_properties(namespace).addErrback(packageFailure) | ||||
def remote_get_properties(self, keys=None): | ||||
d = self.service.get_properties(keys) | ||||
d.addCallback(pickle.dumps, 2) | ||||
d.addErrback(packageFailure) | ||||
return d | ||||
def remote_has_properties(self, keys): | ||||
d = self.service.has_properties(keys) | ||||
d.addCallback(pickle.dumps, 2) | ||||
d.addErrback(packageFailure) | ||||
return d | ||||
def remote_del_properties(self, keys): | ||||
d = self.service.del_properties(keys) | ||||
d.addErrback(packageFailure) | ||||
return d | ||||
def remote_clear_properties(self): | ||||
d = self.service.clear_properties() | ||||
d.addErrback(packageFailure) | ||||
return d | ||||
components.registerAdapter(FCEngineReferenceFromService, | ||||
IEngineBase, | ||||
IFCEngine) | ||||
#------------------------------------------------------------------------------- | ||||
# Now the server (Controller) side of things | ||||
#------------------------------------------------------------------------------- | ||||
class EngineFromReference(object): | ||||
"""Adapt a `RemoteReference` to an `IEngineBase` implementing object. | ||||
When an engine connects to a controller, it calls the `register_engine` | ||||
method of the controller and passes the controller a `RemoteReference` to | ||||
itself. This class is used to adapt this `RemoteReference` to an object | ||||
that implements the full `IEngineBase` interface. | ||||
See the documentation of `IEngineBase` for details on the methods. | ||||
""" | ||||
implements(IEngineBase) | ||||
def __init__(self, reference): | ||||
self.reference = reference | ||||
self._id = None | ||||
self._properties = StrictDict() | ||||
self.currentCommand = None | ||||
def callRemote(self, *args, **kwargs): | ||||
try: | ||||
return self.reference.callRemote(*args, **kwargs) | ||||
except DeadReferenceError: | ||||
self.notifier() | ||||
self.stopNotifying(self.notifier) | ||||
return defer.fail() | ||||
def get_id(self): | ||||
"""Return the Engines id.""" | ||||
return self._id | ||||
def set_id(self, id): | ||||
"""Set the Engines id.""" | ||||
self._id = id | ||||
return self.callRemote('set_id', id) | ||||
id = property(get_id, set_id) | ||||
def syncProperties(self, r): | ||||
try: | ||||
psync, result = r | ||||
except (ValueError, TypeError): | ||||
return r | ||||
else: | ||||
if psync: | ||||
log.msg("sync properties") | ||||
pick = self.checkReturnForFailure(psync) | ||||
if isinstance(pick, failure.Failure): | ||||
self.properties = pick | ||||
return pick | ||||
else: | ||||
self.properties = pickle.loads(pick) | ||||
return result | ||||
def _set_properties(self, dikt): | ||||
self._properties.clear() | ||||
self._properties.update(dikt) | ||||
def _get_properties(self): | ||||
if isinstance(self._properties, failure.Failure): | ||||
self._properties.raiseException() | ||||
return self._properties | ||||
properties = property(_get_properties, _set_properties) | ||||
#--------------------------------------------------------------------------- | ||||
# Methods from IEngine | ||||
#--------------------------------------------------------------------------- | ||||
#--------------------------------------------------------------------------- | ||||
# execute | ||||
#--------------------------------------------------------------------------- | ||||
def execute(self, lines): | ||||
# self._needProperties = True | ||||
d = self.callRemote('execute', lines) | ||||
d.addCallback(self.syncProperties) | ||||
return d.addCallback(self.checkReturnForFailure) | ||||
#--------------------------------------------------------------------------- | ||||
# push | ||||
#--------------------------------------------------------------------------- | ||||
def push(self, namespace): | ||||
try: | ||||
package = pickle.dumps(namespace, 2) | ||||
except: | ||||
return defer.fail(failure.Failure()) | ||||
else: | ||||
if isinstance(package, failure.Failure): | ||||
return defer.fail(package) | ||||
else: | ||||
d = self.callRemote('push', package) | ||||
return d.addCallback(self.checkReturnForFailure) | ||||
#--------------------------------------------------------------------------- | ||||
# pull | ||||
#--------------------------------------------------------------------------- | ||||
def pull(self, keys): | ||||
d = self.callRemote('pull', keys) | ||||
d.addCallback(self.checkReturnForFailure) | ||||
d.addCallback(pickle.loads) | ||||
return d | ||||
#--------------------------------------------------------------------------- | ||||
# push/pull_function | ||||
#--------------------------------------------------------------------------- | ||||
def push_function(self, namespace): | ||||
try: | ||||
package = pickle.dumps(canDict(namespace), 2) | ||||
except: | ||||
return defer.fail(failure.Failure()) | ||||
else: | ||||
if isinstance(package, failure.Failure): | ||||
return defer.fail(package) | ||||
else: | ||||
d = self.callRemote('push_function', package) | ||||
return d.addCallback(self.checkReturnForFailure) | ||||
def pull_function(self, keys): | ||||
d = self.callRemote('pull_function', keys) | ||||
d.addCallback(self.checkReturnForFailure) | ||||
d.addCallback(pickle.loads) | ||||
# The usage of globals() here is an attempt to bind any pickled functions | ||||
# to the globals of this module. What we really want is to have it bound | ||||
# to the globals of the callers module. This will require walking the | ||||
# stack. BG 10/3/07. | ||||
if len(keys)==1: | ||||
d.addCallback(uncan, globals()) | ||||
elif len(keys)>1: | ||||
d.addCallback(uncanSequence, globals()) | ||||
return d | ||||
#--------------------------------------------------------------------------- | ||||
# Other methods | ||||
#--------------------------------------------------------------------------- | ||||
def get_result(self, i=None): | ||||
return self.callRemote('get_result', i).addCallback(self.checkReturnForFailure) | ||||
def reset(self): | ||||
self._refreshProperties = True | ||||
d = self.callRemote('reset') | ||||
d.addCallback(self.syncProperties) | ||||
return d.addCallback(self.checkReturnForFailure) | ||||
def kill(self): | ||||
#this will raise pb.PBConnectionLost on success | ||||
d = self.callRemote('kill') | ||||
d.addCallback(self.syncProperties) | ||||
d.addCallback(self.checkReturnForFailure) | ||||
d.addErrback(self.killBack) | ||||
return d | ||||
def killBack(self, f): | ||||
log.msg('filling engine: %s' % f) | ||||
return None | ||||
def keys(self): | ||||
return self.callRemote('keys').addCallback(self.checkReturnForFailure) | ||||
#--------------------------------------------------------------------------- | ||||
# Properties methods | ||||
#--------------------------------------------------------------------------- | ||||
def set_properties(self, properties): | ||||
try: | ||||
package = pickle.dumps(properties, 2) | ||||
except: | ||||
return defer.fail(failure.Failure()) | ||||
else: | ||||
if isinstance(package, failure.Failure): | ||||
return defer.fail(package) | ||||
else: | ||||
d = self.callRemote('set_properties', package) | ||||
return d.addCallback(self.checkReturnForFailure) | ||||
return d | ||||
def get_properties(self, keys=None): | ||||
d = self.callRemote('get_properties', keys) | ||||
d.addCallback(self.checkReturnForFailure) | ||||
d.addCallback(pickle.loads) | ||||
return d | ||||
def has_properties(self, keys): | ||||
d = self.callRemote('has_properties', keys) | ||||
d.addCallback(self.checkReturnForFailure) | ||||
d.addCallback(pickle.loads) | ||||
return d | ||||
def del_properties(self, keys): | ||||
d = self.callRemote('del_properties', keys) | ||||
d.addCallback(self.checkReturnForFailure) | ||||
# d.addCallback(pickle.loads) | ||||
return d | ||||
def clear_properties(self): | ||||
d = self.callRemote('clear_properties') | ||||
d.addCallback(self.checkReturnForFailure) | ||||
return d | ||||
#--------------------------------------------------------------------------- | ||||
# push/pull_serialized | ||||
#--------------------------------------------------------------------------- | ||||
def push_serialized(self, namespace): | ||||
"""Older version of pushSerialize.""" | ||||
try: | ||||
package = pickle.dumps(namespace, 2) | ||||
except: | ||||
return defer.fail(failure.Failure()) | ||||
else: | ||||
if isinstance(package, failure.Failure): | ||||
return defer.fail(package) | ||||
else: | ||||
d = self.callRemote('push_serialized', package) | ||||
return d.addCallback(self.checkReturnForFailure) | ||||
def pull_serialized(self, keys): | ||||
d = self.callRemote('pull_serialized', keys) | ||||
d.addCallback(self.checkReturnForFailure) | ||||
d.addCallback(pickle.loads) | ||||
return d | ||||
#--------------------------------------------------------------------------- | ||||
# Misc | ||||
#--------------------------------------------------------------------------- | ||||
def checkReturnForFailure(self, r): | ||||
"""See if a returned value is a pickled Failure object. | ||||
To distinguish between general pickled objects and pickled Failures, the | ||||
other side should prepend the string FAILURE: to any pickled Failure. | ||||
""" | ||||
return unpackageFailure(r) | ||||
components.registerAdapter(EngineFromReference, | ||||
RemoteReference, | ||||
IEngineBase) | ||||
#------------------------------------------------------------------------------- | ||||
# Now adapt an IControllerBase to incoming FC connections | ||||
#------------------------------------------------------------------------------- | ||||
class IFCControllerBase(Interface): | ||||
""" | ||||
Interface that tells how an Engine sees a Controller. | ||||
In our architecture, the Controller listens for Engines to connect | ||||
and register. This interface defines that registration method as it is | ||||
exposed over the Foolscap network protocol | ||||
""" | ||||
def remote_register_engine(self, engineReference, id=None, pid=None, pproperties=None): | ||||
""" | ||||
Register new engine on the controller. | ||||
Engines must call this upon connecting to the controller if they | ||||
want to do work for the controller. | ||||
See the documentation of `IControllerCore` for more details. | ||||
""" | ||||
class FCRemoteEngineRefFromService(Referenceable): | ||||
""" | ||||
Adapt an `IControllerBase` to an `IFCControllerBase`. | ||||
""" | ||||
implements(IFCControllerBase) | ||||
def __init__(self, service): | ||||
assert IControllerBase.providedBy(service), \ | ||||
"IControllerBase is not provided by " + repr(service) | ||||
self.service = service | ||||
def remote_register_engine(self, engine_reference, id=None, pid=None, pproperties=None): | ||||
# First adapt the engine_reference to a basic non-queued engine | ||||
engine = IEngineBase(engine_reference) | ||||
if pproperties: | ||||
engine.properties = pickle.loads(pproperties) | ||||
# Make it an IQueuedEngine before registration | ||||
remote_engine = IEngineQueued(engine) | ||||
# Get the ip/port of the remote side | ||||
peer_address = engine_reference.tracker.broker.transport.getPeer() | ||||
ip = peer_address.host | ||||
port = peer_address.port | ||||
reg_dict = self.service.register_engine(remote_engine, id, ip, port, pid) | ||||
# Now setup callback for disconnect and unregistering the engine | ||||
def notify(*args): | ||||
return self.service.unregister_engine(reg_dict['id']) | ||||
engine_reference.tracker.broker.notifyOnDisconnect(notify) | ||||
engine.notifier = notify | ||||
engine.stopNotifying = engine_reference.tracker.broker.dontNotifyOnDisconnect | ||||
return reg_dict | ||||
components.registerAdapter(FCRemoteEngineRefFromService, | ||||
IControllerBase, | ||||
IFCControllerBase) | ||||