d1ff1526f9
- Renamed actions related to Solutions API - Added openstack-release config to provide solution version Signed-off-by: plumgrid <Junaid Ali>
638 lines
20 KiB
Python
638 lines
20 KiB
Python
# Copyright (c) 2015, PLUMgrid Inc, http://plumgrid.com
|
|
|
|
# This file contains functions used by the hooks to deploy PLUMgrid Director.
|
|
|
|
import pg_dir_context
|
|
import subprocess
|
|
import time
|
|
import os
|
|
import json
|
|
from collections import OrderedDict
|
|
from socket import gethostname as get_unit_hostname
|
|
from copy import deepcopy
|
|
from charmhelpers.contrib.openstack.neutron import neutron_plugin_attribute
|
|
from charmhelpers.contrib.openstack import templating
|
|
from charmhelpers.contrib.storage.linux.ceph import modprobe
|
|
from charmhelpers.core.hookenv import (
|
|
log,
|
|
config,
|
|
unit_get,
|
|
status_set
|
|
)
|
|
from charmhelpers.contrib.network.ip import (
|
|
get_iface_from_addr,
|
|
get_bridges,
|
|
get_bridge_nics,
|
|
is_ip,
|
|
get_iface_addr,
|
|
get_host_ip
|
|
)
|
|
from charmhelpers.core.host import (
|
|
service_start,
|
|
service_stop,
|
|
service_running,
|
|
path_hash,
|
|
set_nic_mtu
|
|
)
|
|
from charmhelpers.fetch import (
|
|
apt_cache,
|
|
apt_install
|
|
)
|
|
from charmhelpers.contrib.openstack.utils import (
|
|
os_release,
|
|
)
|
|
from pg_dir_context import (
|
|
_pg_dir_ips,
|
|
_pg_edge_ips,
|
|
_pg_gateway_ips
|
|
)
|
|
|
|
SOURCES_LIST = '/etc/apt/sources.list'
|
|
LXC_CONF = '/etc/libvirt/lxc.conf'
|
|
TEMPLATES = 'templates/'
|
|
PG_LXC_DATA_PATH = '/var/lib/libvirt/filesystems/plumgrid-data'
|
|
PG_LXC_PATH = '/var/lib/libvirt/filesystems/plumgrid'
|
|
PG_CONF = '%s/conf/pg/plumgrid.conf' % PG_LXC_DATA_PATH
|
|
PG_KA_CONF = '%s/conf/etc/keepalived.conf' % PG_LXC_DATA_PATH
|
|
PG_DEF_CONF = '%s/conf/pg/nginx.conf' % PG_LXC_DATA_PATH
|
|
PG_HN_CONF = '%s/conf/etc/hostname' % PG_LXC_DATA_PATH
|
|
PG_HS_CONF = '%s/conf/etc/hosts' % PG_LXC_DATA_PATH
|
|
PG_IFCS_CONF = '%s/conf/pg/ifcs.conf' % PG_LXC_DATA_PATH
|
|
OPS_CONF = '%s/conf/etc/00-pg.conf' % PG_LXC_DATA_PATH
|
|
AUTH_KEY_PATH = '%s/root/.ssh/authorized_keys' % PG_LXC_DATA_PATH
|
|
TEMP_LICENSE_FILE = '/tmp/license'
|
|
|
|
# Constant values for OpenStack releases as Canonical-Ubuntu
|
|
# doesn't have any specific solution version associated
|
|
OPENSTACK_RELEASE_VERS = {
|
|
'kilo': '10',
|
|
'liberty': '11',
|
|
'mitaka': '12'
|
|
}
|
|
|
|
BASE_RESOURCE_MAP = OrderedDict([
|
|
(PG_KA_CONF, {
|
|
'services': ['plumgrid'],
|
|
'contexts': [pg_dir_context.PGDirContext()],
|
|
}),
|
|
(PG_CONF, {
|
|
'services': ['plumgrid'],
|
|
'contexts': [pg_dir_context.PGDirContext()],
|
|
}),
|
|
(PG_DEF_CONF, {
|
|
'services': ['plumgrid'],
|
|
'contexts': [pg_dir_context.PGDirContext()],
|
|
}),
|
|
(PG_HN_CONF, {
|
|
'services': ['plumgrid'],
|
|
'contexts': [pg_dir_context.PGDirContext()],
|
|
}),
|
|
(PG_HS_CONF, {
|
|
'services': ['plumgrid'],
|
|
'contexts': [pg_dir_context.PGDirContext()],
|
|
}),
|
|
(OPS_CONF, {
|
|
'services': ['plumgrid'],
|
|
'contexts': [pg_dir_context.PGDirContext()],
|
|
}),
|
|
(PG_IFCS_CONF, {
|
|
'services': [],
|
|
'contexts': [pg_dir_context.PGDirContext()],
|
|
}),
|
|
])
|
|
|
|
|
|
def configure_pg_sources():
|
|
'''
|
|
Returns true if install sources is updated in sources.list file
|
|
'''
|
|
try:
|
|
with open(SOURCES_LIST, 'r+') as sources:
|
|
all_lines = sources.readlines()
|
|
sources.seek(0)
|
|
for i in (line for line in all_lines if "plumgrid" not in line):
|
|
sources.write(i)
|
|
sources.truncate()
|
|
sources.close()
|
|
except IOError:
|
|
log('Unable to update /etc/apt/sources.list')
|
|
|
|
|
|
def configure_analyst_opsvm():
|
|
'''
|
|
Configures Anaylyst for OPSVM
|
|
'''
|
|
if not service_running('plumgrid'):
|
|
restart_pg()
|
|
NS_ENTER = ('/opt/local/bin/nsenter -t $(ps ho pid --ppid $(cat '
|
|
'/var/run/libvirt/lxc/plumgrid.pid)) -m -n -u -i -p ')
|
|
sigmund_stop = NS_ENTER + '/usr/bin/service plumgrid-sigmund stop'
|
|
sigmund_status = NS_ENTER \
|
|
+ '/usr/bin/service plumgrid-sigmund status'
|
|
sigmund_autoboot = NS_ENTER \
|
|
+ '/usr/bin/sigmund-configure --ip {0} --start --autoboot' \
|
|
.format(config('opsvm-ip'))
|
|
try:
|
|
status = subprocess.check_output(sigmund_status, shell=True)
|
|
if 'start/running' in status:
|
|
if subprocess.call(sigmund_stop, shell=True):
|
|
log('plumgrid-sigmund couldn\'t be stopped!')
|
|
return
|
|
subprocess.check_call(sigmund_autoboot, shell=True)
|
|
except:
|
|
log('plumgrid-sigmund couldn\'t be started!')
|
|
|
|
|
|
def determine_packages():
|
|
'''
|
|
Returns list of packages required by PLUMgrid director as specified
|
|
in the neutron_plugins dictionary in charmhelpers.
|
|
'''
|
|
pkgs = []
|
|
tag = 'latest'
|
|
for pkg in neutron_plugin_attribute('plumgrid', 'packages', 'neutron'):
|
|
if 'plumgrid' in pkg:
|
|
tag = config('plumgrid-build')
|
|
elif pkg == 'iovisor-dkms':
|
|
tag = config('iovisor-build')
|
|
|
|
if tag == 'latest':
|
|
pkgs.append(pkg)
|
|
else:
|
|
if tag in [i.ver_str for i in apt_cache()[pkg].version_list]:
|
|
pkgs.append('%s=%s' % (pkg, tag))
|
|
else:
|
|
error_msg = \
|
|
"Build version '%s' for package '%s' not available" \
|
|
% (tag, pkg)
|
|
raise ValueError(error_msg)
|
|
return pkgs
|
|
|
|
|
|
def register_configs(release=None):
|
|
'''
|
|
Returns an object of the Openstack Tempating Class which contains the
|
|
the context required for all templates of this charm.
|
|
'''
|
|
release = release or os_release('neutron-common', base='kilo')
|
|
configs = templating.OSConfigRenderer(templates_dir=TEMPLATES,
|
|
openstack_release=release)
|
|
for cfg, rscs in resource_map().iteritems():
|
|
configs.register(cfg, rscs['contexts'])
|
|
return configs
|
|
|
|
|
|
def resource_map():
|
|
'''
|
|
Dynamically generate a map of resources that will be managed for a single
|
|
hook execution.
|
|
'''
|
|
resource_map = deepcopy(BASE_RESOURCE_MAP)
|
|
return resource_map
|
|
|
|
|
|
def restart_map():
|
|
'''
|
|
Constructs a restart map based on charm config settings and relation
|
|
state.
|
|
'''
|
|
return {cfg: rscs['services'] for cfg, rscs in resource_map().iteritems()}
|
|
|
|
|
|
def restart_pg():
|
|
'''
|
|
Stops and Starts PLUMgrid service after flushing iptables.
|
|
'''
|
|
stop_pg()
|
|
service_start('plumgrid')
|
|
time.sleep(3)
|
|
if not service_running('plumgrid'):
|
|
if service_running('libvirt-bin'):
|
|
raise ValueError("plumgrid service couldn't be started")
|
|
else:
|
|
if service_start('libvirt-bin'):
|
|
time.sleep(8)
|
|
if not service_running('plumgrid') \
|
|
and not service_start('plumgrid'):
|
|
raise ValueError("plumgrid service couldn't be started")
|
|
else:
|
|
raise ValueError("libvirt-bin service couldn't be started")
|
|
status_set('active', 'Unit is ready')
|
|
|
|
|
|
def stop_pg():
|
|
'''
|
|
Stops PLUMgrid service.
|
|
'''
|
|
service_stop('plumgrid')
|
|
time.sleep(2)
|
|
|
|
|
|
def load_iovisor():
|
|
'''
|
|
Loads iovisor kernel module.
|
|
'''
|
|
modprobe('iovisor')
|
|
|
|
|
|
def remove_iovisor():
|
|
'''
|
|
Removes iovisor kernel module.
|
|
'''
|
|
_exec_cmd(cmd=['rmmod', 'iovisor'],
|
|
error_msg='Error Removing IOVisor Kernel Module')
|
|
time.sleep(1)
|
|
|
|
|
|
def interface_exists(interface):
|
|
'''
|
|
Checks if interface exists on node.
|
|
'''
|
|
try:
|
|
subprocess.check_call(['ip', 'link', 'show', interface],
|
|
stdout=open(os.devnull, 'w'),
|
|
stderr=subprocess.STDOUT)
|
|
except subprocess.CalledProcessError:
|
|
return False
|
|
return True
|
|
|
|
|
|
def get_mgmt_interface():
|
|
'''
|
|
Returns the managment interface.
|
|
'''
|
|
mgmt_interface = config('mgmt-interface')
|
|
if not mgmt_interface:
|
|
try:
|
|
return get_iface_from_addr(unit_get('private-address'))
|
|
except:
|
|
for bridge_interface in get_bridges():
|
|
if (get_host_ip(unit_get('private-address'))
|
|
in get_iface_addr(bridge_interface)):
|
|
return bridge_interface
|
|
elif interface_exists(mgmt_interface):
|
|
return mgmt_interface
|
|
else:
|
|
log('Provided managment interface %s does not exist'
|
|
% mgmt_interface)
|
|
return get_iface_from_addr(unit_get('private-address'))
|
|
|
|
|
|
def fabric_interface_changed():
|
|
'''
|
|
Returns true if interface for node changed.
|
|
'''
|
|
fabric_interface = get_fabric_interface()
|
|
try:
|
|
with open(PG_IFCS_CONF, 'r') as ifcs:
|
|
for line in ifcs:
|
|
if 'fabric_core' in line:
|
|
if line.split()[0] == fabric_interface:
|
|
return False
|
|
except IOError:
|
|
return True
|
|
return True
|
|
|
|
|
|
def get_fabric_interface():
|
|
'''
|
|
Returns the fabric interface.
|
|
'''
|
|
fabric_interfaces = config('fabric-interfaces')
|
|
if fabric_interfaces == 'MANAGEMENT':
|
|
return get_mgmt_interface()
|
|
else:
|
|
try:
|
|
all_fabric_interfaces = json.loads(fabric_interfaces)
|
|
except ValueError:
|
|
raise ValueError('Invalid json provided for fabric interfaces')
|
|
hostname = get_unit_hostname()
|
|
if hostname in all_fabric_interfaces:
|
|
node_fabric_interface = all_fabric_interfaces[hostname]
|
|
elif 'DEFAULT' in all_fabric_interfaces:
|
|
node_fabric_interface = all_fabric_interfaces['DEFAULT']
|
|
else:
|
|
raise ValueError('No fabric interface provided for node')
|
|
if interface_exists(node_fabric_interface):
|
|
return node_fabric_interface
|
|
else:
|
|
log('Provided fabric interface %s does not exist'
|
|
% node_fabric_interface)
|
|
raise ValueError('Provided fabric interface does not exist')
|
|
return node_fabric_interface
|
|
|
|
|
|
def ensure_mtu():
|
|
'''
|
|
Ensures required MTU of the underlying networking of the node.
|
|
'''
|
|
interface_mtu = config('network-device-mtu')
|
|
fabric_interface = get_fabric_interface()
|
|
if fabric_interface in get_bridges():
|
|
attached_interfaces = get_bridge_nics(fabric_interface)
|
|
for interface in attached_interfaces:
|
|
set_nic_mtu(interface, interface_mtu)
|
|
set_nic_mtu(fabric_interface, interface_mtu)
|
|
|
|
|
|
def _exec_cmd(cmd=None, error_msg='Command exited with ERRORs', fatal=False):
|
|
'''
|
|
Function to execute any bash command on the node.
|
|
'''
|
|
if cmd is None:
|
|
log("No command specified")
|
|
else:
|
|
if fatal:
|
|
subprocess.check_call(cmd)
|
|
else:
|
|
try:
|
|
subprocess.check_call(cmd)
|
|
except subprocess.CalledProcessError:
|
|
log(error_msg)
|
|
|
|
|
|
def add_lcm_key():
|
|
'''
|
|
Adds public key of PLUMgrid-lcm to authorized keys of PLUMgrid Director.
|
|
'''
|
|
key = config('lcm-ssh-key')
|
|
if key == 'null':
|
|
log('lcm key not specified')
|
|
return 0
|
|
file_write_type = 'w+'
|
|
if os.path.isfile(AUTH_KEY_PATH):
|
|
file_write_type = 'a'
|
|
try:
|
|
fr = open(AUTH_KEY_PATH, 'r')
|
|
except IOError:
|
|
log('plumgrid-lxc not installed yet')
|
|
return 0
|
|
for line in fr:
|
|
if key in line:
|
|
log('key already added')
|
|
return 0
|
|
try:
|
|
fa = open(AUTH_KEY_PATH, file_write_type)
|
|
except IOError:
|
|
log('Error opening file to append')
|
|
return 0
|
|
fa.write(key)
|
|
fa.write('\n')
|
|
fa.close()
|
|
return 1
|
|
|
|
|
|
def post_pg_license():
|
|
'''
|
|
Posts PLUMgrid License if it hasnt been posted already.
|
|
'''
|
|
key = config('plumgrid-license-key')
|
|
if key is None:
|
|
log('PLUMgrid License Key not specified')
|
|
return 0
|
|
PG_VIP = config('plumgrid-virtual-ip')
|
|
if not is_ip(PG_VIP):
|
|
raise ValueError('Invalid IP Provided')
|
|
LICENSE_POST_PATH = 'https://%s/0/tenant_manager/license_key' % PG_VIP
|
|
LICENSE_GET_PATH = 'https://%s/0/tenant_manager/licenses' % PG_VIP
|
|
PG_CURL = '%s/opt/pg/scripts/pg_curl.sh' % PG_LXC_PATH
|
|
license = {"key1": {"license": key}}
|
|
licence_post_cmd = [
|
|
PG_CURL,
|
|
'-u',
|
|
'plumgrid:plumgrid',
|
|
LICENSE_POST_PATH,
|
|
'-d',
|
|
json.dumps(license)]
|
|
licence_get_cmd = [PG_CURL, '-u', 'plumgrid:plumgrid', LICENSE_GET_PATH]
|
|
try:
|
|
old_license = subprocess.check_output(licence_get_cmd)
|
|
except subprocess.CalledProcessError:
|
|
log('No response from specified virtual IP')
|
|
return 0
|
|
_exec_cmd(cmd=licence_post_cmd,
|
|
error_msg='Unable to post License', fatal=False)
|
|
new_license = subprocess.check_output(licence_get_cmd)
|
|
if old_license == new_license:
|
|
log('No change in PLUMgrid License')
|
|
return 0
|
|
return 1
|
|
|
|
|
|
def sapi_post_ips():
|
|
"""
|
|
Posts PLUMgrid nodes IPs to solutions api server.
|
|
"""
|
|
pg_edge_ips = _pg_edge_ips()
|
|
pg_dir_ips = _pg_dir_ips()
|
|
pg_gateway_ips = _pg_gateway_ips()
|
|
pg_dir_ips.append(get_host_ip(unit_get('private-address')))
|
|
pg_edge_ips = '"edge_ips"' + ':' \
|
|
+ '"{}"'.format(','.join(str(i) for i in pg_edge_ips))
|
|
pg_dir_ips = '"director_ips"' + ':' \
|
|
+ '"{}"'.format(','.join(str(i) for i in pg_dir_ips))
|
|
pg_gateway_ips = '"gateway_ips"' + ':' \
|
|
+ '"{}"'.format(','.join(str(i) for i in pg_gateway_ips))
|
|
opsvm_ip = '"opsvm_ip"' + ':' + '"{}"'.format(config('opsvm-ip'))
|
|
virtual_ip = '"virtual_ip"' + ':' \
|
|
+ '"{}"'.format(config('plumgrid-virtual-ip'))
|
|
JSON_IPS = ','.join([pg_dir_ips, pg_edge_ips, pg_gateway_ips,
|
|
opsvm_ip, virtual_ip])
|
|
status = (
|
|
'curl -H \'Content-Type: application/json\' -X '
|
|
'PUT -d \'{{{0}}}\' http://{1}' + ':' + '{2}/v1/zones/{3}/allIps'
|
|
).format(JSON_IPS, config('lcm-ip'), config('sapi-port'),
|
|
config('sapi-zone'))
|
|
POST_ZONE_IPs = _exec_cmd_output(
|
|
status,
|
|
'Posting Zone IPs to Solutions API server failed!')
|
|
if POST_ZONE_IPs:
|
|
if 'success' in POST_ZONE_IPs:
|
|
log('Successfully posted Zone IPs to Solutions API server!')
|
|
log(POST_ZONE_IPs)
|
|
|
|
|
|
def _exec_cmd_output(cmd=None, error_msg='Command exited with ERRORs',
|
|
fatal=False):
|
|
'''
|
|
Function to get output from bash command executed on the node.
|
|
'''
|
|
if cmd is None:
|
|
log("No command specified")
|
|
else:
|
|
if fatal:
|
|
return subprocess.check_output(cmd, shell=True)
|
|
else:
|
|
try:
|
|
return subprocess.check_output(cmd, shell=True)
|
|
except subprocess.CalledProcessError:
|
|
log(error_msg)
|
|
return None
|
|
|
|
|
|
def sapi_post_license():
|
|
'''
|
|
Posts PLUMgrid License to solutions api server
|
|
'''
|
|
username = '"user_name":' + '"{}"'.format(config('plumgrid-username'))
|
|
password = '"password":' + '"{}"'.format(config('plumgrid-password'))
|
|
license = '"license":' + '"{}"'.format(config('plumgrid-license-key'))
|
|
JSON_LICENSE = ','.join([username, password, license])
|
|
status = (
|
|
'curl -H \'Content-Type: application/json\' -X '
|
|
'PUT -d \'{{{0}}}\' http://{1}' + ':' + '{2}/v1/zones/{3}/pgLicense'
|
|
).format(JSON_LICENSE, config('lcm-ip'), config('sapi-port'),
|
|
config('sapi-zone'))
|
|
POST_LICENSE = _exec_cmd_output(
|
|
status,
|
|
'Posting PLUMgrid License to Solutions API server failed!')
|
|
if POST_LICENSE:
|
|
if 'success' in POST_LICENSE:
|
|
log('Successfully posted license file for zone "{}"!'
|
|
.format(config('sapi-zone')))
|
|
log(POST_LICENSE)
|
|
|
|
|
|
def sapi_post_zone_info():
|
|
'''
|
|
Posts zone information to solutions api server
|
|
'''
|
|
sol_name = '"solution_name":"Ubuntu OpenStack"'
|
|
release = config('openstack-release')
|
|
for key, value in OPENSTACK_RELEASE_VERS.iteritems():
|
|
if release == value:
|
|
sol_version = value
|
|
else:
|
|
sol_version = 10
|
|
sol_version = '"solution_version":"{}"'.format(sol_version)
|
|
pg_ons_version = _exec_cmd_output(
|
|
'dpkg -l | grep plumgrid | awk \'{print $3}\' | '
|
|
'sed \'s/-/./\' | cut -f1 -d"-"',
|
|
'Unable to obtain PG ONS version'
|
|
).replace('\n', '')
|
|
pg_ons_version = \
|
|
'"pg_ons_version":"{}"'.format(pg_ons_version)
|
|
hypervisor = '"hypervisor":"Ubuntu"'
|
|
hypervisor_version = \
|
|
_exec_cmd_output('lsb_release -r | awk \'{print $2}\'',
|
|
'Unable to obtain solution version'
|
|
).replace('\n', '')
|
|
hypervisor_version = '"hypervisor_version":"{}"' \
|
|
.format(hypervisor_version)
|
|
kernel_version = _exec_cmd_output(
|
|
'uname -r',
|
|
'Unable to obtain kernal version').replace('\n', '')
|
|
kernel_version = \
|
|
'"kernel_version":"{}"'.format(kernel_version)
|
|
cloudapex_path = '/var/lib/libvirt/filesystems/plumgrid/' \
|
|
'opt/pg/web/cloudApex/modules/appCloudApex' \
|
|
'/appCloudApex.js'
|
|
if os.path.isfile(cloudapex_path):
|
|
pg_cloudapex_version = 'cat ' \
|
|
+ '{}'.format(cloudapex_path) \
|
|
+ ' | grep -i appversion | awk \'{print $2}\''
|
|
pg_cloudapex_version = \
|
|
_exec_cmd_output(pg_cloudapex_version,
|
|
'Unable to retrieve CloudApex version'
|
|
).replace('\n', '')
|
|
else:
|
|
log('CloudApex not installed!')
|
|
pg_cloudapex_version = ''
|
|
pg_cloudapex_version = \
|
|
'"pg_cloudapex_version":"{}"'.format(pg_cloudapex_version)
|
|
JSON_ZONE_INFO = ','.join([
|
|
sol_name,
|
|
sol_version,
|
|
pg_ons_version,
|
|
hypervisor,
|
|
hypervisor_version,
|
|
kernel_version,
|
|
pg_cloudapex_version,
|
|
])
|
|
status = (
|
|
'curl -H \'Content-Type: application/json\' -X '
|
|
'PUT -d \'{{{0}}}\' http://{1}:{2}/v1/zones/{3}/zoneinfo'
|
|
).format(JSON_ZONE_INFO, config('lcm-ip'), config('sapi-port'),
|
|
config('sapi-zone'))
|
|
POST_ZONE_INFO = _exec_cmd_output(
|
|
status,
|
|
'Posting Zone Information to Solutions API server failed!')
|
|
if POST_ZONE_INFO:
|
|
if 'success' in POST_ZONE_INFO:
|
|
log('Successfully posted Zone information to Solutions API'
|
|
' server!')
|
|
log(POST_ZONE_INFO)
|
|
|
|
|
|
def load_iptables():
|
|
'''
|
|
Loads iptables rules to allow all PLUMgrid communication.
|
|
'''
|
|
network = get_cidr_from_iface(get_mgmt_interface())
|
|
if network:
|
|
_exec_cmd(['sudo', 'iptables', '-A', 'INPUT', '-p', 'tcp',
|
|
'-j', 'ACCEPT', '-s', network, '-d',
|
|
network, '-m', 'state', '--state', 'NEW'])
|
|
_exec_cmd(['sudo', 'iptables', '-A', 'INPUT', '-p', 'udp', '-j',
|
|
'ACCEPT', '-s', network, '-d', network,
|
|
'-m', 'state', '--state', 'NEW'])
|
|
_exec_cmd(['sudo', 'iptables', '-I', 'INPUT', '-s', network,
|
|
'-d', '224.0.0.18/32', '-j', 'ACCEPT'])
|
|
_exec_cmd(['sudo', 'iptables', '-I', 'INPUT', '-p', 'vrrp', '-j',
|
|
'ACCEPT'])
|
|
_exec_cmd(['sudo', 'iptables', '-A', 'INPUT', '-p', 'tcp', '-j',
|
|
'ACCEPT', '-d', config('plumgrid-virtual-ip'), '-m',
|
|
'state', '--state', 'NEW'])
|
|
apt_install('iptables-persistent')
|
|
|
|
|
|
def get_cidr_from_iface(interface):
|
|
'''
|
|
Determines Network CIDR from interface.
|
|
'''
|
|
if not interface:
|
|
return None
|
|
apt_install('ohai')
|
|
try:
|
|
os_info = subprocess.check_output(['ohai', '-l', 'fatal'])
|
|
except OSError:
|
|
log('Unable to get operating system information')
|
|
return None
|
|
try:
|
|
os_info_json = json.loads(os_info)
|
|
except ValueError:
|
|
log('Unable to determine network')
|
|
return None
|
|
device = os_info_json['network']['interfaces'].get(interface)
|
|
if device is not None:
|
|
if device.get('routes'):
|
|
routes = device['routes']
|
|
for net in routes:
|
|
if 'scope' in net:
|
|
return net.get('destination')
|
|
else:
|
|
return None
|
|
else:
|
|
return None
|
|
|
|
|
|
def director_cluster_ready():
|
|
dirs_count = len(pg_dir_context._pg_dir_ips())
|
|
return True if dirs_count == 2 else False
|
|
|
|
|
|
def restart_on_change(restart_map):
|
|
"""
|
|
Restart services based on configuration files changing
|
|
"""
|
|
def wrap(f):
|
|
def wrapped_f(*args, **kwargs):
|
|
checksums = {path: path_hash(path) for path in restart_map}
|
|
f(*args, **kwargs)
|
|
for path in restart_map:
|
|
if path_hash(path) != checksums[path]:
|
|
restart_pg()
|
|
break
|
|
return wrapped_f
|
|
return wrap
|