Skip to content

Commit

Permalink
[10.x] Fix firstOrNew on HasManyThrough relations (#48542)
Browse files Browse the repository at this point in the history
* Fix the firstOrNew method on HasManyThrough relations

* Add a test for the firstOrNew method when record exists

* Invert the firstOrNew if statement
  • Loading branch information
tonysm authored Sep 26, 2023
1 parent 8647e71 commit 169c976
Show file tree
Hide file tree
Showing 2 changed files with 42 additions and 4 deletions.
9 changes: 5 additions & 4 deletions src/Illuminate/Database/Eloquent/Relations/HasManyThrough.php
Original file line number Diff line number Diff line change
Expand Up @@ -252,15 +252,16 @@ protected function buildDictionary(Collection $results)
* Get the first related model record matching the attributes or instantiate it.
*
* @param array $attributes
* @param array $values
* @return \Illuminate\Database\Eloquent\Model
*/
public function firstOrNew(array $attributes)
public function firstOrNew(array $attributes = [], array $values = [])
{
if (is_null($instance = $this->where($attributes)->first())) {
$instance = $this->related->newInstance($attributes);
if (! is_null($instance = $this->where($attributes)->first())) {
return $instance;
}

return $instance;
return $this->related->newInstance(array_merge($attributes, $values));
}

/**
Expand Down
37 changes: 37 additions & 0 deletions tests/Integration/Database/EloquentHasManyThroughTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -148,6 +148,43 @@ public function testHasSameParentAndThroughParentTable()
$this->assertEquals([1], $categories->pluck('id')->all());
}

public function testFirstOrNewOnMissingRecord()
{
$taylor = User::create(['name' => 'Taylor', 'slug' => 'taylor']);
$team = Team::create(['owner_id' => $taylor->id]);

$user1 = $taylor->teamMates()->firstOrNew(
['slug' => 'tony'],
['name' => 'Tony', 'team_id' => $team->id],
);

$this->assertFalse($user1->exists);
$this->assertEquals($team->id, $user1->team_id);
$this->assertSame('tony', $user1->slug);
$this->assertSame('Tony', $user1->name);
}

public function testFirstOrNewWhenRecordExists()
{
$taylor = User::create(['name' => 'Taylor', 'slug' => 'taylor']);
$team = Team::create(['owner_id' => $taylor->id]);
$existingTony = $team->members()->create(['name' => 'Tony Messias', 'slug' => 'tony']);

$newTony = $taylor->teamMates()->firstOrNew(
['slug' => 'tony'],
['name' => 'Tony', 'team_id' => $team->id],
);

$this->assertTrue($newTony->exists);
$this->assertEquals($team->id, $newTony->team_id);
$this->assertSame('tony', $newTony->slug);
$this->assertSame('Tony Messias', $newTony->name);

$this->assertTrue($existingTony->is($newTony));
$this->assertSame('tony', $existingTony->refresh()->slug);
$this->assertSame('Tony Messias', $existingTony->name);
}

public function testFirstOrCreateWhenModelDoesntExist()
{
$owner = User::create(['name' => 'Taylor']);
Expand Down

0 comments on commit 169c976

Please sign in to comment.