meiko / laravel-filterable
Provides JSON API filter support to models
Installs: 1 287
Dependents: 0
Suggesters: 0
Security: 0
Stars: 1
Watchers: 3
Forks: 1
Open Issues: 0
Requires
- php: >=7.0.0
Requires (Dev)
- laracasts/testdummy: ^2.3
- orchestra/testbench: ^3.6
- phpunit/phpunit: ^7.0
This package is not auto-updated.
Last update: 2022-05-28 00:41:00 UTC
README
Use url parameters to easily sort, filter and search eloquent models.
Table of Contents
Click to expand
Getting started
Installation
Install with composer
composer require meikooy/laravel-filterable
Configuration
Create new configuration file filterable.php
in your config directory (config/filterable.php
).
<?php return [ 'namespace' => '\App\Models' # The namespace which contains your models ];
Usage
To enable usage for your model, you need to apply the provided Filterable trait to your model.
<?php namespace App\Models; use Illuminate\Database\Eloquent\Model; use Meiko\Filterable\Filterable; class Post extends Model { use Filterable; }
Next, call the filters
in your controllers.
Example:
<?php namespace App\Http\Controllers; use Illuminate\Http\Request; use App\Models\Post; class PostController extends Controller { public function index(Request $request) { return Post::filters( function ($filterer) { return $filterer; } ); }
Sorting
To sort your eloquent models, use the sort
url param:
sort=<column-name>
.
Ascending order
Example: Sort post's in ascending order by the posts updated_at
column.
http://localhost/api/posts?sort=updated_at
Descending order
To sort the results in descending order, prefix the url param value with -
.
http://localhost/api/posts?sort=-updated_at
Filtering
To filter your eloquent models, you need to provide the column name
, comparison type
and value
in the url.
<column-name>=<comparison-type>=<value>
Examples:
Fetch posts where the title
column has text like Lorem ipsum
http//localhost/api/posts?title=lk=Lorem ipsum
Fetch posts where the title
is NOT like Lorem ipsum
http//localhost/api/posts?title=nl=Lorem ipsum
Fetch posts where the read_count
column has value bigger than 100.
http//localhost/api/posts?read_count=gt=100
Fetch posts where the title
column contains one of: Lorem ipsum
, Hello
or First
http//localhost/api/posts?title=in=Lorem ipsum|Hello|First
NOTE: When providing multiple values, separate the elements with |
character. Only the bw
, not-bw
, in
, not-in
comparison types support multiple values.
See Field.php for all the supported comparison types.
Id columns
If you need to filter models by one their relations and you use some kind of id obfuscation library (like hashids) then queries with obfuscated id's won't work.
http//localhost/api/posts?user_id=BOaPXaoz # Won't return any results as we are trying to match the `user_id` (in the database) with the obfuscated value: BOaPXaoz
To fix this, you can set custom idResolver
(in config/filterable.php
) to "decode" your id values before they are used in queries.
config/filterable.php
<?php return [ 'namespace' => '\App\Models', 'idResolver' => App\Models\IdResolver::class // The class has to have resolve() method ];
app/Models/IdResolver.php
<?php namespace App\Models; use Hashids\Hashids; class IdResolver { public function resolve($modelName, $id, $routeKeyName) { $hashids = new Hashids(); $decodedId = $hashids->decode($id); return $decodedId; } }
Custom filters
If the provided comparison types are not enough or you wish to have some additional logic for the filtering process you can also create your own filters.
Example: Users have one-to-many relation with posts. We need to filter posts by their creators username.
<?php namespace App\Http\Controllers; use Illuminate\Http\Request; use App\Models\Post; class PostController extends Controller { public function index(Request $request) { return Post::filters( function ($filterer) { $filterer->addFilterColumn( 'username', function ($query, $type, $value) { $query->whereHas( 'user', function ($userQuery) use ($value) { return $userQuery->where('users.username', 'like', $value); } ); } ); return $filterer; } ); }
Now we can use the username
url param in our query.
http://localhost/api/posts?username=meikooy
Searching
Prerequisites
To use the search, you first need configure the columns which can be searched.
<?php namespace App\Models; use Illuminate\Database\Eloquent\Model; use Meiko\Filterable\Filterable; class Post extends Model { use Filterable; /** * Searchable attributes * * @var array */ protected $searchable = [ 'title', 'content', ]; }
Usage
To search, use the _q
url parameter.
http//localhost/api/posts?_q=searchword
The query tries to find matches for searchword
in the Post's title
and content
columns.