iso-filecoin-wallets
Version:
Filecoin Wallet Adapters.
464 lines (417 loc) • 23.3 kB
JavaScript
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: '',
}