See Release Notes
Long Term Support Release
<?php /*< * Copyright 2017 MongoDB, Inc.> * Copyright 2017-present MongoDB, Inc.* * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at *< * http://www.apache.org/licenses/LICENSE-2.0> * https://www.apache.org/licenses/LICENSE-2.0* * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace MongoDB\Model; use Countable;< use Generator;use Iterator;> use IteratorIterator; use Traversable; > use ReturnTypeWillChange;use function count;>use function current;< use function key;use function next; use function reset; /** * Iterator for wrapping a Traversable and caching its results. * * By caching results, this iterators allows a Traversable to be counted and * rewound multiple times, even if the wrapped object does not natively support * those operations (e.g. MongoDB\Driver\Cursor). * * @internal */ class CachingIterator implements Countable, Iterator {> private const FIELD_KEY = 0; /** @var array */ > private const FIELD_VALUE = 1; private $items = []; >< /** @var Generator */> /** @var Iterator */private $iterator; /** @var boolean */ private $iteratorAdvanced = false; /** @var boolean */ private $iteratorExhausted = false; /** * Initialize the iterator and stores the first item in the cache. This< * effectively rewinds the Traversable and the wrapping Generator, which < * will execute up to its first yield statement. Additionally, this mimics < * behavior of the SPL iterators and allows users to omit an explicit call < * to rewind() before using the other methods.> * effectively rewinds the Traversable and the wrapping IteratorIterator. > * Additionally, this mimics behavior of the SPL iterators and allows users > * to omit an explicit call to rewind() before using the other methods.* * @param Traversable $traversable */ public function __construct(Traversable $traversable) {< $this->iterator = $this->wrapTraversable($traversable);> $this->iterator = $traversable instanceof Iterator ? $traversable : new IteratorIterator($traversable); > > $this->iterator->rewind();$this->storeCurrentItem(); } /**< * @see http://php.net/countable.count < * @return integer> * @see https://php.net/countable.count*/< public function count()> public function count(): int{ $this->exhaustIterator(); return count($this->items); } /**< * @see http://php.net/iterator.current> * @see https://php.net/iterator.current* @return mixed */> #[ReturnTypeWillChange]public function current() {< return current($this->items);> $currentItem = current($this->items); > > return $currentItem !== false ? $currentItem[self::FIELD_VALUE] : false;} /**< * @see http://php.net/iterator.key> * @see https://php.net/iterator.key* @return mixed */> #[ReturnTypeWillChange]public function key() {< return key($this->items);> $currentItem = current($this->items); > > return $currentItem !== false ? $currentItem[self::FIELD_KEY] : null;} /**< * @see http://php.net/iterator.next < * @return void> * @see https://php.net/iterator.next*/< public function next()> public function next(): void{ if (! $this->iteratorExhausted) {> $this->iteratorAdvanced = true;$this->iterator->next();>$this->storeCurrentItem();> } > $this->iteratorExhausted = ! $this->iterator->valid();next($this->items); } /**< * @see http://php.net/iterator.rewind < * @return void> * @see https://php.net/iterator.rewind*/< public function rewind()> public function rewind(): void{ /* If the iterator has advanced, exhaust it now so that future iteration * can rely on the cache. */ if ($this->iteratorAdvanced) { $this->exhaustIterator(); } reset($this->items); } /**< * @see http://php.net/iterator.valid < * @return boolean> * @see https://php.net/iterator.valid*/< public function valid()> public function valid(): bool{ return $this->key() !== null; } /** * Ensures that the inner iterator is fully consumed and cached. */< private function exhaustIterator()> private function exhaustIterator(): void{ while (! $this->iteratorExhausted) { $this->next(); } } /** * Stores the current item in the cache. */< private function storeCurrentItem()> private function storeCurrentItem(): void{< $key = $this->iterator->key(); < < if ($key === null) {> if (! $this->iterator->valid()) {return; }< $this->items[$key] = $this->iterator->current(); < } < < /** < * Wraps the Traversable with a Generator. < * < * @param Traversable $traversable < * @return Generator < */ < private function wrapTraversable(Traversable $traversable) < { < foreach ($traversable as $key => $value) { < yield $key => $value; < $this->iteratorAdvanced = true; < } < < $this->iteratorExhausted = true;> // Storing a new item in the internal cache > $this->items[] = [ > self::FIELD_KEY => $this->iterator->key(), > self::FIELD_VALUE => $this->iterator->current(), > ];} }