2023-01-06 10:14:38 +00:00
|
|
|
# Nim-LibP2P
|
2023-01-20 14:47:40 +00:00
|
|
|
# Copyright (c) 2023 Status Research & Development GmbH
|
2023-01-06 10:14:38 +00: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.
|
|
|
|
|
|
|
|
when (NimMajor, NimMinor) < (1, 4):
|
|
|
|
{.push raises: [Defect].}
|
|
|
|
else:
|
|
|
|
{.push raises: [].}
|
|
|
|
|
2023-05-18 08:24:17 +00:00
|
|
|
import std/[options]
|
2023-01-06 10:14:38 +00:00
|
|
|
import stew/[results, objects]
|
|
|
|
import chronos, chronicles
|
|
|
|
import ../../../multiaddress,
|
|
|
|
../../../peerid,
|
|
|
|
../../../errors
|
|
|
|
|
|
|
|
logScope:
|
|
|
|
topics = "libp2p autonat"
|
|
|
|
|
|
|
|
const
|
|
|
|
AutonatCodec* = "/libp2p/autonat/1.0.0"
|
|
|
|
AddressLimit* = 8
|
|
|
|
|
|
|
|
type
|
|
|
|
AutonatError* = object of LPError
|
|
|
|
AutonatUnreachableError* = object of LPError
|
|
|
|
|
|
|
|
MsgType* = enum
|
|
|
|
Dial = 0
|
|
|
|
DialResponse = 1
|
|
|
|
|
|
|
|
ResponseStatus* = enum
|
|
|
|
Ok = 0
|
|
|
|
DialError = 100
|
|
|
|
DialRefused = 101
|
|
|
|
BadRequest = 200
|
|
|
|
InternalError = 300
|
|
|
|
|
|
|
|
AutonatPeerInfo* = object
|
|
|
|
id*: Option[PeerId]
|
|
|
|
addrs*: seq[MultiAddress]
|
|
|
|
|
|
|
|
AutonatDial* = object
|
|
|
|
peerInfo*: Option[AutonatPeerInfo]
|
|
|
|
|
|
|
|
AutonatDialResponse* = object
|
|
|
|
status*: ResponseStatus
|
|
|
|
text*: Option[string]
|
|
|
|
ma*: Option[MultiAddress]
|
|
|
|
|
|
|
|
AutonatMsg* = object
|
|
|
|
msgType*: MsgType
|
|
|
|
dial*: Option[AutonatDial]
|
|
|
|
response*: Option[AutonatDialResponse]
|
|
|
|
|
2023-04-06 13:23:35 +00:00
|
|
|
NetworkReachability* {.pure.} = enum
|
|
|
|
Unknown, NotReachable, Reachable
|
|
|
|
|
2023-01-06 10:14:38 +00:00
|
|
|
proc encode(p: AutonatPeerInfo): ProtoBuffer =
|
|
|
|
result = initProtoBuffer()
|
|
|
|
if p.id.isSome():
|
|
|
|
result.write(1, p.id.get())
|
|
|
|
for ma in p.addrs:
|
|
|
|
result.write(2, ma.data.buffer)
|
|
|
|
result.finish()
|
|
|
|
|
|
|
|
proc encode*(d: AutonatDial): ProtoBuffer =
|
|
|
|
result = initProtoBuffer()
|
|
|
|
result.write(1, MsgType.Dial.uint)
|
|
|
|
var dial = initProtoBuffer()
|
|
|
|
if d.peerInfo.isSome():
|
|
|
|
dial.write(1, encode(d.peerInfo.get()))
|
|
|
|
dial.finish()
|
|
|
|
result.write(2, dial.buffer)
|
|
|
|
result.finish()
|
|
|
|
|
|
|
|
proc encode*(r: AutonatDialResponse): ProtoBuffer =
|
|
|
|
result = initProtoBuffer()
|
|
|
|
result.write(1, MsgType.DialResponse.uint)
|
|
|
|
var bufferResponse = initProtoBuffer()
|
|
|
|
bufferResponse.write(1, r.status.uint)
|
|
|
|
if r.text.isSome():
|
|
|
|
bufferResponse.write(2, r.text.get())
|
|
|
|
if r.ma.isSome():
|
|
|
|
bufferResponse.write(3, r.ma.get())
|
|
|
|
bufferResponse.finish()
|
|
|
|
result.write(3, bufferResponse.buffer)
|
|
|
|
result.finish()
|
|
|
|
|
|
|
|
proc encode*(msg: AutonatMsg): ProtoBuffer =
|
|
|
|
if msg.dial.isSome():
|
|
|
|
return encode(msg.dial.get())
|
|
|
|
if msg.response.isSome():
|
|
|
|
return encode(msg.response.get())
|
|
|
|
|
|
|
|
proc decode*(_: typedesc[AutonatMsg], buf: seq[byte]): Option[AutonatMsg] =
|
|
|
|
var
|
|
|
|
msgTypeOrd: uint32
|
|
|
|
pbDial: ProtoBuffer
|
|
|
|
pbResponse: ProtoBuffer
|
|
|
|
msg: AutonatMsg
|
|
|
|
|
|
|
|
let
|
|
|
|
pb = initProtoBuffer(buf)
|
|
|
|
r1 = pb.getField(1, msgTypeOrd)
|
|
|
|
r2 = pb.getField(2, pbDial)
|
|
|
|
r3 = pb.getField(3, pbResponse)
|
|
|
|
if r1.isErr() or r2.isErr() or r3.isErr(): return none(AutonatMsg)
|
|
|
|
|
|
|
|
if r1.get() and not checkedEnumAssign(msg.msgType, msgTypeOrd):
|
|
|
|
return none(AutonatMsg)
|
|
|
|
if r2.get():
|
|
|
|
var
|
|
|
|
pbPeerInfo: ProtoBuffer
|
|
|
|
dial: AutonatDial
|
|
|
|
let
|
|
|
|
r4 = pbDial.getField(1, pbPeerInfo)
|
|
|
|
if r4.isErr(): return none(AutonatMsg)
|
|
|
|
|
|
|
|
var peerInfo: AutonatPeerInfo
|
|
|
|
if r4.get():
|
|
|
|
var pid: PeerId
|
|
|
|
let
|
|
|
|
r5 = pbPeerInfo.getField(1, pid)
|
|
|
|
r6 = pbPeerInfo.getRepeatedField(2, peerInfo.addrs)
|
|
|
|
if r5.isErr() or r6.isErr(): return none(AutonatMsg)
|
|
|
|
if r5.get(): peerInfo.id = some(pid)
|
|
|
|
dial.peerInfo = some(peerInfo)
|
|
|
|
msg.dial = some(dial)
|
|
|
|
|
|
|
|
if r3.get():
|
|
|
|
var
|
|
|
|
statusOrd: uint
|
|
|
|
text: string
|
|
|
|
ma: MultiAddress
|
|
|
|
response: AutonatDialResponse
|
|
|
|
|
|
|
|
let
|
|
|
|
r4 = pbResponse.getField(1, statusOrd)
|
|
|
|
r5 = pbResponse.getField(2, text)
|
|
|
|
r6 = pbResponse.getField(3, ma)
|
|
|
|
|
|
|
|
if r4.isErr() or r5.isErr() or r6.isErr() or
|
|
|
|
(r4.get() and not checkedEnumAssign(response.status, statusOrd)):
|
|
|
|
return none(AutonatMsg)
|
|
|
|
if r5.get(): response.text = some(text)
|
|
|
|
if r6.get(): response.ma = some(ma)
|
|
|
|
msg.response = some(response)
|
|
|
|
|
|
|
|
return some(msg)
|