gitlab-ci-local 4.53.0 → 4.55.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/utils.js CHANGED
@@ -1,65 +1,36 @@
1
- "use strict";
2
- var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
- if (k2 === undefined) k2 = k;
4
- var desc = Object.getOwnPropertyDescriptor(m, k);
5
- if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
- desc = { enumerable: true, get: function() { return m[k]; } };
7
- }
8
- Object.defineProperty(o, k2, desc);
9
- }) : (function(o, m, k, k2) {
10
- if (k2 === undefined) k2 = k;
11
- o[k2] = m[k];
12
- }));
13
- var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
- Object.defineProperty(o, "default", { enumerable: true, value: v });
15
- }) : function(o, v) {
16
- o["default"] = v;
17
- });
18
- var __importStar = (this && this.__importStar) || function (mod) {
19
- if (mod && mod.__esModule) return mod;
20
- var result = {};
21
- if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
22
- __setModuleDefault(result, mod);
23
- return result;
24
- };
25
- var __importDefault = (this && this.__importDefault) || function (mod) {
26
- return (mod && mod.__esModule) ? mod : { "default": mod };
27
- };
28
- Object.defineProperty(exports, "__esModule", { value: true });
29
- exports.Utils = void 0;
30
- const chalk_1 = __importDefault(require("chalk"));
31
- const job_1 = require("./job");
32
- const fs = __importStar(require("fs-extra"));
33
- const checksum_1 = __importDefault(require("checksum"));
34
- const base64url_1 = __importDefault(require("base64url"));
35
- const execa_1 = __importDefault(require("execa"));
36
- const assert_1 = __importDefault(require("assert"));
37
- const git_data_1 = require("./git-data");
38
- const globby_1 = __importDefault(require("globby"));
39
- const micromatch_1 = __importDefault(require("micromatch"));
40
- const axios_1 = __importDefault(require("axios"));
41
- const path_1 = __importDefault(require("path"));
42
- class Utils {
1
+ import chalk from "chalk";
2
+ import { Job } from "./job.js";
3
+ import fs from "fs-extra";
4
+ import checksum from "checksum";
5
+ import base64url from "base64url";
6
+ import execa from "execa";
7
+ import assert from "assert";
8
+ import { GitData } from "./git-data.js";
9
+ import globby from "globby";
10
+ import micromatch from "micromatch";
11
+ import axios from "axios";
12
+ import path from "path";
13
+ export class Utils {
43
14
  static bash(shellScript, cwd = process.cwd()) {
44
- return (0, execa_1.default)(shellScript, { shell: "bash", cwd });
15
+ return execa(shellScript, { shell: "bash", cwd });
45
16
  }
46
17
  static spawn(cmdArgs, cwd = process.cwd()) {
47
- return (0, execa_1.default)(cmdArgs[0], cmdArgs.slice(1), { cwd });
18
+ return execa(cmdArgs[0], cmdArgs.slice(1), { cwd });
48
19
  }
49
20
  static syncSpawn(cmdArgs, cwd = process.cwd()) {
50
- return execa_1.default.sync(cmdArgs[0], cmdArgs.slice(1), { cwd });
21
+ return execa.sync(cmdArgs[0], cmdArgs.slice(1), { cwd });
51
22
  }
52
23
  static fsUrl(url) {
53
24
  return url.replace(/^https:\/\//g, "").replace(/^http:\/\//g, "");
54
25
  }
55
26
  static safeDockerString(jobName) {
56
27
  return jobName.replace(/[^\w-]+/g, (match) => {
57
- return base64url_1.default.encode(match);
28
+ return base64url.encode(match);
58
29
  });
59
30
  }
60
31
  static forEachRealJob(gitlabData, callback) {
61
32
  for (const [jobName, jobData] of Object.entries(gitlabData)) {
62
- if (job_1.Job.illegalJobNames.has(jobName) || jobName[0].startsWith(".")) {
33
+ if (Job.illegalJobNames.has(jobName) || jobName[0].startsWith(".")) {
63
34
  continue;
64
35
  }
65
36
  callback(jobName, jobData);
@@ -90,10 +61,10 @@ class Utils {
90
61
  }
91
62
  static printJobNames(stream, job, i, arr) {
92
63
  if (i === arr.length - 1) {
93
- stream((0, chalk_1.default) `{blueBright ${job.name}}`);
64
+ stream(chalk `{blueBright ${job.name}}`);
94
65
  }
95
66
  else {
96
- stream((0, chalk_1.default) `{blueBright ${job.name}}, `);
67
+ stream(chalk `{blueBright ${job.name}}, `);
97
68
  }
98
69
  }
99
70
  static expandTextWith(text, expandWith) {
@@ -107,12 +78,8 @@ class Utils {
107
78
  }
108
79
  else {
109
80
  const name = var1 || var2;
110
- (0, assert_1.default)(name, "unexpected unset capture group");
111
- let value = `${expandWith.variable(name)}`;
112
- if (value.startsWith("\"/") && value.endsWith("/\"")) {
113
- value = value.substring(1).slice(0, -1);
114
- }
115
- return `${value}`;
81
+ assert(name, "unexpected unset capture group");
82
+ return `${expandWith.variable(name)}`;
116
83
  }
117
84
  });
118
85
  }
@@ -125,7 +92,7 @@ class Utils {
125
92
  static expandVariables(variables) {
126
93
  let expandedAnyVariables, i = 0;
127
94
  do {
128
- (0, assert_1.default)(i < 100, "Recursive variable expansion reached 100 iterations");
95
+ assert(i < 100, "Recursive variable expansion reached 100 iterations");
129
96
  expandedAnyVariables = false;
130
97
  for (const [k, v] of Object.entries(variables)) {
131
98
  const envsWithoutSelf = { ...variables };
@@ -203,22 +170,77 @@ class Utils {
203
170
  unescape: JSON.stringify("$"),
204
171
  variable: (name) => JSON.stringify(envs[name] ?? null).replaceAll("\\\\", "\\"),
205
172
  });
206
- // Convert =~ to match function
207
- evalStr = evalStr.replace(/\s*=~\s*(\/.*?\/[igmsuy]*)(?:\s|$)/g, ".match($1) != null");
208
- evalStr = evalStr.replace(/\s*=~\s(.+?)(\)*?)(?:\s|$)/g, ".match(new RegExp($1)) != null$2"); // Without forward slashes
209
- // Convert !~ to match function
210
- evalStr = evalStr.replace(/\s*!~\s*(\/.*?\/[igmsuy]*)(?:\s|$)/g, ".match($1) == null");
211
- evalStr = evalStr.replace(/\s*!~\s(.+?)(\)*?)(?:\s|$)/g, ".match(new RegExp($1)) == null$2"); // Without forward slashes
173
+ const expandedEvalStr = evalStr;
174
+ // Scenario when RHS is a <regex>
175
+ // https://regexr.com/85sjo
176
+ const pattern1 = /\s*(?<operator>(?:=~)|(?:!~))\s*(?<rhs>\/.*?\/)(?<flags>[igmsuy]*)(\s|$|\))/g;
177
+ evalStr = evalStr.replace(pattern1, (_, operator, rhs, flags, remainingTokens) => {
178
+ let _operator;
179
+ switch (operator) {
180
+ case "=~":
181
+ _operator = "!=";
182
+ break;
183
+ case "!~":
184
+ _operator = "==";
185
+ break;
186
+ default:
187
+ throw operator;
188
+ }
189
+ return `.match(${rhs}${flags})${remainingTokens} ${_operator} null `;
190
+ });
191
+ // Scenario when RHS is surrounded by double-quotes
192
+ // https://regexr.com/85t0g
193
+ const pattern2 = /\s*(?<operator>(?:=~)|(?:!~))\s*"(?<rhs>[^"\\]*(?:\\.[^"\\]*)*)"/g;
194
+ evalStr = evalStr.replace(pattern2, (_, operator, rhs) => {
195
+ let _operator;
196
+ switch (operator) {
197
+ case "=~":
198
+ _operator = "!=";
199
+ break;
200
+ case "!~":
201
+ _operator = "==";
202
+ break;
203
+ default:
204
+ throw operator;
205
+ }
206
+ if (!/\/(.*)\/([\w]*)/.test(rhs)) {
207
+ throw Error(`RHS (${rhs}) must be a regex pattern. Do not rely on this behavior!
208
+ Refer to https://docs.gitlab.com/ee/ci/jobs/job_rules.html#unexpected-behavior-from-regular-expression-matching-with- for more info...`);
209
+ }
210
+ const regex = /\/(?<pattern>.*)\/(?<flags>[igmsuy]*)/;
211
+ const _rhs = rhs.replace(regex, (_, pattern, flags) => {
212
+ const _pattern = pattern.replace(/(?<!\\)\//g, "\\/"); // escape potentially unescaped `/` that's in the pattern
213
+ return `/${_pattern}/${flags}`;
214
+ });
215
+ return `.match(new RegExp(${_rhs})) ${_operator} null`;
216
+ });
212
217
  // Convert all null.match functions to false
213
- evalStr = evalStr.replace(/null.match\(.+?\) != null/g, "false");
214
- evalStr = evalStr.replace(/null.match\(.+?\) == null/g, "false");
215
- return Boolean(eval(evalStr));
218
+ evalStr = evalStr.replace(/null.match\(.+?\)\s*!=\s*null/g, "false");
219
+ evalStr = evalStr.replace(/null.match\(.+?\)\s*==\s*null/g, "false");
220
+ evalStr = evalStr.trim();
221
+ let res;
222
+ try {
223
+ res = (0, eval)(evalStr); // https://esbuild.github.io/content-types/#direct-eval
224
+ }
225
+ catch (err) {
226
+ console.log(`
227
+ Error attempting to evaluate the following rules:
228
+ rules:
229
+ - if: '${expandedEvalStr}'
230
+ as
231
+ \`\`\`javascript
232
+ ${evalStr}
233
+ \`\`\`
234
+ `);
235
+ throw err;
236
+ }
237
+ return Boolean(res);
216
238
  }
217
239
  static evaluateRuleExist(cwd, ruleExists) {
218
240
  if (ruleExists === undefined)
219
241
  return true;
220
242
  for (const pattern of ruleExists) {
221
- if (globby_1.default.sync(pattern, { dot: true, cwd }).length > 0) {
243
+ if (globby.sync(pattern, { dot: true, cwd }).length > 0) {
222
244
  return true;
223
245
  }
224
246
  }
@@ -233,7 +255,7 @@ class Utils {
233
255
  // NOTE: https://docs.gitlab.com/ee/ci/yaml/#ruleschanges
234
256
  // Glob patterns are interpreted with Ruby's [File.fnmatch](https://docs.ruby-lang.org/en/master/File.html#method-c-fnmatch)
235
257
  // with the flags File::FNM_PATHNAME | File::FNM_DOTMATCH | File::FNM_EXTGLOB.
236
- return micromatch_1.default.some(git_data_1.GitData.changedFiles(`origin/${defaultBranch}`), ruleChanges, {
258
+ return micromatch.some(GitData.changedFiles(`origin/${defaultBranch}`), ruleChanges, {
237
259
  nonegate: true,
238
260
  noextglob: true,
239
261
  posix: false,
@@ -251,8 +273,8 @@ class Utils {
251
273
  files.forEach((file) => {
252
274
  promises.push(new Promise((resolve, reject) => {
253
275
  if (!fs.pathExistsSync(file))
254
- resolve(path_1.default.relative(cwd, file)); // must use relative path here, so that checksum can be deterministic when running the unit tests
255
- checksum_1.default.file(file, (err, hash) => {
276
+ resolve(path.relative(cwd, file)); // must use relative path here, so that checksum can be deterministic when running the unit tests
277
+ checksum.file(file, (err, hash) => {
256
278
  if (err) {
257
279
  return reject(err);
258
280
  }
@@ -261,7 +283,7 @@ class Utils {
261
283
  }));
262
284
  });
263
285
  const result = await Promise.all(promises);
264
- return (0, checksum_1.default)(result.join(""));
286
+ return checksum(result.join(""));
265
287
  }
266
288
  static isObject(v) {
267
289
  return Object.getPrototypeOf(v) === Object.prototype;
@@ -282,7 +304,7 @@ class Utils {
282
304
  case "http":
283
305
  case "https": {
284
306
  try {
285
- const { status } = await axios_1.default.get(`${protocol}://${domain}/${projectPath}/-/raw/${ref}/${file}`);
307
+ const { status } = await axios.get(`${protocol}://${domain}:${port}/${projectPath}/-/raw/${ref}/${file}`);
286
308
  return (status === 200);
287
309
  }
288
310
  catch (e) {
@@ -302,5 +324,4 @@ class Utils {
302
324
  throw new Error(`Unhandled case ${param}`);
303
325
  }
304
326
  }
305
- exports.Utils = Utils;
306
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["utils.ts"],"names":[],"mappings":"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","sourcesContent":["import chalk from \"chalk\";\nimport {Job, JobRule} from \"./job\";\nimport * as fs from \"fs-extra\";\nimport checksum from \"checksum\";\nimport base64url from \"base64url\";\nimport execa from \"execa\";\nimport assert from \"assert\";\nimport {CICDVariable} from \"./variables-from-files\";\nimport {GitData, GitSchema} from \"./git-data\";\nimport globby from \"globby\";\nimport micromatch from \"micromatch\";\nimport axios from \"axios\";\nimport path from \"path\";\n\ntype RuleResultOpt = {\n    cwd: string;\n    rules: JobRule[];\n    variables: {[key: string]: string};\n};\n\ntype ExpandWith = {\n    unescape: string;\n    variable: (name: string) => string;\n};\n\nexport class Utils {\n    static bash (shellScript: string, cwd = process.cwd()): Promise<{stdout: string; stderr: string; exitCode: number}> {\n        return execa(shellScript, {shell: \"bash\", cwd});\n    }\n\n    static spawn (cmdArgs: string[], cwd = process.cwd()): Promise<{stdout: string; stderr: string}> {\n        return execa(cmdArgs[0], cmdArgs.slice(1), {cwd});\n    }\n\n    static syncSpawn (cmdArgs: string[], cwd = process.cwd()): {stdout: string; stderr: string} {\n        return execa.sync(cmdArgs[0], cmdArgs.slice(1), {cwd});\n    }\n\n    static fsUrl (url: string): string {\n        return url.replace(/^https:\\/\\//g, \"\").replace(/^http:\\/\\//g, \"\");\n    }\n\n    static safeDockerString (jobName: string) {\n        return jobName.replace(/[^\\w-]+/g, (match) => {\n            return base64url.encode(match);\n        });\n    }\n\n    static forEachRealJob (gitlabData: any, callback: (jobName: string, jobData: any) => void) {\n        for (const [jobName, jobData] of Object.entries<any>(gitlabData)) {\n            if (Job.illegalJobNames.has(jobName) || jobName[0].startsWith(\".\")) {\n                continue;\n            }\n            callback(jobName, jobData);\n        }\n    }\n\n    static getJobNamesFromPreviousStages (jobs: ReadonlyArray<Job>, stages: readonly string[], currentJob: Job) {\n        const jobNames: string[] = [];\n        const currentStageIndex = stages.indexOf(currentJob.stage);\n        jobs.forEach(job => {\n            const stageIndex = stages.indexOf(job.stage);\n            if (stageIndex < currentStageIndex) {\n                jobNames.push(job.name);\n            }\n        });\n        return jobNames;\n    }\n\n    static async getCoveragePercent (cwd: string, stateDir: string, coverageRegex: string, jobName: string) {\n        const content = await fs.readFile(`${cwd}/${stateDir}/output/${jobName}.log`, \"utf8\");\n\n        const regex = new RegExp(coverageRegex.replace(/^\\//, \"\").replace(/\\/$/, \"\"), \"gm\");\n        const matches = Array.from(content.matchAll(regex));\n        if (matches.length === 0) return \"0\";\n\n        const lastMatch = matches[matches.length - 1];\n        const digits = /\\d+(?:\\.\\d+)?/.exec(lastMatch[1] ?? lastMatch[0]);\n        if (!digits) return \"0\";\n        return digits[0] ?? \"0\";\n    }\n\n    static printJobNames (stream: (txt: string) => void, job: {name: string}, i: number, arr: {name: string}[]) {\n        if (i === arr.length - 1) {\n            stream(chalk`{blueBright ${job.name}}`);\n        } else {\n            stream(chalk`{blueBright ${job.name}}, `);\n        }\n    }\n\n    private static expandTextWith (text: any, expandWith: ExpandWith) {\n        if (typeof text !== \"string\") {\n            return text;\n        }\n\n        return text.replace(\n            /(\\$\\$)|\\$\\{([a-zA-Z_]\\w*)}|\\$([a-zA-Z_]\\w*)/g, // https://regexr.com/7s4ka\n            (_match, escape, var1, var2) => {\n                if (typeof escape !== \"undefined\") {\n                    return expandWith.unescape;\n                } else {\n                    const name = var1 || var2;\n                    assert(name, \"unexpected unset capture group\");\n                    let value = `${expandWith.variable(name)}`;\n                    if (value.startsWith(\"\\\"/\") && value.endsWith(\"/\\\"\")) {\n                        value = value.substring(1).slice(0, -1);\n                    }\n                    return `${value}`;\n                }\n            }\n        );\n    }\n\n    static expandText (text: any, envs: {[key: string]: string}) {\n        return this.expandTextWith(text, {\n            unescape: \"$\",\n            variable: (name) => envs[name] ?? \"\",\n        });\n    }\n\n    static expandVariables (variables: {[key: string]: string}) {\n        let expandedAnyVariables, i = 0;\n        do {\n            assert(i < 100, \"Recursive variable expansion reached 100 iterations\");\n            expandedAnyVariables = false;\n            for (const [k, v] of Object.entries(variables)) {\n                const envsWithoutSelf = {...variables};\n                delete envsWithoutSelf[k];\n                // If the $$'s are converted to single $'s now, then the next\n                // iteration, they might be interpreted as variables, even\n                // though they were *explicitly* escaped. To work around this,\n                // leave the '$$'s as the same value, then only unescape them at\n                // the very end.\n                variables[k] = Utils.expandTextWith(v, {\n                    unescape: \"$$\",\n                    variable: (name) => envsWithoutSelf[name] ?? \"\",\n                });\n                expandedAnyVariables ||= variables[k] !== v;\n            }\n            i++;\n        } while (expandedAnyVariables);\n\n        return variables;\n    }\n\n    static unscape$$Variables (variables: {[key: string]: string}) {\n        for (const [k, v] of Object.entries(variables)) {\n            variables[k] = Utils.expandText(v, {});\n        }\n\n        return variables;\n    }\n\n    static findEnvMatchedVariables (variables: {[name: string]: CICDVariable}, fileVariablesDir?: string, environment?: {name: string}) {\n        const envMatchedVariables: {[key: string]: string} = {};\n        for (const [k, v] of Object.entries(variables)) {\n            for (const entry of v.environments) {\n                if (environment?.name.match(entry.regexp) || entry.regexp.source === \".*\") {\n                    if (fileVariablesDir != null && v.type === \"file\" && !entry.fileSource) {\n                        envMatchedVariables[k] = `${fileVariablesDir}/${k}`;\n                        fs.mkdirpSync(`${fileVariablesDir}`);\n                        fs.writeFileSync(`${fileVariablesDir}/${k}`, entry.content);\n                    } else if (fileVariablesDir != null && v.type === \"file\" && entry.fileSource) {\n                        envMatchedVariables[k] = `${fileVariablesDir}/${k}`;\n                        fs.mkdirpSync(`${fileVariablesDir}`);\n                        fs.copyFileSync(entry.fileSource, `${fileVariablesDir}/${k}`);\n                    } else {\n                        envMatchedVariables[k] = entry.content;\n                    }\n                    break;\n                }\n            }\n        }\n        return envMatchedVariables;\n    }\n\n    static getRulesResult (opt: RuleResultOpt, gitData: GitData, jobWhen: string = \"on_success\", jobAllowFailure: boolean | {exit_codes: number | number[]} = false): {when: string; allowFailure: boolean | {exit_codes: number | number[]}; variables?: {[name: string]: string}} {\n        let when = \"never\";\n\n        // optional manual jobs allowFailure defaults to true https://docs.gitlab.com/ee/ci/jobs/job_control.html#types-of-manual-jobs\n        let allowFailure = jobWhen === \"manual\" ? true : jobAllowFailure;\n        let ruleVariable: {[name: string]: string} | undefined;\n\n        for (const rule of opt.rules) {\n            if (!Utils.evaluateRuleIf(rule.if, opt.variables)) continue;\n            if (!Utils.evaluateRuleExist(opt.cwd, rule.exists)) continue;\n            if (!Utils.evaluateRuleChanges(gitData.branches.default, rule.changes)) continue;\n\n            when = rule.when ? rule.when : jobWhen;\n            allowFailure = rule.allow_failure ?? allowFailure;\n            ruleVariable = rule.variables;\n\n            break; // Early return, will not evaluate the remaining rules\n        }\n\n        return {when, allowFailure, variables: ruleVariable};\n    }\n\n    static evaluateRuleIf (ruleIf: string | undefined, envs: {[key: string]: string}): boolean {\n        if (ruleIf === undefined) return true;\n        let evalStr = ruleIf;\n\n        // Expand all variables\n        evalStr = this.expandTextWith(evalStr, {\n            unescape: JSON.stringify(\"$\"),\n            variable: (name) => JSON.stringify(envs[name] ?? null).replaceAll(\"\\\\\\\\\", \"\\\\\"),\n        });\n\n        // Convert =~ to match function\n        evalStr = evalStr.replace(/\\s*=~\\s*(\\/.*?\\/[igmsuy]*)(?:\\s|$)/g, \".match($1) != null\");\n        evalStr = evalStr.replace(/\\s*=~\\s(.+?)(\\)*?)(?:\\s|$)/g, \".match(new RegExp($1)) != null$2\"); // Without forward slashes\n\n        // Convert !~ to match function\n        evalStr = evalStr.replace(/\\s*!~\\s*(\\/.*?\\/[igmsuy]*)(?:\\s|$)/g, \".match($1) == null\");\n        evalStr = evalStr.replace(/\\s*!~\\s(.+?)(\\)*?)(?:\\s|$)/g, \".match(new RegExp($1)) == null$2\"); // Without forward slashes\n\n        // Convert all null.match functions to false\n        evalStr = evalStr.replace(/null.match\\(.+?\\) != null/g, \"false\");\n        evalStr = evalStr.replace(/null.match\\(.+?\\) == null/g, \"false\");\n\n        return Boolean(eval(evalStr));\n    }\n\n    static evaluateRuleExist (cwd: string, ruleExists: string[] | undefined): boolean {\n        if (ruleExists === undefined) return true;\n        for (const pattern of ruleExists) {\n            if (globby.sync(pattern, {dot: true, cwd}).length > 0) {\n                return true;\n            }\n        }\n        return false;\n    }\n\n    static evaluateRuleChanges (defaultBranch: string, ruleChanges: string[] | {paths: string[]} | undefined): boolean {\n        if (ruleChanges === undefined) return true;\n\n        // Normalize rules:changes:paths to rules:changes\n        if (!Array.isArray(ruleChanges)) ruleChanges = ruleChanges.paths;\n\n        // NOTE: https://docs.gitlab.com/ee/ci/yaml/#ruleschanges\n        //       Glob patterns are interpreted with Ruby's [File.fnmatch](https://docs.ruby-lang.org/en/master/File.html#method-c-fnmatch)\n        //       with the flags File::FNM_PATHNAME | File::FNM_DOTMATCH | File::FNM_EXTGLOB.\n        return micromatch.some(GitData.changedFiles(`origin/${defaultBranch}`), ruleChanges, {\n            nonegate: true,\n            noextglob: true,\n            posix: false,\n            dot: true,\n        });\n    }\n\n    static async rsyncTrackedFiles (cwd: string, stateDir: string, target: string): Promise<{hrdeltatime: [number, number]}> {\n        const time = process.hrtime();\n        await fs.mkdirp(`${cwd}/${stateDir}/builds/${target}`);\n        await Utils.bash(`rsync -a --delete-excluded --delete --exclude-from=<(git ls-files -o --directory | awk '{print \"/\"$0}') --exclude ${stateDir}/ ./ ${stateDir}/builds/${target}/`, cwd);\n        return {hrdeltatime: process.hrtime(time)};\n    }\n\n    static async checksumFiles (cwd: string, files: string[]): Promise<string> {\n        const promises: Promise<string>[] = [];\n\n        files.forEach((file) => {\n            promises.push(new Promise((resolve, reject) => {\n                if (! fs.pathExistsSync(file)) resolve(path.relative(cwd, file)); // must use relative path here, so that checksum can be deterministic when running the unit tests\n                checksum.file(file, (err, hash) => {\n                    if (err) {\n                        return reject(err);\n                    }\n                    resolve(hash);\n                });\n            }));\n        });\n\n        const result = await Promise.all(promises);\n        return checksum(result.join(\"\"));\n    }\n\n    static isObject (v: any) {\n        return Object.getPrototypeOf(v) === Object.prototype;\n    }\n\n    static async remoteFileExist (cwd: string, file: string, ref: string, domain: string, projectPath: string, protocol: GitSchema, port: string) {\n        switch (protocol) {\n            case \"ssh\":\n            case \"git\":\n                try {\n                    await Utils.spawn(`git archive --remote=ssh://git@${domain}:${port}/${projectPath}.git ${ref} ${file}`.split(\" \"), cwd);\n                    return true;\n                } catch (e: any) {\n                    if (!e.stderr.includes(`remote: fatal: pathspec '${file}' did not match any files`)) throw new Error(e);\n                    return false;\n                }\n\n            case \"http\":\n            case \"https\": {\n                try {\n                    const {status} = await axios.get(`${protocol}://${domain}/${projectPath}/-/raw/${ref}/${file}`);\n                    return (status === 200);\n                } catch (e) {\n                    return false;\n                }\n            }\n            default: {\n                Utils.switchStatementExhaustiveCheck(protocol);\n            }\n        }\n    }\n\n    static trimSuffix (str: string, suffix: string) {\n        return str.endsWith(suffix) ? str.slice(0, -suffix.length) : str;\n    }\n\n    static switchStatementExhaustiveCheck (param: never): never {\n        // https://dev.to/babak/exhaustive-type-checking-with-typescript-4l3f\n        throw new Error(`Unhandled case ${param}`);\n    }\n}\n"]}
327
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["utils.ts"],"names":[],"mappings":"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","sourcesContent":["import chalk from \"chalk\";\nimport {Job, JobRule} from \"./job.js\";\nimport fs from \"fs-extra\";\nimport checksum from \"checksum\";\nimport base64url from \"base64url\";\nimport execa from \"execa\";\nimport assert from \"assert\";\nimport {CICDVariable} from \"./variables-from-files.js\";\nimport {GitData, GitSchema} from \"./git-data.js\";\nimport globby from \"globby\";\nimport micromatch from \"micromatch\";\nimport axios from \"axios\";\nimport path from \"path\";\n\ntype RuleResultOpt = {\n    cwd: string;\n    rules: JobRule[];\n    variables: {[key: string]: string};\n};\n\ntype ExpandWith = {\n    unescape: string;\n    variable: (name: string) => string;\n};\n\nexport class Utils {\n    static bash (shellScript: string, cwd = process.cwd()): Promise<{stdout: string; stderr: string; exitCode: number}> {\n        return execa(shellScript, {shell: \"bash\", cwd});\n    }\n\n    static spawn (cmdArgs: string[], cwd = process.cwd()): Promise<{stdout: string; stderr: string}> {\n        return execa(cmdArgs[0], cmdArgs.slice(1), {cwd});\n    }\n\n    static syncSpawn (cmdArgs: string[], cwd = process.cwd()): {stdout: string; stderr: string} {\n        return execa.sync(cmdArgs[0], cmdArgs.slice(1), {cwd});\n    }\n\n    static fsUrl (url: string): string {\n        return url.replace(/^https:\\/\\//g, \"\").replace(/^http:\\/\\//g, \"\");\n    }\n\n    static safeDockerString (jobName: string) {\n        return jobName.replace(/[^\\w-]+/g, (match) => {\n            return base64url.encode(match);\n        });\n    }\n\n    static forEachRealJob (gitlabData: any, callback: (jobName: string, jobData: any) => void) {\n        for (const [jobName, jobData] of Object.entries<any>(gitlabData)) {\n            if (Job.illegalJobNames.has(jobName) || jobName[0].startsWith(\".\")) {\n                continue;\n            }\n            callback(jobName, jobData);\n        }\n    }\n\n    static getJobNamesFromPreviousStages (jobs: ReadonlyArray<Job>, stages: readonly string[], currentJob: Job) {\n        const jobNames: string[] = [];\n        const currentStageIndex = stages.indexOf(currentJob.stage);\n        jobs.forEach(job => {\n            const stageIndex = stages.indexOf(job.stage);\n            if (stageIndex < currentStageIndex) {\n                jobNames.push(job.name);\n            }\n        });\n        return jobNames;\n    }\n\n    static async getCoveragePercent (cwd: string, stateDir: string, coverageRegex: string, jobName: string) {\n        const content = await fs.readFile(`${cwd}/${stateDir}/output/${jobName}.log`, \"utf8\");\n\n        const regex = new RegExp(coverageRegex.replace(/^\\//, \"\").replace(/\\/$/, \"\"), \"gm\");\n        const matches = Array.from(content.matchAll(regex));\n        if (matches.length === 0) return \"0\";\n\n        const lastMatch = matches[matches.length - 1];\n        const digits = /\\d+(?:\\.\\d+)?/.exec(lastMatch[1] ?? lastMatch[0]);\n        if (!digits) return \"0\";\n        return digits[0] ?? \"0\";\n    }\n\n    static printJobNames (stream: (txt: string) => void, job: {name: string}, i: number, arr: {name: string}[]) {\n        if (i === arr.length - 1) {\n            stream(chalk`{blueBright ${job.name}}`);\n        } else {\n            stream(chalk`{blueBright ${job.name}}, `);\n        }\n    }\n\n    private static expandTextWith (text: any, expandWith: ExpandWith) {\n        if (typeof text !== \"string\") {\n            return text;\n        }\n\n        return text.replace(\n            /(\\$\\$)|\\$\\{([a-zA-Z_]\\w*)}|\\$([a-zA-Z_]\\w*)/g, // https://regexr.com/7s4ka\n            (_match, escape, var1, var2) => {\n                if (typeof escape !== \"undefined\") {\n                    return expandWith.unescape;\n                } else {\n                    const name = var1 || var2;\n                    assert(name, \"unexpected unset capture group\");\n                    return `${expandWith.variable(name)}`;\n                }\n            }\n        );\n    }\n\n    static expandText (text: any, envs: {[key: string]: string}) {\n        return this.expandTextWith(text, {\n            unescape: \"$\",\n            variable: (name) => envs[name] ?? \"\",\n        });\n    }\n\n    static expandVariables (variables: {[key: string]: string}) {\n        let expandedAnyVariables, i = 0;\n        do {\n            assert(i < 100, \"Recursive variable expansion reached 100 iterations\");\n            expandedAnyVariables = false;\n            for (const [k, v] of Object.entries(variables)) {\n                const envsWithoutSelf = {...variables};\n                delete envsWithoutSelf[k];\n                // If the $$'s are converted to single $'s now, then the next\n                // iteration, they might be interpreted as variables, even\n                // though they were *explicitly* escaped. To work around this,\n                // leave the '$$'s as the same value, then only unescape them at\n                // the very end.\n                variables[k] = Utils.expandTextWith(v, {\n                    unescape: \"$$\",\n                    variable: (name) => envsWithoutSelf[name] ?? \"\",\n                });\n                expandedAnyVariables ||= variables[k] !== v;\n            }\n            i++;\n        } while (expandedAnyVariables);\n\n        return variables;\n    }\n\n    static unscape$$Variables (variables: {[key: string]: string}) {\n        for (const [k, v] of Object.entries(variables)) {\n            variables[k] = Utils.expandText(v, {});\n        }\n\n        return variables;\n    }\n\n    static findEnvMatchedVariables (variables: {[name: string]: CICDVariable}, fileVariablesDir?: string, environment?: {name: string}) {\n        const envMatchedVariables: {[key: string]: string} = {};\n        for (const [k, v] of Object.entries(variables)) {\n            for (const entry of v.environments) {\n                if (environment?.name.match(entry.regexp) || entry.regexp.source === \".*\") {\n                    if (fileVariablesDir != null && v.type === \"file\" && !entry.fileSource) {\n                        envMatchedVariables[k] = `${fileVariablesDir}/${k}`;\n                        fs.mkdirpSync(`${fileVariablesDir}`);\n                        fs.writeFileSync(`${fileVariablesDir}/${k}`, entry.content);\n                    } else if (fileVariablesDir != null && v.type === \"file\" && entry.fileSource) {\n                        envMatchedVariables[k] = `${fileVariablesDir}/${k}`;\n                        fs.mkdirpSync(`${fileVariablesDir}`);\n                        fs.copyFileSync(entry.fileSource, `${fileVariablesDir}/${k}`);\n                    } else {\n                        envMatchedVariables[k] = entry.content;\n                    }\n                    break;\n                }\n            }\n        }\n        return envMatchedVariables;\n    }\n\n    static getRulesResult (opt: RuleResultOpt, gitData: GitData, jobWhen: string = \"on_success\", jobAllowFailure: boolean | {exit_codes: number | number[]} = false): {when: string; allowFailure: boolean | {exit_codes: number | number[]}; variables?: {[name: string]: string}} {\n        let when = \"never\";\n\n        // optional manual jobs allowFailure defaults to true https://docs.gitlab.com/ee/ci/jobs/job_control.html#types-of-manual-jobs\n        let allowFailure = jobWhen === \"manual\" ? true : jobAllowFailure;\n        let ruleVariable: {[name: string]: string} | undefined;\n\n        for (const rule of opt.rules) {\n            if (!Utils.evaluateRuleIf(rule.if, opt.variables)) continue;\n            if (!Utils.evaluateRuleExist(opt.cwd, rule.exists)) continue;\n            if (!Utils.evaluateRuleChanges(gitData.branches.default, rule.changes)) continue;\n\n            when = rule.when ? rule.when : jobWhen;\n            allowFailure = rule.allow_failure ?? allowFailure;\n            ruleVariable = rule.variables;\n\n            break; // Early return, will not evaluate the remaining rules\n        }\n\n        return {when, allowFailure, variables: ruleVariable};\n    }\n\n    static evaluateRuleIf (ruleIf: string | undefined, envs: {[key: string]: string}): boolean {\n        if (ruleIf === undefined) return true;\n        let evalStr = ruleIf;\n\n        // Expand all variables\n        evalStr = this.expandTextWith(evalStr, {\n            unescape: JSON.stringify(\"$\"),\n            variable: (name) => JSON.stringify(envs[name] ?? null).replaceAll(\"\\\\\\\\\", \"\\\\\"),\n        });\n        const expandedEvalStr = evalStr;\n\n        // Scenario when RHS is a <regex>\n        // https://regexr.com/85sjo\n        const pattern1 = /\\s*(?<operator>(?:=~)|(?:!~))\\s*(?<rhs>\\/.*?\\/)(?<flags>[igmsuy]*)(\\s|$|\\))/g;\n        evalStr = evalStr.replace(pattern1, (_, operator, rhs, flags, remainingTokens) => {\n            let _operator;\n            switch (operator) {\n                case \"=~\":\n                    _operator = \"!=\";\n                    break;\n                case \"!~\":\n                    _operator = \"==\";\n                    break;\n                default:\n                    throw operator;\n            }\n            return `.match(${rhs}${flags})${remainingTokens} ${_operator} null `;\n        });\n\n        // Scenario when RHS is surrounded by double-quotes\n        // https://regexr.com/85t0g\n        const pattern2 = /\\s*(?<operator>(?:=~)|(?:!~))\\s*\"(?<rhs>[^\"\\\\]*(?:\\\\.[^\"\\\\]*)*)\"/g;\n        evalStr = evalStr.replace(pattern2, (_, operator, rhs) => {\n            let _operator;\n            switch (operator) {\n                case \"=~\":\n                    _operator = \"!=\";\n                    break;\n                case \"!~\":\n                    _operator = \"==\";\n                    break;\n                default:\n                    throw operator;\n            }\n\n            if (!/\\/(.*)\\/([\\w]*)/.test(rhs)) {\n                throw Error(`RHS (${rhs}) must be a regex pattern. Do not rely on this behavior!\nRefer to https://docs.gitlab.com/ee/ci/jobs/job_rules.html#unexpected-behavior-from-regular-expression-matching-with- for more info...`);\n            }\n            const regex = /\\/(?<pattern>.*)\\/(?<flags>[igmsuy]*)/;\n            const _rhs = rhs.replace(regex, (_: string, pattern: string, flags: string) => {\n                const _pattern = pattern.replace(/(?<!\\\\)\\//g, \"\\\\/\"); // escape potentially unescaped `/` that's in the pattern\n                return `/${_pattern}/${flags}`;\n            });\n            return `.match(new RegExp(${_rhs})) ${_operator} null`;\n        });\n\n        // Convert all null.match functions to false\n        evalStr = evalStr.replace(/null.match\\(.+?\\)\\s*!=\\s*null/g, \"false\");\n        evalStr = evalStr.replace(/null.match\\(.+?\\)\\s*==\\s*null/g, \"false\");\n\n        evalStr = evalStr.trim();\n\n        let res;\n        try {\n            res = (0, eval)(evalStr); // https://esbuild.github.io/content-types/#direct-eval\n        } catch (err) {\n            console.log(`\nError attempting to evaluate the following rules:\n  rules:\n    - if: '${expandedEvalStr}'\nas\n\\`\\`\\`javascript\n${evalStr}\n\\`\\`\\`\n`);\n            throw err;\n        }\n        return Boolean(res);\n    }\n\n    static evaluateRuleExist (cwd: string, ruleExists: string[] | undefined): boolean {\n        if (ruleExists === undefined) return true;\n        for (const pattern of ruleExists) {\n            if (globby.sync(pattern, {dot: true, cwd}).length > 0) {\n                return true;\n            }\n        }\n        return false;\n    }\n\n    static evaluateRuleChanges (defaultBranch: string, ruleChanges: string[] | {paths: string[]} | undefined): boolean {\n        if (ruleChanges === undefined) return true;\n\n        // Normalize rules:changes:paths to rules:changes\n        if (!Array.isArray(ruleChanges)) ruleChanges = ruleChanges.paths;\n\n        // NOTE: https://docs.gitlab.com/ee/ci/yaml/#ruleschanges\n        //       Glob patterns are interpreted with Ruby's [File.fnmatch](https://docs.ruby-lang.org/en/master/File.html#method-c-fnmatch)\n        //       with the flags File::FNM_PATHNAME | File::FNM_DOTMATCH | File::FNM_EXTGLOB.\n        return micromatch.some(GitData.changedFiles(`origin/${defaultBranch}`), ruleChanges, {\n            nonegate: true,\n            noextglob: true,\n            posix: false,\n            dot: true,\n        });\n    }\n\n    static async rsyncTrackedFiles (cwd: string, stateDir: string, target: string): Promise<{hrdeltatime: [number, number]}> {\n        const time = process.hrtime();\n        await fs.mkdirp(`${cwd}/${stateDir}/builds/${target}`);\n        await Utils.bash(`rsync -a --delete-excluded --delete --exclude-from=<(git ls-files -o --directory | awk '{print \"/\"$0}') --exclude ${stateDir}/ ./ ${stateDir}/builds/${target}/`, cwd);\n        return {hrdeltatime: process.hrtime(time)};\n    }\n\n    static async checksumFiles (cwd: string, files: string[]): Promise<string> {\n        const promises: Promise<string>[] = [];\n\n        files.forEach((file) => {\n            promises.push(new Promise((resolve, reject) => {\n                if (! fs.pathExistsSync(file)) resolve(path.relative(cwd, file)); // must use relative path here, so that checksum can be deterministic when running the unit tests\n                checksum.file(file, (err, hash) => {\n                    if (err) {\n                        return reject(err);\n                    }\n                    resolve(hash);\n                });\n            }));\n        });\n\n        const result = await Promise.all(promises);\n        return checksum(result.join(\"\"));\n    }\n\n    static isObject (v: any) {\n        return Object.getPrototypeOf(v) === Object.prototype;\n    }\n\n    static async remoteFileExist (cwd: string, file: string, ref: string, domain: string, projectPath: string, protocol: GitSchema, port: string) {\n        switch (protocol) {\n            case \"ssh\":\n            case \"git\":\n                try {\n                    await Utils.spawn(`git archive --remote=ssh://git@${domain}:${port}/${projectPath}.git ${ref} ${file}`.split(\" \"), cwd);\n                    return true;\n                } catch (e: any) {\n                    if (!e.stderr.includes(`remote: fatal: pathspec '${file}' did not match any files`)) throw new Error(e);\n                    return false;\n                }\n\n            case \"http\":\n            case \"https\": {\n                try {\n                    const {status} = await axios.get(`${protocol}://${domain}:${port}/${projectPath}/-/raw/${ref}/${file}`);\n                    return (status === 200);\n                } catch (e) {\n                    return false;\n                }\n            }\n            default: {\n                Utils.switchStatementExhaustiveCheck(protocol);\n            }\n        }\n    }\n\n    static trimSuffix (str: string, suffix: string) {\n        return str.endsWith(suffix) ? str.slice(0, -suffix.length) : str;\n    }\n\n    static switchStatementExhaustiveCheck (param: never): never {\n        // https://dev.to/babak/exhaustive-type-checking-with-typescript-4l3f\n        throw new Error(`Unhandled case ${param}`);\n    }\n}\n"]}
package/src/validator.js CHANGED
@@ -1,19 +1,13 @@
1
- "use strict";
2
- var __importDefault = (this && this.__importDefault) || function (mod) {
3
- return (mod && mod.__esModule) ? mod : { "default": mod };
4
- };
5
- Object.defineProperty(exports, "__esModule", { value: true });
6
- exports.Validator = void 0;
7
- const ajv_1 = __importDefault(require("ajv"));
8
- const assert_1 = __importDefault(require("assert"));
9
- const chalk_1 = __importDefault(require("chalk"));
10
- const schema_1 = __importDefault(require("./schema"));
11
- const schema_error_1 = require("./schema-error");
12
- const terminal_link_1 = __importDefault(require("terminal-link"));
1
+ import Ajv from "ajv";
2
+ import assert from "assert";
3
+ import chalk from "chalk";
4
+ import schema from "./schema/index.js";
5
+ import { betterAjvErrors } from "./schema-error.js";
6
+ import terminalLink from "terminal-link";
13
7
  const MAX_ERRORS = 5;
14
- class Validator {
8
+ export class Validator {
15
9
  static jsonSchemaValidation({ pathToExpandedGitLabCi, gitLabCiConfig }) {
16
- const ajv = new ajv_1.default({
10
+ const ajv = new Ajv({
17
11
  verbose: true,
18
12
  allErrors: true,
19
13
  allowUnionTypes: true,
@@ -21,10 +15,10 @@ class Validator {
21
15
  strictTypes: false, // to suppress the missing types defined in the gitlab-ci json schema
22
16
  keywords: ["markdownDescription"],
23
17
  });
24
- const validate = ajv.compile(schema_1.default);
18
+ const validate = ajv.compile(schema);
25
19
  const valid = validate(gitLabCiConfig);
26
20
  if (!valid) {
27
- const betterErrors = (0, schema_error_1.betterAjvErrors)({
21
+ const betterErrors = betterAjvErrors({
28
22
  data: gitLabCiConfig,
29
23
  errors: validate.errors,
30
24
  });
@@ -34,14 +28,14 @@ class Validator {
34
28
  e += `\t... and ${len - MAX_ERRORS} more`;
35
29
  break;
36
30
  }
37
- e += (0, chalk_1.default) `\t• {redBright ${betterErrors[i].message}} at {blueBright ${betterErrors[i].path}}\n`;
31
+ e += chalk `\t• {redBright ${betterErrors[i].message}} at {blueBright ${betterErrors[i].path}}\n`;
38
32
  }
39
- (0, assert_1.default)(valid, (0, chalk_1.default) `
33
+ assert(valid, chalk `
40
34
  {reset Invalid .gitlab-ci.yml configuration!
41
35
  ${e.trimEnd()}
42
36
 
43
37
  For further troubleshooting, consider either of the following:
44
- \t• Copy the content of {blueBright ${(0, terminal_link_1.default)(".gitlab-ci-local/expanded-gitlab-ci.yml", pathToExpandedGitLabCi)}} to the ${(0, terminal_link_1.default)("pipeline editor", "https://docs.gitlab.com/ee/ci/pipeline_editor/")} to debug it
38
+ \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
45
39
  \t• Use --json-schema-validation=false to disable schema validation (not recommended)}
46
40
  `);
47
41
  }
@@ -63,10 +57,10 @@ For further troubleshooting, consider either of the following:
63
57
  const needJob = jobs.find(j => j.baseName === need.job);
64
58
  if (need.optional && !needJob)
65
59
  continue;
66
- (0, assert_1.default)(needJob != null, (0, chalk_1.default) `needs: [{blueBright ${need.job}}] for {blueBright ${job.baseName}} could not be found`);
60
+ assert(needJob != null, chalk `needs: [{blueBright ${need.job}}] for {blueBright ${job.baseName}} could not be found`);
67
61
  const needJobStageIndex = stages.indexOf(needJob.stage);
68
62
  const jobStageIndex = stages.indexOf(job.stage);
69
- (0, assert_1.default)(needJobStageIndex <= jobStageIndex, (0, chalk_1.default) `needs: [{blueBright ${needJob.name}}] for {blueBright ${job.name}} is in a future stage`);
63
+ assert(needJobStageIndex <= jobStageIndex, chalk `needs: [{blueBright ${needJob.name}}] for {blueBright ${job.name}} is in a future stage`);
70
64
  }
71
65
  }
72
66
  return warnings;
@@ -76,13 +70,13 @@ For further troubleshooting, consider either of the following:
76
70
  if (job.dependencies === null || job.dependencies.length === 0)
77
71
  continue;
78
72
  const undefDeps = job.dependencies.filter((j) => !jobs.some(n => n.baseName === j));
79
- (0, assert_1.default)(undefDeps.length !== job.dependencies.length, (0, chalk_1.default) `dependencies: [{blueBright ${undefDeps.join(",")}}] for {blueBright ${job.name}} cannot be found`);
73
+ assert(undefDeps.length !== job.dependencies.length, chalk `dependencies: [{blueBright ${undefDeps.join(",")}}] for {blueBright ${job.name}} cannot be found`);
80
74
  for (const dep of job.dependencies) {
81
75
  const depJob = jobs.find(j => j.baseName === dep);
82
- (0, assert_1.default)(depJob != null, (0, chalk_1.default) `dependencies: [{blueBright ${dep}}] for {blueBright ${job.baseName}} could not be found`);
76
+ assert(depJob != null, chalk `dependencies: [{blueBright ${dep}}] for {blueBright ${job.baseName}} could not be found`);
83
77
  const depJobStageIndex = stages.indexOf(depJob.stage);
84
78
  const jobStageIndex = stages.indexOf(job.stage);
85
- (0, assert_1.default)(depJobStageIndex <= jobStageIndex, (0, chalk_1.default) `dependencies: [{blueBright ${depJob.name}}] for {blueBright ${job.name}} is in a future stage`);
79
+ assert(depJobStageIndex <= jobStageIndex, chalk `dependencies: [{blueBright ${depJob.name}}] for {blueBright ${job.name}} is in a future stage`);
86
80
  }
87
81
  }
88
82
  }
@@ -98,7 +92,7 @@ For further troubleshooting, consider either of the following:
98
92
  return needs.some(n => n.job === dep);
99
93
  });
100
94
  const assertMsg = `${job.formattedJobName} needs: '${needs.map(n => n.job).join(",")}' doesn't fully contain dependencies: '${dependencies.join(",")}'`;
101
- (0, assert_1.default)(everyIncluded, assertMsg);
95
+ assert(everyIncluded, assertMsg);
102
96
  }
103
97
  }
104
98
  /**
@@ -120,16 +114,16 @@ For further troubleshooting, consider either of the following:
120
114
  for (const job of jobs) {
121
115
  if (job.trigger)
122
116
  continue; // Jobs with trigger are allowed to have empty script
123
- (0, assert_1.default)(job.scripts.length > 0, (0, chalk_1.default) `{blue ${job.name}} has empty script`);
117
+ assert(job.scripts.length > 0, chalk `{blue ${job.name}} has empty script`);
124
118
  }
125
119
  }
126
120
  static arrayOfStrings(jobs) {
127
121
  for (const job of jobs) {
128
122
  if (job.trigger)
129
123
  continue;
130
- job.beforeScripts.forEach((s) => (0, assert_1.default)(typeof s === "string", (0, chalk_1.default) `{blue ${job.name}} before_script contains non string value`));
131
- job.afterScripts.forEach((s) => (0, assert_1.default)(typeof s === "string", (0, chalk_1.default) `{blue ${job.name}} after_script contains non string value`));
132
- job.scripts.forEach((s) => (0, assert_1.default)(typeof s === "string", (0, chalk_1.default) `{blue ${job.name}} script contains non string value`));
124
+ job.beforeScripts.forEach((s) => assert(typeof s === "string", chalk `{blue ${job.name}} before_script contains non string value`));
125
+ job.afterScripts.forEach((s) => assert(typeof s === "string", chalk `{blue ${job.name}} after_script contains non string value`));
126
+ job.scripts.forEach((s) => assert(typeof s === "string", chalk `{blue ${job.name}} script contains non string value`));
133
127
  }
134
128
  }
135
129
  static async run(jobs, stages) {
@@ -153,5 +147,4 @@ For further troubleshooting, consider either of the following:
153
147
  return warnings;
154
148
  }
155
149
  }
156
- exports.Validator = Validator;
157
- //# 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\";\nimport assert from \"assert\";\nimport chalk from \"chalk\";\nimport schema from \"./schema\";\nimport {betterAjvErrors} from \"./schema-error\";\nimport terminalLink from \"terminal-link\";\n\nconst MAX_ERRORS = 5;\n\nexport class Validator {\n    static jsonSchemaValidation ({pathToExpandedGitLabCi, gitLabCiConfig}: {\n        pathToExpandedGitLabCi: string;\n        gitLabCiConfig: object;\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) {\n            const betterErrors = betterAjvErrors({\n                data: gitLabCiConfig,\n                errors: validate.errors,\n            });\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}}\\n`;\n            }\n\n            assert(valid, 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 --json-schema-validation=false to disable schema validation (not recommended)}\n`);\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"]}
150
+ //# 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\";\nimport schema from \"./schema/index.js\";\nimport {betterAjvErrors} from \"./schema-error.js\";\nimport terminalLink from \"terminal-link\";\n\nconst MAX_ERRORS = 5;\n\nexport class Validator {\n    static jsonSchemaValidation ({pathToExpandedGitLabCi, gitLabCiConfig}: {\n        pathToExpandedGitLabCi: string;\n        gitLabCiConfig: object;\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) {\n            const betterErrors = betterAjvErrors({\n                data: gitLabCiConfig,\n                errors: validate.errors,\n            });\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}}\\n`;\n            }\n\n            assert(valid, 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 --json-schema-validation=false to disable schema validation (not recommended)}\n`);\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"]}