import std/times import pkg/chronos import pkg/asynctest import pkg/datastore import pkg/questionable import pkg/questionable/results import pkg/codex/sales import pkg/codex/sales/reservations import pkg/codex/stores/repostore import ./helpers/mockmarket import ./helpers/mockclock import ./helpers/eventually import ./examples import ./sales/teststatemachine import ./sales/testreservations import ./sales/helpers suite "Sales": let availability = Availability.init( size=100.u256, duration=60.u256, minPrice=600.u256 ) var request = StorageRequest( ask: StorageAsk( slots: 4, slotSize: 100.u256, duration: 60.u256, reward: 10.u256, ), content: StorageContent( cid: "some cid" ), expiry: (getTime() + initDuration(hours=1)).toUnix.u256 ) let proof = exampleProof() var sales: Sales market: MockMarket clock: MockClock proving: Proving setup: market = MockMarket.new() clock = MockClock.new() proving = Proving.new() let repoDs = SQLiteDatastore.new(Memory).tryGet() let metaDs = SQLiteDatastore.new(Memory).tryGet() let repo = RepoStore.new(repoDs, metaDs) sales = Sales.new(market, clock, proving, repo) sales.onStore = proc(request: StorageRequest, slot: UInt256, availability: ?Availability) {.async.} = discard sales.onProve = proc(request: StorageRequest, slot: UInt256): Future[seq[byte]] {.async.} = return proof await sales.start() request.expiry = (clock.now() + 42).u256 teardown: await sales.stop() test "makes storage unavailable when matching request comes in": check isOk await sales.reservations.reserve(availability) await market.requestStorage(request) without availability =? await sales.reservations.get(availability.id): fail() check availability.used test "ignores request when no matching storage is available": check isOk await sales.reservations.reserve(availability) var tooBig = request tooBig.ask.slotSize = request.ask.slotSize + 1 await market.requestStorage(tooBig) without availability =? await sales.reservations.get(availability.id): fail() check not availability.used test "ignores request when reward is too low": check isOk await sales.reservations.reserve(availability) var tooCheap = request tooCheap.ask.reward = request.ask.reward - 1 await market.requestStorage(tooCheap) without availability =? await sales.reservations.get(availability.id): fail() check not availability.used test "retrieves and stores data locally": var storingRequest: StorageRequest var storingSlot: UInt256 var storingAvailability: Availability sales.onStore = proc(request: StorageRequest, slot: UInt256, availability: ?Availability) {.async.} = storingRequest = request storingSlot = slot check availability.isSome storingAvailability = !availability check isOk await sales.reservations.reserve(availability) await market.requestStorage(request) check storingRequest == request check storingSlot < request.ask.slots.u256 check storingAvailability == availability test "makes storage available again when data retrieval fails": let error = newException(IOError, "data retrieval failed") sales.onStore = proc(request: StorageRequest, slot: UInt256, availability: ?Availability) {.async.} = raise error check isOk await sales.reservations.reserve(availability) await market.requestStorage(request) without availability =? await sales.reservations.get(availability.id): fail() check not availability.used test "generates proof of storage": var provingRequest: StorageRequest var provingSlot: UInt256 sales.onProve = proc(request: StorageRequest, slot: UInt256): Future[seq[byte]] {.async.} = provingRequest = request provingSlot = slot check isOk await sales.reservations.reserve(availability) await market.requestStorage(request) check provingRequest == request check provingSlot < request.ask.slots.u256 test "fills a slot": check isOk await sales.reservations.reserve(availability) 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() test "calls onSale when slot is filled": var soldAvailability: Availability var soldRequest: StorageRequest var soldSlotIndex: UInt256 sales.onSale = proc(availability: ?Availability, request: StorageRequest, slotIndex: UInt256) = if a =? availability: soldAvailability = a soldRequest = request soldSlotIndex = slotIndex check isOk await sales.reservations.reserve(availability) await market.requestStorage(request) check soldAvailability == availability check soldRequest == request check soldSlotIndex < request.ask.slots.u256 test "calls onClear when storage becomes available again": # fail the proof intentionally to trigger `agent.finish(success=false)`, # which then calls the onClear callback sales.onProve = proc(request: StorageRequest, slot: UInt256): Future[seq[byte]] {.async.} = raise newException(IOError, "proof failed") var clearedAvailability: Availability var clearedRequest: StorageRequest var clearedSlotIndex: UInt256 sales.onClear = proc(availability: ?Availability, request: StorageRequest, slotIndex: UInt256) = if a =? availability: clearedAvailability = a clearedRequest = request clearedSlotIndex = slotIndex check isOk await sales.reservations.reserve(availability) await market.requestStorage(request) check clearedAvailability == availability check clearedRequest == request check clearedSlotIndex < request.ask.slots.u256 test "makes storage available again when other host fills the slot": let otherHost = Address.example sales.onStore = proc(request: StorageRequest, slot: UInt256, availability: ?Availability) {.async.} = await sleepAsync(chronos.hours(1)) check isOk await sales.reservations.reserve(availability) await market.requestStorage(request) for slotIndex in 0..