@obtrace/browser 1.0.0

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.
Files changed (51) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +85 -0
  3. package/dist/auto.js +126 -0
  4. package/dist/browser/breadcrumbs.js +41 -0
  5. package/dist/browser/clicks.js +68 -0
  6. package/dist/browser/console.js +58 -0
  7. package/dist/browser/errors.js +51 -0
  8. package/dist/browser/index.js +352 -0
  9. package/dist/browser/longtasks.js +24 -0
  10. package/dist/browser/memory.js +19 -0
  11. package/dist/browser/offline.js +115 -0
  12. package/dist/browser/replay.js +127 -0
  13. package/dist/browser/resources.js +24 -0
  14. package/dist/browser/vitals.js +81 -0
  15. package/dist/browser_entry.bundle.js +28780 -0
  16. package/dist/browser_entry.bundle.js.map +7 -0
  17. package/dist/browser_entry.js +3 -0
  18. package/dist/core/client.js +80 -0
  19. package/dist/core/otel-web-setup.js +93 -0
  20. package/dist/index.js +6 -0
  21. package/dist/shared/semantic_metrics.js +28 -0
  22. package/dist/shared/types.js +1 -0
  23. package/dist/shared/utils.js +112 -0
  24. package/dist/sourcemaps.js +32 -0
  25. package/dist/types/auto.d.ts +13 -0
  26. package/dist/types/browser/breadcrumbs.d.ts +12 -0
  27. package/dist/types/browser/clicks.d.ts +2 -0
  28. package/dist/types/browser/console.d.ts +2 -0
  29. package/dist/types/browser/errors.d.ts +2 -0
  30. package/dist/types/browser/index.d.ts +25 -0
  31. package/dist/types/browser/longtasks.d.ts +2 -0
  32. package/dist/types/browser/memory.d.ts +2 -0
  33. package/dist/types/browser/offline.d.ts +3 -0
  34. package/dist/types/browser/replay.d.ts +24 -0
  35. package/dist/types/browser/resources.d.ts +2 -0
  36. package/dist/types/browser/vitals.d.ts +2 -0
  37. package/dist/types/browser_entry.d.ts +6 -0
  38. package/dist/types/core/client.d.ts +19 -0
  39. package/dist/types/core/otel-web-setup.d.ts +8 -0
  40. package/dist/types/index.d.ts +9 -0
  41. package/dist/types/shared/semantic_metrics.d.ts +26 -0
  42. package/dist/types/shared/types.d.ts +104 -0
  43. package/dist/types/shared/utils.d.ts +21 -0
  44. package/dist/types/sourcemaps.d.ts +14 -0
  45. package/dist/types/wrappers/frontend/next.d.ts +4 -0
  46. package/dist/types/wrappers/frontend/react.d.ts +8 -0
  47. package/dist/types/wrappers/frontend/vite.d.ts +5 -0
  48. package/dist/wrappers/frontend/next.js +12 -0
  49. package/dist/wrappers/frontend/react.js +20 -0
  50. package/dist/wrappers/frontend/vite.js +19 -0
  51. package/package.json +82 -0
@@ -0,0 +1,81 @@
1
+ import { getElementSelector } from "./breadcrumbs";
2
+ export function installWebVitals(meter, reportAllChanges) {
3
+ if (typeof window === "undefined" || typeof PerformanceObserver === "undefined") {
4
+ return () => undefined;
5
+ }
6
+ const fcpGauge = meter.createGauge("web_vital_fcp_ms", { unit: "ms" });
7
+ const lcpGauge = meter.createGauge("web_vital_lcp_ms", { unit: "ms" });
8
+ const clsGauge = meter.createGauge("web_vital_cls", { unit: "1" });
9
+ const inpGauge = meter.createGauge("web_vital_inp_ms", { unit: "ms" });
10
+ const ttfbGauge = meter.createGauge("web_vital_ttfb_ms", { unit: "ms" });
11
+ const cleanups = [];
12
+ const observe = (type, cb) => {
13
+ try {
14
+ const observer = new PerformanceObserver((list) => {
15
+ for (const entry of list.getEntries()) {
16
+ cb(entry);
17
+ if (!reportAllChanges)
18
+ break;
19
+ }
20
+ });
21
+ observer.observe({ type, buffered: true });
22
+ cleanups.push(() => observer.disconnect());
23
+ }
24
+ catch {
25
+ return;
26
+ }
27
+ };
28
+ observe("paint", (entry) => {
29
+ if (entry.name === "first-contentful-paint") {
30
+ fcpGauge.record(entry.startTime, { vital: "fcp" });
31
+ }
32
+ });
33
+ observe("largest-contentful-paint", (entry) => {
34
+ const lcp = entry;
35
+ const attrs = { vital: "lcp" };
36
+ if (lcp.element) {
37
+ attrs["lcp.element"] = getElementSelector(lcp.element);
38
+ }
39
+ lcpGauge.record(entry.startTime, attrs);
40
+ });
41
+ observe("layout-shift", (entry) => {
42
+ const ls = entry;
43
+ if (ls.hadRecentInput)
44
+ return;
45
+ const attrs = { vital: "cls" };
46
+ if (ls.sources && ls.sources.length > 0 && ls.sources[0].node) {
47
+ attrs["cls.element"] = getElementSelector(ls.sources[0].node);
48
+ }
49
+ clsGauge.record(ls.value ?? 0, attrs);
50
+ });
51
+ const interactionDurations = new Map();
52
+ observe("event", (entry) => {
53
+ const ev = entry;
54
+ if (!ev.interactionId || ev.interactionId === 0)
55
+ return;
56
+ if (!ev.duration || ev.duration <= 0)
57
+ return;
58
+ const existing = interactionDurations.get(ev.interactionId) ?? 0;
59
+ if (ev.duration > existing) {
60
+ interactionDurations.set(ev.interactionId, ev.duration);
61
+ }
62
+ if (interactionDurations.size > 50) {
63
+ const sorted = [...interactionDurations.values()].sort((a, b) => b - a);
64
+ const p98Index = Math.max(0, Math.ceil(sorted.length * 0.02) - 1);
65
+ inpGauge.record(sorted[p98Index], { vital: "inp" });
66
+ }
67
+ });
68
+ const nav = performance.getEntriesByType("navigation")[0];
69
+ if (nav) {
70
+ ttfbGauge.record(nav.responseStart, { vital: "ttfb" });
71
+ }
72
+ return () => {
73
+ if (interactionDurations.size > 0) {
74
+ const sorted = [...interactionDurations.values()].sort((a, b) => b - a);
75
+ const p98Index = Math.max(0, Math.ceil(sorted.length * 0.02) - 1);
76
+ inpGauge.record(sorted[p98Index], { vital: "inp" });
77
+ }
78
+ for (const c of cleanups)
79
+ c();
80
+ };
81
+ }