/usr/lib/python2.7/dist-packages/landscape/amp.py is in landscape-common 16.03-0ubuntu2.
This file is owned by root:root, with mode 0o644.
The actual contents of the file can be viewed below.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 | """Communication between components in different services via twisted AMP.
The Landscape client is composed by several processes that need to talk to
each other. For example the monitor and manager processes need to talk to
the broker in order to ask it to add new messages to the outgoing queue, and
the broker needs to talk to them in order to dispatch them incoming messages
from the server.
This module implements a few conveniences built around L{landscape.lib.amp} to
let the various services connect to each other in an easy and idiomatic way,
and have them respond to standard requests like "ping" or "exit".
"""
import os
import logging
from landscape.lib.amp import (
MethodCallClientFactory, MethodCallServerFactory, RemoteObject)
class ComponentPublisher(object):
"""Publish a Landscape client component using a UNIX socket.
Other Landscape client processes can then connect to the socket and invoke
methods on the component remotely, using L{MethodCall} commands.
@param component: The component to publish. It can be any Python object
implementing the methods listed in the C{methods} class variable.
@param reactor: The L{LandscapeReactor} used to listen to the socket.
@param config: The L{Configuration} object used to build the socket path.
"""
factory = MethodCallServerFactory
def __init__(self, component, reactor, config):
self._reactor = reactor
self._config = config
self._component = component
self._port = None
self.methods = get_remote_methods(type(component)).keys()
def start(self):
"""Start accepting connections."""
factory = MethodCallServerFactory(self._component, self.methods)
socket_path = _get_socket_path(self._component, self._config)
self._port = self._reactor.listen_unix(socket_path, factory)
def stop(self):
"""Stop accepting connections."""
return self._port.stopListening()
def get_remote_methods(klass):
"""Get all the remote methods declared on a class.
@param klass: A class to search for AMP-exposed methods.
"""
remote_methods = {}
for attribute_name in dir(klass):
potential_method = getattr(klass, attribute_name)
name = getattr(potential_method, "amp_exposed", None)
if name is not None:
remote_methods[name] = potential_method
return remote_methods
def remote(method):
"""
A decorator for marking a method as remotely accessible as a method on a
component.
"""
method.amp_exposed = method.__name__
return method
class ComponentConnector(object):
"""Utility superclass for creating connections with a Landscape component.
@cvar component: The class of the component to connect to, it is expected
to define a C{name} class attribute, which will be used to find out
the socket to use. It must be defined by sub-classes.
@cvar factory: The factory class to use for building protocols.
@cvar remote: The L{RemoteObject} class or sub-class used for building
remote objects.
@param reactor: A L{LandscapeReactor} object.
@param config: A L{LandscapeConfiguration}.
@param retry_on_reconnect: If C{True} the remote object built by this
connector will retry L{MethodCall}s that failed due to lost
connections.
@see: L{MethodCallClientFactory}.
"""
factory = MethodCallClientFactory
component = None # Must be defined by sub-classes
remote = RemoteObject
def __init__(self, reactor, config, retry_on_reconnect=False):
self._reactor = reactor
self._config = config
self._retry_on_reconnect = retry_on_reconnect
self._connector = None
def connect(self, max_retries=None, factor=None, quiet=False):
"""Connect to the remote Landscape component.
If the connection is lost after having been established, and then
it is established again by the reconnect mechanism, an event will
be fired.
@param max_retries: If given, the connector will keep trying to connect
up to that number of times, if the first connection attempt fails.
@param factor: Optionally a float indicating by which factor the
delay between subsequent retries should increase. Smaller values
result in a faster reconnection attempts pace.
@param quiet: A boolean indicating whether to log errors.
"""
factory = self.factory(self._reactor._reactor)
factory.initialDelay = factory.delay = 0.05
factory.retryOnReconnect = self._retry_on_reconnect
factory.remote = self.remote
factory.maxRetries = max_retries
if factor:
factory.factor = factor
def fire_reconnect(ignored):
self._reactor.fire("%s-reconnect" % self.component.name)
def connected(remote):
factory.notifyOnConnect(fire_reconnect)
return remote
def log_error(failure):
logging.error("Error while connecting to %s", self.component.name)
return failure
socket_path = _get_socket_path(self.component, self._config)
deferred = factory.getRemoteObject()
self._connector = self._reactor.connect_unix(socket_path, factory)
if not quiet:
deferred.addErrback(log_error)
return deferred.addCallback(connected)
def disconnect(self):
"""Disconnect the L{RemoteObject} that we have created."""
if self._connector is not None:
factory = self._connector.factory
factory.stopTrying()
self._connector.disconnect()
self._connector = None
def _get_socket_path(component, config):
return os.path.join(config.sockets_path, component.name + ".sock")
|