@dollhousemcp/mcp-server 2.0.2 → 2.0.3

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 CHANGED
@@ -1,5 +1,31 @@
1
1
  # Changelog
2
2
 
3
+ ## [2.0.3] - 2026-04-02
4
+
5
+ ### Setup Tab — Interactive Installer
6
+
7
+ One command opens a browser-based setup wizard for installing DollhouseMCP on any MCP client:
8
+
9
+ ```bash
10
+ npx @dollhousemcp/mcp-server@latest --web
11
+ ```
12
+
13
+ #### Features
14
+ - **One-click install** for 9 platforms: Claude Desktop, Claude Code, Cursor, VS Code, Codex, Gemini CLI, Windsurf, Cline, LM Studio
15
+ - **Auto-updating vs Pinned version** toggle — all config snippets update dynamically
16
+ - **Installation detection** — scans existing client configs, shows green/amber state based on whether current settings match
17
+ - **Open config file** buttons — opens client config in the system default editor
18
+ - **Version-aware .mcpb download** — resolves correct versioned Desktop Extension from GitHub API
19
+ - **Install verification** — confirms config was written after install
20
+ - **Keyboard navigation** — arrow keys, Home, End for platform tabs
21
+
22
+ #### Technical
23
+ - Bundled `install-mcp` (MIT, by Dhravya Shah) as runtime dependency
24
+ - 7 of 9 platform panels generated from declarative PLATFORMS registry (zero HTML duplication)
25
+ - `UnicodeValidator.normalize()` on all server-side user input
26
+ - 171 tests including JSDOM DOM validation of generated panels
27
+ - All READMEs and guides updated with interactive setup one-liner
28
+
3
29
  ## [2.0.0] - 2026-04-01
4
30
 
5
31
  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
- ### Choose Your Installation Method
340
+ ### Interactive Setup (Recommended)
341
341
 
342
- <table>
343
- <tr>
344
- <th>Method</th>
345
- <th>Best For</th>
346
- <th>Pros</th>
347
- <th>Cons</th>
348
- </tr>
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
- **Claude Desktop** (one-click install):
61
+ **Interactive Setup** (any platform):
62
+
63
+ ```bash
64
+ npx @dollhousemcp/mcp-server@latest --web
65
+ ```
62
66
 
63
- Download the [DollhouseMCP Desktop Extension](https://github.com/DollhouseMCP/mcp-server/releases/tag/v2.0.0) (`.mcpb` file) and open it. Claude Desktop handles the rest no terminal required.
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
- Current project only:
73
- ```bash
74
- claude mcp add dollhousemcp -- npx -y @dollhousemcp/mcp-server
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) for Claude Desktop manual config, Gemini, Cursor, Codex, local LLMs, and more.
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
- **Claude Desktop** (one-click install):
61
+ **Interactive Setup** (any platform):
62
+
63
+ ```bash
64
+ npx @dollhousemcp/mcp-server@latest --web
65
+ ```
62
66
 
63
- Download the [DollhouseMCP Desktop Extension](https://github.com/DollhouseMCP/mcp-server/releases/tag/v2.0.0) (`.mcpb` file) and open it. Claude Desktop handles the rest no terminal required.
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
- Current project only:
73
- ```bash
74
- claude mcp add dollhousemcp -- npx -y @dollhousemcp/mcp-server
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) for Claude Desktop manual config, Gemini, Cursor, Codex, local LLMs, and more.
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
- **Claude Desktop** (one-click install):
61
+ **Interactive Setup** (any platform):
62
+
63
+ ```bash
64
+ npx @dollhousemcp/mcp-server@latest --web
65
+ ```
62
66
 
63
- Download the [DollhouseMCP Desktop Extension](https://github.com/DollhouseMCP/mcp-server/releases/tag/v2.0.0) (`.mcpb` file) and open it. Claude Desktop handles the rest no terminal required.
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
- Current project only:
73
- ```bash
74
- claude mcp add dollhousemcp -- npx -y @dollhousemcp/mcp-server
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) for Claude Desktop manual config, Gemini, Cursor, Codex, local LLMs, and more.
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,3 @@
1
+ export declare const VERSION = "1.6.5";
2
+ export declare const BUILD_DATE = "2025-08-26T15:30:22.187Z";
3
+ //# sourceMappingURL=version.d.ts.map
@@ -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.2";
6
- export declare const BUILD_TIMESTAMP = "2026-04-02T04:05:19.817Z";
5
+ export declare const PACKAGE_VERSION = "2.0.3";
6
+ export declare const BUILD_TIMESTAMP = "2026-04-02T18:11:11.137Z";
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.2';
6
- export const BUILD_TIMESTAMP = '2026-04-02T04:05:19.817Z';
5
+ export const PACKAGE_VERSION = '2.0.3';
6
+ export const BUILD_TIMESTAMP = '2026-04-02T18:11:11.137Z';
7
7
  export const BUILD_TYPE = 'npm';
8
8
  export const PACKAGE_NAME = '@dollhousemcp/mcp-server';
9
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9nZW5lcmF0ZWQvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFFSCxNQUFNLENBQUMsTUFBTSxlQUFlLEdBQUcsT0FBTyxDQUFDO0FBQ3ZDLE1BQU0sQ0FBQyxNQUFNLGVBQWUsR0FBRywwQkFBMEIsQ0FBQztBQUMxRCxNQUFNLENBQUMsTUFBTSxVQUFVLEdBQWtCLEtBQUssQ0FBQztBQUMvQyxNQUFNLENBQUMsTUFBTSxZQUFZLEdBQUcsMEJBQTBCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEF1dG8tZ2VuZXJhdGVkIGZpbGUgLSBETyBOT1QgRURJVFxuICogR2VuZXJhdGVkIGF0IGJ1aWxkIHRpbWUgYnkgc2NyaXB0cy9nZW5lcmF0ZS12ZXJzaW9uLmpzXG4gKi9cblxuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfVkVSU0lPTiA9ICcyLjAuMic7XG5leHBvcnQgY29uc3QgQlVJTERfVElNRVNUQU1QID0gJzIwMjYtMDQtMDJUMDQ6MDU6MTkuODE3Wic7XG5leHBvcnQgY29uc3QgQlVJTERfVFlQRTogJ25wbScgfCAnZ2l0JyA9ICducG0nO1xuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfTkFNRSA9ICdAZG9sbGhvdXNlbWNwL21jcC1zZXJ2ZXInO1xuIl19
9
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9nZW5lcmF0ZWQvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFFSCxNQUFNLENBQUMsTUFBTSxlQUFlLEdBQUcsT0FBTyxDQUFDO0FBQ3ZDLE1BQU0sQ0FBQyxNQUFNLGVBQWUsR0FBRywwQkFBMEIsQ0FBQztBQUMxRCxNQUFNLENBQUMsTUFBTSxVQUFVLEdBQWtCLEtBQUssQ0FBQztBQUMvQyxNQUFNLENBQUMsTUFBTSxZQUFZLEdBQUcsMEJBQTBCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEF1dG8tZ2VuZXJhdGVkIGZpbGUgLSBETyBOT1QgRURJVFxuICogR2VuZXJhdGVkIGF0IGJ1aWxkIHRpbWUgYnkgc2NyaXB0cy9nZW5lcmF0ZS12ZXJzaW9uLmpzXG4gKi9cblxuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfVkVSU0lPTiA9ICcyLjAuMyc7XG5leHBvcnQgY29uc3QgQlVJTERfVElNRVNUQU1QID0gJzIwMjYtMDQtMDJUMTg6MTE6MTEuMTM3Wic7XG5leHBvcnQgY29uc3QgQlVJTERfVFlQRTogJ25wbScgfCAnZ2l0JyA9ICducG0nO1xuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfTkFNRSA9ICdAZG9sbGhvdXNlbWNwL21jcC1zZXJ2ZXInO1xuIl19
@@ -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"}