1
0
Fork 0
mirror of https://github.com/archtechx/tenancy.git synced 2026-06-20 22:54:05 +00:00
This commit is contained in:
lukinovec 2026-06-12 12:39:53 +02:00 committed by GitHub
commit 5e1cef7399
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
2 changed files with 88 additions and 17 deletions

View file

@ -100,27 +100,51 @@ trait HasPending
*/ */
public static function pullPendingFromPool(bool $firstOrCreate = false, array $attributes = []): ?Tenant public static function pullPendingFromPool(bool $firstOrCreate = false, array $attributes = []): ?Tenant
{ {
$tenant = DB::transaction(function () use ($attributes): ?Tenant { // Attempt pulling a pending tenant.
/** @var (Model&Tenant)|null $tenant */ // The loop handles the case where a single tenant is being pulled by multiple processes at the same time.
$tenant = static::onlyPending()->first(); // If a tenant was pulled by a concurrent process, try pulling the next one in the pool.
while (true) {
/** @var (Model&Tenant)|null $pullCandidate */
$pullCandidate = static::onlyPending()->first();
if ($tenant !== null) { if ($pullCandidate === null) {
event(new PullingPendingTenant($tenant)); return $firstOrCreate ? static::create($attributes) : null;
$tenant->update(array_merge($attributes, [
'pending_since' => null,
]));
} }
// Fired before the claim, so it can fire once per attempt, including for a candidate
// that ends up being claimed by a concurrent process (in which case the loop retries).
// PendingTenantPulled (below) fires exactly once, for the pulled tenant.
event(new PullingPendingTenant($pullCandidate));
$tenant = DB::transaction(function () use ($pullCandidate, $attributes): ?Tenant {
$tenantWasPulled = static::onlyPending()
->whereKey($pullCandidate->getKey())
->update([$pullCandidate->getColumnForQuery('pending_since') => null]) > 0;
if (! $tenantWasPulled) {
return null;
}
// The tenant's pending_since was just cleared, and e.g. a PullingPendingTenant listener
// may have made changes to the tenant, so re-fetch it to get it in the correct state.
/** @var Model&Tenant $pulledTenant */
$pulledTenant = static::findOrFail($pullCandidate->getKey());
if (! empty($attributes)) {
$pulledTenant->update($attributes);
}
return $pulledTenant;
});
if ($tenant === null) {
// If another pull claimed this tenant first, try claiming the next one
continue;
}
event(new PendingTenantPulled($tenant));
return $tenant; return $tenant;
});
if ($tenant === null) {
return $firstOrCreate ? static::create($attributes) : null;
} }
// Only triggered if a tenant that was pulled from the pool is returned
event(new PendingTenantPulled($tenant));
return $tenant;
} }
} }

View file

@ -5,6 +5,7 @@ declare(strict_types=1);
use Illuminate\Database\QueryException; use Illuminate\Database\QueryException;
use Illuminate\Database\Schema\Blueprint; use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Artisan; use Illuminate\Support\Facades\Artisan;
use Illuminate\Support\Facades\DB;
use Illuminate\Support\Facades\Event; use Illuminate\Support\Facades\Event;
use Illuminate\Support\Facades\Schema; use Illuminate\Support\Facades\Schema;
use Illuminate\Support\Str; use Illuminate\Support\Str;
@ -126,6 +127,52 @@ test('a new tenant gets created while pulling a pending tenant if the pending po
expect(Tenant::withPending()->get()->count())->toBe(1); // All tenants expect(Tenant::withPending()->get()->count())->toBe(1); // All tenants
}); });
test('pulling a pending tenant retries when the tenant is claimed concurrently', function () {
Tenant::createPending();
Tenant::createPending();
$stolenId = null;
Event::listen(PullingPendingTenant::class, function (PullingPendingTenant $event) use (&$stolenId) {
if ($stolenId !== null) {
return;
}
$stolenId = $event->tenant->id;
// Steal the tenant like a concurrent process would
Tenant::onlyPending()
->whereKey($event->tenant->id)
->update([$event->tenant->getColumnForQuery('pending_since') => null]);
});
$pulled = Tenant::pullPendingFromPool();
expect($pulled)->not()->toBeNull();
expect($pulled->id)->not()->toBe($stolenId); // Stolen tenant was skipped, the next one was claimed by the pull
expect(Tenant::onlyPending()->count())->toBe(0); // Both tenants claimed
});
test('the pull is rolled back and the tenant stays in the pool if setting attributes fails', function () {
// Pulling a tenant and setting its attributes happen in one transaction,
// so if setting the attributes fails, the whole pull rolls back and the tenant stays in the pool.
Schema::table('tenants', function (Blueprint $table) {
$table->string('slug')->nullable()->unique();
});
Tenant::$extraCustomColumns = ['slug'];
Tenant::create(['slug' => 'taken']);
Tenant::createPending();
// During the pull, set slug to 'taken', which is already used by another tenant to make the attribute update throw
expect(fn () => Tenant::pullPendingFromPool(false, ['slug' => 'taken']))
->toThrow(QueryException::class);
// The pull rolled back, so the tenant is still pending
expect(Tenant::onlyPending()->count())->toBe(1);
});
test('withoutPending chained with where clauses returns correct results', function () { test('withoutPending chained with where clauses returns correct results', function () {
$tenant = Tenant::create(); $tenant = Tenant::create();
$pendingTenant = Tenant::createPending(); $pendingTenant = Tenant::createPending();