2022-05-19 02:29:15 +00:00
|
|
|
import std/sequtils
|
|
|
|
import std/sugar
|
|
|
|
import std/tables
|
|
|
|
|
|
|
|
import pkg/chronos
|
|
|
|
|
2022-05-19 19:56:03 +00:00
|
|
|
import pkg/codex/rng
|
|
|
|
import pkg/codex/stores
|
|
|
|
import pkg/codex/blockexchange
|
|
|
|
import pkg/codex/chunker
|
|
|
|
import pkg/codex/blocktype as bt
|
|
|
|
import pkg/codex/blockexchange/engine
|
2022-05-19 02:29:15 +00:00
|
|
|
|
2024-01-29 20:03:51 +00:00
|
|
|
import ../../../asynctest
|
2022-05-19 02:29:15 +00:00
|
|
|
import ../../helpers
|
2024-01-29 20:03:51 +00:00
|
|
|
import ../../helpers/mockdiscovery
|
2022-05-19 02:29:15 +00:00
|
|
|
import ../../examples
|
|
|
|
|
2023-06-22 18:01:21 +00:00
|
|
|
asyncchecksuite "Test Discovery Engine":
|
2022-05-19 02:29:15 +00:00
|
|
|
let chunker = RandomChunker.new(Rng.instance(), size = 4096, chunkSize = 256)
|
|
|
|
|
|
|
|
var
|
|
|
|
blocks: seq[bt.Block]
|
|
|
|
switch: Switch
|
|
|
|
peerStore: PeerCtxStore
|
|
|
|
blockDiscovery: MockDiscovery
|
|
|
|
pendingBlocks: PendingBlocksManager
|
|
|
|
network: BlockExcNetwork
|
|
|
|
|
|
|
|
setup:
|
|
|
|
while true:
|
|
|
|
let chunk = await chunker.getBytes()
|
|
|
|
if chunk.len <= 0:
|
|
|
|
break
|
|
|
|
|
|
|
|
blocks.add(bt.Block.new(chunk).tryGet())
|
|
|
|
|
|
|
|
switch = newStandardSwitch(transportFlags = {ServerFlags.ReuseAddr})
|
|
|
|
network = BlockExcNetwork.new(switch)
|
|
|
|
peerStore = PeerCtxStore.new()
|
|
|
|
pendingBlocks = PendingBlocksManager.new()
|
|
|
|
blockDiscovery = MockDiscovery.new()
|
|
|
|
|
|
|
|
test "Should Query Wants":
|
|
|
|
var
|
|
|
|
localStore = CacheStore.new()
|
|
|
|
discoveryEngine = DiscoveryEngine.new(
|
|
|
|
localStore,
|
|
|
|
peerStore,
|
|
|
|
network,
|
|
|
|
blockDiscovery,
|
|
|
|
pendingBlocks,
|
|
|
|
discoveryLoopSleep = 100.millis)
|
|
|
|
wants = blocks.mapIt( pendingBlocks.getWantHandle(it.cid) )
|
|
|
|
|
|
|
|
blockDiscovery.findBlockProvidersHandler =
|
|
|
|
proc(d: MockDiscovery, cid: Cid): Future[seq[SignedPeerRecord]] {.async, gcsafe.} =
|
2023-11-14 12:02:17 +00:00
|
|
|
pendingBlocks.resolve(blocks.filterIt( it.cid == cid).mapIt(BlockDelivery(blk: it, address: it.address)))
|
2022-05-19 02:29:15 +00:00
|
|
|
|
|
|
|
await discoveryEngine.start()
|
|
|
|
await allFuturesThrowing(allFinished(wants)).wait(1.seconds)
|
|
|
|
await discoveryEngine.stop()
|
|
|
|
|
|
|
|
test "Should Advertise Haves":
|
|
|
|
var
|
|
|
|
localStore = CacheStore.new(blocks.mapIt( it ))
|
|
|
|
discoveryEngine = DiscoveryEngine.new(
|
|
|
|
localStore,
|
|
|
|
peerStore,
|
|
|
|
network,
|
|
|
|
blockDiscovery,
|
|
|
|
pendingBlocks,
|
|
|
|
discoveryLoopSleep = 100.millis)
|
|
|
|
haves = collect(initTable):
|
|
|
|
for b in blocks:
|
|
|
|
{ b.cid: newFuture[void]() }
|
|
|
|
|
2022-05-26 02:29:31 +00:00
|
|
|
blockDiscovery.publishBlockProvideHandler =
|
2022-05-19 02:29:15 +00:00
|
|
|
proc(d: MockDiscovery, cid: Cid) {.async, gcsafe.} =
|
|
|
|
if not haves[cid].finished:
|
|
|
|
haves[cid].complete
|
|
|
|
|
|
|
|
await discoveryEngine.start()
|
|
|
|
await allFuturesThrowing(
|
2022-11-15 00:01:05 +00:00
|
|
|
allFinished(toSeq(haves.values))).wait(5.seconds)
|
2022-05-19 02:29:15 +00:00
|
|
|
await discoveryEngine.stop()
|
|
|
|
|
|
|
|
test "Should queue discovery request":
|
|
|
|
var
|
|
|
|
localStore = CacheStore.new()
|
|
|
|
discoveryEngine = DiscoveryEngine.new(
|
|
|
|
localStore,
|
|
|
|
peerStore,
|
|
|
|
network,
|
|
|
|
blockDiscovery,
|
|
|
|
pendingBlocks,
|
|
|
|
discoveryLoopSleep = 100.millis)
|
|
|
|
want = newFuture[void]()
|
|
|
|
|
|
|
|
blockDiscovery.findBlockProvidersHandler =
|
|
|
|
proc(d: MockDiscovery, cid: Cid): Future[seq[SignedPeerRecord]] {.async, gcsafe.} =
|
|
|
|
check cid == blocks[0].cid
|
|
|
|
if not want.finished:
|
|
|
|
want.complete()
|
|
|
|
|
|
|
|
await discoveryEngine.start()
|
|
|
|
discoveryEngine.queueFindBlocksReq(@[blocks[0].cid])
|
|
|
|
await want.wait(1.seconds)
|
|
|
|
await discoveryEngine.stop()
|
|
|
|
|
|
|
|
test "Should queue advertise request":
|
|
|
|
var
|
|
|
|
localStore = CacheStore.new(@[blocks[0]])
|
|
|
|
discoveryEngine = DiscoveryEngine.new(
|
|
|
|
localStore,
|
|
|
|
peerStore,
|
|
|
|
network,
|
|
|
|
blockDiscovery,
|
|
|
|
pendingBlocks,
|
|
|
|
discoveryLoopSleep = 100.millis)
|
|
|
|
have = newFuture[void]()
|
|
|
|
|
2022-05-26 02:29:31 +00:00
|
|
|
blockDiscovery.publishBlockProvideHandler =
|
2022-05-19 02:29:15 +00:00
|
|
|
proc(d: MockDiscovery, cid: Cid) {.async, gcsafe.} =
|
|
|
|
check cid == blocks[0].cid
|
|
|
|
if not have.finished:
|
|
|
|
have.complete()
|
|
|
|
|
|
|
|
await discoveryEngine.start()
|
|
|
|
await have.wait(1.seconds)
|
|
|
|
await discoveryEngine.stop()
|
|
|
|
|
|
|
|
test "Should not request more than minPeersPerBlock":
|
|
|
|
var
|
|
|
|
localStore = CacheStore.new()
|
|
|
|
minPeers = 2
|
|
|
|
discoveryEngine = DiscoveryEngine.new(
|
|
|
|
localStore,
|
|
|
|
peerStore,
|
|
|
|
network,
|
|
|
|
blockDiscovery,
|
|
|
|
pendingBlocks,
|
|
|
|
discoveryLoopSleep = 5.minutes,
|
|
|
|
minPeersPerBlock = minPeers)
|
|
|
|
want = newAsyncEvent()
|
|
|
|
|
|
|
|
blockDiscovery.findBlockProvidersHandler =
|
|
|
|
proc(d: MockDiscovery, cid: Cid): Future[seq[SignedPeerRecord]] {.async, gcsafe.} =
|
2022-11-15 15:46:21 +00:00
|
|
|
|
2022-05-19 02:29:15 +00:00
|
|
|
check cid == blocks[0].cid
|
|
|
|
check peerStore.len < minPeers
|
|
|
|
var
|
2023-03-10 07:02:54 +00:00
|
|
|
peerCtx = BlockExcPeerCtx(id: PeerId.example)
|
2022-05-19 02:29:15 +00:00
|
|
|
|
2023-11-14 12:02:17 +00:00
|
|
|
let address = BlockAddress(leaf: false, cid: cid)
|
|
|
|
|
|
|
|
peerCtx.blocks[address] = Presence(address: address, price: 0.u256)
|
2022-05-19 02:29:15 +00:00
|
|
|
peerStore.add(peerCtx)
|
|
|
|
want.fire()
|
|
|
|
|
|
|
|
await discoveryEngine.start()
|
|
|
|
while peerStore.len < minPeers:
|
|
|
|
discoveryEngine.queueFindBlocksReq(@[blocks[0].cid])
|
|
|
|
await want.wait()
|
|
|
|
want.clear()
|
|
|
|
|
|
|
|
check peerStore.len == minPeers
|
|
|
|
await discoveryEngine.stop()
|
|
|
|
|
|
|
|
test "Should not request if there is already an inflight discovery request":
|
|
|
|
var
|
|
|
|
localStore = CacheStore.new()
|
|
|
|
discoveryEngine = DiscoveryEngine.new(
|
|
|
|
localStore,
|
|
|
|
peerStore,
|
|
|
|
network,
|
|
|
|
blockDiscovery,
|
|
|
|
pendingBlocks,
|
|
|
|
discoveryLoopSleep = 100.millis,
|
|
|
|
concurrentDiscReqs = 2)
|
|
|
|
reqs = newFuture[void]()
|
|
|
|
count = 0
|
|
|
|
|
|
|
|
blockDiscovery.findBlockProvidersHandler =
|
|
|
|
proc(d: MockDiscovery, cid: Cid):
|
|
|
|
Future[seq[SignedPeerRecord]] {.gcsafe, async.} =
|
|
|
|
check cid == blocks[0].cid
|
|
|
|
if count > 0:
|
|
|
|
check false
|
|
|
|
count.inc
|
|
|
|
|
|
|
|
await reqs # queue the request
|
|
|
|
|
|
|
|
await discoveryEngine.start()
|
|
|
|
discoveryEngine.queueFindBlocksReq(@[blocks[0].cid])
|
|
|
|
await sleepAsync(200.millis)
|
|
|
|
|
|
|
|
discoveryEngine.queueFindBlocksReq(@[blocks[0].cid])
|
|
|
|
await sleepAsync(200.millis)
|
|
|
|
|
|
|
|
reqs.complete()
|
|
|
|
await discoveryEngine.stop()
|
|
|
|
|
|
|
|
test "Should not request if there is already an inflight advertise request":
|
|
|
|
var
|
|
|
|
localStore = CacheStore.new()
|
|
|
|
discoveryEngine = DiscoveryEngine.new(
|
|
|
|
localStore,
|
|
|
|
peerStore,
|
|
|
|
network,
|
|
|
|
blockDiscovery,
|
|
|
|
pendingBlocks,
|
|
|
|
discoveryLoopSleep = 100.millis,
|
|
|
|
concurrentAdvReqs = 2)
|
|
|
|
reqs = newFuture[void]()
|
|
|
|
count = 0
|
|
|
|
|
2022-05-26 02:29:31 +00:00
|
|
|
blockDiscovery.publishBlockProvideHandler =
|
2022-05-19 02:29:15 +00:00
|
|
|
proc(d: MockDiscovery, cid: Cid) {.async, gcsafe.} =
|
|
|
|
check cid == blocks[0].cid
|
|
|
|
if count > 0:
|
|
|
|
check false
|
|
|
|
count.inc
|
|
|
|
|
|
|
|
await reqs # queue the request
|
|
|
|
|
|
|
|
await discoveryEngine.start()
|
|
|
|
discoveryEngine.queueProvideBlocksReq(@[blocks[0].cid])
|
|
|
|
await sleepAsync(200.millis)
|
|
|
|
|
|
|
|
discoveryEngine.queueProvideBlocksReq(@[blocks[0].cid])
|
|
|
|
await sleepAsync(200.millis)
|
|
|
|
|
|
|
|
reqs.complete()
|
|
|
|
await discoveryEngine.stop()
|