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/env-config.test.ts
DELETED
|
@@ -1,323 +0,0 @@
|
|
|
1
|
-
import { describe, it, expect, beforeEach, afterEach } from 'vitest';
|
|
2
|
-
import {
|
|
3
|
-
AlwaysOffSampler,
|
|
4
|
-
AlwaysOnSampler,
|
|
5
|
-
ParentBasedSampler,
|
|
6
|
-
TraceIdRatioBasedSampler,
|
|
7
|
-
} from '@opentelemetry/sdk-trace-base';
|
|
8
|
-
import {
|
|
9
|
-
resolveOtelEnv,
|
|
10
|
-
parseResourceAttributes,
|
|
11
|
-
parseOtlpHeaders,
|
|
12
|
-
createSamplerFromEnv,
|
|
13
|
-
envToConfig,
|
|
14
|
-
resolveConfigFromEnv,
|
|
15
|
-
} from './env-config';
|
|
16
|
-
|
|
17
|
-
describe('env-config', () => {
|
|
18
|
-
const originalEnv = process.env;
|
|
19
|
-
|
|
20
|
-
beforeEach(() => {
|
|
21
|
-
// Reset environment before each test
|
|
22
|
-
process.env = { ...originalEnv };
|
|
23
|
-
});
|
|
24
|
-
|
|
25
|
-
afterEach(() => {
|
|
26
|
-
// Restore original environment
|
|
27
|
-
process.env = originalEnv;
|
|
28
|
-
});
|
|
29
|
-
|
|
30
|
-
describe('resolveOtelEnv', () => {
|
|
31
|
-
it('should resolve standard OTEL env vars', () => {
|
|
32
|
-
process.env.OTEL_SERVICE_NAME = 'test-service';
|
|
33
|
-
process.env.OTEL_EXPORTER_OTLP_ENDPOINT = 'http://localhost:4318';
|
|
34
|
-
process.env.OTEL_EXPORTER_OTLP_PROTOCOL = 'http';
|
|
35
|
-
|
|
36
|
-
const env = resolveOtelEnv();
|
|
37
|
-
|
|
38
|
-
expect(env.OTEL_SERVICE_NAME).toBe('test-service');
|
|
39
|
-
expect(env.OTEL_EXPORTER_OTLP_ENDPOINT).toBe('http://localhost:4318');
|
|
40
|
-
expect(env.OTEL_EXPORTER_OTLP_PROTOCOL).toBe('http');
|
|
41
|
-
});
|
|
42
|
-
|
|
43
|
-
it('should return undefined for unset env vars', () => {
|
|
44
|
-
const env = resolveOtelEnv();
|
|
45
|
-
|
|
46
|
-
expect(env.OTEL_SERVICE_NAME).toBeUndefined();
|
|
47
|
-
expect(env.OTEL_EXPORTER_OTLP_ENDPOINT).toBeUndefined();
|
|
48
|
-
});
|
|
49
|
-
|
|
50
|
-
it('should validate protocol enum', () => {
|
|
51
|
-
process.env.OTEL_EXPORTER_OTLP_PROTOCOL = 'grpc';
|
|
52
|
-
const env = resolveOtelEnv();
|
|
53
|
-
expect(env.OTEL_EXPORTER_OTLP_PROTOCOL).toBe('grpc');
|
|
54
|
-
});
|
|
55
|
-
|
|
56
|
-
it('should resolve trace sampler env vars', () => {
|
|
57
|
-
process.env.OTEL_TRACES_SAMPLER = 'traceidratio';
|
|
58
|
-
process.env.OTEL_TRACES_SAMPLER_ARG = '0.5';
|
|
59
|
-
|
|
60
|
-
const env = resolveOtelEnv();
|
|
61
|
-
|
|
62
|
-
expect(env.OTEL_TRACES_SAMPLER).toBe('traceidratio');
|
|
63
|
-
expect(env.OTEL_TRACES_SAMPLER_ARG).toBe('0.5');
|
|
64
|
-
});
|
|
65
|
-
});
|
|
66
|
-
|
|
67
|
-
describe('parseResourceAttributes', () => {
|
|
68
|
-
it('should parse comma-separated key=value pairs', () => {
|
|
69
|
-
const input = 'service.version=1.0.0,deployment.environment=production';
|
|
70
|
-
const result = parseResourceAttributes(input);
|
|
71
|
-
|
|
72
|
-
expect(result).toEqual({
|
|
73
|
-
'service.version': '1.0.0',
|
|
74
|
-
'deployment.environment': 'production',
|
|
75
|
-
});
|
|
76
|
-
});
|
|
77
|
-
|
|
78
|
-
it('should handle single attribute', () => {
|
|
79
|
-
const result = parseResourceAttributes('team=backend');
|
|
80
|
-
expect(result).toEqual({ team: 'backend' });
|
|
81
|
-
});
|
|
82
|
-
|
|
83
|
-
it('should handle empty string', () => {
|
|
84
|
-
expect(parseResourceAttributes('')).toEqual({});
|
|
85
|
-
expect(parseResourceAttributes(' ')).toEqual({});
|
|
86
|
-
});
|
|
87
|
-
|
|
88
|
-
it('should handle undefined', () => {
|
|
89
|
-
expect(parseResourceAttributes()).toEqual({});
|
|
90
|
-
});
|
|
91
|
-
|
|
92
|
-
it('should skip invalid pairs without =', () => {
|
|
93
|
-
const result = parseResourceAttributes('valid=value,invalid,another=ok');
|
|
94
|
-
expect(result).toEqual({
|
|
95
|
-
valid: 'value',
|
|
96
|
-
another: 'ok',
|
|
97
|
-
});
|
|
98
|
-
});
|
|
99
|
-
|
|
100
|
-
it('should handle whitespace around keys and values', () => {
|
|
101
|
-
const result = parseResourceAttributes(' key1 = value1 , key2 = value2 ');
|
|
102
|
-
expect(result).toEqual({
|
|
103
|
-
key1: 'value1',
|
|
104
|
-
key2: 'value2',
|
|
105
|
-
});
|
|
106
|
-
});
|
|
107
|
-
|
|
108
|
-
it('should skip empty pairs', () => {
|
|
109
|
-
const result = parseResourceAttributes('key1=value1,,key2=value2');
|
|
110
|
-
expect(result).toEqual({
|
|
111
|
-
key1: 'value1',
|
|
112
|
-
key2: 'value2',
|
|
113
|
-
});
|
|
114
|
-
});
|
|
115
|
-
|
|
116
|
-
it('should handle values with = in them (takes first = as delimiter)', () => {
|
|
117
|
-
const result = parseResourceAttributes('key=value=with=equals');
|
|
118
|
-
expect(result).toEqual({
|
|
119
|
-
key: 'value=with=equals',
|
|
120
|
-
});
|
|
121
|
-
});
|
|
122
|
-
});
|
|
123
|
-
|
|
124
|
-
describe('parseOtlpHeaders', () => {
|
|
125
|
-
it('should parse comma-separated header pairs', () => {
|
|
126
|
-
const input = 'api-key=secret123,x-custom-header=value';
|
|
127
|
-
const result = parseOtlpHeaders(input);
|
|
128
|
-
|
|
129
|
-
expect(result).toEqual({
|
|
130
|
-
'api-key': 'secret123',
|
|
131
|
-
'x-custom-header': 'value',
|
|
132
|
-
});
|
|
133
|
-
});
|
|
134
|
-
|
|
135
|
-
it('should handle single header', () => {
|
|
136
|
-
const result = parseOtlpHeaders('authorization=Bearer token');
|
|
137
|
-
expect(result).toEqual({ authorization: 'Bearer token' });
|
|
138
|
-
});
|
|
139
|
-
|
|
140
|
-
it('should handle empty string', () => {
|
|
141
|
-
expect(parseOtlpHeaders('')).toEqual({});
|
|
142
|
-
expect(parseOtlpHeaders(' ')).toEqual({});
|
|
143
|
-
});
|
|
144
|
-
|
|
145
|
-
it('should handle undefined', () => {
|
|
146
|
-
expect(parseOtlpHeaders()).toEqual({});
|
|
147
|
-
});
|
|
148
|
-
|
|
149
|
-
it('should skip invalid pairs', () => {
|
|
150
|
-
const result = parseOtlpHeaders('valid=value,invalid,another=ok');
|
|
151
|
-
expect(result).toEqual({
|
|
152
|
-
valid: 'value',
|
|
153
|
-
another: 'ok',
|
|
154
|
-
});
|
|
155
|
-
});
|
|
156
|
-
|
|
157
|
-
it('should handle whitespace', () => {
|
|
158
|
-
const result = parseOtlpHeaders(' key1 = value1 , key2 = value2 ');
|
|
159
|
-
expect(result).toEqual({
|
|
160
|
-
key1: 'value1',
|
|
161
|
-
key2: 'value2',
|
|
162
|
-
});
|
|
163
|
-
});
|
|
164
|
-
});
|
|
165
|
-
|
|
166
|
-
describe('envToConfig', () => {
|
|
167
|
-
it('should map OTEL_SERVICE_NAME to service', () => {
|
|
168
|
-
const config = envToConfig({
|
|
169
|
-
OTEL_SERVICE_NAME: 'test-service',
|
|
170
|
-
});
|
|
171
|
-
|
|
172
|
-
expect(config.service).toBe('test-service');
|
|
173
|
-
});
|
|
174
|
-
|
|
175
|
-
it('should map OTEL_EXPORTER_OTLP_ENDPOINT to endpoint', () => {
|
|
176
|
-
const config = envToConfig({
|
|
177
|
-
OTEL_EXPORTER_OTLP_ENDPOINT: 'http://localhost:4318',
|
|
178
|
-
});
|
|
179
|
-
|
|
180
|
-
expect(config.endpoint).toBe('http://localhost:4318');
|
|
181
|
-
});
|
|
182
|
-
|
|
183
|
-
it('should map OTEL_EXPORTER_OTLP_PROTOCOL to protocol', () => {
|
|
184
|
-
const config = envToConfig({
|
|
185
|
-
OTEL_EXPORTER_OTLP_PROTOCOL: 'grpc',
|
|
186
|
-
});
|
|
187
|
-
|
|
188
|
-
expect(config.protocol).toBe('grpc');
|
|
189
|
-
});
|
|
190
|
-
|
|
191
|
-
it('should parse OTEL_EXPORTER_OTLP_HEADERS', () => {
|
|
192
|
-
const config = envToConfig({
|
|
193
|
-
OTEL_EXPORTER_OTLP_HEADERS: 'api-key=secret,x-custom=value',
|
|
194
|
-
});
|
|
195
|
-
|
|
196
|
-
expect(config.headers).toEqual({
|
|
197
|
-
'api-key': 'secret',
|
|
198
|
-
'x-custom': 'value',
|
|
199
|
-
});
|
|
200
|
-
});
|
|
201
|
-
|
|
202
|
-
it('should parse OTEL_RESOURCE_ATTRIBUTES', () => {
|
|
203
|
-
const config = envToConfig({
|
|
204
|
-
OTEL_RESOURCE_ATTRIBUTES:
|
|
205
|
-
'service.version=1.0.0,deployment.environment=prod',
|
|
206
|
-
});
|
|
207
|
-
|
|
208
|
-
expect(config.resourceAttributes).toEqual({
|
|
209
|
-
'service.version': '1.0.0',
|
|
210
|
-
'deployment.environment': 'prod',
|
|
211
|
-
});
|
|
212
|
-
});
|
|
213
|
-
|
|
214
|
-
it('should return empty config for no env vars', () => {
|
|
215
|
-
const config = envToConfig({});
|
|
216
|
-
expect(config).toEqual({});
|
|
217
|
-
});
|
|
218
|
-
|
|
219
|
-
it('should map OTEL_TRACES_SAMPLER to an SDK sampler', () => {
|
|
220
|
-
const config = envToConfig({
|
|
221
|
-
OTEL_TRACES_SAMPLER: 'always_on',
|
|
222
|
-
});
|
|
223
|
-
|
|
224
|
-
expect(config.otelSampler).toBeInstanceOf(AlwaysOnSampler);
|
|
225
|
-
});
|
|
226
|
-
});
|
|
227
|
-
|
|
228
|
-
describe('createSamplerFromEnv', () => {
|
|
229
|
-
it('supports always_on', () => {
|
|
230
|
-
const sampler = createSamplerFromEnv({
|
|
231
|
-
OTEL_TRACES_SAMPLER: 'always_on',
|
|
232
|
-
});
|
|
233
|
-
|
|
234
|
-
expect(sampler).toBeInstanceOf(AlwaysOnSampler);
|
|
235
|
-
});
|
|
236
|
-
|
|
237
|
-
it('supports always_off', () => {
|
|
238
|
-
const sampler = createSamplerFromEnv({
|
|
239
|
-
OTEL_TRACES_SAMPLER: 'always_off',
|
|
240
|
-
});
|
|
241
|
-
|
|
242
|
-
expect(sampler).toBeInstanceOf(AlwaysOffSampler);
|
|
243
|
-
});
|
|
244
|
-
|
|
245
|
-
it('supports traceidratio', () => {
|
|
246
|
-
const sampler = createSamplerFromEnv({
|
|
247
|
-
OTEL_TRACES_SAMPLER: 'traceidratio',
|
|
248
|
-
OTEL_TRACES_SAMPLER_ARG: '0.25',
|
|
249
|
-
});
|
|
250
|
-
|
|
251
|
-
expect(sampler).toBeInstanceOf(TraceIdRatioBasedSampler);
|
|
252
|
-
});
|
|
253
|
-
|
|
254
|
-
it('supports parentbased_traceidratio', () => {
|
|
255
|
-
const sampler = createSamplerFromEnv({
|
|
256
|
-
OTEL_TRACES_SAMPLER: 'parentbased_traceidratio',
|
|
257
|
-
OTEL_TRACES_SAMPLER_ARG: '0.25',
|
|
258
|
-
});
|
|
259
|
-
|
|
260
|
-
expect(sampler).toBeInstanceOf(ParentBasedSampler);
|
|
261
|
-
});
|
|
262
|
-
|
|
263
|
-
it('returns undefined for unsupported remote samplers', () => {
|
|
264
|
-
const sampler = createSamplerFromEnv({
|
|
265
|
-
OTEL_TRACES_SAMPLER: 'jaeger_remote',
|
|
266
|
-
});
|
|
267
|
-
|
|
268
|
-
expect(sampler).toBeUndefined();
|
|
269
|
-
});
|
|
270
|
-
});
|
|
271
|
-
|
|
272
|
-
describe('resolveConfigFromEnv', () => {
|
|
273
|
-
it('should return config from env vars', () => {
|
|
274
|
-
process.env.OTEL_SERVICE_NAME = 'test-service';
|
|
275
|
-
process.env.OTEL_EXPORTER_OTLP_ENDPOINT = 'http://localhost:4318';
|
|
276
|
-
|
|
277
|
-
const config = resolveConfigFromEnv();
|
|
278
|
-
|
|
279
|
-
expect(config.service).toBe('test-service');
|
|
280
|
-
expect(config.endpoint).toBe('http://localhost:4318');
|
|
281
|
-
});
|
|
282
|
-
|
|
283
|
-
it('should work with no env vars set', () => {
|
|
284
|
-
const config = resolveConfigFromEnv();
|
|
285
|
-
expect(config).toEqual({});
|
|
286
|
-
});
|
|
287
|
-
|
|
288
|
-
it('should parse complex real-world scenario', () => {
|
|
289
|
-
process.env.OTEL_SERVICE_NAME = 'my-api';
|
|
290
|
-
process.env.OTEL_EXPORTER_OTLP_ENDPOINT = 'https://api.honeycomb.io';
|
|
291
|
-
process.env.OTEL_EXPORTER_OTLP_HEADERS =
|
|
292
|
-
'x-honeycomb-team=abc123,x-honeycomb-dataset=production';
|
|
293
|
-
process.env.OTEL_RESOURCE_ATTRIBUTES =
|
|
294
|
-
'service.version=1.2.3,deployment.environment=production,team=backend';
|
|
295
|
-
process.env.OTEL_EXPORTER_OTLP_PROTOCOL = 'http';
|
|
296
|
-
|
|
297
|
-
const config = resolveConfigFromEnv();
|
|
298
|
-
|
|
299
|
-
expect(config).toEqual({
|
|
300
|
-
service: 'my-api',
|
|
301
|
-
endpoint: 'https://api.honeycomb.io',
|
|
302
|
-
protocol: 'http',
|
|
303
|
-
headers: {
|
|
304
|
-
'x-honeycomb-team': 'abc123',
|
|
305
|
-
'x-honeycomb-dataset': 'production',
|
|
306
|
-
},
|
|
307
|
-
resourceAttributes: {
|
|
308
|
-
'service.version': '1.2.3',
|
|
309
|
-
'deployment.environment': 'production',
|
|
310
|
-
team: 'backend',
|
|
311
|
-
},
|
|
312
|
-
});
|
|
313
|
-
});
|
|
314
|
-
|
|
315
|
-
it('should include otelSampler when OTEL_TRACES_SAMPLER is set', () => {
|
|
316
|
-
process.env.OTEL_TRACES_SAMPLER = 'parentbased_always_off';
|
|
317
|
-
|
|
318
|
-
const config = resolveConfigFromEnv();
|
|
319
|
-
|
|
320
|
-
expect(config.otelSampler).toBeInstanceOf(ParentBasedSampler);
|
|
321
|
-
});
|
|
322
|
-
});
|
|
323
|
-
});
|
package/src/env-config.ts
DELETED
|
@@ -1,309 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Standard OpenTelemetry environment variables
|
|
3
|
-
*/
|
|
4
|
-
import type { Sampler as OtelSampler } from '@opentelemetry/sdk-trace-base';
|
|
5
|
-
import {
|
|
6
|
-
AlwaysOffSampler,
|
|
7
|
-
AlwaysOnSampler,
|
|
8
|
-
ParentBasedSampler,
|
|
9
|
-
TraceIdRatioBasedSampler,
|
|
10
|
-
} from '@opentelemetry/sdk-trace-base';
|
|
11
|
-
|
|
12
|
-
export interface OtelEnvVars {
|
|
13
|
-
OTEL_SERVICE_NAME?: string;
|
|
14
|
-
OTEL_EXPORTER_OTLP_ENDPOINT?: string;
|
|
15
|
-
OTEL_EXPORTER_OTLP_HEADERS?: string;
|
|
16
|
-
OTEL_RESOURCE_ATTRIBUTES?: string;
|
|
17
|
-
OTEL_EXPORTER_OTLP_PROTOCOL?: 'http' | 'grpc';
|
|
18
|
-
OTEL_TRACES_SAMPLER?: string;
|
|
19
|
-
OTEL_TRACES_SAMPLER_ARG?: string;
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
/**
|
|
23
|
-
* Parsed resource attributes as key-value pairs
|
|
24
|
-
*/
|
|
25
|
-
export interface ResourceAttributes {
|
|
26
|
-
[key: string]: string;
|
|
27
|
-
}
|
|
28
|
-
|
|
29
|
-
/**
|
|
30
|
-
* Parsed OTLP headers as key-value pairs
|
|
31
|
-
*/
|
|
32
|
-
export interface OtlpHeaders {
|
|
33
|
-
[key: string]: string;
|
|
34
|
-
}
|
|
35
|
-
|
|
36
|
-
/**
|
|
37
|
-
* Environment-resolved configuration (subset of AutotelConfig)
|
|
38
|
-
* Defined locally to avoid circular dependency with init.ts
|
|
39
|
-
*/
|
|
40
|
-
export interface EnvConfig {
|
|
41
|
-
service?: string;
|
|
42
|
-
endpoint?: string;
|
|
43
|
-
protocol?: 'http' | 'grpc';
|
|
44
|
-
headers?: Record<string, string>;
|
|
45
|
-
resourceAttributes?: Record<string, string>;
|
|
46
|
-
otelSampler?: OtelSampler;
|
|
47
|
-
}
|
|
48
|
-
|
|
49
|
-
/**
|
|
50
|
-
* Validate URL format
|
|
51
|
-
*/
|
|
52
|
-
function isValidUrl(urlString: string): boolean {
|
|
53
|
-
try {
|
|
54
|
-
const url = new URL(urlString);
|
|
55
|
-
return url.protocol === 'http:' || url.protocol === 'https:';
|
|
56
|
-
} catch {
|
|
57
|
-
return false;
|
|
58
|
-
}
|
|
59
|
-
}
|
|
60
|
-
|
|
61
|
-
/**
|
|
62
|
-
* Resolve OpenTelemetry environment variables from process.env
|
|
63
|
-
*/
|
|
64
|
-
export function resolveOtelEnv(): OtelEnvVars {
|
|
65
|
-
const env: OtelEnvVars = {};
|
|
66
|
-
|
|
67
|
-
// OTEL_SERVICE_NAME - optional string
|
|
68
|
-
if (process.env.OTEL_SERVICE_NAME) {
|
|
69
|
-
const value = process.env.OTEL_SERVICE_NAME.trim();
|
|
70
|
-
if (value) {
|
|
71
|
-
env.OTEL_SERVICE_NAME = value;
|
|
72
|
-
}
|
|
73
|
-
}
|
|
74
|
-
|
|
75
|
-
// OTEL_EXPORTER_OTLP_ENDPOINT - optional URL
|
|
76
|
-
if (process.env.OTEL_EXPORTER_OTLP_ENDPOINT) {
|
|
77
|
-
const value = process.env.OTEL_EXPORTER_OTLP_ENDPOINT.trim();
|
|
78
|
-
if (value && isValidUrl(value)) {
|
|
79
|
-
env.OTEL_EXPORTER_OTLP_ENDPOINT = value;
|
|
80
|
-
}
|
|
81
|
-
}
|
|
82
|
-
|
|
83
|
-
// OTEL_EXPORTER_OTLP_HEADERS - optional string
|
|
84
|
-
if (process.env.OTEL_EXPORTER_OTLP_HEADERS) {
|
|
85
|
-
const value = process.env.OTEL_EXPORTER_OTLP_HEADERS.trim();
|
|
86
|
-
if (value) {
|
|
87
|
-
env.OTEL_EXPORTER_OTLP_HEADERS = value;
|
|
88
|
-
}
|
|
89
|
-
}
|
|
90
|
-
|
|
91
|
-
// OTEL_RESOURCE_ATTRIBUTES - optional string
|
|
92
|
-
if (process.env.OTEL_RESOURCE_ATTRIBUTES) {
|
|
93
|
-
const value = process.env.OTEL_RESOURCE_ATTRIBUTES.trim();
|
|
94
|
-
if (value) {
|
|
95
|
-
env.OTEL_RESOURCE_ATTRIBUTES = value;
|
|
96
|
-
}
|
|
97
|
-
}
|
|
98
|
-
|
|
99
|
-
// OTEL_EXPORTER_OTLP_PROTOCOL - optional enum ('http' | 'grpc')
|
|
100
|
-
if (process.env.OTEL_EXPORTER_OTLP_PROTOCOL) {
|
|
101
|
-
const value = process.env.OTEL_EXPORTER_OTLP_PROTOCOL.trim().toLowerCase();
|
|
102
|
-
if (value === 'http' || value === 'grpc') {
|
|
103
|
-
env.OTEL_EXPORTER_OTLP_PROTOCOL = value;
|
|
104
|
-
}
|
|
105
|
-
}
|
|
106
|
-
|
|
107
|
-
if (process.env.OTEL_TRACES_SAMPLER) {
|
|
108
|
-
const value = process.env.OTEL_TRACES_SAMPLER.trim();
|
|
109
|
-
if (value) {
|
|
110
|
-
env.OTEL_TRACES_SAMPLER = value;
|
|
111
|
-
}
|
|
112
|
-
}
|
|
113
|
-
|
|
114
|
-
if (process.env.OTEL_TRACES_SAMPLER_ARG) {
|
|
115
|
-
const value = process.env.OTEL_TRACES_SAMPLER_ARG.trim();
|
|
116
|
-
if (value) {
|
|
117
|
-
env.OTEL_TRACES_SAMPLER_ARG = value;
|
|
118
|
-
}
|
|
119
|
-
}
|
|
120
|
-
|
|
121
|
-
return env;
|
|
122
|
-
}
|
|
123
|
-
|
|
124
|
-
function parseRatioSamplerArg(
|
|
125
|
-
samplerName: string,
|
|
126
|
-
samplerArg: string | undefined,
|
|
127
|
-
): number {
|
|
128
|
-
if (samplerArg === undefined) {
|
|
129
|
-
return 1.0;
|
|
130
|
-
}
|
|
131
|
-
|
|
132
|
-
const ratio = Number(samplerArg);
|
|
133
|
-
if (!Number.isFinite(ratio) || ratio < 0 || ratio > 1) {
|
|
134
|
-
console.error(
|
|
135
|
-
`[autotel] Invalid OTEL_TRACES_SAMPLER_ARG="${samplerArg}" for ${samplerName}. Expected a number in [0..1]. Falling back to 1.0.`,
|
|
136
|
-
);
|
|
137
|
-
return 1.0;
|
|
138
|
-
}
|
|
139
|
-
|
|
140
|
-
return ratio;
|
|
141
|
-
}
|
|
142
|
-
|
|
143
|
-
function warnOnUnusedSamplerArg(
|
|
144
|
-
samplerName: string,
|
|
145
|
-
samplerArg: string | undefined,
|
|
146
|
-
): void {
|
|
147
|
-
if (samplerArg !== undefined) {
|
|
148
|
-
console.error(
|
|
149
|
-
`[autotel] OTEL_TRACES_SAMPLER_ARG is not used by OTEL_TRACES_SAMPLER="${samplerName}". Ignoring value "${samplerArg}".`,
|
|
150
|
-
);
|
|
151
|
-
}
|
|
152
|
-
}
|
|
153
|
-
|
|
154
|
-
export function createSamplerFromEnv(
|
|
155
|
-
env: Pick<OtelEnvVars, 'OTEL_TRACES_SAMPLER' | 'OTEL_TRACES_SAMPLER_ARG'>,
|
|
156
|
-
): OtelSampler | undefined {
|
|
157
|
-
const samplerName = env.OTEL_TRACES_SAMPLER;
|
|
158
|
-
if (!samplerName) {
|
|
159
|
-
return undefined;
|
|
160
|
-
}
|
|
161
|
-
|
|
162
|
-
switch (samplerName) {
|
|
163
|
-
case 'always_on':
|
|
164
|
-
warnOnUnusedSamplerArg(samplerName, env.OTEL_TRACES_SAMPLER_ARG);
|
|
165
|
-
return new AlwaysOnSampler();
|
|
166
|
-
case 'always_off':
|
|
167
|
-
warnOnUnusedSamplerArg(samplerName, env.OTEL_TRACES_SAMPLER_ARG);
|
|
168
|
-
return new AlwaysOffSampler();
|
|
169
|
-
case 'traceidratio':
|
|
170
|
-
return new TraceIdRatioBasedSampler(
|
|
171
|
-
parseRatioSamplerArg(samplerName, env.OTEL_TRACES_SAMPLER_ARG),
|
|
172
|
-
);
|
|
173
|
-
case 'parentbased_always_on':
|
|
174
|
-
warnOnUnusedSamplerArg(samplerName, env.OTEL_TRACES_SAMPLER_ARG);
|
|
175
|
-
return new ParentBasedSampler({ root: new AlwaysOnSampler() });
|
|
176
|
-
case 'parentbased_always_off':
|
|
177
|
-
warnOnUnusedSamplerArg(samplerName, env.OTEL_TRACES_SAMPLER_ARG);
|
|
178
|
-
return new ParentBasedSampler({ root: new AlwaysOffSampler() });
|
|
179
|
-
case 'parentbased_traceidratio':
|
|
180
|
-
return new ParentBasedSampler({
|
|
181
|
-
root: new TraceIdRatioBasedSampler(
|
|
182
|
-
parseRatioSamplerArg(samplerName, env.OTEL_TRACES_SAMPLER_ARG),
|
|
183
|
-
),
|
|
184
|
-
});
|
|
185
|
-
case 'jaeger_remote':
|
|
186
|
-
case 'parentbased_jaeger_remote':
|
|
187
|
-
case 'xray':
|
|
188
|
-
console.error(
|
|
189
|
-
`[autotel] OTEL_TRACES_SAMPLER="${samplerName}" is not supported yet by autotel. Falling back to the next sampler source.`,
|
|
190
|
-
);
|
|
191
|
-
return undefined;
|
|
192
|
-
default:
|
|
193
|
-
console.error(
|
|
194
|
-
`[autotel] Unknown OTEL_TRACES_SAMPLER="${samplerName}". Falling back to the next sampler source.`,
|
|
195
|
-
);
|
|
196
|
-
return undefined;
|
|
197
|
-
}
|
|
198
|
-
}
|
|
199
|
-
|
|
200
|
-
/**
|
|
201
|
-
* Parse OTEL_RESOURCE_ATTRIBUTES from comma-separated key=value pairs
|
|
202
|
-
* Example: "service.version=1.0.0,deployment.environment=production"
|
|
203
|
-
*/
|
|
204
|
-
export function parseResourceAttributes(
|
|
205
|
-
input: string | undefined,
|
|
206
|
-
): ResourceAttributes {
|
|
207
|
-
if (!input || input.trim() === '') {
|
|
208
|
-
return {};
|
|
209
|
-
}
|
|
210
|
-
|
|
211
|
-
const attributes: ResourceAttributes = {};
|
|
212
|
-
const pairs = input.split(',');
|
|
213
|
-
|
|
214
|
-
for (const pair of pairs) {
|
|
215
|
-
const trimmedPair = pair.trim();
|
|
216
|
-
if (!trimmedPair) continue;
|
|
217
|
-
|
|
218
|
-
const equalIndex = trimmedPair.indexOf('=');
|
|
219
|
-
if (equalIndex === -1) {
|
|
220
|
-
// Invalid format, skip this pair
|
|
221
|
-
continue;
|
|
222
|
-
}
|
|
223
|
-
|
|
224
|
-
const key = trimmedPair.slice(0, equalIndex).trim();
|
|
225
|
-
const value = trimmedPair.slice(equalIndex + 1).trim();
|
|
226
|
-
|
|
227
|
-
if (key && value) {
|
|
228
|
-
attributes[key] = value;
|
|
229
|
-
}
|
|
230
|
-
}
|
|
231
|
-
|
|
232
|
-
return attributes;
|
|
233
|
-
}
|
|
234
|
-
|
|
235
|
-
/**
|
|
236
|
-
* Parse OTEL_EXPORTER_OTLP_HEADERS from comma-separated key=value pairs
|
|
237
|
-
* Example: "api-key=secret123,x-custom-header=value"
|
|
238
|
-
*/
|
|
239
|
-
export function parseOtlpHeaders(input: string | undefined): OtlpHeaders {
|
|
240
|
-
if (!input || input.trim() === '') {
|
|
241
|
-
return {};
|
|
242
|
-
}
|
|
243
|
-
|
|
244
|
-
const headers: OtlpHeaders = {};
|
|
245
|
-
const pairs = input.split(',');
|
|
246
|
-
|
|
247
|
-
for (const pair of pairs) {
|
|
248
|
-
const trimmedPair = pair.trim();
|
|
249
|
-
if (!trimmedPair) continue;
|
|
250
|
-
|
|
251
|
-
const equalIndex = trimmedPair.indexOf('=');
|
|
252
|
-
if (equalIndex === -1) {
|
|
253
|
-
// Invalid format, skip this pair
|
|
254
|
-
continue;
|
|
255
|
-
}
|
|
256
|
-
|
|
257
|
-
const key = trimmedPair.slice(0, equalIndex).trim();
|
|
258
|
-
const value = trimmedPair.slice(equalIndex + 1).trim();
|
|
259
|
-
|
|
260
|
-
if (key && value) {
|
|
261
|
-
headers[key] = value;
|
|
262
|
-
}
|
|
263
|
-
}
|
|
264
|
-
|
|
265
|
-
return headers;
|
|
266
|
-
}
|
|
267
|
-
|
|
268
|
-
/**
|
|
269
|
-
* Convert resolved environment variables to config
|
|
270
|
-
*/
|
|
271
|
-
export function envToConfig(env: OtelEnvVars): EnvConfig {
|
|
272
|
-
const config: EnvConfig = {};
|
|
273
|
-
|
|
274
|
-
if (env.OTEL_SERVICE_NAME) {
|
|
275
|
-
config.service = env.OTEL_SERVICE_NAME;
|
|
276
|
-
}
|
|
277
|
-
|
|
278
|
-
if (env.OTEL_EXPORTER_OTLP_ENDPOINT) {
|
|
279
|
-
config.endpoint = env.OTEL_EXPORTER_OTLP_ENDPOINT;
|
|
280
|
-
}
|
|
281
|
-
|
|
282
|
-
if (env.OTEL_EXPORTER_OTLP_PROTOCOL) {
|
|
283
|
-
config.protocol = env.OTEL_EXPORTER_OTLP_PROTOCOL;
|
|
284
|
-
}
|
|
285
|
-
|
|
286
|
-
if (env.OTEL_EXPORTER_OTLP_HEADERS) {
|
|
287
|
-
config.headers = parseOtlpHeaders(env.OTEL_EXPORTER_OTLP_HEADERS);
|
|
288
|
-
}
|
|
289
|
-
|
|
290
|
-
const resourceAttrs = parseResourceAttributes(env.OTEL_RESOURCE_ATTRIBUTES);
|
|
291
|
-
if (Object.keys(resourceAttrs).length > 0) {
|
|
292
|
-
config.resourceAttributes = resourceAttrs;
|
|
293
|
-
}
|
|
294
|
-
|
|
295
|
-
const sampler = createSamplerFromEnv(env);
|
|
296
|
-
if (sampler) {
|
|
297
|
-
config.otelSampler = sampler;
|
|
298
|
-
}
|
|
299
|
-
|
|
300
|
-
return config;
|
|
301
|
-
}
|
|
302
|
-
|
|
303
|
-
/**
|
|
304
|
-
* Main function to resolve config from environment variables
|
|
305
|
-
*/
|
|
306
|
-
export function resolveConfigFromEnv(): EnvConfig {
|
|
307
|
-
const env = resolveOtelEnv();
|
|
308
|
-
return envToConfig(env);
|
|
309
|
-
}
|