mohen 1.0.0 → 1.1.0

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 CHANGED
@@ -1,6 +1,21 @@
1
- # mohen 墨痕
1
+ <p align="center">
2
+ <img src="logo.png" alt="mohen logo" width="200" />
3
+ </p>
2
4
 
3
- A simple, unified request/response logger for Express and tRPC that writes to a single file with JSON lines format.
5
+ <h1 align="center">mohen 墨痕</h1>
6
+
7
+ <p align="center">
8
+ <strong>A simple, unified request/response logger for Express and tRPC</strong><br>
9
+ Writes to a single file with JSON lines format
10
+ </p>
11
+
12
+ <p align="center">
13
+ <a href="https://www.npmjs.com/package/mohen"><img src="https://img.shields.io/npm/v/mohen.svg" alt="npm version"></a>
14
+ <a href="https://github.com/ivanleomk/mohen/actions"><img src="https://github.com/ivanleomk/mohen/actions/workflows/ci.yml/badge.svg" alt="CI"></a>
15
+ <a href="https://github.com/ivanleomk/mohen/blob/master/LICENSE"><img src="https://img.shields.io/npm/l/mohen.svg" alt="license"></a>
16
+ </p>
17
+
18
+ ---
4
19
 
5
20
  ## Features
6
21
 
@@ -131,9 +146,34 @@ createLogger(filePath, {
131
146
  maxSizeBytes: 10 * 1024 * 1024, // Max file size before truncation (default: 10MB)
132
147
  includeHeaders: false, // Log request headers (default: false)
133
148
  redact: ['password', 'token'], // Fields to redact (default: password, token, authorization, cookie)
149
+ ignorePaths: ['/health', '/health/*', '/metrics'], // Paths to skip logging (supports wildcards)
150
+ includePaths: ['/api/*'], // Only log these paths (supports wildcards)
151
+ });
152
+ ```
153
+
154
+ ### Path Filtering
155
+
156
+ Use `ignorePaths` to skip noisy endpoints like health checks:
157
+
158
+ ```typescript
159
+ const logger = createLogger('./logs/app.log', {
160
+ ignorePaths: ['/health', '/health/*', '/metrics', '/favicon.ico'],
134
161
  });
135
162
  ```
136
163
 
164
+ Or use `includePaths` to only log specific routes:
165
+
166
+ ```typescript
167
+ const logger = createLogger('./logs/app.log', {
168
+ includePaths: ['/api/*', '/trpc/*'],
169
+ });
170
+ ```
171
+
172
+ Wildcard patterns:
173
+ - `/health` - matches exactly `/health`
174
+ - `/health/*` - matches `/health/live`, `/health/ready`, etc.
175
+ - `/api/*` - matches any path starting with `/api/`
176
+
137
177
  ## Log Format
138
178
 
139
179
  Each line is a JSON object with the following structure:
@@ -162,6 +202,20 @@ Each line is a JSON object with the following structure:
162
202
  }
163
203
  ```
164
204
 
205
+ For SSE streaming responses with text-delta events (like LLM responses), the text is automatically aggregated:
206
+
207
+ ```json
208
+ {
209
+ "type": "http",
210
+ "path": "/api/chat",
211
+ "response": {
212
+ "streaming": true,
213
+ "chunks": [{"type": "start"}, {"type": "text-delta", "delta": "Hello"}, ...],
214
+ "text": "Hello world! This is the complete aggregated response."
215
+ }
216
+ }
217
+ ```
218
+
165
219
  ## File Size Management
166
220
 
167
221
  When the log file exceeds `maxSizeBytes`, the oldest 75% of log entries are removed, keeping the most recent 25%. This happens automatically before each write.
package/dist/logger.d.ts CHANGED
@@ -16,6 +16,7 @@ interface LogEntry {
16
16
  body?: unknown;
17
17
  streaming?: boolean;
18
18
  chunks?: unknown[];
19
+ text?: string;
19
20
  };
20
21
  error?: {
21
22
  message: string;
@@ -27,6 +28,8 @@ interface LoggerOptions {
27
28
  maxSizeBytes?: number;
28
29
  includeHeaders?: boolean;
29
30
  redact?: string[];
31
+ ignorePaths?: string[];
32
+ includePaths?: string[];
30
33
  }
31
34
  declare global {
32
35
  namespace Express {
package/dist/logger.js CHANGED
@@ -47,6 +47,8 @@ class UnifiedLogger {
47
47
  this.maxSizeBytes = options.maxSizeBytes ?? 10 * 1024 * 1024; // 10MB default
48
48
  this.includeHeaders = options.includeHeaders ?? false;
49
49
  this.redactFields = new Set(options.redact ?? ['password', 'token', 'authorization', 'cookie']);
50
+ this.ignorePaths = options.ignorePaths ?? [];
51
+ this.includePaths = options.includePaths ?? [];
50
52
  // Ensure directory exists
51
53
  const dir = path.dirname(this.filePath);
52
54
  if (!fs.existsSync(dir)) {
@@ -56,6 +58,29 @@ class UnifiedLogger {
56
58
  generateRequestId() {
57
59
  return `${Date.now().toString(36)}-${Math.random().toString(36).slice(2, 9)}`;
58
60
  }
61
+ matchPath(pattern, requestPath) {
62
+ // Convert wildcard pattern to regex
63
+ // /api/* matches /api/anything
64
+ // /health matches exactly /health
65
+ const regexPattern = pattern
66
+ .replace(/[.+?^${}()|[\]\\]/g, '\\$&') // Escape special regex chars except *
67
+ .replace(/\*/g, '.*'); // Convert * to .*
68
+ const regex = new RegExp(`^${regexPattern}$`);
69
+ return regex.test(requestPath);
70
+ }
71
+ shouldLog(requestPath) {
72
+ // Extract path without query string
73
+ const pathOnly = requestPath.split('?')[0];
74
+ // If includePaths is set, only log matching paths
75
+ if (this.includePaths.length > 0) {
76
+ return this.includePaths.some((pattern) => this.matchPath(pattern, pathOnly));
77
+ }
78
+ // If ignorePaths is set, skip matching paths
79
+ if (this.ignorePaths.length > 0) {
80
+ return !this.ignorePaths.some((pattern) => this.matchPath(pattern, pathOnly));
81
+ }
82
+ return true;
83
+ }
59
84
  redact(obj) {
60
85
  if (obj === null || obj === undefined)
61
86
  return obj;
@@ -112,6 +137,11 @@ class UnifiedLogger {
112
137
  // ===========================================================================
113
138
  expressMiddleware() {
114
139
  return (req, res, next) => {
140
+ const requestPath = req.originalUrl || req.url;
141
+ // Check if we should log this path
142
+ if (!this.shouldLog(requestPath)) {
143
+ return next();
144
+ }
115
145
  const start = Date.now();
116
146
  const requestId = this.generateRequestId();
117
147
  // Initialize metadata object on request
@@ -119,6 +149,7 @@ class UnifiedLogger {
119
149
  // Detect SSE - check both request Accept header and response Content-Type
120
150
  let isSSE = req.headers.accept === 'text/event-stream';
121
151
  const chunks = [];
152
+ const textDeltas = []; // Collect text-delta content
122
153
  // Intercept setHeader to detect SSE by Content-Type
123
154
  const originalSetHeader = res.setHeader.bind(res);
124
155
  res.setHeader = ((name, value) => {
@@ -148,7 +179,7 @@ class UnifiedLogger {
148
179
  // If write is called, treat as streaming
149
180
  if (chunk && isSSE) {
150
181
  const chunkStr = chunk.toString();
151
- const parsed = this.parseSSEChunk(chunkStr);
182
+ const parsed = this.parseSSEChunk(chunkStr, textDeltas);
152
183
  if (parsed) {
153
184
  chunks.push(parsed);
154
185
  }
@@ -163,7 +194,7 @@ class UnifiedLogger {
163
194
  // SSE streaming path
164
195
  if (chunk) {
165
196
  const chunkStr = chunk.toString();
166
- const parsed = this.parseSSEChunk(chunkStr);
197
+ const parsed = this.parseSSEChunk(chunkStr, textDeltas);
167
198
  if (parsed) {
168
199
  chunks.push(parsed);
169
200
  }
@@ -173,7 +204,7 @@ class UnifiedLogger {
173
204
  requestId,
174
205
  type: 'http',
175
206
  method: req.method,
176
- path: req.originalUrl || req.url,
207
+ path: requestPath,
177
208
  statusCode: res.statusCode,
178
209
  duration: Date.now() - start,
179
210
  request: requestInfo,
@@ -182,6 +213,10 @@ class UnifiedLogger {
182
213
  chunks,
183
214
  },
184
215
  };
216
+ // Add aggregated text if we collected text-deltas
217
+ if (textDeltas.length > 0) {
218
+ entry.response.text = textDeltas.join('');
219
+ }
185
220
  if (req.logMetadata && Object.keys(req.logMetadata).length > 0) {
186
221
  entry.metadata = req.logMetadata;
187
222
  }
@@ -194,7 +229,7 @@ class UnifiedLogger {
194
229
  requestId,
195
230
  type: 'http',
196
231
  method: req.method,
197
- path: req.originalUrl || req.url,
232
+ path: requestPath,
198
233
  statusCode: res.statusCode,
199
234
  duration: Date.now() - start,
200
235
  request: requestInfo,
@@ -219,7 +254,7 @@ class UnifiedLogger {
219
254
  requestId,
220
255
  type: 'http',
221
256
  method: req.method,
222
- path: req.originalUrl || req.url,
257
+ path: requestPath,
223
258
  statusCode: res.statusCode,
224
259
  duration: Date.now() - start,
225
260
  request: requestInfo,
@@ -253,22 +288,35 @@ class UnifiedLogger {
253
288
  next();
254
289
  };
255
290
  }
256
- parseSSEChunk(raw) {
291
+ parseSSEChunk(raw, textDeltas) {
257
292
  const lines = raw.split('\n');
293
+ const results = [];
258
294
  for (const line of lines) {
259
295
  if (line.startsWith('data: ')) {
260
296
  const data = line.slice(6).trim();
261
- if (data === '[DONE]')
262
- return { done: true };
297
+ if (data === '[DONE]') {
298
+ results.push({ type: 'done' });
299
+ continue;
300
+ }
263
301
  try {
264
- return JSON.parse(data);
302
+ const parsed = JSON.parse(data);
303
+ // Handle text-delta type - collect the delta text
304
+ if (parsed.type === 'text-delta' && typeof parsed.delta === 'string') {
305
+ textDeltas.push(parsed.delta);
306
+ }
307
+ results.push(parsed);
265
308
  }
266
309
  catch {
267
- return { raw: data };
310
+ results.push({ raw: data });
268
311
  }
269
312
  }
270
313
  }
271
- return null;
314
+ // Return single result or array
315
+ if (results.length === 0)
316
+ return null;
317
+ if (results.length === 1)
318
+ return results[0];
319
+ return results;
272
320
  }
273
321
  // ===========================================================================
274
322
  // tRPC Middleware
@@ -276,6 +324,10 @@ class UnifiedLogger {
276
324
  trpcMiddleware() {
277
325
  const logger = this;
278
326
  return async function loggerMiddleware(opts) {
327
+ // Check if we should log this path
328
+ if (!logger.shouldLog(opts.path)) {
329
+ return opts.next();
330
+ }
279
331
  const start = Date.now();
280
332
  const requestId = logger.generateRequestId();
281
333
  // Initialize metadata on context if not present
@@ -386,4 +438,4 @@ function attachTrpcMetadata(ctx, metadata) {
386
438
  }
387
439
  // Default export for simpler imports
388
440
  exports.default = createLogger;
389
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"logger.js","sourceRoot":"","sources":["../src/logger.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs';\nimport * as path from 'path';\nimport type { Request, Response, NextFunction } from 'express';\n\n// ============================================================================\n// Types\n// ============================================================================\n\ninterface LogEntry {\n  timestamp: string;\n  requestId: string;\n  type: 'http' | 'trpc';\n  method: string;\n  path: string;\n  statusCode?: number;\n  duration: number;\n  request?: {\n    body?: unknown;\n    query?: unknown;\n    headers?: Record<string, string>;\n  };\n  response?: {\n    body?: unknown;\n    streaming?: boolean;\n    chunks?: unknown[];\n  };\n  error?: {\n    message: string;\n    stack?: string;\n  };\n  metadata?: Record<string, unknown>;\n}\n\ninterface LoggerOptions {\n  maxSizeBytes?: number;      // Default: 10MB\n  includeHeaders?: boolean;   // Default: false\n  redact?: string[];          // Fields to redact from logs\n}\n\n// Extend Express Request to include metadata\ndeclare global {\n  namespace Express {\n    interface Request {\n      logMetadata?: Record<string, unknown>;\n    }\n  }\n}\n\n// ============================================================================\n// Core Logger Class\n// ============================================================================\n\nclass UnifiedLogger {\n  private filePath: string;\n  private maxSizeBytes: number;\n  private includeHeaders: boolean;\n  private redactFields: Set<string>;\n\n  constructor(filePath: string, options: LoggerOptions = {}) {\n    this.filePath = path.resolve(filePath);\n    this.maxSizeBytes = options.maxSizeBytes ?? 10 * 1024 * 1024; // 10MB default\n    this.includeHeaders = options.includeHeaders ?? false;\n    this.redactFields = new Set(options.redact ?? ['password', 'token', 'authorization', 'cookie']);\n\n    // Ensure directory exists\n    const dir = path.dirname(this.filePath);\n    if (!fs.existsSync(dir)) {\n      fs.mkdirSync(dir, { recursive: true });\n    }\n  }\n\n  private generateRequestId(): string {\n    return `${Date.now().toString(36)}-${Math.random().toString(36).slice(2, 9)}`;\n  }\n\n  private redact(obj: unknown): unknown {\n    if (obj === null || obj === undefined) return obj;\n    if (typeof obj !== 'object') return obj;\n\n    if (Array.isArray(obj)) {\n      return obj.map((item) => this.redact(item));\n    }\n\n    const result: Record<string, unknown> = {};\n    for (const [key, value] of Object.entries(obj as Record<string, unknown>)) {\n      if (this.redactFields.has(key.toLowerCase())) {\n        result[key] = '[REDACTED]';\n      } else if (typeof value === 'object') {\n        result[key] = this.redact(value);\n      } else {\n        result[key] = value;\n      }\n    }\n    return result;\n  }\n\n  private checkAndRotate(): void {\n    try {\n      if (!fs.existsSync(this.filePath)) return;\n\n      const stats = fs.statSync(this.filePath);\n      if (stats.size > this.maxSizeBytes) {\n        // Read file, keep last 25% of lines\n        const content = fs.readFileSync(this.filePath, 'utf-8');\n        const lines = content.trim().split('\\n');\n        const keepCount = Math.floor(lines.length * 0.25);\n        const newContent = lines.slice(-keepCount).join('\\n') + '\\n';\n        fs.writeFileSync(this.filePath, newContent);\n      }\n    } catch (err) {\n      console.error('Logger rotation error:', err);\n    }\n  }\n\n  write(entry: LogEntry): void {\n    try {\n      this.checkAndRotate();\n      const redactedEntry = this.redact(entry) as LogEntry;\n      const line = JSON.stringify(redactedEntry) + '\\n';\n      fs.appendFileSync(this.filePath, line);\n    } catch (err) {\n      console.error('Logger write error:', err);\n    }\n  }\n\n  // ===========================================================================\n  // Express Middleware\n  // ===========================================================================\n\n  expressMiddleware() {\n    return (req: Request, res: Response, next: NextFunction) => {\n      const start = Date.now();\n      const requestId = this.generateRequestId();\n\n      // Initialize metadata object on request\n      req.logMetadata = {};\n\n      // Detect SSE - check both request Accept header and response Content-Type\n      let isSSE = req.headers.accept === 'text/event-stream';\n      const chunks: unknown[] = [];\n      \n      // Intercept setHeader to detect SSE by Content-Type\n      const originalSetHeader = res.setHeader.bind(res);\n      res.setHeader = ((name: string, value: string | number | readonly string[]): Response => {\n        if (name.toLowerCase() === 'content-type' && \n            typeof value === 'string' && \n            value.includes('text/event-stream')) {\n          isSSE = true;\n        }\n        return originalSetHeader(name, value);\n      }) as typeof res.setHeader;\n\n      // Capture request info\n      const requestInfo: LogEntry['request'] = {\n        body: req.body,\n        query: req.query,\n      };\n\n      if (this.includeHeaders) {\n        requestInfo.headers = req.headers as Record<string, string>;\n      }\n\n      // Intercept write/end for streaming detection\n      const originalWrite = res.write.bind(res);\n      const originalEnd = res.end.bind(res);\n      const originalJson = res.json.bind(res);\n      const originalSend = res.send.bind(res);\n      let responseBody: unknown;\n      let logged = false;\n\n      res.write = ((chunk: any, encodingOrCallback?: any, callback?: any): boolean => {\n        // If write is called, treat as streaming\n        if (chunk && isSSE) {\n          const chunkStr = chunk.toString();\n          const parsed = this.parseSSEChunk(chunkStr);\n          if (parsed) {\n            chunks.push(parsed);\n          }\n        }\n        return originalWrite(chunk, encodingOrCallback, callback);\n      }) as typeof res.write;\n\n      res.end = ((chunk?: any, encodingOrCallback?: any, callback?: any): Response => {\n        if (logged) return originalEnd(chunk, encodingOrCallback, callback);\n        logged = true;\n\n        if (isSSE) {\n          // SSE streaming path\n          if (chunk) {\n            const chunkStr = chunk.toString();\n            const parsed = this.parseSSEChunk(chunkStr);\n            if (parsed) {\n              chunks.push(parsed);\n            }\n          }\n\n          const entry: LogEntry = {\n            timestamp: new Date().toISOString(),\n            requestId,\n            type: 'http',\n            method: req.method,\n            path: req.originalUrl || req.url,\n            statusCode: res.statusCode,\n            duration: Date.now() - start,\n            request: requestInfo,\n            response: {\n              streaming: true,\n              chunks,\n            },\n          };\n\n          if (req.logMetadata && Object.keys(req.logMetadata).length > 0) {\n            entry.metadata = req.logMetadata;\n          }\n\n          this.write(entry);\n        } else {\n          // Regular response path\n          const entry: LogEntry = {\n            timestamp: new Date().toISOString(),\n            requestId,\n            type: 'http',\n            method: req.method,\n            path: req.originalUrl || req.url,\n            statusCode: res.statusCode,\n            duration: Date.now() - start,\n            request: requestInfo,\n            response: {\n              body: responseBody,\n              streaming: false,\n            },\n          };\n\n          if (req.logMetadata && Object.keys(req.logMetadata).length > 0) {\n            entry.metadata = req.logMetadata;\n          }\n\n          this.write(entry);\n        }\n\n        return originalEnd(chunk, encodingOrCallback, callback);\n      }) as typeof res.end;\n\n      const logResponse = () => {\n        if (logged) return;\n        logged = true;\n\n        const entry: LogEntry = {\n          timestamp: new Date().toISOString(),\n          requestId,\n          type: 'http',\n          method: req.method,\n          path: req.originalUrl || req.url,\n          statusCode: res.statusCode,\n          duration: Date.now() - start,\n          request: requestInfo,\n          response: {\n            body: responseBody,\n            streaming: false,\n          },\n        };\n\n        if (req.logMetadata && Object.keys(req.logMetadata).length > 0) {\n          entry.metadata = req.logMetadata;\n        }\n\n        this.write(entry);\n      };\n\n      res.json = (body: any) => {\n        responseBody = body;\n        logResponse();\n        return originalJson(body);\n      };\n\n      res.send = (body: any) => {\n        if (!logged) {\n          try {\n            responseBody = typeof body === 'string' ? JSON.parse(body) : body;\n          } catch {\n            responseBody = body;\n          }\n          logResponse();\n        }\n        return originalSend(body);\n      };\n\n      next();\n    };\n  }\n\n  private parseSSEChunk(raw: string): unknown {\n    const lines = raw.split('\\n');\n    for (const line of lines) {\n      if (line.startsWith('data: ')) {\n        const data = line.slice(6).trim();\n        if (data === '[DONE]') return { done: true };\n        try {\n          return JSON.parse(data);\n        } catch {\n          return { raw: data };\n        }\n      }\n    }\n    return null;\n  }\n\n  // ===========================================================================\n  // tRPC Middleware\n  // ===========================================================================\n\n  trpcMiddleware<TContext extends Record<string, unknown> = Record<string, unknown>>() {\n    const logger = this;\n\n    return async function loggerMiddleware(opts: {\n      path: string;\n      type: 'query' | 'mutation' | 'subscription';\n      input: unknown;\n      ctx: TContext & { logMetadata?: Record<string, unknown> };\n      next: () => Promise<{ ok: boolean; data?: unknown; error?: Error }>;\n    }) {\n      const start = Date.now();\n      const requestId = logger.generateRequestId();\n\n      // Initialize metadata on context if not present\n      if (!opts.ctx.logMetadata) {\n        opts.ctx.logMetadata = {};\n      }\n\n      try {\n        const result = await opts.next();\n\n        const entry: LogEntry = {\n          timestamp: new Date().toISOString(),\n          requestId,\n          type: 'trpc',\n          method: opts.type.toUpperCase(),\n          path: opts.path,\n          statusCode: result.ok ? 200 : 500,\n          duration: Date.now() - start,\n          request: {\n            body: opts.input,\n          },\n          response: {\n            body: result.data,\n            streaming: false,\n          },\n        };\n\n        // Attach metadata if present\n        if (opts.ctx.logMetadata && Object.keys(opts.ctx.logMetadata).length > 0) {\n          entry.metadata = opts.ctx.logMetadata;\n        }\n\n        if (result.error) {\n          entry.error = {\n            message: result.error.message,\n            stack: result.error.stack,\n          };\n        }\n\n        logger.write(entry);\n\n        return result;\n      } catch (error) {\n        const err = error as Error;\n\n        const entry: LogEntry = {\n          timestamp: new Date().toISOString(),\n          requestId,\n          type: 'trpc',\n          method: opts.type.toUpperCase(),\n          path: opts.path,\n          statusCode: 500,\n          duration: Date.now() - start,\n          request: {\n            body: opts.input,\n          },\n          error: {\n            message: err.message,\n            stack: err.stack,\n          },\n        };\n\n        // Attach metadata if present\n        if (opts.ctx.logMetadata && Object.keys(opts.ctx.logMetadata).length > 0) {\n          entry.metadata = opts.ctx.logMetadata;\n        }\n\n        logger.write(entry);\n\n        throw error;\n      }\n    };\n  }\n}\n\n// ============================================================================\n// Factory Function (Main Export)\n// ============================================================================\n\nexport function createLogger(filePath: string, options?: LoggerOptions) {\n  const logger = new UnifiedLogger(filePath, options);\n\n  return {\n    /** Express middleware - use with app.use() */\n    express: () => logger.expressMiddleware(),\n\n    /** tRPC middleware - use with t.procedure.use() */\n    trpc: <TContext extends Record<string, unknown> = Record<string, unknown>>() => \n      logger.trpcMiddleware<TContext>(),\n\n    /** Direct write access for custom logging */\n    write: (entry: Partial<LogEntry>) => logger.write({\n      timestamp: new Date().toISOString(),\n      requestId: `${Date.now().toString(36)}-${Math.random().toString(36).slice(2, 9)}`,\n      type: 'http',\n      method: 'CUSTOM',\n      path: '/',\n      duration: 0,\n      ...entry,\n    } as LogEntry),\n  };\n}\n\n// ============================================================================\n// Helper to attach metadata (for cleaner API)\n// ============================================================================\n\n/**\n * Attach metadata to the current request log entry (Express)\n */\nexport function attachMetadata(req: Request, metadata: Record<string, unknown>): void {\n  if (!req.logMetadata) {\n    req.logMetadata = {};\n  }\n  Object.assign(req.logMetadata, metadata);\n}\n\n/**\n * Attach metadata to the current request log entry (tRPC)\n */\nexport function attachTrpcMetadata<TContext extends { logMetadata?: Record<string, unknown> }>(\n  ctx: TContext,\n  metadata: Record<string, unknown>\n): void {\n  if (!ctx.logMetadata) {\n    ctx.logMetadata = {};\n  }\n  Object.assign(ctx.logMetadata, metadata);\n}\n\n// Default export for simpler imports\nexport default createLogger;\n"]}
441
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"logger.js","sourceRoot":"","sources":["../src/logger.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs';\nimport * as path from 'path';\nimport type { Request, Response, NextFunction } from 'express';\n\n// ============================================================================\n// Types\n// ============================================================================\n\ninterface LogEntry {\n  timestamp: string;\n  requestId: string;\n  type: 'http' | 'trpc';\n  method: string;\n  path: string;\n  statusCode?: number;\n  duration: number;\n  request?: {\n    body?: unknown;\n    query?: unknown;\n    headers?: Record<string, string>;\n  };\n  response?: {\n    body?: unknown;\n    streaming?: boolean;\n    chunks?: unknown[];\n    text?: string; // Aggregated text from text-delta chunks\n  };\n  error?: {\n    message: string;\n    stack?: string;\n  };\n  metadata?: Record<string, unknown>;\n}\n\ninterface LoggerOptions {\n  maxSizeBytes?: number;      // Default: 10MB\n  includeHeaders?: boolean;   // Default: false\n  redact?: string[];          // Fields to redact from logs\n  ignorePaths?: string[];     // Paths to ignore (supports wildcards like /health/*)\n  includePaths?: string[];    // Only log these paths (supports wildcards)\n}\n\n// Extend Express Request to include metadata\ndeclare global {\n  namespace Express {\n    interface Request {\n      logMetadata?: Record<string, unknown>;\n    }\n  }\n}\n\n// ============================================================================\n// Core Logger Class\n// ============================================================================\n\nclass UnifiedLogger {\n  private filePath: string;\n  private maxSizeBytes: number;\n  private includeHeaders: boolean;\n  private redactFields: Set<string>;\n  private ignorePaths: string[];\n  private includePaths: string[];\n\n  constructor(filePath: string, options: LoggerOptions = {}) {\n    this.filePath = path.resolve(filePath);\n    this.maxSizeBytes = options.maxSizeBytes ?? 10 * 1024 * 1024; // 10MB default\n    this.includeHeaders = options.includeHeaders ?? false;\n    this.redactFields = new Set(options.redact ?? ['password', 'token', 'authorization', 'cookie']);\n    this.ignorePaths = options.ignorePaths ?? [];\n    this.includePaths = options.includePaths ?? [];\n\n    // Ensure directory exists\n    const dir = path.dirname(this.filePath);\n    if (!fs.existsSync(dir)) {\n      fs.mkdirSync(dir, { recursive: true });\n    }\n  }\n\n  private generateRequestId(): string {\n    return `${Date.now().toString(36)}-${Math.random().toString(36).slice(2, 9)}`;\n  }\n\n  private matchPath(pattern: string, requestPath: string): boolean {\n    // Convert wildcard pattern to regex\n    // /api/* matches /api/anything\n    // /health matches exactly /health\n    const regexPattern = pattern\n      .replace(/[.+?^${}()|[\\]\\\\]/g, '\\\\$&') // Escape special regex chars except *\n      .replace(/\\*/g, '.*'); // Convert * to .*\n    const regex = new RegExp(`^${regexPattern}$`);\n    return regex.test(requestPath);\n  }\n\n  private shouldLog(requestPath: string): boolean {\n    // Extract path without query string\n    const pathOnly = requestPath.split('?')[0];\n\n    // If includePaths is set, only log matching paths\n    if (this.includePaths.length > 0) {\n      return this.includePaths.some((pattern) => this.matchPath(pattern, pathOnly));\n    }\n\n    // If ignorePaths is set, skip matching paths\n    if (this.ignorePaths.length > 0) {\n      return !this.ignorePaths.some((pattern) => this.matchPath(pattern, pathOnly));\n    }\n\n    return true;\n  }\n\n  private redact(obj: unknown): unknown {\n    if (obj === null || obj === undefined) return obj;\n    if (typeof obj !== 'object') return obj;\n\n    if (Array.isArray(obj)) {\n      return obj.map((item) => this.redact(item));\n    }\n\n    const result: Record<string, unknown> = {};\n    for (const [key, value] of Object.entries(obj as Record<string, unknown>)) {\n      if (this.redactFields.has(key.toLowerCase())) {\n        result[key] = '[REDACTED]';\n      } else if (typeof value === 'object') {\n        result[key] = this.redact(value);\n      } else {\n        result[key] = value;\n      }\n    }\n    return result;\n  }\n\n  private checkAndRotate(): void {\n    try {\n      if (!fs.existsSync(this.filePath)) return;\n\n      const stats = fs.statSync(this.filePath);\n      if (stats.size > this.maxSizeBytes) {\n        // Read file, keep last 25% of lines\n        const content = fs.readFileSync(this.filePath, 'utf-8');\n        const lines = content.trim().split('\\n');\n        const keepCount = Math.floor(lines.length * 0.25);\n        const newContent = lines.slice(-keepCount).join('\\n') + '\\n';\n        fs.writeFileSync(this.filePath, newContent);\n      }\n    } catch (err) {\n      console.error('Logger rotation error:', err);\n    }\n  }\n\n  write(entry: LogEntry): void {\n    try {\n      this.checkAndRotate();\n      const redactedEntry = this.redact(entry) as LogEntry;\n      const line = JSON.stringify(redactedEntry) + '\\n';\n      fs.appendFileSync(this.filePath, line);\n    } catch (err) {\n      console.error('Logger write error:', err);\n    }\n  }\n\n  // ===========================================================================\n  // Express Middleware\n  // ===========================================================================\n\n  expressMiddleware() {\n    return (req: Request, res: Response, next: NextFunction) => {\n      const requestPath = req.originalUrl || req.url;\n\n      // Check if we should log this path\n      if (!this.shouldLog(requestPath)) {\n        return next();\n      }\n\n      const start = Date.now();\n      const requestId = this.generateRequestId();\n\n      // Initialize metadata object on request\n      req.logMetadata = {};\n\n      // Detect SSE - check both request Accept header and response Content-Type\n      let isSSE = req.headers.accept === 'text/event-stream';\n      const chunks: unknown[] = [];\n      const textDeltas: string[] = []; // Collect text-delta content\n      \n      // Intercept setHeader to detect SSE by Content-Type\n      const originalSetHeader = res.setHeader.bind(res);\n      res.setHeader = ((name: string, value: string | number | readonly string[]): Response => {\n        if (name.toLowerCase() === 'content-type' && \n            typeof value === 'string' && \n            value.includes('text/event-stream')) {\n          isSSE = true;\n        }\n        return originalSetHeader(name, value);\n      }) as typeof res.setHeader;\n\n      // Capture request info\n      const requestInfo: LogEntry['request'] = {\n        body: req.body,\n        query: req.query,\n      };\n\n      if (this.includeHeaders) {\n        requestInfo.headers = req.headers as Record<string, string>;\n      }\n\n      // Intercept write/end for streaming detection\n      const originalWrite = res.write.bind(res);\n      const originalEnd = res.end.bind(res);\n      const originalJson = res.json.bind(res);\n      const originalSend = res.send.bind(res);\n      let responseBody: unknown;\n      let logged = false;\n\n      res.write = ((chunk: any, encodingOrCallback?: any, callback?: any): boolean => {\n        // If write is called, treat as streaming\n        if (chunk && isSSE) {\n          const chunkStr = chunk.toString();\n          const parsed = this.parseSSEChunk(chunkStr, textDeltas);\n          if (parsed) {\n            chunks.push(parsed);\n          }\n        }\n        return originalWrite(chunk, encodingOrCallback, callback);\n      }) as typeof res.write;\n\n      res.end = ((chunk?: any, encodingOrCallback?: any, callback?: any): Response => {\n        if (logged) return originalEnd(chunk, encodingOrCallback, callback);\n        logged = true;\n\n        if (isSSE) {\n          // SSE streaming path\n          if (chunk) {\n            const chunkStr = chunk.toString();\n            const parsed = this.parseSSEChunk(chunkStr, textDeltas);\n            if (parsed) {\n              chunks.push(parsed);\n            }\n          }\n\n          const entry: LogEntry = {\n            timestamp: new Date().toISOString(),\n            requestId,\n            type: 'http',\n            method: req.method,\n            path: requestPath,\n            statusCode: res.statusCode,\n            duration: Date.now() - start,\n            request: requestInfo,\n            response: {\n              streaming: true,\n              chunks,\n            },\n          };\n\n          // Add aggregated text if we collected text-deltas\n          if (textDeltas.length > 0) {\n            entry.response!.text = textDeltas.join('');\n          }\n\n          if (req.logMetadata && Object.keys(req.logMetadata).length > 0) {\n            entry.metadata = req.logMetadata;\n          }\n\n          this.write(entry);\n        } else {\n          // Regular response path\n          const entry: LogEntry = {\n            timestamp: new Date().toISOString(),\n            requestId,\n            type: 'http',\n            method: req.method,\n            path: requestPath,\n            statusCode: res.statusCode,\n            duration: Date.now() - start,\n            request: requestInfo,\n            response: {\n              body: responseBody,\n              streaming: false,\n            },\n          };\n\n          if (req.logMetadata && Object.keys(req.logMetadata).length > 0) {\n            entry.metadata = req.logMetadata;\n          }\n\n          this.write(entry);\n        }\n\n        return originalEnd(chunk, encodingOrCallback, callback);\n      }) as typeof res.end;\n\n      const logResponse = () => {\n        if (logged) return;\n        logged = true;\n\n        const entry: LogEntry = {\n          timestamp: new Date().toISOString(),\n          requestId,\n          type: 'http',\n          method: req.method,\n          path: requestPath,\n          statusCode: res.statusCode,\n          duration: Date.now() - start,\n          request: requestInfo,\n          response: {\n            body: responseBody,\n            streaming: false,\n          },\n        };\n\n        if (req.logMetadata && Object.keys(req.logMetadata).length > 0) {\n          entry.metadata = req.logMetadata;\n        }\n\n        this.write(entry);\n      };\n\n      res.json = (body: any) => {\n        responseBody = body;\n        logResponse();\n        return originalJson(body);\n      };\n\n      res.send = (body: any) => {\n        if (!logged) {\n          try {\n            responseBody = typeof body === 'string' ? JSON.parse(body) : body;\n          } catch {\n            responseBody = body;\n          }\n          logResponse();\n        }\n        return originalSend(body);\n      };\n\n      next();\n    };\n  }\n\n  private parseSSEChunk(raw: string, textDeltas: string[]): unknown {\n    const lines = raw.split('\\n');\n    const results: unknown[] = [];\n\n    for (const line of lines) {\n      if (line.startsWith('data: ')) {\n        const data = line.slice(6).trim();\n        if (data === '[DONE]') {\n          results.push({ type: 'done' });\n          continue;\n        }\n        try {\n          const parsed = JSON.parse(data);\n          \n          // Handle text-delta type - collect the delta text\n          if (parsed.type === 'text-delta' && typeof parsed.delta === 'string') {\n            textDeltas.push(parsed.delta);\n          }\n          \n          results.push(parsed);\n        } catch {\n          results.push({ raw: data });\n        }\n      }\n    }\n\n    // Return single result or array\n    if (results.length === 0) return null;\n    if (results.length === 1) return results[0];\n    return results;\n  }\n\n  // ===========================================================================\n  // tRPC Middleware\n  // ===========================================================================\n\n  trpcMiddleware<TContext extends Record<string, unknown> = Record<string, unknown>>() {\n    const logger = this;\n\n    return async function loggerMiddleware(opts: {\n      path: string;\n      type: 'query' | 'mutation' | 'subscription';\n      input: unknown;\n      ctx: TContext & { logMetadata?: Record<string, unknown> };\n      next: () => Promise<{ ok: boolean; data?: unknown; error?: Error }>;\n    }) {\n      // Check if we should log this path\n      if (!logger.shouldLog(opts.path)) {\n        return opts.next();\n      }\n\n      const start = Date.now();\n      const requestId = logger.generateRequestId();\n\n      // Initialize metadata on context if not present\n      if (!opts.ctx.logMetadata) {\n        opts.ctx.logMetadata = {};\n      }\n\n      try {\n        const result = await opts.next();\n\n        const entry: LogEntry = {\n          timestamp: new Date().toISOString(),\n          requestId,\n          type: 'trpc',\n          method: opts.type.toUpperCase(),\n          path: opts.path,\n          statusCode: result.ok ? 200 : 500,\n          duration: Date.now() - start,\n          request: {\n            body: opts.input,\n          },\n          response: {\n            body: result.data,\n            streaming: false,\n          },\n        };\n\n        // Attach metadata if present\n        if (opts.ctx.logMetadata && Object.keys(opts.ctx.logMetadata).length > 0) {\n          entry.metadata = opts.ctx.logMetadata;\n        }\n\n        if (result.error) {\n          entry.error = {\n            message: result.error.message,\n            stack: result.error.stack,\n          };\n        }\n\n        logger.write(entry);\n\n        return result;\n      } catch (error) {\n        const err = error as Error;\n\n        const entry: LogEntry = {\n          timestamp: new Date().toISOString(),\n          requestId,\n          type: 'trpc',\n          method: opts.type.toUpperCase(),\n          path: opts.path,\n          statusCode: 500,\n          duration: Date.now() - start,\n          request: {\n            body: opts.input,\n          },\n          error: {\n            message: err.message,\n            stack: err.stack,\n          },\n        };\n\n        // Attach metadata if present\n        if (opts.ctx.logMetadata && Object.keys(opts.ctx.logMetadata).length > 0) {\n          entry.metadata = opts.ctx.logMetadata;\n        }\n\n        logger.write(entry);\n\n        throw error;\n      }\n    };\n  }\n}\n\n// ============================================================================\n// Factory Function (Main Export)\n// ============================================================================\n\nexport function createLogger(filePath: string, options?: LoggerOptions) {\n  const logger = new UnifiedLogger(filePath, options);\n\n  return {\n    /** Express middleware - use with app.use() */\n    express: () => logger.expressMiddleware(),\n\n    /** tRPC middleware - use with t.procedure.use() */\n    trpc: <TContext extends Record<string, unknown> = Record<string, unknown>>() => \n      logger.trpcMiddleware<TContext>(),\n\n    /** Direct write access for custom logging */\n    write: (entry: Partial<LogEntry>) => logger.write({\n      timestamp: new Date().toISOString(),\n      requestId: `${Date.now().toString(36)}-${Math.random().toString(36).slice(2, 9)}`,\n      type: 'http',\n      method: 'CUSTOM',\n      path: '/',\n      duration: 0,\n      ...entry,\n    } as LogEntry),\n  };\n}\n\n// ============================================================================\n// Helper to attach metadata (for cleaner API)\n// ============================================================================\n\n/**\n * Attach metadata to the current request log entry (Express)\n */\nexport function attachMetadata(req: Request, metadata: Record<string, unknown>): void {\n  if (!req.logMetadata) {\n    req.logMetadata = {};\n  }\n  Object.assign(req.logMetadata, metadata);\n}\n\n/**\n * Attach metadata to the current request log entry (tRPC)\n */\nexport function attachTrpcMetadata<TContext extends { logMetadata?: Record<string, unknown> }>(\n  ctx: TContext,\n  metadata: Record<string, unknown>\n): void {\n  if (!ctx.logMetadata) {\n    ctx.logMetadata = {};\n  }\n  Object.assign(ctx.logMetadata, metadata);\n}\n\n// Default export for simpler imports\nexport default createLogger;\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "mohen",
3
- "version": "1.0.0",
3
+ "version": "1.1.0",
4
4
  "description": "Unified request/response logger for Express and tRPC with SSE support (墨痕 - ink trace)",
5
5
  "main": "dist/logger.js",
6
6
  "types": "dist/logger.d.ts",
package/src/logger.ts CHANGED
@@ -23,6 +23,7 @@ interface LogEntry {
23
23
  body?: unknown;
24
24
  streaming?: boolean;
25
25
  chunks?: unknown[];
26
+ text?: string; // Aggregated text from text-delta chunks
26
27
  };
27
28
  error?: {
28
29
  message: string;
@@ -35,6 +36,8 @@ interface LoggerOptions {
35
36
  maxSizeBytes?: number; // Default: 10MB
36
37
  includeHeaders?: boolean; // Default: false
37
38
  redact?: string[]; // Fields to redact from logs
39
+ ignorePaths?: string[]; // Paths to ignore (supports wildcards like /health/*)
40
+ includePaths?: string[]; // Only log these paths (supports wildcards)
38
41
  }
39
42
 
40
43
  // Extend Express Request to include metadata
@@ -55,12 +58,16 @@ class UnifiedLogger {
55
58
  private maxSizeBytes: number;
56
59
  private includeHeaders: boolean;
57
60
  private redactFields: Set<string>;
61
+ private ignorePaths: string[];
62
+ private includePaths: string[];
58
63
 
59
64
  constructor(filePath: string, options: LoggerOptions = {}) {
60
65
  this.filePath = path.resolve(filePath);
61
66
  this.maxSizeBytes = options.maxSizeBytes ?? 10 * 1024 * 1024; // 10MB default
62
67
  this.includeHeaders = options.includeHeaders ?? false;
63
68
  this.redactFields = new Set(options.redact ?? ['password', 'token', 'authorization', 'cookie']);
69
+ this.ignorePaths = options.ignorePaths ?? [];
70
+ this.includePaths = options.includePaths ?? [];
64
71
 
65
72
  // Ensure directory exists
66
73
  const dir = path.dirname(this.filePath);
@@ -73,6 +80,34 @@ class UnifiedLogger {
73
80
  return `${Date.now().toString(36)}-${Math.random().toString(36).slice(2, 9)}`;
74
81
  }
75
82
 
83
+ private matchPath(pattern: string, requestPath: string): boolean {
84
+ // Convert wildcard pattern to regex
85
+ // /api/* matches /api/anything
86
+ // /health matches exactly /health
87
+ const regexPattern = pattern
88
+ .replace(/[.+?^${}()|[\]\\]/g, '\\$&') // Escape special regex chars except *
89
+ .replace(/\*/g, '.*'); // Convert * to .*
90
+ const regex = new RegExp(`^${regexPattern}$`);
91
+ return regex.test(requestPath);
92
+ }
93
+
94
+ private shouldLog(requestPath: string): boolean {
95
+ // Extract path without query string
96
+ const pathOnly = requestPath.split('?')[0];
97
+
98
+ // If includePaths is set, only log matching paths
99
+ if (this.includePaths.length > 0) {
100
+ return this.includePaths.some((pattern) => this.matchPath(pattern, pathOnly));
101
+ }
102
+
103
+ // If ignorePaths is set, skip matching paths
104
+ if (this.ignorePaths.length > 0) {
105
+ return !this.ignorePaths.some((pattern) => this.matchPath(pattern, pathOnly));
106
+ }
107
+
108
+ return true;
109
+ }
110
+
76
111
  private redact(obj: unknown): unknown {
77
112
  if (obj === null || obj === undefined) return obj;
78
113
  if (typeof obj !== 'object') return obj;
@@ -129,6 +164,13 @@ class UnifiedLogger {
129
164
 
130
165
  expressMiddleware() {
131
166
  return (req: Request, res: Response, next: NextFunction) => {
167
+ const requestPath = req.originalUrl || req.url;
168
+
169
+ // Check if we should log this path
170
+ if (!this.shouldLog(requestPath)) {
171
+ return next();
172
+ }
173
+
132
174
  const start = Date.now();
133
175
  const requestId = this.generateRequestId();
134
176
 
@@ -138,6 +180,7 @@ class UnifiedLogger {
138
180
  // Detect SSE - check both request Accept header and response Content-Type
139
181
  let isSSE = req.headers.accept === 'text/event-stream';
140
182
  const chunks: unknown[] = [];
183
+ const textDeltas: string[] = []; // Collect text-delta content
141
184
 
142
185
  // Intercept setHeader to detect SSE by Content-Type
143
186
  const originalSetHeader = res.setHeader.bind(res);
@@ -172,7 +215,7 @@ class UnifiedLogger {
172
215
  // If write is called, treat as streaming
173
216
  if (chunk && isSSE) {
174
217
  const chunkStr = chunk.toString();
175
- const parsed = this.parseSSEChunk(chunkStr);
218
+ const parsed = this.parseSSEChunk(chunkStr, textDeltas);
176
219
  if (parsed) {
177
220
  chunks.push(parsed);
178
221
  }
@@ -188,7 +231,7 @@ class UnifiedLogger {
188
231
  // SSE streaming path
189
232
  if (chunk) {
190
233
  const chunkStr = chunk.toString();
191
- const parsed = this.parseSSEChunk(chunkStr);
234
+ const parsed = this.parseSSEChunk(chunkStr, textDeltas);
192
235
  if (parsed) {
193
236
  chunks.push(parsed);
194
237
  }
@@ -199,7 +242,7 @@ class UnifiedLogger {
199
242
  requestId,
200
243
  type: 'http',
201
244
  method: req.method,
202
- path: req.originalUrl || req.url,
245
+ path: requestPath,
203
246
  statusCode: res.statusCode,
204
247
  duration: Date.now() - start,
205
248
  request: requestInfo,
@@ -209,6 +252,11 @@ class UnifiedLogger {
209
252
  },
210
253
  };
211
254
 
255
+ // Add aggregated text if we collected text-deltas
256
+ if (textDeltas.length > 0) {
257
+ entry.response!.text = textDeltas.join('');
258
+ }
259
+
212
260
  if (req.logMetadata && Object.keys(req.logMetadata).length > 0) {
213
261
  entry.metadata = req.logMetadata;
214
262
  }
@@ -221,7 +269,7 @@ class UnifiedLogger {
221
269
  requestId,
222
270
  type: 'http',
223
271
  method: req.method,
224
- path: req.originalUrl || req.url,
272
+ path: requestPath,
225
273
  statusCode: res.statusCode,
226
274
  duration: Date.now() - start,
227
275
  request: requestInfo,
@@ -250,7 +298,7 @@ class UnifiedLogger {
250
298
  requestId,
251
299
  type: 'http',
252
300
  method: req.method,
253
- path: req.originalUrl || req.url,
301
+ path: requestPath,
254
302
  statusCode: res.statusCode,
255
303
  duration: Date.now() - start,
256
304
  request: requestInfo,
@@ -289,20 +337,36 @@ class UnifiedLogger {
289
337
  };
290
338
  }
291
339
 
292
- private parseSSEChunk(raw: string): unknown {
340
+ private parseSSEChunk(raw: string, textDeltas: string[]): unknown {
293
341
  const lines = raw.split('\n');
342
+ const results: unknown[] = [];
343
+
294
344
  for (const line of lines) {
295
345
  if (line.startsWith('data: ')) {
296
346
  const data = line.slice(6).trim();
297
- if (data === '[DONE]') return { done: true };
347
+ if (data === '[DONE]') {
348
+ results.push({ type: 'done' });
349
+ continue;
350
+ }
298
351
  try {
299
- return JSON.parse(data);
352
+ const parsed = JSON.parse(data);
353
+
354
+ // Handle text-delta type - collect the delta text
355
+ if (parsed.type === 'text-delta' && typeof parsed.delta === 'string') {
356
+ textDeltas.push(parsed.delta);
357
+ }
358
+
359
+ results.push(parsed);
300
360
  } catch {
301
- return { raw: data };
361
+ results.push({ raw: data });
302
362
  }
303
363
  }
304
364
  }
305
- return null;
365
+
366
+ // Return single result or array
367
+ if (results.length === 0) return null;
368
+ if (results.length === 1) return results[0];
369
+ return results;
306
370
  }
307
371
 
308
372
  // ===========================================================================
@@ -319,6 +383,11 @@ class UnifiedLogger {
319
383
  ctx: TContext & { logMetadata?: Record<string, unknown> };
320
384
  next: () => Promise<{ ok: boolean; data?: unknown; error?: Error }>;
321
385
  }) {
386
+ // Check if we should log this path
387
+ if (!logger.shouldLog(opts.path)) {
388
+ return opts.next();
389
+ }
390
+
322
391
  const start = Date.now();
323
392
  const requestId = logger.generateRequestId();
324
393
 
@@ -196,7 +196,7 @@ describe('mohen logger', () => {
196
196
  { count: 1 },
197
197
  { count: 2 },
198
198
  { count: 3 },
199
- { done: true },
199
+ { type: 'done' },
200
200
  ],
201
201
  },
202
202
  });
@@ -468,6 +468,99 @@ describe('mohen logger', () => {
468
468
  });
469
469
  });
470
470
 
471
+ describe('Path Filtering', () => {
472
+ it('should ignore paths matching ignorePaths patterns', async () => {
473
+ vi.useRealTimers();
474
+ clearLogFile();
475
+
476
+ const filteredLogger = createLogger(TEST_LOG_FILE, {
477
+ ignorePaths: ['/health', '/health/*', '/metrics'],
478
+ });
479
+
480
+ const filteredApp = express();
481
+ filteredApp.use(express.json());
482
+ filteredApp.use(filteredLogger.express());
483
+ filteredApp.get('/health', (req, res) => res.json({ status: 'ok' }));
484
+ filteredApp.get('/health/live', (req, res) => res.json({ live: true }));
485
+ filteredApp.get('/metrics', (req, res) => res.json({ cpu: 50 }));
486
+ filteredApp.get('/api/users', (req, res) => res.json({ users: [] }));
487
+
488
+ await request(filteredApp).get('/health').expect(200);
489
+ await request(filteredApp).get('/health/live').expect(200);
490
+ await request(filteredApp).get('/metrics').expect(200);
491
+ await request(filteredApp).get('/api/users').expect(200);
492
+
493
+ const entries = readLogEntries();
494
+ expect(entries).toHaveLength(1);
495
+ expect(entries[0].path).toBe('/api/users');
496
+ });
497
+
498
+ it('should only log paths matching includePaths patterns', async () => {
499
+ vi.useRealTimers();
500
+ clearLogFile();
501
+
502
+ const filteredLogger = createLogger(TEST_LOG_FILE, {
503
+ includePaths: ['/api/*'],
504
+ });
505
+
506
+ const filteredApp = express();
507
+ filteredApp.use(express.json());
508
+ filteredApp.use(filteredLogger.express());
509
+ filteredApp.get('/health', (req, res) => res.json({ status: 'ok' }));
510
+ filteredApp.get('/api/users', (req, res) => res.json({ users: [] }));
511
+ filteredApp.get('/api/orders', (req, res) => res.json({ orders: [] }));
512
+
513
+ await request(filteredApp).get('/health').expect(200);
514
+ await request(filteredApp).get('/api/users').expect(200);
515
+ await request(filteredApp).get('/api/orders').expect(200);
516
+
517
+ const entries = readLogEntries();
518
+ expect(entries).toHaveLength(2);
519
+ expect(entries[0].path).toBe('/api/users');
520
+ expect(entries[1].path).toBe('/api/orders');
521
+ });
522
+ });
523
+
524
+ describe('SSE Text Delta Parsing', () => {
525
+ it('should aggregate text-delta chunks into text field', async () => {
526
+ app.get('/api/stream', (req, res) => {
527
+ res.setHeader('Content-Type', 'text/event-stream');
528
+ res.setHeader('Cache-Control', 'no-cache');
529
+ res.setHeader('Connection', 'keep-alive');
530
+
531
+ res.write('data: {"type":"start"}\n\n');
532
+ res.write('data: {"type":"text-delta","id":"0","delta":"Hello"}\n\n');
533
+ res.write('data: {"type":"text-delta","id":"0","delta":" world"}\n\n');
534
+ res.write('data: {"type":"text-delta","id":"0","delta":"!"}\n\n');
535
+ res.write('data: {"type":"finish","finishReason":"stop"}\n\n');
536
+ res.write('data: [DONE]\n\n');
537
+ res.end();
538
+ });
539
+
540
+ await request(app)
541
+ .get('/api/stream')
542
+ .set('Accept', 'text/event-stream')
543
+ .expect(200);
544
+
545
+ const entries = readLogEntries();
546
+ expect(entries).toHaveLength(1);
547
+ expect(entries[0]).toMatchObject({
548
+ timestamp: '2024-01-15T10:30:00.000Z',
549
+ type: 'http',
550
+ method: 'GET',
551
+ path: '/api/stream',
552
+ statusCode: 200,
553
+ response: {
554
+ streaming: true,
555
+ text: 'Hello world!',
556
+ },
557
+ });
558
+ expect(entries[0].response.chunks).toContainEqual({ type: 'start' });
559
+ expect(entries[0].response.chunks).toContainEqual({ type: 'text-delta', id: '0', delta: 'Hello' });
560
+ expect(entries[0].response.chunks).toContainEqual({ type: 'done' });
561
+ });
562
+ });
563
+
471
564
  describe('File Size Management', () => {
472
565
  it('should truncate log file when exceeding max size', async () => {
473
566
  vi.useRealTimers(); // Use real timers for this test