projen
Version:
CDK for software projects
302 lines • 41.9 kB
JavaScript
"use strict";
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.CiConfiguration = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const path = require("path");
const case_1 = require("case");
const component_1 = require("../component");
const yaml_1 = require("../yaml");
/**
* CI for GitLab.
* A CI is a configurable automated process made up of one or more stages/jobs.
* @see https://docs.gitlab.com/ee/ci/yaml/
*/
class CiConfiguration extends component_1.Component {
get defaultCache() {
return this._defaultCache;
}
constructor(project, name, options) {
super(project);
/**
* Defines default scripts that should run *after* all jobs. Can be overriden by the job level `afterScript`.
*/
this.defaultAfterScript = [];
/**
* Defines default scripts that should run *before* all jobs. Can be overriden by the job level `afterScript`.
*/
this.defaultBeforeScript = [];
/**
* A default list of additional Docker images to run scripts in. The service image is linked to the image specified in the image parameter.
*/
this.defaultServices = [];
/**
* Used to select a specific runner from the list of all runners that are available for the project.
*/
this.defaultTags = [];
/**
* Can be `Include` or `Include[]`. Each `Include` will be a string, or an
* object with properties for the method if including external YAML file. The external
* content will be fetched, included and evaluated along the `.gitlab-ci.yml`.
*/
this.include = [];
/**
* Groups jobs into stages. All jobs in one stage must complete before next stage is
* executed. Defaults to ['build', 'test', 'deploy'].
*/
this.stages = [];
/**
* Global variables that are passed to jobs.
* If the job already has that variable defined, the job-level variable takes precedence.
*/
this.variables = {};
/**
* The jobs in the CI configuration.
*/
this.jobs = {};
this.name = path.parse(name).name;
const derivedPath = this.name === "gitlab-ci"
? ".gitlab-ci.yml"
: `.gitlab/ci-templates/${name.toLocaleLowerCase()}.yml`;
this.path = options?.path ?? derivedPath;
this.file = new yaml_1.YamlFile(this.project, this.path, {
obj: () => this.renderCI(),
// GitLab needs to read the file from the repository in order to work.
committed: true,
});
const defaults = options?.default;
if (defaults) {
this.defaultAfterScript.push(...(defaults.afterScript ?? []));
this.defaultArtifacts = defaults.artifacts;
defaults.beforeScript &&
this.defaultBeforeScript.push(...defaults.beforeScript);
defaults.cache && this.addDefaultCaches(defaults.cache);
this.defaultIdTokens = defaults.idTokens;
this.defaultImage = defaults.image;
this.defaultInterruptible = defaults.interruptible;
this.defaultRetry = defaults.retry;
defaults.services && this.addServices(...defaults.services);
defaults.tags && this.defaultTags.push(...defaults.tags);
this.defaultTimeout = defaults.timeout;
}
this.pages = options?.pages;
this.workflow = options?.workflow;
if (options?.stages) {
this.addStages(...options.stages);
}
if (options?.variables) {
this.addGlobalVariables(options.variables);
}
if (options?.jobs) {
this.addJobs(options.jobs);
}
}
/**
* Add additional yml/yaml files to the CI includes
* @param includes The includes to add.
*/
addIncludes(...includes) {
for (const additional of includes) {
this.assertIsValidInclude(additional);
for (const existing of this.include) {
if (this.areEqualIncludes(existing, additional)) {
throw new Error(`${this.name}: GitLab CI ${existing} already contains one or more templates specified in ${additional}.`);
}
}
this.include.push(additional);
}
}
/**
* Throw an error if the provided Include is invalid.
* @see https://gitlab.com/gitlab-org/gitlab/-/blob/master/lib/gitlab/ci/config/external/mapper.rb
* @param include the Include to validate.
*/
assertIsValidInclude(include) {
const combos = [
include.local,
include.file && include.project,
include.remote,
include.template,
];
const len = combos.filter((x) => Boolean(x)).length;
if (len !== 1) {
throw new Error(`${this.name}: GitLab CI include ${include} contains ${len} property combination(s).
A valid include configuration specifies *one* of the following property combinations.
* local
* file, project
* remote
* template
`);
}
}
/**
* Check if the equality of Includes.
* @see https://gitlab.com/gitlab-org/gitlab/-/blob/master/lib/gitlab/ci/config/external/mapper.rb
* @param x First include to compare.
* @param y Second include to compare.
* @returns Whether the includes are equal.
*/
areEqualIncludes(x, y) {
if (x.local === y.local && x.local !== undefined) {
return true;
}
else if (x.template === y.template && x.template !== undefined) {
return true;
}
else if (x.remote === y.remote && x.remote !== undefined) {
return true;
}
else if (x.project === y.project && x.ref === y.ref) {
const xFiles = x.file ? x.file : [];
const yFiles = y.file ? y.file : [];
const allFiles = xFiles.concat(yFiles);
return new Set(allFiles).size !== allFiles.length;
}
return false;
}
/**
* Add additional services.
* @param services The services to add.
*/
addServices(...services) {
for (const additional of services) {
for (const existing of this.defaultServices) {
if (additional.name === existing.name &&
additional.alias === existing.alias) {
throw new Error(`${this.name}: GitLab CI already contains service ${additional}.`);
}
}
this.defaultServices.push(additional);
}
}
/**
* Add a globally defined variable to the CI configuration.
* @param variables The variables to add.
*/
addGlobalVariables(variables) {
for (const [key, value] of Object.entries(variables)) {
if (this.variables[key] !== undefined) {
throw new Error(`${this.name}: GitLab CI already contains variable ${key}.`);
}
this.variables[key] = value;
}
}
/**
* Add stages to the CI configuration if not already present.
* @param stages stages to add.
*/
addStages(...stages) {
for (const stage of stages) {
if (!this.stages.includes(stage)) {
this.stages.push(stage);
}
}
}
/**
* Add jobs and their stages to the CI configuration.
* @param jobs Jobs to add.
*/
addJobs(jobs) {
for (const [key, value] of Object.entries(jobs)) {
if (this.jobs[key] !== undefined) {
throw new Error(`${this.name}: GitLab CI already contains job ${key}.`);
}
this.jobs[key] = value;
if (value.stage) {
this.addStages(value.stage);
}
if (value.cache) {
this.assertIsValidCacheSetup(value.cache);
}
}
}
isValidCacheSetup(caches) {
const MAX_CONFIGURABLE_CACHES = 4;
return caches.length <= MAX_CONFIGURABLE_CACHES;
}
assertIsValidCacheSetup(caches) {
if (!this.isValidCacheSetup(caches)) {
throw new Error(`${this.name}: GitLab CI can only define up to 4 caches, got: ${caches.length}`);
}
}
/**
* Adds up to 4 default caches configuration to the CI configuration.
* @param caches Caches to add.
*/
addDefaultCaches(caches) {
this.assertIsValidCacheSetup(caches);
this._defaultCache = caches;
}
/**
* Specify a list of commands to execute on the runner before cloning the Git repository and any submodules
* https://docs.gitlab.com/ci/yaml/#hookspre_get_sources_script
* @param hooks
*/
addDefaultHooks(hooks) {
this.defaultHooks = hooks;
}
renderCI() {
return {
default: this.renderDefault(),
include: this.include.length > 0 ? snakeCaseKeys(this.include) : undefined,
pages: snakeCaseKeys(this.pages),
services: this.defaultServices.length > 0
? snakeCaseKeys(this.defaultServices)
: undefined,
variables: Object.entries(this.variables).length > 0 ? this.variables : undefined,
workflow: snakeCaseKeys(this.workflow),
stages: this.stages.length > 0 ? this.stages : undefined,
// we do not want to change job names
// as they can be hidden (https://docs.gitlab.com/ee/ci/jobs/index.html#hide-jobs)
// or referenced in extends
...snakeCaseKeys(this.jobs, true),
};
}
renderDefault() {
const defaults = {
afterScript: this.defaultAfterScript.length > 0
? this.defaultAfterScript
: undefined,
artifacts: this.defaultArtifacts,
beforeScript: this.defaultBeforeScript.length > 0
? this.defaultBeforeScript
: undefined,
cache: this.defaultCache,
idTokens: this.defaultIdTokens,
image: this.defaultImage,
interruptible: this.defaultInterruptible,
retry: this.defaultRetry,
services: this.defaultServices.length > 0 ? this.defaultServices : undefined,
tags: this.defaultTags.length > 0 ? this.defaultTags : undefined,
timeout: this.defaultTimeout,
hooks: this.defaultHooks,
};
return Object.values(defaults).filter((x) => x).length
? snakeCaseKeys(defaults)
: undefined;
}
}
exports.CiConfiguration = CiConfiguration;
_a = JSII_RTTI_SYMBOL_1;
CiConfiguration[_a] = { fqn: "projen.gitlab.CiConfiguration", version: "0.99.51" };
function snakeCaseKeys(obj, skipTopLevel = false) {
if (typeof obj !== "object" || obj == null) {
return obj;
}
if (Array.isArray(obj)) {
return obj.map((o) => snakeCaseKeys(o));
}
const result = {};
for (let [k, v] of Object.entries(obj)) {
if (typeof v === "object" &&
v != null &&
k !== "variables" &&
k !== "idTokens" &&
k !== "inputs") {
v = snakeCaseKeys(v);
}
result[skipTopLevel ? k : (0, case_1.snake)(k)] = v;
}
return result;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"configuration.js","sourceRoot":"","sources":["../../src/gitlab/configuration.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from \"path\";\nimport { snake } from \"case\";\nimport type {\n  Artifacts,\n  Cache,\n  Default,\n  DefaultHooks,\n  IDToken,\n  Image,\n  Include,\n  Job,\n  Retry,\n  Service,\n  VariableConfig,\n  Workflow,\n} from \"./configuration-model\";\nimport { Component } from \"../component\";\nimport type { Project } from \"../project\";\nimport { YamlFile } from \"../yaml\";\n\n/**\n * Options for `CiConfiguration`.\n */\nexport interface CiConfigurationOptions {\n  /**\n   * Default settings for the CI Configuration. Jobs that do not define one or more of the listed keywords use the value defined in the default section.\n   */\n  readonly default?: Default;\n  /**\n   * A special job used to upload static sites to Gitlab pages. Requires a `public/` directory\n   * with `artifacts.path` pointing to it.\n   */\n  readonly pages?: Job;\n  /**\n   * Used to control pipeline behavior.\n   */\n  readonly workflow?: Workflow;\n  /**\n   * Groups jobs into stages. All jobs in one stage must complete before next stage is\n   * executed. If no stages are specified. Defaults to ['build', 'test', 'deploy'].\n   */\n  readonly stages?: string[];\n  /**\n   * Global variables that are passed to jobs.\n   * If the job already has that variable defined, the job-level variable takes precedence.\n   */\n  readonly variables?: Record<string, any>;\n  /**\n   * An initial set of jobs to add to the configuration.\n   */\n  readonly jobs?: Record<string, Job>;\n  /**\n   * The path of the file to generate.\n   */\n  readonly path?: string;\n}\n\n/**\n * CI for GitLab.\n * A CI is a configurable automated process made up of one or more stages/jobs.\n * @see https://docs.gitlab.com/ee/ci/yaml/\n */\nexport class CiConfiguration extends Component {\n  /**\n   * The name of the configuration.\n   */\n  public readonly name: string;\n  /**\n   * Path to CI file generated by the configuration.\n   */\n  public readonly path: string;\n  /**\n   * The workflow YAML file.\n   */\n  public readonly file: YamlFile;\n  /**\n   * Defines default scripts that should run *after* all jobs. Can be overriden by the job level `afterScript`.\n   */\n  public readonly defaultAfterScript: string[] = [];\n  /**\n   * Default list of files and directories that should be attached to the job if it succeeds. Artifacts are sent to Gitlab where they can be downloaded.\n   */\n  public readonly defaultArtifacts?: Artifacts;\n  /**\n   * Defines default scripts that should run *before* all jobs. Can be overriden by the job level `afterScript`.\n   */\n  public readonly defaultBeforeScript: string[] = [];\n  /**\n   * A default list of cache definitions (máx. 4) with the files and directories to cache between jobs. You can only use paths that are in the local working copy.\n   */\n  private _defaultCache?: Cache[];\n\n  public get defaultCache(): Cache[] | undefined {\n    return this._defaultCache;\n  }\n  /**\n   * Specifies the default docker image to use globally for all jobs.\n   */\n  public readonly defaultImage?: Image;\n  /**\n   * The default behavior for whether a job should be canceled when a newer pipeline starts before the job completes (Default: false).\n   */\n  public readonly defaultInterruptible?: boolean;\n  /**\n   * How many times a job is retried if it fails. If not defined, defaults to 0 and jobs do not retry.\n   */\n  public readonly defaultRetry?: Retry;\n  /**\n   * A default list of additional Docker images to run scripts in. The service image is linked to the image specified in the  image parameter.\n   */\n  private defaultServices: Service[] = [];\n  /**\n   * Used to select a specific runner from the list of all runners that are available for the project.\n   */\n  readonly defaultTags: string[] = [];\n  /**\n   * A default timeout job written in natural language (Ex. one hour, 3600 seconds, 60 minutes).\n   */\n  readonly defaultTimeout?: string;\n  /**\n   * Default ID tokens (JSON Web Tokens) that are used for CI/CD authentication to use globally for all jobs.\n   */\n  readonly defaultIdTokens?: Record<string, IDToken>;\n  /**\n   * Specify a list of commands to execute on the runner before cloning the Git repository and any submodules https://docs.gitlab.com/ci/yaml/#hookspre_get_sources_script\n   */\n  private defaultHooks?: Default[\"hooks\"];\n  /**\n   * Can be `Include` or `Include[]`. Each `Include` will be a string, or an\n   * object with properties for the method if including external YAML file. The external\n   * content will be fetched, included and evaluated along the `.gitlab-ci.yml`.\n   */\n  private include: Include[] = [];\n  /**\n   * A special job used to upload static sites to Gitlab pages. Requires a `public/` directory\n   * with `artifacts.path` pointing to it.\n   */\n  public readonly pages?: Job;\n  /**\n   * Groups jobs into stages. All jobs in one stage must complete before next stage is\n   * executed. Defaults to ['build', 'test', 'deploy'].\n   */\n  public readonly stages: string[] = [];\n  /**\n   * Global variables that are passed to jobs.\n   * If the job already has that variable defined, the job-level variable takes precedence.\n   */\n  public readonly variables: Record<string, number | VariableConfig | string> =\n    {};\n  /**\n   * Used to control pipeline behavior.\n   */\n  public readonly workflow?: Workflow;\n  /**\n   * The jobs in the CI configuration.\n   */\n  public readonly jobs: Record<string, Job> = {};\n\n  constructor(\n    project: Project,\n    name: string,\n    options?: CiConfigurationOptions,\n  ) {\n    super(project);\n    this.name = path.parse(name).name;\n    const derivedPath =\n      this.name === \"gitlab-ci\"\n        ? \".gitlab-ci.yml\"\n        : `.gitlab/ci-templates/${name.toLocaleLowerCase()}.yml`;\n    this.path = options?.path ?? derivedPath;\n    this.file = new YamlFile(this.project, this.path, {\n      obj: () => this.renderCI(),\n      // GitLab needs to read the file from the repository in order to work.\n      committed: true,\n    });\n    const defaults = options?.default;\n    if (defaults) {\n      this.defaultAfterScript.push(...(defaults.afterScript ?? []));\n      this.defaultArtifacts = defaults.artifacts;\n      defaults.beforeScript &&\n        this.defaultBeforeScript.push(...defaults.beforeScript);\n      defaults.cache && this.addDefaultCaches(defaults.cache);\n      this.defaultIdTokens = defaults.idTokens;\n      this.defaultImage = defaults.image;\n      this.defaultInterruptible = defaults.interruptible;\n      this.defaultRetry = defaults.retry;\n      defaults.services && this.addServices(...defaults.services);\n      defaults.tags && this.defaultTags.push(...defaults.tags);\n      this.defaultTimeout = defaults.timeout;\n    }\n    this.pages = options?.pages;\n    this.workflow = options?.workflow;\n    if (options?.stages) {\n      this.addStages(...options.stages);\n    }\n    if (options?.variables) {\n      this.addGlobalVariables(options.variables);\n    }\n    if (options?.jobs) {\n      this.addJobs(options.jobs);\n    }\n  }\n\n  /**\n   * Add additional yml/yaml files to the CI includes\n   * @param includes The includes to add.\n   */\n  public addIncludes(...includes: Include[]) {\n    for (const additional of includes) {\n      this.assertIsValidInclude(additional);\n      for (const existing of this.include) {\n        if (this.areEqualIncludes(existing, additional)) {\n          throw new Error(\n            `${this.name}: GitLab CI ${existing} already contains one or more templates specified in ${additional}.`,\n          );\n        }\n      }\n      this.include.push(additional);\n    }\n  }\n\n  /**\n   * Throw an error if the provided Include is invalid.\n   * @see https://gitlab.com/gitlab-org/gitlab/-/blob/master/lib/gitlab/ci/config/external/mapper.rb\n   * @param include the Include to validate.\n   */\n  private assertIsValidInclude(include: Include) {\n    const combos = [\n      include.local,\n      include.file && include.project,\n      include.remote,\n      include.template,\n    ];\n    const len = combos.filter((x) => Boolean(x)).length;\n    if (len !== 1) {\n      throw new Error(\n        `${this.name}: GitLab CI include ${include} contains ${len} property combination(s).\n        A valid include configuration specifies *one* of the following property combinations.\n        * local\n        * file, project\n        * remote\n        * template\n        `,\n      );\n    }\n  }\n\n  /**\n   * Check if the equality of Includes.\n   * @see https://gitlab.com/gitlab-org/gitlab/-/blob/master/lib/gitlab/ci/config/external/mapper.rb\n   * @param x First include to compare.\n   * @param y Second include to compare.\n   * @returns Whether the includes are equal.\n   */\n  private areEqualIncludes(x: Include, y: Include): boolean {\n    if (x.local === y.local && x.local !== undefined) {\n      return true;\n    } else if (x.template === y.template && x.template !== undefined) {\n      return true;\n    } else if (x.remote === y.remote && x.remote !== undefined) {\n      return true;\n    } else if (x.project === y.project && x.ref === y.ref) {\n      const xFiles = x.file ? x.file : [];\n      const yFiles = y.file ? y.file : [];\n      const allFiles = xFiles.concat(yFiles);\n      return new Set(allFiles).size !== allFiles.length;\n    }\n    return false;\n  }\n\n  /**\n   * Add additional services.\n   * @param services The services to add.\n   */\n  public addServices(...services: Service[]) {\n    for (const additional of services) {\n      for (const existing of this.defaultServices) {\n        if (\n          additional.name === existing.name &&\n          additional.alias === existing.alias\n        ) {\n          throw new Error(\n            `${this.name}: GitLab CI already contains service ${additional}.`,\n          );\n        }\n      }\n      this.defaultServices.push(additional);\n    }\n  }\n\n  /**\n   * Add a globally defined variable to the CI configuration.\n   * @param variables The variables to add.\n   */\n  public addGlobalVariables(variables: Record<string, any>) {\n    for (const [key, value] of Object.entries(variables)) {\n      if (this.variables[key] !== undefined) {\n        throw new Error(\n          `${this.name}: GitLab CI already contains variable ${key}.`,\n        );\n      }\n      this.variables[key] = value;\n    }\n  }\n\n  /**\n   * Add stages to the CI configuration if not already present.\n   * @param stages stages to add.\n   */\n  public addStages(...stages: string[]) {\n    for (const stage of stages) {\n      if (!this.stages.includes(stage)) {\n        this.stages.push(stage);\n      }\n    }\n  }\n\n  /**\n   * Add jobs and their stages to the CI configuration.\n   * @param jobs Jobs to add.\n   */\n  public addJobs(jobs: Record<string, Job>) {\n    for (const [key, value] of Object.entries(jobs)) {\n      if (this.jobs[key] !== undefined) {\n        throw new Error(`${this.name}: GitLab CI already contains job ${key}.`);\n      }\n      this.jobs[key] = value;\n      if (value.stage) {\n        this.addStages(value.stage);\n      }\n      if (value.cache) {\n        this.assertIsValidCacheSetup(value.cache);\n      }\n    }\n  }\n\n  private isValidCacheSetup(caches: Cache[]): Boolean {\n    const MAX_CONFIGURABLE_CACHES = 4;\n    return caches.length <= MAX_CONFIGURABLE_CACHES;\n  }\n\n  private assertIsValidCacheSetup(caches: Cache[]) {\n    if (!this.isValidCacheSetup(caches)) {\n      throw new Error(\n        `${this.name}: GitLab CI can only define up to 4 caches, got: ${caches.length}`,\n      );\n    }\n  }\n\n  /**\n   * Adds up to 4 default caches configuration to the CI configuration.\n   * @param caches Caches to add.\n   */\n  public addDefaultCaches(caches: Cache[]) {\n    this.assertIsValidCacheSetup(caches);\n    this._defaultCache = caches;\n  }\n\n  /**\n   * Specify a list of commands to execute on the runner before cloning the Git repository and any submodules\n   * https://docs.gitlab.com/ci/yaml/#hookspre_get_sources_script\n   * @param hooks\n   */\n  public addDefaultHooks(hooks: DefaultHooks) {\n    this.defaultHooks = hooks;\n  }\n\n  private renderCI() {\n    return {\n      default: this.renderDefault(),\n      include:\n        this.include.length > 0 ? snakeCaseKeys(this.include) : undefined,\n      pages: snakeCaseKeys(this.pages),\n      services:\n        this.defaultServices.length > 0\n          ? snakeCaseKeys(this.defaultServices)\n          : undefined,\n      variables:\n        Object.entries(this.variables).length > 0 ? this.variables : undefined,\n      workflow: snakeCaseKeys(this.workflow),\n      stages: this.stages.length > 0 ? this.stages : undefined,\n      // we do not want to change job names\n      // as they can be hidden (https://docs.gitlab.com/ee/ci/jobs/index.html#hide-jobs)\n      // or referenced in extends\n      ...snakeCaseKeys(this.jobs, true),\n    };\n  }\n\n  private renderDefault() {\n    const defaults: Default = {\n      afterScript:\n        this.defaultAfterScript.length > 0\n          ? this.defaultAfterScript\n          : undefined,\n      artifacts: this.defaultArtifacts,\n      beforeScript:\n        this.defaultBeforeScript.length > 0\n          ? this.defaultBeforeScript\n          : undefined,\n      cache: this.defaultCache,\n      idTokens: this.defaultIdTokens,\n      image: this.defaultImage,\n      interruptible: this.defaultInterruptible,\n      retry: this.defaultRetry,\n      services:\n        this.defaultServices.length > 0 ? this.defaultServices : undefined,\n      tags: this.defaultTags.length > 0 ? this.defaultTags : undefined,\n      timeout: this.defaultTimeout,\n      hooks: this.defaultHooks,\n    };\n    return Object.values(defaults).filter((x) => x).length\n      ? snakeCaseKeys(defaults)\n      : undefined;\n  }\n}\n\nfunction snakeCaseKeys<T = unknown>(obj: T, skipTopLevel: boolean = false): T {\n  if (typeof obj !== \"object\" || obj == null) {\n    return obj;\n  }\n\n  if (Array.isArray(obj)) {\n    return obj.map((o) => snakeCaseKeys(o)) as any;\n  }\n\n  const result: Record<string, unknown> = {};\n  for (let [k, v] of Object.entries(obj)) {\n    if (\n      typeof v === \"object\" &&\n      v != null &&\n      k !== \"variables\" &&\n      k !== \"idTokens\" &&\n      k !== \"inputs\"\n    ) {\n      v = snakeCaseKeys(v);\n    }\n    result[skipTopLevel ? k : snake(k)] = v;\n  }\n  return result as any;\n}\n"]}