UNPKG

gitlab-ci-local

Version:

Tired of pushing to test your .gitlab-ci.yml?

136 lines 28.3 kB
import fs from "fs-extra"; import * as yaml from "js-yaml"; import chalk from "chalk"; import assert from "assert"; import { Utils } from "./utils.js"; import dotenv from "dotenv"; export class VariablesFromFiles { static async init(argv, writeStreams, gitData) { const cwd = argv.cwd; const stateDir = argv.stateDir; const homeDir = argv.home; const remoteVariables = argv.remoteVariables; const autoCompleting = argv.autoCompleting; const homeVariablesFile = `${homeDir}/${stateDir}/variables.yml`; const variables = {}; let remoteFileData = {}; let homeFileData = {}; if (remoteVariables && !autoCompleting) { const match = /(?<url>git@.*?)=(?<file>.*?)=(?<ref>.*)/.exec(remoteVariables); assert(match != null, "--remote-variables is malformed use 'git@gitlab.com:firecow/example.git=gitlab-variables.yml=master' syntax"); const url = match.groups?.url; const file = match.groups?.file; const ref = match.groups?.ref; const res = await Utils.bash(`set -eou pipefail; git archive --remote=${url} ${ref} ${file} | tar -xO ${file}`, cwd); remoteFileData = yaml.load(`${res.stdout}`); } if (await fs.pathExists(homeVariablesFile)) { homeFileData = yaml.load(await fs.readFile(homeVariablesFile, "utf8"), { schema: yaml.FAILSAFE_SCHEMA }); } const unpack = (v) => { if (typeof v === "string") { const catchAll = { values: {}, type: null }; catchAll.values = {}; catchAll.values["*"] = v; return catchAll; } else { v.type = v.type ?? "variable"; } return v; }; const addToVariables = async (key, val, scopePriority, isDotEnv = false) => { const { type, values } = unpack(val); for (const [matcher, content] of Object.entries(values)) { assert(typeof content == "string", `${key}.${matcher} content must be text or multiline text`); if (isDotEnv || type === "variable" || (type === null && !/^[/~]/.exec(content))) { const regexp = matcher === "*" ? /.*/g : new RegExp(`^${matcher.replace(/\*/g, ".*")}$`, "g"); variables[key] = variables[key] ?? { type: "variable", environments: [] }; variables[key].environments.push({ content, regexp, regexpPriority: matcher.length, scopePriority }); } else if (type === null && /^[/~]/.exec(content)) { const fileSource = content.replace(/^~\/(.*)/, `${homeDir}/$1`); const regexp = matcher === "*" ? /.*/g : new RegExp(`^${matcher.replace(/\*/g, ".*")}$`, "g"); variables[key] = variables[key] ?? { type: "file", environments: [] }; if (fs.existsSync(fileSource)) { variables[key].environments.push({ content, regexp, regexpPriority: matcher.length, scopePriority, fileSource }); } else { variables[key].environments.push({ content: `warn: ${key} is pointing to invalid path\n`, regexp, regexpPriority: matcher.length, scopePriority }); } } else if (type === "file") { const regexp = matcher === "*" ? /.*/g : new RegExp(`^${matcher.replace(/\*/g, ".*")}$`, "g"); variables[key] = variables[key] ?? { type: "file", environments: [] }; variables[key].environments.push({ content, regexp, regexpPriority: matcher.length, scopePriority }); } else { assert(false, `${key} was not handled properly`); } } }; const addVariableFileToVariables = async (fileData, filePriority) => { for (const [globalKey, globalEntry] of Object.entries(fileData?.global ?? {})) { await addToVariables(globalKey, globalEntry, 1 + filePriority); } const groupUrl = `${gitData.remote.host}/${gitData.remote.group}/`; for (const [groupKey, groupEntries] of Object.entries(fileData?.group ?? {})) { if (!groupUrl.includes(this.normalizeProjectKey(groupKey, writeStreams))) continue; assert(groupEntries != null, "groupEntries cannot be null/undefined"); assert(Utils.isObject(groupEntries), "group entries in variable files must be an object"); for (const [k, v] of Object.entries(groupEntries)) { await addToVariables(k, v, 2 + filePriority); } } const projectUrl = `${gitData.remote.host}/${gitData.remote.group}/${gitData.remote.project}.git`; for (const [projectKey, projectEntries] of Object.entries(fileData?.project ?? [])) { if (!projectUrl.includes(this.normalizeProjectKey(projectKey, writeStreams))) continue; assert(projectEntries != null, "projectEntries cannot be null/undefined"); assert(Utils.isObject(projectEntries), "project entries in variable files must be an object"); for (const [k, v] of Object.entries(projectEntries)) { await addToVariables(k, v, 3 + filePriority); } } }; await addVariableFileToVariables(remoteFileData, 0); await addVariableFileToVariables(homeFileData, 10); const projectVariablesFile = `${argv.cwd}/${argv.variablesFile}`; if (fs.existsSync(projectVariablesFile)) { let isDotEnvFormat = false; const projectVariablesFileRawContent = await fs.readFile(projectVariablesFile, "utf8"); let projectVariablesFileData; try { projectVariablesFileData = yaml.load(projectVariablesFileRawContent, { schema: yaml.FAILSAFE_SCHEMA }) ?? {}; if (typeof (projectVariablesFileData) === "string") { isDotEnvFormat = true; projectVariablesFileData = dotenv.parse(projectVariablesFileRawContent); } } catch (e) { if (e instanceof yaml.YAMLException) { isDotEnvFormat = true; projectVariablesFileData = dotenv.parse(projectVariablesFileRawContent); } } assert(projectVariablesFileData != null, "projectEntries cannot be null/undefined"); assert(Utils.isObject(projectVariablesFileData), `${argv.cwd}/.gitlab-ci-local-variables.yml must contain an object`); for (const [k, v] of Object.entries(projectVariablesFileData)) { await addToVariables(k, v, 24, isDotEnvFormat); } } for (const varObj of Object.values(variables)) { varObj.environments.sort((a, b) => b.scopePriority - a.scopePriority); varObj.environments.sort((a, b) => b.regexpPriority - a.regexpPriority); } return variables; } static normalizeProjectKey(key, writeStreams) { if (!key.includes(":")) return key; writeStreams.stderr(chalk `{yellow WARNING: Interpreting '${key}' as '${key.replace(":", "/")}'}\n`); return key.replace(":", "/"); } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"variables-from-files.js","sourceRoot":"","sources":["variables-from-files.ts"],"names":[],"mappings":"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","sourcesContent":["import {WriteStreams} from \"./write-streams.js\";\nimport {GitData} from \"./git-data.js\";\nimport fs from \"fs-extra\";\nimport * as yaml from \"js-yaml\";\nimport chalk from \"chalk\";\nimport {Argv} from \"./argv.js\";\nimport assert from \"assert\";\nimport {Utils} from \"./utils.js\";\nimport dotenv from \"dotenv\";\n\nexport interface CICDVariable {\n    type: \"file\" | \"variable\";\n    environments: {\n        content: string;\n        regexp: RegExp;\n        regexpPriority: number;\n        scopePriority: number;\n        fileSource?: string;\n    }[];\n}\n\nexport class VariablesFromFiles {\n\n    static async init (argv: Argv, writeStreams: WriteStreams, gitData: GitData): Promise<{[name: string]: CICDVariable}> {\n        const cwd = argv.cwd;\n        const stateDir = argv.stateDir;\n        const homeDir = argv.home;\n        const remoteVariables = argv.remoteVariables;\n        const autoCompleting = argv.autoCompleting;\n        const homeVariablesFile = `${homeDir}/${stateDir}/variables.yml`;\n        const variables: {[name: string]: CICDVariable} = {};\n        let remoteFileData: any = {};\n        let homeFileData: any = {};\n\n        if (remoteVariables && !autoCompleting) {\n            const match = /(?<url>git@.*?)=(?<file>.*?)=(?<ref>.*)/.exec(remoteVariables);\n            assert(match != null, \"--remote-variables is malformed use 'git@gitlab.com:firecow/example.git=gitlab-variables.yml=master' syntax\");\n            const url = match.groups?.url;\n            const file = match.groups?.file;\n            const ref = match.groups?.ref;\n            const res = await Utils.bash(`set -eou pipefail; git archive --remote=${url} ${ref} ${file} | tar -xO ${file}`, cwd);\n            remoteFileData = yaml.load(`${res.stdout}`);\n        }\n\n        if (await fs.pathExists(homeVariablesFile)) {\n            homeFileData = yaml.load(await fs.readFile(homeVariablesFile, \"utf8\"), {schema: yaml.FAILSAFE_SCHEMA});\n        }\n\n        const unpack = (v: any): {values: any; type: \"file\" | \"variable\" | null} => {\n            if (typeof v === \"string\") {\n                const catchAll: {values: any; type: \"file\" | \"variable\" | null} = {values: {}, type: null};\n                catchAll.values = {};\n                catchAll.values[\"*\"] = v;\n                return catchAll;\n            } else {\n                v.type = v.type ?? \"variable\";\n            }\n            return v;\n        };\n        const addToVariables = async (key: string, val: any, scopePriority: number, isDotEnv = false) => {\n            const {type, values} = unpack(val);\n            for (const [matcher, content] of Object.entries(values)) {\n                assert(typeof content == \"string\", `${key}.${matcher} content must be text or multiline text`);\n                if (isDotEnv || type === \"variable\" || (type === null && !/^[/~]/.exec(content))) {\n                    const regexp = matcher === \"*\" ? /.*/g : new RegExp(`^${matcher.replace(/\\*/g, \".*\")}$`, \"g\");\n                    variables[key] = variables[key] ?? {type: \"variable\", environments: []};\n                    variables[key].environments.push({content, regexp, regexpPriority: matcher.length, scopePriority});\n                } else if (type === null && /^[/~]/.exec(content)) {\n                    const fileSource = content.replace(/^~\\/(.*)/, `${homeDir}/$1`);\n                    const regexp = matcher === \"*\" ? /.*/g : new RegExp(`^${matcher.replace(/\\*/g, \".*\")}$`, \"g\");\n                    variables[key] = variables[key] ?? {type: \"file\", environments: []};\n                    if (fs.existsSync(fileSource)) {\n                        variables[key].environments.push({content, regexp, regexpPriority: matcher.length, scopePriority, fileSource});\n                    } else {\n                        variables[key].environments.push({content: `warn: ${key} is pointing to invalid path\\n`, regexp, regexpPriority: matcher.length, scopePriority});\n                    }\n                } else if (type === \"file\") {\n                    const regexp = matcher === \"*\" ? /.*/g : new RegExp(`^${matcher.replace(/\\*/g, \".*\")}$`, \"g\");\n                    variables[key] = variables[key] ?? {type: \"file\", environments: []};\n                    variables[key].environments.push({content, regexp, regexpPriority: matcher.length, scopePriority});\n                } else {\n                    assert(false, `${key} was not handled properly`);\n                }\n            }\n        };\n\n        const addVariableFileToVariables = async (fileData: any, filePriority: number) => {\n            for (const [globalKey, globalEntry] of Object.entries(fileData?.global ?? {})) {\n                await addToVariables(globalKey, globalEntry, 1 + filePriority);\n            }\n\n            const groupUrl = `${gitData.remote.host}/${gitData.remote.group}/`;\n            for (const [groupKey, groupEntries] of Object.entries(fileData?.group ?? {})) {\n                if (!groupUrl.includes(this.normalizeProjectKey(groupKey, writeStreams))) continue;\n                assert(groupEntries != null, \"groupEntries cannot be null/undefined\");\n                assert(Utils.isObject(groupEntries), \"group entries in variable files must be an object\");\n                for (const [k, v] of Object.entries(groupEntries)) {\n                    await addToVariables(k, v, 2 + filePriority);\n                }\n            }\n\n            const projectUrl = `${gitData.remote.host}/${gitData.remote.group}/${gitData.remote.project}.git`;\n            for (const [projectKey, projectEntries] of Object.entries(fileData?.project ?? [])) {\n                if (!projectUrl.includes(this.normalizeProjectKey(projectKey, writeStreams))) continue;\n                assert(projectEntries != null, \"projectEntries cannot be null/undefined\");\n                assert(Utils.isObject(projectEntries), \"project entries in variable files must be an object\");\n                for (const [k, v] of Object.entries(projectEntries)) {\n                    await addToVariables(k, v, 3 + filePriority);\n                }\n            }\n        };\n\n        await addVariableFileToVariables(remoteFileData, 0);\n        await addVariableFileToVariables(homeFileData, 10);\n\n        const projectVariablesFile = `${argv.cwd}/${argv.variablesFile}`;\n        if (fs.existsSync(projectVariablesFile)) {\n            let isDotEnvFormat = false;\n            const projectVariablesFileRawContent = await fs.readFile(projectVariablesFile, \"utf8\");\n            let projectVariablesFileData;\n            try {\n                projectVariablesFileData = yaml.load(projectVariablesFileRawContent, {schema: yaml.FAILSAFE_SCHEMA}) ?? {};\n\n                if (typeof(projectVariablesFileData) === \"string\") {\n                    isDotEnvFormat = true;\n                    projectVariablesFileData = dotenv.parse(projectVariablesFileRawContent);\n                }\n            } catch (e) {\n                if (e instanceof yaml.YAMLException) {\n                    isDotEnvFormat = true;\n                    projectVariablesFileData = dotenv.parse(projectVariablesFileRawContent);\n                }\n            }\n            assert(projectVariablesFileData != null, \"projectEntries cannot be null/undefined\");\n            assert(Utils.isObject(projectVariablesFileData), `${argv.cwd}/.gitlab-ci-local-variables.yml must contain an object`);\n            for (const [k, v] of Object.entries(projectVariablesFileData)) {\n                await addToVariables(k, v, 24, isDotEnvFormat);\n            }\n        }\n\n        for (const varObj of Object.values(variables)) {\n            varObj.environments.sort((a, b) => b.scopePriority - a.scopePriority);\n            varObj.environments.sort((a, b) => b.regexpPriority - a.regexpPriority);\n        }\n\n        return variables;\n    }\n\n    static normalizeProjectKey (key: string, writeStreams: WriteStreams): string {\n        if (!key.includes(\":\")) return key;\n        writeStreams.stderr(chalk`{yellow WARNING: Interpreting '${key}' as '${key.replace(\":\", \"/\")}'}\\n`);\n        return key.replace(\":\", \"/\");\n    }\n}\n"]}