gitlab-ci-local 4.65.1 → 4.66.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/src/validator.js DELETED
@@ -1,151 +0,0 @@
1
- import Ajv from "ajv";
2
- import assert from "assert";
3
- import chalk from "chalk-template";
4
- import schema from "./schema/index.js";
5
- import { betterAjvErrors } from "./schema-error.js";
6
- import terminalLink from "terminal-link";
7
- const MAX_ERRORS = 5;
8
- export class Validator {
9
- static jsonSchemaValidation({ pathToExpandedGitLabCi, gitLabCiConfig, argv }) {
10
- const ajv = new Ajv({
11
- verbose: true,
12
- allErrors: true,
13
- allowUnionTypes: true,
14
- validateFormats: false,
15
- strictTypes: false, // to suppress the missing types defined in the gitlab-ci json schema
16
- keywords: ["markdownDescription"],
17
- });
18
- const validate = ajv.compile(schema);
19
- const valid = validate(gitLabCiConfig);
20
- if (valid)
21
- return;
22
- const betterErrors = betterAjvErrors({
23
- data: gitLabCiConfig,
24
- errors: validate.errors,
25
- }).filter(betterError => !argv.ignoreSchemaPaths.includes(betterError.schemaPath));
26
- let e = "";
27
- for (let i = 0, len = betterErrors.length; i < len; i++) {
28
- if (i + 1 > MAX_ERRORS) {
29
- e += `\t... and ${len - MAX_ERRORS} more`;
30
- break;
31
- }
32
- e += chalk `\t• {redBright ${betterErrors[i].message}} at {blueBright ${betterErrors[i].path}} {grey [${betterErrors[i].schemaPath}]}\n`;
33
- }
34
- assert(valid || betterErrors.length == 0, chalk `
35
- {reset Invalid .gitlab-ci.yml configuration!
36
- ${e.trimEnd()}
37
-
38
- For further troubleshooting, consider either of the following:
39
- \t• Copy the content of {blueBright ${terminalLink(".gitlab-ci-local/expanded-gitlab-ci.yml", pathToExpandedGitLabCi)}} to the ${terminalLink("pipeline editor", "https://docs.gitlab.com/ee/ci/pipeline_editor/")} to debug it
40
- \t• Use --ignore-schema-paths "#/definitions/tags/minItems" --ignore-schema-paths "#/additionalProperties" to partially disable certain validation rule
41
- \t• Use --json-schema-validation=false to disable schema validation (not recommended)}
42
- `);
43
- }
44
- static needs(jobs, stages) {
45
- const warnings = [];
46
- for (const job of jobs) {
47
- if (job.needs === null || job.needs.length === 0)
48
- continue;
49
- for (const [i, need] of job.needs.entries()) {
50
- if (need.pipeline) {
51
- warnings.push(`${job.name}.needs[${i}].job:${need.job} ignored, pipeline key not supported`);
52
- continue;
53
- }
54
- if (need.project) {
55
- warnings.push(`${job.name}.needs[${i}] ignored, project key not supported`);
56
- continue;
57
- }
58
- const needJob = jobs.find(j => j.baseName === need.job);
59
- if (need.optional && !needJob)
60
- continue;
61
- assert(needJob != null, chalk `needs: [{blueBright ${need.job}}] for {blueBright ${job.baseName}} could not be found`);
62
- const needJobStageIndex = stages.indexOf(needJob.stage);
63
- const jobStageIndex = stages.indexOf(job.stage);
64
- assert(needJobStageIndex <= jobStageIndex, chalk `needs: [{blueBright ${needJob.name}}] for {blueBright ${job.name}} is in a future stage`);
65
- }
66
- }
67
- return warnings;
68
- }
69
- static dependencies(jobs, stages) {
70
- for (const job of jobs) {
71
- if (job.dependencies === null || job.dependencies.length === 0)
72
- continue;
73
- const undefDeps = job.dependencies.filter((j) => !jobs.some(n => n.baseName === j));
74
- assert(undefDeps.length !== job.dependencies.length, chalk `dependencies: [{blueBright ${undefDeps.join(",")}}] for {blueBright ${job.name}} cannot be found`);
75
- for (const dep of job.dependencies) {
76
- const depJob = jobs.find(j => j.baseName === dep);
77
- assert(depJob != null, chalk `dependencies: [{blueBright ${dep}}] for {blueBright ${job.baseName}} could not be found`);
78
- const depJobStageIndex = stages.indexOf(depJob.stage);
79
- const jobStageIndex = stages.indexOf(job.stage);
80
- assert(depJobStageIndex <= jobStageIndex, chalk `dependencies: [{blueBright ${depJob.name}}] for {blueBright ${job.name}} is in a future stage`);
81
- }
82
- }
83
- }
84
- static dependenciesContainment(jobs) {
85
- for (const job of jobs) {
86
- const needs = job.needs;
87
- const dependencies = job.dependencies;
88
- if (needs && needs.length === 0)
89
- continue;
90
- if (!dependencies || !needs)
91
- continue;
92
- const everyIncluded = dependencies.every((dep) => {
93
- return needs.some(n => n.job === dep);
94
- });
95
- const assertMsg = `${job.formattedJobName} needs: '${needs.map(n => n.job).join(",")}' doesn't fully contain dependencies: '${dependencies.join(",")}'`;
96
- assert(everyIncluded, assertMsg);
97
- }
98
- }
99
- /**
100
- * These jobs named are reserved keywords in GitLab CI but does not prevent the pipeline from running
101
- * https://github.com/firecow/gitlab-ci-local/issues/1263
102
- * @param jobsNames
103
- * @private
104
- */
105
- static potentialIllegalJobName(jobsNames) {
106
- const warnings = [];
107
- for (const jobName of jobsNames) {
108
- if (new Set(["types", "true", "false", "nil"]).has(jobName)) {
109
- warnings.push(`Job name "${jobName}" is a reserved keyword. (https://docs.gitlab.com/ee/ci/jobs/#job-name-limitations)`);
110
- }
111
- }
112
- return warnings;
113
- }
114
- static scriptBlank(jobs) {
115
- for (const job of jobs) {
116
- if (job.trigger)
117
- continue; // Jobs with trigger are allowed to have empty script
118
- assert(job.scripts.length > 0, chalk `{blue ${job.name}} has empty script`);
119
- }
120
- }
121
- static arrayOfStrings(jobs) {
122
- for (const job of jobs) {
123
- if (job.trigger)
124
- continue;
125
- job.beforeScripts.forEach((s) => assert(typeof s === "string", chalk `{blue ${job.name}} before_script contains non string value`));
126
- job.afterScripts.forEach((s) => assert(typeof s === "string", chalk `{blue ${job.name}} after_script contains non string value`));
127
- job.scripts.forEach((s) => assert(typeof s === "string", chalk `{blue ${job.name}} script contains non string value`));
128
- }
129
- }
130
- static async run(jobs, stages) {
131
- const warnings = [];
132
- this.scriptBlank(jobs);
133
- this.arrayOfStrings(jobs);
134
- warnings.push(...this.needs(jobs, stages));
135
- this.dependencies(jobs, stages);
136
- this.dependenciesContainment(jobs);
137
- warnings.push(...this.potentialIllegalJobName(jobs.map(j => j.baseName)));
138
- warnings.push(...this.artifacts(jobs));
139
- return warnings;
140
- }
141
- static artifacts(jobs) {
142
- const warnings = [];
143
- for (const job of jobs) {
144
- if (job.artifacts === null) {
145
- warnings.push(`${job.name}.artifacts is null, ignoring.`);
146
- }
147
- }
148
- return warnings;
149
- }
150
- }
151
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"validator.js","sourceRoot":"","sources":["validator.ts"],"names":[],"mappings":"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","sourcesContent":["import Ajv from \"ajv\";\nimport {Job} from \"./job.js\";\nimport assert from \"assert\";\nimport chalk from \"chalk-template\";\nimport schema from \"./schema/index.js\";\nimport {betterAjvErrors} from \"./schema-error.js\";\nimport terminalLink from \"terminal-link\";\nimport {Argv} from \"./argv.js\";\n\nconst MAX_ERRORS = 5;\n\nexport class Validator {\n    static jsonSchemaValidation ({pathToExpandedGitLabCi, gitLabCiConfig, argv}: {\n        pathToExpandedGitLabCi: string;\n        gitLabCiConfig: object;\n        argv: Argv;\n    }) {\n        const ajv = new Ajv({\n            verbose: true,\n            allErrors: true,\n            allowUnionTypes: true,\n            validateFormats: false,\n            strictTypes: false, // to suppress the missing types defined in the gitlab-ci json schema\n            keywords: [\"markdownDescription\"],\n        });\n        const validate = ajv.compile(schema);\n        const valid = validate(gitLabCiConfig);\n        if (valid) return;\n        const betterErrors = betterAjvErrors({\n            data: gitLabCiConfig,\n            errors: validate.errors,\n        }).filter(betterError => !argv.ignoreSchemaPaths.includes(betterError.schemaPath));\n\n        let e: string = \"\";\n        for (let i = 0, len = betterErrors.length; i < len; i++) {\n            if (i + 1 > MAX_ERRORS) {\n                e += `\\t... and ${len - MAX_ERRORS} more`;\n                break;\n            }\n            e += chalk`\\t• {redBright ${betterErrors[i].message}} at {blueBright ${betterErrors[i].path}} {grey [${betterErrors[i].schemaPath}]}\\n`;\n        }\n\n        assert(valid || betterErrors.length == 0, chalk`\n{reset Invalid .gitlab-ci.yml configuration!\n${e.trimEnd()}\n\nFor further troubleshooting, consider either of the following:\n\\t• Copy the content of {blueBright ${terminalLink(\".gitlab-ci-local/expanded-gitlab-ci.yml\", pathToExpandedGitLabCi)}} to the ${terminalLink(\"pipeline editor\", \"https://docs.gitlab.com/ee/ci/pipeline_editor/\")} to debug it\n\\t• Use --ignore-schema-paths \"#/definitions/tags/minItems\" --ignore-schema-paths \"#/additionalProperties\" to partially disable certain validation rule\n\\t• Use --json-schema-validation=false to disable schema validation (not recommended)}\n`);\n    }\n\n    private static needs (jobs: ReadonlyArray<Job>, stages: readonly string[]): string[] {\n        const warnings: string[] = [];\n        for (const job of jobs) {\n            if (job.needs === null || job.needs.length === 0) continue;\n\n            for (const [i, need] of job.needs.entries()) {\n                if (need.pipeline) {\n                    warnings.push(`${job.name}.needs[${i}].job:${need.job} ignored, pipeline key not supported`);\n                    continue;\n                }\n                if (need.project) {\n                    warnings.push(`${job.name}.needs[${i}] ignored, project key not supported`);\n                    continue;\n                }\n                const needJob = jobs.find(j => j.baseName === need.job);\n                if (need.optional && !needJob) continue;\n                assert(needJob != null, chalk`needs: [{blueBright ${need.job}}] for {blueBright ${job.baseName}} could not be found`);\n                const needJobStageIndex = stages.indexOf(needJob.stage);\n                const jobStageIndex = stages.indexOf(job.stage);\n                assert(needJobStageIndex <= jobStageIndex, chalk`needs: [{blueBright ${needJob.name}}] for {blueBright ${job.name}} is in a future stage`);\n            }\n\n        }\n        return warnings;\n    }\n\n    private static dependencies (jobs: ReadonlyArray<Job>, stages: readonly string[]) {\n        for (const job of jobs) {\n            if (job.dependencies === null || job.dependencies.length === 0) continue;\n\n            const undefDeps = job.dependencies.filter((j) => !jobs.some(n => n.baseName === j));\n            assert(undefDeps.length !== job.dependencies.length, chalk`dependencies: [{blueBright ${undefDeps.join(\",\")}}] for {blueBright ${job.name}} cannot be found`);\n\n            for (const dep of job.dependencies) {\n                const depJob = jobs.find(j => j.baseName === dep);\n                assert(depJob != null, chalk`dependencies: [{blueBright ${dep}}] for {blueBright ${job.baseName}} could not be found`);\n                const depJobStageIndex = stages.indexOf(depJob.stage);\n                const jobStageIndex = stages.indexOf(job.stage);\n                assert(depJobStageIndex <= jobStageIndex, chalk`dependencies: [{blueBright ${depJob.name}}] for {blueBright ${job.name}} is in a future stage`);\n            }\n        }\n    }\n\n    private static dependenciesContainment (jobs: ReadonlyArray<Job>) {\n        for (const job of jobs) {\n            const needs = job.needs;\n            const dependencies = job.dependencies;\n            if (needs && needs.length === 0) continue;\n            if (!dependencies || !needs) continue;\n\n\n            const everyIncluded = dependencies.every((dep: string) => {\n                return needs.some(n => n.job === dep);\n            });\n            const assertMsg = `${job.formattedJobName} needs: '${needs.map(n => n.job).join(\",\")}' doesn't fully contain dependencies: '${dependencies.join(\",\")}'`;\n            assert(everyIncluded, assertMsg);\n        }\n    }\n\n    /**\n     * These jobs named are reserved keywords in GitLab CI but does not prevent the pipeline from running\n     * https://github.com/firecow/gitlab-ci-local/issues/1263\n     * @param jobsNames\n     * @private\n     */\n    private static potentialIllegalJobName (jobsNames: string[]) {\n        const warnings = [];\n        for (const jobName of jobsNames) {\n            if (new Set([\"types\", \"true\", \"false\", \"nil\"]).has(jobName)) {\n                warnings.push(`Job name \"${jobName}\" is a reserved keyword. (https://docs.gitlab.com/ee/ci/jobs/#job-name-limitations)`);\n            }\n        }\n        return warnings;\n    }\n\n    private static scriptBlank (jobs: ReadonlyArray<Job>) {\n        for (const job of jobs) {\n            if (job.trigger) continue; // Jobs with trigger are allowed to have empty script\n            assert(job.scripts.length > 0, chalk`{blue ${job.name}} has empty script`);\n        }\n    }\n\n    private static arrayOfStrings (jobs: ReadonlyArray<Job>) {\n        for (const job of jobs) {\n            if (job.trigger) continue;\n            job.beforeScripts.forEach((s: any) => assert(typeof s === \"string\", chalk`{blue ${job.name}} before_script contains non string value`));\n            job.afterScripts.forEach((s: any) => assert(typeof s === \"string\", chalk`{blue ${job.name}} after_script contains non string value`));\n            job.scripts.forEach((s: any) => assert(typeof s === \"string\", chalk`{blue ${job.name}} script contains non string value`));\n        }\n    }\n\n    static async run (jobs: ReadonlyArray<Job>, stages: readonly string[]) {\n        const warnings: string[] = [];\n        this.scriptBlank(jobs);\n        this.arrayOfStrings(jobs);\n        warnings.push(...this.needs(jobs, stages));\n        this.dependencies(jobs, stages);\n        this.dependenciesContainment(jobs);\n        warnings.push(...this.potentialIllegalJobName(jobs.map(j => j.baseName)));\n        warnings.push(...this.artifacts(jobs));\n        return warnings;\n    }\n\n    private static artifacts (jobs: ReadonlyArray<Job>) {\n        const warnings: string[] = [];\n        for (const job of jobs) {\n            if (job.artifacts === null) {\n                warnings.push(`${job.name}.artifacts is null, ignoring.`);\n            }\n        }\n        return warnings;\n    }\n}\n"]}
@@ -1,143 +0,0 @@
1
- import fs from "fs-extra";
2
- import * as yaml from "js-yaml";
3
- import chalk from "chalk-template";
4
- import assert from "assert";
5
- import { Utils } from "./utils.js";
6
- import dotenv from "dotenv";
7
- import deepExtend from "deep-extend";
8
- export class VariablesFromFiles {
9
- static async init(argv, writeStreams, gitData) {
10
- const cwd = argv.cwd;
11
- const stateDir = argv.stateDir;
12
- const homeDir = argv.home;
13
- const remoteVariables = argv.remoteVariables;
14
- const autoCompleting = argv.autoCompleting;
15
- const homeVariablesFile = `${homeDir}/${stateDir}/variables.yml`;
16
- const variables = {};
17
- let remoteFileData = {};
18
- let homeFileData = {};
19
- if (remoteVariables && !autoCompleting) {
20
- for (let i = 0; i < remoteVariables.length; i++) {
21
- const match = /(?<url>git@.*?)=(?<file>.*?)=(?<ref>.*)/.exec(remoteVariables[i]);
22
- assert(match != null, "--remote-variables is malformed use 'git@gitlab.com:firecow/example.git=gitlab-variables.yml=master' syntax");
23
- const url = match.groups?.url;
24
- const file = match.groups?.file;
25
- const ref = match.groups?.ref;
26
- const res = await Utils.bash(`set -eou pipefail; git archive --remote=${url} ${ref} ${file} | tar -xO ${file}`, cwd);
27
- const loadedYaml = yaml.load(`${res.stdout}`);
28
- // Check if loadedYaml is an object
29
- if (typeof loadedYaml === "object" && loadedYaml !== null) {
30
- remoteFileData = deepExtend(remoteFileData, loadedYaml);
31
- }
32
- }
33
- }
34
- if (await fs.pathExists(homeVariablesFile)) {
35
- homeFileData = yaml.load(await fs.readFile(homeVariablesFile, "utf8"), { schema: yaml.FAILSAFE_SCHEMA });
36
- }
37
- const unpack = (v) => {
38
- if (typeof v === "string") {
39
- const catchAll = { values: {}, type: null };
40
- catchAll.values = {};
41
- catchAll.values["*"] = v;
42
- return catchAll;
43
- }
44
- else {
45
- v.type = v.type ?? "variable";
46
- }
47
- return v;
48
- };
49
- const addToVariables = async (key, val, scopePriority, isDotEnv = false) => {
50
- const { type, values } = unpack(val);
51
- for (const [matcher, content] of Object.entries(values)) {
52
- assert(typeof content == "string", `${key}.${matcher} content must be text or multiline text`);
53
- if (isDotEnv || type === "variable" || (type === null && !/^[/~]/.exec(content))) {
54
- const regexp = matcher === "*" ? /.*/g : new RegExp(`^${matcher.replace(/\*/g, ".*")}$`, "g");
55
- variables[key] = variables[key] ?? { type: "variable", environments: [] };
56
- variables[key].environments.push({ content, regexp, regexpPriority: matcher.length, scopePriority });
57
- }
58
- else if (type === null && /^[/~]/.exec(content)) {
59
- const fileSource = content.replace(/^~\/(.*)/, `${homeDir}/$1`);
60
- const regexp = matcher === "*" ? /.*/g : new RegExp(`^${matcher.replace(/\*/g, ".*")}$`, "g");
61
- variables[key] = variables[key] ?? { type: "file", environments: [] };
62
- if (fs.existsSync(fileSource)) {
63
- variables[key].environments.push({ content, regexp, regexpPriority: matcher.length, scopePriority, fileSource });
64
- }
65
- else {
66
- variables[key].environments.push({ content: `warn: ${key} is pointing to invalid path\n`, regexp, regexpPriority: matcher.length, scopePriority });
67
- }
68
- }
69
- else if (type === "file") {
70
- const regexp = matcher === "*" ? /.*/g : new RegExp(`^${matcher.replace(/\*/g, ".*")}$`, "g");
71
- variables[key] = variables[key] ?? { type: "file", environments: [] };
72
- variables[key].environments.push({ content, regexp, regexpPriority: matcher.length, scopePriority });
73
- }
74
- else {
75
- assert(false, `${key} was not handled properly`);
76
- }
77
- }
78
- };
79
- const addVariableFileToVariables = async (fileData, filePriority) => {
80
- for (const [globalKey, globalEntry] of Object.entries(fileData?.global ?? {})) {
81
- await addToVariables(globalKey, globalEntry, 1 + filePriority);
82
- }
83
- const groupUrl = `${gitData.remote.host}/${gitData.remote.group}/`;
84
- for (const [groupKey, groupEntries] of Object.entries(fileData?.group ?? {})) {
85
- if (!groupUrl.includes(this.normalizeProjectKey(groupKey, writeStreams)))
86
- continue;
87
- assert(groupEntries != null, "groupEntries cannot be null/undefined");
88
- assert(Utils.isObject(groupEntries), "group entries in variable files must be an object");
89
- for (const [k, v] of Object.entries(groupEntries)) {
90
- await addToVariables(k, v, 2 + filePriority);
91
- }
92
- }
93
- const projectUrl = `${gitData.remote.host}/${gitData.remote.group}/${gitData.remote.project}.git`;
94
- for (const [projectKey, projectEntries] of Object.entries(fileData?.project ?? [])) {
95
- if (!projectUrl.includes(this.normalizeProjectKey(projectKey, writeStreams)))
96
- continue;
97
- assert(projectEntries != null, "projectEntries cannot be null/undefined");
98
- assert(Utils.isObject(projectEntries), "project entries in variable files must be an object");
99
- for (const [k, v] of Object.entries(projectEntries)) {
100
- await addToVariables(k, v, 3 + filePriority);
101
- }
102
- }
103
- };
104
- await addVariableFileToVariables(remoteFileData, 0);
105
- await addVariableFileToVariables(homeFileData, 10);
106
- const projectVariablesFile = `${argv.cwd}/${argv.variablesFile}`;
107
- if (fs.existsSync(projectVariablesFile)) {
108
- let isDotEnvFormat = false;
109
- const projectVariablesFileRawContent = await fs.readFile(projectVariablesFile, "utf8");
110
- let projectVariablesFileData;
111
- try {
112
- projectVariablesFileData = yaml.load(projectVariablesFileRawContent, { schema: yaml.FAILSAFE_SCHEMA }) ?? {};
113
- if (typeof (projectVariablesFileData) === "string") {
114
- isDotEnvFormat = true;
115
- projectVariablesFileData = dotenv.parse(projectVariablesFileRawContent);
116
- }
117
- }
118
- catch (e) {
119
- if (e instanceof yaml.YAMLException) {
120
- isDotEnvFormat = true;
121
- projectVariablesFileData = dotenv.parse(projectVariablesFileRawContent);
122
- }
123
- }
124
- assert(projectVariablesFileData != null, "projectEntries cannot be null/undefined");
125
- assert(Utils.isObject(projectVariablesFileData), `${argv.cwd}/.gitlab-ci-local-variables.yml must contain an object`);
126
- for (const [k, v] of Object.entries(projectVariablesFileData)) {
127
- await addToVariables(k, v, 24, isDotEnvFormat);
128
- }
129
- }
130
- for (const varObj of Object.values(variables)) {
131
- varObj.environments.sort((a, b) => b.scopePriority - a.scopePriority);
132
- varObj.environments.sort((a, b) => b.regexpPriority - a.regexpPriority);
133
- }
134
- return variables;
135
- }
136
- static normalizeProjectKey(key, writeStreams) {
137
- if (!key.includes(":"))
138
- return key;
139
- writeStreams.stderr(chalk `{yellow WARNING: Interpreting '${key}' as '${key.replace(":", "/")}'}\n`);
140
- return key.replace(":", "/");
141
- }
142
- }
143
- //# 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-template\";\nimport {Argv} from \"./argv.js\";\nimport assert from \"assert\";\nimport {Utils} from \"./utils.js\";\nimport dotenv from \"dotenv\";\nimport deepExtend from \"deep-extend\";\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            for (let i = 0; i < remoteVariables.length; i++) {\n                const match = /(?<url>git@.*?)=(?<file>.*?)=(?<ref>.*)/.exec(remoteVariables[i]);\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                const loadedYaml = yaml.load(`${res.stdout}`);\n                // Check if loadedYaml is an object\n                if (typeof loadedYaml === \"object\" && loadedYaml !== null) {\n                    remoteFileData = deepExtend(remoteFileData, loadedYaml);\n                }\n            }\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"]}
@@ -1,51 +0,0 @@
1
- class AbstractWriteStreams {
2
- memoStdout = (() => {
3
- const cache = new Map();
4
- return (message) => {
5
- if (cache.has(message))
6
- return;
7
- cache.set(message, null);
8
- this.stdout(message);
9
- };
10
- })();
11
- }
12
- export class WriteStreamsProcess extends AbstractWriteStreams {
13
- stderr(txt) {
14
- process.stderr.write(txt);
15
- }
16
- stdout(txt) {
17
- process.stdout.write(txt);
18
- }
19
- flush() {
20
- // Process write streams flushes themselves.
21
- }
22
- }
23
- export class WriteStreamsMock extends AbstractWriteStreams {
24
- currentStderr = "";
25
- currentStdout = "";
26
- stderrLines = [];
27
- stdoutLines = [];
28
- stderr(txt) {
29
- this.currentStderr += txt;
30
- if (txt.endsWith("\n")) {
31
- this.stderrLines.push(this.currentStderr.slice(0, -1));
32
- this.currentStderr = "";
33
- }
34
- }
35
- stdout(txt) {
36
- this.currentStdout += txt;
37
- if (txt.endsWith("\n")) {
38
- this.stdoutLines.push(this.currentStdout.slice(0, -1));
39
- this.currentStdout = "";
40
- }
41
- }
42
- flush() {
43
- if (this.currentStdout.length != 0) {
44
- this.stdout("\n");
45
- }
46
- if (this.currentStderr.length != 0) {
47
- this.stderr("\n");
48
- }
49
- }
50
- }
51
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoid3JpdGUtc3RyZWFtcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIndyaXRlLXN0cmVhbXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBT0EsTUFBZSxvQkFBb0I7SUFLL0IsVUFBVSxHQUFHLENBQUMsR0FBRyxFQUFFO1FBQ2YsTUFBTSxLQUFLLEdBQUcsSUFBSSxHQUFHLEVBQUUsQ0FBQztRQUN4QixPQUFPLENBQUMsT0FBZSxFQUFFLEVBQUU7WUFDdkIsSUFBSSxLQUFLLENBQUMsR0FBRyxDQUFDLE9BQU8sQ0FBQztnQkFBRSxPQUFPO1lBQy9CLEtBQUssQ0FBQyxHQUFHLENBQUMsT0FBTyxFQUFFLElBQUksQ0FBQyxDQUFDO1lBQ3pCLElBQUksQ0FBQyxNQUFNLENBQUMsT0FBTyxDQUFDLENBQUM7UUFDekIsQ0FBQyxDQUFDO0lBQ04sQ0FBQyxDQUFDLEVBQUUsQ0FBQztDQUNSO0FBRUQsTUFBTSxPQUFPLG1CQUFvQixTQUFRLG9CQUFvQjtJQUN6RCxNQUFNLENBQUUsR0FBVztRQUNmLE9BQU8sQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDO0lBQzlCLENBQUM7SUFFRCxNQUFNLENBQUUsR0FBVztRQUNmLE9BQU8sQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDO0lBQzlCLENBQUM7SUFFRCxLQUFLO1FBQ0QsNENBQTRDO0lBQ2hELENBQUM7Q0FDSjtBQUVELE1BQU0sT0FBTyxnQkFBaUIsU0FBUSxvQkFBb0I7SUFDOUMsYUFBYSxHQUFHLEVBQUUsQ0FBQztJQUNuQixhQUFhLEdBQUcsRUFBRSxDQUFDO0lBRWxCLFdBQVcsR0FBYSxFQUFFLENBQUM7SUFDM0IsV0FBVyxHQUFhLEVBQUUsQ0FBQztJQUVwQyxNQUFNLENBQUUsR0FBVztRQUNmLElBQUksQ0FBQyxhQUFhLElBQUksR0FBRyxDQUFDO1FBQzFCLElBQUksR0FBRyxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDO1lBQ3JCLElBQUksQ0FBQyxXQUFXLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxhQUFhLENBQUMsS0FBSyxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUM7WUFDdkQsSUFBSSxDQUFDLGFBQWEsR0FBRyxFQUFFLENBQUM7UUFDNUIsQ0FBQztJQUNMLENBQUM7SUFFRCxNQUFNLENBQUUsR0FBVztRQUNmLElBQUksQ0FBQyxhQUFhLElBQUksR0FBRyxDQUFDO1FBQzFCLElBQUksR0FBRyxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDO1lBQ3JCLElBQUksQ0FBQyxXQUFXLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxhQUFhLENBQUMsS0FBSyxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUM7WUFDdkQsSUFBSSxDQUFDLGFBQWEsR0FBRyxFQUFFLENBQUM7UUFDNUIsQ0FBQztJQUNMLENBQUM7SUFFRCxLQUFLO1FBQ0QsSUFBSSxJQUFJLENBQUMsYUFBYSxDQUFDLE1BQU0sSUFBSSxDQUFDLEVBQUUsQ0FBQztZQUNqQyxJQUFJLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxDQUFDO1FBQ3RCLENBQUM7UUFDRCxJQUFJLElBQUksQ0FBQyxhQUFhLENBQUMsTUFBTSxJQUFJLENBQUMsRUFBRSxDQUFDO1lBQ2pDLElBQUksQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDdEIsQ0FBQztJQUNMLENBQUM7Q0FDSiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBpbnRlcmZhY2UgV3JpdGVTdHJlYW1zIHtcbiAgICBzdGRvdXQ6ICh0eHQ6IHN0cmluZykgPT4gdm9pZDtcbiAgICBtZW1vU3Rkb3V0OiAodHh0OiBzdHJpbmcpID0+IHZvaWQ7XG4gICAgc3RkZXJyOiAodHh0OiBzdHJpbmcpID0+IHZvaWQ7XG4gICAgZmx1c2g6ICgpID0+IHZvaWQ7XG59XG5cbmFic3RyYWN0IGNsYXNzIEFic3RyYWN0V3JpdGVTdHJlYW1zIGltcGxlbWVudHMgV3JpdGVTdHJlYW1zIHtcbiAgICBhYnN0cmFjdCBzdGRvdXQgKHR4dDogc3RyaW5nKTogdm9pZDtcbiAgICBhYnN0cmFjdCBzdGRlcnIgKHR4dDogc3RyaW5nKTogdm9pZDtcbiAgICBhYnN0cmFjdCBmbHVzaCAoKTogdm9pZDtcblxuICAgIG1lbW9TdGRvdXQgPSAoKCkgPT4ge1xuICAgICAgICBjb25zdCBjYWNoZSA9IG5ldyBNYXAoKTtcbiAgICAgICAgcmV0dXJuIChtZXNzYWdlOiBzdHJpbmcpID0+IHtcbiAgICAgICAgICAgIGlmIChjYWNoZS5oYXMobWVzc2FnZSkpIHJldHVybjtcbiAgICAgICAgICAgIGNhY2hlLnNldChtZXNzYWdlLCBudWxsKTtcbiAgICAgICAgICAgIHRoaXMuc3Rkb3V0KG1lc3NhZ2UpO1xuICAgICAgICB9O1xuICAgIH0pKCk7XG59XG5cbmV4cG9ydCBjbGFzcyBXcml0ZVN0cmVhbXNQcm9jZXNzIGV4dGVuZHMgQWJzdHJhY3RXcml0ZVN0cmVhbXMge1xuICAgIHN0ZGVyciAodHh0OiBzdHJpbmcpOiB2b2lkIHtcbiAgICAgICAgcHJvY2Vzcy5zdGRlcnIud3JpdGUodHh0KTtcbiAgICB9XG5cbiAgICBzdGRvdXQgKHR4dDogc3RyaW5nKTogdm9pZCB7XG4gICAgICAgIHByb2Nlc3Muc3Rkb3V0LndyaXRlKHR4dCk7XG4gICAgfVxuXG4gICAgZmx1c2ggKCk6IHZvaWQge1xuICAgICAgICAvLyBQcm9jZXNzIHdyaXRlIHN0cmVhbXMgZmx1c2hlcyB0aGVtc2VsdmVzLlxuICAgIH1cbn1cblxuZXhwb3J0IGNsYXNzIFdyaXRlU3RyZWFtc01vY2sgZXh0ZW5kcyBBYnN0cmFjdFdyaXRlU3RyZWFtcyB7XG4gICAgcHJpdmF0ZSBjdXJyZW50U3RkZXJyID0gXCJcIjtcbiAgICBwcml2YXRlIGN1cnJlbnRTdGRvdXQgPSBcIlwiO1xuXG4gICAgcmVhZG9ubHkgc3RkZXJyTGluZXM6IHN0cmluZ1tdID0gW107XG4gICAgcmVhZG9ubHkgc3Rkb3V0TGluZXM6IHN0cmluZ1tdID0gW107XG5cbiAgICBzdGRlcnIgKHR4dDogc3RyaW5nKTogdm9pZCB7XG4gICAgICAgIHRoaXMuY3VycmVudFN0ZGVyciArPSB0eHQ7XG4gICAgICAgIGlmICh0eHQuZW5kc1dpdGgoXCJcXG5cIikpIHtcbiAgICAgICAgICAgIHRoaXMuc3RkZXJyTGluZXMucHVzaCh0aGlzLmN1cnJlbnRTdGRlcnIuc2xpY2UoMCwgLTEpKTtcbiAgICAgICAgICAgIHRoaXMuY3VycmVudFN0ZGVyciA9IFwiXCI7XG4gICAgICAgIH1cbiAgICB9XG5cbiAgICBzdGRvdXQgKHR4dDogc3RyaW5nKTogdm9pZCB7XG4gICAgICAgIHRoaXMuY3VycmVudFN0ZG91dCArPSB0eHQ7XG4gICAgICAgIGlmICh0eHQuZW5kc1dpdGgoXCJcXG5cIikpIHtcbiAgICAgICAgICAgIHRoaXMuc3Rkb3V0TGluZXMucHVzaCh0aGlzLmN1cnJlbnRTdGRvdXQuc2xpY2UoMCwgLTEpKTtcbiAgICAgICAgICAgIHRoaXMuY3VycmVudFN0ZG91dCA9IFwiXCI7XG4gICAgICAgIH1cbiAgICB9XG5cbiAgICBmbHVzaCAoKTogdm9pZCB7XG4gICAgICAgIGlmICh0aGlzLmN1cnJlbnRTdGRvdXQubGVuZ3RoICE9IDApIHtcbiAgICAgICAgICAgIHRoaXMuc3Rkb3V0KFwiXFxuXCIpO1xuICAgICAgICB9XG4gICAgICAgIGlmICh0aGlzLmN1cnJlbnRTdGRlcnIubGVuZ3RoICE9IDApIHtcbiAgICAgICAgICAgIHRoaXMuc3RkZXJyKFwiXFxuXCIpO1xuICAgICAgICB9XG4gICAgfVxufVxuIl19