@cartridge/controller 0.6.0 → 0.7.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/.turbo/turbo-build$colon$deps.log +52 -76
- package/.turbo/turbo-build.log +53 -77
- package/dist/controller.cjs +860 -0
- package/dist/controller.cjs.map +1 -0
- package/dist/controller.d.cts +33 -0
- package/dist/controller.d.ts +4 -4
- package/dist/controller.js +84 -68
- package/dist/controller.js.map +1 -1
- package/dist/index.cjs +2200 -0
- package/dist/index.cjs.map +1 -0
- package/dist/index.d.cts +17 -0
- package/dist/index.d.ts +12 -7
- package/dist/index.js +161 -7601
- package/dist/index.js.map +1 -1
- package/dist/lookup.cjs +59 -0
- package/dist/lookup.cjs.map +1 -0
- package/dist/lookup.d.cts +4 -0
- package/dist/lookup.js +7 -7
- package/dist/lookup.js.map +1 -1
- package/dist/{provider.js → node/index.cjs} +499 -25
- package/dist/node/index.cjs.map +1 -0
- package/dist/node/index.d.cts +56 -0
- package/dist/node/index.d.ts +56 -0
- package/dist/{telegram/provider.js → node/index.js} +354 -135
- package/dist/node/index.js.map +1 -0
- package/dist/{policies.d.ts → policies-DD1aPjQ4.d.cts} +6 -4
- package/dist/policies-DD1aPjQ4.d.ts +21 -0
- package/dist/{types-CVnDQVqD.d.ts → provider-ap1C1ypF.d.cts} +27 -10
- package/dist/provider-ap1C1ypF.d.ts +201 -0
- package/dist/session/{provider.js → index.cjs} +82 -73
- package/dist/session/index.cjs.map +1 -0
- package/dist/session/{provider.d.ts → index.d.cts} +6 -3
- package/dist/session/index.d.ts +37 -8
- package/dist/session/index.js +47 -64
- package/dist/session/index.js.map +1 -1
- package/package.json +35 -16
- package/src/controller.ts +43 -15
- package/src/iframe/base.ts +1 -11
- package/src/node/account.ts +72 -0
- package/src/node/backend.ts +159 -0
- package/src/node/index.ts +4 -0
- package/src/node/provider.ts +178 -0
- package/src/node/server.ts +89 -0
- package/src/session/account.ts +1 -1
- package/src/session/provider.ts +0 -16
- package/src/types.ts +3 -6
- package/tsconfig.json +2 -1
- package/dist/__tests__/parseChainId.test.d.ts +0 -2
- package/dist/__tests__/parseChainId.test.js +0 -89
- package/dist/__tests__/parseChainId.test.js.map +0 -1
- package/dist/account.d.ts +0 -38
- package/dist/account.js +0 -110
- package/dist/account.js.map +0 -1
- package/dist/constants.d.ts +0 -5
- package/dist/constants.js +0 -10
- package/dist/constants.js.map +0 -1
- package/dist/errors.d.ts +0 -5
- package/dist/errors.js +0 -11
- package/dist/errors.js.map +0 -1
- package/dist/icon.d.ts +0 -3
- package/dist/icon.js +0 -6
- package/dist/icon.js.map +0 -1
- package/dist/iframe/base.d.ts +0 -5
- package/dist/iframe/base.js +0 -126
- package/dist/iframe/base.js.map +0 -1
- package/dist/iframe/index.d.ts +0 -5
- package/dist/iframe/index.js +0 -188
- package/dist/iframe/index.js.map +0 -1
- package/dist/iframe/keychain.d.ts +0 -5
- package/dist/iframe/keychain.js +0 -147
- package/dist/iframe/keychain.js.map +0 -1
- package/dist/iframe/profile.d.ts +0 -5
- package/dist/iframe/profile.js +0 -167
- package/dist/iframe/profile.js.map +0 -1
- package/dist/index.d-BbTUPBeO.d.ts +0 -68
- package/dist/mutex.d.ts +0 -14
- package/dist/mutex.js +0 -22
- package/dist/mutex.js.map +0 -1
- package/dist/policies.js +0 -26
- package/dist/policies.js.map +0 -1
- package/dist/provider.d.ts +0 -24
- package/dist/provider.js.map +0 -1
- package/dist/session/account.d.ts +0 -37
- package/dist/session/account.js +0 -94
- package/dist/session/account.js.map +0 -1
- package/dist/session/backend.d.ts +0 -60
- package/dist/session/backend.js +0 -39
- package/dist/session/backend.js.map +0 -1
- package/dist/session/provider.js.map +0 -1
- package/dist/telegram/backend.d.ts +0 -33
- package/dist/telegram/backend.js +0 -40
- package/dist/telegram/backend.js.map +0 -1
- package/dist/telegram/provider.d.ts +0 -27
- package/dist/telegram/provider.js.map +0 -1
- package/dist/types.d.ts +0 -5
- package/dist/types.js +0 -13
- package/dist/types.js.map +0 -1
- package/dist/utils.d.ts +0 -19
- package/dist/utils.js +0 -141
- package/dist/utils.js.map +0 -1
|
@@ -0,0 +1,860 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
var starknet = require('starknet');
|
|
4
|
+
var penpal = require('@cartridge/penpal');
|
|
5
|
+
var typesJs = require('@starknet-io/types-js');
|
|
6
|
+
|
|
7
|
+
// src/account.ts
|
|
8
|
+
function toArray(val) {
|
|
9
|
+
return Array.isArray(val) ? val : [val];
|
|
10
|
+
}
|
|
11
|
+
function parseChainId(url) {
|
|
12
|
+
const parts = url.pathname.split("/");
|
|
13
|
+
if (parts.includes("starknet")) {
|
|
14
|
+
if (parts.includes("mainnet")) {
|
|
15
|
+
return starknet.constants.StarknetChainId.SN_MAIN;
|
|
16
|
+
} else if (parts.includes("sepolia")) {
|
|
17
|
+
return starknet.constants.StarknetChainId.SN_SEPOLIA;
|
|
18
|
+
}
|
|
19
|
+
} else if (parts.length >= 3) {
|
|
20
|
+
const projectName = parts[2];
|
|
21
|
+
if (parts.includes("katana")) {
|
|
22
|
+
return starknet.shortString.encodeShortString(
|
|
23
|
+
`WP_${projectName.toUpperCase().replace(/-/g, "_")}`
|
|
24
|
+
);
|
|
25
|
+
} else if (parts.includes("mainnet")) {
|
|
26
|
+
return starknet.shortString.encodeShortString(
|
|
27
|
+
`GG_${projectName.toUpperCase().replace(/-/g, "_")}`
|
|
28
|
+
);
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
throw new Error(`Chain ${url.toString()} not supported`);
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
// src/account.ts
|
|
35
|
+
var ControllerAccount = class extends starknet.WalletAccount {
|
|
36
|
+
address;
|
|
37
|
+
keychain;
|
|
38
|
+
modal;
|
|
39
|
+
options;
|
|
40
|
+
constructor(provider, rpcUrl, address, keychain, options, modal) {
|
|
41
|
+
super({ nodeUrl: rpcUrl }, provider);
|
|
42
|
+
this.address = address;
|
|
43
|
+
this.keychain = keychain;
|
|
44
|
+
this.options = options;
|
|
45
|
+
this.modal = modal;
|
|
46
|
+
}
|
|
47
|
+
/**
|
|
48
|
+
* Invoke execute function in account contract
|
|
49
|
+
*
|
|
50
|
+
* @param calls the invocation object or an array of them, containing:
|
|
51
|
+
* - contractAddress - the address of the contract
|
|
52
|
+
* - entrypoint - the entrypoint of the contract
|
|
53
|
+
* - calldata - (defaults to []) the calldata
|
|
54
|
+
* - signature - (defaults to []) the signature
|
|
55
|
+
* @param abis (optional) the abi of the contract for better displaying
|
|
56
|
+
*
|
|
57
|
+
* @returns response from addTransaction
|
|
58
|
+
*/
|
|
59
|
+
async execute(calls) {
|
|
60
|
+
calls = toArray(calls);
|
|
61
|
+
return new Promise(async (resolve, reject) => {
|
|
62
|
+
const sessionExecute = await this.keychain.execute(
|
|
63
|
+
calls,
|
|
64
|
+
undefined,
|
|
65
|
+
undefined,
|
|
66
|
+
false
|
|
67
|
+
);
|
|
68
|
+
if (sessionExecute.code === "SUCCESS" /* SUCCESS */) {
|
|
69
|
+
resolve(sessionExecute);
|
|
70
|
+
return;
|
|
71
|
+
}
|
|
72
|
+
if (this.options?.propagateSessionErrors) {
|
|
73
|
+
reject(sessionExecute.error);
|
|
74
|
+
return;
|
|
75
|
+
}
|
|
76
|
+
this.modal.open();
|
|
77
|
+
const manualExecute = await this.keychain.execute(
|
|
78
|
+
calls,
|
|
79
|
+
undefined,
|
|
80
|
+
undefined,
|
|
81
|
+
true,
|
|
82
|
+
sessionExecute.error
|
|
83
|
+
);
|
|
84
|
+
if (manualExecute.code === "SUCCESS" /* SUCCESS */) {
|
|
85
|
+
resolve(manualExecute);
|
|
86
|
+
this.modal.close();
|
|
87
|
+
return;
|
|
88
|
+
}
|
|
89
|
+
reject(manualExecute.error);
|
|
90
|
+
return;
|
|
91
|
+
});
|
|
92
|
+
}
|
|
93
|
+
/**
|
|
94
|
+
* Sign an JSON object for off-chain usage with the starknet private key and return the signature
|
|
95
|
+
* This adds a message prefix so it cant be interchanged with transactions
|
|
96
|
+
*
|
|
97
|
+
* @param json - JSON object to be signed
|
|
98
|
+
* @returns the signature of the JSON object
|
|
99
|
+
* @throws {Error} if the JSON object is not a valid JSON
|
|
100
|
+
*/
|
|
101
|
+
async signMessage(typedData2) {
|
|
102
|
+
return new Promise(async (resolve, reject) => {
|
|
103
|
+
const sessionSign = await this.keychain.signMessage(typedData2, "", true);
|
|
104
|
+
if (!("code" in sessionSign)) {
|
|
105
|
+
resolve(sessionSign);
|
|
106
|
+
return;
|
|
107
|
+
}
|
|
108
|
+
this.modal.open();
|
|
109
|
+
const manualSign = await this.keychain.signMessage(typedData2, "", false);
|
|
110
|
+
if (!("code" in manualSign)) {
|
|
111
|
+
resolve(manualSign);
|
|
112
|
+
} else {
|
|
113
|
+
reject(manualSign.error);
|
|
114
|
+
}
|
|
115
|
+
this.modal.close();
|
|
116
|
+
});
|
|
117
|
+
}
|
|
118
|
+
};
|
|
119
|
+
var account_default = ControllerAccount;
|
|
120
|
+
var IFrame = class {
|
|
121
|
+
url;
|
|
122
|
+
iframe;
|
|
123
|
+
container;
|
|
124
|
+
onClose;
|
|
125
|
+
constructor({
|
|
126
|
+
id,
|
|
127
|
+
url,
|
|
128
|
+
preset,
|
|
129
|
+
onClose,
|
|
130
|
+
onConnect,
|
|
131
|
+
methods = {}
|
|
132
|
+
}) {
|
|
133
|
+
if (typeof document === "undefined") {
|
|
134
|
+
return;
|
|
135
|
+
}
|
|
136
|
+
if (preset) {
|
|
137
|
+
url.searchParams.set("preset", preset);
|
|
138
|
+
}
|
|
139
|
+
this.url = url;
|
|
140
|
+
const iframe = document.createElement("iframe");
|
|
141
|
+
iframe.src = url.toString();
|
|
142
|
+
iframe.id = id;
|
|
143
|
+
iframe.style.border = "none";
|
|
144
|
+
iframe.sandbox.add("allow-forms");
|
|
145
|
+
iframe.sandbox.add("allow-popups");
|
|
146
|
+
iframe.sandbox.add("allow-scripts");
|
|
147
|
+
iframe.sandbox.add("allow-same-origin");
|
|
148
|
+
iframe.allow = "publickey-credentials-create *; publickey-credentials-get *; clipboard-write";
|
|
149
|
+
if (!!document.hasStorageAccess) {
|
|
150
|
+
iframe.sandbox.add("allow-storage-access-by-user-activation");
|
|
151
|
+
}
|
|
152
|
+
const container = document.createElement("div");
|
|
153
|
+
container.id = "controller";
|
|
154
|
+
container.style.position = "fixed";
|
|
155
|
+
container.style.height = "100%";
|
|
156
|
+
container.style.width = "100%";
|
|
157
|
+
container.style.top = "0";
|
|
158
|
+
container.style.left = "0";
|
|
159
|
+
container.style.zIndex = "10000";
|
|
160
|
+
container.style.backgroundColor = "rgba(0,0,0,0.6)";
|
|
161
|
+
container.style.display = "flex";
|
|
162
|
+
container.style.alignItems = "center";
|
|
163
|
+
container.style.justifyContent = "center";
|
|
164
|
+
container.style.visibility = "hidden";
|
|
165
|
+
container.style.opacity = "0";
|
|
166
|
+
container.style.transition = "opacity 0.2s ease";
|
|
167
|
+
container.appendChild(iframe);
|
|
168
|
+
this.iframe = iframe;
|
|
169
|
+
this.container = container;
|
|
170
|
+
penpal.connectToChild({
|
|
171
|
+
iframe: this.iframe,
|
|
172
|
+
methods: { close: () => this.close(), ...methods }
|
|
173
|
+
}).promise.then(onConnect);
|
|
174
|
+
this.resize();
|
|
175
|
+
window.addEventListener("resize", () => this.resize());
|
|
176
|
+
const observer = new MutationObserver(() => {
|
|
177
|
+
const existingController2 = document.getElementById("controller");
|
|
178
|
+
if (document.body) {
|
|
179
|
+
if (id === "controller-keychain" && !existingController2 || id === "controller-profile") {
|
|
180
|
+
document.body.appendChild(container);
|
|
181
|
+
observer.disconnect();
|
|
182
|
+
}
|
|
183
|
+
}
|
|
184
|
+
});
|
|
185
|
+
observer.observe(document.documentElement, {
|
|
186
|
+
childList: true,
|
|
187
|
+
subtree: true
|
|
188
|
+
});
|
|
189
|
+
const existingController = document.getElementById("controller");
|
|
190
|
+
if (document.body) {
|
|
191
|
+
if (id === "controller-keychain" && !existingController || id === "controller-profile") {
|
|
192
|
+
document.body.appendChild(container);
|
|
193
|
+
}
|
|
194
|
+
}
|
|
195
|
+
this.onClose = onClose;
|
|
196
|
+
}
|
|
197
|
+
open() {
|
|
198
|
+
if (!this.container) return;
|
|
199
|
+
document.body.style.overflow = "hidden";
|
|
200
|
+
this.container.style.visibility = "visible";
|
|
201
|
+
this.container.style.opacity = "1";
|
|
202
|
+
}
|
|
203
|
+
close() {
|
|
204
|
+
if (!this.container) return;
|
|
205
|
+
this.onClose?.();
|
|
206
|
+
document.body.style.overflow = "auto";
|
|
207
|
+
this.container.style.visibility = "hidden";
|
|
208
|
+
this.container.style.opacity = "0";
|
|
209
|
+
}
|
|
210
|
+
sendBackward() {
|
|
211
|
+
if (!this.container) return;
|
|
212
|
+
this.container.style.zIndex = "9999";
|
|
213
|
+
}
|
|
214
|
+
sendForward() {
|
|
215
|
+
if (!this.container) return;
|
|
216
|
+
this.container.style.zIndex = "10000";
|
|
217
|
+
}
|
|
218
|
+
resize() {
|
|
219
|
+
if (!this.iframe) return;
|
|
220
|
+
this.iframe.style.userSelect = "none";
|
|
221
|
+
if (window.innerWidth < 768) {
|
|
222
|
+
this.iframe.style.height = "100%";
|
|
223
|
+
this.iframe.style.width = "100%";
|
|
224
|
+
this.iframe.style.borderRadius = "0";
|
|
225
|
+
return;
|
|
226
|
+
}
|
|
227
|
+
this.iframe.style.height = "600px";
|
|
228
|
+
this.iframe.style.width = "432px";
|
|
229
|
+
this.iframe.style.borderRadius = "8px";
|
|
230
|
+
}
|
|
231
|
+
};
|
|
232
|
+
|
|
233
|
+
// src/constants.ts
|
|
234
|
+
var KEYCHAIN_URL = "https://x.cartridge.gg";
|
|
235
|
+
var PROFILE_URL = "https://profile.cartridge.gg";
|
|
236
|
+
|
|
237
|
+
// src/iframe/keychain.ts
|
|
238
|
+
var KeychainIFrame = class extends IFrame {
|
|
239
|
+
constructor({ url, policies, ...iframeOptions }) {
|
|
240
|
+
const _url = new URL(url ?? KEYCHAIN_URL);
|
|
241
|
+
if (policies) {
|
|
242
|
+
_url.searchParams.set(
|
|
243
|
+
"policies",
|
|
244
|
+
encodeURIComponent(JSON.stringify(policies))
|
|
245
|
+
);
|
|
246
|
+
}
|
|
247
|
+
super({
|
|
248
|
+
...iframeOptions,
|
|
249
|
+
id: "controller-keychain",
|
|
250
|
+
url: _url
|
|
251
|
+
});
|
|
252
|
+
}
|
|
253
|
+
};
|
|
254
|
+
|
|
255
|
+
// src/iframe/profile.ts
|
|
256
|
+
var ProfileIFrame = class extends IFrame {
|
|
257
|
+
constructor({
|
|
258
|
+
profileUrl,
|
|
259
|
+
rpcUrl,
|
|
260
|
+
version,
|
|
261
|
+
username,
|
|
262
|
+
slot,
|
|
263
|
+
namespace,
|
|
264
|
+
tokens,
|
|
265
|
+
...iframeOptions
|
|
266
|
+
}) {
|
|
267
|
+
const _profileUrl = (profileUrl || PROFILE_URL).replace(/\/$/, "");
|
|
268
|
+
let _url = new URL(
|
|
269
|
+
slot ? namespace ? `${_profileUrl}/account/${username}/slot/${slot}?ps=${encodeURIComponent(
|
|
270
|
+
slot
|
|
271
|
+
)}&ns=${encodeURIComponent(namespace)}` : `${_profileUrl}/account/${username}/slot/${slot}?ps=${encodeURIComponent(
|
|
272
|
+
slot
|
|
273
|
+
)}` : `${_profileUrl}/account/${username}`
|
|
274
|
+
);
|
|
275
|
+
if (version) {
|
|
276
|
+
_url.searchParams.set("v", encodeURIComponent(version));
|
|
277
|
+
}
|
|
278
|
+
_url.searchParams.set("rpcUrl", encodeURIComponent(rpcUrl));
|
|
279
|
+
if (tokens?.erc20) {
|
|
280
|
+
_url.searchParams.set(
|
|
281
|
+
"erc20",
|
|
282
|
+
encodeURIComponent(tokens.erc20.toString())
|
|
283
|
+
);
|
|
284
|
+
}
|
|
285
|
+
super({
|
|
286
|
+
...iframeOptions,
|
|
287
|
+
id: "controller-profile",
|
|
288
|
+
url: _url
|
|
289
|
+
});
|
|
290
|
+
}
|
|
291
|
+
};
|
|
292
|
+
|
|
293
|
+
// src/errors.ts
|
|
294
|
+
var NotReadyToConnect = class _NotReadyToConnect extends Error {
|
|
295
|
+
constructor() {
|
|
296
|
+
super("Not ready to connect");
|
|
297
|
+
Object.setPrototypeOf(this, _NotReadyToConnect.prototype);
|
|
298
|
+
}
|
|
299
|
+
};
|
|
300
|
+
|
|
301
|
+
// package.json
|
|
302
|
+
var package_default = {
|
|
303
|
+
name: "@cartridge/controller",
|
|
304
|
+
version: "0.7.0",
|
|
305
|
+
description: "Cartridge Controller",
|
|
306
|
+
module: "dist/index.js",
|
|
307
|
+
types: "dist/index.d.ts",
|
|
308
|
+
type: "module",
|
|
309
|
+
scripts: {
|
|
310
|
+
"build:deps": "tsup",
|
|
311
|
+
build: "pnpm build:deps",
|
|
312
|
+
format: 'prettier --write "src/**/*.ts"',
|
|
313
|
+
"format:check": 'prettier --check "src/**/*.ts"',
|
|
314
|
+
test: "jest",
|
|
315
|
+
version: "pnpm pkg get version"
|
|
316
|
+
},
|
|
317
|
+
exports: {
|
|
318
|
+
".": {
|
|
319
|
+
types: "./dist/index.d.ts",
|
|
320
|
+
import: "./dist/index.js",
|
|
321
|
+
require: "./dist/index.cjs"
|
|
322
|
+
},
|
|
323
|
+
"./session": {
|
|
324
|
+
types: "./dist/session/index.d.ts",
|
|
325
|
+
import: "./dist/session/index.js",
|
|
326
|
+
require: "./dist/session/index.cjs"
|
|
327
|
+
},
|
|
328
|
+
"./session/node": {
|
|
329
|
+
types: "./dist/node/index.d.ts",
|
|
330
|
+
import: "./dist/node/index.js",
|
|
331
|
+
require: "./dist/node/index.cjs"
|
|
332
|
+
},
|
|
333
|
+
"./provider": {
|
|
334
|
+
types: "./dist/provider/index.d.ts",
|
|
335
|
+
import: "./dist/provider/index.js"
|
|
336
|
+
},
|
|
337
|
+
"./types": {
|
|
338
|
+
types: "./dist/types/index.d.ts",
|
|
339
|
+
import: "./dist/types/index.js"
|
|
340
|
+
}
|
|
341
|
+
},
|
|
342
|
+
tsup: {
|
|
343
|
+
entry: [
|
|
344
|
+
"src/index.ts",
|
|
345
|
+
"src/controller.ts",
|
|
346
|
+
"src/lookup.ts",
|
|
347
|
+
"src/session/index.ts",
|
|
348
|
+
"src/node/index.ts"
|
|
349
|
+
],
|
|
350
|
+
format: [
|
|
351
|
+
"esm",
|
|
352
|
+
"cjs"
|
|
353
|
+
],
|
|
354
|
+
splitting: false,
|
|
355
|
+
sourcemap: true,
|
|
356
|
+
clean: true,
|
|
357
|
+
dts: true,
|
|
358
|
+
treeshake: {
|
|
359
|
+
preset: "recommended"
|
|
360
|
+
},
|
|
361
|
+
exports: "named"
|
|
362
|
+
},
|
|
363
|
+
peerDependencies: {
|
|
364
|
+
starknet: "catalog:",
|
|
365
|
+
open: "^10.1.0"
|
|
366
|
+
},
|
|
367
|
+
dependencies: {
|
|
368
|
+
"@cartridge/account-wasm": "workspace:*",
|
|
369
|
+
"@cartridge/penpal": "catalog:",
|
|
370
|
+
"@starknet-io/types-js": "catalog:",
|
|
371
|
+
"@telegram-apps/sdk": "^2.4.0",
|
|
372
|
+
base64url: "catalog:",
|
|
373
|
+
"cbor-x": "^1.5.0",
|
|
374
|
+
"fast-deep-equal": "catalog:"
|
|
375
|
+
},
|
|
376
|
+
devDependencies: {
|
|
377
|
+
"@cartridge/tsconfig": "workspace:*",
|
|
378
|
+
"@types/jest": "^29.5.14",
|
|
379
|
+
"@types/node": "catalog:",
|
|
380
|
+
jest: "^29.7.0",
|
|
381
|
+
prettier: "catalog:",
|
|
382
|
+
"ts-jest": "^29.2.5",
|
|
383
|
+
tsup: "catalog:",
|
|
384
|
+
typescript: "catalog:"
|
|
385
|
+
}
|
|
386
|
+
};
|
|
387
|
+
|
|
388
|
+
// src/icon.ts
|
|
389
|
+
var icon = "";
|
|
390
|
+
|
|
391
|
+
// src/mutex.ts
|
|
392
|
+
function releaseStub() {
|
|
393
|
+
}
|
|
394
|
+
var Mutex = class {
|
|
395
|
+
m_lastPromise = Promise.resolve();
|
|
396
|
+
/**
|
|
397
|
+
* Acquire lock
|
|
398
|
+
* @param [bypass=false] option to skip lock acquisition
|
|
399
|
+
*/
|
|
400
|
+
async obtain(bypass = false) {
|
|
401
|
+
let release = releaseStub;
|
|
402
|
+
if (bypass) return release;
|
|
403
|
+
const lastPromise = this.m_lastPromise;
|
|
404
|
+
this.m_lastPromise = new Promise((resolve) => release = resolve);
|
|
405
|
+
await lastPromise;
|
|
406
|
+
return release;
|
|
407
|
+
}
|
|
408
|
+
};
|
|
409
|
+
|
|
410
|
+
// src/provider.ts
|
|
411
|
+
var mutex = new Mutex();
|
|
412
|
+
var BaseProvider = class {
|
|
413
|
+
id = "controller";
|
|
414
|
+
name = "Controller";
|
|
415
|
+
version = package_default.version;
|
|
416
|
+
icon = icon;
|
|
417
|
+
account;
|
|
418
|
+
subscriptions = [];
|
|
419
|
+
_probePromise = null;
|
|
420
|
+
async safeProbe() {
|
|
421
|
+
if (this.account) {
|
|
422
|
+
return this.account;
|
|
423
|
+
}
|
|
424
|
+
if (this._probePromise) {
|
|
425
|
+
return this._probePromise;
|
|
426
|
+
}
|
|
427
|
+
const release = await mutex.obtain();
|
|
428
|
+
return await new Promise(async (resolve) => {
|
|
429
|
+
try {
|
|
430
|
+
this._probePromise = this.probe();
|
|
431
|
+
const result = await this._probePromise;
|
|
432
|
+
resolve(result);
|
|
433
|
+
} finally {
|
|
434
|
+
this._probePromise = null;
|
|
435
|
+
}
|
|
436
|
+
}).finally(() => {
|
|
437
|
+
release();
|
|
438
|
+
});
|
|
439
|
+
}
|
|
440
|
+
request = async (call) => {
|
|
441
|
+
switch (call.type) {
|
|
442
|
+
case "wallet_getPermissions":
|
|
443
|
+
await this.safeProbe();
|
|
444
|
+
if (this.account) {
|
|
445
|
+
return [typesJs.Permission.ACCOUNTS];
|
|
446
|
+
}
|
|
447
|
+
return [];
|
|
448
|
+
case "wallet_requestAccounts": {
|
|
449
|
+
if (this.account) {
|
|
450
|
+
return [this.account.address];
|
|
451
|
+
}
|
|
452
|
+
const silentMode = call.params && call.params.silent_mode;
|
|
453
|
+
this.account = await this.safeProbe();
|
|
454
|
+
if (!this.account && !silentMode) {
|
|
455
|
+
this.account = await this.connect();
|
|
456
|
+
}
|
|
457
|
+
if (this.account) {
|
|
458
|
+
return [this.account.address];
|
|
459
|
+
}
|
|
460
|
+
return [];
|
|
461
|
+
}
|
|
462
|
+
case "wallet_watchAsset":
|
|
463
|
+
throw {
|
|
464
|
+
code: 63,
|
|
465
|
+
message: "An unexpected error occurred",
|
|
466
|
+
data: "wallet_watchAsset not implemented"
|
|
467
|
+
};
|
|
468
|
+
case "wallet_addStarknetChain": {
|
|
469
|
+
let params2 = call.params;
|
|
470
|
+
return this.addStarknetChain(params2);
|
|
471
|
+
}
|
|
472
|
+
case "wallet_switchStarknetChain": {
|
|
473
|
+
let params2 = call.params;
|
|
474
|
+
return this.switchStarknetChain(params2.chainId);
|
|
475
|
+
}
|
|
476
|
+
case "wallet_requestChainId":
|
|
477
|
+
if (!this.account) {
|
|
478
|
+
throw {
|
|
479
|
+
code: 63,
|
|
480
|
+
message: "An unexpected error occurred",
|
|
481
|
+
data: "Account not initialized"
|
|
482
|
+
};
|
|
483
|
+
}
|
|
484
|
+
return await this.account.getChainId();
|
|
485
|
+
case "wallet_deploymentData":
|
|
486
|
+
throw {
|
|
487
|
+
code: 63,
|
|
488
|
+
message: "An unexpected error occurred",
|
|
489
|
+
data: "wallet_deploymentData not implemented"
|
|
490
|
+
};
|
|
491
|
+
case "wallet_addInvokeTransaction":
|
|
492
|
+
if (!this.account) {
|
|
493
|
+
throw {
|
|
494
|
+
code: 63,
|
|
495
|
+
message: "An unexpected error occurred",
|
|
496
|
+
data: "Account not initialized"
|
|
497
|
+
};
|
|
498
|
+
}
|
|
499
|
+
let params = call.params;
|
|
500
|
+
return await this.account.execute(
|
|
501
|
+
params.calls.map((call2) => ({
|
|
502
|
+
contractAddress: call2.contract_address,
|
|
503
|
+
entrypoint: call2.entry_point,
|
|
504
|
+
calldata: call2.calldata
|
|
505
|
+
}))
|
|
506
|
+
);
|
|
507
|
+
case "wallet_addDeclareTransaction":
|
|
508
|
+
throw {
|
|
509
|
+
code: 63,
|
|
510
|
+
message: "An unexpected error occurred",
|
|
511
|
+
data: "wallet_addDeclareTransaction not implemented"
|
|
512
|
+
};
|
|
513
|
+
case "wallet_signTypedData": {
|
|
514
|
+
if (!this.account) {
|
|
515
|
+
throw {
|
|
516
|
+
code: 63,
|
|
517
|
+
message: "An unexpected error occurred",
|
|
518
|
+
data: "Account not initialized"
|
|
519
|
+
};
|
|
520
|
+
}
|
|
521
|
+
return await this.account.signMessage(call.params);
|
|
522
|
+
}
|
|
523
|
+
case "wallet_supportedSpecs":
|
|
524
|
+
return [];
|
|
525
|
+
case "wallet_supportedWalletApi":
|
|
526
|
+
return [];
|
|
527
|
+
default:
|
|
528
|
+
throw {
|
|
529
|
+
code: 63,
|
|
530
|
+
message: "An unexpected error occurred",
|
|
531
|
+
data: `Unknown RPC call type: ${call.type}`
|
|
532
|
+
};
|
|
533
|
+
}
|
|
534
|
+
};
|
|
535
|
+
on = (event, handler) => {
|
|
536
|
+
if (event !== "accountsChanged" && event !== "networkChanged") {
|
|
537
|
+
throw new Error(`Unknown event: ${event}`);
|
|
538
|
+
}
|
|
539
|
+
this.subscriptions.push({ type: event, handler });
|
|
540
|
+
};
|
|
541
|
+
off = (event, handler) => {
|
|
542
|
+
if (event !== "accountsChanged" && event !== "networkChanged") {
|
|
543
|
+
throw new Error(`Unknown event: ${event}`);
|
|
544
|
+
}
|
|
545
|
+
const idx = this.subscriptions.findIndex(
|
|
546
|
+
(sub) => sub.type === event && sub.handler === handler
|
|
547
|
+
);
|
|
548
|
+
if (idx >= 0) {
|
|
549
|
+
this.subscriptions.splice(idx, 1);
|
|
550
|
+
}
|
|
551
|
+
};
|
|
552
|
+
emitNetworkChanged(chainId) {
|
|
553
|
+
this.subscriptions.filter((sub) => sub.type === "networkChanged").forEach((sub) => {
|
|
554
|
+
sub.handler(chainId);
|
|
555
|
+
});
|
|
556
|
+
}
|
|
557
|
+
emitAccountsChanged(accounts) {
|
|
558
|
+
this.subscriptions.filter((sub) => sub.type === "accountsChanged").forEach((sub) => {
|
|
559
|
+
sub.handler(accounts);
|
|
560
|
+
});
|
|
561
|
+
}
|
|
562
|
+
};
|
|
563
|
+
|
|
564
|
+
// src/controller.ts
|
|
565
|
+
var ControllerProvider = class extends BaseProvider {
|
|
566
|
+
keychain;
|
|
567
|
+
profile;
|
|
568
|
+
options;
|
|
569
|
+
iframes;
|
|
570
|
+
selectedChain;
|
|
571
|
+
chains;
|
|
572
|
+
constructor(options) {
|
|
573
|
+
super();
|
|
574
|
+
const chains = /* @__PURE__ */ new Map();
|
|
575
|
+
for (const chain of options.chains) {
|
|
576
|
+
const url = new URL(chain.rpcUrl);
|
|
577
|
+
const chainId = parseChainId(url);
|
|
578
|
+
chains.set(chainId, chain);
|
|
579
|
+
}
|
|
580
|
+
if (options.policies?.messages?.length && options.policies.messages.length !== chains.size) {
|
|
581
|
+
console.warn(
|
|
582
|
+
"Each message policy is associated with a specific chain. The number of message policies does not match the number of chains specified - session message signing may not work on some chains."
|
|
583
|
+
);
|
|
584
|
+
}
|
|
585
|
+
this.chains = chains;
|
|
586
|
+
this.selectedChain = options.defaultChainId;
|
|
587
|
+
if (!this.chains.has(this.selectedChain)) {
|
|
588
|
+
throw new Error(
|
|
589
|
+
`Chain ${this.selectedChain} not found in configured chains`
|
|
590
|
+
);
|
|
591
|
+
}
|
|
592
|
+
this.iframes = {
|
|
593
|
+
keychain: new KeychainIFrame({
|
|
594
|
+
...options,
|
|
595
|
+
onClose: this.keychain?.reset,
|
|
596
|
+
onConnect: (keychain) => {
|
|
597
|
+
this.keychain = keychain;
|
|
598
|
+
}
|
|
599
|
+
})
|
|
600
|
+
};
|
|
601
|
+
this.options = options;
|
|
602
|
+
if (typeof window !== "undefined") {
|
|
603
|
+
window.starknet_controller = this;
|
|
604
|
+
}
|
|
605
|
+
}
|
|
606
|
+
async probe() {
|
|
607
|
+
try {
|
|
608
|
+
await this.waitForKeychain();
|
|
609
|
+
if (!this.keychain) {
|
|
610
|
+
console.error(new NotReadyToConnect().message);
|
|
611
|
+
return;
|
|
612
|
+
}
|
|
613
|
+
const response = await this.keychain.probe(this.rpcUrl());
|
|
614
|
+
let rpcUrl = response?.rpcUrl || this.rpcUrl();
|
|
615
|
+
this.account = new account_default(
|
|
616
|
+
this,
|
|
617
|
+
rpcUrl,
|
|
618
|
+
response.address,
|
|
619
|
+
this.keychain,
|
|
620
|
+
this.options,
|
|
621
|
+
this.iframes.keychain
|
|
622
|
+
);
|
|
623
|
+
} catch (e) {
|
|
624
|
+
console.error(e);
|
|
625
|
+
return;
|
|
626
|
+
}
|
|
627
|
+
if (!this.iframes.profile) {
|
|
628
|
+
const username = await this.keychain.username();
|
|
629
|
+
this.iframes.profile = new ProfileIFrame({
|
|
630
|
+
...this.options,
|
|
631
|
+
onConnect: (profile) => {
|
|
632
|
+
this.profile = profile;
|
|
633
|
+
},
|
|
634
|
+
methods: {
|
|
635
|
+
openSettings: () => this.openSettings.bind(this),
|
|
636
|
+
openPurchaseCredits: () => this.openPurchaseCredits.bind(this),
|
|
637
|
+
openExecute: () => this.openExecute.bind(this)
|
|
638
|
+
},
|
|
639
|
+
rpcUrl: this.rpcUrl(),
|
|
640
|
+
username,
|
|
641
|
+
version: this.version
|
|
642
|
+
});
|
|
643
|
+
}
|
|
644
|
+
return this.account;
|
|
645
|
+
}
|
|
646
|
+
async connect() {
|
|
647
|
+
if (this.account) {
|
|
648
|
+
return this.account;
|
|
649
|
+
}
|
|
650
|
+
if (!this.keychain || !this.iframes.keychain) {
|
|
651
|
+
console.error(new NotReadyToConnect().message);
|
|
652
|
+
return;
|
|
653
|
+
}
|
|
654
|
+
if (!!document.hasStorageAccess) {
|
|
655
|
+
const ok = await document.hasStorageAccess();
|
|
656
|
+
if (!ok) {
|
|
657
|
+
await document.requestStorageAccess();
|
|
658
|
+
}
|
|
659
|
+
}
|
|
660
|
+
this.iframes.keychain.open();
|
|
661
|
+
try {
|
|
662
|
+
let response = await this.keychain.connect(
|
|
663
|
+
this.options.policies || {},
|
|
664
|
+
this.rpcUrl()
|
|
665
|
+
);
|
|
666
|
+
if (response.code !== "SUCCESS" /* SUCCESS */) {
|
|
667
|
+
throw new Error(response.message);
|
|
668
|
+
}
|
|
669
|
+
response = response;
|
|
670
|
+
this.account = new account_default(
|
|
671
|
+
this,
|
|
672
|
+
this.rpcUrl(),
|
|
673
|
+
response.address,
|
|
674
|
+
this.keychain,
|
|
675
|
+
this.options,
|
|
676
|
+
this.iframes.keychain
|
|
677
|
+
);
|
|
678
|
+
return this.account;
|
|
679
|
+
} catch (e) {
|
|
680
|
+
console.log(e);
|
|
681
|
+
} finally {
|
|
682
|
+
this.iframes.keychain.close();
|
|
683
|
+
}
|
|
684
|
+
}
|
|
685
|
+
async switchStarknetChain(chainId) {
|
|
686
|
+
if (!this.keychain || !this.iframes.keychain) {
|
|
687
|
+
console.error(new NotReadyToConnect().message);
|
|
688
|
+
return false;
|
|
689
|
+
}
|
|
690
|
+
try {
|
|
691
|
+
this.selectedChain = chainId;
|
|
692
|
+
const response = await this.keychain.probe(this.rpcUrl());
|
|
693
|
+
if (response.rpcUrl === this.rpcUrl()) {
|
|
694
|
+
return true;
|
|
695
|
+
}
|
|
696
|
+
await this.keychain.switchChain(this.rpcUrl());
|
|
697
|
+
} catch (e) {
|
|
698
|
+
console.error(e);
|
|
699
|
+
return false;
|
|
700
|
+
}
|
|
701
|
+
this.emitNetworkChanged(chainId);
|
|
702
|
+
return true;
|
|
703
|
+
}
|
|
704
|
+
addStarknetChain(_chain) {
|
|
705
|
+
return Promise.resolve(true);
|
|
706
|
+
}
|
|
707
|
+
async disconnect() {
|
|
708
|
+
if (!this.keychain) {
|
|
709
|
+
console.error(new NotReadyToConnect().message);
|
|
710
|
+
return;
|
|
711
|
+
}
|
|
712
|
+
if (!!document.hasStorageAccess) {
|
|
713
|
+
const ok = await document.hasStorageAccess();
|
|
714
|
+
if (!ok) {
|
|
715
|
+
await document.requestStorageAccess();
|
|
716
|
+
}
|
|
717
|
+
}
|
|
718
|
+
this.account = undefined;
|
|
719
|
+
return this.keychain.disconnect();
|
|
720
|
+
}
|
|
721
|
+
async openProfile(tab = "inventory") {
|
|
722
|
+
if (!this.profile || !this.iframes.profile?.url) {
|
|
723
|
+
console.error("Profile is not ready");
|
|
724
|
+
return;
|
|
725
|
+
}
|
|
726
|
+
if (!this.account) {
|
|
727
|
+
console.error("Account is not ready");
|
|
728
|
+
return;
|
|
729
|
+
}
|
|
730
|
+
this.profile.navigate(`${this.iframes.profile.url?.pathname}/${tab}`);
|
|
731
|
+
this.iframes.profile.open();
|
|
732
|
+
}
|
|
733
|
+
async openProfileTo(to) {
|
|
734
|
+
if (!this.profile || !this.iframes.profile?.url) {
|
|
735
|
+
console.error("Profile is not ready");
|
|
736
|
+
return;
|
|
737
|
+
}
|
|
738
|
+
if (!this.account) {
|
|
739
|
+
console.error("Account is not ready");
|
|
740
|
+
return;
|
|
741
|
+
}
|
|
742
|
+
this.profile.navigate(`${this.iframes.profile.url?.pathname}/${to}`);
|
|
743
|
+
this.iframes.profile.open();
|
|
744
|
+
}
|
|
745
|
+
async openProfileAt(at) {
|
|
746
|
+
if (!this.profile || !this.iframes.profile?.url) {
|
|
747
|
+
console.error("Profile is not ready");
|
|
748
|
+
return;
|
|
749
|
+
}
|
|
750
|
+
if (!this.account) {
|
|
751
|
+
console.error("Account is not ready");
|
|
752
|
+
return;
|
|
753
|
+
}
|
|
754
|
+
this.profile.navigate(at);
|
|
755
|
+
this.iframes.profile.open();
|
|
756
|
+
}
|
|
757
|
+
async openSettings() {
|
|
758
|
+
if (!this.keychain || !this.iframes.keychain) {
|
|
759
|
+
console.error(new NotReadyToConnect().message);
|
|
760
|
+
return null;
|
|
761
|
+
}
|
|
762
|
+
if (this.iframes.profile?.sendBackward) {
|
|
763
|
+
this.iframes.profile?.sendBackward();
|
|
764
|
+
} else {
|
|
765
|
+
this.iframes.profile?.close();
|
|
766
|
+
}
|
|
767
|
+
this.iframes.keychain.open();
|
|
768
|
+
const res = await this.keychain.openSettings();
|
|
769
|
+
this.iframes.keychain.close();
|
|
770
|
+
this.iframes.profile?.sendForward?.();
|
|
771
|
+
if (res && res.code === "NOT_CONNECTED" /* NOT_CONNECTED */) {
|
|
772
|
+
return false;
|
|
773
|
+
}
|
|
774
|
+
return true;
|
|
775
|
+
}
|
|
776
|
+
revoke(origin, _policy) {
|
|
777
|
+
if (!this.keychain) {
|
|
778
|
+
console.error(new NotReadyToConnect().message);
|
|
779
|
+
return null;
|
|
780
|
+
}
|
|
781
|
+
return this.keychain.revoke(origin);
|
|
782
|
+
}
|
|
783
|
+
rpcUrl() {
|
|
784
|
+
return this.chains.get(this.selectedChain).rpcUrl;
|
|
785
|
+
}
|
|
786
|
+
username() {
|
|
787
|
+
if (!this.keychain) {
|
|
788
|
+
console.error(new NotReadyToConnect().message);
|
|
789
|
+
return;
|
|
790
|
+
}
|
|
791
|
+
return this.keychain.username();
|
|
792
|
+
}
|
|
793
|
+
openPurchaseCredits() {
|
|
794
|
+
if (!this.keychain || !this.iframes.keychain) {
|
|
795
|
+
console.error(new NotReadyToConnect().message);
|
|
796
|
+
return;
|
|
797
|
+
}
|
|
798
|
+
if (!this.iframes.profile) {
|
|
799
|
+
console.error("Profile is not ready");
|
|
800
|
+
return;
|
|
801
|
+
}
|
|
802
|
+
this.iframes.profile.close();
|
|
803
|
+
this.iframes.keychain.open();
|
|
804
|
+
this.keychain.openPurchaseCredits();
|
|
805
|
+
}
|
|
806
|
+
async openExecute(calls, chainId) {
|
|
807
|
+
if (!this.keychain || !this.iframes.keychain) {
|
|
808
|
+
console.error(new NotReadyToConnect().message);
|
|
809
|
+
return;
|
|
810
|
+
}
|
|
811
|
+
if (!this.iframes.profile) {
|
|
812
|
+
console.error("Profile is not ready");
|
|
813
|
+
return;
|
|
814
|
+
}
|
|
815
|
+
let currentChainId = this.selectedChain;
|
|
816
|
+
if (chainId) {
|
|
817
|
+
this.switchStarknetChain(chainId);
|
|
818
|
+
}
|
|
819
|
+
this.iframes.profile?.sendBackward();
|
|
820
|
+
this.iframes.keychain.open();
|
|
821
|
+
this.iframes.profile?.close();
|
|
822
|
+
const res = await this.keychain.execute(calls, undefined, undefined, true);
|
|
823
|
+
this.iframes.profile?.open();
|
|
824
|
+
this.iframes.keychain.close();
|
|
825
|
+
this.iframes.profile?.sendForward();
|
|
826
|
+
if (chainId) {
|
|
827
|
+
this.switchStarknetChain(currentChainId);
|
|
828
|
+
}
|
|
829
|
+
return !(res && (res.code === "NOT_CONNECTED" /* NOT_CONNECTED */ || res.code === "CANCELED" /* CANCELED */));
|
|
830
|
+
}
|
|
831
|
+
async delegateAccount() {
|
|
832
|
+
if (!this.keychain) {
|
|
833
|
+
console.error(new NotReadyToConnect().message);
|
|
834
|
+
return null;
|
|
835
|
+
}
|
|
836
|
+
return await this.keychain.delegateAccount();
|
|
837
|
+
}
|
|
838
|
+
waitForKeychain({
|
|
839
|
+
timeout = 5e4,
|
|
840
|
+
interval = 100
|
|
841
|
+
} = {}) {
|
|
842
|
+
return new Promise((resolve, reject) => {
|
|
843
|
+
const startTime = Date.now();
|
|
844
|
+
const id = setInterval(() => {
|
|
845
|
+
if (Date.now() - startTime > timeout) {
|
|
846
|
+
clearInterval(id);
|
|
847
|
+
reject(new Error("Timeout waiting for keychain"));
|
|
848
|
+
return;
|
|
849
|
+
}
|
|
850
|
+
if (!this.keychain) return;
|
|
851
|
+
clearInterval(id);
|
|
852
|
+
resolve();
|
|
853
|
+
}, interval);
|
|
854
|
+
});
|
|
855
|
+
}
|
|
856
|
+
};
|
|
857
|
+
|
|
858
|
+
module.exports = ControllerProvider;
|
|
859
|
+
//# sourceMappingURL=controller.cjs.map
|
|
860
|
+
//# sourceMappingURL=controller.cjs.map
|