 36961d8ae8
			
		
	
	36961d8ae8
	
	
	
		
			
			- docker cleanup was always running on deletion instead of using the settings set in the deletion modal
		
			
				
	
	
		
			167 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			167 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| 
 | |
| namespace App\Jobs;
 | |
| 
 | |
| use App\Actions\Application\StopApplication;
 | |
| use App\Actions\Database\StopDatabase;
 | |
| use App\Actions\Server\CleanupDocker;
 | |
| use App\Actions\Service\DeleteService;
 | |
| use App\Actions\Service\StopService;
 | |
| use App\Models\Application;
 | |
| use App\Models\ApplicationPreview;
 | |
| use App\Models\Service;
 | |
| use App\Models\StandaloneClickhouse;
 | |
| use App\Models\StandaloneDragonfly;
 | |
| use App\Models\StandaloneKeydb;
 | |
| use App\Models\StandaloneMariadb;
 | |
| use App\Models\StandaloneMongodb;
 | |
| use App\Models\StandaloneMysql;
 | |
| use App\Models\StandalonePostgresql;
 | |
| use App\Models\StandaloneRedis;
 | |
| use Illuminate\Bus\Queueable;
 | |
| use Illuminate\Contracts\Queue\ShouldBeEncrypted;
 | |
| use Illuminate\Contracts\Queue\ShouldQueue;
 | |
| use Illuminate\Foundation\Bus\Dispatchable;
 | |
| use Illuminate\Queue\InteractsWithQueue;
 | |
| use Illuminate\Queue\SerializesModels;
 | |
| use Illuminate\Support\Facades\Artisan;
 | |
| 
 | |
| class DeleteResourceJob implements ShouldBeEncrypted, ShouldQueue
 | |
| {
 | |
|     use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
 | |
| 
 | |
|     public function __construct(
 | |
|         public Application|ApplicationPreview|Service|StandalonePostgresql|StandaloneRedis|StandaloneMongodb|StandaloneMysql|StandaloneMariadb|StandaloneKeydb|StandaloneDragonfly|StandaloneClickhouse $resource,
 | |
|         public bool $deleteVolumes = true,
 | |
|         public bool $deleteConnectedNetworks = true,
 | |
|         public bool $deleteConfigurations = true,
 | |
|         public bool $dockerCleanup = true
 | |
|     ) {
 | |
|         $this->onQueue('high');
 | |
|     }
 | |
| 
 | |
|     public function handle()
 | |
|     {
 | |
|         try {
 | |
|             // Handle ApplicationPreview instances separately
 | |
|             if ($this->resource instanceof ApplicationPreview) {
 | |
|                 $this->deleteApplicationPreview();
 | |
| 
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             switch ($this->resource->type()) {
 | |
|                 case 'application':
 | |
|                     StopApplication::run($this->resource, previewDeployments: true, dockerCleanup: $this->dockerCleanup);
 | |
|                     break;
 | |
|                 case 'standalone-postgresql':
 | |
|                 case 'standalone-redis':
 | |
|                 case 'standalone-mongodb':
 | |
|                 case 'standalone-mysql':
 | |
|                 case 'standalone-mariadb':
 | |
|                 case 'standalone-keydb':
 | |
|                 case 'standalone-dragonfly':
 | |
|                 case 'standalone-clickhouse':
 | |
|                     StopDatabase::run($this->resource, dockerCleanup: $this->dockerCleanup);
 | |
|                     break;
 | |
|                 case 'service':
 | |
|                     StopService::run($this->resource, $this->deleteConnectedNetworks, $this->dockerCleanup);
 | |
|                     DeleteService::run($this->resource, $this->deleteVolumes, $this->deleteConnectedNetworks, $this->deleteConfigurations, $this->dockerCleanup);
 | |
| 
 | |
|                     return;
 | |
|             }
 | |
| 
 | |
|             if ($this->deleteConfigurations) {
 | |
|                 $this->resource->deleteConfigurations();
 | |
|             }
 | |
|             if ($this->deleteVolumes) {
 | |
|                 $this->resource->deleteVolumes();
 | |
|                 $this->resource->persistentStorages()->delete();
 | |
|             }
 | |
|             $this->resource->fileStorages()->delete(); // these are file mounts which should probably have their own flag
 | |
| 
 | |
|             $isDatabase = $this->resource instanceof StandalonePostgresql
 | |
|             || $this->resource instanceof StandaloneRedis
 | |
|             || $this->resource instanceof StandaloneMongodb
 | |
|             || $this->resource instanceof StandaloneMysql
 | |
|             || $this->resource instanceof StandaloneMariadb
 | |
|             || $this->resource instanceof StandaloneKeydb
 | |
|             || $this->resource instanceof StandaloneDragonfly
 | |
|             || $this->resource instanceof StandaloneClickhouse;
 | |
| 
 | |
|             if ($isDatabase) {
 | |
|                 $this->resource->sslCertificates()->delete();
 | |
|                 $this->resource->scheduledBackups()->delete();
 | |
|                 $this->resource->tags()->detach();
 | |
|             }
 | |
|             $this->resource->environment_variables()->delete();
 | |
| 
 | |
|             if ($this->deleteConnectedNetworks && $this->resource->type() === 'application') {
 | |
|                 $this->resource->deleteConnectedNetworks();
 | |
|             }
 | |
|         } catch (\Throwable $e) {
 | |
|             throw $e;
 | |
|         } finally {
 | |
|             $this->resource->forceDelete();
 | |
|             if ($this->dockerCleanup) {
 | |
|                 $server = data_get($this->resource, 'server') ?? data_get($this->resource, 'destination.server');
 | |
|                 if ($server) {
 | |
|                     CleanupDocker::dispatch($server, false, false);
 | |
|                 }
 | |
|             }
 | |
|             Artisan::queue('cleanup:stucked-resources');
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     private function deleteApplicationPreview()
 | |
|     {
 | |
|         $application = $this->resource->application;
 | |
|         $server = $application->destination->server;
 | |
|         $pull_request_id = $this->resource->pull_request_id;
 | |
| 
 | |
|         // Ensure the preview is soft deleted (may already be done in Livewire component)
 | |
|         if (! $this->resource->trashed()) {
 | |
|             $this->resource->delete();
 | |
|         }
 | |
| 
 | |
|         try {
 | |
|             if ($server->isSwarm()) {
 | |
|                 instant_remote_process(["docker stack rm {$application->uuid}-{$pull_request_id}"], $server);
 | |
|             } else {
 | |
|                 $containers = getCurrentApplicationContainerStatus($server, $application->id, $pull_request_id)->toArray();
 | |
|                 $this->stopPreviewContainers($containers, $server);
 | |
|             }
 | |
|         } catch (\Throwable $e) {
 | |
|             // Log the error but don't fail the job
 | |
|             ray('Error stopping preview containers: '.$e->getMessage());
 | |
|         }
 | |
| 
 | |
|         // Finally, force delete to trigger resource cleanup
 | |
|         $this->resource->forceDelete();
 | |
|     }
 | |
| 
 | |
|     private function stopPreviewContainers(array $containers, $server, int $timeout = 30)
 | |
|     {
 | |
|         if (empty($containers)) {
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         $containerNames = [];
 | |
|         foreach ($containers as $container) {
 | |
|             $containerNames[] = str_replace('/', '', $container['Names']);
 | |
|         }
 | |
| 
 | |
|         $containerList = implode(' ', array_map('escapeshellarg', $containerNames));
 | |
|         $commands = [
 | |
|             "docker stop --time=$timeout $containerList",
 | |
|             "docker rm -f $containerList",
 | |
|         ];
 | |
| 
 | |
|         instant_remote_process(
 | |
|             command: $commands,
 | |
|             server: $server,
 | |
|             throwError: false
 | |
|         );
 | |
|     }
 | |
| }
 |