@solana-mobile/mobile-wallet-adapter-protocol 2.2.1 → 2.2.2-hotfix.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/src/transact.ts DELETED
@@ -1,593 +0,0 @@
1
- import { fromUint8Array, toUint8Array } from './base64Utils.js';
2
- import createHelloReq from './createHelloReq.js';
3
- import createMobileWalletProxy from './createMobileWalletProxy.js';
4
- import { SEQUENCE_NUMBER_BYTES } from './createSequenceNumberVector.js';
5
- import { ENCODED_PUBLIC_KEY_LENGTH_BYTES } from './encryptedMessage.js';
6
- import {
7
- SolanaMobileWalletAdapterError,
8
- SolanaMobileWalletAdapterErrorCode,
9
- SolanaMobileWalletAdapterProtocolError,
10
- } from './errors.js';
11
- import generateAssociationKeypair from './generateAssociationKeypair.js';
12
- import generateECDHKeypair from './generateECDHKeypair.js';
13
- import { getRemoteAssociateAndroidIntentURL } from './getAssociateAndroidIntentURL.js';
14
- import { decryptJsonRpcMessage, encryptJsonRpcMessage } from './jsonRpcMessage.js';
15
- import parseHelloRsp, { SharedSecret } from './parseHelloRsp.js';
16
- import parseSessionProps from './parseSessionProps.js';
17
- import { startSession } from './startSession.js';
18
- import {
19
- AssociationKeypair,
20
- MobileWallet,
21
- RemoteMobileWallet,
22
- RemoteScenario,
23
- RemoteWalletAssociationConfig,
24
- SessionProperties,
25
- WalletAssociationConfig
26
- } from './types.js';
27
-
28
- const WEBSOCKET_CONNECTION_CONFIG = {
29
- /**
30
- * 300 milliseconds is a generally accepted threshold for what someone
31
- * would consider an acceptable response time for a user interface
32
- * after having performed a low-attention tapping task. We set the initial
33
- * interval at which we wait for the wallet to set up the websocket at
34
- * half this, as per the Nyquist frequency, with a progressive backoff
35
- * sequence from there. The total wait time is 30s, which allows for the
36
- * user to be presented with a disambiguation dialog, select a wallet, and
37
- * for the wallet app to subsequently start.
38
- */
39
- retryDelayScheduleMs: [150, 150, 200, 500, 500, 750, 750, 1000],
40
- timeoutMs: 30000,
41
- } as const;
42
- const WEBSOCKET_PROTOCOL_BINARY = 'com.solana.mobilewalletadapter.v1';
43
- const WEBSOCKET_PROTOCOL_BASE64 = 'com.solana.mobilewalletadapter.v1.base64';
44
- type PROTOCOL_ENCODING = 'binary' | 'base64';
45
-
46
- type JsonResponsePromises<T> = Record<
47
- number,
48
- Readonly<{ resolve: (value?: T | PromiseLike<T>) => void; reject: (reason?: unknown) => void }>
49
- >;
50
-
51
- type State =
52
- | { __type: 'connected'; sharedSecret: SharedSecret; sessionProperties: SessionProperties }
53
- | { __type: 'connecting'; associationKeypair: AssociationKeypair }
54
- | { __type: 'disconnected' }
55
- | { __type: 'hello_req_sent'; associationPublicKey: CryptoKey; ecdhPrivateKey: CryptoKey };
56
-
57
- type RemoteState = State | { __type: 'reflector_id_received'; reflectorId: ArrayBuffer };
58
-
59
- function assertSecureContext() {
60
- if (typeof window === 'undefined' || window.isSecureContext !== true) {
61
- throw new SolanaMobileWalletAdapterError(
62
- SolanaMobileWalletAdapterErrorCode.ERROR_SECURE_CONTEXT_REQUIRED,
63
- 'The mobile wallet adapter protocol must be used in a secure context (`https`).',
64
- );
65
- }
66
- }
67
-
68
- function assertSecureEndpointSpecificURI(walletUriBase: string) {
69
- let url: URL;
70
- try {
71
- url = new URL(walletUriBase);
72
- } catch {
73
- throw new SolanaMobileWalletAdapterError(
74
- SolanaMobileWalletAdapterErrorCode.ERROR_FORBIDDEN_WALLET_BASE_URL,
75
- 'Invalid base URL supplied by wallet',
76
- );
77
- }
78
- if (url.protocol !== 'https:') {
79
- throw new SolanaMobileWalletAdapterError(
80
- SolanaMobileWalletAdapterErrorCode.ERROR_FORBIDDEN_WALLET_BASE_URL,
81
- 'Base URLs supplied by wallets must be valid `https` URLs',
82
- );
83
- }
84
- }
85
-
86
- function getSequenceNumberFromByteArray(byteArray: ArrayBuffer): number {
87
- const view = new DataView(byteArray);
88
- return view.getUint32(0, /* littleEndian */ false);
89
- }
90
-
91
- function decodeVarLong(byteArray: ArrayBuffer): { value: number, offset: number} {
92
- var bytes = new Uint8Array(byteArray),
93
- l = byteArray.byteLength,
94
- limit = 10,
95
- value = 0,
96
- offset = 0,
97
- b;
98
- do {
99
- if (offset >= l || offset > limit) throw new RangeError('Failed to decode varint');
100
- b = bytes[offset++];
101
- value |= (b & 0x7F) << (7 * offset);
102
- } while (b >= 0x80);
103
-
104
- return { value, offset };
105
- }
106
-
107
- function getReflectorIdFromByteArray(byteArray: ArrayBuffer): Uint8Array {
108
- let { value: length, offset } = decodeVarLong(byteArray);
109
- return new Uint8Array(byteArray.slice(offset, offset + length));
110
- }
111
-
112
- export async function transact<TReturn>(
113
- callback: (wallet: MobileWallet) => TReturn,
114
- config?: WalletAssociationConfig,
115
- ): Promise<TReturn> {
116
- assertSecureContext();
117
- const associationKeypair = await generateAssociationKeypair();
118
- const sessionPort = await startSession(associationKeypair.publicKey, config?.baseUri);
119
- const websocketURL = `ws://localhost:${sessionPort}/solana-wallet`;
120
- let connectionStartTime: number;
121
- const getNextRetryDelayMs = (() => {
122
- const schedule = [...WEBSOCKET_CONNECTION_CONFIG.retryDelayScheduleMs];
123
- return () => (schedule.length > 1 ? (schedule.shift() as number) : schedule[0]);
124
- })();
125
- let nextJsonRpcMessageId = 1;
126
- let lastKnownInboundSequenceNumber = 0;
127
- let state: State = { __type: 'disconnected' };
128
- return new Promise((resolve, reject) => {
129
- let socket: WebSocket;
130
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
131
- const jsonRpcResponsePromises: JsonResponsePromises<any> = {};
132
- const handleOpen = async () => {
133
- if (state.__type !== 'connecting') {
134
- console.warn(
135
- 'Expected adapter state to be `connecting` at the moment the websocket opens. ' +
136
- `Got \`${state.__type}\`.`,
137
- );
138
- return;
139
- }
140
- socket.removeEventListener('open', handleOpen);
141
-
142
- // previous versions of this library and walletlib incorrectly implemented the MWA session
143
- // establishment protocol for local connections. The dapp is supposed to wait for the
144
- // APP_PING message before sending the HELLO_REQ. Instead, the dapp was sending the HELLO_REQ
145
- // immediately upon connection to the websocket server regardless of wether or not an
146
- // APP_PING was sent by the wallet/websocket server. We must continue to support this behavior
147
- // in case the user is using a wallet that has not updated their walletlib implementation.
148
- const { associationKeypair } = state;
149
- const ecdhKeypair = await generateECDHKeypair();
150
- socket.send(await createHelloReq(ecdhKeypair.publicKey, associationKeypair.privateKey));
151
- state = {
152
- __type: 'hello_req_sent',
153
- associationPublicKey: associationKeypair.publicKey,
154
- ecdhPrivateKey: ecdhKeypair.privateKey,
155
- };
156
- };
157
- const handleClose = (evt: CloseEvent) => {
158
- if (evt.wasClean) {
159
- state = { __type: 'disconnected' };
160
- } else {
161
- reject(
162
- new SolanaMobileWalletAdapterError(
163
- SolanaMobileWalletAdapterErrorCode.ERROR_SESSION_CLOSED,
164
- `The wallet session dropped unexpectedly (${evt.code}: ${evt.reason}).`,
165
- { closeEvent: evt },
166
- ),
167
- );
168
- }
169
- disposeSocket();
170
- };
171
- const handleError = async (_evt: Event) => {
172
- disposeSocket();
173
- if (Date.now() - connectionStartTime >= WEBSOCKET_CONNECTION_CONFIG.timeoutMs) {
174
- reject(
175
- new SolanaMobileWalletAdapterError(
176
- SolanaMobileWalletAdapterErrorCode.ERROR_SESSION_TIMEOUT,
177
- `Failed to connect to the wallet websocket at ${websocketURL}.`,
178
- ),
179
- );
180
- } else {
181
- await new Promise((resolve) => {
182
- const retryDelayMs = getNextRetryDelayMs();
183
- retryWaitTimeoutId = window.setTimeout(resolve, retryDelayMs);
184
- });
185
- attemptSocketConnection();
186
- }
187
- };
188
- const handleMessage = async (evt: MessageEvent<Blob>) => {
189
- const responseBuffer = await evt.data.arrayBuffer();
190
- switch (state.__type) {
191
- case 'connecting':
192
- if (responseBuffer.byteLength !== 0) {
193
- throw new Error('Encountered unexpected message while connecting');
194
- }
195
- const ecdhKeypair = await generateECDHKeypair();
196
- socket.send(await createHelloReq(ecdhKeypair.publicKey, associationKeypair.privateKey));
197
- state = {
198
- __type: 'hello_req_sent',
199
- associationPublicKey: associationKeypair.publicKey,
200
- ecdhPrivateKey: ecdhKeypair.privateKey,
201
- };
202
- break;
203
- case 'connected':
204
- try {
205
- const sequenceNumberVector = responseBuffer.slice(0, SEQUENCE_NUMBER_BYTES);
206
- const sequenceNumber = getSequenceNumberFromByteArray(sequenceNumberVector);
207
- if (sequenceNumber !== (lastKnownInboundSequenceNumber + 1)) {
208
- throw new Error('Encrypted message has invalid sequence number');
209
- }
210
- lastKnownInboundSequenceNumber = sequenceNumber;
211
- const jsonRpcMessage = await decryptJsonRpcMessage(responseBuffer, state.sharedSecret);
212
- const responsePromise = jsonRpcResponsePromises[jsonRpcMessage.id];
213
- delete jsonRpcResponsePromises[jsonRpcMessage.id];
214
- responsePromise.resolve(jsonRpcMessage.result);
215
- } catch (e) {
216
- if (e instanceof SolanaMobileWalletAdapterProtocolError) {
217
- const responsePromise = jsonRpcResponsePromises[e.jsonRpcMessageId];
218
- delete jsonRpcResponsePromises[e.jsonRpcMessageId];
219
- responsePromise.reject(e);
220
- } else {
221
- throw e;
222
- }
223
- }
224
- break;
225
- case 'hello_req_sent': {
226
- // if we receive an APP_PING message (empty message), resend the HELLO_REQ (see above)
227
- if (responseBuffer.byteLength === 0) {
228
- const ecdhKeypair = await generateECDHKeypair();
229
- socket.send(await createHelloReq(ecdhKeypair.publicKey, associationKeypair.privateKey));
230
- state = {
231
- __type: 'hello_req_sent',
232
- associationPublicKey: associationKeypair.publicKey,
233
- ecdhPrivateKey: ecdhKeypair.privateKey,
234
- };
235
- break;
236
- }
237
- const sharedSecret = await parseHelloRsp(
238
- responseBuffer,
239
- state.associationPublicKey,
240
- state.ecdhPrivateKey,
241
- );
242
- const sessionPropertiesBuffer = responseBuffer.slice(ENCODED_PUBLIC_KEY_LENGTH_BYTES);
243
- const sessionProperties = sessionPropertiesBuffer.byteLength !== 0
244
- ? await (async () => {
245
- const sequenceNumberVector = sessionPropertiesBuffer.slice(0, SEQUENCE_NUMBER_BYTES);
246
- const sequenceNumber = getSequenceNumberFromByteArray(sequenceNumberVector);
247
- if (sequenceNumber !== (lastKnownInboundSequenceNumber + 1)) {
248
- throw new Error('Encrypted message has invalid sequence number');
249
- }
250
- lastKnownInboundSequenceNumber = sequenceNumber;
251
- return parseSessionProps(sessionPropertiesBuffer, sharedSecret);
252
- })() : <SessionProperties> { protocol_version: 'legacy' };
253
- state = { __type: 'connected', sharedSecret, sessionProperties };
254
- const wallet = createMobileWalletProxy(sessionProperties.protocol_version,
255
- async (method, params) => {
256
- const id = nextJsonRpcMessageId++;
257
- socket.send(
258
- await encryptJsonRpcMessage(
259
- {
260
- id,
261
- jsonrpc: '2.0' as const,
262
- method,
263
- params: params ?? {},
264
- },
265
- sharedSecret,
266
- ),
267
- );
268
- return new Promise((resolve, reject) => {
269
- jsonRpcResponsePromises[id] = {
270
- resolve(result) {
271
- switch (method) {
272
- case 'authorize':
273
- case 'reauthorize': {
274
- const { wallet_uri_base } = result as Awaited<
275
- ReturnType<MobileWallet['authorize' | 'reauthorize']>
276
- >;
277
- if (wallet_uri_base != null) {
278
- try {
279
- assertSecureEndpointSpecificURI(wallet_uri_base);
280
- } catch (e) {
281
- reject(e);
282
- return;
283
- }
284
- }
285
- break;
286
- }
287
- }
288
- resolve(result);
289
- },
290
- reject,
291
- };
292
- });
293
- })
294
- try {
295
- resolve(await callback(wallet));
296
- } catch (e) {
297
- reject(e);
298
- } finally {
299
- disposeSocket();
300
- socket.close();
301
- }
302
- break;
303
- }
304
- }
305
- };
306
- let disposeSocket: () => void;
307
- let retryWaitTimeoutId: number;
308
- const attemptSocketConnection = () => {
309
- if (disposeSocket) {
310
- disposeSocket();
311
- }
312
- state = { __type: 'connecting', associationKeypair };
313
- if (connectionStartTime === undefined) {
314
- connectionStartTime = Date.now();
315
- }
316
- socket = new WebSocket(websocketURL, [WEBSOCKET_PROTOCOL_BINARY]);
317
- socket.addEventListener('open', handleOpen);
318
- socket.addEventListener('close', handleClose);
319
- socket.addEventListener('error', handleError);
320
- socket.addEventListener('message', handleMessage);
321
- disposeSocket = () => {
322
- window.clearTimeout(retryWaitTimeoutId);
323
- socket.removeEventListener('open', handleOpen);
324
- socket.removeEventListener('close', handleClose);
325
- socket.removeEventListener('error', handleError);
326
- socket.removeEventListener('message', handleMessage);
327
- };
328
- };
329
- attemptSocketConnection();
330
- });
331
- }
332
-
333
- export async function startRemoteScenario(
334
- config: RemoteWalletAssociationConfig,
335
- ): Promise<RemoteScenario> {
336
- assertSecureContext();
337
- const associationKeypair = await generateAssociationKeypair();
338
- const websocketURL = `wss://${config?.remoteHostAuthority}/reflect`;
339
- let connectionStartTime: number;
340
- const getNextRetryDelayMs = (() => {
341
- const schedule = [...WEBSOCKET_CONNECTION_CONFIG.retryDelayScheduleMs];
342
- return () => (schedule.length > 1 ? (schedule.shift() as number) : schedule[0]);
343
- })();
344
- let nextJsonRpcMessageId = 1;
345
- let lastKnownInboundSequenceNumber = 0;
346
- let encoding: PROTOCOL_ENCODING;
347
- let state: RemoteState = { __type: 'disconnected' };
348
- let socket: WebSocket;
349
- let disposeSocket: () => void;
350
- let decodeBytes = async (evt: MessageEvent<string | Blob>) => {
351
- if (encoding == 'base64') { // base64 encoding
352
- const message = await evt.data as string;
353
- return toUint8Array(message).buffer;
354
- } else {
355
- return await (evt.data as Blob).arrayBuffer();
356
- }
357
- };
358
- // Reflector Connection Phase
359
- // here we connect to the reflector and wait for the REFLECTOR_ID message
360
- // so we build the association URL and return that back to the caller
361
- const associationUrl = await new Promise<URL>((resolve, reject) => {
362
- const handleOpen = async () => {
363
- if (state.__type !== 'connecting') {
364
- console.warn(
365
- 'Expected adapter state to be `connecting` at the moment the websocket opens. ' +
366
- `Got \`${state.__type}\`.`,
367
- );
368
- return;
369
- }
370
- if (socket.protocol.includes(WEBSOCKET_PROTOCOL_BASE64)) {
371
- encoding = 'base64';
372
- } else {
373
- encoding = 'binary';
374
- }
375
- socket.removeEventListener('open', handleOpen);
376
- };
377
- const handleClose = (evt: CloseEvent) => {
378
- if (evt.wasClean) {
379
- state = { __type: 'disconnected' };
380
- } else {
381
- reject(
382
- new SolanaMobileWalletAdapterError(
383
- SolanaMobileWalletAdapterErrorCode.ERROR_SESSION_CLOSED,
384
- `The wallet session dropped unexpectedly (${evt.code}: ${evt.reason}).`,
385
- { closeEvent: evt },
386
- ),
387
- );
388
- }
389
- disposeSocket();
390
- };
391
- const handleError = async (_evt: Event) => {
392
- disposeSocket();
393
- if (Date.now() - connectionStartTime >= WEBSOCKET_CONNECTION_CONFIG.timeoutMs) {
394
- reject(
395
- new SolanaMobileWalletAdapterError(
396
- SolanaMobileWalletAdapterErrorCode.ERROR_SESSION_TIMEOUT,
397
- `Failed to connect to the wallet websocket at ${websocketURL}.`,
398
- ),
399
- );
400
- } else {
401
- await new Promise((resolve) => {
402
- const retryDelayMs = getNextRetryDelayMs();
403
- retryWaitTimeoutId = window.setTimeout(resolve, retryDelayMs);
404
- });
405
- attemptSocketConnection();
406
- }
407
- };
408
- const handleReflectorIdMessage = async (evt: MessageEvent<string | Blob>) => {
409
- const responseBuffer = await decodeBytes(evt);
410
- if (state.__type === 'connecting') {
411
- if (responseBuffer.byteLength == 0) {
412
- throw new Error('Encountered unexpected message while connecting');
413
- }
414
- const reflectorId = getReflectorIdFromByteArray(responseBuffer);
415
- state = {
416
- __type: 'reflector_id_received',
417
- reflectorId: reflectorId
418
- };
419
- const associationUrl = await getRemoteAssociateAndroidIntentURL(
420
- associationKeypair.publicKey,
421
- config.remoteHostAuthority,
422
- reflectorId,
423
- config?.baseUri
424
- );
425
- socket.removeEventListener('message', handleReflectorIdMessage);
426
- resolve(associationUrl);
427
- }
428
- };
429
- let retryWaitTimeoutId: number;
430
- const attemptSocketConnection = () => {
431
- if (disposeSocket) {
432
- disposeSocket();
433
- }
434
- state = { __type: 'connecting', associationKeypair };
435
- if (connectionStartTime === undefined) {
436
- connectionStartTime = Date.now();
437
- }
438
- socket = new WebSocket(websocketURL,
439
- [WEBSOCKET_PROTOCOL_BINARY, WEBSOCKET_PROTOCOL_BASE64]);
440
- socket.addEventListener('open', handleOpen);
441
- socket.addEventListener('close', handleClose);
442
- socket.addEventListener('error', handleError);
443
- socket.addEventListener('message', handleReflectorIdMessage);
444
- disposeSocket = () => {
445
- window.clearTimeout(retryWaitTimeoutId);
446
- socket.removeEventListener('open', handleOpen);
447
- socket.removeEventListener('close', handleClose);
448
- socket.removeEventListener('error', handleError);
449
- socket.removeEventListener('message', handleReflectorIdMessage);
450
- };
451
- };
452
- attemptSocketConnection();
453
- });
454
- // Wallet Connection Phase
455
- // here we return the association URL (containing the reflector ID) to the caller +
456
- // a promise that will resolve the MobileWallet object once the wallet connects.
457
- let sessionEstablished = false;
458
- let handleClose: () => void;
459
- return { associationUrl, close: () => {
460
- socket.close();
461
- handleClose();
462
- }, wallet: new Promise((resolve, reject) => {
463
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
464
- const jsonRpcResponsePromises: JsonResponsePromises<any> = {};
465
- const handleMessage = async (evt: MessageEvent<string | Blob>) => {
466
- const responseBuffer = await decodeBytes(evt);
467
- switch (state.__type) {
468
- case 'reflector_id_received':
469
- if (responseBuffer.byteLength !== 0) {
470
- throw new Error('Encountered unexpected message while awaiting reflection');
471
- }
472
- const ecdhKeypair = await generateECDHKeypair();
473
- const binaryMsg = await createHelloReq(ecdhKeypair.publicKey, associationKeypair.privateKey);
474
- if (encoding == 'base64') {
475
- socket.send(fromUint8Array(binaryMsg));
476
- } else {
477
- socket.send(binaryMsg);
478
- }
479
- state = {
480
- __type: 'hello_req_sent',
481
- associationPublicKey: associationKeypair.publicKey,
482
- ecdhPrivateKey: ecdhKeypair.privateKey,
483
- };
484
- break;
485
- case 'connected':
486
- try {
487
- const sequenceNumberVector = responseBuffer.slice(0, SEQUENCE_NUMBER_BYTES);
488
- const sequenceNumber = getSequenceNumberFromByteArray(sequenceNumberVector);
489
- if (sequenceNumber !== (lastKnownInboundSequenceNumber + 1)) {
490
- throw new Error('Encrypted message has invalid sequence number');
491
- }
492
- lastKnownInboundSequenceNumber = sequenceNumber;
493
- const jsonRpcMessage = await decryptJsonRpcMessage(responseBuffer, state.sharedSecret);
494
- const responsePromise = jsonRpcResponsePromises[jsonRpcMessage.id];
495
- delete jsonRpcResponsePromises[jsonRpcMessage.id];
496
- responsePromise.resolve(jsonRpcMessage.result);
497
- } catch (e) {
498
- if (e instanceof SolanaMobileWalletAdapterProtocolError) {
499
- const responsePromise = jsonRpcResponsePromises[e.jsonRpcMessageId];
500
- delete jsonRpcResponsePromises[e.jsonRpcMessageId];
501
- responsePromise.reject(e);
502
- } else {
503
- throw e;
504
- }
505
- }
506
- break;
507
- case 'hello_req_sent': {
508
- const sharedSecret = await parseHelloRsp(
509
- responseBuffer,
510
- state.associationPublicKey,
511
- state.ecdhPrivateKey,
512
- );
513
- const sessionPropertiesBuffer = responseBuffer.slice(ENCODED_PUBLIC_KEY_LENGTH_BYTES);
514
- const sessionProperties = sessionPropertiesBuffer.byteLength !== 0
515
- ? await (async () => {
516
- const sequenceNumberVector = sessionPropertiesBuffer.slice(0, SEQUENCE_NUMBER_BYTES);
517
- const sequenceNumber = getSequenceNumberFromByteArray(sequenceNumberVector);
518
- if (sequenceNumber !== (lastKnownInboundSequenceNumber + 1)) {
519
- throw new Error('Encrypted message has invalid sequence number');
520
- }
521
- lastKnownInboundSequenceNumber = sequenceNumber;
522
- return parseSessionProps(sessionPropertiesBuffer, sharedSecret);
523
- })() : <SessionProperties> { protocol_version: 'legacy' };
524
- state = { __type: 'connected', sharedSecret, sessionProperties };
525
- const wallet = createMobileWalletProxy(sessionProperties.protocol_version,
526
- async (method, params) => {
527
- const id = nextJsonRpcMessageId++;
528
- const binaryMsg = await encryptJsonRpcMessage(
529
- {
530
- id,
531
- jsonrpc: '2.0' as const,
532
- method,
533
- params: params ?? {},
534
- },
535
- sharedSecret,
536
- )
537
- if (encoding == 'base64') {
538
- socket.send(fromUint8Array(binaryMsg));
539
- } else {
540
- socket.send(binaryMsg);
541
- }
542
- return new Promise((resolve, reject) => {
543
- jsonRpcResponsePromises[id] = {
544
- resolve(result) {
545
- switch (method) {
546
- case 'authorize':
547
- case 'reauthorize': {
548
- const { wallet_uri_base } = result as Awaited<
549
- ReturnType<MobileWallet['authorize' | 'reauthorize']>
550
- >;
551
- if (wallet_uri_base != null) {
552
- try {
553
- assertSecureEndpointSpecificURI(wallet_uri_base);
554
- } catch (e) {
555
- reject(e);
556
- return;
557
- }
558
- }
559
- break;
560
- }
561
- }
562
- resolve(result);
563
- },
564
- reject,
565
- };
566
- });
567
- })
568
- sessionEstablished = true;
569
- try {
570
- resolve(wallet);
571
- } catch (e) {
572
- reject(e);
573
- }
574
- break;
575
- }
576
- }
577
- }
578
- socket.addEventListener('message', handleMessage);
579
- handleClose = () => {
580
- socket.removeEventListener('message', handleMessage);
581
- disposeSocket();
582
- if (!sessionEstablished) {
583
- reject(
584
- new SolanaMobileWalletAdapterError(
585
- SolanaMobileWalletAdapterErrorCode.ERROR_SESSION_CLOSED,
586
- `The wallet session was closed before connection.`,
587
- { closeEvent: new CloseEvent('socket was closed before connection') },
588
- ),
589
- );
590
- }
591
- };
592
- })};
593
- }