356 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			356 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
<?php
 | 
						|
 | 
						|
use App\Actions\Application\StopApplication;
 | 
						|
use App\Enums\ApplicationDeploymentStatus;
 | 
						|
use App\Jobs\ApplicationDeploymentJob;
 | 
						|
use App\Jobs\VolumeCloneJob;
 | 
						|
use App\Models\Application;
 | 
						|
use App\Models\ApplicationDeploymentQueue;
 | 
						|
use App\Models\Server;
 | 
						|
use App\Models\StandaloneDocker;
 | 
						|
use Spatie\Url\Url;
 | 
						|
use Visus\Cuid2\Cuid2;
 | 
						|
 | 
						|
function queue_application_deployment(Application $application, string $deployment_uuid, ?int $pull_request_id = 0, string $commit = 'HEAD', bool $force_rebuild = false, bool $is_webhook = false, bool $is_api = false, bool $restart_only = false, ?string $git_type = null, bool $no_questions_asked = false, ?Server $server = null, ?StandaloneDocker $destination = null, bool $only_this_server = false, bool $rollback = false)
 | 
						|
{
 | 
						|
    $application_id = $application->id;
 | 
						|
    $deployment_link = Url::fromString($application->link()."/deployment/{$deployment_uuid}");
 | 
						|
    $deployment_url = $deployment_link->getPath();
 | 
						|
    $server_id = $application->destination->server->id;
 | 
						|
    $server_name = $application->destination->server->name;
 | 
						|
    $destination_id = $application->destination->id;
 | 
						|
 | 
						|
    if ($server) {
 | 
						|
        $server_id = $server->id;
 | 
						|
        $server_name = $server->name;
 | 
						|
    }
 | 
						|
    if ($destination) {
 | 
						|
        $destination_id = $destination->id;
 | 
						|
    }
 | 
						|
 | 
						|
    // Check if there's already a deployment in progress or queued for this application and commit
 | 
						|
    $existing_deployment = ApplicationDeploymentQueue::where('application_id', $application_id)
 | 
						|
        ->where('commit', $commit)
 | 
						|
        ->where('pull_request_id', $pull_request_id)
 | 
						|
        ->whereIn('status', [ApplicationDeploymentStatus::IN_PROGRESS->value, ApplicationDeploymentStatus::QUEUED->value])
 | 
						|
        ->first();
 | 
						|
 | 
						|
    if ($existing_deployment) {
 | 
						|
        // If force_rebuild is true or rollback is true or no_questions_asked is true, we'll still create a new deployment
 | 
						|
        if (! $force_rebuild && ! $rollback && ! $no_questions_asked) {
 | 
						|
            // Return the existing deployment's details
 | 
						|
            return [
 | 
						|
                'status' => 'skipped',
 | 
						|
                'message' => 'Deployment already queued for this commit.',
 | 
						|
                'deployment_uuid' => $existing_deployment->deployment_uuid,
 | 
						|
                'existing_deployment' => $existing_deployment,
 | 
						|
            ];
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    $deployment = ApplicationDeploymentQueue::create([
 | 
						|
        'application_id' => $application_id,
 | 
						|
        'application_name' => $application->name,
 | 
						|
        'server_id' => $server_id,
 | 
						|
        'server_name' => $server_name,
 | 
						|
        'destination_id' => $destination_id,
 | 
						|
        'deployment_uuid' => $deployment_uuid,
 | 
						|
        'deployment_url' => $deployment_url,
 | 
						|
        'pull_request_id' => $pull_request_id,
 | 
						|
        'force_rebuild' => $force_rebuild,
 | 
						|
        'is_webhook' => $is_webhook,
 | 
						|
        'is_api' => $is_api,
 | 
						|
        'restart_only' => $restart_only,
 | 
						|
        'commit' => $commit,
 | 
						|
        'rollback' => $rollback,
 | 
						|
        'git_type' => $git_type,
 | 
						|
        'only_this_server' => $only_this_server,
 | 
						|
    ]);
 | 
						|
 | 
						|
    if ($no_questions_asked) {
 | 
						|
        ApplicationDeploymentJob::dispatch(
 | 
						|
            application_deployment_queue_id: $deployment->id,
 | 
						|
        );
 | 
						|
    } elseif (next_queuable($server_id, $application_id, $commit, $pull_request_id)) {
 | 
						|
        ApplicationDeploymentJob::dispatch(
 | 
						|
            application_deployment_queue_id: $deployment->id,
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    return [
 | 
						|
        'status' => 'queued',
 | 
						|
        'message' => 'Deployment queued.',
 | 
						|
        'deployment_uuid' => $deployment_uuid,
 | 
						|
    ];
 | 
						|
}
 | 
						|
function force_start_deployment(ApplicationDeploymentQueue $deployment)
 | 
						|
{
 | 
						|
    $deployment->update([
 | 
						|
        'status' => ApplicationDeploymentStatus::IN_PROGRESS->value,
 | 
						|
    ]);
 | 
						|
 | 
						|
    ApplicationDeploymentJob::dispatch(
 | 
						|
        application_deployment_queue_id: $deployment->id,
 | 
						|
    );
 | 
						|
}
 | 
						|
function queue_next_deployment(Application $application)
 | 
						|
{
 | 
						|
    $server_id = $application->destination->server_id;
 | 
						|
    $queued_deployments = ApplicationDeploymentQueue::where('server_id', $server_id)
 | 
						|
        ->where('status', ApplicationDeploymentStatus::QUEUED)
 | 
						|
        ->get()
 | 
						|
        ->sortBy('created_at');
 | 
						|
 | 
						|
    foreach ($queued_deployments as $next_deployment) {
 | 
						|
        // Check if this queued deployment can actually run
 | 
						|
        if (next_queuable($next_deployment->server_id, $next_deployment->application_id, $next_deployment->commit, $next_deployment->pull_request_id)) {
 | 
						|
            $next_deployment->update([
 | 
						|
                'status' => ApplicationDeploymentStatus::IN_PROGRESS->value,
 | 
						|
            ]);
 | 
						|
 | 
						|
            ApplicationDeploymentJob::dispatch(
 | 
						|
                application_deployment_queue_id: $next_deployment->id,
 | 
						|
            );
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
function next_queuable(string $server_id, string $application_id, string $commit = 'HEAD', int $pull_request_id = 0): bool
 | 
						|
{
 | 
						|
    // Check if there's already a deployment in progress for this application with the same pull_request_id
 | 
						|
    // This allows normal deployments and PR deployments to run concurrently
 | 
						|
    $in_progress = ApplicationDeploymentQueue::where('application_id', $application_id)
 | 
						|
        ->where('pull_request_id', $pull_request_id)
 | 
						|
        ->where('status', ApplicationDeploymentStatus::IN_PROGRESS->value)
 | 
						|
        ->exists();
 | 
						|
 | 
						|
    if ($in_progress) {
 | 
						|
        return false;
 | 
						|
    }
 | 
						|
 | 
						|
    // Check server's concurrent build limit
 | 
						|
    $server = Server::find($server_id);
 | 
						|
    $concurrent_builds = $server->settings->concurrent_builds;
 | 
						|
    $active_deployments = ApplicationDeploymentQueue::where('server_id', $server_id)
 | 
						|
        ->where('status', ApplicationDeploymentStatus::IN_PROGRESS->value)
 | 
						|
        ->count();
 | 
						|
 | 
						|
    if ($active_deployments >= $concurrent_builds) {
 | 
						|
        return false;
 | 
						|
    }
 | 
						|
 | 
						|
    return true;
 | 
						|
}
 | 
						|
function next_after_cancel(?Server $server = null)
 | 
						|
{
 | 
						|
    if ($server) {
 | 
						|
        $next_found = ApplicationDeploymentQueue::where('server_id', data_get($server, 'id'))
 | 
						|
            ->where('status', ApplicationDeploymentStatus::QUEUED)
 | 
						|
            ->get()
 | 
						|
            ->sortBy('created_at');
 | 
						|
 | 
						|
        if ($next_found->count() > 0) {
 | 
						|
            foreach ($next_found as $next) {
 | 
						|
                // Use next_queuable to properly check if this deployment can run
 | 
						|
                if (next_queuable($next->server_id, $next->application_id, $next->commit, $next->pull_request_id)) {
 | 
						|
                    $next->update([
 | 
						|
                        'status' => ApplicationDeploymentStatus::IN_PROGRESS->value,
 | 
						|
                    ]);
 | 
						|
 | 
						|
                    ApplicationDeploymentJob::dispatch(
 | 
						|
                        application_deployment_queue_id: $next->id,
 | 
						|
                    );
 | 
						|
                }
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
function clone_application(Application $source, $destination, array $overrides = [], bool $cloneVolumeData = false): Application
 | 
						|
{
 | 
						|
    $uuid = $overrides['uuid'] ?? (string) new Cuid2;
 | 
						|
    $server = $destination->server;
 | 
						|
 | 
						|
    // Prepare name and URL
 | 
						|
    $name = $overrides['name'] ?? 'clone-of-'.str($source->name)->limit(20).'-'.$uuid;
 | 
						|
    $applicationSettings = $source->settings;
 | 
						|
    $url = $overrides['fqdn'] ?? $source->fqdn;
 | 
						|
 | 
						|
    if ($server->proxyType() !== 'NONE' && $applicationSettings->is_container_label_readonly_enabled === true) {
 | 
						|
        $url = generateUrl(server: $server, random: $uuid);
 | 
						|
    }
 | 
						|
 | 
						|
    // Clone the application
 | 
						|
    $newApplication = $source->replicate([
 | 
						|
        'id',
 | 
						|
        'created_at',
 | 
						|
        'updated_at',
 | 
						|
        'additional_servers_count',
 | 
						|
        'additional_networks_count',
 | 
						|
    ])->fill(array_merge([
 | 
						|
        'uuid' => $uuid,
 | 
						|
        'name' => $name,
 | 
						|
        'fqdn' => $url,
 | 
						|
        'status' => 'exited',
 | 
						|
        'destination_id' => $destination->id,
 | 
						|
    ], $overrides));
 | 
						|
    $newApplication->save();
 | 
						|
 | 
						|
    // Update custom labels if needed
 | 
						|
    if ($newApplication->destination->server->proxyType() !== 'NONE' && $applicationSettings->is_container_label_readonly_enabled === true) {
 | 
						|
        $customLabels = str(implode('|coolify|', generateLabelsApplication($newApplication)))->replace('|coolify|', "\n");
 | 
						|
        $newApplication->custom_labels = base64_encode($customLabels);
 | 
						|
        $newApplication->save();
 | 
						|
    }
 | 
						|
 | 
						|
    // Clone settings
 | 
						|
    $newApplication->settings()->delete();
 | 
						|
    if ($applicationSettings) {
 | 
						|
        $newApplicationSettings = $applicationSettings->replicate([
 | 
						|
            'id',
 | 
						|
            'created_at',
 | 
						|
            'updated_at',
 | 
						|
        ])->fill([
 | 
						|
            'application_id' => $newApplication->id,
 | 
						|
        ]);
 | 
						|
        $newApplicationSettings->save();
 | 
						|
    }
 | 
						|
 | 
						|
    // Clone tags
 | 
						|
    $tags = $source->tags;
 | 
						|
    foreach ($tags as $tag) {
 | 
						|
        $newApplication->tags()->attach($tag->id);
 | 
						|
    }
 | 
						|
 | 
						|
    // Clone scheduled tasks
 | 
						|
    $scheduledTasks = $source->scheduled_tasks()->get();
 | 
						|
    foreach ($scheduledTasks as $task) {
 | 
						|
        $newTask = $task->replicate([
 | 
						|
            'id',
 | 
						|
            'created_at',
 | 
						|
            'updated_at',
 | 
						|
        ])->fill([
 | 
						|
            'uuid' => (string) new Cuid2,
 | 
						|
            'application_id' => $newApplication->id,
 | 
						|
            'team_id' => currentTeam()->id,
 | 
						|
        ]);
 | 
						|
        $newTask->save();
 | 
						|
    }
 | 
						|
 | 
						|
    // Clone previews with FQDN regeneration
 | 
						|
    $applicationPreviews = $source->previews()->get();
 | 
						|
    foreach ($applicationPreviews as $preview) {
 | 
						|
        $newPreview = $preview->replicate([
 | 
						|
            'id',
 | 
						|
            'created_at',
 | 
						|
            'updated_at',
 | 
						|
        ])->fill([
 | 
						|
            'uuid' => (string) new Cuid2,
 | 
						|
            'application_id' => $newApplication->id,
 | 
						|
            'status' => 'exited',
 | 
						|
            'fqdn' => null,
 | 
						|
            'docker_compose_domains' => null,
 | 
						|
        ]);
 | 
						|
        $newPreview->save();
 | 
						|
 | 
						|
        // Regenerate FQDN for the cloned preview
 | 
						|
        if ($newApplication->build_pack === 'dockercompose') {
 | 
						|
            $newPreview->generate_preview_fqdn_compose();
 | 
						|
        } else {
 | 
						|
            $newPreview->generate_preview_fqdn();
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    // Clone persistent volumes
 | 
						|
    $persistentVolumes = $source->persistentStorages()->get();
 | 
						|
    foreach ($persistentVolumes as $volume) {
 | 
						|
        $newName = '';
 | 
						|
        if (str_starts_with($volume->name, $source->uuid)) {
 | 
						|
            $newName = str($volume->name)->replace($source->uuid, $newApplication->uuid);
 | 
						|
        } else {
 | 
						|
            $newName = $newApplication->uuid.'-'.str($volume->name)->afterLast('-');
 | 
						|
        }
 | 
						|
 | 
						|
        $newPersistentVolume = $volume->replicate([
 | 
						|
            'id',
 | 
						|
            'created_at',
 | 
						|
            'updated_at',
 | 
						|
        ])->fill([
 | 
						|
            'name' => $newName,
 | 
						|
            'resource_id' => $newApplication->id,
 | 
						|
        ]);
 | 
						|
        $newPersistentVolume->save();
 | 
						|
 | 
						|
        if ($cloneVolumeData) {
 | 
						|
            try {
 | 
						|
                StopApplication::dispatch($source, false, false);
 | 
						|
                $sourceVolume = $volume->name;
 | 
						|
                $targetVolume = $newPersistentVolume->name;
 | 
						|
                $sourceServer = $source->destination->server;
 | 
						|
                $targetServer = $newApplication->destination->server;
 | 
						|
 | 
						|
                VolumeCloneJob::dispatch($sourceVolume, $targetVolume, $sourceServer, $targetServer, $newPersistentVolume);
 | 
						|
 | 
						|
                queue_application_deployment(
 | 
						|
                    deployment_uuid: (string) new Cuid2,
 | 
						|
                    application: $source,
 | 
						|
                    server: $sourceServer,
 | 
						|
                    destination: $source->destination,
 | 
						|
                    no_questions_asked: true
 | 
						|
                );
 | 
						|
            } catch (\Exception $e) {
 | 
						|
                \Log::error('Failed to copy volume data for '.$volume->name.': '.$e->getMessage());
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    // Clone file storages
 | 
						|
    $fileStorages = $source->fileStorages()->get();
 | 
						|
    foreach ($fileStorages as $storage) {
 | 
						|
        $newStorage = $storage->replicate([
 | 
						|
            'id',
 | 
						|
            'created_at',
 | 
						|
            'updated_at',
 | 
						|
        ])->fill([
 | 
						|
            'resource_id' => $newApplication->id,
 | 
						|
        ]);
 | 
						|
        $newStorage->save();
 | 
						|
    }
 | 
						|
 | 
						|
    // Clone production environment variables without triggering the created hook
 | 
						|
    $environmentVariables = $source->environment_variables()->get();
 | 
						|
    foreach ($environmentVariables as $environmentVariable) {
 | 
						|
        \App\Models\EnvironmentVariable::withoutEvents(function () use ($environmentVariable, $newApplication) {
 | 
						|
            $newEnvironmentVariable = $environmentVariable->replicate([
 | 
						|
                'id',
 | 
						|
                'created_at',
 | 
						|
                'updated_at',
 | 
						|
            ])->fill([
 | 
						|
                'resourceable_id' => $newApplication->id,
 | 
						|
                'resourceable_type' => $newApplication->getMorphClass(),
 | 
						|
                'is_preview' => false,
 | 
						|
            ]);
 | 
						|
            $newEnvironmentVariable->save();
 | 
						|
        });
 | 
						|
    }
 | 
						|
 | 
						|
    // Clone preview environment variables
 | 
						|
    $previewEnvironmentVariables = $source->environment_variables_preview()->get();
 | 
						|
    foreach ($previewEnvironmentVariables as $previewEnvironmentVariable) {
 | 
						|
        \App\Models\EnvironmentVariable::withoutEvents(function () use ($previewEnvironmentVariable, $newApplication) {
 | 
						|
            $newPreviewEnvironmentVariable = $previewEnvironmentVariable->replicate([
 | 
						|
                'id',
 | 
						|
                'created_at',
 | 
						|
                'updated_at',
 | 
						|
            ])->fill([
 | 
						|
                'resourceable_id' => $newApplication->id,
 | 
						|
                'resourceable_type' => $newApplication->getMorphClass(),
 | 
						|
                'is_preview' => true,
 | 
						|
            ]);
 | 
						|
            $newPreviewEnvironmentVariable->save();
 | 
						|
        });
 | 
						|
    }
 | 
						|
 | 
						|
    return $newApplication;
 | 
						|
}
 |