For the project I’m working on, I find myself performing a search query against multiple Eloquent models, where I search for a term in multiple and different attributes within each Model.
There is going to be many ways to solve this, I’m sure. But, as far as I can see, there are two practical approaches to tackle this.
The first would be to extend the Eloquent Query Builder using macros
. macros will let you add the search feature to ALL your Models. If this sounds like what you’re looking for, then Freek Van der Herten got your back, Freek has a great blog post explaining this approach. As a matter of fact, this blog post is highly based on his approach.
The second approach is to use Local Query Scope. Local scopes allow you to define common sets of query constraints that you may easily re-use throughout your application. And to make it even more re-usable I’m wrapping the search scope inside a trait
so it’s easy to apply to every Model we want to make searchable.
In this post, I will share with you my experience with the approach I chose to improve the readability and the reusability of my search functionality through a trait
I named Searchable
.
But before getting to the “real stuff”, here is some use cases.
Usage
class User extends Model
{
use Searchable;
}
snippet 1
User::search('searchTerm', ['name', 'email'])->get();
/**
* SELECT * FROM `users`
* WHERE `name` LIKE '%searchTerm%'
* OR `email` LIKE '%searchTerm%'
*/
The order of the arguments is not really important, as long as you respect some conventions, of which, the searchTerm is always a string
, and the attributes is an array
. With that in mind, the above snippet could easily be written like below and still got the same results:
snippet 2
User::search(['name', 'email'], 'searchTerm')->get();
/**
* SELECT * FROM `users`
* WHERE `name` LIKE '%searchTerm%'
* OR `email` LIKE '%searchTerm%'
*/
Sometimes you may only want to pass searchTerm as the only argument, that’s fine too. As long as your Model has either a public $searchable
property or a public searchable
method returning an array
of attributes you wish to perform the search against, everything is going to be fine.
snippet 3
class User extends Model
{
use Searchable;
/**
* Searchable attributes
*
* @return string[]
*/
public $searchable = ['name', 'email'];
// OR
/**
* Searchable attributes
*
* @return string[]
*/
public static function searchable()
{
return ['name', 'email'];
}
}
//...
User::search('searchTerm')->get();
/**
* SELECT * FROM `users`
* WHERE `name` LIKE '%searchTerm%'
* OR `email` LIKE '%searchTerm%'
*/
Or maybe you only want to pass attributes, and let the “magic” guess for the searchTerm. As I’m going to explain next, that’s possible too.
snippet 4
User::search(['name', 'email', 'phone'])->get();
/**
* SELECT * FROM `users`
* WHERE `name` LIKE '%searchTerm%'
* OR `email` LIKE '%searchTerm%'
* OR `phone` LIKE '%searchTerm%'
*/
The last snippets works, because behind the scenes, I check if a specific query parameter key is present in the Request
, if it’s the case I use it to get the searchTerm. More to come later on.
What about the case where you don’t want to pass any arguments at all. Well, that works too.
snippet 5
User::search()->get();
/**
* SELECT * FROM `users`
* WHERE `name` LIKE '%searchTerm%'
* OR `email` LIKE '%searchTerm%'
*/
It’s a sort of snippet 3 and 4 combined.
Support for relations
You can also search for attributes in related models.
snippet 6
User::search('searchTerm', ['name', 'country.name'])->get();
/**
* SELECT * FROM `users`
* WHERE (
* `name` LIKE '%searchTerm%' OR
* EXISTS (
* SELECT * FROM `countries` WHERE `users`.`country_id` = `countries`.`id`
* AND `name` LIKE '%searchTerm%'
* )
*)
*/
As you can conclude from the generated SQL, we are looking for users where name contains searchTerm OR
users with country name contains searchTerm.
By this far, I hope this opens your appetite to follow along with me to implement this feature.
Implementation
I will start by creating Searchable
trait, I usually keep the Model’s traits in the App\Concerns\Models
namespace, but this is just a personal preference, you can put it wherever you think suits your project the best.
<?php
namespace App\Concerns\Models;
trait Searchable
{
/**
* Scope a query to search for term in the attributes
*
* @param \Illuminate\Database\Eloquent\Builder $query
* @return \Illuminate\Database\Eloquent\Builder
*/
protected function scopeSearch($query)
{
// search logic...
}
}
By now you know that there are two optional arguments we can pass to search scope, a $searchTerm string
and an array
of $attributes.
While $query
argument is always present as it’s passed by Laravel Query Builder.
It seems to me that I should first parse search scope arguments, for that reason I created parseArguments
method to handle the parsing.
<?php
namespace App\Concerns\Models;
trait Searchable
{
/**
* Scope a query to search for term in the attributes
*
* @param \Illuminate\Database\Eloquent\Builder $query
* @return \Illuminate\Database\Eloquent\Builder
*/
protected function scopeSearch($query)
{
[$searchTerm, $attributes] = $this->parseArguments(func_get_args());
}
/**
* Parse search scope arguments
*
* @param array $arguments
* @return array
*/
private function parseArguments(array $arguments)
{
// parsing...
}
}
func_get_args
gets an array of the function’s argument list. In our case it’s a list of argument passed to scopeSearch which are $query Builder
, $searchTerm string
and $attributes array
.
Here is the entire parseArguments
definition alongside the searchableAttributes
method definition.
N.B. parseArguments
returns an array
where the $searchTerm is at index 0
and $attrubutes at index 1
.
<?php
//...
/**
* Parse search scope arguments
*
* @param array $arguments
* @return array
*/
private function parseArguments(array $arguments)
{
$args_count = count($arguments);
switch ($args_count) {
case 1:
return [request(config('searchable.key')), $this->searchableAttributes()];
break;
case 2:
return is_string($arguments[1])
? [$arguments[1], $this->searchableAttributes()]
: [request(config('searchable.key')), $arguments[1]];
break;
case 3:
return is_string($arguments[1])
? [$arguments[1], $arguments[2]]
: [$arguments[2], $arguments[1]];
break;
default:
return [null, []];
break;
}
}
/**
* Get searchable columns
*
* @return array
*/
public function searchableAttributes()
{
if (method_exists($this, 'searchable')) {
return $this->searchable();
}
return property_exists($this, 'searchable') ? $this->searchable : [];
}
searchableAttributes
main job is to get searchable attributes array difined in the Model (e.g: User). It’s used in the case where the $attributes argument is not provided in search scope (see snippet 3 and 5). searchableAttributes
first looks for a method in the Model named searchable if it exist. If it does, it returns the result. If it does not, it looks for a property named $searchable. Again if it exists it returns it’s content, if does not, an empty array is returned instead.
For parseArguments
, first we count how many arguments passed to the search scope. The number of arguments is important here as it will tell us how our scope is been used (see the usage section).
One thing to note is that the value of $args_count
will never be 0
(that’s why you won’t find a case
statement for 0
in the switch
) because $query Builder
is always passed by Laravel as the first argument, which also means, it is the argument at index 0 of $arguments
array.
Ok, let’s break it down case by case.
case 1
case 1:
return [request(config('searchable.key')), $this->searchableAttributes()];
break;
Correspond to snippet 5. In this case, no arguments are passed to search scope, so it’s up to us get the searchTerm from the Request
and the attributes from the Model
.
Let’s take a closer look at request(config('searchable.key'))
. As a Laravel user you know, as the name implies, that config access a value in a config file using the “dot” syntax, which includes the name of the file and the option you wish to access.
So, to get the query parameter key that presumably holds your searchTerm value, you need to have configuration file (searchable.php
) in config
directory.
return [
// query parameter key
// e.g http://example.com/search?q=searchTerm
'key' => 'q',
];
In this example, the query parameter key that will be used to get searchTerm value is q, so as long as you Request
has a filled q parameter you’re good to go.
case 2
case 2:
return is_string($arguments[1])
? [$arguments[1], $this->searchableAttributes()]
: [request(config('searchable.key')), $arguments[1]];
break;
Correspond to snippet 3 OR snippet 4. In this case, either $searchTerm string
OR $attributes array
alongside $query Builder
are passed to the search scope. This convention of string/array makes it easy for us to distinguich what type of argument is passed to the search scope.
So, if it’s the $searchTerm string
we get the $attributes array
by involking searchableAttributes
method, else, it’s the $attributes array
we get the $searchTerm from the Request
by the configured query parameter key.
case 3
case 3:
return is_string($arguments[1])
? [$arguments[1], $arguments[2]]
: [$arguments[2], $arguments[1]];
break;
Correspond to snippet 1 OR snippet 2 OR snippet 6. both $searchTerm and
$attributes are present as arguements to our search scope, regardless of theire order.
We make sure the order of arguments in the returned array is as expected: [string $searchTerm, array $attributes]
.
After argument parsing, the logical next step is to build the query, and it’s exaclty what we are going to do next.
Build the query
Here is the entire definition of scopeSearch
:
/**
* Scope a query to search for a term in the attributes
*
* @param \Illuminate\Database\Eloquent\Builder $query
* @return \Illuminate\Database\Eloquent\Builder
*/
protected function scopeSearch($query)
{
[$searchTerm, $attributes] = $this->parseArguments(func_get_args());
if (!$searchTerm || !$attributes) {
return $query;
}
return $query->where(function (Builder $query) use ($searchTerm, $attributes) {
foreach ($attributes as $attribute) {
$query->when(
str_contains($attribute, '.'),
function (Builder $query) use ($searchTerm, $attribute) {
[$relationName, $relationAttribute] = explode('.', $attribute);
$query->orWhereHas($relationName, function (Builder $query) use ($searchTerm, $relationAttribute) {
$query->where($relationAttribute, 'LIKE', "%{$searchTerm}%");
});
},
function (Builder $query) use ($searchTerm, $attribute) {
$query->orWhere($attribute, 'LIKE', "%{$searchTerm}%");
}
);
}
});
}
If parseArguments
method didn’t manage to parse a valid $searchTerm nor a valid array of $attributes then, no need to continue down the road just return the $query Builder
.
Else, we have valid $searchTerm and a valid array of $attributes, we have to perform a logical grouping by passing a closure to where
method. Inside the closure we go through our $attributes using a foreach
loop. Remember that, as we have seen in snippet 6, we may search for attributes in related model. So, to cover this usecase, we to perform different orWhere
clause depending on the type of attribute. For that reason we are using conditional clauses.
Final results
One configuration file config/searchable.php
return [
// query parameter key
// e.g http://example.com/search?q=searchTerm
'key' => 'q',
];
One searchable trait app/Concerns/Models/Searchable.php
<?php
namespace App\Concerns\Models;
use Illuminate\Support\Arr;
use Illuminate\Database\Eloquent\Builder;
trait Searchable
{
/**
* Scope a query to search for a term in the attributes
*
* @param \Illuminate\Database\Eloquent\Builder $query
* @return \Illuminate\Database\Eloquent\Builder
*/
protected function scopeSearch($query)
{
[$searchTerm, $attributes] = $this->parseArguments(func_get_args());
if (!$searchTerm || !$attributes) {
return $query;
}
return $query->where(function (Builder $query) use ($attributes, $searchTerm) {
foreach (Arr::wrap($attributes) as $attribute) {
$query->when(
str_contains($attribute, '.'),
function (Builder $query) use ($attribute, $searchTerm) {
[$relationName, $relationAttribute] = explode('.', $attribute);
$query->orWhereHas($relationName, function (Builder $query) use ($relationAttribute, $searchTerm) {
$query->where($relationAttribute, 'LIKE', "%{$searchTerm}%");
});
},
function (Builder $query) use ($attribute, $searchTerm) {
$query->orWhere($attribute, 'LIKE', "%{$searchTerm}%");
}
);
}
});
}
/**
* Parse search scope arguments
*
* @param array $arguments
* @return array
*/
private function parseArguments(array $arguments)
{
$args_count = count($arguments);
switch ($args_count) {
case 1:
return [request(config('searchable.key')), $this->searchableAttributes()];
break;
case 2:
return is_string($arguments[1])
? [$arguments[1], $this->searchableAttributes()]
: [request(config('searchable.key')), $arguments[1]];
break;
case 3:
return is_string($arguments[1])
? [$arguments[1], $arguments[2]]
: [$arguments[2], $arguments[1]];
break;
default:
return [null, []];
break;
}
}
/**
* Get searchable columns
*
* @return array
*/
public function searchableAttributes()
{
if (method_exists($this, 'searchable')) {
return $this->searchable();
}
return property_exists($this, 'searchable') ? $this->searchable : [];
}
}