2018-04-27 08:53:53 +00:00
|
|
|
# Nimbus
|
|
|
|
# Copyright (c) 2018 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.
|
|
|
|
|
2018-07-22 21:34:43 +00:00
|
|
|
import
|
2019-02-05 19:15:50 +00:00
|
|
|
os, strutils, net, eth/keys, db/[storage_types, db_chain, select_backend],
|
|
|
|
eth/common as eth_common, eth/p2p as eth_p2p,
|
2019-02-06 18:03:36 +00:00
|
|
|
chronos, json_rpc/rpcserver, chronicles,
|
2019-02-05 19:15:50 +00:00
|
|
|
eth/p2p/rlpx_protocols/[eth_protocol, les_protocol],
|
|
|
|
eth/p2p/blockchain_sync,
|
2018-11-22 06:40:09 +00:00
|
|
|
config, genesis, rpc/[common, p2p, debug, whisper], p2p/chain,
|
2019-02-05 19:15:50 +00:00
|
|
|
eth/trie/db
|
2018-07-26 20:08:43 +00:00
|
|
|
|
2018-06-20 17:27:32 +00:00
|
|
|
## TODO:
|
|
|
|
## * No IPv6 support
|
|
|
|
## * No multiple bind addresses support
|
|
|
|
## * No database support
|
|
|
|
|
2018-07-12 11:14:04 +00:00
|
|
|
const
|
|
|
|
nimbusClientId = "nimbus 0.1.0"
|
|
|
|
|
2018-06-20 17:27:32 +00:00
|
|
|
when not defined(windows):
|
|
|
|
from posix import SIGINT, SIGTERM
|
|
|
|
|
|
|
|
type
|
|
|
|
NimbusState = enum
|
|
|
|
Starting, Running, Stopping, Stopped
|
|
|
|
|
|
|
|
NimbusObject = ref object
|
2018-07-20 17:02:19 +00:00
|
|
|
rpcServer*: RpcHttpServer
|
2018-07-12 11:14:04 +00:00
|
|
|
ethNode*: EthereumNode
|
2018-06-20 17:27:32 +00:00
|
|
|
state*: NimbusState
|
|
|
|
|
|
|
|
proc start(): NimbusObject =
|
|
|
|
var nimbus = NimbusObject()
|
|
|
|
var conf = getConfiguration()
|
|
|
|
|
2019-03-20 13:41:25 +00:00
|
|
|
setLogLevel(conf.debug.logLevel)
|
|
|
|
if len(conf.debug.logFile) != 0:
|
|
|
|
discard defaultChroniclesStream.output.open(conf.debug.logFile, fmAppend)
|
|
|
|
else:
|
|
|
|
discard defaultChroniclesStream.output.open(stdout)
|
|
|
|
|
2018-06-20 17:27:32 +00:00
|
|
|
## Creating RPC Server
|
|
|
|
if RpcFlags.Enabled in conf.rpc.flags:
|
2018-07-20 17:02:19 +00:00
|
|
|
nimbus.rpcServer = newRpcHttpServer(conf.rpc.binds)
|
2018-06-20 17:27:32 +00:00
|
|
|
setupCommonRpc(nimbus.rpcServer)
|
|
|
|
|
|
|
|
## Creating P2P Server
|
|
|
|
if conf.net.nodekey.isZeroKey():
|
|
|
|
conf.net.nodekey = newPrivateKey()
|
|
|
|
|
|
|
|
var keypair: KeyPair
|
|
|
|
keypair.seckey = conf.net.nodekey
|
|
|
|
keypair.pubkey = conf.net.nodekey.getPublicKey()
|
|
|
|
|
|
|
|
var address: Address
|
|
|
|
address.ip = parseIpAddress("0.0.0.0")
|
|
|
|
address.tcpPort = Port(conf.net.bindPort)
|
|
|
|
address.udpPort = Port(conf.net.discPort)
|
|
|
|
|
2018-09-24 22:25:17 +00:00
|
|
|
createDir(conf.dataDir)
|
|
|
|
let trieDB = trieDB newChainDb(conf.dataDir)
|
2018-11-30 10:07:20 +00:00
|
|
|
let chainDB = newBaseChainDB(trieDB, conf.prune == PruneMode.Full)
|
2018-07-26 20:08:43 +00:00
|
|
|
|
|
|
|
if canonicalHeadHashKey().toOpenArray notin trieDB:
|
|
|
|
initializeEmptyDb(chainDb)
|
2019-03-13 21:36:54 +00:00
|
|
|
doAssert(canonicalHeadHashKey().toOpenArray in trieDB)
|
2018-07-26 20:08:43 +00:00
|
|
|
|
2018-07-12 11:14:04 +00:00
|
|
|
nimbus.ethNode = newEthereumNode(keypair, address, conf.net.networkId,
|
|
|
|
nil, nimbusClientId)
|
2018-06-20 17:27:32 +00:00
|
|
|
|
2018-08-29 08:49:01 +00:00
|
|
|
nimbus.ethNode.chain = newChain(chainDB)
|
2018-07-26 20:08:43 +00:00
|
|
|
|
2018-11-22 06:40:09 +00:00
|
|
|
if RpcFlags.Eth in conf.rpc.flags:
|
2018-08-29 08:49:01 +00:00
|
|
|
setupEthRpc(nimbus.ethNode, chainDB, nimbus.rpcServer)
|
2018-06-20 17:27:32 +00:00
|
|
|
|
2018-11-22 06:40:09 +00:00
|
|
|
if RpcFlags.Shh in conf.rpc.flags:
|
|
|
|
setupWhisperRPC(nimbus.rpcServer)
|
|
|
|
|
|
|
|
if RpcFlags.Debug in conf.rpc.flags:
|
|
|
|
setupDebugRpc(chainDB, nimbus.rpcServer)
|
|
|
|
|
2018-06-20 17:27:32 +00:00
|
|
|
## Starting servers
|
|
|
|
nimbus.state = Starting
|
|
|
|
if RpcFlags.Enabled in conf.rpc.flags:
|
|
|
|
nimbus.rpcServer.rpc("admin_quit") do() -> string:
|
|
|
|
nimbus.state = Stopping
|
|
|
|
result = "EXITING"
|
|
|
|
nimbus.rpcServer.start()
|
2018-07-12 11:14:04 +00:00
|
|
|
|
2018-09-25 13:30:44 +00:00
|
|
|
waitFor nimbus.ethNode.connectToNetwork(conf.net.bootNodes,
|
|
|
|
enableDiscovery = NoDiscover notin conf.net.flags)
|
2018-07-22 21:34:43 +00:00
|
|
|
|
|
|
|
# TODO: temp code until the CLI/RPC interface is fleshed out
|
2018-08-29 08:49:01 +00:00
|
|
|
let status = waitFor nimbus.ethNode.fastBlockchainSync()
|
|
|
|
if status != syncSuccess:
|
2018-12-06 23:16:34 +00:00
|
|
|
debug "Block sync failed: ", status
|
2018-07-12 11:14:04 +00:00
|
|
|
|
2018-06-20 17:27:32 +00:00
|
|
|
nimbus.state = Running
|
|
|
|
result = nimbus
|
|
|
|
|
|
|
|
proc stop*(nimbus: NimbusObject) {.async.} =
|
2018-12-06 23:16:34 +00:00
|
|
|
trace "Graceful shutdown"
|
2018-06-20 17:27:32 +00:00
|
|
|
nimbus.rpcServer.stop()
|
|
|
|
|
|
|
|
proc process*(nimbus: NimbusObject) =
|
|
|
|
if nimbus.state == Running:
|
|
|
|
when not defined(windows):
|
|
|
|
proc signalBreak(udata: pointer) =
|
|
|
|
nimbus.state = Stopping
|
|
|
|
# Adding SIGINT, SIGTERM handlers
|
2018-08-29 08:49:01 +00:00
|
|
|
# discard addSignal(SIGINT, signalBreak)
|
|
|
|
# discard addSignal(SIGTERM, signalBreak)
|
2018-06-20 17:27:32 +00:00
|
|
|
|
|
|
|
# Main loop
|
|
|
|
while nimbus.state == Running:
|
|
|
|
poll()
|
|
|
|
|
|
|
|
# Stop loop
|
|
|
|
waitFor nimbus.stop()
|
2018-04-27 08:53:53 +00:00
|
|
|
|
|
|
|
when isMainModule:
|
|
|
|
var message: string
|
2018-06-20 17:27:32 +00:00
|
|
|
|
|
|
|
## Pring Nimbus header
|
|
|
|
echo NimbusHeader
|
|
|
|
|
|
|
|
## Processing command line arguments
|
2018-05-02 15:01:10 +00:00
|
|
|
if processArguments(message) != ConfigStatus.Success:
|
2018-04-27 08:53:53 +00:00
|
|
|
echo message
|
|
|
|
quit(QuitFailure)
|
|
|
|
else:
|
|
|
|
if len(message) > 0:
|
|
|
|
echo message
|
2018-06-20 17:27:32 +00:00
|
|
|
quit(QuitSuccess)
|
2018-04-27 08:53:53 +00:00
|
|
|
|
2018-06-20 17:27:32 +00:00
|
|
|
var nimbus = start()
|
|
|
|
nimbus.process()
|