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()
|
||||
{
|
||||
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'));
|
||||
}
|
||||
}
|
||||
|
|
|
|||
|
|
@ -7,20 +7,20 @@ use Stancl\Tenancy\Database\Models\Tenant;
|
|||
|
||||
return [
|
||||
'tenant_model' => Tenant::class,
|
||||
'internal_prefix' => 'tenancy_',
|
||||
'id_generator' => Stancl\Tenancy\UUIDGenerator::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' => [
|
||||
'127.0.0.1',
|
||||
'localhost',
|
||||
],
|
||||
|
||||
/**
|
||||
* Controller namespace used by routes in routes/tenant.php.
|
||||
*/
|
||||
'tenant_route_namespace' => 'App\Http\Controllers',
|
||||
|
||||
/**
|
||||
* Tenancy bootstrappers are executed when tenancy is initialized.
|
||||
* 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.
|
||||
*
|
||||
|
|
@ -151,6 +135,22 @@ return [
|
|||
'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
|
||||
* 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()
|
||||
{
|
||||
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));
|
||||
}
|
||||
});
|
||||
|
|
@ -19,6 +19,11 @@ trait GeneratesIds
|
|||
|
||||
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 Stancl\Tenancy\Contracts;
|
||||
use Stancl\Tenancy\Contracts\Tenant;
|
||||
use Stancl\Tenancy\Database\Concerns\CentralConnection;
|
||||
use Stancl\Tenancy\Database\Concerns;
|
||||
use Stancl\Tenancy\Events;
|
||||
use Stancl\Tenancy\Exceptions\DomainOccupiedByOtherTenantException;
|
||||
|
||||
/**
|
||||
* @property string $domain
|
||||
|
|
@ -19,38 +18,24 @@ use Stancl\Tenancy\Exceptions\DomainOccupiedByOtherTenantException;
|
|||
*/
|
||||
class Domain extends Model implements Contracts\Domain
|
||||
{
|
||||
use CentralConnection;
|
||||
use Concerns\CentralConnection,
|
||||
Concerns\EnsuresDomainIsNotOccupied;
|
||||
|
||||
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()
|
||||
{
|
||||
return $this->belongsTo(config('tenancy.tenant_model'));
|
||||
}
|
||||
|
||||
public $dispatchesEvents = [
|
||||
'saved' => Events\DomainSaved::class,
|
||||
protected $dispatchesEvents = [
|
||||
'saving' => Events\SavingDomain::class,
|
||||
'created' => Events\DomainCreated::class,
|
||||
'saved' => Events\DomainSaved::class,
|
||||
'creating' => Events\CreatingDomain::class,
|
||||
'updated' => Events\DomainUpdated::class,
|
||||
'created' => Events\DomainCreated::class,
|
||||
'updating' => Events\UpdatingDomain::class,
|
||||
'deleted' => Events\DomainDeleted::class,
|
||||
'updated' => Events\DomainUpdated::class,
|
||||
'deleting' => Events\DeletingDomain::class,
|
||||
'deleted' => Events\DomainDeleted::class,
|
||||
];
|
||||
}
|
||||
|
|
|
|||
|
|
@ -23,7 +23,9 @@ class Tenant extends Model implements Contracts\Tenant
|
|||
{
|
||||
use Concerns\CentralConnection,
|
||||
Concerns\GeneratesIds,
|
||||
Concerns\HasDataColumn;
|
||||
Concerns\HasDataColumn,
|
||||
Concerns\HasInternalKeys,
|
||||
Concerns\TenantRun;
|
||||
|
||||
protected $table = 'tenants';
|
||||
protected $primaryKey = 'id';
|
||||
|
|
@ -44,53 +46,14 @@ class Tenant extends Model implements Contracts\Tenant
|
|||
return new TenantCollection($models);
|
||||
}
|
||||
|
||||
public static function internalPrefix(): string
|
||||
{
|
||||
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,
|
||||
protected $dispatchesEvents = [
|
||||
'saving' => Events\SavingTenant::class,
|
||||
'created' => Events\TenantCreated::class,
|
||||
'saved' => Events\TenantSaved::class,
|
||||
'creating' => Events\CreatingTenant::class,
|
||||
'updated' => Events\TenantUpdated::class,
|
||||
'created' => Events\TenantCreated::class,
|
||||
'updating' => Events\UpdatingTenant::class,
|
||||
'deleted' => Events\TenantDeleted::class,
|
||||
'updated' => Events\TenantUpdated::class,
|
||||
'deleting' => Events\DeletingTenant::class,
|
||||
'deleted' => Events\TenantDeleted::class,
|
||||
];
|
||||
}
|
||||
|
|
|
|||
|
|
@ -36,7 +36,6 @@ class TenantDatabaseManagerTest extends TestCase
|
|||
|
||||
config()->set([
|
||||
"tenancy.database.managers.$driver" => $databaseManager,
|
||||
'tenancy.internal_prefix' => 'tenancy_',
|
||||
]);
|
||||
|
||||
$name = 'db' . $this->randomString();
|
||||
|
|
@ -65,8 +64,6 @@ class TenantDatabaseManagerTest extends TestCase
|
|||
return $event->tenant;
|
||||
})->toListener());
|
||||
|
||||
config(['tenancy.internal_prefix' => 'tenancy_']);
|
||||
|
||||
$database = 'db' . $this->randomString();
|
||||
|
||||
$mysqlmanager = app(MySQLDatabaseManager::class);
|
||||
|
|
@ -113,7 +110,6 @@ class TenantDatabaseManagerTest extends TestCase
|
|||
}
|
||||
config([
|
||||
'database.connections.fooconn.driver' => 'sqlite',
|
||||
'tenancy.internal_prefix' => 'tenancy_',
|
||||
]);
|
||||
|
||||
$tenant = Tenant::create([
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue