nimbus-eth1/fluffy/tools/portal_bridge/portal_bridge_common.nim
bhartnett 56724536a4
Fluffy State Bridge: Improvements to websocket rpc connections (#2694)
Handle reconnect for websocket rpc client and create rpc client instance per worker in order to improve websocket throughput. (#2694)
2024-10-04 14:17:18 +08:00

80 lines
2.6 KiB
Nim

# Fluffy
# Copyright (c) 2024 Status Research & Development GmbH
# Licensed and distributed under either of
# * MIT license (license terms in the root directory or at https://opensource.org/licenses/MIT).
# * Apache v2 license (license terms in the root directory or at https://www.apache.org/licenses/LICENSE-2.0).
# at your option. This file may not be copied, modified, or distributed except according to those terms.
{.push raises: [].}
import
chronicles,
json_rpc/rpcclient,
web3/[eth_api, eth_api_types],
../../rpc/rpc_calls/rpc_trace_calls,
./portal_bridge_conf
export rpcclient
proc newRpcClientConnect*(url: JsonRpcUrl): RpcClient =
## Instantiate a new JSON-RPC client and try to connect. Will quit on failure.
case url.kind
of HttpUrl:
let client = newRpcHttpClient()
try:
waitFor client.connect(url.value)
except CatchableError as e:
fatal "Failed to connect to JSON-RPC server", error = $e.msg, url = url.value
quit QuitFailure
client
of WsUrl:
let client = newRpcWebSocketClient()
try:
waitFor client.connect(url.value)
except CatchableError as e:
fatal "Failed to connect to JSON-RPC server", error = $e.msg, url = url.value
quit QuitFailure
client
proc tryReconnect*(client: RpcClient, url: JsonRpcUrl) {.async: (raises: []).} =
if url.kind == WsUrl:
doAssert client of RpcWebSocketClient
let wsClient = RpcWebSocketClient(client)
if wsClient.transport.isNil:
# disconnected
try:
await wsClient.connect(url.value)
except CatchableError as e:
warn "Failed to reconnect to JSON-RPC server", error = $e.msg, url = url.value
proc getBlockByNumber*(
client: RpcClient, blockId: BlockIdentifier, fullTransactions: bool = true
): Future[Result[BlockObject, string]] {.async: (raises: []).} =
let blck =
try:
let res = await client.eth_getBlockByNumber(blockId, fullTransactions)
if res.isNil:
return err("EL failed to provide requested block")
res
except CatchableError as e:
return err("EL JSON-RPC eth_getBlockByNumber failed: " & e.msg)
return ok(blck)
proc getUncleByBlockNumberAndIndex*(
client: RpcClient, blockId: BlockIdentifier, index: Quantity
): Future[Result[BlockObject, string]] {.async: (raises: []).} =
let blck =
try:
let res = await client.eth_getUncleByBlockNumberAndIndex(blockId, index)
if res.isNil:
return err("EL failed to provide requested uncle block")
res
except CatchableError as e:
return err("EL JSON-RPC eth_getUncleByBlockNumberAndIndex failed: " & e.msg)
return ok(blck)