- Add retry configuration to CoolifyTask (3 tries, 600s timeout) - Add retry configuration to ScheduledTaskJob (3 tries, configurable timeout) - Add retry configuration to DatabaseBackupJob (2 tries) - Implement exponential backoff for all jobs (30s, 60s, 120s intervals) - Add failed() handlers with comprehensive error logging to scheduled-errors channel - Add execution tracking: started_at, retry_count, duration (decimal), error_details - Add configurable timeout field to scheduled tasks (60-3600s, default 300s) - Update UI to include timeout configuration in task creation/editing forms - Increase ScheduledJobManager lock expiration from 60s to 90s for high-load environments - Implement safe queue cleanup with restart vs runtime modes - Restart mode: aggressive cleanup (marks all processing jobs as failed) - Runtime mode: conservative cleanup (only marks jobs >12h as failed, skips deployments) - Add cleanup:redis --restart flag for system startup - Integrate cleanup into Dev.php init() for development environment - Increase scheduled-errors log retention from 7 to 14 days - Create comprehensive test suite (unit and feature tests) - Add TESTING_GUIDE.md with manual testing instructions Fixes issues with jobs failing after single attempt and "attempted too many times" errors
60 lines
1.6 KiB
PHP
60 lines
1.6 KiB
PHP
<?php
|
|
|
|
namespace App\Console\Commands;
|
|
|
|
use App\Jobs\CheckHelperImageJob;
|
|
use App\Models\InstanceSettings;
|
|
use Illuminate\Console\Command;
|
|
use Illuminate\Support\Facades\Artisan;
|
|
|
|
class Dev extends Command
|
|
{
|
|
protected $signature = 'dev {--init}';
|
|
|
|
protected $description = 'Helper commands for development.';
|
|
|
|
public function handle()
|
|
{
|
|
if ($this->option('init')) {
|
|
$this->init();
|
|
|
|
return;
|
|
}
|
|
}
|
|
|
|
public function init()
|
|
{
|
|
// Generate APP_KEY if not exists
|
|
|
|
if (empty(config('app.key'))) {
|
|
echo "Generating APP_KEY.\n";
|
|
Artisan::call('key:generate');
|
|
}
|
|
|
|
// Generate STORAGE link if not exists
|
|
if (! file_exists(public_path('storage'))) {
|
|
echo "Generating STORAGE link.\n";
|
|
Artisan::call('storage:link');
|
|
}
|
|
|
|
// Seed database if it's empty
|
|
$settings = InstanceSettings::find(0);
|
|
if (! $settings) {
|
|
echo "Initializing instance, seeding database.\n";
|
|
Artisan::call('migrate --seed');
|
|
} else {
|
|
echo "Instance already initialized.\n";
|
|
}
|
|
|
|
// Clean up stuck jobs and stale locks on development startup
|
|
try {
|
|
echo "Cleaning up Redis (stuck jobs and stale locks)...\n";
|
|
Artisan::call('cleanup:redis', ['--restart' => true, '--clear-locks' => true]);
|
|
echo "Redis cleanup completed.\n";
|
|
} catch (\Throwable $e) {
|
|
echo "Error in cleanup:redis: {$e->getMessage()}\n";
|
|
}
|
|
|
|
CheckHelperImageJob::dispatch();
|
|
}
|
|
}
|