1488 lines
		
	
	
		
			63 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			1488 lines
		
	
	
		
			63 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| 
 | |
| use App\Models\Application;
 | |
| use App\Models\ApplicationPreview;
 | |
| use App\Models\EnvironmentVariable;
 | |
| use App\Models\InstanceSettings;
 | |
| use App\Models\LocalFileVolume;
 | |
| use App\Models\LocalPersistentVolume;
 | |
| use App\Models\Server;
 | |
| use App\Models\Service;
 | |
| use App\Models\ServiceApplication;
 | |
| use App\Models\ServiceDatabase;
 | |
| use App\Models\StandaloneMariadb;
 | |
| use App\Models\StandaloneMongodb;
 | |
| use App\Models\StandaloneMysql;
 | |
| use App\Models\StandalonePostgresql;
 | |
| use App\Models\StandaloneRedis;
 | |
| use App\Models\Team;
 | |
| use App\Models\User;
 | |
| use App\Notifications\Channels\DiscordChannel;
 | |
| use App\Notifications\Channels\EmailChannel;
 | |
| use App\Notifications\Channels\TelegramChannel;
 | |
| use App\Notifications\Internal\GeneralNotification;
 | |
| use DanHarrin\LivewireRateLimiting\Exceptions\TooManyRequestsException;
 | |
| use Illuminate\Database\QueryException;
 | |
| use Illuminate\Mail\Message;
 | |
| use Illuminate\Notifications\Messages\MailMessage;
 | |
| use Illuminate\Support\Collection;
 | |
| use Illuminate\Support\Facades\Cache;
 | |
| use Illuminate\Support\Facades\File;
 | |
| use Illuminate\Support\Facades\Http;
 | |
| use Illuminate\Support\Facades\Mail;
 | |
| use Illuminate\Support\Facades\Route;
 | |
| use Illuminate\Support\Str;
 | |
| use Illuminate\Support\Stringable;
 | |
| use Poliander\Cron\CronExpression;
 | |
| use Visus\Cuid2\Cuid2;
 | |
| use phpseclib3\Crypt\RSA;
 | |
| use Spatie\Url\Url;
 | |
| use Symfony\Component\Yaml\Yaml;
 | |
| 
 | |
| function base_configuration_dir(): string
 | |
| {
 | |
|     return '/data/coolify';
 | |
| }
 | |
| function application_configuration_dir(): string
 | |
| {
 | |
|     return base_configuration_dir() . "/applications";
 | |
| }
 | |
| function service_configuration_dir(): string
 | |
| {
 | |
|     return base_configuration_dir() . "/services";
 | |
| }
 | |
| function database_configuration_dir(): string
 | |
| {
 | |
|     return base_configuration_dir() . "/databases";
 | |
| }
 | |
| function database_proxy_dir($uuid): string
 | |
| {
 | |
|     return base_configuration_dir() . "/databases/$uuid/proxy";
 | |
| }
 | |
| function backup_dir(): string
 | |
| {
 | |
|     return base_configuration_dir() . "/backups";
 | |
| }
 | |
| 
 | |
| function generate_readme_file(string $name, string $updated_at): string
 | |
| {
 | |
|     return "Resource name: $name\nLatest Deployment Date: $updated_at";
 | |
| }
 | |
| 
 | |
| function isInstanceAdmin()
 | |
| {
 | |
|     return auth()?->user()?->isInstanceAdmin() ?? false;
 | |
| }
 | |
| 
 | |
| function currentTeam()
 | |
| {
 | |
|     return auth()?->user()?->currentTeam() ?? null;
 | |
| }
 | |
| 
 | |
| function showBoarding(): bool
 | |
| {
 | |
|     return currentTeam()->show_boarding ?? false;
 | |
| }
 | |
| function refreshSession(?Team $team = null): void
 | |
| {
 | |
|     if (!$team) {
 | |
|         if (auth()->user()?->currentTeam()) {
 | |
|             $team = Team::find(auth()->user()->currentTeam()->id);
 | |
|         } else {
 | |
|             $team = User::find(auth()->user()->id)->teams->first();
 | |
|         }
 | |
|     }
 | |
|     Cache::forget('team:' . auth()->user()->id);
 | |
|     Cache::remember('team:' . auth()->user()->id, 3600, function () use ($team) {
 | |
|         return $team;
 | |
|     });
 | |
|     session(['currentTeam' => $team]);
 | |
| }
 | |
| function handleError(?Throwable $error = null, ?Livewire\Component $livewire = null, ?string $customErrorMessage = null)
 | |
| {
 | |
|     if ($error instanceof TooManyRequestsException) {
 | |
|         if (isset($livewire)) {
 | |
|             return $livewire->emit('error', "Too many requests. Please try again in {$error->secondsUntilAvailable} seconds.");
 | |
|         }
 | |
|         return "Too many requests. Please try again in {$error->secondsUntilAvailable} seconds.";
 | |
|     }
 | |
| 
 | |
|     if ($error instanceof Throwable) {
 | |
|         $message = $error->getMessage();
 | |
|     } else {
 | |
|         $message = null;
 | |
|     }
 | |
|     if ($customErrorMessage) {
 | |
|         $message = $customErrorMessage . ' ' . $message;
 | |
|     }
 | |
| 
 | |
|     if (isset($livewire)) {
 | |
|         return $livewire->emit('error', $message);
 | |
|     }
 | |
|     throw new Exception($message);
 | |
| }
 | |
| function get_route_parameters(): array
 | |
| {
 | |
|     return Route::current()->parameters();
 | |
| }
 | |
| 
 | |
| function get_latest_version_of_coolify(): string
 | |
| {
 | |
|     try {
 | |
|         $response = Http::get('https://cdn.coollabs.io/coolify/versions.json');
 | |
|         $versions = $response->json();
 | |
|         return data_get($versions, 'coolify.v4.version');
 | |
|     } catch (\Throwable $e) {
 | |
|         //throw $e;
 | |
|         ray($e->getMessage());
 | |
|         return '0.0.0';
 | |
|     }
 | |
| }
 | |
| 
 | |
| function generate_random_name(?string $cuid = null): string
 | |
| {
 | |
|     $generator = new \Nubs\RandomNameGenerator\All(
 | |
|         [
 | |
|             new \Nubs\RandomNameGenerator\Alliteration(),
 | |
|         ]
 | |
|     );
 | |
|     if (is_null($cuid)) {
 | |
|         $cuid = new Cuid2(7);
 | |
|     }
 | |
|     return Str::kebab("{$generator->getName()}-$cuid");
 | |
| }
 | |
| function generateSSHKey()
 | |
| {
 | |
|     $key = RSA::createKey();
 | |
|     return [
 | |
|         'private' => $key->toString('PKCS1'),
 | |
|         'public' => $key->getPublicKey()->toString('OpenSSH', ['comment' => 'coolify-generated-ssh-key'])
 | |
|     ];
 | |
| }
 | |
| function formatPrivateKey(string $privateKey)
 | |
| {
 | |
|     $privateKey = trim($privateKey);
 | |
|     if (!str_ends_with($privateKey, "\n")) {
 | |
|         $privateKey .= "\n";
 | |
|     }
 | |
|     return $privateKey;
 | |
| }
 | |
| function generate_application_name(string $git_repository, string $git_branch, ?string $cuid = null): string
 | |
| {
 | |
|     if (is_null($cuid)) {
 | |
|         $cuid = new Cuid2(7);
 | |
|     }
 | |
|     return Str::kebab("$git_repository:$git_branch-$cuid");
 | |
| }
 | |
| 
 | |
| function is_transactional_emails_active(): bool
 | |
| {
 | |
|     return isEmailEnabled(InstanceSettings::get());
 | |
| }
 | |
| 
 | |
| function set_transanctional_email_settings(InstanceSettings | null $settings = null): string|null
 | |
| {
 | |
|     if (!$settings) {
 | |
|         $settings = InstanceSettings::get();
 | |
|     }
 | |
|     config()->set('mail.from.address', data_get($settings, 'smtp_from_address'));
 | |
|     config()->set('mail.from.name', data_get($settings, 'smtp_from_name'));
 | |
|     if (data_get($settings, 'resend_enabled')) {
 | |
|         config()->set('mail.default', 'resend');
 | |
|         config()->set('resend.api_key', data_get($settings, 'resend_api_key'));
 | |
|         return 'resend';
 | |
|     }
 | |
|     if (data_get($settings, 'smtp_enabled')) {
 | |
|         config()->set('mail.default', 'smtp');
 | |
|         config()->set('mail.mailers.smtp', [
 | |
|             "transport" => "smtp",
 | |
|             "host" => data_get($settings, 'smtp_host'),
 | |
|             "port" => data_get($settings, 'smtp_port'),
 | |
|             "encryption" => data_get($settings, 'smtp_encryption'),
 | |
|             "username" => data_get($settings, 'smtp_username'),
 | |
|             "password" => data_get($settings, 'smtp_password'),
 | |
|             "timeout" => data_get($settings, 'smtp_timeout'),
 | |
|             "local_domain" => null,
 | |
|         ]);
 | |
|         return 'smtp';
 | |
|     }
 | |
|     return null;
 | |
| }
 | |
| 
 | |
| function base_ip(): string
 | |
| {
 | |
|     if (isDev()) {
 | |
|         return "localhost";
 | |
|     }
 | |
|     $settings = InstanceSettings::get();
 | |
|     if ($settings->public_ipv4) {
 | |
|         return "$settings->public_ipv4";
 | |
|     }
 | |
|     if ($settings->public_ipv6) {
 | |
|         return "$settings->public_ipv6";
 | |
|     }
 | |
|     return "localhost";
 | |
| }
 | |
| function getFqdnWithoutPort(String $fqdn)
 | |
| {
 | |
|     $url = Url::fromString($fqdn);
 | |
|     $host = $url->getHost();
 | |
|     $scheme = $url->getScheme();
 | |
|     $path = $url->getPath();
 | |
|     return "$scheme://$host$path";
 | |
| }
 | |
| /**
 | |
|  * If fqdn is set, return it, otherwise return public ip.
 | |
|  */
 | |
| function base_url(bool $withPort = true): string
 | |
| {
 | |
|     $settings = InstanceSettings::get();
 | |
|     if ($settings->fqdn) {
 | |
|         return $settings->fqdn;
 | |
|     }
 | |
|     $port = config('app.port');
 | |
|     if ($settings->public_ipv4) {
 | |
|         if ($withPort) {
 | |
|             if (isDev()) {
 | |
|                 return "http://localhost:$port";
 | |
|             }
 | |
|             return "http://$settings->public_ipv4:$port";
 | |
|         }
 | |
|         if (isDev()) {
 | |
|             return "http://localhost";
 | |
|         }
 | |
|         return "http://$settings->public_ipv4";
 | |
|     }
 | |
|     if ($settings->public_ipv6) {
 | |
|         if ($withPort) {
 | |
|             return "http://$settings->public_ipv6:$port";
 | |
|         }
 | |
|         return "http://$settings->public_ipv6";
 | |
|     }
 | |
|     return url('/');
 | |
| }
 | |
| 
 | |
| function isDev(): bool
 | |
| {
 | |
|     return config('app.env') === 'local';
 | |
| }
 | |
| 
 | |
| function isCloud(): bool
 | |
| {
 | |
|     return !config('coolify.self_hosted');
 | |
| }
 | |
| 
 | |
| function validate_cron_expression($expression_to_validate): bool
 | |
| {
 | |
|     $isValid = false;
 | |
|     $expression = new CronExpression($expression_to_validate);
 | |
|     $isValid = $expression->isValid();
 | |
| 
 | |
|     if (isset(VALID_CRON_STRINGS[$expression_to_validate])) {
 | |
|         $isValid = true;
 | |
|     }
 | |
|     return $isValid;
 | |
| }
 | |
| function send_internal_notification(string $message): void
 | |
| {
 | |
|     try {
 | |
|         $baseUrl = config('app.name');
 | |
|         $team = Team::find(0);
 | |
|         $team?->notify(new GeneralNotification("👀 {$baseUrl}: " . $message));
 | |
|         ray("👀 {$baseUrl}: " . $message);
 | |
|     } catch (\Throwable $e) {
 | |
|         ray($e->getMessage());
 | |
|     }
 | |
| }
 | |
| function send_user_an_email(MailMessage $mail, string $email, ?string $cc = null): void
 | |
| {
 | |
|     $settings = InstanceSettings::get();
 | |
|     $type = set_transanctional_email_settings($settings);
 | |
|     if (!$type) {
 | |
|         throw new Exception('No email settings found.');
 | |
|     }
 | |
|     if ($cc) {
 | |
|         Mail::send(
 | |
|             [],
 | |
|             [],
 | |
|             fn (Message $message) => $message
 | |
|                 ->to($email)
 | |
|                 ->replyTo($email)
 | |
|                 ->cc($cc)
 | |
|                 ->subject($mail->subject)
 | |
|                 ->html((string) $mail->render())
 | |
|         );
 | |
|     } else {
 | |
|         Mail::send(
 | |
|             [],
 | |
|             [],
 | |
|             fn (Message $message) => $message
 | |
|                 ->to($email)
 | |
|                 ->subject($mail->subject)
 | |
|                 ->html((string) $mail->render())
 | |
|         );
 | |
|     }
 | |
| }
 | |
| function isTestEmailEnabled($notifiable)
 | |
| {
 | |
|     if (data_get($notifiable, 'use_instance_email_settings') && isInstanceAdmin()) {
 | |
|         return true;
 | |
|     } else if (data_get($notifiable, 'smtp_enabled') || data_get($notifiable, 'resend_enabled') && auth()->user()->isAdminFromSession()) {
 | |
|         return true;
 | |
|     }
 | |
|     return false;
 | |
| }
 | |
| function isEmailEnabled($notifiable)
 | |
| {
 | |
|     return data_get($notifiable, 'smtp_enabled') || data_get($notifiable, 'resend_enabled') || data_get($notifiable, 'use_instance_email_settings');
 | |
| }
 | |
| function setNotificationChannels($notifiable, $event)
 | |
| {
 | |
|     $channels = [];
 | |
|     $isEmailEnabled = isEmailEnabled($notifiable);
 | |
|     $isDiscordEnabled = data_get($notifiable, 'discord_enabled');
 | |
|     $isTelegramEnabled = data_get($notifiable, 'telegram_enabled');
 | |
|     $isSubscribedToEmailEvent = data_get($notifiable, "smtp_notifications_$event");
 | |
|     $isSubscribedToDiscordEvent = data_get($notifiable, "discord_notifications_$event");
 | |
|     $isSubscribedToTelegramEvent = data_get($notifiable, "telegram_notifications_$event");
 | |
| 
 | |
|     if ($isDiscordEnabled && $isSubscribedToDiscordEvent) {
 | |
|         $channels[] = DiscordChannel::class;
 | |
|     }
 | |
|     if ($isEmailEnabled && $isSubscribedToEmailEvent) {
 | |
|         $channels[] = EmailChannel::class;
 | |
|     }
 | |
|     if ($isTelegramEnabled && $isSubscribedToTelegramEvent) {
 | |
|         $channels[] = TelegramChannel::class;
 | |
|     }
 | |
|     return $channels;
 | |
| }
 | |
| function parseEnvFormatToArray($env_file_contents)
 | |
| {
 | |
|     $env_array = array();
 | |
|     $lines = explode("\n", $env_file_contents);
 | |
|     foreach ($lines as $line) {
 | |
|         if ($line === '' || substr($line, 0, 1) === '#') {
 | |
|             continue;
 | |
|         }
 | |
|         $equals_pos = strpos($line, '=');
 | |
|         if ($equals_pos !== false) {
 | |
|             $key = substr($line, 0, $equals_pos);
 | |
|             $value = substr($line, $equals_pos + 1);
 | |
|             if (substr($value, 0, 1) === '"' && substr($value, -1) === '"') {
 | |
|                 $value = substr($value, 1, -1);
 | |
|             } elseif (substr($value, 0, 1) === "'" && substr($value, -1) === "'") {
 | |
|                 $value = substr($value, 1, -1);
 | |
|             }
 | |
|             $env_array[$key] = $value;
 | |
|         }
 | |
|     }
 | |
|     return $env_array;
 | |
| }
 | |
| 
 | |
| function data_get_str($data, $key, $default = null): Stringable
 | |
| {
 | |
|     $str = data_get($data, $key, $default) ?? $default;
 | |
|     return Str::of($str);
 | |
| }
 | |
| 
 | |
| function generateFqdn(Server $server, string $random)
 | |
| {
 | |
|     $wildcard = data_get($server, 'settings.wildcard_domain');
 | |
|     if (is_null($wildcard) || $wildcard === '') {
 | |
|         $wildcard = sslip($server);
 | |
|     }
 | |
|     $url = Url::fromString($wildcard);
 | |
|     $host = $url->getHost();
 | |
|     $path = $url->getPath() === '/' ? '' : $url->getPath();
 | |
|     $scheme = $url->getScheme();
 | |
|     $finalFqdn = "$scheme://{$random}.$host$path";
 | |
|     return $finalFqdn;
 | |
| }
 | |
| function sslip(Server $server)
 | |
| {
 | |
|     if (isDev()) {
 | |
|         return "http://127.0.0.1.sslip.io";
 | |
|     }
 | |
|     if ($server->ip === 'host.docker.internal') {
 | |
|         $baseIp = base_ip();
 | |
|         return "http://$baseIp.sslip.io";
 | |
|     }
 | |
|     return "http://{$server->ip}.sslip.io";
 | |
| }
 | |
| 
 | |
| function getServiceTemplates()
 | |
| {
 | |
|     if (isDev()) {
 | |
|         $services = File::get(base_path('templates/service-templates.json'));
 | |
|         $services = collect(json_decode($services))->sortKeys();
 | |
|     } else {
 | |
|         try {
 | |
|             $response = Http::retry(3, 50)->get(config('constants.services.official'));
 | |
|             if ($response->failed()) {
 | |
|                 return collect([]);
 | |
|             }
 | |
|             $services = $response->json();
 | |
|             $services = collect($services)->sortKeys();
 | |
|         } catch (\Throwable $e) {
 | |
|             $services = collect([]);
 | |
|         }
 | |
|     }
 | |
|     // $version = config('version');
 | |
|     // $services = $services->map(function ($service) use ($version) {
 | |
|     //     if (version_compare($version, data_get($service, 'minVersion', '0.0.0'), '<')) {
 | |
|     //         $service->disabled = true;
 | |
|     //     }
 | |
|     //     return $service;
 | |
|     // });
 | |
|     return $services;
 | |
| }
 | |
| 
 | |
| function getResourceByUuid(string $uuid, ?int $teamId = null)
 | |
| {
 | |
|     $resource = queryResourcesByUuid($uuid);
 | |
|     if (!is_null($teamId)) {
 | |
|         if (!is_null($resource) && $resource->environment->project->team_id === $teamId) {
 | |
|             return $resource;
 | |
|         }
 | |
|         return null;
 | |
|     } else {
 | |
|         return $resource;
 | |
|     }
 | |
| }
 | |
| function queryResourcesByUuid(string $uuid)
 | |
| {
 | |
|     $resource = null;
 | |
|     $application = Application::whereUuid($uuid)->first();
 | |
|     if ($application) return $application;
 | |
|     $service = Service::whereUuid($uuid)->first();
 | |
|     if ($service) return $service;
 | |
|     $postgresql = StandalonePostgresql::whereUuid($uuid)->first();
 | |
|     if ($postgresql) return $postgresql;
 | |
|     $redis = StandaloneRedis::whereUuid($uuid)->first();
 | |
|     if ($redis) return $redis;
 | |
|     $mongodb = StandaloneMongodb::whereUuid($uuid)->first();
 | |
|     if ($mongodb) return $mongodb;
 | |
|     $mysql = StandaloneMysql::whereUuid($uuid)->first();
 | |
|     if ($mysql) return $mysql;
 | |
|     $mariadb = StandaloneMariadb::whereUuid($uuid)->first();
 | |
|     if ($mariadb) return $mariadb;
 | |
|     return $resource;
 | |
| }
 | |
| 
 | |
| function generateDeployWebhook($resource)
 | |
| {
 | |
|     $baseUrl = base_url();
 | |
|     $api = Url::fromString($baseUrl) . '/api/v1';
 | |
|     $endpoint = '/deploy';
 | |
|     $uuid = data_get($resource, 'uuid');
 | |
|     $url = $api . $endpoint . "?uuid=$uuid&force=false";
 | |
|     return $url;
 | |
| }
 | |
| function generateGitManualWebhook($resource, $type)
 | |
| {
 | |
|     if ($resource->source_id !== 0 && !is_null($resource->source_id)) {
 | |
|         return null;
 | |
|     }
 | |
|     if ($resource->getMorphClass() === 'App\Models\Application') {
 | |
|         $baseUrl = base_url();
 | |
|         $api = Url::fromString($baseUrl) . "/webhooks/source/$type/events/manual";
 | |
|         return $api;
 | |
|     }
 | |
|     return null;
 | |
| }
 | |
| function removeAnsiColors($text)
 | |
| {
 | |
|     return preg_replace('/\e[[][A-Za-z0-9];?[0-9]*m?/', '', $text);
 | |
| }
 | |
| 
 | |
| function getTopLevelNetworks(Service|Application $resource)
 | |
| {
 | |
|     if ($resource->getMorphClass() === 'App\Models\Service') {
 | |
|         if ($resource->docker_compose_raw) {
 | |
|             try {
 | |
|                 $yaml = Yaml::parse($resource->docker_compose_raw);
 | |
|             } catch (\Exception $e) {
 | |
|                 throw new \Exception($e->getMessage());
 | |
|             }
 | |
|             $services = data_get($yaml, 'services');
 | |
|             $topLevelNetworks = collect(data_get($yaml, 'networks', []));
 | |
|             $definedNetwork = collect([$resource->uuid]);
 | |
|             $services = collect($services)->map(function ($service, $_) use ($topLevelNetworks, $definedNetwork) {
 | |
|                 $serviceNetworks = collect(data_get($service, 'networks', []));
 | |
| 
 | |
|                 // Collect/create/update networks
 | |
|                 if ($serviceNetworks->count() > 0) {
 | |
|                     foreach ($serviceNetworks as $networkName => $networkDetails) {
 | |
|                         $networkExists = $topLevelNetworks->contains(function ($value, $key) use ($networkName) {
 | |
|                             return $value == $networkName || $key == $networkName;
 | |
|                         });
 | |
|                         if (!$networkExists) {
 | |
|                             $topLevelNetworks->put($networkDetails, null);
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 $definedNetworkExists = $topLevelNetworks->contains(function ($value, $_) use ($definedNetwork) {
 | |
|                     return $value == $definedNetwork;
 | |
|                 });
 | |
|                 if (!$definedNetworkExists) {
 | |
|                     foreach ($definedNetwork as $network) {
 | |
|                         $topLevelNetworks->put($network,  [
 | |
|                             'name' => $network,
 | |
|                             'external' => true
 | |
|                         ]);
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 return $service;
 | |
|             });
 | |
|             return $topLevelNetworks->keys();
 | |
|         }
 | |
|     } else if ($resource->getMorphClass() === 'App\Models\Application') {
 | |
|         try {
 | |
|             $yaml = Yaml::parse($resource->docker_compose_raw);
 | |
|         } catch (\Exception $e) {
 | |
|             throw new \Exception($e->getMessage());
 | |
|         }
 | |
|         $server = $resource->destination->server;
 | |
|         $topLevelNetworks = collect(data_get($yaml, 'networks', []));
 | |
|         $services = data_get($yaml, 'services');
 | |
|         $definedNetwork = collect([$resource->uuid]);
 | |
|         $services = collect($services)->map(function ($service, $_) use ($topLevelNetworks, $definedNetwork) {
 | |
|             $serviceNetworks = collect(data_get($service, 'networks', []));
 | |
| 
 | |
|             // Collect/create/update networks
 | |
|             if ($serviceNetworks->count() > 0) {
 | |
|                 foreach ($serviceNetworks as $networkName => $networkDetails) {
 | |
|                     $networkExists = $topLevelNetworks->contains(function ($value, $key) use ($networkName) {
 | |
|                         return $value == $networkName || $key == $networkName;
 | |
|                     });
 | |
|                     if (!$networkExists) {
 | |
|                         $topLevelNetworks->put($networkDetails, null);
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|             $definedNetworkExists = $topLevelNetworks->contains(function ($value, $_) use ($definedNetwork) {
 | |
|                 return $value == $definedNetwork;
 | |
|             });
 | |
|             if (!$definedNetworkExists) {
 | |
|                 foreach ($definedNetwork as $network) {
 | |
|                     $topLevelNetworks->put($network,  [
 | |
|                         'name' => $network,
 | |
|                         'external' => true
 | |
|                     ]);
 | |
|                 }
 | |
|             }
 | |
|             return $service;
 | |
|         });
 | |
|         return $topLevelNetworks->keys();
 | |
|     }
 | |
| }
 | |
| function parseDockerComposeFile(Service|Application $resource, bool $isNew = false, int $pull_request_id = 0, bool $is_pr = false)
 | |
| {
 | |
|     // ray()->clearAll();
 | |
|     if ($resource->getMorphClass() === 'App\Models\Service') {
 | |
|         if ($resource->docker_compose_raw) {
 | |
|             try {
 | |
|                 $yaml = Yaml::parse($resource->docker_compose_raw);
 | |
|             } catch (\Exception $e) {
 | |
|                 throw new \Exception($e->getMessage());
 | |
|             }
 | |
| 
 | |
|             $topLevelVolumes = collect(data_get($yaml, 'volumes', []));
 | |
|             $topLevelNetworks = collect(data_get($yaml, 'networks', []));
 | |
|             $dockerComposeVersion = data_get($yaml, 'version') ?? '3.8';
 | |
|             $services = data_get($yaml, 'services');
 | |
| 
 | |
|             $generatedServiceFQDNS = collect([]);
 | |
|             if (is_null($resource->destination)) {
 | |
|                 $destination = $resource->server->destinations()->first();
 | |
|                 if ($destination) {
 | |
|                     $resource->destination()->associate($destination);
 | |
|                     $resource->save();
 | |
|                 }
 | |
|             }
 | |
|             $definedNetwork = collect([$resource->uuid]);
 | |
| 
 | |
|             $services = collect($services)->map(function ($service, $serviceName) use ($topLevelVolumes, $topLevelNetworks, $definedNetwork, $isNew, $generatedServiceFQDNS, $resource) {
 | |
|                 $serviceVolumes = collect(data_get($service, 'volumes', []));
 | |
|                 $servicePorts = collect(data_get($service, 'ports', []));
 | |
|                 $serviceNetworks = collect(data_get($service, 'networks', []));
 | |
|                 $serviceVariables = collect(data_get($service, 'environment', []));
 | |
|                 $serviceLabels = collect(data_get($service, 'labels', []));
 | |
|                 if ($serviceLabels->count() > 0) {
 | |
|                     $removedLabels = collect([]);
 | |
|                     $serviceLabels = $serviceLabels->filter(function ($serviceLabel, $serviceLabelName) use ($removedLabels) {
 | |
|                         if (!str($serviceLabel)->contains('=')) {
 | |
|                             $removedLabels->put($serviceLabelName, $serviceLabel);
 | |
|                             return false;
 | |
|                         }
 | |
|                         return $serviceLabel;
 | |
|                     });
 | |
|                     foreach ($removedLabels as $removedLabelName => $removedLabel) {
 | |
|                         $serviceLabels->push("$removedLabelName=$removedLabel");
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 $containerName = "$serviceName-{$resource->uuid}";
 | |
| 
 | |
|                 // Decide if the service is a database
 | |
|                 $isDatabase = isDatabaseImage(data_get_str($service, 'image'));
 | |
|                 $image = data_get_str($service, 'image');
 | |
|                 data_set($service, 'is_database', $isDatabase);
 | |
| 
 | |
|                 // Create new serviceApplication or serviceDatabase
 | |
|                 if ($isDatabase) {
 | |
|                     if ($isNew) {
 | |
|                         $savedService = ServiceDatabase::create([
 | |
|                             'name' => $serviceName,
 | |
|                             'image' => $image,
 | |
|                             'service_id' => $resource->id
 | |
|                         ]);
 | |
|                     } else {
 | |
|                         $savedService = ServiceDatabase::where([
 | |
|                             'name' => $serviceName,
 | |
|                             'service_id' => $resource->id
 | |
|                         ])->first();
 | |
|                     }
 | |
|                 } else {
 | |
|                     if ($isNew) {
 | |
|                         $savedService = ServiceApplication::create([
 | |
|                             'name' => $serviceName,
 | |
|                             'image' => $image,
 | |
|                             'service_id' => $resource->id
 | |
|                         ]);
 | |
|                     } else {
 | |
|                         $savedService = ServiceApplication::where([
 | |
|                             'name' => $serviceName,
 | |
|                             'service_id' => $resource->id
 | |
|                         ])->first();
 | |
|                     }
 | |
|                 }
 | |
|                 if (is_null($savedService)) {
 | |
|                     if ($isDatabase) {
 | |
|                         $savedService = ServiceDatabase::create([
 | |
|                             'name' => $serviceName,
 | |
|                             'image' => $image,
 | |
|                             'service_id' => $resource->id
 | |
|                         ]);
 | |
|                     } else {
 | |
|                         $savedService = ServiceApplication::create([
 | |
|                             'name' => $serviceName,
 | |
|                             'image' => $image,
 | |
|                             'service_id' => $resource->id
 | |
|                         ]);
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 // Check if image changed
 | |
|                 if ($savedService->image !== $image) {
 | |
|                     $savedService->image = $image;
 | |
|                     $savedService->save();
 | |
|                 }
 | |
| 
 | |
|                 // Collect/create/update networks
 | |
|                 if ($serviceNetworks->count() > 0) {
 | |
|                     foreach ($serviceNetworks as $networkName => $networkDetails) {
 | |
|                         $networkExists = $topLevelNetworks->contains(function ($value, $key) use ($networkName) {
 | |
|                             return $value == $networkName || $key == $networkName;
 | |
|                         });
 | |
|                         if (!$networkExists) {
 | |
|                             $topLevelNetworks->put($networkDetails, null);
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 // Collect/create/update ports
 | |
|                 $collectedPorts = collect([]);
 | |
|                 if ($servicePorts->count() > 0) {
 | |
|                     foreach ($servicePorts as $sport) {
 | |
|                         if (is_string($sport) || is_numeric($sport)) {
 | |
|                             $collectedPorts->push($sport);
 | |
|                         }
 | |
|                         if (is_array($sport)) {
 | |
|                             $target = data_get($sport, 'target');
 | |
|                             $published = data_get($sport, 'published');
 | |
|                             $protocol = data_get($sport, 'protocol');
 | |
|                             $collectedPorts->push("$target:$published/$protocol");
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|                 $savedService->ports = $collectedPorts->implode(',');
 | |
|                 $savedService->save();
 | |
| 
 | |
|                 // Add Coolify specific networks
 | |
|                 $definedNetworkExists = $topLevelNetworks->contains(function ($value, $_) use ($definedNetwork) {
 | |
|                     return $value == $definedNetwork;
 | |
|                 });
 | |
|                 if (!$definedNetworkExists) {
 | |
|                     foreach ($definedNetwork as $network) {
 | |
|                         $topLevelNetworks->put($network,  [
 | |
|                             'name' => $network,
 | |
|                             'external' => true
 | |
|                         ]);
 | |
|                     }
 | |
|                 }
 | |
|                 $networks = collect();
 | |
|                 foreach ($serviceNetworks as $key => $serviceNetwork) {
 | |
|                     if (gettype($serviceNetwork) === 'string') {
 | |
|                         // networks:
 | |
|                         //  - appwrite
 | |
|                         $networks->put($serviceNetwork, null);
 | |
|                     } else if (gettype($serviceNetwork) === 'array') {
 | |
|                         // networks:
 | |
|                         //   default:
 | |
|                         //     ipv4_address: 192.168.203.254
 | |
|                         // $networks->put($serviceNetwork, null);
 | |
|                         ray($key);
 | |
|                         $networks->put($key, $serviceNetwork);
 | |
|                     }
 | |
|                 }
 | |
|                 foreach ($definedNetwork as $key => $network) {
 | |
|                     $networks->put($network, null);
 | |
|                 }
 | |
|                 data_set($service, 'networks', $networks->toArray());
 | |
| 
 | |
|                 // Collect/create/update volumes
 | |
|                 if ($serviceVolumes->count() > 0) {
 | |
|                     $serviceVolumes = $serviceVolumes->map(function ($volume) use ($savedService, $topLevelVolumes) {
 | |
|                         $type = null;
 | |
|                         $source = null;
 | |
|                         $target = null;
 | |
|                         $content = null;
 | |
|                         $isDirectory = false;
 | |
|                         if (is_string($volume)) {
 | |
|                             $source = Str::of($volume)->before(':');
 | |
|                             $target = Str::of($volume)->after(':')->beforeLast(':');
 | |
|                             if ($source->startsWith('./') || $source->startsWith('/') || $source->startsWith('~')) {
 | |
|                                 $type = Str::of('bind');
 | |
|                             } else {
 | |
|                                 $type = Str::of('volume');
 | |
|                             }
 | |
|                         } else if (is_array($volume)) {
 | |
|                             $type = data_get_str($volume, 'type');
 | |
|                             $source = data_get_str($volume, 'source');
 | |
|                             $target = data_get_str($volume, 'target');
 | |
|                             $content = data_get($volume, 'content');
 | |
|                             $isDirectory = (bool) data_get($volume, 'isDirectory', false);
 | |
|                             $foundConfig = $savedService->fileStorages()->whereMountPath($target)->first();
 | |
|                             if ($foundConfig) {
 | |
|                                 $contentNotNull = data_get($foundConfig, 'content');
 | |
|                                 if ($contentNotNull) {
 | |
|                                     $content = $contentNotNull;
 | |
|                                 }
 | |
|                                 $isDirectory = (bool) data_get($foundConfig, 'is_directory');
 | |
|                             }
 | |
|                         }
 | |
|                         if ($type->value() === 'bind') {
 | |
|                             if ($source->value() === "/var/run/docker.sock") {
 | |
|                                 return $volume;
 | |
|                             }
 | |
|                             if ($source->value() === '/tmp' || $source->value() === '/tmp/') {
 | |
|                                 return $volume;
 | |
|                             }
 | |
|                             LocalFileVolume::updateOrCreate(
 | |
|                                 [
 | |
|                                     'mount_path' => $target,
 | |
|                                     'resource_id' => $savedService->id,
 | |
|                                     'resource_type' => get_class($savedService)
 | |
|                                 ],
 | |
|                                 [
 | |
|                                     'fs_path' => $source,
 | |
|                                     'mount_path' => $target,
 | |
|                                     'content' => $content,
 | |
|                                     'is_directory' => $isDirectory,
 | |
|                                     'resource_id' => $savedService->id,
 | |
|                                     'resource_type' => get_class($savedService)
 | |
|                                 ]
 | |
|                             );
 | |
|                         } else if ($type->value() === 'volume') {
 | |
|                             $slugWithoutUuid = Str::slug($source, '-');
 | |
|                             $name = "{$savedService->service->uuid}_{$slugWithoutUuid}";
 | |
|                             if (is_string($volume)) {
 | |
|                                 $source = Str::of($volume)->before(':');
 | |
|                                 $target = Str::of($volume)->after(':')->beforeLast(':');
 | |
|                                 $source = $name;
 | |
|                                 $volume = "$source:$target";
 | |
|                             } else if (is_array($volume)) {
 | |
|                                 data_set($volume, 'source', $name);
 | |
|                             }
 | |
|                             $topLevelVolumes->put($name, [
 | |
|                                 'name' => $name,
 | |
|                             ]);
 | |
|                             LocalPersistentVolume::updateOrCreate(
 | |
|                                 [
 | |
|                                     'mount_path' => $target,
 | |
|                                     'resource_id' => $savedService->id,
 | |
|                                     'resource_type' => get_class($savedService)
 | |
|                                 ],
 | |
|                                 [
 | |
|                                     'name' => $name,
 | |
|                                     'mount_path' => $target,
 | |
|                                     'resource_id' => $savedService->id,
 | |
|                                     'resource_type' => get_class($savedService)
 | |
|                                 ]
 | |
|                             );
 | |
|                         }
 | |
|                         $savedService->getFilesFromServer(isInit: true);
 | |
|                         return $volume;
 | |
|                     });
 | |
|                     data_set($service, 'volumes', $serviceVolumes->toArray());
 | |
|                 }
 | |
| 
 | |
|                 // Add env_file with at least .env to the service
 | |
|                 // $envFile = collect(data_get($service, 'env_file', []));
 | |
|                 // if ($envFile->count() > 0) {
 | |
|                 //     if (!$envFile->contains('.env')) {
 | |
|                 //         $envFile->push('.env');
 | |
|                 //     }
 | |
|                 // } else {
 | |
|                 //     $envFile = collect(['.env']);
 | |
|                 // }
 | |
|                 // data_set($service, 'env_file', $envFile->toArray());
 | |
| 
 | |
| 
 | |
|                 // Get variables from the service
 | |
|                 foreach ($serviceVariables as $variableName => $variable) {
 | |
|                     if (is_numeric($variableName)) {
 | |
|                         $variable = Str::of($variable);
 | |
|                         if ($variable->contains('=')) {
 | |
|                             // - SESSION_SECRET=123
 | |
|                             // - SESSION_SECRET=
 | |
|                             $key = $variable->before('=');
 | |
|                             $value = $variable->after('=');
 | |
|                         } else {
 | |
|                             // - SESSION_SECRET
 | |
|                             $key = $variable;
 | |
|                             $value = null;
 | |
|                         }
 | |
|                     } else {
 | |
|                         // SESSION_SECRET: 123
 | |
|                         // SESSION_SECRET:
 | |
|                         $key = Str::of($variableName);
 | |
|                         $value = Str::of($variable);
 | |
|                     }
 | |
|                     // TODO: here is the problem
 | |
|                     if ($key->startsWith('SERVICE_FQDN')) {
 | |
|                         if ($isNew || $savedService->fqdn === null) {
 | |
|                             $name = $key->after('SERVICE_FQDN_')->beforeLast('_')->lower();
 | |
|                             $fqdn = generateFqdn($resource->server, "{$name->value()}-{$resource->uuid}");
 | |
|                             if (substr_count($key->value(), '_') === 3) {
 | |
|                                 // SERVICE_FQDN_UMAMI_1000
 | |
|                                 $port = $key->afterLast('_');
 | |
|                             } else {
 | |
|                                 // SERVICE_FQDN_UMAMI
 | |
|                                 $port = null;
 | |
|                             }
 | |
|                             if ($port) {
 | |
|                                 $fqdn = "$fqdn:$port";
 | |
|                             }
 | |
|                             if (substr_count($key->value(), '_') >= 2) {
 | |
|                                 if (is_null($value)) {
 | |
|                                     $value = Str::of('/');
 | |
|                                 }
 | |
|                                 $path = $value->value();
 | |
|                                 if ($generatedServiceFQDNS->count() > 0) {
 | |
|                                     $alreadyGenerated = $generatedServiceFQDNS->has($key->value());
 | |
|                                     if ($alreadyGenerated) {
 | |
|                                         $fqdn = $generatedServiceFQDNS->get($key->value());
 | |
|                                     } else {
 | |
|                                         $generatedServiceFQDNS->put($key->value(), $fqdn);
 | |
|                                     }
 | |
|                                 } else {
 | |
|                                     $generatedServiceFQDNS->put($key->value(), $fqdn);
 | |
|                                 }
 | |
|                                 $fqdn = "$fqdn$path";
 | |
|                             }
 | |
| 
 | |
|                             if (!$isDatabase) {
 | |
|                                 if ($savedService->fqdn) {
 | |
|                                     $fqdn = $savedService->fqdn . ',' . $fqdn;
 | |
|                                 } else {
 | |
|                                     $fqdn = $fqdn;
 | |
|                                 }
 | |
|                                 $savedService->fqdn = $fqdn;
 | |
|                                 $savedService->save();
 | |
|                             }
 | |
|                         }
 | |
|                         // data_forget($service, "environment.$variableName");
 | |
|                         // $yaml = data_forget($yaml, "services.$serviceName.environment.$variableName");
 | |
|                         // if (count(data_get($yaml, 'services.' . $serviceName . '.environment')) === 0) {
 | |
|                         //     $yaml = data_forget($yaml, "services.$serviceName.environment");
 | |
|                         // }
 | |
|                         continue;
 | |
|                     }
 | |
|                     if ($value?->startsWith('$')) {
 | |
|                         $value = Str::of(replaceVariables($value));
 | |
|                         $key = $value;
 | |
|                         $foundEnv = EnvironmentVariable::where([
 | |
|                             'key' => $key,
 | |
|                             'service_id' => $resource->id,
 | |
|                         ])->first();
 | |
|                         if ($value->startsWith('SERVICE_')) {
 | |
|                             ['command' => $command, 'forService' => $forService, 'generatedValue' => $generatedValue, 'port' => $port] = parseEnvVariable($value);
 | |
|                             if ($command->value() === 'FQDN' || $command->value() === 'URL') {
 | |
|                                 if (Str::lower($forService) === $serviceName) {
 | |
|                                     $fqdn = generateFqdn($resource->server, $containerName);
 | |
|                                 } else {
 | |
|                                     $fqdn = generateFqdn($resource->server, Str::lower($forService) . '-' . $resource->uuid);
 | |
|                                 }
 | |
|                                 if ($port) {
 | |
|                                     $fqdn = "$fqdn:$port";
 | |
|                                 }
 | |
|                                 if ($foundEnv) {
 | |
|                                     $fqdn = data_get($foundEnv, 'value');
 | |
|                                 } else {
 | |
|                                     if ($command->value() === 'URL') {
 | |
|                                         $fqdn = Str::of($fqdn)->after('://')->value();
 | |
|                                     }
 | |
|                                     EnvironmentVariable::create([
 | |
|                                         'key' => $key,
 | |
|                                         'value' => $fqdn,
 | |
|                                         'is_build_time' => false,
 | |
|                                         'service_id' => $resource->id,
 | |
|                                         'is_preview' => false,
 | |
|                                     ]);
 | |
|                                 }
 | |
|                                 if (!$isDatabase) {
 | |
|                                     if ($command->value() === 'FQDN' && is_null($savedService->fqdn)) {
 | |
|                                         $savedService->fqdn = $fqdn;
 | |
|                                         $savedService->save();
 | |
|                                     }
 | |
|                                 }
 | |
|                             } else {
 | |
|                                 $generatedValue = generateEnvValue($command);
 | |
|                                 if (!$foundEnv) {
 | |
|                                     EnvironmentVariable::create([
 | |
|                                         'key' => $key,
 | |
|                                         'value' => $generatedValue,
 | |
|                                         'is_build_time' => false,
 | |
|                                         'service_id' => $resource->id,
 | |
|                                         'is_preview' => false,
 | |
|                                     ]);
 | |
|                                 }
 | |
|                             }
 | |
|                         } else {
 | |
|                             if ($value->contains(':-')) {
 | |
|                                 $key = $value->before(':');
 | |
|                                 $defaultValue = $value->after(':-');
 | |
|                             } else if ($value->contains('-')) {
 | |
|                                 $key = $value->before('-');
 | |
|                                 $defaultValue = $value->after('-');
 | |
|                             } else if ($value->contains(':?')) {
 | |
|                                 $key = $value->before(':');
 | |
|                                 $defaultValue = $value->after(':?');
 | |
|                             } else if ($value->contains('?')) {
 | |
|                                 $key = $value->before('?');
 | |
|                                 $defaultValue = $value->after('?');
 | |
|                             } else {
 | |
|                                 $key = $value;
 | |
|                                 $defaultValue = null;
 | |
|                             }
 | |
|                             if ($foundEnv) {
 | |
|                                 $defaultValue = data_get($foundEnv, 'value');
 | |
|                             }
 | |
|                             EnvironmentVariable::updateOrCreate([
 | |
|                                 'key' => $key,
 | |
|                                 'service_id' => $resource->id,
 | |
|                             ], [
 | |
|                                 'value' => $defaultValue,
 | |
|                                 'is_build_time' => false,
 | |
|                                 'service_id' => $resource->id,
 | |
|                                 'is_preview' => false,
 | |
|                             ]);
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 // Add labels to the service
 | |
|                 if ($savedService->serviceType()) {
 | |
|                     $fqdns = generateServiceSpecificFqdns($savedService, forTraefik: true);
 | |
|                 } else {
 | |
|                     $fqdns = collect(data_get($savedService, 'fqdns'));
 | |
|                 }
 | |
|                 $defaultLabels = defaultLabels($resource->id, $containerName, type: 'service', subType: $isDatabase ? 'database' : 'application', subId: $savedService->id);
 | |
|                 $serviceLabels = $serviceLabels->merge($defaultLabels);
 | |
|                 if (!$isDatabase && $fqdns->count() > 0) {
 | |
|                     if ($fqdns) {
 | |
|                         $serviceLabels = $serviceLabels->merge(fqdnLabelsForTraefik($resource->uuid, $fqdns, true));
 | |
|                     }
 | |
|                 }
 | |
|                 if ($resource->server->isLogDrainEnabled() && $savedService->isLogDrainEnabled()) {
 | |
|                     data_set($service, 'logging', [
 | |
|                         'driver' => 'fluentd',
 | |
|                         'options' => [
 | |
|                             'fluentd-address' => "tcp://127.0.0.1:24224",
 | |
|                             'fluentd-async' => "true",
 | |
|                             'fluentd-sub-second-precision' => "true",
 | |
|                         ]
 | |
|                     ]);
 | |
|                 }
 | |
|                 data_set($service, 'labels', $serviceLabels->toArray());
 | |
|                 data_forget($service, 'is_database');
 | |
|                 data_set($service, 'restart', RESTART_MODE);
 | |
|                 data_set($service, 'container_name', $containerName);
 | |
|                 data_forget($service, 'volumes.*.content');
 | |
|                 data_forget($service, 'volumes.*.isDirectory');
 | |
|                 // Remove unnecessary variables from service.environment
 | |
|                 // $withoutServiceEnvs = collect([]);
 | |
|                 // collect(data_get($service, 'environment'))->each(function ($value, $key) use ($withoutServiceEnvs) {
 | |
|                 //     ray($key, $value);
 | |
|                 //     if (!Str::of($key)->startsWith('$SERVICE_') && !Str::of($value)->startsWith('SERVICE_')) {
 | |
|                 //         $k = Str::of($value)->before("=");
 | |
|                 //         $v = Str::of($value)->after("=");
 | |
|                 //         $withoutServiceEnvs->put($k->value(), $v->value());
 | |
|                 //     }
 | |
|                 // });
 | |
|                 // ray($withoutServiceEnvs);
 | |
|                 // data_set($service, 'environment', $withoutServiceEnvs->toArray());
 | |
|                 return $service;
 | |
|             });
 | |
|             $finalServices = [
 | |
|                 'version' => $dockerComposeVersion,
 | |
|                 'services' => $services->toArray(),
 | |
|                 'volumes' => $topLevelVolumes->toArray(),
 | |
|                 'networks' => $topLevelNetworks->toArray(),
 | |
|             ];
 | |
|             $resource->docker_compose_raw = Yaml::dump($yaml, 10, 2);
 | |
|             $resource->docker_compose = Yaml::dump($finalServices, 10, 2);
 | |
|             $resource->save();
 | |
|             $resource->saveComposeConfigs();
 | |
|             return collect($finalServices);
 | |
|         } else {
 | |
|             return collect([]);
 | |
|         }
 | |
|     } else if ($resource->getMorphClass() === 'App\Models\Application') {
 | |
|         $isSameDockerComposeFile = false;
 | |
|         if ($resource->dockerComposePrLocation() === $resource->dockerComposeLocation()) {
 | |
|             $isSameDockerComposeFile = true;
 | |
|             $is_pr = false;
 | |
|         }
 | |
|         if ($is_pr) {
 | |
|             try {
 | |
|                 $yaml = Yaml::parse($resource->docker_compose_pr_raw);
 | |
|             } catch (\Exception $e) {
 | |
|                 throw new \Exception($e->getMessage());
 | |
|             }
 | |
|         } else {
 | |
|             try {
 | |
|                 $yaml = Yaml::parse($resource->docker_compose_raw);
 | |
|             } catch (\Exception $e) {
 | |
|                 throw new \Exception($e->getMessage());
 | |
|             }
 | |
|         }
 | |
|         $server = $resource->destination->server;
 | |
|         $topLevelVolumes = collect(data_get($yaml, 'volumes', []));
 | |
|         if ($pull_request_id !== 0) {
 | |
|             $topLevelVolumes = collect([]);
 | |
|         }
 | |
|         $topLevelNetworks = collect(data_get($yaml, 'networks', []));
 | |
|         $dockerComposeVersion = data_get($yaml, 'version') ?? '3.8';
 | |
|         $services = data_get($yaml, 'services');
 | |
| 
 | |
|         $generatedServiceFQDNS = collect([]);
 | |
|         if (is_null($resource->destination)) {
 | |
|             $destination = $server->destinations()->first();
 | |
|             if ($destination) {
 | |
|                 $resource->destination()->associate($destination);
 | |
|                 $resource->save();
 | |
|             }
 | |
|         }
 | |
|         $definedNetwork = collect([$resource->uuid]);
 | |
|         if ($pull_request_id !== 0) {
 | |
|             $definedNetwork = collect(["{$resource->uuid}-$pull_request_id"]);
 | |
|         }
 | |
|         $services = collect($services)->map(function ($service, $serviceName) use ($topLevelVolumes, $topLevelNetworks, $definedNetwork, $isNew, $generatedServiceFQDNS, $resource, $server, $pull_request_id) {
 | |
|             $serviceVolumes = collect(data_get($service, 'volumes', []));
 | |
|             $servicePorts = collect(data_get($service, 'ports', []));
 | |
|             $serviceNetworks = collect(data_get($service, 'networks', []));
 | |
|             $serviceVariables = collect(data_get($service, 'environment', []));
 | |
|             $serviceLabels = collect(data_get($service, 'labels', []));
 | |
|             if ($serviceLabels->count() > 0) {
 | |
|                 $removedLabels = collect([]);
 | |
|                 $serviceLabels = $serviceLabels->filter(function ($serviceLabel, $serviceLabelName) use ($removedLabels) {
 | |
|                     if (!str($serviceLabel)->contains('=')) {
 | |
|                         $removedLabels->put($serviceLabelName, $serviceLabel);
 | |
|                         return false;
 | |
|                     }
 | |
|                     return $serviceLabel;
 | |
|                 });
 | |
|                 foreach ($removedLabels as $removedLabelName => $removedLabel) {
 | |
|                     $serviceLabels->push("$removedLabelName=$removedLabel");
 | |
|                 }
 | |
|             }
 | |
|             $baseName = generateApplicationContainerName($resource, $pull_request_id);
 | |
|             $containerName = "$serviceName-$baseName";
 | |
|             if ($pull_request_id !== 0) {
 | |
|                 if (count($serviceVolumes) > 0) {
 | |
|                     $serviceVolumes = $serviceVolumes->map(function ($volume) use ($resource, $pull_request_id, $topLevelVolumes) {
 | |
|                         if (is_string($volume)) {
 | |
|                             $volume = str($volume);
 | |
|                             if ($volume->contains(':')) {
 | |
|                                 $name = $volume->before(':');
 | |
|                                 $mount = $volume->after(':');
 | |
|                                 $newName = $resource->uuid . "-{$name}-pr-$pull_request_id";
 | |
|                                 $volume = str("$newName:$mount");
 | |
|                                 $topLevelVolumes->put($newName, [
 | |
|                                     'name' => $newName,
 | |
|                                 ]);
 | |
|                             }
 | |
|                         } else if (is_array($volume)) {
 | |
|                             $source = data_get($volume, 'source');
 | |
|                             if ($source) {
 | |
|                                 $newSource = $resource->uuid . "-{$source}-pr-$pull_request_id";
 | |
|                                 data_set($volume, 'source', $newSource);
 | |
|                                 $topLevelVolumes->put($newSource, [
 | |
|                                     'name' => $newSource,
 | |
|                                 ]);
 | |
|                             }
 | |
|                         }
 | |
| 
 | |
| 
 | |
|                         return $volume->value();
 | |
|                     });
 | |
|                     data_set($service, 'volumes', $serviceVolumes->toArray());
 | |
|                 }
 | |
|             } else {
 | |
|             }
 | |
|             // Decide if the service is a database
 | |
|             $isDatabase = isDatabaseImage(data_get_str($service, 'image'));
 | |
|             data_set($service, 'is_database', $isDatabase);
 | |
| 
 | |
|             // Collect/create/update networks
 | |
|             if ($serviceNetworks->count() > 0) {
 | |
|                 foreach ($serviceNetworks as $networkName => $networkDetails) {
 | |
|                     ray($networkDetails);
 | |
|                     $networkExists = $topLevelNetworks->contains(function ($value, $key) use ($networkName) {
 | |
|                         return $value == $networkName || $key == $networkName;
 | |
|                     });
 | |
|                     if (!$networkExists) {
 | |
|                         $topLevelNetworks->put($networkDetails, null);
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|             // Collect/create/update ports
 | |
|             $collectedPorts = collect([]);
 | |
|             if ($servicePorts->count() > 0) {
 | |
|                 foreach ($servicePorts as $sport) {
 | |
|                     if (is_string($sport) || is_numeric($sport)) {
 | |
|                         $collectedPorts->push($sport);
 | |
|                     }
 | |
|                     if (is_array($sport)) {
 | |
|                         $target = data_get($sport, 'target');
 | |
|                         $published = data_get($sport, 'published');
 | |
|                         $protocol = data_get($sport, 'protocol');
 | |
|                         $collectedPorts->push("$target:$published/$protocol");
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|             if ($collectedPorts->count() > 0) {
 | |
|                 // ray($collectedPorts->implode(','));
 | |
|             }
 | |
|             $definedNetworkExists = $topLevelNetworks->contains(function ($value, $_) use ($definedNetwork) {
 | |
|                 return $value == $definedNetwork;
 | |
|             });
 | |
|             if (!$definedNetworkExists) {
 | |
|                 foreach ($definedNetwork as $network) {
 | |
|                     if ($pull_request_id !== 0) {
 | |
|                         $topLevelNetworks->put($network,  [
 | |
|                             'name' => $network,
 | |
|                             'external' => true
 | |
|                         ]);
 | |
|                     } else {
 | |
|                         $topLevelNetworks->put($network,  [
 | |
|                             'name' => $network,
 | |
|                             'external' => true
 | |
|                         ]);
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|             $networks = collect();
 | |
|             foreach ($serviceNetworks as $key => $serviceNetwork) {
 | |
|                 if (gettype($serviceNetwork) === 'string') {
 | |
|                     // networks:
 | |
|                     //  - appwrite
 | |
|                     $networks->put($serviceNetwork, null);
 | |
|                 } else if (gettype($serviceNetwork) === 'array') {
 | |
|                     // networks:
 | |
|                     //   default:
 | |
|                     //     ipv4_address: 192.168.203.254
 | |
|                     // $networks->put($serviceNetwork, null);
 | |
|                     $networks->put($key, $serviceNetwork);
 | |
|                 }
 | |
|             }
 | |
|             foreach ($definedNetwork as $key => $network) {
 | |
|                 $networks->put($network, null);
 | |
|             }
 | |
|             data_set($service, 'networks', $networks->toArray());
 | |
|             // Get variables from the service
 | |
|             foreach ($serviceVariables as $variableName => $variable) {
 | |
|                 if (is_numeric($variableName)) {
 | |
|                     $variable = Str::of($variable);
 | |
|                     if ($variable->contains('=')) {
 | |
|                         // - SESSION_SECRET=123
 | |
|                         // - SESSION_SECRET=
 | |
|                         $key = $variable->before('=');
 | |
|                         $value = $variable->after('=');
 | |
|                     } else {
 | |
|                         // - SESSION_SECRET
 | |
|                         $key = $variable;
 | |
|                         $value = null;
 | |
|                     }
 | |
|                 } else {
 | |
|                     // SESSION_SECRET: 123
 | |
|                     // SESSION_SECRET:
 | |
|                     $key = Str::of($variableName);
 | |
|                     $value = Str::of($variable);
 | |
|                 }
 | |
|                 if ($key->startsWith('SERVICE_FQDN')) {
 | |
|                     if ($isNew) {
 | |
|                         $name = $key->after('SERVICE_FQDN_')->beforeLast('_')->lower();
 | |
|                         $fqdn = generateFqdn($server, "{$name->value()}-{$resource->uuid}");
 | |
|                         if (substr_count($key->value(), '_') === 3) {
 | |
|                             // SERVICE_FQDN_UMAMI_1000
 | |
|                             $port = $key->afterLast('_');
 | |
|                         } else {
 | |
|                             // SERVICE_FQDN_UMAMI
 | |
|                             $port = null;
 | |
|                         }
 | |
|                         if ($port) {
 | |
|                             $fqdn = "$fqdn:$port";
 | |
|                         }
 | |
|                         if (substr_count($key->value(), '_') >= 2) {
 | |
|                             if (is_null($value)) {
 | |
|                                 $value = Str::of('/');
 | |
|                             }
 | |
|                             $path = $value->value();
 | |
|                             if ($generatedServiceFQDNS->count() > 0) {
 | |
|                                 $alreadyGenerated = $generatedServiceFQDNS->has($key->value());
 | |
|                                 if ($alreadyGenerated) {
 | |
|                                     $fqdn = $generatedServiceFQDNS->get($key->value());
 | |
|                                 } else {
 | |
|                                     $generatedServiceFQDNS->put($key->value(), $fqdn);
 | |
|                                 }
 | |
|                             } else {
 | |
|                                 $generatedServiceFQDNS->put($key->value(), $fqdn);
 | |
|                             }
 | |
|                             $fqdn = "$fqdn$path";
 | |
|                         }
 | |
|                     }
 | |
|                     continue;
 | |
|                 }
 | |
|                 if ($value?->startsWith('$')) {
 | |
|                     $value = Str::of(replaceVariables($value));
 | |
|                     $key = $value;
 | |
|                     $foundEnv = EnvironmentVariable::where([
 | |
|                         'key' => $key,
 | |
|                         'application_id' => $resource->id,
 | |
|                     ])->first();
 | |
|                     if ($value->startsWith('SERVICE_')) {
 | |
|                         ['command' => $command, 'forService' => $forService, 'generatedValue' => $generatedValue, 'port' => $port] = parseEnvVariable($value);
 | |
|                         if ($command->value() === 'FQDN' || $command->value() === 'URL') {
 | |
|                             if (Str::lower($forService) === $serviceName) {
 | |
|                                 $fqdn = generateFqdn($server, $containerName);
 | |
|                             } else {
 | |
|                                 $fqdn = generateFqdn($server, Str::lower($forService) . '-' . $resource->uuid);
 | |
|                             }
 | |
|                             if ($port) {
 | |
|                                 $fqdn = "$fqdn:$port";
 | |
|                             }
 | |
|                             if ($foundEnv) {
 | |
|                                 $fqdn = data_get($foundEnv, 'value');
 | |
|                             } else {
 | |
|                                 if ($command->value() === 'URL') {
 | |
|                                     $fqdn = Str::of($fqdn)->after('://')->value();
 | |
|                                 }
 | |
|                                 EnvironmentVariable::create([
 | |
|                                     'key' => $key,
 | |
|                                     'value' => $fqdn,
 | |
|                                     'is_build_time' => false,
 | |
|                                     'application_id' => $resource->id,
 | |
|                                     'is_preview' => false,
 | |
|                                 ]);
 | |
|                             }
 | |
|                         } else {
 | |
|                             $generatedValue = generateEnvValue($command);
 | |
|                             if (!$foundEnv) {
 | |
|                                 EnvironmentVariable::create([
 | |
|                                     'key' => $key,
 | |
|                                     'value' => $generatedValue,
 | |
|                                     'is_build_time' => false,
 | |
|                                     'application_id' => $resource->id,
 | |
|                                     'is_preview' => false,
 | |
|                                 ]);
 | |
|                             }
 | |
|                         }
 | |
|                     } else {
 | |
|                         if ($value->contains(':-')) {
 | |
|                             $key = $value->before(':');
 | |
|                             $defaultValue = $value->after(':-');
 | |
|                         } else if ($value->contains('-')) {
 | |
|                             $key = $value->before('-');
 | |
|                             $defaultValue = $value->after('-');
 | |
|                         } else if ($value->contains(':?')) {
 | |
|                             $key = $value->before(':');
 | |
|                             $defaultValue = $value->after(':?');
 | |
|                         } else if ($value->contains('?')) {
 | |
|                             $key = $value->before('?');
 | |
|                             $defaultValue = $value->after('?');
 | |
|                         } else {
 | |
|                             $key = $value;
 | |
|                             $defaultValue = null;
 | |
|                         }
 | |
|                         if ($foundEnv) {
 | |
|                             $defaultValue = data_get($foundEnv, 'value');
 | |
|                         }
 | |
|                         $isBuildTime = data_get($foundEnv, 'is_build_time', false);
 | |
|                         EnvironmentVariable::updateOrCreate([
 | |
|                             'key' => $key->value(),
 | |
|                             'application_id' => $resource->id,
 | |
|                         ], [
 | |
|                             'value' => $defaultValue,
 | |
|                             'is_build_time' => $isBuildTime,
 | |
|                             'application_id' => $resource->id,
 | |
|                         ]);
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|             // Add labels to the service
 | |
|             if ($resource->serviceType()) {
 | |
|                 $fqdns = generateServiceSpecificFqdns($resource, forTraefik: true);
 | |
|             } else {
 | |
|                 $domains = collect(json_decode($resource->docker_compose_domains)) ?? [];
 | |
|                 if ($domains) {
 | |
|                     $fqdns = data_get($domains, "$serviceName.domain");
 | |
|                     if ($fqdns) {
 | |
|                         $fqdns = str($fqdns)->explode(',');
 | |
|                         $uuid = new Cuid2(7);
 | |
|                         if ($pull_request_id !== 0) {
 | |
|                             $fqdns = $fqdns->map(function ($fqdn) use ($pull_request_id, $resource) {
 | |
|                                 $preview = ApplicationPreview::findPreviewByApplicationAndPullId($resource->id, $pull_request_id);
 | |
|                                 $url = Url::fromString($fqdn);
 | |
|                                 $template = $resource->preview_url_template;
 | |
|                                 $host = $url->getHost();
 | |
|                                 $schema = $url->getScheme();
 | |
|                                 $random = new Cuid2(7);
 | |
|                                 $preview_fqdn = str_replace('{{random}}', $random, $template);
 | |
|                                 $preview_fqdn = str_replace('{{domain}}', $host, $preview_fqdn);
 | |
|                                 $preview_fqdn = str_replace('{{pr_id}}', $pull_request_id, $preview_fqdn);
 | |
|                                 $preview_fqdn = "$schema://$preview_fqdn";
 | |
|                                 $preview->fqdn = $preview_fqdn;
 | |
|                                 $preview->save();
 | |
|                                 return $preview_fqdn;
 | |
|                             });
 | |
|                         }
 | |
|                         $serviceLabels = $serviceLabels->merge(fqdnLabelsForTraefik($uuid, $fqdns));
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|             $defaultLabels = defaultLabels($resource->id, $containerName, $pull_request_id, type: 'application');
 | |
|             $serviceLabels = $serviceLabels->merge($defaultLabels);
 | |
| 
 | |
|             if ($server->isLogDrainEnabled() && $resource->isLogDrainEnabled()) {
 | |
|                 data_set($service, 'logging', [
 | |
|                     'driver' => 'fluentd',
 | |
|                     'options' => [
 | |
|                         'fluentd-address' => "tcp://127.0.0.1:24224",
 | |
|                         'fluentd-async' => "true",
 | |
|                         'fluentd-sub-second-precision' => "true",
 | |
|                     ]
 | |
|                 ]);
 | |
|             }
 | |
|             data_set($service, 'labels', $serviceLabels->toArray());
 | |
|             data_forget($service, 'is_database');
 | |
|             data_set($service, 'restart', RESTART_MODE);
 | |
|             data_set($service, 'container_name', $containerName);
 | |
|             data_forget($service, 'volumes.*.content');
 | |
|             data_forget($service, 'volumes.*.isDirectory');
 | |
| 
 | |
|             return $service;
 | |
|         });
 | |
|         $finalServices = [
 | |
|             'version' => $dockerComposeVersion,
 | |
|             'services' => $services->toArray(),
 | |
|             'volumes' => $topLevelVolumes->toArray(),
 | |
|             'networks' => $topLevelNetworks->toArray(),
 | |
|         ];
 | |
|         if ($isSameDockerComposeFile) {
 | |
|             $resource->docker_compose_pr_raw = Yaml::dump($yaml, 10, 2);
 | |
|             $resource->docker_compose_pr = Yaml::dump($finalServices, 10, 2);
 | |
|             $resource->docker_compose_raw = Yaml::dump($yaml, 10, 2);
 | |
|             $resource->docker_compose = Yaml::dump($finalServices, 10, 2);
 | |
|         } else {
 | |
|             if ($is_pr) {
 | |
|                 $resource->docker_compose_pr_raw = Yaml::dump($yaml, 10, 2);
 | |
|                 $resource->docker_compose_pr = Yaml::dump($finalServices, 10, 2);
 | |
|             } else {
 | |
|                 $resource->docker_compose_raw = Yaml::dump($yaml, 10, 2);
 | |
|                 $resource->docker_compose = Yaml::dump($finalServices, 10, 2);
 | |
|             }
 | |
|         }
 | |
|         $resource->save();
 | |
|         return collect($finalServices);
 | |
|     }
 | |
| }
 | |
| 
 | |
| function parseEnvVariable(Str|string $value)
 | |
| {
 | |
|     $value = str($value);
 | |
|     $count = substr_count($value->value(), '_');
 | |
|     $command = null;
 | |
|     $forService = null;
 | |
|     $generatedValue = null;
 | |
|     $port = null;
 | |
| 
 | |
|     if ($count === 2) {
 | |
|         if ($value->startsWith('SERVICE_FQDN') || $value->startsWith('SERVICE_URL')) {
 | |
|             // SERVICE_FQDN_UMAMI
 | |
|             $command = $value->after('SERVICE_')->beforeLast('_');
 | |
|             $forService = $value->afterLast('_');
 | |
|         } else {
 | |
|             // SERVICE_BASE64_UMAMI
 | |
|             $command = $value->after('SERVICE_')->beforeLast('_');
 | |
|         }
 | |
|     }
 | |
|     if ($count === 3) {
 | |
|         if ($value->startsWith('SERVICE_FQDN') || $value->startsWith('SERVICE_URL')) {
 | |
|             // SERVICE_FQDN_UMAMI_1000
 | |
|             $command = $value->after('SERVICE_')->before('_');
 | |
|             $forService = $value->after('SERVICE_')->after('_')->before('_');
 | |
|             $port = $value->afterLast('_');
 | |
|         } else {
 | |
|             // SERVICE_BASE64_64_UMAMI
 | |
|             $command = $value->after('SERVICE_')->beforeLast('_');
 | |
|         }
 | |
|     }
 | |
|     return [
 | |
|         'command' => $command,
 | |
|         'forService' => $forService,
 | |
|         'generatedValue' => $generatedValue,
 | |
|         'port' => $port,
 | |
|     ];
 | |
| }
 | |
| function generateEnvValue(string $command)
 | |
| {
 | |
|     switch ($command) {
 | |
|         case 'PASSWORD':
 | |
|             $generatedValue = Str::password(symbols: false);
 | |
|             break;
 | |
|         case 'PASSWORD_64':
 | |
|             $generatedValue = Str::password(length: 64, symbols: false);
 | |
|             break;
 | |
|         case 'BASE64_64':
 | |
|             $generatedValue = Str::random(64);
 | |
|             break;
 | |
|         case 'BASE64_128':
 | |
|             $generatedValue = Str::random(128);
 | |
|             break;
 | |
|         case 'BASE64':
 | |
|             $generatedValue = Str::random(32);
 | |
|             break;
 | |
|         case 'USER':
 | |
|             $generatedValue = Str::random(16);
 | |
|             break;
 | |
|     }
 | |
|     return $generatedValue;
 | |
| }
 | 
