autotel 4.1.0 → 4.2.1
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/auto.cjs +5 -3
- package/dist/auto.cjs.map +1 -1
- package/dist/auto.js +3 -3
- package/dist/auto.js.map +1 -1
- package/dist/chunk-C_NdSu1c.cjs +34 -0
- package/dist/correlation-id.cjs +1 -1
- package/dist/correlation-id.d.cts.map +1 -1
- package/dist/correlation-id.d.ts.map +1 -1
- package/dist/correlation-id.js +1 -1
- package/dist/decorators.cjs +1 -1
- package/dist/decorators.js +1 -1
- package/dist/{event-ByBTV9M2.js → event-531asIM6.js} +4 -4
- package/dist/{event-ByBTV9M2.js.map → event-531asIM6.js.map} +1 -1
- package/dist/{event-BhHREDJk.cjs → event-CcZYwp50.cjs} +4 -4
- package/dist/{event-BhHREDJk.cjs.map → event-CcZYwp50.cjs.map} +1 -1
- package/dist/event.cjs +1 -1
- package/dist/event.js +1 -1
- package/dist/{functional-zpzNLhky.cjs → functional-C8B0Qa7o.cjs} +10 -7
- package/dist/functional-C8B0Qa7o.cjs.map +1 -0
- package/dist/{functional-DtI0u4vx.js → functional-r-AUIRy_.js} +9 -9
- package/dist/functional-r-AUIRy_.js.map +1 -0
- package/dist/functional.cjs +1 -1
- package/dist/functional.js +1 -1
- package/dist/http.cjs +1 -1
- package/dist/http.js +1 -1
- package/dist/index.cjs +15 -13
- package/dist/index.cjs.map +1 -1
- package/dist/index.d.cts.map +1 -1
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +14 -14
- package/dist/index.js.map +1 -1
- package/dist/{init-D-jnNMix.js → init-BS2JVkrL.js} +2 -2
- package/dist/{init-D-jnNMix.js.map → init-BS2JVkrL.js.map} +1 -1
- package/dist/{init-BX7AmFRl.cjs → init-BXiuPK6j.cjs} +3 -3
- package/dist/{init-BX7AmFRl.cjs.map → init-BXiuPK6j.cjs.map} +1 -1
- package/dist/instrumentation.cjs +2 -2
- package/dist/instrumentation.js +2 -2
- package/dist/logger.cjs +236 -8
- package/dist/logger.cjs.map +1 -0
- package/dist/messaging.cjs +1 -1
- package/dist/messaging.js +1 -1
- package/dist/{node-require-DF5QBX6z.cjs → node-require-CZ_PU448.cjs} +6 -4
- package/dist/node-require-CZ_PU448.cjs.map +1 -0
- package/dist/{node-require-Db1oDpLj.js → node-require-vROmTeJ8.js} +5 -5
- package/dist/node-require-vROmTeJ8.js.map +1 -0
- package/dist/{operation-context-C-2hmmtP.js → operation-context-CKBoA4Qy.js} +3 -3
- package/dist/operation-context-CKBoA4Qy.js.map +1 -0
- package/dist/{operation-context-n4_obUwq.cjs → operation-context-D6LDf4W_.cjs} +3 -1
- package/dist/operation-context-D6LDf4W_.cjs.map +1 -0
- package/dist/register.cjs +3 -1
- package/dist/register.cjs.map +1 -1
- package/dist/register.js +2 -2
- package/dist/register.js.map +1 -1
- package/dist/semantic-helpers.cjs +1 -1
- package/dist/semantic-helpers.js +1 -1
- package/dist/{stable-hash-Cg5cT34Q.js → stable-hash-ChFBIhNt.js} +3 -3
- package/dist/stable-hash-ChFBIhNt.js.map +1 -0
- package/dist/{stable-hash-BNTMrmdB.cjs → stable-hash-brKISGf1.cjs} +4 -2
- package/dist/stable-hash-brKISGf1.cjs.map +1 -0
- package/dist/trace-context-Cijqoi6e.d.cts.map +1 -1
- package/dist/trace-context-Cijqoi6e.d.ts.map +1 -1
- package/dist/trace-helpers.cjs +1 -1
- package/dist/trace-helpers.js +1 -1
- package/dist/{track-wc0HafS_.js → track-COUuU48p.js} +5 -5
- package/dist/track-COUuU48p.js.map +1 -0
- package/dist/{track-D59FfpL0.cjs → track-Cb3Q4QmS.cjs} +4 -2
- package/dist/track-Cb3Q4QmS.cjs.map +1 -0
- package/dist/validate.cjs +1 -1
- package/dist/validate.js +1 -1
- package/dist/webhook.cjs +1 -1
- package/dist/webhook.js +1 -1
- package/dist/workflow-distributed.cjs +1 -1
- package/dist/workflow-distributed.js +1 -1
- package/dist/workflow.cjs +3 -1
- package/dist/workflow.cjs.map +1 -1
- package/dist/workflow.d.cts.map +1 -1
- package/dist/workflow.d.ts.map +1 -1
- package/dist/workflow.js +3 -3
- package/dist/workflow.js.map +1 -1
- package/dist/yaml-config.cjs +233 -4
- package/dist/yaml-config.cjs.map +1 -0
- package/dist/yaml-config.d.cts.map +1 -1
- package/dist/yaml-config.d.ts.map +1 -1
- package/dist/yaml-config.js +8 -7
- package/dist/yaml-config.js.map +1 -1
- package/package.json +1 -2
- package/dist/functional-DtI0u4vx.js.map +0 -1
- package/dist/functional-zpzNLhky.cjs.map +0 -1
- package/dist/logger-thMPLpOG.cjs +0 -487
- package/dist/logger-thMPLpOG.cjs.map +0 -1
- package/dist/node-require-DF5QBX6z.cjs.map +0 -1
- package/dist/node-require-Db1oDpLj.js.map +0 -1
- package/dist/operation-context-C-2hmmtP.js.map +0 -1
- package/dist/operation-context-n4_obUwq.cjs.map +0 -1
- package/dist/stable-hash-BNTMrmdB.cjs.map +0 -1
- package/dist/stable-hash-Cg5cT34Q.js.map +0 -1
- package/dist/track-D59FfpL0.cjs.map +0 -1
- package/dist/track-wc0HafS_.js.map +0 -1
- package/dist/yaml-config-Ck2uB0Dp.cjs +0 -273
- package/dist/yaml-config-Ck2uB0Dp.cjs.map +0 -1
- package/src/attribute-redacting-processor.test.ts +0 -763
- package/src/attribute-redacting-processor.ts +0 -621
- package/src/attributes/attachers.ts +0 -161
- package/src/attributes/builders.ts +0 -529
- package/src/attributes/domains.ts +0 -42
- package/src/attributes/index.ts +0 -81
- package/src/attributes/registry.ts +0 -323
- package/src/attributes/types.ts +0 -211
- package/src/attributes/utils.ts +0 -64
- package/src/attributes/validators.ts +0 -266
- package/src/attributes.test.ts +0 -292
- package/src/auto.ts +0 -67
- package/src/autotel-logger.test.ts +0 -548
- package/src/autotel-logger.ts +0 -364
- package/src/baggage-span-processor.test.ts +0 -202
- package/src/baggage-span-processor.ts +0 -100
- package/src/business-baggage.test.ts +0 -500
- package/src/business-baggage.ts +0 -669
- package/src/circuit-breaker.test.ts +0 -341
- package/src/circuit-breaker.ts +0 -184
- package/src/config.test.ts +0 -94
- package/src/config.ts +0 -172
- package/src/correlated-events.test.ts +0 -151
- package/src/correlated-events.ts +0 -47
- package/src/correlation-id.test.ts +0 -163
- package/src/correlation-id.ts +0 -206
- package/src/db.test.ts +0 -252
- package/src/db.ts +0 -447
- package/src/decorators.test.ts +0 -153
- package/src/decorators.ts +0 -188
- package/src/define-event.test.ts +0 -41
- package/src/define-event.ts +0 -58
- package/src/devtools.ts +0 -60
- package/src/drain-pipeline.test.ts +0 -68
- package/src/drain-pipeline.ts +0 -199
- package/src/drain-toolkit.test.ts +0 -113
- package/src/drain-toolkit.ts +0 -129
- package/src/enricher-toolkit.test.ts +0 -67
- package/src/enricher-toolkit.ts +0 -79
- package/src/enrichers.test.ts +0 -150
- package/src/enrichers.ts +0 -145
- package/src/env-config.test.ts +0 -323
- package/src/env-config.ts +0 -309
- package/src/error-catalog.test.ts +0 -133
- package/src/error-catalog.ts +0 -262
- package/src/event-queue.test.ts +0 -864
- package/src/event-queue.ts +0 -699
- package/src/event-subscriber.ts +0 -262
- package/src/event-testing.ts +0 -197
- package/src/event.test.ts +0 -1104
- package/src/event.ts +0 -988
- package/src/events-config.ts +0 -235
- package/src/exporters.ts +0 -165
- package/src/filtering-span-processor.test.ts +0 -281
- package/src/filtering-span-processor.ts +0 -111
- package/src/flatten-attributes.test.ts +0 -76
- package/src/flatten-attributes.ts +0 -80
- package/src/functional.strict-types.typecheck.ts +0 -53
- package/src/functional.test.ts +0 -1464
- package/src/functional.ts +0 -2539
- package/src/functional.types.test.ts +0 -135
- package/src/hook.mjs +0 -15
- package/src/http.test.ts +0 -485
- package/src/http.ts +0 -424
- package/src/index.ts +0 -433
- package/src/init-auto-redactor.test.ts +0 -53
- package/src/init-redactor.test.ts +0 -8
- package/src/init.customization.test.ts +0 -665
- package/src/init.integrations.test.ts +0 -399
- package/src/init.openllmetry.test.ts +0 -194
- package/src/init.protocol.test.ts +0 -215
- package/src/init.ts +0 -2439
- package/src/instrumentation.test.ts +0 -108
- package/src/instrumentation.ts +0 -319
- package/src/logger.test.ts +0 -125
- package/src/logger.ts +0 -341
- package/src/messaging-adapters.test.ts +0 -595
- package/src/messaging-adapters.ts +0 -583
- package/src/messaging-testing.test.ts +0 -573
- package/src/messaging-testing.ts +0 -935
- package/src/messaging.test.ts +0 -1646
- package/src/messaging.ts +0 -2245
- package/src/metric-helpers.ts +0 -47
- package/src/metric-testing.ts +0 -197
- package/src/metric.ts +0 -446
- package/src/metrics.test.ts +0 -241
- package/src/node-require.ts +0 -123
- package/src/operation-context.ts +0 -93
- package/src/parse-error.test.ts +0 -73
- package/src/parse-error.ts +0 -112
- package/src/posthog-logs.test.ts +0 -115
- package/src/posthog-logs.ts +0 -77
- package/src/pretty-console-exporter.test.ts +0 -545
- package/src/pretty-console-exporter.ts +0 -413
- package/src/pretty-log-formatter.test.ts +0 -123
- package/src/pretty-log-formatter.ts +0 -210
- package/src/processors/canonical-log-line-processor.test.ts +0 -523
- package/src/processors/canonical-log-line-processor.ts +0 -396
- package/src/processors.ts +0 -152
- package/src/rate-limiter.test.ts +0 -199
- package/src/rate-limiter.ts +0 -98
- package/src/redact-values.test.ts +0 -90
- package/src/redact-values.ts +0 -34
- package/src/register.ts +0 -37
- package/src/request-logger.test.ts +0 -545
- package/src/request-logger.ts +0 -342
- package/src/sampling.test.ts +0 -1060
- package/src/sampling.ts +0 -737
- package/src/security-schema.test.ts +0 -45
- package/src/security-schema.ts +0 -107
- package/src/semantic-conventions.ts +0 -15
- package/src/semantic-helpers.test.ts +0 -226
- package/src/semantic-helpers.ts +0 -438
- package/src/shutdown.test.ts +0 -364
- package/src/shutdown.ts +0 -246
- package/src/span-name-normalizer.test.ts +0 -377
- package/src/span-name-normalizer.ts +0 -213
- package/src/stable-hash.ts +0 -27
- package/src/structured-error.test.ts +0 -191
- package/src/structured-error.ts +0 -157
- package/src/stub.integration.test.ts +0 -361
- package/src/tail-sampling-processor.test.ts +0 -230
- package/src/tail-sampling-processor.ts +0 -55
- package/src/test-span-collector.test.ts +0 -234
- package/src/test-span-collector.ts +0 -150
- package/src/testing.ts +0 -705
- package/src/trace-context.test.ts +0 -73
- package/src/trace-context.ts +0 -567
- package/src/trace-helpers.new.test.ts +0 -278
- package/src/trace-helpers.test.ts +0 -290
- package/src/trace-helpers.ts +0 -710
- package/src/trace-hybrid.test.ts +0 -42
- package/src/trace-hybrid.ts +0 -37
- package/src/tracer-provider.test.ts +0 -183
- package/src/tracer-provider.ts +0 -266
- package/src/track.test.ts +0 -154
- package/src/track.ts +0 -216
- package/src/validate.test.ts +0 -287
- package/src/validate.ts +0 -307
- package/src/validation-attributes.ts +0 -43
- package/src/validation.test.ts +0 -330
- package/src/validation.ts +0 -246
- package/src/variable-name-inference.test.ts +0 -178
- package/src/variable-name-inference.ts +0 -242
- package/src/webhook.test.ts +0 -649
- package/src/webhook.ts +0 -637
- package/src/workflow-distributed.test.ts +0 -786
- package/src/workflow-distributed.ts +0 -916
- package/src/workflow.async-safety.integration.test.ts +0 -345
- package/src/workflow.test.ts +0 -647
- package/src/workflow.ts +0 -810
- package/src/yaml-config.test.ts +0 -373
- package/src/yaml-config.ts +0 -351
package/src/autotel-logger.ts
DELETED
|
@@ -1,364 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Zero-dependency structured logger for autotel
|
|
3
|
-
*
|
|
4
|
-
* This logger provides:
|
|
5
|
-
* - Structured JSON logging (production) or pretty print (development)
|
|
6
|
-
* - Auto trace context injection (traceId, spanId, correlationId)
|
|
7
|
-
* - Dynamic log level control (per-request via OTel context)
|
|
8
|
-
* - Level support (debug, info, warn, error, none)
|
|
9
|
-
* - Zero additional dependencies (uses @opentelemetry/api, already a dep)
|
|
10
|
-
*
|
|
11
|
-
* Uses Pino-compatible signature supporting both patterns:
|
|
12
|
-
* - `log.info('simple message')` - string-only
|
|
13
|
-
* - `log.info({ userId: '123' }, 'message')` - object first with optional message
|
|
14
|
-
*
|
|
15
|
-
* Used as the default fallback when users don't provide Pino/Bunyan.
|
|
16
|
-
* Can also be used directly: import { createBuiltinLogger } from 'autotel/logger'
|
|
17
|
-
*
|
|
18
|
-
* @example
|
|
19
|
-
* ```typescript
|
|
20
|
-
* import { createBuiltinLogger, runWithLogLevel } from 'autotel/logger';
|
|
21
|
-
*
|
|
22
|
-
* const log = createBuiltinLogger('my-service');
|
|
23
|
-
*
|
|
24
|
-
* // Simple message (no metadata)
|
|
25
|
-
* log.info('Server started');
|
|
26
|
-
*
|
|
27
|
-
* // With metadata (Pino-style: object first, message second)
|
|
28
|
-
* log.info({ userId: '123' }, 'User created');
|
|
29
|
-
* // Output: {"level":"info","service":"my-service","msg":"User created","userId":"123","traceId":"..."}
|
|
30
|
-
*
|
|
31
|
-
* // Dynamic log level per-request
|
|
32
|
-
* runWithLogLevel('debug', () => {
|
|
33
|
-
* log.debug('This will log even if default level is "info"');
|
|
34
|
-
* });
|
|
35
|
-
* ```
|
|
36
|
-
*/
|
|
37
|
-
|
|
38
|
-
import {
|
|
39
|
-
trace,
|
|
40
|
-
context as api_context,
|
|
41
|
-
createContextKey,
|
|
42
|
-
} from '@opentelemetry/api';
|
|
43
|
-
import type { Logger } from './logger';
|
|
44
|
-
|
|
45
|
-
export type BuiltinLogLevel = 'debug' | 'info' | 'warn' | 'error' | 'none';
|
|
46
|
-
|
|
47
|
-
/**
|
|
48
|
-
* Context key for storing active log level (enables per-request log levels)
|
|
49
|
-
*/
|
|
50
|
-
const LOG_LEVEL_KEY = createContextKey('autotel-log-level');
|
|
51
|
-
|
|
52
|
-
/**
|
|
53
|
-
* Get the active log level from context (if set)
|
|
54
|
-
* Falls back to undefined if no log level is set in context
|
|
55
|
-
*/
|
|
56
|
-
export function getActiveLogLevel(): BuiltinLogLevel | undefined {
|
|
57
|
-
return api_context.active().getValue(LOG_LEVEL_KEY) as
|
|
58
|
-
| BuiltinLogLevel
|
|
59
|
-
| undefined;
|
|
60
|
-
}
|
|
61
|
-
|
|
62
|
-
/**
|
|
63
|
-
* Run a function with a specific log level
|
|
64
|
-
* The log level is stored in OpenTelemetry context and applies to all logger calls within the callback
|
|
65
|
-
*
|
|
66
|
-
* @example
|
|
67
|
-
* ```typescript
|
|
68
|
-
* // Enable debug logging for a specific request
|
|
69
|
-
* runWithLogLevel('debug', () => {
|
|
70
|
-
* log.debug('This will be logged');
|
|
71
|
-
* processRequest();
|
|
72
|
-
* });
|
|
73
|
-
*
|
|
74
|
-
* // Disable logging temporarily
|
|
75
|
-
* runWithLogLevel('none', () => {
|
|
76
|
-
* log.info('This will NOT be logged');
|
|
77
|
-
* });
|
|
78
|
-
* ```
|
|
79
|
-
*/
|
|
80
|
-
export function runWithLogLevel<T>(
|
|
81
|
-
level: BuiltinLogLevel,
|
|
82
|
-
callback: () => T,
|
|
83
|
-
): T {
|
|
84
|
-
const ctx = api_context.active().setValue(LOG_LEVEL_KEY, level);
|
|
85
|
-
return api_context.with(ctx, callback);
|
|
86
|
-
}
|
|
87
|
-
|
|
88
|
-
/**
|
|
89
|
-
* Get current trace context from active span
|
|
90
|
-
* Returns null if no active span exists
|
|
91
|
-
*/
|
|
92
|
-
function getTraceContextInternal(): {
|
|
93
|
-
traceId: string;
|
|
94
|
-
spanId: string;
|
|
95
|
-
correlationId: string;
|
|
96
|
-
} | null {
|
|
97
|
-
const span = trace.getActiveSpan();
|
|
98
|
-
if (!span) return null;
|
|
99
|
-
|
|
100
|
-
const ctx = span.spanContext();
|
|
101
|
-
return {
|
|
102
|
-
traceId: ctx.traceId,
|
|
103
|
-
spanId: ctx.spanId,
|
|
104
|
-
correlationId: ctx.traceId.slice(0, 16), // First 16 chars for grouping
|
|
105
|
-
};
|
|
106
|
-
}
|
|
107
|
-
|
|
108
|
-
/**
|
|
109
|
-
* Helper to get trace context (useful for BYOL - Bring Your Own Logger)
|
|
110
|
-
*
|
|
111
|
-
* @example
|
|
112
|
-
* ```typescript
|
|
113
|
-
* import bunyan from 'bunyan';
|
|
114
|
-
* import { getTraceContext } from 'autotel/logger';
|
|
115
|
-
*
|
|
116
|
-
* const bunyanLogger = bunyan.createLogger({ name: 'myapp' });
|
|
117
|
-
* const ctx = getTraceContext();
|
|
118
|
-
* bunyanLogger.info({ ...ctx, email: 'test@example.com' }, 'Creating user');
|
|
119
|
-
* ```
|
|
120
|
-
*/
|
|
121
|
-
export function getTraceContext() {
|
|
122
|
-
return getTraceContextInternal();
|
|
123
|
-
}
|
|
124
|
-
|
|
125
|
-
export interface BuiltinLoggerOptions {
|
|
126
|
-
/** Minimum log level. Default: 'info' */
|
|
127
|
-
level?: BuiltinLogLevel;
|
|
128
|
-
/** Pretty print for development. Default: false (JSON output) */
|
|
129
|
-
pretty?: boolean;
|
|
130
|
-
}
|
|
131
|
-
|
|
132
|
-
/**
|
|
133
|
-
* Create a lightweight structured logger
|
|
134
|
-
*
|
|
135
|
-
* @param service - Service name for logging
|
|
136
|
-
* @param options - Optional configuration
|
|
137
|
-
*
|
|
138
|
-
* @example
|
|
139
|
-
* ```typescript
|
|
140
|
-
* const log = createBuiltinLogger('user-service');
|
|
141
|
-
*
|
|
142
|
-
* log.info('Creating user', { email: 'test@example.com' });
|
|
143
|
-
* // Output: {"level":"info","service":"user-service","msg":"Creating user",
|
|
144
|
-
* // "email":"test@example.com","traceId":"...","spanId":"..."}
|
|
145
|
-
*
|
|
146
|
-
* // Dynamic log level control per-request
|
|
147
|
-
* runWithLogLevel('debug', () => {
|
|
148
|
-
* log.debug('This will be logged even if logger was created with level: "info"');
|
|
149
|
-
* });
|
|
150
|
-
* ```
|
|
151
|
-
*/
|
|
152
|
-
export function createBuiltinLogger(
|
|
153
|
-
service: string,
|
|
154
|
-
options?: BuiltinLoggerOptions,
|
|
155
|
-
): Logger {
|
|
156
|
-
const defaultLevel = options?.level || 'info';
|
|
157
|
-
const pretty = options?.pretty || false;
|
|
158
|
-
|
|
159
|
-
const levelPriority: Record<BuiltinLogLevel, number> = {
|
|
160
|
-
none: -1,
|
|
161
|
-
debug: 0,
|
|
162
|
-
info: 1,
|
|
163
|
-
warn: 2,
|
|
164
|
-
error: 3,
|
|
165
|
-
};
|
|
166
|
-
|
|
167
|
-
const shouldLog = (level: BuiltinLogLevel): boolean => {
|
|
168
|
-
// Priority: context level > options level > 'info' default
|
|
169
|
-
const activeLevel = getActiveLogLevel() ?? defaultLevel;
|
|
170
|
-
|
|
171
|
-
// 'none' means suppress all logging
|
|
172
|
-
if (activeLevel === 'none') return false;
|
|
173
|
-
|
|
174
|
-
return levelPriority[level] >= levelPriority[activeLevel];
|
|
175
|
-
};
|
|
176
|
-
|
|
177
|
-
const log = (
|
|
178
|
-
level: 'info' | 'error' | 'warn' | 'debug',
|
|
179
|
-
msg: string,
|
|
180
|
-
attrs?: Record<string, unknown>,
|
|
181
|
-
) => {
|
|
182
|
-
if (!shouldLog(level)) return;
|
|
183
|
-
|
|
184
|
-
const ctx = getTraceContextInternal();
|
|
185
|
-
const logEntry: Record<string, unknown> = {
|
|
186
|
-
level,
|
|
187
|
-
service,
|
|
188
|
-
msg,
|
|
189
|
-
...attrs,
|
|
190
|
-
...ctx, // Auto-inject traceId, spanId, correlationId
|
|
191
|
-
timestamp: new Date().toISOString(),
|
|
192
|
-
};
|
|
193
|
-
|
|
194
|
-
if (pretty) {
|
|
195
|
-
// Pretty print for development
|
|
196
|
-
const traceInfo = ctx
|
|
197
|
-
? ` [${ctx.traceId.slice(0, 8)}.../${ctx.spanId.slice(0, 8)}...]`
|
|
198
|
-
: '';
|
|
199
|
-
console.log(
|
|
200
|
-
`[${level.toUpperCase()}]${traceInfo} ${service}: ${msg}`,
|
|
201
|
-
attrs || '',
|
|
202
|
-
);
|
|
203
|
-
} else {
|
|
204
|
-
// Structured JSON for production
|
|
205
|
-
console.log(JSON.stringify(logEntry));
|
|
206
|
-
}
|
|
207
|
-
};
|
|
208
|
-
|
|
209
|
-
// Pino-compatible signature: supports both:
|
|
210
|
-
// - logger.info('message') - string-only
|
|
211
|
-
// - logger.info({ extra }, 'message') - Pino style with metadata
|
|
212
|
-
// Also auto-detects and handles legacy Winston-style: logger.info('message', { extra })
|
|
213
|
-
const createLogMethod = (level: 'info' | 'warn' | 'debug') => {
|
|
214
|
-
return (
|
|
215
|
-
extraOrMessage: Record<string, unknown> | string,
|
|
216
|
-
message?: string | Record<string, unknown>,
|
|
217
|
-
) => {
|
|
218
|
-
if (typeof extraOrMessage === 'string') {
|
|
219
|
-
// First arg is string - could be:
|
|
220
|
-
// 1. String-only: logger.info('message')
|
|
221
|
-
// 2. Legacy Winston-style: logger.info('message', { extra })
|
|
222
|
-
if (message !== undefined && typeof message === 'object') {
|
|
223
|
-
// Legacy Winston-style detected - auto-swap for backward compatibility
|
|
224
|
-
if (process.env.NODE_ENV !== 'production') {
|
|
225
|
-
console.warn(
|
|
226
|
-
`[autotel] Legacy logger pattern detected: logger.${level}('message', metadata). ` +
|
|
227
|
-
`Autotel recommends Pino signature: logger.${level}({ ...metadata }, 'message'). ` +
|
|
228
|
-
`Auto-swapping arguments for compatibility.`,
|
|
229
|
-
);
|
|
230
|
-
}
|
|
231
|
-
// Swap: treat first arg as message, second as metadata
|
|
232
|
-
log(level, extraOrMessage, message as Record<string, unknown>);
|
|
233
|
-
} else {
|
|
234
|
-
// Pure string-only call: logger.info('message')
|
|
235
|
-
log(level, extraOrMessage);
|
|
236
|
-
}
|
|
237
|
-
} else {
|
|
238
|
-
// Pino style: logger.info({ extra }, 'message')
|
|
239
|
-
log(level, (message as string) || '', extraOrMessage);
|
|
240
|
-
}
|
|
241
|
-
};
|
|
242
|
-
};
|
|
243
|
-
|
|
244
|
-
return {
|
|
245
|
-
info: createLogMethod('info'),
|
|
246
|
-
warn: createLogMethod('warn'),
|
|
247
|
-
debug: createLogMethod('debug'),
|
|
248
|
-
|
|
249
|
-
error: (
|
|
250
|
-
extraOrMessage: Record<string, unknown> | string,
|
|
251
|
-
message?: string | Record<string, unknown> | Error,
|
|
252
|
-
) => {
|
|
253
|
-
if (typeof extraOrMessage === 'string') {
|
|
254
|
-
// First arg is string - could be:
|
|
255
|
-
// 1. String-only: logger.error('message')
|
|
256
|
-
// 2. Legacy: logger.error('message', error) - Error as second arg
|
|
257
|
-
// 3. Legacy: logger.error('message', { extra }) - object as second arg
|
|
258
|
-
|
|
259
|
-
// Handle legacy logger.error('message', error) pattern
|
|
260
|
-
if (message instanceof Error) {
|
|
261
|
-
if (process.env.NODE_ENV !== 'production') {
|
|
262
|
-
console.warn(
|
|
263
|
-
`[autotel] Legacy logger pattern detected: logger.error('message', error). ` +
|
|
264
|
-
`Autotel recommends Pino signature: logger.error({ err: error }, 'message'). ` +
|
|
265
|
-
`Auto-swapping arguments for compatibility.`,
|
|
266
|
-
);
|
|
267
|
-
}
|
|
268
|
-
log('error', extraOrMessage, {
|
|
269
|
-
error: message.message,
|
|
270
|
-
stack: message.stack,
|
|
271
|
-
name: message.name,
|
|
272
|
-
});
|
|
273
|
-
return;
|
|
274
|
-
}
|
|
275
|
-
|
|
276
|
-
// Handle legacy logger.error('message', { extra }) pattern
|
|
277
|
-
if (message !== undefined && typeof message === 'object') {
|
|
278
|
-
if (process.env.NODE_ENV !== 'production') {
|
|
279
|
-
console.warn(
|
|
280
|
-
`[autotel] Legacy logger pattern detected: logger.error('message', metadata). ` +
|
|
281
|
-
`Autotel recommends Pino signature: logger.error({ ...metadata }, 'message'). ` +
|
|
282
|
-
`Auto-swapping arguments for compatibility.`,
|
|
283
|
-
);
|
|
284
|
-
}
|
|
285
|
-
// Swap: treat first arg as message, second as metadata (handle err extraction)
|
|
286
|
-
const extra = message as Record<string, unknown>;
|
|
287
|
-
const { err, ...rest } = extra as Record<string, unknown> & {
|
|
288
|
-
err?: unknown;
|
|
289
|
-
};
|
|
290
|
-
let errorAttrs: Record<string, unknown> = rest;
|
|
291
|
-
if (err instanceof Error) {
|
|
292
|
-
errorAttrs = {
|
|
293
|
-
error: err.message,
|
|
294
|
-
stack: err.stack,
|
|
295
|
-
name: err.name,
|
|
296
|
-
...rest,
|
|
297
|
-
};
|
|
298
|
-
} else if (err !== undefined) {
|
|
299
|
-
errorAttrs = { err, ...rest };
|
|
300
|
-
}
|
|
301
|
-
log('error', extraOrMessage, errorAttrs);
|
|
302
|
-
return;
|
|
303
|
-
}
|
|
304
|
-
|
|
305
|
-
// Pure string-only call: logger.error('message')
|
|
306
|
-
log('error', extraOrMessage);
|
|
307
|
-
return;
|
|
308
|
-
}
|
|
309
|
-
|
|
310
|
-
// Pino style: logger.error({ err, ...extra }, 'message')
|
|
311
|
-
// Extract err from extra if present (Pino convention)
|
|
312
|
-
const { err, ...rest } = extraOrMessage as Record<string, unknown> & {
|
|
313
|
-
err?: unknown;
|
|
314
|
-
};
|
|
315
|
-
let errorAttrs: Record<string, unknown> = rest;
|
|
316
|
-
if (err instanceof Error) {
|
|
317
|
-
// err is an Error - extract message, stack, name for structured logging
|
|
318
|
-
errorAttrs = {
|
|
319
|
-
error: err.message,
|
|
320
|
-
stack: err.stack,
|
|
321
|
-
name: err.name,
|
|
322
|
-
...rest,
|
|
323
|
-
};
|
|
324
|
-
} else if (err !== undefined) {
|
|
325
|
-
// err is not an Error but exists - preserve it as-is
|
|
326
|
-
errorAttrs = {
|
|
327
|
-
err,
|
|
328
|
-
...rest,
|
|
329
|
-
};
|
|
330
|
-
}
|
|
331
|
-
log('error', (message as string) || '', errorAttrs);
|
|
332
|
-
},
|
|
333
|
-
};
|
|
334
|
-
}
|
|
335
|
-
|
|
336
|
-
/**
|
|
337
|
-
* Pino-like factory function for creating an autotel logger
|
|
338
|
-
*
|
|
339
|
-
* @example
|
|
340
|
-
* ```typescript
|
|
341
|
-
* import { autotelLogger } from 'autotel/logger';
|
|
342
|
-
*
|
|
343
|
-
* const log = autotelLogger();
|
|
344
|
-
*
|
|
345
|
-
* // Simple message
|
|
346
|
-
* log.info('Server started');
|
|
347
|
-
*
|
|
348
|
-
* // With metadata (Pino-style: object first, message second)
|
|
349
|
-
* log.info({ userId: '123' }, 'User created');
|
|
350
|
-
*
|
|
351
|
-
* // With options
|
|
352
|
-
* const devLog = autotelLogger({ service: 'my-app', level: 'debug', pretty: true });
|
|
353
|
-
* ```
|
|
354
|
-
*/
|
|
355
|
-
export function autotelLogger(options?: {
|
|
356
|
-
service?: string;
|
|
357
|
-
level?: BuiltinLogLevel;
|
|
358
|
-
pretty?: boolean;
|
|
359
|
-
}): Logger {
|
|
360
|
-
return createBuiltinLogger(options?.service || 'app', {
|
|
361
|
-
level: options?.level,
|
|
362
|
-
pretty: options?.pretty,
|
|
363
|
-
});
|
|
364
|
-
}
|
|
@@ -1,202 +0,0 @@
|
|
|
1
|
-
import { describe, it, expect, beforeEach } from 'vitest';
|
|
2
|
-
import { BaggageSpanProcessor } from './baggage-span-processor';
|
|
3
|
-
import { context, propagation, trace } from '@opentelemetry/api';
|
|
4
|
-
import { BasicTracerProvider } from '@opentelemetry/sdk-trace-base';
|
|
5
|
-
import {
|
|
6
|
-
InMemorySpanExporter,
|
|
7
|
-
SimpleSpanProcessor,
|
|
8
|
-
} from '@opentelemetry/sdk-trace-base';
|
|
9
|
-
import { AsyncLocalStorageContextManager } from '@opentelemetry/context-async-hooks';
|
|
10
|
-
|
|
11
|
-
// Set up context manager globally
|
|
12
|
-
const contextManager = new AsyncLocalStorageContextManager();
|
|
13
|
-
contextManager.enable();
|
|
14
|
-
context.setGlobalContextManager(contextManager);
|
|
15
|
-
|
|
16
|
-
describe('BaggageSpanProcessor', () => {
|
|
17
|
-
beforeEach(() => {
|
|
18
|
-
// Set up context manager to ensure context propagation works
|
|
19
|
-
trace.setGlobalTracerProvider(undefined as any);
|
|
20
|
-
});
|
|
21
|
-
|
|
22
|
-
it('should copy baggage entries to span attributes with default prefix', async () => {
|
|
23
|
-
const exporter = new InMemorySpanExporter();
|
|
24
|
-
const baggageProcessor = new BaggageSpanProcessor();
|
|
25
|
-
const provider = new BasicTracerProvider({
|
|
26
|
-
spanProcessors: [baggageProcessor, new SimpleSpanProcessor(exporter)],
|
|
27
|
-
});
|
|
28
|
-
|
|
29
|
-
// Set as global provider
|
|
30
|
-
trace.setGlobalTracerProvider(provider);
|
|
31
|
-
|
|
32
|
-
const tracer = provider.getTracer('test');
|
|
33
|
-
|
|
34
|
-
// Set baggage
|
|
35
|
-
const activeContext = context.active();
|
|
36
|
-
let baggage = propagation.createBaggage();
|
|
37
|
-
baggage = baggage.setEntry('tenant.id', { value: 'tenant-123' });
|
|
38
|
-
baggage = baggage.setEntry('user.id', { value: 'user-456' });
|
|
39
|
-
const contextWithBaggage = propagation.setBaggage(activeContext, baggage);
|
|
40
|
-
|
|
41
|
-
// Create span within baggage context
|
|
42
|
-
await new Promise<void>((resolve) => {
|
|
43
|
-
context.with(contextWithBaggage, () => {
|
|
44
|
-
tracer.startActiveSpan('test-span', (span) => {
|
|
45
|
-
span.end();
|
|
46
|
-
resolve();
|
|
47
|
-
});
|
|
48
|
-
});
|
|
49
|
-
});
|
|
50
|
-
|
|
51
|
-
// Flush to ensure spans are exported
|
|
52
|
-
await provider.forceFlush();
|
|
53
|
-
|
|
54
|
-
const spans = exporter.getFinishedSpans();
|
|
55
|
-
expect(spans).toHaveLength(1);
|
|
56
|
-
expect(spans[0]!.attributes['baggage.tenant.id']).toBe('tenant-123');
|
|
57
|
-
expect(spans[0]!.attributes['baggage.user.id']).toBe('user-456');
|
|
58
|
-
});
|
|
59
|
-
|
|
60
|
-
it('should use custom prefix when provided', async () => {
|
|
61
|
-
const exporter = new InMemorySpanExporter();
|
|
62
|
-
const baggageProcessor = new BaggageSpanProcessor({ prefix: 'ctx.' });
|
|
63
|
-
const provider = new BasicTracerProvider({
|
|
64
|
-
spanProcessors: [baggageProcessor, new SimpleSpanProcessor(exporter)],
|
|
65
|
-
});
|
|
66
|
-
|
|
67
|
-
// Set as global provider
|
|
68
|
-
trace.setGlobalTracerProvider(provider);
|
|
69
|
-
|
|
70
|
-
const tracer = provider.getTracer('test');
|
|
71
|
-
|
|
72
|
-
// Set baggage
|
|
73
|
-
const activeContext = context.active();
|
|
74
|
-
let baggage = propagation.createBaggage();
|
|
75
|
-
baggage = baggage.setEntry('tenant.id', { value: 'tenant-123' });
|
|
76
|
-
const contextWithBaggage = propagation.setBaggage(activeContext, baggage);
|
|
77
|
-
|
|
78
|
-
// Create span within baggage context
|
|
79
|
-
await new Promise<void>((resolve) => {
|
|
80
|
-
context.with(contextWithBaggage, () => {
|
|
81
|
-
tracer.startActiveSpan('test-span', (span) => {
|
|
82
|
-
span.end();
|
|
83
|
-
resolve();
|
|
84
|
-
});
|
|
85
|
-
});
|
|
86
|
-
});
|
|
87
|
-
|
|
88
|
-
// Flush to ensure spans are exported
|
|
89
|
-
await provider.forceFlush();
|
|
90
|
-
|
|
91
|
-
const spans = exporter.getFinishedSpans();
|
|
92
|
-
expect(spans).toHaveLength(1);
|
|
93
|
-
expect(spans[0]!.attributes['ctx.tenant.id']).toBe('tenant-123');
|
|
94
|
-
});
|
|
95
|
-
|
|
96
|
-
it('should use no prefix when empty string provided', async () => {
|
|
97
|
-
const exporter = new InMemorySpanExporter();
|
|
98
|
-
const baggageProcessor = new BaggageSpanProcessor({ prefix: '' });
|
|
99
|
-
const provider = new BasicTracerProvider({
|
|
100
|
-
spanProcessors: [baggageProcessor, new SimpleSpanProcessor(exporter)],
|
|
101
|
-
});
|
|
102
|
-
|
|
103
|
-
// Set as global provider
|
|
104
|
-
trace.setGlobalTracerProvider(provider);
|
|
105
|
-
|
|
106
|
-
const tracer = provider.getTracer('test');
|
|
107
|
-
|
|
108
|
-
// Set baggage
|
|
109
|
-
const activeContext = context.active();
|
|
110
|
-
let baggage = propagation.createBaggage();
|
|
111
|
-
baggage = baggage.setEntry('tenant.id', { value: 'tenant-123' });
|
|
112
|
-
const contextWithBaggage = propagation.setBaggage(activeContext, baggage);
|
|
113
|
-
|
|
114
|
-
// Create span within baggage context
|
|
115
|
-
await new Promise<void>((resolve) => {
|
|
116
|
-
context.with(contextWithBaggage, () => {
|
|
117
|
-
tracer.startActiveSpan('test-span', (span) => {
|
|
118
|
-
span.end();
|
|
119
|
-
resolve();
|
|
120
|
-
});
|
|
121
|
-
});
|
|
122
|
-
});
|
|
123
|
-
|
|
124
|
-
// Flush to ensure spans are exported
|
|
125
|
-
await provider.forceFlush();
|
|
126
|
-
|
|
127
|
-
const spans = exporter.getFinishedSpans();
|
|
128
|
-
expect(spans).toHaveLength(1);
|
|
129
|
-
expect(spans[0]!.attributes['tenant.id']).toBe('tenant-123');
|
|
130
|
-
});
|
|
131
|
-
|
|
132
|
-
it('should handle spans with no baggage gracefully', async () => {
|
|
133
|
-
const exporter = new InMemorySpanExporter();
|
|
134
|
-
const baggageProcessor = new BaggageSpanProcessor();
|
|
135
|
-
const provider = new BasicTracerProvider({
|
|
136
|
-
spanProcessors: [baggageProcessor, new SimpleSpanProcessor(exporter)],
|
|
137
|
-
});
|
|
138
|
-
|
|
139
|
-
// Set as global provider
|
|
140
|
-
trace.setGlobalTracerProvider(provider);
|
|
141
|
-
|
|
142
|
-
const tracer = provider.getTracer('test');
|
|
143
|
-
|
|
144
|
-
// Create span without baggage
|
|
145
|
-
await new Promise<void>((resolve) => {
|
|
146
|
-
tracer.startActiveSpan('test-span', (span) => {
|
|
147
|
-
span.end();
|
|
148
|
-
resolve();
|
|
149
|
-
});
|
|
150
|
-
});
|
|
151
|
-
|
|
152
|
-
// Flush to ensure spans are exported
|
|
153
|
-
await provider.forceFlush();
|
|
154
|
-
|
|
155
|
-
const spans = exporter.getFinishedSpans();
|
|
156
|
-
expect(spans).toHaveLength(1);
|
|
157
|
-
// Should have no baggage attributes
|
|
158
|
-
expect(
|
|
159
|
-
Object.keys(spans[0]!.attributes).filter((k) => k.startsWith('baggage.')),
|
|
160
|
-
).toHaveLength(0);
|
|
161
|
-
});
|
|
162
|
-
|
|
163
|
-
it('should copy multiple baggage entries', async () => {
|
|
164
|
-
const exporter = new InMemorySpanExporter();
|
|
165
|
-
const baggageProcessor = new BaggageSpanProcessor();
|
|
166
|
-
const provider = new BasicTracerProvider({
|
|
167
|
-
spanProcessors: [baggageProcessor, new SimpleSpanProcessor(exporter)],
|
|
168
|
-
});
|
|
169
|
-
|
|
170
|
-
// Set as global provider
|
|
171
|
-
trace.setGlobalTracerProvider(provider);
|
|
172
|
-
|
|
173
|
-
const tracer = provider.getTracer('test');
|
|
174
|
-
|
|
175
|
-
// Set multiple baggage entries
|
|
176
|
-
const activeContext = context.active();
|
|
177
|
-
let baggage = propagation.createBaggage();
|
|
178
|
-
baggage = baggage.setEntry('key1', { value: 'value1' });
|
|
179
|
-
baggage = baggage.setEntry('key2', { value: 'value2' });
|
|
180
|
-
baggage = baggage.setEntry('key3', { value: 'value3' });
|
|
181
|
-
const contextWithBaggage = propagation.setBaggage(activeContext, baggage);
|
|
182
|
-
|
|
183
|
-
// Create span within baggage context
|
|
184
|
-
await new Promise<void>((resolve) => {
|
|
185
|
-
context.with(contextWithBaggage, () => {
|
|
186
|
-
tracer.startActiveSpan('test-span', (span) => {
|
|
187
|
-
span.end();
|
|
188
|
-
resolve();
|
|
189
|
-
});
|
|
190
|
-
});
|
|
191
|
-
});
|
|
192
|
-
|
|
193
|
-
// Flush to ensure spans are exported
|
|
194
|
-
await provider.forceFlush();
|
|
195
|
-
|
|
196
|
-
const spans = exporter.getFinishedSpans();
|
|
197
|
-
expect(spans).toHaveLength(1);
|
|
198
|
-
expect(spans[0]!.attributes['baggage.key1']).toBe('value1');
|
|
199
|
-
expect(spans[0]!.attributes['baggage.key2']).toBe('value2');
|
|
200
|
-
expect(spans[0]!.attributes['baggage.key3']).toBe('value3');
|
|
201
|
-
});
|
|
202
|
-
});
|
|
@@ -1,100 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Span processor that copies baggage entries to span attributes
|
|
3
|
-
*
|
|
4
|
-
* This makes baggage visible in trace UIs without manual attribute setting.
|
|
5
|
-
* Enabled via init({ baggage: true }) or init({ baggage: 'custom-prefix' })
|
|
6
|
-
*/
|
|
7
|
-
|
|
8
|
-
import type { Span, Context } from '@opentelemetry/api';
|
|
9
|
-
import { propagation, context as otelContext } from '@opentelemetry/api';
|
|
10
|
-
import type { SpanProcessor } from '@opentelemetry/sdk-trace-base';
|
|
11
|
-
import type { ReadableSpan } from '@opentelemetry/sdk-trace-base';
|
|
12
|
-
import { requireModule } from './node-require';
|
|
13
|
-
|
|
14
|
-
export interface BaggageSpanProcessorOptions {
|
|
15
|
-
/**
|
|
16
|
-
* Prefix for baggage attributes
|
|
17
|
-
* @default 'baggage.'
|
|
18
|
-
*/
|
|
19
|
-
prefix?: string;
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
/**
|
|
23
|
-
* Span processor that automatically copies baggage entries to span attributes
|
|
24
|
-
*
|
|
25
|
-
* This makes baggage visible in trace UIs (Jaeger, Grafana, DataDog, etc.)
|
|
26
|
-
* without manually calling ctx.setAttribute() for each baggage entry.
|
|
27
|
-
*
|
|
28
|
-
* @example Enable in init()
|
|
29
|
-
* ```typescript
|
|
30
|
-
* init({
|
|
31
|
-
* service: 'my-app',
|
|
32
|
-
* baggage: true // Uses default 'baggage.' prefix
|
|
33
|
-
* });
|
|
34
|
-
*
|
|
35
|
-
* // Now baggage automatically appears as span attributes
|
|
36
|
-
* await withBaggage({
|
|
37
|
-
* baggage: { 'tenant.id': 't1', 'user.id': 'u1' },
|
|
38
|
-
* fn: async () => {
|
|
39
|
-
* // Span has baggage.tenant.id and baggage.user.id attributes!
|
|
40
|
-
* }
|
|
41
|
-
* });
|
|
42
|
-
* ```
|
|
43
|
-
*
|
|
44
|
-
* @example Custom prefix
|
|
45
|
-
* ```typescript
|
|
46
|
-
* init({
|
|
47
|
-
* service: 'my-app',
|
|
48
|
-
* baggage: 'ctx' // Uses 'ctx.' prefix
|
|
49
|
-
* });
|
|
50
|
-
* // Creates attributes: ctx.tenant.id, ctx.user.id
|
|
51
|
-
* ```
|
|
52
|
-
*/
|
|
53
|
-
export class BaggageSpanProcessor implements SpanProcessor {
|
|
54
|
-
private readonly prefix: string;
|
|
55
|
-
|
|
56
|
-
constructor(options: BaggageSpanProcessorOptions = {}) {
|
|
57
|
-
this.prefix = options.prefix ?? 'baggage.';
|
|
58
|
-
}
|
|
59
|
-
|
|
60
|
-
onStart(span: Span, parentContext: Context): void {
|
|
61
|
-
// Read baggage from parentContext first (spans created with explicit context)
|
|
62
|
-
// Then fall back to active context (spans created without explicit context)
|
|
63
|
-
// Also check getActiveContextWithBaggage() to see baggage set via ctx.setBaggage()
|
|
64
|
-
let baggage = propagation.getBaggage(parentContext);
|
|
65
|
-
if (!baggage) {
|
|
66
|
-
baggage = propagation.getBaggage(otelContext.active());
|
|
67
|
-
}
|
|
68
|
-
// Check stored context from ctx.setBaggage() if still no baggage
|
|
69
|
-
if (!baggage) {
|
|
70
|
-
try {
|
|
71
|
-
const { getActiveContextWithBaggage } = requireModule<{
|
|
72
|
-
getActiveContextWithBaggage: () => Context;
|
|
73
|
-
}>('./trace-context');
|
|
74
|
-
const storedContext = getActiveContextWithBaggage();
|
|
75
|
-
baggage = propagation.getBaggage(storedContext);
|
|
76
|
-
} catch {
|
|
77
|
-
// Fallback if trace-context isn't available
|
|
78
|
-
}
|
|
79
|
-
}
|
|
80
|
-
if (!baggage) return;
|
|
81
|
-
|
|
82
|
-
// Copy all baggage entries to span attributes
|
|
83
|
-
for (const [key, entry] of baggage.getAllEntries()) {
|
|
84
|
-
span.setAttribute(`${this.prefix}${key}`, entry.value);
|
|
85
|
-
}
|
|
86
|
-
}
|
|
87
|
-
|
|
88
|
-
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
89
|
-
onEnd(_span: ReadableSpan): void {
|
|
90
|
-
// No-op - required by SpanProcessor interface
|
|
91
|
-
}
|
|
92
|
-
|
|
93
|
-
async shutdown(): Promise<void> {
|
|
94
|
-
// No-op
|
|
95
|
-
}
|
|
96
|
-
|
|
97
|
-
async forceFlush(): Promise<void> {
|
|
98
|
-
// No-op
|
|
99
|
-
}
|
|
100
|
-
}
|