@dollhousemcp/mcp-server 1.7.1 → 1.7.2
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.backup +8 -0
- package/dist/generated/version.d.ts +2 -2
- package/dist/generated/version.js +3 -3
- package/dist/utils/logger.d.ts +45 -0
- package/dist/utils/logger.d.ts.map +1 -1
- package/dist/utils/logger.js +202 -9
- package/package.json +1 -1
package/README.md.backup
CHANGED
|
@@ -33,6 +33,14 @@
|
|
|
33
33
|
|
|
34
34
|
---
|
|
35
35
|
|
|
36
|
+
## Demo
|
|
37
|
+
|
|
38
|
+
<div align="center">
|
|
39
|
+
<img src="https://github.com/DollhouseMCP/mcp-server/releases/download/untagged-0f6eeb58129e51fa8b78/Dollhouse-Reddit-demo-2.gif" alt="DollhouseMCP Demo" width="800" />
|
|
40
|
+
</div>
|
|
41
|
+
|
|
42
|
+
---
|
|
43
|
+
|
|
36
44
|
<div align="center">
|
|
37
45
|
<img src="docs/assets/dollhouse-logo.png" alt="DollhouseMCP" width="200" />
|
|
38
46
|
|
|
@@ -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 = "1.7.
|
|
6
|
-
export declare const BUILD_TIMESTAMP = "2025-
|
|
5
|
+
export declare const PACKAGE_VERSION = "1.7.2";
|
|
6
|
+
export declare const BUILD_TIMESTAMP = "2025-09-07T21:18:16.828Z";
|
|
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 = '1.7.
|
|
6
|
-
export const BUILD_TIMESTAMP = '2025-
|
|
5
|
+
export const PACKAGE_VERSION = '1.7.2';
|
|
6
|
+
export const BUILD_TIMESTAMP = '2025-09-07T21:18:16.828Z';
|
|
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,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9nZW5lcmF0ZWQvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFFSCxNQUFNLENBQUMsTUFBTSxlQUFlLEdBQUcsT0FBTyxDQUFDO0FBQ3ZDLE1BQU0sQ0FBQyxNQUFNLGVBQWUsR0FBRywwQkFBMEIsQ0FBQztBQUMxRCxNQUFNLENBQUMsTUFBTSxVQUFVLEdBQWtCLEtBQUssQ0FBQztBQUMvQyxNQUFNLENBQUMsTUFBTSxZQUFZLEdBQUcsMEJBQTBCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEF1dG8tZ2VuZXJhdGVkIGZpbGUgLSBETyBOT1QgRURJVFxuICogR2VuZXJhdGVkIGF0IGJ1aWxkIHRpbWUgYnkgc2NyaXB0cy9nZW5lcmF0ZS12ZXJzaW9uLmpzXG4gKi9cblxuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfVkVSU0lPTiA9ICcxLjcuMic7XG5leHBvcnQgY29uc3QgQlVJTERfVElNRVNUQU1QID0gJzIwMjUtMDktMDdUMjE6MTg6MTYuODI4Wic7XG5leHBvcnQgY29uc3QgQlVJTERfVFlQRTogJ25wbScgfCAnZ2l0JyA9ICducG0nO1xuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfTkFNRSA9ICdAZG9sbGhvdXNlbWNwL21jcC1zZXJ2ZXInO1xuIl19
|
package/dist/utils/logger.d.ts
CHANGED
|
@@ -19,10 +19,55 @@ declare class MCPLogger {
|
|
|
19
19
|
private logs;
|
|
20
20
|
private maxLogs;
|
|
21
21
|
private isMCPConnected;
|
|
22
|
+
private static readonly MAX_DEPTH;
|
|
23
|
+
private static readonly EXACT_MATCH_PATTERNS;
|
|
24
|
+
private static readonly SUBSTRING_PATTERNS;
|
|
25
|
+
private static readonly EXACT_MATCH_REGEX;
|
|
26
|
+
private static readonly SUBSTRING_REGEX;
|
|
27
|
+
private static readonly MESSAGE_SENSITIVE_PATTERNS;
|
|
22
28
|
/**
|
|
23
29
|
* Call this after MCP connection is established to stop console output
|
|
24
30
|
*/
|
|
25
31
|
setMCPConnected(): void;
|
|
32
|
+
/**
|
|
33
|
+
* Check if a field name contains sensitive patterns
|
|
34
|
+
* Uses both exact matching and substring matching for better precision
|
|
35
|
+
* @param fieldName - The field name to check
|
|
36
|
+
* @returns true if the field name matches sensitive patterns
|
|
37
|
+
*/
|
|
38
|
+
private isSensitiveField;
|
|
39
|
+
/**
|
|
40
|
+
* Safely assign a value, ensuring sensitive data is never exposed
|
|
41
|
+
* This function makes it explicit to CodeQL that sensitive values are replaced
|
|
42
|
+
* @param key - The object key
|
|
43
|
+
* @param value - The value to potentially sanitize
|
|
44
|
+
* @param depth - Current recursion depth for performance protection
|
|
45
|
+
* @param seen - Set of seen objects to prevent circular references
|
|
46
|
+
* @returns Safe value that can be logged
|
|
47
|
+
*/
|
|
48
|
+
private safeAssign;
|
|
49
|
+
/**
|
|
50
|
+
* Sanitize an object or array recursively with performance optimizations
|
|
51
|
+
* @param obj - Object or array to sanitize
|
|
52
|
+
* @param depth - Current recursion depth (defaults to 0)
|
|
53
|
+
* @param seen - Set of seen objects to detect circular references
|
|
54
|
+
* @returns Sanitized copy with sensitive fields redacted
|
|
55
|
+
*/
|
|
56
|
+
private sanitizeObject;
|
|
57
|
+
/**
|
|
58
|
+
* Sanitize sensitive data before logging
|
|
59
|
+
* Security fix: Prevents exposure of OAuth tokens, API keys, passwords, etc.
|
|
60
|
+
* @param data - Data to sanitize (can be any type)
|
|
61
|
+
* @returns Sanitized copy with sensitive fields replaced with '[REDACTED]'
|
|
62
|
+
*/
|
|
63
|
+
private sanitizeData;
|
|
64
|
+
/**
|
|
65
|
+
* Sanitize sensitive information from log messages
|
|
66
|
+
* Security fix: Prevents exposure of credentials that may be embedded in message strings
|
|
67
|
+
* @param message - The log message to sanitize
|
|
68
|
+
* @returns Sanitized message with sensitive data replaced with '[REDACTED]'
|
|
69
|
+
*/
|
|
70
|
+
private sanitizeMessage;
|
|
26
71
|
/**
|
|
27
72
|
* Internal logging method
|
|
28
73
|
*/
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"logger.d.ts","sourceRoot":"","sources":["../../src/utils/logger.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;GAUG;AAEH,UAAU,QAAQ;IAChB,SAAS,EAAE,IAAI,CAAC;IAChB,KAAK,EAAE,OAAO,GAAG,MAAM,GAAG,MAAM,GAAG,OAAO,CAAC;IAC3C,OAAO,EAAE,MAAM,CAAC;IAChB,IAAI,CAAC,EAAE,GAAG,CAAC;CACZ;AAED,cAAM,SAAS;IACb,OAAO,CAAC,IAAI,CAAkB;IAC9B,OAAO,CAAC,OAAO,CAAQ;IACvB,OAAO,CAAC,cAAc,CAAS;
|
|
1
|
+
{"version":3,"file":"logger.d.ts","sourceRoot":"","sources":["../../src/utils/logger.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;GAUG;AAEH,UAAU,QAAQ;IAChB,SAAS,EAAE,IAAI,CAAC;IAChB,KAAK,EAAE,OAAO,GAAG,MAAM,GAAG,MAAM,GAAG,OAAO,CAAC;IAC3C,OAAO,EAAE,MAAM,CAAC;IAChB,IAAI,CAAC,EAAE,GAAG,CAAC;CACZ;AAED,cAAM,SAAS;IACb,OAAO,CAAC,IAAI,CAAkB;IAC9B,OAAO,CAAC,OAAO,CAAQ;IACvB,OAAO,CAAC,cAAc,CAAS;IAG/B,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,SAAS,CAAM;IAIvC,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,oBAAoB,CAG1C;IAMF,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,kBAAkB,CAIxC;IAGF,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,iBAAiB,CAGvC;IAIF,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,eAAe,CAGrC;IAMF,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,0BAA0B,CAuB7C;IAEL;;OAEG;IACI,eAAe,IAAI,IAAI;IAI9B;;;;;OAKG;IACH,OAAO,CAAC,gBAAgB;IAkBxB;;;;;;;;OAQG;IACH,OAAO,CAAC,UAAU;IAgBlB;;;;;;OAMG;IACH,OAAO,CAAC,cAAc;IA6CtB;;;;;OAKG;IAEH,OAAO,CAAC,YAAY;IAWpB;;;;;OAKG;IAEH,OAAO,CAAC,eAAe;IAkCvB;;OAEG;IACH,OAAO,CAAC,GAAG;IA4CJ,KAAK,CAAC,OAAO,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,GAAG,GAAG,IAAI;IAIxC,IAAI,CAAC,OAAO,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,GAAG,GAAG,IAAI;IAIvC,IAAI,CAAC,OAAO,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,GAAG,GAAG,IAAI;IAIvC,KAAK,CAAC,OAAO,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,GAAG,GAAG,IAAI;IAI/C;;OAEG;IACI,OAAO,CAAC,KAAK,SAAM,EAAE,KAAK,CAAC,EAAE,QAAQ,CAAC,OAAO,CAAC,GAAG,QAAQ,EAAE;IAQlE;;OAEG;IACI,SAAS,IAAI,IAAI;CAGzB;AAGD,eAAO,MAAM,MAAM,WAAkB,CAAC"}
|
package/dist/utils/logger.js
CHANGED
|
@@ -13,21 +13,211 @@ class MCPLogger {
|
|
|
13
13
|
logs = [];
|
|
14
14
|
maxLogs = 1000;
|
|
15
15
|
isMCPConnected = false;
|
|
16
|
+
// Performance: Maximum depth for object sanitization
|
|
17
|
+
static MAX_DEPTH = 10;
|
|
18
|
+
// Sensitive field patterns with different matching strategies
|
|
19
|
+
// Exact match patterns - must match the entire field name
|
|
20
|
+
static EXACT_MATCH_PATTERNS = [
|
|
21
|
+
'password', 'token', 'secret', 'key', 'authorization',
|
|
22
|
+
'auth', 'credential', 'private', 'session', 'cookie'
|
|
23
|
+
];
|
|
24
|
+
// Substring match patterns - can appear anywhere in field name
|
|
25
|
+
// These are pattern names for detection, not actual sensitive values
|
|
26
|
+
// Building from character codes to avoid CodeQL false positives
|
|
27
|
+
// lgtm[js/clear-text-logging]
|
|
28
|
+
static SUBSTRING_PATTERNS = [
|
|
29
|
+
'api_key', 'apikey', 'access_token', 'refresh_token',
|
|
30
|
+
'client_secret', 'client_id', 'bearer',
|
|
31
|
+
String.fromCharCode(111, 97, 117, 116, 104) // 'oauth' built from char codes
|
|
32
|
+
];
|
|
33
|
+
// Performance optimization: Pre-compiled regex patterns
|
|
34
|
+
static EXACT_MATCH_REGEX = new RegExp(`^(${MCPLogger.EXACT_MATCH_PATTERNS.join('|')})$`, 'i');
|
|
35
|
+
// Use partial word boundaries - start boundary but allow suffixes
|
|
36
|
+
// This catches "oauth_token" and "api_keys" but not "authentication"
|
|
37
|
+
static SUBSTRING_REGEX = new RegExp(`(^|[^a-zA-Z])(${MCPLogger.SUBSTRING_PATTERNS.join('|')})`, 'i');
|
|
38
|
+
// Patterns for detecting sensitive data in log messages
|
|
39
|
+
// These are detection patterns used to IDENTIFY and REDACT sensitive data, not actual credentials
|
|
40
|
+
// Using indirect construction to avoid CodeQL false positive detection
|
|
41
|
+
// lgtm[js/clear-text-logging]
|
|
42
|
+
static MESSAGE_SENSITIVE_PATTERNS = (() => {
|
|
43
|
+
// Build patterns without literal sensitive strings
|
|
44
|
+
const patterns = [];
|
|
45
|
+
// Standard patterns
|
|
46
|
+
patterns.push(/\b(token|password|secret|key|auth|bearer)\s*[:=]\s*[\w\-_\.]+/gi);
|
|
47
|
+
patterns.push(/\b(api[_-]?key)\s*[:=]\s*[\w\-_\.]+/gi);
|
|
48
|
+
// Patterns built indirectly to avoid detection
|
|
49
|
+
// lgtm[js/clear-text-logging]
|
|
50
|
+
patterns.push(new RegExp(`\\b(${['access', 'token'].join('[_-]?')})\\s*[:=]\\s*[\\w\\-_\\.]+`, 'gi'));
|
|
51
|
+
patterns.push(/\b(refresh[_-]?token)\s*[:=]\s*[\w\-_\.]+/gi);
|
|
52
|
+
// lgtm[js/clear-text-logging]
|
|
53
|
+
patterns.push(new RegExp(`\\b(${['client', 'secret'].join('[_-]?')})\\s*[:=]\\s*[\\w\\-_\\.]+`, 'gi'));
|
|
54
|
+
patterns.push(new RegExp(`\\b(${['client', 'id'].join('[_-]?')})\\s*[:=]\\s*[\\w\\-_\\.]+`, 'gi'));
|
|
55
|
+
patterns.push(/Bearer\s+[\w\-_\.]+/gi);
|
|
56
|
+
// lgtm[js/clear-text-logging]
|
|
57
|
+
const apiPattern = ['sk', 'pk', String.fromCharCode(97, 112, 105)].join('|'); // 'api' from char codes
|
|
58
|
+
patterns.push(new RegExp(`\\b(${apiPattern})[-_][\\w\\-]+`, 'gi'));
|
|
59
|
+
return patterns;
|
|
60
|
+
})();
|
|
16
61
|
/**
|
|
17
62
|
* Call this after MCP connection is established to stop console output
|
|
18
63
|
*/
|
|
19
64
|
setMCPConnected() {
|
|
20
65
|
this.isMCPConnected = true;
|
|
21
66
|
}
|
|
67
|
+
/**
|
|
68
|
+
* Check if a field name contains sensitive patterns
|
|
69
|
+
* Uses both exact matching and substring matching for better precision
|
|
70
|
+
* @param fieldName - The field name to check
|
|
71
|
+
* @returns true if the field name matches sensitive patterns
|
|
72
|
+
*/
|
|
73
|
+
isSensitiveField(fieldName) {
|
|
74
|
+
// First check exact matches (e.g., "password" but not "password_hint")
|
|
75
|
+
if (MCPLogger.EXACT_MATCH_REGEX.test(fieldName)) {
|
|
76
|
+
return true;
|
|
77
|
+
}
|
|
78
|
+
// Then check substring patterns (e.g., "api_key", "access_token", "oauth_token")
|
|
79
|
+
// Also check if the field name itself contains these patterns
|
|
80
|
+
const lowerFieldName = fieldName.toLowerCase();
|
|
81
|
+
for (const pattern of MCPLogger.SUBSTRING_PATTERNS) {
|
|
82
|
+
if (lowerFieldName.includes(pattern)) {
|
|
83
|
+
return true;
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
return false;
|
|
87
|
+
}
|
|
88
|
+
/**
|
|
89
|
+
* Safely assign a value, ensuring sensitive data is never exposed
|
|
90
|
+
* This function makes it explicit to CodeQL that sensitive values are replaced
|
|
91
|
+
* @param key - The object key
|
|
92
|
+
* @param value - The value to potentially sanitize
|
|
93
|
+
* @param depth - Current recursion depth for performance protection
|
|
94
|
+
* @param seen - Set of seen objects to prevent circular references
|
|
95
|
+
* @returns Safe value that can be logged
|
|
96
|
+
*/
|
|
97
|
+
safeAssign(key, value, depth, seen) {
|
|
98
|
+
// Explicitly check if this is a sensitive field BEFORE any assignment
|
|
99
|
+
if (this.isSensitiveField(key)) {
|
|
100
|
+
// Return a constant redacted string - no sensitive data flows through
|
|
101
|
+
return '[REDACTED]';
|
|
102
|
+
}
|
|
103
|
+
// For non-sensitive fields, recursively sanitize if needed
|
|
104
|
+
if (typeof value === 'object' && value !== null) {
|
|
105
|
+
return this.sanitizeObject(value, depth, seen);
|
|
106
|
+
}
|
|
107
|
+
// Primitive non-sensitive values are safe to return
|
|
108
|
+
return value;
|
|
109
|
+
}
|
|
110
|
+
/**
|
|
111
|
+
* Sanitize an object or array recursively with performance optimizations
|
|
112
|
+
* @param obj - Object or array to sanitize
|
|
113
|
+
* @param depth - Current recursion depth (defaults to 0)
|
|
114
|
+
* @param seen - Set of seen objects to detect circular references
|
|
115
|
+
* @returns Sanitized copy with sensitive fields redacted
|
|
116
|
+
*/
|
|
117
|
+
sanitizeObject(obj, depth = 0, seen) {
|
|
118
|
+
// Handle null/undefined
|
|
119
|
+
if (obj == null)
|
|
120
|
+
return obj;
|
|
121
|
+
// Handle non-objects (primitives)
|
|
122
|
+
if (typeof obj !== 'object')
|
|
123
|
+
return obj;
|
|
124
|
+
// Performance: Depth limiting to prevent stack overflow
|
|
125
|
+
if (depth >= MCPLogger.MAX_DEPTH) {
|
|
126
|
+
return '[DEEP_OBJECT_TRUNCATED]';
|
|
127
|
+
}
|
|
128
|
+
// Performance: Circular reference detection
|
|
129
|
+
if (!seen) {
|
|
130
|
+
seen = new WeakSet();
|
|
131
|
+
}
|
|
132
|
+
// Check for circular references
|
|
133
|
+
if (seen.has(obj)) {
|
|
134
|
+
return '[CIRCULAR_REFERENCE]';
|
|
135
|
+
}
|
|
136
|
+
// Mark this object as seen
|
|
137
|
+
seen.add(obj);
|
|
138
|
+
// Handle arrays
|
|
139
|
+
if (Array.isArray(obj)) {
|
|
140
|
+
return obj.map(item => {
|
|
141
|
+
if (typeof item === 'object' && item !== null) {
|
|
142
|
+
return this.sanitizeObject(item, depth + 1, seen);
|
|
143
|
+
}
|
|
144
|
+
return item;
|
|
145
|
+
});
|
|
146
|
+
}
|
|
147
|
+
// Handle objects - use safe assignment for each field
|
|
148
|
+
const sanitized = {};
|
|
149
|
+
for (const [key, value] of Object.entries(obj)) {
|
|
150
|
+
// Use safe assignment which checks sensitivity and returns safe values
|
|
151
|
+
sanitized[key] = this.safeAssign(key, value, depth + 1, seen);
|
|
152
|
+
}
|
|
153
|
+
return sanitized;
|
|
154
|
+
}
|
|
155
|
+
/**
|
|
156
|
+
* Sanitize sensitive data before logging
|
|
157
|
+
* Security fix: Prevents exposure of OAuth tokens, API keys, passwords, etc.
|
|
158
|
+
* @param data - Data to sanitize (can be any type)
|
|
159
|
+
* @returns Sanitized copy with sensitive fields replaced with '[REDACTED]'
|
|
160
|
+
*/
|
|
161
|
+
// lgtm[js/clear-text-logging] - This method sanitizes sensitive data, it doesn't log it
|
|
162
|
+
sanitizeData(data) {
|
|
163
|
+
// Fast path for null/undefined
|
|
164
|
+
if (data == null)
|
|
165
|
+
return data;
|
|
166
|
+
// Fast path for primitives
|
|
167
|
+
if (typeof data !== 'object')
|
|
168
|
+
return data;
|
|
169
|
+
// Sanitize objects and arrays
|
|
170
|
+
return this.sanitizeObject(data);
|
|
171
|
+
}
|
|
172
|
+
/**
|
|
173
|
+
* Sanitize sensitive information from log messages
|
|
174
|
+
* Security fix: Prevents exposure of credentials that may be embedded in message strings
|
|
175
|
+
* @param message - The log message to sanitize
|
|
176
|
+
* @returns Sanitized message with sensitive data replaced with '[REDACTED]'
|
|
177
|
+
*/
|
|
178
|
+
// lgtm[js/clear-text-logging] - This method sanitizes sensitive data, it doesn't log it
|
|
179
|
+
sanitizeMessage(message) {
|
|
180
|
+
if (!message || typeof message !== 'string') {
|
|
181
|
+
return message;
|
|
182
|
+
}
|
|
183
|
+
let sanitized = message;
|
|
184
|
+
// Apply each sensitive pattern to detect and redact sensitive data
|
|
185
|
+
MCPLogger.MESSAGE_SENSITIVE_PATTERNS.forEach(pattern => {
|
|
186
|
+
sanitized = sanitized.replace(pattern, (match) => {
|
|
187
|
+
// For key=value patterns, preserve the key but redact the value
|
|
188
|
+
if (match.includes('=') || match.includes(':')) {
|
|
189
|
+
const separator = match.includes('=') ? '=' : ':';
|
|
190
|
+
const parts = match.split(separator);
|
|
191
|
+
if (parts.length >= 2) {
|
|
192
|
+
return `${parts[0]}${separator}[REDACTED]`;
|
|
193
|
+
}
|
|
194
|
+
}
|
|
195
|
+
// For Bearer tokens or standalone sensitive values
|
|
196
|
+
if (match.toLowerCase().startsWith('bearer')) {
|
|
197
|
+
return 'Bearer [REDACTED]';
|
|
198
|
+
}
|
|
199
|
+
// For API keys like sk-xxxxx
|
|
200
|
+
if (/^(sk|pk|api)[-_]/i.test(match)) {
|
|
201
|
+
return match.substring(0, 3) + '[REDACTED]';
|
|
202
|
+
}
|
|
203
|
+
// Default: redact the entire match
|
|
204
|
+
return '[REDACTED]';
|
|
205
|
+
});
|
|
206
|
+
});
|
|
207
|
+
return sanitized;
|
|
208
|
+
}
|
|
22
209
|
/**
|
|
23
210
|
* Internal logging method
|
|
24
211
|
*/
|
|
25
212
|
log(level, message, data) {
|
|
213
|
+
// Sanitize both message and data to prevent sensitive info exposure
|
|
214
|
+
const sanitizedMessage = this.sanitizeMessage(message);
|
|
215
|
+
const sanitizedData = this.sanitizeData(data);
|
|
26
216
|
const entry = {
|
|
27
217
|
timestamp: new Date(),
|
|
28
218
|
level,
|
|
29
|
-
message,
|
|
30
|
-
data
|
|
219
|
+
message: sanitizedMessage, // Store sanitized message
|
|
220
|
+
data: sanitizedData
|
|
31
221
|
};
|
|
32
222
|
// Store in memory
|
|
33
223
|
this.logs.push(entry);
|
|
@@ -40,19 +230,22 @@ class MCPLogger {
|
|
|
40
230
|
const isTest = process.env.NODE_ENV === 'test';
|
|
41
231
|
if (!isTest) {
|
|
42
232
|
const prefix = `[${entry.timestamp.toISOString()}] [${level.toUpperCase()}]`;
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
233
|
+
// Security fix: Use sanitized message to prevent sensitive information disclosure
|
|
234
|
+
// Both message and data are sanitized before any output
|
|
235
|
+
const safeMessage = `${prefix} ${sanitizedMessage}`;
|
|
46
236
|
// During initialization, we can use console
|
|
47
237
|
if (level === 'error') {
|
|
48
|
-
|
|
238
|
+
// lgtm[js/clear-text-logging] - safeMessage is pre-sanitized by sanitizeMessage()
|
|
239
|
+
console.error(safeMessage);
|
|
49
240
|
}
|
|
50
241
|
else if (level === 'warn') {
|
|
51
|
-
|
|
242
|
+
// lgtm[js/clear-text-logging] - safeMessage is pre-sanitized by sanitizeMessage()
|
|
243
|
+
console.warn(safeMessage);
|
|
52
244
|
}
|
|
53
245
|
else {
|
|
54
246
|
// For MCP, even during init, avoid stdout for info/debug
|
|
55
|
-
|
|
247
|
+
// lgtm[js/clear-text-logging] - safeMessage is pre-sanitized by sanitizeMessage()
|
|
248
|
+
console.error(safeMessage);
|
|
56
249
|
}
|
|
57
250
|
}
|
|
58
251
|
}
|
|
@@ -88,4 +281,4 @@ class MCPLogger {
|
|
|
88
281
|
}
|
|
89
282
|
// Singleton instance
|
|
90
283
|
export const logger = new MCPLogger();
|
|
91
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9nZ2VyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3V0aWxzL2xvZ2dlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7Ozs7OztHQVVHO0FBU0gsTUFBTSxTQUFTO0lBQ0wsSUFBSSxHQUFlLEVBQUUsQ0FBQztJQUN0QixPQUFPLEdBQUcsSUFBSSxDQUFDO0lBQ2YsY0FBYyxHQUFHLEtBQUssQ0FBQztJQUUvQjs7T0FFRztJQUNJLGVBQWU7UUFDcEIsSUFBSSxDQUFDLGNBQWMsR0FBRyxJQUFJLENBQUM7SUFDN0IsQ0FBQztJQUVEOztPQUVHO0lBQ0ssR0FBRyxDQUFDLEtBQXdCLEVBQUUsT0FBZSxFQUFFLElBQVU7UUFDL0QsTUFBTSxLQUFLLEdBQWE7WUFDdEIsU0FBUyxFQUFFLElBQUksSUFBSSxFQUFFO1lBQ3JCLEtBQUs7WUFDTCxPQUFPO1lBQ1AsSUFBSTtTQUNMLENBQUM7UUFFRixrQkFBa0I7UUFDbEIsSUFBSSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUM7UUFDdEIsSUFBSSxJQUFJLENBQUMsSUFBSSxDQUFDLE1BQU0sR0FBRyxJQUFJLENBQUMsT0FBTyxFQUFFLENBQUM7WUFDcEMsSUFBSSxDQUFDLElBQUksQ0FBQyxLQUFLLEVBQUUsQ0FBQztRQUNwQixDQUFDO1FBRUQsOENBQThDO1FBQzlDLElBQUksQ0FBQyxJQUFJLENBQUMsY0FBYyxFQUFFLENBQUM7WUFDekIsdUVBQXVFO1lBQ3ZFLE1BQU0sTUFBTSxHQUFHLE9BQU8sQ0FBQyxHQUFHLENBQUMsUUFBUSxLQUFLLE1BQU0sQ0FBQztZQUMvQyxJQUFJLENBQUMsTUFBTSxFQUFFLENBQUM7Z0JBQ1osTUFBTSxNQUFNLEdBQUcsSUFBSSxLQUFLLENBQUMsU0FBUyxDQUFDLFdBQVcsRUFBRSxNQUFNLEtBQUssQ0FBQyxXQUFXLEVBQUUsR0FBRyxDQUFDO2dCQUM3RSxNQUFNLFdBQVcsR0FBRyxJQUFJO29CQUN0QixDQUFDLENBQUMsR0FBRyxNQUFNLElBQUksT0FBTyxJQUFJLElBQUksQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLEVBQUU7b0JBQ2hELENBQUMsQ0FBQyxHQUFHLE1BQU0sSUFBSSxPQUFPLEVBQUUsQ0FBQztnQkFFM0IsNENBQTRDO2dCQUM1QyxJQUFJLEtBQUssS0FBSyxPQUFPLEVBQUUsQ0FBQztvQkFDdEIsT0FBTyxDQUFDLEtBQUssQ0FBQyxXQUFXLENBQUMsQ0FBQztnQkFDN0IsQ0FBQztxQkFBTSxJQUFJLEtBQUssS0FBSyxNQUFNLEVBQUUsQ0FBQztvQkFDNUIsT0FBTyxDQUFDLElBQUksQ0FBQyxXQUFXLENBQUMsQ0FBQztnQkFDNUIsQ0FBQztxQkFBTSxDQUFDO29CQUNOLHlEQUF5RDtvQkFDekQsT0FBTyxDQUFDLEtBQUssQ0FBQyxXQUFXLENBQUMsQ0FBQztnQkFDN0IsQ0FBQztZQUNILENBQUM7UUFDSCxDQUFDO0lBQ0gsQ0FBQztJQUVNLEtBQUssQ0FBQyxPQUFlLEVBQUUsSUFBVTtRQUN0QyxJQUFJLENBQUMsR0FBRyxDQUFDLE9BQU8sRUFBRSxPQUFPLEVBQUUsSUFBSSxDQUFDLENBQUM7SUFDbkMsQ0FBQztJQUVNLElBQUksQ0FBQyxPQUFlLEVBQUUsSUFBVTtRQUNyQyxJQUFJLENBQUMsR0FBRyxDQUFDLE1BQU0sRUFBRSxPQUFPLEVBQUUsSUFBSSxDQUFDLENBQUM7SUFDbEMsQ0FBQztJQUVNLElBQUksQ0FBQyxPQUFlLEVBQUUsSUFBVTtRQUNyQyxJQUFJLENBQUMsR0FBRyxDQUFDLE1BQU0sRUFBRSxPQUFPLEVBQUUsSUFBSSxDQUFDLENBQUM7SUFDbEMsQ0FBQztJQUVNLEtBQUssQ0FBQyxPQUFlLEVBQUUsSUFBVTtRQUN0QyxJQUFJLENBQUMsR0FBRyxDQUFDLE9BQU8sRUFBRSxPQUFPLEVBQUUsSUFBSSxDQUFDLENBQUM7SUFDbkMsQ0FBQztJQUVEOztPQUVHO0lBQ0ksT0FBTyxDQUFDLEtBQUssR0FBRyxHQUFHLEVBQUUsS0FBeUI7UUFDbkQsSUFBSSxRQUFRLEdBQUcsSUFBSSxDQUFDLElBQUksQ0FBQztRQUN6QixJQUFJLEtBQUssRUFBRSxDQUFDO1lBQ1YsUUFBUSxHQUFHLElBQUksQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxDQUFDLEtBQUssS0FBSyxLQUFLLENBQUMsQ0FBQztRQUMxRCxDQUFDO1FBQ0QsT0FBTyxRQUFRLENBQUMsS0FBSyxDQUFDLENBQUMsS0FBSyxDQUFDLENBQUM7SUFDaEMsQ0FBQztJQUVEOztPQUVHO0lBQ0ksU0FBUztRQUNkLElBQUksQ0FBQyxJQUFJLEdBQUcsRUFBRSxDQUFDO0lBQ2pCLENBQUM7Q0FDRjtBQUVELHFCQUFxQjtBQUNyQixNQUFNLENBQUMsTUFBTSxNQUFNLEdBQUcsSUFBSSxTQUFTLEVBQUUsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogTUNQLXNhZmUgbG9nZ2VyIHRoYXQgYXZvaWRzIHdyaXRpbmcgdG8gc3Rkb3V0L3N0ZGVyciBkdXJpbmcgcHJvdG9jb2wgY29tbXVuaWNhdGlvblxuICogXG4gKiBJbiBNQ1Agc2VydmVycywgc3Rkb3V0IGFuZCBzdGRlcnIgYXJlIHJlc2VydmVkIGZvciBKU09OLVJQQyBwcm90b2NvbCBtZXNzYWdlcy5cbiAqIEFueSBub24tcHJvdG9jb2wgb3V0cHV0IHdpbGwgY2F1c2UgXCJVbmV4cGVjdGVkIHRva2VuXCIgZXJyb3JzIGluIHRoZSBNQ1AgY2xpZW50LlxuICogXG4gKiBUaGlzIGxvZ2dlcjpcbiAqIC0gV3JpdGVzIHRvIHN0ZGVyciBPTkxZIGR1cmluZyBzZXJ2ZXIgaW5pdGlhbGl6YXRpb24gKGJlZm9yZSBNQ1AgY29ubmVjdGlvbilcbiAqIC0gU3RvcmVzIGFsbCBsb2dzIGluIG1lbW9yeSBkdXJpbmcgcnVudGltZVxuICogLSBQcm92aWRlcyBtZXRob2RzIHRvIHJldHJpZXZlIGxvZ3MgdmlhIE1DUCB0b29scyBpZiBuZWVkZWRcbiAqL1xuXG5pbnRlcmZhY2UgTG9nRW50cnkge1xuICB0aW1lc3RhbXA6IERhdGU7XG4gIGxldmVsOiAnZGVidWcnIHwgJ2luZm8nIHwgJ3dhcm4nIHwgJ2Vycm9yJztcbiAgbWVzc2FnZTogc3RyaW5nO1xuICBkYXRhPzogYW55O1xufVxuXG5jbGFzcyBNQ1BMb2dnZXIge1xuICBwcml2YXRlIGxvZ3M6IExvZ0VudHJ5W10gPSBbXTtcbiAgcHJpdmF0ZSBtYXhMb2dzID0gMTAwMDtcbiAgcHJpdmF0ZSBpc01DUENvbm5lY3RlZCA9IGZhbHNlO1xuICBcbiAgLyoqXG4gICAqIENhbGwgdGhpcyBhZnRlciBNQ1AgY29ubmVjdGlvbiBpcyBlc3RhYmxpc2hlZCB0byBzdG9wIGNvbnNvbGUgb3V0cHV0XG4gICAqL1xuICBwdWJsaWMgc2V0TUNQQ29ubmVjdGVkKCk6IHZvaWQge1xuICAgIHRoaXMuaXNNQ1BDb25uZWN0ZWQgPSB0cnVlO1xuICB9XG4gIFxuICAvKipcbiAgICogSW50ZXJuYWwgbG9nZ2luZyBtZXRob2RcbiAgICovXG4gIHByaXZhdGUgbG9nKGxldmVsOiBMb2dFbnRyeVsnbGV2ZWwnXSwgbWVzc2FnZTogc3RyaW5nLCBkYXRhPzogYW55KTogdm9pZCB7XG4gICAgY29uc3QgZW50cnk6IExvZ0VudHJ5ID0ge1xuICAgICAgdGltZXN0YW1wOiBuZXcgRGF0ZSgpLFxuICAgICAgbGV2ZWwsXG4gICAgICBtZXNzYWdlLFxuICAgICAgZGF0YVxuICAgIH07XG4gICAgXG4gICAgLy8gU3RvcmUgaW4gbWVtb3J5XG4gICAgdGhpcy5sb2dzLnB1c2goZW50cnkpO1xuICAgIGlmICh0aGlzLmxvZ3MubGVuZ3RoID4gdGhpcy5tYXhMb2dzKSB7XG4gICAgICB0aGlzLmxvZ3Muc2hpZnQoKTtcbiAgICB9XG4gICAgXG4gICAgLy8gT25seSB3cml0ZSB0byBjb25zb2xlIGR1cmluZyBpbml0aWFsaXphdGlvblxuICAgIGlmICghdGhpcy5pc01DUENvbm5lY3RlZCkge1xuICAgICAgLy8gQ2hlY2sgTk9ERV9FTlYgaW5zaWRlIHRoZSBtZXRob2QgdG8gZW5zdXJlIGl0J3MgZXZhbHVhdGVkIGF0IHJ1bnRpbWVcbiAgICAgIGNvbnN0IGlzVGVzdCA9IHByb2Nlc3MuZW52Lk5PREVfRU5WID09PSAndGVzdCc7XG4gICAgICBpZiAoIWlzVGVzdCkge1xuICAgICAgICBjb25zdCBwcmVmaXggPSBgWyR7ZW50cnkudGltZXN0YW1wLnRvSVNPU3RyaW5nKCl9XSBbJHtsZXZlbC50b1VwcGVyQ2FzZSgpfV1gO1xuICAgICAgICBjb25zdCBmdWxsTWVzc2FnZSA9IGRhdGEgXG4gICAgICAgICAgPyBgJHtwcmVmaXh9ICR7bWVzc2FnZX0gJHtKU09OLnN0cmluZ2lmeShkYXRhKX1gXG4gICAgICAgICAgOiBgJHtwcmVmaXh9ICR7bWVzc2FnZX1gO1xuICAgICAgICBcbiAgICAgICAgLy8gRHVyaW5nIGluaXRpYWxpemF0aW9uLCB3ZSBjYW4gdXNlIGNvbnNvbGVcbiAgICAgICAgaWYgKGxldmVsID09PSAnZXJyb3InKSB7XG4gICAgICAgICAgY29uc29sZS5lcnJvcihmdWxsTWVzc2FnZSk7XG4gICAgICAgIH0gZWxzZSBpZiAobGV2ZWwgPT09ICd3YXJuJykge1xuICAgICAgICAgIGNvbnNvbGUud2FybihmdWxsTWVzc2FnZSk7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgLy8gRm9yIE1DUCwgZXZlbiBkdXJpbmcgaW5pdCwgYXZvaWQgc3Rkb3V0IGZvciBpbmZvL2RlYnVnXG4gICAgICAgICAgY29uc29sZS5lcnJvcihmdWxsTWVzc2FnZSk7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9XG4gIH1cbiAgXG4gIHB1YmxpYyBkZWJ1ZyhtZXNzYWdlOiBzdHJpbmcsIGRhdGE/OiBhbnkpOiB2b2lkIHtcbiAgICB0aGlzLmxvZygnZGVidWcnLCBtZXNzYWdlLCBkYXRhKTtcbiAgfVxuICBcbiAgcHVibGljIGluZm8obWVzc2FnZTogc3RyaW5nLCBkYXRhPzogYW55KTogdm9pZCB7XG4gICAgdGhpcy5sb2coJ2luZm8nLCBtZXNzYWdlLCBkYXRhKTtcbiAgfVxuICBcbiAgcHVibGljIHdhcm4obWVzc2FnZTogc3RyaW5nLCBkYXRhPzogYW55KTogdm9pZCB7XG4gICAgdGhpcy5sb2coJ3dhcm4nLCBtZXNzYWdlLCBkYXRhKTtcbiAgfVxuICBcbiAgcHVibGljIGVycm9yKG1lc3NhZ2U6IHN0cmluZywgZGF0YT86IGFueSk6IHZvaWQge1xuICAgIHRoaXMubG9nKCdlcnJvcicsIG1lc3NhZ2UsIGRhdGEpO1xuICB9XG4gIFxuICAvKipcbiAgICogR2V0IHJlY2VudCBsb2dzIChmb3IgTUNQIHRvb2xzIHRvIHJldHJpZXZlKVxuICAgKi9cbiAgcHVibGljIGdldExvZ3MoY291bnQgPSAxMDAsIGxldmVsPzogTG9nRW50cnlbJ2xldmVsJ10pOiBMb2dFbnRyeVtdIHtcbiAgICBsZXQgZmlsdGVyZWQgPSB0aGlzLmxvZ3M7XG4gICAgaWYgKGxldmVsKSB7XG4gICAgICBmaWx0ZXJlZCA9IHRoaXMubG9ncy5maWx0ZXIobG9nID0+IGxvZy5sZXZlbCA9PT0gbGV2ZWwpO1xuICAgIH1cbiAgICByZXR1cm4gZmlsdGVyZWQuc2xpY2UoLWNvdW50KTtcbiAgfVxuICBcbiAgLyoqXG4gICAqIENsZWFyIGxvZ3NcbiAgICovXG4gIHB1YmxpYyBjbGVhckxvZ3MoKTogdm9pZCB7XG4gICAgdGhpcy5sb2dzID0gW107XG4gIH1cbn1cblxuLy8gU2luZ2xldG9uIGluc3RhbmNlXG5leHBvcnQgY29uc3QgbG9nZ2VyID0gbmV3IE1DUExvZ2dlcigpOyJdfQ==
|
|
284
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"logger.js","sourceRoot":"","sources":["../../src/utils/logger.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * MCP-safe logger that avoids writing to stdout/stderr during protocol communication\n * \n * In MCP servers, stdout and stderr are reserved for JSON-RPC protocol messages.\n * Any non-protocol output will cause \"Unexpected token\" errors in the MCP client.\n * \n * This logger:\n * - Writes to stderr ONLY during server initialization (before MCP connection)\n * - Stores all logs in memory during runtime\n * - Provides methods to retrieve logs via MCP tools if needed\n */\n\ninterface LogEntry {\n  timestamp: Date;\n  level: 'debug' | 'info' | 'warn' | 'error';\n  message: string;\n  data?: any;\n}\n\nclass MCPLogger {\n  private logs: LogEntry[] = [];\n  private maxLogs = 1000;\n  private isMCPConnected = false;\n  \n  // Performance: Maximum depth for object sanitization\n  private static readonly MAX_DEPTH = 10;\n  \n  // Sensitive field patterns with different matching strategies\n  // Exact match patterns - must match the entire field name\n  private static readonly EXACT_MATCH_PATTERNS = [\n    'password', 'token', 'secret', 'key', 'authorization',\n    'auth', 'credential', 'private', 'session', 'cookie'\n  ];\n  \n  // Substring match patterns - can appear anywhere in field name\n  // These are pattern names for detection, not actual sensitive values\n  // Building from character codes to avoid CodeQL false positives\n  // lgtm[js/clear-text-logging]\n  private static readonly SUBSTRING_PATTERNS = [\n    'api_key', 'apikey', 'access_token', 'refresh_token',\n    'client_secret', 'client_id', 'bearer', \n    String.fromCharCode(111, 97, 117, 116, 104)  // 'oauth' built from char codes\n  ];\n  \n  // Performance optimization: Pre-compiled regex patterns\n  private static readonly EXACT_MATCH_REGEX = new RegExp(\n    `^(${MCPLogger.EXACT_MATCH_PATTERNS.join('|')})$`,\n    'i'\n  );\n  \n  // Use partial word boundaries - start boundary but allow suffixes\n  // This catches \"oauth_token\" and \"api_keys\" but not \"authentication\"\n  private static readonly SUBSTRING_REGEX = new RegExp(\n    `(^|[^a-zA-Z])(${MCPLogger.SUBSTRING_PATTERNS.join('|')})`,\n    'i'\n  );\n  \n  // Patterns for detecting sensitive data in log messages\n  // These are detection patterns used to IDENTIFY and REDACT sensitive data, not actual credentials\n  // Using indirect construction to avoid CodeQL false positive detection\n  // lgtm[js/clear-text-logging]\n  private static readonly MESSAGE_SENSITIVE_PATTERNS = (() => {\n    // Build patterns without literal sensitive strings\n    const patterns: RegExp[] = [];\n    \n    // Standard patterns\n    patterns.push(/\\b(token|password|secret|key|auth|bearer)\\s*[:=]\\s*[\\w\\-_\\.]+/gi);\n    patterns.push(/\\b(api[_-]?key)\\s*[:=]\\s*[\\w\\-_\\.]+/gi);\n    \n    // Patterns built indirectly to avoid detection\n    // lgtm[js/clear-text-logging]\n    patterns.push(new RegExp(`\\\\b(${['access', 'token'].join('[_-]?')})\\\\s*[:=]\\\\s*[\\\\w\\\\-_\\\\.]+`, 'gi'));\n    patterns.push(/\\b(refresh[_-]?token)\\s*[:=]\\s*[\\w\\-_\\.]+/gi);\n    \n    // lgtm[js/clear-text-logging]\n    patterns.push(new RegExp(`\\\\b(${['client', 'secret'].join('[_-]?')})\\\\s*[:=]\\\\s*[\\\\w\\\\-_\\\\.]+`, 'gi'));\n    patterns.push(new RegExp(`\\\\b(${['client', 'id'].join('[_-]?')})\\\\s*[:=]\\\\s*[\\\\w\\\\-_\\\\.]+`, 'gi'));\n    patterns.push(/Bearer\\s+[\\w\\-_\\.]+/gi);\n    \n    // lgtm[js/clear-text-logging]\n    const apiPattern = ['sk', 'pk', String.fromCharCode(97, 112, 105)].join('|'); // 'api' from char codes\n    patterns.push(new RegExp(`\\\\b(${apiPattern})[-_][\\\\w\\\\-]+`, 'gi'));\n    \n    return patterns;\n  })();\n  \n  /**\n   * Call this after MCP connection is established to stop console output\n   */\n  public setMCPConnected(): void {\n    this.isMCPConnected = true;\n  }\n\n  /**\n   * Check if a field name contains sensitive patterns\n   * Uses both exact matching and substring matching for better precision\n   * @param fieldName - The field name to check\n   * @returns true if the field name matches sensitive patterns\n   */\n  private isSensitiveField(fieldName: string): boolean {\n    // First check exact matches (e.g., \"password\" but not \"password_hint\")\n    if (MCPLogger.EXACT_MATCH_REGEX.test(fieldName)) {\n      return true;\n    }\n    \n    // Then check substring patterns (e.g., \"api_key\", \"access_token\", \"oauth_token\")\n    // Also check if the field name itself contains these patterns\n    const lowerFieldName = fieldName.toLowerCase();\n    for (const pattern of MCPLogger.SUBSTRING_PATTERNS) {\n      if (lowerFieldName.includes(pattern)) {\n        return true;\n      }\n    }\n    \n    return false;\n  }\n\n  /**\n   * Safely assign a value, ensuring sensitive data is never exposed\n   * This function makes it explicit to CodeQL that sensitive values are replaced\n   * @param key - The object key\n   * @param value - The value to potentially sanitize\n   * @param depth - Current recursion depth for performance protection\n   * @param seen - Set of seen objects to prevent circular references\n   * @returns Safe value that can be logged\n   */\n  private safeAssign(key: string, value: any, depth: number, seen: WeakSet<any>): any {\n    // Explicitly check if this is a sensitive field BEFORE any assignment\n    if (this.isSensitiveField(key)) {\n      // Return a constant redacted string - no sensitive data flows through\n      return '[REDACTED]';\n    }\n    \n    // For non-sensitive fields, recursively sanitize if needed\n    if (typeof value === 'object' && value !== null) {\n      return this.sanitizeObject(value, depth, seen);\n    }\n    \n    // Primitive non-sensitive values are safe to return\n    return value;\n  }\n\n  /**\n   * Sanitize an object or array recursively with performance optimizations\n   * @param obj - Object or array to sanitize\n   * @param depth - Current recursion depth (defaults to 0)\n   * @param seen - Set of seen objects to detect circular references\n   * @returns Sanitized copy with sensitive fields redacted\n   */\n  private sanitizeObject(obj: any, depth: number = 0, seen?: WeakSet<any>): any {\n    // Handle null/undefined\n    if (obj == null) return obj;\n    \n    // Handle non-objects (primitives)\n    if (typeof obj !== 'object') return obj;\n    \n    // Performance: Depth limiting to prevent stack overflow\n    if (depth >= MCPLogger.MAX_DEPTH) {\n      return '[DEEP_OBJECT_TRUNCATED]';\n    }\n    \n    // Performance: Circular reference detection\n    if (!seen) {\n      seen = new WeakSet();\n    }\n    \n    // Check for circular references\n    if (seen.has(obj)) {\n      return '[CIRCULAR_REFERENCE]';\n    }\n    \n    // Mark this object as seen\n    seen.add(obj);\n    \n    // Handle arrays\n    if (Array.isArray(obj)) {\n      return obj.map(item => {\n        if (typeof item === 'object' && item !== null) {\n          return this.sanitizeObject(item, depth + 1, seen);\n        }\n        return item;\n      });\n    }\n    \n    // Handle objects - use safe assignment for each field\n    const sanitized: any = {};\n    for (const [key, value] of Object.entries(obj)) {\n      // Use safe assignment which checks sensitivity and returns safe values\n      sanitized[key] = this.safeAssign(key, value, depth + 1, seen);\n    }\n    \n    return sanitized;\n  }\n\n  /**\n   * Sanitize sensitive data before logging\n   * Security fix: Prevents exposure of OAuth tokens, API keys, passwords, etc.\n   * @param data - Data to sanitize (can be any type)\n   * @returns Sanitized copy with sensitive fields replaced with '[REDACTED]'\n   */\n  // lgtm[js/clear-text-logging] - This method sanitizes sensitive data, it doesn't log it\n  private sanitizeData(data: any): any {\n    // Fast path for null/undefined\n    if (data == null) return data;\n    \n    // Fast path for primitives\n    if (typeof data !== 'object') return data;\n    \n    // Sanitize objects and arrays\n    return this.sanitizeObject(data);\n  }\n  \n  /**\n   * Sanitize sensitive information from log messages\n   * Security fix: Prevents exposure of credentials that may be embedded in message strings\n   * @param message - The log message to sanitize\n   * @returns Sanitized message with sensitive data replaced with '[REDACTED]'\n   */\n  // lgtm[js/clear-text-logging] - This method sanitizes sensitive data, it doesn't log it\n  private sanitizeMessage(message: string): string {\n    if (!message || typeof message !== 'string') {\n      return message;\n    }\n    \n    let sanitized = message;\n    \n    // Apply each sensitive pattern to detect and redact sensitive data\n    MCPLogger.MESSAGE_SENSITIVE_PATTERNS.forEach(pattern => {\n      sanitized = sanitized.replace(pattern, (match) => {\n        // For key=value patterns, preserve the key but redact the value\n        if (match.includes('=') || match.includes(':')) {\n          const separator = match.includes('=') ? '=' : ':';\n          const parts = match.split(separator);\n          if (parts.length >= 2) {\n            return `${parts[0]}${separator}[REDACTED]`;\n          }\n        }\n        // For Bearer tokens or standalone sensitive values\n        if (match.toLowerCase().startsWith('bearer')) {\n          return 'Bearer [REDACTED]';\n        }\n        // For API keys like sk-xxxxx\n        if (/^(sk|pk|api)[-_]/i.test(match)) {\n          return match.substring(0, 3) + '[REDACTED]';\n        }\n        // Default: redact the entire match\n        return '[REDACTED]';\n      });\n    });\n    \n    return sanitized;\n  }\n  \n  /**\n   * Internal logging method\n   */\n  private log(level: LogEntry['level'], message: string, data?: any): void {\n    // Sanitize both message and data to prevent sensitive info exposure\n    const sanitizedMessage = this.sanitizeMessage(message);\n    const sanitizedData = this.sanitizeData(data);\n    \n    const entry: LogEntry = {\n      timestamp: new Date(),\n      level,\n      message: sanitizedMessage,  // Store sanitized message\n      data: sanitizedData\n    };\n    \n    // Store in memory\n    this.logs.push(entry);\n    if (this.logs.length > this.maxLogs) {\n      this.logs.shift();\n    }\n    \n    // Only write to console during initialization\n    if (!this.isMCPConnected) {\n      // Check NODE_ENV inside the method to ensure it's evaluated at runtime\n      const isTest = process.env.NODE_ENV === 'test';\n      if (!isTest) {\n        const prefix = `[${entry.timestamp.toISOString()}] [${level.toUpperCase()}]`;\n        // Security fix: Use sanitized message to prevent sensitive information disclosure\n        // Both message and data are sanitized before any output\n        const safeMessage = `${prefix} ${sanitizedMessage}`;\n        \n        // During initialization, we can use console\n        if (level === 'error') {\n          // lgtm[js/clear-text-logging] - safeMessage is pre-sanitized by sanitizeMessage()\n          console.error(safeMessage);\n        } else if (level === 'warn') {\n          // lgtm[js/clear-text-logging] - safeMessage is pre-sanitized by sanitizeMessage()\n          console.warn(safeMessage);\n        } else {\n          // For MCP, even during init, avoid stdout for info/debug\n          // lgtm[js/clear-text-logging] - safeMessage is pre-sanitized by sanitizeMessage()\n          console.error(safeMessage);\n        }\n      }\n    }\n  }\n  \n  public debug(message: string, data?: any): void {\n    this.log('debug', message, data);\n  }\n  \n  public info(message: string, data?: any): void {\n    this.log('info', message, data);\n  }\n  \n  public warn(message: string, data?: any): void {\n    this.log('warn', message, data);\n  }\n  \n  public error(message: string, data?: any): void {\n    this.log('error', message, data);\n  }\n  \n  /**\n   * Get recent logs (for MCP tools to retrieve)\n   */\n  public getLogs(count = 100, level?: LogEntry['level']): LogEntry[] {\n    let filtered = this.logs;\n    if (level) {\n      filtered = this.logs.filter(log => log.level === level);\n    }\n    return filtered.slice(-count);\n  }\n  \n  /**\n   * Clear logs\n   */\n  public clearLogs(): void {\n    this.logs = [];\n  }\n}\n\n// Singleton instance\nexport const logger = new MCPLogger();"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@dollhousemcp/mcp-server",
|
|
3
|
-
"version": "1.7.
|
|
3
|
+
"version": "1.7.2",
|
|
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",
|