Sync charm-helpers.

This commit is contained in:
Corey Bryant 2014-12-10 20:28:53 +00:00
parent 89d9e54b48
commit dc5b65699a
30 changed files with 1198 additions and 523 deletions

View File

@ -13,9 +13,10 @@ clustering-related helpers.
import subprocess import subprocess
import os import os
from socket import gethostname as get_unit_hostname from socket import gethostname as get_unit_hostname
import six
from charmhelpers.core.hookenv import ( from charmhelpers.core.hookenv import (
log, log,
relation_ids, relation_ids,
@ -77,7 +78,7 @@ def is_crm_leader(resource):
"show", resource "show", resource
] ]
try: try:
status = subprocess.check_output(cmd) status = subprocess.check_output(cmd).decode('UTF-8')
except subprocess.CalledProcessError: except subprocess.CalledProcessError:
return False return False
else: else:
@ -150,34 +151,42 @@ def https():
return False return False
def determine_api_port(public_port): def determine_api_port(public_port, singlenode_mode=False):
''' '''
Determine correct API server listening port based on Determine correct API server listening port based on
existence of HTTPS reverse proxy and/or haproxy. existence of HTTPS reverse proxy and/or haproxy.
public_port: int: standard public port for given service public_port: int: standard public port for given service
singlenode_mode: boolean: Shuffle ports when only a single unit is present
returns: int: the correct listening port for the API service returns: int: the correct listening port for the API service
''' '''
i = 0 i = 0
if len(peer_units()) > 0 or is_clustered(): if singlenode_mode:
i += 1
elif len(peer_units()) > 0 or is_clustered():
i += 1 i += 1
if https(): if https():
i += 1 i += 1
return public_port - (i * 10) return public_port - (i * 10)
def determine_apache_port(public_port): def determine_apache_port(public_port, singlenode_mode=False):
''' '''
Description: Determine correct apache listening port based on public IP + Description: Determine correct apache listening port based on public IP +
state of the cluster. state of the cluster.
public_port: int: standard public port for given service public_port: int: standard public port for given service
singlenode_mode: boolean: Shuffle ports when only a single unit is present
returns: int: the correct listening port for the HAProxy service returns: int: the correct listening port for the HAProxy service
''' '''
i = 0 i = 0
if len(peer_units()) > 0 or is_clustered(): if singlenode_mode:
i += 1
elif len(peer_units()) > 0 or is_clustered():
i += 1 i += 1
return public_port - (i * 10) return public_port - (i * 10)
@ -197,7 +206,7 @@ def get_hacluster_config():
for setting in settings: for setting in settings:
conf[setting] = config_get(setting) conf[setting] = config_get(setting)
missing = [] missing = []
[missing.append(s) for s, v in conf.iteritems() if v is None] [missing.append(s) for s, v in six.iteritems(conf) if v is None]
if missing: if missing:
log('Insufficient config data to configure hacluster.', level=ERROR) log('Insufficient config data to configure hacluster.', level=ERROR)
raise HAIncompleteConfig raise HAIncompleteConfig

View File

@ -1,15 +1,12 @@
import glob import glob
import re import re
import subprocess import subprocess
import sys
from functools import partial from functools import partial
from charmhelpers.core.hookenv import unit_get from charmhelpers.core.hookenv import unit_get
from charmhelpers.fetch import apt_install from charmhelpers.fetch import apt_install
from charmhelpers.core.hookenv import ( from charmhelpers.core.hookenv import (
WARNING,
ERROR,
log log
) )
@ -34,31 +31,28 @@ def _validate_cidr(network):
network) network)
def no_ip_found_error_out(network):
errmsg = ("No IP address found in network: %s" % network)
raise ValueError(errmsg)
def get_address_in_network(network, fallback=None, fatal=False): def get_address_in_network(network, fallback=None, fatal=False):
""" """Get an IPv4 or IPv6 address within the network from the host.
Get an IPv4 or IPv6 address within the network from the host.
:param network (str): CIDR presentation format. For example, :param network (str): CIDR presentation format. For example,
'192.168.1.0/24'. '192.168.1.0/24'.
:param fallback (str): If no address is found, return fallback. :param fallback (str): If no address is found, return fallback.
:param fatal (boolean): If no address is found, fallback is not :param fatal (boolean): If no address is found, fallback is not
set and fatal is True then exit(1). set and fatal is True then exit(1).
""" """
def not_found_error_out():
log("No IP address found in network: %s" % network,
level=ERROR)
sys.exit(1)
if network is None: if network is None:
if fallback is not None: if fallback is not None:
return fallback return fallback
if fatal:
no_ip_found_error_out(network)
else: else:
if fatal: return None
not_found_error_out()
else:
return None
_validate_cidr(network) _validate_cidr(network)
network = netaddr.IPNetwork(network) network = netaddr.IPNetwork(network)
@ -70,6 +64,7 @@ def get_address_in_network(network, fallback=None, fatal=False):
cidr = netaddr.IPNetwork("%s/%s" % (addr, netmask)) cidr = netaddr.IPNetwork("%s/%s" % (addr, netmask))
if cidr in network: if cidr in network:
return str(cidr.ip) return str(cidr.ip)
if network.version == 6 and netifaces.AF_INET6 in addresses: if network.version == 6 and netifaces.AF_INET6 in addresses:
for addr in addresses[netifaces.AF_INET6]: for addr in addresses[netifaces.AF_INET6]:
if not addr['addr'].startswith('fe80'): if not addr['addr'].startswith('fe80'):
@ -82,20 +77,20 @@ def get_address_in_network(network, fallback=None, fatal=False):
return fallback return fallback
if fatal: if fatal:
not_found_error_out() no_ip_found_error_out(network)
return None return None
def is_ipv6(address): def is_ipv6(address):
'''Determine whether provided address is IPv6 or not''' """Determine whether provided address is IPv6 or not."""
try: try:
address = netaddr.IPAddress(address) address = netaddr.IPAddress(address)
except netaddr.AddrFormatError: except netaddr.AddrFormatError:
# probably a hostname - so not an address at all! # probably a hostname - so not an address at all!
return False return False
else:
return address.version == 6 return address.version == 6
def is_address_in_network(network, address): def is_address_in_network(network, address):
@ -113,11 +108,13 @@ def is_address_in_network(network, address):
except (netaddr.core.AddrFormatError, ValueError): except (netaddr.core.AddrFormatError, ValueError):
raise ValueError("Network (%s) is not in CIDR presentation format" % raise ValueError("Network (%s) is not in CIDR presentation format" %
network) network)
try: try:
address = netaddr.IPAddress(address) address = netaddr.IPAddress(address)
except (netaddr.core.AddrFormatError, ValueError): except (netaddr.core.AddrFormatError, ValueError):
raise ValueError("Address (%s) is not in correct presentation format" % raise ValueError("Address (%s) is not in correct presentation format" %
address) address)
if address in network: if address in network:
return True return True
else: else:
@ -147,6 +144,7 @@ def _get_for_address(address, key):
return iface return iface
else: else:
return addresses[netifaces.AF_INET][0][key] return addresses[netifaces.AF_INET][0][key]
if address.version == 6 and netifaces.AF_INET6 in addresses: if address.version == 6 and netifaces.AF_INET6 in addresses:
for addr in addresses[netifaces.AF_INET6]: for addr in addresses[netifaces.AF_INET6]:
if not addr['addr'].startswith('fe80'): if not addr['addr'].startswith('fe80'):
@ -160,41 +158,42 @@ def _get_for_address(address, key):
return str(cidr).split('/')[1] return str(cidr).split('/')[1]
else: else:
return addr[key] return addr[key]
return None return None
get_iface_for_address = partial(_get_for_address, key='iface') get_iface_for_address = partial(_get_for_address, key='iface')
get_netmask_for_address = partial(_get_for_address, key='netmask') get_netmask_for_address = partial(_get_for_address, key='netmask')
def format_ipv6_addr(address): def format_ipv6_addr(address):
""" """If address is IPv6, wrap it in '[]' otherwise return None.
IPv6 needs to be wrapped with [] in url link to parse correctly.
This is required by most configuration files when specifying IPv6
addresses.
""" """
if is_ipv6(address): if is_ipv6(address):
address = "[%s]" % address return "[%s]" % address
else:
log("Not a valid ipv6 address: %s" % address, level=WARNING)
address = None
return address return None
def get_iface_addr(iface='eth0', inet_type='AF_INET', inc_aliases=False, def get_iface_addr(iface='eth0', inet_type='AF_INET', inc_aliases=False,
fatal=True, exc_list=None): fatal=True, exc_list=None):
""" """Return the assigned IP address for a given interface, if any."""
Return the assigned IP address for a given interface, if any, or [].
"""
# Extract nic if passed /dev/ethX # Extract nic if passed /dev/ethX
if '/' in iface: if '/' in iface:
iface = iface.split('/')[-1] iface = iface.split('/')[-1]
if not exc_list: if not exc_list:
exc_list = [] exc_list = []
try: try:
inet_num = getattr(netifaces, inet_type) inet_num = getattr(netifaces, inet_type)
except AttributeError: except AttributeError:
raise Exception('Unknown inet type ' + str(inet_type)) raise Exception("Unknown inet type '%s'" % str(inet_type))
interfaces = netifaces.interfaces() interfaces = netifaces.interfaces()
if inc_aliases: if inc_aliases:
@ -202,15 +201,18 @@ def get_iface_addr(iface='eth0', inet_type='AF_INET', inc_aliases=False,
for _iface in interfaces: for _iface in interfaces:
if iface == _iface or _iface.split(':')[0] == iface: if iface == _iface or _iface.split(':')[0] == iface:
ifaces.append(_iface) ifaces.append(_iface)
if fatal and not ifaces: if fatal and not ifaces:
raise Exception("Invalid interface '%s'" % iface) raise Exception("Invalid interface '%s'" % iface)
ifaces.sort() ifaces.sort()
else: else:
if iface not in interfaces: if iface not in interfaces:
if fatal: if fatal:
raise Exception("%s not found " % (iface)) raise Exception("Interface '%s' not found " % (iface))
else: else:
return [] return []
else: else:
ifaces = [iface] ifaces = [iface]
@ -221,10 +223,13 @@ def get_iface_addr(iface='eth0', inet_type='AF_INET', inc_aliases=False,
for entry in net_info[inet_num]: for entry in net_info[inet_num]:
if 'addr' in entry and entry['addr'] not in exc_list: if 'addr' in entry and entry['addr'] not in exc_list:
addresses.append(entry['addr']) addresses.append(entry['addr'])
if fatal and not addresses: if fatal and not addresses:
raise Exception("Interface '%s' doesn't have any %s addresses." % raise Exception("Interface '%s' doesn't have any %s addresses." %
(iface, inet_type)) (iface, inet_type))
return addresses
return sorted(addresses)
get_ipv4_addr = partial(get_iface_addr, inet_type='AF_INET') get_ipv4_addr = partial(get_iface_addr, inet_type='AF_INET')
@ -241,6 +246,7 @@ def get_iface_from_addr(addr):
raw = re.match(ll_key, _addr) raw = re.match(ll_key, _addr)
if raw: if raw:
_addr = raw.group(1) _addr = raw.group(1)
if _addr == addr: if _addr == addr:
log("Address '%s' is configured on iface '%s'" % log("Address '%s' is configured on iface '%s'" %
(addr, iface)) (addr, iface))
@ -251,8 +257,9 @@ def get_iface_from_addr(addr):
def sniff_iface(f): def sniff_iface(f):
"""If no iface provided, inject net iface inferred from unit private """Ensure decorated function is called with a value for iface.
address.
If no iface provided, inject net iface inferred from unit private address.
""" """
def iface_sniffer(*args, **kwargs): def iface_sniffer(*args, **kwargs):
if not kwargs.get('iface', None): if not kwargs.get('iface', None):
@ -295,7 +302,7 @@ def get_ipv6_addr(iface=None, inc_aliases=False, fatal=True, exc_list=None,
if global_addrs: if global_addrs:
# Make sure any found global addresses are not temporary # Make sure any found global addresses are not temporary
cmd = ['ip', 'addr', 'show', iface] cmd = ['ip', 'addr', 'show', iface]
out = subprocess.check_output(cmd) out = subprocess.check_output(cmd).decode('UTF-8')
if dynamic_only: if dynamic_only:
key = re.compile("inet6 (.+)/[0-9]+ scope global dynamic.*") key = re.compile("inet6 (.+)/[0-9]+ scope global dynamic.*")
else: else:
@ -317,33 +324,28 @@ def get_ipv6_addr(iface=None, inc_aliases=False, fatal=True, exc_list=None,
return addrs return addrs
if fatal: if fatal:
raise Exception("Interface '%s' doesn't have a scope global " raise Exception("Interface '%s' does not have a scope global "
"non-temporary ipv6 address." % iface) "non-temporary ipv6 address." % iface)
return [] return []
def get_bridges(vnic_dir='/sys/devices/virtual/net'): def get_bridges(vnic_dir='/sys/devices/virtual/net'):
""" """Return a list of bridges on the system."""
Return a list of bridges on the system or [] b_regex = "%s/*/bridge" % vnic_dir
""" return [x.replace(vnic_dir, '').split('/')[1] for x in glob.glob(b_regex)]
b_rgex = vnic_dir + '/*/bridge'
return [x.replace(vnic_dir, '').split('/')[1] for x in glob.glob(b_rgex)]
def get_bridge_nics(bridge, vnic_dir='/sys/devices/virtual/net'): def get_bridge_nics(bridge, vnic_dir='/sys/devices/virtual/net'):
""" """Return a list of nics comprising a given bridge on the system."""
Return a list of nics comprising a given bridge on the system or [] brif_regex = "%s/%s/brif/*" % (vnic_dir, bridge)
""" return [x.split('/')[-1] for x in glob.glob(brif_regex)]
brif_rgex = "%s/%s/brif/*" % (vnic_dir, bridge)
return [x.split('/')[-1] for x in glob.glob(brif_rgex)]
def is_bridge_member(nic): def is_bridge_member(nic):
""" """Check if a given nic is a member of a bridge."""
Check if a given nic is a member of a bridge
"""
for bridge in get_bridges(): for bridge in get_bridges():
if nic in get_bridge_nics(bridge): if nic in get_bridge_nics(bridge):
return True return True
return False return False

View File

@ -0,0 +1,189 @@
"""
This module contains helpers to add and remove ufw rules.
Examples:
- open SSH port for subnet 10.0.3.0/24:
>>> from charmhelpers.contrib.network import ufw
>>> ufw.enable()
>>> ufw.grant_access(src='10.0.3.0/24', dst='any', port='22', proto='tcp')
- open service by name as defined in /etc/services:
>>> from charmhelpers.contrib.network import ufw
>>> ufw.enable()
>>> ufw.service('ssh', 'open')
- close service by port number:
>>> from charmhelpers.contrib.network import ufw
>>> ufw.enable()
>>> ufw.service('4949', 'close') # munin
"""
__author__ = "Felipe Reyes <felipe.reyes@canonical.com>"
import re
import os
import subprocess
from charmhelpers.core import hookenv
def is_enabled():
"""
Check if `ufw` is enabled
:returns: True if ufw is enabled
"""
output = subprocess.check_output(['ufw', 'status'],
env={'LANG': 'en_US',
'PATH': os.environ['PATH']})
m = re.findall(r'^Status: active\n', output, re.M)
return len(m) >= 1
def enable():
"""
Enable ufw
:returns: True if ufw is successfully enabled
"""
if is_enabled():
return True
output = subprocess.check_output(['ufw', 'enable'],
env={'LANG': 'en_US',
'PATH': os.environ['PATH']})
m = re.findall('^Firewall is active and enabled on system startup\n',
output, re.M)
hookenv.log(output, level='DEBUG')
if len(m) == 0:
hookenv.log("ufw couldn't be enabled", level='WARN')
return False
else:
hookenv.log("ufw enabled", level='INFO')
return True
def disable():
"""
Disable ufw
:returns: True if ufw is successfully disabled
"""
if not is_enabled():
return True
output = subprocess.check_output(['ufw', 'disable'],
env={'LANG': 'en_US',
'PATH': os.environ['PATH']})
m = re.findall(r'^Firewall stopped and disabled on system startup\n',
output, re.M)
hookenv.log(output, level='DEBUG')
if len(m) == 0:
hookenv.log("ufw couldn't be disabled", level='WARN')
return False
else:
hookenv.log("ufw disabled", level='INFO')
return True
def modify_access(src, dst='any', port=None, proto=None, action='allow'):
"""
Grant access to an address or subnet
:param src: address (e.g. 192.168.1.234) or subnet
(e.g. 192.168.1.0/24).
:param dst: destiny of the connection, if the machine has multiple IPs and
connections to only one of those have to accepted this is the
field has to be set.
:param port: destiny port
:param proto: protocol (tcp or udp)
:param action: `allow` or `delete`
"""
if not is_enabled():
hookenv.log('ufw is disabled, skipping modify_access()', level='WARN')
return
if action == 'delete':
cmd = ['ufw', 'delete', 'allow']
else:
cmd = ['ufw', action]
if src is not None:
cmd += ['from', src]
if dst is not None:
cmd += ['to', dst]
if port is not None:
cmd += ['port', port]
if proto is not None:
cmd += ['proto', proto]
hookenv.log('ufw {}: {}'.format(action, ' '.join(cmd)), level='DEBUG')
p = subprocess.Popen(cmd, stdout=subprocess.PIPE)
(stdout, stderr) = p.communicate()
hookenv.log(stdout, level='INFO')
if p.returncode != 0:
hookenv.log(stderr, level='ERROR')
hookenv.log('Error running: {}, exit code: {}'.format(' '.join(cmd),
p.returncode),
level='ERROR')
def grant_access(src, dst='any', port=None, proto=None):
"""
Grant access to an address or subnet
:param src: address (e.g. 192.168.1.234) or subnet
(e.g. 192.168.1.0/24).
:param dst: destiny of the connection, if the machine has multiple IPs and
connections to only one of those have to accepted this is the
field has to be set.
:param port: destiny port
:param proto: protocol (tcp or udp)
"""
return modify_access(src, dst=dst, port=port, proto=proto, action='allow')
def revoke_access(src, dst='any', port=None, proto=None):
"""
Revoke access to an address or subnet
:param src: address (e.g. 192.168.1.234) or subnet
(e.g. 192.168.1.0/24).
:param dst: destiny of the connection, if the machine has multiple IPs and
connections to only one of those have to accepted this is the
field has to be set.
:param port: destiny port
:param proto: protocol (tcp or udp)
"""
return modify_access(src, dst=dst, port=port, proto=proto, action='delete')
def service(name, action):
"""
Open/close access to a service
:param name: could be a service name defined in `/etc/services` or a port
number.
:param action: `open` or `close`
"""
if action == 'open':
subprocess.check_output(['ufw', 'allow', name])
elif action == 'close':
subprocess.check_output(['ufw', 'delete', 'allow', name])
else:
raise Exception(("'{}' not supported, use 'allow' "
"or 'delete'").format(action))

View File

@ -1,3 +1,4 @@
import six
from charmhelpers.contrib.amulet.deployment import ( from charmhelpers.contrib.amulet.deployment import (
AmuletDeployment AmuletDeployment
) )
@ -69,7 +70,7 @@ class OpenStackAmuletDeployment(AmuletDeployment):
def _configure_services(self, configs): def _configure_services(self, configs):
"""Configure all of the services.""" """Configure all of the services."""
for service, config in configs.iteritems(): for service, config in six.iteritems(configs):
self.d.configure(service, config) self.d.configure(service, config)
def _get_openstack_release(self): def _get_openstack_release(self):

View File

@ -7,6 +7,8 @@ import glanceclient.v1.client as glance_client
import keystoneclient.v2_0 as keystone_client import keystoneclient.v2_0 as keystone_client
import novaclient.v1_1.client as nova_client import novaclient.v1_1.client as nova_client
import six
from charmhelpers.contrib.amulet.utils import ( from charmhelpers.contrib.amulet.utils import (
AmuletUtils AmuletUtils
) )
@ -60,7 +62,7 @@ class OpenStackAmuletUtils(AmuletUtils):
expected service catalog endpoints. expected service catalog endpoints.
""" """
self.log.debug('actual: {}'.format(repr(actual))) self.log.debug('actual: {}'.format(repr(actual)))
for k, v in expected.iteritems(): for k, v in six.iteritems(expected):
if k in actual: if k in actual:
ret = self._validate_dict_data(expected[k][0], actual[k][0]) ret = self._validate_dict_data(expected[k][0], actual[k][0])
if ret: if ret:

View File

@ -1,20 +1,18 @@
import json import json
import os import os
import time import time
from base64 import b64decode from base64 import b64decode
from subprocess import check_call
from subprocess import ( import six
check_call
)
from charmhelpers.fetch import ( from charmhelpers.fetch import (
apt_install, apt_install,
filter_installed_packages, filter_installed_packages,
) )
from charmhelpers.core.hookenv import ( from charmhelpers.core.hookenv import (
config, config,
is_relation_made,
local_unit, local_unit,
log, log,
relation_get, relation_get,
@ -23,43 +21,40 @@ from charmhelpers.core.hookenv import (
relation_set, relation_set,
unit_get, unit_get,
unit_private_ip, unit_private_ip,
DEBUG,
INFO,
WARNING,
ERROR, ERROR,
INFO
) )
from charmhelpers.core.host import ( from charmhelpers.core.host import (
mkdir, mkdir,
write_file write_file,
) )
from charmhelpers.contrib.hahelpers.cluster import ( from charmhelpers.contrib.hahelpers.cluster import (
determine_apache_port, determine_apache_port,
determine_api_port, determine_api_port,
https, https,
is_clustered is_clustered,
) )
from charmhelpers.contrib.hahelpers.apache import ( from charmhelpers.contrib.hahelpers.apache import (
get_cert, get_cert,
get_ca_cert, get_ca_cert,
install_ca_cert, install_ca_cert,
) )
from charmhelpers.contrib.openstack.neutron import ( from charmhelpers.contrib.openstack.neutron import (
neutron_plugin_attribute, neutron_plugin_attribute,
) )
from charmhelpers.contrib.network.ip import ( from charmhelpers.contrib.network.ip import (
get_address_in_network, get_address_in_network,
get_ipv6_addr, get_ipv6_addr,
get_netmask_for_address, get_netmask_for_address,
format_ipv6_addr, format_ipv6_addr,
is_address_in_network is_address_in_network,
) )
from charmhelpers.contrib.openstack.utils import get_host_ip from charmhelpers.contrib.openstack.utils import get_host_ip
CA_CERT_PATH = '/usr/local/share/ca-certificates/keystone_juju_ca_cert.crt' CA_CERT_PATH = '/usr/local/share/ca-certificates/keystone_juju_ca_cert.crt'
ADDRESS_TYPES = ['admin', 'internal', 'public']
class OSContextError(Exception): class OSContextError(Exception):
@ -67,7 +62,7 @@ class OSContextError(Exception):
def ensure_packages(packages): def ensure_packages(packages):
'''Install but do not upgrade required plugin packages''' """Install but do not upgrade required plugin packages."""
required = filter_installed_packages(packages) required = filter_installed_packages(packages)
if required: if required:
apt_install(required, fatal=True) apt_install(required, fatal=True)
@ -75,20 +70,27 @@ def ensure_packages(packages):
def context_complete(ctxt): def context_complete(ctxt):
_missing = [] _missing = []
for k, v in ctxt.iteritems(): for k, v in six.iteritems(ctxt):
if v is None or v == '': if v is None or v == '':
_missing.append(k) _missing.append(k)
if _missing: if _missing:
log('Missing required data: %s' % ' '.join(_missing), level='INFO') log('Missing required data: %s' % ' '.join(_missing), level=INFO)
return False return False
return True return True
def config_flags_parser(config_flags): def config_flags_parser(config_flags):
"""Parses config flags string into dict.
The provided config_flags string may be a list of comma-separated values
which themselves may be comma-separated list of values.
"""
if config_flags.find('==') >= 0: if config_flags.find('==') >= 0:
log("config_flags is not in expected format (key=value)", log("config_flags is not in expected format (key=value)", level=ERROR)
level=ERROR)
raise OSContextError raise OSContextError
# strip the following from each value. # strip the following from each value.
post_strippers = ' ,' post_strippers = ' ,'
# we strip any leading/trailing '=' or ' ' from the string then # we strip any leading/trailing '=' or ' ' from the string then
@ -96,7 +98,7 @@ def config_flags_parser(config_flags):
split = config_flags.strip(' =').split('=') split = config_flags.strip(' =').split('=')
limit = len(split) limit = len(split)
flags = {} flags = {}
for i in xrange(0, limit - 1): for i in range(0, limit - 1):
current = split[i] current = split[i]
next = split[i + 1] next = split[i + 1]
vindex = next.rfind(',') vindex = next.rfind(',')
@ -111,17 +113,18 @@ def config_flags_parser(config_flags):
# if this not the first entry, expect an embedded key. # if this not the first entry, expect an embedded key.
index = current.rfind(',') index = current.rfind(',')
if index < 0: if index < 0:
log("invalid config value(s) at index %s" % (i), log("Invalid config value(s) at index %s" % (i), level=ERROR)
level=ERROR)
raise OSContextError raise OSContextError
key = current[index + 1:] key = current[index + 1:]
# Add to collection. # Add to collection.
flags[key.strip(post_strippers)] = value.rstrip(post_strippers) flags[key.strip(post_strippers)] = value.rstrip(post_strippers)
return flags return flags
class OSContextGenerator(object): class OSContextGenerator(object):
"""Base class for all context generators."""
interfaces = [] interfaces = []
def __call__(self): def __call__(self):
@ -133,11 +136,11 @@ class SharedDBContext(OSContextGenerator):
def __init__(self, def __init__(self,
database=None, user=None, relation_prefix=None, ssl_dir=None): database=None, user=None, relation_prefix=None, ssl_dir=None):
''' """Allows inspecting relation for settings prefixed with
Allows inspecting relation for settings prefixed with relation_prefix. relation_prefix. This is useful for parsing access for multiple
This is useful for parsing access for multiple databases returned via databases returned via the shared-db interface (eg, nova_password,
the shared-db interface (eg, nova_password, quantum_password) quantum_password)
''' """
self.relation_prefix = relation_prefix self.relation_prefix = relation_prefix
self.database = database self.database = database
self.user = user self.user = user
@ -147,9 +150,8 @@ class SharedDBContext(OSContextGenerator):
self.database = self.database or config('database') self.database = self.database or config('database')
self.user = self.user or config('database-user') self.user = self.user or config('database-user')
if None in [self.database, self.user]: if None in [self.database, self.user]:
log('Could not generate shared_db context. ' log("Could not generate shared_db context. Missing required charm "
'Missing required charm config options. ' "config options. (database name and user)", level=ERROR)
'(database name and user)')
raise OSContextError raise OSContextError
ctxt = {} ctxt = {}
@ -202,23 +204,24 @@ class PostgresqlDBContext(OSContextGenerator):
def __call__(self): def __call__(self):
self.database = self.database or config('database') self.database = self.database or config('database')
if self.database is None: if self.database is None:
log('Could not generate postgresql_db context. ' log('Could not generate postgresql_db context. Missing required '
'Missing required charm config options. ' 'charm config options. (database name)', level=ERROR)
'(database name)')
raise OSContextError raise OSContextError
ctxt = {}
ctxt = {}
for rid in relation_ids(self.interfaces[0]): for rid in relation_ids(self.interfaces[0]):
for unit in related_units(rid): for unit in related_units(rid):
ctxt = { rel_host = relation_get('host', rid=rid, unit=unit)
'database_host': relation_get('host', rid=rid, unit=unit), rel_user = relation_get('user', rid=rid, unit=unit)
'database': self.database, rel_passwd = relation_get('password', rid=rid, unit=unit)
'database_user': relation_get('user', rid=rid, unit=unit), ctxt = {'database_host': rel_host,
'database_password': relation_get('password', rid=rid, unit=unit), 'database': self.database,
'database_type': 'postgresql', 'database_user': rel_user,
} 'database_password': rel_passwd,
'database_type': 'postgresql'}
if context_complete(ctxt): if context_complete(ctxt):
return ctxt return ctxt
return {} return {}
@ -227,23 +230,29 @@ def db_ssl(rdata, ctxt, ssl_dir):
ca_path = os.path.join(ssl_dir, 'db-client.ca') ca_path = os.path.join(ssl_dir, 'db-client.ca')
with open(ca_path, 'w') as fh: with open(ca_path, 'w') as fh:
fh.write(b64decode(rdata['ssl_ca'])) fh.write(b64decode(rdata['ssl_ca']))
ctxt['database_ssl_ca'] = ca_path ctxt['database_ssl_ca'] = ca_path
elif 'ssl_ca' in rdata: elif 'ssl_ca' in rdata:
log("Charm not setup for ssl support but ssl ca found") log("Charm not setup for ssl support but ssl ca found", level=INFO)
return ctxt return ctxt
if 'ssl_cert' in rdata: if 'ssl_cert' in rdata:
cert_path = os.path.join( cert_path = os.path.join(
ssl_dir, 'db-client.cert') ssl_dir, 'db-client.cert')
if not os.path.exists(cert_path): if not os.path.exists(cert_path):
log("Waiting 1m for ssl client cert validity") log("Waiting 1m for ssl client cert validity", level=INFO)
time.sleep(60) time.sleep(60)
with open(cert_path, 'w') as fh: with open(cert_path, 'w') as fh:
fh.write(b64decode(rdata['ssl_cert'])) fh.write(b64decode(rdata['ssl_cert']))
ctxt['database_ssl_cert'] = cert_path ctxt['database_ssl_cert'] = cert_path
key_path = os.path.join(ssl_dir, 'db-client.key') key_path = os.path.join(ssl_dir, 'db-client.key')
with open(key_path, 'w') as fh: with open(key_path, 'w') as fh:
fh.write(b64decode(rdata['ssl_key'])) fh.write(b64decode(rdata['ssl_key']))
ctxt['database_ssl_key'] = key_path ctxt['database_ssl_key'] = key_path
return ctxt return ctxt
@ -251,9 +260,8 @@ class IdentityServiceContext(OSContextGenerator):
interfaces = ['identity-service'] interfaces = ['identity-service']
def __call__(self): def __call__(self):
log('Generating template context for identity-service') log('Generating template context for identity-service', level=DEBUG)
ctxt = {} ctxt = {}
for rid in relation_ids('identity-service'): for rid in relation_ids('identity-service'):
for unit in related_units(rid): for unit in related_units(rid):
rdata = relation_get(rid=rid, unit=unit) rdata = relation_get(rid=rid, unit=unit)
@ -261,26 +269,24 @@ class IdentityServiceContext(OSContextGenerator):
serv_host = format_ipv6_addr(serv_host) or serv_host serv_host = format_ipv6_addr(serv_host) or serv_host
auth_host = rdata.get('auth_host') auth_host = rdata.get('auth_host')
auth_host = format_ipv6_addr(auth_host) or auth_host auth_host = format_ipv6_addr(auth_host) or auth_host
svc_protocol = rdata.get('service_protocol') or 'http'
ctxt = { auth_protocol = rdata.get('auth_protocol') or 'http'
'service_port': rdata.get('service_port'), ctxt = {'service_port': rdata.get('service_port'),
'service_host': serv_host, 'service_host': serv_host,
'auth_host': auth_host, 'auth_host': auth_host,
'auth_port': rdata.get('auth_port'), 'auth_port': rdata.get('auth_port'),
'admin_tenant_name': rdata.get('service_tenant'), 'admin_tenant_name': rdata.get('service_tenant'),
'admin_user': rdata.get('service_username'), 'admin_user': rdata.get('service_username'),
'admin_password': rdata.get('service_password'), 'admin_password': rdata.get('service_password'),
'service_protocol': 'service_protocol': svc_protocol,
rdata.get('service_protocol') or 'http', 'auth_protocol': auth_protocol}
'auth_protocol':
rdata.get('auth_protocol') or 'http',
}
if context_complete(ctxt): if context_complete(ctxt):
# NOTE(jamespage) this is required for >= icehouse # NOTE(jamespage) this is required for >= icehouse
# so a missing value just indicates keystone needs # so a missing value just indicates keystone needs
# upgrading # upgrading
ctxt['admin_tenant_id'] = rdata.get('service_tenant_id') ctxt['admin_tenant_id'] = rdata.get('service_tenant_id')
return ctxt return ctxt
return {} return {}
@ -293,21 +299,23 @@ class AMQPContext(OSContextGenerator):
self.interfaces = [rel_name] self.interfaces = [rel_name]
def __call__(self): def __call__(self):
log('Generating template context for amqp') log('Generating template context for amqp', level=DEBUG)
conf = config() conf = config()
user_setting = 'rabbit-user'
vhost_setting = 'rabbit-vhost'
if self.relation_prefix: if self.relation_prefix:
user_setting = self.relation_prefix + '-rabbit-user' user_setting = '%s-rabbit-user' % (self.relation_prefix)
vhost_setting = self.relation_prefix + '-rabbit-vhost' vhost_setting = '%s-rabbit-vhost' % (self.relation_prefix)
else:
user_setting = 'rabbit-user'
vhost_setting = 'rabbit-vhost'
try: try:
username = conf[user_setting] username = conf[user_setting]
vhost = conf[vhost_setting] vhost = conf[vhost_setting]
except KeyError as e: except KeyError as e:
log('Could not generate shared_db context. ' log('Could not generate shared_db context. Missing required charm '
'Missing required charm config options: %s.' % e) 'config options: %s.' % e, level=ERROR)
raise OSContextError raise OSContextError
ctxt = {} ctxt = {}
for rid in relation_ids(self.rel_name): for rid in relation_ids(self.rel_name):
ha_vip_only = False ha_vip_only = False
@ -321,6 +329,7 @@ class AMQPContext(OSContextGenerator):
host = relation_get('private-address', rid=rid, unit=unit) host = relation_get('private-address', rid=rid, unit=unit)
host = format_ipv6_addr(host) or host host = format_ipv6_addr(host) or host
ctxt['rabbitmq_host'] = host ctxt['rabbitmq_host'] = host
ctxt.update({ ctxt.update({
'rabbitmq_user': username, 'rabbitmq_user': username,
'rabbitmq_password': relation_get('password', rid=rid, 'rabbitmq_password': relation_get('password', rid=rid,
@ -331,6 +340,7 @@ class AMQPContext(OSContextGenerator):
ssl_port = relation_get('ssl_port', rid=rid, unit=unit) ssl_port = relation_get('ssl_port', rid=rid, unit=unit)
if ssl_port: if ssl_port:
ctxt['rabbit_ssl_port'] = ssl_port ctxt['rabbit_ssl_port'] = ssl_port
ssl_ca = relation_get('ssl_ca', rid=rid, unit=unit) ssl_ca = relation_get('ssl_ca', rid=rid, unit=unit)
if ssl_ca: if ssl_ca:
ctxt['rabbit_ssl_ca'] = ssl_ca ctxt['rabbit_ssl_ca'] = ssl_ca
@ -344,41 +354,45 @@ class AMQPContext(OSContextGenerator):
if context_complete(ctxt): if context_complete(ctxt):
if 'rabbit_ssl_ca' in ctxt: if 'rabbit_ssl_ca' in ctxt:
if not self.ssl_dir: if not self.ssl_dir:
log(("Charm not setup for ssl support " log("Charm not setup for ssl support but ssl ca "
"but ssl ca found")) "found", level=INFO)
break break
ca_path = os.path.join( ca_path = os.path.join(
self.ssl_dir, 'rabbit-client-ca.pem') self.ssl_dir, 'rabbit-client-ca.pem')
with open(ca_path, 'w') as fh: with open(ca_path, 'w') as fh:
fh.write(b64decode(ctxt['rabbit_ssl_ca'])) fh.write(b64decode(ctxt['rabbit_ssl_ca']))
ctxt['rabbit_ssl_ca'] = ca_path ctxt['rabbit_ssl_ca'] = ca_path
# Sufficient information found = break out! # Sufficient information found = break out!
break break
# Used for active/active rabbitmq >= grizzly # Used for active/active rabbitmq >= grizzly
if ('clustered' not in ctxt or ha_vip_only) \ if (('clustered' not in ctxt or ha_vip_only) and
and len(related_units(rid)) > 1: len(related_units(rid)) > 1):
rabbitmq_hosts = [] rabbitmq_hosts = []
for unit in related_units(rid): for unit in related_units(rid):
host = relation_get('private-address', rid=rid, unit=unit) host = relation_get('private-address', rid=rid, unit=unit)
host = format_ipv6_addr(host) or host host = format_ipv6_addr(host) or host
rabbitmq_hosts.append(host) rabbitmq_hosts.append(host)
ctxt['rabbitmq_hosts'] = ','.join(rabbitmq_hosts)
ctxt['rabbitmq_hosts'] = ','.join(sorted(rabbitmq_hosts))
if not context_complete(ctxt): if not context_complete(ctxt):
return {} return {}
else:
return ctxt return ctxt
class CephContext(OSContextGenerator): class CephContext(OSContextGenerator):
"""Generates context for /etc/ceph/ceph.conf templates."""
interfaces = ['ceph'] interfaces = ['ceph']
def __call__(self): def __call__(self):
'''This generates context for /etc/ceph/ceph.conf templates'''
if not relation_ids('ceph'): if not relation_ids('ceph'):
return {} return {}
log('Generating template context for ceph') log('Generating template context for ceph', level=DEBUG)
mon_hosts = [] mon_hosts = []
auth = None auth = None
key = None key = None
@ -387,18 +401,18 @@ class CephContext(OSContextGenerator):
for unit in related_units(rid): for unit in related_units(rid):
auth = relation_get('auth', rid=rid, unit=unit) auth = relation_get('auth', rid=rid, unit=unit)
key = relation_get('key', rid=rid, unit=unit) key = relation_get('key', rid=rid, unit=unit)
ceph_addr = \ ceph_pub_addr = relation_get('ceph-public-address', rid=rid,
relation_get('ceph-public-address', rid=rid, unit=unit) or \ unit=unit)
relation_get('private-address', rid=rid, unit=unit) unit_priv_addr = relation_get('private-address', rid=rid,
unit=unit)
ceph_addr = ceph_pub_addr or unit_priv_addr
ceph_addr = format_ipv6_addr(ceph_addr) or ceph_addr ceph_addr = format_ipv6_addr(ceph_addr) or ceph_addr
mon_hosts.append(ceph_addr) mon_hosts.append(ceph_addr)
ctxt = { ctxt = {'mon_hosts': ' '.join(sorted(mon_hosts)),
'mon_hosts': ' '.join(mon_hosts), 'auth': auth,
'auth': auth, 'key': key,
'key': key, 'use_syslog': use_syslog}
'use_syslog': use_syslog
}
if not os.path.isdir('/etc/ceph'): if not os.path.isdir('/etc/ceph'):
os.mkdir('/etc/ceph') os.mkdir('/etc/ceph')
@ -407,79 +421,68 @@ class CephContext(OSContextGenerator):
return {} return {}
ensure_packages(['ceph-common']) ensure_packages(['ceph-common'])
return ctxt return ctxt
ADDRESS_TYPES = ['admin', 'internal', 'public']
class HAProxyContext(OSContextGenerator): class HAProxyContext(OSContextGenerator):
"""Provides half a context for the haproxy template, which describes
all peers to be included in the cluster. Each charm needs to include
its own context generator that describes the port mapping.
"""
interfaces = ['cluster'] interfaces = ['cluster']
def __call__(self): def __init__(self, singlenode_mode=False):
''' self.singlenode_mode = singlenode_mode
Builds half a context for the haproxy template, which describes
all peers to be included in the cluster. Each charm needs to include
its own context generator that describes the port mapping.
'''
if not relation_ids('cluster'):
return {}
l_unit = local_unit().replace('/', '-') def __call__(self):
if not relation_ids('cluster') and not self.singlenode_mode:
return {}
if config('prefer-ipv6'): if config('prefer-ipv6'):
addr = get_ipv6_addr(exc_list=[config('vip')])[0] addr = get_ipv6_addr(exc_list=[config('vip')])[0]
else: else:
addr = get_host_ip(unit_get('private-address')) addr = get_host_ip(unit_get('private-address'))
l_unit = local_unit().replace('/', '-')
cluster_hosts = {} cluster_hosts = {}
# NOTE(jamespage): build out map of configured network endpoints # NOTE(jamespage): build out map of configured network endpoints
# and associated backends # and associated backends
for addr_type in ADDRESS_TYPES: for addr_type in ADDRESS_TYPES:
laddr = get_address_in_network( cfg_opt = 'os-{}-network'.format(addr_type)
config('os-{}-network'.format(addr_type))) laddr = get_address_in_network(config(cfg_opt))
if laddr: if laddr:
cluster_hosts[laddr] = {} netmask = get_netmask_for_address(laddr)
cluster_hosts[laddr]['network'] = "{}/{}".format( cluster_hosts[laddr] = {'network': "{}/{}".format(laddr,
laddr, netmask),
get_netmask_for_address(laddr) 'backends': {l_unit: laddr}}
)
cluster_hosts[laddr]['backends'] = {}
cluster_hosts[laddr]['backends'][l_unit] = laddr
for rid in relation_ids('cluster'): for rid in relation_ids('cluster'):
for unit in related_units(rid): for unit in related_units(rid):
_unit = unit.replace('/', '-')
_laddr = relation_get('{}-address'.format(addr_type), _laddr = relation_get('{}-address'.format(addr_type),
rid=rid, unit=unit) rid=rid, unit=unit)
if _laddr: if _laddr:
_unit = unit.replace('/', '-')
cluster_hosts[laddr]['backends'][_unit] = _laddr cluster_hosts[laddr]['backends'][_unit] = _laddr
# NOTE(jamespage) no split configurations found, just use # NOTE(jamespage) no split configurations found, just use
# private addresses # private addresses
if not cluster_hosts: if not cluster_hosts:
cluster_hosts[addr] = {} netmask = get_netmask_for_address(addr)
cluster_hosts[addr]['network'] = "{}/{}".format( cluster_hosts[addr] = {'network': "{}/{}".format(addr, netmask),
addr, 'backends': {l_unit: addr}}
get_netmask_for_address(addr)
)
cluster_hosts[addr]['backends'] = {}
cluster_hosts[addr]['backends'][l_unit] = addr
for rid in relation_ids('cluster'): for rid in relation_ids('cluster'):
for unit in related_units(rid): for unit in related_units(rid):
_unit = unit.replace('/', '-')
_laddr = relation_get('private-address', _laddr = relation_get('private-address',
rid=rid, unit=unit) rid=rid, unit=unit)
if _laddr: if _laddr:
_unit = unit.replace('/', '-')
cluster_hosts[addr]['backends'][_unit] = _laddr cluster_hosts[addr]['backends'][_unit] = _laddr
ctxt = { ctxt = {'frontends': cluster_hosts}
'frontends': cluster_hosts,
}
if config('haproxy-server-timeout'): if config('haproxy-server-timeout'):
ctxt['haproxy_server_timeout'] = config('haproxy-server-timeout') ctxt['haproxy_server_timeout'] = config('haproxy-server-timeout')
if config('haproxy-client-timeout'): if config('haproxy-client-timeout'):
ctxt['haproxy_client_timeout'] = config('haproxy-client-timeout') ctxt['haproxy_client_timeout'] = config('haproxy-client-timeout')
@ -493,13 +496,18 @@ class HAProxyContext(OSContextGenerator):
ctxt['stat_port'] = ':8888' ctxt['stat_port'] = ':8888'
for frontend in cluster_hosts: for frontend in cluster_hosts:
if len(cluster_hosts[frontend]['backends']) > 1: if (len(cluster_hosts[frontend]['backends']) > 1 or
self.singlenode_mode):
# Enable haproxy when we have enough peers. # Enable haproxy when we have enough peers.
log('Ensuring haproxy enabled in /etc/default/haproxy.') log('Ensuring haproxy enabled in /etc/default/haproxy.',
level=DEBUG)
with open('/etc/default/haproxy', 'w') as out: with open('/etc/default/haproxy', 'w') as out:
out.write('ENABLED=1\n') out.write('ENABLED=1\n')
return ctxt return ctxt
log('HAProxy context is incomplete, this unit has no peers.')
log('HAProxy context is incomplete, this unit has no peers.',
level=INFO)
return {} return {}
@ -507,29 +515,28 @@ class ImageServiceContext(OSContextGenerator):
interfaces = ['image-service'] interfaces = ['image-service']
def __call__(self): def __call__(self):
''' """Obtains the glance API server from the image-service relation.
Obtains the glance API server from the image-service relation. Useful Useful in nova and cinder (currently).
in nova and cinder (currently). """
''' log('Generating template context for image-service.', level=DEBUG)
log('Generating template context for image-service.')
rids = relation_ids('image-service') rids = relation_ids('image-service')
if not rids: if not rids:
return {} return {}
for rid in rids: for rid in rids:
for unit in related_units(rid): for unit in related_units(rid):
api_server = relation_get('glance-api-server', api_server = relation_get('glance-api-server',
rid=rid, unit=unit) rid=rid, unit=unit)
if api_server: if api_server:
return {'glance_api_servers': api_server} return {'glance_api_servers': api_server}
log('ImageService context is incomplete. '
'Missing required relation data.') log("ImageService context is incomplete. Missing required relation "
"data.", level=INFO)
return {} return {}
class ApacheSSLContext(OSContextGenerator): class ApacheSSLContext(OSContextGenerator):
"""Generates a context for an apache vhost configuration that configures
"""
Generates a context for an apache vhost configuration that configures
HTTPS reverse proxying for one or many endpoints. Generated context HTTPS reverse proxying for one or many endpoints. Generated context
looks something like:: looks something like::
@ -563,6 +570,7 @@ class ApacheSSLContext(OSContextGenerator):
else: else:
cert_filename = 'cert' cert_filename = 'cert'
key_filename = 'key' key_filename = 'key'
write_file(path=os.path.join(ssl_dir, cert_filename), write_file(path=os.path.join(ssl_dir, cert_filename),
content=b64decode(cert)) content=b64decode(cert))
write_file(path=os.path.join(ssl_dir, key_filename), write_file(path=os.path.join(ssl_dir, key_filename),
@ -574,7 +582,8 @@ class ApacheSSLContext(OSContextGenerator):
install_ca_cert(b64decode(ca_cert)) install_ca_cert(b64decode(ca_cert))
def canonical_names(self): def canonical_names(self):
'''Figure out which canonical names clients will access this service''' """Figure out which canonical names clients will access this service.
"""
cns = [] cns = []
for r_id in relation_ids('identity-service'): for r_id in relation_ids('identity-service'):
for unit in related_units(r_id): for unit in related_units(r_id):
@ -582,55 +591,80 @@ class ApacheSSLContext(OSContextGenerator):
for k in rdata: for k in rdata:
if k.startswith('ssl_key_'): if k.startswith('ssl_key_'):
cns.append(k.lstrip('ssl_key_')) cns.append(k.lstrip('ssl_key_'))
return list(set(cns))
return sorted(list(set(cns)))
def get_network_addresses(self):
"""For each network configured, return corresponding address and vip
(if available).
Returns a list of tuples of the form:
[(address_in_net_a, vip_in_net_a),
(address_in_net_b, vip_in_net_b),
...]
or, if no vip(s) available:
[(address_in_net_a, address_in_net_a),
(address_in_net_b, address_in_net_b),
...]
"""
addresses = []
if config('vip'):
vips = config('vip').split()
else:
vips = []
for net_type in ['os-internal-network', 'os-admin-network',
'os-public-network']:
addr = get_address_in_network(config(net_type),
unit_get('private-address'))
if len(vips) > 1 and is_clustered():
if not config(net_type):
log("Multiple networks configured but net_type "
"is None (%s)." % net_type, level=WARNING)
continue
for vip in vips:
if is_address_in_network(config(net_type), vip):
addresses.append((addr, vip))
break
elif is_clustered() and config('vip'):
addresses.append((addr, config('vip')))
else:
addresses.append((addr, addr))
return sorted(addresses)
def __call__(self): def __call__(self):
if isinstance(self.external_ports, basestring): if isinstance(self.external_ports, six.string_types):
self.external_ports = [self.external_ports] self.external_ports = [self.external_ports]
if (not self.external_ports or not https()):
if not self.external_ports or not https():
return {} return {}
self.configure_ca() self.configure_ca()
self.enable_modules() self.enable_modules()
ctxt = { ctxt = {'namespace': self.service_namespace,
'namespace': self.service_namespace, 'endpoints': [],
'endpoints': [], 'ext_ports': []}
'ext_ports': []
}
for cn in self.canonical_names(): for cn in self.canonical_names():
self.configure_cert(cn) self.configure_cert(cn)
addresses = [] addresses = self.get_network_addresses()
vips = [] for address, endpoint in sorted(set(addresses)):
if config('vip'):
vips = config('vip').split()
for network_type in ['os-internal-network',
'os-admin-network',
'os-public-network']:
address = get_address_in_network(config(network_type),
unit_get('private-address'))
if len(vips) > 0 and is_clustered():
for vip in vips:
if is_address_in_network(config(network_type),
vip):
addresses.append((address, vip))
break
elif is_clustered():
addresses.append((address, config('vip')))
else:
addresses.append((address, address))
for address, endpoint in set(addresses):
for api_port in self.external_ports: for api_port in self.external_ports:
ext_port = determine_apache_port(api_port) ext_port = determine_apache_port(api_port)
int_port = determine_api_port(api_port) int_port = determine_api_port(api_port)
portmap = (address, endpoint, int(ext_port), int(int_port)) portmap = (address, endpoint, int(ext_port), int(int_port))
ctxt['endpoints'].append(portmap) ctxt['endpoints'].append(portmap)
ctxt['ext_ports'].append(int(ext_port)) ctxt['ext_ports'].append(int(ext_port))
ctxt['ext_ports'] = list(set(ctxt['ext_ports']))
ctxt['ext_ports'] = sorted(list(set(ctxt['ext_ports'])))
return ctxt return ctxt
@ -647,21 +681,23 @@ class NeutronContext(OSContextGenerator):
@property @property
def packages(self): def packages(self):
return neutron_plugin_attribute( return neutron_plugin_attribute(self.plugin, 'packages',
self.plugin, 'packages', self.network_manager) self.network_manager)
@property @property
def neutron_security_groups(self): def neutron_security_groups(self):
return None return None
def _ensure_packages(self): def _ensure_packages(self):
[ensure_packages(pkgs) for pkgs in self.packages] for pkgs in self.packages:
ensure_packages(pkgs)
def _save_flag_file(self): def _save_flag_file(self):
if self.network_manager == 'quantum': if self.network_manager == 'quantum':
_file = '/etc/nova/quantum_plugin.conf' _file = '/etc/nova/quantum_plugin.conf'
else: else:
_file = '/etc/nova/neutron_plugin.conf' _file = '/etc/nova/neutron_plugin.conf'
with open(_file, 'wb') as out: with open(_file, 'wb') as out:
out.write(self.plugin + '\n') out.write(self.plugin + '\n')
@ -670,13 +706,11 @@ class NeutronContext(OSContextGenerator):
self.network_manager) self.network_manager)
config = neutron_plugin_attribute(self.plugin, 'config', config = neutron_plugin_attribute(self.plugin, 'config',
self.network_manager) self.network_manager)
ovs_ctxt = { ovs_ctxt = {'core_plugin': driver,
'core_plugin': driver, 'neutron_plugin': 'ovs',
'neutron_plugin': 'ovs', 'neutron_security_groups': self.neutron_security_groups,
'neutron_security_groups': self.neutron_security_groups, 'local_ip': unit_private_ip(),
'local_ip': unit_private_ip(), 'config': config}
'config': config
}
return ovs_ctxt return ovs_ctxt
@ -685,13 +719,11 @@ class NeutronContext(OSContextGenerator):
self.network_manager) self.network_manager)
config = neutron_plugin_attribute(self.plugin, 'config', config = neutron_plugin_attribute(self.plugin, 'config',
self.network_manager) self.network_manager)
nvp_ctxt = { nvp_ctxt = {'core_plugin': driver,
'core_plugin': driver, 'neutron_plugin': 'nvp',
'neutron_plugin': 'nvp', 'neutron_security_groups': self.neutron_security_groups,
'neutron_security_groups': self.neutron_security_groups, 'local_ip': unit_private_ip(),
'local_ip': unit_private_ip(), 'config': config}
'config': config
}
return nvp_ctxt return nvp_ctxt
@ -700,35 +732,50 @@ class NeutronContext(OSContextGenerator):
self.network_manager) self.network_manager)
n1kv_config = neutron_plugin_attribute(self.plugin, 'config', n1kv_config = neutron_plugin_attribute(self.plugin, 'config',
self.network_manager) self.network_manager)
n1kv_ctxt = { n1kv_user_config_flags = config('n1kv-config-flags')
'core_plugin': driver, restrict_policy_profiles = config('n1kv-restrict-policy-profiles')
'neutron_plugin': 'n1kv', n1kv_ctxt = {'core_plugin': driver,
'neutron_security_groups': self.neutron_security_groups, 'neutron_plugin': 'n1kv',
'local_ip': unit_private_ip(), 'neutron_security_groups': self.neutron_security_groups,
'config': n1kv_config, 'local_ip': unit_private_ip(),
'vsm_ip': config('n1kv-vsm-ip'), 'config': n1kv_config,
'vsm_username': config('n1kv-vsm-username'), 'vsm_ip': config('n1kv-vsm-ip'),
'vsm_password': config('n1kv-vsm-password'), 'vsm_username': config('n1kv-vsm-username'),
'restrict_policy_profiles': config( 'vsm_password': config('n1kv-vsm-password'),
'n1kv_restrict_policy_profiles'), 'restrict_policy_profiles': restrict_policy_profiles}
}
if n1kv_user_config_flags:
flags = config_flags_parser(n1kv_user_config_flags)
n1kv_ctxt['user_config_flags'] = flags
return n1kv_ctxt return n1kv_ctxt
def calico_ctxt(self):
driver = neutron_plugin_attribute(self.plugin, 'driver',
self.network_manager)
config = neutron_plugin_attribute(self.plugin, 'config',
self.network_manager)
calico_ctxt = {'core_plugin': driver,
'neutron_plugin': 'Calico',
'neutron_security_groups': self.neutron_security_groups,
'local_ip': unit_private_ip(),
'config': config}
return calico_ctxt
def neutron_ctxt(self): def neutron_ctxt(self):
if https(): if https():
proto = 'https' proto = 'https'
else: else:
proto = 'http' proto = 'http'
if is_clustered(): if is_clustered():
host = config('vip') host = config('vip')
else: else:
host = unit_get('private-address') host = unit_get('private-address')
url = '%s://%s:%s' % (proto, host, '9696')
ctxt = { ctxt = {'network_manager': self.network_manager,
'network_manager': self.network_manager, 'neutron_url': '%s://%s:%s' % (proto, host, '9696')}
'neutron_url': url,
}
return ctxt return ctxt
def __call__(self): def __call__(self):
@ -748,6 +795,8 @@ class NeutronContext(OSContextGenerator):
ctxt.update(self.nvp_ctxt()) ctxt.update(self.nvp_ctxt())
elif self.plugin == 'n1kv': elif self.plugin == 'n1kv':
ctxt.update(self.n1kv_ctxt()) ctxt.update(self.n1kv_ctxt())
elif self.plugin == 'Calico':
ctxt.update(self.calico_ctxt())
alchemy_flags = config('neutron-alchemy-flags') alchemy_flags = config('neutron-alchemy-flags')
if alchemy_flags: if alchemy_flags:
@ -759,23 +808,40 @@ class NeutronContext(OSContextGenerator):
class OSConfigFlagContext(OSContextGenerator): class OSConfigFlagContext(OSContextGenerator):
"""Provides support for user-defined config flags.
""" Users can define a comma-seperated list of key=value pairs
Responsible for adding user-defined config-flags in charm config to a in the charm configuration and apply them at any point in
template context. any file by using a template flag.
Sometimes users might want config flags inserted within a
specific section so this class allows users to specify the
template flag name, allowing for multiple template flags
(sections) within the same context.
NOTE: the value of config-flags may be a comma-separated list of NOTE: the value of config-flags may be a comma-separated list of
key=value pairs and some Openstack config files support key=value pairs and some Openstack config files support
comma-separated lists as values. comma-separated lists as values.
""" """
def __init__(self, charm_flag='config-flags',
template_flag='user_config_flags'):
"""
:param charm_flag: config flags in charm configuration.
:param template_flag: insert point for user-defined flags in template
file.
"""
super(OSConfigFlagContext, self).__init__()
self._charm_flag = charm_flag
self._template_flag = template_flag
def __call__(self): def __call__(self):
config_flags = config('config-flags') config_flags = config(self._charm_flag)
if not config_flags: if not config_flags:
return {} return {}
flags = config_flags_parser(config_flags) return {self._template_flag:
return {'user_config_flags': flags} config_flags_parser(config_flags)}
class SubordinateConfigContext(OSContextGenerator): class SubordinateConfigContext(OSContextGenerator):
@ -819,7 +885,6 @@ class SubordinateConfigContext(OSContextGenerator):
}, },
} }
} }
""" """
def __init__(self, service, config_file, interface): def __init__(self, service, config_file, interface):
@ -849,26 +914,28 @@ class SubordinateConfigContext(OSContextGenerator):
if self.service not in sub_config: if self.service not in sub_config:
log('Found subordinate_config on %s but it contained' log('Found subordinate_config on %s but it contained'
'nothing for %s service' % (rid, self.service)) 'nothing for %s service' % (rid, self.service),
level=INFO)
continue continue
sub_config = sub_config[self.service] sub_config = sub_config[self.service]
if self.config_file not in sub_config: if self.config_file not in sub_config:
log('Found subordinate_config on %s but it contained' log('Found subordinate_config on %s but it contained'
'nothing for %s' % (rid, self.config_file)) 'nothing for %s' % (rid, self.config_file),
level=INFO)
continue continue
sub_config = sub_config[self.config_file] sub_config = sub_config[self.config_file]
for k, v in sub_config.iteritems(): for k, v in six.iteritems(sub_config):
if k == 'sections': if k == 'sections':
for section, config_dict in v.iteritems(): for section, config_dict in six.iteritems(v):
log("adding section '%s'" % (section)) log("adding section '%s'" % (section),
level=DEBUG)
ctxt[k][section] = config_dict ctxt[k][section] = config_dict
else: else:
ctxt[k] = v ctxt[k] = v
log("%d section(s) found" % (len(ctxt['sections'])), level=INFO) log("%d section(s) found" % (len(ctxt['sections'])), level=DEBUG)
return ctxt return ctxt
@ -880,15 +947,14 @@ class LogLevelContext(OSContextGenerator):
False if config('debug') is None else config('debug') False if config('debug') is None else config('debug')
ctxt['verbose'] = \ ctxt['verbose'] = \
False if config('verbose') is None else config('verbose') False if config('verbose') is None else config('verbose')
return ctxt return ctxt
class SyslogContext(OSContextGenerator): class SyslogContext(OSContextGenerator):
def __call__(self): def __call__(self):
ctxt = { ctxt = {'use_syslog': config('use-syslog')}
'use_syslog': config('use-syslog')
}
return ctxt return ctxt
@ -896,13 +962,9 @@ class BindHostContext(OSContextGenerator):
def __call__(self): def __call__(self):
if config('prefer-ipv6'): if config('prefer-ipv6'):
return { return {'bind_host': '::'}
'bind_host': '::'
}
else: else:
return { return {'bind_host': '0.0.0.0'}
'bind_host': '0.0.0.0'
}
class WorkerConfigContext(OSContextGenerator): class WorkerConfigContext(OSContextGenerator):
@ -914,11 +976,42 @@ class WorkerConfigContext(OSContextGenerator):
except ImportError: except ImportError:
apt_install('python-psutil', fatal=True) apt_install('python-psutil', fatal=True)
from psutil import NUM_CPUS from psutil import NUM_CPUS
return NUM_CPUS return NUM_CPUS
def __call__(self): def __call__(self):
multiplier = config('worker-multiplier') or 1 multiplier = config('worker-multiplier') or 0
ctxt = { ctxt = {"workers": self.num_cpus * multiplier}
"workers": self.num_cpus * multiplier return ctxt
}
class ZeroMQContext(OSContextGenerator):
interfaces = ['zeromq-configuration']
def __call__(self):
ctxt = {}
if is_relation_made('zeromq-configuration', 'host'):
for rid in relation_ids('zeromq-configuration'):
for unit in related_units(rid):
ctxt['zmq_nonce'] = relation_get('nonce', unit, rid)
ctxt['zmq_host'] = relation_get('host', unit, rid)
return ctxt
class NotificationDriverContext(OSContextGenerator):
def __init__(self, zmq_relation='zeromq-configuration',
amqp_relation='amqp'):
"""
:param zmq_relation: Name of Zeromq relation to check
"""
self.zmq_relation = zmq_relation
self.amqp_relation = amqp_relation
def __call__(self):
ctxt = {'notifications': 'False'}
if is_relation_made(self.amqp_relation):
ctxt['notifications'] = "True"
return ctxt return ctxt

View File

@ -2,21 +2,19 @@ from charmhelpers.core.hookenv import (
config, config,
unit_get, unit_get,
) )
from charmhelpers.contrib.network.ip import ( from charmhelpers.contrib.network.ip import (
get_address_in_network, get_address_in_network,
is_address_in_network, is_address_in_network,
is_ipv6, is_ipv6,
get_ipv6_addr, get_ipv6_addr,
) )
from charmhelpers.contrib.hahelpers.cluster import is_clustered from charmhelpers.contrib.hahelpers.cluster import is_clustered
PUBLIC = 'public' PUBLIC = 'public'
INTERNAL = 'int' INTERNAL = 'int'
ADMIN = 'admin' ADMIN = 'admin'
_address_map = { ADDRESS_MAP = {
PUBLIC: { PUBLIC: {
'config': 'os-public-network', 'config': 'os-public-network',
'fallback': 'public-address' 'fallback': 'public-address'
@ -33,16 +31,14 @@ _address_map = {
def canonical_url(configs, endpoint_type=PUBLIC): def canonical_url(configs, endpoint_type=PUBLIC):
''' """Returns the correct HTTP URL to this host given the state of HTTPS
Returns the correct HTTP URL to this host given the state of HTTPS
configuration, hacluster and charm configuration. configuration, hacluster and charm configuration.
:configs OSTemplateRenderer: A config tempating object to inspect for :param configs: OSTemplateRenderer config templating object to inspect
a complete https context. for a complete https context.
:endpoint_type str: The endpoint type to resolve. :param endpoint_type: str endpoint type to resolve.
:param returns: str base URL for services on the current service unit.
:returns str: Base URL for services on the current service unit. """
'''
scheme = 'http' scheme = 'http'
if 'https' in configs.complete_contexts(): if 'https' in configs.complete_contexts():
scheme = 'https' scheme = 'https'
@ -53,27 +49,45 @@ def canonical_url(configs, endpoint_type=PUBLIC):
def resolve_address(endpoint_type=PUBLIC): def resolve_address(endpoint_type=PUBLIC):
"""Return unit address depending on net config.
If unit is clustered with vip(s) and has net splits defined, return vip on
correct network. If clustered with no nets defined, return primary vip.
If not clustered, return unit address ensuring address is on configured net
split if one is configured.
:param endpoint_type: Network endpoing type
"""
resolved_address = None resolved_address = None
if is_clustered(): vips = config('vip')
if config(_address_map[endpoint_type]['config']) is None: if vips:
# Assume vip is simple and pass back directly vips = vips.split()
resolved_address = config('vip')
net_type = ADDRESS_MAP[endpoint_type]['config']
net_addr = config(net_type)
net_fallback = ADDRESS_MAP[endpoint_type]['fallback']
clustered = is_clustered()
if clustered:
if not net_addr:
# If no net-splits defined, we expect a single vip
resolved_address = vips[0]
else: else:
for vip in config('vip').split(): for vip in vips:
if is_address_in_network( if is_address_in_network(net_addr, vip):
config(_address_map[endpoint_type]['config']),
vip):
resolved_address = vip resolved_address = vip
break
else: else:
if config('prefer-ipv6'): if config('prefer-ipv6'):
fallback_addr = get_ipv6_addr(exc_list=[config('vip')])[0] fallback_addr = get_ipv6_addr(exc_list=vips)[0]
else: else:
fallback_addr = unit_get(_address_map[endpoint_type]['fallback']) fallback_addr = unit_get(net_fallback)
resolved_address = get_address_in_network(
config(_address_map[endpoint_type]['config']), fallback_addr) resolved_address = get_address_in_network(net_addr, fallback_addr)
if resolved_address is None: if resolved_address is None:
raise ValueError('Unable to resolve a suitable IP address' raise ValueError("Unable to resolve a suitable IP address based on "
' based on charm state and configuration') "charm state and configuration. (net_type=%s, "
else: "clustered=%s)" % (net_type, clustered))
return resolved_address
return resolved_address

View File

@ -14,7 +14,7 @@ from charmhelpers.contrib.openstack.utils import os_release
def headers_package(): def headers_package():
"""Ensures correct linux-headers for running kernel are installed, """Ensures correct linux-headers for running kernel are installed,
for building DKMS package""" for building DKMS package"""
kver = check_output(['uname', '-r']).strip() kver = check_output(['uname', '-r']).decode('UTF-8').strip()
return 'linux-headers-%s' % kver return 'linux-headers-%s' % kver
QUANTUM_CONF_DIR = '/etc/quantum' QUANTUM_CONF_DIR = '/etc/quantum'
@ -22,7 +22,7 @@ QUANTUM_CONF_DIR = '/etc/quantum'
def kernel_version(): def kernel_version():
""" Retrieve the current major kernel version as a tuple e.g. (3, 13) """ """ Retrieve the current major kernel version as a tuple e.g. (3, 13) """
kver = check_output(['uname', '-r']).strip() kver = check_output(['uname', '-r']).decode('UTF-8').strip()
kver = kver.split('.') kver = kver.split('.')
return (int(kver[0]), int(kver[1])) return (int(kver[0]), int(kver[1]))
@ -138,10 +138,25 @@ def neutron_plugins():
relation_prefix='neutron', relation_prefix='neutron',
ssl_dir=NEUTRON_CONF_DIR)], ssl_dir=NEUTRON_CONF_DIR)],
'services': [], 'services': [],
'packages': [['neutron-plugin-cisco']], 'packages': [[headers_package()] + determine_dkms_package(),
['neutron-plugin-cisco']],
'server_packages': ['neutron-server', 'server_packages': ['neutron-server',
'neutron-plugin-cisco'], 'neutron-plugin-cisco'],
'server_services': ['neutron-server'] 'server_services': ['neutron-server']
},
'Calico': {
'config': '/etc/neutron/plugins/ml2/ml2_conf.ini',
'driver': 'neutron.plugins.ml2.plugin.Ml2Plugin',
'contexts': [
context.SharedDBContext(user=config('neutron-database-user'),
database=config('neutron-database'),
relation_prefix='neutron',
ssl_dir=NEUTRON_CONF_DIR)],
'services': ['calico-compute', 'bird', 'neutron-dhcp-agent'],
'packages': [[headers_package()] + determine_dkms_package(),
['calico-compute', 'bird', 'neutron-dhcp-agent']],
'server_packages': ['neutron-server', 'calico-control'],
'server_services': ['neutron-server']
} }
} }
if release >= 'icehouse': if release >= 'icehouse':
@ -162,7 +177,8 @@ def neutron_plugin_attribute(plugin, attr, net_manager=None):
elif manager == 'neutron': elif manager == 'neutron':
plugins = neutron_plugins() plugins = neutron_plugins()
else: else:
log('Error: Network manager does not support plugins.') log("Network manager '%s' does not support plugins." % (manager),
level=ERROR)
raise Exception raise Exception
try: try:

View File

@ -1,13 +1,13 @@
import os import os
from charmhelpers.fetch import apt_install import six
from charmhelpers.fetch import apt_install
from charmhelpers.core.hookenv import ( from charmhelpers.core.hookenv import (
log, log,
ERROR, ERROR,
INFO INFO
) )
from charmhelpers.contrib.openstack.utils import OPENSTACK_CODENAMES from charmhelpers.contrib.openstack.utils import OPENSTACK_CODENAMES
try: try:
@ -43,7 +43,7 @@ def get_loader(templates_dir, os_release):
order by OpenStack release. order by OpenStack release.
""" """
tmpl_dirs = [(rel, os.path.join(templates_dir, rel)) tmpl_dirs = [(rel, os.path.join(templates_dir, rel))
for rel in OPENSTACK_CODENAMES.itervalues()] for rel in six.itervalues(OPENSTACK_CODENAMES)]
if not os.path.isdir(templates_dir): if not os.path.isdir(templates_dir):
log('Templates directory not found @ %s.' % templates_dir, log('Templates directory not found @ %s.' % templates_dir,
@ -258,7 +258,7 @@ class OSConfigRenderer(object):
""" """
Write out all registered config files. Write out all registered config files.
""" """
[self.write(k) for k in self.templates.iterkeys()] [self.write(k) for k in six.iterkeys(self.templates)]
def set_release(self, openstack_release): def set_release(self, openstack_release):
""" """
@ -275,5 +275,5 @@ class OSConfigRenderer(object):
''' '''
interfaces = [] interfaces = []
[interfaces.extend(i.complete_contexts()) [interfaces.extend(i.complete_contexts())
for i in self.templates.itervalues()] for i in six.itervalues(self.templates)]
return interfaces return interfaces

View File

@ -2,6 +2,7 @@
# Common python helper functions used for OpenStack charms. # Common python helper functions used for OpenStack charms.
from collections import OrderedDict from collections import OrderedDict
from functools import wraps
import subprocess import subprocess
import json import json
@ -9,11 +10,13 @@ import os
import socket import socket
import sys import sys
import six
import yaml
from charmhelpers.core.hookenv import ( from charmhelpers.core.hookenv import (
config, config,
log as juju_log, log as juju_log,
charm_dir, charm_dir,
ERROR,
INFO, INFO,
relation_ids, relation_ids,
relation_set relation_set
@ -30,7 +33,8 @@ from charmhelpers.contrib.network.ip import (
) )
from charmhelpers.core.host import lsb_release, mounts, umount from charmhelpers.core.host import lsb_release, mounts, umount
from charmhelpers.fetch import apt_install, apt_cache from charmhelpers.fetch import apt_install, apt_cache, install_remote
from charmhelpers.contrib.python.packages import pip_install
from charmhelpers.contrib.storage.linux.utils import is_block_device, zap_disk from charmhelpers.contrib.storage.linux.utils import is_block_device, zap_disk
from charmhelpers.contrib.storage.linux.loopback import ensure_loopback_device from charmhelpers.contrib.storage.linux.loopback import ensure_loopback_device
@ -112,7 +116,7 @@ def get_os_codename_install_source(src):
# Best guess match based on deb string provided # Best guess match based on deb string provided
if src.startswith('deb') or src.startswith('ppa'): if src.startswith('deb') or src.startswith('ppa'):
for k, v in OPENSTACK_CODENAMES.iteritems(): for k, v in six.iteritems(OPENSTACK_CODENAMES):
if v in src: if v in src:
return v return v
@ -133,7 +137,7 @@ def get_os_codename_version(vers):
def get_os_version_codename(codename): def get_os_version_codename(codename):
'''Determine OpenStack version number from codename.''' '''Determine OpenStack version number from codename.'''
for k, v in OPENSTACK_CODENAMES.iteritems(): for k, v in six.iteritems(OPENSTACK_CODENAMES):
if v == codename: if v == codename:
return k return k
e = 'Could not derive OpenStack version for '\ e = 'Could not derive OpenStack version for '\
@ -193,7 +197,7 @@ def get_os_version_package(pkg, fatal=True):
else: else:
vers_map = OPENSTACK_CODENAMES vers_map = OPENSTACK_CODENAMES
for version, cname in vers_map.iteritems(): for version, cname in six.iteritems(vers_map):
if cname == codename: if cname == codename:
return version return version
# e = "Could not determine OpenStack version for package: %s" % pkg # e = "Could not determine OpenStack version for package: %s" % pkg
@ -317,7 +321,7 @@ def save_script_rc(script_path="scripts/scriptrc", **env_vars):
rc_script.write( rc_script.write(
"#!/bin/bash\n") "#!/bin/bash\n")
[rc_script.write('export %s=%s\n' % (u, p)) [rc_script.write('export %s=%s\n' % (u, p))
for u, p in env_vars.iteritems() if u != "script_path"] for u, p in six.iteritems(env_vars) if u != "script_path"]
def openstack_upgrade_available(package): def openstack_upgrade_available(package):
@ -350,8 +354,8 @@ def ensure_block_device(block_device):
''' '''
_none = ['None', 'none', None] _none = ['None', 'none', None]
if (block_device in _none): if (block_device in _none):
error_out('prepare_storage(): Missing required input: ' error_out('prepare_storage(): Missing required input: block_device=%s.'
'block_device=%s.' % block_device, level=ERROR) % block_device)
if block_device.startswith('/dev/'): if block_device.startswith('/dev/'):
bdev = block_device bdev = block_device
@ -367,8 +371,7 @@ def ensure_block_device(block_device):
bdev = '/dev/%s' % block_device bdev = '/dev/%s' % block_device
if not is_block_device(bdev): if not is_block_device(bdev):
error_out('Failed to locate valid block device at %s' % bdev, error_out('Failed to locate valid block device at %s' % bdev)
level=ERROR)
return bdev return bdev
@ -417,7 +420,7 @@ def ns_query(address):
if isinstance(address, dns.name.Name): if isinstance(address, dns.name.Name):
rtype = 'PTR' rtype = 'PTR'
elif isinstance(address, basestring): elif isinstance(address, six.string_types):
rtype = 'A' rtype = 'A'
else: else:
return None return None
@ -468,6 +471,14 @@ def get_hostname(address, fqdn=True):
return result.split('.')[0] return result.split('.')[0]
def get_matchmaker_map(mm_file='/etc/oslo/matchmaker_ring.json'):
mm_map = {}
if os.path.isfile(mm_file):
with open(mm_file, 'r') as f:
mm_map = json.load(f)
return mm_map
def sync_db_with_multi_ipv6_addresses(database, database_user, def sync_db_with_multi_ipv6_addresses(database, database_user,
relation_prefix=None): relation_prefix=None):
hosts = get_ipv6_addr(dynamic_only=False) hosts = get_ipv6_addr(dynamic_only=False)
@ -477,10 +488,132 @@ def sync_db_with_multi_ipv6_addresses(database, database_user,
'hostname': json.dumps(hosts)} 'hostname': json.dumps(hosts)}
if relation_prefix: if relation_prefix:
keys = kwargs.keys() for key in list(kwargs.keys()):
for key in keys:
kwargs["%s_%s" % (relation_prefix, key)] = kwargs[key] kwargs["%s_%s" % (relation_prefix, key)] = kwargs[key]
del kwargs[key] del kwargs[key]
for rid in relation_ids('shared-db'): for rid in relation_ids('shared-db'):
relation_set(relation_id=rid, **kwargs) relation_set(relation_id=rid, **kwargs)
def os_requires_version(ostack_release, pkg):
"""
Decorator for hook to specify minimum supported release
"""
def wrap(f):
@wraps(f)
def wrapped_f(*args):
if os_release(pkg) < ostack_release:
raise Exception("This hook is not supported on releases"
" before %s" % ostack_release)
f(*args)
return wrapped_f
return wrap
def git_install_requested():
"""Returns true if openstack-origin-git is specified."""
return config('openstack-origin-git') != "None"
requirements_dir = None
def git_clone_and_install(file_name, core_project):
"""Clone/install all OpenStack repos specified in yaml config file."""
global requirements_dir
if file_name == "None":
return
yaml_file = os.path.join(charm_dir(), file_name)
# clone/install the requirements project first
installed = _git_clone_and_install_subset(yaml_file,
whitelist=['requirements'])
if 'requirements' not in installed:
error_out('requirements git repository must be specified')
# clone/install all other projects except requirements and the core project
blacklist = ['requirements', core_project]
_git_clone_and_install_subset(yaml_file, blacklist=blacklist,
update_requirements=True)
# clone/install the core project
whitelist = [core_project]
installed = _git_clone_and_install_subset(yaml_file, whitelist=whitelist,
update_requirements=True)
if core_project not in installed:
error_out('{} git repository must be specified'.format(core_project))
def _git_clone_and_install_subset(yaml_file, whitelist=[], blacklist=[],
update_requirements=False):
"""Clone/install subset of OpenStack repos specified in yaml config file."""
global requirements_dir
installed = []
with open(yaml_file, 'r') as fd:
projects = yaml.load(fd)
for proj, val in projects.items():
# The project subset is chosen based on the following 3 rules:
# 1) If project is in blacklist, we don't clone/install it, period.
# 2) If whitelist is empty, we clone/install everything else.
# 3) If whitelist is not empty, we clone/install everything in the
# whitelist.
if proj in blacklist:
continue
if whitelist and proj not in whitelist:
continue
repo = val['repository']
branch = val['branch']
repo_dir = _git_clone_and_install_single(repo, branch,
update_requirements)
if proj == 'requirements':
requirements_dir = repo_dir
installed.append(proj)
return installed
def _git_clone_and_install_single(repo, branch, update_requirements=False):
"""Clone and install a single git repository."""
dest_parent_dir = "/mnt/openstack-git/"
dest_dir = os.path.join(dest_parent_dir, os.path.basename(repo))
if not os.path.exists(dest_parent_dir):
juju_log('Host dir not mounted at {}. '
'Creating directory there instead.'.format(dest_parent_dir))
os.mkdir(dest_parent_dir)
if not os.path.exists(dest_dir):
juju_log('Cloning git repo: {}, branch: {}'.format(repo, branch))
repo_dir = install_remote(repo, dest=dest_parent_dir, branch=branch)
else:
repo_dir = dest_dir
if update_requirements:
if not requirements_dir:
error_out('requirements repo must be cloned before '
'updating from global requirements.')
_git_update_requirements(repo_dir, requirements_dir)
juju_log('Installing git repo from dir: {}'.format(repo_dir))
pip_install(repo_dir)
return repo_dir
def _git_update_requirements(package_dir, reqs_dir):
"""Update from global requirements.
Update an OpenStack git directory's requirements.txt and
test-requirements.txt from global-requirements.txt."""
orig_dir = os.getcwd()
os.chdir(reqs_dir)
cmd = "python update.py {}".format(package_dir)
try:
subprocess.check_call(cmd.split(' '))
except subprocess.CalledProcessError:
package = os.path.basename(package_dir)
error_out("Error updating {} from global-requirements.txt".format(package))
os.chdir(orig_dir)

View File

@ -0,0 +1,77 @@
#!/usr/bin/env python
# coding: utf-8
__author__ = "Jorge Niedbalski <jorge.niedbalski@canonical.com>"
from charmhelpers.fetch import apt_install, apt_update
from charmhelpers.core.hookenv import log
try:
from pip import main as pip_execute
except ImportError:
apt_update()
apt_install('python-pip')
from pip import main as pip_execute
def parse_options(given, available):
"""Given a set of options, check if available"""
for key, value in sorted(given.items()):
if key in available:
yield "--{0}={1}".format(key, value)
def pip_install_requirements(requirements, **options):
"""Install a requirements file """
command = ["install"]
available_options = ('proxy', 'src', 'log', )
for option in parse_options(options, available_options):
command.append(option)
command.append("-r {0}".format(requirements))
log("Installing from file: {} with options: {}".format(requirements,
command))
pip_execute(command)
def pip_install(package, fatal=False, **options):
"""Install a python package"""
command = ["install"]
available_options = ('proxy', 'src', 'log', "index-url", )
for option in parse_options(options, available_options):
command.append(option)
if isinstance(package, list):
command.extend(package)
else:
command.append(package)
log("Installing {} package with options: {}".format(package,
command))
pip_execute(command)
def pip_uninstall(package, **options):
"""Uninstall a python package"""
command = ["uninstall", "-q", "-y"]
available_options = ('proxy', 'log', )
for option in parse_options(options, available_options):
command.append(option)
if isinstance(package, list):
command.extend(package)
else:
command.append(package)
log("Uninstalling {} package with options: {}".format(package,
command))
pip_execute(command)
def pip_list():
"""Returns the list of current python installed packages
"""
return pip_execute(["list"])

View File

@ -16,19 +16,18 @@ import time
from subprocess import ( from subprocess import (
check_call, check_call,
check_output, check_output,
CalledProcessError CalledProcessError,
) )
from charmhelpers.core.hookenv import ( from charmhelpers.core.hookenv import (
relation_get, relation_get,
relation_ids, relation_ids,
related_units, related_units,
log, log,
DEBUG,
INFO, INFO,
WARNING, WARNING,
ERROR ERROR,
) )
from charmhelpers.core.host import ( from charmhelpers.core.host import (
mount, mount,
mounts, mounts,
@ -37,7 +36,6 @@ from charmhelpers.core.host import (
service_running, service_running,
umount, umount,
) )
from charmhelpers.fetch import ( from charmhelpers.fetch import (
apt_install, apt_install,
) )
@ -56,99 +54,85 @@ CEPH_CONF = """[global]
def install(): def install():
''' Basic Ceph client installation ''' """Basic Ceph client installation."""
ceph_dir = "/etc/ceph" ceph_dir = "/etc/ceph"
if not os.path.exists(ceph_dir): if not os.path.exists(ceph_dir):
os.mkdir(ceph_dir) os.mkdir(ceph_dir)
apt_install('ceph-common', fatal=True) apt_install('ceph-common', fatal=True)
def rbd_exists(service, pool, rbd_img): def rbd_exists(service, pool, rbd_img):
''' Check to see if a RADOS block device exists ''' """Check to see if a RADOS block device exists."""
try: try:
out = check_output(['rbd', 'list', '--id', service, out = check_output(['rbd', 'list', '--id',
'--pool', pool]) service, '--pool', pool]).decode('UTF-8')
except CalledProcessError: except CalledProcessError:
return False return False
else:
return rbd_img in out return rbd_img in out
def create_rbd_image(service, pool, image, sizemb): def create_rbd_image(service, pool, image, sizemb):
''' Create a new RADOS block device ''' """Create a new RADOS block device."""
cmd = [ cmd = ['rbd', 'create', image, '--size', str(sizemb), '--id', service,
'rbd', '--pool', pool]
'create',
image,
'--size',
str(sizemb),
'--id',
service,
'--pool',
pool
]
check_call(cmd) check_call(cmd)
def pool_exists(service, name): def pool_exists(service, name):
''' Check to see if a RADOS pool already exists ''' """Check to see if a RADOS pool already exists."""
try: try:
out = check_output(['rados', '--id', service, 'lspools']) out = check_output(['rados', '--id', service,
'lspools']).decode('UTF-8')
except CalledProcessError: except CalledProcessError:
return False return False
else:
return name in out return name in out
def get_osds(service): def get_osds(service):
''' """Return a list of all Ceph Object Storage Daemons currently in the
Return a list of all Ceph Object Storage Daemons cluster.
currently in the cluster """
'''
version = ceph_version() version = ceph_version()
if version and version >= '0.56': if version and version >= '0.56':
return json.loads(check_output(['ceph', '--id', service, return json.loads(check_output(['ceph', '--id', service,
'osd', 'ls', '--format=json'])) 'osd', 'ls',
else: '--format=json']).decode('UTF-8'))
return None
return None
def create_pool(service, name, replicas=2): def create_pool(service, name, replicas=3):
''' Create a new RADOS pool ''' """Create a new RADOS pool."""
if pool_exists(service, name): if pool_exists(service, name):
log("Ceph pool {} already exists, skipping creation".format(name), log("Ceph pool {} already exists, skipping creation".format(name),
level=WARNING) level=WARNING)
return return
# Calculate the number of placement groups based # Calculate the number of placement groups based
# on upstream recommended best practices. # on upstream recommended best practices.
osds = get_osds(service) osds = get_osds(service)
if osds: if osds:
pgnum = (len(osds) * 100 / replicas) pgnum = (len(osds) * 100 // replicas)
else: else:
# NOTE(james-page): Default to 200 for older ceph versions # NOTE(james-page): Default to 200 for older ceph versions
# which don't support OSD query from cli # which don't support OSD query from cli
pgnum = 200 pgnum = 200
cmd = [
'ceph', '--id', service, cmd = ['ceph', '--id', service, 'osd', 'pool', 'create', name, str(pgnum)]
'osd', 'pool', 'create',
name, str(pgnum)
]
check_call(cmd) check_call(cmd)
cmd = [
'ceph', '--id', service, cmd = ['ceph', '--id', service, 'osd', 'pool', 'set', name, 'size',
'osd', 'pool', 'set', name, str(replicas)]
'size', str(replicas)
]
check_call(cmd) check_call(cmd)
def delete_pool(service, name): def delete_pool(service, name):
''' Delete a RADOS pool from ceph ''' """Delete a RADOS pool from ceph."""
cmd = [ cmd = ['ceph', '--id', service, 'osd', 'pool', 'delete', name,
'ceph', '--id', service, '--yes-i-really-really-mean-it']
'osd', 'pool', 'delete',
name, '--yes-i-really-really-mean-it'
]
check_call(cmd) check_call(cmd)
@ -161,44 +145,43 @@ def _keyring_path(service):
def create_keyring(service, key): def create_keyring(service, key):
''' Create a new Ceph keyring containing key''' """Create a new Ceph keyring containing key."""
keyring = _keyring_path(service) keyring = _keyring_path(service)
if os.path.exists(keyring): if os.path.exists(keyring):
log('ceph: Keyring exists at %s.' % keyring, level=WARNING) log('Ceph keyring exists at %s.' % keyring, level=WARNING)
return return
cmd = [
'ceph-authtool', cmd = ['ceph-authtool', keyring, '--create-keyring',
keyring, '--name=client.{}'.format(service), '--add-key={}'.format(key)]
'--create-keyring',
'--name=client.{}'.format(service),
'--add-key={}'.format(key)
]
check_call(cmd) check_call(cmd)
log('ceph: Created new ring at %s.' % keyring, level=INFO) log('Created new ceph keyring at %s.' % keyring, level=DEBUG)
def create_key_file(service, key): def create_key_file(service, key):
''' Create a file containing key ''' """Create a file containing key."""
keyfile = _keyfile_path(service) keyfile = _keyfile_path(service)
if os.path.exists(keyfile): if os.path.exists(keyfile):
log('ceph: Keyfile exists at %s.' % keyfile, level=WARNING) log('Keyfile exists at %s.' % keyfile, level=WARNING)
return return
with open(keyfile, 'w') as fd: with open(keyfile, 'w') as fd:
fd.write(key) fd.write(key)
log('ceph: Created new keyfile at %s.' % keyfile, level=INFO)
log('Created new keyfile at %s.' % keyfile, level=INFO)
def get_ceph_nodes(): def get_ceph_nodes():
''' Query named relation 'ceph' to detemine current nodes ''' """Query named relation 'ceph' to determine current nodes."""
hosts = [] hosts = []
for r_id in relation_ids('ceph'): for r_id in relation_ids('ceph'):
for unit in related_units(r_id): for unit in related_units(r_id):
hosts.append(relation_get('private-address', unit=unit, rid=r_id)) hosts.append(relation_get('private-address', unit=unit, rid=r_id))
return hosts return hosts
def configure(service, key, auth, use_syslog): def configure(service, key, auth, use_syslog):
''' Perform basic configuration of Ceph ''' """Perform basic configuration of Ceph."""
create_keyring(service, key) create_keyring(service, key)
create_key_file(service, key) create_key_file(service, key)
hosts = get_ceph_nodes() hosts = get_ceph_nodes()
@ -211,17 +194,17 @@ def configure(service, key, auth, use_syslog):
def image_mapped(name): def image_mapped(name):
''' Determine whether a RADOS block device is mapped locally ''' """Determine whether a RADOS block device is mapped locally."""
try: try:
out = check_output(['rbd', 'showmapped']) out = check_output(['rbd', 'showmapped']).decode('UTF-8')
except CalledProcessError: except CalledProcessError:
return False return False
else:
return name in out return name in out
def map_block_storage(service, pool, image): def map_block_storage(service, pool, image):
''' Map a RADOS block device for local use ''' """Map a RADOS block device for local use."""
cmd = [ cmd = [
'rbd', 'rbd',
'map', 'map',
@ -235,31 +218,32 @@ def map_block_storage(service, pool, image):
def filesystem_mounted(fs): def filesystem_mounted(fs):
''' Determine whether a filesytems is already mounted ''' """Determine whether a filesytems is already mounted."""
return fs in [f for f, m in mounts()] return fs in [f for f, m in mounts()]
def make_filesystem(blk_device, fstype='ext4', timeout=10): def make_filesystem(blk_device, fstype='ext4', timeout=10):
''' Make a new filesystem on the specified block device ''' """Make a new filesystem on the specified block device."""
count = 0 count = 0
e_noent = os.errno.ENOENT e_noent = os.errno.ENOENT
while not os.path.exists(blk_device): while not os.path.exists(blk_device):
if count >= timeout: if count >= timeout:
log('ceph: gave up waiting on block device %s' % blk_device, log('Gave up waiting on block device %s' % blk_device,
level=ERROR) level=ERROR)
raise IOError(e_noent, os.strerror(e_noent), blk_device) raise IOError(e_noent, os.strerror(e_noent), blk_device)
log('ceph: waiting for block device %s to appear' % blk_device,
level=INFO) log('Waiting for block device %s to appear' % blk_device,
level=DEBUG)
count += 1 count += 1
time.sleep(1) time.sleep(1)
else: else:
log('ceph: Formatting block device %s as filesystem %s.' % log('Formatting block device %s as filesystem %s.' %
(blk_device, fstype), level=INFO) (blk_device, fstype), level=INFO)
check_call(['mkfs', '-t', fstype, blk_device]) check_call(['mkfs', '-t', fstype, blk_device])
def place_data_on_block_device(blk_device, data_src_dst): def place_data_on_block_device(blk_device, data_src_dst):
''' Migrate data in data_src_dst to blk_device and then remount ''' """Migrate data in data_src_dst to blk_device and then remount."""
# mount block device into /mnt # mount block device into /mnt
mount(blk_device, '/mnt') mount(blk_device, '/mnt')
# copy data to /mnt # copy data to /mnt
@ -279,8 +263,8 @@ def place_data_on_block_device(blk_device, data_src_dst):
# TODO: re-use # TODO: re-use
def modprobe(module): def modprobe(module):
''' Load a kernel module and configure for auto-load on reboot ''' """Load a kernel module and configure for auto-load on reboot."""
log('ceph: Loading kernel module', level=INFO) log('Loading kernel module', level=INFO)
cmd = ['modprobe', module] cmd = ['modprobe', module]
check_call(cmd) check_call(cmd)
with open('/etc/modules', 'r+') as modules: with open('/etc/modules', 'r+') as modules:
@ -289,7 +273,7 @@ def modprobe(module):
def copy_files(src, dst, symlinks=False, ignore=None): def copy_files(src, dst, symlinks=False, ignore=None):
''' Copy files from src to dst ''' """Copy files from src to dst."""
for item in os.listdir(src): for item in os.listdir(src):
s = os.path.join(src, item) s = os.path.join(src, item)
d = os.path.join(dst, item) d = os.path.join(dst, item)
@ -300,9 +284,9 @@ def copy_files(src, dst, symlinks=False, ignore=None):
def ensure_ceph_storage(service, pool, rbd_img, sizemb, mount_point, def ensure_ceph_storage(service, pool, rbd_img, sizemb, mount_point,
blk_device, fstype, system_services=[]): blk_device, fstype, system_services=[],
""" replicas=3):
NOTE: This function must only be called from a single service unit for """NOTE: This function must only be called from a single service unit for
the same rbd_img otherwise data loss will occur. the same rbd_img otherwise data loss will occur.
Ensures given pool and RBD image exists, is mapped to a block device, Ensures given pool and RBD image exists, is mapped to a block device,
@ -316,15 +300,16 @@ def ensure_ceph_storage(service, pool, rbd_img, sizemb, mount_point,
""" """
# Ensure pool, RBD image, RBD mappings are in place. # Ensure pool, RBD image, RBD mappings are in place.
if not pool_exists(service, pool): if not pool_exists(service, pool):
log('ceph: Creating new pool {}.'.format(pool)) log('Creating new pool {}.'.format(pool), level=INFO)
create_pool(service, pool) create_pool(service, pool, replicas=replicas)
if not rbd_exists(service, pool, rbd_img): if not rbd_exists(service, pool, rbd_img):
log('ceph: Creating RBD image ({}).'.format(rbd_img)) log('Creating RBD image ({}).'.format(rbd_img), level=INFO)
create_rbd_image(service, pool, rbd_img, sizemb) create_rbd_image(service, pool, rbd_img, sizemb)
if not image_mapped(rbd_img): if not image_mapped(rbd_img):
log('ceph: Mapping RBD Image {} as a Block Device.'.format(rbd_img)) log('Mapping RBD Image {} as a Block Device.'.format(rbd_img),
level=INFO)
map_block_storage(service, pool, rbd_img) map_block_storage(service, pool, rbd_img)
# make file system # make file system
@ -339,45 +324,47 @@ def ensure_ceph_storage(service, pool, rbd_img, sizemb, mount_point,
for svc in system_services: for svc in system_services:
if service_running(svc): if service_running(svc):
log('ceph: Stopping services {} prior to migrating data.' log('Stopping services {} prior to migrating data.'
.format(svc)) .format(svc), level=DEBUG)
service_stop(svc) service_stop(svc)
place_data_on_block_device(blk_device, mount_point) place_data_on_block_device(blk_device, mount_point)
for svc in system_services: for svc in system_services:
log('ceph: Starting service {} after migrating data.' log('Starting service {} after migrating data.'
.format(svc)) .format(svc), level=DEBUG)
service_start(svc) service_start(svc)
def ensure_ceph_keyring(service, user=None, group=None): def ensure_ceph_keyring(service, user=None, group=None):
''' """Ensures a ceph keyring is created for a named service and optionally
Ensures a ceph keyring is created for a named service ensures user and group ownership.
and optionally ensures user and group ownership.
Returns False if no ceph key is available in relation state. Returns False if no ceph key is available in relation state.
''' """
key = None key = None
for rid in relation_ids('ceph'): for rid in relation_ids('ceph'):
for unit in related_units(rid): for unit in related_units(rid):
key = relation_get('key', rid=rid, unit=unit) key = relation_get('key', rid=rid, unit=unit)
if key: if key:
break break
if not key: if not key:
return False return False
create_keyring(service=service, key=key) create_keyring(service=service, key=key)
keyring = _keyring_path(service) keyring = _keyring_path(service)
if user and group: if user and group:
check_call(['chown', '%s.%s' % (user, group), keyring]) check_call(['chown', '%s.%s' % (user, group), keyring])
return True return True
def ceph_version(): def ceph_version():
''' Retrieve the local version of ceph ''' """Retrieve the local version of ceph."""
if os.path.exists('/usr/bin/ceph'): if os.path.exists('/usr/bin/ceph'):
cmd = ['ceph', '-v'] cmd = ['ceph', '-v']
output = check_output(cmd) output = check_output(cmd).decode('US-ASCII')
output = output.split() output = output.split()
if len(output) > 3: if len(output) > 3:
return output[2] return output[2]

View File

@ -1,12 +1,12 @@
import os import os
import re import re
from subprocess import ( from subprocess import (
check_call, check_call,
check_output, check_output,
) )
import six
################################################## ##################################################
# loopback device helpers. # loopback device helpers.
@ -37,7 +37,7 @@ def create_loopback(file_path):
''' '''
file_path = os.path.abspath(file_path) file_path = os.path.abspath(file_path)
check_call(['losetup', '--find', file_path]) check_call(['losetup', '--find', file_path])
for d, f in loopback_devices().iteritems(): for d, f in six.iteritems(loopback_devices()):
if f == file_path: if f == file_path:
return d return d
@ -51,7 +51,7 @@ def ensure_loopback_device(path, size):
:returns: str: Full path to the ensured loopback device (eg, /dev/loop0) :returns: str: Full path to the ensured loopback device (eg, /dev/loop0)
''' '''
for d, f in loopback_devices().iteritems(): for d, f in six.iteritems(loopback_devices()):
if f == path: if f == path:
return d return d

View File

@ -61,6 +61,7 @@ def list_lvm_volume_group(block_device):
vg = None vg = None
pvd = check_output(['pvdisplay', block_device]).splitlines() pvd = check_output(['pvdisplay', block_device]).splitlines()
for l in pvd: for l in pvd:
l = l.decode('UTF-8')
if l.strip().startswith('VG Name'): if l.strip().startswith('VG Name'):
vg = ' '.join(l.strip().split()[2:]) vg = ' '.join(l.strip().split()[2:])
return vg return vg

View File

@ -30,7 +30,8 @@ def zap_disk(block_device):
# sometimes sgdisk exits non-zero; this is OK, dd will clean up # sometimes sgdisk exits non-zero; this is OK, dd will clean up
call(['sgdisk', '--zap-all', '--mbrtogpt', call(['sgdisk', '--zap-all', '--mbrtogpt',
'--clear', block_device]) '--clear', block_device])
dev_end = check_output(['blockdev', '--getsz', block_device]) dev_end = check_output(['blockdev', '--getsz',
block_device]).decode('UTF-8')
gpt_end = int(dev_end.split()[0]) - 100 gpt_end = int(dev_end.split()[0]) - 100
check_call(['dd', 'if=/dev/zero', 'of=%s' % (block_device), check_call(['dd', 'if=/dev/zero', 'of=%s' % (block_device),
'bs=1M', 'count=1']) 'bs=1M', 'count=1'])
@ -47,7 +48,7 @@ def is_device_mounted(device):
it doesn't. it doesn't.
''' '''
is_partition = bool(re.search(r".*[0-9]+\b", device)) is_partition = bool(re.search(r".*[0-9]+\b", device))
out = check_output(['mount']) out = check_output(['mount']).decode('UTF-8')
if is_partition: if is_partition:
return bool(re.search(device + r"\b", out)) return bool(re.search(device + r"\b", out))
return bool(re.search(device + r"[0-9]+\b", out)) return bool(re.search(device + r"[0-9]+\b", out))

View File

@ -3,10 +3,11 @@
__author__ = 'Jorge Niedbalski R. <jorge.niedbalski@canonical.com>' __author__ = 'Jorge Niedbalski R. <jorge.niedbalski@canonical.com>'
import io
import os import os
class Fstab(file): class Fstab(io.FileIO):
"""This class extends file in order to implement a file reader/writer """This class extends file in order to implement a file reader/writer
for file `/etc/fstab` for file `/etc/fstab`
""" """
@ -24,8 +25,8 @@ class Fstab(file):
options = "defaults" options = "defaults"
self.options = options self.options = options
self.d = d self.d = int(d)
self.p = p self.p = int(p)
def __eq__(self, o): def __eq__(self, o):
return str(self) == str(o) return str(self) == str(o)
@ -45,7 +46,7 @@ class Fstab(file):
self._path = path self._path = path
else: else:
self._path = self.DEFAULT_PATH self._path = self.DEFAULT_PATH
file.__init__(self, self._path, 'r+') super(Fstab, self).__init__(self._path, 'rb+')
def _hydrate_entry(self, line): def _hydrate_entry(self, line):
# NOTE: use split with no arguments to split on any # NOTE: use split with no arguments to split on any
@ -58,8 +59,9 @@ class Fstab(file):
def entries(self): def entries(self):
self.seek(0) self.seek(0)
for line in self.readlines(): for line in self.readlines():
line = line.decode('us-ascii')
try: try:
if not line.startswith("#"): if line.strip() and not line.startswith("#"):
yield self._hydrate_entry(line) yield self._hydrate_entry(line)
except ValueError: except ValueError:
pass pass
@ -75,14 +77,14 @@ class Fstab(file):
if self.get_entry_by_attr('device', entry.device): if self.get_entry_by_attr('device', entry.device):
return False return False
self.write(str(entry) + '\n') self.write((str(entry) + '\n').encode('us-ascii'))
self.truncate() self.truncate()
return entry return entry
def remove_entry(self, entry): def remove_entry(self, entry):
self.seek(0) self.seek(0)
lines = self.readlines() lines = [l.decode('us-ascii') for l in self.readlines()]
found = False found = False
for index, line in enumerate(lines): for index, line in enumerate(lines):
@ -97,7 +99,7 @@ class Fstab(file):
lines.remove(line) lines.remove(line)
self.seek(0) self.seek(0)
self.write(''.join(lines)) self.write(''.join(lines).encode('us-ascii'))
self.truncate() self.truncate()
return True return True

View File

@ -9,9 +9,14 @@ import json
import yaml import yaml
import subprocess import subprocess
import sys import sys
import UserDict
from subprocess import CalledProcessError from subprocess import CalledProcessError
import six
if not six.PY3:
from UserDict import UserDict
else:
from collections import UserDict
CRITICAL = "CRITICAL" CRITICAL = "CRITICAL"
ERROR = "ERROR" ERROR = "ERROR"
WARNING = "WARNING" WARNING = "WARNING"
@ -63,16 +68,18 @@ def log(message, level=None):
command = ['juju-log'] command = ['juju-log']
if level: if level:
command += ['-l', level] command += ['-l', level]
if not isinstance(message, six.string_types):
message = repr(message)
command += [message] command += [message]
subprocess.call(command) subprocess.call(command)
class Serializable(UserDict.IterableUserDict): class Serializable(UserDict):
"""Wrapper, an object that can be serialized to yaml or json""" """Wrapper, an object that can be serialized to yaml or json"""
def __init__(self, obj): def __init__(self, obj):
# wrap the object # wrap the object
UserDict.IterableUserDict.__init__(self) UserDict.__init__(self)
self.data = obj self.data = obj
def __getattr__(self, attr): def __getattr__(self, attr):
@ -214,6 +221,12 @@ class Config(dict):
except KeyError: except KeyError:
return (self._prev_dict or {})[key] return (self._prev_dict or {})[key]
def keys(self):
prev_keys = []
if self._prev_dict is not None:
prev_keys = self._prev_dict.keys()
return list(set(prev_keys + list(dict.keys(self))))
def load_previous(self, path=None): def load_previous(self, path=None):
"""Load previous copy of config from disk. """Load previous copy of config from disk.
@ -263,7 +276,7 @@ class Config(dict):
""" """
if self._prev_dict: if self._prev_dict:
for k, v in self._prev_dict.iteritems(): for k, v in six.iteritems(self._prev_dict):
if k not in self: if k not in self:
self[k] = v self[k] = v
with open(self.path, 'w') as f: with open(self.path, 'w') as f:
@ -278,7 +291,8 @@ def config(scope=None):
config_cmd_line.append(scope) config_cmd_line.append(scope)
config_cmd_line.append('--format=json') config_cmd_line.append('--format=json')
try: try:
config_data = json.loads(subprocess.check_output(config_cmd_line)) config_data = json.loads(
subprocess.check_output(config_cmd_line).decode('UTF-8'))
if scope is not None: if scope is not None:
return config_data return config_data
return Config(config_data) return Config(config_data)
@ -297,10 +311,10 @@ def relation_get(attribute=None, unit=None, rid=None):
if unit: if unit:
_args.append(unit) _args.append(unit)
try: try:
return json.loads(subprocess.check_output(_args)) return json.loads(subprocess.check_output(_args).decode('UTF-8'))
except ValueError: except ValueError:
return None return None
except CalledProcessError, e: except CalledProcessError as e:
if e.returncode == 2: if e.returncode == 2:
return None return None
raise raise
@ -312,7 +326,7 @@ def relation_set(relation_id=None, relation_settings=None, **kwargs):
relation_cmd_line = ['relation-set'] relation_cmd_line = ['relation-set']
if relation_id is not None: if relation_id is not None:
relation_cmd_line.extend(('-r', relation_id)) relation_cmd_line.extend(('-r', relation_id))
for k, v in (relation_settings.items() + kwargs.items()): for k, v in (list(relation_settings.items()) + list(kwargs.items())):
if v is None: if v is None:
relation_cmd_line.append('{}='.format(k)) relation_cmd_line.append('{}='.format(k))
else: else:
@ -329,7 +343,8 @@ def relation_ids(reltype=None):
relid_cmd_line = ['relation-ids', '--format=json'] relid_cmd_line = ['relation-ids', '--format=json']
if reltype is not None: if reltype is not None:
relid_cmd_line.append(reltype) relid_cmd_line.append(reltype)
return json.loads(subprocess.check_output(relid_cmd_line)) or [] return json.loads(
subprocess.check_output(relid_cmd_line).decode('UTF-8')) or []
return [] return []
@ -340,7 +355,8 @@ def related_units(relid=None):
units_cmd_line = ['relation-list', '--format=json'] units_cmd_line = ['relation-list', '--format=json']
if relid is not None: if relid is not None:
units_cmd_line.extend(('-r', relid)) units_cmd_line.extend(('-r', relid))
return json.loads(subprocess.check_output(units_cmd_line)) or [] return json.loads(
subprocess.check_output(units_cmd_line).decode('UTF-8')) or []
@cached @cached
@ -449,7 +465,7 @@ def unit_get(attribute):
"""Get the unit ID for the remote unit""" """Get the unit ID for the remote unit"""
_args = ['unit-get', '--format=json', attribute] _args = ['unit-get', '--format=json', attribute]
try: try:
return json.loads(subprocess.check_output(_args)) return json.loads(subprocess.check_output(_args).decode('UTF-8'))
except ValueError: except ValueError:
return None return None

View File

@ -13,13 +13,13 @@ import random
import string import string
import subprocess import subprocess
import hashlib import hashlib
import shutil
from contextlib import contextmanager from contextlib import contextmanager
from collections import OrderedDict from collections import OrderedDict
from hookenv import log import six
from fstab import Fstab
from .hookenv import log
from .fstab import Fstab
def service_start(service_name): def service_start(service_name):
@ -55,7 +55,9 @@ def service(action, service_name):
def service_running(service): def service_running(service):
"""Determine whether a system service is running""" """Determine whether a system service is running"""
try: try:
output = subprocess.check_output(['service', service, 'status'], stderr=subprocess.STDOUT) output = subprocess.check_output(
['service', service, 'status'],
stderr=subprocess.STDOUT).decode('UTF-8')
except subprocess.CalledProcessError: except subprocess.CalledProcessError:
return False return False
else: else:
@ -68,7 +70,9 @@ def service_running(service):
def service_available(service_name): def service_available(service_name):
"""Determine whether a system service is available""" """Determine whether a system service is available"""
try: try:
subprocess.check_output(['service', service_name, 'status'], stderr=subprocess.STDOUT) subprocess.check_output(
['service', service_name, 'status'],
stderr=subprocess.STDOUT).decode('UTF-8')
except subprocess.CalledProcessError as e: except subprocess.CalledProcessError as e:
return 'unrecognized service' not in e.output return 'unrecognized service' not in e.output
else: else:
@ -97,6 +101,26 @@ def adduser(username, password=None, shell='/bin/bash', system_user=False):
return user_info return user_info
def add_group(group_name, system_group=False):
"""Add a group to the system"""
try:
group_info = grp.getgrnam(group_name)
log('group {0} already exists!'.format(group_name))
except KeyError:
log('creating group {0}'.format(group_name))
cmd = ['addgroup']
if system_group:
cmd.append('--system')
else:
cmd.extend([
'--group',
])
cmd.append(group_name)
subprocess.check_call(cmd)
group_info = grp.getgrnam(group_name)
return group_info
def add_user_to_group(username, group): def add_user_to_group(username, group):
"""Add a user to a group""" """Add a user to a group"""
cmd = [ cmd = [
@ -116,7 +140,7 @@ def rsync(from_path, to_path, flags='-r', options=None):
cmd.append(from_path) cmd.append(from_path)
cmd.append(to_path) cmd.append(to_path)
log(" ".join(cmd)) log(" ".join(cmd))
return subprocess.check_output(cmd).strip() return subprocess.check_output(cmd).decode('UTF-8').strip()
def symlink(source, destination): def symlink(source, destination):
@ -131,7 +155,7 @@ def symlink(source, destination):
subprocess.check_call(cmd) subprocess.check_call(cmd)
def mkdir(path, owner='root', group='root', perms=0555, force=False): def mkdir(path, owner='root', group='root', perms=0o555, force=False):
"""Create a directory""" """Create a directory"""
log("Making dir {} {}:{} {:o}".format(path, owner, group, log("Making dir {} {}:{} {:o}".format(path, owner, group,
perms)) perms))
@ -147,7 +171,7 @@ def mkdir(path, owner='root', group='root', perms=0555, force=False):
os.chown(realpath, uid, gid) os.chown(realpath, uid, gid)
def write_file(path, content, owner='root', group='root', perms=0444): def write_file(path, content, owner='root', group='root', perms=0o444):
"""Create or overwrite a file with the contents of a string""" """Create or overwrite a file with the contents of a string"""
log("Writing file {} {}:{} {:o}".format(path, owner, group, perms)) log("Writing file {} {}:{} {:o}".format(path, owner, group, perms))
uid = pwd.getpwnam(owner).pw_uid uid = pwd.getpwnam(owner).pw_uid
@ -178,7 +202,7 @@ def mount(device, mountpoint, options=None, persist=False, filesystem="ext3"):
cmd_args.extend([device, mountpoint]) cmd_args.extend([device, mountpoint])
try: try:
subprocess.check_output(cmd_args) subprocess.check_output(cmd_args)
except subprocess.CalledProcessError, e: except subprocess.CalledProcessError as e:
log('Error mounting {} at {}\n{}'.format(device, mountpoint, e.output)) log('Error mounting {} at {}\n{}'.format(device, mountpoint, e.output))
return False return False
@ -192,7 +216,7 @@ def umount(mountpoint, persist=False):
cmd_args = ['umount', mountpoint] cmd_args = ['umount', mountpoint]
try: try:
subprocess.check_output(cmd_args) subprocess.check_output(cmd_args)
except subprocess.CalledProcessError, e: except subprocess.CalledProcessError as e:
log('Error unmounting {}\n{}'.format(mountpoint, e.output)) log('Error unmounting {}\n{}'.format(mountpoint, e.output))
return False return False
@ -219,8 +243,8 @@ def file_hash(path, hash_type='md5'):
""" """
if os.path.exists(path): if os.path.exists(path):
h = getattr(hashlib, hash_type)() h = getattr(hashlib, hash_type)()
with open(path, 'r') as source: with open(path, 'rb') as source:
h.update(source.read()) # IGNORE:E1101 - it does have update h.update(source.read())
return h.hexdigest() return h.hexdigest()
else: else:
return None return None
@ -298,7 +322,7 @@ def pwgen(length=None):
if length is None: if length is None:
length = random.choice(range(35, 45)) length = random.choice(range(35, 45))
alphanumeric_chars = [ alphanumeric_chars = [
l for l in (string.letters + string.digits) l for l in (string.ascii_letters + string.digits)
if l not in 'l0QD1vAEIOUaeiou'] if l not in 'l0QD1vAEIOUaeiou']
random_chars = [ random_chars = [
random.choice(alphanumeric_chars) for _ in range(length)] random.choice(alphanumeric_chars) for _ in range(length)]
@ -307,14 +331,14 @@ def pwgen(length=None):
def list_nics(nic_type): def list_nics(nic_type):
'''Return a list of nics of given type(s)''' '''Return a list of nics of given type(s)'''
if isinstance(nic_type, basestring): if isinstance(nic_type, six.string_types):
int_types = [nic_type] int_types = [nic_type]
else: else:
int_types = nic_type int_types = nic_type
interfaces = [] interfaces = []
for int_type in int_types: for int_type in int_types:
cmd = ['ip', 'addr', 'show', 'label', int_type + '*'] cmd = ['ip', 'addr', 'show', 'label', int_type + '*']
ip_output = subprocess.check_output(cmd).split('\n') ip_output = subprocess.check_output(cmd).decode('UTF-8').split('\n')
ip_output = (line for line in ip_output if line) ip_output = (line for line in ip_output if line)
for line in ip_output: for line in ip_output:
if line.split()[1].startswith(int_type): if line.split()[1].startswith(int_type):
@ -336,7 +360,7 @@ def set_nic_mtu(nic, mtu):
def get_nic_mtu(nic): def get_nic_mtu(nic):
cmd = ['ip', 'addr', 'show', nic] cmd = ['ip', 'addr', 'show', nic]
ip_output = subprocess.check_output(cmd).split('\n') ip_output = subprocess.check_output(cmd).decode('UTF-8').split('\n')
mtu = "" mtu = ""
for line in ip_output: for line in ip_output:
words = line.split() words = line.split()
@ -347,7 +371,7 @@ def get_nic_mtu(nic):
def get_nic_hwaddr(nic): def get_nic_hwaddr(nic):
cmd = ['ip', '-o', '-0', 'addr', 'show', nic] cmd = ['ip', '-o', '-0', 'addr', 'show', nic]
ip_output = subprocess.check_output(cmd) ip_output = subprocess.check_output(cmd).decode('UTF-8')
hwaddr = "" hwaddr = ""
words = ip_output.split() words = ip_output.split()
if 'link/ether' in words: if 'link/ether' in words:
@ -364,8 +388,8 @@ def cmp_pkgrevno(package, revno, pkgcache=None):
''' '''
import apt_pkg import apt_pkg
from charmhelpers.fetch import apt_cache
if not pkgcache: if not pkgcache:
from charmhelpers.fetch import apt_cache
pkgcache = apt_cache() pkgcache = apt_cache()
pkg = pkgcache[package] pkg = pkgcache[package]
return apt_pkg.version_compare(pkg.current_ver.ver_str, revno) return apt_pkg.version_compare(pkg.current_ver.ver_str, revno)

View File

@ -1,2 +1,2 @@
from .base import * from .base import * # NOQA
from .helpers import * from .helpers import * # NOQA

View File

@ -196,7 +196,7 @@ class StoredContext(dict):
if not os.path.isabs(file_name): if not os.path.isabs(file_name):
file_name = os.path.join(hookenv.charm_dir(), file_name) file_name = os.path.join(hookenv.charm_dir(), file_name)
with open(file_name, 'w') as file_stream: with open(file_name, 'w') as file_stream:
os.fchmod(file_stream.fileno(), 0600) os.fchmod(file_stream.fileno(), 0o600)
yaml.dump(config_data, file_stream) yaml.dump(config_data, file_stream)
def read_context(self, file_name): def read_context(self, file_name):
@ -211,15 +211,19 @@ class StoredContext(dict):
class TemplateCallback(ManagerCallback): class TemplateCallback(ManagerCallback):
""" """
Callback class that will render a Jinja2 template, for use as a ready action. Callback class that will render a Jinja2 template, for use as a ready
action.
:param str source: The template source file, relative to
`$CHARM_DIR/templates`
:param str source: The template source file, relative to `$CHARM_DIR/templates`
:param str target: The target to write the rendered template to :param str target: The target to write the rendered template to
:param str owner: The owner of the rendered file :param str owner: The owner of the rendered file
:param str group: The group of the rendered file :param str group: The group of the rendered file
:param int perms: The permissions of the rendered file :param int perms: The permissions of the rendered file
""" """
def __init__(self, source, target, owner='root', group='root', perms=0444): def __init__(self, source, target,
owner='root', group='root', perms=0o444):
self.source = source self.source = source
self.target = target self.target = target
self.owner = owner self.owner = owner

View File

@ -4,7 +4,8 @@ from charmhelpers.core import host
from charmhelpers.core import hookenv from charmhelpers.core import hookenv
def render(source, target, context, owner='root', group='root', perms=0444, templates_dir=None): def render(source, target, context, owner='root', group='root',
perms=0o444, templates_dir=None):
""" """
Render a template. Render a template.

View File

@ -5,10 +5,6 @@ from yaml import safe_load
from charmhelpers.core.host import ( from charmhelpers.core.host import (
lsb_release lsb_release
) )
from urlparse import (
urlparse,
urlunparse,
)
import subprocess import subprocess
from charmhelpers.core.hookenv import ( from charmhelpers.core.hookenv import (
config, config,
@ -16,6 +12,12 @@ from charmhelpers.core.hookenv import (
) )
import os import os
import six
if six.PY3:
from urllib.parse import urlparse, urlunparse
else:
from urlparse import urlparse, urlunparse
CLOUD_ARCHIVE = """# Ubuntu Cloud Archive CLOUD_ARCHIVE = """# Ubuntu Cloud Archive
deb http://ubuntu-cloud.archive.canonical.com/ubuntu {} main deb http://ubuntu-cloud.archive.canonical.com/ubuntu {} main
@ -72,6 +74,7 @@ CLOUD_ARCHIVE_POCKETS = {
FETCH_HANDLERS = ( FETCH_HANDLERS = (
'charmhelpers.fetch.archiveurl.ArchiveUrlFetchHandler', 'charmhelpers.fetch.archiveurl.ArchiveUrlFetchHandler',
'charmhelpers.fetch.bzrurl.BzrUrlFetchHandler', 'charmhelpers.fetch.bzrurl.BzrUrlFetchHandler',
'charmhelpers.fetch.giturl.GitUrlFetchHandler',
) )
APT_NO_LOCK = 100 # The return code for "couldn't acquire lock" in APT. APT_NO_LOCK = 100 # The return code for "couldn't acquire lock" in APT.
@ -148,7 +151,7 @@ def apt_install(packages, options=None, fatal=False):
cmd = ['apt-get', '--assume-yes'] cmd = ['apt-get', '--assume-yes']
cmd.extend(options) cmd.extend(options)
cmd.append('install') cmd.append('install')
if isinstance(packages, basestring): if isinstance(packages, six.string_types):
cmd.append(packages) cmd.append(packages)
else: else:
cmd.extend(packages) cmd.extend(packages)
@ -181,7 +184,7 @@ def apt_update(fatal=False):
def apt_purge(packages, fatal=False): def apt_purge(packages, fatal=False):
"""Purge one or more packages""" """Purge one or more packages"""
cmd = ['apt-get', '--assume-yes', 'purge'] cmd = ['apt-get', '--assume-yes', 'purge']
if isinstance(packages, basestring): if isinstance(packages, six.string_types):
cmd.append(packages) cmd.append(packages)
else: else:
cmd.extend(packages) cmd.extend(packages)
@ -192,7 +195,7 @@ def apt_purge(packages, fatal=False):
def apt_hold(packages, fatal=False): def apt_hold(packages, fatal=False):
"""Hold one or more packages""" """Hold one or more packages"""
cmd = ['apt-mark', 'hold'] cmd = ['apt-mark', 'hold']
if isinstance(packages, basestring): if isinstance(packages, six.string_types):
cmd.append(packages) cmd.append(packages)
else: else:
cmd.extend(packages) cmd.extend(packages)
@ -218,6 +221,7 @@ def add_source(source, key=None):
pocket for the release. pocket for the release.
'cloud:' may be used to activate official cloud archive pockets, 'cloud:' may be used to activate official cloud archive pockets,
such as 'cloud:icehouse' such as 'cloud:icehouse'
'distro' may be used as a noop
@param key: A key to be added to the system's APT keyring and used @param key: A key to be added to the system's APT keyring and used
to verify the signatures on packages. Ideally, this should be an to verify the signatures on packages. Ideally, this should be an
@ -251,12 +255,14 @@ def add_source(source, key=None):
release = lsb_release()['DISTRIB_CODENAME'] release = lsb_release()['DISTRIB_CODENAME']
with open('/etc/apt/sources.list.d/proposed.list', 'w') as apt: with open('/etc/apt/sources.list.d/proposed.list', 'w') as apt:
apt.write(PROPOSED_POCKET.format(release)) apt.write(PROPOSED_POCKET.format(release))
elif source == 'distro':
pass
else: else:
raise SourceConfigError("Unknown source: {!r}".format(source)) log("Unknown source: {!r}".format(source))
if key: if key:
if '-----BEGIN PGP PUBLIC KEY BLOCK-----' in key: if '-----BEGIN PGP PUBLIC KEY BLOCK-----' in key:
with NamedTemporaryFile() as key_file: with NamedTemporaryFile('w+') as key_file:
key_file.write(key) key_file.write(key)
key_file.flush() key_file.flush()
key_file.seek(0) key_file.seek(0)
@ -293,14 +299,14 @@ def configure_sources(update=False,
sources = safe_load((config(sources_var) or '').strip()) or [] sources = safe_load((config(sources_var) or '').strip()) or []
keys = safe_load((config(keys_var) or '').strip()) or None keys = safe_load((config(keys_var) or '').strip()) or None
if isinstance(sources, basestring): if isinstance(sources, six.string_types):
sources = [sources] sources = [sources]
if keys is None: if keys is None:
for source in sources: for source in sources:
add_source(source, None) add_source(source, None)
else: else:
if isinstance(keys, basestring): if isinstance(keys, six.string_types):
keys = [keys] keys = [keys]
if len(sources) != len(keys): if len(sources) != len(keys):
@ -397,7 +403,7 @@ def _run_apt_command(cmd, fatal=False):
while result is None or result == APT_NO_LOCK: while result is None or result == APT_NO_LOCK:
try: try:
result = subprocess.check_call(cmd, env=env) result = subprocess.check_call(cmd, env=env)
except subprocess.CalledProcessError, e: except subprocess.CalledProcessError as e:
retry_count = retry_count + 1 retry_count = retry_count + 1
if retry_count > APT_NO_LOCK_RETRY_COUNT: if retry_count > APT_NO_LOCK_RETRY_COUNT:
raise raise

View File

@ -1,8 +1,23 @@
import os import os
import urllib2
from urllib import urlretrieve
import urlparse
import hashlib import hashlib
import re
import six
if six.PY3:
from urllib.request import (
build_opener, install_opener, urlopen, urlretrieve,
HTTPPasswordMgrWithDefaultRealm, HTTPBasicAuthHandler,
)
from urllib.parse import urlparse, urlunparse, parse_qs
from urllib.error import URLError
else:
from urllib import urlretrieve
from urllib2 import (
build_opener, install_opener, urlopen,
HTTPPasswordMgrWithDefaultRealm, HTTPBasicAuthHandler,
URLError
)
from urlparse import urlparse, urlunparse, parse_qs
from charmhelpers.fetch import ( from charmhelpers.fetch import (
BaseFetchHandler, BaseFetchHandler,
@ -15,6 +30,24 @@ from charmhelpers.payload.archive import (
from charmhelpers.core.host import mkdir, check_hash from charmhelpers.core.host import mkdir, check_hash
def splituser(host):
'''urllib.splituser(), but six's support of this seems broken'''
_userprog = re.compile('^(.*)@(.*)$')
match = _userprog.match(host)
if match:
return match.group(1, 2)
return None, host
def splitpasswd(user):
'''urllib.splitpasswd(), but six's support of this is missing'''
_passwdprog = re.compile('^([^:]*):(.*)$', re.S)
match = _passwdprog.match(user)
if match:
return match.group(1, 2)
return user, None
class ArchiveUrlFetchHandler(BaseFetchHandler): class ArchiveUrlFetchHandler(BaseFetchHandler):
""" """
Handler to download archive files from arbitrary URLs. Handler to download archive files from arbitrary URLs.
@ -42,20 +75,20 @@ class ArchiveUrlFetchHandler(BaseFetchHandler):
""" """
# propogate all exceptions # propogate all exceptions
# URLError, OSError, etc # URLError, OSError, etc
proto, netloc, path, params, query, fragment = urlparse.urlparse(source) proto, netloc, path, params, query, fragment = urlparse(source)
if proto in ('http', 'https'): if proto in ('http', 'https'):
auth, barehost = urllib2.splituser(netloc) auth, barehost = splituser(netloc)
if auth is not None: if auth is not None:
source = urlparse.urlunparse((proto, barehost, path, params, query, fragment)) source = urlunparse((proto, barehost, path, params, query, fragment))
username, password = urllib2.splitpasswd(auth) username, password = splitpasswd(auth)
passman = urllib2.HTTPPasswordMgrWithDefaultRealm() passman = HTTPPasswordMgrWithDefaultRealm()
# Realm is set to None in add_password to force the username and password # Realm is set to None in add_password to force the username and password
# to be used whatever the realm # to be used whatever the realm
passman.add_password(None, source, username, password) passman.add_password(None, source, username, password)
authhandler = urllib2.HTTPBasicAuthHandler(passman) authhandler = HTTPBasicAuthHandler(passman)
opener = urllib2.build_opener(authhandler) opener = build_opener(authhandler)
urllib2.install_opener(opener) install_opener(opener)
response = urllib2.urlopen(source) response = urlopen(source)
try: try:
with open(dest, 'w') as dest_file: with open(dest, 'w') as dest_file:
dest_file.write(response.read()) dest_file.write(response.read())
@ -91,17 +124,21 @@ class ArchiveUrlFetchHandler(BaseFetchHandler):
url_parts = self.parse_url(source) url_parts = self.parse_url(source)
dest_dir = os.path.join(os.environ.get('CHARM_DIR'), 'fetched') dest_dir = os.path.join(os.environ.get('CHARM_DIR'), 'fetched')
if not os.path.exists(dest_dir): if not os.path.exists(dest_dir):
mkdir(dest_dir, perms=0755) mkdir(dest_dir, perms=0o755)
dld_file = os.path.join(dest_dir, os.path.basename(url_parts.path)) dld_file = os.path.join(dest_dir, os.path.basename(url_parts.path))
try: try:
self.download(source, dld_file) self.download(source, dld_file)
except urllib2.URLError as e: except URLError as e:
raise UnhandledSource(e.reason) raise UnhandledSource(e.reason)
except OSError as e: except OSError as e:
raise UnhandledSource(e.strerror) raise UnhandledSource(e.strerror)
options = urlparse.parse_qs(url_parts.fragment) options = parse_qs(url_parts.fragment)
for key, value in options.items(): for key, value in options.items():
if key in hashlib.algorithms: if not six.PY3:
algorithms = hashlib.algorithms
else:
algorithms = hashlib.algorithms_available
if key in algorithms:
check_hash(dld_file, value, key) check_hash(dld_file, value, key)
if checksum: if checksum:
check_hash(dld_file, checksum, hash_type) check_hash(dld_file, checksum, hash_type)

View File

@ -5,6 +5,10 @@ from charmhelpers.fetch import (
) )
from charmhelpers.core.host import mkdir from charmhelpers.core.host import mkdir
import six
if six.PY3:
raise ImportError('bzrlib does not support Python3')
try: try:
from bzrlib.branch import Branch from bzrlib.branch import Branch
except ImportError: except ImportError:
@ -42,7 +46,7 @@ class BzrUrlFetchHandler(BaseFetchHandler):
dest_dir = os.path.join(os.environ.get('CHARM_DIR'), "fetched", dest_dir = os.path.join(os.environ.get('CHARM_DIR'), "fetched",
branch_name) branch_name)
if not os.path.exists(dest_dir): if not os.path.exists(dest_dir):
mkdir(dest_dir, perms=0755) mkdir(dest_dir, perms=0o755)
try: try:
self.branch(source, dest_dir) self.branch(source, dest_dir)
except OSError as e: except OSError as e:

View File

@ -0,0 +1,51 @@
import os
from charmhelpers.fetch import (
BaseFetchHandler,
UnhandledSource
)
from charmhelpers.core.host import mkdir
import six
if six.PY3:
raise ImportError('GitPython does not support Python 3')
try:
from git import Repo
except ImportError:
from charmhelpers.fetch import apt_install
apt_install("python-git")
from git import Repo
class GitUrlFetchHandler(BaseFetchHandler):
"""Handler for git branches via generic and github URLs"""
def can_handle(self, source):
url_parts = self.parse_url(source)
# TODO (mattyw) no support for ssh git@ yet
if url_parts.scheme not in ('http', 'https', 'git'):
return False
else:
return True
def clone(self, source, dest, branch):
if not self.can_handle(source):
raise UnhandledSource("Cannot handle {}".format(source))
repo = Repo.clone_from(source, dest)
repo.git.checkout(branch)
def install(self, source, branch="master", dest=None):
url_parts = self.parse_url(source)
branch_name = url_parts.path.strip("/").split("/")[-1]
if dest:
dest_dir = os.path.join(dest, branch_name)
else:
dest_dir = os.path.join(os.environ.get('CHARM_DIR'), "fetched",
branch_name)
if not os.path.exists(dest_dir):
mkdir(dest_dir, perms=0o755)
try:
self.clone(source, dest_dir, branch)
except OSError as e:
raise UnhandledSource(e.strerror)
return dest_dir

View File

@ -1,6 +1,6 @@
import amulet import amulet
import os import os
import six
class AmuletDeployment(object): class AmuletDeployment(object):
@ -52,12 +52,12 @@ class AmuletDeployment(object):
def _add_relations(self, relations): def _add_relations(self, relations):
"""Add all of the relations for the services.""" """Add all of the relations for the services."""
for k, v in relations.iteritems(): for k, v in six.iteritems(relations):
self.d.relate(k, v) self.d.relate(k, v)
def _configure_services(self, configs): def _configure_services(self, configs):
"""Configure all of the services.""" """Configure all of the services."""
for service, config in configs.iteritems(): for service, config in six.iteritems(configs):
self.d.configure(service, config) self.d.configure(service, config)
def _deploy(self): def _deploy(self):

View File

@ -5,6 +5,8 @@ import re
import sys import sys
import time import time
import six
class AmuletUtils(object): class AmuletUtils(object):
"""Amulet utilities. """Amulet utilities.
@ -58,7 +60,7 @@ class AmuletUtils(object):
Verify the specified services are running on the corresponding Verify the specified services are running on the corresponding
service units. service units.
""" """
for k, v in commands.iteritems(): for k, v in six.iteritems(commands):
for cmd in v: for cmd in v:
output, code = k.run(cmd) output, code = k.run(cmd)
if code != 0: if code != 0:
@ -100,11 +102,11 @@ class AmuletUtils(object):
longs, or can be a function that evaluate a variable and returns a longs, or can be a function that evaluate a variable and returns a
bool. bool.
""" """
for k, v in expected.iteritems(): for k, v in six.iteritems(expected):
if k in actual: if k in actual:
if (isinstance(v, basestring) or if (isinstance(v, six.string_types) or
isinstance(v, bool) or isinstance(v, bool) or
isinstance(v, (int, long))): isinstance(v, six.integer_types)):
if v != actual[k]: if v != actual[k]:
return "{}:{}".format(k, actual[k]) return "{}:{}".format(k, actual[k])
elif not v(actual[k]): elif not v(actual[k]):

View File

@ -1,3 +1,4 @@
import six
from charmhelpers.contrib.amulet.deployment import ( from charmhelpers.contrib.amulet.deployment import (
AmuletDeployment AmuletDeployment
) )
@ -69,7 +70,7 @@ class OpenStackAmuletDeployment(AmuletDeployment):
def _configure_services(self, configs): def _configure_services(self, configs):
"""Configure all of the services.""" """Configure all of the services."""
for service, config in configs.iteritems(): for service, config in six.iteritems(configs):
self.d.configure(service, config) self.d.configure(service, config)
def _get_openstack_release(self): def _get_openstack_release(self):

View File

@ -7,6 +7,8 @@ import glanceclient.v1.client as glance_client
import keystoneclient.v2_0 as keystone_client import keystoneclient.v2_0 as keystone_client
import novaclient.v1_1.client as nova_client import novaclient.v1_1.client as nova_client
import six
from charmhelpers.contrib.amulet.utils import ( from charmhelpers.contrib.amulet.utils import (
AmuletUtils AmuletUtils
) )
@ -60,7 +62,7 @@ class OpenStackAmuletUtils(AmuletUtils):
expected service catalog endpoints. expected service catalog endpoints.
""" """
self.log.debug('actual: {}'.format(repr(actual))) self.log.debug('actual: {}'.format(repr(actual)))
for k, v in expected.iteritems(): for k, v in six.iteritems(expected):
if k in actual: if k in actual:
ret = self._validate_dict_data(expected[k][0], actual[k][0]) ret = self._validate_dict_data(expected[k][0], actual[k][0])
if ret: if ret: