Skip to content

Commit d3778f0

Browse files
committed
fix(model): Removed manual dirty _id check and moved test to Ticket/GH3326
1 parent 211b639 commit d3778f0

File tree

5 files changed

+43
-123
lines changed

5 files changed

+43
-123
lines changed

src/Eloquent/Model.php

-59
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,6 @@
44

55
namespace MongoDB\Laravel\Eloquent;
66

7-
use Illuminate\Database\Eloquent\Builder;
87
use Illuminate\Database\Eloquent\Model as BaseModel;
98
use MongoDB\Laravel\Auth\User;
109

@@ -58,62 +57,4 @@ final public static function isDocumentModel(string|object $class): bool
5857
// Document models must use the DocumentModel trait.
5958
return self::$documentModelClasses[$class] = array_key_exists(DocumentModel::class, class_uses_recursive($class));
6059
}
61-
62-
/**
63-
* Override of Laravel's performInsert() to handle MongoDB _id immutability.
64-
*
65-
* Ensures syncOriginal() is called immediately after insert, so that
66-
* subsequent save() calls (e.g., inside the "created" model event)
67-
* don't treat the primary key (_id) as dirty.
68-
*
69-
* @param Builder $query
70-
*
71-
* @return bool
72-
*/
73-
protected function performInsert(Builder $query)
74-
{
75-
if ($this->usesUniqueIds()) {
76-
$this->setUniqueIds();
77-
}
78-
79-
if ($this->fireModelEvent('creating') === false) {
80-
return false;
81-
}
82-
83-
// First we'll need to create a fresh query instance and touch the creation and
84-
// update timestamps on this model, which are maintained by us for developer
85-
// convenience. After, we will just continue saving these model instances.
86-
if ($this->usesTimestamps()) {
87-
$this->updateTimestamps();
88-
}
89-
90-
// If the model has an incrementing key, we can use the "insertGetId" method on
91-
// the query builder, which will give us back the final inserted ID for this
92-
// table from the database. Not all tables have to be incrementing though.
93-
$attributes = $this->getAttributesForInsert();
94-
95-
if ($this->getIncrementing()) {
96-
$this->insertAndSetId($query, $attributes);
97-
} else {
98-
if (empty($attributes)) {
99-
return true;
100-
}
101-
102-
$query->insert($attributes);
103-
}
104-
105-
// MongoDB-specific fix: prevent _id from being considered dirty after insert
106-
$this->syncOriginal();
107-
108-
// We will go ahead and set the exists property to true, so that it is set when
109-
// the created event is fired, just in case the developer tries to update it
110-
// during the event. This will allow them to do so and run an update here.
111-
$this->exists = true;
112-
113-
$this->wasRecentlyCreated = true;
114-
115-
$this->fireModelEvent('created', false);
116-
117-
return true;
118-
}
11960
}

src/Query/Builder.php

-7
Original file line numberDiff line numberDiff line change
@@ -783,13 +783,6 @@ public function update(array $values, array $options = [])
783783
unset($values[$key]);
784784
}
785785

786-
// Since "id" is an alias for "_id", we prevent updating it
787-
foreach ($values as $fields) {
788-
if (array_key_exists('id', $fields)) {
789-
throw new InvalidArgumentException('Cannot update "id" field.');
790-
}
791-
}
792-
793786
return $this->performUpdate($values, $options);
794787
}
795788

tests/Eloquent/Fixtures/CreatedEventTestModel.php

-20
This file was deleted.

tests/Eloquent/ModelTest.php

+1-37
Original file line numberDiff line numberDiff line change
@@ -7,13 +7,11 @@
77
use MongoDB\Laravel\Auth\User;
88
use MongoDB\Laravel\Eloquent\DocumentModel;
99
use MongoDB\Laravel\Eloquent\Model;
10-
use MongoDB\Laravel\MongoDBServiceProvider;
11-
use MongoDB\Laravel\Tests\Eloquent\Fixtures\CreatedEventTestModel;
1210
use MongoDB\Laravel\Tests\Models\Book;
1311
use MongoDB\Laravel\Tests\Models\Casting;
1412
use MongoDB\Laravel\Tests\Models\SqlBook;
15-
use Orchestra\Testbench\TestCase;
1613
use PHPUnit\Framework\Attributes\DataProvider;
14+
use PHPUnit\Framework\TestCase;
1715

1816
class ModelTest extends TestCase
1917
{
@@ -23,18 +21,6 @@ public function testIsDocumentModel(bool $expected, string|object $classOrObject
2321
$this->assertSame($expected, Model::isDocumentModel($classOrObject));
2422
}
2523

26-
public function testCreatedEventCanSafelyCallSave(): void
27-
{
28-
$model = new CreatedEventTestModel();
29-
$model->foo = 'bar';
30-
$model->save();
31-
32-
$fresh = $model->fresh();
33-
34-
$this->assertEquals('bar', $fresh->foo);
35-
$this->assertEquals('written-in-created', $fresh->extra);
36-
}
37-
3824
public static function provideDocumentModelClasses(): Generator
3925
{
4026
// Test classes
@@ -73,26 +59,4 @@ public static function provideDocumentModelClasses(): Generator
7359
},
7460
];
7561
}
76-
77-
protected function getEnvironmentSetUp($app): void
78-
{
79-
$app['config']->set('database.default', 'mongodb');
80-
81-
$app['config']->set('database.connections.mongodb', [
82-
'driver' => 'mongodb',
83-
'host' => env('DB_HOST', 'localhost'),
84-
'port' => env('DB_PORT', 27017),
85-
'database' => env('DB_DATABASE', 'testing'),
86-
'username' => env('DB_USERNAME', null),
87-
'password' => env('DB_PASSWORD', null),
88-
'options' => [],
89-
]);
90-
}
91-
92-
protected function getPackageProviders($app): array
93-
{
94-
return [
95-
MongoDBServiceProvider::class,
96-
];
97-
}
9862
}

tests/Ticket/GH3326Test.php

+42
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,42 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace MongoDB\Laravel\Tests\Ticket;
6+
7+
use MongoDB\Laravel\Eloquent\Model;
8+
use MongoDB\Laravel\Tests\TestCase;
9+
10+
/**
11+
* @see https://github.com/mongodb/laravel-mongodb/issues/3326
12+
* @see https://jira.mongodb.org/browse/PHPORM-309
13+
*/
14+
class GH3326Test extends TestCase
15+
{
16+
public function testCreatedEventCanSafelyCallSave(): void
17+
{
18+
$model = new GH3326Model();
19+
$model->foo = 'bar';
20+
$model->save();
21+
22+
$fresh = $model->fresh();
23+
24+
$this->assertEquals('bar', $fresh->foo);
25+
$this->assertEquals('written-in-created', $fresh->extra);
26+
}
27+
}
28+
29+
class GH3326Model extends Model
30+
{
31+
protected $connection = 'mongodb';
32+
protected $collection = 'test_gh3326';
33+
protected $guarded = [];
34+
35+
protected static function booted(): void
36+
{
37+
static::created(function ($model) {
38+
$model->extra = 'written-in-created';
39+
$model->saveQuietly();
40+
});
41+
}
42+
}

0 commit comments

Comments
 (0)