diff options
Diffstat (limited to 'cloudinit')
| -rw-r--r-- | cloudinit/config/cc_chef.py | 305 | ||||
| -rw-r--r-- | cloudinit/config/cc_debug.py | 18 | ||||
| -rw-r--r-- | cloudinit/distros/__init__.py | 16 | ||||
| -rw-r--r-- | cloudinit/distros/net_util.py | 11 | ||||
| -rw-r--r-- | cloudinit/ssh_util.py | 5 | ||||
| -rw-r--r-- | cloudinit/util.py | 20 | 
6 files changed, 296 insertions, 79 deletions
| diff --git a/cloudinit/config/cc_chef.py b/cloudinit/config/cc_chef.py index 806deed9..fc837363 100644 --- a/cloudinit/config/cc_chef.py +++ b/cloudinit/config/cc_chef.py @@ -18,6 +18,57 @@  #    You should have received a copy of the GNU General Public License  #    along with this program.  If not, see <http://www.gnu.org/licenses/>. +""" +**Summary:** module that configures, starts and installs chef. + +**Description:** This module enables chef to be installed (from packages or +from gems, or from omnibus). Before this occurs chef configurations are +written to disk (validation.pem, client.pem, firstboot.json, client.rb), +and needed chef folders/directories are created (/etc/chef and /var/log/chef +and so-on). Then once installing proceeds correctly if configured chef will +be started (in daemon mode or in non-daemon mode) and then once that has +finished (if ran in non-daemon mode this will be when chef finishes +converging, if ran in daemon mode then no further actions are possible since +chef will have forked into its own process) then a post run function can +run that can do finishing activities (such as removing the validation pem +file). + +It can be configured with the following option structure:: + +    chef: +       directories: (defaulting to /etc/chef, /var/log/chef, /var/lib/chef, +                     /var/cache/chef, /var/backups/chef, /var/run/chef) +       validation_key or validation_cert: (optional string to be written to +                                           /etc/chef/validation.pem) +       firstboot_path: (path to write run_list and initial_attributes keys that +                        should also be present in this configuration, defaults +                        to /etc/chef/firstboot.json) +       exec: boolean to run or not run chef (defaults to false, unless +                                             a gem installed is requested +                                             where this will then default +                                             to true) + +    chef.rb template keys (if falsey, then will be skipped and not +                           written to /etc/chef/client.rb) + +    chef: +      client_key: +      environment: +      file_backup_path: +      file_cache_path: +      json_attribs: +      log_level: +      log_location: +      node_name: +      pid_file: +      server_url: +      show_time: +      ssl_verify_mode: +      validation_key: +      validation_name: +""" + +import itertools  import json  import os @@ -27,19 +78,112 @@ from cloudinit import util  RUBY_VERSION_DEFAULT = "1.8" -CHEF_DIRS = [ +CHEF_DIRS = tuple([      '/etc/chef',      '/var/log/chef',      '/var/lib/chef',      '/var/cache/chef',      '/var/backups/chef',      '/var/run/chef', -] +]) +REQUIRED_CHEF_DIRS = tuple([ +    '/etc/chef', +]) + +# Used if fetching chef from a omnibus style package +OMNIBUS_URL = "https://www.getchef.com/chef/install.sh" +OMNIBUS_URL_RETRIES = 5 + +CHEF_VALIDATION_PEM_PATH = '/etc/chef/validation.pem' +CHEF_FB_PATH = '/etc/chef/firstboot.json' +CHEF_RB_TPL_DEFAULTS = { +    # These are ruby symbols... +    'ssl_verify_mode': ':verify_none', +    'log_level': ':info', +    # These are not symbols... +    'log_location': '/var/log/chef/client.log', +    'validation_key': CHEF_VALIDATION_PEM_PATH, +    'client_key': "/etc/chef/client.pem", +    'json_attribs': CHEF_FB_PATH, +    'file_cache_path': "/var/cache/chef", +    'file_backup_path': "/var/backups/chef", +    'pid_file': "/var/run/chef/client.pid", +    'show_time': True, +} +CHEF_RB_TPL_BOOL_KEYS = frozenset(['show_time']) +CHEF_RB_TPL_PATH_KEYS = frozenset([ +    'log_location', +    'validation_key', +    'client_key', +    'file_cache_path', +    'json_attribs', +    'file_cache_path', +    'pid_file', +]) +CHEF_RB_TPL_KEYS = list(CHEF_RB_TPL_DEFAULTS.keys()) +CHEF_RB_TPL_KEYS.extend(CHEF_RB_TPL_BOOL_KEYS) +CHEF_RB_TPL_KEYS.extend(CHEF_RB_TPL_PATH_KEYS) +CHEF_RB_TPL_KEYS.extend([ +    'server_url', +    'node_name', +    'environment', +    'validation_name', +]) +CHEF_RB_TPL_KEYS = frozenset(CHEF_RB_TPL_KEYS) +CHEF_RB_PATH = '/etc/chef/client.rb' +CHEF_EXEC_PATH = '/usr/bin/chef-client' +CHEF_EXEC_DEF_ARGS = tuple(['-d', '-i', '1800', '-s', '20']) + + +def is_installed(): +    if not os.path.isfile(CHEF_EXEC_PATH): +        return False +    if not os.access(CHEF_EXEC_PATH, os.X_OK): +        return False +    return True + + +def post_run_chef(chef_cfg, log): +    delete_pem = util.get_cfg_option_bool(chef_cfg, +                                          'delete_validation_post_exec', +                                          default=False) +    if delete_pem and os.path.isfile(CHEF_VALIDATION_PEM_PATH): +        os.unlink(CHEF_VALIDATION_PEM_PATH) -OMNIBUS_URL = "https://www.opscode.com/chef/install.sh" + +def get_template_params(iid, chef_cfg, log): +    params = CHEF_RB_TPL_DEFAULTS.copy() +    # Allow users to overwrite any of the keys they want (if they so choose), +    # when a value is None, then the value will be set to None and no boolean +    # or string version will be populated... +    for (k, v) in chef_cfg.items(): +        if k not in CHEF_RB_TPL_KEYS: +            log.debug("Skipping unknown chef template key '%s'", k) +            continue +        if v is None: +            params[k] = None +        else: +            # This will make the value a boolean or string... +            if k in CHEF_RB_TPL_BOOL_KEYS: +                params[k] = util.get_cfg_option_bool(chef_cfg, k) +            else: +                params[k] = util.get_cfg_option_str(chef_cfg, k) +    # These ones are overwritten to be exact values... +    params.update({ +        'generated_by': util.make_header(), +        'node_name': util.get_cfg_option_str(chef_cfg, 'node_name', +                                             default=iid), +        'environment': util.get_cfg_option_str(chef_cfg, 'environment', +                                               default='_default'), +        # These two are mandatory... +        'server_url': chef_cfg['server_url'], +        'validation_name': chef_cfg['validation_name'], +    }) +    return params  def handle(name, cfg, cloud, log, _args): +    """Handler method activated by cloud-init."""      # If there isn't a chef key in the configuration don't do anything      if 'chef' not in cfg: @@ -49,7 +193,10 @@ def handle(name, cfg, cloud, log, _args):      chef_cfg = cfg['chef']      # Ensure the chef directories we use exist -    for d in CHEF_DIRS: +    chef_dirs = util.get_cfg_option_list(chef_cfg, 'directories') +    if not chef_dirs: +        chef_dirs = list(CHEF_DIRS) +    for d in itertools.chain(chef_dirs, REQUIRED_CHEF_DIRS):          util.ensure_dir(d)      # Set the validation key based on the presence of either 'validation_key' @@ -57,64 +204,108 @@ def handle(name, cfg, cloud, log, _args):      # takes precedence      for key in ('validation_key', 'validation_cert'):          if key in chef_cfg and chef_cfg[key]: -            util.write_file('/etc/chef/validation.pem', chef_cfg[key]) +            util.write_file(CHEF_VALIDATION_PEM_PATH, chef_cfg[key])              break      # Create the chef config from template      template_fn = cloud.get_template_filename('chef_client.rb')      if template_fn:          iid = str(cloud.datasource.get_instance_id()) -        params = { -            'server_url': chef_cfg['server_url'], -            'node_name': util.get_cfg_option_str(chef_cfg, 'node_name', iid), -            'environment': util.get_cfg_option_str(chef_cfg, 'environment', -                                                   '_default'), -            'validation_name': chef_cfg['validation_name'] -        } -        templater.render_to_file(template_fn, '/etc/chef/client.rb', params) +        params = get_template_params(iid, chef_cfg, log) +        # Do a best effort attempt to ensure that the template values that +        # are associated with paths have there parent directory created +        # before they are used by the chef-client itself. +        param_paths = set() +        for (k, v) in params.items(): +            if k in CHEF_RB_TPL_PATH_KEYS and v: +                param_paths.add(os.path.dirname(v)) +        util.ensure_dirs(param_paths) +        templater.render_to_file(template_fn, CHEF_RB_PATH, params)      else: -        log.warn("No template found, not rendering to /etc/chef/client.rb") - -    # set the firstboot json -    initial_json = {} -    if 'run_list' in chef_cfg: -        initial_json['run_list'] = chef_cfg['run_list'] -    if 'initial_attributes' in chef_cfg: -        initial_attributes = chef_cfg['initial_attributes'] -        for k in list(initial_attributes.keys()): -            initial_json[k] = initial_attributes[k] -    util.write_file('/etc/chef/firstboot.json', json.dumps(initial_json)) +        log.warn("No template found, not rendering to %s", +                 CHEF_RB_PATH) -    # If chef is not installed, we install chef based on 'install_type' -    if (not os.path.isfile('/usr/bin/chef-client') or -            util.get_cfg_option_bool(chef_cfg, -                'force_install', default=False)): - -        install_type = util.get_cfg_option_str(chef_cfg, 'install_type', -                                               'packages') -        if install_type == "gems": -            # this will install and run the chef-client from gems -            chef_version = util.get_cfg_option_str(chef_cfg, 'version', None) -            ruby_version = util.get_cfg_option_str(chef_cfg, 'ruby_version', -                                                   RUBY_VERSION_DEFAULT) -            install_chef_from_gems(cloud.distro, ruby_version, chef_version) -            # and finally, run chef-client -            log.debug('Running chef-client') -            util.subp(['/usr/bin/chef-client', -                       '-d', '-i', '1800', '-s', '20'], capture=False) -        elif install_type == 'packages': -            # this will install and run the chef-client from packages -            cloud.distro.install_packages(('chef',)) -        elif install_type == 'omnibus': -            url = util.get_cfg_option_str(chef_cfg, "omnibus_url", OMNIBUS_URL) -            content = url_helper.readurl(url=url, retries=5) -            with util.tempdir() as tmpd: -                # use tmpd over tmpfile to avoid 'Text file busy' on execute -                tmpf = "%s/chef-omnibus-install" % tmpd -                util.write_file(tmpf, str(content), mode=0700) -                util.subp([tmpf], capture=False) +    # Set the firstboot json +    fb_filename = util.get_cfg_option_str(chef_cfg, 'firstboot_path', +                                          default=CHEF_FB_PATH) +    if not fb_filename: +        log.info("First boot path empty, not writing first boot json file") +    else: +        initial_json = {} +        if 'run_list' in chef_cfg: +            initial_json['run_list'] = chef_cfg['run_list'] +        if 'initial_attributes' in chef_cfg: +            initial_attributes = chef_cfg['initial_attributes'] +            for k in list(initial_attributes.keys()): +                initial_json[k] = initial_attributes[k] +        util.write_file(fb_filename, json.dumps(initial_json)) + +    # Try to install chef, if its not already installed... +    force_install = util.get_cfg_option_bool(chef_cfg, +                                             'force_install', default=False) +    if not is_installed() or force_install: +        run = install_chef(cloud, chef_cfg, log) +    elif is_installed(): +        run = util.get_cfg_option_bool(chef_cfg, 'exec', default=False) +    else: +        run = False +    if run: +        run_chef(chef_cfg, log) +        post_run_chef(chef_cfg, log) + + +def run_chef(chef_cfg, log): +    log.debug('Running chef-client') +    cmd = [CHEF_EXEC_PATH] +    if 'exec_arguments' in chef_cfg: +        cmd_args = chef_cfg['exec_arguments'] +        if isinstance(cmd_args, (list, tuple)): +            cmd.extend(cmd_args) +        elif isinstance(cmd_args, (str, basestring)): +            cmd.append(cmd_args)          else: -            log.warn("Unknown chef install type %s", install_type) +            log.warn("Unknown type %s provided for chef" +                     " 'exec_arguments' expected list, tuple," +                     " or string", type(cmd_args)) +            cmd.extend(CHEF_EXEC_DEF_ARGS) +    else: +        cmd.extend(CHEF_EXEC_DEF_ARGS) +    util.subp(cmd, capture=False) + + +def install_chef(cloud, chef_cfg, log): +    # If chef is not installed, we install chef based on 'install_type' +    install_type = util.get_cfg_option_str(chef_cfg, 'install_type', +                                           'packages') +    run = util.get_cfg_option_bool(chef_cfg, 'exec', default=False) +    if install_type == "gems": +        # This will install and run the chef-client from gems +        chef_version = util.get_cfg_option_str(chef_cfg, 'version', None) +        ruby_version = util.get_cfg_option_str(chef_cfg, 'ruby_version', +                                               RUBY_VERSION_DEFAULT) +        install_chef_from_gems(cloud.distro, ruby_version, chef_version) +        # Retain backwards compat, by preferring True instead of False +        # when not provided/overriden... +        run = util.get_cfg_option_bool(chef_cfg, 'exec', default=True) +    elif install_type == 'packages': +        # This will install and run the chef-client from packages +        cloud.distro.install_packages(('chef',)) +    elif install_type == 'omnibus': +        # This will install as a omnibus unified package +        url = util.get_cfg_option_str(chef_cfg, "omnibus_url", OMNIBUS_URL) +        retries = max(0, util.get_cfg_option_int(chef_cfg, +                                                 "omnibus_url_retries", +                                                 default=OMNIBUS_URL_RETRIES)) +        content = url_helper.readurl(url=url, retries=retries) +        with util.tempdir() as tmpd: +            # Use tmpdir over tmpfile to avoid 'text file busy' on execute +            tmpf = "%s/chef-omnibus-install" % tmpd +            util.write_file(tmpf, str(content), mode=0700) +            util.subp([tmpf], capture=False) +    else: +        log.warn("Unknown chef install type '%s'", install_type) +        run = False +    return run  def get_ruby_packages(version): @@ -133,9 +324,9 @@ def install_chef_from_gems(ruby_version, chef_version, distro):          util.sym_link('/usr/bin/ruby%s' % ruby_version, '/usr/bin/ruby')      if chef_version:          util.subp(['/usr/bin/gem', 'install', 'chef', -                  '-v %s' % chef_version, '--no-ri', -                  '--no-rdoc', '--bindir', '/usr/bin', '-q'], capture=False) +                   '-v %s' % chef_version, '--no-ri', +                   '--no-rdoc', '--bindir', '/usr/bin', '-q'], capture=False)      else:          util.subp(['/usr/bin/gem', 'install', 'chef', -                  '--no-ri', '--no-rdoc', '--bindir', -                  '/usr/bin', '-q'], capture=False) +                   '--no-ri', '--no-rdoc', '--bindir', +                   '/usr/bin', '-q'], capture=False) diff --git a/cloudinit/config/cc_debug.py b/cloudinit/config/cc_debug.py index a3af4500..8c489426 100644 --- a/cloudinit/config/cc_debug.py +++ b/cloudinit/config/cc_debug.py @@ -33,11 +33,14 @@ It can be configured with the following option structure::      Log configurations are not output.  """ -from cloudinit import type_utils -from cloudinit import util  import copy  from StringIO import StringIO +from cloudinit import type_utils +from cloudinit import util + +SKIP_KEYS = frozenset(['log_cfgs']) +  def _make_header(text):      header = StringIO() @@ -50,6 +53,11 @@ def _make_header(text):      return header.getvalue() +def _dumps(obj): +    text = util.yaml_dumps(obj, explicit_start=False, explicit_end=False) +    return text.rstrip() + +  def handle(name, cfg, cloud, log, args):      """Handler method activated by cloud-init.""" @@ -67,7 +75,7 @@ def handle(name, cfg, cloud, log, args):          return      # Clean out some keys that we just don't care about showing...      dump_cfg = copy.deepcopy(cfg) -    for k in ['log_cfgs']: +    for k in SKIP_KEYS:          dump_cfg.pop(k, None)      all_keys = list(dump_cfg.keys())      for k in all_keys: @@ -76,10 +84,10 @@ def handle(name, cfg, cloud, log, args):      # Now dump it...      to_print = StringIO()      to_print.write(_make_header("Config")) -    to_print.write(util.yaml_dumps(dump_cfg)) +    to_print.write(_dumps(dump_cfg))      to_print.write("\n")      to_print.write(_make_header("MetaData")) -    to_print.write(util.yaml_dumps(cloud.datasource.metadata)) +    to_print.write(_dumps(cloud.datasource.metadata))      to_print.write("\n")      to_print.write(_make_header("Misc"))      to_print.write("Datasource: %s\n" % diff --git a/cloudinit/distros/__init__.py b/cloudinit/distros/__init__.py index 2599d9f2..83c2eebf 100644 --- a/cloudinit/distros/__init__.py +++ b/cloudinit/distros/__init__.py @@ -387,8 +387,20 @@ class Distro(object):          # Import SSH keys          if 'ssh_authorized_keys' in kwargs: -            keys = set(kwargs['ssh_authorized_keys']) or [] -            ssh_util.setup_user_keys(keys, name, options=None) +            # Try to handle this in a smart manner. +            keys = kwargs['ssh_authorized_keys'] +            if isinstance(keys, (basestring, str)): +                keys = [keys] +            if isinstance(keys, dict): +                keys = list(keys.values()) +            if keys is not None: +                if not isinstance(keys, (tuple, list, set)): +                    LOG.warn("Invalid type '%s' detected for" +                             " 'ssh_authorized_keys', expected list," +                             " string, dict, or set.", type(keys)) +                else: +                    keys = set(keys) or [] +                    ssh_util.setup_user_keys(keys, name, options=None)          return True diff --git a/cloudinit/distros/net_util.py b/cloudinit/distros/net_util.py index f8c34846..dd63a6a3 100644 --- a/cloudinit/distros/net_util.py +++ b/cloudinit/distros/net_util.py @@ -113,11 +113,11 @@ def translate_network(settings):      for info in ifaces:          if 'iface' not in info:              continue +        iface_details = info['iface'].split(None) +        # Check if current device *may* have an ipv6 IP          use_ipv6 = False -        # Check if current device has an ipv6 IP -        if 'inet6' in info['iface']: +        if 'inet6' in iface_details:              use_ipv6 = True -        iface_details = info['iface'].split(None)          dev_name = None          if len(iface_details) >= 1:              dev = iface_details[0].strip().lower() @@ -160,9 +160,8 @@ def translate_network(settings):                      hw_addr = hw_split[1]                      if hw_addr:                          iface_info['hwaddress'] = hw_addr - -        # If ipv6 is enabled, device will have multiple IPs. -        # Update the dictionary instead of overwriting it +        # If ipv6 is enabled, device will have multiple IPs, so we need to +        # update the dictionary instead of overwriting it...          if dev_name in real_ifaces:              real_ifaces[dev_name].update(iface_info)          else: diff --git a/cloudinit/ssh_util.py b/cloudinit/ssh_util.py index 70a577bc..14d0cb0f 100644 --- a/cloudinit/ssh_util.py +++ b/cloudinit/ssh_util.py @@ -293,7 +293,10 @@ def parse_ssh_config(fname):          if not line or line.startswith("#"):              lines.append(SshdConfigLine(line))              continue -        (key, val) = line.split(None, 1) +        try: +            key, val = line.split(None, 1) +        except ValueError: +            key, val = line.split('=', 1)          lines.append(SshdConfigLine(line, key, val))      return lines diff --git a/cloudinit/util.py b/cloudinit/util.py index 4bb73c11..ee5e5c0a 100644 --- a/cloudinit/util.py +++ b/cloudinit/util.py @@ -399,6 +399,10 @@ def get_cfg_option_str(yobj, key, default=None):      return val +def get_cfg_option_int(yobj, key, default=0): +    return int(get_cfg_option_str(yobj, key, default=default)) + +  def system_info():      return {          'platform': platform.platform(), @@ -1270,14 +1274,14 @@ def read_write_cmdline_url(target_fn):              logexc(LOG, "Failed writing url content to %s", target_fn) -def yaml_dumps(obj): -    formatted = yaml.dump(obj, -                    line_break="\n", -                    indent=4, -                    explicit_start=True, -                    explicit_end=True, -                    default_flow_style=False) -    return formatted +def yaml_dumps(obj, explicit_start=True, explicit_end=True): +    return yaml.safe_dump(obj, +                          line_break="\n", +                          indent=4, +                          explicit_start=explicit_start, +                          explicit_end=explicit_end, +                          default_flow_style=False, +                          allow_unicode=True)  def ensure_dir(path, mode=None): | 
