UNPKG

projen

Version:

CDK for software projects

120 lines • 16.5 kB
"use strict"; 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"]}