evlog 1.5.0 → 1.7.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +112 -0
- package/dist/_utils-DZA9nou3.mjs +23 -0
- package/dist/_utils-DZA9nou3.mjs.map +1 -0
- package/dist/adapters/axiom.d.mts +16 -15
- package/dist/adapters/axiom.d.mts.map +1 -0
- package/dist/adapters/axiom.mjs +96 -57
- package/dist/adapters/axiom.mjs.map +1 -0
- package/dist/adapters/better-stack.d.mts +62 -0
- package/dist/adapters/better-stack.d.mts.map +1 -0
- package/dist/adapters/better-stack.mjs +109 -0
- package/dist/adapters/better-stack.mjs.map +1 -0
- package/dist/adapters/otlp.d.mts +31 -30
- package/dist/adapters/otlp.d.mts.map +1 -0
- package/dist/adapters/otlp.mjs +198 -184
- package/dist/adapters/otlp.mjs.map +1 -0
- package/dist/adapters/posthog.d.mts +20 -19
- package/dist/adapters/posthog.d.mts.map +1 -0
- package/dist/adapters/posthog.mjs +110 -69
- package/dist/adapters/posthog.mjs.map +1 -0
- package/dist/adapters/sentry.d.mts +79 -0
- package/dist/adapters/sentry.d.mts.map +1 -0
- package/dist/adapters/sentry.mjs +233 -0
- package/dist/adapters/sentry.mjs.map +1 -0
- package/dist/enrichers.d.mts +74 -0
- package/dist/enrichers.d.mts.map +1 -0
- package/dist/enrichers.mjs +172 -0
- package/dist/enrichers.mjs.map +1 -0
- package/dist/error.d.mts +24 -22
- package/dist/error.d.mts.map +1 -0
- package/dist/error.mjs +107 -76
- package/dist/error.mjs.map +1 -0
- package/dist/index.d.mts +6 -5
- package/dist/index.mjs +6 -6
- package/dist/logger.d.mts +5 -3
- package/dist/logger.d.mts.map +1 -0
- package/dist/logger.mjs +204 -173
- package/dist/logger.mjs.map +1 -0
- package/dist/nitro/errorHandler.d.mts +4 -2
- package/dist/nitro/errorHandler.d.mts.map +1 -0
- package/dist/nitro/errorHandler.mjs +49 -38
- package/dist/nitro/errorHandler.mjs.map +1 -0
- package/dist/nitro/plugin.d.mts +4 -2
- package/dist/nitro/plugin.d.mts.map +1 -0
- package/dist/nitro/plugin.mjs +155 -137
- package/dist/nitro/plugin.mjs.map +1 -0
- package/dist/nuxt/module.d.mts +149 -122
- package/dist/nuxt/module.d.mts.map +1 -0
- package/dist/nuxt/module.mjs +66 -65
- package/dist/nuxt/module.mjs.map +1 -0
- package/dist/pipeline.d.mts +46 -0
- package/dist/pipeline.d.mts.map +1 -0
- package/dist/pipeline.mjs +122 -0
- package/dist/pipeline.mjs.map +1 -0
- package/dist/runtime/client/log.d.mts +7 -5
- package/dist/runtime/client/log.d.mts.map +1 -0
- package/dist/runtime/client/log.mjs +57 -62
- package/dist/runtime/client/log.mjs.map +1 -0
- package/dist/runtime/client/plugin.d.mts +3 -1
- package/dist/runtime/client/plugin.d.mts.map +1 -0
- package/dist/runtime/client/plugin.mjs +13 -12
- package/dist/runtime/client/plugin.mjs.map +1 -0
- package/dist/runtime/server/routes/_evlog/ingest.post.d.mts +4 -2
- package/dist/runtime/server/routes/_evlog/ingest.post.d.mts.map +1 -0
- package/dist/runtime/server/routes/_evlog/ingest.post.mjs +113 -77
- package/dist/runtime/server/routes/_evlog/ingest.post.mjs.map +1 -0
- package/dist/runtime/server/useLogger.d.mts +14 -3
- package/dist/runtime/server/useLogger.d.mts.map +1 -0
- package/dist/runtime/server/useLogger.mjs +39 -10
- package/dist/runtime/server/useLogger.mjs.map +1 -0
- package/dist/runtime/utils/parseError.d.mts +5 -3
- package/dist/runtime/utils/parseError.d.mts.map +1 -0
- package/dist/runtime/utils/parseError.mjs +25 -26
- package/dist/runtime/utils/parseError.mjs.map +1 -0
- package/dist/types.d.mts +348 -246
- package/dist/types.d.mts.map +1 -0
- package/dist/types.mjs +1 -1
- package/dist/utils.d.mts +19 -14
- package/dist/utils.d.mts.map +1 -0
- package/dist/utils.mjs +59 -50
- package/dist/utils.mjs.map +1 -0
- package/dist/workers.d.mts +10 -9
- package/dist/workers.d.mts.map +1 -0
- package/dist/workers.mjs +68 -40
- package/dist/workers.mjs.map +1 -0
- package/package.json +37 -9
- package/dist/adapters/axiom.d.ts +0 -62
- package/dist/adapters/otlp.d.ts +0 -83
- package/dist/adapters/posthog.d.ts +0 -72
- package/dist/error.d.ts +0 -63
- package/dist/index.d.ts +0 -5
- package/dist/logger.d.ts +0 -40
- package/dist/nitro/errorHandler.d.ts +0 -13
- package/dist/nitro/plugin.d.ts +0 -5
- package/dist/nuxt/module.d.ts +0 -125
- package/dist/runtime/client/log.d.ts +0 -10
- package/dist/runtime/client/plugin.d.ts +0 -3
- package/dist/runtime/server/routes/_evlog/ingest.post.d.ts +0 -5
- package/dist/runtime/server/useLogger.d.ts +0 -28
- package/dist/runtime/utils/parseError.d.ts +0 -5
- package/dist/types.d.ts +0 -364
- package/dist/utils.d.ts +0 -29
- package/dist/workers.d.ts +0 -45
package/dist/logger.mjs
CHANGED
|
@@ -1,203 +1,234 @@
|
|
|
1
|
-
import {
|
|
2
|
-
import { isDev, detectEnvironment, formatDuration, matchesPattern, getConsoleMethod, colors, getLevelColor } from './utils.mjs';
|
|
1
|
+
import { colors, detectEnvironment, formatDuration, getConsoleMethod, getLevelColor, isDev, matchesPattern } from "./utils.mjs";
|
|
3
2
|
|
|
3
|
+
//#region src/logger.ts
|
|
4
|
+
function isPlainObject(val) {
|
|
5
|
+
return val !== null && typeof val === "object" && !Array.isArray(val);
|
|
6
|
+
}
|
|
7
|
+
function deepDefaults(base, defaults) {
|
|
8
|
+
const result = { ...base };
|
|
9
|
+
for (const key in defaults) {
|
|
10
|
+
const baseVal = result[key];
|
|
11
|
+
const defaultVal = defaults[key];
|
|
12
|
+
if (baseVal === void 0 || baseVal === null) result[key] = defaultVal;
|
|
13
|
+
else if (isPlainObject(baseVal) && isPlainObject(defaultVal)) result[key] = deepDefaults(baseVal, defaultVal);
|
|
14
|
+
}
|
|
15
|
+
return result;
|
|
16
|
+
}
|
|
4
17
|
let globalEnv = {
|
|
5
|
-
|
|
6
|
-
|
|
18
|
+
service: "app",
|
|
19
|
+
environment: "development"
|
|
7
20
|
};
|
|
8
21
|
let globalPretty = isDev();
|
|
9
22
|
let globalSampling = {};
|
|
10
23
|
let globalStringify = true;
|
|
24
|
+
let globalDrain;
|
|
25
|
+
/**
|
|
26
|
+
* Initialize the logger with configuration.
|
|
27
|
+
* Call this once at application startup.
|
|
28
|
+
*/
|
|
11
29
|
function initLogger(config = {}) {
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
30
|
+
const detected = detectEnvironment();
|
|
31
|
+
globalEnv = {
|
|
32
|
+
service: config.env?.service ?? detected.service ?? "app",
|
|
33
|
+
environment: config.env?.environment ?? detected.environment ?? "development",
|
|
34
|
+
version: config.env?.version ?? detected.version,
|
|
35
|
+
commitHash: config.env?.commitHash ?? detected.commitHash,
|
|
36
|
+
region: config.env?.region ?? detected.region
|
|
37
|
+
};
|
|
38
|
+
globalPretty = config.pretty ?? isDev();
|
|
39
|
+
globalSampling = config.sampling ?? {};
|
|
40
|
+
globalStringify = config.stringify ?? true;
|
|
41
|
+
globalDrain = config.drain;
|
|
23
42
|
}
|
|
43
|
+
/**
|
|
44
|
+
* Determine if a log at the given level should be emitted based on sampling config.
|
|
45
|
+
* Error level defaults to 100% (always logged) unless explicitly configured otherwise.
|
|
46
|
+
*/
|
|
24
47
|
function shouldSample(level) {
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
if (percentage >= 100) return true;
|
|
32
|
-
return Math.random() * 100 < percentage;
|
|
48
|
+
const { rates } = globalSampling;
|
|
49
|
+
if (!rates) return true;
|
|
50
|
+
const percentage = level === "error" && rates.error === void 0 ? 100 : rates[level] ?? 100;
|
|
51
|
+
if (percentage <= 0) return false;
|
|
52
|
+
if (percentage >= 100) return true;
|
|
53
|
+
return Math.random() * 100 < percentage;
|
|
33
54
|
}
|
|
55
|
+
/**
|
|
56
|
+
* Evaluate tail sampling conditions to determine if a log should be force-kept.
|
|
57
|
+
* Returns true if ANY condition matches (OR logic).
|
|
58
|
+
*/
|
|
34
59
|
function shouldKeep(ctx) {
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
}
|
|
44
|
-
if (condition.path && ctx.path && matchesPattern(ctx.path, condition.path)) {
|
|
45
|
-
return true;
|
|
46
|
-
}
|
|
47
|
-
return false;
|
|
48
|
-
});
|
|
60
|
+
const { keep } = globalSampling;
|
|
61
|
+
if (!keep?.length) return false;
|
|
62
|
+
return keep.some((condition) => {
|
|
63
|
+
if (condition.status !== void 0 && ctx.status !== void 0 && ctx.status >= condition.status) return true;
|
|
64
|
+
if (condition.duration !== void 0 && ctx.duration !== void 0 && ctx.duration >= condition.duration) return true;
|
|
65
|
+
if (condition.path && ctx.path && matchesPattern(ctx.path, condition.path)) return true;
|
|
66
|
+
return false;
|
|
67
|
+
});
|
|
49
68
|
}
|
|
50
69
|
function emitWideEvent(level, event, skipSamplingCheck = false) {
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
console[getConsoleMethod(level)](formatted);
|
|
66
|
-
}
|
|
67
|
-
return formatted;
|
|
70
|
+
if (!skipSamplingCheck && !shouldSample(level)) return null;
|
|
71
|
+
const formatted = {
|
|
72
|
+
timestamp: (/* @__PURE__ */ new Date()).toISOString(),
|
|
73
|
+
level,
|
|
74
|
+
...globalEnv,
|
|
75
|
+
...event
|
|
76
|
+
};
|
|
77
|
+
if (globalPretty) prettyPrintWideEvent(formatted);
|
|
78
|
+
else if (globalStringify) console[getConsoleMethod(level)](JSON.stringify(formatted));
|
|
79
|
+
else console[getConsoleMethod(level)](formatted);
|
|
80
|
+
if (globalDrain) Promise.resolve(globalDrain({ event: formatted })).catch((err) => {
|
|
81
|
+
console.error("[evlog] drain failed:", err);
|
|
82
|
+
});
|
|
83
|
+
return formatted;
|
|
68
84
|
}
|
|
69
85
|
function emitTaggedLog(level, tag, message) {
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
86
|
+
if (globalPretty) {
|
|
87
|
+
if (!shouldSample(level)) return;
|
|
88
|
+
const color = getLevelColor(level);
|
|
89
|
+
const timestamp = (/* @__PURE__ */ new Date()).toISOString().slice(11, 23);
|
|
90
|
+
console.log(`${colors.dim}${timestamp}${colors.reset} ${color}[${tag}]${colors.reset} ${message}`);
|
|
91
|
+
return;
|
|
92
|
+
}
|
|
93
|
+
emitWideEvent(level, {
|
|
94
|
+
tag,
|
|
95
|
+
message
|
|
96
|
+
});
|
|
80
97
|
}
|
|
81
98
|
function formatValue(value) {
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
pairs.push(`${k}=${JSON.stringify(v)}`);
|
|
91
|
-
} else {
|
|
92
|
-
pairs.push(`${k}=${v}`);
|
|
93
|
-
}
|
|
94
|
-
}
|
|
95
|
-
}
|
|
96
|
-
return pairs.join(" ");
|
|
97
|
-
}
|
|
98
|
-
return String(value);
|
|
99
|
+
if (value === null || value === void 0) return String(value);
|
|
100
|
+
if (typeof value === "object") {
|
|
101
|
+
const pairs = [];
|
|
102
|
+
for (const [k, v] of Object.entries(value)) if (v !== void 0 && v !== null) if (typeof v === "object") pairs.push(`${k}=${JSON.stringify(v)}`);
|
|
103
|
+
else pairs.push(`${k}=${v}`);
|
|
104
|
+
return pairs.join(" ");
|
|
105
|
+
}
|
|
106
|
+
return String(value);
|
|
99
107
|
}
|
|
100
108
|
function prettyPrintWideEvent(event) {
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
});
|
|
109
|
+
const { timestamp, level, service, environment, version, ...rest } = event;
|
|
110
|
+
const levelColor = getLevelColor(level);
|
|
111
|
+
const ts = timestamp.slice(11, 23);
|
|
112
|
+
let header = `${colors.dim}${ts}${colors.reset} ${levelColor}${level.toUpperCase()}${colors.reset}`;
|
|
113
|
+
header += ` ${colors.cyan}[${service}]${colors.reset}`;
|
|
114
|
+
if (rest.method && rest.path) {
|
|
115
|
+
header += ` ${rest.method} ${rest.path}`;
|
|
116
|
+
delete rest.method;
|
|
117
|
+
delete rest.path;
|
|
118
|
+
}
|
|
119
|
+
if (rest.status) {
|
|
120
|
+
const statusColor = rest.status >= 400 ? colors.red : colors.green;
|
|
121
|
+
header += ` ${statusColor}${rest.status}${colors.reset}`;
|
|
122
|
+
delete rest.status;
|
|
123
|
+
}
|
|
124
|
+
if (rest.duration) {
|
|
125
|
+
header += ` ${colors.dim}in ${rest.duration}${colors.reset}`;
|
|
126
|
+
delete rest.duration;
|
|
127
|
+
}
|
|
128
|
+
console.log(header);
|
|
129
|
+
const entries = Object.entries(rest).filter(([_, v]) => v !== void 0);
|
|
130
|
+
const lastIndex = entries.length - 1;
|
|
131
|
+
entries.forEach(([key, value], index) => {
|
|
132
|
+
const prefix = index === lastIndex ? "└─" : "├─";
|
|
133
|
+
const formatted = formatValue(value);
|
|
134
|
+
console.log(` ${colors.dim}${prefix}${colors.reset} ${colors.cyan}${key}:${colors.reset} ${formatted}`);
|
|
135
|
+
});
|
|
129
136
|
}
|
|
130
137
|
function createLogMethod(level) {
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
} else {
|
|
137
|
-
emitTaggedLog(level, "log", String(tagOrEvent));
|
|
138
|
-
}
|
|
139
|
-
};
|
|
138
|
+
return function logMethod(tagOrEvent, message) {
|
|
139
|
+
if (typeof tagOrEvent === "string" && message !== void 0) emitTaggedLog(level, tagOrEvent, message);
|
|
140
|
+
else if (typeof tagOrEvent === "object") emitWideEvent(level, tagOrEvent);
|
|
141
|
+
else emitTaggedLog(level, "log", String(tagOrEvent));
|
|
142
|
+
};
|
|
140
143
|
}
|
|
144
|
+
/**
|
|
145
|
+
* Simple logging API - as easy as console.log
|
|
146
|
+
*
|
|
147
|
+
* @example
|
|
148
|
+
* ```ts
|
|
149
|
+
* log.info('auth', 'User logged in')
|
|
150
|
+
* log.error({ action: 'payment', error: 'failed' })
|
|
151
|
+
* ```
|
|
152
|
+
*/
|
|
141
153
|
const log = {
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
154
|
+
info: createLogMethod("info"),
|
|
155
|
+
error: createLogMethod("error"),
|
|
156
|
+
warn: createLogMethod("warn"),
|
|
157
|
+
debug: createLogMethod("debug")
|
|
146
158
|
};
|
|
159
|
+
/**
|
|
160
|
+
* Create a request-scoped logger for building wide events.
|
|
161
|
+
*
|
|
162
|
+
* @example
|
|
163
|
+
* ```ts
|
|
164
|
+
* const log = createRequestLogger({ method: 'POST', path: '/checkout' })
|
|
165
|
+
* log.set({ user: { id: '123' } })
|
|
166
|
+
* log.set({ cart: { items: 3 } })
|
|
167
|
+
* log.emit()
|
|
168
|
+
* ```
|
|
169
|
+
*/
|
|
147
170
|
function createRequestLogger(options = {}) {
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
171
|
+
const startTime = Date.now();
|
|
172
|
+
let context = {
|
|
173
|
+
method: options.method,
|
|
174
|
+
path: options.path,
|
|
175
|
+
requestId: options.requestId
|
|
176
|
+
};
|
|
177
|
+
let hasError = false;
|
|
178
|
+
return {
|
|
179
|
+
set(data) {
|
|
180
|
+
context = deepDefaults(data, context);
|
|
181
|
+
},
|
|
182
|
+
error(error, errorContext) {
|
|
183
|
+
hasError = true;
|
|
184
|
+
const err = typeof error === "string" ? new Error(error) : error;
|
|
185
|
+
context = deepDefaults({
|
|
186
|
+
...errorContext,
|
|
187
|
+
error: {
|
|
188
|
+
name: err.name,
|
|
189
|
+
message: err.message,
|
|
190
|
+
stack: err.stack,
|
|
191
|
+
..."statusCode" in err && { statusCode: err.statusCode },
|
|
192
|
+
..."statusMessage" in err && { statusMessage: err.statusMessage },
|
|
193
|
+
..."data" in err && { data: err.data },
|
|
194
|
+
..."cause" in err && { cause: err.cause }
|
|
195
|
+
}
|
|
196
|
+
}, context);
|
|
197
|
+
},
|
|
198
|
+
emit(overrides) {
|
|
199
|
+
const durationMs = Date.now() - startTime;
|
|
200
|
+
const duration = formatDuration(durationMs);
|
|
201
|
+
const level = hasError ? "error" : "info";
|
|
202
|
+
const { _forceKeep, ...restOverrides } = overrides ?? {};
|
|
203
|
+
const tailCtx = {
|
|
204
|
+
status: context.status ?? restOverrides.status,
|
|
205
|
+
duration: durationMs,
|
|
206
|
+
path: context.path,
|
|
207
|
+
method: context.method,
|
|
208
|
+
context: {
|
|
209
|
+
...context,
|
|
210
|
+
...restOverrides
|
|
211
|
+
}
|
|
212
|
+
};
|
|
213
|
+
if (!(_forceKeep || shouldKeep(tailCtx)) && !shouldSample(level)) return null;
|
|
214
|
+
return emitWideEvent(level, {
|
|
215
|
+
...context,
|
|
216
|
+
...restOverrides,
|
|
217
|
+
duration
|
|
218
|
+
}, true);
|
|
219
|
+
},
|
|
220
|
+
getContext() {
|
|
221
|
+
return { ...context };
|
|
222
|
+
}
|
|
223
|
+
};
|
|
198
224
|
}
|
|
225
|
+
/**
|
|
226
|
+
* Get the current environment context.
|
|
227
|
+
*/
|
|
199
228
|
function getEnvironment() {
|
|
200
|
-
|
|
229
|
+
return { ...globalEnv };
|
|
201
230
|
}
|
|
202
231
|
|
|
232
|
+
//#endregion
|
|
203
233
|
export { createRequestLogger, getEnvironment, initLogger, log, shouldKeep };
|
|
234
|
+
//# sourceMappingURL=logger.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"logger.mjs","names":[],"sources":["../src/logger.ts"],"sourcesContent":["import type { DrainContext, EnvironmentContext, FieldContext, Log, LogLevel, LoggerConfig, RequestLogger, RequestLoggerOptions, SamplingConfig, TailSamplingContext, WideEvent } from './types'\nimport { colors, detectEnvironment, formatDuration, getConsoleMethod, getLevelColor, isDev, matchesPattern } from './utils'\n\nfunction isPlainObject(val: unknown): val is Record<string, unknown> {\n return val !== null && typeof val === 'object' && !Array.isArray(val)\n}\n\nfunction deepDefaults(base: Record<string, unknown>, defaults: Record<string, unknown>): Record<string, unknown> {\n const result = { ...base }\n for (const key in defaults) {\n const baseVal = result[key]\n const defaultVal = defaults[key]\n if (baseVal === undefined || baseVal === null) {\n result[key] = defaultVal\n } else if (isPlainObject(baseVal) && isPlainObject(defaultVal)) {\n result[key] = deepDefaults(baseVal, defaultVal)\n }\n }\n return result\n}\n\nlet globalEnv: EnvironmentContext = {\n service: 'app',\n environment: 'development',\n}\n\nlet globalPretty = isDev()\nlet globalSampling: SamplingConfig = {}\nlet globalStringify = true\nlet globalDrain: ((ctx: DrainContext) => void | Promise<void>) | undefined\n\n/**\n * Initialize the logger with configuration.\n * Call this once at application startup.\n */\nexport function initLogger(config: LoggerConfig = {}): void {\n const detected = detectEnvironment()\n\n globalEnv = {\n service: config.env?.service ?? detected.service ?? 'app',\n environment: config.env?.environment ?? detected.environment ?? 'development',\n version: config.env?.version ?? detected.version,\n commitHash: config.env?.commitHash ?? detected.commitHash,\n region: config.env?.region ?? detected.region,\n }\n\n globalPretty = config.pretty ?? isDev()\n globalSampling = config.sampling ?? {}\n globalStringify = config.stringify ?? true\n globalDrain = config.drain\n}\n\n/**\n * Determine if a log at the given level should be emitted based on sampling config.\n * Error level defaults to 100% (always logged) unless explicitly configured otherwise.\n */\nfunction shouldSample(level: LogLevel): boolean {\n const { rates } = globalSampling\n if (!rates) {\n return true // No sampling configured, log everything\n }\n\n // Error defaults to 100% unless explicitly set\n const percentage = level === 'error' && rates.error === undefined\n ? 100\n : rates[level] ?? 100\n\n // 0% = never log, 100% = always log\n if (percentage <= 0) return false\n if (percentage >= 100) return true\n\n return Math.random() * 100 < percentage\n}\n\n/**\n * Evaluate tail sampling conditions to determine if a log should be force-kept.\n * Returns true if ANY condition matches (OR logic).\n */\nexport function shouldKeep(ctx: TailSamplingContext): boolean {\n const { keep } = globalSampling\n if (!keep?.length) return false\n\n return keep.some((condition) => {\n if (condition.status !== undefined && ctx.status !== undefined && ctx.status >= condition.status) {\n return true\n }\n if (condition.duration !== undefined && ctx.duration !== undefined && ctx.duration >= condition.duration) {\n return true\n }\n if (condition.path && ctx.path && matchesPattern(ctx.path, condition.path)) {\n return true\n }\n return false\n })\n}\n\nfunction emitWideEvent(level: LogLevel, event: Record<string, unknown>, skipSamplingCheck = false): WideEvent | null {\n if (!skipSamplingCheck && !shouldSample(level)) {\n return null\n }\n\n const formatted: WideEvent = {\n timestamp: new Date().toISOString(),\n level,\n ...globalEnv,\n ...event,\n }\n\n if (globalPretty) {\n prettyPrintWideEvent(formatted)\n } else if (globalStringify) {\n console[getConsoleMethod(level)](JSON.stringify(formatted))\n } else {\n console[getConsoleMethod(level)](formatted)\n }\n\n if (globalDrain) {\n Promise.resolve(globalDrain({ event: formatted })).catch((err) => {\n console.error('[evlog] drain failed:', err)\n })\n }\n\n return formatted\n}\n\nfunction emitTaggedLog(level: LogLevel, tag: string, message: string): void {\n if (globalPretty) {\n if (!shouldSample(level)) {\n return\n }\n const color = getLevelColor(level)\n const timestamp = new Date().toISOString().slice(11, 23)\n console.log(`${colors.dim}${timestamp}${colors.reset} ${color}[${tag}]${colors.reset} ${message}`)\n return\n }\n emitWideEvent(level, { tag, message })\n}\n\nfunction formatValue(value: unknown): string {\n if (value === null || value === undefined) {\n return String(value)\n }\n if (typeof value === 'object') {\n // Flatten object to key=value pairs\n const pairs: string[] = []\n for (const [k, v] of Object.entries(value as Record<string, unknown>)) {\n if (v !== undefined && v !== null) {\n if (typeof v === 'object') {\n // For nested objects, show as JSON\n pairs.push(`${k}=${JSON.stringify(v)}`)\n } else {\n pairs.push(`${k}=${v}`)\n }\n }\n }\n return pairs.join(' ')\n }\n return String(value)\n}\n\nfunction prettyPrintWideEvent(event: Record<string, unknown>): void {\n const { timestamp, level, service, environment, version, ...rest } = event\n const levelColor = getLevelColor(level as string)\n const ts = (timestamp as string).slice(11, 23)\n\n let header = `${colors.dim}${ts}${colors.reset} ${levelColor}${(level as string).toUpperCase()}${colors.reset}`\n header += ` ${colors.cyan}[${service}]${colors.reset}`\n\n if (rest.method && rest.path) {\n header += ` ${rest.method} ${rest.path}`\n delete rest.method\n delete rest.path\n }\n\n if (rest.status) {\n const statusColor = (rest.status as number) >= 400 ? colors.red : colors.green\n header += ` ${statusColor}${rest.status}${colors.reset}`\n delete rest.status\n }\n\n if (rest.duration) {\n header += ` ${colors.dim}in ${rest.duration}${colors.reset}`\n delete rest.duration\n }\n\n console.log(header)\n\n const entries = Object.entries(rest).filter(([_, v]) => v !== undefined)\n const lastIndex = entries.length - 1\n\n entries.forEach(([key, value], index) => {\n const isLast = index === lastIndex\n const prefix = isLast ? '└─' : '├─'\n const formatted = formatValue(value)\n console.log(` ${colors.dim}${prefix}${colors.reset} ${colors.cyan}${key}:${colors.reset} ${formatted}`)\n })\n}\n\nfunction createLogMethod(level: LogLevel) {\n return function logMethod(tagOrEvent: string | Record<string, unknown>, message?: string): void {\n if (typeof tagOrEvent === 'string' && message !== undefined) {\n emitTaggedLog(level, tagOrEvent, message)\n } else if (typeof tagOrEvent === 'object') {\n emitWideEvent(level, tagOrEvent)\n } else {\n emitTaggedLog(level, 'log', String(tagOrEvent))\n }\n }\n}\n\n/**\n * Simple logging API - as easy as console.log\n *\n * @example\n * ```ts\n * log.info('auth', 'User logged in')\n * log.error({ action: 'payment', error: 'failed' })\n * ```\n */\nexport const log: Log = {\n info: createLogMethod('info'),\n error: createLogMethod('error'),\n warn: createLogMethod('warn'),\n debug: createLogMethod('debug'),\n}\n\n/**\n * Create a request-scoped logger for building wide events.\n *\n * @example\n * ```ts\n * const log = createRequestLogger({ method: 'POST', path: '/checkout' })\n * log.set({ user: { id: '123' } })\n * log.set({ cart: { items: 3 } })\n * log.emit()\n * ```\n */\nexport function createRequestLogger<T extends object = Record<string, unknown>>(options: RequestLoggerOptions = {}): RequestLogger<T> {\n const startTime = Date.now()\n let context: Record<string, unknown> = {\n method: options.method,\n path: options.path,\n requestId: options.requestId,\n }\n let hasError = false\n\n return {\n set(data: FieldContext<T>): void {\n context = deepDefaults(data as Record<string, unknown>, context) as Record<string, unknown>\n },\n\n error(error: Error | string, errorContext?: FieldContext<T>): void {\n hasError = true\n const err = typeof error === 'string' ? new Error(error) : error\n\n const errorData = {\n ...(errorContext as Record<string, unknown>),\n error: {\n name: err.name,\n message: err.message,\n stack: err.stack,\n ...('statusCode' in err && { statusCode: (err as Record<string, unknown>).statusCode }),\n ...('statusMessage' in err && { statusMessage: (err as Record<string, unknown>).statusMessage }),\n ...('data' in err && { data: (err as Record<string, unknown>).data }),\n ...('cause' in err && { cause: (err as unknown as Record<string, unknown>).cause }),\n },\n }\n context = deepDefaults(errorData, context) as Record<string, unknown>\n },\n\n emit(overrides?: FieldContext<T> & { _forceKeep?: boolean }): WideEvent | null {\n const durationMs = Date.now() - startTime\n const duration = formatDuration(durationMs)\n const level: LogLevel = hasError ? 'error' : 'info'\n\n // Extract _forceKeep from overrides (set by evlog:emit:keep hook)\n const { _forceKeep, ...restOverrides } = (overrides ?? {}) as Record<string, unknown> & { _forceKeep?: boolean }\n\n // Build tail sampling context\n const tailCtx: TailSamplingContext = {\n status: (context.status ?? restOverrides.status) as number | undefined,\n duration: durationMs,\n path: context.path as string | undefined,\n method: context.method as string | undefined,\n context: { ...context, ...restOverrides },\n }\n\n // Tail sampling: force keep if hook or built-in conditions match\n const forceKeep = _forceKeep || shouldKeep(tailCtx)\n\n // Apply head sampling only if not force-kept\n if (!forceKeep && !shouldSample(level)) {\n return null\n }\n\n return emitWideEvent(level, {\n ...context,\n ...restOverrides,\n duration,\n }, true)\n },\n\n getContext(): FieldContext<T> & Record<string, unknown> {\n return { ...context }\n },\n }\n}\n\n/**\n * Get the current environment context.\n */\nexport function getEnvironment(): EnvironmentContext {\n return { ...globalEnv }\n}\n"],"mappings":"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"}
|
|
@@ -1,5 +1,6 @@
|
|
|
1
|
-
import * as nitropack from
|
|
1
|
+
import * as nitropack from "nitropack";
|
|
2
2
|
|
|
3
|
+
//#region src/nitro/errorHandler.d.ts
|
|
3
4
|
/**
|
|
4
5
|
* Custom Nitro error handler that properly serializes EvlogError.
|
|
5
6
|
* This ensures that 'data' (containing 'why', 'fix', 'link') is preserved
|
|
@@ -9,5 +10,6 @@ import * as nitropack from 'nitropack';
|
|
|
9
10
|
* sanitizing internal error details in production for 5xx errors.
|
|
10
11
|
*/
|
|
11
12
|
declare const _default: nitropack.NitroErrorHandler;
|
|
12
|
-
|
|
13
|
+
//#endregion
|
|
13
14
|
export { _default as default };
|
|
15
|
+
//# sourceMappingURL=errorHandler.d.mts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"errorHandler.d.mts","names":[],"sources":["../../src/nitro/errorHandler.ts"],"mappings":""}
|
|
@@ -1,41 +1,52 @@
|
|
|
1
|
-
import { defineNitroErrorHandler } from
|
|
2
|
-
import { getRequestURL,
|
|
1
|
+
import { defineNitroErrorHandler } from "nitropack/runtime";
|
|
2
|
+
import { getRequestURL, send, setResponseHeader, setResponseStatus } from "h3";
|
|
3
3
|
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
4
|
+
//#region src/nitro/errorHandler.ts
|
|
5
|
+
/**
|
|
6
|
+
* Custom Nitro error handler that properly serializes EvlogError.
|
|
7
|
+
* This ensures that 'data' (containing 'why', 'fix', 'link') is preserved
|
|
8
|
+
* in the JSON response regardless of the underlying HTTP framework.
|
|
9
|
+
*
|
|
10
|
+
* For non-EvlogError, it preserves Nitro's default response shape while
|
|
11
|
+
* sanitizing internal error details in production for 5xx errors.
|
|
12
|
+
*/
|
|
13
|
+
var errorHandler_default = defineNitroErrorHandler((error, event) => {
|
|
14
|
+
const evlogError = error.name === "EvlogError" ? error : error.cause?.name === "EvlogError" ? error.cause : null;
|
|
15
|
+
const isDev = process.env.NODE_ENV === "development";
|
|
16
|
+
const url = getRequestURL(event, { xForwardedHost: true }).pathname;
|
|
17
|
+
if (!evlogError) {
|
|
18
|
+
const status = error.statusCode ?? error.status ?? 500;
|
|
19
|
+
const rawMessage = (error.statusText ?? error.statusMessage ?? error.message) || "Internal Server Error";
|
|
20
|
+
const message = isDev ? rawMessage : status >= 500 ? "Internal Server Error" : rawMessage;
|
|
21
|
+
setResponseStatus(event, status);
|
|
22
|
+
setResponseHeader(event, "Content-Type", "application/json");
|
|
23
|
+
return send(event, JSON.stringify({
|
|
24
|
+
url,
|
|
25
|
+
status,
|
|
26
|
+
statusCode: status,
|
|
27
|
+
statusText: message,
|
|
28
|
+
statusMessage: message,
|
|
29
|
+
message,
|
|
30
|
+
error: true
|
|
31
|
+
}));
|
|
32
|
+
}
|
|
33
|
+
const status = evlogError.status ?? evlogError.statusCode ?? 500;
|
|
34
|
+
setResponseStatus(event, status);
|
|
35
|
+
setResponseHeader(event, "Content-Type", "application/json");
|
|
36
|
+
const { data } = evlogError;
|
|
37
|
+
const statusMessage = evlogError.statusMessage || evlogError.message;
|
|
38
|
+
return send(event, JSON.stringify({
|
|
39
|
+
url,
|
|
40
|
+
status,
|
|
41
|
+
statusCode: status,
|
|
42
|
+
statusText: statusMessage,
|
|
43
|
+
statusMessage,
|
|
44
|
+
message: evlogError.message,
|
|
45
|
+
error: true,
|
|
46
|
+
...data !== void 0 && { data }
|
|
47
|
+
}));
|
|
39
48
|
});
|
|
40
49
|
|
|
41
|
-
|
|
50
|
+
//#endregion
|
|
51
|
+
export { errorHandler_default as default };
|
|
52
|
+
//# sourceMappingURL=errorHandler.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"errorHandler.mjs","names":[],"sources":["../../src/nitro/errorHandler.ts"],"sourcesContent":["import { defineNitroErrorHandler } from 'nitropack/runtime'\nimport { getRequestURL, setResponseHeader, setResponseStatus, send } from 'h3'\n\n/**\n * Custom Nitro error handler that properly serializes EvlogError.\n * This ensures that 'data' (containing 'why', 'fix', 'link') is preserved\n * in the JSON response regardless of the underlying HTTP framework.\n *\n * For non-EvlogError, it preserves Nitro's default response shape while\n * sanitizing internal error details in production for 5xx errors.\n */\nexport default defineNitroErrorHandler((error, event) => {\n // Check if this is an EvlogError (by name or by checking cause)\n const evlogError = error.name === 'EvlogError'\n ? error\n : (error.cause as Error)?.name === 'EvlogError'\n ? error.cause as Error\n : null\n\n const isDev = process.env.NODE_ENV === 'development'\n const url = getRequestURL(event, { xForwardedHost: true }).pathname\n\n // For non-EvlogError, preserve Nitro's default response shape\n if (!evlogError) {\n const status = (error as { statusCode?: number }).statusCode\n ?? (error as { status?: number }).status\n ?? 500\n\n // Derive message from statusText/statusMessage/message for cross-version compatibility\n const rawMessage = ((error as { statusText?: string }).statusText\n ?? (error as { statusMessage?: string }).statusMessage\n ?? error.message) || 'Internal Server Error'\n\n // Sanitize internal error details in production for 5xx errors\n const message = isDev\n ? rawMessage\n : (status >= 500 ? 'Internal Server Error' : rawMessage)\n\n setResponseStatus(event, status)\n setResponseHeader(event, 'Content-Type', 'application/json')\n\n // Preserve Nitro's default response shape with both legacy and modern fields\n return send(event, JSON.stringify({\n url,\n status,\n statusCode: status,\n statusText: message,\n statusMessage: message,\n message,\n error: true,\n }))\n }\n\n // Derive status from evlogError to ensure consistency between\n // HTTP response status and response body\n const status = (evlogError as { status?: number }).status\n ?? (evlogError as { statusCode?: number }).statusCode\n ?? 500\n\n setResponseStatus(event, status)\n setResponseHeader(event, 'Content-Type', 'application/json')\n\n // Serialize EvlogError with all its data, preserving Nitro's response shape\n const { data } = evlogError as { data?: unknown }\n const statusMessage = (evlogError as { statusMessage?: string }).statusMessage || evlogError.message\n return send(event, JSON.stringify({\n url,\n status,\n statusCode: status,\n statusText: statusMessage,\n statusMessage,\n message: evlogError.message,\n error: true,\n ...(data !== undefined && { data }),\n }))\n})\n"],"mappings":";;;;;;;;;;;;AAWA,2BAAe,yBAAyB,OAAO,UAAU;CAEvD,MAAM,aAAa,MAAM,SAAS,eAC9B,QACC,MAAM,OAAiB,SAAS,eAC/B,MAAM,QACN;CAEN,MAAM,QAAQ,QAAQ,IAAI,aAAa;CACvC,MAAM,MAAM,cAAc,OAAO,EAAE,gBAAgB,MAAM,CAAC,CAAC;AAG3D,KAAI,CAAC,YAAY;EACf,MAAM,SAAU,MAAkC,cAC5C,MAA8B,UAC/B;EAGL,MAAM,cAAe,MAAkC,cACjD,MAAqC,iBACtC,MAAM,YAAY;EAGvB,MAAM,UAAU,QACZ,aACC,UAAU,MAAM,0BAA0B;AAE/C,oBAAkB,OAAO,OAAO;AAChC,oBAAkB,OAAO,gBAAgB,mBAAmB;AAG5D,SAAO,KAAK,OAAO,KAAK,UAAU;GAChC;GACA;GACA,YAAY;GACZ,YAAY;GACZ,eAAe;GACf;GACA,OAAO;GACR,CAAC,CAAC;;CAKL,MAAM,SAAU,WAAmC,UAC7C,WAAuC,cACxC;AAEL,mBAAkB,OAAO,OAAO;AAChC,mBAAkB,OAAO,gBAAgB,mBAAmB;CAG5D,MAAM,EAAE,SAAS;CACjB,MAAM,gBAAiB,WAA0C,iBAAiB,WAAW;AAC7F,QAAO,KAAK,OAAO,KAAK,UAAU;EAChC;EACA;EACA,YAAY;EACZ,YAAY;EACZ;EACA,SAAS,WAAW;EACpB,OAAO;EACP,GAAI,SAAS,UAAa,EAAE,MAAM;EACnC,CAAC,CAAC;EACH"}
|
package/dist/nitro/plugin.d.mts
CHANGED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"plugin.d.mts","names":[],"sources":["../../src/nitro/plugin.ts"],"mappings":""}
|