mirror of
https://github.com/coollabsio/coolify.git
synced 2025-12-28 05:34:50 +00:00
Fixes the "Snapshot missing on Livewire component" error that occurs when toggling the "Backup includes all databases" checkbox during MariaDB database import operations. Root Cause: - ActivityMonitor component was initialized without proper lifecycle hooks - When parent Import component re-rendered (via checkbox toggle), the ActivityMonitor's Livewire snapshot became stale - Missing null checks caused errors when querying with undefined activityId - No state cleanup when slide-over closed, causing issues on subsequent opens Changes: - Add updatedActivityId() lifecycle hook to ActivityMonitor for proper hydration - Add defensive null check in hydrateActivity() to prevent query errors - Track activityId in Import component for state management - Add slideOverClosed event dispatch in slide-over component - Add event listener in Import component to reset activityId on close Testing: - Manually verify checkbox toggle doesn't trigger popup - Verify actual restore operations work correctly - Test both file-based and S3-based restore methods - Ensure X button properly closes the modal - Verify no console errors or Livewire warnings 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com>
108 lines
3.1 KiB
PHP
108 lines
3.1 KiB
PHP
<?php
|
|
|
|
namespace App\Livewire;
|
|
|
|
use App\Models\User;
|
|
use Livewire\Component;
|
|
use Spatie\Activitylog\Models\Activity;
|
|
|
|
class ActivityMonitor extends Component
|
|
{
|
|
public ?string $header = null;
|
|
|
|
public $activityId = null;
|
|
|
|
public $eventToDispatch = 'activityFinished';
|
|
|
|
public $eventData = null;
|
|
|
|
public $isPollingActive = false;
|
|
|
|
public bool $fullHeight = false;
|
|
|
|
public $activity;
|
|
|
|
public bool $showWaiting = true;
|
|
|
|
public static $eventDispatched = false;
|
|
|
|
protected $listeners = ['activityMonitor' => 'newMonitorActivity'];
|
|
|
|
public function newMonitorActivity($activityId, $eventToDispatch = 'activityFinished', $eventData = null, $header = null)
|
|
{
|
|
// Reset event dispatched flag for new activity
|
|
self::$eventDispatched = false;
|
|
|
|
$this->activityId = $activityId;
|
|
$this->eventToDispatch = $eventToDispatch;
|
|
$this->eventData = $eventData;
|
|
|
|
// Update header if provided
|
|
if ($header !== null) {
|
|
$this->header = $header;
|
|
}
|
|
|
|
$this->hydrateActivity();
|
|
|
|
$this->isPollingActive = true;
|
|
}
|
|
|
|
public function hydrateActivity()
|
|
{
|
|
if ($this->activityId === null) {
|
|
$this->activity = null;
|
|
|
|
return;
|
|
}
|
|
|
|
$this->activity = Activity::find($this->activityId);
|
|
}
|
|
|
|
public function updatedActivityId($value)
|
|
{
|
|
if ($value) {
|
|
$this->hydrateActivity();
|
|
$this->isPollingActive = true;
|
|
self::$eventDispatched = false;
|
|
}
|
|
}
|
|
|
|
public function polling()
|
|
{
|
|
$this->hydrateActivity();
|
|
$exit_code = data_get($this->activity, 'properties.exitCode');
|
|
if ($exit_code !== null) {
|
|
$this->isPollingActive = false;
|
|
if ($exit_code === 0) {
|
|
if ($this->eventToDispatch !== null) {
|
|
if (str($this->eventToDispatch)->startsWith('App\\Events\\')) {
|
|
$causer_id = data_get($this->activity, 'causer_id');
|
|
$user = User::find($causer_id);
|
|
if ($user) {
|
|
$teamId = $user->currentTeam()->id;
|
|
if (! self::$eventDispatched) {
|
|
if (filled($this->eventData)) {
|
|
$this->eventToDispatch::dispatch($teamId, $this->eventData);
|
|
} else {
|
|
$this->eventToDispatch::dispatch($teamId);
|
|
}
|
|
self::$eventDispatched = true;
|
|
}
|
|
}
|
|
|
|
return;
|
|
}
|
|
if (! self::$eventDispatched) {
|
|
if (filled($this->eventData)) {
|
|
$this->dispatch($this->eventToDispatch, $this->eventData);
|
|
} else {
|
|
$this->dispatch($this->eventToDispatch);
|
|
}
|
|
self::$eventDispatched = true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|