feat/fix: Show exited containers on UI & better UX
This commit is contained in:
		| @@ -109,6 +109,7 @@ export async function buildImage({ | ||||
| 		} | ||||
| 	); | ||||
| 	await streamEvents({ stream, docker, buildId, applicationId, debug }); | ||||
| 	await saveBuildLog({ line: `Building image successful!`, buildId, applicationId }); | ||||
| } | ||||
|  | ||||
| export function dockerInstance({ destinationDocker }): { engine: Dockerode; network: string } { | ||||
|   | ||||
| @@ -126,8 +126,8 @@ export async function startTcpProxy( | ||||
| 	const host = getEngine(engine); | ||||
|  | ||||
| 	const containerName = `haproxy-for-${publicPort}`; | ||||
| 	const found = await checkContainer(engine, containerName); | ||||
| 	const foundDependentContainer = await checkContainer(engine, id); | ||||
| 	const found = await checkContainer(engine, containerName, true); | ||||
| 	const foundDependentContainer = await checkContainer(engine, id, true); | ||||
|  | ||||
| 	try { | ||||
| 		if (foundDependentContainer && !found) { | ||||
| @@ -161,8 +161,8 @@ export async function startHttpProxy( | ||||
| 	const host = getEngine(engine); | ||||
|  | ||||
| 	const containerName = `haproxy-for-${publicPort}`; | ||||
| 	const found = await checkContainer(engine, containerName); | ||||
| 	const foundDependentContainer = await checkContainer(engine, id); | ||||
| 	const found = await checkContainer(engine, containerName, true); | ||||
| 	const foundDependentContainer = await checkContainer(engine, id, true); | ||||
|  | ||||
| 	try { | ||||
| 		if (foundDependentContainer && !found) { | ||||
| @@ -186,7 +186,7 @@ export async function startHttpProxy( | ||||
|  | ||||
| export async function startCoolifyProxy(engine: string): Promise<void> { | ||||
| 	const host = getEngine(engine); | ||||
| 	const found = await checkContainer(engine, 'coolify-haproxy'); | ||||
| 	const found = await checkContainer(engine, 'coolify-haproxy', true); | ||||
| 	const { proxyPassword, proxyUser, id } = await db.listSettings(); | ||||
| 	if (!found) { | ||||
| 		const { stdout: Config } = await asyncExecShell( | ||||
| @@ -201,7 +201,25 @@ export async function startCoolifyProxy(engine: string): Promise<void> { | ||||
| 	await configureNetworkCoolifyProxy(engine); | ||||
| } | ||||
|  | ||||
| export async function checkContainer(engine: string, container: string): Promise<boolean> { | ||||
| export async function isContainerExited(engine: string, containerName: string): Promise<boolean> { | ||||
| 	let isExited = false; | ||||
| 	const host = getEngine(engine); | ||||
| 	try { | ||||
| 		const { stdout } = await asyncExecShell( | ||||
| 			`DOCKER_HOST="${host}" docker inspect -f '{{.State.Status}}' ${containerName}` | ||||
| 		); | ||||
| 		if (stdout.trim() === 'exited') { | ||||
| 			isExited = true; | ||||
| 		} | ||||
| 	} catch (error) {} | ||||
|  | ||||
| 	return isExited; | ||||
| } | ||||
| export async function checkContainer( | ||||
| 	engine: string, | ||||
| 	container: string, | ||||
| 	remove: boolean = false | ||||
| ): Promise<boolean> { | ||||
| 	const host = getEngine(engine); | ||||
| 	let containerFound = false; | ||||
|  | ||||
| @@ -212,7 +230,10 @@ export async function checkContainer(engine: string, container: string): Promise | ||||
| 		const parsedStdout = JSON.parse(stdout); | ||||
| 		const status = parsedStdout.Status; | ||||
| 		const isRunning = status === 'running'; | ||||
| 		if (status === 'exited' || status === 'created') { | ||||
| 		if (status === 'created') { | ||||
| 			await asyncExecShell(`DOCKER_HOST="${host}" docker rm ${container}`); | ||||
| 		} | ||||
| 		if (remove && status === 'exited') { | ||||
| 			await asyncExecShell(`DOCKER_HOST="${host}" docker rm ${container}`); | ||||
| 		} | ||||
| 		if (isRunning) { | ||||
|   | ||||
| @@ -5,3 +5,4 @@ export const gitTokens: Writable<{ githubToken: string | null; gitlabToken: stri | ||||
| 		githubToken: null, | ||||
| 		gitlabToken: null | ||||
| 	}); | ||||
| export const disabledButton: Writable<boolean> = writable(false); | ||||
|   | ||||
| @@ -17,7 +17,7 @@ | ||||
| 		const endpoint = `/applications/${params.id}.json`; | ||||
| 		const res = await fetch(endpoint); | ||||
| 		if (res.ok) { | ||||
| 			let { application, isRunning, appId, githubToken, gitlabToken } = await res.json(); | ||||
| 			let { application, isRunning, isExited, appId, githubToken, gitlabToken } = await res.json(); | ||||
| 			if (!application || Object.entries(application).length === 0) { | ||||
| 				return { | ||||
| 					status: 302, | ||||
| @@ -46,6 +46,7 @@ | ||||
| 				props: { | ||||
| 					application, | ||||
| 					isRunning, | ||||
| 					isExited, | ||||
| 					githubToken, | ||||
| 					gitlabToken | ||||
| 				}, | ||||
| @@ -67,6 +68,7 @@ | ||||
| <script lang="ts"> | ||||
| 	export let application; | ||||
| 	export let isRunning; | ||||
| 	export let isExited; | ||||
| 	export let githubToken; | ||||
| 	export let gitlabToken; | ||||
| 	import { page, session } from '$app/stores'; | ||||
| @@ -77,11 +79,18 @@ | ||||
| 	import { goto } from '$app/navigation'; | ||||
| 	import { gitTokens } from '$lib/store'; | ||||
| 	import { toast } from '@zerodevx/svelte-toast'; | ||||
|  | ||||
| 	import { disabledButton } from '$lib/store'; | ||||
| 	if (githubToken) $gitTokens.githubToken = githubToken; | ||||
| 	if (gitlabToken) $gitTokens.gitlabToken = gitlabToken; | ||||
|  | ||||
| 	let loading = false; | ||||
| 	$disabledButton = | ||||
| 		!$session.isAdmin || | ||||
| 		!application.fqdn || | ||||
| 		!application.gitSource || | ||||
| 		!application.repository || | ||||
| 		!application.destinationDocker || | ||||
| 		!application.buildPack; | ||||
| 	const { id } = $page.params; | ||||
|  | ||||
| 	async function handleDeploySubmit() { | ||||
| @@ -127,136 +136,67 @@ | ||||
| 	{#if loading} | ||||
| 		<Loading fullscreen cover /> | ||||
| 	{:else} | ||||
| 		{#if application.fqdn && application.gitSource && application.repository && application.destinationDocker && application.buildPack} | ||||
| 			{#if isRunning} | ||||
| 		{#if isExited} | ||||
| 			<a | ||||
| 				href={!$disabledButton ? `/applications/${id}/logs` : null} | ||||
| 				class=" icons bg-transparent tooltip-bottom text-sm flex items-center text-red-500 tooltip-red-500" | ||||
| 				data-tooltip="Application exited with an error!" | ||||
| 				sveltekit:prefetch | ||||
| 			> | ||||
| 				<svg | ||||
| 					xmlns="http://www.w3.org/2000/svg" | ||||
| 					class="w-6 h-6" | ||||
| 					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="M8.7 3h6.6c.3 0 .5 .1 .7 .3l4.7 4.7c.2 .2 .3 .4 .3 .7v6.6c0 .3 -.1 .5 -.3 .7l-4.7 4.7c-.2 .2 -.4 .3 -.7 .3h-6.6c-.3 0 -.5 -.1 -.7 -.3l-4.7 -4.7c-.2 -.2 -.3 -.4 -.3 -.7v-6.6c0 -.3 .1 -.5 .3 -.7l4.7 -4.7c.2 -.2 .4 -.3 .7 -.3z" | ||||
| 					/> | ||||
| 					<line x1="12" y1="8" x2="12" y2="12" /> | ||||
| 					<line x1="12" y1="16" x2="12.01" y2="16" /> | ||||
| 				</svg> | ||||
| 			</a> | ||||
| 		{/if} | ||||
| 		{#if isRunning} | ||||
| 			<button | ||||
| 				on:click={stopApplication} | ||||
| 				title="Stop application" | ||||
| 				type="submit" | ||||
| 				disabled={$disabledButton} | ||||
| 				class="icons bg-transparent tooltip-bottom text-sm flex items-center space-x-2 text-red-500" | ||||
| 				data-tooltip={$session.isAdmin | ||||
| 					? 'Stop application' | ||||
| 					: 'You do not have permission to stop the application.'} | ||||
| 			> | ||||
| 				<svg | ||||
| 					xmlns="http://www.w3.org/2000/svg" | ||||
| 					class="w-6 h-6" | ||||
| 					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" /> | ||||
| 					<rect x="6" y="5" width="4" height="14" rx="1" /> | ||||
| 					<rect x="14" y="5" width="4" height="14" rx="1" /> | ||||
| 				</svg> | ||||
| 			</button> | ||||
| 			<form on:submit|preventDefault={handleDeploySubmit}> | ||||
| 				<button | ||||
| 					on:click={stopApplication} | ||||
| 					title="Stop application" | ||||
| 					title="Rebuild application" | ||||
| 					type="submit" | ||||
| 					disabled={!$session.isAdmin} | ||||
| 					class="icons bg-transparent tooltip-bottom text-sm flex items-center space-x-2 text-red-500" | ||||
| 					disabled={$disabledButton} | ||||
| 					class="icons bg-transparent tooltip-bottom text-sm flex items-center space-x-2 hover:text-green-500" | ||||
| 					data-tooltip={$session.isAdmin | ||||
| 						? 'Stop application' | ||||
| 						: 'You do not have permission to stop the application.'} | ||||
| 				> | ||||
| 					<svg | ||||
| 						xmlns="http://www.w3.org/2000/svg" | ||||
| 						class="w-6 h-6" | ||||
| 						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" /> | ||||
| 						<rect x="6" y="5" width="4" height="14" rx="1" /> | ||||
| 						<rect x="14" y="5" width="4" height="14" rx="1" /> | ||||
| 					</svg> | ||||
| 				</button> | ||||
| 				<form on:submit|preventDefault={handleDeploySubmit}> | ||||
| 					<button | ||||
| 						title="Rebuild application" | ||||
| 						type="submit" | ||||
| 						disabled={!$session.isAdmin} | ||||
| 						class="icons bg-transparent tooltip-bottom text-sm flex items-center space-x-2 hover:text-green-500" | ||||
| 						data-tooltip={$session.isAdmin | ||||
| 							? 'Rebuild application' | ||||
| 							: 'You do not have permission to rebuild application.'} | ||||
| 					> | ||||
| 						<svg | ||||
| 							xmlns="http://www.w3.org/2000/svg" | ||||
| 							class="w-6 h-6" | ||||
| 							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="M16.3 5h.7a2 2 0 0 1 2 2v10a2 2 0 0 1 -2 2h-10a2 2 0 0 1 -2 -2v-10a2 2 0 0 1 2 -2h5l-2.82 -2.82m0 5.64l2.82 -2.82" | ||||
| 								transform="rotate(-45 12 12)" | ||||
| 							/> | ||||
| 						</svg> | ||||
| 					</button> | ||||
| 				</form> | ||||
| 			{:else} | ||||
| 				<form on:submit|preventDefault={handleDeploySubmit}> | ||||
| 					<button | ||||
| 						title="Build and start application" | ||||
| 						type="submit" | ||||
| 						disabled={!$session.isAdmin} | ||||
| 						class="icons bg-transparent tooltip-bottom text-sm flex items-center space-x-2 text-green-500" | ||||
| 						data-tooltip={$session.isAdmin | ||||
| 							? 'Build and start application' | ||||
| 							: 'You do not have permission to Build and start application.'} | ||||
| 					> | ||||
| 						<svg | ||||
| 							xmlns="http://www.w3.org/2000/svg" | ||||
| 							class="w-6 h-6" | ||||
| 							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="M7 4v16l13 -8z" /> | ||||
| 						</svg> | ||||
| 					</button> | ||||
| 				</form> | ||||
| 			{/if} | ||||
|  | ||||
| 			<div class="border border-stone-700 h-8" /> | ||||
| 			<a | ||||
| 				href="/applications/{id}" | ||||
| 				sveltekit:prefetch | ||||
| 				class="hover:text-yellow-500 rounded" | ||||
| 				class:text-yellow-500={$page.url.pathname === `/applications/${id}`} | ||||
| 				class:bg-coolgray-500={$page.url.pathname === `/applications/${id}`} | ||||
| 			> | ||||
| 				<button | ||||
| 					title="Configurations" | ||||
| 					class="icons bg-transparent tooltip-bottom text-sm disabled:text-red-500" | ||||
| 					data-tooltip="Configurations" | ||||
| 				> | ||||
| 					<svg | ||||
| 						xmlns="http://www.w3.org/2000/svg" | ||||
| 						class="h-6 w-6" | ||||
| 						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" /> | ||||
| 						<rect x="4" y="8" width="4" height="4" /> | ||||
| 						<line x1="6" y1="4" x2="6" y2="8" /> | ||||
| 						<line x1="6" y1="12" x2="6" y2="20" /> | ||||
| 						<rect x="10" y="14" width="4" height="4" /> | ||||
| 						<line x1="12" y1="4" x2="12" y2="14" /> | ||||
| 						<line x1="12" y1="18" x2="12" y2="20" /> | ||||
| 						<rect x="16" y="5" width="4" height="4" /> | ||||
| 						<line x1="18" y1="4" x2="18" y2="5" /> | ||||
| 						<line x1="18" y1="9" x2="18" y2="20" /> | ||||
| 					</svg></button | ||||
| 				></a | ||||
| 			> | ||||
| 			<a | ||||
| 				href="/applications/{id}/secrets" | ||||
| 				sveltekit:prefetch | ||||
| 				class="hover:text-pink-500 rounded" | ||||
| 				class:text-pink-500={$page.url.pathname === `/applications/${id}/secrets`} | ||||
| 				class:bg-coolgray-500={$page.url.pathname === `/applications/${id}/secrets`} | ||||
| 			> | ||||
| 				<button | ||||
| 					title="Secret" | ||||
| 					class="icons bg-transparent tooltip-bottom text-sm disabled:text-red-500" | ||||
| 					data-tooltip="Secret" | ||||
| 						? 'Rebuild application' | ||||
| 						: 'You do not have permission to rebuild application.'} | ||||
| 				> | ||||
| 					<svg | ||||
| 						xmlns="http://www.w3.org/2000/svg" | ||||
| @@ -270,24 +210,22 @@ | ||||
| 					> | ||||
| 						<path stroke="none" d="M0 0h24v24H0z" fill="none" /> | ||||
| 						<path | ||||
| 							d="M12 3a12 12 0 0 0 8.5 3a12 12 0 0 1 -8.5 15a12 12 0 0 1 -8.5 -15a12 12 0 0 0 8.5 -3" | ||||
| 							d="M16.3 5h.7a2 2 0 0 1 2 2v10a2 2 0 0 1 -2 2h-10a2 2 0 0 1 -2 -2v-10a2 2 0 0 1 2 -2h5l-2.82 -2.82m0 5.64l2.82 -2.82" | ||||
| 							transform="rotate(-45 12 12)" | ||||
| 						/> | ||||
| 						<circle cx="12" cy="11" r="1" /> | ||||
| 						<line x1="12" y1="12" x2="12" y2="14.5" /> | ||||
| 					</svg></button | ||||
| 				></a | ||||
| 			> | ||||
| 			<a | ||||
| 				href="/applications/{id}/storage" | ||||
| 				sveltekit:prefetch | ||||
| 				class="hover:text-pink-500 rounded" | ||||
| 				class:text-pink-500={$page.url.pathname === `/applications/${id}/storage`} | ||||
| 				class:bg-coolgray-500={$page.url.pathname === `/applications/${id}/storage`} | ||||
| 			> | ||||
| 					</svg> | ||||
| 				</button> | ||||
| 			</form> | ||||
| 		{:else} | ||||
| 			<form on:submit|preventDefault={handleDeploySubmit}> | ||||
| 				<button | ||||
| 					title="Persistent Storage" | ||||
| 					class="icons bg-transparent tooltip-bottom text-sm disabled:text-red-500" | ||||
| 					data-tooltip="Persistent Storage" | ||||
| 					title="Build and start application" | ||||
| 					type="submit" | ||||
| 					disabled={$disabledButton} | ||||
| 					class="icons bg-transparent tooltip-bottom text-sm flex items-center space-x-2 text-green-500" | ||||
| 					data-tooltip={$session.isAdmin | ||||
| 						? 'Build and start application' | ||||
| 						: 'You do not have permission to Build and start application.'} | ||||
| 				> | ||||
| 					<svg | ||||
| 						xmlns="http://www.w3.org/2000/svg" | ||||
| @@ -300,112 +238,213 @@ | ||||
| 						stroke-linejoin="round" | ||||
| 					> | ||||
| 						<path stroke="none" d="M0 0h24v24H0z" fill="none" /> | ||||
| 						<ellipse cx="12" cy="6" rx="8" ry="3" /> | ||||
| 						<path d="M4 6v6a8 3 0 0 0 16 0v-6" /> | ||||
| 						<path d="M4 12v6a8 3 0 0 0 16 0v-6" /> | ||||
| 						<path d="M7 4v16l13 -8z" /> | ||||
| 					</svg> | ||||
| 				</button></a | ||||
| 			> | ||||
| 			<a | ||||
| 				href="/applications/{id}/previews" | ||||
| 				sveltekit:prefetch | ||||
| 				class="hover:text-orange-500 rounded" | ||||
| 				class:text-orange-500={$page.url.pathname === `/applications/${id}/previews`} | ||||
| 				class:bg-coolgray-500={$page.url.pathname === `/applications/${id}/previews`} | ||||
| 			> | ||||
| 				<button | ||||
| 					title="Previews" | ||||
| 					class="icons bg-transparent tooltip-bottom text-sm disabled:text-red-500" | ||||
| 					data-tooltip="Previews" | ||||
| 				> | ||||
| 					<svg | ||||
| 						xmlns="http://www.w3.org/2000/svg" | ||||
| 						class="w-6 h-6" | ||||
| 						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" /> | ||||
| 						<circle cx="7" cy="18" r="2" /> | ||||
| 						<circle cx="7" cy="6" r="2" /> | ||||
| 						<circle cx="17" cy="12" r="2" /> | ||||
| 						<line x1="7" y1="8" x2="7" y2="16" /> | ||||
| 						<path d="M7 8a4 4 0 0 0 4 4h4" /> | ||||
| 					</svg></button | ||||
| 				></a | ||||
| 			> | ||||
| 			<div class="border border-stone-700 h-8" /> | ||||
| 			<a | ||||
| 				href="/applications/{id}/logs" | ||||
| 				sveltekit:prefetch | ||||
| 				class="hover:text-sky-500 rounded" | ||||
| 				class:text-sky-500={$page.url.pathname === `/applications/${id}/logs`} | ||||
| 				class:bg-coolgray-500={$page.url.pathname === `/applications/${id}/logs`} | ||||
| 			> | ||||
| 				<button | ||||
| 					title="Application Logs" | ||||
| 					class="icons bg-transparent tooltip-bottom text-sm disabled:text-red-500 " | ||||
| 					data-tooltip="Application Logs" | ||||
| 				> | ||||
| 					<svg | ||||
| 						xmlns="http://www.w3.org/2000/svg" | ||||
| 						class="h-6 w-6" | ||||
| 						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="M3 19a9 9 0 0 1 9 0a9 9 0 0 1 9 0" /> | ||||
| 						<path d="M3 6a9 9 0 0 1 9 0a9 9 0 0 1 9 0" /> | ||||
| 						<line x1="3" y1="6" x2="3" y2="19" /> | ||||
| 						<line x1="12" y1="6" x2="12" y2="19" /> | ||||
| 						<line x1="21" y1="6" x2="21" y2="19" /> | ||||
| 					</svg> | ||||
| 				</button></a | ||||
| 			> | ||||
| 			<a | ||||
| 				href="/applications/{id}/logs/build" | ||||
| 				sveltekit:prefetch | ||||
| 				class="hover:text-red-500 rounded" | ||||
| 				class:text-red-500={$page.url.pathname === `/applications/${id}/logs/build`} | ||||
| 				class:bg-coolgray-500={$page.url.pathname === `/applications/${id}/logs/build`} | ||||
| 			> | ||||
| 				<button | ||||
| 					title="Build Logs" | ||||
| 					class="icons bg-transparent tooltip-bottom text-sm disabled:text-red-500 " | ||||
| 					data-tooltip="Build Logs" | ||||
| 				> | ||||
| 					<svg | ||||
| 						xmlns="http://www.w3.org/2000/svg" | ||||
| 						class="h-6 w-6" | ||||
| 						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" /> | ||||
| 						<circle cx="19" cy="13" r="2" /> | ||||
| 						<circle cx="4" cy="17" r="2" /> | ||||
| 						<circle cx="13" cy="17" r="2" /> | ||||
| 						<line x1="13" y1="19" x2="4" y2="19" /> | ||||
| 						<line x1="4" y1="15" x2="13" y2="15" /> | ||||
| 						<path d="M8 12v-5h2a3 3 0 0 1 3 3v5" /> | ||||
| 						<path d="M5 15v-2a1 1 0 0 1 1 -1h7" /> | ||||
| 						<path d="M19 11v-7l-6 7" /> | ||||
| 					</svg> | ||||
| 				</button></a | ||||
| 			> | ||||
| 			<div class="border border-stone-700 h-8" /> | ||||
| 				</button> | ||||
| 			</form> | ||||
| 		{/if} | ||||
|  | ||||
| 		<div class="border border-coolgray-500 h-8" /> | ||||
| 		<a | ||||
| 			href={!$disabledButton ? `/applications/${id}` : null} | ||||
| 			sveltekit:prefetch | ||||
| 			class="hover:text-yellow-500 rounded" | ||||
| 			class:text-yellow-500={$page.url.pathname === `/applications/${id}`} | ||||
| 			class:bg-coolgray-500={$page.url.pathname === `/applications/${id}`} | ||||
| 		> | ||||
| 			<button | ||||
| 				title="Configurations" | ||||
| 				disabled={$disabledButton} | ||||
| 				class="icons bg-transparent tooltip-bottom text-sm" | ||||
| 				data-tooltip="Configurations" | ||||
| 			> | ||||
| 				<svg | ||||
| 					xmlns="http://www.w3.org/2000/svg" | ||||
| 					class="h-6 w-6" | ||||
| 					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" /> | ||||
| 					<rect x="4" y="8" width="4" height="4" /> | ||||
| 					<line x1="6" y1="4" x2="6" y2="8" /> | ||||
| 					<line x1="6" y1="12" x2="6" y2="20" /> | ||||
| 					<rect x="10" y="14" width="4" height="4" /> | ||||
| 					<line x1="12" y1="4" x2="12" y2="14" /> | ||||
| 					<line x1="12" y1="18" x2="12" y2="20" /> | ||||
| 					<rect x="16" y="5" width="4" height="4" /> | ||||
| 					<line x1="18" y1="4" x2="18" y2="5" /> | ||||
| 					<line x1="18" y1="9" x2="18" y2="20" /> | ||||
| 				</svg></button | ||||
| 			></a | ||||
| 		> | ||||
| 		<a | ||||
| 			href={!$disabledButton ? `/applications/${id}/secrets` : null} | ||||
| 			sveltekit:prefetch | ||||
| 			class="hover:text-pink-500 rounded" | ||||
| 			class:text-pink-500={$page.url.pathname === `/applications/${id}/secrets`} | ||||
| 			class:bg-coolgray-500={$page.url.pathname === `/applications/${id}/secrets`} | ||||
| 		> | ||||
| 			<button | ||||
| 				title="Secret" | ||||
| 				disabled={$disabledButton} | ||||
| 				class="icons bg-transparent tooltip-bottom text-sm" | ||||
| 				data-tooltip="Secret" | ||||
| 			> | ||||
| 				<svg | ||||
| 					xmlns="http://www.w3.org/2000/svg" | ||||
| 					class="w-6 h-6" | ||||
| 					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 3a12 12 0 0 0 8.5 3a12 12 0 0 1 -8.5 15a12 12 0 0 1 -8.5 -15a12 12 0 0 0 8.5 -3" | ||||
| 					/> | ||||
| 					<circle cx="12" cy="11" r="1" /> | ||||
| 					<line x1="12" y1="12" x2="12" y2="14.5" /> | ||||
| 				</svg></button | ||||
| 			></a | ||||
| 		> | ||||
| 		<a | ||||
| 			href={!$disabledButton ? `/applications/${id}/storage` : null} | ||||
| 			sveltekit:prefetch | ||||
| 			class="hover:text-pink-500 rounded" | ||||
| 			class:text-pink-500={$page.url.pathname === `/applications/${id}/storage`} | ||||
| 			class:bg-coolgray-500={$page.url.pathname === `/applications/${id}/storage`} | ||||
| 		> | ||||
| 			<button | ||||
| 				title="Persistent Storage" | ||||
| 				disabled={$disabledButton} | ||||
| 				class="icons bg-transparent tooltip-bottom text-sm" | ||||
| 				data-tooltip="Persistent Storage" | ||||
| 			> | ||||
| 				<svg | ||||
| 					xmlns="http://www.w3.org/2000/svg" | ||||
| 					class="w-6 h-6" | ||||
| 					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" /> | ||||
| 					<ellipse cx="12" cy="6" rx="8" ry="3" /> | ||||
| 					<path d="M4 6v6a8 3 0 0 0 16 0v-6" /> | ||||
| 					<path d="M4 12v6a8 3 0 0 0 16 0v-6" /> | ||||
| 				</svg> | ||||
| 			</button></a | ||||
| 		> | ||||
| 		<a | ||||
| 			href={!$disabledButton ? `/applications/${id}/previews` : null} | ||||
| 			sveltekit:prefetch | ||||
| 			class="hover:text-orange-500 rounded" | ||||
| 			class:text-orange-500={$page.url.pathname === `/applications/${id}/previews`} | ||||
| 			class:bg-coolgray-500={$page.url.pathname === `/applications/${id}/previews`} | ||||
| 		> | ||||
| 			<button | ||||
| 				title="Previews" | ||||
| 				disabled={$disabledButton} | ||||
| 				class="icons bg-transparent tooltip-bottom text-sm" | ||||
| 				data-tooltip="Previews" | ||||
| 			> | ||||
| 				<svg | ||||
| 					xmlns="http://www.w3.org/2000/svg" | ||||
| 					class="w-6 h-6" | ||||
| 					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" /> | ||||
| 					<circle cx="7" cy="18" r="2" /> | ||||
| 					<circle cx="7" cy="6" r="2" /> | ||||
| 					<circle cx="17" cy="12" r="2" /> | ||||
| 					<line x1="7" y1="8" x2="7" y2="16" /> | ||||
| 					<path d="M7 8a4 4 0 0 0 4 4h4" /> | ||||
| 				</svg></button | ||||
| 			></a | ||||
| 		> | ||||
| 		<div class="border border-coolgray-500 h-8" /> | ||||
| 		<a | ||||
| 			href={!$disabledButton ? `/applications/${id}/logs` : null} | ||||
| 			sveltekit:prefetch | ||||
| 			class="hover:text-sky-500 rounded" | ||||
| 			class:text-sky-500={$page.url.pathname === `/applications/${id}/logs`} | ||||
| 			class:bg-coolgray-500={$page.url.pathname === `/applications/${id}/logs`} | ||||
| 		> | ||||
| 			<button | ||||
| 				title="Application Logs" | ||||
| 				disabled={$disabledButton} | ||||
| 				class="icons bg-transparent tooltip-bottom text-sm" | ||||
| 				data-tooltip="Application Logs" | ||||
| 			> | ||||
| 				<svg | ||||
| 					xmlns="http://www.w3.org/2000/svg" | ||||
| 					class="h-6 w-6" | ||||
| 					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="M3 19a9 9 0 0 1 9 0a9 9 0 0 1 9 0" /> | ||||
| 					<path d="M3 6a9 9 0 0 1 9 0a9 9 0 0 1 9 0" /> | ||||
| 					<line x1="3" y1="6" x2="3" y2="19" /> | ||||
| 					<line x1="12" y1="6" x2="12" y2="19" /> | ||||
| 					<line x1="21" y1="6" x2="21" y2="19" /> | ||||
| 				</svg> | ||||
| 			</button></a | ||||
| 		> | ||||
| 		<a | ||||
| 			href={!$disabledButton ? `/applications/${id}/logs/build` : null} | ||||
| 			sveltekit:prefetch | ||||
| 			class="hover:text-red-500 rounded" | ||||
| 			class:text-red-500={$page.url.pathname === `/applications/${id}/logs/build`} | ||||
| 			class:bg-coolgray-500={$page.url.pathname === `/applications/${id}/logs/build`} | ||||
| 		> | ||||
| 			<button | ||||
| 				title="Build Logs" | ||||
| 				disabled={$disabledButton} | ||||
| 				class="icons bg-transparent tooltip-bottom text-sm" | ||||
| 				data-tooltip="Build Logs" | ||||
| 			> | ||||
| 				<svg | ||||
| 					xmlns="http://www.w3.org/2000/svg" | ||||
| 					class="h-6 w-6" | ||||
| 					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" /> | ||||
| 					<circle cx="19" cy="13" r="2" /> | ||||
| 					<circle cx="4" cy="17" r="2" /> | ||||
| 					<circle cx="13" cy="17" r="2" /> | ||||
| 					<line x1="13" y1="19" x2="4" y2="19" /> | ||||
| 					<line x1="4" y1="15" x2="13" y2="15" /> | ||||
| 					<path d="M8 12v-5h2a3 3 0 0 1 3 3v5" /> | ||||
| 					<path d="M5 15v-2a1 1 0 0 1 1 -1h7" /> | ||||
| 					<path d="M19 11v-7l-6 7" /> | ||||
| 				</svg> | ||||
| 			</button></a | ||||
| 		> | ||||
| 		<div class="border border-coolgray-500 h-8" /> | ||||
|  | ||||
| 		<button | ||||
| 			on:click={() => deleteApplication(application.name)} | ||||
| 			title="Delete application" | ||||
|   | ||||
| @@ -19,13 +19,14 @@ | ||||
| 			const tempBuildPack = JSON.parse( | ||||
| 				JSON.stringify(findBuildPack(buildPack.name, packageManager)) | ||||
| 			); | ||||
|  | ||||
| 			delete tempBuildPack.name; | ||||
| 			delete tempBuildPack.fancyName; | ||||
| 			delete tempBuildPack.color; | ||||
| 			delete tempBuildPack.hoverColor; | ||||
|  | ||||
| 			if (foundConfig.buildPack !== name) { | ||||
| 				await post(`/applications/${id}.json`, { ...tempBuildPack }); | ||||
| 				await post(`/applications/${id}.json`, { ...tempBuildPack, buildPack: name }); | ||||
| 			} | ||||
| 			await post(`/applications/${id}/configuration/buildpack.json`, { buildPack: name }); | ||||
| 			return await goto(from || `/applications/${id}`); | ||||
|   | ||||
| @@ -31,7 +31,7 @@ | ||||
|  | ||||
| 	import { buildPacks, findBuildPack, scanningTemplates } from '$lib/components/templates'; | ||||
| 	import BuildPack from './_BuildPack.svelte'; | ||||
| 	import { page, session } from '$app/stores'; | ||||
| 	import { page } from '$app/stores'; | ||||
| 	import { get } from '$lib/api'; | ||||
| 	import { errorNotification } from '$lib/form'; | ||||
| 	import { gitTokens } from '$lib/store'; | ||||
| @@ -221,7 +221,7 @@ | ||||
| 	<div class="max-w-7xl mx-auto flex flex-wrap justify-center"> | ||||
| 		{#each buildPacks as buildPack} | ||||
| 			<div class="p-2"> | ||||
| 				<BuildPack {buildPack} {scanning} {packageManager} bind:foundConfig /> | ||||
| 				<BuildPack {buildPack} {scanning} bind:foundConfig /> | ||||
| 			</div> | ||||
| 		{/each} | ||||
| 	</div> | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| import { getUserDetails } from '$lib/common'; | ||||
| import * as db from '$lib/database'; | ||||
| import { ErrorHandler } from '$lib/database'; | ||||
| import { checkContainer } from '$lib/haproxy'; | ||||
| import { checkContainer, isContainerExited } from '$lib/haproxy'; | ||||
| import type { RequestHandler } from '@sveltejs/kit'; | ||||
| import jsonwebtoken from 'jsonwebtoken'; | ||||
| import { get as getRequest } from '$lib/api'; | ||||
| @@ -15,17 +15,20 @@ export const get: RequestHandler = async (event) => { | ||||
|  | ||||
| 	const appId = process.env['COOLIFY_APP_ID']; | ||||
| 	let isRunning = false; | ||||
| 	let isExited = false; | ||||
| 	let githubToken = event.locals.cookies?.githubToken || null; | ||||
| 	let gitlabToken = event.locals.cookies?.gitlabToken || null; | ||||
| 	try { | ||||
| 		const application = await db.getApplication({ id, teamId }); | ||||
| 		if (application.destinationDockerId) { | ||||
| 			isRunning = await checkContainer(application.destinationDocker.engine, id); | ||||
| 			isExited = await isContainerExited(application.destinationDocker.engine, id); | ||||
| 		} | ||||
| 		return { | ||||
| 			status: 200, | ||||
| 			body: { | ||||
| 				isRunning, | ||||
| 				isExited, | ||||
| 				application, | ||||
| 				appId, | ||||
| 				githubToken, | ||||
| @@ -57,11 +60,14 @@ export const post: RequestHandler = async (event) => { | ||||
| 		pythonWSGI, | ||||
| 		pythonModule, | ||||
| 		pythonVariable, | ||||
| 		dockerFileLocation | ||||
| 		dockerFileLocation, | ||||
| 		denoMainFile, | ||||
| 		denoOptions | ||||
| 	} = await event.request.json(); | ||||
| 	if (port) port = Number(port); | ||||
|  | ||||
| 	try { | ||||
| 		console.log(buildPack); | ||||
| 		const defaultConfiguration = await setDefaultConfiguration({ | ||||
| 			buildPack, | ||||
| 			port, | ||||
| @@ -70,7 +76,8 @@ export const post: RequestHandler = async (event) => { | ||||
| 			buildCommand, | ||||
| 			publishDirectory, | ||||
| 			baseDirectory, | ||||
| 			dockerFileLocation | ||||
| 			dockerFileLocation, | ||||
| 			denoMainFile | ||||
| 		}); | ||||
| 		await db.configureApplication({ | ||||
| 			id, | ||||
| @@ -87,6 +94,8 @@ export const post: RequestHandler = async (event) => { | ||||
| 			pythonModule, | ||||
| 			pythonVariable, | ||||
| 			dockerFileLocation, | ||||
| 			denoMainFile, | ||||
| 			denoOptions, | ||||
| 			...defaultConfiguration | ||||
| 		}); | ||||
| 		return { status: 201 }; | ||||
|   | ||||
| @@ -48,6 +48,7 @@ | ||||
| 	import { post } from '$lib/api'; | ||||
| 	import cuid from 'cuid'; | ||||
| 	import { browser } from '$app/env'; | ||||
| 	import { disabledButton } from '$lib/store'; | ||||
| 	const { id } = $page.params; | ||||
|  | ||||
| 	let domainEl: HTMLInputElement; | ||||
| @@ -122,7 +123,8 @@ | ||||
| 		try { | ||||
| 			await post(`/applications/${id}/check.json`, { fqdn: application.fqdn, forceSave }); | ||||
| 			await post(`/applications/${id}.json`, { ...application }); | ||||
| 			return window.location.reload(); | ||||
| 			$disabledButton = false; | ||||
| 			return toast.push('Configurations saved.'); | ||||
| 		} catch ({ error }) { | ||||
| 			if (error?.startsWith('DNS not set')) { | ||||
| 				forceSave = true; | ||||
| @@ -434,13 +436,27 @@ | ||||
| 			{/if} | ||||
| 			{#if application.buildPack === 'deno'} | ||||
| 				<div class="grid grid-cols-2 items-center"> | ||||
| 					<label for="startCommand" class="text-base font-bold text-stone-100">Start Command</label> | ||||
| 					<label for="denoMainFile" class="text-base font-bold text-stone-100">Main File</label> | ||||
| 					<input | ||||
| 						readonly={!$session.isAdmin} | ||||
| 						name="startCommand" | ||||
| 						id="startCommand" | ||||
| 						bind:value={application.startCommand} | ||||
| 						placeholder="default: deno run main.js" | ||||
| 						name="denoMainFile" | ||||
| 						id="denoMainFile" | ||||
| 						bind:value={application.denoMainFile} | ||||
| 						placeholder="default: main.ts" | ||||
| 					/> | ||||
| 				</div> | ||||
| 				<div class="grid grid-cols-2 items-center"> | ||||
| 					<label for="denoOptions" class="text-base font-bold text-stone-100">List of Options</label | ||||
| 					> | ||||
| 					<input | ||||
| 						readonly={!$session.isAdmin} | ||||
| 						name="denoOptions" | ||||
| 						id="denoOptions" | ||||
| 						bind:value={application.denoOptions} | ||||
| 						placeholder="eg: --allow-net --allow-hrtime --config path/to/file.json" | ||||
| 					/> | ||||
| 					<Explainer | ||||
| 						text="List of options to pass to <span class='text-green-500 font-bold'>deno</span> command. Could include permissions, configurations files, etc." | ||||
| 					/> | ||||
| 				</div> | ||||
| 			{/if} | ||||
|   | ||||
| @@ -119,7 +119,7 @@ a { | ||||
| } | ||||
|  | ||||
| .icons { | ||||
| 	@apply rounded p-2 transition duration-200 hover:bg-coolgray-500 disabled:bg-coolblack !important; | ||||
| 	@apply rounded p-2 transition duration-200 hover:bg-coolgray-500 disabled:bg-coolblack disabled:text-coolgray-500 !important; | ||||
| } | ||||
|  | ||||
| .arrow-right-applications { | ||||
|   | ||||
		Reference in New Issue
	
	Block a user
	 Andras Bacsai
					Andras Bacsai