Skip to content

Commit

Permalink
Unpack nested IteratorAggregate objects for Count
Browse files Browse the repository at this point in the history
This allows generators (#2149) and internal Traversables (#2642) to be properly detected before attempting to restore the Iterator's position after counting (#1125).
  • Loading branch information
jmikola authored and sebastianbergmann committed Jun 1, 2017
1 parent 7727a49 commit a9439d3
Show file tree
Hide file tree
Showing 3 changed files with 42 additions and 4 deletions.
8 changes: 4 additions & 4 deletions src/Framework/Constraint/Count.php
Original file line number Diff line number Diff line change
Expand Up @@ -56,12 +56,12 @@ protected function getCountOf($other)
}

if ($other instanceof Traversable) {
if ($other instanceof IteratorAggregate) {
$iterator = $other->getIterator();
} else {
$iterator = $other;
while ($other instanceof IteratorAggregate) {
$other = $other->getIterator();
}

$iterator = $other;

if ($iterator instanceof Generator) {
return $this->getCountOfGenerator($iterator);
}
Expand Down
19 changes: 19 additions & 0 deletions tests/Framework/Constraint/CountTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -25,9 +25,11 @@ public function testCount()
$countConstraint = new Count(2);
$it = new \TestIterator([1, 2]);
$ia = new \TestIteratorAggregate($it);
$ia2 = new \TestIteratorAggregate2($ia);

$this->assertTrue($countConstraint->evaluate($it, '', true));
$this->assertTrue($countConstraint->evaluate($ia, '', true));
$this->assertTrue($countConstraint->evaluate($ia2, '', true));
}

public function testCountDoesNotChangeIteratorKey()
Expand Down Expand Up @@ -78,6 +80,23 @@ public function testCountDoesNotChangeIteratorKey()
$it->next();
$countConstraint->evaluate($ia, '', true);
$this->assertFalse($it->valid());

// test with nested IteratorAggregate
$it = new \TestIterator([1, 2]);
$ia = new \TestIteratorAggregate($it);
$ia2 = new \TestIteratorAggregate2($ia);

$countConstraint = new Count(2);
$countConstraint->evaluate($ia2, '', true);
$this->assertEquals(1, $it->current());

$it->next();
$countConstraint->evaluate($ia2, '', true);
$this->assertEquals(2, $it->current());

$it->next();
$countConstraint->evaluate($ia2, '', true);
$this->assertFalse($it->valid());
}

public function testCountGeneratorsDoNotRewind()
Expand Down
19 changes: 19 additions & 0 deletions tests/_files/TestIteratorAggregate2.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
<?php

/* This class is used for testing a chain of IteratorAggregate objects, since
* PHP does allow IteratorAggregate::getIterator() to return an instance of the
* same class. */
class TestIteratorAggregate2 implements IteratorAggregate
{
private $traversable;

public function __construct(\Traversable $traversable)
{
$this->traversable = $traversable;
}

public function getIterator()
{
return $this->traversable;
}
}

0 comments on commit a9439d3

Please sign in to comment.