@provablehq/aleo-wallet-adaptor-puzzle 0.1.1-alpha.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.
@@ -0,0 +1,382 @@
1
+ import {
2
+ Account,
3
+ TransactionOptions,
4
+ Network,
5
+ TransactionStatusResponse,
6
+ TransactionStatus,
7
+ } from '@provablehq/aleo-types';
8
+ import {
9
+ AleoDeployment,
10
+ WalletDecryptPermission,
11
+ WalletName,
12
+ WalletReadyState,
13
+ } from '@provablehq/aleo-wallet-standard';
14
+ import {
15
+ BaseAleoWalletAdapter,
16
+ DecryptPermission,
17
+ MethodNotImplementedError,
18
+ WalletConnectionError,
19
+ WalletDecryptionError,
20
+ WalletDecryptionNotAllowedError,
21
+ WalletDisconnectionError,
22
+ WalletError,
23
+ WalletNotConnectedError,
24
+ WalletSignMessageError,
25
+ WalletTransactionError,
26
+ } from '@provablehq/aleo-wallet-adaptor-core';
27
+ import {
28
+ connect,
29
+ disconnect,
30
+ requestCreateEvent,
31
+ requestSignature,
32
+ EventType,
33
+ decrypt as puzzleDecrypt,
34
+ getRecords,
35
+ getEvent,
36
+ } from '@puzzlehq/sdk-core';
37
+ import { PuzzleWindow, PuzzleWalletAdapterConfig, PUZZLE_NETWORK_MAP } from './types';
38
+ import { PuzzleIcon } from './icon';
39
+
40
+ /**
41
+ * Puzzle wallet adapter
42
+ */
43
+ export class PuzzleWalletAdapter extends BaseAleoWalletAdapter {
44
+ /**
45
+ * The wallet name
46
+ */
47
+ name = 'Puzzle Wallet' as WalletName<'Puzzle Wallet'>;
48
+
49
+ /**
50
+ * The wallet URL
51
+ */
52
+ url = 'https://puzzle.online/wallet';
53
+
54
+ /**
55
+ * The wallet icon (base64-encoded SVG)
56
+ */
57
+ icon = PuzzleIcon;
58
+ /**
59
+ * The window object
60
+ */
61
+ private _window: PuzzleWindow | undefined;
62
+
63
+ /**
64
+ * App name
65
+ */
66
+ private _appName: string;
67
+
68
+ /**
69
+ * App icon URL
70
+ */
71
+ private _appIconUrl?: string;
72
+
73
+ /**
74
+ * App description
75
+ */
76
+ private _appDescription?: string;
77
+
78
+ /**
79
+ * Current network
80
+ */
81
+ network: Network = Network.TESTNET3;
82
+
83
+ /**
84
+ * The wallet's decrypt permission
85
+ */
86
+ decryptPermission: WalletDecryptPermission = WalletDecryptPermission.NoDecrypt;
87
+
88
+ _readyState: WalletReadyState =
89
+ typeof window === 'undefined' || typeof document === 'undefined'
90
+ ? WalletReadyState.UNSUPPORTED
91
+ : WalletReadyState.NOT_DETECTED;
92
+
93
+ /**
94
+ * Public key
95
+ */
96
+ private _publicKey: string = '';
97
+
98
+ /**
99
+ * Create a new Puzzle wallet adapter
100
+ * @param config Adapter configuration
101
+ */
102
+ constructor(config?: PuzzleWalletAdapterConfig) {
103
+ super();
104
+ this._appName = config?.appName || 'Aleo App';
105
+ this._appIconUrl = config?.appIconUrl;
106
+ this._appDescription = config?.appDescription;
107
+ this._checkAvailability();
108
+ }
109
+
110
+ /**
111
+ * Check if Puzzle wallet is available
112
+ */
113
+ private _checkAvailability(): void {
114
+ if (typeof window === 'undefined') {
115
+ this.readyState = WalletReadyState.UNSUPPORTED;
116
+ return;
117
+ }
118
+
119
+ this._window = window as PuzzleWindow;
120
+
121
+ if (this._window.puzzle) {
122
+ this.readyState = WalletReadyState.INSTALLED;
123
+ } else {
124
+ // Check if user is on a mobile device
125
+ const isMobile = /iPhone|iPad|iPod|Android/i.test(navigator.userAgent);
126
+ if (isMobile) {
127
+ this.readyState = WalletReadyState.LOADABLE;
128
+ }
129
+ }
130
+ }
131
+
132
+ /**
133
+ * Connect to Puzzle wallet
134
+ * @param network The network to connect to
135
+ * @returns The connected account
136
+ */
137
+ async connect(
138
+ network: Network,
139
+ decryptPermission: WalletDecryptPermission,
140
+ programs?: string[],
141
+ ): Promise<Account> {
142
+ try {
143
+ if (this.readyState !== WalletReadyState.INSTALLED) {
144
+ throw new WalletConnectionError('Puzzle Wallet is not available');
145
+ }
146
+
147
+ // Call connect and extract address safely
148
+ const response = await connect({
149
+ dAppInfo: {
150
+ name: this._appName,
151
+ description: this._appDescription,
152
+ iconUrl: this._appIconUrl,
153
+ },
154
+ permissions: {
155
+ programIds: {
156
+ [PUZZLE_NETWORK_MAP[network]]: programs,
157
+ },
158
+ },
159
+ });
160
+
161
+ // Type guard to check response structure
162
+ if (!response || typeof response !== 'object' || !('connection' in response)) {
163
+ throw new WalletConnectionError('Invalid response from wallet');
164
+ }
165
+
166
+ this.network = network;
167
+ this.decryptPermission = decryptPermission;
168
+ const address = (response as { connection: { address: string } }).connection?.address;
169
+
170
+ if (!address) {
171
+ throw new WalletConnectionError('No address returned from wallet');
172
+ }
173
+
174
+ this._publicKey = address;
175
+
176
+ const account: Account = {
177
+ address: this._publicKey,
178
+ };
179
+
180
+ this.account = account;
181
+ this.emit('connect', account);
182
+
183
+ return account;
184
+ } catch (err: Error | unknown) {
185
+ this.emit('error', err instanceof Error ? err : new Error(String(err)));
186
+ throw new WalletConnectionError(err instanceof Error ? err.message : 'Connection failed');
187
+ }
188
+ }
189
+
190
+ /**
191
+ * Disconnect from Puzzle wallet
192
+ */
193
+ async disconnect(): Promise<void> {
194
+ try {
195
+ await disconnect();
196
+ this._publicKey = '';
197
+ this.account = undefined;
198
+ this.emit('disconnect');
199
+ } catch (err: Error | unknown) {
200
+ this.emit('error', err instanceof Error ? err : new Error(String(err)));
201
+ throw new WalletDisconnectionError(
202
+ err instanceof Error ? err.message : 'Disconnection failed',
203
+ );
204
+ }
205
+ }
206
+
207
+ /**
208
+ * Sign a message with Puzzle wallet
209
+ * @param message The message to sign
210
+ * @returns The signed message
211
+ */
212
+ async signMessage(message: Uint8Array): Promise<Uint8Array> {
213
+ if (!this._publicKey || !this.account) {
214
+ throw new WalletNotConnectedError();
215
+ }
216
+
217
+ try {
218
+ // Pass only the parameters expected by the Puzzle SDK
219
+ const signature = await requestSignature({
220
+ message: message.toString(),
221
+ address: this._publicKey,
222
+ });
223
+
224
+ return new TextEncoder().encode(signature.signature);
225
+ } catch (error: Error | unknown) {
226
+ throw new WalletSignMessageError(
227
+ error instanceof Error ? error.message : 'Failed to sign message',
228
+ );
229
+ }
230
+ }
231
+
232
+ async decrypt(cipherText: string) {
233
+ if (!this._publicKey || !this.account) {
234
+ throw new WalletNotConnectedError();
235
+ }
236
+ switch (this.decryptPermission) {
237
+ case WalletDecryptPermission.NoDecrypt:
238
+ throw new WalletDecryptionNotAllowedError();
239
+
240
+ case WalletDecryptPermission.UponRequest:
241
+ case DecryptPermission.AutoDecrypt:
242
+ case DecryptPermission.OnChainHistory: {
243
+ try {
244
+ const text = await puzzleDecrypt({ ciphertexts: [cipherText] });
245
+ return text.plaintexts![0];
246
+ } catch (error: Error | unknown) {
247
+ throw new WalletDecryptionError(
248
+ error instanceof Error ? error.message : 'Failed to decrypt',
249
+ );
250
+ }
251
+ }
252
+ default:
253
+ throw new WalletDecryptionError();
254
+ }
255
+ }
256
+
257
+ /**
258
+ * Execute a transaction with Puzzle wallet
259
+ * @param options Transaction options
260
+ * @returns The executed temporary transaction ID
261
+ */
262
+ async executeTransaction(options: TransactionOptions): Promise<{ transactionId: string }> {
263
+ if (!this._publicKey || !this.account) {
264
+ throw new WalletNotConnectedError();
265
+ }
266
+
267
+ if (options.privateFee) {
268
+ throw new WalletTransactionError('Private fee is not supported by Puzzle wallet');
269
+ }
270
+
271
+ try {
272
+ const fee = options.fee ? options.fee / 1000000 : 0.001;
273
+
274
+ const requestData = {
275
+ type: EventType.Execute,
276
+ programId: options.program,
277
+ functionId: options.function,
278
+ fee,
279
+ inputs: options.inputs,
280
+ address: this._publicKey,
281
+ network: PUZZLE_NETWORK_MAP[this.network],
282
+ };
283
+
284
+ const result = await requestCreateEvent(requestData);
285
+
286
+ if (result.error) {
287
+ throw new WalletTransactionError(result.error);
288
+ }
289
+
290
+ if (!result.eventId) {
291
+ throw new WalletTransactionError('Could not create transaction');
292
+ }
293
+
294
+ return {
295
+ transactionId: result.eventId,
296
+ };
297
+ } catch (error: Error | unknown) {
298
+ console.error('Puzzle Wallet executeTransaction error', error);
299
+ if (error instanceof WalletError) {
300
+ throw error;
301
+ }
302
+ throw new WalletTransactionError(
303
+ error instanceof Error ? error.message : 'Failed to execute transaction',
304
+ );
305
+ }
306
+ }
307
+
308
+ /**
309
+ * Get transaction status
310
+ * @param transactionId The transaction ID
311
+ * @returns The transaction status
312
+ */
313
+ async transactionStatus(transactionId: string): Promise<TransactionStatusResponse> {
314
+ if (!this._publicKey || !this.account) {
315
+ throw new WalletNotConnectedError();
316
+ }
317
+
318
+ try {
319
+ const result = await getEvent({
320
+ id: transactionId,
321
+ address: this._publicKey,
322
+ network: PUZZLE_NETWORK_MAP[this.network],
323
+ });
324
+
325
+ return {
326
+ status: result.event?.status || TransactionStatus.PENDING,
327
+ transactionId: result.event?.transactionId,
328
+ error: result.event?.error,
329
+ };
330
+ } catch (error: Error | unknown) {
331
+ throw new WalletTransactionError(
332
+ error instanceof Error ? error.message : 'Failed to get transaction status',
333
+ );
334
+ }
335
+ }
336
+ /**
337
+ * Request records from Leo wallet
338
+ * @param program The program to request records from
339
+ * @param includePlaintext Whether to include plaintext on each record
340
+ * @returns The records
341
+ */
342
+ async requestRecords(program: string): Promise<unknown[]> {
343
+ if (!this._publicKey || !this.account) {
344
+ throw new WalletNotConnectedError();
345
+ }
346
+
347
+ try {
348
+ const result = await getRecords({
349
+ filter: {
350
+ programIds: [program],
351
+ status: 'All',
352
+ },
353
+ address: this._publicKey,
354
+ network: PUZZLE_NETWORK_MAP[this.network],
355
+ });
356
+
357
+ return result?.records || [];
358
+ } catch (error: Error | unknown) {
359
+ throw new WalletError(error instanceof Error ? error.message : 'Failed to request records');
360
+ }
361
+ }
362
+
363
+ /**
364
+ * Switch the network
365
+ * @param network The network to switch to
366
+ */
367
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
368
+ async switchNetwork(_network: Network): Promise<void> {
369
+ console.error('Puzzle Wallet does not support switching networks');
370
+ throw new MethodNotImplementedError('switchNetwork');
371
+ }
372
+
373
+ /**
374
+ * Execute a deployment
375
+ * @param deployment The deployment to execute
376
+ * @returns The executed transaction ID
377
+ */
378
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
379
+ async executeDeployment(_deployment: AleoDeployment): Promise<{ transactionId: string }> {
380
+ throw new MethodNotImplementedError('executeDeployment');
381
+ }
382
+ }
package/src/icon.ts ADDED
@@ -0,0 +1,2 @@
1
+ export const PuzzleIcon =
2
+ '';
package/src/index.ts ADDED
@@ -0,0 +1,2 @@
1
+ export * from './PuzzleWalletAdapter';
2
+ export * from './types';
package/src/types.ts ADDED
@@ -0,0 +1,44 @@
1
+ import { Network } from '@provablehq/aleo-types';
2
+ import { Network as PuzzleNetwork } from '@puzzlehq/sdk-core';
3
+
4
+ /**
5
+ * Puzzle window interface
6
+ */
7
+ export interface PuzzleWindow extends Window {
8
+ puzzle?: {
9
+ connected: boolean;
10
+ connect(): Promise<unknown>;
11
+ disconnect(): Promise<void>;
12
+ };
13
+ }
14
+
15
+ /**
16
+ * Puzzle wallet adapter configuration
17
+ */
18
+ export interface PuzzleWalletAdapterConfig {
19
+ /**
20
+ * Application name
21
+ */
22
+ appName?: string;
23
+
24
+ /**
25
+ * Application icon URL
26
+ */
27
+ appIconUrl?: string;
28
+
29
+ /**
30
+ * Application description
31
+ */
32
+ appDescription?: string;
33
+
34
+ /**
35
+ * Program ID permissions by network
36
+ */
37
+ programIdPermissions?: Record<string, string[]>;
38
+ }
39
+
40
+ export const PUZZLE_NETWORK_MAP: Record<Network, PuzzleNetwork> = {
41
+ [Network.MAINNET]: PuzzleNetwork.AleoMainnet,
42
+ [Network.TESTNET3]: PuzzleNetwork.AleoTestnet,
43
+ [Network.CANARY]: PuzzleNetwork.AleoTestnet,
44
+ };