39

给定以下课程:

<?php
class Example {
    private $Other;

    public function __construct ($Other)
    {
        $this->Other = $Other;
    }

    public function query ()
    {   
        $params = array(
            'key1' => 'Value 1'
            , 'key2' => 'Value 2'
        );

        $this->Other->post($params);
    }
}

这个测试用例:

<?php
require_once 'Example.php';
require_once 'PHPUnit/Framework.php';

class ExampleTest extends PHPUnit_Framework_TestCase {

    public function test_query_key1_value ()
    {   
        $Mock = $this->getMock('Other', array('post'));

        $Mock->expects($this->once())
              ->method('post')
              ->with(YOUR_IDEA_HERE);

        $Example = new Example($Mock);
        $Example->query();
    }

如何验证$params(这是一个数组)并传递给$Other->post()包含一个名为“key1”的键,其值为“值 1”?

我不想验证所有的数组——这只是一个示例代码,在实际代码中,传递的数组有更多的值,我只想验证其中的一个键/值对。

$this->arrayHasKey('keyname')可以使用它来验证密钥是否存在。

还有$this->contains('Value 1'), 可以用来验证数组是否有这个值。

我什至可以将这两者与$this->logicalAnd. 但这当然不会产生预期的结果。

到目前为止,我一直在使用 returnCallback,捕获整个 $params,然后对其进行断言,但也许还有另一种方法可以做我想做的事?

4

6 回答 6

53

$this->arrayHasKey('keyname');方法存在,但它的名称是assertArrayHasKey

// In your PHPUnit test method
$hi = array(
    'fr' => 'Bonjour',
    'en' => 'Hello'
);

$this->assertArrayHasKey('en', $hi);    // Succeeds
$this->assertArrayHasKey('de', $hi);    // Fails
于 2014-06-05T13:24:36.770 回答
16

代替创建可重用的约束类,我能够使用 PHPUnit 中现有的回调约束来断言数组键的值。在我的用例中,我需要检查模拟方法(MongoCollection::ensureIndex(),如果有人好奇的话)的第二个参数中的数组值。这是我想出的:

$mockedObject->expects($this->once())
    ->method('mockedMethod')
    ->with($this->anything(), $this->callback(function($o) {
        return isset($o['timeout']) && $o['timeout'] === 10000;
    }));

回调约束在其构造函数中需要一个可调用对象,并在评估期间简单地调用它。断言通过或失败取决于可调用对象是返回真还是假。

对于一个大型项目,我当然建议创建一个可重用的约束(如上述解决方案)或请求将PR #312合并到 PHPUnit 中,但这确实满足了一次性需求。很容易看出回调约束对更复杂的断言也可能有用。

于 2013-02-12T16:40:23.593 回答
8

我最终基于属性一创建了自己的约束类

<?php
class Test_Constraint_ArrayHas extends PHPUnit_Framework_Constraint
{
    protected $arrayKey;

    protected $constraint;

    protected $value;

    /**
     * @param PHPUnit_Framework_Constraint $constraint
     * @param string                       $arrayKey
     */
    public function __construct(PHPUnit_Framework_Constraint $constraint, $arrayKey)
    {
        $this->constraint  = $constraint;
        $this->arrayKey    = $arrayKey;
    }


    /**
     * Evaluates the constraint for parameter $other. Returns TRUE if the
     * constraint is met, FALSE otherwise.
     *
     * @param mixed $other Value or object to evaluate.
     * @return bool
     */
    public function evaluate($other)
    {
        if (!array_key_exists($this->arrayKey, $other)) {
            return false;
        }

        $this->value = $other[$this->arrayKey];

        return $this->constraint->evaluate($other[$this->arrayKey]);
    }

    /**
     * @param   mixed   $other The value passed to evaluate() which failed the
     *                         constraint check.
     * @param   string  $description A string with extra description of what was
     *                               going on while the evaluation failed.
     * @param   boolean $not Flag to indicate negation.
     * @throws  PHPUnit_Framework_ExpectationFailedException
     */
    public function fail($other, $description, $not = FALSE)
    {
        parent::fail($other[$this->arrayKey], $description, $not);
    }


    /**
     * Returns a string representation of the constraint.
     *
     * @return string
     */
    public function toString ()
    {
        return 'the value of key "' . $this->arrayKey . '"(' . $this->value . ') ' .  $this->constraint->toString();
    }


    /**
     * Counts the number of constraint elements.
     *
     * @return integer
     */
    public function count ()
    {
        return count($this->constraint) + 1;
    }


    protected function customFailureDescription ($other, $description, $not)
    {
        return sprintf('Failed asserting that %s.', $this->toString());
    }

它可以这样使用:

 ... ->with(new Test_Constraint_ArrayHas($this->equalTo($value), $key));
于 2009-09-29T08:25:51.240 回答
3

如果您希望对参数进行一些复杂的测试,并且还有有用的消息和比较,则始终可以选择在回调中放置断言。

例如

$clientMock->expects($this->once())->method('post')->with($this->callback(function($input) {
    $this->assertNotEmpty($input['txn_id']);
    unset($input['txn_id']);
    $this->assertEquals($input, array(
        //...
    ));
    return true;
}));

请注意,回调返回 true。否则,它总是会失败。

于 2016-02-25T17:27:28.513 回答
0

如果您不介意在密钥不存在时出现错误(而不是失败),那么您可以使用以下内容。

$this->assertEquals('Value 1', $params['key1']);
于 2021-11-10T13:52:31.960 回答
-4

对不起,我不会说英语。

我认为您可以使用 array_key_exists 函数测试数组中是否存在键,并且可以使用 array_search 测试该值是否存在

例如:

function checkKeyAndValueExists($key,$value,$arr){
    return array_key_exists($key, $arr) && array_search($value,$arr)!==false;
}

使用!==是因为 array_search 返回该值的键(如果存在)并且它可能为 0。

于 2009-09-28T15:51:49.110 回答