giastfader / laravel-query-builder

Easily build Eloquent queries from API requests

Home Page:https://spatie.be/en/opensource/laravel

Geek Repo:Geek Repo

Github PK Tool:Github PK Tool

Easily build Eloquent queries from API requests

Latest Version on Packagist Build Status StyleCI Quality Score Total Downloads

This package allows you to filter, sort and include eloquent relations based on a request. The QueryBuilder used in this package extends Laravel's default Eloquent builder. This means all your favorite methods and macros are still available. Query parameter names follow the JSON API specification as closely as possible.

Basic usage

Filtering an API request: /users?filter[name]=John:

$users = QueryBuilder::for(User::class)
    ->allowedFilters('name')
    ->get();
// all `User`s that contain the string "John" in their name

Requesting relations from an API request: /users?include=posts:

$users = QueryBuilder::for(User::class)
    ->allowedIncludes('posts')
    ->get();
// all `User`s with their `posts` loaded

Works together nicely with existing queries:

$query = User::where('active', true);

$user = QueryBuilder::for($query)
    ->allowedIncludes('posts', 'permissions')
    ->where('score', '>', 42) // chain on any of Laravel's query builder methods
    ->first();

Sorting an API request: /users?sort=name:

$users = QueryBuilder::for(User::class)->get();
// all `User`s sorted by name

Have a look at the usage section below for advanced examples and features.

Installation

You can install the package via composer:

composer require spatie/laravel-query-builder

Usage

Including relationships

The include query parameter will load any Eloquent relation on the results collection. By default no includes are allowed. All includes must be specified using allowedIncludes().

// GET /users?include=posts
$users = QueryBuilder::for(User::class)
    ->allowedIncludes('posts')
    ->get();

// $users will contain all users with their posts loaded

You can load multiple relationship by separating them with a comma:

// GET /users?include=posts,permissions
$users = QueryBuilder::for(User::class)
    ->allowedIncludes('posts', 'permissions')
    ->get();

// $users will contain all users with their posts and permissions loaded

When trying to include relationships that have not been allowed using allowedIncludes() an InvalidQuery exception will be thrown.

Relation/include names will converted to camelCase when looking for the corresponding relationship on the model. This means /users?include=blog-posts will try to load the blogPosts() relationship on the User model.

Once the relationships are loaded on the results collection you can include them in your response by using Eloquent API resources and conditional relationships.

Filtering

The filter query parmeters can be used to filter results by partial property value, exact property value or if a property value exists in a given array of values. You can also specify custom filters for more advanced queries.

By default no filters are allowed. All filters have to be specified using allowedFilters().

// GET /users?filter[name]=john&filter[email]=gmail
$users = QueryBuilder::for(User::class)
    ->allowedFilters('name', 'email')
    ->get();
// $users will contain all users with "john" in their name AND "gmail" in their email address

You can specify multiple matching filter values by passing a comma separated list of values:

// GET /users?filter[name]=seb,freek
$users = QueryBuilder::for(User::class)
    ->allowedFilters('name')
    ->get();
// $users will contain all users that contain "seb" OR "freek" in their name

Exact filters

When filtering models based on their IDs, a boolean value or a literal string, you'll want to use exact filters. This way /users?filter[id]=1 won't match all users containing the digit 1 in their ID.

Exact filters can be added using Spatie\QueryBuilder\Filter::exact('property_name') in the allowedFilters() method.

use Spatie\QueryBuilder\Filter;

// GET /users?filter[name]=John%20Doe
$users = QueryBuilder::for(User::class)
    ->allowedFilters(Filter::exact('name'))
    ->get();
// all users with the exact name "John Doe"

The query builder will automatically map 'true' and 'false' as booleans and a comma separated list of values as an array:

use Spatie\QueryBuilder\Filter;

// GET /users?filter[id]=1,2,3,4,5&filter[admin]=true
$users = QueryBuilder::for(User::class)
    ->allowedFilters(Filter::exact('id'), Filter::exact('admin'))
    ->get();
// $users will contain all admin users with id 1, 2, 3, 4 or 5

Custom filters

You can specify custom filters using the Filter::custom() method. Custom filters are simple, invokable classes that implement the \Spatie\QueryBuilder\Filters\Filter interface. This way you can create any query your heart desires.

For example:

use Spatie\QueryBuilder\Filter;
use Illuminate\Database\Eloquent\Builder;

class FiltersUserPermission
{
    public function __invoke(Builder $query, $value, string $property)
    {
        return $query->whereHas('permissions', function (Builder $query) use ($value) {
            $query->where('name', $value);
        });
    }
}

// GET /users?filter[permission]=createPosts
$users = QueryBuilder::for(User::class)
    ->allowedFilters(Filter::custom('permission', FiltersUserPermission::class))
    ->get();
// $users will contain all users that have the `createPosts` permission

Sorting

The sort query parameter is used to determine by which property the results collection will be ordered. Sorting is ascending by default. Adding a hyphen (-) to the start of the property name will reverse the results collection.

// GET /users?sort=-name
$users = QueryBuilder::for(User::class)->get();

// $users will be sorted by name and descending (Z -> A)

By default all model properties can be used to sort the results. However, you can use the allowedSorts method to limit which properties are allowed to be used in the request.

When trying to sort by a property that's not specified in allowedSorts() an InvalidQuery exception will be thrown.

// GET /users?sort=password
$users = QueryBuilder::for(User::class)
    ->allowedSorts('name')
    ->get();

// Will throw an `InvalidQuery` exception as `password` is not an allowed sorting property

To define a default sort parameter that should be applied without explicitly adding it to the request, you can use the defaultSort method.

// GET /users
$users = QueryBuilder::for(User::class)
    ->defaultSort('name')
    ->allowedSorts('name', 'street')
    ->get();

// Will retrieve the users sorted by name

Other query methods

As the QueryBuilder extends Laravel's default Eloquent query builder you can use any method or macro you like. You can also specify a base query instead of the model FQCN:

QueryBuilder::for(User::where('id', 42)) // base query instead of model
    ->allowedIncludes('posts')
    ->where('activated', true) // chain on any of Laravel's query methods
    ->first(); // we only need one specific user

Pagination

This package doesn't provide any methods to help you paginate responses. However as documented above you can use Laravel's default paginate() method.

If you want to completly adhere to the JSON API specification you can also use our own spatie/json-api-paginate!

Testing

composer test

Changelog

Please see CHANGELOG for more information what has changed recently.

Contributing

Please see CONTRIBUTING for details.

Security

If you discover any security related issues, please email freek@spatie.be instead of using the issue tracker.

Postcardware

You're free to use this package, but if it makes it to your production environment we highly appreciate you sending us a postcard from your hometown, mentioning which of our package(s) you are using.

Our address is: Spatie, Samberstraat 69D, 2060 Antwerp, Belgium.

We publish all received postcards on our company website.

Credits

Support us

Spatie is a webdesign agency based in Antwerp, Belgium. You'll find an overview of all our open source projects on our website.

Does your business depend on our contributions? Reach out and support us on Patreon. All pledges will be dedicated to allocating workforce on maintenance and new awesome stuff.

License

The MIT License (MIT). Please see License File for more information.

About

Easily build Eloquent queries from API requests

https://spatie.be/en/opensource/laravel

License:MIT License


Languages

Language:PHP 100.0%