tim-logger 0.0.37 → 0.0.38
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/object.d.ts.map +1 -1
- package/dist/server.cjs.js +3 -3
- package/dist/server.cjs.js.map +1 -1
- package/dist/server.d.ts.map +1 -1
- package/dist/server.es.js +33 -33
- package/dist/server.es.js.map +1 -1
- package/package.json +1 -1
package/dist/object.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"object.d.ts","sourceRoot":"","sources":["../src/object.ts"],"names":[],"mappings":"AAAA,wBAAgB,WAAW,CAAC,CAAC,EAAE,KAAK,EAAE,CAAC,GAAG,CAAC,
|
|
1
|
+
{"version":3,"file":"object.d.ts","sourceRoot":"","sources":["../src/object.ts"],"names":[],"mappings":"AAAA,wBAAgB,WAAW,CAAC,CAAC,EAAE,KAAK,EAAE,CAAC,GAAG,CAAC,CAwB1C"}
|
package/dist/server.cjs.js
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
"use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const d=require("kleur"),D=require("node:fs/promises"),O=require("node:path"),k=require("node:util"),l=require("winston"),_=require("winston-transport"),q=require("./console-intercept-CL_BfKQo.cjs");function
|
|
2
|
-
`,"utf8")})().catch(n=>{const s=n instanceof Error?n.message:y.format("%o",n);console.warn("[tim-logger] Failed to write log file",{logPath:o,error:s})}).finally(r)}}const A=l.format.combine(l.format.timestamp({format:()=>new Date().toISOString()}),l.format.printf(t=>{const e=t.logEvent??{},r=t.timestamp??new Date().toISOString(),o=String(e.source??"LOG"),a=String(e.message??t.message??""),n=e.data??{},s=n&&typeof n=="object"&&Object.keys(n).length?` | ${JSON.stringify(n)}`:"";return`[${r}] ${o.padStart(12)}: ${a}${s}`})),I=["console"],G=(t,e,r,o)=>typeof t=="string"?{message:t,level:e,source:r,data:o}:{...t,level:t.level??e,source:t.source??r,data:t.data??o},N=t=>{if(!t)return"";try{return JSON.stringify(t)}catch{return"[unserializable data]"}},U=l.format.combine(l.format.timestamp(),l.format.errors({stack:!0}),l.format.printf(t=>{const{timestamp:e,level:r="info",message:o="",stack:a,logEvent:n}=t,s=n??{},m=s.source??"LOG",p=s.verb??r?.toUpperCase?.()??"LOG",w=[s.machineId,s.sessionId].filter(Boolean),
|
|
3
|
-
${a}`:f}))
|
|
1
|
+
"use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const d=require("kleur"),D=require("node:fs/promises"),O=require("node:path"),k=require("node:util"),l=require("winston"),_=require("winston-transport"),q=require("./console-intercept-CL_BfKQo.cjs");function S(t){const e=Object.create(null,{[Symbol.toStringTag]:{value:"Module"}});if(t){for(const r in t)if(r!=="default"){const o=Object.getOwnPropertyDescriptor(t,r);Object.defineProperty(e,r,o.get?o:{enumerable:!0,get:()=>t[r]})}}return e.default=t,Object.freeze(e)}const h=S(D),j=S(O),y=S(k);function L(t){if(t===void 0)return t;if(Array.isArray(t))return t.filter(r=>r!==void 0).map(r=>L(r));if(t&&typeof t=="object"){const e={};for(const[r,o]of Object.entries(t))o!==void 0&&(e[r]=L(o));return e}return t}const T=Symbol.for("message"),x=t=>{if(!t||typeof t!="object")return;const e=t.logPath;return typeof e=="string"&&e.length?e:void 0};class R extends _{defaultLogPath;ensuredDirs=new Set;lineFormat;constructor({defaultLogPath:e,format:r}){super(),this.defaultLogPath=e,this.lineFormat=r}log(e,r){setImmediate(()=>this.emit("logged",e));const o=x(e.logEvent?.data)??this.defaultLogPath;if(!o){r();return}(async()=>{const n=j.dirname(o);this.ensuredDirs.has(n)||(this.ensuredDirs.add(n),await h.mkdir(n,{recursive:!0}).catch(()=>{}));const s=this.lineFormat.transform({...e},this.lineFormat.options??{}),m=s[T]??s.message??"";await h.appendFile(o,`${m}
|
|
2
|
+
`,"utf8")})().catch(n=>{const s=n instanceof Error?n.message:y.format("%o",n);console.warn("[tim-logger] Failed to write log file",{logPath:o,error:s})}).finally(r)}}const A=l.format.combine(l.format.timestamp({format:()=>new Date().toISOString()}),l.format.printf(t=>{const e=t.logEvent??{},r=t.timestamp??new Date().toISOString(),o=String(e.source??"LOG"),a=String(e.message??t.message??""),n=e.data??{},s=n&&typeof n=="object"&&Object.keys(n).length?` | ${JSON.stringify(n)}`:"";return`[${r}] ${o.padStart(12)}: ${a}${s}`})),I=["console"],G=(t,e,r,o)=>typeof t=="string"?{message:t,level:e,source:r,data:o}:{...t,level:t.level??e,source:t.source??r,data:t.data??o},N=t=>{if(!t)return"";try{return JSON.stringify(t)}catch{return"[unserializable data]"}},U=l.format.combine(l.format.timestamp(),l.format.errors({stack:!0}),l.format.printf(t=>{const{timestamp:e,level:r="info",message:o="",stack:a,logEvent:n}=t,s=n??{},m=s.source??"LOG",p=s.verb??r?.toUpperCase?.()??"LOG",w=[s.machineId,s.sessionId].filter(Boolean),P=w.length?d.gray(` [${w.join(" ")}]`):"",i=s.data?d.gray(` ${N(s.data)}`):"",c=typeof s.duration=="number"?d.yellow(` ${s.duration}ms`):"",g=typeof o=="string"?o.trimStart():String(o),f=`${e} ${d.bold(`[${m}]`)} ${d.cyan(p)} ${g}${c}${P}${i}`;return a?`${f}
|
|
3
|
+
${a}`:f})),b=(t={})=>{const e=t.transports?.length?t.transports:I;let r=!1;const o=[];e.includes("console")&&o.push(new l.transports.Console({format:U}));const a=t.logPath??t.fsPath??"log.txt";e.includes("fs")&&o.push(new R({defaultLogPath:a,format:A}));const n=l.createLogger({level:"info",transports:o.length?o:[new l.transports.Console]}),s=async(i,c)=>{const g=G(i,"info",t.source,c),f=g.level??"info",v={...g,level:f,timestamp:g.timestamp??Date.now()};if(n.log({level:f,message:v.message,logEvent:v}),e.includes("custom"))if(t.addLog){const C=L({...v});await Promise.resolve(t.addLog(C))}else r||(r=!0,console.warn("[tim-logger] Custom transport requested without addLog handler."))},m=async(i,c)=>s({message:i,level:"info",data:c,source:t.source}),p=async(i,c)=>s({message:i,level:"warn",data:c,source:t.source});return{log:s,info:m,warn:p,error:async(i,c)=>s({message:i,level:"error",data:c,source:t.source}),clearLogs:async(i=()=>!0)=>{if(e.includes("fs"))try{await h.mkdir(j.dirname(a),{recursive:!0}).catch(()=>{}),await h.writeFile(a,"",{flag:"a"}),await h.truncate(a,0),await m("🧹 Logs cleared",{logPath:a})}catch(c){const g=c instanceof Error?c.message:y.format("%o",c);await p("Failed to clear logs",{logPath:a,error:g})}e.includes("custom")&&t.clearLogs&&await t.clearLogs(i)}}};let u=b();const z=(t={})=>(u=b(t),u),H=(t,e)=>u.log(t,e),J=(t,e)=>u.info(t,e),B=(t,e)=>u.warn(t,e),$=(t,e)=>u.error(t,e),V=()=>u.clearLogs?.()??Promise.resolve();let E=!1;function F(t,e){const r=e instanceof Error?e.stack||e.message:y.format("%o",e);$(`❌ ${t}: ${r}`)}function K({onCrash:t}={}){if(E)return;E=!0;const e=r=>o=>{if(F(r,o),typeof t=="function")try{t(o)}catch(a){const n=a instanceof Error?a.stack||a.message:y.format("%o",a);$(`❌ Crash handler failed: ${n}`)}};process.on("uncaughtException",e("Uncaught Exception")),process.on("unhandledRejection",e("Unhandled Rejection"))}exports.interceptConsoleLogs=q.interceptConsoleLogs;exports.clearLogs=V;exports.createServerLogger=b;exports.error=$;exports.info=J;exports.init=z;exports.log=H;exports.logCrash=F;exports.registerCrashHandlers=K;exports.warn=B;
|
|
4
4
|
//# sourceMappingURL=server.cjs.js.map
|
package/dist/server.cjs.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"server.cjs.js","sources":["../src/object.ts","../src/server.ts"],"sourcesContent":["export function cleanObject<T>(input: T): T {\n if (input === undefined) return input;\n\n // Arrays: clean items and drop undefineds\n if (Array.isArray(input)) {\n const arr = (input as unknown[])\n .filter((v) => v !== undefined)\n .map((v) => cleanObject(v));\n return arr as unknown as T;\n }\n\n // Objects: clean keys recursively, drop undefined values\n if (input && typeof input === 'object') {\n const out: Record<string, unknown> = {};\n for (const [k, v] of Object.entries(input as Record<string, unknown>)) {\n if (v === undefined) continue;\n out[k] = cleanObject(v);\n }\n return out as unknown as T;\n }\n\n // Primitives / functions / etc: return as-is\n return input;\n}\n","import kleur from 'kleur';\nimport type { TransformableInfo } from 'logform';\nimport * as fs from 'node:fs/promises';\nimport * as path from 'node:path';\nimport * as util from 'node:util';\nimport {\n createLogger as createWinstonLogger,\n transports,\n format as winstonFormat,\n} from 'winston';\nimport TransportStream from 'winston-transport';\nimport { cleanObject } from './object';\nimport type { Logger, ServerLoggerConfig, ServerTransport } from './types';\n\ntype FileInfo = TransformableInfo & {\n logEvent?: Partial<LogEvent>;\n};\n\nconst MESSAGE = Symbol.for('message');\n\nconst getLogPathFromData = (data?: Json) => {\n if (!data || typeof data !== 'object') return undefined;\n const maybeLogPath = (data as Record<string, unknown>).logPath;\n return typeof maybeLogPath === 'string' && maybeLogPath.length\n ? maybeLogPath\n : undefined;\n};\n\nclass DynamicFileTransport extends TransportStream {\n private defaultLogPath: string;\n private ensuredDirs = new Set<string>();\n private lineFormat: ReturnType<typeof winstonFormat.combine>;\n\n constructor({\n defaultLogPath,\n format,\n }: {\n defaultLogPath: string;\n format: ReturnType<typeof winstonFormat.combine>;\n }) {\n // IMPORTANT: don't pass `format` into super if you're going to transform manually\n super();\n this.defaultLogPath = defaultLogPath;\n this.lineFormat = format;\n }\n\n override log(info: FileInfo, callback: () => void) {\n setImmediate(() => this.emit('logged', info));\n\n const logPath =\n getLogPathFromData(info.logEvent?.data) ?? this.defaultLogPath;\n\n if (!logPath) {\n callback();\n return;\n }\n\n const write = async () => {\n const dir = path.dirname(logPath);\n if (!this.ensuredDirs.has(dir)) {\n this.ensuredDirs.add(dir);\n await fs.mkdir(dir, { recursive: true }).catch(() => {});\n }\n\n // clone because logform formats mutate `info`\n const formatted = this.lineFormat.transform(\n { ...info },\n this.lineFormat.options ?? {}\n ) as FileInfo;\n\n const message =\n (formatted as any)[MESSAGE] ?? formatted.message ?? '';\n\n await fs.appendFile(logPath, `${message}\\n`, 'utf8');\n };\n\n void write()\n .catch((err) => {\n const msg =\n err instanceof Error ? err.message : util.format('%o', err);\n console.warn('[tim-logger] Failed to write log file', {\n logPath,\n error: msg,\n });\n })\n .finally(callback);\n }\n}\n\nconst fileLineFormat = winstonFormat.combine(\n winstonFormat.timestamp({ format: () => new Date().toISOString() }),\n winstonFormat.printf((info: FileInfo) => {\n const entry = info.logEvent ?? {};\n\n // Use the formatter timestamp (or fallback), instead of generating a new one.\n const timestamp =\n (info.timestamp as string) ?? new Date().toISOString();\n\n const name = String(entry.source ?? 'LOG');\n const text = String(entry.message ?? info.message ?? '');\n\n const data = (entry.data ?? {}) as Record<string, unknown>;\n const dataString =\n data && typeof data === 'object' && Object.keys(data).length\n ? ` | ${JSON.stringify(data)}`\n : '';\n\n return `[${timestamp}] ${name.padStart(12)}: ${text}${dataString}`;\n })\n);\n\nconst DEFAULT_SERVER_TRANSPORTS: ServerTransport[] = ['console'];\n\ntype ConsoleInfo = TransformableInfo & {\n timestamp?: string;\n stack?: string;\n logEvent?: Partial<LogEvent>;\n};\n\nconst normalizeEvent = (\n eventOrMessage: LogEvent | string,\n level?: LogLevel,\n source?: string,\n data?: Json\n): LogEvent => {\n if (typeof eventOrMessage === 'string') {\n return { message: eventOrMessage, level, source, data } as LogEvent;\n }\n\n return {\n ...eventOrMessage,\n level: eventOrMessage.level ?? level,\n source: eventOrMessage.source ?? source,\n data: eventOrMessage.data ?? data,\n };\n};\n\nconst stringifyData = (data?: Json) => {\n if (!data) return '';\n try {\n return JSON.stringify(data);\n } catch {\n return '[unserializable data]';\n }\n};\n\nconst consoleFormat = winstonFormat.combine(\n winstonFormat.timestamp(),\n winstonFormat.errors({ stack: true }),\n winstonFormat.printf((info: ConsoleInfo) => {\n const {\n timestamp,\n level = 'info',\n message = '',\n stack,\n logEvent,\n } = info;\n\n const entry: Partial<LogEvent> = logEvent ?? {};\n const source = entry.source ?? 'LOG';\n const verb = entry.verb ?? level?.toUpperCase?.() ?? 'LOG';\n\n const contextParts = [entry.machineId, entry.sessionId].filter(Boolean);\n const ctx = contextParts.length\n ? kleur.gray(` [${contextParts.join(' ')}]`)\n : '';\n\n const data = entry.data\n ? kleur.gray(` ${stringifyData(entry.data)}`)\n : '';\n\n const duration =\n typeof entry.duration === 'number'\n ? kleur.yellow(` ${entry.duration}ms`)\n : '';\n\n const cleanMessage =\n typeof message === 'string' ? message.trimStart() : String(message);\n\n const base = `${timestamp} ${kleur.bold(`[${source}]`)} ${kleur.cyan(\n verb\n )} ${cleanMessage}${duration}${ctx}${data}`;\n\n return stack ? `${base}\\n${stack}` : base;\n })\n);\n\nexport const createServerLogger = (config: ServerLoggerConfig = {}): Logger => {\n const transportsArr = config.transports?.length\n ? config.transports\n : DEFAULT_SERVER_TRANSPORTS;\n let warnedMissingCustom = false;\n\n const winstonTransports: TransportStream[] = [];\n\n if (transportsArr.includes('console')) {\n winstonTransports.push(\n new transports.Console({\n format: consoleFormat,\n })\n );\n }\n\n const defaultLogPath = config.logPath ?? config.fsPath ?? 'log.txt';\n\n if (transportsArr.includes('fs')) {\n winstonTransports.push(\n new DynamicFileTransport({\n defaultLogPath,\n format: fileLineFormat,\n })\n );\n }\n\n const baseLogger = createWinstonLogger({\n level: 'info',\n transports: winstonTransports.length\n ? winstonTransports\n : [new transports.Console()],\n });\n\n const log = async (eventOrMessage: LogEvent | string, data?: Json) => {\n const event = normalizeEvent(\n eventOrMessage,\n 'info',\n config.source,\n data\n );\n const resolvedLevel = event.level ?? 'info';\n\n const payload = {\n ...event,\n level: resolvedLevel,\n timestamp: event.timestamp ?? Date.now(),\n };\n\n baseLogger.log({\n level: resolvedLevel,\n message: payload.message,\n logEvent: payload,\n });\n\n if (transportsArr.includes('custom')) {\n if (config.addLog) {\n const clean = { ...payload };\n clean.data = cleanObject(clean.data);\n await Promise.resolve(config.addLog(clean));\n } else if (!warnedMissingCustom) {\n warnedMissingCustom = true;\n console.warn(\n '[tim-logger] Custom transport requested without addLog handler.'\n );\n }\n }\n };\n\n const info = async (message: string, data?: Json) =>\n log({\n message,\n level: 'info',\n data,\n source: config.source,\n } as LogEvent);\n\n const warn = async (message: string, data?: Json) =>\n log({\n message,\n level: 'warn',\n data,\n source: config.source,\n } as LogEvent);\n\n const error = async (message: string, data?: Json) =>\n log({\n message,\n level: 'error',\n data,\n source: config.source,\n } as LogEvent);\n\n /**\n * Clears the fs transport logs by truncating the file.\n * No-op if 'fs' transport is not enabled.\n */\n const clearLogs = async (predicate = () => true) => {\n if (transportsArr.includes('fs')) {\n try {\n // Ensure file exists, then truncate.\n await fs\n .mkdir(path.dirname(defaultLogPath), {\n recursive: true,\n })\n .catch(() => {});\n await fs.writeFile(defaultLogPath, '', { flag: 'a' }); // touch\n await fs.truncate(defaultLogPath, 0);\n await info('🧹 Logs cleared', { logPath: defaultLogPath });\n } catch (err) {\n // Don’t throw; log the failure instead.\n const msg =\n err instanceof Error ? err.message : util.format('%o', err);\n await warn('Failed to clear logs', {\n logPath: defaultLogPath,\n error: msg,\n });\n }\n }\n\n if (transportsArr.includes('custom') && config.clearLogs) {\n await config.clearLogs(predicate);\n }\n };\n\n return { log, info, warn, error, clearLogs };\n};\n\nlet defaultLogger = createServerLogger();\n\nexport const init = (config: ServerLoggerConfig = {}) => {\n defaultLogger = createServerLogger(config);\n return defaultLogger;\n};\n\nexport const log = (eventOrMessage: LogEvent | string, data?: Json) =>\n defaultLogger.log(eventOrMessage, data);\nexport const info = (message: string, data?: Json) =>\n defaultLogger.info(message, data);\nexport const warn = (message: string, data?: Json) =>\n defaultLogger.warn(message, data);\nexport const error = (message: string, data?: Json) =>\n defaultLogger.error(message, data);\nexport const clearLogs = () => defaultLogger.clearLogs?.() ?? Promise.resolve();\nexport { interceptConsoleLogs } from './console-intercept';\n\n// newly added\nlet crashHandlersRegistered = false;\n\nexport function logCrash(type: string, err: unknown) {\n const message =\n err instanceof Error\n ? err.stack || err.message\n : util.format('%o', err);\n void error(`❌ ${type}: ${message}`);\n}\n\nexport function registerCrashHandlers({\n onCrash,\n}: { onCrash?: (err: unknown) => void } = {}) {\n if (crashHandlersRegistered) return;\n crashHandlersRegistered = true;\n\n const handleCrash = (type: string) => (err: unknown) => {\n logCrash(type, err);\n if (typeof onCrash === 'function') {\n try {\n onCrash(err);\n } catch (handlerErr) {\n // FIX: was `log.error(...)` (wrong symbol). Use our logger.\n const msg =\n handlerErr instanceof Error\n ? handlerErr.stack || handlerErr.message\n : util.format('%o', handlerErr);\n void error(`❌ Crash handler failed: ${msg}`);\n }\n }\n };\n\n process.on('uncaughtException', handleCrash('Uncaught Exception'));\n process.on('unhandledRejection', handleCrash('Unhandled Rejection'));\n}\n"],"names":["cleanObject","input","v","out","k","MESSAGE","getLogPathFromData","data","maybeLogPath","DynamicFileTransport","TransportStream","defaultLogPath","format","info","callback","logPath","dir","path","fs","formatted","message","err","msg","util","fileLineFormat","winstonFormat","entry","timestamp","name","text","dataString","DEFAULT_SERVER_TRANSPORTS","normalizeEvent","eventOrMessage","level","source","stringifyData","consoleFormat","stack","logEvent","verb","contextParts","ctx","kleur","duration","cleanMessage","base","createServerLogger","config","transportsArr","warnedMissingCustom","winstonTransports","transports","baseLogger","createWinstonLogger","log","event","resolvedLevel","payload","clean","warn","predicate","clearLogs","defaultLogger","init","error","crashHandlersRegistered","logCrash","type","registerCrashHandlers","onCrash","handleCrash","handlerErr"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"server.cjs.js","sources":["../src/object.ts","../src/server.ts"],"sourcesContent":["export function cleanObject<T>(input: T): T {\n if (input === undefined) return input;\n\n // Arrays: clean items and drop undefineds\n if (Array.isArray(input)) {\n const arr = (input as unknown[])\n .filter((v) => v !== undefined)\n .map((v) => cleanObject(v));\n return arr as unknown as T;\n }\n\n // Objects: clean keys recursively, drop undefined values\n if (input && typeof input === 'object') {\n const out: Record<string, unknown> = {};\n for (const [k, v] of Object.entries(input as Record<string, unknown>)) {\n if (v === undefined) continue;\n out[k] = cleanObject(v);\n }\n return out as unknown as T;\n }\n\n // Primitives / functions / etc: return as-is\n\n return input;\n}\n","import kleur from 'kleur';\nimport type { TransformableInfo } from 'logform';\nimport * as fs from 'node:fs/promises';\nimport * as path from 'node:path';\nimport * as util from 'node:util';\nimport {\n createLogger as createWinstonLogger,\n transports,\n format as winstonFormat,\n} from 'winston';\nimport TransportStream from 'winston-transport';\nimport { cleanObject } from './object';\nimport type { Logger, ServerLoggerConfig, ServerTransport } from './types';\n\ntype FileInfo = TransformableInfo & {\n logEvent?: Partial<LogEvent>;\n};\n\nconst MESSAGE = Symbol.for('message');\n\nconst getLogPathFromData = (data?: Json) => {\n if (!data || typeof data !== 'object') return undefined;\n const maybeLogPath = (data as Record<string, unknown>).logPath;\n return typeof maybeLogPath === 'string' && maybeLogPath.length\n ? maybeLogPath\n : undefined;\n};\n\nclass DynamicFileTransport extends TransportStream {\n private defaultLogPath: string;\n private ensuredDirs = new Set<string>();\n private lineFormat: ReturnType<typeof winstonFormat.combine>;\n\n constructor({\n defaultLogPath,\n format,\n }: {\n defaultLogPath: string;\n format: ReturnType<typeof winstonFormat.combine>;\n }) {\n // IMPORTANT: don't pass `format` into super if you're going to transform manually\n super();\n this.defaultLogPath = defaultLogPath;\n this.lineFormat = format;\n }\n\n override log(info: FileInfo, callback: () => void) {\n setImmediate(() => this.emit('logged', info));\n\n const logPath =\n getLogPathFromData(info.logEvent?.data) ?? this.defaultLogPath;\n\n if (!logPath) {\n callback();\n return;\n }\n\n const write = async () => {\n const dir = path.dirname(logPath);\n if (!this.ensuredDirs.has(dir)) {\n this.ensuredDirs.add(dir);\n await fs.mkdir(dir, { recursive: true }).catch(() => {});\n }\n\n // clone because logform formats mutate `info`\n const formatted = this.lineFormat.transform(\n { ...info },\n this.lineFormat.options ?? {}\n ) as FileInfo;\n\n const message =\n (formatted as any)[MESSAGE] ?? formatted.message ?? '';\n\n await fs.appendFile(logPath, `${message}\\n`, 'utf8');\n };\n\n void write()\n .catch((err) => {\n const msg =\n err instanceof Error ? err.message : util.format('%o', err);\n console.warn('[tim-logger] Failed to write log file', {\n logPath,\n error: msg,\n });\n })\n .finally(callback);\n }\n}\n\nconst fileLineFormat = winstonFormat.combine(\n winstonFormat.timestamp({ format: () => new Date().toISOString() }),\n winstonFormat.printf((info: FileInfo) => {\n const entry = info.logEvent ?? {};\n\n // Use the formatter timestamp (or fallback), instead of generating a new one.\n const timestamp =\n (info.timestamp as string) ?? new Date().toISOString();\n\n const name = String(entry.source ?? 'LOG');\n const text = String(entry.message ?? info.message ?? '');\n\n const data = (entry.data ?? {}) as Record<string, unknown>;\n const dataString =\n data && typeof data === 'object' && Object.keys(data).length\n ? ` | ${JSON.stringify(data)}`\n : '';\n\n return `[${timestamp}] ${name.padStart(12)}: ${text}${dataString}`;\n })\n);\n\nconst DEFAULT_SERVER_TRANSPORTS: ServerTransport[] = ['console'];\n\ntype ConsoleInfo = TransformableInfo & {\n timestamp?: string;\n stack?: string;\n logEvent?: Partial<LogEvent>;\n};\n\nconst normalizeEvent = (\n eventOrMessage: LogEvent | string,\n level?: LogLevel,\n source?: string,\n data?: Json\n): LogEvent => {\n if (typeof eventOrMessage === 'string') {\n return { message: eventOrMessage, level, source, data } as LogEvent;\n }\n\n return {\n ...eventOrMessage,\n level: eventOrMessage.level ?? level,\n source: eventOrMessage.source ?? source,\n data: eventOrMessage.data ?? data,\n };\n};\n\nconst stringifyData = (data?: Json) => {\n if (!data) return '';\n try {\n return JSON.stringify(data);\n } catch {\n return '[unserializable data]';\n }\n};\n\nconst consoleFormat = winstonFormat.combine(\n winstonFormat.timestamp(),\n winstonFormat.errors({ stack: true }),\n winstonFormat.printf((info: ConsoleInfo) => {\n const {\n timestamp,\n level = 'info',\n message = '',\n stack,\n logEvent,\n } = info;\n\n const entry: Partial<LogEvent> = logEvent ?? {};\n const source = entry.source ?? 'LOG';\n const verb = entry.verb ?? level?.toUpperCase?.() ?? 'LOG';\n\n const contextParts = [entry.machineId, entry.sessionId].filter(Boolean);\n const ctx = contextParts.length\n ? kleur.gray(` [${contextParts.join(' ')}]`)\n : '';\n\n const data = entry.data\n ? kleur.gray(` ${stringifyData(entry.data)}`)\n : '';\n\n const duration =\n typeof entry.duration === 'number'\n ? kleur.yellow(` ${entry.duration}ms`)\n : '';\n\n const cleanMessage =\n typeof message === 'string' ? message.trimStart() : String(message);\n\n const base = `${timestamp} ${kleur.bold(`[${source}]`)} ${kleur.cyan(\n verb\n )} ${cleanMessage}${duration}${ctx}${data}`;\n\n return stack ? `${base}\\n${stack}` : base;\n })\n);\n\nexport const createServerLogger = (config: ServerLoggerConfig = {}): Logger => {\n const transportsArr = config.transports?.length\n ? config.transports\n : DEFAULT_SERVER_TRANSPORTS;\n let warnedMissingCustom = false;\n\n const winstonTransports: TransportStream[] = [];\n\n if (transportsArr.includes('console')) {\n winstonTransports.push(\n new transports.Console({\n format: consoleFormat,\n })\n );\n }\n\n const defaultLogPath = config.logPath ?? config.fsPath ?? 'log.txt';\n\n if (transportsArr.includes('fs')) {\n winstonTransports.push(\n new DynamicFileTransport({\n defaultLogPath,\n format: fileLineFormat,\n })\n );\n }\n\n const baseLogger = createWinstonLogger({\n level: 'info',\n transports: winstonTransports.length\n ? winstonTransports\n : [new transports.Console()],\n });\n\n const log = async (eventOrMessage: LogEvent | string, data?: Json) => {\n const event = normalizeEvent(\n eventOrMessage,\n 'info',\n config.source,\n data\n );\n const resolvedLevel = event.level ?? 'info';\n\n const payload = {\n ...event,\n level: resolvedLevel,\n timestamp: event.timestamp ?? Date.now(),\n };\n\n baseLogger.log({\n level: resolvedLevel,\n message: payload.message,\n logEvent: payload,\n });\n\n if (transportsArr.includes('custom')) {\n if (config.addLog) {\n const clean = cleanObject({ ...payload });\n await Promise.resolve(config.addLog(clean));\n } else if (!warnedMissingCustom) {\n warnedMissingCustom = true;\n console.warn(\n '[tim-logger] Custom transport requested without addLog handler.'\n );\n }\n }\n };\n\n const info = async (message: string, data?: Json) =>\n log({\n message,\n level: 'info',\n data,\n source: config.source,\n } as LogEvent);\n\n const warn = async (message: string, data?: Json) =>\n log({\n message,\n level: 'warn',\n data,\n source: config.source,\n } as LogEvent);\n\n const error = async (message: string, data?: Json) =>\n log({\n message,\n level: 'error',\n data,\n source: config.source,\n } as LogEvent);\n\n /**\n * Clears the fs transport logs by truncating the file.\n * No-op if 'fs' transport is not enabled.\n */\n const clearLogs = async (predicate = () => true) => {\n if (transportsArr.includes('fs')) {\n try {\n // Ensure file exists, then truncate.\n await fs\n .mkdir(path.dirname(defaultLogPath), {\n recursive: true,\n })\n .catch(() => {});\n await fs.writeFile(defaultLogPath, '', { flag: 'a' }); // touch\n await fs.truncate(defaultLogPath, 0);\n await info('🧹 Logs cleared', { logPath: defaultLogPath });\n } catch (err) {\n // Don’t throw; log the failure instead.\n const msg =\n err instanceof Error ? err.message : util.format('%o', err);\n await warn('Failed to clear logs', {\n logPath: defaultLogPath,\n error: msg,\n });\n }\n }\n\n if (transportsArr.includes('custom') && config.clearLogs) {\n await config.clearLogs(predicate);\n }\n };\n\n return { log, info, warn, error, clearLogs };\n};\n\nlet defaultLogger = createServerLogger();\n\nexport const init = (config: ServerLoggerConfig = {}) => {\n defaultLogger = createServerLogger(config);\n return defaultLogger;\n};\n\nexport const log = (eventOrMessage: LogEvent | string, data?: Json) =>\n defaultLogger.log(eventOrMessage, data);\nexport const info = (message: string, data?: Json) =>\n defaultLogger.info(message, data);\nexport const warn = (message: string, data?: Json) =>\n defaultLogger.warn(message, data);\nexport const error = (message: string, data?: Json) =>\n defaultLogger.error(message, data);\nexport const clearLogs = () => defaultLogger.clearLogs?.() ?? Promise.resolve();\nexport { interceptConsoleLogs } from './console-intercept';\n\n// newly added\nlet crashHandlersRegistered = false;\n\nexport function logCrash(type: string, err: unknown) {\n const message =\n err instanceof Error\n ? err.stack || err.message\n : util.format('%o', err);\n void error(`❌ ${type}: ${message}`);\n}\n\nexport function registerCrashHandlers({\n onCrash,\n}: { onCrash?: (err: unknown) => void } = {}) {\n if (crashHandlersRegistered) return;\n crashHandlersRegistered = true;\n\n const handleCrash = (type: string) => (err: unknown) => {\n logCrash(type, err);\n if (typeof onCrash === 'function') {\n try {\n onCrash(err);\n } catch (handlerErr) {\n // FIX: was `log.error(...)` (wrong symbol). Use our logger.\n const msg =\n handlerErr instanceof Error\n ? handlerErr.stack || handlerErr.message\n : util.format('%o', handlerErr);\n void error(`❌ Crash handler failed: ${msg}`);\n }\n }\n };\n\n process.on('uncaughtException', handleCrash('Uncaught Exception'));\n process.on('unhandledRejection', handleCrash('Unhandled Rejection'));\n}\n"],"names":["cleanObject","input","v","out","k","MESSAGE","getLogPathFromData","data","maybeLogPath","DynamicFileTransport","TransportStream","defaultLogPath","format","info","callback","logPath","dir","path","fs","formatted","message","err","msg","util","fileLineFormat","winstonFormat","entry","timestamp","name","text","dataString","DEFAULT_SERVER_TRANSPORTS","normalizeEvent","eventOrMessage","level","source","stringifyData","consoleFormat","stack","logEvent","verb","contextParts","ctx","kleur","duration","cleanMessage","base","createServerLogger","config","transportsArr","warnedMissingCustom","winstonTransports","transports","baseLogger","createWinstonLogger","log","event","resolvedLevel","payload","clean","warn","predicate","clearLogs","defaultLogger","init","error","crashHandlersRegistered","logCrash","type","registerCrashHandlers","onCrash","handleCrash","handlerErr"],"mappings":"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"}
|
package/dist/server.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"server.d.ts","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"AAYA,OAAO,KAAK,EAAE,MAAM,EAAE,kBAAkB,EAAmB,MAAM,SAAS,CAAC;AA+K3E,eAAO,MAAM,kBAAkB,GAAI,SAAQ,kBAAuB,KAAG,
|
|
1
|
+
{"version":3,"file":"server.d.ts","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"AAYA,OAAO,KAAK,EAAE,MAAM,EAAE,kBAAkB,EAAmB,MAAM,SAAS,CAAC;AA+K3E,eAAO,MAAM,kBAAkB,GAAI,SAAQ,kBAAuB,KAAG,MA6HpE,CAAC;AAIF,eAAO,MAAM,IAAI,GAAI,SAAQ,kBAAuB,WAGnD,CAAC;AAEF,eAAO,MAAM,GAAG,GAAI,gBAAgB,QAAQ,GAAG,MAAM,EAAE,OAAO,IAAI,kBACvB,CAAC;AAC5C,eAAO,MAAM,IAAI,GAAI,SAAS,MAAM,EAAE,OAAO,IAAI,kBACZ,CAAC;AACtC,eAAO,MAAM,IAAI,GAAI,SAAS,MAAM,EAAE,OAAO,IAAI,kBACZ,CAAC;AACtC,eAAO,MAAM,KAAK,GAAI,SAAS,MAAM,EAAE,OAAO,IAAI,kBACZ,CAAC;AACvC,eAAO,MAAM,SAAS,qBAAyD,CAAC;AAChF,OAAO,EAAE,oBAAoB,EAAE,MAAM,qBAAqB,CAAC;AAK3D,wBAAgB,QAAQ,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,EAAE,OAAO,QAMlD;AAED,wBAAgB,qBAAqB,CAAC,EAClC,OAAO,GACV,GAAE;IAAE,OAAO,CAAC,EAAE,CAAC,GAAG,EAAE,OAAO,KAAK,IAAI,CAAA;CAAO,QAsB3C"}
|
package/dist/server.es.js
CHANGED
|
@@ -1,18 +1,18 @@
|
|
|
1
1
|
import u from "kleur";
|
|
2
2
|
import * as h from "node:fs/promises";
|
|
3
|
-
import * as
|
|
3
|
+
import * as b from "node:path";
|
|
4
4
|
import * as y from "node:util";
|
|
5
|
-
import { format as m, transports as
|
|
5
|
+
import { format as m, transports as S, createLogger as j } from "winston";
|
|
6
6
|
import x from "winston-transport";
|
|
7
7
|
import { i as X } from "./console-intercept-ByhsZ0kO.js";
|
|
8
|
-
function
|
|
8
|
+
function L(t) {
|
|
9
9
|
if (t === void 0) return t;
|
|
10
10
|
if (Array.isArray(t))
|
|
11
|
-
return t.filter((o) => o !== void 0).map((o) =>
|
|
11
|
+
return t.filter((o) => o !== void 0).map((o) => L(o));
|
|
12
12
|
if (t && typeof t == "object") {
|
|
13
13
|
const e = {};
|
|
14
14
|
for (const [o, r] of Object.entries(t))
|
|
15
|
-
r !== void 0 && (e[o] =
|
|
15
|
+
r !== void 0 && (e[o] = L(r));
|
|
16
16
|
return e;
|
|
17
17
|
}
|
|
18
18
|
return t;
|
|
@@ -40,14 +40,14 @@ class R extends x {
|
|
|
40
40
|
return;
|
|
41
41
|
}
|
|
42
42
|
(async () => {
|
|
43
|
-
const a =
|
|
43
|
+
const a = b.dirname(r);
|
|
44
44
|
this.ensuredDirs.has(a) || (this.ensuredDirs.add(a), await h.mkdir(a, { recursive: !0 }).catch(() => {
|
|
45
45
|
}));
|
|
46
46
|
const s = this.lineFormat.transform(
|
|
47
47
|
{ ...e },
|
|
48
48
|
this.lineFormat.options ?? {}
|
|
49
|
-
),
|
|
50
|
-
await h.appendFile(r, `${
|
|
49
|
+
), f = s[C] ?? s.message ?? "";
|
|
50
|
+
await h.appendFile(r, `${f}
|
|
51
51
|
`, "utf8");
|
|
52
52
|
})().catch((a) => {
|
|
53
53
|
const s = a instanceof Error ? a.message : y.format("%o", a);
|
|
@@ -86,18 +86,18 @@ const A = m.combine(
|
|
|
86
86
|
message: r = "",
|
|
87
87
|
stack: n,
|
|
88
88
|
logEvent: a
|
|
89
|
-
} = t, s = a ?? {},
|
|
89
|
+
} = t, s = a ?? {}, f = s.source ?? "LOG", p = s.verb ?? o?.toUpperCase?.() ?? "LOG", w = [s.machineId, s.sessionId].filter(Boolean), $ = w.length ? u.gray(` [${w.join(" ")}]`) : "", i = s.data ? u.gray(` ${I(s.data)}`) : "", c = typeof s.duration == "number" ? u.yellow(` ${s.duration}ms`) : "", l = typeof r == "string" ? r.trimStart() : String(r), d = `${e} ${u.bold(`[${f}]`)} ${u.cyan(
|
|
90
90
|
p
|
|
91
|
-
)} ${l}${c}${
|
|
92
|
-
return n ? `${
|
|
93
|
-
${n}` :
|
|
91
|
+
)} ${l}${c}${$}${i}`;
|
|
92
|
+
return n ? `${d}
|
|
93
|
+
${n}` : d;
|
|
94
94
|
})
|
|
95
|
-
),
|
|
95
|
+
), P = (t = {}) => {
|
|
96
96
|
const e = t.transports?.length ? t.transports : O;
|
|
97
97
|
let o = !1;
|
|
98
98
|
const r = [];
|
|
99
99
|
e.includes("console") && r.push(
|
|
100
|
-
new
|
|
100
|
+
new S.Console({
|
|
101
101
|
format: G
|
|
102
102
|
})
|
|
103
103
|
);
|
|
@@ -110,30 +110,30 @@ ${n}` : f;
|
|
|
110
110
|
);
|
|
111
111
|
const a = j({
|
|
112
112
|
level: "info",
|
|
113
|
-
transports: r.length ? r : [new
|
|
113
|
+
transports: r.length ? r : [new S.Console()]
|
|
114
114
|
}), s = async (i, c) => {
|
|
115
115
|
const l = T(
|
|
116
116
|
i,
|
|
117
117
|
"info",
|
|
118
118
|
t.source,
|
|
119
119
|
c
|
|
120
|
-
),
|
|
120
|
+
), d = l.level ?? "info", v = {
|
|
121
121
|
...l,
|
|
122
|
-
level:
|
|
122
|
+
level: d,
|
|
123
123
|
timestamp: l.timestamp ?? Date.now()
|
|
124
124
|
};
|
|
125
125
|
if (a.log({
|
|
126
|
-
level:
|
|
126
|
+
level: d,
|
|
127
127
|
message: v.message,
|
|
128
128
|
logEvent: v
|
|
129
129
|
}), e.includes("custom"))
|
|
130
130
|
if (t.addLog) {
|
|
131
|
-
const
|
|
132
|
-
|
|
131
|
+
const D = L({ ...v });
|
|
132
|
+
await Promise.resolve(t.addLog(D));
|
|
133
133
|
} else o || (o = !0, console.warn(
|
|
134
134
|
"[tim-logger] Custom transport requested without addLog handler."
|
|
135
135
|
));
|
|
136
|
-
},
|
|
136
|
+
}, f = async (i, c) => s({
|
|
137
137
|
message: i,
|
|
138
138
|
level: "info",
|
|
139
139
|
data: c,
|
|
@@ -144,7 +144,7 @@ ${n}` : f;
|
|
|
144
144
|
data: c,
|
|
145
145
|
source: t.source
|
|
146
146
|
});
|
|
147
|
-
return { log: s, info:
|
|
147
|
+
return { log: s, info: f, warn: p, error: async (i, c) => s({
|
|
148
148
|
message: i,
|
|
149
149
|
level: "error",
|
|
150
150
|
data: c,
|
|
@@ -152,10 +152,10 @@ ${n}` : f;
|
|
|
152
152
|
}), clearLogs: async (i = () => !0) => {
|
|
153
153
|
if (e.includes("fs"))
|
|
154
154
|
try {
|
|
155
|
-
await h.mkdir(
|
|
155
|
+
await h.mkdir(b.dirname(n), {
|
|
156
156
|
recursive: !0
|
|
157
157
|
}).catch(() => {
|
|
158
|
-
}), await h.writeFile(n, "", { flag: "a" }), await h.truncate(n, 0), await
|
|
158
|
+
}), await h.writeFile(n, "", { flag: "a" }), await h.truncate(n, 0), await f("🧹 Logs cleared", { logPath: n });
|
|
159
159
|
} catch (c) {
|
|
160
160
|
const l = c instanceof Error ? c.message : y.format("%o", c);
|
|
161
161
|
await p("Failed to clear logs", {
|
|
@@ -166,33 +166,33 @@ ${n}` : f;
|
|
|
166
166
|
e.includes("custom") && t.clearLogs && await t.clearLogs(i);
|
|
167
167
|
} };
|
|
168
168
|
};
|
|
169
|
-
let g =
|
|
170
|
-
const J = (t = {}) => (g =
|
|
171
|
-
let
|
|
169
|
+
let g = P();
|
|
170
|
+
const J = (t = {}) => (g = P(t), g), _ = (t, e) => g.log(t, e), q = (t, e) => g.info(t, e), B = (t, e) => g.warn(t, e), F = (t, e) => g.error(t, e), V = () => g.clearLogs?.() ?? Promise.resolve();
|
|
171
|
+
let E = !1;
|
|
172
172
|
function U(t, e) {
|
|
173
173
|
const o = e instanceof Error ? e.stack || e.message : y.format("%o", e);
|
|
174
|
-
|
|
174
|
+
F(`❌ ${t}: ${o}`);
|
|
175
175
|
}
|
|
176
176
|
function K({
|
|
177
177
|
onCrash: t
|
|
178
178
|
} = {}) {
|
|
179
|
-
if (
|
|
180
|
-
|
|
179
|
+
if (E) return;
|
|
180
|
+
E = !0;
|
|
181
181
|
const e = (o) => (r) => {
|
|
182
182
|
if (U(o, r), typeof t == "function")
|
|
183
183
|
try {
|
|
184
184
|
t(r);
|
|
185
185
|
} catch (n) {
|
|
186
186
|
const a = n instanceof Error ? n.stack || n.message : y.format("%o", n);
|
|
187
|
-
|
|
187
|
+
F(`❌ Crash handler failed: ${a}`);
|
|
188
188
|
}
|
|
189
189
|
};
|
|
190
190
|
process.on("uncaughtException", e("Uncaught Exception")), process.on("unhandledRejection", e("Unhandled Rejection"));
|
|
191
191
|
}
|
|
192
192
|
export {
|
|
193
193
|
V as clearLogs,
|
|
194
|
-
|
|
195
|
-
|
|
194
|
+
P as createServerLogger,
|
|
195
|
+
F as error,
|
|
196
196
|
q as info,
|
|
197
197
|
J as init,
|
|
198
198
|
X as interceptConsoleLogs,
|
package/dist/server.es.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"server.es.js","sources":["../src/object.ts","../src/server.ts"],"sourcesContent":["export function cleanObject<T>(input: T): T {\n if (input === undefined) return input;\n\n // Arrays: clean items and drop undefineds\n if (Array.isArray(input)) {\n const arr = (input as unknown[])\n .filter((v) => v !== undefined)\n .map((v) => cleanObject(v));\n return arr as unknown as T;\n }\n\n // Objects: clean keys recursively, drop undefined values\n if (input && typeof input === 'object') {\n const out: Record<string, unknown> = {};\n for (const [k, v] of Object.entries(input as Record<string, unknown>)) {\n if (v === undefined) continue;\n out[k] = cleanObject(v);\n }\n return out as unknown as T;\n }\n\n // Primitives / functions / etc: return as-is\n return input;\n}\n","import kleur from 'kleur';\nimport type { TransformableInfo } from 'logform';\nimport * as fs from 'node:fs/promises';\nimport * as path from 'node:path';\nimport * as util from 'node:util';\nimport {\n createLogger as createWinstonLogger,\n transports,\n format as winstonFormat,\n} from 'winston';\nimport TransportStream from 'winston-transport';\nimport { cleanObject } from './object';\nimport type { Logger, ServerLoggerConfig, ServerTransport } from './types';\n\ntype FileInfo = TransformableInfo & {\n logEvent?: Partial<LogEvent>;\n};\n\nconst MESSAGE = Symbol.for('message');\n\nconst getLogPathFromData = (data?: Json) => {\n if (!data || typeof data !== 'object') return undefined;\n const maybeLogPath = (data as Record<string, unknown>).logPath;\n return typeof maybeLogPath === 'string' && maybeLogPath.length\n ? maybeLogPath\n : undefined;\n};\n\nclass DynamicFileTransport extends TransportStream {\n private defaultLogPath: string;\n private ensuredDirs = new Set<string>();\n private lineFormat: ReturnType<typeof winstonFormat.combine>;\n\n constructor({\n defaultLogPath,\n format,\n }: {\n defaultLogPath: string;\n format: ReturnType<typeof winstonFormat.combine>;\n }) {\n // IMPORTANT: don't pass `format` into super if you're going to transform manually\n super();\n this.defaultLogPath = defaultLogPath;\n this.lineFormat = format;\n }\n\n override log(info: FileInfo, callback: () => void) {\n setImmediate(() => this.emit('logged', info));\n\n const logPath =\n getLogPathFromData(info.logEvent?.data) ?? this.defaultLogPath;\n\n if (!logPath) {\n callback();\n return;\n }\n\n const write = async () => {\n const dir = path.dirname(logPath);\n if (!this.ensuredDirs.has(dir)) {\n this.ensuredDirs.add(dir);\n await fs.mkdir(dir, { recursive: true }).catch(() => {});\n }\n\n // clone because logform formats mutate `info`\n const formatted = this.lineFormat.transform(\n { ...info },\n this.lineFormat.options ?? {}\n ) as FileInfo;\n\n const message =\n (formatted as any)[MESSAGE] ?? formatted.message ?? '';\n\n await fs.appendFile(logPath, `${message}\\n`, 'utf8');\n };\n\n void write()\n .catch((err) => {\n const msg =\n err instanceof Error ? err.message : util.format('%o', err);\n console.warn('[tim-logger] Failed to write log file', {\n logPath,\n error: msg,\n });\n })\n .finally(callback);\n }\n}\n\nconst fileLineFormat = winstonFormat.combine(\n winstonFormat.timestamp({ format: () => new Date().toISOString() }),\n winstonFormat.printf((info: FileInfo) => {\n const entry = info.logEvent ?? {};\n\n // Use the formatter timestamp (or fallback), instead of generating a new one.\n const timestamp =\n (info.timestamp as string) ?? new Date().toISOString();\n\n const name = String(entry.source ?? 'LOG');\n const text = String(entry.message ?? info.message ?? '');\n\n const data = (entry.data ?? {}) as Record<string, unknown>;\n const dataString =\n data && typeof data === 'object' && Object.keys(data).length\n ? ` | ${JSON.stringify(data)}`\n : '';\n\n return `[${timestamp}] ${name.padStart(12)}: ${text}${dataString}`;\n })\n);\n\nconst DEFAULT_SERVER_TRANSPORTS: ServerTransport[] = ['console'];\n\ntype ConsoleInfo = TransformableInfo & {\n timestamp?: string;\n stack?: string;\n logEvent?: Partial<LogEvent>;\n};\n\nconst normalizeEvent = (\n eventOrMessage: LogEvent | string,\n level?: LogLevel,\n source?: string,\n data?: Json\n): LogEvent => {\n if (typeof eventOrMessage === 'string') {\n return { message: eventOrMessage, level, source, data } as LogEvent;\n }\n\n return {\n ...eventOrMessage,\n level: eventOrMessage.level ?? level,\n source: eventOrMessage.source ?? source,\n data: eventOrMessage.data ?? data,\n };\n};\n\nconst stringifyData = (data?: Json) => {\n if (!data) return '';\n try {\n return JSON.stringify(data);\n } catch {\n return '[unserializable data]';\n }\n};\n\nconst consoleFormat = winstonFormat.combine(\n winstonFormat.timestamp(),\n winstonFormat.errors({ stack: true }),\n winstonFormat.printf((info: ConsoleInfo) => {\n const {\n timestamp,\n level = 'info',\n message = '',\n stack,\n logEvent,\n } = info;\n\n const entry: Partial<LogEvent> = logEvent ?? {};\n const source = entry.source ?? 'LOG';\n const verb = entry.verb ?? level?.toUpperCase?.() ?? 'LOG';\n\n const contextParts = [entry.machineId, entry.sessionId].filter(Boolean);\n const ctx = contextParts.length\n ? kleur.gray(` [${contextParts.join(' ')}]`)\n : '';\n\n const data = entry.data\n ? kleur.gray(` ${stringifyData(entry.data)}`)\n : '';\n\n const duration =\n typeof entry.duration === 'number'\n ? kleur.yellow(` ${entry.duration}ms`)\n : '';\n\n const cleanMessage =\n typeof message === 'string' ? message.trimStart() : String(message);\n\n const base = `${timestamp} ${kleur.bold(`[${source}]`)} ${kleur.cyan(\n verb\n )} ${cleanMessage}${duration}${ctx}${data}`;\n\n return stack ? `${base}\\n${stack}` : base;\n })\n);\n\nexport const createServerLogger = (config: ServerLoggerConfig = {}): Logger => {\n const transportsArr = config.transports?.length\n ? config.transports\n : DEFAULT_SERVER_TRANSPORTS;\n let warnedMissingCustom = false;\n\n const winstonTransports: TransportStream[] = [];\n\n if (transportsArr.includes('console')) {\n winstonTransports.push(\n new transports.Console({\n format: consoleFormat,\n })\n );\n }\n\n const defaultLogPath = config.logPath ?? config.fsPath ?? 'log.txt';\n\n if (transportsArr.includes('fs')) {\n winstonTransports.push(\n new DynamicFileTransport({\n defaultLogPath,\n format: fileLineFormat,\n })\n );\n }\n\n const baseLogger = createWinstonLogger({\n level: 'info',\n transports: winstonTransports.length\n ? winstonTransports\n : [new transports.Console()],\n });\n\n const log = async (eventOrMessage: LogEvent | string, data?: Json) => {\n const event = normalizeEvent(\n eventOrMessage,\n 'info',\n config.source,\n data\n );\n const resolvedLevel = event.level ?? 'info';\n\n const payload = {\n ...event,\n level: resolvedLevel,\n timestamp: event.timestamp ?? Date.now(),\n };\n\n baseLogger.log({\n level: resolvedLevel,\n message: payload.message,\n logEvent: payload,\n });\n\n if (transportsArr.includes('custom')) {\n if (config.addLog) {\n const clean = { ...payload };\n clean.data = cleanObject(clean.data);\n await Promise.resolve(config.addLog(clean));\n } else if (!warnedMissingCustom) {\n warnedMissingCustom = true;\n console.warn(\n '[tim-logger] Custom transport requested without addLog handler.'\n );\n }\n }\n };\n\n const info = async (message: string, data?: Json) =>\n log({\n message,\n level: 'info',\n data,\n source: config.source,\n } as LogEvent);\n\n const warn = async (message: string, data?: Json) =>\n log({\n message,\n level: 'warn',\n data,\n source: config.source,\n } as LogEvent);\n\n const error = async (message: string, data?: Json) =>\n log({\n message,\n level: 'error',\n data,\n source: config.source,\n } as LogEvent);\n\n /**\n * Clears the fs transport logs by truncating the file.\n * No-op if 'fs' transport is not enabled.\n */\n const clearLogs = async (predicate = () => true) => {\n if (transportsArr.includes('fs')) {\n try {\n // Ensure file exists, then truncate.\n await fs\n .mkdir(path.dirname(defaultLogPath), {\n recursive: true,\n })\n .catch(() => {});\n await fs.writeFile(defaultLogPath, '', { flag: 'a' }); // touch\n await fs.truncate(defaultLogPath, 0);\n await info('🧹 Logs cleared', { logPath: defaultLogPath });\n } catch (err) {\n // Don’t throw; log the failure instead.\n const msg =\n err instanceof Error ? err.message : util.format('%o', err);\n await warn('Failed to clear logs', {\n logPath: defaultLogPath,\n error: msg,\n });\n }\n }\n\n if (transportsArr.includes('custom') && config.clearLogs) {\n await config.clearLogs(predicate);\n }\n };\n\n return { log, info, warn, error, clearLogs };\n};\n\nlet defaultLogger = createServerLogger();\n\nexport const init = (config: ServerLoggerConfig = {}) => {\n defaultLogger = createServerLogger(config);\n return defaultLogger;\n};\n\nexport const log = (eventOrMessage: LogEvent | string, data?: Json) =>\n defaultLogger.log(eventOrMessage, data);\nexport const info = (message: string, data?: Json) =>\n defaultLogger.info(message, data);\nexport const warn = (message: string, data?: Json) =>\n defaultLogger.warn(message, data);\nexport const error = (message: string, data?: Json) =>\n defaultLogger.error(message, data);\nexport const clearLogs = () => defaultLogger.clearLogs?.() ?? Promise.resolve();\nexport { interceptConsoleLogs } from './console-intercept';\n\n// newly added\nlet crashHandlersRegistered = false;\n\nexport function logCrash(type: string, err: unknown) {\n const message =\n err instanceof Error\n ? err.stack || err.message\n : util.format('%o', err);\n void error(`❌ ${type}: ${message}`);\n}\n\nexport function registerCrashHandlers({\n onCrash,\n}: { onCrash?: (err: unknown) => void } = {}) {\n if (crashHandlersRegistered) return;\n crashHandlersRegistered = true;\n\n const handleCrash = (type: string) => (err: unknown) => {\n logCrash(type, err);\n if (typeof onCrash === 'function') {\n try {\n onCrash(err);\n } catch (handlerErr) {\n // FIX: was `log.error(...)` (wrong symbol). Use our logger.\n const msg =\n handlerErr instanceof Error\n ? handlerErr.stack || handlerErr.message\n : util.format('%o', handlerErr);\n void error(`❌ Crash handler failed: ${msg}`);\n }\n }\n };\n\n process.on('uncaughtException', handleCrash('Uncaught Exception'));\n process.on('unhandledRejection', handleCrash('Unhandled Rejection'));\n}\n"],"names":["cleanObject","input","v","out","k","MESSAGE","getLogPathFromData","data","maybeLogPath","DynamicFileTransport","TransportStream","defaultLogPath","format","info","callback","logPath","dir","path","fs","formatted","message","err","msg","util","fileLineFormat","winstonFormat","entry","timestamp","name","text","dataString","DEFAULT_SERVER_TRANSPORTS","normalizeEvent","eventOrMessage","level","source","stringifyData","consoleFormat","stack","logEvent","verb","contextParts","ctx","kleur","duration","cleanMessage","base","createServerLogger","config","transportsArr","warnedMissingCustom","winstonTransports","transports","baseLogger","createWinstonLogger","log","event","resolvedLevel","payload","clean","warn","predicate","clearLogs","defaultLogger","init","error","crashHandlersRegistered","logCrash","type","registerCrashHandlers","onCrash","handleCrash","handlerErr"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"server.es.js","sources":["../src/object.ts","../src/server.ts"],"sourcesContent":["export function cleanObject<T>(input: T): T {\n if (input === undefined) return input;\n\n // Arrays: clean items and drop undefineds\n if (Array.isArray(input)) {\n const arr = (input as unknown[])\n .filter((v) => v !== undefined)\n .map((v) => cleanObject(v));\n return arr as unknown as T;\n }\n\n // Objects: clean keys recursively, drop undefined values\n if (input && typeof input === 'object') {\n const out: Record<string, unknown> = {};\n for (const [k, v] of Object.entries(input as Record<string, unknown>)) {\n if (v === undefined) continue;\n out[k] = cleanObject(v);\n }\n return out as unknown as T;\n }\n\n // Primitives / functions / etc: return as-is\n\n return input;\n}\n","import kleur from 'kleur';\nimport type { TransformableInfo } from 'logform';\nimport * as fs from 'node:fs/promises';\nimport * as path from 'node:path';\nimport * as util from 'node:util';\nimport {\n createLogger as createWinstonLogger,\n transports,\n format as winstonFormat,\n} from 'winston';\nimport TransportStream from 'winston-transport';\nimport { cleanObject } from './object';\nimport type { Logger, ServerLoggerConfig, ServerTransport } from './types';\n\ntype FileInfo = TransformableInfo & {\n logEvent?: Partial<LogEvent>;\n};\n\nconst MESSAGE = Symbol.for('message');\n\nconst getLogPathFromData = (data?: Json) => {\n if (!data || typeof data !== 'object') return undefined;\n const maybeLogPath = (data as Record<string, unknown>).logPath;\n return typeof maybeLogPath === 'string' && maybeLogPath.length\n ? maybeLogPath\n : undefined;\n};\n\nclass DynamicFileTransport extends TransportStream {\n private defaultLogPath: string;\n private ensuredDirs = new Set<string>();\n private lineFormat: ReturnType<typeof winstonFormat.combine>;\n\n constructor({\n defaultLogPath,\n format,\n }: {\n defaultLogPath: string;\n format: ReturnType<typeof winstonFormat.combine>;\n }) {\n // IMPORTANT: don't pass `format` into super if you're going to transform manually\n super();\n this.defaultLogPath = defaultLogPath;\n this.lineFormat = format;\n }\n\n override log(info: FileInfo, callback: () => void) {\n setImmediate(() => this.emit('logged', info));\n\n const logPath =\n getLogPathFromData(info.logEvent?.data) ?? this.defaultLogPath;\n\n if (!logPath) {\n callback();\n return;\n }\n\n const write = async () => {\n const dir = path.dirname(logPath);\n if (!this.ensuredDirs.has(dir)) {\n this.ensuredDirs.add(dir);\n await fs.mkdir(dir, { recursive: true }).catch(() => {});\n }\n\n // clone because logform formats mutate `info`\n const formatted = this.lineFormat.transform(\n { ...info },\n this.lineFormat.options ?? {}\n ) as FileInfo;\n\n const message =\n (formatted as any)[MESSAGE] ?? formatted.message ?? '';\n\n await fs.appendFile(logPath, `${message}\\n`, 'utf8');\n };\n\n void write()\n .catch((err) => {\n const msg =\n err instanceof Error ? err.message : util.format('%o', err);\n console.warn('[tim-logger] Failed to write log file', {\n logPath,\n error: msg,\n });\n })\n .finally(callback);\n }\n}\n\nconst fileLineFormat = winstonFormat.combine(\n winstonFormat.timestamp({ format: () => new Date().toISOString() }),\n winstonFormat.printf((info: FileInfo) => {\n const entry = info.logEvent ?? {};\n\n // Use the formatter timestamp (or fallback), instead of generating a new one.\n const timestamp =\n (info.timestamp as string) ?? new Date().toISOString();\n\n const name = String(entry.source ?? 'LOG');\n const text = String(entry.message ?? info.message ?? '');\n\n const data = (entry.data ?? {}) as Record<string, unknown>;\n const dataString =\n data && typeof data === 'object' && Object.keys(data).length\n ? ` | ${JSON.stringify(data)}`\n : '';\n\n return `[${timestamp}] ${name.padStart(12)}: ${text}${dataString}`;\n })\n);\n\nconst DEFAULT_SERVER_TRANSPORTS: ServerTransport[] = ['console'];\n\ntype ConsoleInfo = TransformableInfo & {\n timestamp?: string;\n stack?: string;\n logEvent?: Partial<LogEvent>;\n};\n\nconst normalizeEvent = (\n eventOrMessage: LogEvent | string,\n level?: LogLevel,\n source?: string,\n data?: Json\n): LogEvent => {\n if (typeof eventOrMessage === 'string') {\n return { message: eventOrMessage, level, source, data } as LogEvent;\n }\n\n return {\n ...eventOrMessage,\n level: eventOrMessage.level ?? level,\n source: eventOrMessage.source ?? source,\n data: eventOrMessage.data ?? data,\n };\n};\n\nconst stringifyData = (data?: Json) => {\n if (!data) return '';\n try {\n return JSON.stringify(data);\n } catch {\n return '[unserializable data]';\n }\n};\n\nconst consoleFormat = winstonFormat.combine(\n winstonFormat.timestamp(),\n winstonFormat.errors({ stack: true }),\n winstonFormat.printf((info: ConsoleInfo) => {\n const {\n timestamp,\n level = 'info',\n message = '',\n stack,\n logEvent,\n } = info;\n\n const entry: Partial<LogEvent> = logEvent ?? {};\n const source = entry.source ?? 'LOG';\n const verb = entry.verb ?? level?.toUpperCase?.() ?? 'LOG';\n\n const contextParts = [entry.machineId, entry.sessionId].filter(Boolean);\n const ctx = contextParts.length\n ? kleur.gray(` [${contextParts.join(' ')}]`)\n : '';\n\n const data = entry.data\n ? kleur.gray(` ${stringifyData(entry.data)}`)\n : '';\n\n const duration =\n typeof entry.duration === 'number'\n ? kleur.yellow(` ${entry.duration}ms`)\n : '';\n\n const cleanMessage =\n typeof message === 'string' ? message.trimStart() : String(message);\n\n const base = `${timestamp} ${kleur.bold(`[${source}]`)} ${kleur.cyan(\n verb\n )} ${cleanMessage}${duration}${ctx}${data}`;\n\n return stack ? `${base}\\n${stack}` : base;\n })\n);\n\nexport const createServerLogger = (config: ServerLoggerConfig = {}): Logger => {\n const transportsArr = config.transports?.length\n ? config.transports\n : DEFAULT_SERVER_TRANSPORTS;\n let warnedMissingCustom = false;\n\n const winstonTransports: TransportStream[] = [];\n\n if (transportsArr.includes('console')) {\n winstonTransports.push(\n new transports.Console({\n format: consoleFormat,\n })\n );\n }\n\n const defaultLogPath = config.logPath ?? config.fsPath ?? 'log.txt';\n\n if (transportsArr.includes('fs')) {\n winstonTransports.push(\n new DynamicFileTransport({\n defaultLogPath,\n format: fileLineFormat,\n })\n );\n }\n\n const baseLogger = createWinstonLogger({\n level: 'info',\n transports: winstonTransports.length\n ? winstonTransports\n : [new transports.Console()],\n });\n\n const log = async (eventOrMessage: LogEvent | string, data?: Json) => {\n const event = normalizeEvent(\n eventOrMessage,\n 'info',\n config.source,\n data\n );\n const resolvedLevel = event.level ?? 'info';\n\n const payload = {\n ...event,\n level: resolvedLevel,\n timestamp: event.timestamp ?? Date.now(),\n };\n\n baseLogger.log({\n level: resolvedLevel,\n message: payload.message,\n logEvent: payload,\n });\n\n if (transportsArr.includes('custom')) {\n if (config.addLog) {\n const clean = cleanObject({ ...payload });\n await Promise.resolve(config.addLog(clean));\n } else if (!warnedMissingCustom) {\n warnedMissingCustom = true;\n console.warn(\n '[tim-logger] Custom transport requested without addLog handler.'\n );\n }\n }\n };\n\n const info = async (message: string, data?: Json) =>\n log({\n message,\n level: 'info',\n data,\n source: config.source,\n } as LogEvent);\n\n const warn = async (message: string, data?: Json) =>\n log({\n message,\n level: 'warn',\n data,\n source: config.source,\n } as LogEvent);\n\n const error = async (message: string, data?: Json) =>\n log({\n message,\n level: 'error',\n data,\n source: config.source,\n } as LogEvent);\n\n /**\n * Clears the fs transport logs by truncating the file.\n * No-op if 'fs' transport is not enabled.\n */\n const clearLogs = async (predicate = () => true) => {\n if (transportsArr.includes('fs')) {\n try {\n // Ensure file exists, then truncate.\n await fs\n .mkdir(path.dirname(defaultLogPath), {\n recursive: true,\n })\n .catch(() => {});\n await fs.writeFile(defaultLogPath, '', { flag: 'a' }); // touch\n await fs.truncate(defaultLogPath, 0);\n await info('🧹 Logs cleared', { logPath: defaultLogPath });\n } catch (err) {\n // Don’t throw; log the failure instead.\n const msg =\n err instanceof Error ? err.message : util.format('%o', err);\n await warn('Failed to clear logs', {\n logPath: defaultLogPath,\n error: msg,\n });\n }\n }\n\n if (transportsArr.includes('custom') && config.clearLogs) {\n await config.clearLogs(predicate);\n }\n };\n\n return { log, info, warn, error, clearLogs };\n};\n\nlet defaultLogger = createServerLogger();\n\nexport const init = (config: ServerLoggerConfig = {}) => {\n defaultLogger = createServerLogger(config);\n return defaultLogger;\n};\n\nexport const log = (eventOrMessage: LogEvent | string, data?: Json) =>\n defaultLogger.log(eventOrMessage, data);\nexport const info = (message: string, data?: Json) =>\n defaultLogger.info(message, data);\nexport const warn = (message: string, data?: Json) =>\n defaultLogger.warn(message, data);\nexport const error = (message: string, data?: Json) =>\n defaultLogger.error(message, data);\nexport const clearLogs = () => defaultLogger.clearLogs?.() ?? Promise.resolve();\nexport { interceptConsoleLogs } from './console-intercept';\n\n// newly added\nlet crashHandlersRegistered = false;\n\nexport function logCrash(type: string, err: unknown) {\n const message =\n err instanceof Error\n ? err.stack || err.message\n : util.format('%o', err);\n void error(`❌ ${type}: ${message}`);\n}\n\nexport function registerCrashHandlers({\n onCrash,\n}: { onCrash?: (err: unknown) => void } = {}) {\n if (crashHandlersRegistered) return;\n crashHandlersRegistered = true;\n\n const handleCrash = (type: string) => (err: unknown) => {\n logCrash(type, err);\n if (typeof onCrash === 'function') {\n try {\n onCrash(err);\n } catch (handlerErr) {\n // FIX: was `log.error(...)` (wrong symbol). Use our logger.\n const msg =\n handlerErr instanceof Error\n ? handlerErr.stack || handlerErr.message\n : util.format('%o', handlerErr);\n void error(`❌ Crash handler failed: ${msg}`);\n }\n }\n };\n\n process.on('uncaughtException', handleCrash('Uncaught Exception'));\n process.on('unhandledRejection', handleCrash('Unhandled Rejection'));\n}\n"],"names":["cleanObject","input","v","out","k","MESSAGE","getLogPathFromData","data","maybeLogPath","DynamicFileTransport","TransportStream","defaultLogPath","format","info","callback","logPath","dir","path","fs","formatted","message","err","msg","util","fileLineFormat","winstonFormat","entry","timestamp","name","text","dataString","DEFAULT_SERVER_TRANSPORTS","normalizeEvent","eventOrMessage","level","source","stringifyData","consoleFormat","stack","logEvent","verb","contextParts","ctx","kleur","duration","cleanMessage","base","createServerLogger","config","transportsArr","warnedMissingCustom","winstonTransports","transports","baseLogger","createWinstonLogger","log","event","resolvedLevel","payload","clean","warn","predicate","clearLogs","defaultLogger","init","error","crashHandlersRegistered","logCrash","type","registerCrashHandlers","onCrash","handleCrash","handlerErr"],"mappings":"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"}
|