despatates / laravie-query-filter
Database/Eloquent Query Builder filters for Laravel
Fund package maintenance!
Liberapay
paypal.me/crynobone
Requires
- php: ^7.3 || ^8.0
- illuminate/database: ^8.67 || ^9.0
- illuminate/support: ^8.67 || ^9.0
Requires (Dev)
- nunomaduro/larastan: ^1.0.1
- orchestra/canvas: ^6.7 || ^7.0
- orchestra/testbench: ^6.22 || ^7.0
README
ℹ️ Forked from https://github.com/laravie/query-filter. I just released a v3.3.0 to work with Laravel 9.x.
Database/Eloquent Query Builder filters for Laravel
Installation
To install through composer, run the following command from terminal:
composer require "laravie/query-filter"
Usages
Order Queries
new Laravie\QueryFilter\Orderable(?string $column, string $direction = 'asc', array $config = []);
The class provides a simple interface to handle ORDER BY
queries to Laravel Eloquent/Query Builder.
use App\User; use Laravie\QueryFilter\Orderable; $query = User::query(); $orderable = new Orderable( 'name', 'desc' ); return $orderable->apply($query)->get();
select * from `users` order by `name` desc;
The code will validate the column name before trying to apply
orderBy()
to the query, this would prevent SQL injection especially when column is given by the user.
Search Queries
new Laravie\QueryFilter\Searchable(?string $keyword, array $columns = []);
The class provides a simple interface to LIKE
queries to Laravel Eloquent/Query Builder.
use App\User; use Laravie\QueryFilter\Searchable; $query = User::query(); $searchable = new Searchable( 'crynobone', ['name', 'email'] ); return $searchable->apply($query)->get();
select * from `users` where ( ( `name` like 'crynobone' or `name` like 'crynobone%' or `name` like '%crynobone' or `name` like '%crynobone%' ) or ( `email` like 'crynobone' or `email` like 'crynobone%' or `email` like '%crynobone' or `email` like '%crynobone%' ) );
Search with wildcard
Set specific %
or *
wildcard to reduce the possible LIKE
s variations.
use App\User; use Laravie\QueryFilter\Searchable; $query = User::query(); $searchable = new Searchable( 'crynobone*gmail', ['name', 'email'] ); return $searchable->apply($query)->get();
select * from `users` where ( ( `name` like 'crynobone%gmail' ) or ( `email` like 'crynobone%gmail' ) );
Search with exact wildcard
Use noWildcardSearching()
to disable adding additional search condition.
use App\User; use Laravie\QueryFilter\Searchable; $query = User::query(); $searchable = (new Searchable( 'crynobone@gmail', ['name', 'email'] ))->noWildcardSearching(); return $searchable->apply($query)->get();
select * from `users` where ( ( `name` like 'crynobone@gmail' ) or ( `email` like 'crynobone@gmail' ) );
Search with JSON path
This would allow you to query JSON path using LIKE
with case insensitive.
use App\User; use Laravie\QueryFilter\Searchable; $query = User::query(); $searchable = new Searchable( 'Malaysia', ['address->country'] ); return $searchable->apply($query)->get();
select * from `users` where ( ( lower(json_unquote(json_extract(`meta`, '$."country"'))) like 'malaysia' or lower(json_unquote(json_extract(`meta`, '$."country"'))) like 'malaysia%' or lower(json_unquote(json_extract(`meta`, '$."country"'))) like '%malaysia' or lower(json_unquote(json_extract(`meta`, '$."country"'))) like '%malaysia%' ) );
Search with Relations
This would make it easy to search results not only in the current model but also it's relations.
use App\User; use Laravie\QueryFilter\Searchable; $query = User::query(); $searchable = new Searchable( 'Administrator', ['name', 'roles.name'] ); return $searchable->apply($query)->get();
select * from `users` where ( ( `name` like 'Administrator' or `name` like 'Administrator%' or `name` like '%Administrator' or `name` like '%Administrator%' ) or exists ( select * from `roles` inner join `user_role` on `roles`.`id` = `user_role`.`role_id` where `users`.`id` = `user_role`.`user_id` and ( `name` like 'Administrator' or `name` like 'Administrator%' or `name` like '%Administrator' or `name` like '%Administrator%' ) ) );
Relations search can only be applied to
Illuminate\Database\Eloquent\Builder
as it need to ensure that the relationship exists viawhereHas()
queries.
Search with Morph Relations
You can use polymorphic relationship search using the following options:
use App\Comment; use Laravie\QueryFilter\Searchable; use Laravie\QueryFilter\Filters\MorphRelationSearch; $query = Comment::query(); $searchable = new Searchable( 'Administrator', ['name', new MorphRelationSearch('commentable', 'name')] ); return $searchable->apply($query)->get();
Taxonomy Queries
new Laravie\QueryFilter\Taxonomy(?string $keyword, array $rules, array $columns = []);
Taxonomy always developers to create a set of rules to group the search keywords using WHERE ... AND
. For any un-grouped keyword it will be executed via Laravie\QueryFilter\Searchable
based on given $columns
.
use App\User; use Laravie\QueryFilter\Taxonomy; $query = User::query(); $taxonomy = new Taxonomy( 'is:admin email:crynobone@gmail.com', [ 'email:*' => static function ($query, $value) { return $query->where('email', '=', $value); }, 'role:[]' => static function ($query, array $value) { return $query->whereIn('role', $value); }, 'is:admin' => static function ($query) { return $query->where('admin', '=', 1); }, ], ); $taxonomy->apply($query)->get();
select * from `user` where `email`='crynobone@gmail.com' and `admin`=1;
Integrations
Query Builder Macro
You can integrate Searchable
with database or eloquent query builder macro by adding the following code to your AppServiceProvider
(under register
method):
<?php namespace App\Providers; use Illuminate\Database\Eloquent\Builder as EloquentBuilder; use Illuminate\Database\Query\Builder as QueryBuilder; use Illuminate\Support\Arr; use Laravie\QueryFilter\Searchable; class AppServiceProvider extends \Illuminate\Support\ServiceProvider { /** * Register any application services. * * This service provider is a great spot to register your various container * bindings with the application. As you can see, we are registering our * "Registrar" implementation here. You can add your own bindings too! * * @return void */ public function register() { QueryBuilder::macro('whereLike', static function ($attributes, string $searchTerm) { return (new Searchable($searchTerm, Arr::wrap($attributes)))->apply($this); }); EloquentBuilder::macro('whereLike', static function ($attributes, string $searchTerm) { return (new Searchable($searchTerm, Arr::wrap($attributes)))->apply($this); }); } }
Using with Laravel Nova
You can override the default Laravel global and local search feature by adding the following methods on app/Nova/Resource.php
:
<?php namespace App\Nova; use Laravel\Nova\Http\Requests\NovaRequest; use Laravel\Nova\Resource as NovaResource; use Laravie\QueryFilter\Searchable; abstract class Resource extends NovaResource { // ... /** * Apply the search query to the query. * * @param \Illuminate\Database\Eloquent\Builder $query * @param string $search * * @return \Illuminate\Database\Eloquent\Builder */ protected static function applySearch($query, $search) { $searchColumns = static::searchableColumns() ?? []; return static::initializeSearch($search, $searchColumns)->apply($query); } /** * Initialize Search. * * @param string $search * @param array $searchColumns * @return \Laravie\QueryFilter\Searchable */ protected static function initializeSearch($search, $searchColumns) { return new Searchable($search, $searchColumns); } }