@frak-labs/core-sdk 0.1.0 → 0.1.1-beta.4dfea079

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 (130) hide show
  1. package/README.md +58 -0
  2. package/cdn/bundle.js +3 -8
  3. package/dist/actions.cjs +1 -1
  4. package/dist/actions.d.cts +3 -1400
  5. package/dist/actions.d.ts +3 -1400
  6. package/dist/actions.js +1 -1
  7. package/dist/bundle.cjs +1 -13
  8. package/dist/bundle.d.cts +4 -1927
  9. package/dist/bundle.d.ts +4 -1927
  10. package/dist/bundle.js +1 -13
  11. package/dist/computeLegacyProductId-CscYhyUi.d.cts +525 -0
  12. package/dist/computeLegacyProductId-WbD1gXV9.d.ts +525 -0
  13. package/dist/index.cjs +1 -13
  14. package/dist/index.d.cts +3 -1269
  15. package/dist/index.d.ts +3 -1269
  16. package/dist/index.js +1 -13
  17. package/dist/openSso-CC1-loUk.d.cts +1019 -0
  18. package/dist/openSso-tkqaDQLV.d.ts +1019 -0
  19. package/dist/setupClient-BjIbK6XJ.cjs +13 -0
  20. package/dist/setupClient-D_HId3e2.js +13 -0
  21. package/dist/siweAuthenticate-B_Z2OZmj.cjs +1 -0
  22. package/dist/siweAuthenticate-CQ4OfPuA.js +1 -0
  23. package/dist/siweAuthenticate-CR4Dpji6.d.cts +467 -0
  24. package/dist/siweAuthenticate-udoruuy9.d.ts +467 -0
  25. package/dist/trackEvent-CGIryq5h.cjs +1 -0
  26. package/dist/trackEvent-YfUh4jrx.js +1 -0
  27. package/package.json +24 -30
  28. package/src/actions/displayEmbeddedWallet.test.ts +194 -0
  29. package/src/actions/displayEmbeddedWallet.ts +20 -0
  30. package/src/actions/displayModal.test.ts +388 -0
  31. package/src/actions/displayModal.ts +120 -0
  32. package/src/actions/getMerchantInformation.test.ts +116 -0
  33. package/src/actions/getMerchantInformation.ts +9 -0
  34. package/src/actions/index.ts +29 -0
  35. package/src/actions/openSso.ts +116 -0
  36. package/src/actions/prepareSso.test.ts +223 -0
  37. package/src/actions/prepareSso.ts +48 -0
  38. package/src/actions/referral/processReferral.test.ts +248 -0
  39. package/src/actions/referral/processReferral.ts +232 -0
  40. package/src/actions/referral/referralInteraction.test.ts +147 -0
  41. package/src/actions/referral/referralInteraction.ts +52 -0
  42. package/src/actions/sendInteraction.ts +24 -0
  43. package/src/actions/trackPurchaseStatus.test.ts +287 -0
  44. package/src/actions/trackPurchaseStatus.ts +56 -0
  45. package/src/actions/watchWalletStatus.test.ts +372 -0
  46. package/src/actions/watchWalletStatus.ts +93 -0
  47. package/src/actions/wrapper/modalBuilder.test.ts +239 -0
  48. package/src/actions/wrapper/modalBuilder.ts +203 -0
  49. package/src/actions/wrapper/sendTransaction.test.ts +164 -0
  50. package/src/actions/wrapper/sendTransaction.ts +62 -0
  51. package/src/actions/wrapper/siweAuthenticate.test.ts +290 -0
  52. package/src/actions/wrapper/siweAuthenticate.ts +94 -0
  53. package/src/bundle.ts +2 -0
  54. package/src/clients/DebugInfo.test.ts +418 -0
  55. package/src/clients/DebugInfo.ts +182 -0
  56. package/src/clients/createIFrameFrakClient.ts +289 -0
  57. package/src/clients/index.ts +3 -0
  58. package/src/clients/setupClient.test.ts +343 -0
  59. package/src/clients/setupClient.ts +73 -0
  60. package/src/clients/transports/iframeLifecycleManager.test.ts +558 -0
  61. package/src/clients/transports/iframeLifecycleManager.ts +174 -0
  62. package/src/constants/interactionTypes.ts +15 -0
  63. package/src/constants/locales.ts +14 -0
  64. package/src/index.ts +110 -0
  65. package/src/types/client.ts +14 -0
  66. package/src/types/compression.ts +22 -0
  67. package/src/types/config.ts +117 -0
  68. package/src/types/context.ts +13 -0
  69. package/src/types/index.ts +75 -0
  70. package/src/types/lifecycle/client.ts +69 -0
  71. package/src/types/lifecycle/iframe.ts +41 -0
  72. package/src/types/lifecycle/index.ts +2 -0
  73. package/src/types/rpc/displayModal.ts +82 -0
  74. package/src/types/rpc/embedded/index.ts +68 -0
  75. package/src/types/rpc/embedded/loggedIn.ts +55 -0
  76. package/src/types/rpc/embedded/loggedOut.ts +28 -0
  77. package/src/types/rpc/interaction.ts +30 -0
  78. package/src/types/rpc/merchantInformation.ts +77 -0
  79. package/src/types/rpc/modal/final.ts +46 -0
  80. package/src/types/rpc/modal/generic.ts +46 -0
  81. package/src/types/rpc/modal/index.ts +16 -0
  82. package/src/types/rpc/modal/login.ts +36 -0
  83. package/src/types/rpc/modal/siweAuthenticate.ts +37 -0
  84. package/src/types/rpc/modal/transaction.ts +33 -0
  85. package/src/types/rpc/sso.ts +80 -0
  86. package/src/types/rpc/walletStatus.ts +29 -0
  87. package/src/types/rpc.ts +146 -0
  88. package/src/types/tracking.ts +60 -0
  89. package/src/types/transport.ts +34 -0
  90. package/src/utils/FrakContext.test.ts +407 -0
  91. package/src/utils/FrakContext.ts +158 -0
  92. package/src/utils/backendUrl.test.ts +83 -0
  93. package/src/utils/backendUrl.ts +62 -0
  94. package/src/utils/clientId.test.ts +41 -0
  95. package/src/utils/clientId.ts +40 -0
  96. package/src/utils/compression/b64.test.ts +181 -0
  97. package/src/utils/compression/b64.ts +29 -0
  98. package/src/utils/compression/compress.test.ts +123 -0
  99. package/src/utils/compression/compress.ts +11 -0
  100. package/src/utils/compression/decompress.test.ts +149 -0
  101. package/src/utils/compression/decompress.ts +11 -0
  102. package/src/utils/compression/index.ts +3 -0
  103. package/src/utils/computeLegacyProductId.ts +11 -0
  104. package/src/utils/constants.test.ts +23 -0
  105. package/src/utils/constants.ts +14 -0
  106. package/src/utils/deepLinkWithFallback.test.ts +243 -0
  107. package/src/utils/deepLinkWithFallback.ts +97 -0
  108. package/src/utils/formatAmount.test.ts +113 -0
  109. package/src/utils/formatAmount.ts +18 -0
  110. package/src/utils/getCurrencyAmountKey.test.ts +44 -0
  111. package/src/utils/getCurrencyAmountKey.ts +15 -0
  112. package/src/utils/getSupportedCurrency.test.ts +51 -0
  113. package/src/utils/getSupportedCurrency.ts +14 -0
  114. package/src/utils/getSupportedLocale.test.ts +64 -0
  115. package/src/utils/getSupportedLocale.ts +16 -0
  116. package/src/utils/iframeHelper.test.ts +450 -0
  117. package/src/utils/iframeHelper.ts +147 -0
  118. package/src/utils/index.ts +36 -0
  119. package/src/utils/merchantId.test.ts +564 -0
  120. package/src/utils/merchantId.ts +122 -0
  121. package/src/utils/sso.ts +126 -0
  122. package/src/utils/ssoUrlListener.test.ts +252 -0
  123. package/src/utils/ssoUrlListener.ts +60 -0
  124. package/src/utils/trackEvent.test.ts +180 -0
  125. package/src/utils/trackEvent.ts +31 -0
  126. package/cdn/bundle.js.LICENSE.txt +0 -10
  127. package/dist/interactions.cjs +0 -1
  128. package/dist/interactions.d.cts +0 -182
  129. package/dist/interactions.d.ts +0 -182
  130. package/dist/interactions.js +0 -1
@@ -0,0 +1,289 @@
1
+ import {
2
+ createRpcClient,
3
+ FrakRpcError,
4
+ type RpcClient,
5
+ RpcErrorCodes,
6
+ } from "@frak-labs/frame-connector";
7
+ import { OpenPanel } from "@openpanel/web";
8
+ import type { FrakLifecycleEvent } from "../types";
9
+ import type { FrakClient } from "../types/client";
10
+ import type { FrakWalletSdkConfig } from "../types/config";
11
+ import type { IFrameRpcSchema } from "../types/rpc";
12
+ import { BACKUP_KEY } from "../utils/constants";
13
+ import { setupSsoUrlListener } from "../utils/ssoUrlListener";
14
+
15
+ import { DebugInfoGatherer } from "./DebugInfo";
16
+ import {
17
+ createIFrameLifecycleManager,
18
+ type IframeLifecycleManager,
19
+ } from "./transports/iframeLifecycleManager";
20
+
21
+ type SdkRpcClient = RpcClient<IFrameRpcSchema, FrakLifecycleEvent>;
22
+
23
+ /**
24
+ * Create a new iframe Frak client
25
+ * @param args
26
+ * @param args.config - The configuration to use for the Frak Wallet SDK
27
+ * @param args.iframe - The iframe to use for the communication
28
+ * @returns The created Frak Client
29
+ *
30
+ * @example
31
+ * const frakConfig: FrakWalletSdkConfig = {
32
+ * metadata: {
33
+ * name: "My app title",
34
+ * },
35
+ * }
36
+ * const iframe = await createIframe({ config: frakConfig });
37
+ * const client = createIFrameFrakClient({ config: frakConfig, iframe });
38
+ */
39
+ export function createIFrameFrakClient({
40
+ config,
41
+ iframe,
42
+ }: {
43
+ config: FrakWalletSdkConfig;
44
+ iframe: HTMLIFrameElement;
45
+ }): FrakClient {
46
+ const frakWalletUrl = config?.walletUrl ?? "https://wallet.frak.id";
47
+
48
+ // Create lifecycle manager
49
+ const lifecycleManager = createIFrameLifecycleManager({
50
+ iframe,
51
+ targetOrigin: frakWalletUrl,
52
+ configDomain: config.domain,
53
+ });
54
+
55
+ // Create our debug info gatherer
56
+ const debugInfo = new DebugInfoGatherer(config, iframe);
57
+
58
+ // Validate iframe
59
+ if (!iframe.contentWindow) {
60
+ throw new FrakRpcError(
61
+ RpcErrorCodes.configError,
62
+ "The iframe does not have a content window"
63
+ );
64
+ }
65
+
66
+ // Create RPC client with middleware and lifecycle handlers
67
+ const rpcClient = createRpcClient<IFrameRpcSchema, FrakLifecycleEvent>({
68
+ emittingTransport: iframe.contentWindow,
69
+ listeningTransport: window,
70
+ targetOrigin: frakWalletUrl,
71
+ middleware: [
72
+ // Ensure we are connected before sending request
73
+ {
74
+ async onRequest(_message, ctx) {
75
+ // Ensure the iframe is connected
76
+ const isConnected = await lifecycleManager.isConnected;
77
+ if (!isConnected) {
78
+ throw new FrakRpcError(
79
+ RpcErrorCodes.clientNotConnected,
80
+ "The iframe provider isn't connected yet"
81
+ );
82
+ }
83
+ return ctx;
84
+ },
85
+ },
86
+ // Save debug info
87
+ {
88
+ onRequest(message, ctx) {
89
+ debugInfo.setLastRequest(message);
90
+ return ctx;
91
+ },
92
+ onResponse(message, response) {
93
+ debugInfo.setLastResponse(message, response);
94
+ return response;
95
+ },
96
+ },
97
+ ],
98
+ // Add lifecycle handlers to process iframe lifecycle events
99
+ lifecycleHandlers: {
100
+ iframeLifecycle: async (event, _context) => {
101
+ // Delegate to lifecycle manager (cast for type compatibility)
102
+ await lifecycleManager.handleEvent(event);
103
+ },
104
+ },
105
+ });
106
+
107
+ // Setup heartbeat
108
+ const stopHeartbeat = setupHeartbeat(rpcClient, lifecycleManager);
109
+
110
+ // Build our destroy function
111
+ const destroy = async () => {
112
+ // Stop heartbeat
113
+ stopHeartbeat();
114
+ // Cleanup the RPC client
115
+ rpcClient.cleanup();
116
+ // Remove the iframe
117
+ iframe.remove();
118
+ };
119
+
120
+ // Init open panel
121
+ let openPanel: OpenPanel | undefined;
122
+ if (
123
+ process.env.OPEN_PANEL_API_URL &&
124
+ process.env.OPEN_PANEL_SDK_CLIENT_ID
125
+ ) {
126
+ console.log("[Frak SDK] Initializing OpenPanel");
127
+ openPanel = new OpenPanel({
128
+ apiUrl: process.env.OPEN_PANEL_API_URL,
129
+ clientId: process.env.OPEN_PANEL_SDK_CLIENT_ID,
130
+ trackScreenViews: true,
131
+ trackOutgoingLinks: true,
132
+ trackAttributes: false,
133
+ // We use a filter to ensure we got the open panel instance initialized
134
+ // A bit hacky, but this way we are sure that we got everything needed for the first event ever sent
135
+ filter: ({ type, payload }) => {
136
+ if (type !== "track") return true;
137
+ if (!payload?.properties) return true;
138
+
139
+ // Check if we we got the properties once initialized
140
+ if (!("sdkVersion" in payload.properties)) {
141
+ payload.properties = {
142
+ ...payload.properties,
143
+ sdkVersion: process.env.SDK_VERSION,
144
+ };
145
+ }
146
+
147
+ return true;
148
+ },
149
+ });
150
+ openPanel.setGlobalProperties({
151
+ sdkVersion: process.env.SDK_VERSION,
152
+ });
153
+ openPanel.init();
154
+ }
155
+
156
+ // Perform the post connection setup
157
+ const waitForSetup = postConnectionSetup({
158
+ config,
159
+ rpcClient,
160
+ lifecycleManager,
161
+ }).then(() => debugInfo.updateSetupStatus(true));
162
+
163
+ return {
164
+ config,
165
+ debugInfo,
166
+ waitForConnection: lifecycleManager.isConnected,
167
+ waitForSetup,
168
+ request: rpcClient.request,
169
+ listenerRequest: rpcClient.listen,
170
+ destroy,
171
+ openPanel,
172
+ };
173
+ }
174
+
175
+ /**
176
+ * Setup the heartbeat
177
+ * @param rpcClient - RPC client to send lifecycle events
178
+ * @param lifecycleManager - Lifecycle manager to track connection
179
+ */
180
+ function setupHeartbeat(
181
+ rpcClient: SdkRpcClient,
182
+ lifecycleManager: IframeLifecycleManager
183
+ ) {
184
+ const HEARTBEAT_INTERVAL = 1_000; // Send heartbeat every 100ms until we are connected
185
+ const HEARTBEAT_TIMEOUT = 30_000; // 30 seconds timeout
186
+ let heartbeatInterval: NodeJS.Timeout;
187
+ let timeoutId: NodeJS.Timeout;
188
+
189
+ const sendHeartbeat = () =>
190
+ rpcClient.sendLifecycle({
191
+ clientLifecycle: "heartbeat",
192
+ });
193
+
194
+ // Start sending heartbeats
195
+ async function startHeartbeat() {
196
+ sendHeartbeat(); // Send initial heartbeat
197
+ heartbeatInterval = setInterval(sendHeartbeat, HEARTBEAT_INTERVAL);
198
+
199
+ // Set up timeout
200
+ timeoutId = setTimeout(() => {
201
+ stopHeartbeat();
202
+ console.log("Heartbeat timeout: connection failed");
203
+ }, HEARTBEAT_TIMEOUT);
204
+
205
+ // Once connected, stop it
206
+ await lifecycleManager.isConnected;
207
+
208
+ // We are now connected, stop the heartbeat
209
+ stopHeartbeat();
210
+ }
211
+
212
+ // Stop sending heartbeats
213
+ function stopHeartbeat() {
214
+ if (heartbeatInterval) {
215
+ clearInterval(heartbeatInterval);
216
+ }
217
+ if (timeoutId) {
218
+ clearTimeout(timeoutId);
219
+ }
220
+ }
221
+
222
+ startHeartbeat();
223
+
224
+ // Return cleanup function
225
+ return stopHeartbeat;
226
+ }
227
+
228
+ /**
229
+ * Perform the post connection setup
230
+ * @param config - SDK configuration
231
+ * @param rpcClient - RPC client to send lifecycle events
232
+ * @param lifecycleManager - Lifecycle manager to track connection
233
+ */
234
+ async function postConnectionSetup({
235
+ config,
236
+ rpcClient,
237
+ lifecycleManager,
238
+ }: {
239
+ config: FrakWalletSdkConfig;
240
+ rpcClient: SdkRpcClient;
241
+ lifecycleManager: IframeLifecycleManager;
242
+ }): Promise<void> {
243
+ // Wait for the handler to be connected
244
+ await lifecycleManager.isConnected;
245
+
246
+ // Setup SSO URL listener to detect and forward SSO redirects
247
+ // This checks for ?sso= parameter and forwards compressed data to iframe
248
+ setupSsoUrlListener(rpcClient, lifecycleManager.isConnected);
249
+
250
+ // Push raw CSS if needed
251
+ async function pushCss() {
252
+ const cssLink = config.customizations?.css;
253
+ if (!cssLink) return;
254
+
255
+ const message = {
256
+ clientLifecycle: "modal-css" as const,
257
+ data: { cssLink },
258
+ };
259
+ rpcClient.sendLifecycle(message);
260
+ }
261
+
262
+ // Push i18n if needed
263
+ async function pushI18n() {
264
+ const i18n = config.customizations?.i18n;
265
+ if (!i18n) return;
266
+
267
+ const message = {
268
+ clientLifecycle: "modal-i18n" as const,
269
+ data: { i18n },
270
+ };
271
+ rpcClient.sendLifecycle(message);
272
+ }
273
+
274
+ // Push local backup if needed
275
+ async function pushBackup() {
276
+ if (typeof window === "undefined") return;
277
+
278
+ const backup = window.localStorage.getItem(BACKUP_KEY);
279
+ if (!backup) return;
280
+
281
+ const message = {
282
+ clientLifecycle: "restore-backup" as const,
283
+ data: { backup },
284
+ };
285
+ rpcClient.sendLifecycle(message);
286
+ }
287
+
288
+ await Promise.allSettled([pushCss(), pushI18n(), pushBackup()]);
289
+ }
@@ -0,0 +1,3 @@
1
+ export { createIFrameFrakClient } from "./createIFrameFrakClient";
2
+ export { DebugInfoGatherer } from "./DebugInfo";
3
+ export { setupClient } from "./setupClient";
@@ -0,0 +1,343 @@
1
+ import { beforeEach, describe, expect, test, vi } from "vitest";
2
+
3
+ // Mock dependencies
4
+ vi.mock("../utils", () => ({
5
+ createIframe: vi.fn(),
6
+ getSupportedCurrency: vi.fn((currency) => currency || "eur"),
7
+ }));
8
+
9
+ vi.mock("./createIFrameFrakClient", () => ({
10
+ createIFrameFrakClient: vi.fn(),
11
+ }));
12
+
13
+ describe("setupClient", () => {
14
+ beforeEach(() => {
15
+ vi.clearAllMocks();
16
+ });
17
+
18
+ describe("client setup", () => {
19
+ test("should create iframe with prepared config", async () => {
20
+ const { setupClient } = await import("./setupClient");
21
+ const { createIframe } = await import("../utils");
22
+ const { createIFrameFrakClient } = await import(
23
+ "./createIFrameFrakClient"
24
+ );
25
+
26
+ const mockIframe = document.createElement("iframe");
27
+ const mockClient = {
28
+ waitForSetup: Promise.resolve(),
29
+ waitForConnection: Promise.resolve(true),
30
+ };
31
+
32
+ vi.mocked(createIframe).mockResolvedValue(mockIframe);
33
+ vi.mocked(createIFrameFrakClient).mockReturnValue(
34
+ mockClient as any
35
+ );
36
+
37
+ const config = {
38
+ metadata: {
39
+ name: "Test App",
40
+ },
41
+ };
42
+
43
+ await setupClient({ config });
44
+
45
+ expect(createIframe).toHaveBeenCalledWith({
46
+ config: expect.objectContaining({
47
+ metadata: expect.objectContaining({
48
+ name: "Test App",
49
+ currency: "eur",
50
+ }),
51
+ }),
52
+ });
53
+ });
54
+
55
+ test("should preserve custom currency in config", async () => {
56
+ const { setupClient } = await import("./setupClient");
57
+ const { createIframe } = await import("../utils");
58
+ const { createIFrameFrakClient } = await import(
59
+ "./createIFrameFrakClient"
60
+ );
61
+ const { getSupportedCurrency } = await import("../utils");
62
+
63
+ const mockIframe = document.createElement("iframe");
64
+ const mockClient = {
65
+ waitForSetup: Promise.resolve(),
66
+ waitForConnection: Promise.resolve(true),
67
+ };
68
+
69
+ vi.mocked(getSupportedCurrency).mockReturnValue("usd");
70
+ vi.mocked(createIframe).mockResolvedValue(mockIframe);
71
+ vi.mocked(createIFrameFrakClient).mockReturnValue(
72
+ mockClient as any
73
+ );
74
+
75
+ const config = {
76
+ metadata: {
77
+ name: "Test App",
78
+ currency: "usd" as const,
79
+ },
80
+ };
81
+
82
+ await setupClient({ config });
83
+
84
+ expect(getSupportedCurrency).toHaveBeenCalledWith("usd");
85
+ expect(createIframe).toHaveBeenCalledWith({
86
+ config: expect.objectContaining({
87
+ metadata: expect.objectContaining({
88
+ currency: "usd",
89
+ }),
90
+ }),
91
+ });
92
+ });
93
+
94
+ test("should return undefined when iframe creation fails", async () => {
95
+ const { setupClient } = await import("./setupClient");
96
+ const { createIframe } = await import("../utils");
97
+
98
+ vi.mocked(createIframe).mockResolvedValue(undefined);
99
+
100
+ const config = {
101
+ metadata: {
102
+ name: "Test App",
103
+ },
104
+ };
105
+
106
+ const result = await setupClient({ config });
107
+
108
+ expect(result).toBeUndefined();
109
+ });
110
+
111
+ test("should create FrakClient with iframe", async () => {
112
+ const { setupClient } = await import("./setupClient");
113
+ const { createIframe } = await import("../utils");
114
+ const { createIFrameFrakClient } = await import(
115
+ "./createIFrameFrakClient"
116
+ );
117
+
118
+ const mockIframe = document.createElement("iframe");
119
+ const mockClient = {
120
+ waitForSetup: Promise.resolve(),
121
+ waitForConnection: Promise.resolve(true),
122
+ };
123
+
124
+ vi.mocked(createIframe).mockResolvedValue(mockIframe);
125
+ vi.mocked(createIFrameFrakClient).mockReturnValue(
126
+ mockClient as any
127
+ );
128
+
129
+ const config = {
130
+ metadata: {
131
+ name: "Test App",
132
+ },
133
+ };
134
+
135
+ await setupClient({ config });
136
+
137
+ expect(createIFrameFrakClient).toHaveBeenCalledWith({
138
+ config: expect.objectContaining({
139
+ metadata: expect.objectContaining({
140
+ name: "Test App",
141
+ }),
142
+ }),
143
+ iframe: mockIframe,
144
+ });
145
+ });
146
+
147
+ test("should wait for client setup", async () => {
148
+ const { setupClient } = await import("./setupClient");
149
+ const { createIframe } = await import("../utils");
150
+ const { createIFrameFrakClient } = await import(
151
+ "./createIFrameFrakClient"
152
+ );
153
+
154
+ const mockIframe = document.createElement("iframe");
155
+ const setupPromise = Promise.resolve();
156
+ const mockClient = {
157
+ waitForSetup: setupPromise,
158
+ waitForConnection: Promise.resolve(true),
159
+ };
160
+
161
+ vi.mocked(createIframe).mockResolvedValue(mockIframe);
162
+ vi.mocked(createIFrameFrakClient).mockReturnValue(
163
+ mockClient as any
164
+ );
165
+
166
+ const config = {
167
+ metadata: {
168
+ name: "Test App",
169
+ },
170
+ };
171
+
172
+ await setupClient({ config });
173
+
174
+ // Verify setup was awaited
175
+ await expect(setupPromise).resolves.toBeUndefined();
176
+ });
177
+
178
+ test("should wait for connection", async () => {
179
+ const { setupClient } = await import("./setupClient");
180
+ const { createIframe } = await import("../utils");
181
+ const { createIFrameFrakClient } = await import(
182
+ "./createIFrameFrakClient"
183
+ );
184
+
185
+ const mockIframe = document.createElement("iframe");
186
+ const connectionPromise = Promise.resolve(true);
187
+ const mockClient = {
188
+ waitForSetup: Promise.resolve(),
189
+ waitForConnection: connectionPromise,
190
+ };
191
+
192
+ vi.mocked(createIframe).mockResolvedValue(mockIframe);
193
+ vi.mocked(createIFrameFrakClient).mockReturnValue(
194
+ mockClient as any
195
+ );
196
+
197
+ const config = {
198
+ metadata: {
199
+ name: "Test App",
200
+ },
201
+ };
202
+
203
+ await setupClient({ config });
204
+
205
+ // Verify connection was awaited
206
+ await expect(connectionPromise).resolves.toBe(true);
207
+ });
208
+
209
+ test("should return undefined when connection fails", async () => {
210
+ const { setupClient } = await import("./setupClient");
211
+ const { createIframe } = await import("../utils");
212
+ const { createIFrameFrakClient } = await import(
213
+ "./createIFrameFrakClient"
214
+ );
215
+
216
+ const mockIframe = document.createElement("iframe");
217
+ const mockClient = {
218
+ waitForSetup: Promise.resolve(),
219
+ waitForConnection: Promise.resolve(false),
220
+ };
221
+
222
+ vi.mocked(createIframe).mockResolvedValue(mockIframe);
223
+ vi.mocked(createIFrameFrakClient).mockReturnValue(
224
+ mockClient as any
225
+ );
226
+
227
+ const config = {
228
+ metadata: {
229
+ name: "Test App",
230
+ },
231
+ };
232
+
233
+ const result = await setupClient({ config });
234
+
235
+ expect(result).toBeUndefined();
236
+ });
237
+
238
+ test("should return client when setup successful", async () => {
239
+ const { setupClient } = await import("./setupClient");
240
+ const { createIframe } = await import("../utils");
241
+ const { createIFrameFrakClient } = await import(
242
+ "./createIFrameFrakClient"
243
+ );
244
+
245
+ const mockIframe = document.createElement("iframe");
246
+ const mockClient = {
247
+ waitForSetup: Promise.resolve(),
248
+ waitForConnection: Promise.resolve(true),
249
+ request: vi.fn(),
250
+ };
251
+
252
+ vi.mocked(createIframe).mockResolvedValue(mockIframe);
253
+ vi.mocked(createIFrameFrakClient).mockReturnValue(
254
+ mockClient as any
255
+ );
256
+
257
+ const config = {
258
+ metadata: {
259
+ name: "Test App",
260
+ },
261
+ };
262
+
263
+ const result = await setupClient({ config });
264
+
265
+ expect(result).toBe(mockClient);
266
+ });
267
+ });
268
+
269
+ describe("config preparation", () => {
270
+ test("should use default currency when none provided", async () => {
271
+ const { setupClient } = await import("./setupClient");
272
+ const { createIframe } = await import("../utils");
273
+ const { createIFrameFrakClient } = await import(
274
+ "./createIFrameFrakClient"
275
+ );
276
+ const { getSupportedCurrency } = await import("../utils");
277
+
278
+ const mockIframe = document.createElement("iframe");
279
+ const mockClient = {
280
+ waitForSetup: Promise.resolve(),
281
+ waitForConnection: Promise.resolve(true),
282
+ };
283
+
284
+ vi.mocked(getSupportedCurrency).mockReturnValue("eur");
285
+ vi.mocked(createIframe).mockResolvedValue(mockIframe);
286
+ vi.mocked(createIFrameFrakClient).mockReturnValue(
287
+ mockClient as any
288
+ );
289
+
290
+ const config = {
291
+ metadata: {
292
+ name: "Test App",
293
+ },
294
+ };
295
+
296
+ await setupClient({ config });
297
+
298
+ expect(getSupportedCurrency).toHaveBeenCalledWith(undefined);
299
+ });
300
+
301
+ test("should merge metadata with prepared config", async () => {
302
+ const { setupClient } = await import("./setupClient");
303
+ const { createIframe } = await import("../utils");
304
+ const { createIFrameFrakClient } = await import(
305
+ "./createIFrameFrakClient"
306
+ );
307
+
308
+ const mockIframe = document.createElement("iframe");
309
+ const mockClient = {
310
+ waitForSetup: Promise.resolve(),
311
+ waitForConnection: Promise.resolve(true),
312
+ };
313
+
314
+ vi.mocked(createIframe).mockResolvedValue(mockIframe);
315
+ vi.mocked(createIFrameFrakClient).mockReturnValue(
316
+ mockClient as any
317
+ );
318
+
319
+ const config = {
320
+ metadata: {
321
+ name: "Test App",
322
+ css: {
323
+ primaryColor: "#ff0000",
324
+ },
325
+ },
326
+ };
327
+
328
+ await setupClient({ config });
329
+
330
+ expect(createIframe).toHaveBeenCalledWith({
331
+ config: expect.objectContaining({
332
+ metadata: expect.objectContaining({
333
+ name: "Test App",
334
+ css: {
335
+ primaryColor: "#ff0000",
336
+ },
337
+ currency: "eur",
338
+ }),
339
+ }),
340
+ });
341
+ });
342
+ });
343
+ });
@@ -0,0 +1,73 @@
1
+ import { createIFrameFrakClient } from "../clients";
2
+ import type { FrakClient, FrakWalletSdkConfig } from "../types";
3
+ import { createIframe, getSupportedCurrency } from "../utils";
4
+
5
+ /**
6
+ * Directly setup the Frak client with an iframe
7
+ * Return when the FrakClient is ready (setup and communication estbalished with the wallet)
8
+ *
9
+ * @param config - The configuration to use for the Frak Wallet SDK
10
+ * @returns a Promise with the Frak Client
11
+ *
12
+ * @example
13
+ * const frakConfig: FrakWalletSdkConfig = {
14
+ * metadata: {
15
+ * name: "My app title",
16
+ * },
17
+ * }
18
+ * const client = await setupClient({ config: frakConfig });
19
+ */
20
+ export async function setupClient({
21
+ config,
22
+ }: {
23
+ config: FrakWalletSdkConfig;
24
+ }): Promise<FrakClient | undefined> {
25
+ // Prepare the config
26
+ const preparedConfig = prepareConfig(config);
27
+
28
+ // Create our iframe
29
+ const iframe = await createIframe({
30
+ config: preparedConfig,
31
+ });
32
+
33
+ if (!iframe) {
34
+ console.error("Failed to create iframe");
35
+ return;
36
+ }
37
+
38
+ // Create our client
39
+ const client = createIFrameFrakClient({
40
+ config: preparedConfig,
41
+ iframe,
42
+ });
43
+
44
+ // Wait for the client to be all setup
45
+ await client.waitForSetup;
46
+
47
+ // Wait for the connection to be established
48
+ const waitForConnection = await client.waitForConnection;
49
+ if (!waitForConnection) {
50
+ console.error("Failed to connect to client");
51
+ return;
52
+ }
53
+
54
+ return client;
55
+ }
56
+
57
+ /**
58
+ * Prepare the config for the Frak Client
59
+ * @param config - The configuration to use for the Frak Wallet SDK
60
+ * @returns The prepared configuration with the supported currency
61
+ */
62
+ function prepareConfig(config: FrakWalletSdkConfig): FrakWalletSdkConfig {
63
+ // Get the supported currency (e.g. "eur")
64
+ const supportedCurrency = getSupportedCurrency(config.metadata?.currency);
65
+
66
+ return {
67
+ ...config,
68
+ metadata: {
69
+ ...config.metadata,
70
+ currency: supportedCurrency,
71
+ },
72
+ };
73
+ }