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

[2.1.0] Fix queue race conditions (#178)

* Add TenantDoesNotExist exception, fix queued migrator serialization

* wip

* Apply fixes from StyleCI

* Use only one config for  queueing

* Apply fixes from StyleCI

* rename test

* rename test
This commit is contained in:
Samuel Štancl 2019-10-17 13:55:10 +02:00 committed by GitHub
parent e98db460ec
commit 3e78410d8a
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
8 changed files with 78 additions and 28 deletions

View file

@ -89,10 +89,9 @@ return [
// 'paypal_api_key' => 'services.paypal.api_key', // 'paypal_api_key' => 'services.paypal.api_key',
], ],
'home_url' => '/app', 'home_url' => '/app',
'migrate_after_creation' => false, // run migrations after creating a tenant
'queue_automatic_migration' => false, // queue the automatic post-tenant-creation migrations
'delete_database_after_tenant_deletion' => false, // delete the tenant's database after deleting the tenant
'queue_database_creation' => false, 'queue_database_creation' => false,
'migrate_after_creation' => false, // run migrations after creating a tenant
'queue_database_deletion' => false, 'queue_database_deletion' => false,
'delete_database_after_tenant_deletion' => false, // delete the tenant's database after deleting the tenant
'unique_id_generator' => Stancl\Tenancy\UniqueIDGenerators\UUIDGenerator::class, 'unique_id_generator' => Stancl\Tenancy\UniqueIDGenerators\UUIDGenerator::class,
]; ];

View file

@ -126,17 +126,21 @@ class DatabaseManager
* Create a database for a tenant. * Create a database for a tenant.
* *
* @param Tenant $tenant * @param Tenant $tenant
* @param \Illuminate\Contracts\Queue\ShouldQueue[]|callable[] $afterCreating
* @return void * @return void
*/ */
public function createDatabase(Tenant $tenant) public function createDatabase(Tenant $tenant, array $afterCreating = [])
{ {
$database = $tenant->getDatabaseName(); $database = $tenant->getDatabaseName();
$manager = $this->getTenantDatabaseManager($tenant); $manager = $this->getTenantDatabaseManager($tenant);
if ($this->app['config']['tenancy.queue_database_creation'] ?? false) { if ($this->app['config']['tenancy.queue_database_creation'] ?? false) {
QueuedTenantDatabaseCreator::dispatch($manager, $database); QueuedTenantDatabaseCreator::withChain($afterCreating)->dispatch($manager, $database);
} else { } else {
$manager->createDatabase($database); $manager->createDatabase($database);
foreach ($afterCreating as $callback) {
$callback();
}
} }
} }

View file

@ -0,0 +1,15 @@
<?php
declare(strict_types=1);
namespace Stancl\Tenancy\Exceptions;
use Exception;
class TenantDoesNotExistException extends Exception
{
public function __construct(string $id)
{
$this->message = "Tenant with this id does not exist: $id";
}
}

View file

@ -16,12 +16,12 @@ class QueuedTenantDatabaseMigrator implements ShouldQueue
{ {
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels; use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
/** @var Tenant */ /** @var string */
protected $tenant; protected $tenantId;
public function __construct(Tenant $tenant) public function __construct(Tenant $tenant)
{ {
$this->tenant = $tenant; $this->tenantId = $tenant->id;
} }
/** /**
@ -32,7 +32,7 @@ class QueuedTenantDatabaseMigrator implements ShouldQueue
public function handle() public function handle()
{ {
Artisan::call('tenants:migrate', [ Artisan::call('tenants:migrate', [
'--tenants' => [$this->tenant->id], '--tenants' => [$this->tenantId],
]); ]);
} }
} }

View file

@ -10,6 +10,7 @@ use Stancl\Tenancy\Contracts\StorageDriver;
use Stancl\Tenancy\DatabaseManager; use Stancl\Tenancy\DatabaseManager;
use Stancl\Tenancy\Exceptions\DomainsOccupiedByOtherTenantException; use Stancl\Tenancy\Exceptions\DomainsOccupiedByOtherTenantException;
use Stancl\Tenancy\Exceptions\TenantCouldNotBeIdentifiedException; use Stancl\Tenancy\Exceptions\TenantCouldNotBeIdentifiedException;
use Stancl\Tenancy\Exceptions\TenantDoesNotExistException;
use Stancl\Tenancy\Exceptions\TenantWithThisIdAlreadyExistsException; use Stancl\Tenancy\Exceptions\TenantWithThisIdAlreadyExistsException;
use Stancl\Tenancy\StorageDrivers\Database\DomainModel as Domains; use Stancl\Tenancy\StorageDrivers\Database\DomainModel as Domains;
use Stancl\Tenancy\StorageDrivers\Database\TenantModel as Tenants; use Stancl\Tenancy\StorageDrivers\Database\TenantModel as Tenants;
@ -59,7 +60,13 @@ class DatabaseStorageDriver implements StorageDriver
public function findById(string $id): Tenant public function findById(string $id): Tenant
{ {
return Tenant::fromStorage(Tenants::find($id)->decoded()) $tenant = Tenants::find($id);
if (! $tenant) {
throw new TenantDoesNotExistException($id);
}
return Tenant::fromStorage($tenant->decoded())
->withDomains($this->getTenantDomains($id)); ->withDomains($this->getTenantDomains($id));
} }

View file

@ -9,6 +9,7 @@ use Illuminate\Foundation\Application;
use Stancl\Tenancy\Contracts\StorageDriver; use Stancl\Tenancy\Contracts\StorageDriver;
use Stancl\Tenancy\Exceptions\DomainsOccupiedByOtherTenantException; use Stancl\Tenancy\Exceptions\DomainsOccupiedByOtherTenantException;
use Stancl\Tenancy\Exceptions\TenantCouldNotBeIdentifiedException; use Stancl\Tenancy\Exceptions\TenantCouldNotBeIdentifiedException;
use Stancl\Tenancy\Exceptions\TenantDoesNotExistException;
use Stancl\Tenancy\Exceptions\TenantWithThisIdAlreadyExistsException; use Stancl\Tenancy\Exceptions\TenantWithThisIdAlreadyExistsException;
use Stancl\Tenancy\Tenant; use Stancl\Tenancy\Tenant;
@ -73,7 +74,13 @@ class RedisStorageDriver implements StorageDriver
public function findById(string $id): Tenant public function findById(string $id): Tenant
{ {
return $this->makeTenant($this->redis->hgetall("tenants:$id")); $data = $this->redis->hgetall("tenants:$id");
if (! $data) {
throw new TenantDoesNotExistException($id);
}
return $this->makeTenant($data);
} }
public function getTenantIdByDomain(string $domain): ?string public function getTenantIdByDomain(string $domain): ?string

View file

@ -62,18 +62,24 @@ class TenantManager
$this->ensureTenantCanBeCreated($tenant); $this->ensureTenantCanBeCreated($tenant);
$this->storage->createTenant($tenant); $this->storage->createTenant($tenant);
$this->database->createDatabase($tenant);
/** @var \Illuminate\Contracts\Queue\ShouldQueue[]|callable[] $afterCreating */
$afterCreating = [];
if ($this->shouldMigrateAfterCreation()) { if ($this->shouldMigrateAfterCreation()) {
if ($this->shouldQueueMigration()) { $afterCreating += $this->databaseCreationQueued() ? [
QueuedTenantDatabaseMigrator::dispatch($tenant); new QueuedTenantDatabaseMigrator($tenant),
} else { ] : [
$this->artisan->call('tenants:migrate', [ function () use ($tenant) {
'--tenants' => [$tenant['id']], $this->artisan->call('tenants:migrate', [
]); '--tenants' => [$tenant['id']],
} ]);
},
];
} }
$this->database->createDatabase($tenant, $afterCreating);
return $this; return $this;
} }
@ -315,9 +321,9 @@ class TenantManager
return $this->app['config']['tenancy.migrate_after_creation'] ?? false; return $this->app['config']['tenancy.migrate_after_creation'] ?? false;
} }
public function shouldQueueMigration(): bool public function databaseCreationQueued(): bool
{ {
return $this->app['config']['tenancy.queue_automatic_migration'] ?? false; return $this->app['config']['tenancy.queue_database_creation'] ?? false;
} }
public function shouldDeleteDatabase(): bool public function shouldDeleteDatabase(): bool

View file

@ -8,7 +8,9 @@ use Illuminate\Support\Facades\DB;
use Illuminate\Support\Facades\Queue; use Illuminate\Support\Facades\Queue;
use Illuminate\Support\Facades\Storage; use Illuminate\Support\Facades\Storage;
use Stancl\Tenancy\Exceptions\DomainsOccupiedByOtherTenantException; use Stancl\Tenancy\Exceptions\DomainsOccupiedByOtherTenantException;
use Stancl\Tenancy\Exceptions\TenantDoesNotExistException;
use Stancl\Tenancy\Exceptions\TenantWithThisIdAlreadyExistsException; use Stancl\Tenancy\Exceptions\TenantWithThisIdAlreadyExistsException;
use Stancl\Tenancy\Jobs\QueuedTenantDatabaseCreator;
use Stancl\Tenancy\Jobs\QueuedTenantDatabaseMigrator; use Stancl\Tenancy\Jobs\QueuedTenantDatabaseMigrator;
use Stancl\Tenancy\Tenant; use Stancl\Tenancy\Tenant;
use Stancl\Tenancy\TenantManager; use Stancl\Tenancy\TenantManager;
@ -248,22 +250,32 @@ class TenantManagerTest extends TestCase
} }
/** @test */ /** @test */
public function automigration_can_be_queued() public function automigration_is_queued_when_db_creation_is_queued()
{ {
Queue::fake(); Queue::fake();
config([ config([
'tenancy.queue_database_creation' => true,
'tenancy.migrate_after_creation' => true, 'tenancy.migrate_after_creation' => true,
'tenancy.queue_automatic_migration' => true,
]); ]);
$tenant = Tenant::new()->save(); $tenant = Tenant::new()->save();
tenancy()->initialize($tenant);
Queue::assertPushed(QueuedTenantDatabaseMigrator::class); Queue::assertPushedWithChain(QueuedTenantDatabaseCreator::class, [
QueuedTenantDatabaseMigrator::class,
]);
$this->assertFalse(\Schema::hasTable('users')); // foreach (Queue::pushedJobs() as $job) {
(new QueuedTenantDatabaseMigrator($tenant))->handle(); // $job[0]['job']->handle(); // this doesn't execute the chained job
$this->assertTrue(\Schema::hasTable('users')); // }
// tenancy()->initialize($tenant);
// $this->assertTrue(\Schema::hasTable('users'));
}
/** @test */
public function TenantDoesNotExistException_is_thrown_when_find_is_called_on_an_id_that_does_not_belong_to_any_tenant()
{
$this->expectException(TenantDoesNotExistException::class);
tenancy()->find('gjnfdgf');
} }
} }