mirror of
https://github.com/codex-storage/nim-codex.git
synced 2025-01-24 19:59:51 +00:00
de88fd2c53
* implement a logging proxy The logging proxy: - prevents the need to import chronicles (as well as export except toJson), - prevents the need to override `writeValue` or use or import nim-json-seralization elsewhere in the codebase, allowing for sole use of utils/json for de/serialization, - and handles json formatting correctly in chronicles json sinks * Rename logging -> logutils to avoid ambiguity with common names * clean up * add setProperty for JsonRecord, remove nim-json-serialization conflict * Allow specifying textlines and json format separately Not specifying a LogFormat will apply the formatting to both textlines and json sinks. Specifying a LogFormat will apply the formatting to only that sink. * remove unneeded usages of std/json We only need to import utils/json instead of std/json * move serialization from rest/json to utils/json so it can be shared * fix NoColors ambiguity Was causing unit tests to fail on Windows. * Remove nre usage to fix Windows error Windows was erroring with `could not load: pcre64.dll`. Instead of fixing that error, remove the pcre usage :) * Add logutils module doc * Shorten logutils.formatIt for `NBytes` Both json and textlines formatIt were not needed, and could be combined into one formatIt * remove debug integration test config debug output and logformat of json for integration test logs * Use ## module doc to support docgen * bump nim-poseidon2 to export fromBytes Before the changes in this branch, fromBytes was likely being resolved by nim-stew, or other dependency. With the changes in this branch, that dependency was removed and fromBytes could no longer be resolved. By exporting fromBytes from nim-poseidon, the correct resolution is now happening. * fixes to get compiling after rebasing master * Add support for Result types being logged using formatIt
137 lines
4.0 KiB
Nim
137 lines
4.0 KiB
Nim
import std/options
|
|
import pkg/questionable/results
|
|
import ../../clock
|
|
import ../../logutils
|
|
import ../statemachine
|
|
import ../salesagent
|
|
import ../salescontext
|
|
import ./errorhandling
|
|
import ./cancelled
|
|
import ./failed
|
|
import ./errored
|
|
import ./payout
|
|
|
|
logScope:
|
|
topics = "marketplace sales proving"
|
|
|
|
type
|
|
SlotNotFilledError* = object of CatchableError
|
|
SaleProving* = ref object of ErrorHandlingState
|
|
loop: Future[void]
|
|
|
|
method prove*(
|
|
state: SaleProving,
|
|
slot: Slot,
|
|
challenge: ProofChallenge,
|
|
onProve: OnProve,
|
|
market: Market,
|
|
currentPeriod: Period
|
|
) {.base, async.} =
|
|
try:
|
|
without proof =? (await onProve(slot, challenge)), err:
|
|
error "Failed to generate proof", error = err.msg
|
|
# In this state, there's nothing we can do except try again next time.
|
|
return
|
|
debug "Submitting proof", currentPeriod = currentPeriod, slotId = slot.id
|
|
await market.submitProof(slot.id, proof)
|
|
except CatchableError as e:
|
|
error "Submitting proof failed", msg = e.msg
|
|
|
|
proc proveLoop(
|
|
state: SaleProving,
|
|
market: Market,
|
|
clock: Clock,
|
|
request: StorageRequest,
|
|
slotIndex: UInt256,
|
|
onProve: OnProve
|
|
) {.async.} =
|
|
|
|
let slot = Slot(request: request, slotIndex: slotIndex)
|
|
let slotId = slot.id
|
|
|
|
logScope:
|
|
period = currentPeriod
|
|
requestId = request.id
|
|
slotIndex
|
|
slotId = slot.id
|
|
|
|
proc getCurrentPeriod(): Future[Period] {.async.} =
|
|
let periodicity = await market.periodicity()
|
|
return periodicity.periodOf(clock.now().u256)
|
|
|
|
proc waitUntilPeriod(period: Period) {.async.} =
|
|
let periodicity = await market.periodicity()
|
|
await clock.waitUntil(periodicity.periodStart(period).truncate(int64))
|
|
|
|
while true:
|
|
let currentPeriod = await getCurrentPeriod()
|
|
let slotState = await market.slotState(slot.id)
|
|
if slotState == SlotState.Finished:
|
|
debug "Slot reached finished state", period = currentPeriod
|
|
return
|
|
|
|
if slotState != SlotState.Filled:
|
|
raise newException(SlotNotFilledError, "Slot is not in Filled state!")
|
|
|
|
debug "Proving for new period", period = currentPeriod
|
|
|
|
if (await market.isProofRequired(slotId)) or (await market.willProofBeRequired(slotId)):
|
|
let challenge = await market.getChallenge(slotId)
|
|
debug "Proof is required", period = currentPeriod, challenge = challenge
|
|
await state.prove(slot, challenge, onProve, market, currentPeriod)
|
|
|
|
await waitUntilPeriod(currentPeriod + 1)
|
|
|
|
method `$`*(state: SaleProving): string = "SaleProving"
|
|
|
|
method onCancelled*(state: SaleProving, request: StorageRequest): ?State =
|
|
# state.loop cancellation happens automatically when run is cancelled due to
|
|
# state change
|
|
return some State(SaleCancelled())
|
|
|
|
method onFailed*(state: SaleProving, request: StorageRequest): ?State =
|
|
# state.loop cancellation happens automatically when run is cancelled due to
|
|
# state change
|
|
return some State(SaleFailed())
|
|
|
|
method run*(state: SaleProving, machine: Machine): Future[?State] {.async.} =
|
|
let data = SalesAgent(machine).data
|
|
let context = SalesAgent(machine).context
|
|
|
|
without request =? data.request:
|
|
raiseAssert "no sale request"
|
|
|
|
without onProve =? context.onProve:
|
|
raiseAssert "onProve callback not set"
|
|
|
|
without market =? context.market:
|
|
raiseAssert("market not set")
|
|
|
|
without clock =? context.clock:
|
|
raiseAssert("clock not set")
|
|
|
|
debug "Start proving", requestId = data.requestId, slotIndex = data.slotIndex
|
|
try:
|
|
let loop = state.proveLoop(market, clock, request, data.slotIndex, onProve)
|
|
state.loop = loop
|
|
await loop
|
|
except CancelledError:
|
|
discard
|
|
except CatchableError as e:
|
|
error "Proving failed", msg = e.msg
|
|
return some State(SaleErrored(error: e))
|
|
finally:
|
|
# Cleanup of the proving loop
|
|
debug "Stopping proving.", requestId = data.requestId, slotIndex = data.slotIndex
|
|
|
|
if not state.loop.isNil:
|
|
if not state.loop.finished:
|
|
try:
|
|
await state.loop.cancelAndWait()
|
|
except CatchableError as e:
|
|
error "Error during cancelation of prooving loop", msg = e.msg
|
|
|
|
state.loop = nil
|
|
|
|
return some State(SalePayout())
|