Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

chore: Circuit relay #3112

Merged
merged 23 commits into from
Oct 28, 2024
Merged
Show file tree
Hide file tree
Changes from 15 commits
Commits
Show all changes
23 commits
Select commit Hold shift + click to select a range
46721c1
circuit relay first commit
Ivansete-status Oct 10, 2024
034b9dc
undo apt install libpcre
Ivansete-status Oct 15, 2024
7fe1057
nat.nim: protect against possible exceptions when calling getExternalIP
Ivansete-status Oct 16, 2024
2fc9637
progress in circuit-relay addition
Ivansete-status Oct 19, 2024
f415a98
Minor change to adapt the isRelayClient param description
Ivansete-status Oct 19, 2024
7c30a04
circuit relay progress
Ivansete-status Oct 22, 2024
dfe23b4
waku_node: should start waku relay protocol
Ivansete-status Oct 22, 2024
b0b48bd
waku factory change to mount circuit hop proto by default
Ivansete-status Oct 22, 2024
8d2fcd3
fix compilation issue in tests
Ivansete-status Oct 23, 2024
3e2a1d5
fix compilation issue
Ivansete-status Oct 23, 2024
7112afc
update tests to use getPrimaryIPAddr
Ivansete-status Oct 23, 2024
748b083
test_app compilation fix
Ivansete-status Oct 23, 2024
da6bdc1
builer: rm unnecessary line
Ivansete-status Oct 23, 2024
7e8511c
Update tests/wakunode_rest/test_rest_admin.nim
Ivansete-status Oct 24, 2024
16bdcb0
Update waku/discovery/waku_discv5.nim
Ivansete-status Oct 24, 2024
28f317b
avoid default relay in setupNode proc
Ivansete-status Oct 25, 2024
50905f0
remove leftover unneeded code
Ivansete-status Oct 25, 2024
49771f6
fix nph format issue
Ivansete-status Oct 25, 2024
7af2425
missing import in test_node_factory
Ivansete-status Oct 25, 2024
3f784cc
waku_node: better import of autonat_service
Ivansete-status Oct 25, 2024
d69b626
waku: avoid compilation issue
Ivansete-status Oct 25, 2024
16cf26b
waku.nim: protect against possible crash if discv5 is not enabled
Ivansete-status Oct 25, 2024
eae4e8b
apply Zoltan suggestion to simplify tests
Ivansete-status Oct 26, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 0 additions & 3 deletions .github/workflows/ci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -111,9 +111,6 @@ jobs:
run: |
postgres_enabled=0
if [ ${{ runner.os }} == "Linux" ]; then
sudo apt-get update
sudo apt-get install -y libpcre3 libpcre3-dev

sudo docker run --rm -d -e POSTGRES_PASSWORD=test123 -p 5432:5432 postgres:15.4-alpine3.18
postgres_enabled=1
fi
Expand Down
2 changes: 1 addition & 1 deletion apps/liteprotocoltester/liteprotocoltester.nim
Original file line number Diff line number Diff line change
Expand Up @@ -124,7 +124,7 @@ when isMainModule:
error "Starting esential REST server failed.", error = $error
quit(QuitFailure)

var wakuApp = Waku.init(wakuConf).valueOr:
var wakuApp = Waku.new(wakuConf).valueOr:
error "Waku initialization failed", error = error
quit(QuitFailure)

Expand Down
4 changes: 2 additions & 2 deletions apps/wakunode2/wakunode2.nim
Original file line number Diff line number Diff line change
Expand Up @@ -42,7 +42,7 @@ when isMainModule:
error "failure while loading the configuration", error = error
quit(QuitFailure)

## Also called within Waku.init. The call to startRestServerEsentials needs the following line
## Also called within Waku.new. The call to startRestServerEsentials needs the following line
logging.setupLog(conf.logLevel, conf.logFormat)

case conf.cmd
Expand All @@ -66,7 +66,7 @@ when isMainModule:
error "Starting esential REST server failed.", error = $error
quit(QuitFailure)

var waku = Waku.init(confCopy).valueOr:
var waku = Waku.new(confCopy).valueOr:
error "Waku initialization failed", error = error
quit(QuitFailure)

Expand Down
2 changes: 1 addition & 1 deletion examples/wakustealthcommitments/node_spec.nim
Original file line number Diff line number Diff line change
Expand Up @@ -48,7 +48,7 @@ proc setup*(): Waku =
conf.rlnRelay = twnClusterConf.rlnRelay

debug "Starting node"
var waku = Waku.init(conf).valueOr:
var waku = Waku.new(conf).valueOr:
error "Waku initialization failed", error = error
quit(QuitFailure)

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -59,7 +59,7 @@ proc createWaku(configJson: cstring): Future[Result[Waku, string]] {.async.} =
formattedString & ". expected type: " & $typeof(confValue)
)

let wakuRes = Waku.init(conf).valueOr:
let wakuRes = Waku.new(conf).valueOr:
error "waku initialization failed", error = error
return err("Failed setting up Waku: " & $error)

Expand Down
10 changes: 5 additions & 5 deletions tests/test_peer_manager.nim
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
{.used.}

import
std/[options, sequtils, times, sugar],
std/[options, sequtils, times, sugar, net],
stew/shims/net as stewNet,
testutils/unittests,
chronos,
Expand Down Expand Up @@ -270,12 +270,12 @@ procSuite "Peer Manager":
storage = WakuPeerStorage.new(database)[]
node1 = newTestWakuNode(
generateSecp256k1Key(),
ValidIpAddress.init("127.0.0.1"),
ValidIpAddress.init($getPrimaryIPAddr()),
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think getPrimaryIPAddr can be used here directly... no need to convert to string and than from string.

Port(44048),
peerStorage = storage,
)
node2 = newTestWakuNode(
generateSecp256k1Key(), ValidIpAddress.init("127.0.0.1"), Port(34023)
generateSecp256k1Key(), ValidIpAddress.init($getPrimaryIPAddr()), Port(34023)
)

node1.mountMetadata(0).expect("Mounted Waku Metadata")
Expand Down Expand Up @@ -345,12 +345,12 @@ procSuite "Peer Manager":
storage = WakuPeerStorage.new(database)[]
node1 = newTestWakuNode(
generateSecp256k1Key(),
ValidIpAddress.init("127.0.0.1"),
ValidIpAddress.init($getPrimaryIPAddr()),
Port(44048),
peerStorage = storage,
)
node2 = newTestWakuNode(
generateSecp256k1Key(), ValidIpAddress.init("127.0.0.1"), Port(34023)
generateSecp256k1Key(), ValidIpAddress.init($getPrimaryIPAddr()), Port(34023)
)

node1.mountMetadata(0).expect("Mounted Waku Metadata")
Expand Down
4 changes: 2 additions & 2 deletions tests/test_waku_switch.nim
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@ suite "Waku Switch":
## Given
let
sourceSwitch = newTestSwitch()
wakuSwitch = newWakuSwitch(rng = rng())
wakuSwitch = newWakuSwitch(rng = rng(), circuitRelay = Relay.new())
gabrielmer marked this conversation as resolved.
Show resolved Hide resolved
await sourceSwitch.start()
await wakuSwitch.start()

Expand All @@ -46,7 +46,7 @@ suite "Waku Switch":
asyncTest "Waku Switch acts as circuit relayer":
## Setup
let
wakuSwitch = newWakuSwitch(rng = rng())
wakuSwitch = newWakuSwitch(rng = rng(), circuitRelay = Relay.new())
sourceClient = RelayClient.new()
destClient = RelayClient.new()
sourceSwitch = newCircuitRelayClientSwitch(sourceClient)
Expand Down
6 changes: 3 additions & 3 deletions tests/test_wakunode.nim
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
{.used.}

import
std/[sequtils, strutils],
std/[sequtils, strutils, net],
stew/byteutils,
stew/shims/net as stewNet,
testutils/unittests,
Expand Down Expand Up @@ -169,7 +169,7 @@ suite "WakuNode":
nodeKey = generateSecp256k1Key()
bindIp = parseIpAddress("0.0.0.0")
bindPort = Port(61006)
extIp = some(parseIpAddress("127.0.0.1"))
extIp = some(parseIpAddress($getPrimaryIPAddr()))
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Same here, with getPrimaryIPAddr, no need to conver vica-versa, that shall give a valid IP IMHO.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks so much! I've applied your suggestions and much more cleaner now:)

extPort = some(Port(61008))
node = newTestWakuNode(nodeKey, bindIp, bindPort, extIp, extPort)

Expand Down Expand Up @@ -205,7 +205,7 @@ suite "WakuNode":
nodeKey = generateSecp256k1Key()
bindIp = parseIpAddress("0.0.0.0")
bindPort = Port(61010)
extIp = some(parseIpAddress("127.0.0.1"))
extIp = some(parseIpAddress($getPrimaryIPAddr()))
extPort = some(Port(61012))
domainName = "example.com"
expectedDns4Addr =
Expand Down
8 changes: 4 additions & 4 deletions tests/wakunode2/test_app.nim
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,7 @@ suite "Wakunode2 - Waku":
## Given
var conf = defaultTestWakuNodeConf()

let waku = Waku.init(conf).valueOr:
let waku = Waku.new(conf).valueOr:
raiseAssert error

## When
Expand All @@ -35,7 +35,7 @@ suite "Wakunode2 - Waku initialization":
var conf = defaultTestWakuNodeConf()
conf.peerPersistence = true

let waku = Waku.init(conf).valueOr:
let waku = Waku.new(conf).valueOr:
raiseAssert error

check:
Expand All @@ -46,7 +46,7 @@ suite "Wakunode2 - Waku initialization":
var conf = defaultTestWakuNodeConf()

## When
var waku = Waku.init(conf).valueOr:
var waku = Waku.new(conf).valueOr:
raiseAssert error

(waitFor startWaku(addr waku)).isOkOr:
Expand All @@ -73,7 +73,7 @@ suite "Wakunode2 - Waku initialization":
conf.tcpPort = Port(0)

## When
var waku = Waku.init(conf).valueOr:
var waku = Waku.new(conf).valueOr:
raiseAssert error

(waitFor startWaku(addr waku)).isOkOr:
Expand Down
17 changes: 10 additions & 7 deletions tests/wakunode_rest/test_rest_admin.nim
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
{.used.}

import
std/[sequtils, strformat],
std/[sequtils, strformat, net],
stew/shims/net,
testutils/unittests,
presto,
Expand Down Expand Up @@ -38,12 +38,15 @@ suite "Waku v2 Rest API - Admin":
var client {.threadvar.}: RestClientRef

asyncSetup:
node1 =
newTestWakuNode(generateSecp256k1Key(), parseIpAddress("127.0.0.1"), Port(60600))
node2 =
newTestWakuNode(generateSecp256k1Key(), parseIpAddress("127.0.0.1"), Port(60602))
node3 =
newTestWakuNode(generateSecp256k1Key(), parseIpAddress("127.0.0.1"), Port(60604))
node1 = newTestWakuNode(
generateSecp256k1Key(), getPrimaryIPAddr(), Port(60600)
)
node2 = newTestWakuNode(
generateSecp256k1Key(), getPrimaryIPAddr(), Port(60602)
)
node3 = newTestWakuNode(
generateSecp256k1Key(), getPrimaryIPAddr(), Port(60604)
)

await allFutures(node1.start(), node2.start(), node3.start())
await allFutures(
Expand Down
7 changes: 6 additions & 1 deletion waku/common/utils/nat.nim
Original file line number Diff line number Diff line change
Expand Up @@ -39,7 +39,12 @@ proc setupNat*(
warn "NAT already initialized, skipping as cannot be done multiple times"
else:
singletonNat = true
let extIp = getExternalIP(strategy)
var extIp = none(IpAddress)
try:
extIp = getExternalIP(strategy)
except Exception:
warn "exception in setupNat", error = getCurrentExceptionMsg()

if extIP.isSome():
endpoint.ip = some(extIp.get())
# RedirectPorts in considered a gcsafety violation
Expand Down
36 changes: 36 additions & 0 deletions waku/discovery/autonat_service.nim
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
import
chronos,
chronicles,
bearssl/rand,
libp2p/protocols/connectivity/autonat/client,
libp2p/protocols/connectivity/autonat/service,
libp2p/protocols/connectivity/autonat/core

const AutonatCheckInterval = Opt.some(chronos.seconds(30))

proc getAutonatService*(rng: ref HmacDrbgContext): AutonatService =
## AutonatService request other peers to dial us back
## flagging us as Reachable or NotReachable.
## minConfidence is used as threshold to determine the state.
## If maxQueueSize > numPeersToAsk past samples are considered
## in the calculation.
let autonatService = AutonatService.new(
autonatClient = AutonatClient.new(),
rng = rng,
scheduleInterval = AutonatCheckInterval,
askNewConnectedPeers = false,
numPeersToAsk = 3,
maxQueueSize = 3,
minConfidence = 0.7,
)

proc statusAndConfidenceHandler(
networkReachability: NetworkReachability, confidence: Opt[float]
): Future[void] {.async.} =
if confidence.isSome():
info "Peer reachability status",
networkReachability = networkReachability, confidence = confidence.get()

autonatService.statusAndConfidenceHandler(statusAndConfidenceHandler)

return autonatService
20 changes: 18 additions & 2 deletions waku/discovery/waku_discv5.nim
Original file line number Diff line number Diff line change
Expand Up @@ -108,6 +108,19 @@ proc new*(
topicSubscriptionQueue: queue,
)

proc updateAnnouncedMultiAddress*(
wd: WakuDiscoveryV5, addresses: seq[MultiAddress]
): Result[void, string] =
let encodedAddrs = multiaddr.encodeMultiaddrs(addresses)

wd.protocol.updateRecord([(MultiaddrEnrField, encodedAddrs)]).isOkOr:
return err("failed to update multiaddress in ENR: " & $error)

debug "ENR updated successfully with new multiaddress",
enrUri = wd.protocol.localNode.record.toUri(), enr = $(wd.protocol.localNode.record)

return ok()

proc updateENRShards(
wd: WakuDiscoveryV5, newTopics: seq[PubsubTopic], add: bool
): Result[void, string] =
Expand Down Expand Up @@ -286,7 +299,9 @@ proc subscriptionsListener(wd: WakuDiscoveryV5) {.async.} =
if subRes.isErr() and unsubRes.isErr():
continue

debug "ENR updated successfully"
debug "ENR updated successfully",
enrUri = wd.protocol.localNode.record.toUri(),
enr = $(wd.protocol.localNode.record)

wd.predicate =
shardingPredicate(wd.protocol.localNode.record, wd.protocol.bootstrapRecords)
Expand Down Expand Up @@ -314,7 +329,8 @@ proc start*(wd: WakuDiscoveryV5): Future[Result[void, string]] {.async: (raises:
asyncSpawn wd.subscriptionsListener()

debug "Successfully started discovery v5 service"
info "Discv5: discoverable ENR ", enr = wd.protocol.localNode.record.toUri()
info "Discv5: discoverable ENR ",
enrUri = wd.protocol.localNode.record.toUri(), enr = $(wd.protocol.localNode.record)

ok()

Expand Down
16 changes: 14 additions & 2 deletions waku/factory/builder.nim
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,9 @@ import
libp2p/crypto/crypto,
libp2p/builders,
libp2p/nameresolving/nameresolver,
libp2p/transports/wstransport
libp2p/transports/wstransport,
libp2p/protocols/connectivity/relay/client,
libp2p/protocols/connectivity/relay/relay
import
../waku_enr,
../discovery/waku_discv5,
Expand Down Expand Up @@ -38,6 +40,7 @@ type
switchSslSecureKey: Option[string]
switchSslSecureCert: Option[string]
switchSendSignedPeerRecord: Option[bool]
circuitRelay: Relay

#Rate limit configs for non-relay req-resp protocols
rateLimitSettings: Option[seq[string]]
Expand Down Expand Up @@ -116,6 +119,9 @@ proc withColocationLimit*(builder: var WakuNodeBuilder, colocationLimit: int) =
proc withRateLimit*(builder: var WakuNodeBuilder, limits: seq[string]) =
builder.rateLimitSettings = some(limits)

proc withCircuitRelay*(builder: var WakuNodeBuilder, circuitRelay: Relay) =
builder.circuitRelay = circuitRelay

## Waku switch

proc withSwitchConfiguration*(
Expand Down Expand Up @@ -154,6 +160,12 @@ proc build*(builder: WakuNodeBuilder): Result[WakuNode, string] =
if builder.record.isNone():
return err("node record is required")

let circuitRelay =
if builder.circuitRelay.isNil():
Relay.new()
else:
builder.circuitRelay

var switch: Switch
try:
switch = newWakuSwitch(
Expand All @@ -170,7 +182,7 @@ proc build*(builder: WakuNodeBuilder): Result[WakuNode, string] =
sendSignedPeerRecord = builder.switchSendSignedPeerRecord.get(false),
agentString = builder.switchAgentString,
peerStoreCapacity = builder.peerStorageCapacity,
services = @[Service(getAutonatService(rng))],
circuitRelay = circuitRelay,
)
except CatchableError:
return err("failed to create switch: " & getCurrentExceptionMsg())
Expand Down
10 changes: 10 additions & 0 deletions waku/factory/external_config.nim
Original file line number Diff line number Diff line change
Expand Up @@ -245,6 +245,16 @@ type WakuNodeConf* = object
name: "dns4-domain-name"
.}: string

## Circuit-relay config
isRelayClient* {.
desc:
"""Set the node as a relay-client.
Set it to true for nodes that run behind a NAT or firewall and
hence would have reachability issues.""",
defaultValue: false,
name: "relay-client"
.}: bool

## Relay config
relay* {.
desc: "Enable relay protocol: true|false", defaultValue: true, name: "relay"
Expand Down
Loading
Loading