@goliapkg/sentori-react-native 1.2.0 → 2.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.
- package/README.md +63 -3
- package/lib/compat/sentry.d.ts +112 -0
- package/lib/compat/sentry.d.ts.map +1 -0
- package/lib/compat/sentry.js +326 -0
- package/lib/compat/sentry.js.map +1 -0
- package/lib/config.d.ts +5 -0
- package/lib/config.d.ts.map +1 -1
- package/lib/config.js.map +1 -1
- package/lib/feedback.d.ts +2 -0
- package/lib/feedback.d.ts.map +1 -0
- package/lib/feedback.js +17 -0
- package/lib/feedback.js.map +1 -0
- package/lib/init.d.ts +18 -0
- package/lib/init.d.ts.map +1 -1
- package/lib/init.js +10 -3
- package/lib/init.js.map +1 -1
- package/lib/metrics.d.ts +2 -4
- package/lib/metrics.d.ts.map +1 -1
- package/lib/metrics.js.map +1 -1
- package/lib/track.d.ts.map +1 -1
- package/lib/track.js +9 -0
- package/lib/track.js.map +1 -1
- package/package.json +19 -3
- package/src/compat/sentry.ts +482 -0
- package/src/config.ts +5 -0
- package/src/feedback.ts +21 -0
- package/src/init.ts +28 -3
- package/src/metrics.ts +3 -1
- package/src/track.ts +9 -0
package/lib/init.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"init.js","sourceRoot":"","sources":["../src/init.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,wBAAwB,CAAC;AAErD,OAAO,EAAkB,SAAS,EAAE,MAAM,UAAU,CAAC;AACrD,OAAO,EAAE,oBAAoB,EAAE,MAAM,mBAAmB,CAAC;AACzD,OAAO,EAAE,uBAAuB,EAAE,MAAM,sBAAsB,CAAC;AAC/D,OAAO,EAAE,qBAAqB,EAAE,MAAM,oBAAoB,CAAC;AAC3D,OAAO,EAAE,qBAAqB,EAAE,MAAM,oBAAoB,CAAC;AAC3D,OAAO,EAAE,aAAa,EAAE,MAAM,eAAe,CAAC;AAC9C,OAAO,EACL,mBAAmB,EACnB,mBAAmB,GACpB,MAAM,sBAAsB,CAAC;AAC9B,OAAO,EAAE,YAAY,EAAE,MAAM,cAAc,CAAC;AAC5C,OAAO,EAAE,iBAAiB,EAAE,MAAM,WAAW,CAAC;AAC9C,OAAO,EAAE,eAAe,EAAE,MAAM,SAAS,CAAC;AAC1C,OAAO,EAAE,kBAAkB,EAAE,uBAAuB,EAAE,eAAe,EAAE,MAAM,UAAU,CAAC;AACxF,OAAO,EAAE,cAAc,EAAE,MAAM,iBAAiB,CAAC;AACjD,OAAO,EAAE,cAAc,EAAyB,MAAM,aAAa,CAAC;AACpE,OAAO,EAAE,SAAS,EAAE,MAAM,wBAAwB,CAAC;AACnD,OAAO,EAAE,mBAAmB,EAAE,MAAM,mBAAmB,CAAC;AACxD,OAAO,EAAE,oBAAoB,EAAE,MAAM,qBAAqB,CAAC;AAC3D,OAAO,EAAE,qBAAqB,EAAE,MAAM,WAAW,CAAC;AAClD,OAAO,EAAE,qBAAqB,EAAwB,MAAM,sBAAsB,CAAC;AACnF,OAAO,EAAE,WAAW,EAAE,MAAM,UAAU,CAAC;AACvC,OAAO,EAAE,mBAAmB,EAAE,MAAM,mBAAmB,CAAC;AACxD,OAAO,EAAE,YAAY,EAAE,MAAM,mBAAmB,CAAC;AACjD,OAAO,EACL,iBAAiB,EACjB,OAAO,EACP,cAAc,EACd,gBAAgB,GACjB,MAAM,aAAa,CAAC;
|
|
1
|
+
{"version":3,"file":"init.js","sourceRoot":"","sources":["../src/init.ts"],"names":[],"mappings":"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"}
|
package/lib/metrics.d.ts
CHANGED
|
@@ -1,3 +1,4 @@
|
|
|
1
|
+
import type { SpanContextLike } from '@goliapkg/sentori-core';
|
|
1
2
|
/**
|
|
2
3
|
* Record a numeric metric point. Cheap to call from a render hook —
|
|
3
4
|
* pushes into a 500-slot ring drained every 30 s (or on overflow).
|
|
@@ -11,10 +12,7 @@
|
|
|
11
12
|
* span.end({ status: 'ok' })
|
|
12
13
|
*/
|
|
13
14
|
export declare function recordMetric(name: string, value: number, tags?: Record<string, string>, opts?: {
|
|
14
|
-
parent?:
|
|
15
|
-
spanId: string;
|
|
16
|
-
traceId: string;
|
|
17
|
-
};
|
|
15
|
+
parent?: SpanContextLike;
|
|
18
16
|
}): void;
|
|
19
17
|
export declare function flushMetrics(): Promise<void>;
|
|
20
18
|
/**
|
package/lib/metrics.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"metrics.d.ts","sourceRoot":"","sources":["../src/metrics.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"metrics.d.ts","sourceRoot":"","sources":["../src/metrics.ts"],"names":[],"mappings":"AAaA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,wBAAwB,CAAC;AAkB9D;;;;;;;;;;;GAWG;AACH,wBAAgB,YAAY,CAC1B,IAAI,EAAE,MAAM,EACZ,KAAK,EAAE,MAAM,EACb,IAAI,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,EAC7B,IAAI,CAAC,EAAE;IAAE,MAAM,CAAC,EAAE,eAAe,CAAA;CAAE,GAClC,IAAI,CAcN;AAED,wBAAsB,YAAY,IAAI,OAAO,CAAC,IAAI,CAAC,CAOlD;AAED;;;GAGG;AACH,wBAAgB,iBAAiB,IAAI,IAAI,CAQxC;AAED,wBAAgB,sBAAsB,IAAI,IAAI,CAM7C"}
|
package/lib/metrics.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"metrics.js","sourceRoot":"","sources":["../src/metrics.ts"],"names":[],"mappings":"AAAA,kCAAkC;AAClC,EAAE;AACF,sEAAsE;AACtE,mEAAmE;AACnE,qEAAqE;AACrE,sEAAsE;AACtE,oEAAoE;AACpE,EAAE;AACF,uEAAuE;AACvE,qEAAqE;AACrE,oEAAoE;AACpE,2BAA2B;
|
|
1
|
+
{"version":3,"file":"metrics.js","sourceRoot":"","sources":["../src/metrics.ts"],"names":[],"mappings":"AAAA,kCAAkC;AAClC,EAAE;AACF,sEAAsE;AACtE,mEAAmE;AACnE,qEAAqE;AACrE,sEAAsE;AACtE,oEAAoE;AACpE,EAAE;AACF,uEAAuE;AACvE,qEAAqE;AACrE,oEAAoE;AACpE,2BAA2B;AAI3B,OAAO,EAAE,SAAS,EAAE,aAAa,EAAE,MAAM,UAAU,CAAC;AACpD,OAAO,EAAE,gBAAgB,EAAE,MAAM,aAAa,CAAC;AAS/C,MAAM,UAAU,GAAG,GAAG,CAAC;AACvB,MAAM,iBAAiB,GAAG,MAAM,CAAC;AAEjC,IAAI,IAAI,GAAY,EAAE,CAAC;AACvB,IAAI,MAAM,GAA0C,IAAI,CAAC;AAEzD;;;;;;;;;;;GAWG;AACH,MAAM,UAAU,YAAY,CAC1B,IAAY,EACZ,KAAa,EACb,IAA6B,EAC7B,IAAmC;IAEnC,IAAI,CAAC,aAAa,EAAE;QAAE,OAAO;IAC7B,IAAI,OAAO,IAAI,KAAK,QAAQ,IAAI,IAAI,CAAC,MAAM,KAAK,CAAC,IAAI,IAAI,CAAC,MAAM,GAAG,GAAG;QAAE,OAAO;IAC/E,IAAI,OAAO,KAAK,KAAK,QAAQ,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,KAAK,CAAC;QAAE,OAAO;IACjE,IAAI,IAAI,IAAI,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,MAAM,GAAG,EAAE;QAAE,OAAO;IAClD,kEAAkE;IAClE,mEAAmE;IACnE,MAAM,SAAS,GAAuC,IAAI,EAAE,MAAM;QAChE,CAAC,CAAC,EAAE,GAAG,CAAC,IAAI,IAAI,EAAE,CAAC,EAAE,OAAO,EAAE,IAAI,CAAC,MAAM,CAAC,MAAM,EAAE,QAAQ,EAAE,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE;QACjF,CAAC,CAAC,IAAI,CAAC;IACT,IAAI,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,IAAI,EAAE,SAAS,EAAE,EAAE,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,EAAE,KAAK,EAAE,CAAC,CAAC;IAC1E,IAAI,IAAI,CAAC,MAAM,IAAI,UAAU,EAAE,CAAC;QAC9B,KAAK,YAAY,EAAE,CAAC;IACtB,CAAC;AACH,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,YAAY;IAChC,IAAI,IAAI,CAAC,MAAM,KAAK,CAAC;QAAE,OAAO;IAC9B,MAAM,MAAM,GAAG,SAAS,EAAE,CAAC;IAC3B,IAAI,CAAC,MAAM;QAAE,OAAO;IACpB,MAAM,KAAK,GAAG,IAAI,CAAC;IACnB,IAAI,GAAG,EAAE,CAAC;IACV,MAAM,gBAAgB,CAAC,MAAM,CAAC,SAAS,EAAE,MAAM,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC;AAChE,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,iBAAiB;IAC/B,IAAI,MAAM,KAAK,IAAI;QAAE,OAAO;IAC5B,MAAM,GAAG,WAAW,CAAC,GAAG,EAAE;QACxB,KAAK,YAAY,EAAE,CAAC;IACtB,CAAC,EAAE,iBAAiB,CAAC,CAAC;IACtB,mEAAmE;IACnE,4DAA4D;IAC3D,MAA4C,CAAC,KAAK,EAAE,EAAE,CAAC;AAC1D,CAAC;AAED,MAAM,UAAU,sBAAsB;IACpC,IAAI,MAAM,KAAK,IAAI,EAAE,CAAC;QACpB,aAAa,CAAC,MAAM,CAAC,CAAC;QACtB,MAAM,GAAG,IAAI,CAAC;IAChB,CAAC;IACD,IAAI,GAAG,EAAE,CAAC;AACZ,CAAC"}
|
package/lib/track.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"track.d.ts","sourceRoot":"","sources":["../src/track.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"track.d.ts","sourceRoot":"","sources":["../src/track.ts"],"names":[],"mappings":"AAwBA,MAAM,MAAM,UAAU,GAAG,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;AAEjD,MAAM,MAAM,UAAU,GAAG;IACvB,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,CAAC,EAAE,UAAU,CAAC;IACnB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,EAAE,EAAE,MAAM,CAAC;IACX,MAAM,CAAC,EAAE,MAAM,CAAC;CACjB,CAAC;AAUF;;;;;;;;;;;;GAYG;AACH,wBAAgB,KAAK,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,CAAC,EAAE,UAAU,EAAE,KAAK,CAAC,EAAE,MAAM,GAAG,IAAI,CA8B5E;AAED,wBAAsB,UAAU,IAAI,OAAO,CAAC,IAAI,CAAC,CAOhD;AAED;;;GAGG;AACH,wBAAgB,eAAe,IAAI,IAAI,CAOtC;AAED,wBAAgB,iBAAiB,IAAI,SAAS,UAAU,EAAE,CAEzD;AAED,wBAAgB,oBAAoB,IAAI,IAAI,CAM3C"}
|
package/lib/track.js
CHANGED
|
@@ -16,6 +16,7 @@
|
|
|
16
16
|
// in navigation.ts whenever react-navigation swaps the active route.
|
|
17
17
|
// Hosts that don't use react-navigation can call
|
|
18
18
|
// `sentori.track('$pageview', { route: 'Cart' })` themselves.
|
|
19
|
+
import { addInternalBreadcrumb } from './breadcrumbs';
|
|
19
20
|
import { getCurrentUserId } from './capture';
|
|
20
21
|
import { getConfig, isInitialized } from './config';
|
|
21
22
|
import { sendTrackBatch } from './transport';
|
|
@@ -58,6 +59,14 @@ export function track(name, props, route) {
|
|
|
58
59
|
userId: getCurrentUserId(),
|
|
59
60
|
};
|
|
60
61
|
_buf.push(ev);
|
|
62
|
+
// v2.0 W3 — auto-breadcrumb. When `init.capture.trackAutoBreadcrumb`
|
|
63
|
+
// is `true`, push a `{ type: 'track', data: { name, props } }`
|
|
64
|
+
// breadcrumb so the customer journey leading up to a later
|
|
65
|
+
// `captureException` / `captureMessage` is visible in the dashboard.
|
|
66
|
+
// Defaults off — see Config.trackAutoBreadcrumb docstring.
|
|
67
|
+
if (config?.trackAutoBreadcrumb === true) {
|
|
68
|
+
addInternalBreadcrumb('track', props ? { name, props } : { name });
|
|
69
|
+
}
|
|
61
70
|
if (_buf.length >= MAX_BUFFER) {
|
|
62
71
|
void flushTrack();
|
|
63
72
|
}
|
package/lib/track.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"track.js","sourceRoot":"","sources":["../src/track.ts"],"names":[],"mappings":"AAAA,2DAA2D;AAC3D,EAAE;AACF,sEAAsE;AACtE,kEAAkE;AAClE,oEAAoE;AACpE,8DAA8D;AAC9D,aAAa;AACb,EAAE;AACF,iEAAiE;AACjE,kEAAkE;AAClE,mEAAmE;AACnE,gEAAgE;AAChE,SAAS;AACT,EAAE;AACF,mEAAmE;AACnE,qEAAqE;AACrE,iDAAiD;AACjD,8DAA8D;AAE9D,OAAO,EAAE,gBAAgB,EAAE,MAAM,WAAW,CAAC;AAC7C,OAAO,EAAE,SAAS,EAAE,aAAa,EAAE,MAAM,UAAU,CAAC;AACpD,OAAO,EAAE,cAAc,EAAE,MAAM,aAAa,CAAC;AAe7C,MAAM,UAAU,GAAG,GAAG,CAAC;AACvB,MAAM,QAAQ,GAAG,GAAG,CAAC;AACrB,MAAM,cAAc,GAAG,EAAE,CAAC;AAC1B,MAAM,iBAAiB,GAAG,MAAM,CAAC;AAEjC,IAAI,IAAI,GAAiB,EAAE,CAAC;AAC5B,IAAI,MAAM,GAA0C,IAAI,CAAC;AAEzD;;;;;;;;;;;;GAYG;AACH,MAAM,UAAU,KAAK,CAAC,IAAY,EAAE,KAAkB,EAAE,KAAc;IACpE,IAAI,CAAC,aAAa,EAAE;QAAE,OAAO;IAC7B,IAAI,OAAO,IAAI,KAAK,QAAQ,IAAI,IAAI,CAAC,MAAM,KAAK,CAAC,IAAI,IAAI,CAAC,MAAM,GAAG,QAAQ,EAAE,CAAC;QAC5E,OAAO;IACT,CAAC;IACD,IAAI,KAAK,IAAI,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,MAAM,GAAG,cAAc,EAAE,CAAC;QACxD,OAAO;IACT,CAAC;IACD,MAAM,MAAM,GAAG,SAAS,EAAE,CAAC;IAC3B,MAAM,EAAE,GAAe;QACrB,IAAI;QACJ,KAAK;QACL,OAAO,EAAE,MAAM,EAAE,OAAO;QACxB,WAAW,EAAE,MAAM,EAAE,WAAW;QAChC,KAAK;QACL,EAAE,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;QAC5B,MAAM,EAAE,gBAAgB,EAAE;KAC3B,CAAC;IACF,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IACd,IAAI,IAAI,CAAC,MAAM,IAAI,UAAU,EAAE,CAAC;QAC9B,KAAK,UAAU,EAAE,CAAC;IACpB,CAAC;AACH,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,UAAU;IAC9B,IAAI,IAAI,CAAC,MAAM,KAAK,CAAC;QAAE,OAAO;IAC9B,MAAM,MAAM,GAAG,SAAS,EAAE,CAAC;IAC3B,IAAI,CAAC,MAAM;QAAE,OAAO;IACpB,MAAM,KAAK,GAAG,IAAI,CAAC;IACnB,IAAI,GAAG,EAAE,CAAC;IACV,MAAM,cAAc,CAAC,MAAM,CAAC,SAAS,EAAE,MAAM,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC;AAC9D,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,eAAe;IAC7B,IAAI,MAAM,KAAK,IAAI;QAAE,OAAO;IAC5B,MAAM,GAAG,WAAW,CAAC,GAAG,EAAE;QACxB,KAAK,UAAU,EAAE,CAAC;IACpB,CAAC,EAAE,iBAAiB,CAAC,CAAC;IACtB,sDAAsD;IACrD,MAA4C,CAAC,KAAK,EAAE,EAAE,CAAC;AAC1D,CAAC;AAED,MAAM,UAAU,iBAAiB;IAC/B,OAAO,IAAI,CAAC;AACd,CAAC;AAED,MAAM,UAAU,oBAAoB;IAClC,IAAI,MAAM,KAAK,IAAI,EAAE,CAAC;QACpB,aAAa,CAAC,MAAM,CAAC,CAAC;QACtB,MAAM,GAAG,IAAI,CAAC;IAChB,CAAC;IACD,IAAI,GAAG,EAAE,CAAC;AACZ,CAAC"}
|
|
1
|
+
{"version":3,"file":"track.js","sourceRoot":"","sources":["../src/track.ts"],"names":[],"mappings":"AAAA,2DAA2D;AAC3D,EAAE;AACF,sEAAsE;AACtE,kEAAkE;AAClE,oEAAoE;AACpE,8DAA8D;AAC9D,aAAa;AACb,EAAE;AACF,iEAAiE;AACjE,kEAAkE;AAClE,mEAAmE;AACnE,gEAAgE;AAChE,SAAS;AACT,EAAE;AACF,mEAAmE;AACnE,qEAAqE;AACrE,iDAAiD;AACjD,8DAA8D;AAE9D,OAAO,EAAE,qBAAqB,EAAE,MAAM,eAAe,CAAC;AACtD,OAAO,EAAE,gBAAgB,EAAE,MAAM,WAAW,CAAC;AAC7C,OAAO,EAAE,SAAS,EAAE,aAAa,EAAE,MAAM,UAAU,CAAC;AACpD,OAAO,EAAE,cAAc,EAAE,MAAM,aAAa,CAAC;AAe7C,MAAM,UAAU,GAAG,GAAG,CAAC;AACvB,MAAM,QAAQ,GAAG,GAAG,CAAC;AACrB,MAAM,cAAc,GAAG,EAAE,CAAC;AAC1B,MAAM,iBAAiB,GAAG,MAAM,CAAC;AAEjC,IAAI,IAAI,GAAiB,EAAE,CAAC;AAC5B,IAAI,MAAM,GAA0C,IAAI,CAAC;AAEzD;;;;;;;;;;;;GAYG;AACH,MAAM,UAAU,KAAK,CAAC,IAAY,EAAE,KAAkB,EAAE,KAAc;IACpE,IAAI,CAAC,aAAa,EAAE;QAAE,OAAO;IAC7B,IAAI,OAAO,IAAI,KAAK,QAAQ,IAAI,IAAI,CAAC,MAAM,KAAK,CAAC,IAAI,IAAI,CAAC,MAAM,GAAG,QAAQ,EAAE,CAAC;QAC5E,OAAO;IACT,CAAC;IACD,IAAI,KAAK,IAAI,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,MAAM,GAAG,cAAc,EAAE,CAAC;QACxD,OAAO;IACT,CAAC;IACD,MAAM,MAAM,GAAG,SAAS,EAAE,CAAC;IAC3B,MAAM,EAAE,GAAe;QACrB,IAAI;QACJ,KAAK;QACL,OAAO,EAAE,MAAM,EAAE,OAAO;QACxB,WAAW,EAAE,MAAM,EAAE,WAAW;QAChC,KAAK;QACL,EAAE,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;QAC5B,MAAM,EAAE,gBAAgB,EAAE;KAC3B,CAAC;IACF,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IACd,qEAAqE;IACrE,+DAA+D;IAC/D,2DAA2D;IAC3D,qEAAqE;IACrE,2DAA2D;IAC3D,IAAI,MAAM,EAAE,mBAAmB,KAAK,IAAI,EAAE,CAAC;QACzC,qBAAqB,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC,CAAC,EAAE,IAAI,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC;IACrE,CAAC;IACD,IAAI,IAAI,CAAC,MAAM,IAAI,UAAU,EAAE,CAAC;QAC9B,KAAK,UAAU,EAAE,CAAC;IACpB,CAAC;AACH,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,UAAU;IAC9B,IAAI,IAAI,CAAC,MAAM,KAAK,CAAC;QAAE,OAAO;IAC9B,MAAM,MAAM,GAAG,SAAS,EAAE,CAAC;IAC3B,IAAI,CAAC,MAAM;QAAE,OAAO;IACpB,MAAM,KAAK,GAAG,IAAI,CAAC;IACnB,IAAI,GAAG,EAAE,CAAC;IACV,MAAM,cAAc,CAAC,MAAM,CAAC,SAAS,EAAE,MAAM,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC;AAC9D,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,eAAe;IAC7B,IAAI,MAAM,KAAK,IAAI;QAAE,OAAO;IAC5B,MAAM,GAAG,WAAW,CAAC,GAAG,EAAE;QACxB,KAAK,UAAU,EAAE,CAAC;IACpB,CAAC,EAAE,iBAAiB,CAAC,CAAC;IACtB,sDAAsD;IACrD,MAA4C,CAAC,KAAK,EAAE,EAAE,CAAC;AAC1D,CAAC;AAED,MAAM,UAAU,iBAAiB;IAC/B,OAAO,IAAI,CAAC;AACd,CAAC;AAED,MAAM,UAAU,oBAAoB;IAClC,IAAI,MAAM,KAAK,IAAI,EAAE,CAAC;QACpB,aAAa,CAAC,MAAM,CAAC,CAAC;QACtB,MAAM,GAAG,IAAI,CAAC;IAChB,CAAC;IACD,IAAI,GAAG,EAAE,CAAC;AACZ,CAAC"}
|
package/package.json
CHANGED
|
@@ -1,8 +1,9 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@goliapkg/sentori-react-native",
|
|
3
|
-
"version": "
|
|
3
|
+
"version": "2.0.0",
|
|
4
4
|
"description": "Sentori SDK for React Native — JS-layer error capture, native crash handlers (iOS / Android), batched transport, fetch + react-navigation tracing.",
|
|
5
|
-
"license": "MIT",
|
|
5
|
+
"license": "Apache-2.0 OR MIT",
|
|
6
|
+
"author": "GOLIA K.K. <takagi@golia.jp> (https://golia.jp)",
|
|
6
7
|
"homepage": "https://sentori.golia.jp",
|
|
7
8
|
"repository": {
|
|
8
9
|
"type": "git",
|
|
@@ -22,6 +23,21 @@
|
|
|
22
23
|
],
|
|
23
24
|
"main": "lib/index.js",
|
|
24
25
|
"types": "lib/index.d.ts",
|
|
26
|
+
"exports": {
|
|
27
|
+
".": {
|
|
28
|
+
"types": "./lib/index.d.ts",
|
|
29
|
+
"default": "./lib/index.js"
|
|
30
|
+
},
|
|
31
|
+
"./compat": {
|
|
32
|
+
"types": "./lib/compat/sentry.d.ts",
|
|
33
|
+
"default": "./lib/compat/sentry.js"
|
|
34
|
+
},
|
|
35
|
+
"./feedback": {
|
|
36
|
+
"types": "./lib/feedback.d.ts",
|
|
37
|
+
"default": "./lib/feedback.js"
|
|
38
|
+
},
|
|
39
|
+
"./package.json": "./package.json"
|
|
40
|
+
},
|
|
25
41
|
"bin": {
|
|
26
42
|
"sentori-rn-upload-source-bundle": "bin/sentori-rn-upload-source-bundle.cjs"
|
|
27
43
|
},
|
|
@@ -76,6 +92,6 @@
|
|
|
76
92
|
"access": "public"
|
|
77
93
|
},
|
|
78
94
|
"dependencies": {
|
|
79
|
-
"@goliapkg/sentori-core": "^0.
|
|
95
|
+
"@goliapkg/sentori-core": "^1.0.0"
|
|
80
96
|
}
|
|
81
97
|
}
|
|
@@ -0,0 +1,482 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* v2.3 W6.3 — Sentry-compatible API surface.
|
|
3
|
+
*
|
|
4
|
+
* Drop-in for code (or LLM-generated code) written against
|
|
5
|
+
* `@sentry/react-native`. Every Sentry call maps to exactly one
|
|
6
|
+
* Sentori-native call internally. Translation differences (e.g.
|
|
7
|
+
* `Sentry.setUser({ ip_address })` — Sentori never stores IP) fire
|
|
8
|
+
* a one-shot console hint at `info` level, deduplicated per
|
|
9
|
+
* (api, dropped_field).
|
|
10
|
+
*
|
|
11
|
+
* Why this exists: LLMs have seen a LOT of Sentry code; letting
|
|
12
|
+
* them write the same syntax against Sentori is one less thing
|
|
13
|
+
* Sentori asks the host to think about. Combined with the v2.3
|
|
14
|
+
* "free bonus" stance — host adds Sentori without unlearning
|
|
15
|
+
* anything.
|
|
16
|
+
*
|
|
17
|
+
* Usage:
|
|
18
|
+
*
|
|
19
|
+
* import * as Sentry from '@goliapkg/sentori-react-native/compat'
|
|
20
|
+
*
|
|
21
|
+
* Sentry.init({ dsn: 'https://<token>@<host>/<projectId>', ... })
|
|
22
|
+
* Sentry.captureException(err)
|
|
23
|
+
* Sentry.setUser({ id, email }) // email → linkBy.email (hashed)
|
|
24
|
+
*
|
|
25
|
+
* The compat layer holds NO state of its own — it's a thin shim
|
|
26
|
+
* over the same Sentori internals. Mixing `Sentry.*` and
|
|
27
|
+
* `sentori.*` calls in the same app works fine.
|
|
28
|
+
*
|
|
29
|
+
* See `docs/design/sdk-v2.3-redesign.md` §4 for the full
|
|
30
|
+
* translation table + design rationale.
|
|
31
|
+
*/
|
|
32
|
+
|
|
33
|
+
import { logger } from '@goliapkg/sentori-core'
|
|
34
|
+
|
|
35
|
+
import { addBreadcrumb as nativeAddBreadcrumb } from '../breadcrumbs'
|
|
36
|
+
import {
|
|
37
|
+
captureException as nativeCaptureException,
|
|
38
|
+
captureMessage as nativeCaptureMessage,
|
|
39
|
+
setTag as nativeSetTag,
|
|
40
|
+
setTags as nativeSetTags,
|
|
41
|
+
setUser as nativeSetUser,
|
|
42
|
+
} from '../capture'
|
|
43
|
+
import { type InitOptions, init as nativeInit } from '../init'
|
|
44
|
+
|
|
45
|
+
// ── one-shot warn dedup ───────────────────────────────────────────────────
|
|
46
|
+
|
|
47
|
+
const _warnedOnce = new Set<string>()
|
|
48
|
+
function warnOnce(key: string, msg: string): void {
|
|
49
|
+
if (_warnedOnce.has(key)) return
|
|
50
|
+
_warnedOnce.add(key)
|
|
51
|
+
logger.info('compat', msg)
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
// ── DSN parsing ───────────────────────────────────────────────────────────
|
|
55
|
+
|
|
56
|
+
type SentryInitOpts = {
|
|
57
|
+
dsn: string
|
|
58
|
+
environment?: string
|
|
59
|
+
release?: string
|
|
60
|
+
tracesSampleRate?: number
|
|
61
|
+
sampleRate?: number
|
|
62
|
+
attachStacktrace?: boolean
|
|
63
|
+
autoSessionTracking?: boolean
|
|
64
|
+
/** Sentry's `debug: true` ≈ Sentori's `logLevel: 'debug'`. */
|
|
65
|
+
debug?: boolean
|
|
66
|
+
/** Catch-all for fields Sentori either ignores or doesn't map yet. */
|
|
67
|
+
[other: string]: unknown
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
function parseDsn(dsn: string): { token: string; ingestUrl: string } {
|
|
71
|
+
// Sentry DSN shape: `https://<key>@<host>[:port][/<projectId>]`
|
|
72
|
+
// Sentori cares about `<key>` (must be `st_pk_…`) and `<host>`.
|
|
73
|
+
let url: URL
|
|
74
|
+
try {
|
|
75
|
+
url = new URL(dsn)
|
|
76
|
+
} catch {
|
|
77
|
+
throw new Error(`Sentory compat: dsn is not a valid URL: ${dsn}`)
|
|
78
|
+
}
|
|
79
|
+
const key = url.username
|
|
80
|
+
if (!key) {
|
|
81
|
+
throw new Error(`Sentory compat: dsn missing token in user-info component`)
|
|
82
|
+
}
|
|
83
|
+
if (!key.startsWith('st_pk_')) {
|
|
84
|
+
throw new Error(
|
|
85
|
+
`Sentory compat: dsn token must start with 'st_pk_' (got prefix '${key.slice(0, 8)}…'). ` +
|
|
86
|
+
`Sentori does not parse Sentry-issued tokens — generate a Sentori project token via the dashboard.`,
|
|
87
|
+
)
|
|
88
|
+
}
|
|
89
|
+
// strip user-info to reconstruct the ingest origin
|
|
90
|
+
const ingestUrl = `${url.protocol}//${url.host}`
|
|
91
|
+
return { token: key, ingestUrl }
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
// ── Sentry.init ───────────────────────────────────────────────────────────
|
|
95
|
+
|
|
96
|
+
export function init(opts: SentryInitOpts): void {
|
|
97
|
+
const { token, ingestUrl } = parseDsn(opts.dsn)
|
|
98
|
+
|
|
99
|
+
const sentoriOpts: InitOptions = {
|
|
100
|
+
token,
|
|
101
|
+
release: opts.release ?? '',
|
|
102
|
+
ingestUrl,
|
|
103
|
+
...(opts.environment ? { environment: opts.environment } : {}),
|
|
104
|
+
sample: {
|
|
105
|
+
...(opts.tracesSampleRate !== undefined ? { traces: opts.tracesSampleRate } : {}),
|
|
106
|
+
...(opts.sampleRate !== undefined ? { errors: opts.sampleRate } : {}),
|
|
107
|
+
},
|
|
108
|
+
...(opts.debug ? { logLevel: 'debug' as const } : {}),
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
// Empty release is the most common Sentry-init mistake; warn but
|
|
112
|
+
// continue.
|
|
113
|
+
if (!sentoriOpts.release) {
|
|
114
|
+
warnOnce(
|
|
115
|
+
'init:no-release',
|
|
116
|
+
'Sentry.init() with no `release` — Sentori requires release for grouping + drop-down menus to make sense. Set `release: "myapp@1.2.3"` for production cuts.',
|
|
117
|
+
)
|
|
118
|
+
// Sentori's init throws when release is empty; provide a
|
|
119
|
+
// reasonable fallback so the rest of the code path works in dev.
|
|
120
|
+
sentoriOpts.release = `unspecified@${Date.now()}`
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
// Pass-through informational hints for ignored fields.
|
|
124
|
+
for (const ignored of [
|
|
125
|
+
'attachStacktrace',
|
|
126
|
+
'autoSessionTracking',
|
|
127
|
+
'integrations',
|
|
128
|
+
'beforeSend',
|
|
129
|
+
'beforeBreadcrumb',
|
|
130
|
+
'maxBreadcrumbs',
|
|
131
|
+
]) {
|
|
132
|
+
if (ignored in opts) {
|
|
133
|
+
warnOnce(
|
|
134
|
+
`init:ignored:${ignored}`,
|
|
135
|
+
`Sentry.init({ ${ignored} }) ignored. ` +
|
|
136
|
+
`${getIgnoredHint(ignored)}`,
|
|
137
|
+
)
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
nativeInit(sentoriOpts)
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
function getIgnoredHint(field: string): string {
|
|
145
|
+
switch (field) {
|
|
146
|
+
case 'attachStacktrace':
|
|
147
|
+
return 'Sentori always sends stack traces — no toggle.'
|
|
148
|
+
case 'autoSessionTracking':
|
|
149
|
+
return "Sentori sessions are on by default; toggle via `init({ capture: { sessions: true|false } })`."
|
|
150
|
+
case 'integrations':
|
|
151
|
+
return 'Sentori uses `init({ capture: {...} })` toggles instead of Integration classes — see the docs.'
|
|
152
|
+
case 'beforeSend':
|
|
153
|
+
case 'beforeBreadcrumb':
|
|
154
|
+
return 'Sentori does not support an arbitrary beforeSend hook today. Server-side PII scrubbing is automatic.'
|
|
155
|
+
case 'maxBreadcrumbs':
|
|
156
|
+
return 'Sentori uses a fixed 100-slot ring buffer.'
|
|
157
|
+
default:
|
|
158
|
+
return ''
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
// ── Severity / level enum ─────────────────────────────────────────────────
|
|
163
|
+
|
|
164
|
+
/** Sentry's severity values, surfaced as strings (Sentori only
|
|
165
|
+
* uses 5 levels). `Log` and `Critical` collapse to `'info'` and
|
|
166
|
+
* `'fatal'` respectively. */
|
|
167
|
+
export const Severity = {
|
|
168
|
+
Fatal: 'fatal' as const,
|
|
169
|
+
Critical: 'fatal' as const,
|
|
170
|
+
Error: 'error' as const,
|
|
171
|
+
Warning: 'warning' as const,
|
|
172
|
+
Log: 'info' as const,
|
|
173
|
+
Info: 'info' as const,
|
|
174
|
+
Debug: 'debug' as const,
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
type SentryLevelString =
|
|
178
|
+
| 'critical'
|
|
179
|
+
| 'debug'
|
|
180
|
+
| 'error'
|
|
181
|
+
| 'fatal'
|
|
182
|
+
| 'info'
|
|
183
|
+
| 'log'
|
|
184
|
+
| 'warning'
|
|
185
|
+
|
|
186
|
+
type SentoriLevel = 'debug' | 'error' | 'fatal' | 'info' | 'warning'
|
|
187
|
+
|
|
188
|
+
function mapLevel(level: SentryLevelString | undefined): SentoriLevel | undefined {
|
|
189
|
+
if (!level) return undefined
|
|
190
|
+
switch (level) {
|
|
191
|
+
case 'critical':
|
|
192
|
+
warnOnce(
|
|
193
|
+
'severity:critical',
|
|
194
|
+
"Sentry.Severity.Critical → mapped to 'fatal' (Sentori's 5-level syslog-style scale).",
|
|
195
|
+
)
|
|
196
|
+
return 'fatal'
|
|
197
|
+
case 'log':
|
|
198
|
+
warnOnce(
|
|
199
|
+
'severity:log',
|
|
200
|
+
"Sentry.Severity.Log → mapped to 'info' (Sentori's 5-level scale has no separate Log).",
|
|
201
|
+
)
|
|
202
|
+
return 'info'
|
|
203
|
+
default:
|
|
204
|
+
return level as SentoriLevel
|
|
205
|
+
}
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
// ── captureException ──────────────────────────────────────────────────────
|
|
209
|
+
|
|
210
|
+
type SentryCaptureContext = {
|
|
211
|
+
tags?: Record<string, string>
|
|
212
|
+
extra?: Record<string, unknown>
|
|
213
|
+
level?: SentryLevelString
|
|
214
|
+
fingerprint?: string[]
|
|
215
|
+
user?: SentrySetUserInput
|
|
216
|
+
}
|
|
217
|
+
|
|
218
|
+
export function captureException(
|
|
219
|
+
err: unknown,
|
|
220
|
+
hint?: { captureContext?: SentryCaptureContext } | SentryCaptureContext,
|
|
221
|
+
): void {
|
|
222
|
+
// Sentry v8+ takes the context inline (Hint); earlier versions
|
|
223
|
+
// wrapped it in `{ captureContext: {...} }`. Accept both.
|
|
224
|
+
const ctx: SentryCaptureContext | undefined = (() => {
|
|
225
|
+
if (!hint) return undefined
|
|
226
|
+
if ('captureContext' in (hint as { captureContext?: unknown })) {
|
|
227
|
+
return (hint as { captureContext?: SentryCaptureContext }).captureContext
|
|
228
|
+
}
|
|
229
|
+
return hint as SentryCaptureContext
|
|
230
|
+
})()
|
|
231
|
+
|
|
232
|
+
if (ctx?.extra) {
|
|
233
|
+
warnOnce(
|
|
234
|
+
'captureException:extra',
|
|
235
|
+
'Sentry.captureException(err, { extra }) → `extra` mapped to `tags` (Sentori does not have a separate extra namespace).',
|
|
236
|
+
)
|
|
237
|
+
}
|
|
238
|
+
if (ctx?.user) {
|
|
239
|
+
// Apply the per-call user via setUser (Sentori takes the
|
|
240
|
+
// current scope user automatically).
|
|
241
|
+
setUser(ctx.user)
|
|
242
|
+
}
|
|
243
|
+
|
|
244
|
+
const mergedTags = {
|
|
245
|
+
...(ctx?.tags ?? {}),
|
|
246
|
+
...(ctx?.extra
|
|
247
|
+
? Object.fromEntries(
|
|
248
|
+
Object.entries(ctx.extra).map(([k, v]) => [k, String(v)]),
|
|
249
|
+
)
|
|
250
|
+
: {}),
|
|
251
|
+
}
|
|
252
|
+
|
|
253
|
+
nativeCaptureException(err as Error, {
|
|
254
|
+
...(Object.keys(mergedTags).length > 0 ? { tags: mergedTags } : {}),
|
|
255
|
+
...(ctx?.fingerprint ? { fingerprint: ctx.fingerprint } : {}),
|
|
256
|
+
})
|
|
257
|
+
}
|
|
258
|
+
|
|
259
|
+
// ── captureMessage ────────────────────────────────────────────────────────
|
|
260
|
+
|
|
261
|
+
export function captureMessage(
|
|
262
|
+
msg: string,
|
|
263
|
+
levelOrCtx?: SentryCaptureContext | SentryLevelString,
|
|
264
|
+
): void {
|
|
265
|
+
let level: SentoriLevel | undefined
|
|
266
|
+
let tags: Record<string, string> | undefined
|
|
267
|
+
if (typeof levelOrCtx === 'string') {
|
|
268
|
+
level = mapLevel(levelOrCtx)
|
|
269
|
+
} else if (levelOrCtx) {
|
|
270
|
+
level = mapLevel(levelOrCtx.level)
|
|
271
|
+
tags = levelOrCtx.tags
|
|
272
|
+
}
|
|
273
|
+
nativeCaptureMessage(msg, {
|
|
274
|
+
...(level ? { level } : {}),
|
|
275
|
+
...(tags ? { tags } : {}),
|
|
276
|
+
})
|
|
277
|
+
}
|
|
278
|
+
|
|
279
|
+
// ── setUser ───────────────────────────────────────────────────────────────
|
|
280
|
+
|
|
281
|
+
type SentrySetUserInput = {
|
|
282
|
+
id?: string
|
|
283
|
+
email?: string
|
|
284
|
+
username?: string
|
|
285
|
+
ip_address?: string
|
|
286
|
+
segment?: string
|
|
287
|
+
[other: string]: unknown
|
|
288
|
+
} | null
|
|
289
|
+
|
|
290
|
+
export function setUser(user: SentrySetUserInput): void {
|
|
291
|
+
if (user == null) {
|
|
292
|
+
nativeSetUser(null)
|
|
293
|
+
return
|
|
294
|
+
}
|
|
295
|
+
const { id, email, username, ip_address, segment, ...rest } = user
|
|
296
|
+
|
|
297
|
+
if (ip_address !== undefined) {
|
|
298
|
+
warnOnce(
|
|
299
|
+
'setUser:ip_address',
|
|
300
|
+
'Sentry.setUser({ ip_address }) → dropped. Sentori never stores IP (privacy by design).',
|
|
301
|
+
)
|
|
302
|
+
}
|
|
303
|
+
if (segment !== undefined) {
|
|
304
|
+
warnOnce(
|
|
305
|
+
'setUser:segment',
|
|
306
|
+
'Sentry.setUser({ segment }) → mapped to tag `user.segment`. Set via setTag for clarity.',
|
|
307
|
+
)
|
|
308
|
+
if (typeof segment === 'string') nativeSetTag('user.segment', segment)
|
|
309
|
+
}
|
|
310
|
+
|
|
311
|
+
const linkBy: Record<string, string> = {}
|
|
312
|
+
if (email) linkBy.email = email
|
|
313
|
+
if (username) linkBy.username = username
|
|
314
|
+
|
|
315
|
+
// Surface any other fields the host bolted on (Sentry historically
|
|
316
|
+
// accepted arbitrary keys) — they pass through as tags.
|
|
317
|
+
for (const [k, v] of Object.entries(rest)) {
|
|
318
|
+
if (v !== undefined && v !== null) nativeSetTag(`user.${k}`, String(v))
|
|
319
|
+
}
|
|
320
|
+
|
|
321
|
+
nativeSetUser({
|
|
322
|
+
...(id ? { id } : {}),
|
|
323
|
+
...(Object.keys(linkBy).length > 0 ? { linkBy } : {}),
|
|
324
|
+
})
|
|
325
|
+
}
|
|
326
|
+
|
|
327
|
+
// ── setTag / setTags ──────────────────────────────────────────────────────
|
|
328
|
+
|
|
329
|
+
// Re-export Sentori-native semantics; identical signatures.
|
|
330
|
+
export const setTag = nativeSetTag
|
|
331
|
+
export const setTags = nativeSetTags
|
|
332
|
+
|
|
333
|
+
// ── addBreadcrumb ─────────────────────────────────────────────────────────
|
|
334
|
+
|
|
335
|
+
type SentryBreadcrumb = {
|
|
336
|
+
category?: string
|
|
337
|
+
message?: string
|
|
338
|
+
level?: SentryLevelString
|
|
339
|
+
type?: 'default' | 'error' | 'http' | 'info' | 'navigation' | 'query' | 'user' | string
|
|
340
|
+
data?: Record<string, unknown>
|
|
341
|
+
timestamp?: number
|
|
342
|
+
}
|
|
343
|
+
|
|
344
|
+
type SentoriBreadcrumbType = 'custom' | 'log' | 'nav' | 'net' | 'user'
|
|
345
|
+
|
|
346
|
+
function mapCategoryToType(category: string | undefined): SentoriBreadcrumbType | undefined {
|
|
347
|
+
if (!category) return undefined
|
|
348
|
+
if (['auth', 'click', 'gesture', 'input', 'touch', 'ui'].includes(category)) return 'user'
|
|
349
|
+
if (['fetch', 'http', 'xhr'].includes(category)) return 'net'
|
|
350
|
+
if (['nav', 'navigation', 'route'].includes(category)) return 'nav'
|
|
351
|
+
if (['console', 'log', 'sentry'].includes(category)) return 'log'
|
|
352
|
+
return 'custom'
|
|
353
|
+
}
|
|
354
|
+
|
|
355
|
+
function mapSentryType(t: string | undefined): SentoriBreadcrumbType | undefined {
|
|
356
|
+
if (!t) return undefined
|
|
357
|
+
switch (t) {
|
|
358
|
+
case 'http':
|
|
359
|
+
return 'net'
|
|
360
|
+
case 'navigation':
|
|
361
|
+
return 'nav'
|
|
362
|
+
case 'user':
|
|
363
|
+
case 'log':
|
|
364
|
+
case 'custom':
|
|
365
|
+
return t
|
|
366
|
+
default:
|
|
367
|
+
return 'custom'
|
|
368
|
+
}
|
|
369
|
+
}
|
|
370
|
+
|
|
371
|
+
export function addBreadcrumb(crumb: SentryBreadcrumb): void {
|
|
372
|
+
if (!crumb.message) {
|
|
373
|
+
crumb.message = crumb.category ?? crumb.type ?? '(no message)'
|
|
374
|
+
}
|
|
375
|
+
if (crumb.category && !crumb.type) {
|
|
376
|
+
warnOnce(
|
|
377
|
+
'breadcrumb:category',
|
|
378
|
+
'Sentry.addBreadcrumb({ category }) → mapped to `type` via well-known table; the category string itself is preserved under `data.category`.',
|
|
379
|
+
)
|
|
380
|
+
}
|
|
381
|
+
// RN SDK shape: { type, data }. No top-level `message` —
|
|
382
|
+
// Sentry's message goes into data.message.
|
|
383
|
+
nativeAddBreadcrumb({
|
|
384
|
+
type: mapSentryType(crumb.type) ?? mapCategoryToType(crumb.category) ?? 'custom',
|
|
385
|
+
data: {
|
|
386
|
+
message: crumb.message,
|
|
387
|
+
...(crumb.data ?? {}),
|
|
388
|
+
...(crumb.category ? { category: crumb.category } : {}),
|
|
389
|
+
...(crumb.level ? { level: mapLevel(crumb.level) } : {}),
|
|
390
|
+
},
|
|
391
|
+
})
|
|
392
|
+
}
|
|
393
|
+
|
|
394
|
+
// ── flush / close ─────────────────────────────────────────────────────────
|
|
395
|
+
|
|
396
|
+
// Re-export Sentori native flush + close — same signatures.
|
|
397
|
+
export { close, flush } from '../lifecycle'
|
|
398
|
+
|
|
399
|
+
// ── startTransaction / startSpan / withScope ──────────────────────────────
|
|
400
|
+
|
|
401
|
+
// Trace mapping is non-trivial (Sentry's transaction object exposes
|
|
402
|
+
// startChild, etc.). v2.3 ships a minimum-viable Sentry trace
|
|
403
|
+
// surface that supports startTransaction returning a Sentori Span
|
|
404
|
+
// object with a partial Sentry-style API (.startChild, .finish).
|
|
405
|
+
// Anything beyond that throws a clear error directing to the native
|
|
406
|
+
// `sentori.startSpan` / `sentori.withScopedSpan`.
|
|
407
|
+
|
|
408
|
+
import { startSpan } from '@goliapkg/sentori-core'
|
|
409
|
+
|
|
410
|
+
type SentrySpanOpts = {
|
|
411
|
+
op?: string
|
|
412
|
+
name?: string
|
|
413
|
+
description?: string
|
|
414
|
+
tags?: Record<string, string>
|
|
415
|
+
}
|
|
416
|
+
|
|
417
|
+
export function startTransaction(opts: SentrySpanOpts): {
|
|
418
|
+
finish: (status?: 'cancelled' | 'error' | 'ok') => void
|
|
419
|
+
setStatus: (status: 'cancelled' | 'error' | 'ok') => void
|
|
420
|
+
setTag: (k: string, v: string) => void
|
|
421
|
+
startChild: (childOpts: SentrySpanOpts) => unknown
|
|
422
|
+
} {
|
|
423
|
+
warnOnce(
|
|
424
|
+
'startTransaction',
|
|
425
|
+
'Sentry.startTransaction() → mapped to sentori.startSpan() with op as name. Native equivalent: sentori.startTrace(name) or sentori.startSpan({ name }).',
|
|
426
|
+
)
|
|
427
|
+
const name = opts.name ?? opts.op ?? 'transaction'
|
|
428
|
+
const span = startSpan(name, {
|
|
429
|
+
parent: null,
|
|
430
|
+
tags: opts.tags,
|
|
431
|
+
})
|
|
432
|
+
return {
|
|
433
|
+
finish: (status) => span.finish({ status: status === 'ok' ? 'ok' : 'error' }),
|
|
434
|
+
setStatus: (status) => { span.setTag('status', status) },
|
|
435
|
+
setTag: (k, v) => { span.setTag(k, v) },
|
|
436
|
+
startChild: (childOpts) => {
|
|
437
|
+
return startSpan(childOpts.name ?? childOpts.op ?? 'child', {
|
|
438
|
+
tags: childOpts.tags,
|
|
439
|
+
})
|
|
440
|
+
},
|
|
441
|
+
}
|
|
442
|
+
}
|
|
443
|
+
|
|
444
|
+
// ── withScope / configureScope (no-op scoping; same state as native) ─────
|
|
445
|
+
|
|
446
|
+
type ScopeProxy = {
|
|
447
|
+
setTag: (k: string, v: string) => void
|
|
448
|
+
setTags: (rec: Record<string, string>) => void
|
|
449
|
+
setUser: (u: SentrySetUserInput) => void
|
|
450
|
+
setExtra: (k: string, v: unknown) => void
|
|
451
|
+
setLevel: (l: SentryLevelString) => void
|
|
452
|
+
}
|
|
453
|
+
|
|
454
|
+
function scopeProxy(): ScopeProxy {
|
|
455
|
+
return {
|
|
456
|
+
setTag: nativeSetTag,
|
|
457
|
+
setTags: nativeSetTags,
|
|
458
|
+
setUser,
|
|
459
|
+
setExtra: (k, v) => nativeSetTag(`extra.${k}`, String(v)),
|
|
460
|
+
setLevel: () => {
|
|
461
|
+
warnOnce(
|
|
462
|
+
'scope:setLevel',
|
|
463
|
+
'Sentry.withScope(s => s.setLevel(…)) → not supported. Sentori levels travel on capture call, not on scope.',
|
|
464
|
+
)
|
|
465
|
+
},
|
|
466
|
+
}
|
|
467
|
+
}
|
|
468
|
+
|
|
469
|
+
export function withScope<T>(fn: (scope: ScopeProxy) => T): T {
|
|
470
|
+
// Sentori has no Hub; tags set inside `fn` persist (best-effort
|
|
471
|
+
// approximation of Sentry semantics). For most callers this is
|
|
472
|
+
// fine; tighter isolation needs an actual Hub which we don't ship.
|
|
473
|
+
warnOnce(
|
|
474
|
+
'withScope',
|
|
475
|
+
'Sentry.withScope() → tag mutations are NOT auto-reverted on scope exit. Use sentori.setTag/clearTags explicitly for tight isolation.',
|
|
476
|
+
)
|
|
477
|
+
return fn(scopeProxy())
|
|
478
|
+
}
|
|
479
|
+
|
|
480
|
+
export function configureScope(fn: (scope: ScopeProxy) => void): void {
|
|
481
|
+
fn(scopeProxy())
|
|
482
|
+
}
|
package/src/config.ts
CHANGED
|
@@ -36,6 +36,11 @@ export type Config = {
|
|
|
36
36
|
* session-trail buffer and uploads it as a `sessionTrail`
|
|
37
37
|
* attachment. Defaults to false. */
|
|
38
38
|
sessionTrailEnabled: boolean;
|
|
39
|
+
/** v2.0 W3 — when true, every `track(name, props)` call also
|
|
40
|
+
* pushes a `{ type: 'track', data: { name, props } }` breadcrumb
|
|
41
|
+
* so a subsequent capture carries the customer journey. Defaults
|
|
42
|
+
* to false to preserve v1 customer breadcrumb shape on upgrade. */
|
|
43
|
+
trackAutoBreadcrumb: boolean;
|
|
39
44
|
/** v2.3 — Sentori console output gate.
|
|
40
45
|
*
|
|
41
46
|
* Default `warn`: SDK is silent on host's console unless
|
package/src/feedback.ts
ADDED
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
// v2.0 W3 — `feedback` subpath. The feedback widget pulls in a
|
|
2
|
+
// component tree (react + tsx) and an internal viewer route, so
|
|
3
|
+
// importing it eagerly from the SDK's top-level barrel pays a
|
|
4
|
+
// bundle cost every consumer pays — even ones that never render
|
|
5
|
+
// the widget.
|
|
6
|
+
//
|
|
7
|
+
// Subpath import `@goliapkg/sentori-react-native/feedback`
|
|
8
|
+
// resolves to just the widget surface. Hosts that don't render
|
|
9
|
+
// the feedback button pay zero bundle delta; hosts that do reach
|
|
10
|
+
// for it import via this module and get exactly what they need.
|
|
11
|
+
//
|
|
12
|
+
// Top-level (`index.ts`) still re-exports `FeedbackButton` for one
|
|
13
|
+
// more release cycle so v1 callsites don't break on upgrade. v3
|
|
14
|
+
// will drop the top-level re-export — see
|
|
15
|
+
// `docs/recipes/v1-to-v2-migration.md` for the deprecation timeline.
|
|
16
|
+
|
|
17
|
+
export {
|
|
18
|
+
FeedbackButton,
|
|
19
|
+
type FeedbackButtonHandle,
|
|
20
|
+
type FeedbackButtonProps,
|
|
21
|
+
} from './feedback-widget'
|