mirror of
https://github.com/status-im/nimbus-eth2.git
synced 2025-01-28 07:15:57 +00:00
beacon_node: graceful shutdown (#1033)
* beacon_node: graceful shutdown * separate BeaconNodeStatus and BeaconNode instances
This commit is contained in:
parent
f4d38611ef
commit
c4462af4ab
@ -36,6 +36,13 @@ type
|
|||||||
RpcServer* = RpcHttpServer
|
RpcServer* = RpcHttpServer
|
||||||
KeyPair = eth2_network.KeyPair
|
KeyPair = eth2_network.KeyPair
|
||||||
|
|
||||||
|
# "state" is already taken by BeaconState
|
||||||
|
BeaconNodeStatus* = enum
|
||||||
|
Starting, Running, Stopping
|
||||||
|
|
||||||
|
# this needs to be global, so it can be set in the Ctrl+C signal handler
|
||||||
|
var status = BeaconNodeStatus.Starting
|
||||||
|
|
||||||
template init(T: type RpcHttpServer, ip: IpAddress, port: Port): T =
|
template init(T: type RpcHttpServer, ip: IpAddress, port: Port): T =
|
||||||
newRpcHttpServer([initTAddress(ip, port)])
|
newRpcHttpServer([initTAddress(ip, port)])
|
||||||
|
|
||||||
@ -722,44 +729,61 @@ proc installAttestationHandlers(node: BeaconNode) =
|
|||||||
|
|
||||||
waitFor allFutures(attestationSubscriptions)
|
waitFor allFutures(attestationSubscriptions)
|
||||||
|
|
||||||
|
proc stop*(node: BeaconNode) =
|
||||||
|
status = BeaconNodeStatus.Stopping
|
||||||
|
info "Graceful shutdown"
|
||||||
|
waitFor node.network.stop()
|
||||||
|
|
||||||
proc run*(node: BeaconNode) =
|
proc run*(node: BeaconNode) =
|
||||||
if node.rpcServer != nil:
|
if status == BeaconNodeStatus.Starting:
|
||||||
node.rpcServer.installRpcHandlers(node)
|
# it might have been set to "Stopping" with Ctrl+C
|
||||||
node.rpcServer.start()
|
status = BeaconNodeStatus.Running
|
||||||
|
|
||||||
waitFor node.network.subscribe(node.topicBeaconBlocks) do (signedBlock: SignedBeaconBlock):
|
if node.rpcServer != nil:
|
||||||
onBeaconBlock(node, signedBlock)
|
node.rpcServer.installRpcHandlers(node)
|
||||||
do (signedBlock: SignedBeaconBlock) -> bool:
|
node.rpcServer.start()
|
||||||
let (afterGenesis, slot) = node.beaconClock.now.toSlot()
|
|
||||||
if not afterGenesis:
|
|
||||||
return false
|
|
||||||
node.blockPool.isValidBeaconBlock(signedBlock, slot, {})
|
|
||||||
|
|
||||||
installAttestationHandlers(node)
|
waitFor node.network.subscribe(node.topicBeaconBlocks) do (signedBlock: SignedBeaconBlock):
|
||||||
|
onBeaconBlock(node, signedBlock)
|
||||||
|
do (signedBlock: SignedBeaconBlock) -> bool:
|
||||||
|
let (afterGenesis, slot) = node.beaconClock.now.toSlot()
|
||||||
|
if not afterGenesis:
|
||||||
|
return false
|
||||||
|
node.blockPool.isValidBeaconBlock(signedBlock, slot, {})
|
||||||
|
|
||||||
let
|
installAttestationHandlers(node)
|
||||||
t = node.beaconClock.now().toSlot()
|
|
||||||
curSlot = if t.afterGenesis: t.slot
|
|
||||||
else: GENESIS_SLOT
|
|
||||||
nextSlot = curSlot + 1 # No earlier than GENESIS_SLOT + 1
|
|
||||||
fromNow = saturate(node.beaconClock.fromNow(nextSlot))
|
|
||||||
|
|
||||||
info "Scheduling first slot action",
|
let
|
||||||
beaconTime = shortLog(node.beaconClock.now()),
|
t = node.beaconClock.now().toSlot()
|
||||||
nextSlot = shortLog(nextSlot),
|
curSlot = if t.afterGenesis: t.slot
|
||||||
fromNow = shortLog(fromNow),
|
else: GENESIS_SLOT
|
||||||
cat = "scheduling"
|
nextSlot = curSlot + 1 # No earlier than GENESIS_SLOT + 1
|
||||||
|
fromNow = saturate(node.beaconClock.fromNow(nextSlot))
|
||||||
|
|
||||||
addTimer(fromNow) do (p: pointer):
|
info "Scheduling first slot action",
|
||||||
asyncCheck node.onSlotStart(curSlot, nextSlot)
|
beaconTime = shortLog(node.beaconClock.now()),
|
||||||
|
nextSlot = shortLog(nextSlot),
|
||||||
|
fromNow = shortLog(fromNow),
|
||||||
|
cat = "scheduling"
|
||||||
|
|
||||||
let second = Moment.now() + chronos.seconds(1)
|
addTimer(fromNow) do (p: pointer):
|
||||||
discard setTimer(second) do (p: pointer):
|
asyncCheck node.onSlotStart(curSlot, nextSlot)
|
||||||
asyncCheck node.onSecond(second)
|
|
||||||
|
|
||||||
node.syncLoop = runSyncLoop(node)
|
let second = Moment.now() + chronos.seconds(1)
|
||||||
|
discard setTimer(second) do (p: pointer):
|
||||||
|
asyncCheck node.onSecond(second)
|
||||||
|
|
||||||
runForever()
|
node.syncLoop = runSyncLoop(node)
|
||||||
|
|
||||||
|
# main event loop
|
||||||
|
while status == BeaconNodeStatus.Running:
|
||||||
|
try:
|
||||||
|
poll()
|
||||||
|
except CatchableError as e:
|
||||||
|
debug "Exception in poll()", exc = e.name, err = e.msg
|
||||||
|
|
||||||
|
# time to say goodbye
|
||||||
|
node.stop()
|
||||||
|
|
||||||
var gPidFile: string
|
var gPidFile: string
|
||||||
proc createPidFile(filename: string) =
|
proc createPidFile(filename: string) =
|
||||||
@ -955,15 +979,6 @@ programMain:
|
|||||||
stderr.write "Invalid value for --log-level. " & err.msg
|
stderr.write "Invalid value for --log-level. " & err.msg
|
||||||
quit 1
|
quit 1
|
||||||
|
|
||||||
## Ctrl+C handling
|
|
||||||
proc controlCHandler() {.noconv.} =
|
|
||||||
when defined(windows):
|
|
||||||
# workaround for https://github.com/nim-lang/Nim/issues/4057
|
|
||||||
setupForeignThreadGc()
|
|
||||||
debug "Shutting down after having received SIGINT"
|
|
||||||
quit(QuitFailure)
|
|
||||||
setControlCHook(controlCHandler)
|
|
||||||
|
|
||||||
case config.cmd
|
case config.cmd
|
||||||
of createTestnet:
|
of createTestnet:
|
||||||
var deposits: seq[Deposit]
|
var deposits: seq[Deposit]
|
||||||
@ -1043,6 +1058,16 @@ programMain:
|
|||||||
createPidFile(config.dataDir.string / "beacon_node.pid")
|
createPidFile(config.dataDir.string / "beacon_node.pid")
|
||||||
|
|
||||||
var node = waitFor BeaconNode.init(config)
|
var node = waitFor BeaconNode.init(config)
|
||||||
|
|
||||||
|
## Ctrl+C handling
|
||||||
|
proc controlCHandler() {.noconv.} =
|
||||||
|
when defined(windows):
|
||||||
|
# workaround for https://github.com/nim-lang/Nim/issues/4057
|
||||||
|
setupForeignThreadGc()
|
||||||
|
info "Shutting down after having received SIGINT"
|
||||||
|
status = BeaconNodeStatus.Stopping
|
||||||
|
setControlCHook(controlCHandler)
|
||||||
|
|
||||||
when hasPrompt:
|
when hasPrompt:
|
||||||
initPrompt(node)
|
initPrompt(node)
|
||||||
|
|
||||||
|
@ -718,6 +718,13 @@ proc start*(node: Eth2Node) {.async.} =
|
|||||||
node.discoveryLoop = node.runDiscoveryLoop()
|
node.discoveryLoop = node.runDiscoveryLoop()
|
||||||
traceAsyncErrors node.discoveryLoop
|
traceAsyncErrors node.discoveryLoop
|
||||||
|
|
||||||
|
proc stop*(node: Eth2Node) {.async.} =
|
||||||
|
# ignore errors in futures, since we're shutting down
|
||||||
|
await allFutures(@[
|
||||||
|
node.discovery.closeWait(),
|
||||||
|
node.switch.stop(),
|
||||||
|
])
|
||||||
|
|
||||||
proc init*(T: type Peer, network: Eth2Node, info: PeerInfo): Peer =
|
proc init*(T: type Peer, network: Eth2Node, info: PeerInfo): Peer =
|
||||||
new result
|
new result
|
||||||
result.info = info
|
result.info = info
|
||||||
|
@ -442,6 +442,7 @@ proc getGenesis*(m: MainchainMonitor): Future[BeaconStateRef] {.async.} =
|
|||||||
if m.genesisState != nil:
|
if m.genesisState != nil:
|
||||||
return m.genesisState
|
return m.genesisState
|
||||||
else:
|
else:
|
||||||
|
result = new BeaconStateRef # make the compiler happy
|
||||||
raiseAssert "Unreachable code"
|
raiseAssert "Unreachable code"
|
||||||
|
|
||||||
method getBlockByHash*(p: Web3DataProviderRef, hash: BlockHash): Future[BlockObject] =
|
method getBlockByHash*(p: Web3DataProviderRef, hash: BlockHash): Future[BlockObject] =
|
||||||
|
@ -110,7 +110,7 @@ if [[ "$USE_TMUX" != "no" ]]; then
|
|||||||
$TMUX select-window -t "${TMUX_SESSION_NAME}:sim"
|
$TMUX select-window -t "${TMUX_SESSION_NAME}:sim"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
$MAKE -j3 NIMFLAGS="$CUSTOM_NIMFLAGS $DEFS" LOG_LEVEL="${LOG_LEVEL:-DEBUG}" beacon_node process_dashboard deposit_contract
|
$MAKE -j3 --no-print-directory NIMFLAGS="$CUSTOM_NIMFLAGS $DEFS" LOG_LEVEL="${LOG_LEVEL:-DEBUG}" beacon_node process_dashboard deposit_contract
|
||||||
|
|
||||||
if [ ! -f "${LAST_VALIDATOR}" ]; then
|
if [ ! -f "${LAST_VALIDATOR}" ]; then
|
||||||
if [ "$WEB3_ARG" != "" ]; then
|
if [ "$WEB3_ARG" != "" ]; then
|
||||||
|
Loading…
x
Reference in New Issue
Block a user