@frak-labs/core-sdk 0.1.0-beta.afa252b0 → 0.1.0-beta.c7e026e5

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 (143) hide show
  1. package/cdn/bundle.iife.js +14 -0
  2. package/dist/actions-CEEObPYc.js +1 -0
  3. package/dist/actions-DbQhWYx8.cjs +1 -0
  4. package/dist/actions.cjs +1 -1
  5. package/dist/actions.d.cts +3 -1481
  6. package/dist/actions.d.ts +3 -1481
  7. package/dist/actions.js +1 -1
  8. package/dist/bundle.cjs +1 -13
  9. package/dist/bundle.d.cts +6 -2087
  10. package/dist/bundle.d.ts +6 -2087
  11. package/dist/bundle.js +1 -13
  12. package/dist/index-7OZ39x1U.d.ts +195 -0
  13. package/dist/index-C6FxkWPC.d.cts +511 -0
  14. package/dist/index-UFX7xCg3.d.ts +351 -0
  15. package/dist/index-d8xS4ryI.d.ts +511 -0
  16. package/dist/index-p4FqSp8z.d.cts +351 -0
  17. package/dist/index-zDq-VlKx.d.cts +195 -0
  18. package/dist/index.cjs +1 -13
  19. package/dist/index.d.cts +4 -1387
  20. package/dist/index.d.ts +4 -1387
  21. package/dist/index.js +1 -13
  22. package/dist/interaction-DMJ3ZfaF.d.cts +45 -0
  23. package/dist/interaction-KX1h9a7V.d.ts +45 -0
  24. package/dist/interactions-DnfM3oe0.js +1 -0
  25. package/dist/interactions-EIXhNLf6.cjs +1 -0
  26. package/dist/interactions.cjs +1 -1
  27. package/dist/interactions.d.cts +2 -182
  28. package/dist/interactions.d.ts +2 -182
  29. package/dist/interactions.js +1 -1
  30. package/dist/openSso-D--Airj6.d.cts +1018 -0
  31. package/dist/openSso-DsKJ4y0j.d.ts +1018 -0
  32. package/dist/productTypes-BUkXJKZ7.cjs +1 -0
  33. package/dist/productTypes-CGb1MmBF.js +1 -0
  34. package/dist/src-B_xO0AR6.cjs +13 -0
  35. package/dist/src-D2d52OZa.js +13 -0
  36. package/dist/trackEvent-CHnYa85W.js +1 -0
  37. package/dist/trackEvent-GuQm_1Nm.cjs +1 -0
  38. package/package.json +23 -18
  39. package/src/actions/displayEmbeddedWallet.test.ts +194 -0
  40. package/src/actions/displayEmbeddedWallet.ts +20 -0
  41. package/src/actions/displayModal.test.ts +387 -0
  42. package/src/actions/displayModal.ts +131 -0
  43. package/src/actions/getProductInformation.test.ts +133 -0
  44. package/src/actions/getProductInformation.ts +14 -0
  45. package/src/actions/index.ts +29 -0
  46. package/src/actions/openSso.test.ts +407 -0
  47. package/src/actions/openSso.ts +116 -0
  48. package/src/actions/prepareSso.test.ts +223 -0
  49. package/src/actions/prepareSso.ts +48 -0
  50. package/src/actions/referral/processReferral.ts +230 -0
  51. package/src/actions/referral/referralInteraction.test.ts +153 -0
  52. package/src/actions/referral/referralInteraction.ts +57 -0
  53. package/src/actions/sendInteraction.test.ts +219 -0
  54. package/src/actions/sendInteraction.ts +32 -0
  55. package/src/actions/trackPurchaseStatus.test.ts +287 -0
  56. package/src/actions/trackPurchaseStatus.ts +53 -0
  57. package/src/actions/watchWalletStatus.test.ts +372 -0
  58. package/src/actions/watchWalletStatus.ts +94 -0
  59. package/src/actions/wrapper/modalBuilder.test.ts +253 -0
  60. package/src/actions/wrapper/modalBuilder.ts +212 -0
  61. package/src/actions/wrapper/sendTransaction.test.ts +164 -0
  62. package/src/actions/wrapper/sendTransaction.ts +62 -0
  63. package/src/actions/wrapper/siweAuthenticate.test.ts +290 -0
  64. package/src/actions/wrapper/siweAuthenticate.ts +94 -0
  65. package/src/bundle.ts +3 -0
  66. package/src/clients/DebugInfo.ts +182 -0
  67. package/src/clients/createIFrameFrakClient.ts +287 -0
  68. package/src/clients/index.ts +3 -0
  69. package/src/clients/setupClient.test.ts +343 -0
  70. package/src/clients/setupClient.ts +73 -0
  71. package/src/clients/transports/iframeLifecycleManager.test.ts +399 -0
  72. package/src/clients/transports/iframeLifecycleManager.ts +90 -0
  73. package/src/constants/interactionTypes.ts +44 -0
  74. package/src/constants/locales.ts +14 -0
  75. package/src/constants/productTypes.ts +33 -0
  76. package/src/index.ts +101 -0
  77. package/src/interactions/index.ts +5 -0
  78. package/src/interactions/pressEncoder.test.ts +215 -0
  79. package/src/interactions/pressEncoder.ts +53 -0
  80. package/src/interactions/purchaseEncoder.test.ts +291 -0
  81. package/src/interactions/purchaseEncoder.ts +99 -0
  82. package/src/interactions/referralEncoder.test.ts +170 -0
  83. package/src/interactions/referralEncoder.ts +47 -0
  84. package/src/interactions/retailEncoder.test.ts +107 -0
  85. package/src/interactions/retailEncoder.ts +37 -0
  86. package/src/interactions/webshopEncoder.test.ts +56 -0
  87. package/src/interactions/webshopEncoder.ts +30 -0
  88. package/src/types/client.ts +14 -0
  89. package/src/types/compression.ts +22 -0
  90. package/src/types/config.ts +111 -0
  91. package/src/types/context.ts +13 -0
  92. package/src/types/index.ts +71 -0
  93. package/src/types/lifecycle/client.ts +46 -0
  94. package/src/types/lifecycle/iframe.ts +35 -0
  95. package/src/types/lifecycle/index.ts +2 -0
  96. package/src/types/rpc/displayModal.ts +84 -0
  97. package/src/types/rpc/embedded/index.ts +68 -0
  98. package/src/types/rpc/embedded/loggedIn.ts +55 -0
  99. package/src/types/rpc/embedded/loggedOut.ts +28 -0
  100. package/src/types/rpc/interaction.ts +43 -0
  101. package/src/types/rpc/modal/final.ts +46 -0
  102. package/src/types/rpc/modal/generic.ts +46 -0
  103. package/src/types/rpc/modal/index.ts +20 -0
  104. package/src/types/rpc/modal/login.ts +32 -0
  105. package/src/types/rpc/modal/openSession.ts +25 -0
  106. package/src/types/rpc/modal/siweAuthenticate.ts +37 -0
  107. package/src/types/rpc/modal/transaction.ts +33 -0
  108. package/src/types/rpc/productInformation.ts +59 -0
  109. package/src/types/rpc/sso.ts +80 -0
  110. package/src/types/rpc/walletStatus.ts +35 -0
  111. package/src/types/rpc.ts +158 -0
  112. package/src/types/transport.ts +34 -0
  113. package/src/utils/FrakContext.test.ts +407 -0
  114. package/src/utils/FrakContext.ts +158 -0
  115. package/src/utils/compression/b64.test.ts +181 -0
  116. package/src/utils/compression/b64.ts +29 -0
  117. package/src/utils/compression/compress.test.ts +123 -0
  118. package/src/utils/compression/compress.ts +11 -0
  119. package/src/utils/compression/decompress.test.ts +145 -0
  120. package/src/utils/compression/decompress.ts +11 -0
  121. package/src/utils/compression/index.ts +3 -0
  122. package/src/utils/computeProductId.test.ts +80 -0
  123. package/src/utils/computeProductId.ts +11 -0
  124. package/src/utils/constants.test.ts +23 -0
  125. package/src/utils/constants.ts +4 -0
  126. package/src/utils/formatAmount.test.ts +113 -0
  127. package/src/utils/formatAmount.ts +18 -0
  128. package/src/utils/getCurrencyAmountKey.test.ts +44 -0
  129. package/src/utils/getCurrencyAmountKey.ts +15 -0
  130. package/src/utils/getSupportedCurrency.test.ts +51 -0
  131. package/src/utils/getSupportedCurrency.ts +14 -0
  132. package/src/utils/getSupportedLocale.test.ts +64 -0
  133. package/src/utils/getSupportedLocale.ts +16 -0
  134. package/src/utils/iframeHelper.test.ts +450 -0
  135. package/src/utils/iframeHelper.ts +143 -0
  136. package/src/utils/index.ts +21 -0
  137. package/src/utils/sso.test.ts +361 -0
  138. package/src/utils/sso.ts +119 -0
  139. package/src/utils/ssoUrlListener.ts +60 -0
  140. package/src/utils/trackEvent.test.ts +162 -0
  141. package/src/utils/trackEvent.ts +26 -0
  142. package/cdn/bundle.js +0 -19
  143. package/cdn/bundle.js.LICENSE.txt +0 -10
@@ -0,0 +1,290 @@
1
+ import { beforeEach, vi } from "vitest";
2
+
3
+ vi.mock("../displayModal", () => ({
4
+ displayModal: vi.fn(),
5
+ }));
6
+
7
+ vi.mock("viem/siwe", () => ({
8
+ generateSiweNonce: vi.fn(() => "mock-nonce-123456"),
9
+ }));
10
+
11
+ import type { Address, Hex } from "viem";
12
+ import { describe, expect, it } from "../../../tests/vitest-fixtures";
13
+ import type { FrakClient } from "../../types";
14
+ import { siweAuthenticate } from "./siweAuthenticate";
15
+
16
+ describe("siweAuthenticate", () => {
17
+ const mockClient = {
18
+ config: {
19
+ domain: "example.com",
20
+ metadata: {
21
+ name: "Test App",
22
+ },
23
+ },
24
+ request: vi.fn(),
25
+ } as unknown as FrakClient;
26
+
27
+ const mockClientWithoutDomain = {
28
+ config: {
29
+ metadata: {
30
+ name: "Test App",
31
+ },
32
+ },
33
+ request: vi.fn(),
34
+ } as unknown as FrakClient;
35
+
36
+ beforeEach(() => {
37
+ vi.stubGlobal("window", {
38
+ location: {
39
+ host: "window.example.com",
40
+ },
41
+ });
42
+ });
43
+
44
+ describe("basic usage", () => {
45
+ it("should call displayModal with SIWE params", async () => {
46
+ const { displayModal } = await import("../displayModal");
47
+
48
+ const mockResponse = {
49
+ login: {
50
+ wallet: "0x1234567890123456789012345678901234567890" as Address,
51
+ },
52
+ siweAuthenticate: {
53
+ message: "Sign in to Test App",
54
+ signature: "0xsignature" as Hex,
55
+ },
56
+ };
57
+ vi.mocked(displayModal).mockResolvedValue(mockResponse as any);
58
+
59
+ await siweAuthenticate(mockClient, {});
60
+
61
+ expect(displayModal).toHaveBeenCalledWith(mockClient, {
62
+ metadata: undefined,
63
+ steps: {
64
+ login: {},
65
+ siweAuthenticate: {
66
+ siwe: expect.objectContaining({
67
+ domain: "example.com",
68
+ nonce: "mock-nonce-123456",
69
+ uri: "https://example.com",
70
+ version: "1",
71
+ }),
72
+ },
73
+ },
74
+ });
75
+ });
76
+
77
+ it("should return SIWE authentication result", async () => {
78
+ const { displayModal } = await import("../displayModal");
79
+
80
+ const mockSiweResult = {
81
+ message: "I confirm that I want to use my Frak wallet",
82
+ signature:
83
+ "0xsig1234567890123456789012345678901234567890123456789012345678901234" as Hex,
84
+ };
85
+ vi.mocked(displayModal).mockResolvedValue({
86
+ login: {},
87
+ siweAuthenticate: mockSiweResult,
88
+ } as any);
89
+
90
+ const result = await siweAuthenticate(mockClient, {});
91
+
92
+ expect(result).toEqual(mockSiweResult);
93
+ });
94
+ });
95
+
96
+ describe("SIWE parameter handling", () => {
97
+ it("should use default statement when not provided", async () => {
98
+ const { displayModal } = await import("../displayModal");
99
+
100
+ vi.mocked(displayModal).mockResolvedValue({
101
+ login: {},
102
+ siweAuthenticate: { message: "", signature: "0x" as Hex },
103
+ } as any);
104
+
105
+ await siweAuthenticate(mockClient, {});
106
+
107
+ expect(displayModal).toHaveBeenCalledWith(
108
+ mockClient,
109
+ expect.objectContaining({
110
+ steps: expect.objectContaining({
111
+ siweAuthenticate: expect.objectContaining({
112
+ siwe: expect.objectContaining({
113
+ statement:
114
+ "I confirm that I want to use my Frak wallet on: Test App",
115
+ }),
116
+ }),
117
+ }),
118
+ })
119
+ );
120
+ });
121
+
122
+ it("should use custom statement when provided", async () => {
123
+ const { displayModal } = await import("../displayModal");
124
+
125
+ vi.mocked(displayModal).mockResolvedValue({
126
+ login: {},
127
+ siweAuthenticate: { message: "", signature: "0x" as Hex },
128
+ } as any);
129
+
130
+ await siweAuthenticate(mockClient, {
131
+ siwe: { statement: "Custom sign in message" },
132
+ });
133
+
134
+ expect(displayModal).toHaveBeenCalledWith(
135
+ mockClient,
136
+ expect.objectContaining({
137
+ steps: expect.objectContaining({
138
+ siweAuthenticate: expect.objectContaining({
139
+ siwe: expect.objectContaining({
140
+ statement: "Custom sign in message",
141
+ }),
142
+ }),
143
+ }),
144
+ })
145
+ );
146
+ });
147
+
148
+ it("should use custom nonce when provided", async () => {
149
+ const { displayModal } = await import("../displayModal");
150
+
151
+ vi.mocked(displayModal).mockResolvedValue({
152
+ login: {},
153
+ siweAuthenticate: { message: "", signature: "0x" as Hex },
154
+ } as any);
155
+
156
+ await siweAuthenticate(mockClient, {
157
+ siwe: { nonce: "custom-nonce" },
158
+ });
159
+
160
+ expect(displayModal).toHaveBeenCalledWith(
161
+ mockClient,
162
+ expect.objectContaining({
163
+ steps: expect.objectContaining({
164
+ siweAuthenticate: expect.objectContaining({
165
+ siwe: expect.objectContaining({
166
+ nonce: "custom-nonce",
167
+ }),
168
+ }),
169
+ }),
170
+ })
171
+ );
172
+ });
173
+
174
+ it("should use window.location.host when domain not in config", async () => {
175
+ const { displayModal } = await import("../displayModal");
176
+
177
+ vi.mocked(displayModal).mockResolvedValue({
178
+ login: {},
179
+ siweAuthenticate: { message: "", signature: "0x" as Hex },
180
+ } as any);
181
+
182
+ await siweAuthenticate(mockClientWithoutDomain, {});
183
+
184
+ expect(displayModal).toHaveBeenCalledWith(
185
+ mockClientWithoutDomain,
186
+ expect.objectContaining({
187
+ steps: expect.objectContaining({
188
+ siweAuthenticate: expect.objectContaining({
189
+ siwe: expect.objectContaining({
190
+ domain: "window.example.com",
191
+ uri: "https://window.example.com",
192
+ }),
193
+ }),
194
+ }),
195
+ })
196
+ );
197
+ });
198
+
199
+ it("should use custom uri when provided", async () => {
200
+ const { displayModal } = await import("../displayModal");
201
+
202
+ vi.mocked(displayModal).mockResolvedValue({
203
+ login: {},
204
+ siweAuthenticate: { message: "", signature: "0x" as Hex },
205
+ } as any);
206
+
207
+ await siweAuthenticate(mockClient, {
208
+ siwe: { uri: "https://custom-uri.com" },
209
+ });
210
+
211
+ expect(displayModal).toHaveBeenCalledWith(
212
+ mockClient,
213
+ expect.objectContaining({
214
+ steps: expect.objectContaining({
215
+ siweAuthenticate: expect.objectContaining({
216
+ siwe: expect.objectContaining({
217
+ uri: "https://custom-uri.com",
218
+ }),
219
+ }),
220
+ }),
221
+ })
222
+ );
223
+ });
224
+
225
+ it("should use custom version when provided", async () => {
226
+ const { displayModal } = await import("../displayModal");
227
+
228
+ vi.mocked(displayModal).mockResolvedValue({
229
+ login: {},
230
+ siweAuthenticate: { message: "", signature: "0x" as Hex },
231
+ } as any);
232
+
233
+ await siweAuthenticate(mockClient, {
234
+ siwe: { version: "2" as any },
235
+ });
236
+
237
+ expect(displayModal).toHaveBeenCalledWith(
238
+ mockClient,
239
+ expect.objectContaining({
240
+ steps: expect.objectContaining({
241
+ siweAuthenticate: expect.objectContaining({
242
+ siwe: expect.objectContaining({
243
+ version: "2",
244
+ }),
245
+ }),
246
+ }),
247
+ })
248
+ );
249
+ });
250
+ });
251
+
252
+ describe("with metadata", () => {
253
+ it("should pass metadata to displayModal", async () => {
254
+ const { displayModal } = await import("../displayModal");
255
+
256
+ vi.mocked(displayModal).mockResolvedValue({
257
+ login: {},
258
+ siweAuthenticate: { message: "", signature: "0x" as Hex },
259
+ } as any);
260
+
261
+ const metadata = {
262
+ header: {
263
+ title: "Sign In",
264
+ },
265
+ context: "Sign in to access your account",
266
+ };
267
+
268
+ await siweAuthenticate(mockClient, { metadata });
269
+
270
+ expect(displayModal).toHaveBeenCalledWith(mockClient, {
271
+ metadata,
272
+ steps: expect.any(Object),
273
+ });
274
+ });
275
+ });
276
+
277
+ describe("error handling", () => {
278
+ it("should propagate errors from displayModal", async () => {
279
+ const { displayModal } = await import("../displayModal");
280
+
281
+ vi.mocked(displayModal).mockRejectedValue(
282
+ new Error("SIWE authentication rejected")
283
+ );
284
+
285
+ await expect(siweAuthenticate(mockClient, {})).rejects.toThrow(
286
+ "SIWE authentication rejected"
287
+ );
288
+ });
289
+ });
290
+ });
@@ -0,0 +1,94 @@
1
+ import { generateSiweNonce } from "viem/siwe";
2
+ import type {
3
+ FrakClient,
4
+ ModalRpcMetadata,
5
+ SiweAuthenticateReturnType,
6
+ SiweAuthenticationParams,
7
+ } from "../../types";
8
+ import { displayModal } from "../displayModal";
9
+
10
+ /**
11
+ * Parameter used to directly show a modal used to authenticate with SIWE
12
+ * @inline
13
+ */
14
+ export type SiweAuthenticateModalParams = {
15
+ /**
16
+ * Partial SIWE params, since we can rebuild them from the SDK if they are empty
17
+ *
18
+ * If no parameters provider, some fields will be recomputed from the current configuration and environment.
19
+ * - `statement` will be set to a default value
20
+ * - `nonce` will be generated
21
+ * - `uri` will be set to the current domain
22
+ * - `version` will be set to "1"
23
+ * - `domain` will be set to the current window domain
24
+ *
25
+ * @default {}
26
+ */
27
+ siwe?: Partial<SiweAuthenticationParams>;
28
+ /**
29
+ * Custom metadata to be passed to the modal
30
+ */
31
+ metadata?: ModalRpcMetadata;
32
+ };
33
+
34
+ /**
35
+ * Function used to launch a siwe authentication
36
+ * @param client - The current Frak Client
37
+ * @param args - The parameters
38
+ * @returns The SIWE authentication result (message + signature) in a promise
39
+ *
40
+ * @description This function will display a modal to the user with the provided SIWE parameters and metadata.
41
+ *
42
+ * @example
43
+ * import { siweAuthenticate } from "@frak-labs/core-sdk/actions";
44
+ * import { parseSiweMessage } from "viem/siwe";
45
+ *
46
+ * const { signature, message } = await siweAuthenticate(frakConfig, {
47
+ * siwe: {
48
+ * statement: "Sign in to My App",
49
+ * domain: "my-app.com",
50
+ * expirationTimeTimestamp: Date.now() + 1000 * 60 * 5,
51
+ * },
52
+ * metadata: {
53
+ * header: {
54
+ * title: "Sign in",
55
+ * },
56
+ * context: "Sign in to My App",
57
+ * },
58
+ * });
59
+ * console.log("Parsed final message:", parseSiweMessage(message));
60
+ * console.log("Siwe signature:", signature);
61
+ */
62
+ export async function siweAuthenticate(
63
+ client: FrakClient,
64
+ { siwe, metadata }: SiweAuthenticateModalParams
65
+ ): Promise<SiweAuthenticateReturnType> {
66
+ const effectiveDomain = client.config?.domain ?? window.location.host;
67
+ const realStatement =
68
+ siwe?.statement ??
69
+ `I confirm that I want to use my Frak wallet on: ${client.config.metadata.name}`;
70
+
71
+ // Fill up the siwe request params
72
+ const builtSiwe: SiweAuthenticationParams = {
73
+ ...siwe,
74
+ statement: realStatement,
75
+ nonce: siwe?.nonce ?? generateSiweNonce(),
76
+ uri: siwe?.uri ?? `https://${effectiveDomain}`,
77
+ version: siwe?.version ?? "1",
78
+ domain: effectiveDomain,
79
+ };
80
+
81
+ // Trigger a modal with login options
82
+ const result = await displayModal(client, {
83
+ metadata,
84
+ steps: {
85
+ login: {},
86
+ siweAuthenticate: {
87
+ siwe: builtSiwe,
88
+ },
89
+ },
90
+ });
91
+
92
+ // Return the SIWE result only
93
+ return result.siweAuthenticate;
94
+ }
package/src/bundle.ts ADDED
@@ -0,0 +1,3 @@
1
+ export * from "./actions";
2
+ export * from "./index";
3
+ export * from "./interactions";
@@ -0,0 +1,182 @@
1
+ import {
2
+ FrakRpcError,
3
+ type RpcMessage,
4
+ type RpcResponse,
5
+ } from "@frak-labs/frame-connector";
6
+ import type { FrakWalletSdkConfig } from "../types";
7
+
8
+ type IframeStatus = {
9
+ loading: boolean;
10
+ url: string | null;
11
+ readyState: number;
12
+ contentWindow: boolean;
13
+ isConnected: boolean;
14
+ };
15
+
16
+ type DebugInfo = {
17
+ timestamp: string;
18
+ encodedUrl: string;
19
+ navigatorInfo: string;
20
+ encodedConfig: string;
21
+ iframeStatus: string;
22
+ lastRequest: string;
23
+ lastResponse: string;
24
+ clientStatus: string;
25
+ error: string;
26
+ };
27
+
28
+ type NavigatorInfo = {
29
+ userAgent: string;
30
+ language: string;
31
+ onLine: boolean;
32
+ screenWidth: number;
33
+ screenHeight: number;
34
+ pixelRatio: number;
35
+ };
36
+
37
+ /** @ignore */
38
+ export class DebugInfoGatherer {
39
+ private config?: FrakWalletSdkConfig;
40
+ private iframe?: HTMLIFrameElement;
41
+ private isSetupDone = false;
42
+ private lastResponse: null | {
43
+ message: RpcMessage;
44
+ response: RpcResponse;
45
+ timestamp: number;
46
+ } = null;
47
+ private lastRequest: null | {
48
+ event: RpcMessage;
49
+ timestamp: number;
50
+ } = null;
51
+
52
+ constructor(config?: FrakWalletSdkConfig, iframe?: HTMLIFrameElement) {
53
+ this.config = config;
54
+ this.iframe = iframe;
55
+ this.lastRequest = null;
56
+ this.lastResponse = null;
57
+ }
58
+
59
+ // Update communication logs
60
+ public setLastResponse(message: RpcMessage, response: RpcResponse) {
61
+ this.lastResponse = {
62
+ message,
63
+ response,
64
+ timestamp: Date.now(),
65
+ };
66
+ }
67
+ public setLastRequest(event: RpcMessage) {
68
+ this.lastRequest = { event, timestamp: Date.now() };
69
+ }
70
+
71
+ // Update connection status
72
+ public updateSetupStatus(status: boolean) {
73
+ this.isSetupDone = status;
74
+ }
75
+
76
+ private base64Encode(data: object): string {
77
+ try {
78
+ return btoa(JSON.stringify(data));
79
+ } catch (err) {
80
+ console.warn("Failed to encode debug data", err);
81
+ return btoa("Failed to encode data");
82
+ }
83
+ }
84
+
85
+ /**
86
+ * Extract information from the iframe status
87
+ */
88
+ private getIframeStatus(): IframeStatus | null {
89
+ if (!this.iframe) {
90
+ return null;
91
+ }
92
+ return {
93
+ loading: this.iframe.hasAttribute("loading"),
94
+ url: this.iframe.src,
95
+ readyState: this.iframe.contentDocument?.readyState
96
+ ? this.iframe.contentDocument.readyState === "complete"
97
+ ? 1
98
+ : 0
99
+ : -1,
100
+ contentWindow: !!this.iframe.contentWindow,
101
+ isConnected: this.iframe.isConnected,
102
+ };
103
+ }
104
+
105
+ private getNavigatorInfo(): NavigatorInfo | null {
106
+ if (!navigator) {
107
+ return null;
108
+ }
109
+ return {
110
+ userAgent: navigator.userAgent,
111
+ language: navigator.language,
112
+ onLine: navigator.onLine,
113
+ screenWidth: window.screen.width,
114
+ screenHeight: window.screen.height,
115
+ pixelRatio: window.devicePixelRatio,
116
+ };
117
+ }
118
+
119
+ private gatherDebugInfo(error: Error | unknown): DebugInfo {
120
+ const iframeStatus = this.getIframeStatus();
121
+ const navigatorInfo = this.getNavigatorInfo();
122
+
123
+ // Format the error in a readable format
124
+ let formattedError = "Unknown";
125
+ if (error instanceof FrakRpcError) {
126
+ formattedError = `FrakRpcError: ${error.code} '${error.message}'`;
127
+ } else if (error instanceof Error) {
128
+ formattedError = error.message;
129
+ } else if (typeof error === "string") {
130
+ formattedError = error;
131
+ }
132
+
133
+ // Craft the debug info
134
+ const debugInfo: DebugInfo = {
135
+ timestamp: new Date().toISOString(),
136
+ encodedUrl: btoa(window.location.href),
137
+ encodedConfig: this.config
138
+ ? this.base64Encode(this.config)
139
+ : "no-config",
140
+ navigatorInfo: navigatorInfo
141
+ ? this.base64Encode(navigatorInfo)
142
+ : "no-navigator",
143
+ iframeStatus: iframeStatus
144
+ ? this.base64Encode(iframeStatus)
145
+ : "not-iframe",
146
+ lastRequest: this.lastRequest
147
+ ? this.base64Encode(this.lastRequest)
148
+ : "No Frak request logged",
149
+ lastResponse: this.lastResponse
150
+ ? this.base64Encode(this.lastResponse)
151
+ : "No Frak response logged",
152
+ clientStatus: this.isSetupDone ? "setup" : "not-setup",
153
+ error: formattedError,
154
+ };
155
+
156
+ return debugInfo;
157
+ }
158
+
159
+ public static empty(): DebugInfoGatherer {
160
+ return new DebugInfoGatherer();
161
+ }
162
+
163
+ /**
164
+ * Format Frak debug information
165
+ */
166
+ public formatDebugInfo(error: Error | unknown | string): string {
167
+ const debugInfo = this.gatherDebugInfo(error);
168
+ return `
169
+ Debug Information:
170
+ -----------------
171
+ Timestamp: ${debugInfo.timestamp}
172
+ URL: ${debugInfo.encodedUrl}
173
+ Config: ${debugInfo.encodedConfig}
174
+ Navigator Info: ${debugInfo.navigatorInfo}
175
+ IFrame Status: ${debugInfo.iframeStatus}
176
+ Last Request: ${debugInfo.lastRequest}
177
+ Last Response: ${debugInfo.lastResponse}
178
+ Client Status: ${debugInfo.clientStatus}
179
+ Error: ${debugInfo.error}
180
+ `.trim();
181
+ }
182
+ }