intracom-telecom-sdn/multinet

View on GitHub
multi/master.py

Summary

Maintainability
D
2 days
Test Coverage
#!/usr/bin/env python

# Copyright (c) 2015 Intracom S.A. Telecom Solutions. All rights reserved.
#
# This program and the accompanying materials are made available under the
# terms of the Eclipse Public License v1.0 which accompanies this distribution,
# and is available at http://www.eclipse.org/legal/epl-v10.html

"""
With this module we create the master REST server
to manage the distributed topologies
"""

import bottle
import logging
import util.multinet_requests as m_util
import time

# We must define logging level separately because this module runs
# independently.
logging.basicConfig(level=logging.DEBUG)

WORKER_PORT_LIST = []
WORKER_IP_LIST = []

@bottle.route('/init', method='POST')
def init():
    """
    Broadcast the POST request to the 'init' endpoint of the workers
    Aggregate the responses

    Args:
      controller_ip_address (str): The IP address of the controller
      controller_of_port (int): The OpenFlow port of the controller
      switch_type (str): The type of soft switch to use for the emulation
      topo_type (str): The type of the topology we want to build
      topo_size (int): The size of the topology we want to build
      group_size (int): The number of switches in a gorup for gradual bootup
      group_delay (int): The delay between the bootup of each group
      hosts_per_switch (int): The number of hosts connected to each switch

    Returns:
        requests.models.Response: An HTTP Response with the aggregated
        status codes and bodies of the broadcasted requests
    """

    logging.info('[ip list] {0}'.format(WORKER_IP_LIST))
    data = bottle.request.json
    logging.info('[init] topology type: {0}'.format(data['topo']['topo_type']))
    reqs = m_util.broadcast_cmd(WORKER_IP_LIST, WORKER_PORT_LIST, 'init',
                                data)

    stat, bod = m_util.aggregate_broadcast_response(reqs)
    return bottle.HTTPResponse(status=stat, body=bod)


@bottle.route('/start', method='POST')
def start():
    """
    Broadcast the POST request to the 'start' endpoint of the workers
    Aggregate the responses

    Returns:
        requests.models.Response: An HTTP Response with the aggregated
        status codes and bodies of the broadcasted requests
    """
    data = bottle.request.json
    reqs = m_util.broadcast_cmd(WORKER_IP_LIST, WORKER_PORT_LIST, 'start', data)
    stat, bod = m_util.aggregate_broadcast_response(reqs)
    return bottle.HTTPResponse(status=stat, body=bod)


@bottle.route('/detect_hosts', method='POST')
def detect_hosts():
    """
    Broadcast the POST request to the 'detect_hosts' endpoint of the workers
    Aggregate the responses

    Returns:
        requests.models.Response: An HTTP Response with the aggregated
        status codes and bodies of the broadcasted requests
    """
    data = bottle.request.json
    reqs = m_util.broadcast_cmd(WORKER_IP_LIST, WORKER_PORT_LIST,
                                'detect_hosts', data)
    stat, bod = m_util.aggregate_broadcast_response(reqs)
    return bottle.HTTPResponse(status=stat, body=bod)


@bottle.route('/get_switches', method='POST')
def get_switches():
    """
    Broadcast the POST request to the 'get_switches' endpoint of the workers
    Aggregate the responses

    Returns:
        requests.models.Response: An HTTP Response with the aggregated
        status codes and bodies of the broadcasted requests
    """
    data = bottle.request.json
    reqs = m_util.broadcast_cmd(WORKER_IP_LIST, WORKER_PORT_LIST,
                                'get_switches', data)
    stat, bod = m_util.aggregate_broadcast_response(reqs)
    return bottle.HTTPResponse(status=stat, body=bod)


@bottle.route('/get_flows', method='POST')
def get_flows():
    """
    Broadcast the POST request to the 'get_flows' endpoint of the workers
    Aggregate the responses

    Returns:
        requests.models.Response: An HTTP Response with the aggregated
        status codes and bodies of the broadcasted requests
    """
    data = bottle.request.json
    t_start = time.time()
    reqs = m_util.broadcast_cmd(WORKER_IP_LIST, WORKER_PORT_LIST,
                                'get_flows', data)
    stat, bod = m_util.aggregate_broadcast_response(reqs)
    get_flow_latency = time.time() - t_start
    logging.info('[get_flows] Flow latency interval on master: {0} [sec]'.
                 format(get_flow_latency))
    return bottle.HTTPResponse(status=stat, body=bod)


@bottle.route('/stop', method='POST')
def stop():
    """
    Broadcast the POST request to the 'stop' endpoint of the workers
    Aggregate the responses

    Returns:
        requests.models.Response: An HTTP Response with the aggregated
        status codes and bodies of the broadcasted requests
    """
    data = bottle.request.json
    reqs = m_util.broadcast_cmd(WORKER_IP_LIST, WORKER_PORT_LIST, 'stop', data)
    stat, bod = m_util.aggregate_broadcast_response(reqs)
    return bottle.HTTPResponse(status=stat, body=bod)


@bottle.route('/ping_all', method='POST')
def ping_all():
    """
    Broadcast the POST request to the 'ping_all' endpoint of the workers
    Aggregate the responses

    Returns:
        requests.models.Response: An HTTP Response with the aggregated
        status codes and bodies of the broadcasted requests
    """
    data = bottle.request.json
    reqs = m_util.broadcast_cmd(WORKER_IP_LIST, WORKER_PORT_LIST, 'ping_all',
                                data)
    stat, bod = m_util.aggregate_broadcast_response(reqs)
    return bottle.HTTPResponse(status=stat, body=bod)


@bottle.route('/generate_traffic', method='POST')
def generate_traffic():
    """
    Broadcast the POST request to the 'generate_traffic' endpoint of the workers
    Aggregate the responses

    Returns:
        requests.models.Response: An HTTP Response with the aggregated
        status codes and bodies of the broadcasted requests
    """
    data = bottle.request.json
    reqs = m_util.broadcast_cmd(WORKER_IP_LIST, WORKER_PORT_LIST,
                                'generate_traffic', data)
    stat, bod = m_util.aggregate_broadcast_response(reqs)
    return bottle.HTTPResponse(status=stat, body=bod)


def rest_start():
    """
    Parse the command line arguments and start the master server
    """
    global WORKER_PORT_LIST
    global WORKER_IP_LIST

    args = m_util.parse_arguments()
    runtime_config = m_util.parse_json_conf(args.json_config)

    master_ip = runtime_config['master_ip']
    master_port = runtime_config['master_port']
    WORKER_IP_LIST = runtime_config['worker_ip_list']
    WORKER_PORT_LIST = runtime_config['worker_port_list']

    bottle.run(host=master_ip, port=master_port, debug=True)

if __name__ == '__main__':
    logging.getLogger().setLevel(logging.DEBUG)
    rest_start()