diff options
Diffstat (limited to 'tools/patman/settings.py')
| -rw-r--r-- | tools/patman/settings.py | 180 | 
1 files changed, 176 insertions, 4 deletions
diff --git a/tools/patman/settings.py b/tools/patman/settings.py index 4dda17bf5..084d1b80e 100644 --- a/tools/patman/settings.py +++ b/tools/patman/settings.py @@ -26,6 +26,140 @@ import re  import command  import gitutil +"""Default settings per-project. + +These are used by _ProjectConfigParser.  Settings names should match +the "dest" of the option parser from patman.py. +""" +_default_settings = { +    "u-boot": {}, +    "linux": { +        "process_tags": "False", +    } +} + +class _ProjectConfigParser(ConfigParser.SafeConfigParser): +    """ConfigParser that handles projects. + +    There are two main goals of this class: +    - Load project-specific default settings. +    - Merge general default settings/aliases with project-specific ones. + +    # Sample config used for tests below... +    >>> import StringIO +    >>> sample_config = ''' +    ... [alias] +    ... me: Peter P. <likesspiders@example.com> +    ... enemies: Evil <evil@example.com> +    ... +    ... [sm_alias] +    ... enemies: Green G. <ugly@example.com> +    ... +    ... [sm2_alias] +    ... enemies: Doc O. <pus@example.com> +    ... +    ... [settings] +    ... am_hero: True +    ... ''' + +    # Check to make sure that bogus project gets general alias. +    >>> config = _ProjectConfigParser("zzz") +    >>> config.readfp(StringIO.StringIO(sample_config)) +    >>> config.get("alias", "enemies") +    'Evil <evil@example.com>' + +    # Check to make sure that alias gets overridden by project. +    >>> config = _ProjectConfigParser("sm") +    >>> config.readfp(StringIO.StringIO(sample_config)) +    >>> config.get("alias", "enemies") +    'Green G. <ugly@example.com>' + +    # Check to make sure that settings get merged with project. +    >>> config = _ProjectConfigParser("linux") +    >>> config.readfp(StringIO.StringIO(sample_config)) +    >>> sorted(config.items("settings")) +    [('am_hero', 'True'), ('process_tags', 'False')] + +    # Check to make sure that settings works with unknown project. +    >>> config = _ProjectConfigParser("unknown") +    >>> config.readfp(StringIO.StringIO(sample_config)) +    >>> sorted(config.items("settings")) +    [('am_hero', 'True')] +    """ +    def __init__(self, project_name): +        """Construct _ProjectConfigParser. + +        In addition to standard SafeConfigParser initialization, this also loads +        project defaults. + +        Args: +            project_name: The name of the project. +        """ +        self._project_name = project_name +        ConfigParser.SafeConfigParser.__init__(self) + +        # Update the project settings in the config based on +        # the _default_settings global. +        project_settings = "%s_settings" % project_name +        if not self.has_section(project_settings): +            self.add_section(project_settings) +        project_defaults = _default_settings.get(project_name, {}) +        for setting_name, setting_value in project_defaults.iteritems(): +            self.set(project_settings, setting_name, setting_value) + +    def get(self, section, option, *args, **kwargs): +        """Extend SafeConfigParser to try project_section before section. + +        Args: +            See SafeConfigParser. +        Returns: +            See SafeConfigParser. +        """ +        try: +            return ConfigParser.SafeConfigParser.get( +                self, "%s_%s" % (self._project_name, section), option, +                *args, **kwargs +            ) +        except (ConfigParser.NoSectionError, ConfigParser.NoOptionError): +            return ConfigParser.SafeConfigParser.get( +                self, section, option, *args, **kwargs +            ) + +    def items(self, section, *args, **kwargs): +        """Extend SafeConfigParser to add project_section to section. + +        Args: +            See SafeConfigParser. +        Returns: +            See SafeConfigParser. +        """ +        project_items = [] +        has_project_section = False +        top_items = [] + +        # Get items from the project section +        try: +            project_items = ConfigParser.SafeConfigParser.items( +                self, "%s_%s" % (self._project_name, section), *args, **kwargs +            ) +            has_project_section = True +        except ConfigParser.NoSectionError: +            pass + +        # Get top-level items +        try: +            top_items = ConfigParser.SafeConfigParser.items( +                self, section, *args, **kwargs +            ) +        except ConfigParser.NoSectionError: +            # If neither section exists raise the error on... +            if not has_project_section: +                raise + +        item_dict = dict(top_items) +        item_dict.update(project_items) +        return item_dict.items() +  def ReadGitAliases(fname):      """Read a git alias file. This is in the form used by git: @@ -88,13 +222,45 @@ def CreatePatmanConfigFile(config_fname):      print >>f, "[alias]\nme: %s <%s>" % (name, email)      f.close(); -def Setup(config_fname=''): +def _UpdateDefaults(parser, config): +    """Update the given OptionParser defaults based on config. + +    We'll walk through all of the settings from the parser +    For each setting we'll look for a default in the option parser. +    If it's found we'll update the option parser default. + +    The idea here is that the .patman file should be able to update +    defaults but that command line flags should still have the final +    say. + +    Args: +        parser: An instance of an OptionParser whose defaults will be +            updated. +        config: An instance of _ProjectConfigParser that we will query +            for settings. +    """ +    defaults = parser.get_default_values() +    for name, val in config.items('settings'): +        if hasattr(defaults, name): +            default_val = getattr(defaults, name) +            if isinstance(default_val, bool): +                val = config.getboolean('settings', name) +            elif isinstance(default_val, int): +                val = config.getint('settings', name) +            parser.set_default(name, val) +        else: +            print "WARNING: Unknown setting %s" % name + +def Setup(parser, project_name, config_fname=''):      """Set up the settings module by reading config files.      Args: +        parser:         The parser to update +        project_name:   Name of project that we're working on; we'll look +            for sections named "project_section" as well.          config_fname:   Config filename to read ('' for default)      """ -    settings = ConfigParser.SafeConfigParser() +    config = _ProjectConfigParser(project_name)      if config_fname == '':          config_fname = '%s/.patman' % os.getenv('HOME') @@ -102,11 +268,17 @@ def Setup(config_fname=''):          print "No config file found ~/.patman\nCreating one...\n"          CreatePatmanConfigFile(config_fname) -    settings.read(config_fname) +    config.read(config_fname) -    for name, value in settings.items('alias'): +    for name, value in config.items('alias'):          alias[name] = value.split(',') +    _UpdateDefaults(parser, config)  # These are the aliases we understand, indexed by alias. Each member is a list.  alias = {} + +if __name__ == "__main__": +    import doctest + +    doctest.testmod()  |