gitlab-ci-local
Version:
Tired of pushing to test your .gitlab-ci.yml?
218 lines • 34 kB
JavaScript
import chalk from "chalk";
import deepExtend from "deep-extend";
import assert, { AssertionError } from "assert";
import { Job } from "./job.js";
import { traverse } from "object-traversal";
import { Utils } from "./utils.js";
const extendsMaxDepth = 11;
const extendsRecurse = (gitlabData, jobName, jobData, parents, depth) => {
assert(depth < extendsMaxDepth, chalk `{blueBright ${jobName}}: circular dependency detected in \`extends\``);
depth++;
jobData.extends = typeof jobData.extends === "string" ? [jobData.extends] : jobData.extends;
jobData.extends = jobData.extends ?? [];
for (const parentName of jobData.extends) {
const parentData = gitlabData[parentName];
assert(parentData != null, chalk `{blueBright ${parentName}} is unspecified, used by {blueBright ${jobName}} extends`);
extendsRecurse(gitlabData, parentName, parentData, parents, depth);
parents.push(parentData);
}
return parents;
};
export function jobExtends(gitlabData) {
for (const [jobName, jobData] of Object.entries(gitlabData)) {
if (Job.illegalJobNames.has(jobName))
continue;
if (!Utils.isObject(jobData))
continue;
const parentDatas = extendsRecurse(gitlabData, jobName, jobData, [], 0);
gitlabData[jobName] = deepExtend({}, ...parentDatas, jobData);
}
for (const [jobName, jobData] of Object.entries(gitlabData)) {
if (Job.illegalJobNames.has(jobName))
continue;
if (!Utils.isObject(jobData))
continue;
delete jobData.extends;
}
}
export function reference(gitlabData, recurseData) {
for (const [key, value] of Object.entries(recurseData || {})) {
if (value?.referenceData) {
recurseData[key] = getSubDataByReference(gitlabData, value.referenceData);
}
else if (typeof value === "object") {
reference(gitlabData, value);
}
if (hasCircularChain(recurseData)) {
throw new AssertionError({ message: `!reference circular chain detected [${value.referenceData}]` });
}
}
}
const getSubDataByReference = (gitlabData, referenceData) => {
let gitlabSubData = gitlabData;
referenceData.forEach((referencePointer) => {
assert(gitlabSubData[referencePointer] != null, `!reference [${referenceData.join(", ")}] is undefined`);
gitlabSubData = gitlabSubData[referencePointer];
});
return gitlabSubData;
};
function hasCircularChain(data) {
try {
JSON.stringify(data);
}
catch (e) {
if (e instanceof TypeError && e.message.startsWith("Converting circular structure to JSON")) {
return true;
}
}
return false;
}
/**
Transform the globally defined ["image", "services", "cache", "before_script", "after_script"] into the default.x syntax
https://docs.gitlab.com/ee/ci/yaml/index.html#globally-defined-image-services-cache-before_script-after_script
*/
export function transformDeprecatedGlobalDefaultSyntax(gitlabData) {
const GITLAB_DEPRECATED_GLOBALLY_DEFINED_KEYWORDS = ["image", "services", "cache", "before_script", "after_script"];
gitlabData.default = gitlabData.default ?? {};
for (const g of GITLAB_DEPRECATED_GLOBALLY_DEFINED_KEYWORDS) {
if (gitlabData[g] !== undefined) {
gitlabData.default[g] = gitlabData[g];
delete gitlabData[g]; // Since this is deprecated, deleting it to prevent us from using it internally
}
}
}
export function normalize(gitlabData) {
normalizeGlobalVariables(gitlabData);
for (const [jobName, jobData] of Object.entries(gitlabData)) {
if (Job.illegalJobNames.has(jobName) || jobName.startsWith("."))
continue;
needsEach(jobName, gitlabData);
cacheEach(jobName, gitlabData);
servicesEach(jobName, gitlabData);
imageEach(jobName, gitlabData);
jobData.after_script = (typeof jobData.after_script === "string" && jobData.after_script !== "") ? [jobData.after_script] : jobData.after_script;
jobData.before_script = (typeof jobData.before_script === "string" && jobData.before_script !== "") ? [jobData.before_script] : jobData.before_script;
jobData.script = (typeof jobData.script === "string" && jobData.script !== "") ? [jobData.script] : jobData.script;
}
}
export function needsComplex(data) {
const needs = {
job: data.job ?? data,
artifacts: data.artifacts ?? true,
...(data.pipeline ? { pipeline: data.pipeline } : {}),
...(data.project ? { project: data.project } : {}),
...(data.ref ? { ref: data.ref } : {}),
...(data.optional ? { optional: data.optional } : {}),
};
// In needs:project/needs:pipeline, `optional` is not an allowed property
if (!data.project && !data.pipeline && data.optional === undefined) {
needs.optional = false;
}
return needs;
}
export function needsEach(jobName, gitlabData) {
const jobData = gitlabData[jobName];
if (!jobData.needs)
return;
for (const [i, n] of Object.entries(jobData.needs)) {
jobData.needs[i] = needsComplex(n);
}
}
export function cacheComplex(data) {
return {
key: data.key,
paths: data.paths,
policy: data.policy ?? "pull-push",
when: data.when ?? "on_success",
};
}
export function cacheEach(jobName, gitlabData) {
const jobData = gitlabData[jobName];
const cache = jobData.cache;
if (!cache)
return;
jobData.cache = Array.isArray(cache) ? cache : [cache];
for (const [i, c] of Object.entries(jobData.cache)) {
if (c.key?.files instanceof Array) {
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}`);
}
jobData.cache[i] = cacheComplex(c);
}
// Remove cache elements with empty paths array (gitlab.com works the same way.)
jobData.cache = jobData.cache.filter((c) => c.paths?.length !== undefined);
}
export function servicesComplex(data) {
return {
name: typeof data === "string" ? data : data.name,
entrypoint: data.entrypoint,
command: data.command,
alias: data.alias,
variables: data.variables,
};
}
export function servicesEach(jobName, gitlabData) {
const jobData = gitlabData[jobName];
const services = jobData.services;
if (!services)
return;
jobData.services = Array.isArray(services) ? services : [services];
for (const [i, s] of Object.entries(jobData.services)) {
jobData.services[i] = servicesComplex(s);
}
}
export function imageComplex(data) {
if (data == null)
return data;
return {
name: typeof data === "string" ? data : data.name,
entrypoint: data.entrypoint,
...(data.docker?.user ? { docker: { user: data.docker?.user } } : {}),
};
}
export function imageEach(jobName, gitlabData) {
const jobData = gitlabData[jobName];
const image = jobData.image;
if (!image)
return;
jobData.image = imageComplex(jobData.image);
}
export function inheritDefault(gitlabData) {
for (const [jobName, jobData] of Object.entries(gitlabData)) {
if (jobData.inherit?.default === false)
continue;
if (Job.illegalJobNames.has(jobName) || jobName.startsWith(".")) {
// skip hidden jobs as they might just contain shared yaml
// see https://github.com/firecow/gitlab-ci-local/issues/1277
continue;
}
const keywordsToInheritFrom = (Array.isArray(jobData.inherit?.default))
? jobData.inherit.default
: ["artifacts", "cache", "services", "image", "before_script", "after_script"];
for (const keyword of keywordsToInheritFrom) {
if (gitlabData.default[keyword] !== undefined)
jobData[keyword] = jobData[keyword] ?? gitlabData.default[keyword];
}
}
}
function normalizeGlobalVariables(gitlabData) {
for (const [key, value] of Object.entries(gitlabData.variables ?? {})) {
if (value === null) {
gitlabData.variables[key] = ""; // variable's values are nullable
}
else if (Utils.isObject(value)) {
gitlabData.variables[key] = String(value["value"]);
}
else {
gitlabData.variables[key] = String(value);
}
}
}
export function flattenLists(gitlabData) {
traverse(gitlabData, ({ parent, key, value, meta }) => {
if (parent != null && key != null && Array.isArray(value)) {
parent[key] = value.flat(9);
assert(!parent[key].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`);
}
}, { cycleHandling: true });
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"data-expander.js","sourceRoot":"","sources":["data-expander.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,UAAU,MAAM,aAAa,CAAC;AACrC,OAAO,MAAM,EAAE,EAAC,cAAc,EAAC,MAAM,QAAQ,CAAC;AAC9C,OAAO,EAAC,GAAG,EAAO,MAAM,UAAU,CAAC;AACnC,OAAO,EAAC,QAAQ,EAAC,MAAM,kBAAkB,CAAC;AAC1C,OAAO,EAAC,KAAK,EAAC,MAAM,YAAY,CAAC;AAEjC,MAAM,eAAe,GAAG,EAAE,CAAC;AAC3B,MAAM,cAAc,GAAG,CAAC,UAAe,EAAE,OAAe,EAAE,OAAY,EAAE,OAAc,EAAE,KAAa,EAAE,EAAE;IACrG,MAAM,CAAC,KAAK,GAAG,eAAe,EAAE,KAAK,CAAA,eAAe,OAAO,gDAAgD,CAAC,CAAC;IAC7G,KAAK,EAAE,CAAC;IAER,OAAO,CAAC,OAAO,GAAG,OAAO,OAAO,CAAC,OAAO,KAAK,QAAQ,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC;IAC5F,OAAO,CAAC,OAAO,GAAG,OAAO,CAAC,OAAO,IAAI,EAAE,CAAC;IAExC,KAAK,MAAM,UAAU,IAAI,OAAO,CAAC,OAAO,EAAE,CAAC;QACvC,MAAM,UAAU,GAAG,UAAU,CAAC,UAAU,CAAC,CAAC;QAC1C,MAAM,CAAC,UAAU,IAAI,IAAI,EAAE,KAAK,CAAA,eAAe,UAAU,yCAAyC,OAAO,WAAW,CAAC,CAAC;QACtH,cAAc,CAAC,UAAU,EAAE,UAAU,EAAE,UAAU,EAAE,OAAO,EAAE,KAAK,CAAC,CAAC;QACnE,OAAO,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IAC7B,CAAC;IACD,OAAO,OAAO,CAAC;AACnB,CAAC,CAAC;AAEF,MAAM,UAAU,UAAU,CAAE,UAAe;IACvC,KAAK,MAAM,CAAC,OAAO,EAAE,OAAO,CAAC,IAAI,MAAM,CAAC,OAAO,CAAM,UAAU,CAAC,EAAE,CAAC;QAC/D,IAAI,GAAG,CAAC,eAAe,CAAC,GAAG,CAAC,OAAO,CAAC;YAAE,SAAS;QAC/C,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,OAAO,CAAC;YAAE,SAAS;QACvC,MAAM,WAAW,GAAG,cAAc,CAAC,UAAU,EAAE,OAAO,EAAE,OAAO,EAAE,EAAE,EAAE,CAAC,CAAC,CAAC;QACxE,UAAU,CAAC,OAAO,CAAC,GAAG,UAAU,CAAC,EAAE,EAAE,GAAG,WAAW,EAAE,OAAO,CAAC,CAAC;IAClE,CAAC;IAED,KAAK,MAAM,CAAC,OAAO,EAAE,OAAO,CAAC,IAAI,MAAM,CAAC,OAAO,CAAM,UAAU,CAAC,EAAE,CAAC;QAC/D,IAAI,GAAG,CAAC,eAAe,CAAC,GAAG,CAAC,OAAO,CAAC;YAAE,SAAS;QAC/C,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,OAAO,CAAC;YAAE,SAAS;QACvC,OAAO,OAAO,CAAC,OAAO,CAAC;IAC3B,CAAC;AACL,CAAC;AAED,MAAM,UAAU,SAAS,CAAE,UAAe,EAAE,WAAgB;IACxD,KAAK,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAM,WAAW,IAAI,EAAE,CAAC,EAAE,CAAC;QAChE,IAAI,KAAK,EAAE,aAAa,EAAE,CAAC;YACvB,WAAW,CAAC,GAAG,CAAC,GAAG,qBAAqB,CAAC,UAAU,EAAE,KAAK,CAAC,aAAa,CAAC,CAAC;QAC9E,CAAC;aAAM,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE,CAAC;YACnC,SAAS,CAAC,UAAU,EAAE,KAAK,CAAC,CAAC;QACjC,CAAC;QAED,IAAI,gBAAgB,CAAC,WAAW,CAAC,EAAE,CAAC;YAChC,MAAM,IAAI,cAAc,CAAC,EAAC,OAAO,EAAE,uCAAuC,KAAK,CAAC,aAAa,GAAG,EAAC,CAAC,CAAC;QACvG,CAAC;IACL,CAAC;AACL,CAAC;AAED,MAAM,qBAAqB,GAAG,CAAC,UAAe,EAAE,aAAuB,EAAE,EAAE;IACvE,IAAI,aAAa,GAAG,UAAU,CAAC;IAC/B,aAAa,CAAC,OAAO,CAAC,CAAC,gBAAgB,EAAE,EAAE;QACvC,MAAM,CAAC,aAAa,CAAC,gBAAgB,CAAC,IAAI,IAAI,EAAE,eAAe,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;QACzG,aAAa,GAAG,aAAa,CAAC,gBAAgB,CAAC,CAAC;IACpD,CAAC,CAAC,CAAC;IACH,OAAO,aAAa,CAAC;AACzB,CAAC,CAAC;AAEF,SAAS,gBAAgB,CAAE,IAAS;IAChC,IAAI,CAAC;QACD,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;IACzB,CAAC;IAAC,OAAO,CAAC,EAAE,CAAC;QACT,IAAI,CAAC,YAAY,SAAS,IAAI,CAAC,CAAC,OAAO,CAAC,UAAU,CAAC,uCAAuC,CAAC,EAAE,CAAC;YAC1F,OAAO,IAAI,CAAC;QAChB,CAAC;IACL,CAAC;IACD,OAAO,KAAK,CAAC;AACjB,CAAC;AAED;;;EAGE;AACF,MAAM,UAAU,sCAAsC,CAAE,UAAe;IACnE,MAAM,2CAA2C,GAAG,CAAC,OAAO,EAAE,UAAU,EAAE,OAAO,EAAE,eAAe,EAAE,cAAc,CAAC,CAAC;IAEpH,UAAU,CAAC,OAAO,GAAG,UAAU,CAAC,OAAO,IAAI,EAAE,CAAC;IAC9C,KAAK,MAAM,CAAC,IAAI,2CAA2C,EAAE,CAAC;QAC1D,IAAI,UAAU,CAAC,CAAC,CAAC,KAAK,SAAS,EAAE,CAAC;YAC9B,UAAU,CAAC,OAAO,CAAC,CAAC,CAAC,GAAG,UAAU,CAAC,CAAC,CAAC,CAAC;YACtC,OAAO,UAAU,CAAC,CAAC,CAAC,CAAC,CAAC,+EAA+E;QACzG,CAAC;IACL,CAAC;AACL,CAAC;AAED,MAAM,UAAU,SAAS,CAAE,UAAe;IACtC,wBAAwB,CAAC,UAAU,CAAC,CAAC;IAErC,KAAK,MAAM,CAAC,OAAO,EAAE,OAAO,CAAC,IAAI,MAAM,CAAC,OAAO,CAAM,UAAU,CAAC,EAAE,CAAC;QAC/D,IAAI,GAAG,CAAC,eAAe,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,OAAO,CAAC,UAAU,CAAC,GAAG,CAAC;YAAE,SAAS;QAC1E,SAAS,CAAC,OAAO,EAAE,UAAU,CAAC,CAAC;QAC/B,SAAS,CAAC,OAAO,EAAE,UAAU,CAAC,CAAC;QAC/B,YAAY,CAAC,OAAO,EAAE,UAAU,CAAC,CAAC;QAClC,SAAS,CAAC,OAAO,EAAE,UAAU,CAAC,CAAC;QAE/B,OAAO,CAAC,YAAY,GAAG,CAAC,OAAO,OAAO,CAAC,YAAY,KAAK,QAAQ,IAAI,OAAO,CAAC,YAAY,KAAK,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,YAAY,CAAC;QACjJ,OAAO,CAAC,aAAa,GAAG,CAAC,OAAO,OAAO,CAAC,aAAa,KAAK,QAAQ,IAAI,OAAO,CAAC,aAAa,KAAK,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,aAAa,CAAC;QACtJ,OAAO,CAAC,MAAM,GAAG,CAAC,OAAO,OAAO,CAAC,MAAM,KAAK,QAAQ,IAAI,OAAO,CAAC,MAAM,KAAK,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,MAAM,CAAC;IACvH,CAAC;AACL,CAAC;AAED,MAAM,UAAU,YAAY,CAAE,IAAS;IACnC,MAAM,KAAK,GAAS;QAChB,GAAG,EAAE,IAAI,CAAC,GAAG,IAAI,IAAI;QACrB,SAAS,EAAE,IAAI,CAAC,SAAS,IAAI,IAAI;QACjC,GAAG,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,EAAC,QAAQ,EAAE,IAAI,CAAC,QAAQ,EAAC,CAAC,CAAC,CAAC,EAAE,CAAC;QACnD,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,EAAC,OAAO,EAAE,IAAI,CAAC,OAAO,EAAC,CAAC,CAAC,CAAC,EAAE,CAAC;QAChD,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,EAAC,GAAG,EAAE,IAAI,CAAC,GAAG,EAAC,CAAC,CAAC,CAAC,EAAE,CAAC;QACpC,GAAG,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,EAAC,QAAQ,EAAE,IAAI,CAAC,QAAQ,EAAC,CAAC,CAAC,CAAC,EAAE,CAAC;KACtD,CAAC;IAEF,yEAAyE;IACzE,IAAI,CAAC,IAAI,CAAC,OAAO,IAAI,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,QAAQ,KAAK,SAAS,EAAE,CAAC;QACjE,KAAK,CAAC,QAAQ,GAAG,KAAK,CAAC;IAC3B,CAAC;IACD,OAAO,KAAK,CAAC;AACjB,CAAC;AAED,MAAM,UAAU,SAAS,CAAE,OAAe,EAAE,UAAe;IACvD,MAAM,OAAO,GAAG,UAAU,CAAC,OAAO,CAAC,CAAC;IACpC,IAAI,CAAC,OAAO,CAAC,KAAK;QAAE,OAAO;IAE3B,KAAK,MAAM,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,MAAM,CAAC,OAAO,CAAM,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC;QACtD,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,GAAG,YAAY,CAAC,CAAC,CAAC,CAAC;IACvC,CAAC;AACL,CAAC;AAED,MAAM,UAAU,YAAY,CAAE,IAAS;IACnC,OAAO;QACH,GAAG,EAAE,IAAI,CAAC,GAAG;QACb,KAAK,EAAE,IAAI,CAAC,KAAK;QACjB,MAAM,EAAE,IAAI,CAAC,MAAM,IAAI,WAAW;QAClC,IAAI,EAAE,IAAI,CAAC,IAAI,IAAI,YAAY;KAClC,CAAC;AACN,CAAC;AAED,MAAM,UAAU,SAAS,CAAE,OAAe,EAAE,UAAe;IACvD,MAAM,OAAO,GAAG,UAAU,CAAC,OAAO,CAAC,CAAC;IACpC,MAAM,KAAK,GAAG,OAAO,CAAC,KAAK,CAAC;IAC5B,IAAI,CAAC,KAAK;QAAE,OAAO;IAEnB,OAAO,CAAC,KAAK,GAAG,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;IACvD,KAAK,MAAM,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,MAAM,CAAC,OAAO,CAAM,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC;QACtD,IAAI,CAAC,CAAC,GAAG,EAAE,KAAK,YAAY,KAAK,EAAE,CAAC;YAChC,MAAM,CAAC,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,MAAM,KAAK,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,MAAM,KAAK,CAAC,EAAE,oEAAoE,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC,CAAC;QAC3J,CAAC;QACD,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,GAAG,YAAY,CAAC,CAAC,CAAC,CAAC;IACvC,CAAC;IAED,gFAAgF;IAChF,OAAO,CAAC,KAAK,GAAG,OAAO,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,KAAK,EAAE,MAAM,KAAK,SAAS,CAAC,CAAC;AACpF,CAAC;AAED,MAAM,UAAU,eAAe,CAAE,IAAS;IACtC,OAAO;QACH,IAAI,EAAE,OAAO,IAAI,KAAK,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI;QACjD,UAAU,EAAE,IAAI,CAAC,UAAU;QAC3B,OAAO,EAAE,IAAI,CAAC,OAAO;QACrB,KAAK,EAAE,IAAI,CAAC,KAAK;QACjB,SAAS,EAAE,IAAI,CAAC,SAAS;KAC5B,CAAC;AACN,CAAC;AAED,MAAM,UAAU,YAAY,CAAE,OAAe,EAAE,UAAe;IAC1D,MAAM,OAAO,GAAG,UAAU,CAAC,OAAO,CAAC,CAAC;IACpC,MAAM,QAAQ,GAAG,OAAO,CAAC,QAAQ,CAAC;IAClC,IAAI,CAAC,QAAQ;QAAE,OAAO;IAEtB,OAAO,CAAC,QAAQ,GAAG,KAAK,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC;IAEnE,KAAK,MAAM,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,MAAM,CAAC,OAAO,CAAM,OAAO,CAAC,QAAQ,CAAC,EAAE,CAAC;QACzD,OAAO,CAAC,QAAQ,CAAC,CAAC,CAAC,GAAG,eAAe,CAAC,CAAC,CAAC,CAAC;IAC7C,CAAC;AACL,CAAC;AAED,MAAM,UAAU,YAAY,CAAE,IAAS;IACnC,IAAI,IAAI,IAAI,IAAI;QAAE,OAAO,IAAI,CAAC;IAC9B,OAAO;QACH,IAAI,EAAE,OAAO,IAAI,KAAK,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI;QACjD,UAAU,EAAE,IAAI,CAAC,UAAU;QAC3B,GAAG,CAAC,IAAI,CAAC,MAAM,EAAE,IAAI,CAAC,CAAC,CAAC,EAAC,MAAM,EAAE,EAAC,IAAI,EAAE,IAAI,CAAC,MAAM,EAAE,IAAI,EAAC,EAAC,CAAC,CAAC,CAAC,EAAE,CAAC;KACpE,CAAC;AACN,CAAC;AAED,MAAM,UAAU,SAAS,CAAE,OAAe,EAAE,UAAe;IACvD,MAAM,OAAO,GAAG,UAAU,CAAC,OAAO,CAAC,CAAC;IACpC,MAAM,KAAK,GAAG,OAAO,CAAC,KAAK,CAAC;IAC5B,IAAI,CAAC,KAAK;QAAE,OAAO;IAEnB,OAAO,CAAC,KAAK,GAAG,YAAY,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;AAChD,CAAC;AAED,MAAM,UAAU,cAAc,CAAE,UAAe;IAC3C,KAAK,MAAM,CAAC,OAAO,EAAE,OAAO,CAAC,IAAI,MAAM,CAAC,OAAO,CAAM,UAAU,CAAC,EAAE,CAAC;QAC/D,IAAI,OAAO,CAAC,OAAO,EAAE,OAAO,KAAK,KAAK;YAAE,SAAS;QAEjD,IAAI,GAAG,CAAC,eAAe,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,OAAO,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC;YAC9D,0DAA0D;YAC1D,6DAA6D;YAC7D,SAAS;QACb,CAAC;QAED,MAAM,qBAAqB,GAAG,CAAC,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;YACnE,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,OAAO;YACzB,CAAC,CAAC,CAAC,WAAW,EAAE,OAAO,EAAE,UAAU,EAAE,OAAO,EAAE,eAAe,EAAE,cAAc,CAAC,CAAC;QAEnF,KAAK,MAAM,OAAO,IAAI,qBAAqB,EAAE,CAAC;YAC1C,IAAI,UAAU,CAAC,OAAO,CAAC,OAAO,CAAC,KAAK,SAAS;gBAAE,OAAO,CAAC,OAAO,CAAC,GAAG,OAAO,CAAC,OAAO,CAAC,IAAI,UAAU,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;QACtH,CAAC;IACL,CAAC;AACL,CAAC;AAED,SAAS,wBAAwB,CAAE,UAAe;IAC9C,KAAK,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAM,UAAU,CAAC,SAAS,IAAI,EAAE,CAAC,EAAE,CAAC;QACzE,IAAI,KAAK,KAAK,IAAI,EAAE,CAAC;YACjB,UAAU,CAAC,SAAS,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,CAAC,iCAAiC;QACrE,CAAC;aAAM,IAAI,KAAK,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE,CAAC;YAC/B,UAAU,CAAC,SAAS,CAAC,GAAG,CAAC,GAAG,MAAM,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC;QACvD,CAAC;aAAM,CAAC;YACJ,UAAU,CAAC,SAAS,CAAC,GAAG,CAAC,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;QAC9C,CAAC;IACL,CAAC;AACL,CAAC;AAED,MAAM,UAAU,YAAY,CAAE,UAAe;IACzC,QAAQ,CAAC,UAAU,EAAE,CAAC,EAAC,MAAM,EAAE,GAAG,EAAE,KAAK,EAAE,IAAI,EAAC,EAAE,EAAE;QAChD,IAAI,MAAM,IAAI,IAAI,IAAI,GAAG,IAAI,IAAI,IAAI,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC;YACxD,MAAM,CAAC,GAAG,CAAC,GAAG,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YAC5B,MAAM,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,EAAE,KAAK,CAAA,wDAAwD,IAAI,CAAC,QAAQ,4EAA4E,CAAC,CAAC;QACrM,CAAC;IACL,CAAC,EAAE,EAAC,aAAa,EAAE,IAAI,EAAC,CAAC,CAAC;AAC9B,CAAC","sourcesContent":["import chalk from \"chalk\";\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            recurseData[key] = getSubDataByReference(gitlabData, value.referenceData);\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        if (value === null) {\n            gitlabData.variables[key] = \"\"; // variable's values are nullable\n        } else if (Utils.isObject(value)) {\n            gitlabData.variables[key] = String(value[\"value\"]);\n        } else {\n            gitlabData.variables[key] = String(value);\n        }\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            parent[key] = value.flat(9);\n            assert(!parent[key].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"]}