mirror of
https://github.com/archtechx/tenancy.git
synced 2025-12-12 10:54:04 +00:00
Refactor models & config
This commit is contained in:
parent
f0ff8248e7
commit
2839f45196
9 changed files with 123 additions and 97 deletions
|
|
@ -110,7 +110,7 @@ class TenancyServiceProvider extends ServiceProvider
|
||||||
protected function mapRoutes()
|
protected function mapRoutes()
|
||||||
{
|
{
|
||||||
if (file_exists(base_path('routes/tenant.php'))) {
|
if (file_exists(base_path('routes/tenant.php'))) {
|
||||||
Route::namespace($this->app['config']['tenancy.tenant_route_namespace'] ?? 'App\Http\Controllers')
|
Route::namespace('App\Http\Controllers')
|
||||||
->group(base_path('routes/tenant.php'));
|
->group(base_path('routes/tenant.php'));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -7,20 +7,20 @@ use Stancl\Tenancy\Database\Models\Tenant;
|
||||||
|
|
||||||
return [
|
return [
|
||||||
'tenant_model' => Tenant::class,
|
'tenant_model' => Tenant::class,
|
||||||
'internal_prefix' => 'tenancy_',
|
|
||||||
'id_generator' => Stancl\Tenancy\UUIDGenerator::class,
|
'id_generator' => Stancl\Tenancy\UUIDGenerator::class,
|
||||||
|
|
||||||
'domain_model' => Domain::class,
|
'domain_model' => Domain::class,
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The list of domains hosting your central app.
|
||||||
|
*
|
||||||
|
* Only relevant if you're using the domain or subdomain identification middleware.
|
||||||
|
*/
|
||||||
'central_domains' => [
|
'central_domains' => [
|
||||||
'127.0.0.1',
|
'127.0.0.1',
|
||||||
'localhost',
|
'localhost',
|
||||||
],
|
],
|
||||||
|
|
||||||
/**
|
|
||||||
* Controller namespace used by routes in routes/tenant.php.
|
|
||||||
*/
|
|
||||||
'tenant_route_namespace' => 'App\Http\Controllers',
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Tenancy bootstrappers are executed when tenancy is initialized.
|
* Tenancy bootstrappers are executed when tenancy is initialized.
|
||||||
* Their responsibility is making Laravel features tenant-aware.
|
* Their responsibility is making Laravel features tenant-aware.
|
||||||
|
|
@ -75,22 +75,6 @@ return [
|
||||||
],
|
],
|
||||||
],
|
],
|
||||||
|
|
||||||
/**
|
|
||||||
* Redis tenancy config. Used by RedisTenancyBoostrapper.
|
|
||||||
*
|
|
||||||
* Note: You need phpredis to use Redis tenancy.
|
|
||||||
*
|
|
||||||
* Note: You don't need to use this if you're using Redis only for cache.
|
|
||||||
* Redis tenancy is only relevant if you're making direct Redis calls,
|
|
||||||
* either using the Redis facade or by injecting it as a dependency.
|
|
||||||
*/
|
|
||||||
'redis' => [
|
|
||||||
'prefix_base' => 'tenant', // Each key in Redis will be prepended by this prefix_base, followed by the tenant id.
|
|
||||||
'prefixed_connections' => [ // Redis connections whose keys are prefixed, to separate one tenant's keys from another.
|
|
||||||
// 'default',
|
|
||||||
],
|
|
||||||
],
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Cache tenancy config. Used by CacheTenancyBootstrapper.
|
* Cache tenancy config. Used by CacheTenancyBootstrapper.
|
||||||
*
|
*
|
||||||
|
|
@ -151,6 +135,22 @@ return [
|
||||||
'asset_helper_tenancy' => true,
|
'asset_helper_tenancy' => true,
|
||||||
],
|
],
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Redis tenancy config. Used by RedisTenancyBoostrapper.
|
||||||
|
*
|
||||||
|
* Note: You need phpredis to use Redis tenancy.
|
||||||
|
*
|
||||||
|
* Note: You don't need to use this if you're using Redis only for cache.
|
||||||
|
* Redis tenancy is only relevant if you're making direct Redis calls,
|
||||||
|
* either using the Redis facade or by injecting it as a dependency.
|
||||||
|
*/
|
||||||
|
'redis' => [
|
||||||
|
'prefix_base' => 'tenant', // Each key in Redis will be prepended by this prefix_base, followed by the tenant id.
|
||||||
|
'prefixed_connections' => [ // Redis connections whose keys are prefixed, to separate one tenant's keys from another.
|
||||||
|
// 'default',
|
||||||
|
],
|
||||||
|
],
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Features are classes that provide additional functionality
|
* Features are classes that provide additional functionality
|
||||||
* not needed for tenancy to be bootstrapped. They are run
|
* not needed for tenancy to be bootstrapped. They are run
|
||||||
|
|
|
||||||
19
src/Database/Concerns/EnsuresDomainIsNotOccupied.php
Normal file
19
src/Database/Concerns/EnsuresDomainIsNotOccupied.php
Normal file
|
|
@ -0,0 +1,19 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Stancl\Tenancy\Database\Concerns;
|
||||||
|
|
||||||
|
use Stancl\Tenancy\Exceptions\DomainOccupiedByOtherTenantException;
|
||||||
|
|
||||||
|
trait EnsuresDomainIsNotOccupied
|
||||||
|
{
|
||||||
|
public static function bootEnsuresDomainIsNotOccupied()
|
||||||
|
{
|
||||||
|
static::saving(function ($self) {
|
||||||
|
if ($domain = $self->newQuery()->where('domain', $self->domain)->first()) {
|
||||||
|
if ($domain->getKey() !== $self->getKey()) {
|
||||||
|
throw new DomainOccupiedByOtherTenantException($self->domain);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -11,7 +11,7 @@ trait GeneratesIds
|
||||||
public static function bootGeneratesIds()
|
public static function bootGeneratesIds()
|
||||||
{
|
{
|
||||||
static::creating(function (self $model) {
|
static::creating(function (self $model) {
|
||||||
if (! $model->getTenantKey() && app()->bound(UniqueIdentifierGenerator::class)) {
|
if (! $model->getTenantKey() && $model->shouldGenerateId()) {
|
||||||
$model->setAttribute($model->getTenantKeyName(), app(UniqueIdentifierGenerator::class)->generate($model));
|
$model->setAttribute($model->getTenantKeyName(), app(UniqueIdentifierGenerator::class)->generate($model));
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
@ -19,6 +19,11 @@ trait GeneratesIds
|
||||||
|
|
||||||
public function getIncrementing()
|
public function getIncrementing()
|
||||||
{
|
{
|
||||||
return ! app()->bound(UniqueIdentifierGenerator::class);
|
return ! $this->shouldGenerateId();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function shouldGenerateId(): bool
|
||||||
|
{
|
||||||
|
return app()->bound(UniqueIdentifierGenerator::class);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
32
src/Database/Concerns/HasInternalKeys.php
Normal file
32
src/Database/Concerns/HasInternalKeys.php
Normal file
|
|
@ -0,0 +1,32 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Stancl\Tenancy\Database\Concerns;
|
||||||
|
|
||||||
|
trait HasInternalKeys
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Get the internal prefix.
|
||||||
|
*/
|
||||||
|
public static function internalPrefix(): string
|
||||||
|
{
|
||||||
|
return 'tenancy_';
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get an internal key.
|
||||||
|
*/
|
||||||
|
public function getInternal(string $key)
|
||||||
|
{
|
||||||
|
return $this->getAttribute(static::internalPrefix() . $key);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set internal key.
|
||||||
|
*/
|
||||||
|
public function setInternal(string $key, $value)
|
||||||
|
{
|
||||||
|
$this->setAttribute(static::internalPrefix() . $key, $value);
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
}
|
||||||
26
src/Database/Concerns/TenantRun.php
Normal file
26
src/Database/Concerns/TenantRun.php
Normal file
|
|
@ -0,0 +1,26 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Stancl\Tenancy\Database\Concerns;
|
||||||
|
|
||||||
|
use Stancl\Tenancy\Contracts\Tenant;
|
||||||
|
|
||||||
|
trait TenantRun
|
||||||
|
{
|
||||||
|
public function run(callable $callback)
|
||||||
|
{
|
||||||
|
/** @var Tenant $this */
|
||||||
|
|
||||||
|
$originalTenant = tenant();
|
||||||
|
|
||||||
|
tenancy()->initialize($this);
|
||||||
|
$result = $callback($this);
|
||||||
|
|
||||||
|
if ($originalTenant) {
|
||||||
|
tenancy()->initialize($originalTenant);
|
||||||
|
} else {
|
||||||
|
tenancy()->end();
|
||||||
|
}
|
||||||
|
|
||||||
|
return $result;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -7,9 +7,8 @@ namespace Stancl\Tenancy\Database\Models;
|
||||||
use Illuminate\Database\Eloquent\Model;
|
use Illuminate\Database\Eloquent\Model;
|
||||||
use Stancl\Tenancy\Contracts;
|
use Stancl\Tenancy\Contracts;
|
||||||
use Stancl\Tenancy\Contracts\Tenant;
|
use Stancl\Tenancy\Contracts\Tenant;
|
||||||
use Stancl\Tenancy\Database\Concerns\CentralConnection;
|
use Stancl\Tenancy\Database\Concerns;
|
||||||
use Stancl\Tenancy\Events;
|
use Stancl\Tenancy\Events;
|
||||||
use Stancl\Tenancy\Exceptions\DomainOccupiedByOtherTenantException;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @property string $domain
|
* @property string $domain
|
||||||
|
|
@ -19,38 +18,24 @@ use Stancl\Tenancy\Exceptions\DomainOccupiedByOtherTenantException;
|
||||||
*/
|
*/
|
||||||
class Domain extends Model implements Contracts\Domain
|
class Domain extends Model implements Contracts\Domain
|
||||||
{
|
{
|
||||||
use CentralConnection;
|
use Concerns\CentralConnection,
|
||||||
|
Concerns\EnsuresDomainIsNotOccupied;
|
||||||
|
|
||||||
protected $guarded = [];
|
protected $guarded = [];
|
||||||
|
|
||||||
public static function boot()
|
|
||||||
{
|
|
||||||
parent::boot();
|
|
||||||
|
|
||||||
$ensureDomainIsNotOccupied = function (self $self) {
|
|
||||||
if ($domain = self::where('domain', $self->domain)->first()) {
|
|
||||||
if ($domain->getKey() !== $self->getKey()) {
|
|
||||||
throw new DomainOccupiedByOtherTenantException($self->domain);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
static::saving($ensureDomainIsNotOccupied);
|
|
||||||
}
|
|
||||||
|
|
||||||
public function tenant()
|
public function tenant()
|
||||||
{
|
{
|
||||||
return $this->belongsTo(config('tenancy.tenant_model'));
|
return $this->belongsTo(config('tenancy.tenant_model'));
|
||||||
}
|
}
|
||||||
|
|
||||||
public $dispatchesEvents = [
|
protected $dispatchesEvents = [
|
||||||
'saved' => Events\DomainSaved::class,
|
|
||||||
'saving' => Events\SavingDomain::class,
|
'saving' => Events\SavingDomain::class,
|
||||||
'created' => Events\DomainCreated::class,
|
'saved' => Events\DomainSaved::class,
|
||||||
'creating' => Events\CreatingDomain::class,
|
'creating' => Events\CreatingDomain::class,
|
||||||
'updated' => Events\DomainUpdated::class,
|
'created' => Events\DomainCreated::class,
|
||||||
'updating' => Events\UpdatingDomain::class,
|
'updating' => Events\UpdatingDomain::class,
|
||||||
'deleted' => Events\DomainDeleted::class,
|
'updated' => Events\DomainUpdated::class,
|
||||||
'deleting' => Events\DeletingDomain::class,
|
'deleting' => Events\DeletingDomain::class,
|
||||||
|
'deleted' => Events\DomainDeleted::class,
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -23,7 +23,9 @@ class Tenant extends Model implements Contracts\Tenant
|
||||||
{
|
{
|
||||||
use Concerns\CentralConnection,
|
use Concerns\CentralConnection,
|
||||||
Concerns\GeneratesIds,
|
Concerns\GeneratesIds,
|
||||||
Concerns\HasDataColumn;
|
Concerns\HasDataColumn,
|
||||||
|
Concerns\HasInternalKeys,
|
||||||
|
Concerns\TenantRun;
|
||||||
|
|
||||||
protected $table = 'tenants';
|
protected $table = 'tenants';
|
||||||
protected $primaryKey = 'id';
|
protected $primaryKey = 'id';
|
||||||
|
|
@ -44,53 +46,14 @@ class Tenant extends Model implements Contracts\Tenant
|
||||||
return new TenantCollection($models);
|
return new TenantCollection($models);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function internalPrefix(): string
|
protected $dispatchesEvents = [
|
||||||
{
|
|
||||||
return config('tenancy.internal_prefix');
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get an internal key.
|
|
||||||
*/
|
|
||||||
public function getInternal(string $key)
|
|
||||||
{
|
|
||||||
return $this->getAttribute(static::internalPrefix() . $key);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set internal key.
|
|
||||||
*/
|
|
||||||
public function setInternal(string $key, $value)
|
|
||||||
{
|
|
||||||
$this->setAttribute(static::internalPrefix() . $key, $value);
|
|
||||||
|
|
||||||
return $this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function run(callable $callback)
|
|
||||||
{
|
|
||||||
$originalTenant = tenant();
|
|
||||||
|
|
||||||
tenancy()->initialize($this);
|
|
||||||
$result = $callback($this);
|
|
||||||
|
|
||||||
if ($originalTenant) {
|
|
||||||
tenancy()->initialize($originalTenant);
|
|
||||||
} else {
|
|
||||||
tenancy()->end();
|
|
||||||
}
|
|
||||||
|
|
||||||
return $result;
|
|
||||||
}
|
|
||||||
|
|
||||||
public $dispatchesEvents = [
|
|
||||||
'saved' => Events\TenantSaved::class,
|
|
||||||
'saving' => Events\SavingTenant::class,
|
'saving' => Events\SavingTenant::class,
|
||||||
'created' => Events\TenantCreated::class,
|
'saved' => Events\TenantSaved::class,
|
||||||
'creating' => Events\CreatingTenant::class,
|
'creating' => Events\CreatingTenant::class,
|
||||||
'updated' => Events\TenantUpdated::class,
|
'created' => Events\TenantCreated::class,
|
||||||
'updating' => Events\UpdatingTenant::class,
|
'updating' => Events\UpdatingTenant::class,
|
||||||
'deleted' => Events\TenantDeleted::class,
|
'updated' => Events\TenantUpdated::class,
|
||||||
'deleting' => Events\DeletingTenant::class,
|
'deleting' => Events\DeletingTenant::class,
|
||||||
|
'deleted' => Events\TenantDeleted::class,
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -36,7 +36,6 @@ class TenantDatabaseManagerTest extends TestCase
|
||||||
|
|
||||||
config()->set([
|
config()->set([
|
||||||
"tenancy.database.managers.$driver" => $databaseManager,
|
"tenancy.database.managers.$driver" => $databaseManager,
|
||||||
'tenancy.internal_prefix' => 'tenancy_',
|
|
||||||
]);
|
]);
|
||||||
|
|
||||||
$name = 'db' . $this->randomString();
|
$name = 'db' . $this->randomString();
|
||||||
|
|
@ -65,8 +64,6 @@ class TenantDatabaseManagerTest extends TestCase
|
||||||
return $event->tenant;
|
return $event->tenant;
|
||||||
})->toListener());
|
})->toListener());
|
||||||
|
|
||||||
config(['tenancy.internal_prefix' => 'tenancy_']);
|
|
||||||
|
|
||||||
$database = 'db' . $this->randomString();
|
$database = 'db' . $this->randomString();
|
||||||
|
|
||||||
$mysqlmanager = app(MySQLDatabaseManager::class);
|
$mysqlmanager = app(MySQLDatabaseManager::class);
|
||||||
|
|
@ -113,7 +110,6 @@ class TenantDatabaseManagerTest extends TestCase
|
||||||
}
|
}
|
||||||
config([
|
config([
|
||||||
'database.connections.fooconn.driver' => 'sqlite',
|
'database.connections.fooconn.driver' => 'sqlite',
|
||||||
'tenancy.internal_prefix' => 'tenancy_',
|
|
||||||
]);
|
]);
|
||||||
|
|
||||||
$tenant = Tenant::create([
|
$tenant = Tenant::create([
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue