projen
Version:
CDK for software projects
148 lines • 23 kB
JavaScript
;
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.99.16" };
/**
* 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"]}