gitlab-ci-local
Version:
Tired of pushing to test your .gitlab-ci.yml?
151 lines (150 loc) • 26.1 kB
JavaScript
import Ajv from "ajv";
import assert from "assert";
import chalk from "chalk";
import schema from "./schema/index.js";
import { betterAjvErrors } from "./schema-error.js";
import terminalLink from "terminal-link";
const MAX_ERRORS = 5;
export class Validator {
static jsonSchemaValidation({ pathToExpandedGitLabCi, gitLabCiConfig, argv }) {
const ajv = new Ajv({
verbose: true,
allErrors: true,
allowUnionTypes: true,
validateFormats: false,
strictTypes: false, // to suppress the missing types defined in the gitlab-ci json schema
keywords: ["markdownDescription"],
});
const validate = ajv.compile(schema);
const valid = validate(gitLabCiConfig);
if (valid)
return;
const betterErrors = betterAjvErrors({
data: gitLabCiConfig,
errors: validate.errors,
}).filter(betterError => !argv.ignoreSchemaPaths.includes(betterError.schemaPath));
let e = "";
for (let i = 0, len = betterErrors.length; i < len; i++) {
if (i + 1 > MAX_ERRORS) {
e += `\t... and ${len - MAX_ERRORS} more`;
break;
}
e += chalk `\t• {redBright ${betterErrors[i].message}} at {blueBright ${betterErrors[i].path}} {grey [${betterErrors[i].schemaPath}]}\n`;
}
assert(valid || betterErrors.length == 0, chalk `
{reset Invalid .gitlab-ci.yml configuration!
${e.trimEnd()}
For further troubleshooting, consider either of the following:
\t• Copy the content of {blueBright ${terminalLink(".gitlab-ci-local/expanded-gitlab-ci.yml", pathToExpandedGitLabCi)}} to the ${terminalLink("pipeline editor", "https://docs.gitlab.com/ee/ci/pipeline_editor/")} to debug it
\t• Use --ignore-schema-paths= "#/definitions/tags/minItems" --ignore-schema-paths "#/additionalProperties" to partially disable certain validation rule
\t• Use --json-schema-validation=false to disable schema validation (not recommended)}
`);
}
static needs(jobs, stages) {
const warnings = [];
for (const job of jobs) {
if (job.needs === null || job.needs.length === 0)
continue;
for (const [i, need] of job.needs.entries()) {
if (need.pipeline) {
warnings.push(`${job.name}.needs[${i}].job:${need.job} ignored, pipeline key not supported`);
continue;
}
if (need.project) {
warnings.push(`${job.name}.needs[${i}] ignored, project key not supported`);
continue;
}
const needJob = jobs.find(j => j.baseName === need.job);
if (need.optional && !needJob)
continue;
assert(needJob != null, chalk `needs: [{blueBright ${need.job}}] for {blueBright ${job.baseName}} could not be found`);
const needJobStageIndex = stages.indexOf(needJob.stage);
const jobStageIndex = stages.indexOf(job.stage);
assert(needJobStageIndex <= jobStageIndex, chalk `needs: [{blueBright ${needJob.name}}] for {blueBright ${job.name}} is in a future stage`);
}
}
return warnings;
}
static dependencies(jobs, stages) {
for (const job of jobs) {
if (job.dependencies === null || job.dependencies.length === 0)
continue;
const undefDeps = job.dependencies.filter((j) => !jobs.some(n => n.baseName === j));
assert(undefDeps.length !== job.dependencies.length, chalk `dependencies: [{blueBright ${undefDeps.join(",")}}] for {blueBright ${job.name}} cannot be found`);
for (const dep of job.dependencies) {
const depJob = jobs.find(j => j.baseName === dep);
assert(depJob != null, chalk `dependencies: [{blueBright ${dep}}] for {blueBright ${job.baseName}} could not be found`);
const depJobStageIndex = stages.indexOf(depJob.stage);
const jobStageIndex = stages.indexOf(job.stage);
assert(depJobStageIndex <= jobStageIndex, chalk `dependencies: [{blueBright ${depJob.name}}] for {blueBright ${job.name}} is in a future stage`);
}
}
}
static dependenciesContainment(jobs) {
for (const job of jobs) {
const needs = job.needs;
const dependencies = job.dependencies;
if (needs && needs.length === 0)
continue;
if (!dependencies || !needs)
continue;
const everyIncluded = dependencies.every((dep) => {
return needs.some(n => n.job === dep);
});
const assertMsg = `${job.formattedJobName} needs: '${needs.map(n => n.job).join(",")}' doesn't fully contain dependencies: '${dependencies.join(",")}'`;
assert(everyIncluded, assertMsg);
}
}
/**
* These jobs named are reserved keywords in GitLab CI but does not prevent the pipeline from running
* https://github.com/firecow/gitlab-ci-local/issues/1263
* @param jobsNames
* @private
*/
static potentialIllegalJobName(jobsNames) {
const warnings = [];
for (const jobName of jobsNames) {
if (new Set(["types", "true", "false", "nil"]).has(jobName)) {
warnings.push(`Job name "${jobName}" is a reserved keyword. (https://docs.gitlab.com/ee/ci/jobs/#job-name-limitations)`);
}
}
return warnings;
}
static scriptBlank(jobs) {
for (const job of jobs) {
if (job.trigger)
continue; // Jobs with trigger are allowed to have empty script
assert(job.scripts.length > 0, chalk `{blue ${job.name}} has empty script`);
}
}
static arrayOfStrings(jobs) {
for (const job of jobs) {
if (job.trigger)
continue;
job.beforeScripts.forEach((s) => assert(typeof s === "string", chalk `{blue ${job.name}} before_script contains non string value`));
job.afterScripts.forEach((s) => assert(typeof s === "string", chalk `{blue ${job.name}} after_script contains non string value`));
job.scripts.forEach((s) => assert(typeof s === "string", chalk `{blue ${job.name}} script contains non string value`));
}
}
static async run(jobs, stages) {
const warnings = [];
this.scriptBlank(jobs);
this.arrayOfStrings(jobs);
warnings.push(...this.needs(jobs, stages));
this.dependencies(jobs, stages);
this.dependenciesContainment(jobs);
warnings.push(...this.potentialIllegalJobName(jobs.map(j => j.baseName)));
warnings.push(...this.artifacts(jobs));
return warnings;
}
static artifacts(jobs) {
const warnings = [];
for (const job of jobs) {
if (job.artifacts === null) {
warnings.push(`${job.name}.artifacts is null, ignoring.`);
}
}
return warnings;
}
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"validator.js","sourceRoot":"","sources":["validator.ts"],"names":[],"mappings":"AAAA,OAAO,GAAG,MAAM,KAAK,CAAC;AAEtB,OAAO,MAAM,MAAM,QAAQ,CAAC;AAC5B,OAAO,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,MAAM,MAAM,mBAAmB,CAAC;AACvC,OAAO,EAAC,eAAe,EAAC,MAAM,mBAAmB,CAAC;AAClD,OAAO,YAAY,MAAM,eAAe,CAAC;AAGzC,MAAM,UAAU,GAAG,CAAC,CAAC;AAErB,MAAM,OAAO,SAAS;IAClB,MAAM,CAAC,oBAAoB,CAAE,EAAC,sBAAsB,EAAE,cAAc,EAAE,IAAI,EAIzE;QACG,MAAM,GAAG,GAAG,IAAI,GAAG,CAAC;YAChB,OAAO,EAAE,IAAI;YACb,SAAS,EAAE,IAAI;YACf,eAAe,EAAE,IAAI;YACrB,eAAe,EAAE,KAAK;YACtB,WAAW,EAAE,KAAK,EAAE,qEAAqE;YACzF,QAAQ,EAAE,CAAC,qBAAqB,CAAC;SACpC,CAAC,CAAC;QACH,MAAM,QAAQ,GAAG,GAAG,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;QACrC,MAAM,KAAK,GAAG,QAAQ,CAAC,cAAc,CAAC,CAAC;QACvC,IAAI,KAAK;YAAE,OAAO;QAClB,MAAM,YAAY,GAAG,eAAe,CAAC;YACjC,IAAI,EAAE,cAAc;YACpB,MAAM,EAAE,QAAQ,CAAC,MAAM;SAC1B,CAAC,CAAC,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC,QAAQ,CAAC,WAAW,CAAC,UAAU,CAAC,CAAC,CAAC;QAEnF,IAAI,CAAC,GAAW,EAAE,CAAC;QACnB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,GAAG,GAAG,YAAY,CAAC,MAAM,EAAE,CAAC,GAAG,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC;YACtD,IAAI,CAAC,GAAG,CAAC,GAAG,UAAU,EAAE,CAAC;gBACrB,CAAC,IAAI,aAAa,GAAG,GAAG,UAAU,OAAO,CAAC;gBAC1C,MAAM;YACV,CAAC;YACD,CAAC,IAAI,KAAK,CAAA,kBAAkB,YAAY,CAAC,CAAC,CAAC,CAAC,OAAO,oBAAoB,YAAY,CAAC,CAAC,CAAC,CAAC,IAAI,YAAY,YAAY,CAAC,CAAC,CAAC,CAAC,UAAU,MAAM,CAAC;QAC5I,CAAC;QAED,MAAM,CAAC,KAAK,IAAI,YAAY,CAAC,MAAM,IAAI,CAAC,EAAE,KAAK,CAAA;;EAErD,CAAC,CAAC,OAAO,EAAE;;;sCAGyB,YAAY,CAAC,yCAAyC,EAAE,sBAAsB,CAAC,YAAY,YAAY,CAAC,iBAAiB,EAAE,gDAAgD,CAAC;;;CAGjN,CAAC,CAAC;IACC,CAAC;IAEO,MAAM,CAAC,KAAK,CAAE,IAAwB,EAAE,MAAyB;QACrE,MAAM,QAAQ,GAAa,EAAE,CAAC;QAC9B,KAAK,MAAM,GAAG,IAAI,IAAI,EAAE,CAAC;YACrB,IAAI,GAAG,CAAC,KAAK,KAAK,IAAI,IAAI,GAAG,CAAC,KAAK,CAAC,MAAM,KAAK,CAAC;gBAAE,SAAS;YAE3D,KAAK,MAAM,CAAC,CAAC,EAAE,IAAI,CAAC,IAAI,GAAG,CAAC,KAAK,CAAC,OAAO,EAAE,EAAE,CAAC;gBAC1C,IAAI,IAAI,CAAC,QAAQ,EAAE,CAAC;oBAChB,QAAQ,CAAC,IAAI,CAAC,GAAG,GAAG,CAAC,IAAI,UAAU,CAAC,SAAS,IAAI,CAAC,GAAG,sCAAsC,CAAC,CAAC;oBAC7F,SAAS;gBACb,CAAC;gBACD,IAAI,IAAI,CAAC,OAAO,EAAE,CAAC;oBACf,QAAQ,CAAC,IAAI,CAAC,GAAG,GAAG,CAAC,IAAI,UAAU,CAAC,sCAAsC,CAAC,CAAC;oBAC5E,SAAS;gBACb,CAAC;gBACD,MAAM,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,KAAK,IAAI,CAAC,GAAG,CAAC,CAAC;gBACxD,IAAI,IAAI,CAAC,QAAQ,IAAI,CAAC,OAAO;oBAAE,SAAS;gBACxC,MAAM,CAAC,OAAO,IAAI,IAAI,EAAE,KAAK,CAAA,uBAAuB,IAAI,CAAC,GAAG,sBAAsB,GAAG,CAAC,QAAQ,sBAAsB,CAAC,CAAC;gBACtH,MAAM,iBAAiB,GAAG,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;gBACxD,MAAM,aAAa,GAAG,MAAM,CAAC,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;gBAChD,MAAM,CAAC,iBAAiB,IAAI,aAAa,EAAE,KAAK,CAAA,uBAAuB,OAAO,CAAC,IAAI,sBAAsB,GAAG,CAAC,IAAI,wBAAwB,CAAC,CAAC;YAC/I,CAAC;QAEL,CAAC;QACD,OAAO,QAAQ,CAAC;IACpB,CAAC;IAEO,MAAM,CAAC,YAAY,CAAE,IAAwB,EAAE,MAAyB;QAC5E,KAAK,MAAM,GAAG,IAAI,IAAI,EAAE,CAAC;YACrB,IAAI,GAAG,CAAC,YAAY,KAAK,IAAI,IAAI,GAAG,CAAC,YAAY,CAAC,MAAM,KAAK,CAAC;gBAAE,SAAS;YAEzE,MAAM,SAAS,GAAG,GAAG,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,KAAK,CAAC,CAAC,CAAC,CAAC;YACpF,MAAM,CAAC,SAAS,CAAC,MAAM,KAAK,GAAG,CAAC,YAAY,CAAC,MAAM,EAAE,KAAK,CAAA,8BAA8B,SAAS,CAAC,IAAI,CAAC,GAAG,CAAC,sBAAsB,GAAG,CAAC,IAAI,mBAAmB,CAAC,CAAC;YAE9J,KAAK,MAAM,GAAG,IAAI,GAAG,CAAC,YAAY,EAAE,CAAC;gBACjC,MAAM,MAAM,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,KAAK,GAAG,CAAC,CAAC;gBAClD,MAAM,CAAC,MAAM,IAAI,IAAI,EAAE,KAAK,CAAA,8BAA8B,GAAG,sBAAsB,GAAG,CAAC,QAAQ,sBAAsB,CAAC,CAAC;gBACvH,MAAM,gBAAgB,GAAG,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;gBACtD,MAAM,aAAa,GAAG,MAAM,CAAC,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;gBAChD,MAAM,CAAC,gBAAgB,IAAI,aAAa,EAAE,KAAK,CAAA,8BAA8B,MAAM,CAAC,IAAI,sBAAsB,GAAG,CAAC,IAAI,wBAAwB,CAAC,CAAC;YACpJ,CAAC;QACL,CAAC;IACL,CAAC;IAEO,MAAM,CAAC,uBAAuB,CAAE,IAAwB;QAC5D,KAAK,MAAM,GAAG,IAAI,IAAI,EAAE,CAAC;YACrB,MAAM,KAAK,GAAG,GAAG,CAAC,KAAK,CAAC;YACxB,MAAM,YAAY,GAAG,GAAG,CAAC,YAAY,CAAC;YACtC,IAAI,KAAK,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC;gBAAE,SAAS;YAC1C,IAAI,CAAC,YAAY,IAAI,CAAC,KAAK;gBAAE,SAAS;YAGtC,MAAM,aAAa,GAAG,YAAY,CAAC,KAAK,CAAC,CAAC,GAAW,EAAE,EAAE;gBACrD,OAAO,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC;YAC1C,CAAC,CAAC,CAAC;YACH,MAAM,SAAS,GAAG,GAAG,GAAG,CAAC,gBAAgB,YAAY,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,0CAA0C,YAAY,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC;YACxJ,MAAM,CAAC,aAAa,EAAE,SAAS,CAAC,CAAC;QACrC,CAAC;IACL,CAAC;IAED;;;;;OAKG;IACK,MAAM,CAAC,uBAAuB,CAAE,SAAmB;QACvD,MAAM,QAAQ,GAAG,EAAE,CAAC;QACpB,KAAK,MAAM,OAAO,IAAI,SAAS,EAAE,CAAC;YAC9B,IAAI,IAAI,GAAG,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,OAAO,EAAE,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,EAAE,CAAC;gBAC1D,QAAQ,CAAC,IAAI,CAAC,aAAa,OAAO,qFAAqF,CAAC,CAAC;YAC7H,CAAC;QACL,CAAC;QACD,OAAO,QAAQ,CAAC;IACpB,CAAC;IAEO,MAAM,CAAC,WAAW,CAAE,IAAwB;QAChD,KAAK,MAAM,GAAG,IAAI,IAAI,EAAE,CAAC;YACrB,IAAI,GAAG,CAAC,OAAO;gBAAE,SAAS,CAAC,qDAAqD;YAChF,MAAM,CAAC,GAAG,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE,KAAK,CAAA,SAAS,GAAG,CAAC,IAAI,oBAAoB,CAAC,CAAC;QAC/E,CAAC;IACL,CAAC;IAEO,MAAM,CAAC,cAAc,CAAE,IAAwB;QACnD,KAAK,MAAM,GAAG,IAAI,IAAI,EAAE,CAAC;YACrB,IAAI,GAAG,CAAC,OAAO;gBAAE,SAAS;YAC1B,GAAG,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,MAAM,CAAC,OAAO,CAAC,KAAK,QAAQ,EAAE,KAAK,CAAA,SAAS,GAAG,CAAC,IAAI,2CAA2C,CAAC,CAAC,CAAC;YACxI,GAAG,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,MAAM,CAAC,OAAO,CAAC,KAAK,QAAQ,EAAE,KAAK,CAAA,SAAS,GAAG,CAAC,IAAI,0CAA0C,CAAC,CAAC,CAAC;YACtI,GAAG,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,MAAM,CAAC,OAAO,CAAC,KAAK,QAAQ,EAAE,KAAK,CAAA,SAAS,GAAG,CAAC,IAAI,oCAAoC,CAAC,CAAC,CAAC;QAC/H,CAAC;IACL,CAAC;IAED,MAAM,CAAC,KAAK,CAAC,GAAG,CAAE,IAAwB,EAAE,MAAyB;QACjE,MAAM,QAAQ,GAAa,EAAE,CAAC;QAC9B,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC;QACvB,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC;QAC1B,QAAQ,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC,CAAC;QAC3C,IAAI,CAAC,YAAY,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC;QAChC,IAAI,CAAC,uBAAuB,CAAC,IAAI,CAAC,CAAC;QACnC,QAAQ,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,uBAAuB,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC;QAC1E,QAAQ,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC;QACvC,OAAO,QAAQ,CAAC;IACpB,CAAC;IAEO,MAAM,CAAC,SAAS,CAAE,IAAwB;QAC9C,MAAM,QAAQ,GAAa,EAAE,CAAC;QAC9B,KAAK,MAAM,GAAG,IAAI,IAAI,EAAE,CAAC;YACrB,IAAI,GAAG,CAAC,SAAS,KAAK,IAAI,EAAE,CAAC;gBACzB,QAAQ,CAAC,IAAI,CAAC,GAAG,GAAG,CAAC,IAAI,+BAA+B,CAAC,CAAC;YAC9D,CAAC;QACL,CAAC;QACD,OAAO,QAAQ,CAAC;IACpB,CAAC;CACJ","sourcesContent":["import Ajv from \"ajv\";\nimport {Job} from \"./job.js\";\nimport assert from \"assert\";\nimport chalk from \"chalk\";\nimport schema from \"./schema/index.js\";\nimport {betterAjvErrors} from \"./schema-error.js\";\nimport terminalLink from \"terminal-link\";\nimport {Argv} from \"./argv.js\";\n\nconst MAX_ERRORS = 5;\n\nexport class Validator {\n    static jsonSchemaValidation ({pathToExpandedGitLabCi, gitLabCiConfig, argv}: {\n        pathToExpandedGitLabCi: string;\n        gitLabCiConfig: object;\n        argv: Argv;\n    }) {\n        const ajv = new Ajv({\n            verbose: true,\n            allErrors: true,\n            allowUnionTypes: true,\n            validateFormats: false,\n            strictTypes: false, // to suppress the missing types defined in the gitlab-ci json schema\n            keywords: [\"markdownDescription\"],\n        });\n        const validate = ajv.compile(schema);\n        const valid = validate(gitLabCiConfig);\n        if (valid) return;\n        const betterErrors = betterAjvErrors({\n            data: gitLabCiConfig,\n            errors: validate.errors,\n        }).filter(betterError => !argv.ignoreSchemaPaths.includes(betterError.schemaPath));\n\n        let e: string = \"\";\n        for (let i = 0, len = betterErrors.length; i < len; i++) {\n            if (i + 1 > MAX_ERRORS) {\n                e += `\\t... and ${len - MAX_ERRORS} more`;\n                break;\n            }\n            e += chalk`\\t• {redBright ${betterErrors[i].message}} at {blueBright ${betterErrors[i].path}} {grey [${betterErrors[i].schemaPath}]}\\n`;\n        }\n\n        assert(valid || betterErrors.length == 0, chalk`\n{reset Invalid .gitlab-ci.yml configuration!\n${e.trimEnd()}\n\nFor further troubleshooting, consider either of the following:\n\\t• Copy the content of {blueBright ${terminalLink(\".gitlab-ci-local/expanded-gitlab-ci.yml\", pathToExpandedGitLabCi)}} to the ${terminalLink(\"pipeline editor\", \"https://docs.gitlab.com/ee/ci/pipeline_editor/\")} to debug it\n\\t• Use --ignore-schema-paths= \"#/definitions/tags/minItems\" --ignore-schema-paths \"#/additionalProperties\" to partially disable certain validation rule\n\\t• Use --json-schema-validation=false to disable schema validation (not recommended)}\n`);\n    }\n\n    private static needs (jobs: ReadonlyArray<Job>, stages: readonly string[]): string[] {\n        const warnings: string[] = [];\n        for (const job of jobs) {\n            if (job.needs === null || job.needs.length === 0) continue;\n\n            for (const [i, need] of job.needs.entries()) {\n                if (need.pipeline) {\n                    warnings.push(`${job.name}.needs[${i}].job:${need.job} ignored, pipeline key not supported`);\n                    continue;\n                }\n                if (need.project) {\n                    warnings.push(`${job.name}.needs[${i}] ignored, project key not supported`);\n                    continue;\n                }\n                const needJob = jobs.find(j => j.baseName === need.job);\n                if (need.optional && !needJob) continue;\n                assert(needJob != null, chalk`needs: [{blueBright ${need.job}}] for {blueBright ${job.baseName}} could not be found`);\n                const needJobStageIndex = stages.indexOf(needJob.stage);\n                const jobStageIndex = stages.indexOf(job.stage);\n                assert(needJobStageIndex <= jobStageIndex, chalk`needs: [{blueBright ${needJob.name}}] for {blueBright ${job.name}} is in a future stage`);\n            }\n\n        }\n        return warnings;\n    }\n\n    private static dependencies (jobs: ReadonlyArray<Job>, stages: readonly string[]) {\n        for (const job of jobs) {\n            if (job.dependencies === null || job.dependencies.length === 0) continue;\n\n            const undefDeps = job.dependencies.filter((j) => !jobs.some(n => n.baseName === j));\n            assert(undefDeps.length !== job.dependencies.length, chalk`dependencies: [{blueBright ${undefDeps.join(\",\")}}] for {blueBright ${job.name}} cannot be found`);\n\n            for (const dep of job.dependencies) {\n                const depJob = jobs.find(j => j.baseName === dep);\n                assert(depJob != null, chalk`dependencies: [{blueBright ${dep}}] for {blueBright ${job.baseName}} could not be found`);\n                const depJobStageIndex = stages.indexOf(depJob.stage);\n                const jobStageIndex = stages.indexOf(job.stage);\n                assert(depJobStageIndex <= jobStageIndex, chalk`dependencies: [{blueBright ${depJob.name}}] for {blueBright ${job.name}} is in a future stage`);\n            }\n        }\n    }\n\n    private static dependenciesContainment (jobs: ReadonlyArray<Job>) {\n        for (const job of jobs) {\n            const needs = job.needs;\n            const dependencies = job.dependencies;\n            if (needs && needs.length === 0) continue;\n            if (!dependencies || !needs) continue;\n\n\n            const everyIncluded = dependencies.every((dep: string) => {\n                return needs.some(n => n.job === dep);\n            });\n            const assertMsg = `${job.formattedJobName} needs: '${needs.map(n => n.job).join(\",\")}' doesn't fully contain dependencies: '${dependencies.join(\",\")}'`;\n            assert(everyIncluded, assertMsg);\n        }\n    }\n\n    /**\n     * These jobs named are reserved keywords in GitLab CI but does not prevent the pipeline from running\n     * https://github.com/firecow/gitlab-ci-local/issues/1263\n     * @param jobsNames\n     * @private\n     */\n    private static potentialIllegalJobName (jobsNames: string[]) {\n        const warnings = [];\n        for (const jobName of jobsNames) {\n            if (new Set([\"types\", \"true\", \"false\", \"nil\"]).has(jobName)) {\n                warnings.push(`Job name \"${jobName}\" is a reserved keyword. (https://docs.gitlab.com/ee/ci/jobs/#job-name-limitations)`);\n            }\n        }\n        return warnings;\n    }\n\n    private static scriptBlank (jobs: ReadonlyArray<Job>) {\n        for (const job of jobs) {\n            if (job.trigger) continue; // Jobs with trigger are allowed to have empty script\n            assert(job.scripts.length > 0, chalk`{blue ${job.name}} has empty script`);\n        }\n    }\n\n    private static arrayOfStrings (jobs: ReadonlyArray<Job>) {\n        for (const job of jobs) {\n            if (job.trigger) continue;\n            job.beforeScripts.forEach((s: any) => assert(typeof s === \"string\", chalk`{blue ${job.name}} before_script contains non string value`));\n            job.afterScripts.forEach((s: any) => assert(typeof s === \"string\", chalk`{blue ${job.name}} after_script contains non string value`));\n            job.scripts.forEach((s: any) => assert(typeof s === \"string\", chalk`{blue ${job.name}} script contains non string value`));\n        }\n    }\n\n    static async run (jobs: ReadonlyArray<Job>, stages: readonly string[]) {\n        const warnings: string[] = [];\n        this.scriptBlank(jobs);\n        this.arrayOfStrings(jobs);\n        warnings.push(...this.needs(jobs, stages));\n        this.dependencies(jobs, stages);\n        this.dependenciesContainment(jobs);\n        warnings.push(...this.potentialIllegalJobName(jobs.map(j => j.baseName)));\n        warnings.push(...this.artifacts(jobs));\n        return warnings;\n    }\n\n    private static artifacts (jobs: ReadonlyArray<Job>) {\n        const warnings: string[] = [];\n        for (const job of jobs) {\n            if (job.artifacts === null) {\n                warnings.push(`${job.name}.artifacts is null, ignoring.`);\n            }\n        }\n        return warnings;\n    }\n}\n"]}