UNPKG

iso-filecoin-wallets

Version:
464 lines (417 loc) 23.3 kB
import { AdapterBlueprint } from '@reown/appkit/adapters' import { CoreHelperUtil, StorageUtil } from '@reown/appkit-controllers' import { RPC } from 'iso-filecoin/rpc' import { Token } from 'iso-filecoin/token' /** * @import { AppKit, AppKitOptions, CaipNetwork } from '@reown/appkit' * @import { ChainNamespace } from '@reown/appkit-common' * @import { AccountNetwork, WalletAdapter } from 'iso-filecoin-wallets/types' * @import { ConnectorType, Provider } from '@reown/appkit-controllers' */ /** * @description Filecoin namespace */ export const filNamespace = /** @type {ChainNamespace} */ ('fil') /** * TODO: * - auth provider for SIWX * * @description Filecoin adapter for AppKit * @extends {AdapterBlueprint<FilecoinConnector>} */ export class FilecoinAppKitAdapter extends AdapterBlueprint { /** * @type {WalletAdapter | undefined} */ #adapter /** * @type {WalletAdapter[]} */ adapters /** * @param {{ adapters: WalletAdapter[] }} params */ constructor(params) { super({ namespace: filNamespace, adapterType: 'filecoin', }) this.adapters = params.adapters } /** * @param {AdapterBlueprint.Params} params */ construct(params) { super.construct({ namespace: filNamespace, networks: params.networks?.filter( (n) => n.chainNamespace === filNamespace ), projectId: params.projectId, }) this.caipNetworks = params.networks?.filter( (n) => n.chainNamespace === filNamespace ) } /** * @param {string} id */ #getFilConnector(id) { return this.connectors.find((c) => c.id === id) } /** * Get accounts * * @param {AdapterBlueprint.GetAccountsParams} params * @returns {Promise<AdapterBlueprint.GetAccountsResult>} */ getAccounts(params) { const connector = this.#getFilConnector(params.id) if (!connector || !connector.adapter.account) { return Promise.resolve({ accounts: [], }) } const acc = connector.adapter.account const account = CoreHelperUtil.createAccount( filNamespace, acc.address.toString(), 'eoa', undefined, acc.path ) return Promise.resolve({ accounts: [account], }) } /** * Sync connectors * * @param {AppKitOptions} _options * @param {AppKit} _appKit * @returns {void | Promise<void>} */ syncConnectors(_options, _appKit) { for (const adapter of this.adapters) { this.addConnector(new FilecoinConnector(adapter)) } } /** * Sync connections * * @param {AdapterBlueprint.SyncConnectionsParams} _params * @returns {void | Promise<void>} */ syncConnections(_params) { return Promise.resolve() } /** * Connect * * @param {AdapterBlueprint.ConnectParams} params * @returns {Promise<AdapterBlueprint.ConnectResult>} */ async connect(params) { const connector = this.#getFilConnector(params.id) if (!connector) { throw new Error(`No connector for id: ${params.id}`) } const { account } = await connector.adapter.connect({ network: params.chainId === 'f' ? 'mainnet' : 'testnet', }) this.#adapter = connector.adapter this.#listenToAdapter(connector.adapter) return { id: connector.id, type: connector.type, chainId: params.chainId ?? 'f', address: account.address.toString(), provider: connector.provider, } } /** * Listen to adapter * * @param {WalletAdapter} adapter */ #listenToAdapter(adapter) { const handleConnect = (/** @type {CustomEvent<AccountNetwork>} */ evt) => { this.emit('accountChanged', { address: evt.detail.account.address.toString(), chainId: evt.detail.network === 'mainnet' ? 'f' : 't', }) } const handleAccountChanged = ( /** @type {CustomEvent<import('iso-filecoin/types').IAccount>} */ evt ) => { this.emit('accountChanged', { address: evt.detail.address.toString(), }) } const handleDisconnect = () => { // remove event listener adapter.off('connect', handleConnect) adapter.off('disconnect', handleDisconnect) adapter.off('accountChanged', handleAccountChanged) adapter.off('networkChanged', handleConnect) this.emit('disconnect') } adapter.on('connect', handleConnect) adapter.on('accountChanged', handleAccountChanged) adapter.on('networkChanged', handleConnect) adapter.on('disconnect', handleDisconnect) } /** * Sync connection * * @param {AdapterBlueprint.SyncConnectionParams} params * @returns {Promise<AdapterBlueprint.ConnectResult>} */ syncConnection(params) { return this.connect({ ...params, type: '', }) } /** * Disconnect * * @param {AdapterBlueprint.DisconnectParams} _params * @returns {Promise<AdapterBlueprint.DisconnectResult>} */ async disconnect(_params) { if (this.#adapter) { await this.#adapter.disconnect() this.#adapter = undefined } return { connections: [], } } /** * Switch network * * @param {AdapterBlueprint.SwitchNetworkParams} params * @returns {Promise<void>} */ async switchNetwork(params) { if (!this.#adapter) { throw new Error('Not connected') } await this.#adapter.changeNetwork( params.caipNetwork.id === 'f' ? 'mainnet' : 'testnet' ) } /** * Get balance * * @param {AdapterBlueprint.GetBalanceParams} params * @returns {Promise<AdapterBlueprint.GetBalanceResult>} */ async getBalance(params) { if (!this.#adapter || !params.caipNetwork || !params.address) { return { balance: '0.00', symbol: 'FIL', } } // Switch network if needed const network = params.caipNetwork.id === 'f' ? 'mainnet' : 'testnet' let address = params.address if (network !== this.#adapter.network) { const { account } = await this.#adapter.changeNetwork(network) address = account.address.toString() } // Check cache const caipAddress = `${params.caipNetwork.caipNetworkId}:${address}` const cachedBalance = StorageUtil.getNativeBalanceCacheForCaipAddress(caipAddress) if (cachedBalance) { return { balance: cachedBalance.balance, symbol: cachedBalance.symbol } } // Get balance from RPC const rpc = new RPC({ api: params.caipNetwork?.rpcUrls?.default?.http?.[0], network: network, }) const balance = await rpc.balance(address) if (balance.error) { throw new Error(balance.error.message) } // Format balance const formattedBalance = Token.fromAttoFIL(balance.result) .toFIL() .toFormat({ decimalPlaces: 1, }) // Update cache StorageUtil.updateNativeBalanceCache({ caipAddress, balance: formattedBalance, symbol: params.caipNetwork.nativeCurrency.symbol, timestamp: Date.now(), }) return { balance: formattedBalance, symbol: params.caipNetwork.nativeCurrency.symbol, } } /** * Sign message * * @param {AdapterBlueprint.SignMessageParams} _params * @returns {Promise<AdapterBlueprint.SignMessageResult>} */ signMessage(_params) { return Promise.resolve({ signature: '0x', }) } /** * Estimate gas * * @param {AdapterBlueprint.EstimateGasTransactionArgs} _params * @returns {Promise<AdapterBlueprint.EstimateGasTransactionResult>} */ estimateGas(_params) { return Promise.resolve({ gas: 0n, }) } /** * Send transaction * * @param {AdapterBlueprint.SendTransactionParams} _params * @returns {Promise<AdapterBlueprint.SendTransactionResult>} */ sendTransaction(_params) { return Promise.resolve({ hash: '0x', }) } /** * Parse units * * @param {AdapterBlueprint.ParseUnitsParams} _params * @returns {AdapterBlueprint.ParseUnitsResult} */ parseUnits(_params) { return 0n } /** * Format units * * @param {AdapterBlueprint.FormatUnitsParams} _params * @returns {AdapterBlueprint.FormatUnitsResult} */ formatUnits(_params) { return '0' } getProfile() { return Promise.resolve({ profileName: undefined, profileImage: undefined, }) } getCapabilities() { return Promise.resolve({}) } grantPermissions() { return Promise.resolve({}) } revokePermissions() { return Promise.resolve(/** @type {`0x${string}`} */ ('0x')) } walletGetAssets() { return Promise.resolve({}) } /** * Write contract * * @returns {Promise<AdapterBlueprint.WriteContractResult>} */ writeContract() { return Promise.resolve({ hash: '', }) } setUniversalProvider() { return Promise.resolve() } /** * @param {{ provider: any; }} params */ getWalletConnectProvider(params) { return params.provider } } class FilecoinConnector { /** * @type {WalletAdapter} */ adapter /** * @type {ChainNamespace} */ chain /** * @type {string} */ id /** * @type {ConnectorType} */ type /** * @type {string} */ name /** * @type {string} */ imageUrl /** * @type {CaipNetwork[]} */ chains /** @type {Provider} */ provider /** * * @param {WalletAdapter} adapter */ constructor(adapter) { this.id = adapter.id this.type = 'INJECTED' this.name = adapter.name this.adapter = adapter this.chain = /** @type {ChainNamespace} */ ('fil') this.imageUrl = connectorImages[adapter.id] this.chains = [] // @ts-ignore this.provider = adapter } } /** * Appkit connector images * * @type {Record<string, string>} */ export const connectorImages = { filsnap: '', ledger: '', } /** * Appkit chain images * * @type {Record<string, string>} */ export const chainImages = { 314: '', 314_159: '', f: '', t: '', }