From 70aa21ec55ea92889bd5adfdc65329897036b03f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E2=80=9CJamesHenry=E2=80=9D?= Date: Sun, 19 Nov 2023 23:43:25 +0400 Subject: [PATCH] fix: ensure resolved BASE_SHA was not disconnected from branch --- .husky/pre-commit | 8 +- README.md | 36 ++-- dist/index.js | 113 ++++++----- find-successful-workflow.ts | 260 ++++++++++++++++---------- package.json | 16 +- yarn.lock | 362 +++++++++++++++++++++++++++++++++++- 6 files changed, 628 insertions(+), 167 deletions(-) diff --git a/.husky/pre-commit b/.husky/pre-commit index 73de882..8e0da16 100755 --- a/.husky/pre-commit +++ b/.husky/pre-commit @@ -1,3 +1,7 @@ -#!/bin/sh +#!/usr/bin/env sh +. "$(dirname -- "$0")/_/husky.sh" + +npx --no-install lint-staged --allow-empty + +node tools/pre-commit.js -node tools/pre-commit.js \ No newline at end of file diff --git a/README.md b/README.md index 134ce35..e0441af 100644 --- a/README.md +++ b/README.md @@ -19,6 +19,7 @@ width="100%" alt="Nx - Smart, Extensible Build Framework">

**.github/workflows/ci.yml** + ```yaml # ... more CI config ... @@ -40,7 +41,7 @@ jobs: # =========================================================================== - name: Derive appropriate SHAs for base and head for `nx affected` commands uses: nrwl/nx-set-shas@v4 - + - run: | echo "BASE: ${{ env.NX_BASE }}" echo "HEAD: ${{ env.NX_HEAD }}" @@ -51,18 +52,20 @@ jobs: - name: Derive appropriate SHAs for base and head for `nx affected` commands id: setSHAs uses: nrwl/nx-set-shas@v4 - + - run: | echo "BASE: ${{ steps.setSHAs.outputs.base }}" echo "HEAD: ${{ steps.setSHAs.outputs.head }}" # ... more CI config ... ``` + ## Configuration Options + ```yaml - uses: nrwl/nx-set-shas@v4 with: @@ -70,33 +73,34 @@ jobs: # Common names for this branch include main and master. # # Default: main - main-branch-name: '' + main-branch-name: "" # Applies the derived SHAs for base and head as NX_BASE and NX_HEAD environment variables within the current Job. # # Default: true - set-environment-variables-for-job: '' + set-environment-variables-for-job: "" # By default, if no successful workflow run is found on the main branch to determine the SHA, we will log a warning and use HEAD~1. Enable this option to error and exit instead. # # Default: false - error-on-no-successful-workflow: '' + error-on-no-successful-workflow: "" # The type of event to check for the last successful commit corresponding to that workflow-id, e.g. push, pull_request, release etc. # # Default: push - last-successful-event: '' + last-successful-event: "" # The path where your repository is. This is only required for cases where the repository code is checked out or moved to a specific path. # # Default: . - working-directory: '' + working-directory: "" - # The ID of the github action workflow to check for successful run or the name of the file name containing the workflow. + # The ID of the github action workflow to check for successful run or the name of the file name containing the workflow. # E.g. 'ci.yml'. If not provided, current workflow id will be used # - workflow-id: '' + workflow-id: "" ``` + ## Permissions in v2+ @@ -104,15 +108,17 @@ jobs: This Action uses Github API to find the last successful workflow run. If your `GITHUB_TOKEN` has restrictions set please ensure you override them for the workflow to enable read access to `actions` and `contents`: + ```yaml jobs: myjob: runs-on: ubuntu-latest name: My Job permissions: - contents: 'read' - actions: 'read' + contents: "read" + actions: "read" ``` + ## Self-hosted runners @@ -120,6 +126,7 @@ jobs: This Action supports usage of your own self-hosted runners, but since it uses GitHub APIs you will need to grant it explicit access rights: + ```yaml # ... more CI config ... @@ -149,6 +156,7 @@ jobs: # ... more CI config ... ``` + ## Background @@ -157,7 +165,6 @@ When we run the `affected` command on [Nx](https://nx.dev/), we can specify 2 gi This makes it easy to set up a CI system that scales well with the continuous growth of your repository, as you add more and more projects. - ### Problem Figuring out what these two git commits are might not be as simple as it seems. @@ -174,8 +181,9 @@ Conceptually, what we want is to use the absolute latest commit on the `master` The commits therefore can't just be `HEAD` and `HEAD~1`. If a few deployments fail one after another, that means that we're accumulating a list of affected projects that are not getting deployed. Anytime we retry the deployment, we want to include **every commit since the last time we deployed successfully**. That way we ensure we don't accidentally skip deploying a project that has changed. This action enables you to find: -* Commit SHA from which PR originated (in the case of `pull_request`) -* Commit SHA of the last successful CI run + +- Commit SHA from which PR originated (in the case of `pull_request`) +- Commit SHA of the last successful CI run ## License diff --git a/dist/index.js b/dist/index.js index 6a3e9d6..9c49a01 100644 --- a/dist/index.js +++ b/dist/index.js @@ -37853,21 +37853,21 @@ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, ge }); }; Object.defineProperty(exports, "__esModule", ({ value: true })); -const action_1 = __nccwpck_require__(1231); const core = __nccwpck_require__(2186); const github = __nccwpck_require__(5438); +const action_1 = __nccwpck_require__(1231); const child_process_1 = __nccwpck_require__(2081); const fs_1 = __nccwpck_require__(7147); const https_proxy_agent_1 = __nccwpck_require__(7219); const proxy_from_env_1 = __nccwpck_require__(3329); -const { runId, repo: { repo, owner }, eventName } = github.context; +const { runId, repo: { repo, owner }, eventName, } = github.context; process.env.GITHUB_TOKEN = process.argv[2]; const mainBranchName = process.argv[3]; const errorOnNoSuccessfulWorkflow = process.argv[4]; const lastSuccessfulEvent = process.argv[5]; const workingDirectory = process.argv[6]; const workflowId = process.argv[7]; -const defaultWorkingDirectory = '.'; +const defaultWorkingDirectory = "."; const ProxifiedClient = action_1.Octokit.plugin(proxyPlugin); let BASE_SHA; (() => __awaiter(void 0, void 0, void 0, function* () { @@ -37876,17 +37876,20 @@ let BASE_SHA; process.chdir(workingDirectory); } else { - process.stdout.write('\n'); + process.stdout.write("\n"); process.stdout.write(`WARNING: Working directory '${workingDirectory}' doesn't exist.\n`); } } - const headResult = (0, child_process_1.spawnSync)('git', ['rev-parse', 'HEAD'], { encoding: 'utf-8' }); + const headResult = (0, child_process_1.spawnSync)("git", ["rev-parse", "HEAD"], { + encoding: "utf-8", + }); const HEAD_SHA = headResult.stdout; - if ((['pull_request', 'pull_request_target'].includes(eventName) && !github.context.payload.pull_request.merged) || - eventName == 'merge_group') { + if ((["pull_request", "pull_request_target"].includes(eventName) && + !github.context.payload.pull_request.merged) || + eventName == "merge_group") { try { const mergeBaseRef = yield findMergeBaseRef(); - const baseResult = (0, child_process_1.spawnSync)('git', ['merge-base', `origin/${mainBranchName}`, mergeBaseRef], { encoding: 'utf-8' }); + const baseResult = (0, child_process_1.spawnSync)("git", ["merge-base", `origin/${mainBranchName}`, mergeBaseRef], { encoding: "utf-8" }); BASE_SHA = baseResult.stdout; } catch (e) { @@ -37903,32 +37906,35 @@ let BASE_SHA; return; } if (!BASE_SHA) { - if (errorOnNoSuccessfulWorkflow === 'true') { + if (errorOnNoSuccessfulWorkflow === "true") { reportFailure(mainBranchName); return; } else { - process.stdout.write('\n'); - process.stdout.write(`WARNING: Unable to find a successful workflow run on 'origin/${mainBranchName}'\n`); + process.stdout.write("\n"); + process.stdout.write(`WARNING: Unable to find a successful workflow run on 'origin/${mainBranchName}', or the latest successful workflow was connected to a commit which no longer exists on that branch (e.g. if that branch was rebased)\n`); process.stdout.write(`We are therefore defaulting to use HEAD~1 on 'origin/${mainBranchName}'\n`); - process.stdout.write('\n'); + process.stdout.write("\n"); process.stdout.write(`NOTE: You can instead make this a hard error by setting 'error-on-no-successful-workflow' on the action in your workflow.\n`); - const commitCountOutput = (0, child_process_1.spawnSync)('git', ['rev-list', '--count', `origin/${mainBranchName}`], { encoding: 'utf-8' }).stdout; + process.stdout.write("\n"); + const commitCountOutput = (0, child_process_1.spawnSync)("git", ["rev-list", "--count", `origin/${mainBranchName}`], { encoding: "utf-8" }).stdout; const commitCount = parseInt(stripNewLineEndings(commitCountOutput), 10); - const LAST_COMMIT_CMD = `origin/${mainBranchName}${commitCount > 1 ? '~1' : ''}`; - const baseRes = (0, child_process_1.spawnSync)('git', ['rev-parse', LAST_COMMIT_CMD], { encoding: 'utf-8' }); + const LAST_COMMIT_CMD = `origin/${mainBranchName}${commitCount > 1 ? "~1" : ""}`; + const baseRes = (0, child_process_1.spawnSync)("git", ["rev-parse", LAST_COMMIT_CMD], { + encoding: "utf-8", + }); BASE_SHA = baseRes.stdout; - core.setOutput('noPreviousBuild', 'true'); + core.setOutput("noPreviousBuild", "true"); } } else { - process.stdout.write('\n'); + process.stdout.write("\n"); process.stdout.write(`Found the last successful workflow run on 'origin/${mainBranchName}'\n`); process.stdout.write(`Commit: ${BASE_SHA}\n`); } } - core.setOutput('base', stripNewLineEndings(BASE_SHA)); - core.setOutput('head', stripNewLineEndings(HEAD_SHA)); + core.setOutput("base", stripNewLineEndings(BASE_SHA)); + core.setOutput("head", stripNewLineEndings(HEAD_SHA)); }))(); function reportFailure(branchName) { core.setFailed(` @@ -37940,7 +37946,7 @@ function reportFailure(branchName) { - If no, then you might have changed your git history and those commits no longer exist.`); } function proxyPlugin(octokit) { - octokit.hook.before('request', options => { + octokit.hook.before("request", (options) => { const proxy = (0, proxy_from_env_1.getProxyForUrl)(options.baseUrl); if (proxy) { options.request.agent = new https_proxy_agent_1.HttpsProxyAgent(proxy); @@ -37949,47 +37955,45 @@ function proxyPlugin(octokit) { } /** * Find last successful workflow run on the repo - * @param {string?} workflow_id - * @param {number} run_id - * @param {string} owner - * @param {string} repo - * @param {string} branch - * @returns */ function findSuccessfulCommit(workflow_id, run_id, owner, repo, branch, lastSuccessfulEvent) { return __awaiter(this, void 0, void 0, function* () { const octokit = new ProxifiedClient(); if (!workflow_id) { - workflow_id = yield octokit.request(`GET /repos/${owner}/${repo}/actions/runs/${run_id}`, { + workflow_id = yield octokit + .request(`GET /repos/${owner}/${repo}/actions/runs/${run_id}`, { owner, repo, branch, - run_id - }).then(({ data: { workflow_id } }) => workflow_id); - process.stdout.write('\n'); + run_id, + }) + .then(({ data: { workflow_id } }) => workflow_id); + process.stdout.write("\n"); process.stdout.write(`Workflow Id not provided. Using workflow '${workflow_id}'\n`); } // fetch all workflow runs on a given repo/branch/workflow with push and success - const shas = yield octokit.request(`GET /repos/${owner}/${repo}/actions/workflows/${workflow_id}/runs`, { + const shas = yield octokit + .request(`GET /repos/${owner}/${repo}/actions/workflows/${workflow_id}/runs`, { owner, repo, // on non-push workflow runs we do not have branch property - branch: lastSuccessfulEvent !== 'push' ? undefined : branch, + branch: lastSuccessfulEvent !== "push" ? undefined : branch, workflow_id, event: lastSuccessfulEvent, - status: 'success' - }).then(({ data: { workflow_runs } }) => workflow_runs.map(run => run.head_sha)); - return yield findExistingCommit(shas); + status: "success", + }) + .then(({ data: { workflow_runs } }) => workflow_runs.map((run) => run.head_sha)); + return yield findExistingCommit(octokit, branch, shas); }); } function findMergeBaseRef() { return __awaiter(this, void 0, void 0, function* () { - if (eventName == 'merge_group') { + if (eventName == "merge_group") { const mergeQueueBranch = yield findMergeQueueBranch(); return `origin/${mergeQueueBranch}`; } else { - return 'HEAD'; + return "HEAD"; } }); } @@ -38002,9 +38006,9 @@ function findMergeQueueBranch() { return __awaiter(this, void 0, void 0, function* () { const pull_number = findMergeQueuePr(); if (!pull_number) { - throw new Error('Failed to determine PR number'); + throw new Error("Failed to determine PR number"); } - process.stdout.write('\n'); + process.stdout.write("\n"); process.stdout.write(`Found PR #${pull_number} from merge queue branch\n`); const octokit = new ProxifiedClient(); const result = yield octokit.request(`GET /repos/${owner}/${repo}/pulls/${pull_number}`, { owner, repo, pull_number: +pull_number }); @@ -38013,13 +38017,11 @@ function findMergeQueueBranch() { } /** * Get first existing commit - * @param {string[]} commit_shas - * @returns {string?} */ -function findExistingCommit(shas) { +function findExistingCommit(octokit, branchName, shas) { return __awaiter(this, void 0, void 0, function* () { for (const commitSha of shas) { - if (yield commitExists(commitSha)) { + if (yield commitExists(octokit, branchName, commitSha)) { return commitSha; } } @@ -38028,14 +38030,26 @@ function findExistingCommit(shas) { } /** * Check if given commit is valid - * @param {string} commitSha - * @returns {boolean} */ -function commitExists(commitSha) { +function commitExists(octokit, branchName, commitSha) { return __awaiter(this, void 0, void 0, function* () { try { - (0, child_process_1.spawnSync)('git', ['cat-file', '-e', commitSha], { stdio: ['pipe', 'pipe', null] }); - return true; + (0, child_process_1.spawnSync)("git", ["cat-file", "-e", commitSha], { + stdio: ["pipe", "pipe", null], + }); + // Check the commit exists in general + yield octokit.request("GET /repos/{owner}/{repo}/commits/{commit_sha}", { + owner, + repo, + commit_sha: commitSha, + }); + // Check the commit exists on the expected main branch (it will not in the case of a rebased main branch) + const commits = yield octokit.request("GET /repos/{owner}/{repo}/commits", { + owner, + repo, + sha: branchName, + }); + return commits.data.some((commit) => commit.sha === commitSha); } catch (_a) { return false; @@ -38044,10 +38058,9 @@ function commitExists(commitSha) { } /** * Strips LF line endings from given string - * @param {string} string */ function stripNewLineEndings(string) { - return string.replace('\n', ''); + return string.replace("\n", ""); } diff --git a/find-successful-workflow.ts b/find-successful-workflow.ts index 6982b53..bb52bb6 100644 --- a/find-successful-workflow.ts +++ b/find-successful-workflow.ts @@ -1,48 +1,56 @@ -import { Octokit } from '@octokit/action'; -import * as core from '@actions/core'; -import * as github from '@actions/github'; -import { spawnSync } from 'child_process'; -import { existsSync } from 'fs'; -import { HttpsProxyAgent } from 'https-proxy-agent' -import { getProxyForUrl } from 'proxy-from-env'; - -const { runId, repo: { repo, owner }, eventName } = github.context; +import * as core from "@actions/core"; +import * as github from "@actions/github"; +import { Octokit } from "@octokit/action"; +import { spawnSync } from "child_process"; +import { existsSync } from "fs"; +import { HttpsProxyAgent } from "https-proxy-agent"; +import { getProxyForUrl } from "proxy-from-env"; + +const { + runId, + repo: { repo, owner }, + eventName, +} = github.context; process.env.GITHUB_TOKEN = process.argv[2]; const mainBranchName = process.argv[3]; const errorOnNoSuccessfulWorkflow = process.argv[4]; const lastSuccessfulEvent = process.argv[5]; const workingDirectory = process.argv[6]; const workflowId = process.argv[7]; -const defaultWorkingDirectory = '.'; - - +const defaultWorkingDirectory = "."; -const ProxifiedClient = Octokit.plugin( - proxyPlugin -); +const ProxifiedClient = Octokit.plugin(proxyPlugin); -let BASE_SHA; +let BASE_SHA: string; (async () => { if (workingDirectory !== defaultWorkingDirectory) { if (existsSync(workingDirectory)) { process.chdir(workingDirectory); } else { - process.stdout.write('\n'); - process.stdout.write(`WARNING: Working directory '${workingDirectory}' doesn't exist.\n`); + process.stdout.write("\n"); + process.stdout.write( + `WARNING: Working directory '${workingDirectory}' doesn't exist.\n` + ); } } - const headResult = spawnSync('git', ['rev-parse', 'HEAD'], { encoding: 'utf-8' }); + const headResult = spawnSync("git", ["rev-parse", "HEAD"], { + encoding: "utf-8", + }); const HEAD_SHA = headResult.stdout; - if ( - (['pull_request', 'pull_request_target'].includes(eventName) && !github.context.payload.pull_request.merged) || - eventName == 'merge_group' + (["pull_request", "pull_request_target"].includes(eventName) && + !github.context.payload.pull_request.merged) || + eventName == "merge_group" ) { try { const mergeBaseRef = await findMergeBaseRef(); - const baseResult = spawnSync('git', ['merge-base', `origin/${mainBranchName}`, mergeBaseRef], { encoding: 'utf-8' }); + const baseResult = spawnSync( + "git", + ["merge-base", `origin/${mainBranchName}`, mergeBaseRef], + { encoding: "utf-8" } + ); BASE_SHA = baseResult.stdout; } catch (e) { core.setFailed(e.message); @@ -50,43 +58,69 @@ let BASE_SHA; } } else { try { - BASE_SHA = await findSuccessfulCommit(workflowId, runId, owner, repo, mainBranchName, lastSuccessfulEvent); + BASE_SHA = await findSuccessfulCommit( + workflowId, + runId, + owner, + repo, + mainBranchName, + lastSuccessfulEvent + ); } catch (e) { core.setFailed(e.message); return; } if (!BASE_SHA) { - if (errorOnNoSuccessfulWorkflow === 'true') { + if (errorOnNoSuccessfulWorkflow === "true") { reportFailure(mainBranchName); return; } else { - process.stdout.write('\n'); - process.stdout.write(`WARNING: Unable to find a successful workflow run on 'origin/${mainBranchName}'\n`); - process.stdout.write(`We are therefore defaulting to use HEAD~1 on 'origin/${mainBranchName}'\n`); - process.stdout.write('\n'); - process.stdout.write(`NOTE: You can instead make this a hard error by setting 'error-on-no-successful-workflow' on the action in your workflow.\n`); + process.stdout.write( "\n"); + process.stdout.write( + `WARNING: Unable to find a successful workflow run on 'origin/${mainBranchName}', or the latest successful workflow was connected to a commit which no longer exists on that branch (e.g. if that branch was rebased)\n` + ); + process.stdout.write( + `We are therefore defaulting to use HEAD~1 on 'origin/${mainBranchName}'\n` + ); + process.stdout.write("\n"); + process.stdout.write( + `NOTE: You can instead make this a hard error by setting 'error-on-no-successful-workflow' on the action in your workflow.\n` + ); + process.stdout.write("\n"); - const commitCountOutput = spawnSync('git', ['rev-list', '--count', `origin/${mainBranchName}`], { encoding: 'utf-8' }).stdout; - const commitCount = parseInt(stripNewLineEndings(commitCountOutput), 10); + const commitCountOutput = spawnSync( + "git", + ["rev-list", "--count", `origin/${mainBranchName}`], + { encoding: "utf-8" } + ).stdout; + const commitCount = parseInt( + stripNewLineEndings(commitCountOutput), + 10 + ); - const LAST_COMMIT_CMD = `origin/${mainBranchName}${commitCount > 1 ? '~1' : ''}` - const baseRes = spawnSync('git', ['rev-parse', LAST_COMMIT_CMD], { encoding: 'utf-8' }); + const LAST_COMMIT_CMD = `origin/${mainBranchName}${ + commitCount > 1 ? "~1" : "" + }`; + const baseRes = spawnSync("git", ["rev-parse", LAST_COMMIT_CMD], { + encoding: "utf-8", + }); BASE_SHA = baseRes.stdout; - core.setOutput('noPreviousBuild', 'true'); + core.setOutput("noPreviousBuild", "true"); } } else { - process.stdout.write('\n'); - process.stdout.write(`Found the last successful workflow run on 'origin/${mainBranchName}'\n`); + process.stdout.write("\n"); + process.stdout.write( + `Found the last successful workflow run on 'origin/${mainBranchName}'\n` + ); process.stdout.write(`Commit: ${BASE_SHA}\n`); } - } - core.setOutput('base', stripNewLineEndings(BASE_SHA)); - core.setOutput('head', stripNewLineEndings(HEAD_SHA)); + core.setOutput("base", stripNewLineEndings(BASE_SHA)); + core.setOutput("head", stripNewLineEndings(HEAD_SHA)); })(); -function reportFailure(branchName) { +function reportFailure(branchName: string): void { core.setFailed(` Unable to find a successful workflow run on 'origin/${branchName}' NOTE: You have set 'error-on-no-successful-workflow' on the action so this is a hard error. @@ -96,85 +130,104 @@ function reportFailure(branchName) { - If no, then you might have changed your git history and those commits no longer exist.`); } -function proxyPlugin(octokit: Octokit) { - octokit.hook.before('request', options => { - const proxy: URL = getProxyForUrl(options.baseUrl) +function proxyPlugin(octokit: Octokit): void { + octokit.hook.before("request", (options) => { + const proxy: URL = getProxyForUrl(options.baseUrl); if (proxy) { - options.request.agent = new HttpsProxyAgent(proxy) + options.request.agent = new HttpsProxyAgent(proxy); } - }) + }); } /** * Find last successful workflow run on the repo - * @param {string?} workflow_id - * @param {number} run_id - * @param {string} owner - * @param {string} repo - * @param {string} branch - * @returns */ -async function findSuccessfulCommit(workflow_id, run_id, owner, repo, branch, lastSuccessfulEvent) { +async function findSuccessfulCommit( + workflow_id: string | undefined, + run_id: number, + owner: string, + repo: string, + branch: string, + lastSuccessfulEvent: string +): Promise { const octokit = new ProxifiedClient(); if (!workflow_id) { - workflow_id = await octokit.request(`GET /repos/${owner}/${repo}/actions/runs/${run_id}`, { - owner, - repo, - branch, - run_id - }).then(({ data: { workflow_id } }) => workflow_id); - process.stdout.write('\n'); - process.stdout.write(`Workflow Id not provided. Using workflow '${workflow_id}'\n`); + workflow_id = await octokit + .request(`GET /repos/${owner}/${repo}/actions/runs/${run_id}`, { + owner, + repo, + branch, + run_id, + }) + .then(({ data: { workflow_id } }) => workflow_id); + process.stdout.write("\n"); + process.stdout.write( + `Workflow Id not provided. Using workflow '${workflow_id}'\n` + ); } // fetch all workflow runs on a given repo/branch/workflow with push and success - const shas = await octokit.request(`GET /repos/${owner}/${repo}/actions/workflows/${workflow_id}/runs`, { - owner, - repo, - // on non-push workflow runs we do not have branch property - branch: lastSuccessfulEvent !== 'push' ? undefined : branch, - workflow_id, - event: lastSuccessfulEvent, - status: 'success' - }).then(({ data: { workflow_runs } }) => workflow_runs.map(run => run.head_sha)); - - return await findExistingCommit(shas); + const shas = await octokit + .request( + `GET /repos/${owner}/${repo}/actions/workflows/${workflow_id}/runs`, + { + owner, + repo, + // on non-push workflow runs we do not have branch property + branch: lastSuccessfulEvent !== "push" ? undefined : branch, + workflow_id, + event: lastSuccessfulEvent, + status: "success", + } + ) + .then(({ data: { workflow_runs } }) => + workflow_runs.map((run: { head_sha: any }) => run.head_sha) + ); + + return await findExistingCommit(octokit, branch, shas); } -async function findMergeBaseRef() { - if (eventName == 'merge_group') { +async function findMergeBaseRef(): Promise { + if (eventName == "merge_group") { const mergeQueueBranch = await findMergeQueueBranch(); return `origin/${mergeQueueBranch}`; } else { - return 'HEAD' + return "HEAD"; } } -function findMergeQueuePr() { +function findMergeQueuePr(): string { const { head_ref, base_sha } = github.context.payload.merge_group; - const result = new RegExp(`^refs/heads/gh-readonly-queue/${mainBranchName}/pr-(\\d+)-${base_sha}$`).exec(head_ref); + const result = new RegExp( + `^refs/heads/gh-readonly-queue/${mainBranchName}/pr-(\\d+)-${base_sha}$` + ).exec(head_ref); return result ? result.at(1) : undefined; } -async function findMergeQueueBranch() { +async function findMergeQueueBranch(): Promise { const pull_number = findMergeQueuePr(); if (!pull_number) { - throw new Error('Failed to determine PR number') + throw new Error("Failed to determine PR number"); } - process.stdout.write('\n'); + process.stdout.write("\n"); process.stdout.write(`Found PR #${pull_number} from merge queue branch\n`); const octokit = new ProxifiedClient(); - const result = await octokit.request(`GET /repos/${owner}/${repo}/pulls/${pull_number}`, { owner, repo, pull_number: +pull_number }); + const result = await octokit.request( + `GET /repos/${owner}/${repo}/pulls/${pull_number}`, + { owner, repo, pull_number: +pull_number } + ); return result.data.head.ref; } /** * Get first existing commit - * @param {string[]} commit_shas - * @returns {string?} */ -async function findExistingCommit(shas) { +async function findExistingCommit( + octokit: Octokit, + branchName: string, + shas: string[] +): Promise { for (const commitSha of shas) { - if (await commitExists(commitSha)) { + if (await commitExists(octokit, branchName, commitSha)) { return commitSha; } } @@ -183,13 +236,34 @@ async function findExistingCommit(shas) { /** * Check if given commit is valid - * @param {string} commitSha - * @returns {boolean} */ -async function commitExists(commitSha) { +async function commitExists( + octokit: Octokit, + branchName: string, + commitSha: string +): Promise { try { - spawnSync('git', ['cat-file', '-e', commitSha], { stdio: ['pipe', 'pipe', null] }); - return true; + spawnSync("git", ["cat-file", "-e", commitSha], { + stdio: ["pipe", "pipe", null], + }); + + // Check the commit exists in general + await octokit.request("GET /repos/{owner}/{repo}/commits/{commit_sha}", { + owner, + repo, + commit_sha: commitSha, + }); + + // Check the commit exists on the expected main branch (it will not in the case of a rebased main branch) + const commits = await octokit.request("GET /repos/{owner}/{repo}/commits", { + owner, + repo, + sha: branchName, + }); + + return commits.data.some( + (commit: { sha: string }) => commit.sha === commitSha + ); } catch { return false; } @@ -197,9 +271,7 @@ async function commitExists(commitSha) { /** * Strips LF line endings from given string - * @param {string} string */ -function stripNewLineEndings(string) { - return string.replace('\n', ''); +function stripNewLineEndings(string: string): string { + return string.replace("\n", ""); } - diff --git a/package.json b/package.json index 4488d06..3722b2b 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "private": true, - "version": "4.0.1", + "version": "4.0.2", "license": "MIT", "description": "This package.json is here purely to control the version of the Action, in combination with https://github.com/JamesHenry/publish-shell-action", "scripts": { @@ -13,6 +13,11 @@ "engines": { "node": ">=18" }, + "volta": { + "node": "20.9.0", + "yarn": "1.22.21" + }, + "packageManager": "yarn@1.22.21", "homepage": "https://github.com/nrwl/nx-set-shas#readme", "dependencies": { "@actions/core": "^1.10.0", @@ -22,11 +27,18 @@ "proxy-from-env": "1.1.0" }, "devDependencies": { - "@types/node": "^20.5.9", + "@types/node": "^20.9.2", "@vercel/ncc": "^0.36.1", "chalk": "^4.1.2", "husky": "^8.0.1", "is-ci": "^3.0.1", + "lint-staged": "15.1.0", + "prettier": "^3.1.0", "typescript": "^5.2.0" + }, + "lint-staged": { + "*.{ts,json,yml,md}": [ + "npx prettier --write" + ] } } diff --git a/yarn.lock b/yarn.lock index a22a3c9..b2a942e 100644 --- a/yarn.lock +++ b/yarn.lock @@ -220,10 +220,12 @@ dependencies: "@octokit/openapi-types" "^12.11.0" -"@types/node@^20.5.9": - version "20.5.9" - resolved "https://registry.yarnpkg.com/@types/node/-/node-20.5.9.tgz#a70ec9d8fa0180a314c3ede0e20ea56ff71aed9a" - integrity sha512-PcGNd//40kHAS3sTlzKB9C9XL4K0sTup8nbG5lC14kzEteTNuAFh9u5nA0o5TWnSG2r/JNPRXFVcHJIIeRlmqQ== +"@types/node@^20.9.2": + version "20.9.2" + resolved "https://registry.yarnpkg.com/@types/node/-/node-20.9.2.tgz#002815c8e87fe0c9369121c78b52e800fadc0ac6" + integrity sha512-WHZXKFCEyIUJzAwh3NyyTHYSR35SevJ6mZ1nWwJafKtiQbqRTIKSRcw3Ma3acqgsent3RRDqeVwpHntMk+9irg== + dependencies: + undici-types "~5.26.4" "@vercel/ncc@^0.36.1": version "0.36.1" @@ -237,6 +239,18 @@ agent-base@^7.0.2: dependencies: debug "^4.3.4" +ansi-escapes@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/ansi-escapes/-/ansi-escapes-5.0.0.tgz#b6a0caf0eef0c41af190e9a749e0c00ec04bb2a6" + integrity sha512-5GFMVX8HqE/TB+FuBJGuO5XG0WrsA6ptUqoODaT/n9mmUaZFkqnBueB4leqGBCmrUHnCnC4PCZTCd0E7QQ83bA== + dependencies: + type-fest "^1.0.2" + +ansi-regex@^6.0.1: + version "6.0.1" + resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-6.0.1.tgz#3183e38fae9a65d7cb5e53945cd5897d0260a06a" + integrity sha512-n5M855fKb2SsfMIiFFoVrABHJC8QtHwVx+mHWP3QcEqBHYienj5dHSgjbxtC0WEZXYt4wcD6zrQElDPhFuZgfA== + ansi-styles@^4.1.0: version "4.3.0" resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-4.3.0.tgz#edd803628ae71c04c85ae7a0906edad34b648937" @@ -244,11 +258,28 @@ ansi-styles@^4.1.0: dependencies: color-convert "^2.0.1" +ansi-styles@^6.0.0, ansi-styles@^6.1.0: + version "6.2.1" + resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-6.2.1.tgz#0e62320cf99c21afff3b3012192546aacbfb05c5" + integrity sha512-bN798gFfQX+viw3R7yrGWRqnrN2oRkEkUjjl4JNn4E8GxxbjtG3FbrEIIY3l8/hrwUwIeCZvi4QuOTP4MErVug== + before-after-hook@^2.2.0: version "2.2.3" resolved "https://registry.yarnpkg.com/before-after-hook/-/before-after-hook-2.2.3.tgz#c51e809c81a4e354084422b9b26bad88249c517c" integrity sha512-NzUnlZexiaH/46WDhANlyR2bXRopNg4F/zuSA3OpZnllCUgRaOF2znDioDWrmbNVsuZk6l9pMquQB38cfBZwkQ== +braces@^3.0.2: + version "3.0.2" + resolved "https://registry.yarnpkg.com/braces/-/braces-3.0.2.tgz#3454e1a462ee8d599e236df336cd9ea4f8afe107" + integrity sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A== + dependencies: + fill-range "^7.0.1" + +chalk@5.3.0: + version "5.3.0" + resolved "https://registry.yarnpkg.com/chalk/-/chalk-5.3.0.tgz#67c20a7ebef70e7f3970a01f90fa210cb6860385" + integrity sha512-dLitG79d+GV1Nb/VYcCDFivJeK1hiukt9QjRNVOsUtTy1rR1YJsmpGGTZ3qJos+uw7WmWF4wUwBd9jxjocFC2w== + chalk@^4.1.2: version "4.1.2" resolved "https://registry.yarnpkg.com/chalk/-/chalk-4.1.2.tgz#aac4e2b7734a740867aeb16bf02aad556a1e7a01" @@ -262,6 +293,21 @@ ci-info@^3.2.0: resolved "https://registry.yarnpkg.com/ci-info/-/ci-info-3.8.0.tgz#81408265a5380c929f0bc665d62256628ce9ef91" integrity sha512-eXTggHWSooYhq49F2opQhuHWgzucfF2YgODK4e1566GQs5BIfP30B0oenwBJHfWxAs2fyPB1s7Mg949zLf61Yw== +cli-cursor@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/cli-cursor/-/cli-cursor-4.0.0.tgz#3cecfe3734bf4fe02a8361cbdc0f6fe28c6a57ea" + integrity sha512-VGtlMu3x/4DOtIUwEkRezxUZ2lBacNJCHash0N0WeZDBS+7Ux1dm3XWAgWYxLJFMMdOeXMHXorshEFhbMSGelg== + dependencies: + restore-cursor "^4.0.0" + +cli-truncate@^3.1.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/cli-truncate/-/cli-truncate-3.1.0.tgz#3f23ab12535e3d73e839bb43e73c9de487db1389" + integrity sha512-wfOBkjXteqSnI59oPcJkcPl/ZmwvMMOj340qUIY1SKZCv0B9Cf4D4fAucRkIKQmsIuYK3x1rrgU7MeGRruiuiA== + dependencies: + slice-ansi "^5.0.0" + string-width "^5.0.0" + color-convert@^2.0.1: version "2.0.1" resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-2.0.1.tgz#72d3a68d598c9bdb3af2ad1e84f21d896abd4de3" @@ -274,7 +320,26 @@ color-name@~1.1.4: resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.4.tgz#c2a09a87acbde69543de6f63fa3995c826c536a2" integrity sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA== -debug@4, debug@^4.3.4: +colorette@^2.0.20: + version "2.0.20" + resolved "https://registry.yarnpkg.com/colorette/-/colorette-2.0.20.tgz#9eb793e6833067f7235902fcd3b09917a000a95a" + integrity sha512-IfEDxwoWIjkeXL1eXcDiow4UbKjhLdq6/EuSVR9GMN7KVH3r9gQ83e73hsz1Nd1T3ijd5xv1wcWRYO+D6kCI2w== + +commander@11.1.0: + version "11.1.0" + resolved "https://registry.yarnpkg.com/commander/-/commander-11.1.0.tgz#62fdce76006a68e5c1ab3314dc92e800eb83d906" + integrity sha512-yPVavfyCcRhmorC7rWlkHn15b4wDVgVmBA7kV4QVBsF7kv/9TKJAbAXVTxvTnwP8HHKjRCJDClKbciiYS7p0DQ== + +cross-spawn@^7.0.3: + version "7.0.3" + resolved "https://registry.yarnpkg.com/cross-spawn/-/cross-spawn-7.0.3.tgz#f73a85b9d5d41d045551c177e2882d4ac85728a6" + integrity sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w== + dependencies: + path-key "^3.1.0" + shebang-command "^2.0.0" + which "^2.0.1" + +debug@4, debug@4.3.4, debug@^4.3.4: version "4.3.4" resolved "https://registry.yarnpkg.com/debug/-/debug-4.3.4.tgz#1319f6579357f2338d3337d2cdd4914bb5dcc865" integrity sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ== @@ -286,6 +351,48 @@ deprecation@^2.0.0, deprecation@^2.3.1: resolved "https://registry.yarnpkg.com/deprecation/-/deprecation-2.3.1.tgz#6368cbdb40abf3373b525ac87e4a260c3a700919" integrity sha512-xmHIy4F3scKVwMsQ4WnVaS8bHOx0DmVwRywosKhaILI0ywMDWPtBSku2HNxRvF7jtwDRsoEwYQSfbxj8b7RlJQ== +eastasianwidth@^0.2.0: + version "0.2.0" + resolved "https://registry.yarnpkg.com/eastasianwidth/-/eastasianwidth-0.2.0.tgz#696ce2ec0aa0e6ea93a397ffcf24aa7840c827cb" + integrity sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA== + +emoji-regex@^9.2.2: + version "9.2.2" + resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-9.2.2.tgz#840c8803b0d8047f4ff0cf963176b32d4ef3ed72" + integrity sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg== + +eventemitter3@^5.0.1: + version "5.0.1" + resolved "https://registry.yarnpkg.com/eventemitter3/-/eventemitter3-5.0.1.tgz#53f5ffd0a492ac800721bb42c66b841de96423c4" + integrity sha512-GWkBvjiSZK87ELrYOSESUYeVIc9mvLLf/nXalMOS5dYrgZq9o5OVkbZAVM06CVxYsCwH9BDZFPlQTlPA1j4ahA== + +execa@8.0.1: + version "8.0.1" + resolved "https://registry.yarnpkg.com/execa/-/execa-8.0.1.tgz#51f6a5943b580f963c3ca9c6321796db8cc39b8c" + integrity sha512-VyhnebXciFV2DESc+p6B+y0LjSm0krU4OgJN44qFAhBY0TJ+1V61tYD2+wHusZ6F9n5K+vl8k0sTy7PEfV4qpg== + dependencies: + cross-spawn "^7.0.3" + get-stream "^8.0.1" + human-signals "^5.0.0" + is-stream "^3.0.0" + merge-stream "^2.0.0" + npm-run-path "^5.1.0" + onetime "^6.0.0" + signal-exit "^4.1.0" + strip-final-newline "^3.0.0" + +fill-range@^7.0.1: + version "7.0.1" + resolved "https://registry.yarnpkg.com/fill-range/-/fill-range-7.0.1.tgz#1919a6a7c75fe38b2c7c77e5198535da9acdda40" + integrity sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ== + dependencies: + to-regex-range "^5.0.1" + +get-stream@^8.0.1: + version "8.0.1" + resolved "https://registry.yarnpkg.com/get-stream/-/get-stream-8.0.1.tgz#def9dfd71742cd7754a7761ed43749a27d02eca2" + integrity sha512-VaUJspBffn/LMCJVoMvSAdmscJyS1auj5Zulnn5UoYcY531UWmdwhRWkcGKnGU93m5HSXP9LP2usOryrBtQowA== + has-flag@^4.0.0: version "4.0.0" resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-4.0.0.tgz#944771fd9c81c81265c4d6941860da06bb59479b" @@ -299,6 +406,11 @@ https-proxy-agent@7.0.2: agent-base "^7.0.2" debug "4" +human-signals@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/human-signals/-/human-signals-5.0.0.tgz#42665a284f9ae0dade3ba41ebc37eb4b852f3a28" + integrity sha512-AXcZb6vzzrFAUE61HnN4mpLqd/cSIwNQjtNWR0euPm6y0iqx3G4gOXaIDdtdDwZmhwe82LA6+zinmW4UBWVePQ== + husky@^8.0.1: version "8.0.3" resolved "https://registry.yarnpkg.com/husky/-/husky-8.0.3.tgz#4936d7212e46d1dea28fef29bb3a108872cd9184" @@ -311,11 +423,98 @@ is-ci@^3.0.1: dependencies: ci-info "^3.2.0" +is-fullwidth-code-point@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/is-fullwidth-code-point/-/is-fullwidth-code-point-4.0.0.tgz#fae3167c729e7463f8461ce512b080a49268aa88" + integrity sha512-O4L094N2/dZ7xqVdrXhh9r1KODPJpFms8B5sGdJLPy664AgvXsreZUyCQQNItZRDlYug4xStLjNp/sz3HvBowQ== + +is-number@^7.0.0: + version "7.0.0" + resolved "https://registry.yarnpkg.com/is-number/-/is-number-7.0.0.tgz#7535345b896734d5f80c4d06c50955527a14f12b" + integrity sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng== + is-plain-object@^5.0.0: version "5.0.0" resolved "https://registry.yarnpkg.com/is-plain-object/-/is-plain-object-5.0.0.tgz#4427f50ab3429e9025ea7d52e9043a9ef4159344" integrity sha512-VRSzKkbMm5jMDoKLbltAkFQ5Qr7VDiTFGXxYFXXowVj387GeGNOCsOH6Msy00SGZ3Fp84b1Naa1psqgcCIEP5Q== +is-stream@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/is-stream/-/is-stream-3.0.0.tgz#e6bfd7aa6bef69f4f472ce9bb681e3e57b4319ac" + integrity sha512-LnQR4bZ9IADDRSkvpqMGvt/tEJWclzklNgSw48V5EAaAeDd6qGvN8ei6k5p0tvxSR171VmGyHuTiAOfxAbr8kA== + +isexe@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/isexe/-/isexe-2.0.0.tgz#e8fbf374dc556ff8947a10dcb0572d633f2cfa10" + integrity sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw== + +lilconfig@2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/lilconfig/-/lilconfig-2.1.0.tgz#78e23ac89ebb7e1bfbf25b18043de756548e7f52" + integrity sha512-utWOt/GHzuUxnLKxB6dk81RoOeoNeHgbrXiuGk4yyF5qlRz+iIVWu56E2fqGHFrXz0QNUhLB/8nKqvRH66JKGQ== + +lint-staged@15.1.0: + version "15.1.0" + resolved "https://registry.yarnpkg.com/lint-staged/-/lint-staged-15.1.0.tgz#c0f8e4d96ac3c09beac5c76d08524d6000c207b4" + integrity sha512-ZPKXWHVlL7uwVpy8OZ7YQjYDAuO5X4kMh0XgZvPNxLcCCngd0PO5jKQyy3+s4TL2EnHoIXIzP1422f/l3nZKMw== + dependencies: + chalk "5.3.0" + commander "11.1.0" + debug "4.3.4" + execa "8.0.1" + lilconfig "2.1.0" + listr2 "7.0.2" + micromatch "4.0.5" + pidtree "0.6.0" + string-argv "0.3.2" + yaml "2.3.4" + +listr2@7.0.2: + version "7.0.2" + resolved "https://registry.yarnpkg.com/listr2/-/listr2-7.0.2.tgz#3aa3e1549dfaf3c57ab5eeaba754da3b87f33063" + integrity sha512-rJysbR9GKIalhTbVL2tYbF2hVyDnrf7pFUZBwjPaMIdadYHmeT+EVi/Bu3qd7ETQPahTotg2WRCatXwRBW554g== + dependencies: + cli-truncate "^3.1.0" + colorette "^2.0.20" + eventemitter3 "^5.0.1" + log-update "^5.0.1" + rfdc "^1.3.0" + wrap-ansi "^8.1.0" + +log-update@^5.0.1: + version "5.0.1" + resolved "https://registry.yarnpkg.com/log-update/-/log-update-5.0.1.tgz#9e928bf70cb183c1f0c9e91d9e6b7115d597ce09" + integrity sha512-5UtUDQ/6edw4ofyljDNcOVJQ4c7OjDro4h3y8e1GQL5iYElYclVHJ3zeWchylvMaKnDbDilC8irOVyexnA/Slw== + dependencies: + ansi-escapes "^5.0.0" + cli-cursor "^4.0.0" + slice-ansi "^5.0.0" + strip-ansi "^7.0.1" + wrap-ansi "^8.0.1" + +merge-stream@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/merge-stream/-/merge-stream-2.0.0.tgz#52823629a14dd00c9770fb6ad47dc6310f2c1f60" + integrity sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w== + +micromatch@4.0.5: + version "4.0.5" + resolved "https://registry.yarnpkg.com/micromatch/-/micromatch-4.0.5.tgz#bc8999a7cbbf77cdc89f132f6e467051b49090c6" + integrity sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA== + dependencies: + braces "^3.0.2" + picomatch "^2.3.1" + +mimic-fn@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/mimic-fn/-/mimic-fn-2.1.0.tgz#7ed2c2ccccaf84d3ffcb7a69b57711fc2083401b" + integrity sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg== + +mimic-fn@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/mimic-fn/-/mimic-fn-4.0.0.tgz#60a90550d5cb0b239cca65d893b1a53b29871ecc" + integrity sha512-vqiC06CuhBTUdZH+RYl8sFrL096vA45Ok5ISO6sE/Mr1jRbGH4Csnhi8f3wKVl7x8mO4Au7Ir9D3Oyv1VYMFJw== + ms@2.1.2: version "2.1.2" resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.2.tgz#d09d1f357b443f493382a8eb3ccd183872ae6009" @@ -328,6 +527,13 @@ node-fetch@^2.6.7: dependencies: whatwg-url "^5.0.0" +npm-run-path@^5.1.0: + version "5.1.0" + resolved "https://registry.yarnpkg.com/npm-run-path/-/npm-run-path-5.1.0.tgz#bc62f7f3f6952d9894bd08944ba011a6ee7b7e00" + integrity sha512-sJOdmRGrY2sjNTRMbSvluQqg+8X7ZK61yvzBEIDhz4f8z1TZFYABsqjjCBd/0PUNE9M6QDgHJXQkGUEm7Q+l9Q== + dependencies: + path-key "^4.0.0" + once@^1.4.0: version "1.4.0" resolved "https://registry.yarnpkg.com/once/-/once-1.4.0.tgz#583b1aa775961d4b113ac17d9c50baef9dd76bd1" @@ -335,11 +541,119 @@ once@^1.4.0: dependencies: wrappy "1" +onetime@^5.1.0: + version "5.1.2" + resolved "https://registry.yarnpkg.com/onetime/-/onetime-5.1.2.tgz#d0e96ebb56b07476df1dd9c4806e5237985ca45e" + integrity sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg== + dependencies: + mimic-fn "^2.1.0" + +onetime@^6.0.0: + version "6.0.0" + resolved "https://registry.yarnpkg.com/onetime/-/onetime-6.0.0.tgz#7c24c18ed1fd2e9bca4bd26806a33613c77d34b4" + integrity sha512-1FlR+gjXK7X+AsAHso35MnyN5KqGwJRi/31ft6x0M194ht7S+rWAvd7PHss9xSKMzE0asv1pyIHaJYq+BbacAQ== + dependencies: + mimic-fn "^4.0.0" + +path-key@^3.1.0: + version "3.1.1" + resolved "https://registry.yarnpkg.com/path-key/-/path-key-3.1.1.tgz#581f6ade658cbba65a0d3380de7753295054f375" + integrity sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q== + +path-key@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/path-key/-/path-key-4.0.0.tgz#295588dc3aee64154f877adb9d780b81c554bf18" + integrity sha512-haREypq7xkM7ErfgIyA0z+Bj4AGKlMSdlQE2jvJo6huWD1EdkKYV+G/T4nq0YEF2vgTT8kqMFKo1uHn950r4SQ== + +picomatch@^2.3.1: + version "2.3.1" + resolved "https://registry.yarnpkg.com/picomatch/-/picomatch-2.3.1.tgz#3ba3833733646d9d3e4995946c1365a67fb07a42" + integrity sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA== + +pidtree@0.6.0: + version "0.6.0" + resolved "https://registry.yarnpkg.com/pidtree/-/pidtree-0.6.0.tgz#90ad7b6d42d5841e69e0a2419ef38f8883aa057c" + integrity sha512-eG2dWTVw5bzqGRztnHExczNxt5VGsE6OwTeCG3fdUf9KBsZzO3R5OIIIzWR+iZA0NtZ+RDVdaoE2dK1cn6jH4g== + +prettier@^3.1.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/prettier/-/prettier-3.1.0.tgz#c6d16474a5f764ea1a4a373c593b779697744d5e" + integrity sha512-TQLvXjq5IAibjh8EpBIkNKxO749UEWABoiIZehEPiY4GNpVdhaFKqSTu+QrlU6D2dPAfubRmtJTi4K4YkQ5eXw== + proxy-from-env@1.1.0: version "1.1.0" resolved "https://registry.yarnpkg.com/proxy-from-env/-/proxy-from-env-1.1.0.tgz#e102f16ca355424865755d2c9e8ea4f24d58c3e2" integrity sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg== +restore-cursor@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/restore-cursor/-/restore-cursor-4.0.0.tgz#519560a4318975096def6e609d44100edaa4ccb9" + integrity sha512-I9fPXU9geO9bHOt9pHHOhOkYerIMsmVaWB0rA2AI9ERh/+x/i7MV5HKBNrg+ljO5eoPVgCcnFuRjJ9uH6I/3eg== + dependencies: + onetime "^5.1.0" + signal-exit "^3.0.2" + +rfdc@^1.3.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/rfdc/-/rfdc-1.3.0.tgz#d0b7c441ab2720d05dc4cf26e01c89631d9da08b" + integrity sha512-V2hovdzFbOi77/WajaSMXk2OLm+xNIeQdMMuB7icj7bk6zi2F8GGAxigcnDFpJHbNyNcgyJDiP+8nOrY5cZGrA== + +shebang-command@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/shebang-command/-/shebang-command-2.0.0.tgz#ccd0af4f8835fbdc265b82461aaf0c36663f34ea" + integrity sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA== + dependencies: + shebang-regex "^3.0.0" + +shebang-regex@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/shebang-regex/-/shebang-regex-3.0.0.tgz#ae16f1644d873ecad843b0307b143362d4c42172" + integrity sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A== + +signal-exit@^3.0.2: + version "3.0.7" + resolved "https://registry.yarnpkg.com/signal-exit/-/signal-exit-3.0.7.tgz#a9a1767f8af84155114eaabd73f99273c8f59ad9" + integrity sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ== + +signal-exit@^4.1.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/signal-exit/-/signal-exit-4.1.0.tgz#952188c1cbd546070e2dd20d0f41c0ae0530cb04" + integrity sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw== + +slice-ansi@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/slice-ansi/-/slice-ansi-5.0.0.tgz#b73063c57aa96f9cd881654b15294d95d285c42a" + integrity sha512-FC+lgizVPfie0kkhqUScwRu1O/lF6NOgJmlCgK+/LYxDCTk8sGelYaHDhFcDN+Sn3Cv+3VSa4Byeo+IMCzpMgQ== + dependencies: + ansi-styles "^6.0.0" + is-fullwidth-code-point "^4.0.0" + +string-argv@0.3.2: + version "0.3.2" + resolved "https://registry.yarnpkg.com/string-argv/-/string-argv-0.3.2.tgz#2b6d0ef24b656274d957d54e0a4bbf6153dc02b6" + integrity sha512-aqD2Q0144Z+/RqG52NeHEkZauTAUWJO8c6yTftGJKO3Tja5tUgIfmIl6kExvhtxSDP7fXB6DvzkfMpCd/F3G+Q== + +string-width@^5.0.0, string-width@^5.0.1: + version "5.1.2" + resolved "https://registry.yarnpkg.com/string-width/-/string-width-5.1.2.tgz#14f8daec6d81e7221d2a357e668cab73bdbca794" + integrity sha512-HnLOCR3vjcY8beoNLtcjZ5/nxn2afmME6lhrDrebokqMap+XbeW8n9TXpPDOqdGK5qcI3oT0GKTW6wC7EMiVqA== + dependencies: + eastasianwidth "^0.2.0" + emoji-regex "^9.2.2" + strip-ansi "^7.0.1" + +strip-ansi@^7.0.1: + version "7.1.0" + resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-7.1.0.tgz#d5b6568ca689d8561370b0707685d22434faff45" + integrity sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ== + dependencies: + ansi-regex "^6.0.1" + +strip-final-newline@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/strip-final-newline/-/strip-final-newline-3.0.0.tgz#52894c313fbff318835280aed60ff71ebf12b8fd" + integrity sha512-dOESqjYr96iWYylGObzd39EuNTa5VJxyvVAEm5Jnh7KGo75V43Hk1odPQkNDyXNmUR6k+gEiDVXnjB8HJ3crXw== + supports-color@^7.1.0: version "7.2.0" resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-7.2.0.tgz#1b7dcdcb32b8138801b3e478ba6a51caa89648da" @@ -347,6 +661,13 @@ supports-color@^7.1.0: dependencies: has-flag "^4.0.0" +to-regex-range@^5.0.1: + version "5.0.1" + resolved "https://registry.yarnpkg.com/to-regex-range/-/to-regex-range-5.0.1.tgz#1648c44aae7c8d988a326018ed72f5b4dd0392e4" + integrity sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ== + dependencies: + is-number "^7.0.0" + tr46@~0.0.3: version "0.0.3" resolved "https://registry.yarnpkg.com/tr46/-/tr46-0.0.3.tgz#8184fd347dac9cdc185992f3a6622e14b9d9ab6a" @@ -357,11 +678,21 @@ tunnel@^0.0.6: resolved "https://registry.yarnpkg.com/tunnel/-/tunnel-0.0.6.tgz#72f1314b34a5b192db012324df2cc587ca47f92c" integrity sha512-1h/Lnq9yajKY2PEbBadPXj3VxsDDu844OnaAo52UVmIzIvwwtBPIuNvkjuzBlTWpfJyUbG3ez0KSBibQkj4ojg== +type-fest@^1.0.2: + version "1.4.0" + resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-1.4.0.tgz#e9fb813fe3bf1744ec359d55d1affefa76f14be1" + integrity sha512-yGSza74xk0UG8k+pLh5oeoYirvIiWo5t0/o3zHHAO2tRDiZcxWP7fywNlXhqb6/r6sWvwi+RsyQMWhVLe4BVuA== + typescript@^5.2.0: version "5.2.2" resolved "https://registry.yarnpkg.com/typescript/-/typescript-5.2.2.tgz#5ebb5e5a5b75f085f22bc3f8460fba308310fa78" integrity sha512-mI4WrpHsbCIcwT9cF4FZvr80QUeKvsUsUvKDoR+X/7XHQH98xYD8YHZg7ANtz2GtZt/CBq2QJ0thkGJMHfqc1w== +undici-types@~5.26.4: + version "5.26.5" + resolved "https://registry.yarnpkg.com/undici-types/-/undici-types-5.26.5.tgz#bcd539893d00b56e964fd2657a4866b221a65617" + integrity sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA== + undici@^5.0.0: version "5.25.4" resolved "https://registry.yarnpkg.com/undici/-/undici-5.25.4.tgz#7d8ef81d94f84cd384986271e5e5599b6dff4296" @@ -392,7 +723,28 @@ whatwg-url@^5.0.0: tr46 "~0.0.3" webidl-conversions "^3.0.0" +which@^2.0.1: + version "2.0.2" + resolved "https://registry.yarnpkg.com/which/-/which-2.0.2.tgz#7c6a8dd0a636a0327e10b59c9286eee93f3f51b1" + integrity sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA== + dependencies: + isexe "^2.0.0" + +wrap-ansi@^8.0.1, wrap-ansi@^8.1.0: + version "8.1.0" + resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-8.1.0.tgz#56dc22368ee570face1b49819975d9b9a5ead214" + integrity sha512-si7QWI6zUMq56bESFvagtmzMdGOtoxfR+Sez11Mobfc7tm+VkUckk9bW2UeffTGVUbOksxmSw0AA2gs8g71NCQ== + dependencies: + ansi-styles "^6.1.0" + string-width "^5.0.1" + strip-ansi "^7.0.1" + wrappy@1: version "1.0.2" resolved "https://registry.yarnpkg.com/wrappy/-/wrappy-1.0.2.tgz#b5243d8f3ec1aa35f1364605bc0d1036e30ab69f" integrity sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ== + +yaml@2.3.4: + version "2.3.4" + resolved "https://registry.yarnpkg.com/yaml/-/yaml-2.3.4.tgz#53fc1d514be80aabf386dc6001eb29bf3b7523b2" + integrity sha512-8aAvwVUSHpfEqTQ4w/KMlf3HcRdt50E5ODIQJBw1fQ5RL34xabzxtUlzTXVqc4rkZsPbvrXKWnABCD7kWSmocA==