Files
jabali-panel/app/Filament/Admin/Pages/ServerUpdates.php
2026-02-01 01:01:23 +02:00

334 lines
11 KiB
PHP

<?php
declare(strict_types=1);
namespace App\Filament\Admin\Pages;
use App\Services\Agent\AgentClient;
use BackedEnum;
use Filament\Actions\Action;
use Filament\Actions\Concerns\InteractsWithActions;
use Filament\Actions\Contracts\HasActions;
use Filament\Notifications\Notification;
use Filament\Pages\Page;
use Filament\Support\Icons\Heroicon;
use Filament\Tables\Columns\TextColumn;
use Filament\Tables\Concerns\InteractsWithTable;
use Filament\Tables\Contracts\HasTable;
use Filament\Tables\Table;
use Illuminate\Contracts\Support\Htmlable;
use Illuminate\Support\Facades\Artisan;
use Illuminate\Support\Facades\File;
use Illuminate\Support\Facades\View;
class ServerUpdates extends Page implements HasActions, HasTable
{
use InteractsWithActions;
use InteractsWithTable;
protected static string|BackedEnum|null $navigationIcon = Heroicon::OutlinedArrowPathRoundedSquare;
protected static ?int $navigationSort = 16;
protected static ?string $slug = 'server-updates';
protected string $view = 'filament.admin.pages.server-updates';
public array $packages = [];
public ?string $currentVersion = null;
public ?int $behindCount = null;
public array $recentChanges = [];
public ?string $lastCheckedAt = null;
public ?string $jabaliOutput = null;
public ?string $jabaliOutputTitle = null;
public ?string $jabaliOutputAt = null;
public bool $isChecking = false;
public bool $isUpgrading = false;
public array $refreshOutput = [];
public ?string $refreshOutputAt = null;
public ?string $refreshOutputTitle = null;
protected ?AgentClient $agent = null;
protected bool $updatesLoaded = false;
public function getTitle(): string|Htmlable
{
return __('System Updates');
}
public static function getNavigationLabel(): string
{
return __('System Updates');
}
public function mount(): void
{
$this->loadUpdates(false);
$this->loadVersionInfo();
}
public function getUpdateStatusLabelProperty(): string
{
if ($this->behindCount === null) {
return __('Not checked');
}
if ($this->behindCount === 0) {
return __('Up to date');
}
return __(':count commit(s) behind', ['count' => $this->behindCount]);
}
protected function getAgent(): AgentClient
{
return $this->agent ??= new AgentClient;
}
public function loadUpdates(bool $refreshTable = true, bool $refreshApt = false): void
{
try {
$result = $this->getAgent()->updatesList($refreshApt);
$this->packages = $result['packages'] ?? [];
$this->updatesLoaded = true;
if ($refreshApt) {
$refreshOutput = $result['refresh_output'] ?? [];
$refreshLines = is_array($refreshOutput) ? $refreshOutput : [$refreshOutput];
$this->refreshOutput = ! empty(array_filter($refreshLines, static fn ($line) => $line !== null && $line !== ''))
? $refreshLines
: [__('No output captured.')];
$this->refreshOutputAt = now()->format('Y-m-d H:i:s');
$this->refreshOutputTitle = __('Update Refresh Output');
}
$warnings = $result['warnings'] ?? [];
if (! empty($warnings)) {
Notification::make()
->title(__('Update check completed with warnings'))
->body(implode("\n", array_filter($warnings)))
->warning()
->send();
}
} catch (\Exception $e) {
$this->packages = [];
$this->updatesLoaded = true;
if ($refreshApt) {
$this->refreshOutput = [$e->getMessage()];
$this->refreshOutputAt = now()->format('Y-m-d H:i:s');
$this->refreshOutputTitle = __('Update Refresh Output');
}
Notification::make()
->title(__('Failed to load updates'))
->body($e->getMessage())
->danger()
->send();
}
if ($refreshTable) {
$this->resetTable();
}
}
public function loadVersionInfo(): void
{
$this->currentVersion = $this->readVersion();
}
public function checkForUpdates(): void
{
$this->isChecking = true;
try {
$exitCode = Artisan::call('jabali:upgrade', ['--check' => true]);
$output = trim(Artisan::output());
if ($exitCode !== 0) {
throw new \RuntimeException($output !== '' ? $output : __('Update check failed.'));
}
$this->jabaliOutput = $output !== '' ? $output : __('No output captured.');
$this->jabaliOutputTitle = __('Update Check Output');
$this->jabaliOutputAt = now()->format('Y-m-d H:i:s');
$this->lastCheckedAt = $this->jabaliOutputAt;
$this->parseUpdateOutput($output);
Notification::make()
->title(__('Update check completed'))
->success()
->send();
} catch (\Throwable $e) {
$this->jabaliOutput = $e->getMessage();
$this->jabaliOutputTitle = __('Update Check Output');
$this->jabaliOutputAt = now()->format('Y-m-d H:i:s');
Notification::make()
->title(__('Update check failed'))
->body($e->getMessage())
->danger()
->send();
} finally {
$this->isChecking = false;
}
}
public function performUpgrade(): void
{
$this->isUpgrading = true;
try {
$exitCode = Artisan::call('jabali:upgrade', ['--force' => true]);
$output = trim(Artisan::output());
if ($exitCode !== 0) {
throw new \RuntimeException($output !== '' ? $output : __('Upgrade failed.'));
}
$this->jabaliOutput = $output !== '' ? $output : __('No output captured.');
$this->jabaliOutputTitle = __('Upgrade Output');
$this->jabaliOutputAt = now()->format('Y-m-d H:i:s');
$this->loadVersionInfo();
$this->behindCount = 0;
$this->recentChanges = [];
Notification::make()
->title(__('Upgrade completed'))
->success()
->send();
} catch (\Throwable $e) {
$this->jabaliOutput = $e->getMessage();
$this->jabaliOutputTitle = __('Upgrade Output');
$this->jabaliOutputAt = now()->format('Y-m-d H:i:s');
Notification::make()
->title(__('Upgrade failed'))
->body($e->getMessage())
->danger()
->send();
} finally {
$this->isUpgrading = false;
}
}
public function runUpdates(): void
{
try {
$result = $this->getAgent()->updatesRun();
$output = $result['output'] ?? [];
$outputLines = is_array($output) ? $output : [$output];
$this->refreshOutput = ! empty(array_filter($outputLines, static fn ($line) => $line !== null && $line !== ''))
? $outputLines
: [__('No output captured.')];
$this->refreshOutputAt = now()->format('Y-m-d H:i:s');
$this->refreshOutputTitle = __('System Update Output');
Notification::make()
->title(__('Updates completed'))
->success()
->send();
} catch (\Exception $e) {
$this->refreshOutput = [$e->getMessage()];
$this->refreshOutputAt = now()->format('Y-m-d H:i:s');
$this->refreshOutputTitle = __('System Update Output');
Notification::make()
->title(__('Update failed'))
->body($e->getMessage())
->danger()
->send();
}
$this->loadUpdates();
$this->dispatch('$refresh');
}
protected function parseUpdateOutput(string $output): void
{
$this->behindCount = null;
$this->recentChanges = [];
if (preg_match('/Updates available:\s+(\d+)/', $output, $matches)) {
$this->behindCount = (int) $matches[1];
} elseif (str_contains(strtolower($output), 'up to date')) {
$this->behindCount = 0;
}
if (preg_match('/Recent changes:\s*(.+)$/s', $output, $matches)) {
$lines = preg_split('/\r?\n/', trim($matches[1]));
$this->recentChanges = array_values(array_filter($lines, static fn (string $line): bool => trim($line) !== ''));
}
}
protected function readVersion(): string
{
$versionFile = base_path('VERSION');
if (! File::exists($versionFile)) {
return 'unknown';
}
$content = File::get($versionFile);
if (preg_match('/VERSION=(.+)/', $content, $matches)) {
return trim($matches[1]);
}
return 'unknown';
}
public function table(Table $table): Table
{
return $table
->records(function () {
if (! $this->updatesLoaded) {
$this->loadUpdates(false);
}
return collect($this->packages)
->mapWithKeys(function (array $package, int $index): array {
$keyParts = [
$package['name'] ?? (string) $index,
$package['current_version'] ?? '',
$package['new_version'] ?? '',
];
$key = implode('|', array_filter($keyParts, fn (string $part): bool => $part !== ''));
return [$key !== '' ? $key : (string) $index => $package];
})
->all();
})
->columns([
TextColumn::make('name')
->label(__('Package'))
->searchable(),
TextColumn::make('current_version')
->label(__('Current Version')),
TextColumn::make('new_version')
->label(__('New Version')),
])
->emptyStateHeading(__('No updates available'))
->emptyStateDescription(__('Your system packages are up to date.'))
->headerActions([
Action::make('refresh')
->label(__('Refresh'))
->icon('heroicon-o-arrow-path')
->action(fn () => $this->loadUpdates(true, true)),
Action::make('runUpdates')
->label(__('Run Updates'))
->icon('heroicon-o-arrow-path-rounded-square')
->color('primary')
->requiresConfirmation()
->modalHeading(__('Install updates'))
->modalDescription(__('This will run apt-get upgrade on the server. Continue?'))
->action(fn () => $this->runUpdates()),
]);
}
}