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