Skip to content
Snippets Groups Projects
pingpong.py 7.91 KiB
Newer Older
israelad's avatar
israelad committed
from charmhelpers.core.hookenv import (
    action_get,
    action_fail,
    action_set,
    config,
    status_set,
)

from charms.reactive import (
    remove_state as remove_flag,
    set_state as set_flag,
    when,
israelad's avatar
israelad committed
    when_not,
israelad's avatar
israelad committed
)
import charms.sshproxy
israelad's avatar
israelad committed
# from subprocess import (
#     Popen,
#     CalledProcessError,
#     PIPE,
# )
israelad's avatar
israelad committed


cfg = config()


@when_not('pingpong.configured')
def not_configured():
    """Check the current configuration.
    Check the current values in config to see if we have enough
    information to continue.
    """
    config_changed()


israelad's avatar
israelad committed
@when('config.changed', 'sshproxy.configured')
israelad's avatar
israelad committed
def config_changed():
israelad's avatar
israelad committed
    """Verify the configuration.
israelad's avatar
israelad committed
    Verify that the charm has been configured
    """

    try:
        status_set('maintenance', 'Verifying configuration data...')

        (validated, output) = charms.sshproxy.verify_ssh_credentials()
        if not validated:
            status_set('blocked', 'Unable to verify SSH credentials: {}'.format(
                output
            ))
israelad's avatar
israelad committed
            return

        if all(k in cfg for k in ['mode']):
            if cfg['mode'] in ['ping', 'pong']:
                set_flag('pingpong.configured')
                status_set('active', 'ready!')
                return
        status_set('blocked', 'Waiting for configuration')

    except Exception as err:
        status_set('blocked', 'Waiting for valid configuration ({})'.format(err))
israelad's avatar
israelad committed


israelad's avatar
israelad committed
@when('config.changed')
@when_not('sshproxy.configured')
def invalid_credentials():
    status_set('blocked', 'Waiting for SSH credentials.')
    pass


israelad's avatar
israelad committed
def is_ping():
    if cfg['mode'] == 'ping':
        return True
    return False


def is_pong():
    return not is_ping()


def get_port():
    port = 18888
    if is_pong():
        port = 18889
    return port

israelad's avatar
israelad committed
@when('pingpong.configured')
@when('actions.start')
def start():
    try:
Philip Joseph's avatar
Philip Joseph committed
        # Bring up the eth1 interface.
        # The selinux label on the file needs to be set correctly
        cmd = "sudo timeout 5 /sbin/restorecon -v /etc/sysconfig/network-scripts/ifcfg-eth1"
Philip Joseph's avatar
Philip Joseph committed
        result, err = charms.sshproxy._run(cmd)
    except Exception as e:
        err = "{}".format(e)
        action_fail('command failed: {}, errors: {}'.format(err, e.output))
        remove_flag('actions.start')
        return

israelad's avatar
israelad committed
    # Attempt to raise the non-mgmt interface, but ignore failures if
    # the interface is already up.
Philip Joseph's avatar
Philip Joseph committed
    try:
israelad's avatar
israelad committed
        cmd = "sudo timeout 30 /sbin/ifup eth1"
Philip Joseph's avatar
Philip Joseph committed
        result, err = charms.sshproxy._run(cmd)
    except Exception as e:
israelad's avatar
israelad committed
        pass
israelad's avatar
israelad committed
        cmd = "sudo timeout 30 /usr/bin/systemctl start {}". \
Philip Joseph's avatar
Philip Joseph committed
              format(cfg['mode'])
israelad's avatar
israelad committed
        result, err = charms.sshproxy._run(cmd)
Philip Joseph's avatar
Philip Joseph committed
    except Exception as e:
        action_fail('command failed: {}, errors: {}'.format(e, e.output))
israelad's avatar
israelad committed
    else:
Philip Joseph's avatar
Philip Joseph committed
        action_set({'stdout': result,
                    'errors': err})
israelad's avatar
israelad committed
    finally:
        remove_flag('actions.start')


@when('pingpong.configured')
@when('actions.stop')
def stop():
    try:
        # Enter the command to stop your service(s)
        cmd = "sudo timeout 30 /usr/bin/systemctl stop {}".format(cfg['mode'])
israelad's avatar
israelad committed
        result, err = charms.sshproxy._run(cmd)
Philip Joseph's avatar
Philip Joseph committed
    except Exception as e:
        action_fail('command failed: {}, errors: {}'.format(e, e.output))
israelad's avatar
israelad committed
    else:
Philip Joseph's avatar
Philip Joseph committed
        action_set({'stdout': result,
                    'errors': err})
israelad's avatar
israelad committed
    finally:
        remove_flag('actions.stop')


@when('pingpong.configured')
@when('actions.restart')
def restart():
    try:
        # Enter the command to restart your service(s)
        cmd = "sudo timeout 30 /usr/bin/systemctl restart {}".format(cfg['mode'])
israelad's avatar
israelad committed
        result, err = charms.sshproxy._run(cmd)
Philip Joseph's avatar
Philip Joseph committed
    except Exception as e:
        action_fail('command failed: {}, errors: {}'.format(e, e.output))
israelad's avatar
israelad committed
    else:
Philip Joseph's avatar
Philip Joseph committed
        action_set({'stdout': result,
                    'errors': err})
israelad's avatar
israelad committed
    finally:
        remove_flag('actions.restart')


@when('pingpong.configured')
@when('actions.set-server')
def set_server():
    try:
        # Get the target service info
        target_ip = action_get('server-ip')
        target_port = action_get('server-port')

Philip Joseph's avatar
Philip Joseph committed
        data = '{{"ip" : "{}", "port" : {} }}'. \
               format(target_ip, target_port)
israelad's avatar
israelad committed

        cmd = format_curl(
            'POST',
            '/server',
            data,
        )

        result, err = charms.sshproxy._run(cmd)
Philip Joseph's avatar
Philip Joseph committed
    except Exception as e:
        action_fail('command failed: {}, errors: {}'.format(e, e.output))
israelad's avatar
israelad committed
    else:
Philip Joseph's avatar
Philip Joseph committed
        action_set({'stdout': result,
                    'errors': err})
israelad's avatar
israelad committed
    finally:
        remove_flag('actions.set-server')


@when('pingpong.configured')
@when('actions.set-rate')
def set_rate():
    try:
        if is_ping():
            rate = action_get('rate')
Philip Joseph's avatar
Philip Joseph committed
            cmd = format_curl('POST', '/rate', '{{"rate" : {}}}'.format(rate))
israelad's avatar
israelad committed

            result, err = charms.sshproxy._run(cmd)
Philip Joseph's avatar
Philip Joseph committed
    except Exception as e:
        err = "{}".format(e)
        action_fail('command failed: {}, errors: {}'.format(err, e.output))
israelad's avatar
israelad committed
    else:
Philip Joseph's avatar
Philip Joseph committed
        action_set({'stdout': result,
                    'errors': err})
israelad's avatar
israelad committed
    finally:
        remove_flag('actions.set-rate')


@when('pingpong.configured')
@when('actions.get-rate')
def get_rate():
    try:
        if is_ping():
            cmd = format_curl('GET', '/rate')

            result, err = charms.sshproxy._run(cmd)
Philip Joseph's avatar
Philip Joseph committed
    except Exception as e:
        action_fail('command failed: {}, errors: {}'.format(e, e.output))
israelad's avatar
israelad committed
    else:
Philip Joseph's avatar
Philip Joseph committed
        action_set({'stdout': result,
                    'errors': err})
israelad's avatar
israelad committed
    finally:
        remove_flag('actions.get-rate')


@when('pingpong.configured')
@when('actions.get-state')
def get_state():
    try:
        cmd = format_curl('GET', '/state')

        result, err = charms.sshproxy._run(cmd)
Philip Joseph's avatar
Philip Joseph committed
    except Exception as e:
        action_fail('command failed: {}, errors: {}'.format(e, e.output))
israelad's avatar
israelad committed
    else:
Philip Joseph's avatar
Philip Joseph committed
        action_set({'stdout': result,
                    'errors': err})
israelad's avatar
israelad committed
    finally:
        remove_flag('actions.get-state')


@when('pingpong.configured')
@when('actions.get-stats')
def get_stats():
    try:
        cmd = format_curl('GET', '/stats')

        result, err = charms.sshproxy._run(cmd)
Philip Joseph's avatar
Philip Joseph committed
    except Exception as e:
        action_fail('command failed: {}, errors: {}'.format(e, e.output))
israelad's avatar
israelad committed
    else:
Philip Joseph's avatar
Philip Joseph committed
        action_set({'stdout': result,
                    'errors': err})
israelad's avatar
israelad committed
    finally:
        remove_flag('actions.get-stats')


@when('pingpong.configured')
Philip Joseph's avatar
Philip Joseph committed
@when('actions.start-traffic')
def start_traffic():
israelad's avatar
israelad committed
    try:
Philip Joseph's avatar
Philip Joseph committed
        cmd = format_curl('POST', '/adminstatus/state', '{"enable" : true}')
israelad's avatar
israelad committed

        result, err = charms.sshproxy._run(cmd)
Philip Joseph's avatar
Philip Joseph committed
    except Exception as e:
        action_fail('command failed: {}, errors: {}'.format(e, e.output))
israelad's avatar
israelad committed
    else:
Philip Joseph's avatar
Philip Joseph committed
        action_set({'stdout': result,
                    'errors': err})
israelad's avatar
israelad committed
    finally:
Philip Joseph's avatar
Philip Joseph committed
        remove_flag('actions.start-traffic')
israelad's avatar
israelad committed


@when('pingpong.configured')
Philip Joseph's avatar
Philip Joseph committed
@when('actions.stop-traffic')
def stop_traffic():
israelad's avatar
israelad committed
    try:
Philip Joseph's avatar
Philip Joseph committed
        cmd = format_curl('POST', '/adminstatus/state', '{"enable" : false}')
israelad's avatar
israelad committed

        result, err = charms.sshproxy._run(cmd)
Philip Joseph's avatar
Philip Joseph committed
    except Exception as e:
        action_fail('command failed: {}, errors: {}'.format(e, e.output))
israelad's avatar
israelad committed
    else:
Philip Joseph's avatar
Philip Joseph committed
        action_set({'stdout': result,
                    'errors': err})
israelad's avatar
israelad committed
    finally:
Philip Joseph's avatar
Philip Joseph committed
        remove_flag('actions.stop-traffic')
israelad's avatar
israelad committed


def format_curl(method, path, data=None):
    """ A utility function to build the curl command line. """

    # method must be GET or POST
    if method not in ['GET', 'POST']:
        # Throw exception
        return None

    # Get our service info
israelad's avatar
israelad committed
    port = get_port()
    mode = cfg['mode']

    cmd = ['curl',
           # '-D', '/dev/stdout',
Philip Joseph's avatar
Philip Joseph committed
           '-H', 'Accept: application/vnd.yang.data+xml',
           '-H', 'Content-Type: application/vnd.yang.data+json',
israelad's avatar
israelad committed
           '-X', method]

    if method == "POST" and data:
        cmd.append('-d')
Philip Joseph's avatar
Philip Joseph committed
        cmd.append('{}'.format(data))
israelad's avatar
israelad committed

    cmd.append(
        'http://{}:{}/api/v1/{}{}'.format(host, port, mode, path)
    )
    return cmd