logixlysia 6.0.0 → 6.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md ADDED
@@ -0,0 +1,49 @@
1
+ <div align="center">
2
+ <h1><code>🦊</code> Logixlysia</h1>
3
+ <strong>Logixlysia is a logging library for ElysiaJS</strong>
4
+ <img src="../../apps/docs/app/opengraph-image.png" alt="Logixlysia" width="100%" height="auto" />
5
+ </div>
6
+
7
+ ## `📩` Installation
8
+
9
+ ```bash
10
+ bun add logixlysia
11
+ ```
12
+
13
+ ## `📝` Usage
14
+
15
+ ```ts
16
+ import { Elysia } from 'elysia'
17
+ import logixlysia from 'logixlysia' // or import { logixlysia } from 'logixlysia'
18
+
19
+ const app = new Elysia({
20
+ name: "Elysia with Logixlysia"
21
+ })
22
+ .use(
23
+ logixlysia({
24
+ config: {
25
+ showStartupMessage: true,
26
+ startupMessageFormat: 'simple',
27
+ timestamp: {
28
+ translateTime: 'yyyy-mm-dd HH:MM:ss.SSS'
29
+ },
30
+ logFilePath: './logs/example.log',
31
+ ip: true,
32
+ customLogFormat:
33
+ '🦊 {now} {level} {duration} {method} {pathname} {status} {message} {ip}'
34
+ }
35
+ }))
36
+ .get('/', () => {
37
+ return { message: 'Welcome to Basic Elysia with Logixlysia' }
38
+ })
39
+
40
+ app.listen(3000)
41
+ ```
42
+
43
+ ## `📚` Documentation
44
+
45
+ Check out the [website](https://logixlysia.vercel.app) for more detailed documentation and examples.
46
+
47
+ ## `📄` License
48
+
49
+ Licensed under the [MIT License](LICENSE).
package/dist/index.d.ts CHANGED
@@ -1,20 +1,21 @@
1
1
  import { Elysia, SingletonBase } from "elysia";
2
2
  import { Logger as PinoLogger, LoggerOptions as PinoLoggerOptions } from "pino";
3
3
  type Pino = PinoLogger<never, boolean>;
4
+ type RequestInfo = Request;
4
5
  type LogLevel = "DEBUG" | "INFO" | "WARNING" | "ERROR";
5
- type StoreData = {
6
+ interface StoreData {
6
7
  beforeTime: bigint;
7
- };
8
- type LogixlysiaStore = {
8
+ }
9
+ interface LogixlysiaStore {
9
10
  logger: Logger;
10
11
  pino: Pino;
11
12
  beforeTime?: bigint;
12
13
  [key: string]: unknown;
13
- };
14
- type Transport = {
14
+ }
15
+ interface Transport {
15
16
  log: (level: LogLevel, message: string, meta?: Record<string, unknown>) => void | Promise<void>;
16
- };
17
- type LogRotationConfig = {
17
+ }
18
+ interface LogRotationConfig {
18
19
  /**
19
20
  * Max log file size before rotation, e.g. '10m', '5k', or a byte count.
20
21
  */
@@ -29,8 +30,15 @@ type LogRotationConfig = {
29
30
  interval?: string;
30
31
  compress?: boolean;
31
32
  compression?: "gzip";
32
- };
33
- type Options = {
33
+ }
34
+ interface LogFilter {
35
+ /**
36
+ * Array of log levels to allow. If specified, only logs with these levels will be processed.
37
+ * If not specified, all log levels will be allowed.
38
+ */
39
+ level?: LogLevel[];
40
+ }
41
+ interface Options {
34
42
  config?: {
35
43
  showStartupMessage?: boolean;
36
44
  startupMessageFormat?: "simple" | "banner";
@@ -40,6 +48,7 @@ type Options = {
40
48
  translateTime?: string;
41
49
  };
42
50
  customLogFormat?: string;
51
+ logFilter?: LogFilter;
43
52
  transports?: Transport[];
44
53
  useTransportsOnly?: boolean;
45
54
  disableInternalLogger?: boolean;
@@ -50,8 +59,8 @@ type Options = {
50
59
  prettyPrint?: boolean;
51
60
  }) | undefined;
52
61
  };
53
- };
54
- type Logger = {
62
+ }
63
+ interface Logger {
55
64
  pino: Pino;
56
65
  log: (level: LogLevel, request: RequestInfo, data: Record<string, unknown>, store: StoreData) => void;
57
66
  handleHttpError: (request: RequestInfo, error: unknown, store: StoreData) => void;
@@ -59,11 +68,11 @@ type Logger = {
59
68
  info: (request: RequestInfo, message: string, context?: Record<string, unknown>) => void;
60
69
  warn: (request: RequestInfo, message: string, context?: Record<string, unknown>) => void;
61
70
  error: (request: RequestInfo, message: string, context?: Record<string, unknown>) => void;
62
- };
63
- type LogixlysiaContext = {
71
+ }
72
+ interface LogixlysiaContext {
64
73
  request: Request;
65
74
  store: LogixlysiaStore;
66
- };
75
+ }
67
76
  type Logixlysia = Elysia<"Logixlysia", SingletonBase & {
68
77
  store: LogixlysiaStore;
69
78
  }>;
package/dist/index.js CHANGED
@@ -1,6 +1,6 @@
1
- import{Elysia as Hn}from"elysia";import v from"elysia/package.json";var u=(n,o)=>{if(n.length>=o)return n.slice(0,o);let r=Math.floor((o-n.length)/2),b=o-n.length-r;return`${" ".repeat(r)}${n}${" ".repeat(b)}`},B=(n)=>{let o=`Elysia v${v.version}`,b=Math.max(n.length,o.length)+4,c=`┌${"─".repeat(b)}┐`,i=`└${"─".repeat(b)}┘`,f=`│${" ".repeat(b)}│`,m=`│${u(o,b)}│`,L=`│ ${n}${" ".repeat(Math.max(0,b-n.length-4))} │`;return[c,f,m,f,L,f,i].join(`
2
- `)};var V=(n,o)=>{if(!(o.config?.showStartupMessage??!0))return;let{port:b,hostname:c,protocol:i}=n;if(b===void 0||!c||!i)return;let m=`\uD83E\uDD8A Elysia is running at ${`${i}://${c}:${b}`}`;if((o.config?.startupMessageFormat??"banner")==="simple"){console.log(m);return}console.log(B(m))};import p from"pino";var j=(...n)=>{let o=typeof n[0]==="string"?{level:n[0],request:n[1],data:n[2],store:n[3],options:n[4]}:n[0],{level:r,request:b,data:c,store:i,options:f}=o,m=f.config?.transports??[];if(m.length===0)return;let L=typeof c.message==="string"?c.message:"",O={request:{method:b.method,url:b.url},...c,beforeTime:i.beforeTime};for(let $ of m)try{let R=$.log(r,L,O);if(R&&typeof R.catch==="function")R.catch(()=>{})}catch{}};import{appendFile as In}from"node:fs/promises";import{dirname as On}from"node:path";import{promises as d}from"node:fs";var Y=async(n)=>{await d.mkdir(n,{recursive:!0})};import{promises as T}from"node:fs";import{promisify as cn}from"node:util";import{gzip as bn}from"node:zlib";import{promises as Z}from"node:fs";import{basename as l,dirname as a}from"node:path";var e=/^(\d+(?:\.\d+)?)(k|kb|m|mb|g|gb)$/i,nn=/^(\d+)(h|d|w)$/i,on=/\.(\d{4}-\d{2}-\d{2}-\d{2}-\d{2}-\d{2})(?:\.gz)?$/,_=(n)=>{if(typeof n==="number")return n;let o=n.trim(),r=Number(o);if(Number.isFinite(r))return r;let b=o.match(e);if(!b)throw Error(`Invalid size format: ${n}`);let c=Number(b[1]),i=b[2].toLowerCase(),f=1024;if(i.startsWith("m"))f=1048576;else if(i.startsWith("g"))f=1073741824;return Math.floor(c*f)},rn=(n)=>{let o=n.trim().match(nn);if(!o)throw Error(`Invalid interval format: ${n}`);let r=Number(o[1]),b=o[2].toLowerCase(),c=3600000;if(b==="d")c=86400000;else if(b==="w")c=604800000;return r*c},H=(n)=>{if(typeof n==="number")return{type:"count",value:n};return{type:"time",value:rn(n)}},A=async(n,o)=>{try{return(await Z.stat(n)).size>o}catch{return!1}},U=async(n)=>{let o=a(n),r=l(n),b;try{b=await Z.readdir(o)}catch{return[]}return b.filter((c)=>c.startsWith(`${r}.`)&&on.test(c)).map((c)=>`${o}/${c}`)};var fn=cn(bn),D=(n)=>String(n).padStart(2,"0"),mn=(n,o)=>{let r=o.getFullYear(),b=D(o.getMonth()+1),c=D(o.getDate()),i=D(o.getHours()),f=D(o.getMinutes()),m=D(o.getSeconds());return`${n}.${r}-${b}-${c}-${i}-${f}-${m}`},Rn=async(n)=>{try{if((await T.stat(n)).size===0)return""}catch{return""}let o=mn(n,new Date);return await T.rename(n,o),o},$n=async(n)=>{let o=await T.readFile(n),r=await fn(o);await T.writeFile(`${n}.gz`,r),await T.rm(n,{force:!0})},S=async(n,o)=>{if(o.maxSize===void 0)return!1;let r=_(o.maxSize);return await A(n,r)},Ln=async(n,o)=>{let r=await U(n);if(r.length<=o)return;let b=await Promise.all(r.map(async(i)=>({path:i,stat:await T.stat(i)})));b.sort((i,f)=>f.stat.mtimeMs-i.stat.mtimeMs);let c=b.slice(o);await Promise.all(c.map(({path:i})=>T.rm(i,{force:!0})))},wn=async(n,o)=>{let r=await U(n);if(r.length===0)return;let b=Date.now(),i=(await Promise.all(r.map(async(f)=>({path:f,stat:await T.stat(f)})))).filter(({stat:f})=>b-f.mtimeMs>o);await Promise.all(i.map(({path:f})=>T.rm(f,{force:!0})))},k=async(n,o)=>{let r=await Rn(n);if(!r)return;if(o.compress===!0){if((o.compression??"gzip")==="gzip")await $n(r)}if(o.maxFiles!==void 0){let c=H(o.maxFiles);if(c.type==="count")await Ln(n,c.value);else await wn(n,c.value)}};var N=async(...n)=>{let o=typeof n[0]==="string"?(()=>{let[G,F,K,W,J,Q]=n;return{filePath:G,level:F,request:K,data:W,store:J,options:Q}})():n[0],{filePath:r,level:b,request:c,data:i,store:f,options:m}=o,L=m.config,O=L?.useTransportsOnly===!0,$=L?.disableFileLogging===!0;if(O||$)return;let R=typeof i.message==="string"?i.message:"",w=f.beforeTime===BigInt(0)?0:Number(process.hrtime.bigint()-f.beforeTime)/1e6,g=`${b} ${w.toFixed(2)}ms ${c.method} ${new URL(c.url).pathname} ${R}
3
- `;await Y(On(r)),await In(r,g,{encoding:"utf-8"});let y=L?.logRotation;if(!y)return;if(await S(r,y))await k(r,y)};import I from"chalk";import{StatusMap as yn}from"elysia";var Tn=/^\d+$/,gn=/[_-]+/g,Fn=/([a-z0-9])([A-Z])/g,En=/([A-Z])([A-Z][a-z])/g,Kn=/['’]/g,Gn=/[^a-z0-9\s]+/g,Dn=/\s+/g,M=(n)=>{let o=n.trim();if(!o)return"";return o.replace(gn," ").replace(Fn,"$1 $2").replace(En,"$1 $2").replace(Kn,"").toLowerCase().replace(Gn," ").replace(Dn," ").trim()},Xn=(()=>{let n=new Map;for(let[o,r]of Object.entries(yn))n.set(M(o),r);return n})(),s=(n)=>{if(typeof n==="number"&&Number.isFinite(n))return n;if(typeof n==="string"){let o=n.trim();if(Tn.test(o))return Number(o);return Xn.get(M(o))??500}return 500};var X=(n)=>String(n).padStart(2,"0"),jn=(n)=>String(n).padStart(3,"0"),Nn=(n)=>{let r=n.config?.useColors??!0,b=typeof process<"u"&&process.stdout?.isTTY===!0;return r&&b},Wn=(n,o)=>{if(!o)return n.toISOString();let r=String(n.getFullYear()),b=X(n.getMonth()+1),c=X(n.getDate()),i=X(n.getHours()),f=X(n.getMinutes()),m=X(n.getSeconds()),L=jn(n.getMilliseconds());return o.replaceAll("yyyy",r).replaceAll("mm",b).replaceAll("dd",c).replaceAll("HH",i).replaceAll("MM",f).replaceAll("ss",m).replaceAll("SSS",L)},Jn=(n)=>{let o=n.headers.get("x-forwarded-for");if(o)return o.split(",")[0]?.trim()??"";return n.headers.get("x-real-ip")??""},Qn=(n,o)=>{if(!o)return n;if(n==="ERROR")return I.bgRed.black(n);if(n==="WARNING")return I.bgYellow.black(n);if(n==="DEBUG")return I.bgBlue.black(n);return I.bgGreen.black(n)},Zn=(n,o)=>{if(!o)return n;let r=n.toUpperCase();if(r==="GET")return I.green.bold(r);if(r==="POST")return I.blue.bold(r);if(r==="PUT")return I.yellow.bold(r);if(r==="PATCH")return I.yellowBright.bold(r);if(r==="DELETE")return I.red.bold(r);if(r==="OPTIONS")return I.cyan.bold(r);if(r==="HEAD")return I.greenBright.bold(r);if(r==="TRACE")return I.magenta.bold(r);if(r==="CONNECT")return I.cyanBright.bold(r);return I.white.bold(r)},Un=(n,o)=>{if(!o)return n;let r=Number.parseInt(n,10);if(!Number.isFinite(r))return n;if(r>=500)return I.red(n);if(r>=400)return I.yellow(n);if(r>=300)return I.cyan(n);if(r>=200)return I.green(n);return I.gray(n)},zn=(n,o)=>{if(!o)return n;return I.gray(n)},Bn=(n,o)=>{if(!o)return n;return I.bgHex("#FFA500").black(n)},Vn=(n,o)=>{if(!o)return n;return I.whiteBright(n)},Yn=(n)=>{if(typeof n==="object"&&n!==null)return JSON.stringify(n);return""},P=({level:n,request:o,data:r,store:b,options:c})=>{let i=c.config,f=Nn(c),m=i?.customLogFormat??"\uD83E\uDD8A {now} {level} {duration} {method} {pathname} {status} {message} {ip} {context}",L=new Date,O=String(L.getTime()),$=Wn(L,i?.timestamp?.translateTime),R=Bn($,f),w=typeof r.message==="string"?r.message:"",g=b.beforeTime===BigInt(0)?0:Number(process.hrtime.bigint()-b.beforeTime)/1e6,y=new URL(o.url).pathname,E=r.status,G=E===null||E===void 0?200:s(E),F=String(G),K=i?.ip===!0?Jn(o):"",W=Yn(r.context),J=Qn(n,f),Q=Zn(o.method,f),z=Vn(y,f),C=Un(F,f),h=zn(`${g.toFixed(2)}ms`,f);return m.replaceAll("{now}",R).replaceAll("{epoch}",O).replaceAll("{level}",J).replaceAll("{duration}",h).replaceAll("{method}",Q).replaceAll("{pathname}",z).replaceAll("{path}",z).replaceAll("{status}",C).replaceAll("{message}",w).replaceAll("{ip}",K).replaceAll("{context}",W)};var x=(n)=>{let o="An error occurred";if(n instanceof Error)o=n.message;else if(n&&typeof n==="object"&&"message"in n)o=n.message;else o=String(n);return console.error(`Parsing error: ${o}`),o};var _n=(n)=>typeof n==="object"&&n!==null&&("status"in n)&&typeof n.status==="number",t=(n,o,r,b)=>{let c=b.config,i=c?.useTransportsOnly===!0,f=c?.disableInternalLogger===!0,m=c?.disableFileLogging===!0,L=_n(o)?o.status:500,O=x(o),$="ERROR",R={status:L,message:O,error:o};if(j({level:"ERROR",request:n,data:R,store:r,options:b}),!(i||m)){let w=c?.logFilePath;if(w)N({filePath:w,level:"ERROR",request:n,data:R,store:r,options:b}).catch(()=>{})}if(i||f)return;console.error(`ERROR ${n.method} ${new URL(n.url).pathname} ${O}`)};var q=(n={})=>{let o=n.config,r=o?.pino,{prettyPrint:b,...c}=r??{},f=b===!0&&c.transport===void 0?p.transport({target:"pino-pretty",options:{colorize:process.stdout?.isTTY===!0,translateTime:o?.timestamp?.translateTime,messageKey:c.messageKey,errorKey:c.errorKey}}):c.transport,m=p({...c,level:c.level??"info",messageKey:c.messageKey,errorKey:c.errorKey,transport:f}),L=($,R,w,g)=>{j({level:$,request:R,data:w,store:g,options:n});let y=o?.useTransportsOnly===!0,E=o?.disableInternalLogger===!0,G=o?.disableFileLogging===!0;if(!(y||G)){let K=o?.logFilePath;if(K)N({filePath:K,level:$,request:R,data:w,store:g,options:n}).catch(()=>{})}if(y||E)return;let F=P({level:$,request:R,data:w,store:g,options:n});switch($){case"DEBUG":{console.debug(F);break}case"INFO":{console.info(F);break}case"WARNING":{console.warn(F);break}case"ERROR":{console.error(F);break}default:{console.log(F);break}}},O=($,R,w,g)=>{let y={beforeTime:process.hrtime.bigint()};L($,R,{message:w,context:g},y)};return{pino:m,log:L,handleHttpError:($,R,w)=>{t($,R,w,n)},debug:($,R,w)=>{O("DEBUG",$,R,w)},info:($,R,w)=>{O("INFO",$,R,w)},warn:($,R,w)=>{O("WARNING",$,R,w)},error:($,R,w)=>{O("ERROR",$,R,w)}}};var An=(n={})=>{let o=new WeakSet,r=q(n),b={...r,debug:(i,f,m)=>{o.add(i),r.debug(i,f,m)},info:(i,f,m)=>{o.add(i),r.info(i,f,m)},warn:(i,f,m)=>{o.add(i),r.warn(i,f,m)},error:(i,f,m)=>{o.add(i),r.error(i,f,m)}};return new Hn({name:"Logixlysia",detail:{description:"Logixlysia is a plugin for Elysia that provides a logger and pino logger.",tags:["logging","pino"]}}).state("logger",b).state("pino",b.pino).state("beforeTime",BigInt(0)).onStart(({server:i})=>{if(i)V(i,n)}).onRequest(({store:i})=>{i.beforeTime=process.hrtime.bigint()}).onAfterHandle(({request:i,set:f,store:m})=>{if(o.has(i))return;let L=typeof f.status==="number"?f.status:200,O="INFO";if(L>=500)O="ERROR";else if(L>=400)O="WARNING";b.log(O,i,{status:L},m)}).onError(({request:i,error:f,store:m})=>{b.handleHttpError(i,f,m)}).as("scoped")},jo=An;export{An as logixlysia,jo as default};
1
+ import{Elysia as m0}from"elysia";import o from"elysia/package.json";var s=(n,b)=>{if(n.length>=b)return n.slice(0,b);let $=Math.floor((b-n.length)/2),R=b-n.length-$;return`${" ".repeat($)}${n}${" ".repeat(R)}`},M=(n)=>{let b=`Elysia v${o.version}`,R=Math.max(n.length,b.length)+4,I=`┌${"─".repeat(R)}┐`,w=`└${"─".repeat(R)}┘`,E=`│${" ".repeat(R)}│`,X=`│${s(b,R)}│`,J=`│ ${n}${" ".repeat(Math.max(0,R-n.length-4))} │`;return[I,E,X,E,J,E,w].join(`
2
+ `)};var S=(n,b)=>{if(!(b.config?.showStartupMessage??!0))return;let{port:R,hostname:I,protocol:w}=n;if(R===void 0||!I||!w)return;let X=`\uD83E\uDD8A Elysia is running at ${`${w}://${I}:${R}`}`;if((b.config?.startupMessageFormat??"banner")==="simple"){console.log(X);return}console.log(M(X))};import g from"pino";var Y=(...n)=>{let b=typeof n[0]==="string"?{level:n[0],request:n[1],data:n[2],store:n[3],options:n[4]}:n[0],{level:$,request:R,data:I,store:w,options:E}=b,X=E.config?.transports??[];if(X.length===0)return;let J=typeof I.message==="string"?I.message:"",K={request:{method:R.method,url:R.url},...I,beforeTime:w.beforeTime};for(let L of X)try{let f=L.log($,J,K);if(f&&typeof f.catch==="function")f.catch(()=>{})}catch{}};import{appendFile as D0}from"node:fs/promises";import{dirname as J0}from"node:path";import{promises as l}from"node:fs";var m=async(n)=>{await l.mkdir(n,{recursive:!0})};import{promises as Z}from"node:fs";import{promisify as w0}from"node:util";import{gzip as R0}from"node:zlib";import{promises as W}from"node:fs";import{basename as t,dirname as a}from"node:path";var e=/^(\d+(?:\.\d+)?)(k|kb|m|mb|g|gb)$/i,n0=/^(\d+)(h|d|w)$/i,b0=/\.(\d{4}-\d{2}-\d{2}-\d{2}-\d{2}-\d{2})(?:\.gz)?$/,k=(n)=>{if(typeof n==="number")return n;let b=n.trim(),$=Number(b);if(Number.isFinite($))return $;let R=b.match(e);if(!R)throw Error(`Invalid size format: ${n}`);let I=Number(R[1]),w=R[2].toLowerCase(),E=1024;if(w.startsWith("m"))E=1048576;else if(w.startsWith("g"))E=1073741824;return Math.floor(I*E)},$0=(n)=>{let b=n.trim().match(n0);if(!b)throw Error(`Invalid interval format: ${n}`);let $=Number(b[1]),R=b[2].toLowerCase(),I=3600000;if(R==="d")I=86400000;else if(R==="w")I=604800000;return $*I},P=(n)=>{if(typeof n==="number")return{type:"count",value:n};return{type:"time",value:$0(n)}},x=async(n,b)=>{try{return(await W.stat(n)).size>b}catch{return!1}},A=async(n)=>{let b=a(n),$=t(n),R;try{R=await W.readdir(b)}catch{return[]}return R.filter((I)=>I.startsWith(`${$}.`)&&b0.test(I)).map((I)=>`${b}/${I}`)};var I0=w0(R0),O=(n)=>String(n).padStart(2,"0"),E0=(n,b)=>{let $=b.getFullYear(),R=O(b.getMonth()+1),I=O(b.getDate()),w=O(b.getHours()),E=O(b.getMinutes()),X=O(b.getSeconds());return`${n}.${$}-${R}-${I}-${w}-${E}-${X}`},X0=async(n)=>{try{if((await Z.stat(n)).size===0)return""}catch{return""}let b=E0(n,new Date);return await Z.rename(n,b),b},f0=async(n)=>{let b=await Z.readFile(n),$=await I0(b);await Z.writeFile(`${n}.gz`,$),await Z.rm(n,{force:!0})},C=async(n,b)=>{if(b.maxSize===void 0)return!1;let $=k(b.maxSize);return await x(n,$)},G0=async(n,b)=>{let $=await A(n);if($.length<=b)return;let R=await Promise.all($.map(async(w)=>({path:w,stat:await Z.stat(w)})));R.sort((w,E)=>E.stat.mtimeMs-w.stat.mtimeMs);let I=R.slice(b);await Promise.all(I.map(({path:w})=>Z.rm(w,{force:!0})))},j0=async(n,b)=>{let $=await A(n);if($.length===0)return;let R=Date.now(),w=(await Promise.all($.map(async(E)=>({path:E,stat:await Z.stat(E)})))).filter(({stat:E})=>R-E.mtimeMs>b);await Promise.all(w.map(({path:E})=>Z.rm(E,{force:!0})))},q=async(n,b)=>{let $=await X0(n);if(!$)return;if(b.compress===!0){if((b.compression??"gzip")==="gzip")await f0($)}if(b.maxFiles!==void 0){let I=P(b.maxFiles);if(I.type==="count")await G0(n,I.value);else await j0(n,I.value)}};var _=async(...n)=>{let b=typeof n[0]==="string"?(()=>{let[z,B,T,V,N,c]=n;return{filePath:z,level:B,request:T,data:V,store:N,options:c}})():n[0],{filePath:$,level:R,request:I,data:w,store:E,options:X}=b,J=X.config,K=J?.useTransportsOnly===!0,L=J?.disableFileLogging===!0;if(K||L)return;let f=typeof w.message==="string"?w.message:"",G=E.beforeTime===BigInt(0)?0:Number(process.hrtime.bigint()-E.beforeTime)/1e6,j=`${R} ${G.toFixed(2)}ms ${I.method} ${new URL(I.url).pathname} ${f}
3
+ `;await m(J0($)),await D0($,j,{encoding:"utf-8"});let Q=J?.logRotation;if(!Q)return;if(await C($,Q))await q($,Q)};import D from"chalk";import{StatusMap as K0}from"elysia";var L0=/^\d+$/,Q0=/[_-]+/g,Z0=/([a-z0-9])([A-Z])/g,F0=/([A-Z])([A-Z][a-z])/g,T0=/['’]/g,V0=/[^a-z0-9\s]+/g,z0=/\s+/g,i=(n)=>{let b=n.trim();if(!b)return"";return b.replace(Q0," ").replace(Z0,"$1 $2").replace(F0,"$1 $2").replace(T0,"").toLowerCase().replace(V0," ").replace(z0," ").trim()},B0=(()=>{let n=new Map;for(let[b,$]of Object.entries(K0))n.set(i(b),$);return n})(),v=(n)=>{if(typeof n==="number"&&Number.isFinite(n))return n;if(typeof n==="string"){let b=n.trim();if(L0.test(b))return Number(b);return B0.get(i(b))??500}return 500};var U=(n)=>String(n).padStart(2,"0"),O0=(n)=>String(n).padStart(3,"0"),U0=(n)=>{let $=n.config?.useColors??!0,R=typeof process<"u"&&process.stdout?.isTTY===!0;return $&&R},Y0=(n,b)=>{if(!b)return n.toISOString();let $=String(n.getFullYear()),R=U(n.getMonth()+1),I=U(n.getDate()),w=U(n.getHours()),E=U(n.getMinutes()),X=U(n.getSeconds()),J=O0(n.getMilliseconds());return b.replaceAll("yyyy",$).replaceAll("mm",R).replaceAll("dd",I).replaceAll("HH",w).replaceAll("MM",E).replaceAll("ss",X).replaceAll("SSS",J)},_0=(n)=>{let b=n.headers.get("x-forwarded-for");if(b)return b.split(",")[0]?.trim()??"";return n.headers.get("x-real-ip")??""},y0=(n,b)=>{if(!b)return n;if(n==="ERROR")return D.bgRed.black(n);if(n==="WARNING")return D.bgYellow.black(n);if(n==="DEBUG")return D.bgBlue.black(n);return D.bgGreen.black(n)},N0=(n,b)=>{if(!b)return n;let $=n.toUpperCase();if($==="GET")return D.green.bold($);if($==="POST")return D.blue.bold($);if($==="PUT")return D.yellow.bold($);if($==="PATCH")return D.yellowBright.bold($);if($==="DELETE")return D.red.bold($);if($==="OPTIONS")return D.cyan.bold($);if($==="HEAD")return D.greenBright.bold($);if($==="TRACE")return D.magenta.bold($);if($==="CONNECT")return D.cyanBright.bold($);return D.white.bold($)},c0=(n,b)=>{if(!b)return n;let $=Number.parseInt(n,10);if(!Number.isFinite($))return n;if($>=500)return D.red(n);if($>=400)return D.yellow(n);if($>=300)return D.cyan(n);if($>=200)return D.green(n);return D.gray(n)},W0=(n,b)=>{if(!b)return n;return D.gray(n)},A0=(n,b)=>{if(!b)return n;return D.bgHex("#FFA500").black(n)},H0=(n,b)=>{if(!b)return n;return D.whiteBright(n)},M0=(n)=>{if(typeof n==="object"&&n!==null)return JSON.stringify(n);return""},y=({level:n,request:b,data:$,store:R,options:I})=>{let w=I.config,E=U0(I),X=w?.customLogFormat??"\uD83E\uDD8A {now} {level} {duration} {method} {pathname} {status} {message} {ip} {context}",J=new Date,K=String(J.getTime()),L=Y0(J,w?.timestamp?.translateTime),f=A0(L,E),G=typeof $.message==="string"?$.message:"",j=R.beforeTime===BigInt(0)?0:Number(process.hrtime.bigint()-R.beforeTime)/1e6,Q=new URL(b.url).pathname,F=$.status,z=F===null||F===void 0?200:v(F),B=String(z),T=w?.ip===!0?_0(b):"",V=M0($.context),N=y0(n,E),c=N0(b.method,E),H=H0(Q,E),u=c0(B,E),d=W0(`${j.toFixed(2)}ms`,E);return X.replaceAll("{now}",f).replaceAll("{epoch}",K).replaceAll("{level}",N).replaceAll("{duration}",d).replaceAll("{method}",c).replaceAll("{pathname}",H).replaceAll("{path}",H).replaceAll("{status}",u).replaceAll("{message}",G).replaceAll("{ip}",T).replaceAll("{context}",V)};var h=(n)=>{let b="An error occurred";if(n instanceof Error)b=n.message;else if(n&&typeof n==="object"&&"message"in n)b=n.message;else b=String(n);return b};var S0=(n)=>typeof n==="object"&&n!==null&&("status"in n)&&typeof n.status==="number",r=(n,b,$,R)=>{let I=R.config,w=I?.useTransportsOnly===!0,E=I?.disableInternalLogger===!0,X=I?.disableFileLogging===!0,J=S0(b)?b.status:500,K=h(b),L="ERROR",f={status:J,message:K,error:b};if(Y({level:"ERROR",request:n,data:f,store:$,options:R}),!(w||X)){let j=I?.logFilePath;if(j)_({filePath:j,level:"ERROR",request:n,data:f,store:$,options:R}).catch(()=>{})}if(w||E)return;let G=y({level:"ERROR",request:n,data:f,store:$,options:R});console.error(G)};var p=(n={})=>{let b=n.config,$=b?.pino,{prettyPrint:R,...I}=$??{},E=R===!0&&I.transport===void 0?g.transport({target:"pino-pretty",options:{colorize:process.stdout?.isTTY===!0,translateTime:b?.timestamp?.translateTime,messageKey:I.messageKey,errorKey:I.errorKey}}):I.transport,X=g({...I,level:I.level??"info",messageKey:I.messageKey,errorKey:I.errorKey,transport:E}),J=(f,G)=>{if(!G?.level||G.level.length===0)return!0;return G.level.includes(f)},K=(f,G,j,Q)=>{if(!J(f,b?.logFilter))return;Y({level:f,request:G,data:j,store:Q,options:n});let F=b?.useTransportsOnly===!0,z=b?.disableInternalLogger===!0,B=b?.disableFileLogging===!0;if(!(F||B)){let V=b?.logFilePath;if(V)_({filePath:V,level:f,request:G,data:j,store:Q,options:n}).catch(()=>{})}if(F||z)return;let T=y({level:f,request:G,data:j,store:Q,options:n});switch(f){case"DEBUG":{console.debug(T);break}case"INFO":{console.info(T);break}case"WARNING":{console.warn(T);break}case"ERROR":{console.error(T);break}default:{console.log(T);break}}},L=(f,G,j,Q)=>{let F={beforeTime:process.hrtime.bigint()};K(f,G,{message:j,context:Q},F)};return{pino:X,log:K,handleHttpError:(f,G,j)=>{r(f,G,j,n)},debug:(f,G,j)=>{L("DEBUG",f,G,j)},info:(f,G,j)=>{L("INFO",f,G,j)},warn:(f,G,j)=>{L("WARNING",f,G,j)},error:(f,G,j)=>{L("ERROR",f,G,j)}}};var k0=(n={})=>{let b=new WeakSet,$=p(n),R={...$,debug:(w,E,X)=>{b.add(w),$.debug(w,E,X)},info:(w,E,X)=>{b.add(w),$.info(w,E,X)},warn:(w,E,X)=>{b.add(w),$.warn(w,E,X)},error:(w,E,X)=>{b.add(w),$.error(w,E,X)}};return new m0({name:"Logixlysia",detail:{description:"Logixlysia is a plugin for Elysia that provides a logger and pino logger.",tags:["logging","pino"]}}).state("logger",R).state("pino",R.pino).state("beforeTime",BigInt(0)).onStart(({server:w})=>{if(w)S(w,n)}).onRequest(({store:w})=>{w.beforeTime=process.hrtime.bigint()}).onAfterHandle(({request:w,set:E,store:X})=>{if(b.has(w))return;let J=typeof E.status==="number"?E.status:200,K="INFO";if(J>=500)K="ERROR";else if(J>=400)K="WARNING";R.log(K,w,{status:J},X)}).onError(({request:w,error:E,store:X})=>{R.handleHttpError(w,E,X)}).as("scoped")},Un=k0;export{k0 as logixlysia,Un as default};
4
4
 
5
- //# debugId=7FAFEF1E34BEF02764756E2164756E21
6
- //# 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, type SingletonBase } from 'elysia'\nimport { startServer } from './extensions'\nimport type { LogixlysiaStore, Options } from './interfaces'\nimport { createLogger } from './logger'\n\nexport type Logixlysia = Elysia<\n  'Logixlysia',\n  SingletonBase & { store: LogixlysiaStore }\n>\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 }) => {\n        if (server) {\n          startServer(server, 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 unknown 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 elysiaPkg from 'elysia/package.json'\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\nexport const renderBanner = (message: string): string => {\n  const versionLine = `Elysia v${elysiaPkg.version}`\n  const contentWidth = Math.max(message.length, versionLine.length)\n  const innerWidth = contentWidth + 4 // 2 spaces padding on both sides\n\n  const top = `┌${'─'.repeat(innerWidth)}┐`\n  const bot = `└${'─'.repeat(innerWidth)}┘`\n  const empty = `│${' '.repeat(innerWidth)}│`\n\n  const versionRow = `│${centerText(versionLine, innerWidth)}│`\n  const messageRow = `│  ${message}${' '.repeat(Math.max(0, innerWidth - message.length - 4))}  │`\n\n  return [top, empty, versionRow, empty, messageRow, empty, bot].join('\\n')\n}\n",
    "import type { Options } from '../interfaces'\nimport { 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\n  const format = options.config?.startupMessageFormat ?? 'banner'\n  if (format === 'simple') {\n    console.log(message)\n    return\n  }\n\n  console.log(renderBanner(message))\n}\n",
    "import pino from 'pino'\nimport type {\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 { formatLine } from './create-logger'\nimport { handleHttpError } from './handle-http-error'\n\nexport const createLogger = (options: Options = {}): Logger => {\n  const config = options.config\n\n  const pinoConfig = config?.pino\n  const { prettyPrint, ...pinoOptions } = pinoConfig ?? {}\n\n  const shouldPrettyPrint =\n    prettyPrint === true && pinoOptions.transport === undefined\n\n  const transport = shouldPrettyPrint\n    ? pino.transport({\n        target: 'pino-pretty',\n        options: {\n          colorize: process.stdout?.isTTY === true,\n          translateTime: config?.timestamp?.translateTime,\n          messageKey: pinoOptions.messageKey,\n          errorKey: pinoOptions.errorKey\n        }\n      })\n    : pinoOptions.transport\n\n  const pinoLogger: Pino = pino({\n    ...pinoOptions,\n    level: pinoOptions.level ?? 'info',\n    messageKey: pinoOptions.messageKey,\n    errorKey: pinoOptions.errorKey,\n    transport\n  })\n\n  const log = (\n    level: LogLevel,\n    request: RequestInfo,\n    data: Record<string, unknown>,\n    store: StoreData\n  ): void => {\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 message = formatLine({ level, request, data, store, options })\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\ntype LogToTransportsInput = {\n  level: LogLevel\n  request: RequestInfo\n  data: Record<string, unknown>\n  store: StoreData\n  options: Options\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\ntype LogToFileInput = {\n  filePath: string\n  level: LogLevel\n  request: RequestInfo\n  data: Record<string, unknown>\n  store: StoreData\n  options: Options\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 chalk from 'chalk'\nimport { getStatusCode } from '../helpers/status'\nimport type {\n  LogLevel,\n  Options,\n  Pino,\n  RequestInfo,\n  StoreData\n} from '../interfaces'\n\nconst pad2 = (value: number): string => String(value).padStart(2, '0')\nconst pad3 = (value: number): string => String(value).padStart(3, '0')\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\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\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 getColoredDuration = (duration: string, useColors: boolean): string => {\n  if (!useColors) {\n    return duration\n  }\n\n  return chalk.gray(duration)\n}\n\nconst getColoredTimestamp = (timestamp: string, useColors: boolean): string => {\n  if (!useColors) {\n    return timestamp\n  }\n\n  return chalk.bgHex('#FFA500').black(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 getContextString = (value: unknown): string => {\n  if (typeof value === 'object' && value !== null) {\n    return JSON.stringify(value)\n  }\n\n  return ''\n}\n\nexport const formatLine = ({\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}): string => {\n  const config = options.config\n  const useColors = shouldUseColors(options)\n  const format =\n    config?.customLogFormat ??\n    '🦊 {now} {level} {duration} {method} {pathname} {status} {message} {ip} {context}'\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  const ctxString = getContextString(data.context)\n  const coloredLevel = getColoredLevel(level, useColors)\n  const coloredMethod = getColoredMethod(request.method, useColors)\n  const coloredPathname = getColoredPathname(pathname, useColors)\n  const coloredStatus = getColoredStatus(status, useColors)\n  const coloredDuration = getColoredDuration(\n    `${durationMs.toFixed(2)}ms`,\n    useColors\n  )\n\n  return format\n    .replaceAll('{now}', timestamp)\n    .replaceAll('{epoch}', epoch)\n    .replaceAll('{level}', coloredLevel)\n    .replaceAll('{duration}', coloredDuration)\n    .replaceAll('{method}', coloredMethod)\n    .replaceAll('{pathname}', coloredPathname)\n    .replaceAll('{path}', coloredPathname)\n    .replaceAll('{status}', coloredStatus)\n    .replaceAll('{message}', message)\n    .replaceAll('{ip}', ip)\n    .replaceAll('{context}', ctxString)\n}\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  console.error(`Parsing error: ${message}`)\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'\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  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  console.error(\n    `${level} ${request.method} ${new URL(request.url).pathname} ${message}`\n  )\n}\n"
  ],
  "mappings": "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",
  "debugId": "7FAFEF1E34BEF02764756E2164756E21",
  "names": []
}
5
+ //# debugId=E21AEBE3A9BD8B9764756E2164756E21
6
+ //# 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, type SingletonBase } from 'elysia'\nimport { startServer } from './extensions'\nimport type { LogixlysiaStore, Options } from './interfaces'\nimport { createLogger } from './logger'\n\nexport type Logixlysia = Elysia<\n  'Logixlysia',\n  SingletonBase & { store: LogixlysiaStore }\n>\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 }) => {\n        if (server) {\n          startServer(server, 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 unknown 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 elysiaPkg from 'elysia/package.json'\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\nexport const renderBanner = (message: string): string => {\n  const versionLine = `Elysia v${elysiaPkg.version}`\n  const contentWidth = Math.max(message.length, versionLine.length)\n  const innerWidth = contentWidth + 4 // 2 spaces padding on both sides\n\n  const top = `┌${'─'.repeat(innerWidth)}┐`\n  const bot = `└${'─'.repeat(innerWidth)}┘`\n  const empty = `│${' '.repeat(innerWidth)}│`\n\n  const versionRow = `│${centerText(versionLine, innerWidth)}│`\n  const messageRow = `│  ${message}${' '.repeat(Math.max(0, innerWidth - message.length - 4))}  │`\n\n  return [top, empty, versionRow, empty, messageRow, empty, bot].join('\\n')\n}\n",
    "import type { Options } from '../interfaces'\nimport { 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\n  const format = options.config?.startupMessageFormat ?? 'banner'\n  if (format === 'simple') {\n    console.log(message)\n    return\n  }\n\n  console.log(renderBanner(message))\n}\n",
    "import pino from 'pino'\nimport type {\n  Logger,\n  LogLevel,\n  Options,\n  Pino,\n  RequestInfo,\n  StoreData,\n  LogFilter\n} from '../interfaces'\nimport { logToTransports } from '../output'\nimport { logToFile } from '../output/file'\nimport { formatLine } from './create-logger'\nimport { handleHttpError } from './handle-http-error'\n\nexport const createLogger = (options: Options = {}): Logger => {\n  const config = options.config\n\n  const pinoConfig = config?.pino\n  const { prettyPrint, ...pinoOptions } = pinoConfig ?? {}\n\n  const shouldPrettyPrint =\n    prettyPrint === true && pinoOptions.transport === undefined\n\n  const transport = shouldPrettyPrint\n    ? pino.transport({\n        target: 'pino-pretty',\n        options: {\n          colorize: process.stdout?.isTTY === true,\n          translateTime: config?.timestamp?.translateTime,\n          messageKey: pinoOptions.messageKey,\n          errorKey: pinoOptions.errorKey\n        }\n      })\n    : pinoOptions.transport\n\n  const pinoLogger: Pino = pino({\n    ...pinoOptions,\n    level: pinoOptions.level ?? 'info',\n    messageKey: pinoOptions.messageKey,\n    errorKey: pinoOptions.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 message = formatLine({ level, request, data, store, options })\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  level: LogLevel\n  request: RequestInfo\n  data: Record<string, unknown>\n  store: StoreData\n  options: Options\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  filePath: string\n  level: LogLevel\n  request: RequestInfo\n  data: Record<string, unknown>\n  store: StoreData\n  options: Options\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 chalk from 'chalk'\nimport { getStatusCode } from '../helpers/status'\nimport type {\n  LogLevel,\n  Options,\n  Pino,\n  RequestInfo,\n  StoreData\n} from '../interfaces'\n\nconst pad2 = (value: number): string => String(value).padStart(2, '0')\nconst pad3 = (value: number): string => String(value).padStart(3, '0')\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\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\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 getColoredDuration = (duration: string, useColors: boolean): string => {\n  if (!useColors) {\n    return duration\n  }\n\n  return chalk.gray(duration)\n}\n\nconst getColoredTimestamp = (timestamp: string, useColors: boolean): string => {\n  if (!useColors) {\n    return timestamp\n  }\n\n  return chalk.bgHex('#FFA500').black(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 getContextString = (value: unknown): string => {\n  if (typeof value === 'object' && value !== null) {\n    return JSON.stringify(value)\n  }\n\n  return ''\n}\n\nexport const formatLine = ({\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}): string => {\n  const config = options.config\n  const useColors = shouldUseColors(options)\n  const format =\n    config?.customLogFormat ??\n    '🦊 {now} {level} {duration} {method} {pathname} {status} {message} {ip} {context}'\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  const ctxString = getContextString(data.context)\n  const coloredLevel = getColoredLevel(level, useColors)\n  const coloredMethod = getColoredMethod(request.method, useColors)\n  const coloredPathname = getColoredPathname(pathname, useColors)\n  const coloredStatus = getColoredStatus(status, useColors)\n  const coloredDuration = getColoredDuration(\n    `${durationMs.toFixed(2)}ms`,\n    useColors\n  )\n\n  return format\n    .replaceAll('{now}', timestamp)\n    .replaceAll('{epoch}', epoch)\n    .replaceAll('{level}', coloredLevel)\n    .replaceAll('{duration}', coloredDuration)\n    .replaceAll('{method}', coloredMethod)\n    .replaceAll('{pathname}', coloredPathname)\n    .replaceAll('{path}', coloredPathname)\n    .replaceAll('{status}', coloredStatus)\n    .replaceAll('{message}', message)\n    .replaceAll('{ip}', ip)\n    .replaceAll('{context}', ctxString)\n}\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 { formatLine } 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  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 formattedMessage = formatLine({ level, request, data, store, options })\n  console.error(formattedMessage)\n}\n"
  ],
  "mappings": "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",
  "debugId": "E21AEBE3A9BD8B9764756E2164756E21",
  "names": []
}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "logixlysia",
3
- "version": "6.0.0",
3
+ "version": "6.1.0",
4
4
  "description": "🦊 Logixlysia is a logger for Elysia",
5
5
  "type": "module",
6
6
  "main": "dist/index.js",
@@ -38,15 +38,16 @@
38
38
  },
39
39
  "dependencies": {
40
40
  "chalk": "^5.6.2",
41
- "elysia": "^1.4.19",
42
41
  "pino": "^10.1.0",
43
42
  "pino-pretty": "^13.1.3"
44
43
  },
45
44
  "devDependencies": {
46
45
  "@types/bun": "^1.3.5",
47
- "bunup": "^0.16.11"
46
+ "bunup": "^0.16.17",
47
+ "elysia": "^1.4.19"
48
48
  },
49
49
  "peerDependencies": {
50
+ "elysia": "^1.4.19",
50
51
  "typescript": "^5.9.3"
51
52
  }
52
53
  }
package/src/interfaces.ts CHANGED
@@ -5,20 +5,22 @@ import type {
5
5
 
6
6
  export type Pino = PinoLogger<never, boolean>
7
7
 
8
+ export type RequestInfo = Request
9
+
8
10
  export type LogLevel = 'DEBUG' | 'INFO' | 'WARNING' | 'ERROR'
9
11
 
10
- export type StoreData = {
12
+ export interface StoreData {
11
13
  beforeTime: bigint
12
14
  }
13
15
 
14
- export type LogixlysiaStore = {
16
+ export interface LogixlysiaStore {
15
17
  logger: Logger
16
18
  pino: Pino
17
19
  beforeTime?: bigint
18
20
  [key: string]: unknown
19
21
  }
20
22
 
21
- export type Transport = {
23
+ export interface Transport {
22
24
  log: (
23
25
  level: LogLevel,
24
26
  message: string,
@@ -26,7 +28,7 @@ export type Transport = {
26
28
  ) => void | Promise<void>
27
29
  }
28
30
 
29
- export type LogRotationConfig = {
31
+ export interface LogRotationConfig {
30
32
  /**
31
33
  * Max log file size before rotation, e.g. '10m', '5k', or a byte count.
32
34
  */
@@ -43,7 +45,15 @@ export type LogRotationConfig = {
43
45
  compression?: 'gzip'
44
46
  }
45
47
 
46
- export type Options = {
48
+ export interface LogFilter {
49
+ /**
50
+ * Array of log levels to allow. If specified, only logs with these levels will be processed.
51
+ * If not specified, all log levels will be allowed.
52
+ */
53
+ level?: LogLevel[]
54
+ }
55
+
56
+ export interface Options {
47
57
  config?: {
48
58
  showStartupMessage?: boolean
49
59
  startupMessageFormat?: 'simple' | 'banner'
@@ -54,6 +64,9 @@ export type Options = {
54
64
  }
55
65
  customLogFormat?: string
56
66
 
67
+ // Filtering
68
+ logFilter?: LogFilter
69
+
57
70
  // Outputs
58
71
  transports?: Transport[]
59
72
  useTransportsOnly?: boolean
@@ -76,7 +89,7 @@ export class HttpError extends Error {
76
89
  }
77
90
  }
78
91
 
79
- export type Logger = {
92
+ export interface Logger {
80
93
  pino: Pino
81
94
  log: (
82
95
  level: LogLevel,
@@ -111,7 +124,7 @@ export type Logger = {
111
124
  ) => void
112
125
  }
113
126
 
114
- export type LogixlysiaContext = {
127
+ export interface LogixlysiaContext {
115
128
  request: Request
116
129
  store: LogixlysiaStore
117
130
  }
@@ -2,6 +2,7 @@ import type { LogLevel, Options, RequestInfo, StoreData } from '../interfaces'
2
2
  import { logToTransports } from '../output'
3
3
  import { logToFile } from '../output/file'
4
4
  import { parseError } from '../utils/error'
5
+ import { formatLine } from './create-logger'
5
6
 
6
7
  const isErrorWithStatus = (
7
8
  value: unknown
@@ -45,7 +46,6 @@ export const handleHttpError = (
45
46
  return
46
47
  }
47
48
 
48
- console.error(
49
- `${level} ${request.method} ${new URL(request.url).pathname} ${message}`
50
- )
49
+ const formattedMessage = formatLine({ level, request, data, store, options })
50
+ console.error(formattedMessage)
51
51
  }
@@ -5,7 +5,8 @@ import type {
5
5
  Options,
6
6
  Pino,
7
7
  RequestInfo,
8
- StoreData
8
+ StoreData,
9
+ LogFilter
9
10
  } from '../interfaces'
10
11
  import { logToTransports } from '../output'
11
12
  import { logToFile } from '../output/file'
@@ -41,12 +42,24 @@ export const createLogger = (options: Options = {}): Logger => {
41
42
  transport
42
43
  })
43
44
 
45
+ const shouldLog = (level: LogLevel, logFilter?: LogFilter): boolean => {
46
+ if (!logFilter?.level || logFilter.level.length === 0) {
47
+ return true
48
+ }
49
+ return logFilter.level.includes(level)
50
+ }
51
+
44
52
  const log = (
45
53
  level: LogLevel,
46
54
  request: RequestInfo,
47
55
  data: Record<string, unknown>,
48
56
  store: StoreData
49
57
  ): void => {
58
+ // Check if this log level should be filtered
59
+ if (!shouldLog(level, config?.logFilter)) {
60
+ return
61
+ }
62
+
50
63
  logToTransports({ level, request, data, store, options })
51
64
 
52
65
  const useTransportsOnly = config?.useTransportsOnly === true
@@ -4,7 +4,7 @@ import type { LogLevel, Options, RequestInfo, StoreData } from '../interfaces'
4
4
  import { ensureDir } from './fs'
5
5
  import { performRotation, shouldRotate } from './rotation-manager'
6
6
 
7
- type LogToFileInput = {
7
+ interface LogToFileInput {
8
8
  filePath: string
9
9
  level: LogLevel
10
10
  request: RequestInfo
@@ -1,6 +1,6 @@
1
1
  import type { LogLevel, Options, RequestInfo, StoreData } from '../interfaces'
2
2
 
3
- type LogToTransportsInput = {
3
+ interface LogToTransportsInput {
4
4
  level: LogLevel
5
5
  request: RequestInfo
6
6
  data: Record<string, unknown>
@@ -9,7 +9,5 @@ export const parseError = (error: unknown): string => {
9
9
  message = String(error)
10
10
  }
11
11
 
12
- console.error(`Parsing error: ${message}`)
13
-
14
12
  return message
15
13
  }