@@ -3,6 +3,8 @@
 | 
			
		||||
namespace App\Actions\Application;
 | 
			
		||||
 | 
			
		||||
use App\Models\Application;
 | 
			
		||||
use App\Models\StandaloneDocker;
 | 
			
		||||
use App\Notifications\Application\StatusChanged;
 | 
			
		||||
use Lorisleiva\Actions\Concerns\AsAction;
 | 
			
		||||
 | 
			
		||||
class StopApplication
 | 
			
		||||
@@ -10,13 +12,20 @@ class StopApplication
 | 
			
		||||
    use AsAction;
 | 
			
		||||
    public function handle(Application $application)
 | 
			
		||||
    {
 | 
			
		||||
        $server = $application->destination->server;
 | 
			
		||||
        if (!$server->isFunctional()) {
 | 
			
		||||
            return 'Server is not functional';
 | 
			
		||||
        if ($application->destination->server->isSwarm()) {
 | 
			
		||||
            instant_remote_process(["docker stack rm {$application->uuid}"], $application->destination->server);
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
        if ($server->isSwarm()) {
 | 
			
		||||
            instant_remote_process(["docker stack rm {$application->uuid}" ], $server);
 | 
			
		||||
        } else {
 | 
			
		||||
 | 
			
		||||
        $servers = collect([]);
 | 
			
		||||
        $servers->push($application->destination->server);
 | 
			
		||||
        $application->additional_servers->map(function ($server) use ($servers) {
 | 
			
		||||
            $servers->push($server);
 | 
			
		||||
        });
 | 
			
		||||
        foreach ($servers as $server) {
 | 
			
		||||
            if (!$server->isFunctional()) {
 | 
			
		||||
                return 'Server is not functional';
 | 
			
		||||
            }
 | 
			
		||||
            $containers = getCurrentApplicationContainerStatus($server, $application->id, 0);
 | 
			
		||||
            if ($containers->count() > 0) {
 | 
			
		||||
                foreach ($containers as $container) {
 | 
			
		||||
@@ -28,20 +37,7 @@ class StopApplication
 | 
			
		||||
                        );
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
                // TODO: make notification for application
 | 
			
		||||
                // $application->environment->project->team->notify(new StatusChanged($application));
 | 
			
		||||
            }
 | 
			
		||||
            // Delete Preview Deployments
 | 
			
		||||
            $previewDeployments = $application->previews;
 | 
			
		||||
            foreach ($previewDeployments as $previewDeployment) {
 | 
			
		||||
                $containers = getCurrentApplicationContainerStatus($server, $application->id, $previewDeployment->pull_request_id);
 | 
			
		||||
                foreach ($containers as $container) {
 | 
			
		||||
                    $name = str_replace('/', '', $container['Names']);
 | 
			
		||||
                    instant_remote_process(["docker rm -f $name"], $application->destination->server, throwError: false);
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										38
									
								
								app/Actions/Application/StopApplicationOneServer.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								app/Actions/Application/StopApplicationOneServer.php
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,38 @@
 | 
			
		||||
<?php
 | 
			
		||||
 | 
			
		||||
namespace App\Actions\Application;
 | 
			
		||||
 | 
			
		||||
use App\Models\Application;
 | 
			
		||||
use App\Models\Server;
 | 
			
		||||
use Lorisleiva\Actions\Concerns\AsAction;
 | 
			
		||||
 | 
			
		||||
class StopApplicationOneServer
 | 
			
		||||
{
 | 
			
		||||
    use AsAction;
 | 
			
		||||
    public function handle(Application $application, Server $server)
 | 
			
		||||
    {
 | 
			
		||||
        if ($application->destination->server->isSwarm()) {
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
        if (!$server->isFunctional()) {
 | 
			
		||||
            return 'Server is not functional';
 | 
			
		||||
        }
 | 
			
		||||
        try {
 | 
			
		||||
            $containers = getCurrentApplicationContainerStatus($server, $application->id, 0);
 | 
			
		||||
            if ($containers->count() > 0) {
 | 
			
		||||
                foreach ($containers as $container) {
 | 
			
		||||
                    $containerName = data_get($container, 'Names');
 | 
			
		||||
                    if ($containerName) {
 | 
			
		||||
                        instant_remote_process(
 | 
			
		||||
                            ["docker rm -f {$containerName}"],
 | 
			
		||||
                            $server
 | 
			
		||||
                        );
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        } catch (\Exception $e) {
 | 
			
		||||
            ray($e->getMessage());
 | 
			
		||||
            return $e->getMessage();
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -10,24 +10,31 @@ class StopService
 | 
			
		||||
    use AsAction;
 | 
			
		||||
    public function handle(Service $service)
 | 
			
		||||
    {
 | 
			
		||||
        $server = $service->destination->server;
 | 
			
		||||
        if (!$server->isFunctional()) {
 | 
			
		||||
            return 'Server is not functional';
 | 
			
		||||
        try {
 | 
			
		||||
            $server = $service->destination->server;
 | 
			
		||||
            if (!$server->isFunctional()) {
 | 
			
		||||
                return 'Server is not functional';
 | 
			
		||||
            }
 | 
			
		||||
            ray('Stopping service: ' . $service->name);
 | 
			
		||||
            $applications = $service->applications()->get();
 | 
			
		||||
            foreach ($applications as $application) {
 | 
			
		||||
                instant_remote_process(["docker rm -f {$application->name}-{$service->uuid}"], $service->server);
 | 
			
		||||
                $application->update(['status' => 'exited']);
 | 
			
		||||
            }
 | 
			
		||||
            $dbs = $service->databases()->get();
 | 
			
		||||
            foreach ($dbs as $db) {
 | 
			
		||||
                instant_remote_process(["docker rm -f {$db->name}-{$service->uuid}"], $service->server);
 | 
			
		||||
                $db->update(['status' => 'exited']);
 | 
			
		||||
            }
 | 
			
		||||
            instant_remote_process(["docker network disconnect {$service->uuid} coolify-proxy 2>/dev/null"], $service->server, false);
 | 
			
		||||
            instant_remote_process(["docker network rm {$service->uuid} 2>/dev/null"], $service->server, false);
 | 
			
		||||
            // TODO: make notification for databases
 | 
			
		||||
            // $service->environment->project->team->notify(new StatusChanged($service));
 | 
			
		||||
        } catch (\Exception $e) {
 | 
			
		||||
            echo $e->getMessage();
 | 
			
		||||
            ray($e->getMessage());
 | 
			
		||||
            return $e->getMessage();
 | 
			
		||||
        }
 | 
			
		||||
        ray('Stopping service: ' . $service->name);
 | 
			
		||||
        $applications = $service->applications()->get();
 | 
			
		||||
        foreach ($applications as $application) {
 | 
			
		||||
            instant_remote_process(["docker rm -f {$application->name}-{$service->uuid}"], $service->server);
 | 
			
		||||
            $application->update(['status' => 'exited']);
 | 
			
		||||
        }
 | 
			
		||||
        $dbs = $service->databases()->get();
 | 
			
		||||
        foreach ($dbs as $db) {
 | 
			
		||||
            instant_remote_process(["docker rm -f {$db->name}-{$service->uuid}"], $service->server);
 | 
			
		||||
            $db->update(['status' => 'exited']);
 | 
			
		||||
        }
 | 
			
		||||
        instant_remote_process(["docker network disconnect {$service->uuid} coolify-proxy 2>/dev/null"], $service->server, false);
 | 
			
		||||
        instant_remote_process(["docker network rm {$service->uuid} 2>/dev/null"], $service->server, false);
 | 
			
		||||
        // TODO: make notification for databases
 | 
			
		||||
        // $service->environment->project->team->notify(new StatusChanged($service));
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										55
									
								
								app/Actions/Shared/ComplexStatusCheck.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								app/Actions/Shared/ComplexStatusCheck.php
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,55 @@
 | 
			
		||||
<?php
 | 
			
		||||
 | 
			
		||||
namespace App\Actions\Shared;
 | 
			
		||||
 | 
			
		||||
use App\Models\Application;
 | 
			
		||||
use Lorisleiva\Actions\Concerns\AsAction;
 | 
			
		||||
 | 
			
		||||
class ComplexStatusCheck
 | 
			
		||||
{
 | 
			
		||||
    use AsAction;
 | 
			
		||||
    public function handle(Application $application)
 | 
			
		||||
    {
 | 
			
		||||
        $servers = $application->additional_servers;
 | 
			
		||||
        $servers->push($application->destination->server);
 | 
			
		||||
        foreach ($servers as $server) {
 | 
			
		||||
            $is_main_server = $application->destination->server->id === $server->id;
 | 
			
		||||
            if (!$server->isFunctional()) {
 | 
			
		||||
                if ($is_main_server) {
 | 
			
		||||
                    $application->update(['status' => 'exited:unhealthy']);
 | 
			
		||||
                    continue;
 | 
			
		||||
                } else {
 | 
			
		||||
                    $application->additional_servers()->updateExistingPivot($server->id, ['status' => 'exited:unhealthy']);
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            $container = instant_remote_process(["docker container inspect $(docker container ls -q --filter 'label=coolify.applicationId={$application->id}' --filter 'label=coolify.pullRequestId=0') --format '{{json .}}'"], $server, false);
 | 
			
		||||
            $container = format_docker_command_output_to_json($container);
 | 
			
		||||
            if ($container->count() === 1) {
 | 
			
		||||
                $container = $container->first();
 | 
			
		||||
                $containerStatus = data_get($container, 'State.Status');
 | 
			
		||||
                $containerHealth = data_get($container, 'State.Health.Status', 'unhealthy');
 | 
			
		||||
                if ($is_main_server) {
 | 
			
		||||
                    $statusFromDb = $application->status;
 | 
			
		||||
                    if ($statusFromDb !== $containerStatus) {
 | 
			
		||||
                        $application->update(['status' => "$containerStatus:$containerHealth"]);
 | 
			
		||||
                    }
 | 
			
		||||
                } else {
 | 
			
		||||
                    $additional_server = $application->additional_servers()->wherePivot('server_id', $server->id);
 | 
			
		||||
                    $statusFromDb = $additional_server->first()->pivot->status;
 | 
			
		||||
                    if ($statusFromDb !== $containerStatus) {
 | 
			
		||||
                        $additional_server->updateExistingPivot($server->id, ['status' => "$containerStatus:$containerHealth"]);
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            } else {
 | 
			
		||||
                if ($is_main_server) {
 | 
			
		||||
                    $application->update(['status' => 'exited:unhealthy']);
 | 
			
		||||
                    continue;
 | 
			
		||||
                } else {
 | 
			
		||||
                    $application->additional_servers()->updateExistingPivot($server->id, ['status' => 'exited:unhealthy']);
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -20,7 +20,8 @@ class CleanupStuckedResources extends Command
 | 
			
		||||
 | 
			
		||||
    public function handle()
 | 
			
		||||
    {
 | 
			
		||||
        echo "Running cleanup stucked...\n";
 | 
			
		||||
        ray('Running cleanup stucked resources.');
 | 
			
		||||
        echo "Running cleanup stucked resources.\n";
 | 
			
		||||
        $this->cleanup_stucked_resources();
 | 
			
		||||
    }
 | 
			
		||||
    private function cleanup_stucked_resources()
 | 
			
		||||
@@ -113,18 +114,18 @@ class CleanupStuckedResources extends Command
 | 
			
		||||
            $applications = Application::all();
 | 
			
		||||
            foreach ($applications as $application) {
 | 
			
		||||
                if (!data_get($application, 'environment')) {
 | 
			
		||||
                    echo 'Application without environment: ' . $application->name . ' soft deleting\n';
 | 
			
		||||
                    $application->delete();
 | 
			
		||||
                    echo 'Application without environment: ' . $application->name . '\n';
 | 
			
		||||
                    $application->forceDelete();
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
                if (!$application->destination()) {
 | 
			
		||||
                    echo 'Application without destination: ' . $application->name . ' soft deleting\n';
 | 
			
		||||
                    $application->delete();
 | 
			
		||||
                    echo 'Application without destination: ' . $application->name . '\n';
 | 
			
		||||
                    $application->forceDelete();
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
                if (!data_get($application, 'destination.server')) {
 | 
			
		||||
                    echo 'Application without server: ' . $application->name . ' soft deleting\n';
 | 
			
		||||
                    $application->delete();
 | 
			
		||||
                    echo 'Application without server: ' . $application->name . '\n';
 | 
			
		||||
                    $application->forceDelete();
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
@@ -135,18 +136,18 @@ class CleanupStuckedResources extends Command
 | 
			
		||||
            $postgresqls = StandalonePostgresql::all()->where('id', '!=', 0);
 | 
			
		||||
            foreach ($postgresqls as $postgresql) {
 | 
			
		||||
                if (!data_get($postgresql, 'environment')) {
 | 
			
		||||
                    echo 'Postgresql without environment: ' . $postgresql->name . ' soft deleting\n';
 | 
			
		||||
                    $postgresql->delete();
 | 
			
		||||
                    echo 'Postgresql without environment: ' . $postgresql->name . '\n';
 | 
			
		||||
                    $postgresql->forceDelete();
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
                if (!$postgresql->destination()) {
 | 
			
		||||
                    echo 'Postgresql without destination: ' . $postgresql->name . ' soft deleting\n';
 | 
			
		||||
                    $postgresql->delete();
 | 
			
		||||
                    echo 'Postgresql without destination: ' . $postgresql->name . '\n';
 | 
			
		||||
                    $postgresql->forceDelete();
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
                if (!data_get($postgresql, 'destination.server')) {
 | 
			
		||||
                    echo 'Postgresql without server: ' . $postgresql->name . ' soft deleting\n';
 | 
			
		||||
                    $postgresql->delete();
 | 
			
		||||
                    echo 'Postgresql without server: ' . $postgresql->name . '\n';
 | 
			
		||||
                    $postgresql->forceDelete();
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
@@ -157,18 +158,18 @@ class CleanupStuckedResources extends Command
 | 
			
		||||
            $redis = StandaloneRedis::all();
 | 
			
		||||
            foreach ($redis as $redis) {
 | 
			
		||||
                if (!data_get($redis, 'environment')) {
 | 
			
		||||
                    echo 'Redis without environment: ' . $redis->name . ' soft deleting\n';
 | 
			
		||||
                    $redis->delete();
 | 
			
		||||
                    echo 'Redis without environment: ' . $redis->name . '\n';
 | 
			
		||||
                    $redis->forceDelete();
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
                if (!$redis->destination()) {
 | 
			
		||||
                    echo 'Redis without destination: ' . $redis->name . ' soft deleting\n';
 | 
			
		||||
                    $redis->delete();
 | 
			
		||||
                    echo 'Redis without destination: ' . $redis->name . '\n';
 | 
			
		||||
                    $redis->forceDelete();
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
                if (!data_get($redis, 'destination.server')) {
 | 
			
		||||
                    echo 'Redis without server: ' . $redis->name . ' soft deleting\n';
 | 
			
		||||
                    $redis->delete();
 | 
			
		||||
                    echo 'Redis without server: ' . $redis->name . '\n';
 | 
			
		||||
                    $redis->forceDelete();
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
@@ -180,18 +181,18 @@ class CleanupStuckedResources extends Command
 | 
			
		||||
            $mongodbs = StandaloneMongodb::all();
 | 
			
		||||
            foreach ($mongodbs as $mongodb) {
 | 
			
		||||
                if (!data_get($mongodb, 'environment')) {
 | 
			
		||||
                    echo 'Mongodb without environment: ' . $mongodb->name . ' soft deleting\n';
 | 
			
		||||
                    $mongodb->delete();
 | 
			
		||||
                    echo 'Mongodb without environment: ' . $mongodb->name . '\n';
 | 
			
		||||
                    $mongodb->forceDelete();
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
                if (!$mongodb->destination()) {
 | 
			
		||||
                    echo 'Mongodb without destination: ' . $mongodb->name . ' soft deleting\n';
 | 
			
		||||
                    $mongodb->delete();
 | 
			
		||||
                    echo 'Mongodb without destination: ' . $mongodb->name . '\n';
 | 
			
		||||
                    $mongodb->forceDelete();
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
                if (!data_get($mongodb, 'destination.server')) {
 | 
			
		||||
                    echo 'Mongodb without server:  ' . $mongodb->name . ' soft deleting\n';
 | 
			
		||||
                    $mongodb->delete();
 | 
			
		||||
                    echo 'Mongodb without server:  ' . $mongodb->name . '\n';
 | 
			
		||||
                    $mongodb->forceDelete();
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
@@ -203,18 +204,18 @@ class CleanupStuckedResources extends Command
 | 
			
		||||
            $mysqls = StandaloneMysql::all();
 | 
			
		||||
            foreach ($mysqls as $mysql) {
 | 
			
		||||
                if (!data_get($mysql, 'environment')) {
 | 
			
		||||
                    echo 'Mysql without environment: ' . $mysql->name . ' soft deleting\n';
 | 
			
		||||
                    $mysql->delete();
 | 
			
		||||
                    echo 'Mysql without environment: ' . $mysql->name . '\n';
 | 
			
		||||
                    $mysql->forceDelete();
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
                if (!$mysql->destination()) {
 | 
			
		||||
                    echo 'Mysql without destination: ' . $mysql->name . ' soft deleting\n';
 | 
			
		||||
                    $mysql->delete();
 | 
			
		||||
                    echo 'Mysql without destination: ' . $mysql->name . '\n';
 | 
			
		||||
                    $mysql->forceDelete();
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
                if (!data_get($mysql, 'destination.server')) {
 | 
			
		||||
                    echo 'Mysql without server: ' . $mysql->name . ' soft deleting\n';
 | 
			
		||||
                    $mysql->delete();
 | 
			
		||||
                    echo 'Mysql without server: ' . $mysql->name . '\n';
 | 
			
		||||
                    $mysql->forceDelete();
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
@@ -226,18 +227,18 @@ class CleanupStuckedResources extends Command
 | 
			
		||||
            $mariadbs = StandaloneMariadb::all();
 | 
			
		||||
            foreach ($mariadbs as $mariadb) {
 | 
			
		||||
                if (!data_get($mariadb, 'environment')) {
 | 
			
		||||
                    echo 'Mariadb without environment: ' . $mariadb->name . ' soft deleting\n';
 | 
			
		||||
                    $mariadb->delete();
 | 
			
		||||
                    echo 'Mariadb without environment: ' . $mariadb->name . '\n';
 | 
			
		||||
                    $mariadb->forceDelete();
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
                if (!$mariadb->destination()) {
 | 
			
		||||
                    echo 'Mariadb without destination: ' . $mariadb->name . ' soft deleting\n';
 | 
			
		||||
                    $mariadb->delete();
 | 
			
		||||
                    echo 'Mariadb without destination: ' . $mariadb->name . '\n';
 | 
			
		||||
                    $mariadb->forceDelete();
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
                if (!data_get($mariadb, 'destination.server')) {
 | 
			
		||||
                    echo 'Mariadb without server: ' . $mariadb->name . ' soft deleting\n';
 | 
			
		||||
                    $mariadb->delete();
 | 
			
		||||
                    echo 'Mariadb without server: ' . $mariadb->name . '\n';
 | 
			
		||||
                    $mariadb->forceDelete();
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
@@ -249,18 +250,18 @@ class CleanupStuckedResources extends Command
 | 
			
		||||
            $services = Service::all();
 | 
			
		||||
            foreach ($services as $service) {
 | 
			
		||||
                if (!data_get($service, 'environment')) {
 | 
			
		||||
                    echo 'Service without environment: ' . $service->name . ' soft deleting\n';
 | 
			
		||||
                    $service->delete();
 | 
			
		||||
                    echo 'Service without environment: ' . $service->name . '\n';
 | 
			
		||||
                    $service->forceDelete();
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
                if (!$service->destination()) {
 | 
			
		||||
                    echo 'Service without destination: ' . $service->name . ' soft deleting\n';
 | 
			
		||||
                    $service->delete();
 | 
			
		||||
                    echo 'Service without destination: ' . $service->name . '\n';
 | 
			
		||||
                    $service->forceDelete();
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
                if (!data_get($service, 'server')) {
 | 
			
		||||
                    echo 'Service without server: ' . $service->name . ' soft deleting\n';
 | 
			
		||||
                    $service->delete();
 | 
			
		||||
                    echo 'Service without server: ' . $service->name . '\n';
 | 
			
		||||
                    $service->forceDelete();
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
@@ -271,8 +272,8 @@ class CleanupStuckedResources extends Command
 | 
			
		||||
            $serviceApplications = ServiceApplication::all();
 | 
			
		||||
            foreach ($serviceApplications as $service) {
 | 
			
		||||
                if (!data_get($service, 'service')) {
 | 
			
		||||
                    echo 'ServiceApplication without service: ' . $service->name . ' soft deleting\n';
 | 
			
		||||
                    $service->delete();
 | 
			
		||||
                    echo 'ServiceApplication without service: ' . $service->name . '\n';
 | 
			
		||||
                    $service->forceDelete();
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
@@ -283,8 +284,8 @@ class CleanupStuckedResources extends Command
 | 
			
		||||
            $serviceDatabases = ServiceDatabase::all();
 | 
			
		||||
            foreach ($serviceDatabases as $service) {
 | 
			
		||||
                if (!data_get($service, 'service')) {
 | 
			
		||||
                    echo 'ServiceDatabase without service: ' . $service->name . ' soft deleting\n';
 | 
			
		||||
                    $service->delete();
 | 
			
		||||
                    echo 'ServiceDatabase without service: ' . $service->name . '\n';
 | 
			
		||||
                    $service->forceDelete();
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
 
 | 
			
		||||
@@ -14,39 +14,44 @@ use Illuminate\Support\Facades\Http;
 | 
			
		||||
 | 
			
		||||
class Init extends Command
 | 
			
		||||
{
 | 
			
		||||
    protected $signature = 'app:init {--cleanup}';
 | 
			
		||||
    protected $signature = 'app:init {--full-cleanup} {--cleanup-deployments}';
 | 
			
		||||
    protected $description = 'Cleanup instance related stuffs';
 | 
			
		||||
 | 
			
		||||
    public function handle()
 | 
			
		||||
    {
 | 
			
		||||
        $this->alive();
 | 
			
		||||
        $cleanup = $this->option('cleanup');
 | 
			
		||||
        if ($cleanup) {
 | 
			
		||||
            echo "Running cleanups...\n";
 | 
			
		||||
            $this->call('cleanup:stucked-resources');
 | 
			
		||||
        $full_cleanup = $this->option('full-cleanup');
 | 
			
		||||
        $cleanup_deployments = $this->option('cleanup-deployments');
 | 
			
		||||
        if ($cleanup_deployments) {
 | 
			
		||||
            echo "Running cleanup deployments.\n";
 | 
			
		||||
            $this->cleanup_in_progress_application_deployments();
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
        if ($full_cleanup) {
 | 
			
		||||
            // Required for falsely deleted coolify db
 | 
			
		||||
            $this->restore_coolify_db_backup();
 | 
			
		||||
 | 
			
		||||
            // $this->cleanup_ssh();
 | 
			
		||||
        }
 | 
			
		||||
        $this->cleanup_in_progress_application_deployments();
 | 
			
		||||
        $this->cleanup_stucked_helper_containers();
 | 
			
		||||
 | 
			
		||||
        try {
 | 
			
		||||
            setup_dynamic_configuration();
 | 
			
		||||
        } catch (\Throwable $e) {
 | 
			
		||||
            echo "Could not setup dynamic configuration: {$e->getMessage()}\n";
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        $settings = InstanceSettings::get();
 | 
			
		||||
        if (!is_null(env('AUTOUPDATE', null))) {
 | 
			
		||||
            if (env('AUTOUPDATE') == true) {
 | 
			
		||||
                $settings->update(['is_auto_update_enabled' => true]);
 | 
			
		||||
            } else {
 | 
			
		||||
                $settings->update(['is_auto_update_enabled' => false]);
 | 
			
		||||
            $this->cleanup_in_progress_application_deployments();
 | 
			
		||||
            $this->cleanup_stucked_helper_containers();
 | 
			
		||||
            $this->call('cleanup:queue');
 | 
			
		||||
            $this->call('cleanup:stucked-resources');
 | 
			
		||||
            try {
 | 
			
		||||
                setup_dynamic_configuration();
 | 
			
		||||
            } catch (\Throwable $e) {
 | 
			
		||||
                echo "Could not setup dynamic configuration: {$e->getMessage()}\n";
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            $settings = InstanceSettings::get();
 | 
			
		||||
            if (!is_null(env('AUTOUPDATE', null))) {
 | 
			
		||||
                if (env('AUTOUPDATE') == true) {
 | 
			
		||||
                    $settings->update(['is_auto_update_enabled' => true]);
 | 
			
		||||
                } else {
 | 
			
		||||
                    $settings->update(['is_auto_update_enabled' => false]);
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
        $this->call('cleanup:queue');
 | 
			
		||||
        $this->cleanup_stucked_helper_containers();
 | 
			
		||||
        $this->call('cleanup:stucked-resources');
 | 
			
		||||
    }
 | 
			
		||||
    private function restore_coolify_db_backup()
 | 
			
		||||
    {
 | 
			
		||||
@@ -120,8 +125,10 @@ class Init extends Command
 | 
			
		||||
        // Cleanup any failed deployments
 | 
			
		||||
 | 
			
		||||
        try {
 | 
			
		||||
            $halted_deployments = ApplicationDeploymentQueue::where('status', '==', ApplicationDeploymentStatus::IN_PROGRESS)->where('status', '==', ApplicationDeploymentStatus::QUEUED)->get();
 | 
			
		||||
            foreach ($halted_deployments as $deployment) {
 | 
			
		||||
            $queued_inprogress_deployments = ApplicationDeploymentQueue::whereIn('status', [ApplicationDeploymentStatus::IN_PROGRESS->value, ApplicationDeploymentStatus::QUEUED->value])->get();
 | 
			
		||||
            foreach ($queued_inprogress_deployments as $deployment) {
 | 
			
		||||
                ray($deployment->id, $deployment->status);
 | 
			
		||||
                echo "Cleaning up deployment: {$deployment->id}\n";
 | 
			
		||||
                $deployment->status = ApplicationDeploymentStatus::FAILED->value;
 | 
			
		||||
                $deployment->save();
 | 
			
		||||
            }
 | 
			
		||||
@@ -129,5 +136,4 @@ class Init extends Command
 | 
			
		||||
            echo "Error: {$e->getMessage()}\n";
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -4,6 +4,7 @@ namespace App\Console;
 | 
			
		||||
 | 
			
		||||
use App\Jobs\CheckLogDrainContainerJob;
 | 
			
		||||
use App\Jobs\CleanupInstanceStuffsJob;
 | 
			
		||||
use App\Jobs\ComplexContainerStatusJob;
 | 
			
		||||
use App\Jobs\DatabaseBackupJob;
 | 
			
		||||
use App\Jobs\ScheduledTaskJob;
 | 
			
		||||
use App\Jobs\InstanceAutoUpdateJob;
 | 
			
		||||
@@ -91,7 +92,6 @@ class Kernel extends ConsoleKernel
 | 
			
		||||
    {
 | 
			
		||||
        $scheduled_backups = ScheduledDatabaseBackup::all();
 | 
			
		||||
        if ($scheduled_backups->isEmpty()) {
 | 
			
		||||
            ray('no scheduled backups');
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
        foreach ($scheduled_backups as $scheduled_backup) {
 | 
			
		||||
@@ -117,7 +117,6 @@ class Kernel extends ConsoleKernel
 | 
			
		||||
    {
 | 
			
		||||
        $scheduled_tasks = ScheduledTask::all();
 | 
			
		||||
        if ($scheduled_tasks->isEmpty()) {
 | 
			
		||||
            ray('no scheduled tasks');
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
        foreach ($scheduled_tasks as $scheduled_task) {
 | 
			
		||||
 
 | 
			
		||||
@@ -123,6 +123,7 @@ class ApplicationDeploymentJob implements ShouldQueue, ShouldBeEncrypted
 | 
			
		||||
            $this->source = $source->getMorphClass()::where('id', $this->application->source->id)->first();
 | 
			
		||||
        }
 | 
			
		||||
        $this->server = Server::find($this->application_deployment_queue->server_id);
 | 
			
		||||
        $this->timeout = $this->server->settings->dynamic_timeout;
 | 
			
		||||
        $this->destination = $this->server->destinations()->where('id', $this->application_deployment_queue->destination_id)->first();
 | 
			
		||||
        $this->server = $this->mainServer = $this->destination->server;
 | 
			
		||||
        $this->serverUser = $this->server->user;
 | 
			
		||||
@@ -248,9 +249,14 @@ class ApplicationDeploymentJob implements ShouldQueue, ShouldBeEncrypted
 | 
			
		||||
                dispatch(new ContainerStatusJob($this->server));
 | 
			
		||||
            }
 | 
			
		||||
            // Otherwise built image needs to be pushed before from the build server.
 | 
			
		||||
            if (!$this->use_build_server) {
 | 
			
		||||
                $this->push_to_docker_registry();
 | 
			
		||||
            }
 | 
			
		||||
            // ray($this->use_build_server);
 | 
			
		||||
            // if (!$this->use_build_server) {
 | 
			
		||||
            //     if ($this->application->additional_servers->count() > 0) {
 | 
			
		||||
            //         $this->push_to_docker_registry(forceFail: true);
 | 
			
		||||
            //     } else {
 | 
			
		||||
            //         $this->push_to_docker_registry();
 | 
			
		||||
            //     }
 | 
			
		||||
            // }
 | 
			
		||||
            $this->next(ApplicationDeploymentStatus::FINISHED->value);
 | 
			
		||||
            if ($this->pull_request_id !== 0) {
 | 
			
		||||
                if ($this->application->is_github_based()) {
 | 
			
		||||
@@ -288,162 +294,13 @@ class ApplicationDeploymentJob implements ShouldQueue, ShouldBeEncrypted
 | 
			
		||||
            ApplicationStatusChanged::dispatch(data_get($this->application, 'environment.project.team.id'));
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    private function write_deployment_configurations()
 | 
			
		||||
    {
 | 
			
		||||
        if (isset($this->docker_compose_base64)) {
 | 
			
		||||
            if ($this->use_build_server) {
 | 
			
		||||
                $this->server = $this->original_server;
 | 
			
		||||
            }
 | 
			
		||||
            $readme = generate_readme_file($this->application->name, $this->application_deployment_queue->updated_at);
 | 
			
		||||
            $composeFileName = "$this->configuration_dir/docker-compose.yml";
 | 
			
		||||
            if ($this->pull_request_id !== 0) {
 | 
			
		||||
                $composeFileName = "$this->configuration_dir/docker-compose-pr-{$this->pull_request_id}.yml";
 | 
			
		||||
            }
 | 
			
		||||
            $this->execute_remote_command(
 | 
			
		||||
                [
 | 
			
		||||
                    "mkdir -p $this->configuration_dir"
 | 
			
		||||
                ],
 | 
			
		||||
                [
 | 
			
		||||
                    "echo '{$this->docker_compose_base64}' | base64 -d > $composeFileName",
 | 
			
		||||
                ],
 | 
			
		||||
                [
 | 
			
		||||
                    "echo '{$readme}' > $this->configuration_dir/README.md",
 | 
			
		||||
                ]
 | 
			
		||||
            );
 | 
			
		||||
            if ($this->use_build_server) {
 | 
			
		||||
                $this->server = $this->build_server;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    private function push_to_docker_registry($forceFail = false)
 | 
			
		||||
    {
 | 
			
		||||
        if (
 | 
			
		||||
            $this->application->docker_registry_image_name &&
 | 
			
		||||
            $this->application->build_pack !== 'dockerimage' &&
 | 
			
		||||
            !$this->application->destination->server->isSwarm() &&
 | 
			
		||||
            !$this->restart_only &&
 | 
			
		||||
            !(str($this->saved_outputs->get('local_image_found'))->isNotEmpty() && !$this->application->isConfigurationChanged())
 | 
			
		||||
        ) {
 | 
			
		||||
            try {
 | 
			
		||||
                instant_remote_process(["docker images --format '{{json .}}' {$this->production_image_name}"], $this->server);
 | 
			
		||||
                $this->application_deployment_queue->addLogEntry("----------------------------------------");
 | 
			
		||||
                $this->application_deployment_queue->addLogEntry("Pushing image to docker registry ({$this->production_image_name}).");
 | 
			
		||||
                $this->execute_remote_command(
 | 
			
		||||
                    [
 | 
			
		||||
                        executeInDocker($this->deployment_uuid, "docker push {$this->production_image_name}"), 'hidden' => true
 | 
			
		||||
                    ],
 | 
			
		||||
                );
 | 
			
		||||
                if ($this->application->docker_registry_image_tag) {
 | 
			
		||||
                    // Tag image with latest
 | 
			
		||||
                    $this->application_deployment_queue->addLogEntry("Tagging and pushing image with latest tag.");
 | 
			
		||||
                    $this->execute_remote_command(
 | 
			
		||||
                        [
 | 
			
		||||
                            executeInDocker($this->deployment_uuid, "docker tag {$this->production_image_name} {$this->application->docker_registry_image_name}:{$this->application->docker_registry_image_tag}"), 'ignore_errors' => true, 'hidden' => true
 | 
			
		||||
                        ],
 | 
			
		||||
                        [
 | 
			
		||||
                            executeInDocker($this->deployment_uuid, "docker push {$this->application->docker_registry_image_name}:{$this->application->docker_registry_image_tag}"), 'ignore_errors' => true, 'hidden' => true
 | 
			
		||||
                        ],
 | 
			
		||||
                    );
 | 
			
		||||
                }
 | 
			
		||||
                $this->application_deployment_queue->addLogEntry("Image pushed to docker registry.");
 | 
			
		||||
            } catch (Exception $e) {
 | 
			
		||||
                $this->application_deployment_queue->addLogEntry("Failed to push image to docker registry. Please check debug logs for more information.");
 | 
			
		||||
                if ($forceFail) {
 | 
			
		||||
                    throw $e;
 | 
			
		||||
                }
 | 
			
		||||
                ray($e);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    private function generate_image_names()
 | 
			
		||||
    {
 | 
			
		||||
        if ($this->application->dockerfile) {
 | 
			
		||||
            if ($this->application->docker_registry_image_name) {
 | 
			
		||||
                $this->build_image_name = Str::lower("{$this->application->docker_registry_image_name}:build");
 | 
			
		||||
                $this->production_image_name = Str::lower("{$this->application->docker_registry_image_name}:latest");
 | 
			
		||||
            } else {
 | 
			
		||||
                $this->build_image_name = Str::lower("{$this->application->uuid}:build");
 | 
			
		||||
                $this->production_image_name = Str::lower("{$this->application->uuid}:latest");
 | 
			
		||||
            }
 | 
			
		||||
        } else if ($this->application->build_pack === 'dockerimage') {
 | 
			
		||||
            $this->production_image_name = Str::lower("{$this->dockerImage}:{$this->dockerImageTag}");
 | 
			
		||||
        } else if ($this->pull_request_id !== 0) {
 | 
			
		||||
            if ($this->application->docker_registry_image_name) {
 | 
			
		||||
                $this->build_image_name = Str::lower("{$this->application->docker_registry_image_name}:pr-{$this->pull_request_id}-build");
 | 
			
		||||
                $this->production_image_name = Str::lower("{$this->application->docker_registry_image_name}:pr-{$this->pull_request_id}");
 | 
			
		||||
            } else {
 | 
			
		||||
                $this->build_image_name = Str::lower("{$this->application->uuid}:pr-{$this->pull_request_id}-build");
 | 
			
		||||
                $this->production_image_name = Str::lower("{$this->application->uuid}:pr-{$this->pull_request_id}");
 | 
			
		||||
            }
 | 
			
		||||
        } else {
 | 
			
		||||
            $this->dockerImageTag = str($this->commit)->substr(0, 128);
 | 
			
		||||
            if ($this->application->docker_registry_image_name) {
 | 
			
		||||
                $this->build_image_name = Str::lower("{$this->application->docker_registry_image_name}:{$this->dockerImageTag}-build");
 | 
			
		||||
                $this->production_image_name = Str::lower("{$this->application->docker_registry_image_name}:{$this->dockerImageTag}");
 | 
			
		||||
            } else {
 | 
			
		||||
                $this->build_image_name = Str::lower("{$this->application->uuid}:{$this->dockerImageTag}-build");
 | 
			
		||||
                $this->production_image_name = Str::lower("{$this->application->uuid}:{$this->dockerImageTag}");
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    private function just_restart()
 | 
			
		||||
    {
 | 
			
		||||
        $this->application_deployment_queue->addLogEntry("Starting deployment of {$this->customRepository}:{$this->application->git_branch}.");
 | 
			
		||||
        $this->prepare_builder_image();
 | 
			
		||||
        $this->check_git_if_build_needed();
 | 
			
		||||
        $this->set_base_dir();
 | 
			
		||||
        $this->generate_image_names();
 | 
			
		||||
        $this->check_image_locally_or_remotely();
 | 
			
		||||
        if (str($this->saved_outputs->get('local_image_found'))->isNotEmpty()) {
 | 
			
		||||
            $this->application_deployment_queue->addLogEntry("Image found ({$this->production_image_name}) with the same Git Commit SHA. Restarting container.");
 | 
			
		||||
            $this->create_workdir();
 | 
			
		||||
            $this->generate_compose_file();
 | 
			
		||||
            $this->rolling_update();
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
        throw new RuntimeException('Cannot find image anywhere. Please redeploy the application.');
 | 
			
		||||
    }
 | 
			
		||||
    private function check_image_locally_or_remotely()
 | 
			
		||||
    {
 | 
			
		||||
        $this->execute_remote_command([
 | 
			
		||||
            "docker images -q {$this->production_image_name} 2>/dev/null", "hidden" => true, "save" => "local_image_found"
 | 
			
		||||
        ]);
 | 
			
		||||
        if (str($this->saved_outputs->get('local_image_found'))->isEmpty() && $this->application->docker_registry_image_name) {
 | 
			
		||||
            $this->execute_remote_command([
 | 
			
		||||
                "docker pull {$this->production_image_name} 2>/dev/null", "ignore_errors" => true, "hidden" => true
 | 
			
		||||
            ]);
 | 
			
		||||
            $this->execute_remote_command([
 | 
			
		||||
                "docker images -q {$this->production_image_name} 2>/dev/null", "hidden" => true, "save" => "local_image_found"
 | 
			
		||||
            ]);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    private function save_environment_variables()
 | 
			
		||||
    {
 | 
			
		||||
        $envs = collect([]);
 | 
			
		||||
        if ($this->pull_request_id !== 0) {
 | 
			
		||||
            foreach ($this->application->environment_variables_preview as $env) {
 | 
			
		||||
                $envs->push($env->key . '=' . $env->real_value);
 | 
			
		||||
            }
 | 
			
		||||
        } else {
 | 
			
		||||
            foreach ($this->application->environment_variables as $env) {
 | 
			
		||||
                $envs->push($env->key . '=' . $env->real_value);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        $envs_base64 = base64_encode($envs->implode("\n"));
 | 
			
		||||
        $this->execute_remote_command(
 | 
			
		||||
            [
 | 
			
		||||
                executeInDocker($this->deployment_uuid, "echo '$envs_base64' | base64 -d > $this->workdir/.env")
 | 
			
		||||
            ],
 | 
			
		||||
        );
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private function deploy_simple_dockerfile()
 | 
			
		||||
    {
 | 
			
		||||
        if ($this->use_build_server) {
 | 
			
		||||
            $this->server = $this->build_server;
 | 
			
		||||
        }
 | 
			
		||||
        $dockerfile_base64 = base64_encode($this->application->dockerfile);
 | 
			
		||||
        $this->application_deployment_queue->addLogEntry("Starting deployment of {$this->application->name}.");
 | 
			
		||||
        $this->application_deployment_queue->addLogEntry("Starting deployment of {$this->application->name} to {$this->server->name}.");
 | 
			
		||||
        $this->prepare_builder_image();
 | 
			
		||||
        $this->execute_remote_command(
 | 
			
		||||
            [
 | 
			
		||||
@@ -451,19 +308,30 @@ class ApplicationDeploymentJob implements ShouldQueue, ShouldBeEncrypted
 | 
			
		||||
            ],
 | 
			
		||||
        );
 | 
			
		||||
        $this->generate_image_names();
 | 
			
		||||
        if (!$this->force_rebuild) {
 | 
			
		||||
            $this->check_image_locally_or_remotely();
 | 
			
		||||
            if (str($this->saved_outputs->get('local_image_found'))->isNotEmpty() && !$this->application->isConfigurationChanged()) {
 | 
			
		||||
                $this->create_workdir();
 | 
			
		||||
                $this->application_deployment_queue->addLogEntry("No configuration changed & image found ({$this->production_image_name}) with the same Git Commit SHA. Build step skipped.");
 | 
			
		||||
                $this->generate_compose_file();
 | 
			
		||||
                $this->push_to_docker_registry();
 | 
			
		||||
                $this->rolling_update();
 | 
			
		||||
                return;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        $this->generate_compose_file();
 | 
			
		||||
        $this->generate_build_env_variables();
 | 
			
		||||
        $this->add_build_env_variables_to_dockerfile();
 | 
			
		||||
        $this->build_image();
 | 
			
		||||
        $this->push_to_docker_registry();
 | 
			
		||||
        $this->rolling_update();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private function deploy_dockerimage_buildpack()
 | 
			
		||||
    {
 | 
			
		||||
        $this->dockerImage = $this->application->docker_registry_image_name;
 | 
			
		||||
        $this->dockerImageTag = $this->application->docker_registry_image_tag;
 | 
			
		||||
        ray("echo 'Starting deployment of {$this->dockerImage}:{$this->dockerImageTag}.'");
 | 
			
		||||
        $this->application_deployment_queue->addLogEntry("Starting deployment of {$this->dockerImage}:{$this->dockerImageTag}.");
 | 
			
		||||
        ray("echo 'Starting deployment of {$this->dockerImage}:{$this->dockerImageTag} to {$this->server->name}.'");
 | 
			
		||||
        $this->application_deployment_queue->addLogEntry("Starting deployment of {$this->dockerImage}:{$this->dockerImageTag} to {$this->server->name}.");
 | 
			
		||||
        $this->generate_image_names();
 | 
			
		||||
        $this->prepare_builder_image();
 | 
			
		||||
        $this->generate_compose_file();
 | 
			
		||||
@@ -481,9 +349,9 @@ class ApplicationDeploymentJob implements ShouldQueue, ShouldBeEncrypted
 | 
			
		||||
            $this->docker_compose_custom_build_command = $this->application->docker_compose_custom_build_command;
 | 
			
		||||
        }
 | 
			
		||||
        if ($this->pull_request_id === 0) {
 | 
			
		||||
            $this->application_deployment_queue->addLogEntry("Starting deployment of {$this->application->name}.");
 | 
			
		||||
            $this->application_deployment_queue->addLogEntry("Starting deployment of {$this->application->name} to {$this->server->name}.");
 | 
			
		||||
        } else {
 | 
			
		||||
            $this->application_deployment_queue->addLogEntry("Starting pull request (#{$this->pull_request_id}) deployment of {$this->customRepository}:{$this->application->git_branch}.");
 | 
			
		||||
            $this->application_deployment_queue->addLogEntry("Starting pull request (#{$this->pull_request_id}) deployment of {$this->customRepository}:{$this->application->git_branch} to {$this->server->name}.");
 | 
			
		||||
        }
 | 
			
		||||
        $this->prepare_builder_image();
 | 
			
		||||
        $this->check_git_if_build_needed();
 | 
			
		||||
@@ -551,25 +419,7 @@ class ApplicationDeploymentJob implements ShouldQueue, ShouldBeEncrypted
 | 
			
		||||
        if (data_get($this->application, 'dockerfile_location')) {
 | 
			
		||||
            $this->dockerfile_location = $this->application->dockerfile_location;
 | 
			
		||||
        }
 | 
			
		||||
        $this->application_deployment_queue->addLogEntry("Starting deployment of {$this->customRepository}:{$this->application->git_branch}.");
 | 
			
		||||
        $this->prepare_builder_image();
 | 
			
		||||
        $this->check_git_if_build_needed();
 | 
			
		||||
        $this->clone_repository();
 | 
			
		||||
        $this->set_base_dir();
 | 
			
		||||
        $this->generate_image_names();
 | 
			
		||||
        $this->cleanup_git();
 | 
			
		||||
        $this->generate_compose_file();
 | 
			
		||||
        $this->generate_build_env_variables();
 | 
			
		||||
        $this->add_build_env_variables_to_dockerfile();
 | 
			
		||||
        $this->build_image();
 | 
			
		||||
        $this->rolling_update();
 | 
			
		||||
    }
 | 
			
		||||
    private function deploy_nixpacks_buildpack()
 | 
			
		||||
    {
 | 
			
		||||
        if ($this->use_build_server) {
 | 
			
		||||
            $this->server = $this->build_server;
 | 
			
		||||
        }
 | 
			
		||||
        $this->application_deployment_queue->addLogEntry("Starting deployment of {$this->customRepository}:{$this->application->git_branch}.");
 | 
			
		||||
        $this->application_deployment_queue->addLogEntry("Starting deployment of {$this->customRepository}:{$this->application->git_branch} to {$this->server->name}.");
 | 
			
		||||
        $this->prepare_builder_image();
 | 
			
		||||
        $this->check_git_if_build_needed();
 | 
			
		||||
        $this->set_base_dir();
 | 
			
		||||
@@ -580,6 +430,38 @@ class ApplicationDeploymentJob implements ShouldQueue, ShouldBeEncrypted
 | 
			
		||||
                $this->create_workdir();
 | 
			
		||||
                $this->application_deployment_queue->addLogEntry("No configuration changed & image found ({$this->production_image_name}) with the same Git Commit SHA. Build step skipped.");
 | 
			
		||||
                $this->generate_compose_file();
 | 
			
		||||
                $this->push_to_docker_registry();
 | 
			
		||||
                $this->rolling_update();
 | 
			
		||||
                return;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        $this->clone_repository();
 | 
			
		||||
        $this->cleanup_git();
 | 
			
		||||
        $this->generate_compose_file();
 | 
			
		||||
        $this->generate_build_env_variables();
 | 
			
		||||
        $this->add_build_env_variables_to_dockerfile();
 | 
			
		||||
        $this->build_image();
 | 
			
		||||
        $this->push_to_docker_registry();
 | 
			
		||||
        $this->rolling_update();
 | 
			
		||||
    }
 | 
			
		||||
    private function deploy_nixpacks_buildpack()
 | 
			
		||||
    {
 | 
			
		||||
        if ($this->use_build_server) {
 | 
			
		||||
            $this->server = $this->build_server;
 | 
			
		||||
        }
 | 
			
		||||
        $this->application_deployment_queue->addLogEntry("Starting deployment of {$this->customRepository}:{$this->application->git_branch} to {$this->server->name}.");
 | 
			
		||||
        $this->prepare_builder_image();
 | 
			
		||||
        $this->check_git_if_build_needed();
 | 
			
		||||
        $this->set_base_dir();
 | 
			
		||||
        $this->generate_image_names();
 | 
			
		||||
        if (!$this->force_rebuild) {
 | 
			
		||||
            $this->check_image_locally_or_remotely();
 | 
			
		||||
            if (str($this->saved_outputs->get('local_image_found'))->isNotEmpty() && !$this->application->isConfigurationChanged()) {
 | 
			
		||||
                $this->create_workdir();
 | 
			
		||||
                $this->application_deployment_queue->addLogEntry("No configuration changed & image found ({$this->production_image_name}) with the same Git Commit SHA. Build step skipped.");
 | 
			
		||||
                $this->generate_compose_file();
 | 
			
		||||
                ray('pushing to docker registry');
 | 
			
		||||
                $this->push_to_docker_registry();
 | 
			
		||||
                $this->rolling_update();
 | 
			
		||||
                return;
 | 
			
		||||
            }
 | 
			
		||||
@@ -592,8 +474,8 @@ class ApplicationDeploymentJob implements ShouldQueue, ShouldBeEncrypted
 | 
			
		||||
        $this->generate_nixpacks_confs();
 | 
			
		||||
        $this->generate_compose_file();
 | 
			
		||||
        $this->generate_build_env_variables();
 | 
			
		||||
        // $this->add_build_env_variables_to_dockerfile();
 | 
			
		||||
        $this->build_image();
 | 
			
		||||
        $this->push_to_docker_registry();
 | 
			
		||||
        $this->rolling_update();
 | 
			
		||||
    }
 | 
			
		||||
    private function deploy_static_buildpack()
 | 
			
		||||
@@ -601,18 +483,202 @@ class ApplicationDeploymentJob implements ShouldQueue, ShouldBeEncrypted
 | 
			
		||||
        if ($this->use_build_server) {
 | 
			
		||||
            $this->server = $this->build_server;
 | 
			
		||||
        }
 | 
			
		||||
        $this->application_deployment_queue->addLogEntry("Starting deployment of {$this->customRepository}:{$this->application->git_branch}.");
 | 
			
		||||
        $this->application_deployment_queue->addLogEntry("Starting deployment of {$this->customRepository}:{$this->application->git_branch} to {$this->server->name}.");
 | 
			
		||||
        $this->prepare_builder_image();
 | 
			
		||||
        $this->check_git_if_build_needed();
 | 
			
		||||
        $this->set_base_dir();
 | 
			
		||||
        $this->generate_image_names();
 | 
			
		||||
        if (!$this->force_rebuild) {
 | 
			
		||||
            $this->check_image_locally_or_remotely();
 | 
			
		||||
            if (str($this->saved_outputs->get('local_image_found'))->isNotEmpty() && !$this->application->isConfigurationChanged()) {
 | 
			
		||||
                $this->create_workdir();
 | 
			
		||||
                $this->application_deployment_queue->addLogEntry("No configuration changed & image found ({$this->production_image_name}) with the same Git Commit SHA. Build step skipped.");
 | 
			
		||||
                $this->generate_compose_file();
 | 
			
		||||
                $this->push_to_docker_registry();
 | 
			
		||||
                $this->rolling_update();
 | 
			
		||||
                return;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        $this->clone_repository();
 | 
			
		||||
        $this->cleanup_git();
 | 
			
		||||
        $this->build_image();
 | 
			
		||||
        $this->generate_compose_file();
 | 
			
		||||
        $this->build_image();
 | 
			
		||||
        $this->push_to_docker_registry();
 | 
			
		||||
        $this->rolling_update();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private function write_deployment_configurations()
 | 
			
		||||
    {
 | 
			
		||||
        if (isset($this->docker_compose_base64)) {
 | 
			
		||||
            if ($this->use_build_server) {
 | 
			
		||||
                $this->server = $this->original_server;
 | 
			
		||||
            }
 | 
			
		||||
            $readme = generate_readme_file($this->application->name, $this->application_deployment_queue->updated_at);
 | 
			
		||||
            $composeFileName = "$this->configuration_dir/docker-compose.yml";
 | 
			
		||||
            if ($this->pull_request_id !== 0) {
 | 
			
		||||
                $composeFileName = "$this->configuration_dir/docker-compose-pr-{$this->pull_request_id}.yml";
 | 
			
		||||
            }
 | 
			
		||||
            $this->execute_remote_command(
 | 
			
		||||
                [
 | 
			
		||||
                    "mkdir -p $this->configuration_dir"
 | 
			
		||||
                ],
 | 
			
		||||
                [
 | 
			
		||||
                    "echo '{$this->docker_compose_base64}' | base64 -d > $composeFileName",
 | 
			
		||||
                ],
 | 
			
		||||
                [
 | 
			
		||||
                    "echo '{$readme}' > $this->configuration_dir/README.md",
 | 
			
		||||
                ]
 | 
			
		||||
            );
 | 
			
		||||
            if ($this->use_build_server) {
 | 
			
		||||
                $this->server = $this->build_server;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    private function push_to_docker_registry()
 | 
			
		||||
    {
 | 
			
		||||
        $forceFail = true;
 | 
			
		||||
        if (str($this->application->docker_registry_image_name)->isEmpty()) {
 | 
			
		||||
            ray('empty docker_registry_image_name');
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
        if ($this->restart_only) {
 | 
			
		||||
            ray('restart_only');
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
        if ($this->application->build_pack === 'dockerimage') {
 | 
			
		||||
            ray('dockerimage');
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
        if ($this->use_build_server) {
 | 
			
		||||
            ray('use_build_server');
 | 
			
		||||
            $forceFail = true;
 | 
			
		||||
        }
 | 
			
		||||
        if ($this->server->isSwarm() && $this->build_pack !== 'dockerimage') {
 | 
			
		||||
            ray('isSwarm');
 | 
			
		||||
            $forceFail = true;
 | 
			
		||||
        }
 | 
			
		||||
        if ($this->application->additional_servers->count() > 0) {
 | 
			
		||||
            ray('additional_servers');
 | 
			
		||||
            $forceFail = true;
 | 
			
		||||
        }
 | 
			
		||||
        if ($this->application->additional_servers()->wherePivot('server_id', $this->server->id)->count() > 0) {
 | 
			
		||||
            ray('this is an additional_servers, no pushy pushy');
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
        ray('push_to_docker_registry noww: ' . $this->production_image_name);
 | 
			
		||||
        try {
 | 
			
		||||
            instant_remote_process(["docker images --format '{{json .}}' {$this->production_image_name}"], $this->server);
 | 
			
		||||
            $this->application_deployment_queue->addLogEntry("----------------------------------------");
 | 
			
		||||
            $this->application_deployment_queue->addLogEntry("Pushing image to docker registry ({$this->production_image_name}).");
 | 
			
		||||
            $this->execute_remote_command(
 | 
			
		||||
                [
 | 
			
		||||
                    executeInDocker($this->deployment_uuid, "docker push {$this->production_image_name}"), 'hidden' => true
 | 
			
		||||
                ],
 | 
			
		||||
            );
 | 
			
		||||
            if ($this->application->docker_registry_image_tag) {
 | 
			
		||||
                // Tag image with latest
 | 
			
		||||
                $this->application_deployment_queue->addLogEntry("Tagging and pushing image with latest tag.");
 | 
			
		||||
                $this->execute_remote_command(
 | 
			
		||||
                    [
 | 
			
		||||
                        executeInDocker($this->deployment_uuid, "docker tag {$this->production_image_name} {$this->application->docker_registry_image_name}:{$this->application->docker_registry_image_tag}"), 'ignore_errors' => true, 'hidden' => true
 | 
			
		||||
                    ],
 | 
			
		||||
                    [
 | 
			
		||||
                        executeInDocker($this->deployment_uuid, "docker push {$this->application->docker_registry_image_name}:{$this->application->docker_registry_image_tag}"), 'ignore_errors' => true, 'hidden' => true
 | 
			
		||||
                    ],
 | 
			
		||||
                );
 | 
			
		||||
            }
 | 
			
		||||
            $this->application_deployment_queue->addLogEntry("Image pushed to docker registry.");
 | 
			
		||||
        } catch (Exception $e) {
 | 
			
		||||
            $this->application_deployment_queue->addLogEntry("Failed to push image to docker registry. Please check debug logs for more information.");
 | 
			
		||||
            if ($forceFail) {
 | 
			
		||||
                throw new RuntimeException($e->getMessage(), 69420);
 | 
			
		||||
            }
 | 
			
		||||
            ray($e);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    private function generate_image_names()
 | 
			
		||||
    {
 | 
			
		||||
        if ($this->application->dockerfile) {
 | 
			
		||||
            if ($this->application->docker_registry_image_name) {
 | 
			
		||||
                $this->build_image_name = Str::lower("{$this->application->docker_registry_image_name}:build");
 | 
			
		||||
                $this->production_image_name = Str::lower("{$this->application->docker_registry_image_name}:latest");
 | 
			
		||||
            } else {
 | 
			
		||||
                $this->build_image_name = Str::lower("{$this->application->uuid}:build");
 | 
			
		||||
                $this->production_image_name = Str::lower("{$this->application->uuid}:latest");
 | 
			
		||||
            }
 | 
			
		||||
        } else if ($this->application->build_pack === 'dockerimage') {
 | 
			
		||||
            $this->production_image_name = Str::lower("{$this->dockerImage}:{$this->dockerImageTag}");
 | 
			
		||||
        } else if ($this->pull_request_id !== 0) {
 | 
			
		||||
            if ($this->application->docker_registry_image_name) {
 | 
			
		||||
                $this->build_image_name = Str::lower("{$this->application->docker_registry_image_name}:pr-{$this->pull_request_id}-build");
 | 
			
		||||
                $this->production_image_name = Str::lower("{$this->application->docker_registry_image_name}:pr-{$this->pull_request_id}");
 | 
			
		||||
            } else {
 | 
			
		||||
                $this->build_image_name = Str::lower("{$this->application->uuid}:pr-{$this->pull_request_id}-build");
 | 
			
		||||
                $this->production_image_name = Str::lower("{$this->application->uuid}:pr-{$this->pull_request_id}");
 | 
			
		||||
            }
 | 
			
		||||
        } else {
 | 
			
		||||
            $this->dockerImageTag = str($this->commit)->substr(0, 128);
 | 
			
		||||
            if ($this->application->docker_registry_image_name) {
 | 
			
		||||
                $this->build_image_name = Str::lower("{$this->application->docker_registry_image_name}:{$this->dockerImageTag}-build");
 | 
			
		||||
                $this->production_image_name = Str::lower("{$this->application->docker_registry_image_name}:{$this->dockerImageTag}");
 | 
			
		||||
            } else {
 | 
			
		||||
                $this->build_image_name = Str::lower("{$this->application->uuid}:{$this->dockerImageTag}-build");
 | 
			
		||||
                $this->production_image_name = Str::lower("{$this->application->uuid}:{$this->dockerImageTag}");
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    private function just_restart()
 | 
			
		||||
    {
 | 
			
		||||
        $this->application_deployment_queue->addLogEntry("Starting deployment of {$this->customRepository}:{$this->application->git_branch} to {$this->server->name}.");
 | 
			
		||||
        $this->prepare_builder_image();
 | 
			
		||||
        $this->check_git_if_build_needed();
 | 
			
		||||
        $this->set_base_dir();
 | 
			
		||||
        $this->generate_image_names();
 | 
			
		||||
        $this->check_image_locally_or_remotely();
 | 
			
		||||
        if (str($this->saved_outputs->get('local_image_found'))->isNotEmpty()) {
 | 
			
		||||
            $this->application_deployment_queue->addLogEntry("Image found ({$this->production_image_name}) with the same Git Commit SHA. Restarting container.");
 | 
			
		||||
            $this->create_workdir();
 | 
			
		||||
            $this->generate_compose_file();
 | 
			
		||||
            $this->rolling_update();
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
        throw new RuntimeException('Cannot find image anywhere. Please redeploy the application.');
 | 
			
		||||
    }
 | 
			
		||||
    private function check_image_locally_or_remotely()
 | 
			
		||||
    {
 | 
			
		||||
        $this->execute_remote_command([
 | 
			
		||||
            "docker images -q {$this->production_image_name} 2>/dev/null", "hidden" => true, "save" => "local_image_found"
 | 
			
		||||
        ]);
 | 
			
		||||
        if (str($this->saved_outputs->get('local_image_found'))->isEmpty() && $this->application->docker_registry_image_name) {
 | 
			
		||||
            $this->execute_remote_command([
 | 
			
		||||
                "docker pull {$this->production_image_name} 2>/dev/null", "ignore_errors" => true, "hidden" => true
 | 
			
		||||
            ]);
 | 
			
		||||
            $this->execute_remote_command([
 | 
			
		||||
                "docker images -q {$this->production_image_name} 2>/dev/null", "hidden" => true, "save" => "local_image_found"
 | 
			
		||||
            ]);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    private function save_environment_variables()
 | 
			
		||||
    {
 | 
			
		||||
        $envs = collect([]);
 | 
			
		||||
        if ($this->pull_request_id !== 0) {
 | 
			
		||||
            foreach ($this->application->environment_variables_preview as $env) {
 | 
			
		||||
                $envs->push($env->key . '=' . $env->real_value);
 | 
			
		||||
            }
 | 
			
		||||
        } else {
 | 
			
		||||
            foreach ($this->application->environment_variables as $env) {
 | 
			
		||||
                $envs->push($env->key . '=' . $env->real_value);
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        $envs_base64 = base64_encode($envs->implode("\n"));
 | 
			
		||||
        $this->execute_remote_command(
 | 
			
		||||
            [
 | 
			
		||||
                executeInDocker($this->deployment_uuid, "echo '$envs_base64' | base64 -d > $this->workdir/.env")
 | 
			
		||||
            ],
 | 
			
		||||
        );
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    private function framework_based_notification()
 | 
			
		||||
    {
 | 
			
		||||
        // Laravel old env variables
 | 
			
		||||
@@ -630,9 +696,6 @@ class ApplicationDeploymentJob implements ShouldQueue, ShouldBeEncrypted
 | 
			
		||||
    private function rolling_update()
 | 
			
		||||
    {
 | 
			
		||||
        if ($this->server->isSwarm()) {
 | 
			
		||||
            if ($this->build_pack !== 'dockerimage') {
 | 
			
		||||
                $this->push_to_docker_registry(forceFail: true);
 | 
			
		||||
            }
 | 
			
		||||
            $this->application_deployment_queue->addLogEntry("Rolling update started.");
 | 
			
		||||
            $this->execute_remote_command(
 | 
			
		||||
                [
 | 
			
		||||
@@ -642,7 +705,6 @@ class ApplicationDeploymentJob implements ShouldQueue, ShouldBeEncrypted
 | 
			
		||||
            $this->application_deployment_queue->addLogEntry("Rolling update completed.");
 | 
			
		||||
        } else {
 | 
			
		||||
            if ($this->use_build_server) {
 | 
			
		||||
                $this->push_to_docker_registry(forceFail: true);
 | 
			
		||||
                $this->write_deployment_configurations();
 | 
			
		||||
                $this->server = $this->original_server;
 | 
			
		||||
            }
 | 
			
		||||
@@ -787,10 +849,10 @@ class ApplicationDeploymentJob implements ShouldQueue, ShouldBeEncrypted
 | 
			
		||||
    }
 | 
			
		||||
    private function deploy_to_additional_destinations()
 | 
			
		||||
    {
 | 
			
		||||
        if (str($this->application->additional_destinations)->isEmpty()) {
 | 
			
		||||
        if ($this->application->additional_networks->count() === 0) {
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
        $destination_ids = collect(str($this->application->additional_destinations)->explode(','));
 | 
			
		||||
        $destination_ids = $this->application->additional_networks->pluck('id');
 | 
			
		||||
        if ($this->server->isSwarm()) {
 | 
			
		||||
            $this->application_deployment_queue->addLogEntry("Additional destinations are not supported in swarm mode.");
 | 
			
		||||
            return;
 | 
			
		||||
@@ -1529,7 +1591,7 @@ COPY ./nginx.conf /etc/nginx/conf.d/default.conf");
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
        if ($status === ApplicationDeploymentStatus::FINISHED->value) {
 | 
			
		||||
            // $this->deploy_to_additional_destinations();
 | 
			
		||||
            $this->deploy_to_additional_destinations();
 | 
			
		||||
            $this->application->environment->project->team?->notify(new DeploymentSuccess($this->application, $this->deployment_uuid, $this->preview));
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
@@ -1542,10 +1604,14 @@ COPY ./nginx.conf /etc/nginx/conf.d/default.conf");
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if ($this->application->build_pack !== 'dockercompose') {
 | 
			
		||||
            $this->application_deployment_queue->addLogEntry("Deployment failed. Removing the new version of your application.", 'stderr');
 | 
			
		||||
            $this->execute_remote_command(
 | 
			
		||||
                [executeInDocker($this->deployment_uuid, "docker rm -f $this->container_name >/dev/null 2>&1"), "hidden" => true, "ignore_errors" => true]
 | 
			
		||||
            );
 | 
			
		||||
            $code = $exception->getCode();
 | 
			
		||||
            if ($code !== 69420) {
 | 
			
		||||
                // 69420 means failed to push the image to the registry, so we don't need to remove the new version as it is the currently running one
 | 
			
		||||
                $this->application_deployment_queue->addLogEntry("Deployment failed. Removing the new version of your application.", 'stderr');
 | 
			
		||||
                $this->execute_remote_command(
 | 
			
		||||
                    [executeInDocker($this->deployment_uuid, "docker rm -f $this->container_name >/dev/null 2>&1"), "hidden" => true, "ignore_errors" => true]
 | 
			
		||||
                );
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        $this->next(ApplicationDeploymentStatus::FAILED->value);
 | 
			
		||||
 
 | 
			
		||||
@@ -5,6 +5,7 @@ namespace App\Jobs;
 | 
			
		||||
use App\Actions\Database\StartDatabaseProxy;
 | 
			
		||||
use App\Actions\Proxy\CheckProxy;
 | 
			
		||||
use App\Actions\Proxy\StartProxy;
 | 
			
		||||
use App\Actions\Shared\ComplexStatusCheck;
 | 
			
		||||
use App\Models\ApplicationPreview;
 | 
			
		||||
use App\Models\Server;
 | 
			
		||||
use App\Notifications\Container\ContainerRestarted;
 | 
			
		||||
@@ -42,6 +43,19 @@ class ContainerStatusJob implements ShouldQueue, ShouldBeEncrypted
 | 
			
		||||
 | 
			
		||||
    public function handle()
 | 
			
		||||
    {
 | 
			
		||||
        $applications = $this->server->applications();
 | 
			
		||||
        foreach ($applications as $application) {
 | 
			
		||||
            if ($application->additional_servers->count() > 0) {
 | 
			
		||||
                $is_main_server = $application->destination->server->id === $this->server->id;
 | 
			
		||||
                if ($is_main_server) {
 | 
			
		||||
                    ComplexStatusCheck::run($application);
 | 
			
		||||
                    $applications = $applications->filter(function ($value, $key) use ($application) {
 | 
			
		||||
                        return $value->id !== $application->id;
 | 
			
		||||
                    });
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (!$this->server->isFunctional()) {
 | 
			
		||||
            return 'Server is not ready.';
 | 
			
		||||
        };
 | 
			
		||||
@@ -83,7 +97,6 @@ class ContainerStatusJob implements ShouldQueue, ShouldBeEncrypted
 | 
			
		||||
                    });
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
            $applications = $this->server->applications();
 | 
			
		||||
            $databases = $this->server->databases();
 | 
			
		||||
            $services = $this->server->services()->get();
 | 
			
		||||
            $previews = $this->server->previews();
 | 
			
		||||
@@ -160,10 +173,9 @@ class ContainerStatusJob implements ShouldQueue, ShouldBeEncrypted
 | 
			
		||||
                            // Notify user that this container should not be there.
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
                    if (data_get($container,'Name') === '/coolify-db') {
 | 
			
		||||
                    if (data_get($container, 'Name') === '/coolify-db') {
 | 
			
		||||
                        $foundDatabases[] = 0;
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                }
 | 
			
		||||
                $serviceLabelId = data_get($labels, 'coolify.serviceId');
 | 
			
		||||
                if ($serviceLabelId) {
 | 
			
		||||
@@ -209,7 +221,7 @@ class ContainerStatusJob implements ShouldQueue, ShouldBeEncrypted
 | 
			
		||||
            }
 | 
			
		||||
            $exitedServices = $exitedServices->unique('id');
 | 
			
		||||
            foreach ($exitedServices as $exitedService) {
 | 
			
		||||
                if ($exitedService->status === 'exited') {
 | 
			
		||||
                if (str($exitedService->status)->startsWith('exited')) {
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
                $name = data_get($exitedService, 'name');
 | 
			
		||||
@@ -231,7 +243,7 @@ class ContainerStatusJob implements ShouldQueue, ShouldBeEncrypted
 | 
			
		||||
            $notRunningApplications = $applications->pluck('id')->diff($foundApplications);
 | 
			
		||||
            foreach ($notRunningApplications as $applicationId) {
 | 
			
		||||
                $application = $applications->where('id', $applicationId)->first();
 | 
			
		||||
                if ($application->status === 'exited') {
 | 
			
		||||
                if (str($application->status)->startsWith('exited')) {
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
                $application->update(['status' => 'exited']);
 | 
			
		||||
@@ -256,7 +268,7 @@ class ContainerStatusJob implements ShouldQueue, ShouldBeEncrypted
 | 
			
		||||
            $notRunningApplicationPreviews = $previews->pluck('id')->diff($foundApplicationPreviews);
 | 
			
		||||
            foreach ($notRunningApplicationPreviews as $previewId) {
 | 
			
		||||
                $preview = $previews->where('id', $previewId)->first();
 | 
			
		||||
                if ($preview->status === 'exited') {
 | 
			
		||||
                if (str($preview->status)->startsWith('exited')) {
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
                $preview->update(['status' => 'exited']);
 | 
			
		||||
@@ -281,7 +293,7 @@ class ContainerStatusJob implements ShouldQueue, ShouldBeEncrypted
 | 
			
		||||
            $notRunningDatabases = $databases->pluck('id')->diff($foundDatabases);
 | 
			
		||||
            foreach ($notRunningDatabases as $database) {
 | 
			
		||||
                $database = $databases->where('id', $database)->first();
 | 
			
		||||
                if ($database->status === 'exited') {
 | 
			
		||||
                if (str($database->status)->startsWith('exited')) {
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
                $database->update(['status' => 'exited']);
 | 
			
		||||
 
 | 
			
		||||
@@ -19,6 +19,7 @@ use Illuminate\Contracts\Queue\ShouldQueue;
 | 
			
		||||
use Illuminate\Foundation\Bus\Dispatchable;
 | 
			
		||||
use Illuminate\Queue\InteractsWithQueue;
 | 
			
		||||
use Illuminate\Queue\SerializesModels;
 | 
			
		||||
use Illuminate\Support\Facades\Artisan;
 | 
			
		||||
 | 
			
		||||
class DeleteResourceJob implements ShouldQueue, ShouldBeEncrypted
 | 
			
		||||
{
 | 
			
		||||
@@ -49,8 +50,11 @@ class DeleteResourceJob implements ShouldQueue, ShouldBeEncrypted
 | 
			
		||||
                    break;
 | 
			
		||||
            }
 | 
			
		||||
        } catch (\Throwable $e) {
 | 
			
		||||
            ray($e->getMessage());
 | 
			
		||||
            send_internal_notification('ContainerStoppingJob failed with: ' . $e->getMessage());
 | 
			
		||||
            throw $e;
 | 
			
		||||
        } finally {
 | 
			
		||||
            Artisan::queue('cleanup:stucked-resources');
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -41,6 +41,15 @@ class ServerStatusJob implements ShouldQueue, ShouldBeEncrypted
 | 
			
		||||
            throw new \RuntimeException('Server is not ready.');
 | 
			
		||||
        };
 | 
			
		||||
        try {
 | 
			
		||||
            // $this->server->validateConnection();
 | 
			
		||||
            // $this->server->validateOS();
 | 
			
		||||
            // $docker_installed = $this->server->validateDockerEngine();
 | 
			
		||||
            // if (!$docker_installed) {
 | 
			
		||||
            //     $this->server->installDocker();
 | 
			
		||||
            //     $this->server->validateDockerEngine();
 | 
			
		||||
            // }
 | 
			
		||||
 | 
			
		||||
            // $this->server->validateDockerEngineVersion();
 | 
			
		||||
            if ($this->server->isFunctional()) {
 | 
			
		||||
                $this->cleanup(notify: false);
 | 
			
		||||
            }
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										41
									
								
								app/Livewire/Admin/Index.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								app/Livewire/Admin/Index.php
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,41 @@
 | 
			
		||||
<?php
 | 
			
		||||
 | 
			
		||||
namespace App\Livewire\Admin;
 | 
			
		||||
 | 
			
		||||
use App\Models\User;
 | 
			
		||||
use Illuminate\Support\Facades\Crypt;
 | 
			
		||||
use Livewire\Component;
 | 
			
		||||
 | 
			
		||||
class Index extends Component
 | 
			
		||||
{
 | 
			
		||||
    public $users = [];
 | 
			
		||||
    public function mount()
 | 
			
		||||
    {
 | 
			
		||||
        if (auth()->user()->id !== 0 && session('adminToken') === null) {
 | 
			
		||||
            return redirect()->route('dashboard');
 | 
			
		||||
        }
 | 
			
		||||
        $this->users = User::whereHas('teams', function ($query) {
 | 
			
		||||
            $query->whereRelation('subscription', 'stripe_subscription_id', '!=', null);
 | 
			
		||||
        })->get();
 | 
			
		||||
    }
 | 
			
		||||
    public function switchUser(int $user_id)
 | 
			
		||||
    {
 | 
			
		||||
        $user = User::find($user_id);
 | 
			
		||||
        auth()->login($user);
 | 
			
		||||
 | 
			
		||||
        if ($user_id === 0) {
 | 
			
		||||
            session()->forget('adminToken');
 | 
			
		||||
        } else {
 | 
			
		||||
            $token_payload = [
 | 
			
		||||
                'valid' => true,
 | 
			
		||||
            ];
 | 
			
		||||
            $token = Crypt::encrypt($token_payload);
 | 
			
		||||
            session(['adminToken' => $token]);
 | 
			
		||||
        }
 | 
			
		||||
        return refreshSession();
 | 
			
		||||
    }
 | 
			
		||||
    public function render()
 | 
			
		||||
    {
 | 
			
		||||
        return view('livewire.admin.index');
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -6,6 +6,7 @@ use App\Models\ApplicationDeploymentQueue;
 | 
			
		||||
use App\Models\Project;
 | 
			
		||||
use App\Models\Server;
 | 
			
		||||
use Illuminate\Support\Collection;
 | 
			
		||||
use Illuminate\Support\Facades\Artisan;
 | 
			
		||||
use Livewire\Component;
 | 
			
		||||
 | 
			
		||||
class Dashboard extends Component
 | 
			
		||||
@@ -19,6 +20,12 @@ class Dashboard extends Component
 | 
			
		||||
        $this->projects = Project::ownedByCurrentTeam()->get();
 | 
			
		||||
        $this->get_deployments();
 | 
			
		||||
    }
 | 
			
		||||
    public function cleanup_queue()
 | 
			
		||||
    {
 | 
			
		||||
        Artisan::queue('app:init', [
 | 
			
		||||
            '--cleanup-deployments' => 'true'
 | 
			
		||||
        ]);
 | 
			
		||||
    }
 | 
			
		||||
    public function get_deployments()
 | 
			
		||||
    {
 | 
			
		||||
        $this->deployments_per_server = ApplicationDeploymentQueue::whereIn("status", ["in_progress", "queued"])->whereIn("server_id", $this->servers->pluck("id"))->get([
 | 
			
		||||
 
 | 
			
		||||
@@ -48,6 +48,8 @@ class DeploymentNavbar extends Component
 | 
			
		||||
    {
 | 
			
		||||
        try {
 | 
			
		||||
            $kill_command = "docker rm -f {$this->application_deployment_queue->deployment_uuid}";
 | 
			
		||||
            $server_id = $this->application_deployment_queue->server_id ?? $this->application->destination->server_id;
 | 
			
		||||
            $server = Server::find($server_id);
 | 
			
		||||
            if ($this->application_deployment_queue->logs) {
 | 
			
		||||
                $previous_logs = json_decode($this->application_deployment_queue->logs, associative: true, flags: JSON_THROW_ON_ERROR);
 | 
			
		||||
 | 
			
		||||
@@ -63,8 +65,8 @@ class DeploymentNavbar extends Component
 | 
			
		||||
                $this->application_deployment_queue->update([
 | 
			
		||||
                    'logs' => json_encode($previous_logs, flags: JSON_THROW_ON_ERROR),
 | 
			
		||||
                ]);
 | 
			
		||||
                instant_remote_process([$kill_command], $this->server);
 | 
			
		||||
            }
 | 
			
		||||
            instant_remote_process([$kill_command], $server);
 | 
			
		||||
        } catch (\Throwable $e) {
 | 
			
		||||
            ray($e);
 | 
			
		||||
            return handleError($e, $this);
 | 
			
		||||
 
 | 
			
		||||
@@ -243,9 +243,11 @@ class General extends Component
 | 
			
		||||
                    return str($domain)->trim()->lower();
 | 
			
		||||
                });
 | 
			
		||||
                $domains = $domains->unique();
 | 
			
		||||
                foreach ($domains as $domain) {
 | 
			
		||||
                    if (!validate_dns_entry($domain, $this->application->destination->server)) {
 | 
			
		||||
                        $showToaster && $this->dispatch('error', "Validating DNS ($domain) failed.","Make sure you have added the DNS records correctly.<br><br>Check this <a target='_blank' class='text-white underline' href='https://coolify.io/docs/dns-settings'>documentation</a> for further help.");
 | 
			
		||||
                if ($this->application->additional_servers->count() === 0) {
 | 
			
		||||
                    foreach ($domains as $domain) {
 | 
			
		||||
                        if (!validate_dns_entry($domain, $this->application->destination->server)) {
 | 
			
		||||
                            $showToaster && $this->dispatch('error', "Validating DNS ($domain) failed.","Make sure you have added the DNS records correctly.<br><br>Check this <a target='_blank' class='text-white underline' href='https://coolify.io/docs/dns-settings'>documentation</a> for further help.");
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
                check_fqdn_usage($this->application);
 | 
			
		||||
 
 | 
			
		||||
@@ -3,6 +3,8 @@
 | 
			
		||||
namespace App\Livewire\Project\Application;
 | 
			
		||||
 | 
			
		||||
use App\Actions\Application\StopApplication;
 | 
			
		||||
use App\Events\ApplicationStatusChanged;
 | 
			
		||||
use App\Jobs\ComplexContainerStatusJob;
 | 
			
		||||
use App\Jobs\ContainerStatusJob;
 | 
			
		||||
use App\Jobs\ServerStatusJob;
 | 
			
		||||
use App\Models\Application;
 | 
			
		||||
@@ -31,13 +33,14 @@ class Heading extends Component
 | 
			
		||||
    {
 | 
			
		||||
        if ($this->application->destination->server->isFunctional()) {
 | 
			
		||||
            dispatch(new ContainerStatusJob($this->application->destination->server));
 | 
			
		||||
            $this->application->refresh();
 | 
			
		||||
            $this->application->previews->each(function ($preview) {
 | 
			
		||||
                $preview->refresh();
 | 
			
		||||
            });
 | 
			
		||||
            // $this->application->refresh();
 | 
			
		||||
            // $this->application->previews->each(function ($preview) {
 | 
			
		||||
            //     $preview->refresh();
 | 
			
		||||
            // });
 | 
			
		||||
        } else {
 | 
			
		||||
            dispatch(new ServerStatusJob($this->application->destination->server));
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if ($showNotification) $this->dispatch('success', "Application status updated.");
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
@@ -49,15 +52,19 @@ class Heading extends Component
 | 
			
		||||
    public function deploy(bool $force_rebuild = false)
 | 
			
		||||
    {
 | 
			
		||||
        if ($this->application->build_pack === 'dockercompose' && is_null($this->application->docker_compose_raw)) {
 | 
			
		||||
            $this->dispatch('error', 'Please load a Compose file first.');
 | 
			
		||||
            $this->dispatch('error', 'Failed to deploy', 'Please load a Compose file first.');
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
        if ($this->application->destination->server->isSwarm() && is_null($this->application->docker_registry_image_name)) {
 | 
			
		||||
            $this->dispatch('error', 'To deploy to a Swarm cluster you must set a Docker image name first.');
 | 
			
		||||
        if ($this->application->destination->server->isSwarm() && str($this->application->docker_registry_image_name)->isEmpty()) {
 | 
			
		||||
            $this->dispatch('error', 'Failed to deploy', 'To deploy to a Swarm cluster you must set a Docker image name first.');
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
        if (data_get($this->application, 'settings.is_build_server_enabled') && is_null($this->application->docker_registry_image_name)) {
 | 
			
		||||
            $this->dispatch('error', 'To use a build server you must set a Docker image name first.<br>More information here: <a target="_blank" class="underline" href="https://coolify.io/docs/server/build-server">documentation</a>');
 | 
			
		||||
        if (data_get($this->application, 'settings.is_build_server_enabled') && str($this->application->docker_registry_image_name)->isEmpty()) {
 | 
			
		||||
            $this->dispatch('error', 'Failed to deploy', 'To use a build server, you must first set a Docker image.<br>More information here: <a target="_blank" class="underline" href="https://coolify.io/docs/server/build-server">documentation</a>');
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
        if ($this->application->additional_servers->count() > 0 && str($this->application->docker_registry_image_name)->isEmpty()) {
 | 
			
		||||
            $this->dispatch('error', 'Failed to deploy', 'To deploy to more than one server, you must first set a Docker image.<br>More information here: <a target="_blank" class="underline" href="https://coolify.io/docs/server/build-server">documentation</a>');
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
        $this->setDeploymentUuid();
 | 
			
		||||
@@ -85,26 +92,20 @@ class Heading extends Component
 | 
			
		||||
        StopApplication::run($this->application);
 | 
			
		||||
        $this->application->status = 'exited';
 | 
			
		||||
        $this->application->save();
 | 
			
		||||
        $this->application->refresh();
 | 
			
		||||
    }
 | 
			
		||||
    public function restartNew()
 | 
			
		||||
    {
 | 
			
		||||
        $this->setDeploymentUuid();
 | 
			
		||||
        queue_application_deployment(
 | 
			
		||||
            application: $this->application,
 | 
			
		||||
            deployment_uuid: $this->deploymentUuid,
 | 
			
		||||
            restart_only: true,
 | 
			
		||||
            is_new_deployment: true,
 | 
			
		||||
        );
 | 
			
		||||
        return redirect()->route('project.application.deployment.show', [
 | 
			
		||||
            'project_uuid' => $this->parameters['project_uuid'],
 | 
			
		||||
            'application_uuid' => $this->parameters['application_uuid'],
 | 
			
		||||
            'deployment_uuid' => $this->deploymentUuid,
 | 
			
		||||
            'environment_name' => $this->parameters['environment_name'],
 | 
			
		||||
        ]);
 | 
			
		||||
        if ($this->application->additional_servers->count() > 0) {
 | 
			
		||||
            $this->application->additional_servers->each(function ($server) {
 | 
			
		||||
                $server->pivot->status = "exited:unhealthy";
 | 
			
		||||
                $server->pivot->save();
 | 
			
		||||
            });
 | 
			
		||||
        }
 | 
			
		||||
        ApplicationStatusChanged::dispatch(data_get($this->application, 'environment.project.team.id'));
 | 
			
		||||
    }
 | 
			
		||||
    public function restart()
 | 
			
		||||
    {
 | 
			
		||||
        if ($this->application->additional_servers->count() > 0 && str($this->application->docker_registry_image_name)->isEmpty()) {
 | 
			
		||||
            $this->dispatch('error', 'Failed to deploy', 'To deploy to more than one server, you must first set a Docker image.<br>More information here: <a target="_blank" class="underline" href="https://coolify.io/docs/server/build-server">documentation</a>');
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
        $this->setDeploymentUuid();
 | 
			
		||||
        queue_application_deployment(
 | 
			
		||||
            application: $this->application,
 | 
			
		||||
 
 | 
			
		||||
@@ -2,11 +2,80 @@
 | 
			
		||||
 | 
			
		||||
namespace App\Livewire\Project\Shared;
 | 
			
		||||
 | 
			
		||||
use App\Actions\Application\StopApplicationOneServer;
 | 
			
		||||
use App\Events\ApplicationStatusChanged;
 | 
			
		||||
use App\Models\Server;
 | 
			
		||||
use App\Models\StandaloneDocker;
 | 
			
		||||
use Livewire\Component;
 | 
			
		||||
use Visus\Cuid2\Cuid2;
 | 
			
		||||
 | 
			
		||||
class Destination extends Component
 | 
			
		||||
{
 | 
			
		||||
    public $resource;
 | 
			
		||||
    public $servers = [];
 | 
			
		||||
    public $additional_servers = [];
 | 
			
		||||
    public $networks = [];
 | 
			
		||||
 | 
			
		||||
    public function getListeners()
 | 
			
		||||
    {
 | 
			
		||||
        $teamId = auth()->user()->currentTeam()->id;
 | 
			
		||||
        return [
 | 
			
		||||
            "echo-private:team.{$teamId},ApplicationStatusChanged" => 'loadData',
 | 
			
		||||
        ];
 | 
			
		||||
    }
 | 
			
		||||
    public function mount()
 | 
			
		||||
    {
 | 
			
		||||
        $this->loadData();
 | 
			
		||||
    }
 | 
			
		||||
    public function loadData()
 | 
			
		||||
    {
 | 
			
		||||
        $all_networks = collect([]);
 | 
			
		||||
        $all_networks = $all_networks->push($this->resource->destination);
 | 
			
		||||
        $all_networks = $all_networks->merge($this->resource->additional_networks);
 | 
			
		||||
 | 
			
		||||
        $this->networks = Server::isUsable()->get()->map(function ($server) {
 | 
			
		||||
            return $server->standaloneDockers;
 | 
			
		||||
        })->flatten();
 | 
			
		||||
        $this->networks = $this->networks->reject(function ($network) use ($all_networks) {
 | 
			
		||||
            return $all_networks->pluck('id')->contains($network->id);
 | 
			
		||||
        });
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
    public function redeploy(int $network_id, int $server_id)
 | 
			
		||||
    {
 | 
			
		||||
        $deployment_uuid = new Cuid2(7);
 | 
			
		||||
        $server = Server::find($server_id);
 | 
			
		||||
        $destination = StandaloneDocker::find($network_id);
 | 
			
		||||
        queue_application_deployment(
 | 
			
		||||
            deployment_uuid: $deployment_uuid,
 | 
			
		||||
            application: $this->resource,
 | 
			
		||||
            server: $server,
 | 
			
		||||
            destination: $destination,
 | 
			
		||||
            no_questions_asked: true,
 | 
			
		||||
        );
 | 
			
		||||
        return redirect()->route('project.application.deployment.show', [
 | 
			
		||||
            'project_uuid' => data_get($this->resource, 'environment.project.uuid'),
 | 
			
		||||
            'application_uuid' => data_get($this->resource, 'uuid'),
 | 
			
		||||
            'deployment_uuid' => $deployment_uuid,
 | 
			
		||||
            'environment_name' => data_get($this->resource, 'environment.name'),
 | 
			
		||||
        ]);
 | 
			
		||||
    }
 | 
			
		||||
    public function addServer(int $network_id, int $server_id)
 | 
			
		||||
    {
 | 
			
		||||
        $this->resource->additional_networks()->attach($network_id, ['server_id' => $server_id]);
 | 
			
		||||
        $this->resource->load(['additional_networks']);
 | 
			
		||||
        ApplicationStatusChanged::dispatch(data_get($this->resource, 'environment.project.team.id'));
 | 
			
		||||
        $this->loadData();
 | 
			
		||||
    }
 | 
			
		||||
    public function removeServer(int $network_id, int $server_id)
 | 
			
		||||
    {
 | 
			
		||||
        if ($this->resource->destination->server->id == $server_id) {
 | 
			
		||||
            $this->dispatch('error', 'You cannot remove this destination server.', 'You are trying to remove the main server.');
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
        $server = Server::find($server_id);
 | 
			
		||||
        StopApplicationOneServer::run($this->resource, $server);
 | 
			
		||||
        $this->resource->additional_networks()->detach($network_id, ['server_id' => $server_id]);
 | 
			
		||||
        $this->resource->load(['additional_networks']);
 | 
			
		||||
        ApplicationStatusChanged::dispatch(data_get($this->resource, 'environment.project.team.id'));
 | 
			
		||||
        $this->loadData();
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -171,7 +171,7 @@ class All extends Component
 | 
			
		||||
            }
 | 
			
		||||
            $environment->save();
 | 
			
		||||
            $this->refreshEnvs();
 | 
			
		||||
            $this->dispatch('success', 'Environment variable added successfully.');
 | 
			
		||||
            $this->dispatch('success', 'Environment variable added.');
 | 
			
		||||
        } catch (\Throwable $e) {
 | 
			
		||||
            return handleError($e, $this);
 | 
			
		||||
        }
 | 
			
		||||
 
 | 
			
		||||
@@ -28,6 +28,7 @@ class Form extends Component
 | 
			
		||||
        'server.settings.is_swarm_worker' => 'required|boolean',
 | 
			
		||||
        'server.settings.is_build_server' => 'required|boolean',
 | 
			
		||||
        'server.settings.concurrent_builds' => 'required|integer|min:1',
 | 
			
		||||
        'server.settings.dynamic_timeout' => 'required|integer|min:1',
 | 
			
		||||
        'wildcard_domain' => 'nullable|url',
 | 
			
		||||
    ];
 | 
			
		||||
    protected $validationAttributes = [
 | 
			
		||||
@@ -42,6 +43,8 @@ class Form extends Component
 | 
			
		||||
        'server.settings.is_swarm_worker' => 'Swarm Worker',
 | 
			
		||||
        'server.settings.is_build_server' => 'Build Server',
 | 
			
		||||
        'server.settings.concurrent_builds' => 'Concurrent Builds',
 | 
			
		||||
        'server.settings.dynamic_timeout' => 'Dynamic Timeout',
 | 
			
		||||
 | 
			
		||||
    ];
 | 
			
		||||
 | 
			
		||||
    public function mount()
 | 
			
		||||
 
 | 
			
		||||
@@ -15,7 +15,6 @@ class Application extends BaseModel
 | 
			
		||||
{
 | 
			
		||||
    use SoftDeletes;
 | 
			
		||||
    protected $guarded = [];
 | 
			
		||||
 | 
			
		||||
    protected static function booted()
 | 
			
		||||
    {
 | 
			
		||||
        static::saving(function ($application) {
 | 
			
		||||
@@ -53,6 +52,16 @@ class Application extends BaseModel
 | 
			
		||||
        });
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public function additional_servers()
 | 
			
		||||
    {
 | 
			
		||||
        return $this->belongsToMany(Server::class, 'additional_destinations')
 | 
			
		||||
            ->withPivot('standalone_docker_id', 'status');
 | 
			
		||||
    }
 | 
			
		||||
    public function additional_networks()
 | 
			
		||||
    {
 | 
			
		||||
        return $this->belongsToMany(StandaloneDocker::class, 'additional_destinations')
 | 
			
		||||
            ->withPivot('server_id', 'status');
 | 
			
		||||
    }
 | 
			
		||||
    public function is_github_based(): bool
 | 
			
		||||
    {
 | 
			
		||||
        if (data_get($this, 'source')) {
 | 
			
		||||
@@ -203,6 +212,79 @@ class Application extends BaseModel
 | 
			
		||||
 | 
			
		||||
        );
 | 
			
		||||
    }
 | 
			
		||||
    public function realStatus()
 | 
			
		||||
    {
 | 
			
		||||
       return $this->getRawOriginal('status');
 | 
			
		||||
    }
 | 
			
		||||
    public function status(): Attribute
 | 
			
		||||
    {
 | 
			
		||||
        return Attribute::make(
 | 
			
		||||
            set: function ($value) {
 | 
			
		||||
                if ($this->additional_servers->count() === 0) {
 | 
			
		||||
                    if (str($value)->contains('(')) {
 | 
			
		||||
                        $status = str($value)->before('(')->trim()->value();
 | 
			
		||||
                        $health = str($value)->after('(')->before(')')->trim()->value() ?? 'unhealthy';
 | 
			
		||||
                    } else if (str($value)->contains(':')) {
 | 
			
		||||
                        $status = str($value)->before(':')->trim()->value();
 | 
			
		||||
                        $health = str($value)->after(':')->trim()->value() ?? 'unhealthy';
 | 
			
		||||
                    } else {
 | 
			
		||||
                        $status = $value;
 | 
			
		||||
                        $health = 'unhealthy';
 | 
			
		||||
                    }
 | 
			
		||||
                    return "$status:$health";
 | 
			
		||||
                } else {
 | 
			
		||||
                    if (str($value)->contains('(')) {
 | 
			
		||||
                        $status = str($value)->before('(')->trim()->value();
 | 
			
		||||
                        $health = str($value)->after('(')->before(')')->trim()->value() ?? 'unhealthy';
 | 
			
		||||
                    } else if (str($value)->contains(':')) {
 | 
			
		||||
                        $status = str($value)->before(':')->trim()->value();
 | 
			
		||||
                        $health = str($value)->after(':')->trim()->value() ?? 'unhealthy';
 | 
			
		||||
                    } else {
 | 
			
		||||
                        $status = $value;
 | 
			
		||||
                        $health = 'unhealthy';
 | 
			
		||||
                    }
 | 
			
		||||
                    return "$status:$health";
 | 
			
		||||
                }
 | 
			
		||||
            },
 | 
			
		||||
            get: function ($value) {
 | 
			
		||||
                if ($this->additional_servers->count() === 0) {
 | 
			
		||||
                    //running (healthy)
 | 
			
		||||
                    if (str($value)->contains('(')) {
 | 
			
		||||
                        $status = str($value)->before('(')->trim()->value();
 | 
			
		||||
                        $health = str($value)->after('(')->before(')')->trim()->value() ?? 'unhealthy';
 | 
			
		||||
                    } else if (str($value)->contains(':')) {
 | 
			
		||||
                        $status = str($value)->before(':')->trim()->value();
 | 
			
		||||
                        $health = str($value)->after(':')->trim()->value() ?? 'unhealthy';
 | 
			
		||||
                    } else {
 | 
			
		||||
                        $status = $value;
 | 
			
		||||
                        $health = 'unhealthy';
 | 
			
		||||
                    }
 | 
			
		||||
                    return "$status:$health";
 | 
			
		||||
                } else {
 | 
			
		||||
                    $complex_status = null;
 | 
			
		||||
                    $complex_health = null;
 | 
			
		||||
                    $complex_status = $main_server_status = str($value)->before(':')->value();
 | 
			
		||||
                    $complex_health = $main_server_health = str($value)->after(':')->value() ?? 'unhealthy';
 | 
			
		||||
                    $additional_servers_status = $this->additional_servers->pluck('pivot.status');
 | 
			
		||||
                    foreach ($additional_servers_status as $status) {
 | 
			
		||||
                        $server_status = str($status)->before(':')->value();
 | 
			
		||||
                        $server_health = str($status)->after(':')->value() ?? 'unhealthy';
 | 
			
		||||
                        if ($server_status !== 'running') {
 | 
			
		||||
                            if ($main_server_status !== $server_status) {
 | 
			
		||||
                                $complex_status = 'degraded';
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                        if ($server_health !== 'healthy') {
 | 
			
		||||
                            if ($main_server_health !== $server_health) {
 | 
			
		||||
                                $complex_health = 'unhealthy';
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
                    return "$complex_status:$complex_health";
 | 
			
		||||
                }
 | 
			
		||||
            },
 | 
			
		||||
        );
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public function portsExposesArray(): Attribute
 | 
			
		||||
    {
 | 
			
		||||
@@ -216,7 +298,8 @@ class Application extends BaseModel
 | 
			
		||||
    {
 | 
			
		||||
        return $this->morphToMany(Tag::class, 'taggable');
 | 
			
		||||
    }
 | 
			
		||||
    public function project() {
 | 
			
		||||
    public function project()
 | 
			
		||||
    {
 | 
			
		||||
        return data_get($this, 'environment.project');
 | 
			
		||||
    }
 | 
			
		||||
    public function team()
 | 
			
		||||
@@ -435,7 +518,7 @@ class Application extends BaseModel
 | 
			
		||||
    {
 | 
			
		||||
        return "/artifacts/{$uuid}";
 | 
			
		||||
    }
 | 
			
		||||
       function setGitImportSettings(string $deployment_uuid, string $git_clone_command)
 | 
			
		||||
    function setGitImportSettings(string $deployment_uuid, string $git_clone_command)
 | 
			
		||||
    {
 | 
			
		||||
        $baseDir = $this->generateBaseDir($deployment_uuid);
 | 
			
		||||
        if ($this->git_commit_sha !== 'HEAD') {
 | 
			
		||||
 
 | 
			
		||||
@@ -9,6 +9,7 @@ use App\Notifications\Server\Revived;
 | 
			
		||||
use App\Notifications\Server\Unreachable;
 | 
			
		||||
use Illuminate\Database\Eloquent\Builder;
 | 
			
		||||
use Illuminate\Database\Eloquent\Casts\Attribute;
 | 
			
		||||
use Illuminate\Support\Facades\DB;
 | 
			
		||||
use Spatie\SchemalessAttributes\Casts\SchemalessAttributes;
 | 
			
		||||
use Spatie\SchemalessAttributes\SchemalessAttributesTrait;
 | 
			
		||||
use Illuminate\Support\Str;
 | 
			
		||||
@@ -248,9 +249,17 @@ class Server extends BaseModel
 | 
			
		||||
    }
 | 
			
		||||
    public function applications()
 | 
			
		||||
    {
 | 
			
		||||
        return $this->destinations()->map(function ($standaloneDocker) {
 | 
			
		||||
        $applications = $this->destinations()->map(function ($standaloneDocker) {
 | 
			
		||||
            return $standaloneDocker->applications;
 | 
			
		||||
        })->flatten();
 | 
			
		||||
        $additionalApplicationIds = DB::table('additional_destinations')->where('server_id', $this->id)->get('application_id');
 | 
			
		||||
        $additionalApplicationIds = collect($additionalApplicationIds)->map(function ($item) {
 | 
			
		||||
            return $item->application_id;
 | 
			
		||||
        });
 | 
			
		||||
        Application::whereIn('id', $additionalApplicationIds)->get()->each(function ($application) use ($applications) {
 | 
			
		||||
            $applications->push($application);
 | 
			
		||||
        });
 | 
			
		||||
        return $applications;
 | 
			
		||||
    }
 | 
			
		||||
    public function dockerComposeBasedApplications()
 | 
			
		||||
    {
 | 
			
		||||
@@ -300,7 +309,8 @@ class Server extends BaseModel
 | 
			
		||||
    {
 | 
			
		||||
        $standalone_docker = $this->hasMany(StandaloneDocker::class)->get();
 | 
			
		||||
        $swarm_docker = $this->hasMany(SwarmDocker::class)->get();
 | 
			
		||||
        return $standalone_docker->concat($swarm_docker);
 | 
			
		||||
        $asd = $this->belongsToMany(StandaloneDocker::class, 'additional_destinations')->withPivot('server_id')->get();
 | 
			
		||||
        return $standalone_docker->concat($swarm_docker)->concat($asd);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public function standaloneDockers()
 | 
			
		||||
 
 | 
			
		||||
@@ -43,7 +43,41 @@ class StandaloneMariadb extends BaseModel
 | 
			
		||||
            $database->tags()->detach();
 | 
			
		||||
        });
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public function realStatus()
 | 
			
		||||
    {
 | 
			
		||||
       return $this->getRawOriginal('status');
 | 
			
		||||
    }
 | 
			
		||||
    public function status(): Attribute
 | 
			
		||||
    {
 | 
			
		||||
        return Attribute::make(
 | 
			
		||||
            set: function ($value) {
 | 
			
		||||
                if (str($value)->contains('(')) {
 | 
			
		||||
                    $status = str($value)->before('(')->trim()->value();
 | 
			
		||||
                    $health = str($value)->after('(')->before(')')->trim()->value() ?? 'unhealthy';
 | 
			
		||||
                } else if (str($value)->contains(':')) {
 | 
			
		||||
                    $status = str($value)->before(':')->trim()->value();
 | 
			
		||||
                    $health = str($value)->after(':')->trim()->value() ?? 'unhealthy';
 | 
			
		||||
                } else {
 | 
			
		||||
                    $status = $value;
 | 
			
		||||
                    $health = 'unhealthy';
 | 
			
		||||
                }
 | 
			
		||||
                return "$status:$health";
 | 
			
		||||
            },
 | 
			
		||||
            get: function ($value) {
 | 
			
		||||
                if (str($value)->contains('(')) {
 | 
			
		||||
                    $status = str($value)->before('(')->trim()->value();
 | 
			
		||||
                    $health = str($value)->after('(')->before(')')->trim()->value() ?? 'unhealthy';
 | 
			
		||||
                } else if (str($value)->contains(':')) {
 | 
			
		||||
                    $status = str($value)->before(':')->trim()->value();
 | 
			
		||||
                    $health = str($value)->after(':')->trim()->value() ?? 'unhealthy';
 | 
			
		||||
                } else {
 | 
			
		||||
                    $status = $value;
 | 
			
		||||
                    $health = 'unhealthy';
 | 
			
		||||
                }
 | 
			
		||||
                return "$status:$health";
 | 
			
		||||
            },
 | 
			
		||||
        );
 | 
			
		||||
    }
 | 
			
		||||
    public function tags()
 | 
			
		||||
    {
 | 
			
		||||
        return $this->morphToMany(Tag::class, 'taggable');
 | 
			
		||||
 
 | 
			
		||||
@@ -46,7 +46,41 @@ class StandaloneMongodb extends BaseModel
 | 
			
		||||
            $database->tags()->detach();
 | 
			
		||||
        });
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public function realStatus()
 | 
			
		||||
    {
 | 
			
		||||
       return $this->getRawOriginal('status');
 | 
			
		||||
    }
 | 
			
		||||
    public function status(): Attribute
 | 
			
		||||
    {
 | 
			
		||||
        return Attribute::make(
 | 
			
		||||
            set: function ($value) {
 | 
			
		||||
                if (str($value)->contains('(')) {
 | 
			
		||||
                    $status = str($value)->before('(')->trim()->value();
 | 
			
		||||
                    $health = str($value)->after('(')->before(')')->trim()->value() ?? 'unhealthy';
 | 
			
		||||
                } else if (str($value)->contains(':')) {
 | 
			
		||||
                    $status = str($value)->before(':')->trim()->value();
 | 
			
		||||
                    $health = str($value)->after(':')->trim()->value() ?? 'unhealthy';
 | 
			
		||||
                } else {
 | 
			
		||||
                    $status = $value;
 | 
			
		||||
                    $health = 'unhealthy';
 | 
			
		||||
                }
 | 
			
		||||
                return "$status:$health";
 | 
			
		||||
            },
 | 
			
		||||
            get: function ($value) {
 | 
			
		||||
                if (str($value)->contains('(')) {
 | 
			
		||||
                    $status = str($value)->before('(')->trim()->value();
 | 
			
		||||
                    $health = str($value)->after('(')->before(')')->trim()->value() ?? 'unhealthy';
 | 
			
		||||
                } else if (str($value)->contains(':')) {
 | 
			
		||||
                    $status = str($value)->before(':')->trim()->value();
 | 
			
		||||
                    $health = str($value)->after(':')->trim()->value() ?? 'unhealthy';
 | 
			
		||||
                } else {
 | 
			
		||||
                    $status = $value;
 | 
			
		||||
                    $health = 'unhealthy';
 | 
			
		||||
                }
 | 
			
		||||
                return "$status:$health";
 | 
			
		||||
            },
 | 
			
		||||
        );
 | 
			
		||||
    }
 | 
			
		||||
    public function tags()
 | 
			
		||||
    {
 | 
			
		||||
        return $this->morphToMany(Tag::class, 'taggable');
 | 
			
		||||
 
 | 
			
		||||
@@ -43,7 +43,41 @@ class StandaloneMysql extends BaseModel
 | 
			
		||||
            $database->tags()->detach();
 | 
			
		||||
        });
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public function realStatus()
 | 
			
		||||
    {
 | 
			
		||||
       return $this->getRawOriginal('status');
 | 
			
		||||
    }
 | 
			
		||||
    public function status(): Attribute
 | 
			
		||||
    {
 | 
			
		||||
        return Attribute::make(
 | 
			
		||||
            set: function ($value) {
 | 
			
		||||
                if (str($value)->contains('(')) {
 | 
			
		||||
                    $status = str($value)->before('(')->trim()->value();
 | 
			
		||||
                    $health = str($value)->after('(')->before(')')->trim()->value() ?? 'unhealthy';
 | 
			
		||||
                } else if (str($value)->contains(':')) {
 | 
			
		||||
                    $status = str($value)->before(':')->trim()->value();
 | 
			
		||||
                    $health = str($value)->after(':')->trim()->value() ?? 'unhealthy';
 | 
			
		||||
                } else {
 | 
			
		||||
                    $status = $value;
 | 
			
		||||
                    $health = 'unhealthy';
 | 
			
		||||
                }
 | 
			
		||||
                return "$status:$health";
 | 
			
		||||
            },
 | 
			
		||||
            get: function ($value) {
 | 
			
		||||
                if (str($value)->contains('(')) {
 | 
			
		||||
                    $status = str($value)->before('(')->trim()->value();
 | 
			
		||||
                    $health = str($value)->after('(')->before(')')->trim()->value() ?? 'unhealthy';
 | 
			
		||||
                } else if (str($value)->contains(':')) {
 | 
			
		||||
                    $status = str($value)->before(':')->trim()->value();
 | 
			
		||||
                    $health = str($value)->after(':')->trim()->value() ?? 'unhealthy';
 | 
			
		||||
                } else {
 | 
			
		||||
                    $status = $value;
 | 
			
		||||
                    $health = 'unhealthy';
 | 
			
		||||
                }
 | 
			
		||||
                return "$status:$health";
 | 
			
		||||
            },
 | 
			
		||||
        );
 | 
			
		||||
    }
 | 
			
		||||
    public function tags()
 | 
			
		||||
    {
 | 
			
		||||
        return $this->morphToMany(Tag::class, 'taggable');
 | 
			
		||||
 
 | 
			
		||||
@@ -43,7 +43,41 @@ class StandalonePostgresql extends BaseModel
 | 
			
		||||
            $database->tags()->detach();
 | 
			
		||||
        });
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public function realStatus()
 | 
			
		||||
    {
 | 
			
		||||
        return $this->getRawOriginal('status');
 | 
			
		||||
    }
 | 
			
		||||
    public function status(): Attribute
 | 
			
		||||
    {
 | 
			
		||||
        return Attribute::make(
 | 
			
		||||
            set: function ($value) {
 | 
			
		||||
                if (str($value)->contains('(')) {
 | 
			
		||||
                    $status = str($value)->before('(')->trim()->value();
 | 
			
		||||
                    $health = str($value)->after('(')->before(')')->trim()->value() ?? 'unhealthy';
 | 
			
		||||
                } else if (str($value)->contains(':')) {
 | 
			
		||||
                    $status = str($value)->before(':')->trim()->value();
 | 
			
		||||
                    $health = str($value)->after(':')->trim()->value() ?? 'unhealthy';
 | 
			
		||||
                } else {
 | 
			
		||||
                    $status = $value;
 | 
			
		||||
                    $health = 'unhealthy';
 | 
			
		||||
                }
 | 
			
		||||
                return "$status:$health";
 | 
			
		||||
            },
 | 
			
		||||
            get: function ($value) {
 | 
			
		||||
                if (str($value)->contains('(')) {
 | 
			
		||||
                    $status = str($value)->before('(')->trim()->value();
 | 
			
		||||
                    $health = str($value)->after('(')->before(')')->trim()->value() ?? 'unhealthy';
 | 
			
		||||
                } else if (str($value)->contains(':')) {
 | 
			
		||||
                    $status = str($value)->before(':')->trim()->value();
 | 
			
		||||
                    $health = str($value)->after(':')->trim()->value() ?? 'unhealthy';
 | 
			
		||||
                } else {
 | 
			
		||||
                    $status = $value;
 | 
			
		||||
                    $health = 'unhealthy';
 | 
			
		||||
                }
 | 
			
		||||
                return "$status:$health";
 | 
			
		||||
            },
 | 
			
		||||
        );
 | 
			
		||||
    }
 | 
			
		||||
    public function tags()
 | 
			
		||||
    {
 | 
			
		||||
        return $this->morphToMany(Tag::class, 'taggable');
 | 
			
		||||
 
 | 
			
		||||
@@ -38,7 +38,41 @@ class StandaloneRedis extends BaseModel
 | 
			
		||||
            $database->tags()->detach();
 | 
			
		||||
        });
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public function realStatus()
 | 
			
		||||
    {
 | 
			
		||||
        return $this->getRawOriginal('status');
 | 
			
		||||
    }
 | 
			
		||||
    public function status(): Attribute
 | 
			
		||||
    {
 | 
			
		||||
        return Attribute::make(
 | 
			
		||||
            set: function ($value) {
 | 
			
		||||
                if (str($value)->contains('(')) {
 | 
			
		||||
                    $status = str($value)->before('(')->trim()->value();
 | 
			
		||||
                    $health = str($value)->after('(')->before(')')->trim()->value() ?? 'unhealthy';
 | 
			
		||||
                } else if (str($value)->contains(':')) {
 | 
			
		||||
                    $status = str($value)->before(':')->trim()->value();
 | 
			
		||||
                    $health = str($value)->after(':')->trim()->value() ?? 'unhealthy';
 | 
			
		||||
                } else {
 | 
			
		||||
                    $status = $value;
 | 
			
		||||
                    $health = 'unhealthy';
 | 
			
		||||
                }
 | 
			
		||||
                return "$status:$health";
 | 
			
		||||
            },
 | 
			
		||||
            get: function ($value) {
 | 
			
		||||
                if (str($value)->contains('(')) {
 | 
			
		||||
                    $status = str($value)->before('(')->trim()->value();
 | 
			
		||||
                    $health = str($value)->after('(')->before(')')->trim()->value() ?? 'unhealthy';
 | 
			
		||||
                } else if (str($value)->contains(':')) {
 | 
			
		||||
                    $status = str($value)->before(':')->trim()->value();
 | 
			
		||||
                    $health = str($value)->after(':')->trim()->value() ?? 'unhealthy';
 | 
			
		||||
                } else {
 | 
			
		||||
                    $status = $value;
 | 
			
		||||
                    $health = 'unhealthy';
 | 
			
		||||
                }
 | 
			
		||||
                return "$status:$health";
 | 
			
		||||
            },
 | 
			
		||||
        );
 | 
			
		||||
    }
 | 
			
		||||
    public function tags()
 | 
			
		||||
    {
 | 
			
		||||
        return $this->morphToMany(Tag::class, 'taggable');
 | 
			
		||||
 
 | 
			
		||||
@@ -104,7 +104,7 @@ function handleError(?Throwable $error = null, ?Livewire\Component $livewire = n
 | 
			
		||||
    ray($error);
 | 
			
		||||
    if ($error instanceof TooManyRequestsException) {
 | 
			
		||||
        if (isset($livewire)) {
 | 
			
		||||
            return $livewire->dispatch('error', "Too many requests.","Please try again in {$error->secondsUntilAvailable} seconds.");
 | 
			
		||||
            return $livewire->dispatch('error', "Too many requests.", "Please try again in {$error->secondsUntilAvailable} seconds.");
 | 
			
		||||
        }
 | 
			
		||||
        return "Too many requests. Please try again in {$error->secondsUntilAvailable} seconds.";
 | 
			
		||||
    }
 | 
			
		||||
@@ -1690,7 +1690,7 @@ function check_fqdn_usage(ServiceApplication|Application $own_resource)
 | 
			
		||||
            $naked_domain = str($domain)->replace('http://', '')->replace('https://', '')->value();
 | 
			
		||||
            if ($domains->contains($naked_domain)) {
 | 
			
		||||
                if ($app->uuid !== $own_resource->uuid) {
 | 
			
		||||
                    throw new \RuntimeException("Domain $naked_domain is already in use by another resource.");
 | 
			
		||||
                    throw new \RuntimeException("Domain $naked_domain is already in use by another resource:<br> {$app->name}.");
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 
 | 
			
		||||
@@ -65,7 +65,7 @@ return [
 | 
			
		||||
            'driver' => 'redis',
 | 
			
		||||
            'connection' => 'default',
 | 
			
		||||
            'queue' => env('REDIS_QUEUE', 'default'),
 | 
			
		||||
            'retry_after' => 3600,
 | 
			
		||||
            'retry_after' => 86400,
 | 
			
		||||
            'block_for' => null,
 | 
			
		||||
            'after_commit' => true,
 | 
			
		||||
        ],
 | 
			
		||||
 
 | 
			
		||||
@@ -7,7 +7,7 @@ return [
 | 
			
		||||
 | 
			
		||||
    // The release version of your application
 | 
			
		||||
    // Example with dynamic git hash: trim(exec('git --git-dir ' . base_path('.git') . ' log --pretty="%h" -n1 HEAD'))
 | 
			
		||||
    'release' => '4.0.0-beta.211',
 | 
			
		||||
    'release' => '4.0.0-beta.212',
 | 
			
		||||
    // When left empty or `null` the Laravel environment will be used
 | 
			
		||||
    'environment' => config('app.env'),
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -1,3 +1,3 @@
 | 
			
		||||
<?php
 | 
			
		||||
 | 
			
		||||
return '4.0.0-beta.211';
 | 
			
		||||
return '4.0.0-beta.212';
 | 
			
		||||
 
 | 
			
		||||
@@ -24,7 +24,6 @@ return new class extends Migration
 | 
			
		||||
            $table->string('taggable_type');
 | 
			
		||||
            $table->foreign('tag_id')->references('id')->on('tags')->onDelete('cascade');
 | 
			
		||||
            $table->unique(['tag_id', 'taggable_id', 'taggable_type'], 'taggable_unique'); // Composite unique index
 | 
			
		||||
 | 
			
		||||
        });
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -0,0 +1,37 @@
 | 
			
		||||
<?php
 | 
			
		||||
 | 
			
		||||
use Illuminate\Database\Migrations\Migration;
 | 
			
		||||
use Illuminate\Database\Schema\Blueprint;
 | 
			
		||||
use Illuminate\Support\Facades\Schema;
 | 
			
		||||
 | 
			
		||||
return new class extends Migration
 | 
			
		||||
{
 | 
			
		||||
    /**
 | 
			
		||||
     * Run the migrations.
 | 
			
		||||
     */
 | 
			
		||||
    public function up(): void
 | 
			
		||||
    {
 | 
			
		||||
        Schema::create('additional_destinations', function (Blueprint $table) {
 | 
			
		||||
            $table->id();
 | 
			
		||||
            $table->foreignId('application_id')->constrained()->onDelete('cascade');
 | 
			
		||||
            $table->foreignId('server_id')->constrained()->onDelete('cascade');
 | 
			
		||||
            $table->string('status')->default('exited');
 | 
			
		||||
            $table->foreignId('standalone_docker_id')->constrained()->onDelete('cascade');
 | 
			
		||||
            $table->timestamps();
 | 
			
		||||
        });
 | 
			
		||||
        Schema::table('applications', function (Blueprint $table) {
 | 
			
		||||
            $table->dropColumn('additional_destinations');
 | 
			
		||||
        });
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Reverse the migrations.
 | 
			
		||||
     */
 | 
			
		||||
    public function down(): void
 | 
			
		||||
    {
 | 
			
		||||
        Schema::dropIfExists('additional_destinations');
 | 
			
		||||
        Schema::table('applications', function (Blueprint $table) {
 | 
			
		||||
            $table->string('additional_destinations')->nullable()->after('destination');
 | 
			
		||||
        });
 | 
			
		||||
    }
 | 
			
		||||
};
 | 
			
		||||
@@ -0,0 +1,28 @@
 | 
			
		||||
<?php
 | 
			
		||||
 | 
			
		||||
use Illuminate\Database\Migrations\Migration;
 | 
			
		||||
use Illuminate\Database\Schema\Blueprint;
 | 
			
		||||
use Illuminate\Support\Facades\Schema;
 | 
			
		||||
 | 
			
		||||
return new class extends Migration
 | 
			
		||||
{
 | 
			
		||||
    /**
 | 
			
		||||
     * Run the migrations.
 | 
			
		||||
     */
 | 
			
		||||
    public function up(): void
 | 
			
		||||
    {
 | 
			
		||||
        Schema::table('server_settings', function (Blueprint $table) {
 | 
			
		||||
            $table->integer('dynamic_timeout')->default(3600);
 | 
			
		||||
        });
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * Reverse the migrations.
 | 
			
		||||
     */
 | 
			
		||||
    public function down(): void
 | 
			
		||||
    {
 | 
			
		||||
        Schema::table('server_settings', function (Blueprint $table) {
 | 
			
		||||
            $table->dropColumn('dynamic_timeout');
 | 
			
		||||
        });
 | 
			
		||||
    }
 | 
			
		||||
};
 | 
			
		||||
@@ -1,3 +1,3 @@
 | 
			
		||||
#!/command/execlineb -P
 | 
			
		||||
s6-setuidgid webuser
 | 
			
		||||
php /var/www/html/artisan app:init --cleanup
 | 
			
		||||
php /var/www/html/artisan app:init --full-cleanup
 | 
			
		||||
 
 | 
			
		||||
@@ -22,7 +22,7 @@
 | 
			
		||||
        </a>
 | 
			
		||||
    @endif
 | 
			
		||||
    <div class="flex-1"></div>
 | 
			
		||||
    @if ($database->status !== 'exited')
 | 
			
		||||
    @if (!str($database->status)->startsWith('exited'))
 | 
			
		||||
        <button wire:click='stop' class="flex items-center gap-2 cursor-pointer hover:text-white text-neutral-400">
 | 
			
		||||
            <svg xmlns="http://www.w3.org/2000/svg" class="w-5 h-5 text-error" viewBox="0 0 24 24" stroke-width="2"
 | 
			
		||||
                stroke="currentColor" fill="none" stroke-linecap="round" stroke-linejoin="round">
 | 
			
		||||
 
 | 
			
		||||
@@ -28,21 +28,16 @@
 | 
			
		||||
                <div class="justify-center" wire:click="help" onclick="help.showModal()">
 | 
			
		||||
                    <svg class="icon" viewBox="0 0 256 256" xmlns="http://www.w3.org/2000/svg">
 | 
			
		||||
                        <path fill="currentColor"
 | 
			
		||||
                            d="M144 180a16 16 0 1 1-16-16a16 16 0 0 1 16 16m92-52A108 108 0 1 1 128 20a108.12 108.12 0 0 1 108 108m-24 0a84 84 0 1 0-84 84a84.09 84.09 0 0 0 84-84m-84-64c-24.26 0-44 17.94-44 40v4a12 12 0 0 0 24 0v-4c0-8.82 9-16 20-16s20 7.18 20 16s-9 16-20 16a12 12 0 0 0-12 12v8a12 12 0 0 0 23.73 2.56C158.31 137.88 172 122.37 172 104c0-22.06-19.74-40-44-40" />
 | 
			
		||||
                            d="M140 180a12 12 0 1 1-12-12a12 12 0 0 1 12 12M128 72c-22.06 0-40 16.15-40 36v4a8 8 0 0 0 16 0v-4c0-11 10.77-20 24-20s24 9 24 20s-10.77 20-24 20a8 8 0 0 0-8 8v8a8 8 0 0 0 16 0v-.72c18.24-3.35 32-17.9 32-35.28c0-19.85-17.94-36-40-36m104 56A104 104 0 1 1 128 24a104.11 104.11 0 0 1 104 104m-16 0a88 88 0 1 0-88 88a88.1 88.1 0 0 0 88-88" />
 | 
			
		||||
                    </svg>
 | 
			
		||||
                </div>
 | 
			
		||||
            </li>
 | 
			
		||||
            <li class="pb-6" title="Logout">
 | 
			
		||||
                <form action="/logout" method="POST" class=" hover:bg-transparent">
 | 
			
		||||
                <form action="/logout" method="POST" class="hover:bg-transparent">
 | 
			
		||||
                    @csrf
 | 
			
		||||
                    <button class="flex items-center gap-2 rounded-none hover:text-white hover:bg-transparent">
 | 
			
		||||
                        <svg xmlns="http://www.w3.org/2000/svg" class="icon" viewBox="0 0 24 24" stroke-width="1.5"
 | 
			
		||||
                            stroke="currentColor" fill="none" stroke-linecap="round" stroke-linejoin="round">
 | 
			
		||||
                            <path stroke="none" d="M0 0h24v24H0z" fill="none" />
 | 
			
		||||
                            <path d="M13 12v.01" />
 | 
			
		||||
                            <path d="M3 21h18" />
 | 
			
		||||
                            <path d="M5 21v-16a2 2 0 0 1 2 -2h7.5m2.5 10.5v7.5" />
 | 
			
		||||
                            <path d="M14 7h7m-3 -3l3 3l-3 3" />
 | 
			
		||||
                        <svg class="icon" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg">
 | 
			
		||||
                            <path fill="currentColor" d="M12 22C6.477 22 2 17.523 2 12S6.477 2 12 2a9.985 9.985 0 0 1 8 4h-2.71a8 8 0 1 0 .001 12h2.71A9.985 9.985 0 0 1 12 22m7-6v-3h-8v-2h8V8l5 4z"/>
 | 
			
		||||
                        </svg>
 | 
			
		||||
                    </button>
 | 
			
		||||
                </form>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,5 +1,5 @@
 | 
			
		||||
@auth
 | 
			
		||||
    <nav class="fixed h-full overflow-hidden overflow-y-auto pt-28 scrollbar">
 | 
			
		||||
    <nav class="fixed h-full pt-28 scrollbar">
 | 
			
		||||
        <a href="/" class="fixed top-0 z-50 mx-3 mt-3 bg-transparent cursor-pointer"><img
 | 
			
		||||
                class="transition rounded w-11 h-11" src="{{ asset('coolify-transparent.png') }}"></a>
 | 
			
		||||
        <ul class="flex flex-col h-full gap-4 menu flex-nowrap">
 | 
			
		||||
@@ -40,62 +40,118 @@
 | 
			
		||||
                    </svg>
 | 
			
		||||
                </a>
 | 
			
		||||
            </li>
 | 
			
		||||
            <li title="Command Center">
 | 
			
		||||
                <a class="hover:bg-transparent" href="/command-center">
 | 
			
		||||
                    <svg xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                        class="{{ request()->is('command-center') ? 'text-warning icon' : 'icon' }}" viewBox="0 0 24 24"
 | 
			
		||||
                        stroke-width="1.5" stroke="currentColor" fill="none" stroke-linecap="round"
 | 
			
		||||
                        stroke-linejoin="round">
 | 
			
		||||
                        <path stroke="none" d="M0 0h24v24H0z" fill="none" />
 | 
			
		||||
                        <path d="M5 7l5 5l-5 5" />
 | 
			
		||||
                        <path d="M12 19l7 0" />
 | 
			
		||||
                    </svg>
 | 
			
		||||
                </a>
 | 
			
		||||
            </li>
 | 
			
		||||
            <li title="Source">
 | 
			
		||||
                <a class="hover:bg-transparent" href="{{ route('source.all') }}">
 | 
			
		||||
                    <svg class="icon" viewBox="0 0 15 15" xmlns="http://www.w3.org/2000/svg">
 | 
			
		||||
                        <path fill="currentColor"
 | 
			
		||||
                            d="m6.793 1.207l.353.354l-.353-.354ZM1.207 6.793l-.353-.354l.353.354Zm0 1.414l.354-.353l-.354.353Zm5.586 5.586l-.354.353l.354-.353Zm1.414 0l-.353-.354l.353.354Zm5.586-5.586l.353.354l-.353-.354Zm0-1.414l-.354.353l.354-.353ZM8.207 1.207l.354-.353l-.354.353ZM6.44.854L.854 6.439l.707.707l5.585-5.585L6.44.854ZM.854 8.56l5.585 5.585l.707-.707l-5.585-5.585l-.707.707Zm7.707 5.585l5.585-5.585l-.707-.707l-5.585 5.585l.707.707Zm5.585-7.707L8.561.854l-.707.707l5.585 5.585l.707-.707Zm0 2.122a1.5 1.5 0 0 0 0-2.122l-.707.707a.5.5 0 0 1 0 .708l.707.707ZM6.44 14.146a1.5 1.5 0 0 0 2.122 0l-.707-.707a.5.5 0 0 1-.708 0l-.707.707ZM.854 6.44a1.5 1.5 0 0 0 0 2.122l.707-.707a.5.5 0 0 1 0-.708L.854 6.44Zm6.292-4.878a.5.5 0 0 1 .708 0L8.56.854a1.5 1.5 0 0 0-2.122 0l.707.707Zm-2 1.293l1 1l.708-.708l-1-1l-.708.708ZM7.5 5a.5.5 0 0 1-.5-.5H6A1.5 1.5 0 0 0 7.5 6V5Zm.5-.5a.5.5 0 0 1-.5.5v1A1.5 1.5 0 0 0 9 4.5H8ZM7.5 4a.5.5 0 0 1 .5.5h1A1.5 1.5 0 0 0 7.5 3v1Zm0-1A1.5 1.5 0 0 0 6 4.5h1a.5.5 0 0 1 .5-.5V3Zm.646 2.854l1.5 1.5l.707-.708l-1.5-1.5l-.707.708ZM10.5 8a.5.5 0 0 1-.5-.5H9A1.5 1.5 0 0 0 10.5 9V8Zm.5-.5a.5.5 0 0 1-.5.5v1A1.5 1.5 0 0 0 12 7.5h-1Zm-.5-.5a.5.5 0 0 1 .5.5h1A1.5 1.5 0 0 0 10.5 6v1Zm0-1A1.5 1.5 0 0 0 9 7.5h1a.5.5 0 0 1 .5-.5V6ZM7 5.5v4h1v-4H7Zm.5 5.5a.5.5 0 0 1-.5-.5H6A1.5 1.5 0 0 0 7.5 12v-1Zm.5-.5a.5.5 0 0 1-.5.5v1A1.5 1.5 0 0 0 9 10.5H8Zm-.5-.5a.5.5 0 0 1 .5.5h1A1.5 1.5 0 0 0 7.5 9v1Zm0-1A1.5 1.5 0 0 0 6 10.5h1a.5.5 0 0 1 .5-.5V9Z" />
 | 
			
		||||
                    </svg>
 | 
			
		||||
                </a>
 | 
			
		||||
            </li>
 | 
			
		||||
            <li title="Security">
 | 
			
		||||
                <a class="hover:bg-transparent" href="{{ route('security.private-key.index') }}">
 | 
			
		||||
                    <svg class="icon" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg">
 | 
			
		||||
                        <path fill="none" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round"
 | 
			
		||||
                            stroke-width="2"
 | 
			
		||||
                            d="m16.555 3.843l3.602 3.602a2.877 2.877 0 0 1 0 4.069l-2.643 2.643a2.877 2.877 0 0 1-4.069 0l-.301-.301l-6.558 6.558a2 2 0 0 1-1.239.578L5.172 21H4a1 1 0 0 1-.993-.883L3 20v-1.172a2 2 0 0 1 .467-1.284l.119-.13L4 17h2v-2h2v-2l2.144-2.144l-.301-.301a2.877 2.877 0 0 1 0-4.069l2.643-2.643a2.877 2.877 0 0 1 4.069 0zM15 9h.01" />
 | 
			
		||||
                    </svg>
 | 
			
		||||
                </a>
 | 
			
		||||
            </li>
 | 
			
		||||
            <li title="Teams">
 | 
			
		||||
                <a class="hover:bg-transparent" href="{{ route('team.index') }}">
 | 
			
		||||
                    <svg xmlns="http://www.w3.org/2000/svg" class="icon" viewBox="0 0 24 24" stroke-width="1.5"
 | 
			
		||||
                        stroke="currentColor" fill="none" stroke-linecap="round" stroke-linejoin="round">
 | 
			
		||||
                        <path stroke="none" d="M0 0h24v24H0z" fill="none" />
 | 
			
		||||
                        <path d="M10 13a2 2 0 1 0 4 0a2 2 0 0 0 -4 0" />
 | 
			
		||||
                        <path d="M8 21v-1a2 2 0 0 1 2 -2h4a2 2 0 0 1 2 2v1" />
 | 
			
		||||
                        <path d="M15 5a2 2 0 1 0 4 0a2 2 0 0 0 -4 0" />
 | 
			
		||||
                        <path d="M17 10h2a2 2 0 0 1 2 2v1" />
 | 
			
		||||
                        <path d="M5 5a2 2 0 1 0 4 0a2 2 0 0 0 -4 0" />
 | 
			
		||||
                        <path d="M3 13v-1a2 2 0 0 1 2 -2h2" />
 | 
			
		||||
                    </svg>
 | 
			
		||||
                </a>
 | 
			
		||||
            </li>
 | 
			
		||||
            <li title="Tags">
 | 
			
		||||
                <a class="hover:bg-transparent" href="{{ route('tags.index') }}">
 | 
			
		||||
                    <svg class="icon" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg">
 | 
			
		||||
                        <g fill="none" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round"
 | 
			
		||||
                            stroke-width="2">
 | 
			
		||||
                            <path
 | 
			
		||||
                                d="M3 8v4.172a2 2 0 0 0 .586 1.414l5.71 5.71a2.41 2.41 0 0 0 3.408 0l3.592-3.592a2.41 2.41 0 0 0 0-3.408l-5.71-5.71A2 2 0 0 0 9.172 6H5a2 2 0 0 0-2 2" />
 | 
			
		||||
                            <path d="m18 19l1.592-1.592a4.82 4.82 0 0 0 0-6.816L15 6m-8 4h-.01" />
 | 
			
		||||
                        </g>
 | 
			
		||||
                    </svg>
 | 
			
		||||
                </a>
 | 
			
		||||
            </li>
 | 
			
		||||
 | 
			
		||||
            <details x-data="{ open: false }" class="dropdown dropdown-right" x-bind:open="open">
 | 
			
		||||
                <summary class="bg-transparent border-none btn hover:bg-transparent no-animation"
 | 
			
		||||
                    x-on:click.prevent="open = !open" x-on:click.away="open = false"> <svg class="icon"
 | 
			
		||||
                        viewBox="0 0 256 256" xmlns="http://www.w3.org/2000/svg">
 | 
			
		||||
                        <path fill="currentColor"
 | 
			
		||||
                            d="M224 128a8 8 0 0 1-8 8h-80v80a8 8 0 0 1-16 0v-80H40a8 8 0 0 1 0-16h80V40a8 8 0 0 1 16 0v80h80a8 8 0 0 1 8 8" />
 | 
			
		||||
                    </svg></summary>
 | 
			
		||||
                <ul tabindex="0" class="w-64 p-2 border border-coolgray-200 dropdown-content menu bg-coolgray-100 ">
 | 
			
		||||
                    <li title="Tags" class="border-transparent hover:bg-coolgray-200 ">
 | 
			
		||||
                        <a class=" hover:bg-transparent hover:no-underline" href="{{ route('tags.index') }}">
 | 
			
		||||
                            <svg class="icon" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg">
 | 
			
		||||
                                <g fill="none" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round"
 | 
			
		||||
                                    stroke-width="2">
 | 
			
		||||
                                    <path
 | 
			
		||||
                                        d="M3 8v4.172a2 2 0 0 0 .586 1.414l5.71 5.71a2.41 2.41 0 0 0 3.408 0l3.592-3.592a2.41 2.41 0 0 0 0-3.408l-5.71-5.71A2 2 0 0 0 9.172 6H5a2 2 0 0 0-2 2" />
 | 
			
		||||
                                    <path d="m18 19l1.592-1.592a4.82 4.82 0 0 0 0-6.816L15 6m-8 4h-.01" />
 | 
			
		||||
                                </g>
 | 
			
		||||
                            </svg>
 | 
			
		||||
                            Tags
 | 
			
		||||
                        </a>
 | 
			
		||||
                    </li>
 | 
			
		||||
                    <li title="Command Center" class="hover:bg-coolgray-200">
 | 
			
		||||
                        <a class="hover:bg-transparent hover:no-underline" href="{{ route('command-center') }}">
 | 
			
		||||
                            <svg xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                                class="{{ request()->is('command-center') ? ' icon' : 'icon' }}" viewBox="0 0 24 24"
 | 
			
		||||
                                stroke-width="1.5" stroke="currentColor" fill="none" stroke-linecap="round"
 | 
			
		||||
                                stroke-linejoin="round">
 | 
			
		||||
                                <path stroke="none" d="M0 0h24v24H0z" fill="none" />
 | 
			
		||||
                                <path d="M5 7l5 5l-5 5" />
 | 
			
		||||
                                <path d="M12 19l7 0" />
 | 
			
		||||
                            </svg>
 | 
			
		||||
                            Command Center
 | 
			
		||||
                        </a>
 | 
			
		||||
                    </li>
 | 
			
		||||
                    <li title="Source" class="hover:bg-coolgray-200">
 | 
			
		||||
                        <a class="hover:bg-transparent hover:no-underline" href="{{ route('source.all') }}">
 | 
			
		||||
                            <svg class="icon" viewBox="0 0 15 15" xmlns="http://www.w3.org/2000/svg">
 | 
			
		||||
                                <path fill="currentColor"
 | 
			
		||||
                                    d="m6.793 1.207l.353.354l-.353-.354ZM1.207 6.793l-.353-.354l.353.354Zm0 1.414l.354-.353l-.354.353Zm5.586 5.586l-.354.353l.354-.353Zm1.414 0l-.353-.354l.353.354Zm5.586-5.586l.353.354l-.353-.354Zm0-1.414l-.354.353l.354-.353ZM8.207 1.207l.354-.353l-.354.353ZM6.44.854L.854 6.439l.707.707l5.585-5.585L6.44.854ZM.854 8.56l5.585 5.585l.707-.707l-5.585-5.585l-.707.707Zm7.707 5.585l5.585-5.585l-.707-.707l-5.585 5.585l.707.707Zm5.585-7.707L8.561.854l-.707.707l5.585 5.585l.707-.707Zm0 2.122a1.5 1.5 0 0 0 0-2.122l-.707.707a.5.5 0 0 1 0 .708l.707.707ZM6.44 14.146a1.5 1.5 0 0 0 2.122 0l-.707-.707a.5.5 0 0 1-.708 0l-.707.707ZM.854 6.44a1.5 1.5 0 0 0 0 2.122l.707-.707a.5.5 0 0 1 0-.708L.854 6.44Zm6.292-4.878a.5.5 0 0 1 .708 0L8.56.854a1.5 1.5 0 0 0-2.122 0l.707.707Zm-2 1.293l1 1l.708-.708l-1-1l-.708.708ZM7.5 5a.5.5 0 0 1-.5-.5H6A1.5 1.5 0 0 0 7.5 6V5Zm.5-.5a.5.5 0 0 1-.5.5v1A1.5 1.5 0 0 0 9 4.5H8ZM7.5 4a.5.5 0 0 1 .5.5h1A1.5 1.5 0 0 0 7.5 3v1Zm0-1A1.5 1.5 0 0 0 6 4.5h1a.5.5 0 0 1 .5-.5V3Zm.646 2.854l1.5 1.5l.707-.708l-1.5-1.5l-.707.708ZM10.5 8a.5.5 0 0 1-.5-.5H9A1.5 1.5 0 0 0 10.5 9V8Zm.5-.5a.5.5 0 0 1-.5.5v1A1.5 1.5 0 0 0 12 7.5h-1Zm-.5-.5a.5.5 0 0 1 .5.5h1A1.5 1.5 0 0 0 10.5 6v1Zm0-1A1.5 1.5 0 0 0 9 7.5h1a.5.5 0 0 1 .5-.5V6ZM7 5.5v4h1v-4H7Zm.5 5.5a.5.5 0 0 1-.5-.5H6A1.5 1.5 0 0 0 7.5 12v-1Zm.5-.5a.5.5 0 0 1-.5.5v1A1.5 1.5 0 0 0 9 10.5H8Zm-.5-.5a.5.5 0 0 1 .5.5h1A1.5 1.5 0 0 0 7.5 9v1Zm0-1A1.5 1.5 0 0 0 6 10.5h1a.5.5 0 0 1 .5-.5V9Z" />
 | 
			
		||||
                            </svg>
 | 
			
		||||
                            Sources
 | 
			
		||||
                        </a>
 | 
			
		||||
                    </li>
 | 
			
		||||
                    <li title="Security" class="hover:bg-coolgray-200">
 | 
			
		||||
                        <a class="hover:bg-transparent hover:no-underline" href="{{ route('security.private-key.index') }}">
 | 
			
		||||
                            <svg class="icon" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg">
 | 
			
		||||
                                <path fill="none" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round"
 | 
			
		||||
                                    stroke-width="2"
 | 
			
		||||
                                    d="m16.555 3.843l3.602 3.602a2.877 2.877 0 0 1 0 4.069l-2.643 2.643a2.877 2.877 0 0 1-4.069 0l-.301-.301l-6.558 6.558a2 2 0 0 1-1.239.578L5.172 21H4a1 1 0 0 1-.993-.883L3 20v-1.172a2 2 0 0 1 .467-1.284l.119-.13L4 17h2v-2h2v-2l2.144-2.144l-.301-.301a2.877 2.877 0 0 1 0-4.069l2.643-2.643a2.877 2.877 0 0 1 4.069 0zM15 9h.01" />
 | 
			
		||||
                            </svg>
 | 
			
		||||
                            Security
 | 
			
		||||
                        </a>
 | 
			
		||||
                    </li>
 | 
			
		||||
                    <li title="Profile" class="hover:bg-coolgray-200">
 | 
			
		||||
                        <a class="hover:bg-transparent hover:no-underline" href="{{ route('profile') }}">
 | 
			
		||||
                            <svg xmlns="http://www.w3.org/2000/svg" class="icon" viewBox="0 0 24 24"
 | 
			
		||||
                                stroke-width="1.5" stroke="currentColor" fill="none" stroke-linecap="round"
 | 
			
		||||
                                stroke-linejoin="round">
 | 
			
		||||
                                <path stroke="none" d="M0 0h24v24H0z" fill="none" />
 | 
			
		||||
                                <path d="M12 12m-9 0a9 9 0 1 0 18 0a9 9 0 1 0 -18 0" />
 | 
			
		||||
                                <path d="M12 10m-3 0a3 3 0 1 0 6 0a3 3 0 1 0 -6 0" />
 | 
			
		||||
                                <path d="M6.168 18.849a4 4 0 0 1 3.832 -2.849h4a4 4 0 0 1 3.834 2.855" />
 | 
			
		||||
                            </svg>
 | 
			
		||||
                            Profile
 | 
			
		||||
                        </a>
 | 
			
		||||
                    </li>
 | 
			
		||||
                    <li title="Teams" class="hover:bg-coolgray-200">
 | 
			
		||||
                        <a class="hover:bg-transparent hover:no-underline" href="{{ route('team.index') }}">
 | 
			
		||||
                            <svg xmlns="http://www.w3.org/2000/svg" class="icon" viewBox="0 0 24 24"
 | 
			
		||||
                                stroke-width="1.5" stroke="currentColor" fill="none" stroke-linecap="round"
 | 
			
		||||
                                stroke-linejoin="round">
 | 
			
		||||
                                <path stroke="none" d="M0 0h24v24H0z" fill="none" />
 | 
			
		||||
                                <path d="M10 13a2 2 0 1 0 4 0a2 2 0 0 0 -4 0" />
 | 
			
		||||
                                <path d="M8 21v-1a2 2 0 0 1 2 -2h4a2 2 0 0 1 2 2v1" />
 | 
			
		||||
                                <path d="M15 5a2 2 0 1 0 4 0a2 2 0 0 0 -4 0" />
 | 
			
		||||
                                <path d="M17 10h2a2 2 0 0 1 2 2v1" />
 | 
			
		||||
                                <path d="M5 5a2 2 0 1 0 4 0a2 2 0 0 0 -4 0" />
 | 
			
		||||
                                <path d="M3 13v-1a2 2 0 0 1 2 -2h2" />
 | 
			
		||||
                            </svg>
 | 
			
		||||
                            Teams
 | 
			
		||||
                        </a>
 | 
			
		||||
                    </li>
 | 
			
		||||
                    @if (isInstanceAdmin())
 | 
			
		||||
                        <li title="Settings" class="hover:bg-coolgray-200">
 | 
			
		||||
                            <a class="hover:bg-transparent hover:no-underline" href="/settings">
 | 
			
		||||
                                <svg xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                                    class="{{ request()->is('settings*') ? 'text-warning icon' : 'icon' }}"
 | 
			
		||||
                                    viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" fill="none"
 | 
			
		||||
                                    stroke-linecap="round" stroke-linejoin="round">
 | 
			
		||||
                                    <path stroke="none" d="M0 0h24v24H0z" fill="none" />
 | 
			
		||||
                                    <path
 | 
			
		||||
                                        d="M10.325 4.317c.426 -1.756 2.924 -1.756 3.35 0a1.724 1.724 0 0 0 2.573 1.066c1.543 -.94 3.31 .826 2.37 2.37a1.724 1.724 0 0 0 1.065 2.572c1.756 .426 1.756 2.924 0 3.35a1.724 1.724 0 0 0 -1.066 2.573c.94 1.543 -.826 3.31 -2.37 2.37a1.724 1.724 0 0 0 -2.572 1.065c-.426 1.756 -2.924 1.756 -3.35 0a1.724 1.724 0 0 0 -2.573 -1.066c-1.543 .94 -3.31 -.826 -2.37 -2.37a1.724 1.724 0 0 0 -1.065 -2.572c-1.756 -.426 -1.756 -2.924 0 -3.35a1.724 1.724 0 0 0 1.066 -2.573c-.94 -1.543 .826 -3.31 2.37 -2.37c1 .608 2.296 .07 2.572 -1.065z" />
 | 
			
		||||
                                    <path d="M9 12a3 3 0 1 0 6 0a3 3 0 0 0 -6 0" />
 | 
			
		||||
                                </svg>
 | 
			
		||||
                                Settings
 | 
			
		||||
                            </a>
 | 
			
		||||
                        </li>
 | 
			
		||||
                    @endif
 | 
			
		||||
 | 
			
		||||
                </ul>
 | 
			
		||||
            </details>
 | 
			
		||||
            @if (isCloud())
 | 
			
		||||
                <li title="Admin">
 | 
			
		||||
                    <a class="hover:bg-transparent" href="/admin">
 | 
			
		||||
                        <svg class="text-pink-600 icon" viewBox="0 0 256 256" xmlns="http://www.w3.org/2000/svg">
 | 
			
		||||
                            <path fill="currentColor"
 | 
			
		||||
                                d="M177.62 159.6a52 52 0 0 1-34 34a12.2 12.2 0 0 1-3.6.55a12 12 0 0 1-3.6-23.45a28 28 0 0 0 18.32-18.32a12 12 0 0 1 22.9 7.2ZM220 144a92 92 0 0 1-184 0c0-28.81 11.27-58.18 33.48-87.28a12 12 0 0 1 17.9-1.33l19.69 19.11L127 19.89a12 12 0 0 1 18.94-5.12C168.2 33.25 220 82.85 220 144m-24 0c0-41.71-30.61-78.39-52.52-99.29l-20.21 55.4a12 12 0 0 1-19.63 4.5L80.71 82.36C67 103.38 60 124.06 60 144a68 68 0 0 0 136 0" />
 | 
			
		||||
                        </svg>
 | 
			
		||||
                    </a>
 | 
			
		||||
                </li>
 | 
			
		||||
            @endif
 | 
			
		||||
            <div class="flex-1"></div>
 | 
			
		||||
            @if (isInstanceAdmin() && !isCloud())
 | 
			
		||||
                @persist('upgrade')
 | 
			
		||||
@@ -114,58 +170,25 @@
 | 
			
		||||
                    </svg>
 | 
			
		||||
                </a>
 | 
			
		||||
            </li>
 | 
			
		||||
            <li title="Profile">
 | 
			
		||||
                <a class="hover:bg-transparent" href="/profile">
 | 
			
		||||
                    <svg xmlns="http://www.w3.org/2000/svg" class="icon" viewBox="0 0 24 24" stroke-width="1.5"
 | 
			
		||||
                        stroke="currentColor" fill="none" stroke-linecap="round" stroke-linejoin="round">
 | 
			
		||||
                        <path stroke="none" d="M0 0h24v24H0z" fill="none" />
 | 
			
		||||
                        <path d="M12 12m-9 0a9 9 0 1 0 18 0a9 9 0 1 0 -18 0" />
 | 
			
		||||
                        <path d="M12 10m-3 0a3 3 0 1 0 6 0a3 3 0 1 0 -6 0" />
 | 
			
		||||
                        <path d="M6.168 18.849a4 4 0 0 1 3.832 -2.849h4a4 4 0 0 1 3.834 2.855" />
 | 
			
		||||
                    </svg>
 | 
			
		||||
                </a>
 | 
			
		||||
            </li>
 | 
			
		||||
 | 
			
		||||
            @if (isInstanceAdmin())
 | 
			
		||||
                <li title="Settings" class="mt-auto">
 | 
			
		||||
                    <a class="hover:bg-transparent" href="/settings">
 | 
			
		||||
                        <svg xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                            class="{{ request()->is('settings*') ? 'text-warning icon' : 'icon' }}" viewBox="0 0 24 24"
 | 
			
		||||
                            stroke-width="1.5" stroke="currentColor" fill="none" stroke-linecap="round"
 | 
			
		||||
                            stroke-linejoin="round">
 | 
			
		||||
                            <path stroke="none" d="M0 0h24v24H0z" fill="none" />
 | 
			
		||||
                            <path
 | 
			
		||||
                                d="M10.325 4.317c.426 -1.756 2.924 -1.756 3.35 0a1.724 1.724 0 0 0 2.573 1.066c1.543 -.94 3.31 .826 2.37 2.37a1.724 1.724 0 0 0 1.065 2.572c1.756 .426 1.756 2.924 0 3.35a1.724 1.724 0 0 0 -1.066 2.573c.94 1.543 -.826 3.31 -2.37 2.37a1.724 1.724 0 0 0 -2.572 1.065c-.426 1.756 -2.924 1.756 -3.35 0a1.724 1.724 0 0 0 -2.573 -1.066c-1.543 .94 -3.31 -.826 -2.37 -2.37a1.724 1.724 0 0 0 -1.065 -2.572c-1.756 -.426 -1.756 -2.924 0 -3.35a1.724 1.724 0 0 0 1.066 -2.573c-.94 -1.543 .826 -3.31 2.37 -2.37c1 .608 2.296 .07 2.572 -1.065z" />
 | 
			
		||||
                            <path d="M9 12a3 3 0 1 0 6 0a3 3 0 0 0 -6 0" />
 | 
			
		||||
                        </svg>
 | 
			
		||||
                    </a>
 | 
			
		||||
                </li>
 | 
			
		||||
            @endif
 | 
			
		||||
            @if (isSubscriptionActive() || isDev())
 | 
			
		||||
                <li title="Send us feedback or get help!" class="hover:bg-transparent">
 | 
			
		||||
                    <div class="justify-center" wire:click="help" onclick="help.showModal()">
 | 
			
		||||
                        <svg class="icon" viewBox="0 0 256 256" xmlns="http://www.w3.org/2000/svg">
 | 
			
		||||
                            <path fill="currentColor"
 | 
			
		||||
                                d="M144 180a16 16 0 1 1-16-16a16 16 0 0 1 16 16m92-52A108 108 0 1 1 128 20a108.12 108.12 0 0 1 108 108m-24 0a84 84 0 1 0-84 84a84.09 84.09 0 0 0 84-84m-84-64c-24.26 0-44 17.94-44 40v4a12 12 0 0 0 24 0v-4c0-8.82 9-16 20-16s20 7.18 20 16s-9 16-20 16a12 12 0 0 0-12 12v8a12 12 0 0 0 23.73 2.56C158.31 137.88 172 122.37 172 104c0-22.06-19.74-40-44-40" />
 | 
			
		||||
                        </svg>
 | 
			
		||||
                    </div>
 | 
			
		||||
                </li>
 | 
			
		||||
            @endif
 | 
			
		||||
            <li class="pb-6" title="Logout">
 | 
			
		||||
                <form action="/logout" method="POST" class=" hover:bg-transparent">
 | 
			
		||||
            <li title="Send us feedback or get help!" class="hover:bg-transparent">
 | 
			
		||||
                <div class="justify-center" wire:click="help" onclick="help.showModal()">
 | 
			
		||||
                    <svg class="icon" viewBox="0 0 256 256" xmlns="http://www.w3.org/2000/svg">
 | 
			
		||||
                        <path fill="currentColor"
 | 
			
		||||
                            d="M140 180a12 12 0 1 1-12-12a12 12 0 0 1 12 12M128 72c-22.06 0-40 16.15-40 36v4a8 8 0 0 0 16 0v-4c0-11 10.77-20 24-20s24 9 24 20s-10.77 20-24 20a8 8 0 0 0-8 8v8a8 8 0 0 0 16 0v-.72c18.24-3.35 32-17.9 32-35.28c0-19.85-17.94-36-40-36m104 56A104 104 0 1 1 128 24a104.11 104.11 0 0 1 104 104m-16 0a88 88 0 1 0-88 88a88.1 88.1 0 0 0 88-88" />
 | 
			
		||||
                    </svg>
 | 
			
		||||
                </div>
 | 
			
		||||
            </li>
 | 
			
		||||
            <form action="/logout" method="POST" class="hover:bg-transparent">
 | 
			
		||||
            <li title="Logout" class="mb-6 hover:transparent">
 | 
			
		||||
                    @csrf
 | 
			
		||||
                    <button type="submit" class="rounded-none hover:text-white hover:bg-transparent">
 | 
			
		||||
                        <svg xmlns="http://www.w3.org/2000/svg" class="icon" viewBox="0 0 24 24" stroke-width="1.5"
 | 
			
		||||
                            stroke="currentColor" fill="none" stroke-linecap="round" stroke-linejoin="round">
 | 
			
		||||
                            <path stroke="none" d="M0 0h24v24H0z" fill="none" />
 | 
			
		||||
                            <path d="M13 12v.01" />
 | 
			
		||||
                            <path d="M3 21h18" />
 | 
			
		||||
                            <path d="M5 21v-16a2 2 0 0 1 2 -2h7.5m2.5 10.5v7.5" />
 | 
			
		||||
                            <path d="M14 7h7m-3 -3l3 3l-3 3" />
 | 
			
		||||
                        <svg class="icon" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg">
 | 
			
		||||
                            <path fill="currentColor" d="M12 22C6.477 22 2 17.523 2 12S6.477 2 12 2a9.985 9.985 0 0 1 8 4h-2.71a8 8 0 1 0 .001 12h2.71A9.985 9.985 0 0 1 12 22m7-6v-3h-8v-2h8V8l5 4z"/>
 | 
			
		||||
                        </svg>
 | 
			
		||||
                    </button>
 | 
			
		||||
                </form>
 | 
			
		||||
            </li>
 | 
			
		||||
                </li>
 | 
			
		||||
            </form>
 | 
			
		||||
        </ul>
 | 
			
		||||
    </nav>
 | 
			
		||||
@endauth
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,6 @@
 | 
			
		||||
@if (Str::of($status)->startsWith('running'))
 | 
			
		||||
@if (str($status)->startsWith('running'))
 | 
			
		||||
    <x-status.running :status="$status" />
 | 
			
		||||
@elseif(Str::of($status)->startsWith('restarting') || Str::of($status)->startsWith('starting'))
 | 
			
		||||
@elseif(str($status)->startsWith('restarting') || str($status)->startsWith('starting') || str($status)->startsWith('degraded'))
 | 
			
		||||
    <x-status.restarting :status="$status" />
 | 
			
		||||
@else
 | 
			
		||||
    <x-status.stopped :status="$status" />
 | 
			
		||||
 
 | 
			
		||||
@@ -2,7 +2,12 @@
 | 
			
		||||
    'status' => 'Restarting',
 | 
			
		||||
])
 | 
			
		||||
<x-loading wire:loading.delay.longer />
 | 
			
		||||
<div class="flex items-center gap-2" wire:loading.remove.delay.longer>
 | 
			
		||||
<div class="flex items-center " wire:loading.remove.delay.longer>
 | 
			
		||||
    <div class="badge badge-warning badge-xs"></div>
 | 
			
		||||
    <div class="text-xs font-medium tracking-wide text-warning">{{ Str::headline($status) }}</div>
 | 
			
		||||
    <div class="pl-2 pr-1 text-xs font-bold tracking-widerr text-warning">
 | 
			
		||||
        {{ str($status)->before(':')->headline() }}
 | 
			
		||||
    </div>
 | 
			
		||||
    @if (!str($status)->startsWith('Proxy'))
 | 
			
		||||
        <div class="text-xs text-warning">({{ str($status)->after(':') }})</div>
 | 
			
		||||
    @endif
 | 
			
		||||
</div>
 | 
			
		||||
 
 | 
			
		||||
@@ -2,7 +2,12 @@
 | 
			
		||||
    'status' => 'Running',
 | 
			
		||||
])
 | 
			
		||||
<x-loading wire:loading.delay.longer />
 | 
			
		||||
<div class="flex items-center gap-2 " wire:loading.remove.delay.longer>
 | 
			
		||||
<div class="flex items-center" wire:loading.remove.delay.longer>
 | 
			
		||||
    <div class="badge badge-success badge-xs"></div>
 | 
			
		||||
    <div class="text-xs font-medium tracking-wide text-success">{{ Str::headline($status) }}</div>
 | 
			
		||||
    <div class="pl-2 pr-1 text-xs font-bold tracking-wider text-success">
 | 
			
		||||
        {{ str($status)->before(':')->headline() }}
 | 
			
		||||
    </div>
 | 
			
		||||
    @if (!str($status)->startsWith('Proxy'))
 | 
			
		||||
        <div class="text-xs text-success">({{ str($status)->after(':') }})</div>
 | 
			
		||||
    @endif
 | 
			
		||||
</div>
 | 
			
		||||
 
 | 
			
		||||
@@ -2,7 +2,7 @@
 | 
			
		||||
    'status' => 'Stopped',
 | 
			
		||||
])
 | 
			
		||||
<x-loading wire:loading.delay.longer />
 | 
			
		||||
<div class="flex items-center gap-2 " wire:loading.remove.delay.longer>
 | 
			
		||||
<div class="flex items-center" wire:loading.remove.delay.longer>
 | 
			
		||||
    <div class="badge badge-error badge-xs"></div>
 | 
			
		||||
    <div class="text-xs font-medium tracking-wide text-error">{{ Str::headline($status) }}</div>
 | 
			
		||||
    <div class="pl-2 pr-1 text-xs font-bold tracking-wider text-error">{{ str($status)->before(':')->headline() }}</div>
 | 
			
		||||
</div>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,15 +1,12 @@
 | 
			
		||||
@extends('layouts.base')
 | 
			
		||||
@section('body')
 | 
			
		||||
    @if (isSubscriptionActive() || isDev())
 | 
			
		||||
        <div title="Send us feedback or get help!" class="fixed top-0 right-0 p-2 px-4 pt-4 mt-auto text-xs">
 | 
			
		||||
            <button class="flex items-center justify-center gap-2" wire:click="help" onclick="help.showModal()">
 | 
			
		||||
                <svg class="icon" viewBox="0 0 256 256" xmlns="http://www.w3.org/2000/svg">
 | 
			
		||||
                    <path fill="currentColor"
 | 
			
		||||
                        d="M144 180a16 16 0 1 1-16-16a16 16 0 0 1 16 16m92-52A108 108 0 1 1 128 20a108.12 108.12 0 0 1 108 108m-24 0a84 84 0 1 0-84 84a84.09 84.09 0 0 0 84-84m-84-64c-24.26 0-44 17.94-44 40v4a12 12 0 0 0 24 0v-4c0-8.82 9-16 20-16s20 7.18 20 16s-9 16-20 16a12 12 0 0 0-12 12v8a12 12 0 0 0 23.73 2.56C158.31 137.88 172 122.37 172 104c0-22.06-19.74-40-44-40" />
 | 
			
		||||
                </svg>
 | 
			
		||||
            </button>
 | 
			
		||||
        </div>
 | 
			
		||||
    @endif
 | 
			
		||||
    <div title="Send us feedback or get help!" class="fixed top-0 right-0 p-2 px-4 pt-4 mt-auto text-xs">
 | 
			
		||||
        <button class="flex items-center justify-center gap-2" wire:click="help" onclick="help.showModal()">
 | 
			
		||||
            <svg class="icon" viewBox="0 0 256 256" xmlns="http://www.w3.org/2000/svg">
 | 
			
		||||
                <path fill="currentColor" d="M140 180a12 12 0 1 1-12-12a12 12 0 0 1 12 12M128 72c-22.06 0-40 16.15-40 36v4a8 8 0 0 0 16 0v-4c0-11 10.77-20 24-20s24 9 24 20s-10.77 20-24 20a8 8 0 0 0-8 8v8a8 8 0 0 0 16 0v-.72c18.24-3.35 32-17.9 32-35.28c0-19.85-17.94-36-40-36m104 56A104 104 0 1 1 128 24a104.11 104.11 0 0 1 104 104m-16 0a88 88 0 1 0-88 88a88.1 88.1 0 0 0 88-88"/>
 | 
			
		||||
            </svg>
 | 
			
		||||
        </button>
 | 
			
		||||
    </div>
 | 
			
		||||
    <main class="min-h-screen hero">
 | 
			
		||||
        <div class="hero-content">
 | 
			
		||||
            {{ $slot }}
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										17
									
								
								resources/views/livewire/admin/index.blade.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								resources/views/livewire/admin/index.blade.php
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,17 @@
 | 
			
		||||
<div>
 | 
			
		||||
    <h1>Admin Dashboard</h1>
 | 
			
		||||
    <h3 class="pt-4">Who am I now?</h3>
 | 
			
		||||
    {{ auth()->user()->name }}
 | 
			
		||||
    <h3 class="pt-4">Users</h3>
 | 
			
		||||
    <div class="flex flex-wrap gap-2">
 | 
			
		||||
        <div class="w-96 box" wire:click="switchUser('0')">
 | 
			
		||||
            Root
 | 
			
		||||
        </div>
 | 
			
		||||
        @foreach ($users as $user)
 | 
			
		||||
            <div class="w-96 box" wire:click="switchUser('{{ $user->id }}')">
 | 
			
		||||
                <p>{{ $user->name }}</p>
 | 
			
		||||
                <p>{{ $user->email }}</p>
 | 
			
		||||
            </div>
 | 
			
		||||
        @endforeach
 | 
			
		||||
    </div>
 | 
			
		||||
</div>
 | 
			
		||||
@@ -29,7 +29,7 @@
 | 
			
		||||
    @endif
 | 
			
		||||
    @foreach ($projects as $project)
 | 
			
		||||
        <div class="gap-2 border border-transparent cursor-pointer box group">
 | 
			
		||||
            @if (data_get($project, 'environments.0.name'))
 | 
			
		||||
            @if (data_get($project, 'environments')->count() === 1)
 | 
			
		||||
                <a class="flex flex-col flex-1 mx-6 hover:no-underline"
 | 
			
		||||
                    href="{{ route('project.resource.index', ['project_uuid' => data_get($project, 'uuid'), 'environment_name' => data_get($project, 'environments.0.name', 'production')]) }}">
 | 
			
		||||
                    <div class="font-bold text-white">{{ $project->name }}</div>
 | 
			
		||||
@@ -106,6 +106,7 @@
 | 
			
		||||
    @if (count($deployments_per_server) > 0)
 | 
			
		||||
        <x-loading />
 | 
			
		||||
    @endif
 | 
			
		||||
    <x-forms.button wire:click='cleanup_queue'>Cleanup Queues</x-forms.button>
 | 
			
		||||
</div>
 | 
			
		||||
<div wire:poll.1000ms="get_deployments" class="grid grid-cols-1">
 | 
			
		||||
    @forelse ($deployments_per_server as $server_name => $deployments)
 | 
			
		||||
 
 | 
			
		||||
@@ -27,8 +27,8 @@
 | 
			
		||||
                <a :class="activeTab === 'source' && 'text-white'"
 | 
			
		||||
                    @click.prevent="activeTab = 'source'; window.location.hash = 'source'" href="#">Source</a>
 | 
			
		||||
            @endif
 | 
			
		||||
            <a :class="activeTab === 'server' && 'text-white'"
 | 
			
		||||
                @click.prevent="activeTab = 'server'; window.location.hash = 'server'" href="#">Server
 | 
			
		||||
            <a :class="activeTab === 'servers' && 'text-white'"
 | 
			
		||||
                @click.prevent="activeTab = 'servers'; window.location.hash = 'servers'" href="#">Servers
 | 
			
		||||
            </a>
 | 
			
		||||
 | 
			
		||||
            <a :class="activeTab === 'scheduled-tasks' && 'text-white'"
 | 
			
		||||
@@ -88,7 +88,7 @@
 | 
			
		||||
                    <livewire:project.application.source :application="$application" />
 | 
			
		||||
                </div>
 | 
			
		||||
            @endif
 | 
			
		||||
            <div x-cloak x-show="activeTab === 'server'">
 | 
			
		||||
            <div x-cloak x-show="activeTab === 'servers'">
 | 
			
		||||
                <livewire:project.shared.destination :resource="$application" :servers="$servers" />
 | 
			
		||||
            </div>
 | 
			
		||||
            <div x-cloak x-show="activeTab === 'storages'">
 | 
			
		||||
 
 | 
			
		||||
@@ -77,6 +77,11 @@
 | 
			
		||||
                            Manual
 | 
			
		||||
                        </div>
 | 
			
		||||
                    @endif
 | 
			
		||||
                    @if (data_get($deployment, 'server_name'))
 | 
			
		||||
                        <div class="flex gap-1">
 | 
			
		||||
                            Server: {{ data_get($deployment, 'server_name') }}
 | 
			
		||||
                        </div>
 | 
			
		||||
                    @endif
 | 
			
		||||
                </div>
 | 
			
		||||
 | 
			
		||||
                <div class="flex flex-col" x-data="elapsedTime('{{ $deployment->deployment_uuid }}', '{{ $deployment->status }}', '{{ $deployment->created_at }}', '{{ $deployment->updated_at }}')">
 | 
			
		||||
 
 | 
			
		||||
@@ -97,7 +97,7 @@
 | 
			
		||||
                            <x-forms.input id="application.docker_registry_image_tag" label="Docker Image Tag" />
 | 
			
		||||
                        @endif
 | 
			
		||||
                    @else
 | 
			
		||||
                        @if ($application->destination->server->isSwarm())
 | 
			
		||||
                        @if ($application->destination->server->isSwarm() || $application->additional_servers->count() > 0)
 | 
			
		||||
                            <x-forms.input id="application.docker_registry_image_name" required label="Docker Image" />
 | 
			
		||||
                            <x-forms.input id="application.docker_registry_image_tag"
 | 
			
		||||
                                helper="If set, it will tag the built image with this tag too. <br><br>Example: If you set it to 'latest', it will push the image with the commit sha tag + with the latest tag."
 | 
			
		||||
 
 | 
			
		||||
@@ -27,7 +27,7 @@
 | 
			
		||||
            @if (!$application->destination->server->isSwarm())
 | 
			
		||||
                <x-applications.advanced :application="$application" />
 | 
			
		||||
            @endif
 | 
			
		||||
            @if ($application->status !== 'exited')
 | 
			
		||||
            @if (!str($application->status)->startsWith('exited'))
 | 
			
		||||
                @if (!$application->destination->server->isSwarm())
 | 
			
		||||
                    <button title="With rolling update if possible" wire:click='deploy'
 | 
			
		||||
                        class="flex items-center gap-2 cursor-pointer hover:text-white text-neutral-400">
 | 
			
		||||
 
 | 
			
		||||
@@ -21,10 +21,10 @@
 | 
			
		||||
                @click.prevent="activeTab = 'environment-variables'; window.location.hash = 'environment-variables'"
 | 
			
		||||
                href="#">Environment
 | 
			
		||||
                Variables</a>
 | 
			
		||||
            <a :class="activeTab === 'server' && 'text-white'"
 | 
			
		||||
                @click.prevent="activeTab = 'server';
 | 
			
		||||
                window.location.hash = 'server'"
 | 
			
		||||
                href="#">Server
 | 
			
		||||
            <a :class="activeTab === 'servers' && 'text-white'"
 | 
			
		||||
                @click.prevent="activeTab = 'servers';
 | 
			
		||||
                window.location.hash = 'servers'"
 | 
			
		||||
                href="#">Servers
 | 
			
		||||
            </a>
 | 
			
		||||
            <a :class="activeTab === 'storages' && 'text-white'"
 | 
			
		||||
                @click.prevent="activeTab = 'storages';
 | 
			
		||||
@@ -74,7 +74,7 @@
 | 
			
		||||
            <div x-cloak x-show="activeTab === 'environment-variables'">
 | 
			
		||||
                <livewire:project.shared.environment-variable.all :resource="$database" />
 | 
			
		||||
            </div>
 | 
			
		||||
            <div x-cloak x-show="activeTab === 'server'">
 | 
			
		||||
            <div x-cloak x-show="activeTab === 'servers'">
 | 
			
		||||
                <livewire:project.shared.destination :resource="$database" />
 | 
			
		||||
            </div>
 | 
			
		||||
            <div x-cloak x-show="activeTab === 'storages'">
 | 
			
		||||
 
 | 
			
		||||
@@ -1,5 +1,5 @@
 | 
			
		||||
<div x-data x-init="$wire.loadServers">
 | 
			
		||||
    <div class="flex gap-2 ">
 | 
			
		||||
    <div class="flex gap-4 ">
 | 
			
		||||
        <h1>New Resource</h1>
 | 
			
		||||
        <div class="w-96">
 | 
			
		||||
            <x-forms.select wire:model="selectedEnvironment">
 | 
			
		||||
@@ -10,7 +10,7 @@
 | 
			
		||||
        </div>
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class="pb-4 ">Deploy resources, like Applications, Databases, Services...</div>
 | 
			
		||||
    <div class="flex flex-col gap-2 pt-10">
 | 
			
		||||
    <div class="flex flex-col gap-4 pt-10">
 | 
			
		||||
        @if ($current_step === 'type')
 | 
			
		||||
            <ul class="pb-10 steps">
 | 
			
		||||
                <li class="step step-secondary">Select Resource Type</li>
 | 
			
		||||
@@ -18,7 +18,7 @@
 | 
			
		||||
                <li class="step">Select a Destination</li>
 | 
			
		||||
            </ul>
 | 
			
		||||
            <h2>Applications</h2>
 | 
			
		||||
            <div class="grid justify-start grid-cols-1 gap-2 text-left xl:grid-cols-3">
 | 
			
		||||
            <div class="grid justify-start grid-cols-1 gap-4 text-left xl:grid-cols-3">
 | 
			
		||||
                <div class="box group" wire:click="setType('public')">
 | 
			
		||||
                    <div class="flex flex-col mx-6">
 | 
			
		||||
                        <div class="font-bold text-white group-hover:text-white">
 | 
			
		||||
@@ -50,7 +50,7 @@
 | 
			
		||||
                    </div>
 | 
			
		||||
                </div>
 | 
			
		||||
            </div>
 | 
			
		||||
            <div class="grid justify-start grid-cols-1 gap-2 text-left xl:grid-cols-3">
 | 
			
		||||
            <div class="grid justify-start grid-cols-1 gap-4 text-left xl:grid-cols-3">
 | 
			
		||||
                <div class="box group" wire:click="setType('dockerfile')">
 | 
			
		||||
                    <div class="flex flex-col mx-6">
 | 
			
		||||
                        <div class="font-bold text-white group-hover:text-white">
 | 
			
		||||
@@ -83,7 +83,7 @@
 | 
			
		||||
                </div>
 | 
			
		||||
            </div>
 | 
			
		||||
            <h2 class="py-4">Databases</h2>
 | 
			
		||||
            <div class="grid justify-start grid-cols-1 gap-2 text-left xl:grid-cols-3">
 | 
			
		||||
            <div class="grid justify-start grid-cols-1 gap-4 text-left xl:grid-cols-5">
 | 
			
		||||
                <div class="box group" wire:click="setType('postgresql')">
 | 
			
		||||
                    <div class="flex flex-col mx-6">
 | 
			
		||||
                        <div class="font-bold text-white group-hover:text-white">
 | 
			
		||||
@@ -151,14 +151,14 @@
 | 
			
		||||
                    </div>
 | 
			
		||||
                </div> --}}
 | 
			
		||||
            </div>
 | 
			
		||||
            <div class="flex items-center gap-2" wire:init='loadServices'>
 | 
			
		||||
            <div class="flex items-center gap-4" wire:init='loadServices'>
 | 
			
		||||
                <h2 class="py-4">Services</h2>
 | 
			
		||||
                <x-forms.button wire:click='loadServices'>Reload Services List</x-forms.button>
 | 
			
		||||
                <input
 | 
			
		||||
                    class="w-full text-white rounded input input-sm bg-coolgray-200 disabled:bg-coolgray-200/50 disabled:border-none placeholder:text-coolgray-500 read-only:text-neutral-500 read-only:bg-coolgray-200/50"
 | 
			
		||||
                    wire:model.live.debounce.200ms="search" placeholder="Search...">
 | 
			
		||||
            </div>
 | 
			
		||||
            <div class="grid justify-start grid-cols-1 gap-2 text-left xl:grid-cols-3">
 | 
			
		||||
            <div class="grid justify-start grid-cols-1 gap-4 text-left xl:grid-cols-5">
 | 
			
		||||
                @if ($loadingServices)
 | 
			
		||||
                    <span class="loading loading-xs loading-spinner"></span>
 | 
			
		||||
                @else
 | 
			
		||||
@@ -211,7 +211,7 @@
 | 
			
		||||
                        label="Include Swarm Clusters" />
 | 
			
		||||
                </div>
 | 
			
		||||
            @endif --}}
 | 
			
		||||
            <div class="flex flex-col justify-center gap-2 text-left xl:flex-row xl:flex-wrap">
 | 
			
		||||
            <div class="flex flex-col justify-center gap-4 text-left xl:flex-row xl:flex-wrap">
 | 
			
		||||
                @forelse($servers as $server)
 | 
			
		||||
                    <div class="box group" wire:click="setServer({{ $server }})">
 | 
			
		||||
                        <div class="flex flex-col mx-6">
 | 
			
		||||
@@ -244,7 +244,7 @@
 | 
			
		||||
                <li class="step step-secondary">Select a Destination</li>
 | 
			
		||||
            </ul>
 | 
			
		||||
 | 
			
		||||
            <div class="flex flex-col justify-center gap-2 text-left xl:flex-row xl:flex-wrap">
 | 
			
		||||
            <div class="flex flex-col justify-center gap-4 text-left xl:flex-row xl:flex-wrap">
 | 
			
		||||
                @if ($server->isSwarm())
 | 
			
		||||
                    @foreach ($swarmDockers as $swarmDocker)
 | 
			
		||||
                        <div class="box group" wire:click="setDestination('{{ $swarmDocker->uuid }}')">
 | 
			
		||||
@@ -279,7 +279,7 @@
 | 
			
		||||
            </div>
 | 
			
		||||
        @endif
 | 
			
		||||
        @if ($current_step === 'existing-postgresql')
 | 
			
		||||
            <form wire:submit='addExistingPostgresql' class="flex items-end gap-2">
 | 
			
		||||
            <form wire:submit='addExistingPostgresql' class="flex items-end gap-4">
 | 
			
		||||
                <x-forms.input placeholder="postgres://username:password@database:5432" label="Database URL"
 | 
			
		||||
                    id="existingPostgresqlUrl" />
 | 
			
		||||
                <x-forms.button type="submit">Add Database</x-forms.button>
 | 
			
		||||
 
 | 
			
		||||
@@ -61,7 +61,10 @@
 | 
			
		||||
                            <template x-if="item.status.startsWith('exited')">
 | 
			
		||||
                                <div class="absolute bg-error -top-1 -left-1 badge badge-xs"></div>
 | 
			
		||||
                            </template>
 | 
			
		||||
                            <template x-if="item.status.startsWith('restarting')">
 | 
			
		||||
                            <template x-if="item.status.startsWith('restarting')" >
 | 
			
		||||
                                <div class="absolute bg-warning -top-1 -left-1 badge badge-xs"></div>
 | 
			
		||||
                            </template>
 | 
			
		||||
                            <template x-if="item.status.startsWith('degraded')">
 | 
			
		||||
                                <div class="absolute bg-warning -top-1 -left-1 badge badge-xs"></div>
 | 
			
		||||
                            </template>
 | 
			
		||||
                        </a>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,38 +1,75 @@
 | 
			
		||||
<div>
 | 
			
		||||
    <h2>Server</h2>
 | 
			
		||||
    <h2>Servers</h2>
 | 
			
		||||
    <div class="">Server related configurations.</div>
 | 
			
		||||
    <h3 class="pt-4">Destination Server & Network</h3>
 | 
			
		||||
    <div class="py-4">
 | 
			
		||||
        <a class="box"
 | 
			
		||||
            href="{{ route('server.show', ['server_uuid' => data_get($resource, 'destination.server.uuid')]) }}">On
 | 
			
		||||
            server <span class="px-1 text-warning">{{ data_get($resource, 'destination.server.name') }}</span>
 | 
			
		||||
            in <span class="px-1 text-warning"> {{ data_get($resource, 'destination.network') }} </span> network.</a>
 | 
			
		||||
    </div>
 | 
			
		||||
    {{-- Additional Destinations:
 | 
			
		||||
    {{$resource->additional_destinations}} --}}
 | 
			
		||||
    {{-- @if (count($servers) > 0)
 | 
			
		||||
        <div>
 | 
			
		||||
            <h3>Additional Servers</h3>
 | 
			
		||||
            @foreach ($servers as $server)
 | 
			
		||||
                <form wire:submit='submit' class="p-2 border border-coolgray-400">
 | 
			
		||||
                    <h4>{{ $server->name }}</h4>
 | 
			
		||||
                    <div class="text-sm text-coolgray-600">{{ $server->description }}</div>
 | 
			
		||||
                    <x-forms.checkbox id="additionalServers.{{ $loop->index }}.enabled" label="Enabled">
 | 
			
		||||
                    </x-forms.checkbox>
 | 
			
		||||
                    <x-forms.select label="Destination" id="additionalServers.{{ $loop->index }}.destination" required>
 | 
			
		||||
                        @foreach ($server->destinations() as $destination)
 | 
			
		||||
                            @if ($loop->first)
 | 
			
		||||
                                <option selected value="{{ $destination->uuid }}">{{ $destination->name }}</option>
 | 
			
		||||
                                <option value="{{ $destination->uuid }}">{{ $destination->name }}</option>
 | 
			
		||||
                            @else
 | 
			
		||||
                                <option value="{{ $destination->uuid }}">{{ $destination->name }}</option>
 | 
			
		||||
                                <option value="{{ $destination->uuid }}">{{ $destination->name }}</option>
 | 
			
		||||
                            @endif
 | 
			
		||||
                        @endforeach
 | 
			
		||||
                    </x-forms.select>
 | 
			
		||||
                    <x-forms.button type="submit">Save</x-forms.button>
 | 
			
		||||
                </form>
 | 
			
		||||
            @endforeach
 | 
			
		||||
    <div class="grid grid-cols-1 gap-4 py-4">
 | 
			
		||||
        <div class="flex gap-2">
 | 
			
		||||
            <div class="relative flex flex-col text-white cursor-default box-without-bg bg-coolgray-100 w-96">
 | 
			
		||||
                <div class="font-bold">Main Server</div>
 | 
			
		||||
                @if (str($resource->realStatus())->startsWith('running'))
 | 
			
		||||
                    <div title="{{ $resource->realStatus() }}" class="absolute bg-success -top-1 -left-1 badge badge-xs">
 | 
			
		||||
                    </div>
 | 
			
		||||
                @elseif (str($resource->realStatus())->startsWith('exited'))
 | 
			
		||||
                    <div title="{{ $resource->realStatus() }}" class="absolute bg-error -top-1 -left-1 badge badge-xs">
 | 
			
		||||
                    </div>
 | 
			
		||||
                @endif
 | 
			
		||||
                <div>
 | 
			
		||||
                    Server: {{ data_get($resource, 'destination.server.name') }}
 | 
			
		||||
                </div>
 | 
			
		||||
                <div>
 | 
			
		||||
                    Network: {{ data_get($resource, 'destination.network') }}
 | 
			
		||||
                </div>
 | 
			
		||||
            </div>
 | 
			
		||||
            @if ($resource?->additional_networks?->count() > 0)
 | 
			
		||||
                <x-forms.button
 | 
			
		||||
                    wire:click="redeploy('{{ data_get($resource, 'destination.id') }}','{{ data_get($resource, 'destination.server.id') }}')">Redeploy</x-forms.button>
 | 
			
		||||
            @endif
 | 
			
		||||
        </div>
 | 
			
		||||
        @if ($resource?->additional_networks?->count() > 0)
 | 
			
		||||
            @foreach ($resource->additional_networks as $destination)
 | 
			
		||||
                <div class="flex gap-2">
 | 
			
		||||
                    <div class="relative flex flex-col box w-96">
 | 
			
		||||
                        @if (str(data_get($destination, 'pivot.status'))->startsWith('running'))
 | 
			
		||||
                            <div title="{{ data_get($destination, 'pivot.status') }}"
 | 
			
		||||
                                class="absolute bg-success -top-1 -left-1 badge badge-xs"></div>
 | 
			
		||||
                        @elseif (str(data_get($destination, 'pivot.status'))->startsWith('exited'))
 | 
			
		||||
                            <div title="{{ data_get($destination, 'pivot.status') }}"
 | 
			
		||||
                                class="absolute bg-error -top-1 -left-1 badge badge-xs"></div>
 | 
			
		||||
                        @endif
 | 
			
		||||
                        <div>
 | 
			
		||||
                            Server: {{ data_get($destination, 'server.name') }}
 | 
			
		||||
                        </div>
 | 
			
		||||
                        <div>
 | 
			
		||||
                            Network: {{ data_get($destination, 'network') }}
 | 
			
		||||
                        </div>
 | 
			
		||||
 | 
			
		||||
                    </div>
 | 
			
		||||
                    <x-forms.button
 | 
			
		||||
                        wire:click="redeploy('{{ data_get($destination, 'id') }}','{{ data_get($destination, 'server.id') }}')">Redeploy</x-forms.button>
 | 
			
		||||
                    <x-new-modal
 | 
			
		||||
                        action="removeServer({{ data_get($destination, 'id') }},{{ data_get($destination, 'server.id') }})"
 | 
			
		||||
                        isErrorButton buttonTitle="Remove Server">
 | 
			
		||||
                        This will stop the running application in this server and remove it as a deployment
 | 
			
		||||
                        destination.<br><br>Please think again.
 | 
			
		||||
                    </x-new-modal>
 | 
			
		||||
                </div>
 | 
			
		||||
            @endforeach
 | 
			
		||||
        @endif
 | 
			
		||||
    </div>
 | 
			
		||||
    {{-- @if ($resource->getMorphClass() === 'App\Models\Application')
 | 
			
		||||
        @if (count($networks) > 0)
 | 
			
		||||
            <h4>Choose another server</h4>
 | 
			
		||||
            <div class="pb-4 description">(experimental) </div>
 | 
			
		||||
            <div class="grid grid-cols-1 gap-4 ">
 | 
			
		||||
                @foreach ($networks as $network)
 | 
			
		||||
                    <div wire:click="addServer('{{ $network->id }}','{{ data_get($network, 'server.id') }}')"
 | 
			
		||||
                        class="box w-96">
 | 
			
		||||
                        {{ data_get($network, 'server.name') }}
 | 
			
		||||
                        {{ $network->name }}
 | 
			
		||||
                    </div>
 | 
			
		||||
                @endforeach
 | 
			
		||||
            </div>
 | 
			
		||||
        @else
 | 
			
		||||
            <div class="text-neutral-500">No additional servers available to attach.</div>
 | 
			
		||||
        @endif
 | 
			
		||||
    @endif --}}
 | 
			
		||||
</div>
 | 
			
		||||
 
 | 
			
		||||
@@ -42,7 +42,7 @@
 | 
			
		||||
                <x-forms.input id="server.description" label="Description" />
 | 
			
		||||
                @if (!$server->settings->is_swarm_worker && !$server->settings->is_build_server)
 | 
			
		||||
                    <x-forms.input placeholder="https://example.com" id="wildcard_domain" label="Wildcard Domain"
 | 
			
		||||
                        helper="Wildcard domain for your applications. If you set this, you will get a random generated domain for your new applications.<br><span class='font-bold text-white'>Example:</span><br>In case you set:<span class='text-helper'>https://example.com</span> your applications will get:<br> <span class='text-helper'>https://randomId.example.com</span>" />
 | 
			
		||||
                        helper='A wildcard domain allows you to receive a randomly generated domain for your new applications. <br><br>For instance, if you set "https://example.com" as your wildcard domain, your applications will receive domains like "https://randomId.example.com".' />
 | 
			
		||||
                @endif
 | 
			
		||||
 | 
			
		||||
            </div>
 | 
			
		||||
@@ -61,7 +61,7 @@
 | 
			
		||||
                            label="Use it as a build server?" />
 | 
			
		||||
                    @else
 | 
			
		||||
                        <x-forms.checkbox instantSave
 | 
			
		||||
                            helper="If you are using Cloudflare Tunnels, enable this. It will proxy all ssh requests to your server through Cloudflare.<br><span class='text-warning'>Coolify does not install/setup Cloudflare (cloudflared) on your server.</span>"
 | 
			
		||||
                            helper="If you are using Cloudflare Tunnels, enable this. It will proxy all SSH requests to your server through Cloudflare.<br><span class='text-warning'>Coolify does not install or set up Cloudflare (cloudflared) on your server.</span>"
 | 
			
		||||
                            id="server.settings.is_cloudflare_tunnel" label="Cloudflare Tunnel" />
 | 
			
		||||
                        @if ($server->isSwarm())
 | 
			
		||||
                            <div class="pt-6"> Swarm support is experimental. </div>
 | 
			
		||||
@@ -93,9 +93,11 @@
 | 
			
		||||
            <h3 class="py-4">Settings</h3>
 | 
			
		||||
            <div class="flex gap-2">
 | 
			
		||||
                <x-forms.input id="cleanup_after_percentage" label="Disk cleanup threshold (%)" required
 | 
			
		||||
                    helper="Disk cleanup job will be executed if disk usage is more than this number." />
 | 
			
		||||
                    helper="The disk cleanup task will run when the disk usage exceeds this threshold." />
 | 
			
		||||
                <x-forms.input id="server.settings.concurrent_builds" label="Number of concurrent builds" required
 | 
			
		||||
                    helper="You can define how many concurrent builds processes / deployments should run at the same time." />
 | 
			
		||||
                    helper="You can specify the number of simultaneous build processes/deployments that should run concurrently." />
 | 
			
		||||
                <x-forms.input id="server.settings.dynamic_timeout" label="Deployment timeout (seconds)" required
 | 
			
		||||
                    helper="You can define the maximum duration for a deployment to run before timing it out." />
 | 
			
		||||
            </div>
 | 
			
		||||
        @endif
 | 
			
		||||
    </form>
 | 
			
		||||
 
 | 
			
		||||
@@ -15,7 +15,9 @@
 | 
			
		||||
                        <x-forms.input type="password" required id="server.settings.logdrain_newrelic_license_key"
 | 
			
		||||
                            label="License Key" />
 | 
			
		||||
                        <x-forms.input required id="server.settings.logdrain_newrelic_base_uri"
 | 
			
		||||
                            placeholder="https://log-api.eu.newrelic.com/log/v1" label="Endpoint (EU / US)" />
 | 
			
		||||
                            placeholder="https://log-api.eu.newrelic.com/log/v1"
 | 
			
		||||
                            helper="For EU use: https://log-api.eu.newrelic.com/log/v1<br>For US use: https://log-api.newrelic.com/log/v1"
 | 
			
		||||
                            label="Endpoint" />
 | 
			
		||||
                    </div>
 | 
			
		||||
                </div>
 | 
			
		||||
                <div class="flex justify-end gap-4 pt-6">
 | 
			
		||||
 
 | 
			
		||||
@@ -11,6 +11,8 @@ use Illuminate\Support\Facades\Route;
 | 
			
		||||
 | 
			
		||||
use App\Http\Controllers\Controller;
 | 
			
		||||
use App\Http\Controllers\MagicController;
 | 
			
		||||
 | 
			
		||||
use App\Livewire\Admin\Index as AdminIndex;
 | 
			
		||||
use App\Livewire\Dev\Compose as Compose;
 | 
			
		||||
 | 
			
		||||
use App\Livewire\Dashboard;
 | 
			
		||||
@@ -77,6 +79,9 @@ use App\Livewire\Waitlist\Index as WaitlistIndex;
 | 
			
		||||
if (isDev()) {
 | 
			
		||||
    Route::get('/dev/compose', Compose::class)->name('dev.compose');
 | 
			
		||||
}
 | 
			
		||||
if (isCloud()) {
 | 
			
		||||
    Route::get('/admin', AdminIndex::class)->name('admin.index');
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Route::post('/forgot-password', [Controller::class, 'forgot_password'])->name('password.forgot');
 | 
			
		||||
Route::get('/api/v1/test/realtime', [Controller::class, 'realtime_test'])->middleware('auth');
 | 
			
		||||
 
 | 
			
		||||
@@ -4,7 +4,7 @@
 | 
			
		||||
            "version": "3.12.36"
 | 
			
		||||
        },
 | 
			
		||||
        "v4": {
 | 
			
		||||
            "version": "4.0.0-beta.211"
 | 
			
		||||
            "version": "4.0.0-beta.212"
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user