diff options
| author | Ryan Harper <ryan.harper@canonical.com> | 2016-03-23 11:05:22 -0500 | 
|---|---|---|
| committer | Ryan Harper <ryan.harper@canonical.com> | 2016-03-23 11:05:22 -0500 | 
| commit | 2b85dabb802766e0b3b1949d744c8860c0cb838a (patch) | |
| tree | b35a267b5e18822613af225e7af98295674e8ef7 /cloudinit/sources/DataSourceConfigDrive.py | |
| parent | 7f0871dc5b141ff4bf601b6d96021eba8a3bb0ec (diff) | |
| download | vyos-cloud-init-2b85dabb802766e0b3b1949d744c8860c0cb838a.tar.gz vyos-cloud-init-2b85dabb802766e0b3b1949d744c8860c0cb838a.zip | |
configdata: parse and convert openstack network_data json to network_config
Diffstat (limited to 'cloudinit/sources/DataSourceConfigDrive.py')
| -rw-r--r-- | cloudinit/sources/DataSourceConfigDrive.py | 137 | 
1 files changed, 137 insertions, 0 deletions
| diff --git a/cloudinit/sources/DataSourceConfigDrive.py b/cloudinit/sources/DataSourceConfigDrive.py index 6fc9e05b..d84fab54 100644 --- a/cloudinit/sources/DataSourceConfigDrive.py +++ b/cloudinit/sources/DataSourceConfigDrive.py @@ -18,6 +18,7 @@  #    You should have received a copy of the GNU General Public License  #    along with this program.  If not, see <http://www.gnu.org/licenses/>. +import copy  import os  from cloudinit import log as logging @@ -50,6 +51,8 @@ class DataSourceConfigDrive(openstack.SourceMixin, sources.DataSource):          self.seed_dir = os.path.join(paths.seed_dir, 'config_drive')          self.version = None          self.ec2_metadata = None +        self._network_config = None +        self.network_json = None          self.files = {}      def __str__(self): @@ -144,12 +147,27 @@ class DataSourceConfigDrive(openstack.SourceMixin, sources.DataSource):              LOG.warn("Invalid content in vendor-data: %s", e)              self.vendordata_raw = None +        nd = results.get('networkdata') +        self.networkdata_pure = nd +        try: +            self.network_json = openstack.convert_networkdata_json(nd) +        except ValueError as e: +            LOG.warn("Invalid content in network-data: %s", e) +            self.network_json = None + +        if self.network_json: +            self._network_config = convert_network_data(self.network_json) +          return True      def check_instance_id(self):          # quickly (local check only) if self.instance_id is still valid          return sources.instance_id_matches_system_uuid(self.get_instance_id()) +    @property +    def network_config(self): +        return self._network_config +  class DataSourceConfigDriveNet(DataSourceConfigDrive):      def __init__(self, sys_cfg, distro, paths): @@ -287,3 +305,122 @@ datasources = [  # Return a list of data sources that match this set of dependencies  def get_datasource_list(depends):      return sources.list_from_depends(depends, datasources) + + +# Convert OpenStack ConfigDrive NetworkData json to network_config yaml +def convert_network_data(network_json=None): +    """Return a dictionary of network_config by parsing provided +       OpenStack ConfigDrive NetworkData json format + +    OpenStack network_data.json provides a 3 element dictionary +      - "links" (links are network devices, physical or virtual) +      - "networks" (networks are ip network configurations for one or more +                    links) +      -  services (non-ip services, like dns) + +    networks and links are combined via network items referencing specific +    links via a 'link_id' which maps to a links 'id' field. + +    To convert this format to network_config yaml, we first iterate over the +    links and then walk the network list to determine if any of the networks +    utilize the current link; if so we generate a subnet entry for the device + +    We also need to map network_data.json fields to network_config fields. For +    example, the network_data links 'id' field is equivalent to network_config +    'name' field for devices.  We apply more of this mapping to the various +    link types that we encounter. + +    There are additional fields that are populated in the network_data.json +    from OpenStack that are not relevant to network_config yaml, so we +    enumerate a dictionary of valid keys for network_yaml and apply filtering +    to drop these superflous keys from the network_config yaml. +    """ +    if network_json is None: +        return None + +    # dict of network_config key for filtering network_json +    valid_keys = { +        'physical': [ +            'name', +            'type', +            'mac_address', +            'subnets', +            'params', +        ], +        'subnet': [ +            'type', +            'address', +            'netmask', +            'broadcast', +            'metric', +            'gateway', +            'pointopoint', +            'mtu', +            'scope', +            'dns_nameservers', +            'dns_search', +            'routes', +        ], +    } + +    links = network_json.get('links', []) +    networks = network_json.get('networks', []) +    services = network_json.get('services', []) + +    config = [] +    for link in links: +        subnets = [] +        cfg = {k: v for k, v in link.items() +               if k in valid_keys['physical']} +        cfg.update({'name': link['id']}) +        for network in [net for net in networks +                        if net['link'] == link['id']]: +            subnet = {k: v for k, v in network.items() +                      if k in valid_keys['subnet']} +            if 'dhcp' in network['type']: +                t = 'dhcp6' if network['type'].startswith('ipv6') else 'dhcp4' +                subnet.update({ +                    'type': t, +                }) +            else: +                subnet.update({ +                   'type': 'static', +                   'address': network.get('ip_address'), +                }) +            subnets.append(subnet) +        cfg.update({'subnets': subnets}) +        if link['type'] in ['ethernet', 'vif', 'ovs']: +            cfg.update({ +                'type': 'physical', +                'mac_address': link['ethernet_mac_address']}) +        elif link['type'] in ['bond']: +            params = {} +            for k, v in link.items(): +                if k == 'bond_links': +                    continue +                elif k.startswith('bond'): +                    params.update({k: v}) +            cfg.update({ +                'bond_interfaces': copy.deepcopy(link['bond_links']), +                'params': params, +            }) +        elif link['type'] in ['vlan']: +            cfg.update({ +                'name': "%s.%s" % (link['vlan_link'], +                                   link['vlan_id']), +                'vlan_link': link['vlan_link'], +                'vlan_id': link['vlan_id'], +                'mac_address': link['vlan_mac_address'], +            }) +        else: +            raise ValueError( +                    'Unknown network_data link type: %s' % link['type']) + +        config.append(cfg) + +    for service in services: +        cfg = service +        cfg.update({'type': 'nameserver'}) +        config.append(cfg) + +    return {'version': 1, 'config': config} | 
