From 9be80a16aa1f225e799a0f3023d4ce3590d7b819 Mon Sep 17 00:00:00 2001 From: Volodymyr Vreshch Date: Fri, 12 Jun 2026 20:54:51 +0200 Subject: [PATCH 1/2] feat!: reboot CLI as the agentage Memory client (setup + status) BREAKING CHANGE: all agent-runtime commands and the local daemon are removed. - setup: OAuth 2.1 sign-in (DCR + PKCE, localhost callback); --disconnect, --reauth, --no-browser - status: target, sign-in state (introspection), endpoint health; --json - stateless CLI: ~/.agentage holds credentials only (auth.json, 0600) - deps reduced to chalk + commander + open; package guard test keeps it that way - 50 unit tests, coverage 92/83/85/93 --- CHANGELOG.md | 12 + CLAUDE.md | 95 +- README.md | 201 +- package-lock.json | 1785 +---------------- package.json | 20 +- src/cli.ts | 73 +- src/commands/agents.test.ts | 168 -- src/commands/agents.ts | 122 -- src/commands/completions.test.ts | 57 - src/commands/completions.ts | 126 -- src/commands/config-cmd.test.ts | 94 - src/commands/config-cmd.ts | 82 - src/commands/create.test.ts | 178 -- src/commands/create.ts | 149 -- src/commands/daemon-cmd.test.ts | 64 - src/commands/daemon-cmd.ts | 29 - src/commands/logs.test.ts | 97 - src/commands/logs.ts | 53 - src/commands/machines.test.ts | 123 -- src/commands/machines.ts | 75 - src/commands/projects.test.ts | 311 --- src/commands/projects.ts | 211 -- src/commands/run.test.ts | 533 ----- src/commands/run.ts | 420 ---- src/commands/runs.test.ts | 207 -- src/commands/runs.ts | 109 - src/commands/schedules.test.ts | 209 -- src/commands/schedules.ts | 209 -- src/commands/setup-mcp.test.ts | 213 -- src/commands/setup-mcp.ts | 210 -- src/commands/setup.test.ts | 923 +-------- src/commands/setup.ts | 674 +------ src/commands/status.test.ts | 407 +--- src/commands/status.ts | 216 +- src/commands/update.test.ts | 228 --- src/commands/update.ts | 89 - src/commands/vault.test.ts | 333 --- src/commands/vault.ts | 313 --- src/commands/whoami.test.ts | 105 - src/commands/whoami.ts | 49 - src/daemon-entry.ts | 106 - src/daemon/actions.test.ts | 53 - src/daemon/actions.ts | 55 - src/daemon/actions/actions.test.ts | 119 -- src/daemon/actions/agent-install.ts | 56 - src/daemon/actions/cli-update.ts | 63 - src/daemon/actions/index.ts | 37 - src/daemon/actions/project-add-from-origin.ts | 70 - src/daemon/actions/settings-update.test.ts | 105 - src/daemon/actions/settings-update.ts | 130 -- src/daemon/actions/types.ts | 11 - src/daemon/actions/vault-actions.test.ts | 268 --- src/daemon/actions/vault-add.ts | 108 - .../actions/vault-content-actions.test.ts | 381 ---- src/daemon/actions/vault-edit.ts | 89 - src/daemon/actions/vault-files.ts | 67 - src/daemon/actions/vault-list.ts | 32 - src/daemon/actions/vault-read.ts | 73 - src/daemon/actions/vault-reindex.ts | 47 - src/daemon/actions/vault-remove.ts | 46 - src/daemon/actions/vault-search.ts | 66 - src/daemon/autostart/darwin.test.ts | 107 - src/daemon/autostart/darwin.ts | 69 - src/daemon/autostart/index.test.ts | 24 - src/daemon/autostart/index.ts | 58 - src/daemon/autostart/linux.test.ts | 110 - src/daemon/autostart/linux.ts | 65 - src/daemon/autostart/types.ts | 24 - src/daemon/autostart/win32.test.ts | 76 - src/daemon/autostart/win32.ts | 36 - src/daemon/config.test.ts | 332 --- src/daemon/config.ts | 279 --- src/daemon/daemon.test.ts | 82 - src/daemon/daemon.ts | 115 -- src/daemon/logger.test.ts | 71 - src/daemon/logger.ts | 52 - src/daemon/metrics.test.ts | 25 - src/daemon/metrics.ts | 82 - src/daemon/routes.actions.test.ts | 154 -- src/daemon/routes.test.ts | 584 ------ src/daemon/routes.ts | 397 ---- src/daemon/run-manager.test.ts | 578 ------ src/daemon/run-manager.ts | 321 --- src/daemon/scheduler.test.ts | 147 -- src/daemon/scheduler.ts | 203 -- src/daemon/server.test.ts | 207 -- src/daemon/server.ts | 78 - src/daemon/websocket.test.ts | 116 -- src/daemon/websocket.ts | 204 -- src/discovery/code-factory.test.ts | 154 -- src/discovery/code-factory.ts | 56 - src/discovery/markdown-factory.test.ts | 147 -- src/discovery/markdown-factory.ts | 118 -- src/discovery/scanner.test.ts | 158 -- src/discovery/scanner.ts | 87 - src/discovery/watcher.test.ts | 95 - src/discovery/watcher.ts | 58 - src/e2e/daemon-lifecycle.test.ts | 428 ---- src/examples/claude-agent.agent.test.ts | 10 - src/examples/claude-agent.agent.ts | 13 - src/examples/code-reviewer.agent.test.ts | 43 - src/examples/code-reviewer.agent.ts | 11 - src/examples/copilot.agent.test.ts | 10 - src/examples/copilot.agent.ts | 9 - src/examples/countdown.agent.test.ts | 73 - src/examples/countdown.agent.ts | 14 - src/examples/hello.agent.test.ts | 25 - src/examples/hello.agent.ts | 9 - src/examples/shell.agent.test.ts | 39 - src/examples/shell.agent.ts | 9 - src/hub/auth-callback.test.ts | 91 - src/hub/auth-callback.ts | 328 --- src/hub/auth.test.ts | 55 - src/hub/auth.ts | 46 - src/hub/command-dispatch.test.ts | 127 -- src/hub/command-dispatch.ts | 112 -- src/hub/hub-client.test.ts | 323 --- src/hub/hub-client.ts | 295 --- src/hub/hub-sync.test.ts | 580 ------ src/hub/hub-sync.ts | 400 ---- src/hub/hub-ws.test.ts | 15 - src/hub/hub-ws.ts | 298 --- src/hub/reconnection.test.ts | 77 - src/hub/reconnection.ts | 67 - src/hub/token-refresh.test.ts | 248 --- src/hub/token-refresh.ts | 85 - src/index.ts | 3 +- src/lib/api.test.ts | 100 + src/lib/api.ts | 57 + src/lib/callback-server.test.ts | 74 + src/lib/callback-server.ts | 88 + src/lib/config.test.ts | 62 + src/lib/config.ts | 49 + src/lib/oauth.test.ts | 147 ++ src/lib/oauth.ts | 113 ++ src/lib/origins.test.ts | 61 + src/lib/origins.ts | 42 + src/lib/status-info.test.ts | 74 + src/lib/status-info.ts | 46 + src/package-guard.test.ts | 46 + src/projects/projects.test.ts | 803 -------- src/projects/projects.ts | 414 ---- src/utils/action-client.ts | 38 - src/utils/daemon-client.test.ts | 89 - src/utils/daemon-client.ts | 50 - src/utils/ensure-daemon.test.ts | 62 - src/utils/ensure-daemon.ts | 33 - src/utils/render.test.ts | 408 ---- src/utils/render.ts | 103 - src/utils/schema-input.test.ts | 122 -- src/utils/schema-input.ts | 90 - src/utils/update-checker.test.ts | 108 - src/utils/update-checker.ts | 112 -- src/utils/version.test.ts | 15 - src/vaults/instance.ts | 22 - src/vaults/path-safety.test.ts | 25 - src/vaults/path-safety.ts | 13 - src/vaults/reconciler.test.ts | 111 - src/vaults/reconciler.ts | 106 - src/vaults/registry.test.ts | 111 - src/vaults/registry.ts | 128 -- src/vaults/sqlite-fts5-index.test.ts | 173 -- src/vaults/sqlite-fts5-index.ts | 159 -- src/vaults/types.ts | 72 - src/vaults/writer.test.ts | 103 - src/vaults/writer.ts | 82 - vitest.config.ts | 10 +- 167 files changed, 1354 insertions(+), 24033 deletions(-) delete mode 100644 src/commands/agents.test.ts delete mode 100644 src/commands/agents.ts delete mode 100644 src/commands/completions.test.ts delete mode 100644 src/commands/completions.ts delete mode 100644 src/commands/config-cmd.test.ts delete mode 100644 src/commands/config-cmd.ts delete mode 100644 src/commands/create.test.ts delete mode 100644 src/commands/create.ts delete mode 100644 src/commands/daemon-cmd.test.ts delete mode 100644 src/commands/daemon-cmd.ts delete mode 100644 src/commands/logs.test.ts delete mode 100644 src/commands/logs.ts delete mode 100644 src/commands/machines.test.ts delete mode 100644 src/commands/machines.ts delete mode 100644 src/commands/projects.test.ts delete mode 100644 src/commands/projects.ts delete mode 100644 src/commands/run.test.ts delete mode 100644 src/commands/run.ts delete mode 100644 src/commands/runs.test.ts delete mode 100644 src/commands/runs.ts delete mode 100644 src/commands/schedules.test.ts delete mode 100644 src/commands/schedules.ts delete mode 100644 src/commands/setup-mcp.test.ts delete mode 100644 src/commands/setup-mcp.ts delete mode 100644 src/commands/update.test.ts delete mode 100644 src/commands/update.ts delete mode 100644 src/commands/vault.test.ts delete mode 100644 src/commands/vault.ts delete mode 100644 src/commands/whoami.test.ts delete mode 100644 src/commands/whoami.ts delete mode 100644 src/daemon-entry.ts delete mode 100644 src/daemon/actions.test.ts delete mode 100644 src/daemon/actions.ts delete mode 100644 src/daemon/actions/actions.test.ts delete mode 100644 src/daemon/actions/agent-install.ts delete mode 100644 src/daemon/actions/cli-update.ts delete mode 100644 src/daemon/actions/index.ts delete mode 100644 src/daemon/actions/project-add-from-origin.ts delete mode 100644 src/daemon/actions/settings-update.test.ts delete mode 100644 src/daemon/actions/settings-update.ts delete mode 100644 src/daemon/actions/types.ts delete mode 100644 src/daemon/actions/vault-actions.test.ts delete mode 100644 src/daemon/actions/vault-add.ts delete mode 100644 src/daemon/actions/vault-content-actions.test.ts delete mode 100644 src/daemon/actions/vault-edit.ts delete mode 100644 src/daemon/actions/vault-files.ts delete mode 100644 src/daemon/actions/vault-list.ts delete mode 100644 src/daemon/actions/vault-read.ts delete mode 100644 src/daemon/actions/vault-reindex.ts delete mode 100644 src/daemon/actions/vault-remove.ts delete mode 100644 src/daemon/actions/vault-search.ts delete mode 100644 src/daemon/autostart/darwin.test.ts delete mode 100644 src/daemon/autostart/darwin.ts delete mode 100644 src/daemon/autostart/index.test.ts delete mode 100644 src/daemon/autostart/index.ts delete mode 100644 src/daemon/autostart/linux.test.ts delete mode 100644 src/daemon/autostart/linux.ts delete mode 100644 src/daemon/autostart/types.ts delete mode 100644 src/daemon/autostart/win32.test.ts delete mode 100644 src/daemon/autostart/win32.ts delete mode 100644 src/daemon/config.test.ts delete mode 100644 src/daemon/config.ts delete mode 100644 src/daemon/daemon.test.ts delete mode 100644 src/daemon/daemon.ts delete mode 100644 src/daemon/logger.test.ts delete mode 100644 src/daemon/logger.ts delete mode 100644 src/daemon/metrics.test.ts delete mode 100644 src/daemon/metrics.ts delete mode 100644 src/daemon/routes.actions.test.ts delete mode 100644 src/daemon/routes.test.ts delete mode 100644 src/daemon/routes.ts delete mode 100644 src/daemon/run-manager.test.ts delete mode 100644 src/daemon/run-manager.ts delete mode 100644 src/daemon/scheduler.test.ts delete mode 100644 src/daemon/scheduler.ts delete mode 100644 src/daemon/server.test.ts delete mode 100644 src/daemon/server.ts delete mode 100644 src/daemon/websocket.test.ts delete mode 100644 src/daemon/websocket.ts delete mode 100644 src/discovery/code-factory.test.ts delete mode 100644 src/discovery/code-factory.ts delete mode 100644 src/discovery/markdown-factory.test.ts delete mode 100644 src/discovery/markdown-factory.ts delete mode 100644 src/discovery/scanner.test.ts delete mode 100644 src/discovery/scanner.ts delete mode 100644 src/discovery/watcher.test.ts delete mode 100644 src/discovery/watcher.ts delete mode 100644 src/e2e/daemon-lifecycle.test.ts delete mode 100644 src/examples/claude-agent.agent.test.ts delete mode 100644 src/examples/claude-agent.agent.ts delete mode 100644 src/examples/code-reviewer.agent.test.ts delete mode 100644 src/examples/code-reviewer.agent.ts delete mode 100644 src/examples/copilot.agent.test.ts delete mode 100644 src/examples/copilot.agent.ts delete mode 100644 src/examples/countdown.agent.test.ts delete mode 100644 src/examples/countdown.agent.ts delete mode 100644 src/examples/hello.agent.test.ts delete mode 100644 src/examples/hello.agent.ts delete mode 100644 src/examples/shell.agent.test.ts delete mode 100644 src/examples/shell.agent.ts delete mode 100644 src/hub/auth-callback.test.ts delete mode 100644 src/hub/auth-callback.ts delete mode 100644 src/hub/auth.test.ts delete mode 100644 src/hub/auth.ts delete mode 100644 src/hub/command-dispatch.test.ts delete mode 100644 src/hub/command-dispatch.ts delete mode 100644 src/hub/hub-client.test.ts delete mode 100644 src/hub/hub-client.ts delete mode 100644 src/hub/hub-sync.test.ts delete mode 100644 src/hub/hub-sync.ts delete mode 100644 src/hub/hub-ws.test.ts delete mode 100644 src/hub/hub-ws.ts delete mode 100644 src/hub/reconnection.test.ts delete mode 100644 src/hub/reconnection.ts delete mode 100644 src/hub/token-refresh.test.ts delete mode 100644 src/hub/token-refresh.ts create mode 100644 src/lib/api.test.ts create mode 100644 src/lib/api.ts create mode 100644 src/lib/callback-server.test.ts create mode 100644 src/lib/callback-server.ts create mode 100644 src/lib/config.test.ts create mode 100644 src/lib/config.ts create mode 100644 src/lib/oauth.test.ts create mode 100644 src/lib/oauth.ts create mode 100644 src/lib/origins.test.ts create mode 100644 src/lib/origins.ts create mode 100644 src/lib/status-info.test.ts create mode 100644 src/lib/status-info.ts create mode 100644 src/package-guard.test.ts delete mode 100644 src/projects/projects.test.ts delete mode 100644 src/projects/projects.ts delete mode 100644 src/utils/action-client.ts delete mode 100644 src/utils/daemon-client.test.ts delete mode 100644 src/utils/daemon-client.ts delete mode 100644 src/utils/ensure-daemon.test.ts delete mode 100644 src/utils/ensure-daemon.ts delete mode 100644 src/utils/render.test.ts delete mode 100644 src/utils/render.ts delete mode 100644 src/utils/schema-input.test.ts delete mode 100644 src/utils/schema-input.ts delete mode 100644 src/utils/update-checker.test.ts delete mode 100644 src/utils/update-checker.ts delete mode 100644 src/utils/version.test.ts delete mode 100644 src/vaults/instance.ts delete mode 100644 src/vaults/path-safety.test.ts delete mode 100644 src/vaults/path-safety.ts delete mode 100644 src/vaults/reconciler.test.ts delete mode 100644 src/vaults/reconciler.ts delete mode 100644 src/vaults/registry.test.ts delete mode 100644 src/vaults/registry.ts delete mode 100644 src/vaults/sqlite-fts5-index.test.ts delete mode 100644 src/vaults/sqlite-fts5-index.ts delete mode 100644 src/vaults/types.ts delete mode 100644 src/vaults/writer.test.ts delete mode 100644 src/vaults/writer.ts diff --git a/CHANGELOG.md b/CHANGELOG.md index b1df70f..75789eb 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -7,6 +7,18 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 --- +## [0.25.0] - 2026-06-12 + +### Changed +- **BREAKING - Memory client reboot.** The CLI is now the agentage Memory client. + All agent-runtime commands were removed: run, agents, runs, machines, schedules, + projects, vault, daemon, logs, create, config, whoami (folded into status), + completions, update, setup mcp. The local daemon (:4243) is gone - the CLI is stateless. +- `setup` now signs in via OAuth 2.1 (Dynamic Client Registration + PKCE) against + auth.agentage.io with a localhost callback; flags: `--disconnect`, `--reauth`, `--no-browser`. +- `status` shows account, plan, memories, and endpoint health (`--json` supported). +- Dependencies reduced to chalk + commander + open. + ## [0.24.6] - 2026-04-26 ### New Features diff --git a/CLAUDE.md b/CLAUDE.md index 386f027..dbed5d5 100644 --- a/CLAUDE.md +++ b/CLAUDE.md @@ -1,66 +1,29 @@ -# CLAUDE.md - -This file provides guidance to Claude Code (claude.ai/code) when working with code in this repository. - -## Project Overview - -**@agentage/cli** — CLI and daemon for the Agentage platform. Provides the `agentage` command-line tool and a local daemon process that discovers, runs, and manages AI agents. - -**Repository:** `agentage/cli` -**Default Branch:** `master` -**Package:** Single npm package (ESM) - -## Development Commands - -```bash -npm install # Install dependencies -npm run build # Compile TypeScript (tsc) -npm test # Run Vitest unit tests -npm run type-check # TypeScript checking -npm run lint # ESLint -npm run format:check # Prettier check -npm run verify # Full pipeline: type-check + lint + format:check + build + test -npm run clean # Clean dist/ -``` - -## Architecture - -``` -src/ -├── cli.ts # CLI entrypoint (Commander.js) -├── commands/ # CLI commands (init, publish, list, etc.) -├── daemon/ # Local daemon (Express + WebSocket) -├── hub/ # Hub client for cloud interactions -├── discovery/ # Agent discovery from filesystem -└── utils/ # Shared utilities -``` - -### Key Patterns - -- **CLI:** Commander.js command definitions in `src/commands/` -- **Daemon:** Express.js HTTP + WebSocket server for agent lifecycle management -- **Discovery:** Filesystem scanning for `.agent.md` and `.agent.ts` files -- **Hub Integration:** Supabase client for cloud platform communication - -### Key Dependencies - -- `@agentage/core`, `@agentage/platform` — Agent framework -- `commander` — CLI framework -- `express`, `ws` — Daemon server -- `@supabase/supabase-js` — Cloud platform -- `gray-matter` — Agent manifest parsing -- `chalk` — Terminal output - -## Testing - -- **Framework:** Vitest -- **Pattern:** `*.test.ts` colocated with source -- **Coverage:** 70% minimum threshold - -## Publishing - -Published to npm as `@agentage/cli`. **Never run `npm publish` manually** — use the automated release pipeline via GitHub Actions. - -## Standards - -See [root CLAUDE.md](../../CLAUDE.md) and [agentage CLAUDE.md](../CLAUDE.md) for cross-repo conventions, branching strategy, and tech standards. +# CLAUDE.md - @agentage/cli + +Terminal client for agentage Memory. v0.25 reboot: `setup` (OAuth sign-in) + `status` only. +The pre-0.25 agent-runtime CLI (daemon, run/agents/machines/...) lives in git history only - +do not resurrect patterns from it. + +## Layout +- `src/cli.ts` - commander entry (excluded from coverage; keep logic out of it) +- `src/commands/` - thin command wiring; flow logic takes injected `Deps` for testability +- `src/lib/` - origins (one FQDN -> service URLs), config (`~/.agentage`, 0600 auth.json), + oauth (DCR + PKCE), callback-server (one-shot localhost), api (bearer + refresh-once), + status-info +- `src/package-guard.test.ts` - CI guard: no daemon/agent-runtime remnants, runtime deps + stay exactly `chalk + commander + open` + +## Auth model +Fresh DCR public client per `setup` run (the redirect URI binds the ephemeral callback +port); `client_id` stored in auth.json for the refresh grant. Tokens are opaque; `status` +validates them via the OAuth introspection endpoint (`/api/auth/mcp/get-session`). +Account details (email/plan/memories) in `status` are pending OAuth-bearer support in the +backend REST API - it currently accepts session cookies only. + +## Conventions +- `AGENTAGE_SITE_FQDN` selects the target host (production default); `AGENTAGE_CONFIG_DIR` + overrides the config dir - test isolation depends on it, never remove +- `npm run verify` before any push; CI also runs `test:coverage` (thresholds 65/70/70/70) +- Conventional commits; publish only via the release workflow, never local `npm publish` +- This is a PUBLIC repo: no secrets, no internal hosts, no roadmap, no provider names in + code, docs, comments, or commit messages diff --git a/README.md b/README.md index 6b27e6e..4ad0d63 100644 --- a/README.md +++ b/README.md @@ -3,207 +3,56 @@ [![CI](https://github.com/agentage/cli/actions/workflows/ci.yml/badge.svg)](https://github.com/agentage/cli/actions/workflows/ci.yml) [![npm version](https://badge.fury.io/js/%40agentage%2Fcli.svg)](https://badge.fury.io/js/%40agentage%2Fcli) -CLI and daemon for the [Agentage](https://agentage.io) control plane. Discovers local agents, executes them, and connects to the hub for multi-machine orchestration. +The terminal client for [agentage Memory](https://agentage.io) - one memory, every AI, owned by you. -## Installation +> v0.25 is a reboot: the CLI is now the agentage **Memory client**. All agent-runtime +> commands from 0.24.x and earlier (daemon, run, agents, machines, ...) were removed. -```bash -npm install -g @agentage/cli -``` - -## Quick Start — 5 minutes to your first agent - -From `npm install` to a running agent in five commands. No hub, no login, no API key required. - -### 1. Install +## Install ```bash npm install -g @agentage/cli ``` -### 2. Write an agent +Requires Node.js >= 22. -Create `~/agents/hello.agent.md` (the default agents directory — see `agentage status` to confirm or change): +## Commands -```markdown ---- -description: Simple greeting agent ---- -You greet the user by name. Reply in exactly one short sentence. -``` +### `agentage setup` -That's a complete agent — YAML frontmatter plus a system prompt. The daemon picks up any `.agent.md` file it finds. - -### 3. Verify discovery +Signs this machine in to your agentage memory. Opens a browser for OAuth 2.1 +sign-in (PKCE) and stores the resulting tokens in `~/.agentage/auth.json` +(mode 0600). No passwords ever touch the terminal. ```bash -agentage agents -``` - +agentage setup # browser sign-in, then prints status +agentage setup --no-browser # print the sign-in URL instead of opening a browser +agentage setup --reauth # force a fresh sign-in +agentage setup --disconnect # sign out and remove local credentials ``` -NAME DESCRIPTION PATH -hello Simple greeting agent ~/agents/hello.agent.md -1 agents discovered -``` - -The daemon auto-starts on first command and listens on `localhost:4243`. +### `agentage status` -### 4. Run it +Shows this machine's connection, one line per fact: CLI version, target, +sign-in state, and endpoint reachability. ```bash -agentage run hello "Volodymyr" +agentage status +agentage status --json ``` -Without an LLM adapter configured, markdown agents echo the prompt plus the task — enough to confirm the pipeline is wired. For a real model response, scaffold a code agent that uses the Claude adapter: - -```bash -agentage create greeter -t claude # writes greeter.agent.ts -npm install @anthropic-ai/claude-agent-sdk -export ANTHROPIC_API_KEY=sk-… -agentage run greeter "Hi" -``` - -### 5. Explore - -```bash -agentage status # daemon + hub health, discovery dirs -agentage runs # history of recent runs -agentage create --help # other templates: simple, shell, claude, copilot, llm -``` - -### Real-world example - -The [`pr-list`](https://github.com/vreshch/agents/tree/master/plugins/pr-list) agent (private repo) is a deterministic utility that queries `gh pr list` across a set of repos and returns a structured table. It's the smallest example of a code agent (`.agent.ts`) that uses a config object and has no LLM dependency — a good reference for your first "real" agent. - -## Daemon - -The daemon is a lightweight Express server that runs on each machine. It discovers local agents, executes them, and optionally syncs with a central hub. - -### API - -| Method | Endpoint | Description | -| ------ | ----------------------- | --------------------------------------------------- | -| `GET` | `/api/health` | Status, version, uptime, machine ID, hub connection | -| `GET` | `/api/agents` | List discovered agent manifests | -| `POST` | `/api/agents/refresh` | Re-scan agent directories | -| `POST` | `/api/agents/:name/run` | Execute an agent (`{ task, config?, context? }`) | -| `GET` | `/api/runs` | List all runs | -| `GET` | `/api/runs/:id` | Get run details + output | -| `POST` | `/api/runs/:id/cancel` | Cancel a running execution | -| `POST` | `/api/runs/:id/input` | Send input to a waiting agent | +## Environment -Default port: **4243** - -### Agent Discovery - -The daemon scans configured directories for agents: - -``` -~/.agentage/agents/*.agent.md # Global agents -.agentage/agents/*.agent.md # Workspace agents -``` - -Two built-in factories: **markdown** (`.agent.md` files with YAML frontmatter) and **code** (TypeScript/JavaScript modules exporting an Agent). - -### Hub Sync - -When authenticated (`agentage setup`), the daemon connects to the hub via WebSocket — registering the machine, syncing agents, and relaying run events. - -| Method | Endpoint | Description | -| ------ | ------------------- | ------------------------------------- | -| `GET` | `/api/hub/machines` | List all hub-connected machines | -| `GET` | `/api/hub/agents` | Aggregated agents across all machines | -| `POST` | `/api/hub/runs` | Start a run on any connected machine | - -#### Resilience - -The daemon is designed to keep its hub connection alive across transient network failures, auth expiry, and hub restarts — without requiring a restart of the daemon process. - -| Behavior | Value | Source | -| ----------------------- | --------------------------------------------------------------- | --------------------------------------------------------- | -| Heartbeat interval | **30 s** | `HEARTBEAT_INTERVAL_MS` in `src/hub/hub-sync.ts` | -| Heartbeat scheduling | Sequential (next scheduled after current resolves — no overlap) | `src/hub/hub-sync.ts` | -| Reconnect initial delay | **1 s** | `initialDelayMs` in `src/hub/reconnection.ts` | -| Reconnect max delay | **30 s** | `maxDelayMs` in `src/hub/reconnection.ts` | -| Reconnect backoff | Exponential (×2 per failed attempt, capped at max) | `src/hub/reconnection.ts` | -| Token refresh threshold | **5 min** before expiry | `TOKEN_REFRESH_THRESHOLD_S` in `src/hub/token-refresh.ts` | -| Token refresh trigger | Pre-heartbeat check + reactive on 401 | `token-refresh.ts`, `hub-client.ts` | - -**Heartbeat payload.** Each heartbeat POSTs `{ agents, activeRunIds, daemonVersion }` to `/api/machines/:id/heartbeat`. The response may contain `pendingCommands` (e.g. queued `cancel` or `input` requests from the hub) which are applied to the local run-manager inline before the next heartbeat. - -**Reconnection flow.** - -1. Hub WS `close` event fires → `connected` set to `false`, reconnector started. -2. Reconnector calls `connectAll` (re-register machine + reopen WS) with exponential backoff. -3. On success, backoff delay resets to 1 s and the heartbeat loop resumes. -4. `stop()` (daemon shutdown) cancels the reconnector and deregisters the machine best-effort. - -**Auth handling.** Token refresh is attempted both proactively (before each heartbeat, if within 5 min of expiry) and reactively (on any hub API 401 response). Refresh uses the Supabase refresh token fetched from `/api/health`. A failed refresh surfaces as a warning but does not kill the daemon — the next reconnect cycle will retry. - -**Offline mode.** If no auth is present at startup, the daemon runs in standalone mode: local agent execution still works via the REST API and local WebSocket (`/ws`), but no hub sync is attempted. Running `agentage setup` after the fact requires a daemon restart to pick up the new auth. - -## CLI Commands - -| Command | Description | -| ---------------------------- | --------------------------- | -| `agentage daemon` | Start the daemon server | -| `agentage agents` | List discovered agents | -| `agentage run [task]` | Execute an agent | -| `agentage runs` | List runs | -| `agentage machines` | List hub-connected machines | -| `agentage status` | Show daemon and hub status | -| `agentage logs` | View daemon logs | -| `agentage setup` | Configure machine + hub + auth (interactive or headless) | -| `agentage setup --disconnect` | Deregister and remove credentials | - -## Project Structure - -``` -src/ -├── cli.ts # CLI entry point (commander) -├── daemon-entry.ts # Daemon process entry point -├── commands/ # CLI command handlers -│ ├── daemon-cmd.ts # agentage daemon -│ ├── agents.ts # agentage agents -│ ├── run.ts # agentage run -│ ├── runs.ts # agentage runs -│ ├── machines.ts # agentage machines -│ ├── status.ts # agentage status -│ ├── logs.ts # agentage logs -│ └── setup.ts # agentage setup -├── daemon/ # Daemon server -│ ├── server.ts # Express + HTTP server setup -│ ├── routes.ts # REST API routes -│ ├── config.ts # Daemon configuration -│ ├── run-manager.ts # Agent execution + run lifecycle -│ ├── websocket.ts # WebSocket for streaming -│ └── logger.ts # Structured logging -├── discovery/ # Agent discovery -│ ├── scanner.ts # Directory scanning -│ ├── markdown-factory.ts # .agent.md parser (gray-matter) -│ └── code-factory.ts # JS/TS module loader (jiti) -├── hub/ # Hub connection -│ ├── auth.ts # Auth token storage -│ ├── auth-callback.ts # OAuth callback server -│ ├── hub-client.ts # Hub REST client -│ ├── hub-sync.ts # Machine/agent sync -│ ├── hub-ws.ts # Hub WebSocket client -│ └── reconnection.ts # Auto-reconnect logic -└── utils/ - ├── daemon-client.ts # CLI → daemon HTTP client - ├── ensure-daemon.ts # Auto-start daemon if not running - └── render.ts # Terminal output formatting -``` +| Variable | Purpose | Default | +|---|---|---| +| `AGENTAGE_SITE_FQDN` | Target host | `agentage.io` | +| `AGENTAGE_CONFIG_DIR` | Credential/config directory | `~/.agentage` | ## Development -Requires Node.js >= 22.0.0. - ```bash npm ci -npm run verify # type-check + lint + format + test + build -npm run build && npm link # test CLI locally +npm run verify # type-check + lint + format + test + build ``` ## License diff --git a/package-lock.json b/package-lock.json index a802428..49eaa74 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,103 +1,36 @@ { "name": "@agentage/cli", - "version": "0.24.6", + "version": "0.25.0", "lockfileVersion": 3, "requires": true, "packages": { "": { "name": "@agentage/cli", - "version": "0.24.6", + "version": "0.25.0", "license": "MIT", "dependencies": { - "@agentage/core": "^0.10.0", - "@agentage/platform": "^0.6.1", - "@supabase/supabase-js": "2.106.2", - "ajv": "8.20.0", - "better-sqlite3": "*", - "chalk": "*", - "commander": "*", - "croner": "*", - "express": "*", - "gray-matter": "*", - "jiti": "*", - "open": "*", - "ws": "*" + "chalk": "latest", + "commander": "latest", + "open": "latest" }, "bin": { "agentage": "dist/cli.js" }, "devDependencies": { - "@anthropic-ai/sdk": "*", - "@types/better-sqlite3": "*", - "@types/express": "*", - "@types/node": "*", - "@types/ws": "*", - "@typescript-eslint/eslint-plugin": "*", - "@typescript-eslint/parser": "*", - "@vitest/coverage-v8": "*", - "eslint": "*", - "eslint-config-prettier": "*", + "@types/node": "latest", + "@typescript-eslint/eslint-plugin": "latest", + "@typescript-eslint/parser": "latest", + "@vitest/coverage-v8": "latest", + "eslint": "latest", + "eslint-config-prettier": "latest", "eslint-plugin-prettier": "latest", - "prettier": "*", - "typescript": "*", - "vitest": "*" - }, - "engines": { - "node": ">=22.0.0", - "npm": ">=10.0.0" - } - }, - "node_modules/@agentage/core": { - "version": "0.10.1", - "resolved": "https://registry.npmjs.org/@agentage/core/-/core-0.10.1.tgz", - "integrity": "sha512-hzSDEapUUmPJxVOJUv4BXNZ6o8ODZzo2vdbsqOLeTjvqYsEE+EF4mNeacs++lHz9yVFvrTd0HcC7ILbFt3+B4g==", - "license": "MIT", - "engines": { - "node": ">=22.0.0", - "npm": ">=10.0.0" - }, - "peerDependencies": { - "zod": ">=4.0.0" + "prettier": "latest", + "typescript": "latest", + "vitest": "latest" }, - "peerDependenciesMeta": { - "zod": { - "optional": true - } - } - }, - "node_modules/@agentage/platform": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/@agentage/platform/-/platform-0.6.1.tgz", - "integrity": "sha512-jaCODocqYMvh6CpuWMk1Lnepb+TYtcNeZTGjQz2X8FfeJRgPTDNbxrMtL9R9cEIYTKR7HgSWfzOZ5W5/K6zjMQ==", - "license": "MIT", "engines": { "node": ">=22.0.0", "npm": ">=10.0.0" - }, - "peerDependencies": { - "@agentage/core": ">=0.2.0" - } - }, - "node_modules/@anthropic-ai/sdk": { - "version": "0.100.1", - "resolved": "https://registry.npmjs.org/@anthropic-ai/sdk/-/sdk-0.100.1.tgz", - "integrity": "sha512-RANcEe7LpiLczkKGOwoXOTuFdPhuubS0i4xaAKOMpcqc55YO0mukgxppV7eygx3DXNjxWT6RYOLPyOy0aIAmwg==", - "dev": true, - "license": "MIT", - "dependencies": { - "json-schema-to-ts": "^3.1.1", - "standardwebhooks": "^1.0.0" - }, - "bin": { - "anthropic-ai-sdk": "bin/cli" - }, - "peerDependencies": { - "zod": "^3.25.0 || ^4.0.0" - }, - "peerDependenciesMeta": { - "zod": { - "optional": true - } } }, "node_modules/@babel/helper-string-parser": { @@ -136,16 +69,6 @@ "node": ">=6.0.0" } }, - "node_modules/@babel/runtime": { - "version": "7.29.2", - "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.29.2.tgz", - "integrity": "sha512-JiDShH45zKHWyGe4ZNVRrCjBz8Nh9TMmZG1kh4QTK8hCBTWBi8Da+i7s1fJw7/lYpM4ccepSNfqzZ/QvABBi5g==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=6.9.0" - } - }, "node_modules/@babel/types": { "version": "7.29.0", "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.29.0.tgz", @@ -684,13 +607,6 @@ "dev": true, "license": "MIT" }, - "node_modules/@stablelib/base64": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/@stablelib/base64/-/base64-1.0.1.tgz", - "integrity": "sha512-1bnPQqSxSuc3Ii6MhBysoWCg58j97aUjuCSZrGSmDxNqtytIi0k8utUenAwTZN4V5mXXYGsVUI9zeBqy+jBOSQ==", - "dev": true, - "license": "MIT" - }, "node_modules/@standard-schema/spec": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/@standard-schema/spec/-/spec-1.1.0.tgz", @@ -698,90 +614,6 @@ "dev": true, "license": "MIT" }, - "node_modules/@supabase/auth-js": { - "version": "2.106.2", - "resolved": "https://registry.npmjs.org/@supabase/auth-js/-/auth-js-2.106.2.tgz", - "integrity": "sha512-VcAjUErkHkhC5Jaf+g/G1qbkQrFh8edaCdHa7pxJmHUjkWKjT7UnYCtPA89XV0N0GIYRkEqJZw5V62CtOxTmBQ==", - "license": "MIT", - "dependencies": { - "tslib": "2.8.1" - }, - "engines": { - "node": ">=20.0.0" - } - }, - "node_modules/@supabase/functions-js": { - "version": "2.106.2", - "resolved": "https://registry.npmjs.org/@supabase/functions-js/-/functions-js-2.106.2.tgz", - "integrity": "sha512-oRnr0QrL8H+zTO1YyQ1QjiHZU/957jvubbxSJTUm2XLAgzoGGV9Tahfyd+uvLsBLRVmXLtpU3oyCjdQIvkGMOA==", - "license": "MIT", - "dependencies": { - "tslib": "2.8.1" - }, - "engines": { - "node": ">=20.0.0" - } - }, - "node_modules/@supabase/phoenix": { - "version": "0.4.2", - "resolved": "https://registry.npmjs.org/@supabase/phoenix/-/phoenix-0.4.2.tgz", - "integrity": "sha512-YSAGnmDAfuleFCVt3CeurQZAhxRfXWeZIIkwp7NhYzQ1UwW6ePSnzsFAiUm/mbCkfoCf70QQHKW/K6RKh52a4A==", - "license": "MIT" - }, - "node_modules/@supabase/postgrest-js": { - "version": "2.106.2", - "resolved": "https://registry.npmjs.org/@supabase/postgrest-js/-/postgrest-js-2.106.2.tgz", - "integrity": "sha512-tDOzyPgp9pIRMR2x6C9+uDSJrnXSzxLtt3d7nC+Lrsy3jnJDHYfdQC/xcRyhJE/TOBJ0heSqRKR3UmejDjZxsw==", - "license": "MIT", - "dependencies": { - "tslib": "2.8.1" - }, - "engines": { - "node": ">=20.0.0" - } - }, - "node_modules/@supabase/realtime-js": { - "version": "2.106.2", - "resolved": "https://registry.npmjs.org/@supabase/realtime-js/-/realtime-js-2.106.2.tgz", - "integrity": "sha512-LdRGT7DNhyZkPjubUv5bSdAZ0jSEX8wTHvx7htj7+K59TOZRvz4TuQK7tL2RWxyIZVeFMRluL04SzWS61rKnUA==", - "license": "MIT", - "dependencies": { - "@supabase/phoenix": "^0.4.2", - "tslib": "2.8.1" - }, - "engines": { - "node": ">=20.0.0" - } - }, - "node_modules/@supabase/storage-js": { - "version": "2.106.2", - "resolved": "https://registry.npmjs.org/@supabase/storage-js/-/storage-js-2.106.2.tgz", - "integrity": "sha512-xgKCSYuev1YarV+iVqr+zlfgSyremnJtn8T0NCT8L4XmMv1CLtESc0Q6kNp8+mKWdX/8ND0nzm7OMKx08kwNAw==", - "license": "MIT", - "dependencies": { - "iceberg-js": "^0.8.1", - "tslib": "2.8.1" - }, - "engines": { - "node": ">=20.0.0" - } - }, - "node_modules/@supabase/supabase-js": { - "version": "2.106.2", - "resolved": "https://registry.npmjs.org/@supabase/supabase-js/-/supabase-js-2.106.2.tgz", - "integrity": "sha512-2/RZ/1fmJx/MRSEDG2Xk8+J4JVk5clM9V0uSI6kUTrcS32KA89DtqI5RUOC9r6mzY3WBC9qexLjssIHjbLyVJA==", - "license": "MIT", - "dependencies": { - "@supabase/auth-js": "2.106.2", - "@supabase/functions-js": "2.106.2", - "@supabase/postgrest-js": "2.106.2", - "@supabase/realtime-js": "2.106.2", - "@supabase/storage-js": "2.106.2" - }, - "engines": { - "node": ">=20.0.0" - } - }, "node_modules/@tybys/wasm-util": { "version": "0.10.2", "resolved": "https://registry.npmjs.org/@tybys/wasm-util/-/wasm-util-0.10.2.tgz", @@ -793,27 +625,6 @@ "tslib": "^2.4.0" } }, - "node_modules/@types/better-sqlite3": { - "version": "7.6.13", - "resolved": "https://registry.npmjs.org/@types/better-sqlite3/-/better-sqlite3-7.6.13.tgz", - "integrity": "sha512-NMv9ASNARoKksWtsq/SHakpYAYnhBrQgGD8zkLYk/jaK8jUGn08CfEdTRgYhMypUQAfzSP8W6gNLe0q19/t4VA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/node": "*" - } - }, - "node_modules/@types/body-parser": { - "version": "1.19.6", - "resolved": "https://registry.npmjs.org/@types/body-parser/-/body-parser-1.19.6.tgz", - "integrity": "sha512-HLFeCYgz89uk22N5Qg3dvGvsv46B8GLvKKo1zKG4NybA8U2DiEO3w9lqGg29t/tfLRJpJ6iQxnVw4OnB7MoM9g==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/connect": "*", - "@types/node": "*" - } - }, "node_modules/@types/chai": { "version": "5.2.3", "resolved": "https://registry.npmjs.org/@types/chai/-/chai-5.2.3.tgz", @@ -825,16 +636,6 @@ "assertion-error": "^2.0.1" } }, - "node_modules/@types/connect": { - "version": "3.4.38", - "resolved": "https://registry.npmjs.org/@types/connect/-/connect-3.4.38.tgz", - "integrity": "sha512-K6uROf1LD88uDQqJCktA4yzL1YYAK6NgfsI0v/mTgyPKWsX1CnJ0XPSDhViejru1GcRkLWb8RlzFYJRqGUbaug==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/node": "*" - } - }, "node_modules/@types/deep-eql": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/@types/deep-eql/-/deep-eql-4.0.2.tgz", @@ -856,38 +657,6 @@ "dev": true, "license": "MIT" }, - "node_modules/@types/express": { - "version": "5.0.6", - "resolved": "https://registry.npmjs.org/@types/express/-/express-5.0.6.tgz", - "integrity": "sha512-sKYVuV7Sv9fbPIt/442koC7+IIwK5olP1KWeD88e/idgoJqDm3JV/YUiPwkoKK92ylff2MGxSz1CSjsXelx0YA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/body-parser": "*", - "@types/express-serve-static-core": "^5.0.0", - "@types/serve-static": "^2" - } - }, - "node_modules/@types/express-serve-static-core": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/@types/express-serve-static-core/-/express-serve-static-core-5.1.1.tgz", - "integrity": "sha512-v4zIMr/cX7/d2BpAEX3KNKL/JrT1s43s96lLvvdTmza1oEvDudCqK9aF/djc/SWgy8Yh0h30TZx5VpzqFCxk5A==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/node": "*", - "@types/qs": "*", - "@types/range-parser": "*", - "@types/send": "*" - } - }, - "node_modules/@types/http-errors": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/@types/http-errors/-/http-errors-2.0.5.tgz", - "integrity": "sha512-r8Tayk8HJnX0FztbZN7oVqGccWgw98T/0neJphO91KkmOzug1KkofZURD4UaD5uH8AqcFLfdPErnBod0u71/qg==", - "dev": true, - "license": "MIT" - }, "node_modules/@types/json-schema": { "version": "7.0.15", "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz", @@ -905,51 +674,6 @@ "undici-types": ">=7.24.0 <7.24.7" } }, - "node_modules/@types/qs": { - "version": "6.15.0", - "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.15.0.tgz", - "integrity": "sha512-JawvT8iBVWpzTrz3EGw9BTQFg3BQNmwERdKE22vlTxawwtbyUSlMppvZYKLZzB5zgACXdXxbD3m1bXaMqP/9ow==", - "dev": true, - "license": "MIT" - }, - "node_modules/@types/range-parser": { - "version": "1.2.7", - "resolved": "https://registry.npmjs.org/@types/range-parser/-/range-parser-1.2.7.tgz", - "integrity": "sha512-hKormJbkJqzQGhziax5PItDUTMAM9uE2XXQmM37dyd4hVM+5aVl7oVxMVUiVQn2oCQFN/LKCZdvSM0pFRqbSmQ==", - "dev": true, - "license": "MIT" - }, - "node_modules/@types/send": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/@types/send/-/send-1.2.1.tgz", - "integrity": "sha512-arsCikDvlU99zl1g69TcAB3mzZPpxgw0UQnaHeC1Nwb015xp8bknZv5rIfri9xTOcMuaVgvabfIRA7PSZVuZIQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/node": "*" - } - }, - "node_modules/@types/serve-static": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/@types/serve-static/-/serve-static-2.2.0.tgz", - "integrity": "sha512-8mam4H1NHLtu7nmtalF7eyBH14QyOASmcxHhSfEoRyr0nP/YdoesEtU+uSRvMe96TW/HPTtkoKqQLl53N7UXMQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/http-errors": "*", - "@types/node": "*" - } - }, - "node_modules/@types/ws": { - "version": "8.18.1", - "resolved": "https://registry.npmjs.org/@types/ws/-/ws-8.18.1.tgz", - "integrity": "sha512-ThVF6DCVhA8kUGy+aazFQ4kXQ7E1Ty7A3ypFOe0IcJV8O/M511G99AW24irKrW56Wt44yG9+ij8FaqoBGkuBXg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/node": "*" - } - }, "node_modules/@typescript-eslint/eslint-plugin": { "version": "8.60.0", "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-8.60.0.tgz", @@ -1327,19 +1051,6 @@ "url": "https://opencollective.com/vitest" } }, - "node_modules/accepts": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/accepts/-/accepts-2.0.0.tgz", - "integrity": "sha512-5cvg6CtKwfgdmVqY1WIiXKc3Q1bkRqGLi+2W/6ao+6Y7gu/RCwRuAhGEzh5B4KlszSuTLgZYuqFqo5bImjNKng==", - "license": "MIT", - "dependencies": { - "mime-types": "^3.0.0", - "negotiator": "^1.0.0" - }, - "engines": { - "node": ">= 0.6" - } - }, "node_modules/acorn": { "version": "8.16.0", "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.16.0.tgz", @@ -1363,31 +1074,6 @@ "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" } }, - "node_modules/ajv": { - "version": "8.20.0", - "resolved": "https://registry.npmjs.org/ajv/-/ajv-8.20.0.tgz", - "integrity": "sha512-Thbli+OlOj+iMPYFBVBfJ3OmCAnaSyNn4M1vz9T6Gka5Jt9ba/HIR56joy65tY6kx/FCF5VXNB819Y7/GUrBGA==", - "license": "MIT", - "dependencies": { - "fast-deep-equal": "^3.1.3", - "fast-uri": "^3.0.1", - "json-schema-traverse": "^1.0.0", - "require-from-string": "^2.0.2" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/epoberezkin" - } - }, - "node_modules/argparse": { - "version": "1.0.10", - "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", - "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", - "license": "MIT", - "dependencies": { - "sprintf-js": "~1.0.2" - } - }, "node_modules/assertion-error": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/assertion-error/-/assertion-error-2.0.1.tgz", @@ -1420,84 +1106,6 @@ "node": "18 || 20 || >=22" } }, - "node_modules/base64-js": { - "version": "1.5.1", - "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz", - "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==", - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ], - "license": "MIT" - }, - "node_modules/better-sqlite3": { - "version": "12.10.0", - "resolved": "https://registry.npmjs.org/better-sqlite3/-/better-sqlite3-12.10.0.tgz", - "integrity": "sha512-CyzaZRQKyHkB2ZInfTTl2nvT33EbDpjkLEbE8/Zck3Ll6O0qqvuGdrJ45HgtH+HykRg88ITY3AdreBGN70aBSQ==", - "hasInstallScript": true, - "license": "MIT", - "dependencies": { - "bindings": "^1.5.0", - "prebuild-install": "^7.1.1" - }, - "engines": { - "node": "20.x || 22.x || 23.x || 24.x || 25.x || 26.x" - } - }, - "node_modules/bindings": { - "version": "1.5.0", - "resolved": "https://registry.npmjs.org/bindings/-/bindings-1.5.0.tgz", - "integrity": "sha512-p2q/t/mhvuOj/UeLlV6566GD/guowlr0hHxClI0W9m7MWYkL1F0hLo+0Aexs9HSPCtR1SXQ0TD3MMKrXZajbiQ==", - "license": "MIT", - "dependencies": { - "file-uri-to-path": "1.0.0" - } - }, - "node_modules/bl": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/bl/-/bl-4.1.0.tgz", - "integrity": "sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==", - "license": "MIT", - "dependencies": { - "buffer": "^5.5.0", - "inherits": "^2.0.4", - "readable-stream": "^3.4.0" - } - }, - "node_modules/body-parser": { - "version": "2.2.2", - "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-2.2.2.tgz", - "integrity": "sha512-oP5VkATKlNwcgvxi0vM0p/D3n2C3EReYVX+DNYs5TjZFn/oQt2j+4sVJtSMr18pdRr8wjTcBl6LoV+FUwzPmNA==", - "license": "MIT", - "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" - }, - "engines": { - "node": ">=18" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/express" - } - }, "node_modules/brace-expansion": { "version": "5.0.4", "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-5.0.4.tgz", @@ -1511,30 +1119,6 @@ "node": "18 || 20 || >=22" } }, - "node_modules/buffer": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/buffer/-/buffer-5.7.1.tgz", - "integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==", - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ], - "license": "MIT", - "dependencies": { - "base64-js": "^1.3.1", - "ieee754": "^1.1.13" - } - }, "node_modules/bundle-name": { "version": "4.1.0", "resolved": "https://registry.npmjs.org/bundle-name/-/bundle-name-4.1.0.tgz", @@ -1550,44 +1134,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/bytes": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz", - "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==", - "license": "MIT", - "engines": { - "node": ">= 0.8" - } - }, - "node_modules/call-bind-apply-helpers": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.2.tgz", - "integrity": "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==", - "license": "MIT", - "dependencies": { - "es-errors": "^1.3.0", - "function-bind": "^1.1.2" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/call-bound": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/call-bound/-/call-bound-1.0.4.tgz", - "integrity": "sha512-+ys997U96po4Kx/ABpBCqhA9EuxJaQWDQg7295H4hBphv3IZg0boBKuwYpt4YXp6MZ5AmZQnU/tyMTlRpaSejg==", - "license": "MIT", - "dependencies": { - "call-bind-apply-helpers": "^1.0.2", - "get-intrinsic": "^1.3.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/chai": { "version": "6.2.2", "resolved": "https://registry.npmjs.org/chai/-/chai-6.2.2.tgz", @@ -1610,12 +1156,6 @@ "url": "https://github.com/chalk/chalk?sponsor=1" } }, - "node_modules/chownr": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/chownr/-/chownr-1.1.4.tgz", - "integrity": "sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==", - "license": "ISC" - }, "node_modules/commander": { "version": "14.0.3", "resolved": "https://registry.npmjs.org/commander/-/commander-14.0.3.tgz", @@ -1625,28 +1165,6 @@ "node": ">=20" } }, - "node_modules/content-disposition": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-1.0.1.tgz", - "integrity": "sha512-oIXISMynqSqm241k6kcQ5UwttDILMK4BiurCfGEREw6+X9jkkpEe5T9FZaApyLGGOnFuyMWZpdolTXMtvEJ08Q==", - "license": "MIT", - "engines": { - "node": ">=18" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/express" - } - }, - "node_modules/content-type": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.5.tgz", - "integrity": "sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==", - "license": "MIT", - "engines": { - "node": ">= 0.6" - } - }, "node_modules/convert-source-map": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-2.0.0.tgz", @@ -1654,62 +1172,26 @@ "dev": true, "license": "MIT" }, - "node_modules/cookie": { - "version": "0.7.2", - "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.7.2.tgz", - "integrity": "sha512-yki5XnKuf750l50uGTllt6kKILY4nQ1eNIQatoXEByZ5dWgnKqbnqmTrBE5B4N7lrMJKQ2ytWMiTO2o0v6Ew/w==", - "license": "MIT", - "engines": { - "node": ">= 0.6" - } - }, - "node_modules/cookie-signature": { - "version": "1.2.2", - "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.2.2.tgz", - "integrity": "sha512-D76uU73ulSXrD1UXF4KE2TMxVVwhsnCgfAyTg9k8P6KGZjlXKrOLe4dJQKI3Bxi5wjesZoFXJWElNWBjPZMbhg==", + "node_modules/cross-spawn": { + "version": "7.0.6", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", + "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==", + "dev": true, "license": "MIT", + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, "engines": { - "node": ">=6.6.0" - } - }, - "node_modules/croner": { - "version": "10.0.1", - "resolved": "https://registry.npmjs.org/croner/-/croner-10.0.1.tgz", - "integrity": "sha512-ixNtAJndqh173VQ4KodSdJEI6nuioBWI0V1ITNKhZZsO0pEMoDxz539T4FTTbSZ/xIOSuDnzxLVRqBVSvPNE2g==", - "funding": [ - { - "type": "other", - "url": "https://paypal.me/hexagonpp" - }, - { - "type": "github", - "url": "https://github.com/sponsors/hexagon" - } - ], - "license": "MIT", - "engines": { - "node": ">=18.0" - } - }, - "node_modules/cross-spawn": { - "version": "7.0.6", - "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", - "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==", - "dev": true, - "license": "MIT", - "dependencies": { - "path-key": "^3.1.0", - "shebang-command": "^2.0.0", - "which": "^2.0.1" - }, - "engines": { - "node": ">= 8" + "node": ">= 8" } }, "node_modules/debug": { "version": "4.4.3", "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz", "integrity": "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==", + "dev": true, "license": "MIT", "dependencies": { "ms": "^2.1.3" @@ -1723,30 +1205,6 @@ } } }, - "node_modules/decompress-response": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-6.0.0.tgz", - "integrity": "sha512-aW35yZM6Bb/4oJlZncMH2LCoZtJXTRxES17vE3hoRiowU2kWHaJKFkSBDnDR+cm9J+9QhXmREyIfv0pji9ejCQ==", - "license": "MIT", - "dependencies": { - "mimic-response": "^3.1.0" - }, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/deep-extend": { - "version": "0.6.0", - "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz", - "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==", - "license": "MIT", - "engines": { - "node": ">=4.0.0" - } - }, "node_modules/deep-is": { "version": "0.1.4", "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", @@ -1794,80 +1252,16 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/depd": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz", - "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==", - "license": "MIT", - "engines": { - "node": ">= 0.8" - } - }, "node_modules/detect-libc": { "version": "2.1.2", "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.1.2.tgz", "integrity": "sha512-Btj2BOOO83o3WyH59e8MgXsxEQVcarkUOpEYrubB0urwnN10yQ364rsiByU11nZlqWYZm05i/of7io4mzihBtQ==", + "dev": true, "license": "Apache-2.0", "engines": { "node": ">=8" } }, - "node_modules/dunder-proto": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz", - "integrity": "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==", - "license": "MIT", - "dependencies": { - "call-bind-apply-helpers": "^1.0.1", - "es-errors": "^1.3.0", - "gopd": "^1.2.0" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/ee-first": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", - "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==", - "license": "MIT" - }, - "node_modules/encodeurl": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-2.0.0.tgz", - "integrity": "sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg==", - "license": "MIT", - "engines": { - "node": ">= 0.8" - } - }, - "node_modules/end-of-stream": { - "version": "1.4.5", - "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.5.tgz", - "integrity": "sha512-ooEGc6HP26xXq/N+GCGOT0JKCLDGrq2bQUZrQ7gyrJiZANJ/8YDTxTpQBXGMn+WbIQXNVpyWymm7KYVICQnyOg==", - "license": "MIT", - "dependencies": { - "once": "^1.4.0" - } - }, - "node_modules/es-define-property": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz", - "integrity": "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==", - "license": "MIT", - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/es-errors": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz", - "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==", - "license": "MIT", - "engines": { - "node": ">= 0.4" - } - }, "node_modules/es-module-lexer": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/es-module-lexer/-/es-module-lexer-2.0.0.tgz", @@ -1875,24 +1269,6 @@ "dev": true, "license": "MIT" }, - "node_modules/es-object-atoms": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.1.1.tgz", - "integrity": "sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA==", - "license": "MIT", - "dependencies": { - "es-errors": "^1.3.0" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/escape-html": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", - "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==", - "license": "MIT" - }, "node_modules/escape-string-regexp": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", @@ -2119,19 +1495,6 @@ "url": "https://opencollective.com/eslint" } }, - "node_modules/esprima": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", - "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", - "license": "BSD-2-Clause", - "bin": { - "esparse": "bin/esparse.js", - "esvalidate": "bin/esvalidate.js" - }, - "engines": { - "node": ">=4" - } - }, "node_modules/esquery": { "version": "1.7.0", "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.7.0.tgz", @@ -2188,24 +1551,6 @@ "node": ">=0.10.0" } }, - "node_modules/etag": { - "version": "1.8.1", - "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", - "integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==", - "license": "MIT", - "engines": { - "node": ">= 0.6" - } - }, - "node_modules/expand-template": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/expand-template/-/expand-template-2.0.3.tgz", - "integrity": "sha512-XYfuKMvj4O35f/pOXLObndIRvyQ+/+6AhODh+OKWj9S9498pHHn/IMszH+gt0fBCRWMNfk1ZSp5x3AifmnI2vg==", - "license": "(MIT OR WTFPL)", - "engines": { - "node": ">=6" - } - }, "node_modules/expect-type": { "version": "1.3.0", "resolved": "https://registry.npmjs.org/expect-type/-/expect-type-1.3.0.tgz", @@ -2216,65 +1561,11 @@ "node": ">=12.0.0" } }, - "node_modules/express": { - "version": "5.2.1", - "resolved": "https://registry.npmjs.org/express/-/express-5.2.1.tgz", - "integrity": "sha512-hIS4idWWai69NezIdRt2xFVofaF4j+6INOpJlVOLDO8zXGpUVEVzIYk12UUi2JzjEzWL3IOAxcTubgz9Po0yXw==", - "license": "MIT", - "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" - }, - "engines": { - "node": ">= 18" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/express" - } - }, - "node_modules/extend-shallow": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", - "integrity": "sha512-zCnTtlxNoAiDc3gqY2aYAWFx7XWWiasuF2K8Me5WbN8otHKTUKBwjPtNpRs/rbUZm7KxWAaNj7P1a/p52GbVug==", - "license": "MIT", - "dependencies": { - "is-extendable": "^0.1.0" - }, - "engines": { - "node": ">=0.10.0" - } - }, "node_modules/fast-deep-equal": { "version": "3.1.3", "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", + "dev": true, "license": "MIT" }, "node_modules/fast-diff": { @@ -2298,29 +1589,6 @@ "dev": true, "license": "MIT" }, - "node_modules/fast-sha256": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/fast-sha256/-/fast-sha256-1.3.0.tgz", - "integrity": "sha512-n11RGP/lrWEFI/bWdygLxhI+pVeo1ZYIVwvvPkW7azl/rOy+F3HYRZ2K5zeE9mmkhQppyv9sQFx0JM9UabnpPQ==", - "dev": true, - "license": "Unlicense" - }, - "node_modules/fast-uri": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/fast-uri/-/fast-uri-3.1.0.tgz", - "integrity": "sha512-iPeeDKJSWf4IEOasVVrknXpaBV0IApz/gp7S2bb7Z4Lljbl2MGJRqInZiUrQwV16cpzw/D3S5j5Julj/gT52AA==", - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/fastify" - }, - { - "type": "opencollective", - "url": "https://opencollective.com/fastify" - } - ], - "license": "BSD-3-Clause" - }, "node_modules/fdir": { "version": "6.5.0", "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.5.0.tgz", @@ -2352,33 +1620,6 @@ "node": ">=16.0.0" } }, - "node_modules/file-uri-to-path": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/file-uri-to-path/-/file-uri-to-path-1.0.0.tgz", - "integrity": "sha512-0Zt+s3L7Vf1biwWZ29aARiVYLx7iMGnEUl9x33fbB/j3jR81u/O2LbqK+Bm1CDSNDKVtJ/YjwY7TUd5SkeLQLw==", - "license": "MIT" - }, - "node_modules/finalhandler": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-2.1.1.tgz", - "integrity": "sha512-S8KoZgRZN+a5rNwqTxlZZePjT/4cnm0ROV70LedRHZ0p8u9fRID0hJUZQpkKLzro8LfmC8sx23bY6tVNxv8pQA==", - "license": "MIT", - "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" - }, - "engines": { - "node": ">= 18.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/express" - } - }, "node_modules/find-up": { "version": "5.0.0", "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", @@ -2417,30 +1658,6 @@ "dev": true, "license": "ISC" }, - "node_modules/forwarded": { - "version": "0.2.0", - "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz", - "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==", - "license": "MIT", - "engines": { - "node": ">= 0.6" - } - }, - "node_modules/fresh": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/fresh/-/fresh-2.0.0.tgz", - "integrity": "sha512-Rx/WycZ60HOaqLKAi6cHRKKI7zxWbJ31MhntmtwMoaTeF7XFH9hhBp8vITaMidfljRQ6eYWCKkaTK+ykVJHP2A==", - "license": "MIT", - "engines": { - "node": ">= 0.8" - } - }, - "node_modules/fs-constants": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/fs-constants/-/fs-constants-1.0.0.tgz", - "integrity": "sha512-y6OAwoSIf7FyjMIv94u+b5rdheZEjzR63GTyZJm5qh4Bi+2YgwLCcI/fPFZkL5PSixOt6ZNKm+w+Hfp/Bciwow==", - "license": "MIT" - }, "node_modules/fsevents": { "version": "2.3.3", "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz", @@ -2456,58 +1673,6 @@ "node": "^8.16.0 || ^10.6.0 || >=11.0.0" } }, - "node_modules/function-bind": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", - "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", - "license": "MIT", - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/get-intrinsic": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz", - "integrity": "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==", - "license": "MIT", - "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" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/get-proto": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz", - "integrity": "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==", - "license": "MIT", - "dependencies": { - "dunder-proto": "^1.0.1", - "es-object-atoms": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/github-from-package": { - "version": "0.0.0", - "resolved": "https://registry.npmjs.org/github-from-package/-/github-from-package-0.0.0.tgz", - "integrity": "sha512-SyHy3T1v2NUXn29OsWdxmK6RwHD+vkj3v8en8AOBZ1wBQ/hCAQ5bAQTD02kW4W9tUp/3Qh6J8r9EvntiyCmOOw==", - "license": "MIT" - }, "node_modules/glob-parent": { "version": "6.0.2", "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", @@ -2521,33 +1686,6 @@ "node": ">=10.13.0" } }, - "node_modules/gopd": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz", - "integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==", - "license": "MIT", - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/gray-matter": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/gray-matter/-/gray-matter-4.0.3.tgz", - "integrity": "sha512-5v6yZd4JK3eMI3FqqCouswVqwugaA9r4dNZB1wwcmrD02QkV5H0y7XBQW8QwQqEaZY1pM9aqORSORhJRdNK44Q==", - "license": "MIT", - "dependencies": { - "js-yaml": "^3.13.1", - "kind-of": "^6.0.2", - "section-matter": "^1.0.0", - "strip-bom-string": "^1.0.0" - }, - "engines": { - "node": ">=6.0" - } - }, "node_modules/has-flag": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", @@ -2558,30 +1696,6 @@ "node": ">=8" } }, - "node_modules/has-symbols": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz", - "integrity": "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==", - "license": "MIT", - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/hasown": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", - "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", - "license": "MIT", - "dependencies": { - "function-bind": "^1.1.2" - }, - "engines": { - "node": ">= 0.4" - } - }, "node_modules/html-escaper": { "version": "2.0.2", "resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.2.tgz", @@ -2589,71 +1703,6 @@ "dev": true, "license": "MIT" }, - "node_modules/http-errors": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.1.tgz", - "integrity": "sha512-4FbRdAX+bSdmo4AUFuS0WNiPz8NgFt+r8ThgNWmlrjQjt1Q7ZR9+zTlce2859x4KSXrwIsaeTqDoKQmtP8pLmQ==", - "license": "MIT", - "dependencies": { - "depd": "~2.0.0", - "inherits": "~2.0.4", - "setprototypeof": "~1.2.0", - "statuses": "~2.0.2", - "toidentifier": "~1.0.1" - }, - "engines": { - "node": ">= 0.8" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/express" - } - }, - "node_modules/iceberg-js": { - "version": "0.8.1", - "resolved": "https://registry.npmjs.org/iceberg-js/-/iceberg-js-0.8.1.tgz", - "integrity": "sha512-1dhVQZXhcHje7798IVM+xoo/1ZdVfzOMIc8/rgVSijRK38EDqOJoGula9N/8ZI5RD8QTxNQtK/Gozpr+qUqRRA==", - "license": "MIT", - "engines": { - "node": ">=20.0.0" - } - }, - "node_modules/iconv-lite": { - "version": "0.7.2", - "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.7.2.tgz", - "integrity": "sha512-im9DjEDQ55s9fL4EYzOAv0yMqmMBSZp6G0VvFyTMPKWxiSBHUj9NW/qqLmXUwXrrM7AvqSlTCfvqRb0cM8yYqw==", - "license": "MIT", - "dependencies": { - "safer-buffer": ">= 2.1.2 < 3.0.0" - }, - "engines": { - "node": ">=0.10.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/express" - } - }, - "node_modules/ieee754": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz", - "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==", - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ], - "license": "BSD-3-Clause" - }, "node_modules/ignore": { "version": "7.0.5", "resolved": "https://registry.npmjs.org/ignore/-/ignore-7.0.5.tgz", @@ -2674,27 +1723,6 @@ "node": ">=0.8.19" } }, - "node_modules/inherits": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", - "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", - "license": "ISC" - }, - "node_modules/ini": { - "version": "1.3.8", - "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.8.tgz", - "integrity": "sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==", - "license": "ISC" - }, - "node_modules/ipaddr.js": { - "version": "1.9.1", - "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", - "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==", - "license": "MIT", - "engines": { - "node": ">= 0.10" - } - }, "node_modules/is-docker": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/is-docker/-/is-docker-3.0.0.tgz", @@ -2710,15 +1738,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/is-extendable": { - "version": "0.1.1", - "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-0.1.1.tgz", - "integrity": "sha512-5BMULNob1vgFX6EjQw5izWDxrecWK9AM72rugNr0TFldMOi0fj6Jk+zeKIt0xGj4cEfQIJth4w3OKWOJ4f+AFw==", - "license": "MIT", - "engines": { - "node": ">=0.10.0" - } - }, "node_modules/is-extglob": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", @@ -2772,12 +1791,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/is-promise": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-4.0.0.tgz", - "integrity": "sha512-hvpoI6korhJMnej285dSg6nu1+e6uxs7zG3BYAm5byqDsgJNWwxzM6z6iZiAgQR4TJ30JmBTOwqZUw3WlyH3AQ==", - "license": "MIT" - }, "node_modules/is-wsl": { "version": "3.1.1", "resolved": "https://registry.npmjs.org/is-wsl/-/is-wsl-3.1.1.tgz", @@ -2839,15 +1852,6 @@ "node": ">=8" } }, - "node_modules/jiti": { - "version": "2.7.0", - "resolved": "https://registry.npmjs.org/jiti/-/jiti-2.7.0.tgz", - "integrity": "sha512-AC/7JofJvZGrrneWNaEnJeOLUx+JlGt7tNa0wZiRPT4MY1wmfKjt2+6O2p2uz2+skll8OZZmJMNqeke7kKbNgQ==", - "license": "MIT", - "bin": { - "jiti": "lib/jiti-cli.mjs" - } - }, "node_modules/js-tokens": { "version": "10.0.0", "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-10.0.0.tgz", @@ -2855,19 +1859,6 @@ "dev": true, "license": "MIT" }, - "node_modules/js-yaml": { - "version": "3.14.2", - "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.2.tgz", - "integrity": "sha512-PMSmkqxr106Xa156c2M265Z+FTrPl+oxd/rgOQy2tijQeK5TxQ43psO1ZCwhVOSdnn+RzkzlRz/eY4BgJBYVpg==", - "license": "MIT", - "dependencies": { - "argparse": "^1.0.7", - "esprima": "^4.0.0" - }, - "bin": { - "js-yaml": "bin/js-yaml.js" - } - }, "node_modules/json-buffer": { "version": "3.0.1", "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.1.tgz", @@ -2875,26 +1866,6 @@ "dev": true, "license": "MIT" }, - "node_modules/json-schema-to-ts": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/json-schema-to-ts/-/json-schema-to-ts-3.1.1.tgz", - "integrity": "sha512-+DWg8jCJG2TEnpy7kOm/7/AxaYoaRbjVB4LFZLySZlWn8exGs3A4OLJR966cVvU26N7X9TWxl+Jsw7dzAqKT6g==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/runtime": "^7.18.3", - "ts-algebra": "^2.0.0" - }, - "engines": { - "node": ">=16" - } - }, - "node_modules/json-schema-traverse": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz", - "integrity": "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==", - "license": "MIT" - }, "node_modules/json-stable-stringify-without-jsonify": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", @@ -2912,15 +1883,6 @@ "json-buffer": "3.0.1" } }, - "node_modules/kind-of": { - "version": "6.0.3", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz", - "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==", - "license": "MIT", - "engines": { - "node": ">=0.10.0" - } - }, "node_modules/levn": { "version": "0.4.1", "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", @@ -3250,73 +2212,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/math-intrinsics": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz", - "integrity": "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==", - "license": "MIT", - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/media-typer": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-1.1.0.tgz", - "integrity": "sha512-aisnrDP4GNe06UcKFnV5bfMNPBUw4jsLGaWwWfnH3v02GnBuXX2MCVn5RbrWo0j3pczUilYblq7fQ7Nw2t5XKw==", - "license": "MIT", - "engines": { - "node": ">= 0.8" - } - }, - "node_modules/merge-descriptors": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-2.0.0.tgz", - "integrity": "sha512-Snk314V5ayFLhp3fkUREub6WtjBfPdCPY1Ln8/8munuLuiYhsABgBVWsozAG+MWMbVEvcdcpbi9R7ww22l9Q3g==", - "license": "MIT", - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/mime-db": { - "version": "1.54.0", - "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.54.0.tgz", - "integrity": "sha512-aU5EJuIN2WDemCcAp2vFBfp/m4EAhWJnUNSSw0ixs7/kXbd6Pg64EmwJkNdFhB8aWt1sH2CTXrLxo/iAGV3oPQ==", - "license": "MIT", - "engines": { - "node": ">= 0.6" - } - }, - "node_modules/mime-types": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-3.0.2.tgz", - "integrity": "sha512-Lbgzdk0h4juoQ9fCKXW4by0UJqj+nOOrI9MJ1sSj4nI8aI2eo1qmvQEie4VD1glsS250n15LsWsYtCugiStS5A==", - "license": "MIT", - "dependencies": { - "mime-db": "^1.54.0" - }, - "engines": { - "node": ">=18" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/express" - } - }, - "node_modules/mimic-response": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-3.1.0.tgz", - "integrity": "sha512-z0yWI+4FDrrweS8Zmt4Ej5HdJmky15+L2e6Wgn3+iK5fWzb6T3fhNFq2+MeTRb064c6Wr4N/wv0DzQTjNzHNGQ==", - "license": "MIT", - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, "node_modules/minimatch": { "version": "10.2.4", "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-10.2.4.tgz", @@ -3333,25 +2228,11 @@ "url": "https://github.com/sponsors/isaacs" } }, - "node_modules/minimist": { - "version": "1.2.8", - "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.8.tgz", - "integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==", - "license": "MIT", - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/mkdirp-classic": { - "version": "0.5.3", - "resolved": "https://registry.npmjs.org/mkdirp-classic/-/mkdirp-classic-0.5.3.tgz", - "integrity": "sha512-gKLcREMhtuZRwRAfqP3RFW+TK4JqApVBtOIftVgjuABpAtpxhPGaDcfvbhNvD0B8iD1oUr/txX35NjcaY6Ns/A==", - "license": "MIT" - }, "node_modules/ms": { "version": "2.1.3", "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "dev": true, "license": "MIT" }, "node_modules/nanoid": { @@ -3373,12 +2254,6 @@ "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1" } }, - "node_modules/napi-build-utils": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/napi-build-utils/-/napi-build-utils-2.0.0.tgz", - "integrity": "sha512-GEbrYkbfF7MoNaoh2iGG84Mnf/WZfB0GdGEsM8wz7Expx/LlWf5U8t9nvJKXSp3qr5IsEbK04cBGhol/KwOsWA==", - "license": "MIT" - }, "node_modules/natural-compare": { "version": "1.4.0", "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", @@ -3386,39 +2261,6 @@ "dev": true, "license": "MIT" }, - "node_modules/negotiator": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-1.0.0.tgz", - "integrity": "sha512-8Ofs/AUQh8MaEcrlq5xOX0CQ9ypTF5dl78mjlMNfOK08fzpgTHQRQPBxcPlEtIw0yRpws+Zo/3r+5WRby7u3Gg==", - "license": "MIT", - "engines": { - "node": ">= 0.6" - } - }, - "node_modules/node-abi": { - "version": "3.89.0", - "resolved": "https://registry.npmjs.org/node-abi/-/node-abi-3.89.0.tgz", - "integrity": "sha512-6u9UwL0HlAl21+agMN3YAMXcKByMqwGx+pq+P76vii5f7hTPtKDp08/H9py6DY+cfDw7kQNTGEj/rly3IgbNQA==", - "license": "MIT", - "dependencies": { - "semver": "^7.3.5" - }, - "engines": { - "node": ">=10" - } - }, - "node_modules/object-inspect": { - "version": "1.13.4", - "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.4.tgz", - "integrity": "sha512-W67iLl4J2EXEGTbfeHCffrjDfitvLANg0UlX3wFUUSTx92KXRFegMHUVgSqE+wvhAbi4WqjGg9czysTV2Epbew==", - "license": "MIT", - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/obug": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/obug/-/obug-2.1.1.tgz", @@ -3430,27 +2272,6 @@ ], "license": "MIT" }, - "node_modules/on-finished": { - "version": "2.4.1", - "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz", - "integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==", - "license": "MIT", - "dependencies": { - "ee-first": "1.1.1" - }, - "engines": { - "node": ">= 0.8" - } - }, - "node_modules/once": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", - "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", - "license": "ISC", - "dependencies": { - "wrappy": "1" - } - }, "node_modules/open": { "version": "11.0.0", "resolved": "https://registry.npmjs.org/open/-/open-11.0.0.tgz", @@ -3521,15 +2342,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/parseurl": { - "version": "1.3.3", - "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", - "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==", - "license": "MIT", - "engines": { - "node": ">= 0.8" - } - }, "node_modules/path-exists": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", @@ -3550,16 +2362,6 @@ "node": ">=8" } }, - "node_modules/path-to-regexp": { - "version": "8.3.0", - "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-8.3.0.tgz", - "integrity": "sha512-7jdwVIRtsP8MYpdXSwOS0YdD0Du+qOoF/AEPIt88PcCFrZCzx41oxku1jD88hZBwbNUIEfpqvuhjFaMAqMTWnA==", - "license": "MIT", - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/express" - } - }, "node_modules/pathe": { "version": "2.0.3", "resolved": "https://registry.npmjs.org/pathe/-/pathe-2.0.3.tgz", @@ -3628,33 +2430,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/prebuild-install": { - "version": "7.1.3", - "resolved": "https://registry.npmjs.org/prebuild-install/-/prebuild-install-7.1.3.tgz", - "integrity": "sha512-8Mf2cbV7x1cXPUILADGI3wuhfqWvtiLA1iclTDbFRZkgRQS0NqsPZphna9V+HyTEadheuPmjaJMsbzKQFOzLug==", - "deprecated": "No longer maintained. Please contact the author of the relevant native addon; alternatives are available.", - "license": "MIT", - "dependencies": { - "detect-libc": "^2.0.0", - "expand-template": "^2.0.3", - "github-from-package": "0.0.0", - "minimist": "^1.2.3", - "mkdirp-classic": "^0.5.3", - "napi-build-utils": "^2.0.0", - "node-abi": "^3.3.0", - "pump": "^3.0.0", - "rc": "^1.2.7", - "simple-get": "^4.0.0", - "tar-fs": "^2.0.0", - "tunnel-agent": "^0.6.0" - }, - "bin": { - "prebuild-install": "bin.js" - }, - "engines": { - "node": ">=10" - } - }, "node_modules/prelude-ls": { "version": "1.2.1", "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", @@ -3694,29 +2469,6 @@ "node": ">=6.0.0" } }, - "node_modules/proxy-addr": { - "version": "2.0.7", - "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", - "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", - "license": "MIT", - "dependencies": { - "forwarded": "0.2.0", - "ipaddr.js": "1.9.1" - }, - "engines": { - "node": ">= 0.10" - } - }, - "node_modules/pump": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.4.tgz", - "integrity": "sha512-VS7sjc6KR7e1ukRFhQSY5LM2uBWAUPiOPa/A3mkKmiMwSmRFUITt0xuj+/lesgnCv+dPIEYlkzrcyXgquIHMcA==", - "license": "MIT", - "dependencies": { - "end-of-stream": "^1.1.0", - "once": "^1.3.1" - } - }, "node_modules/punycode": { "version": "2.3.1", "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz", @@ -3727,83 +2479,6 @@ "node": ">=6" } }, - "node_modules/qs": { - "version": "6.15.0", - "resolved": "https://registry.npmjs.org/qs/-/qs-6.15.0.tgz", - "integrity": "sha512-mAZTtNCeetKMH+pSjrb76NAM8V9a05I9aBZOHztWy/UqcJdQYNsf59vrRKWnojAT9Y+GbIvoTBC++CPHqpDBhQ==", - "license": "BSD-3-Clause", - "dependencies": { - "side-channel": "^1.1.0" - }, - "engines": { - "node": ">=0.6" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/range-parser": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", - "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==", - "license": "MIT", - "engines": { - "node": ">= 0.6" - } - }, - "node_modules/raw-body": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-3.0.2.tgz", - "integrity": "sha512-K5zQjDllxWkf7Z5xJdV0/B0WTNqx6vxG70zJE4N0kBs4LovmEYWJzQGxC9bS9RAKu3bgM40lrd5zoLJ12MQ5BA==", - "license": "MIT", - "dependencies": { - "bytes": "~3.1.2", - "http-errors": "~2.0.1", - "iconv-lite": "~0.7.0", - "unpipe": "~1.0.0" - }, - "engines": { - "node": ">= 0.10" - } - }, - "node_modules/rc": { - "version": "1.2.8", - "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz", - "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==", - "license": "(BSD-2-Clause OR MIT OR Apache-2.0)", - "dependencies": { - "deep-extend": "^0.6.0", - "ini": "~1.3.0", - "minimist": "^1.2.0", - "strip-json-comments": "~2.0.1" - }, - "bin": { - "rc": "cli.js" - } - }, - "node_modules/readable-stream": { - "version": "3.6.2", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.2.tgz", - "integrity": "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==", - "license": "MIT", - "dependencies": { - "inherits": "^2.0.3", - "string_decoder": "^1.1.1", - "util-deprecate": "^1.0.1" - }, - "engines": { - "node": ">= 6" - } - }, - "node_modules/require-from-string": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/require-from-string/-/require-from-string-2.0.2.tgz", - "integrity": "sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw==", - "license": "MIT", - "engines": { - "node": ">=0.10.0" - } - }, "node_modules/rolldown": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/rolldown/-/rolldown-1.0.2.tgz", @@ -3838,22 +2513,6 @@ "@rolldown/binding-win32-x64-msvc": "1.0.2" } }, - "node_modules/router": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/router/-/router-2.2.0.tgz", - "integrity": "sha512-nLTrUKm2UyiL7rlhapu/Zl45FwNgkZGaCpZbIHajDYgwlJCOzLSk+cIPAnsEqV955GjILJnKbdQC1nVPz+gAYQ==", - "license": "MIT", - "dependencies": { - "debug": "^4.4.0", - "depd": "^2.0.0", - "is-promise": "^4.0.0", - "parseurl": "^1.3.3", - "path-to-regexp": "^8.0.0" - }, - "engines": { - "node": ">= 18" - } - }, "node_modules/run-applescript": { "version": "7.1.0", "resolved": "https://registry.npmjs.org/run-applescript/-/run-applescript-7.1.0.tgz", @@ -3866,49 +2525,11 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/safe-buffer": { - "version": "5.2.1", - "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", - "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ], - "license": "MIT" - }, - "node_modules/safer-buffer": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", - "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", - "license": "MIT" - }, - "node_modules/section-matter": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/section-matter/-/section-matter-1.0.0.tgz", - "integrity": "sha512-vfD3pmTzGpufjScBh50YHKzEu2lxBWhVEHsNGoEXmCmn2hKGfeNLYMzCJpe8cD7gqX7TJluOVpBkAequ6dgMmA==", - "license": "MIT", - "dependencies": { - "extend-shallow": "^2.0.1", - "kind-of": "^6.0.0" - }, - "engines": { - "node": ">=4" - } - }, "node_modules/semver": { "version": "7.7.4", "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.4.tgz", "integrity": "sha512-vFKC2IEtQnVhpT78h1Yp8wzwrf8CM+MzKMHGJZfBtzhZNycRFnXsHk6E5TxIkkMsgNS7mdX3AGB7x2QM2di4lA==", + "dev": true, "license": "ISC", "bin": { "semver": "bin/semver.js" @@ -3917,57 +2538,6 @@ "node": ">=10" } }, - "node_modules/send": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/send/-/send-1.2.1.tgz", - "integrity": "sha512-1gnZf7DFcoIcajTjTwjwuDjzuz4PPcY2StKPlsGAQ1+YH20IRVrBaXSWmdjowTJ6u8Rc01PoYOGHXfP1mYcZNQ==", - "license": "MIT", - "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" - }, - "engines": { - "node": ">= 18" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/express" - } - }, - "node_modules/serve-static": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-2.2.1.tgz", - "integrity": "sha512-xRXBn0pPqQTVQiC8wyQrKs2MOlX24zQ0POGaj0kultvoOCstBQM5yvOhAVSUwOMjQtTvsPWoNCHfPGwaaQJhTw==", - "license": "MIT", - "dependencies": { - "encodeurl": "^2.0.0", - "escape-html": "^1.0.3", - "parseurl": "^1.3.3", - "send": "^1.2.0" - }, - "engines": { - "node": ">= 18" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/express" - } - }, - "node_modules/setprototypeof": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz", - "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==", - "license": "ISC" - }, "node_modules/shebang-command": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", @@ -3991,78 +2561,6 @@ "node": ">=8" } }, - "node_modules/side-channel": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.1.0.tgz", - "integrity": "sha512-ZX99e6tRweoUXqR+VBrslhda51Nh5MTQwou5tnUDgbtyM0dBgmhEDtWGP/xbKn6hqfPRHujUNwz5fy/wbbhnpw==", - "license": "MIT", - "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" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/side-channel-list": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/side-channel-list/-/side-channel-list-1.0.0.tgz", - "integrity": "sha512-FCLHtRD/gnpCiCHEiJLOwdmFP+wzCmDEkc9y7NsYxeF4u7Btsn1ZuwgwJGxImImHicJArLP4R0yX4c2KCrMrTA==", - "license": "MIT", - "dependencies": { - "es-errors": "^1.3.0", - "object-inspect": "^1.13.3" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/side-channel-map": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/side-channel-map/-/side-channel-map-1.0.1.tgz", - "integrity": "sha512-VCjCNfgMsby3tTdo02nbjtM/ewra6jPHmpThenkTYh8pG9ucZ/1P8So4u4FGBek/BjpOVsDCMoLA/iuBKIFXRA==", - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2", - "es-errors": "^1.3.0", - "get-intrinsic": "^1.2.5", - "object-inspect": "^1.13.3" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/side-channel-weakmap": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/side-channel-weakmap/-/side-channel-weakmap-1.0.2.tgz", - "integrity": "sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A==", - "license": "MIT", - "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" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/siginfo": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/siginfo/-/siginfo-2.0.0.tgz", @@ -4070,51 +2568,6 @@ "dev": true, "license": "ISC" }, - "node_modules/simple-concat": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/simple-concat/-/simple-concat-1.0.1.tgz", - "integrity": "sha512-cSFtAPtRhljv69IK0hTVZQ+OfE9nePi/rtJmw5UjHeVyVroEqJXP1sFztKUy1qU+xvz3u/sfYJLa947b7nAN2Q==", - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ], - "license": "MIT" - }, - "node_modules/simple-get": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/simple-get/-/simple-get-4.0.1.tgz", - "integrity": "sha512-brv7p5WgH0jmQJr1ZDDfKDOSeWWg+OVypG99A/5vYGPqJ6pxiaHLy8nxtFjBA7oMa01ebA9gfh1uMCFqOuXxvA==", - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ], - "license": "MIT", - "dependencies": { - "decompress-response": "^6.0.0", - "once": "^1.3.1", - "simple-concat": "^1.0.0" - } - }, "node_modules/source-map-js": { "version": "1.2.1", "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.1.tgz", @@ -4125,12 +2578,6 @@ "node": ">=0.10.0" } }, - "node_modules/sprintf-js": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", - "integrity": "sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==", - "license": "BSD-3-Clause" - }, "node_modules/stackback": { "version": "0.0.2", "resolved": "https://registry.npmjs.org/stackback/-/stackback-0.0.2.tgz", @@ -4138,26 +2585,6 @@ "dev": true, "license": "MIT" }, - "node_modules/standardwebhooks": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/standardwebhooks/-/standardwebhooks-1.0.0.tgz", - "integrity": "sha512-BbHGOQK9olHPMvQNHWul6MYlrRTAOKn03rOe4A8O3CLWhNf4YHBqq2HJKKC+sfqpxiBY52pNeesD6jIiLDz8jg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@stablelib/base64": "^1.0.0", - "fast-sha256": "^1.3.0" - } - }, - "node_modules/statuses": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.2.tgz", - "integrity": "sha512-DvEy55V3DB7uknRo+4iOGT5fP1slR8wQohVdknigZPMpMstaKJQWhwiYBACJE3Ul2pTnATihhBYnRhZQHGBiRw==", - "license": "MIT", - "engines": { - "node": ">= 0.8" - } - }, "node_modules/std-env": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/std-env/-/std-env-4.0.0.tgz", @@ -4165,33 +2592,6 @@ "dev": true, "license": "MIT" }, - "node_modules/string_decoder": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz", - "integrity": "sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==", - "license": "MIT", - "dependencies": { - "safe-buffer": "~5.2.0" - } - }, - "node_modules/strip-bom-string": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/strip-bom-string/-/strip-bom-string-1.0.0.tgz", - "integrity": "sha512-uCC2VHvQRYu+lMh4My/sFNmF2klFymLX1wHJeXnbEJERpV/ZsVuonzerjfrGpIGF7LBVa1O7i9kjiWvJiFck8g==", - "license": "MIT", - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/strip-json-comments": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", - "integrity": "sha512-4gB8na07fecVVkOI6Rs4e7T6NOTki5EmL7TUduTs6bu3EdnSycntVJ4re8kgZA+wx9IueI2Y11bfbgwtzuE0KQ==", - "license": "MIT", - "engines": { - "node": ">=0.10.0" - } - }, "node_modules/supports-color": { "version": "7.2.0", "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", @@ -4221,34 +2621,6 @@ "url": "https://opencollective.com/synckit" } }, - "node_modules/tar-fs": { - "version": "2.1.4", - "resolved": "https://registry.npmjs.org/tar-fs/-/tar-fs-2.1.4.tgz", - "integrity": "sha512-mDAjwmZdh7LTT6pNleZ05Yt65HC3E+NiQzl672vQG38jIrehtJk/J3mNwIg+vShQPcLF/LV7CMnDW6vjj6sfYQ==", - "license": "MIT", - "dependencies": { - "chownr": "^1.1.1", - "mkdirp-classic": "^0.5.2", - "pump": "^3.0.0", - "tar-stream": "^2.1.4" - } - }, - "node_modules/tar-stream": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/tar-stream/-/tar-stream-2.2.0.tgz", - "integrity": "sha512-ujeqbceABgwMZxEJnk2HDY2DlnUZ+9oEcb1KzTVfYHio0UE6dG71n60d8D2I4qNvleWrrXpmjpt7vZeF1LnMZQ==", - "license": "MIT", - "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" - }, - "engines": { - "node": ">=6" - } - }, "node_modules/tinybench": { "version": "2.9.0", "resolved": "https://registry.npmjs.org/tinybench/-/tinybench-2.9.0.tgz", @@ -4293,22 +2665,6 @@ "node": ">=14.0.0" } }, - "node_modules/toidentifier": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz", - "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==", - "license": "MIT", - "engines": { - "node": ">=0.6" - } - }, - "node_modules/ts-algebra": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ts-algebra/-/ts-algebra-2.0.0.tgz", - "integrity": "sha512-FPAhNPFMrkwz76P7cdjdmiShwMynZYN6SgOujD1urY4oNm80Ou9oMdmbR45LotcKOXoy7wSmHkRFE6Mxbrhefw==", - "dev": true, - "license": "MIT" - }, "node_modules/ts-api-utils": { "version": "2.5.0", "resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-2.5.0.tgz", @@ -4326,19 +2682,9 @@ "version": "2.8.1", "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.1.tgz", "integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==", - "license": "0BSD" - }, - "node_modules/tunnel-agent": { - "version": "0.6.0", - "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz", - "integrity": "sha512-McnNiV1l8RYeY8tBgEpuodCC1mLUdbSN+CYBL7kJsJNInOP8UjDDEwdk6Mw60vdLLrr5NHKZhMAOSrR2NZuQ+w==", - "license": "Apache-2.0", - "dependencies": { - "safe-buffer": "^5.0.1" - }, - "engines": { - "node": "*" - } + "dev": true, + "license": "0BSD", + "optional": true }, "node_modules/type-check": { "version": "0.4.0", @@ -4353,20 +2699,6 @@ "node": ">= 0.8.0" } }, - "node_modules/type-is": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/type-is/-/type-is-2.0.1.tgz", - "integrity": "sha512-OZs6gsjF4vMp32qrCbiVSkrFmXtG/AZhY3t0iAMrMBiAZyV9oALtXO8hsrHbMXF9x6L3grlFuwW2oAz7cav+Gw==", - "license": "MIT", - "dependencies": { - "content-type": "^1.0.5", - "media-typer": "^1.1.0", - "mime-types": "^3.0.0" - }, - "engines": { - "node": ">= 0.6" - } - }, "node_modules/typescript": { "version": "5.9.3", "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.9.3.tgz", @@ -4388,15 +2720,6 @@ "dev": true, "license": "MIT" }, - "node_modules/unpipe": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", - "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==", - "license": "MIT", - "engines": { - "node": ">= 0.8" - } - }, "node_modules/uri-js": { "version": "4.4.1", "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", @@ -4407,21 +2730,6 @@ "punycode": "^2.1.0" } }, - "node_modules/util-deprecate": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", - "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==", - "license": "MIT" - }, - "node_modules/vary": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", - "integrity": "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==", - "license": "MIT", - "engines": { - "node": ">= 0.8" - } - }, "node_modules/vite": { "version": "8.0.14", "resolved": "https://registry.npmjs.org/vite/-/vite-8.0.14.tgz", @@ -4633,33 +2941,6 @@ "node": ">=0.10.0" } }, - "node_modules/wrappy": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", - "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", - "license": "ISC" - }, - "node_modules/ws": { - "version": "8.21.0", - "resolved": "https://registry.npmjs.org/ws/-/ws-8.21.0.tgz", - "integrity": "sha512-Vsp28b7DRcimFQvrqu2Wek3z1iYxDCWqHYB8Qsnk/S4RfaCQzPGPyBNuVjJV3cd6UiKtUtp6sNM77gWvzcCH+g==", - "license": "MIT", - "engines": { - "node": ">=10.0.0" - }, - "peerDependencies": { - "bufferutil": "^4.0.1", - "utf-8-validate": ">=5.0.2" - }, - "peerDependenciesMeta": { - "bufferutil": { - "optional": true - }, - "utf-8-validate": { - "optional": true - } - } - }, "node_modules/wsl-utils": { "version": "0.3.1", "resolved": "https://registry.npmjs.org/wsl-utils/-/wsl-utils-0.3.1.tgz", diff --git a/package.json b/package.json index 9538bf5..2034400 100644 --- a/package.json +++ b/package.json @@ -1,7 +1,7 @@ { "name": "@agentage/cli", - "version": "0.24.6", - "description": "Agentage CLI and daemon — control plane for AI agents", + "version": "0.25.0", + "description": "agentage Memory CLI - connect, clone, and manage your memory from the terminal", "type": "module", "main": "./dist/index.js", "bin": { @@ -29,26 +29,12 @@ "prepublishOnly": "npm run verify" }, "dependencies": { - "@agentage/core": "^0.10.0", - "@agentage/platform": "^0.6.1", - "@supabase/supabase-js": "2.106.2", - "ajv": "8.20.0", - "better-sqlite3": "latest", "chalk": "latest", "commander": "latest", - "croner": "*", - "express": "latest", - "gray-matter": "latest", - "jiti": "latest", - "open": "latest", - "ws": "latest" + "open": "latest" }, "devDependencies": { - "@anthropic-ai/sdk": "latest", - "@types/better-sqlite3": "latest", - "@types/express": "latest", "@types/node": "latest", - "@types/ws": "latest", "@typescript-eslint/eslint-plugin": "latest", "@typescript-eslint/parser": "latest", "@vitest/coverage-v8": "latest", diff --git a/src/cli.ts b/src/cli.ts index 8bcab2d..3136e41 100644 --- a/src/cli.ts +++ b/src/cli.ts @@ -1,74 +1,21 @@ #!/usr/bin/env node import { Command } from 'commander'; -import { VERSION } from './utils/version.js'; -import { registerRun } from './commands/run.js'; -import { registerAgents } from './commands/agents.js'; -import { registerRuns } from './commands/runs.js'; -import { registerMachines } from './commands/machines.js'; -import { registerStatus } from './commands/status.js'; -import { registerLogs } from './commands/logs.js'; -import { registerDaemon } from './commands/daemon-cmd.js'; -import { registerWhoami } from './commands/whoami.js'; -import { registerCompletions } from './commands/completions.js'; -import { registerConfig } from './commands/config-cmd.js'; import { registerSetup } from './commands/setup.js'; -import { createCreateCommand } from './commands/create.js'; -import { registerUpdate } from './commands/update.js'; -import { registerProjects } from './commands/projects.js'; -import { registerSchedules } from './commands/schedules.js'; -import { registerVaults } from './commands/vault.js'; -import { checkForUpdateSafe, type UpdateCheckResult } from './utils/update-checker.js'; +import { registerStatus } from './commands/status.js'; +import { VERSION } from './utils/version.js'; const program = new Command(); -// Kick off background update check (non-blocking) -const updateCheckPromise: Promise = checkForUpdateSafe(); - -program.name('agentage').description('Agentage CLI — control plane for AI agents').version(VERSION); +program + .name('agentage') + .description('agentage Memory CLI - one memory, every AI, owned by you') + .version(VERSION); -registerRun(program); -registerAgents(program); -registerRuns(program); -registerMachines(program); -registerStatus(program); -registerLogs(program); -registerDaemon(program); -registerWhoami(program); -registerCompletions(program); -registerConfig(program); registerSetup(program); -program.addCommand(createCreateCommand()); -registerUpdate(program); -registerProjects(program); -registerSchedules(program); -registerVaults(program); - -// Show the update notice only on an interactive terminal. Appending it to -// non-TTY stdout (CI, pipes, `--json` consumers) corrupts machine-readable -// output — the e2e nightly's standalone suite hit exactly this when -// `latest` advanced past the installed version mid-run. The user can also -// opt out explicitly with NO_UPDATE_NOTIFIER (matching the npm convention). -const shouldShowUpdateNotice = (): boolean => - process.stdout.isTTY === true && !process.env['NO_UPDATE_NOTIFIER']; - -program.parseAsync().then(async () => { - if (shouldShowUpdateNotice()) { - const result = await updateCheckPromise; - if (result?.updateAvailable) { - const { default: chalk } = await import('chalk'); - // Route to stderr — the update notice is meta-information, not command - // output. Even with the TTY guard above, keeping stdout pure means - // consumers that capture stdout (e.g. `agentage status | jq`) never - // see this string in their pipeline. - console.error( - chalk.yellow( - `\nUpdate available: ${result.currentVersion} → ${result.latestVersion} — run ${chalk.white('agentage update')} to install.` - ) - ); - } - } +registerStatus(program); - // Force exit — forked daemon process can keep the event loop alive - setTimeout(() => process.exit(process.exitCode ?? 0), 100); +program.parseAsync().catch((err: unknown) => { + console.error(err instanceof Error ? err.message : String(err)); + process.exit(1); }); diff --git a/src/commands/agents.test.ts b/src/commands/agents.test.ts deleted file mode 100644 index 3dc67a8..0000000 --- a/src/commands/agents.test.ts +++ /dev/null @@ -1,168 +0,0 @@ -import { describe, it, expect, vi, beforeEach } from 'vitest'; -import { Command } from 'commander'; - -vi.mock('../utils/ensure-daemon.js', () => ({ - ensureDaemon: vi.fn(), -})); - -vi.mock('../utils/daemon-client.js', () => ({ - get: vi.fn(), - post: vi.fn(), -})); - -import { get, post } from '../utils/daemon-client.js'; -import { registerAgents } from './agents.js'; - -const mockGet = vi.mocked(get); -const mockPost = vi.mocked(post); - -describe('agents command', () => { - let program: Command; - let logs: string[]; - let errorLogs: string[]; - - beforeEach(() => { - vi.clearAllMocks(); - logs = []; - errorLogs = []; - vi.spyOn(console, 'log').mockImplementation((...args: unknown[]) => { - logs.push(args.map(String).join(' ')); - }); - vi.spyOn(console, 'error').mockImplementation((...args: unknown[]) => { - errorLogs.push(args.map(String).join(' ')); - }); - - program = new Command(); - program.exitOverride(); - registerAgents(program); - }); - - it('lists local agents from daemon', async () => { - mockGet.mockResolvedValue([ - { name: 'hello', description: 'A greeting agent', path: '/agents/hello.agent.md' }, - ]); - - await program.parseAsync(['node', 'agentage', 'agents']); - - expect(mockGet).toHaveBeenCalledWith('/api/agents'); - expect(logs.some((l) => l.includes('hello'))).toBe(true); - expect(logs.some((l) => l.includes('A greeting agent'))).toBe(true); - }); - - it('refreshes agents with --refresh', async () => { - mockPost.mockResolvedValue([ - { name: 'refreshed', description: 'Refreshed', path: '/agents/refreshed.agent.md' }, - ]); - - await program.parseAsync(['node', 'agentage', 'agents', '--refresh']); - - expect(mockPost).toHaveBeenCalledWith('/api/agents/refresh'); - }); - - it('outputs JSON with --json', async () => { - const agents = [{ name: 'test', description: 'Test', path: '/test' }]; - mockGet.mockResolvedValue(agents); - - await program.parseAsync(['node', 'agentage', 'agents', '--json']); - - expect(logs[0]).toBe(JSON.stringify(agents, null, 2)); - }); - - it('prints message when no agents found', async () => { - mockGet.mockResolvedValue([]); - - await program.parseAsync(['node', 'agentage', 'agents']); - - expect(logs.some((l) => l.includes('No agents discovered'))).toBe(true); - }); - - describe('--all (hub mode)', () => { - it('fetches agents from hub', async () => { - mockGet.mockResolvedValue([ - { - name: 'remote-agent', - description: 'Hub agent', - machines: { name: 'my-pc', status: 'online' }, - }, - ]); - - await program.parseAsync(['node', 'agentage', 'agents', '--all']); - - expect(mockGet).toHaveBeenCalledWith('/api/hub/agents'); - expect(logs.some((l) => l.includes('remote-agent'))).toBe(true); - }); - - it('shows error when not connected to hub', async () => { - mockGet.mockRejectedValue(new Error('Unauthorized')); - - await program.parseAsync(['node', 'agentage', 'agents', '--all']); - - expect(errorLogs.some((l) => l.includes('Not connected to hub'))).toBe(true); - expect(process.exitCode).toBe(1); - process.exitCode = undefined; - }); - - it('outputs JSON in hub mode', async () => { - const agents = [ - { name: 'hub-agent', description: 'Hub', machines: { name: 'pc', status: 'online' } }, - ]; - mockGet.mockResolvedValue(agents); - - await program.parseAsync(['node', 'agentage', 'agents', '--all', '--json']); - - expect(logs[0]).toBe(JSON.stringify(agents, null, 2)); - }); - - it('prints message when no hub agents found', async () => { - mockGet.mockResolvedValue([]); - - await program.parseAsync(['node', 'agentage', 'agents', '--all']); - - expect(logs.some((l) => l.includes('No agents found across machines'))).toBe(true); - }); - - it('shows offline status for offline machines', async () => { - mockGet.mockResolvedValue([ - { - name: 'offline-agent', - description: 'Agent', - machines: { name: 'old-pc', status: 'offline' }, - }, - ]); - - await program.parseAsync(['node', 'agentage', 'agents', '--all']); - - expect(logs.some((l) => l.includes('offline'))).toBe(true); - }); - }); - - it('shows table header with correct columns', async () => { - mockGet.mockResolvedValue([{ name: 'a', description: 'desc', path: '/p' }]); - - await program.parseAsync(['node', 'agentage', 'agents']); - - expect(logs[0]).toContain('NAME'); - expect(logs[0]).toContain('DESCRIPTION'); - expect(logs[0]).toContain('PATH'); - }); - - it('handles agents without descriptions', async () => { - mockGet.mockResolvedValue([{ name: 'nodesc', path: '/test' }]); - - await program.parseAsync(['node', 'agentage', 'agents']); - - expect(logs.some((l) => l.includes('nodesc'))).toBe(true); - }); - - it('shows agent count at bottom', async () => { - mockGet.mockResolvedValue([ - { name: 'a1', description: 'Agent 1', path: '/a1' }, - { name: 'a2', description: 'Agent 2', path: '/a2' }, - { name: 'a3', description: 'Agent 3', path: '/a3' }, - ]); - - await program.parseAsync(['node', 'agentage', 'agents']); - - expect(logs.some((l) => l.includes('3 agents discovered'))).toBe(true); - }); -}); diff --git a/src/commands/agents.ts b/src/commands/agents.ts deleted file mode 100644 index 1347121..0000000 --- a/src/commands/agents.ts +++ /dev/null @@ -1,122 +0,0 @@ -import { type Command } from 'commander'; -import chalk from 'chalk'; -import { type AgentManifest } from '@agentage/core'; -import { ensureDaemon } from '../utils/ensure-daemon.js'; -import { get, post } from '../utils/daemon-client.js'; -import { type ScanWarning } from '../discovery/scanner.js'; - -interface HubAgent { - name: string; - description?: string; - version?: string; - machines?: { name: string; status: string }; -} - -export const registerAgents = (program: Command): void => { - program - .command('agents') - .description('List discovered agents') - .option('--refresh', 'Rescan directories first') - .option('--all', 'Show agents from all connected machines') - .option('--json', 'JSON output') - .action(async (opts: { refresh?: boolean; all?: boolean; json?: boolean }) => { - await ensureDaemon(); - - if (opts.all) { - await listHubAgents(opts.json ?? false); - return; - } - - let agents: AgentManifest[]; - if (opts.refresh) { - agents = await post('/api/agents/refresh'); - } else { - agents = await get('/api/agents'); - } - - if (opts.json) { - console.log(JSON.stringify(agents, null, 2)); - return; - } - - if (agents.length === 0) { - console.log(chalk.gray('No agents discovered.')); - return; - } - - const nameWidth = Math.max(12, ...agents.map((a) => a.name.length)) + 2; - const descWidth = Math.max(12, ...agents.map((a) => (a.description || '').length)) + 2; - - console.log( - chalk.bold('NAME'.padEnd(nameWidth)) + - chalk.bold('DESCRIPTION'.padEnd(descWidth)) + - chalk.bold('PATH') - ); - - for (const agent of agents) { - const desc = (agent.description || '').substring(0, 60); - console.log(agent.name.padEnd(nameWidth) + desc.padEnd(descWidth) + chalk.gray(agent.path)); - } - - console.log(chalk.dim(`\n${agents.length} agents discovered`)); - - try { - const warnings = await get('/api/agents/warnings'); - if (warnings.length > 0) { - console.log(chalk.yellow(`\n⚠ Failed to load ${warnings.length} agent(s):`)); - for (const w of warnings) { - console.log(chalk.yellow(` ${w.file}`)); - console.log(chalk.dim(` ${w.message}`)); - } - } - } catch { - // Daemon may not support warnings endpoint yet - } - }); -}; - -const listHubAgents = async (jsonMode: boolean): Promise => { - let agents: HubAgent[]; - try { - agents = await get('/api/hub/agents'); - } catch { - console.error(chalk.red("Not connected to hub. Run 'agentage setup' first.")); - process.exitCode = 1; - return; - } - - if (jsonMode) { - console.log(JSON.stringify(agents, null, 2)); - return; - } - - if (agents.length === 0) { - console.log(chalk.gray('No agents found across machines.')); - return; - } - - const nameWidth = Math.max(12, ...agents.map((a) => a.name.length)) + 2; - const machineWidth = Math.max(10, ...agents.map((a) => (a.machines?.name || '').length)) + 2; - const descWidth = - Math.max(12, ...agents.map((a) => (a.description || '').length).slice(0, 40)) + 2; - - console.log( - chalk.bold('NAME'.padEnd(nameWidth)) + - chalk.bold('MACHINE'.padEnd(machineWidth)) + - chalk.bold('DESCRIPTION'.padEnd(descWidth)) + - chalk.bold('STATUS') - ); - - for (const agent of agents) { - const machineName = agent.machines?.name || ''; - const status = - agent.machines?.status === 'online' ? chalk.green('online') : chalk.gray('offline'); - - console.log( - agent.name.padEnd(nameWidth) + - machineName.padEnd(machineWidth) + - (agent.description || '').substring(0, 38).padEnd(descWidth) + - status - ); - } -}; diff --git a/src/commands/completions.test.ts b/src/commands/completions.test.ts deleted file mode 100644 index 005e844..0000000 --- a/src/commands/completions.test.ts +++ /dev/null @@ -1,57 +0,0 @@ -import { describe, it, expect, vi, beforeEach } from 'vitest'; -import { Command } from 'commander'; - -import { registerCompletions } from './completions.js'; - -describe('completions command', () => { - let program: Command; - let logs: string[]; - - beforeEach(() => { - vi.clearAllMocks(); - logs = []; - vi.spyOn(console, 'log').mockImplementation((...args: unknown[]) => { - logs.push(args.map(String).join(' ')); - }); - - program = new Command(); - program.exitOverride(); - registerCompletions(program); - }); - - it('generates bash completions with _agentage function', async () => { - await program.parseAsync(['node', 'agentage', 'completions', 'bash']); - - const output = logs.join('\n'); - expect(output).toContain('_agentage()'); - expect(output).toContain('complete -F _agentage agentage'); - expect(output).toContain('COMPREPLY'); - }); - - it('generates zsh completions with compdef', async () => { - await program.parseAsync(['node', 'agentage', 'completions', 'zsh']); - - const output = logs.join('\n'); - expect(output).toContain('#compdef agentage'); - expect(output).toContain('_agentage'); - }); - - it('generates fish completions with complete -c', async () => { - await program.parseAsync(['node', 'agentage', 'completions', 'fish']); - - const output = logs.join('\n'); - expect(output).toContain('complete -c agentage'); - expect(output).toContain('run'); - expect(output).toContain('agents'); - expect(output).toContain('status'); - }); - - it('bash completions include all commands', async () => { - await program.parseAsync(['node', 'agentage', 'completions', 'bash']); - - const output = logs.join('\n'); - for (const cmd of ['run', 'agents', 'runs', 'machines', 'status', 'login', 'whoami']) { - expect(output).toContain(cmd); - } - }); -}); diff --git a/src/commands/completions.ts b/src/commands/completions.ts deleted file mode 100644 index 5e36a6d..0000000 --- a/src/commands/completions.ts +++ /dev/null @@ -1,126 +0,0 @@ -import { type Command } from 'commander'; - -const COMMANDS = [ - 'run', - 'agents', - 'runs', - 'machines', - 'status', - 'login', - 'logout', - 'logs', - 'daemon', - 'whoami', - 'config', - 'completions', - 'init', -]; - -const generateBash = (): string => { - const cmds = COMMANDS.join(' '); - return `_agentage() { - local cur prev commands - COMPREPLY=() - cur="\${COMP_WORDS[COMP_CWORD]}" - prev="\${COMP_WORDS[COMP_CWORD-1]}" - commands="${cmds}" - - if [[ \${COMP_CWORD} -eq 1 ]]; then - COMPREPLY=( $(compgen -W "\${commands}" -- "\${cur}") ) - return 0 - fi - - case "\${prev}" in - agents) - COMPREPLY=( $(compgen -W "--refresh --all --json" -- "\${cur}") ) - ;; - runs) - COMPREPLY=( $(compgen -W "--all --json --filter --last" -- "\${cur}") ) - ;; - status) - COMPREPLY=( $(compgen -W "--add-dir --remove-dir --json" -- "\${cur}") ) - ;; - machines) - COMPREPLY=( $(compgen -W "--json" -- "\${cur}") ) - ;; - login) - COMPREPLY=( $(compgen -W "--hub --token" -- "\${cur}") ) - ;; - whoami) - COMPREPLY=( $(compgen -W "--json" -- "\${cur}") ) - ;; - config) - COMPREPLY=( $(compgen -W "list set" -- "\${cur}") ) - ;; - completions) - COMPREPLY=( $(compgen -W "bash zsh fish" -- "\${cur}") ) - ;; - init) - COMPREPLY=( $(compgen -W "--hub --name --dir --no-login" -- "\${cur}") ) - ;; - daemon) - COMPREPLY=( $(compgen -W "start stop restart status" -- "\${cur}") ) - ;; - esac - return 0 -} -complete -F _agentage agentage -`; -}; - -const generateZsh = (): string => { - const cmds = COMMANDS.map((c) => `'${c}:${c} command'`).join('\n '); - return `#compdef agentage - -_agentage() { - local -a commands - commands=( - ${cmds} - ) - - _arguments -C \\ - '1:command:->cmd' \\ - '*::arg:->args' - - case $state in - cmd) - _describe 'command' commands - ;; - esac -} - -_agentage "$@" -`; -}; - -const generateFish = (): string => { - const lines = COMMANDS.map( - (c) => `complete -c agentage -n '__fish_use_subcommand' -a '${c}' -d '${c} command'` - ); - return lines.join('\n') + '\n'; -}; - -export const registerCompletions = (program: Command): void => { - const cmd = program.command('completions').description('Generate shell completions'); - - cmd - .command('bash') - .description('Generate bash completions') - .action(() => { - console.log(generateBash()); - }); - - cmd - .command('zsh') - .description('Generate zsh completions') - .action(() => { - console.log(generateZsh()); - }); - - cmd - .command('fish') - .description('Generate fish completions') - .action(() => { - console.log(generateFish()); - }); -}; diff --git a/src/commands/config-cmd.test.ts b/src/commands/config-cmd.test.ts deleted file mode 100644 index 0fad825..0000000 --- a/src/commands/config-cmd.test.ts +++ /dev/null @@ -1,94 +0,0 @@ -import { describe, it, expect, vi, beforeEach } from 'vitest'; -import { Command } from 'commander'; - -vi.mock('../daemon/config.js', () => ({ - loadConfig: vi.fn(), - saveConfig: vi.fn(), -})); - -import { loadConfig, saveConfig } from '../daemon/config.js'; -import { registerConfig } from './config-cmd.js'; - -const mockLoadConfig = vi.mocked(loadConfig); -const mockSaveConfig = vi.mocked(saveConfig); - -describe('config command', () => { - let program: Command; - let logs: string[]; - - const defaultConfig = { - machine: { id: 'machine-123', name: 'test-host' }, - daemon: { port: 4243 }, - agents: { default: '/home/user/agents', additional: [] }, - projects: { default: '/home/user/projects', additional: [] }, - sync: { - events: { - state: true, - result: true, - error: true, - input_required: true, - 'output.llm.delta': true, - 'output.llm.tool_call': true, - 'output.llm.usage': true, - 'output.progress': true, - }, - }, - }; - - beforeEach(() => { - vi.clearAllMocks(); - logs = []; - vi.spyOn(console, 'log').mockImplementation((...args: unknown[]) => { - logs.push(args.map(String).join(' ')); - }); - - mockLoadConfig.mockReturnValue(structuredClone(defaultConfig)); - - program = new Command(); - program.exitOverride(); - registerConfig(program); - }); - - describe('list', () => { - it('lists all config as key=value pairs', async () => { - await program.parseAsync(['node', 'agentage', 'config', 'list']); - - expect(logs.some((l) => l.includes('machine.id=machine-123'))).toBe(true); - expect(logs.some((l) => l.includes('machine.name=test-host'))).toBe(true); - expect(logs.some((l) => l.includes('daemon.port=4243'))).toBe(true); - }); - - it('outputs JSON with --json', async () => { - await program.parseAsync(['node', 'agentage', 'config', 'list', '--json']); - - const parsed = JSON.parse(logs[0]!); - expect(parsed.machine.name).toBe('test-host'); - expect(parsed.daemon.port).toBe(4243); - }); - }); - - describe('set', () => { - it('sets a config value with dot notation', async () => { - await program.parseAsync(['node', 'agentage', 'config', 'set', 'daemon.port', '5000']); - - expect(mockSaveConfig).toHaveBeenCalledTimes(1); - const saved = mockSaveConfig.mock.calls[0]![0]; - expect(saved.daemon.port).toBe(5000); - expect(logs.some((l) => l.includes('Set daemon.port=5000'))).toBe(true); - }); - - it('sets a string value', async () => { - await program.parseAsync(['node', 'agentage', 'config', 'set', 'machine.name', 'new-host']); - - const saved = mockSaveConfig.mock.calls[0]![0]; - expect(saved.machine.name).toBe('new-host'); - }); - - it('sets a boolean value', async () => { - await program.parseAsync(['node', 'agentage', 'config', 'set', 'sync.events.state', 'false']); - - const saved = mockSaveConfig.mock.calls[0]![0]; - expect(saved.sync.events.state).toBe(false); - }); - }); -}); diff --git a/src/commands/config-cmd.ts b/src/commands/config-cmd.ts deleted file mode 100644 index 93b1c65..0000000 --- a/src/commands/config-cmd.ts +++ /dev/null @@ -1,82 +0,0 @@ -import { type Command } from 'commander'; -import chalk from 'chalk'; -import { loadConfig, saveConfig, type DaemonConfig } from '../daemon/config.js'; - -const flattenConfig = ( - obj: Record, - prefix = '' -): Array<{ key: string; value: string }> => { - const entries: Array<{ key: string; value: string }> = []; - - for (const [k, v] of Object.entries(obj)) { - const key = prefix ? `${prefix}.${k}` : k; - - if (v !== null && typeof v === 'object' && !Array.isArray(v)) { - entries.push(...flattenConfig(v as Record, key)); - } else { - entries.push({ key, value: Array.isArray(v) ? JSON.stringify(v) : String(v) }); - } - } - - return entries; -}; - -const setNestedValue = (obj: Record, path: string, value: string): void => { - const parts = path.split('.'); - let current: Record = obj; - - for (let i = 0; i < parts.length - 1; i++) { - const part = parts[i]!; - if (!(part in current) || typeof current[part] !== 'object') { - current[part] = {}; - } - current = current[part] as Record; - } - - const lastKey = parts[parts.length - 1]!; - - // Try to parse as number or boolean - if (value === 'true') { - current[lastKey] = true; - } else if (value === 'false') { - current[lastKey] = false; - } else if (/^\d+$/.test(value)) { - current[lastKey] = parseInt(value, 10); - } else { - current[lastKey] = value; - } -}; - -export const registerConfig = (program: Command): void => { - const cmd = program.command('config').description('View and update configuration'); - - cmd - .command('list') - .description('List all configuration values') - .option('--json', 'JSON output') - .action((opts: { json?: boolean }) => { - const config = loadConfig(); - - if (opts.json) { - console.log(JSON.stringify(config, null, 2)); - return; - } - - const entries = flattenConfig(config as unknown as Record); - for (const { key, value } of entries) { - console.log(`${chalk.bold(key)}=${value}`); - } - }); - - cmd - .command('set') - .description('Set a configuration value') - .argument('', 'Config key (dot notation: daemon.port, hub.url)') - .argument('', 'Value to set') - .action((key: string, value: string) => { - const config = loadConfig(); - setNestedValue(config as unknown as Record, key, value); - saveConfig(config as DaemonConfig); - console.log(chalk.green(`Set ${key}=${value}`)); - }); -}; diff --git a/src/commands/create.test.ts b/src/commands/create.test.ts deleted file mode 100644 index 4dc0336..0000000 --- a/src/commands/create.test.ts +++ /dev/null @@ -1,178 +0,0 @@ -import { describe, it, expect, beforeEach, afterEach, vi } from 'vitest'; -import { existsSync, mkdirSync, readFileSync, rmSync, writeFileSync } from 'node:fs'; -import { join } from 'node:path'; -import { tmpdir } from 'node:os'; - -const testDir = join(tmpdir(), `agentage-test-create-${Date.now()}`); -const configDir = join(testDir, '.agentage'); - -describe('create command', () => { - beforeEach(() => { - mkdirSync(testDir, { recursive: true }); - mkdirSync(configDir, { recursive: true }); - process.env['AGENTAGE_CONFIG_DIR'] = configDir; - }); - - afterEach(() => { - delete process.env['AGENTAGE_CONFIG_DIR']; - rmSync(testDir, { recursive: true, force: true }); - }); - - it('creates agent file with simple template', async () => { - const { createCreateCommand } = await import('./create.js'); - const cmd = createCreateCommand(); - await cmd.parseAsync(['node', 'test', 'my-agent', '--dir', testDir]); - - const filePath = join(testDir, 'my-agent.agent.ts'); - expect(existsSync(filePath)).toBe(true); - - const content = readFileSync(filePath, 'utf-8'); - expect(content).toContain("name: 'my-agent'"); - expect(content).toContain('import { agent'); - }); - - it('creates agent with shell template', async () => { - const { createCreateCommand } = await import('./create.js'); - const cmd = createCreateCommand(); - await cmd.parseAsync(['node', 'test', 'my-shell', '--template', 'shell', '--dir', testDir]); - - const content = readFileSync(join(testDir, 'my-shell.agent.ts'), 'utf-8'); - expect(content).toContain("name: 'my-shell'"); - expect(content).toContain('yield* shell('); - }); - - it('creates agent with llm template', async () => { - const { createCreateCommand } = await import('./create.js'); - const cmd = createCreateCommand(); - await cmd.parseAsync(['node', 'test', 'my-llm', '--template', 'llm', '--dir', testDir]); - - const content = readFileSync(join(testDir, 'my-llm.agent.ts'), 'utf-8'); - expect(content).toContain("name: 'my-llm'"); - expect(content).toContain("model: 'claude-sonnet-4-6'"); - expect(content).not.toContain('async *run'); - }); - - it('rejects invalid name', async () => { - const spy = vi.spyOn(console, 'error').mockImplementation(() => {}); - const { createCreateCommand } = await import('./create.js'); - const cmd = createCreateCommand(); - await cmd.parseAsync(['node', 'test', 'NotKebab', '--dir', testDir]); - - expect(spy).toHaveBeenCalled(); - spy.mockRestore(); - }); - - it('rejects unknown template', async () => { - const spy = vi.spyOn(console, 'error').mockImplementation(() => {}); - const { createCreateCommand } = await import('./create.js'); - const cmd = createCreateCommand(); - await cmd.parseAsync(['node', 'test', 'my-agent', '--template', 'unknown', '--dir', testDir]); - - expect(spy).toHaveBeenCalled(); - spy.mockRestore(); - }); - - it('rejects if file already exists', async () => { - const { createCreateCommand } = await import('./create.js'); - const cmd = createCreateCommand(); - - // Create first - await cmd.parseAsync(['node', 'test', 'existing', '--dir', testDir]); - expect(existsSync(join(testDir, 'existing.agent.ts'))).toBe(true); - - // Try again - const spy = vi.spyOn(console, 'error').mockImplementation(() => {}); - const cmd2 = createCreateCommand(); - await cmd2.parseAsync(['node', 'test', 'existing', '--dir', testDir]); - expect(spy).toHaveBeenCalled(); - spy.mockRestore(); - }); - - // ─── QW-1: defaults to discovery dir ─────────────────────── - - it('defaults to first discovery dir when no --dir given', async () => { - // Set up config with discovery dirs - const agentsDir = join(configDir, 'agents'); - mkdirSync(agentsDir, { recursive: true }); - process.env['AGENTAGE_CONFIG_DIR'] = configDir; - writeFileSync( - join(configDir, 'config.json'), - JSON.stringify({ - machine: { id: 'test', name: 'test' }, - daemon: { port: 4243 }, - agents: { default: agentsDir, additional: [] }, - projects: { default: '/tmp/projects', additional: [] }, - sync: { events: {} }, - }) - ); - - const { createCreateCommand } = await import('./create.js'); - const cmd = createCreateCommand(); - await cmd.parseAsync(['node', 'test', 'default-dir-agent']); - - expect(existsSync(join(agentsDir, 'default-dir-agent.agent.ts'))).toBe(true); - delete process.env['AGENTAGE_CONFIG_DIR']; - }); - - // ─── QW-2: smart post-create message ─────────────────────── - - it('shows auto-discovered message when created in discovery dir', async () => { - const agentsDir = join(configDir, 'agents'); - mkdirSync(agentsDir, { recursive: true }); - process.env['AGENTAGE_CONFIG_DIR'] = configDir; - writeFileSync( - join(configDir, 'config.json'), - JSON.stringify({ - machine: { id: 'test', name: 'test' }, - daemon: { port: 4243 }, - agents: { default: agentsDir, additional: [] }, - projects: { default: '/tmp/projects', additional: [] }, - sync: { events: {} }, - }) - ); - - const spy = vi.spyOn(console, 'log').mockImplementation(() => {}); - const { createCreateCommand } = await import('./create.js'); - const cmd = createCreateCommand(); - await cmd.parseAsync(['node', 'test', 'auto-agent', '--dir', agentsDir]); - - const output = spy.mock.calls.map((c) => c[0]).join('\n'); - expect(output).toContain('auto-discovered'); - expect(output).toContain('agentage run auto-agent'); - expect(output).not.toContain('cp '); - - spy.mockRestore(); - delete process.env['AGENTAGE_CONFIG_DIR']; - }); - - it('shows cp instructions when created outside discovery dir', async () => { - const agentsDir = join(configDir, 'agents'); - mkdirSync(agentsDir, { recursive: true }); - process.env['AGENTAGE_CONFIG_DIR'] = configDir; - writeFileSync( - join(configDir, 'config.json'), - JSON.stringify({ - machine: { id: 'test', name: 'test' }, - daemon: { port: 4243 }, - agents: { default: agentsDir, additional: [] }, - projects: { default: '/tmp/projects', additional: [] }, - sync: { events: {} }, - }) - ); - - const otherDir = join(testDir, 'other'); - mkdirSync(otherDir, { recursive: true }); - - const spy = vi.spyOn(console, 'log').mockImplementation(() => {}); - const { createCreateCommand } = await import('./create.js'); - const cmd = createCreateCommand(); - await cmd.parseAsync(['node', 'test', 'other-agent', '--dir', otherDir]); - - const output = spy.mock.calls.map((c) => c[0]).join('\n'); - expect(output).toContain('cp '); - expect(output).not.toContain('auto-discovered'); - - spy.mockRestore(); - delete process.env['AGENTAGE_CONFIG_DIR']; - }); -}); diff --git a/src/commands/create.ts b/src/commands/create.ts deleted file mode 100644 index b140c2d..0000000 --- a/src/commands/create.ts +++ /dev/null @@ -1,149 +0,0 @@ -import { existsSync, mkdirSync, writeFileSync } from 'node:fs'; -import { join, resolve } from 'node:path'; -import { Command } from 'commander'; -import chalk from 'chalk'; -import { loadConfig, getAgentsDirs, getDefaultAgentsDir } from '../daemon/config.js'; - -const TEMPLATES: Record string; deps?: string[] }> = { - simple: { - content: (name) => `import { agent, output } from '@agentage/core'; - -export default agent({ - name: '${name}', - description: 'A simple agent', - async *run({ task }) { - yield output(\`Running: \${task}\`); - }, -}); -`, - }, - shell: { - content: (name) => `import { agent, shell } from '@agentage/core'; - -export default agent({ - name: '${name}', - description: 'Executes a shell command and streams output', - async *run({ task }) { - yield* shell(task); - }, -}); -`, - }, - claude: { - deps: ['@anthropic-ai/claude-agent-sdk'], - content: (name) => `import { agent, claude } from '@agentage/core'; - -export default agent({ - name: '${name}', - description: 'Runs a task using Claude Code', - async *run({ task }, { signal }) { - yield* claude(task, { - signal, - tools: ['Read', 'Write', 'Edit', 'Glob', 'Grep', 'Bash'], - maxTurns: 10, - }); - }, -}); -`, - }, - copilot: { - deps: ['@github/copilot-sdk'], - content: (name) => `import { agent, copilot } from '@agentage/core'; - -export default agent({ - name: '${name}', - description: 'Runs a task using GitHub Copilot', - async *run({ task }, { signal }) { - yield* copilot(task, { signal, model: 'gpt-4o' }); - }, -}); -`, - }, - llm: { - content: (name) => `import { agent } from '@agentage/core'; - -export default agent({ - name: '${name}', - description: 'An LLM-powered agent', - model: 'claude-sonnet-4-6', - tools: ['read', 'glob', 'grep'], - prompt: \`You are a helpful assistant. -Respond concisely and cite sources when possible.\`, -}); -`, - }, -}; - -const KEBAB_CASE = /^[a-z][a-z0-9]*(-[a-z0-9]+)*$/; - -const resolveInstallDir = (): string => { - try { - const config = loadConfig(); - return config.agents.default; - } catch { - return getDefaultAgentsDir(); - } -}; - -export const createCreateCommand = (): Command => { - const cmd = new Command('create') - .description('Scaffold a new agent from a template') - .argument('', 'Agent name (kebab-case)') - .option('-t, --template