# Copyright 2014-2016 ARM Limited # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. import os from copy import copy from collections import OrderedDict from wlauto.exceptions import ConfigError from wlauto.utils.misc import (get_article, merge_config_values) from wlauto.utils.types import (identifier, integer, boolean, list_of_strings, toggle_set) from wlauto.core.configuration.tree import SectionNode ########################## ### CONFIG POINT TYPES ### ########################## class RebootPolicy(object): """ Represents the reboot policy for the execution -- at what points the device should be rebooted. This, in turn, is controlled by the policy value that is passed in on construction and would typically be read from the user's settings. Valid policy values are: :never: The device will never be rebooted. :as_needed: Only reboot the device if it becomes unresponsive, or needs to be flashed, etc. :initial: The device will be rebooted when the execution first starts, just before executing the first workload spec. :each_spec: The device will be rebooted before running a new workload spec. :each_iteration: The device will be rebooted before each new iteration. """ valid_policies = ['never', 'as_needed', 'initial', 'each_spec', 'each_iteration'] def __init__(self, policy): policy = policy.strip().lower().replace(' ', '_') if policy not in self.valid_policies: message = 'Invalid reboot policy {}; must be one of {}'.format(policy, ', '.join(self.valid_policies)) raise ConfigError(message) self.policy = policy @property def can_reboot(self): return self.policy != 'never' @property def perform_initial_boot(self): return self.policy not in ['never', 'as_needed'] @property def reboot_on_each_spec(self): return self.policy in ['each_spec', 'each_iteration'] @property def reboot_on_each_iteration(self): return self.policy == 'each_iteration' def __str__(self): return self.policy __repr__ = __str__ def __cmp__(self, other): if isinstance(other, RebootPolicy): return cmp(self.policy, other.policy) else: return cmp(self.policy, other) def to_pod(self): return self.policy @staticmethod def from_pod(pod): return RebootPolicy(pod) class status_list(list): def append(self, item): list.append(self, str(item).upper()) class LoggingConfig(dict): defaults = { 'file_format': '%(asctime)s %(levelname)-8s %(name)s: %(message)s', 'verbose_format': '%(asctime)s %(levelname)-8s %(name)s: %(message)s', 'regular_format': '%(levelname)-8s %(message)s', 'color': True, } def __init__(self, config=None): dict.__init__(self) if isinstance(config, dict): config = {identifier(k.lower()): v for k, v in config.iteritems()} self['regular_format'] = config.pop('regular_format', self.defaults['regular_format']) self['verbose_format'] = config.pop('verbose_format', self.defaults['verbose_format']) self['file_format'] = config.pop('file_format', self.defaults['file_format']) self['color'] = config.pop('colour_enabled', self.defaults['color']) # legacy self['color'] = config.pop('color', self.defaults['color']) if config: message = 'Unexpected logging configuation parameters: {}' raise ValueError(message.format(bad_vals=', '.join(config.keys()))) elif config is None: for k, v in self.defaults.iteritems(): self[k] = v else: raise ValueError(config) class ConfigurationPoint(object): """ This defines a generic configuration point for workload automation. This is used to handle global settings, plugin parameters, etc. """ # Mapping for kind conversion; see docs for convert_types below kind_map = { int: integer, bool: boolean, dict: OrderedDict, } def __init__(self, name, kind=None, mandatory=None, default=None, override=False, allowed_values=None, description=None, constraint=None, merge=False, aliases=None, convert_types=True): """ Create a new Parameter object. :param name: The name of the parameter. This will become an instance member of the plugin object to which the parameter is applied, so it must be a valid python identifier. This is the only mandatory parameter. :param kind: The type of parameter this is. This must be a callable that takes an arbitrary object and converts it to the expected type, or raised ``ValueError`` if such conversion is not possible. Most Python standard types -- ``str``, ``int``, ``bool``, etc. -- can be used here. This defaults to ``str`` if not specified. :param mandatory: If set to ``True``, then a non-``None`` value for this parameter *must* be provided on plugin object construction, otherwise ``ConfigError`` will be raised. :param default: The default value for this parameter. If no value is specified on plugin construction, this value will be used instead. (Note: if this is specified and is not ``None``, then ``mandatory`` parameter will be ignored). :param override: A ``bool`` that specifies whether a parameter of the same name further up the hierarchy should be overridden. If this is ``False`` (the default), an exception will be raised by the ``AttributeCollection`` instead. :param allowed_values: This should be the complete list of allowed values for this parameter. Note: ``None`` value will always be allowed, even if it is not in this list. If you want to disallow ``None``, set ``mandatory`` to ``True``. :param constraint: If specified, this must be a callable that takes the parameter value as an argument and return a boolean indicating whether the constraint has been satisfied. Alternatively, can be a two-tuple with said callable as the first element and a string describing the constraint as the second. :param merge: The default behaviour when setting a value on an object that already has that attribute is to overrided with the new value. If this is set to ``True`` then the two values will be merged instead. The rules by which the values are merged will be determined by the types of the existing and new values -- see ``merge_config_values`` documentation for details. :param aliases: Alternative names for the same configuration point. These are largely for backwards compatibility. :param convert_types: If ``True`` (the default), will automatically convert ``kind`` values from native Python types to WA equivalents. This allows more ituitive interprestation of parameter values, e.g. the string ``"false"`` being interpreted as ``False`` when specifed as the value for a boolean Parameter. """ self.name = identifier(name) if kind is not None and not callable(kind): raise ValueError('Kind must be callable.') if convert_types and kind in self.kind_map: kind = self.kind_map[kind] self.kind = kind self.mandatory = mandatory self.default = default self.override = override self.allowed_values = allowed_values self.description = description if self.kind is None and not self.override: self.kind = str if constraint is not None and not callable(constraint) and not isinstance(constraint, tuple): raise ValueError('Constraint must be callable or a (callable, str) tuple.') self.constraint = constraint self.merge = merge self.aliases = aliases or [] def match(self, name): if name == self.name: return True elif name in self.aliases: return True return False def set_value(self, obj, value=None, check_mandatory=True): if value is None: if self.default is not None: value = self.default elif check_mandatory and self.mandatory: msg = 'No values specified for mandatory parameter "{}" in {}' raise ConfigError(msg.format(self.name, obj.name)) else: try: value = self.kind(value) except (ValueError, TypeError): typename = self.get_type_name() msg = 'Bad value "{}" for {}; must be {} {}' article = get_article(typename) raise ConfigError(msg.format(value, self.name, article, typename)) if value is not None: self.validate_value(obj.name, value) if self.merge and hasattr(obj, self.name): value = merge_config_values(getattr(obj, self.name), value) setattr(obj, self.name, value) def get_type_name(self): typename = str(self.kind) if '\'' in typename: typename = typename.split('\'')[1] elif typename.startswith('<function'): typename = typename.split()[1] return typename def validate(self, obj): value = getattr(obj, self.name, None) if value is not None: self.validate_value(obj.name, value) else: if self.mandatory: msg = 'No value specified for mandatory parameter "{}" in {}.' raise ConfigError(msg.format(self.name, obj.name)) def validate_value(self, name, value): if self.allowed_values: self.validate_allowed_values(name, value) if self.constraint: self.validate_constraint(name, value) def validate_allowed_values(self, name, value): if 'list' in str(self.kind): for v in value: if v not in self.allowed_values: msg = 'Invalid value {} for {} in {}; must be in {}' raise ConfigError(msg.format(v, self.name, name, self.allowed_values)) else: if value not in self.allowed_values: msg = 'Invalid value {} for {} in {}; must be in {}' raise ConfigError(msg.format(value, self.name, name, self.allowed_values)) def validate_constraint(self, name, value): msg_vals = {'value': value, 'param': self.name, 'plugin': name} if isinstance(self.constraint, tuple) and len(self.constraint) == 2: constraint, msg = self.constraint # pylint: disable=unpacking-non-sequence elif callable(self.constraint): constraint = self.constraint msg = '"{value}" failed constraint validation for "{param}" in "{plugin}".' else: raise ValueError('Invalid constraint for "{}": must be callable or a 2-tuple'.format(self.name)) if not constraint(value): raise ConfigError(value, msg.format(**msg_vals)) def __repr__(self): d = copy(self.__dict__) del d['description'] return 'ConfPoint({})'.format(d) __str__ = __repr__ ##################### ### Configuration ### ##################### class Configuration(object): __configuration = [] name = "" # The below line must be added to all subclasses configuration = {cp.name: cp for cp in __configuration} def __init__(self): self._finalized = False for confpoint in self.configuration.itervalues(): confpoint.set_value(self, check_mandatory=False) def set(self, name, value): if self._finalized: raise RuntimeError("Cannot set configuration after it has been finalized.") if name not in self.configuration: raise ConfigError('Unknown {} configuration "{}"'.format(self.name, name)) self.configuration[name].set_value(self, value) def update_config(self, values): for k, v in values.iteritems(): self.set(k, v) def finalize(self): for c in self.configuration.itervalues(): c.validate(self) self._finalized = True # This configuration for the core WA framework class WAConfiguration(Configuration): name = "WA Configuration" __configuration = [ ConfigurationPoint( 'user_directory', description=""" Path to the user directory. This is the location WA will look for user configuration, additional plugins and plugin dependencies. """, kind=str, default=os.path.join(os.path.expanduser('~'), '.workload_automation'), ), ConfigurationPoint( 'plugin_packages', kind=list_of_strings, default=[ 'wlauto.commands', 'wlauto.workloads', 'wlauto.instrumentation', 'wlauto.result_processors', 'wlauto.managers', 'wlauto.resource_getters', ], description=""" List of packages that will be scanned for WA plugins. """, ), ConfigurationPoint( 'plugin_paths', kind=list_of_strings, default=[ 'workloads', 'instruments', 'targets', 'processors', # Legacy 'managers', 'result_processors', ], description=""" List of paths that will be scanned for WA plugins. """, merge=True ), ConfigurationPoint( 'plugin_ignore_paths', kind=list_of_strings, default=[], description=""" List of (sub)paths that will be ignored when scanning ``plugin_paths`` for WA plugins. """, ), ConfigurationPoint( 'assets_repository', description=""" The local mount point for the filer hosting WA assets. """, ), ConfigurationPoint( 'logging', kind=LoggingConfig, default=LoggingConfig.defaults, description=""" WA logging configuration. This should be a dict with a subset of the following keys:: :normal_format: Logging format used for console output :verbose_format: Logging format used for verbose console output :file_format: Logging format used for run.log :color: If ``True`` (the default), console logging output will contain bash color escape codes. Set this to ``False`` if console output will be piped somewhere that does not know how to handle those. """, ), ConfigurationPoint( 'verbosity', kind=int, default=0, description=""" Verbosity of console output. """, ), ConfigurationPoint( # TODO: Needs some format for dates ect/ comes from cfg 'default_output_directory', default="wa_output", description=""" The default output directory that will be created if not specified when invoking a run. """, ), ] configuration = {cp.name: cp for cp in __configuration} dependencies_directory = None # TODO: What was this for? # This is generic top-level configuration for WA runs. class RunConfiguration(Configuration): name = "Run Configuration" __configuration = [ ConfigurationPoint('run_name', kind=str, # TODO: Can only come from an agenda description=''' A descriptive name for this WA run. '''), ConfigurationPoint('output_directory', kind=str, # default=settings.default_output_directory, description=''' The path where WA will output its results. '''), ConfigurationPoint('project', kind=str, description=''' The project this WA run belongs too. '''), ConfigurationPoint('project_stage', kind=dict, description=''' The stage of the project this WA run is from. '''), ConfigurationPoint('execution_order', kind=str, default='by_iteration', allowed_values=None, # TODO: description=''' The order that workload specs will be executed '''), ConfigurationPoint('reboot_policy', kind=str, default='as_needed', allowed_values=RebootPolicy.valid_policies, description=''' How the device will be rebooted during the run. '''), ConfigurationPoint('device', kind=str, description=''' The type of device this WA run will be executed on. '''), ConfigurationPoint('retry_on_status', kind=status_list, default=status_list(['FAILED', 'PARTIAL']), allowed_values=None, # TODO: - can it even be done? description=''' Which iteration results will lead to WA retrying. '''), ConfigurationPoint('max_retries', kind=int, default=3, description=''' The number of times WA will attempt to retry a failed iteration. '''), ] configuration = {cp.name: cp for cp in __configuration} # This is the configuration for WA jobs class JobSpec(Configuration): __configuration = [ ConfigurationPoint('iterations', kind=int, default=1, description=''' How many times to repeat this workload spec '''), ConfigurationPoint('workload_name', kind=str, mandatory=True, aliases=["name"], description=''' The name of the workload to run. '''), ConfigurationPoint('label', kind=str, description=''' Similar to IDs but do not have the uniqueness restriction. If specified, labels will be used by some result processes instead of (or in addition to) the workload name. For example, the csv result processor will put the label in the "workload" column of the CSV file. '''), ConfigurationPoint('runtime_parameters', kind=dict, merge=True, aliases=["runtime_params"], description=''' Rather than configuration for the workload, `runtime_parameters` allow you to change the configuration of the underlying device this particular workload spec. E.g. CPU frequencies, governor ect. '''), ConfigurationPoint('boot_parameters', kind=dict, merge=True, aliases=["boot_params"], description=''' These work in a similar way to runtime_parameters, but they get passed to the device when it reboots. '''), ConfigurationPoint('instrumentation', kind=toggle_set, merge=True, aliases=["instruments"], description=''' The instruments to enable (or disabled using a ~) during this workload spec. '''), ConfigurationPoint('flash', kind=dict, merge=True, description=''' '''), ConfigurationPoint('classifiers', kind=dict, merge=True, description=''' Classifiers allow you to tag metrics from this workload spec to help in post processing them. Theses are often used to help identify what runtime_parameters were used for results when post processing. '''), ] configuration = {cp.name: cp for cp in __configuration} #section id #id mergering id_parts = [] # pointers to entries # This is used to construct the WA configuration tree class JobsConfiguration(object): name = "Jobs Configuration" def __init__(self): self.sections = [] self.workloads = [] self.disabled_instruments = [] self.root_node = Node(global_section()) self._global_finalized = False def set_global_config(self, name, value): if self._global_finalized: raise RuntimeError("Cannot add global config once it has been finalized") if name not in JobSpec.configuration: raise ConfigError('Unknown global configuration "{}"'.format(name)) JobSpec.configuration[name].set_value(self.root_node.config, value, check_mandatory=False) def finalise_global_config(self): for cfg_point in JobSpec.configuration.itervalues(): cfg_point.validate(self.root_node.config) self._global_finalized = True def add_section(self, section, workloads): new_node = self.root_node.add_section(section) for workload in workloads: new_node.add_workload(workload) def add_workload(self, workload): self.root_node.add_workload(workload) def disable_instruments(self, instruments): self.disabled_instruments = instruments def only_run_ids(self): pass def generate_job_specs(self): for leaf in self.root_node.leaves(): workloads = leaf.workloads sections = [leaf.config] for ancestor in leaf.ancestors(): workloads += ancestor.workloads sections.insert(ancestor.config, 0) for workload in workloads: job_spec = JobSpec() for section in sections: job_spec.id_parts.append(section.pop("id")) job_spec.update_config(section) job_spec.id_parts.append(workload.pop("id")) job_spec.update_config(workload) yield job_spec class global_section(object): name = "Global Configuration" def to_pod(self): return self.__dict__.copy() settings = WAConfiguration()