diff --git a/.project b/.project
new file mode 100644
index 00000000..a146609a
--- /dev/null
+++ b/.project
@@ -0,0 +1,17 @@
+
+
+ neutron-openvswitch
+
+
+
+
+
+ org.python.pydev.PyDevBuilder
+
+
+
+
+
+ org.python.pydev.pythonNature
+
+
diff --git a/.pydevproject b/.pydevproject
new file mode 100644
index 00000000..d6fbb946
--- /dev/null
+++ b/.pydevproject
@@ -0,0 +1,9 @@
+
+
+python 2.7
+Default
+
+/neutron-openvswitch/hooks
+/neutron-openvswitch/unit_tests
+
+
diff --git a/charm-helpers-sync.yaml b/charm-helpers-sync.yaml
index d5fab032..2439ece8 100644
--- a/charm-helpers-sync.yaml
+++ b/charm-helpers-sync.yaml
@@ -1,4 +1,4 @@
-branch: lp:charm-helpers
+branch: lp:~james-page/charm-helpers/network-splits
destination: hooks/charmhelpers
include:
- core
@@ -8,3 +8,4 @@ include:
- contrib.network.ovs
- contrib.storage.linux
- payload.execd
+ - contrib.network.ip
diff --git a/config.yaml b/config.yaml
index 3033d6f0..9b916905 100644
--- a/config.yaml
+++ b/config.yaml
@@ -21,3 +21,14 @@ options:
default: False
type: boolean
description: Enable verbose logging
+ # Network configuration options
+ # by default all access is over 'private-address'
+ os-data-network:
+ type: string
+ description: |
+ The IP address and netmask of the OpenStack Data network (e.g.,
+ 192.168.0.0/24)
+ .
+ This network will be used for tenant network traffic in overlay
+ networks.
+
diff --git a/hooks/charmhelpers/contrib/hahelpers/cluster.py b/hooks/charmhelpers/contrib/hahelpers/cluster.py
index bf832f7d..dd89f347 100644
--- a/hooks/charmhelpers/contrib/hahelpers/cluster.py
+++ b/hooks/charmhelpers/contrib/hahelpers/cluster.py
@@ -163,13 +163,14 @@ def get_hacluster_config():
return conf
-def canonical_url(configs, vip_setting='vip'):
+def canonical_url(configs, vip_setting='vip', address=None):
'''
Returns the correct HTTP URL to this host given the state of HTTPS
configuration and hacluster.
:configs : OSTemplateRenderer: A config tempating object to inspect for
a complete https context.
+
:vip_setting: str: Setting in charm config that specifies
VIP address.
'''
@@ -179,5 +180,5 @@ def canonical_url(configs, vip_setting='vip'):
if is_clustered():
addr = config_get(vip_setting)
else:
- addr = unit_get('private-address')
+ addr = address or unit_get('private-address')
return '%s://%s' % (scheme, addr)
diff --git a/hooks/charmhelpers/contrib/network/ip.py b/hooks/charmhelpers/contrib/network/ip.py
new file mode 100644
index 00000000..15a6731c
--- /dev/null
+++ b/hooks/charmhelpers/contrib/network/ip.py
@@ -0,0 +1,69 @@
+import sys
+
+from charmhelpers.fetch import apt_install
+from charmhelpers.core.hookenv import (
+ ERROR, log,
+)
+
+try:
+ import netifaces
+except ImportError:
+ apt_install('python-netifaces')
+ import netifaces
+
+try:
+ import netaddr
+except ImportError:
+ apt_install('python-netaddr')
+ import netaddr
+
+
+def _validate_cidr(network):
+ try:
+ netaddr.IPNetwork(network)
+ except (netaddr.core.AddrFormatError, ValueError):
+ raise ValueError("Network (%s) is not in CIDR presentation format" %
+ network)
+
+
+def get_address_in_network(network, fallback=None, fatal=False):
+ """
+ Get an IPv4 address within the network from the host.
+
+ :param network (str): CIDR presentation format. For example,
+ '192.168.1.0/24'.
+ :param fallback (str): If no address is found, return fallback.
+ :param fatal (boolean): If no address is found, fallback is not
+ 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 fallback is not None:
+ return fallback
+ else:
+ if fatal:
+ not_found_error_out()
+
+ _validate_cidr(network)
+ for iface in netifaces.interfaces():
+ addresses = netifaces.ifaddresses(iface)
+ if netifaces.AF_INET in addresses:
+ addr = addresses[netifaces.AF_INET][0]['addr']
+ netmask = addresses[netifaces.AF_INET][0]['netmask']
+ cidr = netaddr.IPNetwork("%s/%s" % (addr, netmask))
+ if cidr in netaddr.IPNetwork(network):
+ return str(cidr.ip)
+
+ if fallback is not None:
+ return fallback
+
+ if fatal:
+ not_found_error_out()
+
+ return None
diff --git a/hooks/charmhelpers/contrib/openstack/context.py b/hooks/charmhelpers/contrib/openstack/context.py
index e0526d89..aea11b34 100644
--- a/hooks/charmhelpers/contrib/openstack/context.py
+++ b/hooks/charmhelpers/contrib/openstack/context.py
@@ -340,10 +340,12 @@ class CephContext(OSContextGenerator):
use_syslog = str(config('use-syslog')).lower()
for rid in relation_ids('ceph'):
for unit in related_units(rid):
- mon_hosts.append(relation_get('private-address', rid=rid,
- unit=unit))
auth = relation_get('auth', rid=rid, unit=unit)
key = relation_get('key', rid=rid, unit=unit)
+ ceph_addr = \
+ relation_get('ceph-public-address', rid=rid, unit=unit) or \
+ relation_get('private-address', rid=rid, unit=unit)
+ mon_hosts.append(ceph_addr)
ctxt = {
'mon_hosts': ' '.join(mon_hosts),
@@ -426,12 +428,13 @@ class ApacheSSLContext(OSContextGenerator):
"""
Generates a context for an apache vhost configuration that configures
HTTPS reverse proxying for one or many endpoints. Generated context
- looks something like:
- {
- 'namespace': 'cinder',
- 'private_address': 'iscsi.mycinderhost.com',
- 'endpoints': [(8776, 8766), (8777, 8767)]
- }
+ looks something like::
+
+ {
+ 'namespace': 'cinder',
+ 'private_address': 'iscsi.mycinderhost.com',
+ 'endpoints': [(8776, 8766), (8777, 8767)]
+ }
The endpoints list consists of a tuples mapping external ports
to internal ports.
@@ -641,7 +644,7 @@ class SubordinateConfigContext(OSContextGenerator):
The subordinate interface allows subordinates to export their
configuration requirements to the principle for multiple config
files and multiple serivces. Ie, a subordinate that has interfaces
- to both glance and nova may export to following yaml blob as json:
+ to both glance and nova may export to following yaml blob as json::
glance:
/etc/glance/glance-api.conf:
@@ -660,7 +663,8 @@ class SubordinateConfigContext(OSContextGenerator):
It is then up to the principle charms to subscribe this context to
the service+config file it is interestd in. Configuration data will
- be available in the template context, in glance's case, as:
+ be available in the template context, in glance's case, as::
+
ctxt = {
... other context ...
'subordinate_config': {
diff --git a/hooks/charmhelpers/contrib/openstack/templating.py b/hooks/charmhelpers/contrib/openstack/templating.py
index 4595778c..f5442712 100644
--- a/hooks/charmhelpers/contrib/openstack/templating.py
+++ b/hooks/charmhelpers/contrib/openstack/templating.py
@@ -30,17 +30,17 @@ def get_loader(templates_dir, os_release):
loading dir.
A charm may also ship a templates dir with this module
- and it will be appended to the bottom of the search list, eg:
- hooks/charmhelpers/contrib/openstack/templates.
+ and it will be appended to the bottom of the search list, eg::
- :param templates_dir: str: Base template directory containing release
- sub-directories.
- :param os_release : str: OpenStack release codename to construct template
- loader.
+ hooks/charmhelpers/contrib/openstack/templates
- :returns : jinja2.ChoiceLoader constructed with a list of
- jinja2.FilesystemLoaders, ordered in descending
- order by OpenStack release.
+ :param templates_dir (str): Base template directory containing release
+ sub-directories.
+ :param os_release (str): OpenStack release codename to construct template
+ loader.
+ :returns: jinja2.ChoiceLoader constructed with a list of
+ jinja2.FilesystemLoaders, ordered in descending
+ order by OpenStack release.
"""
tmpl_dirs = [(rel, os.path.join(templates_dir, rel))
for rel in OPENSTACK_CODENAMES.itervalues()]
@@ -111,7 +111,8 @@ class OSConfigRenderer(object):
and ease the burden of managing config templates across multiple OpenStack
releases.
- Basic usage:
+ Basic usage::
+
# import some common context generates from charmhelpers
from charmhelpers.contrib.openstack import context
@@ -131,21 +132,19 @@ class OSConfigRenderer(object):
# write out all registered configs
configs.write_all()
- Details:
+ **OpenStack Releases and template loading**
- OpenStack Releases and template loading
- ---------------------------------------
When the object is instantiated, it is associated with a specific OS
release. This dictates how the template loader will be constructed.
The constructed loader attempts to load the template from several places
in the following order:
- - from the most recent OS release-specific template dir (if one exists)
- - the base templates_dir
- - a template directory shipped in the charm with this helper file.
+ - from the most recent OS release-specific template dir (if one exists)
+ - the base templates_dir
+ - a template directory shipped in the charm with this helper file.
+ For the example above, '/tmp/templates' contains the following structure::
- For the example above, '/tmp/templates' contains the following structure:
/tmp/templates/nova.conf
/tmp/templates/api-paste.ini
/tmp/templates/grizzly/api-paste.ini
@@ -169,8 +168,8 @@ class OSConfigRenderer(object):
$CHARM/hooks/charmhelpers/contrib/openstack/templates. This allows
us to ship common templates (haproxy, apache) with the helpers.
- Context generators
- ---------------------------------------
+ **Context generators**
+
Context generators are used to generate template contexts during hook
execution. Doing so may require inspecting service relations, charm
config, etc. When registered, a config file is associated with a list
diff --git a/hooks/charmhelpers/contrib/storage/linux/ceph.py b/hooks/charmhelpers/contrib/storage/linux/ceph.py
index 12417410..768438a4 100644
--- a/hooks/charmhelpers/contrib/storage/linux/ceph.py
+++ b/hooks/charmhelpers/contrib/storage/linux/ceph.py
@@ -303,7 +303,7 @@ def ensure_ceph_storage(service, pool, rbd_img, sizemb, mount_point,
blk_device, fstype, system_services=[]):
"""
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,
and the device is formatted and mounted at the given mount_point.
diff --git a/hooks/charmhelpers/core/hookenv.py b/hooks/charmhelpers/core/hookenv.py
index c2e66f66..c9530433 100644
--- a/hooks/charmhelpers/core/hookenv.py
+++ b/hooks/charmhelpers/core/hookenv.py
@@ -25,7 +25,7 @@ cache = {}
def cached(func):
"""Cache return values for multiple executions of func + args
- For example:
+ For example::
@cached
def unit_get(attribute):
@@ -445,18 +445,19 @@ class UnregisteredHookError(Exception):
class Hooks(object):
"""A convenient handler for hook functions.
- Example:
+ Example::
+
hooks = Hooks()
# register a hook, taking its name from the function name
@hooks.hook()
def install():
- ...
+ pass # your code here
# register a hook, providing a custom hook name
@hooks.hook("config-changed")
def config_changed():
- ...
+ pass # your code here
if __name__ == "__main__":
# execute a hook based on the name the program is called by
diff --git a/hooks/charmhelpers/core/host.py b/hooks/charmhelpers/core/host.py
index 59f8facc..8b617a42 100644
--- a/hooks/charmhelpers/core/host.py
+++ b/hooks/charmhelpers/core/host.py
@@ -211,13 +211,13 @@ def file_hash(path):
def restart_on_change(restart_map, stopstart=False):
"""Restart services based on configuration files changing
- This function is used a decorator, for example
+ This function is used a decorator, for example::
@restart_on_change({
'/etc/ceph/ceph.conf': [ 'cinder-api', 'cinder-volume' ]
})
def ceph_client_changed():
- ...
+ pass # your code here
In this example, the cinder-api and cinder-volume services
would be restarted if /etc/ceph/ceph.conf is changed by the
@@ -313,9 +313,11 @@ def get_nic_hwaddr(nic):
def cmp_pkgrevno(package, revno, pkgcache=None):
'''Compare supplied revno with the revno of the installed package
- 1 => Installed revno is greater than supplied arg
- 0 => Installed revno is the same as supplied arg
- -1 => Installed revno is less than supplied arg
+
+ * 1 => Installed revno is greater than supplied arg
+ * 0 => Installed revno is the same as supplied arg
+ * -1 => Installed revno is less than supplied arg
+
'''
import apt_pkg
if not pkgcache:
diff --git a/hooks/charmhelpers/fetch/__init__.py b/hooks/charmhelpers/fetch/__init__.py
index b5cb48ef..5be512ce 100644
--- a/hooks/charmhelpers/fetch/__init__.py
+++ b/hooks/charmhelpers/fetch/__init__.py
@@ -235,31 +235,39 @@ def configure_sources(update=False,
sources_var='install_sources',
keys_var='install_keys'):
"""
- Configure multiple sources from charm configuration
+ Configure multiple sources from charm configuration.
+
+ The lists are encoded as yaml fragments in the configuration.
+ The frament needs to be included as a string.
Example config:
- install_sources:
+ install_sources: |
- "ppa:foo"
- "http://example.com/repo precise main"
- install_keys:
+ install_keys: |
- null
- "a1b2c3d4"
Note that 'null' (a.k.a. None) should not be quoted.
"""
- sources = safe_load(config(sources_var))
- keys = config(keys_var)
- if keys is not None:
- keys = safe_load(keys)
- if isinstance(sources, basestring) and (
- keys is None or isinstance(keys, basestring)):
- add_source(sources, keys)
+ sources = safe_load((config(sources_var) or '').strip()) or []
+ keys = safe_load((config(keys_var) or '').strip()) or None
+
+ if isinstance(sources, basestring):
+ sources = [sources]
+
+ if keys is None:
+ for source in sources:
+ add_source(source, None)
else:
- if not len(sources) == len(keys):
- msg = 'Install sources and keys lists are different lengths'
- raise SourceConfigError(msg)
- for src_num in range(len(sources)):
- add_source(sources[src_num], keys[src_num])
+ if isinstance(keys, basestring):
+ keys = [keys]
+
+ if len(sources) != len(keys):
+ raise SourceConfigError(
+ 'Install sources and keys lists are different lengths')
+ for source, key in zip(sources, keys):
+ add_source(source, key)
if update:
apt_update(fatal=True)
diff --git a/hooks/neutron_ovs_context.py b/hooks/neutron_ovs_context.py
index d8b373a1..cbfa9035 100644
--- a/hooks/neutron_ovs_context.py
+++ b/hooks/neutron_ovs_context.py
@@ -10,6 +10,8 @@ from charmhelpers.contrib.openstack import context
from charmhelpers.core.host import service_running, service_start
from charmhelpers.contrib.network.ovs import add_bridge
from charmhelpers.contrib.openstack.utils import get_host_ip
+from charmhelpers.contrib.network.ip import get_address_in_network
+
OVS_BRIDGE = 'br-int'
@@ -59,7 +61,9 @@ class OVSPluginContext(context.NeutronContext):
self._ensure_bridge()
conf = config()
- ovs_ctxt['local_ip'] = get_host_ip(unit_get('private-address'))
+ ovs_ctxt['local_ip'] = \
+ get_address_in_network(config('os-data-network'),
+ get_host_ip(unit_get('private-address')))
ovs_ctxt['neutron_security_groups'] = self.neutron_security_groups
# TODO: We need to sort out the syslog and debug/verbose options as a
# general context helper