autotel-playwright 0.4.35 → 0.4.36

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/index.cjs CHANGED
@@ -1,238 +1,296 @@
1
- 'use strict';
2
-
3
- var test$1 = require('@playwright/test');
4
- var autotel = require('autotel');
5
- var testSpanCollector = require('autotel/test-span-collector');
6
- var processors = require('autotel/processors');
7
-
8
- // src/index.ts
9
- var TRACER_NAME = "playwright-tests";
10
- var TRACER_VERSION = "0.1.0";
11
- var collector = null;
1
+ Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
2
+ let _playwright_test = require("@playwright/test");
3
+ let autotel = require("autotel");
4
+ let autotel_test_span_collector = require("autotel/test-span-collector");
5
+ let autotel_processors = require("autotel/processors");
6
+ //#region src/index.ts
7
+ /**
8
+ * autotel-playwright
9
+ *
10
+ * Playwright fixture that creates one OTel span per test and injects W3C trace
11
+ * context into requests to your API so "test → API" appears as one trace.
12
+ *
13
+ * @example
14
+ * // globalSetup.ts: init({ service: 'e2e-tests' });
15
+ * // In spec:
16
+ * import { test, expect } from 'autotel-playwright';
17
+ * test('checks health', async ({ page }) => {
18
+ * await page.goto(API_BASE_URL + '/health'); // request gets traceparent
19
+ * });
20
+ * // Node-side API calls with trace context:
21
+ * test('api health', async ({ requestWithTrace }) => {
22
+ * const res = await requestWithTrace.get(API_BASE_URL + '/health');
23
+ * expect(res.ok()).toBeTruthy();
24
+ * });
25
+ */
26
+ const TRACER_NAME = "playwright-tests";
27
+ const TRACER_VERSION = "0.1.0";
28
+ let collector = null;
12
29
  function ensureCollector() {
13
- if (!collector) {
14
- collector = new testSpanCollector.TestSpanCollector();
15
- const provider = autotel.getAutotelTracerProvider();
16
- if ("addSpanProcessor" in provider) {
17
- provider.addSpanProcessor(
18
- new processors.SimpleSpanProcessor(collector)
19
- );
20
- }
21
- }
22
- return collector;
30
+ if (!collector) {
31
+ collector = new autotel_test_span_collector.TestSpanCollector();
32
+ const provider = (0, autotel.getAutotelTracerProvider)();
33
+ if ("addSpanProcessor" in provider) provider.addSpanProcessor(new autotel_processors.SimpleSpanProcessor(collector));
34
+ }
35
+ return collector;
23
36
  }
24
- var ENV_API_BASE_URL = "API_BASE_URL";
25
- var ENV_API_ORIGIN = "AUTOTEL_PLAYWRIGHT_API_ORIGIN";
37
+ /** Env keys for API base URL (requests to this origin get trace context injected). */
38
+ const ENV_API_BASE_URL = "API_BASE_URL";
39
+ const ENV_API_ORIGIN = "AUTOTEL_PLAYWRIGHT_API_ORIGIN";
26
40
  function getApiBaseUrls() {
27
- const a = process.env[ENV_API_BASE_URL];
28
- const b = process.env[ENV_API_ORIGIN];
29
- const urls = [];
30
- if (a) urls.push(a.replace(/\/$/, ""));
31
- if (b) urls.push(b.replace(/\/$/, ""));
32
- return [...new Set(urls)];
41
+ const a = process.env[ENV_API_BASE_URL];
42
+ const b = process.env[ENV_API_ORIGIN];
43
+ const urls = [];
44
+ if (a) urls.push(a.replace(/\/$/, ""));
45
+ if (b) urls.push(b.replace(/\/$/, ""));
46
+ return [...new Set(urls)];
33
47
  }
48
+ /**
49
+ * Returns true if requestUrl should receive trace headers for the given apiBaseUrls.
50
+ * When a base URL includes a path (e.g. http://localhost:3000/api), only requests
51
+ * whose path starts with that path segment match; same-origin but different path
52
+ * (e.g. /health) must not match to avoid leaking trace context to unrelated endpoints.
53
+ */
34
54
  function urlMatchesApiOrigin(requestUrl, apiBaseUrls) {
35
- if (apiBaseUrls.length === 0) return false;
36
- try {
37
- const u = new URL(requestUrl);
38
- const requestOrigin = u.origin;
39
- const requestPathname = u.pathname;
40
- return apiBaseUrls.some((base2) => {
41
- try {
42
- const b = new URL(base2);
43
- if (requestOrigin !== b.origin) return false;
44
- const basePathname = b.pathname.replace(/\/$/, "") || "/";
45
- if (basePathname === "/") return true;
46
- return requestPathname === basePathname || requestPathname.startsWith(basePathname + "/");
47
- } catch {
48
- return requestUrl.startsWith(base2);
49
- }
50
- });
51
- } catch {
52
- return apiBaseUrls.some((base2) => requestUrl.startsWith(base2));
53
- }
55
+ if (apiBaseUrls.length === 0) return false;
56
+ try {
57
+ const u = new URL(requestUrl);
58
+ const requestOrigin = u.origin;
59
+ const requestPathname = u.pathname;
60
+ return apiBaseUrls.some((base) => {
61
+ try {
62
+ const b = new URL(base);
63
+ if (requestOrigin !== b.origin) return false;
64
+ const basePathname = b.pathname.replace(/\/$/, "") || "/";
65
+ if (basePathname === "/") return true;
66
+ return requestPathname === basePathname || requestPathname.startsWith(basePathname + "/");
67
+ } catch {
68
+ return requestUrl.startsWith(base);
69
+ }
70
+ });
71
+ } catch {
72
+ return apiBaseUrls.some((base) => requestUrl.startsWith(base));
73
+ }
54
74
  }
55
- var AUTOTEL_ATTRIBUTE_ANNOTATION = "autotel.attribute";
75
+ /** Annotation type for custom span attributes: description should be "key=value" or "key=value1;key2=value2". */
76
+ const AUTOTEL_ATTRIBUTE_ANNOTATION = "autotel.attribute";
56
77
  function setAttributesFromAnnotations(span, testInfo) {
57
- for (const a of testInfo.annotations) {
58
- if (a.type !== AUTOTEL_ATTRIBUTE_ANNOTATION || !a.description) continue;
59
- const entries = a.description.split(";");
60
- for (const entry of entries) {
61
- const parts = entry.split("=");
62
- if (parts.length >= 2) {
63
- const key = parts[0].trim();
64
- const value = parts.slice(1).join("=").trim();
65
- span.setAttribute(key, value);
66
- }
67
- }
68
- }
78
+ for (const a of testInfo.annotations) {
79
+ if (a.type !== "autotel.attribute" || !a.description) continue;
80
+ const entries = a.description.split(";");
81
+ for (const entry of entries) {
82
+ const parts = entry.split("=");
83
+ if (parts.length >= 2) {
84
+ const key = parts[0].trim();
85
+ const value = parts.slice(1).join("=").trim();
86
+ span.setAttribute(key, value);
87
+ }
88
+ }
89
+ }
69
90
  }
70
91
  function mergeTraceHeaders(url, options, apiBaseUrls, carrier, testName) {
71
- const opts = options ?? {};
72
- if (!urlMatchesApiOrigin(url, apiBaseUrls)) return opts;
73
- return {
74
- ...opts,
75
- headers: { ...opts.headers, ...carrier, "x-test-name": testName }
76
- };
92
+ const opts = options ?? {};
93
+ if (!urlMatchesApiOrigin(url, apiBaseUrls)) return opts;
94
+ return {
95
+ ...opts,
96
+ headers: {
97
+ ...opts.headers,
98
+ ...carrier,
99
+ "x-test-name": testName
100
+ }
101
+ };
77
102
  }
103
+ /** Wraps APIRequestContext so requests to API_BASE_URL get trace context injected. */
78
104
  function createRequestWithTrace(request, apiBaseUrls, carrier, testInfo) {
79
- const merge = (url, options) => mergeTraceHeaders(url, options, apiBaseUrls, carrier, testInfo.title);
80
- return {
81
- get: (url, options) => request.get(url, merge(url, options)),
82
- post: (url, options) => request.post(url, merge(url, options)),
83
- put: (url, options) => request.put(url, merge(url, options)),
84
- patch: (url, options) => request.patch(url, merge(url, options)),
85
- delete: (url, options) => request.delete(url, merge(url, options)),
86
- head: (url, options) => request.head(url, merge(url, options)),
87
- fetch: (urlOrRequest, options) => request.fetch(urlOrRequest, merge(typeof urlOrRequest === "string" ? urlOrRequest : urlOrRequest.url(), options)),
88
- storageState: (options) => request.storageState(options),
89
- dispose: () => request.dispose()
90
- };
105
+ const merge = (url, options) => mergeTraceHeaders(url, options, apiBaseUrls, carrier, testInfo.title);
106
+ return {
107
+ get: (url, options) => request.get(url, merge(url, options)),
108
+ post: (url, options) => request.post(url, merge(url, options)),
109
+ put: (url, options) => request.put(url, merge(url, options)),
110
+ patch: (url, options) => request.patch(url, merge(url, options)),
111
+ delete: (url, options) => request.delete(url, merge(url, options)),
112
+ head: (url, options) => request.head(url, merge(url, options)),
113
+ fetch: (urlOrRequest, options) => request.fetch(urlOrRequest, merge(typeof urlOrRequest === "string" ? urlOrRequest : urlOrRequest.url(), options)),
114
+ storageState: (options) => request.storageState(options),
115
+ dispose: () => request.dispose()
116
+ };
91
117
  }
92
- var test = test$1.test.extend({
93
- _otelTestSpan: [
94
- // eslint-disable-next-line no-empty-pattern
95
- async ({}, use, testInfo) => {
96
- ensureCollector();
97
- const apiBaseUrls = getApiBaseUrls();
98
- const tracer = autotel.getTracer(TRACER_NAME, TRACER_VERSION);
99
- const spanName = `e2e:${testInfo.title}`;
100
- const span = tracer.startSpan(spanName, {
101
- attributes: {
102
- "test.title": testInfo.title,
103
- "test.project": testInfo.project.name,
104
- "test.file": testInfo.file ?? "",
105
- "test.line": testInfo.line ?? 0
106
- }
107
- });
108
- setAttributesFromAnnotations(span, testInfo);
109
- const ctx = autotel.otelTrace.setSpan(autotel.context.active(), span);
110
- const carrier = {};
111
- autotel.context.with(ctx, () => {
112
- autotel.propagation.inject(autotel.context.active(), carrier);
113
- });
114
- try {
115
- await autotel.context.with(ctx, () => use({ carrier, apiBaseUrls, testInfo }));
116
- } catch (error) {
117
- span.setStatus({ code: autotel.SpanStatusCode.ERROR, message: error instanceof Error ? error.message : "Unknown error" });
118
- span.recordException(error instanceof Error ? error : new Error(String(error)));
119
- throw error;
120
- } finally {
121
- span.end();
122
- const traceId = span.spanContext().traceId;
123
- const rootSpanId = span.spanContext().spanId;
124
- const spans = collector.drainTrace(traceId, rootSpanId);
125
- if (spans.length > 0) {
126
- testInfo.annotations.push({
127
- type: "otel-spans",
128
- description: JSON.stringify(spans)
129
- });
130
- }
131
- }
132
- },
133
- { scope: "test" }
134
- ],
135
- page: async ({ page, _otelTestSpan }, use) => {
136
- const { carrier, apiBaseUrls, testInfo } = _otelTestSpan;
137
- if (apiBaseUrls.length > 0) {
138
- await page.route("**/*", async (route) => {
139
- const request = route.request();
140
- const url = request.url();
141
- if (urlMatchesApiOrigin(url, apiBaseUrls)) {
142
- const headers = {
143
- ...request.headers(),
144
- ...carrier,
145
- "x-test-name": testInfo.title
146
- };
147
- await route.continue({ headers });
148
- } else {
149
- await route.continue();
150
- }
151
- });
152
- }
153
- await use(page);
154
- },
155
- requestWithTrace: async ({ request, _otelTestSpan }, use) => {
156
- const wrapped = createRequestWithTrace(
157
- request,
158
- _otelTestSpan.apiBaseUrls,
159
- _otelTestSpan.carrier,
160
- _otelTestSpan.testInfo
161
- );
162
- await use(wrapped);
163
- }
118
+ const test = _playwright_test.test.extend({
119
+ _otelTestSpan: [async ({}, use, testInfo) => {
120
+ ensureCollector();
121
+ const apiBaseUrls = getApiBaseUrls();
122
+ const tracer = (0, autotel.getTracer)(TRACER_NAME, TRACER_VERSION);
123
+ const spanName = `e2e:${testInfo.title}`;
124
+ const span = tracer.startSpan(spanName, { attributes: {
125
+ "test.title": testInfo.title,
126
+ "test.project": testInfo.project.name,
127
+ "test.file": testInfo.file ?? "",
128
+ "test.line": testInfo.line ?? 0
129
+ } });
130
+ setAttributesFromAnnotations(span, testInfo);
131
+ const ctx = autotel.otelTrace.setSpan(autotel.context.active(), span);
132
+ const carrier = {};
133
+ autotel.context.with(ctx, () => {
134
+ autotel.propagation.inject(autotel.context.active(), carrier);
135
+ });
136
+ try {
137
+ await autotel.context.with(ctx, () => use({
138
+ carrier,
139
+ apiBaseUrls,
140
+ testInfo
141
+ }));
142
+ } catch (error) {
143
+ span.setStatus({
144
+ code: autotel.SpanStatusCode.ERROR,
145
+ message: error instanceof Error ? error.message : "Unknown error"
146
+ });
147
+ span.recordException(error instanceof Error ? error : new Error(String(error)));
148
+ throw error;
149
+ } finally {
150
+ span.end();
151
+ const traceId = span.spanContext().traceId;
152
+ const rootSpanId = span.spanContext().spanId;
153
+ const spans = collector.drainTrace(traceId, rootSpanId);
154
+ if (spans.length > 0) testInfo.annotations.push({
155
+ type: "otel-spans",
156
+ description: JSON.stringify(spans)
157
+ });
158
+ }
159
+ }, { scope: "test" }],
160
+ page: async ({ page, _otelTestSpan }, use) => {
161
+ const { carrier, apiBaseUrls, testInfo } = _otelTestSpan;
162
+ if (apiBaseUrls.length > 0) await page.route("**/*", async (route) => {
163
+ const request = route.request();
164
+ if (urlMatchesApiOrigin(request.url(), apiBaseUrls)) {
165
+ const headers = {
166
+ ...request.headers(),
167
+ ...carrier,
168
+ "x-test-name": testInfo.title
169
+ };
170
+ await route.continue({ headers });
171
+ } else await route.continue();
172
+ });
173
+ await use(page);
174
+ },
175
+ requestWithTrace: async ({ request, _otelTestSpan }, use) => {
176
+ await use(createRequestWithTrace(request, _otelTestSpan.apiBaseUrls, _otelTestSpan.carrier, _otelTestSpan.testInfo));
177
+ }
164
178
  });
179
+ /**
180
+ * Runs a named step as a child span of the current test span. Use inside a test to get
181
+ * step-level spans (e.g. "step:login", "step:navigate") under the test span in the same trace.
182
+ *
183
+ * @example
184
+ * test('user flow', async ({ page }) => {
185
+ * await step('login', async () => {
186
+ * await page.click('button[type=submit]');
187
+ * });
188
+ * await step('open profile', async () => {
189
+ * await page.goto('/profile');
190
+ * });
191
+ * });
192
+ */
165
193
  async function step(name, fn) {
166
- const tracer = autotel.getTracer(TRACER_NAME, TRACER_VERSION);
167
- const span = tracer.startSpan(`step:${name}`, {
168
- attributes: { "step.name": name }
169
- });
170
- try {
171
- return await autotel.context.with(autotel.otelTrace.setSpan(autotel.context.active(), span), fn);
172
- } catch (error) {
173
- span.setStatus({ code: autotel.SpanStatusCode.ERROR, message: error instanceof Error ? error.message : "Unknown error" });
174
- span.recordException(error instanceof Error ? error : new Error(String(error)));
175
- throw error;
176
- } finally {
177
- span.end();
178
- }
194
+ const span = (0, autotel.getTracer)(TRACER_NAME, TRACER_VERSION).startSpan(`step:${name}`, { attributes: { "step.name": name } });
195
+ try {
196
+ return await autotel.context.with(autotel.otelTrace.setSpan(autotel.context.active(), span), fn);
197
+ } catch (error) {
198
+ span.setStatus({
199
+ code: autotel.SpanStatusCode.ERROR,
200
+ message: error instanceof Error ? error.message : "Unknown error"
201
+ });
202
+ span.recordException(error instanceof Error ? error : new Error(String(error)));
203
+ throw error;
204
+ } finally {
205
+ span.end();
206
+ }
179
207
  }
208
+ /**
209
+ * Returns a function suitable for Playwright globalSetup that inits autotel.
210
+ * Call autotel.init() with the given options (or defaults) so test spans are exported.
211
+ */
180
212
  function createGlobalSetup(initOptions) {
181
- return async () => {
182
- const { init } = await import('autotel');
183
- init({
184
- service: "e2e-tests",
185
- debug: true,
186
- ...initOptions
187
- });
188
- };
213
+ return async () => {
214
+ const { init } = await import("autotel");
215
+ init({
216
+ service: "e2e-tests",
217
+ debug: true,
218
+ ...initOptions
219
+ });
220
+ };
189
221
  }
222
+ /**
223
+ * Creates a typed client for the test-spans HTTP endpoint.
224
+ *
225
+ * Pairs with `createTestSpansHandlers()` from `autotel-tanstack/testing`.
226
+ *
227
+ * @param baseUrl - Base URL of the app under test (e.g. 'http://localhost:3100')
228
+ * @param options.path - Path of the test-spans endpoint (default: '/api/test-spans')
229
+ *
230
+ * @example
231
+ * ```typescript
232
+ * const spansClient = createTestSpansClient('http://localhost:3100');
233
+ *
234
+ * test('server function is traced', async ({ request }) => {
235
+ * await spansClient.clearSpans(request);
236
+ * await page.goto('/');
237
+ * // ... trigger action ...
238
+ * const spans = await spansClient.getSpans(request);
239
+ * expect(spans.find(s => s.name === 'sendMoney.handler')).toBeDefined();
240
+ * });
241
+ * ```
242
+ */
190
243
  function createTestSpansClient(baseUrl, options) {
191
- const base2 = baseUrl.replace(/\/$/, "");
192
- const path = options?.path ?? "/api/test-spans";
193
- const url = `${base2}${path}`;
194
- return {
195
- async getSpans(request) {
196
- const res = await request.get(url);
197
- if (!res.ok()) {
198
- throw new Error(`GET ${path} failed: ${res.status()}`);
199
- }
200
- const body = await res.json();
201
- return body.spans;
202
- },
203
- async clearSpans(request) {
204
- const res = await request.delete(url);
205
- if (!res.ok()) {
206
- throw new Error(`DELETE ${path} failed: ${res.status()}`);
207
- }
208
- }
209
- };
244
+ const base = baseUrl.replace(/\/$/, "");
245
+ const path = options?.path ?? "/api/test-spans";
246
+ const url = `${base}${path}`;
247
+ return {
248
+ async getSpans(request) {
249
+ const res = await request.get(url);
250
+ if (!res.ok()) throw new Error(`GET ${path} failed: ${res.status()}`);
251
+ return (await res.json()).spans;
252
+ },
253
+ async clearSpans(request) {
254
+ const res = await request.delete(url);
255
+ if (!res.ok()) throw new Error(`DELETE ${path} failed: ${res.status()}`);
256
+ }
257
+ };
210
258
  }
211
-
212
- Object.defineProperty(exports, "expect", {
213
- enumerable: true,
214
- get: function () { return test$1.expect; }
215
- });
259
+ //#endregion
260
+ exports.AUTOTEL_ATTRIBUTE_ANNOTATION = AUTOTEL_ATTRIBUTE_ANNOTATION;
261
+ exports.createGlobalSetup = createGlobalSetup;
262
+ exports.createTestSpansClient = createTestSpansClient;
216
263
  Object.defineProperty(exports, "enrichWithTraceContext", {
217
- enumerable: true,
218
- get: function () { return autotel.enrichWithTraceContext; }
264
+ enumerable: true,
265
+ get: function() {
266
+ return autotel.enrichWithTraceContext;
267
+ }
268
+ });
269
+ Object.defineProperty(exports, "expect", {
270
+ enumerable: true,
271
+ get: function() {
272
+ return _playwright_test.expect;
273
+ }
219
274
  });
220
275
  Object.defineProperty(exports, "getTraceContext", {
221
- enumerable: true,
222
- get: function () { return autotel.getTraceContext; }
276
+ enumerable: true,
277
+ get: function() {
278
+ return autotel.getTraceContext;
279
+ }
223
280
  });
224
281
  Object.defineProperty(exports, "isTracing", {
225
- enumerable: true,
226
- get: function () { return autotel.isTracing; }
282
+ enumerable: true,
283
+ get: function() {
284
+ return autotel.isTracing;
285
+ }
227
286
  });
228
287
  Object.defineProperty(exports, "resolveTraceUrl", {
229
- enumerable: true,
230
- get: function () { return autotel.resolveTraceUrl; }
288
+ enumerable: true,
289
+ get: function() {
290
+ return autotel.resolveTraceUrl;
291
+ }
231
292
  });
232
- exports.AUTOTEL_ATTRIBUTE_ANNOTATION = AUTOTEL_ATTRIBUTE_ANNOTATION;
233
- exports.createGlobalSetup = createGlobalSetup;
234
- exports.createTestSpansClient = createTestSpansClient;
235
293
  exports.step = step;
236
294
  exports.test = test;
237
- //# sourceMappingURL=index.cjs.map
295
+
238
296
  //# sourceMappingURL=index.cjs.map
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/index.ts"],"names":["TestSpanCollector","getAutotelTracerProvider","SimpleSpanProcessor","base","getTracer","otelTrace","otelContext","propagation","SpanStatusCode"],"mappings":"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","file":"index.cjs","sourcesContent":["/**\n * autotel-playwright\n *\n * Playwright fixture that creates one OTel span per test and injects W3C trace\n * context into requests to your API so \"test → API\" appears as one trace.\n *\n * @example\n * // globalSetup.ts: init({ service: 'e2e-tests' });\n * // In spec:\n * import { test, expect } from 'autotel-playwright';\n * test('checks health', async ({ page }) => {\n * await page.goto(API_BASE_URL + '/health'); // request gets traceparent\n * });\n * // Node-side API calls with trace context:\n * test('api health', async ({ requestWithTrace }) => {\n * const res = await requestWithTrace.get(API_BASE_URL + '/health');\n * expect(res.ok()).toBeTruthy();\n * });\n */\n\nimport { test as base } from '@playwright/test';\nimport type { Page, APIRequestContext, Request as PlaywrightRequest } from '@playwright/test';\nimport type { TestInfo } from '@playwright/test';\nimport type { AutotelConfig } from 'autotel';\nimport {\n getTracer,\n getAutotelTracerProvider,\n context as otelContext,\n propagation,\n otelTrace,\n SpanStatusCode,\n} from 'autotel';\nimport { TestSpanCollector } from 'autotel/test-span-collector';\nimport { SimpleSpanProcessor } from 'autotel/processors';\n\nconst TRACER_NAME = 'playwright-tests';\nconst TRACER_VERSION = '0.1.0';\n\nlet collector: TestSpanCollector | null = null;\n\ninterface TracerProviderWithProcessor {\n addSpanProcessor(processor: unknown): void;\n}\n\nfunction ensureCollector(): TestSpanCollector {\n if (!collector) {\n collector = new TestSpanCollector();\n const provider = getAutotelTracerProvider();\n if ('addSpanProcessor' in provider) {\n (provider as TracerProviderWithProcessor).addSpanProcessor(\n new SimpleSpanProcessor(collector),\n );\n }\n }\n return collector;\n}\n\n/** Env keys for API base URL (requests to this origin get trace context injected). */\nconst ENV_API_BASE_URL = 'API_BASE_URL';\nconst ENV_API_ORIGIN = 'AUTOTEL_PLAYWRIGHT_API_ORIGIN';\n\nfunction getApiBaseUrls(): string[] {\n const a = process.env[ENV_API_BASE_URL];\n const b = process.env[ENV_API_ORIGIN];\n const urls: string[] = [];\n if (a) urls.push(a.replace(/\\/$/, ''));\n if (b) urls.push(b.replace(/\\/$/, ''));\n return [...new Set(urls)];\n}\n\n/**\n * Returns true if requestUrl should receive trace headers for the given apiBaseUrls.\n * When a base URL includes a path (e.g. http://localhost:3000/api), only requests\n * whose path starts with that path segment match; same-origin but different path\n * (e.g. /health) must not match to avoid leaking trace context to unrelated endpoints.\n */\nfunction urlMatchesApiOrigin(requestUrl: string, apiBaseUrls: string[]): boolean {\n if (apiBaseUrls.length === 0) return false;\n try {\n const u = new URL(requestUrl);\n const requestOrigin = u.origin;\n const requestPathname = u.pathname;\n return apiBaseUrls.some((base) => {\n try {\n const b = new URL(base);\n if (requestOrigin !== b.origin) return false;\n const basePathname = b.pathname.replace(/\\/$/, '') || '/';\n if (basePathname === '/') return true;\n return (\n requestPathname === basePathname || requestPathname.startsWith(basePathname + '/')\n );\n } catch {\n return requestUrl.startsWith(base);\n }\n });\n } catch {\n return apiBaseUrls.some((base) => requestUrl.startsWith(base));\n }\n}\n\n/** Annotation type for custom span attributes: description should be \"key=value\" or \"key=value1;key2=value2\". */\nexport const AUTOTEL_ATTRIBUTE_ANNOTATION = 'autotel.attribute';\n\nfunction setAttributesFromAnnotations(\n span: { setAttribute: (k: string, v: string | number | boolean) => void },\n testInfo: { annotations: Array<{ type: string; description?: string }> },\n): void {\n for (const a of testInfo.annotations) {\n if (a.type !== AUTOTEL_ATTRIBUTE_ANNOTATION || !a.description) continue;\n const entries = a.description.split(';');\n for (const entry of entries) {\n const parts = entry.split('=');\n if (parts.length >= 2) {\n const key = parts[0].trim();\n const value = parts.slice(1).join('=').trim();\n span.setAttribute(key, value);\n }\n }\n }\n}\n\n/** Internal: options for get/post/put/patch/delete/head/fetch that may include headers. */\ntype RequestOptions = Record<string, unknown> & { headers?: Record<string, string> };\n\nfunction mergeTraceHeaders(\n url: string,\n options: RequestOptions | undefined,\n apiBaseUrls: string[],\n carrier: Record<string, string>,\n testName: string,\n): RequestOptions {\n const opts = options ?? {};\n if (!urlMatchesApiOrigin(url, apiBaseUrls)) return opts;\n return {\n ...opts,\n headers: { ...(opts.headers as Record<string, string>), ...carrier, 'x-test-name': testName },\n };\n}\n\n/** Wraps APIRequestContext so requests to API_BASE_URL get trace context injected. */\nfunction createRequestWithTrace(\n request: APIRequestContext,\n apiBaseUrls: string[],\n carrier: Record<string, string>,\n testInfo: TestInfo,\n): APIRequestContext {\n const merge = (url: string, options?: RequestOptions) =>\n mergeTraceHeaders(url, options, apiBaseUrls, carrier, testInfo.title);\n\n return {\n get: (url: string, options?: RequestOptions) => request.get(url, merge(url, options)),\n post: (url: string, options?: RequestOptions) => request.post(url, merge(url, options)),\n put: (url: string, options?: RequestOptions) => request.put(url, merge(url, options)),\n patch: (url: string, options?: RequestOptions) => request.patch(url, merge(url, options)),\n delete: (url: string, options?: RequestOptions) => request.delete(url, merge(url, options)),\n head: (url: string, options?: RequestOptions) => request.head(url, merge(url, options)),\n fetch: (urlOrRequest: string | PlaywrightRequest, options?: RequestOptions) =>\n request.fetch(urlOrRequest, merge(typeof urlOrRequest === 'string' ? urlOrRequest : urlOrRequest.url(), options)),\n storageState: (options?: { path?: string }) => request.storageState(options),\n dispose: () => request.dispose(),\n } as APIRequestContext;\n}\n\ntype OtelTestSpan = {\n carrier: Record<string, string>;\n apiBaseUrls: string[];\n testInfo: TestInfo;\n};\n\nexport const test = base.extend<{\n page: Page;\n requestWithTrace: APIRequestContext;\n _otelTestSpan: OtelTestSpan;\n}>({\n _otelTestSpan: [\n // eslint-disable-next-line no-empty-pattern\n async ({}, use, testInfo) => {\n ensureCollector();\n const apiBaseUrls = getApiBaseUrls();\n const tracer = getTracer(TRACER_NAME, TRACER_VERSION);\n const spanName = `e2e:${testInfo.title}`;\n const span = tracer.startSpan(spanName, {\n attributes: {\n 'test.title': testInfo.title,\n 'test.project': testInfo.project.name,\n 'test.file': testInfo.file ?? '',\n 'test.line': testInfo.line ?? 0,\n },\n });\n setAttributesFromAnnotations(span, testInfo);\n const ctx = otelTrace.setSpan(otelContext.active(), span);\n const carrier: Record<string, string> = {};\n otelContext.with(ctx, () => {\n propagation.inject(otelContext.active(), carrier);\n });\n try {\n await otelContext.with(ctx, () => use({ carrier, apiBaseUrls, testInfo }));\n } catch (error) {\n span.setStatus({ code: SpanStatusCode.ERROR, message: error instanceof Error ? error.message : 'Unknown error' });\n span.recordException(error instanceof Error ? error : new Error(String(error)));\n throw error;\n } finally {\n span.end();\n const traceId = span.spanContext().traceId;\n const rootSpanId = span.spanContext().spanId;\n const spans = collector!.drainTrace(traceId, rootSpanId);\n if (spans.length > 0) {\n testInfo.annotations.push({\n type: 'otel-spans',\n description: JSON.stringify(spans),\n });\n }\n }\n },\n { scope: 'test' },\n ],\n\n page: async ({ page, _otelTestSpan }, use) => {\n const { carrier, apiBaseUrls, testInfo } = _otelTestSpan;\n if (apiBaseUrls.length > 0) {\n await page.route('**/*', async (route) => {\n const request = route.request();\n const url = request.url();\n if (urlMatchesApiOrigin(url, apiBaseUrls)) {\n const headers = {\n ...request.headers(),\n ...carrier,\n 'x-test-name': testInfo.title,\n };\n await route.continue({ headers });\n } else {\n await route.continue();\n }\n });\n }\n await use(page);\n },\n\n requestWithTrace: async ({ request, _otelTestSpan }, use) => {\n const wrapped = createRequestWithTrace(\n request,\n _otelTestSpan.apiBaseUrls,\n _otelTestSpan.carrier,\n _otelTestSpan.testInfo,\n );\n await use(wrapped);\n },\n});\n\nexport { expect } from '@playwright/test';\n\n// Re-export trace context helpers for DX convenience\nexport {\n getTraceContext,\n resolveTraceUrl,\n isTracing,\n enrichWithTraceContext,\n} from 'autotel';\n\nexport type { OtelTraceContext } from 'autotel';\n\n/**\n * Runs a named step as a child span of the current test span. Use inside a test to get\n * step-level spans (e.g. \"step:login\", \"step:navigate\") under the test span in the same trace.\n *\n * @example\n * test('user flow', async ({ page }) => {\n * await step('login', async () => {\n * await page.click('button[type=submit]');\n * });\n * await step('open profile', async () => {\n * await page.goto('/profile');\n * });\n * });\n */\nexport async function step<T>(name: string, fn: () => Promise<T>): Promise<T> {\n const tracer = getTracer(TRACER_NAME, TRACER_VERSION);\n const span = tracer.startSpan(`step:${name}`, {\n attributes: { 'step.name': name },\n });\n try {\n return await otelContext.with(otelTrace.setSpan(otelContext.active(), span), fn);\n } catch (error) {\n span.setStatus({ code: SpanStatusCode.ERROR, message: error instanceof Error ? error.message : 'Unknown error' });\n span.recordException(error instanceof Error ? error : new Error(String(error)));\n throw error;\n } finally {\n span.end();\n }\n}\n\n/**\n * Returns a function suitable for Playwright globalSetup that inits autotel.\n * Call autotel.init() with the given options (or defaults) so test spans are exported.\n */\nexport function createGlobalSetup(initOptions?: AutotelConfig): () => Promise<void> {\n return async () => {\n const { init } = await import('autotel');\n init({\n service: 'e2e-tests',\n debug: true,\n ...initOptions,\n });\n };\n}\n\n/**\n * Serialized span returned by the test-spans endpoint (matches autotel-tanstack/testing SerializedSpan).\n */\nexport interface SerializedSpan {\n name: string;\n spanId: string;\n traceId: string;\n parentSpanId?: string;\n attributes?: Record<string, unknown>;\n status: { code: number; message?: string };\n durationMs: number;\n}\n\n/**\n * Creates a typed client for the test-spans HTTP endpoint.\n *\n * Pairs with `createTestSpansHandlers()` from `autotel-tanstack/testing`.\n *\n * @param baseUrl - Base URL of the app under test (e.g. 'http://localhost:3100')\n * @param options.path - Path of the test-spans endpoint (default: '/api/test-spans')\n *\n * @example\n * ```typescript\n * const spansClient = createTestSpansClient('http://localhost:3100');\n *\n * test('server function is traced', async ({ request }) => {\n * await spansClient.clearSpans(request);\n * await page.goto('/');\n * // ... trigger action ...\n * const spans = await spansClient.getSpans(request);\n * expect(spans.find(s => s.name === 'sendMoney.handler')).toBeDefined();\n * });\n * ```\n */\nexport function createTestSpansClient(\n baseUrl: string,\n options?: { path?: string },\n): {\n getSpans(request: APIRequestContext): Promise<SerializedSpan[]>;\n clearSpans(request: APIRequestContext): Promise<void>;\n} {\n const base = baseUrl.replace(/\\/$/, '');\n const path = options?.path ?? '/api/test-spans';\n const url = `${base}${path}`;\n\n return {\n async getSpans(request: APIRequestContext): Promise<SerializedSpan[]> {\n const res = await request.get(url);\n if (!res.ok()) {\n throw new Error(`GET ${path} failed: ${res.status()}`);\n }\n const body = await res.json() as { spans: SerializedSpan[] };\n return body.spans;\n },\n\n async clearSpans(request: APIRequestContext): Promise<void> {\n const res = await request.delete(url);\n if (!res.ok()) {\n throw new Error(`DELETE ${path} failed: ${res.status()}`);\n }\n },\n };\n}\n"]}
1
+ {"version":3,"file":"index.cjs","names":["TestSpanCollector","SimpleSpanProcessor","base","otelTrace","otelContext","SpanStatusCode"],"sources":["../src/index.ts"],"sourcesContent":["/**\n * autotel-playwright\n *\n * Playwright fixture that creates one OTel span per test and injects W3C trace\n * context into requests to your API so \"test → API\" appears as one trace.\n *\n * @example\n * // globalSetup.ts: init({ service: 'e2e-tests' });\n * // In spec:\n * import { test, expect } from 'autotel-playwright';\n * test('checks health', async ({ page }) => {\n * await page.goto(API_BASE_URL + '/health'); // request gets traceparent\n * });\n * // Node-side API calls with trace context:\n * test('api health', async ({ requestWithTrace }) => {\n * const res = await requestWithTrace.get(API_BASE_URL + '/health');\n * expect(res.ok()).toBeTruthy();\n * });\n */\n\nimport { test as base } from '@playwright/test';\nimport type { Page, APIRequestContext, Request as PlaywrightRequest } from '@playwright/test';\nimport type { TestInfo } from '@playwright/test';\nimport type { AutotelConfig } from 'autotel';\nimport {\n getTracer,\n getAutotelTracerProvider,\n context as otelContext,\n propagation,\n otelTrace,\n SpanStatusCode,\n} from 'autotel';\nimport { TestSpanCollector } from 'autotel/test-span-collector';\nimport { SimpleSpanProcessor } from 'autotel/processors';\n\nconst TRACER_NAME = 'playwright-tests';\nconst TRACER_VERSION = '0.1.0';\n\nlet collector: TestSpanCollector | null = null;\n\ninterface TracerProviderWithProcessor {\n addSpanProcessor(processor: unknown): void;\n}\n\nfunction ensureCollector(): TestSpanCollector {\n if (!collector) {\n collector = new TestSpanCollector();\n const provider = getAutotelTracerProvider();\n if ('addSpanProcessor' in provider) {\n (provider as TracerProviderWithProcessor).addSpanProcessor(\n new SimpleSpanProcessor(collector),\n );\n }\n }\n return collector;\n}\n\n/** Env keys for API base URL (requests to this origin get trace context injected). */\nconst ENV_API_BASE_URL = 'API_BASE_URL';\nconst ENV_API_ORIGIN = 'AUTOTEL_PLAYWRIGHT_API_ORIGIN';\n\nfunction getApiBaseUrls(): string[] {\n const a = process.env[ENV_API_BASE_URL];\n const b = process.env[ENV_API_ORIGIN];\n const urls: string[] = [];\n if (a) urls.push(a.replace(/\\/$/, ''));\n if (b) urls.push(b.replace(/\\/$/, ''));\n return [...new Set(urls)];\n}\n\n/**\n * Returns true if requestUrl should receive trace headers for the given apiBaseUrls.\n * When a base URL includes a path (e.g. http://localhost:3000/api), only requests\n * whose path starts with that path segment match; same-origin but different path\n * (e.g. /health) must not match to avoid leaking trace context to unrelated endpoints.\n */\nfunction urlMatchesApiOrigin(requestUrl: string, apiBaseUrls: string[]): boolean {\n if (apiBaseUrls.length === 0) return false;\n try {\n const u = new URL(requestUrl);\n const requestOrigin = u.origin;\n const requestPathname = u.pathname;\n return apiBaseUrls.some((base) => {\n try {\n const b = new URL(base);\n if (requestOrigin !== b.origin) return false;\n const basePathname = b.pathname.replace(/\\/$/, '') || '/';\n if (basePathname === '/') return true;\n return (\n requestPathname === basePathname || requestPathname.startsWith(basePathname + '/')\n );\n } catch {\n return requestUrl.startsWith(base);\n }\n });\n } catch {\n return apiBaseUrls.some((base) => requestUrl.startsWith(base));\n }\n}\n\n/** Annotation type for custom span attributes: description should be \"key=value\" or \"key=value1;key2=value2\". */\nexport const AUTOTEL_ATTRIBUTE_ANNOTATION = 'autotel.attribute';\n\nfunction setAttributesFromAnnotations(\n span: { setAttribute: (k: string, v: string | number | boolean) => void },\n testInfo: { annotations: Array<{ type: string; description?: string }> },\n): void {\n for (const a of testInfo.annotations) {\n if (a.type !== AUTOTEL_ATTRIBUTE_ANNOTATION || !a.description) continue;\n const entries = a.description.split(';');\n for (const entry of entries) {\n const parts = entry.split('=');\n if (parts.length >= 2) {\n const key = parts[0].trim();\n const value = parts.slice(1).join('=').trim();\n span.setAttribute(key, value);\n }\n }\n }\n}\n\n/** Internal: options for get/post/put/patch/delete/head/fetch that may include headers. */\ntype RequestOptions = Record<string, unknown> & { headers?: Record<string, string> };\n\nfunction mergeTraceHeaders(\n url: string,\n options: RequestOptions | undefined,\n apiBaseUrls: string[],\n carrier: Record<string, string>,\n testName: string,\n): RequestOptions {\n const opts = options ?? {};\n if (!urlMatchesApiOrigin(url, apiBaseUrls)) return opts;\n return {\n ...opts,\n headers: { ...(opts.headers as Record<string, string>), ...carrier, 'x-test-name': testName },\n };\n}\n\n/** Wraps APIRequestContext so requests to API_BASE_URL get trace context injected. */\nfunction createRequestWithTrace(\n request: APIRequestContext,\n apiBaseUrls: string[],\n carrier: Record<string, string>,\n testInfo: TestInfo,\n): APIRequestContext {\n const merge = (url: string, options?: RequestOptions) =>\n mergeTraceHeaders(url, options, apiBaseUrls, carrier, testInfo.title);\n\n return {\n get: (url: string, options?: RequestOptions) => request.get(url, merge(url, options)),\n post: (url: string, options?: RequestOptions) => request.post(url, merge(url, options)),\n put: (url: string, options?: RequestOptions) => request.put(url, merge(url, options)),\n patch: (url: string, options?: RequestOptions) => request.patch(url, merge(url, options)),\n delete: (url: string, options?: RequestOptions) => request.delete(url, merge(url, options)),\n head: (url: string, options?: RequestOptions) => request.head(url, merge(url, options)),\n fetch: (urlOrRequest: string | PlaywrightRequest, options?: RequestOptions) =>\n request.fetch(urlOrRequest, merge(typeof urlOrRequest === 'string' ? urlOrRequest : urlOrRequest.url(), options)),\n storageState: (options?: { path?: string }) => request.storageState(options),\n dispose: () => request.dispose(),\n } as APIRequestContext;\n}\n\ntype OtelTestSpan = {\n carrier: Record<string, string>;\n apiBaseUrls: string[];\n testInfo: TestInfo;\n};\n\nexport const test = base.extend<{\n page: Page;\n requestWithTrace: APIRequestContext;\n _otelTestSpan: OtelTestSpan;\n}>({\n _otelTestSpan: [\n // eslint-disable-next-line no-empty-pattern\n async ({}, use, testInfo) => {\n ensureCollector();\n const apiBaseUrls = getApiBaseUrls();\n const tracer = getTracer(TRACER_NAME, TRACER_VERSION);\n const spanName = `e2e:${testInfo.title}`;\n const span = tracer.startSpan(spanName, {\n attributes: {\n 'test.title': testInfo.title,\n 'test.project': testInfo.project.name,\n 'test.file': testInfo.file ?? '',\n 'test.line': testInfo.line ?? 0,\n },\n });\n setAttributesFromAnnotations(span, testInfo);\n const ctx = otelTrace.setSpan(otelContext.active(), span);\n const carrier: Record<string, string> = {};\n otelContext.with(ctx, () => {\n propagation.inject(otelContext.active(), carrier);\n });\n try {\n await otelContext.with(ctx, () => use({ carrier, apiBaseUrls, testInfo }));\n } catch (error) {\n span.setStatus({ code: SpanStatusCode.ERROR, message: error instanceof Error ? error.message : 'Unknown error' });\n span.recordException(error instanceof Error ? error : new Error(String(error)));\n throw error;\n } finally {\n span.end();\n const traceId = span.spanContext().traceId;\n const rootSpanId = span.spanContext().spanId;\n const spans = collector!.drainTrace(traceId, rootSpanId);\n if (spans.length > 0) {\n testInfo.annotations.push({\n type: 'otel-spans',\n description: JSON.stringify(spans),\n });\n }\n }\n },\n { scope: 'test' },\n ],\n\n page: async ({ page, _otelTestSpan }, use) => {\n const { carrier, apiBaseUrls, testInfo } = _otelTestSpan;\n if (apiBaseUrls.length > 0) {\n await page.route('**/*', async (route) => {\n const request = route.request();\n const url = request.url();\n if (urlMatchesApiOrigin(url, apiBaseUrls)) {\n const headers = {\n ...request.headers(),\n ...carrier,\n 'x-test-name': testInfo.title,\n };\n await route.continue({ headers });\n } else {\n await route.continue();\n }\n });\n }\n await use(page);\n },\n\n requestWithTrace: async ({ request, _otelTestSpan }, use) => {\n const wrapped = createRequestWithTrace(\n request,\n _otelTestSpan.apiBaseUrls,\n _otelTestSpan.carrier,\n _otelTestSpan.testInfo,\n );\n await use(wrapped);\n },\n});\n\nexport { expect } from '@playwright/test';\n\n// Re-export trace context helpers for DX convenience\nexport {\n getTraceContext,\n resolveTraceUrl,\n isTracing,\n enrichWithTraceContext,\n} from 'autotel';\n\nexport type { OtelTraceContext } from 'autotel';\n\n/**\n * Runs a named step as a child span of the current test span. Use inside a test to get\n * step-level spans (e.g. \"step:login\", \"step:navigate\") under the test span in the same trace.\n *\n * @example\n * test('user flow', async ({ page }) => {\n * await step('login', async () => {\n * await page.click('button[type=submit]');\n * });\n * await step('open profile', async () => {\n * await page.goto('/profile');\n * });\n * });\n */\nexport async function step<T>(name: string, fn: () => Promise<T>): Promise<T> {\n const tracer = getTracer(TRACER_NAME, TRACER_VERSION);\n const span = tracer.startSpan(`step:${name}`, {\n attributes: { 'step.name': name },\n });\n try {\n return await otelContext.with(otelTrace.setSpan(otelContext.active(), span), fn);\n } catch (error) {\n span.setStatus({ code: SpanStatusCode.ERROR, message: error instanceof Error ? error.message : 'Unknown error' });\n span.recordException(error instanceof Error ? error : new Error(String(error)));\n throw error;\n } finally {\n span.end();\n }\n}\n\n/**\n * Returns a function suitable for Playwright globalSetup that inits autotel.\n * Call autotel.init() with the given options (or defaults) so test spans are exported.\n */\nexport function createGlobalSetup(initOptions?: AutotelConfig): () => Promise<void> {\n return async () => {\n const { init } = await import('autotel');\n init({\n service: 'e2e-tests',\n debug: true,\n ...initOptions,\n });\n };\n}\n\n/**\n * Serialized span returned by the test-spans endpoint (matches autotel-tanstack/testing SerializedSpan).\n */\nexport interface SerializedSpan {\n name: string;\n spanId: string;\n traceId: string;\n parentSpanId?: string;\n attributes?: Record<string, unknown>;\n status: { code: number; message?: string };\n durationMs: number;\n}\n\n/**\n * Creates a typed client for the test-spans HTTP endpoint.\n *\n * Pairs with `createTestSpansHandlers()` from `autotel-tanstack/testing`.\n *\n * @param baseUrl - Base URL of the app under test (e.g. 'http://localhost:3100')\n * @param options.path - Path of the test-spans endpoint (default: '/api/test-spans')\n *\n * @example\n * ```typescript\n * const spansClient = createTestSpansClient('http://localhost:3100');\n *\n * test('server function is traced', async ({ request }) => {\n * await spansClient.clearSpans(request);\n * await page.goto('/');\n * // ... trigger action ...\n * const spans = await spansClient.getSpans(request);\n * expect(spans.find(s => s.name === 'sendMoney.handler')).toBeDefined();\n * });\n * ```\n */\nexport function createTestSpansClient(\n baseUrl: string,\n options?: { path?: string },\n): {\n getSpans(request: APIRequestContext): Promise<SerializedSpan[]>;\n clearSpans(request: APIRequestContext): Promise<void>;\n} {\n const base = baseUrl.replace(/\\/$/, '');\n const path = options?.path ?? '/api/test-spans';\n const url = `${base}${path}`;\n\n return {\n async getSpans(request: APIRequestContext): Promise<SerializedSpan[]> {\n const res = await request.get(url);\n if (!res.ok()) {\n throw new Error(`GET ${path} failed: ${res.status()}`);\n }\n const body = await res.json() as { spans: SerializedSpan[] };\n return body.spans;\n },\n\n async clearSpans(request: APIRequestContext): Promise<void> {\n const res = await request.delete(url);\n if (!res.ok()) {\n throw new Error(`DELETE ${path} failed: ${res.status()}`);\n }\n },\n };\n}\n"],"mappings":"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"}