You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 

209 lines
7.4 KiB

"""
Author: tikitu
"""
import ConfigParser
from argparse import _SubParsersAction, _StoreAction, _StoreConstAction
import argparse
__version__ = '0.5.1'
def get_config_parser(filename):
config_parser = ConfigParser.SafeConfigParser(allow_no_value=True)
config_parser.read([filename])
return config_parser
def read_config_file(arg_parser, filename):
config_parser = get_config_parser(filename)
read_config_parser(arg_parser, config_parser)
def read_config_parser(arg_parser, config_parser):
ReadConfig(config_parser=config_parser).walk_parser(arg_parser)
def add_config_block_subcommand(arg_parser, subparsers,
config_parser=None,
only_non_defaults=False):
"""
Add a subcommand "config-block" to the arg_parser, to be used as follows:
In myprog.py:
subparsers = arg_parser.add_subparsers(..., dest='command')
add_config_block_subcommand(arg_parser, subparsers)
# ...
parsed_args = arg_parser.parse_args()
if parsed_args.command == 'config':
print parsed_args.func(parsed_args)
exit(0)
On the commandline:
$ myprog.py config default --username tikitu --secret xyzzy
[default]
username: tikitu
secret: xyzzy
:param arg_parser:
:param config_parser:
:param dest:
:param only_non_defaults:
:return: None
"""
config_command_parser = subparsers.add_parser('config')
config_command_parser.add_argument('block')
config_command_parser.add_argument('commandline', nargs=argparse.REMAINDER)
def handle_args(orig_parsed_args):
if config_parser is not None:
read_config_parser(arg_parser, config_parser)
args_for_commandline = list(orig_parsed_args.commandline)
if orig_parsed_args.block == 'default':
subparsers.add_parser('dummy-command')
args_for_commandline.append('dummy-command')
else:
args_for_commandline.insert(0, orig_parsed_args.block)
parsed_args = arg_parser.parse_args(args_for_commandline)
return generate_config(arg_parser, parsed_args,
section=orig_parsed_args.block,
only_non_defaults=only_non_defaults)
config_command_parser.set_defaults(func=handle_args)
def generate_config(arg_parser, parsed_args, section='default',
only_non_defaults=False):
action = GenerateConfig(parsed_args, section,
only_non_defaults=only_non_defaults)
action.walk_parser(arg_parser)
return action.contents
class ArgParserWalker(object):
def start_section(self, section_name):
raise NotImplementedError()
def end_section(self):
raise NotImplementedError()
def process_parser_action(self, action, is_store_const=False):
raise NotImplementedError()
def walk_parser(self, arg_parser):
try:
self.start_section('default')
for action in arg_parser._actions:
if isinstance(action, _StoreAction):
self.process_parser_action(action)
elif isinstance(action, _StoreConstAction):
self.process_parser_action(action, is_store_const=True)
elif isinstance(action, _SubParsersAction):
for command, sub_parser in action.choices.items():
self.start_section(command)
for sub_action in sub_parser._actions:
self.process_parser_action(sub_action)
self.end_section()
self.end_section()
except DefaultError as e:
arg_parser.error(
u'[{section_name}] config option "{option_string}" '
u'must be {type_transformer}() value, got: {value}'.format(
section_name=e.section_name,
option_string=e.option_string,
type_transformer=e.type_transformer.__name__,
value=e.value
))
class GenerateConfig(ArgParserWalker):
def __init__(self, parsed_args, section, only_non_defaults=False):
self.parsed_args = parsed_args
self._contents = []
self._only_non_defaults = only_non_defaults
self._section = section
self._in_sections = []
def start_section(self, section_name):
self._in_sections.append(section_name)
if section_name == self._section:
if self._contents:
self._contents.append(u'')
self._contents.append(u'[{0}]'.format(section_name))
def end_section(self):
self._in_sections.pop()
@property
def contents(self):
return u'\n'.join(self._contents + [u''])
def process_parser_action(self, action, is_store_const=False):
if self._in_sections[-1] != self._section:
return
# take the longest string, likely the most informative
action_name = list(action.option_strings)
action_name.sort(key=lambda s: len(s), reverse=True)
action_name = _convert_option_string(action_name[0])
action_value = getattr(self.parsed_args, action.dest, None)
if self._only_non_defaults and action_value == action.default:
action_value = None
if action_value is not None:
if is_store_const:
self._contents.append(action_name)
else:
self._contents.append(u'{action_name}: {default_value}'.format(
action_name=action_name,
default_value=action_value, # hope it prints as wanted...
))
class ReadConfig(ArgParserWalker):
def __init__(self, config_parser=None):
self.sections = []
self.config_parser = config_parser
def start_section(self, section_name):
self.sections.append(section_name)
def end_section(self):
self.sections.pop()
@property
def current_section(self):
return self.sections[-1] if self.sections else None
def process_parser_action(self, action, is_store_const=False):
for option_string in action.option_strings:
option_string = _convert_option_string(option_string)
if self.config_parser.has_option(self.current_section,
option_string):
if is_store_const:
action.default = action.const
else:
value = self.config_parser.get(self.current_section,
option_string)
type_transformer = (action.type if action.type is not None
else lambda x: x)
try:
action.default = type_transformer(value)
except:
raise DefaultError(self.current_section,
option_string,
value,
type_transformer)
action.required = False
class DefaultError(Exception):
def __init__(self, section_name, option_string, value, type_transformer):
self.section_name = section_name
self.option_string = option_string
self.value = value
self.type_transformer = type_transformer
def _convert_option_string(op_s):
return op_s.lstrip('-').replace('-', '_')