evlog 2.11.0 → 2.12.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 +4 -2
- package/dist/{_drain-YH8ERc5l.mjs → _drain-CmCtsuF6.mjs} +1 -1
- package/dist/{_drain-YH8ERc5l.mjs.map → _drain-CmCtsuF6.mjs.map} +1 -1
- package/dist/{_http-C_2wbJw3.mjs → _http-CHSsrWDJ.mjs} +2 -2
- package/dist/{_http-C_2wbJw3.mjs.map → _http-CHSsrWDJ.mjs.map} +1 -1
- package/dist/{_severity-BZhz3f9e.mjs → _severity-CQijvfhU.mjs} +1 -1
- package/dist/{_severity-BZhz3f9e.mjs.map → _severity-CQijvfhU.mjs.map} +1 -1
- package/dist/adapters/axiom.d.mts +1 -1
- package/dist/adapters/axiom.mjs +2 -2
- package/dist/adapters/better-stack.d.mts +1 -1
- package/dist/adapters/better-stack.mjs +2 -2
- package/dist/adapters/datadog.d.mts +1 -1
- package/dist/adapters/datadog.mjs +2 -2
- package/dist/adapters/fs.d.mts +1 -1
- package/dist/adapters/fs.mjs +1 -1
- package/dist/adapters/hyperdx.d.mts +1 -1
- package/dist/adapters/hyperdx.mjs +2 -2
- package/dist/adapters/otlp.d.mts +1 -1
- package/dist/adapters/otlp.mjs +3 -3
- package/dist/adapters/posthog.d.mts +1 -1
- package/dist/adapters/posthog.mjs +2 -2
- package/dist/adapters/sentry.d.mts +1 -1
- package/dist/adapters/sentry.mjs +3 -3
- package/dist/ai/index.d.mts +144 -5
- package/dist/ai/index.d.mts.map +1 -1
- package/dist/ai/index.mjs +108 -5
- package/dist/ai/index.mjs.map +1 -1
- package/dist/browser.d.mts +13 -52
- package/dist/browser.d.mts.map +1 -1
- package/dist/browser.mjs +5 -81
- package/dist/browser.mjs.map +1 -1
- package/dist/client.d.mts +2 -2
- package/dist/client.mjs +2 -2
- package/dist/{dist-BFn8qsRC.mjs → dist-Do8P4zWd.mjs} +1 -1
- package/dist/{dist-BFn8qsRC.mjs.map → dist-Do8P4zWd.mjs.map} +1 -1
- package/dist/elysia/index.d.mts +2 -2
- package/dist/elysia/index.mjs +1 -1
- package/dist/enrichers.d.mts +1 -1
- package/dist/{error-plrBYLQk.d.mts → error-WRz4_F3W.d.mts} +2 -2
- package/dist/{error-plrBYLQk.d.mts.map → error-WRz4_F3W.d.mts.map} +1 -1
- package/dist/error.d.mts +1 -1
- package/dist/{errors-gH4C9KSC.mjs → errors-BJRXUfMg.mjs} +1 -1
- package/dist/{errors-gH4C9KSC.mjs.map → errors-BJRXUfMg.mjs.map} +1 -1
- package/dist/{errors-bPoj9UZk.d.mts → errors-J2kt7mZh.d.mts} +2 -2
- package/dist/{errors-bPoj9UZk.d.mts.map → errors-J2kt7mZh.d.mts.map} +1 -1
- package/dist/express/index.d.mts +2 -2
- package/dist/express/index.mjs +2 -2
- package/dist/fastify/index.d.mts +2 -2
- package/dist/fastify/index.mjs +2 -2
- package/dist/{headers-BSi3UHKL.mjs → headers-ht4yS2mx.mjs} +6 -4
- package/dist/headers-ht4yS2mx.mjs.map +1 -0
- package/dist/hono/index.d.mts +2 -2
- package/dist/hono/index.mjs +1 -1
- package/dist/http.d.mts +65 -0
- package/dist/http.d.mts.map +1 -0
- package/dist/http.mjs +94 -0
- package/dist/http.mjs.map +1 -0
- package/dist/index.d.mts +7 -6
- package/dist/index.mjs +3 -2
- package/dist/{logger-CG1eop2_.d.mts → logger-Bm0k3Hf3.d.mts} +2 -2
- package/dist/{logger-CG1eop2_.d.mts.map → logger-Bm0k3Hf3.d.mts.map} +1 -1
- package/dist/logger-DY0X5oQd.mjs +704 -0
- package/dist/logger-DY0X5oQd.mjs.map +1 -0
- package/dist/logger.d.mts +1 -1
- package/dist/logger.mjs +1 -361
- package/dist/{middleware-DojmTj9Y.d.mts → middleware-D_igVy93.d.mts} +9 -2
- package/dist/middleware-D_igVy93.d.mts.map +1 -0
- package/dist/nestjs/index.d.mts +2 -2
- package/dist/nestjs/index.mjs +2 -2
- package/dist/next/client.d.mts +9 -3
- package/dist/next/client.d.mts.map +1 -1
- package/dist/next/client.mjs +5 -3
- package/dist/next/client.mjs.map +1 -1
- package/dist/next/index.d.mts +9 -4
- package/dist/next/index.d.mts.map +1 -1
- package/dist/next/index.mjs +3 -2
- package/dist/next/index.mjs.map +1 -1
- package/dist/next/instrumentation.d.mts +3 -1
- package/dist/next/instrumentation.d.mts.map +1 -1
- package/dist/next/instrumentation.mjs +2 -1
- package/dist/next/instrumentation.mjs.map +1 -1
- package/dist/nitro/errorHandler.mjs +2 -2
- package/dist/nitro/module.d.mts +2 -2
- package/dist/nitro/plugin.mjs +23 -5
- package/dist/nitro/plugin.mjs.map +1 -1
- package/dist/nitro/v3/errorHandler.mjs +3 -3
- package/dist/nitro/v3/index.d.mts +2 -2
- package/dist/nitro/v3/module.d.mts +1 -1
- package/dist/nitro/v3/plugin.mjs +25 -6
- package/dist/nitro/v3/plugin.mjs.map +1 -1
- package/dist/nitro/v3/useLogger.d.mts +1 -1
- package/dist/{nitro-CfGx0wDJ.d.mts → nitro-BeRXZcBd.d.mts} +13 -2
- package/dist/nitro-BeRXZcBd.d.mts.map +1 -0
- package/dist/{nitro-Dpq5ZmcM.mjs → nitro-OmT_M4Pb.mjs} +2 -2
- package/dist/nitro-OmT_M4Pb.mjs.map +1 -0
- package/dist/{nitroConfigBridge-fidbf-Y_.mjs → nitroConfigBridge-C37lXaNm.mjs} +1 -1
- package/dist/{nitroConfigBridge-fidbf-Y_.mjs.map → nitroConfigBridge-C37lXaNm.mjs.map} +1 -1
- package/dist/nuxt/module.d.mts +26 -1
- package/dist/nuxt/module.d.mts.map +1 -1
- package/dist/nuxt/module.mjs +10 -3
- package/dist/nuxt/module.mjs.map +1 -1
- package/dist/{parseError-B_qXj8x4.d.mts → parseError-DhXS_vzM.d.mts} +2 -2
- package/dist/parseError-DhXS_vzM.d.mts.map +1 -0
- package/dist/react-router/index.d.mts +2 -2
- package/dist/react-router/index.mjs +2 -2
- package/dist/{routes-CE3_c-iZ.mjs → routes-CGPmbzCZ.mjs} +1 -1
- package/dist/{routes-CE3_c-iZ.mjs.map → routes-CGPmbzCZ.mjs.map} +1 -1
- package/dist/runtime/client/log.d.mts +7 -2
- package/dist/runtime/client/log.d.mts.map +1 -1
- package/dist/runtime/client/log.mjs +24 -6
- package/dist/runtime/client/log.mjs.map +1 -1
- package/dist/runtime/client/plugin.mjs +1 -0
- package/dist/runtime/client/plugin.mjs.map +1 -1
- package/dist/runtime/server/routes/_evlog/ingest.post.mjs +1 -1
- package/dist/runtime/server/useLogger.d.mts +1 -1
- package/dist/runtime/utils/parseError.d.mts +2 -2
- package/dist/runtime/utils/parseError.mjs +1 -1
- package/dist/{source-location-B1VVgXkh.mjs → source-location-DRvDDqfq.mjs} +1 -1
- package/dist/{source-location-B1VVgXkh.mjs.map → source-location-DRvDDqfq.mjs.map} +1 -1
- package/dist/{storage-B6NPh8rV.mjs → storage-DpLJYMoc.mjs} +1 -1
- package/dist/{storage-B6NPh8rV.mjs.map → storage-DpLJYMoc.mjs.map} +1 -1
- package/dist/sveltekit/index.d.mts +2 -2
- package/dist/sveltekit/index.mjs +4 -4
- package/dist/toolkit.d.mts +3 -3
- package/dist/toolkit.mjs +4 -4
- package/dist/{types-v_JkG_D7.d.mts → types-D5OwxZCw.d.mts} +71 -2
- package/dist/types-D5OwxZCw.d.mts.map +1 -0
- package/dist/types.d.mts +2 -2
- package/dist/{useLogger-TjKH37BO.d.mts → useLogger-Dcj1Nrsa.d.mts} +2 -2
- package/dist/{useLogger-TjKH37BO.d.mts.map → useLogger-Dcj1Nrsa.d.mts.map} +1 -1
- package/dist/utils-Bnc95-VC.d.mts +54 -0
- package/dist/utils-Bnc95-VC.d.mts.map +1 -0
- package/dist/utils.d.mts +2 -50
- package/dist/utils.mjs +13 -1
- package/dist/utils.mjs.map +1 -1
- package/dist/vite/index.d.mts +5 -1
- package/dist/vite/index.d.mts.map +1 -1
- package/dist/vite/index.mjs +3 -1
- package/dist/vite/index.mjs.map +1 -1
- package/dist/workers.d.mts +1 -1
- package/dist/workers.mjs +1 -1
- package/package.json +24 -16
- package/dist/headers-BSi3UHKL.mjs.map +0 -1
- package/dist/logger.mjs.map +0 -1
- package/dist/middleware-DojmTj9Y.d.mts.map +0 -1
- package/dist/nitro-CfGx0wDJ.d.mts.map +0 -1
- package/dist/nitro-Dpq5ZmcM.mjs.map +0 -1
- package/dist/parseError-B_qXj8x4.d.mts.map +0 -1
- package/dist/types-v_JkG_D7.d.mts.map +0 -1
- package/dist/utils.d.mts.map +0 -1
|
@@ -0,0 +1,704 @@
|
|
|
1
|
+
import { colors, cssColors, detectEnvironment, escapeFormatString, formatDuration, getConsoleMethod, getCssLevelColor, getLevelColor, isClient, isDev, isLevelEnabled, matchesPattern } from "./utils.mjs";
|
|
2
|
+
//#region src/redact.ts
|
|
3
|
+
const DEFAULT_REPLACEMENT = "[REDACTED]";
|
|
4
|
+
/**
|
|
5
|
+
* Built-in PII detection patterns with smart masking.
|
|
6
|
+
* Each builtin preserves just enough signal for debugging while scrubbing PII.
|
|
7
|
+
*/
|
|
8
|
+
const builtinPatterns = {
|
|
9
|
+
creditCard: {
|
|
10
|
+
pattern: /\b\d{4}[\s-]?\d{4}[\s-]?\d{4}[\s-]?\d{4}\b/g,
|
|
11
|
+
mask: (m) => `****${m.replace(/[\s-]/g, "").slice(-4)}`
|
|
12
|
+
},
|
|
13
|
+
email: {
|
|
14
|
+
pattern: /[\w.+-]+@[\w-]+\.[\w.]+/g,
|
|
15
|
+
mask: (m) => {
|
|
16
|
+
if (m.indexOf("@") < 1) return "***@***";
|
|
17
|
+
const tld = m.slice(m.lastIndexOf("."));
|
|
18
|
+
return `${m[0]}***@***${tld}`;
|
|
19
|
+
}
|
|
20
|
+
},
|
|
21
|
+
ipv4: {
|
|
22
|
+
pattern: /\b(?!0\.0\.0\.0\b)(?!127\.0\.0\.1\b)\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}\b/g,
|
|
23
|
+
mask: (m) => `***.***.***.${m.split(".").pop()}`
|
|
24
|
+
},
|
|
25
|
+
phone: {
|
|
26
|
+
pattern: /(?:\+\d{1,3}[\s.-]?)?\(?\d{1,4}\)?[\s.-]?\d{2,4}[\s.-]?\d{2,4}[\s.-]?\d{2,4}\b/g,
|
|
27
|
+
mask: (m) => {
|
|
28
|
+
const digits = m.replace(/[^\d]/g, "");
|
|
29
|
+
if (m.startsWith("+") && digits.length > 4) {
|
|
30
|
+
const ccMatch = m.match(/^\+\d{1,3}/);
|
|
31
|
+
return `${ccMatch ? ccMatch[0] : "+"}******${digits.slice(-2)}`;
|
|
32
|
+
}
|
|
33
|
+
if (digits.length > 2) return `${"*".repeat(digits.length - 2)}${digits.slice(-2)}`;
|
|
34
|
+
return "***";
|
|
35
|
+
}
|
|
36
|
+
},
|
|
37
|
+
jwt: {
|
|
38
|
+
pattern: /\beyJ[\w-]*\.[\w-]*\.[\w-]*\b/g,
|
|
39
|
+
mask: () => "eyJ***.***"
|
|
40
|
+
},
|
|
41
|
+
bearer: {
|
|
42
|
+
pattern: /\bBearer\s+[\w\-.~+/]{8,}=*/gi,
|
|
43
|
+
mask: () => "Bearer ***"
|
|
44
|
+
},
|
|
45
|
+
iban: {
|
|
46
|
+
pattern: /\b[A-Z]{2}\d{2}[\s-]?[\dA-Z]{4}[\s-]?[\dA-Z]{4}[\s-]?[\dA-Z]{4}[\s-]?[\dA-Z]{0,4}[\s-]?[\dA-Z]{0,4}[\s-]?[\dA-Z]{0,4}\b/g,
|
|
47
|
+
mask: (m) => {
|
|
48
|
+
const clean = m.replace(/[\s-]/g, "");
|
|
49
|
+
return `${clean.slice(0, 4)}****${clean.slice(-3)}`;
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
};
|
|
53
|
+
/**
|
|
54
|
+
* Resolve a `redact` option (boolean or object) into a concrete `RedactConfig`.
|
|
55
|
+
*
|
|
56
|
+
* - `true` → all built-in patterns with smart masking, no custom paths
|
|
57
|
+
* - `{ ... }` → built-in maskers merged with user config (opt-out: `builtins: false`)
|
|
58
|
+
* - `false` / `undefined` → `undefined` (no redaction)
|
|
59
|
+
*/
|
|
60
|
+
function resolveRedactConfig(input) {
|
|
61
|
+
if (input === void 0 || input === false) return void 0;
|
|
62
|
+
if (input === true) return { _maskers: allBuiltinMaskers() };
|
|
63
|
+
if (input.builtins === false) return input;
|
|
64
|
+
const maskers = Array.isArray(input.builtins) ? input.builtins.map((name) => builtinPatterns[name]).filter(Boolean).map((b) => [cloneRegex(b.pattern), b.mask]) : allBuiltinMaskers();
|
|
65
|
+
return {
|
|
66
|
+
...input,
|
|
67
|
+
_maskers: maskers
|
|
68
|
+
};
|
|
69
|
+
}
|
|
70
|
+
function allBuiltinMaskers() {
|
|
71
|
+
return Object.values(builtinPatterns).map((b) => [cloneRegex(b.pattern), b.mask]);
|
|
72
|
+
}
|
|
73
|
+
function cloneRegex(re) {
|
|
74
|
+
return new RegExp(re.source, re.flags);
|
|
75
|
+
}
|
|
76
|
+
/**
|
|
77
|
+
* Redact sensitive data from a wide event in-place.
|
|
78
|
+
*
|
|
79
|
+
* Three strategies applied in order:
|
|
80
|
+
* 1. **Path-based**: dot-notation paths — the leaf value is replaced with `replacement`.
|
|
81
|
+
* 2. **Masker-based**: built-in patterns with smart partial masking (e.g. `****1111`).
|
|
82
|
+
* 3. **Pattern-based**: custom RegExp patterns replaced with `replacement`.
|
|
83
|
+
*
|
|
84
|
+
* @param event - The wide event object (mutated in-place).
|
|
85
|
+
* @param config - Redaction configuration.
|
|
86
|
+
*/
|
|
87
|
+
function redactEvent(event, config) {
|
|
88
|
+
const replacement = config.replacement ?? DEFAULT_REPLACEMENT;
|
|
89
|
+
if (config.paths?.length) for (const path of config.paths) redactPath(event, path.split("."), replacement);
|
|
90
|
+
if (config._maskers?.length) applyMaskersToTree(event, config._maskers);
|
|
91
|
+
if (config.patterns?.length) redactPatterns(event, config.patterns, replacement);
|
|
92
|
+
}
|
|
93
|
+
function redactPath(obj, segments, replacement) {
|
|
94
|
+
let current = obj;
|
|
95
|
+
for (let i = 0; i < segments.length - 1; i++) {
|
|
96
|
+
if (current === null || current === void 0 || typeof current !== "object") return;
|
|
97
|
+
current = current[segments[i]];
|
|
98
|
+
}
|
|
99
|
+
if (current === null || current === void 0 || typeof current !== "object") return;
|
|
100
|
+
const leaf = segments[segments.length - 1];
|
|
101
|
+
if (leaf in current) current[leaf] = replacement;
|
|
102
|
+
}
|
|
103
|
+
function redactPatterns(obj, patterns, replacement) {
|
|
104
|
+
if (obj === null || obj === void 0) return;
|
|
105
|
+
if (Array.isArray(obj)) {
|
|
106
|
+
for (let i = 0; i < obj.length; i++) if (typeof obj[i] === "string") obj[i] = applyPatterns(obj[i], patterns, replacement);
|
|
107
|
+
else if (typeof obj[i] === "object") redactPatterns(obj[i], patterns, replacement);
|
|
108
|
+
return;
|
|
109
|
+
}
|
|
110
|
+
if (typeof obj === "object") {
|
|
111
|
+
const record = obj;
|
|
112
|
+
for (const key in record) {
|
|
113
|
+
const val = record[key];
|
|
114
|
+
if (typeof val === "string") record[key] = applyPatterns(val, patterns, replacement);
|
|
115
|
+
else if (typeof val === "object") redactPatterns(val, patterns, replacement);
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
function applyPatterns(value, patterns, replacement) {
|
|
120
|
+
let result = value;
|
|
121
|
+
for (const pattern of patterns) {
|
|
122
|
+
pattern.lastIndex = 0;
|
|
123
|
+
result = result.replace(pattern, replacement);
|
|
124
|
+
}
|
|
125
|
+
return result;
|
|
126
|
+
}
|
|
127
|
+
function applyMaskersToTree(obj, maskers) {
|
|
128
|
+
if (obj === null || obj === void 0) return;
|
|
129
|
+
if (Array.isArray(obj)) {
|
|
130
|
+
for (let i = 0; i < obj.length; i++) if (typeof obj[i] === "string") obj[i] = applyMaskers(obj[i], maskers);
|
|
131
|
+
else if (typeof obj[i] === "object") applyMaskersToTree(obj[i], maskers);
|
|
132
|
+
return;
|
|
133
|
+
}
|
|
134
|
+
if (typeof obj === "object") {
|
|
135
|
+
const record = obj;
|
|
136
|
+
for (const key in record) {
|
|
137
|
+
const val = record[key];
|
|
138
|
+
if (typeof val === "string") record[key] = applyMaskers(val, maskers);
|
|
139
|
+
else if (typeof val === "object") applyMaskersToTree(val, maskers);
|
|
140
|
+
}
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
function applyMaskers(value, maskers) {
|
|
144
|
+
let result = value;
|
|
145
|
+
for (const [pattern, mask] of maskers) {
|
|
146
|
+
pattern.lastIndex = 0;
|
|
147
|
+
result = result.replace(pattern, mask);
|
|
148
|
+
}
|
|
149
|
+
return result;
|
|
150
|
+
}
|
|
151
|
+
/**
|
|
152
|
+
* Normalize a redact config that may have been deserialized from JSON
|
|
153
|
+
* (e.g. via `process.env.__EVLOG_CONFIG`). Converts pattern strings
|
|
154
|
+
* back to RegExp instances, then resolves built-in patterns.
|
|
155
|
+
*/
|
|
156
|
+
function normalizeRedactConfig(raw) {
|
|
157
|
+
if (raw === void 0 || raw === false) return void 0;
|
|
158
|
+
if (raw === true) return resolveRedactConfig(true);
|
|
159
|
+
const config = {};
|
|
160
|
+
if (Array.isArray(raw.paths)) config.paths = raw.paths;
|
|
161
|
+
if (typeof raw.replacement === "string") config.replacement = raw.replacement;
|
|
162
|
+
if (raw.builtins === false) config.builtins = false;
|
|
163
|
+
else if (Array.isArray(raw.builtins)) config.builtins = raw.builtins;
|
|
164
|
+
if (Array.isArray(raw.patterns)) config.patterns = raw.patterns.map((p) => {
|
|
165
|
+
if (p instanceof RegExp) return p;
|
|
166
|
+
if (typeof p === "string") return new RegExp(p, "g");
|
|
167
|
+
if (typeof p === "object" && p !== null) {
|
|
168
|
+
const obj = p;
|
|
169
|
+
return new RegExp(obj.source, obj.flags ?? "g");
|
|
170
|
+
}
|
|
171
|
+
return null;
|
|
172
|
+
}).filter((p) => p !== null);
|
|
173
|
+
return resolveRedactConfig(config);
|
|
174
|
+
}
|
|
175
|
+
//#endregion
|
|
176
|
+
//#region src/logger.ts
|
|
177
|
+
function isPlainObject(val) {
|
|
178
|
+
return val !== null && typeof val === "object" && !Array.isArray(val);
|
|
179
|
+
}
|
|
180
|
+
const _tsDate = /* @__PURE__ */ new Date();
|
|
181
|
+
function isoNow() {
|
|
182
|
+
_tsDate.setTime(Date.now());
|
|
183
|
+
return _tsDate.toISOString();
|
|
184
|
+
}
|
|
185
|
+
function mergeInto(target, source) {
|
|
186
|
+
for (const key in source) {
|
|
187
|
+
const sourceVal = source[key];
|
|
188
|
+
if (sourceVal === void 0 || sourceVal === null) continue;
|
|
189
|
+
const targetVal = target[key];
|
|
190
|
+
if (isPlainObject(sourceVal) && isPlainObject(targetVal)) mergeInto(targetVal, sourceVal);
|
|
191
|
+
else target[key] = sourceVal;
|
|
192
|
+
}
|
|
193
|
+
}
|
|
194
|
+
let globalEnv = {
|
|
195
|
+
service: "app",
|
|
196
|
+
environment: "development"
|
|
197
|
+
};
|
|
198
|
+
let globalPretty = isDev();
|
|
199
|
+
let globalSampling = {};
|
|
200
|
+
let globalStringify = true;
|
|
201
|
+
let globalDrain;
|
|
202
|
+
let globalRedact;
|
|
203
|
+
let globalEnabled = true;
|
|
204
|
+
let globalSilent = false;
|
|
205
|
+
/** Minimum level for the global `log` API only (`ownsEvent === false`). Default: all levels. */
|
|
206
|
+
let globalMinLevel = "debug";
|
|
207
|
+
let _locked = false;
|
|
208
|
+
/**
|
|
209
|
+
* Initialize the logger with configuration.
|
|
210
|
+
* Call this once at application startup.
|
|
211
|
+
*/
|
|
212
|
+
function initLogger(config = {}) {
|
|
213
|
+
globalEnabled = config.enabled ?? true;
|
|
214
|
+
const detected = detectEnvironment();
|
|
215
|
+
globalEnv = {
|
|
216
|
+
service: config.env?.service ?? detected.service ?? "app",
|
|
217
|
+
environment: config.env?.environment ?? detected.environment ?? "development",
|
|
218
|
+
version: config.env?.version ?? detected.version,
|
|
219
|
+
commitHash: config.env?.commitHash ?? detected.commitHash,
|
|
220
|
+
region: config.env?.region ?? detected.region
|
|
221
|
+
};
|
|
222
|
+
globalPretty = config.pretty ?? isDev();
|
|
223
|
+
globalSampling = config.sampling ?? {};
|
|
224
|
+
globalStringify = config.stringify ?? true;
|
|
225
|
+
globalDrain = config.drain;
|
|
226
|
+
globalRedact = resolveRedactConfig(config.redact ?? !isDev());
|
|
227
|
+
globalSilent = config.silent ?? false;
|
|
228
|
+
globalMinLevel = config.minLevel ?? "debug";
|
|
229
|
+
if (globalSilent && !globalDrain && !config._suppressDrainWarning) console.warn("[evlog] silent mode is enabled but no drain is configured. Events will be built and sampled but not output anywhere. Set a drain via initLogger({ drain }) or a framework hook (evlog:drain).");
|
|
230
|
+
}
|
|
231
|
+
/**
|
|
232
|
+
* Check if logging is globally enabled.
|
|
233
|
+
*/
|
|
234
|
+
function isEnabled() {
|
|
235
|
+
return globalEnabled;
|
|
236
|
+
}
|
|
237
|
+
/**
|
|
238
|
+
* @internal Lock the logger to prevent re-initialization.
|
|
239
|
+
* Called by instrumentation register() after setting up the logger with drain.
|
|
240
|
+
* Prevents configureHandler() from overwriting the drain config.
|
|
241
|
+
*/
|
|
242
|
+
function lockLogger() {
|
|
243
|
+
_locked = true;
|
|
244
|
+
}
|
|
245
|
+
/**
|
|
246
|
+
* @internal Check if the logger has been locked by instrumentation.
|
|
247
|
+
*/
|
|
248
|
+
function isLoggerLocked() {
|
|
249
|
+
return _locked;
|
|
250
|
+
}
|
|
251
|
+
/**
|
|
252
|
+
* @internal Get the globally configured drain callback.
|
|
253
|
+
* Used by framework middleware to fall back to the global drain
|
|
254
|
+
* when no middleware-level drain is provided.
|
|
255
|
+
*/
|
|
256
|
+
function getGlobalDrain() {
|
|
257
|
+
return globalDrain;
|
|
258
|
+
}
|
|
259
|
+
/**
|
|
260
|
+
* Determine if a log at the given level should be emitted based on sampling config.
|
|
261
|
+
* Error level defaults to 100% (always logged) unless explicitly configured otherwise.
|
|
262
|
+
*/
|
|
263
|
+
function shouldSample(level) {
|
|
264
|
+
const { rates } = globalSampling;
|
|
265
|
+
if (!rates) return true;
|
|
266
|
+
const percentage = level === "error" && rates.error === void 0 ? 100 : rates[level] ?? 100;
|
|
267
|
+
if (percentage <= 0) return false;
|
|
268
|
+
if (percentage >= 100) return true;
|
|
269
|
+
return Math.random() * 100 < percentage;
|
|
270
|
+
}
|
|
271
|
+
/**
|
|
272
|
+
* Evaluate tail sampling conditions to determine if a log should be force-kept.
|
|
273
|
+
* Returns true if ANY condition matches (OR logic).
|
|
274
|
+
*/
|
|
275
|
+
function shouldKeep(ctx) {
|
|
276
|
+
const { keep } = globalSampling;
|
|
277
|
+
if (!keep?.length) return false;
|
|
278
|
+
return keep.some((condition) => {
|
|
279
|
+
if (condition.status !== void 0 && ctx.status !== void 0 && ctx.status >= condition.status) return true;
|
|
280
|
+
if (condition.duration !== void 0 && ctx.duration !== void 0 && ctx.duration >= condition.duration) return true;
|
|
281
|
+
if (condition.path && ctx.path && matchesPattern(ctx.path, condition.path)) return true;
|
|
282
|
+
return false;
|
|
283
|
+
});
|
|
284
|
+
}
|
|
285
|
+
function emitWideEvent(level, event, deferDrain = false, ownsEvent = false) {
|
|
286
|
+
if (!globalEnabled) return null;
|
|
287
|
+
if (!ownsEvent) {
|
|
288
|
+
if (!isLevelEnabled(level, globalMinLevel)) return null;
|
|
289
|
+
if (!shouldSample(level)) return null;
|
|
290
|
+
}
|
|
291
|
+
let formatted;
|
|
292
|
+
if (ownsEvent) {
|
|
293
|
+
event.timestamp = isoNow();
|
|
294
|
+
event.level = level;
|
|
295
|
+
if (event.service === void 0) event.service = globalEnv.service;
|
|
296
|
+
if (event.environment === void 0) event.environment = globalEnv.environment;
|
|
297
|
+
if (globalEnv.version !== void 0 && event.version === void 0) event.version = globalEnv.version;
|
|
298
|
+
if (globalEnv.commitHash !== void 0 && event.commitHash === void 0) event.commitHash = globalEnv.commitHash;
|
|
299
|
+
if (globalEnv.region !== void 0 && event.region === void 0) event.region = globalEnv.region;
|
|
300
|
+
formatted = event;
|
|
301
|
+
} else formatted = {
|
|
302
|
+
timestamp: isoNow(),
|
|
303
|
+
level,
|
|
304
|
+
...globalEnv,
|
|
305
|
+
...event
|
|
306
|
+
};
|
|
307
|
+
if (globalRedact) redactEvent(formatted, globalRedact);
|
|
308
|
+
if (!globalSilent) if (globalPretty) prettyPrintWideEvent(formatted);
|
|
309
|
+
else if (globalStringify) console[getConsoleMethod(level)](JSON.stringify(formatted));
|
|
310
|
+
else console[getConsoleMethod(level)](formatted);
|
|
311
|
+
if (globalDrain && !deferDrain) Promise.resolve(globalDrain({ event: formatted })).catch((err) => {
|
|
312
|
+
console.error("[evlog] drain failed:", err);
|
|
313
|
+
});
|
|
314
|
+
return formatted;
|
|
315
|
+
}
|
|
316
|
+
function emitTaggedLog(level, tag, message) {
|
|
317
|
+
if (!globalEnabled) return;
|
|
318
|
+
if (globalPretty && !globalSilent) {
|
|
319
|
+
if (!isLevelEnabled(level, globalMinLevel)) return;
|
|
320
|
+
if (!shouldSample(level)) return;
|
|
321
|
+
if (isClient()) {
|
|
322
|
+
const levelColor = getCssLevelColor(level);
|
|
323
|
+
const timestamp = isoNow().slice(11, 23);
|
|
324
|
+
console.log(`%c${timestamp}%c %c[${escapeFormatString(tag)}]%c ${escapeFormatString(message)}`, cssColors.dim, cssColors.reset, levelColor, cssColors.reset);
|
|
325
|
+
} else {
|
|
326
|
+
const color = getLevelColor(level);
|
|
327
|
+
const timestamp = isoNow().slice(11, 23);
|
|
328
|
+
console.log(`${colors.dim}${timestamp}${colors.reset} ${color}[${tag}]${colors.reset} ${message}`);
|
|
329
|
+
}
|
|
330
|
+
return;
|
|
331
|
+
}
|
|
332
|
+
emitWideEvent(level, {
|
|
333
|
+
tag,
|
|
334
|
+
message
|
|
335
|
+
});
|
|
336
|
+
}
|
|
337
|
+
function formatValue(value) {
|
|
338
|
+
if (value === null || value === void 0) return String(value);
|
|
339
|
+
if (typeof value === "object") {
|
|
340
|
+
const pairs = [];
|
|
341
|
+
for (const [k, v] of Object.entries(value)) if (v !== void 0 && v !== null) if (typeof v === "object") pairs.push(`${k}=${JSON.stringify(v)}`);
|
|
342
|
+
else pairs.push(`${k}=${v}`);
|
|
343
|
+
return pairs.join(" ");
|
|
344
|
+
}
|
|
345
|
+
return String(value);
|
|
346
|
+
}
|
|
347
|
+
function formatCost(cost) {
|
|
348
|
+
if (cost < .01) return `$${cost.toFixed(6)}`;
|
|
349
|
+
if (cost < 1) return `$${cost.toFixed(4)}`;
|
|
350
|
+
return `$${cost.toFixed(2)}`;
|
|
351
|
+
}
|
|
352
|
+
function buildAIEntries(ai) {
|
|
353
|
+
const entries = [];
|
|
354
|
+
const headerParts = [];
|
|
355
|
+
if (ai.model) {
|
|
356
|
+
let m = String(ai.model);
|
|
357
|
+
if (ai.provider) m += ` (${ai.provider})`;
|
|
358
|
+
headerParts.push(m);
|
|
359
|
+
}
|
|
360
|
+
if (ai.calls) headerParts.push(`${ai.calls} call${ai.calls > 1 ? "s" : ""}`);
|
|
361
|
+
if (ai.steps && ai.steps > 1) headerParts.push(`${ai.steps} steps`);
|
|
362
|
+
entries.push({
|
|
363
|
+
key: "ai",
|
|
364
|
+
value: headerParts.join(" · ")
|
|
365
|
+
});
|
|
366
|
+
const inputTokens = ai.inputTokens;
|
|
367
|
+
const outputTokens = ai.outputTokens;
|
|
368
|
+
const totalTokens = ai.totalTokens;
|
|
369
|
+
if (inputTokens !== void 0 && outputTokens !== void 0) {
|
|
370
|
+
let tokLine = `${inputTokens} in → ${outputTokens} out`;
|
|
371
|
+
if (totalTokens) tokLine += ` (${totalTokens} total)`;
|
|
372
|
+
const extras = [];
|
|
373
|
+
if (ai.cacheReadTokens) extras.push(`${ai.cacheReadTokens} cache read`);
|
|
374
|
+
if (ai.cacheWriteTokens) extras.push(`${ai.cacheWriteTokens} cache write`);
|
|
375
|
+
if (ai.reasoningTokens) extras.push(`${ai.reasoningTokens} reasoning`);
|
|
376
|
+
if (extras.length) tokLine += ` · ${extras.join(" · ")}`;
|
|
377
|
+
entries.push({
|
|
378
|
+
key: "ai.tokens",
|
|
379
|
+
value: tokLine
|
|
380
|
+
});
|
|
381
|
+
}
|
|
382
|
+
const msFirst = ai.msToFirstChunk;
|
|
383
|
+
const msFinish = ai.msToFinish;
|
|
384
|
+
const tps = ai.tokensPerSecond;
|
|
385
|
+
if (msFirst !== void 0 || msFinish !== void 0) {
|
|
386
|
+
const parts = [];
|
|
387
|
+
if (msFirst !== void 0) parts.push(`${formatDuration(msFirst)} to first chunk`);
|
|
388
|
+
if (msFinish !== void 0) parts.push(`${formatDuration(msFinish)} total`);
|
|
389
|
+
let streamLine = parts.join(" → ");
|
|
390
|
+
if (tps) streamLine += ` · ${tps} tok/s`;
|
|
391
|
+
entries.push({
|
|
392
|
+
key: "ai.streaming",
|
|
393
|
+
value: streamLine
|
|
394
|
+
});
|
|
395
|
+
}
|
|
396
|
+
if (ai.estimatedCost !== void 0) entries.push({
|
|
397
|
+
key: "ai.cost",
|
|
398
|
+
value: formatCost(ai.estimatedCost)
|
|
399
|
+
});
|
|
400
|
+
if (ai.totalDurationMs !== void 0) entries.push({
|
|
401
|
+
key: "ai.totalDuration",
|
|
402
|
+
value: formatDuration(ai.totalDurationMs)
|
|
403
|
+
});
|
|
404
|
+
const toolCalls = ai.toolCalls;
|
|
405
|
+
const tools = ai.tools;
|
|
406
|
+
const hasInputs = toolCalls?.length ? typeof toolCalls[0] === "object" : false;
|
|
407
|
+
if (tools?.length) {
|
|
408
|
+
const children = tools.map((t, idx) => {
|
|
409
|
+
const mark = t.success ? "✓" : "✗";
|
|
410
|
+
let line = `${t.name} ${formatDuration(t.durationMs)} ${mark}`;
|
|
411
|
+
if (t.error) line += ` ${t.error}`;
|
|
412
|
+
if (hasInputs && toolCalls && idx < toolCalls.length) {
|
|
413
|
+
const tc = toolCalls[idx];
|
|
414
|
+
const inputStr = typeof tc.input === "string" ? tc.input : JSON.stringify(tc.input);
|
|
415
|
+
const truncated = inputStr.length > 100 ? `${inputStr.slice(0, 100)}…` : inputStr;
|
|
416
|
+
line += ` ${truncated}`;
|
|
417
|
+
}
|
|
418
|
+
return line;
|
|
419
|
+
});
|
|
420
|
+
entries.push({
|
|
421
|
+
key: "ai.tools",
|
|
422
|
+
value: "",
|
|
423
|
+
children
|
|
424
|
+
});
|
|
425
|
+
} else if (toolCalls?.length) if (hasInputs) {
|
|
426
|
+
const children = toolCalls.map((tc) => {
|
|
427
|
+
const inputStr = typeof tc.input === "string" ? tc.input : JSON.stringify(tc.input);
|
|
428
|
+
const truncated = inputStr.length > 100 ? `${inputStr.slice(0, 100)}…` : inputStr;
|
|
429
|
+
return `${tc.name}(${truncated})`;
|
|
430
|
+
});
|
|
431
|
+
entries.push({
|
|
432
|
+
key: "ai.tools",
|
|
433
|
+
value: "",
|
|
434
|
+
children
|
|
435
|
+
});
|
|
436
|
+
} else entries.push({
|
|
437
|
+
key: "ai.tools",
|
|
438
|
+
value: toolCalls.join(", ")
|
|
439
|
+
});
|
|
440
|
+
const stepsUsage = ai.stepsUsage;
|
|
441
|
+
if (stepsUsage?.length) {
|
|
442
|
+
const allSameModel = stepsUsage.every((s) => s.model === stepsUsage[0].model);
|
|
443
|
+
const children = stepsUsage.map((s) => {
|
|
444
|
+
let line = `${allSameModel ? "" : `${s.model} `}${s.inputTokens} in → ${s.outputTokens} out`;
|
|
445
|
+
const stepTools = s.toolCalls;
|
|
446
|
+
if (stepTools?.length) line += ` [${stepTools.join(", ")}]`;
|
|
447
|
+
return line;
|
|
448
|
+
});
|
|
449
|
+
entries.push({
|
|
450
|
+
key: "ai.steps",
|
|
451
|
+
value: "",
|
|
452
|
+
children
|
|
453
|
+
});
|
|
454
|
+
} else if (ai.steps && ai.steps > 1) entries.push({
|
|
455
|
+
key: "ai.steps",
|
|
456
|
+
value: String(ai.steps)
|
|
457
|
+
});
|
|
458
|
+
const embedding = ai.embedding;
|
|
459
|
+
if (embedding) {
|
|
460
|
+
const parts = [];
|
|
461
|
+
if (embedding.model) parts.push(String(embedding.model));
|
|
462
|
+
parts.push(`${embedding.tokens} tokens`);
|
|
463
|
+
if (embedding.dimensions) parts.push(`${embedding.dimensions}d`);
|
|
464
|
+
if (embedding.count) parts.push(`${embedding.count} items`);
|
|
465
|
+
entries.push({
|
|
466
|
+
key: "ai.embedding",
|
|
467
|
+
value: parts.join(" · ")
|
|
468
|
+
});
|
|
469
|
+
}
|
|
470
|
+
if (ai.finishReason) entries.push({
|
|
471
|
+
key: "ai.finishReason",
|
|
472
|
+
value: String(ai.finishReason)
|
|
473
|
+
});
|
|
474
|
+
if (ai.error) entries.push({
|
|
475
|
+
key: "ai.error",
|
|
476
|
+
value: String(ai.error)
|
|
477
|
+
});
|
|
478
|
+
if (ai.responseId) entries.push({
|
|
479
|
+
key: "ai.responseId",
|
|
480
|
+
value: String(ai.responseId)
|
|
481
|
+
});
|
|
482
|
+
return entries;
|
|
483
|
+
}
|
|
484
|
+
function prettyPrintWideEvent(event) {
|
|
485
|
+
const { timestamp, level, service, environment, version, ...rest } = event;
|
|
486
|
+
const ts = timestamp.slice(11, 23);
|
|
487
|
+
const browser = isClient();
|
|
488
|
+
const parts = [];
|
|
489
|
+
const styles = [];
|
|
490
|
+
if (browser) {
|
|
491
|
+
const lc = getCssLevelColor(level);
|
|
492
|
+
parts.push(`%c${ts}%c %c${level.toUpperCase()}%c %c[${escapeFormatString(String(service))}]%c`);
|
|
493
|
+
styles.push(cssColors.dim, cssColors.reset, lc, cssColors.reset, cssColors.cyan, cssColors.reset);
|
|
494
|
+
} else {
|
|
495
|
+
const lc = getLevelColor(level);
|
|
496
|
+
parts.push(`${colors.dim}${ts}${colors.reset} ${lc}${level.toUpperCase()}${colors.reset} ${colors.cyan}[${service}]${colors.reset}`);
|
|
497
|
+
}
|
|
498
|
+
if (rest.method && rest.path) {
|
|
499
|
+
parts.push(browser ? ` ${escapeFormatString(String(rest.method))} ${escapeFormatString(String(rest.path))}` : ` ${rest.method} ${rest.path}`);
|
|
500
|
+
delete rest.method;
|
|
501
|
+
delete rest.path;
|
|
502
|
+
}
|
|
503
|
+
if (rest.status) {
|
|
504
|
+
const sc = browser ? rest.status >= 400 ? cssColors.red : cssColors.green : rest.status >= 400 ? colors.red : colors.green;
|
|
505
|
+
if (browser) {
|
|
506
|
+
parts.push(` %c${rest.status}%c`);
|
|
507
|
+
styles.push(sc, cssColors.reset);
|
|
508
|
+
} else parts.push(` ${sc}${rest.status}${colors.reset}`);
|
|
509
|
+
delete rest.status;
|
|
510
|
+
}
|
|
511
|
+
if (rest.duration) {
|
|
512
|
+
if (browser) {
|
|
513
|
+
parts.push(` %c${escapeFormatString(`in ${rest.duration}`)}%c`);
|
|
514
|
+
styles.push(cssColors.dim, cssColors.reset);
|
|
515
|
+
} else parts.push(` ${colors.dim}in ${rest.duration}${colors.reset}`);
|
|
516
|
+
delete rest.duration;
|
|
517
|
+
}
|
|
518
|
+
console.log(parts.join(""), ...styles);
|
|
519
|
+
const aiData = rest.ai;
|
|
520
|
+
if (aiData && typeof aiData === "object") delete rest.ai;
|
|
521
|
+
const restEntries = Object.entries(rest).filter(([_, v]) => v !== void 0);
|
|
522
|
+
const aiEntries = aiData ? buildAIEntries(aiData) : [];
|
|
523
|
+
const allEntries = [...restEntries.map(([key, value]) => ({
|
|
524
|
+
key,
|
|
525
|
+
value: formatValue(value)
|
|
526
|
+
})), ...aiEntries];
|
|
527
|
+
for (let i = 0; i < allEntries.length; i++) {
|
|
528
|
+
const entry = allEntries[i];
|
|
529
|
+
const hasChildren = entry.children && entry.children.length > 0;
|
|
530
|
+
const prefix = i === allEntries.length - 1 && !hasChildren ? "└─" : "├─";
|
|
531
|
+
if (browser) {
|
|
532
|
+
const val = entry.value ? ` ${escapeFormatString(entry.value)}` : "";
|
|
533
|
+
console.log(` %c${prefix}%c %c${escapeFormatString(entry.key)}:%c${val}`, cssColors.dim, cssColors.reset, cssColors.cyan, cssColors.reset);
|
|
534
|
+
} else {
|
|
535
|
+
const val = entry.value ? ` ${entry.value}` : "";
|
|
536
|
+
console.log(` ${colors.dim}${prefix}${colors.reset} ${colors.cyan}${entry.key}:${colors.reset}${val}`);
|
|
537
|
+
}
|
|
538
|
+
if (hasChildren) {
|
|
539
|
+
const connector = i === allEntries.length - 1 ? " " : "│";
|
|
540
|
+
for (let j = 0; j < entry.children.length; j++) {
|
|
541
|
+
const child = entry.children[j];
|
|
542
|
+
const childPrefix = j === entry.children.length - 1 ? "└─" : "├─";
|
|
543
|
+
if (browser) console.log(` %c${connector} ${childPrefix}%c ${escapeFormatString(child)}`, cssColors.dim, cssColors.reset);
|
|
544
|
+
else console.log(` ${colors.dim}${connector} ${childPrefix}${colors.reset} ${child}`);
|
|
545
|
+
}
|
|
546
|
+
}
|
|
547
|
+
}
|
|
548
|
+
}
|
|
549
|
+
function createLogMethod(level) {
|
|
550
|
+
return function logMethod(tagOrEvent, message) {
|
|
551
|
+
if (typeof tagOrEvent === "string" && message !== void 0) emitTaggedLog(level, tagOrEvent, message);
|
|
552
|
+
else if (typeof tagOrEvent === "object") emitWideEvent(level, tagOrEvent);
|
|
553
|
+
else emitTaggedLog(level, "log", String(tagOrEvent));
|
|
554
|
+
};
|
|
555
|
+
}
|
|
556
|
+
/**
|
|
557
|
+
* Simple logging API - as easy as console.log
|
|
558
|
+
*
|
|
559
|
+
* @example
|
|
560
|
+
* ```ts
|
|
561
|
+
* log.info('auth', 'User logged in')
|
|
562
|
+
* log.error({ action: 'payment', error: 'failed' })
|
|
563
|
+
* ```
|
|
564
|
+
*/
|
|
565
|
+
const _log = {
|
|
566
|
+
info: createLogMethod("info"),
|
|
567
|
+
error: createLogMethod("error"),
|
|
568
|
+
warn: createLogMethod("warn"),
|
|
569
|
+
debug: createLogMethod("debug")
|
|
570
|
+
};
|
|
571
|
+
const noopLogger = {
|
|
572
|
+
set() {},
|
|
573
|
+
error() {},
|
|
574
|
+
info() {},
|
|
575
|
+
warn() {},
|
|
576
|
+
emit() {
|
|
577
|
+
return null;
|
|
578
|
+
},
|
|
579
|
+
getContext() {
|
|
580
|
+
return {};
|
|
581
|
+
}
|
|
582
|
+
};
|
|
583
|
+
/**
|
|
584
|
+
* Create a scoped logger for building wide events.
|
|
585
|
+
* Use this for any context: workflows, jobs, scripts, queues, etc.
|
|
586
|
+
*
|
|
587
|
+
* @example
|
|
588
|
+
* ```ts
|
|
589
|
+
* const log = createLogger({ jobId: job.id, queue: 'emails' })
|
|
590
|
+
* log.set({ batch: { size: 50, processed: 12 } })
|
|
591
|
+
* log.emit()
|
|
592
|
+
* ```
|
|
593
|
+
*/
|
|
594
|
+
function createLogger(initialContext = {}, internalOptions) {
|
|
595
|
+
if (!globalEnabled) return noopLogger;
|
|
596
|
+
const deferDrain = internalOptions?._deferDrain ?? false;
|
|
597
|
+
const startTime = Date.now();
|
|
598
|
+
const context = { ...initialContext };
|
|
599
|
+
let hasError = false;
|
|
600
|
+
let hasWarn = false;
|
|
601
|
+
function addLog(level, message) {
|
|
602
|
+
if (!Array.isArray(context.requestLogs)) context.requestLogs = [];
|
|
603
|
+
context.requestLogs.push({
|
|
604
|
+
level,
|
|
605
|
+
message,
|
|
606
|
+
timestamp: isoNow()
|
|
607
|
+
});
|
|
608
|
+
}
|
|
609
|
+
return {
|
|
610
|
+
set(data) {
|
|
611
|
+
mergeInto(context, data);
|
|
612
|
+
},
|
|
613
|
+
error(error, errorContext) {
|
|
614
|
+
hasError = true;
|
|
615
|
+
const err = typeof error === "string" ? new Error(error) : error;
|
|
616
|
+
if (errorContext) mergeInto(context, errorContext);
|
|
617
|
+
const errorObj = {
|
|
618
|
+
name: err.name,
|
|
619
|
+
message: err.message,
|
|
620
|
+
stack: err.stack
|
|
621
|
+
};
|
|
622
|
+
const errRecord = err;
|
|
623
|
+
for (const k of [
|
|
624
|
+
"status",
|
|
625
|
+
"statusText",
|
|
626
|
+
"statusCode",
|
|
627
|
+
"statusMessage",
|
|
628
|
+
"data",
|
|
629
|
+
"cause",
|
|
630
|
+
"internal"
|
|
631
|
+
]) if (k in err) errorObj[k] = errRecord[k];
|
|
632
|
+
if (isPlainObject(context.error)) mergeInto(context.error, errorObj);
|
|
633
|
+
else context.error = errorObj;
|
|
634
|
+
},
|
|
635
|
+
info(message, infoContext) {
|
|
636
|
+
addLog("info", message);
|
|
637
|
+
if (infoContext) {
|
|
638
|
+
const { requestLogs: _, ...rest } = infoContext;
|
|
639
|
+
mergeInto(context, rest);
|
|
640
|
+
}
|
|
641
|
+
},
|
|
642
|
+
warn(message, warnContext) {
|
|
643
|
+
hasWarn = true;
|
|
644
|
+
addLog("warn", message);
|
|
645
|
+
if (warnContext) {
|
|
646
|
+
const { requestLogs: _, ...rest } = warnContext;
|
|
647
|
+
mergeInto(context, rest);
|
|
648
|
+
}
|
|
649
|
+
},
|
|
650
|
+
emit(overrides) {
|
|
651
|
+
const durationMs = Date.now() - startTime;
|
|
652
|
+
const level = hasError ? "error" : hasWarn ? "warn" : "info";
|
|
653
|
+
let forceKeep = false;
|
|
654
|
+
if (overrides?._forceKeep) forceKeep = true;
|
|
655
|
+
else if (globalSampling.keep?.length) forceKeep = shouldKeep({
|
|
656
|
+
status: overrides?.status ?? context.status,
|
|
657
|
+
duration: durationMs,
|
|
658
|
+
path: context.path,
|
|
659
|
+
method: context.method,
|
|
660
|
+
context
|
|
661
|
+
});
|
|
662
|
+
if (!forceKeep && !shouldSample(level)) return null;
|
|
663
|
+
if (overrides) {
|
|
664
|
+
const obj = overrides;
|
|
665
|
+
for (const key in obj) if (key !== "_forceKeep") context[key] = obj[key];
|
|
666
|
+
}
|
|
667
|
+
context.duration = formatDuration(durationMs);
|
|
668
|
+
return emitWideEvent(level, context, deferDrain, true);
|
|
669
|
+
},
|
|
670
|
+
getContext() {
|
|
671
|
+
return { ...context };
|
|
672
|
+
}
|
|
673
|
+
};
|
|
674
|
+
}
|
|
675
|
+
/**
|
|
676
|
+
* Create a request-scoped logger for building wide events.
|
|
677
|
+
* Convenience wrapper around `createLogger` that pre-populates HTTP request fields.
|
|
678
|
+
*
|
|
679
|
+
* @example
|
|
680
|
+
* ```ts
|
|
681
|
+
* const log = createRequestLogger({ method: 'POST', path: '/checkout' })
|
|
682
|
+
* log.set({ user: { id: '123' } })
|
|
683
|
+
* log.set({ cart: { items: 3 } })
|
|
684
|
+
* log.emit()
|
|
685
|
+
* ```
|
|
686
|
+
*/
|
|
687
|
+
function createRequestLogger(options = {}, internalOptions) {
|
|
688
|
+
const initial = {};
|
|
689
|
+
if (options.method !== void 0) initial.method = options.method;
|
|
690
|
+
if (options.path !== void 0) initial.path = options.path;
|
|
691
|
+
if (options.requestId !== void 0) initial.requestId = options.requestId;
|
|
692
|
+
return createLogger(initial, internalOptions);
|
|
693
|
+
}
|
|
694
|
+
/**
|
|
695
|
+
* Get the current environment context.
|
|
696
|
+
*/
|
|
697
|
+
function getEnvironment() {
|
|
698
|
+
return { ...globalEnv };
|
|
699
|
+
}
|
|
700
|
+
if (typeof __EVLOG_CONFIG__ !== "undefined") initLogger(__EVLOG_CONFIG__);
|
|
701
|
+
//#endregion
|
|
702
|
+
export { getGlobalDrain as a, isLoggerLocked as c, normalizeRedactConfig as d, redactEvent as f, getEnvironment as i, lockLogger as l, createLogger as n, initLogger as o, resolveRedactConfig as p, createRequestLogger as r, isEnabled as s, _log as t, shouldKeep as u };
|
|
703
|
+
|
|
704
|
+
//# sourceMappingURL=logger-DY0X5oQd.mjs.map
|