Use dynamically Eloquent scopes for search process
travelbuild/eloquent-filterable is a Laravel package for use dynamically eloquent scopes for search process.
It currently has 1 GitHub stars and 326 downloads on Packagist (latest version v1.0.1).
Install it with composer require travelbuild/eloquent-filterable.
Discover more Laravel packages by travelbuild
or browse all Laravel packages to compare alternatives.
Last updated
With this package, you can call local scopes with filter scope method.
PHP 7.1.3+ or HHVM, and Composer are required.
To get thepackage, simply add the following line to the require block of your composer.json file:
"require": {
"travelbuild/eloquent-filterable": "~1.0"
}
You'll then need to run composer install or composer update to download it and have the autoloader updated. Or use to shortcut installed through terminal:
composer require travelbuild/eloquent-filterable
Each model must be implement the Filterable trait.
<?php
namespace App;
use Travelbuild\Filterable\Filterable;
use Illuminate\Database\Eloquent\Model;
class Post extends Model
{
use Filterable;
}
Then, you can define local scopes in your model. If you need help to local scopes, please follow the Laravel documentation.
/**
* Scope a query to only include only active posts.
*
* @param Builder $query
* @return Builder
*/
public function scopeActive(Builder $query): Builder
{
return $query->where('active', true);
}
And finally, you must be define usable local scopes to the filterable property in your model if you need. Why need it? We don't want some scopes to be used by the client.
/**
* The usable scope filters.
*
* @return array
*/
protected $filterable = [
'active',
];
if you want all scopes to be usable to the client, do not define filterable property or define as follows:
This way is not recommended. Take control, access to all scopes may be risky.
/**
* The usable scope filters.
*
* @return array
*/
protected $filterable = ['*'];
That's it. Now, you can call dynamically all callable scopes with filter scope.
$users = User::filter(['active' => true])->paginate();
We needed this package to quickly respond HTTP requests. Our client send to us some conditions using by search endpoint. We wanted to handle these filter conditions dynamically for simplicity and reusability.
<?php
namespace App\Http\Controllers;
use App\Post;
use Illuminate\Http\Request;
class PostSearchController extends Controller
{
/**
* Respond the post search request.
*
* @param Request $request
* @return JsonResponse
*/
public function __invoke(Request $request)
{
$posts = Post::filter($request->all())->paginate(
$request->get('limit', 25)
);
return response()->json($posts);
}
}
Create a new route for post search in routes/api.php:
Route::get('/posts', 'PostSearchController');
Define model and filterable local scopes:
<?php
namespace App;
use Travelbuild\Filterable\Filterable;
use Illuminate\Database\Eloquent\Model;
use Illuminate\Database\Eloquent\Builder;
class Post extends Model
{
use Filterable;
/**
* The usable scope filters.
*
* @return array
*/
protected $filterable = [
'active',
'author',
];
/**
* Scope a query to only include only active posts.
*
* @param Builder $query
* @return Builder
*/
public function scopeActive(Builder $query): Builder
{
return $query->where('active', true);
}
/**
* Scope a query to only include author's posts.
*
* @param Builder $query
* @param int $authorId
* @return Builder
*/
public function scopeAuthor(Builder $query, int $authorId): Builder
{
return $query->where('author_id', $authorId);
}
}
Now, the client may be send to server some search conditions. Here is some example endpoints:
/posts
/posts?active
/posts?author=5
/posts?active&author=5
/posts?active&author=5&limit=10
...
Our model will be apply the filter when exists local scope.
Love innovation and simplicity. Please use issues all errors or suggestions reporting. Follow the steps for contributing:
Please create your tests and check before PR. Use the command:
$ phpunit
Eloquent Filterable is licensed under The MIT License (MIT).