From 4aa854e8ace262c8ee97b78884890644e90d84ae Mon Sep 17 00:00:00 2001 From: Georges Garnier Date: Sun, 26 Apr 2026 04:04:39 +0200 Subject: [PATCH 01/14] feat(docker): build agent-forge/base image with node 22 and dev tooling Add the base sandbox image (debian bookworm-slim + Node 22 LTS + git, curl, ripgrep, jq) and a build helper script. Non-root user `agent` owns /workspace so it can write there. Closes #1. --- docker/base.Dockerfile | 4 +++- scripts/docker/build-base.sh | 11 +++++++++++ 2 files changed, 14 insertions(+), 1 deletion(-) create mode 100755 scripts/docker/build-base.sh diff --git a/docker/base.Dockerfile b/docker/base.Dockerfile index 594cec3..a0e0c6f 100644 --- a/docker/base.Dockerfile +++ b/docker/base.Dockerfile @@ -23,7 +23,9 @@ RUN curl -fsSL https://deb.nodesource.com/setup_${NODE_VERSION}.x | bash - \ && rm -rf /var/lib/apt/lists/* # ─── Non-root user ─────────────────────────────────────────────── -RUN useradd -m -s /bin/bash agent +RUN useradd -m -s /bin/bash agent \ + && mkdir -p /workspace \ + && chown agent:agent /workspace USER agent WORKDIR /workspace diff --git a/scripts/docker/build-base.sh b/scripts/docker/build-base.sh new file mode 100755 index 0000000..9b7c49e --- /dev/null +++ b/scripts/docker/build-base.sh @@ -0,0 +1,11 @@ +#!/usr/bin/env bash +set -euo pipefail + +IMAGE="agent-forge/base:latest" +DOCKERFILE="docker/base.Dockerfile" + +echo "Building ${IMAGE} from ${DOCKERFILE}..." +docker build -t "${IMAGE}" -f "${DOCKERFILE}" . +echo +echo "✓ Built ${IMAGE}" +echo " Try: docker run --rm ${IMAGE} node --version" From 5c02b3f0c651eeb79a6a887e07247ffa6e3d4948 Mon Sep 17 00:00:00 2001 From: Georges Garnier Date: Sun, 26 Apr 2026 12:06:07 +0200 Subject: [PATCH 02/14] =?UTF-8?q?feat(runtime):=20minimal=20stdin=20?= =?UTF-8?q?=E2=86=92=20llm=20=E2=86=92=20stdout=20via=20vercel=20ai=20sdk?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Implement the runtime entry point that reads a prompt from stdin, calls an OpenAI-compatible LLM endpoint via Vercel AI SDK, writes the assistant text to stdout. Defaults to a local MLX server (mlx_lm.server on 127.0.0.1:8080) so dev runs cost nothing. Switching to OpenAI cloud or any other OpenAI-compatible provider only requires FORGE_BASE_URL + FORGE_API_KEY + FORGE_MODEL. Also includes the bun lockfile generated on first install. Closes #2. --- bun.lock | 594 ++++++++++++++++++++++++++++++++++ packages/runtime/package.json | 5 +- packages/runtime/src/index.ts | 48 ++- 3 files changed, 641 insertions(+), 6 deletions(-) create mode 100644 bun.lock diff --git a/bun.lock b/bun.lock new file mode 100644 index 0000000..bbf93d2 --- /dev/null +++ b/bun.lock @@ -0,0 +1,594 @@ +{ + "lockfileVersion": 1, + "configVersion": 1, + "workspaces": { + "": { + "name": "@agent-forge/monorepo", + "devDependencies": { + "@biomejs/biome": "^1.9.0", + "@types/bun": "latest", + "typescript": "^5.6.0", + }, + }, + "packages/cli": { + "name": "@agent-forge/cli", + "version": "0.0.0", + "bin": { + "forge": "./dist/cli.mjs", + }, + "dependencies": { + "@agent-forge/core": "workspace:*", + "@agent-forge/tools-core": "workspace:*", + "chalk": "^5.3.0", + "commander": "^12.1.0", + "ink": "^5.0.0", + "react": "^18.3.0", + }, + "devDependencies": { + "@types/react": "^18.3.0", + }, + }, + "packages/core": { + "name": "@agent-forge/core", + "version": "0.0.0", + "dependencies": { + "@anthropic-ai/sdk": "^0.32.0", + "@modelcontextprotocol/sdk": "^1.0.0", + "dockerode": "^4.0.0", + "yaml": "^2.6.0", + "zod": "^3.23.0", + }, + "devDependencies": { + "@types/dockerode": "^3.3.0", + }, + }, + "packages/runtime": { + "name": "@agent-forge/runtime", + "version": "0.0.0", + "bin": { + "forge-runtime": "./dist/runtime.mjs", + }, + "dependencies": { + "@agent-forge/core": "workspace:*", + "@agent-forge/tools-core": "workspace:*", + "@ai-sdk/openai": "^1.0.0", + "@modelcontextprotocol/sdk": "^1.0.0", + "ai": "^4.0.0", + "yaml": "^2.6.0", + }, + }, + "packages/tools-core": { + "name": "@agent-forge/tools-core", + "version": "0.0.0", + "dependencies": { + "@agent-forge/core": "workspace:*", + "zod": "^3.23.0", + }, + }, + }, + "packages": { + "@agent-forge/cli": ["@agent-forge/cli@workspace:packages/cli"], + + "@agent-forge/core": ["@agent-forge/core@workspace:packages/core"], + + "@agent-forge/runtime": ["@agent-forge/runtime@workspace:packages/runtime"], + + "@agent-forge/tools-core": ["@agent-forge/tools-core@workspace:packages/tools-core"], + + "@ai-sdk/openai": ["@ai-sdk/openai@1.3.24", "", { "dependencies": { "@ai-sdk/provider": "1.1.3", "@ai-sdk/provider-utils": "2.2.8" }, "peerDependencies": { "zod": "^3.0.0" } }, "sha512-GYXnGJTHRTZc4gJMSmFRgEQudjqd4PUN0ZjQhPwOAYH1yOAvQoG/Ikqs+HyISRbLPCrhbZnPKCNHuRU4OfpW0Q=="], + + "@ai-sdk/provider": ["@ai-sdk/provider@1.1.3", "", { "dependencies": { "json-schema": "^0.4.0" } }, "sha512-qZMxYJ0qqX/RfnuIaab+zp8UAeJn/ygXXAffR5I4N0n1IrvA6qBsjc8hXLmBiMV2zoXlifkacF7sEFnYnjBcqg=="], + + "@ai-sdk/provider-utils": ["@ai-sdk/provider-utils@2.2.8", "", { "dependencies": { "@ai-sdk/provider": "1.1.3", "nanoid": "^3.3.8", "secure-json-parse": "^2.7.0" }, "peerDependencies": { "zod": "^3.23.8" } }, "sha512-fqhG+4sCVv8x7nFzYnFo19ryhAa3w096Kmc3hWxMQfW/TubPOmt3A6tYZhl4mUfQWWQMsuSkLrtjlWuXBVSGQA=="], + + "@ai-sdk/react": ["@ai-sdk/react@1.2.12", "", { "dependencies": { "@ai-sdk/provider-utils": "2.2.8", "@ai-sdk/ui-utils": "1.2.11", "swr": "^2.2.5", "throttleit": "2.1.0" }, "peerDependencies": { "react": "^18 || ^19 || ^19.0.0-rc", "zod": "^3.23.8" }, "optionalPeers": ["zod"] }, "sha512-jK1IZZ22evPZoQW3vlkZ7wvjYGYF+tRBKXtrcolduIkQ/m/sOAVcVeVDUDvh1T91xCnWCdUGCPZg2avZ90mv3g=="], + + "@ai-sdk/ui-utils": ["@ai-sdk/ui-utils@1.2.11", "", { "dependencies": { "@ai-sdk/provider": "1.1.3", "@ai-sdk/provider-utils": "2.2.8", "zod-to-json-schema": "^3.24.1" }, "peerDependencies": { "zod": "^3.23.8" } }, "sha512-3zcwCc8ezzFlwp3ZD15wAPjf2Au4s3vAbKsXQVyhxODHcmu0iyPO2Eua6D/vicq/AUm/BAo60r97O6HU+EI0+w=="], + + "@alcalzone/ansi-tokenize": ["@alcalzone/ansi-tokenize@0.1.3", "", { "dependencies": { "ansi-styles": "^6.2.1", "is-fullwidth-code-point": "^4.0.0" } }, "sha512-3yWxPTq3UQ/FY9p1ErPxIyfT64elWaMvM9lIHnaqpyft63tkxodF5aUElYHrdisWve5cETkh1+KBw1yJuW0aRw=="], + + "@anthropic-ai/sdk": ["@anthropic-ai/sdk@0.32.1", "", { "dependencies": { "@types/node": "^18.11.18", "@types/node-fetch": "^2.6.4", "abort-controller": "^3.0.0", "agentkeepalive": "^4.2.1", "form-data-encoder": "1.7.2", "formdata-node": "^4.3.2", "node-fetch": "^2.6.7" } }, "sha512-U9JwTrDvdQ9iWuABVsMLj8nJVwAyQz6QXvgLsVhryhCEPkLsbcP/MXxm+jYcAwLoV8ESbaTTjnD4kuAFa+Hyjg=="], + + "@balena/dockerignore": ["@balena/dockerignore@1.0.2", "", {}, "sha512-wMue2Sy4GAVTk6Ic4tJVcnfdau+gx2EnG7S+uAEe+TWJFqE4YoWN4/H8MSLj4eYJKxGg26lZwboEniNiNwZQ6Q=="], + + "@biomejs/biome": ["@biomejs/biome@1.9.4", "", { "optionalDependencies": { "@biomejs/cli-darwin-arm64": "1.9.4", "@biomejs/cli-darwin-x64": "1.9.4", "@biomejs/cli-linux-arm64": "1.9.4", "@biomejs/cli-linux-arm64-musl": "1.9.4", "@biomejs/cli-linux-x64": "1.9.4", "@biomejs/cli-linux-x64-musl": "1.9.4", "@biomejs/cli-win32-arm64": "1.9.4", "@biomejs/cli-win32-x64": "1.9.4" }, "bin": { "biome": "bin/biome" } }, "sha512-1rkd7G70+o9KkTn5KLmDYXihGoTaIGO9PIIN2ZB7UJxFrWw04CZHPYiMRjYsaDvVV7hP1dYNRLxSANLaBFGpog=="], + + "@biomejs/cli-darwin-arm64": ["@biomejs/cli-darwin-arm64@1.9.4", "", { "os": "darwin", "cpu": "arm64" }, "sha512-bFBsPWrNvkdKrNCYeAp+xo2HecOGPAy9WyNyB/jKnnedgzl4W4Hb9ZMzYNbf8dMCGmUdSavlYHiR01QaYR58cw=="], + + "@biomejs/cli-darwin-x64": ["@biomejs/cli-darwin-x64@1.9.4", "", { "os": "darwin", "cpu": "x64" }, "sha512-ngYBh/+bEedqkSevPVhLP4QfVPCpb+4BBe2p7Xs32dBgs7rh9nY2AIYUL6BgLw1JVXV8GlpKmb/hNiuIxfPfZg=="], + + "@biomejs/cli-linux-arm64": ["@biomejs/cli-linux-arm64@1.9.4", "", { "os": "linux", "cpu": "arm64" }, "sha512-fJIW0+LYujdjUgJJuwesP4EjIBl/N/TcOX3IvIHJQNsAqvV2CHIogsmA94BPG6jZATS4Hi+xv4SkBBQSt1N4/g=="], + + "@biomejs/cli-linux-arm64-musl": ["@biomejs/cli-linux-arm64-musl@1.9.4", "", { "os": "linux", "cpu": "arm64" }, "sha512-v665Ct9WCRjGa8+kTr0CzApU0+XXtRgwmzIf1SeKSGAv+2scAlW6JR5PMFo6FzqqZ64Po79cKODKf3/AAmECqA=="], + + "@biomejs/cli-linux-x64": ["@biomejs/cli-linux-x64@1.9.4", "", { "os": "linux", "cpu": "x64" }, "sha512-lRCJv/Vi3Vlwmbd6K+oQ0KhLHMAysN8lXoCI7XeHlxaajk06u7G+UsFSO01NAs5iYuWKmVZjmiOzJ0OJmGsMwg=="], + + "@biomejs/cli-linux-x64-musl": ["@biomejs/cli-linux-x64-musl@1.9.4", "", { "os": "linux", "cpu": "x64" }, "sha512-gEhi/jSBhZ2m6wjV530Yy8+fNqG8PAinM3oV7CyO+6c3CEh16Eizm21uHVsyVBEB6RIM8JHIl6AGYCv6Q6Q9Tg=="], + + "@biomejs/cli-win32-arm64": ["@biomejs/cli-win32-arm64@1.9.4", "", { "os": "win32", "cpu": "arm64" }, "sha512-tlbhLk+WXZmgwoIKwHIHEBZUwxml7bRJgk0X2sPyNR3S93cdRq6XulAZRQJ17FYGGzWne0fgrXBKpl7l4M87Hg=="], + + "@biomejs/cli-win32-x64": ["@biomejs/cli-win32-x64@1.9.4", "", { "os": "win32", "cpu": "x64" }, "sha512-8Y5wMhVIPaWe6jw2H+KlEm4wP/f7EW3810ZLmDlrEEy5KvBsb9ECEfu/kMWD484ijfQ8+nIi0giMgu9g1UAuuA=="], + + "@grpc/grpc-js": ["@grpc/grpc-js@1.14.3", "", { "dependencies": { "@grpc/proto-loader": "^0.8.0", "@js-sdsl/ordered-map": "^4.4.2" } }, "sha512-Iq8QQQ/7X3Sac15oB6p0FmUg/klxQvXLeileoqrTRGJYLV+/9tubbr9ipz0GKHjmXVsgFPo/+W+2cA8eNcR+XA=="], + + "@grpc/proto-loader": ["@grpc/proto-loader@0.7.15", "", { "dependencies": { "lodash.camelcase": "^4.3.0", "long": "^5.0.0", "protobufjs": "^7.2.5", "yargs": "^17.7.2" }, "bin": { "proto-loader-gen-types": "build/bin/proto-loader-gen-types.js" } }, "sha512-tMXdRCfYVixjuFK+Hk0Q1s38gV9zDiDJfWL3h1rv4Qc39oILCu1TRTDt7+fGUI8K4G1Fj125Hx/ru3azECWTyQ=="], + + "@hono/node-server": ["@hono/node-server@1.19.14", "", { "peerDependencies": { "hono": "^4" } }, "sha512-GwtvgtXxnWsucXvbQXkRgqksiH2Qed37H9xHZocE5sA3N8O8O8/8FA3uclQXxXVzc9XBZuEOMK7+r02FmSpHtw=="], + + "@js-sdsl/ordered-map": ["@js-sdsl/ordered-map@4.4.2", "", {}, "sha512-iUKgm52T8HOE/makSxjqoWhe95ZJA1/G1sYsGev2JDKUSS14KAgg1LHb+Ba+IPow0xflbnSkOsZcO08C7w1gYw=="], + + "@modelcontextprotocol/sdk": ["@modelcontextprotocol/sdk@1.29.0", "", { "dependencies": { "@hono/node-server": "^1.19.9", "ajv": "^8.17.1", "ajv-formats": "^3.0.1", "content-type": "^1.0.5", "cors": "^2.8.5", "cross-spawn": "^7.0.5", "eventsource": "^3.0.2", "eventsource-parser": "^3.0.0", "express": "^5.2.1", "express-rate-limit": "^8.2.1", "hono": "^4.11.4", "jose": "^6.1.3", "json-schema-typed": "^8.0.2", "pkce-challenge": "^5.0.0", "raw-body": "^3.0.0", "zod": "^3.25 || ^4.0", "zod-to-json-schema": "^3.25.1" }, "peerDependencies": { "@cfworker/json-schema": "^4.1.1" }, "optionalPeers": ["@cfworker/json-schema"] }, "sha512-zo37mZA9hJWpULgkRpowewez1y6ML5GsXJPY8FI0tBBCd77HEvza4jDqRKOXgHNn867PVGCyTdzqpz0izu5ZjQ=="], + + "@opentelemetry/api": ["@opentelemetry/api@1.9.0", "", {}, "sha512-3giAOQvZiH5F9bMlMiv8+GSPMeqg0dbaeo58/0SlA9sxSqZhnUtxzX9/2FzyhS9sWQf5S0GJE0AKBrFqjpeYcg=="], + + "@protobufjs/aspromise": ["@protobufjs/aspromise@1.1.2", "", {}, "sha512-j+gKExEuLmKwvz3OgROXtrJ2UG2x8Ch2YZUxahh+s1F2HZ+wAceUNLkvy6zKCPVRkU++ZWQrdxsUeQXmcg4uoQ=="], + + "@protobufjs/base64": ["@protobufjs/base64@1.1.2", "", {}, "sha512-AZkcAA5vnN/v4PDqKyMR5lx7hZttPDgClv83E//FMNhR2TMcLUhfRUBHCmSl0oi9zMgDDqRUJkSxO3wm85+XLg=="], + + "@protobufjs/codegen": ["@protobufjs/codegen@2.0.4", "", {}, "sha512-YyFaikqM5sH0ziFZCN3xDC7zeGaB/d0IUb9CATugHWbd1FRFwWwt4ld4OYMPWu5a3Xe01mGAULCdqhMlPl29Jg=="], + + "@protobufjs/eventemitter": ["@protobufjs/eventemitter@1.1.0", "", {}, "sha512-j9ednRT81vYJ9OfVuXG6ERSTdEL1xVsNgqpkxMsbIabzSo3goCjDIveeGv5d03om39ML71RdmrGNjG5SReBP/Q=="], + + "@protobufjs/fetch": ["@protobufjs/fetch@1.1.0", "", { "dependencies": { "@protobufjs/aspromise": "^1.1.1", "@protobufjs/inquire": "^1.1.0" } }, "sha512-lljVXpqXebpsijW71PZaCYeIcE5on1w5DlQy5WH6GLbFryLUrBD4932W/E2BSpfRJWseIL4v/KPgBFxDOIdKpQ=="], + + "@protobufjs/float": ["@protobufjs/float@1.0.2", "", {}, "sha512-Ddb+kVXlXst9d+R9PfTIxh1EdNkgoRe5tOX6t01f1lYWOvJnSPDBlG241QLzcyPdoNTsblLUdujGSE4RzrTZGQ=="], + + "@protobufjs/inquire": ["@protobufjs/inquire@1.1.0", "", {}, "sha512-kdSefcPdruJiFMVSbn801t4vFK7KB/5gd2fYvrxhuJYg8ILrmn9SKSX2tZdV6V+ksulWqS7aXjBcRXl3wHoD9Q=="], + + "@protobufjs/path": ["@protobufjs/path@1.1.2", "", {}, "sha512-6JOcJ5Tm08dOHAbdR3GrvP+yUUfkjG5ePsHYczMFLq3ZmMkAD98cDgcT2iA1lJ9NVwFd4tH/iSSoe44YWkltEA=="], + + "@protobufjs/pool": ["@protobufjs/pool@1.1.0", "", {}, "sha512-0kELaGSIDBKvcgS4zkjz1PeddatrjYcmMWOlAuAPwAeccUrPHdUqo/J6LiymHHEiJT5NrF1UVwxY14f+fy4WQw=="], + + "@protobufjs/utf8": ["@protobufjs/utf8@1.1.0", "", {}, "sha512-Vvn3zZrhQZkkBE8LSuW3em98c0FwgO4nxzv6OdSxPKJIEKY2bGbHn+mhGIPerzI4twdxaP8/0+06HBpwf345Lw=="], + + "@types/bun": ["@types/bun@1.3.13", "", { "dependencies": { "bun-types": "1.3.13" } }, "sha512-9fqXWk5YIHGGnUau9TEi+qdlTYDAnOj+xLCmSTwXfAIqXr2x4tytJb43E9uCvt09zJURKXwAtkoH4nLQfzeTXw=="], + + "@types/diff-match-patch": ["@types/diff-match-patch@1.0.36", "", {}, "sha512-xFdR6tkm0MWvBfO8xXCSsinYxHcqkQUlcHeSpMC2ukzOb6lwQAfDmW+Qt0AvlGd8HpsS28qKsB+oPeJn9I39jg=="], + + "@types/docker-modem": ["@types/docker-modem@3.0.6", "", { "dependencies": { "@types/node": "*", "@types/ssh2": "*" } }, "sha512-yKpAGEuKRSS8wwx0joknWxsmLha78wNMe9R2S3UNsVOkZded8UqOrV8KoeDXoXsjndxwyF3eIhyClGbO1SEhEg=="], + + "@types/dockerode": ["@types/dockerode@3.3.47", "", { "dependencies": { "@types/docker-modem": "*", "@types/node": "*", "@types/ssh2": "*" } }, "sha512-ShM1mz7rCjdssXt7Xz0u1/R2BJC7piWa3SJpUBiVjCf2A3XNn4cP6pUVaD8bLanpPVVn4IKzJuw3dOvkJ8IbYw=="], + + "@types/node": ["@types/node@18.19.130", "", { "dependencies": { "undici-types": "~5.26.4" } }, "sha512-GRaXQx6jGfL8sKfaIDD6OupbIHBr9jv7Jnaml9tB7l4v068PAOXqfcujMMo5PhbIs6ggR1XODELqahT2R8v0fg=="], + + "@types/node-fetch": ["@types/node-fetch@2.6.13", "", { "dependencies": { "@types/node": "*", "form-data": "^4.0.4" } }, "sha512-QGpRVpzSaUs30JBSGPjOg4Uveu384erbHBoT1zeONvyCfwQxIkUshLAOqN/k9EjGviPRmWTTe6aH2qySWKTVSw=="], + + "@types/prop-types": ["@types/prop-types@15.7.15", "", {}, "sha512-F6bEyamV9jKGAFBEmlQnesRPGOQqS2+Uwi0Em15xenOxHaf2hv6L8YCVn3rPdPJOiJfPiCnLIRyvwVaqMY3MIw=="], + + "@types/react": ["@types/react@18.3.28", "", { "dependencies": { "@types/prop-types": "*", "csstype": "^3.2.2" } }, "sha512-z9VXpC7MWrhfWipitjNdgCauoMLRdIILQsAEV+ZesIzBq/oUlxk0m3ApZuMFCXdnS4U7KrI+l3WRUEGQ8K1QKw=="], + + "@types/ssh2": ["@types/ssh2@1.15.5", "", { "dependencies": { "@types/node": "^18.11.18" } }, "sha512-N1ASjp/nXH3ovBHddRJpli4ozpk6UdDYIX4RJWFa9L1YKnzdhTlVmiGHm4DZnj/jLbqZpes4aeR30EFGQtvhQQ=="], + + "abort-controller": ["abort-controller@3.0.0", "", { "dependencies": { "event-target-shim": "^5.0.0" } }, "sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg=="], + + "accepts": ["accepts@2.0.0", "", { "dependencies": { "mime-types": "^3.0.0", "negotiator": "^1.0.0" } }, "sha512-5cvg6CtKwfgdmVqY1WIiXKc3Q1bkRqGLi+2W/6ao+6Y7gu/RCwRuAhGEzh5B4KlszSuTLgZYuqFqo5bImjNKng=="], + + "agentkeepalive": ["agentkeepalive@4.6.0", "", { "dependencies": { "humanize-ms": "^1.2.1" } }, "sha512-kja8j7PjmncONqaTsB8fQ+wE2mSU2DJ9D4XKoJ5PFWIdRMa6SLSN1ff4mOr4jCbfRSsxR4keIiySJU0N9T5hIQ=="], + + "ai": ["ai@4.3.19", "", { "dependencies": { "@ai-sdk/provider": "1.1.3", "@ai-sdk/provider-utils": "2.2.8", "@ai-sdk/react": "1.2.12", "@ai-sdk/ui-utils": "1.2.11", "@opentelemetry/api": "1.9.0", "jsondiffpatch": "0.6.0" }, "peerDependencies": { "react": "^18 || ^19 || ^19.0.0-rc", "zod": "^3.23.8" }, "optionalPeers": ["react"] }, "sha512-dIE2bfNpqHN3r6IINp9znguYdhIOheKW2LDigAMrgt/upT3B8eBGPSCblENvaZGoq+hxaN9fSMzjWpbqloP+7Q=="], + + "ajv": ["ajv@8.20.0", "", { "dependencies": { "fast-deep-equal": "^3.1.3", "fast-uri": "^3.0.1", "json-schema-traverse": "^1.0.0", "require-from-string": "^2.0.2" } }, "sha512-Thbli+OlOj+iMPYFBVBfJ3OmCAnaSyNn4M1vz9T6Gka5Jt9ba/HIR56joy65tY6kx/FCF5VXNB819Y7/GUrBGA=="], + + "ajv-formats": ["ajv-formats@3.0.1", "", { "dependencies": { "ajv": "^8.0.0" } }, "sha512-8iUql50EUR+uUcdRQ3HDqa6EVyo3docL8g5WJ3FNcWmu62IbkGUue/pEyLBW8VGKKucTPgqeks4fIU1DA4yowQ=="], + + "ansi-escapes": ["ansi-escapes@7.3.0", "", { "dependencies": { "environment": "^1.0.0" } }, "sha512-BvU8nYgGQBxcmMuEeUEmNTvrMVjJNSH7RgW24vXexN4Ven6qCvy4TntnvlnwnMLTVlcRQQdbRY8NKnaIoeWDNg=="], + + "ansi-regex": ["ansi-regex@6.2.2", "", {}, "sha512-Bq3SmSpyFHaWjPk8If9yc6svM8c56dB5BAtW4Qbw5jHTwwXXcTLoRMkpDJp6VL0XzlWaCHTXrkFURMYmD0sLqg=="], + + "ansi-styles": ["ansi-styles@6.2.3", "", {}, "sha512-4Dj6M28JB+oAH8kFkTLUo+a2jwOFkuqb3yucU0CANcRRUbxS0cP0nZYCGjcc3BNXwRIsUVmDGgzawme7zvJHvg=="], + + "asn1": ["asn1@0.2.6", "", { "dependencies": { "safer-buffer": "~2.1.0" } }, "sha512-ix/FxPn0MDjeyJ7i/yoHGFt/EX6LyNbxSEhPPXODPL+KB0VPk86UYfL0lMdy+KCnv+fmvIzySwaK5COwqVbWTQ=="], + + "asynckit": ["asynckit@0.4.0", "", {}, "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q=="], + + "auto-bind": ["auto-bind@5.0.1", "", {}, "sha512-ooviqdwwgfIfNmDwo94wlshcdzfO64XV0Cg6oDsDYBJfITDz1EngD2z7DkbvCWn+XIMsIqW27sEVF6qcpJrRcg=="], + + "base64-js": ["base64-js@1.5.1", "", {}, "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA=="], + + "bcrypt-pbkdf": ["bcrypt-pbkdf@1.0.2", "", { "dependencies": { "tweetnacl": "^0.14.3" } }, "sha512-qeFIXtP4MSoi6NLqO12WfqARWWuCKi2Rn/9hJLEmtB5yTNr9DqFWkJRCf2qShWzPeAMRnOgCrq0sg/KLv5ES9w=="], + + "bl": ["bl@4.1.0", "", { "dependencies": { "buffer": "^5.5.0", "inherits": "^2.0.4", "readable-stream": "^3.4.0" } }, "sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w=="], + + "body-parser": ["body-parser@2.2.2", "", { "dependencies": { "bytes": "^3.1.2", "content-type": "^1.0.5", "debug": "^4.4.3", "http-errors": "^2.0.0", "iconv-lite": "^0.7.0", "on-finished": "^2.4.1", "qs": "^6.14.1", "raw-body": "^3.0.1", "type-is": "^2.0.1" } }, "sha512-oP5VkATKlNwcgvxi0vM0p/D3n2C3EReYVX+DNYs5TjZFn/oQt2j+4sVJtSMr18pdRr8wjTcBl6LoV+FUwzPmNA=="], + + "buffer": ["buffer@5.7.1", "", { "dependencies": { "base64-js": "^1.3.1", "ieee754": "^1.1.13" } }, "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ=="], + + "buildcheck": ["buildcheck@0.0.7", "", {}, "sha512-lHblz4ahamxpTmnsk+MNTRWsjYKv965MwOrSJyeD588rR3Jcu7swE+0wN5F+PbL5cjgu/9ObkhfzEPuofEMwLA=="], + + "bun-types": ["bun-types@1.3.13", "", { "dependencies": { "@types/node": "*" } }, "sha512-QXKeHLlOLqQX9LgYaHJfzdBaV21T63HhFJnvuRCcjZiaUDpbs5ED1MgxbMra71CsryN/1dAoXuJJJwIv/2drVA=="], + + "bytes": ["bytes@3.1.2", "", {}, "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg=="], + + "call-bind-apply-helpers": ["call-bind-apply-helpers@1.0.2", "", { "dependencies": { "es-errors": "^1.3.0", "function-bind": "^1.1.2" } }, "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ=="], + + "call-bound": ["call-bound@1.0.4", "", { "dependencies": { "call-bind-apply-helpers": "^1.0.2", "get-intrinsic": "^1.3.0" } }, "sha512-+ys997U96po4Kx/ABpBCqhA9EuxJaQWDQg7295H4hBphv3IZg0boBKuwYpt4YXp6MZ5AmZQnU/tyMTlRpaSejg=="], + + "chalk": ["chalk@5.6.2", "", {}, "sha512-7NzBL0rN6fMUW+f7A6Io4h40qQlG+xGmtMxfbnH/K7TAtt8JQWVQK+6g0UXKMeVJoyV5EkkNsErQ8pVD3bLHbA=="], + + "chownr": ["chownr@1.1.4", "", {}, "sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg=="], + + "cli-boxes": ["cli-boxes@3.0.0", "", {}, "sha512-/lzGpEWL/8PfI0BmBOPRwp0c/wFNX1RdUML3jK/RcSBA9T8mZDdQpqYBKtCFTOfQbwPqWEOpjqW+Fnayc0969g=="], + + "cli-cursor": ["cli-cursor@4.0.0", "", { "dependencies": { "restore-cursor": "^4.0.0" } }, "sha512-VGtlMu3x/4DOtIUwEkRezxUZ2lBacNJCHash0N0WeZDBS+7Ux1dm3XWAgWYxLJFMMdOeXMHXorshEFhbMSGelg=="], + + "cli-truncate": ["cli-truncate@4.0.0", "", { "dependencies": { "slice-ansi": "^5.0.0", "string-width": "^7.0.0" } }, "sha512-nPdaFdQ0h/GEigbPClz11D0v/ZJEwxmeVZGeMo3Z5StPtUTkA9o1lD6QwoirYiSDzbcwn2XcjwmCp68W1IS4TA=="], + + "cliui": ["cliui@8.0.1", "", { "dependencies": { "string-width": "^4.2.0", "strip-ansi": "^6.0.1", "wrap-ansi": "^7.0.0" } }, "sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ=="], + + "code-excerpt": ["code-excerpt@4.0.0", "", { "dependencies": { "convert-to-spaces": "^2.0.1" } }, "sha512-xxodCmBen3iy2i0WtAK8FlFNrRzjUqjRsMfho58xT/wvZU1YTM3fCnRjcy1gJPMepaRlgm/0e6w8SpWHpn3/cA=="], + + "color-convert": ["color-convert@2.0.1", "", { "dependencies": { "color-name": "~1.1.4" } }, "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ=="], + + "color-name": ["color-name@1.1.4", "", {}, "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA=="], + + "combined-stream": ["combined-stream@1.0.8", "", { "dependencies": { "delayed-stream": "~1.0.0" } }, "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg=="], + + "commander": ["commander@12.1.0", "", {}, "sha512-Vw8qHK3bZM9y/P10u3Vib8o/DdkvA2OtPtZvD871QKjy74Wj1WSKFILMPRPSdUSx5RFK1arlJzEtA4PkFgnbuA=="], + + "content-disposition": ["content-disposition@1.1.0", "", {}, "sha512-5jRCH9Z/+DRP7rkvY83B+yGIGX96OYdJmzngqnw2SBSxqCFPd0w2km3s5iawpGX8krnwSGmF0FW5Nhr0Hfai3g=="], + + "content-type": ["content-type@1.0.5", "", {}, "sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA=="], + + "convert-to-spaces": ["convert-to-spaces@2.0.1", "", {}, "sha512-rcQ1bsQO9799wq24uE5AM2tAILy4gXGIK/njFWcVQkGNZ96edlpY+A7bjwvzjYvLDyzmG1MmMLZhpcsb+klNMQ=="], + + "cookie": ["cookie@0.7.2", "", {}, "sha512-yki5XnKuf750l50uGTllt6kKILY4nQ1eNIQatoXEByZ5dWgnKqbnqmTrBE5B4N7lrMJKQ2ytWMiTO2o0v6Ew/w=="], + + "cookie-signature": ["cookie-signature@1.2.2", "", {}, "sha512-D76uU73ulSXrD1UXF4KE2TMxVVwhsnCgfAyTg9k8P6KGZjlXKrOLe4dJQKI3Bxi5wjesZoFXJWElNWBjPZMbhg=="], + + "cors": ["cors@2.8.6", "", { "dependencies": { "object-assign": "^4", "vary": "^1" } }, "sha512-tJtZBBHA6vjIAaF6EnIaq6laBBP9aq/Y3ouVJjEfoHbRBcHBAHYcMh/w8LDrk2PvIMMq8gmopa5D4V8RmbrxGw=="], + + "cpu-features": ["cpu-features@0.0.10", "", { "dependencies": { "buildcheck": "~0.0.6", "nan": "^2.19.0" } }, "sha512-9IkYqtX3YHPCzoVg1Py+o9057a3i0fp7S530UWokCSaFVTc7CwXPRiOjRjBQQ18ZCNafx78YfnG+HALxtVmOGA=="], + + "cross-spawn": ["cross-spawn@7.0.6", "", { "dependencies": { "path-key": "^3.1.0", "shebang-command": "^2.0.0", "which": "^2.0.1" } }, "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA=="], + + "csstype": ["csstype@3.2.3", "", {}, "sha512-z1HGKcYy2xA8AGQfwrn0PAy+PB7X/GSj3UVJW9qKyn43xWa+gl5nXmU4qqLMRzWVLFC8KusUX8T/0kCiOYpAIQ=="], + + "debug": ["debug@4.4.3", "", { "dependencies": { "ms": "^2.1.3" } }, "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA=="], + + "delayed-stream": ["delayed-stream@1.0.0", "", {}, "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ=="], + + "depd": ["depd@2.0.0", "", {}, "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw=="], + + "dequal": ["dequal@2.0.3", "", {}, "sha512-0je+qPKHEMohvfRTCEo3CrPG6cAzAYgmzKyxRiYSSDkS6eGJdyVJm7WaYA5ECaAD9wLB2T4EEeymA5aFVcYXCA=="], + + "diff-match-patch": ["diff-match-patch@1.0.5", "", {}, "sha512-IayShXAgj/QMXgB0IWmKx+rOPuGMhqm5w6jvFxmVenXKIzRqTAAsbBPT3kWQeGANj3jGgvcvv4yK6SxqYmikgw=="], + + "docker-modem": ["docker-modem@5.0.7", "", { "dependencies": { "debug": "^4.1.1", "readable-stream": "^3.5.0", "split-ca": "^1.0.1", "ssh2": "^1.15.0" } }, "sha512-XJgGhoR/CLpqshm4d3L7rzH6t8NgDFUIIpztYlLHIApeJjMZKYJMz2zxPsYxnejq5h3ELYSw/RBsi3t5h7gNTA=="], + + "dockerode": ["dockerode@4.0.12", "", { "dependencies": { "@balena/dockerignore": "^1.0.2", "@grpc/grpc-js": "^1.11.1", "@grpc/proto-loader": "^0.7.13", "docker-modem": "^5.0.7", "protobufjs": "^7.3.2", "tar-fs": "^2.1.4", "uuid": "^10.0.0" } }, "sha512-/bCZd6KlGcjZO8Buqmi/vXuqEGVEZ0PNjx/biBNqJD3MhK9DmdiAuKxqfNhflgDESDIiBz3qF+0e55+CpnrUcw=="], + + "dunder-proto": ["dunder-proto@1.0.1", "", { "dependencies": { "call-bind-apply-helpers": "^1.0.1", "es-errors": "^1.3.0", "gopd": "^1.2.0" } }, "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A=="], + + "ee-first": ["ee-first@1.1.1", "", {}, "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow=="], + + "emoji-regex": ["emoji-regex@10.6.0", "", {}, "sha512-toUI84YS5YmxW219erniWD0CIVOo46xGKColeNQRgOzDorgBi1v4D71/OFzgD9GO2UGKIv1C3Sp8DAn0+j5w7A=="], + + "encodeurl": ["encodeurl@2.0.0", "", {}, "sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg=="], + + "end-of-stream": ["end-of-stream@1.4.5", "", { "dependencies": { "once": "^1.4.0" } }, "sha512-ooEGc6HP26xXq/N+GCGOT0JKCLDGrq2bQUZrQ7gyrJiZANJ/8YDTxTpQBXGMn+WbIQXNVpyWymm7KYVICQnyOg=="], + + "environment": ["environment@1.1.0", "", {}, "sha512-xUtoPkMggbz0MPyPiIWr1Kp4aeWJjDZ6SMvURhimjdZgsRuDplF5/s9hcgGhyXMhs+6vpnuoiZ2kFiu3FMnS8Q=="], + + "es-define-property": ["es-define-property@1.0.1", "", {}, "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g=="], + + "es-errors": ["es-errors@1.3.0", "", {}, "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw=="], + + "es-object-atoms": ["es-object-atoms@1.1.1", "", { "dependencies": { "es-errors": "^1.3.0" } }, "sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA=="], + + "es-set-tostringtag": ["es-set-tostringtag@2.1.0", "", { "dependencies": { "es-errors": "^1.3.0", "get-intrinsic": "^1.2.6", "has-tostringtag": "^1.0.2", "hasown": "^2.0.2" } }, "sha512-j6vWzfrGVfyXxge+O0x5sh6cvxAog0a/4Rdd2K36zCMV5eJ+/+tOAngRO8cODMNWbVRdVlmGZQL2YS3yR8bIUA=="], + + "es-toolkit": ["es-toolkit@1.46.0", "", {}, "sha512-IToJ6ct9OLl5zz6WsC/1vZEwfSZ7Myil+ygl5Tf30Xjn9AEkzNB4kqp2G7VUJKF1DtTx/ra5M5KLlXvzOg51BA=="], + + "escalade": ["escalade@3.2.0", "", {}, "sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA=="], + + "escape-html": ["escape-html@1.0.3", "", {}, "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow=="], + + "escape-string-regexp": ["escape-string-regexp@2.0.0", "", {}, "sha512-UpzcLCXolUWcNu5HtVMHYdXJjArjsF9C0aNnquZYY4uW/Vu0miy5YoWvbV345HauVvcAUnpRuhMMcqTcGOY2+w=="], + + "etag": ["etag@1.8.1", "", {}, "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg=="], + + "event-target-shim": ["event-target-shim@5.0.1", "", {}, "sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ=="], + + "eventsource": ["eventsource@3.0.7", "", { "dependencies": { "eventsource-parser": "^3.0.1" } }, "sha512-CRT1WTyuQoD771GW56XEZFQ/ZoSfWid1alKGDYMmkt2yl8UXrVR4pspqWNEcqKvVIzg6PAltWjxcSSPrboA4iA=="], + + "eventsource-parser": ["eventsource-parser@3.0.8", "", {}, "sha512-70QWGkr4snxr0OXLRWsFLeRBIRPuQOvt4s8QYjmUlmlkyTZkRqS7EDVRZtzU3TiyDbXSzaOeF0XUKy8PchzukQ=="], + + "express": ["express@5.2.1", "", { "dependencies": { "accepts": "^2.0.0", "body-parser": "^2.2.1", "content-disposition": "^1.0.0", "content-type": "^1.0.5", "cookie": "^0.7.1", "cookie-signature": "^1.2.1", "debug": "^4.4.0", "depd": "^2.0.0", "encodeurl": "^2.0.0", "escape-html": "^1.0.3", "etag": "^1.8.1", "finalhandler": "^2.1.0", "fresh": "^2.0.0", "http-errors": "^2.0.0", "merge-descriptors": "^2.0.0", "mime-types": "^3.0.0", "on-finished": "^2.4.1", "once": "^1.4.0", "parseurl": "^1.3.3", "proxy-addr": "^2.0.7", "qs": "^6.14.0", "range-parser": "^1.2.1", "router": "^2.2.0", "send": "^1.1.0", "serve-static": "^2.2.0", "statuses": "^2.0.1", "type-is": "^2.0.1", "vary": "^1.1.2" } }, "sha512-hIS4idWWai69NezIdRt2xFVofaF4j+6INOpJlVOLDO8zXGpUVEVzIYk12UUi2JzjEzWL3IOAxcTubgz9Po0yXw=="], + + "express-rate-limit": ["express-rate-limit@8.4.1", "", { "dependencies": { "ip-address": "10.1.0" }, "peerDependencies": { "express": ">= 4.11" } }, "sha512-NGVYwQSAyEQgzxX1iCM978PP9AdO/hW93gMcF6ZwQCm+rFvLsBH6w4xcXWTcliS8La5EPRN3p9wzItqBwJrfNw=="], + + "fast-deep-equal": ["fast-deep-equal@3.1.3", "", {}, "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q=="], + + "fast-uri": ["fast-uri@3.1.0", "", {}, "sha512-iPeeDKJSWf4IEOasVVrknXpaBV0IApz/gp7S2bb7Z4Lljbl2MGJRqInZiUrQwV16cpzw/D3S5j5Julj/gT52AA=="], + + "finalhandler": ["finalhandler@2.1.1", "", { "dependencies": { "debug": "^4.4.0", "encodeurl": "^2.0.0", "escape-html": "^1.0.3", "on-finished": "^2.4.1", "parseurl": "^1.3.3", "statuses": "^2.0.1" } }, "sha512-S8KoZgRZN+a5rNwqTxlZZePjT/4cnm0ROV70LedRHZ0p8u9fRID0hJUZQpkKLzro8LfmC8sx23bY6tVNxv8pQA=="], + + "form-data": ["form-data@4.0.5", "", { "dependencies": { "asynckit": "^0.4.0", "combined-stream": "^1.0.8", "es-set-tostringtag": "^2.1.0", "hasown": "^2.0.2", "mime-types": "^2.1.12" } }, "sha512-8RipRLol37bNs2bhoV67fiTEvdTrbMUYcFTiy3+wuuOnUog2QBHCZWXDRijWQfAkhBj2Uf5UnVaiWwA5vdd82w=="], + + "form-data-encoder": ["form-data-encoder@1.7.2", "", {}, "sha512-qfqtYan3rxrnCk1VYaA4H+Ms9xdpPqvLZa6xmMgFvhO32x7/3J/ExcTd6qpxM0vH2GdMI+poehyBZvqfMTto8A=="], + + "formdata-node": ["formdata-node@4.4.1", "", { "dependencies": { "node-domexception": "1.0.0", "web-streams-polyfill": "4.0.0-beta.3" } }, "sha512-0iirZp3uVDjVGt9p49aTaqjk84TrglENEDuqfdlZQ1roC9CWlPk6Avf8EEnZNcAqPonwkG35x4n3ww/1THYAeQ=="], + + "forwarded": ["forwarded@0.2.0", "", {}, "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow=="], + + "fresh": ["fresh@2.0.0", "", {}, "sha512-Rx/WycZ60HOaqLKAi6cHRKKI7zxWbJ31MhntmtwMoaTeF7XFH9hhBp8vITaMidfljRQ6eYWCKkaTK+ykVJHP2A=="], + + "fs-constants": ["fs-constants@1.0.0", "", {}, "sha512-y6OAwoSIf7FyjMIv94u+b5rdheZEjzR63GTyZJm5qh4Bi+2YgwLCcI/fPFZkL5PSixOt6ZNKm+w+Hfp/Bciwow=="], + + "function-bind": ["function-bind@1.1.2", "", {}, "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA=="], + + "get-caller-file": ["get-caller-file@2.0.5", "", {}, "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg=="], + + "get-east-asian-width": ["get-east-asian-width@1.5.0", "", {}, "sha512-CQ+bEO+Tva/qlmw24dCejulK5pMzVnUOFOijVogd3KQs07HnRIgp8TGipvCCRT06xeYEbpbgwaCxglFyiuIcmA=="], + + "get-intrinsic": ["get-intrinsic@1.3.0", "", { "dependencies": { "call-bind-apply-helpers": "^1.0.2", "es-define-property": "^1.0.1", "es-errors": "^1.3.0", "es-object-atoms": "^1.1.1", "function-bind": "^1.1.2", "get-proto": "^1.0.1", "gopd": "^1.2.0", "has-symbols": "^1.1.0", "hasown": "^2.0.2", "math-intrinsics": "^1.1.0" } }, "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ=="], + + "get-proto": ["get-proto@1.0.1", "", { "dependencies": { "dunder-proto": "^1.0.1", "es-object-atoms": "^1.0.0" } }, "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g=="], + + "gopd": ["gopd@1.2.0", "", {}, "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg=="], + + "has-symbols": ["has-symbols@1.1.0", "", {}, "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ=="], + + "has-tostringtag": ["has-tostringtag@1.0.2", "", { "dependencies": { "has-symbols": "^1.0.3" } }, "sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw=="], + + "hasown": ["hasown@2.0.3", "", { "dependencies": { "function-bind": "^1.1.2" } }, "sha512-ej4AhfhfL2Q2zpMmLo7U1Uv9+PyhIZpgQLGT1F9miIGmiCJIoCgSmczFdrc97mWT4kVY72KA+WnnhJ5pghSvSg=="], + + "hono": ["hono@4.12.15", "", {}, "sha512-qM0jDhFEaCBb4TxoW7f53Qrpv9RBiayUHo0S52JudprkhvpjIrGoU1mnnr29Fvd1U335ZFPZQY1wlkqgfGXyLg=="], + + "http-errors": ["http-errors@2.0.1", "", { "dependencies": { "depd": "~2.0.0", "inherits": "~2.0.4", "setprototypeof": "~1.2.0", "statuses": "~2.0.2", "toidentifier": "~1.0.1" } }, "sha512-4FbRdAX+bSdmo4AUFuS0WNiPz8NgFt+r8ThgNWmlrjQjt1Q7ZR9+zTlce2859x4KSXrwIsaeTqDoKQmtP8pLmQ=="], + + "humanize-ms": ["humanize-ms@1.2.1", "", { "dependencies": { "ms": "^2.0.0" } }, "sha512-Fl70vYtsAFb/C06PTS9dZBo7ihau+Tu/DNCk/OyHhea07S+aeMWpFFkUaXRa8fI+ScZbEI8dfSxwY7gxZ9SAVQ=="], + + "iconv-lite": ["iconv-lite@0.7.2", "", { "dependencies": { "safer-buffer": ">= 2.1.2 < 3.0.0" } }, "sha512-im9DjEDQ55s9fL4EYzOAv0yMqmMBSZp6G0VvFyTMPKWxiSBHUj9NW/qqLmXUwXrrM7AvqSlTCfvqRb0cM8yYqw=="], + + "ieee754": ["ieee754@1.2.1", "", {}, "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA=="], + + "indent-string": ["indent-string@5.0.0", "", {}, "sha512-m6FAo/spmsW2Ab2fU35JTYwtOKa2yAwXSwgjSv1TJzh4Mh7mC3lzAOVLBprb72XsTrgkEIsl7YrFNAiDiRhIGg=="], + + "inherits": ["inherits@2.0.4", "", {}, "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ=="], + + "ink": ["ink@5.2.1", "", { "dependencies": { "@alcalzone/ansi-tokenize": "^0.1.3", "ansi-escapes": "^7.0.0", "ansi-styles": "^6.2.1", "auto-bind": "^5.0.1", "chalk": "^5.3.0", "cli-boxes": "^3.0.0", "cli-cursor": "^4.0.0", "cli-truncate": "^4.0.0", "code-excerpt": "^4.0.0", "es-toolkit": "^1.22.0", "indent-string": "^5.0.0", "is-in-ci": "^1.0.0", "patch-console": "^2.0.0", "react-reconciler": "^0.29.0", "scheduler": "^0.23.0", "signal-exit": "^3.0.7", "slice-ansi": "^7.1.0", "stack-utils": "^2.0.6", "string-width": "^7.2.0", "type-fest": "^4.27.0", "widest-line": "^5.0.0", "wrap-ansi": "^9.0.0", "ws": "^8.18.0", "yoga-layout": "~3.2.1" }, "peerDependencies": { "@types/react": ">=18.0.0", "react": ">=18.0.0", "react-devtools-core": "^4.19.1" }, "optionalPeers": ["@types/react", "react-devtools-core"] }, "sha512-BqcUyWrG9zq5HIwW6JcfFHsIYebJkWWb4fczNah1goUO0vv5vneIlfwuS85twyJ5hYR/y18FlAYUxrO9ChIWVg=="], + + "ip-address": ["ip-address@10.1.0", "", {}, "sha512-XXADHxXmvT9+CRxhXg56LJovE+bmWnEWB78LB83VZTprKTmaC5QfruXocxzTZ2Kl0DNwKuBdlIhjL8LeY8Sf8Q=="], + + "ipaddr.js": ["ipaddr.js@1.9.1", "", {}, "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g=="], + + "is-fullwidth-code-point": ["is-fullwidth-code-point@4.0.0", "", {}, "sha512-O4L094N2/dZ7xqVdrXhh9r1KODPJpFms8B5sGdJLPy664AgvXsreZUyCQQNItZRDlYug4xStLjNp/sz3HvBowQ=="], + + "is-in-ci": ["is-in-ci@1.0.0", "", { "bin": { "is-in-ci": "cli.js" } }, "sha512-eUuAjybVTHMYWm/U+vBO1sY/JOCgoPCXRxzdju0K+K0BiGW0SChEL1MLC0PoCIR1OlPo5YAp8HuQoUlsWEICwg=="], + + "is-promise": ["is-promise@4.0.0", "", {}, "sha512-hvpoI6korhJMnej285dSg6nu1+e6uxs7zG3BYAm5byqDsgJNWwxzM6z6iZiAgQR4TJ30JmBTOwqZUw3WlyH3AQ=="], + + "isexe": ["isexe@2.0.0", "", {}, "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw=="], + + "jose": ["jose@6.2.2", "", {}, "sha512-d7kPDd34KO/YnzaDOlikGpOurfF0ByC2sEV4cANCtdqLlTfBlw2p14O/5d/zv40gJPbIQxfES3nSx1/oYNyuZQ=="], + + "js-tokens": ["js-tokens@4.0.0", "", {}, "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ=="], + + "json-schema": ["json-schema@0.4.0", "", {}, "sha512-es94M3nTIfsEPisRafak+HDLfHXnKBhV3vU5eqPcS3flIWqcxJWgXHXiey3YrpaNsanY5ei1VoYEbOzijuq9BA=="], + + "json-schema-traverse": ["json-schema-traverse@1.0.0", "", {}, "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug=="], + + "json-schema-typed": ["json-schema-typed@8.0.2", "", {}, "sha512-fQhoXdcvc3V28x7C7BMs4P5+kNlgUURe2jmUT1T//oBRMDrqy1QPelJimwZGo7Hg9VPV3EQV5Bnq4hbFy2vetA=="], + + "jsondiffpatch": ["jsondiffpatch@0.6.0", "", { "dependencies": { "@types/diff-match-patch": "^1.0.36", "chalk": "^5.3.0", "diff-match-patch": "^1.0.5" }, "bin": { "jsondiffpatch": "bin/jsondiffpatch.js" } }, "sha512-3QItJOXp2AP1uv7waBkao5nCvhEv+QmJAd38Ybq7wNI74Q+BBmnLn4EDKz6yI9xGAIQoUF87qHt+kc1IVxB4zQ=="], + + "lodash.camelcase": ["lodash.camelcase@4.3.0", "", {}, "sha512-TwuEnCnxbc3rAvhf/LbG7tJUDzhqXyFnv3dtzLOPgCG/hODL7WFnsbwktkD7yUV0RrreP/l1PALq/YSg6VvjlA=="], + + "long": ["long@5.3.2", "", {}, "sha512-mNAgZ1GmyNhD7AuqnTG3/VQ26o760+ZYBPKjPvugO8+nLbYfX6TVpJPseBvopbdY+qpZ/lKUnmEc1LeZYS3QAA=="], + + "loose-envify": ["loose-envify@1.4.0", "", { "dependencies": { "js-tokens": "^3.0.0 || ^4.0.0" }, "bin": { "loose-envify": "cli.js" } }, "sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q=="], + + "math-intrinsics": ["math-intrinsics@1.1.0", "", {}, "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g=="], + + "media-typer": ["media-typer@1.1.0", "", {}, "sha512-aisnrDP4GNe06UcKFnV5bfMNPBUw4jsLGaWwWfnH3v02GnBuXX2MCVn5RbrWo0j3pczUilYblq7fQ7Nw2t5XKw=="], + + "merge-descriptors": ["merge-descriptors@2.0.0", "", {}, "sha512-Snk314V5ayFLhp3fkUREub6WtjBfPdCPY1Ln8/8munuLuiYhsABgBVWsozAG+MWMbVEvcdcpbi9R7ww22l9Q3g=="], + + "mime-db": ["mime-db@1.54.0", "", {}, "sha512-aU5EJuIN2WDemCcAp2vFBfp/m4EAhWJnUNSSw0ixs7/kXbd6Pg64EmwJkNdFhB8aWt1sH2CTXrLxo/iAGV3oPQ=="], + + "mime-types": ["mime-types@3.0.2", "", { "dependencies": { "mime-db": "^1.54.0" } }, "sha512-Lbgzdk0h4juoQ9fCKXW4by0UJqj+nOOrI9MJ1sSj4nI8aI2eo1qmvQEie4VD1glsS250n15LsWsYtCugiStS5A=="], + + "mimic-fn": ["mimic-fn@2.1.0", "", {}, "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg=="], + + "mkdirp-classic": ["mkdirp-classic@0.5.3", "", {}, "sha512-gKLcREMhtuZRwRAfqP3RFW+TK4JqApVBtOIftVgjuABpAtpxhPGaDcfvbhNvD0B8iD1oUr/txX35NjcaY6Ns/A=="], + + "ms": ["ms@2.1.3", "", {}, "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA=="], + + "nan": ["nan@2.26.2", "", {}, "sha512-0tTvBTYkt3tdGw22nrAy50x7gpbGCCFH3AFcyS5WiUu7Eu4vWlri1woE6qHBSfy11vksDqkiwjOnlR7WV8G1Hw=="], + + "nanoid": ["nanoid@3.3.11", "", { "bin": { "nanoid": "bin/nanoid.cjs" } }, "sha512-N8SpfPUnUp1bK+PMYW8qSWdl9U+wwNWI4QKxOYDy9JAro3WMX7p2OeVRF9v+347pnakNevPmiHhNmZ2HbFA76w=="], + + "negotiator": ["negotiator@1.0.0", "", {}, "sha512-8Ofs/AUQh8MaEcrlq5xOX0CQ9ypTF5dl78mjlMNfOK08fzpgTHQRQPBxcPlEtIw0yRpws+Zo/3r+5WRby7u3Gg=="], + + "node-domexception": ["node-domexception@1.0.0", "", {}, "sha512-/jKZoMpw0F8GRwl4/eLROPA3cfcXtLApP0QzLmUT/HuPCZWyB7IY9ZrMeKw2O/nFIqPQB3PVM9aYm0F312AXDQ=="], + + "node-fetch": ["node-fetch@2.7.0", "", { "dependencies": { "whatwg-url": "^5.0.0" }, "peerDependencies": { "encoding": "^0.1.0" }, "optionalPeers": ["encoding"] }, "sha512-c4FRfUm/dbcWZ7U+1Wq0AwCyFL+3nt2bEw05wfxSz+DWpWsitgmSgYmy2dQdWyKC1694ELPqMs/YzUSNozLt8A=="], + + "object-assign": ["object-assign@4.1.1", "", {}, "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg=="], + + "object-inspect": ["object-inspect@1.13.4", "", {}, "sha512-W67iLl4J2EXEGTbfeHCffrjDfitvLANg0UlX3wFUUSTx92KXRFegMHUVgSqE+wvhAbi4WqjGg9czysTV2Epbew=="], + + "on-finished": ["on-finished@2.4.1", "", { "dependencies": { "ee-first": "1.1.1" } }, "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg=="], + + "once": ["once@1.4.0", "", { "dependencies": { "wrappy": "1" } }, "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w=="], + + "onetime": ["onetime@5.1.2", "", { "dependencies": { "mimic-fn": "^2.1.0" } }, "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg=="], + + "parseurl": ["parseurl@1.3.3", "", {}, "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ=="], + + "patch-console": ["patch-console@2.0.0", "", {}, "sha512-0YNdUceMdaQwoKce1gatDScmMo5pu/tfABfnzEqeG0gtTmd7mh/WcwgUjtAeOU7N8nFFlbQBnFK2gXW5fGvmMA=="], + + "path-key": ["path-key@3.1.1", "", {}, "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q=="], + + "path-to-regexp": ["path-to-regexp@8.4.2", "", {}, "sha512-qRcuIdP69NPm4qbACK+aDogI5CBDMi1jKe0ry5rSQJz8JVLsC7jV8XpiJjGRLLol3N+R5ihGYcrPLTno6pAdBA=="], + + "pkce-challenge": ["pkce-challenge@5.0.1", "", {}, "sha512-wQ0b/W4Fr01qtpHlqSqspcj3EhBvimsdh0KlHhH8HRZnMsEa0ea2fTULOXOS9ccQr3om+GcGRk4e+isrZWV8qQ=="], + + "protobufjs": ["protobufjs@7.5.5", "", { "dependencies": { "@protobufjs/aspromise": "^1.1.2", "@protobufjs/base64": "^1.1.2", "@protobufjs/codegen": "^2.0.4", "@protobufjs/eventemitter": "^1.1.0", "@protobufjs/fetch": "^1.1.0", "@protobufjs/float": "^1.0.2", "@protobufjs/inquire": "^1.1.0", "@protobufjs/path": "^1.1.2", "@protobufjs/pool": "^1.1.0", "@protobufjs/utf8": "^1.1.0", "@types/node": ">=13.7.0", "long": "^5.0.0" } }, "sha512-3wY1AxV+VBNW8Yypfd1yQY9pXnqTAN+KwQxL8iYm3/BjKYMNg4i0owhEe26PWDOMaIrzeeF98Lqd5NGz4omiIg=="], + + "proxy-addr": ["proxy-addr@2.0.7", "", { "dependencies": { "forwarded": "0.2.0", "ipaddr.js": "1.9.1" } }, "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg=="], + + "pump": ["pump@3.0.4", "", { "dependencies": { "end-of-stream": "^1.1.0", "once": "^1.3.1" } }, "sha512-VS7sjc6KR7e1ukRFhQSY5LM2uBWAUPiOPa/A3mkKmiMwSmRFUITt0xuj+/lesgnCv+dPIEYlkzrcyXgquIHMcA=="], + + "qs": ["qs@6.15.1", "", { "dependencies": { "side-channel": "^1.1.0" } }, "sha512-6YHEFRL9mfgcAvql/XhwTvf5jKcOiiupt2FiJxHkiX1z4j7WL8J/jRHYLluORvc1XxB5rV20KoeK00gVJamspg=="], + + "range-parser": ["range-parser@1.2.1", "", {}, "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg=="], + + "raw-body": ["raw-body@3.0.2", "", { "dependencies": { "bytes": "~3.1.2", "http-errors": "~2.0.1", "iconv-lite": "~0.7.0", "unpipe": "~1.0.0" } }, "sha512-K5zQjDllxWkf7Z5xJdV0/B0WTNqx6vxG70zJE4N0kBs4LovmEYWJzQGxC9bS9RAKu3bgM40lrd5zoLJ12MQ5BA=="], + + "react": ["react@18.3.1", "", { "dependencies": { "loose-envify": "^1.1.0" } }, "sha512-wS+hAgJShR0KhEvPJArfuPVN1+Hz1t0Y6n5jLrGQbkb4urgPE/0Rve+1kMB1v/oWgHgm4WIcV+i7F2pTVj+2iQ=="], + + "react-reconciler": ["react-reconciler@0.29.2", "", { "dependencies": { "loose-envify": "^1.1.0", "scheduler": "^0.23.2" }, "peerDependencies": { "react": "^18.3.1" } }, "sha512-zZQqIiYgDCTP/f1N/mAR10nJGrPD2ZR+jDSEsKWJHYC7Cm2wodlwbR3upZRdC3cjIjSlTLNVyO7Iu0Yy7t2AYg=="], + + "readable-stream": ["readable-stream@3.6.2", "", { "dependencies": { "inherits": "^2.0.3", "string_decoder": "^1.1.1", "util-deprecate": "^1.0.1" } }, "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA=="], + + "require-directory": ["require-directory@2.1.1", "", {}, "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q=="], + + "require-from-string": ["require-from-string@2.0.2", "", {}, "sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw=="], + + "restore-cursor": ["restore-cursor@4.0.0", "", { "dependencies": { "onetime": "^5.1.0", "signal-exit": "^3.0.2" } }, "sha512-I9fPXU9geO9bHOt9pHHOhOkYerIMsmVaWB0rA2AI9ERh/+x/i7MV5HKBNrg+ljO5eoPVgCcnFuRjJ9uH6I/3eg=="], + + "router": ["router@2.2.0", "", { "dependencies": { "debug": "^4.4.0", "depd": "^2.0.0", "is-promise": "^4.0.0", "parseurl": "^1.3.3", "path-to-regexp": "^8.0.0" } }, "sha512-nLTrUKm2UyiL7rlhapu/Zl45FwNgkZGaCpZbIHajDYgwlJCOzLSk+cIPAnsEqV955GjILJnKbdQC1nVPz+gAYQ=="], + + "safe-buffer": ["safe-buffer@5.2.1", "", {}, "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ=="], + + "safer-buffer": ["safer-buffer@2.1.2", "", {}, "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg=="], + + "scheduler": ["scheduler@0.23.2", "", { "dependencies": { "loose-envify": "^1.1.0" } }, "sha512-UOShsPwz7NrMUqhR6t0hWjFduvOzbtv7toDH1/hIrfRNIDBnnBWd0CwJTGvTpngVlmwGCdP9/Zl/tVrDqcuYzQ=="], + + "secure-json-parse": ["secure-json-parse@2.7.0", "", {}, "sha512-6aU+Rwsezw7VR8/nyvKTx8QpWH9FrcYiXXlqC4z5d5XQBDRqtbfsRjnwGyqbi3gddNtWHuEk9OANUotL26qKUw=="], + + "send": ["send@1.2.1", "", { "dependencies": { "debug": "^4.4.3", "encodeurl": "^2.0.0", "escape-html": "^1.0.3", "etag": "^1.8.1", "fresh": "^2.0.0", "http-errors": "^2.0.1", "mime-types": "^3.0.2", "ms": "^2.1.3", "on-finished": "^2.4.1", "range-parser": "^1.2.1", "statuses": "^2.0.2" } }, "sha512-1gnZf7DFcoIcajTjTwjwuDjzuz4PPcY2StKPlsGAQ1+YH20IRVrBaXSWmdjowTJ6u8Rc01PoYOGHXfP1mYcZNQ=="], + + "serve-static": ["serve-static@2.2.1", "", { "dependencies": { "encodeurl": "^2.0.0", "escape-html": "^1.0.3", "parseurl": "^1.3.3", "send": "^1.2.0" } }, "sha512-xRXBn0pPqQTVQiC8wyQrKs2MOlX24zQ0POGaj0kultvoOCstBQM5yvOhAVSUwOMjQtTvsPWoNCHfPGwaaQJhTw=="], + + "setprototypeof": ["setprototypeof@1.2.0", "", {}, "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw=="], + + "shebang-command": ["shebang-command@2.0.0", "", { "dependencies": { "shebang-regex": "^3.0.0" } }, "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA=="], + + "shebang-regex": ["shebang-regex@3.0.0", "", {}, "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A=="], + + "side-channel": ["side-channel@1.1.0", "", { "dependencies": { "es-errors": "^1.3.0", "object-inspect": "^1.13.3", "side-channel-list": "^1.0.0", "side-channel-map": "^1.0.1", "side-channel-weakmap": "^1.0.2" } }, "sha512-ZX99e6tRweoUXqR+VBrslhda51Nh5MTQwou5tnUDgbtyM0dBgmhEDtWGP/xbKn6hqfPRHujUNwz5fy/wbbhnpw=="], + + "side-channel-list": ["side-channel-list@1.0.1", "", { "dependencies": { "es-errors": "^1.3.0", "object-inspect": "^1.13.4" } }, "sha512-mjn/0bi/oUURjc5Xl7IaWi/OJJJumuoJFQJfDDyO46+hBWsfaVM65TBHq2eoZBhzl9EchxOijpkbRC8SVBQU0w=="], + + "side-channel-map": ["side-channel-map@1.0.1", "", { "dependencies": { "call-bound": "^1.0.2", "es-errors": "^1.3.0", "get-intrinsic": "^1.2.5", "object-inspect": "^1.13.3" } }, "sha512-VCjCNfgMsby3tTdo02nbjtM/ewra6jPHmpThenkTYh8pG9ucZ/1P8So4u4FGBek/BjpOVsDCMoLA/iuBKIFXRA=="], + + "side-channel-weakmap": ["side-channel-weakmap@1.0.2", "", { "dependencies": { "call-bound": "^1.0.2", "es-errors": "^1.3.0", "get-intrinsic": "^1.2.5", "object-inspect": "^1.13.3", "side-channel-map": "^1.0.1" } }, "sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A=="], + + "signal-exit": ["signal-exit@3.0.7", "", {}, "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ=="], + + "slice-ansi": ["slice-ansi@7.1.2", "", { "dependencies": { "ansi-styles": "^6.2.1", "is-fullwidth-code-point": "^5.0.0" } }, "sha512-iOBWFgUX7caIZiuutICxVgX1SdxwAVFFKwt1EvMYYec/NWO5meOJ6K5uQxhrYBdQJne4KxiqZc+KptFOWFSI9w=="], + + "split-ca": ["split-ca@1.0.1", "", {}, "sha512-Q5thBSxp5t8WPTTJQS59LrGqOZqOsrhDGDVm8azCqIBjSBd7nd9o2PM+mDulQQkh8h//4U6hFZnc/mul8t5pWQ=="], + + "ssh2": ["ssh2@1.17.0", "", { "dependencies": { "asn1": "^0.2.6", "bcrypt-pbkdf": "^1.0.2" }, "optionalDependencies": { "cpu-features": "~0.0.10", "nan": "^2.23.0" } }, "sha512-wPldCk3asibAjQ/kziWQQt1Wh3PgDFpC0XpwclzKcdT1vql6KeYxf5LIt4nlFkUeR8WuphYMKqUA56X4rjbfgQ=="], + + "stack-utils": ["stack-utils@2.0.6", "", { "dependencies": { "escape-string-regexp": "^2.0.0" } }, "sha512-XlkWvfIm6RmsWtNJx+uqtKLS8eqFbxUg0ZzLXqY0caEy9l7hruX8IpiDnjsLavoBgqCCR71TqWO8MaXYheJ3RQ=="], + + "statuses": ["statuses@2.0.2", "", {}, "sha512-DvEy55V3DB7uknRo+4iOGT5fP1slR8wQohVdknigZPMpMstaKJQWhwiYBACJE3Ul2pTnATihhBYnRhZQHGBiRw=="], + + "string-width": ["string-width@7.2.0", "", { "dependencies": { "emoji-regex": "^10.3.0", "get-east-asian-width": "^1.0.0", "strip-ansi": "^7.1.0" } }, "sha512-tsaTIkKW9b4N+AEj+SVA+WhJzV7/zMhcSu78mLKWSk7cXMOSHsBKFWUs0fWwq8QyK3MgJBQRX6Gbi4kYbdvGkQ=="], + + "string_decoder": ["string_decoder@1.3.0", "", { "dependencies": { "safe-buffer": "~5.2.0" } }, "sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA=="], + + "strip-ansi": ["strip-ansi@7.2.0", "", { "dependencies": { "ansi-regex": "^6.2.2" } }, "sha512-yDPMNjp4WyfYBkHnjIRLfca1i6KMyGCtsVgoKe/z1+6vukgaENdgGBZt+ZmKPc4gavvEZ5OgHfHdrazhgNyG7w=="], + + "swr": ["swr@2.4.1", "", { "dependencies": { "dequal": "^2.0.3", "use-sync-external-store": "^1.6.0" }, "peerDependencies": { "react": "^16.11.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" } }, "sha512-2CC6CiKQtEwaEeNiqWTAw9PGykW8SR5zZX8MZk6TeAvEAnVS7Visz8WzphqgtQ8v2xz/4Q5K+j+SeMaKXeeQIA=="], + + "tar-fs": ["tar-fs@2.1.4", "", { "dependencies": { "chownr": "^1.1.1", "mkdirp-classic": "^0.5.2", "pump": "^3.0.0", "tar-stream": "^2.1.4" } }, "sha512-mDAjwmZdh7LTT6pNleZ05Yt65HC3E+NiQzl672vQG38jIrehtJk/J3mNwIg+vShQPcLF/LV7CMnDW6vjj6sfYQ=="], + + "tar-stream": ["tar-stream@2.2.0", "", { "dependencies": { "bl": "^4.0.3", "end-of-stream": "^1.4.1", "fs-constants": "^1.0.0", "inherits": "^2.0.3", "readable-stream": "^3.1.1" } }, "sha512-ujeqbceABgwMZxEJnk2HDY2DlnUZ+9oEcb1KzTVfYHio0UE6dG71n60d8D2I4qNvleWrrXpmjpt7vZeF1LnMZQ=="], + + "throttleit": ["throttleit@2.1.0", "", {}, "sha512-nt6AMGKW1p/70DF/hGBdJB57B8Tspmbp5gfJ8ilhLnt7kkr2ye7hzD6NVG8GGErk2HWF34igrL2CXmNIkzKqKw=="], + + "toidentifier": ["toidentifier@1.0.1", "", {}, "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA=="], + + "tr46": ["tr46@0.0.3", "", {}, "sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw=="], + + "tweetnacl": ["tweetnacl@0.14.5", "", {}, "sha512-KXXFFdAbFXY4geFIwoyNK+f5Z1b7swfXABfL7HXCmoIWMKU3dmS26672A4EeQtDzLKy7SXmfBu51JolvEKwtGA=="], + + "type-fest": ["type-fest@4.41.0", "", {}, "sha512-TeTSQ6H5YHvpqVwBRcnLDCBnDOHWYu7IvGbHT6N8AOymcr9PJGjc1GTtiWZTYg0NCgYwvnYWEkVChQAr9bjfwA=="], + + "type-is": ["type-is@2.0.1", "", { "dependencies": { "content-type": "^1.0.5", "media-typer": "^1.1.0", "mime-types": "^3.0.0" } }, "sha512-OZs6gsjF4vMp32qrCbiVSkrFmXtG/AZhY3t0iAMrMBiAZyV9oALtXO8hsrHbMXF9x6L3grlFuwW2oAz7cav+Gw=="], + + "typescript": ["typescript@5.9.3", "", { "bin": { "tsc": "bin/tsc", "tsserver": "bin/tsserver" } }, "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw=="], + + "undici-types": ["undici-types@5.26.5", "", {}, "sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA=="], + + "unpipe": ["unpipe@1.0.0", "", {}, "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ=="], + + "use-sync-external-store": ["use-sync-external-store@1.6.0", "", { "peerDependencies": { "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" } }, "sha512-Pp6GSwGP/NrPIrxVFAIkOQeyw8lFenOHijQWkUTrDvrF4ALqylP2C/KCkeS9dpUM3KvYRQhna5vt7IL95+ZQ9w=="], + + "util-deprecate": ["util-deprecate@1.0.2", "", {}, "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw=="], + + "uuid": ["uuid@10.0.0", "", { "bin": { "uuid": "dist/bin/uuid" } }, "sha512-8XkAphELsDnEGrDxUOHB3RGvXz6TeuYSGEZBOjtTtPm2lwhGBjLgOzLHB63IUWfBpNucQjND6d3AOudO+H3RWQ=="], + + "vary": ["vary@1.1.2", "", {}, "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg=="], + + "web-streams-polyfill": ["web-streams-polyfill@4.0.0-beta.3", "", {}, "sha512-QW95TCTaHmsYfHDybGMwO5IJIM93I/6vTRk+daHTWFPhwh+C8Cg7j7XyKrwrj8Ib6vYXe0ocYNrmzY4xAAN6ug=="], + + "webidl-conversions": ["webidl-conversions@3.0.1", "", {}, "sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ=="], + + "whatwg-url": ["whatwg-url@5.0.0", "", { "dependencies": { "tr46": "~0.0.3", "webidl-conversions": "^3.0.0" } }, "sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw=="], + + "which": ["which@2.0.2", "", { "dependencies": { "isexe": "^2.0.0" }, "bin": { "node-which": "./bin/node-which" } }, "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA=="], + + "widest-line": ["widest-line@5.0.0", "", { "dependencies": { "string-width": "^7.0.0" } }, "sha512-c9bZp7b5YtRj2wOe6dlj32MK+Bx/M/d+9VB2SHM1OtsUHR0aV0tdP6DWh/iMt0kWi1t5g1Iudu6hQRNd1A4PVA=="], + + "wrap-ansi": ["wrap-ansi@9.0.2", "", { "dependencies": { "ansi-styles": "^6.2.1", "string-width": "^7.0.0", "strip-ansi": "^7.1.0" } }, "sha512-42AtmgqjV+X1VpdOfyTGOYRi0/zsoLqtXQckTmqTeybT+BDIbM/Guxo7x3pE2vtpr1ok6xRqM9OpBe+Jyoqyww=="], + + "wrappy": ["wrappy@1.0.2", "", {}, "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ=="], + + "ws": ["ws@8.20.0", "", { "peerDependencies": { "bufferutil": "^4.0.1", "utf-8-validate": ">=5.0.2" }, "optionalPeers": ["bufferutil", "utf-8-validate"] }, "sha512-sAt8BhgNbzCtgGbt2OxmpuryO63ZoDk/sqaB/znQm94T4fCEsy/yV+7CdC1kJhOU9lboAEU7R3kquuycDoibVA=="], + + "y18n": ["y18n@5.0.8", "", {}, "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA=="], + + "yaml": ["yaml@2.8.3", "", { "bin": { "yaml": "bin.mjs" } }, "sha512-AvbaCLOO2Otw/lW5bmh9d/WEdcDFdQp2Z2ZUH3pX9U2ihyUY0nvLv7J6TrWowklRGPYbB/IuIMfYgxaCPg5Bpg=="], + + "yargs": ["yargs@17.7.2", "", { "dependencies": { "cliui": "^8.0.1", "escalade": "^3.1.1", "get-caller-file": "^2.0.5", "require-directory": "^2.1.1", "string-width": "^4.2.3", "y18n": "^5.0.5", "yargs-parser": "^21.1.1" } }, "sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w=="], + + "yargs-parser": ["yargs-parser@21.1.1", "", {}, "sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw=="], + + "yoga-layout": ["yoga-layout@3.2.1", "", {}, "sha512-0LPOt3AxKqMdFBZA3HBAt/t/8vIKq7VaQYbuA8WxCgung+p9TVyKRYdpvCb80HcdTN2NkbIKbhNwKUfm3tQywQ=="], + + "zod": ["zod@3.25.76", "", {}, "sha512-gzUt/qt81nXsFGKIFcC3YnfEAx5NkunCfnDlvuBSSFS02bcXu4Lmea0AFIUwbLWxWPx3d9p8S5QoaujKcNQxcQ=="], + + "zod-to-json-schema": ["zod-to-json-schema@3.25.2", "", { "peerDependencies": { "zod": "^3.25.28 || ^4" } }, "sha512-O/PgfnpT1xKSDeQYSCfRI5Gy3hPf91mKVDuYLUHZJMiDFptvP41MSnWofm8dnCm0256ZNfZIM7DSzuSMAFnjHA=="], + + "@grpc/grpc-js/@grpc/proto-loader": ["@grpc/proto-loader@0.8.0", "", { "dependencies": { "lodash.camelcase": "^4.3.0", "long": "^5.0.0", "protobufjs": "^7.5.3", "yargs": "^17.7.2" }, "bin": { "proto-loader-gen-types": "build/bin/proto-loader-gen-types.js" } }, "sha512-rc1hOQtjIWGxcxpb9aHAfLpIctjEnsDehj0DAiVfBlmT84uvR0uUtN2hEi/ecvWVjXUGf5qPF4qEgiLOx1YIMQ=="], + + "cli-truncate/slice-ansi": ["slice-ansi@5.0.0", "", { "dependencies": { "ansi-styles": "^6.0.0", "is-fullwidth-code-point": "^4.0.0" } }, "sha512-FC+lgizVPfie0kkhqUScwRu1O/lF6NOgJmlCgK+/LYxDCTk8sGelYaHDhFcDN+Sn3Cv+3VSa4Byeo+IMCzpMgQ=="], + + "cliui/string-width": ["string-width@4.2.3", "", { "dependencies": { "emoji-regex": "^8.0.0", "is-fullwidth-code-point": "^3.0.0", "strip-ansi": "^6.0.1" } }, "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g=="], + + "cliui/strip-ansi": ["strip-ansi@6.0.1", "", { "dependencies": { "ansi-regex": "^5.0.1" } }, "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A=="], + + "cliui/wrap-ansi": ["wrap-ansi@7.0.0", "", { "dependencies": { "ansi-styles": "^4.0.0", "string-width": "^4.1.0", "strip-ansi": "^6.0.0" } }, "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q=="], + + "form-data/mime-types": ["mime-types@2.1.35", "", { "dependencies": { "mime-db": "1.52.0" } }, "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw=="], + + "slice-ansi/is-fullwidth-code-point": ["is-fullwidth-code-point@5.1.0", "", { "dependencies": { "get-east-asian-width": "^1.3.1" } }, "sha512-5XHYaSyiqADb4RnZ1Bdad6cPp8Toise4TzEjcOYDHZkTCbKgiUl7WTUCpNWHuxmDt91wnsZBc9xinNzopv3JMQ=="], + + "yargs/string-width": ["string-width@4.2.3", "", { "dependencies": { "emoji-regex": "^8.0.0", "is-fullwidth-code-point": "^3.0.0", "strip-ansi": "^6.0.1" } }, "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g=="], + + "cliui/string-width/emoji-regex": ["emoji-regex@8.0.0", "", {}, "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A=="], + + "cliui/string-width/is-fullwidth-code-point": ["is-fullwidth-code-point@3.0.0", "", {}, "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg=="], + + "cliui/strip-ansi/ansi-regex": ["ansi-regex@5.0.1", "", {}, "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ=="], + + "cliui/wrap-ansi/ansi-styles": ["ansi-styles@4.3.0", "", { "dependencies": { "color-convert": "^2.0.1" } }, "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg=="], + + "form-data/mime-types/mime-db": ["mime-db@1.52.0", "", {}, "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg=="], + + "yargs/string-width/emoji-regex": ["emoji-regex@8.0.0", "", {}, "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A=="], + + "yargs/string-width/is-fullwidth-code-point": ["is-fullwidth-code-point@3.0.0", "", {}, "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg=="], + + "yargs/string-width/strip-ansi": ["strip-ansi@6.0.1", "", { "dependencies": { "ansi-regex": "^5.0.1" } }, "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A=="], + + "yargs/string-width/strip-ansi/ansi-regex": ["ansi-regex@5.0.1", "", {}, "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ=="], + } +} diff --git a/packages/runtime/package.json b/packages/runtime/package.json index 8e58405..aadc74d 100644 --- a/packages/runtime/package.json +++ b/packages/runtime/package.json @@ -8,15 +8,16 @@ "forge-runtime": "./dist/runtime.mjs" }, "scripts": { - "build": "bun build ./src/index.ts --outdir ./dist --target bun", + "build": "bun build ./src/index.ts --outfile ./dist/runtime.mjs --target bun", "test": "bun test", "typecheck": "tsc --noEmit" }, "dependencies": { "@agent-forge/core": "workspace:*", "@agent-forge/tools-core": "workspace:*", - "@anthropic-ai/sdk": "^0.32.0", + "@ai-sdk/openai": "^1.0.0", "@modelcontextprotocol/sdk": "^1.0.0", + "ai": "^4.0.0", "yaml": "^2.6.0" } } diff --git a/packages/runtime/src/index.ts b/packages/runtime/src/index.ts index cf0d594..d473b05 100644 --- a/packages/runtime/src/index.ts +++ b/packages/runtime/src/index.ts @@ -1,8 +1,48 @@ // @agent-forge/runtime — entry point // -// Runs inside the Docker container. Reads its AGENT.md, boots the query loop, -// connects to claude-presence MCP, streams events to the host via stdio. +// Reads a prompt from stdin, calls an OpenAI-compatible LLM endpoint +// via the Vercel AI SDK, writes the assistant text to stdout. // -// Phase POC : not implemented yet. +// Defaults to a local MLX server (mlx_lm.server) so dev runs cost nothing. +// To use OpenAI cloud or another OpenAI-compatible provider : +// FORGE_BASE_URL=https://api.openai.com/v1 FORGE_API_KEY=sk-... -export {} +import { createOpenAI } from '@ai-sdk/openai' +import { generateText } from 'ai' + +const BASE_URL = process.env.FORGE_BASE_URL ?? 'http://127.0.0.1:8080/v1' +const API_KEY = process.env.FORGE_API_KEY ?? 'not-needed' +const MODEL = process.env.FORGE_MODEL ?? 'mlx-community/Llama-3.2-3B-Instruct-4bit' +const MAX_TOKENS = Number(process.env.FORGE_MAX_TOKENS ?? '1024') + +async function readStdin(): Promise { + const chunks: Uint8Array[] = [] + for await (const chunk of process.stdin) { + chunks.push(chunk as Uint8Array) + } + return Buffer.concat(chunks).toString('utf8').trim() +} + +async function main(): Promise { + const prompt = await readStdin() + if (!prompt) { + console.error('✗ no prompt received on stdin') + process.exit(1) + } + + const provider = createOpenAI({ baseURL: BASE_URL, apiKey: API_KEY }) + + const { text } = await generateText({ + model: provider(MODEL), + prompt, + maxTokens: MAX_TOKENS, + }) + + process.stdout.write(`${text}\n`) +} + +main().catch((err) => { + const msg = err instanceof Error ? err.message : String(err) + console.error(`✗ runtime error: ${msg}`) + process.exit(1) +}) From ebab0c4650291af86c8de4ed70fa6831609db428 Mon Sep 17 00:00:00 2001 From: Georges Garnier Date: Sun, 26 Apr 2026 13:14:12 +0200 Subject: [PATCH 03/14] feat(cli): add poc:p1 script orchestrating runtime in docker MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Adds packages/cli/src/poc-p1.ts and a root npm script `poc:p1` that demonstrates the full P1 round-trip : spawn `docker run --rm -i agent-forge/base:latest`, mount the runtime bundle as a read-only volume, pipe a hardcoded prompt to its stdin, capture stdout, and let `--rm` reclaim the container. The container is pointed at the host MLX server via `FORGE_BASE_URL=http://host.docker.internal:8080/v1` so the round-trip costs nothing. Implementation note: shells out to the docker CLI rather than using the Engine API (dockerode hangs on `attach` upgrade under Bun). The CLI is plenty for P1 — we will switch to the API later when finer-grained control is needed. Closes #3. --- bun.lock | 3 ++ package.json | 4 +++ packages/cli/package.json | 6 ++-- packages/cli/src/poc-p1.ts | 74 ++++++++++++++++++++++++++++++++++++++ 4 files changed, 84 insertions(+), 3 deletions(-) create mode 100644 packages/cli/src/poc-p1.ts diff --git a/bun.lock b/bun.lock index bbf93d2..ae44298 100644 --- a/bun.lock +++ b/bun.lock @@ -66,6 +66,9 @@ }, }, }, + "trustedDependencies": [ + "@biomejs/biome", + ], "packages": { "@agent-forge/cli": ["@agent-forge/cli@workspace:packages/cli"], diff --git a/package.json b/package.json index 09d5ee2..7ba5838 100644 --- a/package.json +++ b/package.json @@ -20,6 +20,7 @@ "format": "biome format --write .", "typecheck": "bun run --filter '*' typecheck", "mockup": "node demo-sprites/forge-mockup-v3.mjs", + "poc:p1": "bun run packages/cli/src/poc-p1.ts", "hooks:install": "bash scripts/install-hooks.sh", "prepare": "bash scripts/install-hooks.sh" }, @@ -40,5 +41,8 @@ "anthropic", "mcp", "orchestration" + ], + "trustedDependencies": [ + "@biomejs/biome" ] } diff --git a/packages/cli/package.json b/packages/cli/package.json index 223b2b0..71872aa 100644 --- a/packages/cli/package.json +++ b/packages/cli/package.json @@ -16,10 +16,10 @@ "dependencies": { "@agent-forge/core": "workspace:*", "@agent-forge/tools-core": "workspace:*", - "ink": "^5.0.0", - "react": "^18.3.0", + "chalk": "^5.3.0", "commander": "^12.1.0", - "chalk": "^5.3.0" + "ink": "^5.0.0", + "react": "^18.3.0" }, "devDependencies": { "@types/react": "^18.3.0" diff --git a/packages/cli/src/poc-p1.ts b/packages/cli/src/poc-p1.ts new file mode 100644 index 0000000..6ebe107 --- /dev/null +++ b/packages/cli/src/poc-p1.ts @@ -0,0 +1,74 @@ +// @agent-forge/cli — P1 proof of concept +// +// Smallest end-to-end demo of the architecture : +// host orchestrator → docker run base image → mount runtime via volume → +// pipe prompt to stdin → capture stdout → container removed (--rm). +// +// Requires : +// - Docker daemon running +// - Image agent-forge/base:latest built (see scripts/docker/build-base.sh) +// - packages/runtime/dist/runtime.mjs built (cd packages/runtime && bun run build) +// - An OpenAI-compatible LLM endpoint reachable from the container +// (defaults to MLX at host.docker.internal:8080, see P1.2) +// +// Implementation note : we shell out to the docker CLI rather than using the +// Docker Engine API (e.g. dockerode) because dockerode's `attach` upgrade +// hangs under Bun. The CLI is plenty for P1 — we will switch to the API later +// when we need finer-grained control (resource limits, healthchecks, …). + +import { spawn } from 'node:child_process' + +const PROMPT = 'Write a haiku about Docker' +const IMAGE = 'agent-forge/base:latest' +const RUNTIME_DIST = new URL('../../runtime/dist', import.meta.url).pathname + +async function main(): Promise { + const args = [ + 'run', + '--rm', + '-i', + '-v', + `${RUNTIME_DIST}:/runtime:ro`, + '-e', + 'FORGE_BASE_URL=http://host.docker.internal:8080/v1', + IMAGE, + 'node', + '/runtime/runtime.mjs', + ] + + const child = spawn('docker', args, { + stdio: ['pipe', 'pipe', 'pipe'], + }) + + const stdoutChunks: Buffer[] = [] + const stderrChunks: Buffer[] = [] + child.stdout.on('data', (c: Buffer) => stdoutChunks.push(c)) + child.stderr.on('data', (c: Buffer) => stderrChunks.push(c)) + + child.stdin.write(PROMPT) + child.stdin.end() + + const exitCode: number = await new Promise((res, rej) => { + child.on('error', rej) + child.on('close', (code) => res(code ?? 1)) + }) + + const stdout = Buffer.concat(stdoutChunks).toString('utf8').trim() + const stderr = Buffer.concat(stderrChunks).toString('utf8').trim() + + if (stderr) { + process.stderr.write(`${stderr}\n`) + } + if (stdout) { + process.stdout.write(`${stdout}\n`) + } + if (exitCode !== 0) { + process.exit(exitCode) + } +} + +main().catch((err) => { + const msg = err instanceof Error ? err.message : String(err) + console.error(`✗ poc-p1 error: ${msg}`) + process.exit(1) +}) From b82e62cc46550ba8e33af68d26180f70ced30c08 Mon Sep 17 00:00:00 2001 From: Georges Garnier Date: Sun, 26 Apr 2026 16:18:37 +0200 Subject: [PATCH 04/14] feat(cli): add preflight checks to poc:p1 MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Before launching the container, check three preconditions and surface a clear actionable message on failure : 1. Docker daemon is reachable (`docker info`) 2. Image agent-forge/base:latest is built locally 3. Runtime bundle packages/runtime/dist/runtime.mjs exists Each failure prints `✗ ` followed by the exact remediation command and exits 1. The user never has to guess what to fix. LLM-side errors (endpoint unreachable, model load failure, etc.) are already surfaced verbatim by the runtime via stderr — no extra plumbing needed here. Closes #4. --- packages/cli/src/poc-p1.ts | 41 +++++++++++++++++++++++++++++++++++++- 1 file changed, 40 insertions(+), 1 deletion(-) diff --git a/packages/cli/src/poc-p1.ts b/packages/cli/src/poc-p1.ts index 6ebe107..47b70d5 100644 --- a/packages/cli/src/poc-p1.ts +++ b/packages/cli/src/poc-p1.ts @@ -16,13 +16,52 @@ // hangs under Bun. The CLI is plenty for P1 — we will switch to the API later // when we need finer-grained control (resource limits, healthchecks, …). -import { spawn } from 'node:child_process' +import { spawn, spawnSync } from 'node:child_process' +import { existsSync } from 'node:fs' +import { join } from 'node:path' const PROMPT = 'Write a haiku about Docker' const IMAGE = 'agent-forge/base:latest' const RUNTIME_DIST = new URL('../../runtime/dist', import.meta.url).pathname +const RUNTIME_BUNDLE = join(RUNTIME_DIST, 'runtime.mjs') + +function fail(message: string): never { + process.stderr.write(`✗ ${message}\n`) + process.exit(1) +} + +function preflight(): void { + // 1. Docker daemon reachable + const docker = spawnSync('docker', ['info'], { stdio: 'ignore' }) + if (docker.error || docker.status !== 0) { + fail( + 'Docker daemon is not reachable.\n' + + ' Start Docker Desktop (or `colima start`) and try again.', + ) + } + + // 2. Base image present locally (we cannot `docker pull` because the image + // is not published to a registry yet — it is built locally). + const image = spawnSync('docker', ['image', 'inspect', IMAGE], { stdio: 'ignore' }) + if (image.status !== 0) { + fail( + `Image ${IMAGE} is not built locally.\n` + + ' Run: bash scripts/docker/build-base.sh', + ) + } + + // 3. Runtime bundle built + if (!existsSync(RUNTIME_BUNDLE)) { + fail( + `Runtime bundle missing: ${RUNTIME_BUNDLE}\n` + + ' Run: cd packages/runtime && bun run build', + ) + } +} async function main(): Promise { + preflight() + const args = [ 'run', '--rm', From 4a9a09909d7d90425017909351d04039bd1fa89f Mon Sep 17 00:00:00 2001 From: Georges Garnier Date: Sun, 26 Apr 2026 17:00:32 +0200 Subject: [PATCH 05/14] feat(cli): always cleanup container on signal, error or timeout Guarantees that no container survives a poc:p1 run, no matter how it ends : - explicit --name agent-forge-poc- so we can target it after a signal even if the docker client child has not yet propagated SIGTERM - SIGINT and SIGTERM handlers force-remove the container and exit 130 - 60s hard timeout (overridable via FORGE_POC_TIMEOUT_MS) kills the container and exits 1 with a clear message - try/finally wraps the spawn so any exception still triggers cleanup - FORGE_BASE_URL is now overridable from the host (default unchanged) Verified by sending SIGINT mid-run (no leftover) and by pointing FORGE_BASE_URL at a nc :9999 sink (timeout fires, no leftover). Closes #5. --- packages/cli/src/poc-p1.ts | 75 +++++++++++++++++++++++++++----------- 1 file changed, 54 insertions(+), 21 deletions(-) diff --git a/packages/cli/src/poc-p1.ts b/packages/cli/src/poc-p1.ts index 47b70d5..de6b246 100644 --- a/packages/cli/src/poc-p1.ts +++ b/packages/cli/src/poc-p1.ts @@ -24,6 +24,9 @@ const PROMPT = 'Write a haiku about Docker' const IMAGE = 'agent-forge/base:latest' const RUNTIME_DIST = new URL('../../runtime/dist', import.meta.url).pathname const RUNTIME_BUNDLE = join(RUNTIME_DIST, 'runtime.mjs') +const CONTAINER_NAME = `agent-forge-poc-${process.pid}` +const TIMEOUT_MS = Number(process.env.FORGE_POC_TIMEOUT_MS ?? '60000') +const RUNTIME_BASE_URL = process.env.FORGE_BASE_URL ?? 'http://host.docker.internal:8080/v1' function fail(message: string): never { process.stderr.write(`✗ ${message}\n`) @@ -59,6 +62,11 @@ function preflight(): void { } } +function forceRemoveContainer(): void { + // Best-effort sync cleanup. Used in signal handlers and on timeout. + spawnSync('docker', ['rm', '-f', CONTAINER_NAME], { stdio: 'ignore' }) +} + async function main(): Promise { preflight() @@ -66,10 +74,12 @@ async function main(): Promise { 'run', '--rm', '-i', + '--name', + CONTAINER_NAME, '-v', `${RUNTIME_DIST}:/runtime:ro`, '-e', - 'FORGE_BASE_URL=http://host.docker.internal:8080/v1', + `FORGE_BASE_URL=${RUNTIME_BASE_URL}`, IMAGE, 'node', '/runtime/runtime.mjs', @@ -79,30 +89,53 @@ async function main(): Promise { stdio: ['pipe', 'pipe', 'pipe'], }) - const stdoutChunks: Buffer[] = [] - const stderrChunks: Buffer[] = [] - child.stdout.on('data', (c: Buffer) => stdoutChunks.push(c)) - child.stderr.on('data', (c: Buffer) => stderrChunks.push(c)) + // Cleanup on Ctrl+C / SIGTERM. Also kill the child docker client. + const onSignal = (signal: NodeJS.Signals) => { + forceRemoveContainer() + child.kill('SIGTERM') + process.stderr.write(`\n✗ interrupted by ${signal}\n`) + process.exit(130) // 128 + SIGINT (2) + } + process.on('SIGINT', onSignal) + process.on('SIGTERM', onSignal) - child.stdin.write(PROMPT) - child.stdin.end() + // Hard timeout : kill the container if the runtime takes too long. + const timeout = setTimeout(() => { + forceRemoveContainer() + child.kill('SIGTERM') + process.stderr.write(`✗ runtime timeout after ${TIMEOUT_MS / 1000}s\n`) + process.exit(1) + }, TIMEOUT_MS) - const exitCode: number = await new Promise((res, rej) => { - child.on('error', rej) - child.on('close', (code) => res(code ?? 1)) - }) + try { + const stdoutChunks: Buffer[] = [] + const stderrChunks: Buffer[] = [] + child.stdout.on('data', (c: Buffer) => stdoutChunks.push(c)) + child.stderr.on('data', (c: Buffer) => stderrChunks.push(c)) - const stdout = Buffer.concat(stdoutChunks).toString('utf8').trim() - const stderr = Buffer.concat(stderrChunks).toString('utf8').trim() + child.stdin.write(PROMPT) + child.stdin.end() - if (stderr) { - process.stderr.write(`${stderr}\n`) - } - if (stdout) { - process.stdout.write(`${stdout}\n`) - } - if (exitCode !== 0) { - process.exit(exitCode) + const exitCode: number = await new Promise((res, rej) => { + child.on('error', rej) + child.on('close', (code) => res(code ?? 1)) + }) + + const stdout = Buffer.concat(stdoutChunks).toString('utf8').trim() + const stderr = Buffer.concat(stderrChunks).toString('utf8').trim() + + if (stderr) { + process.stderr.write(`${stderr}\n`) + } + if (stdout) { + process.stdout.write(`${stdout}\n`) + } + if (exitCode !== 0) { + process.exit(exitCode) + } + } finally { + clearTimeout(timeout) + forceRemoveContainer() } } From d4799a61ff4f730f6b1cf7c2115c3052eeb40b85 Mon Sep 17 00:00:00 2001 From: Georges Garnier Date: Sun, 26 Apr 2026 17:03:30 +0200 Subject: [PATCH 06/14] docs(readme): add Try P1 section and bump status to P1 done Adds a Try P1 walkthrough under Try the mockup in both English and French READMEs : prerequisites, default MLX path (free, local on Apple Silicon), alternative OpenAI cloud path, and the preflight troubleshooting messages. Status badge moves from "POC" to "P1 done", and the status sentence now reflects that P1 is runnable end-to-end with P2 next. Closes #6. --- README.fr.md | 84 +++++++++++++++++++++++++++++++++++++++++++++++++--- README.md | 84 +++++++++++++++++++++++++++++++++++++++++++++++++--- 2 files changed, 160 insertions(+), 8 deletions(-) diff --git a/README.fr.md b/README.fr.md index 2f5c015..7a5453c 100644 --- a/README.fr.md +++ b/README.fr.md @@ -7,7 +7,7 @@ **Forgez, lancez et orchestrez des agents LLM en sandbox.** [![License: Apache 2.0](https://img.shields.io/badge/License-Apache_2.0-blue.svg)](./LICENSE) - ![Status: POC](https://img.shields.io/badge/status-POC-orange) + ![Status: P1 done](https://img.shields.io/badge/status-P1%20done-green) ![Stack: TypeScript + Bun](https://img.shields.io/badge/stack-TypeScript_+_Bun-3178c6) 🇫🇷 Version française · [🇬🇧 English version](./README.md) @@ -16,7 +16,7 @@ --- -> 🚧 **Statut — Phase de conception.** L'architecture est posée, le mockup interactif est fonctionnel. **Pas encore de code de production.** Le premier jalon exécutable (P1 — *Hello agent in Docker*) est le prochain livrable. Mettez une ⭐ pour suivre l'évolution. +> 🚧 **Statut — POC précoce.** Architecture posée, mockup interactif fonctionnel, **premier jalon livré (P1 — *Hello agent in Docker*) : un script orchestre un container Docker exécutant un round-trip LLM de bout en bout.** Prochain jalon : P2 — CLI conversationnelle. Mettez une ⭐ pour suivre l'évolution. ## Qu'est-ce qu'Agent Forge ? @@ -28,9 +28,9 @@ Une CLI conversationnelle qui vous permet de **décrire** un projet logiciel en ## Statut -🚧 **Phase POC.** Phase de conception active. **Pas encore de code de production.** +🚧 **Phase POC, P1 terminé.** -Un mockup interactif complet existe (`demo-sprites/`), et l'architecture est entièrement préparée. Le premier jalon exécutable (P1 — *Hello agent in Docker*) arrive ensuite. +Un mockup interactif complet existe (`demo-sprites/`), l'architecture est entièrement préparée, et le premier jalon (P1 — *Hello agent in Docker*) tourne de bout en bout : un script orchestre un container Docker, lui envoie un prompt, et affiche la réponse du LLM. Prochain jalon : P2 — CLI conversationnelle. ## Tester le mockup @@ -42,6 +42,82 @@ Parcourt les 7 écrans du produit : splash, welcome, chat, mission control, focu Appuyez sur `SPACE` pour avancer, `B` pour reculer, `R` pour redémarrer. +## Tester P1 — *Hello agent in Docker* + +Le premier jalon exécutable : un script lance un container Docker, y monte un runtime Node minimal, lui envoie un prompt via stdin, et affiche la réponse du LLM. + +### Prérequis + +- **Docker** lancé (Docker Desktop, OrbStack ou `colima`) +- **Bun** ≥ 1.1 — `curl -fsSL https://bun.sh/install | bash` +- **Un endpoint LLM compatible OpenAI joignable depuis le container.** Le défaut pointe sur un serveur [MLX](https://github.com/ml-explore/mlx) local, gratuit sur Mac Apple Silicon. Tout autre endpoint compatible OpenAI (Ollama, OpenAI cloud, vLLM, …) fonctionne en surchargeant deux variables d'env — voir plus bas. + +### Voie par défaut : serveur MLX local (Apple Silicon, gratuit) + +```bash +# 1. Installer MLX-LM dans un venv isolé (une seule fois) +python3 -m venv ~/.agent-forge/mlx-venv +~/.agent-forge/mlx-venv/bin/pip install mlx-lm + +# 2. Télécharger un petit modèle instruct (~2 Go, une seule fois) +~/.agent-forge/mlx-venv/bin/hf download mlx-community/Llama-3.2-3B-Instruct-4bit + +# 3. Démarrer le serveur HTTP MLX (laissez-le tourner dans un terminal) +~/.agent-forge/mlx-venv/bin/mlx_lm.server \ + --model mlx-community/Llama-3.2-3B-Instruct-4bit \ + --port 8080 +``` + +Dans un autre terminal : + +```bash +# 4. Builder l'image Docker base (une seule fois, ~600 Mo, ~1 min) +bash scripts/docker/build-base.sh + +# 5. Installer les deps JS et builder le bundle runtime +bun install +cd packages/runtime && bun run build && cd - + +# 6. Lancer le round-trip +bun run poc:p1 +``` + +Sortie attendue : + +``` +Containers at sea +Docker's gentle guiding hand +Freedom in the code +``` + +Le container est supprimé automatiquement (`--rm`), même sur Ctrl+C ou timeout. + +### Alternative : OpenAI cloud (ou tout endpoint compatible OpenAI) + +```bash +FORGE_BASE_URL=https://api.openai.com/v1 \ +FORGE_API_KEY=sk-... \ +FORGE_MODEL=gpt-4o-mini \ +bun run poc:p1 +``` + +Même script, endpoint différent. Le runtime est provider-agnostic via le [Vercel AI SDK](https://sdk.vercel.ai). + +### Dépannage + +Le script effectue trois preflight checks et indique exactement quoi corriger : + +``` +✗ Docker daemon is not reachable. + Start Docker Desktop (or `colima start`) and try again. + +✗ Image agent-forge/base:latest is not built locally. + Run: bash scripts/docker/build-base.sh + +✗ Runtime bundle missing: …/packages/runtime/dist/runtime.mjs + Run: cd packages/runtime && bun run build +``` + ## Concept Agent Forge unifie cinq primitives : diff --git a/README.md b/README.md index 6185434..ab06612 100644 --- a/README.md +++ b/README.md @@ -7,7 +7,7 @@ **Forge, run, and orchestrate sandboxed LLM agents.** [![License: Apache 2.0](https://img.shields.io/badge/License-Apache_2.0-blue.svg)](./LICENSE) - ![Status: POC](https://img.shields.io/badge/status-POC-orange) + ![Status: P1 done](https://img.shields.io/badge/status-P1%20done-green) ![Stack: TypeScript + Bun](https://img.shields.io/badge/stack-TypeScript_+_Bun-3178c6) 🇬🇧 English version · [🇫🇷 Version française](./README.fr.md) @@ -16,7 +16,7 @@ --- -> 🚧 **Status — Design phase.** Architecture is complete, the interactive mockup is runnable. **No production code yet.** First runnable milestone (P1 — *Hello agent in Docker*) is the next deliverable. Star the repo to follow along. +> 🚧 **Status — Early POC.** Architecture complete, interactive mockup runnable, **first milestone (P1 — *Hello agent in Docker*) lands : a host script orchestrates a Docker container running an LLM round-trip end-to-end.** Next milestone : P2 — Conversational CLI. Star the repo to follow along. ## What is Agent Forge ? @@ -28,9 +28,9 @@ A conversational CLI that lets you **describe** a software project in natural la ## Status -🚧 **Phase POC.** Active design phase. **No production code yet.** +🚧 **Phase POC, P1 done.** -A complete interactive mockup exists (`demo-sprites/`), and the architecture is fully scaffolded. The first runnable milestone (P1 — *Hello agent in Docker*) comes next. +A complete interactive mockup exists (`demo-sprites/`), the architecture is fully scaffolded, and the first milestone (P1 — *Hello agent in Docker*) is runnable end-to-end : a host script spins up a Docker container, feeds it a prompt, and prints the LLM response. Next : P2 — Conversational CLI. ## Try the mockup @@ -42,6 +42,82 @@ Walks through the 7 screens of the product : splash, welcome, chat, mission cont Press `SPACE` to advance, `B` to go back, `R` to restart. +## Try P1 — *Hello agent in Docker* + +The first runnable milestone : a host script launches a Docker container, mounts a minimal Node runtime inside it, pipes a prompt through stdin, and prints the LLM response back. + +### Prerequisites + +- **Docker** running (Docker Desktop, OrbStack, or `colima`) +- **Bun** ≥ 1.1 — `curl -fsSL https://bun.sh/install | bash` +- **An OpenAI-compatible LLM endpoint reachable from the container.** The default points at a local [MLX](https://github.com/ml-explore/mlx) server, which is free to run on Apple Silicon Macs. Any other OpenAI-compatible endpoint (Ollama, OpenAI cloud, vLLM, …) works by overriding two env vars — see below. + +### Default path : local MLX server (Apple Silicon, free) + +```bash +# 1. Install MLX-LM in an isolated venv (one-time) +python3 -m venv ~/.agent-forge/mlx-venv +~/.agent-forge/mlx-venv/bin/pip install mlx-lm + +# 2. Pull a small instruction-tuned model (~2 GB, one-time) +~/.agent-forge/mlx-venv/bin/hf download mlx-community/Llama-3.2-3B-Instruct-4bit + +# 3. Start the MLX HTTP server (keep this running in a terminal) +~/.agent-forge/mlx-venv/bin/mlx_lm.server \ + --model mlx-community/Llama-3.2-3B-Instruct-4bit \ + --port 8080 +``` + +In another terminal : + +```bash +# 4. Build the base Docker image (one-time, ~600 MB, ~1 min) +bash scripts/docker/build-base.sh + +# 5. Install JS deps and build the runtime bundle +bun install +cd packages/runtime && bun run build && cd - + +# 6. Run the round-trip +bun run poc:p1 +``` + +Expected output : + +``` +Containers at sea +Docker's gentle guiding hand +Freedom in the code +``` + +The container is removed automatically (`--rm`), even on Ctrl+C or timeout. + +### Alternative : OpenAI cloud (or any OpenAI-compatible endpoint) + +```bash +FORGE_BASE_URL=https://api.openai.com/v1 \ +FORGE_API_KEY=sk-... \ +FORGE_MODEL=gpt-4o-mini \ +bun run poc:p1 +``` + +Same script, different endpoint. The runtime is provider-agnostic via the [Vercel AI SDK](https://sdk.vercel.ai). + +### Troubleshooting + +The script runs three preflight checks and tells you exactly what to fix : + +``` +✗ Docker daemon is not reachable. + Start Docker Desktop (or `colima start`) and try again. + +✗ Image agent-forge/base:latest is not built locally. + Run: bash scripts/docker/build-base.sh + +✗ Runtime bundle missing: …/packages/runtime/dist/runtime.mjs + Run: cd packages/runtime && bun run build +``` + ## Concept Agent Forge unifies five primitives : From db91b81ca4688adb31c5a1dacbe5e447ffbc540b Mon Sep 17 00:00:00 2001 From: Georges Garnier Date: Sun, 26 Apr 2026 18:33:27 +0200 Subject: [PATCH 07/14] feat(cli): ink TUI skeleton with splash, language picker, and welcome MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Bootstraps the conversational REPL with a permanent two-zone layout : - Splash on top (logo, tagline, animated preflight checks). Stays visible as a session header. On first run, an inline language picker appears below the checks (English / Français, side-by-side, arrow keys or E/F shortcuts). - Welcome pinned to the bottom of the terminal (header bar, big question, suggestions, prompt input, footer hints). Empty space in the middle will host the transcript / build progress later. User language preference is persisted to ~/.agent-forge/config.json. All UI strings are bilingual via a small i18n table indexed by `useT()`. The bin script `forge` (root package.json) runs the TUI directly with bun, no link required for development. Closes #7. --- bun.lock | 3 + package.json | 1 + packages/cli/package.json | 1 + packages/cli/src/components/App.tsx | 34 +++++ packages/cli/src/components/Footer.tsx | 43 ++++++ packages/cli/src/components/Header.tsx | 42 ++++++ .../cli/src/components/LanguagePicker.tsx | 78 ++++++++++ packages/cli/src/components/Splash.tsx | 80 +++++++++++ packages/cli/src/components/Welcome.tsx | 102 +++++++++++++ packages/cli/src/config/store.ts | 33 +++++ packages/cli/src/hooks/usePreflight.ts | 84 +++++++++++ packages/cli/src/i18n/LanguageContext.tsx | 53 +++++++ packages/cli/src/i18n/strings.ts | 134 ++++++++++++++++++ packages/cli/src/index.ts | 9 -- packages/cli/src/index.tsx | 15 ++ packages/cli/src/theme/colors.ts | 33 +++++ packages/cli/src/theme/logo.ts | 21 +++ 17 files changed, 757 insertions(+), 9 deletions(-) create mode 100644 packages/cli/src/components/App.tsx create mode 100644 packages/cli/src/components/Footer.tsx create mode 100644 packages/cli/src/components/Header.tsx create mode 100644 packages/cli/src/components/LanguagePicker.tsx create mode 100644 packages/cli/src/components/Splash.tsx create mode 100644 packages/cli/src/components/Welcome.tsx create mode 100644 packages/cli/src/config/store.ts create mode 100644 packages/cli/src/hooks/usePreflight.ts create mode 100644 packages/cli/src/i18n/LanguageContext.tsx create mode 100644 packages/cli/src/i18n/strings.ts delete mode 100644 packages/cli/src/index.ts create mode 100644 packages/cli/src/index.tsx create mode 100644 packages/cli/src/theme/colors.ts create mode 100644 packages/cli/src/theme/logo.ts diff --git a/bun.lock b/bun.lock index ae44298..b7497a6 100644 --- a/bun.lock +++ b/bun.lock @@ -22,6 +22,7 @@ "chalk": "^5.3.0", "commander": "^12.1.0", "ink": "^5.0.0", + "ink-text-input": "^6.0.0", "react": "^18.3.0", }, "devDependencies": { @@ -352,6 +353,8 @@ "ink": ["ink@5.2.1", "", { "dependencies": { "@alcalzone/ansi-tokenize": "^0.1.3", "ansi-escapes": "^7.0.0", "ansi-styles": "^6.2.1", "auto-bind": "^5.0.1", "chalk": "^5.3.0", "cli-boxes": "^3.0.0", "cli-cursor": "^4.0.0", "cli-truncate": "^4.0.0", "code-excerpt": "^4.0.0", "es-toolkit": "^1.22.0", "indent-string": "^5.0.0", "is-in-ci": "^1.0.0", "patch-console": "^2.0.0", "react-reconciler": "^0.29.0", "scheduler": "^0.23.0", "signal-exit": "^3.0.7", "slice-ansi": "^7.1.0", "stack-utils": "^2.0.6", "string-width": "^7.2.0", "type-fest": "^4.27.0", "widest-line": "^5.0.0", "wrap-ansi": "^9.0.0", "ws": "^8.18.0", "yoga-layout": "~3.2.1" }, "peerDependencies": { "@types/react": ">=18.0.0", "react": ">=18.0.0", "react-devtools-core": "^4.19.1" }, "optionalPeers": ["@types/react", "react-devtools-core"] }, "sha512-BqcUyWrG9zq5HIwW6JcfFHsIYebJkWWb4fczNah1goUO0vv5vneIlfwuS85twyJ5hYR/y18FlAYUxrO9ChIWVg=="], + "ink-text-input": ["ink-text-input@6.0.0", "", { "dependencies": { "chalk": "^5.3.0", "type-fest": "^4.18.2" }, "peerDependencies": { "ink": ">=5", "react": ">=18" } }, "sha512-Fw64n7Yha5deb1rHY137zHTAbSTNelUKuB5Kkk2HACXEtwIHBCf9OH2tP/LQ9fRYTl1F0dZgbW0zPnZk6FA9Lw=="], + "ip-address": ["ip-address@10.1.0", "", {}, "sha512-XXADHxXmvT9+CRxhXg56LJovE+bmWnEWB78LB83VZTprKTmaC5QfruXocxzTZ2Kl0DNwKuBdlIhjL8LeY8Sf8Q=="], "ipaddr.js": ["ipaddr.js@1.9.1", "", {}, "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g=="], diff --git a/package.json b/package.json index 7ba5838..1ed794c 100644 --- a/package.json +++ b/package.json @@ -20,6 +20,7 @@ "format": "biome format --write .", "typecheck": "bun run --filter '*' typecheck", "mockup": "node demo-sprites/forge-mockup-v3.mjs", + "forge": "bun run packages/cli/src/index.tsx", "poc:p1": "bun run packages/cli/src/poc-p1.ts", "hooks:install": "bash scripts/install-hooks.sh", "prepare": "bash scripts/install-hooks.sh" diff --git a/packages/cli/package.json b/packages/cli/package.json index 71872aa..a9d4809 100644 --- a/packages/cli/package.json +++ b/packages/cli/package.json @@ -19,6 +19,7 @@ "chalk": "^5.3.0", "commander": "^12.1.0", "ink": "^5.0.0", + "ink-text-input": "^6.0.0", "react": "^18.3.0" }, "devDependencies": { diff --git a/packages/cli/src/components/App.tsx b/packages/cli/src/components/App.tsx new file mode 100644 index 0000000..cbeb3a7 --- /dev/null +++ b/packages/cli/src/components/App.tsx @@ -0,0 +1,34 @@ +// Top-level layout : permanent two-zone screen. +// - Splash on top : logo, checks, language picker (first run). +// Stays visible as a session header — will be cleared later (P3+) when +// the build phase begins (mission control replaces this view). +// - Welcome PINNED to the bottom of the terminal : header, question, +// suggestions, prompt, footer. The middle is empty space (will host the +// transcript / build progress in P2.2+ / P3+). + +import { Box } from 'ink' +import React from 'react' +import { useLanguage } from '../i18n/LanguageContext.tsx' +import { Splash } from './Splash.tsx' +import { Welcome } from './Welcome.tsx' + +export function App(): React.JSX.Element { + const { lang } = useLanguage() + const rows = process.stdout.rows ?? 30 + + return ( + + + + {lang ? ( + { + // P2.2 will route this to the LLM. For P2.1 we just echo to + // stderr so we can confirm the input pipeline works end-to-end. + process.stderr.write(`[echo] ${prompt}\n`) + }} + /> + ) : null} + + ) +} diff --git a/packages/cli/src/components/Footer.tsx b/packages/cli/src/components/Footer.tsx new file mode 100644 index 0000000..3a07fc9 --- /dev/null +++ b/packages/cli/src/components/Footer.tsx @@ -0,0 +1,43 @@ +// Bottom bar: keyboard hints on the left, screen info on the right. +// Mirrors drawFooter() from demo-sprites/forge-mockup-v3.mjs. + +import { Box, Text } from 'ink' +import React from 'react' +import { C } from '../theme/colors.ts' + +export type Hint = { key: string; label: string } + +export function Footer({ + hints, + info, +}: { + hints: Hint[] + info?: string +}): React.JSX.Element { + return ( + + + {'─'.repeat(process.stdout.columns ?? 80)} + + + + {hints.map((h, i) => ( + + {i > 0 ? {' '} : null} + + {h.key} + + + {h.label} + + ))} + + {info ? ( + + {info} + + ) : null} + + + ) +} diff --git a/packages/cli/src/components/Header.tsx b/packages/cli/src/components/Header.tsx new file mode 100644 index 0000000..8d693e3 --- /dev/null +++ b/packages/cli/src/components/Header.tsx @@ -0,0 +1,42 @@ +// Top bar: "▌▌ AGENT FORGE ▐▐" left, optional center label, optional right info. +// Mirrors drawHeader() from demo-sprites/forge-mockup-v3.mjs. + +import { Box, Text } from 'ink' +import React from 'react' +import { C } from '../theme/colors.ts' + +const VERSION = '0.0.0' + +export function Header({ + label, + info, +}: { + label?: string + info?: string +}): React.JSX.Element { + return ( + + + + + {' ▌▌ AGENT FORGE ▐▐ '} + + + v{VERSION} + + {label ? ( + {` ${label}`} + ) : null} + + {info ? ( + + {info} + + ) : null} + + + {'─'.repeat(process.stdout.columns ?? 80)} + + + ) +} diff --git a/packages/cli/src/components/LanguagePicker.tsx b/packages/cli/src/components/LanguagePicker.tsx new file mode 100644 index 0000000..6be7294 --- /dev/null +++ b/packages/cli/src/components/LanguagePicker.tsx @@ -0,0 +1,78 @@ +// Inline language picker block — designed to be embedded inside Splash, NOT +// rendered as its own full screen. Bilingual (EN + FR) since we do not yet +// know which language the user wants. Selection is persisted by the parent. + +import { Box, Text, useInput, useStdin } from 'ink' +import React, { useState } from 'react' +import { type Lang } from '../config/store.ts' +import { useT } from '../i18n/LanguageContext.tsx' +import { C } from '../theme/colors.ts' + +const OPTIONS: ReadonlyArray<{ lang: Lang; label: string }> = [ + { lang: 'en', label: 'English' }, + { lang: 'fr', label: 'Français' }, +] + +export function LanguagePicker({ + onPick, +}: { + onPick: (lang: Lang) => void +}): React.JSX.Element { + const t = useT() + const { isRawModeSupported } = useStdin() + const [index, setIndex] = useState(0) + + useInput( + (input, key) => { + if (key.leftArrow) { + setIndex((i) => (i - 1 + OPTIONS.length) % OPTIONS.length) + } else if (key.rightArrow) { + setIndex((i) => (i + 1) % OPTIONS.length) + } else if (input === 'e' || input === 'E') { + setIndex(0) + } else if (input === 'f' || input === 'F') { + setIndex(1) + } else if (key.return) { + const picked = OPTIONS[index] + if (picked) onPick(picked.lang) + } + }, + { isActive: isRawModeSupported }, + ) + + return ( + + + + {t('langPickerTitleEN')} + + + {t('langPickerTitleFR')} + + + + + {OPTIONS.map((opt, i) => { + const selected = i === index + return ( + + + {selected ? '▸ ' : ' '} + + + {opt.label} + + + ) + })} + + + + + [←→] {t('langPickerHintNavigate')} [E/F] shortcut [⏎]{' '} + {t('langPickerHintSelect')} + + + + ) +} diff --git a/packages/cli/src/components/Splash.tsx b/packages/cli/src/components/Splash.tsx new file mode 100644 index 0000000..529246d --- /dev/null +++ b/packages/cli/src/components/Splash.tsx @@ -0,0 +1,80 @@ +// Boot splash : centered ASCII logo, tagline, animated preflight checks, +// and (on first run) an inline language picker below the checks. +// +// Stays mounted as a session header — App.tsx renders Welcome BELOW it. +// Will be cleared in P3+ when the build phase replaces this view. +// +// Mirrors screenSplash() from demo-sprites/forge-mockup-v3.mjs. + +import { Box, Text } from 'ink' +import React from 'react' +import { type Lang } from '../config/store.ts' +import { usePreflight } from '../hooks/usePreflight.ts' +import { useLanguage } from '../i18n/LanguageContext.tsx' +import { C } from '../theme/colors.ts' +import { LOGO_AGENT, LOGO_FORGE } from '../theme/logo.ts' +import { LanguagePicker } from './LanguagePicker.tsx' + +const VERSION = '0.0.0' + +export function Splash(): React.JSX.Element { + const { checks, allDone } = usePreflight() + const { lang, setLang } = useLanguage() + + const handlePick = (picked: Lang): void => { + setLang(picked) + } + + return ( + + + {LOGO_AGENT.map((line, i) => ( + + {line} + + ))} + {LOGO_FORGE.map((line, i) => ( + + {line} + + ))} + + + + + Forge, run, and orchestrate sandboxed LLM agents + + + v{VERSION} · by @garniergeorges · Apache 2.0 + + + + + {checks.map((c) => { + const symbol = + c.status === 'ok' ? '✓' : c.status === 'fail' ? '✗' : '·' + const symbolColor = + c.status === 'ok' ? C.green : c.status === 'fail' ? C.red : C.grey + const trail = + c.status === 'pending' ? '' : c.status === 'running' ? '...' : '' + return ( + + {symbol} + + {' '} + {c.label} + {trail} + + + ) + })} + + + {allDone && lang === null ? ( + + + + ) : null} + + ) +} diff --git a/packages/cli/src/components/Welcome.tsx b/packages/cli/src/components/Welcome.tsx new file mode 100644 index 0000000..3b6fed5 --- /dev/null +++ b/packages/cli/src/components/Welcome.tsx @@ -0,0 +1,102 @@ +// Welcome screen: header, big question, suggestions, prompt, footer. +// Mirrors screenWelcome() from demo-sprites/forge-mockup-v3.mjs. + +import { Box, Text, useStdin } from 'ink' +import TextInput from 'ink-text-input' +import React, { useState } from 'react' +import { useT } from '../i18n/LanguageContext.tsx' +import { C } from '../theme/colors.ts' +import { Footer } from './Footer.tsx' +import { Header } from './Header.tsx' + +const FORGE_MODEL = + process.env.FORGE_MODEL ?? 'mlx-community/Llama-3.2-3B-Instruct-4bit' + +function shortModel(name: string): string { + // "mlx-community/Llama-3.2-3B-Instruct-4bit" → "Llama-3.2-3B-Instruct-4bit" + const slash = name.lastIndexOf('/') + return slash >= 0 ? name.slice(slash + 1) : name +} + +export function Welcome({ + onSubmit, +}: { + onSubmit: (prompt: string) => void +}): React.JSX.Element { + const t = useT() + const { isRawModeSupported } = useStdin() + const [input, setInput] = useState('') + + const handleSubmit = (value: string): void => { + const trimmed = value.trim() + if (!trimmed) return + setInput('') + onSubmit(trimmed) + } + + return ( + +
+ + + + {t('welcomeTitle')} + + + {t('welcomeSubtitle')} + + + + + {t('welcomeSuggestion1')} + + + {t('welcomeSuggestion2')} + + + {t('welcomeSuggestion3')} + + + {t('welcomeSuggestion4')} + + + + + + + {' '} + {'─'.repeat(Math.max(0, (process.stdout.columns ?? 80) - 2))} + + + {' ❯ '} + {isRawModeSupported ? ( + + ) : ( + + {t('welcomeRawModeDisabled')} + + )} + + + + +