164 lines
5.1 KiB
PHP
164 lines
5.1 KiB
PHP
<?php
|
|
|
|
/**
|
|
* ============================================================
|
|
*
|
|
* @project biiproject
|
|
*
|
|
* @author Andika Debi Putra
|
|
*
|
|
* @email andikadebiputra@gmail.com
|
|
*
|
|
* @website https://biiproject.com
|
|
*
|
|
* @copyright Copyright (c) 2026 Andika Debi Putra
|
|
* @license Proprietary - All Rights Reserved
|
|
*
|
|
* @version 1.0.0
|
|
*
|
|
* @created 2026-05-01
|
|
* ============================================================
|
|
*
|
|
* Unauthorized copying, modification, distribution, or use
|
|
* of this file is strictly prohibited without prior written
|
|
* permission from the author.
|
|
* ============================================================
|
|
*/
|
|
|
|
namespace App\Console\Commands;
|
|
|
|
use App\Models\User;
|
|
use App\Notifications\BackupFailedNotification;
|
|
use App\Services\Notification\TelegramService;
|
|
use Illuminate\Console\Command;
|
|
use Illuminate\Support\Facades\Log;
|
|
use Illuminate\Support\Facades\Storage;
|
|
use Illuminate\Support\Str;
|
|
|
|
class VerifyBackups extends Command
|
|
{
|
|
/**
|
|
* The name and signature of the console command.
|
|
*
|
|
* @var string
|
|
*/
|
|
protected $signature = 'backups:verify
|
|
{--disk=local : Storage disk to check}
|
|
{--max-age=7 : Alert if newest backup is older than N days}
|
|
{--min-size=1024 : Alert if backup is smaller than N bytes}';
|
|
|
|
/**
|
|
* The description of the console command.
|
|
*
|
|
* @var string
|
|
*/
|
|
protected $description = 'Verify backups exist, are recent, and have non-trivial file sizes';
|
|
|
|
/**
|
|
* Execute the console command.
|
|
*/
|
|
public function handle(TelegramService $telegram): int
|
|
{
|
|
$disk = $this->option('disk');
|
|
$maxAgeDays = (int) $this->option('max-age');
|
|
$minBytes = (int) $this->option('min-size');
|
|
|
|
$this->info("Verifying backups on disk: {$disk}");
|
|
|
|
$storage = Storage::disk($disk);
|
|
|
|
if (! $storage->exists('Laravel')) {
|
|
$reason = 'Backup directory "Laravel" not found on disk.';
|
|
$this->error($reason);
|
|
$this->notifyFailure($reason, $disk, $telegram);
|
|
|
|
return self::FAILURE;
|
|
}
|
|
|
|
$files = collect($storage->allFiles('Laravel'))
|
|
->filter(fn ($f) => Str::endsWith($f, ['.zip', '.gz', '.sql']))
|
|
->sortByDesc(fn ($f) => $storage->lastModified($f))
|
|
->values();
|
|
|
|
if ($files->isEmpty()) {
|
|
$reason = 'No backup files found.';
|
|
$this->error($reason);
|
|
$this->notifyFailure($reason, $disk, $telegram);
|
|
|
|
return self::FAILURE;
|
|
}
|
|
|
|
$latest = $files->first();
|
|
$latestModified = $storage->lastModified($latest);
|
|
$latestSize = $storage->size($latest);
|
|
$ageDays = (int) round((time() - $latestModified) / 86400);
|
|
|
|
$this->table(['File', 'Age (days)', 'Size (KB)', 'Status'], [[
|
|
basename($latest),
|
|
$ageDays,
|
|
round($latestSize / 1024, 1),
|
|
$this->statusLabel($ageDays, $latestSize, $maxAgeDays, $minBytes),
|
|
]]);
|
|
|
|
$problems = [];
|
|
|
|
if ($ageDays > $maxAgeDays) {
|
|
$problems[] = "Latest backup is {$ageDays} days old (threshold: {$maxAgeDays} days)";
|
|
}
|
|
|
|
if ($latestSize < $minBytes) {
|
|
$problems[] = "Latest backup is suspiciously small ({$latestSize} bytes)";
|
|
}
|
|
|
|
if (! empty($problems)) {
|
|
foreach ($problems as $problem) {
|
|
$this->warn($problem);
|
|
}
|
|
$this->notifyFailure(implode('; ', $problems), $disk, $telegram);
|
|
|
|
return self::FAILURE;
|
|
}
|
|
|
|
$this->info("✓ Backup verification passed. {$files->count()} backup(s) found.");
|
|
Log::channel('single')->info('[VerifyBackups] OK', ['count' => $files->count(), 'age_days' => $ageDays]);
|
|
|
|
return self::SUCCESS;
|
|
}
|
|
|
|
private function statusLabel(int $age, int $size, int $maxAge, int $minBytes): string
|
|
{
|
|
if ($age > $maxAge || $size < $minBytes) {
|
|
return '⚠ WARN';
|
|
}
|
|
|
|
return '✓ OK';
|
|
}
|
|
|
|
private function notifyFailure(string $reason, string $disk, TelegramService $telegram): void
|
|
{
|
|
Log::channel('single')->error('[VerifyBackups] FAILED', compact('reason', 'disk'));
|
|
|
|
// Prepare Telegram Alert
|
|
$hostname = gethostname();
|
|
$message = "🚨 <b>BACKUP VERIFICATION FAILED</b> 🚨\n";
|
|
$message .= "--------------------------------\n";
|
|
$message .= "<b>Host:</b> {$hostname}\n";
|
|
$message .= "<b>Disk:</b> {$disk}\n";
|
|
$message .= "<b>Reason:</b> {$reason}\n";
|
|
$message .= "--------------------------------\n";
|
|
$message .= '<i>Please check the backup logs immediately!</i>';
|
|
|
|
$telegram->sendMessage($message);
|
|
|
|
// Notify developer role via database notification
|
|
try {
|
|
$developers = User::role('Developer')->get();
|
|
foreach ($developers as $dev) {
|
|
$dev->notify(new BackupFailedNotification($reason));
|
|
}
|
|
} catch (\Throwable) {
|
|
// Silently skip if notification classes don't exist yet
|
|
}
|
|
}
|
|
}
|