gitlab-ci-local
Version:
Tired of pushing to test your .gitlab-ci.yml?
136 lines • 28.3 kB
JavaScript
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"]}