UNPKG

projen

Version:

CDK for software projects

106 lines • 18.3 kB
"use strict"; 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, '\\"'); const credentials = workflowEngine.projenCredentials; 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}`, environment: credentials.environment, steps: [ ...credentials.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: credentials.tokenRef, auto_backport_label_prefix: labelPrefix, }, }, ], }); } } exports.PullRequestBackport = PullRequestBackport; _a = JSII_RTTI_SYMBOL_1; PullRequestBackport[_a] = { fqn: "projen.github.PullRequestBackport", version: "0.99.16" }; //# 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) =>\n        `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    const credentials = workflowEngine.projenCredentials;\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      environment: credentials.environment,\n      steps: [\n        ...credentials.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: credentials.tokenRef,\n            auto_backport_label_prefix: labelPrefix,\n          },\n        },\n      ],\n    });\n  }\n}\n"]}