1326 lines
45 KiB
Nim
Raw Permalink Normal View History

2022-07-01 20:19:57 +02:00
# Nim-LibP2P
2023-01-20 15:47:40 +01:00
# Copyright (c) 2023 Status Research & Development GmbH
2022-07-01 20:19:57 +02:00
# 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.
2018-11-19 04:52:11 +02:00
2023-06-07 13:12:49 +02:00
{.push raises: [].}
2018-11-19 04:52:11 +02:00
## This module implementes API for `go-libp2p-daemon`.
import std/[os, osproc, strutils, tables, strtabs, sequtils]
import pkg/[chronos, chronicles]
import ../varint, ../multiaddress, ../multicodec, ../cid, ../peerid
import ../wire, ../multihash, ../protobuf/minprotobuf, ../errors
import ../crypto/crypto, ../utility
2018-11-19 04:52:11 +02:00
export
peerid, multiaddress, multicodec, multihash, cid, crypto, wire, errors
2019-03-05 01:57:18 +02:00
2018-11-19 04:52:11 +02:00
when not defined(windows):
import posix
const
DefaultSocketPath* = "/unix/tmp/p2pd.sock"
DefaultUnixSocketPattern* = "/unix/tmp/nim-p2pd-$1-$2.sock"
DefaultIpSocketPattern* = "/ip4/127.0.0.1/tcp/$2"
DefaultUnixChildPattern* = "/unix/tmp/nim-p2pd-handle-$1-$2.sock"
DefaultIpChildPattern* = "/ip4/127.0.0.1/tcp/$2"
2018-11-19 04:52:11 +02:00
DefaultDaemonFile* = "p2pd"
type
IpfsLogLevel* {.pure.} = enum
Critical, Error, Warning, Notice, Info, Debug, Trace
2018-11-19 04:52:11 +02:00
RequestType* {.pure.} = enum
IDENTIFY = 0,
2018-11-19 04:52:11 +02:00
CONNECT = 1,
STREAM_OPEN = 2,
STREAM_HANDLER = 3,
DHT = 4,
LIST_PEERS = 5,
CONNMANAGER = 6,
DISCONNECT = 7
2018-12-04 19:53:36 +02:00
PUBSUB = 8
2018-11-19 04:52:11 +02:00
DHTRequestType* {.pure.} = enum
FIND_PEER = 0,
FIND_PEERS_CONNECTED_TO_PEER = 1,
FIND_PROVIDERS = 2,
GET_CLOSEST_PEERS = 3,
GET_PUBLIC_KEY = 4,
GET_VALUE = 5,
SEARCH_VALUE = 6,
PUT_VALUE = 7,
PROVIDE = 8
ConnManagerRequestType* {.pure.} = enum
TAG_PEER = 0,
UNTAG_PEER = 1,
TRIM = 2
PSRequestType* {.pure.} = enum
GET_TOPICS = 0,
LIST_PEERS = 1,
PUBLISH = 2,
SUBSCRIBE = 3
2018-11-19 04:52:11 +02:00
ResponseKind* = enum
Malformed,
Error,
Success
ResponseType* {.pure.} = enum
ERROR = 2,
STREAMINFO = 3,
IDENTITY = 4,
DHT = 5,
PEERINFO = 6
2018-12-04 19:53:36 +02:00
PUBSUB = 7
2018-11-19 04:52:11 +02:00
DHTResponseType* {.pure.} = enum
BEGIN = 0,
VALUE = 1,
END = 2
MultiProtocol* = string
DHTValue* = seq[byte]
P2PStreamFlags* {.pure.} = enum
None, Closed, Inbound, Outbound
P2PDaemonFlags* = enum
DHTClient, ## Start daemon in DHT client mode
DHTFull, ## Start daemon with full DHT support
Bootstrap, ## Start daemon with bootstrap
WaitBootstrap, ## Start daemon with bootstrap and wait until daemon
## establish connection to at least 2 peers
PSFloodSub, ## Enable `FloodSub` protocol in daemon
PSGossipSub, ## Enable `GossipSub` protocol in daemon
PSNoSign, ## Disable pubsub message signing (default true)
2018-12-17 22:39:25 +02:00
PSStrictSign, ## Force strict checking pubsub message signature
NATPortMap, ## Force daemon to use NAT-PMP.
AutoNAT, ## Force daemon to use AutoNAT.
AutoRelay, ## Enables autorelay mode.
RelayActive, ## Enables active mode for relay.
RelayDiscovery,## Enables passive discovery for relay.
RelayHop, ## Enables hop for relay.
2021-12-16 11:05:20 +01:00
NoInlinePeerId,## Disable inlining of peer ID (not yet in #master).
NoProcessCtrl ## Process was not spawned.
2018-11-19 04:52:11 +02:00
P2PStream* = ref object
flags*: set[P2PStreamFlags]
2021-12-16 11:05:20 +01:00
peer*: PeerId
2018-11-19 04:52:11 +02:00
raddress*: MultiAddress
protocol*: string
transp*: StreamTransport
2018-12-10 12:38:12 +02:00
P2PServer = object
server*: StreamServer
address*: MultiAddress
2018-12-10 12:38:12 +02:00
2018-11-19 04:52:11 +02:00
DaemonAPI* = ref object
2018-12-10 22:55:06 +02:00
# pool*: TransportPool
2018-11-19 04:52:11 +02:00
flags*: set[P2PDaemonFlags]
address*: MultiAddress
2018-11-19 04:52:11 +02:00
pattern*: string
ucounter*: int
process*: Process
handlers*: Table[string, P2PStreamCallback]
2018-12-10 12:38:12 +02:00
servers*: seq[P2PServer]
userData*: RootRef
2018-11-19 04:52:11 +02:00
PeerInfo* = object
2021-12-16 11:05:20 +01:00
peer*: PeerId
addresses*: seq[MultiAddress]
2018-11-19 04:52:11 +02:00
PubsubTicket* = ref object
topic*: string
handler*: P2PPubSubCallback
transp*: StreamTransport
PubSubMessage* = object
2021-12-16 11:05:20 +01:00
peer*: PeerId
data*: seq[byte]
seqno*: seq[byte]
topics*: seq[string]
signature*: Signature
key*: PublicKey
2018-11-19 04:52:11 +02:00
P2PStreamCallback* = proc(api: DaemonAPI,
2023-06-07 13:12:49 +02:00
stream: P2PStream): Future[void] {.gcsafe, raises: [CatchableError].}
P2PPubSubCallback* = proc(api: DaemonAPI,
ticket: PubsubTicket,
2023-06-07 13:12:49 +02:00
message: PubSubMessage): Future[bool] {.gcsafe, raises: [CatchableError].}
2018-11-19 04:52:11 +02:00
DaemonError* = object of LPError
DaemonRemoteError* = object of DaemonError
DaemonLocalError* = object of DaemonError
2018-11-19 04:52:11 +02:00
var daemonsCount {.threadvar.}: int
chronicles.formatIt(PeerInfo): shortLog(it)
2018-11-19 04:52:11 +02:00
proc requestIdentity(): ProtoBuffer =
## https://github.com/libp2p/go-libp2p-daemon/blob/master/conn.go
## Processing function `doIdentify(req *pb.Request)`.
result = initProtoBuffer({WithVarintLength})
result.write(1, safeConvert[uint](RequestType.IDENTIFY))
2018-11-19 04:52:11 +02:00
result.finish()
2021-12-16 11:05:20 +01:00
proc requestConnect(peerid: PeerId,
addresses: openArray[MultiAddress],
2018-11-23 12:16:35 +02:00
timeout = 0): ProtoBuffer =
2018-11-19 04:52:11 +02:00
## https://github.com/libp2p/go-libp2p-daemon/blob/master/conn.go
## Processing function `doConnect(req *pb.Request)`.
result = initProtoBuffer({WithVarintLength})
var msg = initProtoBuffer()
msg.write(1, peerid)
2018-11-19 04:52:11 +02:00
for item in addresses:
msg.write(2, item.data.buffer)
2018-11-23 12:16:35 +02:00
if timeout > 0:
msg.write(3, hint64(timeout))
result.write(1, safeConvert[uint](RequestType.CONNECT))
result.write(2, msg)
2018-11-19 04:52:11 +02:00
result.finish()
2021-12-16 11:05:20 +01:00
proc requestDisconnect(peerid: PeerId): ProtoBuffer =
2018-11-19 04:52:11 +02:00
## https://github.com/libp2p/go-libp2p-daemon/blob/master/conn.go
## Processing function `doDisconnect(req *pb.Request)`.
result = initProtoBuffer({WithVarintLength})
var msg = initProtoBuffer()
msg.write(1, peerid)
result.write(1, safeConvert[uint](RequestType.DISCONNECT))
result.write(7, msg)
2018-11-19 04:52:11 +02:00
result.finish()
2021-12-16 11:05:20 +01:00
proc requestStreamOpen(peerid: PeerId,
protocols: openArray[string],
2018-11-23 12:16:35 +02:00
timeout = 0): ProtoBuffer =
2018-11-19 04:52:11 +02:00
## https://github.com/libp2p/go-libp2p-daemon/blob/master/conn.go
## Processing function `doStreamOpen(req *pb.Request)`.
result = initProtoBuffer({WithVarintLength})
var msg = initProtoBuffer()
msg.write(1, peerid)
2018-11-19 04:52:11 +02:00
for item in protocols:
msg.write(2, item)
2018-11-23 12:16:35 +02:00
if timeout > 0:
msg.write(3, hint64(timeout))
result.write(1, safeConvert[uint](RequestType.STREAM_OPEN))
result.write(3, msg)
2018-11-19 04:52:11 +02:00
result.finish()
proc requestStreamHandler(address: MultiAddress,
2021-12-16 11:05:20 +01:00
protocols: openArray[MultiProtocol]): ProtoBuffer =
2018-11-19 04:52:11 +02:00
## https://github.com/libp2p/go-libp2p-daemon/blob/master/conn.go
## Processing function `doStreamHandler(req *pb.Request)`.
result = initProtoBuffer({WithVarintLength})
var msg = initProtoBuffer()
msg.write(1, address.data.buffer)
2018-11-19 04:52:11 +02:00
for item in protocols:
msg.write(2, item)
result.write(1, safeConvert[uint](RequestType.STREAM_HANDLER))
result.write(4, msg)
2018-11-19 04:52:11 +02:00
result.finish()
proc requestListPeers(): ProtoBuffer =
## https://github.com/libp2p/go-libp2p-daemon/blob/master/conn.go
## Processing function `doListPeers(req *pb.Request)`
result = initProtoBuffer({WithVarintLength})
result.write(1, safeConvert[uint](RequestType.LIST_PEERS))
2018-11-19 04:52:11 +02:00
result.finish()
2021-12-16 11:05:20 +01:00
proc requestDHTFindPeer(peer: PeerId, timeout = 0): ProtoBuffer =
2018-11-19 04:52:11 +02:00
## https://github.com/libp2p/go-libp2p-daemon/blob/master/dht.go
## Processing function `doDHTFindPeer(req *pb.DHTRequest)`.
let msgid = safeConvert[uint](DHTRequestType.FIND_PEER)
2018-11-19 04:52:11 +02:00
result = initProtoBuffer({WithVarintLength})
var msg = initProtoBuffer()
msg.write(1, msgid)
msg.write(2, peer)
2018-11-19 04:52:11 +02:00
if timeout > 0:
msg.write(7, hint64(timeout))
2018-11-19 04:52:11 +02:00
msg.finish()
result.write(1, safeConvert[uint](RequestType.DHT))
result.write(5, msg)
2018-11-19 04:52:11 +02:00
result.finish()
2021-12-16 11:05:20 +01:00
proc requestDHTFindPeersConnectedToPeer(peer: PeerId,
2018-11-19 04:52:11 +02:00
timeout = 0): ProtoBuffer =
## https://github.com/libp2p/go-libp2p-daemon/blob/master/dht.go
## Processing function `doDHTFindPeersConnectedToPeer(req *pb.DHTRequest)`.
let msgid = safeConvert[uint](DHTRequestType.FIND_PEERS_CONNECTED_TO_PEER)
2018-11-19 04:52:11 +02:00
result = initProtoBuffer({WithVarintLength})
var msg = initProtoBuffer()
msg.write(1, msgid)
msg.write(2, peer)
2018-11-19 04:52:11 +02:00
if timeout > 0:
msg.write(7, hint64(timeout))
2018-11-19 04:52:11 +02:00
msg.finish()
result.write(1, safeConvert[uint](RequestType.DHT))
result.write(5, msg)
2018-11-19 04:52:11 +02:00
result.finish()
proc requestDHTFindProviders(cid: Cid,
2018-11-19 04:52:11 +02:00
count: uint32, timeout = 0): ProtoBuffer =
## https://github.com/libp2p/go-libp2p-daemon/blob/master/dht.go
## Processing function `doDHTFindProviders(req *pb.DHTRequest)`.
let msgid = safeConvert[uint](DHTRequestType.FIND_PROVIDERS)
2018-11-19 04:52:11 +02:00
result = initProtoBuffer({WithVarintLength})
var msg = initProtoBuffer()
msg.write(1, msgid)
msg.write(3, cid.data.buffer)
msg.write(6, count)
2018-11-19 04:52:11 +02:00
if timeout > 0:
msg.write(7, hint64(timeout))
2018-11-19 04:52:11 +02:00
msg.finish()
result.write(1, safeConvert[uint](RequestType.DHT))
result.write(5, msg)
2018-11-19 04:52:11 +02:00
result.finish()
proc requestDHTGetClosestPeers(key: string, timeout = 0): ProtoBuffer =
## https://github.com/libp2p/go-libp2p-daemon/blob/master/dht.go
## Processing function `doDHTGetClosestPeers(req *pb.DHTRequest)`.
let msgid = safeConvert[uint](DHTRequestType.GET_CLOSEST_PEERS)
2018-11-19 04:52:11 +02:00
result = initProtoBuffer({WithVarintLength})
var msg = initProtoBuffer()
msg.write(1, msgid)
msg.write(4, key)
2018-11-19 04:52:11 +02:00
if timeout > 0:
msg.write(7, hint64(timeout))
2018-11-19 04:52:11 +02:00
msg.finish()
result.write(1, safeConvert[uint](RequestType.DHT))
result.write(5, msg)
2018-11-19 04:52:11 +02:00
result.finish()
2021-12-16 11:05:20 +01:00
proc requestDHTGetPublicKey(peer: PeerId, timeout = 0): ProtoBuffer =
2018-11-19 04:52:11 +02:00
## https://github.com/libp2p/go-libp2p-daemon/blob/master/dht.go
## Processing function `doDHTGetPublicKey(req *pb.DHTRequest)`.
let msgid = safeConvert[uint](DHTRequestType.GET_PUBLIC_KEY)
2018-11-19 04:52:11 +02:00
result = initProtoBuffer({WithVarintLength})
var msg = initProtoBuffer()
msg.write(1, msgid)
msg.write(2, peer)
2018-11-19 04:52:11 +02:00
if timeout > 0:
msg.write(7, hint64(timeout))
2018-11-19 04:52:11 +02:00
msg.finish()
result.write(1, safeConvert[uint](RequestType.DHT))
result.write(5, msg)
2018-11-19 04:52:11 +02:00
result.finish()
proc requestDHTGetValue(key: string, timeout = 0): ProtoBuffer =
## https://github.com/libp2p/go-libp2p-daemon/blob/master/dht.go
## Processing function `doDHTGetValue(req *pb.DHTRequest)`.
let msgid = safeConvert[uint](DHTRequestType.GET_VALUE)
2018-11-19 04:52:11 +02:00
result = initProtoBuffer({WithVarintLength})
var msg = initProtoBuffer()
msg.write(1, msgid)
msg.write(4, key)
2018-11-19 04:52:11 +02:00
if timeout > 0:
msg.write(7, hint64(timeout))
2018-11-19 04:52:11 +02:00
msg.finish()
result.write(1, safeConvert[uint](RequestType.DHT))
result.write(5, msg)
2018-11-19 04:52:11 +02:00
result.finish()
proc requestDHTSearchValue(key: string, timeout = 0): ProtoBuffer =
## https://github.com/libp2p/go-libp2p-daemon/blob/master/dht.go
## Processing function `doDHTSearchValue(req *pb.DHTRequest)`.
let msgid = safeConvert[uint](DHTRequestType.SEARCH_VALUE)
2018-11-19 04:52:11 +02:00
result = initProtoBuffer({WithVarintLength})
var msg = initProtoBuffer()
msg.write(1, msgid)
msg.write(4, key)
2018-11-19 04:52:11 +02:00
if timeout > 0:
msg.write(7, hint64(timeout))
2018-11-19 04:52:11 +02:00
msg.finish()
result.write(1, safeConvert[uint](RequestType.DHT))
result.write(5, msg)
2018-11-19 04:52:11 +02:00
result.finish()
2021-12-16 11:05:20 +01:00
proc requestDHTPutValue(key: string, value: openArray[byte],
2018-11-19 04:52:11 +02:00
timeout = 0): ProtoBuffer =
## https://github.com/libp2p/go-libp2p-daemon/blob/master/dht.go
## Processing function `doDHTPutValue(req *pb.DHTRequest)`.
let msgid = safeConvert[uint](DHTRequestType.PUT_VALUE)
2018-11-19 04:52:11 +02:00
result = initProtoBuffer({WithVarintLength})
var msg = initProtoBuffer()
msg.write(1, msgid)
msg.write(4, key)
msg.write(5, value)
2018-11-19 04:52:11 +02:00
if timeout > 0:
msg.write(7, hint64(timeout))
2018-11-19 04:52:11 +02:00
msg.finish()
result.write(1, uint(RequestType.DHT))
result.write(5, msg)
2018-11-19 04:52:11 +02:00
result.finish()
proc requestDHTProvide(cid: Cid, timeout = 0): ProtoBuffer =
2018-11-19 04:52:11 +02:00
## https://github.com/libp2p/go-libp2p-daemon/blob/master/dht.go
## Processing function `doDHTProvide(req *pb.DHTRequest)`.
let msgid = safeConvert[uint](DHTRequestType.PROVIDE)
2018-11-19 04:52:11 +02:00
result = initProtoBuffer({WithVarintLength})
var msg = initProtoBuffer()
msg.write(1, msgid)
msg.write(3, cid.data.buffer)
2018-11-19 04:52:11 +02:00
if timeout > 0:
msg.write(7, hint64(timeout))
2018-11-19 04:52:11 +02:00
msg.finish()
result.write(1, safeConvert[uint](RequestType.DHT))
result.write(5, msg)
2018-11-19 04:52:11 +02:00
result.finish()
2021-12-16 11:05:20 +01:00
proc requestCMTagPeer(peer: PeerId, tag: string, weight: int): ProtoBuffer =
2018-11-19 04:52:11 +02:00
## https://github.com/libp2p/go-libp2p-daemon/blob/master/connmgr.go#L18
let msgid = safeConvert[uint](ConnManagerRequestType.TAG_PEER)
2018-11-19 04:52:11 +02:00
result = initProtoBuffer({WithVarintLength})
var msg = initProtoBuffer()
msg.write(1, msgid)
msg.write(2, peer)
msg.write(3, tag)
msg.write(4, hint64(weight))
2018-11-19 04:52:11 +02:00
msg.finish()
result.write(1, safeConvert[uint](RequestType.CONNMANAGER))
result.write(6, msg)
2018-11-19 04:52:11 +02:00
result.finish()
2021-12-16 11:05:20 +01:00
proc requestCMUntagPeer(peer: PeerId, tag: string): ProtoBuffer =
2018-11-19 04:52:11 +02:00
## https://github.com/libp2p/go-libp2p-daemon/blob/master/connmgr.go#L33
let msgid = safeConvert[uint](ConnManagerRequestType.UNTAG_PEER)
2018-11-19 04:52:11 +02:00
result = initProtoBuffer({WithVarintLength})
var msg = initProtoBuffer()
msg.write(1, msgid)
msg.write(2, peer)
msg.write(3, tag)
2018-11-19 04:52:11 +02:00
msg.finish()
result.write(1, safeConvert[uint](RequestType.CONNMANAGER))
result.write(6, msg)
2018-11-19 04:52:11 +02:00
result.finish()
proc requestCMTrim(): ProtoBuffer =
## https://github.com/libp2p/go-libp2p-daemon/blob/master/connmgr.go#L47
let msgid = safeConvert[uint](ConnManagerRequestType.TRIM)
2018-11-19 04:52:11 +02:00
result = initProtoBuffer({WithVarintLength})
var msg = initProtoBuffer()
msg.write(1, msgid)
2018-11-19 04:52:11 +02:00
msg.finish()
result.write(1, safeConvert[uint](RequestType.CONNMANAGER))
result.write(6, msg)
2018-11-19 04:52:11 +02:00
result.finish()
proc requestPSGetTopics(): ProtoBuffer =
## https://github.com/libp2p/go-libp2p-daemon/blob/master/pubsub.go
## Processing function `doPubsubGetTopics(req *pb.PSRequest)`.
let msgid = safeConvert[uint](PSRequestType.GET_TOPICS)
result = initProtoBuffer({WithVarintLength})
var msg = initProtoBuffer()
msg.write(1, msgid)
msg.finish()
result.write(1, safeConvert[uint](RequestType.PUBSUB))
result.write(8, msg)
result.finish()
proc requestPSListPeers(topic: string): ProtoBuffer =
## https://github.com/libp2p/go-libp2p-daemon/blob/master/pubsub.go
## Processing function `doPubsubListPeers(req *pb.PSRequest)`.
let msgid = safeConvert[uint](PSRequestType.LIST_PEERS)
result = initProtoBuffer({WithVarintLength})
var msg = initProtoBuffer()
msg.write(1, msgid)
msg.write(2, topic)
msg.finish()
result.write(1, safeConvert[uint](RequestType.PUBSUB))
result.write(8, msg)
result.finish()
2021-12-16 11:05:20 +01:00
proc requestPSPublish(topic: string, data: openArray[byte]): ProtoBuffer =
## https://github.com/libp2p/go-libp2p-daemon/blob/master/pubsub.go
## Processing function `doPubsubPublish(req *pb.PSRequest)`.
let msgid = safeConvert[uint](PSRequestType.PUBLISH)
result = initProtoBuffer({WithVarintLength})
var msg = initProtoBuffer()
msg.write(1, msgid)
msg.write(2, topic)
msg.write(3, data)
msg.finish()
result.write(1, safeConvert[uint](RequestType.PUBSUB))
result.write(8, msg)
result.finish()
proc requestPSSubscribe(topic: string): ProtoBuffer =
## https://github.com/libp2p/go-libp2p-daemon/blob/master/pubsub.go
## Processing function `doPubsubSubscribe(req *pb.PSRequest)`.
let msgid = safeConvert[uint](PSRequestType.SUBSCRIBE)
result = initProtoBuffer({WithVarintLength})
var msg = initProtoBuffer()
msg.write(1, msgid)
msg.write(2, topic)
msg.finish()
result.write(1, safeConvert[uint](RequestType.PUBSUB))
result.write(8, msg)
result.finish()
proc checkResponse(pb: ProtoBuffer): ResponseKind {.inline.} =
2018-11-19 04:52:11 +02:00
result = ResponseKind.Malformed
var value: uint64
if getRequiredField(pb, 1, value).isOk():
2018-11-19 04:52:11 +02:00
if value == 0:
result = ResponseKind.Success
else:
result = ResponseKind.Error
2023-06-07 13:12:49 +02:00
proc getErrorMessage(pb: ProtoBuffer): string {.inline, raises: [DaemonLocalError].} =
var error: seq[byte]
if pb.getRequiredField(ResponseType.ERROR.int, error).isOk():
if initProtoBuffer(error).getRequiredField(1, result).isErr():
2018-11-19 04:52:11 +02:00
raise newException(DaemonLocalError, "Error message is missing!")
proc recvMessage(conn: StreamTransport): Future[seq[byte]] {.async.} =
var
size: uint
length: int
res: VarintResult[void]
2018-11-19 04:52:11 +02:00
var buffer = newSeq[byte](10)
try:
for i in 0..<len(buffer):
await conn.readExactly(addr buffer[i], 1)
res = PB.getUVarint(buffer.toOpenArray(0, i), length, size)
if res.isOk():
break
if res.isErr() or size > 1'u shl 22:
buffer.setLen(0)
result = buffer
return
buffer.setLen(size)
await conn.readExactly(addr buffer[0], int(size))
except TransportIncompleteError:
buffer.setLen(0)
2018-11-19 04:52:11 +02:00
result = buffer
proc newConnection*(api: DaemonAPI): Future[StreamTransport]
2023-06-07 13:12:49 +02:00
{.raises: [LPError].} =
2018-12-10 22:55:06 +02:00
result = connect(api.address)
2019-08-22 13:01:28 +03:00
proc closeConnection*(api: DaemonAPI, transp: StreamTransport): Future[void] =
result = transp.closeWait()
2018-12-10 22:55:06 +02:00
proc socketExists(address: MultiAddress): Future[bool] {.async.} =
try:
var transp = await connect(address)
await transp.closeWait()
result = true
2023-06-07 13:12:49 +02:00
except CatchableError:
result = false
when defined(windows):
2019-06-25 11:27:38 +03:00
proc getCurrentProcessId(): uint32 {.stdcall, dynlib: "kernel32",
importc: "GetCurrentProcessId".}
proc getProcessId(): int =
result = cast[int](getCurrentProcessId())
else:
proc getProcessId(): int =
result = int(posix.getpid())
proc getSocket(pattern: string,
count: ptr int): Future[MultiAddress] {.async.} =
var sockname = ""
var pid = $getProcessId()
sockname = pattern % [pid, $(count[])]
let tmpma = MultiAddress.init(sockname).tryGet()
if UNIX.match(tmpma):
while true:
count[] = count[] + 1
sockname = pattern % [pid, $(count[])]
var ma = MultiAddress.init(sockname).tryGet()
let res = await socketExists(ma)
if not res:
result = ma
break
elif TCP.match(tmpma):
sockname = pattern % [pid, "0"]
var ma = MultiAddress.init(sockname).tryGet()
var sock = createAsyncSocket(ma)
if sock.bindAsyncSocket(ma):
# Socket was successfully bound, then its free to use
count[] = count[] + 1
var ta = sock.getLocalAddress()
sockname = pattern % [pid, $ta.port]
result = MultiAddress.init(sockname).tryGet()
closeSocket(sock)
# This is forward declaration needed for newDaemonApi()
proc listPeers*(api: DaemonAPI): Future[seq[PeerInfo]] {.async, gcsafe.}
proc copyEnv(): StringTableRef =
## This procedure copy all environment variables into StringTable.
result = newStringTable(modeStyleInsensitive)
for key, val in envPairs():
result[key] = val
2018-11-19 04:52:11 +02:00
proc newDaemonApi*(flags: set[P2PDaemonFlags] = {},
bootstrapNodes: seq[string] = @[],
id: string = "",
hostAddresses: seq[MultiAddress] = @[],
announcedAddresses: seq[MultiAddress] = @[],
2018-11-19 04:52:11 +02:00
daemon = DefaultDaemonFile,
sockpath = "",
patternSock = "",
patternHandler = "",
poolSize = 10,
gossipsubHeartbeatInterval = 0,
gossipsubHeartbeatDelay = 0,
peersRequired = 2,
logFile = "",
logLevel = IpfsLogLevel.Debug): Future[DaemonAPI] {.async.} =
## Initialize connection to `go-libp2p-daemon` control socket.
##
## ``flags`` - set of P2PDaemonFlags.
##
## ``bootstrapNodes`` - list of bootnode's addresses in MultiAddress format.
## (default: @[], which means usage of default nodes inside of
## `go-libp2p-daemon`).
##
## ``id`` - path to file with identification information (default: "" which
## means - generate new random identity).
##
## ``hostAddresses`` - list of multiaddrs the host should listen on.
## (default: @[], the daemon will pick a listening port at random).
##
## ``announcedAddresses`` - list of multiaddrs the host should announce to
## the network (default: @[], the daemon will announce its own listening
## address).
##
## ``daemon`` - name of ``go-libp2p-daemon`` executable (default: "p2pd").
##
## ``sockpath`` - default control socket MultiAddress
## (default: "/unix/tmp/p2pd.sock").
##
## ``patternSock`` - MultiAddress pattern string, used to start multiple
## daemons (default on Unix: "/unix/tmp/nim-p2pd-$1-$2.sock", on Windows:
## "/ip4/127.0.0.1/tcp/$2").
##
## ``patternHandler`` - MultiAddress pattern string, used to establish
## incoming channels (default on Unix: "/unix/tmp/nim-p2pd-handle-$1-$2.sock",
## on Windows: "/ip4/127.0.0.1/tcp/$2").
##
## ``poolSize`` - size of connections pool (default: 10).
##
## ``gossipsubHeartbeatInterval`` - GossipSub protocol heartbeat interval in
## milliseconds (default: 0, use default `go-libp2p-daemon` values).
##
## ``gossipsubHeartbeatDelay`` - GossipSub protocol heartbeat delay in
## millseconds (default: 0, use default `go-libp2p-daemon` values).
##
## ``peersRequired`` - Wait until `go-libp2p-daemon` will connect to at least
## ``peersRequired`` peers before return from `newDaemonApi()` procedure
## (default: 2).
##
## ``logFile`` - Enable ``go-libp2p-daemon`` logging and store it to file
## ``logFile`` (default: "", no logging)
##
## ``logLevel`` - Set ``go-libp2p-daemon`` logging verbosity level to
2019-08-22 13:29:54 +03:00
## ``logLevel`` (default: Debug)
var api = new DaemonAPI
2018-12-10 22:55:06 +02:00
var args = newSeq[string]()
var env: StringTableRef
when defined(windows):
var patternForSocket = if len(patternSock) > 0:
patternSock
else:
DefaultIpSocketPattern
var patternForChild = if len(patternHandler) > 0:
patternHandler
else:
DefaultIpChildPattern
else:
var patternForSocket = if len(patternSock) > 0:
patternSock
else:
DefaultUnixSocketPattern
var patternForChild = if len(patternHandler) > 0:
patternHandler
else:
DefaultUnixChildPattern
api.flags = flags
2018-12-10 12:38:12 +02:00
api.servers = newSeq[P2PServer]()
api.pattern = patternForChild
api.ucounter = 1
if len(sockpath) == 0:
api.flags.excl(NoProcessCtrl)
api.address = await getSocket(patternForSocket, addr daemonsCount)
else:
api.address = MultiAddress.init(sockpath).tryGet()
api.flags.incl(NoProcessCtrl)
let res = await socketExists(api.address)
if not res:
raise newException(DaemonLocalError, "Could not connect to remote daemon")
result = api
return
# DHTFull and DHTClient could not be present at the same time
if DHTFull in flags and DHTClient in flags:
api.flags.excl(DHTClient)
# PSGossipSub and PSFloodSub could not be present at the same time
if PSGossipSub in flags and PSFloodSub in flags:
api.flags.excl(PSFloodSub)
if DHTFull in api.flags:
args.add("-dht")
if DHTClient in api.flags:
args.add("-dhtClient")
if {Bootstrap, WaitBootstrap} * api.flags != {}:
args.add("-b")
if len(logFile) != 0:
env = copyEnv()
env["IPFS_LOGGING_FMT"] = "nocolor"
env["GOLOG_FILE"] = logFile
case logLevel
of IpfsLogLevel.Critical:
env["IPFS_LOGGING"] = "CRITICAL"
of IpfsLogLevel.Error:
env["IPFS_LOGGING"] = "ERROR"
of IpfsLogLevel.Warning:
env["IPFS_LOGGING"] = "WARNING"
of IpfsLogLevel.Notice:
env["IPFS_LOGGING"] = "NOTICE"
of IpfsLogLevel.Info:
env["IPFS_LOGGING"] = "INFO"
of IpfsLogLevel.Debug:
env["IPFS_LOGGING"] = "DEBUG"
of IpfsLogLevel.Trace:
env["IPFS_LOGGING"] = "DEBUG"
env["GOLOG_TRACING_FILE"] = logFile
if PSGossipSub in api.flags:
args.add("-pubsub")
args.add("-pubsubRouter=gossipsub")
if gossipsubHeartbeatInterval != 0:
let param = $gossipsubHeartbeatInterval & "ms"
args.add("-gossipsubHeartbeatInterval=" & param)
if gossipsubHeartbeatDelay != 0:
let param = $gossipsubHeartbeatDelay & "ms"
args.add("-gossipsubHeartbeatInitialDelay=" & param)
if PSFloodSub in api.flags:
args.add("-pubsub")
args.add("-pubsubRouter=floodsub")
if api.flags * {PSFloodSub, PSGossipSub} != {}:
if PSNoSign in api.flags:
args.add("-pubsubSign=false")
if PSStrictSign in api.flags:
args.add("-pubsubSignStrict=true")
if NATPortMap in api.flags:
args.add("-natPortMap=true")
if AutoNAT in api.flags:
args.add("-autonat=true")
if AutoRelay in api.flags:
args.add("-autoRelay=true")
if RelayActive in api.flags:
args.add("-relayActive=true")
if RelayDiscovery in api.flags:
args.add("-relayDiscovery=true")
if RelayHop in api.flags:
args.add("-relayHop=true")
2021-12-16 11:05:20 +01:00
if NoInlinePeerId in api.flags:
args.add("-noInlinePeerId=true")
if len(bootstrapNodes) > 0:
args.add("-bootstrapPeers=" & bootstrapNodes.join(","))
if len(id) != 0:
args.add("-id=" & id)
if len(hostAddresses) > 0:
var opt = "-hostAddrs="
for i, address in hostAddresses:
if i > 0: opt.add ","
opt.add $address
args.add(opt)
if len(announcedAddresses) > 0:
var opt = "-announceAddrs="
for i, address in announcedAddresses:
if i > 0: opt.add ","
opt.add $address
args.add(opt)
args.add("-noise=true")
Gossip one one (#240) * allow multiple codecs per protocol (without breaking things) * add 1.1 protocol to gossip * explicit peering part 1 * explicit peering part 2 * explicit peering part 3 * PeerInfo and ControlPrune protocols * fix encodePrune * validated always, even explicit peers * prune by score (score is stub still) * add a way to pass parameters to gossip * standard setup fixes * take into account explicit direct peers in publish * add floodPublish logic * small fixes, publish still half broken * make sure to waitsub in sparse test * use var semantics to optimize table access * wip... lvalues don't work properly sadly... * big publish refactor, replenish and balance * fix internal tests * use g.peers for fanout (todo: don't include flood peers) * exclude non gossip from fanout * internal test fixes * fix flood tests * fix test's trypublish * test interop fixes * make sure to not remove peers from gossip table * restore old replenishFanout * cleanups * restore utility module import * restore trace vs debug in gossip * improve fanout replenish behavior further * triage publish nil peers (issue is on master too but just hidden behind a if/in) * getGossipPeers fixes * remove topics from pubsubpeer (was unused) * simplify rebalanceMesh (following spec) and make it finally reach D_high * better diagnostics * merge new pubsubpeer, copy 1.1 to new module * fix up merge * conditional enable gossip11 module * add back topics in peers, re-enable flood publish * add more heartbeat locking to prevent races * actually lock the heartbeat * minor fixes * with sugar * merge 1.0 * remove assertion in publish * fix multistream 1.1 multi proto * Fix merge oops * wip * fix gossip 11 upstream * gossipsub11 -> gossipsub * support interop testing * tests fixing * fix directchat build * control prune updates (pb) * wip parameters * gossip internal tests fixes * parameters wip * finishup with params * cleanups/wip * small sugar * grafted and pruned procs * wip updateScores * wip * fix logging issue * pubsubpeer, chronicles explicit override * fix internal gossip tests * wip * tables troubleshooting * score wip * score wip * fixes * fix test utils generateNodes * don't delete while iterating in score update * fix grafted defect * add a handleConnect in subscribeTopic * pruning improvements * wip * score fixes * post merge - builds gossip tests * further merge fixes * rebalance improvements and opportunistic grafting * fix test for now * restore explicit peering * implement peer exchange graft message * add an hard cap to PX * backoff time management * IWANT cap/budget * Adaptive gossip dissemination * outbound mesh quota, internal tests fixing * oversub prune score based, finish outbound quota * finishup with score and ihave budget * use go daemon 0.3.0 * import fixes * byScore cleanup score sorting * remove pointless scaling in `/` Duration operator * revert using libp2p org for daemon * interop fixes * fixes and cleanup * remove heartbeat assertion, minor debug fixes * logging improvements and cleaning up * (to revert) add some traces * add explicit topic to gossip rpcs * pubsub merge fixes and type fix in switch * Revert "(to revert) add some traces" This reverts commit 4663eaab6cc336c81cee50bc54025cf0b7bcbd99. * cleanup some now irrelevant todo * shuffle peers anyway as score might be disabled * add missing shuffle * old merge fix * more merge fixes * debug improvements * re-enable gossip internal tests * add gossip10 fallback (dormant but tested) * split gossipsub internal tests into 1.0 and 1.1 Co-authored-by: Dmitriy Ryajov <dryajov@gmail.com>
2020-09-21 18:16:29 +09:00
args.add("-quic=false")
args.add("-listen=" & $api.address)
2018-12-10 22:55:06 +02:00
# We are trying to get absolute daemon path.
let cmd = findExe(daemon)
Gossip one one (#240) * allow multiple codecs per protocol (without breaking things) * add 1.1 protocol to gossip * explicit peering part 1 * explicit peering part 2 * explicit peering part 3 * PeerInfo and ControlPrune protocols * fix encodePrune * validated always, even explicit peers * prune by score (score is stub still) * add a way to pass parameters to gossip * standard setup fixes * take into account explicit direct peers in publish * add floodPublish logic * small fixes, publish still half broken * make sure to waitsub in sparse test * use var semantics to optimize table access * wip... lvalues don't work properly sadly... * big publish refactor, replenish and balance * fix internal tests * use g.peers for fanout (todo: don't include flood peers) * exclude non gossip from fanout * internal test fixes * fix flood tests * fix test's trypublish * test interop fixes * make sure to not remove peers from gossip table * restore old replenishFanout * cleanups * restore utility module import * restore trace vs debug in gossip * improve fanout replenish behavior further * triage publish nil peers (issue is on master too but just hidden behind a if/in) * getGossipPeers fixes * remove topics from pubsubpeer (was unused) * simplify rebalanceMesh (following spec) and make it finally reach D_high * better diagnostics * merge new pubsubpeer, copy 1.1 to new module * fix up merge * conditional enable gossip11 module * add back topics in peers, re-enable flood publish * add more heartbeat locking to prevent races * actually lock the heartbeat * minor fixes * with sugar * merge 1.0 * remove assertion in publish * fix multistream 1.1 multi proto * Fix merge oops * wip * fix gossip 11 upstream * gossipsub11 -> gossipsub * support interop testing * tests fixing * fix directchat build * control prune updates (pb) * wip parameters * gossip internal tests fixes * parameters wip * finishup with params * cleanups/wip * small sugar * grafted and pruned procs * wip updateScores * wip * fix logging issue * pubsubpeer, chronicles explicit override * fix internal gossip tests * wip * tables troubleshooting * score wip * score wip * fixes * fix test utils generateNodes * don't delete while iterating in score update * fix grafted defect * add a handleConnect in subscribeTopic * pruning improvements * wip * score fixes * post merge - builds gossip tests * further merge fixes * rebalance improvements and opportunistic grafting * fix test for now * restore explicit peering * implement peer exchange graft message * add an hard cap to PX * backoff time management * IWANT cap/budget * Adaptive gossip dissemination * outbound mesh quota, internal tests fixing * oversub prune score based, finish outbound quota * finishup with score and ihave budget * use go daemon 0.3.0 * import fixes * byScore cleanup score sorting * remove pointless scaling in `/` Duration operator * revert using libp2p org for daemon * interop fixes * fixes and cleanup * remove heartbeat assertion, minor debug fixes * logging improvements and cleaning up * (to revert) add some traces * add explicit topic to gossip rpcs * pubsub merge fixes and type fix in switch * Revert "(to revert) add some traces" This reverts commit 4663eaab6cc336c81cee50bc54025cf0b7bcbd99. * cleanup some now irrelevant todo * shuffle peers anyway as score might be disabled * add missing shuffle * old merge fix * more merge fixes * debug improvements * re-enable gossip internal tests * add gossip10 fallback (dormant but tested) * split gossipsub internal tests into 1.0 and 1.1 Co-authored-by: Dmitriy Ryajov <dryajov@gmail.com>
2020-09-21 18:16:29 +09:00
trace "p2pd cmd", cmd, args
2018-12-10 22:55:06 +02:00
if len(cmd) == 0:
raise newException(DaemonLocalError, "Could not find daemon executable!")
2018-12-10 22:55:06 +02:00
# Starting daemon process
# echo "Starting ", cmd, " ", args.join(" ")
api.process =
exceptionToAssert:
startProcess(cmd, "", args, env, {poParentStreams})
2018-12-10 22:55:06 +02:00
# Waiting until daemon will not be bound to control socket.
while true:
if not api.process.running():
raise newException(DaemonLocalError,
"Daemon executable could not be started!")
let res = await socketExists(api.address)
if res:
2018-12-10 22:55:06 +02:00
break
2019-03-31 01:32:04 +02:00
await sleepAsync(500.milliseconds)
if WaitBootstrap in api.flags:
while true:
var peers = await listPeers(api)
if len(peers) >= peersRequired:
break
2019-03-31 01:32:04 +02:00
await sleepAsync(1.seconds)
result = api
2018-11-19 04:52:11 +02:00
2018-12-10 12:38:12 +02:00
proc close*(stream: P2PStream) {.async.} =
2018-11-19 04:52:11 +02:00
## Close ``stream``.
if P2PStreamFlags.Closed notin stream.flags:
2019-08-22 13:01:28 +03:00
await stream.transp.closeWait()
2018-11-19 04:52:11 +02:00
stream.transp = nil
stream.flags.incl(P2PStreamFlags.Closed)
else:
raise newException(DaemonLocalError, "Stream is already closed!")
proc close*(api: DaemonAPI) {.async.} =
## Shutdown connections to `go-libp2p-daemon` control socket.
2018-12-10 22:55:06 +02:00
# await api.pool.close()
2018-11-19 04:52:11 +02:00
# Closing all pending servers.
if len(api.servers) > 0:
var pending = newSeq[Future[void]]()
for server in api.servers:
2018-12-10 12:38:12 +02:00
server.server.stop()
server.server.close()
pending.add(server.server.join())
await allFutures(pending)
2018-12-10 12:38:12 +02:00
for server in api.servers:
let address = initTAddress(server.address).tryGet()
discard tryRemoveFile($address)
2018-12-10 12:38:12 +02:00
api.servers.setLen(0)
2018-11-19 04:52:11 +02:00
# Closing daemon's process.
2019-08-22 13:01:28 +03:00
if NoProcessCtrl notin api.flags:
2019-08-22 08:56:36 +03:00
when defined(windows):
api.process.kill()
else:
api.process.terminate()
discard api.process.waitForExit()
# Attempt to delete unix socket endpoint.
let address = initTAddress(api.address).tryGet()
if address.family == AddressFamily.Unix:
discard tryRemoveFile($address)
2018-11-19 04:52:11 +02:00
template withMessage(m, body: untyped): untyped =
let kind = m.checkResponse()
if kind == ResponseKind.Error:
raise newException(DaemonRemoteError, m.getErrorMessage())
elif kind == ResponseKind.Malformed:
raise newException(DaemonLocalError, "Malformed message received!")
else:
body
proc transactMessage(transp: StreamTransport,
pb: ProtoBuffer): Future[ProtoBuffer] {.async.} =
let length = pb.getLen()
let res = await transp.write(pb.getPtr(), length)
if res != length:
raise newException(DaemonLocalError, "Could not send message to daemon!")
var message = await transp.recvMessage()
if len(message) == 0:
raise newException(DaemonLocalError, "Incorrect or empty message received!")
2018-11-19 04:52:11 +02:00
result = initProtoBuffer(message)
proc getPeerInfo(pb: ProtoBuffer): PeerInfo
2023-06-07 13:12:49 +02:00
{.raises: [DaemonLocalError].} =
2018-11-19 04:52:11 +02:00
## Get PeerInfo object from ``pb``.
result.addresses = newSeq[MultiAddress]()
if pb.getRequiredField(1, result.peer).isErr():
raise newException(DaemonLocalError, "Incorrect or empty message received!")
discard pb.getRepeatedField(2, result.addresses)
2018-11-19 04:52:11 +02:00
proc identity*(api: DaemonAPI): Future[PeerInfo] {.async.} =
## Get Node identity information
2018-12-10 22:55:06 +02:00
var transp = await api.newConnection()
2018-11-19 04:52:11 +02:00
try:
var pb = await transactMessage(transp, requestIdentity())
pb.withMessage() do:
var res: seq[byte]
if pb.getRequiredField(ResponseType.IDENTITY.int, res).isOk():
var resPb = initProtoBuffer(res)
result = getPeerInfo(resPb)
2018-11-19 04:52:11 +02:00
finally:
2018-12-10 22:55:06 +02:00
await api.closeConnection(transp)
2018-11-19 04:52:11 +02:00
2021-12-16 11:05:20 +01:00
proc connect*(api: DaemonAPI, peer: PeerId,
addresses: seq[MultiAddress],
timeout = 0) {.async.} =
2018-11-19 04:52:11 +02:00
## Connect to remote peer with id ``peer`` and addresses ``addresses``.
2018-12-10 22:55:06 +02:00
var transp = await api.newConnection()
2018-11-19 04:52:11 +02:00
try:
var pb = await transp.transactMessage(requestConnect(peer, addresses,
timeout))
2018-11-19 04:52:11 +02:00
pb.withMessage() do:
discard
2023-06-07 13:12:49 +02:00
except CatchableError:
2018-12-10 22:55:06 +02:00
await api.closeConnection(transp)
2018-11-19 04:52:11 +02:00
2021-12-16 11:05:20 +01:00
proc disconnect*(api: DaemonAPI, peer: PeerId) {.async.} =
2018-11-19 04:52:11 +02:00
## Disconnect from remote peer with id ``peer``.
2018-12-10 22:55:06 +02:00
var transp = await api.newConnection()
2018-11-19 04:52:11 +02:00
try:
var pb = await transp.transactMessage(requestDisconnect(peer))
pb.withMessage() do:
discard
finally:
2018-12-10 22:55:06 +02:00
await api.closeConnection(transp)
2018-11-19 04:52:11 +02:00
2021-12-16 11:05:20 +01:00
proc openStream*(api: DaemonAPI, peer: PeerId,
protocols: seq[string],
timeout = 0): Future[P2PStream] {.async.} =
2018-11-19 04:52:11 +02:00
## Open new stream to peer ``peer`` using one of the protocols in
## ``protocols``. Returns ``StreamTransport`` for the stream.
2018-12-10 22:55:06 +02:00
var transp = await api.newConnection()
2018-11-19 04:52:11 +02:00
var stream = new P2PStream
try:
var pb = await transp.transactMessage(requestStreamOpen(peer, protocols,
timeout))
2018-11-19 04:52:11 +02:00
pb.withMessage() do:
var res: seq[byte]
if pb.getRequiredField(ResponseType.STREAMINFO.int, res).isOk():
let resPb = initProtoBuffer(res)
# stream.peer = newSeq[byte]()
var raddress = newSeq[byte]()
2018-11-19 04:52:11 +02:00
stream.protocol = ""
resPb.getRequiredField(1, stream.peer).tryGet()
resPb.getRequiredField(2, raddress).tryGet()
stream.raddress = MultiAddress.init(raddress).tryGet()
resPb.getRequiredField(3, stream.protocol).tryGet()
2018-11-19 04:52:11 +02:00
stream.flags.incl(Outbound)
stream.transp = transp
result = stream
except CatchableError as exc:
2018-12-10 22:55:06 +02:00
await api.closeConnection(transp)
raise exc
2018-11-19 04:52:11 +02:00
proc streamHandler(server: StreamServer, transp: StreamTransport) {.async.} =
var api = getUserData[DaemonAPI](server)
var message = await transp.recvMessage()
var pb = initProtoBuffer(message)
var stream = new P2PStream
var raddress = newSeq[byte]()
2018-11-19 04:52:11 +02:00
stream.protocol = ""
pb.getRequiredField(1, stream.peer).tryGet()
pb.getRequiredField(2, raddress).tryGet()
stream.raddress = MultiAddress.init(raddress).tryGet()
pb.getRequiredField(3, stream.protocol).tryGet()
2018-11-19 04:52:11 +02:00
stream.flags.incl(Inbound)
stream.transp = transp
if len(stream.protocol) > 0:
var handler = api.handlers.getOrDefault(stream.protocol)
if not isNil(handler):
asyncSpawn handler(api, stream)
2018-11-19 04:52:11 +02:00
proc addHandler*(api: DaemonAPI, protocols: seq[string],
2023-06-07 13:12:49 +02:00
handler: P2PStreamCallback) {.async, raises: [LPError].} =
2018-11-19 04:52:11 +02:00
## Add stream handler ``handler`` for set of protocols ``protocols``.
2018-12-10 22:55:06 +02:00
var transp = await api.newConnection()
let maddress = await getSocket(api.pattern, addr api.ucounter)
var server = createStreamServer(maddress, streamHandler, udata = api)
2018-11-19 04:52:11 +02:00
try:
for item in protocols:
api.handlers[item] = handler
server.start()
var pb = await transp.transactMessage(requestStreamHandler(maddress,
2018-11-19 04:52:11 +02:00
protocols))
pb.withMessage() do:
api.servers.add(P2PServer(server: server, address: maddress))
except CatchableError as exc:
2018-11-19 04:52:11 +02:00
for item in protocols:
api.handlers.del(item)
server.stop()
server.close()
await server.join()
raise exc
finally:
2018-12-10 22:55:06 +02:00
await api.closeConnection(transp)
2018-11-19 04:52:11 +02:00
proc listPeers*(api: DaemonAPI): Future[seq[PeerInfo]] {.async.} =
## Get list of remote peers to which we are currently connected.
2018-12-10 22:55:06 +02:00
var transp = await api.newConnection()
2018-11-19 04:52:11 +02:00
try:
var pb = await transp.transactMessage(requestListPeers())
pb.withMessage() do:
result = newSeq[PeerInfo]()
var ress: seq[seq[byte]]
if pb.getRequiredRepeatedField(ResponseType.PEERINFO.int, ress).isOk():
for p in ress:
let peer = initProtoBuffer(p).getPeerInfo()
2018-11-19 04:52:11 +02:00
result.add(peer)
finally:
2018-12-10 22:55:06 +02:00
await api.closeConnection(transp)
2018-11-19 04:52:11 +02:00
2021-12-16 11:05:20 +01:00
proc cmTagPeer*(api: DaemonAPI, peer: PeerId, tag: string,
2018-11-19 04:52:11 +02:00
weight: int) {.async.} =
## Tag peer with id ``peer`` using ``tag`` and ``weight``.
2018-12-10 22:55:06 +02:00
var transp = await api.newConnection()
2018-11-19 04:52:11 +02:00
try:
var pb = await transp.transactMessage(requestCMTagPeer(peer, tag, weight))
withMessage(pb) do:
discard
finally:
2018-12-10 22:55:06 +02:00
await api.closeConnection(transp)
2018-11-19 04:52:11 +02:00
2021-12-16 11:05:20 +01:00
proc cmUntagPeer*(api: DaemonAPI, peer: PeerId, tag: string) {.async.} =
2018-11-19 04:52:11 +02:00
## Remove tag ``tag`` from peer with id ``peer``.
2018-12-10 22:55:06 +02:00
var transp = await api.newConnection()
2018-11-19 04:52:11 +02:00
try:
var pb = await transp.transactMessage(requestCMUntagPeer(peer, tag))
withMessage(pb) do:
discard
finally:
2018-12-10 22:55:06 +02:00
await api.closeConnection(transp)
2018-11-19 04:52:11 +02:00
proc cmTrimPeers*(api: DaemonAPI) {.async.} =
## Trim all connections.
2018-12-10 22:55:06 +02:00
var transp = await api.newConnection()
2018-11-19 04:52:11 +02:00
try:
var pb = await transp.transactMessage(requestCMTrim())
withMessage(pb) do:
discard
finally:
2018-12-10 22:55:06 +02:00
await api.closeConnection(transp)
2018-11-19 04:52:11 +02:00
proc dhtGetSinglePeerInfo(pb: ProtoBuffer): PeerInfo
2023-06-07 13:12:49 +02:00
{.raises: [DaemonLocalError].} =
var res: seq[byte]
if pb.getRequiredField(2, res).isOk():
result = initProtoBuffer(res).getPeerInfo()
2018-11-19 04:52:11 +02:00
else:
raise newException(DaemonLocalError, "Missing required field `peer`!")
proc dhtGetSingleValue(pb: ProtoBuffer): seq[byte]
2023-06-07 13:12:49 +02:00
{.raises: [DaemonLocalError].} =
2018-11-19 04:52:11 +02:00
result = newSeq[byte]()
if pb.getRequiredField(3, result).isErr():
2018-11-19 04:52:11 +02:00
raise newException(DaemonLocalError, "Missing field `value`!")
proc dhtGetSinglePublicKey(pb: ProtoBuffer): PublicKey
2023-06-07 13:12:49 +02:00
{.raises: [DaemonLocalError].} =
if pb.getRequiredField(3, result).isErr():
raise newException(DaemonLocalError, "Missing field `value`!")
2021-12-16 11:05:20 +01:00
proc dhtGetSinglePeerId(pb: ProtoBuffer): PeerId
2023-06-07 13:12:49 +02:00
{.raises: [DaemonLocalError].} =
if pb.getRequiredField(3, result).isErr():
raise newException(DaemonLocalError, "Missing field `value`!")
proc enterDhtMessage(pb: ProtoBuffer, rt: DHTResponseType): ProtoBuffer
2023-06-07 13:12:49 +02:00
{.inline, raises: [DaemonLocalError].} =
var dhtResponse: seq[byte]
if pb.getRequiredField(ResponseType.DHT.int, dhtResponse).isOk():
var pbDhtResponse = initProtoBuffer(dhtResponse)
var dtype: uint
if pbDhtResponse.getRequiredField(1, dtype).isErr():
2018-11-19 04:52:11 +02:00
raise newException(DaemonLocalError, "Missing required DHT field `type`!")
if dtype != safeConvert[uint](rt):
2018-11-19 04:52:11 +02:00
raise newException(DaemonLocalError, "Wrong DHT answer type! ")
var value: seq[byte]
if pbDhtResponse.getRequiredField(3, value).isErr():
raise newException(DaemonLocalError, "Missing required DHT field `value`!")
return initProtoBuffer(value)
2018-11-19 04:52:11 +02:00
else:
raise newException(DaemonLocalError, "Wrong message type!")
proc enterPsMessage(pb: ProtoBuffer): ProtoBuffer
2023-06-07 13:12:49 +02:00
{.inline, raises: [DaemonLocalError].} =
var res: seq[byte]
if pb.getRequiredField(ResponseType.PUBSUB.int, res).isErr():
raise newException(DaemonLocalError, "Wrong message type!")
initProtoBuffer(res)
proc getDhtMessageType(pb: ProtoBuffer): DHTResponseType
2023-06-07 13:12:49 +02:00
{.inline, raises: [DaemonLocalError].} =
2018-11-19 04:52:11 +02:00
var dtype: uint
if pb.getRequiredField(1, dtype).isErr():
2018-11-19 04:52:11 +02:00
raise newException(DaemonLocalError, "Missing required DHT field `type`!")
if dtype == safeConvert[uint](DHTResponseType.VALUE):
2018-11-19 04:52:11 +02:00
result = DHTResponseType.VALUE
elif dtype == safeConvert[uint](DHTResponseType.END):
2018-11-19 04:52:11 +02:00
result = DHTResponseType.END
else:
raise newException(DaemonLocalError, "Wrong DHT answer type!")
2021-12-16 11:05:20 +01:00
proc dhtFindPeer*(api: DaemonAPI, peer: PeerId,
2018-11-19 04:52:11 +02:00
timeout = 0): Future[PeerInfo] {.async.} =
## Find peer with id ``peer`` and return peer information ``PeerInfo``.
##
## You can specify timeout for DHT request with ``timeout`` value. ``0`` value
## means no timeout.
2018-12-10 22:55:06 +02:00
var transp = await api.newConnection()
2018-11-19 04:52:11 +02:00
try:
var pb = await transp.transactMessage(requestDHTFindPeer(peer, timeout))
withMessage(pb) do:
result = pb.enterDhtMessage(DHTResponseType.VALUE).dhtGetSinglePeerInfo()
2018-11-19 04:52:11 +02:00
finally:
2018-12-10 22:55:06 +02:00
await api.closeConnection(transp)
2018-11-19 04:52:11 +02:00
2021-12-16 11:05:20 +01:00
proc dhtGetPublicKey*(api: DaemonAPI, peer: PeerId,
timeout = 0): Future[PublicKey] {.async.} =
2018-11-19 04:52:11 +02:00
## Get peer's public key from peer with id ``peer``.
##
## You can specify timeout for DHT request with ``timeout`` value. ``0`` value
## means no timeout.
2018-12-10 22:55:06 +02:00
var transp = await api.newConnection()
2018-11-19 04:52:11 +02:00
try:
var pb = await transp.transactMessage(requestDHTGetPublicKey(peer, timeout))
withMessage(pb) do:
result = pb.enterDhtMessage(DHTResponseType.VALUE).dhtGetSinglePublicKey()
2018-11-19 04:52:11 +02:00
finally:
2018-12-10 22:55:06 +02:00
await api.closeConnection(transp)
2018-11-19 04:52:11 +02:00
proc dhtGetValue*(api: DaemonAPI, key: string,
timeout = 0): Future[seq[byte]] {.async.} =
## Get value associated with ``key``.
##
## You can specify timeout for DHT request with ``timeout`` value. ``0`` value
## means no timeout.
2018-12-10 22:55:06 +02:00
var transp = await api.newConnection()
2018-11-19 04:52:11 +02:00
try:
var pb = await transp.transactMessage(requestDHTGetValue(key, timeout))
withMessage(pb) do:
result = pb.enterDhtMessage(DHTResponseType.VALUE).dhtGetSingleValue()
2018-11-19 04:52:11 +02:00
finally:
2018-12-10 22:55:06 +02:00
await api.closeConnection(transp)
2018-11-19 04:52:11 +02:00
proc dhtPutValue*(api: DaemonAPI, key: string, value: seq[byte],
timeout = 0) {.async.} =
## Associate ``value`` with ``key``.
##
## You can specify timeout for DHT request with ``timeout`` value. ``0`` value
## means no timeout.
2018-12-10 22:55:06 +02:00
var transp = await api.newConnection()
2018-11-19 04:52:11 +02:00
try:
var pb = await transp.transactMessage(requestDHTPutValue(key, value,
timeout))
withMessage(pb) do:
discard
finally:
2018-12-10 22:55:06 +02:00
await api.closeConnection(transp)
2018-11-19 04:52:11 +02:00
proc dhtProvide*(api: DaemonAPI, cid: Cid, timeout = 0) {.async.} =
2018-11-19 04:52:11 +02:00
## Provide content with id ``cid``.
##
## You can specify timeout for DHT request with ``timeout`` value. ``0`` value
## means no timeout.
2018-12-10 22:55:06 +02:00
var transp = await api.newConnection()
2018-11-19 04:52:11 +02:00
try:
var pb = await transp.transactMessage(requestDHTProvide(cid, timeout))
withMessage(pb) do:
discard
finally:
2018-12-10 22:55:06 +02:00
await api.closeConnection(transp)
2018-11-19 04:52:11 +02:00
2021-12-16 11:05:20 +01:00
proc dhtFindPeersConnectedToPeer*(api: DaemonAPI, peer: PeerId,
2018-11-19 04:52:11 +02:00
timeout = 0): Future[seq[PeerInfo]] {.async.} =
## Find peers which are connected to peer with id ``peer``.
##
## You can specify timeout for DHT request with ``timeout`` value. ``0`` value
## means no timeout.
2018-12-10 22:55:06 +02:00
var transp = await api.newConnection()
2018-11-19 04:52:11 +02:00
var list = newSeq[PeerInfo]()
try:
let spb = requestDHTFindPeersConnectedToPeer(peer, timeout)
var pb = await transp.transactMessage(spb)
withMessage(pb) do:
discard pb.enterDhtMessage(DHTResponseType.BEGIN)
2018-11-19 04:52:11 +02:00
while true:
var message = await transp.recvMessage()
if len(message) == 0:
break
2018-11-19 04:52:11 +02:00
var cpb = initProtoBuffer(message)
if cpb.getDhtMessageType() == DHTResponseType.END:
break
list.add(cpb.dhtGetSinglePeerInfo())
result = list
finally:
2018-12-10 22:55:06 +02:00
await api.closeConnection(transp)
2018-11-19 04:52:11 +02:00
proc dhtGetClosestPeers*(api: DaemonAPI, key: string,
2021-12-16 11:05:20 +01:00
timeout = 0): Future[seq[PeerId]] {.async.} =
2018-11-19 04:52:11 +02:00
## Get closest peers for ``key``.
##
## You can specify timeout for DHT request with ``timeout`` value. ``0`` value
## means no timeout.
2018-12-10 22:55:06 +02:00
var transp = await api.newConnection()
2021-12-16 11:05:20 +01:00
var list = newSeq[PeerId]()
2018-11-19 04:52:11 +02:00
try:
let spb = requestDHTGetClosestPeers(key, timeout)
var pb = await transp.transactMessage(spb)
withMessage(pb) do:
discard pb.enterDhtMessage(DHTResponseType.BEGIN)
2018-11-19 04:52:11 +02:00
while true:
var message = await transp.recvMessage()
if len(message) == 0:
break
2018-11-19 04:52:11 +02:00
var cpb = initProtoBuffer(message)
if cpb.getDhtMessageType() == DHTResponseType.END:
break
2021-12-16 11:05:20 +01:00
list.add(cpb.dhtGetSinglePeerId())
2018-11-19 04:52:11 +02:00
result = list
finally:
2018-12-10 22:55:06 +02:00
await api.closeConnection(transp)
2018-11-19 04:52:11 +02:00
proc dhtFindProviders*(api: DaemonAPI, cid: Cid, count: uint32,
timeout = 0): Future[seq[PeerInfo]] {.async.} =
2018-11-19 04:52:11 +02:00
## Get ``count`` providers for content with id ``cid``.
##
## You can specify timeout for DHT request with ``timeout`` value. ``0`` value
## means no timeout.
2018-12-10 22:55:06 +02:00
var transp = await api.newConnection()
2018-11-19 04:52:11 +02:00
var list = newSeq[PeerInfo]()
try:
let spb = requestDHTFindProviders(cid, count, timeout)
var pb = await transp.transactMessage(spb)
withMessage(pb) do:
discard pb.enterDhtMessage(DHTResponseType.BEGIN)
2018-11-19 04:52:11 +02:00
while true:
var message = await transp.recvMessage()
if len(message) == 0:
break
2018-11-19 04:52:11 +02:00
var cpb = initProtoBuffer(message)
if cpb.getDhtMessageType() == DHTResponseType.END:
break
list.add(cpb.dhtGetSinglePeerInfo())
result = list
finally:
2018-12-10 22:55:06 +02:00
await api.closeConnection(transp)
2018-11-19 04:52:11 +02:00
proc dhtSearchValue*(api: DaemonAPI, key: string,
timeout = 0): Future[seq[seq[byte]]] {.async.} =
## Search for value with ``key``, return list of values found.
##
## You can specify timeout for DHT request with ``timeout`` value. ``0`` value
## means no timeout.
2018-12-10 22:55:06 +02:00
var transp = await api.newConnection()
2018-11-19 04:52:11 +02:00
var list = newSeq[seq[byte]]()
try:
var pb = await transp.transactMessage(requestDHTSearchValue(key, timeout))
withMessage(pb) do:
discard pb.enterDhtMessage(DHTResponseType.BEGIN)
2018-11-19 04:52:11 +02:00
while true:
var message = await transp.recvMessage()
if len(message) == 0:
break
2018-11-19 04:52:11 +02:00
var cpb = initProtoBuffer(message)
if cpb.getDhtMessageType() == DHTResponseType.END:
break
list.add(cpb.dhtGetSingleValue())
result = list
finally:
2018-12-10 22:55:06 +02:00
await api.closeConnection(transp)
proc pubsubGetTopics*(api: DaemonAPI): Future[seq[string]] {.async.} =
## Get list of topics this node is subscribed to.
2018-12-10 22:55:06 +02:00
var transp = await api.newConnection()
try:
var pb = await transp.transactMessage(requestPSGetTopics())
withMessage(pb) do:
let innerPb = pb.enterPsMessage()
var topics = newSeq[string]()
discard innerPb.getRepeatedField(1, topics)
result = topics
finally:
2018-12-10 22:55:06 +02:00
await api.closeConnection(transp)
proc pubsubListPeers*(api: DaemonAPI,
2021-12-16 11:05:20 +01:00
topic: string): Future[seq[PeerId]] {.async.} =
## Get list of peers we are connected to and which also subscribed to topic
## ``topic``.
2018-12-10 22:55:06 +02:00
var transp = await api.newConnection()
try:
var pb = await transp.transactMessage(requestPSListPeers(topic))
withMessage(pb) do:
2021-12-16 11:05:20 +01:00
var peer: PeerId
let innerPb = pb.enterPsMessage()
var peers = newSeq[seq[byte]]()
discard innerPb.getRepeatedField(2, peers)
result = peers.mapIt(PeerId.init(it).get())
finally:
2018-12-10 22:55:06 +02:00
await api.closeConnection(transp)
proc pubsubPublish*(api: DaemonAPI, topic: string,
value: seq[byte]) {.async.} =
## Get list of peer identifiers which are subscribed to topic ``topic``.
2018-12-10 22:55:06 +02:00
var transp = await api.newConnection()
try:
var pb = await transp.transactMessage(requestPSPublish(topic, value))
withMessage(pb) do:
discard
finally:
2018-12-10 22:55:06 +02:00
await api.closeConnection(transp)
proc getPubsubMessage*(pb: ProtoBuffer): PubSubMessage =
result.data = newSeq[byte]()
result.seqno = newSeq[byte]()
discard pb.getField(1, result.peer)
discard pb.getField(2, result.data)
discard pb.getField(3, result.seqno)
discard pb.getRepeatedField(4, result.topics)
discard pb.getField(5, result.signature)
discard pb.getField(6, result.key)
proc pubsubLoop(api: DaemonAPI, ticket: PubsubTicket) {.async.} =
while true:
var pbmessage = await ticket.transp.recvMessage()
if len(pbmessage) == 0:
break
var pb = initProtoBuffer(pbmessage)
var message = pb.getPubsubMessage()
## We can do here `await` too
let res = await ticket.handler(api, ticket, message)
if not res:
ticket.transp.close()
await ticket.transp.join()
break
proc pubsubSubscribe*(api: DaemonAPI, topic: string,
handler: P2PPubSubCallback): Future[PubsubTicket] {.async.} =
## Subscribe to topic ``topic``.
2018-12-10 22:55:06 +02:00
var transp = await api.newConnection()
try:
var pb = await transp.transactMessage(requestPSSubscribe(topic))
pb.withMessage() do:
var ticket = new PubsubTicket
ticket.topic = topic
ticket.handler = handler
ticket.transp = transp
asyncSpawn pubsubLoop(api, ticket)
result = ticket
except CatchableError as exc:
await api.closeConnection(transp)
raise exc
proc shortLog*(pinfo: PeerInfo): string =
## Get string representation of ``PeerInfo`` object.
result = newStringOfCap(128)
2021-12-16 11:05:20 +01:00
result.add("{PeerId: '")
result.add($pinfo.peer.shortLog())
result.add("' Addresses: [")
let length = len(pinfo.addresses)
for i in 0..<length:
result.add("'")
result.add($pinfo.addresses[i])
result.add("'")
if i < length - 1:
result.add(", ")
result.add("]}")
if len(pinfo.addresses) > 0:
result = result