mirror of
https://github.com/archtechx/tenancy.git
synced 2025-12-12 15:54:03 +00:00
Tenant contract, Tenancy bootstrappers, drop predis
This commit is contained in:
parent
98cb49b1c0
commit
f04ca349bd
13 changed files with 158 additions and 47 deletions
|
|
@ -26,11 +26,10 @@ return [
|
||||||
'suffix' => '',
|
'suffix' => '',
|
||||||
],
|
],
|
||||||
'redis' => [
|
'redis' => [
|
||||||
'tenancy' => false, // to enable Redis tenancy, you must use phpredis
|
|
||||||
'prefix_base' => 'tenant',
|
'prefix_base' => 'tenant',
|
||||||
'prefixed_connections' => [
|
'prefixed_connections' => [
|
||||||
'default',
|
// 'default',
|
||||||
'cache',
|
// 'cache',
|
||||||
],
|
],
|
||||||
],
|
],
|
||||||
'cache' => [
|
'cache' => [
|
||||||
|
|
@ -51,10 +50,14 @@ return [
|
||||||
],
|
],
|
||||||
],
|
],
|
||||||
'database_managers' => [
|
'database_managers' => [
|
||||||
|
// Tenant database managers handle the creation & deletion of tenant databases.
|
||||||
'sqlite' => 'Stancl\Tenancy\TenantDatabaseManagers\SQLiteDatabaseManager',
|
'sqlite' => 'Stancl\Tenancy\TenantDatabaseManagers\SQLiteDatabaseManager',
|
||||||
'mysql' => 'Stancl\Tenancy\TenantDatabaseManagers\MySQLDatabaseManager',
|
'mysql' => 'Stancl\Tenancy\TenantDatabaseManagers\MySQLDatabaseManager',
|
||||||
'pgsql' => 'Stancl\Tenancy\TenantDatabaseManagers\PostgreSQLDatabaseManager',
|
'pgsql' => 'Stancl\Tenancy\TenantDatabaseManagers\PostgreSQLDatabaseManager',
|
||||||
],
|
],
|
||||||
|
'tenancy_bootstrappers' => [
|
||||||
|
''
|
||||||
|
],
|
||||||
'queue_database_creation' => false,
|
'queue_database_creation' => false,
|
||||||
'queue_database_deletion' => false,
|
'queue_database_deletion' => false,
|
||||||
'unique_id_generator' => 'Stancl\Tenancy\UUIDGenerator',
|
'unique_id_generator' => 'Stancl\Tenancy\UUIDGenerator',
|
||||||
|
|
|
||||||
|
|
@ -11,8 +11,7 @@
|
||||||
],
|
],
|
||||||
"require": {
|
"require": {
|
||||||
"illuminate/support": "5.8.*",
|
"illuminate/support": "5.8.*",
|
||||||
"webpatser/laravel-uuid": "^3.0",
|
"webpatser/laravel-uuid": "^3.0"
|
||||||
"predis/predis": "^1.1"
|
|
||||||
},
|
},
|
||||||
"require-dev": {
|
"require-dev": {
|
||||||
"vlucas/phpdotenv": "^3.3",
|
"vlucas/phpdotenv": "^3.3",
|
||||||
|
|
|
||||||
6
src/Contracts/Tenant.php
Normal file
6
src/Contracts/Tenant.php
Normal file
|
|
@ -0,0 +1,6 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Stancl\Tenancy\Contracts;
|
||||||
|
|
||||||
|
/** Empty interface implemented by Stancl\Tenancy\Tenant have a dependency-injectable contract for the current tenant. */
|
||||||
|
interface Tenant {}
|
||||||
|
|
@ -1,10 +0,0 @@
|
||||||
<?php
|
|
||||||
|
|
||||||
declare(strict_types=1);
|
|
||||||
|
|
||||||
namespace Stancl\Tenancy\Exceptions;
|
|
||||||
|
|
||||||
class PhpRedisNotInstalledException extends \Exception
|
|
||||||
{
|
|
||||||
protected $message = 'PhpRedis is not installed. PhpRedis is required for Redis multi-tenancy because Predis does not support prefixes.';
|
|
||||||
}
|
|
||||||
26
src/TenancyBootstrappers/CacheTenancyBoostrapper.php
Normal file
26
src/TenancyBootstrappers/CacheTenancyBoostrapper.php
Normal file
|
|
@ -0,0 +1,26 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Stancl\Tenancy\TenancyBoostrappers;
|
||||||
|
|
||||||
|
use Stancl\Tenancy\Contracts\TenancyBootstrapper;
|
||||||
|
|
||||||
|
class CacheTenancyBootstrapped implements TenancyBootstrapper
|
||||||
|
{
|
||||||
|
/** @var \Illuminate\Cache\CacheManager */
|
||||||
|
protected $originalCache;
|
||||||
|
|
||||||
|
public function start()
|
||||||
|
{
|
||||||
|
$this->originalCache = $this->originalCache ?? $this->app['cache'];
|
||||||
|
$this->app->extend('cache', function () {
|
||||||
|
return new CacheManager($this->app);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public function end()
|
||||||
|
{
|
||||||
|
$this->app->extend('cache', function () {
|
||||||
|
return $this->originalCache;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
57
src/TenancyBootstrappers/FilesystemTenancyBootstrapper.php
Normal file
57
src/TenancyBootstrappers/FilesystemTenancyBootstrapper.php
Normal file
|
|
@ -0,0 +1,57 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Stancl\Tenancy\TenancyBoostrappers;
|
||||||
|
|
||||||
|
use Stancl\Tenancy\Contracts\TenancyBootstrapper;
|
||||||
|
|
||||||
|
// todo better solution than tenant_asset?
|
||||||
|
|
||||||
|
class FilesystemTenancyBootstrapper implements TenancyBootstrapper
|
||||||
|
{
|
||||||
|
protected $originalPaths = [];
|
||||||
|
|
||||||
|
/** @var Application */
|
||||||
|
protected $app;
|
||||||
|
|
||||||
|
public function __construct(Application $app)
|
||||||
|
{
|
||||||
|
$this->app = $app;
|
||||||
|
$this->originalPaths = [
|
||||||
|
'disks' => [],
|
||||||
|
'path' => $this->app->storagePath(),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
public function start()
|
||||||
|
{
|
||||||
|
// todo revisit this
|
||||||
|
$suffix = $this->app['config']['tenancy.filesystem.suffix_base'] . tenant('uuid');
|
||||||
|
|
||||||
|
// storage_path()
|
||||||
|
$this->app->useStoragePath($this->originalPaths['path'] . "/{$suffix}");
|
||||||
|
|
||||||
|
// Storage facade
|
||||||
|
foreach ($this->app['config']['tenancy.filesystem.disks'] as $disk) {
|
||||||
|
$this->originalPaths['disks'][$disk] = Storage::disk($disk)->getAdapter()->getPathPrefix();
|
||||||
|
|
||||||
|
if ($root = \str_replace('%storage_path%', storage_path(), $this->app['config']["tenancy.filesystem.root_override.{$disk}"])) {
|
||||||
|
Storage::disk($disk)->getAdapter()->setPathPrefix($root);
|
||||||
|
} else {
|
||||||
|
$root = $this->app['config']["filesystems.disks.{$disk}.root"];
|
||||||
|
|
||||||
|
Storage::disk($disk)->getAdapter()->setPathPrefix($root . "/{$suffix}");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function end()
|
||||||
|
{
|
||||||
|
// storage_path()
|
||||||
|
$this->app->useStoragePath($this->originalPaths['path']);
|
||||||
|
|
||||||
|
// Storage facade
|
||||||
|
foreach ($this->app['config']['tenancy.filesystem.disks'] as $disk) {
|
||||||
|
Storage::disk($disk)->getAdapter()->setPathPrefix($this->originalPaths['disks'][$disk]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
44
src/TenancyBootstrappers/RedisTenancyBootstrapper.php
Normal file
44
src/TenancyBootstrappers/RedisTenancyBootstrapper.php
Normal file
|
|
@ -0,0 +1,44 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Stancl\Tenancy\TenantDatabaseManagers;
|
||||||
|
|
||||||
|
use Stancl\Tenancy\Contracts\TenancyBootstrapper;
|
||||||
|
|
||||||
|
class RedisTenancyBootstrapper implements TenancyBootstrapper
|
||||||
|
{
|
||||||
|
/** @var string[string] Original prefixes of connections */
|
||||||
|
protected $originalPrefixes = [];
|
||||||
|
|
||||||
|
/** @var Application */
|
||||||
|
protected $app;
|
||||||
|
|
||||||
|
public function __construct(Application $app)
|
||||||
|
{
|
||||||
|
$this->app = $app;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function start()
|
||||||
|
{
|
||||||
|
foreach ($this->prefixedConnections() as $connection) {
|
||||||
|
$prefix = $this->app['config']['tenancy.redis.prefix_base'] . $this->tenant['uuid'];
|
||||||
|
$client = Redis::connection($connection)->client();
|
||||||
|
|
||||||
|
$this->originalPrefixes[$connection] = $client->getOption($client::OPT_PREFIX);
|
||||||
|
$client->setOption($client::OPT_PREFIX, $prefix);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function end()
|
||||||
|
{
|
||||||
|
foreach ($this->prefixedConnections() as $connection) {
|
||||||
|
$client = Redis::connection($connection)->client();
|
||||||
|
|
||||||
|
$client->setOption($client::OPT_PREFIX, $this->originalPrefixes[$connection]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function prefixedConnections()
|
||||||
|
{
|
||||||
|
return config('tenancy.redis.prefixed_connections');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -132,6 +132,8 @@ class TenancyServiceProvider extends ServiceProvider
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// todo foreach bootstrappers, singleton
|
||||||
|
|
||||||
$this->app->singleton(Migrate::class, function ($app) {
|
$this->app->singleton(Migrate::class, function ($app) {
|
||||||
return new Migrate($app['migrator'], $app[DatabaseManager::class]);
|
return new Migrate($app['migrator'], $app[DatabaseManager::class]);
|
||||||
});
|
});
|
||||||
|
|
|
||||||
|
|
@ -6,12 +6,10 @@ namespace Stancl\Tenancy;
|
||||||
|
|
||||||
use ArrayAccess;
|
use ArrayAccess;
|
||||||
|
|
||||||
// todo laravel events instead of custom events?
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @internal Class is subject to breaking changes in minor and patch versions.
|
* @internal Class is subject to breaking changes in minor and patch versions.
|
||||||
*/
|
*/
|
||||||
class Tenant implements ArrayAccess
|
class Tenant implements ArrayAccess, Contracts\Tenant
|
||||||
{
|
{
|
||||||
use Traits\HasArrayAccess;
|
use Traits\HasArrayAccess;
|
||||||
|
|
||||||
|
|
@ -56,7 +54,7 @@ class Tenant implements ArrayAccess
|
||||||
return app(static::class)->withData($data)->persisted();
|
return app(static::class)->withData($data)->persisted();
|
||||||
}
|
}
|
||||||
|
|
||||||
public function persisted()
|
protected function persisted()
|
||||||
{
|
{
|
||||||
$this->persisted = true;
|
$this->persisted = true;
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -55,7 +55,16 @@ class TenantManagerv2
|
||||||
public function bootstrapTenancy(Tenant $tenant): self
|
public function bootstrapTenancy(Tenant $tenant): self
|
||||||
{
|
{
|
||||||
foreach($this->tenancyBootstrappers() as $bootstrapper) {
|
foreach($this->tenancyBootstrappers() as $bootstrapper) {
|
||||||
$bootstrapper::start($tenant);
|
$this->app[$bootstrapper]->start($tenant);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function endTenancy(): self
|
||||||
|
{
|
||||||
|
foreach($this->tenancyBootstrappers() as $bootstrapper) {
|
||||||
|
$this->app[$bootstrapper]->end();
|
||||||
}
|
}
|
||||||
|
|
||||||
return $this;
|
return $this;
|
||||||
|
|
|
||||||
|
|
@ -6,6 +6,8 @@ namespace Stancl\Tenancy;
|
||||||
|
|
||||||
use Illuminate\Database\Eloquent\Model;
|
use Illuminate\Database\Eloquent\Model;
|
||||||
|
|
||||||
|
// todo move this to a database driver domain?
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @final Class is subject to breaking changes in minor and patch versions.
|
* @final Class is subject to breaking changes in minor and patch versions.
|
||||||
*/
|
*/
|
||||||
|
|
|
||||||
6
test
6
test
|
|
@ -4,9 +4,7 @@ set -e
|
||||||
# for development
|
# for development
|
||||||
docker-compose up -d
|
docker-compose up -d
|
||||||
printf "Variant 1\n\n"
|
printf "Variant 1\n\n"
|
||||||
docker-compose exec test env TENANCY_TEST_REDIS_TENANCY=1 TENANCY_TEST_REDIS_CLIENT=phpredis TENANCY_TEST_STORAGE_DRIVER=redis vendor/bin/phpunit --coverage-php coverage/1.cov "$@"
|
docker-compose exec test env TENANCY_TEST_STORAGE_DRIVER=redis vendor/bin/phpunit --coverage-php coverage/1.cov "$@"
|
||||||
printf "Variant 2\n\n"
|
printf "Variant 2\n\n"
|
||||||
docker-compose exec test env TENANCY_TEST_REDIS_TENANCY=0 TENANCY_TEST_REDIS_CLIENT=predis TENANCY_TEST_STORAGE_DRIVER=redis vendor/bin/phpunit --coverage-php coverage/2.cov "$@"
|
docker-compose exec test env TENANCY_TEST_STORAGE_DRIVER=db vendor/bin/phpunit --coverage-php coverage/3.cov "$@"
|
||||||
printf "Variant 3\n\n"
|
|
||||||
docker-compose exec test env TENANCY_TEST_REDIS_TENANCY=1 TENANCY_TEST_REDIS_CLIENT=phpredis TENANCY_TEST_STORAGE_DRIVER=db vendor/bin/phpunit --coverage-php coverage/3.cov "$@"
|
|
||||||
docker-compose exec test vendor/bin/phpcov merge --clover clover.xml coverage/
|
docker-compose exec test vendor/bin/phpcov merge --clover clover.xml coverage/
|
||||||
|
|
|
||||||
|
|
@ -6,7 +6,6 @@ namespace Stancl\Tenancy\Tests;
|
||||||
|
|
||||||
use Illuminate\Support\Facades\Redis;
|
use Illuminate\Support\Facades\Redis;
|
||||||
use Illuminate\Support\Facades\Config;
|
use Illuminate\Support\Facades\Config;
|
||||||
use Stancl\Tenancy\Exceptions\PhpRedisNotInstalledException;
|
|
||||||
|
|
||||||
class BootstrapsTenancyTest extends TestCase
|
class BootstrapsTenancyTest extends TestCase
|
||||||
{
|
{
|
||||||
|
|
@ -38,28 +37,6 @@ class BootstrapsTenancyTest extends TestCase
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/** @test */
|
|
||||||
public function predis_is_supported()
|
|
||||||
{
|
|
||||||
Config::set('database.redis.client', 'predis');
|
|
||||||
Redis::setDriver('predis');
|
|
||||||
Config::set('tenancy.redis.tenancy', false);
|
|
||||||
|
|
||||||
// assert no exception is thrown from initializing tenancy
|
|
||||||
$this->assertNotNull($this->initTenancy());
|
|
||||||
}
|
|
||||||
|
|
||||||
/** @test */
|
|
||||||
public function predis_is_not_supported_without_disabling_redis_multitenancy()
|
|
||||||
{
|
|
||||||
Config::set('database.redis.client', 'predis');
|
|
||||||
Redis::setDriver('predis');
|
|
||||||
Config::set('tenancy.redis.tenancy', true);
|
|
||||||
|
|
||||||
$this->expectException(PhpRedisNotInstalledException::class);
|
|
||||||
$this->initTenancy();
|
|
||||||
}
|
|
||||||
|
|
||||||
/** @test */
|
/** @test */
|
||||||
public function filesystem_is_suffixed()
|
public function filesystem_is_suffixed()
|
||||||
{
|
{
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue