nim-codex/tests/codex/testsales.nim

161 lines
5.4 KiB
Nim
Raw Normal View History

2022-03-30 12:51:28 +02:00
import pkg/asynctest
import pkg/chronos
2022-05-19 14:56:03 -05:00
import pkg/codex/sales
2022-03-30 12:51:28 +02:00
import ./helpers/mockmarket
2022-05-17 17:02:03 +02:00
import ./helpers/mockclock
2022-03-30 12:51:28 +02:00
import ./examples
suite "Sales":
let availability = Availability.init(
size=100.u256,
duration=60.u256,
minPrice=42.u256
)
2022-07-05 15:04:25 +02:00
var request = StorageRequest(
2022-07-05 09:39:59 +02:00
ask: StorageAsk(
duration: 60.u256,
size: 100.u256,
reward:42.u256,
slots: 4
2022-07-05 09:39:59 +02:00
),
content: StorageContent(
cid: "some cid"
)
)
2022-07-05 10:37:55 +02:00
let proof = seq[byte].example
2022-03-31 11:34:33 +02:00
2022-03-30 12:51:28 +02:00
var sales: Sales
var market: MockMarket
2022-05-17 17:02:03 +02:00
var clock: MockClock
2022-03-30 12:51:28 +02:00
setup:
market = MockMarket.new()
2022-05-17 17:02:03 +02:00
clock = MockClock.new()
sales = Sales.new(market, clock)
2022-07-07 16:18:45 +02:00
sales.onStore = proc(cid: string, availability: Availability) {.async.} =
discard
2022-07-07 16:18:45 +02:00
sales.onProve = proc(cid: string): Future[seq[byte]] {.async.} =
return proof
await sales.start()
2022-07-05 15:04:25 +02:00
request.expiry = (clock.now() + 42).u256
2022-03-31 11:34:33 +02:00
teardown:
await sales.stop()
2022-03-30 12:51:28 +02:00
test "has no availability initially":
check sales.available.len == 0
test "can add available storage":
let availability1 = Availability.example
let availability2 = Availability.example
sales.add(availability1)
check sales.available.contains(availability1)
sales.add(availability2)
check sales.available.contains(availability1)
check sales.available.contains(availability2)
test "can remove available storage":
sales.add(availability)
sales.remove(availability)
check sales.available.len == 0
test "generates unique ids for storage availability":
let availability1 = Availability.init(1.u256, 2.u256, 3.u256)
let availability2 = Availability.init(1.u256, 2.u256, 3.u256)
2022-03-30 12:51:28 +02:00
check availability1.id != availability2.id
2022-07-05 10:29:02 +02:00
test "makes storage unavailable when matching request comes in":
2022-07-05 09:39:59 +02:00
sales.add(availability)
discard await market.requestStorage(request)
2022-07-05 10:29:02 +02:00
check sales.available.len == 0
2022-07-05 09:39:59 +02:00
test "ignores request when no matching storage is available":
sales.add(availability)
var tooBig = request
tooBig.ask.size = request.ask.size + 1
discard await market.requestStorage(tooBig)
2022-07-05 10:29:02 +02:00
check sales.available == @[availability]
2022-03-30 12:51:28 +02:00
test "retrieves and stores data locally":
var storingCid: string
var storingAvailability: Availability
2022-07-07 16:18:45 +02:00
sales.onStore = proc(cid: string, availability: Availability) {.async.} =
storingCid = cid
storingAvailability = availability
2022-03-30 12:51:28 +02:00
sales.add(availability)
discard await market.requestStorage(request)
check storingCid == request.content.cid
2022-03-30 17:28:56 +02:00
test "makes storage available again when data retrieval fails":
let error = newException(IOError, "data retrieval failed")
2022-07-07 16:18:45 +02:00
sales.onStore = proc(cid: string, availability: Availability) {.async.} =
raise error
sales.add(availability)
discard await market.requestStorage(request)
check sales.available == @[availability]
2022-07-05 10:24:33 +02:00
test "generates proof of storage":
var provingCid: string
2022-07-07 16:18:45 +02:00
sales.onProve = proc(cid: string): Future[seq[byte]] {.async.} = provingCid = cid
2022-07-05 10:24:33 +02:00
sales.add(availability)
discard await market.requestStorage(request)
check provingCid == request.content.cid
test "fills a slot":
2022-07-05 10:37:55 +02:00
sales.add(availability)
discard await market.requestStorage(request)
check market.filled.len == 1
check market.filled[0].requestId == request.id
check market.filled[0].slotIndex < request.ask.slots.u256
check market.filled[0].proof == proof
check market.filled[0].host == await market.getSigner()
2022-07-05 10:37:55 +02:00
test "calls onSale when slot is filled":
2022-07-05 10:51:01 +02:00
var soldAvailability: Availability
var soldRequest: StorageRequest
var soldSlotIndex: UInt256
sales.onSale = proc(availability: Availability,
request: StorageRequest,
slotIndex: UInt256) =
2022-07-05 10:51:01 +02:00
soldAvailability = availability
soldRequest = request
soldSlotIndex = slotIndex
2022-07-05 10:51:01 +02:00
sales.add(availability)
discard await market.requestStorage(request)
check soldAvailability == availability
check soldRequest == request
check soldSlotIndex < request.ask.slots.u256
test "calls onClear when storage becomes available again":
2022-07-07 16:18:45 +02:00
sales.onProve = proc(cid: string): Future[seq[byte]] {.async.} =
raise newException(IOError, "proof failed")
var clearedAvailability: Availability
var clearedRequest: StorageRequest
sales.onClear = proc(availability: Availability, request: StorageRequest) =
clearedAvailability = availability
clearedRequest = request
sales.add(availability)
discard await market.requestStorage(request)
check clearedAvailability == availability
check clearedRequest == request
test "makes storage available again when other host fills the slot":
let otherHost = Address.example
2022-07-07 16:18:45 +02:00
sales.onStore = proc(cid: string, availability: Availability) {.async.} =
await sleepAsync(1.hours)
sales.add(availability)
discard await market.requestStorage(request)
for slotIndex in 0..<request.ask.slots:
market.fillSlot(request.id, slotIndex.u256, proof, otherHost)
check sales.available == @[availability]
2022-07-05 15:04:25 +02:00
test "makes storage available again when request expires":
2022-07-07 16:18:45 +02:00
sales.onStore = proc(cid: string, availability: Availability) {.async.} =
await sleepAsync(1.hours)
2022-07-05 15:04:25 +02:00
sales.add(availability)
discard await market.requestStorage(request)
clock.set(request.expiry.truncate(int64))
await sleepAsync(2.seconds)
check sales.available == @[availability]