projen
Version:
CDK for software projects
120 lines • 16.5 kB
JavaScript
;
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.isObjectContainingFieldExactly = exports.Circleci = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const component_1 = require("../component");
const util_1 = require("../util");
const yaml_1 = require("../yaml");
/**
* Circleci Class to manage `.circleci/config.yml`.
* Check projen's docs for more information.
*
* @see https://circleci.com/docs/2.0/configuration-reference/
*/
class Circleci extends component_1.Component {
constructor(project, options = {}) {
super(project);
/**
* reduce objects with `identifier` field for WorkflowJobs.
* A workflow job can contain `orbParameter` which are passed to orbs.
* This map is directly added as `Record<string,any>` to job.
* So we gonna add those after the default field of WorkflowJob.
* @see https://circleci.com/developer/orbs/orb/circleci/node#usage-install_nodejs
* @param jobs
*/
this.renderJobs = (jobs = []) => {
let result = [];
for (const job of jobs ?? []) {
const { identifier, orbParameters, ...reduced } = job;
if ((0, exports.isObjectContainingFieldExactly)(job, "identifier")) {
result = [...result, identifier];
}
else {
result = [
...result,
{ [identifier]: { ...reduced, ...orbParameters } },
];
}
}
return result;
};
/**
* Snake case for listed keys. There are too many exceptions needed to do it recursive without a whitelist.
* This list needs to be updated once we add field with snake case like `aws_auth`.
* @param input
*/
this.snakeCase = (input) => {
const snakeCaseKeyWords = [
"awsAuth",
"workingDirectory",
"resourceClass",
"dockerLayerCaching",
"noOutputTimeout",
"circleciIpRanges",
];
return (0, util_1.snakeCaseKeys)(input, true, snakeCaseKeyWords);
};
this.options = options;
this.orbs = options.orbs ?? {};
this.workflows = options.workflows ?? [];
this.jobs = options.jobs ?? [];
this.file = new yaml_1.YamlFile(project, ".circleci/config.yml", {
obj: () => this.renderCircleCi(),
});
}
/**
* function to prepare the rendering of the yaml file.
* Objects with dynamic keys like workflows and jobs required the field `identifier` to be set.
* Those object will be reduced by `identifier` field before rendering
* @private
*/
renderCircleCi() {
// render dynamic keys for workflow
const workflowRecords = {};
for (const workflow of this.workflows) {
const { identifier, ...reduced } = workflow;
reduced.jobs = this.renderJobs(workflow.jobs);
workflowRecords[identifier] = this.snakeCase(reduced);
}
// render dynamic keys for jobs
const jobRecords = {};
for (const job of this.jobs) {
const { identifier, ...reduced } = job;
jobRecords[identifier] = this.snakeCase(reduced);
}
return {
version: this.options.version || 2.1,
setup: this.options.setup,
orbs: this.orbs,
jobs: Object.keys(jobRecords).length > 0 ? jobRecords : undefined,
workflows: workflowRecords,
};
}
/**
* add new workflow to existing pipeline
* @param workflow
*/
addWorkflow(workflow) {
this.workflows = [...this.workflows, workflow];
}
/**
* Add a Circleci Orb to pipeline. Will throw error if the orb already exists
* @param name
* @param orb
*/
addOrb(name, orb) {
if (this.orbs[name] !== undefined) {
throw new Error(`Circleci Config already contains an orb named ${name}.`);
}
this.orbs[name] = orb;
}
}
exports.Circleci = Circleci;
_a = JSII_RTTI_SYMBOL_1;
Circleci[_a] = { fqn: "projen.circleci.Circleci", version: "0.95.2" };
const isObjectContainingFieldExactly = (obj, field) => {
return Object.keys(obj).length == 1 && Object.keys(obj).includes(field);
};
exports.isObjectContainingFieldExactly = isObjectContainingFieldExactly;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"circleci.js","sourceRoot":"","sources":["../../src/circleci/circleci.ts"],"names":[],"mappings":"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","sourcesContent":["import { WorkflowJob, Workflow, Job } from \"./model\";\nimport { Component } from \"../component\";\nimport { Project } from \"../project\";\nimport { snakeCaseKeys } from \"../util\";\nimport { YamlFile } from \"../yaml\";\n\n/**\n * Options for class {@link Circleci}\n *\n * @see https://circleci.com/docs/2.0/configuration-reference/\n */\nexport interface CircleCiProps {\n  /**\n   * Contains a map of CirclCi Orbs\n   * ```json\n   * orbs: {\n   *  node: \"circleci/node@5.0.1\"\n   *  slack: \"circleci/slack@4.8.3\"\n   * }\n   * ```\n   */\n  readonly orbs?: Record<string, string>;\n  /**\n   * pipeline version\n   *\n   * @default 2.1\n   * @see https://circleci.com/docs/2.0/configuration-reference/#version\n   */\n  readonly version?: number;\n  /**\n   * List of Workflows of pipeline, e.g.\n   * ```json\n   * workflows: {\n   *   {\n   *     identifier: \"build\",\n   *       jobs: [{\n   *          identifier: \"node/install\",\n   *          context: [\"npm\"],\n   *       }]\n   *   }\n   * }\n   * ```\n   *\n   * @see https://circleci.com/docs/2.0/configuration-reference/#workflows\n   */\n  readonly workflows?: Workflow[];\n  /**\n   * List of Jobs to create unique steps per pipeline, e.g.\n   * ```json\n   * jobs: [{\n   *  identifier: \"compile\",\n   *  docker: { image: \"golang:alpine\" }\n   *  steps: [\"checkout\", run: {command: \"go build .\"}]\n   * }]\n   * ```\n   *\n   * @see https://circleci.com/docs/2.0/configuration-reference/#jobs\n   */\n  readonly jobs?: Job[];\n  /**\n   * The setup field enables you to conditionally trigger configurations from outside\n   * the primary .circleci parent directory, update pipeline parameters, or generate customized configurations.\n   *\n   * @see https://circleci.com/docs/2.0/configuration-reference/#setup\n   */\n  readonly setup?: boolean;\n}\n\n/**\n * Circleci Class to manage `.circleci/config.yml`.\n * Check projen's docs for more information.\n *\n * @see https://circleci.com/docs/2.0/configuration-reference/\n */\nexport class Circleci extends Component {\n  /**\n   * The yaml file for the Circleci pipeline\n   * */\n  public readonly file: YamlFile;\n  /**\n   * internal copy of options to share options between functions\n   * @private\n   */\n  private options: CircleCiProps;\n  /**\n   * internal map of orbs\n   * @private\n   */\n  private readonly orbs: Record<string, string>;\n  /**\n   * internal list of workflows\n   * @private\n   */\n  private workflows: Workflow[];\n  /**\n   * internal list of jobs\n   * @private\n   */\n  private readonly jobs: Job[];\n\n  constructor(project: Project, options: CircleCiProps = {}) {\n    super(project);\n    this.options = options;\n    this.orbs = options.orbs ?? {};\n    this.workflows = options.workflows ?? [];\n    this.jobs = options.jobs ?? [];\n    this.file = new YamlFile(project, \".circleci/config.yml\", {\n      obj: () => this.renderCircleCi(),\n    });\n  }\n\n  /**\n   * function to prepare the rendering of the yaml file.\n   * Objects with dynamic keys like workflows and jobs required the field `identifier` to be set.\n   * Those object will be reduced by `identifier` field before rendering\n   * @private\n   */\n  private renderCircleCi() {\n    // render dynamic keys for workflow\n    const workflowRecords: Record<string, any> = {};\n    for (const workflow of this.workflows) {\n      const { identifier, ...reduced } = workflow;\n      reduced.jobs = this.renderJobs(workflow.jobs);\n      workflowRecords[identifier] = this.snakeCase(reduced);\n    }\n\n    // render dynamic keys for jobs\n    const jobRecords: Record<string, any> = {};\n    for (const job of this.jobs) {\n      const { identifier, ...reduced } = job;\n      jobRecords[identifier] = this.snakeCase(reduced);\n    }\n\n    return {\n      version: this.options.version || 2.1,\n      setup: this.options.setup,\n      orbs: this.orbs,\n      jobs: Object.keys(jobRecords).length > 0 ? jobRecords : undefined,\n      workflows: workflowRecords,\n    };\n  }\n\n  /**\n   * add new workflow to existing pipeline\n   * @param workflow\n   */\n  public addWorkflow(workflow: Workflow) {\n    this.workflows = [...this.workflows, workflow];\n  }\n\n  /**\n   * reduce objects with `identifier` field for WorkflowJobs.\n   * A workflow job can contain `orbParameter` which are passed to orbs.\n   * This map is directly added as `Record<string,any>` to job.\n   * So we gonna add those after the default field of WorkflowJob.\n   * @see https://circleci.com/developer/orbs/orb/circleci/node#usage-install_nodejs\n   * @param jobs\n   */\n  private renderJobs = (jobs: WorkflowJob[] = []): any => {\n    let result: any = [];\n    for (const job of jobs ?? []) {\n      const { identifier, orbParameters, ...reduced } = job;\n      if (isObjectContainingFieldExactly(job, \"identifier\")) {\n        result = [...result, identifier];\n      } else {\n        result = [\n          ...result,\n          { [identifier]: { ...reduced, ...orbParameters } },\n        ];\n      }\n    }\n    return result;\n  };\n\n  /**\n   * Add a Circleci Orb to pipeline. Will throw error if the orb already exists\n   * @param name\n   * @param orb\n   */\n  public addOrb(name: string, orb: string) {\n    if (this.orbs[name] !== undefined) {\n      throw new Error(`Circleci Config already contains an orb named ${name}.`);\n    }\n    this.orbs[name] = orb;\n  }\n\n  /**\n   * Snake case for listed keys. There are too many exceptions needed to do it recursive without a whitelist.\n   * This list needs to be updated once we add field with snake case like `aws_auth`.\n   * @param input\n   */\n  private snakeCase = (input: any): any => {\n    const snakeCaseKeyWords = [\n      \"awsAuth\",\n      \"workingDirectory\",\n      \"resourceClass\",\n      \"dockerLayerCaching\",\n      \"noOutputTimeout\",\n      \"circleciIpRanges\",\n    ];\n    return snakeCaseKeys(input, true, snakeCaseKeyWords);\n  };\n}\n\nexport const isObjectContainingFieldExactly = (\n  obj: any,\n  field: string\n): boolean => {\n  return Object.keys(obj).length == 1 && Object.keys(obj).includes(field);\n};\n"]}