mirror of
https://github.com/codex-storage/nim-codex.git
synced 2025-01-11 13:34:16 +00:00
d70ab59004
* refactor multi node test suite Refactor the multinode test suite into the marketplace test suite. - Arbitrary number of nodes can be started with each test: clients, providers, validators - Hardhat can also be started locally with each test, usually for the purpose of saving and inspecting its log file. - Log files for all nodes can be persisted on disk, with configuration at the test-level - Log files, if persisted (as specified in the test), will be persisted to a CI artifact - Node config is specified at the test-level instead of the suite-level - Node/Hardhat process starting/stopping is now async, and runs much faster - Per-node config includes: - simulating proof failures - logging to file - log level - log topics - storage quota - debug (print logs to stdout) - Tests find next available ports when starting nodes, as closing ports on Windows can lag - Hardhat is no longer required to be running prior to starting the integration tests (as long as Hardhat is configured to run in the tests). - If Hardhat is already running, a snapshot will be taken and reverted before and after each test, respectively. - If Hardhat is not already running and configured to run at the test-level, a Hardhat process will be spawned and torn down before and after each test, respectively. * additional logging for debug purposes * address PR feedback - fix spelling - revert change from catching ProviderError to SignerError -- this should be handled more consistently in the Market abstraction, and will be handled in another PR. - remove method label from raiseAssert - remove unused import * Use API instead of command exec to test for free port Use chronos `createStreamServer` API to test for free port by binding localhost address and port. Use `ServerFlags.ReuseAddr` to enable reuse of same IP/Port on multiple test runs. * clean up * remove upraises annotations from tests * Update tests to work with updated erasure coding slot sizes * update dataset size, nodes, tolerance to match valid ec params Integration tests now have valid dataset sizes (blocks), tolerances, and number of nodes, to work with valid ec params. These values are validated when requested storage. Print the rest api failure message (via doAssert) when a rest api call fails (eg the rest api may validate some ec params). All integration tests pass when the async `clock.now` changes are reverted. * dont use async clock for now * fix workflow * move integration logs uplod to reusable --------- Co-authored-by: Dmitriy Ryajov <dryajov@gmail.com>
139 lines
4.1 KiB
Nim
139 lines
4.1 KiB
Nim
import std/options
|
|
import pkg/questionable/results
|
|
import ../../clock
|
|
import ../../logutils
|
|
import ../../utils/exceptions
|
|
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.msgDetail
|
|
|
|
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)
|
|
|
|
debug "waiting until next period"
|
|
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 cancellation of proving loop", msg = e.msg
|
|
|
|
state.loop = nil
|
|
|
|
return some State(SalePayout())
|