Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/protonemedia/laravel-eloquent-scope-as-select
Stop duplicating your Eloquent query scopes and constraints in PHP. This package lets you re-use your query scopes and constraints by adding them as a subquery.
https://github.com/protonemedia/laravel-eloquent-scope-as-select
laravel laravel-packages php subquery
Last synced: 4 days ago
JSON representation
Stop duplicating your Eloquent query scopes and constraints in PHP. This package lets you re-use your query scopes and constraints by adding them as a subquery.
- Host: GitHub
- URL: https://github.com/protonemedia/laravel-eloquent-scope-as-select
- Owner: protonemedia
- License: mit
- Created: 2020-11-30T08:30:47.000Z (about 4 years ago)
- Default Branch: main
- Last Pushed: 2024-11-20T11:04:20.000Z (about 2 months ago)
- Last Synced: 2024-12-28T22:02:53.808Z (11 days ago)
- Topics: laravel, laravel-packages, php, subquery
- Language: PHP
- Homepage: https://protone.media/blog/stop-duplicating-your-eloquent-query-scopes-and-constraints-re-use-them-as-select-statements-with-a-new-laravel-package
- Size: 45.9 KB
- Stars: 134
- Watchers: 7
- Forks: 7
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG.md
- Contributing: CONTRIBUTING.md
- Funding: .github/FUNDING.yml
- License: LICENSE.md
Awesome Lists containing this project
README
# Laravel Eloquent Scope as Select
[![Latest Version on Packagist](https://img.shields.io/packagist/v/protonemedia/laravel-eloquent-scope-as-select.svg?style=flat-square)](https://packagist.org/packages/protonemedia/laravel-eloquent-scope-as-select)
![run-tests](https://github.com/protonemedia/laravel-eloquent-scope-as-select/workflows/run-tests/badge.svg)
[![Quality Score](https://img.shields.io/scrutinizer/g/protonemedia/laravel-eloquent-scope-as-select.svg?style=flat-square)](https://scrutinizer-ci.com/g/protonemedia/laravel-eloquent-scope-as-select)
[![Total Downloads](https://img.shields.io/packagist/dt/protonemedia/laravel-eloquent-scope-as-select.svg?style=flat-square)](https://packagist.org/packages/protonemedia/laravel-eloquent-scope-as-select)
[![Buy us a tree](https://img.shields.io/badge/Treeware-%F0%9F%8C%B3-lightgreen)](https://plant.treeware.earth/protonemedia/laravel-eloquent-scope-as-select)Stop duplicating your Eloquent query scopes and constraints in PHP. This package lets you re-use your query scopes and constraints by adding them as a subquery.
### 📺 Want to see this package in action? Join the live stream on December 3 at 14:00 CET: [https://youtu.be/0vR8IQSFsfQ](https://youtu.be/0vR8IQSFsfQ)
## Requirements
* PHP 8.2+
* Laravel 10.0This package is tested with GitHub Actions using MySQL 8.0, PostgreSQL 10.8 and SQLite.
## Features
* Add a subquery based on a [query scope](https://laravel.com/docs/8.x/eloquent#query-scopes).
* Add a subquery using a Closure.
* Shortcuts for calling scopes by using a string or array.
* Support for more than one subquery.
* Support for flipping the result.
* Zero third-party dependencies.Related package: [Laravel Eloquent Where Not](https://github.com/protonemedia/laravel-eloquent-where-not)
## Sponsor Us
[](https://inertiaui.com/inertia-table?utm_source=github&utm_campaign=laravel-eloquent-scope-as-select)
❤️ We proudly support the community by developing Laravel packages and giving them away for free. If this package saves you time or if you're relying on it professionally, please consider [sponsoring the maintenance and development](https://github.com/sponsors/pascalbaljet) and check out our latest premium package: [Inertia Table](https://inertiaui.com/inertia-table?utm_source=github&utm_campaign=laravel-eloquent-scope-as-select). Keeping track of issues and pull requests takes time, but we're happy to help!
## Blogpost
If you want to know more about the background of this package, please read the blogpost: [Stop duplicating your Eloquent query scopes and constraints. Re-use them as select statements with a new Laravel package](https://protone.media/blog/stop-duplicating-your-eloquent-query-scopes-and-constraints-re-use-them-as-select-statements-with-a-new-laravel-package).
## Installation
You can install the package via composer:
```bash
composer require protonemedia/laravel-eloquent-scope-as-select
```Add the `macro` to the query builder, for example, in your `AppServiceProvider`. By default, the name of the macro is `addScopeAsSelect`, but you can customize it with the first parameter of the `addMacro` method.
```php
use ProtoneMedia\LaravelEloquentScopeAsSelect\ScopeAsSelect;public function boot()
{
ScopeAsSelect::addMacro();// or use a custom method name:
ScopeAsSelect::addMacro('withScopeAsSubQuery');
}
```## Short API description
*For a more practical explanation, check out the [usage](#usage) section below.*
Add a select using a Closure. Each `Post` model, published or not, will have an `is_published` attribute.
```php
Post::addScopeAsSelect('is_published', function ($query) {
$query->published();
})->get();
```The example above can be shortened by using a string, where the second argument is the name of the scope:
```php
Post::addScopeAsSelect('is_published', 'published')->get();
```You can use an array to call multiple scopes:
```php
Post::addScopeAsSelect('is_popular_and_published', ['popular', 'published'])->get();
```Use an associative array to call dynamic scopes:
```php
Post::addScopeAsSelect('is_announcement', ['ofType' => 'announcement'])->get();
```If your dynamic scopes require multiple arguments, you can use an associative array:
```php
Post::addScopeAsSelect('is_announcement', ['publishedBetween' => [2010, 2020]])->get();
```You can also mix dynamic and non-dynmaic scopes:
```php
Post::addScopeAsSelect('is_published_announcement', [
'published',
'ofType' => 'announcement'
])->get();
```The method has an optional third argument that flips the result.
```php
Post::addScopeAsSelect('is_not_announcement', ['ofType' => 'announcement'], false)->get();
```## Usage
Imagine you have a `Post` Eloquent model with a query scope.
```php
class Post extends Model
{
public function scopePublished($query)
{
return $query->whereNotNull('published_at');
}
}
```Now you can fetch all published posts by calling the scope method on the query:
```php
$allPublishedPosts = Post::published()->get();
```But what if you want to fetch *all* posts and *then* check if the post is published? This scope is quite simple, so you can easily mimic the scope's outcome by checking the `published_at` attribute:
```php
Post::get()->each(function (Post $post) {
$isPublished = !is_null($post->published_at);
});
```This is harder to achieve when scopes get more complicated or when you chain various scopes. Let's add a relationship and another scope to the `Post` model:
```php
class Post extends Model
{
public function comments()
{
return $this->hasMany(Comment::class);
}public function scopePublished($query)
{
return $query->whereNotNull('published_at');
}public function scopePublishedInCurrentYear($query)
{
return $query->whereYear('published_at', date('Y'));
}
}
```Using Eloquent, we can fetch all posts from this year with at least ten comments.
```php
$recentPopularPosts = Post::query()
->publishedInCurrentYear()
->has('comments', '>=', 10)
->get();
```Great! Now we want to fetch all posts again, and then check if the post was published this year and has at least ten comments.
```php
Post::get()->each(function (Post $post) {
$isRecentAndPopular = $post->comments()->count() >= 10
&& optional($post->published_at)->isCurrentYear();
});
```Well, you get the idea. This is bound to get messy and you're duplicating logic as well.
### Solution
Using the power of this package, you can re-use your scopes when fetching data. The first example (`published` scope) can be narrowed down to:
```php
$posts = Post::addScopeAsSelect('is_published', function ($query) {
$query->published();
})->get();
```With short closures, a feature which was introduced in PHP 7.4, this can be even shorter:
```php
$posts = Post::addScopeAsSelect('is_published', fn ($query) => $query->published())->get();
```Now every `Post` model will have an `is_published` boolean attribute.
```php
$posts->each(function (Post $post) {
$isPublished = $post->is_published;
});
```You can add multiple selects as well, for example, to combine both scenarios:
```php
Post::query()
->addScopeAsSelect('is_published', function ($query) {
$query->published();
})
->addScopeAsSelect('is_recent_and_popular', function ($query) {
$query->publishedInCurrentYear()->has('comments', '>=', 10);
})
->get()
->each(function (Post $post) {
$isPublished = $post->is_published;$isRecentAndPopular = $post->is_recent_and_popular;
});
```### Shortcuts
Instead of using a Closure, there are some shortcuts you could use (see also: [Short API description](#short-api-description)):
Using a string instead of a Closure:
```php
Post::addScopeAsSelect('is_published', function ($query) {
$query->published();
});// is the same as:
Post::addScopeAsSelect('is_published', 'published');
```Using an array instead of Closure, to support multiple scopes and dynamic scopes:
```php
Post::addScopeAsSelect('is_announcement', function ($query) {
$query->ofType('announcement');
});// is the same as:
Post::addScopeAsSelect('is_announcement', ['ofType' => 'announcement']);
```You can also flip the result with the optional third parameter (it defaults to `true`):
```php
$postA = Post::addScopeAsSelect('is_announcement', ['ofType' => 'announcement'])->first();
$postB = Post::addScopeAsSelect('is_not_announcement', ['ofType' => 'announcement'], false)->first();$this->assertTrue($postA->is_announcement)
$this->assertFalse($postB->is_not_announcement);
```### Testing
``` bash
composer test
```### Changelog
Please see [CHANGELOG](CHANGELOG.md) for more information about what has changed recently.
## Contributing
Please see [CONTRIBUTING](CONTRIBUTING.md) for details.
## Other Laravel packages
* [`Inertia Table`](https://inertiaui.com/inertia-table?utm_source=github&utm_campaign=laravel-eloquent-scope-as-select): The Ultimate Table for Inertia.js with built-in Query Builder.
* [`Laravel Blade On Demand`](https://github.com/protonemedia/laravel-blade-on-demand): Laravel package to compile Blade templates in memory.
* [`Laravel Cross Eloquent Search`](https://github.com/protonemedia/laravel-cross-eloquent-search): Laravel package to search through multiple Eloquent models.
* [`Laravel FFMpeg`](https://github.com/protonemedia/laravel-ffmpeg): This package provides an integration with FFmpeg for Laravel. The storage of the files is handled by Laravel's Filesystem.
* [`Laravel MinIO Testing Tools`](https://github.com/protonemedia/laravel-minio-testing-tools): Run your tests against a MinIO S3 server.
* [`Laravel Mixins`](https://github.com/protonemedia/laravel-mixins): A collection of Laravel goodies.
* [`Laravel Paddle`](https://github.com/protonemedia/laravel-paddle): Paddle.com API integration for Laravel with support for webhooks/events.
* [`Laravel Task Runner`](https://github.com/protonemedia/laravel-task-runner): Write Shell scripts like Blade Components and run them locally or on a remote server.
* [`Laravel Verify New Email`](https://github.com/protonemedia/laravel-verify-new-email): This package adds support for verifying new email addresses: when a user updates its email address, it won't replace the old one until the new one is verified.
* [`Laravel XSS Protection`](https://github.com/protonemedia/laravel-xss-protection): Laravel Middleware to protect your app against Cross-site scripting (XSS). It sanitizes request input, and it can sanatize Blade echo statements.### Security
If you discover any security related issues, please email [email protected] instead of using the issue tracker.
## Credits
- [Pascal Baljet](https://github.com/protonemedia)
- [All Contributors](../../contributors)## License
The MIT License (MIT). Please see [License File](LICENSE.md) for more information.
## Treeware
This package is [Treeware](https://treeware.earth). If you use it in production, then we ask that you [**buy the world a tree**](https://plant.treeware.earth/pascalbaljetmedia/laravel-eloquent-scope-as-select) to thank us for our work. By contributing to the Treeware forest you’ll be creating employment for local families and restoring wildlife habitats.