This Laravel Eloquent extension adds support for JSON foreign keys to BelongsTo
, HasOne
, HasMany
, HasManyThrough
, MorphTo
, MorphOne
and MorphMany
relationships.
It also provides many-to-many relationships with JSON arrays.
Database | Laravel |
---|---|
MySQL 5.7+ | 5.5.29+ |
MariaDB 10.2+ | 5.8+ (2019) |
PostgreSQL 9.3+ | 5.5.29+ |
SQLite 3.18+ | 5.6.35+ |
SQL Server 2016+ | 5.6.25+ |
composer require staudenmeir/eloquent-json-relations
In this example, User
has a BelongsTo
relationship with Locale
. There is no dedicated column, but the foreign key (locale_id
) is stored as a property in a JSON field (users.options
):
class User extends Model
{
use \Staudenmeir\EloquentJsonRelations\HasJsonRelationships;
protected $casts = [
'options' => 'json',
];
public function locale()
{
return $this->belongsTo('App\Locale', 'options->locale_id');
}
}
class Locale extends Model
{
use \Staudenmeir\EloquentJsonRelations\HasJsonRelationships;
public function users()
{
return $this->hasMany('App\User', 'options->locale_id');
}
}
Remember to use the HasJsonRelationships
trait in both the parent and the related model.
This package also introduces two new relationship types: BelongsToJson
and HasManyJson
On Laravel 5.6.25+, you can use them to implement many-to-many relationships with JSON arrays.
In this example, User
has a BelongsToMany
relationship with Role
. There is no pivot table, but the foreign keys are stored as an array in a JSON field (users.options
):
By default, the relationship stores the records as an array of IDs:
class User extends Model
{
use \Staudenmeir\EloquentJsonRelations\HasJsonRelationships;
protected $casts = [
'options' => 'json',
];
public function roles()
{
return $this->belongsToJson('App\Role', 'options->role_ids');
}
}
class Role extends Model
{
use \Staudenmeir\EloquentJsonRelations\HasJsonRelationships;
public function users()
{
return $this->hasManyJson('App\User', 'options->role_ids');
}
}
On the side of the BelongsToJson
relationship, you can use attach()
, detach()
, sync()
and toggle()
:
$user = new User;
$user->roles()->attach([1, 2])->save(); // Now: [1, 2]
$user->roles()->detach([2])->save(); // Now: [1]
$user->roles()->sync([1, 3])->save(); // Now: [1, 3]
$user->roles()->toggle([2, 3])->save(); // Now: [1, 2]
You can also store the records as objects with additional pivot attributes:
class User extends Model
{
use \Staudenmeir\EloquentJsonRelations\HasJsonRelationships;
protected $casts = [
'options' => 'json',
];
public function roles()
{
return $this->belongsToJson('App\Role', 'options->roles[]->role_id');
}
}
class Role extends Model
{
use \Staudenmeir\EloquentJsonRelations\HasJsonRelationships;
public function users()
{
return $this->hasManyJson('App\User', 'options->roles[]->role_id');
}
}
Here, options->roles
is the path to the JSON array. role_id
is the name of the foreign key property inside the record object:
$user = new User;
$user->roles()->attach([1 => ['active' => true], 2 => ['active' => false]])->save();
// Now: [{"role_id":1,"active":true},{"role_id":2,"active":false}]
$user->roles()->detach([2])->save();
// Now: [{"role_id":1,"active":true}]
$user->roles()->sync([1 => ['active' => false], 3 => ['active' => true]])->save();
// Now: [{"role_id":1,"active":false},{"role_id":3,"active":true}]
$user->roles()->toggle([2 => ['active' => true], 3])->save();
// Now: [{"role_id":1,"active":false},{"role_id":2,"active":true}]
Limitations: On SQLite and SQL Server, these relationships only work partially.
On one-to-many relationships, you can still ensure referential integrity.
MySQL and SQL Server support foreign keys on JSON columns with generated/computed columns.
Laravel migrations support this feature on MySQL:
Schema::create('users', function (Blueprint $table) {
$table->increments('id');
$table->json('options');
$locale_id = DB::connection()->getQueryGrammar()->wrap('options->locale_id');
$table->unsignedInteger('locale_id')->storedAs($locale_id);
$table->foreign('locale_id')->references('id')->on('locales');
});
On SQL Server, the migration requires raw SQL:
Schema::create('users', function (Blueprint $table) {
$table->increments('id');
$table->json('options');
});
$locale_id = DB::connection()->getQueryGrammar()->wrap('options->locale_id');
DB::statement('ALTER TABLE [users] ADD "locale_id" AS CAST('.$locale_id.' AS INT) PERSISTED');
Schema::table('users', function (Blueprint $table) {
$table->foreign('locale_id')->references('id')->on('locales');
});