commit
						5adbd5e784
					
				| @ -1,7 +1,7 @@ | |||||||
| { | { | ||||||
| 	"name": "coolify", | 	"name": "coolify", | ||||||
| 	"description": "An open-source & self-hostable Heroku / Netlify alternative.", | 	"description": "An open-source & self-hostable Heroku / Netlify alternative.", | ||||||
| 	"version": "2.0.27", | 	"version": "2.0.28", | ||||||
| 	"license": "AGPL-3.0", | 	"license": "AGPL-3.0", | ||||||
| 	"scripts": { | 	"scripts": { | ||||||
| 		"dev": "docker-compose -f docker-compose-dev.yaml up -d && NODE_ENV=development svelte-kit dev --host 0.0.0.0", | 		"dev": "docker-compose -f docker-compose-dev.yaml up -d && NODE_ENV=development svelte-kit dev --host 0.0.0.0", | ||||||
|  | |||||||
| @ -0,0 +1,13 @@ | |||||||
|  | -- CreateTable | ||||||
|  | CREATE TABLE "ServiceSecret" ( | ||||||
|  |     "id" TEXT NOT NULL PRIMARY KEY, | ||||||
|  |     "name" TEXT NOT NULL, | ||||||
|  |     "value" TEXT NOT NULL, | ||||||
|  |     "createdAt" DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, | ||||||
|  |     "updatedAt" DATETIME NOT NULL, | ||||||
|  |     "serviceId" TEXT NOT NULL, | ||||||
|  |     CONSTRAINT "ServiceSecret_serviceId_fkey" FOREIGN KEY ("serviceId") REFERENCES "Service" ("id") ON DELETE RESTRICT ON UPDATE CASCADE | ||||||
|  | ); | ||||||
|  | 
 | ||||||
|  | -- CreateIndex | ||||||
|  | CREATE UNIQUE INDEX "ServiceSecret_name_serviceId_key" ON "ServiceSecret"("name", "serviceId"); | ||||||
| @ -122,6 +122,18 @@ model Secret { | |||||||
|   @@unique([name, applicationId, isPRMRSecret]) |   @@unique([name, applicationId, isPRMRSecret]) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | model ServiceSecret { | ||||||
|  |   id        String   @id @default(cuid()) | ||||||
|  |   name      String | ||||||
|  |   value     String | ||||||
|  |   createdAt DateTime @default(now()) | ||||||
|  |   updatedAt DateTime @updatedAt | ||||||
|  |   service   Service  @relation(fields: [serviceId], references: [id]) | ||||||
|  |   serviceId String | ||||||
|  | 
 | ||||||
|  |   @@unique([name, serviceId]) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| model BuildLog { | model BuildLog { | ||||||
|   id            String  @id @default(cuid()) |   id            String  @id @default(cuid()) | ||||||
|   applicationId String? |   applicationId String? | ||||||
| @ -252,6 +264,7 @@ model Service { | |||||||
|   minio               Minio? |   minio               Minio? | ||||||
|   vscodeserver        Vscodeserver? |   vscodeserver        Vscodeserver? | ||||||
|   wordpress           Wordpress? |   wordpress           Wordpress? | ||||||
|  |   serviceSecret       ServiceSecret[] | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| model PlausibleAnalytics { | model PlausibleAnalytics { | ||||||
|  | |||||||
| @ -15,6 +15,9 @@ export async function isDockerNetworkExists({ network }) { | |||||||
| 	return await prisma.destinationDocker.findFirst({ where: { network } }); | 	return await prisma.destinationDocker.findFirst({ where: { network } }); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | export async function isServiceSecretExists({ id, name }) { | ||||||
|  | 	return await prisma.serviceSecret.findFirst({ where: { name, serviceId: id } }); | ||||||
|  | } | ||||||
| export async function isSecretExists({ id, name, isPRMRSecret }) { | export async function isSecretExists({ id, name, isPRMRSecret }) { | ||||||
| 	return await prisma.secret.findFirst({ where: { name, applicationId: id, isPRMRSecret } }); | 	return await prisma.secret.findFirst({ where: { name, applicationId: id, isPRMRSecret } }); | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,6 +1,19 @@ | |||||||
| import { encrypt, decrypt } from '$lib/crypto'; | import { encrypt, decrypt } from '$lib/crypto'; | ||||||
| import { prisma } from './common'; | import { prisma } from './common'; | ||||||
| 
 | 
 | ||||||
|  | export async function listServiceSecrets(serviceId: string) { | ||||||
|  | 	let secrets = await prisma.serviceSecret.findMany({ | ||||||
|  | 		where: { serviceId }, | ||||||
|  | 		orderBy: { createdAt: 'desc' } | ||||||
|  | 	}); | ||||||
|  | 	secrets = secrets.map((secret) => { | ||||||
|  | 		secret.value = decrypt(secret.value); | ||||||
|  | 		return secret; | ||||||
|  | 	}); | ||||||
|  | 
 | ||||||
|  | 	return secrets; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| export async function listSecrets(applicationId: string) { | export async function listSecrets(applicationId: string) { | ||||||
| 	let secrets = await prisma.secret.findMany({ | 	let secrets = await prisma.secret.findMany({ | ||||||
| 		where: { applicationId }, | 		where: { applicationId }, | ||||||
| @ -14,6 +27,12 @@ export async function listSecrets(applicationId: string) { | |||||||
| 	return secrets; | 	return secrets; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | export async function createServiceSecret({ id, name, value }) { | ||||||
|  | 	value = encrypt(value); | ||||||
|  | 	return await prisma.serviceSecret.create({ | ||||||
|  | 		data: { name, value, service: { connect: { id } } } | ||||||
|  | 	}); | ||||||
|  | } | ||||||
| export async function createSecret({ id, name, value, isBuildSecret, isPRMRSecret }) { | export async function createSecret({ id, name, value, isBuildSecret, isPRMRSecret }) { | ||||||
| 	value = encrypt(value); | 	value = encrypt(value); | ||||||
| 	return await prisma.secret.create({ | 	return await prisma.secret.create({ | ||||||
| @ -21,10 +40,24 @@ export async function createSecret({ id, name, value, isBuildSecret, isPRMRSecre | |||||||
| 	}); | 	}); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | export async function updateServiceSecret({ id, name, value }) { | ||||||
|  | 	value = encrypt(value); | ||||||
|  | 	const found = await prisma.serviceSecret.findFirst({ where: { serviceId: id, name } }); | ||||||
|  | 
 | ||||||
|  | 	if (found) { | ||||||
|  | 		return await prisma.serviceSecret.updateMany({ | ||||||
|  | 			where: { serviceId: id, name }, | ||||||
|  | 			data: { value } | ||||||
|  | 		}); | ||||||
|  | 	} else { | ||||||
|  | 		return await prisma.serviceSecret.create({ | ||||||
|  | 			data: { name, value, service: { connect: { id } } } | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  | } | ||||||
| export async function updateSecret({ id, name, value, isBuildSecret, isPRMRSecret }) { | export async function updateSecret({ id, name, value, isBuildSecret, isPRMRSecret }) { | ||||||
| 	value = encrypt(value); | 	value = encrypt(value); | ||||||
| 	const found = await prisma.secret.findFirst({ where: { applicationId: id, name, isPRMRSecret } }); | 	const found = await prisma.secret.findFirst({ where: { applicationId: id, name, isPRMRSecret } }); | ||||||
| 	console.log(found); |  | ||||||
| 
 | 
 | ||||||
| 	if (found) { | 	if (found) { | ||||||
| 		return await prisma.secret.updateMany({ | 		return await prisma.secret.updateMany({ | ||||||
| @ -38,6 +71,10 @@ export async function updateSecret({ id, name, value, isBuildSecret, isPRMRSecre | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | export async function removeServiceSecret({ id, name }) { | ||||||
|  | 	return await prisma.serviceSecret.deleteMany({ where: { serviceId: id, name } }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| export async function removeSecret({ id, name }) { | export async function removeSecret({ id, name }) { | ||||||
| 	return await prisma.secret.deleteMany({ where: { applicationId: id, name } }); | 	return await prisma.secret.deleteMany({ where: { applicationId: id, name } }); | ||||||
| } | } | ||||||
|  | |||||||
| @ -19,7 +19,8 @@ export async function getService({ id, teamId }) { | |||||||
| 			plausibleAnalytics: true, | 			plausibleAnalytics: true, | ||||||
| 			minio: true, | 			minio: true, | ||||||
| 			vscodeserver: true, | 			vscodeserver: true, | ||||||
| 			wordpress: true | 			wordpress: true, | ||||||
|  | 			serviceSecret: true | ||||||
| 		} | 		} | ||||||
| 	}); | 	}); | ||||||
| 
 | 
 | ||||||
| @ -42,6 +43,12 @@ export async function getService({ id, teamId }) { | |||||||
| 	if (body.wordpress?.mysqlRootUserPassword) | 	if (body.wordpress?.mysqlRootUserPassword) | ||||||
| 		body.wordpress.mysqlRootUserPassword = decrypt(body.wordpress.mysqlRootUserPassword); | 		body.wordpress.mysqlRootUserPassword = decrypt(body.wordpress.mysqlRootUserPassword); | ||||||
| 
 | 
 | ||||||
|  | 	if (body?.serviceSecret.length > 0) { | ||||||
|  | 		body.serviceSecret = body.serviceSecret.map((s) => { | ||||||
|  | 			s.value = decrypt(s.value); | ||||||
|  | 			return s; | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
| 	return { ...body }; | 	return { ...body }; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -159,5 +166,7 @@ export async function removeService({ id }) { | |||||||
| 	await prisma.minio.deleteMany({ where: { serviceId: id } }); | 	await prisma.minio.deleteMany({ where: { serviceId: id } }); | ||||||
| 	await prisma.vscodeserver.deleteMany({ where: { serviceId: id } }); | 	await prisma.vscodeserver.deleteMany({ where: { serviceId: id } }); | ||||||
| 	await prisma.wordpress.deleteMany({ where: { serviceId: id } }); | 	await prisma.wordpress.deleteMany({ where: { serviceId: id } }); | ||||||
|  | 	await prisma.serviceSecret.deleteMany({ where: { serviceId: id } }); | ||||||
|  | 
 | ||||||
| 	await prisma.service.delete({ where: { id } }); | 	await prisma.service.delete({ where: { id } }); | ||||||
| } | } | ||||||
|  | |||||||
| @ -112,9 +112,15 @@ export async function haproxyInstance() { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export async function configureHAProxy() { | export async function configureHAProxy() { | ||||||
|  | 	const haproxy = await haproxyInstance(); | ||||||
|  | 
 | ||||||
| 	try { | 	try { | ||||||
| 		const haproxy = await haproxyInstance(); |  | ||||||
| 		await checkHAProxy(haproxy); | 		await checkHAProxy(haproxy); | ||||||
|  | 	} catch (error) { | ||||||
|  | 		return 'Error: HAProxy is not running'; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	try { | ||||||
| 		const data = { | 		const data = { | ||||||
| 			applications: [], | 			applications: [], | ||||||
| 			services: [], | 			services: [], | ||||||
|  | |||||||
| @ -49,7 +49,12 @@ export async function completeTransaction(transactionId) { | |||||||
| } | } | ||||||
| export async function deleteProxy({ id }) { | export async function deleteProxy({ id }) { | ||||||
| 	const haproxy = await haproxyInstance(); | 	const haproxy = await haproxyInstance(); | ||||||
| 	await checkHAProxy(haproxy); | 	try { | ||||||
|  | 		await checkHAProxy(haproxy); | ||||||
|  | 	} catch (error) { | ||||||
|  | 		return 'Error: HAProxy is not running'; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	let transactionId; | 	let transactionId; | ||||||
| 	try { | 	try { | ||||||
| 		await haproxy.get(`v2/services/haproxy/configuration/backends/${id}`).json(); | 		await haproxy.get(`v2/services/haproxy/configuration/backends/${id}`).json(); | ||||||
|  | |||||||
| @ -1,4 +1,3 @@ | |||||||
| import { dev } from '$app/env'; |  | ||||||
| import { ErrorHandler } from '$lib/database'; | import { ErrorHandler } from '$lib/database'; | ||||||
| import { configureHAProxy } from '$lib/haproxy/configuration'; | import { configureHAProxy } from '$lib/haproxy/configuration'; | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -25,7 +25,7 @@ | |||||||
| define('WP_ALLOW_MULTISITE', true); | define('WP_ALLOW_MULTISITE', true); | ||||||
| define('MULTISITE', true); | define('MULTISITE', true); | ||||||
| define('SUBDOMAIN_INSTALL', false);` | define('SUBDOMAIN_INSTALL', false);` | ||||||
| 			: null}>{service.wordpress.extraConfig || 'N/A'}</textarea | 			: 'N/A'}>{service.wordpress.extraConfig}</textarea | ||||||
| 	> | 	> | ||||||
| </div> | </div> | ||||||
| <div class="flex space-x-1 py-5 font-bold"> | <div class="flex space-x-1 py-5 font-bold"> | ||||||
|  | |||||||
| @ -57,13 +57,13 @@ | |||||||
| </script> | </script> | ||||||
| 
 | 
 | ||||||
| <script> | <script> | ||||||
| 	import { session } from '$app/stores'; | 	import { page, session } from '$app/stores'; | ||||||
| 	import { errorNotification } from '$lib/form'; | 	import { errorNotification } from '$lib/form'; | ||||||
| 	import DeleteIcon from '$lib/components/DeleteIcon.svelte'; | 	import DeleteIcon from '$lib/components/DeleteIcon.svelte'; | ||||||
| 	import Loading from '$lib/components/Loading.svelte'; | 	import Loading from '$lib/components/Loading.svelte'; | ||||||
| 	import { del, post } from '$lib/api'; | 	import { del, post } from '$lib/api'; | ||||||
| 	import { goto } from '$app/navigation'; | 	import { goto } from '$app/navigation'; | ||||||
| 	import { onMount } from 'svelte'; | 	const { id } = $page.params; | ||||||
| 
 | 
 | ||||||
| 	export let service; | 	export let service; | ||||||
| 	export let isRunning; | 	export let isRunning; | ||||||
| @ -168,6 +168,76 @@ | |||||||
| 					</svg> | 					</svg> | ||||||
| 				</button> | 				</button> | ||||||
| 			{/if} | 			{/if} | ||||||
|  | 			<div class="border border-stone-700 h-8" /> | ||||||
|  | 		{/if} | ||||||
|  | 		{#if service.type && service.destinationDockerId && service.version} | ||||||
|  | 			<a | ||||||
|  | 				href="/services/{id}" | ||||||
|  | 				sveltekit:prefetch | ||||||
|  | 				class="hover:text-yellow-500 rounded" | ||||||
|  | 				class:text-yellow-500={$page.url.pathname === `/services/${id}`} | ||||||
|  | 				class:bg-coolgray-500={$page.url.pathname === `/services/${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="/services/{id}/secrets" | ||||||
|  | 				sveltekit:prefetch | ||||||
|  | 				class="hover:text-pink-500 rounded" | ||||||
|  | 				class:text-pink-500={$page.url.pathname === `/services/${id}/secrets`} | ||||||
|  | 				class:bg-coolgray-500={$page.url.pathname === `/services/${id}/secrets`} | ||||||
|  | 			> | ||||||
|  | 				<button | ||||||
|  | 					title="Secrets" | ||||||
|  | 					class="icons bg-transparent tooltip-bottom text-sm disabled:text-red-500" | ||||||
|  | 					data-tooltip="Secrets" | ||||||
|  | 				> | ||||||
|  | 					<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 | ||||||
|  | 			> | ||||||
|  | 			<div class="border border-stone-700 h-8" /> | ||||||
| 		{/if} | 		{/if} | ||||||
| 		<button | 		<button | ||||||
| 			on:click={deleteService} | 			on:click={deleteService} | ||||||
|  | |||||||
| @ -14,19 +14,32 @@ export const post: RequestHandler = async (event) => { | |||||||
| 
 | 
 | ||||||
| 	try { | 	try { | ||||||
| 		const service = await db.getService({ id, teamId }); | 		const service = await db.getService({ id, teamId }); | ||||||
| 		const { type, version, destinationDockerId, destinationDocker } = service; | 		const { type, version, destinationDockerId, destinationDocker, serviceSecret } = service; | ||||||
| 		const network = destinationDockerId && destinationDocker.network; | 		const network = destinationDockerId && destinationDocker.network; | ||||||
| 		const host = getEngine(destinationDocker.engine); | 		const host = getEngine(destinationDocker.engine); | ||||||
| 
 | 
 | ||||||
| 		const { workdir } = await createDirectories({ repository: type, buildId: id }); | 		const { workdir } = await createDirectories({ repository: type, buildId: id }); | ||||||
| 		const image = getServiceImage(type); | 		const image = getServiceImage(type); | ||||||
|  | 
 | ||||||
|  | 		const config = { | ||||||
|  | 			image: `${image}:${version}`, | ||||||
|  | 			volume: `${id}-ngrams:/ngrams`, | ||||||
|  | 			environmentVariables: {} | ||||||
|  | 		}; | ||||||
|  | 
 | ||||||
|  | 		if (serviceSecret.length > 0) { | ||||||
|  | 			serviceSecret.forEach((secret) => { | ||||||
|  | 				config.environmentVariables[secret.name] = secret.value; | ||||||
|  | 			}); | ||||||
|  | 		} | ||||||
| 		const composeFile = { | 		const composeFile = { | ||||||
| 			version: '3.8', | 			version: '3.8', | ||||||
| 			services: { | 			services: { | ||||||
| 				[id]: { | 				[id]: { | ||||||
| 					container_name: id, | 					container_name: id, | ||||||
| 					image: `${image}:${version}`, | 					image: config.image, | ||||||
| 					networks: [network], | 					networks: [network], | ||||||
|  | 					environment: config.environmentVariables, | ||||||
| 					restart: 'always', | 					restart: 'always', | ||||||
| 					volumes: [`${id}-ngrams:/ngrams`], | 					volumes: [`${id}-ngrams:/ngrams`], | ||||||
| 					labels: makeLabelForServices('languagetool') | 					labels: makeLabelForServices('languagetool') | ||||||
|  | |||||||
| @ -6,7 +6,7 @@ import type { RequestHandler } from '@sveltejs/kit'; | |||||||
| import { startHttpProxy } from '$lib/haproxy'; | import { startHttpProxy } from '$lib/haproxy'; | ||||||
| import getPort, { portNumbers } from 'get-port'; | import getPort, { portNumbers } from 'get-port'; | ||||||
| import { getDomain } from '$lib/components/common'; | import { getDomain } from '$lib/components/common'; | ||||||
| import { ErrorHandler } from '$lib/database'; | import { ErrorHandler, getServiceImage } from '$lib/database'; | ||||||
| import { makeLabelForServices } from '$lib/buildPacks/common'; | import { makeLabelForServices } from '$lib/buildPacks/common'; | ||||||
| 
 | 
 | ||||||
| export const post: RequestHandler = async (event) => { | export const post: RequestHandler = async (event) => { | ||||||
| @ -23,7 +23,8 @@ export const post: RequestHandler = async (event) => { | |||||||
| 			fqdn, | 			fqdn, | ||||||
| 			destinationDockerId, | 			destinationDockerId, | ||||||
| 			destinationDocker, | 			destinationDocker, | ||||||
| 			minio: { rootUser, rootUserPassword } | 			minio: { rootUser, rootUserPassword }, | ||||||
|  | 			serviceSecret | ||||||
| 		} = service; | 		} = service; | ||||||
| 
 | 
 | ||||||
| 		const data = await db.prisma.setting.findFirst(); | 		const data = await db.prisma.setting.findFirst(); | ||||||
| @ -38,9 +39,10 @@ export const post: RequestHandler = async (event) => { | |||||||
| 		const apiPort = 9000; | 		const apiPort = 9000; | ||||||
| 
 | 
 | ||||||
| 		const { workdir } = await createDirectories({ repository: type, buildId: id }); | 		const { workdir } = await createDirectories({ repository: type, buildId: id }); | ||||||
|  | 		const image = getServiceImage(type); | ||||||
| 
 | 
 | ||||||
| 		const config = { | 		const config = { | ||||||
| 			image: `minio/minio:${version}`, | 			image: `${image}:${version}`, | ||||||
| 			volume: `${id}-minio-data:/data`, | 			volume: `${id}-minio-data:/data`, | ||||||
| 			environmentVariables: { | 			environmentVariables: { | ||||||
| 				MINIO_ROOT_USER: rootUser, | 				MINIO_ROOT_USER: rootUser, | ||||||
| @ -48,12 +50,17 @@ export const post: RequestHandler = async (event) => { | |||||||
| 				MINIO_BROWSER_REDIRECT_URL: fqdn | 				MINIO_BROWSER_REDIRECT_URL: fqdn | ||||||
| 			} | 			} | ||||||
| 		}; | 		}; | ||||||
|  | 		if (serviceSecret.length > 0) { | ||||||
|  | 			serviceSecret.forEach((secret) => { | ||||||
|  | 				config.environmentVariables[secret.name] = secret.value; | ||||||
|  | 			}); | ||||||
|  | 		} | ||||||
| 		const composeFile = { | 		const composeFile = { | ||||||
| 			version: '3.8', | 			version: '3.8', | ||||||
| 			services: { | 			services: { | ||||||
| 				[id]: { | 				[id]: { | ||||||
| 					container_name: id, | 					container_name: id, | ||||||
| 					image: `minio/minio:${version}`, | 					image: config.image, | ||||||
| 					command: `server /data --console-address ":${consolePort}"`, | 					command: `server /data --console-address ":${consolePort}"`, | ||||||
| 					environment: config.environmentVariables, | 					environment: config.environmentVariables, | ||||||
| 					networks: [network], | 					networks: [network], | ||||||
|  | |||||||
| @ -3,7 +3,7 @@ import * as db from '$lib/database'; | |||||||
| import { promises as fs } from 'fs'; | import { promises as fs } from 'fs'; | ||||||
| import yaml from 'js-yaml'; | import yaml from 'js-yaml'; | ||||||
| import type { RequestHandler } from '@sveltejs/kit'; | import type { RequestHandler } from '@sveltejs/kit'; | ||||||
| import { ErrorHandler } from '$lib/database'; | import { ErrorHandler, getServiceImage } from '$lib/database'; | ||||||
| import { makeLabelForServices } from '$lib/buildPacks/common'; | import { makeLabelForServices } from '$lib/buildPacks/common'; | ||||||
| 
 | 
 | ||||||
| export const post: RequestHandler = async (event) => { | export const post: RequestHandler = async (event) => { | ||||||
| @ -14,19 +14,30 @@ export const post: RequestHandler = async (event) => { | |||||||
| 
 | 
 | ||||||
| 	try { | 	try { | ||||||
| 		const service = await db.getService({ id, teamId }); | 		const service = await db.getService({ id, teamId }); | ||||||
| 		const { type, version, destinationDockerId, destinationDocker } = service; | 		const { type, version, destinationDockerId, destinationDocker, serviceSecret } = service; | ||||||
| 		const network = destinationDockerId && destinationDocker.network; | 		const network = destinationDockerId && destinationDocker.network; | ||||||
| 		const host = getEngine(destinationDocker.engine); | 		const host = getEngine(destinationDocker.engine); | ||||||
| 
 | 
 | ||||||
| 		const { workdir } = await createDirectories({ repository: type, buildId: id }); | 		const { workdir } = await createDirectories({ repository: type, buildId: id }); | ||||||
|  | 		const image = getServiceImage(type); | ||||||
| 
 | 
 | ||||||
|  | 		const config = { | ||||||
|  | 			image: `${image}:${version}`, | ||||||
|  | 			environmentVariables: {} | ||||||
|  | 		}; | ||||||
|  | 		if (serviceSecret.length > 0) { | ||||||
|  | 			serviceSecret.forEach((secret) => { | ||||||
|  | 				config.environmentVariables[secret.name] = secret.value; | ||||||
|  | 			}); | ||||||
|  | 		} | ||||||
| 		const composeFile = { | 		const composeFile = { | ||||||
| 			version: '3.8', | 			version: '3.8', | ||||||
| 			services: { | 			services: { | ||||||
| 				[id]: { | 				[id]: { | ||||||
| 					container_name: id, | 					container_name: id, | ||||||
| 					image: `nocodb/nocodb:${version}`, | 					image: config.image, | ||||||
| 					networks: [network], | 					networks: [network], | ||||||
|  | 					environment: config.environmentVariables, | ||||||
| 					restart: 'always', | 					restart: 'always', | ||||||
| 					labels: makeLabelForServices('nocodb') | 					labels: makeLabelForServices('nocodb') | ||||||
| 				} | 				} | ||||||
|  | |||||||
| @ -3,7 +3,7 @@ import * as db from '$lib/database'; | |||||||
| import { promises as fs } from 'fs'; | import { promises as fs } from 'fs'; | ||||||
| import yaml from 'js-yaml'; | import yaml from 'js-yaml'; | ||||||
| import type { RequestHandler } from '@sveltejs/kit'; | import type { RequestHandler } from '@sveltejs/kit'; | ||||||
| import { ErrorHandler } from '$lib/database'; | import { ErrorHandler, getServiceImage } from '$lib/database'; | ||||||
| import { makeLabelForServices } from '$lib/buildPacks/common'; | import { makeLabelForServices } from '$lib/buildPacks/common'; | ||||||
| 
 | 
 | ||||||
| export const post: RequestHandler = async (event) => { | export const post: RequestHandler = async (event) => { | ||||||
| @ -20,6 +20,7 @@ export const post: RequestHandler = async (event) => { | |||||||
| 			fqdn, | 			fqdn, | ||||||
| 			destinationDockerId, | 			destinationDockerId, | ||||||
| 			destinationDocker, | 			destinationDocker, | ||||||
|  | 			serviceSecret, | ||||||
| 			plausibleAnalytics: { | 			plausibleAnalytics: { | ||||||
| 				id: plausibleDbId, | 				id: plausibleDbId, | ||||||
| 				username, | 				username, | ||||||
| @ -31,10 +32,11 @@ export const post: RequestHandler = async (event) => { | |||||||
| 				secretKeyBase | 				secretKeyBase | ||||||
| 			} | 			} | ||||||
| 		} = service; | 		} = service; | ||||||
|  | 		const image = getServiceImage(type); | ||||||
| 
 | 
 | ||||||
| 		const config = { | 		const config = { | ||||||
| 			plausibleAnalytics: { | 			plausibleAnalytics: { | ||||||
| 				image: `plausible/analytics:${version}`, | 				image: `${image}:${version}`, | ||||||
| 				environmentVariables: { | 				environmentVariables: { | ||||||
| 					ADMIN_USER_EMAIL: email, | 					ADMIN_USER_EMAIL: email, | ||||||
| 					ADMIN_USER_NAME: username, | 					ADMIN_USER_NAME: username, | ||||||
| @ -68,6 +70,11 @@ export const post: RequestHandler = async (event) => { | |||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 		}; | 		}; | ||||||
|  | 		if (serviceSecret.length > 0) { | ||||||
|  | 			serviceSecret.forEach((secret) => { | ||||||
|  | 				config.plausibleAnalytics.environmentVariables[secret.name] = secret.value; | ||||||
|  | 			}); | ||||||
|  | 		} | ||||||
| 		const network = destinationDockerId && destinationDocker.network; | 		const network = destinationDockerId && destinationDocker.network; | ||||||
| 		const host = getEngine(destinationDocker.engine); | 		const host = getEngine(destinationDocker.engine); | ||||||
| 
 | 
 | ||||||
|  | |||||||
							
								
								
									
										87
									
								
								src/routes/services/[id]/secrets/_Secret.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								src/routes/services/[id]/secrets/_Secret.svelte
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,87 @@ | |||||||
|  | <script> | ||||||
|  | 	export let name = ''; | ||||||
|  | 	export let value = ''; | ||||||
|  | 	export let isNewSecret = false; | ||||||
|  | 
 | ||||||
|  | 	import { page } from '$app/stores'; | ||||||
|  | 	import { del, post } from '$lib/api'; | ||||||
|  | 	import CopyPasswordField from '$lib/components/CopyPasswordField.svelte'; | ||||||
|  | 	import { errorNotification } from '$lib/form'; | ||||||
|  | 	import { toast } from '@zerodevx/svelte-toast'; | ||||||
|  | 	import { createEventDispatcher } from 'svelte'; | ||||||
|  | 
 | ||||||
|  | 	const dispatch = createEventDispatcher(); | ||||||
|  | 	const { id } = $page.params; | ||||||
|  | 	async function removeSecret() { | ||||||
|  | 		try { | ||||||
|  | 			await del(`/services/${id}/secrets.json`, { name }); | ||||||
|  | 			dispatch('refresh'); | ||||||
|  | 			if (isNewSecret) { | ||||||
|  | 				name = ''; | ||||||
|  | 				value = ''; | ||||||
|  | 			} | ||||||
|  | 		} catch ({ error }) { | ||||||
|  | 			return errorNotification(error); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	async function saveSecret(isNew = false) { | ||||||
|  | 		if (!name) return errorNotification('Name is required.'); | ||||||
|  | 		if (!value) return errorNotification('Value is required.'); | ||||||
|  | 		try { | ||||||
|  | 			await post(`/services/${id}/secrets.json`, { | ||||||
|  | 				name, | ||||||
|  | 				value, | ||||||
|  | 
 | ||||||
|  | 				isNew | ||||||
|  | 			}); | ||||||
|  | 			dispatch('refresh'); | ||||||
|  | 			if (isNewSecret) { | ||||||
|  | 				name = ''; | ||||||
|  | 				value = ''; | ||||||
|  | 			} | ||||||
|  | 			toast.push('Secret saved.'); | ||||||
|  | 		} catch ({ error }) { | ||||||
|  | 			return errorNotification(error); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | </script> | ||||||
|  | 
 | ||||||
|  | <td> | ||||||
|  | 	<input | ||||||
|  | 		id={isNewSecret ? 'secretName' : 'secretNameNew'} | ||||||
|  | 		bind:value={name} | ||||||
|  | 		required | ||||||
|  | 		placeholder="EXAMPLE_VARIABLE" | ||||||
|  | 		class=" border border-dashed border-coolgray-300" | ||||||
|  | 		readonly={!isNewSecret} | ||||||
|  | 		class:bg-transparent={!isNewSecret} | ||||||
|  | 		class:cursor-not-allowed={!isNewSecret} | ||||||
|  | 	/> | ||||||
|  | </td> | ||||||
|  | <td> | ||||||
|  | 	<CopyPasswordField | ||||||
|  | 		id={isNewSecret ? 'secretValue' : 'secretValueNew'} | ||||||
|  | 		name={isNewSecret ? 'secretValue' : 'secretValueNew'} | ||||||
|  | 		isPasswordField={true} | ||||||
|  | 		bind:value | ||||||
|  | 		required | ||||||
|  | 		placeholder="J$#@UIO%HO#$U%H" | ||||||
|  | 	/> | ||||||
|  | </td> | ||||||
|  | 
 | ||||||
|  | <td> | ||||||
|  | 	{#if isNewSecret} | ||||||
|  | 		<div class="flex items-center justify-center"> | ||||||
|  | 			<button class="bg-green-600 hover:bg-green-500" on:click={() => saveSecret(true)}>Add</button> | ||||||
|  | 		</div> | ||||||
|  | 	{:else} | ||||||
|  | 		<div class="flex flex-row justify-center space-x-2"> | ||||||
|  | 			<div class="flex items-center justify-center"> | ||||||
|  | 				<button class="" on:click={() => saveSecret(false)}>Set</button> | ||||||
|  | 			</div> | ||||||
|  | 			<div class="flex justify-center items-end"> | ||||||
|  | 				<button class="bg-red-600 hover:bg-red-500" on:click={removeSecret}>Remove</button> | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 	{/if} | ||||||
|  | </td> | ||||||
							
								
								
									
										70
									
								
								src/routes/services/[id]/secrets/index.json.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										70
									
								
								src/routes/services/[id]/secrets/index.json.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,70 @@ | |||||||
|  | import { getTeam, getUserDetails } from '$lib/common'; | ||||||
|  | import * as db from '$lib/database'; | ||||||
|  | import { ErrorHandler } from '$lib/database'; | ||||||
|  | import type { RequestHandler } from '@sveltejs/kit'; | ||||||
|  | 
 | ||||||
|  | export const get: RequestHandler = async (event) => { | ||||||
|  | 	const { teamId, status, body } = await getUserDetails(event); | ||||||
|  | 	if (status === 401) return { status, body }; | ||||||
|  | 
 | ||||||
|  | 	const { id } = event.params; | ||||||
|  | 	try { | ||||||
|  | 		const secrets = await db.listServiceSecrets(id); | ||||||
|  | 		return { | ||||||
|  | 			status: 200, | ||||||
|  | 			body: { | ||||||
|  | 				secrets: secrets.sort((a, b) => { | ||||||
|  | 					return ('' + a.name).localeCompare(b.name); | ||||||
|  | 				}) | ||||||
|  | 			} | ||||||
|  | 		}; | ||||||
|  | 	} catch (error) { | ||||||
|  | 		return ErrorHandler(error); | ||||||
|  | 	} | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | export const post: RequestHandler = async (event) => { | ||||||
|  | 	const { teamId, status, body } = await getUserDetails(event); | ||||||
|  | 	if (status === 401) return { status, body }; | ||||||
|  | 
 | ||||||
|  | 	const { id } = event.params; | ||||||
|  | 	const { name, value, isBuildSecret, isPRMRSecret, isNew } = await event.request.json(); | ||||||
|  | 	try { | ||||||
|  | 		if (isNew) { | ||||||
|  | 			const found = await db.isServiceSecretExists({ id, name }); | ||||||
|  | 			if (found) { | ||||||
|  | 				throw { | ||||||
|  | 					error: `Secret ${name} already exists.` | ||||||
|  | 				}; | ||||||
|  | 			} else { | ||||||
|  | 				await db.createServiceSecret({ id, name, value }); | ||||||
|  | 				return { | ||||||
|  | 					status: 201 | ||||||
|  | 				}; | ||||||
|  | 			} | ||||||
|  | 		} else { | ||||||
|  | 			await db.updateServiceSecret({ id, name, value }); | ||||||
|  | 			return { | ||||||
|  | 				status: 201 | ||||||
|  | 			}; | ||||||
|  | 		} | ||||||
|  | 	} catch (error) { | ||||||
|  | 		return ErrorHandler(error); | ||||||
|  | 	} | ||||||
|  | }; | ||||||
|  | export const del: RequestHandler = async (event) => { | ||||||
|  | 	const { teamId, status, body } = await getUserDetails(event); | ||||||
|  | 	if (status === 401) return { status, body }; | ||||||
|  | 
 | ||||||
|  | 	const { id } = event.params; | ||||||
|  | 	const { name } = await event.request.json(); | ||||||
|  | 
 | ||||||
|  | 	try { | ||||||
|  | 		await db.removeServiceSecret({ id, name }); | ||||||
|  | 		return { | ||||||
|  | 			status: 200 | ||||||
|  | 		}; | ||||||
|  | 	} catch (error) { | ||||||
|  | 		return ErrorHandler(error); | ||||||
|  | 	} | ||||||
|  | }; | ||||||
							
								
								
									
										67
									
								
								src/routes/services/[id]/secrets/index.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										67
									
								
								src/routes/services/[id]/secrets/index.svelte
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,67 @@ | |||||||
|  | <script context="module" lang="ts"> | ||||||
|  | 	import type { Load } from '@sveltejs/kit'; | ||||||
|  | 	export const load: Load = async ({ fetch, params, stuff }) => { | ||||||
|  | 		let endpoint = `/services/${params.id}/secrets.json`; | ||||||
|  | 		const res = await fetch(endpoint); | ||||||
|  | 		if (res.ok) { | ||||||
|  | 			return { | ||||||
|  | 				props: { | ||||||
|  | 					service: stuff.service, | ||||||
|  | 					...(await res.json()) | ||||||
|  | 				} | ||||||
|  | 			}; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		return { | ||||||
|  | 			status: res.status, | ||||||
|  | 			error: new Error(`Could not load ${endpoint}`) | ||||||
|  | 		}; | ||||||
|  | 	}; | ||||||
|  | </script> | ||||||
|  | 
 | ||||||
|  | <script lang="ts"> | ||||||
|  | 	export let secrets; | ||||||
|  | 	export let service; | ||||||
|  | 	import Secret from './_Secret.svelte'; | ||||||
|  | 	import { getDomain } from '$lib/components/common'; | ||||||
|  | 	import { page } from '$app/stores'; | ||||||
|  | 	import { get } from '$lib/api'; | ||||||
|  | 
 | ||||||
|  | 	const { id } = $page.params; | ||||||
|  | 
 | ||||||
|  | 	async function refreshSecrets() { | ||||||
|  | 		const data = await get(`/services/${id}/secrets.json`); | ||||||
|  | 		secrets = [...data.secrets]; | ||||||
|  | 	} | ||||||
|  | </script> | ||||||
|  | 
 | ||||||
|  | <div class="flex space-x-1 p-6 font-bold"> | ||||||
|  | 	<div class="mr-4 text-2xl tracking-tight"> | ||||||
|  | 		Secrets {#if service.fqdn} | ||||||
|  | 			<a href={service.fqdn} target="_blank">{getDomain(service.fqdn)}</a> | ||||||
|  | 		{/if} | ||||||
|  | 	</div> | ||||||
|  | </div> | ||||||
|  | <div class="mx-auto max-w-6xl rounded-xl px-6 pt-4"> | ||||||
|  | 	<table class="mx-auto border-separate text-left"> | ||||||
|  | 		<thead> | ||||||
|  | 			<tr class="h-12"> | ||||||
|  | 				<th scope="col">Name</th> | ||||||
|  | 				<th scope="col">Value</th> | ||||||
|  | 				<th scope="col" class="w-96 text-center">Action</th> | ||||||
|  | 			</tr> | ||||||
|  | 		</thead> | ||||||
|  | 		<tbody> | ||||||
|  | 			{#each secrets as secret} | ||||||
|  | 				{#key secret.id} | ||||||
|  | 					<tr> | ||||||
|  | 						<Secret name={secret.name} value={secret.value} on:refresh={refreshSecrets} /> | ||||||
|  | 					</tr> | ||||||
|  | 				{/key} | ||||||
|  | 			{/each} | ||||||
|  | 			<tr> | ||||||
|  | 				<Secret isNewSecret on:refresh={refreshSecrets} /> | ||||||
|  | 			</tr> | ||||||
|  | 		</tbody> | ||||||
|  | 	</table> | ||||||
|  | </div> | ||||||
| @ -14,25 +14,31 @@ export const post: RequestHandler = async (event) => { | |||||||
| 
 | 
 | ||||||
| 	try { | 	try { | ||||||
| 		const service = await db.getService({ id, teamId }); | 		const service = await db.getService({ id, teamId }); | ||||||
| 		const { type, version, destinationDockerId, destinationDocker } = service; | 		const { type, version, destinationDockerId, destinationDocker, serviceSecret } = service; | ||||||
| 
 | 
 | ||||||
| 		const network = destinationDockerId && destinationDocker.network; | 		const network = destinationDockerId && destinationDocker.network; | ||||||
| 		const host = getEngine(destinationDocker.engine); | 		const host = getEngine(destinationDocker.engine); | ||||||
| 
 | 
 | ||||||
| 		const { workdir } = await createDirectories({ repository: type, buildId: id }); | 		const { workdir } = await createDirectories({ repository: type, buildId: id }); | ||||||
| 		const baseImage = getServiceImage(type); | 		const image = getServiceImage(type); | ||||||
| 
 | 
 | ||||||
| 		const config = { | 		const config = { | ||||||
| 			image: `${baseImage}:${version}`, | 			image: `${image}:${version}`, | ||||||
| 			volume: `${id}-vaultwarden-data:/data/` | 			volume: `${id}-vaultwarden-data:/data/`, | ||||||
|  | 			environmentVariables: {} | ||||||
| 		}; | 		}; | ||||||
| 
 | 		if (serviceSecret.length > 0) { | ||||||
|  | 			serviceSecret.forEach((secret) => { | ||||||
|  | 				config.environmentVariables[secret.name] = secret.value; | ||||||
|  | 			}); | ||||||
|  | 		} | ||||||
| 		const composeFile = { | 		const composeFile = { | ||||||
| 			version: '3.8', | 			version: '3.8', | ||||||
| 			services: { | 			services: { | ||||||
| 				[id]: { | 				[id]: { | ||||||
| 					container_name: id, | 					container_name: id, | ||||||
| 					image: config.image, | 					image: config.image, | ||||||
|  | 					environment: config.environmentVariables, | ||||||
| 					networks: [network], | 					networks: [network], | ||||||
| 					volumes: [config.volume], | 					volumes: [config.volume], | ||||||
| 					restart: 'always', | 					restart: 'always', | ||||||
|  | |||||||
| @ -3,7 +3,7 @@ import * as db from '$lib/database'; | |||||||
| import { promises as fs } from 'fs'; | import { promises as fs } from 'fs'; | ||||||
| import yaml from 'js-yaml'; | import yaml from 'js-yaml'; | ||||||
| import type { RequestHandler } from '@sveltejs/kit'; | import type { RequestHandler } from '@sveltejs/kit'; | ||||||
| import { ErrorHandler } from '$lib/database'; | import { ErrorHandler, getServiceImage } from '$lib/database'; | ||||||
| import { makeLabelForServices } from '$lib/buildPacks/common'; | import { makeLabelForServices } from '$lib/buildPacks/common'; | ||||||
| 
 | 
 | ||||||
| export const post: RequestHandler = async (event) => { | export const post: RequestHandler = async (event) => { | ||||||
| @ -19,6 +19,7 @@ export const post: RequestHandler = async (event) => { | |||||||
| 			version, | 			version, | ||||||
| 			destinationDockerId, | 			destinationDockerId, | ||||||
| 			destinationDocker, | 			destinationDocker, | ||||||
|  | 			serviceSecret, | ||||||
| 			vscodeserver: { password } | 			vscodeserver: { password } | ||||||
| 		} = service; | 		} = service; | ||||||
| 
 | 
 | ||||||
| @ -26,13 +27,20 @@ export const post: RequestHandler = async (event) => { | |||||||
| 		const host = getEngine(destinationDocker.engine); | 		const host = getEngine(destinationDocker.engine); | ||||||
| 
 | 
 | ||||||
| 		const { workdir } = await createDirectories({ repository: type, buildId: id }); | 		const { workdir } = await createDirectories({ repository: type, buildId: id }); | ||||||
|  | 		const image = getServiceImage(type); | ||||||
|  | 
 | ||||||
| 		const config = { | 		const config = { | ||||||
| 			image: `codercom/code-server:${version}`, | 			image: `${image}:${version}`, | ||||||
| 			volume: `${id}-vscodeserver-data:/home/coder`, | 			volume: `${id}-vscodeserver-data:/home/coder`, | ||||||
| 			environmentVariables: { | 			environmentVariables: { | ||||||
| 				PASSWORD: password | 				PASSWORD: password | ||||||
| 			} | 			} | ||||||
| 		}; | 		}; | ||||||
|  | 		if (serviceSecret.length > 0) { | ||||||
|  | 			serviceSecret.forEach((secret) => { | ||||||
|  | 				config.environmentVariables[secret.name] = secret.value; | ||||||
|  | 			}); | ||||||
|  | 		} | ||||||
| 		const composeFile = { | 		const composeFile = { | ||||||
| 			version: '3.8', | 			version: '3.8', | ||||||
| 			services: { | 			services: { | ||||||
|  | |||||||
| @ -3,7 +3,7 @@ import * as db from '$lib/database'; | |||||||
| import { promises as fs } from 'fs'; | import { promises as fs } from 'fs'; | ||||||
| import yaml from 'js-yaml'; | import yaml from 'js-yaml'; | ||||||
| import type { RequestHandler } from '@sveltejs/kit'; | import type { RequestHandler } from '@sveltejs/kit'; | ||||||
| import { ErrorHandler } from '$lib/database'; | import { ErrorHandler, getServiceImage } from '$lib/database'; | ||||||
| import { makeLabelForServices } from '$lib/buildPacks/common'; | import { makeLabelForServices } from '$lib/buildPacks/common'; | ||||||
| 
 | 
 | ||||||
| export const post: RequestHandler = async (event) => { | export const post: RequestHandler = async (event) => { | ||||||
| @ -19,6 +19,7 @@ export const post: RequestHandler = async (event) => { | |||||||
| 			version, | 			version, | ||||||
| 			fqdn, | 			fqdn, | ||||||
| 			destinationDockerId, | 			destinationDockerId, | ||||||
|  | 			serviceSecret, | ||||||
| 			destinationDocker, | 			destinationDocker, | ||||||
| 			wordpress: { | 			wordpress: { | ||||||
| 				mysqlDatabase, | 				mysqlDatabase, | ||||||
| @ -32,11 +33,12 @@ export const post: RequestHandler = async (event) => { | |||||||
| 
 | 
 | ||||||
| 		const network = destinationDockerId && destinationDocker.network; | 		const network = destinationDockerId && destinationDocker.network; | ||||||
| 		const host = getEngine(destinationDocker.engine); | 		const host = getEngine(destinationDocker.engine); | ||||||
|  | 		const image = getServiceImage(type); | ||||||
| 
 | 
 | ||||||
| 		const { workdir } = await createDirectories({ repository: type, buildId: id }); | 		const { workdir } = await createDirectories({ repository: type, buildId: id }); | ||||||
| 		const config = { | 		const config = { | ||||||
| 			wordpress: { | 			wordpress: { | ||||||
| 				image: `wordpress:${version}`, | 				image: `${image}:${version}`, | ||||||
| 				volume: `${id}-wordpress-data:/var/www/html`, | 				volume: `${id}-wordpress-data:/var/www/html`, | ||||||
| 				environmentVariables: { | 				environmentVariables: { | ||||||
| 					WORDPRESS_DB_HOST: `${id}-mysql`, | 					WORDPRESS_DB_HOST: `${id}-mysql`, | ||||||
| @ -58,6 +60,11 @@ export const post: RequestHandler = async (event) => { | |||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 		}; | 		}; | ||||||
|  | 		if (serviceSecret.length > 0) { | ||||||
|  | 			serviceSecret.forEach((secret) => { | ||||||
|  | 				config.wordpress.environmentVariables[secret.name] = secret.value; | ||||||
|  | 			}); | ||||||
|  | 		} | ||||||
| 		const composeFile = { | 		const composeFile = { | ||||||
| 			version: '3.8', | 			version: '3.8', | ||||||
| 			services: { | 			services: { | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user