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/parser.js CHANGED
@@ -1,52 +1,23 @@
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.Parser = void 0;
30
- const chalk_1 = __importDefault(require("chalk"));
31
- const path_1 = __importDefault(require("path"));
32
- const deep_extend_1 = __importDefault(require("deep-extend"));
33
- const fs = __importStar(require("fs-extra"));
34
- const yaml = __importStar(require("js-yaml"));
35
- const pretty_hrtime_1 = __importDefault(require("pretty-hrtime"));
36
- const job_1 = require("./job");
37
- const DataExpander = __importStar(require("./data-expander"));
38
- const utils_1 = require("./utils");
39
- const assert_1 = __importDefault(require("assert"));
40
- const validator_1 = require("./validator");
41
- const parallel = __importStar(require("./parallel"));
42
- const git_data_1 = require("./git-data");
43
- const parser_includes_1 = require("./parser-includes");
44
- const producers_1 = require("./producers");
45
- const variables_from_files_1 = require("./variables-from-files");
46
- const predefined_variables_1 = require("./predefined-variables");
1
+ import chalk from "chalk";
2
+ import path from "path";
3
+ import deepExtend from "deep-extend";
4
+ import fs from "fs-extra";
5
+ import * as yaml from "js-yaml";
6
+ import prettyHrtime from "pretty-hrtime";
7
+ import { Job } from "./job.js";
8
+ import * as DataExpander from "./data-expander.js";
9
+ import { Utils } from "./utils.js";
10
+ import assert from "assert";
11
+ import { Validator } from "./validator.js";
12
+ import * as parallel from "./parallel.js";
13
+ import { GitData } from "./git-data.js";
14
+ import { ParserIncludes } from "./parser-includes.js";
15
+ import { Producers } from "./producers.js";
16
+ import { VariablesFromFiles } from "./variables-from-files.js";
17
+ import { init as initPredefinedVariables } from "./predefined-variables.js";
47
18
  const MAX_FUNCTIONS = 3;
48
19
  const INCLUDE_INPUTS_SUPPORTED_TYPES = ["string", "boolean", "number", "array"];
49
- class Parser {
20
+ export class Parser {
50
21
  _stages = [];
51
22
  _gitlabData;
52
23
  _jobNamePad = null;
@@ -75,28 +46,28 @@ class Parser {
75
46
  const parser = new Parser(argv, writeStreams, pipelineIid, jobs, expandVariables);
76
47
  const time = process.hrtime();
77
48
  await parser.init();
78
- const warnings = await validator_1.Validator.run(parser.jobs, parser.stages);
49
+ const warnings = await Validator.run(parser.jobs, parser.stages);
79
50
  for (const job of parser.jobs) {
80
51
  if (job.artifacts === null) {
81
52
  job.deleteArtifacts();
82
53
  }
83
54
  }
84
55
  const parsingTime = process.hrtime(time);
85
- const pathToExpandedGitLabCi = path_1.default.join(argv.cwd, argv.stateDir, "expanded-gitlab-ci.yml");
86
- fs.mkdirpSync(path_1.default.join(argv.cwd, argv.stateDir));
56
+ const pathToExpandedGitLabCi = path.join(argv.cwd, argv.stateDir, "expanded-gitlab-ci.yml");
57
+ fs.mkdirpSync(path.join(argv.cwd, argv.stateDir));
87
58
  fs.writeFileSync(pathToExpandedGitLabCi, yaml.dump(parser.gitlabData));
88
- writeStreams.stderr((0, chalk_1.default) `{grey parsing and downloads finished in ${(0, pretty_hrtime_1.default)(parsingTime)}.}\n`);
59
+ writeStreams.stderr(chalk `{grey parsing and downloads finished in ${prettyHrtime(parsingTime)}.}\n`);
89
60
  for (const warning of warnings) {
90
- writeStreams.stderr((0, chalk_1.default) `{yellow ${warning}}\n`);
61
+ writeStreams.stderr(chalk `{yellow ${warning}}\n`);
91
62
  }
92
63
  // # Second layer of check for errors that are not caught in Validator.run
93
64
  if (parser.argv.jsonSchemaValidation) {
94
65
  const time = process.hrtime();
95
- validator_1.Validator.jsonSchemaValidation({
66
+ Validator.jsonSchemaValidation({
96
67
  pathToExpandedGitLabCi,
97
68
  gitLabCiConfig: parser.gitlabData,
98
69
  });
99
- writeStreams.stderr((0, chalk_1.default) `{grey json schema validated in ${(0, pretty_hrtime_1.default)(process.hrtime(time))}}\n`);
70
+ writeStreams.stderr(chalk `{grey json schema validated in ${prettyHrtime(process.hrtime(time))}}\n`);
100
71
  }
101
72
  return parser;
102
73
  }
@@ -108,28 +79,28 @@ class Parser {
108
79
  const file = argv.file;
109
80
  const pipelineIid = this.pipelineIid;
110
81
  const fetchIncludes = argv.fetchIncludes;
111
- const gitData = await git_data_1.GitData.init(cwd, writeStreams);
112
- const variablesFromFiles = await variables_from_files_1.VariablesFromFiles.init(argv, writeStreams, gitData);
113
- const predefinedVariables = (0, predefined_variables_1.init)({ gitData, argv });
114
- const envMatchedVariables = utils_1.Utils.findEnvMatchedVariables(variablesFromFiles);
82
+ const gitData = await GitData.init(cwd, writeStreams);
83
+ const variablesFromFiles = await VariablesFromFiles.init(argv, writeStreams, gitData);
84
+ const envMatchedVariables = Utils.findEnvMatchedVariables(variablesFromFiles);
85
+ const predefinedVariables = initPredefinedVariables({ gitData, argv, envMatchedVariables });
115
86
  const variables = { ...predefinedVariables, ...envMatchedVariables, ...argv.variable };
116
- const expanded = utils_1.Utils.expandVariables(variables);
87
+ const expanded = Utils.expandVariables(variables);
117
88
  let yamlDataList = [{ stages: [".pre", "build", "test", "deploy", ".post"] }];
118
89
  const gitlabCiData = await Parser.loadYaml(`${cwd}/${file}`, {}, this.expandVariables);
119
- yamlDataList = yamlDataList.concat(await parser_includes_1.ParserIncludes.init(gitlabCiData, { cwd, stateDir, writeStreams, gitData, fetchIncludes, variables: expanded, expandVariables: this.expandVariables, maximumIncludes: argv.maximumIncludes }));
120
- parser_includes_1.ParserIncludes.resetCount();
90
+ yamlDataList = yamlDataList.concat(await ParserIncludes.init(gitlabCiData, { cwd, stateDir, writeStreams, gitData, fetchIncludes, variables: expanded, expandVariables: this.expandVariables, maximumIncludes: argv.maximumIncludes }));
91
+ ParserIncludes.resetCount();
121
92
  const gitlabCiLocalData = await Parser.loadYaml(`${cwd}/.gitlab-ci-local.yml`, {}, this.expandVariables);
122
- yamlDataList = yamlDataList.concat(await parser_includes_1.ParserIncludes.init(gitlabCiLocalData, { cwd, stateDir, writeStreams, gitData, fetchIncludes, variables: expanded, expandVariables: this.expandVariables, maximumIncludes: argv.maximumIncludes }));
123
- parser_includes_1.ParserIncludes.resetCount();
124
- const gitlabData = (0, deep_extend_1.default)({}, ...yamlDataList);
93
+ yamlDataList = yamlDataList.concat(await ParserIncludes.init(gitlabCiLocalData, { cwd, stateDir, writeStreams, gitData, fetchIncludes, variables: expanded, expandVariables: this.expandVariables, maximumIncludes: argv.maximumIncludes }));
94
+ ParserIncludes.resetCount();
95
+ const gitlabData = deepExtend({}, ...yamlDataList);
125
96
  // Expand various fields in gitlabData
126
97
  DataExpander.jobExtends(gitlabData);
127
98
  DataExpander.reference(gitlabData, gitlabData);
99
+ DataExpander.flattenLists(gitlabData);
128
100
  DataExpander.complexObjects(gitlabData);
129
101
  DataExpander.defaults(gitlabData);
130
102
  DataExpander.globalVariables(gitlabData);
131
- DataExpander.flattenLists(gitlabData);
132
- (0, assert_1.default)(gitlabData.stages && Array.isArray(gitlabData.stages), (0, chalk_1.default) `{yellow stages:} must be an array`);
103
+ assert(gitlabData.stages && Array.isArray(gitlabData.stages), chalk `{yellow stages:} must be an array`);
133
104
  if (!gitlabData.stages.includes(".pre")) {
134
105
  gitlabData.stages.unshift(".pre");
135
106
  }
@@ -138,27 +109,28 @@ class Parser {
138
109
  }
139
110
  this._stages = gitlabData.stages;
140
111
  // Check job variables for invalid hash of key value pairs, and cast numbers to strings
141
- utils_1.Utils.forEachRealJob(gitlabData, (jobName, jobData) => {
112
+ Utils.forEachRealJob(gitlabData, (jobName, jobData) => {
113
+ assert(jobData.when !== "never", chalk `This GitLab CI configuration is invalid: jobs:${jobName} when:never can only be used in a rules section or workflow:rules`);
142
114
  for (const [key, _value] of Object.entries(jobData.variables || {})) {
143
115
  let value = _value;
144
116
  if (value === null)
145
117
  value = ""; // variable's values are nullable
146
- (0, assert_1.default)(typeof value === "string" || typeof value === "number" || typeof value === "boolean", (0, chalk_1.default) `{blueBright ${jobName}} has invalid variables hash of key value pairs. ${key}=${value}`);
118
+ assert(typeof value === "string" || typeof value === "number" || typeof value === "boolean", chalk `{blueBright ${jobName}} has invalid variables hash of key value pairs. ${key}=${value}`);
147
119
  jobData.variables[key] = String(value);
148
120
  }
149
121
  for (let i = 0; i < (jobData.services ?? []).length; i++) {
150
122
  const service = jobData.services[i];
151
123
  for (const [key, value] of Object.entries(service.variables || {})) {
152
- (0, assert_1.default)(typeof value === "string" || typeof value === "number" || typeof value === "boolean", (0, chalk_1.default) `{blueBright ${jobName}.services[${i}]} has invalid variables hash of key value pairs. ${key}=${value}`);
124
+ assert(typeof value === "string" || typeof value === "number" || typeof value === "boolean", chalk `{blueBright ${jobName}.services[${i}]} has invalid variables hash of key value pairs. ${key}=${value}`);
153
125
  jobData.services[i].variables[key] = String(value);
154
126
  }
155
127
  }
156
128
  });
157
129
  this._gitlabData = gitlabData;
158
130
  // Generate jobs and put them into stages
159
- utils_1.Utils.forEachRealJob(gitlabData, (jobName, jobData) => {
160
- (0, assert_1.default)(gitData != null, "gitData must be set");
161
- (0, assert_1.default)(variablesFromFiles != null, "homeVariables must be set");
131
+ Utils.forEachRealJob(gitlabData, (jobName, jobData) => {
132
+ assert(gitData != null, "gitData must be set");
133
+ assert(variablesFromFiles != null, "homeVariables must be set");
162
134
  let nodeIndex = 1;
163
135
  const parallelMatrixVariablesList = parallel.matrixVariablesList(jobData, jobName);
164
136
  for (const parallelMatrixVariables of parallelMatrixVariablesList) {
@@ -169,7 +141,7 @@ class Parser {
169
141
  else if (parallel.isPlainParallel(jobData)) {
170
142
  matrixJobName = `${jobName}: [${nodeIndex}/${parallelMatrixVariablesList.length}]`;
171
143
  }
172
- const job = new job_1.Job({
144
+ const job = new Job({
173
145
  argv,
174
146
  writeStreams,
175
147
  data: jobData,
@@ -186,7 +158,7 @@ class Parser {
186
158
  expandVariables: this.expandVariables,
187
159
  });
188
160
  const foundStage = this.stages.includes(job.stage);
189
- (0, assert_1.default)(foundStage, (0, chalk_1.default) `{yellow stage:${job.stage}} not found for {blueBright ${job.name}}`);
161
+ assert(foundStage, chalk `{yellow stage:${job.stage}} not found for {blueBright ${job.name}}`);
190
162
  this.jobs.push(job);
191
163
  nodeIndex++;
192
164
  }
@@ -219,7 +191,7 @@ class Parser {
219
191
  });
220
192
  // Generate producers for each job
221
193
  this.jobs.forEach((job) => {
222
- job.producers = producers_1.Producers.init(this.jobs, this.stages, job);
194
+ job.producers = Producers.init(this.jobs, this.stages, job);
223
195
  });
224
196
  }
225
197
  static async loadYaml(filePath, ctx = {}, expandVariables = true) {
@@ -280,7 +252,7 @@ class Parser {
280
252
  }
281
253
  catch (e) {
282
254
  if (e instanceof yaml.YAMLException && e.reason === "duplicated mapping key") {
283
- console.log((0, chalk_1.default) `{black.bgYellowBright WARN } duplicated mapping key detected! Values will be overwritten!`);
255
+ console.log(chalk `{black.bgYellowBright WARN } duplicated mapping key detected! Values will be overwritten!`);
284
256
  fileData = yaml.loadAll(fileSplitClone.join("\n"), null, { schema, json: true });
285
257
  }
286
258
  else {
@@ -295,7 +267,7 @@ class Parser {
295
267
  const interpolatedConfigurations = JSON.stringify(uninterpolatedConfigurations)
296
268
  .replace(/(?<firstChar>.)?(?<secondChar>.)?\$\[\[\s*inputs.(?<interpolationKey>[\w-]+)\s*\|?\s*(?<interpolationFunctions>.*?)\s*\]\](?<lastChar>[^$])?/g // https://regexr.com/81c16
297
269
  , (_, firstChar, secondChar, interpolationKey, interpolationFunctions, lastChar) => {
298
- const configFilePath = path_1.default.relative(process.cwd(), filePath);
270
+ const configFilePath = path.relative(process.cwd(), filePath);
299
271
  const context = {
300
272
  interpolationKey,
301
273
  interpolationFunctions,
@@ -327,7 +299,7 @@ class Parser {
327
299
  }
328
300
  return firstTwoChar + inputValue + lastChar;
329
301
  default:
330
- utils_1.Utils.switchStatementExhaustiveCheck(inputType);
302
+ Utils.switchStatementExhaustiveCheck(inputType);
331
303
  }
332
304
  });
333
305
  return JSON.parse(interpolatedConfigurations);
@@ -335,36 +307,35 @@ class Parser {
335
307
  return fileData[0];
336
308
  }
337
309
  }
338
- exports.Parser = Parser;
339
310
  function isGitlabSpecFile(fileData) {
340
311
  return "spec" in fileData;
341
312
  }
342
313
  function validateInterpolationKey(ctx) {
343
314
  const { configFilePath, interpolationKey, inputsSpecification } = ctx;
344
- const invalidInterpolationKeyErr = (0, chalk_1.default) `This GitLab CI configuration is invalid: \`{blueBright ${configFilePath}}\`: unknown interpolation key: \`${interpolationKey}\`.`;
345
- (0, assert_1.default)(inputsSpecification.spec.inputs?.[interpolationKey] !== undefined, invalidInterpolationKeyErr);
315
+ const invalidInterpolationKeyErr = chalk `This GitLab CI configuration is invalid: \`{blueBright ${configFilePath}}\`: unknown interpolation key: \`${interpolationKey}\`.`;
316
+ assert(inputsSpecification.spec.inputs?.[interpolationKey] !== undefined, invalidInterpolationKeyErr);
346
317
  }
347
318
  function validateInterpolationFunctions(ctx) {
348
319
  const { interpolationFunctions, configFilePath } = ctx;
349
320
  if (interpolationFunctions != "") {
350
- console.log((0, chalk_1.default) `{black.bgYellowBright WARN } interpolation functions is currently not supported via gitlab-ci-local. Functions will just be a no-op.`);
321
+ console.log(chalk `{black.bgYellowBright WARN } interpolation functions is currently not supported via gitlab-ci-local. Functions will just be a no-op.`);
351
322
  }
352
- (0, assert_1.default)(interpolationFunctions.split("|").length <= MAX_FUNCTIONS, (0, chalk_1.default) `This GitLab CI configuration is invalid: \`{blueBright ${configFilePath}}\`: too many functions in interpolation block.`);
323
+ assert(interpolationFunctions.split("|").length <= MAX_FUNCTIONS, chalk `This GitLab CI configuration is invalid: \`{blueBright ${configFilePath}}\`: too many functions in interpolation block.`);
353
324
  }
354
325
  function validateInput(ctx) {
355
326
  const { configFilePath, interpolationKey, inputsSpecification } = ctx;
356
327
  const inputValue = getInputValue(ctx);
357
328
  const options = inputsSpecification.spec.inputs[interpolationKey]?.options;
358
329
  if (options) {
359
- (0, assert_1.default)(options.includes(inputValue), (0, chalk_1.default) `This GitLab CI configuration is invalid: \`{blueBright ${configFilePath}}\`: \`{blueBright ${interpolationKey}}\` input: \`{blueBright ${inputValue}}\` cannot be used because it is not in the list of allowed options.`);
330
+ assert(options.includes(inputValue), chalk `This GitLab CI configuration is invalid: \`{blueBright ${configFilePath}}\`: \`{blueBright ${interpolationKey}}\` input: \`{blueBright ${inputValue}}\` cannot be used because it is not in the list of allowed options.`);
360
331
  }
361
332
  const expectedInputType = getExpectedInputType(ctx);
362
- (0, assert_1.default)(INCLUDE_INPUTS_SUPPORTED_TYPES.includes(expectedInputType), (0, chalk_1.default) `This GitLab CI configuration is invalid: \`{blueBright ${configFilePath}}\`: header:spec:inputs:{blueBright ${interpolationKey}} input type unknown value: {blueBright ${expectedInputType}}.`);
333
+ assert(INCLUDE_INPUTS_SUPPORTED_TYPES.includes(expectedInputType), chalk `This GitLab CI configuration is invalid: \`{blueBright ${configFilePath}}\`: header:spec:inputs:{blueBright ${interpolationKey}} input type unknown value: {blueBright ${expectedInputType}}.`);
363
334
  const inputType = Array.isArray(inputValue) ? "array" : typeof inputValue;
364
- (0, assert_1.default)(inputType === expectedInputType, (0, chalk_1.default) `This GitLab CI configuration is invalid: \`{blueBright ${configFilePath}}\`: \`{blueBright ${interpolationKey}}\` input: provided value is not a {blueBright ${expectedInputType}}.`);
335
+ assert(inputType === expectedInputType, chalk `This GitLab CI configuration is invalid: \`{blueBright ${configFilePath}}\`: \`{blueBright ${interpolationKey}}\` input: provided value is not a {blueBright ${expectedInputType}}.`);
365
336
  const regex = inputsSpecification.spec.inputs[interpolationKey]?.regex;
366
337
  if (regex) {
367
- console.log((0, chalk_1.default) `{black.bgYellowBright WARN } spec:inputs:regex is currently not supported via gitlab-ci-local. This will just be a no-op.`);
338
+ console.log(chalk `{black.bgYellowBright WARN } spec:inputs:regex is currently not supported via gitlab-ci-local. This will just be a no-op.`);
368
339
  }
369
340
  }
370
341
  function parseIncludeInputs(ctx) {
@@ -376,11 +347,11 @@ function parseIncludeInputs(ctx) {
376
347
  function getInputValue(ctx) {
377
348
  const { inputs, interpolationKey, configFilePath, inputsSpecification } = ctx;
378
349
  const inputValue = inputs[interpolationKey] || inputsSpecification.spec.inputs[interpolationKey]?.default;
379
- (0, assert_1.default)(inputValue !== undefined, (0, chalk_1.default) `This GitLab CI configuration is invalid: \`{blueBright ${configFilePath}}\`: \`{blueBright ${interpolationKey}}\` input: required value has not been provided.`);
350
+ assert(inputValue !== undefined, chalk `This GitLab CI configuration is invalid: \`{blueBright ${configFilePath}}\`: \`{blueBright ${interpolationKey}}\` input: required value has not been provided.`);
380
351
  return inputValue;
381
352
  }
382
353
  function getExpectedInputType(ctx) {
383
354
  const { interpolationKey, inputsSpecification } = ctx;
384
355
  return inputsSpecification.spec.inputs[interpolationKey]?.type || "string";
385
356
  }
386
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"parser.js","sourceRoot":"","sources":["parser.ts"],"names":[],"mappings":"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","sourcesContent":["import chalk from \"chalk\";\nimport path from \"path\";\nimport deepExtend from \"deep-extend\";\nimport * as fs from \"fs-extra\";\nimport * as yaml from \"js-yaml\";\nimport prettyHrtime from \"pretty-hrtime\";\nimport {Job} from \"./job\";\nimport * as DataExpander from \"./data-expander\";\nimport {Utils} from \"./utils\";\nimport assert from \"assert\";\nimport {Validator} from \"./validator\";\nimport * as parallel from \"./parallel\";\nimport {GitData} from \"./git-data\";\nimport {ParserIncludes} from \"./parser-includes\";\nimport {Producers} from \"./producers\";\nimport {VariablesFromFiles} from \"./variables-from-files\";\nimport {Argv} from \"./argv\";\nimport {WriteStreams} from \"./write-streams\";\nimport {init as initPredefinedVariables} from \"./predefined-variables\";\n\nconst MAX_FUNCTIONS = 3;\nconst INCLUDE_INPUTS_SUPPORTED_TYPES = [\"string\", \"boolean\", \"number\", \"array\"] as const;\nexport type InputType = typeof INCLUDE_INPUTS_SUPPORTED_TYPES[number];\n\nexport class Parser {\n\n    private _stages: string[] = [];\n    private _gitlabData: any;\n    private _jobNamePad: number | null = null;\n\n    readonly jobs: Job[];\n    readonly argv: Argv;\n    readonly writeStreams: WriteStreams;\n    readonly pipelineIid: number;\n    readonly expandVariables: boolean;\n\n    private constructor (argv: Argv, writeStreams: WriteStreams, pipelineIid: number, jobs: Job[], expandVariables: boolean) {\n        this.argv = argv;\n        this.writeStreams = writeStreams;\n        this.pipelineIid = pipelineIid;\n        this.jobs = jobs;\n        this.expandVariables = expandVariables;\n    }\n\n    get stages (): readonly string[] {\n        return this._stages;\n    }\n\n    get gitlabData () {\n        return this._gitlabData;\n    }\n\n    get jobNamePad (): number {\n        return this._jobNamePad ?? 0;\n    }\n\n    static async create (argv: Argv, writeStreams: WriteStreams, pipelineIid: number, jobs: Job[], expandVariables: boolean = true) {\n        const parser = new Parser(argv, writeStreams, pipelineIid, jobs, expandVariables);\n        const time = process.hrtime();\n        await parser.init();\n        const warnings = await Validator.run(parser.jobs, parser.stages);\n\n        for (const job of parser.jobs) {\n            if (job.artifacts === null) {\n                job.deleteArtifacts();\n            }\n        }\n\n        const parsingTime = process.hrtime(time);\n        const pathToExpandedGitLabCi = path.join(argv.cwd, argv.stateDir, \"expanded-gitlab-ci.yml\");\n        fs.mkdirpSync(path.join(argv.cwd, argv.stateDir));\n        fs.writeFileSync(pathToExpandedGitLabCi, yaml.dump(parser.gitlabData));\n        writeStreams.stderr(chalk`{grey parsing and downloads finished in ${prettyHrtime(parsingTime)}.}\\n`);\n\n        for (const warning of warnings) {\n            writeStreams.stderr(chalk`{yellow ${warning}}\\n`);\n        }\n\n        // # Second layer of check for errors that are not caught in Validator.run\n        if (parser.argv.jsonSchemaValidation) {\n            const time = process.hrtime();\n            Validator.jsonSchemaValidation({\n                pathToExpandedGitLabCi,\n                gitLabCiConfig: parser.gitlabData,\n            });\n            writeStreams.stderr(chalk`{grey json schema validated in ${prettyHrtime(process.hrtime(time))}}\\n`);\n        }\n        return parser;\n    }\n\n    async init () {\n        const argv = this.argv;\n        const cwd = argv.cwd;\n        const stateDir = argv.stateDir;\n        const writeStreams = this.writeStreams;\n        const file = argv.file;\n        const pipelineIid = this.pipelineIid;\n        const fetchIncludes = argv.fetchIncludes;\n        const gitData = await GitData.init(cwd, writeStreams);\n        const variablesFromFiles = await VariablesFromFiles.init(argv, writeStreams, gitData);\n        const predefinedVariables = initPredefinedVariables({gitData, argv});\n        const envMatchedVariables = Utils.findEnvMatchedVariables(variablesFromFiles);\n        const variables = {...predefinedVariables, ...envMatchedVariables, ...argv.variable};\n        const expanded = Utils.expandVariables(variables);\n\n        let yamlDataList: any[] = [{stages: [\".pre\", \"build\", \"test\", \"deploy\", \".post\"]}];\n        const gitlabCiData = await Parser.loadYaml(`${cwd}/${file}`, {}, this.expandVariables);\n\n        yamlDataList = yamlDataList.concat(await ParserIncludes.init(gitlabCiData, {cwd, stateDir, writeStreams, gitData, fetchIncludes, variables: expanded, expandVariables: this.expandVariables, maximumIncludes: argv.maximumIncludes}));\n        ParserIncludes.resetCount();\n\n        const gitlabCiLocalData = await Parser.loadYaml(`${cwd}/.gitlab-ci-local.yml`, {}, this.expandVariables);\n        yamlDataList = yamlDataList.concat(await ParserIncludes.init(gitlabCiLocalData, {cwd, stateDir, writeStreams, gitData, fetchIncludes, variables: expanded, expandVariables: this.expandVariables, maximumIncludes: argv.maximumIncludes}));\n        ParserIncludes.resetCount();\n\n        const gitlabData: any = deepExtend({}, ...yamlDataList);\n\n        // Expand various fields in gitlabData\n        DataExpander.jobExtends(gitlabData);\n        DataExpander.reference(gitlabData, gitlabData);\n        DataExpander.complexObjects(gitlabData);\n        DataExpander.defaults(gitlabData);\n        DataExpander.globalVariables(gitlabData);\n        DataExpander.flattenLists(gitlabData);\n\n        assert(gitlabData.stages && Array.isArray(gitlabData.stages), chalk`{yellow stages:} must be an array`);\n        if (!gitlabData.stages.includes(\".pre\")) {\n            gitlabData.stages.unshift(\".pre\");\n        }\n        if (!gitlabData.stages.includes(\".post\")) {\n            gitlabData.stages.push(\".post\");\n        }\n        this._stages = gitlabData.stages;\n\n        // Check job variables for invalid hash of key value pairs, and cast numbers to strings\n        Utils.forEachRealJob(gitlabData, (jobName, jobData) => {\n            for (const [key, _value] of Object.entries(jobData.variables || {})) {\n                let value = _value;\n                if (value === null) value = \"\"; // variable's values are nullable\n                assert(\n                    typeof value === \"string\" || typeof value === \"number\" || typeof value === \"boolean\",\n                    chalk`{blueBright ${jobName}} has invalid variables hash of key value pairs. ${key}=${value}`\n                );\n                jobData.variables[key] = String(value);\n            }\n\n            for (let i = 0; i < (jobData.services ?? []).length; i++) {\n                const service = jobData.services[i];\n                for (const [key, value] of Object.entries(service.variables || {})) {\n                    assert(\n                        typeof value === \"string\" || typeof value === \"number\" || typeof value === \"boolean\",\n                        chalk`{blueBright ${jobName}.services[${i}]} has invalid variables hash of key value pairs. ${key}=${value}`\n                    );\n                    jobData.services[i].variables[key] = String(value);\n                }\n            }\n        });\n\n        this._gitlabData = gitlabData;\n\n        // Generate jobs and put them into stages\n        Utils.forEachRealJob(gitlabData, (jobName, jobData) => {\n            assert(gitData != null, \"gitData must be set\");\n            assert(variablesFromFiles != null, \"homeVariables must be set\");\n\n            let nodeIndex = 1;\n            const parallelMatrixVariablesList = parallel.matrixVariablesList(jobData, jobName);\n            for (const parallelMatrixVariables of parallelMatrixVariablesList) {\n                let matrixJobName = jobName;\n                if (parallelMatrixVariables) {\n                    matrixJobName = `${jobName}: [${Object.values(parallelMatrixVariables ?? []).join(\",\")}]`;\n                } else if (parallel.isPlainParallel(jobData)) {\n                    matrixJobName = `${jobName}: [${nodeIndex}/${parallelMatrixVariablesList.length}]`;\n                }\n\n                const job = new Job({\n                    argv,\n                    writeStreams,\n                    data: jobData,\n                    name: matrixJobName,\n                    baseName: jobName,\n                    globalVariables: gitlabData.variables,\n                    pipelineIid: pipelineIid,\n                    predefinedVariables,\n                    gitData,\n                    variablesFromFiles,\n                    matrixVariables: parallelMatrixVariables,\n                    nodeIndex: (jobData.parallel != null) ? nodeIndex : null,\n                    nodesTotal: parallelMatrixVariablesList.length,\n                    expandVariables: this.expandVariables,\n                });\n                const foundStage = this.stages.includes(job.stage);\n                assert(foundStage, chalk`{yellow stage:${job.stage}} not found for {blueBright ${job.name}}`);\n                this.jobs.push(job);\n                nodeIndex++;\n            }\n        });\n\n        // Add some padding so that job logs are nicely aligned\n        // allow users to override this in case they have really long job name (see #840)\n        if (this.argv.maxJobNamePadding !== null && this.argv.maxJobNamePadding <= 0) {\n            this._jobNamePad = 0;\n        } else {\n            const jobs = this.argv.job.length !== 0 ? this.argv.job : this.jobs;\n            jobs.forEach((job) => {\n                let jobNeedsLength: number[] = [];\n\n                if (this.argv.needs && this.argv.job.length > 0) {\n                    const found = this.jobs.find(j => j.baseName === job);\n                    if (found?.needs) {\n                        jobNeedsLength = found.needs.map(f => f.job.length);\n                    }\n                }\n                const jobLength = typeof job == \"string\" ? job.length : job.name.length;\n                this._jobNamePad = Math.max(jobLength, this._jobNamePad ?? 0, ...jobNeedsLength);\n            });\n            if (this.argv.maxJobNamePadding !== null) {\n                this._jobNamePad = Math.min(this.argv.maxJobNamePadding ?? 0, this._jobNamePad ?? 0);\n            }\n        }\n\n        // Set jobNamePad on all jobs\n        this.jobs.forEach((job) => {\n            job.jobNamePad = this.jobNamePad;\n        });\n\n        // Generate producers for each job\n        this.jobs.forEach((job) => {\n            job.producers = Producers.init(this.jobs, this.stages, job);\n        });\n    }\n\n    static async loadYaml (filePath: string, ctx: any = {}, expandVariables: boolean = true): Promise<any> {\n        const ymlPath = `${filePath}`;\n        if (!fs.existsSync(ymlPath)) {\n            return {};\n        }\n\n        const fileContent = await fs.readFile(`${filePath}`, \"utf8\");\n        const fileSplit = fileContent.split(/\\r?\\n/g);\n        const fileSplitClone = fileSplit.slice();\n\n        let interactiveMatch = null;\n        let descriptionMatch = null;\n        let injectSSHAgent = null;\n        let noArtifactsToSourceMatch = null;\n        let index = 0;\n        if (expandVariables) {\n            for (const line of fileSplit) {\n                interactiveMatch = !interactiveMatch ? /#\\s?@\\s?[Ii]nteractive/.exec(line) : interactiveMatch;\n                injectSSHAgent = !injectSSHAgent ? /#\\s?@\\s?[Ii]njectSSHAgent/.exec(line) : injectSSHAgent;\n                noArtifactsToSourceMatch = !noArtifactsToSourceMatch ? /#\\s?@\\s?NoArtifactsToSource/i.exec(line) : noArtifactsToSourceMatch;\n                descriptionMatch = !descriptionMatch ? /#\\s?@\\s?[Dd]escription (?<description>.*)/.exec(line) : descriptionMatch;\n\n                const jobMatch = /\\w:/.exec(line);\n                if (jobMatch && (interactiveMatch || descriptionMatch || injectSSHAgent || noArtifactsToSourceMatch)) {\n                    if (interactiveMatch) {\n                        fileSplitClone.splice(index + 1, 0, \"  gclInteractive: true\");\n                        index++;\n                    }\n                    if (injectSSHAgent) {\n                        fileSplitClone.splice(index + 1, 0, \"  gclInjectSSHAgent: true\");\n                        index++;\n                    }\n                    if (noArtifactsToSourceMatch) {\n                        fileSplitClone.splice(index + 1, 0, \"  gclArtifactsToSource: false\");\n                        index++;\n                    }\n                    if (descriptionMatch) {\n                        fileSplitClone.splice(index + 1, 0, `  gclDescription: ${descriptionMatch?.groups?.description ?? \"\"}`);\n                        index++;\n                    }\n                    interactiveMatch = null;\n                    descriptionMatch = null;\n                    injectSSHAgent = null;\n                    noArtifactsToSourceMatch = null;\n                }\n                index++;\n            }\n        }\n\n        const referenceType = new yaml.Type(\"!reference\", {\n            kind: \"sequence\",\n            construct: function (data) {\n                return {referenceData: data};\n            },\n        });\n        const schema = yaml.DEFAULT_SCHEMA.extend([referenceType]);\n        let fileData;\n\n        try {\n            fileData = yaml.loadAll(fileSplitClone.join(\"\\n\"), null, {schema}) as any[];\n        } catch (e: any) {\n            if (e instanceof yaml.YAMLException && e.reason === \"duplicated mapping key\") {\n                console.log(chalk`{black.bgYellowBright  WARN } duplicated mapping key detected! Values will be overwritten!`);\n                fileData = yaml.loadAll(fileSplitClone.join(\"\\n\"), null, {schema, json: true}) as any[];\n            } else {\n                throw e;\n            }\n        }\n\n        if (fileData.length <= 1) return fileData[0];\n\n        if (isGitlabSpecFile(fileData[0])) {\n            const inputsSpecification: any = fileData[0];\n            const uninterpolatedConfigurations: any = fileData[1];\n\n            const interpolatedConfigurations = JSON.stringify(uninterpolatedConfigurations)\n                .replace(\n                    /(?<firstChar>.)?(?<secondChar>.)?\\$\\[\\[\\s*inputs.(?<interpolationKey>[\\w-]+)\\s*\\|?\\s*(?<interpolationFunctions>.*?)\\s*\\]\\](?<lastChar>[^$])?/g // https://regexr.com/81c16\n                    , (_: string, firstChar: string, secondChar: string, interpolationKey: string, interpolationFunctions: string, lastChar: string) => {\n                        const configFilePath = path.relative(process.cwd(), filePath);\n                        const context = {\n                            interpolationKey,\n                            interpolationFunctions,\n                            inputsSpecification,\n                            configFilePath,\n                            ...ctx,\n                        };\n                        firstChar ??= \"\";\n                        secondChar ??= \"\";\n                        lastChar ??= \"\";\n\n                        const {inputValue, inputType} = parseIncludeInputs(context);\n                        const firstTwoChar = firstChar + secondChar;\n                        switch (inputType) {\n                            case \"array\":\n                                if ((secondChar == \"\\\"\" && lastChar == \"\\\"\") && firstChar != \"\\\\\") {\n                                    return firstChar + JSON.stringify(inputValue);\n                                }\n\n                                // NOTE: This behaves slightly differently from gitlab.com. I can't come up with practical use case so i don't think it's worth the effort to mimic this\n                                return firstTwoChar + JSON.stringify(JSON.stringify(inputValue)).slice(1, -1) + lastChar;\n                            case \"string\":\n                                return firstTwoChar\n                                    + JSON.stringify(inputValue) // ensure a valid json string\n                                        .slice(1, -1) // remove the surrounding \"\n                                    + lastChar;\n\n                            case \"number\":\n                            case \"boolean\":\n                                if ((secondChar == \"\\\"\" && lastChar == \"\\\"\") && firstChar != \"\\\\\") {\n                                    return firstChar + inputValue;\n                                }\n                                return firstTwoChar + inputValue + lastChar;\n\n                            default:\n                                Utils.switchStatementExhaustiveCheck(inputType);\n                        }\n                    });\n            return JSON.parse(interpolatedConfigurations);\n        }\n        return fileData[0];\n    }\n}\n\nfunction isGitlabSpecFile (fileData: any) {\n    return \"spec\" in fileData;\n}\n\nfunction validateInterpolationKey (ctx: any) {\n    const {configFilePath, interpolationKey, inputsSpecification} = ctx;\n    const invalidInterpolationKeyErr = chalk`This GitLab CI configuration is invalid: \\`{blueBright ${configFilePath}}\\`: unknown interpolation key: \\`${interpolationKey}\\`.`;\n    assert(inputsSpecification.spec.inputs?.[interpolationKey] !== undefined, invalidInterpolationKeyErr);\n}\n\nfunction validateInterpolationFunctions (ctx: any) {\n    const {interpolationFunctions, configFilePath} = ctx;\n    if (interpolationFunctions != \"\") {\n        console.log(chalk`{black.bgYellowBright  WARN } interpolation functions is currently not supported via gitlab-ci-local. Functions will just be a no-op.`);\n    }\n    assert(interpolationFunctions.split(\"|\").length <= MAX_FUNCTIONS, chalk`This GitLab CI configuration is invalid: \\`{blueBright ${configFilePath}}\\`: too many functions in interpolation block.`);\n}\n\nfunction validateInput (ctx: any) {\n    const {configFilePath, interpolationKey, inputsSpecification} = ctx;\n    const inputValue = getInputValue(ctx);\n\n    const options = inputsSpecification.spec.inputs[interpolationKey]?.options;\n    if (options) {\n        assert(options.includes(inputValue),\n            chalk`This GitLab CI configuration is invalid: \\`{blueBright ${configFilePath}}\\`: \\`{blueBright ${interpolationKey}}\\` input: \\`{blueBright ${inputValue}}\\` cannot be used because it is not in the list of allowed options.`);\n    }\n\n    const expectedInputType = getExpectedInputType(ctx);\n    assert(INCLUDE_INPUTS_SUPPORTED_TYPES.includes(expectedInputType),\n        chalk`This GitLab CI configuration is invalid: \\`{blueBright ${configFilePath}}\\`: header:spec:inputs:{blueBright ${interpolationKey}} input type unknown value: {blueBright ${expectedInputType}}.`);\n\n    const inputType = Array.isArray(inputValue) ? \"array\" : typeof inputValue;\n    assert(inputType === expectedInputType,\n        chalk`This GitLab CI configuration is invalid: \\`{blueBright ${configFilePath}}\\`: \\`{blueBright ${interpolationKey}}\\` input: provided value is not a {blueBright ${expectedInputType}}.`);\n\n    const regex = inputsSpecification.spec.inputs[interpolationKey]?.regex;\n    if (regex) {\n        console.log(chalk`{black.bgYellowBright  WARN } spec:inputs:regex is currently not supported via gitlab-ci-local. This will just be a no-op.`);\n    }\n}\n\nfunction parseIncludeInputs (ctx: any): {inputValue: any; inputType: InputType} {\n    validateInterpolationKey(ctx);\n    validateInterpolationFunctions(ctx);\n    validateInput(ctx);\n    return {inputValue: getInputValue(ctx), inputType: getExpectedInputType(ctx)};\n}\n\nfunction getInputValue (ctx: any) {\n    const {inputs, interpolationKey, configFilePath, inputsSpecification} = ctx;\n    const inputValue = inputs[interpolationKey] || inputsSpecification.spec.inputs[interpolationKey]?.default;\n    assert(inputValue !== undefined, chalk`This GitLab CI configuration is invalid: \\`{blueBright ${configFilePath}}\\`: \\`{blueBright ${interpolationKey}}\\` input: required value has not been provided.`);\n    return inputValue;\n}\n\nfunction getExpectedInputType (ctx: any): InputType {\n    const {interpolationKey, inputsSpecification} = ctx;\n    return inputsSpecification.spec.inputs[interpolationKey]?.type || \"string\";\n}\n"]}
357
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"parser.js","sourceRoot":"","sources":["parser.ts"],"names":[],"mappings":"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","sourcesContent":["import chalk from \"chalk\";\nimport path from \"path\";\nimport deepExtend from \"deep-extend\";\nimport fs from \"fs-extra\";\nimport * as yaml from \"js-yaml\";\nimport prettyHrtime from \"pretty-hrtime\";\nimport {Job} from \"./job.js\";\nimport * as DataExpander from \"./data-expander.js\";\nimport {Utils} from \"./utils.js\";\nimport assert from \"assert\";\nimport {Validator} from \"./validator.js\";\nimport * as parallel from \"./parallel.js\";\nimport {GitData} from \"./git-data.js\";\nimport {ParserIncludes} from \"./parser-includes.js\";\nimport {Producers} from \"./producers.js\";\nimport {VariablesFromFiles} from \"./variables-from-files.js\";\nimport {Argv} from \"./argv.js\";\nimport {WriteStreams} from \"./write-streams.js\";\nimport {init as initPredefinedVariables} from \"./predefined-variables.js\";\n\nconst MAX_FUNCTIONS = 3;\nconst INCLUDE_INPUTS_SUPPORTED_TYPES = [\"string\", \"boolean\", \"number\", \"array\"] as const;\nexport type InputType = typeof INCLUDE_INPUTS_SUPPORTED_TYPES[number];\n\nexport class Parser {\n\n    private _stages: string[] = [];\n    private _gitlabData: any;\n    private _jobNamePad: number | null = null;\n\n    readonly jobs: Job[];\n    readonly argv: Argv;\n    readonly writeStreams: WriteStreams;\n    readonly pipelineIid: number;\n    readonly expandVariables: boolean;\n\n    private constructor (argv: Argv, writeStreams: WriteStreams, pipelineIid: number, jobs: Job[], expandVariables: boolean) {\n        this.argv = argv;\n        this.writeStreams = writeStreams;\n        this.pipelineIid = pipelineIid;\n        this.jobs = jobs;\n        this.expandVariables = expandVariables;\n    }\n\n    get stages (): readonly string[] {\n        return this._stages;\n    }\n\n    get gitlabData () {\n        return this._gitlabData;\n    }\n\n    get jobNamePad (): number {\n        return this._jobNamePad ?? 0;\n    }\n\n    static async create (argv: Argv, writeStreams: WriteStreams, pipelineIid: number, jobs: Job[], expandVariables: boolean = true) {\n        const parser = new Parser(argv, writeStreams, pipelineIid, jobs, expandVariables);\n        const time = process.hrtime();\n        await parser.init();\n        const warnings = await Validator.run(parser.jobs, parser.stages);\n\n        for (const job of parser.jobs) {\n            if (job.artifacts === null) {\n                job.deleteArtifacts();\n            }\n        }\n\n        const parsingTime = process.hrtime(time);\n        const pathToExpandedGitLabCi = path.join(argv.cwd, argv.stateDir, \"expanded-gitlab-ci.yml\");\n        fs.mkdirpSync(path.join(argv.cwd, argv.stateDir));\n        fs.writeFileSync(pathToExpandedGitLabCi, yaml.dump(parser.gitlabData));\n        writeStreams.stderr(chalk`{grey parsing and downloads finished in ${prettyHrtime(parsingTime)}.}\\n`);\n\n        for (const warning of warnings) {\n            writeStreams.stderr(chalk`{yellow ${warning}}\\n`);\n        }\n\n        // # Second layer of check for errors that are not caught in Validator.run\n        if (parser.argv.jsonSchemaValidation) {\n            const time = process.hrtime();\n            Validator.jsonSchemaValidation({\n                pathToExpandedGitLabCi,\n                gitLabCiConfig: parser.gitlabData,\n            });\n            writeStreams.stderr(chalk`{grey json schema validated in ${prettyHrtime(process.hrtime(time))}}\\n`);\n        }\n        return parser;\n    }\n\n    async init () {\n        const argv = this.argv;\n        const cwd = argv.cwd;\n        const stateDir = argv.stateDir;\n        const writeStreams = this.writeStreams;\n        const file = argv.file;\n        const pipelineIid = this.pipelineIid;\n        const fetchIncludes = argv.fetchIncludes;\n        const gitData = await GitData.init(cwd, writeStreams);\n        const variablesFromFiles = await VariablesFromFiles.init(argv, writeStreams, gitData);\n        const envMatchedVariables = Utils.findEnvMatchedVariables(variablesFromFiles);\n        const predefinedVariables = initPredefinedVariables({gitData, argv, envMatchedVariables});\n        const variables = {...predefinedVariables, ...envMatchedVariables, ...argv.variable};\n        const expanded = Utils.expandVariables(variables);\n\n        let yamlDataList: any[] = [{stages: [\".pre\", \"build\", \"test\", \"deploy\", \".post\"]}];\n        const gitlabCiData = await Parser.loadYaml(`${cwd}/${file}`, {}, this.expandVariables);\n\n        yamlDataList = yamlDataList.concat(await ParserIncludes.init(gitlabCiData, {cwd, stateDir, writeStreams, gitData, fetchIncludes, variables: expanded, expandVariables: this.expandVariables, maximumIncludes: argv.maximumIncludes}));\n        ParserIncludes.resetCount();\n\n        const gitlabCiLocalData = await Parser.loadYaml(`${cwd}/.gitlab-ci-local.yml`, {}, this.expandVariables);\n        yamlDataList = yamlDataList.concat(await ParserIncludes.init(gitlabCiLocalData, {cwd, stateDir, writeStreams, gitData, fetchIncludes, variables: expanded, expandVariables: this.expandVariables, maximumIncludes: argv.maximumIncludes}));\n        ParserIncludes.resetCount();\n\n        const gitlabData: any = deepExtend({}, ...yamlDataList);\n\n        // Expand various fields in gitlabData\n        DataExpander.jobExtends(gitlabData);\n        DataExpander.reference(gitlabData, gitlabData);\n        DataExpander.flattenLists(gitlabData);\n        DataExpander.complexObjects(gitlabData);\n        DataExpander.defaults(gitlabData);\n        DataExpander.globalVariables(gitlabData);\n\n        assert(gitlabData.stages && Array.isArray(gitlabData.stages), chalk`{yellow stages:} must be an array`);\n        if (!gitlabData.stages.includes(\".pre\")) {\n            gitlabData.stages.unshift(\".pre\");\n        }\n        if (!gitlabData.stages.includes(\".post\")) {\n            gitlabData.stages.push(\".post\");\n        }\n        this._stages = gitlabData.stages;\n\n        // Check job variables for invalid hash of key value pairs, and cast numbers to strings\n        Utils.forEachRealJob(gitlabData, (jobName, jobData) => {\n            assert(jobData.when !== \"never\",\n                chalk`This GitLab CI configuration is invalid: jobs:${jobName} when:never can only be used in a rules section or workflow:rules`\n            );\n            for (const [key, _value] of Object.entries(jobData.variables || {})) {\n                let value = _value;\n                if (value === null) value = \"\"; // variable's values are nullable\n                assert(\n                    typeof value === \"string\" || typeof value === \"number\" || typeof value === \"boolean\",\n                    chalk`{blueBright ${jobName}} has invalid variables hash of key value pairs. ${key}=${value}`\n                );\n                jobData.variables[key] = String(value);\n            }\n\n            for (let i = 0; i < (jobData.services ?? []).length; i++) {\n                const service = jobData.services[i];\n                for (const [key, value] of Object.entries(service.variables || {})) {\n                    assert(\n                        typeof value === \"string\" || typeof value === \"number\" || typeof value === \"boolean\",\n                        chalk`{blueBright ${jobName}.services[${i}]} has invalid variables hash of key value pairs. ${key}=${value}`\n                    );\n                    jobData.services[i].variables[key] = String(value);\n                }\n            }\n        });\n\n        this._gitlabData = gitlabData;\n\n        // Generate jobs and put them into stages\n        Utils.forEachRealJob(gitlabData, (jobName, jobData) => {\n            assert(gitData != null, \"gitData must be set\");\n            assert(variablesFromFiles != null, \"homeVariables must be set\");\n\n            let nodeIndex = 1;\n            const parallelMatrixVariablesList = parallel.matrixVariablesList(jobData, jobName);\n            for (const parallelMatrixVariables of parallelMatrixVariablesList) {\n                let matrixJobName = jobName;\n                if (parallelMatrixVariables) {\n                    matrixJobName = `${jobName}: [${Object.values(parallelMatrixVariables ?? []).join(\",\")}]`;\n                } else if (parallel.isPlainParallel(jobData)) {\n                    matrixJobName = `${jobName}: [${nodeIndex}/${parallelMatrixVariablesList.length}]`;\n                }\n\n                const job = new Job({\n                    argv,\n                    writeStreams,\n                    data: jobData,\n                    name: matrixJobName,\n                    baseName: jobName,\n                    globalVariables: gitlabData.variables,\n                    pipelineIid: pipelineIid,\n                    predefinedVariables,\n                    gitData,\n                    variablesFromFiles,\n                    matrixVariables: parallelMatrixVariables,\n                    nodeIndex: (jobData.parallel != null) ? nodeIndex : null,\n                    nodesTotal: parallelMatrixVariablesList.length,\n                    expandVariables: this.expandVariables,\n                });\n                const foundStage = this.stages.includes(job.stage);\n                assert(foundStage, chalk`{yellow stage:${job.stage}} not found for {blueBright ${job.name}}`);\n                this.jobs.push(job);\n                nodeIndex++;\n            }\n        });\n\n        // Add some padding so that job logs are nicely aligned\n        // allow users to override this in case they have really long job name (see #840)\n        if (this.argv.maxJobNamePadding !== null && this.argv.maxJobNamePadding <= 0) {\n            this._jobNamePad = 0;\n        } else {\n            const jobs = this.argv.job.length !== 0 ? this.argv.job : this.jobs;\n            jobs.forEach((job) => {\n                let jobNeedsLength: number[] = [];\n\n                if (this.argv.needs && this.argv.job.length > 0) {\n                    const found = this.jobs.find(j => j.baseName === job);\n                    if (found?.needs) {\n                        jobNeedsLength = found.needs.map(f => f.job.length);\n                    }\n                }\n                const jobLength = typeof job == \"string\" ? job.length : job.name.length;\n                this._jobNamePad = Math.max(jobLength, this._jobNamePad ?? 0, ...jobNeedsLength);\n            });\n            if (this.argv.maxJobNamePadding !== null) {\n                this._jobNamePad = Math.min(this.argv.maxJobNamePadding ?? 0, this._jobNamePad ?? 0);\n            }\n        }\n\n        // Set jobNamePad on all jobs\n        this.jobs.forEach((job) => {\n            job.jobNamePad = this.jobNamePad;\n        });\n\n        // Generate producers for each job\n        this.jobs.forEach((job) => {\n            job.producers = Producers.init(this.jobs, this.stages, job);\n        });\n    }\n\n    static async loadYaml (filePath: string, ctx: any = {}, expandVariables: boolean = true): Promise<any> {\n        const ymlPath = `${filePath}`;\n        if (!fs.existsSync(ymlPath)) {\n            return {};\n        }\n\n        const fileContent = await fs.readFile(`${filePath}`, \"utf8\");\n        const fileSplit = fileContent.split(/\\r?\\n/g);\n        const fileSplitClone = fileSplit.slice();\n\n        let interactiveMatch = null;\n        let descriptionMatch = null;\n        let injectSSHAgent = null;\n        let noArtifactsToSourceMatch = null;\n        let index = 0;\n        if (expandVariables) {\n            for (const line of fileSplit) {\n                interactiveMatch = !interactiveMatch ? /#\\s?@\\s?[Ii]nteractive/.exec(line) : interactiveMatch;\n                injectSSHAgent = !injectSSHAgent ? /#\\s?@\\s?[Ii]njectSSHAgent/.exec(line) : injectSSHAgent;\n                noArtifactsToSourceMatch = !noArtifactsToSourceMatch ? /#\\s?@\\s?NoArtifactsToSource/i.exec(line) : noArtifactsToSourceMatch;\n                descriptionMatch = !descriptionMatch ? /#\\s?@\\s?[Dd]escription (?<description>.*)/.exec(line) : descriptionMatch;\n\n                const jobMatch = /\\w:/.exec(line);\n                if (jobMatch && (interactiveMatch || descriptionMatch || injectSSHAgent || noArtifactsToSourceMatch)) {\n                    if (interactiveMatch) {\n                        fileSplitClone.splice(index + 1, 0, \"  gclInteractive: true\");\n                        index++;\n                    }\n                    if (injectSSHAgent) {\n                        fileSplitClone.splice(index + 1, 0, \"  gclInjectSSHAgent: true\");\n                        index++;\n                    }\n                    if (noArtifactsToSourceMatch) {\n                        fileSplitClone.splice(index + 1, 0, \"  gclArtifactsToSource: false\");\n                        index++;\n                    }\n                    if (descriptionMatch) {\n                        fileSplitClone.splice(index + 1, 0, `  gclDescription: ${descriptionMatch?.groups?.description ?? \"\"}`);\n                        index++;\n                    }\n                    interactiveMatch = null;\n                    descriptionMatch = null;\n                    injectSSHAgent = null;\n                    noArtifactsToSourceMatch = null;\n                }\n                index++;\n            }\n        }\n\n        const referenceType = new yaml.Type(\"!reference\", {\n            kind: \"sequence\",\n            construct: function (data) {\n                return {referenceData: data};\n            },\n        });\n        const schema = yaml.DEFAULT_SCHEMA.extend([referenceType]);\n        let fileData;\n\n        try {\n            fileData = yaml.loadAll(fileSplitClone.join(\"\\n\"), null, {schema}) as any[];\n        } catch (e: any) {\n            if (e instanceof yaml.YAMLException && e.reason === \"duplicated mapping key\") {\n                console.log(chalk`{black.bgYellowBright  WARN } duplicated mapping key detected! Values will be overwritten!`);\n                fileData = yaml.loadAll(fileSplitClone.join(\"\\n\"), null, {schema, json: true}) as any[];\n            } else {\n                throw e;\n            }\n        }\n\n        if (fileData.length <= 1) return fileData[0];\n\n        if (isGitlabSpecFile(fileData[0])) {\n            const inputsSpecification: any = fileData[0];\n            const uninterpolatedConfigurations: any = fileData[1];\n\n            const interpolatedConfigurations = JSON.stringify(uninterpolatedConfigurations)\n                .replace(\n                    /(?<firstChar>.)?(?<secondChar>.)?\\$\\[\\[\\s*inputs.(?<interpolationKey>[\\w-]+)\\s*\\|?\\s*(?<interpolationFunctions>.*?)\\s*\\]\\](?<lastChar>[^$])?/g // https://regexr.com/81c16\n                    , (_: string, firstChar: string, secondChar: string, interpolationKey: string, interpolationFunctions: string, lastChar: string) => {\n                        const configFilePath = path.relative(process.cwd(), filePath);\n                        const context = {\n                            interpolationKey,\n                            interpolationFunctions,\n                            inputsSpecification,\n                            configFilePath,\n                            ...ctx,\n                        };\n                        firstChar ??= \"\";\n                        secondChar ??= \"\";\n                        lastChar ??= \"\";\n\n                        const {inputValue, inputType} = parseIncludeInputs(context);\n                        const firstTwoChar = firstChar + secondChar;\n                        switch (inputType) {\n                            case \"array\":\n                                if ((secondChar == \"\\\"\" && lastChar == \"\\\"\") && firstChar != \"\\\\\") {\n                                    return firstChar + JSON.stringify(inputValue);\n                                }\n\n                                // NOTE: This behaves slightly differently from gitlab.com. I can't come up with practical use case so i don't think it's worth the effort to mimic this\n                                return firstTwoChar + JSON.stringify(JSON.stringify(inputValue)).slice(1, -1) + lastChar;\n                            case \"string\":\n                                return firstTwoChar\n                                    + JSON.stringify(inputValue) // ensure a valid json string\n                                        .slice(1, -1) // remove the surrounding \"\n                                    + lastChar;\n\n                            case \"number\":\n                            case \"boolean\":\n                                if ((secondChar == \"\\\"\" && lastChar == \"\\\"\") && firstChar != \"\\\\\") {\n                                    return firstChar + inputValue;\n                                }\n                                return firstTwoChar + inputValue + lastChar;\n\n                            default:\n                                Utils.switchStatementExhaustiveCheck(inputType);\n                        }\n                    });\n            return JSON.parse(interpolatedConfigurations);\n        }\n        return fileData[0];\n    }\n}\n\nfunction isGitlabSpecFile (fileData: any) {\n    return \"spec\" in fileData;\n}\n\nfunction validateInterpolationKey (ctx: any) {\n    const {configFilePath, interpolationKey, inputsSpecification} = ctx;\n    const invalidInterpolationKeyErr = chalk`This GitLab CI configuration is invalid: \\`{blueBright ${configFilePath}}\\`: unknown interpolation key: \\`${interpolationKey}\\`.`;\n    assert(inputsSpecification.spec.inputs?.[interpolationKey] !== undefined, invalidInterpolationKeyErr);\n}\n\nfunction validateInterpolationFunctions (ctx: any) {\n    const {interpolationFunctions, configFilePath} = ctx;\n    if (interpolationFunctions != \"\") {\n        console.log(chalk`{black.bgYellowBright  WARN } interpolation functions is currently not supported via gitlab-ci-local. Functions will just be a no-op.`);\n    }\n    assert(interpolationFunctions.split(\"|\").length <= MAX_FUNCTIONS, chalk`This GitLab CI configuration is invalid: \\`{blueBright ${configFilePath}}\\`: too many functions in interpolation block.`);\n}\n\nfunction validateInput (ctx: any) {\n    const {configFilePath, interpolationKey, inputsSpecification} = ctx;\n    const inputValue = getInputValue(ctx);\n\n    const options = inputsSpecification.spec.inputs[interpolationKey]?.options;\n    if (options) {\n        assert(options.includes(inputValue),\n            chalk`This GitLab CI configuration is invalid: \\`{blueBright ${configFilePath}}\\`: \\`{blueBright ${interpolationKey}}\\` input: \\`{blueBright ${inputValue}}\\` cannot be used because it is not in the list of allowed options.`);\n    }\n\n    const expectedInputType = getExpectedInputType(ctx);\n    assert(INCLUDE_INPUTS_SUPPORTED_TYPES.includes(expectedInputType),\n        chalk`This GitLab CI configuration is invalid: \\`{blueBright ${configFilePath}}\\`: header:spec:inputs:{blueBright ${interpolationKey}} input type unknown value: {blueBright ${expectedInputType}}.`);\n\n    const inputType = Array.isArray(inputValue) ? \"array\" : typeof inputValue;\n    assert(inputType === expectedInputType,\n        chalk`This GitLab CI configuration is invalid: \\`{blueBright ${configFilePath}}\\`: \\`{blueBright ${interpolationKey}}\\` input: provided value is not a {blueBright ${expectedInputType}}.`);\n\n    const regex = inputsSpecification.spec.inputs[interpolationKey]?.regex;\n    if (regex) {\n        console.log(chalk`{black.bgYellowBright  WARN } spec:inputs:regex is currently not supported via gitlab-ci-local. This will just be a no-op.`);\n    }\n}\n\nfunction parseIncludeInputs (ctx: any): {inputValue: any; inputType: InputType} {\n    validateInterpolationKey(ctx);\n    validateInterpolationFunctions(ctx);\n    validateInput(ctx);\n    return {inputValue: getInputValue(ctx), inputType: getExpectedInputType(ctx)};\n}\n\nfunction getInputValue (ctx: any) {\n    const {inputs, interpolationKey, configFilePath, inputsSpecification} = ctx;\n    const inputValue = inputs[interpolationKey] || inputsSpecification.spec.inputs[interpolationKey]?.default;\n    assert(inputValue !== undefined, chalk`This GitLab CI configuration is invalid: \\`{blueBright ${configFilePath}}\\`: \\`{blueBright ${interpolationKey}}\\` input: required value has not been provided.`);\n    return inputValue;\n}\n\nfunction getExpectedInputType (ctx: any): InputType {\n    const {interpolationKey, inputsSpecification} = ctx;\n    return inputsSpecification.spec.inputs[interpolationKey]?.type || \"string\";\n}\n"]}