UNPKG

projen

Version:

CDK for software projects

271 lines • 40.6 kB
"use strict"; var _a; Object.defineProperty(exports, "__esModule", { value: true }); exports.BuildWorkflow = void 0; const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const posixPath = require("node:path/posix"); 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"); 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 = (0, util_1.projectPathRelativeToRepoRoot)(this.project); 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: 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 ? posixPath.join(projectPathRelativeToRoot, this.artifactsDirectory) : this.artifactsDirectory, }, }), ]), ]; } } exports.BuildWorkflow = BuildWorkflow; _a = JSII_RTTI_SYMBOL_1; BuildWorkflow[_a] = { fqn: "projen.build.BuildWorkflow", version: "0.99.3" }; //# 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 posixPath from \"node:path/posix\";\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 {\n  ensureNotHiddenPath,\n  projectPathRelativeToRepoRoot,\n} 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\";\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 = projectPathRelativeToRepoRoot(\n      this.project\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: projectPathRelativeToRoot,\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                  ? posixPath.join(\n                      projectPathRelativeToRoot,\n                      this.artifactsDirectory\n                    )\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"]}