@goliapkg/sentori-react-native 0.5.7 → 0.6.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/lib/capture.d.ts +6 -0
- package/lib/capture.d.ts.map +1 -1
- package/lib/capture.js +65 -9
- package/lib/capture.js.map +1 -1
- package/lib/config.d.ts +2 -0
- package/lib/config.d.ts.map +1 -1
- package/lib/config.js.map +1 -1
- package/lib/handlers/screenshot.d.ts +12 -0
- package/lib/handlers/screenshot.d.ts.map +1 -0
- package/lib/handlers/screenshot.js +85 -0
- package/lib/handlers/screenshot.js.map +1 -0
- package/lib/index.d.ts +5 -0
- package/lib/index.d.ts.map +1 -1
- package/lib/index.js +5 -0
- package/lib/index.js.map +1 -1
- package/lib/init.d.ts +5 -0
- package/lib/init.d.ts.map +1 -1
- package/lib/init.js +1 -0
- package/lib/init.js.map +1 -1
- package/lib/mask.d.ts +30 -0
- package/lib/mask.d.ts.map +1 -0
- package/lib/mask.js +77 -0
- package/lib/mask.js.map +1 -0
- package/lib/transport.d.ts +22 -0
- package/lib/transport.d.ts.map +1 -1
- package/lib/transport.js +62 -0
- package/lib/transport.js.map +1 -1
- package/lib/types.d.ts +1 -1
- package/lib/types.d.ts.map +1 -1
- package/package.json +9 -4
- package/src/__tests__/screenshot.test.ts +88 -0
- package/src/capture.ts +79 -9
- package/src/config.ts +2 -0
- package/src/handlers/screenshot.ts +115 -0
- package/src/index.ts +5 -0
- package/src/init.ts +6 -0
- package/src/mask.tsx +95 -0
- package/src/transport.ts +77 -0
- package/src/types.ts +3 -0
package/lib/capture.d.ts
CHANGED
|
@@ -1,4 +1,5 @@
|
|
|
1
1
|
import type { Tags, User } from './types';
|
|
2
|
+
export declare const __resetScreenshotBudgetForTests: () => void;
|
|
2
3
|
/**
|
|
3
4
|
* Attach a stable user identifier to events captured after this call.
|
|
4
5
|
*
|
|
@@ -17,6 +18,11 @@ export type CaptureExtras = {
|
|
|
17
18
|
tags?: Tags;
|
|
18
19
|
user?: User;
|
|
19
20
|
fingerprint?: string[];
|
|
21
|
+
/** Phase 42 sub-D.07: per-call screenshot override. `false` skips
|
|
22
|
+
* screenshot capture even when `init({ capture: { screenshot:
|
|
23
|
+
* true } })` is on — handy for sensitive screens. Defaults to
|
|
24
|
+
* whatever `config.screenshotsEnabled` says. */
|
|
25
|
+
screenshot?: boolean;
|
|
20
26
|
};
|
|
21
27
|
export declare const captureError: (error: Error, extras?: CaptureExtras) => void;
|
|
22
28
|
export declare const captureException: (error: Error, extras?: CaptureExtras) => void;
|
package/lib/capture.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"capture.d.ts","sourceRoot":"","sources":["../src/capture.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"capture.d.ts","sourceRoot":"","sources":["../src/capture.ts"],"names":[],"mappings":"AAQA,OAAO,KAAK,EAAoD,IAAI,EAAE,IAAI,EAAE,MAAM,SAAS,CAAC;AAgB5F,eAAO,MAAM,+BAA+B,QAAO,IAElD,CAAC;AAEF;;;;;;;;;;;GAWG;AACH,eAAO,MAAM,OAAO,GAAI,MAAM,IAAI,GAAG,IAAI,KAAG,IAE3C,CAAC;AAEF,eAAO,MAAM,OAAO,QAAO,IAAI,GAAG,IAAa,CAAC;AAEhD,MAAM,MAAM,aAAa,GAAG;IAC1B,IAAI,CAAC,EAAE,IAAI,CAAC;IACZ,IAAI,CAAC,EAAE,IAAI,CAAC;IACZ,WAAW,CAAC,EAAE,MAAM,EAAE,CAAC;IACvB;;;qDAGiD;IACjD,UAAU,CAAC,EAAE,OAAO,CAAC;CACtB,CAAC;AAEF,eAAO,MAAM,YAAY,GAAI,OAAO,KAAK,EAAE,SAAS,aAAa,KAAG,IA6CnE,CAAC;AAEF,eAAO,MAAM,gBAAgB,UA/CO,KAAK,WAAW,aAAa,KAAG,IA+CxB,CAAC"}
|
package/lib/capture.js
CHANGED
|
@@ -1,11 +1,23 @@
|
|
|
1
|
+
import { addBreadcrumb, getBreadcrumbs } from './breadcrumbs';
|
|
1
2
|
import { getConfig, isInitialized } from './config';
|
|
2
|
-
import { getBreadcrumbs } from './breadcrumbs';
|
|
3
3
|
import { symbolicateErrorViaMetro } from './handlers/dev-symbolicate';
|
|
4
|
+
import { captureScreenshot } from './handlers/screenshot';
|
|
4
5
|
import { markSessionErrored } from './session-tracker';
|
|
5
6
|
import { parseStack } from './stack';
|
|
6
|
-
import { enqueue } from './transport';
|
|
7
|
+
import { enqueue, uploadAttachment } from './transport';
|
|
7
8
|
import { uuidV7 } from './uuid';
|
|
8
9
|
let _user = null;
|
|
10
|
+
// Phase 42 sub-D.08 — per-session screenshot quota. Defaults: 10 in
|
|
11
|
+
// prod, unlimited (-1 sentinel) in dev so test loops + react-error-
|
|
12
|
+
// overlay reruns don't run out partway through the session.
|
|
13
|
+
const SCREENSHOT_PROD_LIMIT = 10;
|
|
14
|
+
let _screenshotsTaken = 0;
|
|
15
|
+
function screenshotBudget() {
|
|
16
|
+
return typeof __DEV__ !== 'undefined' && __DEV__ ? -1 : SCREENSHOT_PROD_LIMIT;
|
|
17
|
+
}
|
|
18
|
+
export const __resetScreenshotBudgetForTests = () => {
|
|
19
|
+
_screenshotsTaken = 0;
|
|
20
|
+
};
|
|
9
21
|
/**
|
|
10
22
|
* Attach a stable user identifier to events captured after this call.
|
|
11
23
|
*
|
|
@@ -46,20 +58,64 @@ export const captureError = (error, extras) => {
|
|
|
46
58
|
// Phase 26 sub-B: a captured error promotes the current session to
|
|
47
59
|
// `errored` so the next AppState=background ping reports unhealthy.
|
|
48
60
|
markSessionErrored();
|
|
61
|
+
// Phase 42 sub-D.07: opt-in screenshot. Default off; per-call
|
|
62
|
+
// `extras.screenshot: false` always wins so callers can mute it
|
|
63
|
+
// on a sensitive flow even when init has it on globally.
|
|
64
|
+
const wantScreenshot = config.screenshotsEnabled && extras?.screenshot !== false && allowScreenshot();
|
|
49
65
|
// Phase 40 sub-E: in dev there's no uploaded source map, so ask
|
|
50
66
|
// Metro to symbolicate the stack before we send it (best-effort,
|
|
51
67
|
// short timeout). Release builds skip straight to enqueue and let
|
|
52
68
|
// the server symbolicate at ingest against the uploaded map.
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
.catch(() => { })
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
69
|
+
const pipeline = async () => {
|
|
70
|
+
if (typeof __DEV__ !== 'undefined' && __DEV__) {
|
|
71
|
+
await symbolicateErrorViaMetro(event.error).catch(() => { });
|
|
72
|
+
}
|
|
73
|
+
if (wantScreenshot) {
|
|
74
|
+
await captureAndAttachScreenshot(event);
|
|
75
|
+
}
|
|
59
76
|
enqueue(event);
|
|
60
|
-
}
|
|
77
|
+
};
|
|
78
|
+
void pipeline();
|
|
61
79
|
};
|
|
62
80
|
export const captureException = captureError;
|
|
81
|
+
/** Phase 42 sub-D.08: per-session screenshot quota gate. */
|
|
82
|
+
function allowScreenshot() {
|
|
83
|
+
const budget = screenshotBudget();
|
|
84
|
+
if (budget < 0)
|
|
85
|
+
return true; // dev: unlimited
|
|
86
|
+
if (_screenshotsTaken >= budget)
|
|
87
|
+
return false;
|
|
88
|
+
_screenshotsTaken += 1;
|
|
89
|
+
return true;
|
|
90
|
+
}
|
|
91
|
+
/**
|
|
92
|
+
* Phase 42 sub-D.06/07: take a screenshot, upload it, push the
|
|
93
|
+
* server-issued ref into `event.attachments`. Every step is
|
|
94
|
+
* best-effort — on any failure we leave a breadcrumb and let the
|
|
95
|
+
* event ship without a thumbnail.
|
|
96
|
+
*/
|
|
97
|
+
async function captureAndAttachScreenshot(event) {
|
|
98
|
+
let blob = null;
|
|
99
|
+
try {
|
|
100
|
+
blob = await captureScreenshot();
|
|
101
|
+
}
|
|
102
|
+
catch {
|
|
103
|
+
// capture itself shouldn't throw — `captureScreenshot` already
|
|
104
|
+
// catches — but be defensive.
|
|
105
|
+
}
|
|
106
|
+
if (!blob) {
|
|
107
|
+
addBreadcrumb({ type: 'custom', data: { reason: 'screenshot-capture-failed' } });
|
|
108
|
+
return;
|
|
109
|
+
}
|
|
110
|
+
const attachment = await uploadAttachment(event.id, 'screenshot', blob, { source: 'js' });
|
|
111
|
+
if (!attachment) {
|
|
112
|
+
addBreadcrumb({ type: 'custom', data: { reason: 'screenshot-upload-failed' } });
|
|
113
|
+
return;
|
|
114
|
+
}
|
|
115
|
+
if (!event.attachments)
|
|
116
|
+
event.attachments = [];
|
|
117
|
+
event.attachments.push(attachment);
|
|
118
|
+
}
|
|
63
119
|
const errorToObject = (error) => {
|
|
64
120
|
const causeRaw = error.cause;
|
|
65
121
|
let cause = null;
|
package/lib/capture.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"capture.js","sourceRoot":"","sources":["../src/capture.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,aAAa,EAAE,MAAM,UAAU,CAAC;AACpD,OAAO,EAAE,
|
|
1
|
+
{"version":3,"file":"capture.js","sourceRoot":"","sources":["../src/capture.ts"],"names":[],"mappings":"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"}
|
package/lib/config.d.ts
CHANGED
|
@@ -4,6 +4,8 @@ export type Config = {
|
|
|
4
4
|
environment: string;
|
|
5
5
|
ingestUrl: string;
|
|
6
6
|
enabled: boolean;
|
|
7
|
+
/** Phase 42 sub-D.07: opt-in screenshot capture on captureException. */
|
|
8
|
+
screenshotsEnabled: boolean;
|
|
7
9
|
};
|
|
8
10
|
export declare const setConfig: (config: Config) => void;
|
|
9
11
|
export declare const getConfig: () => Config | null;
|
package/lib/config.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"config.d.ts","sourceRoot":"","sources":["../src/config.ts"],"names":[],"mappings":"AAAA,MAAM,MAAM,MAAM,GAAG;IACnB,KAAK,EAAE,MAAM,CAAC;IACd,OAAO,EAAE,MAAM,CAAC;IAChB,WAAW,EAAE,MAAM,CAAC;IACpB,SAAS,EAAE,MAAM,CAAC;IAClB,OAAO,EAAE,OAAO,CAAC;
|
|
1
|
+
{"version":3,"file":"config.d.ts","sourceRoot":"","sources":["../src/config.ts"],"names":[],"mappings":"AAAA,MAAM,MAAM,MAAM,GAAG;IACnB,KAAK,EAAE,MAAM,CAAC;IACd,OAAO,EAAE,MAAM,CAAC;IAChB,WAAW,EAAE,MAAM,CAAC;IACpB,SAAS,EAAE,MAAM,CAAC;IAClB,OAAO,EAAE,OAAO,CAAC;IACjB,wEAAwE;IACxE,kBAAkB,EAAE,OAAO,CAAC;CAC7B,CAAC;AAIF,eAAO,MAAM,SAAS,GAAI,QAAQ,MAAM,KAAG,IAE1C,CAAC;AAEF,eAAO,MAAM,SAAS,QAAO,MAAM,GAAG,IAAe,CAAC;AAEtD,eAAO,MAAM,aAAa,QAAO,OAA2B,CAAC;AAE7D,eAAO,MAAM,eAAe,QAAO,IAElC,CAAC"}
|
package/lib/config.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"config.js","sourceRoot":"","sources":["../src/config.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"config.js","sourceRoot":"","sources":["../src/config.ts"],"names":[],"mappings":"AAUA,IAAI,OAAO,GAAkB,IAAI,CAAC;AAElC,MAAM,CAAC,MAAM,SAAS,GAAG,CAAC,MAAc,EAAQ,EAAE;IAChD,OAAO,GAAG,MAAM,CAAC;AACnB,CAAC,CAAC;AAEF,MAAM,CAAC,MAAM,SAAS,GAAG,GAAkB,EAAE,CAAC,OAAO,CAAC;AAEtD,MAAM,CAAC,MAAM,aAAa,GAAG,GAAY,EAAE,CAAC,OAAO,KAAK,IAAI,CAAC;AAE7D,MAAM,CAAC,MAAM,eAAe,GAAG,GAAS,EAAE;IACxC,OAAO,GAAG,IAAI,CAAC;AACjB,CAAC,CAAC"}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
/** What `captureScreenshot()` hands back when it succeeds. */
|
|
2
|
+
export type ScreenshotBlob = {
|
|
3
|
+
base64: string;
|
|
4
|
+
mediaType: string;
|
|
5
|
+
};
|
|
6
|
+
/**
|
|
7
|
+
* Take one screenshot, yielding the JS thread first. Returns null on
|
|
8
|
+
* any error (missing peer dep, native side refused, timeout, etc.).
|
|
9
|
+
* Caller is responsible for opt-in checks (`config.screenshotsEnabled`).
|
|
10
|
+
*/
|
|
11
|
+
export declare function captureScreenshot(): Promise<ScreenshotBlob | null>;
|
|
12
|
+
//# sourceMappingURL=screenshot.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"screenshot.d.ts","sourceRoot":"","sources":["../../src/handlers/screenshot.ts"],"names":[],"mappings":"AAoDA,8DAA8D;AAC9D,MAAM,MAAM,cAAc,GAAG;IAC3B,MAAM,EAAE,MAAM,CAAC;IACf,SAAS,EAAE,MAAM,CAAC;CACnB,CAAC;AAEF;;;;GAIG;AACH,wBAAsB,iBAAiB,IAAI,OAAO,CAAC,cAAc,GAAG,IAAI,CAAC,CAmCxE"}
|
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
// Phase 42 sub-D.03/04 — capture a screenshot of the current view tree
|
|
2
|
+
// on `captureException`. Off-main-thread, best-effort, opt-in.
|
|
3
|
+
//
|
|
4
|
+
// Performance contract (sub-D.04):
|
|
5
|
+
// - Wait for the in-flight RN interaction batch to drain before
|
|
6
|
+
// touching the view shot (`InteractionManager.runAfterInteractions`)
|
|
7
|
+
// so we never extend the active gesture / animation by a frame.
|
|
8
|
+
// - Yield one paint by chaining a `requestAnimationFrame` so the
|
|
9
|
+
// screenshot reflects post-error UI state, not the frame that
|
|
10
|
+
// was already half-laid-out.
|
|
11
|
+
// - Capped output: 480 px on the longest edge, WebP q=70. Typical
|
|
12
|
+
// payload 30-80 KB; multipart hard cap is 500 KB.
|
|
13
|
+
// - On any failure we silently return null. The error event still
|
|
14
|
+
// goes to the server; the user just doesn't see a thumbnail.
|
|
15
|
+
//
|
|
16
|
+
// `react-native-view-shot` is an OPTIONAL peer. We `require()` it
|
|
17
|
+
// lazily so apps that don't install it never pay the bundle cost
|
|
18
|
+
// or fail at import time. Without it, `captureScreenshot()` returns
|
|
19
|
+
// `null` immediately.
|
|
20
|
+
import { InteractionManager } from 'react-native';
|
|
21
|
+
function loadCaptureRef() {
|
|
22
|
+
try {
|
|
23
|
+
// eslint-disable-next-line @typescript-eslint/no-require-imports
|
|
24
|
+
const mod = require('react-native-view-shot');
|
|
25
|
+
return mod.captureRef ?? mod.default?.captureRef ?? null;
|
|
26
|
+
}
|
|
27
|
+
catch {
|
|
28
|
+
return null;
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
const MAX_LONG_EDGE_PX = 480;
|
|
32
|
+
const WEBP_QUALITY = 0.7;
|
|
33
|
+
const CAPTURE_TIMEOUT_MS = 1500;
|
|
34
|
+
/**
|
|
35
|
+
* Take one screenshot, yielding the JS thread first. Returns null on
|
|
36
|
+
* any error (missing peer dep, native side refused, timeout, etc.).
|
|
37
|
+
* Caller is responsible for opt-in checks (`config.screenshotsEnabled`).
|
|
38
|
+
*/
|
|
39
|
+
export async function captureScreenshot() {
|
|
40
|
+
const captureRef = loadCaptureRef();
|
|
41
|
+
if (!captureRef)
|
|
42
|
+
return null;
|
|
43
|
+
// Wait for the in-flight RN interaction batch to drain. This is
|
|
44
|
+
// why screenshot capture doesn't visibly stall the user's last
|
|
45
|
+
// action — we let React commit before we ask the OS to render.
|
|
46
|
+
await new Promise((resolve) => {
|
|
47
|
+
InteractionManager.runAfterInteractions(() => resolve());
|
|
48
|
+
});
|
|
49
|
+
await new Promise((resolve) => {
|
|
50
|
+
requestAnimationFrame(() => resolve());
|
|
51
|
+
});
|
|
52
|
+
try {
|
|
53
|
+
const base64 = await withTimeout(captureRef(undefined, {
|
|
54
|
+
format: 'jpg',
|
|
55
|
+
quality: WEBP_QUALITY,
|
|
56
|
+
result: 'base64',
|
|
57
|
+
// Long-edge cap. RN view-shot scales preserving aspect ratio
|
|
58
|
+
// when only one dimension is set.
|
|
59
|
+
width: MAX_LONG_EDGE_PX,
|
|
60
|
+
}), CAPTURE_TIMEOUT_MS);
|
|
61
|
+
if (!base64)
|
|
62
|
+
return null;
|
|
63
|
+
// view-shot doesn't ship a WebP encoder on every RN version.
|
|
64
|
+
// JPEG q=70 fits the budget too (typical 40-100 KB) and every
|
|
65
|
+
// version handles it identically. We can swap to WebP once the
|
|
66
|
+
// RN minimum we support has it everywhere.
|
|
67
|
+
return { base64, mediaType: 'image/jpeg' };
|
|
68
|
+
}
|
|
69
|
+
catch {
|
|
70
|
+
return null;
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
function withTimeout(p, ms) {
|
|
74
|
+
return new Promise((resolve) => {
|
|
75
|
+
const t = setTimeout(() => resolve(null), ms);
|
|
76
|
+
p.then((v) => {
|
|
77
|
+
clearTimeout(t);
|
|
78
|
+
resolve(v);
|
|
79
|
+
}, () => {
|
|
80
|
+
clearTimeout(t);
|
|
81
|
+
resolve(null);
|
|
82
|
+
});
|
|
83
|
+
});
|
|
84
|
+
}
|
|
85
|
+
//# sourceMappingURL=screenshot.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"screenshot.js","sourceRoot":"","sources":["../../src/handlers/screenshot.ts"],"names":[],"mappings":"AAAA,uEAAuE;AACvE,+DAA+D;AAC/D,EAAE;AACF,mCAAmC;AACnC,kEAAkE;AAClE,yEAAyE;AACzE,oEAAoE;AACpE,mEAAmE;AACnE,kEAAkE;AAClE,iCAAiC;AACjC,oEAAoE;AACpE,sDAAsD;AACtD,oEAAoE;AACpE,iEAAiE;AACjE,EAAE;AACF,kEAAkE;AAClE,iEAAiE;AACjE,oEAAoE;AACpE,sBAAsB;AAEtB,OAAO,EAAE,kBAAkB,EAAE,MAAM,cAAc,CAAC;AAkBlD,SAAS,cAAc;IACrB,IAAI,CAAC;QACH,iEAAiE;QACjE,MAAM,GAAG,GAAG,OAAO,CAAC,wBAAwB,CAAmB,CAAC;QAChE,OAAO,GAAG,CAAC,UAAU,IAAI,GAAG,CAAC,OAAO,EAAE,UAAU,IAAI,IAAI,CAAC;IAC3D,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,IAAI,CAAC;IACd,CAAC;AACH,CAAC;AAED,MAAM,gBAAgB,GAAG,GAAG,CAAC;AAC7B,MAAM,YAAY,GAAG,GAAG,CAAC;AACzB,MAAM,kBAAkB,GAAG,IAAI,CAAC;AAQhC;;;;GAIG;AACH,MAAM,CAAC,KAAK,UAAU,iBAAiB;IACrC,MAAM,UAAU,GAAG,cAAc,EAAE,CAAC;IACpC,IAAI,CAAC,UAAU;QAAE,OAAO,IAAI,CAAC;IAE7B,gEAAgE;IAChE,+DAA+D;IAC/D,+DAA+D;IAC/D,MAAM,IAAI,OAAO,CAAO,CAAC,OAAO,EAAE,EAAE;QAClC,kBAAkB,CAAC,oBAAoB,CAAC,GAAG,EAAE,CAAC,OAAO,EAAE,CAAC,CAAC;IAC3D,CAAC,CAAC,CAAC;IACH,MAAM,IAAI,OAAO,CAAO,CAAC,OAAO,EAAE,EAAE;QAClC,qBAAqB,CAAC,GAAG,EAAE,CAAC,OAAO,EAAE,CAAC,CAAC;IACzC,CAAC,CAAC,CAAC;IAEH,IAAI,CAAC;QACH,MAAM,MAAM,GAAG,MAAM,WAAW,CAC9B,UAAU,CAAC,SAAS,EAAE;YACpB,MAAM,EAAE,KAAK;YACb,OAAO,EAAE,YAAY;YACrB,MAAM,EAAE,QAAQ;YAChB,6DAA6D;YAC7D,kCAAkC;YAClC,KAAK,EAAE,gBAAgB;SACxB,CAAC,EACF,kBAAkB,CACnB,CAAC;QACF,IAAI,CAAC,MAAM;YAAE,OAAO,IAAI,CAAC;QACzB,6DAA6D;QAC7D,8DAA8D;QAC9D,+DAA+D;QAC/D,2CAA2C;QAC3C,OAAO,EAAE,MAAM,EAAE,SAAS,EAAE,YAAY,EAAE,CAAC;IAC7C,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,IAAI,CAAC;IACd,CAAC;AACH,CAAC;AAED,SAAS,WAAW,CAAI,CAAa,EAAE,EAAU;IAC/C,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE;QAC7B,MAAM,CAAC,GAAG,UAAU,CAAC,GAAG,EAAE,CAAC,OAAO,CAAC,IAAoB,CAAC,EAAE,EAAE,CAAC,CAAC;QAC9D,CAAC,CAAC,IAAI,CACJ,CAAC,CAAC,EAAE,EAAE;YACJ,YAAY,CAAC,CAAC,CAAC,CAAC;YAChB,OAAO,CAAC,CAAC,CAAC,CAAC;QACb,CAAC,EACD,GAAG,EAAE;YACH,YAAY,CAAC,CAAC,CAAC,CAAC;YAChB,OAAO,CAAC,IAAoB,CAAC,CAAC;QAChC,CAAC,CACF,CAAC;IACJ,CAAC,CAAC,CAAC;AACL,CAAC"}
|
package/lib/index.d.ts
CHANGED
|
@@ -1,4 +1,5 @@
|
|
|
1
1
|
import { ErrorBoundary } from './error-boundary';
|
|
2
|
+
import { MaskRegion, setMaskedNode, unsetMaskedNode } from './mask';
|
|
2
3
|
export declare const sentori: {
|
|
3
4
|
init: (options: import("./init").InitOptions) => void;
|
|
4
5
|
addBreadcrumb: (input: import("./breadcrumbs").AddBreadcrumbInput) => void;
|
|
@@ -7,6 +8,9 @@ export declare const sentori: {
|
|
|
7
8
|
captureError: (error: Error, extras?: import("./capture").CaptureExtras) => void;
|
|
8
9
|
captureException: (error: Error, extras?: import("./capture").CaptureExtras) => void;
|
|
9
10
|
ErrorBoundary: typeof ErrorBoundary;
|
|
11
|
+
MaskRegion: typeof MaskRegion;
|
|
12
|
+
setMaskedNode: typeof setMaskedNode;
|
|
13
|
+
unsetMaskedNode: typeof unsetMaskedNode;
|
|
10
14
|
startSession: () => void;
|
|
11
15
|
endSession: (status?: "exited") => void;
|
|
12
16
|
markSessionCrashed: () => void;
|
|
@@ -16,6 +20,7 @@ export { init, init as initSentori } from './init';
|
|
|
16
20
|
export { addBreadcrumb } from './breadcrumbs';
|
|
17
21
|
export { setUser, getUser, captureError, captureException } from './capture';
|
|
18
22
|
export { ErrorBoundary } from './error-boundary';
|
|
23
|
+
export { MaskRegion, setMaskedNode, unsetMaskedNode } from './mask';
|
|
19
24
|
export { startAnrWatchdog, stopAnrWatchdog, triggerNativeCrash, } from './native';
|
|
20
25
|
export { endSession, markSessionCrashed, startSession, } from './session-tracker';
|
|
21
26
|
export { type NavigationRefLike, useTraceNavigation } from './navigation';
|
package/lib/index.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,aAAa,EAAE,MAAM,kBAAkB,CAAC;
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,aAAa,EAAE,MAAM,kBAAkB,CAAC;AACjD,OAAO,EAAE,UAAU,EAAE,aAAa,EAAE,eAAe,EAAE,MAAM,QAAQ,CAAC;AAOpE,eAAO,MAAM,OAAO;;;;;;;;;;;;;;CAcnB,CAAC;AAEF,eAAe,OAAO,CAAC;AAEvB,OAAO,EAAE,IAAI,EAAE,IAAI,IAAI,WAAW,EAAE,MAAM,QAAQ,CAAC;AACnD,OAAO,EAAE,aAAa,EAAE,MAAM,eAAe,CAAC;AAC9C,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,YAAY,EAAE,gBAAgB,EAAE,MAAM,WAAW,CAAC;AAC7E,OAAO,EAAE,aAAa,EAAE,MAAM,kBAAkB,CAAC;AACjD,OAAO,EAAE,UAAU,EAAE,aAAa,EAAE,eAAe,EAAE,MAAM,QAAQ,CAAC;AACpE,OAAO,EACL,gBAAgB,EAChB,eAAe,EACf,kBAAkB,GACnB,MAAM,UAAU,CAAC;AAClB,OAAO,EACL,UAAU,EACV,kBAAkB,EAClB,YAAY,GACb,MAAM,mBAAmB,CAAC;AAC3B,OAAO,EAAE,KAAK,iBAAiB,EAAE,kBAAkB,EAAE,MAAM,cAAc,CAAC;AAE1E,YAAY,EACV,KAAK,EACL,YAAY,EACZ,KAAK,EACL,UAAU,EACV,cAAc,EACd,MAAM,EACN,QAAQ,EACR,GAAG,EACH,IAAI,EACJ,IAAI,EACJ,SAAS,EACT,QAAQ,GACT,MAAM,SAAS,CAAC"}
|
package/lib/index.js
CHANGED
|
@@ -2,6 +2,7 @@ import { init } from './init';
|
|
|
2
2
|
import { addBreadcrumb } from './breadcrumbs';
|
|
3
3
|
import { setUser, getUser, captureError, captureException } from './capture';
|
|
4
4
|
import { ErrorBoundary } from './error-boundary';
|
|
5
|
+
import { MaskRegion, setMaskedNode, unsetMaskedNode } from './mask';
|
|
5
6
|
import { endSession, markSessionCrashed, startSession, } from './session-tracker';
|
|
6
7
|
export const sentori = {
|
|
7
8
|
init,
|
|
@@ -11,6 +12,9 @@ export const sentori = {
|
|
|
11
12
|
captureError,
|
|
12
13
|
captureException,
|
|
13
14
|
ErrorBoundary,
|
|
15
|
+
MaskRegion,
|
|
16
|
+
setMaskedNode,
|
|
17
|
+
unsetMaskedNode,
|
|
14
18
|
startSession,
|
|
15
19
|
endSession,
|
|
16
20
|
markSessionCrashed,
|
|
@@ -20,6 +24,7 @@ export { init, init as initSentori } from './init';
|
|
|
20
24
|
export { addBreadcrumb } from './breadcrumbs';
|
|
21
25
|
export { setUser, getUser, captureError, captureException } from './capture';
|
|
22
26
|
export { ErrorBoundary } from './error-boundary';
|
|
27
|
+
export { MaskRegion, setMaskedNode, unsetMaskedNode } from './mask';
|
|
23
28
|
export { startAnrWatchdog, stopAnrWatchdog, triggerNativeCrash, } from './native';
|
|
24
29
|
export { endSession, markSessionCrashed, startSession, } from './session-tracker';
|
|
25
30
|
export { useTraceNavigation } from './navigation';
|
package/lib/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,QAAQ,CAAC;AAC9B,OAAO,EAAE,aAAa,EAAE,MAAM,eAAe,CAAC;AAC9C,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,YAAY,EAAE,gBAAgB,EAAE,MAAM,WAAW,CAAC;AAC7E,OAAO,EAAE,aAAa,EAAE,MAAM,kBAAkB,CAAC;AACjD,OAAO,EACL,UAAU,EACV,kBAAkB,EAClB,YAAY,GACb,MAAM,mBAAmB,CAAC;AAE3B,MAAM,CAAC,MAAM,OAAO,GAAG;IACrB,IAAI;IACJ,aAAa;IACb,OAAO;IACP,OAAO;IACP,YAAY;IACZ,gBAAgB;IAChB,aAAa;IACb,YAAY;IACZ,UAAU;IACV,kBAAkB;CACnB,CAAC;AAEF,eAAe,OAAO,CAAC;AAEvB,OAAO,EAAE,IAAI,EAAE,IAAI,IAAI,WAAW,EAAE,MAAM,QAAQ,CAAC;AACnD,OAAO,EAAE,aAAa,EAAE,MAAM,eAAe,CAAC;AAC9C,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,YAAY,EAAE,gBAAgB,EAAE,MAAM,WAAW,CAAC;AAC7E,OAAO,EAAE,aAAa,EAAE,MAAM,kBAAkB,CAAC;AACjD,OAAO,EACL,gBAAgB,EAChB,eAAe,EACf,kBAAkB,GACnB,MAAM,UAAU,CAAC;AAClB,OAAO,EACL,UAAU,EACV,kBAAkB,EAClB,YAAY,GACb,MAAM,mBAAmB,CAAC;AAC3B,OAAO,EAA0B,kBAAkB,EAAE,MAAM,cAAc,CAAC"}
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,QAAQ,CAAC;AAC9B,OAAO,EAAE,aAAa,EAAE,MAAM,eAAe,CAAC;AAC9C,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,YAAY,EAAE,gBAAgB,EAAE,MAAM,WAAW,CAAC;AAC7E,OAAO,EAAE,aAAa,EAAE,MAAM,kBAAkB,CAAC;AACjD,OAAO,EAAE,UAAU,EAAE,aAAa,EAAE,eAAe,EAAE,MAAM,QAAQ,CAAC;AACpE,OAAO,EACL,UAAU,EACV,kBAAkB,EAClB,YAAY,GACb,MAAM,mBAAmB,CAAC;AAE3B,MAAM,CAAC,MAAM,OAAO,GAAG;IACrB,IAAI;IACJ,aAAa;IACb,OAAO;IACP,OAAO;IACP,YAAY;IACZ,gBAAgB;IAChB,aAAa;IACb,UAAU;IACV,aAAa;IACb,eAAe;IACf,YAAY;IACZ,UAAU;IACV,kBAAkB;CACnB,CAAC;AAEF,eAAe,OAAO,CAAC;AAEvB,OAAO,EAAE,IAAI,EAAE,IAAI,IAAI,WAAW,EAAE,MAAM,QAAQ,CAAC;AACnD,OAAO,EAAE,aAAa,EAAE,MAAM,eAAe,CAAC;AAC9C,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,YAAY,EAAE,gBAAgB,EAAE,MAAM,WAAW,CAAC;AAC7E,OAAO,EAAE,aAAa,EAAE,MAAM,kBAAkB,CAAC;AACjD,OAAO,EAAE,UAAU,EAAE,aAAa,EAAE,eAAe,EAAE,MAAM,QAAQ,CAAC;AACpE,OAAO,EACL,gBAAgB,EAChB,eAAe,EACf,kBAAkB,GACnB,MAAM,UAAU,CAAC;AAClB,OAAO,EACL,UAAU,EACV,kBAAkB,EAClB,YAAY,GACb,MAAM,mBAAmB,CAAC;AAC3B,OAAO,EAA0B,kBAAkB,EAAE,MAAM,cAAc,CAAC"}
|
package/lib/init.d.ts
CHANGED
|
@@ -16,6 +16,11 @@ export type InitOptions = {
|
|
|
16
16
|
* foreground (`AppState` → `active`), ends it on background.
|
|
17
17
|
* Drives crash-free rate. Set `false` to opt out. */
|
|
18
18
|
sessions?: boolean;
|
|
19
|
+
/** Phase 42 sub-D.07: capture a screenshot of the current screen
|
|
20
|
+
* on `captureException`. Opt-in — requires `react-native-view-shot`
|
|
21
|
+
* installed and `<MaskRegion>` placed over any sensitive UI. The
|
|
22
|
+
* image is webp q=70 480 px max, < 100 KB typical. */
|
|
23
|
+
screenshot?: boolean;
|
|
19
24
|
};
|
|
20
25
|
};
|
|
21
26
|
export declare const init: (options: InitOptions) => void;
|
package/lib/init.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"init.d.ts","sourceRoot":"","sources":["../src/init.ts"],"names":[],"mappings":"AAYA,MAAM,MAAM,WAAW,GAAG;IACxB,sDAAsD;IACtD,KAAK,EAAE,MAAM,CAAC;IACd,4DAA4D;IAC5D,OAAO,EAAE,MAAM,CAAC;IAChB,sEAAsE;IACtE,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,qFAAqF;IACrF,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,iEAAiE;IACjE,OAAO,CAAC,EAAE;QACR,YAAY,CAAC,EAAE,OAAO,CAAC;QACvB,iBAAiB,CAAC,EAAE,OAAO,CAAC;QAC5B,OAAO,CAAC,EAAE,OAAO,CAAC;QAClB;;8DAEsD;QACtD,QAAQ,CAAC,EAAE,OAAO,CAAC;
|
|
1
|
+
{"version":3,"file":"init.d.ts","sourceRoot":"","sources":["../src/init.ts"],"names":[],"mappings":"AAYA,MAAM,MAAM,WAAW,GAAG;IACxB,sDAAsD;IACtD,KAAK,EAAE,MAAM,CAAC;IACd,4DAA4D;IAC5D,OAAO,EAAE,MAAM,CAAC;IAChB,sEAAsE;IACtE,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,qFAAqF;IACrF,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,iEAAiE;IACjE,OAAO,CAAC,EAAE;QACR,YAAY,CAAC,EAAE,OAAO,CAAC;QACvB,iBAAiB,CAAC,EAAE,OAAO,CAAC;QAC5B,OAAO,CAAC,EAAE,OAAO,CAAC;QAClB;;8DAEsD;QACtD,QAAQ,CAAC,EAAE,OAAO,CAAC;QACnB;;;+DAGuD;QACvD,UAAU,CAAC,EAAE,OAAO,CAAC;KACtB,CAAC;CACH,CAAC;AAIF,eAAO,MAAM,IAAI,GAAI,SAAS,WAAW,KAAG,IA0D3C,CAAC"}
|
package/lib/init.js
CHANGED
|
@@ -22,6 +22,7 @@ export const init = (options) => {
|
|
|
22
22
|
environment: env,
|
|
23
23
|
ingestUrl: options.ingestUrl ?? DEFAULT_INGEST_URL,
|
|
24
24
|
enabled: true,
|
|
25
|
+
screenshotsEnabled: options.capture?.screenshot === true,
|
|
25
26
|
});
|
|
26
27
|
// Tell the native crash handler about the config so the JSON it writes
|
|
27
28
|
// on the next NSException / Java uncaught carries release + env.
|
package/lib/init.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"init.js","sourceRoot":"","sources":["../src/init.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,UAAU,CAAC;AACrC,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,kBAAkB,EAAE,eAAe,EAAE,MAAM,UAAU,CAAC;AAC/D,OAAO,EAAE,YAAY,EAAE,MAAM,mBAAmB,CAAC;AACjD,OAAO,EAAE,iBAAiB,EAAE,OAAO,EAAE,cAAc,EAAE,MAAM,aAAa,CAAC;
|
|
1
|
+
{"version":3,"file":"init.js","sourceRoot":"","sources":["../src/init.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,UAAU,CAAC;AACrC,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,kBAAkB,EAAE,eAAe,EAAE,MAAM,UAAU,CAAC;AAC/D,OAAO,EAAE,YAAY,EAAE,MAAM,mBAAmB,CAAC;AACjD,OAAO,EAAE,iBAAiB,EAAE,OAAO,EAAE,cAAc,EAAE,MAAM,aAAa,CAAC;AA+BzE,MAAM,kBAAkB,GAAG,iCAAiC,CAAC;AAE7D,MAAM,CAAC,MAAM,IAAI,GAAG,CAAC,OAAoB,EAAQ,EAAE;IACjD,IAAI,CAAC,OAAO,CAAC,KAAK,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;QAC1D,MAAM,IAAI,KAAK,CAAC,yDAAyD,CAAC,CAAC;IAC7E,CAAC;IACD,IAAI,CAAC,OAAO,CAAC,OAAO,EAAE,CAAC;QACrB,MAAM,IAAI,KAAK,CAAC,8BAA8B,CAAC,CAAC;IAClD,CAAC;IAED,MAAM,GAAG,GACP,OAAO,CAAC,WAAW;QACnB,CAAC,OAAO,OAAO,KAAK,WAAW,IAAI,OAAO,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC;IAE/D,SAAS,CAAC;QACR,KAAK,EAAE,OAAO,CAAC,KAAK;QACpB,OAAO,EAAE,OAAO,CAAC,OAAO;QACxB,WAAW,EAAE,GAAG;QAChB,SAAS,EAAE,OAAO,CAAC,SAAS,IAAI,kBAAkB;QAClD,OAAO,EAAE,IAAI;QACb,kBAAkB,EAAE,OAAO,CAAC,OAAO,EAAE,UAAU,KAAK,IAAI;KACzD,CAAC,CAAC;IAEH,uEAAuE;IACvE,iEAAiE;IACjE,eAAe,CAAC;QACd,KAAK,EAAE,OAAO,CAAC,KAAK;QACpB,OAAO,EAAE,OAAO,CAAC,OAAO;QACxB,WAAW,EAAE,GAAG;KACjB,CAAC,CAAC;IAEH,cAAc,EAAE,CAAC;IAEjB,MAAM,OAAO,GAAG,OAAO,CAAC,OAAO,IAAI,EAAE,CAAC;IACtC,IAAI,OAAO,CAAC,YAAY,KAAK,KAAK;QAAE,oBAAoB,EAAE,CAAC;IAC3D,IAAI,OAAO,CAAC,iBAAiB,KAAK,KAAK;QAAE,qBAAqB,EAAE,CAAC;IACjE,IAAI,OAAO,CAAC,OAAO,KAAK,KAAK;QAAE,qBAAqB,EAAE,CAAC;IACvD,IAAI,OAAO,CAAC,QAAQ,KAAK,KAAK,EAAE,CAAC;QAC/B,+DAA+D;QAC/D,kEAAkE;QAClE,gEAAgE;QAChE,YAAY,EAAE,CAAC;QACf,uBAAuB,EAAE,CAAC;IAC5B,CAAC;IAED,8DAA8D;IAC9D,2DAA2D;IAC3D,iDAAiD;IACjD,kBAAkB,EAAE;SACjB,IAAI,CAAC,CAAC,KAAK,EAAE,EAAE;QACd,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;YACzB,IAAI,CAAC;gBACH,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAU,CAAC,CAAC;YACrC,CAAC;YAAC,MAAM,CAAC;gBACP,iBAAiB;YACnB,CAAC;QACH,CAAC;IACH,CAAC,CAAC;SACD,KAAK,CAAC,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;IACnB,iBAAiB,EAAE,CAAC,KAAK,CAAC,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;AACtC,CAAC,CAAC"}
|
package/lib/mask.d.ts
ADDED
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
import React, { type ReactNode } from 'react';
|
|
2
|
+
import { View, type ViewProps } from 'react-native';
|
|
3
|
+
/**
|
|
4
|
+
* Imperative registration: when you can't wrap the sensitive view
|
|
5
|
+
* in `<MaskRegion>`, drop a ref on it and call `setMaskedNode(ref)`.
|
|
6
|
+
* Future captures will mask the subtree.
|
|
7
|
+
*/
|
|
8
|
+
export declare function setMaskedNode(node: React.Component | View | null | unknown): void;
|
|
9
|
+
/** Removes a previously registered ref. Pair this with mount/unmount
|
|
10
|
+
* lifecycle hooks if the node is short-lived. */
|
|
11
|
+
export declare function unsetMaskedNode(node: React.Component | View | null | unknown): void;
|
|
12
|
+
/** Returns the current set of registered masked nodes + nativeIDs.
|
|
13
|
+
* Read by the native screenshotter layer in sub-E / sub-F. */
|
|
14
|
+
export declare function getMaskedRegions(): {
|
|
15
|
+
refs: Set<unknown>;
|
|
16
|
+
nativeIds: Set<string>;
|
|
17
|
+
};
|
|
18
|
+
/**
|
|
19
|
+
* Declarative redaction. `<MaskRegion>{children}</MaskRegion>` keeps
|
|
20
|
+
* the children visible in normal flight; under capture, the wrapping
|
|
21
|
+
* view is repainted black so the rendered screenshot doesn't leak
|
|
22
|
+
* the underlying pixels.
|
|
23
|
+
*/
|
|
24
|
+
export declare function MaskRegion({ children, nativeID, ...rest }: {
|
|
25
|
+
children: ReactNode;
|
|
26
|
+
nativeID?: string;
|
|
27
|
+
} & ViewProps): React.JSX.Element;
|
|
28
|
+
/** Test-only — flush registration tables. */
|
|
29
|
+
export declare function __resetMaskedRegionsForTests(): void;
|
|
30
|
+
//# sourceMappingURL=mask.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"mask.d.ts","sourceRoot":"","sources":["../src/mask.tsx"],"names":[],"mappings":"AAyBA,OAAO,KAAK,EAAE,EAAE,KAAK,SAAS,EAAqB,MAAM,OAAO,CAAC;AACjE,OAAO,EAAE,IAAI,EAAE,KAAK,SAAS,EAAE,MAAM,cAAc,CAAC;AAMpD;;;;GAIG;AACH,wBAAgB,aAAa,CAAC,IAAI,EAAE,KAAK,CAAC,SAAS,GAAG,IAAI,GAAG,IAAI,GAAG,OAAO,GAAG,IAAI,CAGjF;AAED;kDACkD;AAClD,wBAAgB,eAAe,CAAC,IAAI,EAAE,KAAK,CAAC,SAAS,GAAG,IAAI,GAAG,IAAI,GAAG,OAAO,GAAG,IAAI,CAGnF;AAED;+DAC+D;AAC/D,wBAAgB,gBAAgB,IAAI;IAClC,IAAI,EAAE,GAAG,CAAC,OAAO,CAAC,CAAC;IACnB,SAAS,EAAE,GAAG,CAAC,MAAM,CAAC,CAAC;CACxB,CAEA;AAED;;;;;GAKG;AACH,wBAAgB,UAAU,CAAC,EACzB,QAAQ,EACR,QAAQ,EACR,GAAG,IAAI,EACR,EAAE;IAAE,QAAQ,EAAE,SAAS,CAAC;IAAC,QAAQ,CAAC,EAAE,MAAM,CAAA;CAAE,GAAG,SAAS,GAAG,KAAK,CAAC,GAAG,CAAC,OAAO,CAoB5E;AAED,6CAA6C;AAC7C,wBAAgB,4BAA4B,IAAI,IAAI,CAGnD"}
|
package/lib/mask.js
ADDED
|
@@ -0,0 +1,77 @@
|
|
|
1
|
+
import { jsx as _jsx } from "react/jsx-runtime";
|
|
2
|
+
// Phase 42 sub-D.09/10 — mark UI regions as "do not screenshot".
|
|
3
|
+
//
|
|
4
|
+
// `<MaskRegion>` wraps any subtree the SDK should redact before
|
|
5
|
+
// shipping a crash screenshot. It's purely declarative — the
|
|
6
|
+
// component renders its children as-is in normal flight, but its
|
|
7
|
+
// `View` is tagged with `collapsable={false}` + a sentinel
|
|
8
|
+
// `nativeID` so the platform-level screenshotters
|
|
9
|
+
// (`react-native-view-shot`, the iOS / Android native crash
|
|
10
|
+
// capturers we add in sub-E / sub-F) can find it and paint over.
|
|
11
|
+
//
|
|
12
|
+
// `setMaskedNode(ref)` is the imperative escape hatch: useful
|
|
13
|
+
// when the sensitive view isn't yours to wrap (a third-party
|
|
14
|
+
// modal, a video player, etc.). Pass a React ref obtained via
|
|
15
|
+
// `createRef()` / `useRef()` and the SDK will redact that
|
|
16
|
+
// subtree the next time it captures.
|
|
17
|
+
//
|
|
18
|
+
// `getMaskedRegions()` returns the current set of native tags;
|
|
19
|
+
// `captureScreenshot()` would consult this list, but
|
|
20
|
+
// `react-native-view-shot` doesn't expose a "redact these rects"
|
|
21
|
+
// hook — so this iteration ships the registration API only and
|
|
22
|
+
// the rendered overlay lives behind a default-on
|
|
23
|
+
// `<View style={{ backgroundColor: '#000' }}>` you can wrap
|
|
24
|
+
// yourself. The iOS / Android crash-time screenshotters in
|
|
25
|
+
// sub-E / sub-F will read this list before drawing.
|
|
26
|
+
import { useEffect, useRef } from 'react';
|
|
27
|
+
import { View } from 'react-native';
|
|
28
|
+
/** Component-level node identifiers we've been asked to redact. */
|
|
29
|
+
const _maskedRefs = new Set();
|
|
30
|
+
const _maskedNativeIds = new Set();
|
|
31
|
+
/**
|
|
32
|
+
* Imperative registration: when you can't wrap the sensitive view
|
|
33
|
+
* in `<MaskRegion>`, drop a ref on it and call `setMaskedNode(ref)`.
|
|
34
|
+
* Future captures will mask the subtree.
|
|
35
|
+
*/
|
|
36
|
+
export function setMaskedNode(node) {
|
|
37
|
+
if (!node)
|
|
38
|
+
return;
|
|
39
|
+
_maskedRefs.add(node);
|
|
40
|
+
}
|
|
41
|
+
/** Removes a previously registered ref. Pair this with mount/unmount
|
|
42
|
+
* lifecycle hooks if the node is short-lived. */
|
|
43
|
+
export function unsetMaskedNode(node) {
|
|
44
|
+
if (!node)
|
|
45
|
+
return;
|
|
46
|
+
_maskedRefs.delete(node);
|
|
47
|
+
}
|
|
48
|
+
/** Returns the current set of registered masked nodes + nativeIDs.
|
|
49
|
+
* Read by the native screenshotter layer in sub-E / sub-F. */
|
|
50
|
+
export function getMaskedRegions() {
|
|
51
|
+
return { nativeIds: _maskedNativeIds, refs: _maskedRefs };
|
|
52
|
+
}
|
|
53
|
+
/**
|
|
54
|
+
* Declarative redaction. `<MaskRegion>{children}</MaskRegion>` keeps
|
|
55
|
+
* the children visible in normal flight; under capture, the wrapping
|
|
56
|
+
* view is repainted black so the rendered screenshot doesn't leak
|
|
57
|
+
* the underlying pixels.
|
|
58
|
+
*/
|
|
59
|
+
export function MaskRegion({ children, nativeID, ...rest }) {
|
|
60
|
+
// Auto-generate a stable nativeID per mount so the native
|
|
61
|
+
// screenshotter can find this view by ID at capture time.
|
|
62
|
+
const idRef = useRef(nativeID ?? `sentori-mask-${Math.random().toString(36).slice(2, 10)}`);
|
|
63
|
+
useEffect(() => {
|
|
64
|
+
const id = idRef.current;
|
|
65
|
+
_maskedNativeIds.add(id);
|
|
66
|
+
return () => {
|
|
67
|
+
_maskedNativeIds.delete(id);
|
|
68
|
+
};
|
|
69
|
+
}, []);
|
|
70
|
+
return (_jsx(View, { collapsable: false, nativeID: idRef.current, ...rest, children: children }));
|
|
71
|
+
}
|
|
72
|
+
/** Test-only — flush registration tables. */
|
|
73
|
+
export function __resetMaskedRegionsForTests() {
|
|
74
|
+
_maskedRefs.clear();
|
|
75
|
+
_maskedNativeIds.clear();
|
|
76
|
+
}
|
|
77
|
+
//# sourceMappingURL=mask.js.map
|
package/lib/mask.js.map
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"mask.js","sourceRoot":"","sources":["../src/mask.tsx"],"names":[],"mappings":";AAAA,iEAAiE;AACjE,EAAE;AACF,gEAAgE;AAChE,6DAA6D;AAC7D,iEAAiE;AACjE,2DAA2D;AAC3D,kDAAkD;AAClD,4DAA4D;AAC5D,iEAAiE;AACjE,EAAE;AACF,8DAA8D;AAC9D,6DAA6D;AAC7D,8DAA8D;AAC9D,0DAA0D;AAC1D,qCAAqC;AACrC,EAAE;AACF,+DAA+D;AAC/D,qDAAqD;AACrD,iEAAiE;AACjE,+DAA+D;AAC/D,iDAAiD;AACjD,4DAA4D;AAC5D,2DAA2D;AAC3D,oDAAoD;AAEpD,OAAc,EAAkB,SAAS,EAAE,MAAM,EAAE,MAAM,OAAO,CAAC;AACjE,OAAO,EAAE,IAAI,EAAkB,MAAM,cAAc,CAAC;AAEpD,mEAAmE;AACnE,MAAM,WAAW,GAAG,IAAI,GAAG,EAAoC,CAAC;AAChE,MAAM,gBAAgB,GAAG,IAAI,GAAG,EAAU,CAAC;AAE3C;;;;GAIG;AACH,MAAM,UAAU,aAAa,CAAC,IAA6C;IACzE,IAAI,CAAC,IAAI;QAAE,OAAO;IAClB,WAAW,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;AACxB,CAAC;AAED;kDACkD;AAClD,MAAM,UAAU,eAAe,CAAC,IAA6C;IAC3E,IAAI,CAAC,IAAI;QAAE,OAAO;IAClB,WAAW,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;AAC3B,CAAC;AAED;+DAC+D;AAC/D,MAAM,UAAU,gBAAgB;IAI9B,OAAO,EAAE,SAAS,EAAE,gBAAgB,EAAE,IAAI,EAAE,WAAW,EAAE,CAAC;AAC5D,CAAC;AAED;;;;;GAKG;AACH,MAAM,UAAU,UAAU,CAAC,EACzB,QAAQ,EACR,QAAQ,EACR,GAAG,IAAI,EACgD;IACvD,0DAA0D;IAC1D,0DAA0D;IAC1D,MAAM,KAAK,GAAG,MAAM,CAClB,QAAQ,IAAI,gBAAgB,IAAI,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,EAAE,CACtE,CAAC;IAEF,SAAS,CAAC,GAAG,EAAE;QACb,MAAM,EAAE,GAAG,KAAK,CAAC,OAAO,CAAC;QACzB,gBAAgB,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;QACzB,OAAO,GAAG,EAAE;YACV,gBAAgB,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;QAC9B,CAAC,CAAC;IACJ,CAAC,EAAE,EAAE,CAAC,CAAC;IAEP,OAAO,CACL,KAAC,IAAI,IAAC,WAAW,EAAE,KAAK,EAAE,QAAQ,EAAE,KAAK,CAAC,OAAO,KAAM,IAAI,YACxD,QAAQ,GACJ,CACR,CAAC;AACJ,CAAC;AAED,6CAA6C;AAC7C,MAAM,UAAU,4BAA4B;IAC1C,WAAW,CAAC,KAAK,EAAE,CAAC;IACpB,gBAAgB,CAAC,KAAK,EAAE,CAAC;AAC3B,CAAC"}
|
package/lib/transport.d.ts
CHANGED
|
@@ -14,4 +14,26 @@ export declare const __peekQueue: () => readonly Event[];
|
|
|
14
14
|
* this.
|
|
15
15
|
*/
|
|
16
16
|
export declare const sendSessionPing: (ingestUrl: string, token: string, ping: unknown) => Promise<void>;
|
|
17
|
+
/**
|
|
18
|
+
* Upload a base64-encoded binary blob as an attachment for a known
|
|
19
|
+
* event. The event must NOT have been POSTed yet — the server-side
|
|
20
|
+
* ingest validation in events.rs only honours `event.attachments[].ref`
|
|
21
|
+
* when the matching `event_attachments` row already exists for the
|
|
22
|
+
* same (event_id, project_id). Caller's contract:
|
|
23
|
+
*
|
|
24
|
+
* 1. Generate `event.id` (uuidV7).
|
|
25
|
+
* 2. Build the blob (e.g. via `captureScreenshot`).
|
|
26
|
+
* 3. `await uploadAttachment(...)` → get `{ ref, sizeBytes, mediaType }`.
|
|
27
|
+
* 4. Push `{ ref, kind, ... }` into `event.attachments` then enqueue.
|
|
28
|
+
*
|
|
29
|
+
* Returns `null` on any non-fatal failure (network down, store
|
|
30
|
+
* disabled, 4xx, timeout). The error event still ships without the
|
|
31
|
+
* attachment so we never lose the actual crash.
|
|
32
|
+
*/
|
|
33
|
+
export declare const uploadAttachment: (eventId: string, kind: import("./types").AttachmentMeta["kind"], blob: {
|
|
34
|
+
base64: string;
|
|
35
|
+
mediaType: string;
|
|
36
|
+
}, opts?: {
|
|
37
|
+
source?: "android" | "ios" | "js";
|
|
38
|
+
}) => Promise<import("./types").AttachmentMeta | null>;
|
|
17
39
|
//# sourceMappingURL=transport.d.ts.map
|
package/lib/transport.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"transport.d.ts","sourceRoot":"","sources":["../src/transport.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,KAAK,EAAE,MAAM,SAAS,CAAC;AAqBrC,eAAO,MAAM,OAAO,GAAI,OAAO,KAAK,KAAG,IAUtC,CAAC;AAEF,eAAO,MAAM,cAAc,QAAO,IAOjC,CAAC;AAEF,eAAO,MAAM,UAAU,QAAa,OAAO,CAAC,IAAI,CAe/C,CAAC;AAqBF,eAAO,MAAM,KAAK,QAAa,OAAO,CAAC,IAAI,CAkB1C,CAAC;AA4FF,eAAO,MAAM,iBAAiB,QAAa,OAAO,CAAC,IAAI,CAatD,CAAC;AAEF,eAAO,MAAM,eAAe,QAAO,IAOlC,CAAC;AAEF,eAAO,MAAM,WAAW,QAAO,SAAS,KAAK,EAAY,CAAC;AAE1D;;;;;;GAMG;AACH,eAAO,MAAM,eAAe,GAC1B,WAAW,MAAM,EACjB,OAAO,MAAM,EACb,MAAM,OAAO,KACZ,OAAO,CAAC,IAAI,CAcd,CAAC"}
|
|
1
|
+
{"version":3,"file":"transport.d.ts","sourceRoot":"","sources":["../src/transport.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,KAAK,EAAE,MAAM,SAAS,CAAC;AAqBrC,eAAO,MAAM,OAAO,GAAI,OAAO,KAAK,KAAG,IAUtC,CAAC;AAEF,eAAO,MAAM,cAAc,QAAO,IAOjC,CAAC;AAEF,eAAO,MAAM,UAAU,QAAa,OAAO,CAAC,IAAI,CAe/C,CAAC;AAqBF,eAAO,MAAM,KAAK,QAAa,OAAO,CAAC,IAAI,CAkB1C,CAAC;AA4FF,eAAO,MAAM,iBAAiB,QAAa,OAAO,CAAC,IAAI,CAatD,CAAC;AAEF,eAAO,MAAM,eAAe,QAAO,IAOlC,CAAC;AAEF,eAAO,MAAM,WAAW,QAAO,SAAS,KAAK,EAAY,CAAC;AAE1D;;;;;;GAMG;AACH,eAAO,MAAM,eAAe,GAC1B,WAAW,MAAM,EACjB,OAAO,MAAM,EACb,MAAM,OAAO,KACZ,OAAO,CAAC,IAAI,CAcd,CAAC;AAMF;;;;;;;;;;;;;;;GAeG;AACH,eAAO,MAAM,gBAAgB,GAC3B,SAAS,MAAM,EACf,MAAM,OAAO,SAAS,EAAE,cAAc,CAAC,MAAM,CAAC,EAC9C,MAAM;IAAE,MAAM,EAAE,MAAM,CAAC;IAAC,SAAS,EAAE,MAAM,CAAA;CAAE,EAC3C,OAAM;IAAE,MAAM,CAAC,EAAE,SAAS,GAAG,KAAK,GAAG,IAAI,CAAA;CAAO,KAC/C,OAAO,CAAC,OAAO,SAAS,EAAE,cAAc,GAAG,IAAI,CA6CjD,CAAC"}
|
package/lib/transport.js
CHANGED
|
@@ -213,4 +213,66 @@ export const sendSessionPing = async (ingestUrl, token, ping) => {
|
|
|
213
213
|
// best-effort
|
|
214
214
|
}
|
|
215
215
|
};
|
|
216
|
+
// ──────────────────────────────────────────────────────────────────
|
|
217
|
+
// Phase 42 sub-D.05 — attachment upload pipeline
|
|
218
|
+
// ──────────────────────────────────────────────────────────────────
|
|
219
|
+
/**
|
|
220
|
+
* Upload a base64-encoded binary blob as an attachment for a known
|
|
221
|
+
* event. The event must NOT have been POSTed yet — the server-side
|
|
222
|
+
* ingest validation in events.rs only honours `event.attachments[].ref`
|
|
223
|
+
* when the matching `event_attachments` row already exists for the
|
|
224
|
+
* same (event_id, project_id). Caller's contract:
|
|
225
|
+
*
|
|
226
|
+
* 1. Generate `event.id` (uuidV7).
|
|
227
|
+
* 2. Build the blob (e.g. via `captureScreenshot`).
|
|
228
|
+
* 3. `await uploadAttachment(...)` → get `{ ref, sizeBytes, mediaType }`.
|
|
229
|
+
* 4. Push `{ ref, kind, ... }` into `event.attachments` then enqueue.
|
|
230
|
+
*
|
|
231
|
+
* Returns `null` on any non-fatal failure (network down, store
|
|
232
|
+
* disabled, 4xx, timeout). The error event still ships without the
|
|
233
|
+
* attachment so we never lose the actual crash.
|
|
234
|
+
*/
|
|
235
|
+
export const uploadAttachment = async (eventId, kind, blob, opts = {}) => {
|
|
236
|
+
const config = getConfig();
|
|
237
|
+
if (!config)
|
|
238
|
+
return null;
|
|
239
|
+
const url = `${config.ingestUrl}/v1/events/${encodeURIComponent(eventId)}/attachments/${encodeURIComponent(kind)}`;
|
|
240
|
+
// RN-style multipart: `{ uri, type, name }` is what the native
|
|
241
|
+
// FormData implementation expects for a file part — the bridge
|
|
242
|
+
// serializes a data: URI without us having to allocate a Blob.
|
|
243
|
+
const form = new FormData();
|
|
244
|
+
form.append('file', {
|
|
245
|
+
name: filenameFor(kind, blob.mediaType),
|
|
246
|
+
type: blob.mediaType,
|
|
247
|
+
uri: `data:${blob.mediaType};base64,${blob.base64}`,
|
|
248
|
+
});
|
|
249
|
+
form.append('source', opts.source ?? 'js');
|
|
250
|
+
try {
|
|
251
|
+
const resp = await fetch(url, {
|
|
252
|
+
body: form,
|
|
253
|
+
headers: {
|
|
254
|
+
Authorization: `Bearer ${config.token}`,
|
|
255
|
+
'Sentori-Sdk': `react-native/${SDK_VERSION}`,
|
|
256
|
+
},
|
|
257
|
+
method: 'POST',
|
|
258
|
+
});
|
|
259
|
+
if (resp.status !== 201)
|
|
260
|
+
return null;
|
|
261
|
+
const j = (await resp.json());
|
|
262
|
+
return {
|
|
263
|
+
kind,
|
|
264
|
+
mediaType: j.mediaType,
|
|
265
|
+
ref: j.refId,
|
|
266
|
+
sizeBytes: j.sizeBytes,
|
|
267
|
+
source: opts.source ?? 'js',
|
|
268
|
+
};
|
|
269
|
+
}
|
|
270
|
+
catch {
|
|
271
|
+
return null;
|
|
272
|
+
}
|
|
273
|
+
};
|
|
274
|
+
function filenameFor(kind, mediaType) {
|
|
275
|
+
const ext = mediaType.split('/')[1] ?? 'bin';
|
|
276
|
+
return `${kind}.${ext}`;
|
|
277
|
+
}
|
|
216
278
|
//# sourceMappingURL=transport.js.map
|