mirror of
https://github.com/archtechx/tenancy.git
synced 2026-05-06 14:24:04 +00:00
Skip DB deletion when create_database=false, add ignoreFailures (#1394)
Database deletion is now skipped by default if the tenant has the `create_database` internal attribute set to false, meaning it was likely created without a database. This skip can be opted out of by changing a static property. It also adds an opt-in static property for ignoring any other failures during database deletion, to allow continuing execution of the delete pipeline. --------- Co-authored-by: Samuel Štancl <samuel@archte.ch>
This commit is contained in:
parent
41701aff5f
commit
23b18c93a0
2 changed files with 101 additions and 2 deletions
|
|
@ -22,12 +22,34 @@ class DeleteDatabase implements ShouldQueue
|
||||||
protected TenantWithDatabase&Model $tenant,
|
protected TenantWithDatabase&Model $tenant,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
|
/** Skip database deletion if the create_database internal attribute is false. */
|
||||||
|
public static bool $skipWhenCreateDatabaseIsFalse = true;
|
||||||
|
|
||||||
|
/** Ignore exceptions thrown during database deletion and continue execution. */
|
||||||
|
public static bool $ignoreFailures = false;
|
||||||
|
|
||||||
public function handle(): void
|
public function handle(): void
|
||||||
{
|
{
|
||||||
|
if (static::$skipWhenCreateDatabaseIsFalse && $this->tenant->getInternal('create_database') === false) {
|
||||||
|
// If database creation was skipped, we presume deletion should also be skipped.
|
||||||
|
// To avoid this skip, either unset the `create_database` attribute (or make it true), or
|
||||||
|
// set the $skipWhenCreateDatabaseIsFalse static property to false.
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
event(new DeletingDatabase($this->tenant));
|
event(new DeletingDatabase($this->tenant));
|
||||||
|
|
||||||
$this->tenant->database()->manager()->deleteDatabase($this->tenant);
|
$deleted = false;
|
||||||
|
|
||||||
event(new DatabaseDeleted($this->tenant));
|
try {
|
||||||
|
$this->tenant->database()->manager()->deleteDatabase($this->tenant);
|
||||||
|
$deleted = true;
|
||||||
|
} catch (\Throwable $e) {
|
||||||
|
if (! static::$ignoreFailures) {
|
||||||
|
throw $e;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($deleted) event(new DatabaseDeleted($this->tenant));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -2,17 +2,27 @@
|
||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
use Illuminate\Database\QueryException;
|
||||||
use Illuminate\Support\Facades\Event;
|
use Illuminate\Support\Facades\Event;
|
||||||
use Illuminate\Support\Facades\Schema;
|
use Illuminate\Support\Facades\Schema;
|
||||||
use Stancl\JobPipeline\JobPipeline;
|
use Stancl\JobPipeline\JobPipeline;
|
||||||
use Stancl\Tenancy\Events\TenantCreated;
|
use Stancl\Tenancy\Events\TenantCreated;
|
||||||
|
use Stancl\Tenancy\Events\TenantDeleted;
|
||||||
use Stancl\Tenancy\Jobs\CreateDatabase;
|
use Stancl\Tenancy\Jobs\CreateDatabase;
|
||||||
|
use Stancl\Tenancy\Jobs\DeleteDatabase;
|
||||||
use Stancl\Tenancy\Jobs\MigrateDatabase;
|
use Stancl\Tenancy\Jobs\MigrateDatabase;
|
||||||
use Stancl\Tenancy\Jobs\SeedDatabase;
|
use Stancl\Tenancy\Jobs\SeedDatabase;
|
||||||
use Stancl\Tenancy\Tests\Etc\Tenant;
|
use Stancl\Tenancy\Tests\Etc\Tenant;
|
||||||
use Illuminate\Foundation\Auth\User as Authenticable;
|
use Illuminate\Foundation\Auth\User as Authenticable;
|
||||||
use Stancl\Tenancy\Tests\Etc\TestSeeder;
|
use Stancl\Tenancy\Tests\Etc\TestSeeder;
|
||||||
|
|
||||||
|
beforeEach($cleanup = function () {
|
||||||
|
DeleteDatabase::$ignoreFailures = false;
|
||||||
|
DeleteDatabase::$skipWhenCreateDatabaseIsFalse = true;
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach($cleanup);
|
||||||
|
|
||||||
test('database can be created after tenant creation', function () {
|
test('database can be created after tenant creation', function () {
|
||||||
config(['tenancy.database.template_tenant_connection' => 'mysql']);
|
config(['tenancy.database.template_tenant_connection' => 'mysql']);
|
||||||
|
|
||||||
|
|
@ -82,6 +92,73 @@ test('custom job can be added to the pipeline', function () {
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('database can be deleted after tenant deletion', function () {
|
||||||
|
Event::listen(TenantCreated::class, JobPipeline::make([CreateDatabase::class])->send(function (TenantCreated $event) {
|
||||||
|
return $event->tenant;
|
||||||
|
})->toListener());
|
||||||
|
|
||||||
|
Event::listen(TenantDeleted::class, JobPipeline::make([DeleteDatabase::class])->send(function (TenantDeleted $event) {
|
||||||
|
return $event->tenant;
|
||||||
|
})->toListener());
|
||||||
|
|
||||||
|
$tenant = Tenant::create();
|
||||||
|
$manager = $tenant->database()->manager();
|
||||||
|
|
||||||
|
expect($manager->databaseExists($tenant->database()->getName()))->toBeTrue();
|
||||||
|
|
||||||
|
$tenant->delete();
|
||||||
|
|
||||||
|
expect($manager->databaseExists($tenant->database()->getName()))->toBeFalse();
|
||||||
|
});
|
||||||
|
|
||||||
|
test('database deletion is skipped when create_database is false', function (bool $skipWhenCreateDatabaseIsFalse) {
|
||||||
|
Event::listen(TenantDeleted::class, JobPipeline::make([DeleteDatabase::class])->send(function (TenantDeleted $event) {
|
||||||
|
return $event->tenant;
|
||||||
|
})->toListener());
|
||||||
|
|
||||||
|
// create_database=false means no DB is created (e.g. tenant uses a pre-existing DB)
|
||||||
|
// On deletion, DeleteDatabase should skip rather than attempting DROP DATABASE on a non-existent DB
|
||||||
|
$tenant = Tenant::create(['tenancy_create_database' => false, 'tenancy_db_name' => 'non_existing_db']);
|
||||||
|
|
||||||
|
$manager = $tenant->database()->manager();
|
||||||
|
expect($manager->databaseExists($tenant->database()->getName()))->toBeFalse();
|
||||||
|
|
||||||
|
DeleteDatabase::$skipWhenCreateDatabaseIsFalse = $skipWhenCreateDatabaseIsFalse;
|
||||||
|
|
||||||
|
if ($skipWhenCreateDatabaseIsFalse) {
|
||||||
|
$tenant->delete(); // no exception
|
||||||
|
} else {
|
||||||
|
expect(fn () => $tenant->delete())->toThrow(QueryException::class, "database doesn't exist");
|
||||||
|
}
|
||||||
|
|
||||||
|
expect($manager->databaseExists($tenant->database()->getName()))->toBeFalse();
|
||||||
|
})->with([true, false]);
|
||||||
|
|
||||||
|
test('database deletion failure is ignored when ignoreFailures is true', function (bool $ignoreFailures) {
|
||||||
|
Event::listen(TenantCreated::class, JobPipeline::make([CreateDatabase::class])->send(function (TenantCreated $event) {
|
||||||
|
return $event->tenant;
|
||||||
|
})->toListener());
|
||||||
|
|
||||||
|
Event::listen(TenantDeleted::class, JobPipeline::make([DeleteDatabase::class])->send(function (TenantDeleted $event) {
|
||||||
|
return $event->tenant;
|
||||||
|
})->toListener());
|
||||||
|
|
||||||
|
DeleteDatabase::$ignoreFailures = $ignoreFailures;
|
||||||
|
|
||||||
|
$tenant = Tenant::create();
|
||||||
|
$manager = $tenant->database()->manager();
|
||||||
|
expect($manager->databaseExists($tenant->database()->getName()))->toBeTrue();
|
||||||
|
|
||||||
|
$manager->deleteDatabase($tenant); // manually delete so the job fails
|
||||||
|
expect($manager->databaseExists($tenant->database()->getName()))->toBeFalse();
|
||||||
|
|
||||||
|
if ($ignoreFailures) {
|
||||||
|
$tenant->delete(); // no exception
|
||||||
|
} else {
|
||||||
|
expect(fn () => $tenant->delete())->toThrow(QueryException::class, "database doesn't exist");
|
||||||
|
}
|
||||||
|
})->with([true, false]);
|
||||||
|
|
||||||
class User extends Authenticable
|
class User extends Authenticable
|
||||||
{
|
{
|
||||||
protected $guarded = [];
|
protected $guarded = [];
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue