327 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			327 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| 
 | |
| use App\Jobs\ApplicationDeployDockerImageJob;
 | |
| use App\Jobs\ApplicationDeploymentJob;
 | |
| use App\Jobs\ApplicationDeploymentNewJob;
 | |
| use App\Jobs\ApplicationDeploySimpleDockerfileJob;
 | |
| use App\Jobs\ApplicationRestartJob;
 | |
| use App\Jobs\MultipleApplicationDeploymentJob;
 | |
| use App\Models\Application;
 | |
| use App\Models\ApplicationDeploymentQueue;
 | |
| use App\Models\ApplicationPreview;
 | |
| use App\Models\Server;
 | |
| use Symfony\Component\Yaml\Yaml;
 | |
| 
 | |
| function queue_application_deployment(int $application_id, string $deployment_uuid, int | null $pull_request_id = 0, string $commit = 'HEAD', bool $force_rebuild = false, bool $is_webhook = false, bool $restart_only = false, ?string $git_type = null, bool $is_new_deployment = false)
 | |
| {
 | |
|     $deployment = ApplicationDeploymentQueue::create([
 | |
|         'application_id' => $application_id,
 | |
|         'deployment_uuid' => $deployment_uuid,
 | |
|         'pull_request_id' => $pull_request_id,
 | |
|         'force_rebuild' => $force_rebuild,
 | |
|         'is_webhook' => $is_webhook,
 | |
|         'restart_only' => $restart_only,
 | |
|         'commit' => $commit,
 | |
|         'git_type' => $git_type
 | |
|     ]);
 | |
|     $queued_deployments = ApplicationDeploymentQueue::where('application_id', $application_id)->where('status', 'queued')->get()->sortByDesc('created_at');
 | |
|     $running_deployments = ApplicationDeploymentQueue::where('application_id', $application_id)->where('status', 'in_progress')->get()->sortByDesc('created_at');
 | |
|     ray('Q:' . $queued_deployments->count() . 'R:' . $running_deployments->count() . '| Queuing deployment: ' . $deployment_uuid . ' of applicationID: ' . $application_id . ' pull request: ' . $pull_request_id . ' with commit: ' . $commit . ' and is it forced: ' . $force_rebuild);
 | |
|     if ($queued_deployments->count() > 1) {
 | |
|         $queued_deployments = $queued_deployments->skip(1);
 | |
|         $queued_deployments->each(function ($queued_deployment, $key) {
 | |
|             $queued_deployment->status = 'cancelled by system';
 | |
|             $queued_deployment->save();
 | |
|         });
 | |
|     }
 | |
|     if ($running_deployments->count() > 0) {
 | |
|         return;
 | |
|     }
 | |
|     if ($is_new_deployment) {
 | |
|         dispatch(new ApplicationDeploymentNewJob(
 | |
|             deployment: $deployment,
 | |
|             application: Application::find($application_id)
 | |
|         ));
 | |
|     } else {
 | |
|         dispatch(new ApplicationDeploymentJob(
 | |
|             application_deployment_queue_id: $deployment->id,
 | |
|         ));
 | |
|     }
 | |
| }
 | |
| 
 | |
| function queue_next_deployment(Application $application, bool $isNew = false)
 | |
| {
 | |
|     $next_found = ApplicationDeploymentQueue::where('application_id', $application->id)->where('status', 'queued')->first();
 | |
|     if ($next_found) {
 | |
|         if ($isNew) {
 | |
|             dispatch(new ApplicationDeploymentNewJob(
 | |
|                 deployment: $next_found,
 | |
|                 application: $application
 | |
|             ));
 | |
|         } else {
 | |
|             dispatch(new ApplicationDeploymentJob(
 | |
|                 application_deployment_queue_id: $next_found->id,
 | |
|             ));
 | |
|         }
 | |
|     }
 | |
| }
 | |
| // Deployment things
 | |
| function generateHostIpMapping(Server $server, string $network)
 | |
| {
 | |
|     // Generate custom host<->ip hostnames
 | |
|     $allContainers = instant_remote_process(["docker network inspect {$network} -f '{{json .Containers}}' "], $server);
 | |
|     $allContainers = format_docker_command_output_to_json($allContainers);
 | |
|     $ips = collect([]);
 | |
|     if (count($allContainers) > 0) {
 | |
|         $allContainers = $allContainers[0];
 | |
|         foreach ($allContainers as $container) {
 | |
|             $containerName = data_get($container, 'Name');
 | |
|             if ($containerName === 'coolify-proxy') {
 | |
|                 continue;
 | |
|             }
 | |
|             $containerIp = data_get($container, 'IPv4Address');
 | |
|             if ($containerName && $containerIp) {
 | |
|                 $containerIp = str($containerIp)->before('/');
 | |
|                 $ips->put($containerName, $containerIp->value());
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     return $ips->map(function ($ip, $name) {
 | |
|         return "--add-host $name:$ip";
 | |
|     })->implode(' ');
 | |
| }
 | |
| 
 | |
| function generateBaseDir(string $deplyomentUuid)
 | |
| {
 | |
|     return "/artifacts/$deplyomentUuid";
 | |
| }
 | |
| function generateWorkdir(string $deplyomentUuid, Application $application)
 | |
| {
 | |
|     return generateBaseDir($deplyomentUuid) . rtrim($application->base_directory, '/');
 | |
| }
 | |
| 
 | |
| function prepareHelperContainer(Server $server, string $network, string $deploymentUuid)
 | |
| {
 | |
|     $basedir = generateBaseDir($deploymentUuid);
 | |
|     $helperImage = config('coolify.helper_image');
 | |
| 
 | |
|     $serverUserHomeDir = instant_remote_process(["echo \$HOME"], $server);
 | |
|     $dockerConfigFileExists = instant_remote_process(["test -f {$serverUserHomeDir}/.docker/config.json && echo 'OK' || echo 'NOK'"], $server);
 | |
| 
 | |
|     $commands = collect([]);
 | |
|     if ($dockerConfigFileExists === 'OK') {
 | |
|         $commands->push([
 | |
|             "command" => "docker run -d --network $network --name $deploymentUuid --rm -v {$serverUserHomeDir}/.docker/config.json:/root/.docker/config.json:ro -v /var/run/docker.sock:/var/run/docker.sock $helperImage",
 | |
|             "hidden" => true,
 | |
|         ]);
 | |
|     } else {
 | |
|         $commands->push([
 | |
|             "command" => "docker run -d --network {$network} --name {$deploymentUuid} --rm -v /var/run/docker.sock:/var/run/docker.sock {$helperImage}",
 | |
|             "hidden" => true,
 | |
|         ]);
 | |
|     }
 | |
|     $commands->push([
 | |
|         "command" => executeInDocker($deploymentUuid, "mkdir -p {$basedir}"),
 | |
|         "hidden" => true,
 | |
|     ]);
 | |
|     return $commands;
 | |
| }
 | |
| 
 | |
| function generateComposeFile(string $deploymentUuid, Server $server, string $network, Application $application, string $containerName, string $imageName, ?ApplicationPreview $preview = null, int $pullRequestId = 0)
 | |
| {
 | |
|     $ports = $application->settings->is_static ? [80] : $application->ports_exposes_array;
 | |
|     $workDir = generateWorkdir($deploymentUuid, $application);
 | |
|     $persistent_storages = generateLocalPersistentVolumes($application, $pullRequestId);
 | |
|     $volume_names = generateLocalPersistentVolumesOnlyVolumeNames($application, $pullRequestId);
 | |
|     $environment_variables = generateEnvironmentVariables($application, $ports, $pullRequestId);
 | |
| 
 | |
|     if (data_get($application, 'custom_labels')) {
 | |
|         $labels = collect(str($application->custom_labels)->explode(','));
 | |
|         $labels = $labels->filter(function ($value, $key) {
 | |
|             return !str($value)->startsWith('coolify.');
 | |
|         });
 | |
|         $application->custom_labels = $labels->implode(',');
 | |
|         $application->save();
 | |
|     } else {
 | |
|         $labels = collect(generateLabelsApplication($application, $preview));
 | |
|     }
 | |
|     if ($pullRequestId !== 0) {
 | |
|         $labels = collect(generateLabelsApplication($application, $preview));
 | |
|     }
 | |
|     $labels = $labels->merge(defaultLabels($application->id, $application->uuid, 0))->toArray();
 | |
|     $docker_compose = [
 | |
|         'version' => '3.8',
 | |
|         'services' => [
 | |
|             $containerName => [
 | |
|                 'image' => $imageName,
 | |
|                 'container_name' => $containerName,
 | |
|                 'restart' => RESTART_MODE,
 | |
|                 'environment' => $environment_variables,
 | |
|                 'labels' => $labels,
 | |
|                 'expose' => $ports,
 | |
|                 'networks' => [
 | |
|                     $network,
 | |
|                 ],
 | |
|                 'mem_limit' => $application->limits_memory,
 | |
|                 'memswap_limit' => $application->limits_memory_swap,
 | |
|                 'mem_swappiness' => $application->limits_memory_swappiness,
 | |
|                 'mem_reservation' => $application->limits_memory_reservation,
 | |
|                 'cpus' => (int) $application->limits_cpus,
 | |
|                 'cpuset' => $application->limits_cpuset,
 | |
|                 'cpu_shares' => $application->limits_cpu_shares,
 | |
|             ]
 | |
|         ],
 | |
|         'networks' => [
 | |
|             $network => [
 | |
|                 'external' => true,
 | |
|                 'name' => $network,
 | |
|                 'attachable' => true
 | |
|             ]
 | |
|         ]
 | |
|     ];
 | |
|     if ($server->isLogDrainEnabled() && $application->isLogDrainEnabled()) {
 | |
|         $docker_compose['services'][$containerName]['logging'] = [
 | |
|             'driver' => 'fluentd',
 | |
|             'options' => [
 | |
|                 'fluentd-address' => "tcp://127.0.0.1:24224",
 | |
|                 'fluentd-async' => "true",
 | |
|                 'fluentd-sub-second-precision' => "true",
 | |
|             ]
 | |
|         ];
 | |
|     }
 | |
|     if ($application->settings->is_gpu_enabled) {
 | |
|         $docker_compose['services'][$containerName]['deploy']['resources']['reservations']['devices'] = [
 | |
|             [
 | |
|                 'driver' => data_get($application, 'settings.gpu_driver', 'nvidia'),
 | |
|                 'capabilities' => ['gpu'],
 | |
|                 'options' => data_get($application, 'settings.gpu_options', [])
 | |
|             ]
 | |
|         ];
 | |
|         if (data_get($application, 'settings.gpu_count')) {
 | |
|             $count = data_get($application, 'settings.gpu_count');
 | |
|             if ($count === 'all') {
 | |
|                 $docker_compose['services'][$containerName]['deploy']['resources']['reservations']['devices'][0]['count'] = $count;
 | |
|             } else {
 | |
|                 $docker_compose['services'][$containerName]['deploy']['resources']['reservations']['devices'][0]['count'] = (int) $count;
 | |
|             }
 | |
|         } else if (data_get($application, 'settings.gpu_device_ids')) {
 | |
|             $docker_compose['services'][$containerName]['deploy']['resources']['reservations']['devices'][0]['ids'] = data_get($application, 'settings.gpu_device_ids');
 | |
|         }
 | |
|     }
 | |
|     if ($application->isHealthcheckDisabled()) {
 | |
|         data_forget($docker_compose, 'services.' . $containerName . '.healthcheck');
 | |
|     }
 | |
|     if (count($application->ports_mappings_array) > 0 && $pullRequestId === 0) {
 | |
|         $docker_compose['services'][$containerName]['ports'] = $application->ports_mappings_array;
 | |
|     }
 | |
|     if (count($persistent_storages) > 0) {
 | |
|         $docker_compose['services'][$containerName]['volumes'] = $persistent_storages;
 | |
|     }
 | |
|     if (count($volume_names) > 0) {
 | |
|         $docker_compose['volumes'] = $volume_names;
 | |
|     }
 | |
|     $docker_compose = Yaml::dump($docker_compose, 10);
 | |
|     $docker_compose_base64 = base64_encode($docker_compose);
 | |
|     $commands = collect([]);
 | |
|     $commands->push([
 | |
|         "command" => executeInDocker($deploymentUuid, "echo '{$docker_compose_base64}' | base64 -d > {$workDir}/docker-compose.yml"),
 | |
|         "hidden" => true,
 | |
|     ]);
 | |
|     return $commands;
 | |
| }
 | |
| function generateLocalPersistentVolumes(Application $application, int $pullRequestId = 0)
 | |
| {
 | |
|     $local_persistent_volumes = [];
 | |
|     foreach ($application->persistentStorages as $persistentStorage) {
 | |
|         $volume_name = $persistentStorage->host_path ?? $persistentStorage->name;
 | |
|         if ($pullRequestId !== 0) {
 | |
|             $volume_name = $volume_name . '-pr-' . $pullRequestId;
 | |
|         }
 | |
|         $local_persistent_volumes[] = $volume_name . ':' . $persistentStorage->mount_path;
 | |
|     }
 | |
|     return $local_persistent_volumes;
 | |
| }
 | |
| 
 | |
| function generateLocalPersistentVolumesOnlyVolumeNames(Application $application, int $pullRequestId = 0)
 | |
| {
 | |
|     $local_persistent_volumes_names = [];
 | |
|     foreach ($application->persistentStorages as $persistentStorage) {
 | |
|         if ($persistentStorage->host_path) {
 | |
|             continue;
 | |
|         }
 | |
|         $name = $persistentStorage->name;
 | |
| 
 | |
|         if ($pullRequestId !== 0) {
 | |
|             $name = $name . '-pr-' . $pullRequestId;
 | |
|         }
 | |
| 
 | |
|         $local_persistent_volumes_names[$name] = [
 | |
|             'name' => $name,
 | |
|             'external' => false,
 | |
|         ];
 | |
|     }
 | |
|     return $local_persistent_volumes_names;
 | |
| }
 | |
| function generateEnvironmentVariables(Application $application, $ports, int $pullRequestId = 0)
 | |
| {
 | |
|     $environment_variables = collect();
 | |
|     // ray('Generate Environment Variables')->green();
 | |
|     if ($pullRequestId === 0) {
 | |
|         // ray($this->application->runtime_environment_variables)->green();
 | |
|         foreach ($application->runtime_environment_variables as $env) {
 | |
|             $environment_variables->push("$env->key=$env->value");
 | |
|         }
 | |
|         foreach ($application->nixpacks_environment_variables as $env) {
 | |
|             $environment_variables->push("$env->key=$env->value");
 | |
|         }
 | |
|     } else {
 | |
|         // ray($this->application->runtime_environment_variables_preview)->green();
 | |
|         foreach ($application->runtime_environment_variables_preview as $env) {
 | |
|             $environment_variables->push("$env->key=$env->value");
 | |
|         }
 | |
|         foreach ($application->nixpacks_environment_variables_preview as $env) {
 | |
|             $environment_variables->push("$env->key=$env->value");
 | |
|         }
 | |
|     }
 | |
|     // Add PORT if not exists, use the first port as default
 | |
|     if ($environment_variables->filter(fn ($env) => str($env)->contains('PORT'))->isEmpty()) {
 | |
|         $environment_variables->push("PORT={$ports[0]}");
 | |
|     }
 | |
|     return $environment_variables->all();
 | |
| }
 | |
| 
 | |
| function startNewApplication(Application $application, string $deploymentUuid, ApplicationDeploymentQueue $loggingModel)
 | |
| {
 | |
|     $commands = collect([]);
 | |
|     $workDir = generateWorkdir($deploymentUuid, $application);
 | |
|     if ($application->build_pack === 'dockerimage') {
 | |
|         $loggingModel->addLogEntry('Pulling latest images from the registry.');
 | |
|         $commands->push(
 | |
|             [
 | |
|                 "command" => executeInDocker($deploymentUuid, "docker compose --project-directory {$workDir} pull"),
 | |
|                 "hidden" => true
 | |
|             ],
 | |
|             [
 | |
|                 "command" => executeInDocker($deploymentUuid, "docker compose --project-directory {$workDir} up --build -d"),
 | |
|                 "hidden" => true
 | |
|             ],
 | |
|         );
 | |
|     } else {
 | |
|         $commands->push(
 | |
|             [
 | |
|                 "command" => executeInDocker($deploymentUuid, "docker compose --project-directory {$workDir} up --build -d"),
 | |
|                 "hidden" => true
 | |
|             ],
 | |
|         );
 | |
|     }
 | |
|     return $commands;
 | |
| }
 | |
| function removeOldDeployment(string $containerName)
 | |
| {
 | |
|     $commands = collect([]);
 | |
|     $commands->push(
 | |
|         ["docker rm -f $containerName >/dev/null 2>&1"],
 | |
|     );
 | |
|     return $commands;
 | |
| }
 | 
