logixlysia 5.1.1 โ 5.3.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +6 -0
- package/dist/index.cjs +12 -12
- package/dist/index.d.cts +22 -8
- package/dist/index.d.ts +22 -8
- package/dist/index.js +12 -12
- package/package.json +10 -8
package/README.md
CHANGED
|
@@ -28,6 +28,12 @@ const app = new Elysia({
|
|
|
28
28
|
},
|
|
29
29
|
ip: true,
|
|
30
30
|
logFilePath: './logs/example.log',
|
|
31
|
+
logRotation: {
|
|
32
|
+
maxSize: '10m',
|
|
33
|
+
interval: '1d',
|
|
34
|
+
maxFiles: '7d',
|
|
35
|
+
compress: true
|
|
36
|
+
},
|
|
31
37
|
customLogFormat:
|
|
32
38
|
'๐ฆ {now} {level} {duration} {method} {pathname} {status} {message} {ip} {epoch}',
|
|
33
39
|
logFilter: {
|
package/dist/index.cjs
CHANGED
|
@@ -1,13 +1,13 @@
|
|
|
1
|
-
var
|
|
2
|
-
โ${
|
|
3
|
-
โ${
|
|
4
|
-
โ${
|
|
5
|
-
โ${
|
|
6
|
-
โ${
|
|
7
|
-
โ${
|
|
8
|
-
โ${
|
|
9
|
-
`)}else console.log(`\uD83E\uDD8A Elysia is running at ${
|
|
10
|
-
`;await
|
|
1
|
+
var lr=require("node:module");var pr=Object.create;var{getPrototypeOf:mr,defineProperty:x,getOwnPropertyNames:J,getOwnPropertyDescriptor:ar}=Object,q=Object.prototype.hasOwnProperty;var S=(r,o,n)=>{n=r!=null?pr(mr(r)):{};let t=o||!r||!r.__esModule?x(n,"default",{value:r,enumerable:!0}):n;for(let s of J(r))if(!q.call(t,s))x(t,s,{get:()=>r[s],enumerable:!0});return t},A=new WeakMap,ir=(r)=>{var o=A.get(r),n;if(o)return o;if(o=x({},"__esModule",{value:!0}),r&&typeof r==="object"||typeof r==="function")J(r).map((t)=>!q.call(o,t)&&x(o,t,{get:()=>r[t],enumerable:!(n=ar(r,t))||n.enumerable}));return A.set(r,o),o};var Lr=(r,o)=>{for(var n in o)x(r,n,{get:o[n],enumerable:!0,configurable:!0,set:(t)=>o[n]=()=>t})};var Gr={};Lr(Gr,{logToTransports:()=>l,handleHttpError:()=>E,default:()=>cr,createLogger:()=>R});module.exports=ir(Gr);var gr=require("elysia");var F=(r,o)=>{let n=Math.max(0,(o-r.length)/2),t=" ".repeat(n);return`${t}${r}${t}`.padEnd(o)};function d(r,o){let n=r?.hostname??"localhost",t=r?.port??3000,s=r?.protocol??"http";if(o?.config?.startupMessageFormat!=="simple"){let c=`\uD83E\uDD8A Elysia is running at ${s}://${n}:${t}`,p=Math.max(22,c.length)+4,y="โ".repeat(p),i=F("",p);console.log(`
|
|
2
|
+
โ${y}โ
|
|
3
|
+
โ${i}โ
|
|
4
|
+
โ${F("Elysia with Logixlysia",p)}โ
|
|
5
|
+
โ${i}โ
|
|
6
|
+
โ${F(c,p)}โ
|
|
7
|
+
โ${i}โ
|
|
8
|
+
โ${y}โ
|
|
9
|
+
`)}else console.log(`\uD83E\uDD8A Elysia is running at ${s}://${n}:${t}`)}var f=S(require("chalk")),Q=require("elysia");function N(r){if(typeof r==="number")return r;return Q.StatusMap[r]||500}function v(r,o){let n=r.toString();if(!o)return n;if(r>=500)return f.default.red(n);if(r>=400)return f.default.yellow(n);if(r>=300)return f.default.cyan(n);if(r>=200)return f.default.green(n);return f.default.white(n)}var O=S(require("pino"));var W=S(require("chalk"));var m=S(require("chalk")),G={INFO:m.default.bgGreen.black,WARNING:m.default.bgYellow.black,ERROR:m.default.bgRed.black},K={GET:m.default.green,POST:m.default.yellow,PUT:m.default.blue,PATCH:m.default.magentaBright,DELETE:m.default.red,HEAD:m.default.cyan,OPTIONS:m.default.magenta};var B=S(require("chalk")),fr=[{unit:"s",threshold:1e9,decimalPlaces:2},{unit:"ms",threshold:1e6,decimalPlaces:0},{unit:"ยตs",threshold:1000,decimalPlaces:0},{unit:"ns",threshold:1,decimalPlaces:0}];function M(r,o){let n=Number(process.hrtime.bigint()-r);for(let{unit:t,threshold:s,decimalPlaces:e}of fr)if(n>=s){let c=`${(n/s).toFixed(e)}${t}`.padStart(8).padEnd(16);return o?B.default.gray(c):c}return o?B.default.gray("0ns".padStart(8).padEnd(16)):"0ns".padStart(8).padEnd(16)}function $(r,o){let n=r.toUpperCase();return o?G[n]?.(n.padEnd(7))||n:n.padEnd(7)}function I(r,o){let n=K[r];return o&&n?n(r.padEnd(7)):r.padEnd(7)}function U(r){try{return new URL(r.url).pathname}catch{return}}var a=(r)=>r.toString().padStart(2,"0");function br(r){let o=r.getFullYear(),n=a(r.getMonth()+1),t=a(r.getDate()),s=a(r.getHours()),e=a(r.getMinutes()),g=a(r.getSeconds()),c=r.getMilliseconds().toString().padStart(3,"0");return`${o}-${n}-${t} ${s}:${e}:${g}.${c}`}function Or(r,o){let n={yyyy:r.getFullYear(),yy:r.getFullYear().toString().slice(-2),mm:a(r.getMonth()+1),dd:a(r.getDate()),HH:a(r.getHours()),MM:a(r.getMinutes()),ss:a(r.getSeconds()),SSS:a(r.getMilliseconds()),Z:-r.getTimezoneOffset()/60};return o.replace(/yyyy|yy|mm|dd|HH|MM|ss|SSS|Z/g,(t)=>(n[t]??"").toString())}function H(r,o){if(!o?.translateTime)return r.toISOString();if(o.translateTime===!0||o.translateTime==="SYS:STANDARD")return br(r);return Or(r,o.translateTime)}var yr="\uD83E\uDD8A {now} {level} {duration} {method} {pathname} {status} {message} {context} {ip}";function xr(r,o){if(o?.config?.useColors!==void 0)return o.config.useColors&&process.env.NO_COLOR===void 0;return r&&process.env.NO_COLOR===void 0}function L(r,o,n,t,s,e=!0){let g=xr(e,s),c=new Date,p={now:g?W.default.bgYellow(W.default.black(H(c,s?.config?.timestamp))):H(c,s?.config?.timestamp),epoch:Math.floor(c.getTime()/1000).toString(),level:$(r,e),duration:M(t.beforeTime,e),method:I(o.method,e),pathname:U(o),status:v(n.status||200,e),message:n.message||"",context:n.context?(()=>{try{return JSON.stringify(n.context)}catch(i){return`[Error serializing context: ${i instanceof Error?i.message:"Unknown error"}]`}})():"",ip:s?.config?.ip&&o.headers.get("x-forwarded-for")?`IP: ${o.headers.get("x-forwarded-for")}`:""};return(s?.config?.customLogFormat||yr).replace(/{(\w+)}/g,(i,k)=>{if(k in p)return p[k]||"";return""})}async function l(r,o,n,t,s){if(!s?.config?.transports||s.config.transports.length===0)return;let e=L(r,o,n,t,s,!1),g=s.config.transports.map((c)=>c.log(r,e,{request:o,data:n,store:t}));await Promise.all(g)}var _=require("node:fs"),sr=require("node:path");var C=require("node:fs"),b=require("node:path"),Sr=/^(\d+(?:\.\d+)?)\s*([kmg])?b?$/,wr=/^(\d+)\s*([hdw])$/,Er=/^(\d+)\s*d$/;function X(r){if(typeof r==="number")return r;let o={k:1024,m:1048576,g:1073741824},n=r.toLowerCase().match(Sr);if(!n?.[1])throw Error(`Invalid size format: ${r}`);let t=Number.parseFloat(n[1]),s=n[2]??"";return Math.floor(t*(o[s]??1))}function T(r){let o={h:3600000,d:86400000,w:604800000},n=r.toLowerCase().match(wr);if(!n)throw Error(`Invalid interval format: ${r}`);let t=n[1],s=n[2];if(!t)throw Error(`Invalid interval format: ${r}`);if(!s)throw Error(`Invalid interval format: ${r}`);return Number.parseInt(t,10)*(o[s]??0)}function Z(r){if(typeof r==="number")return{type:"count",value:r};let o=r.toLowerCase().match(Er);if(o?.[1])return{type:"time",value:Number.parseInt(o[1],10)*24*60*60*1000};throw Error(`Invalid retention format: ${r}`)}async function Dr(r){try{return(await C.promises.stat(r)).mtime.getTime()}catch{return Date.now()}}var j=new Map;async function V(r,o){let n=Date.now(),t=j.get(r);if(t!==void 0)return n-t>=o;let s=await Dr(r);return j.set(r,s),n-s>=o}function h(r){j.set(r,Date.now())}async function z(r){let o=b.dirname(r),n=b.basename(r);try{let t=await C.promises.readdir(o),s=new RegExp(`^${n.replace(/[.*+?^${}()|[\]\\]/g,"\\$&")}\\.\\d{4}-\\d{2}-\\d{2}(-\\d{2}-\\d{2}-\\d{2})?(\\.gz)?$`),e=t.filter((c)=>s.test(c)).map((c)=>b.join(o,c));return(await Promise.all(e.map(async(c)=>{let p=await C.promises.stat(c);return{file:c,mtime:p.mtime.getTime()}}))).sort((c,p)=>p.mtime-c.mtime).map((c)=>c.file)}catch{return[]}}var u=require("node:fs"),P=require("node:stream/promises"),rr=require("node:zlib");function Rr(r,o){let n=o.getFullYear(),t=String(o.getMonth()+1).padStart(2,"0"),s=String(o.getDate()).padStart(2,"0"),e=String(o.getHours()).padStart(2,"0"),g=String(o.getMinutes()).padStart(2,"0"),c=String(o.getSeconds()).padStart(2,"0");return`${r}.${n}-${t}-${s}-${e}-${g}-${c}`}async function kr(r){try{if((await u.promises.stat(r)).size===0)return"";let n=Rr(r,new Date);return await u.promises.rename(r,n),h(r),n}catch(o){if(o.code!=="ENOENT")console.error(`Failed to rotate log file ${r}:`,o);return""}}async function dr(r){try{let o=`${r}.gz`,n=u.createReadStream(r),t=u.createWriteStream(o),s=rr.createGzip();await P.pipeline(n,s,t),await u.promises.unlink(r)}catch(o){console.error(`Failed to compress file ${r}:`,o)}}async function vr(r,o){if(!o.maxFiles)return;try{let n=await z(r);if(n.length===0)return;let t=Z(o.maxFiles);if(t.type==="count"){let s=n.slice(t.value);await Promise.all(s.map((e)=>u.promises.unlink(e)))}else if(t.type==="time"){let s=Date.now()-t.value,e=await Promise.all(n.map(async(g)=>{return(await u.promises.stat(g)).mtime.getTime()<s?g:null}));await Promise.all(e.filter((g)=>g!==null).map((g)=>u.promises.unlink(g)))}}catch(n){console.error(`Failed to clean old log files for ${r}:`,n)}}async function or(r,o){try{let n=await kr(r);if(!n)return;if(o.compress)await dr(n);await vr(r,o)}catch(n){console.error(`Failed to perform rotation for ${r}:`,n)}}async function nr(r,o){try{await u.promises.access(r)}catch{return!1}if(o.maxSize)try{let n=X(o.maxSize);if((await u.promises.stat(r)).size>=n)return!0}catch(n){console.error(`Failed to check file size for ${r}:`,n)}if(o.interval)try{let n=T(o.interval),t=await u.promises.stat(r);if(Date.now()-t.mtimeMs>=n)return!0}catch(n){console.error(`Failed to check file age for ${r}:`,n)}return!1}var tr=new Set;async function Mr(r){let o=sr.dirname(r);if(!tr.has(o))await _.promises.mkdir(o,{recursive:!0}),tr.add(o)}async function $r(r,o){let n=o?.config?.logRotation;if(!n)return;let t=!1;if(n.maxSize)t=await nr(r,n);if(!t&&n.interval)try{let s=T(n.interval);t=await V(r,s)}catch(s){console.error("Invalid interval format in log rotation config:",s)}if(t)await or(r,n)}async function w(r,o,n,t,s,e){await Mr(r),await $r(r,e);let g=`${L(o,n,t,s,e,!1)}
|
|
10
|
+
`;await _.promises.appendFile(r,g,{flag:"a"})}var Y=(r,o)=>Array.isArray(r)?r.includes(o):r===o;function er(r,o,n,t){let s=t?.config?.logFilter;if(!s)return!0;return(!s.level||Y(s.level,r))&&(!s.status||Y(s.status,o))&&(!s.method||Y(s.method,n))}async function E(r,o,n,t){let e={status:o.status||500,message:o.message,stack:o.stack},g=[];if(!(t?.config?.useTransportsOnly||t?.config?.disableInternalLogger))console.error(L("ERROR",r,e,n,t));if(!t?.config?.useTransportsOnly&&t?.config?.logFilePath&&!t?.config?.disableFileLogging)g.push(w(t.config.logFilePath,"ERROR",r,e,n,t));if(t?.config?.transports?.length)g.push(l("ERROR",r,e,n,t));await Promise.all(g)}function Ir(){let r=process.memoryUsage().heapUsed/1024/1024,o=process.cpuUsage();return{memoryUsage:r,cpuUsage:o.user/1e6}}function Ur(r){return{level:r.level||"info",timestamp:r.timestamp??!0,messageKey:r.messageKey||"msg",errorKey:r.errorKey||"err",base:r.base||{pid:process.pid}}}function Hr(r){return O.default.transport({target:"pino-pretty",options:{colorize:!0,translateTime:"HH:MM:ss Z",ignore:"pid,hostname",...typeof r==="object"?r:{}}})}function Cr(r){let o=r?.config?.pino||{},{prettyPrint:n,transport:t,...s}=o,e={...Ur(o),...s};if(n)return O.default(e,Hr(n));if(t){if(typeof t==="object"&&"target"in t)return O.default(e,O.default.transport(t));console.warn("Invalid transport configuration provided, falling back to default")}return O.default(e)}function Tr(r){switch(r.toUpperCase()){case"DEBUG":return"debug";case"INFO":return"info";case"WARNING":case"WARN":return"warn";case"ERROR":return"error";default:return console.warn(`Unknown log level "${r}", defaulting to "info"`),"info"}}function Fr(r,o,n,t,s){let e=Tr(o),g=r[e],c=Boolean(s?.config?.pino?.transport||s?.config?.pino?.prettyPrint);if((!s?.config?.useTransportsOnly||c)&&typeof g==="function")g.call(r,n,t)}async function Nr(r,o,n,t,s,e){let g=[];if(!(s?.config?.useTransportsOnly||s?.config?.disableInternalLogger))console.log(e);if(!s?.config?.useTransportsOnly&&s?.config?.logFilePath&&!s?.config?.disableFileLogging)g.push(w(s.config.logFilePath,r,o,n,t,s));if(s?.config?.transports?.length)g.push(l(r,o,n,t,s));await Promise.all(g)}async function D(r,o,n,t,s,e){if(!er(o,t.status||200,n.method,e))return;if(!t.metrics)t.metrics=Ir();if(o==="ERROR"&&!t.stack){let ur=Error(t.message||"Unknown error");t.stack=ur.stack}let g=e?.config?.pino?.errorKey||"err",c=o==="ERROR"&&t.stack?{name:"Error",message:t.message||"Unknown error",stack:t.stack}:void 0,p=n.headers.get("x-forwarded-for"),y=e?.config?.ip&&p?p.split(",")[0]?.trim():void 0,i={method:n.method,url:n.url,status:t.status,message:t.message,context:t.context,metrics:t.metrics,duration:Number(process.hrtime.bigint()-s.beforeTime)/1e6,ip:y,[g]:c};Fr(r,o,i,t.message||"Request processed",e);let k=L(o,n,t,s,e,!0);await Nr(o,n,t,s,e,k)}function R(r){let o=Cr(r),n={store:void 0,pino:o,log:(t,s,e,g)=>D(o,t,s,e,g,r),handleHttpError:async(t,s,e)=>await E(t,s,e,r),customLogFormat:r?.config?.customLogFormat,info:(t,s,e,g)=>{let c=g||n.store||{beforeTime:process.hrtime.bigint()};return c.hasCustomLog=!0,D(o,"INFO",t,{message:s,context:e,status:200},c,r)},error:(t,s,e,g)=>{let c=g||n.store||{beforeTime:process.hrtime.bigint()};return c.hasCustomLog=!0,D(o,"ERROR",t,{message:s,context:e,status:500},c,r)},warn:(t,s,e,g)=>{let c=g||n.store||{beforeTime:process.hrtime.bigint()};return c.hasCustomLog=!0,D(o,"WARNING",t,{message:s,context:e,status:200},c,r)},debug:(t,s,e,g)=>{let c=g||n.store||{beforeTime:process.hrtime.bigint()};return c.hasCustomLog=!0,D(o,"DEBUG",t,{message:s,context:e,status:200},c,r)}};return n}function cr(r){let o=R(r);return new gr.Elysia({name:"Logixlysia"}).onStart((n)=>{if(r?.config?.showStartupMessage??!0)d(n.server,r)}).onRequest((n)=>{let t={...n.store,beforeTime:process.hrtime.bigint(),logger:o,pino:o.pino,hasCustomLog:!1};n.store=t,o.store=t}).onAfterHandle({as:"global"},({request:n,set:t,store:s})=>{let e=s;if(!e.hasCustomLog){let g=N(t.status||200);o.log("INFO",n,{status:g,message:String(t.headers?.["x-message"]||"")},e)}}).onError({as:"global"},async({request:n,error:t,set:s,store:e})=>{let g=N(s.status||500);await o.handleHttpError(n,{...t,status:g},e)})}
|
|
11
11
|
|
|
12
|
-
//# debugId=
|
|
13
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/index.ts", "../src/extensions/start-server.ts", "../src/helpers/status.ts", "../src/logger/build-log-message.ts", "../src/helpers/color-mapping.ts", "../src/helpers/duration.ts", "../src/helpers/log.ts", "../src/helpers/method.ts", "../src/helpers/path.ts", "../src/helpers/timestamp.ts", "../src/output/console.ts", "../src/output/file.ts", "../src/logger/filter.ts", "../src/logger/handle-http-error.ts", "../src/logger/create-logger.ts"],
  "sourcesContent": [
    "import { Elysia } from 'elysia'\n\nimport { startServer } from './extensions'\nimport { getStatusCode } from './helpers/status'\nimport type { HttpError, Options, Server, StoreData } from './interfaces'\nimport { createLogger } from './logger'\n\nexport default function logixlysia(options?: Options): Elysia {\n  const log = createLogger(options)\n\n  return new Elysia({\n    name: 'Logixlysia'\n  })\n    .onStart(ctx => {\n      const showStartupMessage = options?.config?.showStartupMessage ?? true\n      if (showStartupMessage) {\n        startServer(ctx.server as Server, options)\n      }\n    })\n    .onRequest(ctx => {\n      const store = {\n        ...ctx.store,\n        beforeTime: process.hrtime.bigint(),\n        logger: log,\n        hasCustomLog: false\n      }\n      ctx.store = store\n      log.store = store\n    })\n    .onAfterHandle({ as: 'global' }, ({ request, set, store }) => {\n      const storeData = store as StoreData\n\n      if (!storeData.hasCustomLog) {\n        const status = getStatusCode(set.status || 200)\n        log.log(\n          'INFO',\n          request,\n          {\n            status,\n            message: String(set.headers?.['x-message'] || '')\n          },\n          storeData\n        )\n      }\n    })\n    .onError({ as: 'global' }, ({ request, error, set, store }) => {\n      const status = getStatusCode(set.status || 500)\n      log.handleHttpError(\n        request,\n        { ...error, status } as HttpError,\n        store as StoreData\n      )\n    })\n}\n\nexport type {\n  HttpError,\n  LogData,\n  Logger,\n  LogixlysiaContext,\n  LogLevel,\n  Options,\n  RequestInfo,\n  StoreData,\n  Transport\n} from './interfaces'\nexport { createLogger, handleHttpError } from './logger'\nexport { logToTransports } from './output'\n",
    "import type { Options, Server } from '../interfaces'\n\nconst createBoxText = (text: string, width: number): string => {\n  const paddingLength = Math.max(0, (width - text.length) / 2)\n  const padding = ' '.repeat(paddingLength)\n  return `${padding}${text}${padding}`.padEnd(width)\n}\n\nexport default function startServer(config: Server, options?: Options): void {\n  const hostname = config?.hostname ?? 'localhost'\n  const port = config?.port ?? 3000\n  const protocol = config?.protocol ?? 'http'\n  const showBanner = options?.config?.startupMessageFormat !== 'simple'\n\n  if (showBanner) {\n    const title = 'Elysia with Logixlysia'\n    const message = `🦊 Elysia is running at ${protocol}://${hostname}:${port}`\n    const boxWidth = Math.max(title.length, message.length) + 4\n    const border = '─'.repeat(boxWidth)\n    const emptyLine = createBoxText('', boxWidth)\n\n    console.log(`\n      ┌${border}┐\n      │${emptyLine}│\n      │${createBoxText(title, boxWidth)}│\n      │${emptyLine}│\n      │${createBoxText(message, boxWidth)}│\n      │${emptyLine}│\n      └${border}┘\n    `)\n  } else {\n    console.log(`🦊 Elysia is running at ${protocol}://${hostname}:${port}`)\n  }\n}\n",
    "import chalk from 'chalk'\nimport { StatusMap } from 'elysia'\n\nexport function getStatusCode(status: string | number): number {\n  if (typeof status === 'number') {\n    return status\n  }\n  return (StatusMap as Record<string, number>)[status] || 500\n}\n\nexport default function statusString(\n  status: number,\n  useColors: boolean\n): string {\n  const statusStr = status.toString()\n  if (!useColors) {\n    return statusStr\n  }\n\n  if (status >= 500) {\n    return chalk.red(statusStr)\n  }\n  if (status >= 400) {\n    return chalk.yellow(statusStr)\n  }\n  if (status >= 300) {\n    return chalk.cyan(statusStr)\n  }\n  if (status >= 200) {\n    return chalk.green(statusStr)\n  }\n  return chalk.white(statusStr)\n}\n",
    "import chalk from 'chalk'\n\nimport {\n  durationString,\n  formatTimestamp,\n  logString,\n  methodString,\n  pathString,\n  statusString\n} from '../helpers'\nimport type {\n  LogComponents,\n  LogData,\n  LogLevel,\n  Options,\n  RequestInfo,\n  StoreData\n} from '../interfaces'\n\nconst defaultLogFormat =\n  '🦊 {now} {level} {duration} {method} {pathname} {status} {message} {context} {ip}'\n\nfunction shouldUseColors(useColors: boolean, options?: Options): boolean {\n  if (options?.config?.useColors !== undefined) {\n    return options.config.useColors && process.env.NO_COLOR === undefined\n  }\n  return useColors && process.env.NO_COLOR === undefined\n}\n\nexport function buildLogMessage(\n  level: LogLevel,\n  request: RequestInfo,\n  data: LogData,\n  store: StoreData,\n  options?: Options,\n  useColors = true\n): string {\n  const actuallyUseColors = shouldUseColors(useColors, options)\n  const now = new Date()\n  const components: LogComponents = {\n    now: actuallyUseColors\n      ? chalk.bgYellow(\n          chalk.black(formatTimestamp(now, options?.config?.timestamp))\n        )\n      : formatTimestamp(now, options?.config?.timestamp),\n    epoch: Math.floor(now.getTime() / 1000).toString(),\n    level: logString(level, useColors),\n    duration: durationString(store.beforeTime, useColors),\n    method: methodString(request.method, useColors),\n    pathname: pathString(request),\n    status: statusString(data.status || 200, useColors),\n    message: data.message || '',\n    context: data.context\n      ? (() => {\n          try {\n            return JSON.stringify(data.context)\n          } catch (error) {\n            return `[Error serializing context: ${error instanceof Error ? error.message : 'Unknown error'}]`\n          }\n        })()\n      : '',\n    ip:\n      options?.config?.ip && request.headers.get('x-forwarded-for')\n        ? `IP: ${request.headers.get('x-forwarded-for')}`\n        : ''\n  }\n\n  const logFormat = options?.config?.customLogFormat || defaultLogFormat\n\n  return logFormat.replace(/{(\\w+)}/g, (_, key: string) => {\n    if (key in components) {\n      return components[key as keyof LogComponents] || ''\n    }\n    return ''\n  })\n}\n",
    "import chalk from 'chalk'\n\nimport type { ColorMap } from '../interfaces'\n\nexport const LogLevelColorMap: ColorMap = {\n  INFO: chalk.bgGreen.black,\n  WARNING: chalk.bgYellow.black,\n  ERROR: chalk.bgRed.black\n}\n\nexport const HttpMethodColorMap: ColorMap = {\n  GET: chalk.green,\n  POST: chalk.yellow,\n  PUT: chalk.blue,\n  PATCH: chalk.magentaBright,\n  DELETE: chalk.red,\n  HEAD: chalk.cyan,\n  OPTIONS: chalk.magenta\n}\n",
    "import chalk from 'chalk'\n\nconst timeUnits = [\n  { unit: 's', threshold: 1e9, decimalPlaces: 2 },\n  { unit: 'ms', threshold: 1e6, decimalPlaces: 0 },\n  { unit: 'µs', threshold: 1e3, decimalPlaces: 0 },\n  { unit: 'ns', threshold: 1, decimalPlaces: 0 }\n]\n\nexport default function durationString(\n  beforeTime: bigint,\n  useColors: boolean\n): string {\n  const nanoseconds = Number(process.hrtime.bigint() - beforeTime)\n\n  for (const { unit, threshold, decimalPlaces } of timeUnits) {\n    if (nanoseconds >= threshold) {\n      const value = (nanoseconds / threshold).toFixed(decimalPlaces)\n      const timeStr = `${value}${unit}`.padStart(8).padEnd(16)\n      return useColors ? chalk.gray(timeStr) : timeStr\n    }\n  }\n\n  return useColors\n    ? chalk.gray('0ns'.padStart(8).padEnd(16))\n    : '0ns'.padStart(8).padEnd(16)\n}\n",
    "import type { LogLevel } from '../interfaces'\nimport { LogLevelColorMap } from './color-mapping'\n\nexport default function logString(level: LogLevel, useColors: boolean): string {\n  const levelStr = level.toUpperCase()\n  return useColors\n    ? LogLevelColorMap[levelStr]?.(levelStr.padEnd(7)) || levelStr\n    : levelStr.padEnd(7)\n}\n",
    "import { HttpMethodColorMap } from './color-mapping'\n\nexport default function methodString(\n  method: string,\n  useColors: boolean\n): string {\n  const colorFunction = HttpMethodColorMap[method]\n  return useColors && colorFunction\n    ? colorFunction(method.padEnd(7))\n    : method.padEnd(7)\n}\n",
    "import type { RequestInfo } from '../interfaces'\n\nexport default function pathString(\n  requestInfo: RequestInfo\n): string | undefined {\n  try {\n    return new URL(requestInfo.url).pathname\n  } catch {\n    return\n  }\n}\n",
    "import type { TimestampConfig } from '../interfaces'\n\n// const DEFAULT_TIMESTAMP_FORMAT = 'yyyy-mm-dd HH:MM:ss'\nconst SYS_TIME = 'SYS:STANDARD'\n\nconst pad = (n: number): string => n.toString().padStart(2, '0')\n\nfunction formatSystemTime(date: Date): string {\n  const year = date.getFullYear()\n  const month = pad(date.getMonth() + 1)\n  const day = pad(date.getDate())\n  const hours = pad(date.getHours())\n  const minutes = pad(date.getMinutes())\n  const seconds = pad(date.getSeconds())\n  const ms = date.getMilliseconds().toString().padStart(3, '0')\n\n  return `${year}-${month}-${day} ${hours}:${minutes}:${seconds}.${ms}`\n}\n\nfunction formatCustomTime(date: Date, format: string): string {\n  const tokens: { [key: string]: string | number } = {\n    yyyy: date.getFullYear(),\n    yy: date.getFullYear().toString().slice(-2),\n    mm: pad(date.getMonth() + 1),\n    dd: pad(date.getDate()),\n    HH: pad(date.getHours()),\n    MM: pad(date.getMinutes()),\n    ss: pad(date.getSeconds()),\n    SSS: pad(date.getMilliseconds()),\n    Z: -date.getTimezoneOffset() / 60\n  }\n\n  return format.replace(/yyyy|yy|mm|dd|HH|MM|ss|SSS|Z/g, match =>\n    (tokens[match] ?? '').toString()\n  )\n}\n\nexport function formatTimestamp(date: Date, config?: TimestampConfig): string {\n  if (!config?.translateTime) {\n    return date.toISOString()\n  }\n\n  if (config.translateTime === true || config.translateTime === SYS_TIME) {\n    return formatSystemTime(date)\n  }\n\n  return formatCustomTime(date, config.translateTime)\n}\n",
    "import type {\n  LogData,\n  LogLevel,\n  Options,\n  RequestInfo,\n  StoreData\n} from '../interfaces'\nimport { buildLogMessage } from '../logger/build-log-message'\n\nexport async function logToTransports(\n  level: LogLevel,\n  request: RequestInfo,\n  data: LogData,\n  store: StoreData,\n  options?: Options\n): Promise<void> {\n  if (!options?.config?.transports || options.config.transports.length === 0) {\n    return\n  }\n\n  const message = buildLogMessage(level, request, data, store, options, false)\n\n  const promises = options.config.transports.map(transport =>\n    transport.log(level, message, { request, data, store })\n  )\n\n  await Promise.all(promises)\n}\n",
    "import { promises as fs } from 'node:fs'\nimport { dirname } from 'node:path'\n\nimport type {\n  LogData,\n  LogLevel,\n  Options,\n  RequestInfo,\n  StoreData\n} from '../interfaces'\nimport { buildLogMessage } from '../logger/build-log-message'\n\nconst dirCache = new Set<string>()\n\nasync function ensureDirectoryExists(filePath: string): Promise<void> {\n  const dir = dirname(filePath)\n  if (!dirCache.has(dir)) {\n    await fs.mkdir(dir, { recursive: true })\n    dirCache.add(dir)\n  }\n}\n\nexport async function logToFile(\n  filePath: string,\n  level: LogLevel,\n  request: RequestInfo,\n  data: LogData,\n  store: StoreData,\n  options?: Options\n): Promise<void> {\n  await ensureDirectoryExists(filePath)\n  const logMessage = `${buildLogMessage(level, request, data, store, options, false)}\\n`\n  await fs.appendFile(filePath, logMessage, { flag: 'a' })\n}\n",
    "import type { LogLevel, Options } from '../interfaces'\n\nconst checkFilter = (filterValue: unknown, value: unknown) =>\n  Array.isArray(filterValue)\n    ? filterValue.includes(value)\n    : filterValue === value\n\nexport function filterLog(\n  logLevel: LogLevel,\n  status: number,\n  method: string,\n  options?: Options\n): boolean {\n  const filter = options?.config?.logFilter\n  if (!filter) {\n    return true\n  }\n\n  return (\n    (!filter.level || checkFilter(filter.level, logLevel)) &&\n    (!filter.status || checkFilter(filter.status, status)) &&\n    (!filter.method || checkFilter(filter.method, method))\n  )\n}\n",
    "import type { HttpError, Options, RequestInfo, StoreData } from '../interfaces'\nimport { logToFile } from '../output'\nimport { buildLogMessage } from './build-log-message'\n\nexport function handleHttpError(\n  request: RequestInfo,\n  error: HttpError,\n  store: StoreData,\n  options?: Options\n): void {\n  const statusCode = error.status || 500\n  console.error(\n    buildLogMessage('ERROR', request, { status: statusCode }, store, options)\n  )\n\n  const promises: Promise<void>[] = []\n\n  if (options?.config?.logFilePath) {\n    promises.push(\n      logToFile(\n        options.config.logFilePath,\n        'ERROR',\n        request,\n        { status: statusCode },\n        store,\n        options\n      )\n    )\n  }\n}\n",
    "import type {\n  LogData,\n  Logger,\n  LogLevel,\n  Options,\n  RequestInfo,\n  StoreData\n} from '../interfaces'\nimport { logToFile, logToTransports } from '../output'\nimport { buildLogMessage } from './build-log-message'\nimport { filterLog } from './filter'\nimport { handleHttpError } from './handle-http-error'\n\nfunction getMetrics(): LogData['metrics'] {\n  const memoryUsage = process.memoryUsage().heapUsed / 1024 / 1024 // MB\n  const cpuUsage = process.cpuUsage()\n\n  return {\n    memoryUsage,\n    cpuUsage: cpuUsage.user / 1_000_000 // convert to seconds\n  }\n}\n\nasync function log(\n  level: LogLevel,\n  request: RequestInfo,\n  data: LogData,\n  store: StoreData,\n  options?: Options\n): Promise<void> {\n  if (!filterLog(level, data.status || 200, request.method, options)) {\n    return\n  }\n\n  // Add metrics if not provided\n  if (!data.metrics) {\n    data.metrics = getMetrics()\n  }\n\n  // Add stack trace for errors\n  if (level === 'ERROR' && !data.stack) {\n    data.stack = new Error(`Error: ${data.message || 'Unknown error'}`).stack\n  }\n\n  const logMessage = buildLogMessage(level, request, data, store, options, true)\n  console.log(logMessage)\n\n  const promises: Promise<void>[] = []\n\n  if (options?.config?.logFilePath) {\n    promises.push(\n      logToFile(\n        options.config.logFilePath,\n        level,\n        request,\n        data,\n        store,\n        options\n      )\n    )\n  }\n\n  if (options?.config?.transports?.length) {\n    promises.push(logToTransports(level, request, data, store, options))\n  }\n\n  await Promise.all(promises)\n}\n\nexport function createLogger(options?: Options): Logger {\n  const logger: Logger = {\n    store: undefined,\n    log: (level, request, data, store) =>\n      log(level, request, data, store, options),\n    handleHttpError: (request, error, store) =>\n      handleHttpError(request, error, store, options),\n    customLogFormat: options?.config?.customLogFormat,\n    info: (request, message, context, store) => {\n      const storeData = store ||\n        logger.store || { beforeTime: process.hrtime.bigint() }\n      storeData.hasCustomLog = true\n      return log(\n        'INFO',\n        request,\n        { message, context, status: 200 },\n        storeData,\n        options\n      )\n    },\n    error: (request, message, context, store) => {\n      const storeData = store ||\n        logger.store || { beforeTime: process.hrtime.bigint() }\n      storeData.hasCustomLog = true\n      return log(\n        'ERROR',\n        request,\n        { message, context, status: 500 },\n        storeData,\n        options\n      )\n    },\n    warn: (request, message, context, store) => {\n      const storeData = store ||\n        logger.store || { beforeTime: process.hrtime.bigint() }\n      storeData.hasCustomLog = true\n      return log(\n        'WARNING',\n        request,\n        { message, context, status: 200 },\n        storeData,\n        options\n      )\n    },\n    debug: (request, message, context, store) => {\n      const storeData = store ||\n        logger.store || { beforeTime: process.hrtime.bigint() }\n      storeData.hasCustomLog = true\n      return log(\n        'DEBUG',\n        request,\n        { message, context, status: 200 },\n        storeData,\n        options\n      )\n    }\n  }\n  return logger\n}\n"
  ],
  "mappings": "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",
  "debugId": "5019F142C76DE99264756E2164756E21",
  "names": []
}
|
|
12
|
+
//# debugId=4E8C290D9DF35B1E64756E2164756E21
|
|
13
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["src/index.ts", "src/extensions/start-server.ts", "src/helpers/status.ts", "src/logger/create-logger.ts", "src/logger/build-log-message.ts", "src/helpers/color-mapping.ts", "src/helpers/duration.ts", "src/helpers/log.ts", "src/helpers/method.ts", "src/helpers/path.ts", "src/helpers/timestamp.ts", "src/output/console.ts", "src/output/file.ts", "src/utils/rotation.ts", "src/output/rotation-manager.ts", "src/logger/filter.ts", "src/logger/handle-http-error.ts"],
  "sourcesContent": [
    "import { Elysia } from 'elysia'\n\nimport { startServer } from './extensions'\nimport { getStatusCode } from './helpers/status'\nimport type { HttpError, Options, Server, StoreData } from './interfaces'\nimport { createLogger } from './logger'\n\nexport default function logixlysia(options?: Options): Elysia {\n  const log = createLogger(options)\n\n  return new Elysia({\n    name: 'Logixlysia'\n  })\n    .onStart(ctx => {\n      const showStartupMessage = options?.config?.showStartupMessage ?? true\n      if (showStartupMessage) {\n        startServer(ctx.server as Server, options)\n      }\n    })\n    .onRequest(ctx => {\n      const store = {\n        ...ctx.store,\n        beforeTime: process.hrtime.bigint(),\n        logger: log,\n        pino: log.pino, // Expose Pino logger directly\n        hasCustomLog: false\n      }\n      ctx.store = store\n      log.store = store\n    })\n    .onAfterHandle({ as: 'global' }, ({ request, set, store }) => {\n      const storeData = store as StoreData\n\n      if (!storeData.hasCustomLog) {\n        const status = getStatusCode(set.status || 200)\n        log.log(\n          'INFO',\n          request,\n          {\n            status,\n            message: String(set.headers?.['x-message'] || '')\n          },\n          storeData\n        )\n      }\n    })\n    .onError({ as: 'global' }, async ({ request, error, set, store }) => {\n      const status = getStatusCode(set.status || 500)\n      await log.handleHttpError(\n        request,\n        { ...error, status } as HttpError,\n        store as StoreData\n      )\n    })\n}\n\nexport type {\n  HttpError,\n  LogData,\n  Logger,\n  LogixlysiaContext,\n  LogLevel,\n  LogRotationConfig,\n  Options,\n  RequestInfo,\n  StoreData,\n  Transport\n} from './interfaces'\nexport { createLogger, handleHttpError } from './logger'\nexport { logToTransports } from './output'\n",
    "import type { Options, Server } from '../interfaces'\n\nconst createBoxText = (text: string, width: number): string => {\n  const paddingLength = Math.max(0, (width - text.length) / 2)\n  const padding = ' '.repeat(paddingLength)\n  return `${padding}${text}${padding}`.padEnd(width)\n}\n\nexport default function startServer(config: Server, options?: Options): void {\n  const hostname = config?.hostname ?? 'localhost'\n  const port = config?.port ?? 3000\n  const protocol = config?.protocol ?? 'http'\n  const showBanner = options?.config?.startupMessageFormat !== 'simple'\n\n  if (showBanner) {\n    const title = 'Elysia with Logixlysia'\n    const message = `🦊 Elysia is running at ${protocol}://${hostname}:${port}`\n    const boxWidth = Math.max(title.length, message.length) + 4\n    const border = '─'.repeat(boxWidth)\n    const emptyLine = createBoxText('', boxWidth)\n\n    console.log(`\n      ┌${border}┐\n      │${emptyLine}│\n      │${createBoxText(title, boxWidth)}│\n      │${emptyLine}│\n      │${createBoxText(message, boxWidth)}│\n      │${emptyLine}│\n      └${border}┘\n    `)\n  } else {\n    console.log(`🦊 Elysia is running at ${protocol}://${hostname}:${port}`)\n  }\n}\n",
    "import chalk from 'chalk'\nimport { StatusMap } from 'elysia'\n\nexport function getStatusCode(status: string | number): number {\n  if (typeof status === 'number') {\n    return status\n  }\n  return (StatusMap as Record<string, number>)[status] || 500\n}\n\nexport default function statusString(\n  status: number,\n  useColors: boolean\n): string {\n  const statusStr = status.toString()\n  if (!useColors) {\n    return statusStr\n  }\n\n  if (status >= 500) {\n    return chalk.red(statusStr)\n  }\n  if (status >= 400) {\n    return chalk.yellow(statusStr)\n  }\n  if (status >= 300) {\n    return chalk.cyan(statusStr)\n  }\n  if (status >= 200) {\n    return chalk.green(statusStr)\n  }\n  return chalk.white(statusStr)\n}\n",
    "import type { Logger as PinoLogger } from 'pino'\nimport pino from 'pino'\nimport type {\n  LogData,\n  Logger,\n  LogLevel,\n  Options,\n  PinoConfig,\n  RequestInfo,\n  StoreData\n} from '../interfaces'\nimport { logToFile, logToTransports } from '../output'\nimport { buildLogMessage } from './build-log-message'\nimport { filterLog } from './filter'\nimport { handleHttpError } from './handle-http-error'\n\nfunction getMetrics(): LogData['metrics'] {\n  const memoryUsage = process.memoryUsage().heapUsed / 1024 / 1024 // MB\n  const cpuUsage = process.cpuUsage()\n\n  return {\n    memoryUsage,\n    cpuUsage: cpuUsage.user / 1_000_000 // convert to seconds\n  }\n}\n\nfunction buildPinoConfig(pinoConfig: PinoConfig) {\n  return {\n    level: pinoConfig.level || 'info',\n    timestamp: pinoConfig.timestamp ?? true,\n    messageKey: pinoConfig.messageKey || 'msg',\n    errorKey: pinoConfig.errorKey || 'err',\n    base: pinoConfig.base || { pid: process.pid }\n  } as const\n}\n\nfunction createPrettyTransport(prettyPrint: boolean | object) {\n  return pino.transport({\n    target: 'pino-pretty',\n    options: {\n      colorize: true,\n      translateTime: 'HH:MM:ss Z',\n      ignore: 'pid,hostname',\n      ...(typeof prettyPrint === 'object' ? prettyPrint : {})\n    }\n  })\n}\n\nfunction createPinoInstance(options?: Options): PinoLogger {\n  const pinoConfig = options?.config?.pino || {}\n  const { prettyPrint, transport, ...rest } = pinoConfig\n  const config = {\n    ...buildPinoConfig(pinoConfig),\n    ...rest\n  }\n\n  if (prettyPrint && process.env.NODE_ENV !== 'production') {\n    return pino(config, createPrettyTransport(prettyPrint))\n  }\n\n  if (transport) {\n    if (typeof transport === 'object' && 'target' in transport) {\n      return pino(\n        config,\n        pino.transport(\n          transport as pino.TransportSingleOptions | pino.TransportMultiOptions\n        )\n      )\n    }\n    console.warn(\n      'Invalid transport configuration provided, falling back to default'\n    )\n  }\n\n  return pino(config)\n}\n\nfunction mapLogLevelToPino(level: LogLevel): string {\n  switch (level.toUpperCase()) {\n    case 'DEBUG':\n      return 'debug'\n    case 'INFO':\n      return 'info'\n    case 'WARNING':\n    case 'WARN':\n      return 'warn'\n    case 'ERROR':\n      return 'error'\n    default:\n      console.warn(`Unknown log level \"${level}\", defaulting to \"info\"`)\n      return 'info'\n  }\n}\n\nfunction emitPinoLog(\n  pinoLogger: PinoLogger,\n  level: LogLevel,\n  logObject: Record<string, unknown>,\n  message: string,\n  options?: Options\n): void {\n  const pinoLevel = mapLogLevelToPino(level)\n  const logMethod = pinoLogger[pinoLevel as keyof PinoLogger] as (\n    ...args: unknown[]\n  ) => void\n  const hasCustomPinoOutput = Boolean(\n    options?.config?.pino?.transport || options?.config?.pino?.prettyPrint\n  )\n  const shouldEmitPino =\n    !options?.config?.useTransportsOnly || hasCustomPinoOutput\n\n  if (shouldEmitPino && typeof logMethod === 'function') {\n    logMethod.call(pinoLogger, logObject, message)\n  }\n}\n\nasync function handleOutputs(\n  level: LogLevel,\n  request: RequestInfo,\n  data: LogData,\n  store: StoreData,\n  options?: Options,\n  logMessage?: string\n): Promise<void> {\n  const promises: Promise<void>[] = []\n\n  // Handle console logging\n  if (\n    !(\n      options?.config?.useTransportsOnly ||\n      options?.config?.disableInternalLogger\n    )\n  ) {\n    console.log(logMessage)\n  }\n\n  // Handle file logging\n  if (\n    !options?.config?.useTransportsOnly &&\n    options?.config?.logFilePath &&\n    !options?.config?.disableFileLogging\n  ) {\n    promises.push(\n      logToFile(\n        options.config.logFilePath,\n        level,\n        request,\n        data,\n        store,\n        options\n      )\n    )\n  }\n\n  // Handle transport logging\n  if (options?.config?.transports?.length) {\n    promises.push(logToTransports(level, request, data, store, options))\n  }\n\n  await Promise.all(promises)\n}\n\nasync function log(\n  pinoLogger: PinoLogger,\n  level: LogLevel,\n  request: RequestInfo,\n  data: LogData,\n  store: StoreData,\n  options?: Options\n): Promise<void> {\n  if (!filterLog(level, data.status || 200, request.method, options)) {\n    return\n  }\n\n  if (!data.metrics) {\n    data.metrics = getMetrics()\n  }\n\n  if (level === 'ERROR' && !data.stack) {\n    const err = new Error(data.message || 'Unknown error')\n    data.stack = err.stack\n  }\n\n  const errorKey = options?.config?.pino?.errorKey || 'err'\n  const err =\n    level === 'ERROR' && data.stack\n      ? {\n          name: 'Error',\n          message: data.message || 'Unknown error',\n          stack: data.stack\n        }\n      : undefined\n\n  const forwardedFor = request.headers.get('x-forwarded-for')\n  const clientIp =\n    options?.config?.ip && forwardedFor\n      ? forwardedFor.split(',')[0]?.trim()\n      : undefined\n\n  const logObject = {\n    method: request.method,\n    url: request.url,\n    status: data.status,\n    message: data.message,\n    context: data.context,\n    metrics: data.metrics,\n    duration: Number(process.hrtime.bigint() - store.beforeTime) / 1_000_000,\n    ip: clientIp,\n    [errorKey]: err\n  }\n\n  emitPinoLog(\n    pinoLogger,\n    level,\n    logObject,\n    data.message || 'Request processed',\n    options\n  )\n\n  const logMessage = buildLogMessage(level, request, data, store, options, true)\n\n  await handleOutputs(level, request, data, store, options, logMessage)\n}\n\nexport function createLogger(options?: Options): Logger {\n  const pinoLogger = createPinoInstance(options)\n\n  const logger: Logger = {\n    store: undefined,\n    pino: pinoLogger, // Expose the Pino instance\n    log: (level, request, data, store) =>\n      log(pinoLogger, level, request, data, store, options),\n    handleHttpError: async (request, error, store) =>\n      await handleHttpError(request, error, store, options),\n    customLogFormat: options?.config?.customLogFormat,\n    info: (request, message, context, store) => {\n      const storeData = store ||\n        logger.store || { beforeTime: process.hrtime.bigint() }\n      storeData.hasCustomLog = true\n      return log(\n        pinoLogger,\n        'INFO',\n        request,\n        { message, context, status: 200 },\n        storeData,\n        options\n      )\n    },\n    error: (request, message, context, store) => {\n      const storeData = store ||\n        logger.store || { beforeTime: process.hrtime.bigint() }\n      storeData.hasCustomLog = true\n      return log(\n        pinoLogger,\n        'ERROR',\n        request,\n        { message, context, status: 500 },\n        storeData,\n        options\n      )\n    },\n    warn: (request, message, context, store) => {\n      const storeData = store ||\n        logger.store || { beforeTime: process.hrtime.bigint() }\n      storeData.hasCustomLog = true\n      return log(\n        pinoLogger,\n        'WARNING',\n        request,\n        { message, context, status: 200 },\n        storeData,\n        options\n      )\n    },\n    debug: (request, message, context, store) => {\n      const storeData = store ||\n        logger.store || { beforeTime: process.hrtime.bigint() }\n      storeData.hasCustomLog = true\n      return log(\n        pinoLogger,\n        'DEBUG',\n        request,\n        { message, context, status: 200 },\n        storeData,\n        options\n      )\n    }\n  }\n  return logger\n}\n",
    "import chalk from 'chalk'\n\nimport {\n  durationString,\n  formatTimestamp,\n  logString,\n  methodString,\n  pathString,\n  statusString\n} from '../helpers'\nimport type {\n  LogComponents,\n  LogData,\n  LogLevel,\n  Options,\n  RequestInfo,\n  StoreData\n} from '../interfaces'\n\nconst defaultLogFormat =\n  '🦊 {now} {level} {duration} {method} {pathname} {status} {message} {context} {ip}'\n\nfunction shouldUseColors(useColors: boolean, options?: Options): boolean {\n  if (options?.config?.useColors !== undefined) {\n    return options.config.useColors && process.env.NO_COLOR === undefined\n  }\n  return useColors && process.env.NO_COLOR === undefined\n}\n\nexport function buildLogMessage(\n  level: LogLevel,\n  request: RequestInfo,\n  data: LogData,\n  store: StoreData,\n  options?: Options,\n  useColors = true\n): string {\n  const actuallyUseColors = shouldUseColors(useColors, options)\n  const now = new Date()\n  const components: LogComponents = {\n    now: actuallyUseColors\n      ? chalk.bgYellow(\n          chalk.black(formatTimestamp(now, options?.config?.timestamp))\n        )\n      : formatTimestamp(now, options?.config?.timestamp),\n    epoch: Math.floor(now.getTime() / 1000).toString(),\n    level: logString(level, useColors),\n    duration: durationString(store.beforeTime, useColors),\n    method: methodString(request.method, useColors),\n    pathname: pathString(request),\n    status: statusString(data.status || 200, useColors),\n    message: data.message || '',\n    context: data.context\n      ? (() => {\n          try {\n            return JSON.stringify(data.context)\n          } catch (error) {\n            return `[Error serializing context: ${error instanceof Error ? error.message : 'Unknown error'}]`\n          }\n        })()\n      : '',\n    ip:\n      options?.config?.ip && request.headers.get('x-forwarded-for')\n        ? `IP: ${request.headers.get('x-forwarded-for')}`\n        : ''\n  }\n\n  const logFormat = options?.config?.customLogFormat || defaultLogFormat\n\n  return logFormat.replace(/{(\\w+)}/g, (_, key: string) => {\n    if (key in components) {\n      return components[key as keyof LogComponents] || ''\n    }\n    return ''\n  })\n}\n",
    "import chalk from 'chalk'\n\nimport type { ColorMap } from '../interfaces'\n\nexport const LogLevelColorMap: ColorMap = {\n  INFO: chalk.bgGreen.black,\n  WARNING: chalk.bgYellow.black,\n  ERROR: chalk.bgRed.black\n}\n\nexport const HttpMethodColorMap: ColorMap = {\n  GET: chalk.green,\n  POST: chalk.yellow,\n  PUT: chalk.blue,\n  PATCH: chalk.magentaBright,\n  DELETE: chalk.red,\n  HEAD: chalk.cyan,\n  OPTIONS: chalk.magenta\n}\n",
    "import chalk from 'chalk'\n\nconst timeUnits = [\n  { unit: 's', threshold: 1e9, decimalPlaces: 2 },\n  { unit: 'ms', threshold: 1e6, decimalPlaces: 0 },\n  { unit: 'µs', threshold: 1e3, decimalPlaces: 0 },\n  { unit: 'ns', threshold: 1, decimalPlaces: 0 }\n]\n\nexport default function durationString(\n  beforeTime: bigint,\n  useColors: boolean\n): string {\n  const nanoseconds = Number(process.hrtime.bigint() - beforeTime)\n\n  for (const { unit, threshold, decimalPlaces } of timeUnits) {\n    if (nanoseconds >= threshold) {\n      const value = (nanoseconds / threshold).toFixed(decimalPlaces)\n      const timeStr = `${value}${unit}`.padStart(8).padEnd(16)\n      return useColors ? chalk.gray(timeStr) : timeStr\n    }\n  }\n\n  return useColors\n    ? chalk.gray('0ns'.padStart(8).padEnd(16))\n    : '0ns'.padStart(8).padEnd(16)\n}\n",
    "import type { LogLevel } from '../interfaces'\nimport { LogLevelColorMap } from './color-mapping'\n\nexport default function logString(level: LogLevel, useColors: boolean): string {\n  const levelStr = level.toUpperCase()\n  return useColors\n    ? LogLevelColorMap[levelStr]?.(levelStr.padEnd(7)) || levelStr\n    : levelStr.padEnd(7)\n}\n",
    "import { HttpMethodColorMap } from './color-mapping'\n\nexport default function methodString(\n  method: string,\n  useColors: boolean\n): string {\n  const colorFunction = HttpMethodColorMap[method]\n  return useColors && colorFunction\n    ? colorFunction(method.padEnd(7))\n    : method.padEnd(7)\n}\n",
    "import type { RequestInfo } from '../interfaces'\n\nexport default function pathString(\n  requestInfo: RequestInfo\n): string | undefined {\n  try {\n    return new URL(requestInfo.url).pathname\n  } catch {\n    return\n  }\n}\n",
    "import type { TimestampConfig } from '../interfaces'\n\n// const DEFAULT_TIMESTAMP_FORMAT = 'yyyy-mm-dd HH:MM:ss'\nconst SYS_TIME = 'SYS:STANDARD'\n\nconst pad = (n: number): string => n.toString().padStart(2, '0')\n\nfunction formatSystemTime(date: Date): string {\n  const year = date.getFullYear()\n  const month = pad(date.getMonth() + 1)\n  const day = pad(date.getDate())\n  const hours = pad(date.getHours())\n  const minutes = pad(date.getMinutes())\n  const seconds = pad(date.getSeconds())\n  const ms = date.getMilliseconds().toString().padStart(3, '0')\n\n  return `${year}-${month}-${day} ${hours}:${minutes}:${seconds}.${ms}`\n}\n\nfunction formatCustomTime(date: Date, format: string): string {\n  const tokens: { [key: string]: string | number } = {\n    yyyy: date.getFullYear(),\n    yy: date.getFullYear().toString().slice(-2),\n    mm: pad(date.getMonth() + 1),\n    dd: pad(date.getDate()),\n    HH: pad(date.getHours()),\n    MM: pad(date.getMinutes()),\n    ss: pad(date.getSeconds()),\n    SSS: pad(date.getMilliseconds()),\n    Z: -date.getTimezoneOffset() / 60\n  }\n\n  return format.replace(/yyyy|yy|mm|dd|HH|MM|ss|SSS|Z/g, match =>\n    (tokens[match] ?? '').toString()\n  )\n}\n\nexport function formatTimestamp(date: Date, config?: TimestampConfig): string {\n  if (!config?.translateTime) {\n    return date.toISOString()\n  }\n\n  if (config.translateTime === true || config.translateTime === SYS_TIME) {\n    return formatSystemTime(date)\n  }\n\n  return formatCustomTime(date, config.translateTime)\n}\n",
    "import type {\n  LogData,\n  LogLevel,\n  Options,\n  RequestInfo,\n  StoreData\n} from '../interfaces'\nimport { buildLogMessage } from '../logger/build-log-message'\n\nexport async function logToTransports(\n  level: LogLevel,\n  request: RequestInfo,\n  data: LogData,\n  store: StoreData,\n  options?: Options\n): Promise<void> {\n  if (!options?.config?.transports || options.config.transports.length === 0) {\n    return\n  }\n\n  const message = buildLogMessage(level, request, data, store, options, false)\n\n  const promises = options.config.transports.map(transport =>\n    transport.log(level, message, { request, data, store })\n  )\n\n  await Promise.all(promises)\n}\n",
    "import { promises as fs } from 'node:fs'\nimport { dirname } from 'node:path'\n\nimport type {\n  LogData,\n  LogLevel,\n  Options,\n  RequestInfo,\n  StoreData\n} from '../interfaces'\nimport { buildLogMessage } from '../logger/build-log-message'\nimport { parseInterval, shouldRotateByTime } from '../utils/rotation'\nimport { performRotation, shouldRotate } from './rotation-manager'\n\nconst dirCache = new Set<string>()\n\nasync function ensureDirectoryExists(filePath: string): Promise<void> {\n  const dir = dirname(filePath)\n  if (!dirCache.has(dir)) {\n    await fs.mkdir(dir, { recursive: true })\n    dirCache.add(dir)\n  }\n}\n\n/**\n * Check if rotation is needed and perform it\n */\nasync function checkAndRotate(\n  filePath: string,\n  options?: Options\n): Promise<void> {\n  const rotationConfig = options?.config?.logRotation\n  if (!rotationConfig) {\n    return\n  }\n\n  let needsRotation = false\n\n  // Check size-based rotation\n  if (rotationConfig.maxSize) {\n    needsRotation = await shouldRotate(filePath, rotationConfig)\n  }\n\n  // Check time-based rotation\n  if (!needsRotation && rotationConfig.interval) {\n    try {\n      const intervalMs = parseInterval(rotationConfig.interval)\n      needsRotation = await shouldRotateByTime(filePath, intervalMs)\n    } catch (error) {\n      console.error('Invalid interval format in log rotation config:', error)\n    }\n  }\n\n  // Perform rotation if needed\n  if (needsRotation) {\n    await performRotation(filePath, rotationConfig)\n  }\n}\n\nexport async function logToFile(\n  filePath: string,\n  level: LogLevel,\n  request: RequestInfo,\n  data: LogData,\n  store: StoreData,\n  options?: Options\n): Promise<void> {\n  await ensureDirectoryExists(filePath)\n\n  // Check and perform rotation if needed\n  await checkAndRotate(filePath, options)\n\n  const logMessage = `${buildLogMessage(level, request, data, store, options, false)}\\n`\n  await fs.appendFile(filePath, logMessage, { flag: 'a' })\n}\n",
    "import { promises as fs } from 'node:fs'\nimport { basename, dirname, join } from 'node:path'\n\nexport interface ParsedRetention {\n  type: 'count' | 'time'\n  value: number\n}\n\n// Regex patterns defined at top level for performance\nconst SIZE_PATTERN = /^(\\d+(?:\\.\\d+)?)\\s*([kmg])?b?$/\nconst INTERVAL_PATTERN = /^(\\d+)\\s*([hdw])$/\nconst RETENTION_PATTERN = /^(\\d+)\\s*d$/\n\n/**\n * Parse size string to bytes\n * Supports: '10m', '1g', '100k', or raw number\n */\nexport function parseSize(size: string | number): number {\n  if (typeof size === 'number') {\n    return size\n  }\n\n  const units: Record<string, number> = {\n    k: 1024,\n    m: 1024 * 1024,\n    g: 1024 * 1024 * 1024\n  }\n\n  const match = size.toLowerCase().match(SIZE_PATTERN)\n  if (!match?.[1]) {\n    throw new Error(`Invalid size format: ${size}`)\n  }\n\n  const value = Number.parseFloat(match[1])\n  const unit = match[2] ?? ''\n\n  return Math.floor(value * (units[unit] ?? 1))\n}\n\n/**\n * Parse interval string to milliseconds\n * Supports: '1h', '1d', '1w'\n */\nexport function parseInterval(interval: string): number {\n  const units: Record<string, number> = {\n    h: 60 * 60 * 1000, // hour\n    d: 24 * 60 * 60 * 1000, // day\n    w: 7 * 24 * 60 * 60 * 1000 // week\n  }\n\n  const match = interval.toLowerCase().match(INTERVAL_PATTERN)\n  if (!match) {\n    throw new Error(`Invalid interval format: ${interval}`)\n  }\n\n  const valueStr = match[1]\n  const unit = match[2] as string\n\n  if (!valueStr) {\n    throw new Error(`Invalid interval format: ${interval}`)\n  }\n  if (!unit) {\n    throw new Error(`Invalid interval format: ${interval}`)\n  }\n\n  const value = Number.parseInt(valueStr, 10)\n  return value * (units[unit] ?? 0)\n}\n\n/**\n * Parse retention string or number\n * Returns object with type (count or time) and value\n */\nexport function parseRetention(retention: string | number): ParsedRetention {\n  if (typeof retention === 'number') {\n    return { type: 'count', value: retention }\n  }\n\n  // Check if it's a time-based retention (e.g., '7d', '30d')\n  const match = retention.toLowerCase().match(RETENTION_PATTERN)\n  if (match?.[1]) {\n    const days = Number.parseInt(match[1], 10)\n    return { type: 'time', value: days * 24 * 60 * 60 * 1000 } // convert to milliseconds\n  }\n\n  throw new Error(`Invalid retention format: ${retention}`)\n}\n\n/**\n * Check if file should be rotated based on size\n */\nexport async function shouldRotateBySize(\n  filePath: string,\n  maxSize: number\n): Promise<boolean> {\n  try {\n    const stats = await fs.stat(filePath)\n    return stats.size >= maxSize\n  } catch {\n    // File doesn't exist or can't be accessed\n    return false\n  }\n}\n\n/**\n * Get the last rotation time for a file\n * Uses file modification time as a proxy\n */\nasync function getLastRotationTime(filePath: string): Promise<number> {\n  try {\n    const stats = await fs.stat(filePath)\n    return stats.mtime.getTime()\n  } catch {\n    // File doesn't exist, use current time\n    return Date.now()\n  }\n}\n\n// Track last rotation times in memory to avoid excessive file checks\nconst rotationTimeCache = new Map<string, number>()\n\n/**\n * Check if file should be rotated based on time interval\n */\nexport async function shouldRotateByTime(\n  filePath: string,\n  interval: number\n): Promise<boolean> {\n  const now = Date.now()\n\n  // Check cache first\n  const cachedTime = rotationTimeCache.get(filePath)\n  if (cachedTime !== undefined) {\n    return now - cachedTime >= interval\n  }\n\n  // Get last rotation time from file\n  const lastRotation = await getLastRotationTime(filePath)\n  rotationTimeCache.set(filePath, lastRotation)\n\n  return now - lastRotation >= interval\n}\n\n/**\n * Update the last rotation time in cache\n */\nexport function updateRotationTime(filePath: string): void {\n  rotationTimeCache.set(filePath, Date.now())\n}\n\n/**\n * Get rotated files for a given log file\n */\nexport async function getRotatedFiles(filePath: string): Promise<string[]> {\n  const dir = dirname(filePath)\n  const baseName = basename(filePath)\n\n  try {\n    const files = await fs.readdir(dir)\n    // Match files like: app.log.2025-10-10, app.log.2025-10-10.gz\n    const rotatedPattern = new RegExp(\n      `^${baseName.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&')}\\\\.\\\\d{4}-\\\\d{2}-\\\\d{2}(-\\\\d{2}-\\\\d{2}-\\\\d{2})?(\\\\.gz)?$`\n    )\n    const rotatedFiles = files\n      .filter(file => rotatedPattern.test(file))\n      .map(file => join(dir, file))\n\n    // Sort by modification time (newest first)\n    const filesWithStats = await Promise.all(\n      rotatedFiles.map(async file => {\n        const stats = await fs.stat(file)\n        return { file, mtime: stats.mtime.getTime() }\n      })\n    )\n\n    return filesWithStats\n      .sort((a, b) => b.mtime - a.mtime)\n      .map(item => item.file)\n  } catch {\n    return []\n  }\n}\n",
    "import { createReadStream, createWriteStream, promises as fs } from 'node:fs'\nimport { pipeline } from 'node:stream/promises'\nimport { createGzip } from 'node:zlib'\n\nimport type { LogRotationConfig } from '../interfaces'\nimport {\n  getRotatedFiles,\n  parseInterval,\n  parseRetention,\n  parseSize,\n  updateRotationTime\n} from '../utils/rotation'\n\n/**\n * Generate a rotated file name with timestamp\n * Example: app.log -> app.log.2025-10-10-14-30-45\n */\nexport function getRotatedFileName(filePath: string, timestamp: Date): string {\n  const year = timestamp.getFullYear()\n  const month = String(timestamp.getMonth() + 1).padStart(2, '0')\n  const day = String(timestamp.getDate()).padStart(2, '0')\n  const hours = String(timestamp.getHours()).padStart(2, '0')\n  const minutes = String(timestamp.getMinutes()).padStart(2, '0')\n  const seconds = String(timestamp.getSeconds()).padStart(2, '0')\n\n  return `${filePath}.${year}-${month}-${day}-${hours}-${minutes}-${seconds}`\n}\n\n/**\n * Rotate a log file by renaming it with a timestamp\n */\nexport async function rotateFile(filePath: string): Promise<string> {\n  try {\n    // Check if file exists and has content\n    const stats = await fs.stat(filePath)\n    if (stats.size === 0) {\n      // Don't rotate empty files\n      return ''\n    }\n\n    const rotatedPath = getRotatedFileName(filePath, new Date())\n    await fs.rename(filePath, rotatedPath)\n    updateRotationTime(filePath)\n\n    return rotatedPath\n  } catch (error) {\n    // File doesn't exist or can't be rotated\n    if ((error as NodeJS.ErrnoException).code !== 'ENOENT') {\n      console.error(`Failed to rotate log file ${filePath}:`, error)\n    }\n    return ''\n  }\n}\n\n/**\n * Compress a file using gzip\n */\nexport async function compressFile(filePath: string): Promise<void> {\n  try {\n    const compressedPath = `${filePath}.gz`\n    const source = createReadStream(filePath)\n    const destination = createWriteStream(compressedPath)\n    const gzip = createGzip()\n\n    await pipeline(source, gzip, destination)\n\n    // Delete the original file after successful compression\n    await fs.unlink(filePath)\n  } catch (error) {\n    console.error(`Failed to compress file ${filePath}:`, error)\n  }\n}\n\n/**\n * Clean old rotated files based on retention policy\n */\nexport async function cleanOldFiles(\n  filePath: string,\n  config: LogRotationConfig\n): Promise<void> {\n  if (!config.maxFiles) {\n    return\n  }\n\n  try {\n    const rotatedFiles = await getRotatedFiles(filePath)\n    if (rotatedFiles.length === 0) {\n      return\n    }\n\n    const retention = parseRetention(config.maxFiles)\n\n    if (retention.type === 'count') {\n      // Keep only the specified number of files\n      const filesToDelete = rotatedFiles.slice(retention.value)\n      await Promise.all(filesToDelete.map(file => fs.unlink(file)))\n    } else if (retention.type === 'time') {\n      // Delete files older than the specified time\n      const cutoffTime = Date.now() - retention.value\n      const filesToDelete = await Promise.all(\n        rotatedFiles.map(async file => {\n          const stats = await fs.stat(file)\n          return stats.mtime.getTime() < cutoffTime ? file : null\n        })\n      )\n\n      await Promise.all(\n        filesToDelete\n          .filter((file): file is string => file !== null)\n          .map(file => fs.unlink(file))\n      )\n    }\n  } catch (error) {\n    console.error(`Failed to clean old log files for ${filePath}:`, error)\n  }\n}\n\n/**\n * Perform complete rotation: rotate, compress (if enabled), and clean old files\n */\nexport async function performRotation(\n  filePath: string,\n  config: LogRotationConfig\n): Promise<void> {\n  try {\n    // Rotate the file\n    const rotatedPath = await rotateFile(filePath)\n\n    if (!rotatedPath) {\n      return\n    }\n\n    // Compress if enabled (defaults to gzip)\n    if (config.compress) {\n      await compressFile(rotatedPath)\n    }\n\n    // Clean old files\n    await cleanOldFiles(filePath, config)\n  } catch (error) {\n    console.error(`Failed to perform rotation for ${filePath}:`, error)\n  }\n}\n\n/**\n * Check if rotation is needed based on configuration\n */\nexport async function shouldRotate(\n  filePath: string,\n  config: LogRotationConfig\n): Promise<boolean> {\n  try {\n    // Check file existence\n    await fs.access(filePath)\n  } catch {\n    // File doesn't exist, no rotation needed\n    return false\n  }\n\n  // Check size-based rotation\n  if (config.maxSize) {\n    try {\n      const maxSizeBytes = parseSize(config.maxSize)\n      const stats = await fs.stat(filePath)\n      if (stats.size >= maxSizeBytes) {\n        return true\n      }\n    } catch (error) {\n      console.error(`Failed to check file size for ${filePath}:`, error)\n    }\n  }\n\n  // Check time-based rotation\n  if (config.interval) {\n    try {\n      const intervalMs = parseInterval(config.interval)\n      const stats = await fs.stat(filePath)\n      const age = Date.now() - stats.mtimeMs\n      if (age >= intervalMs) {\n        return true\n      }\n    } catch (error) {\n      // Log parse or stat errors and treat as no-op\n      console.error(`Failed to check file age for ${filePath}:`, error)\n    }\n  }\n\n  return false\n}\n",
    "import type { LogLevel, Options } from '../interfaces'\n\nconst checkFilter = (filterValue: unknown, value: unknown) =>\n  Array.isArray(filterValue)\n    ? filterValue.includes(value)\n    : filterValue === value\n\nexport function filterLog(\n  logLevel: LogLevel,\n  status: number,\n  method: string,\n  options?: Options\n): boolean {\n  const filter = options?.config?.logFilter\n  if (!filter) {\n    return true\n  }\n\n  return (\n    (!filter.level || checkFilter(filter.level, logLevel)) &&\n    (!filter.status || checkFilter(filter.status, status)) &&\n    (!filter.method || checkFilter(filter.method, method))\n  )\n}\n",
    "import type { HttpError, Options, RequestInfo, StoreData } from '../interfaces'\nimport { logToFile, logToTransports } from '../output'\nimport { buildLogMessage } from './build-log-message'\n\nexport async function handleHttpError(\n  request: RequestInfo,\n  error: HttpError,\n  store: StoreData,\n  options?: Options\n): Promise<void> {\n  const statusCode = error.status || 500\n  const logData = {\n    status: statusCode,\n    message: error.message,\n    stack: error.stack\n  }\n\n  const promises: Promise<void>[] = []\n\n  // Handle console logging\n  if (\n    !(\n      options?.config?.useTransportsOnly ||\n      options?.config?.disableInternalLogger\n    )\n  ) {\n    console.error(buildLogMessage('ERROR', request, logData, store, options))\n  }\n\n  // Handle file logging\n  if (\n    !options?.config?.useTransportsOnly &&\n    options?.config?.logFilePath &&\n    !options?.config?.disableFileLogging\n  ) {\n    promises.push(\n      logToFile(\n        options.config.logFilePath,\n        'ERROR',\n        request,\n        logData,\n        store,\n        options\n      )\n    )\n  }\n\n  // Handle transport logging\n  if (options?.config?.transports?.length) {\n    promises.push(logToTransports('ERROR', request, logData, store, options))\n  }\n\n  await Promise.all(promises)\n}\n"
  ],
  "mappings": "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",
  "debugId": "4E8C290D9DF35B1E64756E2164756E21",
  "names": []
}
|
package/dist/index.d.cts
CHANGED
|
@@ -1,4 +1,5 @@
|
|
|
1
1
|
import { Elysia } from "elysia";
|
|
2
|
+
import { LoggerOptions, Logger as PinoLogger } from "pino";
|
|
2
3
|
interface RequestInfo {
|
|
3
4
|
headers: {
|
|
4
5
|
get: (key: string) => string | null
|
|
@@ -20,8 +21,9 @@ interface LogData {
|
|
|
20
21
|
}
|
|
21
22
|
interface Logger {
|
|
22
23
|
store?: StoreData;
|
|
24
|
+
pino: PinoLogger;
|
|
23
25
|
log(level: LogLevel, request: RequestInfo, data: LogData, store: StoreData): void;
|
|
24
|
-
handleHttpError(request: RequestInfo, error: HttpError, store: StoreData): void
|
|
26
|
+
handleHttpError(request: RequestInfo, error: HttpError, store: StoreData): Promise<void>;
|
|
25
27
|
customLogFormat?: string;
|
|
26
28
|
info(request: RequestInfo, message: string, context?: Record<string, string | number | boolean | null>, store?: StoreData): void;
|
|
27
29
|
error(request: RequestInfo, message: string, context?: Record<string, string | number | boolean | null>, store?: StoreData): void;
|
|
@@ -31,11 +33,13 @@ interface Logger {
|
|
|
31
33
|
interface StoreData {
|
|
32
34
|
beforeTime: bigint;
|
|
33
35
|
logger?: Logger;
|
|
36
|
+
pino?: PinoLogger;
|
|
34
37
|
hasCustomLog?: boolean;
|
|
35
38
|
}
|
|
36
39
|
interface LogixlysiaContext {
|
|
37
40
|
store: {
|
|
38
41
|
logger: Logger
|
|
42
|
+
pino: PinoLogger
|
|
39
43
|
beforeTime: bigint
|
|
40
44
|
hasCustomLog: boolean
|
|
41
45
|
};
|
|
@@ -56,15 +60,21 @@ interface Transport {
|
|
|
56
60
|
interface TimestampConfig {
|
|
57
61
|
translateTime?: boolean | string;
|
|
58
62
|
}
|
|
63
|
+
type PinoConfig = LoggerOptions & {
|
|
64
|
+
prettyPrint?: boolean | Record<string, unknown>
|
|
65
|
+
};
|
|
66
|
+
interface LogRotationConfig {
|
|
67
|
+
maxSize?: string | number;
|
|
68
|
+
maxFiles?: string | number;
|
|
69
|
+
interval?: string;
|
|
70
|
+
compress?: boolean;
|
|
71
|
+
compression?: "gzip";
|
|
72
|
+
}
|
|
59
73
|
interface Options {
|
|
60
74
|
config?: {
|
|
61
75
|
customLogFormat?: string
|
|
62
76
|
logFilePath?: string
|
|
63
|
-
logRotation?:
|
|
64
|
-
maxSize?: number
|
|
65
|
-
maxFiles?: number
|
|
66
|
-
compress?: boolean
|
|
67
|
-
}
|
|
77
|
+
logRotation?: LogRotationConfig
|
|
68
78
|
logFilter?: {
|
|
69
79
|
level?: LogLevel | LogLevel[]
|
|
70
80
|
method?: string | string[]
|
|
@@ -76,10 +86,14 @@ interface Options {
|
|
|
76
86
|
startupMessageFormat?: "banner" | "simple"
|
|
77
87
|
transports?: Transport[]
|
|
78
88
|
timestamp?: TimestampConfig
|
|
89
|
+
disableInternalLogger?: boolean
|
|
90
|
+
disableFileLogging?: boolean
|
|
91
|
+
useTransportsOnly?: boolean
|
|
92
|
+
pino?: PinoConfig
|
|
79
93
|
};
|
|
80
94
|
}
|
|
81
95
|
declare function createLogger(options?: Options): Logger;
|
|
82
|
-
declare function handleHttpError2(request: RequestInfo, error: HttpError, store: StoreData, options?: Options): void
|
|
96
|
+
declare function handleHttpError2(request: RequestInfo, error: HttpError, store: StoreData, options?: Options): Promise<void>;
|
|
83
97
|
declare function logToTransports(level: LogLevel, request: RequestInfo, data: LogData, store: StoreData, options?: Options): Promise<void>;
|
|
84
98
|
declare function logixlysia(options?: Options): Elysia;
|
|
85
|
-
export { logToTransports, handleHttpError2 as handleHttpError, logixlysia as default, createLogger, Transport, StoreData, RequestInfo, Options, LogixlysiaContext, Logger, LogLevel, LogData, HttpError };
|
|
99
|
+
export { logToTransports, handleHttpError2 as handleHttpError, logixlysia as default, createLogger, Transport, StoreData, RequestInfo, Options, LogixlysiaContext, Logger, LogRotationConfig, LogLevel, LogData, HttpError };
|
package/dist/index.d.ts
CHANGED
|
@@ -1,4 +1,5 @@
|
|
|
1
1
|
import { Elysia } from "elysia";
|
|
2
|
+
import { LoggerOptions, Logger as PinoLogger } from "pino";
|
|
2
3
|
interface RequestInfo {
|
|
3
4
|
headers: {
|
|
4
5
|
get: (key: string) => string | null
|
|
@@ -20,8 +21,9 @@ interface LogData {
|
|
|
20
21
|
}
|
|
21
22
|
interface Logger {
|
|
22
23
|
store?: StoreData;
|
|
24
|
+
pino: PinoLogger;
|
|
23
25
|
log(level: LogLevel, request: RequestInfo, data: LogData, store: StoreData): void;
|
|
24
|
-
handleHttpError(request: RequestInfo, error: HttpError, store: StoreData): void
|
|
26
|
+
handleHttpError(request: RequestInfo, error: HttpError, store: StoreData): Promise<void>;
|
|
25
27
|
customLogFormat?: string;
|
|
26
28
|
info(request: RequestInfo, message: string, context?: Record<string, string | number | boolean | null>, store?: StoreData): void;
|
|
27
29
|
error(request: RequestInfo, message: string, context?: Record<string, string | number | boolean | null>, store?: StoreData): void;
|
|
@@ -31,11 +33,13 @@ interface Logger {
|
|
|
31
33
|
interface StoreData {
|
|
32
34
|
beforeTime: bigint;
|
|
33
35
|
logger?: Logger;
|
|
36
|
+
pino?: PinoLogger;
|
|
34
37
|
hasCustomLog?: boolean;
|
|
35
38
|
}
|
|
36
39
|
interface LogixlysiaContext {
|
|
37
40
|
store: {
|
|
38
41
|
logger: Logger
|
|
42
|
+
pino: PinoLogger
|
|
39
43
|
beforeTime: bigint
|
|
40
44
|
hasCustomLog: boolean
|
|
41
45
|
};
|
|
@@ -56,15 +60,21 @@ interface Transport {
|
|
|
56
60
|
interface TimestampConfig {
|
|
57
61
|
translateTime?: boolean | string;
|
|
58
62
|
}
|
|
63
|
+
type PinoConfig = LoggerOptions & {
|
|
64
|
+
prettyPrint?: boolean | Record<string, unknown>
|
|
65
|
+
};
|
|
66
|
+
interface LogRotationConfig {
|
|
67
|
+
maxSize?: string | number;
|
|
68
|
+
maxFiles?: string | number;
|
|
69
|
+
interval?: string;
|
|
70
|
+
compress?: boolean;
|
|
71
|
+
compression?: "gzip";
|
|
72
|
+
}
|
|
59
73
|
interface Options {
|
|
60
74
|
config?: {
|
|
61
75
|
customLogFormat?: string
|
|
62
76
|
logFilePath?: string
|
|
63
|
-
logRotation?:
|
|
64
|
-
maxSize?: number
|
|
65
|
-
maxFiles?: number
|
|
66
|
-
compress?: boolean
|
|
67
|
-
}
|
|
77
|
+
logRotation?: LogRotationConfig
|
|
68
78
|
logFilter?: {
|
|
69
79
|
level?: LogLevel | LogLevel[]
|
|
70
80
|
method?: string | string[]
|
|
@@ -76,10 +86,14 @@ interface Options {
|
|
|
76
86
|
startupMessageFormat?: "banner" | "simple"
|
|
77
87
|
transports?: Transport[]
|
|
78
88
|
timestamp?: TimestampConfig
|
|
89
|
+
disableInternalLogger?: boolean
|
|
90
|
+
disableFileLogging?: boolean
|
|
91
|
+
useTransportsOnly?: boolean
|
|
92
|
+
pino?: PinoConfig
|
|
79
93
|
};
|
|
80
94
|
}
|
|
81
95
|
declare function createLogger(options?: Options): Logger;
|
|
82
|
-
declare function handleHttpError2(request: RequestInfo, error: HttpError, store: StoreData, options?: Options): void
|
|
96
|
+
declare function handleHttpError2(request: RequestInfo, error: HttpError, store: StoreData, options?: Options): Promise<void>;
|
|
83
97
|
declare function logToTransports(level: LogLevel, request: RequestInfo, data: LogData, store: StoreData, options?: Options): Promise<void>;
|
|
84
98
|
declare function logixlysia(options?: Options): Elysia;
|
|
85
|
-
export { logToTransports, handleHttpError2 as handleHttpError, logixlysia as default, createLogger, Transport, StoreData, RequestInfo, Options, LogixlysiaContext, Logger, LogLevel, LogData, HttpError };
|
|
99
|
+
export { logToTransports, handleHttpError2 as handleHttpError, logixlysia as default, createLogger, Transport, StoreData, RequestInfo, Options, LogixlysiaContext, Logger, LogRotationConfig, LogLevel, LogData, HttpError };
|
package/dist/index.js
CHANGED
|
@@ -1,13 +1,13 @@
|
|
|
1
|
-
import{Elysia as
|
|
2
|
-
โ${
|
|
3
|
-
โ${
|
|
4
|
-
โ${
|
|
5
|
-
โ${
|
|
6
|
-
โ${
|
|
7
|
-
โ${
|
|
8
|
-
โ${
|
|
9
|
-
`)}else console.log(`\uD83E\uDD8A Elysia is running at ${
|
|
10
|
-
`;await
|
|
1
|
+
import{Elysia as Mr}from"elysia";var U=(r,o)=>{let n=Math.max(0,(o-r.length)/2),t=" ".repeat(n);return`${t}${r}${t}`.padEnd(o)};function w(r,o){let n=r?.hostname??"localhost",t=r?.port??3000,s=r?.protocol??"http";if(o?.config?.startupMessageFormat!=="simple"){let c=`\uD83E\uDD8A Elysia is running at ${s}://${n}:${t}`,u=Math.max(22,c.length)+4,f="โ".repeat(u),i=U("",u);console.log(`
|
|
2
|
+
โ${f}โ
|
|
3
|
+
โ${i}โ
|
|
4
|
+
โ${U("Elysia with Logixlysia",u)}โ
|
|
5
|
+
โ${i}โ
|
|
6
|
+
โ${U(c,u)}โ
|
|
7
|
+
โ${i}โ
|
|
8
|
+
โ${f}โ
|
|
9
|
+
`)}else console.log(`\uD83E\uDD8A Elysia is running at ${s}://${n}:${t}`)}import b from"chalk";import{StatusMap as h}from"elysia";function H(r){if(typeof r==="number")return r;return h[r]||500}function E(r,o){let n=r.toString();if(!o)return n;if(r>=500)return b.red(n);if(r>=400)return b.yellow(n);if(r>=300)return b.cyan(n);if(r>=200)return b.green(n);return b.white(n)}import x from"pino";import B from"chalk";import a from"chalk";var C={INFO:a.bgGreen.black,WARNING:a.bgYellow.black,ERROR:a.bgRed.black},T={GET:a.green,POST:a.yellow,PUT:a.blue,PATCH:a.magentaBright,DELETE:a.red,HEAD:a.cyan,OPTIONS:a.magenta};import K from"chalk";var z=[{unit:"s",threshold:1e9,decimalPlaces:2},{unit:"ms",threshold:1e6,decimalPlaces:0},{unit:"ยตs",threshold:1000,decimalPlaces:0},{unit:"ns",threshold:1,decimalPlaces:0}];function D(r,o){let n=Number(process.hrtime.bigint()-r);for(let{unit:t,threshold:s,decimalPlaces:e}of z)if(n>=s){let c=`${(n/s).toFixed(e)}${t}`.padStart(8).padEnd(16);return o?K.gray(c):c}return o?K.gray("0ns".padStart(8).padEnd(16)):"0ns".padStart(8).padEnd(16)}function R(r,o){let n=r.toUpperCase();return o?C[n]?.(n.padEnd(7))||n:n.padEnd(7)}function k(r,o){let n=T[r];return o&&n?n(r.padEnd(7)):r.padEnd(7)}function d(r){try{return new URL(r.url).pathname}catch{return}}var m=(r)=>r.toString().padStart(2,"0");function P(r){let o=r.getFullYear(),n=m(r.getMonth()+1),t=m(r.getDate()),s=m(r.getHours()),e=m(r.getMinutes()),g=m(r.getSeconds()),c=r.getMilliseconds().toString().padStart(3,"0");return`${o}-${n}-${t} ${s}:${e}:${g}.${c}`}function rr(r,o){let n={yyyy:r.getFullYear(),yy:r.getFullYear().toString().slice(-2),mm:m(r.getMonth()+1),dd:m(r.getDate()),HH:m(r.getHours()),MM:m(r.getMinutes()),ss:m(r.getSeconds()),SSS:m(r.getMilliseconds()),Z:-r.getTimezoneOffset()/60};return o.replace(/yyyy|yy|mm|dd|HH|MM|ss|SSS|Z/g,(t)=>(n[t]??"").toString())}function v(r,o){if(!o?.translateTime)return r.toISOString();if(o.translateTime===!0||o.translateTime==="SYS:STANDARD")return P(r);return rr(r,o.translateTime)}var or="\uD83E\uDD8A {now} {level} {duration} {method} {pathname} {status} {message} {context} {ip}";function nr(r,o){if(o?.config?.useColors!==void 0)return o.config.useColors&&process.env.NO_COLOR===void 0;return r&&process.env.NO_COLOR===void 0}function L(r,o,n,t,s,e=!0){let g=nr(e,s),c=new Date,u={now:g?B.bgYellow(B.black(v(c,s?.config?.timestamp))):v(c,s?.config?.timestamp),epoch:Math.floor(c.getTime()/1000).toString(),level:R(r,e),duration:D(t.beforeTime,e),method:k(o.method,e),pathname:d(o),status:E(n.status||200,e),message:n.message||"",context:n.context?(()=>{try{return JSON.stringify(n.context)}catch(i){return`[Error serializing context: ${i instanceof Error?i.message:"Unknown error"}]`}})():"",ip:s?.config?.ip&&o.headers.get("x-forwarded-for")?`IP: ${o.headers.get("x-forwarded-for")}`:""};return(s?.config?.customLogFormat||or).replace(/{(\w+)}/g,(i,S)=>{if(S in u)return u[S]||"";return""})}async function l(r,o,n,t,s){if(!s?.config?.transports||s.config.transports.length===0)return;let e=L(r,o,n,t,s,!1),g=s.config.transports.map((c)=>c.log(r,e,{request:o,data:n,store:t}));await Promise.all(g)}import{promises as X}from"node:fs";import{dirname as yr}from"node:path";import{promises as F}from"node:fs";import{basename as tr,dirname as sr,join as er}from"node:path";var gr=/^(\d+(?:\.\d+)?)\s*([kmg])?b?$/,cr=/^(\d+)\s*([hdw])$/,ur=/^(\d+)\s*d$/;function W(r){if(typeof r==="number")return r;let o={k:1024,m:1048576,g:1073741824},n=r.toLowerCase().match(gr);if(!n?.[1])throw Error(`Invalid size format: ${r}`);let t=Number.parseFloat(n[1]),s=n[2]??"";return Math.floor(t*(o[s]??1))}function M(r){let o={h:3600000,d:86400000,w:604800000},n=r.toLowerCase().match(cr);if(!n)throw Error(`Invalid interval format: ${r}`);let t=n[1],s=n[2];if(!t)throw Error(`Invalid interval format: ${r}`);if(!s)throw Error(`Invalid interval format: ${r}`);return Number.parseInt(t,10)*(o[s]??0)}function j(r){if(typeof r==="number")return{type:"count",value:r};let o=r.toLowerCase().match(ur);if(o?.[1])return{type:"time",value:Number.parseInt(o[1],10)*24*60*60*1000};throw Error(`Invalid retention format: ${r}`)}async function pr(r){try{return(await F.stat(r)).mtime.getTime()}catch{return Date.now()}}var N=new Map;async function _(r,o){let n=Date.now(),t=N.get(r);if(t!==void 0)return n-t>=o;let s=await pr(r);return N.set(r,s),n-s>=o}function Y(r){N.set(r,Date.now())}async function A(r){let o=sr(r),n=tr(r);try{let t=await F.readdir(o),s=new RegExp(`^${n.replace(/[.*+?^${}()|[\]\\]/g,"\\$&")}\\.\\d{4}-\\d{2}-\\d{2}(-\\d{2}-\\d{2}-\\d{2})?(\\.gz)?$`),e=t.filter((c)=>s.test(c)).map((c)=>er(o,c));return(await Promise.all(e.map(async(c)=>{let u=await F.stat(c);return{file:c,mtime:u.mtime.getTime()}}))).sort((c,u)=>u.mtime-c.mtime).map((c)=>c.file)}catch{return[]}}import{createReadStream as mr,createWriteStream as ar,promises as p}from"node:fs";import{pipeline as ir}from"node:stream/promises";import{createGzip as Lr}from"node:zlib";function lr(r,o){let n=o.getFullYear(),t=String(o.getMonth()+1).padStart(2,"0"),s=String(o.getDate()).padStart(2,"0"),e=String(o.getHours()).padStart(2,"0"),g=String(o.getMinutes()).padStart(2,"0"),c=String(o.getSeconds()).padStart(2,"0");return`${r}.${n}-${t}-${s}-${e}-${g}-${c}`}async function fr(r){try{if((await p.stat(r)).size===0)return"";let n=lr(r,new Date);return await p.rename(r,n),Y(r),n}catch(o){if(o.code!=="ENOENT")console.error(`Failed to rotate log file ${r}:`,o);return""}}async function br(r){try{let o=`${r}.gz`,n=mr(r),t=ar(o),s=Lr();await ir(n,s,t),await p.unlink(r)}catch(o){console.error(`Failed to compress file ${r}:`,o)}}async function Or(r,o){if(!o.maxFiles)return;try{let n=await A(r);if(n.length===0)return;let t=j(o.maxFiles);if(t.type==="count"){let s=n.slice(t.value);await Promise.all(s.map((e)=>p.unlink(e)))}else if(t.type==="time"){let s=Date.now()-t.value,e=await Promise.all(n.map(async(g)=>{return(await p.stat(g)).mtime.getTime()<s?g:null}));await Promise.all(e.filter((g)=>g!==null).map((g)=>p.unlink(g)))}}catch(n){console.error(`Failed to clean old log files for ${r}:`,n)}}async function J(r,o){try{let n=await fr(r);if(!n)return;if(o.compress)await br(n);await Or(r,o)}catch(n){console.error(`Failed to perform rotation for ${r}:`,n)}}async function q(r,o){try{await p.access(r)}catch{return!1}if(o.maxSize)try{let n=W(o.maxSize);if((await p.stat(r)).size>=n)return!0}catch(n){console.error(`Failed to check file size for ${r}:`,n)}if(o.interval)try{let n=M(o.interval),t=await p.stat(r);if(Date.now()-t.mtimeMs>=n)return!0}catch(n){console.error(`Failed to check file age for ${r}:`,n)}return!1}var Q=new Set;async function xr(r){let o=yr(r);if(!Q.has(o))await X.mkdir(o,{recursive:!0}),Q.add(o)}async function Sr(r,o){let n=o?.config?.logRotation;if(!n)return;let t=!1;if(n.maxSize)t=await q(r,n);if(!t&&n.interval)try{let s=M(n.interval);t=await _(r,s)}catch(s){console.error("Invalid interval format in log rotation config:",s)}if(t)await J(r,n)}async function O(r,o,n,t,s,e){await xr(r),await Sr(r,e);let g=`${L(o,n,t,s,e,!1)}
|
|
10
|
+
`;await X.appendFile(r,g,{flag:"a"})}var G=(r,o)=>Array.isArray(r)?r.includes(o):r===o;function Z(r,o,n,t){let s=t?.config?.logFilter;if(!s)return!0;return(!s.level||G(s.level,r))&&(!s.status||G(s.status,o))&&(!s.method||G(s.method,n))}async function $(r,o,n,t){let e={status:o.status||500,message:o.message,stack:o.stack},g=[];if(!(t?.config?.useTransportsOnly||t?.config?.disableInternalLogger))console.error(L("ERROR",r,e,n,t));if(!t?.config?.useTransportsOnly&&t?.config?.logFilePath&&!t?.config?.disableFileLogging)g.push(O(t.config.logFilePath,"ERROR",r,e,n,t));if(t?.config?.transports?.length)g.push(l("ERROR",r,e,n,t));await Promise.all(g)}function wr(){let r=process.memoryUsage().heapUsed/1024/1024,o=process.cpuUsage();return{memoryUsage:r,cpuUsage:o.user/1e6}}function Er(r){return{level:r.level||"info",timestamp:r.timestamp??!0,messageKey:r.messageKey||"msg",errorKey:r.errorKey||"err",base:r.base||{pid:process.pid}}}function Dr(r){return x.transport({target:"pino-pretty",options:{colorize:!0,translateTime:"HH:MM:ss Z",ignore:"pid,hostname",...typeof r==="object"?r:{}}})}function Rr(r){let o=r?.config?.pino||{},{prettyPrint:n,transport:t,...s}=o,e={...Er(o),...s};if(n)return x(e,Dr(n));if(t){if(typeof t==="object"&&"target"in t)return x(e,x.transport(t));console.warn("Invalid transport configuration provided, falling back to default")}return x(e)}function kr(r){switch(r.toUpperCase()){case"DEBUG":return"debug";case"INFO":return"info";case"WARNING":case"WARN":return"warn";case"ERROR":return"error";default:return console.warn(`Unknown log level "${r}", defaulting to "info"`),"info"}}function dr(r,o,n,t,s){let e=kr(o),g=r[e],c=Boolean(s?.config?.pino?.transport||s?.config?.pino?.prettyPrint);if((!s?.config?.useTransportsOnly||c)&&typeof g==="function")g.call(r,n,t)}async function vr(r,o,n,t,s,e){let g=[];if(!(s?.config?.useTransportsOnly||s?.config?.disableInternalLogger))console.log(e);if(!s?.config?.useTransportsOnly&&s?.config?.logFilePath&&!s?.config?.disableFileLogging)g.push(O(s.config.logFilePath,r,o,n,t,s));if(s?.config?.transports?.length)g.push(l(r,o,n,t,s));await Promise.all(g)}async function y(r,o,n,t,s,e){if(!Z(o,t.status||200,n.method,e))return;if(!t.metrics)t.metrics=wr();if(o==="ERROR"&&!t.stack){let V=Error(t.message||"Unknown error");t.stack=V.stack}let g=e?.config?.pino?.errorKey||"err",c=o==="ERROR"&&t.stack?{name:"Error",message:t.message||"Unknown error",stack:t.stack}:void 0,u=n.headers.get("x-forwarded-for"),f=e?.config?.ip&&u?u.split(",")[0]?.trim():void 0,i={method:n.method,url:n.url,status:t.status,message:t.message,context:t.context,metrics:t.metrics,duration:Number(process.hrtime.bigint()-s.beforeTime)/1e6,ip:f,[g]:c};dr(r,o,i,t.message||"Request processed",e);let S=L(o,n,t,s,e,!0);await vr(o,n,t,s,e,S)}function I(r){let o=Rr(r),n={store:void 0,pino:o,log:(t,s,e,g)=>y(o,t,s,e,g,r),handleHttpError:async(t,s,e)=>await $(t,s,e,r),customLogFormat:r?.config?.customLogFormat,info:(t,s,e,g)=>{let c=g||n.store||{beforeTime:process.hrtime.bigint()};return c.hasCustomLog=!0,y(o,"INFO",t,{message:s,context:e,status:200},c,r)},error:(t,s,e,g)=>{let c=g||n.store||{beforeTime:process.hrtime.bigint()};return c.hasCustomLog=!0,y(o,"ERROR",t,{message:s,context:e,status:500},c,r)},warn:(t,s,e,g)=>{let c=g||n.store||{beforeTime:process.hrtime.bigint()};return c.hasCustomLog=!0,y(o,"WARNING",t,{message:s,context:e,status:200},c,r)},debug:(t,s,e,g)=>{let c=g||n.store||{beforeTime:process.hrtime.bigint()};return c.hasCustomLog=!0,y(o,"DEBUG",t,{message:s,context:e,status:200},c,r)}};return n}function $r(r){let o=I(r);return new Mr({name:"Logixlysia"}).onStart((n)=>{if(r?.config?.showStartupMessage??!0)w(n.server,r)}).onRequest((n)=>{let t={...n.store,beforeTime:process.hrtime.bigint(),logger:o,pino:o.pino,hasCustomLog:!1};n.store=t,o.store=t}).onAfterHandle({as:"global"},({request:n,set:t,store:s})=>{let e=s;if(!e.hasCustomLog){let g=H(t.status||200);o.log("INFO",n,{status:g,message:String(t.headers?.["x-message"]||"")},e)}}).onError({as:"global"},async({request:n,error:t,set:s,store:e})=>{let g=H(s.status||500);await o.handleHttpError(n,{...t,status:g},e)})}export{l as logToTransports,$ as handleHttpError,$r as default,I as createLogger};
|
|
11
11
|
|
|
12
|
-
//# debugId=
|
|
13
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/index.ts", "../src/extensions/start-server.ts", "../src/helpers/status.ts", "../src/logger/build-log-message.ts", "../src/helpers/color-mapping.ts", "../src/helpers/duration.ts", "../src/helpers/log.ts", "../src/helpers/method.ts", "../src/helpers/path.ts", "../src/helpers/timestamp.ts", "../src/output/console.ts", "../src/output/file.ts", "../src/logger/filter.ts", "../src/logger/handle-http-error.ts", "../src/logger/create-logger.ts"],
  "sourcesContent": [
    "import { Elysia } from 'elysia'\n\nimport { startServer } from './extensions'\nimport { getStatusCode } from './helpers/status'\nimport type { HttpError, Options, Server, StoreData } from './interfaces'\nimport { createLogger } from './logger'\n\nexport default function logixlysia(options?: Options): Elysia {\n  const log = createLogger(options)\n\n  return new Elysia({\n    name: 'Logixlysia'\n  })\n    .onStart(ctx => {\n      const showStartupMessage = options?.config?.showStartupMessage ?? true\n      if (showStartupMessage) {\n        startServer(ctx.server as Server, options)\n      }\n    })\n    .onRequest(ctx => {\n      const store = {\n        ...ctx.store,\n        beforeTime: process.hrtime.bigint(),\n        logger: log,\n        hasCustomLog: false\n      }\n      ctx.store = store\n      log.store = store\n    })\n    .onAfterHandle({ as: 'global' }, ({ request, set, store }) => {\n      const storeData = store as StoreData\n\n      if (!storeData.hasCustomLog) {\n        const status = getStatusCode(set.status || 200)\n        log.log(\n          'INFO',\n          request,\n          {\n            status,\n            message: String(set.headers?.['x-message'] || '')\n          },\n          storeData\n        )\n      }\n    })\n    .onError({ as: 'global' }, ({ request, error, set, store }) => {\n      const status = getStatusCode(set.status || 500)\n      log.handleHttpError(\n        request,\n        { ...error, status } as HttpError,\n        store as StoreData\n      )\n    })\n}\n\nexport type {\n  HttpError,\n  LogData,\n  Logger,\n  LogixlysiaContext,\n  LogLevel,\n  Options,\n  RequestInfo,\n  StoreData,\n  Transport\n} from './interfaces'\nexport { createLogger, handleHttpError } from './logger'\nexport { logToTransports } from './output'\n",
    "import type { Options, Server } from '../interfaces'\n\nconst createBoxText = (text: string, width: number): string => {\n  const paddingLength = Math.max(0, (width - text.length) / 2)\n  const padding = ' '.repeat(paddingLength)\n  return `${padding}${text}${padding}`.padEnd(width)\n}\n\nexport default function startServer(config: Server, options?: Options): void {\n  const hostname = config?.hostname ?? 'localhost'\n  const port = config?.port ?? 3000\n  const protocol = config?.protocol ?? 'http'\n  const showBanner = options?.config?.startupMessageFormat !== 'simple'\n\n  if (showBanner) {\n    const title = 'Elysia with Logixlysia'\n    const message = `🦊 Elysia is running at ${protocol}://${hostname}:${port}`\n    const boxWidth = Math.max(title.length, message.length) + 4\n    const border = '─'.repeat(boxWidth)\n    const emptyLine = createBoxText('', boxWidth)\n\n    console.log(`\n      ┌${border}┐\n      │${emptyLine}│\n      │${createBoxText(title, boxWidth)}│\n      │${emptyLine}│\n      │${createBoxText(message, boxWidth)}│\n      │${emptyLine}│\n      └${border}┘\n    `)\n  } else {\n    console.log(`🦊 Elysia is running at ${protocol}://${hostname}:${port}`)\n  }\n}\n",
    "import chalk from 'chalk'\nimport { StatusMap } from 'elysia'\n\nexport function getStatusCode(status: string | number): number {\n  if (typeof status === 'number') {\n    return status\n  }\n  return (StatusMap as Record<string, number>)[status] || 500\n}\n\nexport default function statusString(\n  status: number,\n  useColors: boolean\n): string {\n  const statusStr = status.toString()\n  if (!useColors) {\n    return statusStr\n  }\n\n  if (status >= 500) {\n    return chalk.red(statusStr)\n  }\n  if (status >= 400) {\n    return chalk.yellow(statusStr)\n  }\n  if (status >= 300) {\n    return chalk.cyan(statusStr)\n  }\n  if (status >= 200) {\n    return chalk.green(statusStr)\n  }\n  return chalk.white(statusStr)\n}\n",
    "import chalk from 'chalk'\n\nimport {\n  durationString,\n  formatTimestamp,\n  logString,\n  methodString,\n  pathString,\n  statusString\n} from '../helpers'\nimport type {\n  LogComponents,\n  LogData,\n  LogLevel,\n  Options,\n  RequestInfo,\n  StoreData\n} from '../interfaces'\n\nconst defaultLogFormat =\n  '🦊 {now} {level} {duration} {method} {pathname} {status} {message} {context} {ip}'\n\nfunction shouldUseColors(useColors: boolean, options?: Options): boolean {\n  if (options?.config?.useColors !== undefined) {\n    return options.config.useColors && process.env.NO_COLOR === undefined\n  }\n  return useColors && process.env.NO_COLOR === undefined\n}\n\nexport function buildLogMessage(\n  level: LogLevel,\n  request: RequestInfo,\n  data: LogData,\n  store: StoreData,\n  options?: Options,\n  useColors = true\n): string {\n  const actuallyUseColors = shouldUseColors(useColors, options)\n  const now = new Date()\n  const components: LogComponents = {\n    now: actuallyUseColors\n      ? chalk.bgYellow(\n          chalk.black(formatTimestamp(now, options?.config?.timestamp))\n        )\n      : formatTimestamp(now, options?.config?.timestamp),\n    epoch: Math.floor(now.getTime() / 1000).toString(),\n    level: logString(level, useColors),\n    duration: durationString(store.beforeTime, useColors),\n    method: methodString(request.method, useColors),\n    pathname: pathString(request),\n    status: statusString(data.status || 200, useColors),\n    message: data.message || '',\n    context: data.context\n      ? (() => {\n          try {\n            return JSON.stringify(data.context)\n          } catch (error) {\n            return `[Error serializing context: ${error instanceof Error ? error.message : 'Unknown error'}]`\n          }\n        })()\n      : '',\n    ip:\n      options?.config?.ip && request.headers.get('x-forwarded-for')\n        ? `IP: ${request.headers.get('x-forwarded-for')}`\n        : ''\n  }\n\n  const logFormat = options?.config?.customLogFormat || defaultLogFormat\n\n  return logFormat.replace(/{(\\w+)}/g, (_, key: string) => {\n    if (key in components) {\n      return components[key as keyof LogComponents] || ''\n    }\n    return ''\n  })\n}\n",
    "import chalk from 'chalk'\n\nimport type { ColorMap } from '../interfaces'\n\nexport const LogLevelColorMap: ColorMap = {\n  INFO: chalk.bgGreen.black,\n  WARNING: chalk.bgYellow.black,\n  ERROR: chalk.bgRed.black\n}\n\nexport const HttpMethodColorMap: ColorMap = {\n  GET: chalk.green,\n  POST: chalk.yellow,\n  PUT: chalk.blue,\n  PATCH: chalk.magentaBright,\n  DELETE: chalk.red,\n  HEAD: chalk.cyan,\n  OPTIONS: chalk.magenta\n}\n",
    "import chalk from 'chalk'\n\nconst timeUnits = [\n  { unit: 's', threshold: 1e9, decimalPlaces: 2 },\n  { unit: 'ms', threshold: 1e6, decimalPlaces: 0 },\n  { unit: 'µs', threshold: 1e3, decimalPlaces: 0 },\n  { unit: 'ns', threshold: 1, decimalPlaces: 0 }\n]\n\nexport default function durationString(\n  beforeTime: bigint,\n  useColors: boolean\n): string {\n  const nanoseconds = Number(process.hrtime.bigint() - beforeTime)\n\n  for (const { unit, threshold, decimalPlaces } of timeUnits) {\n    if (nanoseconds >= threshold) {\n      const value = (nanoseconds / threshold).toFixed(decimalPlaces)\n      const timeStr = `${value}${unit}`.padStart(8).padEnd(16)\n      return useColors ? chalk.gray(timeStr) : timeStr\n    }\n  }\n\n  return useColors\n    ? chalk.gray('0ns'.padStart(8).padEnd(16))\n    : '0ns'.padStart(8).padEnd(16)\n}\n",
    "import type { LogLevel } from '../interfaces'\nimport { LogLevelColorMap } from './color-mapping'\n\nexport default function logString(level: LogLevel, useColors: boolean): string {\n  const levelStr = level.toUpperCase()\n  return useColors\n    ? LogLevelColorMap[levelStr]?.(levelStr.padEnd(7)) || levelStr\n    : levelStr.padEnd(7)\n}\n",
    "import { HttpMethodColorMap } from './color-mapping'\n\nexport default function methodString(\n  method: string,\n  useColors: boolean\n): string {\n  const colorFunction = HttpMethodColorMap[method]\n  return useColors && colorFunction\n    ? colorFunction(method.padEnd(7))\n    : method.padEnd(7)\n}\n",
    "import type { RequestInfo } from '../interfaces'\n\nexport default function pathString(\n  requestInfo: RequestInfo\n): string | undefined {\n  try {\n    return new URL(requestInfo.url).pathname\n  } catch {\n    return\n  }\n}\n",
    "import type { TimestampConfig } from '../interfaces'\n\n// const DEFAULT_TIMESTAMP_FORMAT = 'yyyy-mm-dd HH:MM:ss'\nconst SYS_TIME = 'SYS:STANDARD'\n\nconst pad = (n: number): string => n.toString().padStart(2, '0')\n\nfunction formatSystemTime(date: Date): string {\n  const year = date.getFullYear()\n  const month = pad(date.getMonth() + 1)\n  const day = pad(date.getDate())\n  const hours = pad(date.getHours())\n  const minutes = pad(date.getMinutes())\n  const seconds = pad(date.getSeconds())\n  const ms = date.getMilliseconds().toString().padStart(3, '0')\n\n  return `${year}-${month}-${day} ${hours}:${minutes}:${seconds}.${ms}`\n}\n\nfunction formatCustomTime(date: Date, format: string): string {\n  const tokens: { [key: string]: string | number } = {\n    yyyy: date.getFullYear(),\n    yy: date.getFullYear().toString().slice(-2),\n    mm: pad(date.getMonth() + 1),\n    dd: pad(date.getDate()),\n    HH: pad(date.getHours()),\n    MM: pad(date.getMinutes()),\n    ss: pad(date.getSeconds()),\n    SSS: pad(date.getMilliseconds()),\n    Z: -date.getTimezoneOffset() / 60\n  }\n\n  return format.replace(/yyyy|yy|mm|dd|HH|MM|ss|SSS|Z/g, match =>\n    (tokens[match] ?? '').toString()\n  )\n}\n\nexport function formatTimestamp(date: Date, config?: TimestampConfig): string {\n  if (!config?.translateTime) {\n    return date.toISOString()\n  }\n\n  if (config.translateTime === true || config.translateTime === SYS_TIME) {\n    return formatSystemTime(date)\n  }\n\n  return formatCustomTime(date, config.translateTime)\n}\n",
    "import type {\n  LogData,\n  LogLevel,\n  Options,\n  RequestInfo,\n  StoreData\n} from '../interfaces'\nimport { buildLogMessage } from '../logger/build-log-message'\n\nexport async function logToTransports(\n  level: LogLevel,\n  request: RequestInfo,\n  data: LogData,\n  store: StoreData,\n  options?: Options\n): Promise<void> {\n  if (!options?.config?.transports || options.config.transports.length === 0) {\n    return\n  }\n\n  const message = buildLogMessage(level, request, data, store, options, false)\n\n  const promises = options.config.transports.map(transport =>\n    transport.log(level, message, { request, data, store })\n  )\n\n  await Promise.all(promises)\n}\n",
    "import { promises as fs } from 'node:fs'\nimport { dirname } from 'node:path'\n\nimport type {\n  LogData,\n  LogLevel,\n  Options,\n  RequestInfo,\n  StoreData\n} from '../interfaces'\nimport { buildLogMessage } from '../logger/build-log-message'\n\nconst dirCache = new Set<string>()\n\nasync function ensureDirectoryExists(filePath: string): Promise<void> {\n  const dir = dirname(filePath)\n  if (!dirCache.has(dir)) {\n    await fs.mkdir(dir, { recursive: true })\n    dirCache.add(dir)\n  }\n}\n\nexport async function logToFile(\n  filePath: string,\n  level: LogLevel,\n  request: RequestInfo,\n  data: LogData,\n  store: StoreData,\n  options?: Options\n): Promise<void> {\n  await ensureDirectoryExists(filePath)\n  const logMessage = `${buildLogMessage(level, request, data, store, options, false)}\\n`\n  await fs.appendFile(filePath, logMessage, { flag: 'a' })\n}\n",
    "import type { LogLevel, Options } from '../interfaces'\n\nconst checkFilter = (filterValue: unknown, value: unknown) =>\n  Array.isArray(filterValue)\n    ? filterValue.includes(value)\n    : filterValue === value\n\nexport function filterLog(\n  logLevel: LogLevel,\n  status: number,\n  method: string,\n  options?: Options\n): boolean {\n  const filter = options?.config?.logFilter\n  if (!filter) {\n    return true\n  }\n\n  return (\n    (!filter.level || checkFilter(filter.level, logLevel)) &&\n    (!filter.status || checkFilter(filter.status, status)) &&\n    (!filter.method || checkFilter(filter.method, method))\n  )\n}\n",
    "import type { HttpError, Options, RequestInfo, StoreData } from '../interfaces'\nimport { logToFile } from '../output'\nimport { buildLogMessage } from './build-log-message'\n\nexport function handleHttpError(\n  request: RequestInfo,\n  error: HttpError,\n  store: StoreData,\n  options?: Options\n): void {\n  const statusCode = error.status || 500\n  console.error(\n    buildLogMessage('ERROR', request, { status: statusCode }, store, options)\n  )\n\n  const promises: Promise<void>[] = []\n\n  if (options?.config?.logFilePath) {\n    promises.push(\n      logToFile(\n        options.config.logFilePath,\n        'ERROR',\n        request,\n        { status: statusCode },\n        store,\n        options\n      )\n    )\n  }\n}\n",
    "import type {\n  LogData,\n  Logger,\n  LogLevel,\n  Options,\n  RequestInfo,\n  StoreData\n} from '../interfaces'\nimport { logToFile, logToTransports } from '../output'\nimport { buildLogMessage } from './build-log-message'\nimport { filterLog } from './filter'\nimport { handleHttpError } from './handle-http-error'\n\nfunction getMetrics(): LogData['metrics'] {\n  const memoryUsage = process.memoryUsage().heapUsed / 1024 / 1024 // MB\n  const cpuUsage = process.cpuUsage()\n\n  return {\n    memoryUsage,\n    cpuUsage: cpuUsage.user / 1_000_000 // convert to seconds\n  }\n}\n\nasync function log(\n  level: LogLevel,\n  request: RequestInfo,\n  data: LogData,\n  store: StoreData,\n  options?: Options\n): Promise<void> {\n  if (!filterLog(level, data.status || 200, request.method, options)) {\n    return\n  }\n\n  // Add metrics if not provided\n  if (!data.metrics) {\n    data.metrics = getMetrics()\n  }\n\n  // Add stack trace for errors\n  if (level === 'ERROR' && !data.stack) {\n    data.stack = new Error(`Error: ${data.message || 'Unknown error'}`).stack\n  }\n\n  const logMessage = buildLogMessage(level, request, data, store, options, true)\n  console.log(logMessage)\n\n  const promises: Promise<void>[] = []\n\n  if (options?.config?.logFilePath) {\n    promises.push(\n      logToFile(\n        options.config.logFilePath,\n        level,\n        request,\n        data,\n        store,\n        options\n      )\n    )\n  }\n\n  if (options?.config?.transports?.length) {\n    promises.push(logToTransports(level, request, data, store, options))\n  }\n\n  await Promise.all(promises)\n}\n\nexport function createLogger(options?: Options): Logger {\n  const logger: Logger = {\n    store: undefined,\n    log: (level, request, data, store) =>\n      log(level, request, data, store, options),\n    handleHttpError: (request, error, store) =>\n      handleHttpError(request, error, store, options),\n    customLogFormat: options?.config?.customLogFormat,\n    info: (request, message, context, store) => {\n      const storeData = store ||\n        logger.store || { beforeTime: process.hrtime.bigint() }\n      storeData.hasCustomLog = true\n      return log(\n        'INFO',\n        request,\n        { message, context, status: 200 },\n        storeData,\n        options\n      )\n    },\n    error: (request, message, context, store) => {\n      const storeData = store ||\n        logger.store || { beforeTime: process.hrtime.bigint() }\n      storeData.hasCustomLog = true\n      return log(\n        'ERROR',\n        request,\n        { message, context, status: 500 },\n        storeData,\n        options\n      )\n    },\n    warn: (request, message, context, store) => {\n      const storeData = store ||\n        logger.store || { beforeTime: process.hrtime.bigint() }\n      storeData.hasCustomLog = true\n      return log(\n        'WARNING',\n        request,\n        { message, context, status: 200 },\n        storeData,\n        options\n      )\n    },\n    debug: (request, message, context, store) => {\n      const storeData = store ||\n        logger.store || { beforeTime: process.hrtime.bigint() }\n      storeData.hasCustomLog = true\n      return log(\n        'DEBUG',\n        request,\n        { message, context, status: 200 },\n        storeData,\n        options\n      )\n    }\n  }\n  return logger\n}\n"
  ],
  "mappings": "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",
  "debugId": "F56AA4EFC34A3D6864756E2164756E21",
  "names": []
}
|
|
12
|
+
//# debugId=62A87E578A304FBB64756E2164756E21
|
|
13
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["src/index.ts", "src/extensions/start-server.ts", "src/helpers/status.ts", "src/logger/create-logger.ts", "src/logger/build-log-message.ts", "src/helpers/color-mapping.ts", "src/helpers/duration.ts", "src/helpers/log.ts", "src/helpers/method.ts", "src/helpers/path.ts", "src/helpers/timestamp.ts", "src/output/console.ts", "src/output/file.ts", "src/utils/rotation.ts", "src/output/rotation-manager.ts", "src/logger/filter.ts", "src/logger/handle-http-error.ts"],
  "sourcesContent": [
    "import { Elysia } from 'elysia'\n\nimport { startServer } from './extensions'\nimport { getStatusCode } from './helpers/status'\nimport type { HttpError, Options, Server, StoreData } from './interfaces'\nimport { createLogger } from './logger'\n\nexport default function logixlysia(options?: Options): Elysia {\n  const log = createLogger(options)\n\n  return new Elysia({\n    name: 'Logixlysia'\n  })\n    .onStart(ctx => {\n      const showStartupMessage = options?.config?.showStartupMessage ?? true\n      if (showStartupMessage) {\n        startServer(ctx.server as Server, options)\n      }\n    })\n    .onRequest(ctx => {\n      const store = {\n        ...ctx.store,\n        beforeTime: process.hrtime.bigint(),\n        logger: log,\n        pino: log.pino, // Expose Pino logger directly\n        hasCustomLog: false\n      }\n      ctx.store = store\n      log.store = store\n    })\n    .onAfterHandle({ as: 'global' }, ({ request, set, store }) => {\n      const storeData = store as StoreData\n\n      if (!storeData.hasCustomLog) {\n        const status = getStatusCode(set.status || 200)\n        log.log(\n          'INFO',\n          request,\n          {\n            status,\n            message: String(set.headers?.['x-message'] || '')\n          },\n          storeData\n        )\n      }\n    })\n    .onError({ as: 'global' }, async ({ request, error, set, store }) => {\n      const status = getStatusCode(set.status || 500)\n      await log.handleHttpError(\n        request,\n        { ...error, status } as HttpError,\n        store as StoreData\n      )\n    })\n}\n\nexport type {\n  HttpError,\n  LogData,\n  Logger,\n  LogixlysiaContext,\n  LogLevel,\n  LogRotationConfig,\n  Options,\n  RequestInfo,\n  StoreData,\n  Transport\n} from './interfaces'\nexport { createLogger, handleHttpError } from './logger'\nexport { logToTransports } from './output'\n",
    "import type { Options, Server } from '../interfaces'\n\nconst createBoxText = (text: string, width: number): string => {\n  const paddingLength = Math.max(0, (width - text.length) / 2)\n  const padding = ' '.repeat(paddingLength)\n  return `${padding}${text}${padding}`.padEnd(width)\n}\n\nexport default function startServer(config: Server, options?: Options): void {\n  const hostname = config?.hostname ?? 'localhost'\n  const port = config?.port ?? 3000\n  const protocol = config?.protocol ?? 'http'\n  const showBanner = options?.config?.startupMessageFormat !== 'simple'\n\n  if (showBanner) {\n    const title = 'Elysia with Logixlysia'\n    const message = `🦊 Elysia is running at ${protocol}://${hostname}:${port}`\n    const boxWidth = Math.max(title.length, message.length) + 4\n    const border = '─'.repeat(boxWidth)\n    const emptyLine = createBoxText('', boxWidth)\n\n    console.log(`\n      ┌${border}┐\n      │${emptyLine}│\n      │${createBoxText(title, boxWidth)}│\n      │${emptyLine}│\n      │${createBoxText(message, boxWidth)}│\n      │${emptyLine}│\n      └${border}┘\n    `)\n  } else {\n    console.log(`🦊 Elysia is running at ${protocol}://${hostname}:${port}`)\n  }\n}\n",
    "import chalk from 'chalk'\nimport { StatusMap } from 'elysia'\n\nexport function getStatusCode(status: string | number): number {\n  if (typeof status === 'number') {\n    return status\n  }\n  return (StatusMap as Record<string, number>)[status] || 500\n}\n\nexport default function statusString(\n  status: number,\n  useColors: boolean\n): string {\n  const statusStr = status.toString()\n  if (!useColors) {\n    return statusStr\n  }\n\n  if (status >= 500) {\n    return chalk.red(statusStr)\n  }\n  if (status >= 400) {\n    return chalk.yellow(statusStr)\n  }\n  if (status >= 300) {\n    return chalk.cyan(statusStr)\n  }\n  if (status >= 200) {\n    return chalk.green(statusStr)\n  }\n  return chalk.white(statusStr)\n}\n",
    "import type { Logger as PinoLogger } from 'pino'\nimport pino from 'pino'\nimport type {\n  LogData,\n  Logger,\n  LogLevel,\n  Options,\n  PinoConfig,\n  RequestInfo,\n  StoreData\n} from '../interfaces'\nimport { logToFile, logToTransports } from '../output'\nimport { buildLogMessage } from './build-log-message'\nimport { filterLog } from './filter'\nimport { handleHttpError } from './handle-http-error'\n\nfunction getMetrics(): LogData['metrics'] {\n  const memoryUsage = process.memoryUsage().heapUsed / 1024 / 1024 // MB\n  const cpuUsage = process.cpuUsage()\n\n  return {\n    memoryUsage,\n    cpuUsage: cpuUsage.user / 1_000_000 // convert to seconds\n  }\n}\n\nfunction buildPinoConfig(pinoConfig: PinoConfig) {\n  return {\n    level: pinoConfig.level || 'info',\n    timestamp: pinoConfig.timestamp ?? true,\n    messageKey: pinoConfig.messageKey || 'msg',\n    errorKey: pinoConfig.errorKey || 'err',\n    base: pinoConfig.base || { pid: process.pid }\n  } as const\n}\n\nfunction createPrettyTransport(prettyPrint: boolean | object) {\n  return pino.transport({\n    target: 'pino-pretty',\n    options: {\n      colorize: true,\n      translateTime: 'HH:MM:ss Z',\n      ignore: 'pid,hostname',\n      ...(typeof prettyPrint === 'object' ? prettyPrint : {})\n    }\n  })\n}\n\nfunction createPinoInstance(options?: Options): PinoLogger {\n  const pinoConfig = options?.config?.pino || {}\n  const { prettyPrint, transport, ...rest } = pinoConfig\n  const config = {\n    ...buildPinoConfig(pinoConfig),\n    ...rest\n  }\n\n  if (prettyPrint && process.env.NODE_ENV !== 'production') {\n    return pino(config, createPrettyTransport(prettyPrint))\n  }\n\n  if (transport) {\n    if (typeof transport === 'object' && 'target' in transport) {\n      return pino(\n        config,\n        pino.transport(\n          transport as pino.TransportSingleOptions | pino.TransportMultiOptions\n        )\n      )\n    }\n    console.warn(\n      'Invalid transport configuration provided, falling back to default'\n    )\n  }\n\n  return pino(config)\n}\n\nfunction mapLogLevelToPino(level: LogLevel): string {\n  switch (level.toUpperCase()) {\n    case 'DEBUG':\n      return 'debug'\n    case 'INFO':\n      return 'info'\n    case 'WARNING':\n    case 'WARN':\n      return 'warn'\n    case 'ERROR':\n      return 'error'\n    default:\n      console.warn(`Unknown log level \"${level}\", defaulting to \"info\"`)\n      return 'info'\n  }\n}\n\nfunction emitPinoLog(\n  pinoLogger: PinoLogger,\n  level: LogLevel,\n  logObject: Record<string, unknown>,\n  message: string,\n  options?: Options\n): void {\n  const pinoLevel = mapLogLevelToPino(level)\n  const logMethod = pinoLogger[pinoLevel as keyof PinoLogger] as (\n    ...args: unknown[]\n  ) => void\n  const hasCustomPinoOutput = Boolean(\n    options?.config?.pino?.transport || options?.config?.pino?.prettyPrint\n  )\n  const shouldEmitPino =\n    !options?.config?.useTransportsOnly || hasCustomPinoOutput\n\n  if (shouldEmitPino && typeof logMethod === 'function') {\n    logMethod.call(pinoLogger, logObject, message)\n  }\n}\n\nasync function handleOutputs(\n  level: LogLevel,\n  request: RequestInfo,\n  data: LogData,\n  store: StoreData,\n  options?: Options,\n  logMessage?: string\n): Promise<void> {\n  const promises: Promise<void>[] = []\n\n  // Handle console logging\n  if (\n    !(\n      options?.config?.useTransportsOnly ||\n      options?.config?.disableInternalLogger\n    )\n  ) {\n    console.log(logMessage)\n  }\n\n  // Handle file logging\n  if (\n    !options?.config?.useTransportsOnly &&\n    options?.config?.logFilePath &&\n    !options?.config?.disableFileLogging\n  ) {\n    promises.push(\n      logToFile(\n        options.config.logFilePath,\n        level,\n        request,\n        data,\n        store,\n        options\n      )\n    )\n  }\n\n  // Handle transport logging\n  if (options?.config?.transports?.length) {\n    promises.push(logToTransports(level, request, data, store, options))\n  }\n\n  await Promise.all(promises)\n}\n\nasync function log(\n  pinoLogger: PinoLogger,\n  level: LogLevel,\n  request: RequestInfo,\n  data: LogData,\n  store: StoreData,\n  options?: Options\n): Promise<void> {\n  if (!filterLog(level, data.status || 200, request.method, options)) {\n    return\n  }\n\n  if (!data.metrics) {\n    data.metrics = getMetrics()\n  }\n\n  if (level === 'ERROR' && !data.stack) {\n    const err = new Error(data.message || 'Unknown error')\n    data.stack = err.stack\n  }\n\n  const errorKey = options?.config?.pino?.errorKey || 'err'\n  const err =\n    level === 'ERROR' && data.stack\n      ? {\n          name: 'Error',\n          message: data.message || 'Unknown error',\n          stack: data.stack\n        }\n      : undefined\n\n  const forwardedFor = request.headers.get('x-forwarded-for')\n  const clientIp =\n    options?.config?.ip && forwardedFor\n      ? forwardedFor.split(',')[0]?.trim()\n      : undefined\n\n  const logObject = {\n    method: request.method,\n    url: request.url,\n    status: data.status,\n    message: data.message,\n    context: data.context,\n    metrics: data.metrics,\n    duration: Number(process.hrtime.bigint() - store.beforeTime) / 1_000_000,\n    ip: clientIp,\n    [errorKey]: err\n  }\n\n  emitPinoLog(\n    pinoLogger,\n    level,\n    logObject,\n    data.message || 'Request processed',\n    options\n  )\n\n  const logMessage = buildLogMessage(level, request, data, store, options, true)\n\n  await handleOutputs(level, request, data, store, options, logMessage)\n}\n\nexport function createLogger(options?: Options): Logger {\n  const pinoLogger = createPinoInstance(options)\n\n  const logger: Logger = {\n    store: undefined,\n    pino: pinoLogger, // Expose the Pino instance\n    log: (level, request, data, store) =>\n      log(pinoLogger, level, request, data, store, options),\n    handleHttpError: async (request, error, store) =>\n      await handleHttpError(request, error, store, options),\n    customLogFormat: options?.config?.customLogFormat,\n    info: (request, message, context, store) => {\n      const storeData = store ||\n        logger.store || { beforeTime: process.hrtime.bigint() }\n      storeData.hasCustomLog = true\n      return log(\n        pinoLogger,\n        'INFO',\n        request,\n        { message, context, status: 200 },\n        storeData,\n        options\n      )\n    },\n    error: (request, message, context, store) => {\n      const storeData = store ||\n        logger.store || { beforeTime: process.hrtime.bigint() }\n      storeData.hasCustomLog = true\n      return log(\n        pinoLogger,\n        'ERROR',\n        request,\n        { message, context, status: 500 },\n        storeData,\n        options\n      )\n    },\n    warn: (request, message, context, store) => {\n      const storeData = store ||\n        logger.store || { beforeTime: process.hrtime.bigint() }\n      storeData.hasCustomLog = true\n      return log(\n        pinoLogger,\n        'WARNING',\n        request,\n        { message, context, status: 200 },\n        storeData,\n        options\n      )\n    },\n    debug: (request, message, context, store) => {\n      const storeData = store ||\n        logger.store || { beforeTime: process.hrtime.bigint() }\n      storeData.hasCustomLog = true\n      return log(\n        pinoLogger,\n        'DEBUG',\n        request,\n        { message, context, status: 200 },\n        storeData,\n        options\n      )\n    }\n  }\n  return logger\n}\n",
    "import chalk from 'chalk'\n\nimport {\n  durationString,\n  formatTimestamp,\n  logString,\n  methodString,\n  pathString,\n  statusString\n} from '../helpers'\nimport type {\n  LogComponents,\n  LogData,\n  LogLevel,\n  Options,\n  RequestInfo,\n  StoreData\n} from '../interfaces'\n\nconst defaultLogFormat =\n  '🦊 {now} {level} {duration} {method} {pathname} {status} {message} {context} {ip}'\n\nfunction shouldUseColors(useColors: boolean, options?: Options): boolean {\n  if (options?.config?.useColors !== undefined) {\n    return options.config.useColors && process.env.NO_COLOR === undefined\n  }\n  return useColors && process.env.NO_COLOR === undefined\n}\n\nexport function buildLogMessage(\n  level: LogLevel,\n  request: RequestInfo,\n  data: LogData,\n  store: StoreData,\n  options?: Options,\n  useColors = true\n): string {\n  const actuallyUseColors = shouldUseColors(useColors, options)\n  const now = new Date()\n  const components: LogComponents = {\n    now: actuallyUseColors\n      ? chalk.bgYellow(\n          chalk.black(formatTimestamp(now, options?.config?.timestamp))\n        )\n      : formatTimestamp(now, options?.config?.timestamp),\n    epoch: Math.floor(now.getTime() / 1000).toString(),\n    level: logString(level, useColors),\n    duration: durationString(store.beforeTime, useColors),\n    method: methodString(request.method, useColors),\n    pathname: pathString(request),\n    status: statusString(data.status || 200, useColors),\n    message: data.message || '',\n    context: data.context\n      ? (() => {\n          try {\n            return JSON.stringify(data.context)\n          } catch (error) {\n            return `[Error serializing context: ${error instanceof Error ? error.message : 'Unknown error'}]`\n          }\n        })()\n      : '',\n    ip:\n      options?.config?.ip && request.headers.get('x-forwarded-for')\n        ? `IP: ${request.headers.get('x-forwarded-for')}`\n        : ''\n  }\n\n  const logFormat = options?.config?.customLogFormat || defaultLogFormat\n\n  return logFormat.replace(/{(\\w+)}/g, (_, key: string) => {\n    if (key in components) {\n      return components[key as keyof LogComponents] || ''\n    }\n    return ''\n  })\n}\n",
    "import chalk from 'chalk'\n\nimport type { ColorMap } from '../interfaces'\n\nexport const LogLevelColorMap: ColorMap = {\n  INFO: chalk.bgGreen.black,\n  WARNING: chalk.bgYellow.black,\n  ERROR: chalk.bgRed.black\n}\n\nexport const HttpMethodColorMap: ColorMap = {\n  GET: chalk.green,\n  POST: chalk.yellow,\n  PUT: chalk.blue,\n  PATCH: chalk.magentaBright,\n  DELETE: chalk.red,\n  HEAD: chalk.cyan,\n  OPTIONS: chalk.magenta\n}\n",
    "import chalk from 'chalk'\n\nconst timeUnits = [\n  { unit: 's', threshold: 1e9, decimalPlaces: 2 },\n  { unit: 'ms', threshold: 1e6, decimalPlaces: 0 },\n  { unit: 'µs', threshold: 1e3, decimalPlaces: 0 },\n  { unit: 'ns', threshold: 1, decimalPlaces: 0 }\n]\n\nexport default function durationString(\n  beforeTime: bigint,\n  useColors: boolean\n): string {\n  const nanoseconds = Number(process.hrtime.bigint() - beforeTime)\n\n  for (const { unit, threshold, decimalPlaces } of timeUnits) {\n    if (nanoseconds >= threshold) {\n      const value = (nanoseconds / threshold).toFixed(decimalPlaces)\n      const timeStr = `${value}${unit}`.padStart(8).padEnd(16)\n      return useColors ? chalk.gray(timeStr) : timeStr\n    }\n  }\n\n  return useColors\n    ? chalk.gray('0ns'.padStart(8).padEnd(16))\n    : '0ns'.padStart(8).padEnd(16)\n}\n",
    "import type { LogLevel } from '../interfaces'\nimport { LogLevelColorMap } from './color-mapping'\n\nexport default function logString(level: LogLevel, useColors: boolean): string {\n  const levelStr = level.toUpperCase()\n  return useColors\n    ? LogLevelColorMap[levelStr]?.(levelStr.padEnd(7)) || levelStr\n    : levelStr.padEnd(7)\n}\n",
    "import { HttpMethodColorMap } from './color-mapping'\n\nexport default function methodString(\n  method: string,\n  useColors: boolean\n): string {\n  const colorFunction = HttpMethodColorMap[method]\n  return useColors && colorFunction\n    ? colorFunction(method.padEnd(7))\n    : method.padEnd(7)\n}\n",
    "import type { RequestInfo } from '../interfaces'\n\nexport default function pathString(\n  requestInfo: RequestInfo\n): string | undefined {\n  try {\n    return new URL(requestInfo.url).pathname\n  } catch {\n    return\n  }\n}\n",
    "import type { TimestampConfig } from '../interfaces'\n\n// const DEFAULT_TIMESTAMP_FORMAT = 'yyyy-mm-dd HH:MM:ss'\nconst SYS_TIME = 'SYS:STANDARD'\n\nconst pad = (n: number): string => n.toString().padStart(2, '0')\n\nfunction formatSystemTime(date: Date): string {\n  const year = date.getFullYear()\n  const month = pad(date.getMonth() + 1)\n  const day = pad(date.getDate())\n  const hours = pad(date.getHours())\n  const minutes = pad(date.getMinutes())\n  const seconds = pad(date.getSeconds())\n  const ms = date.getMilliseconds().toString().padStart(3, '0')\n\n  return `${year}-${month}-${day} ${hours}:${minutes}:${seconds}.${ms}`\n}\n\nfunction formatCustomTime(date: Date, format: string): string {\n  const tokens: { [key: string]: string | number } = {\n    yyyy: date.getFullYear(),\n    yy: date.getFullYear().toString().slice(-2),\n    mm: pad(date.getMonth() + 1),\n    dd: pad(date.getDate()),\n    HH: pad(date.getHours()),\n    MM: pad(date.getMinutes()),\n    ss: pad(date.getSeconds()),\n    SSS: pad(date.getMilliseconds()),\n    Z: -date.getTimezoneOffset() / 60\n  }\n\n  return format.replace(/yyyy|yy|mm|dd|HH|MM|ss|SSS|Z/g, match =>\n    (tokens[match] ?? '').toString()\n  )\n}\n\nexport function formatTimestamp(date: Date, config?: TimestampConfig): string {\n  if (!config?.translateTime) {\n    return date.toISOString()\n  }\n\n  if (config.translateTime === true || config.translateTime === SYS_TIME) {\n    return formatSystemTime(date)\n  }\n\n  return formatCustomTime(date, config.translateTime)\n}\n",
    "import type {\n  LogData,\n  LogLevel,\n  Options,\n  RequestInfo,\n  StoreData\n} from '../interfaces'\nimport { buildLogMessage } from '../logger/build-log-message'\n\nexport async function logToTransports(\n  level: LogLevel,\n  request: RequestInfo,\n  data: LogData,\n  store: StoreData,\n  options?: Options\n): Promise<void> {\n  if (!options?.config?.transports || options.config.transports.length === 0) {\n    return\n  }\n\n  const message = buildLogMessage(level, request, data, store, options, false)\n\n  const promises = options.config.transports.map(transport =>\n    transport.log(level, message, { request, data, store })\n  )\n\n  await Promise.all(promises)\n}\n",
    "import { promises as fs } from 'node:fs'\nimport { dirname } from 'node:path'\n\nimport type {\n  LogData,\n  LogLevel,\n  Options,\n  RequestInfo,\n  StoreData\n} from '../interfaces'\nimport { buildLogMessage } from '../logger/build-log-message'\nimport { parseInterval, shouldRotateByTime } from '../utils/rotation'\nimport { performRotation, shouldRotate } from './rotation-manager'\n\nconst dirCache = new Set<string>()\n\nasync function ensureDirectoryExists(filePath: string): Promise<void> {\n  const dir = dirname(filePath)\n  if (!dirCache.has(dir)) {\n    await fs.mkdir(dir, { recursive: true })\n    dirCache.add(dir)\n  }\n}\n\n/**\n * Check if rotation is needed and perform it\n */\nasync function checkAndRotate(\n  filePath: string,\n  options?: Options\n): Promise<void> {\n  const rotationConfig = options?.config?.logRotation\n  if (!rotationConfig) {\n    return\n  }\n\n  let needsRotation = false\n\n  // Check size-based rotation\n  if (rotationConfig.maxSize) {\n    needsRotation = await shouldRotate(filePath, rotationConfig)\n  }\n\n  // Check time-based rotation\n  if (!needsRotation && rotationConfig.interval) {\n    try {\n      const intervalMs = parseInterval(rotationConfig.interval)\n      needsRotation = await shouldRotateByTime(filePath, intervalMs)\n    } catch (error) {\n      console.error('Invalid interval format in log rotation config:', error)\n    }\n  }\n\n  // Perform rotation if needed\n  if (needsRotation) {\n    await performRotation(filePath, rotationConfig)\n  }\n}\n\nexport async function logToFile(\n  filePath: string,\n  level: LogLevel,\n  request: RequestInfo,\n  data: LogData,\n  store: StoreData,\n  options?: Options\n): Promise<void> {\n  await ensureDirectoryExists(filePath)\n\n  // Check and perform rotation if needed\n  await checkAndRotate(filePath, options)\n\n  const logMessage = `${buildLogMessage(level, request, data, store, options, false)}\\n`\n  await fs.appendFile(filePath, logMessage, { flag: 'a' })\n}\n",
    "import { promises as fs } from 'node:fs'\nimport { basename, dirname, join } from 'node:path'\n\nexport interface ParsedRetention {\n  type: 'count' | 'time'\n  value: number\n}\n\n// Regex patterns defined at top level for performance\nconst SIZE_PATTERN = /^(\\d+(?:\\.\\d+)?)\\s*([kmg])?b?$/\nconst INTERVAL_PATTERN = /^(\\d+)\\s*([hdw])$/\nconst RETENTION_PATTERN = /^(\\d+)\\s*d$/\n\n/**\n * Parse size string to bytes\n * Supports: '10m', '1g', '100k', or raw number\n */\nexport function parseSize(size: string | number): number {\n  if (typeof size === 'number') {\n    return size\n  }\n\n  const units: Record<string, number> = {\n    k: 1024,\n    m: 1024 * 1024,\n    g: 1024 * 1024 * 1024\n  }\n\n  const match = size.toLowerCase().match(SIZE_PATTERN)\n  if (!match?.[1]) {\n    throw new Error(`Invalid size format: ${size}`)\n  }\n\n  const value = Number.parseFloat(match[1])\n  const unit = match[2] ?? ''\n\n  return Math.floor(value * (units[unit] ?? 1))\n}\n\n/**\n * Parse interval string to milliseconds\n * Supports: '1h', '1d', '1w'\n */\nexport function parseInterval(interval: string): number {\n  const units: Record<string, number> = {\n    h: 60 * 60 * 1000, // hour\n    d: 24 * 60 * 60 * 1000, // day\n    w: 7 * 24 * 60 * 60 * 1000 // week\n  }\n\n  const match = interval.toLowerCase().match(INTERVAL_PATTERN)\n  if (!match) {\n    throw new Error(`Invalid interval format: ${interval}`)\n  }\n\n  const valueStr = match[1]\n  const unit = match[2] as string\n\n  if (!valueStr) {\n    throw new Error(`Invalid interval format: ${interval}`)\n  }\n  if (!unit) {\n    throw new Error(`Invalid interval format: ${interval}`)\n  }\n\n  const value = Number.parseInt(valueStr, 10)\n  return value * (units[unit] ?? 0)\n}\n\n/**\n * Parse retention string or number\n * Returns object with type (count or time) and value\n */\nexport function parseRetention(retention: string | number): ParsedRetention {\n  if (typeof retention === 'number') {\n    return { type: 'count', value: retention }\n  }\n\n  // Check if it's a time-based retention (e.g., '7d', '30d')\n  const match = retention.toLowerCase().match(RETENTION_PATTERN)\n  if (match?.[1]) {\n    const days = Number.parseInt(match[1], 10)\n    return { type: 'time', value: days * 24 * 60 * 60 * 1000 } // convert to milliseconds\n  }\n\n  throw new Error(`Invalid retention format: ${retention}`)\n}\n\n/**\n * Check if file should be rotated based on size\n */\nexport async function shouldRotateBySize(\n  filePath: string,\n  maxSize: number\n): Promise<boolean> {\n  try {\n    const stats = await fs.stat(filePath)\n    return stats.size >= maxSize\n  } catch {\n    // File doesn't exist or can't be accessed\n    return false\n  }\n}\n\n/**\n * Get the last rotation time for a file\n * Uses file modification time as a proxy\n */\nasync function getLastRotationTime(filePath: string): Promise<number> {\n  try {\n    const stats = await fs.stat(filePath)\n    return stats.mtime.getTime()\n  } catch {\n    // File doesn't exist, use current time\n    return Date.now()\n  }\n}\n\n// Track last rotation times in memory to avoid excessive file checks\nconst rotationTimeCache = new Map<string, number>()\n\n/**\n * Check if file should be rotated based on time interval\n */\nexport async function shouldRotateByTime(\n  filePath: string,\n  interval: number\n): Promise<boolean> {\n  const now = Date.now()\n\n  // Check cache first\n  const cachedTime = rotationTimeCache.get(filePath)\n  if (cachedTime !== undefined) {\n    return now - cachedTime >= interval\n  }\n\n  // Get last rotation time from file\n  const lastRotation = await getLastRotationTime(filePath)\n  rotationTimeCache.set(filePath, lastRotation)\n\n  return now - lastRotation >= interval\n}\n\n/**\n * Update the last rotation time in cache\n */\nexport function updateRotationTime(filePath: string): void {\n  rotationTimeCache.set(filePath, Date.now())\n}\n\n/**\n * Get rotated files for a given log file\n */\nexport async function getRotatedFiles(filePath: string): Promise<string[]> {\n  const dir = dirname(filePath)\n  const baseName = basename(filePath)\n\n  try {\n    const files = await fs.readdir(dir)\n    // Match files like: app.log.2025-10-10, app.log.2025-10-10.gz\n    const rotatedPattern = new RegExp(\n      `^${baseName.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&')}\\\\.\\\\d{4}-\\\\d{2}-\\\\d{2}(-\\\\d{2}-\\\\d{2}-\\\\d{2})?(\\\\.gz)?$`\n    )\n    const rotatedFiles = files\n      .filter(file => rotatedPattern.test(file))\n      .map(file => join(dir, file))\n\n    // Sort by modification time (newest first)\n    const filesWithStats = await Promise.all(\n      rotatedFiles.map(async file => {\n        const stats = await fs.stat(file)\n        return { file, mtime: stats.mtime.getTime() }\n      })\n    )\n\n    return filesWithStats\n      .sort((a, b) => b.mtime - a.mtime)\n      .map(item => item.file)\n  } catch {\n    return []\n  }\n}\n",
    "import { createReadStream, createWriteStream, promises as fs } from 'node:fs'\nimport { pipeline } from 'node:stream/promises'\nimport { createGzip } from 'node:zlib'\n\nimport type { LogRotationConfig } from '../interfaces'\nimport {\n  getRotatedFiles,\n  parseInterval,\n  parseRetention,\n  parseSize,\n  updateRotationTime\n} from '../utils/rotation'\n\n/**\n * Generate a rotated file name with timestamp\n * Example: app.log -> app.log.2025-10-10-14-30-45\n */\nexport function getRotatedFileName(filePath: string, timestamp: Date): string {\n  const year = timestamp.getFullYear()\n  const month = String(timestamp.getMonth() + 1).padStart(2, '0')\n  const day = String(timestamp.getDate()).padStart(2, '0')\n  const hours = String(timestamp.getHours()).padStart(2, '0')\n  const minutes = String(timestamp.getMinutes()).padStart(2, '0')\n  const seconds = String(timestamp.getSeconds()).padStart(2, '0')\n\n  return `${filePath}.${year}-${month}-${day}-${hours}-${minutes}-${seconds}`\n}\n\n/**\n * Rotate a log file by renaming it with a timestamp\n */\nexport async function rotateFile(filePath: string): Promise<string> {\n  try {\n    // Check if file exists and has content\n    const stats = await fs.stat(filePath)\n    if (stats.size === 0) {\n      // Don't rotate empty files\n      return ''\n    }\n\n    const rotatedPath = getRotatedFileName(filePath, new Date())\n    await fs.rename(filePath, rotatedPath)\n    updateRotationTime(filePath)\n\n    return rotatedPath\n  } catch (error) {\n    // File doesn't exist or can't be rotated\n    if ((error as NodeJS.ErrnoException).code !== 'ENOENT') {\n      console.error(`Failed to rotate log file ${filePath}:`, error)\n    }\n    return ''\n  }\n}\n\n/**\n * Compress a file using gzip\n */\nexport async function compressFile(filePath: string): Promise<void> {\n  try {\n    const compressedPath = `${filePath}.gz`\n    const source = createReadStream(filePath)\n    const destination = createWriteStream(compressedPath)\n    const gzip = createGzip()\n\n    await pipeline(source, gzip, destination)\n\n    // Delete the original file after successful compression\n    await fs.unlink(filePath)\n  } catch (error) {\n    console.error(`Failed to compress file ${filePath}:`, error)\n  }\n}\n\n/**\n * Clean old rotated files based on retention policy\n */\nexport async function cleanOldFiles(\n  filePath: string,\n  config: LogRotationConfig\n): Promise<void> {\n  if (!config.maxFiles) {\n    return\n  }\n\n  try {\n    const rotatedFiles = await getRotatedFiles(filePath)\n    if (rotatedFiles.length === 0) {\n      return\n    }\n\n    const retention = parseRetention(config.maxFiles)\n\n    if (retention.type === 'count') {\n      // Keep only the specified number of files\n      const filesToDelete = rotatedFiles.slice(retention.value)\n      await Promise.all(filesToDelete.map(file => fs.unlink(file)))\n    } else if (retention.type === 'time') {\n      // Delete files older than the specified time\n      const cutoffTime = Date.now() - retention.value\n      const filesToDelete = await Promise.all(\n        rotatedFiles.map(async file => {\n          const stats = await fs.stat(file)\n          return stats.mtime.getTime() < cutoffTime ? file : null\n        })\n      )\n\n      await Promise.all(\n        filesToDelete\n          .filter((file): file is string => file !== null)\n          .map(file => fs.unlink(file))\n      )\n    }\n  } catch (error) {\n    console.error(`Failed to clean old log files for ${filePath}:`, error)\n  }\n}\n\n/**\n * Perform complete rotation: rotate, compress (if enabled), and clean old files\n */\nexport async function performRotation(\n  filePath: string,\n  config: LogRotationConfig\n): Promise<void> {\n  try {\n    // Rotate the file\n    const rotatedPath = await rotateFile(filePath)\n\n    if (!rotatedPath) {\n      return\n    }\n\n    // Compress if enabled (defaults to gzip)\n    if (config.compress) {\n      await compressFile(rotatedPath)\n    }\n\n    // Clean old files\n    await cleanOldFiles(filePath, config)\n  } catch (error) {\n    console.error(`Failed to perform rotation for ${filePath}:`, error)\n  }\n}\n\n/**\n * Check if rotation is needed based on configuration\n */\nexport async function shouldRotate(\n  filePath: string,\n  config: LogRotationConfig\n): Promise<boolean> {\n  try {\n    // Check file existence\n    await fs.access(filePath)\n  } catch {\n    // File doesn't exist, no rotation needed\n    return false\n  }\n\n  // Check size-based rotation\n  if (config.maxSize) {\n    try {\n      const maxSizeBytes = parseSize(config.maxSize)\n      const stats = await fs.stat(filePath)\n      if (stats.size >= maxSizeBytes) {\n        return true\n      }\n    } catch (error) {\n      console.error(`Failed to check file size for ${filePath}:`, error)\n    }\n  }\n\n  // Check time-based rotation\n  if (config.interval) {\n    try {\n      const intervalMs = parseInterval(config.interval)\n      const stats = await fs.stat(filePath)\n      const age = Date.now() - stats.mtimeMs\n      if (age >= intervalMs) {\n        return true\n      }\n    } catch (error) {\n      // Log parse or stat errors and treat as no-op\n      console.error(`Failed to check file age for ${filePath}:`, error)\n    }\n  }\n\n  return false\n}\n",
    "import type { LogLevel, Options } from '../interfaces'\n\nconst checkFilter = (filterValue: unknown, value: unknown) =>\n  Array.isArray(filterValue)\n    ? filterValue.includes(value)\n    : filterValue === value\n\nexport function filterLog(\n  logLevel: LogLevel,\n  status: number,\n  method: string,\n  options?: Options\n): boolean {\n  const filter = options?.config?.logFilter\n  if (!filter) {\n    return true\n  }\n\n  return (\n    (!filter.level || checkFilter(filter.level, logLevel)) &&\n    (!filter.status || checkFilter(filter.status, status)) &&\n    (!filter.method || checkFilter(filter.method, method))\n  )\n}\n",
    "import type { HttpError, Options, RequestInfo, StoreData } from '../interfaces'\nimport { logToFile, logToTransports } from '../output'\nimport { buildLogMessage } from './build-log-message'\n\nexport async function handleHttpError(\n  request: RequestInfo,\n  error: HttpError,\n  store: StoreData,\n  options?: Options\n): Promise<void> {\n  const statusCode = error.status || 500\n  const logData = {\n    status: statusCode,\n    message: error.message,\n    stack: error.stack\n  }\n\n  const promises: Promise<void>[] = []\n\n  // Handle console logging\n  if (\n    !(\n      options?.config?.useTransportsOnly ||\n      options?.config?.disableInternalLogger\n    )\n  ) {\n    console.error(buildLogMessage('ERROR', request, logData, store, options))\n  }\n\n  // Handle file logging\n  if (\n    !options?.config?.useTransportsOnly &&\n    options?.config?.logFilePath &&\n    !options?.config?.disableFileLogging\n  ) {\n    promises.push(\n      logToFile(\n        options.config.logFilePath,\n        'ERROR',\n        request,\n        logData,\n        store,\n        options\n      )\n    )\n  }\n\n  // Handle transport logging\n  if (options?.config?.transports?.length) {\n    promises.push(logToTransports('ERROR', request, logData, store, options))\n  }\n\n  await Promise.all(promises)\n}\n"
  ],
  "mappings": "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",
  "debugId": "62A87E578A304FBB64756E2164756E21",
  "names": []
}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "logixlysia",
|
|
3
|
-
"version": "5.
|
|
3
|
+
"version": "5.3.0",
|
|
4
4
|
"description": "๐ฆ Logixlysia is a logger for Elysia",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"main": "dist/index.js",
|
|
@@ -81,16 +81,18 @@
|
|
|
81
81
|
"middleware"
|
|
82
82
|
],
|
|
83
83
|
"dependencies": {
|
|
84
|
-
"chalk": "^5.
|
|
85
|
-
"elysia": "^1.
|
|
84
|
+
"chalk": "^5.6.2",
|
|
85
|
+
"elysia": "^1.4.10",
|
|
86
|
+
"pino": "^10.0.0",
|
|
87
|
+
"pino-pretty": "^13.1.2"
|
|
86
88
|
},
|
|
87
89
|
"devDependencies": {
|
|
88
90
|
"@biomejs/biome": "2.0.5",
|
|
89
|
-
"bun-types": "^1.
|
|
90
|
-
"bunup": "^0.
|
|
91
|
-
"globals": "^16.
|
|
92
|
-
"husky": "^9.1.
|
|
93
|
-
"ultracite": "
|
|
91
|
+
"bun-types": "^1.2.23",
|
|
92
|
+
"bunup": "^0.14.20",
|
|
93
|
+
"globals": "^16.4.0",
|
|
94
|
+
"husky": "^9.1.7",
|
|
95
|
+
"ultracite": "5.0.10"
|
|
94
96
|
},
|
|
95
97
|
"peerDependencies": {
|
|
96
98
|
"typescript": "^5.2.2"
|