-
Notifications
You must be signed in to change notification settings - Fork 1.5k
/
Copy pathutils.ts
99 lines (87 loc) · 3.58 KB
/
utils.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
/**
* @param rpcMap - Map of chainIds to rpc url(s).
* @param timeout - Timeout, in milliseconds, after which to consider network calls failed.
*/
export async function getBestUrlMap(
rpcMap: Record<string, string | string[]>,
timeout: number
): Promise<{ [chainId: string]: string }> {
return Object.fromEntries(
await Promise.all(Object.entries(rpcMap).map(async ([chainId, map]) => [chainId, await getBestUrl(map, timeout)]))
)
}
/**
* @param urls - An array of URLs to try to connect to.
* @param timeout - {@link getBestUrlMap}
*/
async function getBestUrl(urls: string | string[], timeout: number): Promise<string> {
// if we only have 1 url, it's the best!
if (typeof urls === 'string') return urls
if (urls.length === 1) return urls[0]
const [HttpConnection, JsonRpcProvider] = await Promise.all([
import('@walletconnect/jsonrpc-http-connection').then(({ HttpConnection }) => HttpConnection),
import('@walletconnect/jsonrpc-provider').then(({ JsonRpcProvider }) => JsonRpcProvider),
])
// the below returns the first url for which there's been a successful call, prioritized by index
return new Promise((resolve) => {
let resolved = false
const successes: { [index: number]: boolean } = {}
urls.forEach((url, i) => {
const http = new JsonRpcProvider(new HttpConnection(url))
// create a promise that resolves on a successful call, and rejects on a failed call or after timeout milliseconds
const promise = new Promise<void>((resolve, reject) => {
http
.request({ method: 'eth_chainId' })
.then(() => resolve())
.catch(() => reject())
// set a timeout to reject
setTimeout(() => {
reject()
}, timeout)
})
void promise
.then(() => true)
.catch(() => false)
.then((success) => {
// if we already resolved, return
if (resolved) return
// store the result of the call
successes[i] = success
// if this is the last call and we haven't resolved yet - do so
if (Object.keys(successes).length === urls.length) {
const index = Object.keys(successes).findIndex((j) => successes[Number(j)])
// no need to set resolved to true, as this is the last promise
return resolve(urls[index === -1 ? 0 : index])
}
// otherwise, for each prospective index, check if we can resolve
new Array<number>(urls.length).fill(0).forEach((_, prospectiveIndex) => {
// to resolve, we need to:
// a) have successfully made a call
// b) not be waiting on any other higher-index calls
if (
successes[prospectiveIndex] &&
new Array<number>(prospectiveIndex).fill(0).every((_, j) => successes[j] === false)
) {
resolved = true
resolve(urls[prospectiveIndex])
}
})
})
})
})
}
/**
* @param chains - An array of chain IDs.
* @param defaultChainId - The chain ID to treat as the default (it will be the first element in the returned array).
*/
export function getChainsWithDefault(chains: number[], defaultChainId: number) {
const idx = chains.indexOf(defaultChainId)
if (idx === -1) {
throw new Error(
`Invalid chainId ${defaultChainId}. Make sure default chain is included in "chains" - chains specified in "optionalChains" may not be selected as the default, as they may not be supported by the wallet.`
)
}
const ordered = [...chains]
ordered.splice(idx, 1)
return [defaultChainId, ...ordered]
}