@cartridge/controller 0.5.8 → 0.6.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 +77 -110
- package/.turbo/turbo-build.log +78 -111
- package/dist/__tests__/parseChainId.test.d.ts +2 -0
- package/dist/__tests__/parseChainId.test.js +89 -0
- package/dist/__tests__/parseChainId.test.js.map +1 -0
- package/dist/account.js +6 -0
- package/dist/account.js.map +1 -1
- package/dist/controller.d.ts +1 -1
- package/dist/controller.js +218 -136
- package/dist/controller.js.map +1 -1
- package/dist/iframe/base.js +4 -0
- package/dist/iframe/base.js.map +1 -1
- package/dist/iframe/index.js +4 -0
- package/dist/iframe/index.js.map +1 -1
- package/dist/iframe/keychain.js +4 -0
- package/dist/iframe/keychain.js.map +1 -1
- package/dist/iframe/profile.js +4 -0
- package/dist/iframe/profile.js.map +1 -1
- package/dist/index.d.ts +1 -0
- package/dist/index.js +222 -138
- package/dist/index.js.map +1 -1
- package/dist/mutex.d.ts +14 -0
- package/dist/mutex.js +22 -0
- package/dist/mutex.js.map +1 -0
- package/dist/policies.d.ts +19 -0
- package/dist/policies.js +26 -0
- package/dist/policies.js.map +1 -0
- package/dist/provider.d.ts +2 -0
- package/dist/provider.js +167 -109
- package/dist/provider.js.map +1 -1
- package/dist/session/account.js +4 -0
- package/dist/session/account.js.map +1 -1
- package/dist/session/index.d.ts +1 -0
- package/dist/session/index.js +292 -121
- package/dist/session/index.js.map +1 -1
- package/dist/session/provider.d.ts +7 -2
- package/dist/session/provider.js +292 -121
- package/dist/session/provider.js.map +1 -1
- package/dist/telegram/provider.d.ts +2 -1
- package/dist/telegram/provider.js +204 -112
- package/dist/telegram/provider.js.map +1 -1
- package/dist/utils.d.ts +5 -3
- package/dist/utils.js +29 -2
- package/dist/utils.js.map +1 -1
- package/jest.config.ts +13 -0
- package/package.json +26 -8
- package/src/__tests__/parseChainId.test.ts +60 -0
- package/src/controller.ts +25 -29
- package/src/mutex.ts +22 -0
- package/src/policies.ts +49 -0
- package/src/provider.ts +33 -2
- package/src/session/account.ts +1 -0
- package/src/session/provider.ts +139 -10
- package/src/telegram/provider.ts +3 -2
- package/src/utils.ts +32 -1
- package/tsconfig.json +1 -2
package/dist/controller.d.ts
CHANGED
|
@@ -25,7 +25,7 @@ declare class ControllerProvider extends BaseProvider {
|
|
|
25
25
|
username(): Promise<string> | undefined;
|
|
26
26
|
fetchControllers(contractAddresses: string[]): Promise<Record<string, string>>;
|
|
27
27
|
openPurchaseCredits(): void;
|
|
28
|
-
openExecute(calls: any): Promise<boolean | undefined>;
|
|
28
|
+
openExecute(calls: any, chainId?: string): Promise<boolean | undefined>;
|
|
29
29
|
delegateAccount(): Promise<string | null>;
|
|
30
30
|
private waitForKeychain;
|
|
31
31
|
}
|
package/dist/controller.js
CHANGED
|
@@ -7,17 +7,45 @@ import {
|
|
|
7
7
|
import {
|
|
8
8
|
addAddressPadding,
|
|
9
9
|
CallData,
|
|
10
|
+
constants,
|
|
10
11
|
getChecksumAddress,
|
|
11
12
|
hash,
|
|
13
|
+
shortString,
|
|
12
14
|
typedData,
|
|
13
15
|
TypedDataRevision
|
|
14
16
|
} from "starknet";
|
|
15
17
|
function toArray(val) {
|
|
16
18
|
return Array.isArray(val) ? val : [val];
|
|
17
19
|
}
|
|
20
|
+
function parseChainId(url) {
|
|
21
|
+
const parts = url.pathname.split("/");
|
|
22
|
+
if (parts.includes("starknet")) {
|
|
23
|
+
if (parts.includes("mainnet")) {
|
|
24
|
+
return constants.StarknetChainId.SN_MAIN;
|
|
25
|
+
} else if (parts.includes("sepolia")) {
|
|
26
|
+
return constants.StarknetChainId.SN_SEPOLIA;
|
|
27
|
+
}
|
|
28
|
+
} else if (parts.length >= 3) {
|
|
29
|
+
const projectName = parts[2];
|
|
30
|
+
if (parts.includes("katana")) {
|
|
31
|
+
return shortString.encodeShortString(
|
|
32
|
+
`WP_${projectName.toUpperCase().replace(/-/g, "_")}`
|
|
33
|
+
);
|
|
34
|
+
} else if (parts.includes("mainnet")) {
|
|
35
|
+
return shortString.encodeShortString(
|
|
36
|
+
`GG_${projectName.toUpperCase().replace(/-/g, "_")}`
|
|
37
|
+
);
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
throw new Error(`Chain ${url.toString()} not supported`);
|
|
41
|
+
}
|
|
18
42
|
|
|
19
43
|
// src/account.ts
|
|
20
44
|
var ControllerAccount = class extends WalletAccount {
|
|
45
|
+
address;
|
|
46
|
+
keychain;
|
|
47
|
+
modal;
|
|
48
|
+
options;
|
|
21
49
|
constructor(provider, rpcUrl, address, keychain, options, modal) {
|
|
22
50
|
super({ nodeUrl: rpcUrl }, provider);
|
|
23
51
|
this.address = address;
|
|
@@ -102,6 +130,10 @@ var account_default = ControllerAccount;
|
|
|
102
130
|
// src/iframe/base.ts
|
|
103
131
|
import { connectToChild } from "@cartridge/penpal";
|
|
104
132
|
var IFrame = class {
|
|
133
|
+
url;
|
|
134
|
+
iframe;
|
|
135
|
+
container;
|
|
136
|
+
onClose;
|
|
105
137
|
constructor({
|
|
106
138
|
id,
|
|
107
139
|
url,
|
|
@@ -294,7 +326,7 @@ import {
|
|
|
294
326
|
// package.json
|
|
295
327
|
var package_default = {
|
|
296
328
|
name: "@cartridge/controller",
|
|
297
|
-
version: "0.
|
|
329
|
+
version: "0.6.0",
|
|
298
330
|
description: "Cartridge Controller",
|
|
299
331
|
module: "dist/index.js",
|
|
300
332
|
types: "dist/index.d.ts",
|
|
@@ -304,13 +336,26 @@ var package_default = {
|
|
|
304
336
|
build: "pnpm build:deps",
|
|
305
337
|
format: 'prettier --write "src/**/*.ts"',
|
|
306
338
|
"format:check": 'prettier --check "src/**/*.ts"',
|
|
339
|
+
test: "jest",
|
|
307
340
|
version: "pnpm pkg get version"
|
|
308
341
|
},
|
|
309
342
|
exports: {
|
|
310
|
-
".":
|
|
311
|
-
|
|
312
|
-
|
|
313
|
-
|
|
343
|
+
".": {
|
|
344
|
+
types: "./dist/index.d.ts",
|
|
345
|
+
default: "./dist/index.js"
|
|
346
|
+
},
|
|
347
|
+
"./session": {
|
|
348
|
+
types: "./dist/session/index.d.ts",
|
|
349
|
+
default: "./dist/session/index.js"
|
|
350
|
+
},
|
|
351
|
+
"./provider": {
|
|
352
|
+
types: "./dist/provider/index.d.ts",
|
|
353
|
+
default: "./dist/provider/index.js"
|
|
354
|
+
},
|
|
355
|
+
"./types": {
|
|
356
|
+
types: "./dist/types/index.d.ts",
|
|
357
|
+
default: "./dist/types/index.js"
|
|
358
|
+
}
|
|
314
359
|
},
|
|
315
360
|
tsup: {
|
|
316
361
|
entry: [
|
|
@@ -323,6 +368,9 @@ var package_default = {
|
|
|
323
368
|
sourcemap: true,
|
|
324
369
|
clean: true
|
|
325
370
|
},
|
|
371
|
+
peerDependencies: {
|
|
372
|
+
starknet: "^6.21.0"
|
|
373
|
+
},
|
|
326
374
|
dependencies: {
|
|
327
375
|
"@cartridge/account-wasm": "workspace:*",
|
|
328
376
|
"@cartridge/penpal": "^6.2.3",
|
|
@@ -331,12 +379,14 @@ var package_default = {
|
|
|
331
379
|
base64url: "^3.0.1",
|
|
332
380
|
"cbor-x": "^1.5.0",
|
|
333
381
|
"fast-deep-equal": "^3.1.3",
|
|
334
|
-
"query-string": "^7.1.1"
|
|
335
|
-
starknet: "^6.11.0"
|
|
382
|
+
"query-string": "^7.1.1"
|
|
336
383
|
},
|
|
337
384
|
devDependencies: {
|
|
338
385
|
"@cartridge/tsconfig": "workspace:*",
|
|
386
|
+
"@types/jest": "^29.5.14",
|
|
339
387
|
"@types/node": "^20.6.0",
|
|
388
|
+
jest: "^29.7.0",
|
|
389
|
+
"ts-jest": "^29.2.5",
|
|
340
390
|
typescript: "^5.4.5"
|
|
341
391
|
}
|
|
342
392
|
};
|
|
@@ -344,127 +394,167 @@ var package_default = {
|
|
|
344
394
|
// src/icon.ts
|
|
345
395
|
var icon = "";
|
|
346
396
|
|
|
397
|
+
// src/mutex.ts
|
|
398
|
+
function releaseStub() {
|
|
399
|
+
}
|
|
400
|
+
var Mutex = class {
|
|
401
|
+
m_lastPromise = Promise.resolve();
|
|
402
|
+
/**
|
|
403
|
+
* Acquire lock
|
|
404
|
+
* @param [bypass=false] option to skip lock acquisition
|
|
405
|
+
*/
|
|
406
|
+
async obtain(bypass = false) {
|
|
407
|
+
let release = releaseStub;
|
|
408
|
+
if (bypass) return release;
|
|
409
|
+
const lastPromise = this.m_lastPromise;
|
|
410
|
+
this.m_lastPromise = new Promise((resolve) => release = resolve);
|
|
411
|
+
await lastPromise;
|
|
412
|
+
return release;
|
|
413
|
+
}
|
|
414
|
+
};
|
|
415
|
+
|
|
347
416
|
// src/provider.ts
|
|
417
|
+
var mutex = new Mutex();
|
|
348
418
|
var BaseProvider = class {
|
|
349
|
-
|
|
350
|
-
|
|
351
|
-
|
|
352
|
-
|
|
353
|
-
|
|
354
|
-
|
|
355
|
-
|
|
356
|
-
|
|
357
|
-
|
|
358
|
-
|
|
359
|
-
|
|
360
|
-
|
|
361
|
-
|
|
362
|
-
|
|
363
|
-
|
|
364
|
-
|
|
365
|
-
|
|
366
|
-
|
|
367
|
-
|
|
368
|
-
|
|
369
|
-
|
|
370
|
-
|
|
371
|
-
|
|
372
|
-
|
|
373
|
-
|
|
374
|
-
|
|
375
|
-
|
|
419
|
+
id = "controller";
|
|
420
|
+
name = "Controller";
|
|
421
|
+
version = package_default.version;
|
|
422
|
+
icon = icon;
|
|
423
|
+
account;
|
|
424
|
+
subscriptions = [];
|
|
425
|
+
_probePromise = null;
|
|
426
|
+
async safeProbe() {
|
|
427
|
+
if (this.account) {
|
|
428
|
+
return this.account;
|
|
429
|
+
}
|
|
430
|
+
if (this._probePromise) {
|
|
431
|
+
return this._probePromise;
|
|
432
|
+
}
|
|
433
|
+
const release = await mutex.obtain();
|
|
434
|
+
return await new Promise(async (resolve) => {
|
|
435
|
+
try {
|
|
436
|
+
this._probePromise = this.probe();
|
|
437
|
+
const result = await this._probePromise;
|
|
438
|
+
resolve(result);
|
|
439
|
+
} finally {
|
|
440
|
+
this._probePromise = null;
|
|
441
|
+
}
|
|
442
|
+
}).finally(() => {
|
|
443
|
+
release();
|
|
444
|
+
});
|
|
445
|
+
}
|
|
446
|
+
request = async (call) => {
|
|
447
|
+
switch (call.type) {
|
|
448
|
+
case "wallet_getPermissions":
|
|
449
|
+
await this.safeProbe();
|
|
450
|
+
if (this.account) {
|
|
451
|
+
return [Permission.ACCOUNTS];
|
|
376
452
|
}
|
|
377
|
-
|
|
378
|
-
|
|
379
|
-
|
|
380
|
-
|
|
381
|
-
|
|
382
|
-
|
|
383
|
-
|
|
384
|
-
|
|
385
|
-
|
|
453
|
+
return [];
|
|
454
|
+
case "wallet_requestAccounts": {
|
|
455
|
+
if (this.account) {
|
|
456
|
+
return [this.account.address];
|
|
457
|
+
}
|
|
458
|
+
const silentMode = call.params && call.params.silent_mode;
|
|
459
|
+
this.account = await this.safeProbe();
|
|
460
|
+
if (!this.account && !silentMode) {
|
|
461
|
+
this.account = await this.connect();
|
|
386
462
|
}
|
|
387
|
-
|
|
388
|
-
|
|
389
|
-
return this.switchStarknetChain(params2.chainId);
|
|
463
|
+
if (this.account) {
|
|
464
|
+
return [this.account.address];
|
|
390
465
|
}
|
|
391
|
-
|
|
392
|
-
|
|
393
|
-
|
|
394
|
-
|
|
395
|
-
|
|
396
|
-
|
|
397
|
-
|
|
398
|
-
|
|
399
|
-
|
|
400
|
-
|
|
466
|
+
return [];
|
|
467
|
+
}
|
|
468
|
+
case "wallet_watchAsset":
|
|
469
|
+
throw {
|
|
470
|
+
code: 63,
|
|
471
|
+
message: "An unexpected error occurred",
|
|
472
|
+
data: "wallet_watchAsset not implemented"
|
|
473
|
+
};
|
|
474
|
+
case "wallet_addStarknetChain": {
|
|
475
|
+
let params2 = call.params;
|
|
476
|
+
return this.addStarknetChain(params2);
|
|
477
|
+
}
|
|
478
|
+
case "wallet_switchStarknetChain": {
|
|
479
|
+
let params2 = call.params;
|
|
480
|
+
return this.switchStarknetChain(params2.chainId);
|
|
481
|
+
}
|
|
482
|
+
case "wallet_requestChainId":
|
|
483
|
+
if (!this.account) {
|
|
401
484
|
throw {
|
|
402
485
|
code: 63,
|
|
403
486
|
message: "An unexpected error occurred",
|
|
404
|
-
data: "
|
|
487
|
+
data: "Account not initialized"
|
|
405
488
|
};
|
|
406
|
-
|
|
407
|
-
|
|
408
|
-
|
|
409
|
-
|
|
410
|
-
|
|
411
|
-
|
|
412
|
-
|
|
413
|
-
|
|
414
|
-
|
|
415
|
-
|
|
416
|
-
params.calls.map((call2) => ({
|
|
417
|
-
contractAddress: call2.contract_address,
|
|
418
|
-
entrypoint: call2.entry_point,
|
|
419
|
-
calldata: call2.calldata
|
|
420
|
-
}))
|
|
421
|
-
);
|
|
422
|
-
case "wallet_addDeclareTransaction":
|
|
489
|
+
}
|
|
490
|
+
return await this.account.getChainId();
|
|
491
|
+
case "wallet_deploymentData":
|
|
492
|
+
throw {
|
|
493
|
+
code: 63,
|
|
494
|
+
message: "An unexpected error occurred",
|
|
495
|
+
data: "wallet_deploymentData not implemented"
|
|
496
|
+
};
|
|
497
|
+
case "wallet_addInvokeTransaction":
|
|
498
|
+
if (!this.account) {
|
|
423
499
|
throw {
|
|
424
500
|
code: 63,
|
|
425
501
|
message: "An unexpected error occurred",
|
|
426
|
-
data: "
|
|
502
|
+
data: "Account not initialized"
|
|
427
503
|
};
|
|
428
|
-
case "wallet_signTypedData": {
|
|
429
|
-
if (!this.account) {
|
|
430
|
-
throw {
|
|
431
|
-
code: 63,
|
|
432
|
-
message: "An unexpected error occurred",
|
|
433
|
-
data: "Account not initialized"
|
|
434
|
-
};
|
|
435
|
-
}
|
|
436
|
-
return await this.account.signMessage(call.params);
|
|
437
504
|
}
|
|
438
|
-
|
|
439
|
-
|
|
440
|
-
|
|
441
|
-
|
|
442
|
-
|
|
505
|
+
let params = call.params;
|
|
506
|
+
return await this.account.execute(
|
|
507
|
+
params.calls.map((call2) => ({
|
|
508
|
+
contractAddress: call2.contract_address,
|
|
509
|
+
entrypoint: call2.entry_point,
|
|
510
|
+
calldata: call2.calldata
|
|
511
|
+
}))
|
|
512
|
+
);
|
|
513
|
+
case "wallet_addDeclareTransaction":
|
|
514
|
+
throw {
|
|
515
|
+
code: 63,
|
|
516
|
+
message: "An unexpected error occurred",
|
|
517
|
+
data: "wallet_addDeclareTransaction not implemented"
|
|
518
|
+
};
|
|
519
|
+
case "wallet_signTypedData": {
|
|
520
|
+
if (!this.account) {
|
|
443
521
|
throw {
|
|
444
522
|
code: 63,
|
|
445
523
|
message: "An unexpected error occurred",
|
|
446
|
-
data:
|
|
524
|
+
data: "Account not initialized"
|
|
447
525
|
};
|
|
526
|
+
}
|
|
527
|
+
return await this.account.signMessage(call.params);
|
|
448
528
|
}
|
|
449
|
-
|
|
450
|
-
|
|
451
|
-
|
|
452
|
-
|
|
453
|
-
|
|
454
|
-
|
|
455
|
-
|
|
456
|
-
|
|
457
|
-
|
|
458
|
-
|
|
459
|
-
|
|
460
|
-
|
|
461
|
-
|
|
462
|
-
|
|
463
|
-
|
|
464
|
-
|
|
465
|
-
|
|
466
|
-
|
|
467
|
-
|
|
529
|
+
case "wallet_supportedSpecs":
|
|
530
|
+
return [];
|
|
531
|
+
case "wallet_supportedWalletApi":
|
|
532
|
+
return [];
|
|
533
|
+
default:
|
|
534
|
+
throw {
|
|
535
|
+
code: 63,
|
|
536
|
+
message: "An unexpected error occurred",
|
|
537
|
+
data: `Unknown RPC call type: ${call.type}`
|
|
538
|
+
};
|
|
539
|
+
}
|
|
540
|
+
};
|
|
541
|
+
on = (event, handler) => {
|
|
542
|
+
if (event !== "accountsChanged" && event !== "networkChanged") {
|
|
543
|
+
throw new Error(`Unknown event: ${event}`);
|
|
544
|
+
}
|
|
545
|
+
this.subscriptions.push({ type: event, handler });
|
|
546
|
+
};
|
|
547
|
+
off = (event, handler) => {
|
|
548
|
+
if (event !== "accountsChanged" && event !== "networkChanged") {
|
|
549
|
+
throw new Error(`Unknown event: ${event}`);
|
|
550
|
+
}
|
|
551
|
+
const idx = this.subscriptions.findIndex(
|
|
552
|
+
(sub) => sub.type === event && sub.handler === handler
|
|
553
|
+
);
|
|
554
|
+
if (idx >= 0) {
|
|
555
|
+
this.subscriptions.splice(idx, 1);
|
|
556
|
+
}
|
|
557
|
+
};
|
|
468
558
|
emitNetworkChanged(chainId) {
|
|
469
559
|
this.subscriptions.filter((sub) => sub.type === "networkChanged").forEach((sub) => {
|
|
470
560
|
sub.handler(chainId);
|
|
@@ -478,32 +568,19 @@ var BaseProvider = class {
|
|
|
478
568
|
};
|
|
479
569
|
|
|
480
570
|
// src/controller.ts
|
|
481
|
-
import { constants } from "starknet";
|
|
482
571
|
var ControllerProvider = class extends BaseProvider {
|
|
572
|
+
keychain;
|
|
573
|
+
profile;
|
|
574
|
+
options;
|
|
575
|
+
iframes;
|
|
576
|
+
selectedChain;
|
|
577
|
+
chains;
|
|
483
578
|
constructor(options) {
|
|
484
579
|
super();
|
|
485
580
|
const chains = /* @__PURE__ */ new Map();
|
|
486
581
|
for (const chain of options.chains) {
|
|
487
|
-
let chainId;
|
|
488
582
|
const url = new URL(chain.rpcUrl);
|
|
489
|
-
const
|
|
490
|
-
if (parts.includes("starknet")) {
|
|
491
|
-
if (parts.includes("mainnet")) {
|
|
492
|
-
chainId = constants.StarknetChainId.SN_MAIN;
|
|
493
|
-
} else if (parts.includes("sepolia")) {
|
|
494
|
-
chainId = constants.StarknetChainId.SN_SEPOLIA;
|
|
495
|
-
}
|
|
496
|
-
} else if (parts.length >= 3) {
|
|
497
|
-
const projectName = parts[2];
|
|
498
|
-
if (parts.includes("katana")) {
|
|
499
|
-
chainId = `WP_${projectName.toUpperCase()}`;
|
|
500
|
-
} else if (parts.includes("mainnet")) {
|
|
501
|
-
chainId = `GG_${projectName.toUpperCase()}`;
|
|
502
|
-
}
|
|
503
|
-
}
|
|
504
|
-
if (!chainId) {
|
|
505
|
-
throw new Error(`Chain ${chain.rpcUrl} not supported`);
|
|
506
|
-
}
|
|
583
|
+
const chainId = parseChainId(url);
|
|
507
584
|
chains.set(chainId, chain);
|
|
508
585
|
}
|
|
509
586
|
if (options.policies?.messages?.length && options.policies.messages.length !== chains.size) {
|
|
@@ -708,7 +785,7 @@ var ControllerProvider = class extends BaseProvider {
|
|
|
708
785
|
this.iframes.keychain.open();
|
|
709
786
|
this.keychain.openPurchaseCredits();
|
|
710
787
|
}
|
|
711
|
-
async openExecute(calls) {
|
|
788
|
+
async openExecute(calls, chainId) {
|
|
712
789
|
if (!this.keychain || !this.iframes.keychain) {
|
|
713
790
|
console.error(new NotReadyToConnect().message);
|
|
714
791
|
return;
|
|
@@ -717,16 +794,21 @@ var ControllerProvider = class extends BaseProvider {
|
|
|
717
794
|
console.error("Profile is not ready");
|
|
718
795
|
return;
|
|
719
796
|
}
|
|
720
|
-
|
|
721
|
-
|
|
722
|
-
|
|
723
|
-
this.iframes.profile?.close();
|
|
797
|
+
let currentChainId = this.selectedChain;
|
|
798
|
+
if (chainId) {
|
|
799
|
+
this.switchStarknetChain(chainId);
|
|
724
800
|
}
|
|
801
|
+
this.iframes.profile?.sendBackward();
|
|
725
802
|
this.iframes.keychain.open();
|
|
803
|
+
this.iframes.profile?.close();
|
|
726
804
|
const res = await this.keychain.execute(calls, void 0, void 0, true);
|
|
805
|
+
this.iframes.profile?.open();
|
|
727
806
|
this.iframes.keychain.close();
|
|
728
|
-
this.iframes.profile?.sendForward
|
|
729
|
-
|
|
807
|
+
this.iframes.profile?.sendForward();
|
|
808
|
+
if (chainId) {
|
|
809
|
+
this.switchStarknetChain(currentChainId);
|
|
810
|
+
}
|
|
811
|
+
return !(res && (res.code === "NOT_CONNECTED" /* NOT_CONNECTED */ || res.code === "CANCELED" /* CANCELED */));
|
|
730
812
|
}
|
|
731
813
|
async delegateAccount() {
|
|
732
814
|
if (!this.keychain) {
|