summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--cloudinit/sources/DataSourceConfigDrive.py102
-rw-r--r--tests/unittests/test_datasource/test_configdrive.py194
2 files changed, 286 insertions, 10 deletions
diff --git a/cloudinit/sources/DataSourceConfigDrive.py b/cloudinit/sources/DataSourceConfigDrive.py
index 850b281c..677483d0 100644
--- a/cloudinit/sources/DataSourceConfigDrive.py
+++ b/cloudinit/sources/DataSourceConfigDrive.py
@@ -30,7 +30,7 @@ LOG = logging.getLogger(__name__)
# Various defaults/constants...
DEFAULT_IID = "iid-dsconfigdrive"
DEFAULT_MODE = 'pass'
-CFG_DRIVE_FILES = [
+CFG_DRIVE_FILES_V1 = [
"etc/network/interfaces",
"root/.ssh/authorized_keys",
"meta.js",
@@ -49,9 +49,11 @@ class DataSourceConfigDrive(sources.DataSource):
self.cfg = {}
self.dsmode = 'local'
self.seed_dir = os.path.join(paths.seed_dir, 'config_drive')
+ self.version = None
def __str__(self):
- mstr = "%s [%s]" % (util.obj_name(self), self.dsmode)
+ mstr = "%s [%s,ver=%s]" % (util.obj_name(self), self.dsmode,
+ self.version)
mstr += "[seed=%s]" % (self.seed)
return mstr
@@ -62,17 +64,29 @@ class DataSourceConfigDrive(sources.DataSource):
if os.path.isdir(self.seed_dir):
try:
- (md, ud) = read_config_drive_dir(self.seed_dir)
+ (md, ud, files, ver) = read_config_drive_dir(self.seed_dir)
found = self.seed_dir
except NonConfigDriveDir:
util.logexc(LOG, "Failed reading config drive from %s",
self.seed_dir)
if not found:
+ fslist = util.find_devs_with("TYPE=vfat")
+ fslist.extend(util.find_devs_with("TYPE=iso9660"))
+
+ label_list = util.find_devs_with("LABEL=config-2")
+ devlist = list(set(fslist) & set(label_list))
+
dev = find_cfg_drive_device()
- if dev:
+ if dev not in devlist:
+ devlist.append(dev)
+
+ devlist.sort(reverse=True)
+
+ for dev in devlist:
try:
- (md, ud) = util.mount_cb(dev, read_config_drive_dir)
+ (md, ud, files, ver) = util.mount_cb(dev, read_config_drive_dir)
found = dev
+ break
except (NonConfigDriveDir, util.MountFailedError):
pass
@@ -94,6 +108,7 @@ class DataSourceConfigDrive(sources.DataSource):
self.seed = found
self.metadata = md
self.userdata_raw = ud
+ self.version = ver
if md['dsmode'] == self.dsmode:
return True
@@ -122,6 +137,9 @@ class DataSourceConfigDriveNet(DataSourceConfigDrive):
class NonConfigDriveDir(Exception):
pass
+class BrokenConfigDriveDir(Exception):
+ pass
+
def find_cfg_drive_device():
"""Get the config drive device. Return a string like '/dev/vdb'
@@ -154,17 +172,80 @@ def find_cfg_drive_device():
def read_config_drive_dir(source_dir):
+ last_e = NonConfigDriveDir("Not found")
+ for finder in (read_config_drive_dir_v2, read_config_drive_dir_v1):
+ try:
+ data = finder(source_dir)
+ return data
+ except NonConfigDriveDir as exc:
+ last_e = exc
+ raise last_e
+
+def read_config_drive_dir_v2(source_dir, version="latest"):
+ datafiles = (
+ ('metadata',
+ "openstack/%s/meta_data.json" % version, True, json.loads),
+ ('userdata', "openstack/%s/user_data" % version, False, None),
+ ('ec2-metadata', "ec2/latest/metadata.json", False, json.loads),
+ )
+
+ results = {}
+ for (name, path, required, process) in datafiles:
+ fpath = os.path.join(source_dir, path)
+ data = None
+ found = False
+ if os.path.isfile(fpath):
+ try:
+ with open(fpath) as fp:
+ data = fp.read()
+ except Exception as exc:
+ raise BrokenConfigDriveDir("failed to read: %s" % fpath)
+ found = True
+ elif required:
+ raise BrokenConfigDriveDir("missing mandatory %s" % fpath)
+
+ if found and process:
+ try:
+ data = process(data)
+ except Exception as exc:
+ raise BrokenConfigDriveDir("failed to process: %s" % fpath)
+
+ if found:
+ results[name] = data
+
+ def read_content_path(item):
+ # do not use os.path.join here, as content_path starts with /
+ cpath = os.path.sep.join((source_dir, "openstack",
+ "./%s" % item['content_path']))
+ with open(cpath) as fp:
+ return(fp.read())
+
+ files = {}
+ try:
+ for item in results['metadata'].get('files',{}):
+ files[item['path']] = read_content_path(item)
+
+ item = results['metadata'].get("network_config", None)
+ if item:
+ results['network_config'] = read_content_path(item)
+ except Exception as exc:
+ raise BrokenConfigDriveDir("failed to read file %s: %s" % (item, exc))
+
+ results['files'] = files
+ results['cfgdrive_ver'] = 2
+ return results
+
+def read_config_drive_dir_v1(source_dir):
"""
- read_config_drive_dir(source_dir):
- read source_dir, and return a tuple with metadata dict and user-data
- string populated. If not a valid dir, raise a NonConfigDriveDir
+ read source_dir, and return a tuple with metadata dict, user-data,
+ files and version (1). If not a valid dir, raise a NonConfigDriveDir
"""
# TODO(harlowja): fix this for other operating systems...
# Ie: this is where https://fedorahosted.org/netcf/ or similar should
# be hooked in... (or could be)
found = {}
- for af in CFG_DRIVE_FILES:
+ for af in CFG_DRIVE_FILES_V1:
fn = os.path.join(source_dir, af)
if os.path.isfile(fn):
found[af] = fn
@@ -211,7 +292,8 @@ def read_config_drive_dir(source_dir):
if 'user-data' in meta_js:
ud = meta_js['user-data']
- return (md, ud)
+ # metadata, user-data, 'files', 1
+ return {'metadata': md, 'userdata': ud, 'files': [], 'cfgdrive_ver': 1}
# Used to match classes to dependencies
diff --git a/tests/unittests/test_datasource/test_configdrive.py b/tests/unittests/test_datasource/test_configdrive.py
new file mode 100644
index 00000000..f73200e5
--- /dev/null
+++ b/tests/unittests/test_datasource/test_configdrive.py
@@ -0,0 +1,194 @@
+from copy import copy
+import json
+import os
+import os.path
+
+from cloudinit.sources import DataSourceConfigDrive
+from cloudinit import url_helper
+from mocker import MockerTestCase
+
+
+PUBKEY = u'ssh-rsa AAAAB3NzaC1....sIkJhq8wdX+4I3A4cYbYP ubuntu@server-460\n'
+EC2_META = {
+ 'ami-id': 'ami-00000001',
+ 'ami-launch-index': 0,
+ 'ami-manifest-path': 'FIXME',
+ 'block-device-mapping': {
+ 'ami': 'sda1',
+ 'ephemeral0': 'sda2',
+ 'root': '/dev/sda1',
+ 'swap': 'sda3'},
+ 'hostname': 'sm-foo-test.novalocal',
+ 'instance-action': 'none',
+ 'instance-id': 'i-00000001',
+ 'instance-type': 'm1.tiny',
+ 'local-hostname': 'sm-foo-test.novalocal',
+ 'local-ipv4': None,
+ 'placement': {'availability-zone': 'nova'},
+ 'public-hostname': 'sm-foo-test.novalocal',
+ 'public-ipv4': '',
+ 'public-keys': {'0': {'openssh-key': PUBKEY}},
+ 'reservation-id': 'r-iru5qm4m',
+ 'security-groups': ['default']
+}
+USER_DATA = '#!/bin/sh\necho This is user data\n'
+OSTACK_META = {
+ 'availability_zone': 'nova',
+ 'files': [{'content_path': '/content/0000', 'path': '/etc/foo.cfg'},
+ {'content_path': '/content/0001', 'path': '/etc/bar/bar.cfg'}],
+ 'hostname': 'sm-foo-test.novalocal',
+ 'meta': {'dsmode': 'local', 'my-meta': 'my-value'},
+ 'name': 'sm-foo-test',
+ 'public_keys': {'mykey': PUBKEY},
+ 'uuid': 'b0fa911b-69d4-4476-bbe2-1c92bff6535c'}
+
+CONTENT_0 = 'This is contents of /etc/foo.cfg\n'
+CONTENT_1 = '# this is /etc/bar/bar.cfg\n'
+
+CFG_DRIVE_FILES_V2 = {
+ 'ec2/2009-04-04/meta-data.json': json.dumps(EC2_META),
+ 'ec2/2009-04-04/user-data': USER_DATA,
+ 'ec2/latest/meta-data.json': json.dumps(EC2_META),
+ 'ec2/latest/user-data': USER_DATA,
+ 'openstack/2012-08-10/meta_data.json': json.dumps(OSTACK_META),
+ 'openstack/2012-08-10/user_data': USER_DATA,
+ 'openstack/content/0000': CONTENT_0,
+ 'openstack/content/0001': CONTENT_1,
+ 'openstack/latest/meta_data.json': json.dumps(OSTACK_META),
+ 'openstack/latest/user_data': USER_DATA}
+
+
+class TestConfigDriveDataSource(MockerTestCase):
+
+ def setUp(self):
+ super(TestConfigDriveDataSource, self).setUp()
+ # Make a temp directoy for tests to use.
+ self.tmp = self.makeDir()
+
+ def test_dir_valid(self):
+ """Verify a dir is read as such."""
+
+ my_d = os.path.join(self.tmp, "valid")
+ populate_dir(my_d, CFG_DRIVE_FILES_V2)
+
+ found = DataSourceConfigDrive.read_config_drive_dir(my_d)
+
+ self.assertEqual(USER_DATA, found['userdata'])
+ self.assertEqual(OSTACK_META, found['metadata'])
+ self.assertEqual(found['files']['/etc/foo.cfg'], CONTENT_0)
+ self.assertEqual(found['files']['/etc/bar/bar.cfg'], CONTENT_1)
+
+ def test_seed_dir_valid_extra(self):
+ """Verify extra files do not affect datasource validity."""
+
+ my_d = os.path.join(self.tmp, "valid_extra")
+ data = copy(CFG_DRIVE_FILES_V2)
+ data["myfoofile"] = "myfoocontent"
+
+ populate_dir(my_d, data)
+
+# (userdata, metadata) = DataSourceMAAS.read_maas_seed_dir(my_d)
+#
+# self.assertEqual(userdata, data['user-data'])
+# for key in ('instance-id', 'local-hostname'):
+# self.assertEqual(data[key], metadata[key])
+#
+# # additional files should not just appear as keys in metadata atm
+# self.assertFalse(('foo' in metadata))
+#
+# def test_seed_dir_invalid(self):
+# """Verify that invalid seed_dir raises MAASSeedDirMalformed."""
+#
+# valid = {'instance-id': 'i-instanceid',
+# 'local-hostname': 'test-hostname', 'user-data': ''}
+#
+# my_based = os.path.join(self.tmp, "valid_extra")
+#
+# # missing 'userdata' file
+# my_d = "%s-01" % my_based
+# invalid_data = copy(valid)
+# del invalid_data['local-hostname']
+# populate_dir(my_d, invalid_data)
+# self.assertRaises(DataSourceMAAS.MAASSeedDirMalformed,
+# DataSourceMAAS.read_maas_seed_dir, my_d)
+#
+# # missing 'instance-id'
+# my_d = "%s-02" % my_based
+# invalid_data = copy(valid)
+# del invalid_data['instance-id']
+# populate_dir(my_d, invalid_data)
+# self.assertRaises(DataSourceMAAS.MAASSeedDirMalformed,
+# DataSourceMAAS.read_maas_seed_dir, my_d)
+#
+# def test_seed_dir_none(self):
+# """Verify that empty seed_dir raises MAASSeedDirNone."""
+#
+# my_d = os.path.join(self.tmp, "valid_empty")
+# self.assertRaises(DataSourceMAAS.MAASSeedDirNone,
+# DataSourceMAAS.read_maas_seed_dir, my_d)
+#
+# def test_seed_dir_missing(self):
+# """Verify that missing seed_dir raises MAASSeedDirNone."""
+# self.assertRaises(DataSourceMAAS.MAASSeedDirNone,
+# DataSourceMAAS.read_maas_seed_dir,
+# os.path.join(self.tmp, "nonexistantdirectory"))
+#
+# def test_seed_url_valid(self):
+# """Verify that valid seed_url is read as such."""
+# valid = {'meta-data/instance-id': 'i-instanceid',
+# 'meta-data/local-hostname': 'test-hostname',
+# 'meta-data/public-keys': 'test-hostname',
+# 'user-data': 'foodata'}
+# valid_order = [
+# 'meta-data/local-hostname',
+# 'meta-data/instance-id',
+# 'meta-data/public-keys',
+# 'user-data',
+# ]
+# my_seed = "http://example.com/xmeta"
+# my_ver = "1999-99-99"
+# my_headers = {'header1': 'value1', 'header2': 'value2'}
+#
+# def my_headers_cb(url):
+# return my_headers
+#
+# mock_request = self.mocker.replace(url_helper.readurl,
+# passthrough=False)
+#
+# for key in valid_order:
+# url = "%s/%s/%s" % (my_seed, my_ver, key)
+# mock_request(url, headers=my_headers, timeout=None)
+# resp = valid.get(key)
+# self.mocker.result(url_helper.UrlResponse(200, resp))
+# self.mocker.replay()
+#
+# (userdata, metadata) = DataSourceMAAS.read_maas_seed_url(my_seed,
+# header_cb=my_headers_cb, version=my_ver)
+#
+# self.assertEqual("foodata", userdata)
+# self.assertEqual(metadata['instance-id'],
+# valid['meta-data/instance-id'])
+# self.assertEqual(metadata['local-hostname'],
+# valid['meta-data/local-hostname'])
+#
+# def test_seed_url_invalid(self):
+# """Verify that invalid seed_url raises MAASSeedDirMalformed."""
+# pass
+#
+# def test_seed_url_missing(self):
+# """Verify seed_url with no found entries raises MAASSeedDirNone."""
+# pass
+
+
+def populate_dir(seed_dir, files):
+ os.mkdir(seed_dir)
+ for (name, content) in files.iteritems():
+ path = os.path.join(seed_dir, name)
+ dirname = os.path.dirname(path)
+ if not os.path.isdir(dirname):
+ os.makedirs(dirname)
+ with open(path, "w") as fp:
+ fp.write(content)
+ fp.close()
+
+# vi: ts=4 expandtab