@dollhousemcp/mcp-server 2.0.0 → 2.0.1
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/README.md +282 -222
- package/README.npm.md +282 -222
- 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/package.json +1 -1
- package/server.json +2 -2
|
@@ -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.1';
|
|
6
|
+
export const BUILD_TIMESTAMP = '2026-04-02T00:00:54.478Z';
|
|
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,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9nZW5lcmF0ZWQvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFFSCxNQUFNLENBQUMsTUFBTSxlQUFlLEdBQUcsT0FBTyxDQUFDO0FBQ3ZDLE1BQU0sQ0FBQyxNQUFNLGVBQWUsR0FBRywwQkFBMEIsQ0FBQztBQUMxRCxNQUFNLENBQUMsTUFBTSxVQUFVLEdBQWtCLEtBQUssQ0FBQztBQUMvQyxNQUFNLENBQUMsTUFBTSxZQUFZLEdBQUcsMEJBQTBCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEF1dG8tZ2VuZXJhdGVkIGZpbGUgLSBETyBOT1QgRURJVFxuICogR2VuZXJhdGVkIGF0IGJ1aWxkIHRpbWUgYnkgc2NyaXB0cy9nZW5lcmF0ZS12ZXJzaW9uLmpzXG4gKi9cblxuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfVkVSU0lPTiA9ICcyLjAuMSc7XG5leHBvcnQgY29uc3QgQlVJTERfVElNRVNUQU1QID0gJzIwMjYtMDQtMDJUMDA6MDA6NTQuNDc4Wic7XG5leHBvcnQgY29uc3QgQlVJTERfVFlQRTogJ25wbScgfCAnZ2l0JyA9ICducG0nO1xuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfTkFNRSA9ICdAZG9sbGhvdXNlbWNwL21jcC1zZXJ2ZXInO1xuIl19
|
|
@@ -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"}
|
|
@@ -0,0 +1,204 @@
|
|
|
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 function getViewerHtml(port) {
|
|
8
|
+
return /* html */ `<!DOCTYPE html>
|
|
9
|
+
<html lang="en">
|
|
10
|
+
<head>
|
|
11
|
+
<meta charset="utf-8">
|
|
12
|
+
<meta name="viewport" content="width=device-width,initial-scale=1">
|
|
13
|
+
<title>DollhouseMCP Log Viewer</title>
|
|
14
|
+
<style>
|
|
15
|
+
*{box-sizing:border-box;margin:0;padding:0}
|
|
16
|
+
body{background:#1a1a2e;color:#e0e0e0;font-family:'Cascadia Code','Fira Code',monospace;font-size:13px}
|
|
17
|
+
#controls{display:flex;gap:8px;padding:8px 12px;background:#16213e;border-bottom:1px solid #0f3460;align-items:center;flex-wrap:wrap}
|
|
18
|
+
#controls label{color:#94a3b8;font-size:12px}
|
|
19
|
+
#controls select,#controls input{background:#1a1a2e;color:#e0e0e0;border:1px solid #0f3460;border-radius:4px;padding:4px 8px;font-family:inherit;font-size:12px}
|
|
20
|
+
#controls select:focus,#controls input:focus{outline:none;border-color:#e94560}
|
|
21
|
+
button{background:#0f3460;color:#e0e0e0;border:1px solid #0f3460;border-radius:4px;padding:4px 12px;cursor:pointer;font-family:inherit;font-size:12px}
|
|
22
|
+
button:hover{background:#e94560;border-color:#e94560}
|
|
23
|
+
#status{margin-left:auto;font-size:11px;padding:2px 8px;border-radius:10px}
|
|
24
|
+
.connected{background:#064e3b;color:#6ee7b7}
|
|
25
|
+
.disconnected{background:#7f1d1d;color:#fca5a5}
|
|
26
|
+
.paused{background:#78350f;color:#fcd34d}
|
|
27
|
+
#log{overflow-y:auto;height:calc(100vh - 44px);padding:8px 12px}
|
|
28
|
+
.entry{padding:2px 0;white-space:pre-wrap;word-break:break-word;border-bottom:1px solid #1e293b}
|
|
29
|
+
.entry:hover{background:#16213e}
|
|
30
|
+
.lvl-error{color:#f87171}
|
|
31
|
+
.lvl-warn{color:#fbbf24}
|
|
32
|
+
.lvl-info{color:#60a5fa}
|
|
33
|
+
.lvl-debug{color:#9ca3af}
|
|
34
|
+
.ts{color:#64748b}
|
|
35
|
+
.cat{color:#a78bfa}
|
|
36
|
+
.src{color:#2dd4bf}
|
|
37
|
+
.cid{color:#f472b6;font-size:11px;display:inline-block;width:72px;text-align:right;margin-right:4px}
|
|
38
|
+
#search{margin-left:4px}
|
|
39
|
+
</style>
|
|
40
|
+
</head>
|
|
41
|
+
<body>
|
|
42
|
+
<div id="controls">
|
|
43
|
+
<label>Category
|
|
44
|
+
<select id="fCategory">
|
|
45
|
+
<option value="">all</option>
|
|
46
|
+
<option value="application">application</option>
|
|
47
|
+
<option value="security">security</option>
|
|
48
|
+
<option value="performance">performance</option>
|
|
49
|
+
<option value="telemetry">telemetry</option>
|
|
50
|
+
</select>
|
|
51
|
+
</label>
|
|
52
|
+
<label>Level
|
|
53
|
+
<select id="fLevel">
|
|
54
|
+
<option value="">all</option>
|
|
55
|
+
<option value="debug">debug</option>
|
|
56
|
+
<option value="info">info</option>
|
|
57
|
+
<option value="warn">warn</option>
|
|
58
|
+
<option value="error">error</option>
|
|
59
|
+
</select>
|
|
60
|
+
</label>
|
|
61
|
+
<label>Source <input id="fSource" placeholder="substring" size="14"></label>
|
|
62
|
+
<label>RequestId <input id="fCorrelationId" placeholder="correlationId" size="20"></label>
|
|
63
|
+
<label>Search <input id="search" placeholder="message filter" size="18"></label>
|
|
64
|
+
<button id="btnPause">Pause</button>
|
|
65
|
+
<button id="btnClear">Clear</button>
|
|
66
|
+
<span id="status" class="disconnected">disconnected</span>
|
|
67
|
+
</div>
|
|
68
|
+
<div id="log"></div>
|
|
69
|
+
<script>
|
|
70
|
+
(function(){
|
|
71
|
+
var BASE = 'http://127.0.0.1:${port}';
|
|
72
|
+
var MAX_ENTRIES = 1000;
|
|
73
|
+
var log = document.getElementById('log');
|
|
74
|
+
var status = document.getElementById('status');
|
|
75
|
+
var fCategory = document.getElementById('fCategory');
|
|
76
|
+
var fLevel = document.getElementById('fLevel');
|
|
77
|
+
var fSource = document.getElementById('fSource');
|
|
78
|
+
var fCorrelationId = document.getElementById('fCorrelationId');
|
|
79
|
+
var searchBox = document.getElementById('search');
|
|
80
|
+
var btnPause = document.getElementById('btnPause');
|
|
81
|
+
var btnClear = document.getElementById('btnClear');
|
|
82
|
+
var es = null;
|
|
83
|
+
var paused = false;
|
|
84
|
+
var buffer = [];
|
|
85
|
+
|
|
86
|
+
function escHtml(s){
|
|
87
|
+
var d = document.createElement('div');
|
|
88
|
+
d.appendChild(document.createTextNode(s));
|
|
89
|
+
return d.innerHTML;
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
function setStatus(s, cls){
|
|
93
|
+
status.textContent = s;
|
|
94
|
+
status.className = cls;
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
|
|
98
|
+
|
|
99
|
+
var LEVEL_ORDER = {debug:0, info:1, warn:2, error:3};
|
|
100
|
+
|
|
101
|
+
function matchesFilters(entry){
|
|
102
|
+
var cat = fCategory.value;
|
|
103
|
+
if(cat && entry.category !== cat) return false;
|
|
104
|
+
var lvl = fLevel.value;
|
|
105
|
+
if(lvl && (LEVEL_ORDER[entry.level]||0) < (LEVEL_ORDER[lvl]||0)) return false;
|
|
106
|
+
var src = fSource.value.toLowerCase();
|
|
107
|
+
if(src && (!entry.source || entry.source.toLowerCase().indexOf(src) === -1)) return false;
|
|
108
|
+
var cid = fCorrelationId.value;
|
|
109
|
+
if(cid && entry.correlationId !== cid) return false;
|
|
110
|
+
var needle = searchBox.value.toLowerCase();
|
|
111
|
+
if(needle && (!entry.message || entry.message.toLowerCase().indexOf(needle) === -1)) return false;
|
|
112
|
+
return true;
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
function refilter(){
|
|
116
|
+
var els = log.children;
|
|
117
|
+
for(var i = 0; i < els.length; i++){
|
|
118
|
+
var data = els[i]._entryData;
|
|
119
|
+
if(data){
|
|
120
|
+
els[i].style.display = matchesFilters(data) ? '' : 'none';
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
function addEntry(entry){
|
|
126
|
+
if(!matchesFilters(entry)) var hidden = true;
|
|
127
|
+
|
|
128
|
+
var el = document.createElement('div');
|
|
129
|
+
el.className = 'entry lvl-' + entry.level;
|
|
130
|
+
el._entryData = entry;
|
|
131
|
+
if(hidden) el.style.display = 'none';
|
|
132
|
+
var ts = entry.timestamp ? entry.timestamp.slice(11, 23) : '';
|
|
133
|
+
var cid = entry.correlationId ? entry.correlationId.slice(-8) : '';
|
|
134
|
+
el.innerHTML = '<span class="ts">' + escHtml(ts) + '</span> '
|
|
135
|
+
+ '<span class="cid">' + escHtml(cid) + '</span> '
|
|
136
|
+
+ '<span class="cat">[' + escHtml(entry.category) + ']</span> '
|
|
137
|
+
+ '<span class="src">' + escHtml(entry.source) + '</span> '
|
|
138
|
+
+ escHtml(entry.message);
|
|
139
|
+
log.appendChild(el);
|
|
140
|
+
|
|
141
|
+
while(log.children.length > MAX_ENTRIES){
|
|
142
|
+
log.removeChild(log.firstChild);
|
|
143
|
+
}
|
|
144
|
+
if(!hidden) log.scrollTop = log.scrollHeight;
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
function connect(){
|
|
148
|
+
if(es){ es.close(); }
|
|
149
|
+
es = new EventSource(BASE + '/logs/stream');
|
|
150
|
+
es.onopen = function(){ setStatus('connected', 'connected'); };
|
|
151
|
+
es.onmessage = function(e){
|
|
152
|
+
try{
|
|
153
|
+
var entry = JSON.parse(e.data);
|
|
154
|
+
if(paused){ buffer.push(entry); }
|
|
155
|
+
else { addEntry(entry); }
|
|
156
|
+
}catch(err){}
|
|
157
|
+
};
|
|
158
|
+
es.onerror = function(){ setStatus('disconnected', 'disconnected'); };
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
var filterTimer = null;
|
|
162
|
+
function onFilterChange(){
|
|
163
|
+
clearTimeout(filterTimer);
|
|
164
|
+
filterTimer = setTimeout(refilter, 50);
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
fCategory.addEventListener('change', onFilterChange);
|
|
168
|
+
fLevel.addEventListener('change', onFilterChange);
|
|
169
|
+
fSource.addEventListener('input', function(){
|
|
170
|
+
clearTimeout(filterTimer);
|
|
171
|
+
filterTimer = setTimeout(refilter, 400);
|
|
172
|
+
});
|
|
173
|
+
fCorrelationId.addEventListener('input', function(){
|
|
174
|
+
clearTimeout(filterTimer);
|
|
175
|
+
filterTimer = setTimeout(refilter, 400);
|
|
176
|
+
});
|
|
177
|
+
searchBox.addEventListener('input', function(){
|
|
178
|
+
clearTimeout(filterTimer);
|
|
179
|
+
filterTimer = setTimeout(refilter, 400);
|
|
180
|
+
});
|
|
181
|
+
|
|
182
|
+
btnPause.addEventListener('click', function(){
|
|
183
|
+
paused = !paused;
|
|
184
|
+
btnPause.textContent = paused ? 'Resume' : 'Pause';
|
|
185
|
+
if(paused){
|
|
186
|
+
setStatus('paused', 'paused');
|
|
187
|
+
} else {
|
|
188
|
+
setStatus('connected', 'connected');
|
|
189
|
+
buffer.forEach(addEntry);
|
|
190
|
+
buffer = [];
|
|
191
|
+
}
|
|
192
|
+
});
|
|
193
|
+
|
|
194
|
+
btnClear.addEventListener('click', function(){
|
|
195
|
+
log.innerHTML = '';
|
|
196
|
+
});
|
|
197
|
+
|
|
198
|
+
connect();
|
|
199
|
+
})();
|
|
200
|
+
</script>
|
|
201
|
+
</body>
|
|
202
|
+
</html>`;
|
|
203
|
+
}
|
|
204
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"viewerHtml.js","sourceRoot":"","sources":["../../../src/logging/viewer/viewerHtml.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAEH,MAAM,UAAU,aAAa,CAAC,IAAY;IACxC,OAAO,UAAU,CAAC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;iCA+Da,IAAI;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;QAmI7B,CAAC;AACT,CAAC","sourcesContent":["/**\n * Embedded HTML template for the DollhouseMCP Log Viewer.\n *\n * Returns a self-contained vanilla JS/CSS page that connects to the\n * SSELogSink's /logs/stream endpoint via EventSource. See docs/LOGGING-DESIGN.md §4.6.\n */\n\nexport function getViewerHtml(port: number): string {\n  return /* html */ `<!DOCTYPE html>\n<html lang=\"en\">\n<head>\n<meta charset=\"utf-8\">\n<meta name=\"viewport\" content=\"width=device-width,initial-scale=1\">\n<title>DollhouseMCP Log Viewer</title>\n<style>\n*{box-sizing:border-box;margin:0;padding:0}\nbody{background:#1a1a2e;color:#e0e0e0;font-family:'Cascadia Code','Fira Code',monospace;font-size:13px}\n#controls{display:flex;gap:8px;padding:8px 12px;background:#16213e;border-bottom:1px solid #0f3460;align-items:center;flex-wrap:wrap}\n#controls label{color:#94a3b8;font-size:12px}\n#controls select,#controls input{background:#1a1a2e;color:#e0e0e0;border:1px solid #0f3460;border-radius:4px;padding:4px 8px;font-family:inherit;font-size:12px}\n#controls select:focus,#controls input:focus{outline:none;border-color:#e94560}\nbutton{background:#0f3460;color:#e0e0e0;border:1px solid #0f3460;border-radius:4px;padding:4px 12px;cursor:pointer;font-family:inherit;font-size:12px}\nbutton:hover{background:#e94560;border-color:#e94560}\n#status{margin-left:auto;font-size:11px;padding:2px 8px;border-radius:10px}\n.connected{background:#064e3b;color:#6ee7b7}\n.disconnected{background:#7f1d1d;color:#fca5a5}\n.paused{background:#78350f;color:#fcd34d}\n#log{overflow-y:auto;height:calc(100vh - 44px);padding:8px 12px}\n.entry{padding:2px 0;white-space:pre-wrap;word-break:break-word;border-bottom:1px solid #1e293b}\n.entry:hover{background:#16213e}\n.lvl-error{color:#f87171}\n.lvl-warn{color:#fbbf24}\n.lvl-info{color:#60a5fa}\n.lvl-debug{color:#9ca3af}\n.ts{color:#64748b}\n.cat{color:#a78bfa}\n.src{color:#2dd4bf}\n.cid{color:#f472b6;font-size:11px;display:inline-block;width:72px;text-align:right;margin-right:4px}\n#search{margin-left:4px}\n</style>\n</head>\n<body>\n<div id=\"controls\">\n  <label>Category\n    <select id=\"fCategory\">\n      <option value=\"\">all</option>\n      <option value=\"application\">application</option>\n      <option value=\"security\">security</option>\n      <option value=\"performance\">performance</option>\n      <option value=\"telemetry\">telemetry</option>\n    </select>\n  </label>\n  <label>Level\n    <select id=\"fLevel\">\n      <option value=\"\">all</option>\n      <option value=\"debug\">debug</option>\n      <option value=\"info\">info</option>\n      <option value=\"warn\">warn</option>\n      <option value=\"error\">error</option>\n    </select>\n  </label>\n  <label>Source <input id=\"fSource\" placeholder=\"substring\" size=\"14\"></label>\n  <label>RequestId <input id=\"fCorrelationId\" placeholder=\"correlationId\" size=\"20\"></label>\n  <label>Search <input id=\"search\" placeholder=\"message filter\" size=\"18\"></label>\n  <button id=\"btnPause\">Pause</button>\n  <button id=\"btnClear\">Clear</button>\n  <span id=\"status\" class=\"disconnected\">disconnected</span>\n</div>\n<div id=\"log\"></div>\n<script>\n(function(){\n  var BASE = 'http://127.0.0.1:${port}';\n  var MAX_ENTRIES = 1000;\n  var log = document.getElementById('log');\n  var status = document.getElementById('status');\n  var fCategory = document.getElementById('fCategory');\n  var fLevel = document.getElementById('fLevel');\n  var fSource = document.getElementById('fSource');\n  var fCorrelationId = document.getElementById('fCorrelationId');\n  var searchBox = document.getElementById('search');\n  var btnPause = document.getElementById('btnPause');\n  var btnClear = document.getElementById('btnClear');\n  var es = null;\n  var paused = false;\n  var buffer = [];\n\n  function escHtml(s){\n    var d = document.createElement('div');\n    d.appendChild(document.createTextNode(s));\n    return d.innerHTML;\n  }\n\n  function setStatus(s, cls){\n    status.textContent = s;\n    status.className = cls;\n  }\n\n\n\n  var LEVEL_ORDER = {debug:0, info:1, warn:2, error:3};\n\n  function matchesFilters(entry){\n    var cat = fCategory.value;\n    if(cat && entry.category !== cat) return false;\n    var lvl = fLevel.value;\n    if(lvl && (LEVEL_ORDER[entry.level]||0) < (LEVEL_ORDER[lvl]||0)) return false;\n    var src = fSource.value.toLowerCase();\n    if(src && (!entry.source || entry.source.toLowerCase().indexOf(src) === -1)) return false;\n    var cid = fCorrelationId.value;\n    if(cid && entry.correlationId !== cid) return false;\n    var needle = searchBox.value.toLowerCase();\n    if(needle && (!entry.message || entry.message.toLowerCase().indexOf(needle) === -1)) return false;\n    return true;\n  }\n\n  function refilter(){\n    var els = log.children;\n    for(var i = 0; i < els.length; i++){\n      var data = els[i]._entryData;\n      if(data){\n        els[i].style.display = matchesFilters(data) ? '' : 'none';\n      }\n    }\n  }\n\n  function addEntry(entry){\n    if(!matchesFilters(entry)) var hidden = true;\n\n    var el = document.createElement('div');\n    el.className = 'entry lvl-' + entry.level;\n    el._entryData = entry;\n    if(hidden) el.style.display = 'none';\n    var ts = entry.timestamp ? entry.timestamp.slice(11, 23) : '';\n    var cid = entry.correlationId ? entry.correlationId.slice(-8) : '';\n    el.innerHTML = '<span class=\"ts\">' + escHtml(ts) + '</span> '\n      + '<span class=\"cid\">' + escHtml(cid) + '</span> '\n      + '<span class=\"cat\">[' + escHtml(entry.category) + ']</span> '\n      + '<span class=\"src\">' + escHtml(entry.source) + '</span> '\n      + escHtml(entry.message);\n    log.appendChild(el);\n\n    while(log.children.length > MAX_ENTRIES){\n      log.removeChild(log.firstChild);\n    }\n    if(!hidden) log.scrollTop = log.scrollHeight;\n  }\n\n  function connect(){\n    if(es){ es.close(); }\n    es = new EventSource(BASE + '/logs/stream');\n    es.onopen = function(){ setStatus('connected', 'connected'); };\n    es.onmessage = function(e){\n      try{\n        var entry = JSON.parse(e.data);\n        if(paused){ buffer.push(entry); }\n        else { addEntry(entry); }\n      }catch(err){}\n    };\n    es.onerror = function(){ setStatus('disconnected', 'disconnected'); };\n  }\n\n  var filterTimer = null;\n  function onFilterChange(){\n    clearTimeout(filterTimer);\n    filterTimer = setTimeout(refilter, 50);\n  }\n\n  fCategory.addEventListener('change', onFilterChange);\n  fLevel.addEventListener('change', onFilterChange);\n  fSource.addEventListener('input', function(){\n    clearTimeout(filterTimer);\n    filterTimer = setTimeout(refilter, 400);\n  });\n  fCorrelationId.addEventListener('input', function(){\n    clearTimeout(filterTimer);\n    filterTimer = setTimeout(refilter, 400);\n  });\n  searchBox.addEventListener('input', function(){\n    clearTimeout(filterTimer);\n    filterTimer = setTimeout(refilter, 400);\n  });\n\n  btnPause.addEventListener('click', function(){\n    paused = !paused;\n    btnPause.textContent = paused ? 'Resume' : 'Pause';\n    if(paused){\n      setStatus('paused', 'paused');\n    } else {\n      setStatus('connected', 'connected');\n      buffer.forEach(addEntry);\n      buffer = [];\n    }\n  });\n\n  btnClear.addEventListener('click', function(){\n    log.innerHTML = '';\n  });\n\n  connect();\n})();\n</script>\n</body>\n</html>`;\n}\n"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@dollhousemcp/mcp-server",
|
|
3
|
-
"version": "2.0.
|
|
3
|
+
"version": "2.0.1",
|
|
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.
|
|
6
|
+
"version": "2.0.1",
|
|
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.
|
|
32
|
+
"version": "2.0.1",
|
|
33
33
|
"transport": {
|
|
34
34
|
"type": "stdio"
|
|
35
35
|
}
|