如何模拟虚拟二进制文件,以便测试exec()/system()/passthru()函数输出


How do you mock a virtual binary file so that exec() / system() / passthru() function output can be tested?

我有一个有趣的问题,在互联网上搜索了一下,但还没有找到答案。

我在一家不允许员工使用OOP的公司工作,这有点荒谬,但工作经验很有价值。

考虑以下功能:

function get_setting_values_from_file( $parameter )
{
    exec("/usr/var/binary --options $parameter", $output, $return);
    $settings = file( $output[0] );
    foreach( $settings as $setting ) {
        if( strstr( $setting, "color") ) {
            $setting = explode( ":", $setting );
            return $setting[1];
        }
    }
    return false;
}

我需要对类似的功能进行单元测试。我目前正在使用phpUnit进行测试,并使用vfsStream库来模拟文件系统,但当我在无法访问实际系统的情况下进行开发时,如何模拟对exec("/usr/var/binary --options $parameter", $output, $return)的调用?处理这样的测试用例的推荐方法是什么?

感谢所有反馈。

您可以使用函数mock库来模拟exec()。我为您制作了一个(php-mock),它要求您使用名称空间

namespace foo;
use phpmock'phpunit'PHPMock;
class ExecTest extends 'PHPUnit_Framework_TestCase
{
    use PHPMock;
    public function testExec()
    {
        $mock = $this->getFunctionMock(__NAMESPACE__, "exec");
        $mock->expects($this->once())->willReturnCallback(
            function ($command, &$output, &$return_var) {
                $this->assertEquals("foo", $command);
                $output = "failure";
                $return_var = 1;
            }
        );
        exec("foo", $output, $return_var);
        $this->assertEquals("failure", $output);
        $this->assertEquals(1, $return_var);
    }
}

只需模拟此函数,即可返回您试图进入$settings的文本。您不需要调用可执行文件,只需创建文件或返回即可。

例如,假设函数get_setting_values_from_file()以数组的形式返回设置,您可以简单地模拟测试中的函数,以数组的方式返回设置。创建一个测试存根来模拟包含get_setting_values_from_file()方法的对象,并让该mock简单地返回测试假设的FALSE、1或2。

$stub = $this->getMock('GetSettingsClass');
    $stub->expects($this->any())
         ->method('get_settings_from_file')
         ->will($this->returnValue(0));

这来自PHPUnit手册->http://phpunit.de/manual/3.8/en/test-doubles.html#test-双短截线

或者,您甚至可以绕过调用,通过创建数组并将其传递给这些函数,简单地测试处理返回的函数/代码。主代码中的假定示例:

...
$settings = get_setting_values_from_file( 'UserType' );
$UserType = get_user_type($settings);
return $UserType;
function get_user_type($settings)
{
    if($settings !== FALSE)         // Returned from your function if parameter is not found
    {
        switch($settings)
        {
            case 1:
                return 'User';      // Best to use Constants, but for example here only
                break;
            case 2:
                return 'Admin';
                break;
            ...
         }
    }
    else
    {
        return FALSE;
    }
}

现在,在你的测试中,你可以简单地

$this->assertFalse(get_user_type(FALSE, 'Ensure not found data is handled properly as FALSE is returned');
$this->assertEqual('User', get_user_type(1), 'Test UserType=1');
$this->assertEqual('Admin', get_user_type(1), 'Test UserType=2');
...

这些工作原理是,代码不调用必须模拟从操作系统读取的函数,而是通过调用处理设置返回值的函数来处理所有预期的返回。在这里,您只需假设函数"get_setting_values_from_file()"的返回,而不需要文件或任何mock。

然而,这并没有测试从文件中读取,我会在另一个测试中使用setUp和tearDown来实际创建一个具有所需值的文件(fopen/fwrite),然后调用您的函数并确保它返回预期值。

我希望这有助于解释我的想法。