nim-codex/codex/sales.nim
2022-05-19 13:56:03 -06:00

162 lines
4.7 KiB
Nim

import std/sequtils
import pkg/questionable
import pkg/upraises
import pkg/stint
import pkg/nimcrypto
import pkg/chronicles
import ./market
import ./clock
export stint
const DefaultOfferExpiryInterval = (10 * 60).u256
type
Sales* = ref object
market: Market
clock: Clock
subscription: ?Subscription
available*: seq[Availability]
offerExpiryInterval*: UInt256
onSale: ?OnSale
Availability* = object
id*: array[32, byte]
size*: UInt256
duration*: UInt256
minPrice*: UInt256
Negotiation = ref object
sales: Sales
requestId: array[32, byte]
ask: StorageAsk
availability: Availability
offer: ?StorageOffer
subscription: ?Subscription
waiting: ?Future[void]
finished: bool
OnSale = proc(offer: StorageOffer) {.gcsafe, upraises: [].}
func new*(_: type Sales, market: Market, clock: Clock): Sales =
Sales(
market: market,
clock: clock,
offerExpiryInterval: DefaultOfferExpiryInterval
)
proc init*(_: type Availability,
size: UInt256,
duration: UInt256,
minPrice: UInt256): Availability =
var id: array[32, byte]
doAssert randomBytes(id) == 32
Availability(id: id, size: size, duration: duration, minPrice: minPrice)
proc `onSale=`*(sales: Sales, callback: OnSale) =
sales.onSale = some callback
func add*(sales: Sales, availability: Availability) =
sales.available.add(availability)
func remove*(sales: Sales, availability: Availability) =
sales.available.keepItIf(it != availability)
func findAvailability(sales: Sales, ask: StorageAsk): ?Availability =
for availability in sales.available:
if ask.size <= availability.size and
ask.duration <= availability.duration and
ask.maxPrice >= availability.minPrice:
return some availability
proc createOffer(negotiation: Negotiation): StorageOffer =
let sales = negotiation.sales
StorageOffer(
requestId: negotiation.requestId,
price: negotiation.ask.maxPrice,
expiry: sales.clock.now().u256 + sales.offerExpiryInterval
)
proc sendOffer(negotiation: Negotiation) {.async.} =
let offer = negotiation.createOffer()
negotiation.offer = some await negotiation.sales.market.offerStorage(offer)
proc finish(negotiation: Negotiation, success: bool) =
if negotiation.finished:
return
negotiation.finished = true
if subscription =? negotiation.subscription:
asyncSpawn subscription.unsubscribe()
if waiting =? negotiation.waiting:
waiting.cancel()
if success and offer =? negotiation.offer:
if onSale =? negotiation.sales.onSale:
onSale(offer)
else:
negotiation.sales.add(negotiation.availability)
proc onSelect(negotiation: Negotiation, offerId: array[32, byte]) =
if offer =? negotiation.offer and offer.id == offerId:
negotiation.finish(success = true)
else:
negotiation.finish(success = false)
proc subscribeSelect(negotiation: Negotiation) {.async.} =
without offer =? negotiation.offer:
return
proc onSelect(offerId: array[32, byte]) {.gcsafe, upraises:[].} =
negotiation.onSelect(offerId)
let market = negotiation.sales.market
let subscription = await market.subscribeSelection(offer.requestId, onSelect)
negotiation.subscription = some subscription
proc waitForExpiry(negotiation: Negotiation) {.async.} =
without offer =? negotiation.offer:
return
await negotiation.sales.clock.waitUntil(offer.expiry.truncate(int64))
negotiation.finish(success = false)
proc start(negotiation: Negotiation) {.async.} =
try:
let sales = negotiation.sales
let availability = negotiation.availability
sales.remove(availability)
await negotiation.sendOffer()
await negotiation.subscribeSelect()
negotiation.waiting = some negotiation.waitForExpiry()
except CatchableError as e:
error "Negotiation failed", msg = e.msg
proc handleRequest(sales: Sales, requestId: array[32, byte], ask: StorageAsk) =
without availability =? sales.findAvailability(ask):
return
let negotiation = Negotiation(
sales: sales,
requestId: requestId,
ask: ask,
availability: availability
)
asyncSpawn negotiation.start()
proc start*(sales: Sales) {.async.} =
doAssert sales.subscription.isNone, "Sales already started"
proc onRequest(requestId: array[32, byte], ask: StorageAsk) {.gcsafe, upraises:[].} =
sales.handleRequest(requestId, ask)
try:
sales.subscription = some await sales.market.subscribeRequests(onRequest)
except CatchableError as e:
error "Unable to start sales", msg = e.msg
proc stop*(sales: Sales) {.async.} =
if subscription =? sales.subscription:
sales.subscription = Subscription.none
try:
await subscription.unsubscribe()
except CatchableError as e:
warn "Unsubscribe failed", msg = e.msg