summaryrefslogtreecommitdiff
path: root/cloudinit/sources/DataSourceAltCloud.py
diff options
context:
space:
mode:
Diffstat (limited to 'cloudinit/sources/DataSourceAltCloud.py')
-rw-r--r--cloudinit/sources/DataSourceAltCloud.py211
1 files changed, 89 insertions, 122 deletions
diff --git a/cloudinit/sources/DataSourceAltCloud.py b/cloudinit/sources/DataSourceAltCloud.py
index 27a67e6c..0f4cc8e0 100644
--- a/cloudinit/sources/DataSourceAltCloud.py
+++ b/cloudinit/sources/DataSourceAltCloud.py
@@ -4,9 +4,6 @@
# Copyright (C) 2012 Hewlett-Packard Development Company, L.P.
# Copyright (C) 2012 Yahoo! Inc.
#
-# Author: Scott Moser <scott.moser@canonical.com>
-# Author: Juerg Hafliger <juerg.haefliger@hp.com>
-# Author: Joshua Harlow <harlowja@yahoo-inc.com>
# Author: Joe VLcek <JVLcek@RedHat.com>
#
# This program is free software: you can redistribute it and/or modify
@@ -21,9 +18,10 @@
# 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 time
+import errno
import os
import os.path
+import time
from cloudinit import log as logging
from cloudinit import sources
@@ -34,19 +32,10 @@ LOG = logging.getLogger(__name__)
# Needed file paths
CLOUD_INFO_FILE = '/etc/sysconfig/cloud-info'
-MEDIA_DIR = '/media/userdata'
-
-# Deltacloud file name contains deltacloud. Those not using
-# Deltacloud but instead instrumenting the injection, could
-# drop deltacloud from the file name.
-DELTACLOUD_USER_DATA_FILE = MEDIA_DIR + '/deltacloud-user-data.txt'
-USER_DATA_FILE = MEDIA_DIR + '/user-data.txt'
# Shell command lists
CMD_DMI_SYSTEM = ['/usr/sbin/dmidecode', '--string', 'system-product-name']
CMD_PROBE_FLOPPY = ['/sbin/modprobe', 'floppy']
-CMD_MNT_FLOPPY = ['/bin/mount', '/dev/fd0', MEDIA_DIR]
-CMD_MNT_CDROM = ['/bin/mount', '/dev/cdrom', MEDIA_DIR]
# Retry times and sleep secs between each try
RETRY_TIMES = 3
@@ -60,18 +49,50 @@ META_DATA_NOT_SUPPORTED = {
}
+def read_user_data_callback(mount_dir):
+ '''
+ Description:
+ This callback will be applied by util.mount_cb() on the mounted
+ file.
+
+ Deltacloud file name contains deltacloud. Those not using
+ Deltacloud but instead instrumenting the injection, could
+ drop deltacloud from the file name.
+
+ Input:
+ mount_dir - Mount directory
+
+ Returns:
+ User Data
+
+ '''
+
+ deltacloud_user_data_file = mount_dir + '/deltacloud-user-data.txt'
+ user_data_file = mount_dir + '/user-data.txt'
+
+ # First try deltacloud_user_data_file. On failure try user_data_file.
+ try:
+ with open(deltacloud_user_data_file, 'r') as user_data_f:
+ user_data = user_data_f.read().strip()
+ except:
+ try:
+ with open(user_data_file, 'r') as user_data_f:
+ user_data = user_data_f.read().strip()
+ except:
+ util.logexc(LOG, ('Failed accessing user data file.'))
+ return None
+
+ return user_data
+
+
class DataSourceAltCloud(sources.DataSource):
def __init__(self, sys_cfg, distro, paths):
sources.DataSource.__init__(self, sys_cfg, distro, paths)
- self.dsmode = 'local'
self.seed = None
- self.cmdline_id = "ds=nocloud"
- self.seed_dir = os.path.join(paths.seed_dir, 'nocloud')
self.supported_seed_starts = ("/", "file://")
def __str__(self):
- mstr = "%s [seed=%s][dsmode=%s]" % (util.obj_name(self),
- self.seed, self.dsmode)
+ mstr = "%s [seed=%s]" % (util.obj_name(self), self.seed)
return mstr
def get_cloud_type(self):
@@ -117,6 +138,7 @@ class DataSourceAltCloud(sources.DataSource):
'''
Description:
User Data is passed to the launching instance which
+ is used to perform instance configuration.
Cloud providers expose the user data differently.
It is necessary to determine which cloud provider
@@ -172,15 +194,19 @@ class DataSourceAltCloud(sources.DataSource):
RHEVM specific userdata read
If on RHEV-M the user data will be contained on the
- floppy device in file <USER_DATA_FILE>
+ floppy device in file <user_data_file>
To access it:
modprobe floppy
- mkdir <MEDIA_DIR>
- mount /dev/fd0 <MEDIA_DIR>
- mount /dev/fd0 <MEDIA_DIR> # NOTE: -> /dev/
- read <MEDIA_DIR>/<USER_DATA_FILE>
+
+ Leverage util.mount_cb to:
+ mkdir <tmp mount dir>
+ mount /dev/fd0 <tmp mount dir>
+ The call back passed to util.mount_cb will do:
+ read <tmp mount dir>/<user_data_file>
'''
+ return_str = None
+
# modprobe floppy
try:
cmd = CMD_PROBE_FLOPPY
@@ -195,118 +221,59 @@ class DataSourceAltCloud(sources.DataSource):
(' '.join(cmd), _err.message)))
return False
- # mkdir <MEDIA_DIR> dir just in case it isn't already.
- try:
- os.makedirs(MEDIA_DIR)
- except OSError, (_err, strerror):
- if _err is not 17:
- LOG.debug(('makedirs(<MEDIA_DIR>) failed: %s \nError: %s') % \
- (_err, strerror))
- return False
-
- # mount /dev/fd0 <MEDIA_DIR>
- try:
- cmd = CMD_MNT_FLOPPY
- (cmd_out, _err) = util.subp(cmd)
- LOG.debug(('Command: %s\nOutput%s') % (' '.join(cmd), cmd_out))
- except ProcessExecutionError, _err:
- # Ignore failure: already mounted
- if 'ALREADY MOUNTED' not in str(_err.message).upper():
- util.logexc(LOG, (('Failed command: %s\n%s') % \
- (' '.join(cmd), _err.message)))
- return False
- except OSError, _err:
- util.logexc(LOG, (('Failed command: %s\n%s') % \
- (' '.join(cmd), _err.message)))
- return False
-
- # This could be done using "with open()" but that's not available
- # in Python 2.4 as used on RHEL5
- # First try DELTACLOUD_USER_DATA_FILE. If that fails then try
- # USER_DATA_FILE.
+ floppy_dev = '/dev/fd0'
try:
- user_data_file = open(DELTACLOUD_USER_DATA_FILE, 'r')
- user_data = user_data_file.read().strip()
- user_data_file.close()
- except:
- try:
- user_data_file = open(USER_DATA_FILE, 'r')
- user_data = user_data_file.read().strip()
- user_data_file.close()
- except:
- util.logexc(LOG, ('Failed accessing RHEVm user data file.'))
- try:
- user_data_file.close()
- except:
- pass
- return False
-
- self.userdata_raw = user_data
+ return_str = util.mount_cb(floppy_dev, read_user_data_callback)
+ except OSError as err:
+ if err.errno != errno.ENOENT:
+ raise
+ except util.MountFailedError:
+ util.logexc(LOG, ("Failed to mount %s"
+ " when looking for user data"), floppy_dev)
+
+ self.userdata_raw = return_str
self.metadata = META_DATA_NOT_SUPPORTED
- return True
+ if return_str:
+ return True
+ else:
+ return False
def user_data_vsphere(self):
'''
- VSphere specific userdata read
+ vSphere specific userdata read
If on vSphere the user data will be contained on the
- floppy device in file <USER_DATA_FILE>
+ cdrom device in file <user_data_file>
To access it:
- mkdir <MEDIA_DIR> dir just in case it isn't already.
- mount /dev/cdrom <MEDIA_DIR> # NOTE: -> /dev/cdrom
- read <MEDIA_DIR>/<USER_DATA_FILE>
+ Leverage util.mount_cb to:
+ mkdir <tmp mount dir>
+ mount /dev/fd0 <tmp mount dir>
+ The call back passed to util.mount_cb will do:
+ read <tmp mount dir>/<user_data_file>
'''
- # mkdir <MEDIA_DIR> dir just in case it isn't already.
- try:
- os.makedirs(MEDIA_DIR)
- except OSError, (_err, strerror):
- if _err is not 17:
- LOG.debug(('makedirs(<MEDIA_DIR>) failed: %s \nError: %s') % \
- (_err, strerror))
- return False
-
- # mount /dev/cdrom <MEDIA_DIR>
- try:
- cmd = CMD_MNT_CDROM
- (cmd_out, _err) = util.subp(cmd)
- LOG.debug(('Command: %s\nOutput%s') % (' '.join(cmd), cmd_out))
- except ProcessExecutionError, _err:
- # Ignore failure: already mounted
- if 'ALREADY MOUNTED' not in str(_err.message).upper():
- LOG.debug(('Failed command: %s\n%s') % \
- (' '.join(cmd), _err.message))
- return False
- except OSError, _err:
- LOG.debug(('Failed command: %s\n%s') % \
- (' '.join(cmd), _err.message))
- return False
-
- # This could be done using "with open()" but that's not available
- # in Python 2.4 as used on RHEL5
- # First try DELTACLOUD_USER_DATA_FILE. If that fails then try
- # USER_DATA_FILE.
- try:
- user_data_file = open(DELTACLOUD_USER_DATA_FILE, 'r')
- user_data = user_data_file.read().strip()
- user_data_file.close()
- except:
+ return_str = None
+ cdrom_list = util.find_devs_with('LABEL=CDROM')
+ for cdrom_dev in cdrom_list:
try:
- user_data_file = open(USER_DATA_FILE, 'r')
- user_data = user_data_file.read().strip()
- user_data_file.close()
- except:
- LOG.debug('Failed accessing vSphere user data file.')
- try:
- user_data_file.close()
- except:
- pass
- return False
-
- self.userdata_raw = user_data
+ return_str = util.mount_cb(cdrom_dev, read_user_data_callback)
+ if return_str:
+ break
+ except OSError as err:
+ if err.errno != errno.ENOENT:
+ raise
+ except util.MountFailedError:
+ util.logexc(LOG, ("Failed to mount %s"
+ " when looking for user data"), cdrom_dev)
+
+ self.userdata_raw = return_str
self.metadata = META_DATA_NOT_SUPPORTED
- return True
+
+ if return_str:
+ return True
+ else:
+ return False
# Used to match classes to dependencies
datasources = [