├── .gitignore ├── README.md ├── package-lock.json ├── package.json ├── src ├── consts.ts ├── index.ts ├── server.ts ├── types.ts └── weibo.ts └── tsconfig.json /.gitignore: -------------------------------------------------------------------------------- 1 | # Logs 2 | logs 3 | *.log 4 | npm-debug.log* 5 | yarn-debug.log* 6 | yarn-error.log* 7 | lerna-debug.log* 8 | 9 | # Dependency directories 10 | node_modules/ 11 | jspm_packages/ 12 | 13 | # TypeScript cache 14 | *.tsbuildinfo 15 | 16 | # Optional npm cache directory 17 | .npm 18 | 19 | # Optional eslint cache 20 | .eslintcache 21 | 22 | # Output of 'npm pack' 23 | *.tgz 24 | 25 | # dotenv environment variables file 26 | .env 27 | .env.test 28 | 29 | # Build output 30 | dist/ 31 | 32 | # IDE 33 | .idea/ 34 | .vscode/ 35 | *.swp 36 | *.swo -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Weibo MCP Server (TypeScript Version) 2 | 3 | This is a server based on the [Model Context Protocol](https://modelcontextprotocol.io) for scraping Weibo user information, feeds, and search functionality. This server can help retrieve detailed information about Weibo users, feed content, and perform user searches. 4 | 5 | 6 | Weibo Server MCP server 7 | 8 | 9 | ## Installation 10 | 11 | Install from source code: 12 | 13 | ```json 14 | { 15 | "mcpServers": { 16 | "weibo": { 17 | "command": "npx", 18 | "args": [ 19 | "--from", 20 | "git+https://github.com/Selenium39/mcp-server-weibo.git", 21 | "mcp-server-weibo" 22 | ] 23 | } 24 | } 25 | } 26 | ``` 27 | 28 | Install from package manager: 29 | 30 | ```json 31 | { 32 | "mcpServers": { 33 | "weibo": { 34 | "command": "npx", 35 | "args": ["mcp-server-weibo"], 36 | } 37 | } 38 | } 39 | ``` 40 | 41 | ## Components 42 | 43 | ### Tools 44 | 45 | - `search_users(keyword, limit)`: 根据关键词搜索微博用户 46 | - `get_profile(uid)`: 获取用户详细资料信息 47 | - `get_feeds(uid, limit)`: 获取用户微博动态 48 | - `get_hot_search(limit)`: 获取微博热搜榜 49 | - `search_content(keyword, limit, page?)`: 根据关键词搜索微博内容 50 | 51 | ### Resources 52 | 53 | None 54 | 55 | ### Prompts 56 | 57 | None 58 | 59 | ## Requirements 60 | 61 | - Node.js >= 18.0.0 62 | 63 | ## License 64 | 65 | MIT License 66 | 67 | ## Disclaimer 68 | 69 | This project is not affiliated with Weibo and is for learning and research purposes only. -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "mcp-server-weibo", 3 | "version": "1.0.0", 4 | "lockfileVersion": 3, 5 | "requires": true, 6 | "packages": { 7 | "": { 8 | "name": "mcp-server-weibo", 9 | "version": "1.0.0", 10 | "license": "MIT", 11 | "dependencies": { 12 | "@modelcontextprotocol/sdk": "^1.7.0", 13 | "axios": "^1.8.4", 14 | "zod": "^3.22.4" 15 | }, 16 | "devDependencies": { 17 | "@types/node": "^20.10.6", 18 | "ts-node": "^10.9.2", 19 | "typescript": "^5.3.3" 20 | }, 21 | "engines": { 22 | "node": ">=18.0.0" 23 | } 24 | }, 25 | "node_modules/@cspotcode/source-map-support": { 26 | "version": "0.8.1", 27 | "resolved": "https://registry.npmjs.org/@cspotcode/source-map-support/-/source-map-support-0.8.1.tgz", 28 | "integrity": "sha512-IchNf6dN4tHoMFIn/7OE8LWZ19Y6q/67Bmf6vnGREv8RSbBVb9LPJxEcnwrcwX6ixSvaiGoomAUvu4YSxXrVgw==", 29 | "dev": true, 30 | "dependencies": { 31 | "@jridgewell/trace-mapping": "0.3.9" 32 | }, 33 | "engines": { 34 | "node": ">=12" 35 | } 36 | }, 37 | "node_modules/@jridgewell/resolve-uri": { 38 | "version": "3.1.2", 39 | "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz", 40 | "integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==", 41 | "dev": true, 42 | "engines": { 43 | "node": ">=6.0.0" 44 | } 45 | }, 46 | "node_modules/@jridgewell/sourcemap-codec": { 47 | "version": "1.5.0", 48 | "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.0.tgz", 49 | "integrity": "sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ==", 50 | "dev": true 51 | }, 52 | "node_modules/@jridgewell/trace-mapping": { 53 | "version": "0.3.9", 54 | "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.9.tgz", 55 | "integrity": "sha512-3Belt6tdc8bPgAtbcmdtNJlirVoTmEb5e2gC94PnkwEW9jI6CAHUeoG85tjWP5WquqfavoMtMwiG4P926ZKKuQ==", 56 | "dev": true, 57 | "dependencies": { 58 | "@jridgewell/resolve-uri": "^3.0.3", 59 | "@jridgewell/sourcemap-codec": "^1.4.10" 60 | } 61 | }, 62 | "node_modules/@modelcontextprotocol/sdk": { 63 | "version": "1.7.0", 64 | "resolved": "https://registry.npmjs.org/@modelcontextprotocol/sdk/-/sdk-1.7.0.tgz", 65 | "integrity": "sha512-IYPe/FLpvF3IZrd/f5p5ffmWhMc3aEMuM2wGJASDqC2Ge7qatVCdbfPx3n/5xFeb19xN0j/911M2AaFuircsWA==", 66 | "dependencies": { 67 | "content-type": "^1.0.5", 68 | "cors": "^2.8.5", 69 | "eventsource": "^3.0.2", 70 | "express": "^5.0.1", 71 | "express-rate-limit": "^7.5.0", 72 | "pkce-challenge": "^4.1.0", 73 | "raw-body": "^3.0.0", 74 | "zod": "^3.23.8", 75 | "zod-to-json-schema": "^3.24.1" 76 | }, 77 | "engines": { 78 | "node": ">=18" 79 | } 80 | }, 81 | "node_modules/@tsconfig/node10": { 82 | "version": "1.0.11", 83 | "resolved": "https://registry.npmjs.org/@tsconfig/node10/-/node10-1.0.11.tgz", 84 | "integrity": "sha512-DcRjDCujK/kCk/cUe8Xz8ZSpm8mS3mNNpta+jGCA6USEDfktlNvm1+IuZ9eTcDbNk41BHwpHHeW+N1lKCz4zOw==", 85 | "dev": true 86 | }, 87 | "node_modules/@tsconfig/node12": { 88 | "version": "1.0.11", 89 | "resolved": "https://registry.npmjs.org/@tsconfig/node12/-/node12-1.0.11.tgz", 90 | "integrity": "sha512-cqefuRsh12pWyGsIoBKJA9luFu3mRxCA+ORZvA4ktLSzIuCUtWVxGIuXigEwO5/ywWFMZ2QEGKWvkZG1zDMTag==", 91 | "dev": true 92 | }, 93 | "node_modules/@tsconfig/node14": { 94 | "version": "1.0.3", 95 | "resolved": "https://registry.npmjs.org/@tsconfig/node14/-/node14-1.0.3.tgz", 96 | "integrity": "sha512-ysT8mhdixWK6Hw3i1V2AeRqZ5WfXg1G43mqoYlM2nc6388Fq5jcXyr5mRsqViLx/GJYdoL0bfXD8nmF+Zn/Iow==", 97 | "dev": true 98 | }, 99 | "node_modules/@tsconfig/node16": { 100 | "version": "1.0.4", 101 | "resolved": "https://registry.npmjs.org/@tsconfig/node16/-/node16-1.0.4.tgz", 102 | "integrity": "sha512-vxhUy4J8lyeyinH7Azl1pdd43GJhZH/tP2weN8TntQblOY+A0XbT8DJk1/oCPuOOyg/Ja757rG0CgHcWC8OfMA==", 103 | "dev": true 104 | }, 105 | "node_modules/@types/node": { 106 | "version": "20.17.27", 107 | "resolved": "https://registry.npmjs.org/@types/node/-/node-20.17.27.tgz", 108 | "integrity": "sha512-U58sbKhDrthHlxHRJw7ZLiLDZGmAUOZUbpw0S6nL27sYUdhvgBLCRu/keSd6qcTsfArd1sRFCCBxzWATGr/0UA==", 109 | "dev": true, 110 | "dependencies": { 111 | "undici-types": "~6.19.2" 112 | } 113 | }, 114 | "node_modules/accepts": { 115 | "version": "2.0.0", 116 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-2.0.0.tgz", 117 | "integrity": "sha512-5cvg6CtKwfgdmVqY1WIiXKc3Q1bkRqGLi+2W/6ao+6Y7gu/RCwRuAhGEzh5B4KlszSuTLgZYuqFqo5bImjNKng==", 118 | "dependencies": { 119 | "mime-types": "^3.0.0", 120 | "negotiator": "^1.0.0" 121 | }, 122 | "engines": { 123 | "node": ">= 0.6" 124 | } 125 | }, 126 | "node_modules/acorn": { 127 | "version": "8.14.1", 128 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.14.1.tgz", 129 | "integrity": "sha512-OvQ/2pUDKmgfCg++xsTX1wGxfTaszcHVcTctW4UJB4hibJx2HXxxO5UmVgyjMa+ZDsiaf5wWLXYpRWMmBI0QHg==", 130 | "dev": true, 131 | "bin": { 132 | "acorn": "bin/acorn" 133 | }, 134 | "engines": { 135 | "node": ">=0.4.0" 136 | } 137 | }, 138 | "node_modules/acorn-walk": { 139 | "version": "8.3.4", 140 | "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-8.3.4.tgz", 141 | "integrity": "sha512-ueEepnujpqee2o5aIYnvHU6C0A42MNdsIDeqy5BydrkuC5R1ZuUFnm27EeFJGoEHJQgn3uleRvmTXaJgfXbt4g==", 142 | "dev": true, 143 | "dependencies": { 144 | "acorn": "^8.11.0" 145 | }, 146 | "engines": { 147 | "node": ">=0.4.0" 148 | } 149 | }, 150 | "node_modules/arg": { 151 | "version": "4.1.3", 152 | "resolved": "https://registry.npmjs.org/arg/-/arg-4.1.3.tgz", 153 | "integrity": "sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==", 154 | "dev": true 155 | }, 156 | "node_modules/asynckit": { 157 | "version": "0.4.0", 158 | "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", 159 | "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==" 160 | }, 161 | "node_modules/axios": { 162 | "version": "1.8.4", 163 | "resolved": "https://registry.npmjs.org/axios/-/axios-1.8.4.tgz", 164 | "integrity": "sha512-eBSYY4Y68NNlHbHBMdeDmKNtDgXWhQsJcGqzO3iLUM0GraQFSS9cVgPX5I9b3lbdFKyYoAEGAZF1DwhTaljNAw==", 165 | "dependencies": { 166 | "follow-redirects": "^1.15.6", 167 | "form-data": "^4.0.0", 168 | "proxy-from-env": "^1.1.0" 169 | } 170 | }, 171 | "node_modules/body-parser": { 172 | "version": "2.1.0", 173 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-2.1.0.tgz", 174 | "integrity": "sha512-/hPxh61E+ll0Ujp24Ilm64cykicul1ypfwjVttduAiEdtnJFvLePSrIPk+HMImtNv5270wOGCb1Tns2rybMkoQ==", 175 | "dependencies": { 176 | "bytes": "^3.1.2", 177 | "content-type": "^1.0.5", 178 | "debug": "^4.4.0", 179 | "http-errors": "^2.0.0", 180 | "iconv-lite": "^0.5.2", 181 | "on-finished": "^2.4.1", 182 | "qs": "^6.14.0", 183 | "raw-body": "^3.0.0", 184 | "type-is": "^2.0.0" 185 | }, 186 | "engines": { 187 | "node": ">=18" 188 | } 189 | }, 190 | "node_modules/body-parser/node_modules/debug": { 191 | "version": "4.4.0", 192 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.0.tgz", 193 | "integrity": "sha512-6WTZ/IxCY/T6BALoZHaE4ctp9xm+Z5kY/pzYaCHRFeyVhojxlrm+46y68HA6hr0TcwEssoxNiDEUJQjfPZ/RYA==", 194 | "dependencies": { 195 | "ms": "^2.1.3" 196 | }, 197 | "engines": { 198 | "node": ">=6.0" 199 | }, 200 | "peerDependenciesMeta": { 201 | "supports-color": { 202 | "optional": true 203 | } 204 | } 205 | }, 206 | "node_modules/body-parser/node_modules/ms": { 207 | "version": "2.1.3", 208 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 209 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==" 210 | }, 211 | "node_modules/body-parser/node_modules/qs": { 212 | "version": "6.14.0", 213 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.14.0.tgz", 214 | "integrity": "sha512-YWWTjgABSKcvs/nWBi9PycY/JiPJqOD4JA6o9Sej2AtvSGarXxKC3OQSk4pAarbdQlKAh5D4FCQkJNkW+GAn3w==", 215 | "dependencies": { 216 | "side-channel": "^1.1.0" 217 | }, 218 | "engines": { 219 | "node": ">=0.6" 220 | }, 221 | "funding": { 222 | "url": "https://github.com/sponsors/ljharb" 223 | } 224 | }, 225 | "node_modules/bytes": { 226 | "version": "3.1.2", 227 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz", 228 | "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==", 229 | "engines": { 230 | "node": ">= 0.8" 231 | } 232 | }, 233 | "node_modules/call-bind-apply-helpers": { 234 | "version": "1.0.2", 235 | "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.2.tgz", 236 | "integrity": "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==", 237 | "dependencies": { 238 | "es-errors": "^1.3.0", 239 | "function-bind": "^1.1.2" 240 | }, 241 | "engines": { 242 | "node": ">= 0.4" 243 | } 244 | }, 245 | "node_modules/call-bound": { 246 | "version": "1.0.4", 247 | "resolved": "https://registry.npmjs.org/call-bound/-/call-bound-1.0.4.tgz", 248 | "integrity": "sha512-+ys997U96po4Kx/ABpBCqhA9EuxJaQWDQg7295H4hBphv3IZg0boBKuwYpt4YXp6MZ5AmZQnU/tyMTlRpaSejg==", 249 | "dependencies": { 250 | "call-bind-apply-helpers": "^1.0.2", 251 | "get-intrinsic": "^1.3.0" 252 | }, 253 | "engines": { 254 | "node": ">= 0.4" 255 | }, 256 | "funding": { 257 | "url": "https://github.com/sponsors/ljharb" 258 | } 259 | }, 260 | "node_modules/combined-stream": { 261 | "version": "1.0.8", 262 | "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", 263 | "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", 264 | "dependencies": { 265 | "delayed-stream": "~1.0.0" 266 | }, 267 | "engines": { 268 | "node": ">= 0.8" 269 | } 270 | }, 271 | "node_modules/content-disposition": { 272 | "version": "1.0.0", 273 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-1.0.0.tgz", 274 | "integrity": "sha512-Au9nRL8VNUut/XSzbQA38+M78dzP4D+eqg3gfJHMIHHYa3bg067xj1KxMUWj+VULbiZMowKngFFbKczUrNJ1mg==", 275 | "dependencies": { 276 | "safe-buffer": "5.2.1" 277 | }, 278 | "engines": { 279 | "node": ">= 0.6" 280 | } 281 | }, 282 | "node_modules/content-type": { 283 | "version": "1.0.5", 284 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.5.tgz", 285 | "integrity": "sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==", 286 | "engines": { 287 | "node": ">= 0.6" 288 | } 289 | }, 290 | "node_modules/cookie": { 291 | "version": "0.7.1", 292 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.7.1.tgz", 293 | "integrity": "sha512-6DnInpx7SJ2AK3+CTUE/ZM0vWTUboZCegxhC2xiIydHR9jNuTAASBrfEpHhiGOZw/nX51bHt6YQl8jsGo4y/0w==", 294 | "engines": { 295 | "node": ">= 0.6" 296 | } 297 | }, 298 | "node_modules/cookie-signature": { 299 | "version": "1.2.2", 300 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.2.2.tgz", 301 | "integrity": "sha512-D76uU73ulSXrD1UXF4KE2TMxVVwhsnCgfAyTg9k8P6KGZjlXKrOLe4dJQKI3Bxi5wjesZoFXJWElNWBjPZMbhg==", 302 | "engines": { 303 | "node": ">=6.6.0" 304 | } 305 | }, 306 | "node_modules/cors": { 307 | "version": "2.8.5", 308 | "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz", 309 | "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==", 310 | "dependencies": { 311 | "object-assign": "^4", 312 | "vary": "^1" 313 | }, 314 | "engines": { 315 | "node": ">= 0.10" 316 | } 317 | }, 318 | "node_modules/create-require": { 319 | "version": "1.1.1", 320 | "resolved": "https://registry.npmjs.org/create-require/-/create-require-1.1.1.tgz", 321 | "integrity": "sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ==", 322 | "dev": true 323 | }, 324 | "node_modules/debug": { 325 | "version": "4.3.6", 326 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.6.tgz", 327 | "integrity": "sha512-O/09Bd4Z1fBrU4VzkhFqVgpPzaGbw6Sm9FEkBT1A/YBXQFGuuSxa1dN2nxgxS34JmKXqYx8CZAwEVoJFImUXIg==", 328 | "dependencies": { 329 | "ms": "2.1.2" 330 | }, 331 | "engines": { 332 | "node": ">=6.0" 333 | }, 334 | "peerDependenciesMeta": { 335 | "supports-color": { 336 | "optional": true 337 | } 338 | } 339 | }, 340 | "node_modules/delayed-stream": { 341 | "version": "1.0.0", 342 | "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", 343 | "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==", 344 | "engines": { 345 | "node": ">=0.4.0" 346 | } 347 | }, 348 | "node_modules/depd": { 349 | "version": "2.0.0", 350 | "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz", 351 | "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==", 352 | "engines": { 353 | "node": ">= 0.8" 354 | } 355 | }, 356 | "node_modules/destroy": { 357 | "version": "1.2.0", 358 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.2.0.tgz", 359 | "integrity": "sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg==", 360 | "engines": { 361 | "node": ">= 0.8", 362 | "npm": "1.2.8000 || >= 1.4.16" 363 | } 364 | }, 365 | "node_modules/diff": { 366 | "version": "4.0.2", 367 | "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz", 368 | "integrity": "sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==", 369 | "dev": true, 370 | "engines": { 371 | "node": ">=0.3.1" 372 | } 373 | }, 374 | "node_modules/dunder-proto": { 375 | "version": "1.0.1", 376 | "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz", 377 | "integrity": "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==", 378 | "dependencies": { 379 | "call-bind-apply-helpers": "^1.0.1", 380 | "es-errors": "^1.3.0", 381 | "gopd": "^1.2.0" 382 | }, 383 | "engines": { 384 | "node": ">= 0.4" 385 | } 386 | }, 387 | "node_modules/ee-first": { 388 | "version": "1.1.1", 389 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", 390 | "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==" 391 | }, 392 | "node_modules/encodeurl": { 393 | "version": "2.0.0", 394 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-2.0.0.tgz", 395 | "integrity": "sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg==", 396 | "engines": { 397 | "node": ">= 0.8" 398 | } 399 | }, 400 | "node_modules/es-define-property": { 401 | "version": "1.0.1", 402 | "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz", 403 | "integrity": "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==", 404 | "engines": { 405 | "node": ">= 0.4" 406 | } 407 | }, 408 | "node_modules/es-errors": { 409 | "version": "1.3.0", 410 | "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz", 411 | "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==", 412 | "engines": { 413 | "node": ">= 0.4" 414 | } 415 | }, 416 | "node_modules/es-object-atoms": { 417 | "version": "1.1.1", 418 | "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.1.1.tgz", 419 | "integrity": "sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA==", 420 | "dependencies": { 421 | "es-errors": "^1.3.0" 422 | }, 423 | "engines": { 424 | "node": ">= 0.4" 425 | } 426 | }, 427 | "node_modules/es-set-tostringtag": { 428 | "version": "2.1.0", 429 | "resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.1.0.tgz", 430 | "integrity": "sha512-j6vWzfrGVfyXxge+O0x5sh6cvxAog0a/4Rdd2K36zCMV5eJ+/+tOAngRO8cODMNWbVRdVlmGZQL2YS3yR8bIUA==", 431 | "dependencies": { 432 | "es-errors": "^1.3.0", 433 | "get-intrinsic": "^1.2.6", 434 | "has-tostringtag": "^1.0.2", 435 | "hasown": "^2.0.2" 436 | }, 437 | "engines": { 438 | "node": ">= 0.4" 439 | } 440 | }, 441 | "node_modules/escape-html": { 442 | "version": "1.0.3", 443 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", 444 | "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==" 445 | }, 446 | "node_modules/etag": { 447 | "version": "1.8.1", 448 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", 449 | "integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==", 450 | "engines": { 451 | "node": ">= 0.6" 452 | } 453 | }, 454 | "node_modules/eventsource": { 455 | "version": "3.0.5", 456 | "resolved": "https://registry.npmjs.org/eventsource/-/eventsource-3.0.5.tgz", 457 | "integrity": "sha512-LT/5J605bx5SNyE+ITBDiM3FxffBiq9un7Vx0EwMDM3vg8sWKx/tO2zC+LMqZ+smAM0F2hblaDZUVZF0te2pSw==", 458 | "dependencies": { 459 | "eventsource-parser": "^3.0.0" 460 | }, 461 | "engines": { 462 | "node": ">=18.0.0" 463 | } 464 | }, 465 | "node_modules/eventsource-parser": { 466 | "version": "3.0.0", 467 | "resolved": "https://registry.npmjs.org/eventsource-parser/-/eventsource-parser-3.0.0.tgz", 468 | "integrity": "sha512-T1C0XCUimhxVQzW4zFipdx0SficT651NnkR0ZSH3yQwh+mFMdLfgjABVi4YtMTtaL4s168593DaoaRLMqryavA==", 469 | "engines": { 470 | "node": ">=18.0.0" 471 | } 472 | }, 473 | "node_modules/express": { 474 | "version": "5.0.1", 475 | "resolved": "https://registry.npmjs.org/express/-/express-5.0.1.tgz", 476 | "integrity": "sha512-ORF7g6qGnD+YtUG9yx4DFoqCShNMmUKiXuT5oWMHiOvt/4WFbHC6yCwQMTSBMno7AqntNCAzzcnnjowRkTL9eQ==", 477 | "dependencies": { 478 | "accepts": "^2.0.0", 479 | "body-parser": "^2.0.1", 480 | "content-disposition": "^1.0.0", 481 | "content-type": "~1.0.4", 482 | "cookie": "0.7.1", 483 | "cookie-signature": "^1.2.1", 484 | "debug": "4.3.6", 485 | "depd": "2.0.0", 486 | "encodeurl": "~2.0.0", 487 | "escape-html": "~1.0.3", 488 | "etag": "~1.8.1", 489 | "finalhandler": "^2.0.0", 490 | "fresh": "2.0.0", 491 | "http-errors": "2.0.0", 492 | "merge-descriptors": "^2.0.0", 493 | "methods": "~1.1.2", 494 | "mime-types": "^3.0.0", 495 | "on-finished": "2.4.1", 496 | "once": "1.4.0", 497 | "parseurl": "~1.3.3", 498 | "proxy-addr": "~2.0.7", 499 | "qs": "6.13.0", 500 | "range-parser": "~1.2.1", 501 | "router": "^2.0.0", 502 | "safe-buffer": "5.2.1", 503 | "send": "^1.1.0", 504 | "serve-static": "^2.1.0", 505 | "setprototypeof": "1.2.0", 506 | "statuses": "2.0.1", 507 | "type-is": "^2.0.0", 508 | "utils-merge": "1.0.1", 509 | "vary": "~1.1.2" 510 | }, 511 | "engines": { 512 | "node": ">= 18" 513 | } 514 | }, 515 | "node_modules/express-rate-limit": { 516 | "version": "7.5.0", 517 | "resolved": "https://registry.npmjs.org/express-rate-limit/-/express-rate-limit-7.5.0.tgz", 518 | "integrity": "sha512-eB5zbQh5h+VenMPM3fh+nw1YExi5nMr6HUCR62ELSP11huvxm/Uir1H1QEyTkk5QX6A58pX6NmaTMceKZ0Eodg==", 519 | "engines": { 520 | "node": ">= 16" 521 | }, 522 | "funding": { 523 | "url": "https://github.com/sponsors/express-rate-limit" 524 | }, 525 | "peerDependencies": { 526 | "express": "^4.11 || 5 || ^5.0.0-beta.1" 527 | } 528 | }, 529 | "node_modules/finalhandler": { 530 | "version": "2.1.0", 531 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-2.1.0.tgz", 532 | "integrity": "sha512-/t88Ty3d5JWQbWYgaOGCCYfXRwV1+be02WqYYlL6h0lEiUAMPM8o8qKGO01YIkOHzka2up08wvgYD0mDiI+q3Q==", 533 | "dependencies": { 534 | "debug": "^4.4.0", 535 | "encodeurl": "^2.0.0", 536 | "escape-html": "^1.0.3", 537 | "on-finished": "^2.4.1", 538 | "parseurl": "^1.3.3", 539 | "statuses": "^2.0.1" 540 | }, 541 | "engines": { 542 | "node": ">= 0.8" 543 | } 544 | }, 545 | "node_modules/finalhandler/node_modules/debug": { 546 | "version": "4.4.0", 547 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.0.tgz", 548 | "integrity": "sha512-6WTZ/IxCY/T6BALoZHaE4ctp9xm+Z5kY/pzYaCHRFeyVhojxlrm+46y68HA6hr0TcwEssoxNiDEUJQjfPZ/RYA==", 549 | "dependencies": { 550 | "ms": "^2.1.3" 551 | }, 552 | "engines": { 553 | "node": ">=6.0" 554 | }, 555 | "peerDependenciesMeta": { 556 | "supports-color": { 557 | "optional": true 558 | } 559 | } 560 | }, 561 | "node_modules/finalhandler/node_modules/ms": { 562 | "version": "2.1.3", 563 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 564 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==" 565 | }, 566 | "node_modules/follow-redirects": { 567 | "version": "1.15.9", 568 | "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.9.tgz", 569 | "integrity": "sha512-gew4GsXizNgdoRyqmyfMHyAmXsZDk6mHkSxZFCzW9gwlbtOW44CDtYavM+y+72qD/Vq2l550kMF52DT8fOLJqQ==", 570 | "funding": [ 571 | { 572 | "type": "individual", 573 | "url": "https://github.com/sponsors/RubenVerborgh" 574 | } 575 | ], 576 | "engines": { 577 | "node": ">=4.0" 578 | }, 579 | "peerDependenciesMeta": { 580 | "debug": { 581 | "optional": true 582 | } 583 | } 584 | }, 585 | "node_modules/form-data": { 586 | "version": "4.0.2", 587 | "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.2.tgz", 588 | "integrity": "sha512-hGfm/slu0ZabnNt4oaRZ6uREyfCj6P4fT/n6A1rGV+Z0VdGXjfOhVUpkn6qVQONHGIFwmveGXyDs75+nr6FM8w==", 589 | "dependencies": { 590 | "asynckit": "^0.4.0", 591 | "combined-stream": "^1.0.8", 592 | "es-set-tostringtag": "^2.1.0", 593 | "mime-types": "^2.1.12" 594 | }, 595 | "engines": { 596 | "node": ">= 6" 597 | } 598 | }, 599 | "node_modules/form-data/node_modules/mime-db": { 600 | "version": "1.52.0", 601 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", 602 | "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", 603 | "engines": { 604 | "node": ">= 0.6" 605 | } 606 | }, 607 | "node_modules/form-data/node_modules/mime-types": { 608 | "version": "2.1.35", 609 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", 610 | "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", 611 | "dependencies": { 612 | "mime-db": "1.52.0" 613 | }, 614 | "engines": { 615 | "node": ">= 0.6" 616 | } 617 | }, 618 | "node_modules/forwarded": { 619 | "version": "0.2.0", 620 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz", 621 | "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==", 622 | "engines": { 623 | "node": ">= 0.6" 624 | } 625 | }, 626 | "node_modules/fresh": { 627 | "version": "2.0.0", 628 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-2.0.0.tgz", 629 | "integrity": "sha512-Rx/WycZ60HOaqLKAi6cHRKKI7zxWbJ31MhntmtwMoaTeF7XFH9hhBp8vITaMidfljRQ6eYWCKkaTK+ykVJHP2A==", 630 | "engines": { 631 | "node": ">= 0.8" 632 | } 633 | }, 634 | "node_modules/function-bind": { 635 | "version": "1.1.2", 636 | "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", 637 | "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", 638 | "funding": { 639 | "url": "https://github.com/sponsors/ljharb" 640 | } 641 | }, 642 | "node_modules/get-intrinsic": { 643 | "version": "1.3.0", 644 | "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz", 645 | "integrity": "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==", 646 | "dependencies": { 647 | "call-bind-apply-helpers": "^1.0.2", 648 | "es-define-property": "^1.0.1", 649 | "es-errors": "^1.3.0", 650 | "es-object-atoms": "^1.1.1", 651 | "function-bind": "^1.1.2", 652 | "get-proto": "^1.0.1", 653 | "gopd": "^1.2.0", 654 | "has-symbols": "^1.1.0", 655 | "hasown": "^2.0.2", 656 | "math-intrinsics": "^1.1.0" 657 | }, 658 | "engines": { 659 | "node": ">= 0.4" 660 | }, 661 | "funding": { 662 | "url": "https://github.com/sponsors/ljharb" 663 | } 664 | }, 665 | "node_modules/get-proto": { 666 | "version": "1.0.1", 667 | "resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz", 668 | "integrity": "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==", 669 | "dependencies": { 670 | "dunder-proto": "^1.0.1", 671 | "es-object-atoms": "^1.0.0" 672 | }, 673 | "engines": { 674 | "node": ">= 0.4" 675 | } 676 | }, 677 | "node_modules/gopd": { 678 | "version": "1.2.0", 679 | "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz", 680 | "integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==", 681 | "engines": { 682 | "node": ">= 0.4" 683 | }, 684 | "funding": { 685 | "url": "https://github.com/sponsors/ljharb" 686 | } 687 | }, 688 | "node_modules/has-symbols": { 689 | "version": "1.1.0", 690 | "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz", 691 | "integrity": "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==", 692 | "engines": { 693 | "node": ">= 0.4" 694 | }, 695 | "funding": { 696 | "url": "https://github.com/sponsors/ljharb" 697 | } 698 | }, 699 | "node_modules/has-tostringtag": { 700 | "version": "1.0.2", 701 | "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.2.tgz", 702 | "integrity": "sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==", 703 | "dependencies": { 704 | "has-symbols": "^1.0.3" 705 | }, 706 | "engines": { 707 | "node": ">= 0.4" 708 | }, 709 | "funding": { 710 | "url": "https://github.com/sponsors/ljharb" 711 | } 712 | }, 713 | "node_modules/hasown": { 714 | "version": "2.0.2", 715 | "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", 716 | "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", 717 | "dependencies": { 718 | "function-bind": "^1.1.2" 719 | }, 720 | "engines": { 721 | "node": ">= 0.4" 722 | } 723 | }, 724 | "node_modules/http-errors": { 725 | "version": "2.0.0", 726 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.0.tgz", 727 | "integrity": "sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ==", 728 | "dependencies": { 729 | "depd": "2.0.0", 730 | "inherits": "2.0.4", 731 | "setprototypeof": "1.2.0", 732 | "statuses": "2.0.1", 733 | "toidentifier": "1.0.1" 734 | }, 735 | "engines": { 736 | "node": ">= 0.8" 737 | } 738 | }, 739 | "node_modules/iconv-lite": { 740 | "version": "0.5.2", 741 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.5.2.tgz", 742 | "integrity": "sha512-kERHXvpSaB4aU3eANwidg79K8FlrN77m8G9V+0vOR3HYaRifrlwMEpT7ZBJqLSEIHnEgJTHcWK82wwLwwKwtag==", 743 | "dependencies": { 744 | "safer-buffer": ">= 2.1.2 < 3" 745 | }, 746 | "engines": { 747 | "node": ">=0.10.0" 748 | } 749 | }, 750 | "node_modules/inherits": { 751 | "version": "2.0.4", 752 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", 753 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" 754 | }, 755 | "node_modules/ipaddr.js": { 756 | "version": "1.9.1", 757 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", 758 | "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==", 759 | "engines": { 760 | "node": ">= 0.10" 761 | } 762 | }, 763 | "node_modules/is-promise": { 764 | "version": "4.0.0", 765 | "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-4.0.0.tgz", 766 | "integrity": "sha512-hvpoI6korhJMnej285dSg6nu1+e6uxs7zG3BYAm5byqDsgJNWwxzM6z6iZiAgQR4TJ30JmBTOwqZUw3WlyH3AQ==" 767 | }, 768 | "node_modules/make-error": { 769 | "version": "1.3.6", 770 | "resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.6.tgz", 771 | "integrity": "sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==", 772 | "dev": true 773 | }, 774 | "node_modules/math-intrinsics": { 775 | "version": "1.1.0", 776 | "resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz", 777 | "integrity": "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==", 778 | "engines": { 779 | "node": ">= 0.4" 780 | } 781 | }, 782 | "node_modules/media-typer": { 783 | "version": "1.1.0", 784 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-1.1.0.tgz", 785 | "integrity": "sha512-aisnrDP4GNe06UcKFnV5bfMNPBUw4jsLGaWwWfnH3v02GnBuXX2MCVn5RbrWo0j3pczUilYblq7fQ7Nw2t5XKw==", 786 | "engines": { 787 | "node": ">= 0.8" 788 | } 789 | }, 790 | "node_modules/merge-descriptors": { 791 | "version": "2.0.0", 792 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-2.0.0.tgz", 793 | "integrity": "sha512-Snk314V5ayFLhp3fkUREub6WtjBfPdCPY1Ln8/8munuLuiYhsABgBVWsozAG+MWMbVEvcdcpbi9R7ww22l9Q3g==", 794 | "engines": { 795 | "node": ">=18" 796 | }, 797 | "funding": { 798 | "url": "https://github.com/sponsors/sindresorhus" 799 | } 800 | }, 801 | "node_modules/methods": { 802 | "version": "1.1.2", 803 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", 804 | "integrity": "sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==", 805 | "engines": { 806 | "node": ">= 0.6" 807 | } 808 | }, 809 | "node_modules/mime-db": { 810 | "version": "1.54.0", 811 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.54.0.tgz", 812 | "integrity": "sha512-aU5EJuIN2WDemCcAp2vFBfp/m4EAhWJnUNSSw0ixs7/kXbd6Pg64EmwJkNdFhB8aWt1sH2CTXrLxo/iAGV3oPQ==", 813 | "engines": { 814 | "node": ">= 0.6" 815 | } 816 | }, 817 | "node_modules/mime-types": { 818 | "version": "3.0.0", 819 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-3.0.0.tgz", 820 | "integrity": "sha512-XqoSHeCGjVClAmoGFG3lVFqQFRIrTVw2OH3axRqAcfaw+gHWIfnASS92AV+Rl/mk0MupgZTRHQOjxY6YVnzK5w==", 821 | "dependencies": { 822 | "mime-db": "^1.53.0" 823 | }, 824 | "engines": { 825 | "node": ">= 0.6" 826 | } 827 | }, 828 | "node_modules/ms": { 829 | "version": "2.1.2", 830 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 831 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" 832 | }, 833 | "node_modules/negotiator": { 834 | "version": "1.0.0", 835 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-1.0.0.tgz", 836 | "integrity": "sha512-8Ofs/AUQh8MaEcrlq5xOX0CQ9ypTF5dl78mjlMNfOK08fzpgTHQRQPBxcPlEtIw0yRpws+Zo/3r+5WRby7u3Gg==", 837 | "engines": { 838 | "node": ">= 0.6" 839 | } 840 | }, 841 | "node_modules/object-assign": { 842 | "version": "4.1.1", 843 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", 844 | "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==", 845 | "engines": { 846 | "node": ">=0.10.0" 847 | } 848 | }, 849 | "node_modules/object-inspect": { 850 | "version": "1.13.4", 851 | "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.4.tgz", 852 | "integrity": "sha512-W67iLl4J2EXEGTbfeHCffrjDfitvLANg0UlX3wFUUSTx92KXRFegMHUVgSqE+wvhAbi4WqjGg9czysTV2Epbew==", 853 | "engines": { 854 | "node": ">= 0.4" 855 | }, 856 | "funding": { 857 | "url": "https://github.com/sponsors/ljharb" 858 | } 859 | }, 860 | "node_modules/on-finished": { 861 | "version": "2.4.1", 862 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz", 863 | "integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==", 864 | "dependencies": { 865 | "ee-first": "1.1.1" 866 | }, 867 | "engines": { 868 | "node": ">= 0.8" 869 | } 870 | }, 871 | "node_modules/once": { 872 | "version": "1.4.0", 873 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", 874 | "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", 875 | "dependencies": { 876 | "wrappy": "1" 877 | } 878 | }, 879 | "node_modules/parseurl": { 880 | "version": "1.3.3", 881 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", 882 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==", 883 | "engines": { 884 | "node": ">= 0.8" 885 | } 886 | }, 887 | "node_modules/path-to-regexp": { 888 | "version": "8.2.0", 889 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-8.2.0.tgz", 890 | "integrity": "sha512-TdrF7fW9Rphjq4RjrW0Kp2AW0Ahwu9sRGTkS6bvDi0SCwZlEZYmcfDbEsTz8RVk0EHIS/Vd1bv3JhG+1xZuAyQ==", 891 | "engines": { 892 | "node": ">=16" 893 | } 894 | }, 895 | "node_modules/pkce-challenge": { 896 | "version": "4.1.0", 897 | "resolved": "https://registry.npmjs.org/pkce-challenge/-/pkce-challenge-4.1.0.tgz", 898 | "integrity": "sha512-ZBmhE1C9LcPoH9XZSdwiPtbPHZROwAnMy+kIFQVrnMCxY4Cudlz3gBOpzilgc0jOgRaiT3sIWfpMomW2ar2orQ==", 899 | "engines": { 900 | "node": ">=16.20.0" 901 | } 902 | }, 903 | "node_modules/proxy-addr": { 904 | "version": "2.0.7", 905 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", 906 | "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", 907 | "dependencies": { 908 | "forwarded": "0.2.0", 909 | "ipaddr.js": "1.9.1" 910 | }, 911 | "engines": { 912 | "node": ">= 0.10" 913 | } 914 | }, 915 | "node_modules/proxy-from-env": { 916 | "version": "1.1.0", 917 | "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz", 918 | "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==" 919 | }, 920 | "node_modules/qs": { 921 | "version": "6.13.0", 922 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.13.0.tgz", 923 | "integrity": "sha512-+38qI9SOr8tfZ4QmJNplMUxqjbe7LKvvZgWdExBOmd+egZTtjLB67Gu0HRX3u/XOq7UU2Nx6nsjvS16Z9uwfpg==", 924 | "dependencies": { 925 | "side-channel": "^1.0.6" 926 | }, 927 | "engines": { 928 | "node": ">=0.6" 929 | }, 930 | "funding": { 931 | "url": "https://github.com/sponsors/ljharb" 932 | } 933 | }, 934 | "node_modules/range-parser": { 935 | "version": "1.2.1", 936 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", 937 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==", 938 | "engines": { 939 | "node": ">= 0.6" 940 | } 941 | }, 942 | "node_modules/raw-body": { 943 | "version": "3.0.0", 944 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-3.0.0.tgz", 945 | "integrity": "sha512-RmkhL8CAyCRPXCE28MMH0z2PNWQBNk2Q09ZdxM9IOOXwxwZbN+qbWaatPkdkWIKL2ZVDImrN/pK5HTRz2PcS4g==", 946 | "dependencies": { 947 | "bytes": "3.1.2", 948 | "http-errors": "2.0.0", 949 | "iconv-lite": "0.6.3", 950 | "unpipe": "1.0.0" 951 | }, 952 | "engines": { 953 | "node": ">= 0.8" 954 | } 955 | }, 956 | "node_modules/raw-body/node_modules/iconv-lite": { 957 | "version": "0.6.3", 958 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz", 959 | "integrity": "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==", 960 | "dependencies": { 961 | "safer-buffer": ">= 2.1.2 < 3.0.0" 962 | }, 963 | "engines": { 964 | "node": ">=0.10.0" 965 | } 966 | }, 967 | "node_modules/router": { 968 | "version": "2.1.0", 969 | "resolved": "https://registry.npmjs.org/router/-/router-2.1.0.tgz", 970 | "integrity": "sha512-/m/NSLxeYEgWNtyC+WtNHCF7jbGxOibVWKnn+1Psff4dJGOfoXP+MuC/f2CwSmyiHdOIzYnYFp4W6GxWfekaLA==", 971 | "dependencies": { 972 | "is-promise": "^4.0.0", 973 | "parseurl": "^1.3.3", 974 | "path-to-regexp": "^8.0.0" 975 | }, 976 | "engines": { 977 | "node": ">= 18" 978 | } 979 | }, 980 | "node_modules/safe-buffer": { 981 | "version": "5.2.1", 982 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", 983 | "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", 984 | "funding": [ 985 | { 986 | "type": "github", 987 | "url": "https://github.com/sponsors/feross" 988 | }, 989 | { 990 | "type": "patreon", 991 | "url": "https://www.patreon.com/feross" 992 | }, 993 | { 994 | "type": "consulting", 995 | "url": "https://feross.org/support" 996 | } 997 | ] 998 | }, 999 | "node_modules/safer-buffer": { 1000 | "version": "2.1.2", 1001 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", 1002 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" 1003 | }, 1004 | "node_modules/send": { 1005 | "version": "1.1.0", 1006 | "resolved": "https://registry.npmjs.org/send/-/send-1.1.0.tgz", 1007 | "integrity": "sha512-v67WcEouB5GxbTWL/4NeToqcZiAWEq90N888fczVArY8A79J0L4FD7vj5hm3eUMua5EpoQ59wa/oovY6TLvRUA==", 1008 | "dependencies": { 1009 | "debug": "^4.3.5", 1010 | "destroy": "^1.2.0", 1011 | "encodeurl": "^2.0.0", 1012 | "escape-html": "^1.0.3", 1013 | "etag": "^1.8.1", 1014 | "fresh": "^0.5.2", 1015 | "http-errors": "^2.0.0", 1016 | "mime-types": "^2.1.35", 1017 | "ms": "^2.1.3", 1018 | "on-finished": "^2.4.1", 1019 | "range-parser": "^1.2.1", 1020 | "statuses": "^2.0.1" 1021 | }, 1022 | "engines": { 1023 | "node": ">= 18" 1024 | } 1025 | }, 1026 | "node_modules/send/node_modules/fresh": { 1027 | "version": "0.5.2", 1028 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", 1029 | "integrity": "sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q==", 1030 | "engines": { 1031 | "node": ">= 0.6" 1032 | } 1033 | }, 1034 | "node_modules/send/node_modules/mime-db": { 1035 | "version": "1.52.0", 1036 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", 1037 | "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", 1038 | "engines": { 1039 | "node": ">= 0.6" 1040 | } 1041 | }, 1042 | "node_modules/send/node_modules/mime-types": { 1043 | "version": "2.1.35", 1044 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", 1045 | "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", 1046 | "dependencies": { 1047 | "mime-db": "1.52.0" 1048 | }, 1049 | "engines": { 1050 | "node": ">= 0.6" 1051 | } 1052 | }, 1053 | "node_modules/send/node_modules/ms": { 1054 | "version": "2.1.3", 1055 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 1056 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==" 1057 | }, 1058 | "node_modules/serve-static": { 1059 | "version": "2.1.0", 1060 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-2.1.0.tgz", 1061 | "integrity": "sha512-A3We5UfEjG8Z7VkDv6uItWw6HY2bBSBJT1KtVESn6EOoOr2jAxNhxWCLY3jDE2WcuHXByWju74ck3ZgLwL8xmA==", 1062 | "dependencies": { 1063 | "encodeurl": "^2.0.0", 1064 | "escape-html": "^1.0.3", 1065 | "parseurl": "^1.3.3", 1066 | "send": "^1.0.0" 1067 | }, 1068 | "engines": { 1069 | "node": ">= 18" 1070 | } 1071 | }, 1072 | "node_modules/setprototypeof": { 1073 | "version": "1.2.0", 1074 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz", 1075 | "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==" 1076 | }, 1077 | "node_modules/side-channel": { 1078 | "version": "1.1.0", 1079 | "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.1.0.tgz", 1080 | "integrity": "sha512-ZX99e6tRweoUXqR+VBrslhda51Nh5MTQwou5tnUDgbtyM0dBgmhEDtWGP/xbKn6hqfPRHujUNwz5fy/wbbhnpw==", 1081 | "dependencies": { 1082 | "es-errors": "^1.3.0", 1083 | "object-inspect": "^1.13.3", 1084 | "side-channel-list": "^1.0.0", 1085 | "side-channel-map": "^1.0.1", 1086 | "side-channel-weakmap": "^1.0.2" 1087 | }, 1088 | "engines": { 1089 | "node": ">= 0.4" 1090 | }, 1091 | "funding": { 1092 | "url": "https://github.com/sponsors/ljharb" 1093 | } 1094 | }, 1095 | "node_modules/side-channel-list": { 1096 | "version": "1.0.0", 1097 | "resolved": "https://registry.npmjs.org/side-channel-list/-/side-channel-list-1.0.0.tgz", 1098 | "integrity": "sha512-FCLHtRD/gnpCiCHEiJLOwdmFP+wzCmDEkc9y7NsYxeF4u7Btsn1ZuwgwJGxImImHicJArLP4R0yX4c2KCrMrTA==", 1099 | "dependencies": { 1100 | "es-errors": "^1.3.0", 1101 | "object-inspect": "^1.13.3" 1102 | }, 1103 | "engines": { 1104 | "node": ">= 0.4" 1105 | }, 1106 | "funding": { 1107 | "url": "https://github.com/sponsors/ljharb" 1108 | } 1109 | }, 1110 | "node_modules/side-channel-map": { 1111 | "version": "1.0.1", 1112 | "resolved": "https://registry.npmjs.org/side-channel-map/-/side-channel-map-1.0.1.tgz", 1113 | "integrity": "sha512-VCjCNfgMsby3tTdo02nbjtM/ewra6jPHmpThenkTYh8pG9ucZ/1P8So4u4FGBek/BjpOVsDCMoLA/iuBKIFXRA==", 1114 | "dependencies": { 1115 | "call-bound": "^1.0.2", 1116 | "es-errors": "^1.3.0", 1117 | "get-intrinsic": "^1.2.5", 1118 | "object-inspect": "^1.13.3" 1119 | }, 1120 | "engines": { 1121 | "node": ">= 0.4" 1122 | }, 1123 | "funding": { 1124 | "url": "https://github.com/sponsors/ljharb" 1125 | } 1126 | }, 1127 | "node_modules/side-channel-weakmap": { 1128 | "version": "1.0.2", 1129 | "resolved": "https://registry.npmjs.org/side-channel-weakmap/-/side-channel-weakmap-1.0.2.tgz", 1130 | "integrity": "sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A==", 1131 | "dependencies": { 1132 | "call-bound": "^1.0.2", 1133 | "es-errors": "^1.3.0", 1134 | "get-intrinsic": "^1.2.5", 1135 | "object-inspect": "^1.13.3", 1136 | "side-channel-map": "^1.0.1" 1137 | }, 1138 | "engines": { 1139 | "node": ">= 0.4" 1140 | }, 1141 | "funding": { 1142 | "url": "https://github.com/sponsors/ljharb" 1143 | } 1144 | }, 1145 | "node_modules/statuses": { 1146 | "version": "2.0.1", 1147 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.1.tgz", 1148 | "integrity": "sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==", 1149 | "engines": { 1150 | "node": ">= 0.8" 1151 | } 1152 | }, 1153 | "node_modules/toidentifier": { 1154 | "version": "1.0.1", 1155 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz", 1156 | "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==", 1157 | "engines": { 1158 | "node": ">=0.6" 1159 | } 1160 | }, 1161 | "node_modules/ts-node": { 1162 | "version": "10.9.2", 1163 | "resolved": "https://registry.npmjs.org/ts-node/-/ts-node-10.9.2.tgz", 1164 | "integrity": "sha512-f0FFpIdcHgn8zcPSbf1dRevwt047YMnaiJM3u2w2RewrB+fob/zePZcrOyQoLMMO7aBIddLcQIEK5dYjkLnGrQ==", 1165 | "dev": true, 1166 | "dependencies": { 1167 | "@cspotcode/source-map-support": "^0.8.0", 1168 | "@tsconfig/node10": "^1.0.7", 1169 | "@tsconfig/node12": "^1.0.7", 1170 | "@tsconfig/node14": "^1.0.0", 1171 | "@tsconfig/node16": "^1.0.2", 1172 | "acorn": "^8.4.1", 1173 | "acorn-walk": "^8.1.1", 1174 | "arg": "^4.1.0", 1175 | "create-require": "^1.1.0", 1176 | "diff": "^4.0.1", 1177 | "make-error": "^1.1.1", 1178 | "v8-compile-cache-lib": "^3.0.1", 1179 | "yn": "3.1.1" 1180 | }, 1181 | "bin": { 1182 | "ts-node": "dist/bin.js", 1183 | "ts-node-cwd": "dist/bin-cwd.js", 1184 | "ts-node-esm": "dist/bin-esm.js", 1185 | "ts-node-script": "dist/bin-script.js", 1186 | "ts-node-transpile-only": "dist/bin-transpile.js", 1187 | "ts-script": "dist/bin-script-deprecated.js" 1188 | }, 1189 | "peerDependencies": { 1190 | "@swc/core": ">=1.2.50", 1191 | "@swc/wasm": ">=1.2.50", 1192 | "@types/node": "*", 1193 | "typescript": ">=2.7" 1194 | }, 1195 | "peerDependenciesMeta": { 1196 | "@swc/core": { 1197 | "optional": true 1198 | }, 1199 | "@swc/wasm": { 1200 | "optional": true 1201 | } 1202 | } 1203 | }, 1204 | "node_modules/type-is": { 1205 | "version": "2.0.0", 1206 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-2.0.0.tgz", 1207 | "integrity": "sha512-gd0sGezQYCbWSbkZr75mln4YBidWUN60+devscpLF5mtRDUpiaTvKpBNrdaCvel1NdR2k6vclXybU5fBd2i+nw==", 1208 | "dependencies": { 1209 | "content-type": "^1.0.5", 1210 | "media-typer": "^1.1.0", 1211 | "mime-types": "^3.0.0" 1212 | }, 1213 | "engines": { 1214 | "node": ">= 0.6" 1215 | } 1216 | }, 1217 | "node_modules/typescript": { 1218 | "version": "5.8.2", 1219 | "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.8.2.tgz", 1220 | "integrity": "sha512-aJn6wq13/afZp/jT9QZmwEjDqqvSGp1VT5GVg+f/t6/oVyrgXM6BY1h9BRh/O5p3PlUPAe+WuiEZOmb/49RqoQ==", 1221 | "dev": true, 1222 | "bin": { 1223 | "tsc": "bin/tsc", 1224 | "tsserver": "bin/tsserver" 1225 | }, 1226 | "engines": { 1227 | "node": ">=14.17" 1228 | } 1229 | }, 1230 | "node_modules/undici-types": { 1231 | "version": "6.19.8", 1232 | "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.19.8.tgz", 1233 | "integrity": "sha512-ve2KP6f/JnbPBFyobGHuerC9g1FYGn/F8n1LWTwNxCEzd6IfqTwUQcNXgEtmmQ6DlRrC1hrSrBnCZPokRrDHjw==", 1234 | "dev": true 1235 | }, 1236 | "node_modules/unpipe": { 1237 | "version": "1.0.0", 1238 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", 1239 | "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==", 1240 | "engines": { 1241 | "node": ">= 0.8" 1242 | } 1243 | }, 1244 | "node_modules/utils-merge": { 1245 | "version": "1.0.1", 1246 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", 1247 | "integrity": "sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==", 1248 | "engines": { 1249 | "node": ">= 0.4.0" 1250 | } 1251 | }, 1252 | "node_modules/v8-compile-cache-lib": { 1253 | "version": "3.0.1", 1254 | "resolved": "https://registry.npmjs.org/v8-compile-cache-lib/-/v8-compile-cache-lib-3.0.1.tgz", 1255 | "integrity": "sha512-wa7YjyUGfNZngI/vtK0UHAN+lgDCxBPCylVXGp0zu59Fz5aiGtNXaq3DhIov063MorB+VfufLh3JlF2KdTK3xg==", 1256 | "dev": true 1257 | }, 1258 | "node_modules/vary": { 1259 | "version": "1.1.2", 1260 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", 1261 | "integrity": "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==", 1262 | "engines": { 1263 | "node": ">= 0.8" 1264 | } 1265 | }, 1266 | "node_modules/wrappy": { 1267 | "version": "1.0.2", 1268 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", 1269 | "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==" 1270 | }, 1271 | "node_modules/yn": { 1272 | "version": "3.1.1", 1273 | "resolved": "https://registry.npmjs.org/yn/-/yn-3.1.1.tgz", 1274 | "integrity": "sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==", 1275 | "dev": true, 1276 | "engines": { 1277 | "node": ">=6" 1278 | } 1279 | }, 1280 | "node_modules/zod": { 1281 | "version": "3.24.2", 1282 | "resolved": "https://registry.npmjs.org/zod/-/zod-3.24.2.tgz", 1283 | "integrity": "sha512-lY7CDW43ECgW9u1TcT3IoXHflywfVqDYze4waEz812jR/bZ8FHDsl7pFQoSZTz5N+2NqRXs8GBwnAwo3ZNxqhQ==", 1284 | "funding": { 1285 | "url": "https://github.com/sponsors/colinhacks" 1286 | } 1287 | }, 1288 | "node_modules/zod-to-json-schema": { 1289 | "version": "3.24.5", 1290 | "resolved": "https://registry.npmjs.org/zod-to-json-schema/-/zod-to-json-schema-3.24.5.tgz", 1291 | "integrity": "sha512-/AuWwMP+YqiPbsJx5D6TfgRTc4kTLjsh5SOcd4bLsfUg2RcEXrFMJl1DGgdHy2aCfsIA/cr/1JM0xcB2GZji8g==", 1292 | "peerDependencies": { 1293 | "zod": "^3.24.1" 1294 | } 1295 | } 1296 | } 1297 | } 1298 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "mcp-server-weibo", 3 | "version": "1.0.2", 4 | "description": "Model Context Protocol服务器,用于抓取微博用户信息、动态和搜索等功能", 5 | "main": "dist/index.js", 6 | "types": "dist/index.d.ts", 7 | "bin": { 8 | "mcp-server-weibo": "dist/index.js" 9 | }, 10 | "scripts": { 11 | "build": "tsc", 12 | "start": "node dist/index.js", 13 | "dev": "ts-node src/index.ts", 14 | "test": "jest" 15 | }, 16 | "keywords": [ 17 | "weibo", 18 | "mcp", 19 | "model-context-protocol" 20 | ], 21 | "author": "", 22 | "license": "MIT", 23 | "dependencies": { 24 | "@modelcontextprotocol/sdk": "^1.7.0", 25 | "axios": "^1.8.4", 26 | "zod": "^3.22.4" 27 | }, 28 | "devDependencies": { 29 | "@types/node": "^20.10.6", 30 | "ts-node": "^10.9.2", 31 | "typescript": "^5.3.3" 32 | }, 33 | "engines": { 34 | "node": ">=18.0.0" 35 | } 36 | } 37 | -------------------------------------------------------------------------------- /src/consts.ts: -------------------------------------------------------------------------------- 1 | /** 2 | * 微博API请求的默认HTTP头 3 | */ 4 | export const DEFAULT_HEADERS = { 'Content-Type': 'application/json' }; 5 | 6 | /** 7 | * 获取用户资料的URL模板 8 | * {userId}将被替换为实际的用户ID 9 | */ 10 | export const PROFILE_URL = 'https://m.weibo.cn/api/container/getIndex?type=uid&value={userId}'; 11 | 12 | /** 13 | * 获取用户微博动态的URL模板 14 | * {userId}: 用户的唯一标识符 15 | * {containerId}: 用户动态的容器ID 16 | * {sinceId}: 分页信息,上一页最后一条动态的ID 17 | */ 18 | export const FEEDS_URL = 'https://m.weibo.cn/api/container/getIndex?type=uid&value={userId}&containerid={containerId}&since_id={sinceId}'; 19 | 20 | /** 21 | * 获取微博热搜榜的URL 22 | */ 23 | export const HOT_SEARCH_URL = 'https://m.weibo.cn/api/container/getIndex?containerid=106003type%3D25%26t%3D3%26disable_hot%3D1%26filter_type%3Drealtimehot'; 24 | 25 | /** 26 | * 搜索微博内容的URL模板 27 | * {keyword}将被替换为要搜索的关键词 28 | * {page}将被替换为页码 29 | */ 30 | export const SEARCH_CONTENT_URL = 'https://m.weibo.cn/api/container/getIndex?containerid=100103type%3D1%26q%3D{keyword}&page_type=searchall&page={page}'; -------------------------------------------------------------------------------- /src/index.ts: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env node 2 | 3 | import { connectServer } from './server'; 4 | 5 | // 启动MCP服务器,使用标准输入/输出作为通信通道 6 | connectServer().catch(console.error); -------------------------------------------------------------------------------- /src/server.ts: -------------------------------------------------------------------------------- 1 | import { McpServer } from "@modelcontextprotocol/sdk/server/mcp.js"; 2 | import { StdioServerTransport } from "@modelcontextprotocol/sdk/server/stdio.js"; 3 | import { z } from "zod"; 4 | import { WeiboCrawler } from './weibo'; 5 | 6 | // 初始化MCP服务器,名称为"Weibo" 7 | const server = new McpServer({ 8 | name: "Weibo", 9 | version: "1.0.0" 10 | }); 11 | 12 | // 创建WeiboCrawler实例处理微博API操作 13 | const crawler = new WeiboCrawler(); 14 | 15 | /** 16 | * 根据关键词搜索微博用户 17 | */ 18 | server.tool("search_users", 19 | "根据关键词搜索微博用户并返回匹配的用户列表", 20 | { 21 | keyword: z.string().describe("查找用户的搜索词"), 22 | limit: z.number().describe("返回的最大用户数量") 23 | }, 24 | async ({ keyword, limit }) => { 25 | const users = await crawler.searchWeiboUsers(keyword, limit); 26 | return { 27 | content: [{ type: "text", text: JSON.stringify(users) }] 28 | }; 29 | } 30 | ); 31 | 32 | /** 33 | * 获取微博用户的资料信息 34 | */ 35 | server.tool("get_profile", 36 | "获取指定微博用户的详细资料信息", 37 | { 38 | uid: z.number().describe("微博用户的唯一标识符") 39 | }, 40 | async ({ uid }) => { 41 | const profile = await crawler.extractWeiboProfile(uid); 42 | return { 43 | content: [{ type: "text", text: JSON.stringify(profile) }] 44 | }; 45 | } 46 | ); 47 | 48 | /** 49 | * 获取微博用户的动态(帖子) 50 | */ 51 | server.tool("get_feeds", 52 | "获取指定微博用户的最新动态和帖子", 53 | { 54 | uid: z.number().describe("微博用户的唯一标识符"), 55 | limit: z.number().describe("返回的最大动态数量") 56 | }, 57 | async ({ uid, limit }) => { 58 | const feeds = await crawler.extractWeiboFeeds(uid, limit); 59 | return { 60 | content: [{ type: "text", text: JSON.stringify(feeds) }] 61 | }; 62 | } 63 | ); 64 | 65 | /** 66 | * 获取微博热搜榜 67 | */ 68 | server.tool("get_hot_search", 69 | "获取当前微博热搜榜的热门话题列表", 70 | { 71 | limit: z.number().describe("返回的最大热搜条目数量") 72 | }, 73 | async ({ limit }) => { 74 | const hotSearchList = await crawler.getHotSearchList(limit); 75 | return { 76 | content: [{ type: "text", text: JSON.stringify(hotSearchList) }] 77 | }; 78 | } 79 | ); 80 | 81 | /** 82 | * 根据关键词搜索微博内容 83 | */ 84 | server.tool("search_content", 85 | "根据关键词搜索微博内容并返回相关的微博帖子", 86 | { 87 | keyword: z.string().describe("搜索微博内容的关键词"), 88 | limit: z.number().describe("返回的最大微博条目数量"), 89 | page: z.number().optional().describe("起始页码,默认为1") 90 | }, 91 | async ({ keyword, limit, page }) => { 92 | const contents = await crawler.searchWeiboContent(keyword, limit, page || 1); 93 | return { 94 | content: [{ type: "text", text: JSON.stringify(contents) }] 95 | }; 96 | } 97 | ); 98 | 99 | // 导出MCP服务器实例和连接函数 100 | export const connectServer = async () => { 101 | const transport = new StdioServerTransport(); 102 | await server.connect(transport); 103 | }; 104 | 105 | export default server; -------------------------------------------------------------------------------- /src/types.ts: -------------------------------------------------------------------------------- 1 | /** 2 | * 分页微博动态数据模型 3 | */ 4 | export interface PagedFeeds { 5 | /** 6 | * 分页信息,上一页最后一条动态的ID 7 | */ 8 | SinceId: string | number | null; 9 | 10 | /** 11 | * 微博动态条目列表 12 | */ 13 | Feeds: Record[]; 14 | } 15 | 16 | /** 17 | * 微博用户搜索结果数据模型 18 | */ 19 | export interface SearchResult { 20 | /** 21 | * 用户的唯一标识符 22 | */ 23 | id: number; 24 | 25 | /** 26 | * 用户的显示名称 27 | */ 28 | nickName: string; 29 | 30 | /** 31 | * 用户高分辨率头像图片的URL 32 | */ 33 | avatarHD: string; 34 | 35 | /** 36 | * 用户的个人简介 37 | */ 38 | description: string; 39 | } 40 | 41 | /** 42 | * 微博热搜榜条目数据模型 43 | */ 44 | export interface HotSearchItem { 45 | /** 46 | * 热搜关键词 47 | */ 48 | keyword: string; 49 | 50 | /** 51 | * 热搜排名 52 | */ 53 | rank: number; 54 | 55 | /** 56 | * 热搜热度 57 | */ 58 | hotValue: number; 59 | 60 | /** 61 | * 标签类型(如新、热、爆等) 62 | */ 63 | tag?: string; 64 | 65 | /** 66 | * 搜索链接 67 | */ 68 | url?: string; 69 | } 70 | 71 | /** 72 | * 微博内容搜索结果数据模型 73 | */ 74 | export interface ContentSearchResult { 75 | /** 76 | * 微博ID 77 | */ 78 | id: string; 79 | 80 | /** 81 | * 微博文本内容 82 | */ 83 | text: string; 84 | 85 | /** 86 | * 创建时间 87 | */ 88 | created_at: string; 89 | 90 | /** 91 | * 转发数 92 | */ 93 | reposts_count: number; 94 | 95 | /** 96 | * 评论数 97 | */ 98 | comments_count: number; 99 | 100 | /** 101 | * 点赞数 102 | */ 103 | attitudes_count: number; 104 | 105 | /** 106 | * 发布该微博的用户信息 107 | */ 108 | user: { 109 | id: number; 110 | screen_name: string; 111 | profile_image_url: string; 112 | verified: boolean; 113 | }; 114 | 115 | /** 116 | * 图片链接列表(如果有) 117 | */ 118 | pics?: string[]; 119 | 120 | /** 121 | * 视频链接(如果有) 122 | */ 123 | video_url?: string; 124 | } -------------------------------------------------------------------------------- /src/weibo.ts: -------------------------------------------------------------------------------- 1 | import axios from 'axios'; 2 | import { DEFAULT_HEADERS, PROFILE_URL, FEEDS_URL, HOT_SEARCH_URL, SEARCH_CONTENT_URL } from './consts'; 3 | import { PagedFeeds, SearchResult, HotSearchItem, ContentSearchResult } from './types'; 4 | 5 | /** 6 | * 微博爬虫类,用于从微博提取数据 7 | * 提供获取用户资料、动态和搜索用户的功能 8 | */ 9 | export class WeiboCrawler { 10 | /** 11 | * 从微博提取用户资料信息 12 | * 13 | * @param uid 微博用户的唯一标识符 14 | * @returns 用户资料信息,如果提取失败则返回空对象 15 | */ 16 | async extractWeiboProfile(uid: number): Promise> { 17 | try { 18 | const response = await axios.get(PROFILE_URL.replace('{userId}', uid.toString()), { 19 | headers: DEFAULT_HEADERS 20 | }); 21 | return response.data.data.userInfo; 22 | } catch (error) { 23 | console.error(`无法获取UID为'${uid}'的用户资料`, error); 24 | return {}; 25 | } 26 | } 27 | 28 | /** 29 | * 提取用户的微博动态,支持分页 30 | * 31 | * @param uid 微博用户的唯一标识符 32 | * @param limit 最大提取的动态数量 33 | * @returns 用户微博动态列表 34 | */ 35 | async extractWeiboFeeds(uid: number, limit: number): Promise[]> { 36 | const feeds: Record[] = []; 37 | let sinceId = ''; 38 | 39 | try { 40 | const containerId = await this.getContainerId(uid); 41 | if (!containerId) return feeds; 42 | 43 | while (feeds.length < limit) { 44 | const pagedFeeds = await this.extractFeeds(uid, containerId, sinceId); 45 | if (!pagedFeeds.Feeds || pagedFeeds.Feeds.length === 0) { 46 | break; 47 | } 48 | 49 | feeds.push(...pagedFeeds.Feeds); 50 | sinceId = pagedFeeds.SinceId as string; 51 | if (!sinceId) { 52 | break; 53 | } 54 | } 55 | } catch (error) { 56 | console.error(`无法获取UID为'${uid}'的动态`, error); 57 | } 58 | 59 | return feeds.slice(0, limit); 60 | } 61 | 62 | /** 63 | * 根据关键词搜索微博用户 64 | * 65 | * @param keyword 查找用户的搜索词 66 | * @param limit 返回的最大用户数量 67 | * @returns 包含用户信息的SearchResult对象列表 68 | */ 69 | async searchWeiboUsers(keyword: string, limit: number): Promise { 70 | try { 71 | const params = { 'containerid': `100103type=3&q=${keyword}&t=`, 'page_type': 'searchall' }; 72 | const searchParams = new URLSearchParams(); 73 | for (const [key, value] of Object.entries(params)) { 74 | searchParams.append(key, value); 75 | } 76 | const queryString = searchParams.toString(); 77 | 78 | const response = await axios.get(`https://m.weibo.cn/api/container/getIndex?${queryString}`, { 79 | headers: DEFAULT_HEADERS 80 | }); 81 | 82 | const result = response.data; 83 | const cards = result.data.cards; 84 | 85 | if (cards.length < 2) { 86 | return []; 87 | } else { 88 | const cardGroup = cards[1]['card_group']; 89 | return cardGroup.map((item: any) => this.toSearchResult(item.user)).slice(0, limit); 90 | } 91 | } catch (error) { 92 | console.error(`无法搜索关键词为'${keyword}'的用户`, error); 93 | return []; 94 | } 95 | } 96 | 97 | /** 98 | * 将原始用户数据转换为SearchResult对象 99 | * 100 | * @param user 来自微博API的原始用户数据 101 | * @returns 格式化的用户信息 102 | */ 103 | private toSearchResult(user: any): SearchResult { 104 | return { 105 | id: user.id, 106 | nickName: user.screen_name, 107 | avatarHD: user.avatar_hd, 108 | description: user.description 109 | }; 110 | } 111 | 112 | /** 113 | * 获取用户微博动态的容器ID 114 | * 115 | * @param uid 微博用户的唯一标识符 116 | * @returns 用户动态的容器ID,如果提取失败则返回null 117 | */ 118 | private async getContainerId(uid: number): Promise { 119 | try { 120 | const response = await axios.get(PROFILE_URL.replace('{userId}', uid.toString()), { 121 | headers: DEFAULT_HEADERS 122 | }); 123 | 124 | const data = response.data; 125 | const tabsInfo = data?.data?.tabsInfo?.tabs || []; 126 | 127 | for (const tab of tabsInfo) { 128 | if (tab.tabKey === 'weibo') { 129 | return tab.containerid; 130 | } 131 | } 132 | return null; 133 | } catch (error) { 134 | console.error(`无法获取UID为'${uid}'的containerId`, error); 135 | return null; 136 | } 137 | } 138 | 139 | /** 140 | * 提取用户的单页微博动态 141 | * 142 | * @param uid 微博用户的唯一标识符 143 | * @param containerId 用户动态的容器ID 144 | * @param sinceId 分页信息,上一页最后一条动态的ID 145 | * @returns 包含动态和下一页since_id的PagedFeeds对象 146 | */ 147 | private async extractFeeds(uid: number, containerId: string, sinceId: string): Promise { 148 | try { 149 | const url = FEEDS_URL 150 | .replace('{userId}', uid.toString()) 151 | .replace('{containerId}', containerId) 152 | .replace('{sinceId}', sinceId); 153 | 154 | const response = await axios.get(url, { headers: DEFAULT_HEADERS }); 155 | const data = response.data; 156 | 157 | const newSinceId = data?.data?.cardlistInfo?.since_id || ''; 158 | const cards = data?.data?.cards || []; 159 | 160 | if (cards.length > 0) { 161 | return { SinceId: newSinceId, Feeds: cards }; 162 | } else { 163 | return { SinceId: newSinceId, Feeds: [] }; 164 | } 165 | } catch (error) { 166 | console.error(`无法获取UID为'${uid}'的动态`, error); 167 | return { SinceId: null, Feeds: [] }; 168 | } 169 | } 170 | 171 | /** 172 | * 获取微博热搜榜 173 | * 174 | * @param limit 返回的最大热搜条目数量 175 | * @returns 热搜条目列表 176 | */ 177 | async getHotSearchList(limit: number): Promise { 178 | try { 179 | const response = await axios.get(HOT_SEARCH_URL, { 180 | headers: DEFAULT_HEADERS 181 | }); 182 | 183 | const data = response.data; 184 | const cards = data?.data?.cards || []; 185 | 186 | if (cards.length === 0) { 187 | return []; 188 | } 189 | 190 | // 查找包含热搜数据的card 191 | let hotSearchCard = null; 192 | for (const card of cards) { 193 | if (card.card_group && Array.isArray(card.card_group)) { 194 | hotSearchCard = card; 195 | break; 196 | } 197 | } 198 | 199 | if (!hotSearchCard || !hotSearchCard.card_group) { 200 | return []; 201 | } 202 | 203 | // 转换热搜数据为HotSearchItem格式 204 | const hotSearchItems: HotSearchItem[] = []; 205 | let rank = 1; 206 | 207 | for (const item of hotSearchCard.card_group) { 208 | if (item.desc && rank <= limit) { 209 | const hotSearchItem: HotSearchItem = { 210 | keyword: item.desc, 211 | rank: rank, 212 | hotValue: parseInt(item.desc_extr || '0', 10), 213 | tag: item.icon ? item.icon.slice(item.icon.lastIndexOf('/') + 1).replace('.png', '') : undefined, 214 | url: item.scheme 215 | }; 216 | 217 | hotSearchItems.push(hotSearchItem); 218 | rank++; 219 | } 220 | } 221 | 222 | return hotSearchItems; 223 | } catch (error) { 224 | console.error('无法获取微博热搜榜', error); 225 | return []; 226 | } 227 | } 228 | 229 | /** 230 | * 根据关键词搜索微博内容 231 | * 232 | * @param keyword 搜索关键词 233 | * @param limit 返回的最大微博条目数量 234 | * @param page 起始页码(默认为1) 235 | * @returns 微博内容列表 236 | */ 237 | async searchWeiboContent(keyword: string, limit: number, page: number = 1): Promise { 238 | try { 239 | const results: ContentSearchResult[] = []; 240 | let currentPage = page; 241 | 242 | while (results.length < limit) { 243 | const url = SEARCH_CONTENT_URL 244 | .replace('{keyword}', encodeURIComponent(keyword)) 245 | .replace('{page}', currentPage.toString()); 246 | 247 | const response = await axios.get(url, { 248 | headers: DEFAULT_HEADERS 249 | }); 250 | 251 | const data = response.data; 252 | const cards = data?.data?.cards || []; 253 | 254 | // 微博通常会返回多个卡片,我们寻找包含微博内容的卡片组 255 | let contentCards: any[] = []; 256 | for (const card of cards) { 257 | // 微博内容卡片通常有card_type=9 258 | if (card.card_type === 9) { 259 | contentCards.push(card); 260 | } 261 | // 处理卡片组 262 | else if (card.card_group && Array.isArray(card.card_group)) { 263 | const contentGroup = card.card_group.filter((item: any) => item.card_type === 9); 264 | contentCards = contentCards.concat(contentGroup); 265 | } 266 | } 267 | 268 | if (contentCards.length === 0) { 269 | break; // 没有更多内容,退出循环 270 | } 271 | 272 | // 处理每个内容卡片 273 | for (const card of contentCards) { 274 | if (results.length >= limit) { 275 | break; 276 | } 277 | 278 | const mblog = card.mblog; 279 | if (!mblog) continue; 280 | 281 | // 提取图片链接 282 | const pics: string[] = []; 283 | if (mblog.pics && Array.isArray(mblog.pics)) { 284 | for (const pic of mblog.pics) { 285 | if (pic.url) { 286 | pics.push(pic.url); 287 | } 288 | } 289 | } 290 | 291 | // 提取视频链接 292 | let videoUrl = undefined; 293 | if (mblog.page_info && mblog.page_info.type === 'video') { 294 | videoUrl = mblog.page_info.media_info?.stream_url || 295 | mblog.page_info.urls?.mp4_720p_mp4 || 296 | mblog.page_info.urls?.mp4_hd_mp4 || 297 | mblog.page_info.urls?.mp4_ld_mp4; 298 | } 299 | 300 | // 创建内容搜索结果对象 301 | const contentResult: ContentSearchResult = { 302 | id: mblog.id, 303 | text: mblog.text, 304 | created_at: mblog.created_at, 305 | reposts_count: mblog.reposts_count, 306 | comments_count: mblog.comments_count, 307 | attitudes_count: mblog.attitudes_count, 308 | user: { 309 | id: mblog.user.id, 310 | screen_name: mblog.user.screen_name, 311 | profile_image_url: mblog.user.profile_image_url, 312 | verified: mblog.user.verified 313 | }, 314 | pics: pics.length > 0 ? pics : undefined, 315 | video_url: videoUrl 316 | }; 317 | 318 | results.push(contentResult); 319 | } 320 | 321 | currentPage++; 322 | 323 | // 检查是否有下一页 324 | if (!data?.data?.cardlistInfo?.page || data.data.cardlistInfo.page === "1") { 325 | break; 326 | } 327 | } 328 | 329 | return results.slice(0, limit); 330 | } catch (error) { 331 | console.error(`无法搜索关键词为'${keyword}'的微博内容`, error); 332 | return []; 333 | } 334 | } 335 | } -------------------------------------------------------------------------------- /tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "target": "ES2022", 4 | "module": "NodeNext", 5 | "moduleResolution": "NodeNext", 6 | "esModuleInterop": true, 7 | "strict": true, 8 | "declaration": true, 9 | "sourceMap": true, 10 | "outDir": "dist", 11 | "rootDir": "src", 12 | "skipLibCheck": true 13 | }, 14 | "include": ["src/**/*"], 15 | "exclude": ["node_modules", "**/*.test.ts"] 16 | } --------------------------------------------------------------------------------