Просто голова, так как я потратил слишком много часов на это. Я продолжал получать это исключение, хотя я назвал startForeground(..)
первым делом в onCreate(..)
. В конце концов я обнаружил, что проблема была вызвана использованием NOTIFICATION_ID = 0
. Использование любого другого значения, похоже, исправить это.
Согласно документации, --version
с action='version'
(а не с action='store_true'
) автоматически печатает номер версии:
parser.add_argument('--version', action='version', version='%(prog)s 2.0')
Хотя ответ @ eumiro отвечает на параметр --version
, он может это сделать только потому, что это особый случай для optparse. Чтобы разрешить общие вызовы:
prog
prog --verbose
prog --verbose main
prog --verbose db
и prog --version
работают так же, как prog --verbose main
(и prog main --verbose
), вы можете добавить метод в Argumentparser и вызвать это с именем по умолчанию subparser, непосредственно перед вызовом parse_args()
:
import argparse
import sys
def set_default_subparser(self, name, args=None):
"""default subparser selection. Call after setup, just before parse_args()
name: is the name of the subparser to call by default
args: if set is the argument list handed to parse_args()
, tested with 2.7, 3.2, 3.3, 3.4
it works with 2.6 assuming argparse is installed
"""
subparser_found = False
for arg in sys.argv[1:]:
if arg in ['-h', '--help']: # global help if no subparser
break
else:
for x in self._subparsers._actions:
if not isinstance(x, argparse._SubParsersAction):
continue
for sp_name in x._name_parser_map.keys():
if sp_name in sys.argv[1:]:
subparser_found = True
if not subparser_found:
# insert default in first position, this implies no
# global options without a sub_parsers specified
if args is None:
sys.argv.insert(1, name)
else:
args.insert(0, name)
argparse.ArgumentParser.set_default_subparser = set_default_subparser
def do_main(args):
print 'main verbose', args.verbose
def do_db(args):
print 'db verbose:', args.verbose
parser = argparse.ArgumentParser()
parser.add_argument('--verbose', action='store_true')
parser.add_argument('--version', action='version', version='%(prog)s 2.0')
subparsers = parser.add_subparsers()
sp = subparsers.add_parser('main')
sp.set_defaults(func=do_main)
sp.add_argument('--verbose', action='store_true')
sp = subparsers.add_parser('db')
sp.set_defaults(func=do_db)
parser.set_default_subparser('main')
args = parser.parse_args()
if hasattr(args, 'func'):
args.func(args)
Метод set_default_subparser()
является частью пакета ruamel.std.argparse .
Как обсуждалось в http://bugs.python.org/issue9253 (argparse: необязательные подпараметры), как и в Python 3.3, подпарамеры теперь являются необязательными. Это был непреднамеренный результат изменения того, как parse_args
проверял требуемые аргументы.
Я нашел выдумку, которая восстанавливает предыдущее поведение (требуемые подпарамеры), явно устанавливая атрибут required
для subparsers
action.
parser = ArgumentParser(prog='test')
subparsers = parser.add_subparsers()
subparsers.required = True # the fudge
subparsers.dest = 'command'
subparser = subparsers.add_parser("foo", help="run foo")
parser.parse_args()
См. эту проблему для получения более подробной информации. Я ожидаю, что если и когда эта проблема будет исправлена, по умолчанию потребуются подпарамеры, с некоторой опцией установить свой атрибут required
в False
. Но существует большое отставание от патчей argparse
.
Пока мы дождитесь доставки этой функции , мы можем использовать такой код:
# Make sure that main is the default sub-parser
if '-h' not in sys.argv and '--help' not in sys.argv:
if len(sys.argv) < 2:
sys.argv.append('main')
if sys.argv[1] not in ('main', 'test'):
sys.argv = [sys.argv[0], 'main'] + sys.argv[1:]
FWIW, я тоже столкнулся с этим и решил «решить» его, не используя подпараметров (у меня уже была моя собственная система для печати справки, поэтому ничего там не терялось).
Вместо этого , Я делаю это:
parser.add_argument("command", nargs="?",
help="name of command to execute")
args, subcommand_args = parser.parse_known_args()
... и затем подкоманда создает свой собственный синтаксический анализатор (аналогичный подпараметру), который работает только на subcommand_args
.
Это, похоже, реализует основную идею дополнительного подпарамера. Мы разбираем стандартные аргументы, применимые ко всем подкомандам. Затем, если что-то осталось, мы вызываем парсер на остальных. Основными аргументами являются родительский элемент подкоманды, поэтому -h отображается правильно. Я планирую ввести интерактивное приглашение, если нет подкоманд.
import argparse
p1 = argparse.ArgumentParser( add_help = False )
p1.add_argument( ‘–flag1′ )
p2 = argparse.ArgumentParser( parents = [ p1 ] )
s = p2.add_subparsers()
p = s.add_parser( ‘group’ )
p.set_defaults( group=True )
( init_ns, remaining ) = p1.parse_known_args( )
if remaining:
p2.parse_args( args = remaining, namespace=init_ns )
else:
print( ‘Enter interactive loop’ )
print( init_ns )
Да, я только что проверил svn
, который используется в качестве примера объекта в документации add_subparsers()
, и он поддерживает только «-version» в основной команде:
python zacharyyoung$ svn log --version
Subcommand 'log' doesn't accept option '--version'
Type 'svn help log' for usage.
Still:
# create common parser
parent_parser = argparse.ArgumentParser('parent', add_help=False)
parent_parser.add_argument('--version', action='version', version='%(prog)s 2.0')
# create the top-level parser
parser = argparse.ArgumentParser(parents=[parent_parser])
subparsers = parser.add_subparsers()
# create the parser for the "foo" command
parser_foo = subparsers.add_parser('foo', parents=[parent_parser])
Что дает:
python zacharyyoung$ ./arg-test.py --version
arg-test.py 2.0
python zacharyyoung$ ./arg-test.py foo --version
arg-test.py foo 2.0