@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.
- package/LICENSE +674 -0
- package/README.md +31 -0
- package/dist/index.d.mts +146 -0
- package/dist/index.d.ts +146 -0
- package/dist/index.js +321 -0
- package/dist/index.mjs +320 -0
- package/package.json +45 -0
- package/src/PuzzleWalletAdapter.ts +382 -0
- package/src/icon.ts +2 -0
- package/src/index.ts +2 -0
- package/src/types.ts +44 -0
package/dist/index.mjs
ADDED
|
@@ -0,0 +1,320 @@
|
|
|
1
|
+
// src/PuzzleWalletAdapter.ts
|
|
2
|
+
import {
|
|
3
|
+
Network as Network2,
|
|
4
|
+
TransactionStatus
|
|
5
|
+
} from "@provablehq/aleo-types";
|
|
6
|
+
import {
|
|
7
|
+
WalletDecryptPermission,
|
|
8
|
+
WalletReadyState
|
|
9
|
+
} from "@provablehq/aleo-wallet-standard";
|
|
10
|
+
import {
|
|
11
|
+
BaseAleoWalletAdapter,
|
|
12
|
+
DecryptPermission,
|
|
13
|
+
MethodNotImplementedError,
|
|
14
|
+
WalletConnectionError,
|
|
15
|
+
WalletDecryptionError,
|
|
16
|
+
WalletDecryptionNotAllowedError,
|
|
17
|
+
WalletDisconnectionError,
|
|
18
|
+
WalletError,
|
|
19
|
+
WalletNotConnectedError,
|
|
20
|
+
WalletSignMessageError,
|
|
21
|
+
WalletTransactionError
|
|
22
|
+
} from "@provablehq/aleo-wallet-adaptor-core";
|
|
23
|
+
import {
|
|
24
|
+
connect,
|
|
25
|
+
disconnect,
|
|
26
|
+
requestCreateEvent,
|
|
27
|
+
requestSignature,
|
|
28
|
+
EventType,
|
|
29
|
+
decrypt as puzzleDecrypt,
|
|
30
|
+
getRecords,
|
|
31
|
+
getEvent
|
|
32
|
+
} from "@puzzlehq/sdk-core";
|
|
33
|
+
|
|
34
|
+
// src/types.ts
|
|
35
|
+
import { Network } from "@provablehq/aleo-types";
|
|
36
|
+
import { Network as PuzzleNetwork } from "@puzzlehq/sdk-core";
|
|
37
|
+
var PUZZLE_NETWORK_MAP = {
|
|
38
|
+
[Network.MAINNET]: PuzzleNetwork.AleoMainnet,
|
|
39
|
+
[Network.TESTNET3]: PuzzleNetwork.AleoTestnet,
|
|
40
|
+
[Network.CANARY]: PuzzleNetwork.AleoTestnet
|
|
41
|
+
};
|
|
42
|
+
|
|
43
|
+
// src/icon.ts
|
|
44
|
+
var PuzzleIcon = "";
|
|
45
|
+
|
|
46
|
+
// src/PuzzleWalletAdapter.ts
|
|
47
|
+
var PuzzleWalletAdapter = class extends BaseAleoWalletAdapter {
|
|
48
|
+
/**
|
|
49
|
+
* Create a new Puzzle wallet adapter
|
|
50
|
+
* @param config Adapter configuration
|
|
51
|
+
*/
|
|
52
|
+
constructor(config) {
|
|
53
|
+
super();
|
|
54
|
+
/**
|
|
55
|
+
* The wallet name
|
|
56
|
+
*/
|
|
57
|
+
this.name = "Puzzle Wallet";
|
|
58
|
+
/**
|
|
59
|
+
* The wallet URL
|
|
60
|
+
*/
|
|
61
|
+
this.url = "https://puzzle.online/wallet";
|
|
62
|
+
/**
|
|
63
|
+
* The wallet icon (base64-encoded SVG)
|
|
64
|
+
*/
|
|
65
|
+
this.icon = PuzzleIcon;
|
|
66
|
+
/**
|
|
67
|
+
* Current network
|
|
68
|
+
*/
|
|
69
|
+
this.network = Network2.TESTNET3;
|
|
70
|
+
/**
|
|
71
|
+
* The wallet's decrypt permission
|
|
72
|
+
*/
|
|
73
|
+
this.decryptPermission = WalletDecryptPermission.NoDecrypt;
|
|
74
|
+
this._readyState = typeof window === "undefined" || typeof document === "undefined" ? WalletReadyState.UNSUPPORTED : WalletReadyState.NOT_DETECTED;
|
|
75
|
+
/**
|
|
76
|
+
* Public key
|
|
77
|
+
*/
|
|
78
|
+
this._publicKey = "";
|
|
79
|
+
this._appName = config?.appName || "Aleo App";
|
|
80
|
+
this._appIconUrl = config?.appIconUrl;
|
|
81
|
+
this._appDescription = config?.appDescription;
|
|
82
|
+
this._checkAvailability();
|
|
83
|
+
}
|
|
84
|
+
/**
|
|
85
|
+
* Check if Puzzle wallet is available
|
|
86
|
+
*/
|
|
87
|
+
_checkAvailability() {
|
|
88
|
+
if (typeof window === "undefined") {
|
|
89
|
+
this.readyState = WalletReadyState.UNSUPPORTED;
|
|
90
|
+
return;
|
|
91
|
+
}
|
|
92
|
+
this._window = window;
|
|
93
|
+
if (this._window.puzzle) {
|
|
94
|
+
this.readyState = WalletReadyState.INSTALLED;
|
|
95
|
+
} else {
|
|
96
|
+
const isMobile = /iPhone|iPad|iPod|Android/i.test(navigator.userAgent);
|
|
97
|
+
if (isMobile) {
|
|
98
|
+
this.readyState = WalletReadyState.LOADABLE;
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
/**
|
|
103
|
+
* Connect to Puzzle wallet
|
|
104
|
+
* @param network The network to connect to
|
|
105
|
+
* @returns The connected account
|
|
106
|
+
*/
|
|
107
|
+
async connect(network, decryptPermission, programs) {
|
|
108
|
+
try {
|
|
109
|
+
if (this.readyState !== WalletReadyState.INSTALLED) {
|
|
110
|
+
throw new WalletConnectionError("Puzzle Wallet is not available");
|
|
111
|
+
}
|
|
112
|
+
const response = await connect({
|
|
113
|
+
dAppInfo: {
|
|
114
|
+
name: this._appName,
|
|
115
|
+
description: this._appDescription,
|
|
116
|
+
iconUrl: this._appIconUrl
|
|
117
|
+
},
|
|
118
|
+
permissions: {
|
|
119
|
+
programIds: {
|
|
120
|
+
[PUZZLE_NETWORK_MAP[network]]: programs
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
});
|
|
124
|
+
if (!response || typeof response !== "object" || !("connection" in response)) {
|
|
125
|
+
throw new WalletConnectionError("Invalid response from wallet");
|
|
126
|
+
}
|
|
127
|
+
this.network = network;
|
|
128
|
+
this.decryptPermission = decryptPermission;
|
|
129
|
+
const address = response.connection?.address;
|
|
130
|
+
if (!address) {
|
|
131
|
+
throw new WalletConnectionError("No address returned from wallet");
|
|
132
|
+
}
|
|
133
|
+
this._publicKey = address;
|
|
134
|
+
const account = {
|
|
135
|
+
address: this._publicKey
|
|
136
|
+
};
|
|
137
|
+
this.account = account;
|
|
138
|
+
this.emit("connect", account);
|
|
139
|
+
return account;
|
|
140
|
+
} catch (err) {
|
|
141
|
+
this.emit("error", err instanceof Error ? err : new Error(String(err)));
|
|
142
|
+
throw new WalletConnectionError(err instanceof Error ? err.message : "Connection failed");
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
/**
|
|
146
|
+
* Disconnect from Puzzle wallet
|
|
147
|
+
*/
|
|
148
|
+
async disconnect() {
|
|
149
|
+
try {
|
|
150
|
+
await disconnect();
|
|
151
|
+
this._publicKey = "";
|
|
152
|
+
this.account = void 0;
|
|
153
|
+
this.emit("disconnect");
|
|
154
|
+
} catch (err) {
|
|
155
|
+
this.emit("error", err instanceof Error ? err : new Error(String(err)));
|
|
156
|
+
throw new WalletDisconnectionError(
|
|
157
|
+
err instanceof Error ? err.message : "Disconnection failed"
|
|
158
|
+
);
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
/**
|
|
162
|
+
* Sign a message with Puzzle wallet
|
|
163
|
+
* @param message The message to sign
|
|
164
|
+
* @returns The signed message
|
|
165
|
+
*/
|
|
166
|
+
async signMessage(message) {
|
|
167
|
+
if (!this._publicKey || !this.account) {
|
|
168
|
+
throw new WalletNotConnectedError();
|
|
169
|
+
}
|
|
170
|
+
try {
|
|
171
|
+
const signature = await requestSignature({
|
|
172
|
+
message: message.toString(),
|
|
173
|
+
address: this._publicKey
|
|
174
|
+
});
|
|
175
|
+
return new TextEncoder().encode(signature.signature);
|
|
176
|
+
} catch (error) {
|
|
177
|
+
throw new WalletSignMessageError(
|
|
178
|
+
error instanceof Error ? error.message : "Failed to sign message"
|
|
179
|
+
);
|
|
180
|
+
}
|
|
181
|
+
}
|
|
182
|
+
async decrypt(cipherText) {
|
|
183
|
+
if (!this._publicKey || !this.account) {
|
|
184
|
+
throw new WalletNotConnectedError();
|
|
185
|
+
}
|
|
186
|
+
switch (this.decryptPermission) {
|
|
187
|
+
case WalletDecryptPermission.NoDecrypt:
|
|
188
|
+
throw new WalletDecryptionNotAllowedError();
|
|
189
|
+
case WalletDecryptPermission.UponRequest:
|
|
190
|
+
case DecryptPermission.AutoDecrypt:
|
|
191
|
+
case DecryptPermission.OnChainHistory: {
|
|
192
|
+
try {
|
|
193
|
+
const text = await puzzleDecrypt({ ciphertexts: [cipherText] });
|
|
194
|
+
return text.plaintexts[0];
|
|
195
|
+
} catch (error) {
|
|
196
|
+
throw new WalletDecryptionError(
|
|
197
|
+
error instanceof Error ? error.message : "Failed to decrypt"
|
|
198
|
+
);
|
|
199
|
+
}
|
|
200
|
+
}
|
|
201
|
+
default:
|
|
202
|
+
throw new WalletDecryptionError();
|
|
203
|
+
}
|
|
204
|
+
}
|
|
205
|
+
/**
|
|
206
|
+
* Execute a transaction with Puzzle wallet
|
|
207
|
+
* @param options Transaction options
|
|
208
|
+
* @returns The executed temporary transaction ID
|
|
209
|
+
*/
|
|
210
|
+
async executeTransaction(options) {
|
|
211
|
+
if (!this._publicKey || !this.account) {
|
|
212
|
+
throw new WalletNotConnectedError();
|
|
213
|
+
}
|
|
214
|
+
if (options.privateFee) {
|
|
215
|
+
throw new WalletTransactionError("Private fee is not supported by Puzzle wallet");
|
|
216
|
+
}
|
|
217
|
+
try {
|
|
218
|
+
const fee = options.fee ? options.fee / 1e6 : 1e-3;
|
|
219
|
+
const requestData = {
|
|
220
|
+
type: EventType.Execute,
|
|
221
|
+
programId: options.program,
|
|
222
|
+
functionId: options.function,
|
|
223
|
+
fee,
|
|
224
|
+
inputs: options.inputs,
|
|
225
|
+
address: this._publicKey,
|
|
226
|
+
network: PUZZLE_NETWORK_MAP[this.network]
|
|
227
|
+
};
|
|
228
|
+
const result = await requestCreateEvent(requestData);
|
|
229
|
+
if (result.error) {
|
|
230
|
+
throw new WalletTransactionError(result.error);
|
|
231
|
+
}
|
|
232
|
+
if (!result.eventId) {
|
|
233
|
+
throw new WalletTransactionError("Could not create transaction");
|
|
234
|
+
}
|
|
235
|
+
return {
|
|
236
|
+
transactionId: result.eventId
|
|
237
|
+
};
|
|
238
|
+
} catch (error) {
|
|
239
|
+
console.error("Puzzle Wallet executeTransaction error", error);
|
|
240
|
+
if (error instanceof WalletError) {
|
|
241
|
+
throw error;
|
|
242
|
+
}
|
|
243
|
+
throw new WalletTransactionError(
|
|
244
|
+
error instanceof Error ? error.message : "Failed to execute transaction"
|
|
245
|
+
);
|
|
246
|
+
}
|
|
247
|
+
}
|
|
248
|
+
/**
|
|
249
|
+
* Get transaction status
|
|
250
|
+
* @param transactionId The transaction ID
|
|
251
|
+
* @returns The transaction status
|
|
252
|
+
*/
|
|
253
|
+
async transactionStatus(transactionId) {
|
|
254
|
+
if (!this._publicKey || !this.account) {
|
|
255
|
+
throw new WalletNotConnectedError();
|
|
256
|
+
}
|
|
257
|
+
try {
|
|
258
|
+
const result = await getEvent({
|
|
259
|
+
id: transactionId,
|
|
260
|
+
address: this._publicKey,
|
|
261
|
+
network: PUZZLE_NETWORK_MAP[this.network]
|
|
262
|
+
});
|
|
263
|
+
return {
|
|
264
|
+
status: result.event?.status || TransactionStatus.PENDING,
|
|
265
|
+
transactionId: result.event?.transactionId,
|
|
266
|
+
error: result.event?.error
|
|
267
|
+
};
|
|
268
|
+
} catch (error) {
|
|
269
|
+
throw new WalletTransactionError(
|
|
270
|
+
error instanceof Error ? error.message : "Failed to get transaction status"
|
|
271
|
+
);
|
|
272
|
+
}
|
|
273
|
+
}
|
|
274
|
+
/**
|
|
275
|
+
* Request records from Leo wallet
|
|
276
|
+
* @param program The program to request records from
|
|
277
|
+
* @param includePlaintext Whether to include plaintext on each record
|
|
278
|
+
* @returns The records
|
|
279
|
+
*/
|
|
280
|
+
async requestRecords(program) {
|
|
281
|
+
if (!this._publicKey || !this.account) {
|
|
282
|
+
throw new WalletNotConnectedError();
|
|
283
|
+
}
|
|
284
|
+
try {
|
|
285
|
+
const result = await getRecords({
|
|
286
|
+
filter: {
|
|
287
|
+
programIds: [program],
|
|
288
|
+
status: "All"
|
|
289
|
+
},
|
|
290
|
+
address: this._publicKey,
|
|
291
|
+
network: PUZZLE_NETWORK_MAP[this.network]
|
|
292
|
+
});
|
|
293
|
+
return result?.records || [];
|
|
294
|
+
} catch (error) {
|
|
295
|
+
throw new WalletError(error instanceof Error ? error.message : "Failed to request records");
|
|
296
|
+
}
|
|
297
|
+
}
|
|
298
|
+
/**
|
|
299
|
+
* Switch the network
|
|
300
|
+
* @param network The network to switch to
|
|
301
|
+
*/
|
|
302
|
+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
303
|
+
async switchNetwork(_network) {
|
|
304
|
+
console.error("Puzzle Wallet does not support switching networks");
|
|
305
|
+
throw new MethodNotImplementedError("switchNetwork");
|
|
306
|
+
}
|
|
307
|
+
/**
|
|
308
|
+
* Execute a deployment
|
|
309
|
+
* @param deployment The deployment to execute
|
|
310
|
+
* @returns The executed transaction ID
|
|
311
|
+
*/
|
|
312
|
+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
313
|
+
async executeDeployment(_deployment) {
|
|
314
|
+
throw new MethodNotImplementedError("executeDeployment");
|
|
315
|
+
}
|
|
316
|
+
};
|
|
317
|
+
export {
|
|
318
|
+
PUZZLE_NETWORK_MAP,
|
|
319
|
+
PuzzleWalletAdapter
|
|
320
|
+
};
|
package/package.json
ADDED
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@provablehq/aleo-wallet-adaptor-puzzle",
|
|
3
|
+
"version": "0.1.1-alpha.0",
|
|
4
|
+
"description": "Puzzle wallet adapter for Aleo",
|
|
5
|
+
"main": "dist/index.js",
|
|
6
|
+
"module": "dist/index.mjs",
|
|
7
|
+
"types": "dist/index.d.ts",
|
|
8
|
+
"license": "GPL-3.0-or-later",
|
|
9
|
+
"files": [
|
|
10
|
+
"dist",
|
|
11
|
+
"src"
|
|
12
|
+
],
|
|
13
|
+
"dependencies": {
|
|
14
|
+
"@puzzlehq/sdk-core": "^1.0.3",
|
|
15
|
+
"@provablehq/aleo-wallet-adaptor-core": "0.1.1-alpha.0",
|
|
16
|
+
"@provablehq/aleo-types": "0.1.1-alpha.0",
|
|
17
|
+
"@provablehq/aleo-wallet-standard": "0.1.1-alpha.0"
|
|
18
|
+
},
|
|
19
|
+
"repository": {
|
|
20
|
+
"type": "git",
|
|
21
|
+
"url": "git+https://github.com/ProvableHQ/aleo-dev-toolkit.git",
|
|
22
|
+
"directory": "packages/aleo-wallet-adaptor/wallets/puzzle"
|
|
23
|
+
},
|
|
24
|
+
"homepage": "https://provable.com/",
|
|
25
|
+
"keywords": [
|
|
26
|
+
"aleo",
|
|
27
|
+
"wallet",
|
|
28
|
+
"adapter",
|
|
29
|
+
"puzzle"
|
|
30
|
+
],
|
|
31
|
+
"author": "Provable Labs",
|
|
32
|
+
"publishConfig": {
|
|
33
|
+
"access": "public"
|
|
34
|
+
},
|
|
35
|
+
"devDependencies": {
|
|
36
|
+
"rimraf": "^5.0.5",
|
|
37
|
+
"tsup": "^7.0.0",
|
|
38
|
+
"typescript": "^5.0.0"
|
|
39
|
+
},
|
|
40
|
+
"scripts": {
|
|
41
|
+
"build": "tsup",
|
|
42
|
+
"clean": "rimraf dist",
|
|
43
|
+
"dev": "tsup --watch"
|
|
44
|
+
}
|
|
45
|
+
}
|