X7ROOT File Manager
Current Path:
/usr/lib/python2.7/site-packages/cloudinit/config
usr
/
lib
/
python2.7
/
site-packages
/
cloudinit
/
config
/
📁
..
📄
__init__.py
(1.4 KB)
📄
__init__.pyc
(1.31 KB)
📄
__init__.pyo
(1.31 KB)
📄
cc_apt_configure.py
(32.8 KB)
📄
cc_apt_configure.pyc
(30.2 KB)
📄
cc_apt_configure.pyo
(30.2 KB)
📄
cc_apt_pipelining.py
(2.44 KB)
📄
cc_apt_pipelining.pyc
(2.48 KB)
📄
cc_apt_pipelining.pyo
(2.48 KB)
📄
cc_bootcmd.py
(3.5 KB)
📄
cc_bootcmd.pyc
(3.02 KB)
📄
cc_bootcmd.pyo
(3.02 KB)
📄
cc_byobu.py
(3.1 KB)
📄
cc_byobu.pyc
(3.09 KB)
📄
cc_byobu.pyo
(3.09 KB)
📄
cc_ca_certs.py
(4.09 KB)
📄
cc_ca_certs.pyc
(4.1 KB)
📄
cc_ca_certs.pyo
(4.1 KB)
📄
cc_chef.py
(13.17 KB)
📄
cc_chef.pyc
(10.79 KB)
📄
cc_chef.pyo
(10.79 KB)
📄
cc_debug.py
(3.08 KB)
📄
cc_debug.pyc
(3.35 KB)
📄
cc_debug.pyo
(3.35 KB)
📄
cc_disable_ec2_metadata.py
(1.56 KB)
📄
cc_disable_ec2_metadata.pyc
(1.57 KB)
📄
cc_disable_ec2_metadata.pyo
(1.57 KB)
📄
cc_disk_setup.py
(32.8 KB)
📄
cc_disk_setup.pyc
(28.91 KB)
📄
cc_disk_setup.pyo
(28.91 KB)
📄
cc_emit_upstart.py
(2 KB)
📄
cc_emit_upstart.pyc
(2.15 KB)
📄
cc_emit_upstart.pyo
(2.15 KB)
📄
cc_fan.py
(2.83 KB)
📄
cc_fan.pyc
(3.09 KB)
📄
cc_fan.pyo
(3.09 KB)
📄
cc_final_message.py
(2.35 KB)
📄
cc_final_message.pyc
(2.39 KB)
📄
cc_final_message.pyo
(2.39 KB)
📄
cc_foo.py
(2.07 KB)
📄
cc_foo.pyc
(702 B)
📄
cc_foo.pyo
(702 B)
📄
cc_growpart.py
(11.51 KB)
📄
cc_growpart.pyc
(11.12 KB)
📄
cc_growpart.pyo
(11.12 KB)
📄
cc_grub_dpkg.py
(2.87 KB)
📄
cc_grub_dpkg.pyc
(2.63 KB)
📄
cc_grub_dpkg.pyo
(2.63 KB)
📄
cc_keys_to_console.py
(2.36 KB)
📄
cc_keys_to_console.pyc
(2.39 KB)
📄
cc_keys_to_console.pyo
(2.39 KB)
📄
cc_landscape.py
(3.93 KB)
📄
cc_landscape.pyc
(4.02 KB)
📄
cc_landscape.pyo
(4.02 KB)
📄
cc_locale.py
(1.16 KB)
📄
cc_locale.pyc
(1.16 KB)
📄
cc_locale.pyo
(1.16 KB)
📄
cc_lxd.py
(10.27 KB)
📄
cc_lxd.pyc
(8.33 KB)
📄
cc_lxd.pyo
(8.33 KB)
📄
cc_mcollective.py
(5.08 KB)
📄
cc_mcollective.pyc
(3.8 KB)
📄
cc_mcollective.pyo
(3.8 KB)
📄
cc_migrator.py
(3.07 KB)
📄
cc_migrator.pyc
(3.18 KB)
📄
cc_migrator.pyo
(3.18 KB)
📄
cc_mounts.py
(17.24 KB)
📄
cc_mounts.pyc
(14.59 KB)
📄
cc_mounts.pyo
(14.59 KB)
📄
cc_ntp.py
(20.19 KB)
📄
cc_ntp.pyc
(15.74 KB)
📄
cc_ntp.pyo
(15.74 KB)
📄
cc_package_update_upgrade_install.py
(4.11 KB)
📄
cc_package_update_upgrade_install.pyc
(4.01 KB)
📄
cc_package_update_upgrade_install.pyo
(4.01 KB)
📄
cc_phone_home.py
(3.92 KB)
📄
cc_phone_home.pyc
(3.28 KB)
📄
cc_phone_home.pyo
(3.28 KB)
📄
cc_power_state_change.py
(7.65 KB)
📄
cc_power_state_change.pyc
(7.8 KB)
📄
cc_power_state_change.pyo
(7.8 KB)
📄
cc_puppet.py
(8.86 KB)
📄
cc_puppet.pyc
(7.06 KB)
📄
cc_puppet.pyo
(7.06 KB)
📄
cc_resizefs.py
(10.79 KB)
📄
cc_resizefs.pyc
(9.15 KB)
📄
cc_resizefs.pyo
(9.15 KB)
📄
cc_resolv_conf.py
(3.43 KB)
📄
cc_resolv_conf.pyc
(3.44 KB)
📄
cc_resolv_conf.pyo
(3.44 KB)
📄
cc_rh_subscription.py
(15.65 KB)
📄
cc_rh_subscription.pyc
(13.5 KB)
📄
cc_rh_subscription.pyo
(13.5 KB)
📄
cc_rightscale_userdata.py
(3.76 KB)
📄
cc_rightscale_userdata.pyc
(2.82 KB)
📄
cc_rightscale_userdata.pyo
(2.82 KB)
📄
cc_rsyslog.py
(14.1 KB)
📄
cc_rsyslog.pyc
(10.56 KB)
📄
cc_rsyslog.pyo
(10.56 KB)
📄
cc_runcmd.py
(3.11 KB)
📄
cc_runcmd.pyc
(2.73 KB)
📄
cc_runcmd.pyo
(2.73 KB)
📄
cc_salt_minion.py
(4.66 KB)
📄
cc_salt_minion.pyc
(3.78 KB)
📄
cc_salt_minion.pyo
(3.78 KB)
📄
cc_scripts_per_boot.py
(1.2 KB)
📄
cc_scripts_per_boot.pyc
(1.2 KB)
📄
cc_scripts_per_boot.pyo
(1.2 KB)
📄
cc_scripts_per_instance.py
(1.38 KB)
📄
cc_scripts_per_instance.pyc
(1.38 KB)
📄
cc_scripts_per_instance.pyo
(1.38 KB)
📄
cc_scripts_per_once.py
(1.31 KB)
📄
cc_scripts_per_once.pyc
(1.31 KB)
📄
cc_scripts_per_once.pyo
(1.31 KB)
📄
cc_scripts_user.py
(1.42 KB)
📄
cc_scripts_user.pyc
(1.38 KB)
📄
cc_scripts_user.pyo
(1.38 KB)
📄
cc_scripts_vendor.py
(1.38 KB)
📄
cc_scripts_vendor.pyc
(1.46 KB)
📄
cc_scripts_vendor.pyo
(1.46 KB)
📄
cc_seed_random.py
(4.39 KB)
📄
cc_seed_random.pyc
(4.53 KB)
📄
cc_seed_random.pyo
(4.53 KB)
📄
cc_set_hostname.py
(2.95 KB)
📄
cc_set_hostname.pyc
(2.72 KB)
📄
cc_set_hostname.pyo
(2.72 KB)
📄
cc_set_passwords.py
(8.45 KB)
📄
cc_set_passwords.pyc
(7.62 KB)
📄
cc_set_passwords.pyo
(7.62 KB)
📄
cc_snap.py
(8.11 KB)
📄
cc_snap.pyc
(7.54 KB)
📄
cc_snap.pyo
(7.54 KB)
📄
cc_snap_config.py
(5.37 KB)
📄
cc_snap_config.pyc
(5.03 KB)
📄
cc_snap_config.pyo
(5.03 KB)
📄
cc_snappy.py
(9.69 KB)
📄
cc_snappy.pyc
(9.27 KB)
📄
cc_snappy.pyo
(9.27 KB)
📄
cc_spacewalk.py
(2.89 KB)
📄
cc_spacewalk.pyc
(2.91 KB)
📄
cc_spacewalk.pyo
(2.91 KB)
📄
cc_ssh.py
(10.59 KB)
📄
cc_ssh.pyc
(9.11 KB)
📄
cc_ssh.pyo
(9.11 KB)
📄
cc_ssh_authkey_fingerprints.py
(3.43 KB)
📄
cc_ssh_authkey_fingerprints.pyc
(3.94 KB)
📄
cc_ssh_authkey_fingerprints.pyo
(3.94 KB)
📄
cc_ssh_import_id.py
(2.88 KB)
📄
cc_ssh_import_id.pyc
(2.72 KB)
📄
cc_ssh_import_id.pyo
(2.72 KB)
📄
cc_timezone.py
(1.15 KB)
📄
cc_timezone.pyc
(1.15 KB)
📄
cc_timezone.pyo
(1.15 KB)
📄
cc_ubuntu_advantage.py
(6.08 KB)
📄
cc_ubuntu_advantage.pyc
(6.12 KB)
📄
cc_ubuntu_advantage.pyo
(6.12 KB)
📄
cc_ubuntu_drivers.py
(5.66 KB)
📄
cc_ubuntu_drivers.pyc
(4.63 KB)
📄
cc_ubuntu_drivers.pyo
(4.63 KB)
📄
cc_update_etc_hosts.py
(3.33 KB)
📄
cc_update_etc_hosts.pyc
(2.99 KB)
📄
cc_update_etc_hosts.pyo
(2.99 KB)
📄
cc_update_hostname.py
(1.58 KB)
📄
cc_update_hostname.pyc
(1.66 KB)
📄
cc_update_hostname.pyo
(1.66 KB)
📄
cc_users_groups.py
(7.06 KB)
📄
cc_users_groups.pyc
(6.76 KB)
📄
cc_users_groups.pyo
(6.76 KB)
📄
cc_write_files.py
(4.95 KB)
📄
cc_write_files.pyc
(5.13 KB)
📄
cc_write_files.pyo
(5.13 KB)
📄
cc_yum_add_repo.py
(4.27 KB)
📄
cc_yum_add_repo.pyc
(3.88 KB)
📄
cc_yum_add_repo.pyo
(3.88 KB)
📄
cc_zypper_add_repo.py
(7.62 KB)
📄
cc_zypper_add_repo.pyc
(7.12 KB)
📄
cc_zypper_add_repo.pyo
(7.12 KB)
📄
schema.py
(14.06 KB)
📄
schema.pyc
(13.83 KB)
📄
schema.pyo
(13.83 KB)
Editing: schema.py
# This file is part of cloud-init. See LICENSE file for license information. """schema.py: Set of module functions for processing cloud-config schema.""" from __future__ import print_function from cloudinit import importer from cloudinit.util import find_modules, load_file import argparse from collections import defaultdict from copy import deepcopy import logging import os import re import sys import yaml _YAML_MAP = {True: 'true', False: 'false', None: 'null'} SCHEMA_UNDEFINED = b'UNDEFINED' CLOUD_CONFIG_HEADER = b'#cloud-config' SCHEMA_DOC_TMPL = """ {name} {title_underbar} **Summary:** {title} {description} **Internal name:** ``{id}`` **Module frequency:** {frequency} **Supported distros:** {distros} **Config schema**: {property_doc} {examples} """ SCHEMA_PROPERTY_TMPL = '{prefix}**{prop_name}:** ({type}) {description}' SCHEMA_EXAMPLES_HEADER = '\n**Examples**::\n\n' SCHEMA_EXAMPLES_SPACER_TEMPLATE = '\n # --- Example{0} ---' class SchemaValidationError(ValueError): """Raised when validating a cloud-config file against a schema.""" def __init__(self, schema_errors=()): """Init the exception an n-tuple of schema errors. @param schema_errors: An n-tuple of the format: ((flat.config.key, msg),) """ self.schema_errors = schema_errors error_messages = [ '{0}: {1}'.format(config_key, message) for config_key, message in schema_errors] message = "Cloud config schema errors: {0}".format( ', '.join(error_messages)) super(SchemaValidationError, self).__init__(message) def validate_cloudconfig_schema(config, schema, strict=False): """Validate provided config meets the schema definition. @param config: Dict of cloud configuration settings validated against schema. @param schema: jsonschema dict describing the supported schema definition for the cloud config module (config.cc_*). @param strict: Boolean, when True raise SchemaValidationErrors instead of logging warnings. @raises: SchemaValidationError when provided config does not validate against the provided schema. """ try: from jsonschema import Draft4Validator, FormatChecker except ImportError: logging.debug( 'Ignoring schema validation. python-jsonschema is not present') return validator = Draft4Validator(schema, format_checker=FormatChecker()) errors = () for error in sorted(validator.iter_errors(config), key=lambda e: e.path): path = '.'.join([str(p) for p in error.path]) errors += ((path, error.message),) if errors: if strict: raise SchemaValidationError(errors) else: messages = ['{0}: {1}'.format(k, msg) for k, msg in errors] logging.warning('Invalid config:\n%s', '\n'.join(messages)) def annotated_cloudconfig_file(cloudconfig, original_content, schema_errors): """Return contents of the cloud-config file annotated with schema errors. @param cloudconfig: YAML-loaded dict from the original_content or empty dict if unparseable. @param original_content: The contents of a cloud-config file @param schema_errors: List of tuples from a JSONSchemaValidationError. The tuples consist of (schemapath, error_message). """ if not schema_errors: return original_content schemapaths = {} if cloudconfig: schemapaths = _schemapath_for_cloudconfig( cloudconfig, original_content) errors_by_line = defaultdict(list) error_count = 1 error_footer = [] annotated_content = [] for path, msg in schema_errors: match = re.match(r'format-l(?P<line>\d+)\.c(?P<col>\d+).*', path) if match: line, col = match.groups() errors_by_line[int(line)].append(msg) else: col = None errors_by_line[schemapaths[path]].append(msg) if col is not None: msg = 'Line {line} column {col}: {msg}'.format( line=line, col=col, msg=msg) error_footer.append('# E{0}: {1}'.format(error_count, msg)) error_count += 1 lines = original_content.decode().split('\n') error_count = 1 for line_number, line in enumerate(lines): errors = errors_by_line[line_number + 1] if errors: error_label = ','.join( ['E{0}'.format(count + error_count) for count in range(0, len(errors))]) error_count += len(errors) annotated_content.append(line + '\t\t# ' + error_label) else: annotated_content.append(line) annotated_content.append( '# Errors: -------------\n{0}\n\n'.format('\n'.join(error_footer))) return '\n'.join(annotated_content) def validate_cloudconfig_file(config_path, schema, annotate=False): """Validate cloudconfig file adheres to a specific jsonschema. @param config_path: Path to the yaml cloud-config file to parse. @param schema: Dict describing a valid jsonschema to validate against. @param annotate: Boolean set True to print original config file with error annotations on the offending lines. @raises SchemaValidationError containing any of schema_errors encountered. @raises RuntimeError when config_path does not exist. """ if not os.path.exists(config_path): raise RuntimeError('Configfile {0} does not exist'.format(config_path)) content = load_file(config_path, decode=False) if not content.startswith(CLOUD_CONFIG_HEADER): errors = ( ('format-l1.c1', 'File {0} needs to begin with "{1}"'.format( config_path, CLOUD_CONFIG_HEADER.decode())),) error = SchemaValidationError(errors) if annotate: print(annotated_cloudconfig_file({}, content, error.schema_errors)) raise error try: cloudconfig = yaml.safe_load(content) except (yaml.YAMLError) as e: line = column = 1 mark = None if hasattr(e, 'context_mark') and getattr(e, 'context_mark'): mark = getattr(e, 'context_mark') elif hasattr(e, 'problem_mark') and getattr(e, 'problem_mark'): mark = getattr(e, 'problem_mark') if mark: line = mark.line + 1 column = mark.column + 1 errors = (('format-l{line}.c{col}'.format(line=line, col=column), 'File {0} is not valid yaml. {1}'.format( config_path, str(e))),) error = SchemaValidationError(errors) if annotate: print(annotated_cloudconfig_file({}, content, error.schema_errors)) raise error try: validate_cloudconfig_schema( cloudconfig, schema, strict=True) except SchemaValidationError as e: if annotate: print(annotated_cloudconfig_file( cloudconfig, content, e.schema_errors)) raise def _schemapath_for_cloudconfig(config, original_content): """Return a dictionary mapping schemapath to original_content line number. @param config: The yaml.loaded config dictionary of a cloud-config file. @param original_content: The simple file content of the cloud-config file """ # FIXME Doesn't handle multi-line lists or multi-line strings content_lines = original_content.decode().split('\n') schema_line_numbers = {} list_index = 0 RE_YAML_INDENT = r'^(\s*)' scopes = [] for line_number, line in enumerate(content_lines, 1): indent_depth = len(re.match(RE_YAML_INDENT, line).groups()[0]) line = line.strip() if not line or line.startswith('#'): continue if scopes: previous_depth, path_prefix = scopes[-1] else: previous_depth = -1 path_prefix = '' if line.startswith('- '): key = str(list_index) value = line[1:] list_index += 1 else: list_index = 0 key, value = line.split(':', 1) while indent_depth <= previous_depth: if scopes: previous_depth, path_prefix = scopes.pop() else: previous_depth = -1 path_prefix = '' if path_prefix: key = path_prefix + '.' + key scopes.append((indent_depth, key)) if value: value = value.strip() if value.startswith('['): scopes.append((indent_depth + 2, key + '.0')) for inner_list_index in range(0, len(yaml.safe_load(value))): list_key = key + '.' + str(inner_list_index) schema_line_numbers[list_key] = line_number schema_line_numbers[key] = line_number return schema_line_numbers def _get_property_type(property_dict): """Return a string representing a property type from a given jsonschema.""" property_type = property_dict.get('type', SCHEMA_UNDEFINED) if property_type == SCHEMA_UNDEFINED and property_dict.get('enum'): property_type = [ str(_YAML_MAP.get(k, k)) for k in property_dict['enum']] if isinstance(property_type, list): property_type = '/'.join(property_type) items = property_dict.get('items', {}) sub_property_type = items.get('type', '') # Collect each item type for sub_item in items.get('oneOf', {}): if sub_property_type: sub_property_type += '/' sub_property_type += '(' + _get_property_type(sub_item) + ')' if sub_property_type: return '{0} of {1}'.format(property_type, sub_property_type) return property_type def _get_property_doc(schema, prefix=' '): """Return restructured text describing the supported schema properties.""" new_prefix = prefix + ' ' properties = [] for prop_key, prop_config in schema.get('properties', {}).items(): # Define prop_name and dscription for SCHEMA_PROPERTY_TMPL description = prop_config.get('description', '') properties.append(SCHEMA_PROPERTY_TMPL.format( prefix=prefix, prop_name=prop_key, type=_get_property_type(prop_config), description=description.replace('\n', ''))) if 'properties' in prop_config: properties.append( _get_property_doc(prop_config, prefix=new_prefix)) return '\n\n'.join(properties) def _get_schema_examples(schema, prefix=''): """Return restructured text describing the schema examples if present.""" examples = schema.get('examples') if not examples: return '' rst_content = SCHEMA_EXAMPLES_HEADER for count, example in enumerate(examples): # Python2.6 is missing textwrapper.indent lines = example.split('\n') indented_lines = [' {0}'.format(line) for line in lines] if rst_content != SCHEMA_EXAMPLES_HEADER: indented_lines.insert( 0, SCHEMA_EXAMPLES_SPACER_TEMPLATE.format(count + 1)) rst_content += '\n'.join(indented_lines) return rst_content def get_schema_doc(schema): """Return reStructured text rendering the provided jsonschema. @param schema: Dict of jsonschema to render. @raise KeyError: If schema lacks an expected key. """ schema_copy = deepcopy(schema) schema_copy['property_doc'] = _get_property_doc(schema) schema_copy['examples'] = _get_schema_examples(schema) schema_copy['distros'] = ', '.join(schema['distros']) # Need an underbar of the same length as the name schema_copy['title_underbar'] = re.sub(r'.', '-', schema['name']) return SCHEMA_DOC_TMPL.format(**schema_copy) FULL_SCHEMA = None def get_schema(): """Return jsonschema coalesced from all cc_* cloud-config module.""" global FULL_SCHEMA if FULL_SCHEMA: return FULL_SCHEMA full_schema = { '$schema': 'http://json-schema.org/draft-04/schema#', 'id': 'cloud-config-schema', 'allOf': []} configs_dir = os.path.dirname(os.path.abspath(__file__)) potential_handlers = find_modules(configs_dir) for (_fname, mod_name) in potential_handlers.items(): mod_locs, _looked_locs = importer.find_module( mod_name, ['cloudinit.config'], ['schema']) if mod_locs: mod = importer.import_module(mod_locs[0]) full_schema['allOf'].append(mod.schema) FULL_SCHEMA = full_schema return full_schema def error(message): print(message, file=sys.stderr) sys.exit(1) def get_parser(parser=None): """Return a parser for supported cmdline arguments.""" if not parser: parser = argparse.ArgumentParser( prog='cloudconfig-schema', description='Validate cloud-config files or document schema') parser.add_argument('-c', '--config-file', help='Path of the cloud-config yaml file to validate') parser.add_argument('-d', '--doc', action="store_true", default=False, help='Print schema documentation') parser.add_argument('--annotate', action="store_true", default=False, help='Annotate existing cloud-config file with errors') return parser def handle_schema_args(name, args): """Handle provided schema args and perform the appropriate actions.""" exclusive_args = [args.config_file, args.doc] if not any(exclusive_args) or all(exclusive_args): error('Expected either --config-file argument or --doc') full_schema = get_schema() if args.config_file: try: validate_cloudconfig_file( args.config_file, full_schema, args.annotate) except SchemaValidationError as e: if not args.annotate: error(str(e)) except RuntimeError as e: error(str(e)) else: print("Valid cloud-config file {0}".format(args.config_file)) if args.doc: for subschema in full_schema['allOf']: print(get_schema_doc(subschema)) def main(): """Tool to validate schema of a cloud-config file or print schema docs.""" parser = get_parser() handle_schema_args('cloudconfig-schema', parser.parse_args()) return 0 if __name__ == '__main__': sys.exit(main()) # vi: ts=4 expandtab
Upload File
Create Folder