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.
> * 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 Iterator; use IteratorIterator;
> use ReturnTypeWillChange;
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 {
> private const FIELD_KEY = 0; /** @var array */ > private const FIELD_VALUE = 1; private $items = []; >
< /** @var IteratorIterator */
> /** @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 IteratorIterator. * Additionally, this mimics behavior of the SPL iterators and allows users
< * to omit an explicit call * to rewind() before using the other methods.
> * to omit an explicit call to rewind() before using the other methods.
* * @param Traversable $traversable */ public function __construct(Traversable $traversable) {
< $this->iterator = new IteratorIterator($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();
> // Storing a new item in the internal cache > $this->items[] = [ > self::FIELD_KEY => $this->iterator->key(), > self::FIELD_VALUE => $this->iterator->current(), > ];
} }