如何在Python脚本中使用argparse
与shell标签完成合作?
#!/usr/bin/env python
import argparse
def main(**args):
pass
if __name__ == '__main__':
parser = argparse.ArgumentParser()
parser.add_argument('positional', choices=['spam', 'eggs'])
parser.add_argument('--optional', choices=['foo1', 'foo2', 'bar'])
args = parser.parse_args()
main(**vars(args))
在.py文件上设置可执行标志后,预期结果应为:
$ ./example.py sp<tab>
-> completes to "./example.py spam"
$ ./example.py --op<tab>
-> completes to "./example.py --optional"
$ ./example.py --optional b<tab>
-> completes to "./example.py --optional bar"
$ ./example.py --optional f<tab>
-> completes to "./example.py --optional foo"
and, additionally, prints "foo1 foo2" choices on stdout on a new line
答案 0 :(得分:48)
请看安德烈·基斯柳克的argcomplete。
使用以下命令安装:
pip install argcomplete
导入模块并在调用parser.parse_args()
之前在源代码中添加一行:
#!/usr/bin/env python
import argparse as ap
import argcomplete
def main(**args):
pass
if __name__ == '__main__':
parser = ap.ArgumentParser()
parser.add_argument('positional', choices=['spam', 'eggs'])
parser.add_argument('--optional', choices=['foo1', 'foo2', 'bar'])
argcomplete.autocomplete(parser)
args = parser.parse_args()
main(**vars(args))
并确保bash知道此脚本,您可以使用
eval "$(register-python-argcomplete your_script)"
您应该将该行放在~/.bashrc
中,或者按照argcomplete的文档操作并激活'全局'完成。
之后,您完成按要求工作。
这种方法的工作方式是eval行创建一个使用_python_argcomlete
注册的函数complete
。 (运行register-python-argcomplete your_script
只是看看什么被eval编入bash)。自动完成功能查找由bash完成机制设置的环境变量,以查看是否需要执行操作。如果它起作用,它将退出程序。如果它不起作用,这是对程序的正常调用,该函数什么都不做,程序的正常流程继续。
答案 1 :(得分:3)
要使自动完成工作,您需要一个bash函数来生成可能的选项,然后您需要运行complete -F <function_name> <program_name>
这样做的最佳方法是让程序根据自己的解析算法生成完成函数,以避免重复。但是,快速浏览一下argparse,我找不到一种方法来访问它的内部结构,但我建议你去寻找它。
这是一个用于上述程序的bash函数:
function _example_auto() {
local cur=${COMP_WORDS[COMP_CWORD]}
local prev=${COMP_WORDS[COMP_CWORD-1]}
case "$prev" in
--optional )
COMPREPLY=( $(compgen -W "foo1 foo2 bar" -- $cur) )
return 0
;;
*)
COMPREPLY=( $(compgen -W "--optional spam eggs" -- $cur) )
return 0
;;
esac
}