mohen 1.1.0 → 1.2.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/dist/logger.js CHANGED
@@ -132,6 +132,48 @@ class UnifiedLogger {
132
132
  console.error('Logger write error:', err);
133
133
  }
134
134
  }
135
+ /**
136
+ * Decode chunk to string, handling Uint8Array/Buffer from TextEncoderStream
137
+ */
138
+ decodeChunk(chunk) {
139
+ if (chunk === null || chunk === undefined) {
140
+ return '';
141
+ }
142
+ // Handle Uint8Array (from TextEncoderStream in AI SDK)
143
+ if (chunk instanceof Uint8Array) {
144
+ return Buffer.from(chunk).toString('utf-8');
145
+ }
146
+ // Handle Buffer
147
+ if (Buffer.isBuffer(chunk)) {
148
+ return chunk.toString('utf-8');
149
+ }
150
+ // Handle string
151
+ if (typeof chunk === 'string') {
152
+ return chunk;
153
+ }
154
+ // Fallback: try toString but check if it looks like byte array
155
+ const str = chunk.toString();
156
+ // Detect if toString produced a comma-separated byte list like "100,97,116,97"
157
+ // This happens when Uint8Array.toString() is called without proper decoding
158
+ if (/^\d+(,\d+)*$/.test(str) && str.includes(',')) {
159
+ try {
160
+ const bytes = new Uint8Array(str.split(',').map(Number));
161
+ return Buffer.from(bytes).toString('utf-8');
162
+ }
163
+ catch {
164
+ return str;
165
+ }
166
+ }
167
+ return str;
168
+ }
169
+ /**
170
+ * Check if content looks like SSE data
171
+ */
172
+ looksLikeSSE(content) {
173
+ return content.trimStart().startsWith('data:') ||
174
+ content.includes('\ndata:') ||
175
+ content.trimStart().startsWith('event:');
176
+ }
135
177
  // ===========================================================================
136
178
  // Express Middleware
137
179
  // ===========================================================================
@@ -146,10 +188,39 @@ class UnifiedLogger {
146
188
  const requestId = this.generateRequestId();
147
189
  // Initialize metadata object on request
148
190
  req.logMetadata = {};
149
- // Detect SSE - check both request Accept header and response Content-Type
191
+ // Detect SSE - check request Accept header initially
150
192
  let isSSE = req.headers.accept === 'text/event-stream';
151
193
  const chunks = [];
152
194
  const textDeltas = []; // Collect text-delta content
195
+ // Helper to check Content-Type header for SSE
196
+ const checkContentTypeForSSE = (headers) => {
197
+ if (!headers)
198
+ return;
199
+ // headers can be an object or array of [key, value] pairs
200
+ if (Array.isArray(headers)) {
201
+ for (let i = 0; i < headers.length; i += 2) {
202
+ const key = headers[i];
203
+ const value = headers[i + 1];
204
+ if (typeof key === 'string' &&
205
+ key.toLowerCase() === 'content-type' &&
206
+ typeof value === 'string' &&
207
+ value.includes('text/event-stream')) {
208
+ isSSE = true;
209
+ return;
210
+ }
211
+ }
212
+ }
213
+ else if (typeof headers === 'object') {
214
+ for (const [key, value] of Object.entries(headers)) {
215
+ if (key.toLowerCase() === 'content-type' &&
216
+ typeof value === 'string' &&
217
+ value.includes('text/event-stream')) {
218
+ isSSE = true;
219
+ return;
220
+ }
221
+ }
222
+ }
223
+ };
153
224
  // Intercept setHeader to detect SSE by Content-Type
154
225
  const originalSetHeader = res.setHeader.bind(res);
155
226
  res.setHeader = ((name, value) => {
@@ -160,6 +231,20 @@ class UnifiedLogger {
160
231
  }
161
232
  return originalSetHeader(name, value);
162
233
  });
234
+ // Intercept writeHead to detect SSE (used by AI SDK's pipeUIMessageStreamToResponse)
235
+ const originalWriteHead = res.writeHead.bind(res);
236
+ res.writeHead = ((statusCode, statusMessageOrHeaders, maybeHeaders) => {
237
+ // writeHead can be called as:
238
+ // writeHead(statusCode)
239
+ // writeHead(statusCode, headers)
240
+ // writeHead(statusCode, statusMessage, headers)
241
+ let headers = maybeHeaders;
242
+ if (!headers && typeof statusMessageOrHeaders === 'object') {
243
+ headers = statusMessageOrHeaders;
244
+ }
245
+ checkContentTypeForSSE(headers);
246
+ return originalWriteHead(statusCode, statusMessageOrHeaders, maybeHeaders);
247
+ });
163
248
  // Capture request info
164
249
  const requestInfo = {
165
250
  body: req.body,
@@ -176,12 +261,18 @@ class UnifiedLogger {
176
261
  let responseBody;
177
262
  let logged = false;
178
263
  res.write = ((chunk, encodingOrCallback, callback) => {
179
- // If write is called, treat as streaming
180
- if (chunk && isSSE) {
181
- const chunkStr = chunk.toString();
182
- const parsed = this.parseSSEChunk(chunkStr, textDeltas);
183
- if (parsed) {
184
- chunks.push(parsed);
264
+ if (chunk) {
265
+ // Properly decode the chunk (handles Uint8Array from TextEncoderStream)
266
+ const chunkStr = this.decodeChunk(chunk);
267
+ // Auto-detect SSE from content if not already detected
268
+ if (!isSSE && this.looksLikeSSE(chunkStr)) {
269
+ isSSE = true;
270
+ }
271
+ if (isSSE) {
272
+ const parsed = this.parseSSEChunk(chunkStr, textDeltas);
273
+ if (parsed) {
274
+ chunks.push(parsed);
275
+ }
185
276
  }
186
277
  }
187
278
  return originalWrite(chunk, encodingOrCallback, callback);
@@ -193,7 +284,7 @@ class UnifiedLogger {
193
284
  if (isSSE) {
194
285
  // SSE streaming path
195
286
  if (chunk) {
196
- const chunkStr = chunk.toString();
287
+ const chunkStr = this.decodeChunk(chunk);
197
288
  const parsed = this.parseSSEChunk(chunkStr, textDeltas);
198
289
  if (parsed) {
199
290
  chunks.push(parsed);
@@ -438,4 +529,4 @@ function attachTrpcMetadata(ctx, metadata) {
438
529
  }
439
530
  // Default export for simpler imports
440
531
  exports.default = createLogger;
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"]}
532
+ //# 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   * Decode chunk to string, handling Uint8Array/Buffer from TextEncoderStream\n   */\n  private decodeChunk(chunk: any): string {\n    if (chunk === null || chunk === undefined) {\n      return '';\n    }\n    \n    // Handle Uint8Array (from TextEncoderStream in AI SDK)\n    if (chunk instanceof Uint8Array) {\n      return Buffer.from(chunk).toString('utf-8');\n    }\n    \n    // Handle Buffer\n    if (Buffer.isBuffer(chunk)) {\n      return chunk.toString('utf-8');\n    }\n    \n    // Handle string\n    if (typeof chunk === 'string') {\n      return chunk;\n    }\n    \n    // Fallback: try toString but check if it looks like byte array\n    const str = chunk.toString();\n    \n    // Detect if toString produced a comma-separated byte list like \"100,97,116,97\"\n    // This happens when Uint8Array.toString() is called without proper decoding\n    if (/^\\d+(,\\d+)*$/.test(str) && str.includes(',')) {\n      try {\n        const bytes = new Uint8Array(str.split(',').map(Number));\n        return Buffer.from(bytes).toString('utf-8');\n      } catch {\n        return str;\n      }\n    }\n    \n    return str;\n  }\n\n  /**\n   * Check if content looks like SSE data\n   */\n  private looksLikeSSE(content: string): boolean {\n    return content.trimStart().startsWith('data:') || \n           content.includes('\\ndata:') ||\n           content.trimStart().startsWith('event:');\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 request Accept header initially\n      let isSSE = req.headers.accept === 'text/event-stream';\n      const chunks: unknown[] = [];\n      const textDeltas: string[] = []; // Collect text-delta content\n\n      // Helper to check Content-Type header for SSE\n      const checkContentTypeForSSE = (headers: any): void => {\n        if (!headers) return;\n        \n        // headers can be an object or array of [key, value] pairs\n        if (Array.isArray(headers)) {\n          for (let i = 0; i < headers.length; i += 2) {\n            const key = headers[i];\n            const value = headers[i + 1];\n            if (typeof key === 'string' && \n                key.toLowerCase() === 'content-type' && \n                typeof value === 'string' && \n                value.includes('text/event-stream')) {\n              isSSE = true;\n              return;\n            }\n          }\n        } else if (typeof headers === 'object') {\n          for (const [key, value] of Object.entries(headers)) {\n            if (key.toLowerCase() === 'content-type' && \n                typeof value === 'string' && \n                value.includes('text/event-stream')) {\n              isSSE = true;\n              return;\n            }\n          }\n        }\n      };\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      // Intercept writeHead to detect SSE (used by AI SDK's pipeUIMessageStreamToResponse)\n      const originalWriteHead = res.writeHead.bind(res);\n      res.writeHead = ((\n        statusCode: number,\n        statusMessageOrHeaders?: string | any,\n        maybeHeaders?: any\n      ): Response => {\n        // writeHead can be called as:\n        // writeHead(statusCode)\n        // writeHead(statusCode, headers)\n        // writeHead(statusCode, statusMessage, headers)\n        let headers = maybeHeaders;\n        if (!headers && typeof statusMessageOrHeaders === 'object') {\n          headers = statusMessageOrHeaders;\n        }\n        \n        checkContentTypeForSSE(headers);\n        \n        return originalWriteHead(statusCode, statusMessageOrHeaders as any, maybeHeaders);\n      }) as typeof res.writeHead;\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 (chunk) {\n          // Properly decode the chunk (handles Uint8Array from TextEncoderStream)\n          const chunkStr = this.decodeChunk(chunk);\n          \n          // Auto-detect SSE from content if not already detected\n          if (!isSSE && this.looksLikeSSE(chunkStr)) {\n            isSSE = true;\n          }\n          \n          if (isSSE) {\n            const parsed = this.parseSSEChunk(chunkStr, textDeltas);\n            if (parsed) {\n              chunks.push(parsed);\n            }\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 = this.decodeChunk(chunk);\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.1.0",
3
+ "version": "1.2.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
@@ -158,6 +158,55 @@ class UnifiedLogger {
158
158
  }
159
159
  }
160
160
 
161
+ /**
162
+ * Decode chunk to string, handling Uint8Array/Buffer from TextEncoderStream
163
+ */
164
+ private decodeChunk(chunk: any): string {
165
+ if (chunk === null || chunk === undefined) {
166
+ return '';
167
+ }
168
+
169
+ // Handle Uint8Array (from TextEncoderStream in AI SDK)
170
+ if (chunk instanceof Uint8Array) {
171
+ return Buffer.from(chunk).toString('utf-8');
172
+ }
173
+
174
+ // Handle Buffer
175
+ if (Buffer.isBuffer(chunk)) {
176
+ return chunk.toString('utf-8');
177
+ }
178
+
179
+ // Handle string
180
+ if (typeof chunk === 'string') {
181
+ return chunk;
182
+ }
183
+
184
+ // Fallback: try toString but check if it looks like byte array
185
+ const str = chunk.toString();
186
+
187
+ // Detect if toString produced a comma-separated byte list like "100,97,116,97"
188
+ // This happens when Uint8Array.toString() is called without proper decoding
189
+ if (/^\d+(,\d+)*$/.test(str) && str.includes(',')) {
190
+ try {
191
+ const bytes = new Uint8Array(str.split(',').map(Number));
192
+ return Buffer.from(bytes).toString('utf-8');
193
+ } catch {
194
+ return str;
195
+ }
196
+ }
197
+
198
+ return str;
199
+ }
200
+
201
+ /**
202
+ * Check if content looks like SSE data
203
+ */
204
+ private looksLikeSSE(content: string): boolean {
205
+ return content.trimStart().startsWith('data:') ||
206
+ content.includes('\ndata:') ||
207
+ content.trimStart().startsWith('event:');
208
+ }
209
+
161
210
  // ===========================================================================
162
211
  // Express Middleware
163
212
  // ===========================================================================
@@ -177,10 +226,39 @@ class UnifiedLogger {
177
226
  // Initialize metadata object on request
178
227
  req.logMetadata = {};
179
228
 
180
- // Detect SSE - check both request Accept header and response Content-Type
229
+ // Detect SSE - check request Accept header initially
181
230
  let isSSE = req.headers.accept === 'text/event-stream';
182
231
  const chunks: unknown[] = [];
183
232
  const textDeltas: string[] = []; // Collect text-delta content
233
+
234
+ // Helper to check Content-Type header for SSE
235
+ const checkContentTypeForSSE = (headers: any): void => {
236
+ if (!headers) return;
237
+
238
+ // headers can be an object or array of [key, value] pairs
239
+ if (Array.isArray(headers)) {
240
+ for (let i = 0; i < headers.length; i += 2) {
241
+ const key = headers[i];
242
+ const value = headers[i + 1];
243
+ if (typeof key === 'string' &&
244
+ key.toLowerCase() === 'content-type' &&
245
+ typeof value === 'string' &&
246
+ value.includes('text/event-stream')) {
247
+ isSSE = true;
248
+ return;
249
+ }
250
+ }
251
+ } else if (typeof headers === 'object') {
252
+ for (const [key, value] of Object.entries(headers)) {
253
+ if (key.toLowerCase() === 'content-type' &&
254
+ typeof value === 'string' &&
255
+ value.includes('text/event-stream')) {
256
+ isSSE = true;
257
+ return;
258
+ }
259
+ }
260
+ }
261
+ };
184
262
 
185
263
  // Intercept setHeader to detect SSE by Content-Type
186
264
  const originalSetHeader = res.setHeader.bind(res);
@@ -193,6 +271,27 @@ class UnifiedLogger {
193
271
  return originalSetHeader(name, value);
194
272
  }) as typeof res.setHeader;
195
273
 
274
+ // Intercept writeHead to detect SSE (used by AI SDK's pipeUIMessageStreamToResponse)
275
+ const originalWriteHead = res.writeHead.bind(res);
276
+ res.writeHead = ((
277
+ statusCode: number,
278
+ statusMessageOrHeaders?: string | any,
279
+ maybeHeaders?: any
280
+ ): Response => {
281
+ // writeHead can be called as:
282
+ // writeHead(statusCode)
283
+ // writeHead(statusCode, headers)
284
+ // writeHead(statusCode, statusMessage, headers)
285
+ let headers = maybeHeaders;
286
+ if (!headers && typeof statusMessageOrHeaders === 'object') {
287
+ headers = statusMessageOrHeaders;
288
+ }
289
+
290
+ checkContentTypeForSSE(headers);
291
+
292
+ return originalWriteHead(statusCode, statusMessageOrHeaders as any, maybeHeaders);
293
+ }) as typeof res.writeHead;
294
+
196
295
  // Capture request info
197
296
  const requestInfo: LogEntry['request'] = {
198
297
  body: req.body,
@@ -212,12 +311,20 @@ class UnifiedLogger {
212
311
  let logged = false;
213
312
 
214
313
  res.write = ((chunk: any, encodingOrCallback?: any, callback?: any): boolean => {
215
- // If write is called, treat as streaming
216
- if (chunk && isSSE) {
217
- const chunkStr = chunk.toString();
218
- const parsed = this.parseSSEChunk(chunkStr, textDeltas);
219
- if (parsed) {
220
- chunks.push(parsed);
314
+ if (chunk) {
315
+ // Properly decode the chunk (handles Uint8Array from TextEncoderStream)
316
+ const chunkStr = this.decodeChunk(chunk);
317
+
318
+ // Auto-detect SSE from content if not already detected
319
+ if (!isSSE && this.looksLikeSSE(chunkStr)) {
320
+ isSSE = true;
321
+ }
322
+
323
+ if (isSSE) {
324
+ const parsed = this.parseSSEChunk(chunkStr, textDeltas);
325
+ if (parsed) {
326
+ chunks.push(parsed);
327
+ }
221
328
  }
222
329
  }
223
330
  return originalWrite(chunk, encodingOrCallback, callback);
@@ -230,7 +337,7 @@ class UnifiedLogger {
230
337
  if (isSSE) {
231
338
  // SSE streaming path
232
339
  if (chunk) {
233
- const chunkStr = chunk.toString();
340
+ const chunkStr = this.decodeChunk(chunk);
234
341
  const parsed = this.parseSSEChunk(chunkStr, textDeltas);
235
342
  if (parsed) {
236
343
  chunks.push(parsed);
@@ -521,6 +521,118 @@ describe('mohen logger', () => {
521
521
  });
522
522
  });
523
523
 
524
+ describe('AI SDK Style Streaming', () => {
525
+ it('should detect SSE via writeHead (AI SDK pipeUIMessageStreamToResponse)', async () => {
526
+ app.get('/api/ai-stream', (req, res) => {
527
+ // AI SDK uses writeHead instead of setHeader
528
+ res.writeHead(200, {
529
+ 'Content-Type': 'text/event-stream',
530
+ 'Cache-Control': 'no-cache',
531
+ 'Connection': 'keep-alive',
532
+ });
533
+
534
+ res.write('data: {"type":"start"}\n\n');
535
+ res.write('data: {"type":"text-delta","delta":"Hello"}\n\n');
536
+ res.write('data: {"type":"finish"}\n\n');
537
+ res.end();
538
+ });
539
+
540
+ await request(app)
541
+ .get('/api/ai-stream')
542
+ .expect(200);
543
+
544
+ const entries = readLogEntries();
545
+ expect(entries).toHaveLength(1);
546
+ expect(entries[0]).toMatchObject({
547
+ type: 'http',
548
+ method: 'GET',
549
+ path: '/api/ai-stream',
550
+ response: {
551
+ streaming: true,
552
+ text: 'Hello',
553
+ },
554
+ });
555
+ expect(entries[0].response.chunks).toContainEqual({ type: 'start' });
556
+ });
557
+
558
+ it('should handle Uint8Array chunks from TextEncoderStream', async () => {
559
+ app.get('/api/binary-stream', (req, res) => {
560
+ res.writeHead(200, {
561
+ 'Content-Type': 'text/event-stream',
562
+ });
563
+
564
+ // Simulate TextEncoderStream output (Uint8Array)
565
+ const encoder = new TextEncoder();
566
+ res.write(encoder.encode('data: {"type":"start"}\n\n'));
567
+ res.write(encoder.encode('data: {"type":"text-delta","delta":"Binary"}\n\n'));
568
+ res.write(encoder.encode('data: {"type":"text-delta","delta":" works"}\n\n'));
569
+ res.write(encoder.encode('data: [DONE]\n\n'));
570
+ res.end();
571
+ });
572
+
573
+ await request(app)
574
+ .get('/api/binary-stream')
575
+ .expect(200);
576
+
577
+ const entries = readLogEntries();
578
+ expect(entries).toHaveLength(1);
579
+ expect(entries[0]).toMatchObject({
580
+ type: 'http',
581
+ path: '/api/binary-stream',
582
+ response: {
583
+ streaming: true,
584
+ text: 'Binary works',
585
+ },
586
+ });
587
+ });
588
+
589
+ it('should auto-detect SSE from content when headers not set', async () => {
590
+ app.get('/api/auto-detect', (req, res) => {
591
+ // No SSE headers set, but content is SSE format
592
+ res.write('data: {"type":"start"}\n\n');
593
+ res.write('data: {"type":"text-delta","delta":"Auto"}\n\n');
594
+ res.write('data: {"type":"text-delta","delta":" detected"}\n\n');
595
+ res.end();
596
+ });
597
+
598
+ await request(app)
599
+ .get('/api/auto-detect')
600
+ .expect(200);
601
+
602
+ const entries = readLogEntries();
603
+ expect(entries).toHaveLength(1);
604
+ expect(entries[0]).toMatchObject({
605
+ type: 'http',
606
+ path: '/api/auto-detect',
607
+ response: {
608
+ streaming: true,
609
+ text: 'Auto detected',
610
+ },
611
+ });
612
+ });
613
+
614
+ it('should handle writeHead with statusMessage parameter', async () => {
615
+ app.get('/api/writehead-msg', (req, res) => {
616
+ // writeHead(statusCode, statusMessage, headers)
617
+ res.writeHead(200, 'OK', {
618
+ 'Content-Type': 'text/event-stream',
619
+ });
620
+
621
+ res.write('data: {"type":"text-delta","delta":"Test"}\n\n');
622
+ res.end();
623
+ });
624
+
625
+ await request(app)
626
+ .get('/api/writehead-msg')
627
+ .expect(200);
628
+
629
+ const entries = readLogEntries();
630
+ expect(entries).toHaveLength(1);
631
+ expect(entries[0].response.streaming).toBe(true);
632
+ expect(entries[0].response.text).toBe('Test');
633
+ });
634
+ });
635
+
524
636
  describe('SSE Text Delta Parsing', () => {
525
637
  it('should aggregate text-delta chunks into text field', async () => {
526
638
  app.get('/api/stream', (req, res) => {