2

在一个文件中(比如说parser.py)我有:

import argparse

def parse_cmdline(cmdline=None):
    parser = argparse.ArgumentParser()
    parser.add_argument('--first-param',help="Does foo.")
    parser.add_argument('--second-param',help="Does bar.")

    if cmdline is not None:
        args = parser.parse_args(cmdline)
    else:
        args = parser.parse_args()

    return vars(args)

if __name__=='__main__':
    print parse_cmdline()

果然,当从命令行调用它时,它可以工作并给我几乎我所期望的:

$ ./parser.py --first-param 123 --second-param 456
{'first_param': '123', 'second_param': '456'}

但后来我想要unittest它,因此我写了一个test_parser.py文件:

import unittest
from parser import parse_cmdline

class TestParser(unittest.TestCase):
    def test_parse_cmdline(self):
        parsed = parse_cmdline("--first-param 123 --second-param 456")

        self.assertEqual(parsed['first_param'],'123')
        self.assertEqual(parsed['second_param'],'456')

if __name__ == '__main__':
    unittest.main()

然后我收到以下错误:

usage: test_parser.py [-h] [--first-param FIRST_PARAM]
                      [--second-param SECOND_PARAM]
test_parser.py: error: unrecognized arguments: - - f i r s t - p a r a m   1 2 3   - - s e c o n d - p a r a m   4 5 6
E
======================================================================
ERROR: test_parse_cmdline (__main__.TestParser)
----------------------------------------------------------------------
Traceback (most recent call last):
  File "./test_parser.py", line 8, in test_parse_cmdline
    parsed = parse_cmdline("--first-param 123 --second-param 456")
  File "/home/renan/test_argparse/parser.py", line 12, in parse_cmdline
    args = parser.parse_args(cmdline)
  File "/usr/lib/python2.7/argparse.py", line 1691, in parse_args
    self.error(msg % ' '.join(argv))
  File "/usr/lib/python2.7/argparse.py", line 2361, in error
    self.exit(2, _('%s: error: %s\n') % (self.prog, message))
  File "/usr/lib/python2.7/argparse.py", line 2349, in exit
    _sys.exit(status)
SystemExit: 2

----------------------------------------------------------------------
Ran 1 test in 0.004s

FAILED (errors=1)

可以看出,我指定的命令行(--first-param 123 --second-param 456)变成 - - f i r s t - p a r a m 1 2 3 - - s e c o n d - p a r a m 4 5 6了(每个字符用空格隔开)。

我不明白为什么:我做错了什么?

4

2 回答 2

8

argparse想要一个“参数向量”——即一个单独的参数列表——而不是一个“命令行”字符串。

只是打电话split不会解决问题。例如,来自 shell 的这个命令行:

python script.py --first-param '123 456' --second-param 789

... 会正确地给你123 456and 789,但是这行代码:

parse_cmdline("--first-param '123 456' --second-param 789")

将不会; 它会给你'123and ,还有一个它不知道如何处理789的额外内容。456'

事实上,即使这样也是错误的:

parse_cmdline("--first-param '123' --second-param 789")

......因为你会得到'123'而不是123.


有两种方法可以解决这个问题。

首先,如果您知道要传递的内容,则可以首先将其作为列表而不是字符串传递,并且您无需担心繁琐的引用和拆分细节:

parse_cmdline(["--first-param", "123 456", "--second-param", "789"])

或者,如果您不确切知道要通过什么,您只知道它在外壳上的样子,您可以使用shlex

if cmdline is not None:
    args = parser.parse_args(shlex.split(cmdline))

……现在 Python 将像标准 Unix shell 一样分割你的命令行,将 '123 456作为单个参数。

于 2013-08-20T01:08:39.123 回答
1

回答自己(几分钟后我意识到我的错误):

我应该

if cmdline is not None:
    args = parser.parse_args(cmdline.split())
else:
    args = parser.parse_args()

现在测试正确通过了!

于 2013-08-20T00:41:50.397 回答