projen
Version:
CDK for software projects
272 lines • 40.8 kB
JavaScript
"use strict";
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.BuildWorkflow = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const path = require("path");
const component_1 = require("../component");
const github_1 = require("../github");
const constants_1 = require("../github/constants");
const util_1 = require("../github/private/util");
const workflow_actions_1 = require("../github/workflow-actions");
const workflows_model_1 = require("../github/workflows-model");
const javascript_1 = require("../javascript");
const runner_options_1 = require("../runner-options");
const consts_1 = require("./private/consts");
const name_1 = require("../util/name");
const path_1 = require("../util/path");
class BuildWorkflow extends component_1.Component {
constructor(project, options) {
super(project);
this._postBuildJobs = [];
const github = github_1.GitHub.of(this.project.root);
if (!github) {
throw new Error("BuildWorkflow is currently only supported for GitHub projects");
}
this.github = github;
this.preBuildSteps = options.preBuildSteps ?? [];
this.postBuildSteps = options.postBuildSteps ?? [];
this.gitIdentity = options.gitIdentity ?? constants_1.DEFAULT_GITHUB_ACTIONS_USER;
this.buildTask = options.buildTask;
this.artifactsDirectory =
options.artifactsDirectory ?? consts_1.DEFAULT_ARTIFACTS_DIRECTORY;
(0, util_1.ensureNotHiddenPath)(this.artifactsDirectory, "artifactsDirectory");
this.name = options.name ?? (0, name_1.workflowNameForProject)("build", this.project);
const mutableBuilds = options.mutableBuild ?? true;
this.workflow = new github_1.GithubWorkflow(github, this.name);
this.workflow.on(options.workflowTriggers ?? {
pullRequest: {},
workflowDispatch: {}, // allow manual triggering
});
this.addBuildJob(options);
if (mutableBuilds) {
this.addSelfMutationJob(options);
}
if (project instanceof javascript_1.NodeProject) {
project.addPackageIgnore(constants_1.PERMISSION_BACKUP_FILE);
}
}
addBuildJob(options) {
const projectPathRelativeToRoot = path.relative(this.project.root.outdir, this.project.outdir);
const jobConfig = {
...(0, runner_options_1.filteredRunsOnOptions)(options.runsOn, options.runsOnGroup),
container: options.containerImage
? { image: options.containerImage }
: undefined,
env: {
CI: "true",
...options.env,
},
permissions: {
contents: workflows_model_1.JobPermission.WRITE,
...options.permissions,
},
defaults: this.project.parent // is subproject,
? {
run: {
workingDirectory: (0, path_1.ensureRelativePathStartsWithDot)(projectPathRelativeToRoot),
},
}
: undefined,
steps: (() => this.renderBuildSteps(projectPathRelativeToRoot)),
outputs: {
[consts_1.SELF_MUTATION_HAPPENED_OUTPUT]: {
stepId: consts_1.SELF_MUTATION_STEP,
outputName: consts_1.SELF_MUTATION_HAPPENED_OUTPUT,
},
},
};
this.workflow.addJob(consts_1.BUILD_JOBID, jobConfig);
}
/**
* Returns a list of job IDs that are part of the build.
*/
get buildJobIds() {
return [consts_1.BUILD_JOBID, ...this._postBuildJobs];
}
/**
* Adds steps that are executed after the build.
* @param steps The job steps
*/
addPostBuildSteps(...steps) {
this.postBuildSteps.push(...steps);
}
/**
* Adds another job to the build workflow which is executed after the build
* job succeeded.
*
* Jobs are executed _only_ if the build did NOT self mutate. If the build
* self-mutate, the branch will either be updated or the build will fail (in
* forks), so there is no point in executing the post-build job.
*
* @param id The id of the new job
* @param job The job specification
*/
addPostBuildJob(id, job) {
const steps = [];
steps.push({
name: "Download build artifacts",
uses: "actions/download-artifact@v5",
with: {
name: constants_1.BUILD_ARTIFACT_NAME,
path: this.artifactsDirectory,
},
}, {
name: "Restore build artifact permissions",
continueOnError: true,
run: [
`cd ${this.artifactsDirectory} && setfacl --restore=${constants_1.PERMISSION_BACKUP_FILE}`,
].join("\n"),
});
steps.push(...(job.steps ?? []));
this.workflow.addJob(id, {
needs: [consts_1.BUILD_JOBID],
// only run if build did not self-mutate
if: `\${{ !${consts_1.SELF_MUTATION_CONDITION} }}`,
...job,
steps: steps,
});
// add to the list of build job IDs
this._postBuildJobs.push(id);
}
/**
* Run a task as a job within the build workflow which is executed after the
* build job succeeded.
*
* The job will have access to build artifacts and will install project
* dependencies in order to be able to run any commands used in the tasks.
*
* Jobs are executed _only_ if the build did NOT self mutate. If the build
* self-mutate, the branch will either be updated or the build will fail (in
* forks), so there is no point in executing the post-build job.
*
* @param options Specify tools and other options
*/
addPostBuildJobTask(task, options = {}) {
this.addPostBuildJobCommands(`post-build-${task.name}`, [`${this.project.projenCommand} ${task.name}`], {
checkoutRepo: true,
installDeps: true,
tools: options.tools,
...(0, runner_options_1.filteredRunsOnOptions)(options.runsOn, options.runsOnGroup),
});
}
/**
* Run a sequence of commands as a job within the build workflow which is
* executed after the build job succeeded.
*
* Jobs are executed _only_ if the build did NOT self mutate. If the build
* self-mutate, the branch will either be updated or the build will fail (in
* forks), so there is no point in executing the post-build job.
*
* @param options Specify tools and other options
*/
addPostBuildJobCommands(id, commands, options) {
const steps = [];
if (options?.checkoutRepo) {
steps.push(github_1.WorkflowSteps.checkout({
with: {
ref: consts_1.PULL_REQUEST_REF,
repository: consts_1.PULL_REQUEST_REPOSITORY,
...(this.github.downloadLfs ? { lfs: true } : {}),
},
}));
}
if (options?.checkoutRepo &&
options?.installDeps &&
this.project instanceof javascript_1.NodeProject) {
steps.push({
name: "Install dependencies",
run: `${this.project.package.installCommand}`,
});
}
steps.push({ run: commands.join("\n") });
this.addPostBuildJob(id, {
permissions: {
contents: workflows_model_1.JobPermission.READ,
},
tools: options?.tools,
...(0, runner_options_1.filteredRunsOnOptions)(options?.runsOn, options?.runsOnGroup),
steps,
});
}
addSelfMutationJob(options) {
this.workflow.addJob("self-mutation", {
...(0, runner_options_1.filteredRunsOnOptions)(options.runsOn, options.runsOnGroup),
permissions: {
contents: workflows_model_1.JobPermission.WRITE,
},
needs: [consts_1.BUILD_JOBID],
if: `always() && ${consts_1.SELF_MUTATION_CONDITION} && ${consts_1.NOT_FORK}`,
steps: [
...this.workflow.projenCredentials.setupSteps,
...workflow_actions_1.WorkflowActions.checkoutWithPatch({
// we need to use a PAT so that our push will trigger the build workflow
token: this.workflow.projenCredentials.tokenRef,
ref: consts_1.PULL_REQUEST_REF,
repository: consts_1.PULL_REQUEST_REPOSITORY,
lfs: this.github.downloadLfs,
}),
github_1.WorkflowSteps.setupGitIdentity({ gitIdentity: this.gitIdentity }),
{
name: "Push changes",
env: {
PULL_REQUEST_REF: consts_1.PULL_REQUEST_REF,
},
run: [
"git add .",
'git commit -s -m "chore: self mutation"',
`git push origin "HEAD:$PULL_REQUEST_REF"`,
].join("\n"),
},
],
});
}
/**
* Called (lazily) during synth to render the build job steps.
*/
renderBuildSteps(projectPathRelativeToRoot) {
return [
github_1.WorkflowSteps.checkout({
with: {
ref: consts_1.PULL_REQUEST_REF,
repository: consts_1.PULL_REQUEST_REPOSITORY,
...(this.github.downloadLfs ? { lfs: true } : {}),
},
}),
...this.preBuildSteps,
{
name: this.buildTask.name,
run: this.github.project.runTaskCommand(this.buildTask),
},
...this.postBuildSteps,
// check for mutations and upload a git patch file as an artifact
...workflow_actions_1.WorkflowActions.uploadGitPatch({
stepId: consts_1.SELF_MUTATION_STEP,
outputName: consts_1.SELF_MUTATION_HAPPENED_OUTPUT,
mutationError: "Files were changed during build (see build log). If this was triggered from a fork, you will need to update your branch.",
}),
// upload the build artifact only if we have post-build jobs (otherwise, there's no point)
...(this._postBuildJobs.length == 0
? []
: [
{
name: "Backup artifact permissions",
continueOnError: true,
run: `cd ${this.artifactsDirectory} && getfacl -R . > ${constants_1.PERMISSION_BACKUP_FILE}`,
},
github_1.WorkflowSteps.uploadArtifact({
with: {
name: constants_1.BUILD_ARTIFACT_NAME,
path: this.project.parent
? `${projectPathRelativeToRoot}/${this.artifactsDirectory}`
: this.artifactsDirectory,
},
}),
]),
];
}
}
exports.BuildWorkflow = BuildWorkflow;
_a = JSII_RTTI_SYMBOL_1;
BuildWorkflow[_a] = { fqn: "projen.build.BuildWorkflow", version: "0.98.32" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"build-workflow.js","sourceRoot":"","sources":["../../src/build/build-workflow.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from \"path\";\nimport { Task } from \"..\";\nimport { Component } from \"../component\";\nimport {\n  GitHub,\n  GithubWorkflow,\n  GitIdentity,\n  workflows,\n  WorkflowSteps,\n} from \"../github\";\nimport {\n  BUILD_ARTIFACT_NAME,\n  DEFAULT_GITHUB_ACTIONS_USER,\n  PERMISSION_BACKUP_FILE,\n} from \"../github/constants\";\nimport { ensureNotHiddenPath } from \"../github/private/util\";\nimport { WorkflowActions } from \"../github/workflow-actions\";\nimport {\n  Job,\n  JobPermission,\n  JobPermissions,\n  JobStep,\n  Tools,\n  Triggers,\n} from \"../github/workflows-model\";\nimport { NodeProject } from \"../javascript\";\nimport { Project } from \"../project\";\nimport { GroupRunnerOptions, filteredRunsOnOptions } from \"../runner-options\";\nimport {\n  BUILD_JOBID,\n  DEFAULT_ARTIFACTS_DIRECTORY,\n  NOT_FORK,\n  PULL_REQUEST_REF,\n  PULL_REQUEST_REPOSITORY,\n  SELF_MUTATION_CONDITION,\n  SELF_MUTATION_HAPPENED_OUTPUT,\n  SELF_MUTATION_STEP,\n} from \"./private/consts\";\nimport { workflowNameForProject } from \"../util/name\";\nimport { ensureRelativePathStartsWithDot } from \"../util/path\";\n\nexport interface BuildWorkflowCommonOptions {\n  /**\n   * Name of the buildfile (e.g. \"build\" becomes \"build.yml\").\n   *\n   * @default \"build\"\n   */\n  readonly name?: string;\n\n  /**\n   * Steps to execute before the build.\n   * @default []\n   */\n  readonly preBuildSteps?: JobStep[];\n\n  /**\n   * Build workflow triggers\n   * @default \"{ pullRequest: {}, workflowDispatch: {} }\"\n   */\n  readonly workflowTriggers?: Triggers;\n\n  /**\n   * Permissions granted to the build job\n   * To limit job permissions for `contents`, the desired permissions have to be explicitly set, e.g.: `{ contents: JobPermission.NONE }`\n   * @default `{ contents: JobPermission.WRITE }`\n   */\n  readonly permissions?: JobPermissions;\n\n  /**\n   * Build environment variables.\n   * @default {}\n   */\n  readonly env?: { [key: string]: string };\n}\n\nexport interface BuildWorkflowOptions extends BuildWorkflowCommonOptions {\n  /**\n   * The task to execute in order to build the project.\n   */\n  readonly buildTask: Task;\n\n  /**\n   * A name of a directory that includes build artifacts.\n   * @default \"dist\"\n   */\n  readonly artifactsDirectory?: string;\n\n  /**\n   * The container image to use for builds.\n   * @default - the default workflow container\n   */\n  readonly containerImage?: string;\n\n  /**\n   * Automatically update files modified during builds to pull-request branches.\n   * This means that any files synthesized by projen or e.g. test snapshots will\n   * always be up-to-date before a PR is merged.\n   *\n   * Implies that PR builds do not have anti-tamper checks.\n   *\n   * This is enabled by default only if `githubTokenSecret` is set. Otherwise it\n   * is disabled, which implies that file changes that happen during build will\n   * not be pushed back to the branch.\n   *\n   * @default true\n   */\n  readonly mutableBuild?: boolean;\n\n  /**\n   * Steps to execute after build.\n   * @default []\n   */\n  readonly postBuildSteps?: JobStep[];\n\n  /**\n   * Git identity to use for the workflow.\n   * @default - default GitHub Actions user\n   */\n  readonly gitIdentity?: GitIdentity;\n\n  /**\n   * Github Runner selection labels\n   * @default [\"ubuntu-latest\"]\n   * @description Defines a target Runner by labels\n   * @throws {Error} if both `runsOn` and `runsOnGroup` are specified\n   */\n  readonly runsOn?: string[];\n\n  /**\n   * Github Runner Group selection options\n   * @description Defines a target Runner Group by name and/or labels\n   * @throws {Error} if both `runsOn` and `runsOnGroup` are specified\n   */\n  readonly runsOnGroup?: GroupRunnerOptions;\n}\n\nexport class BuildWorkflow extends Component {\n  /**\n   * Name of generated github workflow\n   */\n  public readonly name: string;\n\n  private readonly postBuildSteps: JobStep[];\n  private readonly preBuildSteps: JobStep[];\n  private readonly gitIdentity: GitIdentity;\n  private readonly buildTask: Task;\n  private readonly github: GitHub;\n  private readonly workflow: GithubWorkflow;\n  private readonly artifactsDirectory: string;\n\n  private readonly _postBuildJobs: string[] = [];\n\n  constructor(project: Project, options: BuildWorkflowOptions) {\n    super(project);\n\n    const github = GitHub.of(this.project.root);\n    if (!github) {\n      throw new Error(\n        \"BuildWorkflow is currently only supported for GitHub projects\"\n      );\n    }\n\n    this.github = github;\n    this.preBuildSteps = options.preBuildSteps ?? [];\n    this.postBuildSteps = options.postBuildSteps ?? [];\n    this.gitIdentity = options.gitIdentity ?? DEFAULT_GITHUB_ACTIONS_USER;\n    this.buildTask = options.buildTask;\n    this.artifactsDirectory =\n      options.artifactsDirectory ?? DEFAULT_ARTIFACTS_DIRECTORY;\n    ensureNotHiddenPath(this.artifactsDirectory, \"artifactsDirectory\");\n    this.name = options.name ?? workflowNameForProject(\"build\", this.project);\n    const mutableBuilds = options.mutableBuild ?? true;\n\n    this.workflow = new GithubWorkflow(github, this.name);\n    this.workflow.on(\n      options.workflowTriggers ?? {\n        pullRequest: {},\n        workflowDispatch: {}, // allow manual triggering\n      }\n    );\n\n    this.addBuildJob(options);\n\n    if (mutableBuilds) {\n      this.addSelfMutationJob(options);\n    }\n\n    if (project instanceof NodeProject) {\n      project.addPackageIgnore(PERMISSION_BACKUP_FILE);\n    }\n  }\n\n  private addBuildJob(options: BuildWorkflowOptions) {\n    const projectPathRelativeToRoot = path.relative(\n      this.project.root.outdir,\n      this.project.outdir\n    );\n    const jobConfig: workflows.Job = {\n      ...filteredRunsOnOptions(options.runsOn, options.runsOnGroup),\n      container: options.containerImage\n        ? { image: options.containerImage }\n        : undefined,\n      env: {\n        CI: \"true\",\n        ...options.env,\n      },\n      permissions: {\n        contents: JobPermission.WRITE,\n        ...options.permissions,\n      },\n      defaults: this.project.parent // is subproject,\n        ? {\n            run: {\n              workingDirectory: ensureRelativePathStartsWithDot(\n                projectPathRelativeToRoot\n              ),\n            },\n          }\n        : undefined,\n      steps: (() => this.renderBuildSteps(projectPathRelativeToRoot)) as any,\n      outputs: {\n        [SELF_MUTATION_HAPPENED_OUTPUT]: {\n          stepId: SELF_MUTATION_STEP,\n          outputName: SELF_MUTATION_HAPPENED_OUTPUT,\n        },\n      },\n    };\n\n    this.workflow.addJob(BUILD_JOBID, jobConfig);\n  }\n\n  /**\n   * Returns a list of job IDs that are part of the build.\n   */\n  public get buildJobIds(): string[] {\n    return [BUILD_JOBID, ...this._postBuildJobs];\n  }\n\n  /**\n   * Adds steps that are executed after the build.\n   * @param steps The job steps\n   */\n  public addPostBuildSteps(...steps: JobStep[]): void {\n    this.postBuildSteps.push(...steps);\n  }\n\n  /**\n   * Adds another job to the build workflow which is executed after the build\n   * job succeeded.\n   *\n   * Jobs are executed _only_ if the build did NOT self mutate. If the build\n   * self-mutate, the branch will either be updated or the build will fail (in\n   * forks), so there is no point in executing the post-build job.\n   *\n   * @param id The id of the new job\n   * @param job The job specification\n   */\n  public addPostBuildJob(id: string, job: Job) {\n    const steps = [];\n\n    steps.push(\n      {\n        name: \"Download build artifacts\",\n        uses: \"actions/download-artifact@v5\",\n        with: {\n          name: BUILD_ARTIFACT_NAME,\n          path: this.artifactsDirectory,\n        },\n      },\n      {\n        name: \"Restore build artifact permissions\",\n        continueOnError: true,\n        run: [\n          `cd ${this.artifactsDirectory} && setfacl --restore=${PERMISSION_BACKUP_FILE}`,\n        ].join(\"\\n\"),\n      }\n    );\n\n    steps.push(...(job.steps ?? []));\n\n    this.workflow.addJob(id, {\n      needs: [BUILD_JOBID],\n      // only run if build did not self-mutate\n      if: `\\${{ !${SELF_MUTATION_CONDITION} }}`,\n      ...job,\n      steps: steps,\n    });\n\n    // add to the list of build job IDs\n    this._postBuildJobs.push(id);\n  }\n\n  /**\n   * Run a task as a job within the build workflow which is executed after the\n   * build job succeeded.\n   *\n   * The job will have access to build artifacts and will install project\n   * dependencies in order to be able to run any commands used in the tasks.\n   *\n   * Jobs are executed _only_ if the build did NOT self mutate. If the build\n   * self-mutate, the branch will either be updated or the build will fail (in\n   * forks), so there is no point in executing the post-build job.\n   *\n   * @param options Specify tools and other options\n   */\n  public addPostBuildJobTask(\n    task: Task,\n    options: AddPostBuildJobTaskOptions = {}\n  ) {\n    this.addPostBuildJobCommands(\n      `post-build-${task.name}`,\n      [`${this.project.projenCommand} ${task.name}`],\n      {\n        checkoutRepo: true,\n        installDeps: true,\n        tools: options.tools,\n        ...filteredRunsOnOptions(options.runsOn, options.runsOnGroup),\n      }\n    );\n  }\n\n  /**\n   * Run a sequence of commands as a job within the build workflow which is\n   * executed after the build job succeeded.\n   *\n   * Jobs are executed _only_ if the build did NOT self mutate. If the build\n   * self-mutate, the branch will either be updated or the build will fail (in\n   * forks), so there is no point in executing the post-build job.\n   *\n   * @param options Specify tools and other options\n   */\n  public addPostBuildJobCommands(\n    id: string,\n    commands: string[],\n    options?: AddPostBuildJobCommandsOptions\n  ) {\n    const steps = [];\n\n    if (options?.checkoutRepo) {\n      steps.push(\n        WorkflowSteps.checkout({\n          with: {\n            ref: PULL_REQUEST_REF,\n            repository: PULL_REQUEST_REPOSITORY,\n            ...(this.github.downloadLfs ? { lfs: true } : {}),\n          },\n        })\n      );\n    }\n\n    if (\n      options?.checkoutRepo &&\n      options?.installDeps &&\n      this.project instanceof NodeProject\n    ) {\n      steps.push({\n        name: \"Install dependencies\",\n        run: `${this.project.package.installCommand}`,\n      });\n    }\n\n    steps.push({ run: commands.join(\"\\n\") });\n\n    this.addPostBuildJob(id, {\n      permissions: {\n        contents: JobPermission.READ,\n      },\n      tools: options?.tools,\n      ...filteredRunsOnOptions(options?.runsOn, options?.runsOnGroup),\n      steps,\n    });\n  }\n\n  private addSelfMutationJob(options: BuildWorkflowOptions) {\n    this.workflow.addJob(\"self-mutation\", {\n      ...filteredRunsOnOptions(options.runsOn, options.runsOnGroup),\n      permissions: {\n        contents: JobPermission.WRITE,\n      },\n      needs: [BUILD_JOBID],\n      if: `always() && ${SELF_MUTATION_CONDITION} && ${NOT_FORK}`,\n      steps: [\n        ...this.workflow.projenCredentials.setupSteps,\n        ...WorkflowActions.checkoutWithPatch({\n          // we need to use a PAT so that our push will trigger the build workflow\n          token: this.workflow.projenCredentials.tokenRef,\n          ref: PULL_REQUEST_REF,\n          repository: PULL_REQUEST_REPOSITORY,\n          lfs: this.github.downloadLfs,\n        }),\n        WorkflowSteps.setupGitIdentity({ gitIdentity: this.gitIdentity }),\n        {\n          name: \"Push changes\",\n          env: {\n            PULL_REQUEST_REF,\n          },\n          run: [\n            \"git add .\",\n            'git commit -s -m \"chore: self mutation\"',\n            `git push origin \"HEAD:$PULL_REQUEST_REF\"`,\n          ].join(\"\\n\"),\n        },\n      ],\n    });\n  }\n\n  /**\n   * Called (lazily) during synth to render the build job steps.\n   */\n  private renderBuildSteps(projectPathRelativeToRoot: string): JobStep[] {\n    return [\n      WorkflowSteps.checkout({\n        with: {\n          ref: PULL_REQUEST_REF,\n          repository: PULL_REQUEST_REPOSITORY,\n          ...(this.github.downloadLfs ? { lfs: true } : {}),\n        },\n      }),\n\n      ...this.preBuildSteps,\n\n      {\n        name: this.buildTask.name,\n        run: this.github.project.runTaskCommand(this.buildTask),\n      },\n\n      ...this.postBuildSteps,\n\n      // check for mutations and upload a git patch file as an artifact\n      ...WorkflowActions.uploadGitPatch({\n        stepId: SELF_MUTATION_STEP,\n        outputName: SELF_MUTATION_HAPPENED_OUTPUT,\n        mutationError:\n          \"Files were changed during build (see build log). If this was triggered from a fork, you will need to update your branch.\",\n      }),\n\n      // upload the build artifact only if we have post-build jobs (otherwise, there's no point)\n      ...(this._postBuildJobs.length == 0\n        ? []\n        : [\n            {\n              name: \"Backup artifact permissions\",\n              continueOnError: true,\n              run: `cd ${this.artifactsDirectory} && getfacl -R . > ${PERMISSION_BACKUP_FILE}`,\n            },\n            WorkflowSteps.uploadArtifact({\n              with: {\n                name: BUILD_ARTIFACT_NAME,\n                path: this.project.parent\n                  ? `${projectPathRelativeToRoot}/${this.artifactsDirectory}`\n                  : this.artifactsDirectory,\n              },\n            }),\n          ]),\n    ];\n  }\n}\n\n/**\n * Options for `BuildWorkflow.addPostBuildJobTask`\n */\nexport interface AddPostBuildJobTaskOptions {\n  /**\n   * Tools that should be installed before the task is run.\n   */\n  readonly tools?: Tools;\n\n  /**\n   * Github Runner selection labels\n   * @default [\"ubuntu-latest\"]\n   * @description Defines a target Runner by labels\n   * @throws {Error} if both `runsOn` and `runsOnGroup` are specified\n   */\n  readonly runsOn?: string[];\n\n  /**\n   * Github Runner Group selection options\n   * @description Defines a target Runner Group by name and/or labels\n   * @throws {Error} if both `runsOn` and `runsOnGroup` are specified\n   */\n  readonly runsOnGroup?: GroupRunnerOptions;\n}\n\n/**\n * Options for `BuildWorkflow.addPostBuildJobCommands`\n */\nexport interface AddPostBuildJobCommandsOptions {\n  /**\n   * Tools that should be installed before the commands are run.\n   */\n  readonly tools?: Tools;\n\n  /**\n   * Check out the repository at the pull request branch before commands are\n   * run.\n   *\n   * @default false\n   */\n  readonly checkoutRepo?: boolean;\n\n  /**\n   * Install project dependencies before running commands. `checkoutRepo` must\n   * also be set to true.\n   *\n   * Currently only supported for `NodeProject`.\n   *\n   * @default false\n   */\n  readonly installDeps?: boolean;\n\n  /**\n   * Github Runner selection labels\n   * @default [\"ubuntu-latest\"]\n   * @description Defines a target Runner by labels\n   * @throws {Error} if both `runsOn` and `runsOnGroup` are specified\n   */\n  readonly runsOn?: string[];\n\n  /**\n   * Github Runner Group selection options\n   * @description Defines a target Runner Group by name and/or labels\n   * @throws {Error} if both `runsOn` and `runsOnGroup` are specified\n   */\n  readonly runsOnGroup?: GroupRunnerOptions;\n}\n"]}