logixlysia 6.3.0 → 6.3.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.d.ts CHANGED
@@ -74,6 +74,11 @@ interface Options {
74
74
  disableFileLogging?: boolean;
75
75
  logFilePath?: string;
76
76
  logRotation?: LogRotationConfig;
77
+ /**
78
+ * Automatically redact sensitive information (PII) from logs.
79
+ * Masks emails, IP addresses, Luhn-valid payment card numbers, and JWTs in strings and deeply nested objects.
80
+ */
81
+ autoRedact?: boolean;
77
82
  pino?: (PinoLoggerOptions & {
78
83
  prettyPrint?: PrettyPrintConfig;
79
84
  }) | undefined;
package/dist/index.js CHANGED
@@ -1,10 +1,10 @@
1
- import{Elysia as Q$}from"elysia";import{createRequire as C}from"node:module";var j0=(()=>{try{return C(import.meta.url)("elysia/package.json")}catch{return{}}})(),J0=(()=>{try{return C(import.meta.url)("../../package.json")}catch{return{}}})(),Q0=($,b)=>{if($.length>=b)return $.slice(0,b);let I=Math.floor((b-$.length)/2),j=b-$.length-I;return`${" ".repeat(I)}${$}${" ".repeat(j)}`},n=($,b)=>{let I=j0.version,j=I?`Elysia v${I}`:"Elysia",G=[{kind:"empty"},{kind:"center",text:j},{kind:"empty"},{kind:"left",text:$}];if(b)G.push({kind:"left",text:b});G.push({kind:"empty"});let R=Math.max(j.length,$.length,...b?[b.length]:[0])+4,J=`┌${"─".repeat(R)}┐`,z=`└${"─".repeat(R)}┘`,N=`│${" ".repeat(R)}│`,Y=[J];for(let F of G){if(F.kind==="empty"){Y.push(N);continue}if(F.kind==="center"){Y.push(`│${Q0(F.text,R)}│`);continue}let V=F.text,H=Math.max(0,R-V.length-4);Y.push(`│ ${V}${" ".repeat(H)} │`)}return Y.push(z),Y.join(`
2
- `)},h=()=>{let $=J0.version;if(!$)return null;return` logixlysia v${$}`};var q=($,b)=>{if(!(b.config?.showStartupMessage??!0))return;let{port:j,hostname:G,protocol:X}=$;if(j===void 0||!G||!X)return;let R=`${X}://${G}:${j}`,J=`\uD83E\uDD8A Elysia is running at ${R}`,z=`\uD83E\uDD8A ${R}`;if((b.config?.startupMessageFormat??"banner")==="simple"){console.log(J);return}console.log(n(z,h()))};import J$ from"pino";var S=(...$)=>{let b=typeof $[0]==="string"?{level:$[0],request:$[1],data:$[2],store:$[3],options:$[4]}:$[0],{level:I,request:j,data:G,store:X,options:R}=b,J=R.config?.transports??[];if(J.length===0)return;let z=typeof G.message==="string"?G.message:"",N={request:{method:j.method,url:j.url},...G,beforeTime:X.beforeTime};for(let Y of J)try{let F=Y.log(I,z,N);if(F&&typeof F.catch==="function")F.catch(()=>{})}catch{}};import{appendFile as D0}from"node:fs/promises";import{dirname as M0}from"node:path";import{promises as Z0}from"node:fs";var g=async($)=>{await Z0.mkdir($,{recursive:!0})};import{promises as E}from"node:fs";import{promisify as V0}from"node:util";import{gzip as W0}from"node:zlib";import{promises as c}from"node:fs";import{basename as z0,dirname as K0}from"node:path";var N0=/^(\d+(?:\.\d+)?)(k|kb|m|mb|g|gb)$/i,B0=/^(\d+)(h|d|w)$/i,Y0=/\.(\d{4}-\d{2}-\d{2}-\d{2}-\d{2}-\d{2})(?:\.gz)?$/,v=($)=>{if(typeof $==="number")return $;let b=$.trim(),I=Number(b);if(Number.isFinite(I))return I;let j=b.match(N0);if(!j)throw Error(`Invalid size format: ${$}`);let G=Number(j[1]),X=j[2].toLowerCase(),R=1024;if(X.startsWith("m"))R=1048576;else if(X.startsWith("g"))R=1073741824;return Math.floor(G*R)},F0=($)=>{let b=$.trim().match(B0);if(!b)throw Error(`Invalid interval format: ${$}`);let I=Number(b[1]),j=b[2].toLowerCase(),G=3600000;if(j==="d")G=86400000;else if(j==="w")G=604800000;return I*G},p=($)=>{if(typeof $==="number")return{type:"count",value:$};return{type:"time",value:F0($)}},d=async($,b)=>{try{return(await c.stat($)).size>b}catch{return!1}},x=async($)=>{let b=K0($),I=z0($),j;try{j=await c.readdir(b)}catch{return[]}return j.filter((G)=>G.startsWith(`${I}.`)&&Y0.test(G)).map((G)=>`${b}/${G}`)};var w0=V0(W0),_=($)=>String($).padStart(2,"0"),H0=($,b)=>{let I=b.getFullYear(),j=_(b.getMonth()+1),G=_(b.getDate()),X=_(b.getHours()),R=_(b.getMinutes()),J=_(b.getSeconds());return`${$}.${I}-${j}-${G}-${X}-${R}-${J}`},E0=async($)=>{try{if((await E.stat($)).size===0)return""}catch{return""}let b=H0($,new Date);return await E.rename($,b),b},U0=async($)=>{let b=await E.readFile($),I=await w0(b);await E.writeFile(`${$}.gz`,I),await E.rm($,{force:!0})},u=async($,b)=>{if(b.maxSize===void 0)return!1;let I=v(b.maxSize);return await d($,I)},f0=async($,b)=>{let I=await x($);if(I.length<=b)return;let j=await Promise.all(I.map(async(X)=>({path:X,stat:await E.stat(X)})));j.sort((X,R)=>R.stat.mtimeMs-X.stat.mtimeMs);let G=j.slice(b);await Promise.all(G.map(({path:X})=>E.rm(X,{force:!0})))},A0=async($,b)=>{let I=await x($);if(I.length===0)return;let j=Date.now(),X=(await Promise.all(I.map(async(R)=>({path:R,stat:await E.stat(R)})))).filter(({stat:R})=>j-R.mtimeMs>b);await Promise.all(X.map(({path:R})=>E.rm(R,{force:!0})))},i=async($,b)=>{let I=await E0($);if(!I)return;if(b.compress===!0){if((b.compression??"gzip")==="gzip")await U0(I)}if(b.maxFiles!==void 0){let G=p(b.maxFiles);if(G.type==="count")await f0($,G.value);else await A0($,G.value)}};var k=async(...$)=>{let b=typeof $[0]==="string"?(()=>{let[Z,K,B,U,f,D]=$;return{filePath:Z,level:K,request:B,data:U,store:f,options:D}})():$[0],{filePath:I,level:j,request:G,data:X,store:R,options:J}=b,z=J.config,N=z?.useTransportsOnly===!0,Y=z?.disableFileLogging===!0;if(N||Y)return;let F=typeof X.message==="string"?X.message:"",V=R.beforeTime===BigInt(0)?0:Number(process.hrtime.bigint()-R.beforeTime)/1e6,H=`${j} ${V.toFixed(2)}ms ${G.method} ${new URL(G.url).pathname} ${F}
3
- `;await g(M0(I)),await D0(I,H,{encoding:"utf-8"});let W=z?.logRotation;if(!W)return;if(await u(I,W))await i(I,W)};import{STATUS_CODES as q0}from"node:http";import Q from"chalk";import{StatusMap as _0}from"elysia";var O0=/^\d+$/,T0=/[_-]+/g,L0=/([a-z0-9])([A-Z])/g,S0=/([A-Z])([A-Z][a-z])/g,k0=/['’]/g,y0=/[^a-z0-9\s]+/g,P0=/\s+/g,r=($)=>{let b=$.trim();if(!b)return"";return b.replace(T0," ").replace(L0,"$1 $2").replace(S0,"$1 $2").replace(k0,"").toLowerCase().replace(y0," ").replace(P0," ").trim()},m0=(()=>{let $=new Map;for(let[b,I]of Object.entries(_0))$.set(r(b),I);return $})(),o=($)=>{if(typeof $==="number"&&Number.isFinite($))return $;if(typeof $==="string"){let b=$.trim();if(O0.test(b))return Number(b);return m0.get(r(b))??500}return 500};var y=($)=>{let b="An error occurred";if($ instanceof Error)b=$.message;else if($&&typeof $==="object"&&"message"in $)b=$.message;else b=String($);return b};var O=($)=>String($).padStart(2,"0"),c0=($)=>String($).padStart(3,"0"),x0=500,C0=1000,n0=7,h0="{now} {service}{icon} {method} {pathname} {status} {duration} {message}{speed}",s=($)=>{let I=$.config?.useColors??!0,j=typeof process<"u"&&process.stdout?.isTTY===!0;return I&&j},g0=($,b)=>{if(!b)return $.toISOString();let I=String($.getFullYear()),j=O($.getMonth()+1),G=O($.getDate()),X=O($.getHours()),R=O($.getMinutes()),J=O($.getSeconds()),z=c0($.getMilliseconds());return b.replaceAll("yyyy",I).replaceAll("mm",j).replaceAll("dd",G).replaceAll("HH",X).replaceAll("MM",R).replaceAll("ss",J).replaceAll("SSS",z)},v0=($)=>{let b=$.headers.get("x-forwarded-for");if(b)return b.split(",")[0]?.trim()??"";return $.headers.get("x-real-ip")??""},p0=($)=>{if($>=1000){let b=$/1000;if(b>=10)return`${Math.round(b)}s`;let I=b.toFixed(1);return I.endsWith(".0")?`${Math.round(b)}s`:`${I}s`}if($>0&&$<1)return`${$.toFixed(2)}ms`;return`${Math.round($)}ms`},d0=($)=>{let b=$.config;return{slow:b?.slowThreshold??x0,verySlow:b?.verySlowThreshold??C0}},u0=($,b,I)=>{let j=p0($),{slow:G,verySlow:X}=d0(I),R=$>=X;if(!b)return{text:j,isVerySlow:R};let J=j;if($<G)J=Q.green(j);else if($<X)J=Q.yellow(j);else J=Q.red.bold(j);return{text:J,isVerySlow:R}},i0=($,b)=>{if(!$)return"";let I="⚡ slow";if(!b)return` ${I}`;return` ${Q.yellow(I)}`},r0=($,b)=>{if(!b)return"\uD83E\uDD8A";if($==="ERROR")return Q.bgRed.black(" \uD83E\uDD8A ");if($==="WARNING")return Q.bgYellow.black(" \uD83E\uDD8A ");if($==="DEBUG")return Q.bgBlue.black(" \uD83E\uDD8A ");return Q.bgGreen.black(" \uD83E\uDD8A ")},o0=($,b)=>{if(!b)return $;if($==="ERROR")return Q.bgRed.black($);if($==="WARNING")return Q.bgYellow.black($);if($==="DEBUG")return Q.bgBlue.black($);return Q.bgGreen.black($)},s0=($,b)=>{if(!b)return $;let I=$.toUpperCase();if(I==="GET")return Q.green.bold(I);if(I==="POST")return Q.blue.bold(I);if(I==="PUT")return Q.yellow.bold(I);if(I==="PATCH")return Q.yellowBright.bold(I);if(I==="DELETE")return Q.red.bold(I);if(I==="OPTIONS")return Q.cyan.bold(I);if(I==="HEAD")return Q.greenBright.bold(I);if(I==="TRACE")return Q.magenta.bold(I);if(I==="CONNECT")return Q.cyanBright.bold(I);return Q.white.bold(I)},l0=($,b)=>{if(!b)return $;let I=Number.parseInt($,10);if(!Number.isFinite(I))return $;if(I>=500)return Q.red($);if(I>=400)return Q.yellow($);if(I>=300)return Q.cyan($);if(I>=200)return Q.green($);return Q.gray($)},a0=($,b)=>{if(!b)return $;return Q.gray($)},t0=($,b)=>{if(!b)return $;return Q.whiteBright($)},e0=($)=>{return q0[$]??""},$$=($,b)=>{let I=$.config?.service?.trim();if(!I)return"";let j=`[${I}]`;if(!b)return`${j} `;return`${Q.dim(j)} `},b$=($)=>{if($===null)return"null";if($===void 0)return"undefined";if(typeof $==="string")return $;if(typeof $==="number"||typeof $==="boolean")return String($);if($ instanceof Error)return $.message;try{return JSON.stringify($)}catch{return String($)}},I$=($)=>$!==null&&typeof $==="object"&&!Array.isArray($)&&!($ instanceof Error)&&!($ instanceof Date),l=($,b,I)=>{let j=[];for(let[G,X]of Object.entries($)){let R=b?`${b}.${G}`:G;if(I$(X)&&I>1)j.push(...l(X,R,I-1));else j.push([R,b$(X)])}return j},R$=($,b)=>{if($.length===0)return[];let I=[],j=$.length-1;for(let G=0;G<$.length;G++){let X=G===j?"└─":"├─",R=$[G];if(!R)continue;let[J,z]=R,N=b?Q.cyan(J):J,Y=b?Q.white(z):z;I.push(` ${X} ${N} ${Y}`)}return I};var X$=($,b,I)=>{if(I.config?.showContextTree===!1)return[];let j=s(I),G=I.config?.contextDepth??1,X=[],R=b.context;if(R&&typeof R==="object"&&!Array.isArray(R)&&Object.keys(R).length>0&&G>=1)X.push(...l(R,"",G));if($==="ERROR"&&"error"in b&&b.error!==void 0){let J=y(b.error);if(J)X.push(["error",J])}return R$(X,j)},G$=($)=>{if(typeof $==="object"&&$!==null)return JSON.stringify($);return""},P=({level:$,request:b,data:I,store:j,options:G})=>{let X=G.config,R=s(G),J=X?.customLogFormat??h0,z=new Date,N=String(z.getTime()),Y=g0(z,X?.timestamp?.translateTime),F=a0(Y,R),V=typeof I.message==="string"?I.message:"",H=j.beforeTime===BigInt(0)?0:Number(process.hrtime.bigint()-j.beforeTime)/1e6,W=new URL(b.url).pathname,w=I.status,Z=w===null||w===void 0?200:o(w),K=String(Z),B=X?.ip===!0?v0(b):"",f=X?.showContextTree!==!1&&I.context&&typeof I.context==="object"&&!Array.isArray(I.context)&&Object.keys(I.context).length>0?"":G$(I.context),D=o0($,R),m=b.method.toUpperCase().padEnd(n0),T=s0(m,R),M=t0(W,R),A=l0(K,R),{text:L,isVerySlow:e}=u0(H,R,G),$0=i0(e,R),b0=r0($,R),I0=e0(Z),R0=$$(G,R),X0=J.replaceAll("{now}",F).replaceAll("{epoch}",N).replaceAll("{level}",D).replaceAll("{icon}",b0).replaceAll("{duration}",L).replaceAll("{method}",T).replaceAll("{pathname}",M).replaceAll("{path}",M).replaceAll("{status}",A).replaceAll("{statusText}",I0).replaceAll("{message}",V).replaceAll("{ip}",B).replaceAll("{context}",f).replaceAll("{service}",R0).replaceAll("{speed}",$0),G0=X$($,I,G);return{main:X0,contextLines:G0}};var j$=($)=>typeof $==="object"&&$!==null&&("status"in $)&&typeof $.status==="number",a=($,b,I,j)=>{let G=j.config,X=G?.logFilter;if(X?.level&&X.level.length>0&&!X.level.includes("ERROR"))return;let R=G?.useTransportsOnly===!0,J=G?.disableInternalLogger===!0,z=G?.disableFileLogging===!0,N=j$(b)?b.status:500,Y=y(b),F="ERROR",V={status:N,message:Y,error:b};if(S({level:F,request:$,data:V,store:I,options:j}),!(R||z)){let Z=G?.logFilePath;if(Z)k({filePath:Z,level:F,request:$,data:V,store:I,options:j}).catch(()=>{})}if(R||J)return;let{main:H,contextLines:W}=P({level:F,request:$,data:V,store:I,options:j}),w=W.length>0?`${H}
4
- ${W.join(`
5
- `)}`:H;console.error(w)};var t=($={},b=J$)=>{let I=$.config,j=I?.pino,{prettyPrint:G,...X}=j??{},R=typeof G==="object"&&G!==null?G:void 0,z=(G===!0||R!==void 0)&&X.transport===void 0,N=R?.messageKey??X.messageKey,Y=R?.errorKey??X.errorKey,F=z?{target:"pino-pretty",options:{colorize:process.stdout?.isTTY===!0,translateTime:I?.timestamp?.translateTime,...R,messageKey:N,errorKey:Y}}:X.transport,V=b({...X,level:X.level??"info",messageKey:N,errorKey:Y,transport:F}),H=(Z,K)=>{if(!K?.level||K.level.length===0)return!0;return K.level.includes(Z)},W=(Z,K,B,U)=>{if(!H(Z,I?.logFilter))return;S({level:Z,request:K,data:B,store:U,options:$});let f=I?.useTransportsOnly===!0,D=I?.disableInternalLogger===!0,m=I?.disableFileLogging===!0;if(!(f||m)){let L=I?.logFilePath;if(L)k({filePath:L,level:Z,request:K,data:B,store:U,options:$}).catch(()=>{})}if(f||D)return;let{main:T,contextLines:M}=P({level:Z,request:K,data:B,store:U,options:$}),A=M.length>0?`${T}
6
- ${M.join(`
7
- `)}`:T;switch(Z){case"DEBUG":{console.debug(A);break}case"INFO":{console.info(A);break}case"WARNING":{console.warn(A);break}case"ERROR":{console.error(A);break}default:{console.log(A);break}}},w=(Z,K,B,U)=>{let f={beforeTime:process.hrtime.bigint()};W(Z,K,{message:B,context:U},f)};return{pino:V,log:W,handleHttpError:(Z,K,B)=>{a(Z,K,B,$)},debug:(Z,K,B)=>{w("DEBUG",Z,K,B)},info:(Z,K,B)=>{w("INFO",Z,K,B)},warn:(Z,K,B)=>{w("WARNING",Z,K,B)},error:(Z,K,B)=>{w("ERROR",Z,K,B)}}};var Z$=($={})=>{let b=new WeakSet,I=t($),j={...I,debug:(X,R,J)=>{b.add(X),I.debug(X,R,J)},info:(X,R,J)=>{b.add(X),I.info(X,R,J)},warn:(X,R,J)=>{b.add(X),I.warn(X,R,J)},error:(X,R,J)=>{b.add(X),I.error(X,R,J)}};return new Q$({name:"Logixlysia",detail:{description:"Logixlysia is a plugin for Elysia that provides a logger and pino logger.",tags:["logging","pino"]}}).state("logger",j).state("pino",j.pino).state("beforeTime",BigInt(0)).onStart(({server:X})=>{if(X)q(X,$);else{let R=Number(process.env.PORT)||3000,J=process.env.HOST||"localhost";q({port:R,hostname:J,protocol:"http"},$)}}).onRequest(({store:X})=>{X.beforeTime=process.hrtime.bigint()}).onAfterHandle(({request:X,set:R,store:J})=>{if(b.has(X))return;let z=typeof R.status==="number"?R.status:200,N="INFO";if(z>=500)N="ERROR";else if(z>=400)N="WARNING";j.log(N,X,{status:z},J)}).onError(({request:X,error:R,store:J})=>{j.handleHttpError(X,R,J)}).as("scoped")},e$=Z$;export{Z$ as logixlysia,e$ as default};
1
+ import{Elysia as S$}from"elysia";import{createRequire as s}from"node:module";var H0=(()=>{try{return s(import.meta.url)("elysia/package.json")}catch{return{}}})(),I0=(()=>{try{return s(import.meta.url)("../../package.json")}catch{return{}}})(),V0=($,Q)=>{if($.length>=Q)return $.slice(0,Q);let Z=Math.floor((Q-$.length)/2),J=Q-$.length-Z;return`${" ".repeat(Z)}${$}${" ".repeat(J)}`},l=($,Q)=>{let Z=H0.version,J=Z?`Elysia v${Z}`:"Elysia",X=[{kind:"empty"},{kind:"center",text:J},{kind:"empty"},{kind:"left",text:$}];if(Q)X.push({kind:"left",text:Q});X.push({kind:"empty"});let z=Math.max(J.length,$.length,...Q?[Q.length]:[0])+4,Y=`┌${"─".repeat(z)}┐`,B=`└${"─".repeat(z)}┘`,W=`│${" ".repeat(z)}│`,H=[Y];for(let I of X){if(I.kind==="empty"){H.push(W);continue}if(I.kind==="center"){H.push(`│${V0(I.text,z)}│`);continue}let U=I.text,V=Math.max(0,z-U.length-4);H.push(`│ ${U}${" ".repeat(V)} │`)}return H.push(B),H.join(`
2
+ `)},r=()=>{let $=I0.version;if(!$)return null;return` logixlysia v${$}`};var d=($,Q)=>{if(!(Q.config?.showStartupMessage??!0))return;let{port:J,hostname:X,protocol:K}=$;if(J===void 0||!X||!K)return;let z=`${K}://${X}:${J}`,Y=`\uD83E\uDD8A Elysia is running at ${z}`,B=`\uD83E\uDD8A ${z}`;if((Q.config?.startupMessageFormat??"banner")==="simple"){console.log(Y);return}console.log(l(B,r()))};import E$ from"pino";var q=(...$)=>{let Q=typeof $[0]==="string"?{level:$[0],request:$[1],data:$[2],store:$[3],options:$[4]}:$[0],{level:Z,request:J,data:X,store:K,options:z}=Q,Y=z.config?.transports??[];if(Y.length===0)return;let B=typeof X.message==="string"?X.message:"",W={request:{method:J.method,url:J.url},...X,beforeTime:K.beforeTime};for(let H of Y)try{let I=H.log(Z,B,W);if(I&&typeof I.catch==="function")I.catch(()=>{})}catch{}};import{appendFile as q0}from"node:fs/promises";import{dirname as x0}from"node:path";import{promises as F0}from"node:fs";var a=async($)=>{await F0.mkdir($,{recursive:!0})};import{promises as M}from"node:fs";import{promisify as w0}from"node:util";import{gzip as A0}from"node:zlib";import{promises as n}from"node:fs";import{basename as U0,dirname as M0}from"node:path";var O0=/^(\d+(?:\.\d+)?)(k|kb|m|mb|g|gb)$/i,_0=/^(\d+)(h|d|w)$/i,R0=/\.(\d{4}-\d{2}-\d{2}-\d{2}-\d{2}-\d{2}(?:-\d{3})?(?:-\d+)?)(?:\.gz)?$/,t=($)=>{if(typeof $==="number")return $;let Q=$.trim(),Z=Number(Q);if(Number.isFinite(Z))return Z;let J=Q.match(O0);if(!J)throw Error(`Invalid size format: ${$}`);let X=Number(J[1]),K=J[2].toLowerCase(),z=1024;if(K.startsWith("m"))z=1048576;else if(K.startsWith("g"))z=1073741824;return Math.floor(X*z)},D0=($)=>{let Q=$.trim().match(_0);if(!Q)throw Error(`Invalid interval format: ${$}`);let Z=Number(Q[1]),J=Q[2].toLowerCase(),X=3600000;if(J==="d")X=86400000;else if(J==="w")X=604800000;return Z*X},e=($)=>{if(typeof $==="number")return{type:"count",value:$};return{type:"time",value:D0($)}},$0=async($,Q)=>{try{return(await n.stat($)).size>Q}catch{return!1}},u=async($)=>{let Q=M0($),Z=U0($),J;try{J=await n.readdir(Q)}catch{return[]}return J.filter((X)=>X.startsWith(`${Z}.`)&&R0.test(X)).map((X)=>`${Q}/${X}`)};var T0=w0(A0),x=new Map,E0=($)=>{let Q=x.get($)??Promise.resolve(),Z,J=new Promise((X)=>{Z=X});return Q.then(()=>{return x.set($,J),()=>{if(Z(),x.get($)===J)x.delete($)}})},T=($)=>String($).padStart(2,"0"),S0=($,Q)=>{let Z=Q.getFullYear(),J=T(Q.getMonth()+1),X=T(Q.getDate()),K=T(Q.getHours()),z=T(Q.getMinutes()),Y=T(Q.getSeconds()),B=String(Q.getMilliseconds()).padStart(3,"0");return`${$}.${Z}-${J}-${X}-${K}-${z}-${Y}-${B}`},L0=async($)=>{try{if((await M.stat($)).size===0)return""}catch{return""}let Z=`${S0($,new Date)}-${process.hrtime.bigint()}`;return await M.rename($,Z),Z},y0=async($)=>{let Q=await E0($);try{try{await M.access($)}catch{return}let Z=await M.readFile($),J=await T0(Z);await M.writeFile(`${$}.gz`,J),await M.rm($,{force:!0})}catch(Z){throw console.error(`[logixlysia] Failed to compress file ${$}:`,Z),Z}finally{Q()}},Q0=async($,Q)=>{if(Q.maxSize===void 0)return!1;let Z=t(Q.maxSize);return await $0($,Z)},k0=async($,Q)=>{let Z=await u($);if(Z.length<=Q)return;let X=(await Promise.allSettled(Z.map(async(Y)=>({path:Y,stat:await M.stat(Y)})))).filter((Y)=>Y.status==="fulfilled").map((Y)=>Y.value);if(X.length<=Q)return;X.sort((Y,B)=>B.stat.mtimeMs-Y.stat.mtimeMs);let K=X.slice(Q);(await Promise.allSettled(K.map(({path:Y})=>M.rm(Y,{force:!0})))).forEach((Y,B)=>{if(Y.status==="rejected")console.error(`[logixlysia] Failed to delete rotated log ${K[B].path}:`,Y.reason)})},f0=async($,Q)=>{let Z=await u($);if(Z.length===0)return;let J=Date.now(),z=(await Promise.allSettled(Z.map(async(B)=>({path:B,stat:await M.stat(B)})))).filter((B)=>B.status==="fulfilled").map((B)=>B.value).filter(({stat:B})=>J-B.mtimeMs>Q);(await Promise.allSettled(z.map(({path:B})=>M.rm(B,{force:!0})))).forEach((B,W)=>{if(B.status==="rejected")console.error(`[logixlysia] Failed to delete old log ${z[W].path}:`,B.reason)})},Z0=async($,Q)=>{let Z=await L0($);if(!Z)return;if(Q.compress===!0){if((Q.compression??"gzip")==="gzip")await y0(Z)}if(Q.maxFiles!==void 0){let X=e(Q.maxFiles);if(X.type==="count")await k0($,X.value);else await f0($,X.value)}};var C=new Map,C0=($)=>{let Q=C.get($)??Promise.resolve(),Z,J=new Promise((X)=>{Z=X});return Q.then(()=>{return C.set($,J),()=>{if(Z(),C.get($)===J)C.delete($)}})},m=async(...$)=>{let Q=typeof $[0]==="string"?(()=>{let[N,j,b,D,R,w]=$;return{filePath:N,level:j,request:b,data:D,store:R,options:w}})():$[0],{filePath:Z,level:J,request:X,data:K,store:z,options:Y}=Q,B=Y.config,W=B?.useTransportsOnly===!0,H=B?.disableFileLogging===!0;if(W||H)return;let I=typeof K.message==="string"?K.message:"",U=z.beforeTime===BigInt(0)?0:Number(process.hrtime.bigint()-z.beforeTime)/1e6,V="/";try{V=new URL(X.url).pathname}catch{V=X.url}let O=`${J} ${U.toFixed(2)}ms ${X.method} ${V} ${I}
3
+ `,F=await C0(Z);try{try{await a(x0(Z)),await q0(Z,O,{encoding:"utf-8"})}catch(b){throw console.error(`[logixlysia] Failed to write to log file ${Z}:`,b),b}let N=B?.logRotation;if(!N)return;if(await Q0(Z,N))try{await Z0(Z,N)}catch(b){console.error(`[logixlysia] Failed to rotate log file ${Z}:`,b)}}finally{F()}};var m0=/[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}/g,P0=/\b(?:\d{1,3}\.){3}\d{1,3}\b/g,c0=/\b(?:\d[ -]*?){13,19}\b/g,h0=/eyJ[a-zA-Z0-9_-]+\.eyJ[a-zA-Z0-9_-]+\.[a-zA-Z0-9_-]+/g;var i=($)=>_($).replaceAll("[REDACTED]","redacted"),v0=($)=>{try{let Q=new URL($);if(Q.username!=="")Q.username=i(Q.username);if(Q.password!=="")Q.password=i(Q.password);return Q.hostname=i(Q.hostname),Q.pathname=_(Q.pathname),Q.search=_(Q.search),Q.hash=_(Q.hash),Q.toString()}catch{return _($).replaceAll("[REDACTED]","redacted")}},p0=($)=>{if($.length<13||$.length>19)return!1;let Q=0,Z=!1;for(let J=$.length-1;J>=0;J--){let X=$.charCodeAt(J);if(X<48||X>57)return!1;let K=X-48;if(Z){if(K*=2,K>9)K-=9}Q+=K,Z=!Z}return Q%10===0},g0=($)=>$.replace(c0,(Q)=>{let Z=Q.replace(/\D/g,"");if(Z.length>=13&&Z.length<=19&&p0(Z))return"[REDACTED]";return Q}),_=($)=>{let Q=$;return Q=Q.replace(m0,"[REDACTED]"),Q=Q.replace(P0,"[REDACTED]"),Q=g0(Q),Q=Q.replace(h0,"[REDACTED]"),Q},d0=($,Q)=>{let Z=_($.message),J=Object.getPrototypeOf($),X=Object.create(J);if(X.message=Z,X.name=$.name,$.stack!==void 0)X.stack=_($.stack);let K=$;for(let z of Object.keys(K))if(z!=="message"&&z!=="name"&&z!=="stack")X[z]=P(K[z],Q);return X},n0=($,Q)=>{let Z=Array.from({length:$.length});for(let J=0;J<$.length;J++)Z[J]=P($[J],Q);return Z},u0=($,Q)=>{let Z={};for(let J of Object.keys($))Z[J]=P($[J],Q);return Z},o=($,Q,Z)=>{Q.add($);try{return Z()}finally{Q.delete($)}},P=($,Q)=>{if($===null||$===void 0)return $;if(typeof $==="string")return _($);if(typeof $!=="object")return $;if($ instanceof Date)return new Date($.getTime());let J=$;if(Q.has(J))return"[Circular]";if($ instanceof Error)return o(J,Q,()=>d0($,Q));if(Array.isArray($))return o(J,Q,()=>n0($,Q));return o(J,Q,()=>u0($,Q))},c=($)=>P($,new WeakSet),h=($)=>{let Q=v0($.url),Z=new Headers,J=!1;for(let[B,W]of $.headers.entries()){let H=_(W);if(H!==W)J=!0;Z.set(B,H)}let X=_($.method),K=Q!==$.url,z=X!==$.method;if(!(K||J||z))return $;let Y={method:X,headers:Z,redirect:$.redirect,signal:$.signal};if($.body!==null)Y.body=$.body,Y.duplex="half";return new Request(Q,Y)};import{STATUS_CODES as Q$}from"node:http";import G from"chalk";import{StatusMap as i0}from"elysia";var o0=/^\d+$/,s0=/[_-]+/g,l0=/([a-z0-9])([A-Z])/g,r0=/([A-Z])([A-Z][a-z])/g,a0=/['’]/g,t0=/[^a-z0-9\s]+/g,e0=/\s+/g,J0=($)=>{let Q=$.trim();if(!Q)return"";return Q.replace(s0," ").replace(l0,"$1 $2").replace(r0,"$1 $2").replace(a0,"").toLowerCase().replace(t0," ").replace(e0," ").trim()},$$=(()=>{let $=new Map;for(let[Q,Z]of Object.entries(i0))$.set(J0(Q),Z);return $})(),X0=($)=>{if(typeof $==="number"&&Number.isFinite($))return $;if(typeof $==="string"){let Q=$.trim();if(o0.test(Q))return Number(Q);return $$.get(J0(Q))??500}return 500};var v=($)=>{let Q="An error occurred";if($ instanceof Error)Q=$.message;else if($&&typeof $==="object"&&"message"in $)Q=$.message;else Q=String($);return Q};var E=($)=>String($).padStart(2,"0"),Z$=($)=>String($).padStart(3,"0"),J$=500,X$=1000,z$=7,K$="{now} {service}{icon} {method} {pathname} {status} {duration} {message}{speed}",z0=($)=>{let Z=$.config?.useColors??!0,J=typeof process<"u"&&process.stdout?.isTTY===!0;return Z&&J},Y$=($,Q)=>{if(!Q)return $.toISOString();let Z=String($.getFullYear()),J=E($.getMonth()+1),X=E($.getDate()),K=E($.getHours()),z=E($.getMinutes()),Y=E($.getSeconds()),B=Z$($.getMilliseconds());return Q.replaceAll("yyyy",Z).replaceAll("mm",J).replaceAll("dd",X).replaceAll("HH",K).replaceAll("MM",z).replaceAll("ss",Y).replaceAll("SSS",B)},B$=($)=>{let Q=$.headers.get("x-forwarded-for");if(Q)return Q.split(",")[0]?.trim()??"";return $.headers.get("x-real-ip")??""},G$=($)=>{if($>=1000){let Q=$/1000;if(Q>=10)return`${Math.round(Q)}s`;let Z=Q.toFixed(1);return Z.endsWith(".0")?`${Math.round(Q)}s`:`${Z}s`}if($>0&&$<1)return`${$.toFixed(2)}ms`;return`${Math.round($)}ms`},N$=($)=>{let Q=$.config;return{slow:Q?.slowThreshold??J$,verySlow:Q?.verySlowThreshold??X$}},b$=($,Q,Z)=>{let J=G$($),{slow:X,verySlow:K}=N$(Z),z=$>=K;if(!Q)return{text:J,isVerySlow:z};let Y=J;if($<X)Y=G.green(J);else if($<K)Y=G.yellow(J);else Y=G.red.bold(J);return{text:Y,isVerySlow:z}},j$=($,Q)=>{if(!$)return"";let Z="⚡ slow";if(!Q)return` ${Z}`;return` ${G.yellow(Z)}`},W$=($,Q)=>{if(!Q)return"\uD83E\uDD8A";if($==="ERROR")return G.bgRed.black(" \uD83E\uDD8A ");if($==="WARNING")return G.bgYellow.black(" \uD83E\uDD8A ");if($==="DEBUG")return G.bgBlue.black(" \uD83E\uDD8A ");return G.bgGreen.black(" \uD83E\uDD8A ")},H$=($,Q)=>{if(!Q)return $;if($==="ERROR")return G.bgRed.black($);if($==="WARNING")return G.bgYellow.black($);if($==="DEBUG")return G.bgBlue.black($);return G.bgGreen.black($)},I$=($,Q)=>{if(!Q)return $;let Z=$.toUpperCase();if(Z==="GET")return G.green.bold(Z);if(Z==="POST")return G.blue.bold(Z);if(Z==="PUT")return G.yellow.bold(Z);if(Z==="PATCH")return G.yellowBright.bold(Z);if(Z==="DELETE")return G.red.bold(Z);if(Z==="OPTIONS")return G.cyan.bold(Z);if(Z==="HEAD")return G.greenBright.bold(Z);if(Z==="TRACE")return G.magenta.bold(Z);if(Z==="CONNECT")return G.cyanBright.bold(Z);return G.white.bold(Z)},V$=($,Q)=>{if(!Q)return $;let Z=Number.parseInt($,10);if(!Number.isFinite(Z))return $;if(Z>=500)return G.red($);if(Z>=400)return G.yellow($);if(Z>=300)return G.cyan($);if(Z>=200)return G.green($);return G.gray($)},F$=($,Q)=>{if(!Q)return $;return G.gray($)},U$=($,Q)=>{if(!Q)return $;return G.whiteBright($)},M$=($)=>{return Q$[$]??""},O$=($,Q)=>{let Z=$.config?.service?.trim();if(!Z)return"";let J=`[${Z}]`;if(!Q)return`${J} `;return`${G.dim(J)} `},_$=($)=>{if($===null)return"null";if($===void 0)return"undefined";if(typeof $==="string")return $;if(typeof $==="number"||typeof $==="boolean")return String($);if($ instanceof Error)return $.message;try{return JSON.stringify($)}catch{return String($)}},R$=($)=>$!==null&&typeof $==="object"&&!Array.isArray($)&&!($ instanceof Error)&&!($ instanceof Date),K0=($,Q,Z)=>{let J=[];for(let[X,K]of Object.entries($)){let z=Q?`${Q}.${X}`:X;if(R$(K)&&Z>1)J.push(...K0(K,z,Z-1));else J.push([z,_$(K)])}return J},D$=($,Q)=>{if($.length===0)return[];let Z=[],J=$.length-1;for(let X=0;X<$.length;X++){let K=X===J?"└─":"├─",z=$[X];if(!z)continue;let[Y,B]=z,W=Q?G.cyan(Y):Y,H=Q?G.white(B):B;Z.push(` ${K} ${W} ${H}`)}return Z};var w$=($,Q,Z)=>{if(Z.config?.showContextTree===!1)return[];let J=z0(Z),X=Z.config?.contextDepth??1,K=[],z=Q.context;if(z&&typeof z==="object"&&!Array.isArray(z)&&Object.keys(z).length>0&&X>=1)K.push(...K0(z,"",X));if($==="ERROR"&&"error"in Q&&Q.error!==void 0){let Y=v(Q.error);if(Y)K.push(["error",Y])}return D$(K,J)},A$=($)=>{if(typeof $==="object"&&$!==null)return JSON.stringify($);return""},p=({level:$,request:Q,data:Z,store:J,options:X})=>{let K=X.config,z=z0(X),Y=K?.customLogFormat??K$,B=new Date,W=String(B.getTime()),H=Y$(B,K?.timestamp?.translateTime),I=F$(H,z),U=typeof Z.message==="string"?Z.message:"",V=J.beforeTime===BigInt(0)?0:Number(process.hrtime.bigint()-J.beforeTime)/1e6,O=new URL(Q.url).pathname,F=Z.status,N=F===null||F===void 0?200:X0(F),j=String(N),b=K?.ip===!0?B$(Q):"",R=K?.showContextTree!==!1&&Z.context&&typeof Z.context==="object"&&!Array.isArray(Z.context)&&Object.keys(Z.context).length>0?"":A$(Z.context),w=H$($,z),S=Q.method.toUpperCase().padEnd(z$),g=I$(S,z),L=U$(O,z),y=V$(j,z),{text:k,isVerySlow:A}=b$(V,z,X),f=j$(A,z),G0=W$($,z),N0=M$(N),b0=O$(X,z),j0=Y.replaceAll("{now}",I).replaceAll("{epoch}",W).replaceAll("{level}",w).replaceAll("{icon}",G0).replaceAll("{duration}",k).replaceAll("{method}",g).replaceAll("{pathname}",L).replaceAll("{path}",L).replaceAll("{status}",y).replaceAll("{statusText}",N0).replaceAll("{message}",U).replaceAll("{ip}",b).replaceAll("{context}",R).replaceAll("{service}",b0).replaceAll("{speed}",f),W0=w$($,Z,X);return{main:j0,contextLines:W0}};var T$=($)=>typeof $==="object"&&$!==null&&("status"in $)&&typeof $.status==="number",Y0=($,Q,Z,J)=>{let X=J.config,K=X?.logFilter;if(K?.level&&K.level.length>0&&!K.level.includes("ERROR"))return;let z=X?.useTransportsOnly===!0,Y=X?.disableInternalLogger===!0,B=X?.disableFileLogging===!0,W=T$(Q)?Q.status:500,H=v(Q),I="ERROR",U={status:W,message:H,error:Q},V=X?.autoRedact===!0?c(U):U,O=X?.autoRedact===!0?h($):$;if(q({level:I,request:O,data:V,store:Z,options:J}),!(z||B)){let b=X?.logFilePath;if(b)m({filePath:b,level:I,request:O,data:V,store:Z,options:J}).catch(()=>{})}if(z||Y)return;let{main:F,contextLines:N}=p({level:I,request:O,data:V,store:Z,options:J}),j=N.length>0?`${F}
4
+ ${N.join(`
5
+ `)}`:F;console.error(j)};var B0=($={},Q=E$)=>{let Z=$.config,J=Z?.pino,{prettyPrint:X,...K}=J??{},z=typeof X==="object"&&X!==null?X:void 0,B=(X===!0||z!==void 0)&&K.transport===void 0,W=z?.messageKey??K.messageKey,H=z?.errorKey??K.errorKey,I=B?{target:"pino-pretty",options:{colorize:process.stdout?.isTTY===!0,translateTime:Z?.timestamp?.translateTime,...z,messageKey:W,errorKey:H}}:K.transport,U=Q({...K,level:K.level??"info",messageKey:W,errorKey:H,transport:I}),V=(N,j)=>{if(!j?.level||j.level.length===0)return!0;return j.level.includes(N)},O=(N,j,b,D)=>{if(!V(N,Z?.logFilter))return;let R=Z?.autoRedact===!0?c(b):b,w=Z?.autoRedact===!0?h(j):j;q({level:N,request:w,data:R,store:D,options:$});let S=Z?.useTransportsOnly===!0,g=Z?.disableInternalLogger===!0,L=Z?.disableFileLogging===!0;if(!(S||L)){let f=Z?.logFilePath;if(f)m({filePath:f,level:N,request:w,data:R,store:D,options:$}).catch(()=>{})}if(S||g)return;let{main:y,contextLines:k}=p({level:N,request:w,data:R,store:D,options:$}),A=k.length>0?`${y}
6
+ ${k.join(`
7
+ `)}`:y;switch(N){case"DEBUG":{console.debug(A);break}case"INFO":{console.info(A);break}case"WARNING":{console.warn(A);break}case"ERROR":{console.error(A);break}default:{console.log(A);break}}},F=(N,j,b,D)=>{let R={beforeTime:process.hrtime.bigint()};O(N,j,{message:b,context:D},R)};return{pino:U,log:O,handleHttpError:(N,j,b)=>{Y0(N,j,b,$)},debug:(N,j,b)=>{F("DEBUG",N,j,b)},info:(N,j,b)=>{F("INFO",N,j,b)},warn:(N,j,b)=>{F("WARNING",N,j,b)},error:(N,j,b)=>{F("ERROR",N,j,b)}}};var L$=($={})=>{let Q=new WeakSet,Z=B0($),J={...Z,debug:(K,z,Y)=>{Q.add(K),Z.debug(K,z,Y)},info:(K,z,Y)=>{Q.add(K),Z.info(K,z,Y)},warn:(K,z,Y)=>{Q.add(K),Z.warn(K,z,Y)},error:(K,z,Y)=>{Q.add(K),Z.error(K,z,Y)}};return new S$({name:"Logixlysia",detail:{description:"Logixlysia is a plugin for Elysia that provides a logger and pino logger.",tags:["logging","pino"]}}).state("logger",J).state("pino",J.pino).state("beforeTime",BigInt(0)).onStart(({server:K})=>{if(K)d(K,$);else{let z=Number(process.env.PORT)||3000,Y=process.env.HOST||"localhost";d({port:z,hostname:Y,protocol:"http"},$)}}).onRequest(({store:K})=>{K.beforeTime=process.hrtime.bigint()}).onAfterHandle(({request:K,set:z,store:Y})=>{if(Q.has(K))return;let B=typeof z.status==="number"?z.status:200,W="INFO";if(B>=500)W="ERROR";else if(B>=400)W="WARNING";J.log(W,K,{status:B},Y)}).onError(({request:K,error:z,store:Y})=>{J.handleHttpError(K,z,Y)}).as("scoped")},R2=L$;export{L$ as logixlysia,R2 as default};
8
8
 
9
- //# debugId=F8359047E2C71CA864756E2164756E21
10
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["src/index.ts", "src/extensions/banner.ts", "src/extensions/index.ts", "src/logger/index.ts", "src/output/index.ts", "src/output/file.ts", "src/output/fs.ts", "src/output/rotation-manager.ts", "src/utils/rotation.ts", "src/logger/create-logger.ts", "src/helpers/status.ts", "src/utils/error.ts", "src/logger/handle-http-error.ts"],
  "sourcesContent": [
    "import { Elysia } from 'elysia'\nimport { startServer } from './extensions'\nimport type { LogixlysiaStore, Options } from './interfaces'\nimport { createLogger } from './logger'\n\n/**\n * Empty singleton slots must not use `Record<string, never>`: intersecting that with Elysia's `Context`\n * makes every key (including `store`) become `never` because each key is merged with `never`.\n */\nexport interface EmptyElysiaSlot {\n  readonly __logixlysiaEmpty?: never\n}\n\n/**\n * Explicit singleton without Elysia's `SingletonBase` `Record<string, unknown>` on decorator/derive/resolve so\n * merged `Context` and WebSocket `ws.data` keep precise keys after `.use(logixlysia())`.\n */\nexport interface LogixlysiaSingleton {\n  decorator: EmptyElysiaSlot\n  derive: EmptyElysiaSlot\n  resolve: EmptyElysiaSlot\n  store: LogixlysiaStore\n}\n\n// Elysia's `SingletonBase.store` is `Record<string, unknown>`; `LogixlysiaStore` is intentionally closed (see #220).\n// @ts-expect-error — closed store is correct at runtime and for merged `ws.data` inference.\nexport type Logixlysia = Elysia<'', LogixlysiaSingleton>\n\nexport const logixlysia = (options: Options = {}): Logixlysia => {\n  const didCustomLog = new WeakSet<Request>()\n  const baseLogger = createLogger(options)\n  const logger = {\n    ...baseLogger,\n    debug: (\n      request: Request,\n      message: string,\n      context?: Record<string, unknown>\n    ) => {\n      didCustomLog.add(request)\n      baseLogger.debug(request, message, context)\n    },\n    info: (\n      request: Request,\n      message: string,\n      context?: Record<string, unknown>\n    ) => {\n      didCustomLog.add(request)\n      baseLogger.info(request, message, context)\n    },\n    warn: (\n      request: Request,\n      message: string,\n      context?: Record<string, unknown>\n    ) => {\n      didCustomLog.add(request)\n      baseLogger.warn(request, message, context)\n    },\n    error: (\n      request: Request,\n      message: string,\n      context?: Record<string, unknown>\n    ) => {\n      didCustomLog.add(request)\n      baseLogger.error(request, message, context)\n    }\n  }\n\n  const app = new Elysia({\n    name: 'Logixlysia',\n    detail: {\n      description:\n        'Logixlysia is a plugin for Elysia that provides a logger and pino logger.',\n      tags: ['logging', 'pino']\n    }\n  })\n\n  return (\n    app\n      .state('logger', logger)\n      .state('pino', logger.pino)\n      .state('beforeTime', BigInt(0))\n      .onStart(({ server }): void => {\n        if (server) {\n          startServer(server, options)\n        } else {\n          // Node adapter fallback\n          const port = Number(process.env.PORT) || 3000\n          const hostname = process.env.HOST || 'localhost'\n          startServer({ port, hostname, protocol: 'http' }, options)\n        }\n      })\n      .onRequest(({ store }) => {\n        store.beforeTime = process.hrtime.bigint()\n      })\n      .onAfterHandle(({ request, set, store }) => {\n        if (didCustomLog.has(request)) {\n          return\n        }\n\n        const status = typeof set.status === 'number' ? set.status : 200\n        let level: 'INFO' | 'WARNING' | 'ERROR' = 'INFO'\n        if (status >= 500) {\n          level = 'ERROR'\n        } else if (status >= 400) {\n          level = 'WARNING'\n        }\n\n        logger.log(level, request, { status }, store)\n      })\n      .onError(({ request, error, store }) => {\n        logger.handleHttpError(request, error, store)\n      })\n      // Ensure plugin lifecycle hooks (onRequest/onAfterHandle/onError) apply to the parent app.\n      .as('scoped') as Logixlysia\n  )\n}\n\nexport type {\n  Logger,\n  LogixlysiaContext,\n  LogixlysiaStore,\n  LogLevel,\n  Options,\n  Pino,\n  StoreData,\n  Transport\n} from './interfaces'\n\nexport default logixlysia\n",
    "import { createRequire } from 'node:module'\n\nconst elysiaPkg: { version?: string } = (() => {\n  try {\n    const require = createRequire(import.meta.url)\n    return require('elysia/package.json') as { version?: string }\n  } catch {\n    return {}\n  }\n})()\n\nconst logixlysiaPkg: { version?: string } = (() => {\n  try {\n    const require = createRequire(import.meta.url)\n    return require('../../package.json') as { version?: string }\n  } catch {\n    return {}\n  }\n})()\n\nconst centerText = (text: string, width: number): string => {\n  if (text.length >= width) {\n    return text.slice(0, width)\n  }\n\n  const left = Math.floor((width - text.length) / 2)\n  const right = width - text.length - left\n  return `${' '.repeat(left)}${text}${' '.repeat(right)}`\n}\n\ntype RowSpec =\n  | { kind: 'empty' }\n  | { kind: 'center'; text: string }\n  | { kind: 'left'; text: string }\n\n/**\n * Box banner: Elysia version (centered), URL line, optional logixlysia version line.\n */\nexport const renderBanner = (\n  urlDisplayLine: string,\n  logixlysiaLine: string | null\n): string => {\n  const version = elysiaPkg.version\n  const versionLine = version ? `Elysia v${version}` : 'Elysia'\n\n  const rows: RowSpec[] = [\n    { kind: 'empty' },\n    { kind: 'center', text: versionLine },\n    { kind: 'empty' },\n    { kind: 'left', text: urlDisplayLine }\n  ]\n  if (logixlysiaLine) {\n    rows.push({ kind: 'left', text: logixlysiaLine })\n  }\n  rows.push({ kind: 'empty' })\n\n  const contentWidth = Math.max(\n    versionLine.length,\n    urlDisplayLine.length,\n    ...(logixlysiaLine ? [logixlysiaLine.length] : [0])\n  )\n  const innerWidth = contentWidth + 4\n\n  const top = `┌${'─'.repeat(innerWidth)}┐`\n  const bot = `└${'─'.repeat(innerWidth)}┘`\n  const emptyRow = `│${' '.repeat(innerWidth)}│`\n\n  const out: string[] = [top]\n  for (const row of rows) {\n    if (row.kind === 'empty') {\n      out.push(emptyRow)\n      continue\n    }\n    if (row.kind === 'center') {\n      out.push(`│${centerText(row.text, innerWidth)}│`)\n      continue\n    }\n    const text = row.text\n    const padding = Math.max(0, innerWidth - text.length - 4)\n    out.push(`│  ${text}${' '.repeat(padding)}  │`)\n  }\n  out.push(bot)\n  return out.join('\\n')\n}\n\nexport const getLogixlysiaVersionLine = (): string | null => {\n  const v = logixlysiaPkg.version\n  if (!v) {\n    return null\n  }\n  return `     logixlysia v${v}`\n}\n",
    "import type { Options } from '../interfaces'\nimport { getLogixlysiaVersionLine, renderBanner } from './banner'\n\nexport const startServer = (\n  server: { port?: number; hostname?: string; protocol?: string | null },\n  options: Options\n): void => {\n  const showStartupMessage = options.config?.showStartupMessage ?? true\n  if (!showStartupMessage) {\n    return\n  }\n\n  const { port, hostname, protocol } = server\n  if (port === undefined || !hostname || !protocol) {\n    return\n  }\n\n  const url = `${protocol}://${hostname}:${port}`\n  const message = `🦊 Elysia is running at ${url}`\n  const urlDisplayLine = `🦊  ${url}`\n\n  const format = options.config?.startupMessageFormat ?? 'banner'\n  if (format === 'simple') {\n    console.log(message)\n    return\n  }\n\n  console.log(renderBanner(urlDisplayLine, getLogixlysiaVersionLine()))\n}\n",
    "import pino from 'pino'\nimport type {\n  LogFilter,\n  Logger,\n  LogLevel,\n  Options,\n  Pino,\n  RequestInfo,\n  StoreData\n} from '../interfaces'\nimport { logToTransports } from '../output'\nimport { logToFile } from '../output/file'\nimport { formatLogOutput } from './create-logger'\nimport { handleHttpError } from './handle-http-error'\n\nexport const createLogger = (\n  options: Options = {},\n  pinoFactory: typeof pino = pino\n): Logger => {\n  const config = options.config\n\n  const pinoConfig = config?.pino\n  const { prettyPrint, ...pinoOptions } = pinoConfig ?? {}\n\n  const prettyPrintOptions =\n    typeof prettyPrint === 'object' && prettyPrint !== null\n      ? (prettyPrint as Record<string, unknown>)\n      : undefined\n\n  const enablePrettyPrint =\n    prettyPrint === true || prettyPrintOptions !== undefined\n\n  const shouldPrettyPrint =\n    enablePrettyPrint && pinoOptions.transport === undefined\n\n  const messageKey =\n    (prettyPrintOptions?.messageKey as string | undefined) ??\n    pinoOptions.messageKey\n  const errorKey =\n    (prettyPrintOptions?.errorKey as string | undefined) ?? pinoOptions.errorKey\n\n  const transport = shouldPrettyPrint\n    ? {\n        target: 'pino-pretty',\n        options: {\n          colorize: process.stdout?.isTTY === true,\n          translateTime: config?.timestamp?.translateTime,\n          ...prettyPrintOptions,\n          messageKey,\n          errorKey\n        }\n      }\n    : pinoOptions.transport\n\n  const pinoLogger: Pino = pinoFactory({\n    ...pinoOptions,\n    level: pinoOptions.level ?? 'info',\n    messageKey,\n    errorKey,\n    transport\n  })\n\n  const shouldLog = (level: LogLevel, logFilter?: LogFilter): boolean => {\n    if (!logFilter?.level || logFilter.level.length === 0) {\n      return true\n    }\n    return logFilter.level.includes(level)\n  }\n\n  const log = (\n    level: LogLevel,\n    request: RequestInfo,\n    data: Record<string, unknown>,\n    store: StoreData\n  ): void => {\n    // Check if this log level should be filtered\n    if (!shouldLog(level, config?.logFilter)) {\n      return\n    }\n\n    logToTransports({ level, request, data, store, options })\n\n    const useTransportsOnly = config?.useTransportsOnly === true\n    const disableInternalLogger = config?.disableInternalLogger === true\n    const disableFileLogging = config?.disableFileLogging === true\n\n    if (!(useTransportsOnly || disableFileLogging)) {\n      const filePath = config?.logFilePath\n      if (filePath) {\n        logToFile({ filePath, level, request, data, store, options }).catch(\n          () => {\n            // Ignore errors\n          }\n        )\n      }\n    }\n\n    if (useTransportsOnly || disableInternalLogger) {\n      return\n    }\n\n    const { main, contextLines } = formatLogOutput({\n      level,\n      request,\n      data,\n      store,\n      options\n    })\n    const message =\n      contextLines.length > 0 ? `${main}\\n${contextLines.join('\\n')}` : main\n\n    switch (level) {\n      case 'DEBUG': {\n        console.debug(message)\n        break\n      }\n      case 'INFO': {\n        console.info(message)\n        break\n      }\n      case 'WARNING': {\n        console.warn(message)\n        break\n      }\n      case 'ERROR': {\n        console.error(message)\n        break\n      }\n      default: {\n        console.log(message)\n        break\n      }\n    }\n  }\n\n  const logWithContext = (\n    level: LogLevel,\n    request: RequestInfo,\n    message: string,\n    context?: Record<string, unknown>\n  ): void => {\n    const store: StoreData = { beforeTime: process.hrtime.bigint() }\n    log(level, request, { message, context }, store)\n  }\n\n  return {\n    pino: pinoLogger,\n    log,\n    handleHttpError: (request, error, store) => {\n      handleHttpError(request, error, store, options)\n    },\n    debug: (request, message, context) => {\n      logWithContext('DEBUG', request, message, context)\n    },\n    info: (request, message, context) => {\n      logWithContext('INFO', request, message, context)\n    },\n    warn: (request, message, context) => {\n      logWithContext('WARNING', request, message, context)\n    },\n    error: (request, message, context) => {\n      logWithContext('ERROR', request, message, context)\n    }\n  }\n}\n",
    "import type { LogLevel, Options, RequestInfo, StoreData } from '../interfaces'\n\ninterface LogToTransportsInput {\n  data: Record<string, unknown>\n  level: LogLevel\n  options: Options\n  request: RequestInfo\n  store: StoreData\n}\n\nexport const logToTransports = (\n  ...args:\n    | [LogToTransportsInput]\n    | [LogLevel, RequestInfo, Record<string, unknown>, StoreData, Options]\n): void => {\n  const input: LogToTransportsInput =\n    typeof args[0] === 'string'\n      ? {\n          level: args[0],\n          request: args[1],\n          data: args[2],\n          store: args[3],\n          options: args[4]\n        }\n      : args[0]\n\n  const { level, request, data, store, options } = input\n  const transports = options.config?.transports ?? []\n  if (transports.length === 0) {\n    return\n  }\n\n  const message = typeof data.message === 'string' ? data.message : ''\n  const meta: Record<string, unknown> = {\n    request: {\n      method: request.method,\n      url: request.url\n    },\n    ...data,\n    beforeTime: store.beforeTime\n  }\n\n  for (const transport of transports) {\n    try {\n      const result = transport.log(level, message, meta)\n      if (\n        result &&\n        typeof (result as { catch?: unknown }).catch === 'function'\n      ) {\n        ;(result as Promise<void>).catch(() => {\n          // Ignore errors\n        })\n      }\n    } catch {\n      // Transport failures must never crash application logging.\n    }\n  }\n}\n",
    "import { appendFile } from 'node:fs/promises'\nimport { dirname } from 'node:path'\nimport type { LogLevel, Options, RequestInfo, StoreData } from '../interfaces'\nimport { ensureDir } from './fs'\nimport { performRotation, shouldRotate } from './rotation-manager'\n\ninterface LogToFileInput {\n  data: Record<string, unknown>\n  filePath: string\n  level: LogLevel\n  options: Options\n  request: RequestInfo\n  store: StoreData\n}\n\nexport const logToFile = async (\n  ...args:\n    | [LogToFileInput]\n    | [\n        string,\n        LogLevel,\n        RequestInfo,\n        Record<string, unknown>,\n        StoreData,\n        Options\n      ]\n): Promise<void> => {\n  const input: LogToFileInput =\n    typeof args[0] === 'string'\n      ? (() => {\n          const [\n            filePathArg,\n            levelArg,\n            requestArg,\n            dataArg,\n            storeArg,\n            optionsArg\n          ] = args as [\n            string,\n            LogLevel,\n            RequestInfo,\n            Record<string, unknown>,\n            StoreData,\n            Options\n          ]\n          return {\n            filePath: filePathArg,\n            level: levelArg,\n            request: requestArg,\n            data: dataArg,\n            store: storeArg,\n            options: optionsArg\n          }\n        })()\n      : args[0]\n\n  const { filePath, level, request, data, store, options } = input\n  const config = options.config\n  const useTransportsOnly = config?.useTransportsOnly === true\n  const disableFileLogging = config?.disableFileLogging === true\n  if (useTransportsOnly || disableFileLogging) {\n    return\n  }\n\n  const message = typeof data.message === 'string' ? data.message : ''\n  const durationMs =\n    store.beforeTime === BigInt(0)\n      ? 0\n      : Number(process.hrtime.bigint() - store.beforeTime) / 1_000_000\n\n  const line = `${level} ${durationMs.toFixed(2)}ms ${request.method} ${new URL(request.url).pathname} ${message}\\n`\n\n  await ensureDir(dirname(filePath))\n  await appendFile(filePath, line, { encoding: 'utf-8' })\n\n  const rotation = config?.logRotation\n  if (!rotation) {\n    return\n  }\n\n  const should = await shouldRotate(filePath, rotation)\n  if (should) {\n    await performRotation(filePath, rotation)\n  }\n}\n",
    "import { promises as fs } from 'node:fs'\n\nexport const ensureDir = async (dirPath: string): Promise<void> => {\n  await fs.mkdir(dirPath, { recursive: true })\n}\n",
    "import { promises as fs } from 'node:fs'\nimport { promisify } from 'node:util'\nimport { gzip } from 'node:zlib'\nimport type { LogRotationConfig } from '../interfaces'\nimport {\n  getRotatedFiles,\n  parseRetention,\n  parseSize,\n  shouldRotateBySize\n} from '../utils/rotation'\n\nconst gzipAsync = promisify(gzip)\n\nconst pad2 = (value: number): string => String(value).padStart(2, '0')\n\nexport const getRotatedFileName = (filePath: string, date: Date): string => {\n  const yyyy = date.getFullYear()\n  const mm = pad2(date.getMonth() + 1)\n  const dd = pad2(date.getDate())\n  const HH = pad2(date.getHours())\n  const MM = pad2(date.getMinutes())\n  const ss = pad2(date.getSeconds())\n  return `${filePath}.${yyyy}-${mm}-${dd}-${HH}-${MM}-${ss}`\n}\n\nexport const rotateFile = async (filePath: string): Promise<string> => {\n  try {\n    const stat = await fs.stat(filePath)\n    if (stat.size === 0) {\n      return ''\n    }\n  } catch {\n    return ''\n  }\n\n  const rotated = getRotatedFileName(filePath, new Date())\n  await fs.rename(filePath, rotated)\n  return rotated\n}\n\nexport const compressFile = async (filePath: string): Promise<void> => {\n  const content = await fs.readFile(filePath)\n  const compressed = await gzipAsync(content)\n  await fs.writeFile(`${filePath}.gz`, compressed)\n  await fs.rm(filePath, { force: true })\n}\n\nexport const shouldRotate = async (\n  filePath: string,\n  config: LogRotationConfig\n): Promise<boolean> => {\n  if (config.maxSize === undefined) {\n    return false\n  }\n  const maxSize = parseSize(config.maxSize)\n  return await shouldRotateBySize(filePath, maxSize)\n}\n\nconst cleanupByCount = async (\n  filePath: string,\n  maxFiles: number\n): Promise<void> => {\n  const rotated = await getRotatedFiles(filePath)\n  if (rotated.length <= maxFiles) {\n    return\n  }\n\n  const stats = await Promise.all(\n    rotated.map(async p => ({ path: p, stat: await fs.stat(p) }))\n  )\n\n  stats.sort((a, b) => b.stat.mtimeMs - a.stat.mtimeMs)\n  const toDelete = stats.slice(maxFiles)\n  await Promise.all(toDelete.map(({ path }) => fs.rm(path, { force: true })))\n}\n\nconst cleanupByTime = async (\n  filePath: string,\n  maxAgeMs: number\n): Promise<void> => {\n  const rotated = await getRotatedFiles(filePath)\n  if (rotated.length === 0) {\n    return\n  }\n\n  const now = Date.now()\n  const stats = await Promise.all(\n    rotated.map(async p => ({ path: p, stat: await fs.stat(p) }))\n  )\n\n  const toDelete = stats.filter(({ stat }) => now - stat.mtimeMs > maxAgeMs)\n  await Promise.all(toDelete.map(({ path }) => fs.rm(path, { force: true })))\n}\n\nexport const performRotation = async (\n  filePath: string,\n  config: LogRotationConfig\n): Promise<void> => {\n  const rotated = await rotateFile(filePath)\n  if (!rotated) {\n    return\n  }\n\n  const shouldCompress = config.compress === true\n  if (shouldCompress) {\n    const algo = config.compression ?? 'gzip'\n    if (algo === 'gzip') {\n      await compressFile(rotated)\n    }\n  }\n\n  if (config.maxFiles !== undefined) {\n    const retention = parseRetention(config.maxFiles)\n    if (retention.type === 'count') {\n      await cleanupByCount(filePath, retention.value)\n    } else {\n      await cleanupByTime(filePath, retention.value)\n    }\n  }\n\n  // Optional interval-based rotation cleanup (create interval directories / naming) is not required by tests.\n}\n",
    "import { promises as fs } from 'node:fs'\nimport { basename, dirname } from 'node:path'\n\nconst SIZE_REGEX = /^(\\d+(?:\\.\\d+)?)(k|kb|m|mb|g|gb)$/i\nconst INTERVAL_REGEX = /^(\\d+)(h|d|w)$/i\nconst ROTATED_REGEX = /\\.(\\d{4}-\\d{2}-\\d{2}-\\d{2}-\\d{2}-\\d{2})(?:\\.gz)?$/\n\nexport const parseSize = (value: number | string): number => {\n  if (typeof value === 'number') {\n    return value\n  }\n\n  const trimmed = value.trim()\n  const asNumber = Number(trimmed)\n  if (Number.isFinite(asNumber)) {\n    return asNumber\n  }\n\n  const match = trimmed.match(SIZE_REGEX)\n  if (!match) {\n    throw new Error(`Invalid size format: ${value}`)\n  }\n\n  const amount = Number(match[1])\n  const unit = match[2].toLowerCase()\n\n  let base = 1024\n  if (unit.startsWith('m')) {\n    base = 1024 * 1024\n  } else if (unit.startsWith('g')) {\n    base = 1024 * 1024 * 1024\n  }\n\n  return Math.floor(amount * base)\n}\n\nexport const parseInterval = (value: string): number => {\n  const match = value.trim().match(INTERVAL_REGEX)\n  if (!match) {\n    throw new Error(`Invalid interval format: ${value}`)\n  }\n\n  const amount = Number(match[1])\n  const unit = match[2].toLowerCase()\n\n  let ms = 60 * 60 * 1000\n  if (unit === 'd') {\n    ms = 24 * 60 * 60 * 1000\n  } else if (unit === 'w') {\n    ms = 7 * 24 * 60 * 60 * 1000\n  }\n\n  return amount * ms\n}\n\nexport const parseRetention = (\n  value: number | string\n): { type: 'count' | 'time'; value: number } => {\n  if (typeof value === 'number') {\n    return { type: 'count', value }\n  }\n  return { type: 'time', value: parseInterval(value) }\n}\n\nexport const shouldRotateBySize = async (\n  filePath: string,\n  maxSizeBytes: number\n): Promise<boolean> => {\n  try {\n    const stat = await fs.stat(filePath)\n    return stat.size > maxSizeBytes\n  } catch {\n    return false\n  }\n}\n\nexport const getRotatedFiles = async (filePath: string): Promise<string[]> => {\n  const dir = dirname(filePath)\n  const base = basename(filePath)\n\n  let entries: string[]\n  try {\n    entries = await fs.readdir(dir)\n  } catch {\n    return []\n  }\n\n  return entries\n    .filter(name => name.startsWith(`${base}.`) && ROTATED_REGEX.test(name))\n    .map(name => `${dir}/${name}`)\n}\n",
    "import { STATUS_CODES } from 'node:http'\nimport chalk from 'chalk'\nimport { getStatusCode } from '../helpers/status'\nimport type {\n  LogLevel,\n  Options,\n  Pino,\n  RequestInfo,\n  StoreData\n} from '../interfaces'\nimport { parseError } from '../utils/error'\n\nconst pad2 = (value: number): string => String(value).padStart(2, '0')\nconst pad3 = (value: number): string => String(value).padStart(3, '0')\n\nconst DEFAULT_SLOW_MS = 500\nconst DEFAULT_VERY_SLOW_MS = 1000\nconst METHOD_PAD = 7\n\nconst DEFAULT_LOG_FORMAT =\n  '{now} {service}{icon} {method} {pathname} {status} {duration} {message}{speed}'\n\nexport interface FormattedLogOutput {\n  contextLines: string[]\n  main: string\n}\n\nconst shouldUseColors = (options: Options): boolean => {\n  const config = options.config\n  const enabledByConfig = config?.useColors ?? true\n\n  // Avoid ANSI sequences in non-interactive output (pipes, CI logs, files).\n  const isTty = typeof process !== 'undefined' && process.stdout?.isTTY === true\n  return enabledByConfig && isTty\n}\n\nconst formatTimestamp = (date: Date, pattern?: string): string => {\n  if (!pattern) {\n    return date.toISOString()\n  }\n\n  const yyyy = String(date.getFullYear())\n  const mm = pad2(date.getMonth() + 1)\n  const dd = pad2(date.getDate())\n  const HH = pad2(date.getHours())\n  const MM = pad2(date.getMinutes())\n  const ss = pad2(date.getSeconds())\n  const SSS = pad3(date.getMilliseconds())\n\n  return pattern\n    .replaceAll('yyyy', yyyy)\n    .replaceAll('mm', mm)\n    .replaceAll('dd', dd)\n    .replaceAll('HH', HH)\n    .replaceAll('MM', MM)\n    .replaceAll('ss', ss)\n    .replaceAll('SSS', SSS)\n}\n\n/** Resolves client IP from x-forwarded-for (first IP) or x-real-ip. Empty when neither header is set (e.g. localhost). */\nconst getIp = (request: RequestInfo): string => {\n  const forwarded = request.headers.get('x-forwarded-for')\n  if (forwarded) {\n    return forwarded.split(',')[0]?.trim() ?? ''\n  }\n  return request.headers.get('x-real-ip') ?? ''\n}\n\nexport const formatDuration = (ms: number): string => {\n  if (ms >= 1000) {\n    const sec = ms / 1000\n    if (sec >= 10) {\n      return `${Math.round(sec)}s`\n    }\n    const oneDecimal = sec.toFixed(1)\n    return oneDecimal.endsWith('.0') ? `${Math.round(sec)}s` : `${oneDecimal}s`\n  }\n  if (ms > 0 && ms < 1) {\n    return `${ms.toFixed(2)}ms`\n  }\n  return `${Math.round(ms)}ms`\n}\n\nconst getSlowThresholds = (\n  options: Options\n): { slow: number; verySlow: number } => {\n  const config = options.config\n  return {\n    slow: config?.slowThreshold ?? DEFAULT_SLOW_MS,\n    verySlow: config?.verySlowThreshold ?? DEFAULT_VERY_SLOW_MS\n  }\n}\n\nconst colorDurationText = (\n  ms: number,\n  useColors: boolean,\n  options: Options\n): { text: string; isVerySlow: boolean } => {\n  const raw = formatDuration(ms)\n  const { slow, verySlow } = getSlowThresholds(options)\n  const isVerySlow = ms >= verySlow\n\n  if (!useColors) {\n    return { text: raw, isVerySlow }\n  }\n\n  let colored = raw\n  if (ms < slow) {\n    colored = chalk.green(raw)\n  } else if (ms < verySlow) {\n    colored = chalk.yellow(raw)\n  } else {\n    colored = chalk.red.bold(raw)\n  }\n\n  return { text: colored, isVerySlow }\n}\n\nconst getSpeedToken = (isVerySlow: boolean, useColors: boolean): string => {\n  if (!isVerySlow) {\n    return ''\n  }\n  const badge = '⚡ slow'\n  if (!useColors) {\n    return ` ${badge}`\n  }\n  return ` ${chalk.yellow(badge)}`\n}\n\n/** Logixlysia brand: fox emoji with level-colored background when colors are enabled. */\nconst getLevelIcon = (level: LogLevel, useColors: boolean): string => {\n  if (!useColors) {\n    return '🦊'\n  }\n  if (level === 'ERROR') {\n    return chalk.bgRed.black(' 🦊 ')\n  }\n  if (level === 'WARNING') {\n    return chalk.bgYellow.black(' 🦊 ')\n  }\n  if (level === 'DEBUG') {\n    return chalk.bgBlue.black(' 🦊 ')\n  }\n  return chalk.bgGreen.black(' 🦊 ')\n}\n\nconst getColoredLevel = (level: LogLevel, useColors: boolean): string => {\n  if (!useColors) {\n    return level\n  }\n\n  if (level === 'ERROR') {\n    return chalk.bgRed.black(level)\n  }\n  if (level === 'WARNING') {\n    return chalk.bgYellow.black(level)\n  }\n  if (level === 'DEBUG') {\n    return chalk.bgBlue.black(level)\n  }\n\n  return chalk.bgGreen.black(level)\n}\n\nconst getColoredMethod = (method: string, useColors: boolean): string => {\n  if (!useColors) {\n    return method\n  }\n\n  const upper = method.toUpperCase()\n  if (upper === 'GET') {\n    return chalk.green.bold(upper)\n  }\n  if (upper === 'POST') {\n    return chalk.blue.bold(upper)\n  }\n  if (upper === 'PUT') {\n    return chalk.yellow.bold(upper)\n  }\n  if (upper === 'PATCH') {\n    return chalk.yellowBright.bold(upper)\n  }\n  if (upper === 'DELETE') {\n    return chalk.red.bold(upper)\n  }\n  if (upper === 'OPTIONS') {\n    return chalk.cyan.bold(upper)\n  }\n  if (upper === 'HEAD') {\n    return chalk.greenBright.bold(upper)\n  }\n  if (upper === 'TRACE') {\n    return chalk.magenta.bold(upper)\n  }\n  if (upper === 'CONNECT') {\n    return chalk.cyanBright.bold(upper)\n  }\n\n  return chalk.white.bold(upper)\n}\n\nconst getColoredStatus = (status: string, useColors: boolean): string => {\n  if (!useColors) {\n    return status\n  }\n\n  const numeric = Number.parseInt(status, 10)\n  if (!Number.isFinite(numeric)) {\n    return status\n  }\n\n  if (numeric >= 500) {\n    return chalk.red(status)\n  }\n  if (numeric >= 400) {\n    return chalk.yellow(status)\n  }\n  if (numeric >= 300) {\n    return chalk.cyan(status)\n  }\n  if (numeric >= 200) {\n    return chalk.green(status)\n  }\n  return chalk.gray(status)\n}\n\nconst getColoredTimestamp = (timestamp: string, useColors: boolean): string => {\n  if (!useColors) {\n    return timestamp\n  }\n\n  return chalk.gray(timestamp)\n}\n\nconst getColoredPathname = (pathname: string, useColors: boolean): string => {\n  if (!useColors) {\n    return pathname\n  }\n\n  return chalk.whiteBright(pathname)\n}\n\nconst getStatusText = (statusCode: number): string => {\n  const text = STATUS_CODES[statusCode]\n  return text ?? ''\n}\n\nconst getServiceToken = (options: Options, useColors: boolean): string => {\n  const name = options.config?.service?.trim()\n  if (!name) {\n    return ''\n  }\n  const bracketed = `[${name}]`\n  if (!useColors) {\n    return `${bracketed} `\n  }\n  return `${chalk.dim(bracketed)} `\n}\n\nconst stringifyTreeValue = (value: unknown): string => {\n  if (value === null) {\n    return 'null'\n  }\n  if (value === undefined) {\n    return 'undefined'\n  }\n  if (typeof value === 'string') {\n    return value\n  }\n  if (typeof value === 'number' || typeof value === 'boolean') {\n    return String(value)\n  }\n  if (value instanceof Error) {\n    return value.message\n  }\n  try {\n    return JSON.stringify(value)\n  } catch {\n    return String(value)\n  }\n}\n\n/** Nested objects to expand in the context tree (excludes Arrays, Error, Date). */\nconst isExpandableObject = (value: unknown): value is Record<string, unknown> =>\n  value !== null &&\n  typeof value === 'object' &&\n  !Array.isArray(value) &&\n  !(value instanceof Error) &&\n  !(value instanceof Date)\n\nconst collectContextEntries = (\n  obj: Record<string, unknown>,\n  prefix: string,\n  depthRemaining: number\n): [string, string][] => {\n  const out: [string, string][] = []\n  for (const [k, v] of Object.entries(obj)) {\n    const key = prefix ? `${prefix}.${k}` : k\n    const expandable = isExpandableObject(v) && depthRemaining > 1\n\n    if (expandable) {\n      out.push(...collectContextEntries(v, key, depthRemaining - 1))\n    } else {\n      out.push([key, stringifyTreeValue(v)])\n    }\n  }\n  return out\n}\n\nconst formatEntriesToTreeLines = (\n  entries: [string, string][],\n  useColors: boolean\n): string[] => {\n  if (entries.length === 0) {\n    return []\n  }\n\n  const lines: string[] = []\n  const last = entries.length - 1\n  for (let i = 0; i < entries.length; i++) {\n    const branch = i === last ? '└─' : '├─'\n    const pair = entries[i]\n    if (!pair) {\n      continue\n    }\n    const [k, v] = pair\n    const keyPart = useColors ? chalk.cyan(k) : k\n    const valPart = useColors ? chalk.white(v) : v\n    lines.push(`  ${branch} ${keyPart}  ${valPart}`)\n  }\n  return lines\n}\n\nexport const renderContextTreeLines = (\n  ctx: Record<string, unknown>,\n  options: Options,\n  useColors: boolean\n): string[] => {\n  const depth = options.config?.contextDepth ?? 1\n  if (depth < 1) {\n    return []\n  }\n\n  const entries = collectContextEntries(ctx, '', depth)\n  return formatEntriesToTreeLines(entries, useColors)\n}\n\nexport const buildContextTreeLines = (\n  level: LogLevel,\n  data: Record<string, unknown>,\n  options: Options\n): string[] => {\n  if (options.config?.showContextTree === false) {\n    return []\n  }\n\n  const useColors = shouldUseColors(options)\n  const depth = options.config?.contextDepth ?? 1\n\n  const entries: [string, string][] = []\n\n  const ctx = data.context\n  if (\n    ctx &&\n    typeof ctx === 'object' &&\n    !Array.isArray(ctx) &&\n    Object.keys(ctx as object).length > 0 &&\n    depth >= 1\n  ) {\n    entries.push(\n      ...collectContextEntries(ctx as Record<string, unknown>, '', depth)\n    )\n  }\n\n  if (level === 'ERROR' && 'error' in data && data.error !== undefined) {\n    const msg = parseError(data.error)\n    if (msg) {\n      entries.push(['error', msg])\n    }\n  }\n\n  return formatEntriesToTreeLines(entries, useColors)\n}\n\nconst getContextString = (value: unknown): string => {\n  if (typeof value === 'object' && value !== null) {\n    return JSON.stringify(value)\n  }\n\n  return ''\n}\n\nexport const formatLogOutput = ({\n  level,\n  request,\n  data,\n  store,\n  options\n}: {\n  level: LogLevel\n  request: RequestInfo\n  data: Record<string, unknown>\n  store: StoreData\n  options: Options\n}): FormattedLogOutput => {\n  const config = options.config\n  const useColors = shouldUseColors(options)\n  const format = config?.customLogFormat ?? DEFAULT_LOG_FORMAT\n\n  const now = new Date()\n  const epoch = String(now.getTime())\n  const rawTimestamp = formatTimestamp(now, config?.timestamp?.translateTime)\n  const timestamp = getColoredTimestamp(rawTimestamp, useColors)\n\n  const message = typeof data.message === 'string' ? data.message : ''\n  const durationMs =\n    store.beforeTime === BigInt(0)\n      ? 0\n      : Number(process.hrtime.bigint() - store.beforeTime) / 1_000_000\n\n  const pathname = new URL(request.url).pathname\n  const statusValue = data.status\n  const statusCode =\n    statusValue === null || statusValue === undefined\n      ? 200\n      : getStatusCode(statusValue)\n  const status = String(statusCode)\n  const ip = config?.ip === true ? getIp(request) : ''\n\n  const showTree = config?.showContextTree !== false\n  const ctxString =\n    showTree &&\n    data.context &&\n    typeof data.context === 'object' &&\n    !Array.isArray(data.context) &&\n    Object.keys(data.context as object).length > 0\n      ? ''\n      : getContextString(data.context)\n\n  const coloredLevel = getColoredLevel(level, useColors)\n  const methodPadded = request.method.toUpperCase().padEnd(METHOD_PAD)\n  const coloredMethod = getColoredMethod(methodPadded, useColors)\n  const coloredPathname = getColoredPathname(pathname, useColors)\n  const coloredStatus = getColoredStatus(status, useColors)\n  const { text: coloredDuration, isVerySlow } = colorDurationText(\n    durationMs,\n    useColors,\n    options\n  )\n  const speedToken = getSpeedToken(isVerySlow, useColors)\n  const icon = getLevelIcon(level, useColors)\n  const statusText = getStatusText(statusCode)\n  const serviceToken = getServiceToken(options, useColors)\n\n  const main = format\n    .replaceAll('{now}', timestamp)\n    .replaceAll('{epoch}', epoch)\n    .replaceAll('{level}', coloredLevel)\n    .replaceAll('{icon}', icon)\n    .replaceAll('{duration}', coloredDuration)\n    .replaceAll('{method}', coloredMethod)\n    .replaceAll('{pathname}', coloredPathname)\n    .replaceAll('{path}', coloredPathname)\n    .replaceAll('{status}', coloredStatus)\n    .replaceAll('{statusText}', statusText)\n    .replaceAll('{message}', message)\n    .replaceAll('{ip}', ip)\n    .replaceAll('{context}', ctxString)\n    .replaceAll('{service}', serviceToken)\n    .replaceAll('{speed}', speedToken)\n\n  const contextLines = buildContextTreeLines(level, data, options)\n\n  return { main, contextLines }\n}\n\n/** @deprecated Prefer {@link formatLogOutput} for multi-line context trees. Returns the main line only. */\nexport const formatLine = (input: {\n  level: LogLevel\n  request: RequestInfo\n  data: Record<string, unknown>\n  store: StoreData\n  options: Options\n}): string =>\n  formatLogOutput({\n    ...input,\n    options: {\n      ...input.options,\n      config: {\n        ...(input.options.config ?? {}),\n        showContextTree: false\n      }\n    }\n  }).main\n\nexport const logWithPino = (\n  logger: Pino,\n  level: LogLevel,\n  data: Record<string, unknown>\n): void => {\n  if (level === 'ERROR') {\n    logger.error(data)\n    return\n  }\n  if (level === 'WARNING') {\n    logger.warn(data)\n    return\n  }\n  if (level === 'DEBUG') {\n    logger.debug(data)\n    return\n  }\n  logger.info(data)\n}\n",
    "import { StatusMap } from 'elysia'\n\nconst DIGITS_ONLY = /^\\d+$/\nconst DELIMITERS = /[_-]+/g\nconst CAMEL_BOUNDARY_1 = /([a-z0-9])([A-Z])/g\nconst CAMEL_BOUNDARY_2 = /([A-Z])([A-Z][a-z])/g\nconst APOSTROPHES = /['’]/g\nconst NON_ALPHANUMERIC = /[^a-z0-9\\s]+/g\nconst WHITESPACE = /\\s+/g\n\nconst normalizeStatusName = (value: string): string => {\n  // Handles common variants:\n  // - case differences: \"not found\" vs \"Not Found\"\n  // - spacing/punctuation: \"Not-Found\", \"not_found\"\n  // - camelCase/PascalCase: \"InternalServerError\"\n  const trimmed = value.trim()\n  if (!trimmed) {\n    return ''\n  }\n\n  return trimmed\n    .replace(DELIMITERS, ' ')\n    .replace(CAMEL_BOUNDARY_1, '$1 $2')\n    .replace(CAMEL_BOUNDARY_2, '$1 $2')\n    .replace(APOSTROPHES, '')\n    .toLowerCase()\n    .replace(NON_ALPHANUMERIC, ' ')\n    .replace(WHITESPACE, ' ')\n    .trim()\n}\n\nconst STATUS_BY_NORMALIZED_NAME = (() => {\n  const map = new Map<string, number>()\n\n  for (const [name, code] of Object.entries(StatusMap)) {\n    map.set(normalizeStatusName(name), code)\n  }\n\n  return map\n})()\n\nexport const getStatusCode = (value: unknown): number => {\n  if (typeof value === 'number' && Number.isFinite(value)) {\n    return value\n  }\n\n  if (typeof value === 'string') {\n    const trimmed = value.trim()\n    if (DIGITS_ONLY.test(trimmed)) {\n      return Number(trimmed)\n    }\n\n    const known = STATUS_BY_NORMALIZED_NAME.get(normalizeStatusName(trimmed))\n    return known ?? 500\n  }\n\n  return 500\n}\n",
    "export const parseError = (error: unknown): string => {\n  let message = 'An error occurred'\n\n  if (error instanceof Error) {\n    message = error.message\n  } else if (error && typeof error === 'object' && 'message' in error) {\n    message = error.message as string\n  } else {\n    message = String(error)\n  }\n\n  return message\n}\n",
    "import type { LogLevel, Options, RequestInfo, StoreData } from '../interfaces'\nimport { logToTransports } from '../output'\nimport { logToFile } from '../output/file'\nimport { parseError } from '../utils/error'\nimport { formatLogOutput } from './create-logger'\n\nconst isErrorWithStatus = (\n  value: unknown\n): value is { status: number; message?: string } =>\n  typeof value === 'object' &&\n  value !== null &&\n  'status' in value &&\n  typeof (value as { status?: unknown }).status === 'number'\n\nexport const handleHttpError = (\n  request: RequestInfo,\n  error: unknown,\n  store: StoreData,\n  options: Options\n): void => {\n  const config = options.config\n\n  const logFilter = config?.logFilter\n  if (\n    logFilter?.level &&\n    logFilter.level.length > 0 &&\n    !logFilter.level.includes('ERROR')\n  ) {\n    return\n  }\n\n  const useTransportsOnly = config?.useTransportsOnly === true\n  const disableInternalLogger = config?.disableInternalLogger === true\n  const disableFileLogging = config?.disableFileLogging === true\n\n  const status = isErrorWithStatus(error) ? error.status : 500\n  const message = parseError(error)\n\n  const level: LogLevel = 'ERROR'\n  const data: Record<string, unknown> = { status, message, error }\n\n  logToTransports({ level, request, data, store, options })\n\n  if (!(useTransportsOnly || disableFileLogging)) {\n    const filePath = config?.logFilePath\n    if (filePath) {\n      logToFile({ filePath, level, request, data, store, options }).catch(\n        () => {\n          // Ignore errors\n        }\n      )\n    }\n  }\n\n  if (useTransportsOnly || disableInternalLogger) {\n    return\n  }\n\n  const { main, contextLines } = formatLogOutput({\n    level,\n    request,\n    data,\n    store,\n    options\n  })\n  const formattedMessage =\n    contextLines.length > 0 ? `${main}\\n${contextLines.join('\\n')}` : main\n  console.error(formattedMessage)\n}\n"
  ],
  "mappings": "AAAA,iBAAS,gBCAT,wBAAS,oBAET,IAAM,IAAmC,IAAM,CAC7C,GAAI,CAEF,OADgB,EAAc,YAAY,GAAG,EAC9B,qBAAqB,EACpC,KAAM,CACN,MAAO,CAAC,KAET,EAEG,IAAuC,IAAM,CACjD,GAAI,CAEF,OADgB,EAAc,YAAY,GAAG,EAC9B,oBAAoB,EACnC,KAAM,CACN,MAAO,CAAC,KAET,EAEG,GAAa,CAAC,EAAc,IAA0B,CAC1D,GAAI,EAAK,QAAU,EACjB,OAAO,EAAK,MAAM,EAAG,CAAK,EAG5B,IAAM,EAAO,KAAK,OAAO,EAAQ,EAAK,QAAU,CAAC,EAC3C,EAAQ,EAAQ,EAAK,OAAS,EACpC,MAAO,GAAG,IAAI,OAAO,CAAI,IAAI,IAAO,IAAI,OAAO,CAAK,KAWzC,EAAe,CAC1B,EACA,IACW,CACX,IAAM,EAAU,GAAU,QACpB,EAAc,EAAU,WAAW,IAAY,SAE/C,EAAkB,CACtB,CAAE,KAAM,OAAQ,EAChB,CAAE,KAAM,SAAU,KAAM,CAAY,EACpC,CAAE,KAAM,OAAQ,EAChB,CAAE,KAAM,OAAQ,KAAM,CAAe,CACvC,EACA,GAAI,EACF,EAAK,KAAK,CAAE,KAAM,OAAQ,KAAM,CAAe,CAAC,EAElD,EAAK,KAAK,CAAE,KAAM,OAAQ,CAAC,EAO3B,IAAM,EALe,KAAK,IACxB,EAAY,OACZ,EAAe,OACf,GAAI,EAAiB,CAAC,EAAe,MAAM,EAAI,CAAC,CAAC,CACnD,EACkC,EAE5B,EAAM,IAAG,IAAI,OAAO,CAAU,KAC9B,EAAM,IAAG,IAAI,OAAO,CAAU,KAC9B,EAAW,IAAG,IAAI,OAAO,CAAU,KAEnC,EAAgB,CAAC,CAAG,EAC1B,QAAW,KAAO,EAAM,CACtB,GAAI,EAAI,OAAS,QAAS,CACxB,EAAI,KAAK,CAAQ,EACjB,SAEF,GAAI,EAAI,OAAS,SAAU,CACzB,EAAI,KAAK,IAAG,GAAW,EAAI,KAAM,CAAU,IAAI,EAC/C,SAEF,IAAM,EAAO,EAAI,KACX,EAAU,KAAK,IAAI,EAAG,EAAa,EAAK,OAAS,CAAC,EACxD,EAAI,KAAK,MAAK,IAAO,IAAI,OAAO,CAAO,MAAM,EAG/C,OADA,EAAI,KAAK,CAAG,EACL,EAAI,KAAK;AAAA,CAAI,GAGT,EAA2B,IAAqB,CAC3D,IAAM,EAAI,GAAc,QACxB,GAAI,CAAC,EACH,OAAO,KAET,MAAO,oBAAoB,KCvFtB,IAAM,EAAc,CACzB,EACA,IACS,CAET,GAAI,EADuB,EAAQ,QAAQ,oBAAsB,IAE/D,OAGF,IAAQ,OAAM,WAAU,YAAa,EACrC,GAAI,IAAS,QAAa,CAAC,GAAY,CAAC,EACtC,OAGF,IAAM,EAAM,GAAG,OAAc,KAAY,IACnC,EAAU,qCAA0B,IACpC,EAAiB,iBAAM,IAG7B,IADe,EAAQ,QAAQ,sBAAwB,YACxC,SAAU,CACvB,QAAQ,IAAI,CAAO,EACnB,OAGF,QAAQ,IAAI,EAAa,EAAgB,EAAyB,CAAC,CAAC,GC3BtE,qBCUO,IAAM,EAAkB,IAC1B,IAGM,CACT,IAAM,EACJ,OAAO,EAAK,KAAO,SACf,CACE,MAAO,EAAK,GACZ,QAAS,EAAK,GACd,KAAM,EAAK,GACX,MAAO,EAAK,GACZ,QAAS,EAAK,EAChB,EACA,EAAK,IAEH,QAAO,UAAS,OAAM,QAAO,WAAY,EAC3C,EAAa,EAAQ,QAAQ,YAAc,CAAC,EAClD,GAAI,EAAW,SAAW,EACxB,OAGF,IAAM,EAAU,OAAO,EAAK,UAAY,SAAW,EAAK,QAAU,GAC5D,EAAgC,CACpC,QAAS,CACP,OAAQ,EAAQ,OAChB,IAAK,EAAQ,GACf,KACG,EACH,WAAY,EAAM,UACpB,EAEA,QAAW,KAAa,EACtB,GAAI,CACF,IAAM,EAAS,EAAU,IAAI,EAAO,EAAS,CAAI,EACjD,GACE,GACA,OAAQ,EAA+B,QAAU,WAE/C,EAAyB,MAAM,IAAM,EAEtC,EAEH,KAAM,ICrDZ,qBAAS,0BACT,kBAAS,mBCDT,mBAAS,iBAEF,IAAM,EAAY,MAAO,IAAmC,CACjE,MAAM,GAAG,MAAM,EAAS,CAAE,UAAW,EAAK,CAAC,GCH7C,mBAAS,gBACT,oBAAS,mBACT,eAAS,mBCFT,mBAAS,gBACT,mBAAS,cAAU,mBAEnB,IAAM,GAAa,qCACb,GAAiB,kBACjB,GAAgB,oDAET,EAAY,CAAC,IAAmC,CAC3D,GAAI,OAAO,IAAU,SACnB,OAAO,EAGT,IAAM,EAAU,EAAM,KAAK,EACrB,EAAW,OAAO,CAAO,EAC/B,GAAI,OAAO,SAAS,CAAQ,EAC1B,OAAO,EAGT,IAAM,EAAQ,EAAQ,MAAM,EAAU,EACtC,GAAI,CAAC,EACH,MAAU,MAAM,wBAAwB,GAAO,EAGjD,IAAM,EAAS,OAAO,EAAM,EAAE,EACxB,EAAO,EAAM,GAAG,YAAY,EAE9B,EAAO,KACX,GAAI,EAAK,WAAW,GAAG,EACrB,EAAO,QACF,QAAI,EAAK,WAAW,GAAG,EAC5B,EAAO,WAGT,OAAO,KAAK,MAAM,EAAS,CAAI,GAGpB,GAAgB,CAAC,IAA0B,CACtD,IAAM,EAAQ,EAAM,KAAK,EAAE,MAAM,EAAc,EAC/C,GAAI,CAAC,EACH,MAAU,MAAM,4BAA4B,GAAO,EAGrD,IAAM,EAAS,OAAO,EAAM,EAAE,EACxB,EAAO,EAAM,GAAG,YAAY,EAE9B,EAAK,QACT,GAAI,IAAS,IACX,EAAK,SACA,QAAI,IAAS,IAClB,EAAK,UAGP,OAAO,EAAS,GAGL,EAAiB,CAC5B,IAC8C,CAC9C,GAAI,OAAO,IAAU,SACnB,MAAO,CAAE,KAAM,QAAS,OAAM,EAEhC,MAAO,CAAE,KAAM,OAAQ,MAAO,GAAc,CAAK,CAAE,GAGxC,EAAqB,MAChC,EACA,IACqB,CACrB,GAAI,CAEF,OADa,MAAM,EAAG,KAAK,CAAQ,GACvB,KAAO,EACnB,KAAM,CACN,MAAO,KAIE,EAAkB,MAAO,IAAwC,CAC5E,IAAM,EAAM,GAAQ,CAAQ,EACtB,EAAO,GAAS,CAAQ,EAE1B,EACJ,GAAI,CACF,EAAU,MAAM,EAAG,QAAQ,CAAG,EAC9B,KAAM,CACN,MAAO,CAAC,EAGV,OAAO,EACJ,OAAO,KAAQ,EAAK,WAAW,GAAG,IAAO,GAAK,GAAc,KAAK,CAAI,CAAC,EACtE,IAAI,KAAQ,GAAG,KAAO,GAAM,GD9EjC,IAAM,GAAY,GAAU,EAAI,EAE1B,EAAO,CAAC,IAA0B,OAAO,CAAK,EAAE,SAAS,EAAG,GAAG,EAExD,GAAqB,CAAC,EAAkB,IAAuB,CAC1E,IAAM,EAAO,EAAK,YAAY,EACxB,EAAK,EAAK,EAAK,SAAS,EAAI,CAAC,EAC7B,EAAK,EAAK,EAAK,QAAQ,CAAC,EACxB,EAAK,EAAK,EAAK,SAAS,CAAC,EACzB,EAAK,EAAK,EAAK,WAAW,CAAC,EAC3B,EAAK,EAAK,EAAK,WAAW,CAAC,EACjC,MAAO,GAAG,KAAY,KAAQ,KAAM,KAAM,KAAM,KAAM,KAG3C,GAAa,MAAO,IAAsC,CACrE,GAAI,CAEF,IADa,MAAM,EAAG,KAAK,CAAQ,GAC1B,OAAS,EAChB,MAAO,GAET,KAAM,CACN,MAAO,GAGT,IAAM,EAAU,GAAmB,EAAU,IAAI,IAAM,EAEvD,OADA,MAAM,EAAG,OAAO,EAAU,CAAO,EAC1B,GAGI,GAAe,MAAO,IAAoC,CACrE,IAAM,EAAU,MAAM,EAAG,SAAS,CAAQ,EACpC,EAAa,MAAM,GAAU,CAAO,EAC1C,MAAM,EAAG,UAAU,GAAG,OAAe,CAAU,EAC/C,MAAM,EAAG,GAAG,EAAU,CAAE,MAAO,EAAK,CAAC,GAG1B,EAAe,MAC1B,EACA,IACqB,CACrB,GAAI,EAAO,UAAY,OACrB,MAAO,GAET,IAAM,EAAU,EAAU,EAAO,OAAO,EACxC,OAAO,MAAM,EAAmB,EAAU,CAAO,GAG7C,GAAiB,MACrB,EACA,IACkB,CAClB,IAAM,EAAU,MAAM,EAAgB,CAAQ,EAC9C,GAAI,EAAQ,QAAU,EACpB,OAGF,IAAM,EAAQ,MAAM,QAAQ,IAC1B,EAAQ,IAAI,MAAM,KAAM,CAAE,KAAM,EAAG,KAAM,MAAM,EAAG,KAAK,CAAC,CAAE,EAAE,CAC9D,EAEA,EAAM,KAAK,CAAC,EAAG,IAAM,EAAE,KAAK,QAAU,EAAE,KAAK,OAAO,EACpD,IAAM,EAAW,EAAM,MAAM,CAAQ,EACrC,MAAM,QAAQ,IAAI,EAAS,IAAI,EAAG,UAAW,EAAG,GAAG,EAAM,CAAE,MAAO,EAAK,CAAC,CAAC,CAAC,GAGtE,GAAgB,MACpB,EACA,IACkB,CAClB,IAAM,EAAU,MAAM,EAAgB,CAAQ,EAC9C,GAAI,EAAQ,SAAW,EACrB,OAGF,IAAM,EAAM,KAAK,IAAI,EAKf,GAJQ,MAAM,QAAQ,IAC1B,EAAQ,IAAI,MAAM,KAAM,CAAE,KAAM,EAAG,KAAM,MAAM,EAAG,KAAK,CAAC,CAAE,EAAE,CAC9D,GAEuB,OAAO,EAAG,UAAW,EAAM,EAAK,QAAU,CAAQ,EACzE,MAAM,QAAQ,IAAI,EAAS,IAAI,EAAG,UAAW,EAAG,GAAG,EAAM,CAAE,MAAO,EAAK,CAAC,CAAC,CAAC,GAG/D,EAAkB,MAC7B,EACA,IACkB,CAClB,IAAM,EAAU,MAAM,GAAW,CAAQ,EACzC,GAAI,CAAC,EACH,OAIF,GADuB,EAAO,WAAa,IAGzC,IADa,EAAO,aAAe,UACtB,OACX,MAAM,GAAa,CAAO,EAI9B,GAAI,EAAO,WAAa,OAAW,CACjC,IAAM,EAAY,EAAe,EAAO,QAAQ,EAChD,GAAI,EAAU,OAAS,QACrB,MAAM,GAAe,EAAU,EAAU,KAAK,EAE9C,WAAM,GAAc,EAAU,EAAU,KAAK,IFrG5C,IAAM,EAAY,SACpB,IAUe,CAClB,IAAM,EACJ,OAAO,EAAK,KAAO,UACd,IAAM,CACL,IACE,EACA,EACA,EACA,EACA,EACA,GACE,EAQJ,MAAO,CACL,SAAU,EACV,MAAO,EACP,QAAS,EACT,KAAM,EACN,MAAO,EACP,QAAS,CACX,IACC,EACH,EAAK,IAEH,WAAU,QAAO,UAAS,OAAM,QAAO,WAAY,EACrD,EAAS,EAAQ,OACjB,EAAoB,GAAQ,oBAAsB,GAClD,EAAqB,GAAQ,qBAAuB,GAC1D,GAAI,GAAqB,EACvB,OAGF,IAAM,EAAU,OAAO,EAAK,UAAY,SAAW,EAAK,QAAU,GAC5D,EACJ,EAAM,aAAe,OAAO,CAAC,EACzB,EACA,OAAO,QAAQ,OAAO,OAAO,EAAI,EAAM,UAAU,EAAI,IAErD,EAAO,GAAG,KAAS,EAAW,QAAQ,CAAC,OAAO,EAAQ,UAAU,IAAI,IAAI,EAAQ,GAAG,EAAE,YAAY;AAAA,EAEvG,MAAM,EAAU,GAAQ,CAAQ,CAAC,EACjC,MAAM,GAAW,EAAU,EAAM,CAAE,SAAU,OAAQ,CAAC,EAEtD,IAAM,EAAW,GAAQ,YACzB,GAAI,CAAC,EACH,OAIF,GADe,MAAM,EAAa,EAAU,CAAQ,EAElD,MAAM,EAAgB,EAAU,CAAQ,GIlF5C,uBAAS,mBACT,qBCDA,oBAAS,gBAET,IAAM,GAAc,QACd,GAAa,SACb,GAAmB,qBACnB,GAAmB,uBACnB,GAAc,QACd,GAAmB,gBACnB,GAAa,OAEb,EAAsB,CAAC,IAA0B,CAKrD,IAAM,EAAU,EAAM,KAAK,EAC3B,GAAI,CAAC,EACH,MAAO,GAGT,OAAO,EACJ,QAAQ,GAAY,GAAG,EACvB,QAAQ,GAAkB,OAAO,EACjC,QAAQ,GAAkB,OAAO,EACjC,QAAQ,GAAa,EAAE,EACvB,YAAY,EACZ,QAAQ,GAAkB,GAAG,EAC7B,QAAQ,GAAY,GAAG,EACvB,KAAK,GAGJ,IAA6B,IAAM,CACvC,IAAM,EAAM,IAAI,IAEhB,QAAY,EAAM,KAAS,OAAO,QAAQ,EAAS,EACjD,EAAI,IAAI,EAAoB,CAAI,EAAG,CAAI,EAGzC,OAAO,IACN,EAEU,EAAgB,CAAC,IAA2B,CACvD,GAAI,OAAO,IAAU,UAAY,OAAO,SAAS,CAAK,EACpD,OAAO,EAGT,GAAI,OAAO,IAAU,SAAU,CAC7B,IAAM,EAAU,EAAM,KAAK,EAC3B,GAAI,GAAY,KAAK,CAAO,EAC1B,OAAO,OAAO,CAAO,EAIvB,OADc,GAA0B,IAAI,EAAoB,CAAO,CAAC,GACxD,IAGlB,MAAO,MCxDF,IAAM,EAAa,CAAC,IAA2B,CACpD,IAAI,EAAU,oBAEd,GAAI,aAAiB,MACnB,EAAU,EAAM,QACX,QAAI,GAAS,OAAO,IAAU,UAAY,YAAa,EAC5D,EAAU,EAAM,QAEhB,OAAU,OAAO,CAAK,EAGxB,OAAO,GFCT,IAAM,EAAO,CAAC,IAA0B,OAAO,CAAK,EAAE,SAAS,EAAG,GAAG,EAC/D,GAAO,CAAC,IAA0B,OAAO,CAAK,EAAE,SAAS,EAAG,GAAG,EAE/D,GAAkB,IAClB,GAAuB,KACvB,GAAa,EAEb,GACJ,iFAOI,EAAkB,CAAC,IAA8B,CAErD,IAAM,EADS,EAAQ,QACS,WAAa,GAGvC,EAAQ,OAAO,QAAY,KAAe,QAAQ,QAAQ,QAAU,GAC1E,OAAO,GAAmB,GAGtB,GAAkB,CAAC,EAAY,IAA6B,CAChE,GAAI,CAAC,EACH,OAAO,EAAK,YAAY,EAG1B,IAAM,EAAO,OAAO,EAAK,YAAY,CAAC,EAChC,EAAK,EAAK,EAAK,SAAS,EAAI,CAAC,EAC7B,EAAK,EAAK,EAAK,QAAQ,CAAC,EACxB,EAAK,EAAK,EAAK,SAAS,CAAC,EACzB,EAAK,EAAK,EAAK,WAAW,CAAC,EAC3B,EAAK,EAAK,EAAK,WAAW,CAAC,EAC3B,EAAM,GAAK,EAAK,gBAAgB,CAAC,EAEvC,OAAO,EACJ,WAAW,OAAQ,CAAI,EACvB,WAAW,KAAM,CAAE,EACnB,WAAW,KAAM,CAAE,EACnB,WAAW,KAAM,CAAE,EACnB,WAAW,KAAM,CAAE,EACnB,WAAW,KAAM,CAAE,EACnB,WAAW,MAAO,CAAG,GAIpB,GAAQ,CAAC,IAAiC,CAC9C,IAAM,EAAY,EAAQ,QAAQ,IAAI,iBAAiB,EACvD,GAAI,EACF,OAAO,EAAU,MAAM,GAAG,EAAE,IAAI,KAAK,GAAK,GAE5C,OAAO,EAAQ,QAAQ,IAAI,WAAW,GAAK,IAGhC,GAAiB,CAAC,IAAuB,CACpD,GAAI,GAAM,KAAM,CACd,IAAM,EAAM,EAAK,KACjB,GAAI,GAAO,GACT,MAAO,GAAG,KAAK,MAAM,CAAG,KAE1B,IAAM,EAAa,EAAI,QAAQ,CAAC,EAChC,OAAO,EAAW,SAAS,IAAI,EAAI,GAAG,KAAK,MAAM,CAAG,KAAO,GAAG,KAEhE,GAAI,EAAK,GAAK,EAAK,EACjB,MAAO,GAAG,EAAG,QAAQ,CAAC,MAExB,MAAO,GAAG,KAAK,MAAM,CAAE,OAGnB,GAAoB,CACxB,IACuC,CACvC,IAAM,EAAS,EAAQ,OACvB,MAAO,CACL,KAAM,GAAQ,eAAiB,GAC/B,SAAU,GAAQ,mBAAqB,EACzC,GAGI,GAAoB,CACxB,EACA,EACA,IAC0C,CAC1C,IAAM,EAAM,GAAe,CAAE,GACrB,OAAM,YAAa,GAAkB,CAAO,EAC9C,EAAa,GAAM,EAEzB,GAAI,CAAC,EACH,MAAO,CAAE,KAAM,EAAK,YAAW,EAGjC,IAAI,EAAU,EACd,GAAI,EAAK,EACP,EAAU,EAAM,MAAM,CAAG,EACpB,QAAI,EAAK,EACd,EAAU,EAAM,OAAO,CAAG,EAE1B,OAAU,EAAM,IAAI,KAAK,CAAG,EAG9B,MAAO,CAAE,KAAM,EAAS,YAAW,GAG/B,GAAgB,CAAC,EAAqB,IAA+B,CACzE,GAAI,CAAC,EACH,MAAO,GAET,IAAM,EAAQ,SACd,GAAI,CAAC,EACH,MAAO,IAAI,IAEb,MAAO,IAAI,EAAM,OAAO,CAAK,KAIzB,GAAe,CAAC,EAAiB,IAA+B,CACpE,GAAI,CAAC,EACH,MAAO,eAET,GAAI,IAAU,QACZ,OAAO,EAAM,MAAM,MAAM,gBAAK,EAEhC,GAAI,IAAU,UACZ,OAAO,EAAM,SAAS,MAAM,gBAAK,EAEnC,GAAI,IAAU,QACZ,OAAO,EAAM,OAAO,MAAM,gBAAK,EAEjC,OAAO,EAAM,QAAQ,MAAM,gBAAK,GAG5B,GAAkB,CAAC,EAAiB,IAA+B,CACvE,GAAI,CAAC,EACH,OAAO,EAGT,GAAI,IAAU,QACZ,OAAO,EAAM,MAAM,MAAM,CAAK,EAEhC,GAAI,IAAU,UACZ,OAAO,EAAM,SAAS,MAAM,CAAK,EAEnC,GAAI,IAAU,QACZ,OAAO,EAAM,OAAO,MAAM,CAAK,EAGjC,OAAO,EAAM,QAAQ,MAAM,CAAK,GAG5B,GAAmB,CAAC,EAAgB,IAA+B,CACvE,GAAI,CAAC,EACH,OAAO,EAGT,IAAM,EAAQ,EAAO,YAAY,EACjC,GAAI,IAAU,MACZ,OAAO,EAAM,MAAM,KAAK,CAAK,EAE/B,GAAI,IAAU,OACZ,OAAO,EAAM,KAAK,KAAK,CAAK,EAE9B,GAAI,IAAU,MACZ,OAAO,EAAM,OAAO,KAAK,CAAK,EAEhC,GAAI,IAAU,QACZ,OAAO,EAAM,aAAa,KAAK,CAAK,EAEtC,GAAI,IAAU,SACZ,OAAO,EAAM,IAAI,KAAK,CAAK,EAE7B,GAAI,IAAU,UACZ,OAAO,EAAM,KAAK,KAAK,CAAK,EAE9B,GAAI,IAAU,OACZ,OAAO,EAAM,YAAY,KAAK,CAAK,EAErC,GAAI,IAAU,QACZ,OAAO,EAAM,QAAQ,KAAK,CAAK,EAEjC,GAAI,IAAU,UACZ,OAAO,EAAM,WAAW,KAAK,CAAK,EAGpC,OAAO,EAAM,MAAM,KAAK,CAAK,GAGzB,GAAmB,CAAC,EAAgB,IAA+B,CACvE,GAAI,CAAC,EACH,OAAO,EAGT,IAAM,EAAU,OAAO,SAAS,EAAQ,EAAE,EAC1C,GAAI,CAAC,OAAO,SAAS,CAAO,EAC1B,OAAO,EAGT,GAAI,GAAW,IACb,OAAO,EAAM,IAAI,CAAM,EAEzB,GAAI,GAAW,IACb,OAAO,EAAM,OAAO,CAAM,EAE5B,GAAI,GAAW,IACb,OAAO,EAAM,KAAK,CAAM,EAE1B,GAAI,GAAW,IACb,OAAO,EAAM,MAAM,CAAM,EAE3B,OAAO,EAAM,KAAK,CAAM,GAGpB,GAAsB,CAAC,EAAmB,IAA+B,CAC7E,GAAI,CAAC,EACH,OAAO,EAGT,OAAO,EAAM,KAAK,CAAS,GAGvB,GAAqB,CAAC,EAAkB,IAA+B,CAC3E,GAAI,CAAC,EACH,OAAO,EAGT,OAAO,EAAM,YAAY,CAAQ,GAG7B,GAAgB,CAAC,IAA+B,CAEpD,OADa,GAAa,IACX,IAGX,GAAkB,CAAC,EAAkB,IAA+B,CACxE,IAAM,EAAO,EAAQ,QAAQ,SAAS,KAAK,EAC3C,GAAI,CAAC,EACH,MAAO,GAET,IAAM,EAAY,IAAI,KACtB,GAAI,CAAC,EACH,MAAO,GAAG,KAEZ,MAAO,GAAG,EAAM,IAAI,CAAS,MAGzB,GAAqB,CAAC,IAA2B,CACrD,GAAI,IAAU,KACZ,MAAO,OAET,GAAI,IAAU,OACZ,MAAO,YAET,GAAI,OAAO,IAAU,SACnB,OAAO,EAET,GAAI,OAAO,IAAU,UAAY,OAAO,IAAU,UAChD,OAAO,OAAO,CAAK,EAErB,GAAI,aAAiB,MACnB,OAAO,EAAM,QAEf,GAAI,CACF,OAAO,KAAK,UAAU,CAAK,EAC3B,KAAM,CACN,OAAO,OAAO,CAAK,IAKjB,GAAqB,CAAC,IAC1B,IAAU,MACV,OAAO,IAAU,UACjB,CAAC,MAAM,QAAQ,CAAK,GACpB,EAAE,aAAiB,QACnB,EAAE,aAAiB,MAEf,EAAwB,CAC5B,EACA,EACA,IACuB,CACvB,IAAM,EAA0B,CAAC,EACjC,QAAY,EAAG,KAAM,OAAO,QAAQ,CAAG,EAAG,CACxC,IAAM,EAAM,EAAS,GAAG,KAAU,IAAM,EAGxC,GAFmB,GAAmB,CAAC,GAAK,EAAiB,EAG3D,EAAI,KAAK,GAAG,EAAsB,EAAG,EAAK,EAAiB,CAAC,CAAC,EAE7D,OAAI,KAAK,CAAC,EAAK,GAAmB,CAAC,CAAC,CAAC,EAGzC,OAAO,GAGH,GAA2B,CAC/B,EACA,IACa,CACb,GAAI,EAAQ,SAAW,EACrB,MAAO,CAAC,EAGV,IAAM,EAAkB,CAAC,EACnB,EAAO,EAAQ,OAAS,EAC9B,QAAS,EAAI,EAAG,EAAI,EAAQ,OAAQ,IAAK,CACvC,IAAM,EAAS,IAAM,EAAO,KAAM,KAC5B,EAAO,EAAQ,GACrB,GAAI,CAAC,EACH,SAEF,IAAO,EAAG,GAAK,EACT,EAAU,EAAY,EAAM,KAAK,CAAC,EAAI,EACtC,EAAU,EAAY,EAAM,MAAM,CAAC,EAAI,EAC7C,EAAM,KAAK,KAAK,KAAU,MAAY,GAAS,EAEjD,OAAO,GAiBF,IAAM,GAAwB,CACnC,EACA,EACA,IACa,CACb,GAAI,EAAQ,QAAQ,kBAAoB,GACtC,MAAO,CAAC,EAGV,IAAM,EAAY,EAAgB,CAAO,EACnC,EAAQ,EAAQ,QAAQ,cAAgB,EAExC,EAA8B,CAAC,EAE/B,EAAM,EAAK,QACjB,GACE,GACA,OAAO,IAAQ,UACf,CAAC,MAAM,QAAQ,CAAG,GAClB,OAAO,KAAK,CAAa,EAAE,OAAS,GACpC,GAAS,EAET,EAAQ,KACN,GAAG,EAAsB,EAAgC,GAAI,CAAK,CACpE,EAGF,GAAI,IAAU,SAAW,UAAW,GAAQ,EAAK,QAAU,OAAW,CACpE,IAAM,EAAM,EAAW,EAAK,KAAK,EACjC,GAAI,EACF,EAAQ,KAAK,CAAC,QAAS,CAAG,CAAC,EAI/B,OAAO,GAAyB,EAAS,CAAS,GAG9C,GAAmB,CAAC,IAA2B,CACnD,GAAI,OAAO,IAAU,UAAY,IAAU,KACzC,OAAO,KAAK,UAAU,CAAK,EAG7B,MAAO,IAGI,EAAkB,EAC7B,QACA,UACA,OACA,QACA,aAOwB,CACxB,IAAM,EAAS,EAAQ,OACjB,EAAY,EAAgB,CAAO,EACnC,EAAS,GAAQ,iBAAmB,GAEpC,EAAM,IAAI,KACV,EAAQ,OAAO,EAAI,QAAQ,CAAC,EAC5B,EAAe,GAAgB,EAAK,GAAQ,WAAW,aAAa,EACpE,EAAY,GAAoB,EAAc,CAAS,EAEvD,EAAU,OAAO,EAAK,UAAY,SAAW,EAAK,QAAU,GAC5D,EACJ,EAAM,aAAe,OAAO,CAAC,EACzB,EACA,OAAO,QAAQ,OAAO,OAAO,EAAI,EAAM,UAAU,EAAI,IAErD,EAAW,IAAI,IAAI,EAAQ,GAAG,EAAE,SAChC,EAAc,EAAK,OACnB,EACJ,IAAgB,MAAQ,IAAgB,OACpC,IACA,EAAc,CAAW,EACzB,EAAS,OAAO,CAAU,EAC1B,EAAK,GAAQ,KAAO,GAAO,GAAM,CAAO,EAAI,GAG5C,EADW,GAAQ,kBAAoB,IAG3C,EAAK,SACL,OAAO,EAAK,UAAY,UACxB,CAAC,MAAM,QAAQ,EAAK,OAAO,GAC3B,OAAO,KAAK,EAAK,OAAiB,EAAE,OAAS,EACzC,GACA,GAAiB,EAAK,OAAO,EAE7B,EAAe,GAAgB,EAAO,CAAS,EAC/C,EAAe,EAAQ,OAAO,YAAY,EAAE,OAAO,EAAU,EAC7D,EAAgB,GAAiB,EAAc,CAAS,EACxD,EAAkB,GAAmB,EAAU,CAAS,EACxD,EAAgB,GAAiB,EAAQ,CAAS,GAChD,KAAM,EAAiB,cAAe,GAC5C,EACA,EACA,CACF,EACM,GAAa,GAAc,EAAY,CAAS,EAChD,GAAO,GAAa,EAAO,CAAS,EACpC,GAAa,GAAc,CAAU,EACrC,GAAe,GAAgB,EAAS,CAAS,EAEjD,GAAO,EACV,WAAW,QAAS,CAAS,EAC7B,WAAW,UAAW,CAAK,EAC3B,WAAW,UAAW,CAAY,EAClC,WAAW,SAAU,EAAI,EACzB,WAAW,aAAc,CAAe,EACxC,WAAW,WAAY,CAAa,EACpC,WAAW,aAAc,CAAe,EACxC,WAAW,SAAU,CAAe,EACpC,WAAW,WAAY,CAAa,EACpC,WAAW,eAAgB,EAAU,EACrC,WAAW,YAAa,CAAO,EAC/B,WAAW,OAAQ,CAAE,EACrB,WAAW,YAAa,CAAS,EACjC,WAAW,YAAa,EAAY,EACpC,WAAW,UAAW,EAAU,EAE7B,GAAe,GAAsB,EAAO,EAAM,CAAO,EAE/D,MAAO,CAAE,QAAM,eAAa,GGnd9B,IAAM,GAAoB,CACxB,IAEA,OAAO,IAAU,UACjB,IAAU,OACV,WAAY,IACZ,OAAQ,EAA+B,SAAW,SAEvC,EAAkB,CAC7B,EACA,EACA,EACA,IACS,CACT,IAAM,EAAS,EAAQ,OAEjB,EAAY,GAAQ,UAC1B,GACE,GAAW,OACX,EAAU,MAAM,OAAS,GACzB,CAAC,EAAU,MAAM,SAAS,OAAO,EAEjC,OAGF,IAAM,EAAoB,GAAQ,oBAAsB,GAClD,EAAwB,GAAQ,wBAA0B,GAC1D,EAAqB,GAAQ,qBAAuB,GAEpD,EAAS,GAAkB,CAAK,EAAI,EAAM,OAAS,IACnD,EAAU,EAAW,CAAK,EAE1B,EAAkB,QAClB,EAAgC,CAAE,SAAQ,UAAS,OAAM,EAI/D,GAFA,EAAgB,CAAE,QAAO,UAAS,OAAM,QAAO,SAAQ,CAAC,EAEpD,EAAE,GAAqB,GAAqB,CAC9C,IAAM,EAAW,GAAQ,YACzB,GAAI,EACF,EAAU,CAAE,WAAU,QAAO,UAAS,OAAM,QAAO,SAAQ,CAAC,EAAE,MAC5D,IAAM,EAGR,EAIJ,GAAI,GAAqB,EACvB,OAGF,IAAQ,OAAM,gBAAiB,EAAgB,CAC7C,QACA,UACA,OACA,QACA,SACF,CAAC,EACK,EACJ,EAAa,OAAS,EAAI,GAAG;AAAA,EAAS,EAAa,KAAK;AAAA,CAAI,IAAM,EACpE,QAAQ,MAAM,CAAgB,GTpDzB,IAAM,EAAe,CAC1B,EAAmB,CAAC,EACpB,EAA2B,KAChB,CACX,IAAM,EAAS,EAAQ,OAEjB,EAAa,GAAQ,MACnB,iBAAgB,GAAgB,GAAc,CAAC,EAEjD,EACJ,OAAO,IAAgB,UAAY,IAAgB,KAC9C,EACD,OAKA,GAFJ,IAAgB,IAAQ,IAAuB,SAG1B,EAAY,YAAc,OAE3C,EACH,GAAoB,YACrB,EAAY,WACR,EACH,GAAoB,UAAmC,EAAY,SAEhE,EAAY,EACd,CACE,OAAQ,cACR,QAAS,CACP,SAAU,QAAQ,QAAQ,QAAU,GACpC,cAAe,GAAQ,WAAW,iBAC/B,EACH,aACA,UACF,CACF,EACA,EAAY,UAEV,EAAmB,EAAY,IAChC,EACH,MAAO,EAAY,OAAS,OAC5B,aACA,WACA,WACF,CAAC,EAEK,EAAY,CAAC,EAAiB,IAAmC,CACrE,GAAI,CAAC,GAAW,OAAS,EAAU,MAAM,SAAW,EAClD,MAAO,GAET,OAAO,EAAU,MAAM,SAAS,CAAK,GAGjC,EAAM,CACV,EACA,EACA,EACA,IACS,CAET,GAAI,CAAC,EAAU,EAAO,GAAQ,SAAS,EACrC,OAGF,EAAgB,CAAE,QAAO,UAAS,OAAM,QAAO,SAAQ,CAAC,EAExD,IAAM,EAAoB,GAAQ,oBAAsB,GAClD,EAAwB,GAAQ,wBAA0B,GAC1D,EAAqB,GAAQ,qBAAuB,GAE1D,GAAI,EAAE,GAAqB,GAAqB,CAC9C,IAAM,EAAW,GAAQ,YACzB,GAAI,EACF,EAAU,CAAE,WAAU,QAAO,UAAS,OAAM,QAAO,SAAQ,CAAC,EAAE,MAC5D,IAAM,EAGR,EAIJ,GAAI,GAAqB,EACvB,OAGF,IAAQ,OAAM,gBAAiB,EAAgB,CAC7C,QACA,UACA,OACA,QACA,SACF,CAAC,EACK,EACJ,EAAa,OAAS,EAAI,GAAG;AAAA,EAAS,EAAa,KAAK;AAAA,CAAI,IAAM,EAEpE,OAAQ,OACD,QAAS,CACZ,QAAQ,MAAM,CAAO,EACrB,KACF,KACK,OAAQ,CACX,QAAQ,KAAK,CAAO,EACpB,KACF,KACK,UAAW,CACd,QAAQ,KAAK,CAAO,EACpB,KACF,KACK,QAAS,CACZ,QAAQ,MAAM,CAAO,EACrB,KACF,SACS,CACP,QAAQ,IAAI,CAAO,EACnB,KACF,IAIE,EAAiB,CACrB,EACA,EACA,EACA,IACS,CACT,IAAM,EAAmB,CAAE,WAAY,QAAQ,OAAO,OAAO,CAAE,EAC/D,EAAI,EAAO,EAAS,CAAE,UAAS,SAAQ,EAAG,CAAK,GAGjD,MAAO,CACL,KAAM,EACN,MACA,gBAAiB,CAAC,EAAS,EAAO,IAAU,CAC1C,EAAgB,EAAS,EAAO,EAAO,CAAO,GAEhD,MAAO,CAAC,EAAS,EAAS,IAAY,CACpC,EAAe,QAAS,EAAS,EAAS,CAAO,GAEnD,KAAM,CAAC,EAAS,EAAS,IAAY,CACnC,EAAe,OAAQ,EAAS,EAAS,CAAO,GAElD,KAAM,CAAC,EAAS,EAAS,IAAY,CACnC,EAAe,UAAW,EAAS,EAAS,CAAO,GAErD,MAAO,CAAC,EAAS,EAAS,IAAY,CACpC,EAAe,QAAS,EAAS,EAAS,CAAO,EAErD,GHvIK,IAAM,GAAa,CAAC,EAAmB,CAAC,IAAkB,CAC/D,IAAM,EAAe,IAAI,QACnB,EAAa,EAAa,CAAO,EACjC,EAAS,IACV,EACH,MAAO,CACL,EACA,EACA,IACG,CACH,EAAa,IAAI,CAAO,EACxB,EAAW,MAAM,EAAS,EAAS,CAAO,GAE5C,KAAM,CACJ,EACA,EACA,IACG,CACH,EAAa,IAAI,CAAO,EACxB,EAAW,KAAK,EAAS,EAAS,CAAO,GAE3C,KAAM,CACJ,EACA,EACA,IACG,CACH,EAAa,IAAI,CAAO,EACxB,EAAW,KAAK,EAAS,EAAS,CAAO,GAE3C,MAAO,CACL,EACA,EACA,IACG,CACH,EAAa,IAAI,CAAO,EACxB,EAAW,MAAM,EAAS,EAAS,CAAO,EAE9C,EAWA,OATY,IAAI,GAAO,CACrB,KAAM,aACN,OAAQ,CACN,YACE,4EACF,KAAM,CAAC,UAAW,MAAM,CAC1B,CACF,CAAC,EAII,MAAM,SAAU,CAAM,EACtB,MAAM,OAAQ,EAAO,IAAI,EACzB,MAAM,aAAc,OAAO,CAAC,CAAC,EAC7B,QAAQ,EAAG,YAAmB,CAC7B,GAAI,EACF,EAAY,EAAQ,CAAO,EACtB,KAEL,IAAM,EAAO,OAAO,QAAQ,IAAI,IAAI,GAAK,KACnC,EAAW,QAAQ,IAAI,MAAQ,YACrC,EAAY,CAAE,OAAM,WAAU,SAAU,MAAO,EAAG,CAAO,GAE5D,EACA,UAAU,EAAG,WAAY,CACxB,EAAM,WAAa,QAAQ,OAAO,OAAO,EAC1C,EACA,cAAc,EAAG,UAAS,MAAK,WAAY,CAC1C,GAAI,EAAa,IAAI,CAAO,EAC1B,OAGF,IAAM,EAAS,OAAO,EAAI,SAAW,SAAW,EAAI,OAAS,IACzD,EAAsC,OAC1C,GAAI,GAAU,IACZ,EAAQ,QACH,QAAI,GAAU,IACnB,EAAQ,UAGV,EAAO,IAAI,EAAO,EAAS,CAAE,QAAO,EAAG,CAAK,EAC7C,EACA,QAAQ,EAAG,UAAS,QAAO,WAAY,CACtC,EAAO,gBAAgB,EAAS,EAAO,CAAK,EAC7C,EAEA,GAAG,QAAQ,GAeH",
  "debugId": "F8359047E2C71CA864756E2164756E21",
  "names": []
}
9
+ //# debugId=5ECC4ED287A03C9864756E2164756E21
10
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["src/index.ts", "src/extensions/banner.ts", "src/extensions/index.ts", "src/logger/index.ts", "src/output/index.ts", "src/output/file.ts", "src/output/fs.ts", "src/output/rotation-manager.ts", "src/utils/rotation.ts", "src/utils/redact.ts", "src/logger/create-logger.ts", "src/helpers/status.ts", "src/utils/error.ts", "src/logger/handle-http-error.ts"],
  "sourcesContent": [
    "import { Elysia } from 'elysia'\nimport { startServer } from './extensions'\nimport type { LogixlysiaStore, Options } from './interfaces'\nimport { createLogger } from './logger'\n\n/**\n * Empty singleton slots must not use `Record<string, never>`: intersecting that with Elysia's `Context`\n * makes every key (including `store`) become `never` because each key is merged with `never`.\n */\nexport interface EmptyElysiaSlot {\n  readonly __logixlysiaEmpty?: never\n}\n\n/**\n * Explicit singleton without Elysia's `SingletonBase` `Record<string, unknown>` on decorator/derive/resolve so\n * merged `Context` and WebSocket `ws.data` keep precise keys after `.use(logixlysia())`.\n */\nexport interface LogixlysiaSingleton {\n  decorator: EmptyElysiaSlot\n  derive: EmptyElysiaSlot\n  resolve: EmptyElysiaSlot\n  store: LogixlysiaStore\n}\n\n// Elysia's `SingletonBase.store` is `Record<string, unknown>`; `LogixlysiaStore` is intentionally closed (see #220).\n// @ts-expect-error — closed store is correct at runtime and for merged `ws.data` inference.\nexport type Logixlysia = Elysia<'', LogixlysiaSingleton>\n\nexport const logixlysia = (options: Options = {}): Logixlysia => {\n  const didCustomLog = new WeakSet<Request>()\n  const baseLogger = createLogger(options)\n  const logger = {\n    ...baseLogger,\n    debug: (\n      request: Request,\n      message: string,\n      context?: Record<string, unknown>\n    ) => {\n      didCustomLog.add(request)\n      baseLogger.debug(request, message, context)\n    },\n    info: (\n      request: Request,\n      message: string,\n      context?: Record<string, unknown>\n    ) => {\n      didCustomLog.add(request)\n      baseLogger.info(request, message, context)\n    },\n    warn: (\n      request: Request,\n      message: string,\n      context?: Record<string, unknown>\n    ) => {\n      didCustomLog.add(request)\n      baseLogger.warn(request, message, context)\n    },\n    error: (\n      request: Request,\n      message: string,\n      context?: Record<string, unknown>\n    ) => {\n      didCustomLog.add(request)\n      baseLogger.error(request, message, context)\n    }\n  }\n\n  const app = new Elysia({\n    name: 'Logixlysia',\n    detail: {\n      description:\n        'Logixlysia is a plugin for Elysia that provides a logger and pino logger.',\n      tags: ['logging', 'pino']\n    }\n  })\n\n  return (\n    app\n      .state('logger', logger)\n      .state('pino', logger.pino)\n      .state('beforeTime', BigInt(0))\n      .onStart(({ server }): void => {\n        if (server) {\n          startServer(server, options)\n        } else {\n          // Node adapter fallback\n          const port = Number(process.env.PORT) || 3000\n          const hostname = process.env.HOST || 'localhost'\n          startServer({ port, hostname, protocol: 'http' }, options)\n        }\n      })\n      .onRequest(({ store }) => {\n        store.beforeTime = process.hrtime.bigint()\n      })\n      .onAfterHandle(({ request, set, store }) => {\n        if (didCustomLog.has(request)) {\n          return\n        }\n\n        const status = typeof set.status === 'number' ? set.status : 200\n        let level: 'INFO' | 'WARNING' | 'ERROR' = 'INFO'\n        if (status >= 500) {\n          level = 'ERROR'\n        } else if (status >= 400) {\n          level = 'WARNING'\n        }\n\n        logger.log(level, request, { status }, store)\n      })\n      .onError(({ request, error, store }) => {\n        logger.handleHttpError(request, error, store)\n      })\n      // Ensure plugin lifecycle hooks (onRequest/onAfterHandle/onError) apply to the parent app.\n      .as('scoped') as Logixlysia\n  )\n}\n\nexport type {\n  Logger,\n  LogixlysiaContext,\n  LogixlysiaStore,\n  LogLevel,\n  Options,\n  Pino,\n  StoreData,\n  Transport\n} from './interfaces'\n\nexport default logixlysia\n",
    "import { createRequire } from 'node:module'\n\nconst elysiaPkg: { version?: string } = (() => {\n  try {\n    const require = createRequire(import.meta.url)\n    return require('elysia/package.json') as { version?: string }\n  } catch {\n    return {}\n  }\n})()\n\nconst logixlysiaPkg: { version?: string } = (() => {\n  try {\n    const require = createRequire(import.meta.url)\n    return require('../../package.json') as { version?: string }\n  } catch {\n    return {}\n  }\n})()\n\nconst centerText = (text: string, width: number): string => {\n  if (text.length >= width) {\n    return text.slice(0, width)\n  }\n\n  const left = Math.floor((width - text.length) / 2)\n  const right = width - text.length - left\n  return `${' '.repeat(left)}${text}${' '.repeat(right)}`\n}\n\ntype RowSpec =\n  | { kind: 'empty' }\n  | { kind: 'center'; text: string }\n  | { kind: 'left'; text: string }\n\n/**\n * Box banner: Elysia version (centered), URL line, optional logixlysia version line.\n */\nexport const renderBanner = (\n  urlDisplayLine: string,\n  logixlysiaLine: string | null\n): string => {\n  const version = elysiaPkg.version\n  const versionLine = version ? `Elysia v${version}` : 'Elysia'\n\n  const rows: RowSpec[] = [\n    { kind: 'empty' },\n    { kind: 'center', text: versionLine },\n    { kind: 'empty' },\n    { kind: 'left', text: urlDisplayLine }\n  ]\n  if (logixlysiaLine) {\n    rows.push({ kind: 'left', text: logixlysiaLine })\n  }\n  rows.push({ kind: 'empty' })\n\n  const contentWidth = Math.max(\n    versionLine.length,\n    urlDisplayLine.length,\n    ...(logixlysiaLine ? [logixlysiaLine.length] : [0])\n  )\n  const innerWidth = contentWidth + 4\n\n  const top = `┌${'─'.repeat(innerWidth)}┐`\n  const bot = `└${'─'.repeat(innerWidth)}┘`\n  const emptyRow = `│${' '.repeat(innerWidth)}│`\n\n  const out: string[] = [top]\n  for (const row of rows) {\n    if (row.kind === 'empty') {\n      out.push(emptyRow)\n      continue\n    }\n    if (row.kind === 'center') {\n      out.push(`│${centerText(row.text, innerWidth)}│`)\n      continue\n    }\n    const text = row.text\n    const padding = Math.max(0, innerWidth - text.length - 4)\n    out.push(`│  ${text}${' '.repeat(padding)}  │`)\n  }\n  out.push(bot)\n  return out.join('\\n')\n}\n\nexport const getLogixlysiaVersionLine = (): string | null => {\n  const v = logixlysiaPkg.version\n  if (!v) {\n    return null\n  }\n  return `     logixlysia v${v}`\n}\n",
    "import type { Options } from '../interfaces'\nimport { getLogixlysiaVersionLine, renderBanner } from './banner'\n\nexport const startServer = (\n  server: { port?: number; hostname?: string; protocol?: string | null },\n  options: Options\n): void => {\n  const showStartupMessage = options.config?.showStartupMessage ?? true\n  if (!showStartupMessage) {\n    return\n  }\n\n  const { port, hostname, protocol } = server\n  if (port === undefined || !hostname || !protocol) {\n    return\n  }\n\n  const url = `${protocol}://${hostname}:${port}`\n  const message = `🦊 Elysia is running at ${url}`\n  const urlDisplayLine = `🦊  ${url}`\n\n  const format = options.config?.startupMessageFormat ?? 'banner'\n  if (format === 'simple') {\n    console.log(message)\n    return\n  }\n\n  console.log(renderBanner(urlDisplayLine, getLogixlysiaVersionLine()))\n}\n",
    "import pino from 'pino'\nimport type {\n  LogFilter,\n  Logger,\n  LogLevel,\n  Options,\n  Pino,\n  RequestInfo,\n  StoreData\n} from '../interfaces'\nimport { logToTransports } from '../output'\nimport { logToFile } from '../output/file'\nimport { redact, redactRequest } from '../utils/redact'\nimport { formatLogOutput } from './create-logger'\nimport { handleHttpError } from './handle-http-error'\n\nexport const createLogger = (\n  options: Options = {},\n  pinoFactory: typeof pino = pino\n): Logger => {\n  const config = options.config\n\n  const pinoConfig = config?.pino\n  const { prettyPrint, ...pinoOptions } = pinoConfig ?? {}\n\n  const prettyPrintOptions =\n    typeof prettyPrint === 'object' && prettyPrint !== null\n      ? (prettyPrint as Record<string, unknown>)\n      : undefined\n\n  const enablePrettyPrint =\n    prettyPrint === true || prettyPrintOptions !== undefined\n\n  const shouldPrettyPrint =\n    enablePrettyPrint && pinoOptions.transport === undefined\n\n  const messageKey =\n    (prettyPrintOptions?.messageKey as string | undefined) ??\n    pinoOptions.messageKey\n  const errorKey =\n    (prettyPrintOptions?.errorKey as string | undefined) ?? pinoOptions.errorKey\n\n  const transport = shouldPrettyPrint\n    ? {\n        target: 'pino-pretty',\n        options: {\n          colorize: process.stdout?.isTTY === true,\n          translateTime: config?.timestamp?.translateTime,\n          ...prettyPrintOptions,\n          messageKey,\n          errorKey\n        }\n      }\n    : pinoOptions.transport\n\n  const pinoLogger: Pino = pinoFactory({\n    ...pinoOptions,\n    level: pinoOptions.level ?? 'info',\n    messageKey,\n    errorKey,\n    transport\n  })\n\n  const shouldLog = (level: LogLevel, logFilter?: LogFilter): boolean => {\n    if (!logFilter?.level || logFilter.level.length === 0) {\n      return true\n    }\n    return logFilter.level.includes(level)\n  }\n\n  const log = (\n    level: LogLevel,\n    request: RequestInfo,\n    data: Record<string, unknown>,\n    store: StoreData\n  ): void => {\n    // Check if this log level should be filtered\n    if (!shouldLog(level, config?.logFilter)) {\n      return\n    }\n\n    const logData = config?.autoRedact === true ? redact(data) : data\n    const logRequest =\n      config?.autoRedact === true ? redactRequest(request) : request\n\n    logToTransports({\n      level,\n      request: logRequest,\n      data: logData,\n      store,\n      options\n    })\n\n    const useTransportsOnly = config?.useTransportsOnly === true\n    const disableInternalLogger = config?.disableInternalLogger === true\n    const disableFileLogging = config?.disableFileLogging === true\n\n    if (!(useTransportsOnly || disableFileLogging)) {\n      const filePath = config?.logFilePath\n      if (filePath) {\n        logToFile({\n          filePath,\n          level,\n          request: logRequest,\n          data: logData,\n          store,\n          options\n        }).catch(() => {\n          // Ignore errors\n        })\n      }\n    }\n\n    if (useTransportsOnly || disableInternalLogger) {\n      return\n    }\n\n    const { main, contextLines } = formatLogOutput({\n      level,\n      request: logRequest,\n      data: logData,\n      store,\n      options\n    })\n    const message =\n      contextLines.length > 0 ? `${main}\\n${contextLines.join('\\n')}` : main\n\n    switch (level) {\n      case 'DEBUG': {\n        console.debug(message)\n        break\n      }\n      case 'INFO': {\n        console.info(message)\n        break\n      }\n      case 'WARNING': {\n        console.warn(message)\n        break\n      }\n      case 'ERROR': {\n        console.error(message)\n        break\n      }\n      default: {\n        console.log(message)\n        break\n      }\n    }\n  }\n\n  const logWithContext = (\n    level: LogLevel,\n    request: RequestInfo,\n    message: string,\n    context?: Record<string, unknown>\n  ): void => {\n    const store: StoreData = { beforeTime: process.hrtime.bigint() }\n    log(level, request, { message, context }, store)\n  }\n\n  return {\n    pino: pinoLogger,\n    log,\n    handleHttpError: (request, error, store) => {\n      handleHttpError(request, error, store, options)\n    },\n    debug: (request, message, context) => {\n      logWithContext('DEBUG', request, message, context)\n    },\n    info: (request, message, context) => {\n      logWithContext('INFO', request, message, context)\n    },\n    warn: (request, message, context) => {\n      logWithContext('WARNING', request, message, context)\n    },\n    error: (request, message, context) => {\n      logWithContext('ERROR', request, message, context)\n    }\n  }\n}\n",
    "import type { LogLevel, Options, RequestInfo, StoreData } from '../interfaces'\n\ninterface LogToTransportsInput {\n  data: Record<string, unknown>\n  level: LogLevel\n  options: Options\n  request: RequestInfo\n  store: StoreData\n}\n\nexport const logToTransports = (\n  ...args:\n    | [LogToTransportsInput]\n    | [LogLevel, RequestInfo, Record<string, unknown>, StoreData, Options]\n): void => {\n  const input: LogToTransportsInput =\n    typeof args[0] === 'string'\n      ? {\n          level: args[0],\n          request: args[1],\n          data: args[2],\n          store: args[3],\n          options: args[4]\n        }\n      : args[0]\n\n  const { level, request, data, store, options } = input\n  const transports = options.config?.transports ?? []\n  if (transports.length === 0) {\n    return\n  }\n\n  const message = typeof data.message === 'string' ? data.message : ''\n  const meta: Record<string, unknown> = {\n    request: {\n      method: request.method,\n      url: request.url\n    },\n    ...data,\n    beforeTime: store.beforeTime\n  }\n\n  for (const transport of transports) {\n    try {\n      const result = transport.log(level, message, meta)\n      if (\n        result &&\n        typeof (result as { catch?: unknown }).catch === 'function'\n      ) {\n        ;(result as Promise<void>).catch(() => {\n          // Ignore errors\n        })\n      }\n    } catch {\n      // Transport failures must never crash application logging.\n    }\n  }\n}\n",
    "import { appendFile } from 'node:fs/promises'\nimport { dirname } from 'node:path'\nimport type { LogLevel, Options, RequestInfo, StoreData } from '../interfaces'\nimport { ensureDir } from './fs'\nimport { performRotation, shouldRotate } from './rotation-manager'\n\n// Per-file mutex to prevent race conditions during rotation\nconst fileLocks = new Map<string, Promise<void>>()\n\nconst acquireLock = (filePath: string): Promise<() => void> => {\n  const prior = fileLocks.get(filePath) ?? Promise.resolve()\n\n  let resolveLock: () => void\n  const newLock = new Promise<void>(resolve => {\n    resolveLock = resolve\n  })\n\n  return prior.then(() => {\n    // Only set the lock after acquiring the prior lock to prevent race conditions\n    fileLocks.set(filePath, newLock)\n\n    // Critical section can now proceed\n    return () => {\n      resolveLock!()\n      if (fileLocks.get(filePath) === newLock) {\n        fileLocks.delete(filePath)\n      }\n    }\n  })\n}\n\ninterface LogToFileInput {\n  data: Record<string, unknown>\n  filePath: string\n  level: LogLevel\n  options: Options\n  request: RequestInfo\n  store: StoreData\n}\n\nexport const logToFile = async (\n  ...args:\n    | [LogToFileInput]\n    | [\n        string,\n        LogLevel,\n        RequestInfo,\n        Record<string, unknown>,\n        StoreData,\n        Options\n      ]\n): Promise<void> => {\n  const input: LogToFileInput =\n    typeof args[0] === 'string'\n      ? (() => {\n          const [\n            filePathArg,\n            levelArg,\n            requestArg,\n            dataArg,\n            storeArg,\n            optionsArg\n          ] = args as [\n            string,\n            LogLevel,\n            RequestInfo,\n            Record<string, unknown>,\n            StoreData,\n            Options\n          ]\n          return {\n            filePath: filePathArg,\n            level: levelArg,\n            request: requestArg,\n            data: dataArg,\n            store: storeArg,\n            options: optionsArg\n          }\n        })()\n      : args[0]\n\n  const { filePath, level, request, data, store, options } = input\n  const config = options.config\n  const useTransportsOnly = config?.useTransportsOnly === true\n  const disableFileLogging = config?.disableFileLogging === true\n  if (useTransportsOnly || disableFileLogging) {\n    return\n  }\n\n  const message = typeof data.message === 'string' ? data.message : ''\n  const durationMs =\n    store.beforeTime === BigInt(0)\n      ? 0\n      : Number(process.hrtime.bigint() - store.beforeTime) / 1_000_000\n\n  // Safely parse URL to avoid crashes on malformed URLs\n  let pathname = '/'\n  try {\n    pathname = new URL(request.url).pathname\n  } catch {\n    // Fallback to raw URL if parsing fails\n    pathname = request.url\n  }\n\n  const line = `${level} ${durationMs.toFixed(2)}ms ${request.method} ${pathname} ${message}\\n`\n\n  // Acquire lock before any file operations to prevent race conditions\n  const releaseLock = await acquireLock(filePath)\n\n  try {\n    try {\n      await ensureDir(dirname(filePath))\n      await appendFile(filePath, line, { encoding: 'utf-8' })\n    } catch (error) {\n      // Log file write errors to stderr so they're not completely silent\n      console.error(\n        `[logixlysia] Failed to write to log file ${filePath}:`,\n        error\n      )\n      throw error\n    }\n\n    const rotation = config?.logRotation\n    if (!rotation) {\n      return\n    }\n\n    const should = await shouldRotate(filePath, rotation)\n    if (should) {\n      try {\n        await performRotation(filePath, rotation)\n      } catch (error) {\n        // Log rotation errors but don't crash - log entry was already written\n        console.error(\n          `[logixlysia] Failed to rotate log file ${filePath}:`,\n          error\n        )\n      }\n    }\n  } finally {\n    // Release lock\n    releaseLock()\n  }\n}\n",
    "import { promises as fs } from 'node:fs'\n\nexport const ensureDir = async (dirPath: string): Promise<void> => {\n  await fs.mkdir(dirPath, { recursive: true })\n}\n",
    "import { promises as fs } from 'node:fs'\nimport { promisify } from 'node:util'\nimport { gzip } from 'node:zlib'\nimport type { LogRotationConfig } from '../interfaces'\nimport {\n  getRotatedFiles,\n  parseRetention,\n  parseSize,\n  shouldRotateBySize\n} from '../utils/rotation'\n\nconst gzipAsync = promisify(gzip)\n\n// Compression lock to prevent concurrent compression of the same file\nconst compressionLocks = new Map<string, Promise<void>>()\n\nconst acquireCompressionLock = (filePath: string): Promise<() => void> => {\n  const prior = compressionLocks.get(filePath) ?? Promise.resolve()\n\n  let resolveLock: () => void\n  const newLock = new Promise<void>(resolve => {\n    resolveLock = resolve\n  })\n\n  return prior.then(() => {\n    // Only set the lock after acquiring the prior lock to prevent race conditions\n    compressionLocks.set(filePath, newLock)\n\n    // Critical section can now proceed\n    return () => {\n      resolveLock!()\n      if (compressionLocks.get(filePath) === newLock) {\n        compressionLocks.delete(filePath)\n      }\n    }\n  })\n}\n\nconst pad2 = (value: number): string => String(value).padStart(2, '0')\n\nexport const getRotatedFileName = (filePath: string, date: Date): string => {\n  const yyyy = date.getFullYear()\n  const mm = pad2(date.getMonth() + 1)\n  const dd = pad2(date.getDate())\n  const HH = pad2(date.getHours())\n  const MM = pad2(date.getMinutes())\n  const ss = pad2(date.getSeconds())\n  const SSS = String(date.getMilliseconds()).padStart(3, '0')\n  return `${filePath}.${yyyy}-${mm}-${dd}-${HH}-${MM}-${ss}-${SSS}`\n}\n\nexport const rotateFile = async (filePath: string): Promise<string> => {\n  try {\n    const stat = await fs.stat(filePath)\n    if (stat.size === 0) {\n      return ''\n    }\n  } catch {\n    return ''\n  }\n\n  const baseRotated = getRotatedFileName(filePath, new Date())\n  const rotated = `${baseRotated}-${process.hrtime.bigint()}`\n  await fs.rename(filePath, rotated)\n  return rotated\n}\n\nexport const compressFile = async (filePath: string): Promise<void> => {\n  const release = await acquireCompressionLock(filePath)\n  try {\n    // Check if file still exists (might have been compressed by another process)\n    try {\n      await fs.access(filePath)\n    } catch {\n      // File doesn't exist, already compressed or deleted\n      return\n    }\n\n    const content = await fs.readFile(filePath)\n    const compressed = await gzipAsync(content)\n    await fs.writeFile(`${filePath}.gz`, compressed)\n    await fs.rm(filePath, { force: true })\n  } catch (error) {\n    console.error(`[logixlysia] Failed to compress file ${filePath}:`, error)\n    throw error\n  } finally {\n    release()\n  }\n}\n\nexport const shouldRotate = async (\n  filePath: string,\n  config: LogRotationConfig\n): Promise<boolean> => {\n  if (config.maxSize === undefined) {\n    return false\n  }\n  const maxSize = parseSize(config.maxSize)\n  return await shouldRotateBySize(filePath, maxSize)\n}\n\nconst cleanupByCount = async (\n  filePath: string,\n  maxFiles: number\n): Promise<void> => {\n  const rotated = await getRotatedFiles(filePath)\n  if (rotated.length <= maxFiles) {\n    return\n  }\n\n  // Use Promise.allSettled to handle individual file stat failures gracefully\n  const statsResults = await Promise.allSettled(\n    rotated.map(async p => ({ path: p, stat: await fs.stat(p) }))\n  )\n\n  // Extract successful stats, ignore files that were deleted concurrently\n  const stats = statsResults\n    .filter(\n      (\n        result\n      ): result is PromiseFulfilledResult<{\n        path: string\n        stat: import('node:fs').Stats\n      }> => result.status === 'fulfilled'\n    )\n    .map(result => result.value)\n\n  if (stats.length <= maxFiles) {\n    return\n  }\n\n  stats.sort((a, b) => b.stat.mtimeMs - a.stat.mtimeMs)\n  const toDelete = stats.slice(maxFiles)\n\n  // Delete files individually, continuing even if some fail\n  const deleteResults = await Promise.allSettled(\n    toDelete.map(({ path }) => fs.rm(path, { force: true }))\n  )\n\n  // Log failures but don't crash\n  deleteResults.forEach((result, idx) => {\n    if (result.status === 'rejected') {\n      console.error(\n        `[logixlysia] Failed to delete rotated log ${toDelete[idx].path}:`,\n        result.reason\n      )\n    }\n  })\n}\n\nconst cleanupByTime = async (\n  filePath: string,\n  maxAgeMs: number\n): Promise<void> => {\n  const rotated = await getRotatedFiles(filePath)\n  if (rotated.length === 0) {\n    return\n  }\n\n  const now = Date.now()\n\n  // Use Promise.allSettled to handle individual file stat failures gracefully\n  const statsResults = await Promise.allSettled(\n    rotated.map(async p => ({ path: p, stat: await fs.stat(p) }))\n  )\n\n  // Extract successful stats\n  const stats = statsResults\n    .filter(\n      (\n        result\n      ): result is PromiseFulfilledResult<{\n        path: string\n        stat: import('node:fs').Stats\n      }> => result.status === 'fulfilled'\n    )\n    .map(result => result.value)\n\n  const toDelete = stats.filter(({ stat }) => now - stat.mtimeMs > maxAgeMs)\n\n  // Delete files individually, continuing even if some fail\n  const deleteResults = await Promise.allSettled(\n    toDelete.map(({ path }) => fs.rm(path, { force: true }))\n  )\n\n  // Log failures but don't crash\n  deleteResults.forEach((result, idx) => {\n    if (result.status === 'rejected') {\n      console.error(\n        `[logixlysia] Failed to delete old log ${toDelete[idx].path}:`,\n        result.reason\n      )\n    }\n  })\n}\n\nexport const performRotation = async (\n  filePath: string,\n  config: LogRotationConfig\n): Promise<void> => {\n  const rotated = await rotateFile(filePath)\n  if (!rotated) {\n    return\n  }\n\n  const shouldCompress = config.compress === true\n  if (shouldCompress) {\n    const algo = config.compression ?? 'gzip'\n    if (algo === 'gzip') {\n      await compressFile(rotated)\n    }\n  }\n\n  if (config.maxFiles !== undefined) {\n    const retention = parseRetention(config.maxFiles)\n    if (retention.type === 'count') {\n      await cleanupByCount(filePath, retention.value)\n    } else {\n      await cleanupByTime(filePath, retention.value)\n    }\n  }\n\n  // Optional interval-based rotation cleanup (create interval directories / naming) is not required by tests.\n}\n",
    "import { promises as fs } from 'node:fs'\nimport { basename, dirname } from 'node:path'\n\nconst SIZE_REGEX = /^(\\d+(?:\\.\\d+)?)(k|kb|m|mb|g|gb)$/i\nconst INTERVAL_REGEX = /^(\\d+)(h|d|w)$/i\nconst ROTATED_REGEX =\n  /\\.(\\d{4}-\\d{2}-\\d{2}-\\d{2}-\\d{2}-\\d{2}(?:-\\d{3})?(?:-\\d+)?)(?:\\.gz)?$/\n\nexport const parseSize = (value: number | string): number => {\n  if (typeof value === 'number') {\n    return value\n  }\n\n  const trimmed = value.trim()\n  const asNumber = Number(trimmed)\n  if (Number.isFinite(asNumber)) {\n    return asNumber\n  }\n\n  const match = trimmed.match(SIZE_REGEX)\n  if (!match) {\n    throw new Error(`Invalid size format: ${value}`)\n  }\n\n  const amount = Number(match[1])\n  const unit = match[2].toLowerCase()\n\n  let base = 1024\n  if (unit.startsWith('m')) {\n    base = 1024 * 1024\n  } else if (unit.startsWith('g')) {\n    base = 1024 * 1024 * 1024\n  }\n\n  return Math.floor(amount * base)\n}\n\nexport const parseInterval = (value: string): number => {\n  const match = value.trim().match(INTERVAL_REGEX)\n  if (!match) {\n    throw new Error(`Invalid interval format: ${value}`)\n  }\n\n  const amount = Number(match[1])\n  const unit = match[2].toLowerCase()\n\n  let ms = 60 * 60 * 1000\n  if (unit === 'd') {\n    ms = 24 * 60 * 60 * 1000\n  } else if (unit === 'w') {\n    ms = 7 * 24 * 60 * 60 * 1000\n  }\n\n  return amount * ms\n}\n\nexport const parseRetention = (\n  value: number | string\n): { type: 'count' | 'time'; value: number } => {\n  if (typeof value === 'number') {\n    return { type: 'count', value }\n  }\n  return { type: 'time', value: parseInterval(value) }\n}\n\nexport const shouldRotateBySize = async (\n  filePath: string,\n  maxSizeBytes: number\n): Promise<boolean> => {\n  try {\n    const stat = await fs.stat(filePath)\n    return stat.size > maxSizeBytes\n  } catch {\n    return false\n  }\n}\n\nexport const getRotatedFiles = async (filePath: string): Promise<string[]> => {\n  const dir = dirname(filePath)\n  const base = basename(filePath)\n\n  let entries: string[]\n  try {\n    entries = await fs.readdir(dir)\n  } catch {\n    return []\n  }\n\n  return entries\n    .filter(name => name.startsWith(`${base}.`) && ROTATED_REGEX.test(name))\n    .map(name => `${dir}/${name}`)\n}\n",
    "const EMAIL_REGEX = /[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\\.[a-zA-Z]{2,}/g\nconst IPV4_REGEX = /\\b(?:\\d{1,3}\\.){3}\\d{1,3}\\b/g\n/** Digit runs that may be formatted PANs (spaces/dashes); validated with Luhn before redacting. */\nconst CREDIT_CARD_CANDIDATE_REGEX = /\\b(?:\\d[ -]*?){13,19}\\b/g\nconst JWT_REGEX = /eyJ[a-zA-Z0-9_-]+\\.eyJ[a-zA-Z0-9_-]+\\.[a-zA-Z0-9_-]+/g\n\nconst PAN_MIN_LEN = 13\nconst PAN_MAX_LEN = 19\n\nconst REDACTED_TEXT = '[REDACTED]'\nconst CIRCULAR_REF = '[Circular]'\n\n/**\n * Host and userinfo cannot contain `[REDACTED]` — `[` begins an IPv6 literal in URLs and breaks parsing.\n */\nconst URL_SAFE_REDACT = 'redacted'\n\nconst redactUrlAuthoritySegment = (value: string): string =>\n  redactString(value).replaceAll(REDACTED_TEXT, URL_SAFE_REDACT)\n\n/** Apply PII redaction to a request URL while keeping the result parseable by the URL/Request constructors. */\nconst redactRequestUrl = (urlString: string): string => {\n  try {\n    const u = new URL(urlString)\n    if (u.username !== '') {\n      u.username = redactUrlAuthoritySegment(u.username)\n    }\n    if (u.password !== '') {\n      u.password = redactUrlAuthoritySegment(u.password)\n    }\n    u.hostname = redactUrlAuthoritySegment(u.hostname)\n    u.pathname = redactString(u.pathname)\n    u.search = redactString(u.search)\n    u.hash = redactString(u.hash)\n    return u.toString()\n  } catch {\n    return redactString(urlString).replaceAll(REDACTED_TEXT, URL_SAFE_REDACT)\n  }\n}\n\n/** Luhn checksum; `digits` must contain only `0-9` and length in PAN range. */\nconst passesLuhn = (digits: string): boolean => {\n  if (digits.length < PAN_MIN_LEN || digits.length > PAN_MAX_LEN) {\n    return false\n  }\n\n  let sum = 0\n  let alternate = false\n\n  for (let i = digits.length - 1; i >= 0; i--) {\n    const code = digits.charCodeAt(i)\n    if (code < 48 || code > 57) {\n      return false\n    }\n    let n = code - 48\n    if (alternate) {\n      n *= 2\n      if (n > 9) {\n        n -= 9\n      }\n    }\n    sum += n\n    alternate = !alternate\n  }\n\n  return sum % 10 === 0\n}\n\nconst redactCreditCardCandidates = (text: string): string =>\n  text.replace(CREDIT_CARD_CANDIDATE_REGEX, match => {\n    const digits = match.replace(/\\D/g, '')\n    if (\n      digits.length >= PAN_MIN_LEN &&\n      digits.length <= PAN_MAX_LEN &&\n      passesLuhn(digits)\n    ) {\n      return REDACTED_TEXT\n    }\n    return match\n  })\n\nexport const redactString = (text: string): string => {\n  let result = text\n\n  result = result.replace(EMAIL_REGEX, REDACTED_TEXT)\n  result = result.replace(IPV4_REGEX, REDACTED_TEXT)\n  result = redactCreditCardCandidates(result)\n  result = result.replace(JWT_REGEX, REDACTED_TEXT)\n\n  return result\n}\n\nconst redactErrorClone = (\n  originalError: Error,\n  inProgress: WeakSet<object>\n): Error & Record<string, unknown> => {\n  const redactedMessage = redactString(originalError.message)\n  const proto = Object.getPrototypeOf(originalError) as object\n  const newError = Object.create(proto) as Error & Record<string, unknown>\n\n  newError.message = redactedMessage\n  newError.name = originalError.name\n\n  if (originalError.stack !== undefined) {\n    newError.stack = redactString(originalError.stack)\n  }\n\n  const errorRecord = originalError as unknown as Record<string, unknown>\n\n  for (const key of Object.keys(errorRecord)) {\n    if (key !== 'message' && key !== 'name' && key !== 'stack') {\n      newError[key] = redactInner(errorRecord[key], inProgress)\n    }\n  }\n\n  return newError\n}\n\nconst redactArrayItems = (\n  value: unknown[],\n  inProgress: WeakSet<object>\n): unknown[] => {\n  const redactedArray: unknown[] = Array.from({ length: value.length })\n  for (let i = 0; i < value.length; i++) {\n    redactedArray[i] = redactInner(value[i], inProgress)\n  }\n  return redactedArray\n}\n\nconst redactRecordEntries = (\n  recordValue: Record<string, unknown>,\n  inProgress: WeakSet<object>\n): Record<string, unknown> => {\n  const redactedRecord: Record<string, unknown> = {}\n  for (const key of Object.keys(recordValue)) {\n    redactedRecord[key] = redactInner(recordValue[key], inProgress)\n  }\n  return redactedRecord\n}\n\nconst withReentrancyGuard = <T>(\n  obj: object,\n  inProgress: WeakSet<object>,\n  run: () => T\n): T => {\n  inProgress.add(obj)\n  try {\n    return run()\n  } finally {\n    inProgress.delete(obj)\n  }\n}\n\nconst redactInner = <T>(value: T, inProgress: WeakSet<object>): T => {\n  if (value === null || value === undefined) {\n    return value\n  }\n\n  if (typeof value === 'string') {\n    return redactString(value) as unknown as T\n  }\n\n  const type = typeof value\n  if (type !== 'object') {\n    return value\n  }\n\n  if (value instanceof Date) {\n    return new Date(value.getTime()) as unknown as T\n  }\n\n  const obj = value as object\n  if (inProgress.has(obj)) {\n    return CIRCULAR_REF as unknown as T\n  }\n\n  if (value instanceof Error) {\n    return withReentrancyGuard(obj, inProgress, () =>\n      redactErrorClone(value, inProgress)\n    ) as unknown as T\n  }\n\n  if (Array.isArray(value)) {\n    return withReentrancyGuard(obj, inProgress, () =>\n      redactArrayItems(value, inProgress)\n    ) as unknown as T\n  }\n\n  return withReentrancyGuard(obj, inProgress, () =>\n    redactRecordEntries(value as Record<string, unknown>, inProgress)\n  ) as unknown as T\n}\n\nexport const redact = <T>(value: T): T => redactInner(value, new WeakSet())\n\n/**\n * Clone request URL and headers for logging with the same string redaction as {@link redact}.\n * Preserves body and signal so the original request is still usable.\n */\nexport const redactRequest = (request: Request): Request => {\n  const redactedUrl = redactRequestUrl(request.url)\n  const nextHeaders = new Headers()\n  let headersChanged = false\n\n  for (const [name, value] of request.headers.entries()) {\n    const redacted = redactString(value)\n    if (redacted !== value) {\n      headersChanged = true\n    }\n    nextHeaders.set(name, redacted)\n  }\n\n  const redactedMethod = redactString(request.method)\n  const urlChanged = redactedUrl !== request.url\n  const methodChanged = redactedMethod !== request.method\n\n  if (!(urlChanged || headersChanged || methodChanged)) {\n    return request\n  }\n\n  const init: RequestInit & { duplex?: 'half' } = {\n    method: redactedMethod,\n    headers: nextHeaders,\n    redirect: request.redirect,\n    signal: request.signal\n  }\n\n  if (request.body !== null) {\n    init.body = request.body\n    init.duplex = 'half'\n  }\n\n  return new Request(redactedUrl, init)\n}\n",
    "import { STATUS_CODES } from 'node:http'\nimport chalk from 'chalk'\nimport { getStatusCode } from '../helpers/status'\nimport type {\n  LogLevel,\n  Options,\n  Pino,\n  RequestInfo,\n  StoreData\n} from '../interfaces'\nimport { parseError } from '../utils/error'\n\nconst pad2 = (value: number): string => String(value).padStart(2, '0')\nconst pad3 = (value: number): string => String(value).padStart(3, '0')\n\nconst DEFAULT_SLOW_MS = 500\nconst DEFAULT_VERY_SLOW_MS = 1000\nconst METHOD_PAD = 7\n\nconst DEFAULT_LOG_FORMAT =\n  '{now} {service}{icon} {method} {pathname} {status} {duration} {message}{speed}'\n\nexport interface FormattedLogOutput {\n  contextLines: string[]\n  main: string\n}\n\nconst shouldUseColors = (options: Options): boolean => {\n  const config = options.config\n  const enabledByConfig = config?.useColors ?? true\n\n  // Avoid ANSI sequences in non-interactive output (pipes, CI logs, files).\n  const isTty = typeof process !== 'undefined' && process.stdout?.isTTY === true\n  return enabledByConfig && isTty\n}\n\nconst formatTimestamp = (date: Date, pattern?: string): string => {\n  if (!pattern) {\n    return date.toISOString()\n  }\n\n  const yyyy = String(date.getFullYear())\n  const mm = pad2(date.getMonth() + 1)\n  const dd = pad2(date.getDate())\n  const HH = pad2(date.getHours())\n  const MM = pad2(date.getMinutes())\n  const ss = pad2(date.getSeconds())\n  const SSS = pad3(date.getMilliseconds())\n\n  return pattern\n    .replaceAll('yyyy', yyyy)\n    .replaceAll('mm', mm)\n    .replaceAll('dd', dd)\n    .replaceAll('HH', HH)\n    .replaceAll('MM', MM)\n    .replaceAll('ss', ss)\n    .replaceAll('SSS', SSS)\n}\n\n/** Resolves client IP from x-forwarded-for (first IP) or x-real-ip. Empty when neither header is set (e.g. localhost). */\nconst getIp = (request: RequestInfo): string => {\n  const forwarded = request.headers.get('x-forwarded-for')\n  if (forwarded) {\n    return forwarded.split(',')[0]?.trim() ?? ''\n  }\n  return request.headers.get('x-real-ip') ?? ''\n}\n\nexport const formatDuration = (ms: number): string => {\n  if (ms >= 1000) {\n    const sec = ms / 1000\n    if (sec >= 10) {\n      return `${Math.round(sec)}s`\n    }\n    const oneDecimal = sec.toFixed(1)\n    return oneDecimal.endsWith('.0') ? `${Math.round(sec)}s` : `${oneDecimal}s`\n  }\n  if (ms > 0 && ms < 1) {\n    return `${ms.toFixed(2)}ms`\n  }\n  return `${Math.round(ms)}ms`\n}\n\nconst getSlowThresholds = (\n  options: Options\n): { slow: number; verySlow: number } => {\n  const config = options.config\n  return {\n    slow: config?.slowThreshold ?? DEFAULT_SLOW_MS,\n    verySlow: config?.verySlowThreshold ?? DEFAULT_VERY_SLOW_MS\n  }\n}\n\nconst colorDurationText = (\n  ms: number,\n  useColors: boolean,\n  options: Options\n): { text: string; isVerySlow: boolean } => {\n  const raw = formatDuration(ms)\n  const { slow, verySlow } = getSlowThresholds(options)\n  const isVerySlow = ms >= verySlow\n\n  if (!useColors) {\n    return { text: raw, isVerySlow }\n  }\n\n  let colored = raw\n  if (ms < slow) {\n    colored = chalk.green(raw)\n  } else if (ms < verySlow) {\n    colored = chalk.yellow(raw)\n  } else {\n    colored = chalk.red.bold(raw)\n  }\n\n  return { text: colored, isVerySlow }\n}\n\nconst getSpeedToken = (isVerySlow: boolean, useColors: boolean): string => {\n  if (!isVerySlow) {\n    return ''\n  }\n  const badge = '⚡ slow'\n  if (!useColors) {\n    return ` ${badge}`\n  }\n  return ` ${chalk.yellow(badge)}`\n}\n\n/** Logixlysia brand: fox emoji with level-colored background when colors are enabled. */\nconst getLevelIcon = (level: LogLevel, useColors: boolean): string => {\n  if (!useColors) {\n    return '🦊'\n  }\n  if (level === 'ERROR') {\n    return chalk.bgRed.black(' 🦊 ')\n  }\n  if (level === 'WARNING') {\n    return chalk.bgYellow.black(' 🦊 ')\n  }\n  if (level === 'DEBUG') {\n    return chalk.bgBlue.black(' 🦊 ')\n  }\n  return chalk.bgGreen.black(' 🦊 ')\n}\n\nconst getColoredLevel = (level: LogLevel, useColors: boolean): string => {\n  if (!useColors) {\n    return level\n  }\n\n  if (level === 'ERROR') {\n    return chalk.bgRed.black(level)\n  }\n  if (level === 'WARNING') {\n    return chalk.bgYellow.black(level)\n  }\n  if (level === 'DEBUG') {\n    return chalk.bgBlue.black(level)\n  }\n\n  return chalk.bgGreen.black(level)\n}\n\nconst getColoredMethod = (method: string, useColors: boolean): string => {\n  if (!useColors) {\n    return method\n  }\n\n  const upper = method.toUpperCase()\n  if (upper === 'GET') {\n    return chalk.green.bold(upper)\n  }\n  if (upper === 'POST') {\n    return chalk.blue.bold(upper)\n  }\n  if (upper === 'PUT') {\n    return chalk.yellow.bold(upper)\n  }\n  if (upper === 'PATCH') {\n    return chalk.yellowBright.bold(upper)\n  }\n  if (upper === 'DELETE') {\n    return chalk.red.bold(upper)\n  }\n  if (upper === 'OPTIONS') {\n    return chalk.cyan.bold(upper)\n  }\n  if (upper === 'HEAD') {\n    return chalk.greenBright.bold(upper)\n  }\n  if (upper === 'TRACE') {\n    return chalk.magenta.bold(upper)\n  }\n  if (upper === 'CONNECT') {\n    return chalk.cyanBright.bold(upper)\n  }\n\n  return chalk.white.bold(upper)\n}\n\nconst getColoredStatus = (status: string, useColors: boolean): string => {\n  if (!useColors) {\n    return status\n  }\n\n  const numeric = Number.parseInt(status, 10)\n  if (!Number.isFinite(numeric)) {\n    return status\n  }\n\n  if (numeric >= 500) {\n    return chalk.red(status)\n  }\n  if (numeric >= 400) {\n    return chalk.yellow(status)\n  }\n  if (numeric >= 300) {\n    return chalk.cyan(status)\n  }\n  if (numeric >= 200) {\n    return chalk.green(status)\n  }\n  return chalk.gray(status)\n}\n\nconst getColoredTimestamp = (timestamp: string, useColors: boolean): string => {\n  if (!useColors) {\n    return timestamp\n  }\n\n  return chalk.gray(timestamp)\n}\n\nconst getColoredPathname = (pathname: string, useColors: boolean): string => {\n  if (!useColors) {\n    return pathname\n  }\n\n  return chalk.whiteBright(pathname)\n}\n\nconst getStatusText = (statusCode: number): string => {\n  const text = STATUS_CODES[statusCode]\n  return text ?? ''\n}\n\nconst getServiceToken = (options: Options, useColors: boolean): string => {\n  const name = options.config?.service?.trim()\n  if (!name) {\n    return ''\n  }\n  const bracketed = `[${name}]`\n  if (!useColors) {\n    return `${bracketed} `\n  }\n  return `${chalk.dim(bracketed)} `\n}\n\nconst stringifyTreeValue = (value: unknown): string => {\n  if (value === null) {\n    return 'null'\n  }\n  if (value === undefined) {\n    return 'undefined'\n  }\n  if (typeof value === 'string') {\n    return value\n  }\n  if (typeof value === 'number' || typeof value === 'boolean') {\n    return String(value)\n  }\n  if (value instanceof Error) {\n    return value.message\n  }\n  try {\n    return JSON.stringify(value)\n  } catch {\n    return String(value)\n  }\n}\n\n/** Nested objects to expand in the context tree (excludes Arrays, Error, Date). */\nconst isExpandableObject = (value: unknown): value is Record<string, unknown> =>\n  value !== null &&\n  typeof value === 'object' &&\n  !Array.isArray(value) &&\n  !(value instanceof Error) &&\n  !(value instanceof Date)\n\nconst collectContextEntries = (\n  obj: Record<string, unknown>,\n  prefix: string,\n  depthRemaining: number\n): [string, string][] => {\n  const out: [string, string][] = []\n  for (const [k, v] of Object.entries(obj)) {\n    const key = prefix ? `${prefix}.${k}` : k\n    const expandable = isExpandableObject(v) && depthRemaining > 1\n\n    if (expandable) {\n      out.push(...collectContextEntries(v, key, depthRemaining - 1))\n    } else {\n      out.push([key, stringifyTreeValue(v)])\n    }\n  }\n  return out\n}\n\nconst formatEntriesToTreeLines = (\n  entries: [string, string][],\n  useColors: boolean\n): string[] => {\n  if (entries.length === 0) {\n    return []\n  }\n\n  const lines: string[] = []\n  const last = entries.length - 1\n  for (let i = 0; i < entries.length; i++) {\n    const branch = i === last ? '└─' : '├─'\n    const pair = entries[i]\n    if (!pair) {\n      continue\n    }\n    const [k, v] = pair\n    const keyPart = useColors ? chalk.cyan(k) : k\n    const valPart = useColors ? chalk.white(v) : v\n    lines.push(`  ${branch} ${keyPart}  ${valPart}`)\n  }\n  return lines\n}\n\nexport const renderContextTreeLines = (\n  ctx: Record<string, unknown>,\n  options: Options,\n  useColors: boolean\n): string[] => {\n  const depth = options.config?.contextDepth ?? 1\n  if (depth < 1) {\n    return []\n  }\n\n  const entries = collectContextEntries(ctx, '', depth)\n  return formatEntriesToTreeLines(entries, useColors)\n}\n\nexport const buildContextTreeLines = (\n  level: LogLevel,\n  data: Record<string, unknown>,\n  options: Options\n): string[] => {\n  if (options.config?.showContextTree === false) {\n    return []\n  }\n\n  const useColors = shouldUseColors(options)\n  const depth = options.config?.contextDepth ?? 1\n\n  const entries: [string, string][] = []\n\n  const ctx = data.context\n  if (\n    ctx &&\n    typeof ctx === 'object' &&\n    !Array.isArray(ctx) &&\n    Object.keys(ctx as object).length > 0 &&\n    depth >= 1\n  ) {\n    entries.push(\n      ...collectContextEntries(ctx as Record<string, unknown>, '', depth)\n    )\n  }\n\n  if (level === 'ERROR' && 'error' in data && data.error !== undefined) {\n    const msg = parseError(data.error)\n    if (msg) {\n      entries.push(['error', msg])\n    }\n  }\n\n  return formatEntriesToTreeLines(entries, useColors)\n}\n\nconst getContextString = (value: unknown): string => {\n  if (typeof value === 'object' && value !== null) {\n    return JSON.stringify(value)\n  }\n\n  return ''\n}\n\nexport const formatLogOutput = ({\n  level,\n  request,\n  data,\n  store,\n  options\n}: {\n  level: LogLevel\n  request: RequestInfo\n  data: Record<string, unknown>\n  store: StoreData\n  options: Options\n}): FormattedLogOutput => {\n  const config = options.config\n  const useColors = shouldUseColors(options)\n  const format = config?.customLogFormat ?? DEFAULT_LOG_FORMAT\n\n  const now = new Date()\n  const epoch = String(now.getTime())\n  const rawTimestamp = formatTimestamp(now, config?.timestamp?.translateTime)\n  const timestamp = getColoredTimestamp(rawTimestamp, useColors)\n\n  const message = typeof data.message === 'string' ? data.message : ''\n  const durationMs =\n    store.beforeTime === BigInt(0)\n      ? 0\n      : Number(process.hrtime.bigint() - store.beforeTime) / 1_000_000\n\n  const pathname = new URL(request.url).pathname\n  const statusValue = data.status\n  const statusCode =\n    statusValue === null || statusValue === undefined\n      ? 200\n      : getStatusCode(statusValue)\n  const status = String(statusCode)\n  const ip = config?.ip === true ? getIp(request) : ''\n\n  const showTree = config?.showContextTree !== false\n  const ctxString =\n    showTree &&\n    data.context &&\n    typeof data.context === 'object' &&\n    !Array.isArray(data.context) &&\n    Object.keys(data.context as object).length > 0\n      ? ''\n      : getContextString(data.context)\n\n  const coloredLevel = getColoredLevel(level, useColors)\n  const methodPadded = request.method.toUpperCase().padEnd(METHOD_PAD)\n  const coloredMethod = getColoredMethod(methodPadded, useColors)\n  const coloredPathname = getColoredPathname(pathname, useColors)\n  const coloredStatus = getColoredStatus(status, useColors)\n  const { text: coloredDuration, isVerySlow } = colorDurationText(\n    durationMs,\n    useColors,\n    options\n  )\n  const speedToken = getSpeedToken(isVerySlow, useColors)\n  const icon = getLevelIcon(level, useColors)\n  const statusText = getStatusText(statusCode)\n  const serviceToken = getServiceToken(options, useColors)\n\n  const main = format\n    .replaceAll('{now}', timestamp)\n    .replaceAll('{epoch}', epoch)\n    .replaceAll('{level}', coloredLevel)\n    .replaceAll('{icon}', icon)\n    .replaceAll('{duration}', coloredDuration)\n    .replaceAll('{method}', coloredMethod)\n    .replaceAll('{pathname}', coloredPathname)\n    .replaceAll('{path}', coloredPathname)\n    .replaceAll('{status}', coloredStatus)\n    .replaceAll('{statusText}', statusText)\n    .replaceAll('{message}', message)\n    .replaceAll('{ip}', ip)\n    .replaceAll('{context}', ctxString)\n    .replaceAll('{service}', serviceToken)\n    .replaceAll('{speed}', speedToken)\n\n  const contextLines = buildContextTreeLines(level, data, options)\n\n  return { main, contextLines }\n}\n\n/** @deprecated Prefer {@link formatLogOutput} for multi-line context trees. Returns the main line only. */\nexport const formatLine = (input: {\n  level: LogLevel\n  request: RequestInfo\n  data: Record<string, unknown>\n  store: StoreData\n  options: Options\n}): string =>\n  formatLogOutput({\n    ...input,\n    options: {\n      ...input.options,\n      config: {\n        ...(input.options.config ?? {}),\n        showContextTree: false\n      }\n    }\n  }).main\n\nexport const logWithPino = (\n  logger: Pino,\n  level: LogLevel,\n  data: Record<string, unknown>\n): void => {\n  if (level === 'ERROR') {\n    logger.error(data)\n    return\n  }\n  if (level === 'WARNING') {\n    logger.warn(data)\n    return\n  }\n  if (level === 'DEBUG') {\n    logger.debug(data)\n    return\n  }\n  logger.info(data)\n}\n",
    "import { StatusMap } from 'elysia'\n\nconst DIGITS_ONLY = /^\\d+$/\nconst DELIMITERS = /[_-]+/g\nconst CAMEL_BOUNDARY_1 = /([a-z0-9])([A-Z])/g\nconst CAMEL_BOUNDARY_2 = /([A-Z])([A-Z][a-z])/g\nconst APOSTROPHES = /['’]/g\nconst NON_ALPHANUMERIC = /[^a-z0-9\\s]+/g\nconst WHITESPACE = /\\s+/g\n\nconst normalizeStatusName = (value: string): string => {\n  // Handles common variants:\n  // - case differences: \"not found\" vs \"Not Found\"\n  // - spacing/punctuation: \"Not-Found\", \"not_found\"\n  // - camelCase/PascalCase: \"InternalServerError\"\n  const trimmed = value.trim()\n  if (!trimmed) {\n    return ''\n  }\n\n  return trimmed\n    .replace(DELIMITERS, ' ')\n    .replace(CAMEL_BOUNDARY_1, '$1 $2')\n    .replace(CAMEL_BOUNDARY_2, '$1 $2')\n    .replace(APOSTROPHES, '')\n    .toLowerCase()\n    .replace(NON_ALPHANUMERIC, ' ')\n    .replace(WHITESPACE, ' ')\n    .trim()\n}\n\nconst STATUS_BY_NORMALIZED_NAME = (() => {\n  const map = new Map<string, number>()\n\n  for (const [name, code] of Object.entries(StatusMap)) {\n    map.set(normalizeStatusName(name), code)\n  }\n\n  return map\n})()\n\nexport const getStatusCode = (value: unknown): number => {\n  if (typeof value === 'number' && Number.isFinite(value)) {\n    return value\n  }\n\n  if (typeof value === 'string') {\n    const trimmed = value.trim()\n    if (DIGITS_ONLY.test(trimmed)) {\n      return Number(trimmed)\n    }\n\n    const known = STATUS_BY_NORMALIZED_NAME.get(normalizeStatusName(trimmed))\n    return known ?? 500\n  }\n\n  return 500\n}\n",
    "export const parseError = (error: unknown): string => {\n  let message = 'An error occurred'\n\n  if (error instanceof Error) {\n    message = error.message\n  } else if (error && typeof error === 'object' && 'message' in error) {\n    message = error.message as string\n  } else {\n    message = String(error)\n  }\n\n  return message\n}\n",
    "import type { LogLevel, Options, RequestInfo, StoreData } from '../interfaces'\nimport { logToTransports } from '../output'\nimport { logToFile } from '../output/file'\nimport { parseError } from '../utils/error'\nimport { redact, redactRequest } from '../utils/redact'\nimport { formatLogOutput } from './create-logger'\n\nconst isErrorWithStatus = (\n  value: unknown\n): value is { status: number; message?: string } =>\n  typeof value === 'object' &&\n  value !== null &&\n  'status' in value &&\n  typeof (value as { status?: unknown }).status === 'number'\n\nexport const handleHttpError = (\n  request: RequestInfo,\n  error: unknown,\n  store: StoreData,\n  options: Options\n): void => {\n  const config = options.config\n\n  const logFilter = config?.logFilter\n  if (\n    logFilter?.level &&\n    logFilter.level.length > 0 &&\n    !logFilter.level.includes('ERROR')\n  ) {\n    return\n  }\n\n  const useTransportsOnly = config?.useTransportsOnly === true\n  const disableInternalLogger = config?.disableInternalLogger === true\n  const disableFileLogging = config?.disableFileLogging === true\n\n  const status = isErrorWithStatus(error) ? error.status : 500\n  const message = parseError(error)\n\n  const level: LogLevel = 'ERROR'\n  const data: Record<string, unknown> = { status, message, error }\n  const logData = config?.autoRedact === true ? redact(data) : data\n  const logRequest =\n    config?.autoRedact === true ? redactRequest(request) : request\n\n  logToTransports({ level, request: logRequest, data: logData, store, options })\n\n  if (!(useTransportsOnly || disableFileLogging)) {\n    const filePath = config?.logFilePath\n    if (filePath) {\n      logToFile({\n        filePath,\n        level,\n        request: logRequest,\n        data: logData,\n        store,\n        options\n      }).catch(() => {\n        // Ignore errors\n      })\n    }\n  }\n\n  if (useTransportsOnly || disableInternalLogger) {\n    return\n  }\n\n  const { main, contextLines } = formatLogOutput({\n    level,\n    request: logRequest,\n    data: logData,\n    store,\n    options\n  })\n\n  const formattedMessage =\n    contextLines.length > 0 ? `${main}\\n${contextLines.join('\\n')}` : main\n  console.error(formattedMessage)\n}\n"
  ],
  "mappings": "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",
  "debugId": "5ECC4ED287A03C9864756E2164756E21",
  "names": []
}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "logixlysia",
3
- "version": "6.3.0",
3
+ "version": "6.3.2",
4
4
  "description": "🦊 Logixlysia is a logger for Elysia",
5
5
  "type": "module",
6
6
  "main": "dist/index.js",