summaryrefslogtreecommitdiff
path: root/cloudinit
diff options
context:
space:
mode:
Diffstat (limited to 'cloudinit')
-rw-r--r--cloudinit/settings.py1
-rw-r--r--cloudinit/sources/DataSourceAltCloud.py302
2 files changed, 303 insertions, 0 deletions
diff --git a/cloudinit/settings.py b/cloudinit/settings.py
index 2083cf60..28340e29 100644
--- a/cloudinit/settings.py
+++ b/cloudinit/settings.py
@@ -29,6 +29,7 @@ CLOUD_CONFIG = '/etc/cloud/cloud.cfg'
# What u get if no config is provided
CFG_BUILTIN = {
'datasource_list': [
+ 'AltCloud',
'NoCloud',
'ConfigDrive',
'OVF',
diff --git a/cloudinit/sources/DataSourceAltCloud.py b/cloudinit/sources/DataSourceAltCloud.py
new file mode 100644
index 00000000..121a3c48
--- /dev/null
+++ b/cloudinit/sources/DataSourceAltCloud.py
@@ -0,0 +1,302 @@
+# vi: ts=4 expandtab
+#
+# Copyright (C) 2009-2010 Canonical Ltd.
+# 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>
+#
+# This program is free software: you can redistribute it and/or modify
+# it under the terms of the GNU General Public License version 3, as
+# published by the Free Software Foundation.
+#
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU General Public License for more details.
+#
+# 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 errno
+import os
+import os.path
+import subprocess
+
+from cloudinit import log as logging
+from cloudinit import sources
+from cloudinit import util
+from cloudinit.util import ProcessExecutionError
+
+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]
+
+META_DATA_NOT_SUPPORTED = {
+ 'block-device-mapping' : {},
+ 'instance-id' : 455,
+ 'local-hostname' : 'localhost',
+ 'placement': {},
+ }
+
+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)
+ return mstr
+
+ def get_cloud_type(self):
+ '''
+ Description:
+ Get the type for the cloud back end this instance is running on
+ by examining the string returned by:
+ dmidecode --string system-product-name
+
+ On VMWare/vSphere dmidecode returns: RHEV Hypervisor
+ On VMWare/vSphere dmidecode returns: VMware Virtual Platform
+
+ Input:
+ None
+
+ Returns:
+ One of the following strings:
+ 'RHEV', 'VSPHERE' or 'UNKNOWN'
+
+ '''
+
+ cmd = CMD_DMI_SYSTEM
+ try:
+ (cmd_out, _err) = util.subp(cmd)
+ except ProcessExecutionError, _err:
+ LOG.debug(('Failed command: %s\n%s') % \
+ (' '.join(cmd), _err.message))
+ return 'UNKNOWN'
+ except OSError, _err:
+ LOG.debug(('Failed command: %s\n%s') % \
+ (' '.join(cmd), _err.message))
+ return 'UNKNOWN'
+
+ if cmd_out.upper().startswith('RHEV'):
+ return 'RHEV'
+
+ if cmd_out.upper().startswith('VMWARE'):
+ return 'VSPHERE'
+
+ return 'UNKNOWN'
+
+ def get_data(self):
+ '''
+ Description:
+ User Data is passed to the launching instance which
+
+ Cloud providers expose the user data differently.
+ It is necessary to determine which cloud provider
+ the current instance is running on to determine
+ how to access the user data. Images built with
+ image factory will contain a CLOUD_INFO_FILE which
+ contains a string identifying the cloud provider.
+
+ Images not built with Imagefactory will try to
+ determine what the cloud provider is based on system
+ information.
+ '''
+
+ LOG.debug('Invoked get_data()')
+
+ if os.path.exists(CLOUD_INFO_FILE):
+ try:
+ cloud_info = open(CLOUD_INFO_FILE)
+ cloud_type = cloud_info.read().strip().upper()
+ cloud_info.close()
+ except:
+ util.logexc(LOG, 'Unable to access cloud info file.')
+ return False
+ else:
+ cloud_type = self.get_cloud_type()
+
+ LOG.debug('cloud_type: ' + str(cloud_type))
+
+ if 'RHEV' in cloud_type:
+ return self.user_data_rhevm()
+ elif 'VSPHERE' in cloud_type:
+ return self.user_data_vsphere()
+ else:
+ # there was no recognized alternate cloud type.
+ # suggesting this handler should not be used.
+ return False
+
+ def user_data_rhevm(self):
+ '''
+ RHEVM specific userdata read
+
+ If on RHEV-M the user data will be contained on the
+ 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>
+ '''
+
+ # modprobe floppy
+ try:
+ cmd = CMD_PROBE_FLOPPY
+ (cmd_out, _err) = util.subp(cmd)
+ except ProcessExecutionError, _err:
+ 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
+
+ # mkdir <MEDIA_DIR> dir just in case it isn't already.
+ try:
+ os.makedirs(MEDIA_DIR)
+ except OSError, (errno, strerror):
+ if errno is not 17:
+ LOG.debug(('makedirs(<MEDIA_DIR>) failed: %s \nError: %s') % \
+ (errno, strerror))
+ return False
+
+ # mount /dev/fd0 <MEDIA_DIR>
+ try:
+ cmd = CMD_MNT_FLOPPY
+ (cmd_out, _err) = util.subp(cmd)
+ except ProcessExecutionError, _err:
+ # Ignore failure: already mounted
+ if 'ALREADY MOUNTED' not in _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.
+ 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
+ self.metadata = META_DATA_NOT_SUPPORTED
+
+ return True
+
+ def user_data_vsphere(self):
+ '''
+ VSphere specific userdata read
+
+ If on vSphere the user data will be contained on the
+ floppy 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>
+ '''
+
+ # mkdir <MEDIA_DIR> dir just in case it isn't already.
+ try:
+ os.makedirs(MEDIA_DIR)
+ except OSError, (errno, strerror):
+ if errno is not 17:
+ LOG.debug(('makedirs(<MEDIA_DIR>) failed: %s \nError: %s') % \
+ (errno, strerror))
+ return False
+
+ # mount /dev/cdrom <MEDIA_DIR>
+ try:
+ cmd = CMD_MNT_CDROM
+ (cmd_out, _err) = util.subp(cmd)
+ except ProcessExecutionError, _err:
+ # Ignore failure: already mounted
+ if 'ALREADY MOUNTED' not in _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:
+ 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
+ self.metadata = META_DATA_NOT_SUPPORTED
+ return True
+
+# Used to match classes to dependencies
+datasources = [
+ (DataSourceAltCloud, (sources.DEP_FILESYSTEM, )),
+]
+
+
+# Return a list of data sources that match this set of dependencies
+def get_datasource_list(depends):
+ return sources.list_from_depends(depends, datasources)