Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

WIP: Support version 2.0 of the MongoDB driver #2683

Draft
wants to merge 6 commits into
base: 2.10.x
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
15 changes: 14 additions & 1 deletion .github/workflows/continuous-integration.yml
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ jobs:
runs-on: "ubuntu-20.04"

strategy:
fail-fast: false
matrix:
php-version:
- "8.1"
Expand All @@ -39,7 +40,7 @@ jobs:
- dependencies: "lowest"
php-version: "8.1"
mongodb-version: "5.0"
driver-version: "1.17.0"
driver-version: "1.20.0"
topology: "server"
symfony-version: "stable"
# Test with highest dependencies
Expand All @@ -56,6 +57,13 @@ jobs:
driver-version: "stable"
dependencies: "highest"
symfony-version: "stable"
# Test with ext-2.0
- topology: "server"
php-version: "8.2"
mongodb-version: "7.0"
driver-version: "mongodb/mongo-php-driver@v2.x"
dependencies: "highest"
symfony-version: "7"
# Test with a 5.0 sharded cluster
# Currently disabled due to a bug where MongoDB reports "sharding status unknown"
# - topology: "sharded_cluster"
Expand Down Expand Up @@ -117,6 +125,11 @@ jobs:
dependency-versions: "${{ matrix.dependencies }}"
composer-options: "--prefer-dist"

- name: "Install latest Python version"
uses: actions/setup-python@v5
with:
python-version: '3.13'

- id: setup-mongodb
uses: mongodb-labs/drivers-evergreen-tools@master
with:
Expand Down
4 changes: 2 additions & 2 deletions composer.json
Original file line number Diff line number Diff line change
Expand Up @@ -22,15 +22,15 @@
],
"require": {
"php": "^8.1",
"ext-mongodb": "^1.17",
"ext-mongodb": "^1.20 || ^2.0",
"doctrine/cache": "^1.11 || ^2.0",
"doctrine/collections": "^1.5 || ^2.0",
"doctrine/event-manager": "^1.0 || ^2.0",
"doctrine/instantiator": "^1.1 || ^2",
"doctrine/persistence": "^3.2",
"friendsofphp/proxy-manager-lts": "^1.0",
"jean85/pretty-package-versions": "^1.3.0 || ^2.0.1",
"mongodb/mongodb": "^1.17.0",
"mongodb/mongodb": "^1.20 || ^2.0@dev",
"psr/cache": "^1.0 || ^2.0 || ^3.0",
"symfony/console": "^5.4 || ^6.0 || ^7.0",
"symfony/deprecation-contracts": "^2.2 || ^3.0",
Expand Down
56 changes: 53 additions & 3 deletions tests/Doctrine/ODM/MongoDB/Tests/QueryTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -16,16 +16,22 @@
use Documents\Project;
use Documents\SubProject;
use Documents\User;
use Exception;
use InvalidArgumentException;
use LogicException;
use MongoDB\BSON\Int64;
use MongoDB\BSON\ObjectId;
use MongoDB\Collection;
use MongoDB\Driver\CursorId;
use MongoDB\Driver\CursorInterface;
use MongoDB\Driver\ReadPreference;
use MongoDB\Driver\Server;
use PHPUnit\Framework\Attributes\DataProvider;
use PHPUnit\Framework\MockObject\MockObject;
use Traversable;

use function array_keys;
use function class_exists;
use function iterator_to_array;

use const DOCTRINE_MONGODB_DATABASE;
Expand Down Expand Up @@ -496,7 +502,7 @@

public function testFindWithHint(): void
{
$cursor = $this->createMock(Traversable::class);
$cursor = $this->createCursorMock();

$collection = $this->getMockCollection();
$collection->expects($this->once())
Expand All @@ -523,7 +529,7 @@
$nearest = new ReadPreference('nearest');
$secondaryPreferred = new ReadPreference('secondaryPreferred');

$cursor = $this->createMock(Traversable::class);
$cursor = $this->createCursorMock();

$collection = $this->getMockCollection();
$collection->expects($this->once())
Expand Down Expand Up @@ -552,7 +558,41 @@

public function testNonRewindable(): void
{
$cursor = new ArrayIterator(['foo']);
$cursor = new class ([['foo']]) extends ArrayIterator implements CursorInterface {
public function getId(): Int64

Check failure on line 562 in tests/Doctrine/ODM/MongoDB/Tests/QueryTest.php

View workflow job for this annotation

GitHub Actions / Static Analysis with PHPStan (8.2)

Return type (MongoDB\BSON\Int64) of method class@anonymous/tests/Doctrine/ODM/MongoDB/Tests/QueryTest.php:561::getId() should be compatible with return type (MongoDB\Driver\CursorId) of method MongoDB\Driver\CursorInterface::getId()
{
return new Int64(0);
}

public function getServer(): Server
{
throw new Exception('Not implemented');
}

public function isDead(): bool
{
return false;
}

public function setTypeMap(array $typemap): void

Check failure on line 577 in tests/Doctrine/ODM/MongoDB/Tests/QueryTest.php

View workflow job for this annotation

GitHub Actions / Static Analysis with PHPStan (8.2)

Method class@anonymous/tests/Doctrine/ODM/MongoDB/Tests/QueryTest.php:561::setTypeMap() has parameter $typemap with no value type specified in iterable type array.
{
}

public function toArray(): array

Check failure on line 581 in tests/Doctrine/ODM/MongoDB/Tests/QueryTest.php

View workflow job for this annotation

GitHub Actions / Static Analysis with PHPStan (8.2)

Method class@anonymous/tests/Doctrine/ODM/MongoDB/Tests/QueryTest.php:561::toArray() return type has no value type specified in iterable type array.
{
return iterator_to_array($this);
}

public function current(): array|null

Check failure on line 586 in tests/Doctrine/ODM/MongoDB/Tests/QueryTest.php

View workflow job for this annotation

GitHub Actions / Static Analysis with PHPStan (8.2)

Method class@anonymous/tests/Doctrine/ODM/MongoDB/Tests/QueryTest.php:561::current() return type has no value type specified in iterable type array.
{
return parent::current();
}

public function key(): int|null
{
return parent::key();
}
};

$collection = $this->getMockCollection();
$collection->expects($this->once())
Expand Down Expand Up @@ -582,6 +622,16 @@
{
return $this->createMock(Collection::class);
}

private function createCursorMock(): CursorInterface|Traversable

Check failure on line 626 in tests/Doctrine/ODM/MongoDB/Tests/QueryTest.php

View workflow job for this annotation

GitHub Actions / Static Analysis with PHPStan (8.2)

Method Doctrine\ODM\MongoDB\Tests\QueryTest::createCursorMock() return type has no value type specified in iterable type Traversable.
{
return $this->createMock(
// Use the cursorID class to differentiate between 1.x and 2.x
class_exists(CursorId::class)
? Traversable::class
: CursorInterface::class,
);
}
}

#[ODM\Document(collection: 'people')]
Expand Down
45 changes: 37 additions & 8 deletions tests/Doctrine/ODM/MongoDB/Tests/SchemaManagerTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@
use Documents\Tournament\Tournament;
use Documents\UserName;
use InvalidArgumentException;
use Iterator;
use MongoDB\BSON\Document;
use MongoDB\Client;
use MongoDB\Collection;
Expand All @@ -33,7 +34,10 @@
use MongoDB\Driver\WriteConcern;
use MongoDB\GridFS\Bucket;
use MongoDB\Model\CollectionInfo;
use MongoDB\Model\CollectionInfoCommandIterator;
use MongoDB\Model\CollectionInfoIterator;
use MongoDB\Model\IndexInfo;
use MongoDB\Model\IndexInfoIterator;
use MongoDB\Model\IndexInfoIteratorIterator;
use PHPUnit\Framework\Attributes\DataProvider;
use PHPUnit\Framework\Constraint\ArrayHasKey;
Expand All @@ -46,6 +50,7 @@
use function array_map;
use function assert;
use function in_array;
use function interface_exists;

/**
* @phpstan-import-type IndexMapping from ClassMetadata
Expand Down Expand Up @@ -197,15 +202,15 @@

$filesCollection
->method('listIndexes')
->willReturn([]);
->willReturn($this->createIndexIterator());
$filesCollection
->expects($this->atLeastOnce())
->method('createIndex')
->with(['filename' => 1, 'uploadDate' => 1], $this->writeOptions($expectedWriteOptions));

$chunksCollection
->method('listIndexes')
->willReturn([]);
->willReturn($this->createIndexIterator());
$chunksCollection
->expects($this->atLeastOnce())
->method('createIndex')
Expand Down Expand Up @@ -253,15 +258,15 @@
if ($class === $fileBucket) {
$filesCollection
->method('listIndexes')
->willReturn([]);
->willReturn($this->createIndexIterator());
$filesCollection
->expects($this->once())
->method('createIndex')
->with(['filename' => 1, 'uploadDate' => 1], $this->writeOptions($expectedWriteOptions));

$chunksCollection
->method('listIndexes')
->willReturn([]);
->willReturn($this->createIndexIterator());
$chunksCollection
->expects($this->once())
->method('createIndex')
Expand Down Expand Up @@ -298,7 +303,7 @@
$collection
->expects($this->once())
->method('listIndexes')
->willReturn(new IndexInfoIteratorIterator(new ArrayIterator([])));
->willReturn($this->createIndexIterator());
$collection
->expects($this->once())
->method('createIndex')
Expand Down Expand Up @@ -328,7 +333,7 @@
$collection
->expects($this->once())
->method('listIndexes')
->willReturn(new IndexInfoIteratorIterator(new ArrayIterator($indexes)));
->willReturn($this->createIndexIterator($indexes));
$collection
->expects($this->once())
->method('createIndex')
Expand Down Expand Up @@ -1336,10 +1341,10 @@
$db->method('listCollections')->willReturnCallback(function () {
$collections = [];
foreach ($this->documentCollections as $collectionName => $collection) {
$collections[] = new CollectionInfo(['name' => $collectionName]);
$collections[] = ['name' => $collectionName];
}

return $collections;
return $this->createCollectionIterator($collections);
});

return $db;
Expand Down Expand Up @@ -1372,4 +1377,28 @@
{
return new CommandException('Unrecognized pipeline stage name: \'$listSearchIndexes\'', 40234);
}

private function createIndexIterator(array $indexes = []): Iterator

Check failure on line 1381 in tests/Doctrine/ODM/MongoDB/Tests/SchemaManagerTest.php

View workflow job for this annotation

GitHub Actions / Static Analysis with PHPStan (8.2)

Method Doctrine\ODM\MongoDB\Tests\SchemaManagerTest::createIndexIterator() has parameter $indexes with no value type specified in iterable type array.
{
if (interface_exists(IndexInfoIterator::class)) {
return new IndexInfoIteratorIterator(new ArrayIterator($indexes));
}

return new ArrayIterator(array_map(
static fn (array $indexInfo) => new IndexInfo($indexInfo),
$indexes,
));
}

private function createCollectionIterator(array $collections = []): Iterator

Check failure on line 1393 in tests/Doctrine/ODM/MongoDB/Tests/SchemaManagerTest.php

View workflow job for this annotation

GitHub Actions / Static Analysis with PHPStan (8.2)

Method Doctrine\ODM\MongoDB\Tests\SchemaManagerTest::createCollectionIterator() has parameter $collections with no value type specified in iterable type array.
{
if (interface_exists(CollectionInfoIterator::class)) {
return new CollectionInfoCommandIterator(new ArrayIterator($collections));
}

return new ArrayIterator(array_map(
static fn (array $collectionInfo) => new CollectionInfo($collectionInfo),
$collections,
));
}
}
Loading