|\Illuminate\Database\Query\Builder withPending(bool $withPending = true) * @method static static|\Illuminate\Database\Eloquent\Builder|\Illuminate\Database\Query\Builder onlyPending() * @method static static|\Illuminate\Database\Eloquent\Builder|\Illuminate\Database\Query\Builder withoutPending() */ trait HasPending { public static string $pendingSinceCast = 'timestamp'; /** Boot the trait. */ public static function bootHasPending(): void { static::addGlobalScope(new PendingScope()); } /** Initialize the trait. */ public function initializeHasPending(): void { $this->casts['pending_since'] = static::$pendingSinceCast; } /** Determine if the model instance is in a pending state. */ public function pending(): bool { return ! is_null($this->pending_since); } /** * Create a pending tenant. * * @param array $attributes */ public static function createPending(array $attributes = []): Model&Tenant { $tenant = static::create($attributes); event(new CreatingPendingTenant($tenant)); // Update the pending_since value only after the tenant is created so it's // Not marked as pending until finishing running the migrations, seeders, etc. $tenant->update([ 'pending_since' => now()->timestamp, ]); event(new PendingTenantCreated($tenant)); return $tenant; } /** Pull a pending tenant. */ public static function pullPending(): Model&Tenant { /** @var Model&Tenant $pendingTenant */ $pendingTenant = static::pullPendingFromPool(true); return $pendingTenant; } /** Try to pull a tenant from the pool of pending tenants. */ public static function pullPendingFromPool(bool $firstOrCreate = true, array $attributes = []): ?Tenant { /** @var (Model&Tenant)|null $tenant */ $tenant = static::onlyPending()->first(); if ($tenant === null) { return $firstOrCreate ? static::create($attributes) : null; } event(new PullingPendingTenant($tenant)); $tenant->update(array_merge($attributes, [ 'pending_since' => null, ])); event(new PendingTenantPulled($tenant)); return $tenant; } }