@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.
Files changed (56) hide show
  1. package/.turbo/turbo-build$colon$deps.log +77 -110
  2. package/.turbo/turbo-build.log +78 -111
  3. package/dist/__tests__/parseChainId.test.d.ts +2 -0
  4. package/dist/__tests__/parseChainId.test.js +89 -0
  5. package/dist/__tests__/parseChainId.test.js.map +1 -0
  6. package/dist/account.js +6 -0
  7. package/dist/account.js.map +1 -1
  8. package/dist/controller.d.ts +1 -1
  9. package/dist/controller.js +218 -136
  10. package/dist/controller.js.map +1 -1
  11. package/dist/iframe/base.js +4 -0
  12. package/dist/iframe/base.js.map +1 -1
  13. package/dist/iframe/index.js +4 -0
  14. package/dist/iframe/index.js.map +1 -1
  15. package/dist/iframe/keychain.js +4 -0
  16. package/dist/iframe/keychain.js.map +1 -1
  17. package/dist/iframe/profile.js +4 -0
  18. package/dist/iframe/profile.js.map +1 -1
  19. package/dist/index.d.ts +1 -0
  20. package/dist/index.js +222 -138
  21. package/dist/index.js.map +1 -1
  22. package/dist/mutex.d.ts +14 -0
  23. package/dist/mutex.js +22 -0
  24. package/dist/mutex.js.map +1 -0
  25. package/dist/policies.d.ts +19 -0
  26. package/dist/policies.js +26 -0
  27. package/dist/policies.js.map +1 -0
  28. package/dist/provider.d.ts +2 -0
  29. package/dist/provider.js +167 -109
  30. package/dist/provider.js.map +1 -1
  31. package/dist/session/account.js +4 -0
  32. package/dist/session/account.js.map +1 -1
  33. package/dist/session/index.d.ts +1 -0
  34. package/dist/session/index.js +292 -121
  35. package/dist/session/index.js.map +1 -1
  36. package/dist/session/provider.d.ts +7 -2
  37. package/dist/session/provider.js +292 -121
  38. package/dist/session/provider.js.map +1 -1
  39. package/dist/telegram/provider.d.ts +2 -1
  40. package/dist/telegram/provider.js +204 -112
  41. package/dist/telegram/provider.js.map +1 -1
  42. package/dist/utils.d.ts +5 -3
  43. package/dist/utils.js +29 -2
  44. package/dist/utils.js.map +1 -1
  45. package/jest.config.ts +13 -0
  46. package/package.json +26 -8
  47. package/src/__tests__/parseChainId.test.ts +60 -0
  48. package/src/controller.ts +25 -29
  49. package/src/mutex.ts +22 -0
  50. package/src/policies.ts +49 -0
  51. package/src/provider.ts +33 -2
  52. package/src/session/account.ts +1 -0
  53. package/src/session/provider.ts +139 -10
  54. package/src/telegram/provider.ts +3 -2
  55. package/src/utils.ts +32 -1
  56. package/tsconfig.json +1 -2
@@ -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
  }
@@ -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.5.8",
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
- ".": "./dist/index.js",
311
- "./session": "./dist/session/index.js",
312
- "./provider": "./dist/provider/index.js",
313
- "./types": "./dist/types/index.js"
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
- constructor() {
350
- this.id = "controller";
351
- this.name = "Controller";
352
- this.version = package_default.version;
353
- this.icon = icon;
354
- this.subscriptions = [];
355
- this.request = async (call) => {
356
- switch (call.type) {
357
- case "wallet_getPermissions":
358
- await this.probe();
359
- if (this.account) {
360
- return [Permission.ACCOUNTS];
361
- }
362
- return [];
363
- case "wallet_requestAccounts": {
364
- if (this.account) {
365
- return [this.account.address];
366
- }
367
- const silentMode = call.params && call.params.silent_mode;
368
- this.account = await this.probe();
369
- if (!this.account && !silentMode) {
370
- this.account = await this.connect();
371
- }
372
- if (this.account) {
373
- return [this.account.address];
374
- }
375
- return [];
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
- case "wallet_watchAsset":
378
- throw {
379
- code: 63,
380
- message: "An unexpected error occurred",
381
- data: "wallet_watchAsset not implemented"
382
- };
383
- case "wallet_addStarknetChain": {
384
- let params2 = call.params;
385
- return this.addStarknetChain(params2);
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
- case "wallet_switchStarknetChain": {
388
- let params2 = call.params;
389
- return this.switchStarknetChain(params2.chainId);
463
+ if (this.account) {
464
+ return [this.account.address];
390
465
  }
391
- case "wallet_requestChainId":
392
- if (!this.account) {
393
- throw {
394
- code: 63,
395
- message: "An unexpected error occurred",
396
- data: "Account not initialized"
397
- };
398
- }
399
- return await this.account.getChainId();
400
- case "wallet_deploymentData":
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: "wallet_deploymentData not implemented"
487
+ data: "Account not initialized"
405
488
  };
406
- case "wallet_addInvokeTransaction":
407
- if (!this.account) {
408
- throw {
409
- code: 63,
410
- message: "An unexpected error occurred",
411
- data: "Account not initialized"
412
- };
413
- }
414
- let params = call.params;
415
- return await this.account.execute(
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: "wallet_addDeclareTransaction not implemented"
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
- case "wallet_supportedSpecs":
439
- return [];
440
- case "wallet_supportedWalletApi":
441
- return [];
442
- default:
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: `Unknown RPC call type: ${call.type}`
524
+ data: "Account not initialized"
447
525
  };
526
+ }
527
+ return await this.account.signMessage(call.params);
448
528
  }
449
- };
450
- this.on = (event, handler) => {
451
- if (event !== "accountsChanged" && event !== "networkChanged") {
452
- throw new Error(`Unknown event: ${event}`);
453
- }
454
- this.subscriptions.push({ type: event, handler });
455
- };
456
- this.off = (event, handler) => {
457
- if (event !== "accountsChanged" && event !== "networkChanged") {
458
- throw new Error(`Unknown event: ${event}`);
459
- }
460
- const idx = this.subscriptions.findIndex(
461
- (sub) => sub.type === event && sub.handler === handler
462
- );
463
- if (idx >= 0) {
464
- this.subscriptions.splice(idx, 1);
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 parts = url.pathname.split("/");
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
- if (this.iframes.profile?.sendBackward) {
721
- this.iframes.profile?.sendBackward();
722
- } else {
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
- return !(res && res.code === "NOT_CONNECTED" /* NOT_CONNECTED */);
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) {