15

我想为(相当复杂的)Bash 完成脚本编写单元测试,最好使用 Python - 只是以编程方式获取 Bash 完成值的东西。测试应如下所示:

def test_completion():
  # trigger_completion should return what a user should get on triggering 
  # Bash completion like this: 'pbt createkvm<TAB>' 
  assert trigger_completion('pbt createkvm') == "module1 module2 module3" 

如何以编程方式模拟 Bash 完成以检查我的工具的测试套件内的完成值?

4

2 回答 2

9

假设您在名为 的文件中有一个 bash 完成脚本asdf-completion,其中包含:

_asdf() {
COMPREPLY=()
local cur prev
cur=$(_get_cword)
COMPREPLY=( $( compgen -W "one two three four five six" -- "$cur") )
return 0
}    
complete -F _asdf asdf

这使用 shell 函数_asdf来为虚构的asdf命令提供补全。如果我们设置正确的环境变量(来自 bash 手册页),那么我们可以获得相同的结果,即将潜在扩展放置到COMPREPLY变量中。这是在单元测试中执行此操作的示例:

import subprocess
import unittest

class BashTestCase(unittest.TestCase):
    def test_complete(self):
        completion_file="asdf-completion"
        partial_word="f"
        cmd=["asdf", "other", "arguments", partial_word]
        cmdline = ' '.join(cmd)

        out = subprocess.Popen(['bash', '-i', '-c',
            r'source {compfile}; COMP_LINE="{cmdline}" COMP_WORDS=({cmdline}) COMP_CWORD={cword} COMP_POINT={cmdlen} $(complete -p {cmd} | sed "s/.*-F \\([^ ]*\\) .*/\\1/") && echo ${{COMPREPLY[*]}}'.format(
                compfile=completion_file, cmdline=cmdline, cmdlen=len(cmdline), cmd=cmd[0], cword=cmd.index(partial_word)
                )],
            stdout=subprocess.PIPE)
        stdout, stderr = out.communicate()
        self.assertEqual(stdout, "four five\n")

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

这应该适用于任何使用 的完成-F,但也可能适用于其他人。

je4d 使用的评论expect对于更完整的测试来说是一个很好的评论。

于 2012-02-29T19:01:02.187 回答
1

bonsaiviking 的解决方案几乎对我有用。我不得不更改 bash 字符串脚本。我添加了一个额外的';' 分隔符到已执行的 bash 脚本,否则执行将无法在 Mac OS X 上运行。不太清楚为什么。

我还稍微概括了各种 COMP_ 参数的初始化,以处理我最终遇到的各种情况。

最终的解决方案是一个辅助类来测试从 python 完成的 bash,这样上面的测试就可以写成:

from completion import BashCompletionTest

class AdsfTestCase(BashCompletionTest):
    def test_orig(self):
        self.run_complete("other arguments f", "four five")

    def run_complete(self, command, expected):
        completion_file="adsf-completion"
        program="asdf"
        super(AdsfTestCase, self).run_complete(completion_file, program, command, expected)


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

完成库位于https://github.com/lacostej/unity3d-bash-completion/blob/master/lib/completion.py下

于 2012-04-21T14:24:27.873 回答