mirror of
https://github.com/codex-storage/nim-codex.git
synced 2025-01-09 20:45:38 +00:00
d56eb6aee1
* [contracts] Add SlotFreed event * [integration] allow test node to be stopped twice * [cli] add --validator option * [contracts] remove dead code * [contracts] instantiate OnChainMarket and OnChainClock only once * [contracts] add Validation * [sales] remove duplicate import * [market] add missing import * [market] subscribe to all SlotFilled events * [market] add freeSlot() * [sales] fix warnings * [market] subscribe to SlotFreed events * [contracts] fix warning * [validator] keep track of filled slots * [validation] remove slots that have ended * [proving] absorb Proofs into Market Both Proofs and Market are abstractions around the Marketplace contract, having them separately is more trouble than it's worth at the moment. * [market] add markProofAsMissing() * [clock] speed up waiting for clock in tests * [validator] mark proofs as missing * [timer] fix error on node shutdown * [cli] handle --persistence and --validator separately * [market] allow retrieval of proof timeout value * [validator] do not subscribe to SlotFreed events Freed slots are already handled in removeSlotsThatHaveEnded(), and onSlotsFreed() interfered with its iterator. * [validator] Start validation at the start of a new period To decrease the likelihood that we hit the validation timeout. * [validator] do not mark proofs as missing after timeout * [market] check whether proof can be marked as missing * [validator] simplify validation Simulate a transaction to mark proof as missing, instead of trying to keep track of all the conditions that may lead to a proof being marked as missing. * [build] use nim-ethers PR #40 Uses "pending" blocktag instead of "latest" blocktag for better simulation of transactions before sending them. https://github.com/status-im/nim-ethers/pull/40 * [integration] integration test for validator * [validator] monitor a maximum number of slots Adds cli parameter --validator-max-slots. * [market] fix missing collateral argument After rebasing, add the new argument to fillSlot calls. * [build] update to nim-ethers 0.2.5 * [validator] use Set instead of Table to keep track of slots * [validator] add logging * [validator] add test for slot failure * [market] use "pending" blocktag to use more up to date block time * [contracts] remove unused import * [validator] fix: wait until after period ends The smart contract checks that 'end < block.timestamp', so we need to wait until the block timestamp is greater than the period end.
238 lines
7.3 KiB
Nim
238 lines
7.3 KiB
Nim
## Nim-Codex
|
|
## Copyright (c) 2021 Status Research & Development GmbH
|
|
## Licensed under either of
|
|
## * Apache License, version 2.0, ([LICENSE-APACHE](LICENSE-APACHE))
|
|
## * MIT license ([LICENSE-MIT](LICENSE-MIT))
|
|
## at your option.
|
|
## This file may not be copied, modified, or distributed except according to
|
|
## those terms.
|
|
|
|
import std/sequtils
|
|
import std/os
|
|
import std/tables
|
|
|
|
import pkg/chronicles
|
|
import pkg/chronos
|
|
import pkg/presto
|
|
import pkg/libp2p
|
|
import pkg/confutils
|
|
import pkg/confutils/defs
|
|
import pkg/nitro
|
|
import pkg/stew/io2
|
|
import pkg/stew/shims/net as stewnet
|
|
import pkg/datastore
|
|
|
|
import ./node
|
|
import ./conf
|
|
import ./rng
|
|
import ./rest/api
|
|
import ./stores
|
|
import ./blockexchange
|
|
import ./utils/fileutils
|
|
import ./erasure
|
|
import ./discovery
|
|
import ./contracts
|
|
import ./contracts/clock
|
|
import ./utils/addrutils
|
|
import ./namespaces
|
|
|
|
logScope:
|
|
topics = "codex node"
|
|
|
|
type
|
|
CodexServer* = ref object
|
|
runHandle: Future[void]
|
|
config: CodexConf
|
|
restServer: RestServerRef
|
|
codexNode: CodexNodeRef
|
|
repoStore: RepoStore
|
|
maintenance: BlockMaintainer
|
|
|
|
CodexPrivateKey* = libp2p.PrivateKey # alias
|
|
|
|
proc start*(s: CodexServer) {.async.} =
|
|
notice "Starting codex node"
|
|
|
|
await s.repoStore.start()
|
|
s.restServer.start()
|
|
await s.codexNode.start()
|
|
s.maintenance.start()
|
|
|
|
let
|
|
# TODO: Can't define these as constants, pity
|
|
natIpPart = MultiAddress.init("/ip4/" & $s.config.nat & "/")
|
|
.expect("Should create multiaddress")
|
|
anyAddrIp = MultiAddress.init("/ip4/0.0.0.0/")
|
|
.expect("Should create multiaddress")
|
|
loopBackAddrIp = MultiAddress.init("/ip4/127.0.0.1/")
|
|
.expect("Should create multiaddress")
|
|
|
|
# announce addresses should be set to bound addresses,
|
|
# but the IP should be mapped to the provided nat ip
|
|
announceAddrs = s.codexNode.switch.peerInfo.addrs.mapIt:
|
|
block:
|
|
let
|
|
listenIPPart = it[multiCodec("ip4")].expect("Should get IP")
|
|
|
|
if listenIPPart == anyAddrIp or
|
|
(listenIPPart == loopBackAddrIp and natIpPart != loopBackAddrIp):
|
|
it.remapAddr(s.config.nat.some)
|
|
else:
|
|
it
|
|
|
|
s.codexNode.discovery.updateAnnounceRecord(announceAddrs)
|
|
s.codexNode.discovery.updateDhtRecord(s.config.nat, s.config.discoveryPort)
|
|
|
|
s.runHandle = newFuture[void]("codex.runHandle")
|
|
await s.runHandle
|
|
|
|
proc stop*(s: CodexServer) {.async.} =
|
|
notice "Stopping codex node"
|
|
|
|
await allFuturesThrowing(
|
|
s.restServer.stop(),
|
|
s.codexNode.stop(),
|
|
s.repoStore.stop(),
|
|
s.maintenance.stop())
|
|
|
|
s.runHandle.complete()
|
|
|
|
proc new(_: type Contracts,
|
|
config: CodexConf,
|
|
repo: RepoStore): Contracts =
|
|
|
|
if not config.persistence and not config.validator:
|
|
if config.ethAccount.isSome:
|
|
warn "Ethereum account was set, but neither persistence nor validator is enabled"
|
|
return
|
|
|
|
without account =? config.ethAccount:
|
|
if config.persistence:
|
|
error "Persistence enabled, but no Ethereum account was set"
|
|
if config.validator:
|
|
error "Validator enabled, but no Ethereum account was set"
|
|
quit QuitFailure
|
|
|
|
var deploy: Deployment
|
|
try:
|
|
if deployFile =? config.ethDeployment:
|
|
deploy = Deployment.init(deployFile)
|
|
else:
|
|
deploy = Deployment.init()
|
|
except IOError as e:
|
|
error "Unable to read deployment json"
|
|
quit QuitFailure
|
|
|
|
without marketplaceAddress =? deploy.address(Marketplace):
|
|
error "Marketplace contract address not found in deployment file"
|
|
quit QuitFailure
|
|
|
|
let provider = JsonRpcProvider.new(config.ethProvider)
|
|
let signer = provider.getSigner(account)
|
|
let marketplace = Marketplace.new(marketplaceAddress, signer)
|
|
let market = OnChainMarket.new(marketplace)
|
|
let clock = OnChainClock.new(provider)
|
|
|
|
var client: ?ClientInteractions
|
|
var host: ?HostInteractions
|
|
var validator: ?ValidatorInteractions
|
|
if config.persistence:
|
|
let purchasing = Purchasing.new(market, clock)
|
|
let proving = Proving.new(market, clock)
|
|
let sales = Sales.new(market, clock, proving, repo)
|
|
client = some ClientInteractions.new(clock, purchasing)
|
|
host = some HostInteractions.new(clock, sales, proving)
|
|
if config.validator:
|
|
let validation = Validation.new(clock, market, config.validatorMaxSlots)
|
|
validator = some ValidatorInteractions.new(clock, validation)
|
|
|
|
(client, host, validator)
|
|
|
|
proc new*(T: type CodexServer, config: CodexConf, privateKey: CodexPrivateKey): T =
|
|
|
|
let
|
|
switch = SwitchBuilder
|
|
.new()
|
|
.withPrivateKey(privateKey)
|
|
.withAddresses(config.listenAddrs)
|
|
.withRng(Rng.instance())
|
|
.withNoise()
|
|
.withMplex(5.minutes, 5.minutes)
|
|
.withMaxConnections(config.maxPeers)
|
|
.withAgentVersion(config.agentString)
|
|
.withSignedPeerRecord(true)
|
|
.withTcpTransport({ServerFlags.ReuseAddr})
|
|
.build()
|
|
|
|
var
|
|
cache: CacheStore = nil
|
|
|
|
if config.cacheSize > 0:
|
|
cache = CacheStore.new(cacheSize = config.cacheSize * MiB)
|
|
## Is unused?
|
|
|
|
let
|
|
discoveryDir = config.dataDir / CodexDhtNamespace
|
|
|
|
if io2.createPath(discoveryDir).isErr:
|
|
trace "Unable to create discovery directory for block store", discoveryDir = discoveryDir
|
|
raise (ref Defect)(
|
|
msg: "Unable to create discovery directory for block store: " & discoveryDir)
|
|
|
|
let
|
|
discoveryStore = Datastore(
|
|
SQLiteDatastore.new(config.dataDir / CodexDhtProvidersNamespace)
|
|
.expect("Should create discovery datastore!"))
|
|
|
|
discovery = Discovery.new(
|
|
switch.peerInfo.privateKey,
|
|
announceAddrs = config.listenAddrs,
|
|
bindIp = config.discoveryIp,
|
|
bindPort = config.discoveryPort,
|
|
bootstrapNodes = config.bootstrapNodes,
|
|
store = discoveryStore)
|
|
|
|
wallet = WalletRef.new(EthPrivateKey.random())
|
|
network = BlockExcNetwork.new(switch)
|
|
|
|
repoData = case config.repoKind
|
|
of repoFS: Datastore(FSDatastore.new($config.dataDir, depth = 5)
|
|
.expect("Should create repo file data store!"))
|
|
of repoSQLite: Datastore(SQLiteDatastore.new($config.dataDir)
|
|
.expect("Should create repo SQLite data store!"))
|
|
|
|
repoStore = RepoStore.new(
|
|
repoDs = repoData,
|
|
metaDs = SQLiteDatastore.new(config.dataDir / CodexMetaNamespace)
|
|
.expect("Should create meta data store!"),
|
|
quotaMaxBytes = config.storageQuota.uint,
|
|
blockTtl = config.blockTtlSeconds.seconds)
|
|
|
|
maintenance = BlockMaintainer.new(
|
|
repoStore,
|
|
interval = config.blockMaintenanceIntervalSeconds.seconds,
|
|
numberOfBlocksPerInterval = config.blockMaintenanceNumberOfBlocks)
|
|
|
|
peerStore = PeerCtxStore.new()
|
|
pendingBlocks = PendingBlocksManager.new()
|
|
blockDiscovery = DiscoveryEngine.new(repoStore, peerStore, network, discovery, pendingBlocks)
|
|
engine = BlockExcEngine.new(repoStore, wallet, network, blockDiscovery, peerStore, pendingBlocks)
|
|
store = NetworkStore.new(engine, repoStore)
|
|
erasure = Erasure.new(store, leoEncoderProvider, leoDecoderProvider)
|
|
contracts = Contracts.new(config, repoStore)
|
|
codexNode = CodexNodeRef.new(switch, store, engine, erasure, discovery, contracts)
|
|
restServer = RestServerRef.new(
|
|
codexNode.initRestApi(config),
|
|
initTAddress(config.apiBindAddress , config.apiPort),
|
|
bufferSize = (1024 * 64),
|
|
maxRequestBodySize = int.high)
|
|
.expect("Should start rest server!")
|
|
|
|
switch.mount(network)
|
|
T(
|
|
config: config,
|
|
codexNode: codexNode,
|
|
restServer: restServer,
|
|
repoStore: repoStore,
|
|
maintenance: maintenance)
|