From 7218b3f79b434a5941a9ca883655bf3da39042d1 Mon Sep 17 00:00:00 2001 From: Mauricio Siu <47042324+Siumauricio@users.noreply.github.com> Date: Tue, 11 Mar 2025 00:38:10 -0600 Subject: [PATCH] feat(templates): support array-based environment variable configuration Add support for processing environment variables defined as an array in template configurations, allowing more flexible env var definitions with direct string values and variable interpolation --- .../templates/config.template.test.ts | 26 +++++++++++++++++++ packages/server/src/templates/processors.ts | 20 +++++++++++--- 2 files changed, 43 insertions(+), 3 deletions(-) diff --git a/apps/dokploy/__test__/templates/config.template.test.ts b/apps/dokploy/__test__/templates/config.template.test.ts index 145e1372..4b0e8fac 100644 --- a/apps/dokploy/__test__/templates/config.template.test.ts +++ b/apps/dokploy/__test__/templates/config.template.test.ts @@ -169,6 +169,32 @@ describe("processTemplate", () => { expect(secretKey.split("=")[1]).toHaveLength(64); }); + it("should process env vars when provided as an array", () => { + const template: CompleteTemplate = { + metadata: {} as any, + variables: {}, + config: { + domains: [], + env: [ + 'CLOUDFLARE_TUNNEL_TOKEN=""', + 'ANOTHER_VAR="some value"', + "DOMAIN=${domain}", + ], + mounts: [], + }, + }; + + const result = processTemplate(template, mockSchema); + expect(result.envs).toHaveLength(3); + + // Should preserve exact format for static values + expect(result.envs[0]).toBe('CLOUDFLARE_TUNNEL_TOKEN=""'); + expect(result.envs[1]).toBe('ANOTHER_VAR="some value"'); + + // Should process variables in array items + expect(result.envs[2]).toContain(mockSchema.projectName); + }); + it("should allow using utility functions directly in env vars", () => { const template: CompleteTemplate = { metadata: {} as any, diff --git a/packages/server/src/templates/processors.ts b/packages/server/src/templates/processors.ts index c45a8b0f..e4b54809 100644 --- a/packages/server/src/templates/processors.ts +++ b/packages/server/src/templates/processors.ts @@ -45,7 +45,7 @@ export interface CompleteTemplate { variables: Record; config: { domains: DomainConfig[]; - env: Record; + env: Record | string[]; mounts?: MountConfig[]; }; } @@ -175,7 +175,8 @@ export function processDomains( variables: Record, schema: Schema, ): Template["domains"] { - return template.config.domains.map((domain: DomainConfig) => ({ + if (!template?.config?.domains) return []; + return template?.config?.domains?.map((domain: DomainConfig) => ({ ...domain, host: domain.host ? processValue(domain.host, variables, schema) @@ -191,6 +192,19 @@ export function processEnvVars( variables: Record, schema: Schema, ): Template["envs"] { + if (!template?.config?.env) return []; + + // Handle array of env vars + if (Array.isArray(template.config.env)) { + return template.config.env.map((env) => { + if (typeof env === "string") { + return processValue(env, variables, schema); + } + return env; + }); + } + + // Handle object of env vars return Object.entries(template.config.env).map( ([key, value]: [string, string]) => { const processedValue = processValue(value, variables, schema); @@ -207,7 +221,7 @@ export function processMounts( variables: Record, schema: Schema, ): Template["mounts"] { - if (!template.config.mounts) return []; + if (!template?.config?.mounts) return []; return template.config.mounts.map((mount: MountConfig) => ({ filePath: processValue(mount.filePath, variables, schema),