diff --git a/packages/wallets/ledger/src/polyfills/index.ts b/packages/wallets/ledger/src/polyfills/index.ts index 260f6ed0e..e9db0daf3 100644 --- a/packages/wallets/ledger/src/polyfills/index.ts +++ b/packages/wallets/ledger/src/polyfills/index.ts @@ -1,3 +1 @@ -'use strict'; - import './Buffer.js'; diff --git a/packages/wallets/trezor/src/adapter.ts b/packages/wallets/trezor/src/adapter.ts index 4a91c54c6..785ddd509 100644 --- a/packages/wallets/trezor/src/adapter.ts +++ b/packages/wallets/trezor/src/adapter.ts @@ -1,10 +1,11 @@ import type { WalletName } from '@solana/wallet-adapter-base'; -import type { TrezorConnect } from '@trezor/connect-web'; import { BaseSignerWalletAdapter, + WalletAccountError, WalletConfigError, WalletDisconnectedError, WalletDisconnectionError, + WalletLoadError, WalletNotConnectedError, WalletNotReadyError, WalletPublicKeyError, @@ -14,7 +15,15 @@ import { } from '@solana/wallet-adapter-base'; import type { Transaction, TransactionVersion, VersionedTransaction } from '@solana/web3.js'; import { PublicKey } from '@solana/web3.js'; -import { DEVICE, DEVICE_EVENT } from './constants.js'; +import type { + DeviceEventMessage, + SolanaPublicKey, + SolanaSignedTransaction, + Success, + TrezorConnect, + Unsuccessful, +} from '@trezor/connect-web'; +import './polyfills/index.js'; export interface TrezorWalletAdapterConfig { derivationPath?: string; @@ -27,12 +36,12 @@ export class TrezorWalletAdapter extends BaseSignerWalletAdapter { name = TrezorWalletName; url = 'https://trezor.io'; icon = - ''; + ''; supportedTransactionVersions: ReadonlySet = new Set(['legacy', 0]); private _derivationPath: string; private _wallet: TrezorConnect | null; - private _connectUrl?: string; + private _connectUrl: string | undefined; private _connecting: boolean; private _publicKey: PublicKey | null; private _readyState: WalletReadyState = @@ -47,7 +56,7 @@ export class TrezorWalletAdapter extends BaseSignerWalletAdapter { super(); this._derivationPath = config.derivationPath || `m/44'/501'/0'/0'`; this._wallet = null; - this._connectUrl = config.connectUrl && config.connectUrl + (config.connectUrl.endsWith('/') ? '' : '/'); + this._connectUrl = config.connectUrl?.replace(/\/*$/, '/'); this._connecting = false; this._publicKey = null; } @@ -71,52 +80,56 @@ export class TrezorWalletAdapter extends BaseSignerWalletAdapter { this._connecting = true; + let wallet: TrezorConnect; try { const { default: TrezorConnect } = await import('@trezor/connect-web'); - // @ts-ignore - this._wallet = TrezorConnect.default as TrezorConnect; + // @ts-expect-error // HACK: TrezorConnect.default is undefined. + wallet = TrezorConnect.default as TrezorConnect; } catch (error: any) { - throw new WalletConfigError(error?.message, error); + throw new WalletLoadError(error?.message, error); } - await this._wallet.init({ - manifest: { - email: 'gabriel.kerekes@vacuumlabs.com', - appUrl: 'https://github.com/solana-labs/wallet-adapter', - }, - lazyLoad: true, - ...(this._connectUrl - ? { - connectSrc: this._connectUrl, - iframeSrc: this._connectUrl, - } - : {}), - }); - - let result; try { - result = await this._wallet.solanaGetPublicKey({ - path: this._derivationPath, + await wallet.init({ + manifest: { + email: 'gabriel.kerekes@vacuumlabs.com', + appUrl: window.location.href, + }, + lazyLoad: true, + ...(this._connectUrl + ? { + connectSrc: this._connectUrl, + iframeSrc: this._connectUrl, + } + : {}), }); } catch (error: any) { - throw new WalletPublicKeyError(error?.message, error); + throw new WalletConfigError(error?.message, error); } + let result: Unsuccessful | Success; + try { + result = await wallet.solanaGetPublicKey({ path: this._derivationPath }); + } catch (error: any) { + throw new WalletAccountError(error?.message, error); + } if (!result.success) { - throw new WalletPublicKeyError(result.payload?.error, result.payload); + throw new WalletAccountError(result.payload?.error, result.payload); } - const publicKey = result.payload.publicKey; + let publicKey: PublicKey; + try { + publicKey = new PublicKey(Buffer.from(result.payload.publicKey, 'hex')); + } catch (error: any) { + throw new WalletPublicKeyError(error?.message, error); + } - this._wallet.on(DEVICE_EVENT, (event: any) => { - if (event.type === DEVICE.DISCONNECT) { - this._disconnected(); - } - }); + wallet.on('DEVICE_EVENT', this._onDeviceEvent); - this._publicKey = new PublicKey(Buffer.from(publicKey, 'hex')); + this._wallet = wallet; + this._publicKey = publicKey; - this.emit('connect', this._publicKey); + this.emit('connect', publicKey); } catch (error: any) { this.emit('error', error); throw error; @@ -126,15 +139,20 @@ export class TrezorWalletAdapter extends BaseSignerWalletAdapter { } async disconnect(): Promise { - this._publicKey = null; + const wallet = this._wallet; + if (wallet) { + this._wallet = null; + this._publicKey = null; - try { - await this._wallet?.dispose(); - } catch (error: any) { - this.emit('error', new WalletDisconnectionError(error?.message, error)); - } + try { + wallet.off('DEVICE_EVENT', this._onDeviceEvent); + await wallet.dispose(); + } catch (error: any) { + this.emit('error', new WalletDisconnectionError(error?.message, error)); + } - this.emit('disconnect'); + this.emit('disconnect'); + } } async signTransaction(transaction: T): Promise { @@ -147,7 +165,7 @@ export class TrezorWalletAdapter extends BaseSignerWalletAdapter { ? transaction.message.serialize() : transaction.serializeMessage(); - let result; + let result: Unsuccessful | Success; try { result = await wallet.solanaSignTransaction({ path: this._derivationPath, @@ -161,7 +179,12 @@ export class TrezorWalletAdapter extends BaseSignerWalletAdapter { throw new WalletSignTransactionError(result.payload?.error, result.payload); } - transaction.addSignature(publicKey, Buffer.from(result.payload.signature, 'hex')); + try { + const signature = Buffer.from(result.payload.signature, 'hex'); + transaction.addSignature(publicKey, signature); + } catch (error: any) { + throw new WalletSignTransactionError(error?.message, error); + } return transaction; } catch (error: any) { @@ -170,11 +193,27 @@ export class TrezorWalletAdapter extends BaseSignerWalletAdapter { } } - private _disconnected = () => { - this._wallet?.dispose(); - this._publicKey = null; + private _onDeviceEvent = (event: DeviceEventMessage) => { + if (event.type === 'device-disconnect') { + this._disconnected(); + } + }; + + private _disconnected = async () => { + const wallet = this._wallet; + if (wallet) { + this._wallet = null; + this._publicKey = null; + + try { + wallet.off('DEVICE_EVENT', this._onDeviceEvent); + wallet.dispose(); + } catch (error: any) { + this.emit('error', new WalletDisconnectionError(error?.message, error)); + } - this.emit('error', new WalletDisconnectedError()); - this.emit('disconnect'); + this.emit('error', new WalletDisconnectedError()); + this.emit('disconnect'); + } }; } diff --git a/packages/wallets/trezor/src/constants.ts b/packages/wallets/trezor/src/constants.ts deleted file mode 100644 index 982afc369..000000000 --- a/packages/wallets/trezor/src/constants.ts +++ /dev/null @@ -1,19 +0,0 @@ -// redeclared from TrezorConnect to allow async import -export const DEVICE_EVENT = 'DEVICE_EVENT'; -export const DEVICE = { - CONNECT: 'device-connect', - CONNECT_UNACQUIRED: 'device-connect_unacquired', - DISCONNECT: 'device-disconnect', - CHANGED: 'device-changed', - ACQUIRE: 'device-acquire', - RELEASE: 'device-release', - ACQUIRED: 'device-acquired', - RELEASED: 'device-released', - USED_ELSEWHERE: 'device-used_elsewhere', - LOADING: 'device-loading', - BUTTON: 'button', - PIN: 'pin', - PASSPHRASE: 'passphrase', - PASSPHRASE_ON_DEVICE: 'passphrase_on_device', - WORD: 'word', -}; diff --git a/packages/wallets/trezor/src/polyfills/Buffer.ts b/packages/wallets/trezor/src/polyfills/Buffer.ts new file mode 100644 index 000000000..3a82fd890 --- /dev/null +++ b/packages/wallets/trezor/src/polyfills/Buffer.ts @@ -0,0 +1,7 @@ +import { Buffer } from 'buffer'; + +if (typeof window !== 'undefined' && window.Buffer === undefined) { + (window as any).Buffer = Buffer; +} + +export {}; diff --git a/packages/wallets/trezor/src/polyfills/index.ts b/packages/wallets/trezor/src/polyfills/index.ts new file mode 100644 index 000000000..e9db0daf3 --- /dev/null +++ b/packages/wallets/trezor/src/polyfills/index.ts @@ -0,0 +1 @@ +import './Buffer.js';