UNPKG

projen

Version:

CDK for software projects

148 lines • 23 kB
"use strict"; var _a; Object.defineProperty(exports, "__esModule", { value: true }); exports.PullRequestLint = void 0; const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const _1 = require("."); const workflows_model_1 = require("./workflows-model"); const component_1 = require("../component"); const runner_options_1 = require("../runner-options"); /** * Configure validations to run on GitHub pull requests. * Only generates a file if at least one linter is configured. */ class PullRequestLint extends component_1.Component { constructor(github, options = {}) { super(github.project); this.github = github; this.options = options; const checkSemanticTitle = options.semanticTitle ?? true; const checkContributorStatement = Boolean(options.contributorStatement); // should only create a workflow if one or more linters are enabled if (!checkSemanticTitle && !checkContributorStatement) { return; } const workflow = github.addWorkflow("pull-request-lint"); workflow.on({ pullRequestTarget: { types: [ "labeled", "opened", "synchronize", "reopened", "ready_for_review", "edited", ], }, // run on merge group, but use a condition later to always succeed // needed so the workflow can be a required status check mergeGroup: {}, }); // All checks are run against the PR and can only be evaluated within a PR context // Needed so jobs can be set as required and will run successfully on merge group checks. const prCheck = "(github.event_name == 'pull_request' || github.event_name == 'pull_request_target')"; if (checkSemanticTitle) { const opts = options.semanticTitleOptions ?? {}; const types = opts.types ?? ["feat", "fix", "chore"]; const validateJob = { name: "Validate PR title", if: prCheck, ...(0, runner_options_1.filteredRunsOnOptions)(options.runsOn, options.runsOnGroup), permissions: { pullRequests: workflows_model_1.JobPermission.WRITE, }, steps: [ { uses: "amannn/action-semantic-pull-request@v6", env: { GITHUB_TOKEN: "${{ secrets.GITHUB_TOKEN }}", }, with: { types: types.join("\n"), ...(opts.scopes ? { scopes: opts.scopes.join("\n") } : {}), requireScope: opts.requireScope ?? false, }, }, ], }; workflow.addJobs({ validate: validateJob }); } if (options.contributorStatement) { const opts = options.contributorStatementOptions ?? {}; const users = opts.exemptUsers ?? []; const labels = opts.exemptLabels ?? []; const conditions = [prCheck]; const exclusions = [ ...labels.map((l) => `contains(github.event.pull_request.labels.*.name, '${l}')`), ...users.map((u) => `github.event.pull_request.user.login == '${u}'`), ]; if (exclusions.length) { conditions.push(`!(${exclusions.join(" || ")})`); } const script = (core) => { const actual = process.env.PR_BODY.replace(/\r?\n/g, "\n"); const expected = process.env.EXPECTED.replace(/\r?\n/g, "\n"); if (!actual.includes(expected)) { console.log("%j", actual); console.log("%j", expected); core.setFailed(`${process.env.HELP}: ${expected}`); } }; const helpMessage = "Contributor statement missing from PR description. Please include the following text in the PR description"; const contributorStatement = { name: "Require Contributor Statement", runsOn: options.runsOn ?? ["ubuntu-latest"], permissions: { pullRequests: workflows_model_1.JobPermission.READ, }, if: conditions.join(" && "), env: { PR_BODY: "${{ github.event.pull_request.body }}", EXPECTED: options.contributorStatement, HELP: helpMessage, }, steps: [ { uses: "actions/github-script@v8", with: { script: fnBody(script), }, }, ], }; workflow.addJobs({ contributorStatement }); } } preSynthesize() { if (this.options.contributorStatement) { // Append to PR template in preSynthesize so it's always at the end of the file const prTemplate = _1.PullRequestTemplate.of(this.project) ?? this.github.addPullRequestTemplate(); prTemplate?.addLine(""); prTemplate?.addLine("---"); prTemplate?.addLine(this.options.contributorStatement); prTemplate?.addLine(""); } } } exports.PullRequestLint = PullRequestLint; _a = JSII_RTTI_SYMBOL_1; PullRequestLint[_a] = { fqn: "projen.github.PullRequestLint", version: "0.98.32" }; /** * Helper to generate a JS script as string from a function object * @returns A prettified string of the function's body */ function fnBody(fn) { const def = fn.toString().replace(/\r?\n/g, "\n"); const body = def .substring(def.indexOf("{") + 1, def.lastIndexOf("}")) .split("\n"); const minIndentation = Math.min(...body .filter((l) => l.trim()) // ignore empty lines .map((l) => l.search(/\S|$/))); return body .map((l) => l.replace(" ".repeat(minIndentation), "")) .join("\n") .trim(); } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pull-request-lint.js","sourceRoot":"","sources":["../../src/github/pull-request-lint.ts"],"names":[],"mappings":"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","sourcesContent":["import { GitHub, PullRequestTemplate } from \".\";\nimport { Job, JobPermission } from \"./workflows-model\";\nimport { Component } from \"../component\";\nimport { GroupRunnerOptions, filteredRunsOnOptions } from \"../runner-options\";\n\n/**\n * Options for PullRequestLint\n */\nexport interface PullRequestLintOptions {\n  /**\n   * Validate that pull request titles follow Conventional Commits.\n   *\n   * @default true\n   * @see https://www.conventionalcommits.org/\n   */\n  readonly semanticTitle?: boolean;\n\n  /**\n   * Options for validating the conventional commit title linter.\n   * @default - title must start with \"feat\", \"fix\", or \"chore\"\n   */\n  readonly semanticTitleOptions?: SemanticTitleOptions;\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   * Require a contributor statement to be included in the PR description.\n   * For example confirming that the contribution has been made by the contributor and complies with the project's license.\n   *\n   * Appends the statement to the end of the Pull Request template.\n   *\n   * @default - no contributor statement is required\n   */\n  readonly contributorStatement?: string;\n\n  /**\n   * Options for requiring a contributor statement on Pull Requests\n   * @default - none\n   */\n  readonly contributorStatementOptions?: ContributorStatementOptions;\n}\n\n/**\n * Options for linting that PR titles follow Conventional Commits.\n * @see https://www.conventionalcommits.org/\n */\nexport interface SemanticTitleOptions {\n  /**\n   * Configure a list of commit types that are allowed.\n   * @default [\"feat\", \"fix\", \"chore\"]\n   */\n  readonly types?: string[];\n\n  /**\n   * Configure that a scope must always be provided.\n   * e.g. feat(ui), fix(core)\n   * @default false\n   */\n  readonly requireScope?: boolean;\n\n  /**\n   * Configure which scopes are allowed (newline-delimited).\n   * These are regex patterns auto-wrapped in `^ $`.\n   *\n   * @default - all scopes allowed\n   */\n  readonly scopes?: string[];\n}\n\n/**\n * Options for requiring a contributor statement on Pull Requests\n */\nexport interface ContributorStatementOptions {\n  /**\n   * Pull requests from these GitHub users are exempted from a contributor statement.\n   * @default - no users are exempted\n   */\n  readonly exemptUsers?: string[];\n  /**\n   * Pull requests with one of these labels are exempted from a contributor statement.\n   * @default - no labels are excluded\n   */\n  readonly exemptLabels?: string[];\n}\n\n/**\n * Configure validations to run on GitHub pull requests.\n * Only generates a file if at least one linter is configured.\n */\nexport class PullRequestLint extends Component {\n  constructor(\n    private readonly github: GitHub,\n    private readonly options: PullRequestLintOptions = {}\n  ) {\n    super(github.project);\n\n    const checkSemanticTitle = options.semanticTitle ?? true;\n    const checkContributorStatement = Boolean(options.contributorStatement);\n\n    // should only create a workflow if one or more linters are enabled\n    if (!checkSemanticTitle && !checkContributorStatement) {\n      return;\n    }\n\n    const workflow = github.addWorkflow(\"pull-request-lint\");\n    workflow.on({\n      pullRequestTarget: {\n        types: [\n          \"labeled\",\n          \"opened\",\n          \"synchronize\",\n          \"reopened\",\n          \"ready_for_review\",\n          \"edited\",\n        ],\n      },\n      // run on merge group, but use a condition later to always succeed\n      // needed so the workflow can be a required status check\n      mergeGroup: {},\n    });\n\n    // All checks are run against the PR and can only be evaluated within a PR context\n    // Needed so jobs can be set as required and will run successfully on merge group checks.\n    const prCheck =\n      \"(github.event_name == 'pull_request' || github.event_name == 'pull_request_target')\";\n\n    if (checkSemanticTitle) {\n      const opts = options.semanticTitleOptions ?? {};\n      const types = opts.types ?? [\"feat\", \"fix\", \"chore\"];\n\n      const validateJob: Job = {\n        name: \"Validate PR title\",\n        if: prCheck,\n        ...filteredRunsOnOptions(options.runsOn, options.runsOnGroup),\n        permissions: {\n          pullRequests: JobPermission.WRITE,\n        },\n        steps: [\n          {\n            uses: \"amannn/action-semantic-pull-request@v6\",\n            env: {\n              GITHUB_TOKEN: \"${{ secrets.GITHUB_TOKEN }}\",\n            },\n            with: {\n              types: types.join(\"\\n\"),\n              ...(opts.scopes ? { scopes: opts.scopes.join(\"\\n\") } : {}),\n              requireScope: opts.requireScope ?? false,\n            },\n          },\n        ],\n      };\n\n      workflow.addJobs({ validate: validateJob });\n    }\n\n    if (options.contributorStatement) {\n      const opts = options.contributorStatementOptions ?? {};\n      const users = opts.exemptUsers ?? [];\n      const labels = opts.exemptLabels ?? [];\n\n      const conditions = [prCheck];\n\n      const exclusions: string[] = [\n        ...labels.map(\n          (l) => `contains(github.event.pull_request.labels.*.name, '${l}')`\n        ),\n        ...users.map((u) => `github.event.pull_request.user.login == '${u}'`),\n      ];\n\n      if (exclusions.length) {\n        conditions.push(`!(${exclusions.join(\" || \")})`);\n      }\n\n      const script = (core: any) => {\n        const actual = process.env.PR_BODY!.replace(/\\r?\\n/g, \"\\n\");\n        const expected = process.env.EXPECTED!.replace(/\\r?\\n/g, \"\\n\");\n        if (!actual.includes(expected)) {\n          console.log(\"%j\", actual);\n          console.log(\"%j\", expected);\n          core.setFailed(`${process.env.HELP}: ${expected}`);\n        }\n      };\n\n      const helpMessage =\n        \"Contributor statement missing from PR description. Please include the following text in the PR description\";\n      const contributorStatement: Job = {\n        name: \"Require Contributor Statement\",\n        runsOn: options.runsOn ?? [\"ubuntu-latest\"],\n        permissions: {\n          pullRequests: JobPermission.READ,\n        },\n        if: conditions.join(\" && \"),\n        env: {\n          PR_BODY: \"${{ github.event.pull_request.body }}\",\n          EXPECTED: options.contributorStatement,\n          HELP: helpMessage,\n        },\n        steps: [\n          {\n            uses: \"actions/github-script@v8\",\n            with: {\n              script: fnBody(script),\n            },\n          },\n        ],\n      };\n\n      workflow.addJobs({ contributorStatement });\n    }\n  }\n\n  public preSynthesize(): void {\n    if (this.options.contributorStatement) {\n      // Append to PR template in preSynthesize so it's always at the end of the file\n      const prTemplate =\n        PullRequestTemplate.of(this.project) ??\n        this.github.addPullRequestTemplate();\n      prTemplate?.addLine(\"\");\n      prTemplate?.addLine(\"---\");\n      prTemplate?.addLine(this.options.contributorStatement);\n      prTemplate?.addLine(\"\");\n    }\n  }\n}\n\n/**\n * Helper to generate a JS script as string from a function object\n * @returns A prettified string of the function's body\n */\nfunction fnBody(fn: (...args: any[]) => any) {\n  const def = fn.toString().replace(/\\r?\\n/g, \"\\n\");\n  const body = def\n    .substring(def.indexOf(\"{\") + 1, def.lastIndexOf(\"}\"))\n    .split(\"\\n\");\n  const minIndentation = Math.min(\n    ...body\n      .filter((l) => l.trim()) // ignore empty lines\n      .map((l) => l.search(/\\S|$/))\n  );\n\n  return body\n    .map((l) => l.replace(\" \".repeat(minIndentation), \"\"))\n    .join(\"\\n\")\n    .trim();\n}\n"]}