projen
Version:
CDK for software projects
114 lines • 18.4 kB
JavaScript
;
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.AutoQueue = exports.MergeMethod = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const component_1 = require("../component");
const gh = require("../github");
/**
* The merge method used to add the PR to the merge queue
*
* Behavior can be further configured in repository settings.
*/
var MergeMethod;
(function (MergeMethod) {
MergeMethod["SQUASH"] = "squash";
MergeMethod["MERGE"] = "merge";
MergeMethod["REBASE"] = "rebase";
})(MergeMethod || (exports.MergeMethod = MergeMethod = {}));
/**
* Automatically add pull requests to the merge queue
* PRs will be merged once they pass required checks.
*/
class AutoQueue extends component_1.Component {
constructor(scope, options = {}) {
super(scope);
const workflowEngine = gh.GitHub.of(this.project);
if (!workflowEngine) {
throw new Error(`Cannot add ${new.target.name} to project without GitHub enabled. Please enable GitHub for this project.`);
}
const labels = options.labels ?? [];
const usernames = options.allowedUsernames ?? [];
const conditions = [];
if (labels.length > 0) {
conditions.push("(" +
labels
.map((l) => `contains(github.event.pull_request.labels.*.name, '${l}')`)
.join(" || ") +
")");
}
if (usernames.length > 0) {
conditions.push("(" +
usernames
.map((u) => `github.event.pull_request.user.login == '${u}'`)
.join(" || ") +
")");
}
let needsEditedEvent = false;
if (options.targetBranches) {
// Branch conditions, based off the 'opened' or 'edited' events.
//
// The current workflow will only run if the target branch is one of the intended
// ones, so we only need to check if the event type is correct.
needsEditedEvent = true;
const isOpened = `github.event.action == 'opened'`;
const isBranchChanged = `(github.event.action == 'edited' && github.event.changes.base)`;
conditions.push(`(${isOpened} || ${isBranchChanged})`);
}
const credentials = options.projenCredentials ?? workflowEngine.projenCredentials;
const mergeMethod = options.mergeMethod ?? MergeMethod.SQUASH;
const autoQueueJob = {
name: "Set AutoQueue on PR #${{ github.event.number }}",
runsOn: options.runsOn ?? ["ubuntu-latest"],
permissions: {
pullRequests: gh.workflows.JobPermission.WRITE,
contents: gh.workflows.JobPermission.WRITE,
},
if: conditions.length ? conditions.join(" && ") : undefined,
steps: [
...credentials.setupSteps,
{
uses: "peter-evans/enable-pull-request-automerge@v3",
with: {
token: credentials.tokenRef,
"pull-request-number": "${{ github.event.number }}",
"merge-method": mergeMethod,
},
},
],
};
const workflow = workflowEngine.addWorkflow("auto-queue");
workflow.on({
// The 'pull request' event gives the workflow 'read-only' permissions on some
// pull requests (such as the ones from dependabot) when using the `GITHUB_TOKEN`
// security token. This prevents the workflow from approving these pull requests.
// Github has placed this guard so as to prevent security attacks by simply opening
// a pull request and triggering a workflow on a commit that was not vetted to make
// unintended changes to the repository.
//
// Instead use the 'pull request target' event here that gives the Github workflow
// 'read-write' permissions. This is safe because, this event, unlike the 'pull request'
// event references the BASE commit of the pull request and not the HEAD commit.
//
// We only enable auto-queue when a PR is opened, reopened or moving from Draft to Ready,
// or retargeted to a different branch. Specifically, if a user disables auto-queue we try very hard to avoid
// accidentally re-enabling it.
//
// The 'edited' trigger is only used to detect base branch changes.
pullRequestTarget: {
types: [
"opened",
"reopened",
"ready_for_review",
...(needsEditedEvent ? ["edited"] : []),
],
branches: options.targetBranches,
},
});
workflow.addJobs({ enableAutoQueue: autoQueueJob });
}
}
exports.AutoQueue = AutoQueue;
_a = JSII_RTTI_SYMBOL_1;
AutoQueue[_a] = { fqn: "projen.github.AutoQueue", version: "0.95.2" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"auto-queue.js","sourceRoot":"","sources":["../../src/github/auto-queue.ts"],"names":[],"mappings":"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","sourcesContent":["import { IConstruct } from \"constructs\";\nimport { Component } from \"../component\";\nimport * as gh from \"../github\";\n\n/**\n * The merge method used to add the PR to the merge queue\n *\n * Behavior can be further configured in repository settings.\n */\nexport enum MergeMethod {\n  SQUASH = \"squash\",\n  MERGE = \"merge\",\n  REBASE = \"rebase\",\n}\n\n/**\n * Options for 'AutoQueue'\n */\nexport interface AutoQueueOptions {\n  /**\n   * Only pull requests authored by these Github usernames will have auto-queue enabled.\n   * @default - pull requests from all users are eligible for auto-queuing\n   */\n  readonly allowedUsernames?: string[];\n\n  /**\n   * Only pull requests with one of this labels will have auto-queue enabled.\n   * @default - all pull requests are eligible for auto-queueing\n   */\n  readonly labels?: string[];\n\n  /**\n   * Choose a method for authenticating with GitHub to enable auto-queue on pull requests.\n   *\n   * The workflow cannot use a default github token. Queuing a PR\n   * with the default token will not trigger any merge queue workflows,\n   * which results in the PR just not getting merged at all.\n   *\n   * @see https://projen.io/docs/integrations/github/\n   * @default - uses credentials from the GitHub component\n   */\n  readonly projenCredentials?: gh.GithubCredentials;\n\n  /**\n   * The method used to add the PR to the merge queue\n   * Any branch protection rules must allow this merge method.\n   * @default MergeMethod.SQUASH\n   */\n  readonly mergeMethod?: MergeMethod;\n\n  /**\n   * Github Runner selection labels\n   * @default [\"ubuntu-latest\"]\n   */\n  readonly runsOn?: string[];\n\n  /**\n   * The branch names that we should auto-queue for\n   *\n   * This set of branches should be a subset of `MergeQueueOptions.targetBranches`.\n   *\n   * Be sure not to enable `autoQueue` for branches that don't have branch rules\n   * with merge requirements set up, otherwise new PRs will be merged\n   * immediately after creating without a chance for review.\n   *\n   * ## Automatically merging a set of Stacked PRs\n   *\n   * If you set this to `['main']` you can automatically merge a set of Stacked PRs\n   * in the right order. It works like this:\n   *\n   * - Create PR #1 from branch `a`, targeting `main`.\n   * - Create PR #2 from branch `b`, targeting branch `a`.\n   * - Create PR #3 from branch `c`, targeting branch `b`.\n   *\n   * Initially, PR #1 will be set to auto-merge, PRs #2 and #3 will not.\n   *\n   * Once PR #1 passes all of its requirements it will merge. That will delete\n   * branch `a` and change  the target branch of PR #2 change to `main`. At that\n   * point, auto-queueing will switch on for PR #2 and it gets merged, etc.\n   *\n   * > [!IMPORTANT]\n   * > This component will never disable AutoMerge, only enable it. So if a PR is\n   * > initially targeted at one of the branches in this list, and then\n   * > subsequently retargeted to another branch, *AutoMerge is not\n   * > automatically turned off*.\n   */\n  readonly targetBranches?: string[];\n}\n\n/**\n * Automatically add pull requests to the merge queue\n * PRs will be merged once they pass required checks.\n */\nexport class AutoQueue extends Component {\n  constructor(scope: IConstruct, options: AutoQueueOptions = {}) {\n    super(scope);\n\n    const workflowEngine = gh.GitHub.of(this.project);\n    if (!workflowEngine) {\n      throw new Error(\n        `Cannot add ${\n          new.target.name\n        } to project without GitHub enabled. Please enable GitHub for this project.`\n      );\n    }\n\n    const labels = options.labels ?? [];\n    const usernames = options.allowedUsernames ?? [];\n\n    const conditions: string[] = [];\n    if (labels.length > 0) {\n      conditions.push(\n        \"(\" +\n          labels\n            .map(\n              (l) => `contains(github.event.pull_request.labels.*.name, '${l}')`\n            )\n            .join(\" || \") +\n          \")\"\n      );\n    }\n    if (usernames.length > 0) {\n      conditions.push(\n        \"(\" +\n          usernames\n            .map((u) => `github.event.pull_request.user.login == '${u}'`)\n            .join(\" || \") +\n          \")\"\n      );\n    }\n\n    let needsEditedEvent = false;\n    if (options.targetBranches) {\n      // Branch conditions, based off the 'opened' or 'edited' events.\n      //\n      // The current workflow will only run if the target branch is one of the intended\n      // ones, so we only need to check if the event type is correct.\n      needsEditedEvent = true;\n\n      const isOpened = `github.event.action == 'opened'`;\n      const isBranchChanged = `(github.event.action == 'edited' && github.event.changes.base)`;\n\n      conditions.push(`(${isOpened} || ${isBranchChanged})`);\n    }\n\n    const credentials =\n      options.projenCredentials ?? workflowEngine.projenCredentials;\n    const mergeMethod = options.mergeMethod ?? MergeMethod.SQUASH;\n\n    const autoQueueJob: gh.workflows.Job = {\n      name: \"Set AutoQueue on PR #${{ github.event.number }}\",\n      runsOn: options.runsOn ?? [\"ubuntu-latest\"],\n      permissions: {\n        pullRequests: gh.workflows.JobPermission.WRITE,\n        contents: gh.workflows.JobPermission.WRITE,\n      },\n      if: conditions.length ? conditions.join(\" && \") : undefined,\n      steps: [\n        ...credentials.setupSteps,\n        {\n          uses: \"peter-evans/enable-pull-request-automerge@v3\",\n          with: {\n            token: credentials.tokenRef,\n            \"pull-request-number\": \"${{ github.event.number }}\",\n            \"merge-method\": mergeMethod,\n          },\n        },\n      ],\n    };\n\n    const workflow = workflowEngine.addWorkflow(\"auto-queue\");\n    workflow.on({\n      // The 'pull request' event gives the workflow 'read-only' permissions on some\n      // pull requests (such as the ones from dependabot) when using the `GITHUB_TOKEN`\n      // security token. This prevents the workflow from approving these pull requests.\n      // Github has placed this guard so as to prevent security attacks by simply opening\n      // a pull request and triggering a workflow on a commit that was not vetted to make\n      // unintended changes to the repository.\n      //\n      // Instead use the 'pull request target' event here that gives the Github workflow\n      // 'read-write' permissions. This is safe because, this event, unlike the 'pull request'\n      // event references the BASE commit of the pull request and not the HEAD commit.\n      //\n      // We only enable auto-queue when a PR is opened, reopened or moving from Draft to Ready,\n      // or retargeted to a different branch. Specifically, if a user disables auto-queue we try very hard to avoid\n      // accidentally re-enabling it.\n      //\n      // The 'edited' trigger is only used to detect base branch changes.\n      pullRequestTarget: {\n        types: [\n          \"opened\",\n          \"reopened\",\n          \"ready_for_review\",\n          ...(needsEditedEvent ? [\"edited\" as const] : []),\n        ],\n        branches: options.targetBranches,\n      },\n    });\n    workflow.addJobs({ enableAutoQueue: autoQueueJob });\n  }\n}\n"]}