This commit is contained in:
Andras Bacsai
2023-06-22 14:18:17 +02:00
parent 3af354b343
commit 448fa200a4
5 changed files with 166 additions and 82 deletions

View File

@@ -15,8 +15,13 @@ class Proxy extends Component
public ProxyTypes $selectedProxy = ProxyTypes::TRAEFIK_V2; public ProxyTypes $selectedProxy = ProxyTypes::TRAEFIK_V2;
public $proxy_settings = null; public $proxy_settings = null;
public string|null $redirect_url = null;
protected $listeners = ['serverValidated', 'saveConfiguration']; protected $listeners = ['serverValidated', 'saveConfiguration'];
public function mount()
{
$this->redirect_url = $this->server->proxy->redirect_url;
}
public function serverValidated() public function serverValidated()
{ {
$this->server->refresh(); $this->server->refresh();
@@ -52,17 +57,20 @@ class Proxy extends Component
$this->server->proxy->status = 'exited'; $this->server->proxy->status = 'exited';
$this->server->save(); $this->server->save();
} }
public function saveConfiguration(Server $server) public function saveConfiguration()
{ {
try { try {
$proxy_path = config('coolify.proxy_config_path'); $proxy_path = config('coolify.proxy_config_path');
$this->proxy_settings = Str::of($this->proxy_settings)->trim()->value; $this->proxy_settings = Str::of($this->proxy_settings)->trim()->value;
$docker_compose_yml_base64 = base64_encode($this->proxy_settings); $docker_compose_yml_base64 = base64_encode($this->proxy_settings);
$server->proxy->last_saved_settings = Str::of($docker_compose_yml_base64)->pipe('md5')->value; $this->server->proxy->last_saved_settings = Str::of($docker_compose_yml_base64)->pipe('md5')->value;
$server->save(); $this->server->proxy->redirect_url = $this->redirect_url;
$this->server->save();
instant_remote_process([ instant_remote_process([
"echo '$docker_compose_yml_base64' | base64 -d > $proxy_path/docker-compose.yml", "echo '$docker_compose_yml_base64' | base64 -d > $proxy_path/docker-compose.yml",
], $server); ], $this->server);
$this->server->refresh();
setup_default_redirect_404(redirect_url: $this->server->proxy->redirect_url, server: $this->server);
$this->emit('success', 'Proxy configuration saved.'); $this->emit('success', 'Proxy configuration saved.');
} catch (\Exception $e) { } catch (\Exception $e) {
return general_error_handler(err: $e); return general_error_handler(err: $e);

View File

@@ -15,7 +15,7 @@ class Configuration extends Component
public $do_not_track; public $do_not_track;
public $is_auto_update_enabled; public $is_auto_update_enabled;
public $is_registration_enabled; public $is_registration_enabled;
protected string $dynamic_config_path; protected string $dynamic_config_path = '/data/coolify/proxy/dynamic';
protected Server $server; protected Server $server;
protected $rules = [ protected $rules = [
@@ -117,9 +117,6 @@ class Configuration extends Component
"rm -f $file", "rm -f $file",
], $this->server); ], $this->server);
} else { } else {
$url = Url::fromString($this->settings->default_redirect_404);
$host = $url->getHost();
$schema = $url->getScheme();
$traefik_dynamic_conf = [ $traefik_dynamic_conf = [
'http' => 'http' =>
[ [
@@ -200,10 +197,9 @@ class Configuration extends Component
$this->validate(); $this->validate();
$this->settings->save(); $this->settings->save();
$this->dynamic_config_path = '/data/coolify/proxy/dynamic';
$this->server = Server::findOrFail(0); $this->server = Server::findOrFail(0);
$this->setup_instance_fqdn(); $this->setup_instance_fqdn();
$this->setup_default_redirect_404(); setup_default_redirect_404(redirect_url: $this->settings->default_redirect_404, server: $this->server);
if ($this->settings->fqdn || $this->settings->default_redirect_404) { if ($this->settings->fqdn || $this->settings->default_redirect_404) {
dispatch(new InstanceProxyCheckJob()); dispatch(new InstanceProxyCheckJob());
} }

View File

@@ -87,6 +87,7 @@ class Server extends BaseModel
static public function ownedByCurrentTeam(array $select = ['*']) static public function ownedByCurrentTeam(array $select = ['*'])
{ {
$selectArray = collect($select)->concat(['id']); $selectArray = collect($select)->concat(['id']);
ray(Server::whereTeamId(session('currentTeam')->id)->with('settings')->select($selectArray->all())->get());
return Server::whereTeamId(session('currentTeam')->id)->with('settings')->select($selectArray->all()); return Server::whereTeamId(session('currentTeam')->id)->with('settings')->select($selectArray->all());
} }

View File

@@ -1,85 +1,160 @@
<?php <?php
use App\Models\Server; use App\Models\Server;
use Spatie\Url\Url;
use Symfony\Component\Yaml\Yaml; use Symfony\Component\Yaml\Yaml;
if (!function_exists('getProxyConfiguration')) { function getProxyConfiguration(Server $server)
function getProxyConfiguration(Server $server) {
{ $proxy_path = config('coolify.proxy_config_path');
$proxy_path = config('coolify.proxy_config_path'); if (isDev()) {
if (isDev()) { $proxy_path = $proxy_path . '/testing-host-1/';
$proxy_path = $proxy_path . '/testing-host-1/'; }
} $networks = collect($server->standaloneDockers)->map(function ($docker) {
$networks = collect($server->standaloneDockers)->map(function ($docker) { return $docker['network'];
return $docker['network']; })->unique();
})->unique(); if ($networks->count() === 0) {
if ($networks->count() === 0) { $networks = collect(['coolify']);
$networks = collect(['coolify']); }
} $array_of_networks = collect([]);
$array_of_networks = collect([]); $networks->map(function ($network) use ($array_of_networks) {
$networks->map(function ($network) use ($array_of_networks) { $array_of_networks[$network] = [
$array_of_networks[$network] = [ "external" => true,
"external" => true, ];
]; });
}); $config = [
$config = [ "version" => "3.8",
"version" => "3.8", "networks" => $array_of_networks->toArray(),
"networks" => $array_of_networks->toArray(), "services" => [
"services" => [ "traefik" => [
"traefik" => [ "container_name" => "coolify-proxy",
"container_name" => "coolify-proxy", "image" => "traefik:v2.10",
"image" => "traefik:v2.10", "restart" => "always",
"restart" => "always", "extra_hosts" => [
"extra_hosts" => [ "host.docker.internal:host-gateway",
"host.docker.internal:host-gateway", ],
"networks" => $networks->toArray(),
"ports" => [
"80:80",
"443:443",
"8080:8080",
],
"healthcheck" => [
"test" => "wget -qO- http://localhost:80/ping || exit 1",
"interval" => "4s",
"timeout" => "2s",
"retries" => 5,
],
"volumes" => [
"/var/run/docker.sock:/var/run/docker.sock:ro",
"{$proxy_path}:/traefik",
],
"command" => [
"--ping=true",
"--ping.entrypoint=http",
"--api.dashboard=true",
"--api.insecure=true",
"--entrypoints.http.address=:80",
"--entrypoints.https.address=:443",
"--providers.docker=true",
"--providers.docker.exposedbydefault=false",
"--providers.file.directory=/traefik/dynamic/",
"--providers.file.watch=true",
"--certificatesresolvers.letsencrypt.acme.httpchallenge=true",
"--certificatesresolvers.letsencrypt.acme.storage=/traefik/acme.json",
"--certificatesresolvers.letsencrypt.acme.httpchallenge.entrypoint=http",
],
"labels" => [
"traefik.enable=true",
"traefik.http.routers.traefik.entrypoints=http",
"traefik.http.routers.traefik.middlewares=traefik-basic-auth@file",
"traefik.http.routers.traefik.service=api@internal",
"traefik.http.services.traefik.loadbalancer.server.port=8080",
// Global Middlewares
"traefik.http.middlewares.redirect-to-https.redirectscheme.scheme=https",
"traefik.http.middlewares.gzip.compress=true",
],
],
],
];
if (isDev()) {
$config['services']['traefik']['command'][] = "--log.level=debug";
}
return Yaml::dump($config, 4, 2);
}
function setup_default_redirect_404(string $redirect_url, Server $server)
{
$traefik_dynamic_conf_path = '/data/coolify/proxy/dynamic';
$traefik_default_redirect_file = "$traefik_dynamic_conf_path/default_redirect_404.yaml";
if (empty($redirect_url)) {
remote_process([
"rm -f $traefik_default_redirect_file",
], $server);
} else {
$traefik_dynamic_conf = [
'http' =>
[
'routers' =>
[
'catchall' =>
[
'entryPoints' => [
0 => 'http',
1 => 'https',
],
'service' => 'noop',
'rule' => "HostRegexp(`{catchall:.*}`)",
'priority' => 1,
'middlewares' => [
0 => 'redirect-regexp@file',
],
], ],
"networks" => $networks->toArray(), ],
"ports" => [ 'services' =>
"80:80", [
"443:443", 'noop' =>
"8080:8080", [
'loadBalancer' =>
[
'servers' =>
[
0 =>
[
'url' => '',
],
],
],
], ],
"healthcheck" => [ ],
"test" => "wget -qO- http://localhost:80/ping || exit 1", 'middlewares' =>
"interval" => "4s", [
"timeout" => "2s", 'redirect-regexp' =>
"retries" => 5, [
], 'redirectRegex' =>
"volumes" => [ [
"/var/run/docker.sock:/var/run/docker.sock:ro", 'regex' => '(.*)',
"{$proxy_path}:/traefik", 'replacement' => $redirect_url,
], 'permanent' => false,
"command" => [ ],
"--ping=true",
"--ping.entrypoint=http",
"--api.dashboard=true",
"--api.insecure=true",
"--entrypoints.http.address=:80",
"--entrypoints.https.address=:443",
"--providers.docker=true",
"--providers.docker.exposedbydefault=false",
"--providers.file.directory=/traefik/dynamic/",
"--providers.file.watch=true",
"--certificatesresolvers.letsencrypt.acme.httpchallenge=true",
"--certificatesresolvers.letsencrypt.acme.storage=/traefik/acme.json",
"--certificatesresolvers.letsencrypt.acme.httpchallenge.entrypoint=http",
],
"labels" => [
"traefik.enable=true",
"traefik.http.routers.traefik.entrypoints=http",
"traefik.http.routers.traefik.middlewares=traefik-basic-auth@file",
"traefik.http.routers.traefik.service=api@internal",
"traefik.http.services.traefik.loadbalancer.server.port=8080",
// Global Middlewares
"traefik.http.middlewares.redirect-to-https.redirectscheme.scheme=https",
"traefik.http.middlewares.gzip.compress=true",
], ],
], ],
], ],
]; ];
if (isDev()) { $yaml = Yaml::dump($traefik_dynamic_conf, 12, 2);
$config['services']['traefik']['command'][] = "--log.level=debug"; $yaml =
"# This file is automatically generated by Coolify.\n" .
"# Do not edit it manually (only if you know what are you doing).\n\n" .
$yaml;
$base64 = base64_encode($yaml);
ray("mkdir -p $traefik_dynamic_conf_path");
remote_process([
"mkdir -p $traefik_dynamic_conf_path",
"echo '$base64' | base64 -d > $traefik_default_redirect_file",
], $server);
if (config('app.env') == 'local') {
ray($yaml);
} }
return Yaml::dump($config, 4, 2);
} }
} }

View File

@@ -9,7 +9,7 @@
</div> </div>
@isset($proxy_settings) @isset($proxy_settings)
@if ($selectedProxy->value === 'TRAEFIK_V2') @if ($selectedProxy->value === 'TRAEFIK_V2')
<form wire:submit.prevent='saveConfiguration({{ $server }})'> <form wire:submit.prevent='saveConfiguration'>
<div class="flex items-center gap-2"> <div class="flex items-center gap-2">
<h2>Proxy</h2> <h2>Proxy</h2>
<x-forms.button type="submit">Save</x-forms.button> <x-forms.button type="submit">Save</x-forms.button>
@@ -25,6 +25,10 @@
<div class="text-red-500 ">Configuration out of sync. Restart to get the new configs. <div class="text-red-500 ">Configuration out of sync. Restart to get the new configs.
</div> </div>
@endif @endif
@if ($server->id !== 0)
<x-forms.input id="redirect_url" label="Default redirect"
placeholder="https://coolify.io" />
@endif
<div class="container w-full mx-auto"> <div class="container w-full mx-auto">
<livewire:activity-monitor :header="true" /> <livewire:activity-monitor :header="true" />
</div> </div>