projen
Version:
CDK for software projects
314 lines • 40.8 kB
JavaScript
;
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.99.34" };
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@v6",
with: {
"node-version": tools.node.version,
"package-manager-cache": "false",
},
});
}
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 concurrently, 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@v6\",\n      with: {\n        \"node-version\": tools.node.version,\n        \"package-manager-cache\": \"false\",\n      },\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"]}