- # -*- coding: utf-8 -*-
-
- import errno
- import json
- import logging
- import os
- import shutil
- import six
- import subprocess
- import tempfile
- import uuid
- import yaml
-
- LOG = logging.getLogger(__name__)
-
- class ConfigDriveBuilder(object):
- """Build config drives, optionally as a context manager."""
-
- def __init__(self, image_file):
- self.image_file = image_file
- self.mdfiles=[]
-
- def __enter__(self):
- self._delete_if_exists(self.image_file)
- return self
-
- def __exit__(self, exctype, excval, exctb):
- self.make_drive()
-
- @staticmethod
- def _ensure_tree(path):
- try:
- os.makedirs(path)
- except OSError as e:
- if e.errno == errno.EEXIST and os.path.isdir(path):
- pass
- else:
- raise
-
- @staticmethod
- def _delete_if_exists(path):
- try:
- os.unlink(path)
- except OSError as e:
- if e.errno != errno.ENOENT:
- raise
-
- def add_file(self, path, data):
- self.mdfiles.append((path, data))
-
- def _add_file(self, basedir, path, data):
- filepath = os.path.join(basedir, path)
- dirname = os.path.dirname(filepath)
- self._ensure_tree(dirname)
- with open(filepath, 'wb') as f:
- if isinstance(data, six.text_type):
- data = data.encode('utf-8')
- f.write(data)
-
- def _write_md_files(self, basedir):
- for data in self.mdfiles:
- self._add_file(basedir, data[0], data[1])
-
- def _make_iso9660(self, path, tmpdir):
- cmd = ['mkisofs',
- '-o', path,
- '-ldots',
- '-allow-lowercase',
- '-allow-multidot',
- '-l',
- '-V', 'config-2',
- '-r',
- '-J',
- '-quiet',
- tmpdir]
- try:
- LOG.info('Running cmd (subprocess): %s', cmd)
- _pipe = subprocess.PIPE
- obj = subprocess.Popen(cmd,
- stdin=_pipe,
- stdout=_pipe,
- stderr=_pipe,
- close_fds=True)
- (stdout, stderr) = obj.communicate()
- obj.stdin.close()
- _returncode = obj.returncode
- LOG.debug('Cmd "%s" returned: %s', cmd, _returncode)
- if _returncode != 0:
- output = 'Stdout: %s\nStderr: %s' % (stdout, stderr)
- LOG.error('The command "%s" failed. %s',
- cmd, output)
- raise subprocess.CalledProcessError(cmd=cmd,
- returncode=_returncode,
- output=output)
- except OSError as err:
- LOG.error('Got an OSError in the command: "%s". Errno: %s', cmd,
- err.errno)
- raise
-
- def make_drive(self):
- """Make the config drive.
- :raises CalledProcessError if a helper process has failed.
- """
- try:
- tmpdir = tempfile.mkdtemp()
- self._write_md_files(tmpdir)
- self._make_iso9660(self.image_file, tmpdir)
- finally:
- shutil.rmtree(tmpdir)
-
-
- def generate(dst, hostname, domainname, instance_id=None, user_data=None,
- network_data=None):
-
- ''' Generate config drive
-
- :param dst: destination file to place config drive.
- :param hostname: hostname of Instance.
- :param domainname: instance domain.
- :param instance_id: UUID of the instance.
- :param user_data: custom user data dictionary.
- :param network_data: custom network info dictionary.
-
- '''
- instance_md = {}
- instance_md['uuid'] = instance_id or str(uuid.uuid4())
- instance_md['hostname'] = '%s.%s' % (hostname, domainname)
- instance_md['name'] = hostname
-
- if user_data:
- user_data = '#cloud-config\n\n' + yaml.dump(user_data, default_flow_style=False)
-
- data = json.dumps(instance_md)
- with ConfigDriveBuilder(dst) as cfgdrive:
- cfgdrive.add_file('openstack/latest/meta_data.json', data)
- if user_data:
- cfgdrive.add_file('openstack/latest/user_data', user_data)
- if network_data:
- cfgdrive.add_file('openstack/latest/network_data.json', json.dumps(network_data))
-
- LOG.debug('Config drive was built %s' % dst)
|