├── .eslintrc.cjs ├── .github └── workflows │ ├── publish.yml │ └── test.yml ├── .gitignore ├── .husky └── pre-commit ├── .lintstagedrc.mjs ├── .npmignore ├── .prettierrc ├── CHANGELOG.md ├── CODE_OF_CONDUCT.md ├── CONTRIBUTING.md ├── LICENSE ├── README.md ├── examples ├── test-firecrawl.ts ├── test-playwright.ts └── test-twitter.ts ├── package-lock.json ├── package.json ├── src ├── index.ts ├── server.ts ├── service.ts └── tools.ts └── tsconfig.json /.eslintrc.cjs: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | extends: [ 3 | 'eslint:recommended', 4 | 'plugin:@typescript-eslint/recommended', 5 | 'prettier', 6 | ], 7 | parser: '@typescript-eslint/parser', 8 | plugins: ['@typescript-eslint'], 9 | root: true, 10 | parserOptions: { 11 | project: './tsconfig.json', 12 | tsconfigRootDir: __dirname, 13 | sourceType: 'module', 14 | }, 15 | ignorePatterns: ['dist', '.eslintrc.cjs'], 16 | rules: { 17 | '@typescript-eslint/no-explicit-any': 'warn', 18 | '@typescript-eslint/no-unused-vars': ['warn', { argsIgnorePattern: '^_' }], 19 | '@typescript-eslint/no-floating-promises': 'error', 20 | }, 21 | }; 22 | -------------------------------------------------------------------------------- /.github/workflows/publish.yml: -------------------------------------------------------------------------------- 1 | name: Publish Package 2 | 3 | on: 4 | release: 5 | types: [created] 6 | workflow_dispatch: 7 | 8 | jobs: 9 | publish: 10 | runs-on: ubuntu-latest 11 | steps: 12 | - uses: actions/checkout@v4 13 | 14 | - name: Use Node.js 15 | uses: actions/setup-node@v4 16 | with: 17 | node-version: '20.x' 18 | registry-url: 'https://registry.npmjs.org' 19 | cache: 'npm' 20 | 21 | - name: Install dependencies 22 | run: npm ci 23 | 24 | - name: Build 25 | run: npm run build 26 | 27 | - name: Publish to npm 28 | run: npm publish --access public 29 | env: 30 | NODE_AUTH_TOKEN: ${{ secrets.NPM_TOKEN }} 31 | -------------------------------------------------------------------------------- /.github/workflows/test.yml: -------------------------------------------------------------------------------- 1 | name: Test 2 | 3 | on: 4 | push: 5 | branches: [main] 6 | pull_request: 7 | branches: [main] 8 | 9 | jobs: 10 | test: 11 | runs-on: ubuntu-latest 12 | 13 | steps: 14 | - uses: actions/checkout@v4 15 | 16 | - name: Use Node.js 20.x 17 | uses: actions/setup-node@v4 18 | with: 19 | node-version: 20.x 20 | cache: 'npm' 21 | 22 | - name: Install dependencies 23 | run: npm ci 24 | 25 | - name: Run linter 26 | run: npm run lint 27 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # Dependencies 2 | node_modules/ 3 | .pnp/ 4 | .pnp.js 5 | 6 | # Production build 7 | dist/ 8 | build/ 9 | 10 | # Testing 11 | coverage/ 12 | mcp.config.json 13 | client.md 14 | mcp-ai-sdk-bridge.md 15 | typesdk.md 16 | 17 | # Environment variables 18 | .env 19 | .env.local 20 | .env.*.local 21 | 22 | # Logs 23 | logs/ 24 | *.log 25 | npm-debug.log* 26 | yarn-debug.log* 27 | yarn-error.log* 28 | 29 | # Editor directories and files 30 | .idea/ 31 | .vscode/ 32 | *.swp 33 | *.swo 34 | .DS_Store 35 | *.pem 36 | 37 | # TypeScript cache 38 | *.tsbuildinfo 39 | 40 | # Optional npm cache directory 41 | .npm 42 | 43 | # Optional eslint cache 44 | .eslintcache 45 | 46 | # Optional REPL history 47 | .node_repl_history 48 | 49 | # Output of 'npm pack' 50 | *.tgz 51 | 52 | # Yarn Integrity file 53 | .yarn-integrity 54 | 55 | # dotenv environment variable files 56 | .env 57 | .env.test 58 | .env.production 59 | 60 | # parcel-bundler cache 61 | .cache 62 | .parcel-cache 63 | 64 | # Next.js build output 65 | .next 66 | out 67 | 68 | # Nuxt.js build / generate output 69 | .nuxt 70 | dist 71 | 72 | # Gatsby files 73 | .cache/ 74 | public 75 | 76 | # vuepress build output 77 | .vuepress/dist 78 | 79 | # Serverless directories 80 | .serverless/ 81 | 82 | # FuseBox cache 83 | .fusebox/ 84 | 85 | # DynamoDB Local files 86 | .dynamodb/ 87 | 88 | # TernJS port file 89 | .tern-port 90 | 91 | # Stores VSCode versions used for testing VSCode extensions 92 | .vscode-test -------------------------------------------------------------------------------- /.husky/pre-commit: -------------------------------------------------------------------------------- 1 | npx lint-staged -------------------------------------------------------------------------------- /.lintstagedrc.mjs: -------------------------------------------------------------------------------- 1 | export default { 2 | // Run ESLint on TypeScript files 3 | '**/*.{ts,tsx}': ['eslint --fix'], 4 | 5 | // Run Prettier on all supported files 6 | '**/*.{js,jsx,ts,tsx,json,md}': ['prettier --write'], 7 | 8 | // Run TypeScript compiler check on TypeScript files 9 | '**/*.{ts,tsx}': () => 'tsc --noEmit', 10 | }; 11 | -------------------------------------------------------------------------------- /.npmignore: -------------------------------------------------------------------------------- 1 | # Source 2 | src/ 3 | tests/ 4 | examples/ 5 | docs/ 6 | 7 | # Config files 8 | .eslintrc 9 | .prettierrc 10 | jest.config.js 11 | tsconfig.json 12 | .github/ 13 | .vscode/ 14 | 15 | # Logs 16 | logs/ 17 | *.log 18 | 19 | # Development files 20 | .env 21 | .env.* 22 | coverage/ 23 | .husky/ 24 | .git/ 25 | .gitignore 26 | 27 | # IDE 28 | .idea/ 29 | *.swp 30 | *.swo 31 | 32 | # Build process 33 | node_modules/ 34 | npm-debug.log* 35 | yarn-debug.log* 36 | yarn-error.log* 37 | 38 | # Test files 39 | __tests__/ 40 | *.test.ts 41 | *.spec.ts -------------------------------------------------------------------------------- /.prettierrc: -------------------------------------------------------------------------------- 1 | { 2 | "semi": true, 3 | "trailingComma": "es5", 4 | "singleQuote": true, 5 | "printWidth": 80, 6 | "tabWidth": 2, 7 | "useTabs": false, 8 | "bracketSpacing": true, 9 | "arrowParens": "avoid" 10 | } -------------------------------------------------------------------------------- /CHANGELOG.md: -------------------------------------------------------------------------------- 1 | # Changelog 2 | 3 | All notable changes to this project will be documented in this file. 4 | 5 | The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), 6 | and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). 7 | 8 | ## [Unreleased] 9 | 10 | ### Added 11 | 12 | - Multi-server support with independent configuration 13 | - Twitter MCP server integration 14 | - Firecrawl server integration 15 | - SSE server support 16 | - Comprehensive error handling and logging 17 | - TypeScript type definitions 18 | - GitHub Actions for testing and publishing 19 | 20 | ### Changed 21 | 22 | - Updated Node.js requirement to v20.x 23 | - Improved server cleanup process 24 | - Enhanced error handling for server connections 25 | 26 | ### Fixed 27 | 28 | - Process termination issues during cleanup 29 | - Server connection retry logic 30 | - Tool registration timing issues 31 | 32 | ## [0.1.0] - 2024-02-05 33 | 34 | ### Added 35 | 36 | - Initial release 37 | - Basic MCP server support 38 | - Configuration system 39 | - Tool execution framework 40 | - Documentation 41 | 42 | [Unreleased]: https://github.com/vrknetha/aisdk-mcp-bridge/compare/v0.1.0...HEAD 43 | [0.1.0]: https://github.com/vrknetha/aisdk-mcp-bridge/releases/tag/v0.1.0 44 | -------------------------------------------------------------------------------- /CODE_OF_CONDUCT.md: -------------------------------------------------------------------------------- 1 | # Contributor Covenant Code of Conduct 2 | 3 | ## Our Pledge 4 | 5 | We as members, contributors, and leaders pledge to make participation in our 6 | community a harassment-free experience for everyone, regardless of age, body 7 | size, visible or invisible disability, ethnicity, sex characteristics, gender 8 | identity and expression, level of experience, education, socio-economic status, 9 | nationality, personal appearance, race, religion, or sexual identity 10 | and orientation. 11 | 12 | We pledge to act and interact in ways that contribute to an open, welcoming, 13 | diverse, inclusive, and healthy community. 14 | 15 | ## Our Standards 16 | 17 | Examples of behavior that contributes to a positive environment for our 18 | community include: 19 | 20 | - Demonstrating empathy and kindness toward other people 21 | - Being respectful of differing opinions, viewpoints, and experiences 22 | - Giving and gracefully accepting constructive feedback 23 | - Accepting responsibility and apologizing to those affected by our mistakes, 24 | and learning from the experience 25 | - Focusing on what is best not just for us as individuals, but for the 26 | overall community 27 | 28 | Examples of unacceptable behavior include: 29 | 30 | - The use of sexualized language or imagery, and sexual attention or 31 | advances of any kind 32 | - Trolling, insulting or derogatory comments, and personal or political attacks 33 | - Public or private harassment 34 | - Publishing others' private information, such as a physical or email 35 | address, without their explicit permission 36 | - Other conduct which could reasonably be considered inappropriate in a 37 | professional setting 38 | 39 | ## Enforcement Responsibilities 40 | 41 | Project maintainers are responsible for clarifying and enforcing our standards of 42 | acceptable behavior and will take appropriate and fair corrective action in 43 | response to any behavior that they deem inappropriate, threatening, offensive, 44 | or harmful. 45 | 46 | ## Scope 47 | 48 | This Code of Conduct applies within all community spaces, and also applies when 49 | an individual is officially representing the community in public spaces. 50 | 51 | ## Enforcement 52 | 53 | Instances of abusive, harassing, or otherwise unacceptable behavior may be 54 | reported to the project maintainers responsible for enforcement at 55 | ravi.kiran@yourdomain.com. 56 | 57 | All complaints will be reviewed and investigated promptly and fairly. 58 | 59 | ## Attribution 60 | 61 | This Code of Conduct is adapted from the [Contributor Covenant][homepage], 62 | version 2.0, available at 63 | https://www.contributor-covenant.org/version/2/0/code_of_conduct.html. 64 | 65 | [homepage]: https://www.contributor-covenant.org 66 | -------------------------------------------------------------------------------- /CONTRIBUTING.md: -------------------------------------------------------------------------------- 1 | # Contributing to AISDK MCP Bridge 2 | 3 | First off, thank you for considering contributing to AISDK MCP Bridge! It's people like you that make this project better. 4 | 5 | ## Code of Conduct 6 | 7 | This project and everyone participating in it is governed by our Code of Conduct. By participating, you are expected to uphold this code. 8 | 9 | ## How Can I Contribute? 10 | 11 | ### Reporting Bugs 12 | 13 | Before creating bug reports, please check the existing issues to avoid duplicates. When you create a bug report, include as many details as possible: 14 | 15 | - Use a clear and descriptive title 16 | - Describe the exact steps to reproduce the problem 17 | - Provide specific examples (e.g., sample code, configuration) 18 | - Describe the behavior you observed and what you expected 19 | - Include relevant logs from `logs/mcp-tools.log` 20 | - Note your environment (Node.js version, OS, etc.) 21 | 22 | ### Suggesting Enhancements 23 | 24 | Enhancement suggestions are tracked as GitHub issues. When creating an enhancement suggestion: 25 | 26 | - Use a clear and descriptive title 27 | - Provide a detailed description of the proposed functionality 28 | - Explain why this enhancement would be useful 29 | - List any alternatives you've considered 30 | 31 | ### Pull Requests 32 | 33 | 1. Fork the repo and create your branch from `main` 34 | 2. If you've added code that should be tested, add tests 35 | 3. Ensure the test suite passes 36 | 4. Make sure your code follows the existing style 37 | 5. Update the documentation if needed 38 | 39 | ## Development Process 40 | 41 | 1. **Setup Development Environment** 42 | 43 | ```bash 44 | git clone https://github.com/vrknetha/aisdk-mcp-bridge.git 45 | cd aisdk-mcp-bridge 46 | npm install 47 | ``` 48 | 49 | 2. **Run Tests** 50 | 51 | ```bash 52 | npm test # Run all tests 53 | npm run test:twitter # Run Twitter MCP tests 54 | npm run test:firecrawl # Run Firecrawl tests 55 | ``` 56 | 57 | 3. **Code Style** 58 | - Use TypeScript 59 | - Follow the existing code style 60 | - Use ESLint and Prettier for formatting 61 | ```bash 62 | npm run lint # Check code style 63 | npm run format # Format code 64 | ``` 65 | 66 | ## Project Structure 67 | 68 | ``` 69 | aisdk-mcp-bridge/ 70 | ├── src/ 71 | │ ├── index.ts # Main entry point 72 | │ ├── service.ts # MCP service implementation 73 | │ ├── server.ts # Server management 74 | │ └── tools.ts # Utility functions 75 | ├── examples/ 76 | │ ├── test-twitter.ts # Twitter MCP example 77 | │ └── test-firecrawl.ts# Firecrawl example 78 | ├── tests/ 79 | │ └── ... # Test files 80 | └── logs/ 81 | └── mcp-tools.log # Debug and error logs 82 | ``` 83 | 84 | ## Adding New MCP Servers 85 | 86 | 1. Create a new configuration in `mcp.config.json`: 87 | 88 | ```json 89 | { 90 | "mcpServers": { 91 | "your-server": { 92 | "command": "npx", 93 | "args": ["-y", "your-mcp-server"], 94 | "env": { 95 | "YOUR_API_KEY": "your-api-key" 96 | } 97 | } 98 | } 99 | } 100 | ``` 101 | 102 | 2. Add tests in `examples/test-your-server.ts` 103 | 3. Update documentation to include your server 104 | 4. Add any necessary environment variables to `.env.example` 105 | 106 | ## Commit Messages 107 | 108 | Format your commit messages according to conventional commits: 109 | 110 | - `feat:` New feature 111 | - `fix:` Bug fix 112 | - `docs:` Documentation changes 113 | - `test:` Adding or updating tests 114 | - `chore:` Maintenance tasks 115 | 116 | Example: 117 | 118 | ``` 119 | feat: add support for new MCP server type 120 | 121 | - Added configuration options for new server 122 | - Implemented connection handling 123 | - Added tests and documentation 124 | ``` 125 | 126 | ## Release Process 127 | 128 | 1. Update version in `package.json` 129 | 2. Update CHANGELOG.md 130 | 3. Create a new release on GitHub 131 | 4. GitHub Actions will automatically publish to npm 132 | 133 | ## Getting Help 134 | 135 | - Check the [documentation](README.md) 136 | - Join our [Discord community](https://discord.gg/your-invite) 137 | - Create an issue for complex questions 138 | 139 | ## License 140 | 141 | By contributing, you agree that your contributions will be licensed under the MIT License. 142 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2024 AISDK MCP Bridge 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # AISDK MCP Bridge 2 | 3 | A bridge package that enables seamless integration between the Model Context Protocol (MCP) and AI SDK, allowing for efficient communication and tool execution between MCP servers and AI models. 4 | 5 | [![npm version](https://badge.fury.io/js/aisdk-mcp-bridge.svg)](https://badge.fury.io/js/aisdk-mcp-bridge) 6 | [![License: MIT](https://img.shields.io/badge/License-MIT-yellow.svg)](https://opensource.org/licenses/MIT) 7 | 8 | ## Features 9 | 10 | - Seamless integration between MCP servers and AI SDK 11 | - Support for various MCP server types (Node.js, Python, UVX) 12 | - Multi-server support with independent configuration 13 | - Flexible configuration through `mcp.config.json` 14 | - TypeScript support with full type definitions 15 | - Robust error handling and logging 16 | - Easy-to-use API for tool execution 17 | 18 | ## Installation 19 | 20 | ```bash 21 | npm install aisdk-mcp-bridge 22 | ``` 23 | 24 | ## Quick Start 25 | 26 | 1. Create an `mcp.config.json` file in your project root: 27 | 28 | ```json 29 | { 30 | "mcpServers": { 31 | "twitter-mcp": { 32 | "command": "npx", 33 | "args": ["-y", "@enescinar/twitter-mcp"], 34 | "env": { 35 | "API_KEY": "your-twitter-api-key", 36 | "API_SECRET_KEY": "your-twitter-api-secret", 37 | "ACCESS_TOKEN": "your-twitter-access-token", 38 | "ACCESS_TOKEN_SECRET": "your-twitter-access-token-secret" 39 | } 40 | }, 41 | "firecrawl": { 42 | "command": "npx", 43 | "args": ["-y", "mcp-server-firecrawl"], 44 | "env": { 45 | "FIRE_CRAWL_API_KEY": "your-firecrawl-api-key", 46 | "FIRE_CRAWL_API_URL": "https://api.firecrawl.com" 47 | } 48 | } 49 | } 50 | } 51 | ``` 52 | 53 | 2. Import and use the bridge in your code: 54 | 55 | ```typescript 56 | import { generateText } from 'ai'; 57 | import { google } from '@ai-sdk/google'; 58 | import { getMcpTools, cleanupMcp, initializeMcp } from 'aisdk-mcp-bridge'; 59 | import dotenv from 'dotenv'; 60 | dotenv.config(); 61 | 62 | async function main() { 63 | try { 64 | // Initialize MCP 65 | await initializeMcp({ debug: true }); 66 | 67 | // Get tools from all servers 68 | const allTools = await getMcpTools({ debug: true }); 69 | 70 | // Or get tools from a specific server 71 | const twitterTools = await getMcpTools({ 72 | debug: true, 73 | serverName: 'twitter-mcp', 74 | }); 75 | 76 | // Use tools with AI SDK 77 | const result = await generateText({ 78 | model: google('gemini-1.5-pro'), 79 | messages: [ 80 | { 81 | role: 'system', 82 | content: 83 | 'You are an AI assistant that uses various tools to help users.', 84 | }, 85 | { 86 | role: 'user', 87 | content: 'Your task description here', 88 | }, 89 | ], 90 | tools: twitterTools, // or allTools for all available tools 91 | }); 92 | 93 | console.log('Result:', result.text); 94 | } finally { 95 | // Clean up resources 96 | await cleanupMcp(); 97 | } 98 | } 99 | 100 | main().catch(error => { 101 | console.error('Error:', error); 102 | process.exit(1); 103 | }); 104 | ``` 105 | 106 | ## Configuration 107 | 108 | The `mcp.config.json` file supports multiple servers and communication modes. Each server can be configured independently. 109 | 110 | ### Server Configuration Examples: 111 | 112 | ### Twitter MCP Server 113 | 114 | ```json 115 | { 116 | "mcpServers": { 117 | "twitter-mcp": { 118 | "command": "npx", 119 | "args": ["-y", "@enescinar/twitter-mcp"], 120 | "env": { 121 | "API_KEY": "your-twitter-api-key", 122 | "API_SECRET_KEY": "your-twitter-api-secret", 123 | "ACCESS_TOKEN": "your-twitter-access-token", 124 | "ACCESS_TOKEN_SECRET": "your-twitter-access-token-secret" 125 | } 126 | } 127 | } 128 | } 129 | ``` 130 | 131 | ### Firecrawl Server 132 | 133 | ```json 134 | { 135 | "mcpServers": { 136 | "firecrawl": { 137 | "command": "npx", 138 | "args": ["-y", "mcp-server-firecrawl"], 139 | "env": { 140 | "FIRE_CRAWL_API_KEY": "your-firecrawl-api-key", 141 | "FIRE_CRAWL_API_URL": "https://api.firecrawl.com" 142 | } 143 | } 144 | } 145 | } 146 | ``` 147 | 148 | ### SSE Server 149 | 150 | ```json 151 | { 152 | "mcpServers": { 153 | "sse-server": { 154 | "command": "node", 155 | "args": ["./server.js"], 156 | "mode": "sse", 157 | "sseOptions": { 158 | "endpoint": "http://localhost:3000/events", 159 | "headers": {}, 160 | "reconnectTimeout": 5000 161 | } 162 | } 163 | } 164 | } 165 | ``` 166 | 167 | ## Server Modes 168 | 169 | The bridge supports different communication modes: 170 | 171 | 1. **stdio Mode** (Default) 172 | 173 | - Direct communication through standard input/output 174 | - Best for simple integrations and local development 175 | - Low latency and minimal setup required 176 | 177 | 2. **SSE Mode** (Server-Sent Events) 178 | - Real-time, one-way communication from server to client 179 | - Ideal for streaming updates and long-running operations 180 | - Built-in reconnection handling 181 | 182 | ## API Reference 183 | 184 | ### Core Functions 185 | 186 | #### `initializeMcp(options?: InitOptions): Promise` 187 | 188 | Initialize the MCP service with the provided options. 189 | 190 | ```typescript 191 | interface InitOptions { 192 | configPath?: string; // Path to mcp.config.json 193 | debug?: boolean; // Enable debug logging 194 | } 195 | ``` 196 | 197 | #### `getMcpTools(options?: ToolOptions): Promise` 198 | 199 | Get AI SDK-compatible tools from MCP servers. 200 | 201 | ```typescript 202 | interface ToolOptions { 203 | debug?: boolean; // Enable debug logging 204 | serverName?: string; // Optional server name to get tools from a specific server 205 | } 206 | ``` 207 | 208 | #### `executeMcpFunction(serverName: string, functionName: string, args: Record): Promise` 209 | 210 | Execute a specific function on an MCP server directly. 211 | 212 | ```typescript 213 | // Example 214 | const result = await executeMcpFunction('twitter-mcp', 'postTweet', { 215 | text: 'Hello from MCP!', 216 | }); 217 | ``` 218 | 219 | ### Core Types 220 | 221 | #### `MCPConfig` (alias for `MCPServersConfig`) 222 | 223 | Configuration type for MCP servers. 224 | 225 | ```typescript 226 | interface MCPConfig { 227 | mcpServers: { 228 | [key: string]: ServerConfig; 229 | }; 230 | } 231 | ``` 232 | 233 | #### `ServerConfig` 234 | 235 | Configuration for individual MCP servers. 236 | 237 | ```typescript 238 | interface ServerConfig { 239 | command: string; 240 | args?: string[]; 241 | env?: Record; 242 | mode?: 'stdio' | 'sse'; 243 | sseOptions?: { 244 | endpoint: string; 245 | headers?: Record; 246 | reconnectTimeout?: number; 247 | }; 248 | } 249 | ``` 250 | 251 | #### `MCPToolResult` 252 | 253 | Result type for MCP tool executions. 254 | 255 | ```typescript 256 | interface MCPToolResult { 257 | success: boolean; 258 | data?: unknown; 259 | error?: string; 260 | } 261 | ``` 262 | 263 | #### `cleanupMcp(): Promise` 264 | 265 | Clean up MCP resources and close all server connections. 266 | 267 | ## Error Handling 268 | 269 | The bridge includes comprehensive error handling for: 270 | 271 | - Server initialization failures 272 | - Communication errors 273 | - Tool execution failures 274 | - Configuration issues 275 | - Server connection issues 276 | 277 | ## Logging 278 | 279 | The bridge provides detailed logging through: 280 | 281 | - `mcp-tools.log`: Server-side tool execution logs 282 | - Console output for debugging and errors 283 | 284 | ### Debug Logging 285 | 286 | You can enable detailed debug logging by setting the DEBUG environment variable: 287 | 288 | ```bash 289 | # Enable all debug logs 290 | DEBUG=* npm start 291 | 292 | # Enable MCP debug logs 293 | DEBUG=mcp npm start 294 | 295 | # Enable all MCP namespace logs 296 | DEBUG=mcp:* npm start 297 | ``` 298 | 299 | Debug logs will show: 300 | 301 | - Server initialization and shutdown events 302 | - Tool registration and execution details 303 | - Communication with MCP servers 304 | - Schema conversions and validations 305 | - Error details with stack traces 306 | - Performance metrics and timing information 307 | 308 | ### Log Types 309 | 310 | The logging system supports three types of logs: 311 | 312 | - `info`: General operational information 313 | - `debug`: Detailed debugging information (requires DEBUG env variable) 314 | - `error`: Error messages and stack traces (always logged) 315 | 316 | ### Log File 317 | 318 | All logs are written to `logs/mcp-tools.log` with the following format: 319 | 320 | ``` 321 | [TIMESTAMP] [TYPE] Message 322 | {Optional JSON data} 323 | ``` 324 | 325 | ## Development 326 | 327 | ### Prerequisites 328 | 329 | - Node.js 20.x or higher 330 | - npm 7.x or higher 331 | 332 | ### Setup 333 | 334 | 1. Clone the repository 335 | 2. Install dependencies: 336 | 337 | ```bash 338 | npm install 339 | ``` 340 | 341 | ### Testing 342 | 343 | Run the test suite: 344 | 345 | ```bash 346 | npm test 347 | ``` 348 | 349 | Run specific tests: 350 | 351 | ```bash 352 | 353 | npm run test:twitter 354 | npm run test:firecrawl 355 | ``` 356 | 357 | ## Contributing 358 | 359 | We welcome contributions! Please see our [Contributing Guide](CONTRIBUTING.md) for details on: 360 | 361 | - Setting up the development environment 362 | - Coding standards 363 | - Pull request process 364 | - Adding new MCP servers 365 | 366 | Please note that this project is released with a [Code of Conduct](CODE_OF_CONDUCT.md). By participating in this project you agree to abide by its terms. 367 | 368 | ## Support 369 | 370 | For support: 371 | 372 | 1. Check the [documentation](README.md) 373 | 2. Search [existing issues](https://github.com/vrknetha/aisdk-mcp-bridge/issues) 374 | 3. Create a new issue if your problem persists 375 | 376 | ## Changelog 377 | 378 | See [CHANGELOG.md](CHANGELOG.md) for a list of changes and migration guides. 379 | 380 | ## Security 381 | 382 | For security issues, please email ravi@caw.tech instead of using the public issue tracker. 383 | 384 | ## Authors 385 | 386 | - **Ravi Kiran** - _Initial work_ - [@vrknetha](https://github.com/vrknetha) 387 | 388 | See also the list of [contributors](https://github.com/vrknetha/aisdk-mcp-bridge/contributors) who participated in this project. 389 | 390 | ## Acknowledgments 391 | 392 | - AI SDK team for their excellent SDK 393 | - MCP community for the protocol specification 394 | - All contributors who have helped with the project 395 | 396 | ## License 397 | 398 | This project is licensed under the MIT License - see the [LICENSE](LICENSE) file for details. 399 | -------------------------------------------------------------------------------- /examples/test-firecrawl.ts: -------------------------------------------------------------------------------- 1 | import { generateText } from 'ai'; 2 | import { google } from '@ai-sdk/google'; 3 | import { getMcpTools, initializeMcp, cleanupMcp } from '../src'; 4 | 5 | import dotenv from 'dotenv'; 6 | import { log } from '../src/tools'; 7 | dotenv.config(); 8 | 9 | async function main() { 10 | try { 11 | // Initialize MCP 12 | await initializeMcp({ debug: true }); 13 | 14 | // Test firecrawl server 15 | log('\nTesting firecrawl server...'); 16 | const firecrawlTools = await getMcpTools({ serverName: 'firecrawl' }); 17 | const firecrawlResult = await generateText({ 18 | model: google('gemini-1.5-flash'), 19 | messages: [ 20 | { 21 | role: 'system', 22 | content: 'You are a helpful assistant that analyzes web content.', 23 | }, 24 | { 25 | role: 'user', 26 | content: 27 | 'Please analyze the current trends using https://www.techtarget.com/searchenterpriseai/tip/9-top-AI-and-machine-learning-trends', 28 | }, 29 | ], 30 | tools: firecrawlTools, 31 | }); 32 | log('Firecrawl test result:', firecrawlResult.text); 33 | } catch (error) { 34 | log('Test error:', error); 35 | process.exit(1); 36 | } finally { 37 | // Clean up 38 | await cleanupMcp(); 39 | } 40 | } 41 | 42 | // Run tests 43 | main().catch(error => { 44 | log('Test failed:', error); 45 | }); 46 | -------------------------------------------------------------------------------- /examples/test-playwright.ts: -------------------------------------------------------------------------------- 1 | import { generateText } from 'ai'; 2 | import { google } from '@ai-sdk/google'; 3 | import { getMcpTools, initializeMcp, cleanupMcp } from '../src'; 4 | 5 | import dotenv from 'dotenv'; 6 | dotenv.config(); 7 | 8 | async function main() { 9 | try { 10 | // Initialize MCP 11 | await initializeMcp({ debug: true }); 12 | 13 | // Test playwright server 14 | console.log('\nTesting Playwright server...'); 15 | const playwrightTools = await getMcpTools({ serverName: 'playwright' }); 16 | 17 | // Test browser navigation and screenshot 18 | const result = await generateText({ 19 | model: google('gemini-1.5-flash'), 20 | messages: [ 21 | { 22 | role: 'system', 23 | content: 24 | 'You are a web automation assistant that uses Playwright to navigate websites and extract information. Use the provided tools in sequence: first navigate to the page, then interact with it as needed.', 25 | }, 26 | { 27 | role: 'user', 28 | content: 29 | 'Please navigate to https://news.ycombinator.com/, and then analyze the top 5 stories. Provide a summary of each story with its title and points/comments if available.', 30 | }, 31 | ], 32 | tools: playwrightTools, 33 | }); 34 | 35 | console.log('Playwright test result:', result.text); 36 | } catch (error) { 37 | console.error('Test error:', error); 38 | process.exit(1); 39 | } finally { 40 | // Clean up 41 | await cleanupMcp(); 42 | } 43 | } 44 | 45 | // Run tests 46 | main().catch(error => { 47 | console.error('Test failed:', error); 48 | process.exit(1); 49 | }); 50 | -------------------------------------------------------------------------------- /examples/test-twitter.ts: -------------------------------------------------------------------------------- 1 | import { generateText } from 'ai'; 2 | import { google } from '@ai-sdk/google'; 3 | import { getMcpTools, initializeMcp, cleanupMcp } from '../src'; 4 | 5 | import dotenv from 'dotenv'; 6 | dotenv.config(); 7 | 8 | async function main() { 9 | try { 10 | // Initialize MCP 11 | await initializeMcp({ debug: true }); 12 | 13 | // Test Twitter server 14 | console.log('\nTesting Twitter MCP server...'); 15 | const twitterTools = await getMcpTools({ serverName: 'twitter-mcp' }); 16 | 17 | // Search and analyze tweets 18 | const result = await generateText({ 19 | model: google('gemini-1.5-flash'), 20 | messages: [ 21 | { 22 | role: 'system', 23 | content: 24 | 'You are a social media analyst specializing in AI technology trends. Use the Twitter search tools to find and analyze recent discussions about AI technologies.', 25 | }, 26 | { 27 | role: 'user', 28 | content: `Please perform the following analysis: 29 | 1. Search for recent tweets about "GenAI" and "LLM" (use multiple searches if needed) 30 | 2. Focus on tweets with high engagement (likes, retweets) 31 | 3. Analyze the main themes and sentiments in these discussions 32 | 4. Provide a summary of: 33 | - Key trends and topics being discussed 34 | - Notable opinions or insights 35 | - Common concerns or challenges mentioned 36 | - Any interesting predictions or future outlook`, 37 | }, 38 | ], 39 | tools: twitterTools, 40 | }); 41 | 42 | console.log('Twitter analysis result:', result.text); 43 | } catch (error) { 44 | console.error('Test error:', error); 45 | process.exit(1); 46 | } finally { 47 | // Clean up 48 | await cleanupMcp(); 49 | } 50 | } 51 | 52 | // Run tests 53 | main().catch(error => { 54 | console.error('Test failed:', error); 55 | process.exit(1); 56 | }); 57 | -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "aisdk-mcp-bridge", 3 | "version": "0.1.0", 4 | "lockfileVersion": 3, 5 | "requires": true, 6 | "packages": { 7 | "": { 8 | "name": "aisdk-mcp-bridge", 9 | "version": "0.1.0", 10 | "license": "MIT", 11 | "dependencies": { 12 | "@modelcontextprotocol/sdk": "^1.4.1", 13 | "ai": "^4.1.10", 14 | "axios": "^1.6.7", 15 | "zod": "^3.22.4" 16 | }, 17 | "devDependencies": { 18 | "@ai-sdk/google": "^1.0.0", 19 | "@openrouter/ai-sdk-provider": "^0.2.0", 20 | "@types/cors": "^2.8.13", 21 | "@types/express": "^4.17.17", 22 | "@types/node": "^20.11.16", 23 | "@typescript-eslint/eslint-plugin": "^6.21.0", 24 | "@typescript-eslint/parser": "^6.21.0", 25 | "cors": "^2.8.5", 26 | "dotenv": "^16.0.3", 27 | "eslint": "^8.56.0", 28 | "eslint-config-prettier": "^9.1.0", 29 | "eslint-plugin-prettier": "^5.1.3", 30 | "express": "^4.18.2", 31 | "prettier": "^3.2.5", 32 | "tsx": "^4.19.2", 33 | "typescript": "^5.3.3" 34 | }, 35 | "engines": { 36 | "node": ">=20.0.0" 37 | }, 38 | "peerDependencies": { 39 | "ai": "^4.1.10" 40 | } 41 | }, 42 | "node_modules/@ai-sdk/google": { 43 | "version": "1.1.8", 44 | "resolved": "https://registry.npmjs.org/@ai-sdk/google/-/google-1.1.8.tgz", 45 | "integrity": "sha512-8ZOS4p8Qp/7fXWnwNizlFN5aTBu4u3pNlsc7LDFGUtuJXz6j/pAQhqBh3nEaagX3GNuDFT8rxZ1yu82+fYaRpA==", 46 | "dev": true, 47 | "dependencies": { 48 | "@ai-sdk/provider": "1.0.7", 49 | "@ai-sdk/provider-utils": "2.1.6" 50 | }, 51 | "engines": { 52 | "node": ">=18" 53 | }, 54 | "peerDependencies": { 55 | "zod": "^3.0.0" 56 | } 57 | }, 58 | "node_modules/@ai-sdk/provider": { 59 | "version": "1.0.7", 60 | "resolved": "https://registry.npmjs.org/@ai-sdk/provider/-/provider-1.0.7.tgz", 61 | "integrity": "sha512-q1PJEZ0qD9rVR+8JFEd01/QM++csMT5UVwYXSN2u54BrVw/D8TZLTeg2FEfKK00DgAx0UtWd8XOhhwITP9BT5g==", 62 | "dependencies": { 63 | "json-schema": "^0.4.0" 64 | }, 65 | "engines": { 66 | "node": ">=18" 67 | } 68 | }, 69 | "node_modules/@ai-sdk/provider-utils": { 70 | "version": "2.1.6", 71 | "resolved": "https://registry.npmjs.org/@ai-sdk/provider-utils/-/provider-utils-2.1.6.tgz", 72 | "integrity": "sha512-Pfyaj0QZS22qyVn5Iz7IXcJ8nKIKlu2MeSAdKJzTwkAks7zdLaKVB+396Rqcp1bfQnxl7vaduQVMQiXUrgK8Gw==", 73 | "dependencies": { 74 | "@ai-sdk/provider": "1.0.7", 75 | "eventsource-parser": "^3.0.0", 76 | "nanoid": "^3.3.8", 77 | "secure-json-parse": "^2.7.0" 78 | }, 79 | "engines": { 80 | "node": ">=18" 81 | }, 82 | "peerDependencies": { 83 | "zod": "^3.0.0" 84 | }, 85 | "peerDependenciesMeta": { 86 | "zod": { 87 | "optional": true 88 | } 89 | } 90 | }, 91 | "node_modules/@ai-sdk/react": { 92 | "version": "1.1.8", 93 | "resolved": "https://registry.npmjs.org/@ai-sdk/react/-/react-1.1.8.tgz", 94 | "integrity": "sha512-buHm7hP21xEOksnRQtJX9fKbi7cAUwanEBa5niddTDibCDKd+kIXP2vaJGy8+heB3rff+XSW3BWlA8pscK+n1g==", 95 | "dependencies": { 96 | "@ai-sdk/provider-utils": "2.1.6", 97 | "@ai-sdk/ui-utils": "1.1.8", 98 | "swr": "^2.2.5", 99 | "throttleit": "2.1.0" 100 | }, 101 | "engines": { 102 | "node": ">=18" 103 | }, 104 | "peerDependencies": { 105 | "react": "^18 || ^19 || ^19.0.0-rc", 106 | "zod": "^3.0.0" 107 | }, 108 | "peerDependenciesMeta": { 109 | "react": { 110 | "optional": true 111 | }, 112 | "zod": { 113 | "optional": true 114 | } 115 | } 116 | }, 117 | "node_modules/@ai-sdk/ui-utils": { 118 | "version": "1.1.8", 119 | "resolved": "https://registry.npmjs.org/@ai-sdk/ui-utils/-/ui-utils-1.1.8.tgz", 120 | "integrity": "sha512-nbok53K1EalO2sZjBLFB33cqs+8SxiL6pe7ekZ7+5f2MJTwdvpShl6d9U4O8fO3DnZ9pYLzaVC0XNMxnJt030Q==", 121 | "dependencies": { 122 | "@ai-sdk/provider": "1.0.7", 123 | "@ai-sdk/provider-utils": "2.1.6", 124 | "zod-to-json-schema": "^3.24.1" 125 | }, 126 | "engines": { 127 | "node": ">=18" 128 | }, 129 | "peerDependencies": { 130 | "zod": "^3.0.0" 131 | }, 132 | "peerDependenciesMeta": { 133 | "zod": { 134 | "optional": true 135 | } 136 | } 137 | }, 138 | "node_modules/@esbuild/aix-ppc64": { 139 | "version": "0.23.1", 140 | "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.23.1.tgz", 141 | "integrity": "sha512-6VhYk1diRqrhBAqpJEdjASR/+WVRtfjpqKuNw11cLiaWpAT/Uu+nokB+UJnevzy/P9C/ty6AOe0dwueMrGh/iQ==", 142 | "cpu": [ 143 | "ppc64" 144 | ], 145 | "dev": true, 146 | "optional": true, 147 | "os": [ 148 | "aix" 149 | ], 150 | "engines": { 151 | "node": ">=18" 152 | } 153 | }, 154 | "node_modules/@esbuild/android-arm": { 155 | "version": "0.23.1", 156 | "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.23.1.tgz", 157 | "integrity": "sha512-uz6/tEy2IFm9RYOyvKl88zdzZfwEfKZmnX9Cj1BHjeSGNuGLuMD1kR8y5bteYmwqKm1tj8m4cb/aKEorr6fHWQ==", 158 | "cpu": [ 159 | "arm" 160 | ], 161 | "dev": true, 162 | "optional": true, 163 | "os": [ 164 | "android" 165 | ], 166 | "engines": { 167 | "node": ">=18" 168 | } 169 | }, 170 | "node_modules/@esbuild/android-arm64": { 171 | "version": "0.23.1", 172 | "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.23.1.tgz", 173 | "integrity": "sha512-xw50ipykXcLstLeWH7WRdQuysJqejuAGPd30vd1i5zSyKK3WE+ijzHmLKxdiCMtH1pHz78rOg0BKSYOSB/2Khw==", 174 | "cpu": [ 175 | "arm64" 176 | ], 177 | "dev": true, 178 | "optional": true, 179 | "os": [ 180 | "android" 181 | ], 182 | "engines": { 183 | "node": ">=18" 184 | } 185 | }, 186 | "node_modules/@esbuild/android-x64": { 187 | "version": "0.23.1", 188 | "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.23.1.tgz", 189 | "integrity": "sha512-nlN9B69St9BwUoB+jkyU090bru8L0NA3yFvAd7k8dNsVH8bi9a8cUAUSEcEEgTp2z3dbEDGJGfP6VUnkQnlReg==", 190 | "cpu": [ 191 | "x64" 192 | ], 193 | "dev": true, 194 | "optional": true, 195 | "os": [ 196 | "android" 197 | ], 198 | "engines": { 199 | "node": ">=18" 200 | } 201 | }, 202 | "node_modules/@esbuild/darwin-arm64": { 203 | "version": "0.23.1", 204 | "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.23.1.tgz", 205 | "integrity": "sha512-YsS2e3Wtgnw7Wq53XXBLcV6JhRsEq8hkfg91ESVadIrzr9wO6jJDMZnCQbHm1Guc5t/CdDiFSSfWP58FNuvT3Q==", 206 | "cpu": [ 207 | "arm64" 208 | ], 209 | "dev": true, 210 | "optional": true, 211 | "os": [ 212 | "darwin" 213 | ], 214 | "engines": { 215 | "node": ">=18" 216 | } 217 | }, 218 | "node_modules/@esbuild/darwin-x64": { 219 | "version": "0.23.1", 220 | "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.23.1.tgz", 221 | "integrity": "sha512-aClqdgTDVPSEGgoCS8QDG37Gu8yc9lTHNAQlsztQ6ENetKEO//b8y31MMu2ZaPbn4kVsIABzVLXYLhCGekGDqw==", 222 | "cpu": [ 223 | "x64" 224 | ], 225 | "dev": true, 226 | "optional": true, 227 | "os": [ 228 | "darwin" 229 | ], 230 | "engines": { 231 | "node": ">=18" 232 | } 233 | }, 234 | "node_modules/@esbuild/freebsd-arm64": { 235 | "version": "0.23.1", 236 | "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.23.1.tgz", 237 | "integrity": "sha512-h1k6yS8/pN/NHlMl5+v4XPfikhJulk4G+tKGFIOwURBSFzE8bixw1ebjluLOjfwtLqY0kewfjLSrO6tN2MgIhA==", 238 | "cpu": [ 239 | "arm64" 240 | ], 241 | "dev": true, 242 | "optional": true, 243 | "os": [ 244 | "freebsd" 245 | ], 246 | "engines": { 247 | "node": ">=18" 248 | } 249 | }, 250 | "node_modules/@esbuild/freebsd-x64": { 251 | "version": "0.23.1", 252 | "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.23.1.tgz", 253 | "integrity": "sha512-lK1eJeyk1ZX8UklqFd/3A60UuZ/6UVfGT2LuGo3Wp4/z7eRTRYY+0xOu2kpClP+vMTi9wKOfXi2vjUpO1Ro76g==", 254 | "cpu": [ 255 | "x64" 256 | ], 257 | "dev": true, 258 | "optional": true, 259 | "os": [ 260 | "freebsd" 261 | ], 262 | "engines": { 263 | "node": ">=18" 264 | } 265 | }, 266 | "node_modules/@esbuild/linux-arm": { 267 | "version": "0.23.1", 268 | "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.23.1.tgz", 269 | "integrity": "sha512-CXXkzgn+dXAPs3WBwE+Kvnrf4WECwBdfjfeYHpMeVxWE0EceB6vhWGShs6wi0IYEqMSIzdOF1XjQ/Mkm5d7ZdQ==", 270 | "cpu": [ 271 | "arm" 272 | ], 273 | "dev": true, 274 | "optional": true, 275 | "os": [ 276 | "linux" 277 | ], 278 | "engines": { 279 | "node": ">=18" 280 | } 281 | }, 282 | "node_modules/@esbuild/linux-arm64": { 283 | "version": "0.23.1", 284 | "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.23.1.tgz", 285 | "integrity": "sha512-/93bf2yxencYDnItMYV/v116zff6UyTjo4EtEQjUBeGiVpMmffDNUyD9UN2zV+V3LRV3/on4xdZ26NKzn6754g==", 286 | "cpu": [ 287 | "arm64" 288 | ], 289 | "dev": true, 290 | "optional": true, 291 | "os": [ 292 | "linux" 293 | ], 294 | "engines": { 295 | "node": ">=18" 296 | } 297 | }, 298 | "node_modules/@esbuild/linux-ia32": { 299 | "version": "0.23.1", 300 | "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.23.1.tgz", 301 | "integrity": "sha512-VTN4EuOHwXEkXzX5nTvVY4s7E/Krz7COC8xkftbbKRYAl96vPiUssGkeMELQMOnLOJ8k3BY1+ZY52tttZnHcXQ==", 302 | "cpu": [ 303 | "ia32" 304 | ], 305 | "dev": true, 306 | "optional": true, 307 | "os": [ 308 | "linux" 309 | ], 310 | "engines": { 311 | "node": ">=18" 312 | } 313 | }, 314 | "node_modules/@esbuild/linux-loong64": { 315 | "version": "0.23.1", 316 | "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.23.1.tgz", 317 | "integrity": "sha512-Vx09LzEoBa5zDnieH8LSMRToj7ir/Jeq0Gu6qJ/1GcBq9GkfoEAoXvLiW1U9J1qE/Y/Oyaq33w5p2ZWrNNHNEw==", 318 | "cpu": [ 319 | "loong64" 320 | ], 321 | "dev": true, 322 | "optional": true, 323 | "os": [ 324 | "linux" 325 | ], 326 | "engines": { 327 | "node": ">=18" 328 | } 329 | }, 330 | "node_modules/@esbuild/linux-mips64el": { 331 | "version": "0.23.1", 332 | "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.23.1.tgz", 333 | "integrity": "sha512-nrFzzMQ7W4WRLNUOU5dlWAqa6yVeI0P78WKGUo7lg2HShq/yx+UYkeNSE0SSfSure0SqgnsxPvmAUu/vu0E+3Q==", 334 | "cpu": [ 335 | "mips64el" 336 | ], 337 | "dev": true, 338 | "optional": true, 339 | "os": [ 340 | "linux" 341 | ], 342 | "engines": { 343 | "node": ">=18" 344 | } 345 | }, 346 | "node_modules/@esbuild/linux-ppc64": { 347 | "version": "0.23.1", 348 | "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.23.1.tgz", 349 | "integrity": "sha512-dKN8fgVqd0vUIjxuJI6P/9SSSe/mB9rvA98CSH2sJnlZ/OCZWO1DJvxj8jvKTfYUdGfcq2dDxoKaC6bHuTlgcw==", 350 | "cpu": [ 351 | "ppc64" 352 | ], 353 | "dev": true, 354 | "optional": true, 355 | "os": [ 356 | "linux" 357 | ], 358 | "engines": { 359 | "node": ">=18" 360 | } 361 | }, 362 | "node_modules/@esbuild/linux-riscv64": { 363 | "version": "0.23.1", 364 | "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.23.1.tgz", 365 | "integrity": "sha512-5AV4Pzp80fhHL83JM6LoA6pTQVWgB1HovMBsLQ9OZWLDqVY8MVobBXNSmAJi//Csh6tcY7e7Lny2Hg1tElMjIA==", 366 | "cpu": [ 367 | "riscv64" 368 | ], 369 | "dev": true, 370 | "optional": true, 371 | "os": [ 372 | "linux" 373 | ], 374 | "engines": { 375 | "node": ">=18" 376 | } 377 | }, 378 | "node_modules/@esbuild/linux-s390x": { 379 | "version": "0.23.1", 380 | "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.23.1.tgz", 381 | "integrity": "sha512-9ygs73tuFCe6f6m/Tb+9LtYxWR4c9yg7zjt2cYkjDbDpV/xVn+68cQxMXCjUpYwEkze2RcU/rMnfIXNRFmSoDw==", 382 | "cpu": [ 383 | "s390x" 384 | ], 385 | "dev": true, 386 | "optional": true, 387 | "os": [ 388 | "linux" 389 | ], 390 | "engines": { 391 | "node": ">=18" 392 | } 393 | }, 394 | "node_modules/@esbuild/linux-x64": { 395 | "version": "0.23.1", 396 | "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.23.1.tgz", 397 | "integrity": "sha512-EV6+ovTsEXCPAp58g2dD68LxoP/wK5pRvgy0J/HxPGB009omFPv3Yet0HiaqvrIrgPTBuC6wCH1LTOY91EO5hQ==", 398 | "cpu": [ 399 | "x64" 400 | ], 401 | "dev": true, 402 | "optional": true, 403 | "os": [ 404 | "linux" 405 | ], 406 | "engines": { 407 | "node": ">=18" 408 | } 409 | }, 410 | "node_modules/@esbuild/netbsd-x64": { 411 | "version": "0.23.1", 412 | "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.23.1.tgz", 413 | "integrity": "sha512-aevEkCNu7KlPRpYLjwmdcuNz6bDFiE7Z8XC4CPqExjTvrHugh28QzUXVOZtiYghciKUacNktqxdpymplil1beA==", 414 | "cpu": [ 415 | "x64" 416 | ], 417 | "dev": true, 418 | "optional": true, 419 | "os": [ 420 | "netbsd" 421 | ], 422 | "engines": { 423 | "node": ">=18" 424 | } 425 | }, 426 | "node_modules/@esbuild/openbsd-arm64": { 427 | "version": "0.23.1", 428 | "resolved": "https://registry.npmjs.org/@esbuild/openbsd-arm64/-/openbsd-arm64-0.23.1.tgz", 429 | "integrity": "sha512-3x37szhLexNA4bXhLrCC/LImN/YtWis6WXr1VESlfVtVeoFJBRINPJ3f0a/6LV8zpikqoUg4hyXw0sFBt5Cr+Q==", 430 | "cpu": [ 431 | "arm64" 432 | ], 433 | "dev": true, 434 | "optional": true, 435 | "os": [ 436 | "openbsd" 437 | ], 438 | "engines": { 439 | "node": ">=18" 440 | } 441 | }, 442 | "node_modules/@esbuild/openbsd-x64": { 443 | "version": "0.23.1", 444 | "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.23.1.tgz", 445 | "integrity": "sha512-aY2gMmKmPhxfU+0EdnN+XNtGbjfQgwZj43k8G3fyrDM/UdZww6xrWxmDkuz2eCZchqVeABjV5BpildOrUbBTqA==", 446 | "cpu": [ 447 | "x64" 448 | ], 449 | "dev": true, 450 | "optional": true, 451 | "os": [ 452 | "openbsd" 453 | ], 454 | "engines": { 455 | "node": ">=18" 456 | } 457 | }, 458 | "node_modules/@esbuild/sunos-x64": { 459 | "version": "0.23.1", 460 | "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.23.1.tgz", 461 | "integrity": "sha512-RBRT2gqEl0IKQABT4XTj78tpk9v7ehp+mazn2HbUeZl1YMdaGAQqhapjGTCe7uw7y0frDi4gS0uHzhvpFuI1sA==", 462 | "cpu": [ 463 | "x64" 464 | ], 465 | "dev": true, 466 | "optional": true, 467 | "os": [ 468 | "sunos" 469 | ], 470 | "engines": { 471 | "node": ">=18" 472 | } 473 | }, 474 | "node_modules/@esbuild/win32-arm64": { 475 | "version": "0.23.1", 476 | "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.23.1.tgz", 477 | "integrity": "sha512-4O+gPR5rEBe2FpKOVyiJ7wNDPA8nGzDuJ6gN4okSA1gEOYZ67N8JPk58tkWtdtPeLz7lBnY6I5L3jdsr3S+A6A==", 478 | "cpu": [ 479 | "arm64" 480 | ], 481 | "dev": true, 482 | "optional": true, 483 | "os": [ 484 | "win32" 485 | ], 486 | "engines": { 487 | "node": ">=18" 488 | } 489 | }, 490 | "node_modules/@esbuild/win32-ia32": { 491 | "version": "0.23.1", 492 | "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.23.1.tgz", 493 | "integrity": "sha512-BcaL0Vn6QwCwre3Y717nVHZbAa4UBEigzFm6VdsVdT/MbZ38xoj1X9HPkZhbmaBGUD1W8vxAfffbDe8bA6AKnQ==", 494 | "cpu": [ 495 | "ia32" 496 | ], 497 | "dev": true, 498 | "optional": true, 499 | "os": [ 500 | "win32" 501 | ], 502 | "engines": { 503 | "node": ">=18" 504 | } 505 | }, 506 | "node_modules/@esbuild/win32-x64": { 507 | "version": "0.23.1", 508 | "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.23.1.tgz", 509 | "integrity": "sha512-BHpFFeslkWrXWyUPnbKm+xYYVYruCinGcftSBaa8zoF9hZO4BcSCFUvHVTtzpIY6YzUnYtuEhZ+C9iEXjxnasg==", 510 | "cpu": [ 511 | "x64" 512 | ], 513 | "dev": true, 514 | "optional": true, 515 | "os": [ 516 | "win32" 517 | ], 518 | "engines": { 519 | "node": ">=18" 520 | } 521 | }, 522 | "node_modules/@eslint-community/eslint-utils": { 523 | "version": "4.4.1", 524 | "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.4.1.tgz", 525 | "integrity": "sha512-s3O3waFUrMV8P/XaF/+ZTp1X9XBZW1a4B97ZnjQF2KYWaFD2A8KyFBsrsfSjEmjn3RGWAIuvlneuZm3CUK3jbA==", 526 | "dev": true, 527 | "dependencies": { 528 | "eslint-visitor-keys": "^3.4.3" 529 | }, 530 | "engines": { 531 | "node": "^12.22.0 || ^14.17.0 || >=16.0.0" 532 | }, 533 | "funding": { 534 | "url": "https://opencollective.com/eslint" 535 | }, 536 | "peerDependencies": { 537 | "eslint": "^6.0.0 || ^7.0.0 || >=8.0.0" 538 | } 539 | }, 540 | "node_modules/@eslint-community/regexpp": { 541 | "version": "4.12.1", 542 | "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.12.1.tgz", 543 | "integrity": "sha512-CCZCDJuduB9OUkFkY2IgppNZMi2lBQgD2qzwXkEia16cge2pijY/aXi96CJMquDMn3nJdlPV1A5KrJEXwfLNzQ==", 544 | "dev": true, 545 | "engines": { 546 | "node": "^12.0.0 || ^14.0.0 || >=16.0.0" 547 | } 548 | }, 549 | "node_modules/@eslint/eslintrc": { 550 | "version": "2.1.4", 551 | "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-2.1.4.tgz", 552 | "integrity": "sha512-269Z39MS6wVJtsoUl10L60WdkhJVdPG24Q4eZTH3nnF6lpvSShEK3wQjDX9JRWAUPvPh7COouPpU9IrqaZFvtQ==", 553 | "dev": true, 554 | "dependencies": { 555 | "ajv": "^6.12.4", 556 | "debug": "^4.3.2", 557 | "espree": "^9.6.0", 558 | "globals": "^13.19.0", 559 | "ignore": "^5.2.0", 560 | "import-fresh": "^3.2.1", 561 | "js-yaml": "^4.1.0", 562 | "minimatch": "^3.1.2", 563 | "strip-json-comments": "^3.1.1" 564 | }, 565 | "engines": { 566 | "node": "^12.22.0 || ^14.17.0 || >=16.0.0" 567 | }, 568 | "funding": { 569 | "url": "https://opencollective.com/eslint" 570 | } 571 | }, 572 | "node_modules/@eslint/eslintrc/node_modules/brace-expansion": { 573 | "version": "1.1.11", 574 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 575 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 576 | "dev": true, 577 | "dependencies": { 578 | "balanced-match": "^1.0.0", 579 | "concat-map": "0.0.1" 580 | } 581 | }, 582 | "node_modules/@eslint/eslintrc/node_modules/minimatch": { 583 | "version": "3.1.2", 584 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", 585 | "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", 586 | "dev": true, 587 | "dependencies": { 588 | "brace-expansion": "^1.1.7" 589 | }, 590 | "engines": { 591 | "node": "*" 592 | } 593 | }, 594 | "node_modules/@eslint/js": { 595 | "version": "8.57.1", 596 | "resolved": "https://registry.npmjs.org/@eslint/js/-/js-8.57.1.tgz", 597 | "integrity": "sha512-d9zaMRSTIKDLhctzH12MtXvJKSSUhaHcjV+2Z+GK+EEY7XKpP5yR4x+N3TAcHTcu963nIr+TMcCb4DBCYX1z6Q==", 598 | "dev": true, 599 | "engines": { 600 | "node": "^12.22.0 || ^14.17.0 || >=16.0.0" 601 | } 602 | }, 603 | "node_modules/@humanwhocodes/config-array": { 604 | "version": "0.13.0", 605 | "resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.13.0.tgz", 606 | "integrity": "sha512-DZLEEqFWQFiyK6h5YIeynKx7JlvCYWL0cImfSRXZ9l4Sg2efkFGTuFf6vzXjK1cq6IYkU+Eg/JizXw+TD2vRNw==", 607 | "deprecated": "Use @eslint/config-array instead", 608 | "dev": true, 609 | "dependencies": { 610 | "@humanwhocodes/object-schema": "^2.0.3", 611 | "debug": "^4.3.1", 612 | "minimatch": "^3.0.5" 613 | }, 614 | "engines": { 615 | "node": ">=10.10.0" 616 | } 617 | }, 618 | "node_modules/@humanwhocodes/config-array/node_modules/brace-expansion": { 619 | "version": "1.1.11", 620 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 621 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 622 | "dev": true, 623 | "dependencies": { 624 | "balanced-match": "^1.0.0", 625 | "concat-map": "0.0.1" 626 | } 627 | }, 628 | "node_modules/@humanwhocodes/config-array/node_modules/minimatch": { 629 | "version": "3.1.2", 630 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", 631 | "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", 632 | "dev": true, 633 | "dependencies": { 634 | "brace-expansion": "^1.1.7" 635 | }, 636 | "engines": { 637 | "node": "*" 638 | } 639 | }, 640 | "node_modules/@humanwhocodes/module-importer": { 641 | "version": "1.0.1", 642 | "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz", 643 | "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==", 644 | "dev": true, 645 | "engines": { 646 | "node": ">=12.22" 647 | }, 648 | "funding": { 649 | "type": "github", 650 | "url": "https://github.com/sponsors/nzakas" 651 | } 652 | }, 653 | "node_modules/@humanwhocodes/object-schema": { 654 | "version": "2.0.3", 655 | "resolved": "https://registry.npmjs.org/@humanwhocodes/object-schema/-/object-schema-2.0.3.tgz", 656 | "integrity": "sha512-93zYdMES/c1D69yZiKDBj0V24vqNzB/koF26KPaagAfd3P/4gUlh3Dys5ogAK+Exi9QyzlD8x/08Zt7wIKcDcA==", 657 | "deprecated": "Use @eslint/object-schema instead", 658 | "dev": true 659 | }, 660 | "node_modules/@modelcontextprotocol/sdk": { 661 | "version": "1.4.1", 662 | "resolved": "https://registry.npmjs.org/@modelcontextprotocol/sdk/-/sdk-1.4.1.tgz", 663 | "integrity": "sha512-wS6YC4lkUZ9QpP+/7NBTlVNiEvsnyl0xF7rRusLF+RsG0xDPc/zWR7fEEyhKnnNutGsDAZh59l/AeoWGwIb1+g==", 664 | "dependencies": { 665 | "content-type": "^1.0.5", 666 | "eventsource": "^3.0.2", 667 | "raw-body": "^3.0.0", 668 | "zod": "^3.23.8", 669 | "zod-to-json-schema": "^3.24.1" 670 | }, 671 | "engines": { 672 | "node": ">=18" 673 | } 674 | }, 675 | "node_modules/@nodelib/fs.scandir": { 676 | "version": "2.1.5", 677 | "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", 678 | "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", 679 | "dev": true, 680 | "dependencies": { 681 | "@nodelib/fs.stat": "2.0.5", 682 | "run-parallel": "^1.1.9" 683 | }, 684 | "engines": { 685 | "node": ">= 8" 686 | } 687 | }, 688 | "node_modules/@nodelib/fs.stat": { 689 | "version": "2.0.5", 690 | "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", 691 | "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", 692 | "dev": true, 693 | "engines": { 694 | "node": ">= 8" 695 | } 696 | }, 697 | "node_modules/@nodelib/fs.walk": { 698 | "version": "1.2.8", 699 | "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz", 700 | "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==", 701 | "dev": true, 702 | "dependencies": { 703 | "@nodelib/fs.scandir": "2.1.5", 704 | "fastq": "^1.6.0" 705 | }, 706 | "engines": { 707 | "node": ">= 8" 708 | } 709 | }, 710 | "node_modules/@openrouter/ai-sdk-provider": { 711 | "version": "0.2.0", 712 | "resolved": "https://registry.npmjs.org/@openrouter/ai-sdk-provider/-/ai-sdk-provider-0.2.0.tgz", 713 | "integrity": "sha512-jaEG/T9JKu6yMC0qn91jRYlwtjeddsM7w4tSKUBwXuftK+7HnYMyzqZ2mFNF/WZ6GKPWMN/Ipmk3l4JyJMXUAQ==", 714 | "dev": true, 715 | "dependencies": { 716 | "@ai-sdk/provider": "1.0.6", 717 | "@ai-sdk/provider-utils": "2.1.5" 718 | }, 719 | "engines": { 720 | "node": ">=18" 721 | }, 722 | "peerDependencies": { 723 | "zod": "^3.0.0" 724 | } 725 | }, 726 | "node_modules/@openrouter/ai-sdk-provider/node_modules/@ai-sdk/provider": { 727 | "version": "1.0.6", 728 | "resolved": "https://registry.npmjs.org/@ai-sdk/provider/-/provider-1.0.6.tgz", 729 | "integrity": "sha512-hwj/gFNxpDgEfTaYzCYoslmw01IY9kWLKl/wf8xuPvHtQIzlfXWmmUwc8PnCwxyt8cKzIuV0dfUghCf68HQ0SA==", 730 | "dev": true, 731 | "dependencies": { 732 | "json-schema": "^0.4.0" 733 | }, 734 | "engines": { 735 | "node": ">=18" 736 | } 737 | }, 738 | "node_modules/@openrouter/ai-sdk-provider/node_modules/@ai-sdk/provider-utils": { 739 | "version": "2.1.5", 740 | "resolved": "https://registry.npmjs.org/@ai-sdk/provider-utils/-/provider-utils-2.1.5.tgz", 741 | "integrity": "sha512-PcNR7E4ovZGV/J47gUqaFlvzorgca6uUfN5WzfXJSFWeOeLunN+oxRVwgUOwj0zbmO0yGQTHQD+FHVw8s3Rz8w==", 742 | "dev": true, 743 | "dependencies": { 744 | "@ai-sdk/provider": "1.0.6", 745 | "eventsource-parser": "^3.0.0", 746 | "nanoid": "^3.3.8", 747 | "secure-json-parse": "^2.7.0" 748 | }, 749 | "engines": { 750 | "node": ">=18" 751 | }, 752 | "peerDependencies": { 753 | "zod": "^3.0.0" 754 | }, 755 | "peerDependenciesMeta": { 756 | "zod": { 757 | "optional": true 758 | } 759 | } 760 | }, 761 | "node_modules/@opentelemetry/api": { 762 | "version": "1.9.0", 763 | "resolved": "https://registry.npmjs.org/@opentelemetry/api/-/api-1.9.0.tgz", 764 | "integrity": "sha512-3giAOQvZiH5F9bMlMiv8+GSPMeqg0dbaeo58/0SlA9sxSqZhnUtxzX9/2FzyhS9sWQf5S0GJE0AKBrFqjpeYcg==", 765 | "engines": { 766 | "node": ">=8.0.0" 767 | } 768 | }, 769 | "node_modules/@pkgr/core": { 770 | "version": "0.1.1", 771 | "resolved": "https://registry.npmjs.org/@pkgr/core/-/core-0.1.1.tgz", 772 | "integrity": "sha512-cq8o4cWH0ibXh9VGi5P20Tu9XF/0fFXl9EUinr9QfTM7a7p0oTA4iJRCQWppXR1Pg8dSM0UCItCkPwsk9qWWYA==", 773 | "dev": true, 774 | "engines": { 775 | "node": "^12.20.0 || ^14.18.0 || >=16.0.0" 776 | }, 777 | "funding": { 778 | "url": "https://opencollective.com/unts" 779 | } 780 | }, 781 | "node_modules/@types/body-parser": { 782 | "version": "1.19.5", 783 | "resolved": "https://registry.npmjs.org/@types/body-parser/-/body-parser-1.19.5.tgz", 784 | "integrity": "sha512-fB3Zu92ucau0iQ0JMCFQE7b/dv8Ot07NI3KaZIkIUNXq82k4eBAqUaneXfleGY9JWskeS9y+u0nXMyspcuQrCg==", 785 | "dev": true, 786 | "dependencies": { 787 | "@types/connect": "*", 788 | "@types/node": "*" 789 | } 790 | }, 791 | "node_modules/@types/connect": { 792 | "version": "3.4.38", 793 | "resolved": "https://registry.npmjs.org/@types/connect/-/connect-3.4.38.tgz", 794 | "integrity": "sha512-K6uROf1LD88uDQqJCktA4yzL1YYAK6NgfsI0v/mTgyPKWsX1CnJ0XPSDhViejru1GcRkLWb8RlzFYJRqGUbaug==", 795 | "dev": true, 796 | "dependencies": { 797 | "@types/node": "*" 798 | } 799 | }, 800 | "node_modules/@types/cors": { 801 | "version": "2.8.17", 802 | "resolved": "https://registry.npmjs.org/@types/cors/-/cors-2.8.17.tgz", 803 | "integrity": "sha512-8CGDvrBj1zgo2qE+oS3pOCyYNqCPryMWY2bGfwA0dcfopWGgxs+78df0Rs3rc9THP4JkOhLsAa+15VdpAqkcUA==", 804 | "dev": true, 805 | "dependencies": { 806 | "@types/node": "*" 807 | } 808 | }, 809 | "node_modules/@types/diff-match-patch": { 810 | "version": "1.0.36", 811 | "resolved": "https://registry.npmjs.org/@types/diff-match-patch/-/diff-match-patch-1.0.36.tgz", 812 | "integrity": "sha512-xFdR6tkm0MWvBfO8xXCSsinYxHcqkQUlcHeSpMC2ukzOb6lwQAfDmW+Qt0AvlGd8HpsS28qKsB+oPeJn9I39jg==" 813 | }, 814 | "node_modules/@types/express": { 815 | "version": "4.17.21", 816 | "resolved": "https://registry.npmjs.org/@types/express/-/express-4.17.21.tgz", 817 | "integrity": "sha512-ejlPM315qwLpaQlQDTjPdsUFSc6ZsP4AN6AlWnogPjQ7CVi7PYF3YVz+CY3jE2pwYf7E/7HlDAN0rV2GxTG0HQ==", 818 | "dev": true, 819 | "dependencies": { 820 | "@types/body-parser": "*", 821 | "@types/express-serve-static-core": "^4.17.33", 822 | "@types/qs": "*", 823 | "@types/serve-static": "*" 824 | } 825 | }, 826 | "node_modules/@types/express-serve-static-core": { 827 | "version": "4.19.6", 828 | "resolved": "https://registry.npmjs.org/@types/express-serve-static-core/-/express-serve-static-core-4.19.6.tgz", 829 | "integrity": "sha512-N4LZ2xG7DatVqhCZzOGb1Yi5lMbXSZcmdLDe9EzSndPV2HpWYWzRbaerl2n27irrm94EPpprqa8KpskPT085+A==", 830 | "dev": true, 831 | "dependencies": { 832 | "@types/node": "*", 833 | "@types/qs": "*", 834 | "@types/range-parser": "*", 835 | "@types/send": "*" 836 | } 837 | }, 838 | "node_modules/@types/http-errors": { 839 | "version": "2.0.4", 840 | "resolved": "https://registry.npmjs.org/@types/http-errors/-/http-errors-2.0.4.tgz", 841 | "integrity": "sha512-D0CFMMtydbJAegzOyHjtiKPLlvnm3iTZyZRSZoLq2mRhDdmLfIWOCYPfQJ4cu2erKghU++QvjcUjp/5h7hESpA==", 842 | "dev": true 843 | }, 844 | "node_modules/@types/json-schema": { 845 | "version": "7.0.15", 846 | "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz", 847 | "integrity": "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==", 848 | "dev": true 849 | }, 850 | "node_modules/@types/mime": { 851 | "version": "1.3.5", 852 | "resolved": "https://registry.npmjs.org/@types/mime/-/mime-1.3.5.tgz", 853 | "integrity": "sha512-/pyBZWSLD2n0dcHE3hq8s8ZvcETHtEuF+3E7XVt0Ig2nvsVQXdghHVcEkIWjy9A0wKfTn97a/PSDYohKIlnP/w==", 854 | "dev": true 855 | }, 856 | "node_modules/@types/node": { 857 | "version": "20.17.17", 858 | "resolved": "https://registry.npmjs.org/@types/node/-/node-20.17.17.tgz", 859 | "integrity": "sha512-/WndGO4kIfMicEQLTi/mDANUu/iVUhT7KboZPdEqqHQ4aTS+3qT3U5gIqWDFV+XouorjfgGqvKILJeHhuQgFYg==", 860 | "dev": true, 861 | "dependencies": { 862 | "undici-types": "~6.19.2" 863 | } 864 | }, 865 | "node_modules/@types/qs": { 866 | "version": "6.9.18", 867 | "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.9.18.tgz", 868 | "integrity": "sha512-kK7dgTYDyGqS+e2Q4aK9X3D7q234CIZ1Bv0q/7Z5IwRDoADNU81xXJK/YVyLbLTZCoIwUoDoffFeF+p/eIklAA==", 869 | "dev": true 870 | }, 871 | "node_modules/@types/range-parser": { 872 | "version": "1.2.7", 873 | "resolved": "https://registry.npmjs.org/@types/range-parser/-/range-parser-1.2.7.tgz", 874 | "integrity": "sha512-hKormJbkJqzQGhziax5PItDUTMAM9uE2XXQmM37dyd4hVM+5aVl7oVxMVUiVQn2oCQFN/LKCZdvSM0pFRqbSmQ==", 875 | "dev": true 876 | }, 877 | "node_modules/@types/semver": { 878 | "version": "7.5.8", 879 | "resolved": "https://registry.npmjs.org/@types/semver/-/semver-7.5.8.tgz", 880 | "integrity": "sha512-I8EUhyrgfLrcTkzV3TSsGyl1tSuPrEDzr0yd5m90UgNxQkyDXULk3b6MlQqTCpZpNtWe1K0hzclnZkTcLBe2UQ==", 881 | "dev": true 882 | }, 883 | "node_modules/@types/send": { 884 | "version": "0.17.4", 885 | "resolved": "https://registry.npmjs.org/@types/send/-/send-0.17.4.tgz", 886 | "integrity": "sha512-x2EM6TJOybec7c52BX0ZspPodMsQUd5L6PRwOunVyVUhXiBSKf3AezDL8Dgvgt5o0UfKNfuA0eMLr2wLT4AiBA==", 887 | "dev": true, 888 | "dependencies": { 889 | "@types/mime": "^1", 890 | "@types/node": "*" 891 | } 892 | }, 893 | "node_modules/@types/serve-static": { 894 | "version": "1.15.7", 895 | "resolved": "https://registry.npmjs.org/@types/serve-static/-/serve-static-1.15.7.tgz", 896 | "integrity": "sha512-W8Ym+h8nhuRwaKPaDw34QUkwsGi6Rc4yYqvKFo5rm2FUEhCFbzVWrxXUxuKK8TASjWsysJY0nsmNCGhCOIsrOw==", 897 | "dev": true, 898 | "dependencies": { 899 | "@types/http-errors": "*", 900 | "@types/node": "*", 901 | "@types/send": "*" 902 | } 903 | }, 904 | "node_modules/@typescript-eslint/eslint-plugin": { 905 | "version": "6.21.0", 906 | "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-6.21.0.tgz", 907 | "integrity": "sha512-oy9+hTPCUFpngkEZUSzbf9MxI65wbKFoQYsgPdILTfbUldp5ovUuphZVe4i30emU9M/kP+T64Di0mxl7dSw3MA==", 908 | "dev": true, 909 | "dependencies": { 910 | "@eslint-community/regexpp": "^4.5.1", 911 | "@typescript-eslint/scope-manager": "6.21.0", 912 | "@typescript-eslint/type-utils": "6.21.0", 913 | "@typescript-eslint/utils": "6.21.0", 914 | "@typescript-eslint/visitor-keys": "6.21.0", 915 | "debug": "^4.3.4", 916 | "graphemer": "^1.4.0", 917 | "ignore": "^5.2.4", 918 | "natural-compare": "^1.4.0", 919 | "semver": "^7.5.4", 920 | "ts-api-utils": "^1.0.1" 921 | }, 922 | "engines": { 923 | "node": "^16.0.0 || >=18.0.0" 924 | }, 925 | "funding": { 926 | "type": "opencollective", 927 | "url": "https://opencollective.com/typescript-eslint" 928 | }, 929 | "peerDependencies": { 930 | "@typescript-eslint/parser": "^6.0.0 || ^6.0.0-alpha", 931 | "eslint": "^7.0.0 || ^8.0.0" 932 | }, 933 | "peerDependenciesMeta": { 934 | "typescript": { 935 | "optional": true 936 | } 937 | } 938 | }, 939 | "node_modules/@typescript-eslint/parser": { 940 | "version": "6.21.0", 941 | "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-6.21.0.tgz", 942 | "integrity": "sha512-tbsV1jPne5CkFQCgPBcDOt30ItF7aJoZL997JSF7MhGQqOeT3svWRYxiqlfA5RUdlHN6Fi+EI9bxqbdyAUZjYQ==", 943 | "dev": true, 944 | "dependencies": { 945 | "@typescript-eslint/scope-manager": "6.21.0", 946 | "@typescript-eslint/types": "6.21.0", 947 | "@typescript-eslint/typescript-estree": "6.21.0", 948 | "@typescript-eslint/visitor-keys": "6.21.0", 949 | "debug": "^4.3.4" 950 | }, 951 | "engines": { 952 | "node": "^16.0.0 || >=18.0.0" 953 | }, 954 | "funding": { 955 | "type": "opencollective", 956 | "url": "https://opencollective.com/typescript-eslint" 957 | }, 958 | "peerDependencies": { 959 | "eslint": "^7.0.0 || ^8.0.0" 960 | }, 961 | "peerDependenciesMeta": { 962 | "typescript": { 963 | "optional": true 964 | } 965 | } 966 | }, 967 | "node_modules/@typescript-eslint/scope-manager": { 968 | "version": "6.21.0", 969 | "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-6.21.0.tgz", 970 | "integrity": "sha512-OwLUIWZJry80O99zvqXVEioyniJMa+d2GrqpUTqi5/v5D5rOrppJVBPa0yKCblcigC0/aYAzxxqQ1B+DS2RYsg==", 971 | "dev": true, 972 | "dependencies": { 973 | "@typescript-eslint/types": "6.21.0", 974 | "@typescript-eslint/visitor-keys": "6.21.0" 975 | }, 976 | "engines": { 977 | "node": "^16.0.0 || >=18.0.0" 978 | }, 979 | "funding": { 980 | "type": "opencollective", 981 | "url": "https://opencollective.com/typescript-eslint" 982 | } 983 | }, 984 | "node_modules/@typescript-eslint/type-utils": { 985 | "version": "6.21.0", 986 | "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-6.21.0.tgz", 987 | "integrity": "sha512-rZQI7wHfao8qMX3Rd3xqeYSMCL3SoiSQLBATSiVKARdFGCYSRvmViieZjqc58jKgs8Y8i9YvVVhRbHSTA4VBag==", 988 | "dev": true, 989 | "dependencies": { 990 | "@typescript-eslint/typescript-estree": "6.21.0", 991 | "@typescript-eslint/utils": "6.21.0", 992 | "debug": "^4.3.4", 993 | "ts-api-utils": "^1.0.1" 994 | }, 995 | "engines": { 996 | "node": "^16.0.0 || >=18.0.0" 997 | }, 998 | "funding": { 999 | "type": "opencollective", 1000 | "url": "https://opencollective.com/typescript-eslint" 1001 | }, 1002 | "peerDependencies": { 1003 | "eslint": "^7.0.0 || ^8.0.0" 1004 | }, 1005 | "peerDependenciesMeta": { 1006 | "typescript": { 1007 | "optional": true 1008 | } 1009 | } 1010 | }, 1011 | "node_modules/@typescript-eslint/types": { 1012 | "version": "6.21.0", 1013 | "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-6.21.0.tgz", 1014 | "integrity": "sha512-1kFmZ1rOm5epu9NZEZm1kckCDGj5UJEf7P1kliH4LKu/RkwpsfqqGmY2OOcUs18lSlQBKLDYBOGxRVtrMN5lpg==", 1015 | "dev": true, 1016 | "engines": { 1017 | "node": "^16.0.0 || >=18.0.0" 1018 | }, 1019 | "funding": { 1020 | "type": "opencollective", 1021 | "url": "https://opencollective.com/typescript-eslint" 1022 | } 1023 | }, 1024 | "node_modules/@typescript-eslint/typescript-estree": { 1025 | "version": "6.21.0", 1026 | "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-6.21.0.tgz", 1027 | "integrity": "sha512-6npJTkZcO+y2/kr+z0hc4HwNfrrP4kNYh57ek7yCNlrBjWQ1Y0OS7jiZTkgumrvkX5HkEKXFZkkdFNkaW2wmUQ==", 1028 | "dev": true, 1029 | "dependencies": { 1030 | "@typescript-eslint/types": "6.21.0", 1031 | "@typescript-eslint/visitor-keys": "6.21.0", 1032 | "debug": "^4.3.4", 1033 | "globby": "^11.1.0", 1034 | "is-glob": "^4.0.3", 1035 | "minimatch": "9.0.3", 1036 | "semver": "^7.5.4", 1037 | "ts-api-utils": "^1.0.1" 1038 | }, 1039 | "engines": { 1040 | "node": "^16.0.0 || >=18.0.0" 1041 | }, 1042 | "funding": { 1043 | "type": "opencollective", 1044 | "url": "https://opencollective.com/typescript-eslint" 1045 | }, 1046 | "peerDependenciesMeta": { 1047 | "typescript": { 1048 | "optional": true 1049 | } 1050 | } 1051 | }, 1052 | "node_modules/@typescript-eslint/utils": { 1053 | "version": "6.21.0", 1054 | "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-6.21.0.tgz", 1055 | "integrity": "sha512-NfWVaC8HP9T8cbKQxHcsJBY5YE1O33+jpMwN45qzWWaPDZgLIbo12toGMWnmhvCpd3sIxkpDw3Wv1B3dYrbDQQ==", 1056 | "dev": true, 1057 | "dependencies": { 1058 | "@eslint-community/eslint-utils": "^4.4.0", 1059 | "@types/json-schema": "^7.0.12", 1060 | "@types/semver": "^7.5.0", 1061 | "@typescript-eslint/scope-manager": "6.21.0", 1062 | "@typescript-eslint/types": "6.21.0", 1063 | "@typescript-eslint/typescript-estree": "6.21.0", 1064 | "semver": "^7.5.4" 1065 | }, 1066 | "engines": { 1067 | "node": "^16.0.0 || >=18.0.0" 1068 | }, 1069 | "funding": { 1070 | "type": "opencollective", 1071 | "url": "https://opencollective.com/typescript-eslint" 1072 | }, 1073 | "peerDependencies": { 1074 | "eslint": "^7.0.0 || ^8.0.0" 1075 | } 1076 | }, 1077 | "node_modules/@typescript-eslint/visitor-keys": { 1078 | "version": "6.21.0", 1079 | "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-6.21.0.tgz", 1080 | "integrity": "sha512-JJtkDduxLi9bivAB+cYOVMtbkqdPOhZ+ZI5LC47MIRrDV4Yn2o+ZnW10Nkmr28xRpSpdJ6Sm42Hjf2+REYXm0A==", 1081 | "dev": true, 1082 | "dependencies": { 1083 | "@typescript-eslint/types": "6.21.0", 1084 | "eslint-visitor-keys": "^3.4.1" 1085 | }, 1086 | "engines": { 1087 | "node": "^16.0.0 || >=18.0.0" 1088 | }, 1089 | "funding": { 1090 | "type": "opencollective", 1091 | "url": "https://opencollective.com/typescript-eslint" 1092 | } 1093 | }, 1094 | "node_modules/@ungap/structured-clone": { 1095 | "version": "1.3.0", 1096 | "resolved": "https://registry.npmjs.org/@ungap/structured-clone/-/structured-clone-1.3.0.tgz", 1097 | "integrity": "sha512-WmoN8qaIAo7WTYWbAZuG8PYEhn5fkz7dZrqTBZ7dtt//lL2Gwms1IcnQ5yHqjDfX8Ft5j4YzDM23f87zBfDe9g==", 1098 | "dev": true 1099 | }, 1100 | "node_modules/accepts": { 1101 | "version": "1.3.8", 1102 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.8.tgz", 1103 | "integrity": "sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==", 1104 | "dev": true, 1105 | "dependencies": { 1106 | "mime-types": "~2.1.34", 1107 | "negotiator": "0.6.3" 1108 | }, 1109 | "engines": { 1110 | "node": ">= 0.6" 1111 | } 1112 | }, 1113 | "node_modules/acorn": { 1114 | "version": "8.14.0", 1115 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.14.0.tgz", 1116 | "integrity": "sha512-cl669nCJTZBsL97OF4kUQm5g5hC2uihk0NxY3WENAC0TYdILVkAyHymAntgxGkl7K+t0cXIrH5siy5S4XkFycA==", 1117 | "dev": true, 1118 | "bin": { 1119 | "acorn": "bin/acorn" 1120 | }, 1121 | "engines": { 1122 | "node": ">=0.4.0" 1123 | } 1124 | }, 1125 | "node_modules/acorn-jsx": { 1126 | "version": "5.3.2", 1127 | "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", 1128 | "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", 1129 | "dev": true, 1130 | "peerDependencies": { 1131 | "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" 1132 | } 1133 | }, 1134 | "node_modules/ai": { 1135 | "version": "4.1.17", 1136 | "resolved": "https://registry.npmjs.org/ai/-/ai-4.1.17.tgz", 1137 | "integrity": "sha512-5SW15tXDuxE/wlEOjRKxLxTOUIGD4C9bIee+FCFvXHTTAZhHiQjViC2s7RtMUW+hbFtGya302jUHY1Pe8A/YuQ==", 1138 | "dependencies": { 1139 | "@ai-sdk/provider": "1.0.7", 1140 | "@ai-sdk/provider-utils": "2.1.6", 1141 | "@ai-sdk/react": "1.1.8", 1142 | "@ai-sdk/ui-utils": "1.1.8", 1143 | "@opentelemetry/api": "1.9.0", 1144 | "jsondiffpatch": "0.6.0" 1145 | }, 1146 | "engines": { 1147 | "node": ">=18" 1148 | }, 1149 | "peerDependencies": { 1150 | "react": "^18 || ^19 || ^19.0.0-rc", 1151 | "zod": "^3.0.0" 1152 | }, 1153 | "peerDependenciesMeta": { 1154 | "react": { 1155 | "optional": true 1156 | }, 1157 | "zod": { 1158 | "optional": true 1159 | } 1160 | } 1161 | }, 1162 | "node_modules/ajv": { 1163 | "version": "6.12.6", 1164 | "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", 1165 | "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", 1166 | "dev": true, 1167 | "dependencies": { 1168 | "fast-deep-equal": "^3.1.1", 1169 | "fast-json-stable-stringify": "^2.0.0", 1170 | "json-schema-traverse": "^0.4.1", 1171 | "uri-js": "^4.2.2" 1172 | }, 1173 | "funding": { 1174 | "type": "github", 1175 | "url": "https://github.com/sponsors/epoberezkin" 1176 | } 1177 | }, 1178 | "node_modules/ansi-regex": { 1179 | "version": "5.0.1", 1180 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", 1181 | "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", 1182 | "dev": true, 1183 | "engines": { 1184 | "node": ">=8" 1185 | } 1186 | }, 1187 | "node_modules/ansi-styles": { 1188 | "version": "4.3.0", 1189 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", 1190 | "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", 1191 | "dev": true, 1192 | "dependencies": { 1193 | "color-convert": "^2.0.1" 1194 | }, 1195 | "engines": { 1196 | "node": ">=8" 1197 | }, 1198 | "funding": { 1199 | "url": "https://github.com/chalk/ansi-styles?sponsor=1" 1200 | } 1201 | }, 1202 | "node_modules/argparse": { 1203 | "version": "2.0.1", 1204 | "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", 1205 | "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", 1206 | "dev": true 1207 | }, 1208 | "node_modules/array-flatten": { 1209 | "version": "1.1.1", 1210 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", 1211 | "integrity": "sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg==", 1212 | "dev": true 1213 | }, 1214 | "node_modules/array-union": { 1215 | "version": "2.1.0", 1216 | "resolved": "https://registry.npmjs.org/array-union/-/array-union-2.1.0.tgz", 1217 | "integrity": "sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==", 1218 | "dev": true, 1219 | "engines": { 1220 | "node": ">=8" 1221 | } 1222 | }, 1223 | "node_modules/asynckit": { 1224 | "version": "0.4.0", 1225 | "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", 1226 | "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==" 1227 | }, 1228 | "node_modules/axios": { 1229 | "version": "1.7.9", 1230 | "resolved": "https://registry.npmjs.org/axios/-/axios-1.7.9.tgz", 1231 | "integrity": "sha512-LhLcE7Hbiryz8oMDdDptSrWowmB4Bl6RCt6sIJKpRB4XtVf0iEgewX3au/pJqm+Py1kCASkb/FFKjxQaLtxJvw==", 1232 | "dependencies": { 1233 | "follow-redirects": "^1.15.6", 1234 | "form-data": "^4.0.0", 1235 | "proxy-from-env": "^1.1.0" 1236 | } 1237 | }, 1238 | "node_modules/balanced-match": { 1239 | "version": "1.0.2", 1240 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", 1241 | "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", 1242 | "dev": true 1243 | }, 1244 | "node_modules/body-parser": { 1245 | "version": "1.20.3", 1246 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.20.3.tgz", 1247 | "integrity": "sha512-7rAxByjUMqQ3/bHJy7D6OGXvx/MMc4IqBn/X0fcM1QUcAItpZrBEYhWGem+tzXH90c+G01ypMcYJBO9Y30203g==", 1248 | "dev": true, 1249 | "dependencies": { 1250 | "bytes": "3.1.2", 1251 | "content-type": "~1.0.5", 1252 | "debug": "2.6.9", 1253 | "depd": "2.0.0", 1254 | "destroy": "1.2.0", 1255 | "http-errors": "2.0.0", 1256 | "iconv-lite": "0.4.24", 1257 | "on-finished": "2.4.1", 1258 | "qs": "6.13.0", 1259 | "raw-body": "2.5.2", 1260 | "type-is": "~1.6.18", 1261 | "unpipe": "1.0.0" 1262 | }, 1263 | "engines": { 1264 | "node": ">= 0.8", 1265 | "npm": "1.2.8000 || >= 1.4.16" 1266 | } 1267 | }, 1268 | "node_modules/body-parser/node_modules/debug": { 1269 | "version": "2.6.9", 1270 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 1271 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 1272 | "dev": true, 1273 | "dependencies": { 1274 | "ms": "2.0.0" 1275 | } 1276 | }, 1277 | "node_modules/body-parser/node_modules/ms": { 1278 | "version": "2.0.0", 1279 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 1280 | "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==", 1281 | "dev": true 1282 | }, 1283 | "node_modules/body-parser/node_modules/raw-body": { 1284 | "version": "2.5.2", 1285 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.5.2.tgz", 1286 | "integrity": "sha512-8zGqypfENjCIqGhgXToC8aB2r7YrBX+AQAfIPs/Mlk+BtPTztOvTS01NRW/3Eh60J+a48lt8qsCzirQ6loCVfA==", 1287 | "dev": true, 1288 | "dependencies": { 1289 | "bytes": "3.1.2", 1290 | "http-errors": "2.0.0", 1291 | "iconv-lite": "0.4.24", 1292 | "unpipe": "1.0.0" 1293 | }, 1294 | "engines": { 1295 | "node": ">= 0.8" 1296 | } 1297 | }, 1298 | "node_modules/brace-expansion": { 1299 | "version": "2.0.1", 1300 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz", 1301 | "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==", 1302 | "dev": true, 1303 | "dependencies": { 1304 | "balanced-match": "^1.0.0" 1305 | } 1306 | }, 1307 | "node_modules/braces": { 1308 | "version": "3.0.3", 1309 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz", 1310 | "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==", 1311 | "dev": true, 1312 | "dependencies": { 1313 | "fill-range": "^7.1.1" 1314 | }, 1315 | "engines": { 1316 | "node": ">=8" 1317 | } 1318 | }, 1319 | "node_modules/bytes": { 1320 | "version": "3.1.2", 1321 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz", 1322 | "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==", 1323 | "engines": { 1324 | "node": ">= 0.8" 1325 | } 1326 | }, 1327 | "node_modules/call-bind-apply-helpers": { 1328 | "version": "1.0.1", 1329 | "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.1.tgz", 1330 | "integrity": "sha512-BhYE+WDaywFg2TBWYNXAE+8B1ATnThNBqXHP5nQu0jWJdVvY2hvkpyB3qOmtmDePiS5/BDQ8wASEWGMWRG148g==", 1331 | "dev": true, 1332 | "dependencies": { 1333 | "es-errors": "^1.3.0", 1334 | "function-bind": "^1.1.2" 1335 | }, 1336 | "engines": { 1337 | "node": ">= 0.4" 1338 | } 1339 | }, 1340 | "node_modules/call-bound": { 1341 | "version": "1.0.3", 1342 | "resolved": "https://registry.npmjs.org/call-bound/-/call-bound-1.0.3.tgz", 1343 | "integrity": "sha512-YTd+6wGlNlPxSuri7Y6X8tY2dmm12UMH66RpKMhiX6rsk5wXXnYgbUcOt8kiS31/AjfoTOvCsE+w8nZQLQnzHA==", 1344 | "dev": true, 1345 | "dependencies": { 1346 | "call-bind-apply-helpers": "^1.0.1", 1347 | "get-intrinsic": "^1.2.6" 1348 | }, 1349 | "engines": { 1350 | "node": ">= 0.4" 1351 | }, 1352 | "funding": { 1353 | "url": "https://github.com/sponsors/ljharb" 1354 | } 1355 | }, 1356 | "node_modules/callsites": { 1357 | "version": "3.1.0", 1358 | "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", 1359 | "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", 1360 | "dev": true, 1361 | "engines": { 1362 | "node": ">=6" 1363 | } 1364 | }, 1365 | "node_modules/chalk": { 1366 | "version": "4.1.2", 1367 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", 1368 | "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", 1369 | "dev": true, 1370 | "dependencies": { 1371 | "ansi-styles": "^4.1.0", 1372 | "supports-color": "^7.1.0" 1373 | }, 1374 | "engines": { 1375 | "node": ">=10" 1376 | }, 1377 | "funding": { 1378 | "url": "https://github.com/chalk/chalk?sponsor=1" 1379 | } 1380 | }, 1381 | "node_modules/color-convert": { 1382 | "version": "2.0.1", 1383 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", 1384 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", 1385 | "dev": true, 1386 | "dependencies": { 1387 | "color-name": "~1.1.4" 1388 | }, 1389 | "engines": { 1390 | "node": ">=7.0.0" 1391 | } 1392 | }, 1393 | "node_modules/color-name": { 1394 | "version": "1.1.4", 1395 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", 1396 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", 1397 | "dev": true 1398 | }, 1399 | "node_modules/combined-stream": { 1400 | "version": "1.0.8", 1401 | "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", 1402 | "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", 1403 | "dependencies": { 1404 | "delayed-stream": "~1.0.0" 1405 | }, 1406 | "engines": { 1407 | "node": ">= 0.8" 1408 | } 1409 | }, 1410 | "node_modules/concat-map": { 1411 | "version": "0.0.1", 1412 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 1413 | "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", 1414 | "dev": true 1415 | }, 1416 | "node_modules/content-disposition": { 1417 | "version": "0.5.4", 1418 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.4.tgz", 1419 | "integrity": "sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ==", 1420 | "dev": true, 1421 | "dependencies": { 1422 | "safe-buffer": "5.2.1" 1423 | }, 1424 | "engines": { 1425 | "node": ">= 0.6" 1426 | } 1427 | }, 1428 | "node_modules/content-type": { 1429 | "version": "1.0.5", 1430 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.5.tgz", 1431 | "integrity": "sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==", 1432 | "engines": { 1433 | "node": ">= 0.6" 1434 | } 1435 | }, 1436 | "node_modules/cookie": { 1437 | "version": "0.7.1", 1438 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.7.1.tgz", 1439 | "integrity": "sha512-6DnInpx7SJ2AK3+CTUE/ZM0vWTUboZCegxhC2xiIydHR9jNuTAASBrfEpHhiGOZw/nX51bHt6YQl8jsGo4y/0w==", 1440 | "dev": true, 1441 | "engines": { 1442 | "node": ">= 0.6" 1443 | } 1444 | }, 1445 | "node_modules/cookie-signature": { 1446 | "version": "1.0.6", 1447 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", 1448 | "integrity": "sha512-QADzlaHc8icV8I7vbaJXJwod9HWYp8uCqf1xa4OfNu1T7JVxQIrUgOWtHdNDtPiywmFbiS12VjotIXLrKM3orQ==", 1449 | "dev": true 1450 | }, 1451 | "node_modules/cors": { 1452 | "version": "2.8.5", 1453 | "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz", 1454 | "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==", 1455 | "dev": true, 1456 | "dependencies": { 1457 | "object-assign": "^4", 1458 | "vary": "^1" 1459 | }, 1460 | "engines": { 1461 | "node": ">= 0.10" 1462 | } 1463 | }, 1464 | "node_modules/cross-spawn": { 1465 | "version": "7.0.6", 1466 | "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", 1467 | "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==", 1468 | "dev": true, 1469 | "dependencies": { 1470 | "path-key": "^3.1.0", 1471 | "shebang-command": "^2.0.0", 1472 | "which": "^2.0.1" 1473 | }, 1474 | "engines": { 1475 | "node": ">= 8" 1476 | } 1477 | }, 1478 | "node_modules/debug": { 1479 | "version": "4.4.0", 1480 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.0.tgz", 1481 | "integrity": "sha512-6WTZ/IxCY/T6BALoZHaE4ctp9xm+Z5kY/pzYaCHRFeyVhojxlrm+46y68HA6hr0TcwEssoxNiDEUJQjfPZ/RYA==", 1482 | "dev": true, 1483 | "dependencies": { 1484 | "ms": "^2.1.3" 1485 | }, 1486 | "engines": { 1487 | "node": ">=6.0" 1488 | }, 1489 | "peerDependenciesMeta": { 1490 | "supports-color": { 1491 | "optional": true 1492 | } 1493 | } 1494 | }, 1495 | "node_modules/deep-is": { 1496 | "version": "0.1.4", 1497 | "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", 1498 | "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", 1499 | "dev": true 1500 | }, 1501 | "node_modules/delayed-stream": { 1502 | "version": "1.0.0", 1503 | "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", 1504 | "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==", 1505 | "engines": { 1506 | "node": ">=0.4.0" 1507 | } 1508 | }, 1509 | "node_modules/depd": { 1510 | "version": "2.0.0", 1511 | "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz", 1512 | "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==", 1513 | "engines": { 1514 | "node": ">= 0.8" 1515 | } 1516 | }, 1517 | "node_modules/dequal": { 1518 | "version": "2.0.3", 1519 | "resolved": "https://registry.npmjs.org/dequal/-/dequal-2.0.3.tgz", 1520 | "integrity": "sha512-0je+qPKHEMohvfRTCEo3CrPG6cAzAYgmzKyxRiYSSDkS6eGJdyVJm7WaYA5ECaAD9wLB2T4EEeymA5aFVcYXCA==", 1521 | "engines": { 1522 | "node": ">=6" 1523 | } 1524 | }, 1525 | "node_modules/destroy": { 1526 | "version": "1.2.0", 1527 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.2.0.tgz", 1528 | "integrity": "sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg==", 1529 | "dev": true, 1530 | "engines": { 1531 | "node": ">= 0.8", 1532 | "npm": "1.2.8000 || >= 1.4.16" 1533 | } 1534 | }, 1535 | "node_modules/diff-match-patch": { 1536 | "version": "1.0.5", 1537 | "resolved": "https://registry.npmjs.org/diff-match-patch/-/diff-match-patch-1.0.5.tgz", 1538 | "integrity": "sha512-IayShXAgj/QMXgB0IWmKx+rOPuGMhqm5w6jvFxmVenXKIzRqTAAsbBPT3kWQeGANj3jGgvcvv4yK6SxqYmikgw==" 1539 | }, 1540 | "node_modules/dir-glob": { 1541 | "version": "3.0.1", 1542 | "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-3.0.1.tgz", 1543 | "integrity": "sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==", 1544 | "dev": true, 1545 | "dependencies": { 1546 | "path-type": "^4.0.0" 1547 | }, 1548 | "engines": { 1549 | "node": ">=8" 1550 | } 1551 | }, 1552 | "node_modules/doctrine": { 1553 | "version": "3.0.0", 1554 | "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", 1555 | "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==", 1556 | "dev": true, 1557 | "dependencies": { 1558 | "esutils": "^2.0.2" 1559 | }, 1560 | "engines": { 1561 | "node": ">=6.0.0" 1562 | } 1563 | }, 1564 | "node_modules/dotenv": { 1565 | "version": "16.4.7", 1566 | "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.4.7.tgz", 1567 | "integrity": "sha512-47qPchRCykZC03FhkYAhrvwU4xDBFIj1QPqaarj6mdM/hgUzfPHcpkHJOn3mJAufFeeAxAzeGsr5X0M4k6fLZQ==", 1568 | "dev": true, 1569 | "engines": { 1570 | "node": ">=12" 1571 | }, 1572 | "funding": { 1573 | "url": "https://dotenvx.com" 1574 | } 1575 | }, 1576 | "node_modules/dunder-proto": { 1577 | "version": "1.0.1", 1578 | "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz", 1579 | "integrity": "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==", 1580 | "dev": true, 1581 | "dependencies": { 1582 | "call-bind-apply-helpers": "^1.0.1", 1583 | "es-errors": "^1.3.0", 1584 | "gopd": "^1.2.0" 1585 | }, 1586 | "engines": { 1587 | "node": ">= 0.4" 1588 | } 1589 | }, 1590 | "node_modules/ee-first": { 1591 | "version": "1.1.1", 1592 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", 1593 | "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==", 1594 | "dev": true 1595 | }, 1596 | "node_modules/encodeurl": { 1597 | "version": "2.0.0", 1598 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-2.0.0.tgz", 1599 | "integrity": "sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg==", 1600 | "dev": true, 1601 | "engines": { 1602 | "node": ">= 0.8" 1603 | } 1604 | }, 1605 | "node_modules/es-define-property": { 1606 | "version": "1.0.1", 1607 | "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz", 1608 | "integrity": "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==", 1609 | "dev": true, 1610 | "engines": { 1611 | "node": ">= 0.4" 1612 | } 1613 | }, 1614 | "node_modules/es-errors": { 1615 | "version": "1.3.0", 1616 | "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz", 1617 | "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==", 1618 | "dev": true, 1619 | "engines": { 1620 | "node": ">= 0.4" 1621 | } 1622 | }, 1623 | "node_modules/es-object-atoms": { 1624 | "version": "1.1.1", 1625 | "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.1.1.tgz", 1626 | "integrity": "sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA==", 1627 | "dev": true, 1628 | "dependencies": { 1629 | "es-errors": "^1.3.0" 1630 | }, 1631 | "engines": { 1632 | "node": ">= 0.4" 1633 | } 1634 | }, 1635 | "node_modules/esbuild": { 1636 | "version": "0.23.1", 1637 | "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.23.1.tgz", 1638 | "integrity": "sha512-VVNz/9Sa0bs5SELtn3f7qhJCDPCF5oMEl5cO9/SSinpE9hbPVvxbd572HH5AKiP7WD8INO53GgfDDhRjkylHEg==", 1639 | "dev": true, 1640 | "hasInstallScript": true, 1641 | "bin": { 1642 | "esbuild": "bin/esbuild" 1643 | }, 1644 | "engines": { 1645 | "node": ">=18" 1646 | }, 1647 | "optionalDependencies": { 1648 | "@esbuild/aix-ppc64": "0.23.1", 1649 | "@esbuild/android-arm": "0.23.1", 1650 | "@esbuild/android-arm64": "0.23.1", 1651 | "@esbuild/android-x64": "0.23.1", 1652 | "@esbuild/darwin-arm64": "0.23.1", 1653 | "@esbuild/darwin-x64": "0.23.1", 1654 | "@esbuild/freebsd-arm64": "0.23.1", 1655 | "@esbuild/freebsd-x64": "0.23.1", 1656 | "@esbuild/linux-arm": "0.23.1", 1657 | "@esbuild/linux-arm64": "0.23.1", 1658 | "@esbuild/linux-ia32": "0.23.1", 1659 | "@esbuild/linux-loong64": "0.23.1", 1660 | "@esbuild/linux-mips64el": "0.23.1", 1661 | "@esbuild/linux-ppc64": "0.23.1", 1662 | "@esbuild/linux-riscv64": "0.23.1", 1663 | "@esbuild/linux-s390x": "0.23.1", 1664 | "@esbuild/linux-x64": "0.23.1", 1665 | "@esbuild/netbsd-x64": "0.23.1", 1666 | "@esbuild/openbsd-arm64": "0.23.1", 1667 | "@esbuild/openbsd-x64": "0.23.1", 1668 | "@esbuild/sunos-x64": "0.23.1", 1669 | "@esbuild/win32-arm64": "0.23.1", 1670 | "@esbuild/win32-ia32": "0.23.1", 1671 | "@esbuild/win32-x64": "0.23.1" 1672 | } 1673 | }, 1674 | "node_modules/escape-html": { 1675 | "version": "1.0.3", 1676 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", 1677 | "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==", 1678 | "dev": true 1679 | }, 1680 | "node_modules/escape-string-regexp": { 1681 | "version": "4.0.0", 1682 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", 1683 | "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", 1684 | "dev": true, 1685 | "engines": { 1686 | "node": ">=10" 1687 | }, 1688 | "funding": { 1689 | "url": "https://github.com/sponsors/sindresorhus" 1690 | } 1691 | }, 1692 | "node_modules/eslint": { 1693 | "version": "8.57.1", 1694 | "resolved": "https://registry.npmjs.org/eslint/-/eslint-8.57.1.tgz", 1695 | "integrity": "sha512-ypowyDxpVSYpkXr9WPv2PAZCtNip1Mv5KTW0SCurXv/9iOpcrH9PaqUElksqEB6pChqHGDRCFTyrZlGhnLNGiA==", 1696 | "deprecated": "This version is no longer supported. Please see https://eslint.org/version-support for other options.", 1697 | "dev": true, 1698 | "dependencies": { 1699 | "@eslint-community/eslint-utils": "^4.2.0", 1700 | "@eslint-community/regexpp": "^4.6.1", 1701 | "@eslint/eslintrc": "^2.1.4", 1702 | "@eslint/js": "8.57.1", 1703 | "@humanwhocodes/config-array": "^0.13.0", 1704 | "@humanwhocodes/module-importer": "^1.0.1", 1705 | "@nodelib/fs.walk": "^1.2.8", 1706 | "@ungap/structured-clone": "^1.2.0", 1707 | "ajv": "^6.12.4", 1708 | "chalk": "^4.0.0", 1709 | "cross-spawn": "^7.0.2", 1710 | "debug": "^4.3.2", 1711 | "doctrine": "^3.0.0", 1712 | "escape-string-regexp": "^4.0.0", 1713 | "eslint-scope": "^7.2.2", 1714 | "eslint-visitor-keys": "^3.4.3", 1715 | "espree": "^9.6.1", 1716 | "esquery": "^1.4.2", 1717 | "esutils": "^2.0.2", 1718 | "fast-deep-equal": "^3.1.3", 1719 | "file-entry-cache": "^6.0.1", 1720 | "find-up": "^5.0.0", 1721 | "glob-parent": "^6.0.2", 1722 | "globals": "^13.19.0", 1723 | "graphemer": "^1.4.0", 1724 | "ignore": "^5.2.0", 1725 | "imurmurhash": "^0.1.4", 1726 | "is-glob": "^4.0.0", 1727 | "is-path-inside": "^3.0.3", 1728 | "js-yaml": "^4.1.0", 1729 | "json-stable-stringify-without-jsonify": "^1.0.1", 1730 | "levn": "^0.4.1", 1731 | "lodash.merge": "^4.6.2", 1732 | "minimatch": "^3.1.2", 1733 | "natural-compare": "^1.4.0", 1734 | "optionator": "^0.9.3", 1735 | "strip-ansi": "^6.0.1", 1736 | "text-table": "^0.2.0" 1737 | }, 1738 | "bin": { 1739 | "eslint": "bin/eslint.js" 1740 | }, 1741 | "engines": { 1742 | "node": "^12.22.0 || ^14.17.0 || >=16.0.0" 1743 | }, 1744 | "funding": { 1745 | "url": "https://opencollective.com/eslint" 1746 | } 1747 | }, 1748 | "node_modules/eslint-config-prettier": { 1749 | "version": "9.1.0", 1750 | "resolved": "https://registry.npmjs.org/eslint-config-prettier/-/eslint-config-prettier-9.1.0.tgz", 1751 | "integrity": "sha512-NSWl5BFQWEPi1j4TjVNItzYV7dZXZ+wP6I6ZhrBGpChQhZRUaElihE9uRRkcbRnNb76UMKDF3r+WTmNcGPKsqw==", 1752 | "dev": true, 1753 | "bin": { 1754 | "eslint-config-prettier": "bin/cli.js" 1755 | }, 1756 | "peerDependencies": { 1757 | "eslint": ">=7.0.0" 1758 | } 1759 | }, 1760 | "node_modules/eslint-plugin-prettier": { 1761 | "version": "5.2.3", 1762 | "resolved": "https://registry.npmjs.org/eslint-plugin-prettier/-/eslint-plugin-prettier-5.2.3.tgz", 1763 | "integrity": "sha512-qJ+y0FfCp/mQYQ/vWQ3s7eUlFEL4PyKfAJxsnYTJ4YT73nsJBWqmEpFryxV9OeUiqmsTsYJ5Y+KDNaeP31wrRw==", 1764 | "dev": true, 1765 | "dependencies": { 1766 | "prettier-linter-helpers": "^1.0.0", 1767 | "synckit": "^0.9.1" 1768 | }, 1769 | "engines": { 1770 | "node": "^14.18.0 || >=16.0.0" 1771 | }, 1772 | "funding": { 1773 | "url": "https://opencollective.com/eslint-plugin-prettier" 1774 | }, 1775 | "peerDependencies": { 1776 | "@types/eslint": ">=8.0.0", 1777 | "eslint": ">=8.0.0", 1778 | "eslint-config-prettier": "*", 1779 | "prettier": ">=3.0.0" 1780 | }, 1781 | "peerDependenciesMeta": { 1782 | "@types/eslint": { 1783 | "optional": true 1784 | }, 1785 | "eslint-config-prettier": { 1786 | "optional": true 1787 | } 1788 | } 1789 | }, 1790 | "node_modules/eslint-scope": { 1791 | "version": "7.2.2", 1792 | "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-7.2.2.tgz", 1793 | "integrity": "sha512-dOt21O7lTMhDM+X9mB4GX+DZrZtCUJPL/wlcTqxyrx5IvO0IYtILdtrQGQp+8n5S0gwSVmOf9NQrjMOgfQZlIg==", 1794 | "dev": true, 1795 | "dependencies": { 1796 | "esrecurse": "^4.3.0", 1797 | "estraverse": "^5.2.0" 1798 | }, 1799 | "engines": { 1800 | "node": "^12.22.0 || ^14.17.0 || >=16.0.0" 1801 | }, 1802 | "funding": { 1803 | "url": "https://opencollective.com/eslint" 1804 | } 1805 | }, 1806 | "node_modules/eslint-visitor-keys": { 1807 | "version": "3.4.3", 1808 | "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz", 1809 | "integrity": "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==", 1810 | "dev": true, 1811 | "engines": { 1812 | "node": "^12.22.0 || ^14.17.0 || >=16.0.0" 1813 | }, 1814 | "funding": { 1815 | "url": "https://opencollective.com/eslint" 1816 | } 1817 | }, 1818 | "node_modules/eslint/node_modules/brace-expansion": { 1819 | "version": "1.1.11", 1820 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 1821 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 1822 | "dev": true, 1823 | "dependencies": { 1824 | "balanced-match": "^1.0.0", 1825 | "concat-map": "0.0.1" 1826 | } 1827 | }, 1828 | "node_modules/eslint/node_modules/minimatch": { 1829 | "version": "3.1.2", 1830 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", 1831 | "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", 1832 | "dev": true, 1833 | "dependencies": { 1834 | "brace-expansion": "^1.1.7" 1835 | }, 1836 | "engines": { 1837 | "node": "*" 1838 | } 1839 | }, 1840 | "node_modules/espree": { 1841 | "version": "9.6.1", 1842 | "resolved": "https://registry.npmjs.org/espree/-/espree-9.6.1.tgz", 1843 | "integrity": "sha512-oruZaFkjorTpF32kDSI5/75ViwGeZginGGy2NoOSg3Q9bnwlnmDm4HLnkl0RE3n+njDXR037aY1+x58Z/zFdwQ==", 1844 | "dev": true, 1845 | "dependencies": { 1846 | "acorn": "^8.9.0", 1847 | "acorn-jsx": "^5.3.2", 1848 | "eslint-visitor-keys": "^3.4.1" 1849 | }, 1850 | "engines": { 1851 | "node": "^12.22.0 || ^14.17.0 || >=16.0.0" 1852 | }, 1853 | "funding": { 1854 | "url": "https://opencollective.com/eslint" 1855 | } 1856 | }, 1857 | "node_modules/esquery": { 1858 | "version": "1.6.0", 1859 | "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.6.0.tgz", 1860 | "integrity": "sha512-ca9pw9fomFcKPvFLXhBKUK90ZvGibiGOvRJNbjljY7s7uq/5YO4BOzcYtJqExdx99rF6aAcnRxHmcUHcz6sQsg==", 1861 | "dev": true, 1862 | "dependencies": { 1863 | "estraverse": "^5.1.0" 1864 | }, 1865 | "engines": { 1866 | "node": ">=0.10" 1867 | } 1868 | }, 1869 | "node_modules/esrecurse": { 1870 | "version": "4.3.0", 1871 | "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", 1872 | "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", 1873 | "dev": true, 1874 | "dependencies": { 1875 | "estraverse": "^5.2.0" 1876 | }, 1877 | "engines": { 1878 | "node": ">=4.0" 1879 | } 1880 | }, 1881 | "node_modules/estraverse": { 1882 | "version": "5.3.0", 1883 | "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", 1884 | "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", 1885 | "dev": true, 1886 | "engines": { 1887 | "node": ">=4.0" 1888 | } 1889 | }, 1890 | "node_modules/esutils": { 1891 | "version": "2.0.3", 1892 | "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", 1893 | "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", 1894 | "dev": true, 1895 | "engines": { 1896 | "node": ">=0.10.0" 1897 | } 1898 | }, 1899 | "node_modules/etag": { 1900 | "version": "1.8.1", 1901 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", 1902 | "integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==", 1903 | "dev": true, 1904 | "engines": { 1905 | "node": ">= 0.6" 1906 | } 1907 | }, 1908 | "node_modules/eventsource": { 1909 | "version": "3.0.5", 1910 | "resolved": "https://registry.npmjs.org/eventsource/-/eventsource-3.0.5.tgz", 1911 | "integrity": "sha512-LT/5J605bx5SNyE+ITBDiM3FxffBiq9un7Vx0EwMDM3vg8sWKx/tO2zC+LMqZ+smAM0F2hblaDZUVZF0te2pSw==", 1912 | "dependencies": { 1913 | "eventsource-parser": "^3.0.0" 1914 | }, 1915 | "engines": { 1916 | "node": ">=18.0.0" 1917 | } 1918 | }, 1919 | "node_modules/eventsource-parser": { 1920 | "version": "3.0.0", 1921 | "resolved": "https://registry.npmjs.org/eventsource-parser/-/eventsource-parser-3.0.0.tgz", 1922 | "integrity": "sha512-T1C0XCUimhxVQzW4zFipdx0SficT651NnkR0ZSH3yQwh+mFMdLfgjABVi4YtMTtaL4s168593DaoaRLMqryavA==", 1923 | "engines": { 1924 | "node": ">=18.0.0" 1925 | } 1926 | }, 1927 | "node_modules/express": { 1928 | "version": "4.21.2", 1929 | "resolved": "https://registry.npmjs.org/express/-/express-4.21.2.tgz", 1930 | "integrity": "sha512-28HqgMZAmih1Czt9ny7qr6ek2qddF4FclbMzwhCREB6OFfH+rXAnuNCwo1/wFvrtbgsQDb4kSbX9de9lFbrXnA==", 1931 | "dev": true, 1932 | "dependencies": { 1933 | "accepts": "~1.3.8", 1934 | "array-flatten": "1.1.1", 1935 | "body-parser": "1.20.3", 1936 | "content-disposition": "0.5.4", 1937 | "content-type": "~1.0.4", 1938 | "cookie": "0.7.1", 1939 | "cookie-signature": "1.0.6", 1940 | "debug": "2.6.9", 1941 | "depd": "2.0.0", 1942 | "encodeurl": "~2.0.0", 1943 | "escape-html": "~1.0.3", 1944 | "etag": "~1.8.1", 1945 | "finalhandler": "1.3.1", 1946 | "fresh": "0.5.2", 1947 | "http-errors": "2.0.0", 1948 | "merge-descriptors": "1.0.3", 1949 | "methods": "~1.1.2", 1950 | "on-finished": "2.4.1", 1951 | "parseurl": "~1.3.3", 1952 | "path-to-regexp": "0.1.12", 1953 | "proxy-addr": "~2.0.7", 1954 | "qs": "6.13.0", 1955 | "range-parser": "~1.2.1", 1956 | "safe-buffer": "5.2.1", 1957 | "send": "0.19.0", 1958 | "serve-static": "1.16.2", 1959 | "setprototypeof": "1.2.0", 1960 | "statuses": "2.0.1", 1961 | "type-is": "~1.6.18", 1962 | "utils-merge": "1.0.1", 1963 | "vary": "~1.1.2" 1964 | }, 1965 | "engines": { 1966 | "node": ">= 0.10.0" 1967 | }, 1968 | "funding": { 1969 | "type": "opencollective", 1970 | "url": "https://opencollective.com/express" 1971 | } 1972 | }, 1973 | "node_modules/express/node_modules/debug": { 1974 | "version": "2.6.9", 1975 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 1976 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 1977 | "dev": true, 1978 | "dependencies": { 1979 | "ms": "2.0.0" 1980 | } 1981 | }, 1982 | "node_modules/express/node_modules/ms": { 1983 | "version": "2.0.0", 1984 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 1985 | "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==", 1986 | "dev": true 1987 | }, 1988 | "node_modules/fast-deep-equal": { 1989 | "version": "3.1.3", 1990 | "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", 1991 | "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", 1992 | "dev": true 1993 | }, 1994 | "node_modules/fast-diff": { 1995 | "version": "1.3.0", 1996 | "resolved": "https://registry.npmjs.org/fast-diff/-/fast-diff-1.3.0.tgz", 1997 | "integrity": "sha512-VxPP4NqbUjj6MaAOafWeUn2cXWLcCtljklUtZf0Ind4XQ+QPtmA0b18zZy0jIQx+ExRVCR/ZQpBmik5lXshNsw==", 1998 | "dev": true 1999 | }, 2000 | "node_modules/fast-glob": { 2001 | "version": "3.3.3", 2002 | "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.3.tgz", 2003 | "integrity": "sha512-7MptL8U0cqcFdzIzwOTHoilX9x5BrNqye7Z/LuC7kCMRio1EMSyqRK3BEAUD7sXRq4iT4AzTVuZdhgQ2TCvYLg==", 2004 | "dev": true, 2005 | "dependencies": { 2006 | "@nodelib/fs.stat": "^2.0.2", 2007 | "@nodelib/fs.walk": "^1.2.3", 2008 | "glob-parent": "^5.1.2", 2009 | "merge2": "^1.3.0", 2010 | "micromatch": "^4.0.8" 2011 | }, 2012 | "engines": { 2013 | "node": ">=8.6.0" 2014 | } 2015 | }, 2016 | "node_modules/fast-glob/node_modules/glob-parent": { 2017 | "version": "5.1.2", 2018 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", 2019 | "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", 2020 | "dev": true, 2021 | "dependencies": { 2022 | "is-glob": "^4.0.1" 2023 | }, 2024 | "engines": { 2025 | "node": ">= 6" 2026 | } 2027 | }, 2028 | "node_modules/fast-json-stable-stringify": { 2029 | "version": "2.1.0", 2030 | "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", 2031 | "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", 2032 | "dev": true 2033 | }, 2034 | "node_modules/fast-levenshtein": { 2035 | "version": "2.0.6", 2036 | "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", 2037 | "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==", 2038 | "dev": true 2039 | }, 2040 | "node_modules/fastq": { 2041 | "version": "1.19.0", 2042 | "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.19.0.tgz", 2043 | "integrity": "sha512-7SFSRCNjBQIZH/xZR3iy5iQYR8aGBE0h3VG6/cwlbrpdciNYBMotQav8c1XI3HjHH+NikUpP53nPdlZSdWmFzA==", 2044 | "dev": true, 2045 | "dependencies": { 2046 | "reusify": "^1.0.4" 2047 | } 2048 | }, 2049 | "node_modules/file-entry-cache": { 2050 | "version": "6.0.1", 2051 | "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-6.0.1.tgz", 2052 | "integrity": "sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg==", 2053 | "dev": true, 2054 | "dependencies": { 2055 | "flat-cache": "^3.0.4" 2056 | }, 2057 | "engines": { 2058 | "node": "^10.12.0 || >=12.0.0" 2059 | } 2060 | }, 2061 | "node_modules/fill-range": { 2062 | "version": "7.1.1", 2063 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz", 2064 | "integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==", 2065 | "dev": true, 2066 | "dependencies": { 2067 | "to-regex-range": "^5.0.1" 2068 | }, 2069 | "engines": { 2070 | "node": ">=8" 2071 | } 2072 | }, 2073 | "node_modules/finalhandler": { 2074 | "version": "1.3.1", 2075 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.3.1.tgz", 2076 | "integrity": "sha512-6BN9trH7bp3qvnrRyzsBz+g3lZxTNZTbVO2EV1CS0WIcDbawYVdYvGflME/9QP0h0pYlCDBCTjYa9nZzMDpyxQ==", 2077 | "dev": true, 2078 | "dependencies": { 2079 | "debug": "2.6.9", 2080 | "encodeurl": "~2.0.0", 2081 | "escape-html": "~1.0.3", 2082 | "on-finished": "2.4.1", 2083 | "parseurl": "~1.3.3", 2084 | "statuses": "2.0.1", 2085 | "unpipe": "~1.0.0" 2086 | }, 2087 | "engines": { 2088 | "node": ">= 0.8" 2089 | } 2090 | }, 2091 | "node_modules/finalhandler/node_modules/debug": { 2092 | "version": "2.6.9", 2093 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 2094 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 2095 | "dev": true, 2096 | "dependencies": { 2097 | "ms": "2.0.0" 2098 | } 2099 | }, 2100 | "node_modules/finalhandler/node_modules/ms": { 2101 | "version": "2.0.0", 2102 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 2103 | "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==", 2104 | "dev": true 2105 | }, 2106 | "node_modules/find-up": { 2107 | "version": "5.0.0", 2108 | "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", 2109 | "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", 2110 | "dev": true, 2111 | "dependencies": { 2112 | "locate-path": "^6.0.0", 2113 | "path-exists": "^4.0.0" 2114 | }, 2115 | "engines": { 2116 | "node": ">=10" 2117 | }, 2118 | "funding": { 2119 | "url": "https://github.com/sponsors/sindresorhus" 2120 | } 2121 | }, 2122 | "node_modules/flat-cache": { 2123 | "version": "3.2.0", 2124 | "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.2.0.tgz", 2125 | "integrity": "sha512-CYcENa+FtcUKLmhhqyctpclsq7QF38pKjZHsGNiSQF5r4FtoKDWabFDl3hzaEQMvT1LHEysw5twgLvpYYb4vbw==", 2126 | "dev": true, 2127 | "dependencies": { 2128 | "flatted": "^3.2.9", 2129 | "keyv": "^4.5.3", 2130 | "rimraf": "^3.0.2" 2131 | }, 2132 | "engines": { 2133 | "node": "^10.12.0 || >=12.0.0" 2134 | } 2135 | }, 2136 | "node_modules/flatted": { 2137 | "version": "3.3.2", 2138 | "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.3.2.tgz", 2139 | "integrity": "sha512-AiwGJM8YcNOaobumgtng+6NHuOqC3A7MixFeDafM3X9cIUM+xUXoS5Vfgf+OihAYe20fxqNM9yPBXJzRtZ/4eA==", 2140 | "dev": true 2141 | }, 2142 | "node_modules/follow-redirects": { 2143 | "version": "1.15.9", 2144 | "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.9.tgz", 2145 | "integrity": "sha512-gew4GsXizNgdoRyqmyfMHyAmXsZDk6mHkSxZFCzW9gwlbtOW44CDtYavM+y+72qD/Vq2l550kMF52DT8fOLJqQ==", 2146 | "funding": [ 2147 | { 2148 | "type": "individual", 2149 | "url": "https://github.com/sponsors/RubenVerborgh" 2150 | } 2151 | ], 2152 | "engines": { 2153 | "node": ">=4.0" 2154 | }, 2155 | "peerDependenciesMeta": { 2156 | "debug": { 2157 | "optional": true 2158 | } 2159 | } 2160 | }, 2161 | "node_modules/form-data": { 2162 | "version": "4.0.1", 2163 | "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.1.tgz", 2164 | "integrity": "sha512-tzN8e4TX8+kkxGPK8D5u0FNmjPUjw3lwC9lSLxxoB/+GtsJG91CO8bSWy73APlgAZzZbXEYZJuxjkHH2w+Ezhw==", 2165 | "dependencies": { 2166 | "asynckit": "^0.4.0", 2167 | "combined-stream": "^1.0.8", 2168 | "mime-types": "^2.1.12" 2169 | }, 2170 | "engines": { 2171 | "node": ">= 6" 2172 | } 2173 | }, 2174 | "node_modules/forwarded": { 2175 | "version": "0.2.0", 2176 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz", 2177 | "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==", 2178 | "dev": true, 2179 | "engines": { 2180 | "node": ">= 0.6" 2181 | } 2182 | }, 2183 | "node_modules/fresh": { 2184 | "version": "0.5.2", 2185 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", 2186 | "integrity": "sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q==", 2187 | "dev": true, 2188 | "engines": { 2189 | "node": ">= 0.6" 2190 | } 2191 | }, 2192 | "node_modules/fs.realpath": { 2193 | "version": "1.0.0", 2194 | "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", 2195 | "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==", 2196 | "dev": true 2197 | }, 2198 | "node_modules/fsevents": { 2199 | "version": "2.3.3", 2200 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz", 2201 | "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==", 2202 | "dev": true, 2203 | "hasInstallScript": true, 2204 | "optional": true, 2205 | "os": [ 2206 | "darwin" 2207 | ], 2208 | "engines": { 2209 | "node": "^8.16.0 || ^10.6.0 || >=11.0.0" 2210 | } 2211 | }, 2212 | "node_modules/function-bind": { 2213 | "version": "1.1.2", 2214 | "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", 2215 | "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", 2216 | "dev": true, 2217 | "funding": { 2218 | "url": "https://github.com/sponsors/ljharb" 2219 | } 2220 | }, 2221 | "node_modules/get-intrinsic": { 2222 | "version": "1.2.7", 2223 | "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.2.7.tgz", 2224 | "integrity": "sha512-VW6Pxhsrk0KAOqs3WEd0klDiF/+V7gQOpAvY1jVU/LHmaD/kQO4523aiJuikX/QAKYiW6x8Jh+RJej1almdtCA==", 2225 | "dev": true, 2226 | "dependencies": { 2227 | "call-bind-apply-helpers": "^1.0.1", 2228 | "es-define-property": "^1.0.1", 2229 | "es-errors": "^1.3.0", 2230 | "es-object-atoms": "^1.0.0", 2231 | "function-bind": "^1.1.2", 2232 | "get-proto": "^1.0.0", 2233 | "gopd": "^1.2.0", 2234 | "has-symbols": "^1.1.0", 2235 | "hasown": "^2.0.2", 2236 | "math-intrinsics": "^1.1.0" 2237 | }, 2238 | "engines": { 2239 | "node": ">= 0.4" 2240 | }, 2241 | "funding": { 2242 | "url": "https://github.com/sponsors/ljharb" 2243 | } 2244 | }, 2245 | "node_modules/get-proto": { 2246 | "version": "1.0.1", 2247 | "resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz", 2248 | "integrity": "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==", 2249 | "dev": true, 2250 | "dependencies": { 2251 | "dunder-proto": "^1.0.1", 2252 | "es-object-atoms": "^1.0.0" 2253 | }, 2254 | "engines": { 2255 | "node": ">= 0.4" 2256 | } 2257 | }, 2258 | "node_modules/get-tsconfig": { 2259 | "version": "4.10.0", 2260 | "resolved": "https://registry.npmjs.org/get-tsconfig/-/get-tsconfig-4.10.0.tgz", 2261 | "integrity": "sha512-kGzZ3LWWQcGIAmg6iWvXn0ei6WDtV26wzHRMwDSzmAbcXrTEXxHy6IehI6/4eT6VRKyMP1eF1VqwrVUmE/LR7A==", 2262 | "dev": true, 2263 | "dependencies": { 2264 | "resolve-pkg-maps": "^1.0.0" 2265 | }, 2266 | "funding": { 2267 | "url": "https://github.com/privatenumber/get-tsconfig?sponsor=1" 2268 | } 2269 | }, 2270 | "node_modules/glob": { 2271 | "version": "7.2.3", 2272 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", 2273 | "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", 2274 | "deprecated": "Glob versions prior to v9 are no longer supported", 2275 | "dev": true, 2276 | "dependencies": { 2277 | "fs.realpath": "^1.0.0", 2278 | "inflight": "^1.0.4", 2279 | "inherits": "2", 2280 | "minimatch": "^3.1.1", 2281 | "once": "^1.3.0", 2282 | "path-is-absolute": "^1.0.0" 2283 | }, 2284 | "engines": { 2285 | "node": "*" 2286 | }, 2287 | "funding": { 2288 | "url": "https://github.com/sponsors/isaacs" 2289 | } 2290 | }, 2291 | "node_modules/glob-parent": { 2292 | "version": "6.0.2", 2293 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", 2294 | "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", 2295 | "dev": true, 2296 | "dependencies": { 2297 | "is-glob": "^4.0.3" 2298 | }, 2299 | "engines": { 2300 | "node": ">=10.13.0" 2301 | } 2302 | }, 2303 | "node_modules/glob/node_modules/brace-expansion": { 2304 | "version": "1.1.11", 2305 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 2306 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 2307 | "dev": true, 2308 | "dependencies": { 2309 | "balanced-match": "^1.0.0", 2310 | "concat-map": "0.0.1" 2311 | } 2312 | }, 2313 | "node_modules/glob/node_modules/minimatch": { 2314 | "version": "3.1.2", 2315 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", 2316 | "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", 2317 | "dev": true, 2318 | "dependencies": { 2319 | "brace-expansion": "^1.1.7" 2320 | }, 2321 | "engines": { 2322 | "node": "*" 2323 | } 2324 | }, 2325 | "node_modules/globals": { 2326 | "version": "13.24.0", 2327 | "resolved": "https://registry.npmjs.org/globals/-/globals-13.24.0.tgz", 2328 | "integrity": "sha512-AhO5QUcj8llrbG09iWhPU2B204J1xnPeL8kQmVorSsy+Sjj1sk8gIyh6cUocGmH4L0UuhAJy+hJMRA4mgA4mFQ==", 2329 | "dev": true, 2330 | "dependencies": { 2331 | "type-fest": "^0.20.2" 2332 | }, 2333 | "engines": { 2334 | "node": ">=8" 2335 | }, 2336 | "funding": { 2337 | "url": "https://github.com/sponsors/sindresorhus" 2338 | } 2339 | }, 2340 | "node_modules/globby": { 2341 | "version": "11.1.0", 2342 | "resolved": "https://registry.npmjs.org/globby/-/globby-11.1.0.tgz", 2343 | "integrity": "sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g==", 2344 | "dev": true, 2345 | "dependencies": { 2346 | "array-union": "^2.1.0", 2347 | "dir-glob": "^3.0.1", 2348 | "fast-glob": "^3.2.9", 2349 | "ignore": "^5.2.0", 2350 | "merge2": "^1.4.1", 2351 | "slash": "^3.0.0" 2352 | }, 2353 | "engines": { 2354 | "node": ">=10" 2355 | }, 2356 | "funding": { 2357 | "url": "https://github.com/sponsors/sindresorhus" 2358 | } 2359 | }, 2360 | "node_modules/gopd": { 2361 | "version": "1.2.0", 2362 | "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz", 2363 | "integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==", 2364 | "dev": true, 2365 | "engines": { 2366 | "node": ">= 0.4" 2367 | }, 2368 | "funding": { 2369 | "url": "https://github.com/sponsors/ljharb" 2370 | } 2371 | }, 2372 | "node_modules/graphemer": { 2373 | "version": "1.4.0", 2374 | "resolved": "https://registry.npmjs.org/graphemer/-/graphemer-1.4.0.tgz", 2375 | "integrity": "sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==", 2376 | "dev": true 2377 | }, 2378 | "node_modules/has-flag": { 2379 | "version": "4.0.0", 2380 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", 2381 | "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", 2382 | "dev": true, 2383 | "engines": { 2384 | "node": ">=8" 2385 | } 2386 | }, 2387 | "node_modules/has-symbols": { 2388 | "version": "1.1.0", 2389 | "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz", 2390 | "integrity": "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==", 2391 | "dev": true, 2392 | "engines": { 2393 | "node": ">= 0.4" 2394 | }, 2395 | "funding": { 2396 | "url": "https://github.com/sponsors/ljharb" 2397 | } 2398 | }, 2399 | "node_modules/hasown": { 2400 | "version": "2.0.2", 2401 | "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", 2402 | "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", 2403 | "dev": true, 2404 | "dependencies": { 2405 | "function-bind": "^1.1.2" 2406 | }, 2407 | "engines": { 2408 | "node": ">= 0.4" 2409 | } 2410 | }, 2411 | "node_modules/http-errors": { 2412 | "version": "2.0.0", 2413 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.0.tgz", 2414 | "integrity": "sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ==", 2415 | "dependencies": { 2416 | "depd": "2.0.0", 2417 | "inherits": "2.0.4", 2418 | "setprototypeof": "1.2.0", 2419 | "statuses": "2.0.1", 2420 | "toidentifier": "1.0.1" 2421 | }, 2422 | "engines": { 2423 | "node": ">= 0.8" 2424 | } 2425 | }, 2426 | "node_modules/iconv-lite": { 2427 | "version": "0.4.24", 2428 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", 2429 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", 2430 | "dev": true, 2431 | "dependencies": { 2432 | "safer-buffer": ">= 2.1.2 < 3" 2433 | }, 2434 | "engines": { 2435 | "node": ">=0.10.0" 2436 | } 2437 | }, 2438 | "node_modules/ignore": { 2439 | "version": "5.3.2", 2440 | "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz", 2441 | "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==", 2442 | "dev": true, 2443 | "engines": { 2444 | "node": ">= 4" 2445 | } 2446 | }, 2447 | "node_modules/import-fresh": { 2448 | "version": "3.3.1", 2449 | "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.1.tgz", 2450 | "integrity": "sha512-TR3KfrTZTYLPB6jUjfx6MF9WcWrHL9su5TObK4ZkYgBdWKPOFoSoQIdEuTuR82pmtxH2spWG9h6etwfr1pLBqQ==", 2451 | "dev": true, 2452 | "dependencies": { 2453 | "parent-module": "^1.0.0", 2454 | "resolve-from": "^4.0.0" 2455 | }, 2456 | "engines": { 2457 | "node": ">=6" 2458 | }, 2459 | "funding": { 2460 | "url": "https://github.com/sponsors/sindresorhus" 2461 | } 2462 | }, 2463 | "node_modules/imurmurhash": { 2464 | "version": "0.1.4", 2465 | "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", 2466 | "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", 2467 | "dev": true, 2468 | "engines": { 2469 | "node": ">=0.8.19" 2470 | } 2471 | }, 2472 | "node_modules/inflight": { 2473 | "version": "1.0.6", 2474 | "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", 2475 | "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==", 2476 | "deprecated": "This module is not supported, and leaks memory. Do not use it. Check out lru-cache if you want a good and tested way to coalesce async requests by a key value, which is much more comprehensive and powerful.", 2477 | "dev": true, 2478 | "dependencies": { 2479 | "once": "^1.3.0", 2480 | "wrappy": "1" 2481 | } 2482 | }, 2483 | "node_modules/inherits": { 2484 | "version": "2.0.4", 2485 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", 2486 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" 2487 | }, 2488 | "node_modules/ipaddr.js": { 2489 | "version": "1.9.1", 2490 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", 2491 | "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==", 2492 | "dev": true, 2493 | "engines": { 2494 | "node": ">= 0.10" 2495 | } 2496 | }, 2497 | "node_modules/is-extglob": { 2498 | "version": "2.1.1", 2499 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", 2500 | "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", 2501 | "dev": true, 2502 | "engines": { 2503 | "node": ">=0.10.0" 2504 | } 2505 | }, 2506 | "node_modules/is-glob": { 2507 | "version": "4.0.3", 2508 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", 2509 | "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", 2510 | "dev": true, 2511 | "dependencies": { 2512 | "is-extglob": "^2.1.1" 2513 | }, 2514 | "engines": { 2515 | "node": ">=0.10.0" 2516 | } 2517 | }, 2518 | "node_modules/is-number": { 2519 | "version": "7.0.0", 2520 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", 2521 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", 2522 | "dev": true, 2523 | "engines": { 2524 | "node": ">=0.12.0" 2525 | } 2526 | }, 2527 | "node_modules/is-path-inside": { 2528 | "version": "3.0.3", 2529 | "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz", 2530 | "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==", 2531 | "dev": true, 2532 | "engines": { 2533 | "node": ">=8" 2534 | } 2535 | }, 2536 | "node_modules/isexe": { 2537 | "version": "2.0.0", 2538 | "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", 2539 | "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", 2540 | "dev": true 2541 | }, 2542 | "node_modules/js-yaml": { 2543 | "version": "4.1.0", 2544 | "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", 2545 | "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", 2546 | "dev": true, 2547 | "dependencies": { 2548 | "argparse": "^2.0.1" 2549 | }, 2550 | "bin": { 2551 | "js-yaml": "bin/js-yaml.js" 2552 | } 2553 | }, 2554 | "node_modules/json-buffer": { 2555 | "version": "3.0.1", 2556 | "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.1.tgz", 2557 | "integrity": "sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==", 2558 | "dev": true 2559 | }, 2560 | "node_modules/json-schema": { 2561 | "version": "0.4.0", 2562 | "resolved": "https://registry.npmjs.org/json-schema/-/json-schema-0.4.0.tgz", 2563 | "integrity": "sha512-es94M3nTIfsEPisRafak+HDLfHXnKBhV3vU5eqPcS3flIWqcxJWgXHXiey3YrpaNsanY5ei1VoYEbOzijuq9BA==" 2564 | }, 2565 | "node_modules/json-schema-traverse": { 2566 | "version": "0.4.1", 2567 | "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", 2568 | "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", 2569 | "dev": true 2570 | }, 2571 | "node_modules/json-stable-stringify-without-jsonify": { 2572 | "version": "1.0.1", 2573 | "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", 2574 | "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==", 2575 | "dev": true 2576 | }, 2577 | "node_modules/jsondiffpatch": { 2578 | "version": "0.6.0", 2579 | "resolved": "https://registry.npmjs.org/jsondiffpatch/-/jsondiffpatch-0.6.0.tgz", 2580 | "integrity": "sha512-3QItJOXp2AP1uv7waBkao5nCvhEv+QmJAd38Ybq7wNI74Q+BBmnLn4EDKz6yI9xGAIQoUF87qHt+kc1IVxB4zQ==", 2581 | "dependencies": { 2582 | "@types/diff-match-patch": "^1.0.36", 2583 | "chalk": "^5.3.0", 2584 | "diff-match-patch": "^1.0.5" 2585 | }, 2586 | "bin": { 2587 | "jsondiffpatch": "bin/jsondiffpatch.js" 2588 | }, 2589 | "engines": { 2590 | "node": "^18.0.0 || >=20.0.0" 2591 | } 2592 | }, 2593 | "node_modules/jsondiffpatch/node_modules/chalk": { 2594 | "version": "5.4.1", 2595 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-5.4.1.tgz", 2596 | "integrity": "sha512-zgVZuo2WcZgfUEmsn6eO3kINexW8RAE4maiQ8QNs8CtpPCSyMiYsULR3HQYkm3w8FIA3SberyMJMSldGsW+U3w==", 2597 | "engines": { 2598 | "node": "^12.17.0 || ^14.13 || >=16.0.0" 2599 | }, 2600 | "funding": { 2601 | "url": "https://github.com/chalk/chalk?sponsor=1" 2602 | } 2603 | }, 2604 | "node_modules/keyv": { 2605 | "version": "4.5.4", 2606 | "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz", 2607 | "integrity": "sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==", 2608 | "dev": true, 2609 | "dependencies": { 2610 | "json-buffer": "3.0.1" 2611 | } 2612 | }, 2613 | "node_modules/levn": { 2614 | "version": "0.4.1", 2615 | "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", 2616 | "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", 2617 | "dev": true, 2618 | "dependencies": { 2619 | "prelude-ls": "^1.2.1", 2620 | "type-check": "~0.4.0" 2621 | }, 2622 | "engines": { 2623 | "node": ">= 0.8.0" 2624 | } 2625 | }, 2626 | "node_modules/locate-path": { 2627 | "version": "6.0.0", 2628 | "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", 2629 | "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", 2630 | "dev": true, 2631 | "dependencies": { 2632 | "p-locate": "^5.0.0" 2633 | }, 2634 | "engines": { 2635 | "node": ">=10" 2636 | }, 2637 | "funding": { 2638 | "url": "https://github.com/sponsors/sindresorhus" 2639 | } 2640 | }, 2641 | "node_modules/lodash.merge": { 2642 | "version": "4.6.2", 2643 | "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", 2644 | "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", 2645 | "dev": true 2646 | }, 2647 | "node_modules/math-intrinsics": { 2648 | "version": "1.1.0", 2649 | "resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz", 2650 | "integrity": "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==", 2651 | "dev": true, 2652 | "engines": { 2653 | "node": ">= 0.4" 2654 | } 2655 | }, 2656 | "node_modules/media-typer": { 2657 | "version": "0.3.0", 2658 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", 2659 | "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==", 2660 | "dev": true, 2661 | "engines": { 2662 | "node": ">= 0.6" 2663 | } 2664 | }, 2665 | "node_modules/merge-descriptors": { 2666 | "version": "1.0.3", 2667 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.3.tgz", 2668 | "integrity": "sha512-gaNvAS7TZ897/rVaZ0nMtAyxNyi/pdbjbAwUpFQpN70GqnVfOiXpeUUMKRBmzXaSQ8DdTX4/0ms62r2K+hE6mQ==", 2669 | "dev": true, 2670 | "funding": { 2671 | "url": "https://github.com/sponsors/sindresorhus" 2672 | } 2673 | }, 2674 | "node_modules/merge2": { 2675 | "version": "1.4.1", 2676 | "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz", 2677 | "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==", 2678 | "dev": true, 2679 | "engines": { 2680 | "node": ">= 8" 2681 | } 2682 | }, 2683 | "node_modules/methods": { 2684 | "version": "1.1.2", 2685 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", 2686 | "integrity": "sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==", 2687 | "dev": true, 2688 | "engines": { 2689 | "node": ">= 0.6" 2690 | } 2691 | }, 2692 | "node_modules/micromatch": { 2693 | "version": "4.0.8", 2694 | "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.8.tgz", 2695 | "integrity": "sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==", 2696 | "dev": true, 2697 | "dependencies": { 2698 | "braces": "^3.0.3", 2699 | "picomatch": "^2.3.1" 2700 | }, 2701 | "engines": { 2702 | "node": ">=8.6" 2703 | } 2704 | }, 2705 | "node_modules/mime": { 2706 | "version": "1.6.0", 2707 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", 2708 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==", 2709 | "dev": true, 2710 | "bin": { 2711 | "mime": "cli.js" 2712 | }, 2713 | "engines": { 2714 | "node": ">=4" 2715 | } 2716 | }, 2717 | "node_modules/mime-db": { 2718 | "version": "1.52.0", 2719 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", 2720 | "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", 2721 | "engines": { 2722 | "node": ">= 0.6" 2723 | } 2724 | }, 2725 | "node_modules/mime-types": { 2726 | "version": "2.1.35", 2727 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", 2728 | "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", 2729 | "dependencies": { 2730 | "mime-db": "1.52.0" 2731 | }, 2732 | "engines": { 2733 | "node": ">= 0.6" 2734 | } 2735 | }, 2736 | "node_modules/minimatch": { 2737 | "version": "9.0.3", 2738 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.3.tgz", 2739 | "integrity": "sha512-RHiac9mvaRw0x3AYRgDC1CxAP7HTcNrrECeA8YYJeWnpo+2Q5CegtZjaotWTWxDG3UeGA1coE05iH1mPjT/2mg==", 2740 | "dev": true, 2741 | "dependencies": { 2742 | "brace-expansion": "^2.0.1" 2743 | }, 2744 | "engines": { 2745 | "node": ">=16 || 14 >=14.17" 2746 | }, 2747 | "funding": { 2748 | "url": "https://github.com/sponsors/isaacs" 2749 | } 2750 | }, 2751 | "node_modules/ms": { 2752 | "version": "2.1.3", 2753 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 2754 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", 2755 | "dev": true 2756 | }, 2757 | "node_modules/nanoid": { 2758 | "version": "3.3.8", 2759 | "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.8.tgz", 2760 | "integrity": "sha512-WNLf5Sd8oZxOm+TzppcYk8gVOgP+l58xNy58D0nbUnOxOWRWvlcCV4kUF7ltmI6PsrLl/BgKEyS4mqsGChFN0w==", 2761 | "funding": [ 2762 | { 2763 | "type": "github", 2764 | "url": "https://github.com/sponsors/ai" 2765 | } 2766 | ], 2767 | "bin": { 2768 | "nanoid": "bin/nanoid.cjs" 2769 | }, 2770 | "engines": { 2771 | "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1" 2772 | } 2773 | }, 2774 | "node_modules/natural-compare": { 2775 | "version": "1.4.0", 2776 | "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", 2777 | "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", 2778 | "dev": true 2779 | }, 2780 | "node_modules/negotiator": { 2781 | "version": "0.6.3", 2782 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.3.tgz", 2783 | "integrity": "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==", 2784 | "dev": true, 2785 | "engines": { 2786 | "node": ">= 0.6" 2787 | } 2788 | }, 2789 | "node_modules/object-assign": { 2790 | "version": "4.1.1", 2791 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", 2792 | "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==", 2793 | "dev": true, 2794 | "engines": { 2795 | "node": ">=0.10.0" 2796 | } 2797 | }, 2798 | "node_modules/object-inspect": { 2799 | "version": "1.13.3", 2800 | "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.3.tgz", 2801 | "integrity": "sha512-kDCGIbxkDSXE3euJZZXzc6to7fCrKHNI/hSRQnRuQ+BWjFNzZwiFF8fj/6o2t2G9/jTj8PSIYTfCLelLZEeRpA==", 2802 | "dev": true, 2803 | "engines": { 2804 | "node": ">= 0.4" 2805 | }, 2806 | "funding": { 2807 | "url": "https://github.com/sponsors/ljharb" 2808 | } 2809 | }, 2810 | "node_modules/on-finished": { 2811 | "version": "2.4.1", 2812 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz", 2813 | "integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==", 2814 | "dev": true, 2815 | "dependencies": { 2816 | "ee-first": "1.1.1" 2817 | }, 2818 | "engines": { 2819 | "node": ">= 0.8" 2820 | } 2821 | }, 2822 | "node_modules/once": { 2823 | "version": "1.4.0", 2824 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", 2825 | "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", 2826 | "dev": true, 2827 | "dependencies": { 2828 | "wrappy": "1" 2829 | } 2830 | }, 2831 | "node_modules/optionator": { 2832 | "version": "0.9.4", 2833 | "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.4.tgz", 2834 | "integrity": "sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==", 2835 | "dev": true, 2836 | "dependencies": { 2837 | "deep-is": "^0.1.3", 2838 | "fast-levenshtein": "^2.0.6", 2839 | "levn": "^0.4.1", 2840 | "prelude-ls": "^1.2.1", 2841 | "type-check": "^0.4.0", 2842 | "word-wrap": "^1.2.5" 2843 | }, 2844 | "engines": { 2845 | "node": ">= 0.8.0" 2846 | } 2847 | }, 2848 | "node_modules/p-limit": { 2849 | "version": "3.1.0", 2850 | "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", 2851 | "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", 2852 | "dev": true, 2853 | "dependencies": { 2854 | "yocto-queue": "^0.1.0" 2855 | }, 2856 | "engines": { 2857 | "node": ">=10" 2858 | }, 2859 | "funding": { 2860 | "url": "https://github.com/sponsors/sindresorhus" 2861 | } 2862 | }, 2863 | "node_modules/p-locate": { 2864 | "version": "5.0.0", 2865 | "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", 2866 | "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", 2867 | "dev": true, 2868 | "dependencies": { 2869 | "p-limit": "^3.0.2" 2870 | }, 2871 | "engines": { 2872 | "node": ">=10" 2873 | }, 2874 | "funding": { 2875 | "url": "https://github.com/sponsors/sindresorhus" 2876 | } 2877 | }, 2878 | "node_modules/parent-module": { 2879 | "version": "1.0.1", 2880 | "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", 2881 | "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", 2882 | "dev": true, 2883 | "dependencies": { 2884 | "callsites": "^3.0.0" 2885 | }, 2886 | "engines": { 2887 | "node": ">=6" 2888 | } 2889 | }, 2890 | "node_modules/parseurl": { 2891 | "version": "1.3.3", 2892 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", 2893 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==", 2894 | "dev": true, 2895 | "engines": { 2896 | "node": ">= 0.8" 2897 | } 2898 | }, 2899 | "node_modules/path-exists": { 2900 | "version": "4.0.0", 2901 | "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", 2902 | "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", 2903 | "dev": true, 2904 | "engines": { 2905 | "node": ">=8" 2906 | } 2907 | }, 2908 | "node_modules/path-is-absolute": { 2909 | "version": "1.0.1", 2910 | "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", 2911 | "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==", 2912 | "dev": true, 2913 | "engines": { 2914 | "node": ">=0.10.0" 2915 | } 2916 | }, 2917 | "node_modules/path-key": { 2918 | "version": "3.1.1", 2919 | "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", 2920 | "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", 2921 | "dev": true, 2922 | "engines": { 2923 | "node": ">=8" 2924 | } 2925 | }, 2926 | "node_modules/path-to-regexp": { 2927 | "version": "0.1.12", 2928 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.12.tgz", 2929 | "integrity": "sha512-RA1GjUVMnvYFxuqovrEqZoxxW5NUZqbwKtYz/Tt7nXerk0LbLblQmrsgdeOxV5SFHf0UDggjS/bSeOZwt1pmEQ==", 2930 | "dev": true 2931 | }, 2932 | "node_modules/path-type": { 2933 | "version": "4.0.0", 2934 | "resolved": "https://registry.npmjs.org/path-type/-/path-type-4.0.0.tgz", 2935 | "integrity": "sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==", 2936 | "dev": true, 2937 | "engines": { 2938 | "node": ">=8" 2939 | } 2940 | }, 2941 | "node_modules/picomatch": { 2942 | "version": "2.3.1", 2943 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", 2944 | "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", 2945 | "dev": true, 2946 | "engines": { 2947 | "node": ">=8.6" 2948 | }, 2949 | "funding": { 2950 | "url": "https://github.com/sponsors/jonschlinkert" 2951 | } 2952 | }, 2953 | "node_modules/prelude-ls": { 2954 | "version": "1.2.1", 2955 | "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", 2956 | "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", 2957 | "dev": true, 2958 | "engines": { 2959 | "node": ">= 0.8.0" 2960 | } 2961 | }, 2962 | "node_modules/prettier": { 2963 | "version": "3.4.2", 2964 | "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.4.2.tgz", 2965 | "integrity": "sha512-e9MewbtFo+Fevyuxn/4rrcDAaq0IYxPGLvObpQjiZBMAzB9IGmzlnG9RZy3FFas+eBMu2vA0CszMeduow5dIuQ==", 2966 | "dev": true, 2967 | "bin": { 2968 | "prettier": "bin/prettier.cjs" 2969 | }, 2970 | "engines": { 2971 | "node": ">=14" 2972 | }, 2973 | "funding": { 2974 | "url": "https://github.com/prettier/prettier?sponsor=1" 2975 | } 2976 | }, 2977 | "node_modules/prettier-linter-helpers": { 2978 | "version": "1.0.0", 2979 | "resolved": "https://registry.npmjs.org/prettier-linter-helpers/-/prettier-linter-helpers-1.0.0.tgz", 2980 | "integrity": "sha512-GbK2cP9nraSSUF9N2XwUwqfzlAFlMNYYl+ShE/V+H8a9uNl/oUqB1w2EL54Jh0OlyRSd8RfWYJ3coVS4TROP2w==", 2981 | "dev": true, 2982 | "dependencies": { 2983 | "fast-diff": "^1.1.2" 2984 | }, 2985 | "engines": { 2986 | "node": ">=6.0.0" 2987 | } 2988 | }, 2989 | "node_modules/proxy-addr": { 2990 | "version": "2.0.7", 2991 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", 2992 | "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", 2993 | "dev": true, 2994 | "dependencies": { 2995 | "forwarded": "0.2.0", 2996 | "ipaddr.js": "1.9.1" 2997 | }, 2998 | "engines": { 2999 | "node": ">= 0.10" 3000 | } 3001 | }, 3002 | "node_modules/proxy-from-env": { 3003 | "version": "1.1.0", 3004 | "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz", 3005 | "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==" 3006 | }, 3007 | "node_modules/punycode": { 3008 | "version": "2.3.1", 3009 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz", 3010 | "integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==", 3011 | "dev": true, 3012 | "engines": { 3013 | "node": ">=6" 3014 | } 3015 | }, 3016 | "node_modules/qs": { 3017 | "version": "6.13.0", 3018 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.13.0.tgz", 3019 | "integrity": "sha512-+38qI9SOr8tfZ4QmJNplMUxqjbe7LKvvZgWdExBOmd+egZTtjLB67Gu0HRX3u/XOq7UU2Nx6nsjvS16Z9uwfpg==", 3020 | "dev": true, 3021 | "dependencies": { 3022 | "side-channel": "^1.0.6" 3023 | }, 3024 | "engines": { 3025 | "node": ">=0.6" 3026 | }, 3027 | "funding": { 3028 | "url": "https://github.com/sponsors/ljharb" 3029 | } 3030 | }, 3031 | "node_modules/queue-microtask": { 3032 | "version": "1.2.3", 3033 | "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", 3034 | "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", 3035 | "dev": true, 3036 | "funding": [ 3037 | { 3038 | "type": "github", 3039 | "url": "https://github.com/sponsors/feross" 3040 | }, 3041 | { 3042 | "type": "patreon", 3043 | "url": "https://www.patreon.com/feross" 3044 | }, 3045 | { 3046 | "type": "consulting", 3047 | "url": "https://feross.org/support" 3048 | } 3049 | ] 3050 | }, 3051 | "node_modules/range-parser": { 3052 | "version": "1.2.1", 3053 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", 3054 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==", 3055 | "dev": true, 3056 | "engines": { 3057 | "node": ">= 0.6" 3058 | } 3059 | }, 3060 | "node_modules/raw-body": { 3061 | "version": "3.0.0", 3062 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-3.0.0.tgz", 3063 | "integrity": "sha512-RmkhL8CAyCRPXCE28MMH0z2PNWQBNk2Q09ZdxM9IOOXwxwZbN+qbWaatPkdkWIKL2ZVDImrN/pK5HTRz2PcS4g==", 3064 | "dependencies": { 3065 | "bytes": "3.1.2", 3066 | "http-errors": "2.0.0", 3067 | "iconv-lite": "0.6.3", 3068 | "unpipe": "1.0.0" 3069 | }, 3070 | "engines": { 3071 | "node": ">= 0.8" 3072 | } 3073 | }, 3074 | "node_modules/raw-body/node_modules/iconv-lite": { 3075 | "version": "0.6.3", 3076 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz", 3077 | "integrity": "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==", 3078 | "dependencies": { 3079 | "safer-buffer": ">= 2.1.2 < 3.0.0" 3080 | }, 3081 | "engines": { 3082 | "node": ">=0.10.0" 3083 | } 3084 | }, 3085 | "node_modules/react": { 3086 | "version": "19.0.0", 3087 | "resolved": "https://registry.npmjs.org/react/-/react-19.0.0.tgz", 3088 | "integrity": "sha512-V8AVnmPIICiWpGfm6GLzCR/W5FXLchHop40W4nXBmdlEceh16rCN8O8LNWm5bh5XUX91fh7KpA+W0TgMKmgTpQ==", 3089 | "peer": true, 3090 | "engines": { 3091 | "node": ">=0.10.0" 3092 | } 3093 | }, 3094 | "node_modules/resolve-from": { 3095 | "version": "4.0.0", 3096 | "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", 3097 | "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", 3098 | "dev": true, 3099 | "engines": { 3100 | "node": ">=4" 3101 | } 3102 | }, 3103 | "node_modules/resolve-pkg-maps": { 3104 | "version": "1.0.0", 3105 | "resolved": "https://registry.npmjs.org/resolve-pkg-maps/-/resolve-pkg-maps-1.0.0.tgz", 3106 | "integrity": "sha512-seS2Tj26TBVOC2NIc2rOe2y2ZO7efxITtLZcGSOnHHNOQ7CkiUBfw0Iw2ck6xkIhPwLhKNLS8BO+hEpngQlqzw==", 3107 | "dev": true, 3108 | "funding": { 3109 | "url": "https://github.com/privatenumber/resolve-pkg-maps?sponsor=1" 3110 | } 3111 | }, 3112 | "node_modules/reusify": { 3113 | "version": "1.0.4", 3114 | "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz", 3115 | "integrity": "sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==", 3116 | "dev": true, 3117 | "engines": { 3118 | "iojs": ">=1.0.0", 3119 | "node": ">=0.10.0" 3120 | } 3121 | }, 3122 | "node_modules/rimraf": { 3123 | "version": "3.0.2", 3124 | "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", 3125 | "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", 3126 | "deprecated": "Rimraf versions prior to v4 are no longer supported", 3127 | "dev": true, 3128 | "dependencies": { 3129 | "glob": "^7.1.3" 3130 | }, 3131 | "bin": { 3132 | "rimraf": "bin.js" 3133 | }, 3134 | "funding": { 3135 | "url": "https://github.com/sponsors/isaacs" 3136 | } 3137 | }, 3138 | "node_modules/run-parallel": { 3139 | "version": "1.2.0", 3140 | "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz", 3141 | "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", 3142 | "dev": true, 3143 | "funding": [ 3144 | { 3145 | "type": "github", 3146 | "url": "https://github.com/sponsors/feross" 3147 | }, 3148 | { 3149 | "type": "patreon", 3150 | "url": "https://www.patreon.com/feross" 3151 | }, 3152 | { 3153 | "type": "consulting", 3154 | "url": "https://feross.org/support" 3155 | } 3156 | ], 3157 | "dependencies": { 3158 | "queue-microtask": "^1.2.2" 3159 | } 3160 | }, 3161 | "node_modules/safe-buffer": { 3162 | "version": "5.2.1", 3163 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", 3164 | "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", 3165 | "dev": true, 3166 | "funding": [ 3167 | { 3168 | "type": "github", 3169 | "url": "https://github.com/sponsors/feross" 3170 | }, 3171 | { 3172 | "type": "patreon", 3173 | "url": "https://www.patreon.com/feross" 3174 | }, 3175 | { 3176 | "type": "consulting", 3177 | "url": "https://feross.org/support" 3178 | } 3179 | ] 3180 | }, 3181 | "node_modules/safer-buffer": { 3182 | "version": "2.1.2", 3183 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", 3184 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" 3185 | }, 3186 | "node_modules/secure-json-parse": { 3187 | "version": "2.7.0", 3188 | "resolved": "https://registry.npmjs.org/secure-json-parse/-/secure-json-parse-2.7.0.tgz", 3189 | "integrity": "sha512-6aU+Rwsezw7VR8/nyvKTx8QpWH9FrcYiXXlqC4z5d5XQBDRqtbfsRjnwGyqbi3gddNtWHuEk9OANUotL26qKUw==" 3190 | }, 3191 | "node_modules/semver": { 3192 | "version": "7.7.1", 3193 | "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.1.tgz", 3194 | "integrity": "sha512-hlq8tAfn0m/61p4BVRcPzIGr6LKiMwo4VM6dGi6pt4qcRkmNzTcWq6eCEjEh+qXjkMDvPlOFFSGwQjoEa6gyMA==", 3195 | "dev": true, 3196 | "bin": { 3197 | "semver": "bin/semver.js" 3198 | }, 3199 | "engines": { 3200 | "node": ">=10" 3201 | } 3202 | }, 3203 | "node_modules/send": { 3204 | "version": "0.19.0", 3205 | "resolved": "https://registry.npmjs.org/send/-/send-0.19.0.tgz", 3206 | "integrity": "sha512-dW41u5VfLXu8SJh5bwRmyYUbAoSB3c9uQh6L8h/KtsFREPWpbX1lrljJo186Jc4nmci/sGUZ9a0a0J2zgfq2hw==", 3207 | "dev": true, 3208 | "dependencies": { 3209 | "debug": "2.6.9", 3210 | "depd": "2.0.0", 3211 | "destroy": "1.2.0", 3212 | "encodeurl": "~1.0.2", 3213 | "escape-html": "~1.0.3", 3214 | "etag": "~1.8.1", 3215 | "fresh": "0.5.2", 3216 | "http-errors": "2.0.0", 3217 | "mime": "1.6.0", 3218 | "ms": "2.1.3", 3219 | "on-finished": "2.4.1", 3220 | "range-parser": "~1.2.1", 3221 | "statuses": "2.0.1" 3222 | }, 3223 | "engines": { 3224 | "node": ">= 0.8.0" 3225 | } 3226 | }, 3227 | "node_modules/send/node_modules/debug": { 3228 | "version": "2.6.9", 3229 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 3230 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 3231 | "dev": true, 3232 | "dependencies": { 3233 | "ms": "2.0.0" 3234 | } 3235 | }, 3236 | "node_modules/send/node_modules/debug/node_modules/ms": { 3237 | "version": "2.0.0", 3238 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 3239 | "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==", 3240 | "dev": true 3241 | }, 3242 | "node_modules/send/node_modules/encodeurl": { 3243 | "version": "1.0.2", 3244 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", 3245 | "integrity": "sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w==", 3246 | "dev": true, 3247 | "engines": { 3248 | "node": ">= 0.8" 3249 | } 3250 | }, 3251 | "node_modules/serve-static": { 3252 | "version": "1.16.2", 3253 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.16.2.tgz", 3254 | "integrity": "sha512-VqpjJZKadQB/PEbEwvFdO43Ax5dFBZ2UECszz8bQ7pi7wt//PWe1P6MN7eCnjsatYtBT6EuiClbjSWP2WrIoTw==", 3255 | "dev": true, 3256 | "dependencies": { 3257 | "encodeurl": "~2.0.0", 3258 | "escape-html": "~1.0.3", 3259 | "parseurl": "~1.3.3", 3260 | "send": "0.19.0" 3261 | }, 3262 | "engines": { 3263 | "node": ">= 0.8.0" 3264 | } 3265 | }, 3266 | "node_modules/setprototypeof": { 3267 | "version": "1.2.0", 3268 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz", 3269 | "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==" 3270 | }, 3271 | "node_modules/shebang-command": { 3272 | "version": "2.0.0", 3273 | "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", 3274 | "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", 3275 | "dev": true, 3276 | "dependencies": { 3277 | "shebang-regex": "^3.0.0" 3278 | }, 3279 | "engines": { 3280 | "node": ">=8" 3281 | } 3282 | }, 3283 | "node_modules/shebang-regex": { 3284 | "version": "3.0.0", 3285 | "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", 3286 | "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", 3287 | "dev": true, 3288 | "engines": { 3289 | "node": ">=8" 3290 | } 3291 | }, 3292 | "node_modules/side-channel": { 3293 | "version": "1.1.0", 3294 | "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.1.0.tgz", 3295 | "integrity": "sha512-ZX99e6tRweoUXqR+VBrslhda51Nh5MTQwou5tnUDgbtyM0dBgmhEDtWGP/xbKn6hqfPRHujUNwz5fy/wbbhnpw==", 3296 | "dev": true, 3297 | "dependencies": { 3298 | "es-errors": "^1.3.0", 3299 | "object-inspect": "^1.13.3", 3300 | "side-channel-list": "^1.0.0", 3301 | "side-channel-map": "^1.0.1", 3302 | "side-channel-weakmap": "^1.0.2" 3303 | }, 3304 | "engines": { 3305 | "node": ">= 0.4" 3306 | }, 3307 | "funding": { 3308 | "url": "https://github.com/sponsors/ljharb" 3309 | } 3310 | }, 3311 | "node_modules/side-channel-list": { 3312 | "version": "1.0.0", 3313 | "resolved": "https://registry.npmjs.org/side-channel-list/-/side-channel-list-1.0.0.tgz", 3314 | "integrity": "sha512-FCLHtRD/gnpCiCHEiJLOwdmFP+wzCmDEkc9y7NsYxeF4u7Btsn1ZuwgwJGxImImHicJArLP4R0yX4c2KCrMrTA==", 3315 | "dev": true, 3316 | "dependencies": { 3317 | "es-errors": "^1.3.0", 3318 | "object-inspect": "^1.13.3" 3319 | }, 3320 | "engines": { 3321 | "node": ">= 0.4" 3322 | }, 3323 | "funding": { 3324 | "url": "https://github.com/sponsors/ljharb" 3325 | } 3326 | }, 3327 | "node_modules/side-channel-map": { 3328 | "version": "1.0.1", 3329 | "resolved": "https://registry.npmjs.org/side-channel-map/-/side-channel-map-1.0.1.tgz", 3330 | "integrity": "sha512-VCjCNfgMsby3tTdo02nbjtM/ewra6jPHmpThenkTYh8pG9ucZ/1P8So4u4FGBek/BjpOVsDCMoLA/iuBKIFXRA==", 3331 | "dev": true, 3332 | "dependencies": { 3333 | "call-bound": "^1.0.2", 3334 | "es-errors": "^1.3.0", 3335 | "get-intrinsic": "^1.2.5", 3336 | "object-inspect": "^1.13.3" 3337 | }, 3338 | "engines": { 3339 | "node": ">= 0.4" 3340 | }, 3341 | "funding": { 3342 | "url": "https://github.com/sponsors/ljharb" 3343 | } 3344 | }, 3345 | "node_modules/side-channel-weakmap": { 3346 | "version": "1.0.2", 3347 | "resolved": "https://registry.npmjs.org/side-channel-weakmap/-/side-channel-weakmap-1.0.2.tgz", 3348 | "integrity": "sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A==", 3349 | "dev": true, 3350 | "dependencies": { 3351 | "call-bound": "^1.0.2", 3352 | "es-errors": "^1.3.0", 3353 | "get-intrinsic": "^1.2.5", 3354 | "object-inspect": "^1.13.3", 3355 | "side-channel-map": "^1.0.1" 3356 | }, 3357 | "engines": { 3358 | "node": ">= 0.4" 3359 | }, 3360 | "funding": { 3361 | "url": "https://github.com/sponsors/ljharb" 3362 | } 3363 | }, 3364 | "node_modules/slash": { 3365 | "version": "3.0.0", 3366 | "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", 3367 | "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", 3368 | "dev": true, 3369 | "engines": { 3370 | "node": ">=8" 3371 | } 3372 | }, 3373 | "node_modules/statuses": { 3374 | "version": "2.0.1", 3375 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.1.tgz", 3376 | "integrity": "sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==", 3377 | "engines": { 3378 | "node": ">= 0.8" 3379 | } 3380 | }, 3381 | "node_modules/strip-ansi": { 3382 | "version": "6.0.1", 3383 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", 3384 | "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", 3385 | "dev": true, 3386 | "dependencies": { 3387 | "ansi-regex": "^5.0.1" 3388 | }, 3389 | "engines": { 3390 | "node": ">=8" 3391 | } 3392 | }, 3393 | "node_modules/strip-json-comments": { 3394 | "version": "3.1.1", 3395 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", 3396 | "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", 3397 | "dev": true, 3398 | "engines": { 3399 | "node": ">=8" 3400 | }, 3401 | "funding": { 3402 | "url": "https://github.com/sponsors/sindresorhus" 3403 | } 3404 | }, 3405 | "node_modules/supports-color": { 3406 | "version": "7.2.0", 3407 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", 3408 | "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", 3409 | "dev": true, 3410 | "dependencies": { 3411 | "has-flag": "^4.0.0" 3412 | }, 3413 | "engines": { 3414 | "node": ">=8" 3415 | } 3416 | }, 3417 | "node_modules/swr": { 3418 | "version": "2.3.0", 3419 | "resolved": "https://registry.npmjs.org/swr/-/swr-2.3.0.tgz", 3420 | "integrity": "sha512-NyZ76wA4yElZWBHzSgEJc28a0u6QZvhb6w0azeL2k7+Q1gAzVK+IqQYXhVOC/mzi+HZIozrZvBVeSeOZNR2bqA==", 3421 | "dependencies": { 3422 | "dequal": "^2.0.3", 3423 | "use-sync-external-store": "^1.4.0" 3424 | }, 3425 | "peerDependencies": { 3426 | "react": "^16.11.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" 3427 | } 3428 | }, 3429 | "node_modules/synckit": { 3430 | "version": "0.9.2", 3431 | "resolved": "https://registry.npmjs.org/synckit/-/synckit-0.9.2.tgz", 3432 | "integrity": "sha512-vrozgXDQwYO72vHjUb/HnFbQx1exDjoKzqx23aXEg2a9VIg2TSFZ8FmeZpTjUCFMYw7mpX4BE2SFu8wI7asYsw==", 3433 | "dev": true, 3434 | "dependencies": { 3435 | "@pkgr/core": "^0.1.0", 3436 | "tslib": "^2.6.2" 3437 | }, 3438 | "engines": { 3439 | "node": "^14.18.0 || >=16.0.0" 3440 | }, 3441 | "funding": { 3442 | "url": "https://opencollective.com/unts" 3443 | } 3444 | }, 3445 | "node_modules/text-table": { 3446 | "version": "0.2.0", 3447 | "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", 3448 | "integrity": "sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==", 3449 | "dev": true 3450 | }, 3451 | "node_modules/throttleit": { 3452 | "version": "2.1.0", 3453 | "resolved": "https://registry.npmjs.org/throttleit/-/throttleit-2.1.0.tgz", 3454 | "integrity": "sha512-nt6AMGKW1p/70DF/hGBdJB57B8Tspmbp5gfJ8ilhLnt7kkr2ye7hzD6NVG8GGErk2HWF34igrL2CXmNIkzKqKw==", 3455 | "engines": { 3456 | "node": ">=18" 3457 | }, 3458 | "funding": { 3459 | "url": "https://github.com/sponsors/sindresorhus" 3460 | } 3461 | }, 3462 | "node_modules/to-regex-range": { 3463 | "version": "5.0.1", 3464 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", 3465 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", 3466 | "dev": true, 3467 | "dependencies": { 3468 | "is-number": "^7.0.0" 3469 | }, 3470 | "engines": { 3471 | "node": ">=8.0" 3472 | } 3473 | }, 3474 | "node_modules/toidentifier": { 3475 | "version": "1.0.1", 3476 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz", 3477 | "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==", 3478 | "engines": { 3479 | "node": ">=0.6" 3480 | } 3481 | }, 3482 | "node_modules/ts-api-utils": { 3483 | "version": "1.4.3", 3484 | "resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-1.4.3.tgz", 3485 | "integrity": "sha512-i3eMG77UTMD0hZhgRS562pv83RC6ukSAC2GMNWc+9dieh/+jDM5u5YG+NHX6VNDRHQcHwmsTHctP9LhbC3WxVw==", 3486 | "dev": true, 3487 | "engines": { 3488 | "node": ">=16" 3489 | }, 3490 | "peerDependencies": { 3491 | "typescript": ">=4.2.0" 3492 | } 3493 | }, 3494 | "node_modules/tslib": { 3495 | "version": "2.8.1", 3496 | "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.1.tgz", 3497 | "integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==", 3498 | "dev": true 3499 | }, 3500 | "node_modules/tsx": { 3501 | "version": "4.19.2", 3502 | "resolved": "https://registry.npmjs.org/tsx/-/tsx-4.19.2.tgz", 3503 | "integrity": "sha512-pOUl6Vo2LUq/bSa8S5q7b91cgNSjctn9ugq/+Mvow99qW6x/UZYwzxy/3NmqoT66eHYfCVvFvACC58UBPFf28g==", 3504 | "dev": true, 3505 | "dependencies": { 3506 | "esbuild": "~0.23.0", 3507 | "get-tsconfig": "^4.7.5" 3508 | }, 3509 | "bin": { 3510 | "tsx": "dist/cli.mjs" 3511 | }, 3512 | "engines": { 3513 | "node": ">=18.0.0" 3514 | }, 3515 | "optionalDependencies": { 3516 | "fsevents": "~2.3.3" 3517 | } 3518 | }, 3519 | "node_modules/type-check": { 3520 | "version": "0.4.0", 3521 | "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz", 3522 | "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==", 3523 | "dev": true, 3524 | "dependencies": { 3525 | "prelude-ls": "^1.2.1" 3526 | }, 3527 | "engines": { 3528 | "node": ">= 0.8.0" 3529 | } 3530 | }, 3531 | "node_modules/type-fest": { 3532 | "version": "0.20.2", 3533 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", 3534 | "integrity": "sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==", 3535 | "dev": true, 3536 | "engines": { 3537 | "node": ">=10" 3538 | }, 3539 | "funding": { 3540 | "url": "https://github.com/sponsors/sindresorhus" 3541 | } 3542 | }, 3543 | "node_modules/type-is": { 3544 | "version": "1.6.18", 3545 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", 3546 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", 3547 | "dev": true, 3548 | "dependencies": { 3549 | "media-typer": "0.3.0", 3550 | "mime-types": "~2.1.24" 3551 | }, 3552 | "engines": { 3553 | "node": ">= 0.6" 3554 | } 3555 | }, 3556 | "node_modules/typescript": { 3557 | "version": "5.7.3", 3558 | "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.7.3.tgz", 3559 | "integrity": "sha512-84MVSjMEHP+FQRPy3pX9sTVV/INIex71s9TL2Gm5FG/WG1SqXeKyZ0k7/blY/4FdOzI12CBy1vGc4og/eus0fw==", 3560 | "dev": true, 3561 | "bin": { 3562 | "tsc": "bin/tsc", 3563 | "tsserver": "bin/tsserver" 3564 | }, 3565 | "engines": { 3566 | "node": ">=14.17" 3567 | } 3568 | }, 3569 | "node_modules/undici-types": { 3570 | "version": "6.19.8", 3571 | "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.19.8.tgz", 3572 | "integrity": "sha512-ve2KP6f/JnbPBFyobGHuerC9g1FYGn/F8n1LWTwNxCEzd6IfqTwUQcNXgEtmmQ6DlRrC1hrSrBnCZPokRrDHjw==", 3573 | "dev": true 3574 | }, 3575 | "node_modules/unpipe": { 3576 | "version": "1.0.0", 3577 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", 3578 | "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==", 3579 | "engines": { 3580 | "node": ">= 0.8" 3581 | } 3582 | }, 3583 | "node_modules/uri-js": { 3584 | "version": "4.4.1", 3585 | "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", 3586 | "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==", 3587 | "dev": true, 3588 | "dependencies": { 3589 | "punycode": "^2.1.0" 3590 | } 3591 | }, 3592 | "node_modules/use-sync-external-store": { 3593 | "version": "1.4.0", 3594 | "resolved": "https://registry.npmjs.org/use-sync-external-store/-/use-sync-external-store-1.4.0.tgz", 3595 | "integrity": "sha512-9WXSPC5fMv61vaupRkCKCxsPxBocVnwakBEkMIHHpkTTg6icbJtg6jzgtLDm4bl3cSHAca52rYWih0k4K3PfHw==", 3596 | "peerDependencies": { 3597 | "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" 3598 | } 3599 | }, 3600 | "node_modules/utils-merge": { 3601 | "version": "1.0.1", 3602 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", 3603 | "integrity": "sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==", 3604 | "dev": true, 3605 | "engines": { 3606 | "node": ">= 0.4.0" 3607 | } 3608 | }, 3609 | "node_modules/vary": { 3610 | "version": "1.1.2", 3611 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", 3612 | "integrity": "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==", 3613 | "dev": true, 3614 | "engines": { 3615 | "node": ">= 0.8" 3616 | } 3617 | }, 3618 | "node_modules/which": { 3619 | "version": "2.0.2", 3620 | "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", 3621 | "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", 3622 | "dev": true, 3623 | "dependencies": { 3624 | "isexe": "^2.0.0" 3625 | }, 3626 | "bin": { 3627 | "node-which": "bin/node-which" 3628 | }, 3629 | "engines": { 3630 | "node": ">= 8" 3631 | } 3632 | }, 3633 | "node_modules/word-wrap": { 3634 | "version": "1.2.5", 3635 | "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.5.tgz", 3636 | "integrity": "sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==", 3637 | "dev": true, 3638 | "engines": { 3639 | "node": ">=0.10.0" 3640 | } 3641 | }, 3642 | "node_modules/wrappy": { 3643 | "version": "1.0.2", 3644 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", 3645 | "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", 3646 | "dev": true 3647 | }, 3648 | "node_modules/yocto-queue": { 3649 | "version": "0.1.0", 3650 | "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", 3651 | "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", 3652 | "dev": true, 3653 | "engines": { 3654 | "node": ">=10" 3655 | }, 3656 | "funding": { 3657 | "url": "https://github.com/sponsors/sindresorhus" 3658 | } 3659 | }, 3660 | "node_modules/zod": { 3661 | "version": "3.24.1", 3662 | "resolved": "https://registry.npmjs.org/zod/-/zod-3.24.1.tgz", 3663 | "integrity": "sha512-muH7gBL9sI1nciMZV67X5fTKKBLtwpZ5VBp1vsOQzj1MhrBZ4wlVCm3gedKZWLp0Oyel8sIGfeiz54Su+OVT+A==", 3664 | "funding": { 3665 | "url": "https://github.com/sponsors/colinhacks" 3666 | } 3667 | }, 3668 | "node_modules/zod-to-json-schema": { 3669 | "version": "3.24.1", 3670 | "resolved": "https://registry.npmjs.org/zod-to-json-schema/-/zod-to-json-schema-3.24.1.tgz", 3671 | "integrity": "sha512-3h08nf3Vw3Wl3PK+q3ow/lIil81IT2Oa7YpQyUUDsEWbXveMesdfK1xBd2RhCkynwZndAxixji/7SYJJowr62w==", 3672 | "peerDependencies": { 3673 | "zod": "^3.24.1" 3674 | } 3675 | } 3676 | } 3677 | } 3678 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "aisdk-mcp-bridge", 3 | "version": "0.1.0", 4 | "description": "Bridge between Model Context Protocol (MCP) servers and AI SDK tools", 5 | "main": "./dist/src/index.js", 6 | "types": "./dist/src/index.d.ts", 7 | "exports": { 8 | ".": { 9 | "types": "./dist/src/index.d.ts", 10 | "default": "./dist/src/index.js" 11 | } 12 | }, 13 | "files": [ 14 | "dist", 15 | "README.md", 16 | "LICENSE" 17 | ], 18 | "scripts": { 19 | "build": "tsc", 20 | "lint": "eslint . --ext .ts", 21 | "format": "prettier --write \"src/**/*.ts\"", 22 | "test:twitter": "tsx examples/test-twitter.ts", 23 | "test:playwright": "tsx examples/test-playwright.ts", 24 | "test:firecrawl": "tsx examples/test-firecrawl.ts", 25 | "prepare": "npm run build", 26 | "prepublishOnly": "npm run lint", 27 | "preversion": "npm run lint", 28 | "version": "npm run format && git add -A src", 29 | "postversion": "git push && git push --tags" 30 | }, 31 | "keywords": [ 32 | "ai", 33 | "mcp", 34 | "ai-sdk", 35 | "bridge", 36 | "tools", 37 | "typescript" 38 | ], 39 | "author": "Ravi Kiran (https://github.com/vrknetha)", 40 | "license": "MIT", 41 | "repository": { 42 | "type": "git", 43 | "url": "git+https://github.com/vrknetha/aisdk-mcp-bridge.git" 44 | }, 45 | "bugs": { 46 | "url": "https://github.com/vrknetha/aisdk-mcp-bridge/issues" 47 | }, 48 | "homepage": "https://github.com/vrknetha/aisdk-mcp-bridge#readme", 49 | "dependencies": { 50 | "@modelcontextprotocol/sdk": "^1.4.1", 51 | "ai": "^4.1.10", 52 | "axios": "^1.6.7", 53 | "zod": "^3.22.4" 54 | }, 55 | "peerDependencies": { 56 | "ai": "^4.1.10" 57 | }, 58 | "devDependencies": { 59 | "@ai-sdk/google": "^1.0.0", 60 | "@openrouter/ai-sdk-provider": "^0.2.0", 61 | "@types/cors": "^2.8.13", 62 | "@types/express": "^4.17.17", 63 | "@types/node": "^20.11.16", 64 | "@typescript-eslint/eslint-plugin": "^6.21.0", 65 | "@typescript-eslint/parser": "^6.21.0", 66 | "cors": "^2.8.5", 67 | "dotenv": "^16.0.3", 68 | "eslint": "^8.56.0", 69 | "eslint-config-prettier": "^9.1.0", 70 | "eslint-plugin-prettier": "^5.1.3", 71 | "express": "^4.18.2", 72 | "prettier": "^3.2.5", 73 | "tsx": "^4.19.2", 74 | "typescript": "^5.3.3" 75 | }, 76 | "engines": { 77 | "node": ">=20.0.0" 78 | } 79 | } 80 | -------------------------------------------------------------------------------- /src/index.ts: -------------------------------------------------------------------------------- 1 | import { ToolSet } from 'ai'; 2 | import { MCPService, MCPToolResult } from './service'; 3 | import { 4 | MCPServerManager, 5 | type MCPServersConfig, 6 | type ServerConfig, 7 | ServerConfigSchema, 8 | MCPServersConfigSchema, 9 | } from './server'; 10 | import { log } from './tools'; 11 | import path from 'path'; 12 | import fs from 'fs/promises'; 13 | import { z } from 'zod'; 14 | import { StdioServerParameters } from '@modelcontextprotocol/sdk/client/stdio.js'; 15 | 16 | // Core types 17 | export type { MCPServersConfig as MCPConfig } from './server'; 18 | export type { ServerConfig } from './server'; 19 | export type { MCPToolResult } from './service'; 20 | 21 | // Helper to convert ServerConfig to StdioServerParameters (internal use only) 22 | export function toStdioParams(config: ServerConfig): StdioServerParameters { 23 | // Ensure PATH includes common binary locations 24 | const envPath = process.env.PATH || ''; 25 | const additionalPaths = [ 26 | '/usr/local/bin', 27 | '/usr/bin', 28 | '/bin', 29 | '/opt/homebrew/bin', 30 | './node_modules/.bin', 31 | ].join(':'); 32 | 33 | return { 34 | command: config.command, 35 | args: config.args, 36 | env: { 37 | ...process.env, 38 | NODE_ENV: process.env.NODE_ENV || 'development', 39 | PATH: `${envPath}:${additionalPaths}`, 40 | ...config.env, 41 | }, 42 | stderr: 'inherit', 43 | }; 44 | } 45 | 46 | /** 47 | * Load MCP configuration from a JSON file 48 | */ 49 | async function loadMcpConfig(configPath: string): Promise { 50 | try { 51 | const configContent = await fs.readFile(configPath, 'utf-8'); 52 | const config = JSON.parse(configContent); 53 | return MCPServersConfigSchema.parse(config); 54 | } catch (error) { 55 | log(`Failed to load config from ${configPath}`, error, { type: 'error' }); 56 | throw error; 57 | } 58 | } 59 | 60 | /** 61 | * Save MCP configuration to a JSON file 62 | */ 63 | async function saveMcpConfig( 64 | config: MCPServersConfig, 65 | configPath: string 66 | ): Promise { 67 | try { 68 | await fs.writeFile(configPath, JSON.stringify(config, null, 2)); 69 | log('Saved MCP config', { path: configPath }, { type: 'debug' }); 70 | } catch (error) { 71 | log(`Failed to save config to ${configPath}`, error, { type: 'error' }); 72 | throw error; 73 | } 74 | } 75 | 76 | /** 77 | * Validate server configuration 78 | */ 79 | function validateServerConfig(config: ServerConfig): string[] { 80 | try { 81 | ServerConfigSchema.parse(config); 82 | return []; 83 | } catch (error) { 84 | if (error instanceof z.ZodError) { 85 | const errors = error.errors.map( 86 | err => `${err.path.join('.')}: ${err.message}` 87 | ); 88 | log('Server config validation failed', { errors }, { type: 'error' }); 89 | return errors; 90 | } 91 | log('Invalid server configuration', error, { type: 'error' }); 92 | return ['Invalid server configuration']; 93 | } 94 | } 95 | 96 | /** 97 | * Initialize MCP service 98 | */ 99 | export async function initializeMcp( 100 | options: { 101 | configPath?: string; 102 | debug?: boolean; 103 | } = {} 104 | ): Promise { 105 | const { debug = false } = options; 106 | const service = MCPService.getInstance(); 107 | 108 | // Check if already initialized 109 | if (service.isInitialized()) { 110 | log('MCP service already initialized', undefined, { debug }); 111 | return; 112 | } 113 | 114 | log('Initializing MCP...', undefined, { debug: true }); 115 | 116 | try { 117 | // Load configuration 118 | const configPath = 119 | options.configPath || path.join(process.cwd(), 'mcp.config.json'); 120 | const config = await loadMcpConfig(configPath); 121 | log('Loaded MCP configuration', config, { debug }); 122 | 123 | // Initialize server manager with config 124 | const serverManager = MCPServerManager.getInstance(config); 125 | 126 | // Initialize MCP service with server manager 127 | service.setServerManager(serverManager); 128 | 129 | // Start all servers 130 | const results = await serverManager.startAllServers(); 131 | log('Server startup results:', results, { debug: true }); 132 | 133 | // Check server statuses 134 | const failedServers = []; 135 | const runningServers = []; 136 | 137 | for (const [name, success] of results.entries()) { 138 | if (success) { 139 | runningServers.push(name); 140 | log(`Server ${name} started successfully`, undefined, { type: 'info' }); 141 | } else { 142 | failedServers.push(name); 143 | log(`Server ${name} failed to start`, undefined, { type: 'error' }); 144 | } 145 | } 146 | 147 | if (failedServers.length > 0) { 148 | log( 149 | `Some servers failed to start: ${failedServers.join(', ')}`, 150 | undefined, 151 | { 152 | type: 'error', 153 | } 154 | ); 155 | } 156 | 157 | if (runningServers.length === 0) { 158 | throw new Error('No servers started successfully'); 159 | } 160 | 161 | // Mark service as initialized since we have at least one working server 162 | await service.initialize({ debug }); 163 | log( 164 | `MCP service initialized with servers: ${runningServers.join(', ')}`, 165 | undefined, 166 | { 167 | type: 'info', 168 | } 169 | ); 170 | } catch (error) { 171 | log('Failed to initialize MCP', error, { type: 'error' }); 172 | await service.cleanup(); // Clean up on initialization failure 173 | throw error; 174 | } 175 | } 176 | 177 | /** 178 | * Get tools from MCP servers 179 | */ 180 | export async function getMcpTools( 181 | options: { 182 | debug?: boolean; 183 | serverName?: string; 184 | } = {} 185 | ): Promise { 186 | const { debug = false, serverName } = options; 187 | 188 | try { 189 | log('Getting MCP tools...', serverName ? { serverName } : undefined, { 190 | debug: true, 191 | }); 192 | const service = MCPService.getInstance(); 193 | 194 | if (!service.isInitialized()) { 195 | log('MCP service not initialized', undefined, { type: 'error' }); 196 | throw new Error( 197 | 'MCP service not initialized. Call initializeMcp() first.' 198 | ); 199 | } 200 | 201 | // Check if the specified server exists in configuration 202 | if (serverName) { 203 | const serverManager = MCPServerManager.getInstance(service.getConfig()); 204 | const config = serverManager.getConfig(); 205 | if (!config?.mcpServers[serverName]) { 206 | const error = new Error( 207 | `Server "${serverName}" not found in configuration` 208 | ); 209 | log(`Failed to get MCP tools: ${error.message}`, undefined, { 210 | type: 'error', 211 | }); 212 | throw error; 213 | } 214 | } 215 | 216 | const tools = await service.getTools({ debug, serverName }); 217 | log( 218 | `Retrieved MCP tools${serverName ? ` for server ${serverName}` : ' for all servers'}`, 219 | { toolCount: Object.keys(tools).length }, 220 | { debug: true } 221 | ); 222 | return tools; 223 | } catch (error) { 224 | log( 225 | `Failed to get MCP tools${serverName ? ` for server ${serverName}` : ''}`, 226 | error, 227 | { type: 'error' } 228 | ); 229 | throw error; 230 | } 231 | } 232 | 233 | /** 234 | * Execute a specific function on an MCP server. 235 | * 236 | * @param serverName - Name of the server to execute on 237 | * @param functionName - Name of the function to execute 238 | * @param args - Arguments to pass to the function 239 | */ 240 | export async function executeMcpFunction( 241 | serverName: string, 242 | functionName: string, 243 | args: Record 244 | ): Promise { 245 | const service = MCPService.getInstance(); 246 | return service.executeFunction(serverName, functionName, args); 247 | } 248 | 249 | /** 250 | * Clean up MCP resources 251 | */ 252 | export async function cleanupMcp(): Promise { 253 | try { 254 | log('Cleaning up MCP resources...', undefined, { debug: true }); 255 | const service = MCPService.getInstance(); 256 | 257 | let cleanupComplete = false; 258 | const cleanupTimeout = setTimeout(() => { 259 | if (!cleanupComplete) { 260 | log('Cleanup timeout reached, forcing exit...', undefined, { 261 | type: 'error', 262 | }); 263 | process.exit(1); 264 | } 265 | }, 15000); 266 | 267 | try { 268 | await service.cleanup(); 269 | log('MCP cleanup complete', undefined, { debug: true }); 270 | 271 | // Clear any remaining intervals/timeouts 272 | const intervalIds = getIntervalIds(); 273 | intervalIds.forEach(clearInterval); 274 | const timeoutIds = getTimeoutIds(); 275 | timeoutIds.forEach(clearTimeout); 276 | 277 | cleanupComplete = true; 278 | clearTimeout(cleanupTimeout); 279 | 280 | // Give a small delay for final cleanup operations 281 | await new Promise(resolve => setTimeout(resolve, 1000)); 282 | process.exit(0); 283 | } catch (error) { 284 | log('Cleanup failed', error, { type: 'error' }); 285 | process.exit(1); 286 | } 287 | } catch (error) { 288 | log('Failed to clean up MCP resources', error, { type: 'error' }); 289 | process.exit(1); 290 | } 291 | } 292 | 293 | // Helper function to get all active interval IDs 294 | function getIntervalIds(): NodeJS.Timeout[] { 295 | const ids: NodeJS.Timeout[] = []; 296 | const originalSetInterval = global.setInterval; 297 | global.setInterval = function ( 298 | callback: () => void, 299 | ms?: number 300 | ): NodeJS.Timeout { 301 | const id = originalSetInterval(callback, ms); 302 | ids.push(id); 303 | return id; 304 | } as typeof global.setInterval; 305 | return ids; 306 | } 307 | 308 | // Helper function to get all active timeout IDs 309 | function getTimeoutIds(): NodeJS.Timeout[] { 310 | const ids: NodeJS.Timeout[] = []; 311 | const originalSetTimeout = global.setTimeout; 312 | global.setTimeout = function ( 313 | callback: () => void, 314 | ms?: number 315 | ): NodeJS.Timeout { 316 | const id = originalSetTimeout(callback, ms); 317 | ids.push(id); 318 | return id; 319 | } as typeof global.setTimeout; 320 | return ids; 321 | } 322 | 323 | // Export configuration types and utilities 324 | export { 325 | ServerConfigSchema, 326 | MCPServersConfigSchema, 327 | loadMcpConfig, 328 | saveMcpConfig, 329 | validateServerConfig, 330 | }; 331 | 332 | // Export singleton instances for advanced usage 333 | export const mcpService = MCPService.getInstance(); 334 | -------------------------------------------------------------------------------- /src/server.ts: -------------------------------------------------------------------------------- 1 | import { z } from 'zod'; 2 | import { spawn, ChildProcess } from 'child_process'; 3 | import { log } from './tools'; 4 | 5 | // Server Configuration Schema 6 | export const ServerConfigSchema = z.object({ 7 | command: z.string(), 8 | args: z.array(z.string()), 9 | env: z.record(z.string()).optional(), 10 | mode: z.enum(['stdio', 'sse']).default('stdio'), 11 | autoApprove: z.array(z.string()).optional(), 12 | disabled: z.boolean().optional(), 13 | // SSE specific options 14 | sseOptions: z 15 | .object({ 16 | endpoint: z.string(), 17 | headers: z.record(z.string()).optional(), 18 | reconnectTimeout: z.number().optional(), 19 | }) 20 | .optional(), 21 | }); 22 | 23 | export type ServerConfig = z.infer; 24 | 25 | export const MCPServersConfigSchema = z.object({ 26 | mcpServers: z.record(z.string(), ServerConfigSchema), 27 | }); 28 | 29 | export type MCPServersConfig = z.infer; 30 | 31 | interface RunningServer extends ServerConfig { 32 | startTime: number; 33 | process?: ChildProcess; 34 | } 35 | 36 | export class MCPServerManager { 37 | private static instance: MCPServerManager; 38 | private config: MCPServersConfig; 39 | private runningServers: Map = new Map(); 40 | private startupPromises: Map> = new Map(); 41 | 42 | private constructor(config: MCPServersConfig) { 43 | this.config = config; 44 | } 45 | 46 | public static getInstance(config: MCPServersConfig): MCPServerManager { 47 | if (!MCPServerManager.instance) { 48 | MCPServerManager.instance = new MCPServerManager(config); 49 | } 50 | return MCPServerManager.instance; 51 | } 52 | 53 | public setConfig(config: MCPServersConfig) { 54 | try { 55 | // Validate the config using the schema 56 | const validatedConfig = MCPServersConfigSchema.parse(config); 57 | 58 | // Store the validated config 59 | this.config = validatedConfig; 60 | 61 | // Reset server state when config changes 62 | this.runningServers.clear(); 63 | 64 | return true; 65 | } catch (error) { 66 | console.error('Invalid server configuration:', error); 67 | throw error; 68 | } 69 | } 70 | 71 | public getConfig(): MCPServersConfig { 72 | return this.config; 73 | } 74 | 75 | public getServerInfo(serverName: string): ServerConfig | undefined { 76 | return this.config.mcpServers[serverName]; 77 | } 78 | 79 | private async startStdioServer( 80 | serverName: string, 81 | serverConfig: ServerConfig 82 | ): Promise { 83 | try { 84 | log(`Starting ${serverName} in stdio mode...`, undefined, { 85 | type: 'info', 86 | }); 87 | const server = spawn(serverConfig.command, serverConfig.args, { 88 | env: { 89 | ...process.env, 90 | ...serverConfig.env, 91 | }, 92 | stdio: ['pipe', 'pipe', 'pipe'], 93 | }); 94 | 95 | // Handle server output 96 | this.setupServerLogging(server, serverName); 97 | 98 | server.on('close', code => { 99 | log(`[${serverName}] Server exited with code ${code}`, undefined, { 100 | type: 'info', 101 | }); 102 | this.runningServers.delete(serverName); 103 | }); 104 | 105 | // Store server reference 106 | this.runningServers.set(serverName, { 107 | ...serverConfig, 108 | process: server, 109 | mode: 'stdio', 110 | startTime: Date.now(), 111 | }); 112 | log(`[${serverName}] Server registered in stdio mode`, undefined, { 113 | type: 'info', 114 | }); 115 | 116 | // Wait a bit for the server to initialize 117 | await new Promise(resolve => setTimeout(resolve, 2000)); 118 | 119 | // Verify server is still registered 120 | if (!this.runningServers.has(serverName)) { 121 | log( 122 | `[${serverName}] Server registration lost during initialization`, 123 | undefined, 124 | { type: 'error' } 125 | ); 126 | return false; 127 | } 128 | 129 | log(`[${serverName}] Server initialization complete`, undefined, { 130 | type: 'info', 131 | }); 132 | return true; 133 | } catch (error) { 134 | log(`Failed to start stdio server ${serverName}`, error, { 135 | type: 'error', 136 | }); 137 | return false; 138 | } 139 | } 140 | 141 | private async startSseServer( 142 | serverName: string, 143 | serverConfig: ServerConfig 144 | ): Promise { 145 | try { 146 | log(`Starting ${serverName} in SSE mode...`, undefined, { 147 | type: 'info', 148 | }); 149 | 150 | // Create a promise that resolves when the server is ready 151 | const serverReadyPromise = new Promise(resolve => { 152 | let isReady = false; 153 | const readyTimeout = setTimeout(() => { 154 | if (!isReady) { 155 | log( 156 | `Server ${serverName} failed to initialize within timeout`, 157 | undefined, 158 | { 159 | type: 'error', 160 | } 161 | ); 162 | resolve(false); 163 | } 164 | }, 10000); // 10 second timeout 165 | 166 | const server = spawn(serverConfig.command, serverConfig.args, { 167 | env: { 168 | ...process.env, 169 | ...serverConfig.env, 170 | }, 171 | stdio: ['ignore', 'pipe', 'pipe'], 172 | }); 173 | 174 | // Handle server output 175 | this.setupServerLogging(server, serverName, output => { 176 | // Check for server ready message 177 | if (output.includes('SSE server ready on port')) { 178 | isReady = true; 179 | clearTimeout(readyTimeout); 180 | 181 | // Store server reference only after it's ready 182 | this.runningServers.set(serverName, { 183 | ...serverConfig, 184 | process: server, 185 | mode: 'sse', 186 | startTime: Date.now(), 187 | }); 188 | 189 | resolve(true); 190 | } 191 | }); 192 | 193 | server.on('close', code => { 194 | log(`[${serverName}] Server exited with code ${code}`, undefined, { 195 | type: 'info', 196 | }); 197 | if (!isReady) { 198 | resolve(false); 199 | } 200 | this.runningServers.delete(serverName); 201 | }); 202 | 203 | server.on('error', error => { 204 | log(`[${serverName}] Server error: ${error.message}`, undefined, { 205 | type: 'error', 206 | }); 207 | if (!isReady) { 208 | resolve(false); 209 | } 210 | }); 211 | }); 212 | 213 | // Wait for server to be ready 214 | const success = await serverReadyPromise; 215 | if (!success) { 216 | return false; 217 | } 218 | 219 | // Verify server is still registered 220 | if (!this.runningServers.has(serverName)) { 221 | log( 222 | `[${serverName}] Server registration lost during initialization`, 223 | undefined, 224 | { type: 'error' } 225 | ); 226 | return false; 227 | } 228 | 229 | log(`[${serverName}] Server initialization complete`, undefined, { 230 | type: 'info', 231 | }); 232 | return true; 233 | } catch (error) { 234 | log(`Failed to start SSE server ${serverName}`, error, { 235 | type: 'error', 236 | }); 237 | return false; 238 | } 239 | } 240 | 241 | // Helper method to set up server output logging 242 | private setupServerLogging( 243 | server: ChildProcess, 244 | serverName: string, 245 | onOutput?: (output: string) => void 246 | ): void { 247 | if (server.stdout) { 248 | server.stdout.on('data', data => { 249 | const output = data.toString(); 250 | log(`[${serverName}] ${output}`, undefined, { type: 'info' }); 251 | onOutput?.(output); 252 | }); 253 | } 254 | 255 | if (server.stderr) { 256 | server.stderr.on('data', data => { 257 | const output = data.toString(); 258 | log(`[${serverName}] Error: ${output}`, undefined, { 259 | type: 'error', 260 | }); 261 | onOutput?.(output); 262 | }); 263 | } 264 | } 265 | 266 | public async startAllServers(): Promise> { 267 | const results = new Map(); 268 | const startupPromises: Promise[] = []; 269 | 270 | for (const [serverName, serverConfig] of Object.entries( 271 | this.config.mcpServers 272 | )) { 273 | if (serverConfig.disabled) { 274 | log(`Server ${serverName} is disabled, skipping...`, undefined, { 275 | type: 'info', 276 | }); 277 | results.set(serverName, false); 278 | continue; 279 | } 280 | 281 | if (this.runningServers.has(serverName)) { 282 | log(`Server ${serverName} is already running`, undefined, { 283 | type: 'info', 284 | }); 285 | results.set(serverName, true); 286 | continue; 287 | } 288 | 289 | // Start each server independently 290 | startupPromises.push( 291 | this.startServer(serverName) 292 | .then(success => { 293 | results.set(serverName, success); 294 | if (!success) { 295 | log( 296 | `Server ${serverName} failed to start, continuing with other servers`, 297 | undefined, 298 | { 299 | type: 'error', 300 | } 301 | ); 302 | } 303 | }) 304 | .catch(error => { 305 | log(`Error starting server ${serverName}`, error, { 306 | type: 'error', 307 | }); 308 | results.set(serverName, false); 309 | // Don't rethrow - allow other servers to continue 310 | }) 311 | ); 312 | } 313 | 314 | // Wait for all servers to complete startup attempts 315 | await Promise.allSettled(startupPromises); 316 | return results; 317 | } 318 | 319 | public async startServer(serverName: string): Promise { 320 | const serverConfig = this.getServerInfo(serverName); 321 | if (!serverConfig) { 322 | log(`Server ${serverName} not found in configuration`, undefined, { 323 | type: 'error', 324 | }); 325 | return false; 326 | } 327 | 328 | if (this.isServerRunning(serverName)) { 329 | log(`Server ${serverName} is already running`, undefined, { 330 | type: 'debug', 331 | }); 332 | return true; 333 | } 334 | 335 | try { 336 | // Check if server is already starting 337 | const existingPromise = this.startupPromises.get(serverName); 338 | if (existingPromise) { 339 | log( 340 | `Server ${serverName} is already starting, waiting for completion`, 341 | undefined, 342 | { 343 | type: 'debug', 344 | } 345 | ); 346 | return existingPromise; 347 | } 348 | 349 | // Create startup promise 350 | const startupPromise = new Promise(resolve => { 351 | try { 352 | log(`Starting server ${serverName}...`, undefined, { 353 | type: 'debug', 354 | }); 355 | 356 | const startServer = async () => { 357 | switch (serverConfig.mode) { 358 | case 'stdio': 359 | return await this.startStdioServer(serverName, serverConfig); 360 | case 'sse': 361 | return await this.startSseServer(serverName, serverConfig); 362 | default: 363 | throw new Error( 364 | `Unsupported server mode: ${serverConfig.mode}` 365 | ); 366 | } 367 | }; 368 | 369 | startServer() 370 | .then(resolve) 371 | .catch(error => { 372 | log(`Failed to start server ${serverName}`, error, { 373 | type: 'error', 374 | }); 375 | resolve(false); 376 | }) 377 | .finally(() => { 378 | this.startupPromises.delete(serverName); 379 | }); 380 | } catch (error) { 381 | log(`Failed to start server ${serverName}`, error, { 382 | type: 'error', 383 | }); 384 | this.startupPromises.delete(serverName); 385 | resolve(false); 386 | } 387 | }); 388 | 389 | this.startupPromises.set(serverName, startupPromise); 390 | return startupPromise; 391 | } catch (error) { 392 | log(`Error starting server ${serverName}`, error, { type: 'error' }); 393 | return false; 394 | } 395 | } 396 | 397 | public async stopServer(serverName: string): Promise { 398 | const serverInfo = this.runningServers.get(serverName); 399 | if (!serverInfo) { 400 | return; 401 | } 402 | 403 | try { 404 | if (serverInfo.process) { 405 | // Force kill after timeout if process doesn't exit gracefully 406 | const killTimeout = setTimeout(() => { 407 | try { 408 | if (serverInfo.process) { 409 | serverInfo.process.kill('SIGKILL'); 410 | } 411 | } catch (error) { 412 | log(`Error force killing server ${serverName}`, error, { 413 | type: 'error', 414 | }); 415 | } 416 | }, 5000); 417 | 418 | // Try graceful shutdown first 419 | serverInfo.process.kill('SIGTERM'); 420 | 421 | // Wait for process to exit 422 | await new Promise(resolve => { 423 | if (!serverInfo.process) { 424 | clearTimeout(killTimeout); 425 | resolve(); 426 | return; 427 | } 428 | 429 | serverInfo.process.once('exit', () => { 430 | clearTimeout(killTimeout); 431 | resolve(); 432 | }); 433 | }); 434 | } 435 | this.runningServers.delete(serverName); 436 | log(`Server ${serverName} stopped`, undefined, { type: 'info' }); 437 | } catch (error) { 438 | log(`Error stopping server ${serverName}`, error, { type: 'error' }); 439 | // Don't rethrow - allow cleanup to continue 440 | } 441 | } 442 | 443 | public async stopAllServers(): Promise { 444 | const serverNames = Array.from(this.runningServers.keys()); 445 | const stopPromises = serverNames.map(name => 446 | this.stopServer(name).catch(error => { 447 | log(`Error stopping server ${name}`, error, { type: 'error' }); 448 | }) 449 | ); 450 | 451 | try { 452 | // Wait for all servers to stop with timeout 453 | const timeoutPromise = new Promise(resolve => 454 | setTimeout(resolve, 10000) 455 | ); 456 | await Promise.race([Promise.all(stopPromises), timeoutPromise]); 457 | } finally { 458 | // Force kill any remaining processes 459 | for (const [name, server] of this.runningServers.entries()) { 460 | try { 461 | if (server.process && !server.process.killed) { 462 | server.process.kill('SIGKILL'); 463 | log(`Force killed server ${name}`, undefined, { type: 'info' }); 464 | } 465 | } catch (error) { 466 | log(`Error force killing server ${name}`, error, { type: 'error' }); 467 | } 468 | } 469 | // Clear all server references 470 | this.runningServers.clear(); 471 | } 472 | } 473 | 474 | public isServerRunning(serverName: string): boolean { 475 | return this.runningServers.has(serverName); 476 | } 477 | 478 | public getRunningServers(): string[] { 479 | return Array.from(this.runningServers.keys()); 480 | } 481 | } 482 | -------------------------------------------------------------------------------- /src/service.ts: -------------------------------------------------------------------------------- 1 | import { ToolSet } from 'ai'; 2 | import { Client } from '@modelcontextprotocol/sdk/client/index.js'; 3 | import { StdioClientTransport } from '@modelcontextprotocol/sdk/client/stdio.js'; 4 | import { log, jsonSchemaToZod, createDefaultSchema } from './tools'; 5 | import { toStdioParams } from './index'; 6 | import { MCPServerManager, ServerConfig, MCPServersConfig } from './server'; 7 | import { 8 | CallToolResultSchema, 9 | ListToolsResultSchema, 10 | } from '@modelcontextprotocol/sdk/types.js'; 11 | import type { JSONSchema7 } from 'json-schema'; 12 | import * as z from 'zod'; 13 | 14 | export interface MCPToolResult { 15 | content: Array<{ 16 | type: string; 17 | text: string; 18 | }>; 19 | isError?: boolean; 20 | } 21 | 22 | export class MCPService { 23 | private static instance: MCPService; 24 | private clients: Map; 25 | private transports: Map; 26 | private tools: Map; 27 | private initialized: boolean; 28 | private serverManager: MCPServerManager | null; 29 | 30 | private constructor() { 31 | this.clients = new Map(); 32 | this.transports = new Map(); 33 | this.tools = new Map(); 34 | this.initialized = false; 35 | this.serverManager = null; 36 | 37 | // Handle process signals for graceful shutdown 38 | process.on('SIGTERM', this.handleShutdown.bind(this)); 39 | process.on('SIGINT', this.handleShutdown.bind(this)); 40 | } 41 | 42 | public static getInstance(): MCPService { 43 | if (!MCPService.instance) { 44 | MCPService.instance = new MCPService(); 45 | } 46 | return MCPService.instance; 47 | } 48 | 49 | public isInitialized(): boolean { 50 | return this.initialized; 51 | } 52 | 53 | public getConfig(): MCPServersConfig { 54 | if (!this.serverManager) { 55 | throw new Error('Server manager not initialized'); 56 | } 57 | return this.serverManager.getConfig(); 58 | } 59 | 60 | public setServerManager(serverManager: MCPServerManager): void { 61 | this.serverManager = serverManager; 62 | } 63 | 64 | private async initializeServer( 65 | serverName: string, 66 | serverConfig: ServerConfig, 67 | options: { debug?: boolean } = {} 68 | ): Promise { 69 | const { debug = false } = options; 70 | log(`Initializing server ${serverName}`, undefined, { debug }); 71 | 72 | try { 73 | // Start the server with timeout 74 | const startTimeout = new Promise((_, reject) => { 75 | setTimeout(() => { 76 | reject( 77 | new Error(`Server ${serverName} start timeout after 30 seconds`) 78 | ); 79 | }, 30000); 80 | }); 81 | 82 | // Start server with timeout race 83 | const started = await Promise.race([ 84 | this.serverManager!.startServer(serverName), 85 | startTimeout, 86 | ]); 87 | 88 | if (!started) { 89 | throw new Error(`Failed to start server ${serverName}`); 90 | } 91 | 92 | log(`Server ${serverName} started successfully`, undefined, { debug }); 93 | 94 | // Create client if not exists with timeout 95 | if (!this.clients.has(serverName)) { 96 | log(`Creating new client for ${serverName}`, undefined, { debug }); 97 | const client = new Client({ 98 | name: 'mcp-bridge', 99 | version: '1.0.0', 100 | }); 101 | 102 | // Convert config to StdioServerParameters 103 | const stdioParams = toStdioParams(serverConfig); 104 | 105 | // For stdio servers, we need to handle stderr differently 106 | if (serverConfig.mode === 'stdio') { 107 | stdioParams.stderr = 'pipe'; 108 | } 109 | 110 | const transport = new StdioClientTransport(stdioParams); 111 | 112 | // Set up error handling 113 | transport.onerror = (error: Error) => { 114 | // Only log actual errors, ignore ENOENT for npx as it's expected 115 | if (error.message.includes('spawn npx ENOENT')) { 116 | log('Ignoring expected npx ENOENT error', undefined, { debug }); 117 | return; 118 | } 119 | log(`Transport error for ${serverName}`, error, { 120 | type: 'error', 121 | debug, 122 | }); 123 | }; 124 | 125 | // Connect with timeout and retry 126 | const maxRetries = 3; 127 | let retryCount = 0; 128 | let lastError: Error | null = null; 129 | 130 | while (retryCount < maxRetries) { 131 | try { 132 | const connectTimeout = new Promise((_, reject) => { 133 | setTimeout(() => { 134 | reject( 135 | new Error( 136 | `Client connection timeout for ${serverName} after 15 seconds` 137 | ) 138 | ); 139 | }, 15000); 140 | }); 141 | 142 | await Promise.race([client.connect(transport), connectTimeout]); 143 | 144 | // Connection successful 145 | this.clients.set(serverName, client); 146 | this.transports.set(serverName, transport); 147 | log(`Client created and connected for ${serverName}`, undefined, { 148 | debug, 149 | }); 150 | break; 151 | } catch (error) { 152 | lastError = error as Error; 153 | retryCount++; 154 | if (retryCount < maxRetries) { 155 | log( 156 | `Retrying connection for ${serverName} (attempt ${retryCount + 1}/${maxRetries})`, 157 | undefined, 158 | { debug } 159 | ); 160 | await new Promise(resolve => setTimeout(resolve, 2000)); // Wait 2s between retries 161 | } 162 | } 163 | } 164 | 165 | if (retryCount === maxRetries) { 166 | throw new Error( 167 | `Failed to connect to ${serverName} after ${maxRetries} attempts: ${lastError?.message}` 168 | ); 169 | } 170 | } 171 | 172 | // Get tools with timeout and retry 173 | const client = this.clients.get(serverName)!; 174 | log(`Requesting tools from ${serverName}...`, undefined, { debug }); 175 | 176 | const maxToolRetries = 3; 177 | let toolRetryCount = 0; 178 | let lastToolError: Error | null = null; 179 | 180 | while (toolRetryCount < maxToolRetries) { 181 | try { 182 | const toolsTimeout = new Promise((_, reject) => { 183 | setTimeout(() => { 184 | reject( 185 | new Error( 186 | `Tools request timeout for ${serverName} after 10 seconds` 187 | ) 188 | ); 189 | }, 10000); 190 | }); 191 | 192 | const response = (await Promise.race([ 193 | client.request({ method: 'tools/list' }, ListToolsResultSchema), 194 | toolsTimeout, 195 | ])) as z.infer; 196 | 197 | // Store tools in the map with proper AI SDK format 198 | const toolSet: ToolSet = {}; 199 | for (const tool of response.tools) { 200 | toolSet[tool.name] = { 201 | description: tool.description || '', 202 | parameters: tool.inputSchema 203 | ? jsonSchemaToZod(tool.inputSchema as JSONSchema7) 204 | : createDefaultSchema(), 205 | execute: async (args: unknown) => { 206 | log( 207 | `Executing tool ${tool.name}`, 208 | { arguments: args }, 209 | { debug: true } 210 | ); 211 | const result = await client.request( 212 | { 213 | method: 'tools/call', 214 | params: { 215 | name: tool.name, 216 | arguments: args, 217 | }, 218 | }, 219 | CallToolResultSchema 220 | ); 221 | log(`Tool ${tool.name} response:`, result, { debug: true }); 222 | return { 223 | type: 'function', 224 | function: { 225 | name: tool.name, 226 | arguments: JSON.stringify(args), 227 | }, 228 | content: result.content, 229 | }; 230 | }, 231 | }; 232 | } 233 | this.tools.set(serverName, toolSet); 234 | 235 | log(`Received and processed tools from ${serverName}:`, response, { 236 | debug, 237 | }); 238 | break; 239 | } catch (error) { 240 | lastToolError = error as Error; 241 | toolRetryCount++; 242 | if (toolRetryCount < maxToolRetries) { 243 | log( 244 | `Retrying tools request for ${serverName} (attempt ${toolRetryCount + 1}/${maxToolRetries})`, 245 | undefined, 246 | { debug } 247 | ); 248 | await new Promise(resolve => setTimeout(resolve, 2000)); // Wait 2s between retries 249 | } 250 | } 251 | } 252 | 253 | if (toolRetryCount === maxToolRetries) { 254 | throw new Error( 255 | `Failed to get tools from ${serverName} after ${maxToolRetries} attempts: ${lastToolError?.message}` 256 | ); 257 | } 258 | } catch (error) { 259 | // Clean up resources on initialization failure 260 | await this.cleanup(); 261 | throw error; 262 | } 263 | } 264 | 265 | public async initialize(options: { debug?: boolean } = {}): Promise { 266 | if (this.initialized) { 267 | log('MCP service already initialized', undefined, { 268 | debug: options.debug, 269 | }); 270 | return; 271 | } 272 | 273 | const { debug = false } = options; 274 | const config = this.serverManager!.getConfig(); 275 | 276 | if (!config) { 277 | throw new Error('No MCP server configuration found'); 278 | } 279 | 280 | const initializationErrors: Record = {}; 281 | let anyServerInitialized = false; 282 | 283 | for (const [serverName, serverConfig] of Object.entries( 284 | config.mcpServers 285 | )) { 286 | if (serverConfig.disabled) { 287 | log(`Skipping disabled server: ${serverName}`, undefined, { debug }); 288 | continue; 289 | } 290 | 291 | try { 292 | await this.initializeServer(serverName, serverConfig, { debug }); 293 | anyServerInitialized = true; 294 | } catch (error) { 295 | log(`Failed to initialize server ${serverName}`, error, { 296 | type: 'error', 297 | }); 298 | initializationErrors[serverName] = error as Error; 299 | } 300 | } 301 | 302 | // If all servers failed to initialize, throw error 303 | if (!anyServerInitialized) { 304 | const errorMessage = Object.entries(initializationErrors) 305 | .map(([name, error]) => `${name}: ${error.message}`) 306 | .join('\n'); 307 | throw new Error(`All servers failed to initialize:\n${errorMessage}`); 308 | } 309 | 310 | this.initialized = true; 311 | log('MCP service initialization complete', undefined, { debug }); 312 | } 313 | 314 | public async cleanup(): Promise { 315 | log('Cleaning up MCP service', undefined, { type: 'info' }); 316 | 317 | // First stop all servers 318 | if (this.serverManager) { 319 | await this.serverManager.stopAllServers(); 320 | } 321 | 322 | // Clean up all clients and transports 323 | for (const [serverName, client] of this.clients.entries()) { 324 | try { 325 | await client.close(); 326 | this.clients.delete(serverName); 327 | this.transports.delete(serverName); 328 | this.tools.delete(serverName); 329 | } catch (error) { 330 | log(`Error cleaning up server ${serverName}`, error, { 331 | type: 'error', 332 | }); 333 | } 334 | } 335 | 336 | this.initialized = false; 337 | log('MCP service cleanup complete', undefined, { type: 'info' }); 338 | } 339 | 340 | public async getTools( 341 | options: { debug?: boolean; serverName?: string } = {} 342 | ): Promise { 343 | const { debug = false, serverName } = options; 344 | 345 | if (!this.initialized) { 346 | log('MCP service not initialized, initializing now...', undefined, { 347 | debug, 348 | }); 349 | await this.initialize({ debug }); 350 | } 351 | 352 | const allTools: ToolSet = {}; 353 | const runningServers = serverName 354 | ? [serverName] 355 | : this.serverManager!.getRunningServers(); 356 | 357 | if (serverName && !this.serverManager!.isServerRunning(serverName)) { 358 | throw new Error(`Server ${serverName} is not running`); 359 | } 360 | 361 | log( 362 | `Found ${runningServers.length} running servers: ${runningServers.join(', ')}`, 363 | undefined, 364 | { debug } 365 | ); 366 | 367 | for (const serverName of runningServers) { 368 | try { 369 | const tools = this.tools.get(serverName); 370 | if (!tools) { 371 | log( 372 | `No tools found for server ${serverName}, requesting...`, 373 | undefined, 374 | { debug } 375 | ); 376 | await this.initializeServer( 377 | serverName, 378 | this.serverManager!.getServerInfo(serverName)!, 379 | { debug } 380 | ); 381 | } 382 | Object.assign(allTools, this.tools.get(serverName) || {}); 383 | } catch (error) { 384 | log(`Failed to get tools for server ${serverName}`, error, { 385 | type: 'error', 386 | }); 387 | } 388 | } 389 | 390 | return allTools; 391 | } 392 | 393 | public async executeFunction( 394 | serverName: string, 395 | functionName: string, 396 | args: Record 397 | ): Promise { 398 | const client = this.clients.get(serverName); 399 | if (!client) { 400 | throw new Error(`No client found for server: ${serverName}`); 401 | } 402 | 403 | try { 404 | log( 405 | `Executing function ${functionName} on server ${serverName}`, 406 | { arguments: args }, 407 | { debug: true } 408 | ); 409 | const response = await client.request( 410 | { 411 | method: 'tools/call', 412 | params: { 413 | name: functionName, 414 | arguments: args, 415 | }, 416 | }, 417 | CallToolResultSchema 418 | ); 419 | log(`Function ${functionName} response:`, response, { debug: true }); 420 | return response as MCPToolResult; 421 | } catch (error) { 422 | log(`Failed to execute function ${functionName}`, error, { 423 | type: 'error', 424 | }); 425 | throw error; 426 | } 427 | } 428 | 429 | private async handleShutdown(signal: string): Promise { 430 | log(`${signal} received. Shutting down gracefully...`, undefined, { 431 | type: 'info', 432 | }); 433 | try { 434 | await this.cleanup(); 435 | process.exit(0); 436 | } catch (error) { 437 | log('Error during shutdown:', error, { type: 'error' }); 438 | process.exit(1); 439 | } 440 | } 441 | } 442 | 443 | // Export singleton instance 444 | export const mcpService = MCPService.getInstance(); 445 | -------------------------------------------------------------------------------- /src/tools.ts: -------------------------------------------------------------------------------- 1 | import { z } from 'zod'; 2 | import type { 3 | JSONSchema7, 4 | JSONSchema7Definition, 5 | JSONSchema7TypeName, 6 | } from 'json-schema'; 7 | import * as fs from 'fs'; 8 | import * as path from 'path'; 9 | 10 | // Define server tool type with proper schema types 11 | export interface ServerTool { 12 | name: string; 13 | description?: string; 14 | inputSchema?: JSONSchema7; 15 | outputSchema?: JSONSchema7; 16 | } 17 | 18 | // Add logging utility 19 | export interface LogOptions { 20 | debug?: boolean; 21 | type?: 'info' | 'error' | 'debug'; 22 | skipFile?: boolean; 23 | skipConsole?: boolean; 24 | } 25 | 26 | export function log( 27 | message: string, 28 | data?: unknown, 29 | options: LogOptions = {} 30 | ): void { 31 | // Check DEBUG environment variable and debug namespace 32 | const debugEnabled = 33 | process.env.DEBUG && 34 | (process.env.DEBUG === '*' || 35 | process.env.DEBUG.split(',').some( 36 | ns => ns === 'mcp' || ns === 'mcp:*' || ns === '*' 37 | )); 38 | 39 | // Early return if debug is not enabled and not an error 40 | if (!debugEnabled && !options.debug && options.type !== 'error') { 41 | return; 42 | } 43 | 44 | const timestamp = new Date().toISOString(); 45 | const logType = options.type || 'info'; 46 | const logMessage = `[${timestamp}] [${logType}] ${message}${ 47 | data ? '\n' + JSON.stringify(data, null, 2) : '' 48 | }`; 49 | 50 | // Write to file ONLY if DEBUG is enabled or if it's an error 51 | if (!options.skipFile && (debugEnabled || options.type === 'error')) { 52 | const logDir = path.join(process.cwd(), 'logs'); 53 | if (!fs.existsSync(logDir)) { 54 | fs.mkdirSync(logDir, { recursive: true }); 55 | } 56 | fs.appendFileSync(path.join(logDir, 'mcp-tools.log'), logMessage + '\n'); 57 | } 58 | 59 | // Console log ONLY if DEBUG is enabled or if it's an error 60 | if (!options.skipConsole && (debugEnabled || options.type === 'error')) { 61 | if (options.type === 'error') { 62 | console.error(logMessage); 63 | } else if (options.type === 'debug') { 64 | console.debug(logMessage); 65 | } else { 66 | console.log(logMessage); 67 | } 68 | } 69 | } 70 | 71 | // Helper function to create a default schema that accepts any valid JSON 72 | export function createDefaultSchema(): z.ZodTypeAny { 73 | return z.any(); 74 | } 75 | 76 | // Helper function to convert JSON Schema to Zod schema 77 | export function jsonSchemaToZod( 78 | schema: JSONSchema7Definition | undefined, 79 | debug = false 80 | ): z.ZodTypeAny { 81 | if (!schema || typeof schema === 'boolean') { 82 | return createDefaultSchema(); 83 | } 84 | 85 | // Only log if debug is explicitly true and it's a top-level schema 86 | if (debug && schema.title) { 87 | log('Converting schema', schema, { debug, type: 'debug' }); 88 | } 89 | 90 | try { 91 | // Handle oneOf/anyOf schemas by converting them to a proper type 92 | if (schema.oneOf || schema.anyOf) { 93 | const subSchemas = (schema.oneOf || schema.anyOf || []) 94 | .map(s => { 95 | if (typeof s === 'boolean') return undefined; 96 | return jsonSchemaToZod(s, debug); 97 | }) 98 | .filter((s): s is z.ZodType => s !== undefined); 99 | 100 | if (subSchemas.length === 0) { 101 | // If no valid subschemas, create a type based on the parent schema type 102 | if (schema.type) { 103 | return createSchemaForType(schema as JSONSchema7, debug); 104 | } 105 | return createDefaultSchema(); 106 | } 107 | if (subSchemas.length === 1) return subSchemas[0]; 108 | 109 | // Create a proper union type with all schemas 110 | return z.union( 111 | subSchemas as [z.ZodTypeAny, z.ZodTypeAny, ...z.ZodTypeAny[]] 112 | ); 113 | } 114 | 115 | return createSchemaForType(schema as JSONSchema7, debug); 116 | } catch (error) { 117 | // Only log error if debug is true 118 | if (debug) { 119 | log('Error converting schema', { error, schema }, { type: 'error' }); 120 | } 121 | return createDefaultSchema(); 122 | } 123 | } 124 | 125 | // Helper function to create schema based on type 126 | function createSchemaForType(schema: JSONSchema7, debug = false): z.ZodTypeAny { 127 | const schemaType = schema.type as JSONSchema7TypeName | undefined; 128 | 129 | // Only log debug info if debug is true and it's a non-standard type 130 | if ( 131 | debug && 132 | (!schemaType || 133 | !['string', 'number', 'boolean', 'object', 'array'].includes(schemaType)) 134 | ) { 135 | log(`Using default schema for type: ${schemaType}`, schema, { 136 | debug, 137 | type: 'debug', 138 | }); 139 | } 140 | 141 | switch (schemaType) { 142 | case 'string': { 143 | if (schema.enum && Array.isArray(schema.enum) && schema.enum.length > 0) { 144 | return z.enum(schema.enum as [string, ...string[]]); 145 | } 146 | return z.string(); 147 | } 148 | 149 | case 'number': 150 | case 'integer': { 151 | return z.number(); 152 | } 153 | 154 | case 'boolean': { 155 | return z.boolean(); 156 | } 157 | 158 | case 'object': { 159 | if (!schema.properties) { 160 | return z.record(z.any()); 161 | } 162 | 163 | const shape: Record = {}; 164 | const required = new Set(schema.required || []); 165 | 166 | for (const [key, value] of Object.entries(schema.properties)) { 167 | try { 168 | // Special handling for webhook property 169 | if (key === 'webhook') { 170 | // Instead of using union, use object type with optional headers 171 | shape[key] = z.object({ 172 | url: z.string(), 173 | headers: z.record(z.string()).optional(), 174 | }); 175 | continue; 176 | } 177 | 178 | const propertySchema = jsonSchemaToZod(value, debug); 179 | shape[key] = required.has(key) 180 | ? propertySchema 181 | : propertySchema.optional(); 182 | } catch (error) { 183 | log(`Error converting property ${key}`, error, { 184 | type: 'error', 185 | debug, 186 | }); 187 | shape[key] = z.any(); 188 | } 189 | } 190 | 191 | return z.object(shape).passthrough(); 192 | } 193 | 194 | case 'array': { 195 | const itemSchema = schema.items 196 | ? Array.isArray(schema.items) 197 | ? z.union( 198 | schema.items.map(item => jsonSchemaToZod(item, debug)) as [ 199 | z.ZodTypeAny, 200 | z.ZodTypeAny, 201 | ...z.ZodTypeAny[], 202 | ] 203 | ) 204 | : jsonSchemaToZod(schema.items, debug) 205 | : z.any(); 206 | 207 | return z.array(itemSchema); 208 | } 209 | 210 | case 'null': { 211 | return z.null(); 212 | } 213 | 214 | default: { 215 | log(`Using default schema for type: ${schemaType}`, schema, { 216 | debug, 217 | type: 'debug', 218 | }); 219 | return z.any(); 220 | } 221 | } 222 | } 223 | -------------------------------------------------------------------------------- /tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "target": "es2020", 4 | "module": "commonjs", 5 | "declaration": true, 6 | "declarationMap": true, 7 | "sourceMap": true, 8 | "outDir": "./dist", 9 | "rootDir": ".", 10 | "strict": true, 11 | "moduleResolution": "node", 12 | "esModuleInterop": true, 13 | "skipLibCheck": true, 14 | "forceConsistentCasingInFileNames": true, 15 | "resolveJsonModule": true 16 | }, 17 | "include": ["src/**/*", "examples/**/*"], 18 | "exclude": ["node_modules", "dist", "**/*.test.ts"] 19 | } 20 | --------------------------------------------------------------------------------