beacon_node: graceful shutdown (#1033)

* beacon_node: graceful shutdown

* separate BeaconNodeStatus and BeaconNode instances
This commit is contained in:
Ștefan Talpalaru 2020-05-19 20:57:35 +02:00 committed by GitHub
parent f4d38611ef
commit c4462af4ab
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 72 additions and 39 deletions

View File

@ -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)

View File

@ -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

View File

@ -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] =

View File

@ -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