argparse missing optparse capabilities?

Ian Kelly ian.g.kelly at gmail.com
Thu Jan 5 13:46:02 EST 2012


On Thu, Jan 5, 2012 at 11:14 AM, Ian Kelly <ian.g.kelly at gmail.com> wrote:
> On Thu, Jan 5, 2012 at 1:05 AM, rurpy at yahoo.com <rurpy at yahoo.com> wrote:
>> I have optparse code that parses a command line containing
>> intermixed positional and optional arguments, where the optional
>> arguments set the context for the following positional arguments.
>> For example,
>>
>>  myprogram.py arg1 -c33 arg2 arg3 -c44 arg4
>>
>> 'arg1' is processed in a default context, 'args2' and 'arg3' in
>> context '33', and 'arg4' in context '44'.
>>
>> I am trying to do the same using argparse but it appears to be
>> not doable in a documented way.
>>
>> Here is the working optparse code (which took 30 minutes to write
>> using just the optparse docs):
>>
>>  import optparse
>>  def append_with_pos (option, opt_str, value, parser):
>>        if getattr (parser.values, option.dest, None) is None:
>>            setattr (parser.values, option.dest, [])
>>        getattr (parser.values, option.dest).append ((value, len
>> (parser.largs)))
>>  def opt_parse():
>>        p = optparse.OptionParser()
>>        p.add_option ("-c", type=int,
>>            action='callback', callback=append_with_pos)
>>        opts, args = p.parse_args()
>>        return args, opts
>>  if __name__ == '__main__':
>>        args, opts = opt_parse()
>>        print args, opts
>>
>> Output from the command line above:
>>  ['arg1', 'arg2', 'arg3', 'arg4'] {'c': [(33, 1), (44, 3)]}
>> The -c values are stored as (value, arglist_position) tuples.
>>
>> Here is an attempt to convert to argparse using the guidelines
>> in the argparse docs:
>>
>>  import argparse
>>  class AppendWithPos (argparse.Action):
>>    def __call__ (self, parser, namespace, values,
>> option_string=None):
>>        if getattr (namespace, self.dest, None) is None:
>>            setattr (namespace, self.dest, [])
>>        getattr (namespace, self.dest).extend ((values, len
>> (parser.largs)))
>>  def arg_parse():
>>        p = argparse.ArgumentParser (description='description')
>>        p.add_argument ('src', nargs='*')
>>        p.add_argument ('-c', type=int, action=AppendWithPos)
>>        opts = p.parse_args()
>>        return opts
>>  if __name__ == '__main__':
>>        opts = arg_parse()
>>        print opts
>>
>> This fails with,
>>  AttributeError: 'ArgumentParser' object has no attribute 'largs'
>> and of course, the argparse.parser is not documented beyond how
>> to instantiate it.  Even were that not a problem, argparse complains
>> about "unrecognised arguments" for any positional arguments that
>> occur after an optional one.  I've been farting with this code for
>> a day now.
>>
>> Any suggestions on how I can convince argparse to do what optparse
>> does easily will be very welcome.  (I tried parse_known_args() but
>> that breaks help and requires me to detect truly unknown arguments.)
>>
>> (Python 2.7.1 if it matters and apologies if Google mangles
>> the formatting of this post.)
>
> You have the namespace object in your custom action.  Instead of
> "len(parser.largs)", couldn't you just do "len(namespace.src)"?

Sorry, I missed the second part of that.  You seem to be right, as far
as I can tell from tinkering with it, all the positional arguments
have to be in a single group.  If you have some positional arguments
followed by an option followed by more positional arguments, and any
of the arguments have a loose nargs quantifier ('?' or '*' or '+'),
then you get an error.



More information about the Python-list mailing list