@bugpulse/react-native 0.1.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 (150) hide show
  1. package/README.md +185 -0
  2. package/dist/AnnotationCanvas.d.ts +11 -0
  3. package/dist/AnnotationCanvas.d.ts.map +1 -0
  4. package/dist/AnnotationCanvas.js +154 -0
  5. package/dist/AnnotationCanvas.js.map +1 -0
  6. package/dist/BugReportModal.d.ts +15 -0
  7. package/dist/BugReportModal.d.ts.map +1 -0
  8. package/dist/BugReportModal.js +303 -0
  9. package/dist/BugReportModal.js.map +1 -0
  10. package/dist/BugReportProvider.d.ts +19 -0
  11. package/dist/BugReportProvider.d.ts.map +1 -0
  12. package/dist/BugReportProvider.js +103 -0
  13. package/dist/BugReportProvider.js.map +1 -0
  14. package/dist/DeviceInfo.d.ts +3 -0
  15. package/dist/DeviceInfo.d.ts.map +1 -0
  16. package/dist/DeviceInfo.js +63 -0
  17. package/dist/DeviceInfo.js.map +1 -0
  18. package/dist/ErrorBoundary.d.ts +19 -0
  19. package/dist/ErrorBoundary.d.ts.map +1 -0
  20. package/dist/ErrorBoundary.js +79 -0
  21. package/dist/ErrorBoundary.js.map +1 -0
  22. package/dist/NavigationTracker.d.ts +7 -0
  23. package/dist/NavigationTracker.d.ts.map +1 -0
  24. package/dist/NavigationTracker.js +61 -0
  25. package/dist/NavigationTracker.js.map +1 -0
  26. package/dist/RingBuffer.d.ts +10 -0
  27. package/dist/RingBuffer.d.ts.map +1 -0
  28. package/dist/RingBuffer.js +26 -0
  29. package/dist/RingBuffer.js.map +1 -0
  30. package/dist/ScreenCapture.d.ts +3 -0
  31. package/dist/ScreenCapture.d.ts.map +1 -0
  32. package/dist/ScreenCapture.js +20 -0
  33. package/dist/ScreenCapture.js.map +1 -0
  34. package/dist/ShakeDetector.d.ts +5 -0
  35. package/dist/ShakeDetector.d.ts.map +1 -0
  36. package/dist/ShakeDetector.js +44 -0
  37. package/dist/ShakeDetector.js.map +1 -0
  38. package/dist/StateCapture.d.ts +13 -0
  39. package/dist/StateCapture.d.ts.map +1 -0
  40. package/dist/StateCapture.js +74 -0
  41. package/dist/StateCapture.js.map +1 -0
  42. package/dist/__tests__/DeviceInfo.test.d.ts +2 -0
  43. package/dist/__tests__/DeviceInfo.test.d.ts.map +1 -0
  44. package/dist/__tests__/DeviceInfo.test.js +63 -0
  45. package/dist/__tests__/DeviceInfo.test.js.map +1 -0
  46. package/dist/__tests__/ErrorBoundary.test.d.ts +2 -0
  47. package/dist/__tests__/ErrorBoundary.test.d.ts.map +1 -0
  48. package/dist/__tests__/ErrorBoundary.test.js +64 -0
  49. package/dist/__tests__/ErrorBoundary.test.js.map +1 -0
  50. package/dist/__tests__/NavigationTracker.test.d.ts +2 -0
  51. package/dist/__tests__/NavigationTracker.test.d.ts.map +1 -0
  52. package/dist/__tests__/NavigationTracker.test.js +54 -0
  53. package/dist/__tests__/NavigationTracker.test.js.map +1 -0
  54. package/dist/__tests__/RingBuffer.test.d.ts +2 -0
  55. package/dist/__tests__/RingBuffer.test.d.ts.map +1 -0
  56. package/dist/__tests__/RingBuffer.test.js +43 -0
  57. package/dist/__tests__/RingBuffer.test.js.map +1 -0
  58. package/dist/__tests__/ScreenCapture.test.d.ts +2 -0
  59. package/dist/__tests__/ScreenCapture.test.d.ts.map +1 -0
  60. package/dist/__tests__/ScreenCapture.test.js +27 -0
  61. package/dist/__tests__/ScreenCapture.test.js.map +1 -0
  62. package/dist/__tests__/ShakeDetector.test.d.ts +2 -0
  63. package/dist/__tests__/ShakeDetector.test.d.ts.map +1 -0
  64. package/dist/__tests__/ShakeDetector.test.js +65 -0
  65. package/dist/__tests__/ShakeDetector.test.js.map +1 -0
  66. package/dist/__tests__/StateCapture.test.d.ts +2 -0
  67. package/dist/__tests__/StateCapture.test.d.ts.map +1 -0
  68. package/dist/__tests__/StateCapture.test.js +128 -0
  69. package/dist/__tests__/StateCapture.test.js.map +1 -0
  70. package/dist/__tests__/__mocks__/expo-clipboard.d.ts +3 -0
  71. package/dist/__tests__/__mocks__/expo-clipboard.d.ts.map +1 -0
  72. package/dist/__tests__/__mocks__/expo-clipboard.js +6 -0
  73. package/dist/__tests__/__mocks__/expo-clipboard.js.map +1 -0
  74. package/dist/__tests__/__mocks__/expo-constants.d.ts +17 -0
  75. package/dist/__tests__/__mocks__/expo-constants.d.ts.map +1 -0
  76. package/dist/__tests__/__mocks__/expo-constants.js +8 -0
  77. package/dist/__tests__/__mocks__/expo-constants.js.map +1 -0
  78. package/dist/__tests__/__mocks__/expo-device.d.ts +2 -0
  79. package/dist/__tests__/__mocks__/expo-device.d.ts.map +1 -0
  80. package/dist/__tests__/__mocks__/expo-device.js +5 -0
  81. package/dist/__tests__/__mocks__/expo-device.js.map +1 -0
  82. package/dist/__tests__/__mocks__/expo-file-system-legacy.d.ts +8 -0
  83. package/dist/__tests__/__mocks__/expo-file-system-legacy.d.ts.map +1 -0
  84. package/dist/__tests__/__mocks__/expo-file-system-legacy.js +10 -0
  85. package/dist/__tests__/__mocks__/expo-file-system-legacy.js.map +1 -0
  86. package/dist/__tests__/__mocks__/expo-localization.d.ts +6 -0
  87. package/dist/__tests__/__mocks__/expo-localization.d.ts.map +1 -0
  88. package/dist/__tests__/__mocks__/expo-localization.js +7 -0
  89. package/dist/__tests__/__mocks__/expo-localization.js.map +1 -0
  90. package/dist/__tests__/__mocks__/expo-router.d.ts +5 -0
  91. package/dist/__tests__/__mocks__/expo-router.d.ts.map +1 -0
  92. package/dist/__tests__/__mocks__/expo-router.js +16 -0
  93. package/dist/__tests__/__mocks__/expo-router.js.map +1 -0
  94. package/dist/__tests__/__mocks__/expo-sensors.d.ts +17 -0
  95. package/dist/__tests__/__mocks__/expo-sensors.d.ts.map +1 -0
  96. package/dist/__tests__/__mocks__/expo-sensors.js +22 -0
  97. package/dist/__tests__/__mocks__/expo-sensors.js.map +1 -0
  98. package/dist/__tests__/__mocks__/react-native-gesture-handler.d.ts +11 -0
  99. package/dist/__tests__/__mocks__/react-native-gesture-handler.d.ts.map +1 -0
  100. package/dist/__tests__/__mocks__/react-native-gesture-handler.js +14 -0
  101. package/dist/__tests__/__mocks__/react-native-gesture-handler.js.map +1 -0
  102. package/dist/__tests__/__mocks__/react-native-svg.d.ts +4 -0
  103. package/dist/__tests__/__mocks__/react-native-svg.d.ts.map +1 -0
  104. package/dist/__tests__/__mocks__/react-native-svg.js +7 -0
  105. package/dist/__tests__/__mocks__/react-native-svg.js.map +1 -0
  106. package/dist/__tests__/__mocks__/react-native-view-shot.d.ts +2 -0
  107. package/dist/__tests__/__mocks__/react-native-view-shot.d.ts.map +1 -0
  108. package/dist/__tests__/__mocks__/react-native-view-shot.js +5 -0
  109. package/dist/__tests__/__mocks__/react-native-view-shot.js.map +1 -0
  110. package/dist/__tests__/__mocks__/react-native.d.ts +20 -0
  111. package/dist/__tests__/__mocks__/react-native.d.ts.map +1 -0
  112. package/dist/__tests__/__mocks__/react-native.js +14 -0
  113. package/dist/__tests__/__mocks__/react-native.js.map +1 -0
  114. package/dist/__tests__/fileToBase64.test.d.ts +2 -0
  115. package/dist/__tests__/fileToBase64.test.d.ts.map +1 -0
  116. package/dist/__tests__/fileToBase64.test.js +21 -0
  117. package/dist/__tests__/fileToBase64.test.js.map +1 -0
  118. package/dist/__tests__/integrations/slack.test.d.ts +2 -0
  119. package/dist/__tests__/integrations/slack.test.d.ts.map +1 -0
  120. package/dist/__tests__/integrations/slack.test.js +123 -0
  121. package/dist/__tests__/integrations/slack.test.js.map +1 -0
  122. package/dist/__tests__/integrations/webhook.test.d.ts +2 -0
  123. package/dist/__tests__/integrations/webhook.test.d.ts.map +1 -0
  124. package/dist/__tests__/integrations/webhook.test.js +98 -0
  125. package/dist/__tests__/integrations/webhook.test.js.map +1 -0
  126. package/dist/index.d.ts +8 -0
  127. package/dist/index.d.ts.map +1 -0
  128. package/dist/index.js +18 -0
  129. package/dist/index.js.map +1 -0
  130. package/dist/integrations/index.d.ts +4 -0
  131. package/dist/integrations/index.d.ts.map +1 -0
  132. package/dist/integrations/index.js +8 -0
  133. package/dist/integrations/index.js.map +1 -0
  134. package/dist/integrations/slack.d.ts +9 -0
  135. package/dist/integrations/slack.d.ts.map +1 -0
  136. package/dist/integrations/slack.js +136 -0
  137. package/dist/integrations/slack.js.map +1 -0
  138. package/dist/integrations/types.d.ts +53 -0
  139. package/dist/integrations/types.d.ts.map +1 -0
  140. package/dist/integrations/types.js +3 -0
  141. package/dist/integrations/types.js.map +1 -0
  142. package/dist/integrations/webhook.d.ts +8 -0
  143. package/dist/integrations/webhook.d.ts.map +1 -0
  144. package/dist/integrations/webhook.js +45 -0
  145. package/dist/integrations/webhook.js.map +1 -0
  146. package/dist/utils/fileToBase64.d.ts +2 -0
  147. package/dist/utils/fileToBase64.d.ts.map +1 -0
  148. package/dist/utils/fileToBase64.js +11 -0
  149. package/dist/utils/fileToBase64.js.map +1 -0
  150. package/package.json +77 -0
@@ -0,0 +1,123 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const slack_1 = require("../../integrations/slack");
4
+ const mockReport = {
5
+ screenshot: 'file:///tmp/screenshot.png',
6
+ annotatedScreenshot: null,
7
+ description: 'Button does not work',
8
+ device: {
9
+ model: 'iPhone 15 Pro',
10
+ os: 'ios 17.0',
11
+ appVersion: '1.0.0',
12
+ screenSize: '390x844',
13
+ locale: 'en-US',
14
+ installationId: 'test-id',
15
+ expoConfig: { name: 'TestApp', slug: 'test-app' },
16
+ },
17
+ screen: 'HomeScreen',
18
+ timestamp: '2026-03-28T12:00:00.000Z',
19
+ metadata: { userId: '123' },
20
+ };
21
+ describe('SlackIntegration', () => {
22
+ let originalFetch;
23
+ let consoleWarnSpy;
24
+ beforeEach(() => {
25
+ originalFetch = global.fetch;
26
+ global.fetch = jest.fn();
27
+ consoleWarnSpy = jest.spyOn(console, 'warn').mockImplementation();
28
+ });
29
+ afterEach(() => {
30
+ global.fetch = originalFetch;
31
+ consoleWarnSpy.mockRestore();
32
+ });
33
+ it('sends message to slack webhook', async () => {
34
+ global.fetch.mockResolvedValue({ ok: true });
35
+ const integration = (0, slack_1.SlackIntegration)({ webhookUrl: 'https://hooks.slack.com/test' });
36
+ const result = await integration.send({ ...mockReport, screenshot: null });
37
+ expect(result.success).toBe(true);
38
+ expect(global.fetch).toHaveBeenCalledWith('https://hooks.slack.com/test', expect.objectContaining({ method: 'POST' }));
39
+ });
40
+ it('warns when screenshot exists but no imageUploadKey', async () => {
41
+ global.fetch.mockResolvedValue({ ok: true });
42
+ const integration = (0, slack_1.SlackIntegration)({ webhookUrl: 'https://hooks.slack.com/test' });
43
+ await integration.send(mockReport);
44
+ expect(consoleWarnSpy).toHaveBeenCalledWith(expect.stringContaining('no imageUploadKey configured'));
45
+ });
46
+ it('does not warn when no screenshot', async () => {
47
+ global.fetch.mockResolvedValue({ ok: true });
48
+ const integration = (0, slack_1.SlackIntegration)({ webhookUrl: 'https://hooks.slack.com/test' });
49
+ await integration.send({ ...mockReport, screenshot: null });
50
+ expect(consoleWarnSpy).not.toHaveBeenCalled();
51
+ });
52
+ it('uploads image when imageUploadKey provided', async () => {
53
+ global.fetch
54
+ .mockResolvedValueOnce({ ok: true, json: () => Promise.resolve({ data: { url: 'https://i.imgbb.com/test.png' } }) })
55
+ .mockResolvedValueOnce({ ok: true });
56
+ const integration = (0, slack_1.SlackIntegration)({
57
+ webhookUrl: 'https://hooks.slack.com/test',
58
+ imageUploadKey: 'test-key',
59
+ });
60
+ const result = await integration.send(mockReport);
61
+ expect(result.success).toBe(true);
62
+ expect(global.fetch).toHaveBeenCalledTimes(2);
63
+ });
64
+ it('returns error on slack non-200', async () => {
65
+ global.fetch.mockResolvedValue({ ok: false, status: 403 });
66
+ const integration = (0, slack_1.SlackIntegration)({ webhookUrl: 'https://hooks.slack.com/test' });
67
+ const result = await integration.send({ ...mockReport, screenshot: null });
68
+ expect(result.success).toBe(false);
69
+ expect(result.error).toContain('403');
70
+ });
71
+ it('returns error on network failure', async () => {
72
+ global.fetch.mockRejectedValue(new Error('Network error'));
73
+ const integration = (0, slack_1.SlackIntegration)({ webhookUrl: 'https://hooks.slack.com/test' });
74
+ const result = await integration.send({ ...mockReport, screenshot: null });
75
+ expect(result.success).toBe(false);
76
+ expect(result.error).toBe('Network error');
77
+ });
78
+ it('includes truncated diagnostics in slack message', async () => {
79
+ global.fetch.mockResolvedValue({ ok: true });
80
+ const integration = (0, slack_1.SlackIntegration)({ webhookUrl: 'https://hooks.slack.com/test' });
81
+ const reportWithDiagnostics = {
82
+ ...mockReport,
83
+ screenshot: null,
84
+ diagnostics: {
85
+ stateSnapshots: [
86
+ { name: 'app', state: '{"count":42}', timestamp: '2026-03-28T12:00:00Z', truncated: false },
87
+ ],
88
+ navHistory: [
89
+ { pathname: '/home', segments: ['home'], timestamp: '2026-03-28T12:00:00Z' },
90
+ { pathname: '/settings', segments: ['settings'], timestamp: '2026-03-28T12:01:00Z' },
91
+ ],
92
+ lastError: {
93
+ message: 'TypeError: undefined is not a function',
94
+ stack: 'TypeError: undefined is not a function\n at Object.<anonymous>',
95
+ componentStack: null,
96
+ timestamp: '2026-03-28T12:00:00Z',
97
+ },
98
+ },
99
+ };
100
+ await integration.send(reportWithDiagnostics);
101
+ const body = JSON.parse(global.fetch.mock.calls[0][1].body);
102
+ const blocks = body.blocks;
103
+ // Should have header, main info, diagnostics, no image
104
+ expect(blocks.length).toBeGreaterThanOrEqual(3);
105
+ const diagnosticsBlock = blocks.find((b) => b.type === 'section' && b.text?.text?.includes('Last Error'));
106
+ expect(diagnosticsBlock).toBeDefined();
107
+ expect(diagnosticsBlock.text.text).toContain('TypeError');
108
+ expect(diagnosticsBlock.text.text).toContain('/home');
109
+ expect(diagnosticsBlock.text.text).toContain('app');
110
+ });
111
+ it('handles image upload failure gracefully', async () => {
112
+ global.fetch
113
+ .mockResolvedValueOnce({ ok: false })
114
+ .mockResolvedValueOnce({ ok: true });
115
+ const integration = (0, slack_1.SlackIntegration)({
116
+ webhookUrl: 'https://hooks.slack.com/test',
117
+ imageUploadKey: 'test-key',
118
+ });
119
+ const result = await integration.send(mockReport);
120
+ expect(result.success).toBe(true);
121
+ });
122
+ });
123
+ //# sourceMappingURL=slack.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"slack.test.js","sourceRoot":"","sources":["../../../src/__tests__/integrations/slack.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=webhook.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"webhook.test.d.ts","sourceRoot":"","sources":["../../../src/__tests__/integrations/webhook.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,98 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const webhook_1 = require("../../integrations/webhook");
4
+ const mockReport = {
5
+ screenshot: 'file:///tmp/screenshot.png',
6
+ annotatedScreenshot: 'file:///tmp/annotated.png',
7
+ description: 'Something broke',
8
+ device: {
9
+ model: 'iPhone 15 Pro',
10
+ os: 'ios 17.0',
11
+ appVersion: '1.0.0',
12
+ screenSize: '390x844',
13
+ locale: 'en-US',
14
+ installationId: 'test-id',
15
+ expoConfig: { name: 'TestApp', slug: 'test-app' },
16
+ },
17
+ screen: 'Settings',
18
+ timestamp: '2026-03-28T12:00:00.000Z',
19
+ metadata: {},
20
+ };
21
+ describe('WebhookIntegration', () => {
22
+ let originalFetch;
23
+ beforeEach(() => {
24
+ originalFetch = global.fetch;
25
+ global.fetch = jest.fn();
26
+ });
27
+ afterEach(() => {
28
+ global.fetch = originalFetch;
29
+ });
30
+ it('posts report with base64 screenshots', async () => {
31
+ global.fetch.mockResolvedValue({ ok: true });
32
+ const integration = (0, webhook_1.WebhookIntegration)({ url: 'https://api.example.com/bugs' });
33
+ const result = await integration.send(mockReport);
34
+ expect(result.success).toBe(true);
35
+ const body = JSON.parse(global.fetch.mock.calls[0][1].body);
36
+ expect(body.screenshotBase64).toBe('dGVzdGJhc2U2NA==');
37
+ expect(body.annotatedScreenshotBase64).toBe('dGVzdGJhc2U2NA==');
38
+ expect(body.description).toBe('Something broke');
39
+ });
40
+ it('sends null for screenshots when not provided', async () => {
41
+ global.fetch.mockResolvedValue({ ok: true });
42
+ const integration = (0, webhook_1.WebhookIntegration)({ url: 'https://api.example.com/bugs' });
43
+ const result = await integration.send({ ...mockReport, screenshot: null, annotatedScreenshot: null });
44
+ expect(result.success).toBe(true);
45
+ const body = JSON.parse(global.fetch.mock.calls[0][1].body);
46
+ expect(body.screenshotBase64).toBeNull();
47
+ expect(body.annotatedScreenshotBase64).toBeNull();
48
+ });
49
+ it('includes custom headers', async () => {
50
+ global.fetch.mockResolvedValue({ ok: true });
51
+ const integration = (0, webhook_1.WebhookIntegration)({
52
+ url: 'https://api.example.com/bugs',
53
+ headers: { Authorization: 'Bearer token123' },
54
+ });
55
+ await integration.send({ ...mockReport, screenshot: null, annotatedScreenshot: null });
56
+ const headers = global.fetch.mock.calls[0][1].headers;
57
+ expect(headers.Authorization).toBe('Bearer token123');
58
+ expect(headers['Content-Type']).toBe('application/json');
59
+ });
60
+ it('returns error on non-200', async () => {
61
+ global.fetch.mockResolvedValue({ ok: false, status: 500 });
62
+ const integration = (0, webhook_1.WebhookIntegration)({ url: 'https://api.example.com/bugs' });
63
+ const result = await integration.send({ ...mockReport, screenshot: null, annotatedScreenshot: null });
64
+ expect(result.success).toBe(false);
65
+ expect(result.error).toContain('500');
66
+ });
67
+ it('returns error on network failure', async () => {
68
+ global.fetch.mockRejectedValue(new Error('Connection refused'));
69
+ const integration = (0, webhook_1.WebhookIntegration)({ url: 'https://api.example.com/bugs' });
70
+ const result = await integration.send({ ...mockReport, screenshot: null, annotatedScreenshot: null });
71
+ expect(result.success).toBe(false);
72
+ expect(result.error).toBe('Connection refused');
73
+ });
74
+ it('includes full diagnostics in payload', async () => {
75
+ global.fetch.mockResolvedValue({ ok: true });
76
+ const integration = (0, webhook_1.WebhookIntegration)({ url: 'https://api.example.com/bugs' });
77
+ const diagnostics = {
78
+ stateSnapshots: [
79
+ { name: 'app', state: '{"count":42}', timestamp: '2026-03-28T12:00:00Z', truncated: false },
80
+ ],
81
+ navHistory: [
82
+ { pathname: '/home', segments: ['home'], timestamp: '2026-03-28T12:00:00Z' },
83
+ ],
84
+ lastError: null,
85
+ };
86
+ await integration.send({
87
+ ...mockReport,
88
+ screenshot: null,
89
+ annotatedScreenshot: null,
90
+ diagnostics,
91
+ });
92
+ const body = JSON.parse(global.fetch.mock.calls[0][1].body);
93
+ expect(body.diagnostics).toEqual(diagnostics);
94
+ expect(body.diagnostics.stateSnapshots).toHaveLength(1);
95
+ expect(body.diagnostics.navHistory).toHaveLength(1);
96
+ });
97
+ });
98
+ //# sourceMappingURL=webhook.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"webhook.test.js","sourceRoot":"","sources":["../../../src/__tests__/integrations/webhook.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,8 @@
1
+ export { BugReportProvider, useBugReport } from './BugReportProvider';
2
+ export { SlackIntegration } from './integrations/slack';
3
+ export { WebhookIntegration } from './integrations/webhook';
4
+ export { trackStore, untrackStore } from './StateCapture';
5
+ export { useNavigationTracker } from './NavigationTracker';
6
+ export { BugPulseErrorBoundary } from './ErrorBoundary';
7
+ export type { Integration, BugReport, DeviceInfo, SendResult, Diagnostics, StateSnapshot, NavEntry, ErrorInfo, } from './integrations/types';
8
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,iBAAiB,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AACtE,OAAO,EAAE,gBAAgB,EAAE,MAAM,sBAAsB,CAAC;AACxD,OAAO,EAAE,kBAAkB,EAAE,MAAM,wBAAwB,CAAC;AAC5D,OAAO,EAAE,UAAU,EAAE,YAAY,EAAE,MAAM,gBAAgB,CAAC;AAC1D,OAAO,EAAE,oBAAoB,EAAE,MAAM,qBAAqB,CAAC;AAC3D,OAAO,EAAE,qBAAqB,EAAE,MAAM,iBAAiB,CAAC;AACxD,YAAY,EACV,WAAW,EACX,SAAS,EACT,UAAU,EACV,UAAU,EACV,WAAW,EACX,aAAa,EACb,QAAQ,EACR,SAAS,GACV,MAAM,sBAAsB,CAAC"}
package/dist/index.js ADDED
@@ -0,0 +1,18 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.BugPulseErrorBoundary = exports.useNavigationTracker = exports.untrackStore = exports.trackStore = exports.WebhookIntegration = exports.SlackIntegration = exports.useBugReport = exports.BugReportProvider = void 0;
4
+ var BugReportProvider_1 = require("./BugReportProvider");
5
+ Object.defineProperty(exports, "BugReportProvider", { enumerable: true, get: function () { return BugReportProvider_1.BugReportProvider; } });
6
+ Object.defineProperty(exports, "useBugReport", { enumerable: true, get: function () { return BugReportProvider_1.useBugReport; } });
7
+ var slack_1 = require("./integrations/slack");
8
+ Object.defineProperty(exports, "SlackIntegration", { enumerable: true, get: function () { return slack_1.SlackIntegration; } });
9
+ var webhook_1 = require("./integrations/webhook");
10
+ Object.defineProperty(exports, "WebhookIntegration", { enumerable: true, get: function () { return webhook_1.WebhookIntegration; } });
11
+ var StateCapture_1 = require("./StateCapture");
12
+ Object.defineProperty(exports, "trackStore", { enumerable: true, get: function () { return StateCapture_1.trackStore; } });
13
+ Object.defineProperty(exports, "untrackStore", { enumerable: true, get: function () { return StateCapture_1.untrackStore; } });
14
+ var NavigationTracker_1 = require("./NavigationTracker");
15
+ Object.defineProperty(exports, "useNavigationTracker", { enumerable: true, get: function () { return NavigationTracker_1.useNavigationTracker; } });
16
+ var ErrorBoundary_1 = require("./ErrorBoundary");
17
+ Object.defineProperty(exports, "BugPulseErrorBoundary", { enumerable: true, get: function () { return ErrorBoundary_1.BugPulseErrorBoundary; } });
18
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";;;AAAA,yDAAsE;AAA7D,sHAAA,iBAAiB,OAAA;AAAE,iHAAA,YAAY,OAAA;AACxC,8CAAwD;AAA/C,yGAAA,gBAAgB,OAAA;AACzB,kDAA4D;AAAnD,6GAAA,kBAAkB,OAAA;AAC3B,+CAA0D;AAAjD,0GAAA,UAAU,OAAA;AAAE,4GAAA,YAAY,OAAA;AACjC,yDAA2D;AAAlD,yHAAA,oBAAoB,OAAA;AAC7B,iDAAwD;AAA/C,sHAAA,qBAAqB,OAAA"}
@@ -0,0 +1,4 @@
1
+ export { SlackIntegration } from './slack';
2
+ export { WebhookIntegration } from './webhook';
3
+ export type { Integration, BugReport, DeviceInfo, SendResult } from './types';
4
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/integrations/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,gBAAgB,EAAE,MAAM,SAAS,CAAC;AAC3C,OAAO,EAAE,kBAAkB,EAAE,MAAM,WAAW,CAAC;AAC/C,YAAY,EAAE,WAAW,EAAE,SAAS,EAAE,UAAU,EAAE,UAAU,EAAE,MAAM,SAAS,CAAC"}
@@ -0,0 +1,8 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.WebhookIntegration = exports.SlackIntegration = void 0;
4
+ var slack_1 = require("./slack");
5
+ Object.defineProperty(exports, "SlackIntegration", { enumerable: true, get: function () { return slack_1.SlackIntegration; } });
6
+ var webhook_1 = require("./webhook");
7
+ Object.defineProperty(exports, "WebhookIntegration", { enumerable: true, get: function () { return webhook_1.WebhookIntegration; } });
8
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/integrations/index.ts"],"names":[],"mappings":";;;AAAA,iCAA2C;AAAlC,yGAAA,gBAAgB,OAAA;AACzB,qCAA+C;AAAtC,6GAAA,kBAAkB,OAAA"}
@@ -0,0 +1,9 @@
1
+ import type { Integration } from './types';
2
+ interface SlackConfig {
3
+ webhookUrl: string;
4
+ imageUploadKey?: string;
5
+ imageUploadUrl?: string;
6
+ }
7
+ export declare function SlackIntegration(config: SlackConfig): Integration;
8
+ export {};
9
+ //# sourceMappingURL=slack.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"slack.d.ts","sourceRoot":"","sources":["../../src/integrations/slack.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAa,WAAW,EAAc,MAAM,SAAS,CAAC;AAElE,UAAU,WAAW;IACnB,UAAU,EAAE,MAAM,CAAC;IACnB,cAAc,CAAC,EAAE,MAAM,CAAC;IACxB,cAAc,CAAC,EAAE,MAAM,CAAC;CACzB;AAyHD,wBAAgB,gBAAgB,CAAC,MAAM,EAAE,WAAW,GAAG,WAAW,CAgDjE"}
@@ -0,0 +1,136 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.SlackIntegration = SlackIntegration;
4
+ const fileToBase64_1 = require("../utils/fileToBase64");
5
+ const DEFAULT_IMGBB_URL = 'https://api.imgbb.com/1/upload';
6
+ async function uploadImage(uri, apiKey, uploadUrl) {
7
+ try {
8
+ const base64 = await (0, fileToBase64_1.fileToBase64)(uri);
9
+ const formData = new FormData();
10
+ formData.append('key', apiKey);
11
+ formData.append('image', base64);
12
+ const response = await fetch(uploadUrl, {
13
+ method: 'POST',
14
+ body: formData,
15
+ });
16
+ if (!response.ok)
17
+ return null;
18
+ const data = await response.json();
19
+ return data?.data?.url ?? null;
20
+ }
21
+ catch {
22
+ return null;
23
+ }
24
+ }
25
+ function truncateString(str, maxLen) {
26
+ if (str.length <= maxLen)
27
+ return str;
28
+ return str.slice(0, maxLen) + '...';
29
+ }
30
+ function formatDiagnosticsForSlack(report) {
31
+ const lines = [];
32
+ if (report.diagnostics) {
33
+ const { stateSnapshots, navHistory, lastError } = report.diagnostics;
34
+ if (lastError) {
35
+ lines.push(`*Last Error:* ${lastError.message}`);
36
+ if (lastError.stack) {
37
+ lines.push(`\`${truncateString(lastError.stack.split('\n')[0] ?? '', 200)}\``);
38
+ }
39
+ }
40
+ if (navHistory.length > 0) {
41
+ const recentRoutes = navHistory.slice(-5);
42
+ lines.push(`*Nav History (last ${recentRoutes.length}):*`);
43
+ for (const entry of recentRoutes) {
44
+ lines.push(` ${entry.pathname}`);
45
+ }
46
+ }
47
+ if (stateSnapshots.length > 0) {
48
+ const recentStates = stateSnapshots.slice(-3);
49
+ lines.push(`*State Snapshots (last ${recentStates.length}):*`);
50
+ for (const snap of recentStates) {
51
+ lines.push(` _${snap.name}:_ ${truncateString(snap.state, 200)}${snap.truncated ? ' [TRUNCATED]' : ''}`);
52
+ }
53
+ }
54
+ }
55
+ return lines.join('\n');
56
+ }
57
+ function formatSlackMessage(report, screenshotUrl) {
58
+ const fields = [
59
+ `*Screen:* ${report.screen}`,
60
+ `*Device:* ${report.device.model}`,
61
+ `*OS:* ${report.device.os}`,
62
+ `*App Version:* ${report.device.appVersion}`,
63
+ `*Time:* ${report.timestamp}`,
64
+ ];
65
+ if (report.description) {
66
+ fields.unshift(`*Description:* ${report.description}`);
67
+ }
68
+ const metaEntries = Object.entries(report.metadata);
69
+ if (metaEntries.length > 0) {
70
+ fields.push(`*Metadata:* ${metaEntries.map(([k, v]) => `${k}=${v}`).join(', ')}`);
71
+ }
72
+ const blocks = [
73
+ {
74
+ type: 'header',
75
+ text: { type: 'plain_text', text: 'Bug Report', emoji: true },
76
+ },
77
+ {
78
+ type: 'section',
79
+ text: { type: 'mrkdwn', text: fields.join('\n') },
80
+ },
81
+ ];
82
+ const diagnosticsText = formatDiagnosticsForSlack(report);
83
+ if (diagnosticsText) {
84
+ blocks.push({
85
+ type: 'section',
86
+ text: { type: 'mrkdwn', text: diagnosticsText },
87
+ });
88
+ }
89
+ if (screenshotUrl) {
90
+ blocks.push({
91
+ type: 'image',
92
+ image_url: screenshotUrl,
93
+ alt_text: 'Bug report screenshot',
94
+ });
95
+ }
96
+ return { blocks };
97
+ }
98
+ function SlackIntegration(config) {
99
+ return {
100
+ name: 'slack',
101
+ async send(report) {
102
+ try {
103
+ let screenshotUrl = null;
104
+ const imageUri = report.annotatedScreenshot ?? report.screenshot;
105
+ if (imageUri && !config.imageUploadKey) {
106
+ console.warn('[expo-bug-report] Screenshot available but no imageUploadKey configured. ' +
107
+ 'Bug report will be sent without the screenshot image. ' +
108
+ 'Add imageUploadKey to SlackIntegration config to include screenshots.');
109
+ }
110
+ if (imageUri && config.imageUploadKey) {
111
+ screenshotUrl = await uploadImage(imageUri, config.imageUploadKey, config.imageUploadUrl ?? DEFAULT_IMGBB_URL);
112
+ }
113
+ const message = formatSlackMessage(report, screenshotUrl);
114
+ const response = await fetch(config.webhookUrl, {
115
+ method: 'POST',
116
+ headers: { 'Content-Type': 'application/json' },
117
+ body: JSON.stringify(message),
118
+ });
119
+ if (!response.ok) {
120
+ return {
121
+ success: false,
122
+ error: `Slack returned ${response.status}`,
123
+ };
124
+ }
125
+ return { success: true };
126
+ }
127
+ catch (error) {
128
+ return {
129
+ success: false,
130
+ error: error instanceof Error ? error.message : 'Slack send failed',
131
+ };
132
+ }
133
+ },
134
+ };
135
+ }
136
+ //# sourceMappingURL=slack.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"slack.js","sourceRoot":"","sources":["../../src/integrations/slack.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,53 @@
1
+ export interface StateSnapshot {
2
+ name: string;
3
+ state: string;
4
+ timestamp: string;
5
+ truncated: boolean;
6
+ }
7
+ export interface NavEntry {
8
+ pathname: string;
9
+ segments: string[];
10
+ timestamp: string;
11
+ }
12
+ export interface ErrorInfo {
13
+ message: string;
14
+ stack: string | null;
15
+ componentStack: string | null;
16
+ timestamp: string;
17
+ }
18
+ export interface Diagnostics {
19
+ stateSnapshots: StateSnapshot[];
20
+ navHistory: NavEntry[];
21
+ lastError: ErrorInfo | null;
22
+ }
23
+ export interface BugReport {
24
+ screenshot: string | null;
25
+ annotatedScreenshot: string | null;
26
+ description: string;
27
+ device: DeviceInfo;
28
+ screen: string;
29
+ timestamp: string;
30
+ metadata: Record<string, string>;
31
+ diagnostics?: Diagnostics;
32
+ }
33
+ export interface DeviceInfo {
34
+ model: string;
35
+ os: string;
36
+ appVersion: string;
37
+ screenSize: string;
38
+ locale: string;
39
+ installationId: string;
40
+ expoConfig: {
41
+ name: string;
42
+ slug: string;
43
+ } | null;
44
+ }
45
+ export interface SendResult {
46
+ success: boolean;
47
+ error?: string;
48
+ }
49
+ export interface Integration {
50
+ name: string;
51
+ send(report: BugReport): Promise<SendResult>;
52
+ }
53
+ //# sourceMappingURL=types.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../src/integrations/types.ts"],"names":[],"mappings":"AAAA,MAAM,WAAW,aAAa;IAC5B,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,EAAE,MAAM,CAAC;IACd,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,EAAE,OAAO,CAAC;CACpB;AAED,MAAM,WAAW,QAAQ;IACvB,QAAQ,EAAE,MAAM,CAAC;IACjB,QAAQ,EAAE,MAAM,EAAE,CAAC;IACnB,SAAS,EAAE,MAAM,CAAC;CACnB;AAED,MAAM,WAAW,SAAS;IACxB,OAAO,EAAE,MAAM,CAAC;IAChB,KAAK,EAAE,MAAM,GAAG,IAAI,CAAC;IACrB,cAAc,EAAE,MAAM,GAAG,IAAI,CAAC;IAC9B,SAAS,EAAE,MAAM,CAAC;CACnB;AAED,MAAM,WAAW,WAAW;IAC1B,cAAc,EAAE,aAAa,EAAE,CAAC;IAChC,UAAU,EAAE,QAAQ,EAAE,CAAC;IACvB,SAAS,EAAE,SAAS,GAAG,IAAI,CAAC;CAC7B;AAED,MAAM,WAAW,SAAS;IACxB,UAAU,EAAE,MAAM,GAAG,IAAI,CAAC;IAC1B,mBAAmB,EAAE,MAAM,GAAG,IAAI,CAAC;IACnC,WAAW,EAAE,MAAM,CAAC;IACpB,MAAM,EAAE,UAAU,CAAC;IACnB,MAAM,EAAE,MAAM,CAAC;IACf,SAAS,EAAE,MAAM,CAAC;IAClB,QAAQ,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;IACjC,WAAW,CAAC,EAAE,WAAW,CAAC;CAC3B;AAED,MAAM,WAAW,UAAU;IACzB,KAAK,EAAE,MAAM,CAAC;IACd,EAAE,EAAE,MAAM,CAAC;IACX,UAAU,EAAE,MAAM,CAAC;IACnB,UAAU,EAAE,MAAM,CAAC;IACnB,MAAM,EAAE,MAAM,CAAC;IACf,cAAc,EAAE,MAAM,CAAC;IACvB,UAAU,EAAE;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,IAAI,EAAE,MAAM,CAAA;KAAE,GAAG,IAAI,CAAC;CACnD;AAED,MAAM,WAAW,UAAU;IACzB,OAAO,EAAE,OAAO,CAAC;IACjB,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB;AAED,MAAM,WAAW,WAAW;IAC1B,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,CAAC,MAAM,EAAE,SAAS,GAAG,OAAO,CAAC,UAAU,CAAC,CAAC;CAC9C"}
@@ -0,0 +1,3 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/integrations/types.ts"],"names":[],"mappings":""}
@@ -0,0 +1,8 @@
1
+ import type { Integration } from './types';
2
+ interface WebhookConfig {
3
+ url: string;
4
+ headers?: Record<string, string>;
5
+ }
6
+ export declare function WebhookIntegration(config: WebhookConfig): Integration;
7
+ export {};
8
+ //# sourceMappingURL=webhook.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"webhook.d.ts","sourceRoot":"","sources":["../../src/integrations/webhook.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAa,WAAW,EAAc,MAAM,SAAS,CAAC;AAElE,UAAU,aAAa;IACrB,GAAG,EAAE,MAAM,CAAC;IACZ,OAAO,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;CAClC;AAED,wBAAgB,kBAAkB,CAAC,MAAM,EAAE,aAAa,GAAG,WAAW,CA0CrE"}
@@ -0,0 +1,45 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.WebhookIntegration = WebhookIntegration;
4
+ const fileToBase64_1 = require("../utils/fileToBase64");
5
+ function WebhookIntegration(config) {
6
+ return {
7
+ name: 'webhook',
8
+ async send(report) {
9
+ try {
10
+ const payload = {
11
+ ...report,
12
+ screenshotBase64: report.screenshot
13
+ ? await (0, fileToBase64_1.fileToBase64)(report.screenshot)
14
+ : null,
15
+ annotatedScreenshotBase64: report.annotatedScreenshot
16
+ ? await (0, fileToBase64_1.fileToBase64)(report.annotatedScreenshot)
17
+ : null,
18
+ diagnostics: report.diagnostics ?? null,
19
+ };
20
+ const response = await fetch(config.url, {
21
+ method: 'POST',
22
+ headers: {
23
+ 'Content-Type': 'application/json',
24
+ ...config.headers,
25
+ },
26
+ body: JSON.stringify(payload),
27
+ });
28
+ if (!response.ok) {
29
+ return {
30
+ success: false,
31
+ error: `Webhook returned ${response.status}`,
32
+ };
33
+ }
34
+ return { success: true };
35
+ }
36
+ catch (error) {
37
+ return {
38
+ success: false,
39
+ error: error instanceof Error ? error.message : 'Webhook send failed',
40
+ };
41
+ }
42
+ },
43
+ };
44
+ }
45
+ //# sourceMappingURL=webhook.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"webhook.js","sourceRoot":"","sources":["../../src/integrations/webhook.ts"],"names":[],"mappings":";;AAQA,gDA0CC;AAlDD,wDAAqD;AAQrD,SAAgB,kBAAkB,CAAC,MAAqB;IACtD,OAAO;QACL,IAAI,EAAE,SAAS;QACf,KAAK,CAAC,IAAI,CAAC,MAAiB;YAC1B,IAAI,CAAC;gBACH,MAAM,OAAO,GAA4B;oBACvC,GAAG,MAAM;oBACT,gBAAgB,EAAE,MAAM,CAAC,UAAU;wBACjC,CAAC,CAAC,MAAM,IAAA,2BAAY,EAAC,MAAM,CAAC,UAAU,CAAC;wBACvC,CAAC,CAAC,IAAI;oBACR,yBAAyB,EAAE,MAAM,CAAC,mBAAmB;wBACnD,CAAC,CAAC,MAAM,IAAA,2BAAY,EAAC,MAAM,CAAC,mBAAmB,CAAC;wBAChD,CAAC,CAAC,IAAI;oBACR,WAAW,EAAE,MAAM,CAAC,WAAW,IAAI,IAAI;iBACxC,CAAC;gBAEF,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,MAAM,CAAC,GAAG,EAAE;oBACvC,MAAM,EAAE,MAAM;oBACd,OAAO,EAAE;wBACP,cAAc,EAAE,kBAAkB;wBAClC,GAAG,MAAM,CAAC,OAAO;qBAClB;oBACD,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC;iBAC9B,CAAC,CAAC;gBAEH,IAAI,CAAC,QAAQ,CAAC,EAAE,EAAE,CAAC;oBACjB,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,KAAK,EAAE,oBAAoB,QAAQ,CAAC,MAAM,EAAE;qBAC7C,CAAC;gBACJ,CAAC;gBAED,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC;YAC3B,CAAC;YAAC,OAAO,KAAK,EAAE,CAAC;gBACf,OAAO;oBACL,OAAO,EAAE,KAAK;oBACd,KAAK,EACH,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,qBAAqB;iBACjE,CAAC;YACJ,CAAC;QACH,CAAC;KACF,CAAC;AACJ,CAAC"}
@@ -0,0 +1,2 @@
1
+ export declare function fileToBase64(uri: string): Promise<string>;
2
+ //# sourceMappingURL=fileToBase64.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"fileToBase64.d.ts","sourceRoot":"","sources":["../../src/utils/fileToBase64.ts"],"names":[],"mappings":"AAEA,wBAAsB,YAAY,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC,CAK/D"}
@@ -0,0 +1,11 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.fileToBase64 = fileToBase64;
4
+ const legacy_1 = require("expo-file-system/legacy");
5
+ async function fileToBase64(uri) {
6
+ const base64 = await (0, legacy_1.readAsStringAsync)(uri, {
7
+ encoding: legacy_1.EncodingType.Base64,
8
+ });
9
+ return base64;
10
+ }
11
+ //# sourceMappingURL=fileToBase64.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"fileToBase64.js","sourceRoot":"","sources":["../../src/utils/fileToBase64.ts"],"names":[],"mappings":";;AAEA,oCAKC;AAPD,oDAA0E;AAEnE,KAAK,UAAU,YAAY,CAAC,GAAW;IAC5C,MAAM,MAAM,GAAG,MAAM,IAAA,0BAAiB,EAAC,GAAG,EAAE;QAC1C,QAAQ,EAAE,qBAAY,CAAC,MAAM;KAC9B,CAAC,CAAC;IACH,OAAO,MAAM,CAAC;AAChB,CAAC"}