2022-12-16 11:32:00 +00:00
|
|
|
# Nim-LibP2P
|
|
|
|
# Copyright (c) 2022 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/options
|
|
|
|
import chronos, metrics
|
|
|
|
import unittest2
|
|
|
|
import ../libp2p/[builders,
|
|
|
|
switch,
|
2022-12-23 19:10:15 +00:00
|
|
|
services/autonatservice,
|
|
|
|
protocols/connectivity/autonat]
|
2022-12-16 11:32:00 +00:00
|
|
|
import ./helpers
|
|
|
|
import stubs/autonatstub
|
|
|
|
|
2022-12-23 19:10:15 +00:00
|
|
|
proc createSwitch(autonatSvc: Service = nil, withAutonat = true): Switch =
|
2022-12-16 11:32:00 +00:00
|
|
|
var builder = SwitchBuilder.new()
|
|
|
|
.withRng(newRng())
|
|
|
|
.withAddresses(@[ MultiAddress.init("/ip4/0.0.0.0/tcp/0").tryGet() ])
|
|
|
|
.withTcpTransport()
|
|
|
|
.withMplex()
|
|
|
|
.withNoise()
|
|
|
|
|
2022-12-23 19:10:15 +00:00
|
|
|
if withAutonat:
|
|
|
|
builder = builder.withAutonat()
|
|
|
|
|
2022-12-16 11:32:00 +00:00
|
|
|
if autonatSvc != nil:
|
|
|
|
builder = builder.withServices(@[autonatSvc])
|
|
|
|
|
|
|
|
return builder.build()
|
|
|
|
|
|
|
|
suite "Autonat Service":
|
|
|
|
teardown:
|
|
|
|
checkTrackers()
|
|
|
|
|
2022-12-23 15:49:25 +00:00
|
|
|
asyncTest "Peer must be not reachable":
|
2022-12-16 11:32:00 +00:00
|
|
|
|
|
|
|
let autonatStub = AutonatStub.new(expectedDials = 3)
|
2022-12-23 15:49:25 +00:00
|
|
|
autonatStub.answer = NotReachable
|
2022-12-16 11:32:00 +00:00
|
|
|
|
|
|
|
let autonatService = AutonatService.new(autonatStub, newRng())
|
|
|
|
|
|
|
|
let switch1 = createSwitch(autonatService)
|
|
|
|
let switch2 = createSwitch()
|
|
|
|
let switch3 = createSwitch()
|
|
|
|
let switch4 = createSwitch()
|
|
|
|
|
|
|
|
check autonatService.networkReachability() == NetworkReachability.Unknown
|
|
|
|
|
|
|
|
await switch1.start()
|
|
|
|
await switch2.start()
|
|
|
|
await switch3.start()
|
|
|
|
await switch4.start()
|
|
|
|
|
|
|
|
await switch1.connect(switch2.peerInfo.peerId, switch2.peerInfo.addrs)
|
|
|
|
await switch1.connect(switch3.peerInfo.peerId, switch3.peerInfo.addrs)
|
|
|
|
await switch1.connect(switch4.peerInfo.peerId, switch4.peerInfo.addrs)
|
|
|
|
|
|
|
|
await autonatStub.finished
|
|
|
|
|
|
|
|
check autonatService.networkReachability() == NetworkReachability.NotReachable
|
|
|
|
check libp2p_autonat_reachability_confidence.value(["NotReachable"]) == 0.3
|
|
|
|
|
|
|
|
await allFuturesThrowing(
|
|
|
|
switch1.stop(), switch2.stop(), switch3.stop(), switch4.stop())
|
|
|
|
|
2022-12-23 15:49:25 +00:00
|
|
|
asyncTest "Peer must be reachable":
|
2022-12-16 11:32:00 +00:00
|
|
|
|
2022-12-23 19:10:15 +00:00
|
|
|
let autonat = Autonat.new(switch = nil)
|
2022-12-16 11:32:00 +00:00
|
|
|
|
2022-12-23 19:10:15 +00:00
|
|
|
let autonatService = AutonatService.new(autonat, newRng(), some(1.seconds))
|
2022-12-16 11:32:00 +00:00
|
|
|
|
|
|
|
let switch1 = createSwitch(autonatService)
|
|
|
|
let switch2 = createSwitch()
|
|
|
|
let switch3 = createSwitch()
|
|
|
|
let switch4 = createSwitch()
|
|
|
|
|
2022-12-23 19:10:15 +00:00
|
|
|
let awaiter = newFuture[void]()
|
|
|
|
|
|
|
|
proc statusAndConfidenceHandler(networkReachability: NetworkReachability, confidence: Option[float]) {.gcsafe, async.} =
|
|
|
|
if networkReachability == NetworkReachability.Reachable and confidence.isSome() and confidence.get() >= 0.3:
|
|
|
|
if not awaiter.finished:
|
|
|
|
awaiter.complete()
|
|
|
|
|
|
|
|
autonat.switch = switch1
|
2022-12-16 11:32:00 +00:00
|
|
|
check autonatService.networkReachability() == NetworkReachability.Unknown
|
|
|
|
|
2022-12-23 19:10:15 +00:00
|
|
|
autonatService.statusAndConfidenceHandler(statusAndConfidenceHandler)
|
|
|
|
|
2022-12-16 11:32:00 +00:00
|
|
|
await switch1.start()
|
|
|
|
await switch2.start()
|
|
|
|
await switch3.start()
|
|
|
|
await switch4.start()
|
|
|
|
|
|
|
|
await switch1.connect(switch2.peerInfo.peerId, switch2.peerInfo.addrs)
|
|
|
|
await switch1.connect(switch3.peerInfo.peerId, switch3.peerInfo.addrs)
|
|
|
|
await switch1.connect(switch4.peerInfo.peerId, switch4.peerInfo.addrs)
|
|
|
|
|
2022-12-23 19:10:15 +00:00
|
|
|
await awaiter
|
2022-12-16 11:32:00 +00:00
|
|
|
|
|
|
|
check autonatService.networkReachability() == NetworkReachability.Reachable
|
|
|
|
check libp2p_autonat_reachability_confidence.value(["Reachable"]) == 0.3
|
|
|
|
|
|
|
|
await allFuturesThrowing(
|
|
|
|
switch1.stop(), switch2.stop(), switch3.stop(), switch4.stop())
|
|
|
|
|
2022-12-23 15:49:25 +00:00
|
|
|
asyncTest "Peer must be not reachable and then reachable":
|
2022-12-16 11:32:00 +00:00
|
|
|
|
|
|
|
let autonatStub = AutonatStub.new(expectedDials = 6)
|
2022-12-23 15:49:25 +00:00
|
|
|
autonatStub.answer = NotReachable
|
2022-12-16 11:32:00 +00:00
|
|
|
|
|
|
|
let autonatService = AutonatService.new(autonatStub, newRng(), some(1.seconds))
|
|
|
|
|
|
|
|
let switch1 = createSwitch(autonatService)
|
|
|
|
let switch2 = createSwitch()
|
|
|
|
let switch3 = createSwitch()
|
|
|
|
let switch4 = createSwitch()
|
|
|
|
|
|
|
|
let awaiter = newFuture[void]()
|
|
|
|
|
|
|
|
proc statusAndConfidenceHandler(networkReachability: NetworkReachability, confidence: Option[float]) {.gcsafe, async.} =
|
|
|
|
if networkReachability == NetworkReachability.NotReachable and confidence.isSome() and confidence.get() >= 0.3:
|
|
|
|
if not awaiter.finished:
|
2022-12-23 15:49:25 +00:00
|
|
|
autonatStub.answer = Reachable
|
2022-12-16 11:32:00 +00:00
|
|
|
awaiter.complete()
|
|
|
|
|
|
|
|
check autonatService.networkReachability() == NetworkReachability.Unknown
|
|
|
|
|
|
|
|
autonatService.statusAndConfidenceHandler(statusAndConfidenceHandler)
|
|
|
|
|
|
|
|
await switch1.start()
|
|
|
|
await switch2.start()
|
|
|
|
await switch3.start()
|
|
|
|
await switch4.start()
|
|
|
|
|
|
|
|
await switch1.connect(switch2.peerInfo.peerId, switch2.peerInfo.addrs)
|
|
|
|
await switch1.connect(switch3.peerInfo.peerId, switch3.peerInfo.addrs)
|
|
|
|
await switch1.connect(switch4.peerInfo.peerId, switch4.peerInfo.addrs)
|
|
|
|
|
|
|
|
await awaiter
|
|
|
|
|
|
|
|
check autonatService.networkReachability() == NetworkReachability.NotReachable
|
|
|
|
check libp2p_autonat_reachability_confidence.value(["NotReachable"]) == 0.3
|
|
|
|
|
|
|
|
await autonatStub.finished
|
|
|
|
|
|
|
|
check autonatService.networkReachability() == NetworkReachability.Reachable
|
|
|
|
check libp2p_autonat_reachability_confidence.value(["Reachable"]) == 0.3
|
|
|
|
|
|
|
|
await allFuturesThrowing(switch1.stop(), switch2.stop(), switch3.stop(), switch4.stop())
|
|
|
|
|
2022-12-23 19:10:15 +00:00
|
|
|
asyncTest "Peer must be reachable when one connected peer has autonat disabled":
|
|
|
|
let autonat = Autonat.new(switch = nil)
|
|
|
|
|
|
|
|
let autonatService = AutonatService.new(autonat, newRng(), some(1.seconds), maxQueueSize = 2)
|
|
|
|
|
|
|
|
let switch1 = createSwitch(autonatService)
|
|
|
|
let switch2 = createSwitch(withAutonat = false)
|
|
|
|
let switch3 = createSwitch()
|
|
|
|
let switch4 = createSwitch()
|
|
|
|
|
|
|
|
let awaiter = newFuture[void]()
|
|
|
|
|
|
|
|
proc statusAndConfidenceHandler(networkReachability: NetworkReachability, confidence: Option[float]) {.gcsafe, async.} =
|
|
|
|
if networkReachability == NetworkReachability.Reachable and confidence.isSome() and confidence.get() == 1:
|
|
|
|
if not awaiter.finished:
|
|
|
|
awaiter.complete()
|
|
|
|
|
|
|
|
autonat.switch = switch1
|
|
|
|
check autonatService.networkReachability() == NetworkReachability.Unknown
|
|
|
|
|
|
|
|
autonatService.statusAndConfidenceHandler(statusAndConfidenceHandler)
|
|
|
|
|
|
|
|
await switch1.start()
|
|
|
|
await switch2.start()
|
|
|
|
await switch3.start()
|
|
|
|
await switch4.start()
|
|
|
|
|
|
|
|
await switch1.connect(switch2.peerInfo.peerId, switch2.peerInfo.addrs)
|
|
|
|
await switch1.connect(switch3.peerInfo.peerId, switch3.peerInfo.addrs)
|
|
|
|
await switch1.connect(switch4.peerInfo.peerId, switch4.peerInfo.addrs)
|
|
|
|
|
|
|
|
await awaiter
|
|
|
|
|
|
|
|
check autonatService.networkReachability() == NetworkReachability.Reachable
|
|
|
|
check libp2p_autonat_reachability_confidence.value(["Reachable"]) == 1
|
|
|
|
|
|
|
|
await allFuturesThrowing(
|
|
|
|
switch1.stop(), switch2.stop(), switch3.stop(), switch4.stop())
|
|
|
|
|
2022-12-23 15:49:25 +00:00
|
|
|
asyncTest "Unknown answers must be ignored":
|
2022-12-16 11:32:00 +00:00
|
|
|
|
2022-12-23 15:49:25 +00:00
|
|
|
let autonatStub = AutonatStub.new(expectedDials = 6)
|
|
|
|
autonatStub.answer = NotReachable
|
|
|
|
|
|
|
|
let autonatService = AutonatService.new(autonatStub, newRng(), some(1.seconds), maxQueueSize = 3)
|
|
|
|
|
|
|
|
let switch1 = createSwitch(autonatService)
|
|
|
|
let switch2 = createSwitch()
|
|
|
|
let switch3 = createSwitch()
|
|
|
|
let switch4 = createSwitch()
|
|
|
|
|
|
|
|
let awaiter = newFuture[void]()
|
|
|
|
|
|
|
|
proc statusAndConfidenceHandler(networkReachability: NetworkReachability, confidence: Option[float]) {.gcsafe, async.} =
|
|
|
|
if networkReachability == NetworkReachability.NotReachable and confidence.isSome() and confidence.get() >= 0.3:
|
|
|
|
if not awaiter.finished:
|
|
|
|
autonatStub.answer = Unknown
|
|
|
|
awaiter.complete()
|
|
|
|
|
|
|
|
check autonatService.networkReachability() == NetworkReachability.Unknown
|
|
|
|
|
|
|
|
autonatService.statusAndConfidenceHandler(statusAndConfidenceHandler)
|
|
|
|
|
|
|
|
await switch1.start()
|
|
|
|
await switch2.start()
|
|
|
|
await switch3.start()
|
|
|
|
await switch4.start()
|
|
|
|
|
|
|
|
await switch1.connect(switch2.peerInfo.peerId, switch2.peerInfo.addrs)
|
|
|
|
await switch1.connect(switch3.peerInfo.peerId, switch3.peerInfo.addrs)
|
|
|
|
await switch1.connect(switch4.peerInfo.peerId, switch4.peerInfo.addrs)
|
|
|
|
|
|
|
|
await awaiter
|
|
|
|
|
|
|
|
check autonatService.networkReachability() == NetworkReachability.NotReachable
|
|
|
|
check libp2p_autonat_reachability_confidence.value(["NotReachable"]) == 1/3
|
|
|
|
|
|
|
|
await autonatStub.finished
|
|
|
|
|
|
|
|
check autonatService.networkReachability() == NetworkReachability.NotReachable
|
|
|
|
check libp2p_autonat_reachability_confidence.value(["NotReachable"]) == 1/3
|
|
|
|
|
|
|
|
await allFuturesThrowing(switch1.stop(), switch2.stop(), switch3.stop(), switch4.stop())
|
|
|
|
|
|
|
|
asyncTest "Calling setup and stop twice must work":
|
|
|
|
|
|
|
|
let switch = createSwitch()
|
|
|
|
let autonatService = AutonatService.new(AutonatStub.new(expectedDials = 0), newRng(), some(1.seconds))
|
2022-12-16 11:32:00 +00:00
|
|
|
|
2022-12-23 15:49:25 +00:00
|
|
|
check (await autonatService.setup(switch)) == true
|
|
|
|
check (await autonatService.setup(switch)) == false
|
2022-12-16 11:32:00 +00:00
|
|
|
|
2022-12-23 15:49:25 +00:00
|
|
|
check (await autonatService.stop(switch)) == true
|
|
|
|
check (await autonatService.stop(switch)) == false
|
2022-12-16 11:32:00 +00:00
|
|
|
|
2022-12-23 15:49:25 +00:00
|
|
|
await allFuturesThrowing(switch.stop())
|