@cartridge/controller 0.7.7 → 0.7.8

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 (39) hide show
  1. package/.turbo/turbo-build$colon$deps.log +23 -23
  2. package/.turbo/turbo-build.log +31 -31
  3. package/dist/controller.cjs +472 -3
  4. package/dist/controller.cjs.map +1 -1
  5. package/dist/controller.d.cts +1 -1
  6. package/dist/controller.d.ts +1 -1
  7. package/dist/controller.js +472 -3
  8. package/dist/controller.js.map +1 -1
  9. package/dist/index.cjs +1908 -1214
  10. package/dist/index.cjs.map +1 -1
  11. package/dist/index.d.cts +46 -3
  12. package/dist/index.d.ts +46 -3
  13. package/dist/index.js +1905 -1215
  14. package/dist/index.js.map +1 -1
  15. package/dist/node/index.cjs +1 -1
  16. package/dist/node/index.cjs.map +1 -1
  17. package/dist/node/index.d.cts +2 -2
  18. package/dist/node/index.d.ts +2 -2
  19. package/dist/node/index.js +1 -1
  20. package/dist/node/index.js.map +1 -1
  21. package/dist/{provider-CTDncg8U.d.cts → provider-BeCgS86X.d.cts} +64 -1
  22. package/dist/{provider-CTDncg8U.d.ts → provider-BeCgS86X.d.ts} +64 -1
  23. package/dist/session/index.cjs +1 -1
  24. package/dist/session/index.cjs.map +1 -1
  25. package/dist/session/index.d.cts +2 -2
  26. package/dist/session/index.d.ts +2 -2
  27. package/dist/session/index.js +1 -1
  28. package/dist/session/index.js.map +1 -1
  29. package/package.json +8 -5
  30. package/src/iframe/base.ts +1 -1
  31. package/src/iframe/keychain.ts +11 -0
  32. package/src/index.ts +1 -0
  33. package/src/types.ts +24 -0
  34. package/src/wallets/argent/index.ts +122 -0
  35. package/src/wallets/bridge.ts +165 -0
  36. package/src/wallets/index.ts +5 -0
  37. package/src/wallets/metamask/index.ts +210 -0
  38. package/src/wallets/phantom/index.ts +124 -0
  39. package/src/wallets/types.ts +32 -0
@@ -2,6 +2,9 @@
2
2
 
3
3
  var starknet = require('starknet');
4
4
  var penpal = require('@cartridge/penpal');
5
+ var sdk = require('@metamask/sdk');
6
+ var starknetkit = require('starknetkit');
7
+ var injected = require('starknetkit/injected');
5
8
  var typesJs = require('@starknet-io/types-js');
6
9
 
7
10
  // src/account.ts
@@ -176,7 +179,7 @@ var IFrame = class {
176
179
  this.container = container;
177
180
  penpal.connectToChild({
178
181
  iframe: this.iframe,
179
- methods: { close: () => this.close(), ...methods }
182
+ methods: { close: (_origin) => () => this.close(), ...methods }
180
183
  }).promise.then(onConnect);
181
184
  this.resize();
182
185
  window.addEventListener("resize", () => this.resize());
@@ -240,11 +243,472 @@ var IFrame = class {
240
243
  // src/constants.ts
241
244
  var KEYCHAIN_URL = "https://x.cartridge.gg";
242
245
  var PROFILE_URL = "https://profile.cartridge.gg";
246
+ var MetaMaskWallet = class {
247
+ type = "metamask";
248
+ platform = "ethereum";
249
+ MMSDK;
250
+ account = void 0;
251
+ constructor() {
252
+ this.MMSDK = new sdk.MetaMaskSDK({
253
+ dappMetadata: {
254
+ name: "Cartridge Controller",
255
+ url: window.location.href
256
+ }
257
+ });
258
+ }
259
+ isAvailable() {
260
+ return typeof window !== "undefined" && !!window.ethereum?.isMetaMask;
261
+ }
262
+ getInfo() {
263
+ const available = this.isAvailable();
264
+ return {
265
+ type: this.type,
266
+ available,
267
+ version: available ? window.ethereum?.version || "Unknown" : void 0,
268
+ chainId: available ? window.ethereum?.chainId : void 0,
269
+ name: "MetaMask",
270
+ platform: this.platform
271
+ };
272
+ }
273
+ async connect() {
274
+ if (this.account) {
275
+ return { success: true, wallet: this.type, account: this.account };
276
+ }
277
+ try {
278
+ if (!this.isAvailable()) {
279
+ throw new Error("MetaMask is not available");
280
+ }
281
+ const accounts = await this.MMSDK.connect();
282
+ if (accounts && accounts.length > 0) {
283
+ this.account = accounts[0];
284
+ return { success: true, wallet: this.type, account: this.account };
285
+ }
286
+ throw new Error("No accounts found");
287
+ } catch (error) {
288
+ console.error(`Error connecting to MetaMask:`, error);
289
+ return {
290
+ success: false,
291
+ wallet: this.type,
292
+ error: error.message || "Unknown error"
293
+ };
294
+ }
295
+ }
296
+ async signTransaction(transaction) {
297
+ try {
298
+ if (!this.isAvailable() || !this.account) {
299
+ throw new Error("MetaMask is not connected");
300
+ }
301
+ const ethereum = this.MMSDK.getProvider();
302
+ if (!ethereum) {
303
+ throw new Error("MetaMask is not connected");
304
+ }
305
+ const result = await ethereum.request({
306
+ method: "eth_sendTransaction",
307
+ params: [transaction]
308
+ });
309
+ return { success: true, wallet: this.type, result };
310
+ } catch (error) {
311
+ console.error(`Error signing transaction with MetaMask:`, error);
312
+ return {
313
+ success: false,
314
+ wallet: this.type,
315
+ error: error.message || "Unknown error"
316
+ };
317
+ }
318
+ }
319
+ async signMessage(message) {
320
+ try {
321
+ if (!this.isAvailable() || !this.account) {
322
+ throw new Error("MetaMask is not connected");
323
+ }
324
+ const result = await this.MMSDK.connectAndSign({
325
+ msg: message
326
+ });
327
+ return { success: true, wallet: this.type, result };
328
+ } catch (error) {
329
+ console.error(`Error signing message with MetaMask:`, error);
330
+ return {
331
+ success: false,
332
+ wallet: this.type,
333
+ error: error.message || "Unknown error"
334
+ };
335
+ }
336
+ }
337
+ async signTypedData(data) {
338
+ try {
339
+ if (!this.isAvailable() || !this.account) {
340
+ throw new Error("MetaMask is not connected");
341
+ }
342
+ const ethereum = this.MMSDK.getProvider();
343
+ if (!ethereum) {
344
+ throw new Error("MetaMask is not connected");
345
+ }
346
+ const result = await ethereum.request({
347
+ method: "eth_signTypedData_v4",
348
+ params: [this.account, JSON.stringify(data)]
349
+ });
350
+ return { success: true, wallet: this.type, result };
351
+ } catch (error) {
352
+ console.error(`Error signing typed data with MetaMask:`, error);
353
+ return {
354
+ success: false,
355
+ wallet: this.type,
356
+ error: error.message || "Unknown error"
357
+ };
358
+ }
359
+ }
360
+ async switchChain(chainId) {
361
+ try {
362
+ if (!this.isAvailable()) {
363
+ throw new Error("MetaMask is not available");
364
+ }
365
+ const ethereum = this.MMSDK.getProvider();
366
+ if (!ethereum) {
367
+ throw new Error("MetaMask is not connected");
368
+ }
369
+ try {
370
+ await ethereum.request({
371
+ method: "wallet_switchEthereumChain",
372
+ params: [{ chainId }]
373
+ });
374
+ return true;
375
+ } catch (error) {
376
+ if (error.code === 4902) {
377
+ console.warn("Chain not added to MetaMask");
378
+ }
379
+ throw error;
380
+ }
381
+ } catch (error) {
382
+ console.error(`Error switching chain for MetaMask:`, error);
383
+ return false;
384
+ }
385
+ }
386
+ async getBalance(tokenAddress) {
387
+ try {
388
+ if (!this.isAvailable() || !this.account) {
389
+ throw new Error("MetaMask is not connected");
390
+ }
391
+ if (tokenAddress) {
392
+ return {
393
+ success: false,
394
+ wallet: this.type,
395
+ error: "Not implemented for ERC20"
396
+ };
397
+ } else {
398
+ const ethereum = this.MMSDK.getProvider();
399
+ if (!ethereum) {
400
+ throw new Error("MetaMask is not connected");
401
+ }
402
+ const balance = await ethereum.request({
403
+ method: "eth_getBalance",
404
+ params: [this.account, "latest"]
405
+ });
406
+ return { success: true, wallet: this.type, result: balance };
407
+ }
408
+ } catch (error) {
409
+ console.error(`Error getting balance from MetaMask:`, error);
410
+ return {
411
+ success: false,
412
+ wallet: this.type,
413
+ error: error.message || "Unknown error"
414
+ };
415
+ }
416
+ }
417
+ };
418
+
419
+ // src/wallets/phantom/index.ts
420
+ var PhantomWallet = class {
421
+ type = "phantom";
422
+ platform = "solana";
423
+ account = void 0;
424
+ isAvailable() {
425
+ return typeof window !== "undefined" && !!window.solana?.isPhantom;
426
+ }
427
+ getInfo() {
428
+ const available = this.isAvailable();
429
+ return {
430
+ type: this.type,
431
+ available,
432
+ version: "Unknown",
433
+ name: "Phantom",
434
+ platform: this.platform
435
+ };
436
+ }
437
+ async connect() {
438
+ if (this.account) {
439
+ return { success: true, wallet: this.type, account: this.account };
440
+ }
441
+ try {
442
+ if (!this.isAvailable()) {
443
+ throw new Error("Phantom is not available");
444
+ }
445
+ const response = await window.solana.connect();
446
+ if (response.publicKey) {
447
+ this.account = response.publicKey.toString();
448
+ return { success: true, wallet: this.type, account: this.account };
449
+ }
450
+ throw new Error("No accounts found");
451
+ } catch (error) {
452
+ console.error(`Error connecting to Phantom:`, error);
453
+ return {
454
+ success: false,
455
+ wallet: this.type,
456
+ error: error.message || "Unknown error"
457
+ };
458
+ }
459
+ }
460
+ async signTransaction(transaction) {
461
+ try {
462
+ if (!this.isAvailable() || !this.account) {
463
+ throw new Error("Phantom is not connected");
464
+ }
465
+ const result = await window.solana.signTransaction(transaction);
466
+ return { success: true, wallet: this.type, result };
467
+ } catch (error) {
468
+ console.error(`Error signing transaction with Phantom:`, error);
469
+ return {
470
+ success: false,
471
+ wallet: this.type,
472
+ error: error.message || "Unknown error"
473
+ };
474
+ }
475
+ }
476
+ async signMessage(message) {
477
+ try {
478
+ if (!this.isAvailable() || !this.account) {
479
+ throw new Error("Phantom is not connected");
480
+ }
481
+ const encodedMessage = new TextEncoder().encode(message);
482
+ const result = await window.solana.signMessage(encodedMessage, "utf8");
483
+ return { success: true, wallet: this.type, result };
484
+ } catch (error) {
485
+ console.error(`Error signing message with Phantom:`, error);
486
+ return {
487
+ success: false,
488
+ wallet: this.type,
489
+ error: error.message || "Unknown error"
490
+ };
491
+ }
492
+ }
493
+ async switchChain(_chainId) {
494
+ console.warn("Chain switching not supported for Phantom");
495
+ return false;
496
+ }
497
+ async getBalance(_tokenAddress) {
498
+ try {
499
+ if (!this.isAvailable() || !this.account) {
500
+ throw new Error("Phantom is not connected");
501
+ }
502
+ return {
503
+ success: true,
504
+ wallet: this.type,
505
+ result: "Implement based on Phantom API"
506
+ };
507
+ } catch (error) {
508
+ console.error(`Error getting balance from Phantom:`, error);
509
+ return {
510
+ success: false,
511
+ wallet: this.type,
512
+ error: error.message || "Unknown error"
513
+ };
514
+ }
515
+ }
516
+ };
517
+ var ArgentWallet = class {
518
+ type = "argent";
519
+ platform = "starknet";
520
+ wallet = void 0;
521
+ account = void 0;
522
+ isAvailable() {
523
+ return typeof window !== "undefined" && !!window.starknet_argentX;
524
+ }
525
+ getInfo() {
526
+ const available = this.isAvailable();
527
+ return {
528
+ type: this.type,
529
+ available,
530
+ version: available ? window.starknet_argentX?.version || "Unknown" : void 0,
531
+ chainId: available ? window.starknet_argentX?.chainId : void 0,
532
+ name: "Argent",
533
+ platform: this.platform
534
+ };
535
+ }
536
+ async connect() {
537
+ if (this.account) {
538
+ return { success: true, wallet: this.type, account: this.account };
539
+ }
540
+ try {
541
+ if (!this.isAvailable()) {
542
+ throw new Error("Argent is not available");
543
+ }
544
+ const { wallet, connectorData } = await starknetkit.connect({
545
+ connectors: [new injected.InjectedConnector({ options: { id: "argentX" } })]
546
+ });
547
+ if (!wallet) {
548
+ throw new Error("No wallet found");
549
+ }
550
+ this.wallet = wallet;
551
+ this.account = connectorData?.account;
552
+ return { success: true, wallet: this.type, account: this.account };
553
+ } catch (error) {
554
+ console.error(`Error connecting to Argent:`, error);
555
+ return {
556
+ success: false,
557
+ wallet: this.type,
558
+ error: error.message || "Unknown error"
559
+ };
560
+ }
561
+ }
562
+ async signTypedData(data) {
563
+ try {
564
+ if (!this.isAvailable() || !this.wallet) {
565
+ throw new Error("Argent is not connected");
566
+ }
567
+ console.log("signTypedData", data);
568
+ const sig = await this.wallet.request({
569
+ type: "wallet_signTypedData",
570
+ params: data
571
+ });
572
+ return { success: true, wallet: this.type, result: sig };
573
+ } catch (error) {
574
+ console.error(`Error signing typed data with Argent:`, error);
575
+ return {
576
+ success: false,
577
+ wallet: this.type,
578
+ error: error.message || "Unknown error"
579
+ };
580
+ }
581
+ }
582
+ async switchChain(_chainId) {
583
+ console.warn(
584
+ "Chain switching for Argent may require custom implementation"
585
+ );
586
+ return false;
587
+ }
588
+ async getBalance(_tokenAddress) {
589
+ try {
590
+ if (!this.isAvailable() || !this.wallet) {
591
+ throw new Error("Argent is not connected");
592
+ }
593
+ return {
594
+ success: true,
595
+ wallet: this.type,
596
+ result: "Implement based on Argent API"
597
+ };
598
+ } catch (error) {
599
+ console.error(`Error getting balance from Argent:`, error);
600
+ return {
601
+ success: false,
602
+ wallet: this.type,
603
+ error: error.message || "Unknown error"
604
+ };
605
+ }
606
+ }
607
+ };
608
+
609
+ // src/wallets/bridge.ts
610
+ var WalletBridge = class {
611
+ walletAdapters;
612
+ connectedWallets = /* @__PURE__ */ new Map();
613
+ constructor() {
614
+ this.walletAdapters = /* @__PURE__ */ new Map();
615
+ this.walletAdapters.set("metamask", new MetaMaskWallet());
616
+ this.walletAdapters.set("phantom", new PhantomWallet());
617
+ this.walletAdapters.set("argent", new ArgentWallet());
618
+ if (typeof window !== "undefined") {
619
+ window.wallet_bridge = this;
620
+ }
621
+ }
622
+ getIFrameMethods() {
623
+ return {
624
+ externalDetectWallets: (_origin) => () => this.detectWallets(),
625
+ externalConnectWallet: (_origin) => (type) => this.connectWallet(type),
626
+ externalSignMessage: (_origin) => (type, message) => this.signMessage(type, message),
627
+ externalSignTypedData: (_origin) => (type, data) => this.signTypedData(type, data),
628
+ externalGetBalance: (_origin) => (type, tokenAddress) => this.getBalance(type, tokenAddress)
629
+ };
630
+ }
631
+ async detectWallets() {
632
+ const wallets = Array.from(this.walletAdapters.values()).map(
633
+ (adapter) => adapter.getInfo()
634
+ );
635
+ return wallets;
636
+ }
637
+ getWalletAdapter(type) {
638
+ const adapter = this.walletAdapters.get(type);
639
+ if (!adapter) {
640
+ throw new Error(`Unsupported wallet type: ${type}`);
641
+ }
642
+ return adapter;
643
+ }
644
+ handleError(type, error, operation) {
645
+ const errorMessage = error instanceof Error ? error.message : "Unknown error";
646
+ console.error(`Error ${operation} with ${type} wallet:`, error);
647
+ return { success: false, wallet: type, error: errorMessage };
648
+ }
649
+ async connectWallet(type) {
650
+ try {
651
+ if (this.connectedWallets.has(type)) {
652
+ const wallet2 = this.connectedWallets.get(type);
653
+ return { success: true, wallet: type, account: wallet2.type };
654
+ }
655
+ const wallet = this.getWalletAdapter(type);
656
+ const response = await wallet.connect();
657
+ if (response.success) {
658
+ this.connectedWallets.set(type, wallet);
659
+ }
660
+ return response;
661
+ } catch (error) {
662
+ return this.handleError(type, error, "connecting to");
663
+ }
664
+ }
665
+ async signMessage(type, message) {
666
+ try {
667
+ if (!this.connectedWallets.has(type)) {
668
+ throw new Error(`Wallet ${type} is not connected`);
669
+ }
670
+ const wallet = this.connectedWallets.get(type);
671
+ if (!wallet.signMessage) {
672
+ throw new Error(`Wallet ${type} does not support signing messages`);
673
+ }
674
+ return await wallet.signMessage(message);
675
+ } catch (error) {
676
+ return this.handleError(type, error, "signing message with");
677
+ }
678
+ }
679
+ async signTypedData(type, data) {
680
+ try {
681
+ if (!this.connectedWallets.has(type)) {
682
+ throw new Error(`Wallet ${type} is not connected`);
683
+ }
684
+ const wallet = this.connectedWallets.get(type);
685
+ if (!wallet.signTypedData) {
686
+ throw new Error(`Wallet ${type} does not support signing typed data`);
687
+ }
688
+ return await wallet.signTypedData(data);
689
+ } catch (error) {
690
+ return this.handleError(type, error, "signing typed data with");
691
+ }
692
+ }
693
+ async getBalance(type, tokenAddress) {
694
+ try {
695
+ if (!this.connectedWallets.has(type)) {
696
+ throw new Error(`Wallet ${type} is not connected`);
697
+ }
698
+ const wallet = this.connectedWallets.get(type);
699
+ return await wallet.getBalance(tokenAddress);
700
+ } catch (error) {
701
+ return this.handleError(type, error, "getting balance from");
702
+ }
703
+ }
704
+ };
243
705
 
244
706
  // src/iframe/keychain.ts
245
707
  var KeychainIFrame = class extends IFrame {
708
+ walletBridge;
246
709
  constructor({ url, policies, ...iframeOptions }) {
247
710
  const _url = new URL(url ?? KEYCHAIN_URL);
711
+ const walletBridge = new WalletBridge();
248
712
  if (policies) {
249
713
  _url.searchParams.set(
250
714
  "policies",
@@ -254,8 +718,13 @@ var KeychainIFrame = class extends IFrame {
254
718
  super({
255
719
  ...iframeOptions,
256
720
  id: "controller-keychain",
257
- url: _url
721
+ url: _url,
722
+ methods: walletBridge.getIFrameMethods()
258
723
  });
724
+ this.walletBridge = walletBridge;
725
+ }
726
+ getWalletBridge() {
727
+ return this.walletBridge;
259
728
  }
260
729
  };
261
730
 
@@ -317,7 +786,7 @@ var NotReadyToConnect = class _NotReadyToConnect extends Error {
317
786
 
318
787
  // package.json
319
788
  var package_default = {
320
- version: "0.7.7"};
789
+ version: "0.7.8"};
321
790
 
322
791
  // src/icon.ts
323
792
  var icon = "";