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
|
@@ -1,413 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Pretty Console Exporter
|
|
3
|
-
*
|
|
4
|
-
* A developer-friendly span exporter that displays colorized, hierarchical
|
|
5
|
-
* trace output in the terminal. Zero external dependencies - uses ANSI escape codes.
|
|
6
|
-
*
|
|
7
|
-
* @example Basic usage
|
|
8
|
-
* ```typescript
|
|
9
|
-
* init({
|
|
10
|
-
* service: 'my-app',
|
|
11
|
-
* debug: 'pretty' // Uses PrettyConsoleExporter
|
|
12
|
-
* })
|
|
13
|
-
* ```
|
|
14
|
-
*
|
|
15
|
-
* @example Explicit usage with options
|
|
16
|
-
* ```typescript
|
|
17
|
-
* import { PrettyConsoleExporter } from 'autotel/exporters'
|
|
18
|
-
*
|
|
19
|
-
* init({
|
|
20
|
-
* service: 'my-app',
|
|
21
|
-
* spanExporters: [new PrettyConsoleExporter({
|
|
22
|
-
* colors: true,
|
|
23
|
-
* showAttributes: true,
|
|
24
|
-
* hideAttributes: ['http.user_agent']
|
|
25
|
-
* })]
|
|
26
|
-
* })
|
|
27
|
-
* ```
|
|
28
|
-
*/
|
|
29
|
-
|
|
30
|
-
import type { SpanExporter, ReadableSpan } from '@opentelemetry/sdk-trace-base';
|
|
31
|
-
import { SpanStatusCode } from '@opentelemetry/api';
|
|
32
|
-
|
|
33
|
-
/**
|
|
34
|
-
* Export result code constants (avoid importing @opentelemetry/core)
|
|
35
|
-
*/
|
|
36
|
-
const ExportResultCode = {
|
|
37
|
-
SUCCESS: 0,
|
|
38
|
-
FAILED: 1,
|
|
39
|
-
} as const;
|
|
40
|
-
|
|
41
|
-
/**
|
|
42
|
-
* Export result type for SpanExporter callback
|
|
43
|
-
*/
|
|
44
|
-
interface ExportResult {
|
|
45
|
-
code: number;
|
|
46
|
-
error?: Error;
|
|
47
|
-
}
|
|
48
|
-
|
|
49
|
-
/**
|
|
50
|
-
* ANSI escape codes for terminal colors (zero dependencies)
|
|
51
|
-
*/
|
|
52
|
-
const ANSI = {
|
|
53
|
-
reset: '\u001B[0m',
|
|
54
|
-
bold: '\u001B[1m',
|
|
55
|
-
dim: '\u001B[2m',
|
|
56
|
-
green: '\u001B[32m',
|
|
57
|
-
red: '\u001B[31m',
|
|
58
|
-
yellow: '\u001B[33m',
|
|
59
|
-
blue: '\u001B[34m',
|
|
60
|
-
cyan: '\u001B[36m',
|
|
61
|
-
gray: '\u001B[90m',
|
|
62
|
-
} as const;
|
|
63
|
-
|
|
64
|
-
type AnsiColor = keyof typeof ANSI;
|
|
65
|
-
|
|
66
|
-
/**
|
|
67
|
-
* Configuration options for PrettyConsoleExporter
|
|
68
|
-
*/
|
|
69
|
-
export interface PrettyConsoleExporterOptions {
|
|
70
|
-
/**
|
|
71
|
-
* Enable ANSI colors in output
|
|
72
|
-
* @default auto-detect TTY
|
|
73
|
-
*/
|
|
74
|
-
colors?: boolean;
|
|
75
|
-
|
|
76
|
-
/**
|
|
77
|
-
* Show span attributes in output
|
|
78
|
-
* @default true
|
|
79
|
-
*/
|
|
80
|
-
showAttributes?: boolean;
|
|
81
|
-
|
|
82
|
-
/**
|
|
83
|
-
* Maximum length for attribute values before truncation
|
|
84
|
-
* @default 50
|
|
85
|
-
*/
|
|
86
|
-
maxValueLength?: number;
|
|
87
|
-
|
|
88
|
-
/**
|
|
89
|
-
* Show instrumentation scope name (e.g., [http], [pg])
|
|
90
|
-
* @default true
|
|
91
|
-
*/
|
|
92
|
-
showScope?: boolean;
|
|
93
|
-
|
|
94
|
-
/**
|
|
95
|
-
* Attribute keys to always hide from output
|
|
96
|
-
* @default []
|
|
97
|
-
*/
|
|
98
|
-
hideAttributes?: string[];
|
|
99
|
-
|
|
100
|
-
/**
|
|
101
|
-
* Show trace ID for each root span
|
|
102
|
-
* @default false
|
|
103
|
-
*/
|
|
104
|
-
showTraceId?: boolean;
|
|
105
|
-
}
|
|
106
|
-
|
|
107
|
-
/**
|
|
108
|
-
* Internal node structure for building span trees
|
|
109
|
-
*/
|
|
110
|
-
interface SpanNode {
|
|
111
|
-
span: ReadableSpan;
|
|
112
|
-
children: SpanNode[];
|
|
113
|
-
}
|
|
114
|
-
|
|
115
|
-
/**
|
|
116
|
-
* Pretty Console Exporter - colorized, hierarchical span output for development
|
|
117
|
-
*
|
|
118
|
-
* Features:
|
|
119
|
-
* - Colorized status indicators (✓ green, ✗ red)
|
|
120
|
-
* - Duration with color coding (fast=green, medium=yellow, slow=red)
|
|
121
|
-
* - Hierarchical tree view showing parent-child relationships
|
|
122
|
-
* - Attribute display with truncation
|
|
123
|
-
* - Error message highlighting
|
|
124
|
-
*/
|
|
125
|
-
export class PrettyConsoleExporter implements SpanExporter {
|
|
126
|
-
private readonly options: Required<PrettyConsoleExporterOptions>;
|
|
127
|
-
|
|
128
|
-
constructor(options: PrettyConsoleExporterOptions = {}) {
|
|
129
|
-
this.options = {
|
|
130
|
-
colors: options.colors ?? process.stdout?.isTTY ?? false,
|
|
131
|
-
showAttributes: options.showAttributes ?? true,
|
|
132
|
-
maxValueLength: options.maxValueLength ?? 50,
|
|
133
|
-
showScope: options.showScope ?? true,
|
|
134
|
-
hideAttributes: options.hideAttributes ?? [],
|
|
135
|
-
showTraceId: options.showTraceId ?? false,
|
|
136
|
-
};
|
|
137
|
-
}
|
|
138
|
-
|
|
139
|
-
/**
|
|
140
|
-
* Export spans with pretty formatting
|
|
141
|
-
*/
|
|
142
|
-
export(
|
|
143
|
-
spans: ReadableSpan[],
|
|
144
|
-
resultCallback: (result: ExportResult) => void,
|
|
145
|
-
): void {
|
|
146
|
-
if (spans.length === 0) {
|
|
147
|
-
resultCallback({ code: ExportResultCode.SUCCESS });
|
|
148
|
-
return;
|
|
149
|
-
}
|
|
150
|
-
|
|
151
|
-
try {
|
|
152
|
-
// Group spans by trace ID
|
|
153
|
-
const traceGroups = this.groupByTrace(spans);
|
|
154
|
-
|
|
155
|
-
// Print each trace group
|
|
156
|
-
for (const [traceId, traceSpans] of traceGroups) {
|
|
157
|
-
this.printTrace(traceId, traceSpans);
|
|
158
|
-
}
|
|
159
|
-
|
|
160
|
-
resultCallback({ code: ExportResultCode.SUCCESS });
|
|
161
|
-
} catch {
|
|
162
|
-
// Fail-open: don't crash the app if formatting fails
|
|
163
|
-
resultCallback({ code: ExportResultCode.SUCCESS });
|
|
164
|
-
}
|
|
165
|
-
}
|
|
166
|
-
|
|
167
|
-
/**
|
|
168
|
-
* Group spans by their trace ID
|
|
169
|
-
*/
|
|
170
|
-
private groupByTrace(spans: ReadableSpan[]): Map<string, ReadableSpan[]> {
|
|
171
|
-
const groups = new Map<string, ReadableSpan[]>();
|
|
172
|
-
|
|
173
|
-
for (const span of spans) {
|
|
174
|
-
const traceId = span.spanContext().traceId;
|
|
175
|
-
const group = groups.get(traceId) ?? [];
|
|
176
|
-
group.push(span);
|
|
177
|
-
groups.set(traceId, group);
|
|
178
|
-
}
|
|
179
|
-
|
|
180
|
-
return groups;
|
|
181
|
-
}
|
|
182
|
-
|
|
183
|
-
/**
|
|
184
|
-
* Print a single trace with all its spans as a tree
|
|
185
|
-
*/
|
|
186
|
-
private printTrace(traceId: string, spans: ReadableSpan[]): void {
|
|
187
|
-
// Sort by start time
|
|
188
|
-
const sorted = [...spans].toSorted((a, b) => {
|
|
189
|
-
const aTime = hrTimeToMs(a.startTime);
|
|
190
|
-
const bTime = hrTimeToMs(b.startTime);
|
|
191
|
-
return aTime - bTime;
|
|
192
|
-
});
|
|
193
|
-
|
|
194
|
-
// Build tree structure
|
|
195
|
-
const tree = this.buildSpanTree(sorted);
|
|
196
|
-
|
|
197
|
-
// Print trace ID header if enabled
|
|
198
|
-
if (this.options.showTraceId && tree.length > 0) {
|
|
199
|
-
console.log(this.color(`trace: ${traceId}`, 'gray'));
|
|
200
|
-
}
|
|
201
|
-
|
|
202
|
-
// Print each root span and its children
|
|
203
|
-
for (const node of tree) {
|
|
204
|
-
this.printNode(node, 0, false);
|
|
205
|
-
}
|
|
206
|
-
|
|
207
|
-
// Add blank line between traces
|
|
208
|
-
console.log('');
|
|
209
|
-
}
|
|
210
|
-
|
|
211
|
-
/**
|
|
212
|
-
* Build a tree structure from flat spans using parent-child relationships
|
|
213
|
-
*/
|
|
214
|
-
private buildSpanTree(spans: ReadableSpan[]): SpanNode[] {
|
|
215
|
-
const spanMap = new Map<string, SpanNode>();
|
|
216
|
-
const roots: SpanNode[] = [];
|
|
217
|
-
|
|
218
|
-
// Create nodes for all spans
|
|
219
|
-
for (const span of spans) {
|
|
220
|
-
const spanId = span.spanContext().spanId;
|
|
221
|
-
spanMap.set(spanId, { span, children: [] });
|
|
222
|
-
}
|
|
223
|
-
|
|
224
|
-
// Build parent-child relationships
|
|
225
|
-
for (const span of spans) {
|
|
226
|
-
const spanId = span.spanContext().spanId;
|
|
227
|
-
const parentId = span.parentSpanContext?.spanId;
|
|
228
|
-
const node = spanMap.get(spanId)!;
|
|
229
|
-
|
|
230
|
-
if (parentId && spanMap.has(parentId)) {
|
|
231
|
-
// Has parent in this batch - add as child
|
|
232
|
-
spanMap.get(parentId)!.children.push(node);
|
|
233
|
-
} else {
|
|
234
|
-
// No parent or parent not in batch - treat as root
|
|
235
|
-
roots.push(node);
|
|
236
|
-
}
|
|
237
|
-
}
|
|
238
|
-
|
|
239
|
-
return roots;
|
|
240
|
-
}
|
|
241
|
-
|
|
242
|
-
/**
|
|
243
|
-
* Print a span node with indentation and tree characters
|
|
244
|
-
*/
|
|
245
|
-
private printNode(node: SpanNode, depth: number, isLast: boolean): void {
|
|
246
|
-
const { span } = node;
|
|
247
|
-
|
|
248
|
-
// Build tree prefix
|
|
249
|
-
const prefix =
|
|
250
|
-
depth === 0 ? '' : ' '.repeat(depth - 1) + (isLast ? '└─ ' : '├─ ');
|
|
251
|
-
|
|
252
|
-
// Status indicator
|
|
253
|
-
const isError = span.status.code === SpanStatusCode.ERROR;
|
|
254
|
-
const statusChar = isError ? '✗' : '✓';
|
|
255
|
-
const statusColor: AnsiColor = isError ? 'red' : 'green';
|
|
256
|
-
|
|
257
|
-
// Duration formatting
|
|
258
|
-
const durationMs = hrTimeToMs(span.duration);
|
|
259
|
-
const durationStr = formatDuration(durationMs);
|
|
260
|
-
const durationColor = getDurationColor(durationMs);
|
|
261
|
-
|
|
262
|
-
// Scope name (instrumentation library)
|
|
263
|
-
const scopeName = this.options.showScope
|
|
264
|
-
? this.color(` [${this.getScopeName(span)}]`, 'gray')
|
|
265
|
-
: '';
|
|
266
|
-
|
|
267
|
-
// Build the main line
|
|
268
|
-
const line = [
|
|
269
|
-
prefix,
|
|
270
|
-
this.color(statusChar, statusColor),
|
|
271
|
-
' ',
|
|
272
|
-
span.name.padEnd(Math.max(35 - prefix.length, 10)),
|
|
273
|
-
this.color(durationStr.padStart(8), durationColor),
|
|
274
|
-
scopeName,
|
|
275
|
-
].join('');
|
|
276
|
-
|
|
277
|
-
console.log(line);
|
|
278
|
-
|
|
279
|
-
// Print attributes on next line (indented)
|
|
280
|
-
if (this.options.showAttributes) {
|
|
281
|
-
const attrs = this.formatAttributes(span);
|
|
282
|
-
if (attrs) {
|
|
283
|
-
const attrIndent = ' '.repeat(depth) + ' ';
|
|
284
|
-
console.log(this.color(`${attrIndent}${attrs}`, 'dim'));
|
|
285
|
-
}
|
|
286
|
-
}
|
|
287
|
-
|
|
288
|
-
// Print error message if present
|
|
289
|
-
if (isError && span.status.message) {
|
|
290
|
-
const errorIndent = ' '.repeat(depth) + ' ';
|
|
291
|
-
console.log(
|
|
292
|
-
this.color(`${errorIndent}Error: ${span.status.message}`, 'red'),
|
|
293
|
-
);
|
|
294
|
-
}
|
|
295
|
-
|
|
296
|
-
// Print children
|
|
297
|
-
const childCount = node.children.length;
|
|
298
|
-
let index = 0;
|
|
299
|
-
for (const child of node.children) {
|
|
300
|
-
this.printNode(child, depth + 1, index === childCount - 1);
|
|
301
|
-
index++;
|
|
302
|
-
}
|
|
303
|
-
}
|
|
304
|
-
|
|
305
|
-
/**
|
|
306
|
-
* Get short scope name from instrumentation scope
|
|
307
|
-
*/
|
|
308
|
-
private getScopeName(span: ReadableSpan): string {
|
|
309
|
-
const name = span.instrumentationScope?.name ?? 'unknown';
|
|
310
|
-
// Extract short name from @opentelemetry/instrumentation-xxx format
|
|
311
|
-
const match = name.match(/@opentelemetry\/instrumentation-(.+)/);
|
|
312
|
-
if (match?.[1]) return match[1];
|
|
313
|
-
// Fall back to last part of name or full name
|
|
314
|
-
const lastPart = name.split('/').at(-1);
|
|
315
|
-
return lastPart ?? name;
|
|
316
|
-
}
|
|
317
|
-
|
|
318
|
-
/**
|
|
319
|
-
* Format span attributes as a comma-separated string
|
|
320
|
-
*/
|
|
321
|
-
private formatAttributes(span: ReadableSpan): string {
|
|
322
|
-
const attrs = span.attributes;
|
|
323
|
-
if (!attrs || Object.keys(attrs).length === 0) {
|
|
324
|
-
return '';
|
|
325
|
-
}
|
|
326
|
-
|
|
327
|
-
const pairs: string[] = [];
|
|
328
|
-
for (const [key, value] of Object.entries(attrs)) {
|
|
329
|
-
// Skip hidden attributes
|
|
330
|
-
if (this.options.hideAttributes.includes(key)) continue;
|
|
331
|
-
|
|
332
|
-
// Skip undefined/null values
|
|
333
|
-
if (value === undefined || value === null) continue;
|
|
334
|
-
|
|
335
|
-
// Format value
|
|
336
|
-
const strValue = this.truncate(
|
|
337
|
-
Array.isArray(value) ? `[${value.join(', ')}]` : String(value),
|
|
338
|
-
this.options.maxValueLength,
|
|
339
|
-
);
|
|
340
|
-
pairs.push(`${key}=${strValue}`);
|
|
341
|
-
}
|
|
342
|
-
|
|
343
|
-
return pairs.join(', ');
|
|
344
|
-
}
|
|
345
|
-
|
|
346
|
-
/**
|
|
347
|
-
* Truncate string to max length with ellipsis
|
|
348
|
-
*/
|
|
349
|
-
private truncate(str: string, max: number): string {
|
|
350
|
-
if (str.length <= max) return str;
|
|
351
|
-
return str.slice(0, max - 3) + '...';
|
|
352
|
-
}
|
|
353
|
-
|
|
354
|
-
/**
|
|
355
|
-
* Apply ANSI color if colors are enabled
|
|
356
|
-
*/
|
|
357
|
-
private color(text: string, color: AnsiColor): string {
|
|
358
|
-
if (!this.options.colors) return text;
|
|
359
|
-
return `${ANSI[color]}${text}${ANSI.reset}`;
|
|
360
|
-
}
|
|
361
|
-
|
|
362
|
-
/**
|
|
363
|
-
* Shutdown (no-op for console exporter)
|
|
364
|
-
*/
|
|
365
|
-
shutdown(): Promise<void> {
|
|
366
|
-
return Promise.resolve();
|
|
367
|
-
}
|
|
368
|
-
|
|
369
|
-
/**
|
|
370
|
-
* Force flush (no-op for console exporter)
|
|
371
|
-
*/
|
|
372
|
-
forceFlush(): Promise<void> {
|
|
373
|
-
return Promise.resolve();
|
|
374
|
-
}
|
|
375
|
-
}
|
|
376
|
-
|
|
377
|
-
/**
|
|
378
|
-
* Convert HrTime [seconds, nanoseconds] to milliseconds
|
|
379
|
-
*/
|
|
380
|
-
function hrTimeToMs(hrTime: [number, number]): number {
|
|
381
|
-
const [seconds, nanos] = hrTime;
|
|
382
|
-
return seconds * 1000 + nanos / 1_000_000;
|
|
383
|
-
}
|
|
384
|
-
|
|
385
|
-
/**
|
|
386
|
-
* Format duration with appropriate units
|
|
387
|
-
*/
|
|
388
|
-
function formatDuration(ms: number): string {
|
|
389
|
-
if (ms < 1) {
|
|
390
|
-
// Sub-millisecond: show as microseconds
|
|
391
|
-
return `${(ms * 1000).toFixed(0)}µs`;
|
|
392
|
-
}
|
|
393
|
-
if (ms < 1000) {
|
|
394
|
-
// Under 1 second: show as milliseconds
|
|
395
|
-
return `${ms.toFixed(0)}ms`;
|
|
396
|
-
}
|
|
397
|
-
// 1 second or more: show as seconds
|
|
398
|
-
return `${(ms / 1000).toFixed(2)}s`;
|
|
399
|
-
}
|
|
400
|
-
|
|
401
|
-
/**
|
|
402
|
-
* Get color based on duration (fast=green, medium=yellow, slow=red)
|
|
403
|
-
*/
|
|
404
|
-
function getDurationColor(ms: number): AnsiColor {
|
|
405
|
-
if (ms < 100) return 'green';
|
|
406
|
-
if (ms < 500) return 'yellow';
|
|
407
|
-
return 'red';
|
|
408
|
-
}
|
|
409
|
-
|
|
410
|
-
/**
|
|
411
|
-
* Export utility functions for testing
|
|
412
|
-
*/
|
|
413
|
-
export { formatDuration, getDurationColor, hrTimeToMs };
|
|
@@ -1,123 +0,0 @@
|
|
|
1
|
-
import { describe, expect, it, vi } from 'vitest';
|
|
2
|
-
import { formatDuration, formatPrettyLogLine } from './pretty-log-formatter';
|
|
3
|
-
import type { CanonicalLogLineEvent } from './processors/canonical-log-line-processor';
|
|
4
|
-
|
|
5
|
-
describe('formatDuration', () => {
|
|
6
|
-
it('formats sub-second durations as ms', () => {
|
|
7
|
-
expect(formatDuration(0)).toBe('0ms');
|
|
8
|
-
expect(formatDuration(45)).toBe('45ms');
|
|
9
|
-
expect(formatDuration(999)).toBe('999ms');
|
|
10
|
-
});
|
|
11
|
-
|
|
12
|
-
it('formats seconds with one decimal under 10s', () => {
|
|
13
|
-
expect(formatDuration(1234)).toBe('1.2s');
|
|
14
|
-
expect(formatDuration(9500)).toBe('9.5s');
|
|
15
|
-
});
|
|
16
|
-
|
|
17
|
-
it('rounds seconds above 10s', () => {
|
|
18
|
-
expect(formatDuration(12345)).toBe('12s');
|
|
19
|
-
});
|
|
20
|
-
|
|
21
|
-
it('formats minutes', () => {
|
|
22
|
-
expect(formatDuration(65000)).toBe('1m 5s');
|
|
23
|
-
expect(formatDuration(120000)).toBe('2m');
|
|
24
|
-
});
|
|
25
|
-
});
|
|
26
|
-
|
|
27
|
-
describe('formatPrettyLogLine', () => {
|
|
28
|
-
const originalEnv = process.env.NO_COLOR;
|
|
29
|
-
|
|
30
|
-
beforeEach(() => {
|
|
31
|
-
process.env.NO_COLOR = '1';
|
|
32
|
-
});
|
|
33
|
-
|
|
34
|
-
afterEach(() => {
|
|
35
|
-
if (originalEnv === undefined) {
|
|
36
|
-
delete process.env.NO_COLOR;
|
|
37
|
-
} else {
|
|
38
|
-
process.env.NO_COLOR = originalEnv;
|
|
39
|
-
}
|
|
40
|
-
});
|
|
41
|
-
|
|
42
|
-
it('formats a basic request line', () => {
|
|
43
|
-
const ctx: CanonicalLogLineEvent = {
|
|
44
|
-
span: {} as any,
|
|
45
|
-
level: 'info',
|
|
46
|
-
message: '[checkout] Request completed',
|
|
47
|
-
event: {
|
|
48
|
-
timestamp: '2025-01-24T16:45:31.060Z',
|
|
49
|
-
'service.name': 'my-app',
|
|
50
|
-
'http.request.method': 'GET',
|
|
51
|
-
'http.route': '/api/checkout',
|
|
52
|
-
'http.response.status_code': 200,
|
|
53
|
-
duration_ms: 234,
|
|
54
|
-
status_code: 0,
|
|
55
|
-
operation: 'checkout',
|
|
56
|
-
traceId: 'abc123',
|
|
57
|
-
spanId: 'def456',
|
|
58
|
-
correlationId: 'abc1',
|
|
59
|
-
},
|
|
60
|
-
};
|
|
61
|
-
|
|
62
|
-
const output = formatPrettyLogLine(ctx);
|
|
63
|
-
expect(output).toContain('INFO');
|
|
64
|
-
expect(output).toContain('[my-app]');
|
|
65
|
-
expect(output).toContain('GET');
|
|
66
|
-
expect(output).toContain('/api/checkout');
|
|
67
|
-
expect(output).toContain('200');
|
|
68
|
-
expect(output).toContain('234ms');
|
|
69
|
-
});
|
|
70
|
-
|
|
71
|
-
it('includes context attributes as tree', () => {
|
|
72
|
-
const ctx: CanonicalLogLineEvent = {
|
|
73
|
-
span: {} as any,
|
|
74
|
-
level: 'info',
|
|
75
|
-
message: 'Request completed',
|
|
76
|
-
event: {
|
|
77
|
-
timestamp: '2025-01-24T16:45:31.060Z',
|
|
78
|
-
duration_ms: 100,
|
|
79
|
-
status_code: 0,
|
|
80
|
-
operation: 'checkout',
|
|
81
|
-
traceId: 'abc',
|
|
82
|
-
spanId: 'def',
|
|
83
|
-
correlationId: 'abc',
|
|
84
|
-
'user.id': '123',
|
|
85
|
-
'user.plan': 'premium',
|
|
86
|
-
'cart.items': 3,
|
|
87
|
-
},
|
|
88
|
-
};
|
|
89
|
-
|
|
90
|
-
const output = formatPrettyLogLine(ctx);
|
|
91
|
-
expect(output).toContain('user');
|
|
92
|
-
expect(output).toContain('id=123');
|
|
93
|
-
expect(output).toContain('plan=premium');
|
|
94
|
-
expect(output).toContain('cart');
|
|
95
|
-
expect(output).toContain('items');
|
|
96
|
-
});
|
|
97
|
-
|
|
98
|
-
it('skips internal telemetry attributes', () => {
|
|
99
|
-
const ctx: CanonicalLogLineEvent = {
|
|
100
|
-
span: {} as any,
|
|
101
|
-
level: 'info',
|
|
102
|
-
message: 'Request completed',
|
|
103
|
-
event: {
|
|
104
|
-
timestamp: '2025-01-24T16:45:31.060Z',
|
|
105
|
-
duration_ms: 50,
|
|
106
|
-
status_code: 0,
|
|
107
|
-
operation: 'test',
|
|
108
|
-
traceId: 'abc',
|
|
109
|
-
spanId: 'def',
|
|
110
|
-
correlationId: 'abc',
|
|
111
|
-
'telemetry.sdk.name': 'autotel',
|
|
112
|
-
'otel.scope.name': 'my-scope',
|
|
113
|
-
'service.name': 'my-app',
|
|
114
|
-
custom_field: 'visible',
|
|
115
|
-
},
|
|
116
|
-
};
|
|
117
|
-
|
|
118
|
-
const output = formatPrettyLogLine(ctx);
|
|
119
|
-
expect(output).not.toContain('telemetry.sdk.name');
|
|
120
|
-
expect(output).not.toContain('otel.scope.name');
|
|
121
|
-
expect(output).toContain('custom_field');
|
|
122
|
-
});
|
|
123
|
-
});
|