UNPKG

projen

Version:

CDK for software projects

311 lines • 40.6 kB
"use strict"; var _a; Object.defineProperty(exports, "__esModule", { value: true }); exports.GithubWorkflow = void 0; const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const node_path_1 = require("node:path"); const case_1 = require("case"); const _resolve_1 = require("../_resolve"); const component_1 = require("../component"); const util_1 = require("../util"); const yaml_1 = require("../yaml"); /** * Workflow for GitHub. * * A workflow is a configurable automated process made up of one or more jobs. * * @see https://docs.github.com/en/actions/reference/workflow-syntax-for-github-actions */ class GithubWorkflow extends component_1.Component { /** * All current jobs of the workflow. * * This is a read-only copy, use the respective helper methods to add, update or remove jobs. */ get jobs() { return { ...this._jobs }; } /** * @param github The GitHub component of the project this workflow belongs to. * @param name The name of the workflow, displayed under the repository's "Actions" tab. * @param options Additional options to configure the workflow. */ constructor(github, name, options = {}) { super(github.project, `${new.target.name}#${name}`); this._jobs = {}; this.events = {}; const defaultConcurrency = { cancelInProgress: false, group: "${{ github.workflow }}", }; this.name = name; this.concurrency = options.limitConcurrency ? (0, util_1.deepMerge)([ defaultConcurrency, options.concurrencyOptions, ]) : undefined; this.projenCredentials = github.projenCredentials; this.actions = github.actions; this.env = options.env; const workflowsEnabled = github.workflowsEnabled || options.force; if (workflowsEnabled) { const fileName = options.fileName ?? `${name.toLocaleLowerCase()}.yml`; const extension = (0, node_path_1.extname)(fileName).toLowerCase(); if (![".yml", ".yaml"].includes(extension)) { throw new Error(`GitHub Workflow files must have either a .yml or .yaml file extension, got: ${fileName}`); } this.file = new yaml_1.YamlFile(this.project, `.github/workflows/${fileName}`, { obj: () => this.renderWorkflow(), // GitHub needs to read the file from the repository in order to work. committed: true, }); } } /** * Add events to triggers the workflow. * * @param events The event(s) to trigger the workflow. */ on(events) { this.events = { ...this.events, ...events, }; } /** * Adds a single job to the workflow. * @param id The job name (unique within the workflow) * @param job The job specification */ addJob(id, job) { this.addJobs({ [id]: job }); } /** * Add jobs to the workflow. * * @param jobs Jobs to add. */ addJobs(jobs) { verifyJobConstraints(jobs); Object.assign(this._jobs, { ...jobs }); } /** * Get a single job from the workflow. * @param id The job name (unique within the workflow) */ getJob(id) { return this._jobs[id]; } /** * Updates a single job to the workflow. * @param id The job name (unique within the workflow) */ updateJob(id, job) { this.updateJobs({ [id]: job }); } /** * Updates jobs for this workflow * Does a complete replace, it does not try to merge the jobs * * @param jobs Jobs to update. */ updateJobs(jobs) { verifyJobConstraints(jobs); Object.assign(this._jobs, { ...jobs }); } /** * Removes a single job to the workflow. * @param id The job name (unique within the workflow) */ removeJob(id) { delete this._jobs[id]; } renderWorkflow() { return { name: this.name, "run-name": this.runName, on: snakeCaseKeys(this.events), concurrency: this.concurrency ? { group: this.concurrency?.group, "cancel-in-progress": this.concurrency.cancelInProgress, } : undefined, env: this.env, jobs: renderJobs(this._jobs, this.actions), }; } } exports.GithubWorkflow = GithubWorkflow; _a = JSII_RTTI_SYMBOL_1; GithubWorkflow[_a] = { fqn: "projen.github.GithubWorkflow", version: "0.98.32" }; function snakeCaseKeys(obj) { if (typeof obj !== "object" || obj == null) { return obj; } if (Array.isArray(obj)) { return obj.map(snakeCaseKeys); } const result = {}; for (let [k, v] of Object.entries(obj)) { if (typeof v === "object" && v != null) { v = snakeCaseKeys(v); } result[(0, case_1.snake)(k)] = v; } return result; } function renderJobs(jobs, actions) { const result = {}; for (const [name, job] of Object.entries(jobs)) { result[name] = renderJob(job); } return result; /** @see https://docs.github.com/en/actions/reference/workflow-syntax-for-github-actions */ function renderJob(job) { const steps = new Array(); // https://docs.github.com/en/actions/using-workflows/reusing-workflows#supported-keywords-for-jobs-that-call-a-reusable-workflow if ("uses" in job) { return { name: job.name, needs: arrayOrScalar(job.needs), if: job.if, permissions: (0, util_1.kebabCaseKeys)(job.permissions), concurrency: job.concurrency, uses: job.uses, with: job.with, secrets: job.secrets, strategy: renderJobStrategy(job.strategy), }; } if (job.tools) { steps.push(...setupTools(job.tools)); } const userDefinedSteps = (0, util_1.kebabCaseKeys)((0, _resolve_1.resolve)(job.steps), false); steps.push(...userDefinedSteps); return { name: job.name, needs: arrayOrScalar(job.needs), "runs-on": arrayOrScalar(job.runsOnGroup) ?? arrayOrScalar(job.runsOn), permissions: (0, util_1.kebabCaseKeys)(job.permissions), environment: job.environment, concurrency: job.concurrency, outputs: renderJobOutputs(job.outputs), env: job.env, defaults: (0, util_1.kebabCaseKeys)(job.defaults), if: job.if, steps: steps.map(renderStep), "timeout-minutes": job.timeoutMinutes, strategy: renderJobStrategy(job.strategy), "continue-on-error": job.continueOnError, container: job.container, services: job.services, }; } function renderJobOutputs(output) { if (output == null) { return undefined; } const rendered = {}; for (const [name, { stepId, outputName }] of Object.entries(output)) { rendered[name] = `\${{ steps.${stepId}.outputs.${outputName} }}`; } return rendered; } function renderJobStrategy(strategy) { if (strategy == null) { return undefined; } const rendered = { "max-parallel": strategy.maxParallel, "fail-fast": strategy.failFast, }; if (strategy.matrix) { const matrix = { include: strategy.matrix.include, exclude: strategy.matrix.exclude, }; for (const [key, values] of Object.entries(strategy.matrix.domain ?? {})) { if (key in matrix) { // A domain key was set to `include`, or `exclude`: throw new Error(`Illegal job strategy matrix key: ${key}`); } matrix[key] = values; } rendered.matrix = matrix; } return rendered; } function renderStep(step) { return { name: step.name, id: step.id, if: step.if, uses: step.uses && actions.get(step.uses), env: step.env, run: step.run, shell: step.shell, with: step.with, "continue-on-error": step.continueOnError, "timeout-minutes": step.timeoutMinutes, "working-directory": step.workingDirectory, }; } } function arrayOrScalar(arr) { if (!Array.isArray(arr)) { return arr; } if (arr == null || arr.length === 0) { return arr; } if (arr.length === 1) { return arr[0]; } return arr; } function setupTools(tools) { const steps = []; if (tools.java) { steps.push({ uses: "actions/setup-java@v5", with: { distribution: "corretto", "java-version": tools.java.version }, }); } if (tools.node) { steps.push({ uses: "actions/setup-node@v5", with: { "node-version": tools.node.version }, }); } if (tools.python) { steps.push({ uses: "actions/setup-python@v6", with: { "python-version": tools.python.version }, }); } if (tools.go) { steps.push({ uses: "actions/setup-go@v6", with: { "go-version": tools.go.version }, }); } if (tools.dotnet) { steps.push({ uses: "actions/setup-dotnet@v5", with: { "dotnet-version": tools.dotnet.version }, }); } return steps; } function verifyJobConstraints(jobs) { // verify that job has a "permissions" statement to ensure workflow can // operate in repos with default tokens set to readonly for (const [id, job] of Object.entries(jobs)) { if (!job.permissions) { throw new Error(`${id}: all workflow jobs must have a "permissions" clause to ensure workflow can operate in restricted repositories`); } } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"workflows.js","sourceRoot":"","sources":["../../src/github/workflows.ts"],"names":[],"mappings":"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","sourcesContent":["import { extname } from \"node:path\";\nimport { snake } from \"case\";\nimport { GitHubActionsProvider } from \"./actions-provider\";\nimport { GitHub } from \"./github\";\nimport { GithubCredentials } from \"./github-credentials\";\nimport * as workflows from \"./workflows-model\";\nimport { resolve } from \"../_resolve\";\nimport { Component } from \"../component\";\nimport { deepMerge, kebabCaseKeys } from \"../util\";\nimport { YamlFile } from \"../yaml\";\n\n/**\n * Options for `concurrency`.\n */\nexport interface ConcurrencyOptions {\n  /**\n   * Concurrency group controls which workflow runs will share the same concurrency limit.\n   * For example, if you specify `${{ github.workflow }}-${{ github.ref }}`, workflow runs triggered\n   * on the same branch cannot run concurrenty, but workflows runs triggered on different branches can.\n   *\n   * @default - ${{ github.workflow }}\n   *\n   * @see https://docs.github.com/en/actions/writing-workflows/choosing-what-your-workflow-does/using-concurrency#example-concurrency-groups\n   */\n  readonly group?: string;\n\n  /**\n   * When a workflow is triggered while another one (in the same group) is running, should GitHub cancel\n   * the running workflow?\n   *\n   * @default false\n   */\n  readonly cancelInProgress?: boolean;\n}\n\n/**\n * Options for `GithubWorkflow`.\n */\nexport interface GithubWorkflowOptions {\n  /**\n   * Force the creation of the workflow even if `workflows` is disabled in `GitHub`.\n   *\n   * @default false\n   */\n  readonly force?: boolean;\n\n  /**\n   * Enable concurrency limitations. Use `concurrencyOptions` to configure specific non default values.\n   *\n   * @default false\n   */\n  readonly limitConcurrency?: boolean;\n\n  /**\n   * Concurrency ensures that only a single job or workflow using the same concurrency group will run at a time. Currently in beta.\n   *\n   * @default - { group: ${{ github.workflow }}, cancelInProgress: false }\n   *\n   * @see https://docs.github.com/en/actions/learn-github-actions/workflow-syntax-for-github-actions#concurrency\n   */\n  readonly concurrencyOptions?: ConcurrencyOptions;\n\n  /**\n   * Additional environment variables to set for the workflow.\n   *\n   * @default - no additional environment variables\n   */\n  readonly env?: Record<string, string>;\n\n  /**\n   * Set a custom file name for the workflow definition file. Must include either a .yml or .yaml file extension.\n   *\n   * Use this option to set a file name for the workflow file, that is different than the display name.\n   *\n   * @example \"build-new.yml\"\n   * @example \"my-workflow.yaml\"\n   *\n   * @default - a path-safe version of the workflow name plus the .yml file ending, e.g. build.yml\n   */\n  readonly fileName?: string;\n}\n\n/**\n * Workflow for GitHub.\n *\n * A workflow is a configurable automated process made up of one or more jobs.\n *\n * @see https://docs.github.com/en/actions/reference/workflow-syntax-for-github-actions\n */\nexport class GithubWorkflow extends Component {\n  /**\n   * The name of the workflow. GitHub displays the names of your workflows under your repository's\n   * \"Actions\" tab.\n   * @see https://docs.github.com/en/actions/writing-workflows/workflow-syntax-for-github-actions#name\n   */\n  public readonly name: string;\n\n  /**\n   * All current jobs of the workflow.\n   *\n   * This is a read-only copy, use the respective helper methods to add, update or remove jobs.\n   */\n  public get jobs(): Record<\n    string,\n    workflows.Job | workflows.JobCallingReusableWorkflow\n  > {\n    return { ...this._jobs };\n  }\n\n  /**\n   * The concurrency configuration of the workflow. undefined means no concurrency limitations.\n   */\n  public readonly concurrency?: ConcurrencyOptions;\n\n  /**\n   * Additional environment variables to set for the workflow.\n   */\n  public readonly env?: Record<string, string>;\n\n  /**\n   * The workflow YAML file. May not exist if `workflowsEnabled` is false on `GitHub`.\n   */\n  public readonly file: YamlFile | undefined;\n\n  /**\n   * GitHub API authentication method used by projen workflows.\n   */\n  public readonly projenCredentials: GithubCredentials;\n\n  /**\n   * The name for workflow runs generated from the workflow. GitHub displays the\n   * workflow run name in the list of workflow runs on your repository's\n   * \"Actions\" tab. If `run-name` is omitted or is only whitespace, then the run\n   * name is set to event-specific information for the workflow run. For\n   * example, for a workflow triggered by a `push` or `pull_request` event, it\n   * is set as the commit message.\n   *\n   * This value can include expressions and can reference `github` and `inputs`\n   * contexts.\n   */\n  public runName?: string;\n\n  private readonly _jobs: Record<\n    string,\n    workflows.Job | workflows.JobCallingReusableWorkflow\n  > = {};\n  private actions: GitHubActionsProvider;\n  private events: workflows.Triggers = {};\n\n  /**\n   * @param github The GitHub component of the project this workflow belongs to.\n   * @param name The name of the workflow, displayed under the repository's \"Actions\" tab.\n   * @param options Additional options to configure the workflow.\n   */\n  constructor(\n    github: GitHub,\n    name: string,\n    options: GithubWorkflowOptions = {}\n  ) {\n    super(github.project, `${new.target.name}#${name}`);\n\n    const defaultConcurrency: ConcurrencyOptions = {\n      cancelInProgress: false,\n      group: \"${{ github.workflow }}\",\n    };\n\n    this.name = name;\n    this.concurrency = options.limitConcurrency\n      ? (deepMerge([\n          defaultConcurrency,\n          options.concurrencyOptions,\n        ]) as ConcurrencyOptions)\n      : undefined;\n    this.projenCredentials = github.projenCredentials;\n    this.actions = github.actions;\n\n    this.env = options.env;\n\n    const workflowsEnabled = github.workflowsEnabled || options.force;\n\n    if (workflowsEnabled) {\n      const fileName = options.fileName ?? `${name.toLocaleLowerCase()}.yml`;\n      const extension = extname(fileName).toLowerCase();\n\n      if (![\".yml\", \".yaml\"].includes(extension)) {\n        throw new Error(\n          `GitHub Workflow files must have either a .yml or .yaml file extension, got: ${fileName}`\n        );\n      }\n\n      this.file = new YamlFile(this.project, `.github/workflows/${fileName}`, {\n        obj: () => this.renderWorkflow(),\n        // GitHub needs to read the file from the repository in order to work.\n        committed: true,\n      });\n    }\n  }\n\n  /**\n   * Add events to triggers the workflow.\n   *\n   * @param events The event(s) to trigger the workflow.\n   */\n  public on(events: workflows.Triggers) {\n    this.events = {\n      ...this.events,\n      ...events,\n    };\n  }\n\n  /**\n   * Adds a single job to the workflow.\n   * @param id The job name (unique within the workflow)\n   * @param job The job specification\n   */\n  public addJob(\n    id: string,\n    job: workflows.Job | workflows.JobCallingReusableWorkflow\n  ): void {\n    this.addJobs({ [id]: job });\n  }\n\n  /**\n   * Add jobs to the workflow.\n   *\n   * @param jobs Jobs to add.\n   */\n  public addJobs(\n    jobs: Record<string, workflows.Job | workflows.JobCallingReusableWorkflow>\n  ) {\n    verifyJobConstraints(jobs);\n    Object.assign(this._jobs, { ...jobs });\n  }\n\n  /**\n   * Get a single job from the workflow.\n   * @param id The job name (unique within the workflow)\n   */\n  public getJob(\n    id: string\n  ): workflows.Job | workflows.JobCallingReusableWorkflow {\n    return this._jobs[id];\n  }\n\n  /**\n   * Updates a single job to the workflow.\n   * @param id The job name (unique within the workflow)\n   */\n  public updateJob(\n    id: string,\n    job: workflows.Job | workflows.JobCallingReusableWorkflow\n  ) {\n    this.updateJobs({ [id]: job });\n  }\n\n  /**\n   * Updates jobs for this workflow\n   * Does a complete replace, it does not try to merge the jobs\n   *\n   * @param jobs Jobs to update.\n   */\n  public updateJobs(\n    jobs: Record<string, workflows.Job | workflows.JobCallingReusableWorkflow>\n  ) {\n    verifyJobConstraints(jobs);\n    Object.assign(this._jobs, { ...jobs });\n  }\n\n  /**\n   * Removes a single job to the workflow.\n   * @param id The job name (unique within the workflow)\n   */\n  public removeJob(id: string) {\n    delete this._jobs[id];\n  }\n\n  private renderWorkflow() {\n    return {\n      name: this.name,\n      \"run-name\": this.runName,\n      on: snakeCaseKeys(this.events),\n      concurrency: this.concurrency\n        ? {\n            group: this.concurrency?.group,\n            \"cancel-in-progress\": this.concurrency.cancelInProgress,\n          }\n        : undefined,\n      env: this.env,\n      jobs: renderJobs(this._jobs, this.actions),\n    };\n  }\n}\n\nfunction snakeCaseKeys<T = unknown>(obj: T): T {\n  if (typeof obj !== \"object\" || obj == null) {\n    return obj;\n  }\n\n  if (Array.isArray(obj)) {\n    return obj.map(snakeCaseKeys) as any;\n  }\n\n  const result: Record<string, unknown> = {};\n  for (let [k, v] of Object.entries(obj)) {\n    if (typeof v === \"object\" && v != null) {\n      v = snakeCaseKeys(v);\n    }\n    result[snake(k)] = v;\n  }\n  return result as any;\n}\n\nfunction renderJobs(\n  jobs: Record<string, workflows.Job | workflows.JobCallingReusableWorkflow>,\n  actions: GitHubActionsProvider\n) {\n  const result: Record<string, unknown> = {};\n  for (const [name, job] of Object.entries(jobs)) {\n    result[name] = renderJob(job);\n  }\n  return result;\n\n  /** @see https://docs.github.com/en/actions/reference/workflow-syntax-for-github-actions */\n  function renderJob(\n    job: workflows.Job | workflows.JobCallingReusableWorkflow\n  ) {\n    const steps = new Array<workflows.JobStep>();\n\n    // https://docs.github.com/en/actions/using-workflows/reusing-workflows#supported-keywords-for-jobs-that-call-a-reusable-workflow\n    if (\"uses\" in job) {\n      return {\n        name: job.name,\n        needs: arrayOrScalar(job.needs),\n        if: job.if,\n        permissions: kebabCaseKeys(job.permissions),\n        concurrency: job.concurrency,\n        uses: job.uses,\n        with: job.with,\n        secrets: job.secrets,\n        strategy: renderJobStrategy(job.strategy),\n      };\n    }\n\n    if (job.tools) {\n      steps.push(...setupTools(job.tools));\n    }\n\n    const userDefinedSteps = kebabCaseKeys(resolve(job.steps), false);\n    steps.push(...userDefinedSteps);\n\n    return {\n      name: job.name,\n      needs: arrayOrScalar(job.needs),\n      \"runs-on\": arrayOrScalar(job.runsOnGroup) ?? arrayOrScalar(job.runsOn),\n      permissions: kebabCaseKeys(job.permissions),\n      environment: job.environment,\n      concurrency: job.concurrency,\n      outputs: renderJobOutputs(job.outputs),\n      env: job.env,\n      defaults: kebabCaseKeys(job.defaults),\n      if: job.if,\n      steps: steps.map(renderStep),\n      \"timeout-minutes\": job.timeoutMinutes,\n      strategy: renderJobStrategy(job.strategy),\n      \"continue-on-error\": job.continueOnError,\n      container: job.container,\n      services: job.services,\n    };\n  }\n\n  function renderJobOutputs(output: workflows.Job[\"outputs\"]) {\n    if (output == null) {\n      return undefined;\n    }\n\n    const rendered: Record<string, string> = {};\n    for (const [name, { stepId, outputName }] of Object.entries(output)) {\n      rendered[name] = `\\${{ steps.${stepId}.outputs.${outputName} }}`;\n    }\n    return rendered;\n  }\n\n  function renderJobStrategy(strategy: workflows.Job[\"strategy\"]) {\n    if (strategy == null) {\n      return undefined;\n    }\n\n    const rendered: Record<string, unknown> = {\n      \"max-parallel\": strategy.maxParallel,\n      \"fail-fast\": strategy.failFast,\n    };\n\n    if (strategy.matrix) {\n      const matrix: Record<string, unknown> = {\n        include: strategy.matrix.include,\n        exclude: strategy.matrix.exclude,\n      };\n      for (const [key, values] of Object.entries(\n        strategy.matrix.domain ?? {}\n      )) {\n        if (key in matrix) {\n          // A domain key was set to `include`, or `exclude`:\n          throw new Error(`Illegal job strategy matrix key: ${key}`);\n        }\n        matrix[key] = values;\n      }\n      rendered.matrix = matrix;\n    }\n\n    return rendered;\n  }\n\n  function renderStep(step: workflows.JobStep) {\n    return {\n      name: step.name,\n      id: step.id,\n      if: step.if,\n      uses: step.uses && actions.get(step.uses),\n      env: step.env,\n      run: step.run,\n      shell: step.shell,\n      with: step.with,\n      \"continue-on-error\": step.continueOnError,\n      \"timeout-minutes\": step.timeoutMinutes,\n      \"working-directory\": step.workingDirectory,\n    };\n  }\n}\n\nfunction arrayOrScalar<T>(arr: T | T[] | undefined): T | T[] | undefined {\n  if (!Array.isArray(arr)) {\n    return arr;\n  }\n  if (arr == null || arr.length === 0) {\n    return arr;\n  }\n  if (arr.length === 1) {\n    return arr[0];\n  }\n  return arr;\n}\n\nfunction setupTools(tools: workflows.Tools) {\n  const steps: workflows.JobStep[] = [];\n\n  if (tools.java) {\n    steps.push({\n      uses: \"actions/setup-java@v5\",\n      with: { distribution: \"corretto\", \"java-version\": tools.java.version },\n    });\n  }\n\n  if (tools.node) {\n    steps.push({\n      uses: \"actions/setup-node@v5\",\n      with: { \"node-version\": tools.node.version },\n    });\n  }\n\n  if (tools.python) {\n    steps.push({\n      uses: \"actions/setup-python@v6\",\n      with: { \"python-version\": tools.python.version },\n    });\n  }\n\n  if (tools.go) {\n    steps.push({\n      uses: \"actions/setup-go@v6\",\n      with: { \"go-version\": tools.go.version },\n    });\n  }\n\n  if (tools.dotnet) {\n    steps.push({\n      uses: \"actions/setup-dotnet@v5\",\n      with: { \"dotnet-version\": tools.dotnet.version },\n    });\n  }\n\n  return steps;\n}\n\nfunction verifyJobConstraints(\n  jobs: Record<string, workflows.Job | workflows.JobCallingReusableWorkflow>\n) {\n  // verify that job has a \"permissions\" statement to ensure workflow can\n  // operate in repos with default tokens set to readonly\n  for (const [id, job] of Object.entries(jobs)) {\n    if (!job.permissions) {\n      throw new Error(\n        `${id}: all workflow jobs must have a \"permissions\" clause to ensure workflow can operate in restricted repositories`\n      );\n    }\n  }\n}\n"]}