projen
Version:
CDK for software projects
153 lines • 21.8 kB
JavaScript
;
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.WorkflowActions = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const constants_1 = require("./constants");
const util_1 = require("./private/util");
const workflow_steps_1 = require("./workflow-steps");
const REPO = (0, util_1.context)("github.repository");
const RUN_ID = (0, util_1.context)("github.run_id");
const SERVER_URL = (0, util_1.context)("github.server_url");
const RUN_URL = `${SERVER_URL}/${REPO}/actions/runs/${RUN_ID}`;
const GIT_PATCH_FILE_DEFAULT = "repo.patch";
const RUNNER_TEMP = "${{ runner.temp }}";
/**
* A set of utility functions for creating GitHub actions in workflows.
*/
class WorkflowActions {
/**
* Creates a .patch file from the current git diff and uploads it as an
* artifact. Use `checkoutWithPatch` to download and apply in another job.
*
* If a patch was uploaded, the action can optionally fail the job.
*
* @param options Options
* @returns Job steps
*/
static uploadGitPatch(options) {
const MUTATIONS_FOUND = `steps.${options.stepId}.outputs.${options.outputName}`;
const GIT_PATCH_FILE = options.patchFile ?? GIT_PATCH_FILE_DEFAULT;
const steps = [
{
id: options.stepId,
name: options.stepName ?? "Find mutations",
shell: "bash",
run: [
"git add .",
`git diff --staged --patch --exit-code > ${GIT_PATCH_FILE} || echo "${options.outputName}=true" >> $GITHUB_OUTPUT`,
].join("\n"),
// always run from root of repository
// overrides default working directory which is set by some workflows using this function
workingDirectory: "./",
},
workflow_steps_1.WorkflowSteps.uploadArtifact({
if: MUTATIONS_FOUND,
name: "Upload patch",
with: {
name: GIT_PATCH_FILE,
path: GIT_PATCH_FILE,
includeHiddenFiles: (0, util_1.isHiddenPath)(GIT_PATCH_FILE) ? true : undefined,
},
}),
];
if (options.mutationError) {
steps.push({
name: "Fail build on mutation",
if: MUTATIONS_FOUND,
run: [
`echo "::error::${options.mutationError}"`,
`cat ${GIT_PATCH_FILE}`,
"exit 1",
].join("\n"),
});
}
return steps;
}
/**
* Checks out a repository and applies a git patch that was created using
* `uploadGitPatch`.
*
* @param options Options
* @returns Job steps
*/
static checkoutWithPatch(options = {}) {
const { patchFile, ...restOfOptions } = options;
const GIT_PATCH_FILE = options.patchFile ?? GIT_PATCH_FILE_DEFAULT;
return [
workflow_steps_1.WorkflowSteps.checkout({ with: restOfOptions }),
{
name: "Download patch",
uses: "actions/download-artifact@v8",
with: { name: GIT_PATCH_FILE, path: RUNNER_TEMP },
},
{
name: "Apply patch",
run: `[ -s ${RUNNER_TEMP}/${GIT_PATCH_FILE} ] && git apply ${RUNNER_TEMP}/${GIT_PATCH_FILE} || echo "Empty patch. Skipping."`,
},
];
}
/**
* A step that creates a pull request based on the current repo state.
*
* @param options Options
* @returns Job steps
*/
static createPullRequest(options) {
const workflowName = options.workflowName;
const branchName = options.branchName ?? `github-actions/${workflowName}`;
const stepId = options.stepId ?? "create-pr";
const stepName = options.stepName ?? "Create Pull Request";
const gitIdentity = options.gitIdentity ?? constants_1.DEFAULT_GITHUB_ACTIONS_USER;
const committer = `${gitIdentity.name} <${gitIdentity.email}>`;
const pullRequestDescription = options.pullRequestDescription
.trimEnd()
.endsWith(".")
? options.pullRequestDescription.trimEnd()
: `${options.pullRequestDescription.trimEnd()}.`;
const title = options.pullRequestTitle;
const description = [
`${pullRequestDescription} See details in [workflow run].`,
"",
`[Workflow Run]: ${RUN_URL}`,
"",
"------",
"",
`*Automatically created by projen via the "${workflowName}" workflow*`,
].join("\n");
return [
{
name: stepName,
id: stepId,
uses: "peter-evans/create-pull-request@v8",
with: {
token: options.credentials?.tokenRef,
"commit-message": `${title}\n\n${description}`,
branch: branchName,
base: options.baseBranch,
title: title,
labels: options.labels?.join(",") || undefined,
assignees: options.assignees?.join(",") || undefined,
body: description,
author: committer,
committer: committer,
signoff: options.signoff ?? true,
},
},
];
}
/**
* Configures the git identity (user name and email).
* @param id The identity to use
* @returns Job steps
*
* @deprecated use `WorkflowSteps.setupGitIdentity` instead
*/
static setupGitIdentity(id) {
return [workflow_steps_1.WorkflowSteps.setupGitIdentity({ gitIdentity: id })];
}
}
exports.WorkflowActions = WorkflowActions;
_a = JSII_RTTI_SYMBOL_1;
WorkflowActions[_a] = { fqn: "projen.github.WorkflowActions", version: "0.99.51" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"workflow-actions.js","sourceRoot":"","sources":["../../src/github/workflow-actions.ts"],"names":[],"mappings":"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","sourcesContent":["import type { GitIdentity, GithubCredentials } from \".\";\nimport { DEFAULT_GITHUB_ACTIONS_USER } from \"./constants\";\nimport { context, isHiddenPath } from \"./private/util\";\nimport type { CheckoutWith } from \"./workflow-steps\";\nimport { WorkflowSteps } from \"./workflow-steps\";\nimport type { JobStep } from \"./workflows-model\";\n\nconst REPO = context(\"github.repository\");\nconst RUN_ID = context(\"github.run_id\");\nconst SERVER_URL = context(\"github.server_url\");\nconst RUN_URL = `${SERVER_URL}/${REPO}/actions/runs/${RUN_ID}`;\nconst GIT_PATCH_FILE_DEFAULT = \"repo.patch\";\nconst RUNNER_TEMP = \"${{ runner.temp }}\";\n\n/**\n * A set of utility functions for creating GitHub actions in workflows.\n */\nexport class WorkflowActions {\n  /**\n   * Creates a .patch file from the current git diff and uploads it as an\n   * artifact. Use `checkoutWithPatch` to download and apply in another job.\n   *\n   * If a patch was uploaded, the action can optionally fail the job.\n   *\n   * @param options Options\n   * @returns Job steps\n   */\n  public static uploadGitPatch(options: UploadGitPatchOptions): JobStep[] {\n    const MUTATIONS_FOUND = `steps.${options.stepId}.outputs.${options.outputName}`;\n    const GIT_PATCH_FILE = options.patchFile ?? GIT_PATCH_FILE_DEFAULT;\n\n    const steps: JobStep[] = [\n      {\n        id: options.stepId,\n        name: options.stepName ?? \"Find mutations\",\n        shell: \"bash\",\n        run: [\n          \"git add .\",\n          `git diff --staged --patch --exit-code > ${GIT_PATCH_FILE} || echo \"${options.outputName}=true\" >> $GITHUB_OUTPUT`,\n        ].join(\"\\n\"),\n        // always run from root of repository\n        // overrides default working directory which is set by some workflows using this function\n        workingDirectory: \"./\",\n      },\n      WorkflowSteps.uploadArtifact({\n        if: MUTATIONS_FOUND,\n        name: \"Upload patch\",\n        with: {\n          name: GIT_PATCH_FILE,\n          path: GIT_PATCH_FILE,\n          includeHiddenFiles: isHiddenPath(GIT_PATCH_FILE) ? true : undefined,\n        },\n      }),\n    ];\n\n    if (options.mutationError) {\n      steps.push({\n        name: \"Fail build on mutation\",\n        if: MUTATIONS_FOUND,\n        run: [\n          `echo \"::error::${options.mutationError}\"`,\n          `cat ${GIT_PATCH_FILE}`,\n          \"exit 1\",\n        ].join(\"\\n\"),\n      });\n    }\n\n    return steps;\n  }\n  /**\n   * Checks out a repository and applies a git patch that was created using\n   * `uploadGitPatch`.\n   *\n   * @param options Options\n   * @returns Job steps\n   */\n  public static checkoutWithPatch(\n    options: CheckoutWithPatchOptions = {},\n  ): JobStep[] {\n    const { patchFile, ...restOfOptions } = options;\n    const GIT_PATCH_FILE = options.patchFile ?? GIT_PATCH_FILE_DEFAULT;\n\n    return [\n      WorkflowSteps.checkout({ with: restOfOptions }),\n      {\n        name: \"Download patch\",\n        uses: \"actions/download-artifact@v8\",\n        with: { name: GIT_PATCH_FILE, path: RUNNER_TEMP },\n      },\n      {\n        name: \"Apply patch\",\n        run: `[ -s ${RUNNER_TEMP}/${GIT_PATCH_FILE} ] && git apply ${RUNNER_TEMP}/${GIT_PATCH_FILE} || echo \"Empty patch. Skipping.\"`,\n      },\n    ];\n  }\n\n  /**\n   * A step that creates a pull request based on the current repo state.\n   *\n   * @param options Options\n   * @returns Job steps\n   */\n  public static createPullRequest(\n    options: CreatePullRequestOptions,\n  ): JobStep[] {\n    const workflowName = options.workflowName;\n    const branchName = options.branchName ?? `github-actions/${workflowName}`;\n    const stepId = options.stepId ?? \"create-pr\";\n    const stepName = options.stepName ?? \"Create Pull Request\";\n    const gitIdentity = options.gitIdentity ?? DEFAULT_GITHUB_ACTIONS_USER;\n    const committer = `${gitIdentity.name} <${gitIdentity.email}>`;\n    const pullRequestDescription = options.pullRequestDescription\n      .trimEnd()\n      .endsWith(\".\")\n      ? options.pullRequestDescription.trimEnd()\n      : `${options.pullRequestDescription.trimEnd()}.`;\n\n    const title = options.pullRequestTitle;\n    const description = [\n      `${pullRequestDescription} See details in [workflow run].`,\n      \"\",\n      `[Workflow Run]: ${RUN_URL}`,\n      \"\",\n      \"------\",\n      \"\",\n      `*Automatically created by projen via the \"${workflowName}\" workflow*`,\n    ].join(\"\\n\");\n\n    return [\n      {\n        name: stepName,\n        id: stepId,\n        uses: \"peter-evans/create-pull-request@v8\",\n        with: {\n          token: options.credentials?.tokenRef,\n          \"commit-message\": `${title}\\n\\n${description}`,\n          branch: branchName,\n          base: options.baseBranch,\n          title: title,\n          labels: options.labels?.join(\",\") || undefined,\n          assignees: options.assignees?.join(\",\") || undefined,\n          body: description,\n          author: committer,\n          committer: committer,\n          signoff: options.signoff ?? true,\n        },\n      },\n    ];\n  }\n\n  /**\n   * Configures the git identity (user name and email).\n   * @param id The identity to use\n   * @returns Job steps\n   *\n   * @deprecated use `WorkflowSteps.setupGitIdentity` instead\n   */\n  public static setupGitIdentity(id: GitIdentity): JobStep[] {\n    return [WorkflowSteps.setupGitIdentity({ gitIdentity: id })];\n  }\n}\n\n/**\n * Options for `checkoutWithPatch`.\n */\nexport interface CheckoutWithPatchOptions extends CheckoutWith {\n  /**\n   * The name of the artifact the patch is stored as.\n   * @default \".repo.patch\"\n   */\n  readonly patchFile?: string;\n}\n\n/**\n * Options for `uploadGitPatch`.\n */\nexport interface UploadGitPatchOptions {\n  /**\n   * The step ID which produces the output which indicates if a patch was created.\n   */\n  readonly stepId: string;\n\n  /**\n   * The name of the step.\n   * @default \"Find mutations\"\n   */\n  readonly stepName?: string;\n\n  /**\n   * The name of the artifact the patch is stored as.\n   * @default \".repo.patch\"\n   */\n  readonly patchFile?: string;\n\n  /**\n   * The name of the output to emit. It will be set to `true` if there was a diff.\n   */\n  readonly outputName: string;\n\n  /**\n   * Fail if a mutation was found and print this error message.\n   * @default - do not fail upon mutation\n   */\n  readonly mutationError?: string;\n}\n\nexport interface CreatePullRequestOptions {\n  /**\n   * The step ID which produces the output which indicates if a patch was created.\n   * @default \"create_pr\"\n   */\n  readonly stepId?: string;\n\n  /**\n   * The name of the step displayed on GitHub.\n   * @default \"Create Pull Request\"\n   */\n  readonly stepName?: string;\n\n  /**\n   * The job credentials used to create the pull request.\n   *\n   * Provided credentials must have permissions to create a pull request on the repository.\n   */\n  readonly credentials?: GithubCredentials;\n\n  /**\n   * The name of the workflow that will create the PR\n   */\n  readonly workflowName: string;\n\n  /**\n   * The full title used to create the pull request.\n   *\n   * If PR titles are validated in this repo, the title should comply with the respective rules.\n   */\n  readonly pullRequestTitle: string;\n\n  /**\n   * Description added to the pull request.\n   *\n   * Providence information are automatically added.\n   */\n  readonly pullRequestDescription: string;\n\n  /**\n   * Sets the pull request base branch.\n   *\n   * @default - The branch checked out in the workflow.\n   */\n  readonly baseBranch?: string;\n\n  /**\n   * The pull request branch name.\n   *\n   * @default `github-actions/${options.workflowName}`\n   */\n  readonly branchName?: string;\n\n  /**\n   * The git identity used to create the commit.\n   * @default - default GitHub Actions user\n   */\n  readonly gitIdentity?: GitIdentity;\n\n  /**\n   * Add Signed-off-by line by the committer at the end of the commit log message.\n   *\n   * @default true\n   */\n  readonly signoff?: boolean;\n\n  /**\n   * Labels to apply on the PR.\n   *\n   * @default - no labels.\n   */\n  readonly labels?: string[];\n\n  /**\n   * Assignees to add on the PR.\n   *\n   * @default - no assignees\n   */\n  readonly assignees?: string[];\n}\n"]}