gitlab-ci-local 4.65.1 → 4.66.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +20 -19
- package/dist/index.js +340 -0
- package/dist/index.js.map +480 -0
- package/package.json +26 -80
- package/src/argv.js +0 -307
- package/src/commander.js +0 -247
- package/src/data-expander.js +0 -220
- package/src/executor.js +0 -104
- package/src/git-data.js +0 -168
- package/src/gitlab-preset.js +0 -24
- package/src/global.js +0 -40
- package/src/handler.js +0 -119
- package/src/index.js +0 -364
- package/src/job.js +0 -1463
- package/src/mutex.js +0 -20
- package/src/parallel.js +0 -46
- package/src/parser-includes.js +0 -387
- package/src/parser.js +0 -357
- package/src/predefined-variables.js +0 -67
- package/src/producers.js +0 -42
- package/src/schema/index.js +0 -19
- package/src/schema/schema.js +0 -3190
- package/src/schema-error.js +0 -114
- package/src/state.js +0 -22
- package/src/utils.js +0 -485
- package/src/validator.js +0 -151
- package/src/variables-from-files.js +0 -143
- package/src/write-streams.js +0 -51
package/src/commander.js
DELETED
|
@@ -1,247 +0,0 @@
|
|
|
1
|
-
import chalk from "chalk-template";
|
|
2
|
-
import { Utils } from "./utils.js";
|
|
3
|
-
import { Executor } from "./executor.js";
|
|
4
|
-
import fs from "fs-extra";
|
|
5
|
-
import { AssertionError } from "assert";
|
|
6
|
-
export class Commander {
|
|
7
|
-
static async runPipeline(argv, parser, writeStreams) {
|
|
8
|
-
const jobs = parser.jobs;
|
|
9
|
-
const stages = parser.stages;
|
|
10
|
-
let potentialStarters = [...jobs.values()];
|
|
11
|
-
potentialStarters = potentialStarters.filter(j => j.when !== "never");
|
|
12
|
-
potentialStarters = potentialStarters.filter(j => j.when !== "manual" || argv.manual.includes(j.name));
|
|
13
|
-
await Executor.runLoop(argv, jobs, stages, potentialStarters);
|
|
14
|
-
await Commander.printReport({
|
|
15
|
-
cwd: argv.cwd,
|
|
16
|
-
showTimestamps: argv.showTimestamps,
|
|
17
|
-
stateDir: argv.stateDir,
|
|
18
|
-
writeStreams: writeStreams,
|
|
19
|
-
jobs: jobs,
|
|
20
|
-
stages: stages,
|
|
21
|
-
jobNamePad: parser.jobNamePad,
|
|
22
|
-
});
|
|
23
|
-
}
|
|
24
|
-
static async runJobsInStage(argv, parser, writeStreams) {
|
|
25
|
-
const jobs = parser.jobs.filter(j => j.stage === argv.stage);
|
|
26
|
-
const stages = parser.stages;
|
|
27
|
-
let potentialStarters = [...jobs.values()];
|
|
28
|
-
potentialStarters = potentialStarters.filter(j => j.when !== "never");
|
|
29
|
-
potentialStarters = potentialStarters.filter(j => j.when !== "manual" || argv.manual.includes(j.name));
|
|
30
|
-
potentialStarters = potentialStarters.filter(j => j.stage === argv.stage);
|
|
31
|
-
await Executor.runLoop(argv, jobs, stages, potentialStarters);
|
|
32
|
-
await Commander.printReport({
|
|
33
|
-
cwd: argv.cwd,
|
|
34
|
-
showTimestamps: argv.showTimestamps,
|
|
35
|
-
stateDir: argv.stateDir,
|
|
36
|
-
writeStreams: writeStreams,
|
|
37
|
-
jobs: jobs,
|
|
38
|
-
stages: stages,
|
|
39
|
-
jobNamePad: parser.jobNamePad,
|
|
40
|
-
});
|
|
41
|
-
}
|
|
42
|
-
static async runJobs(argv, parser, writeStreams) {
|
|
43
|
-
const needs = argv.needs || argv.onlyNeeds;
|
|
44
|
-
const jobArgs = argv.job;
|
|
45
|
-
const jobs = parser.jobs;
|
|
46
|
-
const stages = parser.stages;
|
|
47
|
-
const potentialStarters = [];
|
|
48
|
-
const jobSet = needs ? new Set(jobs) : new Set();
|
|
49
|
-
jobArgs.forEach(jobArgName => {
|
|
50
|
-
const baseJobs = jobs.filter(j => j.baseName == jobArgName || j.name === jobArgName);
|
|
51
|
-
for (const b of baseJobs) {
|
|
52
|
-
jobSet.add(b);
|
|
53
|
-
if (needs) {
|
|
54
|
-
potentialStarters.push(...Executor.getPastToWaitFor(jobs, stages, b, argv.manual));
|
|
55
|
-
}
|
|
56
|
-
potentialStarters.push(b);
|
|
57
|
-
}
|
|
58
|
-
});
|
|
59
|
-
if (potentialStarters.length === 0) {
|
|
60
|
-
throw new AssertionError({ message: chalk `{blueBright ${jobArgs.join(",")}} could not be found` });
|
|
61
|
-
}
|
|
62
|
-
const starters = argv.onlyNeeds ?
|
|
63
|
-
potentialStarters.filter(p => !jobArgs.includes(p.name)) :
|
|
64
|
-
potentialStarters;
|
|
65
|
-
await Executor.runLoop(argv, Array.from(jobSet), stages, starters);
|
|
66
|
-
await Commander.printReport({
|
|
67
|
-
cwd: argv.cwd,
|
|
68
|
-
showTimestamps: argv.showTimestamps,
|
|
69
|
-
stateDir: argv.stateDir,
|
|
70
|
-
writeStreams: writeStreams,
|
|
71
|
-
jobs: jobs,
|
|
72
|
-
stages: stages,
|
|
73
|
-
jobNamePad: parser.jobNamePad,
|
|
74
|
-
});
|
|
75
|
-
}
|
|
76
|
-
static async printReport({ cwd, stateDir, showTimestamps, writeStreams, jobs, stages, jobNamePad }) {
|
|
77
|
-
writeStreams.stdout("\n");
|
|
78
|
-
const renderDuration = (duration) => showTimestamps ? ` [${duration.padStart(7)}]` : "";
|
|
79
|
-
const preScripts = {
|
|
80
|
-
successful: [],
|
|
81
|
-
failed: [],
|
|
82
|
-
warned: [],
|
|
83
|
-
};
|
|
84
|
-
const afterScripts = {
|
|
85
|
-
warned: [],
|
|
86
|
-
};
|
|
87
|
-
for (const job of jobs.values()) {
|
|
88
|
-
if (job.started && job.afterScriptsExitCode !== 0) {
|
|
89
|
-
afterScripts.warned.push(job);
|
|
90
|
-
}
|
|
91
|
-
}
|
|
92
|
-
for (const job of jobs.values()) {
|
|
93
|
-
if (!job.started) {
|
|
94
|
-
continue;
|
|
95
|
-
}
|
|
96
|
-
if (job.jobStatus === "success") {
|
|
97
|
-
preScripts.successful.push(job);
|
|
98
|
-
}
|
|
99
|
-
else if (job.jobStatus === "warning") {
|
|
100
|
-
preScripts.warned.push(job);
|
|
101
|
-
}
|
|
102
|
-
else {
|
|
103
|
-
preScripts.failed.push(job);
|
|
104
|
-
}
|
|
105
|
-
}
|
|
106
|
-
if (preScripts.successful.length !== 0) {
|
|
107
|
-
preScripts.successful.sort((a, b) => stages.indexOf(a.stage) - stages.indexOf(b.stage));
|
|
108
|
-
preScripts.successful.forEach(({ argv, coveragePercent, name, prettyDuration }) => {
|
|
109
|
-
let prefix = "";
|
|
110
|
-
if (argv.childPipelineDepth > 0)
|
|
111
|
-
prefix = `[${argv.variable.GCL_TRIGGERER}] -> `;
|
|
112
|
-
const namePad = name.padEnd(jobNamePad);
|
|
113
|
-
writeStreams.stdout(chalk `{black.bgGreenBright PASS }${renderDuration(prettyDuration)} {blueBright ${prefix}${namePad}}`);
|
|
114
|
-
if (coveragePercent) {
|
|
115
|
-
writeStreams.stdout(chalk ` ${coveragePercent}% {grey coverage}`);
|
|
116
|
-
}
|
|
117
|
-
writeStreams.stdout("\n");
|
|
118
|
-
});
|
|
119
|
-
}
|
|
120
|
-
if (preScripts.warned.length !== 0) {
|
|
121
|
-
preScripts.warned.sort((a, b) => stages.indexOf(a.stage) - stages.indexOf(b.stage));
|
|
122
|
-
for (const { name, prettyDuration } of preScripts.warned) {
|
|
123
|
-
const namePad = name.padEnd(jobNamePad);
|
|
124
|
-
const safeName = Utils.safeDockerString(name);
|
|
125
|
-
writeStreams.stdout(chalk `{black.bgYellowBright WARN }${renderDuration(prettyDuration)} {blueBright ${namePad}} pre_script\n`);
|
|
126
|
-
const outputLog = await fs.readFile(`${cwd}/${stateDir}/output/${safeName}.log`, "utf8");
|
|
127
|
-
for (const line of outputLog.split(/\r?\n/).filter(j => !j.includes("[32m$ ")).filter(j => j !== "").slice(-3)) {
|
|
128
|
-
writeStreams.stdout(chalk ` {yellow >} ${line}\n`);
|
|
129
|
-
}
|
|
130
|
-
}
|
|
131
|
-
}
|
|
132
|
-
if (afterScripts.warned.length !== 0) {
|
|
133
|
-
afterScripts.warned.sort((a, b) => stages.indexOf(a.stage) - stages.indexOf(b.stage));
|
|
134
|
-
afterScripts.warned.forEach(({ name, prettyDuration }) => {
|
|
135
|
-
const namePad = name.padEnd(jobNamePad);
|
|
136
|
-
writeStreams.stdout(chalk `{black.bgYellowBright WARN }${renderDuration(prettyDuration)} {blueBright ${namePad}} after_script\n`);
|
|
137
|
-
});
|
|
138
|
-
}
|
|
139
|
-
if (preScripts.failed.length !== 0) {
|
|
140
|
-
preScripts.failed.sort((a, b) => stages.indexOf(a.stage) - stages.indexOf(b.stage));
|
|
141
|
-
for (const { name, prettyDuration } of preScripts.failed) {
|
|
142
|
-
const namePad = name.padEnd(jobNamePad);
|
|
143
|
-
const safeName = Utils.safeDockerString(name);
|
|
144
|
-
writeStreams.stdout(chalk `{black.bgRed FAIL }${renderDuration(prettyDuration)} {blueBright ${namePad}}\n`);
|
|
145
|
-
const outputLog = await fs.readFile(`${cwd}/${stateDir}/output/${safeName}.log`, "utf8");
|
|
146
|
-
for (const line of outputLog.split(/\r?\n/).filter(j => !j.includes("[32m$ ")).filter(j => j !== "").slice(-3)) {
|
|
147
|
-
writeStreams.stdout(chalk ` {red >} ${line}\n`);
|
|
148
|
-
}
|
|
149
|
-
}
|
|
150
|
-
}
|
|
151
|
-
for (const job of preScripts.successful) {
|
|
152
|
-
const e = job.environment;
|
|
153
|
-
if (e == null) {
|
|
154
|
-
continue;
|
|
155
|
-
}
|
|
156
|
-
const name = e.name;
|
|
157
|
-
const url = e.url;
|
|
158
|
-
writeStreams.stdout(chalk `{blueBright ${job.name}} environment: \{ name: {bold ${name}}`);
|
|
159
|
-
if (url != null) {
|
|
160
|
-
writeStreams.stdout(chalk `, url: {bold ${url}}`);
|
|
161
|
-
}
|
|
162
|
-
writeStreams.stdout(" }\n");
|
|
163
|
-
}
|
|
164
|
-
}
|
|
165
|
-
static runList(parser, writeStreams, listAll) {
|
|
166
|
-
const stages = parser.stages;
|
|
167
|
-
let jobs = [...parser.jobs.values()];
|
|
168
|
-
jobs.sort((a, b) => {
|
|
169
|
-
return stages.indexOf(a.stage) - stages.indexOf(b.stage);
|
|
170
|
-
});
|
|
171
|
-
let whenPadEnd = 4;
|
|
172
|
-
jobs.forEach(j => whenPadEnd = Math.max(j.when.length, whenPadEnd));
|
|
173
|
-
let stagePadEnd = 5;
|
|
174
|
-
stages.forEach(s => stagePadEnd = Math.max(s.length, stagePadEnd));
|
|
175
|
-
let descriptionPadEnd = 11;
|
|
176
|
-
jobs.forEach(j => descriptionPadEnd = Math.max(j.description.length, descriptionPadEnd));
|
|
177
|
-
const jobNamePad = parser.jobNamePad;
|
|
178
|
-
if (!listAll) {
|
|
179
|
-
jobs = jobs.filter(j => j.when !== "never");
|
|
180
|
-
}
|
|
181
|
-
writeStreams.stdout(chalk `{grey ${"name".padEnd(jobNamePad)} ${"description".padEnd(descriptionPadEnd)}} `);
|
|
182
|
-
writeStreams.stdout(chalk `{grey ${"stage".padEnd(stagePadEnd)} ${"when".padEnd(whenPadEnd)}} `);
|
|
183
|
-
writeStreams.stdout(chalk `{grey allow_failure needs}\n`);
|
|
184
|
-
const renderLine = (job) => {
|
|
185
|
-
const needs = job.needs?.filter(n => !n.project && !n.pipeline).map(n => n.job);
|
|
186
|
-
const allowFailure = job.allowFailure ? "true " : "false ";
|
|
187
|
-
let jobLine = chalk `{blueBright ${job.name.padEnd(jobNamePad)}} ${job.description.padEnd(descriptionPadEnd)} `;
|
|
188
|
-
jobLine += chalk `{yellow ${job.stage.padEnd(stagePadEnd)}} ${job.when.padEnd(whenPadEnd)} ${allowFailure.padEnd(11)}`;
|
|
189
|
-
if (needs) {
|
|
190
|
-
jobLine += chalk ` [{blueBright ${needs}}]`;
|
|
191
|
-
}
|
|
192
|
-
writeStreams.stdout(`${jobLine}\n`);
|
|
193
|
-
};
|
|
194
|
-
jobs.forEach((job) => renderLine(job));
|
|
195
|
-
}
|
|
196
|
-
static runJson(parser, writeStreams) {
|
|
197
|
-
const jobs = [...parser.jobs.values()];
|
|
198
|
-
const json = [];
|
|
199
|
-
jobs.forEach((job) => {
|
|
200
|
-
json.push({
|
|
201
|
-
name: job.name,
|
|
202
|
-
description: job.description,
|
|
203
|
-
stage: job.stage,
|
|
204
|
-
when: job.when,
|
|
205
|
-
allow_failure: job.allowFailure,
|
|
206
|
-
needs: job.needs?.filter(n => !n.project && !n.pipeline),
|
|
207
|
-
...job.rules ? { rules: job.rules } : {},
|
|
208
|
-
});
|
|
209
|
-
});
|
|
210
|
-
writeStreams.stdout(`${JSON.stringify(json, null, 2)}\n`);
|
|
211
|
-
}
|
|
212
|
-
static runCsv(parser, writeStreams, all) {
|
|
213
|
-
const stages = parser.stages;
|
|
214
|
-
let jobs = [...parser.jobs.values()];
|
|
215
|
-
jobs.sort((a, b) => {
|
|
216
|
-
return stages.indexOf(a.stage) - stages.indexOf(b.stage);
|
|
217
|
-
});
|
|
218
|
-
if (!all) {
|
|
219
|
-
jobs = jobs.filter(j => j.when !== "never");
|
|
220
|
-
}
|
|
221
|
-
writeStreams.stdout("name;description;stage;when;allowFailure;needs\n");
|
|
222
|
-
jobs.forEach((job) => {
|
|
223
|
-
const needs = job.needs?.filter(n => !n.project && !n.pipeline).map(n => n.job).join(",") ?? [];
|
|
224
|
-
writeStreams.stdout(`${job.name};"${job.description}";${job.stage};${job.when};${job.allowFailure ? "true" : "false"};[${needs}]\n`);
|
|
225
|
-
});
|
|
226
|
-
}
|
|
227
|
-
static validateDependencyChain(parser) {
|
|
228
|
-
const allJobs = parser.jobs;
|
|
229
|
-
// This is only the jobs that will actually run
|
|
230
|
-
const activeJobs = allJobs.filter(j => j.when !== "never");
|
|
231
|
-
const stages = parser.stages;
|
|
232
|
-
// This will throw an assertion errror if the dependency chain is broken due to needs keyword on specific events without having to run the full pipeline
|
|
233
|
-
Executor.getStartCandidates(allJobs, stages, activeJobs, []);
|
|
234
|
-
const activeJobNames = new Set(activeJobs.map(job => job.name));
|
|
235
|
-
// This willl throw an assertion error if the dependency chain is broken due to dependencies keyword (a job depending on artifacts from a job that will never run) without having to run the full pipeline
|
|
236
|
-
for (const job of activeJobs) {
|
|
237
|
-
if (job.dependencies) {
|
|
238
|
-
for (const dependency of job.dependencies) {
|
|
239
|
-
if (!activeJobNames.has(dependency)) {
|
|
240
|
-
throw new AssertionError({ message: chalk `{blueBright ${dependency}} is when:never, but its depended on by {blueBright ${job.name}}` });
|
|
241
|
-
}
|
|
242
|
-
}
|
|
243
|
-
}
|
|
244
|
-
}
|
|
245
|
-
}
|
|
246
|
-
}
|
|
247
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"commander.js","sourceRoot":"","sources":["commander.ts"],"names":[],"mappings":"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","sourcesContent":["import chalk from \"chalk-template\";\nimport {Job} from \"./job.js\";\nimport {Parser} from \"./parser.js\";\nimport {Utils} from \"./utils.js\";\nimport {WriteStreams} from \"./write-streams.js\";\nimport {Executor} from \"./executor.js\";\nimport fs from \"fs-extra\";\nimport {Argv} from \"./argv.js\";\nimport {AssertionError} from \"assert\";\n\nexport class Commander {\n\n    static async runPipeline (argv: Argv, parser: Parser, writeStreams: WriteStreams) {\n        const jobs = parser.jobs;\n        const stages = parser.stages;\n\n        let potentialStarters = [...jobs.values()];\n        potentialStarters = potentialStarters.filter(j => j.when !== \"never\");\n        potentialStarters = potentialStarters.filter(j => j.when !== \"manual\" || argv.manual.includes(j.name));\n        await Executor.runLoop(argv, jobs, stages, potentialStarters);\n        await Commander.printReport({\n            cwd: argv.cwd,\n            showTimestamps: argv.showTimestamps,\n            stateDir: argv.stateDir,\n            writeStreams: writeStreams,\n            jobs: jobs,\n            stages: stages,\n            jobNamePad: parser.jobNamePad,\n        });\n    }\n\n    static async runJobsInStage (argv: Argv, parser: Parser, writeStreams: WriteStreams) {\n        const jobs = parser.jobs.filter(j => j.stage === argv.stage);\n        const stages = parser.stages;\n\n        let potentialStarters = [...jobs.values()];\n        potentialStarters = potentialStarters.filter(j => j.when !== \"never\");\n        potentialStarters = potentialStarters.filter(j => j.when !== \"manual\" || argv.manual.includes(j.name));\n        potentialStarters = potentialStarters.filter(j => j.stage === argv.stage);\n        await Executor.runLoop(argv, jobs, stages, potentialStarters);\n        await Commander.printReport({\n            cwd: argv.cwd,\n            showTimestamps: argv.showTimestamps,\n            stateDir: argv.stateDir,\n            writeStreams: writeStreams,\n            jobs: jobs,\n            stages: stages,\n            jobNamePad: parser.jobNamePad,\n        });\n    }\n\n    static async runJobs (argv: Argv, parser: Parser, writeStreams: WriteStreams) {\n        const needs = argv.needs || argv.onlyNeeds;\n        const jobArgs = argv.job;\n        const jobs = parser.jobs;\n        const stages = parser.stages;\n\n        const potentialStarters: Job[] = [];\n        const jobSet = needs ? new Set(jobs) : new Set<Job>();\n        jobArgs.forEach(jobArgName => {\n            const baseJobs = jobs.filter(j => j.baseName == jobArgName || j.name === jobArgName);\n            for (const b of baseJobs) {\n                jobSet.add(b);\n                if (needs) {\n                    potentialStarters.push(...Executor.getPastToWaitFor(jobs, stages, b, argv.manual));\n                }\n                potentialStarters.push(b);\n            }\n        });\n\n        if (potentialStarters.length === 0) {\n            throw new AssertionError({message: chalk`{blueBright ${jobArgs.join(\",\")}} could not be found`});\n        }\n\n        const starters = argv.onlyNeeds ?\n            potentialStarters.filter(p => !jobArgs.includes(p.name)) :\n            potentialStarters;\n\n        await Executor.runLoop(argv, Array.from(jobSet), stages, starters);\n        await Commander.printReport({\n            cwd: argv.cwd,\n            showTimestamps: argv.showTimestamps,\n            stateDir: argv.stateDir,\n            writeStreams: writeStreams,\n            jobs: jobs,\n            stages: stages,\n            jobNamePad: parser.jobNamePad,\n        });\n    }\n\n    static async printReport ({cwd, stateDir, showTimestamps, writeStreams, jobs, stages, jobNamePad}: {\n        cwd: string;\n        showTimestamps: boolean;\n        stateDir: string;\n        writeStreams: WriteStreams;\n        jobs: ReadonlyArray<Job>;\n        stages: readonly string[];\n        jobNamePad: number;\n    }) {\n\n        writeStreams.stdout(\"\\n\");\n\n        const renderDuration = (duration: string) => showTimestamps ? ` [${duration.padStart(7)}]` : \"\";\n\n        const preScripts: {successful: Job[]; failed: Job[]; warned: Job[]} = {\n            successful: [],\n            failed: [],\n            warned: [],\n        };\n        const afterScripts: {warned: Job[]} = {\n            warned: [],\n        };\n\n        for (const job of jobs.values()) {\n            if (job.started && job.afterScriptsExitCode !== 0) {\n                afterScripts.warned.push(job);\n            }\n        }\n\n        for (const job of jobs.values()) {\n            if (!job.started) {\n                continue;\n            }\n\n            if (job.jobStatus === \"success\") {\n                preScripts.successful.push(job);\n            } else if (job.jobStatus === \"warning\") {\n                preScripts.warned.push(job);\n            } else {\n                preScripts.failed.push(job);\n            }\n        }\n\n        if (preScripts.successful.length !== 0) {\n            preScripts.successful.sort((a, b) => stages.indexOf(a.stage) - stages.indexOf(b.stage));\n            preScripts.successful.forEach(({argv, coveragePercent, name, prettyDuration}) => {\n                let prefix = \"\";\n                if (argv.childPipelineDepth > 0) prefix = `[${argv.variable.GCL_TRIGGERER}] -> `;\n\n                const namePad = name.padEnd(jobNamePad);\n                writeStreams.stdout(chalk`{black.bgGreenBright  PASS }${renderDuration(prettyDuration)} {blueBright ${prefix}${namePad}}`);\n                if (coveragePercent) {\n                    writeStreams.stdout(chalk` ${coveragePercent}% {grey coverage}`);\n                }\n                writeStreams.stdout(\"\\n\");\n            });\n        }\n\n        if (preScripts.warned.length !== 0) {\n            preScripts.warned.sort((a, b) => stages.indexOf(a.stage) - stages.indexOf(b.stage));\n            for (const {name, prettyDuration} of preScripts.warned) {\n                const namePad = name.padEnd(jobNamePad);\n                const safeName = Utils.safeDockerString(name);\n                writeStreams.stdout(chalk`{black.bgYellowBright  WARN }${renderDuration(prettyDuration)} {blueBright ${namePad}}  pre_script\\n`);\n                const outputLog = await fs.readFile(`${cwd}/${stateDir}/output/${safeName}.log`, \"utf8\");\n                for (const line of outputLog.split(/\\r?\\n/).filter(j => !j.includes(\"[32m$ \")).filter(j => j !== \"\").slice(-3)) {\n                    writeStreams.stdout(chalk`  {yellow >} ${line}\\n`);\n                }\n            }\n        }\n\n        if (afterScripts.warned.length !== 0) {\n            afterScripts.warned.sort((a, b) => stages.indexOf(a.stage) - stages.indexOf(b.stage));\n            afterScripts.warned.forEach(({name, prettyDuration}) => {\n                const namePad = name.padEnd(jobNamePad);\n                writeStreams.stdout(chalk`{black.bgYellowBright  WARN }${renderDuration(prettyDuration)} {blueBright ${namePad}}  after_script\\n`);\n            });\n        }\n\n        if (preScripts.failed.length !== 0) {\n            preScripts.failed.sort((a, b) => stages.indexOf(a.stage) - stages.indexOf(b.stage));\n            for (const {name, prettyDuration} of preScripts.failed) {\n                const namePad = name.padEnd(jobNamePad);\n                const safeName = Utils.safeDockerString(name);\n                writeStreams.stdout(chalk`{black.bgRed  FAIL }${renderDuration(prettyDuration)} {blueBright ${namePad}}\\n`);\n                const outputLog = await fs.readFile(`${cwd}/${stateDir}/output/${safeName}.log`, \"utf8\");\n                for (const line of outputLog.split(/\\r?\\n/).filter(j => !j.includes(\"[32m$ \")).filter(j => j !== \"\").slice(-3)) {\n                    writeStreams.stdout(chalk`  {red >} ${line}\\n`);\n                }\n            }\n        }\n\n        for (const job of preScripts.successful) {\n            const e = job.environment;\n            if (e == null) {\n                continue;\n            }\n            const name = e.name;\n            const url = e.url;\n            writeStreams.stdout(chalk`{blueBright ${job.name}} environment: \\{ name: {bold ${name}}`);\n            if (url != null) {\n                writeStreams.stdout(chalk`, url: {bold ${url}}`);\n            }\n            writeStreams.stdout(\" }\\n\");\n        }\n    }\n\n    static runList (parser: Parser, writeStreams: WriteStreams, listAll: boolean) {\n        const stages = parser.stages;\n        let jobs = [...parser.jobs.values()];\n        jobs.sort((a, b) => {\n            return stages.indexOf(a.stage) - stages.indexOf(b.stage);\n        });\n\n        let whenPadEnd = 4;\n        jobs.forEach(j => whenPadEnd = Math.max(j.when.length, whenPadEnd));\n\n        let stagePadEnd = 5;\n        stages.forEach(s => stagePadEnd = Math.max(s.length, stagePadEnd));\n\n        let descriptionPadEnd = 11;\n        jobs.forEach(j => descriptionPadEnd = Math.max(j.description.length, descriptionPadEnd));\n\n        const jobNamePad = parser.jobNamePad;\n\n        if (!listAll) {\n            jobs = jobs.filter(j => j.when !== \"never\");\n        }\n\n        writeStreams.stdout(chalk`{grey ${\"name\".padEnd(jobNamePad)}  ${\"description\".padEnd(descriptionPadEnd)}}  `);\n        writeStreams.stdout(chalk`{grey ${\"stage\".padEnd(stagePadEnd)}  ${\"when\".padEnd(whenPadEnd)}}  `);\n        writeStreams.stdout(chalk`{grey allow_failure  needs}\\n`);\n\n        const renderLine = (job: Job) => {\n            const needs = job.needs?.filter(n => !n.project && !n.pipeline).map(n => n.job);\n            const allowFailure = job.allowFailure ? \"true \" : \"false \";\n            let jobLine = chalk`{blueBright ${job.name.padEnd(jobNamePad)}}  ${job.description.padEnd(descriptionPadEnd)}  `;\n            jobLine += chalk`{yellow ${job.stage.padEnd(stagePadEnd)}}  ${job.when.padEnd(whenPadEnd)}  ${allowFailure.padEnd(11)}`;\n            if (needs) {\n                jobLine += chalk`    [{blueBright ${needs}}]`;\n            }\n            writeStreams.stdout(`${jobLine}\\n`);\n        };\n\n        jobs.forEach((job) => renderLine(job));\n    }\n\n    static runJson (parser: Parser, writeStreams: WriteStreams) {\n        const jobs = [...parser.jobs.values()];\n        const json: any[] = [];\n\n        jobs.forEach((job) => {\n            json.push({\n                name: job.name,\n                description: job.description,\n                stage: job.stage,\n                when: job.when,\n                allow_failure: job.allowFailure,\n                needs: job.needs?.filter(n => !n.project && !n.pipeline),\n                ...job.rules ? {rules: job.rules} : {},\n            });\n        });\n\n        writeStreams.stdout(`${JSON.stringify(json, null, 2)}\\n`);\n    }\n\n    static runCsv (parser: Parser, writeStreams: WriteStreams, all: boolean) {\n        const stages = parser.stages;\n        let jobs = [...parser.jobs.values()];\n        jobs.sort((a, b) => {\n            return stages.indexOf(a.stage) - stages.indexOf(b.stage);\n        });\n\n        if (!all) {\n            jobs = jobs.filter(j => j.when !== \"never\");\n        }\n\n        writeStreams.stdout(\"name;description;stage;when;allowFailure;needs\\n\");\n        jobs.forEach((job) => {\n            const needs = job.needs?.filter(n => !n.project && !n.pipeline).map(n => n.job).join(\",\") ?? [];\n            writeStreams.stdout(`${job.name};\"${job.description}\";${job.stage};${job.when};${job.allowFailure ? \"true\" : \"false\"};[${needs}]\\n`);\n        });\n    }\n\n    static validateDependencyChain (parser: Parser) {\n        const allJobs = parser.jobs;\n        // This is only the jobs that will actually run\n        const activeJobs = allJobs.filter(j => j.when !== \"never\");\n        const stages = parser.stages;\n        // This will throw an assertion errror if the dependency chain is broken due to needs keyword on specific events without having to run the full pipeline\n        Executor.getStartCandidates(allJobs, stages, activeJobs, []);\n\n        const activeJobNames = new Set(activeJobs.map(job => job.name));\n        // This willl throw an assertion error if the dependency chain is broken due to dependencies keyword (a job depending on artifacts from a job that will never run) without having to run the full pipeline\n        for (const job of activeJobs) {\n            if (job.dependencies) {\n                for (const dependency of job.dependencies) {\n                    if (!activeJobNames.has(dependency)) {\n                        throw new AssertionError({message: chalk`{blueBright ${dependency}} is when:never, but its depended on by {blueBright ${job.name}}`});\n                    }\n                }\n            }\n        }\n    }\n}\n"]}
|
package/src/data-expander.js
DELETED
|
@@ -1,220 +0,0 @@
|
|
|
1
|
-
import chalk from "chalk-template";
|
|
2
|
-
import deepExtend from "deep-extend";
|
|
3
|
-
import assert, { AssertionError } from "assert";
|
|
4
|
-
import { Job } from "./job.js";
|
|
5
|
-
import { traverse } from "object-traversal";
|
|
6
|
-
import { Utils } from "./utils.js";
|
|
7
|
-
const extendsMaxDepth = 11;
|
|
8
|
-
const extendsRecurse = (gitlabData, jobName, jobData, parents, depth) => {
|
|
9
|
-
assert(depth < extendsMaxDepth, chalk `{blueBright ${jobName}}: circular dependency detected in \`extends\``);
|
|
10
|
-
depth++;
|
|
11
|
-
jobData.extends = typeof jobData.extends === "string" ? [jobData.extends] : jobData.extends;
|
|
12
|
-
jobData.extends = jobData.extends ?? [];
|
|
13
|
-
for (const parentName of jobData.extends) {
|
|
14
|
-
const parentData = gitlabData[parentName];
|
|
15
|
-
assert(parentData != null, chalk `{blueBright ${parentName}} is unspecified, used by {blueBright ${jobName}} extends`);
|
|
16
|
-
extendsRecurse(gitlabData, parentName, parentData, parents, depth);
|
|
17
|
-
parents.push(parentData);
|
|
18
|
-
}
|
|
19
|
-
return parents;
|
|
20
|
-
};
|
|
21
|
-
export function jobExtends(gitlabData) {
|
|
22
|
-
for (const [jobName, jobData] of Object.entries(gitlabData)) {
|
|
23
|
-
if (Job.illegalJobNames.has(jobName))
|
|
24
|
-
continue;
|
|
25
|
-
if (!Utils.isObject(jobData))
|
|
26
|
-
continue;
|
|
27
|
-
const parentDatas = extendsRecurse(gitlabData, jobName, jobData, [], 0);
|
|
28
|
-
gitlabData[jobName] = deepExtend({}, ...parentDatas, jobData);
|
|
29
|
-
}
|
|
30
|
-
for (const [jobName, jobData] of Object.entries(gitlabData)) {
|
|
31
|
-
if (Job.illegalJobNames.has(jobName))
|
|
32
|
-
continue;
|
|
33
|
-
if (!Utils.isObject(jobData))
|
|
34
|
-
continue;
|
|
35
|
-
delete jobData.extends;
|
|
36
|
-
}
|
|
37
|
-
}
|
|
38
|
-
export function reference(gitlabData, recurseData) {
|
|
39
|
-
for (const [key, value] of Object.entries(recurseData || {})) {
|
|
40
|
-
if (value?.referenceData) {
|
|
41
|
-
if (Object.keys(value).length > 1) {
|
|
42
|
-
recurseData[key] = { ...getSubDataByReference(gitlabData, value.referenceData), ...recurseData[key] };
|
|
43
|
-
delete recurseData[key].referenceData;
|
|
44
|
-
}
|
|
45
|
-
else {
|
|
46
|
-
recurseData[key] = getSubDataByReference(gitlabData, value.referenceData);
|
|
47
|
-
}
|
|
48
|
-
}
|
|
49
|
-
else if (typeof value === "object") {
|
|
50
|
-
reference(gitlabData, value);
|
|
51
|
-
}
|
|
52
|
-
if (hasCircularChain(recurseData)) {
|
|
53
|
-
throw new AssertionError({ message: `!reference circular chain detected [${value.referenceData}]` });
|
|
54
|
-
}
|
|
55
|
-
}
|
|
56
|
-
}
|
|
57
|
-
const getSubDataByReference = (gitlabData, referenceData) => {
|
|
58
|
-
let gitlabSubData = gitlabData;
|
|
59
|
-
referenceData.forEach((referencePointer) => {
|
|
60
|
-
assert(gitlabSubData[referencePointer] != null, `!reference [${referenceData.join(", ")}] is undefined`);
|
|
61
|
-
gitlabSubData = gitlabSubData[referencePointer];
|
|
62
|
-
});
|
|
63
|
-
return gitlabSubData;
|
|
64
|
-
};
|
|
65
|
-
function hasCircularChain(data) {
|
|
66
|
-
try {
|
|
67
|
-
JSON.stringify(data);
|
|
68
|
-
}
|
|
69
|
-
catch (e) {
|
|
70
|
-
if (e instanceof TypeError && e.message.startsWith("Converting circular structure to JSON")) {
|
|
71
|
-
return true;
|
|
72
|
-
}
|
|
73
|
-
}
|
|
74
|
-
return false;
|
|
75
|
-
}
|
|
76
|
-
/**
|
|
77
|
-
Transform the globally defined ["image", "services", "cache", "before_script", "after_script"] into the default.x syntax
|
|
78
|
-
https://docs.gitlab.com/ee/ci/yaml/index.html#globally-defined-image-services-cache-before_script-after_script
|
|
79
|
-
*/
|
|
80
|
-
export function transformDeprecatedGlobalDefaultSyntax(gitlabData) {
|
|
81
|
-
const GITLAB_DEPRECATED_GLOBALLY_DEFINED_KEYWORDS = ["image", "services", "cache", "before_script", "after_script"];
|
|
82
|
-
gitlabData.default = gitlabData.default ?? {};
|
|
83
|
-
for (const g of GITLAB_DEPRECATED_GLOBALLY_DEFINED_KEYWORDS) {
|
|
84
|
-
if (gitlabData[g] !== undefined) {
|
|
85
|
-
gitlabData.default[g] = gitlabData[g];
|
|
86
|
-
delete gitlabData[g]; // Since this is deprecated, deleting it to prevent us from using it internally
|
|
87
|
-
}
|
|
88
|
-
}
|
|
89
|
-
}
|
|
90
|
-
export function normalize(gitlabData) {
|
|
91
|
-
normalizeGlobalVariables(gitlabData);
|
|
92
|
-
for (const [jobName, jobData] of Object.entries(gitlabData)) {
|
|
93
|
-
if (Job.illegalJobNames.has(jobName) || jobName.startsWith("."))
|
|
94
|
-
continue;
|
|
95
|
-
needsEach(jobName, gitlabData);
|
|
96
|
-
cacheEach(jobName, gitlabData);
|
|
97
|
-
servicesEach(jobName, gitlabData);
|
|
98
|
-
imageEach(jobName, gitlabData);
|
|
99
|
-
jobData.after_script = (typeof jobData.after_script === "string" && jobData.after_script !== "") ? [jobData.after_script] : jobData.after_script;
|
|
100
|
-
jobData.before_script = (typeof jobData.before_script === "string" && jobData.before_script !== "") ? [jobData.before_script] : jobData.before_script;
|
|
101
|
-
jobData.script = (typeof jobData.script === "string" && jobData.script !== "") ? [jobData.script] : jobData.script;
|
|
102
|
-
}
|
|
103
|
-
}
|
|
104
|
-
export function needsComplex(data) {
|
|
105
|
-
const needs = {
|
|
106
|
-
job: data.job ?? data,
|
|
107
|
-
artifacts: data.artifacts ?? true,
|
|
108
|
-
...(data.pipeline ? { pipeline: data.pipeline } : {}),
|
|
109
|
-
...(data.project ? { project: data.project } : {}),
|
|
110
|
-
...(data.ref ? { ref: data.ref } : {}),
|
|
111
|
-
...(data.optional ? { optional: data.optional } : {}),
|
|
112
|
-
};
|
|
113
|
-
// In needs:project/needs:pipeline, `optional` is not an allowed property
|
|
114
|
-
if (!data.project && !data.pipeline && data.optional === undefined) {
|
|
115
|
-
needs.optional = false;
|
|
116
|
-
}
|
|
117
|
-
return needs;
|
|
118
|
-
}
|
|
119
|
-
export function needsEach(jobName, gitlabData) {
|
|
120
|
-
const jobData = gitlabData[jobName];
|
|
121
|
-
if (!jobData.needs)
|
|
122
|
-
return;
|
|
123
|
-
for (const [i, n] of Object.entries(jobData.needs)) {
|
|
124
|
-
jobData.needs[i] = needsComplex(n);
|
|
125
|
-
}
|
|
126
|
-
}
|
|
127
|
-
export function cacheComplex(data) {
|
|
128
|
-
return {
|
|
129
|
-
key: data.key,
|
|
130
|
-
paths: data.paths,
|
|
131
|
-
policy: data.policy ?? "pull-push",
|
|
132
|
-
when: data.when ?? "on_success",
|
|
133
|
-
};
|
|
134
|
-
}
|
|
135
|
-
export function cacheEach(jobName, gitlabData) {
|
|
136
|
-
const jobData = gitlabData[jobName];
|
|
137
|
-
const cache = jobData.cache;
|
|
138
|
-
if (!cache)
|
|
139
|
-
return;
|
|
140
|
-
jobData.cache = Array.isArray(cache) ? cache : [cache];
|
|
141
|
-
for (const [i, c] of Object.entries(jobData.cache)) {
|
|
142
|
-
if (c.key?.files instanceof Array) {
|
|
143
|
-
assert(c.key.files.length === 1 || c.key.files.length === 2, `cache:key:files should be an array of one or two file paths. Got ${c.key.files.length}`);
|
|
144
|
-
}
|
|
145
|
-
jobData.cache[i] = cacheComplex(c);
|
|
146
|
-
}
|
|
147
|
-
// Remove cache elements with empty paths array (gitlab.com works the same way.)
|
|
148
|
-
jobData.cache = jobData.cache.filter((c) => c.paths?.length !== undefined);
|
|
149
|
-
}
|
|
150
|
-
export function servicesComplex(data) {
|
|
151
|
-
return {
|
|
152
|
-
name: typeof data === "string" ? data : data.name,
|
|
153
|
-
entrypoint: data.entrypoint,
|
|
154
|
-
command: data.command,
|
|
155
|
-
alias: data.alias,
|
|
156
|
-
variables: data.variables,
|
|
157
|
-
};
|
|
158
|
-
}
|
|
159
|
-
export function servicesEach(jobName, gitlabData) {
|
|
160
|
-
const jobData = gitlabData[jobName];
|
|
161
|
-
const services = jobData.services;
|
|
162
|
-
if (!services)
|
|
163
|
-
return;
|
|
164
|
-
jobData.services = Array.isArray(services) ? services : [services];
|
|
165
|
-
for (const [i, s] of Object.entries(jobData.services)) {
|
|
166
|
-
jobData.services[i] = servicesComplex(s);
|
|
167
|
-
}
|
|
168
|
-
}
|
|
169
|
-
export function imageComplex(data) {
|
|
170
|
-
if (data == null)
|
|
171
|
-
return data;
|
|
172
|
-
return {
|
|
173
|
-
name: typeof data === "string" ? data : data.name,
|
|
174
|
-
entrypoint: data.entrypoint,
|
|
175
|
-
...(data.docker?.user ? { docker: { user: data.docker?.user } } : {}),
|
|
176
|
-
};
|
|
177
|
-
}
|
|
178
|
-
export function imageEach(jobName, gitlabData) {
|
|
179
|
-
const jobData = gitlabData[jobName];
|
|
180
|
-
const image = jobData.image;
|
|
181
|
-
if (!image)
|
|
182
|
-
return;
|
|
183
|
-
jobData.image = imageComplex(jobData.image);
|
|
184
|
-
}
|
|
185
|
-
export function inheritDefault(gitlabData) {
|
|
186
|
-
for (const [jobName, jobData] of Object.entries(gitlabData)) {
|
|
187
|
-
if (jobData.inherit?.default === false)
|
|
188
|
-
continue;
|
|
189
|
-
if (Job.illegalJobNames.has(jobName) || jobName.startsWith(".")) {
|
|
190
|
-
// skip hidden jobs as they might just contain shared yaml
|
|
191
|
-
// see https://github.com/firecow/gitlab-ci-local/issues/1277
|
|
192
|
-
continue;
|
|
193
|
-
}
|
|
194
|
-
const keywordsToInheritFrom = (Array.isArray(jobData.inherit?.default)) ?
|
|
195
|
-
jobData.inherit.default :
|
|
196
|
-
["artifacts", "cache", "services", "image", "before_script", "after_script"];
|
|
197
|
-
for (const keyword of keywordsToInheritFrom) {
|
|
198
|
-
if (gitlabData.default[keyword] !== undefined)
|
|
199
|
-
jobData[keyword] = jobData[keyword] ?? gitlabData.default[keyword];
|
|
200
|
-
}
|
|
201
|
-
}
|
|
202
|
-
}
|
|
203
|
-
function normalizeGlobalVariables(gitlabData) {
|
|
204
|
-
for (const [key, value] of Object.entries(gitlabData.variables ?? {})) {
|
|
205
|
-
gitlabData.variables[key] = Utils.normalizeVariables(value);
|
|
206
|
-
}
|
|
207
|
-
}
|
|
208
|
-
export function flattenLists(gitlabData) {
|
|
209
|
-
traverse(gitlabData, ({ parent, key, value, meta }) => {
|
|
210
|
-
if (parent != null && key != null && Array.isArray(value)) {
|
|
211
|
-
assert(!value.flat(9).some(Array.isArray), chalk `This Gitlab CI configuration is invalid: {blueBright ${meta.nodePath}} config should be string or a nested array of strings up to 10 level deep`);
|
|
212
|
-
}
|
|
213
|
-
}, { cycleHandling: true });
|
|
214
|
-
traverse(gitlabData, ({ parent, key, value }) => {
|
|
215
|
-
if (parent != null && key != null && Array.isArray(value)) {
|
|
216
|
-
parent[key] = value.flat(9);
|
|
217
|
-
}
|
|
218
|
-
}, { cycleHandling: false });
|
|
219
|
-
}
|
|
220
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"data-expander.js","sourceRoot":"","sources":["data-expander.ts"],"names":[],"mappings":"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","sourcesContent":["import chalk from \"chalk-template\";\nimport deepExtend from \"deep-extend\";\nimport assert, {AssertionError} from \"assert\";\nimport {Job, Need} from \"./job.js\";\nimport {traverse} from \"object-traversal\";\nimport {Utils} from \"./utils.js\";\n\nconst extendsMaxDepth = 11;\nconst extendsRecurse = (gitlabData: any, jobName: string, jobData: any, parents: any[], depth: number) => {\n    assert(depth < extendsMaxDepth, chalk`{blueBright ${jobName}}: circular dependency detected in \\`extends\\``);\n    depth++;\n\n    jobData.extends = typeof jobData.extends === \"string\" ? [jobData.extends] : jobData.extends;\n    jobData.extends = jobData.extends ?? [];\n\n    for (const parentName of jobData.extends) {\n        const parentData = gitlabData[parentName];\n        assert(parentData != null, chalk`{blueBright ${parentName}} is unspecified, used by {blueBright ${jobName}} extends`);\n        extendsRecurse(gitlabData, parentName, parentData, parents, depth);\n        parents.push(parentData);\n    }\n    return parents;\n};\n\nexport function jobExtends (gitlabData: any) {\n    for (const [jobName, jobData] of Object.entries<any>(gitlabData)) {\n        if (Job.illegalJobNames.has(jobName)) continue;\n        if (!Utils.isObject(jobData)) continue;\n        const parentDatas = extendsRecurse(gitlabData, jobName, jobData, [], 0);\n        gitlabData[jobName] = deepExtend({}, ...parentDatas, jobData);\n    }\n\n    for (const [jobName, jobData] of Object.entries<any>(gitlabData)) {\n        if (Job.illegalJobNames.has(jobName)) continue;\n        if (!Utils.isObject(jobData)) continue;\n        delete jobData.extends;\n    }\n}\n\nexport function reference (gitlabData: any, recurseData: any) {\n    for (const [key, value] of Object.entries<any>(recurseData || {})) {\n        if (value?.referenceData) {\n            if (Object.keys(value).length > 1) {\n                recurseData[key] = {...getSubDataByReference(gitlabData, value.referenceData), ...recurseData[key]};\n                delete recurseData[key].referenceData;\n            } else {\n                recurseData[key] = getSubDataByReference(gitlabData, value.referenceData);\n            }\n        } else if (typeof value === \"object\") {\n            reference(gitlabData, value);\n        }\n\n        if (hasCircularChain(recurseData)) {\n            throw new AssertionError({message: `!reference circular chain detected [${value.referenceData}]`});\n        }\n    }\n}\n\nconst getSubDataByReference = (gitlabData: any, referenceData: string[]) => {\n    let gitlabSubData = gitlabData;\n    referenceData.forEach((referencePointer) => {\n        assert(gitlabSubData[referencePointer] != null, `!reference [${referenceData.join(\", \")}] is undefined`);\n        gitlabSubData = gitlabSubData[referencePointer];\n    });\n    return gitlabSubData;\n};\n\nfunction hasCircularChain (data: any) {\n    try {\n        JSON.stringify(data);\n    } catch (e) {\n        if (e instanceof TypeError && e.message.startsWith(\"Converting circular structure to JSON\")) {\n            return true;\n        }\n    }\n    return false;\n}\n\n/**\n  Transform the globally defined [\"image\", \"services\", \"cache\", \"before_script\", \"after_script\"] into the default.x syntax\n  https://docs.gitlab.com/ee/ci/yaml/index.html#globally-defined-image-services-cache-before_script-after_script\n*/\nexport function transformDeprecatedGlobalDefaultSyntax (gitlabData: any) {\n    const GITLAB_DEPRECATED_GLOBALLY_DEFINED_KEYWORDS = [\"image\", \"services\", \"cache\", \"before_script\", \"after_script\"];\n\n    gitlabData.default = gitlabData.default ?? {};\n    for (const g of GITLAB_DEPRECATED_GLOBALLY_DEFINED_KEYWORDS) {\n        if (gitlabData[g] !== undefined) {\n            gitlabData.default[g] = gitlabData[g];\n            delete gitlabData[g]; // Since this is deprecated, deleting it to prevent us from using it internally\n        }\n    }\n}\n\nexport function normalize (gitlabData: any) {\n    normalizeGlobalVariables(gitlabData);\n\n    for (const [jobName, jobData] of Object.entries<any>(gitlabData)) {\n        if (Job.illegalJobNames.has(jobName) || jobName.startsWith(\".\")) continue;\n        needsEach(jobName, gitlabData);\n        cacheEach(jobName, gitlabData);\n        servicesEach(jobName, gitlabData);\n        imageEach(jobName, gitlabData);\n\n        jobData.after_script = (typeof jobData.after_script === \"string\" && jobData.after_script !== \"\") ? [jobData.after_script] : jobData.after_script;\n        jobData.before_script = (typeof jobData.before_script === \"string\" && jobData.before_script !== \"\") ? [jobData.before_script] : jobData.before_script;\n        jobData.script = (typeof jobData.script === \"string\" && jobData.script !== \"\") ? [jobData.script] : jobData.script;\n    }\n}\n\nexport function needsComplex (data: any) {\n    const needs: Need = {\n        job: data.job ?? data,\n        artifacts: data.artifacts ?? true,\n        ...(data.pipeline ? {pipeline: data.pipeline} : {}),\n        ...(data.project ? {project: data.project} : {}),\n        ...(data.ref ? {ref: data.ref} : {}),\n        ...(data.optional ? {optional: data.optional} : {}),\n    };\n\n    // In needs:project/needs:pipeline, `optional` is not an allowed property\n    if (!data.project && !data.pipeline && data.optional === undefined) {\n        needs.optional = false;\n    }\n    return needs;\n}\n\nexport function needsEach (jobName: string, gitlabData: any) {\n    const jobData = gitlabData[jobName];\n    if (!jobData.needs) return;\n\n    for (const [i, n] of Object.entries<any>(jobData.needs)) {\n        jobData.needs[i] = needsComplex(n);\n    }\n}\n\nexport function cacheComplex (data: any) {\n    return {\n        key: data.key,\n        paths: data.paths,\n        policy: data.policy ?? \"pull-push\",\n        when: data.when ?? \"on_success\",\n    };\n}\n\nexport function cacheEach (jobName: string, gitlabData: any) {\n    const jobData = gitlabData[jobName];\n    const cache = jobData.cache;\n    if (!cache) return;\n\n    jobData.cache = Array.isArray(cache) ? cache : [cache];\n    for (const [i, c] of Object.entries<any>(jobData.cache)) {\n        if (c.key?.files instanceof Array) {\n            assert(c.key.files.length === 1 || c.key.files.length === 2, `cache:key:files should be an array of one or two file paths. Got ${c.key.files.length}`);\n        }\n        jobData.cache[i] = cacheComplex(c);\n    }\n\n    // Remove cache elements with empty paths array (gitlab.com works the same way.)\n    jobData.cache = jobData.cache.filter((c: any) => c.paths?.length !== undefined);\n}\n\nexport function servicesComplex (data: any) {\n    return {\n        name: typeof data === \"string\" ? data : data.name,\n        entrypoint: data.entrypoint,\n        command: data.command,\n        alias: data.alias,\n        variables: data.variables,\n    };\n}\n\nexport function servicesEach (jobName: string, gitlabData: any) {\n    const jobData = gitlabData[jobName];\n    const services = jobData.services;\n    if (!services) return;\n\n    jobData.services = Array.isArray(services) ? services : [services];\n\n    for (const [i, s] of Object.entries<any>(jobData.services)) {\n        jobData.services[i] = servicesComplex(s);\n    }\n}\n\nexport function imageComplex (data: any) {\n    if (data == null) return data;\n    return {\n        name: typeof data === \"string\" ? data : data.name,\n        entrypoint: data.entrypoint,\n        ...(data.docker?.user ? {docker: {user: data.docker?.user}} : {}),\n    };\n}\n\nexport function imageEach (jobName: string, gitlabData: any) {\n    const jobData = gitlabData[jobName];\n    const image = jobData.image;\n    if (!image) return;\n\n    jobData.image = imageComplex(jobData.image);\n}\n\nexport function inheritDefault (gitlabData: any) {\n    for (const [jobName, jobData] of Object.entries<any>(gitlabData)) {\n        if (jobData.inherit?.default === false) continue;\n\n        if (Job.illegalJobNames.has(jobName) || jobName.startsWith(\".\")) {\n            // skip hidden jobs as they might just contain shared yaml\n            // see https://github.com/firecow/gitlab-ci-local/issues/1277\n            continue;\n        }\n\n        const keywordsToInheritFrom = (Array.isArray(jobData.inherit?.default)) ?\n            jobData.inherit.default :\n            [\"artifacts\", \"cache\", \"services\", \"image\", \"before_script\", \"after_script\"];\n\n        for (const keyword of keywordsToInheritFrom) {\n            if (gitlabData.default[keyword] !== undefined) jobData[keyword] = jobData[keyword] ?? gitlabData.default[keyword];\n        }\n    }\n}\n\nfunction normalizeGlobalVariables (gitlabData: any) {\n    for (const [key, value] of Object.entries<any>(gitlabData.variables ?? {})) {\n        gitlabData.variables[key] = Utils.normalizeVariables(value);\n    }\n}\n\nexport function flattenLists (gitlabData: any) {\n    traverse(gitlabData, ({parent, key, value, meta}) => {\n        if (parent != null && key != null && Array.isArray(value)) {\n            assert(!value.flat(9).some(Array.isArray), chalk`This Gitlab CI configuration is invalid: {blueBright ${meta.nodePath}} config should be string or a nested array of strings up to 10 level deep`);\n        }\n    }, {cycleHandling: true});\n\n    traverse(gitlabData, ({parent, key, value}) => {\n        if (parent != null && key != null && Array.isArray(value)) {\n            parent[key] = value.flat(9);\n        }\n    }, {cycleHandling: false});\n}\n"]}
|