@dollhousemcp/mcp-server 2.0.10 → 2.0.11

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.
Files changed (36) hide show
  1. package/dist/auto-dollhouse/portDiscovery.d.ts +23 -0
  2. package/dist/auto-dollhouse/portDiscovery.d.ts.map +1 -0
  3. package/dist/auto-dollhouse/portDiscovery.js +77 -0
  4. package/dist/cli/console-token.d.ts +18 -0
  5. package/dist/cli/console-token.d.ts.map +1 -0
  6. package/dist/cli/console-token.js +187 -0
  7. package/dist/generated/version.d.ts +2 -2
  8. package/dist/generated/version.js +3 -3
  9. package/dist/index.d.ts.map +1 -1
  10. package/dist/index.js +24 -5
  11. package/dist/web/console/consoleToken.d.ts +403 -0
  12. package/dist/web/console/consoleToken.d.ts.map +1 -0
  13. package/dist/web/console/consoleToken.js +930 -0
  14. package/dist/web/middleware/authMiddleware.d.ts +64 -0
  15. package/dist/web/middleware/authMiddleware.d.ts.map +1 -0
  16. package/dist/web/middleware/authMiddleware.js +174 -0
  17. package/dist/web/routes/consoleRouteHelpers.d.ts +33 -0
  18. package/dist/web/routes/consoleRouteHelpers.d.ts.map +1 -0
  19. package/dist/web/routes/consoleRouteHelpers.js +60 -0
  20. package/dist/web/routes/tokenRoutes.d.ts +37 -0
  21. package/dist/web/routes/tokenRoutes.d.ts.map +1 -0
  22. package/dist/web/routes/tokenRoutes.js +95 -0
  23. package/dist/web/routes/totpRoutes.d.ts +45 -0
  24. package/dist/web/routes/totpRoutes.d.ts.map +1 -0
  25. package/dist/web/routes/totpRoutes.js +187 -0
  26. package/package.json +1 -1
  27. package/server.json +2 -2
  28. package/dist/constants/version.d.ts +0 -3
  29. package/dist/constants/version.d.ts.map +0 -1
  30. package/dist/constants/version.js +0 -4
  31. package/dist/logging/sinks/SSELogSink.d.ts +0 -35
  32. package/dist/logging/sinks/SSELogSink.d.ts.map +0 -1
  33. package/dist/logging/sinks/SSELogSink.js +0 -181
  34. package/dist/logging/viewer/viewerHtml.d.ts +0 -8
  35. package/dist/logging/viewer/viewerHtml.d.ts.map +0 -1
  36. package/dist/logging/viewer/viewerHtml.js +0 -204
@@ -0,0 +1,187 @@
1
+ /**
2
+ * TOTP (authenticator) enrollment HTTP routes — Phase 2 of #1780 (#1794).
3
+ *
4
+ * Provides:
5
+ * - GET /api/console/totp/status — enrollment state (no secrets)
6
+ * - POST /api/console/totp/enroll/begin — generate secret, return QR + otpauth URI
7
+ * - POST /api/console/totp/enroll/confirm — verify code, persist, return backup codes (once)
8
+ * - POST /api/console/totp/disable — verify code, clear enrollment
9
+ *
10
+ * Security model:
11
+ * - All endpoints require a valid existing console token. The caller must
12
+ * prove they already hold the token before they can enroll a second
13
+ * factor — otherwise an attacker with local port access could pre-enroll
14
+ * their own authenticator and lock the legitimate user out.
15
+ * - Enforcement happens via an always-on `createAuthMiddleware` instance
16
+ * mounted at the top of this router, independent of the global
17
+ * DOLLHOUSE_WEB_AUTH_ENABLED flag.
18
+ * - Backup codes are returned in plaintext exactly once (confirm response)
19
+ * and only their sha256 hashes are retained by the store.
20
+ * - A sliding-window rate limit throttles confirm/disable attempts on a
21
+ * per-IP basis so a bad actor with a live session can't brute-force a
22
+ * TOTP window by flooding requests.
23
+ *
24
+ * @since v2.1.0 — Issue #1794
25
+ */
26
+ import express, { Router } from 'express';
27
+ import QRCode from 'qrcode';
28
+ import { TotpError } from '../console/consoleToken.js';
29
+ import { createAuthMiddleware } from '../middleware/authMiddleware.js';
30
+ import { SlidingWindowRateLimiter } from '../../utils/SlidingWindowRateLimiter.js';
31
+ import { httpStatusForStoreError, sendStoreError, getNormalizedStringField } from './consoleRouteHelpers.js';
32
+ import { logger } from '../../utils/logger.js';
33
+ /** JSON body size limit — TOTP requests are tiny, cap hard. */
34
+ const BODY_LIMIT = '1kb';
35
+ /**
36
+ * Default rate limit for code-verification endpoints: 10 attempts per minute.
37
+ * TOTP codes are 6 digits (1-in-10^6 guess rate) and the ±30s window gives
38
+ * an attacker up to 3 valid codes per minute. 10 attempts caps brute-force
39
+ * success probability per minute at ~3e-5 even before network latency.
40
+ *
41
+ * Limiters are per-endpoint (confirm vs disable) because they protect
42
+ * different secrets: confirm tests the in-memory pending enrollment secret
43
+ * (bounded lifetime, attacker must also know the pendingId), disable tests
44
+ * the persisted enrollment secret (long-lived). A single shared limiter
45
+ * would let traffic on one endpoint exhaust budget on the other.
46
+ *
47
+ * Limiters are global (not per-IP) because the server binds to 127.0.0.1
48
+ * only — every request comes from the same loopback address, so keying on
49
+ * IP would collapse to a single bucket anyway.
50
+ *
51
+ * Tests construct a fresh router per `buildApp`, which yields fresh
52
+ * limiters, so no cross-test pollution.
53
+ */
54
+ const DEFAULT_RATE_LIMIT_MAX = 10;
55
+ const DEFAULT_RATE_LIMIT_WINDOW_MS = 60_000;
56
+ /**
57
+ * Render an otpauth URI as an SVG data URL suitable for direct embedding
58
+ * in an <img src> or background-image. Separated into a helper so the
59
+ * request handler stays readable.
60
+ */
61
+ async function renderQrDataUrl(otpauthUri) {
62
+ // errorCorrectionLevel 'M' is the default and balances size vs robustness.
63
+ // We emit SVG (not PNG) because it scales to any container size and is
64
+ // smaller on the wire for this particular payload.
65
+ const svg = await QRCode.toString(otpauthUri, { type: 'svg', margin: 1 });
66
+ return `data:image/svg+xml;utf8,${encodeURIComponent(svg)}`;
67
+ }
68
+ /**
69
+ * Build the Express router exposing TOTP endpoints. The returned router
70
+ * should be mounted at `/api/console/totp`; the caller does not need to
71
+ * add additional auth middleware — this router enforces its own auth
72
+ * regardless of the global feature flag.
73
+ */
74
+ export function createTotpRoutes(options) {
75
+ const { store } = options;
76
+ const router = Router();
77
+ const jsonParser = express.json({ limit: BODY_LIMIT, type: 'application/json' });
78
+ // Fresh per-endpoint limiters per router instance. Separate buckets for
79
+ // confirm vs disable — they protect different secrets, so traffic on one
80
+ // should not exhaust the budget of the other. Fresh instances per call
81
+ // keep tests isolated.
82
+ const rateLimitMax = options.rateLimitMax ?? DEFAULT_RATE_LIMIT_MAX;
83
+ const rateLimitWindowMs = options.rateLimitWindowMs ?? DEFAULT_RATE_LIMIT_WINDOW_MS;
84
+ const confirmLimiter = new SlidingWindowRateLimiter(rateLimitMax, rateLimitWindowMs);
85
+ const disableLimiter = new SlidingWindowRateLimiter(rateLimitMax, rateLimitWindowMs);
86
+ // Always-on auth — the global feature flag does not apply here. Even
87
+ // during Phase 1 rollout (flag off), the TOTP endpoints must verify that
88
+ // the caller already holds the console token before letting them enroll
89
+ // a second factor. Otherwise an attacker with local port access could
90
+ // enroll their own authenticator and lock the real user out.
91
+ const auth = createAuthMiddleware({
92
+ store,
93
+ enabled: true,
94
+ label: 'totp',
95
+ });
96
+ router.use(auth);
97
+ /** GET /status — enrollment state (no secret material). */
98
+ router.get('/status', (_req, res) => {
99
+ res.json(store.getTotpStatus());
100
+ });
101
+ /** POST /enroll/begin — generate pending secret, return QR + URI. */
102
+ router.post('/enroll/begin', jsonParser, async (req, res) => {
103
+ // Optional label override — lets the UI label the authenticator entry
104
+ // differently if the user has renamed the console token.
105
+ const label = getNormalizedStringField(req.body, 'label') ?? undefined;
106
+ try {
107
+ const begin = store.beginTotpEnrollment(label);
108
+ const qrSvgDataUrl = await renderQrDataUrl(begin.otpauthUri);
109
+ res.json({
110
+ pendingId: begin.pendingId,
111
+ secret: begin.secret,
112
+ otpauthUri: begin.otpauthUri,
113
+ qrSvgDataUrl,
114
+ expiresAt: begin.expiresAt,
115
+ });
116
+ }
117
+ catch (err) {
118
+ const message = err instanceof Error ? err.message : 'Enrollment could not be started';
119
+ logger.debug(`[TOTP] begin failed: ${message}`);
120
+ if (err instanceof TotpError) {
121
+ sendStoreError(res, httpStatusForStoreError(err.code), err.code, err.message);
122
+ }
123
+ else {
124
+ sendStoreError(res, 500, 'INTERNAL', message);
125
+ }
126
+ }
127
+ });
128
+ /** POST /enroll/confirm — verify code, persist, return backup codes (once). */
129
+ router.post('/enroll/confirm', jsonParser, async (req, res) => {
130
+ if (!confirmLimiter.tryAcquire()) {
131
+ sendStoreError(res, 429, 'RATE_LIMITED', 'Too many confirmation attempts — slow down');
132
+ return;
133
+ }
134
+ const pendingId = getNormalizedStringField(req.body, 'pendingId');
135
+ const code = getNormalizedStringField(req.body, 'code');
136
+ if (!pendingId || !code) {
137
+ sendStoreError(res, 400, 'MISSING_FIELDS', 'pendingId and code are required');
138
+ return;
139
+ }
140
+ try {
141
+ const result = await store.confirmTotpEnrollment(pendingId, code);
142
+ res.json({
143
+ enrolled: true,
144
+ enrolledAt: result.enrolledAt,
145
+ backupCodes: result.backupCodes,
146
+ });
147
+ }
148
+ catch (err) {
149
+ const message = err instanceof Error ? err.message : 'Confirmation failed';
150
+ logger.debug(`[TOTP] confirm failed: ${message}`);
151
+ if (err instanceof TotpError) {
152
+ sendStoreError(res, httpStatusForStoreError(err.code), err.code, err.message);
153
+ }
154
+ else {
155
+ sendStoreError(res, 500, 'INTERNAL', message);
156
+ }
157
+ }
158
+ });
159
+ /** POST /disable — verify code, clear enrollment. */
160
+ router.post('/disable', jsonParser, async (req, res) => {
161
+ if (!disableLimiter.tryAcquire()) {
162
+ sendStoreError(res, 429, 'RATE_LIMITED', 'Too many disable attempts — slow down');
163
+ return;
164
+ }
165
+ const code = getNormalizedStringField(req.body, 'code');
166
+ if (!code) {
167
+ sendStoreError(res, 400, 'MISSING_FIELDS', 'code is required');
168
+ return;
169
+ }
170
+ try {
171
+ await store.disableTotp(code);
172
+ res.json({ enrolled: false });
173
+ }
174
+ catch (err) {
175
+ const message = err instanceof Error ? err.message : 'Disable failed';
176
+ logger.debug(`[TOTP] disable failed: ${message}`);
177
+ if (err instanceof TotpError) {
178
+ sendStoreError(res, httpStatusForStoreError(err.code), err.code, err.message);
179
+ }
180
+ else {
181
+ sendStoreError(res, 500, 'INTERNAL', message);
182
+ }
183
+ }
184
+ });
185
+ return router;
186
+ }
187
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"totpRoutes.js","sourceRoot":"","sources":["../../../src/web/routes/totpRoutes.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * TOTP (authenticator) enrollment HTTP routes — Phase 2 of #1780 (#1794).\n *\n * Provides:\n * - GET    /api/console/totp/status        — enrollment state (no secrets)\n * - POST   /api/console/totp/enroll/begin  — generate secret, return QR + otpauth URI\n * - POST   /api/console/totp/enroll/confirm — verify code, persist, return backup codes (once)\n * - POST   /api/console/totp/disable       — verify code, clear enrollment\n *\n * Security model:\n * - All endpoints require a valid existing console token. The caller must\n *   prove they already hold the token before they can enroll a second\n *   factor — otherwise an attacker with local port access could pre-enroll\n *   their own authenticator and lock the legitimate user out.\n * - Enforcement happens via an always-on `createAuthMiddleware` instance\n *   mounted at the top of this router, independent of the global\n *   DOLLHOUSE_WEB_AUTH_ENABLED flag.\n * - Backup codes are returned in plaintext exactly once (confirm response)\n *   and only their sha256 hashes are retained by the store.\n * - A sliding-window rate limit throttles confirm/disable attempts on a\n *   per-IP basis so a bad actor with a live session can't brute-force a\n *   TOTP window by flooding requests.\n *\n * @since v2.1.0 — Issue #1794\n */\n\nimport express, { Router } from 'express';\nimport type { Request, Response } from 'express';\nimport QRCode from 'qrcode';\nimport { TotpError, type ConsoleTokenStore } from '../console/consoleToken.js';\nimport { createAuthMiddleware } from '../middleware/authMiddleware.js';\nimport { SlidingWindowRateLimiter } from '../../utils/SlidingWindowRateLimiter.js';\nimport { httpStatusForStoreError, sendStoreError, getNormalizedStringField } from './consoleRouteHelpers.js';\nimport { logger } from '../../utils/logger.js';\n\n/** JSON body size limit — TOTP requests are tiny, cap hard. */\nconst BODY_LIMIT = '1kb';\n\n/**\n * Default rate limit for code-verification endpoints: 10 attempts per minute.\n * TOTP codes are 6 digits (1-in-10^6 guess rate) and the ±30s window gives\n * an attacker up to 3 valid codes per minute. 10 attempts caps brute-force\n * success probability per minute at ~3e-5 even before network latency.\n *\n * Limiters are per-endpoint (confirm vs disable) because they protect\n * different secrets: confirm tests the in-memory pending enrollment secret\n * (bounded lifetime, attacker must also know the pendingId), disable tests\n * the persisted enrollment secret (long-lived). A single shared limiter\n * would let traffic on one endpoint exhaust budget on the other.\n *\n * Limiters are global (not per-IP) because the server binds to 127.0.0.1\n * only — every request comes from the same loopback address, so keying on\n * IP would collapse to a single bucket anyway.\n *\n * Tests construct a fresh router per `buildApp`, which yields fresh\n * limiters, so no cross-test pollution.\n */\nconst DEFAULT_RATE_LIMIT_MAX = 10;\nconst DEFAULT_RATE_LIMIT_WINDOW_MS = 60_000;\n\n/**\n * Render an otpauth URI as an SVG data URL suitable for direct embedding\n * in an <img src> or background-image. Separated into a helper so the\n * request handler stays readable.\n */\nasync function renderQrDataUrl(otpauthUri: string): Promise<string> {\n  // errorCorrectionLevel 'M' is the default and balances size vs robustness.\n  // We emit SVG (not PNG) because it scales to any container size and is\n  // smaller on the wire for this particular payload.\n  const svg = await QRCode.toString(otpauthUri, { type: 'svg', margin: 1 });\n  return `data:image/svg+xml;utf8,${encodeURIComponent(svg)}`;\n}\n\n/**\n * Options for the TOTP routes factory.\n */\nexport interface TotpRoutesOptions {\n  store: ConsoleTokenStore;\n  /** Maximum code-verification attempts per window. Default: 10. */\n  rateLimitMax?: number;\n  /** Rate limit window in milliseconds. Default: 60_000 (1 minute). */\n  rateLimitWindowMs?: number;\n}\n\n/**\n * Build the Express router exposing TOTP endpoints. The returned router\n * should be mounted at `/api/console/totp`; the caller does not need to\n * add additional auth middleware — this router enforces its own auth\n * regardless of the global feature flag.\n */\nexport function createTotpRoutes(options: TotpRoutesOptions): Router {\n  const { store } = options;\n  const router = Router();\n  const jsonParser = express.json({ limit: BODY_LIMIT, type: 'application/json' });\n  // Fresh per-endpoint limiters per router instance. Separate buckets for\n  // confirm vs disable — they protect different secrets, so traffic on one\n  // should not exhaust the budget of the other. Fresh instances per call\n  // keep tests isolated.\n  const rateLimitMax = options.rateLimitMax ?? DEFAULT_RATE_LIMIT_MAX;\n  const rateLimitWindowMs = options.rateLimitWindowMs ?? DEFAULT_RATE_LIMIT_WINDOW_MS;\n  const confirmLimiter = new SlidingWindowRateLimiter(rateLimitMax, rateLimitWindowMs);\n  const disableLimiter = new SlidingWindowRateLimiter(rateLimitMax, rateLimitWindowMs);\n\n  // Always-on auth — the global feature flag does not apply here. Even\n  // during Phase 1 rollout (flag off), the TOTP endpoints must verify that\n  // the caller already holds the console token before letting them enroll\n  // a second factor. Otherwise an attacker with local port access could\n  // enroll their own authenticator and lock the real user out.\n  const auth = createAuthMiddleware({\n    store,\n    enabled: true,\n    label: 'totp',\n  });\n  router.use(auth);\n\n  /** GET /status — enrollment state (no secret material). */\n  router.get('/status', (_req: Request, res: Response) => {\n    res.json(store.getTotpStatus());\n  });\n\n  /** POST /enroll/begin — generate pending secret, return QR + URI. */\n  router.post('/enroll/begin', jsonParser, async (req: Request, res: Response) => {\n    // Optional label override — lets the UI label the authenticator entry\n    // differently if the user has renamed the console token.\n    const label = getNormalizedStringField(req.body, 'label') ?? undefined;\n    try {\n      const begin = store.beginTotpEnrollment(label);\n      const qrSvgDataUrl = await renderQrDataUrl(begin.otpauthUri);\n      res.json({\n        pendingId: begin.pendingId,\n        secret: begin.secret,\n        otpauthUri: begin.otpauthUri,\n        qrSvgDataUrl,\n        expiresAt: begin.expiresAt,\n      });\n    } catch (err) {\n      const message = err instanceof Error ? err.message : 'Enrollment could not be started';\n      logger.debug(`[TOTP] begin failed: ${message}`);\n      if (err instanceof TotpError) {\n        sendStoreError(res, httpStatusForStoreError(err.code), err.code, err.message);\n      } else {\n        sendStoreError(res, 500, 'INTERNAL', message);\n      }\n    }\n  });\n\n  /** POST /enroll/confirm — verify code, persist, return backup codes (once). */\n  router.post('/enroll/confirm', jsonParser, async (req: Request, res: Response) => {\n    if (!confirmLimiter.tryAcquire()) {\n      sendStoreError(res, 429, 'RATE_LIMITED', 'Too many confirmation attempts — slow down');\n      return;\n    }\n    const pendingId = getNormalizedStringField(req.body, 'pendingId');\n    const code = getNormalizedStringField(req.body, 'code');\n    if (!pendingId || !code) {\n      sendStoreError(res, 400, 'MISSING_FIELDS', 'pendingId and code are required');\n      return;\n    }\n    try {\n      const result = await store.confirmTotpEnrollment(pendingId, code);\n      res.json({\n        enrolled: true,\n        enrolledAt: result.enrolledAt,\n        backupCodes: result.backupCodes,\n      });\n    } catch (err) {\n      const message = err instanceof Error ? err.message : 'Confirmation failed';\n      logger.debug(`[TOTP] confirm failed: ${message}`);\n      if (err instanceof TotpError) {\n        sendStoreError(res, httpStatusForStoreError(err.code), err.code, err.message);\n      } else {\n        sendStoreError(res, 500, 'INTERNAL', message);\n      }\n    }\n  });\n\n  /** POST /disable — verify code, clear enrollment. */\n  router.post('/disable', jsonParser, async (req: Request, res: Response) => {\n    if (!disableLimiter.tryAcquire()) {\n      sendStoreError(res, 429, 'RATE_LIMITED', 'Too many disable attempts — slow down');\n      return;\n    }\n    const code = getNormalizedStringField(req.body, 'code');\n    if (!code) {\n      sendStoreError(res, 400, 'MISSING_FIELDS', 'code is required');\n      return;\n    }\n    try {\n      await store.disableTotp(code);\n      res.json({ enrolled: false });\n    } catch (err) {\n      const message = err instanceof Error ? err.message : 'Disable failed';\n      logger.debug(`[TOTP] disable failed: ${message}`);\n      if (err instanceof TotpError) {\n        sendStoreError(res, httpStatusForStoreError(err.code), err.code, err.message);\n      } else {\n        sendStoreError(res, 500, 'INTERNAL', message);\n      }\n    }\n  });\n\n  return router;\n}\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@dollhousemcp/mcp-server",
3
- "version": "2.0.10",
3
+ "version": "2.0.11",
4
4
  "description": "DollhouseMCP - A Model Context Protocol (MCP) server that enables dynamic AI persona management from markdown files, allowing Claude and other compatible AI assistants to activate and switch between different behavioral personas.",
5
5
  "type": "module",
6
6
  "main": "dist/index.js",
package/server.json CHANGED
@@ -3,7 +3,7 @@
3
3
  "name": "io.github.DollhouseMCP/mcp-server",
4
4
  "title": "DollhouseMCP",
5
5
  "description": "OSS to create Personas, Skills, Templates, Agents, and Memories to customize your AI experience.",
6
- "version": "2.0.10",
6
+ "version": "2.0.11",
7
7
  "homepage": "https://dollhousemcp.com",
8
8
  "repository": {
9
9
  "type": "git",
@@ -29,7 +29,7 @@
29
29
  {
30
30
  "registryType": "npm",
31
31
  "identifier": "@dollhousemcp/mcp-server",
32
- "version": "2.0.10",
32
+ "version": "2.0.11",
33
33
  "transport": {
34
34
  "type": "stdio"
35
35
  }
@@ -1,3 +0,0 @@
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
@@ -1 +0,0 @@
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"}
@@ -1,4 +0,0 @@
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==
@@ -1,35 +0,0 @@
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
@@ -1 +0,0 @@
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"}
@@ -1,181 +0,0 @@
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"]}
@@ -1,8 +0,0 @@
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
@@ -1 +0,0 @@
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"}