projen
Version:
CDK for software projects
104 lines • 18.1 kB
JavaScript
;
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.PullRequestBackport = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const auto_approve_1 = require("./auto-approve");
const github_1 = require("./github");
const workflows_1 = require("./workflows");
const component_1 = require("../component");
const json_1 = require("../json");
const release_1 = require("../release");
class PullRequestBackport extends component_1.Component {
constructor(scope, options = {}) {
super(scope);
const workflowEngine = github_1.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 branches = options.branches ?? release_1.Release.of(this.project)?.branches ?? [];
if (branches.length === 0) {
this.project.logger.warn("PullRequestBackport could not find any target branches. Backports will not be available. Please add release branches or configure `branches` manually.");
}
const targetPrLabelsRaw = options.backportPRLabels ?? ["backport"];
const targetPrLabels = [...targetPrLabelsRaw];
const shouldAutoApprove = options.autoApproveBackport ?? true;
if (shouldAutoApprove) {
const autoApprove = this.project.components.find((c) => c instanceof auto_approve_1.AutoApprove);
if (autoApprove?.label) {
targetPrLabels.push(autoApprove.label);
}
}
const backportBranchNamePrefix = options.backportBranchNamePrefix ?? "backport/";
const labelPrefix = options.labelPrefix ?? "backport-to-";
// Configuration
this.file = new json_1.JsonFile(this, ".backportrc.json", {
obj: {
commitConflicts: options.createWithConflicts ?? true,
targetPRLabels: targetPrLabels,
backportBranchName: `${backportBranchNamePrefix}{{targetBranch}}-{{refValues}}`,
prTitle: "{{sourcePullRequest.title}} (backport #{{sourcePullRequest.number}})",
targetBranchChoices: branches,
},
// File needs to be available to the GitHub Workflow
committed: true,
});
this.project.addPackageIgnore(this.file.path);
// Workflow
this.workflow = new workflows_1.GithubWorkflow(workflowEngine, options.workflowName ?? "backport");
this.workflow.on({
pullRequestTarget: {
types: ["labeled", "unlabeled", "closed"],
},
});
// condition to detect if the PR is a backport PR
// we prefer to match the PR using labels, but will fallback to matching the branch name prefix
const branchCondition = `startsWith(github.head_ref, '${backportBranchNamePrefix}')`;
const labelConditions = targetPrLabelsRaw.map((label) => `contains(github.event.pull_request.labels.*.name, '${label}')`);
const isBackportPr = labelConditions.length
? `(${labelConditions.join(" && ")})`
: `${branchCondition})`;
const checkStep = "check_labels";
const checkOutput = "matched";
const labelPrefixEscaped = labelPrefix.replace(/"/g, '\\"');
this.workflow.addJob("backport", {
name: "Backport PR",
runsOn: ["ubuntu-latest"],
permissions: {},
// Only ever run this job if the PR is merged and not a backport PR itself
if: `github.event.pull_request.merged == true && !${isBackportPr}`,
steps: [
...workflowEngine.projenCredentials.setupSteps,
// We need a custom step to check if the PR has any of the labels that indicate that the PR should be backported.
// This is not currently possible with GH Actions expression by itself, so we use a bash script.
{
id: checkStep,
name: "Check for backport labels",
shell: "bash",
run: [
"labels='${{ toJSON(github.event.pull_request.labels.*.name) }}'",
`matched=$(echo $labels | jq '.|map(select(startswith("${labelPrefixEscaped}"))) | length')`,
`echo "${checkOutput}=$matched"`,
`echo "${checkOutput}=$matched" >> $GITHUB_OUTPUT`,
].join("\n"),
},
{
name: "Backport Action",
uses: "sqren/backport-github-action@v9.5.1",
// only run this step if we have found matching labels in the previous step
// this is to prevent workflow failures because the action fails when pre-conditions are not met
// and causes any PR to be marked with a red X, leading to error blindness.
if: `fromJSON(steps.${checkStep}.outputs.${checkOutput}) > 0`,
with: {
github_token: workflowEngine.projenCredentials.tokenRef,
auto_backport_label_prefix: labelPrefix,
},
},
],
});
}
}
exports.PullRequestBackport = PullRequestBackport;
_a = JSII_RTTI_SYMBOL_1;
PullRequestBackport[_a] = { fqn: "projen.github.PullRequestBackport", version: "0.95.2" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pull-request-backport.js","sourceRoot":"","sources":["../../src/github/pull-request-backport.ts"],"names":[],"mappings":"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","sourcesContent":["import { IConstruct } from \"constructs\";\nimport { AutoApprove } from \"./auto-approve\";\nimport { GitHub } from \"./github\";\nimport { GithubWorkflow } from \"./workflows\";\nimport { Component } from \"../component\";\nimport { JsonFile } from \"../json\";\nimport { Release } from \"../release\";\n\nexport interface PullRequestBackportOptions {\n  /**\n   * The name of the workflow.\n   *\n   * @default \"backport\"\n   */\n  readonly workflowName?: string;\n\n  /**\n   * Should this created Backport PRs with conflicts.\n   *\n   * Conflicts will have to be resolved manually, but a PR is always created.\n   * Set to `false` to prevent the backport PR from being created if there are conflicts.\n   *\n   * @default true\n   */\n  readonly createWithConflicts?: boolean;\n\n  /**\n   * The labels added to the created backport PR.\n   *\n   * @default [\"backport\"]\n   */\n  readonly backportPRLabels?: string[];\n\n  /**\n   * The prefix used to name backport branches.\n   *\n   * Make sure to include a separator at the end like `/` or `_`.\n   *\n   * @default \"backport/\"\n   */\n  readonly backportBranchNamePrefix?: string;\n\n  /**\n   * Automatically approve backport PRs if the 'auto approve' workflow is available.\n   *\n   * @default true\n   */\n  readonly autoApproveBackport?: boolean;\n\n  /**\n   * List of branches that can be a target for backports\n   *\n   * @default - allow backports to all release branches\n   */\n  readonly branches?: string[];\n\n  /**\n   * The prefix used to detect PRs that should be backported.\n   *\n   * @default \"backport-to-\"\n   */\n  readonly labelPrefix?: string;\n}\n\nexport class PullRequestBackport extends Component {\n  public readonly file: JsonFile;\n  public readonly workflow: GithubWorkflow;\n\n  public constructor(\n    scope: IConstruct,\n    options: PullRequestBackportOptions = {}\n  ) {\n    super(scope);\n\n    const workflowEngine = 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 branches =\n      options.branches ?? Release.of(this.project as any)?.branches ?? [];\n    if (branches.length === 0) {\n      this.project.logger.warn(\n        \"PullRequestBackport could not find any target branches. Backports will not be available. Please add release branches or configure `branches` manually.\"\n      );\n    }\n\n    const targetPrLabelsRaw = options.backportPRLabels ?? [\"backport\"];\n    const targetPrLabels = [...targetPrLabelsRaw];\n\n    const shouldAutoApprove = options.autoApproveBackport ?? true;\n    if (shouldAutoApprove) {\n      const autoApprove = this.project.components.find(\n        (c): c is AutoApprove => c instanceof AutoApprove\n      );\n      if (autoApprove?.label) {\n        targetPrLabels.push(autoApprove.label);\n      }\n    }\n\n    const backportBranchNamePrefix =\n      options.backportBranchNamePrefix ?? \"backport/\";\n    const labelPrefix = options.labelPrefix ?? \"backport-to-\";\n\n    // Configuration\n    this.file = new JsonFile(this, \".backportrc.json\", {\n      obj: {\n        commitConflicts: options.createWithConflicts ?? true,\n        targetPRLabels: targetPrLabels,\n        backportBranchName: `${backportBranchNamePrefix}{{targetBranch}}-{{refValues}}`,\n        prTitle:\n          \"{{sourcePullRequest.title}} (backport #{{sourcePullRequest.number}})\",\n        targetBranchChoices: branches,\n      },\n      // File needs to be available to the GitHub Workflow\n      committed: true,\n    });\n    this.project.addPackageIgnore(this.file.path);\n\n    // Workflow\n    this.workflow = new GithubWorkflow(\n      workflowEngine,\n      options.workflowName ?? \"backport\"\n    );\n    this.workflow.on({\n      pullRequestTarget: {\n        types: [\"labeled\", \"unlabeled\", \"closed\"],\n      },\n    });\n\n    // condition to detect if the PR is a backport PR\n    // we prefer to match the PR using labels, but will fallback to matching the branch name prefix\n    const branchCondition = `startsWith(github.head_ref, '${backportBranchNamePrefix}')`;\n    const labelConditions: string[] = targetPrLabelsRaw.map(\n      (label) => `contains(github.event.pull_request.labels.*.name, '${label}')`\n    );\n    const isBackportPr = labelConditions.length\n      ? `(${labelConditions.join(\" && \")})`\n      : `${branchCondition})`;\n\n    const checkStep = \"check_labels\";\n    const checkOutput = \"matched\";\n    const labelPrefixEscaped = labelPrefix.replace(/\"/g, '\\\\\"');\n\n    this.workflow.addJob(\"backport\", {\n      name: \"Backport PR\",\n      runsOn: [\"ubuntu-latest\"],\n      permissions: {},\n      // Only ever run this job if the PR is merged and not a backport PR itself\n      if: `github.event.pull_request.merged == true && !${isBackportPr}`,\n      steps: [\n        ...workflowEngine.projenCredentials.setupSteps,\n        // We need a custom step to check if the PR has any of the labels that indicate that the PR should be backported.\n        // This is not currently possible with GH Actions expression by itself, so we use a bash script.\n        {\n          id: checkStep,\n          name: \"Check for backport labels\",\n          shell: \"bash\",\n          run: [\n            \"labels='${{ toJSON(github.event.pull_request.labels.*.name) }}'\",\n            `matched=$(echo $labels | jq '.|map(select(startswith(\"${labelPrefixEscaped}\"))) | length')`,\n            `echo \"${checkOutput}=$matched\"`,\n            `echo \"${checkOutput}=$matched\" >> $GITHUB_OUTPUT`,\n          ].join(\"\\n\"),\n        },\n        {\n          name: \"Backport Action\",\n          uses: \"sqren/backport-github-action@v9.5.1\",\n          // only run this step if we have found matching labels in the previous step\n          // this is to prevent workflow failures because the action fails when pre-conditions are not met\n          // and causes any PR to be marked with a red X, leading to error blindness.\n          if: `fromJSON(steps.${checkStep}.outputs.${checkOutput}) > 0`,\n          with: {\n            github_token: workflowEngine.projenCredentials.tokenRef,\n            auto_backport_label_prefix: labelPrefix,\n          },\n        },\n      ],\n    });\n  }\n}\n"]}