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

[12.x] Do not require returning a Builder instance from a local scope method #55246

Merged

Conversation

cosmastech
Copy link
Contributor

@cosmastech cosmastech commented Apr 1, 2025

To resolve the issue that @JeffreyWay reported on Twitter. https://x.com/jeffrey_way/status/1907138791799709883

Related to: #54450


In the current state, a local scope method must return a builder instance when called in a static context. This is counter to how scopes traditionally work, where the return value is essentially ignored.

cc: @shaedrich

Copy link

github-actions bot commented Apr 1, 2025

Thanks for submitting a PR!

Note that draft PR's are not reviewed. If you would like a review, please mark your pull request as ready for review in the GitHub user interface.

Pull requests that are abandoned in draft may be closed due to inactivity.

@cosmastech cosmastech force-pushed the add-named-scope-without-return branch from 351367d to 17f604e Compare April 1, 2025 23:21
@cosmastech cosmastech force-pushed the add-named-scope-without-return branch from 17f604e to e2414e5 Compare April 1, 2025 23:22
@cosmastech cosmastech changed the title [12.x] always return query instance from named scope [12.x] Do not require returning a Builder instance from a local scope method Apr 1, 2025
@cosmastech cosmastech marked this pull request as ready for review April 1, 2025 23:27
@newtonjob
Copy link

Thanks for this @cosmastech! I believe it's not just about the return. We should ideally proxy the static call to the query builder to ensure consistent behavior. The builder adds a few more logic when invoking local scopes that won't be covered here if we call it on the model directly.

See my comment here: #54450 (review)

@cosmastech
Copy link
Contributor Author

cosmastech commented Apr 2, 2025

Thanks for this @cosmastech! I believe it's not just about the return. We should ideally proxy the static call to the query builder to ensure consistent behavior. The builder adds a few more logic when invoking local scopes that won't be covered here if we call it on the model directly.

See my comment here: #54450 (review)

Hey @newtonjob!

I am unable to reproduce the grouping issue you identified.

`User` model code
class User extends Authenticatable
{
    /** @use HasFactory<\Database\Factories\UserFactory> */
    use HasFactory, Notifiable, SoftDeletes;
    protected $fillable = [
        'name',
        'email',
        'password',
    ];
    
    #[\Override]
    protected function casts(): array
    {
        return [
            'email_verified_at' => 'datetime',
            'password' => 'hashed',
        ];
    }

    public function scopeMyScope(Builder $builder)
    {
        $builder->where('email', '=', '[email protected]')->orWhere('id', 1);
    }

    #[Scope]
    protected function other(Builder $builder)
    {
        return $builder->where('email', '=', '[email protected]')->orWhere('id', 1);
    }
}

All of which seem to produce the same query

image

Is there something I'm missing with the setup?

Copy link
Contributor

@shaedrich shaedrich left a comment

Choose a reason for hiding this comment

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

@JeffreyWay Thanks for reporting that on Twitter/X! 👍🏻
@newtonjob Thanks for explaining the problem over in the original PR! 👍🏻
@cosmastech Thanks for opening a PR to solve this and cc-ing me! 👍🏻

Comment on lines 2405 to 2407
$parameters = [$query = static::query(), ...$parameters];

return (new static)->$method(...$parameters) ?? $query;
Copy link
Contributor

Choose a reason for hiding this comment

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

Nice solution! 👍🏻 Is there a reason, you chose this over what @newtonjob suggested in #54450 (comment)?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I didn't actually see his suggestion until after the fact. I have changed it to @newtonjob's suggestion.

@newtonjob
Copy link

Hey @cosmastech! Apparently, the SoftDeletes trait is hiding the issue in your example, so keeping it aside for now.

Here's a better way to reproduce it.

class User extends Authenticatable
{
    protected function scopeOne(Builder $query): void
    {
        $query->where('foo', true)->orWhere('bar', false);
    }

    #[Scope]
    protected function two(Builder $query): Builder
    {
        return $query->where('foo', true)->orWhere('bar', false);
    }
}
Screenshot 2025-04-02 at 10 06 56

The reason why the new scope syntax isn't grouping the OR condition is that when called statically, the scope is invoked directly and doesn't pass through Illuminate\Database\Eloquent\Builder::callNamedScope().

I believe the behavior should be the same whether or not the scope is invoked statically.

@newtonjob
Copy link

@cosmastech just saw you made the change already, thanks! 🙌

@taylorotwell
Copy link
Member

@cosmastech sorry, how does this solve the issue of needing to return? Aren't you still just returning whatever the scope returns in your change?

@cosmastech
Copy link
Contributor Author

@cosmastech sorry, how does this solve the issue of needing to return? Aren't you still just returning whatever the scope returns in your change?

@taylorotwell Through the chain of calls, Builder::callScope() is called. It either returns the Builder instance returned from the scope method (scope* or with the new #[Scope] attribute), or if null, returns the query that was passed as the first parameter.

This is illustrated by the test case which statically calls a local scope that does not return a Builder instance.

@taylorotwell taylorotwell merged commit b8f7003 into laravel:12.x Apr 2, 2025
39 checks passed
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants