-
Notifications
You must be signed in to change notification settings - Fork 2
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
2 changed files
with
205 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,204 @@ | ||
import type { SignPsbtParameters, UTXOWalletProvider } from '@bigmi/core' | ||
import { | ||
type Address, | ||
MethodNotSupportedRpcError, | ||
UserRejectedRequestError, | ||
withRetry, | ||
} from 'viem' | ||
import { ProviderNotFoundError, createConnector } from 'wagmi' | ||
import type { UTXOConnectorParameters } from './types.js' | ||
|
||
export type OKXBitcoinEventMap = { | ||
accountsChanged(accounts: Address[]): void | ||
} | ||
|
||
export type OKXBitcoinEvents = { | ||
addListener<TEvent extends keyof OKXBitcoinEventMap>( | ||
event: TEvent, | ||
listener: OKXBitcoinEventMap[TEvent] | ||
): void | ||
removeListener<TEvent extends keyof OKXBitcoinEventMap>( | ||
event: TEvent, | ||
listener: OKXBitcoinEventMap[TEvent] | ||
): void | ||
} | ||
|
||
type OKXConnectorProperties = { | ||
getAccounts(): Promise<readonly Address[]> | ||
onAccountsChanged(accounts: Address[]): void | ||
getInternalProvider(): Promise<OKXBitcoinProvider> | ||
} & UTXOWalletProvider | ||
|
||
type OKXBitcoinProvider = { | ||
requestAccounts(): Promise<Address[]> | ||
getAccounts(): Promise<Address[]> | ||
signPsbt( | ||
psbtHex: string, | ||
options: { | ||
toSignInputs: { | ||
index: number | ||
address: string | ||
sighashTypes?: number[] | ||
}[] | ||
autoFinalized?: boolean | ||
} | ||
): Promise<string> | ||
} & OKXBitcoinEvents | ||
|
||
okx.type = 'UTXO' as const | ||
export function okx(parameters: UTXOConnectorParameters = {}) { | ||
const { chainId, shimDisconnect = true } = parameters | ||
let accountsChanged: ((accounts: Address[]) => void) | undefined | ||
return createConnector< | ||
UTXOWalletProvider | undefined, | ||
OKXConnectorProperties | ||
>((config) => ({ | ||
id: 'com.okex.wallet.bitcoin', | ||
name: 'OKX Wallet', | ||
type: okx.type, | ||
icon: '', | ||
async setup() { | ||
// | ||
}, | ||
async getInternalProvider() { | ||
if (typeof window === 'undefined') { | ||
return | ||
} | ||
if ('okxwallet' in window) { | ||
const anyWindow: any = window | ||
const internalProvider = anyWindow.okxwallet?.bitcoin | ||
|
||
if (internalProvider?.isOkxWallet) { | ||
return internalProvider | ||
} | ||
} | ||
}, | ||
async getProvider() { | ||
const internalProvider = await this.getInternalProvider() | ||
if (!internalProvider) { | ||
return | ||
} | ||
const provider = { | ||
request: this.request.bind(internalProvider), | ||
} | ||
return provider | ||
}, | ||
async request(this: OKXBitcoinProvider, { method, params }): Promise<any> { | ||
switch (method) { | ||
case 'signPsbt': { | ||
const { psbt, ...options } = params as SignPsbtParameters | ||
const toSignInputs = options.inputsToSign.flatMap( | ||
({ sigHash, address, signingIndexes }) => | ||
signingIndexes.map((index) => ({ | ||
index, | ||
address, | ||
sighashTypes: sigHash !== undefined ? [sigHash] : undefined, | ||
})) | ||
) | ||
const signedPsbt = await this.signPsbt(psbt, { | ||
toSignInputs, | ||
autoFinalized: options.finalize, | ||
}) | ||
return signedPsbt | ||
} | ||
default: | ||
throw new MethodNotSupportedRpcError( | ||
new Error(MethodNotSupportedRpcError.name), | ||
{ | ||
method, | ||
} | ||
) | ||
} | ||
}, | ||
async connect() { | ||
const provider = await this.getInternalProvider() | ||
if (!provider) { | ||
throw new ProviderNotFoundError() | ||
} | ||
try { | ||
const accounts = await provider.requestAccounts() | ||
const chainId = await this.getChainId() | ||
|
||
if (!accountsChanged) { | ||
accountsChanged = this.onAccountsChanged.bind(this) | ||
provider.addListener('accountsChanged', accountsChanged) | ||
} | ||
|
||
// Remove disconnected shim if it exists | ||
if (shimDisconnect) { | ||
await Promise.all([ | ||
config.storage?.setItem(`${this.id}.connected`, true), | ||
config.storage?.removeItem(`${this.id}.disconnected`), | ||
]) | ||
} | ||
return { accounts, chainId } | ||
} catch (error: any) { | ||
throw new UserRejectedRequestError({ | ||
name: UserRejectedRequestError.name, | ||
message: error.message, | ||
}) | ||
} | ||
}, | ||
async disconnect() { | ||
const provider = await this.getInternalProvider() | ||
|
||
if (accountsChanged) { | ||
provider?.removeListener('accountsChanged', accountsChanged) | ||
accountsChanged = undefined | ||
} | ||
|
||
// Add shim signalling connector is disconnected | ||
if (shimDisconnect) { | ||
await Promise.all([ | ||
config.storage?.setItem(`${this.id}.disconnected`, true), | ||
config.storage?.removeItem(`${this.id}.connected`), | ||
]) | ||
} | ||
}, | ||
async getAccounts() { | ||
const provider = await this.getInternalProvider() | ||
if (!provider) { | ||
throw new ProviderNotFoundError() | ||
} | ||
const accounts = await provider.getAccounts() | ||
return accounts as Address[] | ||
}, | ||
async getChainId() { | ||
return chainId! | ||
}, | ||
async isAuthorized() { | ||
try { | ||
const isDisconnected = | ||
shimDisconnect && | ||
// If shim exists in storage, connector is disconnected | ||
(await config.storage?.getItem(`${this.id}.disconnected`)) | ||
if (isDisconnected) { | ||
return false | ||
} | ||
const accounts = await withRetry(() => this.getAccounts()) | ||
return !!accounts.length | ||
} catch { | ||
return false | ||
} | ||
}, | ||
async onAccountsChanged(accounts) { | ||
if (accounts.length === 0) { | ||
this.onDisconnect() | ||
} else { | ||
config.emitter.emit('change', { | ||
accounts: accounts as Address[], | ||
}) | ||
} | ||
}, | ||
onChainChanged(chain) { | ||
const chainId = Number(chain) | ||
config.emitter.emit('change', { chainId }) | ||
}, | ||
async onDisconnect(_error) { | ||
// No need to remove `${this.id}.disconnected` from storage because `onDisconnect` is typically | ||
// only called when the wallet is disconnected through the wallet's interface, meaning the wallet | ||
// actually disconnected and we don't need to simulate it. | ||
config.emitter.emit('disconnect') | ||
}, | ||
})) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters