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
|
2024-05-09 08:03:35 +00:00
|
|
|
import pkg/codex/manifest
|
|
|
|
import pkg/codex/merkletree
|
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
|
|
|
|
|
2024-05-09 08:03:35 +00:00
|
|
|
proc asBlock(m: Manifest): bt.Block =
|
|
|
|
let mdata = m.encode().tryGet()
|
|
|
|
bt.Block.new(data = mdata, codec = ManifestCodec).tryGet()
|
|
|
|
|
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]
|
2024-05-09 08:03:35 +00:00
|
|
|
manifest: Manifest
|
|
|
|
tree: CodexTree
|
|
|
|
manifestBlock: bt.Block
|
2022-05-19 02:29:15 +00:00
|
|
|
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())
|
|
|
|
|
2024-05-09 08:03:35 +00:00
|
|
|
(manifest, tree) = makeManifestAndTree(blocks).tryGet()
|
|
|
|
manifestBlock = manifest.asBlock()
|
|
|
|
blocks.add(manifestBlock)
|
|
|
|
|
2022-05-19 02:29:15 +00:00
|
|
|
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)
|
2024-05-09 08:03:35 +00:00
|
|
|
wants = blocks.mapIt(pendingBlocks.getWantHandle(it.cid) )
|
2022-05-19 02:29:15 +00:00
|
|
|
|
|
|
|
blockDiscovery.findBlockProvidersHandler =
|
|
|
|
proc(d: MockDiscovery, cid: Cid): Future[seq[SignedPeerRecord]] {.async, gcsafe.} =
|
2024-05-09 08:03:35 +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 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 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()
|
|
|
|
|
2024-11-27 07:13:34 +00:00
|
|
|
var pendingCids = newSeq[Cid]()
|
2022-05-19 02:29:15 +00:00
|
|
|
blockDiscovery.findBlockProvidersHandler =
|
|
|
|
proc(d: MockDiscovery, cid: Cid): Future[seq[SignedPeerRecord]] {.async, gcsafe.} =
|
2024-11-27 07:13:34 +00:00
|
|
|
check cid in pendingCids
|
|
|
|
pendingCids.keepItIf(it != cid)
|
2022-05-19 02:29:15 +00:00
|
|
|
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()
|
2024-11-27 07:13:34 +00:00
|
|
|
var idx = 0
|
2022-05-19 02:29:15 +00:00
|
|
|
while peerStore.len < minPeers:
|
2024-11-27 07:13:34 +00:00
|
|
|
let cid = blocks[idx].cid
|
|
|
|
inc idx
|
|
|
|
pendingCids.add(cid)
|
|
|
|
discoveryEngine.queueFindBlocksReq(@[cid])
|
2022-05-19 02:29:15 +00:00
|
|
|
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()
|