bugcatch-sdk 0.1.8 → 0.1.10
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.d.mts +28 -0
- package/dist/index.d.ts +28 -0
- package/dist/index.js +153 -2
- package/dist/index.js.map +1 -1
- package/dist/index.mjs +153 -2
- package/dist/index.mjs.map +1 -1
- package/package.json +1 -1
package/dist/index.d.mts
CHANGED
|
@@ -19,6 +19,16 @@ interface BugCatchOptions {
|
|
|
19
19
|
* Default: true.
|
|
20
20
|
*/
|
|
21
21
|
autoCaptureBreadcrumbs?: boolean;
|
|
22
|
+
/**
|
|
23
|
+
* Automatically intercept fetch() calls and report API timing metrics
|
|
24
|
+
* to BugCatch. Default: false.
|
|
25
|
+
*/
|
|
26
|
+
autoTrackRequests?: boolean;
|
|
27
|
+
/**
|
|
28
|
+
* URL patterns to exclude from request tracking. The BugCatch ingest URL
|
|
29
|
+
* is always excluded automatically.
|
|
30
|
+
*/
|
|
31
|
+
trackIgnoreUrls?: Array<string | RegExp>;
|
|
22
32
|
/**
|
|
23
33
|
* A list of URL patterns (strings or RegExps). Any error originating from
|
|
24
34
|
* a script URL that matches will be ignored.
|
|
@@ -103,11 +113,19 @@ declare class BugCatchClient {
|
|
|
103
113
|
addBreadcrumb(crumb: BreadcrumbEntry): void;
|
|
104
114
|
/** Tear down all global listeners. Call when unmounting in SPAs. */
|
|
105
115
|
destroy(): void;
|
|
116
|
+
/**
|
|
117
|
+
* Manually report an API call timing. Use this when `autoTrackRequests`
|
|
118
|
+
* is off or when you want to track server-side (Node.js) requests explicitly.
|
|
119
|
+
*/
|
|
120
|
+
trackRequest(method: string, route: string, durationMs: number, statusCode: number): void;
|
|
106
121
|
private buildExceptionPayload;
|
|
107
122
|
private buildMessagePayload;
|
|
108
123
|
private buildBase;
|
|
109
124
|
private installErrorHandlers;
|
|
110
125
|
private shouldIgnore;
|
|
126
|
+
private installFetchInterceptor;
|
|
127
|
+
private installXhrInterceptor;
|
|
128
|
+
private sendMetric;
|
|
111
129
|
private send;
|
|
112
130
|
private log;
|
|
113
131
|
}
|
|
@@ -150,6 +168,16 @@ declare const BugCatch: {
|
|
|
150
168
|
* Manually add a breadcrumb.
|
|
151
169
|
*/
|
|
152
170
|
addBreadcrumb(crumb: BreadcrumbEntry): void;
|
|
171
|
+
/**
|
|
172
|
+
* Manually report an API call timing to BugCatch.
|
|
173
|
+
* Use when `autoTrackRequests` is off or for server-side tracking.
|
|
174
|
+
*
|
|
175
|
+
* @example
|
|
176
|
+
* const start = Date.now();
|
|
177
|
+
* await fetch('/api/orders');
|
|
178
|
+
* BugCatch.trackRequest('GET', '/api/orders', Date.now() - start, 200);
|
|
179
|
+
*/
|
|
180
|
+
trackRequest(method: string, route: string, durationMs: number, statusCode: number): void;
|
|
153
181
|
/**
|
|
154
182
|
* Tear down all listeners. Useful in SPA cleanup or hot-reload scenarios.
|
|
155
183
|
*/
|
package/dist/index.d.ts
CHANGED
|
@@ -19,6 +19,16 @@ interface BugCatchOptions {
|
|
|
19
19
|
* Default: true.
|
|
20
20
|
*/
|
|
21
21
|
autoCaptureBreadcrumbs?: boolean;
|
|
22
|
+
/**
|
|
23
|
+
* Automatically intercept fetch() calls and report API timing metrics
|
|
24
|
+
* to BugCatch. Default: false.
|
|
25
|
+
*/
|
|
26
|
+
autoTrackRequests?: boolean;
|
|
27
|
+
/**
|
|
28
|
+
* URL patterns to exclude from request tracking. The BugCatch ingest URL
|
|
29
|
+
* is always excluded automatically.
|
|
30
|
+
*/
|
|
31
|
+
trackIgnoreUrls?: Array<string | RegExp>;
|
|
22
32
|
/**
|
|
23
33
|
* A list of URL patterns (strings or RegExps). Any error originating from
|
|
24
34
|
* a script URL that matches will be ignored.
|
|
@@ -103,11 +113,19 @@ declare class BugCatchClient {
|
|
|
103
113
|
addBreadcrumb(crumb: BreadcrumbEntry): void;
|
|
104
114
|
/** Tear down all global listeners. Call when unmounting in SPAs. */
|
|
105
115
|
destroy(): void;
|
|
116
|
+
/**
|
|
117
|
+
* Manually report an API call timing. Use this when `autoTrackRequests`
|
|
118
|
+
* is off or when you want to track server-side (Node.js) requests explicitly.
|
|
119
|
+
*/
|
|
120
|
+
trackRequest(method: string, route: string, durationMs: number, statusCode: number): void;
|
|
106
121
|
private buildExceptionPayload;
|
|
107
122
|
private buildMessagePayload;
|
|
108
123
|
private buildBase;
|
|
109
124
|
private installErrorHandlers;
|
|
110
125
|
private shouldIgnore;
|
|
126
|
+
private installFetchInterceptor;
|
|
127
|
+
private installXhrInterceptor;
|
|
128
|
+
private sendMetric;
|
|
111
129
|
private send;
|
|
112
130
|
private log;
|
|
113
131
|
}
|
|
@@ -150,6 +168,16 @@ declare const BugCatch: {
|
|
|
150
168
|
* Manually add a breadcrumb.
|
|
151
169
|
*/
|
|
152
170
|
addBreadcrumb(crumb: BreadcrumbEntry): void;
|
|
171
|
+
/**
|
|
172
|
+
* Manually report an API call timing to BugCatch.
|
|
173
|
+
* Use when `autoTrackRequests` is off or for server-side tracking.
|
|
174
|
+
*
|
|
175
|
+
* @example
|
|
176
|
+
* const start = Date.now();
|
|
177
|
+
* await fetch('/api/orders');
|
|
178
|
+
* BugCatch.trackRequest('GET', '/api/orders', Date.now() - start, 200);
|
|
179
|
+
*/
|
|
180
|
+
trackRequest(method: string, route: string, durationMs: number, statusCode: number): void;
|
|
153
181
|
/**
|
|
154
182
|
* Tear down all listeners. Useful in SPA cleanup or hot-reload scenarios.
|
|
155
183
|
*/
|
package/dist/index.js
CHANGED
|
@@ -228,7 +228,17 @@ function parseDsn(dsn) {
|
|
|
228
228
|
throw new Error(`[BugCatch] Could not extract projectId from DSN path: "${dsn}"`);
|
|
229
229
|
}
|
|
230
230
|
const ingestUrl = `${url.origin}${url.pathname}?key=${sdkKey}`;
|
|
231
|
-
|
|
231
|
+
const metricsUrl = `${url.origin}${url.pathname}/metrics?key=${sdkKey}`;
|
|
232
|
+
return { ingestUrl, metricsUrl, projectId, sdkKey };
|
|
233
|
+
}
|
|
234
|
+
function normalizeUrl(rawUrl) {
|
|
235
|
+
let pathname;
|
|
236
|
+
try {
|
|
237
|
+
pathname = new URL(rawUrl).pathname;
|
|
238
|
+
} catch (e) {
|
|
239
|
+
pathname = rawUrl.split("?")[0];
|
|
240
|
+
}
|
|
241
|
+
return pathname.replace(/\/[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}/gi, "/:id").replace(/\/\d{1,20}(?=\/|$)/g, "/:id");
|
|
232
242
|
}
|
|
233
243
|
function matchesPattern(value, patterns) {
|
|
234
244
|
return patterns.some(
|
|
@@ -244,7 +254,8 @@ var BugCatchClient = class {
|
|
|
244
254
|
debug: false,
|
|
245
255
|
maxBreadcrumbs: 100,
|
|
246
256
|
autoCaptureErrors: true,
|
|
247
|
-
autoCaptureBreadcrumbs: true
|
|
257
|
+
autoCaptureBreadcrumbs: true,
|
|
258
|
+
autoTrackRequests: false
|
|
248
259
|
}, options);
|
|
249
260
|
this.dsn = parseDsn(options.dsn);
|
|
250
261
|
this.crumbs = new BreadcrumbBuffer(this.opts.maxBreadcrumbs);
|
|
@@ -256,6 +267,10 @@ var BugCatchClient = class {
|
|
|
256
267
|
this.cleanups.push(installClickBreadcrumbs((c) => this.addBreadcrumb(c)));
|
|
257
268
|
this.cleanups.push(installConsoleBreadcrumbs((c) => this.addBreadcrumb(c)));
|
|
258
269
|
}
|
|
270
|
+
if (this.opts.autoTrackRequests) {
|
|
271
|
+
this.cleanups.push(this.installFetchInterceptor());
|
|
272
|
+
this.cleanups.push(this.installXhrInterceptor());
|
|
273
|
+
}
|
|
259
274
|
this.log("Initialized. Project:", this.dsn.projectId);
|
|
260
275
|
}
|
|
261
276
|
// ─── Public API ─────────────────────────────────────────────────────────────
|
|
@@ -290,6 +305,13 @@ var BugCatchClient = class {
|
|
|
290
305
|
}
|
|
291
306
|
this.cleanups.length = 0;
|
|
292
307
|
}
|
|
308
|
+
/**
|
|
309
|
+
* Manually report an API call timing. Use this when `autoTrackRequests`
|
|
310
|
+
* is off or when you want to track server-side (Node.js) requests explicitly.
|
|
311
|
+
*/
|
|
312
|
+
trackRequest(method, route, durationMs, statusCode) {
|
|
313
|
+
void this.sendMetric({ method: method.toUpperCase(), route, duration_ms: durationMs, status_code: statusCode });
|
|
314
|
+
}
|
|
293
315
|
// ─── Internal: build payloads ────────────────────────────────────────────
|
|
294
316
|
buildExceptionPayload(error, extra) {
|
|
295
317
|
const err = error instanceof Error ? error : new Error(String(error));
|
|
@@ -386,6 +408,123 @@ var BugCatchClient = class {
|
|
|
386
408
|
}
|
|
387
409
|
return false;
|
|
388
410
|
}
|
|
411
|
+
// ─── Internal: request tracking ──────────────────────────────────────────
|
|
412
|
+
installFetchInterceptor() {
|
|
413
|
+
if (typeof globalThis.fetch !== "function") {
|
|
414
|
+
console.warn("[BugCatch] autoTrackRequests is enabled but globalThis.fetch is not available. Metrics will not be tracked.");
|
|
415
|
+
return () => {
|
|
416
|
+
};
|
|
417
|
+
}
|
|
418
|
+
const original = globalThis.fetch.bind(globalThis);
|
|
419
|
+
const self = this;
|
|
420
|
+
this.log("autoTrackRequests: fetch interceptor installed");
|
|
421
|
+
globalThis.fetch = async function patchedFetch(input, init) {
|
|
422
|
+
var _a;
|
|
423
|
+
const url = input instanceof Request ? input.url : String(input);
|
|
424
|
+
if (url.includes(self.dsn.ingestUrl.split("?")[0])) {
|
|
425
|
+
return original(input, init);
|
|
426
|
+
}
|
|
427
|
+
if (self.opts.trackIgnoreUrls && matchesPattern(url, self.opts.trackIgnoreUrls)) {
|
|
428
|
+
self.log("autoTrackRequests: ignored (trackIgnoreUrls match)", url);
|
|
429
|
+
return original(input, init);
|
|
430
|
+
}
|
|
431
|
+
const method = ((_a = init == null ? void 0 : init.method) != null ? _a : input instanceof Request ? input.method : "GET").toUpperCase();
|
|
432
|
+
const route = normalizeUrl(url);
|
|
433
|
+
const start = Date.now();
|
|
434
|
+
self.log("autoTrackRequests: intercepted", method, url);
|
|
435
|
+
try {
|
|
436
|
+
const response = await original(input, init);
|
|
437
|
+
void self.sendMetric({
|
|
438
|
+
method,
|
|
439
|
+
route,
|
|
440
|
+
duration_ms: Date.now() - start,
|
|
441
|
+
status_code: response.status,
|
|
442
|
+
user_id: self.user.id
|
|
443
|
+
});
|
|
444
|
+
return response;
|
|
445
|
+
} catch (err) {
|
|
446
|
+
void self.sendMetric({ method, route, duration_ms: Date.now() - start, status_code: 0 });
|
|
447
|
+
throw err;
|
|
448
|
+
}
|
|
449
|
+
};
|
|
450
|
+
return () => {
|
|
451
|
+
globalThis.fetch = original;
|
|
452
|
+
this.log("autoTrackRequests: fetch interceptor removed");
|
|
453
|
+
};
|
|
454
|
+
}
|
|
455
|
+
installXhrInterceptor() {
|
|
456
|
+
if (typeof XMLHttpRequest === "undefined") {
|
|
457
|
+
return () => {
|
|
458
|
+
};
|
|
459
|
+
}
|
|
460
|
+
const self = this;
|
|
461
|
+
const OriginalXHR = XMLHttpRequest;
|
|
462
|
+
const ingestBase = this.dsn.ingestUrl.split("?")[0];
|
|
463
|
+
function PatchedXHR() {
|
|
464
|
+
const xhr = new OriginalXHR();
|
|
465
|
+
let method = "GET";
|
|
466
|
+
let url = "";
|
|
467
|
+
let start = 0;
|
|
468
|
+
const originalOpen = xhr.open.bind(xhr);
|
|
469
|
+
const originalSend = xhr.send.bind(xhr);
|
|
470
|
+
this.open = function(m, u, async, user, password) {
|
|
471
|
+
method = m ? m.toUpperCase() : "GET";
|
|
472
|
+
url = u || "";
|
|
473
|
+
return originalOpen(m, u, async != null ? async : true, user, password);
|
|
474
|
+
};
|
|
475
|
+
this.send = function(body) {
|
|
476
|
+
const skip = url.includes(ingestBase) || (self.opts.trackIgnoreUrls ? matchesPattern(url, self.opts.trackIgnoreUrls) : false);
|
|
477
|
+
if (!skip) {
|
|
478
|
+
start = Date.now();
|
|
479
|
+
self.log("autoTrackRequests: XHR intercepted", method, url);
|
|
480
|
+
xhr.addEventListener("loadend", () => {
|
|
481
|
+
void self.sendMetric({
|
|
482
|
+
method,
|
|
483
|
+
route: normalizeUrl(url),
|
|
484
|
+
duration_ms: Date.now() - start,
|
|
485
|
+
status_code: xhr.status || 0,
|
|
486
|
+
user_id: self.user.id
|
|
487
|
+
});
|
|
488
|
+
});
|
|
489
|
+
}
|
|
490
|
+
return originalSend(body);
|
|
491
|
+
};
|
|
492
|
+
return new Proxy(this, {
|
|
493
|
+
get(_target, prop) {
|
|
494
|
+
if (prop === "open") return this.open;
|
|
495
|
+
if (prop === "send") return this.send;
|
|
496
|
+
const val = xhr[prop];
|
|
497
|
+
return typeof val === "function" ? val.bind(xhr) : val;
|
|
498
|
+
},
|
|
499
|
+
set(_target, prop, value) {
|
|
500
|
+
xhr[prop] = value;
|
|
501
|
+
return true;
|
|
502
|
+
}
|
|
503
|
+
});
|
|
504
|
+
}
|
|
505
|
+
PatchedXHR.prototype = OriginalXHR.prototype;
|
|
506
|
+
window.XMLHttpRequest = PatchedXHR;
|
|
507
|
+
this.log("autoTrackRequests: XHR interceptor installed");
|
|
508
|
+
return () => {
|
|
509
|
+
window.XMLHttpRequest = OriginalXHR;
|
|
510
|
+
this.log("autoTrackRequests: XHR interceptor removed");
|
|
511
|
+
};
|
|
512
|
+
}
|
|
513
|
+
async sendMetric(metric) {
|
|
514
|
+
this.log("autoTrackRequests: sending metric", metric.method, metric.route, metric.duration_ms + "ms", metric.status_code);
|
|
515
|
+
try {
|
|
516
|
+
const res = await fetch(this.dsn.metricsUrl, {
|
|
517
|
+
method: "POST",
|
|
518
|
+
headers: { "Content-Type": "application/json" },
|
|
519
|
+
body: JSON.stringify(metric)
|
|
520
|
+
});
|
|
521
|
+
if (!res.ok) {
|
|
522
|
+
this.log("autoTrackRequests: metric rejected by server \u2014", res.status, await res.text().catch(() => ""));
|
|
523
|
+
}
|
|
524
|
+
} catch (err) {
|
|
525
|
+
this.log("autoTrackRequests: failed to send metric \u2014", err.message);
|
|
526
|
+
}
|
|
527
|
+
}
|
|
389
528
|
// ─── Internal: transport ─────────────────────────────────────────────────
|
|
390
529
|
async send(event) {
|
|
391
530
|
let finalEvent = event;
|
|
@@ -493,6 +632,18 @@ var BugCatch = {
|
|
|
493
632
|
addBreadcrumb(crumb) {
|
|
494
633
|
getClient().addBreadcrumb(crumb);
|
|
495
634
|
},
|
|
635
|
+
/**
|
|
636
|
+
* Manually report an API call timing to BugCatch.
|
|
637
|
+
* Use when `autoTrackRequests` is off or for server-side tracking.
|
|
638
|
+
*
|
|
639
|
+
* @example
|
|
640
|
+
* const start = Date.now();
|
|
641
|
+
* await fetch('/api/orders');
|
|
642
|
+
* BugCatch.trackRequest('GET', '/api/orders', Date.now() - start, 200);
|
|
643
|
+
*/
|
|
644
|
+
trackRequest(method, route, durationMs, statusCode) {
|
|
645
|
+
getClient().trackRequest(method, route, durationMs, statusCode);
|
|
646
|
+
},
|
|
496
647
|
/**
|
|
497
648
|
* Tear down all listeners. Useful in SPA cleanup or hot-reload scenarios.
|
|
498
649
|
*/
|
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/parse-stack.ts","../src/breadcrumbs.ts","../src/client.ts","../src/index.ts"],"names":[],"mappings":"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","file":"index.js","sourcesContent":["import type { StackFrame } from './types.js';\n\n// Chrome/V8: \" at functionName (filename:line:col)\"\n// \" at filename:line:col\"\nconst CHROME_RE = /^\\s*at (?:(.*?) \\()?(.+?):(\\d+):(\\d+)\\)?$/;\n\n// Firefox/Safari: \"functionName@filename:line:col\"\n// \"@filename:line:col\"\nconst FIREFOX_RE = /^(?:(.+?)@)?(.+?):(\\d+):(\\d+)$/;\n\n// Origins that are not \"in-app\" frames\nconst INTERNAL_PATTERNS = [/node_modules\\//, /bugcatch[-_]?sdk/, /^\\(native\\)$/, /^native code/];\n\nfunction isInApp(filename: string | undefined): boolean {\n if (!filename) return false;\n return !INTERNAL_PATTERNS.some((p) => p.test(filename));\n}\n\nfunction parseChromeLine(line: string): StackFrame | null {\n const m = CHROME_RE.exec(line);\n if (!m) return null;\n const [, fn, filename, lineno, colno] = m;\n return {\n function: fn ?? '<anonymous>',\n filename: filename ?? undefined,\n lineno: lineno ? parseInt(lineno, 10) : undefined,\n colno: colno ? parseInt(colno, 10) : undefined,\n in_app: isInApp(filename ?? undefined),\n };\n}\n\nfunction parseFirefoxLine(line: string): StackFrame | null {\n const m = FIREFOX_RE.exec(line);\n if (!m) return null;\n const [, fn, filename, lineno, colno] = m;\n return {\n function: fn ?? '<anonymous>',\n filename: filename ?? undefined,\n lineno: lineno ? parseInt(lineno, 10) : undefined,\n colno: colno ? parseInt(colno, 10) : undefined,\n in_app: isInApp(filename ?? undefined),\n };\n}\n\nexport function parseStack(error: Error): StackFrame[] {\n const stack = error.stack;\n if (!stack) return [];\n\n const lines = stack.split('\\n');\n const frames: StackFrame[] = [];\n\n for (const line of lines) {\n const trimmed = line.trim();\n if (!trimmed || trimmed.startsWith('Error') || trimmed.startsWith(error.message ?? '')) {\n continue;\n }\n const frame = parseChromeLine(trimmed) ?? parseFirefoxLine(trimmed);\n if (frame) {\n frames.push(frame);\n }\n }\n\n // The API fingerprinting uses the LAST in-app frame, which maps to innermost\n // call. Return in innermost-first order (as captured).\n return frames;\n}\n","import type { BreadcrumbEntry } from './types.js';\n\nexport class BreadcrumbBuffer {\n private readonly buffer: BreadcrumbEntry[] = [];\n private readonly max: number;\n\n constructor(max = 100) {\n this.max = max;\n }\n\n add(crumb: BreadcrumbEntry): void {\n if (this.buffer.length >= this.max) {\n this.buffer.shift();\n }\n this.buffer.push(crumb);\n }\n\n getAll(): BreadcrumbEntry[] {\n return [...this.buffer];\n }\n\n clear(): void {\n this.buffer.length = 0;\n }\n}\n\n// ─── Auto-capture helpers ─────────────────────────────────────────────────────\n\ntype AddFn = (crumb: BreadcrumbEntry) => void;\n\n/**\n * Patch History API (pushState / replaceState / popstate) to capture\n * navigation breadcrumbs.\n */\nexport function installNavigationBreadcrumbs(add: AddFn): () => void {\n if (typeof window === 'undefined') return () => undefined;\n\n const originalPush = history.pushState.bind(history);\n const originalReplace = history.replaceState.bind(history);\n\n history.pushState = function (...args) {\n add({\n timestamp: new Date().toISOString(),\n type: 'navigation',\n category: 'navigation',\n message: `Navigated to ${String(args[2] ?? '')}`,\n data: { to: String(args[2] ?? '') },\n });\n return originalPush(...args);\n };\n\n history.replaceState = function (...args) {\n add({\n timestamp: new Date().toISOString(),\n type: 'navigation',\n category: 'navigation',\n message: `Replaced state: ${String(args[2] ?? '')}`,\n data: { to: String(args[2] ?? '') },\n });\n return originalReplace(...args);\n };\n\n const onPopState = () => {\n add({\n timestamp: new Date().toISOString(),\n type: 'navigation',\n category: 'navigation',\n message: `Navigated back/forward to ${window.location.href}`,\n data: { to: window.location.href },\n });\n };\n\n window.addEventListener('popstate', onPopState);\n\n return () => {\n history.pushState = originalPush;\n history.replaceState = originalReplace;\n window.removeEventListener('popstate', onPopState);\n };\n}\n\n/**\n * Capture click events as breadcrumbs (element tag + text).\n */\nexport function installClickBreadcrumbs(add: AddFn): () => void {\n if (typeof document === 'undefined') return () => undefined;\n\n const handler = (e: MouseEvent) => {\n const target = e.target as HTMLElement | null;\n if (!target) return;\n const tag = target.tagName?.toLowerCase() ?? 'unknown';\n const text = (target.textContent ?? '').trim().slice(0, 50);\n const id = target.id ? `#${target.id}` : '';\n const cls = target.className && typeof target.className === 'string'\n ? `.${target.className.split(' ').join('.')}`\n : '';\n add({\n timestamp: new Date().toISOString(),\n type: 'user',\n category: 'ui.click',\n message: `Click on <${tag}${id}${cls}>`,\n data: { tag, text: text || undefined },\n });\n };\n\n document.addEventListener('click', handler, { capture: true, passive: true });\n\n return () => {\n document.removeEventListener('click', handler, { capture: true });\n };\n}\n\n/**\n * Capture console.warn and console.error calls as breadcrumbs.\n */\nexport function installConsoleBreadcrumbs(add: AddFn): () => void {\n if (typeof console === 'undefined') return () => undefined;\n\n const originalWarn = console.warn.bind(console);\n const originalError = console.error.bind(console);\n\n const formatArgs = (args: unknown[]): string =>\n args.map((a) => (typeof a === 'string' ? a : JSON.stringify(a))).join(' ').slice(0, 200);\n\n console.warn = (...args: unknown[]) => {\n add({\n timestamp: new Date().toISOString(),\n type: 'console',\n category: 'console',\n message: formatArgs(args),\n data: { level: 'warning' },\n });\n originalWarn(...args);\n };\n\n console.error = (...args: unknown[]) => {\n add({\n timestamp: new Date().toISOString(),\n type: 'console',\n category: 'console',\n message: formatArgs(args),\n data: { level: 'error' },\n });\n originalError(...args);\n };\n\n return () => {\n console.warn = originalWarn;\n console.error = originalError;\n };\n}\n","import type {\n BugCatchOptions,\n UserContext,\n BreadcrumbEntry,\n EventPayload,\n ParsedDsn,\n ExceptionValue,\n} from './types.js';\nimport { parseStack } from './parse-stack.js';\nimport {\n BreadcrumbBuffer,\n installNavigationBreadcrumbs,\n installClickBreadcrumbs,\n installConsoleBreadcrumbs,\n} from './breadcrumbs.js';\n\n// ─── UUID ─────────────────────────────────────────────────────────────────────\n\nfunction uuid(): string {\n if (typeof crypto !== 'undefined' && typeof crypto.randomUUID === 'function') {\n return crypto.randomUUID();\n }\n // Fallback for older environments\n return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, (c) => {\n const r = (Math.random() * 16) | 0;\n const v = c === 'x' ? r : (r & 0x3) | 0x8;\n return v.toString(16);\n });\n}\n\n// ─── DSN parser ───────────────────────────────────────────────────────────────\n\nfunction parseDsn(dsn: string): ParsedDsn {\n let url: URL;\n try {\n url = new URL(dsn);\n } catch {\n throw new Error(`[BugCatch] Invalid DSN: \"${dsn}\"`);\n }\n\n const sdkKey = url.searchParams.get('key');\n if (!sdkKey) {\n throw new Error(`[BugCatch] DSN is missing the \"key\" query parameter: \"${dsn}\"`);\n }\n\n // Extract projectId from the last path segment (/ingest/{projectId})\n const segments = url.pathname.split('/').filter(Boolean);\n const projectId = segments[segments.length - 1];\n if (!projectId) {\n throw new Error(`[BugCatch] Could not extract projectId from DSN path: \"${dsn}\"`);\n }\n\n // Reconstruct the URL without extra query params\n const ingestUrl = `${url.origin}${url.pathname}?key=${sdkKey}`;\n\n return { ingestUrl, projectId, sdkKey };\n}\n\n// ─── Pattern matching ─────────────────────────────────────────────────────────\n\nfunction matchesPattern(value: string, patterns: Array<string | RegExp>): boolean {\n return patterns.some((p) =>\n typeof p === 'string' ? value.includes(p) : p.test(value),\n );\n}\n\n// ─── Client ───────────────────────────────────────────────────────────────────\n\nexport class BugCatchClient {\n private readonly opts: Required<\n Pick<BugCatchOptions, 'debug' | 'maxBreadcrumbs' | 'autoCaptureErrors' | 'autoCaptureBreadcrumbs'>\n > & BugCatchOptions;\n\n private readonly dsn: ParsedDsn;\n private readonly crumbs: BreadcrumbBuffer;\n private user: UserContext = {};\n private tags: Record<string, string> = {};\n private readonly cleanups: Array<() => void> = [];\n\n constructor(options: BugCatchOptions) {\n this.opts = {\n debug: false,\n maxBreadcrumbs: 100,\n autoCaptureErrors: true,\n autoCaptureBreadcrumbs: true,\n ...options,\n };\n\n this.dsn = parseDsn(options.dsn);\n this.crumbs = new BreadcrumbBuffer(this.opts.maxBreadcrumbs);\n\n if (this.opts.autoCaptureErrors) {\n this.installErrorHandlers();\n }\n\n if (this.opts.autoCaptureBreadcrumbs) {\n this.cleanups.push(installNavigationBreadcrumbs((c) => this.addBreadcrumb(c)));\n this.cleanups.push(installClickBreadcrumbs((c) => this.addBreadcrumb(c)));\n this.cleanups.push(installConsoleBreadcrumbs((c) => this.addBreadcrumb(c)));\n }\n\n this.log('Initialized. Project:', this.dsn.projectId);\n }\n\n // ─── Public API ─────────────────────────────────────────────────────────────\n\n captureException(error: unknown, extra?: Record<string, unknown>): string {\n const event = this.buildExceptionPayload(error, extra);\n void this.send(event);\n return event.event_id;\n }\n\n captureMessage(message: string, level: string = 'info', extra?: Record<string, unknown>): string {\n const event = this.buildMessagePayload(message, level, extra);\n void this.send(event);\n return event.event_id;\n }\n\n setUser(user: UserContext): void {\n this.user = {\n ...user,\n id: user.id != null ? String(user.id) : undefined,\n };\n }\n\n clearUser(): void {\n this.user = {};\n }\n\n setTag(key: string, value: string): void {\n this.tags[key] = value;\n }\n\n addBreadcrumb(crumb: BreadcrumbEntry): void {\n this.crumbs.add(crumb);\n }\n\n /** Tear down all global listeners. Call when unmounting in SPAs. */\n destroy(): void {\n for (const cleanup of this.cleanups) {\n cleanup();\n }\n this.cleanups.length = 0;\n }\n\n // ─── Internal: build payloads ────────────────────────────────────────────\n\n private buildExceptionPayload(\n error: unknown,\n extra?: Record<string, unknown>,\n ): EventPayload {\n const err = error instanceof Error ? error : new Error(String(error));\n const frames = parseStack(err);\n\n const exceptionValue: ExceptionValue = {\n type: err.name || 'Error',\n value: err.message,\n stacktrace: frames.length > 0 ? { frames } : undefined,\n };\n\n return this.buildBase({\n level: 'error',\n exception: { values: [exceptionValue] },\n extra,\n });\n }\n\n private buildMessagePayload(\n message: string,\n level: string,\n extra?: Record<string, unknown>,\n ): EventPayload {\n return this.buildBase({ level, message, extra });\n }\n\n private buildBase(\n overrides: Partial<EventPayload> & { extra?: Record<string, unknown> },\n ): EventPayload {\n const { extra, ...rest } = overrides;\n const payload: EventPayload = {\n event_id: uuid(),\n timestamp: new Date().toISOString(),\n platform: 'javascript',\n ...rest,\n };\n\n if (this.opts.release) payload.release = this.opts.release;\n if (this.opts.environment) payload.environment = this.opts.environment;\n\n if (Object.keys(this.user).length > 0) {\n payload.user = { ...this.user };\n }\n\n if (Object.keys(this.tags).length > 0) {\n payload.tags = { ...this.tags };\n }\n\n if (extra && Object.keys(extra).length > 0) {\n payload.extra = extra;\n }\n\n const breadcrumbs = this.crumbs.getAll();\n if (breadcrumbs.length > 0) {\n payload.breadcrumbs = breadcrumbs;\n }\n\n // Capture browser request context\n if (typeof window !== 'undefined') {\n payload.request = {\n url: window.location.href,\n headers: {\n 'user-agent': navigator.userAgent,\n },\n };\n }\n\n return payload;\n }\n\n // ─── Internal: error handlers ────────────────────────────────────────────\n\n private installErrorHandlers(): void {\n if (typeof window === 'undefined') {\n // Node.js environment\n const onUncaught = (err: Error) => {\n this.captureException(err, { handler: 'uncaughtException' });\n };\n const onUnhandled = (reason: unknown) => {\n this.captureException(reason, { handler: 'unhandledRejection' });\n };\n process.on('uncaughtException', onUncaught);\n process.on('unhandledRejection', onUnhandled);\n this.cleanups.push(() => {\n process.off('uncaughtException', onUncaught);\n process.off('unhandledRejection', onUnhandled);\n });\n return;\n }\n\n // Browser environment\n const onError = (event: ErrorEvent) => {\n if (!event.error) return;\n if (this.shouldIgnore(event.error, event.filename ?? '')) return;\n this.captureException(event.error, { handler: 'window.onerror' });\n };\n\n const onUnhandledRejection = (event: PromiseRejectionEvent) => {\n const reason = event.reason as unknown;\n if (this.shouldIgnore(reason, '')) return;\n this.captureException(reason, { handler: 'unhandledrejection' });\n };\n\n window.addEventListener('error', onError);\n window.addEventListener('unhandledrejection', onUnhandledRejection);\n\n this.cleanups.push(() => {\n window.removeEventListener('error', onError);\n window.removeEventListener('unhandledrejection', onUnhandledRejection);\n });\n }\n\n private shouldIgnore(error: unknown, scriptUrl: string): boolean {\n const { ignoreUrls, ignoreErrors } = this.opts;\n\n if (ignoreUrls?.length && scriptUrl) {\n if (matchesPattern(scriptUrl, ignoreUrls)) return true;\n }\n\n if (ignoreErrors?.length) {\n const message = error instanceof Error ? error.message : String(error);\n if (matchesPattern(message, ignoreErrors)) return true;\n }\n\n return false;\n }\n\n // ─── Internal: transport ─────────────────────────────────────────────────\n\n private async send(event: EventPayload): Promise<void> {\n let finalEvent: EventPayload | false = event;\n\n if (this.opts.beforeSend) {\n finalEvent = this.opts.beforeSend(event);\n if (finalEvent === false) {\n this.log('Event dropped by beforeSend hook');\n return;\n }\n }\n\n this.log('Sending event:', finalEvent.event_id, finalEvent.level);\n\n let body: string;\n try {\n body = JSON.stringify(finalEvent);\n } catch (err) {\n console.warn('[BugCatch] Failed to serialize event:', err);\n return;\n }\n\n try {\n const fetchOptions: RequestInit = {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body,\n };\n\n // keepalive is browser-only; Node.js fetch (undici) enforces a 64 KB body\n // limit with keepalive: true, which silently drops large payloads.\n if (typeof window !== 'undefined') {\n fetchOptions.keepalive = true;\n }\n\n const res = await fetch(this.dsn.ingestUrl, fetchOptions);\n\n if (!res.ok) {\n console.warn(`[BugCatch] Ingest responded with ${res.status}:`, await res.text());\n }\n } catch (err) {\n console.warn('[BugCatch] Failed to send event:', err);\n }\n }\n\n private log(...args: unknown[]): void {\n if (this.opts.debug) {\n console.debug('[BugCatch]', ...args);\n }\n }\n}\n","import { BugCatchClient } from './client.js';\nimport type { BugCatchOptions, UserContext, BreadcrumbEntry } from './types.js';\n\nexport type { BugCatchOptions, UserContext, BreadcrumbEntry };\nexport type { EventPayload, StackFrame, ParsedDsn } from './types.js';\nexport { BugCatchClient };\n\n// ─── Singleton instance ───────────────────────────────────────────────────────\n\nlet _client: BugCatchClient | null = null;\n\nfunction getClient(): BugCatchClient {\n if (!_client) {\n throw new Error('[BugCatch] SDK not initialized. Call BugCatch.init(options) first.');\n }\n return _client;\n}\n\n// ─── Public singleton API ─────────────────────────────────────────────────────\n\nconst BugCatch = {\n /**\n * Initialize the SDK. Call this once, as early as possible in your app.\n *\n * @example\n * BugCatch.init({\n * dsn: 'http://localhost:3000/ingest/project-id?key=sdk-key',\n * release: '1.0.0',\n * environment: 'production',\n * });\n */\n init(options: BugCatchOptions): BugCatchClient {\n if (_client) {\n console.warn('[BugCatch] init() called more than once. Ignoring.');\n return _client;\n }\n _client = new BugCatchClient(options);\n return _client;\n },\n\n /**\n * Capture an Error object or any value as an error event.\n * Returns the generated event_id.\n */\n captureException(error: unknown, extra?: Record<string, unknown>): string {\n return getClient().captureException(error, extra);\n },\n\n /**\n * Capture a plain text message as an event.\n * Returns the generated event_id.\n */\n captureMessage(message: string, level?: string, extra?: Record<string, unknown>): string {\n return getClient().captureMessage(message, level, extra);\n },\n\n /**\n * Set the current user context. Attached to all subsequent events.\n */\n setUser(user: UserContext): void {\n getClient().setUser(user);\n },\n\n /**\n * Clear the current user context (e.g. on logout).\n */\n clearUser(): void {\n getClient().clearUser();\n },\n\n /**\n * Set a tag that will be attached to all subsequent events.\n */\n setTag(key: string, value: string): void {\n getClient().setTag(key, value);\n },\n\n /**\n * Manually add a breadcrumb.\n */\n addBreadcrumb(crumb: BreadcrumbEntry): void {\n getClient().addBreadcrumb(crumb);\n },\n\n /**\n * Tear down all listeners. Useful in SPA cleanup or hot-reload scenarios.\n */\n destroy(): void {\n _client?.destroy();\n _client = null;\n },\n};\n\n// Named export for CJS consumers: const { BugCatch } = require('bugcatch-sdk')\nexport { BugCatch };\n\n// Default export for ESM consumers: import BugCatch from 'bugcatch-sdk'\nexport default BugCatch;\n"]}
|
|
1
|
+
{"version":3,"sources":["../src/parse-stack.ts","../src/breadcrumbs.ts","../src/client.ts","../src/index.ts"],"names":[],"mappings":"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","file":"index.js","sourcesContent":["import type { StackFrame } from './types.js';\n\n// Chrome/V8: \" at functionName (filename:line:col)\"\n// \" at filename:line:col\"\nconst CHROME_RE = /^\\s*at (?:(.*?) \\()?(.+?):(\\d+):(\\d+)\\)?$/;\n\n// Firefox/Safari: \"functionName@filename:line:col\"\n// \"@filename:line:col\"\nconst FIREFOX_RE = /^(?:(.+?)@)?(.+?):(\\d+):(\\d+)$/;\n\n// Origins that are not \"in-app\" frames\nconst INTERNAL_PATTERNS = [/node_modules\\//, /bugcatch[-_]?sdk/, /^\\(native\\)$/, /^native code/];\n\nfunction isInApp(filename: string | undefined): boolean {\n if (!filename) return false;\n return !INTERNAL_PATTERNS.some((p) => p.test(filename));\n}\n\nfunction parseChromeLine(line: string): StackFrame | null {\n const m = CHROME_RE.exec(line);\n if (!m) return null;\n const [, fn, filename, lineno, colno] = m;\n return {\n function: fn ?? '<anonymous>',\n filename: filename ?? undefined,\n lineno: lineno ? parseInt(lineno, 10) : undefined,\n colno: colno ? parseInt(colno, 10) : undefined,\n in_app: isInApp(filename ?? undefined),\n };\n}\n\nfunction parseFirefoxLine(line: string): StackFrame | null {\n const m = FIREFOX_RE.exec(line);\n if (!m) return null;\n const [, fn, filename, lineno, colno] = m;\n return {\n function: fn ?? '<anonymous>',\n filename: filename ?? undefined,\n lineno: lineno ? parseInt(lineno, 10) : undefined,\n colno: colno ? parseInt(colno, 10) : undefined,\n in_app: isInApp(filename ?? undefined),\n };\n}\n\nexport function parseStack(error: Error): StackFrame[] {\n const stack = error.stack;\n if (!stack) return [];\n\n const lines = stack.split('\\n');\n const frames: StackFrame[] = [];\n\n for (const line of lines) {\n const trimmed = line.trim();\n if (!trimmed || trimmed.startsWith('Error') || trimmed.startsWith(error.message ?? '')) {\n continue;\n }\n const frame = parseChromeLine(trimmed) ?? parseFirefoxLine(trimmed);\n if (frame) {\n frames.push(frame);\n }\n }\n\n // The API fingerprinting uses the LAST in-app frame, which maps to innermost\n // call. Return in innermost-first order (as captured).\n return frames;\n}\n","import type { BreadcrumbEntry } from './types.js';\n\nexport class BreadcrumbBuffer {\n private readonly buffer: BreadcrumbEntry[] = [];\n private readonly max: number;\n\n constructor(max = 100) {\n this.max = max;\n }\n\n add(crumb: BreadcrumbEntry): void {\n if (this.buffer.length >= this.max) {\n this.buffer.shift();\n }\n this.buffer.push(crumb);\n }\n\n getAll(): BreadcrumbEntry[] {\n return [...this.buffer];\n }\n\n clear(): void {\n this.buffer.length = 0;\n }\n}\n\n// ─── Auto-capture helpers ─────────────────────────────────────────────────────\n\ntype AddFn = (crumb: BreadcrumbEntry) => void;\n\n/**\n * Patch History API (pushState / replaceState / popstate) to capture\n * navigation breadcrumbs.\n */\nexport function installNavigationBreadcrumbs(add: AddFn): () => void {\n if (typeof window === 'undefined') return () => undefined;\n\n const originalPush = history.pushState.bind(history);\n const originalReplace = history.replaceState.bind(history);\n\n history.pushState = function (...args) {\n add({\n timestamp: new Date().toISOString(),\n type: 'navigation',\n category: 'navigation',\n message: `Navigated to ${String(args[2] ?? '')}`,\n data: { to: String(args[2] ?? '') },\n });\n return originalPush(...args);\n };\n\n history.replaceState = function (...args) {\n add({\n timestamp: new Date().toISOString(),\n type: 'navigation',\n category: 'navigation',\n message: `Replaced state: ${String(args[2] ?? '')}`,\n data: { to: String(args[2] ?? '') },\n });\n return originalReplace(...args);\n };\n\n const onPopState = () => {\n add({\n timestamp: new Date().toISOString(),\n type: 'navigation',\n category: 'navigation',\n message: `Navigated back/forward to ${window.location.href}`,\n data: { to: window.location.href },\n });\n };\n\n window.addEventListener('popstate', onPopState);\n\n return () => {\n history.pushState = originalPush;\n history.replaceState = originalReplace;\n window.removeEventListener('popstate', onPopState);\n };\n}\n\n/**\n * Capture click events as breadcrumbs (element tag + text).\n */\nexport function installClickBreadcrumbs(add: AddFn): () => void {\n if (typeof document === 'undefined') return () => undefined;\n\n const handler = (e: MouseEvent) => {\n const target = e.target as HTMLElement | null;\n if (!target) return;\n const tag = target.tagName?.toLowerCase() ?? 'unknown';\n const text = (target.textContent ?? '').trim().slice(0, 50);\n const id = target.id ? `#${target.id}` : '';\n const cls = target.className && typeof target.className === 'string'\n ? `.${target.className.split(' ').join('.')}`\n : '';\n add({\n timestamp: new Date().toISOString(),\n type: 'user',\n category: 'ui.click',\n message: `Click on <${tag}${id}${cls}>`,\n data: { tag, text: text || undefined },\n });\n };\n\n document.addEventListener('click', handler, { capture: true, passive: true });\n\n return () => {\n document.removeEventListener('click', handler, { capture: true });\n };\n}\n\n/**\n * Capture console.warn and console.error calls as breadcrumbs.\n */\nexport function installConsoleBreadcrumbs(add: AddFn): () => void {\n if (typeof console === 'undefined') return () => undefined;\n\n const originalWarn = console.warn.bind(console);\n const originalError = console.error.bind(console);\n\n const formatArgs = (args: unknown[]): string =>\n args.map((a) => (typeof a === 'string' ? a : JSON.stringify(a))).join(' ').slice(0, 200);\n\n console.warn = (...args: unknown[]) => {\n add({\n timestamp: new Date().toISOString(),\n type: 'console',\n category: 'console',\n message: formatArgs(args),\n data: { level: 'warning' },\n });\n originalWarn(...args);\n };\n\n console.error = (...args: unknown[]) => {\n add({\n timestamp: new Date().toISOString(),\n type: 'console',\n category: 'console',\n message: formatArgs(args),\n data: { level: 'error' },\n });\n originalError(...args);\n };\n\n return () => {\n console.warn = originalWarn;\n console.error = originalError;\n };\n}\n","import type {\n BugCatchOptions,\n UserContext,\n BreadcrumbEntry,\n EventPayload,\n MetricPayload,\n ParsedDsn,\n ExceptionValue,\n} from './types.js';\nimport { parseStack } from './parse-stack.js';\nimport {\n BreadcrumbBuffer,\n installNavigationBreadcrumbs,\n installClickBreadcrumbs,\n installConsoleBreadcrumbs,\n} from './breadcrumbs.js';\n\n// ─── UUID ─────────────────────────────────────────────────────────────────────\n\nfunction uuid(): string {\n if (typeof crypto !== 'undefined' && typeof crypto.randomUUID === 'function') {\n return crypto.randomUUID();\n }\n // Fallback for older environments\n return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, (c) => {\n const r = (Math.random() * 16) | 0;\n const v = c === 'x' ? r : (r & 0x3) | 0x8;\n return v.toString(16);\n });\n}\n\n// ─── DSN parser ───────────────────────────────────────────────────────────────\n\nfunction parseDsn(dsn: string): ParsedDsn & { metricsUrl: string } {\n let url: URL;\n try {\n url = new URL(dsn);\n } catch {\n throw new Error(`[BugCatch] Invalid DSN: \"${dsn}\"`);\n }\n\n const sdkKey = url.searchParams.get('key');\n if (!sdkKey) {\n throw new Error(`[BugCatch] DSN is missing the \"key\" query parameter: \"${dsn}\"`);\n }\n\n // Extract projectId from the last path segment (/ingest/{projectId})\n const segments = url.pathname.split('/').filter(Boolean);\n const projectId = segments[segments.length - 1];\n if (!projectId) {\n throw new Error(`[BugCatch] Could not extract projectId from DSN path: \"${dsn}\"`);\n }\n\n // Reconstruct the URL without extra query params\n const ingestUrl = `${url.origin}${url.pathname}?key=${sdkKey}`;\n const metricsUrl = `${url.origin}${url.pathname}/metrics?key=${sdkKey}`;\n\n return { ingestUrl, metricsUrl, projectId, sdkKey };\n}\n\n// ─── Route normalizer ─────────────────────────────────────────────────────────\n\n/**\n * Strips the origin and query string from a URL and replaces dynamic path\n * segments (UUIDs, numeric IDs) with `:id` so metrics group by route template.\n *\n * e.g. https://api.example.com/users/42/orders/abc-123?page=1 → /users/:id/orders/:id\n */\nfunction normalizeUrl(rawUrl: string): string {\n let pathname: string;\n try {\n pathname = new URL(rawUrl).pathname;\n } catch {\n // rawUrl is already a path (e.g. \"/api/users/1\")\n pathname = rawUrl.split('?')[0];\n }\n return pathname\n .replace(/\\/[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}/gi, '/:id')\n .replace(/\\/\\d{1,20}(?=\\/|$)/g, '/:id');\n}\n\n// ─── Pattern matching ─────────────────────────────────────────────────────────\n\nfunction matchesPattern(value: string, patterns: Array<string | RegExp>): boolean {\n return patterns.some((p) =>\n typeof p === 'string' ? value.includes(p) : p.test(value),\n );\n}\n\n// ─── Client ───────────────────────────────────────────────────────────────────\n\nexport class BugCatchClient {\n private readonly opts: Required<\n Pick<BugCatchOptions, 'debug' | 'maxBreadcrumbs' | 'autoCaptureErrors' | 'autoCaptureBreadcrumbs' | 'autoTrackRequests'>\n > & BugCatchOptions;\n\n private readonly dsn: ParsedDsn & { metricsUrl: string };\n private readonly crumbs: BreadcrumbBuffer;\n private user: UserContext = {};\n private tags: Record<string, string> = {};\n private readonly cleanups: Array<() => void> = [];\n\n constructor(options: BugCatchOptions) {\n this.opts = {\n debug: false,\n maxBreadcrumbs: 100,\n autoCaptureErrors: true,\n autoCaptureBreadcrumbs: true,\n autoTrackRequests: false,\n ...options,\n };\n\n this.dsn = parseDsn(options.dsn);\n this.crumbs = new BreadcrumbBuffer(this.opts.maxBreadcrumbs);\n\n if (this.opts.autoCaptureErrors) {\n this.installErrorHandlers();\n }\n\n if (this.opts.autoCaptureBreadcrumbs) {\n this.cleanups.push(installNavigationBreadcrumbs((c) => this.addBreadcrumb(c)));\n this.cleanups.push(installClickBreadcrumbs((c) => this.addBreadcrumb(c)));\n this.cleanups.push(installConsoleBreadcrumbs((c) => this.addBreadcrumb(c)));\n }\n\n if (this.opts.autoTrackRequests) {\n this.cleanups.push(this.installFetchInterceptor());\n this.cleanups.push(this.installXhrInterceptor());\n }\n\n this.log('Initialized. Project:', this.dsn.projectId);\n }\n\n // ─── Public API ─────────────────────────────────────────────────────────────\n\n captureException(error: unknown, extra?: Record<string, unknown>): string {\n const event = this.buildExceptionPayload(error, extra);\n void this.send(event);\n return event.event_id;\n }\n\n captureMessage(message: string, level: string = 'info', extra?: Record<string, unknown>): string {\n const event = this.buildMessagePayload(message, level, extra);\n void this.send(event);\n return event.event_id;\n }\n\n setUser(user: UserContext): void {\n this.user = {\n ...user,\n id: user.id != null ? String(user.id) : undefined,\n };\n }\n\n clearUser(): void {\n this.user = {};\n }\n\n setTag(key: string, value: string): void {\n this.tags[key] = value;\n }\n\n addBreadcrumb(crumb: BreadcrumbEntry): void {\n this.crumbs.add(crumb);\n }\n\n /** Tear down all global listeners. Call when unmounting in SPAs. */\n destroy(): void {\n for (const cleanup of this.cleanups) {\n cleanup();\n }\n this.cleanups.length = 0;\n }\n\n /**\n * Manually report an API call timing. Use this when `autoTrackRequests`\n * is off or when you want to track server-side (Node.js) requests explicitly.\n */\n trackRequest(method: string, route: string, durationMs: number, statusCode: number): void {\n void this.sendMetric({ method: method.toUpperCase(), route, duration_ms: durationMs, status_code: statusCode });\n }\n\n // ─── Internal: build payloads ────────────────────────────────────────────\n\n private buildExceptionPayload(\n error: unknown,\n extra?: Record<string, unknown>,\n ): EventPayload {\n const err = error instanceof Error ? error : new Error(String(error));\n const frames = parseStack(err);\n\n const exceptionValue: ExceptionValue = {\n type: err.name || 'Error',\n value: err.message,\n stacktrace: frames.length > 0 ? { frames } : undefined,\n };\n\n return this.buildBase({\n level: 'error',\n exception: { values: [exceptionValue] },\n extra,\n });\n }\n\n private buildMessagePayload(\n message: string,\n level: string,\n extra?: Record<string, unknown>,\n ): EventPayload {\n return this.buildBase({ level, message, extra });\n }\n\n private buildBase(\n overrides: Partial<EventPayload> & { extra?: Record<string, unknown> },\n ): EventPayload {\n const { extra, ...rest } = overrides;\n const payload: EventPayload = {\n event_id: uuid(),\n timestamp: new Date().toISOString(),\n platform: 'javascript',\n ...rest,\n };\n\n if (this.opts.release) payload.release = this.opts.release;\n if (this.opts.environment) payload.environment = this.opts.environment;\n\n if (Object.keys(this.user).length > 0) {\n payload.user = { ...this.user };\n }\n\n if (Object.keys(this.tags).length > 0) {\n payload.tags = { ...this.tags };\n }\n\n if (extra && Object.keys(extra).length > 0) {\n payload.extra = extra;\n }\n\n const breadcrumbs = this.crumbs.getAll();\n if (breadcrumbs.length > 0) {\n payload.breadcrumbs = breadcrumbs;\n }\n\n // Capture browser request context\n if (typeof window !== 'undefined') {\n payload.request = {\n url: window.location.href,\n headers: {\n 'user-agent': navigator.userAgent,\n },\n };\n }\n\n return payload;\n }\n\n // ─── Internal: error handlers ────────────────────────────────────────────\n\n private installErrorHandlers(): void {\n if (typeof window === 'undefined') {\n // Node.js environment\n const onUncaught = (err: Error) => {\n this.captureException(err, { handler: 'uncaughtException' });\n };\n const onUnhandled = (reason: unknown) => {\n this.captureException(reason, { handler: 'unhandledRejection' });\n };\n process.on('uncaughtException', onUncaught);\n process.on('unhandledRejection', onUnhandled);\n this.cleanups.push(() => {\n process.off('uncaughtException', onUncaught);\n process.off('unhandledRejection', onUnhandled);\n });\n return;\n }\n\n // Browser environment\n const onError = (event: ErrorEvent) => {\n if (!event.error) return;\n if (this.shouldIgnore(event.error, event.filename ?? '')) return;\n this.captureException(event.error, { handler: 'window.onerror' });\n };\n\n const onUnhandledRejection = (event: PromiseRejectionEvent) => {\n const reason = event.reason as unknown;\n if (this.shouldIgnore(reason, '')) return;\n this.captureException(reason, { handler: 'unhandledrejection' });\n };\n\n window.addEventListener('error', onError);\n window.addEventListener('unhandledrejection', onUnhandledRejection);\n\n this.cleanups.push(() => {\n window.removeEventListener('error', onError);\n window.removeEventListener('unhandledrejection', onUnhandledRejection);\n });\n }\n\n private shouldIgnore(error: unknown, scriptUrl: string): boolean {\n const { ignoreUrls, ignoreErrors } = this.opts;\n\n if (ignoreUrls?.length && scriptUrl) {\n if (matchesPattern(scriptUrl, ignoreUrls)) return true;\n }\n\n if (ignoreErrors?.length) {\n const message = error instanceof Error ? error.message : String(error);\n if (matchesPattern(message, ignoreErrors)) return true;\n }\n\n return false;\n }\n\n // ─── Internal: request tracking ──────────────────────────────────────────\n\n private installFetchInterceptor(): () => void {\n if (typeof globalThis.fetch !== 'function') {\n console.warn('[BugCatch] autoTrackRequests is enabled but globalThis.fetch is not available. Metrics will not be tracked.');\n return () => {};\n }\n\n const original = globalThis.fetch.bind(globalThis);\n const self = this;\n\n this.log('autoTrackRequests: fetch interceptor installed');\n\n globalThis.fetch = async function patchedFetch(\n input: RequestInfo | URL,\n init?: RequestInit,\n ): Promise<Response> {\n const url = input instanceof Request ? input.url : String(input);\n\n // Never intercept BugCatch's own ingest/metrics requests\n if (url.includes(self.dsn.ingestUrl.split('?')[0])) {\n return original(input, init);\n }\n\n if (self.opts.trackIgnoreUrls && matchesPattern(url, self.opts.trackIgnoreUrls)) {\n self.log('autoTrackRequests: ignored (trackIgnoreUrls match)', url);\n return original(input, init);\n }\n\n const method = (init?.method ?? (input instanceof Request ? input.method : 'GET')).toUpperCase();\n const route = normalizeUrl(url);\n const start = Date.now();\n\n self.log('autoTrackRequests: intercepted', method, url);\n\n try {\n const response = await original(input, init);\n void self.sendMetric({\n method,\n route,\n duration_ms: Date.now() - start,\n status_code: response.status,\n user_id: self.user.id,\n });\n return response;\n } catch (err) {\n // Network error — report as status 0\n void self.sendMetric({ method, route, duration_ms: Date.now() - start, status_code: 0 });\n throw err;\n }\n };\n\n // Return cleanup function that restores original fetch\n return () => {\n globalThis.fetch = original;\n this.log('autoTrackRequests: fetch interceptor removed');\n };\n }\n\n private installXhrInterceptor(): () => void {\n if (typeof XMLHttpRequest === 'undefined') {\n // Node.js environment — XHR not available\n return () => {};\n }\n\n const self = this;\n const OriginalXHR = XMLHttpRequest;\n const ingestBase = this.dsn.ingestUrl.split('?')[0];\n\n function PatchedXHR(this: XMLHttpRequest) {\n const xhr = new OriginalXHR();\n\n let method = 'GET';\n let url = '';\n let start = 0;\n\n const originalOpen = xhr.open.bind(xhr);\n const originalSend = xhr.send.bind(xhr);\n\n (this as any).open = function (m: string, u: string, async?: boolean, user?: string, password?: string) {\n method = m ? m.toUpperCase() : 'GET';\n url = u || '';\n return originalOpen(m, u, async ?? true, user, password);\n };\n\n (this as any).send = function (body?: Document | XMLHttpRequestBodyInit | null) {\n // Skip BugCatch's own requests and ignored URLs\n const skip =\n url.includes(ingestBase) ||\n (self.opts.trackIgnoreUrls ? matchesPattern(url, self.opts.trackIgnoreUrls) : false);\n\n if (!skip) {\n start = Date.now();\n self.log('autoTrackRequests: XHR intercepted', method, url);\n\n xhr.addEventListener('loadend', () => {\n void self.sendMetric({\n method,\n route: normalizeUrl(url),\n duration_ms: Date.now() - start,\n status_code: xhr.status || 0,\n user_id: self.user.id,\n });\n });\n }\n\n return originalSend(body);\n };\n\n // Proxy all other properties/methods to the real XHR instance\n return new Proxy(this, {\n get(_target, prop) {\n if (prop === 'open') return (this as any).open;\n if (prop === 'send') return (this as any).send;\n const val = (xhr as any)[prop];\n return typeof val === 'function' ? val.bind(xhr) : val;\n },\n set(_target, prop, value) {\n (xhr as any)[prop] = value;\n return true;\n },\n });\n }\n\n PatchedXHR.prototype = OriginalXHR.prototype;\n (window as any).XMLHttpRequest = PatchedXHR;\n\n this.log('autoTrackRequests: XHR interceptor installed');\n\n return () => {\n (window as any).XMLHttpRequest = OriginalXHR;\n this.log('autoTrackRequests: XHR interceptor removed');\n };\n }\n\n private async sendMetric(metric: MetricPayload): Promise<void> {\n this.log('autoTrackRequests: sending metric', metric.method, metric.route, metric.duration_ms + 'ms', metric.status_code);\n try {\n const res = await fetch(this.dsn.metricsUrl, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify(metric),\n });\n if (!res.ok) {\n this.log('autoTrackRequests: metric rejected by server —', res.status, await res.text().catch(() => ''));\n }\n } catch (err) {\n this.log('autoTrackRequests: failed to send metric —', (err as Error).message);\n }\n }\n\n // ─── Internal: transport ─────────────────────────────────────────────────\n\n private async send(event: EventPayload): Promise<void> {\n let finalEvent: EventPayload | false = event;\n\n if (this.opts.beforeSend) {\n finalEvent = this.opts.beforeSend(event);\n if (finalEvent === false) {\n this.log('Event dropped by beforeSend hook');\n return;\n }\n }\n\n this.log('Sending event:', finalEvent.event_id, finalEvent.level);\n\n let body: string;\n try {\n body = JSON.stringify(finalEvent);\n } catch (err) {\n console.warn('[BugCatch] Failed to serialize event:', err);\n return;\n }\n\n try {\n const fetchOptions: RequestInit = {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body,\n };\n\n // keepalive is browser-only; Node.js fetch (undici) enforces a 64 KB body\n // limit with keepalive: true, which silently drops large payloads.\n if (typeof window !== 'undefined') {\n fetchOptions.keepalive = true;\n }\n\n const res = await fetch(this.dsn.ingestUrl, fetchOptions);\n\n if (!res.ok) {\n console.warn(`[BugCatch] Ingest responded with ${res.status}:`, await res.text());\n }\n } catch (err) {\n console.warn('[BugCatch] Failed to send event:', err);\n }\n }\n\n private log(...args: unknown[]): void {\n if (this.opts.debug) {\n console.debug('[BugCatch]', ...args);\n }\n }\n}\n","import { BugCatchClient } from './client.js';\nimport type { BugCatchOptions, UserContext, BreadcrumbEntry } from './types.js';\n\nexport type { BugCatchOptions, UserContext, BreadcrumbEntry };\nexport type { EventPayload, StackFrame, ParsedDsn } from './types.js';\nexport { BugCatchClient };\n\n// ─── Singleton instance ───────────────────────────────────────────────────────\n\nlet _client: BugCatchClient | null = null;\n\nfunction getClient(): BugCatchClient {\n if (!_client) {\n throw new Error('[BugCatch] SDK not initialized. Call BugCatch.init(options) first.');\n }\n return _client;\n}\n\n// ─── Public singleton API ─────────────────────────────────────────────────────\n\nconst BugCatch = {\n /**\n * Initialize the SDK. Call this once, as early as possible in your app.\n *\n * @example\n * BugCatch.init({\n * dsn: 'http://localhost:3000/ingest/project-id?key=sdk-key',\n * release: '1.0.0',\n * environment: 'production',\n * });\n */\n init(options: BugCatchOptions): BugCatchClient {\n if (_client) {\n console.warn('[BugCatch] init() called more than once. Ignoring.');\n return _client;\n }\n _client = new BugCatchClient(options);\n return _client;\n },\n\n /**\n * Capture an Error object or any value as an error event.\n * Returns the generated event_id.\n */\n captureException(error: unknown, extra?: Record<string, unknown>): string {\n return getClient().captureException(error, extra);\n },\n\n /**\n * Capture a plain text message as an event.\n * Returns the generated event_id.\n */\n captureMessage(message: string, level?: string, extra?: Record<string, unknown>): string {\n return getClient().captureMessage(message, level, extra);\n },\n\n /**\n * Set the current user context. Attached to all subsequent events.\n */\n setUser(user: UserContext): void {\n getClient().setUser(user);\n },\n\n /**\n * Clear the current user context (e.g. on logout).\n */\n clearUser(): void {\n getClient().clearUser();\n },\n\n /**\n * Set a tag that will be attached to all subsequent events.\n */\n setTag(key: string, value: string): void {\n getClient().setTag(key, value);\n },\n\n /**\n * Manually add a breadcrumb.\n */\n addBreadcrumb(crumb: BreadcrumbEntry): void {\n getClient().addBreadcrumb(crumb);\n },\n\n /**\n * Manually report an API call timing to BugCatch.\n * Use when `autoTrackRequests` is off or for server-side tracking.\n *\n * @example\n * const start = Date.now();\n * await fetch('/api/orders');\n * BugCatch.trackRequest('GET', '/api/orders', Date.now() - start, 200);\n */\n trackRequest(method: string, route: string, durationMs: number, statusCode: number): void {\n getClient().trackRequest(method, route, durationMs, statusCode);\n },\n\n /**\n * Tear down all listeners. Useful in SPA cleanup or hot-reload scenarios.\n */\n destroy(): void {\n _client?.destroy();\n _client = null;\n },\n};\n\n// Named export for CJS consumers: const { BugCatch } = require('bugcatch-sdk')\nexport { BugCatch };\n\n// Default export for ESM consumers: import BugCatch from 'bugcatch-sdk'\nexport default BugCatch;\n"]}
|
package/dist/index.mjs
CHANGED
|
@@ -224,7 +224,17 @@ function parseDsn(dsn) {
|
|
|
224
224
|
throw new Error(`[BugCatch] Could not extract projectId from DSN path: "${dsn}"`);
|
|
225
225
|
}
|
|
226
226
|
const ingestUrl = `${url.origin}${url.pathname}?key=${sdkKey}`;
|
|
227
|
-
|
|
227
|
+
const metricsUrl = `${url.origin}${url.pathname}/metrics?key=${sdkKey}`;
|
|
228
|
+
return { ingestUrl, metricsUrl, projectId, sdkKey };
|
|
229
|
+
}
|
|
230
|
+
function normalizeUrl(rawUrl) {
|
|
231
|
+
let pathname;
|
|
232
|
+
try {
|
|
233
|
+
pathname = new URL(rawUrl).pathname;
|
|
234
|
+
} catch (e) {
|
|
235
|
+
pathname = rawUrl.split("?")[0];
|
|
236
|
+
}
|
|
237
|
+
return pathname.replace(/\/[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}/gi, "/:id").replace(/\/\d{1,20}(?=\/|$)/g, "/:id");
|
|
228
238
|
}
|
|
229
239
|
function matchesPattern(value, patterns) {
|
|
230
240
|
return patterns.some(
|
|
@@ -240,7 +250,8 @@ var BugCatchClient = class {
|
|
|
240
250
|
debug: false,
|
|
241
251
|
maxBreadcrumbs: 100,
|
|
242
252
|
autoCaptureErrors: true,
|
|
243
|
-
autoCaptureBreadcrumbs: true
|
|
253
|
+
autoCaptureBreadcrumbs: true,
|
|
254
|
+
autoTrackRequests: false
|
|
244
255
|
}, options);
|
|
245
256
|
this.dsn = parseDsn(options.dsn);
|
|
246
257
|
this.crumbs = new BreadcrumbBuffer(this.opts.maxBreadcrumbs);
|
|
@@ -252,6 +263,10 @@ var BugCatchClient = class {
|
|
|
252
263
|
this.cleanups.push(installClickBreadcrumbs((c) => this.addBreadcrumb(c)));
|
|
253
264
|
this.cleanups.push(installConsoleBreadcrumbs((c) => this.addBreadcrumb(c)));
|
|
254
265
|
}
|
|
266
|
+
if (this.opts.autoTrackRequests) {
|
|
267
|
+
this.cleanups.push(this.installFetchInterceptor());
|
|
268
|
+
this.cleanups.push(this.installXhrInterceptor());
|
|
269
|
+
}
|
|
255
270
|
this.log("Initialized. Project:", this.dsn.projectId);
|
|
256
271
|
}
|
|
257
272
|
// ─── Public API ─────────────────────────────────────────────────────────────
|
|
@@ -286,6 +301,13 @@ var BugCatchClient = class {
|
|
|
286
301
|
}
|
|
287
302
|
this.cleanups.length = 0;
|
|
288
303
|
}
|
|
304
|
+
/**
|
|
305
|
+
* Manually report an API call timing. Use this when `autoTrackRequests`
|
|
306
|
+
* is off or when you want to track server-side (Node.js) requests explicitly.
|
|
307
|
+
*/
|
|
308
|
+
trackRequest(method, route, durationMs, statusCode) {
|
|
309
|
+
void this.sendMetric({ method: method.toUpperCase(), route, duration_ms: durationMs, status_code: statusCode });
|
|
310
|
+
}
|
|
289
311
|
// ─── Internal: build payloads ────────────────────────────────────────────
|
|
290
312
|
buildExceptionPayload(error, extra) {
|
|
291
313
|
const err = error instanceof Error ? error : new Error(String(error));
|
|
@@ -382,6 +404,123 @@ var BugCatchClient = class {
|
|
|
382
404
|
}
|
|
383
405
|
return false;
|
|
384
406
|
}
|
|
407
|
+
// ─── Internal: request tracking ──────────────────────────────────────────
|
|
408
|
+
installFetchInterceptor() {
|
|
409
|
+
if (typeof globalThis.fetch !== "function") {
|
|
410
|
+
console.warn("[BugCatch] autoTrackRequests is enabled but globalThis.fetch is not available. Metrics will not be tracked.");
|
|
411
|
+
return () => {
|
|
412
|
+
};
|
|
413
|
+
}
|
|
414
|
+
const original = globalThis.fetch.bind(globalThis);
|
|
415
|
+
const self = this;
|
|
416
|
+
this.log("autoTrackRequests: fetch interceptor installed");
|
|
417
|
+
globalThis.fetch = async function patchedFetch(input, init) {
|
|
418
|
+
var _a;
|
|
419
|
+
const url = input instanceof Request ? input.url : String(input);
|
|
420
|
+
if (url.includes(self.dsn.ingestUrl.split("?")[0])) {
|
|
421
|
+
return original(input, init);
|
|
422
|
+
}
|
|
423
|
+
if (self.opts.trackIgnoreUrls && matchesPattern(url, self.opts.trackIgnoreUrls)) {
|
|
424
|
+
self.log("autoTrackRequests: ignored (trackIgnoreUrls match)", url);
|
|
425
|
+
return original(input, init);
|
|
426
|
+
}
|
|
427
|
+
const method = ((_a = init == null ? void 0 : init.method) != null ? _a : input instanceof Request ? input.method : "GET").toUpperCase();
|
|
428
|
+
const route = normalizeUrl(url);
|
|
429
|
+
const start = Date.now();
|
|
430
|
+
self.log("autoTrackRequests: intercepted", method, url);
|
|
431
|
+
try {
|
|
432
|
+
const response = await original(input, init);
|
|
433
|
+
void self.sendMetric({
|
|
434
|
+
method,
|
|
435
|
+
route,
|
|
436
|
+
duration_ms: Date.now() - start,
|
|
437
|
+
status_code: response.status,
|
|
438
|
+
user_id: self.user.id
|
|
439
|
+
});
|
|
440
|
+
return response;
|
|
441
|
+
} catch (err) {
|
|
442
|
+
void self.sendMetric({ method, route, duration_ms: Date.now() - start, status_code: 0 });
|
|
443
|
+
throw err;
|
|
444
|
+
}
|
|
445
|
+
};
|
|
446
|
+
return () => {
|
|
447
|
+
globalThis.fetch = original;
|
|
448
|
+
this.log("autoTrackRequests: fetch interceptor removed");
|
|
449
|
+
};
|
|
450
|
+
}
|
|
451
|
+
installXhrInterceptor() {
|
|
452
|
+
if (typeof XMLHttpRequest === "undefined") {
|
|
453
|
+
return () => {
|
|
454
|
+
};
|
|
455
|
+
}
|
|
456
|
+
const self = this;
|
|
457
|
+
const OriginalXHR = XMLHttpRequest;
|
|
458
|
+
const ingestBase = this.dsn.ingestUrl.split("?")[0];
|
|
459
|
+
function PatchedXHR() {
|
|
460
|
+
const xhr = new OriginalXHR();
|
|
461
|
+
let method = "GET";
|
|
462
|
+
let url = "";
|
|
463
|
+
let start = 0;
|
|
464
|
+
const originalOpen = xhr.open.bind(xhr);
|
|
465
|
+
const originalSend = xhr.send.bind(xhr);
|
|
466
|
+
this.open = function(m, u, async, user, password) {
|
|
467
|
+
method = m ? m.toUpperCase() : "GET";
|
|
468
|
+
url = u || "";
|
|
469
|
+
return originalOpen(m, u, async != null ? async : true, user, password);
|
|
470
|
+
};
|
|
471
|
+
this.send = function(body) {
|
|
472
|
+
const skip = url.includes(ingestBase) || (self.opts.trackIgnoreUrls ? matchesPattern(url, self.opts.trackIgnoreUrls) : false);
|
|
473
|
+
if (!skip) {
|
|
474
|
+
start = Date.now();
|
|
475
|
+
self.log("autoTrackRequests: XHR intercepted", method, url);
|
|
476
|
+
xhr.addEventListener("loadend", () => {
|
|
477
|
+
void self.sendMetric({
|
|
478
|
+
method,
|
|
479
|
+
route: normalizeUrl(url),
|
|
480
|
+
duration_ms: Date.now() - start,
|
|
481
|
+
status_code: xhr.status || 0,
|
|
482
|
+
user_id: self.user.id
|
|
483
|
+
});
|
|
484
|
+
});
|
|
485
|
+
}
|
|
486
|
+
return originalSend(body);
|
|
487
|
+
};
|
|
488
|
+
return new Proxy(this, {
|
|
489
|
+
get(_target, prop) {
|
|
490
|
+
if (prop === "open") return this.open;
|
|
491
|
+
if (prop === "send") return this.send;
|
|
492
|
+
const val = xhr[prop];
|
|
493
|
+
return typeof val === "function" ? val.bind(xhr) : val;
|
|
494
|
+
},
|
|
495
|
+
set(_target, prop, value) {
|
|
496
|
+
xhr[prop] = value;
|
|
497
|
+
return true;
|
|
498
|
+
}
|
|
499
|
+
});
|
|
500
|
+
}
|
|
501
|
+
PatchedXHR.prototype = OriginalXHR.prototype;
|
|
502
|
+
window.XMLHttpRequest = PatchedXHR;
|
|
503
|
+
this.log("autoTrackRequests: XHR interceptor installed");
|
|
504
|
+
return () => {
|
|
505
|
+
window.XMLHttpRequest = OriginalXHR;
|
|
506
|
+
this.log("autoTrackRequests: XHR interceptor removed");
|
|
507
|
+
};
|
|
508
|
+
}
|
|
509
|
+
async sendMetric(metric) {
|
|
510
|
+
this.log("autoTrackRequests: sending metric", metric.method, metric.route, metric.duration_ms + "ms", metric.status_code);
|
|
511
|
+
try {
|
|
512
|
+
const res = await fetch(this.dsn.metricsUrl, {
|
|
513
|
+
method: "POST",
|
|
514
|
+
headers: { "Content-Type": "application/json" },
|
|
515
|
+
body: JSON.stringify(metric)
|
|
516
|
+
});
|
|
517
|
+
if (!res.ok) {
|
|
518
|
+
this.log("autoTrackRequests: metric rejected by server \u2014", res.status, await res.text().catch(() => ""));
|
|
519
|
+
}
|
|
520
|
+
} catch (err) {
|
|
521
|
+
this.log("autoTrackRequests: failed to send metric \u2014", err.message);
|
|
522
|
+
}
|
|
523
|
+
}
|
|
385
524
|
// ─── Internal: transport ─────────────────────────────────────────────────
|
|
386
525
|
async send(event) {
|
|
387
526
|
let finalEvent = event;
|
|
@@ -489,6 +628,18 @@ var BugCatch = {
|
|
|
489
628
|
addBreadcrumb(crumb) {
|
|
490
629
|
getClient().addBreadcrumb(crumb);
|
|
491
630
|
},
|
|
631
|
+
/**
|
|
632
|
+
* Manually report an API call timing to BugCatch.
|
|
633
|
+
* Use when `autoTrackRequests` is off or for server-side tracking.
|
|
634
|
+
*
|
|
635
|
+
* @example
|
|
636
|
+
* const start = Date.now();
|
|
637
|
+
* await fetch('/api/orders');
|
|
638
|
+
* BugCatch.trackRequest('GET', '/api/orders', Date.now() - start, 200);
|
|
639
|
+
*/
|
|
640
|
+
trackRequest(method, route, durationMs, statusCode) {
|
|
641
|
+
getClient().trackRequest(method, route, durationMs, statusCode);
|
|
642
|
+
},
|
|
492
643
|
/**
|
|
493
644
|
* Tear down all listeners. Useful in SPA cleanup or hot-reload scenarios.
|
|
494
645
|
*/
|
package/dist/index.mjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/parse-stack.ts","../src/breadcrumbs.ts","../src/client.ts","../src/index.ts"],"names":[],"mappings":"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","file":"index.mjs","sourcesContent":["import type { StackFrame } from './types.js';\n\n// Chrome/V8: \" at functionName (filename:line:col)\"\n// \" at filename:line:col\"\nconst CHROME_RE = /^\\s*at (?:(.*?) \\()?(.+?):(\\d+):(\\d+)\\)?$/;\n\n// Firefox/Safari: \"functionName@filename:line:col\"\n// \"@filename:line:col\"\nconst FIREFOX_RE = /^(?:(.+?)@)?(.+?):(\\d+):(\\d+)$/;\n\n// Origins that are not \"in-app\" frames\nconst INTERNAL_PATTERNS = [/node_modules\\//, /bugcatch[-_]?sdk/, /^\\(native\\)$/, /^native code/];\n\nfunction isInApp(filename: string | undefined): boolean {\n if (!filename) return false;\n return !INTERNAL_PATTERNS.some((p) => p.test(filename));\n}\n\nfunction parseChromeLine(line: string): StackFrame | null {\n const m = CHROME_RE.exec(line);\n if (!m) return null;\n const [, fn, filename, lineno, colno] = m;\n return {\n function: fn ?? '<anonymous>',\n filename: filename ?? undefined,\n lineno: lineno ? parseInt(lineno, 10) : undefined,\n colno: colno ? parseInt(colno, 10) : undefined,\n in_app: isInApp(filename ?? undefined),\n };\n}\n\nfunction parseFirefoxLine(line: string): StackFrame | null {\n const m = FIREFOX_RE.exec(line);\n if (!m) return null;\n const [, fn, filename, lineno, colno] = m;\n return {\n function: fn ?? '<anonymous>',\n filename: filename ?? undefined,\n lineno: lineno ? parseInt(lineno, 10) : undefined,\n colno: colno ? parseInt(colno, 10) : undefined,\n in_app: isInApp(filename ?? undefined),\n };\n}\n\nexport function parseStack(error: Error): StackFrame[] {\n const stack = error.stack;\n if (!stack) return [];\n\n const lines = stack.split('\\n');\n const frames: StackFrame[] = [];\n\n for (const line of lines) {\n const trimmed = line.trim();\n if (!trimmed || trimmed.startsWith('Error') || trimmed.startsWith(error.message ?? '')) {\n continue;\n }\n const frame = parseChromeLine(trimmed) ?? parseFirefoxLine(trimmed);\n if (frame) {\n frames.push(frame);\n }\n }\n\n // The API fingerprinting uses the LAST in-app frame, which maps to innermost\n // call. Return in innermost-first order (as captured).\n return frames;\n}\n","import type { BreadcrumbEntry } from './types.js';\n\nexport class BreadcrumbBuffer {\n private readonly buffer: BreadcrumbEntry[] = [];\n private readonly max: number;\n\n constructor(max = 100) {\n this.max = max;\n }\n\n add(crumb: BreadcrumbEntry): void {\n if (this.buffer.length >= this.max) {\n this.buffer.shift();\n }\n this.buffer.push(crumb);\n }\n\n getAll(): BreadcrumbEntry[] {\n return [...this.buffer];\n }\n\n clear(): void {\n this.buffer.length = 0;\n }\n}\n\n// ─── Auto-capture helpers ─────────────────────────────────────────────────────\n\ntype AddFn = (crumb: BreadcrumbEntry) => void;\n\n/**\n * Patch History API (pushState / replaceState / popstate) to capture\n * navigation breadcrumbs.\n */\nexport function installNavigationBreadcrumbs(add: AddFn): () => void {\n if (typeof window === 'undefined') return () => undefined;\n\n const originalPush = history.pushState.bind(history);\n const originalReplace = history.replaceState.bind(history);\n\n history.pushState = function (...args) {\n add({\n timestamp: new Date().toISOString(),\n type: 'navigation',\n category: 'navigation',\n message: `Navigated to ${String(args[2] ?? '')}`,\n data: { to: String(args[2] ?? '') },\n });\n return originalPush(...args);\n };\n\n history.replaceState = function (...args) {\n add({\n timestamp: new Date().toISOString(),\n type: 'navigation',\n category: 'navigation',\n message: `Replaced state: ${String(args[2] ?? '')}`,\n data: { to: String(args[2] ?? '') },\n });\n return originalReplace(...args);\n };\n\n const onPopState = () => {\n add({\n timestamp: new Date().toISOString(),\n type: 'navigation',\n category: 'navigation',\n message: `Navigated back/forward to ${window.location.href}`,\n data: { to: window.location.href },\n });\n };\n\n window.addEventListener('popstate', onPopState);\n\n return () => {\n history.pushState = originalPush;\n history.replaceState = originalReplace;\n window.removeEventListener('popstate', onPopState);\n };\n}\n\n/**\n * Capture click events as breadcrumbs (element tag + text).\n */\nexport function installClickBreadcrumbs(add: AddFn): () => void {\n if (typeof document === 'undefined') return () => undefined;\n\n const handler = (e: MouseEvent) => {\n const target = e.target as HTMLElement | null;\n if (!target) return;\n const tag = target.tagName?.toLowerCase() ?? 'unknown';\n const text = (target.textContent ?? '').trim().slice(0, 50);\n const id = target.id ? `#${target.id}` : '';\n const cls = target.className && typeof target.className === 'string'\n ? `.${target.className.split(' ').join('.')}`\n : '';\n add({\n timestamp: new Date().toISOString(),\n type: 'user',\n category: 'ui.click',\n message: `Click on <${tag}${id}${cls}>`,\n data: { tag, text: text || undefined },\n });\n };\n\n document.addEventListener('click', handler, { capture: true, passive: true });\n\n return () => {\n document.removeEventListener('click', handler, { capture: true });\n };\n}\n\n/**\n * Capture console.warn and console.error calls as breadcrumbs.\n */\nexport function installConsoleBreadcrumbs(add: AddFn): () => void {\n if (typeof console === 'undefined') return () => undefined;\n\n const originalWarn = console.warn.bind(console);\n const originalError = console.error.bind(console);\n\n const formatArgs = (args: unknown[]): string =>\n args.map((a) => (typeof a === 'string' ? a : JSON.stringify(a))).join(' ').slice(0, 200);\n\n console.warn = (...args: unknown[]) => {\n add({\n timestamp: new Date().toISOString(),\n type: 'console',\n category: 'console',\n message: formatArgs(args),\n data: { level: 'warning' },\n });\n originalWarn(...args);\n };\n\n console.error = (...args: unknown[]) => {\n add({\n timestamp: new Date().toISOString(),\n type: 'console',\n category: 'console',\n message: formatArgs(args),\n data: { level: 'error' },\n });\n originalError(...args);\n };\n\n return () => {\n console.warn = originalWarn;\n console.error = originalError;\n };\n}\n","import type {\n BugCatchOptions,\n UserContext,\n BreadcrumbEntry,\n EventPayload,\n ParsedDsn,\n ExceptionValue,\n} from './types.js';\nimport { parseStack } from './parse-stack.js';\nimport {\n BreadcrumbBuffer,\n installNavigationBreadcrumbs,\n installClickBreadcrumbs,\n installConsoleBreadcrumbs,\n} from './breadcrumbs.js';\n\n// ─── UUID ─────────────────────────────────────────────────────────────────────\n\nfunction uuid(): string {\n if (typeof crypto !== 'undefined' && typeof crypto.randomUUID === 'function') {\n return crypto.randomUUID();\n }\n // Fallback for older environments\n return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, (c) => {\n const r = (Math.random() * 16) | 0;\n const v = c === 'x' ? r : (r & 0x3) | 0x8;\n return v.toString(16);\n });\n}\n\n// ─── DSN parser ───────────────────────────────────────────────────────────────\n\nfunction parseDsn(dsn: string): ParsedDsn {\n let url: URL;\n try {\n url = new URL(dsn);\n } catch {\n throw new Error(`[BugCatch] Invalid DSN: \"${dsn}\"`);\n }\n\n const sdkKey = url.searchParams.get('key');\n if (!sdkKey) {\n throw new Error(`[BugCatch] DSN is missing the \"key\" query parameter: \"${dsn}\"`);\n }\n\n // Extract projectId from the last path segment (/ingest/{projectId})\n const segments = url.pathname.split('/').filter(Boolean);\n const projectId = segments[segments.length - 1];\n if (!projectId) {\n throw new Error(`[BugCatch] Could not extract projectId from DSN path: \"${dsn}\"`);\n }\n\n // Reconstruct the URL without extra query params\n const ingestUrl = `${url.origin}${url.pathname}?key=${sdkKey}`;\n\n return { ingestUrl, projectId, sdkKey };\n}\n\n// ─── Pattern matching ─────────────────────────────────────────────────────────\n\nfunction matchesPattern(value: string, patterns: Array<string | RegExp>): boolean {\n return patterns.some((p) =>\n typeof p === 'string' ? value.includes(p) : p.test(value),\n );\n}\n\n// ─── Client ───────────────────────────────────────────────────────────────────\n\nexport class BugCatchClient {\n private readonly opts: Required<\n Pick<BugCatchOptions, 'debug' | 'maxBreadcrumbs' | 'autoCaptureErrors' | 'autoCaptureBreadcrumbs'>\n > & BugCatchOptions;\n\n private readonly dsn: ParsedDsn;\n private readonly crumbs: BreadcrumbBuffer;\n private user: UserContext = {};\n private tags: Record<string, string> = {};\n private readonly cleanups: Array<() => void> = [];\n\n constructor(options: BugCatchOptions) {\n this.opts = {\n debug: false,\n maxBreadcrumbs: 100,\n autoCaptureErrors: true,\n autoCaptureBreadcrumbs: true,\n ...options,\n };\n\n this.dsn = parseDsn(options.dsn);\n this.crumbs = new BreadcrumbBuffer(this.opts.maxBreadcrumbs);\n\n if (this.opts.autoCaptureErrors) {\n this.installErrorHandlers();\n }\n\n if (this.opts.autoCaptureBreadcrumbs) {\n this.cleanups.push(installNavigationBreadcrumbs((c) => this.addBreadcrumb(c)));\n this.cleanups.push(installClickBreadcrumbs((c) => this.addBreadcrumb(c)));\n this.cleanups.push(installConsoleBreadcrumbs((c) => this.addBreadcrumb(c)));\n }\n\n this.log('Initialized. Project:', this.dsn.projectId);\n }\n\n // ─── Public API ─────────────────────────────────────────────────────────────\n\n captureException(error: unknown, extra?: Record<string, unknown>): string {\n const event = this.buildExceptionPayload(error, extra);\n void this.send(event);\n return event.event_id;\n }\n\n captureMessage(message: string, level: string = 'info', extra?: Record<string, unknown>): string {\n const event = this.buildMessagePayload(message, level, extra);\n void this.send(event);\n return event.event_id;\n }\n\n setUser(user: UserContext): void {\n this.user = {\n ...user,\n id: user.id != null ? String(user.id) : undefined,\n };\n }\n\n clearUser(): void {\n this.user = {};\n }\n\n setTag(key: string, value: string): void {\n this.tags[key] = value;\n }\n\n addBreadcrumb(crumb: BreadcrumbEntry): void {\n this.crumbs.add(crumb);\n }\n\n /** Tear down all global listeners. Call when unmounting in SPAs. */\n destroy(): void {\n for (const cleanup of this.cleanups) {\n cleanup();\n }\n this.cleanups.length = 0;\n }\n\n // ─── Internal: build payloads ────────────────────────────────────────────\n\n private buildExceptionPayload(\n error: unknown,\n extra?: Record<string, unknown>,\n ): EventPayload {\n const err = error instanceof Error ? error : new Error(String(error));\n const frames = parseStack(err);\n\n const exceptionValue: ExceptionValue = {\n type: err.name || 'Error',\n value: err.message,\n stacktrace: frames.length > 0 ? { frames } : undefined,\n };\n\n return this.buildBase({\n level: 'error',\n exception: { values: [exceptionValue] },\n extra,\n });\n }\n\n private buildMessagePayload(\n message: string,\n level: string,\n extra?: Record<string, unknown>,\n ): EventPayload {\n return this.buildBase({ level, message, extra });\n }\n\n private buildBase(\n overrides: Partial<EventPayload> & { extra?: Record<string, unknown> },\n ): EventPayload {\n const { extra, ...rest } = overrides;\n const payload: EventPayload = {\n event_id: uuid(),\n timestamp: new Date().toISOString(),\n platform: 'javascript',\n ...rest,\n };\n\n if (this.opts.release) payload.release = this.opts.release;\n if (this.opts.environment) payload.environment = this.opts.environment;\n\n if (Object.keys(this.user).length > 0) {\n payload.user = { ...this.user };\n }\n\n if (Object.keys(this.tags).length > 0) {\n payload.tags = { ...this.tags };\n }\n\n if (extra && Object.keys(extra).length > 0) {\n payload.extra = extra;\n }\n\n const breadcrumbs = this.crumbs.getAll();\n if (breadcrumbs.length > 0) {\n payload.breadcrumbs = breadcrumbs;\n }\n\n // Capture browser request context\n if (typeof window !== 'undefined') {\n payload.request = {\n url: window.location.href,\n headers: {\n 'user-agent': navigator.userAgent,\n },\n };\n }\n\n return payload;\n }\n\n // ─── Internal: error handlers ────────────────────────────────────────────\n\n private installErrorHandlers(): void {\n if (typeof window === 'undefined') {\n // Node.js environment\n const onUncaught = (err: Error) => {\n this.captureException(err, { handler: 'uncaughtException' });\n };\n const onUnhandled = (reason: unknown) => {\n this.captureException(reason, { handler: 'unhandledRejection' });\n };\n process.on('uncaughtException', onUncaught);\n process.on('unhandledRejection', onUnhandled);\n this.cleanups.push(() => {\n process.off('uncaughtException', onUncaught);\n process.off('unhandledRejection', onUnhandled);\n });\n return;\n }\n\n // Browser environment\n const onError = (event: ErrorEvent) => {\n if (!event.error) return;\n if (this.shouldIgnore(event.error, event.filename ?? '')) return;\n this.captureException(event.error, { handler: 'window.onerror' });\n };\n\n const onUnhandledRejection = (event: PromiseRejectionEvent) => {\n const reason = event.reason as unknown;\n if (this.shouldIgnore(reason, '')) return;\n this.captureException(reason, { handler: 'unhandledrejection' });\n };\n\n window.addEventListener('error', onError);\n window.addEventListener('unhandledrejection', onUnhandledRejection);\n\n this.cleanups.push(() => {\n window.removeEventListener('error', onError);\n window.removeEventListener('unhandledrejection', onUnhandledRejection);\n });\n }\n\n private shouldIgnore(error: unknown, scriptUrl: string): boolean {\n const { ignoreUrls, ignoreErrors } = this.opts;\n\n if (ignoreUrls?.length && scriptUrl) {\n if (matchesPattern(scriptUrl, ignoreUrls)) return true;\n }\n\n if (ignoreErrors?.length) {\n const message = error instanceof Error ? error.message : String(error);\n if (matchesPattern(message, ignoreErrors)) return true;\n }\n\n return false;\n }\n\n // ─── Internal: transport ─────────────────────────────────────────────────\n\n private async send(event: EventPayload): Promise<void> {\n let finalEvent: EventPayload | false = event;\n\n if (this.opts.beforeSend) {\n finalEvent = this.opts.beforeSend(event);\n if (finalEvent === false) {\n this.log('Event dropped by beforeSend hook');\n return;\n }\n }\n\n this.log('Sending event:', finalEvent.event_id, finalEvent.level);\n\n let body: string;\n try {\n body = JSON.stringify(finalEvent);\n } catch (err) {\n console.warn('[BugCatch] Failed to serialize event:', err);\n return;\n }\n\n try {\n const fetchOptions: RequestInit = {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body,\n };\n\n // keepalive is browser-only; Node.js fetch (undici) enforces a 64 KB body\n // limit with keepalive: true, which silently drops large payloads.\n if (typeof window !== 'undefined') {\n fetchOptions.keepalive = true;\n }\n\n const res = await fetch(this.dsn.ingestUrl, fetchOptions);\n\n if (!res.ok) {\n console.warn(`[BugCatch] Ingest responded with ${res.status}:`, await res.text());\n }\n } catch (err) {\n console.warn('[BugCatch] Failed to send event:', err);\n }\n }\n\n private log(...args: unknown[]): void {\n if (this.opts.debug) {\n console.debug('[BugCatch]', ...args);\n }\n }\n}\n","import { BugCatchClient } from './client.js';\nimport type { BugCatchOptions, UserContext, BreadcrumbEntry } from './types.js';\n\nexport type { BugCatchOptions, UserContext, BreadcrumbEntry };\nexport type { EventPayload, StackFrame, ParsedDsn } from './types.js';\nexport { BugCatchClient };\n\n// ─── Singleton instance ───────────────────────────────────────────────────────\n\nlet _client: BugCatchClient | null = null;\n\nfunction getClient(): BugCatchClient {\n if (!_client) {\n throw new Error('[BugCatch] SDK not initialized. Call BugCatch.init(options) first.');\n }\n return _client;\n}\n\n// ─── Public singleton API ─────────────────────────────────────────────────────\n\nconst BugCatch = {\n /**\n * Initialize the SDK. Call this once, as early as possible in your app.\n *\n * @example\n * BugCatch.init({\n * dsn: 'http://localhost:3000/ingest/project-id?key=sdk-key',\n * release: '1.0.0',\n * environment: 'production',\n * });\n */\n init(options: BugCatchOptions): BugCatchClient {\n if (_client) {\n console.warn('[BugCatch] init() called more than once. Ignoring.');\n return _client;\n }\n _client = new BugCatchClient(options);\n return _client;\n },\n\n /**\n * Capture an Error object or any value as an error event.\n * Returns the generated event_id.\n */\n captureException(error: unknown, extra?: Record<string, unknown>): string {\n return getClient().captureException(error, extra);\n },\n\n /**\n * Capture a plain text message as an event.\n * Returns the generated event_id.\n */\n captureMessage(message: string, level?: string, extra?: Record<string, unknown>): string {\n return getClient().captureMessage(message, level, extra);\n },\n\n /**\n * Set the current user context. Attached to all subsequent events.\n */\n setUser(user: UserContext): void {\n getClient().setUser(user);\n },\n\n /**\n * Clear the current user context (e.g. on logout).\n */\n clearUser(): void {\n getClient().clearUser();\n },\n\n /**\n * Set a tag that will be attached to all subsequent events.\n */\n setTag(key: string, value: string): void {\n getClient().setTag(key, value);\n },\n\n /**\n * Manually add a breadcrumb.\n */\n addBreadcrumb(crumb: BreadcrumbEntry): void {\n getClient().addBreadcrumb(crumb);\n },\n\n /**\n * Tear down all listeners. Useful in SPA cleanup or hot-reload scenarios.\n */\n destroy(): void {\n _client?.destroy();\n _client = null;\n },\n};\n\n// Named export for CJS consumers: const { BugCatch } = require('bugcatch-sdk')\nexport { BugCatch };\n\n// Default export for ESM consumers: import BugCatch from 'bugcatch-sdk'\nexport default BugCatch;\n"]}
|
|
1
|
+
{"version":3,"sources":["../src/parse-stack.ts","../src/breadcrumbs.ts","../src/client.ts","../src/index.ts"],"names":[],"mappings":"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","file":"index.mjs","sourcesContent":["import type { StackFrame } from './types.js';\n\n// Chrome/V8: \" at functionName (filename:line:col)\"\n// \" at filename:line:col\"\nconst CHROME_RE = /^\\s*at (?:(.*?) \\()?(.+?):(\\d+):(\\d+)\\)?$/;\n\n// Firefox/Safari: \"functionName@filename:line:col\"\n// \"@filename:line:col\"\nconst FIREFOX_RE = /^(?:(.+?)@)?(.+?):(\\d+):(\\d+)$/;\n\n// Origins that are not \"in-app\" frames\nconst INTERNAL_PATTERNS = [/node_modules\\//, /bugcatch[-_]?sdk/, /^\\(native\\)$/, /^native code/];\n\nfunction isInApp(filename: string | undefined): boolean {\n if (!filename) return false;\n return !INTERNAL_PATTERNS.some((p) => p.test(filename));\n}\n\nfunction parseChromeLine(line: string): StackFrame | null {\n const m = CHROME_RE.exec(line);\n if (!m) return null;\n const [, fn, filename, lineno, colno] = m;\n return {\n function: fn ?? '<anonymous>',\n filename: filename ?? undefined,\n lineno: lineno ? parseInt(lineno, 10) : undefined,\n colno: colno ? parseInt(colno, 10) : undefined,\n in_app: isInApp(filename ?? undefined),\n };\n}\n\nfunction parseFirefoxLine(line: string): StackFrame | null {\n const m = FIREFOX_RE.exec(line);\n if (!m) return null;\n const [, fn, filename, lineno, colno] = m;\n return {\n function: fn ?? '<anonymous>',\n filename: filename ?? undefined,\n lineno: lineno ? parseInt(lineno, 10) : undefined,\n colno: colno ? parseInt(colno, 10) : undefined,\n in_app: isInApp(filename ?? undefined),\n };\n}\n\nexport function parseStack(error: Error): StackFrame[] {\n const stack = error.stack;\n if (!stack) return [];\n\n const lines = stack.split('\\n');\n const frames: StackFrame[] = [];\n\n for (const line of lines) {\n const trimmed = line.trim();\n if (!trimmed || trimmed.startsWith('Error') || trimmed.startsWith(error.message ?? '')) {\n continue;\n }\n const frame = parseChromeLine(trimmed) ?? parseFirefoxLine(trimmed);\n if (frame) {\n frames.push(frame);\n }\n }\n\n // The API fingerprinting uses the LAST in-app frame, which maps to innermost\n // call. Return in innermost-first order (as captured).\n return frames;\n}\n","import type { BreadcrumbEntry } from './types.js';\n\nexport class BreadcrumbBuffer {\n private readonly buffer: BreadcrumbEntry[] = [];\n private readonly max: number;\n\n constructor(max = 100) {\n this.max = max;\n }\n\n add(crumb: BreadcrumbEntry): void {\n if (this.buffer.length >= this.max) {\n this.buffer.shift();\n }\n this.buffer.push(crumb);\n }\n\n getAll(): BreadcrumbEntry[] {\n return [...this.buffer];\n }\n\n clear(): void {\n this.buffer.length = 0;\n }\n}\n\n// ─── Auto-capture helpers ─────────────────────────────────────────────────────\n\ntype AddFn = (crumb: BreadcrumbEntry) => void;\n\n/**\n * Patch History API (pushState / replaceState / popstate) to capture\n * navigation breadcrumbs.\n */\nexport function installNavigationBreadcrumbs(add: AddFn): () => void {\n if (typeof window === 'undefined') return () => undefined;\n\n const originalPush = history.pushState.bind(history);\n const originalReplace = history.replaceState.bind(history);\n\n history.pushState = function (...args) {\n add({\n timestamp: new Date().toISOString(),\n type: 'navigation',\n category: 'navigation',\n message: `Navigated to ${String(args[2] ?? '')}`,\n data: { to: String(args[2] ?? '') },\n });\n return originalPush(...args);\n };\n\n history.replaceState = function (...args) {\n add({\n timestamp: new Date().toISOString(),\n type: 'navigation',\n category: 'navigation',\n message: `Replaced state: ${String(args[2] ?? '')}`,\n data: { to: String(args[2] ?? '') },\n });\n return originalReplace(...args);\n };\n\n const onPopState = () => {\n add({\n timestamp: new Date().toISOString(),\n type: 'navigation',\n category: 'navigation',\n message: `Navigated back/forward to ${window.location.href}`,\n data: { to: window.location.href },\n });\n };\n\n window.addEventListener('popstate', onPopState);\n\n return () => {\n history.pushState = originalPush;\n history.replaceState = originalReplace;\n window.removeEventListener('popstate', onPopState);\n };\n}\n\n/**\n * Capture click events as breadcrumbs (element tag + text).\n */\nexport function installClickBreadcrumbs(add: AddFn): () => void {\n if (typeof document === 'undefined') return () => undefined;\n\n const handler = (e: MouseEvent) => {\n const target = e.target as HTMLElement | null;\n if (!target) return;\n const tag = target.tagName?.toLowerCase() ?? 'unknown';\n const text = (target.textContent ?? '').trim().slice(0, 50);\n const id = target.id ? `#${target.id}` : '';\n const cls = target.className && typeof target.className === 'string'\n ? `.${target.className.split(' ').join('.')}`\n : '';\n add({\n timestamp: new Date().toISOString(),\n type: 'user',\n category: 'ui.click',\n message: `Click on <${tag}${id}${cls}>`,\n data: { tag, text: text || undefined },\n });\n };\n\n document.addEventListener('click', handler, { capture: true, passive: true });\n\n return () => {\n document.removeEventListener('click', handler, { capture: true });\n };\n}\n\n/**\n * Capture console.warn and console.error calls as breadcrumbs.\n */\nexport function installConsoleBreadcrumbs(add: AddFn): () => void {\n if (typeof console === 'undefined') return () => undefined;\n\n const originalWarn = console.warn.bind(console);\n const originalError = console.error.bind(console);\n\n const formatArgs = (args: unknown[]): string =>\n args.map((a) => (typeof a === 'string' ? a : JSON.stringify(a))).join(' ').slice(0, 200);\n\n console.warn = (...args: unknown[]) => {\n add({\n timestamp: new Date().toISOString(),\n type: 'console',\n category: 'console',\n message: formatArgs(args),\n data: { level: 'warning' },\n });\n originalWarn(...args);\n };\n\n console.error = (...args: unknown[]) => {\n add({\n timestamp: new Date().toISOString(),\n type: 'console',\n category: 'console',\n message: formatArgs(args),\n data: { level: 'error' },\n });\n originalError(...args);\n };\n\n return () => {\n console.warn = originalWarn;\n console.error = originalError;\n };\n}\n","import type {\n BugCatchOptions,\n UserContext,\n BreadcrumbEntry,\n EventPayload,\n MetricPayload,\n ParsedDsn,\n ExceptionValue,\n} from './types.js';\nimport { parseStack } from './parse-stack.js';\nimport {\n BreadcrumbBuffer,\n installNavigationBreadcrumbs,\n installClickBreadcrumbs,\n installConsoleBreadcrumbs,\n} from './breadcrumbs.js';\n\n// ─── UUID ─────────────────────────────────────────────────────────────────────\n\nfunction uuid(): string {\n if (typeof crypto !== 'undefined' && typeof crypto.randomUUID === 'function') {\n return crypto.randomUUID();\n }\n // Fallback for older environments\n return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, (c) => {\n const r = (Math.random() * 16) | 0;\n const v = c === 'x' ? r : (r & 0x3) | 0x8;\n return v.toString(16);\n });\n}\n\n// ─── DSN parser ───────────────────────────────────────────────────────────────\n\nfunction parseDsn(dsn: string): ParsedDsn & { metricsUrl: string } {\n let url: URL;\n try {\n url = new URL(dsn);\n } catch {\n throw new Error(`[BugCatch] Invalid DSN: \"${dsn}\"`);\n }\n\n const sdkKey = url.searchParams.get('key');\n if (!sdkKey) {\n throw new Error(`[BugCatch] DSN is missing the \"key\" query parameter: \"${dsn}\"`);\n }\n\n // Extract projectId from the last path segment (/ingest/{projectId})\n const segments = url.pathname.split('/').filter(Boolean);\n const projectId = segments[segments.length - 1];\n if (!projectId) {\n throw new Error(`[BugCatch] Could not extract projectId from DSN path: \"${dsn}\"`);\n }\n\n // Reconstruct the URL without extra query params\n const ingestUrl = `${url.origin}${url.pathname}?key=${sdkKey}`;\n const metricsUrl = `${url.origin}${url.pathname}/metrics?key=${sdkKey}`;\n\n return { ingestUrl, metricsUrl, projectId, sdkKey };\n}\n\n// ─── Route normalizer ─────────────────────────────────────────────────────────\n\n/**\n * Strips the origin and query string from a URL and replaces dynamic path\n * segments (UUIDs, numeric IDs) with `:id` so metrics group by route template.\n *\n * e.g. https://api.example.com/users/42/orders/abc-123?page=1 → /users/:id/orders/:id\n */\nfunction normalizeUrl(rawUrl: string): string {\n let pathname: string;\n try {\n pathname = new URL(rawUrl).pathname;\n } catch {\n // rawUrl is already a path (e.g. \"/api/users/1\")\n pathname = rawUrl.split('?')[0];\n }\n return pathname\n .replace(/\\/[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}/gi, '/:id')\n .replace(/\\/\\d{1,20}(?=\\/|$)/g, '/:id');\n}\n\n// ─── Pattern matching ─────────────────────────────────────────────────────────\n\nfunction matchesPattern(value: string, patterns: Array<string | RegExp>): boolean {\n return patterns.some((p) =>\n typeof p === 'string' ? value.includes(p) : p.test(value),\n );\n}\n\n// ─── Client ───────────────────────────────────────────────────────────────────\n\nexport class BugCatchClient {\n private readonly opts: Required<\n Pick<BugCatchOptions, 'debug' | 'maxBreadcrumbs' | 'autoCaptureErrors' | 'autoCaptureBreadcrumbs' | 'autoTrackRequests'>\n > & BugCatchOptions;\n\n private readonly dsn: ParsedDsn & { metricsUrl: string };\n private readonly crumbs: BreadcrumbBuffer;\n private user: UserContext = {};\n private tags: Record<string, string> = {};\n private readonly cleanups: Array<() => void> = [];\n\n constructor(options: BugCatchOptions) {\n this.opts = {\n debug: false,\n maxBreadcrumbs: 100,\n autoCaptureErrors: true,\n autoCaptureBreadcrumbs: true,\n autoTrackRequests: false,\n ...options,\n };\n\n this.dsn = parseDsn(options.dsn);\n this.crumbs = new BreadcrumbBuffer(this.opts.maxBreadcrumbs);\n\n if (this.opts.autoCaptureErrors) {\n this.installErrorHandlers();\n }\n\n if (this.opts.autoCaptureBreadcrumbs) {\n this.cleanups.push(installNavigationBreadcrumbs((c) => this.addBreadcrumb(c)));\n this.cleanups.push(installClickBreadcrumbs((c) => this.addBreadcrumb(c)));\n this.cleanups.push(installConsoleBreadcrumbs((c) => this.addBreadcrumb(c)));\n }\n\n if (this.opts.autoTrackRequests) {\n this.cleanups.push(this.installFetchInterceptor());\n this.cleanups.push(this.installXhrInterceptor());\n }\n\n this.log('Initialized. Project:', this.dsn.projectId);\n }\n\n // ─── Public API ─────────────────────────────────────────────────────────────\n\n captureException(error: unknown, extra?: Record<string, unknown>): string {\n const event = this.buildExceptionPayload(error, extra);\n void this.send(event);\n return event.event_id;\n }\n\n captureMessage(message: string, level: string = 'info', extra?: Record<string, unknown>): string {\n const event = this.buildMessagePayload(message, level, extra);\n void this.send(event);\n return event.event_id;\n }\n\n setUser(user: UserContext): void {\n this.user = {\n ...user,\n id: user.id != null ? String(user.id) : undefined,\n };\n }\n\n clearUser(): void {\n this.user = {};\n }\n\n setTag(key: string, value: string): void {\n this.tags[key] = value;\n }\n\n addBreadcrumb(crumb: BreadcrumbEntry): void {\n this.crumbs.add(crumb);\n }\n\n /** Tear down all global listeners. Call when unmounting in SPAs. */\n destroy(): void {\n for (const cleanup of this.cleanups) {\n cleanup();\n }\n this.cleanups.length = 0;\n }\n\n /**\n * Manually report an API call timing. Use this when `autoTrackRequests`\n * is off or when you want to track server-side (Node.js) requests explicitly.\n */\n trackRequest(method: string, route: string, durationMs: number, statusCode: number): void {\n void this.sendMetric({ method: method.toUpperCase(), route, duration_ms: durationMs, status_code: statusCode });\n }\n\n // ─── Internal: build payloads ────────────────────────────────────────────\n\n private buildExceptionPayload(\n error: unknown,\n extra?: Record<string, unknown>,\n ): EventPayload {\n const err = error instanceof Error ? error : new Error(String(error));\n const frames = parseStack(err);\n\n const exceptionValue: ExceptionValue = {\n type: err.name || 'Error',\n value: err.message,\n stacktrace: frames.length > 0 ? { frames } : undefined,\n };\n\n return this.buildBase({\n level: 'error',\n exception: { values: [exceptionValue] },\n extra,\n });\n }\n\n private buildMessagePayload(\n message: string,\n level: string,\n extra?: Record<string, unknown>,\n ): EventPayload {\n return this.buildBase({ level, message, extra });\n }\n\n private buildBase(\n overrides: Partial<EventPayload> & { extra?: Record<string, unknown> },\n ): EventPayload {\n const { extra, ...rest } = overrides;\n const payload: EventPayload = {\n event_id: uuid(),\n timestamp: new Date().toISOString(),\n platform: 'javascript',\n ...rest,\n };\n\n if (this.opts.release) payload.release = this.opts.release;\n if (this.opts.environment) payload.environment = this.opts.environment;\n\n if (Object.keys(this.user).length > 0) {\n payload.user = { ...this.user };\n }\n\n if (Object.keys(this.tags).length > 0) {\n payload.tags = { ...this.tags };\n }\n\n if (extra && Object.keys(extra).length > 0) {\n payload.extra = extra;\n }\n\n const breadcrumbs = this.crumbs.getAll();\n if (breadcrumbs.length > 0) {\n payload.breadcrumbs = breadcrumbs;\n }\n\n // Capture browser request context\n if (typeof window !== 'undefined') {\n payload.request = {\n url: window.location.href,\n headers: {\n 'user-agent': navigator.userAgent,\n },\n };\n }\n\n return payload;\n }\n\n // ─── Internal: error handlers ────────────────────────────────────────────\n\n private installErrorHandlers(): void {\n if (typeof window === 'undefined') {\n // Node.js environment\n const onUncaught = (err: Error) => {\n this.captureException(err, { handler: 'uncaughtException' });\n };\n const onUnhandled = (reason: unknown) => {\n this.captureException(reason, { handler: 'unhandledRejection' });\n };\n process.on('uncaughtException', onUncaught);\n process.on('unhandledRejection', onUnhandled);\n this.cleanups.push(() => {\n process.off('uncaughtException', onUncaught);\n process.off('unhandledRejection', onUnhandled);\n });\n return;\n }\n\n // Browser environment\n const onError = (event: ErrorEvent) => {\n if (!event.error) return;\n if (this.shouldIgnore(event.error, event.filename ?? '')) return;\n this.captureException(event.error, { handler: 'window.onerror' });\n };\n\n const onUnhandledRejection = (event: PromiseRejectionEvent) => {\n const reason = event.reason as unknown;\n if (this.shouldIgnore(reason, '')) return;\n this.captureException(reason, { handler: 'unhandledrejection' });\n };\n\n window.addEventListener('error', onError);\n window.addEventListener('unhandledrejection', onUnhandledRejection);\n\n this.cleanups.push(() => {\n window.removeEventListener('error', onError);\n window.removeEventListener('unhandledrejection', onUnhandledRejection);\n });\n }\n\n private shouldIgnore(error: unknown, scriptUrl: string): boolean {\n const { ignoreUrls, ignoreErrors } = this.opts;\n\n if (ignoreUrls?.length && scriptUrl) {\n if (matchesPattern(scriptUrl, ignoreUrls)) return true;\n }\n\n if (ignoreErrors?.length) {\n const message = error instanceof Error ? error.message : String(error);\n if (matchesPattern(message, ignoreErrors)) return true;\n }\n\n return false;\n }\n\n // ─── Internal: request tracking ──────────────────────────────────────────\n\n private installFetchInterceptor(): () => void {\n if (typeof globalThis.fetch !== 'function') {\n console.warn('[BugCatch] autoTrackRequests is enabled but globalThis.fetch is not available. Metrics will not be tracked.');\n return () => {};\n }\n\n const original = globalThis.fetch.bind(globalThis);\n const self = this;\n\n this.log('autoTrackRequests: fetch interceptor installed');\n\n globalThis.fetch = async function patchedFetch(\n input: RequestInfo | URL,\n init?: RequestInit,\n ): Promise<Response> {\n const url = input instanceof Request ? input.url : String(input);\n\n // Never intercept BugCatch's own ingest/metrics requests\n if (url.includes(self.dsn.ingestUrl.split('?')[0])) {\n return original(input, init);\n }\n\n if (self.opts.trackIgnoreUrls && matchesPattern(url, self.opts.trackIgnoreUrls)) {\n self.log('autoTrackRequests: ignored (trackIgnoreUrls match)', url);\n return original(input, init);\n }\n\n const method = (init?.method ?? (input instanceof Request ? input.method : 'GET')).toUpperCase();\n const route = normalizeUrl(url);\n const start = Date.now();\n\n self.log('autoTrackRequests: intercepted', method, url);\n\n try {\n const response = await original(input, init);\n void self.sendMetric({\n method,\n route,\n duration_ms: Date.now() - start,\n status_code: response.status,\n user_id: self.user.id,\n });\n return response;\n } catch (err) {\n // Network error — report as status 0\n void self.sendMetric({ method, route, duration_ms: Date.now() - start, status_code: 0 });\n throw err;\n }\n };\n\n // Return cleanup function that restores original fetch\n return () => {\n globalThis.fetch = original;\n this.log('autoTrackRequests: fetch interceptor removed');\n };\n }\n\n private installXhrInterceptor(): () => void {\n if (typeof XMLHttpRequest === 'undefined') {\n // Node.js environment — XHR not available\n return () => {};\n }\n\n const self = this;\n const OriginalXHR = XMLHttpRequest;\n const ingestBase = this.dsn.ingestUrl.split('?')[0];\n\n function PatchedXHR(this: XMLHttpRequest) {\n const xhr = new OriginalXHR();\n\n let method = 'GET';\n let url = '';\n let start = 0;\n\n const originalOpen = xhr.open.bind(xhr);\n const originalSend = xhr.send.bind(xhr);\n\n (this as any).open = function (m: string, u: string, async?: boolean, user?: string, password?: string) {\n method = m ? m.toUpperCase() : 'GET';\n url = u || '';\n return originalOpen(m, u, async ?? true, user, password);\n };\n\n (this as any).send = function (body?: Document | XMLHttpRequestBodyInit | null) {\n // Skip BugCatch's own requests and ignored URLs\n const skip =\n url.includes(ingestBase) ||\n (self.opts.trackIgnoreUrls ? matchesPattern(url, self.opts.trackIgnoreUrls) : false);\n\n if (!skip) {\n start = Date.now();\n self.log('autoTrackRequests: XHR intercepted', method, url);\n\n xhr.addEventListener('loadend', () => {\n void self.sendMetric({\n method,\n route: normalizeUrl(url),\n duration_ms: Date.now() - start,\n status_code: xhr.status || 0,\n user_id: self.user.id,\n });\n });\n }\n\n return originalSend(body);\n };\n\n // Proxy all other properties/methods to the real XHR instance\n return new Proxy(this, {\n get(_target, prop) {\n if (prop === 'open') return (this as any).open;\n if (prop === 'send') return (this as any).send;\n const val = (xhr as any)[prop];\n return typeof val === 'function' ? val.bind(xhr) : val;\n },\n set(_target, prop, value) {\n (xhr as any)[prop] = value;\n return true;\n },\n });\n }\n\n PatchedXHR.prototype = OriginalXHR.prototype;\n (window as any).XMLHttpRequest = PatchedXHR;\n\n this.log('autoTrackRequests: XHR interceptor installed');\n\n return () => {\n (window as any).XMLHttpRequest = OriginalXHR;\n this.log('autoTrackRequests: XHR interceptor removed');\n };\n }\n\n private async sendMetric(metric: MetricPayload): Promise<void> {\n this.log('autoTrackRequests: sending metric', metric.method, metric.route, metric.duration_ms + 'ms', metric.status_code);\n try {\n const res = await fetch(this.dsn.metricsUrl, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify(metric),\n });\n if (!res.ok) {\n this.log('autoTrackRequests: metric rejected by server —', res.status, await res.text().catch(() => ''));\n }\n } catch (err) {\n this.log('autoTrackRequests: failed to send metric —', (err as Error).message);\n }\n }\n\n // ─── Internal: transport ─────────────────────────────────────────────────\n\n private async send(event: EventPayload): Promise<void> {\n let finalEvent: EventPayload | false = event;\n\n if (this.opts.beforeSend) {\n finalEvent = this.opts.beforeSend(event);\n if (finalEvent === false) {\n this.log('Event dropped by beforeSend hook');\n return;\n }\n }\n\n this.log('Sending event:', finalEvent.event_id, finalEvent.level);\n\n let body: string;\n try {\n body = JSON.stringify(finalEvent);\n } catch (err) {\n console.warn('[BugCatch] Failed to serialize event:', err);\n return;\n }\n\n try {\n const fetchOptions: RequestInit = {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body,\n };\n\n // keepalive is browser-only; Node.js fetch (undici) enforces a 64 KB body\n // limit with keepalive: true, which silently drops large payloads.\n if (typeof window !== 'undefined') {\n fetchOptions.keepalive = true;\n }\n\n const res = await fetch(this.dsn.ingestUrl, fetchOptions);\n\n if (!res.ok) {\n console.warn(`[BugCatch] Ingest responded with ${res.status}:`, await res.text());\n }\n } catch (err) {\n console.warn('[BugCatch] Failed to send event:', err);\n }\n }\n\n private log(...args: unknown[]): void {\n if (this.opts.debug) {\n console.debug('[BugCatch]', ...args);\n }\n }\n}\n","import { BugCatchClient } from './client.js';\nimport type { BugCatchOptions, UserContext, BreadcrumbEntry } from './types.js';\n\nexport type { BugCatchOptions, UserContext, BreadcrumbEntry };\nexport type { EventPayload, StackFrame, ParsedDsn } from './types.js';\nexport { BugCatchClient };\n\n// ─── Singleton instance ───────────────────────────────────────────────────────\n\nlet _client: BugCatchClient | null = null;\n\nfunction getClient(): BugCatchClient {\n if (!_client) {\n throw new Error('[BugCatch] SDK not initialized. Call BugCatch.init(options) first.');\n }\n return _client;\n}\n\n// ─── Public singleton API ─────────────────────────────────────────────────────\n\nconst BugCatch = {\n /**\n * Initialize the SDK. Call this once, as early as possible in your app.\n *\n * @example\n * BugCatch.init({\n * dsn: 'http://localhost:3000/ingest/project-id?key=sdk-key',\n * release: '1.0.0',\n * environment: 'production',\n * });\n */\n init(options: BugCatchOptions): BugCatchClient {\n if (_client) {\n console.warn('[BugCatch] init() called more than once. Ignoring.');\n return _client;\n }\n _client = new BugCatchClient(options);\n return _client;\n },\n\n /**\n * Capture an Error object or any value as an error event.\n * Returns the generated event_id.\n */\n captureException(error: unknown, extra?: Record<string, unknown>): string {\n return getClient().captureException(error, extra);\n },\n\n /**\n * Capture a plain text message as an event.\n * Returns the generated event_id.\n */\n captureMessage(message: string, level?: string, extra?: Record<string, unknown>): string {\n return getClient().captureMessage(message, level, extra);\n },\n\n /**\n * Set the current user context. Attached to all subsequent events.\n */\n setUser(user: UserContext): void {\n getClient().setUser(user);\n },\n\n /**\n * Clear the current user context (e.g. on logout).\n */\n clearUser(): void {\n getClient().clearUser();\n },\n\n /**\n * Set a tag that will be attached to all subsequent events.\n */\n setTag(key: string, value: string): void {\n getClient().setTag(key, value);\n },\n\n /**\n * Manually add a breadcrumb.\n */\n addBreadcrumb(crumb: BreadcrumbEntry): void {\n getClient().addBreadcrumb(crumb);\n },\n\n /**\n * Manually report an API call timing to BugCatch.\n * Use when `autoTrackRequests` is off or for server-side tracking.\n *\n * @example\n * const start = Date.now();\n * await fetch('/api/orders');\n * BugCatch.trackRequest('GET', '/api/orders', Date.now() - start, 200);\n */\n trackRequest(method: string, route: string, durationMs: number, statusCode: number): void {\n getClient().trackRequest(method, route, durationMs, statusCode);\n },\n\n /**\n * Tear down all listeners. Useful in SPA cleanup or hot-reload scenarios.\n */\n destroy(): void {\n _client?.destroy();\n _client = null;\n },\n};\n\n// Named export for CJS consumers: const { BugCatch } = require('bugcatch-sdk')\nexport { BugCatch };\n\n// Default export for ESM consumers: import BugCatch from 'bugcatch-sdk'\nexport default BugCatch;\n"]}
|