blob: 7c388a9d817e5e91756a95abbda7c6cb04a65a7e [file] [log] [blame] [edit]
#!/usr/bin/python
# Copyright (c) 2012 The Chromium OS Authors. All rights reserved.
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
import atexit
import errno
import logging
import re
import sys
import socket
import threading
import xmlrpclib
import rpm_controller
import rpm_logging_config
from config import rpm_config
from MultiThreadedXMLRPCServer import MultiThreadedXMLRPCServer
from rpm_infrastructure_exception import RPMInfrastructureException
import common
from autotest_lib.site_utils.rpm_control_system import utils
LOG_FILENAME_FORMAT = rpm_config.get('GENERAL','dispatcher_logname_format')
class RPMDispatcher(object):
"""
This class is the RPM dispatcher server and it is responsible for
communicating directly to the RPM devices to change a DUT's outlet status.
When an RPMDispatcher is initialized it registers itself with the frontend
server, who will field out outlet requests to this dispatcher.
Once a request is received the dispatcher looks up the RPMController
instance for the given DUT and then queues up the request and blocks until
it is processed.
@var _address: IP address or Hostname of this dispatcher server.
@var _frontend_server: URI of the frontend server.
@var _lock: Lock used to synchronize access to _worker_dict.
@var _port: Port assigned to this server instance.
@var _worker_dict: Dictionary mapping RPM hostname's to RPMController
instances.
"""
def __init__(self, address, port):
"""
RPMDispatcher constructor.
Initialized instance vars and registers this server with the frontend
server.
@param address: Address of this dispatcher server.
@param port: Port assigned to this dispatcher server.
@raise RPMInfrastructureException: Raised if the dispatch server is
unable to register with the frontend
server.
"""
self._address = address
self._port = port
self._lock = threading.Lock()
self._worker_dict = {}
# We assume that the frontend server and dispatchers are running on the
# same host, and the frontend server is listening for connections from
# the external world.
frontend_server_port = rpm_config.getint('RPM_INFRASTRUCTURE',
'frontend_port')
self._frontend_server = 'http://%s:%d' % (socket.gethostname(),
frontend_server_port)
logging.info('Registering this rpm dispatcher with the frontend '
'server at %s.', self._frontend_server)
client = xmlrpclib.ServerProxy(self._frontend_server)
# De-register with the frontend when the dispatcher exit's.
atexit.register(self._unregister)
try:
client.register_dispatcher(self._get_serveruri())
except socket.error as er:
err_msg = ('Unable to register with frontend server. Error: %s.' %
errno.errorcode[er.errno])
logging.error(err_msg)
raise RPMInfrastructureException(err_msg)
def _worker_dict_put(self, key, value):
"""
Private method used to synchronize access to _worker_dict.
@param key: key value we are using to access _worker_dict.
@param value: value we are putting into _worker_dict.
"""
with self._lock:
self._worker_dict[key] = value
def _worker_dict_get(self, key):
"""
Private method used to synchronize access to _worker_dict.
@param key: key value we are using to access _worker_dict.
@return: value found when accessing _worker_dict
"""
with self._lock:
return self._worker_dict.get(key)
def is_up(self):
"""
Allows the frontend server to see if the dispatcher server is up before
attempting to queue requests.
@return: True. If connection fails, the client proxy will throw a socket
error on the client side.
"""
return True
def queue_request(self, powerunit_info_dict, new_state):
"""
Looks up the appropriate RPMController instance for the device and queues
up the request.
@param powerunit_info_dict: A dictionary, containing the attribute/values
of an unmarshalled PowerUnitInfo instance.
@param new_state: [ON, OFF, CYCLE] state we want to the change the
outlet to.
@return: True if the attempt to change power state was successful,
False otherwise.
"""
powerunit_info = utils.PowerUnitInfo(**powerunit_info_dict)
logging.info('Received request to set device: %s to state: %s',
powerunit_info.device_hostname, new_state)
rpm_controller = self._get_rpm_controller(
powerunit_info.powerunit_hostname,
powerunit_info.hydra_hostname)
return rpm_controller.queue_request(powerunit_info, new_state)
def _get_rpm_controller(self, rpm_hostname, hydra_hostname=None):
"""
Private method that retreives the appropriate RPMController instance
for this RPM Hostname or calls _create_rpm_controller it if it does not
already exist.
@param rpm_hostname: hostname of the RPM whose RPMController we want.
@return: RPMController instance responsible for this RPM.
"""
if not rpm_hostname:
return None
rpm_controller = self._worker_dict_get(rpm_hostname)
if not rpm_controller:
rpm_controller = self._create_rpm_controller(
rpm_hostname, hydra_hostname)
self._worker_dict_put(rpm_hostname, rpm_controller)
return rpm_controller
def _create_rpm_controller(self, rpm_hostname, hydra_hostname):
"""
Determines the type of RPMController required and initializes it.
@param rpm_hostname: Hostname of the RPM we need to communicate with.
@return: RPMController instance responsible for this RPM.
"""
hostname_elements = rpm_hostname.split('-')
if hostname_elements[-2] == 'poe':
# POE switch hostname looks like 'chromeos2-poe-switch1'.
logging.info('The controller is a Cisco POE switch.')
return rpm_controller.CiscoPOEController(rpm_hostname)
else:
# The device is an RPM.
rack_id = hostname_elements[-2]
rpm_typechecker = re.compile('rack[0-9]+[a-z]+')
if rpm_typechecker.match(rack_id):
logging.info('RPM is a webpowered device.')
return rpm_controller.WebPoweredRPMController(rpm_hostname)
else:
logging.info('RPM is a Sentry CDU device.')
return rpm_controller.SentryRPMController(
hostname=rpm_hostname,
hydra_hostname=hydra_hostname)
def _get_serveruri(self):
"""
Formats the _address and _port into a meaningful URI string.
@return: URI of this dispatch server.
"""
return 'http://%s:%d' % (self._address, self._port)
def _unregister(self):
"""
Tells the frontend server that this dispatch server is shutting down and
to unregister it.
Called by atexit.
@raise RPMInfrastructureException: Raised if the dispatch server is
unable to unregister with the
frontend server.
"""
logging.info('Dispatch server shutting down. Unregistering with RPM '
'frontend server.')
client = xmlrpclib.ServerProxy(self._frontend_server)
try:
client.unregister_dispatcher(self._get_serveruri())
except socket.error as er:
err_msg = ('Unable to unregister with frontend server. Error: %s.' %
errno.errorcode[er.errno])
logging.error(err_msg)
raise RPMInfrastructureException(err_msg)
def launch_server_on_unused_port():
"""
Looks up an unused port on this host and launches the xmlrpc server.
Useful for testing by running multiple dispatch servers on the same host.
@return: server,port - server object and the port that which it is listening
to.
"""
address = socket.gethostbyname(socket.gethostname())
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# Set this socket to allow reuse.
sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
sock.bind(('', 0))
port = sock.getsockname()[1]
server = MultiThreadedXMLRPCServer((address, port),
allow_none=True)
sock.close()
return server, port
if __name__ == '__main__':
"""
Main function used to launch the dispatch server. Creates an instance of
RPMDispatcher and registers it to a MultiThreadedXMLRPCServer instance.
"""
if len(sys.argv) != 2:
print 'Usage: ./%s <log_file_name>' % sys.argv[0]
sys.exit(1)
rpm_logging_config.start_log_server(sys.argv[1], LOG_FILENAME_FORMAT)
rpm_logging_config.set_up_logging_to_server()
# Get the local ip _address and set the server to utilize it.
address = socket.gethostbyname(socket.gethostname())
server, port = launch_server_on_unused_port()
rpm_dispatcher = RPMDispatcher(address, port)
server.register_instance(rpm_dispatcher)
server.serve_forever()