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":
|
|
|
|
|
2022-05-09 16:51:08 +02:00
|
|
|
let availability = Availability.init(
|
|
|
|
size=100.u256,
|
|
|
|
duration=60.u256,
|
|
|
|
minPrice=42.u256
|
2022-04-11 20:03:55 +02:00
|
|
|
)
|
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,
|
2022-07-21 14:32:38 +02:00
|
|
|
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.} =
|
2022-07-07 16:14:19 +02:00
|
|
|
discard
|
2022-07-07 16:18:45 +02:00
|
|
|
sales.onProve = proc(cid: string): Future[seq[byte]] {.async.} =
|
2022-07-07 16:14:19 +02:00
|
|
|
return proof
|
2022-05-17 16:40:57 +02:00
|
|
|
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:
|
2022-05-17 16:40:57 +02:00
|
|
|
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":
|
2022-05-09 16:51:08 +02:00
|
|
|
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-04 17:45:53 +02:00
|
|
|
|
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
|
|
|
|
2022-07-07 15:10:53 +02:00
|
|
|
test "retrieves and stores data locally":
|
|
|
|
var storingCid: string
|
2022-07-07 16:14:19 +02:00
|
|
|
var storingAvailability: Availability
|
2022-07-07 16:18:45 +02:00
|
|
|
sales.onStore = proc(cid: string, availability: Availability) {.async.} =
|
2022-07-07 16:14:19 +02:00
|
|
|
storingCid = cid
|
|
|
|
storingAvailability = availability
|
2022-03-30 12:51:28 +02:00
|
|
|
sales.add(availability)
|
2022-03-31 10:46:03 +02:00
|
|
|
discard await market.requestStorage(request)
|
2022-07-07 15:10:53 +02:00
|
|
|
check storingCid == request.content.cid
|
2022-03-30 17:28:56 +02:00
|
|
|
|
2022-07-05 10:32:09 +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.} =
|
2022-07-07 16:14:19 +02:00
|
|
|
raise error
|
2022-07-05 10:32:09 +02:00
|
|
|
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
|
|
|
|
|
2022-07-21 14:32:38 +02:00
|
|
|
test "fills a slot":
|
2022-07-05 10:37:55 +02:00
|
|
|
sales.add(availability)
|
|
|
|
discard await market.requestStorage(request)
|
2022-07-21 14:32:38 +02:00
|
|
|
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
|
|
|
|
2022-07-21 14:32:38 +02:00
|
|
|
test "calls onSale when slot is filled":
|
2022-07-05 10:51:01 +02:00
|
|
|
var soldAvailability: Availability
|
|
|
|
var soldRequest: StorageRequest
|
2022-07-21 14:32:38 +02:00
|
|
|
var soldSlotIndex: UInt256
|
|
|
|
sales.onSale = proc(availability: Availability,
|
|
|
|
request: StorageRequest,
|
|
|
|
slotIndex: UInt256) =
|
2022-07-05 10:51:01 +02:00
|
|
|
soldAvailability = availability
|
|
|
|
soldRequest = request
|
2022-07-21 14:32:38 +02:00
|
|
|
soldSlotIndex = slotIndex
|
2022-07-05 10:51:01 +02:00
|
|
|
sales.add(availability)
|
|
|
|
discard await market.requestStorage(request)
|
|
|
|
check soldAvailability == availability
|
|
|
|
check soldRequest == request
|
2022-07-21 14:32:38 +02:00
|
|
|
check soldSlotIndex < request.ask.slots.u256
|
2022-07-04 17:45:53 +02:00
|
|
|
|
2022-07-07 16:14:19 +02:00
|
|
|
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.} =
|
2022-07-07 16:14:19 +02:00
|
|
|
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
|
|
|
|
|
2022-07-21 14:32:38 +02:00
|
|
|
test "makes storage available again when other host fills the slot":
|
2022-07-05 14:19:01 +02:00
|
|
|
let otherHost = Address.example
|
2022-07-07 16:18:45 +02:00
|
|
|
sales.onStore = proc(cid: string, availability: Availability) {.async.} =
|
2022-07-07 16:14:19 +02:00
|
|
|
await sleepAsync(1.hours)
|
2022-07-05 14:19:01 +02:00
|
|
|
sales.add(availability)
|
|
|
|
discard await market.requestStorage(request)
|
2022-07-21 14:32:38 +02:00
|
|
|
for slotIndex in 0..<request.ask.slots:
|
|
|
|
market.fillSlot(request.id, slotIndex.u256, proof, otherHost)
|
2022-07-05 14:19:01 +02:00
|
|
|
check sales.available == @[availability]
|
2022-07-04 17:45:53 +02:00
|
|
|
|
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.} =
|
2022-07-07 16:14:19 +02:00
|
|
|
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]
|