'$refresh', 'confirmDomainUsage', ]; protected function rules(): array { return [ 'application.name' => ValidationPatterns::nameRules(), 'application.description' => ValidationPatterns::descriptionRules(), 'application.fqdn' => 'nullable', 'application.git_repository' => 'required', 'application.git_branch' => 'required', 'application.git_commit_sha' => 'nullable', 'application.install_command' => 'nullable', 'application.build_command' => 'nullable', 'application.start_command' => 'nullable', 'application.build_pack' => 'required', 'application.static_image' => 'required', 'application.base_directory' => 'required', 'application.publish_directory' => 'nullable', 'application.ports_exposes' => 'required', 'application.ports_mappings' => 'nullable', 'application.custom_network_aliases' => 'nullable', 'application.dockerfile' => 'nullable', 'application.docker_registry_image_name' => 'nullable', 'application.docker_registry_image_tag' => 'nullable', 'application.dockerfile_location' => 'nullable', 'application.docker_compose_location' => 'nullable', 'application.docker_compose' => 'nullable', 'application.docker_compose_raw' => 'nullable', 'application.dockerfile_target_build' => 'nullable', 'application.docker_compose_custom_start_command' => 'nullable', 'application.docker_compose_custom_build_command' => 'nullable', 'application.custom_labels' => 'nullable', 'application.custom_docker_run_options' => 'nullable', 'application.pre_deployment_command' => 'nullable', 'application.pre_deployment_command_container' => 'nullable', 'application.post_deployment_command' => 'nullable', 'application.post_deployment_command_container' => 'nullable', 'application.custom_nginx_configuration' => 'nullable', 'application.settings.is_static' => 'boolean|required', 'application.settings.is_spa' => 'boolean|required', 'application.settings.is_build_server_enabled' => 'boolean|required', 'application.settings.is_container_label_escape_enabled' => 'boolean|required', 'application.settings.is_container_label_readonly_enabled' => 'boolean|required', 'application.settings.is_preserve_repository_enabled' => 'boolean|required', 'application.is_http_basic_auth_enabled' => 'boolean|required', 'application.http_basic_auth_username' => 'string|nullable', 'application.http_basic_auth_password' => 'string|nullable', 'application.watch_paths' => 'nullable', 'application.redirect' => 'string|required', ]; } protected function messages(): array { return array_merge( ValidationPatterns::combinedMessages(), [ 'application.name.required' => 'The Name field is required.', 'application.name.regex' => 'The Name may only contain letters, numbers, spaces, dashes (-), underscores (_), dots (.), slashes (/), colons (:), and parentheses ().', 'application.description.regex' => 'The Description contains invalid characters. Only letters, numbers, spaces, and common punctuation (- _ . : / () \' " , ! ? @ # % & + = [] {} | ~ ` *) are allowed.', 'application.git_repository.required' => 'The Git Repository field is required.', 'application.git_branch.required' => 'The Git Branch field is required.', 'application.build_pack.required' => 'The Build Pack field is required.', 'application.static_image.required' => 'The Static Image field is required.', 'application.base_directory.required' => 'The Base Directory field is required.', 'application.ports_exposes.required' => 'The Exposed Ports field is required.', 'application.settings.is_static.required' => 'The Static setting is required.', 'application.settings.is_static.boolean' => 'The Static setting must be true or false.', 'application.settings.is_spa.required' => 'The SPA setting is required.', 'application.settings.is_spa.boolean' => 'The SPA setting must be true or false.', 'application.settings.is_build_server_enabled.required' => 'The Build Server setting is required.', 'application.settings.is_build_server_enabled.boolean' => 'The Build Server setting must be true or false.', 'application.settings.is_container_label_escape_enabled.required' => 'The Container Label Escape setting is required.', 'application.settings.is_container_label_escape_enabled.boolean' => 'The Container Label Escape setting must be true or false.', 'application.settings.is_container_label_readonly_enabled.required' => 'The Container Label Readonly setting is required.', 'application.settings.is_container_label_readonly_enabled.boolean' => 'The Container Label Readonly setting must be true or false.', 'application.settings.is_preserve_repository_enabled.required' => 'The Preserve Repository setting is required.', 'application.settings.is_preserve_repository_enabled.boolean' => 'The Preserve Repository setting must be true or false.', 'application.is_http_basic_auth_enabled.required' => 'The HTTP Basic Auth setting is required.', 'application.is_http_basic_auth_enabled.boolean' => 'The HTTP Basic Auth setting must be true or false.', 'application.redirect.required' => 'The Redirect setting is required.', 'application.redirect.string' => 'The Redirect setting must be a string.', ] ); } protected $validationAttributes = [ 'application.name' => 'name', 'application.description' => 'description', 'application.fqdn' => 'FQDN', 'application.git_repository' => 'Git repository', 'application.git_branch' => 'Git branch', 'application.git_commit_sha' => 'Git commit SHA', 'application.install_command' => 'Install command', 'application.build_command' => 'Build command', 'application.start_command' => 'Start command', 'application.build_pack' => 'Build pack', 'application.static_image' => 'Static image', 'application.base_directory' => 'Base directory', 'application.publish_directory' => 'Publish directory', 'application.ports_exposes' => 'Ports exposes', 'application.ports_mappings' => 'Ports mappings', 'application.dockerfile' => 'Dockerfile', 'application.docker_registry_image_name' => 'Docker registry image name', 'application.docker_registry_image_tag' => 'Docker registry image tag', 'application.dockerfile_location' => 'Dockerfile location', 'application.docker_compose_location' => 'Docker compose location', 'application.docker_compose' => 'Docker compose', 'application.docker_compose_raw' => 'Docker compose raw', 'application.custom_labels' => 'Custom labels', 'application.dockerfile_target_build' => 'Dockerfile target build', 'application.custom_docker_run_options' => 'Custom docker run commands', 'application.custom_network_aliases' => 'Custom docker network aliases', 'application.docker_compose_custom_start_command' => 'Docker compose custom start command', 'application.docker_compose_custom_build_command' => 'Docker compose custom build command', 'application.custom_nginx_configuration' => 'Custom Nginx configuration', 'application.settings.is_static' => 'Is static', 'application.settings.is_spa' => 'Is SPA', 'application.settings.is_build_server_enabled' => 'Is build server enabled', 'application.settings.is_container_label_escape_enabled' => 'Is container label escape enabled', 'application.settings.is_container_label_readonly_enabled' => 'Is container label readonly', 'application.settings.is_preserve_repository_enabled' => 'Is preserve repository enabled', 'application.watch_paths' => 'Watch paths', 'application.redirect' => 'Redirect', ]; public function mount() { try { $this->parsedServices = $this->application->parse(); if (is_null($this->parsedServices) || empty($this->parsedServices)) { $this->dispatch('error', 'Failed to parse your docker-compose file. Please check the syntax and try again.'); return; } } catch (\Throwable $e) { $this->dispatch('error', $e->getMessage()); } if ($this->application->build_pack === 'dockercompose') { // Only update if user has permission try { $this->authorize('update', $this->application); $this->application->fqdn = null; $this->application->settings->save(); } catch (\Illuminate\Auth\Access\AuthorizationException $e) { // User doesn't have update permission, just continue without saving } } $this->parsedServiceDomains = $this->application->docker_compose_domains ? json_decode($this->application->docker_compose_domains, true) : []; // Convert service names with dots to use underscores for HTML form binding $sanitizedDomains = []; foreach ($this->parsedServiceDomains as $serviceName => $domain) { $sanitizedKey = str($serviceName)->slug('_')->toString(); $sanitizedDomains[$sanitizedKey] = $domain; } $this->parsedServiceDomains = $sanitizedDomains; $this->ports_exposes = $this->application->ports_exposes; $this->is_preserve_repository_enabled = $this->application->settings->is_preserve_repository_enabled; $this->is_container_label_escape_enabled = $this->application->settings->is_container_label_escape_enabled; $this->customLabels = $this->application->parseContainerLabels(); if (! $this->customLabels && $this->application->destination->server->proxyType() !== 'NONE' && $this->application->settings->is_container_label_readonly_enabled === true) { // Only update custom labels if user has permission try { $this->authorize('update', $this->application); $this->customLabels = str(implode('|coolify|', generateLabelsApplication($this->application)))->replace('|coolify|', "\n"); $this->application->custom_labels = base64_encode($this->customLabels); $this->application->save(); } catch (\Illuminate\Auth\Access\AuthorizationException $e) { // User doesn't have update permission, just use existing labels // $this->customLabels = str(implode('|coolify|', generateLabelsApplication($this->application)))->replace('|coolify|', "\n"); } } $this->initialDockerComposeLocation = $this->application->docker_compose_location; if ($this->application->build_pack === 'dockercompose' && ! $this->application->docker_compose_raw) { // Only load compose file if user has update permission try { $this->authorize('update', $this->application); $this->initLoadingCompose = true; $this->dispatch('info', 'Loading docker compose file.'); } catch (\Illuminate\Auth\Access\AuthorizationException $e) { // User doesn't have update permission, skip loading compose file } } if (str($this->application->status)->startsWith('running') && is_null($this->application->config_hash)) { $this->dispatch('configurationChanged'); } } public function instantSave() { try { $this->authorize('update', $this->application); if ($this->application->settings->isDirty('is_spa')) { $this->generateNginxConfiguration($this->application->settings->is_spa ? 'spa' : 'static'); } if ($this->application->isDirty('is_http_basic_auth_enabled')) { $this->application->save(); } $this->application->settings->save(); $this->dispatch('success', 'Settings saved.'); $this->application->refresh(); // If port_exposes changed, reset default labels if ($this->ports_exposes !== $this->application->ports_exposes || $this->is_container_label_escape_enabled !== $this->application->settings->is_container_label_escape_enabled) { $this->resetDefaultLabels(false); } if ($this->is_preserve_repository_enabled !== $this->application->settings->is_preserve_repository_enabled) { if ($this->application->settings->is_preserve_repository_enabled === false) { $this->application->fileStorages->each(function ($storage) { $storage->is_based_on_git = $this->application->settings->is_preserve_repository_enabled; $storage->save(); }); } } if ($this->application->settings->is_container_label_readonly_enabled) { $this->resetDefaultLabels(false); } } catch (\Throwable $e) { return handleError($e, $this); } } public function loadComposeFile($isInit = false, $showToast = true) { try { $this->authorize('update', $this->application); if ($isInit && $this->application->docker_compose_raw) { return; } ['parsedServices' => $this->parsedServices, 'initialDockerComposeLocation' => $this->initialDockerComposeLocation] = $this->application->loadComposeFile($isInit); if (is_null($this->parsedServices)) { $showToast && $this->dispatch('error', 'Failed to parse your docker-compose file. Please check the syntax and try again.'); return; } // Refresh parsedServiceDomains to reflect any changes in docker_compose_domains $this->application->refresh(); $this->parsedServiceDomains = $this->application->docker_compose_domains ? json_decode($this->application->docker_compose_domains, true) : []; // Convert service names with dots to use underscores for HTML form binding $sanitizedDomains = []; foreach ($this->parsedServiceDomains as $serviceName => $domain) { $sanitizedKey = str($serviceName)->slug('_')->toString(); $sanitizedDomains[$sanitizedKey] = $domain; } $this->parsedServiceDomains = $sanitizedDomains; $showToast && $this->dispatch('success', 'Docker compose file loaded.'); $this->dispatch('compose_loaded'); $this->dispatch('refreshStorages'); $this->dispatch('refreshEnvs'); } catch (\Throwable $e) { $this->application->docker_compose_location = $this->initialDockerComposeLocation; $this->application->save(); return handleError($e, $this); } finally { $this->initLoadingCompose = false; } } public function generateDomain(string $serviceName) { try { $this->authorize('update', $this->application); $uuid = new Cuid2; $domain = generateUrl(server: $this->application->destination->server, random: $uuid); $sanitizedKey = str($serviceName)->slug('_')->toString(); $this->parsedServiceDomains[$sanitizedKey]['domain'] = $domain; // Convert back to original service names for storage $originalDomains = []; foreach ($this->parsedServiceDomains as $key => $value) { // Find the original service name by checking parsed services $originalServiceName = $key; if (isset($this->parsedServices['services'])) { foreach ($this->parsedServices['services'] as $originalName => $service) { if (str($originalName)->slug('_')->toString() === $key) { $originalServiceName = $originalName; break; } } } $originalDomains[$originalServiceName] = $value; } $this->application->docker_compose_domains = json_encode($originalDomains); $this->application->save(); $this->dispatch('success', 'Domain generated.'); if ($this->application->build_pack === 'dockercompose') { $this->loadComposeFile(showToast: false); } return $domain; } catch (\Throwable $e) { return handleError($e, $this); } } public function updatedApplicationBaseDirectory() { if ($this->application->build_pack === 'dockercompose') { $this->loadComposeFile(); } } public function updatedApplicationSettingsIsStatic($value) { if ($value) { $this->generateNginxConfiguration(); } } public function updatedApplicationBuildPack() { // Check if user has permission to update try { $this->authorize('update', $this->application); } catch (\Illuminate\Auth\Access\AuthorizationException $e) { // User doesn't have permission, revert the change and return $this->application->refresh(); return; } if ($this->application->build_pack !== 'nixpacks') { $this->application->settings->is_static = false; $this->application->settings->save(); } else { $this->application->ports_exposes = $this->ports_exposes = 3000; $this->resetDefaultLabels(false); } if ($this->application->build_pack === 'dockercompose') { // Only update if user has permission try { $this->authorize('update', $this->application); $this->application->fqdn = null; $this->application->settings->save(); } catch (\Illuminate\Auth\Access\AuthorizationException $e) { // User doesn't have update permission, just continue without saving } } else { // Clear Docker Compose specific data when switching away from dockercompose if ($this->application->getOriginal('build_pack') === 'dockercompose') { $this->application->docker_compose_domains = null; $this->application->docker_compose_raw = null; // Remove SERVICE_FQDN_* and SERVICE_URL_* environment variables $this->application->environment_variables()->where('key', 'LIKE', 'SERVICE_FQDN_%')->delete(); $this->application->environment_variables()->where('key', 'LIKE', 'SERVICE_URL_%')->delete(); $this->application->environment_variables_preview()->where('key', 'LIKE', 'SERVICE_FQDN_%')->delete(); $this->application->environment_variables_preview()->where('key', 'LIKE', 'SERVICE_URL_%')->delete(); } } if ($this->application->build_pack === 'static') { $this->application->ports_exposes = $this->ports_exposes = 80; $this->resetDefaultLabels(false); $this->generateNginxConfiguration(); } $this->submit(); $this->dispatch('buildPackUpdated'); } public function getWildcardDomain() { try { $this->authorize('update', $this->application); $server = data_get($this->application, 'destination.server'); if ($server) { $fqdn = generateUrl(server: $server, random: $this->application->uuid); $this->application->fqdn = $fqdn; $this->application->save(); $this->resetDefaultLabels(); $this->dispatch('success', 'Wildcard domain generated.'); } } catch (\Throwable $e) { return handleError($e, $this); } } public function generateNginxConfiguration($type = 'static') { try { $this->authorize('update', $this->application); $this->application->custom_nginx_configuration = defaultNginxConfiguration($type); $this->application->save(); $this->dispatch('success', 'Nginx configuration generated.'); } catch (\Throwable $e) { return handleError($e, $this); } } public function resetDefaultLabels($manualReset = false) { try { if (! $this->application->settings->is_container_label_readonly_enabled && ! $manualReset) { return; } $this->customLabels = str(implode('|coolify|', generateLabelsApplication($this->application)))->replace('|coolify|', "\n"); $this->ports_exposes = $this->application->ports_exposes; $this->is_container_label_escape_enabled = $this->application->settings->is_container_label_escape_enabled; $this->application->custom_labels = base64_encode($this->customLabels); $this->application->save(); if ($this->application->build_pack === 'dockercompose') { $this->loadComposeFile(showToast: false); } $this->dispatch('configurationChanged'); } catch (\Throwable $e) { return handleError($e, $this); } } public function checkFqdns($showToaster = true) { if (data_get($this->application, 'fqdn')) { $domains = str($this->application->fqdn)->trim()->explode(','); if ($this->application->additional_servers->count() === 0) { foreach ($domains as $domain) { if (! validateDNSEntry($domain, $this->application->destination->server)) { $showToaster && $this->dispatch('error', 'Validating DNS failed.', "Make sure you have added the DNS records correctly.

$domain->{$this->application->destination->server->ip}

Check this documentation for further help."); } } } // Check for domain conflicts if not forcing save if (! $this->forceSaveDomains) { $result = checkDomainUsage(resource: $this->application); if ($result['hasConflicts']) { $this->domainConflicts = $result['conflicts']; $this->showDomainConflictModal = true; return false; } } else { // Reset the force flag after using it $this->forceSaveDomains = false; } $this->application->fqdn = $domains->implode(','); $this->resetDefaultLabels(false); } return true; } public function confirmDomainUsage() { $this->forceSaveDomains = true; $this->showDomainConflictModal = false; $this->submit(); } public function setRedirect() { $this->authorize('update', $this->application); try { $has_www = collect($this->application->fqdns)->filter(fn ($fqdn) => str($fqdn)->contains('www.'))->count(); if ($has_www === 0 && $this->application->redirect === 'www') { $this->dispatch('error', 'You want to redirect to www, but you do not have a www domain set.

Please add www to your domain list and as an A DNS record (if applicable).'); return; } $this->application->save(); $this->resetDefaultLabels(); $this->dispatch('success', 'Redirect updated.'); } catch (\Throwable $e) { return handleError($e, $this); } } public function submit($showToaster = true) { try { $this->authorize('update', $this->application); $this->application->fqdn = str($this->application->fqdn)->replaceEnd(',', '')->trim(); $this->application->fqdn = str($this->application->fqdn)->replaceStart(',', '')->trim(); $this->application->fqdn = str($this->application->fqdn)->trim()->explode(',')->map(function ($domain) { $domain = trim($domain); Url::fromString($domain, ['http', 'https']); return str($domain)->lower(); }); $this->application->fqdn = $this->application->fqdn->unique()->implode(','); $warning = sslipDomainWarning($this->application->fqdn); if ($warning) { $this->dispatch('warning', __('warning.sslipdomain')); } // $this->resetDefaultLabels(); if ($this->application->isDirty('redirect')) { $this->setRedirect(); } if ($this->application->isDirty('dockerfile')) { $this->application->parseHealthcheckFromDockerfile($this->application->dockerfile); } if (! $this->checkFqdns()) { return; // Stop if there are conflicts and user hasn't confirmed } $this->application->save(); if (! $this->customLabels && $this->application->destination->server->proxyType() !== 'NONE' && ! $this->application->settings->is_container_label_readonly_enabled) { $this->customLabels = str(implode('|coolify|', generateLabelsApplication($this->application)))->replace('|coolify|', "\n"); $this->application->custom_labels = base64_encode($this->customLabels); $this->application->save(); } if ($this->application->build_pack === 'dockercompose' && $this->initialDockerComposeLocation !== $this->application->docker_compose_location) { $compose_return = $this->loadComposeFile(showToast: false); if ($compose_return instanceof \Livewire\Features\SupportEvents\Event) { return; } } $this->validate(); if ($this->ports_exposes !== $this->application->ports_exposes || $this->is_container_label_escape_enabled !== $this->application->settings->is_container_label_escape_enabled) { $this->resetDefaultLabels(); } if (data_get($this->application, 'build_pack') === 'dockerimage') { $this->validate([ 'application.docker_registry_image_name' => 'required', ]); } if (data_get($this->application, 'custom_docker_run_options')) { $this->application->custom_docker_run_options = str($this->application->custom_docker_run_options)->trim(); } if (data_get($this->application, 'dockerfile')) { $port = get_port_from_dockerfile($this->application->dockerfile); if ($port && ! $this->application->ports_exposes) { $this->application->ports_exposes = $port; } } if ($this->application->base_directory && $this->application->base_directory !== '/') { $this->application->base_directory = rtrim($this->application->base_directory, '/'); } if ($this->application->publish_directory && $this->application->publish_directory !== '/') { $this->application->publish_directory = rtrim($this->application->publish_directory, '/'); } if ($this->application->build_pack === 'dockercompose') { $this->application->docker_compose_domains = json_encode($this->parsedServiceDomains); if ($this->application->isDirty('docker_compose_domains')) { foreach ($this->parsedServiceDomains as $service) { $domain = data_get($service, 'domain'); if ($domain) { if (! validateDNSEntry($domain, $this->application->destination->server)) { $showToaster && $this->dispatch('error', 'Validating DNS failed.', "Make sure you have added the DNS records correctly.

$domain->{$this->application->destination->server->ip}

Check this documentation for further help."); } } } // Check for domain conflicts if not forcing save if (! $this->forceSaveDomains) { $result = checkDomainUsage(resource: $this->application); if ($result['hasConflicts']) { $this->domainConflicts = $result['conflicts']; $this->showDomainConflictModal = true; return; } } else { // Reset the force flag after using it $this->forceSaveDomains = false; } $this->application->save(); $this->resetDefaultLabels(); } } $this->application->custom_labels = base64_encode($this->customLabels); $this->application->save(); $showToaster && ! $warning && $this->dispatch('success', 'Application settings updated!'); } catch (\Throwable $e) { $originalFqdn = $this->application->getOriginal('fqdn'); if ($originalFqdn !== $this->application->fqdn) { $this->application->fqdn = $originalFqdn; } return handleError($e, $this); } finally { $this->dispatch('configurationChanged'); } } public function downloadConfig() { $config = GenerateConfig::run($this->application, true); $fileName = str($this->application->name)->slug()->append('_config.json'); return response()->streamDownload(function () use ($config) { echo $config; }, $fileName, [ 'Content-Type' => 'application/json', 'Content-Disposition' => 'attachment; filename='.$fileName, ]); } private function updateServiceEnvironmentVariables() { $domains = collect(json_decode($this->application->docker_compose_domains, true)) ?? collect([]); foreach ($domains as $serviceName => $service) { $serviceNameFormatted = str($serviceName)->upper()->replace('-', '_')->replace('.', '_'); $domain = data_get($service, 'domain'); // Delete SERVICE_FQDN_ and SERVICE_URL_ variables if domain is removed $this->application->environment_variables()->where('resourceable_type', Application::class) ->where('resourceable_id', $this->application->id) ->where('key', 'LIKE', "SERVICE_FQDN_{$serviceNameFormatted}%") ->delete(); $this->application->environment_variables()->where('resourceable_type', Application::class) ->where('resourceable_id', $this->application->id) ->where('key', 'LIKE', "SERVICE_URL_{$serviceNameFormatted}%") ->delete(); if ($domain) { // Create or update SERVICE_FQDN_ and SERVICE_URL_ variables $fqdn = Url::fromString($domain); $port = $fqdn->getPort(); $path = $fqdn->getPath(); $urlValue = $fqdn->getScheme().'://'.$fqdn->getHost(); if ($path !== '/') { $urlValue = $urlValue.$path; } $fqdnValue = str($domain)->after('://'); if ($path !== '/') { $fqdnValue = $fqdnValue.$path; } // Create/update SERVICE_FQDN_ $this->application->environment_variables()->updateOrCreate([ 'key' => "SERVICE_FQDN_{$serviceNameFormatted}", ], [ 'value' => $fqdnValue, 'is_preview' => false, ]); // Create/update SERVICE_URL_ $this->application->environment_variables()->updateOrCreate([ 'key' => "SERVICE_URL_{$serviceNameFormatted}", ], [ 'value' => $urlValue, 'is_preview' => false, ]); // Create/update port-specific variables if port exists if (filled($port)) { $this->application->environment_variables()->updateOrCreate([ 'key' => "SERVICE_FQDN_{$serviceNameFormatted}_{$port}", ], [ 'value' => $fqdnValue, 'is_preview' => false, ]); $this->application->environment_variables()->updateOrCreate([ 'key' => "SERVICE_URL_{$serviceNameFormatted}_{$port}", ], [ 'value' => $urlValue, 'is_preview' => false, ]); } } } } }