wip: trpc
This commit is contained in:
		
							
								
								
									
										3
									
								
								apps/client/src/app.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								apps/client/src/app.d.ts
									
									
									
									
										vendored
									
									
								
							| @@ -7,3 +7,6 @@ declare namespace App { | ||||
| 	// interface Error {} | ||||
| 	// interface Platform {} | ||||
| } | ||||
|  | ||||
| declare const GITPOD_WORKSPACE_URL: string; | ||||
| declare const CODESANDBOX_HOST: string; | ||||
|   | ||||
| @@ -1,14 +1,17 @@ | ||||
| import { dev } from '$app/environment'; | ||||
| import { addToast } from './store'; | ||||
|  | ||||
| import Cookies from 'js-cookie'; | ||||
| export const asyncSleep = (delay: number) => new Promise((resolve) => setTimeout(resolve, delay)); | ||||
|  | ||||
| export function errorNotification(error: any | { message: string }): void { | ||||
| 	if (error instanceof Error) { | ||||
| 		console.error(error.message) | ||||
| 		addToast({ | ||||
| 			message: error.message, | ||||
| 			type: 'error' | ||||
| 		}); | ||||
| 	} else { | ||||
| 		console.error(error) | ||||
| 		addToast({ | ||||
| 			message: error, | ||||
| 			type: 'error' | ||||
| @@ -18,3 +21,165 @@ export function errorNotification(error: any | { message: string }): void { | ||||
| export function getRndInteger(min: number, max: number) { | ||||
| 	return Math.floor(Math.random() * (max - min + 1)) + min; | ||||
| } | ||||
|  | ||||
| export function getDomain(domain: string) { | ||||
| 	return domain?.replace('https://', '').replace('http://', ''); | ||||
| } | ||||
|  | ||||
| export const notNodeDeployments = ['php', 'docker', 'rust', 'python', 'deno', 'laravel', 'heroku']; | ||||
| export const staticDeployments = [ | ||||
| 	'react', | ||||
| 	'vuejs', | ||||
| 	'static', | ||||
| 	'svelte', | ||||
| 	'gatsby', | ||||
| 	'php', | ||||
| 	'astro', | ||||
| 	'eleventy' | ||||
| ]; | ||||
|  | ||||
| export function getAPIUrl() { | ||||
| 	if (GITPOD_WORKSPACE_URL) { | ||||
| 		const { href } = new URL(GITPOD_WORKSPACE_URL); | ||||
| 		const newURL = href.replace('https://', 'https://3001-').replace(/\/$/, ''); | ||||
| 		return newURL; | ||||
| 	} | ||||
| 	if (CODESANDBOX_HOST) { | ||||
| 		return `https://${CODESANDBOX_HOST.replace(/\$PORT/, '3001')}`; | ||||
| 	} | ||||
| 	return dev ? `http://${window.location.hostname}:3001` : 'http://localhost:3000'; | ||||
| } | ||||
| export function getWebhookUrl(type: string) { | ||||
| 	if (GITPOD_WORKSPACE_URL) { | ||||
| 		const { href } = new URL(GITPOD_WORKSPACE_URL); | ||||
| 		const newURL = href.replace('https://', 'https://3001-').replace(/\/$/, ''); | ||||
| 		if (type === 'github') { | ||||
| 			return `${newURL}/webhooks/github/events`; | ||||
| 		} | ||||
| 		if (type === 'gitlab') { | ||||
| 			return `${newURL}/webhooks/gitlab/events`; | ||||
| 		} | ||||
| 	} | ||||
| 	if (CODESANDBOX_HOST) { | ||||
| 		const newURL = `https://${CODESANDBOX_HOST.replace(/\$PORT/, '3001')}`; | ||||
| 		if (type === 'github') { | ||||
| 			return `${newURL}/webhooks/github/events`; | ||||
| 		} | ||||
| 		if (type === 'gitlab') { | ||||
| 			return `${newURL}/webhooks/gitlab/events`; | ||||
| 		} | ||||
| 	} | ||||
| 	return `https://webhook.site/0e5beb2c-4e9b-40e2-a89e-32295e570c21/events`; | ||||
| } | ||||
|  | ||||
| async function send({ | ||||
| 	method, | ||||
| 	path, | ||||
| 	data = null, | ||||
| 	headers, | ||||
| 	timeout = 120000 | ||||
| }: { | ||||
| 	method: string; | ||||
| 	path: string; | ||||
| 	data?: any; | ||||
| 	headers?: any; | ||||
| 	timeout?: number; | ||||
| }): Promise<Record<string, unknown>> { | ||||
| 	const token = Cookies.get('token'); | ||||
| 	const controller = new AbortController(); | ||||
| 	const id = setTimeout(() => controller.abort(), timeout); | ||||
| 	const opts: any = { method, headers: {}, body: null, signal: controller.signal }; | ||||
| 	if (data && Object.keys(data).length > 0) { | ||||
| 		const parsedData = data; | ||||
| 		for (const [key, value] of Object.entries(data)) { | ||||
| 			if (value === '') { | ||||
| 				parsedData[key] = null; | ||||
| 			} | ||||
| 		} | ||||
| 		if (parsedData) { | ||||
| 			opts.headers['Content-Type'] = 'application/json'; | ||||
| 			opts.body = JSON.stringify(parsedData); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if (headers) { | ||||
| 		opts.headers = { | ||||
| 			...opts.headers, | ||||
| 			...headers | ||||
| 		}; | ||||
| 	} | ||||
| 	if (token && !path.startsWith('https://')) { | ||||
| 		opts.headers = { | ||||
| 			...opts.headers, | ||||
| 			Authorization: `Bearer ${token}` | ||||
| 		}; | ||||
| 	} | ||||
| 	if (!path.startsWith('https://')) { | ||||
| 		path = `/api/v1${path}`; | ||||
| 	} | ||||
|  | ||||
| 	if (dev && !path.startsWith('https://')) { | ||||
| 		path = `${getAPIUrl()}${path}`; | ||||
| 	} | ||||
| 	if (method === 'POST' && data && !opts.body) { | ||||
| 		opts.body = data; | ||||
| 	} | ||||
| 	const response = await fetch(`${path}`, opts); | ||||
|  | ||||
| 	clearTimeout(id); | ||||
|  | ||||
| 	const contentType = response.headers.get('content-type'); | ||||
|  | ||||
| 	let responseData = {}; | ||||
| 	if (contentType) { | ||||
| 		if (contentType?.indexOf('application/json') !== -1) { | ||||
| 			responseData = await response.json(); | ||||
| 		} else if (contentType?.indexOf('text/plain') !== -1) { | ||||
| 			responseData = await response.text(); | ||||
| 		} else { | ||||
| 			return {}; | ||||
| 		} | ||||
| 	} else { | ||||
| 		return {}; | ||||
| 	} | ||||
| 	if (!response.ok) { | ||||
| 		if ( | ||||
| 			response.status === 401 && | ||||
| 			!path.startsWith('https://api.github') && | ||||
| 			!path.includes('/v4/') | ||||
| 		) { | ||||
| 			Cookies.remove('token'); | ||||
| 		} | ||||
|  | ||||
| 		throw responseData; | ||||
| 	} | ||||
| 	return responseData; | ||||
| } | ||||
|  | ||||
| export function get(path: string, headers?: Record<string, unknown>): Promise<Record<string, any>> { | ||||
| 	return send({ method: 'GET', path, headers }); | ||||
| } | ||||
|  | ||||
| export function del( | ||||
| 	path: string, | ||||
| 	data: Record<string, unknown>, | ||||
| 	headers?: Record<string, unknown> | ||||
| ): Promise<Record<string, any>> { | ||||
| 	return send({ method: 'DELETE', path, data, headers }); | ||||
| } | ||||
|  | ||||
| export function post( | ||||
| 	path: string, | ||||
| 	data: Record<string, unknown> | FormData, | ||||
| 	headers?: Record<string, unknown> | ||||
| ): Promise<Record<string, any>> { | ||||
| 	return send({ method: 'POST', path, data, headers }); | ||||
| } | ||||
|  | ||||
| export function put( | ||||
| 	path: string, | ||||
| 	data: Record<string, unknown>, | ||||
| 	headers?: Record<string, unknown> | ||||
| ): Promise<Record<string, any>> { | ||||
| 	return send({ method: 'PUT', path, data, headers }); | ||||
| } | ||||
|   | ||||
							
								
								
									
										1
									
								
								apps/client/src/lib/components/Beta.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								apps/client/src/lib/components/Beta.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| <span class="badge bg-coollabs-gradient rounded text-white font-normal"> BETA </span> | ||||
							
								
								
									
										156
									
								
								apps/client/src/lib/components/CopyPasswordField.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										156
									
								
								apps/client/src/lib/components/CopyPasswordField.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,156 @@ | ||||
| <script lang="ts"> | ||||
| 	import { browser } from '$app/environment'; | ||||
| 	import { addToast } from '$lib/store'; | ||||
| 	let showPassword = false; | ||||
|  | ||||
| 	export let value: string; | ||||
| 	export let disabled = false; | ||||
| 	export let isPasswordField = false; | ||||
| 	export let readonly = false; | ||||
| 	export let textarea = false; | ||||
| 	export let required = false; | ||||
| 	export let pattern: string | null | undefined = null; | ||||
| 	export let id: string; | ||||
| 	export let name: string; | ||||
| 	export let placeholder = ''; | ||||
| 	export let inputStyle = ''; | ||||
|  | ||||
| 	let disabledClass = 'input input-primary bg-coolback disabled:bg-coolblack w-full'; | ||||
| 	let isHttps = browser && window.location.protocol === 'https:'; | ||||
|  | ||||
| 	function copyToClipboard() { | ||||
| 		if (isHttps && navigator.clipboard) { | ||||
| 			navigator.clipboard.writeText(value); | ||||
| 			addToast({ | ||||
| 				message: 'Copied to clipboard.', | ||||
| 				type: 'success' | ||||
| 			}); | ||||
| 		} | ||||
| 	} | ||||
| </script> | ||||
|  | ||||
| <div class="relative"> | ||||
| 	{#if !isPasswordField || showPassword} | ||||
| 		{#if textarea} | ||||
| 			<textarea | ||||
| 				style={inputStyle} | ||||
| 				rows="5" | ||||
| 				class={disabledClass} | ||||
| 				class:pr-10={true} | ||||
| 				class:pr-20={value && isHttps} | ||||
| 				class:border={required && !value} | ||||
| 				class:border-red-500={required && !value} | ||||
| 				{placeholder} | ||||
| 				type="text" | ||||
| 				{id} | ||||
| 				{pattern} | ||||
| 				{required} | ||||
| 				{readonly} | ||||
| 				{disabled} | ||||
| 				{name}>{value}</textarea | ||||
| 			> | ||||
| 		{:else} | ||||
| 			<input | ||||
| 				style={inputStyle} | ||||
| 				class={disabledClass} | ||||
| 				type="text" | ||||
| 				class:pr-10={true} | ||||
| 				class:pr-20={value && isHttps} | ||||
| 				class:border={required && !value} | ||||
| 				class:border-red-500={required && !value} | ||||
| 				{id} | ||||
| 				{name} | ||||
| 				{required} | ||||
| 				{pattern} | ||||
| 				{readonly} | ||||
| 				bind:value | ||||
| 				{disabled} | ||||
| 				{placeholder} | ||||
| 			/> | ||||
| 		{/if} | ||||
| 	{:else} | ||||
| 		<input | ||||
| 			style={inputStyle} | ||||
| 			class={disabledClass} | ||||
| 			class:pr-10={true} | ||||
| 			class:pr-20={value && isHttps} | ||||
| 			class:border={required && !value} | ||||
| 			class:border-red-500={required && !value} | ||||
| 			type="password" | ||||
| 			{id} | ||||
| 			{name} | ||||
| 			{readonly} | ||||
| 			{pattern} | ||||
| 			{required} | ||||
| 			bind:value | ||||
| 			{disabled} | ||||
| 			{placeholder} | ||||
| 		/> | ||||
| 	{/if} | ||||
|  | ||||
| 	<div class="absolute top-0 right-0 flex justify-center items-center h-full cursor-pointer text-stone-600 hover:text-white mr-3"> | ||||
| 		<div class="flex space-x-2"> | ||||
| 			{#if isPasswordField} | ||||
| 				<!-- svelte-ignore a11y-click-events-have-key-events --> | ||||
| 				<div on:click={() => (showPassword = !showPassword)}> | ||||
| 					{#if showPassword} | ||||
| 						<svg | ||||
| 							xmlns="http://www.w3.org/2000/svg" | ||||
| 							class="h-6 w-6" | ||||
| 							fill="none" | ||||
| 							viewBox="0 0 24 24" | ||||
| 							stroke="currentColor" | ||||
| 						> | ||||
| 							<path | ||||
| 								stroke-linecap="round" | ||||
| 								stroke-linejoin="round" | ||||
| 								stroke-width="2" | ||||
| 								d="M13.875 18.825A10.05 10.05 0 0112 19c-4.478 0-8.268-2.943-9.543-7a9.97 9.97 0 011.563-3.029m5.858.908a3 3 0 114.243 4.243M9.878 9.878l4.242 4.242M9.88 9.88l-3.29-3.29m7.532 7.532l3.29 3.29M3 3l3.59 3.59m0 0A9.953 9.953 0 0112 5c4.478 0 8.268 2.943 9.543 7a10.025 10.025 0 01-4.132 5.411m0 0L21 21" | ||||
| 							/> | ||||
| 						</svg> | ||||
| 					{:else} | ||||
| 						<svg | ||||
| 							xmlns="http://www.w3.org/2000/svg" | ||||
| 							class="h-6 w-6" | ||||
| 							fill="none" | ||||
| 							viewBox="0 0 24 24" | ||||
| 							stroke="currentColor" | ||||
| 						> | ||||
| 							<path | ||||
| 								stroke-linecap="round" | ||||
| 								stroke-linejoin="round" | ||||
| 								stroke-width="2" | ||||
| 								d="M15 12a3 3 0 11-6 0 3 3 0 016 0z" | ||||
| 							/> | ||||
| 							<path | ||||
| 								stroke-linecap="round" | ||||
| 								stroke-linejoin="round" | ||||
| 								stroke-width="2" | ||||
| 								d="M2.458 12C3.732 7.943 7.523 5 12 5c4.478 0 8.268 2.943 9.542 7-1.274 4.057-5.064 7-9.542 7-4.477 0-8.268-2.943-9.542-7z" | ||||
| 							/> | ||||
| 						</svg> | ||||
| 					{/if} | ||||
| 				</div> | ||||
| 			{/if} | ||||
| 			{#if value && isHttps} | ||||
| 				<!-- svelte-ignore a11y-click-events-have-key-events --> | ||||
| 				<div on:click={copyToClipboard}> | ||||
| 					<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="8" y="8" width="12" height="12" rx="2" /> | ||||
| 						<path d="M16 8v-2a2 2 0 0 0 -2 -2h-8a2 2 0 0 0 -2 2v8a2 2 0 0 0 2 2h2" /> | ||||
| 					</svg> | ||||
| 				</div> | ||||
| 			{/if} | ||||
| 		</div> | ||||
| 	</div> | ||||
| </div> | ||||
							
								
								
									
										38
									
								
								apps/client/src/lib/components/Explainer.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								apps/client/src/lib/components/Explainer.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,38 @@ | ||||
| <script lang="ts"> | ||||
| 	// import { onMount } from 'svelte'; | ||||
|  | ||||
| 	// import Tooltip from './Tooltip.svelte'; | ||||
| 	export let explanation = ''; | ||||
| 	export let position = 'dropdown-right'; | ||||
| 	// let id: any; | ||||
| 	// let self: any; | ||||
| 	// onMount(() => { | ||||
| 	// 	id = `info-${self.offsetLeft}-${self.offsetTop}`; | ||||
| 	// }); | ||||
| </script> | ||||
|  | ||||
| <div class={`dropdown dropdown-end ${position}`}> | ||||
| 	<!-- svelte-ignore a11y-label-has-associated-control --> | ||||
| 	<!-- svelte-ignore a11y-no-noninteractive-tabindex --> | ||||
| 	<label tabindex="0" class="btn btn-circle btn-ghost btn-xs text-sky-500"> | ||||
| 		<svg | ||||
| 			xmlns="http://www.w3.org/2000/svg" | ||||
| 			fill="none" | ||||
| 			viewBox="0 0 24 24" | ||||
| 			class="w-4 h-4 stroke-current" | ||||
| 			><path | ||||
| 				stroke-linecap="round" | ||||
| 				stroke-linejoin="round" | ||||
| 				stroke-width="2" | ||||
| 				d="M13 16h-1v-4h-1m1-4h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z" | ||||
| 			/></svg | ||||
| 		> | ||||
| 	</label> | ||||
| 	<!-- svelte-ignore a11y-no-noninteractive-tabindex --> | ||||
| 	<div tabindex="0" class="card compact dropdown-content shadow bg-coolgray-400 rounded w-64"> | ||||
| 		<div class="card-body"> | ||||
| 			<!-- <h2 class="card-title">You needed more info?</h2>  --> | ||||
| 			<p class="text-xs font-normal">{@html explanation}</p> | ||||
| 		</div> | ||||
| 	</div> | ||||
| </div> | ||||
							
								
								
									
										87
									
								
								apps/client/src/lib/components/Setting.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								apps/client/src/lib/components/Setting.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,87 @@ | ||||
| <script lang="ts"> | ||||
| 	import Beta from './Beta.svelte'; | ||||
| 	import Explaner from './Explainer.svelte'; | ||||
| 	import Tooltip from './Tooltip.svelte'; | ||||
|  | ||||
| 	export let id: any; | ||||
| 	export let customClass: any = null; | ||||
| 	export let setting: any; | ||||
| 	export let title: any; | ||||
| 	export let isBeta: any = false; | ||||
| 	export let description: any = null; | ||||
| 	export let isCenter = true; | ||||
| 	export let disabled = false; | ||||
| 	export let dataTooltip: any = null; | ||||
| 	export let loading = false; | ||||
| 	let triggeredBy = `#${id}`; | ||||
| </script> | ||||
|  | ||||
| <div class="flex items-center py-4 pr-8"> | ||||
| 	<div class="flex w-96 flex-col"> | ||||
| 		<!-- svelte-ignore a11y-label-has-associated-control --> | ||||
| 		<label> | ||||
| 			{title} | ||||
| 			{#if isBeta} | ||||
| 				<Beta /> | ||||
| 			{/if} | ||||
| 			{#if description && description !== ''} | ||||
| 				<Explaner explanation={description} /> | ||||
| 			{/if} | ||||
| 		</label> | ||||
| 	</div> | ||||
| </div> | ||||
| <div class:text-center={isCenter} class={`flex justify-center ${customClass}`}> | ||||
| 	<!-- svelte-ignore a11y-click-events-have-key-events --> | ||||
| 	<div | ||||
| 		on:click | ||||
| 		aria-pressed="false" | ||||
| 		class="relative mx-20 inline-flex h-6 w-11 flex-shrink-0 cursor-pointer rounded-full border-2 border-transparent transition-colors duration-200 ease-in-out" | ||||
| 		class:opacity-50={disabled || loading} | ||||
| 		class:bg-green-600={!loading && setting} | ||||
| 		class:bg-stone-700={!loading && !setting} | ||||
| 		class:bg-yellow-500={loading} | ||||
| 		{id} | ||||
| 	> | ||||
| 		<span class="sr-only">Use setting</span> | ||||
| 		<span | ||||
| 			class="pointer-events-none relative inline-block h-5 w-5 transform rounded-full bg-white shadow transition duration-200 ease-in-out" | ||||
| 			class:translate-x-5={setting} | ||||
| 			class:translate-x-0={!setting} | ||||
| 		> | ||||
| 			<span | ||||
| 				class=" absolute inset-0 flex h-full w-full items-center justify-center transition-opacity duration-200 ease-in" | ||||
| 				class:opacity-0={setting} | ||||
| 				class:opacity-100={!setting} | ||||
| 				class:animate-spin={loading} | ||||
| 				aria-hidden="true" | ||||
| 			> | ||||
| 				<svg class="h-3 w-3 bg-white text-red-600" fill="none" viewBox="0 0 12 12"> | ||||
| 					<path | ||||
| 						d="M4 8l2-2m0 0l2-2M6 6L4 4m2 2l2 2" | ||||
| 						stroke="currentColor" | ||||
| 						stroke-width="2" | ||||
| 						stroke-linecap="round" | ||||
| 						stroke-linejoin="round" | ||||
| 					/> | ||||
| 				</svg> | ||||
| 			</span> | ||||
| 			<span | ||||
| 				class="absolute inset-0 flex h-full w-full items-center justify-center transition-opacity duration-100 ease-out" | ||||
| 				aria-hidden="true" | ||||
| 				class:opacity-100={setting} | ||||
| 				class:opacity-0={!setting} | ||||
| 				class:animate-spin={loading} | ||||
| 			> | ||||
| 				<svg class="h-3 w-3 bg-white text-green-600" fill="currentColor" viewBox="0 0 12 12"> | ||||
| 					<path | ||||
| 						d="M3.707 5.293a1 1 0 00-1.414 1.414l1.414-1.414zM5 8l-.707.707a1 1 0 001.414 0L5 8zm4.707-3.293a1 1 0 00-1.414-1.414l1.414 1.414zm-7.414 2l2 2 1.414-1.414-2-2-1.414 1.414zm3.414 2l4-4-1.414-1.414-4 4 1.414 1.414z" | ||||
| 					/> | ||||
| 				</svg> | ||||
| 			</span> | ||||
| 		</span> | ||||
| 	</div> | ||||
| </div> | ||||
|  | ||||
| {#if dataTooltip} | ||||
| 	<Tooltip {triggeredBy} placement="top">{dataTooltip}</Tooltip> | ||||
| {/if} | ||||
| @@ -21,7 +21,8 @@ export const trpc = createTRPCProxyClient<AppRouter>({ | ||||
| 		}) | ||||
| 	] | ||||
| }); | ||||
|  | ||||
| export const disabledButton: Writable<boolean> = writable(false); | ||||
| export const location: Writable<null | string> = writable(null) | ||||
| interface AppSession { | ||||
| 	isRegistrationEnabled: boolean; | ||||
| 	token?: string; | ||||
| @@ -139,3 +140,33 @@ export const status: Writable<any> = writable({ | ||||
| 		isPublic: false | ||||
| 	} | ||||
| }); | ||||
|  | ||||
| export function checkIfDeploymentEnabledApplications(isAdmin: boolean, application: any) { | ||||
| 	return !!( | ||||
| 		(isAdmin && application.buildPack === 'compose') || | ||||
| 		((application.fqdn || application.settings.isBot) && | ||||
| 			((application.gitSource && application.repository && application.buildPack) || | ||||
| 				application.simpleDockerfile) && | ||||
| 			application.destinationDocker) | ||||
| 	); | ||||
| } | ||||
| export const setLocation = (resource: any, settings?: any) => { | ||||
| 	if (resource.settings.isBot && resource.exposePort) { | ||||
| 		disabledButton.set(false); | ||||
| 		return location.set(`http://${dev ? 'localhost' : settings.ipv4}:${resource.exposePort}`); | ||||
| 	} | ||||
| 	if (GITPOD_WORKSPACE_URL && resource.exposePort) { | ||||
| 		const { href } = new URL(GITPOD_WORKSPACE_URL); | ||||
| 		const newURL = href.replace('https://', `https://${resource.exposePort}-`).replace(/\/$/, ''); | ||||
| 		return location.set(newURL); | ||||
| 	} else if (CODESANDBOX_HOST) { | ||||
| 		const newURL = `https://${CODESANDBOX_HOST.replace(/\$PORT/, resource.exposePort)}`; | ||||
| 		return location.set(newURL); | ||||
| 	} | ||||
| 	if (resource.fqdn) { | ||||
| 		return location.set(resource.fqdn); | ||||
| 	} else { | ||||
| 		location.set(null); | ||||
| 		disabledButton.set(false); | ||||
| 	} | ||||
| }; | ||||
|   | ||||
| @@ -1,11 +1,8 @@ | ||||
| import { error } from '@sveltejs/kit'; | ||||
| import { trpc } from '$lib/store'; | ||||
| import type { LayoutLoad } from './$types'; | ||||
| import { redirect } from '@sveltejs/kit'; | ||||
| import Cookies from 'js-cookie'; | ||||
| export const ssr = false; | ||||
|  | ||||
| export const load: LayoutLoad = async ({ url }) => { | ||||
| export const load = async () => { | ||||
| 	try { | ||||
| 		return await trpc.dashboard.resources.query(); | ||||
| 	} catch (err) { | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										118
									
								
								apps/client/src/routes/applications/[id]/features/+page.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										118
									
								
								apps/client/src/routes/applications/[id]/features/+page.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,118 @@ | ||||
| <script lang="ts"> | ||||
| 	import type { PageParentData } from './$types'; | ||||
|  | ||||
| 	export let data: PageParentData; | ||||
| 	const application = data.application.data; | ||||
| 	const settings = data.settings.data; | ||||
| 	import { page } from '$app/stores'; | ||||
| 	const { id } = $page.params; | ||||
| 	import { | ||||
| 		addToast, | ||||
| 		appSession, | ||||
| 		checkIfDeploymentEnabledApplications, | ||||
| 		setLocation, | ||||
| 		status, | ||||
| 		isDeploymentEnabled, | ||||
| 		trpc | ||||
| 	} from '$lib/store'; | ||||
| 	import { errorNotification } from '$lib/common'; | ||||
| 	import Setting from '$lib/components/Setting.svelte'; | ||||
|  | ||||
| 	let previews = application.settings.previews; | ||||
| 	let dualCerts = application.settings.dualCerts; | ||||
| 	let autodeploy = application.settings.autodeploy; | ||||
| 	let isBot = application.settings.isBot; | ||||
| 	let isDBBranching = application.settings.isDBBranching; | ||||
|  | ||||
| 	async function changeSettings(name: any) { | ||||
| 		if (name === 'previews') { | ||||
| 			previews = !previews; | ||||
| 		} | ||||
| 		if (name === 'dualCerts') { | ||||
| 			dualCerts = !dualCerts; | ||||
| 		} | ||||
| 		if (name === 'autodeploy') { | ||||
| 			autodeploy = !autodeploy; | ||||
| 		} | ||||
| 		if (name === 'isBot') { | ||||
| 			if ($status.application.isRunning) return; | ||||
| 			isBot = !isBot; | ||||
| 			application.settings.isBot = isBot; | ||||
| 			application.fqdn = null; | ||||
| 			setLocation(application, settings); | ||||
| 		} | ||||
| 		if (name === 'isDBBranching') { | ||||
| 			isDBBranching = !isDBBranching; | ||||
| 		} | ||||
| 		try { | ||||
| 			await trpc.applications.saveSettings.mutate({ | ||||
| 				id, | ||||
| 				previews, | ||||
| 				dualCerts, | ||||
| 				isBot, | ||||
| 				autodeploy, | ||||
| 				isDBBranching | ||||
| 			}); | ||||
|  | ||||
| 			return addToast({ | ||||
| 				message: 'Settings saved', | ||||
| 				type: 'success' | ||||
| 			}); | ||||
| 		} catch (error) { | ||||
| 			if (name === 'previews') { | ||||
| 				previews = !previews; | ||||
| 			} | ||||
| 			if (name === 'dualCerts') { | ||||
| 				dualCerts = !dualCerts; | ||||
| 			} | ||||
| 			if (name === 'autodeploy') { | ||||
| 				autodeploy = !autodeploy; | ||||
| 			} | ||||
| 			if (name === 'isBot') { | ||||
| 				isBot = !isBot; | ||||
| 			} | ||||
| 			if (name === 'isDBBranching') { | ||||
| 				isDBBranching = !isDBBranching; | ||||
| 			} | ||||
| 			return errorNotification(error); | ||||
| 		} finally { | ||||
| 			$isDeploymentEnabled = checkIfDeploymentEnabledApplications($appSession.isAdmin, application); | ||||
| 		} | ||||
| 	} | ||||
| </script> | ||||
|  | ||||
| <div class="w-full"> | ||||
| 	<div class="mx-auto w-full"> | ||||
| 		<div class="flex flex-row border-b border-coolgray-500 mb-6  space-x-2"> | ||||
| 			<div class="title font-bold pb-3">Features</div> | ||||
| 		</div> | ||||
| 		<div class="px-4 lg:pb-10 pb-6"> | ||||
| 			{#if !application.settings.isPublicRepository} | ||||
| 				<div class="grid grid-cols-2 items-center"> | ||||
| 					<Setting | ||||
| 						id="autodeploy" | ||||
| 						isCenter={false} | ||||
| 						bind:setting={autodeploy} | ||||
| 						on:click={() => changeSettings('autodeploy')} | ||||
| 						title="Enable Automatic Deployment" | ||||
| 						description="Enable automatic deployment through webhooks." | ||||
| 					/> | ||||
| 				</div> | ||||
| 				{#if !application.settings.isBot && !application.simpleDockerfile} | ||||
| 					<div class="grid grid-cols-2 items-center"> | ||||
| 						<Setting | ||||
| 							id="previews" | ||||
| 							isCenter={false} | ||||
| 							bind:setting={previews} | ||||
| 							on:click={() => changeSettings('previews')} | ||||
| 							title="Enable MR/PR Previews" | ||||
| 							description="Enable preview deployments from pull or merge requests." | ||||
| 						/> | ||||
| 					</div> | ||||
| 				{/if} | ||||
| 			{:else} | ||||
| 				No features available for this application | ||||
| 			{/if} | ||||
| 		</div> | ||||
| 	</div> | ||||
| </div> | ||||
							
								
								
									
										138
									
								
								apps/client/src/routes/applications/[id]/secrets/+page.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										138
									
								
								apps/client/src/routes/applications/[id]/secrets/+page.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,138 @@ | ||||
| <script lang="ts"> | ||||
| 	import type { PageData } from './$types'; | ||||
|  | ||||
| 	export let data: PageData; | ||||
| 	let secrets = data.secrets; | ||||
| 	let previewSecrets = data.previewSecrets; | ||||
| 	const application = data.application.data; | ||||
|  | ||||
| 	import pLimit from 'p-limit'; | ||||
| 	import { page } from '$app/stores'; | ||||
| 	import { addToast, trpc } from '$lib/store'; | ||||
| 	import Secret from './_components/Secret.svelte'; | ||||
| 	import PreviewSecret from './_components/PreviewSecret.svelte'; | ||||
| 	import { errorNotification } from '$lib/common'; | ||||
| 	import Explainer from '$lib/components/Explainer.svelte'; | ||||
|  | ||||
| 	const limit = pLimit(1); | ||||
| 	const { id } = $page.params; | ||||
|  | ||||
| 	let batchSecrets = ''; | ||||
| 	async function refreshSecrets() { | ||||
| 		const { data } = await trpc.applications.getSecrets.query({ id }); | ||||
| 		previewSecrets = [...data.previewSecrets]; | ||||
| 		secrets = [...data.secrets]; | ||||
| 	} | ||||
| 	async function getValues() { | ||||
| 		if (!batchSecrets) return; | ||||
| 		const eachValuePair = batchSecrets.split('\n'); | ||||
| 		const batchSecretsPairs = eachValuePair | ||||
| 			.filter((secret) => !secret.startsWith('#') && secret) | ||||
| 			.map((secret) => { | ||||
| 				const [name, ...rest] = secret.split('='); | ||||
| 				const value = rest.join('='); | ||||
| 				return { | ||||
| 					name: name.trim(), | ||||
| 					value: value.trim(), | ||||
| 					createSecret: !secrets.find((secret: any) => name === secret.name) | ||||
| 				}; | ||||
| 			}); | ||||
|  | ||||
| 		await Promise.all( | ||||
| 			batchSecretsPairs.map(({ name, value, createSecret }) => | ||||
| 				limit(async () => { | ||||
| 					try { | ||||
| 						if (!name || !value) return; | ||||
| 						if (createSecret) { | ||||
| 							await trpc.applications.newSecret.mutate({ | ||||
| 								id, | ||||
| 								name, | ||||
| 								value | ||||
| 							}); | ||||
|  | ||||
| 							addToast({ | ||||
| 								message: 'Secret created.', | ||||
| 								type: 'success' | ||||
| 							}); | ||||
| 						} else { | ||||
| 							await trpc.applications.updateSecret.mutate({ | ||||
| 								id, | ||||
| 								name, | ||||
| 								value | ||||
| 							}); | ||||
|  | ||||
| 							addToast({ | ||||
| 								message: 'Secret updated.', | ||||
| 								type: 'success' | ||||
| 							}); | ||||
| 						} | ||||
| 					} catch (error) { | ||||
| 						return errorNotification(error); | ||||
| 					} | ||||
| 				}) | ||||
| 			) | ||||
| 		); | ||||
| 		batchSecrets = ''; | ||||
| 		await refreshSecrets(); | ||||
| 	} | ||||
| </script> | ||||
|  | ||||
| <div class="mx-auto w-full"> | ||||
| 	<div class="flex flex-row border-b border-coolgray-500 mb-6 space-x-2"> | ||||
| 		<div class="title font-bold pb-3">Secrets</div> | ||||
| 	</div> | ||||
| 	{#each secrets as secret, index} | ||||
| 		{#key secret.id} | ||||
| 			<Secret | ||||
| 				{index} | ||||
| 				length={secrets.length} | ||||
| 				name={secret.name} | ||||
| 				value={secret.value} | ||||
| 				isBuildSecret={secret.isBuildSecret} | ||||
| 				on:refresh={refreshSecrets} | ||||
| 			/> | ||||
| 		{/key} | ||||
| 	{/each} | ||||
| 	<div class="lg:pt-0 pt-10"> | ||||
| 		<Secret on:refresh={refreshSecrets} length={secrets.length} isNewSecret /> | ||||
| 	</div> | ||||
| 	{#if !application.settings.isBot && !application.simpleDockerfile} | ||||
| 		<div class="flex flex-row border-b border-coolgray-500 mb-6 space-x-2"> | ||||
| 			<div class="title font-bold pb-3 pt-8"> | ||||
| 				Preview Secrets <Explainer | ||||
| 					explanation="These values overwrite application secrets in PR/MR deployments. <br>Useful for creating <span class='text-green-500 font-bold'>staging</span> environments." | ||||
| 				/> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 		{#if previewSecrets.length !== 0} | ||||
| 			{#each previewSecrets as secret, index} | ||||
| 				{#key index} | ||||
| 					<PreviewSecret | ||||
| 						{index} | ||||
| 						length={secrets.length} | ||||
| 						name={secret.name} | ||||
| 						value={secret.value} | ||||
| 						isBuildSecret={secret.isBuildSecret} | ||||
| 						on:refresh={refreshSecrets} | ||||
| 					/> | ||||
| 				{/key} | ||||
| 			{/each} | ||||
| 		{:else} | ||||
| 			Add secrets first to see Preview Secrets. | ||||
| 		{/if} | ||||
| 	{/if} | ||||
| </div> | ||||
| <form on:submit|preventDefault={getValues} class="mb-12 w-full"> | ||||
| 	<div class="flex flex-row border-b border-coolgray-500 mb-6 space-x-2 pt-10"> | ||||
| 		<div class="flex flex-row space-x-2"> | ||||
| 			<div class="title font-bold pb-3 ">Paste <code>.env</code> file</div> | ||||
| 			<button type="submit" class="btn btn-sm bg-primary">Add Secrets in Batch</button> | ||||
| 		</div> | ||||
| 	</div> | ||||
|  | ||||
| 	<textarea | ||||
| 		placeholder={`PORT=1337\nPASSWORD=supersecret`} | ||||
| 		bind:value={batchSecrets} | ||||
| 		class="mb-2 min-h-[200px] w-full" | ||||
| 	/> | ||||
| </form> | ||||
							
								
								
									
										16
									
								
								apps/client/src/routes/applications/[id]/secrets/+page.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								apps/client/src/routes/applications/[id]/secrets/+page.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| import { error } from '@sveltejs/kit'; | ||||
| import { trpc } from '$lib/store'; | ||||
| import type { PageLoad } from './$types'; | ||||
| export const ssr = false; | ||||
|  | ||||
| export const load: PageLoad = async ({ params }) => { | ||||
| 	try { | ||||
| 		const { id } = params; | ||||
| 		const { data } = await trpc.applications.getSecrets.query({ id }); | ||||
| 		return data; | ||||
| 	} catch (err) { | ||||
| 		throw error(500, { | ||||
| 			message: 'An unexpected error occurred, please try again later.' | ||||
| 		}); | ||||
| 	} | ||||
| }; | ||||
| @@ -0,0 +1,131 @@ | ||||
| <script lang="ts"> | ||||
| 	export let length = 0; | ||||
| 	export let index: number = 0; | ||||
| 	export let name = ''; | ||||
| 	export let value = ''; | ||||
| 	export let isBuildSecret = false; | ||||
|  | ||||
| 	import { page } from '$app/stores'; | ||||
| 	import { errorNotification } from '$lib/common'; | ||||
| 	import CopyPasswordField from '$lib/components/CopyPasswordField.svelte'; | ||||
| 	import { addToast, trpc } from '$lib/store'; | ||||
| 	import { createEventDispatcher } from 'svelte'; | ||||
|  | ||||
| 	const dispatch = createEventDispatcher(); | ||||
| 	const { id } = $page.params; | ||||
|  | ||||
| 	async function updatePreviewSecret() { | ||||
| 		try { | ||||
| 			await trpc.applications.updateSecret.mutate({ | ||||
| 				id, | ||||
| 				name: name.trim(), | ||||
| 				value: value.trim(), | ||||
| 				isPreview: true | ||||
| 			}); | ||||
| 			addToast({ | ||||
| 				message: 'Secret updated.', | ||||
| 				type: 'success' | ||||
| 			}); | ||||
| 		} catch (error) { | ||||
| 			return errorNotification(error); | ||||
| 		} | ||||
| 	} | ||||
| </script> | ||||
|  | ||||
| <div class="w-full grid grid-cols-1 lg:grid-cols-4 gap-2 pb-2"> | ||||
| 	<div class="flex flex-col"> | ||||
| 		{#if index === 0 || length === 0} | ||||
| 			<label for="name" class="pb-2 uppercase font-bold">name</label> | ||||
| 		{/if} | ||||
|  | ||||
| 		<input | ||||
| 			id="secretName" | ||||
| 			readonly | ||||
| 			disabled | ||||
| 			value={name} | ||||
| 			required | ||||
| 			placeholder="EXAMPLE_VARIABLE" | ||||
| 			class=" w-full" | ||||
| 		/> | ||||
| 	</div> | ||||
| 	<div class="flex flex-col"> | ||||
| 		{#if index === 0 || length === 0} | ||||
| 			<label for="value" class="pb-2 uppercase font-bold">value</label> | ||||
| 		{/if} | ||||
|  | ||||
| 		<CopyPasswordField | ||||
| 			id="secretValue" | ||||
| 			name="secretValue" | ||||
| 			isPasswordField={true} | ||||
| 			bind:value | ||||
| 			placeholder="J$#@UIO%HO#$U%H" | ||||
| 		/> | ||||
| 	</div> | ||||
| 	<div class="flex lg:flex-col flex-row justify-start items-center pt-3 lg:pt-0"> | ||||
| 		{#if index === 0 || length === 0} | ||||
| 			<label for="name" class="pb-2 uppercase lg:block hidden font-bold" | ||||
| 				>Need during buildtime?</label | ||||
| 			> | ||||
| 		{/if} | ||||
| 		<label for="name" class="pb-2 uppercase lg:hidden block font-bold">Need during buildtime?</label | ||||
| 		> | ||||
|  | ||||
| 		<div class="flex justify-center h-full items-center pt-0 lg:pt-0 pl-4 lg:pl-0"> | ||||
| 			<button | ||||
| 				aria-pressed="false" | ||||
| 				class="opacity-50 cursor-pointer cursor-not-allowedrelative inline-flex h-6 w-11 flex-shrink-0 rounded-full border-2 border-transparent transition-colors duration-200 ease-in-out " | ||||
| 				class:bg-green-600={isBuildSecret} | ||||
| 				class:bg-stone-700={!isBuildSecret} | ||||
| 			> | ||||
| 				<span class="sr-only">Is build secret?</span> | ||||
| 				<span | ||||
| 					class="pointer-events-none relative inline-block h-5 w-5 transform rounded-full bg-white shadow transition duration-200 ease-in-out" | ||||
| 					class:translate-x-5={isBuildSecret} | ||||
| 					class:translate-x-0={!isBuildSecret} | ||||
| 				> | ||||
| 					<span | ||||
| 						class="absolute inset-0 flex h-full w-full items-center justify-center transition-opacity duration-200 ease-in" | ||||
| 						class:opacity-0={isBuildSecret} | ||||
| 						class:opacity-100={!isBuildSecret} | ||||
| 						aria-hidden="true" | ||||
| 					> | ||||
| 						<svg class="h-3 w-3 bg-white text-red-600" fill="none" viewBox="0 0 12 12"> | ||||
| 							<path | ||||
| 								d="M4 8l2-2m0 0l2-2M6 6L4 4m2 2l2 2" | ||||
| 								stroke="currentColor" | ||||
| 								stroke-width="2" | ||||
| 								stroke-linecap="round" | ||||
| 								stroke-linejoin="round" | ||||
| 							/> | ||||
| 						</svg> | ||||
| 					</span> | ||||
| 					<span | ||||
| 						class="absolute inset-0 flex h-full w-full items-center justify-center transition-opacity duration-100 ease-out" | ||||
| 						aria-hidden="true" | ||||
| 						class:opacity-100={isBuildSecret} | ||||
| 						class:opacity-0={!isBuildSecret} | ||||
| 					> | ||||
| 						<svg class="h-3 w-3 bg-white text-green-600" fill="currentColor" viewBox="0 0 12 12"> | ||||
| 							<path | ||||
| 								d="M3.707 5.293a1 1 0 00-1.414 1.414l1.414-1.414zM5 8l-.707.707a1 1 0 001.414 0L5 8zm4.707-3.293a1 1 0 00-1.414-1.414l1.414 1.414zm-7.414 2l2 2 1.414-1.414-2-2-1.414 1.414zm3.414 2l4-4-1.414-1.414-4 4 1.414 1.414z" | ||||
| 							/> | ||||
| 						</svg> | ||||
| 					</span> | ||||
| 				</span> | ||||
| 			</button> | ||||
| 		</div> | ||||
| 	</div> | ||||
| 	<div class="flex flex-row lg:flex-col lg:items-center items-start"> | ||||
| 		{#if index === 0 || length === 0} | ||||
| 			<label for="name" class="pb-5 uppercase lg:block hidden font-bold" /> | ||||
| 		{/if} | ||||
|  | ||||
| 		<div class="flex justify-center h-full items-center pt-3"> | ||||
| 			<div class="flex flex-row justify-center space-x-2"> | ||||
| 				<div class="flex items-center justify-center"> | ||||
| 					<button class="btn btn-sm btn-primary" on:click={updatePreviewSecret}>Update</button> | ||||
| 				</div> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 	</div> | ||||
| </div> | ||||
| @@ -0,0 +1,193 @@ | ||||
| <script lang="ts"> | ||||
| 	export let length = 0; | ||||
| 	export let index: number = 0; | ||||
| 	export let name = ''; | ||||
| 	export let value = ''; | ||||
| 	export let isBuildSecret = false; | ||||
| 	export let isNewSecret = false; | ||||
|  | ||||
| 	import { page } from '$app/stores'; | ||||
| 	import { errorNotification } from '$lib/common'; | ||||
| 	import CopyPasswordField from '$lib/components/CopyPasswordField.svelte'; | ||||
| 	import { addToast, trpc } from '$lib/store'; | ||||
| 	import { createEventDispatcher } from 'svelte'; | ||||
|  | ||||
| 	const dispatch = createEventDispatcher(); | ||||
| 	const { id } = $page.params; | ||||
| 	function cleanupState() { | ||||
| 		if (isNewSecret) { | ||||
| 			name = ''; | ||||
| 			value = ''; | ||||
| 			isBuildSecret = false; | ||||
| 		} | ||||
| 	} | ||||
| 	async function removeSecret() { | ||||
| 		try { | ||||
| 			await trpc.applications.deleteSecret.mutate({ id, name }); | ||||
| 			cleanupState(); | ||||
| 			addToast({ | ||||
| 				message: 'Secret removed.', | ||||
| 				type: 'success' | ||||
| 			}); | ||||
| 			dispatch('refresh'); | ||||
| 		} catch (error) { | ||||
| 			return errorNotification(error); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	async function addNewSecret() { | ||||
| 		try { | ||||
| 			if (!name.trim()) return errorNotification({ message: 'Name is required.' }); | ||||
| 			if (!value.trim()) return errorNotification({ message: 'Value is required.' }); | ||||
| 			await trpc.applications.newSecret.mutate({ | ||||
| 				id, | ||||
| 				name: name.trim(), | ||||
| 				value: value.trim(), | ||||
| 				isBuildSecret | ||||
| 			}); | ||||
| 			cleanupState(); | ||||
| 			addToast({ | ||||
| 				message: 'Secret added.', | ||||
| 				type: 'success' | ||||
| 			}); | ||||
| 			dispatch('refresh'); | ||||
| 		} catch (error) { | ||||
| 			return errorNotification(error); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	async function updateSecret({ | ||||
| 		changeIsBuildSecret = false | ||||
| 	}: { changeIsBuildSecret?: boolean } = {}) { | ||||
| 		if (changeIsBuildSecret) isBuildSecret = !isBuildSecret; | ||||
| 		if (isNewSecret) return; | ||||
| 		try { | ||||
| 			await trpc.applications.updateSecret.mutate({ | ||||
| 				id, | ||||
| 				name: name.trim(), | ||||
| 				value: value.trim(), | ||||
| 				isBuildSecret, | ||||
| 				isPreview: false | ||||
| 			}); | ||||
| 			addToast({ | ||||
| 				message: 'Secret updated.', | ||||
| 				type: 'success' | ||||
| 			}); | ||||
| 			dispatch('refresh'); | ||||
| 		} catch (error) { | ||||
| 			return errorNotification(error); | ||||
| 		} | ||||
| 	} | ||||
| </script> | ||||
|  | ||||
| <div class="w-full grid grid-cols-1 lg:grid-cols-4 gap-2 pb-2"> | ||||
| 	<div class="flex flex-col"> | ||||
| 		{#if (index === 0 && !isNewSecret) || length === 0} | ||||
| 			<label for="name" class="pb-2 uppercase font-bold">name</label> | ||||
| 		{/if} | ||||
|  | ||||
| 		<input | ||||
| 			id={isNewSecret ? 'secretName' : 'secretNameNew'} | ||||
| 			bind:value={name} | ||||
| 			required | ||||
| 			placeholder="EXAMPLE_VARIABLE" | ||||
| 			readonly={!isNewSecret} | ||||
| 			class="w-full" | ||||
| 			class:bg-coolblack={!isNewSecret} | ||||
| 			class:border={!isNewSecret} | ||||
| 			class:border-dashed={!isNewSecret} | ||||
| 			class:border-coolgray-300={!isNewSecret} | ||||
| 			class:cursor-not-allowed={!isNewSecret} | ||||
| 		/> | ||||
| 	</div> | ||||
| 	<div class="flex flex-col"> | ||||
| 		{#if (index === 0 && !isNewSecret) || length === 0} | ||||
| 			<label for="value" class="pb-2 uppercase font-bold">value</label> | ||||
| 		{/if} | ||||
|  | ||||
| 		<CopyPasswordField | ||||
| 			id={isNewSecret ? 'secretValue' : 'secretValueNew'} | ||||
| 			name={isNewSecret ? 'secretValue' : 'secretValueNew'} | ||||
| 			isPasswordField={true} | ||||
| 			bind:value | ||||
| 			placeholder="J$#@UIO%HO#$U%H" | ||||
| 		/> | ||||
| 	</div> | ||||
| 	<div class="flex lg:flex-col flex-row justify-start items-center pt-3 lg:pt-0"> | ||||
| 		{#if (index === 0 && !isNewSecret) || length === 0} | ||||
| 			<label for="name" class="pb-2 uppercase lg:block hidden font-bold" | ||||
| 				>Need during buildtime?</label | ||||
| 			> | ||||
| 		{/if} | ||||
| 		<label for="name" class="pb-2 uppercase lg:hidden block font-bold">Need during buildtime?</label | ||||
| 		> | ||||
|  | ||||
| 		<div class="flex justify-center h-full items-center pt-0 lg:pt-0 pl-4 lg:pl-0"> | ||||
| 			<button | ||||
| 				on:click={() => updateSecret({ changeIsBuildSecret: true })} | ||||
| 				aria-pressed="false" | ||||
| 				class="relative inline-flex h-6 w-11 flex-shrink-0 rounded-full border-2 border-transparent transition-colors duration-200 ease-in-out " | ||||
| 				class:bg-green-600={isBuildSecret} | ||||
| 				class:bg-stone-700={!isBuildSecret} | ||||
| 			> | ||||
| 				<span class="sr-only">Is build secret?</span> | ||||
| 				<span | ||||
| 					class="pointer-events-none relative inline-block h-5 w-5 transform rounded-full bg-white shadow transition duration-200 ease-in-out" | ||||
| 					class:translate-x-5={isBuildSecret} | ||||
| 					class:translate-x-0={!isBuildSecret} | ||||
| 				> | ||||
| 					<span | ||||
| 						class="absolute inset-0 flex h-full w-full items-center justify-center transition-opacity duration-200 ease-in" | ||||
| 						class:opacity-0={isBuildSecret} | ||||
| 						class:opacity-100={!isBuildSecret} | ||||
| 						aria-hidden="true" | ||||
| 					> | ||||
| 						<svg class="h-3 w-3 bg-white text-red-600" fill="none" viewBox="0 0 12 12"> | ||||
| 							<path | ||||
| 								d="M4 8l2-2m0 0l2-2M6 6L4 4m2 2l2 2" | ||||
| 								stroke="currentColor" | ||||
| 								stroke-width="2" | ||||
| 								stroke-linecap="round" | ||||
| 								stroke-linejoin="round" | ||||
| 							/> | ||||
| 						</svg> | ||||
| 					</span> | ||||
| 					<span | ||||
| 						class="absolute inset-0 flex h-full w-full items-center justify-center transition-opacity duration-100 ease-out" | ||||
| 						aria-hidden="true" | ||||
| 						class:opacity-100={isBuildSecret} | ||||
| 						class:opacity-0={!isBuildSecret} | ||||
| 					> | ||||
| 						<svg class="h-3 w-3 bg-white text-green-600" fill="currentColor" viewBox="0 0 12 12"> | ||||
| 							<path | ||||
| 								d="M3.707 5.293a1 1 0 00-1.414 1.414l1.414-1.414zM5 8l-.707.707a1 1 0 001.414 0L5 8zm4.707-3.293a1 1 0 00-1.414-1.414l1.414 1.414zm-7.414 2l2 2 1.414-1.414-2-2-1.414 1.414zm3.414 2l4-4-1.414-1.414-4 4 1.414 1.414z" | ||||
| 							/> | ||||
| 						</svg> | ||||
| 					</span> | ||||
| 				</span> | ||||
| 			</button> | ||||
| 		</div> | ||||
| 	</div> | ||||
| 	<div class="flex flex-row lg:flex-col lg:items-center items-start"> | ||||
| 		{#if (index === 0 && !isNewSecret) || length === 0} | ||||
| 			<label for="name" class="pb-5 uppercase lg:block hidden font-bold" /> | ||||
| 		{/if} | ||||
|  | ||||
| 		<div class="flex justify-center h-full items-center pt-3"> | ||||
| 			{#if isNewSecret} | ||||
| 				<div class="flex items-center justify-center"> | ||||
| 					<button class="btn btn-sm btn-primary" on:click={addNewSecret}>Add</button> | ||||
| 				</div> | ||||
| 			{:else} | ||||
| 				<div class="flex flex-row justify-center space-x-2"> | ||||
| 					<div class="flex items-center justify-center"> | ||||
| 						<button class="btn btn-sm btn-primary" on:click={() => updateSecret()}>Set</button> | ||||
| 					</div> | ||||
| 					<div class="flex justify-center items-end"> | ||||
| 						<button class="btn btn-sm btn-error" on:click={removeSecret}>Remove</button> | ||||
| 					</div> | ||||
| 				</div> | ||||
| 			{/if} | ||||
| 		</div> | ||||
| 	</div> | ||||
| </div> | ||||
| @@ -0,0 +1,78 @@ | ||||
| <script lang="ts"> | ||||
| 	import type { PageData } from './$types'; | ||||
|  | ||||
| 	export let data: PageData; | ||||
| 	const application = data.application.data; | ||||
| 	let persistentStorages = data.persistentStorages; | ||||
| 	import { page } from '$app/stores'; | ||||
| 	import Storage from './components/Storage.svelte'; | ||||
| 	import Explainer from '$lib/components/Explainer.svelte'; | ||||
| 	import { trpc } from '$lib/store'; | ||||
|  | ||||
| 	let composeJson: any = JSON.parse(application?.dockerComposeFile || '{}'); | ||||
| 	let predefinedVolumes: any[] = []; | ||||
| 	if (composeJson?.services) { | ||||
| 		for (const [_, service] of Object.entries(composeJson.services)) { | ||||
| 			if (service?.volumes) { | ||||
| 				for (const [_, volumeName] of Object.entries(service.volumes)) { | ||||
| 					let [volume, target] = volumeName.split(':'); | ||||
| 					if (volume === '.') { | ||||
| 						volume = target; | ||||
| 					} | ||||
| 					if (!target) { | ||||
| 						target = volume; | ||||
| 						volume = `${application.id}${volume.replace(/\//gi, '-').replace(/\./gi, '')}`; | ||||
| 					} else { | ||||
| 						volume = `${application.id}${volume.replace(/\//gi, '-').replace(/\./gi, '')}`; | ||||
| 					} | ||||
| 					predefinedVolumes.push({ id: volume, path: target, predefined: true }); | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	const { id } = $page.params; | ||||
| 	async function refreshStorage() { | ||||
| 		const { data } = await trpc.applications.getStorages.query({ id }); | ||||
| 		persistentStorages = [...data.persistentStorages]; | ||||
| 	} | ||||
| </script> | ||||
|  | ||||
| <div class="w-full"> | ||||
| 	<div class="mx-auto w-full"> | ||||
| 		<div class="flex flex-row border-b border-coolgray-500 mb-6 space-x-2"> | ||||
| 			<div class="title font-bold pb-3">Persistent Volumes</div> | ||||
| 		</div> | ||||
| 		{#if predefinedVolumes.length > 0} | ||||
| 			<div class="title">Predefined Volumes</div> | ||||
| 			<div class="w-full lg:px-0 px-4"> | ||||
| 				<div class="grid grid-col-1 lg:grid-cols-2 py-2 gap-2"> | ||||
| 					<div class="font-bold uppercase">Volume Id</div> | ||||
| 					<div class="font-bold uppercase">Mount Dir</div> | ||||
| 				</div> | ||||
| 			</div> | ||||
|  | ||||
| 			<div class="gap-4"> | ||||
| 				{#each predefinedVolumes as storage} | ||||
| 					{#key storage.id} | ||||
| 						<Storage on:refresh={refreshStorage} {storage} /> | ||||
| 					{/key} | ||||
| 				{/each} | ||||
| 			</div> | ||||
| 		{/if} | ||||
| 		{#if persistentStorages.length > 0} | ||||
| 			<div class="title" class:pt-10={predefinedVolumes.length > 0}>Custom Volumes</div> | ||||
| 		{/if} | ||||
| 		{#each persistentStorages as storage} | ||||
| 			{#key storage.id} | ||||
| 				<Storage on:refresh={refreshStorage} {storage} /> | ||||
| 			{/key} | ||||
| 		{/each} | ||||
| 		<div class="Preview Secrets" class:pt-10={predefinedVolumes.length > 0}> | ||||
| 			Add New Volume <Explainer | ||||
| 				position="dropdown-bottom" | ||||
| 				explanation="You can specify any folder that you want to be persistent across deployments.<br><br><span class='text-settings '>/example</span> means it will preserve <span class='text-settings '>/example</span> between deployments.<br><br>Your application's data is copied to <span class='text-settings '>/app</span> inside the container, you can preserve data under it as well, like <span class='text-settings '>/app/db</span>.<br><br>This is useful for storing data such as a <span class='text-settings '>database (SQLite)</span> or a <span class='text-settings '>cache</span>." | ||||
| 			/> | ||||
| 		</div> | ||||
| 		<Storage on:refresh={refreshStorage} isNew /> | ||||
| 	</div> | ||||
| </div> | ||||
							
								
								
									
										16
									
								
								apps/client/src/routes/applications/[id]/storages/+page.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								apps/client/src/routes/applications/[id]/storages/+page.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| import { error } from '@sveltejs/kit'; | ||||
| import { trpc } from '$lib/store'; | ||||
| import type { PageLoad } from './$types'; | ||||
| export const ssr = false; | ||||
|  | ||||
| export const load: PageLoad = async ({ params }) => { | ||||
| 	try { | ||||
| 		const { id } = params; | ||||
| 		const { data } = await trpc.applications.getStorages.query({ id }); | ||||
| 		return data; | ||||
| 	} catch (err) { | ||||
| 		throw error(500, { | ||||
| 			message: 'An unexpected error occurred, please try again later.' | ||||
| 		}); | ||||
| 	} | ||||
| }; | ||||
| @@ -0,0 +1,114 @@ | ||||
| <script lang="ts"> | ||||
| 	export let isNew = false; | ||||
| 	export let storage: any = { | ||||
| 		id: null, | ||||
| 		path: null | ||||
| 	}; | ||||
| 	import { page } from '$app/stores'; | ||||
| 	import { createEventDispatcher } from 'svelte'; | ||||
|  | ||||
| 	import { errorNotification } from '$lib/common'; | ||||
| 	import { addToast, trpc } from '$lib/store'; | ||||
| 	const { id } = $page.params; | ||||
|  | ||||
| 	const dispatch = createEventDispatcher(); | ||||
| 	async function saveStorage(newStorage = false) { | ||||
| 		try { | ||||
| 			if (!storage.path) return errorNotification('Path is required'); | ||||
| 			storage.path = storage.path.startsWith('/') ? storage.path : `/${storage.path}`; | ||||
| 			storage.path = storage.path.endsWith('/') ? storage.path.slice(0, -1) : storage.path; | ||||
| 			storage.path.replace(/\/\//g, '/'); | ||||
| 			await trpc.applications.updateStorage.mutate({ | ||||
| 				id, | ||||
| 				path: storage.path, | ||||
| 				storageId: storage.id, | ||||
| 				newStorage | ||||
| 			}); | ||||
|  | ||||
| 			dispatch('refresh'); | ||||
| 			if (isNew) { | ||||
| 				storage.path = null; | ||||
| 				storage.id = null; | ||||
| 			} | ||||
| 			if (newStorage) { | ||||
| 				addToast({ | ||||
| 					message: 'Storage created', | ||||
| 					type: 'success' | ||||
| 				}); | ||||
| 			} else { | ||||
| 				addToast({ | ||||
| 					message: 'Storage updated', | ||||
| 					type: 'success' | ||||
| 				}); | ||||
| 			} | ||||
| 		} catch (error) { | ||||
| 			return errorNotification(error); | ||||
| 		} | ||||
| 	} | ||||
| 	async function removeStorage() { | ||||
| 		try { | ||||
| 			await trpc.applications.deleteStorage.mutate({ | ||||
| 				id, | ||||
| 				path: storage.path | ||||
| 			}); | ||||
| 			dispatch('refresh'); | ||||
| 			addToast({ | ||||
| 				message: 'Storage removed', | ||||
| 				type: 'success' | ||||
| 			}); | ||||
| 		} catch (error) { | ||||
| 			return errorNotification(error); | ||||
| 		} | ||||
| 	} | ||||
| </script> | ||||
|  | ||||
| <div class="w-full lg:px-0 px-4"> | ||||
| 	{#if storage.predefined} | ||||
| 		<div class="flex flex-col lg:flex-row gap-4 pb-2"> | ||||
| 			<input disabled readonly class="w-full" value={storage.id} /> | ||||
| 			<input disabled readonly class="w-full" bind:value={storage.path} /> | ||||
| 		</div> | ||||
| 	{:else} | ||||
| 		<div class="flex gap-4 pb-2" class:pt-8={isNew}> | ||||
| 			{#if storage.applicationId} | ||||
| 				{#if storage.oldPath} | ||||
| 					<input | ||||
| 						disabled | ||||
| 						readonly | ||||
| 						class="w-full" | ||||
| 						value="{storage.applicationId}{storage.path.replace(/\//gi, '-').replace('-app', '')}" | ||||
| 					/> | ||||
| 				{:else} | ||||
| 					<input | ||||
| 						disabled | ||||
| 						readonly | ||||
| 						class="w-full" | ||||
| 						value="{storage.applicationId}{storage.path.replace(/\//gi, '-')}" | ||||
| 					/> | ||||
| 				{/if} | ||||
| 			{/if} | ||||
| 			<input | ||||
| 				disabled={!isNew} | ||||
| 				readonly={!isNew} | ||||
| 				class="w-full" | ||||
| 				bind:value={storage.path} | ||||
| 				required | ||||
| 				placeholder="eg: /data" | ||||
| 			/> | ||||
|  | ||||
| 			<div class="flex items-center justify-center"> | ||||
| 				{#if isNew} | ||||
| 					<div class="w-full lg:w-64"> | ||||
| 						<button class="btn btn-sm btn-primary w-full" on:click={() => saveStorage(true)} | ||||
| 							>Add</button | ||||
| 						> | ||||
| 					</div> | ||||
| 				{:else} | ||||
| 					<div class="flex justify-center"> | ||||
| 						<button class="btn btn-sm btn-error" on:click={removeStorage}>Remove</button> | ||||
| 					</div> | ||||
| 				{/if} | ||||
| 			</div> | ||||
| 		</div> | ||||
| 	{/if} | ||||
| </div> | ||||
| @@ -2,6 +2,60 @@ import { goto } from '$app/navigation'; | ||||
| import { errorNotification } from '$lib/common'; | ||||
| import { trpc } from '$lib/store'; | ||||
|  | ||||
| export async function saveForm() { | ||||
| 	return await trpc.applications.save.mutate(); | ||||
| export async function saveForm(id, application, baseDatabaseBranch, dockerComposeConfiguration) { | ||||
| 	let { | ||||
| 		name, | ||||
| 		buildPack, | ||||
| 		fqdn, | ||||
| 		port, | ||||
| 		exposePort, | ||||
| 		installCommand, | ||||
| 		buildCommand, | ||||
| 		startCommand, | ||||
| 		baseDirectory, | ||||
| 		publishDirectory, | ||||
| 		pythonWSGI, | ||||
| 		pythonModule, | ||||
| 		pythonVariable, | ||||
| 		dockerFileLocation, | ||||
| 		denoMainFile, | ||||
| 		denoOptions, | ||||
| 		gitCommitHash, | ||||
| 		baseImage, | ||||
| 		baseBuildImage, | ||||
| 		deploymentType, | ||||
| 		dockerComposeFile, | ||||
| 		dockerComposeFileLocation, | ||||
| 		simpleDockerfile, | ||||
| 		dockerRegistryImageName | ||||
| 	} = application; | ||||
| 	return await trpc.applications.save.mutate({ | ||||
| 		id, | ||||
| 		name, | ||||
| 		buildPack, | ||||
| 		fqdn, | ||||
| 		port, | ||||
| 		exposePort, | ||||
| 		installCommand, | ||||
| 		buildCommand, | ||||
| 		startCommand, | ||||
| 		baseDirectory, | ||||
| 		publishDirectory, | ||||
| 		pythonWSGI, | ||||
| 		pythonModule, | ||||
| 		pythonVariable, | ||||
| 		dockerFileLocation, | ||||
| 		denoMainFile, | ||||
| 		denoOptions, | ||||
| 		gitCommitHash, | ||||
| 		baseImage, | ||||
| 		baseBuildImage, | ||||
| 		deploymentType, | ||||
| 		dockerComposeFile, | ||||
| 		dockerComposeFileLocation, | ||||
| 		simpleDockerfile, | ||||
| 		dockerRegistryImageName, | ||||
| 		baseDatabaseBranch, | ||||
| 		dockerComposeConfiguration: JSON.stringify(dockerComposeConfiguration) | ||||
| 	}); | ||||
| } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user
	 Andras Bacsai
					Andras Bacsai