Search moodle.org's
Developer Documentation

See Release Notes

  • Bug fixes for general core bugs in 3.11.x will end 14 Nov 2022 (12 months plus 6 months extension).
  • Bug fixes for security issues in 3.11.x will end 13 Nov 2023 (18 months plus 12 months extension).
  • PHP version: minimum PHP 7.3.0 Note: minimum PHP version has increased since Moodle 3.10. PHP 7.4.x is supported too.
<?php
/*
 * Copyright 2017 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
 *
 * 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 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 { /** @var array */ private $items = [];
< /** @var Generator */
> /** @var IteratorIterator */
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 = new IteratorIterator($traversable); > > $this->iterator->rewind();
$this->storeCurrentItem(); } /** * @see http://php.net/countable.count * @return integer */ public function count() { $this->exhaustIterator(); return count($this->items); } /** * @see http://php.net/iterator.current * @return mixed */ public function current() { return current($this->items); } /** * @see http://php.net/iterator.key * @return mixed */ public function key() { return key($this->items); } /** * @see http://php.net/iterator.next * @return void */ public function next() { 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 */ public function rewind() { /* 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 */ public function valid() { return $this->key() !== null; } /** * Ensures that the inner iterator is fully consumed and cached. */ private function exhaustIterator() { while (! $this->iteratorExhausted) { $this->next(); } } /** * Stores the current item in the cache. */ private function storeCurrentItem() { $key = $this->iterator->key(); if ($key === null) { 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;
} }