@dollhousemcp/mcp-server 2.0.2 → 2.0.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +35 -0
- package/README.github.md +8 -33
- package/README.md +10 -8
- package/README.md.backup +10 -8
- package/README.npm.md +10 -8
- package/dist/constants/version.d.ts +3 -0
- package/dist/constants/version.d.ts.map +1 -0
- package/dist/constants/version.js +4 -0
- package/dist/generated/version.d.ts +2 -2
- package/dist/generated/version.js +3 -3
- package/dist/logging/sinks/SSELogSink.d.ts +35 -0
- package/dist/logging/sinks/SSELogSink.d.ts.map +1 -0
- package/dist/logging/sinks/SSELogSink.js +181 -0
- package/dist/logging/viewer/viewerHtml.d.ts +8 -0
- package/dist/logging/viewer/viewerHtml.d.ts.map +1 -0
- package/dist/logging/viewer/viewerHtml.js +204 -0
- package/dist/security/audit/config/suppressions.d.ts.map +1 -1
- package/dist/security/audit/config/suppressions.js +6 -1
- package/dist/seed-elements/memories/dollhousemcp-baseline-knowledge.yaml +149 -0
- package/dist/seed-elements/memories/how-to-create-custom-auto-load-memories.yaml +455 -0
- package/dist/seed-elements/memories/priority-best-practices-for-teams.yaml +542 -0
- package/dist/seed-elements/memories/token-estimation-guidelines.yaml +602 -0
- package/dist/web/public/app.js +29 -10
- package/dist/web/public/fonts/ibmplexmono--F63fjptAgt5VM-kVkqdyU8n1i8q131nj-o.woff2 +0 -0
- package/dist/web/public/fonts/ibmplexmono--F63fjptAgt5VM-kVkqdyU8n1iAq131nj-otFQ.woff2 +0 -0
- package/dist/web/public/fonts/ibmplexmono--F63fjptAgt5VM-kVkqdyU8n1iEq131nj-otFQ.woff2 +0 -0
- package/dist/web/public/fonts/ibmplexmono--F63fjptAgt5VM-kVkqdyU8n1iIq131nj-otFQ.woff2 +0 -0
- package/dist/web/public/fonts/ibmplexmono--F63fjptAgt5VM-kVkqdyU8n1isq131nj-otFQ.woff2 +0 -0
- package/dist/web/public/fonts/ibmplexmono--F6qfjptAgt5VM-kVkqdyU8n3twJwl1FgsAXHNlYzg.woff2 +0 -0
- package/dist/web/public/fonts/ibmplexmono--F6qfjptAgt5VM-kVkqdyU8n3twJwl5FgsAXHNlYzg.woff2 +0 -0
- package/dist/web/public/fonts/ibmplexmono--F6qfjptAgt5VM-kVkqdyU8n3twJwl9FgsAXHNlYzg.woff2 +0 -0
- package/dist/web/public/fonts/ibmplexmono--F6qfjptAgt5VM-kVkqdyU8n3twJwlBFgsAXHNk.woff2 +0 -0
- package/dist/web/public/fonts/ibmplexmono--F6qfjptAgt5VM-kVkqdyU8n3twJwlRFgsAXHNlYzg.woff2 +0 -0
- package/dist/web/public/fonts/manrope-xn7gYHE41ni1AdIRggOxSvfedN62Zw.woff2 +0 -0
- package/dist/web/public/fonts/manrope-xn7gYHE41ni1AdIRggSxSvfedN62Zw.woff2 +0 -0
- package/dist/web/public/fonts/manrope-xn7gYHE41ni1AdIRggexSvfedN4.woff2 +0 -0
- package/dist/web/public/fonts/manrope-xn7gYHE41ni1AdIRggixSvfedN62Zw.woff2 +0 -0
- package/dist/web/public/fonts/manrope-xn7gYHE41ni1AdIRggmxSvfedN62Zw.woff2 +0 -0
- package/dist/web/public/fonts/manrope-xn7gYHE41ni1AdIRggqxSvfedN62Zw.woff2 +0 -0
- package/dist/web/public/fonts/plusjakartasans-LDIoaomQNQcsA88c7O9yZ4KMCoOg4Ko20yygg_vb.woff2 +0 -0
- package/dist/web/public/fonts/plusjakartasans-LDIoaomQNQcsA88c7O9yZ4KMCoOg4Ko40yygg_vbd-E.woff2 +0 -0
- package/dist/web/public/fonts/plusjakartasans-LDIoaomQNQcsA88c7O9yZ4KMCoOg4Ko50yygg_vbd-E.woff2 +0 -0
- package/dist/web/public/fonts/plusjakartasans-LDIoaomQNQcsA88c7O9yZ4KMCoOg4Ko70yygg_vbd-E.woff2 +0 -0
- package/dist/web/public/fonts.css +270 -0
- package/dist/web/public/index.html +365 -0
- package/dist/web/public/logs.css +472 -0
- package/dist/web/public/metrics.css +238 -0
- package/dist/web/public/permissions.css +364 -0
- package/dist/web/public/sessions.css +235 -0
- package/dist/web/public/setup.css +648 -0
- package/dist/web/public/setup.js +752 -0
- package/dist/web/public/styles.css +1717 -0
- package/dist/web/routes/setupRoutes.d.ts +18 -0
- package/dist/web/routes/setupRoutes.d.ts.map +1 -0
- package/dist/web/routes/setupRoutes.js +360 -0
- package/dist/web/server.d.ts.map +1 -1
- package/dist/web/server.js +11 -1
- package/package.json +4 -1
- package/server.json +2 -2
package/CHANGELOG.md
CHANGED
|
@@ -1,5 +1,40 @@
|
|
|
1
1
|
# Changelog
|
|
2
2
|
|
|
3
|
+
## [2.0.4] - 2026-04-02
|
|
4
|
+
|
|
5
|
+
### Hotfix: npm package missing web assets
|
|
6
|
+
|
|
7
|
+
- **Fix**: `package.json` `files` field now includes `dist/web/public/**` and `dist/seed-elements/**`
|
|
8
|
+
- v2.0.3 was missing `.html`, `.css`, and font files from the web console, causing `--web` mode to crash with `NotFoundError`
|
|
9
|
+
- Also missing seed element `.yaml` files, causing seed memory installation to fail
|
|
10
|
+
- Added package inclusion tests to prevent regression
|
|
11
|
+
|
|
12
|
+
## [2.0.3] - 2026-04-02
|
|
13
|
+
|
|
14
|
+
### Setup Tab — Interactive Installer
|
|
15
|
+
|
|
16
|
+
One command opens a browser-based setup wizard for installing DollhouseMCP on any MCP client:
|
|
17
|
+
|
|
18
|
+
```bash
|
|
19
|
+
npx @dollhousemcp/mcp-server@latest --web
|
|
20
|
+
```
|
|
21
|
+
|
|
22
|
+
#### Features
|
|
23
|
+
- **One-click install** for 9 platforms: Claude Desktop, Claude Code, Cursor, VS Code, Codex, Gemini CLI, Windsurf, Cline, LM Studio
|
|
24
|
+
- **Auto-updating vs Pinned version** toggle — all config snippets update dynamically
|
|
25
|
+
- **Installation detection** — scans existing client configs, shows green/amber state based on whether current settings match
|
|
26
|
+
- **Open config file** buttons — opens client config in the system default editor
|
|
27
|
+
- **Version-aware .mcpb download** — resolves correct versioned Desktop Extension from GitHub API
|
|
28
|
+
- **Install verification** — confirms config was written after install
|
|
29
|
+
- **Keyboard navigation** — arrow keys, Home, End for platform tabs
|
|
30
|
+
|
|
31
|
+
#### Technical
|
|
32
|
+
- Bundled `install-mcp` (MIT, by Dhravya Shah) as runtime dependency
|
|
33
|
+
- 7 of 9 platform panels generated from declarative PLATFORMS registry (zero HTML duplication)
|
|
34
|
+
- `UnicodeValidator.normalize()` on all server-side user input
|
|
35
|
+
- 171 tests including JSDOM DOM validation of generated panels
|
|
36
|
+
- All READMEs and guides updated with interactive setup one-liner
|
|
37
|
+
|
|
3
38
|
## [2.0.0] - 2026-04-01
|
|
4
39
|
|
|
5
40
|
DollhouseMCP v2.0.0 is the first stable release of the v2 line. This release brings MCP-AQL (Agent Query Language), Gatekeeper permission system, unified web console, multi-session support, comprehensive metrics, and 9000+ tests across unit, integration, security, and e2e suites.
|
package/README.github.md
CHANGED
|
@@ -337,49 +337,24 @@ Assistant: "I've saved 'Hard SciFi Writer' to your portfolio. You can activate i
|
|
|
337
337
|
|
|
338
338
|
## 📦 Installation
|
|
339
339
|
|
|
340
|
-
###
|
|
340
|
+
### Interactive Setup (Recommended)
|
|
341
341
|
|
|
342
|
-
|
|
343
|
-
|
|
344
|
-
|
|
345
|
-
|
|
346
|
-
|
|
347
|
-
|
|
348
|
-
|
|
349
|
-
<tr>
|
|
350
|
-
<td><strong>Local Install</strong><br>(Recommended)</td>
|
|
351
|
-
<td>Most users, multiple configs, customization</td>
|
|
352
|
-
<td>✅ Multiple setups<br>✅ Easy backup<br>✅ No permissions</td>
|
|
353
|
-
<td>❌ Longer path in config</td>
|
|
354
|
-
</tr>
|
|
355
|
-
<tr>
|
|
356
|
-
<td><strong>npx</strong></td>
|
|
357
|
-
<td>Quick testing, always latest</td>
|
|
358
|
-
<td>✅ No install<br>✅ Always updated</td>
|
|
359
|
-
<td>❌ Slower startup<br>❌ Needs internet</td>
|
|
360
|
-
</tr>
|
|
361
|
-
<tr>
|
|
362
|
-
<td><strong>Global Install</strong></td>
|
|
363
|
-
<td>Single shared instance</td>
|
|
364
|
-
<td>✅ Short command</td>
|
|
365
|
-
<td>❌ Only one version<br>❌ Needs sudo/admin</td>
|
|
366
|
-
</tr>
|
|
367
|
-
</table>
|
|
342
|
+
One command opens a browser-based setup wizard with one-click install for 9 MCP clients:
|
|
343
|
+
|
|
344
|
+
```bash
|
|
345
|
+
npx @dollhousemcp/mcp-server@latest --web
|
|
346
|
+
```
|
|
347
|
+
|
|
348
|
+
Supports Claude Desktop, Claude Code, Cursor, VS Code, Codex, Gemini CLI, Windsurf, Cline, and LM Studio. Detects existing installations, auto-updating and pinned version options.
|
|
368
349
|
|
|
369
350
|
---
|
|
370
351
|
|
|
371
352
|
### Claude Code
|
|
372
353
|
|
|
373
|
-
All projects (recommended):
|
|
374
354
|
```bash
|
|
375
355
|
claude mcp add -s user dollhousemcp -- npx -y @dollhousemcp/mcp-server
|
|
376
356
|
```
|
|
377
357
|
|
|
378
|
-
Current project only:
|
|
379
|
-
```bash
|
|
380
|
-
claude mcp add dollhousemcp -- npx -y @dollhousemcp/mcp-server
|
|
381
|
-
```
|
|
382
|
-
|
|
383
358
|
---
|
|
384
359
|
|
|
385
360
|
### Method 1: Local Installation (Recommended)
|
package/README.md
CHANGED
|
@@ -58,23 +58,25 @@ Your **portfolio** (`~/.dollhouse/portfolio/`) is a local folder that holds all
|
|
|
58
58
|
|
|
59
59
|
DollhouseMCP installs on any MCP-compatible AI client — Claude Code, Claude Desktop, Cursor, Gemini, Codex, and local LLMs. Core element management (create, activate, search, browse) works across all platforms. Advanced features (Gatekeeper confirmation flows, agentic loop execution) have been tested extensively on Claude Code and should work on any client that supports standard MCP tool call/response patterns.
|
|
60
60
|
|
|
61
|
-
**
|
|
61
|
+
**Interactive Setup** (any platform):
|
|
62
|
+
|
|
63
|
+
```bash
|
|
64
|
+
npx @dollhousemcp/mcp-server@latest --web
|
|
65
|
+
```
|
|
62
66
|
|
|
63
|
-
|
|
67
|
+
Opens a browser-based setup wizard with one-click install for Claude Desktop, Claude Code, Cursor, VS Code, Codex, Gemini CLI, Windsurf, Cline, and LM Studio. Detects existing installations, supports auto-updating and pinned versions.
|
|
64
68
|
|
|
65
69
|
**Claude Code** (one command):
|
|
66
70
|
|
|
67
|
-
All projects (recommended):
|
|
68
71
|
```bash
|
|
69
72
|
claude mcp add -s user dollhousemcp -- npx -y @dollhousemcp/mcp-server
|
|
70
73
|
```
|
|
71
74
|
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
```
|
|
75
|
+
**Claude Desktop** (one-click install):
|
|
76
|
+
|
|
77
|
+
Download the [DollhouseMCP Desktop Extension](https://github.com/DollhouseMCP/mcp-server/releases/latest) (`.mcpb` file) and double-click it. Claude Desktop handles the rest — no terminal required.
|
|
76
78
|
|
|
77
|
-
**Other platforms** — see the [Quick Start Guide](docs/guides/quick-start.md)
|
|
79
|
+
**Other platforms** — see the [Quick Start Guide](docs/guides/quick-start.md) or run the interactive setup above.
|
|
78
80
|
|
|
79
81
|
Then start a conversation:
|
|
80
82
|
|
package/README.md.backup
CHANGED
|
@@ -58,23 +58,25 @@ Your **portfolio** (`~/.dollhouse/portfolio/`) is a local folder that holds all
|
|
|
58
58
|
|
|
59
59
|
DollhouseMCP installs on any MCP-compatible AI client — Claude Code, Claude Desktop, Cursor, Gemini, Codex, and local LLMs. Core element management (create, activate, search, browse) works across all platforms. Advanced features (Gatekeeper confirmation flows, agentic loop execution) have been tested extensively on Claude Code and should work on any client that supports standard MCP tool call/response patterns.
|
|
60
60
|
|
|
61
|
-
**
|
|
61
|
+
**Interactive Setup** (any platform):
|
|
62
|
+
|
|
63
|
+
```bash
|
|
64
|
+
npx @dollhousemcp/mcp-server@latest --web
|
|
65
|
+
```
|
|
62
66
|
|
|
63
|
-
|
|
67
|
+
Opens a browser-based setup wizard with one-click install for Claude Desktop, Claude Code, Cursor, VS Code, Codex, Gemini CLI, Windsurf, Cline, and LM Studio. Detects existing installations, supports auto-updating and pinned versions.
|
|
64
68
|
|
|
65
69
|
**Claude Code** (one command):
|
|
66
70
|
|
|
67
|
-
All projects (recommended):
|
|
68
71
|
```bash
|
|
69
72
|
claude mcp add -s user dollhousemcp -- npx -y @dollhousemcp/mcp-server
|
|
70
73
|
```
|
|
71
74
|
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
```
|
|
75
|
+
**Claude Desktop** (one-click install):
|
|
76
|
+
|
|
77
|
+
Download the [DollhouseMCP Desktop Extension](https://github.com/DollhouseMCP/mcp-server/releases/latest) (`.mcpb` file) and double-click it. Claude Desktop handles the rest — no terminal required.
|
|
76
78
|
|
|
77
|
-
**Other platforms** — see the [Quick Start Guide](docs/guides/quick-start.md)
|
|
79
|
+
**Other platforms** — see the [Quick Start Guide](docs/guides/quick-start.md) or run the interactive setup above.
|
|
78
80
|
|
|
79
81
|
Then start a conversation:
|
|
80
82
|
|
package/README.npm.md
CHANGED
|
@@ -58,23 +58,25 @@ Your **portfolio** (`~/.dollhouse/portfolio/`) is a local folder that holds all
|
|
|
58
58
|
|
|
59
59
|
DollhouseMCP installs on any MCP-compatible AI client — Claude Code, Claude Desktop, Cursor, Gemini, Codex, and local LLMs. Core element management (create, activate, search, browse) works across all platforms. Advanced features (Gatekeeper confirmation flows, agentic loop execution) have been tested extensively on Claude Code and should work on any client that supports standard MCP tool call/response patterns.
|
|
60
60
|
|
|
61
|
-
**
|
|
61
|
+
**Interactive Setup** (any platform):
|
|
62
|
+
|
|
63
|
+
```bash
|
|
64
|
+
npx @dollhousemcp/mcp-server@latest --web
|
|
65
|
+
```
|
|
62
66
|
|
|
63
|
-
|
|
67
|
+
Opens a browser-based setup wizard with one-click install for Claude Desktop, Claude Code, Cursor, VS Code, Codex, Gemini CLI, Windsurf, Cline, and LM Studio. Detects existing installations, supports auto-updating and pinned versions.
|
|
64
68
|
|
|
65
69
|
**Claude Code** (one command):
|
|
66
70
|
|
|
67
|
-
All projects (recommended):
|
|
68
71
|
```bash
|
|
69
72
|
claude mcp add -s user dollhousemcp -- npx -y @dollhousemcp/mcp-server
|
|
70
73
|
```
|
|
71
74
|
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
```
|
|
75
|
+
**Claude Desktop** (one-click install):
|
|
76
|
+
|
|
77
|
+
Download the [DollhouseMCP Desktop Extension](https://github.com/DollhouseMCP/mcp-server/releases/latest) (`.mcpb` file) and double-click it. Claude Desktop handles the rest — no terminal required.
|
|
76
78
|
|
|
77
|
-
**Other platforms** — see the [Quick Start Guide](docs/guides/quick-start.md)
|
|
79
|
+
**Other platforms** — see the [Quick Start Guide](docs/guides/quick-start.md) or run the interactive setup above.
|
|
78
80
|
|
|
79
81
|
Then start a conversation:
|
|
80
82
|
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"version.d.ts","sourceRoot":"","sources":["../../src/constants/version.ts"],"names":[],"mappings":"AACA,eAAO,MAAM,OAAO,UAAU,CAAC;AAC/B,eAAO,MAAM,UAAU,6BAA6B,CAAC"}
|
|
@@ -0,0 +1,4 @@
|
|
|
1
|
+
// Auto-generated version constant
|
|
2
|
+
export const VERSION = "1.6.5";
|
|
3
|
+
export const BUILD_DATE = "2025-08-26T15:30:22.187Z";
|
|
4
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9jb25zdGFudHMvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxrQ0FBa0M7QUFDbEMsTUFBTSxDQUFDLE1BQU0sT0FBTyxHQUFHLE9BQU8sQ0FBQztBQUMvQixNQUFNLENBQUMsTUFBTSxVQUFVLEdBQUcsMEJBQTBCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBBdXRvLWdlbmVyYXRlZCB2ZXJzaW9uIGNvbnN0YW50XG5leHBvcnQgY29uc3QgVkVSU0lPTiA9IFwiMS42LjVcIjtcbmV4cG9ydCBjb25zdCBCVUlMRF9EQVRFID0gXCIyMDI1LTA4LTI2VDE1OjMwOjIyLjE4N1pcIjtcbiJdfQ==
|
|
@@ -2,8 +2,8 @@
|
|
|
2
2
|
* Auto-generated file - DO NOT EDIT
|
|
3
3
|
* Generated at build time by scripts/generate-version.js
|
|
4
4
|
*/
|
|
5
|
-
export declare const PACKAGE_VERSION = "2.0.
|
|
6
|
-
export declare const BUILD_TIMESTAMP = "2026-04-
|
|
5
|
+
export declare const PACKAGE_VERSION = "2.0.4";
|
|
6
|
+
export declare const BUILD_TIMESTAMP = "2026-04-02T18:33:53.805Z";
|
|
7
7
|
export declare const BUILD_TYPE: 'npm' | 'git';
|
|
8
8
|
export declare const PACKAGE_NAME = "@dollhousemcp/mcp-server";
|
|
9
9
|
//# sourceMappingURL=version.d.ts.map
|
|
@@ -2,8 +2,8 @@
|
|
|
2
2
|
* Auto-generated file - DO NOT EDIT
|
|
3
3
|
* Generated at build time by scripts/generate-version.js
|
|
4
4
|
*/
|
|
5
|
-
export const PACKAGE_VERSION = '2.0.
|
|
6
|
-
export const BUILD_TIMESTAMP = '2026-04-
|
|
5
|
+
export const PACKAGE_VERSION = '2.0.4';
|
|
6
|
+
export const BUILD_TIMESTAMP = '2026-04-02T18:33:53.805Z';
|
|
7
7
|
export const BUILD_TYPE = 'npm';
|
|
8
8
|
export const PACKAGE_NAME = '@dollhousemcp/mcp-server';
|
|
9
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
9
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9nZW5lcmF0ZWQvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFFSCxNQUFNLENBQUMsTUFBTSxlQUFlLEdBQUcsT0FBTyxDQUFDO0FBQ3ZDLE1BQU0sQ0FBQyxNQUFNLGVBQWUsR0FBRywwQkFBMEIsQ0FBQztBQUMxRCxNQUFNLENBQUMsTUFBTSxVQUFVLEdBQWtCLEtBQUssQ0FBQztBQUMvQyxNQUFNLENBQUMsTUFBTSxZQUFZLEdBQUcsMEJBQTBCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEF1dG8tZ2VuZXJhdGVkIGZpbGUgLSBETyBOT1QgRURJVFxuICogR2VuZXJhdGVkIGF0IGJ1aWxkIHRpbWUgYnkgc2NyaXB0cy9nZW5lcmF0ZS12ZXJzaW9uLmpzXG4gKi9cblxuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfVkVSU0lPTiA9ICcyLjAuNCc7XG5leHBvcnQgY29uc3QgQlVJTERfVElNRVNUQU1QID0gJzIwMjYtMDQtMDJUMTg6MzM6NTMuODA1Wic7XG5leHBvcnQgY29uc3QgQlVJTERfVFlQRTogJ25wbScgfCAnZ2l0JyA9ICducG0nO1xuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfTkFNRSA9ICdAZG9sbGhvdXNlbWNwL21jcC1zZXJ2ZXInO1xuIl19
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* SSE-based real-time log viewer sink.
|
|
3
|
+
*
|
|
4
|
+
* Implements ILogSink and runs an opt-in Express HTTP server that:
|
|
5
|
+
* - Serves a browser-based log viewer at GET /
|
|
6
|
+
* - Streams log entries via SSE at GET /logs/stream
|
|
7
|
+
* - Exposes a JSON query endpoint at GET /logs (delegates to MemoryLogSink)
|
|
8
|
+
* - Provides a health endpoint at GET /health
|
|
9
|
+
*
|
|
10
|
+
* See docs/LOGGING-DESIGN.md §4.6 for the full design.
|
|
11
|
+
*/
|
|
12
|
+
import type { ILogSink, UnifiedLogEntry } from '../types.js';
|
|
13
|
+
import type { MemoryLogSink } from './MemoryLogSink.js';
|
|
14
|
+
export interface SSELogSinkOptions {
|
|
15
|
+
port: number;
|
|
16
|
+
memorySink: MemoryLogSink;
|
|
17
|
+
}
|
|
18
|
+
export declare class SSELogSink implements ILogSink {
|
|
19
|
+
private readonly app;
|
|
20
|
+
private server;
|
|
21
|
+
private readonly clients;
|
|
22
|
+
private readonly memorySink;
|
|
23
|
+
private readonly port;
|
|
24
|
+
private readonly startTime;
|
|
25
|
+
constructor(options: SSELogSinkOptions);
|
|
26
|
+
write(entry: UnifiedLogEntry): void;
|
|
27
|
+
flush(): Promise<void>;
|
|
28
|
+
close(): Promise<void>;
|
|
29
|
+
start(): Promise<void>;
|
|
30
|
+
get clientCount(): number;
|
|
31
|
+
getPort(): number;
|
|
32
|
+
private setupRoutes;
|
|
33
|
+
private matchesFilter;
|
|
34
|
+
}
|
|
35
|
+
//# sourceMappingURL=SSELogSink.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"SSELogSink.d.ts","sourceRoot":"","sources":["../../../src/logging/sinks/SSELogSink.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;GAUG;AAKH,OAAO,KAAK,EAAE,QAAQ,EAAE,eAAe,EAAyB,MAAM,aAAa,CAAC;AAEpF,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,oBAAoB,CAAC;AAGxD,MAAM,WAAW,iBAAiB;IAChC,IAAI,EAAE,MAAM,CAAC;IACb,UAAU,EAAE,aAAa,CAAC;CAC3B;AAcD,qBAAa,UAAW,YAAW,QAAQ;IACzC,OAAO,CAAC,QAAQ,CAAC,GAAG,CAA6B;IACjD,OAAO,CAAC,MAAM,CAAuB;IACrC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAwB;IAChD,OAAO,CAAC,QAAQ,CAAC,UAAU,CAAgB;IAC3C,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAS;IAC9B,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAc;gBAE5B,OAAO,EAAE,iBAAiB;IAWtC,KAAK,CAAC,KAAK,EAAE,eAAe,GAAG,IAAI;IAQ7B,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAItB,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAoBtB,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAS5B,IAAI,WAAW,IAAI,MAAM,CAExB;IAED,OAAO,IAAI,MAAM;IAWjB,OAAO,CAAC,WAAW;IA4FnB,OAAO,CAAC,aAAa;CAkBtB"}
|
|
@@ -0,0 +1,181 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* SSE-based real-time log viewer sink.
|
|
3
|
+
*
|
|
4
|
+
* Implements ILogSink and runs an opt-in Express HTTP server that:
|
|
5
|
+
* - Serves a browser-based log viewer at GET /
|
|
6
|
+
* - Streams log entries via SSE at GET /logs/stream
|
|
7
|
+
* - Exposes a JSON query endpoint at GET /logs (delegates to MemoryLogSink)
|
|
8
|
+
* - Provides a health endpoint at GET /health
|
|
9
|
+
*
|
|
10
|
+
* See docs/LOGGING-DESIGN.md §4.6 for the full design.
|
|
11
|
+
*/
|
|
12
|
+
import express from 'express';
|
|
13
|
+
import { LOG_LEVEL_PRIORITY } from '../types.js';
|
|
14
|
+
import { getViewerHtml } from '../viewer/viewerHtml.js';
|
|
15
|
+
export class SSELogSink {
|
|
16
|
+
app;
|
|
17
|
+
server = null;
|
|
18
|
+
clients = new Set();
|
|
19
|
+
memorySink;
|
|
20
|
+
port;
|
|
21
|
+
startTime = Date.now();
|
|
22
|
+
constructor(options) {
|
|
23
|
+
this.port = options.port;
|
|
24
|
+
this.memorySink = options.memorySink;
|
|
25
|
+
this.app = express();
|
|
26
|
+
this.setupRoutes();
|
|
27
|
+
}
|
|
28
|
+
// ---------------------------------------------------------------------------
|
|
29
|
+
// ILogSink
|
|
30
|
+
// ---------------------------------------------------------------------------
|
|
31
|
+
write(entry) {
|
|
32
|
+
for (const client of this.clients) {
|
|
33
|
+
if (this.matchesFilter(entry, client.filter)) {
|
|
34
|
+
client.res.write(`data: ${JSON.stringify(entry)}\n\n`);
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
async flush() {
|
|
39
|
+
// No-op — SSE writes are immediate.
|
|
40
|
+
}
|
|
41
|
+
async close() {
|
|
42
|
+
// End all client connections
|
|
43
|
+
for (const client of this.clients) {
|
|
44
|
+
client.res.end();
|
|
45
|
+
}
|
|
46
|
+
this.clients.clear();
|
|
47
|
+
// Shut down HTTP server
|
|
48
|
+
if (this.server) {
|
|
49
|
+
await new Promise((resolve) => {
|
|
50
|
+
this.server.close(() => resolve());
|
|
51
|
+
});
|
|
52
|
+
this.server = null;
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
// ---------------------------------------------------------------------------
|
|
56
|
+
// Lifecycle
|
|
57
|
+
// ---------------------------------------------------------------------------
|
|
58
|
+
async start() {
|
|
59
|
+
return new Promise((resolve) => {
|
|
60
|
+
this.server = this.app.listen(this.port, '127.0.0.1', () => {
|
|
61
|
+
this.server.unref();
|
|
62
|
+
resolve();
|
|
63
|
+
});
|
|
64
|
+
});
|
|
65
|
+
}
|
|
66
|
+
get clientCount() {
|
|
67
|
+
return this.clients.size;
|
|
68
|
+
}
|
|
69
|
+
getPort() {
|
|
70
|
+
if (!this.server)
|
|
71
|
+
return this.port;
|
|
72
|
+
const addr = this.server.address();
|
|
73
|
+
if (addr && typeof addr === 'object')
|
|
74
|
+
return addr.port;
|
|
75
|
+
return this.port;
|
|
76
|
+
}
|
|
77
|
+
// ---------------------------------------------------------------------------
|
|
78
|
+
// Routes
|
|
79
|
+
// ---------------------------------------------------------------------------
|
|
80
|
+
setupRoutes() {
|
|
81
|
+
// Viewer HTML
|
|
82
|
+
this.app.get('/', (_req, res) => {
|
|
83
|
+
const actualPort = this.getPort();
|
|
84
|
+
res.type('html').send(getViewerHtml(actualPort));
|
|
85
|
+
});
|
|
86
|
+
// SSE stream
|
|
87
|
+
this.app.get('/logs/stream', (req, res) => {
|
|
88
|
+
res.writeHead(200, {
|
|
89
|
+
'Content-Type': 'text/event-stream',
|
|
90
|
+
'Cache-Control': 'no-cache',
|
|
91
|
+
'Connection': 'keep-alive',
|
|
92
|
+
});
|
|
93
|
+
res.write(':connected\n\n');
|
|
94
|
+
const filter = {};
|
|
95
|
+
if (typeof req.query['category'] === 'string' && req.query['category']) {
|
|
96
|
+
filter.category = req.query['category'];
|
|
97
|
+
}
|
|
98
|
+
if (typeof req.query['level'] === 'string' && req.query['level']) {
|
|
99
|
+
filter.level = req.query['level'];
|
|
100
|
+
}
|
|
101
|
+
if (typeof req.query['source'] === 'string' && req.query['source']) {
|
|
102
|
+
filter.source = req.query['source'];
|
|
103
|
+
}
|
|
104
|
+
if (typeof req.query['correlationId'] === 'string' && req.query['correlationId']) {
|
|
105
|
+
filter.correlationId = req.query['correlationId'];
|
|
106
|
+
}
|
|
107
|
+
const client = { res, filter };
|
|
108
|
+
this.clients.add(client);
|
|
109
|
+
// Backfill recent history so the viewer shows context on connect
|
|
110
|
+
const history = this.memorySink.query({ category: 'all', limit: 500 });
|
|
111
|
+
// Send oldest-first so the viewer displays in chronological order
|
|
112
|
+
const entries = history.entries.slice().reverse();
|
|
113
|
+
for (const entry of entries) {
|
|
114
|
+
res.write(`data: ${JSON.stringify(entry)}\n\n`);
|
|
115
|
+
}
|
|
116
|
+
req.on('close', () => {
|
|
117
|
+
this.clients.delete(client);
|
|
118
|
+
});
|
|
119
|
+
});
|
|
120
|
+
// JSON query (delegates to MemoryLogSink)
|
|
121
|
+
this.app.get('/logs', (req, res) => {
|
|
122
|
+
const options = {};
|
|
123
|
+
if (typeof req.query['category'] === 'string' && req.query['category']) {
|
|
124
|
+
options['category'] = req.query['category'];
|
|
125
|
+
}
|
|
126
|
+
if (typeof req.query['level'] === 'string' && req.query['level']) {
|
|
127
|
+
options['level'] = req.query['level'];
|
|
128
|
+
}
|
|
129
|
+
if (typeof req.query['source'] === 'string' && req.query['source']) {
|
|
130
|
+
options['source'] = req.query['source'];
|
|
131
|
+
}
|
|
132
|
+
if (typeof req.query['message'] === 'string' && req.query['message']) {
|
|
133
|
+
options['message'] = req.query['message'];
|
|
134
|
+
}
|
|
135
|
+
if (typeof req.query['limit'] === 'string') {
|
|
136
|
+
options['limit'] = parseInt(req.query['limit'], 10);
|
|
137
|
+
}
|
|
138
|
+
if (typeof req.query['offset'] === 'string') {
|
|
139
|
+
options['offset'] = parseInt(req.query['offset'], 10);
|
|
140
|
+
}
|
|
141
|
+
if (typeof req.query['since'] === 'string' && req.query['since']) {
|
|
142
|
+
options['since'] = req.query['since'];
|
|
143
|
+
}
|
|
144
|
+
if (typeof req.query['until'] === 'string' && req.query['until']) {
|
|
145
|
+
options['until'] = req.query['until'];
|
|
146
|
+
}
|
|
147
|
+
const result = this.memorySink.query(options);
|
|
148
|
+
res.json(result);
|
|
149
|
+
});
|
|
150
|
+
// Health
|
|
151
|
+
this.app.get('/health', (_req, res) => {
|
|
152
|
+
res.json({
|
|
153
|
+
status: 'ok',
|
|
154
|
+
clients: this.clientCount,
|
|
155
|
+
uptime: Math.floor((Date.now() - this.startTime) / 1000),
|
|
156
|
+
});
|
|
157
|
+
});
|
|
158
|
+
}
|
|
159
|
+
// ---------------------------------------------------------------------------
|
|
160
|
+
// Filter matching
|
|
161
|
+
// ---------------------------------------------------------------------------
|
|
162
|
+
matchesFilter(entry, filter) {
|
|
163
|
+
if (filter.category && entry.category !== filter.category) {
|
|
164
|
+
return false;
|
|
165
|
+
}
|
|
166
|
+
if (filter.level && LOG_LEVEL_PRIORITY[entry.level] < LOG_LEVEL_PRIORITY[filter.level]) {
|
|
167
|
+
return false;
|
|
168
|
+
}
|
|
169
|
+
if (filter.source) {
|
|
170
|
+
const needle = filter.source.toLowerCase();
|
|
171
|
+
if (!entry.source.toLowerCase().includes(needle)) {
|
|
172
|
+
return false;
|
|
173
|
+
}
|
|
174
|
+
}
|
|
175
|
+
if (filter.correlationId && entry.correlationId !== filter.correlationId) {
|
|
176
|
+
return false;
|
|
177
|
+
}
|
|
178
|
+
return true;
|
|
179
|
+
}
|
|
180
|
+
}
|
|
181
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"SSELogSink.js","sourceRoot":"","sources":["../../../src/logging/sinks/SSELogSink.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * SSE-based real-time log viewer sink.\n *\n * Implements ILogSink and runs an opt-in Express HTTP server that:\n * - Serves a browser-based log viewer at GET /\n * - Streams log entries via SSE at GET /logs/stream\n * - Exposes a JSON query endpoint at GET /logs (delegates to MemoryLogSink)\n * - Provides a health endpoint at GET /health\n *\n * See docs/LOGGING-DESIGN.md §4.6 for the full design.\n */\n\nimport express from 'express';\nimport type { Request, Response } from 'express';\nimport type { Server } from 'http';\nimport type { ILogSink, UnifiedLogEntry, LogCategory, LogLevel } from '../types.js';\nimport { LOG_LEVEL_PRIORITY } from '../types.js';\nimport type { MemoryLogSink } from './MemoryLogSink.js';\nimport { getViewerHtml } from '../viewer/viewerHtml.js';\n\nexport interface SSELogSinkOptions {\n  port: number;\n  memorySink: MemoryLogSink;\n}\n\ninterface SSEClientFilter {\n  category?: LogCategory;\n  level?: LogLevel;\n  source?: string;\n  correlationId?: string;\n}\n\ninterface SSEClient {\n  res: Response;\n  filter: SSEClientFilter;\n}\n\nexport class SSELogSink implements ILogSink {\n  private readonly app: ReturnType<typeof express>;\n  private server: Server | null = null;\n  private readonly clients = new Set<SSEClient>();\n  private readonly memorySink: MemoryLogSink;\n  private readonly port: number;\n  private readonly startTime = Date.now();\n\n  constructor(options: SSELogSinkOptions) {\n    this.port = options.port;\n    this.memorySink = options.memorySink;\n    this.app = express();\n    this.setupRoutes();\n  }\n\n  // ---------------------------------------------------------------------------\n  // ILogSink\n  // ---------------------------------------------------------------------------\n\n  write(entry: UnifiedLogEntry): void {\n    for (const client of this.clients) {\n      if (this.matchesFilter(entry, client.filter)) {\n        client.res.write(`data: ${JSON.stringify(entry)}\\n\\n`);\n      }\n    }\n  }\n\n  async flush(): Promise<void> {\n    // No-op — SSE writes are immediate.\n  }\n\n  async close(): Promise<void> {\n    // End all client connections\n    for (const client of this.clients) {\n      client.res.end();\n    }\n    this.clients.clear();\n\n    // Shut down HTTP server\n    if (this.server) {\n      await new Promise<void>((resolve) => {\n        this.server!.close(() => resolve());\n      });\n      this.server = null;\n    }\n  }\n\n  // ---------------------------------------------------------------------------\n  // Lifecycle\n  // ---------------------------------------------------------------------------\n\n  async start(): Promise<void> {\n    return new Promise<void>((resolve) => {\n      this.server = this.app.listen(this.port, '127.0.0.1', () => {\n        this.server!.unref();\n        resolve();\n      });\n    });\n  }\n\n  get clientCount(): number {\n    return this.clients.size;\n  }\n\n  getPort(): number {\n    if (!this.server) return this.port;\n    const addr = this.server.address();\n    if (addr && typeof addr === 'object') return addr.port;\n    return this.port;\n  }\n\n  // ---------------------------------------------------------------------------\n  // Routes\n  // ---------------------------------------------------------------------------\n\n  private setupRoutes(): void {\n    // Viewer HTML\n    this.app.get('/', (_req: Request, res: Response) => {\n      const actualPort = this.getPort();\n      res.type('html').send(getViewerHtml(actualPort));\n    });\n\n    // SSE stream\n    this.app.get('/logs/stream', (req: Request, res: Response) => {\n      res.writeHead(200, {\n        'Content-Type': 'text/event-stream',\n        'Cache-Control': 'no-cache',\n        'Connection': 'keep-alive',\n      });\n      res.write(':connected\\n\\n');\n\n      const filter: SSEClientFilter = {};\n      if (typeof req.query['category'] === 'string' && req.query['category']) {\n        filter.category = req.query['category'] as LogCategory;\n      }\n      if (typeof req.query['level'] === 'string' && req.query['level']) {\n        filter.level = req.query['level'] as LogLevel;\n      }\n      if (typeof req.query['source'] === 'string' && req.query['source']) {\n        filter.source = req.query['source'];\n      }\n      if (typeof req.query['correlationId'] === 'string' && req.query['correlationId']) {\n        filter.correlationId = req.query['correlationId'];\n      }\n\n      const client: SSEClient = { res, filter };\n      this.clients.add(client);\n\n      // Backfill recent history so the viewer shows context on connect\n      const history = this.memorySink.query({ category: 'all', limit: 500 });\n      // Send oldest-first so the viewer displays in chronological order\n      const entries = history.entries.slice().reverse();\n      for (const entry of entries) {\n        res.write(`data: ${JSON.stringify(entry)}\\n\\n`);\n      }\n\n      req.on('close', () => {\n        this.clients.delete(client);\n      });\n    });\n\n    // JSON query (delegates to MemoryLogSink)\n    this.app.get('/logs', (req: Request, res: Response) => {\n      const options: Record<string, unknown> = {};\n      if (typeof req.query['category'] === 'string' && req.query['category']) {\n        options['category'] = req.query['category'];\n      }\n      if (typeof req.query['level'] === 'string' && req.query['level']) {\n        options['level'] = req.query['level'];\n      }\n      if (typeof req.query['source'] === 'string' && req.query['source']) {\n        options['source'] = req.query['source'];\n      }\n      if (typeof req.query['message'] === 'string' && req.query['message']) {\n        options['message'] = req.query['message'];\n      }\n      if (typeof req.query['limit'] === 'string') {\n        options['limit'] = parseInt(req.query['limit'], 10);\n      }\n      if (typeof req.query['offset'] === 'string') {\n        options['offset'] = parseInt(req.query['offset'], 10);\n      }\n      if (typeof req.query['since'] === 'string' && req.query['since']) {\n        options['since'] = req.query['since'];\n      }\n      if (typeof req.query['until'] === 'string' && req.query['until']) {\n        options['until'] = req.query['until'];\n      }\n\n      const result = this.memorySink.query(options);\n      res.json(result);\n    });\n\n    // Health\n    this.app.get('/health', (_req: Request, res: Response) => {\n      res.json({\n        status: 'ok',\n        clients: this.clientCount,\n        uptime: Math.floor((Date.now() - this.startTime) / 1000),\n      });\n    });\n  }\n\n  // ---------------------------------------------------------------------------\n  // Filter matching\n  // ---------------------------------------------------------------------------\n\n  private matchesFilter(entry: UnifiedLogEntry, filter: SSEClientFilter): boolean {\n    if (filter.category && entry.category !== filter.category) {\n      return false;\n    }\n    if (filter.level && LOG_LEVEL_PRIORITY[entry.level] < LOG_LEVEL_PRIORITY[filter.level]) {\n      return false;\n    }\n    if (filter.source) {\n      const needle = filter.source.toLowerCase();\n      if (!entry.source.toLowerCase().includes(needle)) {\n        return false;\n      }\n    }\n    if (filter.correlationId && entry.correlationId !== filter.correlationId) {\n      return false;\n    }\n    return true;\n  }\n}\n"]}
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Embedded HTML template for the DollhouseMCP Log Viewer.
|
|
3
|
+
*
|
|
4
|
+
* Returns a self-contained vanilla JS/CSS page that connects to the
|
|
5
|
+
* SSELogSink's /logs/stream endpoint via EventSource. See docs/LOGGING-DESIGN.md §4.6.
|
|
6
|
+
*/
|
|
7
|
+
export declare function getViewerHtml(port: number): string;
|
|
8
|
+
//# sourceMappingURL=viewerHtml.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"viewerHtml.d.ts","sourceRoot":"","sources":["../../../src/logging/viewer/viewerHtml.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAEH,wBAAgB,aAAa,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,CAoMlD"}
|