2020-12-21 11:45:07 +00:00
|
|
|
## Nim-LibP2P
|
|
|
|
## Copyright (c) 2018 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.
|
|
|
|
|
|
|
|
## This module implementes API for libp2p peer.
|
|
|
|
|
|
|
|
{.push raises: [Defect].}
|
|
|
|
|
2021-02-02 11:46:38 +00:00
|
|
|
import
|
|
|
|
std/[hashes, strutils],
|
|
|
|
stew/[base58, results],
|
|
|
|
chronicles,
|
|
|
|
nimcrypto/utils,
|
|
|
|
./crypto/crypto, ./multicodec, ./multihash, ./vbuffer,
|
|
|
|
./protobuf/minprotobuf
|
2020-12-21 11:45:07 +00:00
|
|
|
|
|
|
|
export results
|
|
|
|
|
|
|
|
const
|
|
|
|
maxInlineKeyLength* = 42
|
|
|
|
|
|
|
|
type
|
|
|
|
PeerID* = object
|
|
|
|
data*: seq[byte]
|
|
|
|
|
|
|
|
func `$`*(pid: PeerID): string =
|
|
|
|
## Return base58 encoded ``pid`` representation.
|
|
|
|
Base58.encode(pid.data)
|
|
|
|
|
|
|
|
func shortLog*(pid: PeerId): string =
|
|
|
|
## Returns compact string representation of ``pid``.
|
|
|
|
var spid = $pid
|
2021-02-02 11:46:38 +00:00
|
|
|
if len(spid) > 10:
|
|
|
|
spid[3] = '*'
|
|
|
|
spid.delete(4, spid.high - 6)
|
|
|
|
|
|
|
|
spid
|
2020-12-21 11:45:07 +00:00
|
|
|
|
|
|
|
chronicles.formatIt(PeerID): shortLog(it)
|
|
|
|
|
2021-02-02 11:46:38 +00:00
|
|
|
func toBytes*(pid: PeerID, data: var openarray[byte]): int =
|
2020-12-21 11:45:07 +00:00
|
|
|
## Store PeerID ``pid`` to array of bytes ``data``.
|
|
|
|
##
|
|
|
|
## Returns number of bytes needed to store ``pid``.
|
|
|
|
result = len(pid.data)
|
|
|
|
if len(data) >= result and result > 0:
|
|
|
|
copyMem(addr data[0], unsafeAddr pid.data[0], result)
|
|
|
|
|
2021-02-02 11:46:38 +00:00
|
|
|
template getBytes*(pid: PeerID): seq[byte] =
|
2020-12-21 11:45:07 +00:00
|
|
|
## Return PeerID ``pid`` as array of bytes.
|
2021-02-02 11:46:38 +00:00
|
|
|
pid.data
|
2020-12-21 11:45:07 +00:00
|
|
|
|
2021-02-02 11:46:38 +00:00
|
|
|
func hex*(pid: PeerID): string =
|
2020-12-21 11:45:07 +00:00
|
|
|
## Returns hexadecimal string representation of ``pid``.
|
2021-02-02 11:46:38 +00:00
|
|
|
toHex(pid.data)
|
2020-12-21 11:45:07 +00:00
|
|
|
|
2021-02-02 11:46:38 +00:00
|
|
|
template len*(pid: PeerID): int =
|
2020-12-21 11:45:07 +00:00
|
|
|
## Returns length of ``pid`` binary representation.
|
2021-02-02 11:46:38 +00:00
|
|
|
len(pid.data)
|
2020-12-21 11:45:07 +00:00
|
|
|
|
2021-02-02 11:46:38 +00:00
|
|
|
func cmp*(a, b: PeerID): int =
|
2020-12-21 11:45:07 +00:00
|
|
|
## Compares two peer ids ``a`` and ``b``.
|
|
|
|
## Returns:
|
|
|
|
##
|
|
|
|
## | 0 iff a == b
|
|
|
|
## | < 0 iff a < b
|
|
|
|
## | > 0 iff a > b
|
|
|
|
var i = 0
|
|
|
|
var m = min(len(a.data), len(b.data))
|
|
|
|
while i < m:
|
|
|
|
result = ord(a.data[i]) - ord(b.data[i])
|
|
|
|
if result != 0: return
|
|
|
|
inc(i)
|
|
|
|
result = len(a.data) - len(b.data)
|
|
|
|
|
2021-02-02 11:46:38 +00:00
|
|
|
template `<=`*(a, b: PeerID): bool =
|
2020-12-21 11:45:07 +00:00
|
|
|
(cmp(a, b) <= 0)
|
|
|
|
|
2021-02-02 11:46:38 +00:00
|
|
|
template `<`*(a, b: PeerID): bool =
|
2020-12-21 11:45:07 +00:00
|
|
|
(cmp(a, b) < 0)
|
|
|
|
|
2021-02-02 11:46:38 +00:00
|
|
|
template `>=`*(a, b: PeerID): bool =
|
2020-12-21 11:45:07 +00:00
|
|
|
(cmp(a, b) >= 0)
|
|
|
|
|
2021-02-02 11:46:38 +00:00
|
|
|
template `>`*(a, b: PeerID): bool =
|
2020-12-21 11:45:07 +00:00
|
|
|
(cmp(a, b) > 0)
|
|
|
|
|
2021-02-02 11:46:38 +00:00
|
|
|
template `==`*(a, b: PeerID): bool =
|
2020-12-21 11:45:07 +00:00
|
|
|
(cmp(a, b) == 0)
|
|
|
|
|
2021-02-02 11:46:38 +00:00
|
|
|
template hash*(pid: PeerID): Hash =
|
|
|
|
hash(pid.data)
|
2020-12-21 11:45:07 +00:00
|
|
|
|
2021-02-02 11:46:38 +00:00
|
|
|
func validate*(pid: PeerID): bool =
|
2020-12-21 11:45:07 +00:00
|
|
|
## Validate check if ``pid`` is empty or not.
|
2021-02-02 11:46:38 +00:00
|
|
|
len(pid.data) > 0 and MultiHash.validate(pid.data)
|
2020-12-21 11:45:07 +00:00
|
|
|
|
2021-02-02 11:46:38 +00:00
|
|
|
func hasPublicKey*(pid: PeerID): bool =
|
2020-12-21 11:45:07 +00:00
|
|
|
## Returns ``true`` if ``pid`` is small enough to hold public key inside.
|
|
|
|
if len(pid.data) > 0:
|
|
|
|
var mh: MultiHash
|
|
|
|
if MultiHash.decode(pid.data, mh).isOk:
|
|
|
|
if mh.mcodec == multiCodec("identity"):
|
|
|
|
result = true
|
|
|
|
|
2021-02-02 11:46:38 +00:00
|
|
|
func extractPublicKey*(pid: PeerID, pubkey: var PublicKey): bool =
|
2020-12-21 11:45:07 +00:00
|
|
|
## Returns ``true`` if public key was successfully decoded from PeerID
|
|
|
|
## ``pid``and stored to ``pubkey``.
|
|
|
|
##
|
|
|
|
## Returns ``false`` otherwise.
|
|
|
|
if len(pid.data) > 0:
|
2021-02-02 11:46:38 +00:00
|
|
|
var mh: MultiHash
|
2020-12-21 11:45:07 +00:00
|
|
|
if MultiHash.decode(pid.data, mh).isOk:
|
|
|
|
if mh.mcodec == multiCodec("identity"):
|
|
|
|
let length = len(mh.data.buffer)
|
|
|
|
result = pubkey.init(mh.data.buffer.toOpenArray(mh.dpos, length - 1))
|
|
|
|
|
2021-02-02 11:46:38 +00:00
|
|
|
func init*(pid: var PeerID, data: openarray[byte]): bool =
|
2020-12-21 11:45:07 +00:00
|
|
|
## Initialize peer id from raw binary representation ``data``.
|
|
|
|
##
|
|
|
|
## Returns ``true`` if peer was successfully initialiazed.
|
|
|
|
var p = PeerID(data: @data)
|
|
|
|
if p.validate():
|
|
|
|
pid = p
|
|
|
|
result = true
|
|
|
|
|
2021-02-02 11:46:38 +00:00
|
|
|
func init*(pid: var PeerID, data: string): bool =
|
2020-12-21 11:45:07 +00:00
|
|
|
## Initialize peer id from base58 encoded string representation.
|
|
|
|
##
|
|
|
|
## Returns ``true`` if peer was successfully initialiazed.
|
|
|
|
var p = newSeq[byte](len(data) + 4)
|
|
|
|
var length = 0
|
|
|
|
if Base58.decode(data, p, length) == Base58Status.Success:
|
|
|
|
p.setLen(length)
|
|
|
|
var opid: PeerID
|
|
|
|
shallowCopy(opid.data, p)
|
|
|
|
if opid.validate():
|
|
|
|
pid = opid
|
|
|
|
result = true
|
|
|
|
|
2021-02-02 11:46:38 +00:00
|
|
|
func init*(t: typedesc[PeerID], data: openarray[byte]): Result[PeerID, cstring] =
|
2020-12-21 11:45:07 +00:00
|
|
|
## Create new peer id from raw binary representation ``data``.
|
|
|
|
var res: PeerID
|
|
|
|
if not init(res, data):
|
|
|
|
err("peerid: incorrect PeerID binary form")
|
|
|
|
else:
|
|
|
|
ok(res)
|
|
|
|
|
2021-02-02 11:46:38 +00:00
|
|
|
func init*(t: typedesc[PeerID], data: string): Result[PeerID, cstring] =
|
2020-12-21 11:45:07 +00:00
|
|
|
## Create new peer id from base58 encoded string representation ``data``.
|
|
|
|
var res: PeerID
|
|
|
|
if not init(res, data):
|
|
|
|
err("peerid: incorrect PeerID string")
|
|
|
|
else:
|
|
|
|
ok(res)
|
|
|
|
|
2021-02-02 11:46:38 +00:00
|
|
|
func init*(t: typedesc[PeerID], pubkey: PublicKey): Result[PeerID, cstring] =
|
2020-12-21 11:45:07 +00:00
|
|
|
## Create new peer id from public key ``pubkey``.
|
2021-02-02 11:46:38 +00:00
|
|
|
var pubraw = ? pubkey.getBytes().orError(
|
|
|
|
cstring("peerid: failed to get bytes from given key"))
|
2020-12-21 11:45:07 +00:00
|
|
|
var mh: MultiHash
|
|
|
|
if len(pubraw) <= maxInlineKeyLength:
|
|
|
|
mh = ? MultiHash.digest("identity", pubraw)
|
|
|
|
else:
|
|
|
|
mh = ? MultiHash.digest("sha2-256", pubraw)
|
|
|
|
ok(PeerID(data: mh.data.buffer))
|
|
|
|
|
2021-02-02 11:46:38 +00:00
|
|
|
func init*(t: typedesc[PeerID], seckey: PrivateKey): Result[PeerID, cstring] =
|
2020-12-21 11:45:07 +00:00
|
|
|
## Create new peer id from private key ``seckey``.
|
2021-02-02 11:46:38 +00:00
|
|
|
PeerID.init(? seckey.getKey().orError(cstring("invalid private key")))
|
2020-12-21 11:45:07 +00:00
|
|
|
|
2021-02-02 11:46:38 +00:00
|
|
|
func match*(pid: PeerID, pubkey: PublicKey): bool =
|
2020-12-21 11:45:07 +00:00
|
|
|
## Returns ``true`` if ``pid`` matches public key ``pubkey``.
|
|
|
|
let p = PeerID.init(pubkey)
|
|
|
|
if p.isErr:
|
|
|
|
false
|
|
|
|
else:
|
|
|
|
pid == p.get()
|
|
|
|
|
2021-02-02 11:46:38 +00:00
|
|
|
func match*(pid: PeerID, seckey: PrivateKey): bool =
|
2020-12-21 11:45:07 +00:00
|
|
|
## Returns ``true`` if ``pid`` matches private key ``seckey``.
|
|
|
|
let p = PeerID.init(seckey)
|
|
|
|
if p.isErr:
|
|
|
|
false
|
|
|
|
else:
|
|
|
|
pid == p.get()
|
|
|
|
|
|
|
|
## Serialization/Deserialization helpers
|
|
|
|
|
2021-02-02 11:46:38 +00:00
|
|
|
func write*(vb: var VBuffer, pid: PeerID) =
|
2020-12-21 11:45:07 +00:00
|
|
|
## Write PeerID value ``peerid`` to buffer ``vb``.
|
|
|
|
vb.writeSeq(pid.data)
|
|
|
|
|
2021-02-02 11:46:38 +00:00
|
|
|
func initProtoField*(index: int, pid: PeerID): ProtoField {.deprecated.} =
|
2020-12-21 11:45:07 +00:00
|
|
|
## Initialize ProtoField with PeerID ``value``.
|
2021-02-02 11:46:38 +00:00
|
|
|
initProtoField(index, pid.data)
|
2020-12-21 11:45:07 +00:00
|
|
|
|
2021-02-02 11:46:38 +00:00
|
|
|
func getValue*(data: var ProtoBuffer, field: int, value: var PeerID): int {.
|
2020-12-21 11:45:07 +00:00
|
|
|
deprecated.} =
|
|
|
|
## Read ``PeerID`` from ProtoBuf's message and validate it.
|
|
|
|
var pid: PeerID
|
|
|
|
result = getLengthValue(data, field, pid.data)
|
|
|
|
if result > 0:
|
|
|
|
if not pid.validate():
|
|
|
|
result = -1
|
|
|
|
else:
|
|
|
|
value = pid
|
|
|
|
|
2021-02-02 11:46:38 +00:00
|
|
|
func write*(pb: var ProtoBuffer, field: int, pid: PeerID) =
|
2020-12-21 11:45:07 +00:00
|
|
|
## Write PeerID value ``peerid`` to object ``pb`` using ProtoBuf's encoding.
|
|
|
|
write(pb, field, pid.data)
|
|
|
|
|
2021-02-02 11:46:38 +00:00
|
|
|
func getField*(pb: ProtoBuffer, field: int,
|
2020-12-21 11:45:07 +00:00
|
|
|
pid: var PeerID): ProtoResult[bool] {.inline.} =
|
|
|
|
## Read ``PeerID`` from ProtoBuf's message and validate it
|
|
|
|
var buffer: seq[byte]
|
|
|
|
let res = ? pb.getField(field, buffer)
|
|
|
|
if not(res):
|
|
|
|
ok(false)
|
|
|
|
else:
|
|
|
|
var peerId: PeerID
|
|
|
|
if peerId.init(buffer):
|
|
|
|
pid = peerId
|
|
|
|
ok(true)
|
|
|
|
else:
|
|
|
|
err(ProtoError.IncorrectBlob)
|