nim-json-rpc/json_rpc/clients/socketclient.nim

80 lines
2.2 KiB
Nim
Raw Normal View History

2023-12-14 08:34:13 +07:00
# json-rpc
# Copyright (c) 2019-2023 Status Research & Development GmbH
# Licensed under either of
# * Apache License, version 2.0, ([LICENSE-APACHE](LICENSE-APACHE))
# * MIT license ([LICENSE-MIT](LICENSE-MIT))
# at your option.
# This file may not be copied, modified, or distributed except according to
# those terms.
import
std/tables,
chronos,
../client
{.push raises: [Defect].}
export client
type
RpcSocketClient* = ref object of RpcClient
transport*: StreamTransport
address*: TransportAddress
2019-05-14 18:42:51 +03:00
loop*: Future[void]
const defaultMaxRequestLength* = 1024 * 128
proc new*(T: type RpcSocketClient): T =
T()
proc newRpcSocketClient*: RpcSocketClient =
## Creates a new client instance.
RpcSocketClient.new()
method call*(self: RpcSocketClient, name: string,
params: JsonNode): Future[Response] {.async, gcsafe.} =
## Remotely calls the specified RPC method.
let id = self.getNextId()
var value = $rpcCallNode(name, params, id) & "\r\n"
if self.transport.isNil:
2018-07-14 11:25:27 +03:00
raise newException(ValueError,
"Transport is not initialised (missing a call to connect?)")
# completed by processMessage.
var newFut = newFuture[Response]()
# add to awaiting responses
self.awaiting[id] = newFut
let res = await self.transport.write(value)
# TODO: Add actions when not full packet was send, e.g. disconnect peer.
2020-03-17 22:05:42 +02:00
doAssert(res == len(value))
return await newFut
proc processData(client: RpcSocketClient) {.async.} =
while true:
2019-05-14 18:42:51 +03:00
while true:
var value = await client.transport.readLine(defaultMaxRequestLength)
if value == "":
# transmission ends
await client.transport.closeWait()
break
# TODO handle exceptions
2019-05-14 18:42:51 +03:00
client.processMessage(value)
2019-05-14 18:42:51 +03:00
# async loop reconnection and waiting
client.transport = await connect(client.address)
proc connect*(client: RpcSocketClient, address: string, port: Port) {.async.} =
let addresses = resolveTAddress(address, port)
client.transport = await connect(addresses[0])
client.address = addresses[0]
2019-05-14 18:42:51 +03:00
client.loop = processData(client)
2019-06-17 19:56:19 +03:00
method close*(client: RpcSocketClient) {.async.} =
2021-04-06 21:48:32 +03:00
await client.loop.cancelAndWait()
if not client.transport.isNil:
client.transport.close()
client.transport = nil