axn / laravel-database-extension
Extension of the database section of the Laravel framework
Installs: 1 865
Dependents: 0
Suggesters: 0
Security: 0
Stars: 0
Watchers: 2
Forks: 0
Open Issues: 0
Requires
- php: ^8.2
- laravel/framework: ^10.0 || ^11.0
Requires (Dev)
- driftingly/rector-laravel: ^1.1
- laravel/pint: ^1.15.0
- rector/rector: ^1.0.3
README
Includes some extensions/improvements to the Database section of Laravel Framework
Installation
With Composer:
composer require axn/laravel-database-extension
Usage
Natural sorting
Method orderByNatural
has been added to QueryBuilder (macro) for natural sorting
(see: http://kumaresan-drupal.blogspot.fr/2012/09/natural-sorting-in-mysql-or.html).
Use it like orderBy
.
Example:
DB::table('appartements')->orderByNatural('numero')->get(); // Descendant DB::table('appartements')->orderByNatural('numero', 'desc')->get(); // or DB::table('appartements')->orderByNaturalDesc('numero')->get();
Default order
Add the global scope DefaultOrderScope
to the model if you want to have select
results automatically sorted:
use Axn\Illuminate\Database\Eloquent\DefaultOrderScope; class MyModel extends Model { protected static function booted() { static::addGlobalScope(new DefaultOrderScope([ 'column' => 'option', ])); } }
option
can be:
- 'asc'
- 'desc'
- 'natural' (apply
orderByNatural()
) - 'natural_asc' (same as 'natural')
- 'natural_desc' (same as 'natural' but descendant)
- 'raw' (apply
orderByRaw()
)
If you don't precise option, it will be "asc" by default.
Example:
use Axn\Illuminate\Database\Eloquent\DefaultOrderScope; class User extends Model { protected static function booted() { static::addGlobalScope(new DefaultOrderScope([ 'lastname' => 'asc', 'firstname' => 'desc', ])); } }
If you don't want the default order applied, simply use the Eloquent method
withoutGlobalScope()
on the model:
$users = User::withoutGlobalScope(DefaultOrderScope::class)->get();
Note that the default order is automatically disabled if you manually set ORDER BY
clause.
Joins using relationships
This is the most important feature of this package: you can do joins using Eloquent relationships!
WARNING: only BelongsTo, HasOne, HasMany, MorphOne and MorphMany relations are supported. So, if you want to use BelongsToMany, you have to go with the HasMany/BelongsTo relations to/from the pivot table.
Example:
// instead of doing joinRel('roles') (User belongs-to-many Role) User::joinRel('userHasRoles') // User has-many UserHasRole ->joinRel('userHasRoles.role') // UserHasRole belongs-to Role ->get(); // with aliases: User::alias('u') ->joinRel('userHasRoles', 'uhr') ->joinRel('uhr.role', 'r') ->get();
You may also want to use:
- leftJoinRel()
- rightJoinRel()
Or if the model uses SoftDeletes and you want to include trashed records:
- joinRelWithTrashed()
- leftJoinRelWithTrashed()
- rightJoinRelWithTrashed()
And to add extra criteria:
User::joinRel('userHasRoles', function ($join) { $join->where('is_main', 1); }) ->joinRel('userHasRoles.role') ->get();
Note that extra criteria are automatically added if they are defined on the relation:
class User extends Model { // joinRel('mainAddress', 'a') will do: // join `addresses` as `a` on `a`.`user_id` = `users`.`id` and `a`.`is_main` = 1 public function mainAddress() { return $this->hasOne('addresses')->where('is_main', 1); } }
WARNING: an instance of JoinRelBuilder is created and attached to the Eloquent Builder instance via WeakMap to handle this feature. If you ever clone the Builder instance, note that there is no cloning of the attached JoinRelBuilder instance. This can be a problem if you use "joinRel" on the cloned instance with a reference to an alias created in the original instance.
For example:
$originalBuilder = User::joinRel('userHasRoles'); $clonedBuilder = clone $originalBuilder; // Produces error: No model with alias "userHasRoles" $clonedBuilder->joinRel('userHasRoles.role');
If you need to handle this case, use the "cloneWithJoinRelBuilder" method instead of clone:
$originalBuilder = User::joinRel('userHasRoles'); $clonedBuilder = $originalBuilder->cloneWithJoinRelBuilder(); $clonedBuilder->joinRel('userHasRoles.role');
Eloquent whereHasIn macro
If you have performance issues with the whereHas
method, you can use whereHasIn
instead.
It uses in
clause instead of exists
to check existence:
// where exists (select * from `comments` where `comments`.`post_id` = `posts`.`id`) Post::whereHas('comments')->get(); // where `posts`.`id` in (select `comments`.`post_id` from `comments`) Post::whereHasIn('comments')->get();
You can use a callback to add extra criteria:
// where `posts`.`id` in ( // select `comments`.`post_id` from `comments` // where `comments`.`content` like "A%" // ) Post::whereHasIn('comments', function ($query) { $query->where('content', 'like', "A%"); })->get();
Note that it does not support "dot" notation, but you can use joins:
// where `posts`.`id` in ( // select `comments`.`post_id` from `comments` // inner join `users` as `author` on `author`.`id` = `comments`.`author_id` // where `author`.`lastname` like "A%" // ) Post::whereHasIn('comments', function ($query) { $query ->joinRel('author') ->where('author.lastname', 'like', "A%"); })->get();
You may also want to use:
- orWhereHasIn()
- whereDoesntHaveIn()
- orWhereDoesntHaveIn()
Eloquent whereLike macro
Source: https://murze.be/searching-models-using-a-where-like-query-in-laravel
Warning! This only works on instances of the Eloquent Builder, not on the generic Query Builder.
A replacement of this:
User::query() ->where('name', 'like', "%{$searchTerm}%") ->orWhere('email', 'like', "%{$searchTerm}%") ->get();
By that:
User::whereLike(['name', 'email'], $searchTerm)->get();
Or more advanced, a replacement of this:
Post::query() ->where('name', 'like', "%{$searchTerm}%") ->orWhere('text', 'like', "%{$searchTerm}%") ->orWhereHas('author', function ($query) use ($searchTerm) { $query->where('name', 'like', "%{$searchTerm}%"); }) ->orWhereHas('tags', function ($query) use ($searchTerm) { $query->where('name', 'like', "%{$searchTerm}%"); }) ->get();
By that:
Post::whereLike(['name', 'text', 'author.name', 'tags.name'], $searchTerm)->get();
SoftDeletes withoutTrashedExcept scope
Our SoftDeletes
trait extends the Eloquent one to provide the withoutTrashedExcept
scope :
$postTypes = PostType::withoutTrashedExcept($post->post_type_id)->get(); // you also can provide multiple ids: $postTypes = PostType::withoutTrashedExcept([1, 2, 3])->get();
To use it, add the trait Axn\Illuminate\Database\Eloquent\SoftDeletes
to models:
namespace App\Models; use Illuminate\Database\Eloquent\Model; use Axn\Illuminate\Database\Eloquent\SoftDeletes; class User extends Model { use SoftDeletes; // ... }