1
0
Fork 0
mirror of https://github.com/archtechx/tenancy.git synced 2025-12-12 14:14:04 +00:00

Refactor models & config

This commit is contained in:
Samuel Štancl 2020-05-24 20:40:10 +02:00
parent f0ff8248e7
commit 2839f45196
9 changed files with 123 additions and 97 deletions

View file

@ -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'));
}
}

View file

@ -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

View 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);
}
}
});
}
}

View file

@ -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);
}
}

View 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;
}
}

View 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;
}
}

View file

@ -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,
];
}

View file

@ -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,
];
}

View file

@ -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([