Skip to content
Merged
Show file tree
Hide file tree
Changes from 1 commit
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
2 changes: 1 addition & 1 deletion CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ All notable changes to this project will be documented in this file.
## [5.0.0] - next

* Remove support for Laravel 10 by @GromNaN in [#3123](https://github.com/mongodb/laravel-mongodb/pull/3123)
* **BREAKING CHANGE** Use `id` as an alias for `_id` in commands and queries for compatibility with Eloquent packages by @GromNaN in [#3040](https://github.com/mongodb/laravel-mongodb/pull/3040)
* **BREAKING CHANGE** Use `id` as an alias for `_id` in commands and queries for compatibility with Eloquent packages by @GromNaN in [#3040](https://github.com/mongodb/laravel-mongodb/pull/3040) and [#3136](https://github.com/mongodb/laravel-mongodb/pull/3136)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I added an association between PHPORM-147 and PHPORM-236 since these appear to be closely related.

* **BREAKING CHANGE** Make Query\Builder return objects instead of array to match Laravel behavior by @GromNaN in [#3107](https://github.com/mongodb/laravel-mongodb/pull/3107)
* **BREAKING CHANGE** In DB query results, convert BSON `UTCDateTime` objects into `Carbon` date with the default timezone by @GromNaN in [#3119](https://github.com/mongodb/laravel-mongodb/pull/3119)
* Remove `MongoFailedJobProvider`, replaced by Laravel `DatabaseFailedJobProvider` by @GromNaN in [#3122](https://github.com/mongodb/laravel-mongodb/pull/3122)
Expand Down
2 changes: 1 addition & 1 deletion docs/includes/usage-examples/FindOneTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,6 @@ public function testFindOne(): void
// end-find-one

$this->assertInstanceOf(Movie::class, $movie);
$this->expectOutputRegex('/^{"_id":"[a-z0-9]{24}","title":"The Shawshank Redemption","directors":\["Frank Darabont","Rob Reiner"\],"id":"[a-z0-9]{24}"}$/');
$this->expectOutputRegex('/^{"title":"The Shawshank Redemption","directors":\["Frank Darabont","Rob Reiner"\],"id":"[a-z0-9]{24}"}$/');
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We can't preserve the key/property order without creating a new object/array. This overhead and complexity is not justified here. "id" will always be last.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I realize this was in the original pattern, so it's not a change from this PR, but what causes the ObjectId to be printed as a plain string? It doesn't look like this library overrides Model::toJson(), so is the ObjectId typing dropped entirely for the model's in-memory representation?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

By default, id is casted into string when Model::toArray() is called.

}
}
8 changes: 4 additions & 4 deletions src/Query/Builder.php
Original file line number Diff line number Diff line change
Expand Up @@ -1616,7 +1616,7 @@ public function orWhereIntegerNotInRaw($column, $values, $boolean = 'and')
private function aliasIdForQuery(array $values): array
{
if (array_key_exists('id', $values)) {
if (array_key_exists('_id', $values)) {
if (array_key_exists('_id', $values) && $values['id'] !== $values['_id']) {
throw new InvalidArgumentException('Cannot have both "id" and "_id" fields.');
}

Expand All @@ -1627,7 +1627,7 @@ private function aliasIdForQuery(array $values): array
foreach ($values as $key => $value) {
if (is_string($key) && str_ends_with($key, '.id')) {
$newkey = substr($key, 0, -3) . '._id';
if (array_key_exists($newkey, $values)) {
if (array_key_exists($newkey, $values) && $value !== $values[$newkey]) {
throw new InvalidArgumentException(sprintf('Cannot have both "%s" and "%s" fields.', $key, $newkey));
}

Expand Down Expand Up @@ -1659,7 +1659,7 @@ private function aliasIdForResult(array|object $values): array|object
if (is_array($values)) {
if (array_key_exists('_id', $values) && ! array_key_exists('id', $values)) {
$values['id'] = $values['_id'];
//unset($values['_id']);
unset($values['_id']);
}

foreach ($values as $key => $value) {
Expand All @@ -1675,7 +1675,7 @@ private function aliasIdForResult(array|object $values): array|object
if ($values instanceof stdClass) {
if (property_exists($values, '_id') && ! property_exists($values, 'id')) {
$values->id = $values->_id;
//unset($values->_id);
unset($values->_id);
}

foreach (get_object_vars($values) as $key => $value) {
Expand Down
13 changes: 11 additions & 2 deletions tests/QueryBuilderTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -449,25 +449,34 @@ public function testDistinct()

public function testCustomId()
{
$tags = [['id' => 'sharp', 'name' => 'Sharp']];
DB::table('items')->insert([
['id' => 'knife', 'type' => 'sharp', 'amount' => 34],
['id' => 'fork', 'type' => 'sharp', 'amount' => 20],
['id' => 'knife', 'type' => 'sharp', 'amount' => 34, 'tags' => $tags],
['id' => 'fork', 'type' => 'sharp', 'amount' => 20, 'tags' => $tags],
['id' => 'spoon', 'type' => 'round', 'amount' => 3],
]);

$item = DB::table('items')->find('knife');
$this->assertEquals('knife', $item->id);
$this->assertObjectNotHasProperty('_id', $item);
$this->assertEquals('sharp', $item->tags[0]['id']);
$this->assertArrayNotHasKey('_id', $item->tags[0]);

$item = DB::table('items')->where('id', 'fork')->first();
$this->assertEquals('fork', $item->id);

// tags.id is translated into tags._id in query
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

There are plenty of assertions in this test for _id not existing on the model object(s), but what actually asserts that _id is stored in the database? Presumably, the query below would work even if id was never translated to _id on the way to the server.

If it's not the intention to test that here, feel free to resolve. This comment made me think otherwise.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I added a query on _id field in this test.

$items = DB::table('items')->whereIn('tags.id', ['sharp'])->get();
$this->assertCount(2, $items);

DB::table('users')->insert([
['id' => 1, 'name' => 'Jane Doe'],
['id' => 2, 'name' => 'John Doe'],
]);

$item = DB::table('users')->find(1);
$this->assertEquals(1, $item->id);
$this->assertObjectNotHasProperty('_id', $item);
}

public function testTake()
Expand Down
4 changes: 2 additions & 2 deletions tests/Queue/Failed/DatabaseFailedJobProviderTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -77,7 +77,7 @@ public function testAll(): void
$all = $this->getProvider()->all();

$this->assertCount(5, $all);
$this->assertEquals(new ObjectId(sprintf('%024d', 5)), $all[0]->_id);
$this->assertEquals(new ObjectId(sprintf('%024d', 5)), $all[0]->id);
$this->assertEquals(sprintf('%024d', 5), $all[0]->id, 'id field is added for compatibility with DatabaseFailedJobProvider');
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Unrelated to the diff, but would this benefit from using assertSame to also check that a string type is used?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

They are not the same. 2 distinct objects line 80.
I add assertions on the type of the "id" property.

}

Expand All @@ -89,7 +89,7 @@ public function testFindAndForget(): void
$found = $provider->find($id);

$this->assertIsObject($found, 'The job is found');
$this->assertEquals(new ObjectId($id), $found->_id);
$this->assertEquals(new ObjectId($id), $found->id);
$this->assertObjectHasProperty('failed_at', $found);

// Delete the job
Expand Down