tim-logger 0.0.27 → 0.0.28

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.
@@ -1,4 +1,4 @@
1
- "use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const m=require("kleur"),l=require("winston"),P=require("winston-transport"),b=require("./console-intercept-CL_BfKQo.cjs"),C=(e={})=>Object.keys(e).reduce((t,o)=>(e[o]!==void 0&&(t[o]=e[o]),t),{}),F=Symbol.for("message"),D=e=>{if(!e||typeof e!="object")return;const t=e.logPath;return typeof t=="string"&&t.length?t:void 0};class j extends P{defaultLogPath;ensuredDirs=new Set;lineFormat;constructor({defaultLogPath:t,format:o}){super(),this.defaultLogPath=t,this.lineFormat=o}log(t,o){setImmediate(()=>this.emit("logged",t));const n=D(t.logEvent?.data)??this.defaultLogPath;if(!n){o();return}(async()=>{const s=(void 0)(n);this.ensuredDirs.has(s)||(this.ensuredDirs.add(s),await(void 0)(s,{recursive:!0}).catch(()=>{}));const r=this.lineFormat.transform({...t},this.lineFormat.options??{}),u=r[F]??r.message??"";await(void 0)(n,`${u}
2
- `,"utf8")})().catch(s=>{const r=s instanceof Error?s.message:(void 0)("%o",s);console.warn("[tim-logger] Failed to write log file",{logPath:n,error:r})}).finally(o)}}const x=l.format.combine(l.format.timestamp({format:()=>new Date().toISOString()}),l.format.printf(e=>{const t=e.logEvent??{},o=e.timestamp??new Date().toISOString(),n=String(t.source??"LOG"),a=String(t.message??e.message??""),s=t.data??{},r=s&&typeof s=="object"&&Object.keys(s).length?` | ${JSON.stringify(s)}`:"";return`[${o}] ${n.padStart(12)}: ${a}${r}`})),O=["console"],R=(e,t,o,n)=>typeof e=="string"?{message:e,level:t,source:o,data:n}:{...e,level:e.level??t,source:e.source??o,data:e.data??n},T=e=>{if(!e)return"";try{return JSON.stringify(e)}catch{return"[unserializable data]"}},I=l.format.combine(l.format.timestamp(),l.format.errors({stack:!0}),l.format.colorize({all:!0}),l.format.printf(e=>{const{timestamp:t,level:o="info",message:n="",stack:a,logEvent:s}=e,r=s??{},u=r.source??"LOG",f=r.verb??o?.toUpperCase?.()??"LOG",h=[r.machineId,r.sessionId].filter(Boolean),v=h.length?m.gray(` [${h.join(" ")}]`):"",i=r.data?m.gray(` ${T(r.data)}`):"",c=typeof r.duration=="number"?m.yellow(` ${r.duration}ms`):"",g=`${t} ${m.bold(`[${u}]`)} ${m.cyan(f)} ${n}${c}${v}${i}`;return a?`${g}
3
- ${a}`:g})),y=(e={})=>{const t=e.transports?.length?e.transports:O;let o=!1;const n=[];t.includes("console")&&n.push(new l.transports.Console({format:I}));const a=e.logPath??e.fsPath??"log.txt";t.includes("fs")&&n.push(new j({defaultLogPath:a,format:x}));const s=l.createLogger({level:"info",transports:n.length?n:[new l.transports.Console]}),r=async(i,c)=>{const g=R(i,"info",e.source,c),S=g.level??"info",p={...g,level:S,timestamp:g.timestamp??Date.now()};if(s.log({level:S,message:p.message,logEvent:p}),t.includes("custom"))if(e.addLog){const w={...p};w.data=C(w.data),await Promise.resolve(e.addLog(w))}else o||(o=!0,console.warn("[tim-logger] Custom transport requested without addLog handler."))},u=async(i,c)=>r({message:i,level:"info",data:c,source:e.source}),f=async(i,c)=>r({message:i,level:"warn",data:c,source:e.source});return{log:r,info:u,warn:f,error:async(i,c)=>r({message:i,level:"error",data:c,source:e.source}),clearLogs:async()=>{if(t.includes("fs"))try{await(void 0)((void 0)(a),{recursive:!0}).catch(()=>{}),await(void 0)(a,"",{flag:"a"}),await(void 0)(a,0),await u("🧹 Logs cleared",{logPath:a})}catch(i){const c=i instanceof Error?i.message:(void 0)("%o",i);await f("Failed to clear logs",{logPath:a,error:c})}}}};let d=y();const q=(e={})=>(d=y(e),d),k=(e,t)=>d.log(e,t),A=(e,t)=>d.info(e,t),G=(e,t)=>d.warn(e,t),L=(e,t)=>d.error(e,t),U=()=>d.clearLogs?.()??Promise.resolve();let $=!1;function E(e,t){const o=t instanceof Error?t.stack||t.message:(void 0)("%o",t);L(`❌ ${e}: ${o}`)}function z({onCrash:e}={}){if($)return;$=!0;const t=o=>n=>{if(E(o,n),typeof e=="function")try{e(n)}catch(a){const s=a instanceof Error?a.stack||a.message:(void 0)("%o",a);L(`❌ Crash handler failed: ${s}`)}};process.on("uncaughtException",t("Uncaught Exception")),process.on("unhandledRejection",t("Unhandled Rejection"))}exports.interceptConsoleLogs=b.interceptConsoleLogs;exports.clearLogs=U;exports.createServerLogger=y;exports.error=L;exports.info=A;exports.init=q;exports.log=k;exports.logCrash=E;exports.registerCrashHandlers=z;exports.warn=G;
1
+ "use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const m=require("kleur"),l=require("winston"),E=require("winston-transport"),b=require("./console-intercept-CL_BfKQo.cjs"),C=(e={})=>Object.keys(e).reduce((t,o)=>(e[o]!==void 0&&(t[o]=e[o]),t),{}),D=Symbol.for("message"),F=e=>{if(!e||typeof e!="object")return;const t=e.logPath;return typeof t=="string"&&t.length?t:void 0};class j extends E{defaultLogPath;ensuredDirs=new Set;lineFormat;constructor({defaultLogPath:t,format:o}){super(),this.defaultLogPath=t,this.lineFormat=o}log(t,o){setImmediate(()=>this.emit("logged",t));const r=F(t.logEvent?.data)??this.defaultLogPath;if(!r){o();return}(async()=>{const a=(void 0)(r);this.ensuredDirs.has(a)||(this.ensuredDirs.add(a),await(void 0)(a,{recursive:!0}).catch(()=>{}));const n=this.lineFormat.transform({...t},this.lineFormat.options??{}),u=n[D]??n.message??"";await(void 0)(r,`${u}
2
+ `,"utf8")})().finally(o)}}const x=l.format.combine(l.format.timestamp({format:()=>new Date().toISOString()}),l.format.printf(e=>{const t=e.logEvent??{},o=e.timestamp??new Date().toISOString(),r=String(t.source??"LOG"),s=String(t.message??e.message??""),a=t.data??{},n=a&&typeof a=="object"&&Object.keys(a).length?` | ${JSON.stringify(a)}`:"";return`[${o}] ${r.padStart(12)}: ${s}${n}`})),O=["console"],R=(e,t,o,r)=>typeof e=="string"?{message:e,level:t,source:o,data:r}:{...e,level:e.level??t,source:e.source??o,data:e.data??r},T=e=>{if(!e)return"";try{return JSON.stringify(e)}catch{return"[unserializable data]"}},I=l.format.combine(l.format.timestamp(),l.format.errors({stack:!0}),l.format.colorize({all:!0}),l.format.printf(e=>{const{timestamp:t,level:o="info",message:r="",stack:s,logEvent:a}=e,n=a??{},u=n.source??"LOG",f=n.verb??o?.toUpperCase?.()??"LOG",h=[n.machineId,n.sessionId].filter(Boolean),v=h.length?m.gray(` [${h.join(" ")}]`):"",i=n.data?m.gray(` ${T(n.data)}`):"",c=typeof n.duration=="number"?m.yellow(` ${n.duration}ms`):"",g=`${t} ${m.bold(`[${u}]`)} ${m.cyan(f)} ${r}${c}${v}${i}`;return s?`${g}
3
+ ${s}`:g})),w=(e={})=>{const t=e.transports?.length?e.transports:O;let o=!1;const r=[];t.includes("console")&&r.push(new l.transports.Console({format:I}));const s=e.logPath??e.fsPath??"log.txt";t.includes("fs")&&r.push(new j({defaultLogPath:s,format:x}));const a=l.createLogger({level:"info",transports:r.length?r:[new l.transports.Console]}),n=async(i,c)=>{const g=R(i,"info",e.source,c),S=g.level??"info",p={...g,level:S,timestamp:g.timestamp??Date.now()};if(a.log({level:S,message:p.message,logEvent:p}),t.includes("custom"))if(e.addLog){const y={...p};y.data=C(y.data),await Promise.resolve(e.addLog(y))}else o||(o=!0,console.warn("[tim-logger] Custom transport requested without addLog handler."))},u=async(i,c)=>n({message:i,level:"info",data:c,source:e.source}),f=async(i,c)=>n({message:i,level:"warn",data:c,source:e.source});return{log:n,info:u,warn:f,error:async(i,c)=>n({message:i,level:"error",data:c,source:e.source}),clearLogs:async()=>{if(t.includes("fs"))try{await(void 0)((void 0)(s),{recursive:!0}).catch(()=>{}),await(void 0)(s,"",{flag:"a"}),await(void 0)(s,0),await u("🧹 Logs cleared",{logPath:s})}catch(i){const c=i instanceof Error?i.message:(void 0)("%o",i);await f("Failed to clear logs",{logPath:s,error:c})}}}};let d=w();const q=(e={})=>(d=w(e),d),k=(e,t)=>d.log(e,t),A=(e,t)=>d.info(e,t),G=(e,t)=>d.warn(e,t),L=(e,t)=>d.error(e,t),U=()=>d.clearLogs?.()??Promise.resolve();let $=!1;function P(e,t){const o=t instanceof Error?t.stack||t.message:(void 0)("%o",t);L(`❌ ${e}: ${o}`)}function z({onCrash:e}={}){if($)return;$=!0;const t=o=>r=>{if(P(o,r),typeof e=="function")try{e(r)}catch(s){const a=s instanceof Error?s.stack||s.message:(void 0)("%o",s);L(`❌ Crash handler failed: ${a}`)}};process.on("uncaughtException",t("Uncaught Exception")),process.on("unhandledRejection",t("Unhandled Rejection"))}exports.interceptConsoleLogs=b.interceptConsoleLogs;exports.clearLogs=U;exports.createServerLogger=w;exports.error=L;exports.info=A;exports.init=q;exports.log=k;exports.logCrash=P;exports.registerCrashHandlers=z;exports.warn=G;
4
4
  //# sourceMappingURL=server.cjs.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"server.cjs.js","sources":["../src/object.ts","../src/server.ts"],"sourcesContent":["export const cleanObject = <T extends Json>(obj: T = {}): Partial<T> => {\n // remove keys with undefined values\n return Object.keys(obj).reduce((acc: T, key) => {\n if (obj[key] !== undefined) {\n // @ts-ignore\n acc[key] = obj[key];\n }\n return acc as T;\n }, {} as T);\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.colorize({ all: 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 base = `${timestamp} ${kleur.bold(`[${source}]`)} ${kleur.cyan(\n verb\n )} ${message}${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 () => {\n if (!transportsArr.includes('fs')) return;\n\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 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","obj","acc","key","MESSAGE","getLogPathFromData","data","maybeLogPath","DynamicFileTransport","TransportStream","defaultLogPath","format","info","callback","logPath","dir","path.dirname","fs.mkdir","formatted","message","fs.appendFile","err","msg","util.format","fileLineFormat","winstonFormat","entry","timestamp","name","text","dataString","DEFAULT_SERVER_TRANSPORTS","normalizeEvent","eventOrMessage","level","source","stringifyData","consoleFormat","stack","logEvent","verb","contextParts","ctx","kleur","duration","base","createServerLogger","config","transportsArr","warnedMissingCustom","winstonTransports","transports","baseLogger","createWinstonLogger","log","event","resolvedLevel","payload","clean","warn","fs.writeFile","fs.truncate","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 const cleanObject = <T extends Json>(obj: T = {}): Partial<T> => {\n // remove keys with undefined values\n return Object.keys(obj).reduce((acc: T, key) => {\n if (obj[key] !== undefined) {\n // @ts-ignore\n acc[key] = obj[key];\n }\n return acc as T;\n }, {} as T);\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.colorize({ all: 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 base = `${timestamp} ${kleur.bold(`[${source}]`)} ${kleur.cyan(\n verb\n )} ${message}${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 () => {\n if (!transportsArr.includes('fs')) return;\n\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 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","obj","acc","key","MESSAGE","getLogPathFromData","data","maybeLogPath","DynamicFileTransport","TransportStream","defaultLogPath","format","info","callback","logPath","dir","path.dirname","fs.mkdir","formatted","message","fs.appendFile","fileLineFormat","winstonFormat","entry","timestamp","name","text","dataString","DEFAULT_SERVER_TRANSPORTS","normalizeEvent","eventOrMessage","level","source","stringifyData","consoleFormat","stack","logEvent","verb","contextParts","ctx","kleur","duration","base","createServerLogger","config","transportsArr","warnedMissingCustom","winstonTransports","transports","baseLogger","createWinstonLogger","log","event","resolvedLevel","payload","clean","warn","fs.writeFile","fs.truncate","err","msg","util.format","clearLogs","defaultLogger","init","error","crashHandlersRegistered","logCrash","type","registerCrashHandlers","onCrash","handleCrash","handlerErr"],"mappings":"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"}
package/dist/server.es.js CHANGED
@@ -2,7 +2,7 @@ import m from "kleur";
2
2
  import { format as l, transports as v, createLogger as P } from "winston";
3
3
  import b from "winston-transport";
4
4
  import { i as K } from "./console-intercept-ByhsZ0kO.js";
5
- const F = (t = {}) => Object.keys(t).reduce((e, o) => (t[o] !== void 0 && (e[o] = t[o]), e), {}), D = Symbol.for("message"), x = (t) => {
5
+ const D = (t = {}) => Object.keys(t).reduce((e, o) => (t[o] !== void 0 && (e[o] = t[o]), e), {}), F = Symbol.for("message"), x = (t) => {
6
6
  if (!t || typeof t != "object") return;
7
7
  const e = t.logPath;
8
8
  return typeof e == "string" && e.length ? e : void 0;
@@ -25,29 +25,23 @@ class C extends b {
25
25
  return;
26
26
  }
27
27
  (async () => {
28
- const s = (void 0)(r);
29
- this.ensuredDirs.has(s) || (this.ensuredDirs.add(s), await (void 0)(s, { recursive: !0 }).catch(() => {
28
+ const a = (void 0)(r);
29
+ this.ensuredDirs.has(a) || (this.ensuredDirs.add(a), await (void 0)(a, { recursive: !0 }).catch(() => {
30
30
  }));
31
- const n = this.lineFormat.transform(
31
+ const s = this.lineFormat.transform(
32
32
  { ...e },
33
33
  this.lineFormat.options ?? {}
34
- ), u = n[D] ?? n.message ?? "";
34
+ ), u = s[F] ?? s.message ?? "";
35
35
  await (void 0)(r, `${u}
36
36
  `, "utf8");
37
- })().catch((s) => {
38
- const n = s instanceof Error ? s.message : (void 0)("%o", s);
39
- console.warn("[tim-logger] Failed to write log file", {
40
- logPath: r,
41
- error: n
42
- });
43
- }).finally(o);
37
+ })().finally(o);
44
38
  }
45
39
  }
46
40
  const R = l.combine(
47
41
  l.timestamp({ format: () => (/* @__PURE__ */ new Date()).toISOString() }),
48
42
  l.printf((t) => {
49
- const e = t.logEvent ?? {}, o = t.timestamp ?? (/* @__PURE__ */ new Date()).toISOString(), r = String(e.source ?? "LOG"), a = String(e.message ?? t.message ?? ""), s = e.data ?? {}, n = s && typeof s == "object" && Object.keys(s).length ? ` | ${JSON.stringify(s)}` : "";
50
- return `[${o}] ${r.padStart(12)}: ${a}${n}`;
43
+ const e = t.logEvent ?? {}, o = t.timestamp ?? (/* @__PURE__ */ new Date()).toISOString(), r = String(e.source ?? "LOG"), n = String(e.message ?? t.message ?? ""), a = e.data ?? {}, s = a && typeof a == "object" && Object.keys(a).length ? ` | ${JSON.stringify(a)}` : "";
44
+ return `[${o}] ${r.padStart(12)}: ${n}${s}`;
51
45
  })
52
46
  ), j = ["console"], O = (t, e, o, r) => typeof t == "string" ? { message: t, level: e, source: o, data: r } : {
53
47
  ...t,
@@ -70,13 +64,13 @@ const R = l.combine(
70
64
  timestamp: e,
71
65
  level: o = "info",
72
66
  message: r = "",
73
- stack: a,
74
- logEvent: s
75
- } = t, n = s ?? {}, u = n.source ?? "LOG", f = n.verb ?? o?.toUpperCase?.() ?? "LOG", h = [n.machineId, n.sessionId].filter(Boolean), y = h.length ? m.gray(` [${h.join(" ")}]`) : "", i = n.data ? m.gray(` ${T(n.data)}`) : "", c = typeof n.duration == "number" ? m.yellow(` ${n.duration}ms`) : "", g = `${e} ${m.bold(`[${u}]`)} ${m.cyan(
67
+ stack: n,
68
+ logEvent: a
69
+ } = t, s = a ?? {}, u = s.source ?? "LOG", f = s.verb ?? o?.toUpperCase?.() ?? "LOG", h = [s.machineId, s.sessionId].filter(Boolean), w = h.length ? m.gray(` [${h.join(" ")}]`) : "", i = s.data ? m.gray(` ${T(s.data)}`) : "", c = typeof s.duration == "number" ? m.yellow(` ${s.duration}ms`) : "", g = `${e} ${m.bold(`[${u}]`)} ${m.cyan(
76
70
  f
77
- )} ${r}${c}${y}${i}`;
78
- return a ? `${g}
79
- ${a}` : g;
71
+ )} ${r}${c}${w}${i}`;
72
+ return n ? `${g}
73
+ ${n}` : g;
80
74
  })
81
75
  ), S = (t = {}) => {
82
76
  const e = t.transports?.length ? t.transports : j;
@@ -87,17 +81,17 @@ ${a}` : g;
87
81
  format: I
88
82
  })
89
83
  );
90
- const a = t.logPath ?? t.fsPath ?? "log.txt";
84
+ const n = t.logPath ?? t.fsPath ?? "log.txt";
91
85
  e.includes("fs") && r.push(
92
86
  new C({
93
- defaultLogPath: a,
87
+ defaultLogPath: n,
94
88
  format: R
95
89
  })
96
90
  );
97
- const s = P({
91
+ const a = P({
98
92
  level: "info",
99
93
  transports: r.length ? r : [new v.Console()]
100
- }), n = async (i, c) => {
94
+ }), s = async (i, c) => {
101
95
  const g = O(
102
96
  i,
103
97
  "info",
@@ -108,29 +102,29 @@ ${a}` : g;
108
102
  level: L,
109
103
  timestamp: g.timestamp ?? Date.now()
110
104
  };
111
- if (s.log({
105
+ if (a.log({
112
106
  level: L,
113
107
  message: p.message,
114
108
  logEvent: p
115
109
  }), e.includes("custom"))
116
110
  if (t.addLog) {
117
- const w = { ...p };
118
- w.data = F(w.data), await Promise.resolve(t.addLog(w));
111
+ const y = { ...p };
112
+ y.data = D(y.data), await Promise.resolve(t.addLog(y));
119
113
  } else o || (o = !0, console.warn(
120
114
  "[tim-logger] Custom transport requested without addLog handler."
121
115
  ));
122
- }, u = async (i, c) => n({
116
+ }, u = async (i, c) => s({
123
117
  message: i,
124
118
  level: "info",
125
119
  data: c,
126
120
  source: t.source
127
- }), f = async (i, c) => n({
121
+ }), f = async (i, c) => s({
128
122
  message: i,
129
123
  level: "warn",
130
124
  data: c,
131
125
  source: t.source
132
126
  });
133
- return { log: n, info: u, warn: f, error: async (i, c) => n({
127
+ return { log: s, info: u, warn: f, error: async (i, c) => s({
134
128
  message: i,
135
129
  level: "error",
136
130
  data: c,
@@ -138,14 +132,14 @@ ${a}` : g;
138
132
  }), clearLogs: async () => {
139
133
  if (e.includes("fs"))
140
134
  try {
141
- await (void 0)((void 0)(a), {
135
+ await (void 0)((void 0)(n), {
142
136
  recursive: !0
143
137
  }).catch(() => {
144
- }), await (void 0)(a, "", { flag: "a" }), await (void 0)(a, 0), await u("🧹 Logs cleared", { logPath: a });
138
+ }), await (void 0)(n, "", { flag: "a" }), await (void 0)(n, 0), await u("🧹 Logs cleared", { logPath: n });
145
139
  } catch (i) {
146
140
  const c = i instanceof Error ? i.message : (void 0)("%o", i);
147
141
  await f("Failed to clear logs", {
148
- logPath: a,
142
+ logPath: n,
149
143
  error: c
150
144
  });
151
145
  }
@@ -167,9 +161,9 @@ function q({
167
161
  if (k(o, r), typeof t == "function")
168
162
  try {
169
163
  t(r);
170
- } catch (a) {
171
- const s = a instanceof Error ? a.stack || a.message : (void 0)("%o", a);
172
- E(`❌ Crash handler failed: ${s}`);
164
+ } catch (n) {
165
+ const a = n instanceof Error ? n.stack || n.message : (void 0)("%o", n);
166
+ E(`❌ Crash handler failed: ${a}`);
173
167
  }
174
168
  };
175
169
  process.on("uncaughtException", e("Uncaught Exception")), process.on("unhandledRejection", e("Unhandled Rejection"));
@@ -1 +1 @@
1
- {"version":3,"file":"server.es.js","sources":["../src/object.ts","../src/server.ts"],"sourcesContent":["export const cleanObject = <T extends Json>(obj: T = {}): Partial<T> => {\n // remove keys with undefined values\n return Object.keys(obj).reduce((acc: T, key) => {\n if (obj[key] !== undefined) {\n // @ts-ignore\n acc[key] = obj[key];\n }\n return acc as T;\n }, {} as T);\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.colorize({ all: 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 base = `${timestamp} ${kleur.bold(`[${source}]`)} ${kleur.cyan(\n verb\n )} ${message}${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 () => {\n if (!transportsArr.includes('fs')) return;\n\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 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","obj","acc","key","MESSAGE","getLogPathFromData","data","maybeLogPath","DynamicFileTransport","TransportStream","defaultLogPath","format","info","callback","logPath","dir","path.dirname","fs.mkdir","formatted","message","fs.appendFile","err","msg","util.format","fileLineFormat","winstonFormat","entry","timestamp","name","text","dataString","DEFAULT_SERVER_TRANSPORTS","normalizeEvent","eventOrMessage","level","source","stringifyData","consoleFormat","stack","logEvent","verb","contextParts","ctx","kleur","duration","base","createServerLogger","config","transportsArr","warnedMissingCustom","winstonTransports","transports","baseLogger","createWinstonLogger","log","event","resolvedLevel","payload","clean","warn","fs.writeFile","fs.truncate","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 const cleanObject = <T extends Json>(obj: T = {}): Partial<T> => {\n // remove keys with undefined values\n return Object.keys(obj).reduce((acc: T, key) => {\n if (obj[key] !== undefined) {\n // @ts-ignore\n acc[key] = obj[key];\n }\n return acc as T;\n }, {} as T);\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.colorize({ all: 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 base = `${timestamp} ${kleur.bold(`[${source}]`)} ${kleur.cyan(\n verb\n )} ${message}${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 () => {\n if (!transportsArr.includes('fs')) return;\n\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 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","obj","acc","key","MESSAGE","getLogPathFromData","data","maybeLogPath","DynamicFileTransport","TransportStream","defaultLogPath","format","info","callback","logPath","dir","path.dirname","fs.mkdir","formatted","message","fs.appendFile","fileLineFormat","winstonFormat","entry","timestamp","name","text","dataString","DEFAULT_SERVER_TRANSPORTS","normalizeEvent","eventOrMessage","level","source","stringifyData","consoleFormat","stack","logEvent","verb","contextParts","ctx","kleur","duration","base","createServerLogger","config","transportsArr","warnedMissingCustom","winstonTransports","transports","baseLogger","createWinstonLogger","log","event","resolvedLevel","payload","clean","warn","fs.writeFile","fs.truncate","err","msg","util.format","clearLogs","defaultLogger","init","error","crashHandlersRegistered","logCrash","type","registerCrashHandlers","onCrash","handleCrash","handlerErr"],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "tim-logger",
3
- "version": "0.0.27",
3
+ "version": "0.0.28",
4
4
  "type": "module",
5
5
  "private": false,
6
6
  "scripts": {