forked from cerc-io/plugeth
cmd/puppeth: integrate blockscout (#18261)
* cmd/puppeth: integrate blockscout * cmd/puppeth: expose debug namespace for blockscout * cmd/puppeth: fix dbdir * cmd/puppeth: run explorer in archive mode * cmd/puppeth: ensure node is synced * cmd/puppeth: fix explorer docker alignment + drop unneeded exec * cmd/puppeth: polish up config saving and reloading * cmd/puppeth: check both web and p2p port for explorer service
This commit is contained in:
parent
fa538ee7ed
commit
16e313699f
@ -30,108 +30,86 @@ import (
|
|||||||
|
|
||||||
// explorerDockerfile is the Dockerfile required to run a block explorer.
|
// explorerDockerfile is the Dockerfile required to run a block explorer.
|
||||||
var explorerDockerfile = `
|
var explorerDockerfile = `
|
||||||
FROM puppeth/explorer:latest
|
FROM puppeth/blockscout:latest
|
||||||
|
|
||||||
ADD ethstats.json /ethstats.json
|
|
||||||
ADD chain.json /chain.json
|
|
||||||
|
|
||||||
|
ADD genesis.json /genesis.json
|
||||||
RUN \
|
RUN \
|
||||||
echo '(cd ../eth-net-intelligence-api && pm2 start /ethstats.json)' > explorer.sh && \
|
echo 'geth --cache 512 init /genesis.json' > explorer.sh && \
|
||||||
echo '(cd ../etherchain-light && npm start &)' >> explorer.sh && \
|
echo $'geth --networkid {{.NetworkID}} --syncmode "full" --gcmode "archive" --port {{.EthPort}} --bootnodes {{.Bootnodes}} --ethstats \'{{.Ethstats}}\' --cache=512 --rpc --rpcapi "net,web3,eth,shh,debug" --rpccorsdomain "*" --rpcvhosts "*" --ws --wsorigins "*" --exitwhensynced' >> explorer.sh && \
|
||||||
echo 'exec /parity/parity --chain=/chain.json --port={{.NodePort}} --tracing=on --fat-db=on --pruning=archive' >> explorer.sh
|
echo $'exec geth --networkid {{.NetworkID}} --syncmode "full" --gcmode "archive" --port {{.EthPort}} --bootnodes {{.Bootnodes}} --ethstats \'{{.Ethstats}}\' --cache=512 --rpc --rpcapi "net,web3,eth,shh,debug" --rpccorsdomain "*" --rpcvhosts "*" --ws --wsorigins "*" &' >> explorer.sh && \
|
||||||
|
echo '/usr/local/bin/docker-entrypoint.sh postgres &' >> explorer.sh && \
|
||||||
|
echo 'sleep 5' >> explorer.sh && \
|
||||||
|
echo 'mix do ecto.drop --force, ecto.create, ecto.migrate' >> explorer.sh && \
|
||||||
|
echo 'mix phx.server' >> explorer.sh
|
||||||
|
|
||||||
ENTRYPOINT ["/bin/sh", "explorer.sh"]
|
ENTRYPOINT ["/bin/sh", "explorer.sh"]
|
||||||
`
|
`
|
||||||
|
|
||||||
// explorerEthstats is the configuration file for the ethstats javascript client.
|
|
||||||
var explorerEthstats = `[
|
|
||||||
{
|
|
||||||
"name" : "node-app",
|
|
||||||
"script" : "app.js",
|
|
||||||
"log_date_format" : "YYYY-MM-DD HH:mm Z",
|
|
||||||
"merge_logs" : false,
|
|
||||||
"watch" : false,
|
|
||||||
"max_restarts" : 10,
|
|
||||||
"exec_interpreter" : "node",
|
|
||||||
"exec_mode" : "fork_mode",
|
|
||||||
"env":
|
|
||||||
{
|
|
||||||
"NODE_ENV" : "production",
|
|
||||||
"RPC_HOST" : "localhost",
|
|
||||||
"RPC_PORT" : "8545",
|
|
||||||
"LISTENING_PORT" : "{{.Port}}",
|
|
||||||
"INSTANCE_NAME" : "{{.Name}}",
|
|
||||||
"CONTACT_DETAILS" : "",
|
|
||||||
"WS_SERVER" : "{{.Host}}",
|
|
||||||
"WS_SECRET" : "{{.Secret}}",
|
|
||||||
"VERBOSITY" : 2
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]`
|
|
||||||
|
|
||||||
// explorerComposefile is the docker-compose.yml file required to deploy and
|
// explorerComposefile is the docker-compose.yml file required to deploy and
|
||||||
// maintain a block explorer.
|
// maintain a block explorer.
|
||||||
var explorerComposefile = `
|
var explorerComposefile = `
|
||||||
version: '2'
|
version: '2'
|
||||||
services:
|
services:
|
||||||
explorer:
|
explorer:
|
||||||
build: .
|
build: .
|
||||||
image: {{.Network}}/explorer
|
image: {{.Network}}/explorer
|
||||||
container_name: {{.Network}}_explorer_1
|
container_name: {{.Network}}_explorer_1
|
||||||
ports:
|
ports:
|
||||||
- "{{.NodePort}}:{{.NodePort}}"
|
- "{{.EthPort}}:{{.EthPort}}"
|
||||||
- "{{.NodePort}}:{{.NodePort}}/udp"{{if not .VHost}}
|
- "{{.EthPort}}:{{.EthPort}}/udp"{{if not .VHost}}
|
||||||
- "{{.WebPort}}:3000"{{end}}
|
- "{{.WebPort}}:4000"{{end}}
|
||||||
volumes:
|
environment:
|
||||||
- {{.Datadir}}:/root/.local/share/io.parity.ethereum
|
- ETH_PORT={{.EthPort}}
|
||||||
environment:
|
- ETH_NAME={{.EthName}}
|
||||||
- NODE_PORT={{.NodePort}}/tcp
|
- BLOCK_TRANSFORMER={{.Transformer}}{{if .VHost}}
|
||||||
- STATS={{.Ethstats}}{{if .VHost}}
|
- VIRTUAL_HOST={{.VHost}}
|
||||||
- VIRTUAL_HOST={{.VHost}}
|
- VIRTUAL_PORT=4000{{end}}
|
||||||
- VIRTUAL_PORT=3000{{end}}
|
volumes:
|
||||||
logging:
|
- {{.Datadir}}:/opt/app/.ethereum
|
||||||
driver: "json-file"
|
- {{.DBDir}}:/var/lib/postgresql/data
|
||||||
options:
|
logging:
|
||||||
max-size: "1m"
|
driver: "json-file"
|
||||||
max-file: "10"
|
options:
|
||||||
restart: always
|
max-size: "1m"
|
||||||
|
max-file: "10"
|
||||||
|
restart: always
|
||||||
`
|
`
|
||||||
|
|
||||||
// deployExplorer deploys a new block explorer container to a remote machine via
|
// deployExplorer deploys a new block explorer container to a remote machine via
|
||||||
// SSH, docker and docker-compose. If an instance with the specified network name
|
// SSH, docker and docker-compose. If an instance with the specified network name
|
||||||
// already exists there, it will be overwritten!
|
// already exists there, it will be overwritten!
|
||||||
func deployExplorer(client *sshClient, network string, chainspec []byte, config *explorerInfos, nocache bool) ([]byte, error) {
|
func deployExplorer(client *sshClient, network string, bootnodes []string, config *explorerInfos, nocache bool, isClique bool) ([]byte, error) {
|
||||||
// Generate the content to upload to the server
|
// Generate the content to upload to the server
|
||||||
workdir := fmt.Sprintf("%d", rand.Int63())
|
workdir := fmt.Sprintf("%d", rand.Int63())
|
||||||
files := make(map[string][]byte)
|
files := make(map[string][]byte)
|
||||||
|
|
||||||
dockerfile := new(bytes.Buffer)
|
dockerfile := new(bytes.Buffer)
|
||||||
template.Must(template.New("").Parse(explorerDockerfile)).Execute(dockerfile, map[string]interface{}{
|
template.Must(template.New("").Parse(explorerDockerfile)).Execute(dockerfile, map[string]interface{}{
|
||||||
"NodePort": config.nodePort,
|
"NetworkID": config.node.network,
|
||||||
|
"Bootnodes": strings.Join(bootnodes, ","),
|
||||||
|
"Ethstats": config.node.ethstats,
|
||||||
|
"EthPort": config.node.port,
|
||||||
})
|
})
|
||||||
files[filepath.Join(workdir, "Dockerfile")] = dockerfile.Bytes()
|
files[filepath.Join(workdir, "Dockerfile")] = dockerfile.Bytes()
|
||||||
|
|
||||||
ethstats := new(bytes.Buffer)
|
transformer := "base"
|
||||||
template.Must(template.New("").Parse(explorerEthstats)).Execute(ethstats, map[string]interface{}{
|
if isClique {
|
||||||
"Port": config.nodePort,
|
transformer = "clique"
|
||||||
"Name": config.ethstats[:strings.Index(config.ethstats, ":")],
|
}
|
||||||
"Secret": config.ethstats[strings.Index(config.ethstats, ":")+1 : strings.Index(config.ethstats, "@")],
|
|
||||||
"Host": config.ethstats[strings.Index(config.ethstats, "@")+1:],
|
|
||||||
})
|
|
||||||
files[filepath.Join(workdir, "ethstats.json")] = ethstats.Bytes()
|
|
||||||
|
|
||||||
composefile := new(bytes.Buffer)
|
composefile := new(bytes.Buffer)
|
||||||
template.Must(template.New("").Parse(explorerComposefile)).Execute(composefile, map[string]interface{}{
|
template.Must(template.New("").Parse(explorerComposefile)).Execute(composefile, map[string]interface{}{
|
||||||
"Datadir": config.datadir,
|
"Network": network,
|
||||||
"Network": network,
|
"VHost": config.host,
|
||||||
"NodePort": config.nodePort,
|
"Ethstats": config.node.ethstats,
|
||||||
"VHost": config.webHost,
|
"Datadir": config.node.datadir,
|
||||||
"WebPort": config.webPort,
|
"DBDir": config.dbdir,
|
||||||
"Ethstats": config.ethstats[:strings.Index(config.ethstats, ":")],
|
"EthPort": config.node.port,
|
||||||
|
"EthName": config.node.ethstats[:strings.Index(config.node.ethstats, ":")],
|
||||||
|
"WebPort": config.port,
|
||||||
|
"Transformer": transformer,
|
||||||
})
|
})
|
||||||
files[filepath.Join(workdir, "docker-compose.yaml")] = composefile.Bytes()
|
files[filepath.Join(workdir, "docker-compose.yaml")] = composefile.Bytes()
|
||||||
|
files[filepath.Join(workdir, "genesis.json")] = config.node.genesis
|
||||||
files[filepath.Join(workdir, "chain.json")] = chainspec
|
|
||||||
|
|
||||||
// Upload the deployment files to the remote server (and clean up afterwards)
|
// Upload the deployment files to the remote server (and clean up afterwards)
|
||||||
if out, err := client.Upload(files); err != nil {
|
if out, err := client.Upload(files); err != nil {
|
||||||
@ -149,22 +127,20 @@ func deployExplorer(client *sshClient, network string, chainspec []byte, config
|
|||||||
// explorerInfos is returned from a block explorer status check to allow reporting
|
// explorerInfos is returned from a block explorer status check to allow reporting
|
||||||
// various configuration parameters.
|
// various configuration parameters.
|
||||||
type explorerInfos struct {
|
type explorerInfos struct {
|
||||||
datadir string
|
node *nodeInfos
|
||||||
ethstats string
|
dbdir string
|
||||||
nodePort int
|
host string
|
||||||
webHost string
|
port int
|
||||||
webPort int
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Report converts the typed struct into a plain string->string map, containing
|
// Report converts the typed struct into a plain string->string map, containing
|
||||||
// most - but not all - fields for reporting to the user.
|
// most - but not all - fields for reporting to the user.
|
||||||
func (info *explorerInfos) Report() map[string]string {
|
func (info *explorerInfos) Report() map[string]string {
|
||||||
report := map[string]string{
|
report := map[string]string{
|
||||||
"Data directory": info.datadir,
|
"Website address ": info.host,
|
||||||
"Node listener port ": strconv.Itoa(info.nodePort),
|
"Website listener port ": strconv.Itoa(info.port),
|
||||||
"Ethstats username": info.ethstats,
|
"Ethereum listener port ": strconv.Itoa(info.node.port),
|
||||||
"Website address ": info.webHost,
|
"Ethstats username": info.node.ethstats,
|
||||||
"Website listener port ": strconv.Itoa(info.webPort),
|
|
||||||
}
|
}
|
||||||
return report
|
return report
|
||||||
}
|
}
|
||||||
@ -172,7 +148,7 @@ func (info *explorerInfos) Report() map[string]string {
|
|||||||
// checkExplorer does a health-check against a block explorer server to verify
|
// checkExplorer does a health-check against a block explorer server to verify
|
||||||
// whether it's running, and if yes, whether it's responsive.
|
// whether it's running, and if yes, whether it's responsive.
|
||||||
func checkExplorer(client *sshClient, network string) (*explorerInfos, error) {
|
func checkExplorer(client *sshClient, network string) (*explorerInfos, error) {
|
||||||
// Inspect a possible block explorer container on the host
|
// Inspect a possible explorer container on the host
|
||||||
infos, err := inspectContainer(client, fmt.Sprintf("%s_explorer_1", network))
|
infos, err := inspectContainer(client, fmt.Sprintf("%s_explorer_1", network))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
@ -181,13 +157,13 @@ func checkExplorer(client *sshClient, network string) (*explorerInfos, error) {
|
|||||||
return nil, ErrServiceOffline
|
return nil, ErrServiceOffline
|
||||||
}
|
}
|
||||||
// Resolve the port from the host, or the reverse proxy
|
// Resolve the port from the host, or the reverse proxy
|
||||||
webPort := infos.portmap["3000/tcp"]
|
port := infos.portmap["4000/tcp"]
|
||||||
if webPort == 0 {
|
if port == 0 {
|
||||||
if proxy, _ := checkNginx(client, network); proxy != nil {
|
if proxy, _ := checkNginx(client, network); proxy != nil {
|
||||||
webPort = proxy.port
|
port = proxy.port
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if webPort == 0 {
|
if port == 0 {
|
||||||
return nil, ErrNotExposed
|
return nil, ErrNotExposed
|
||||||
}
|
}
|
||||||
// Resolve the host from the reverse-proxy and the config values
|
// Resolve the host from the reverse-proxy and the config values
|
||||||
@ -196,17 +172,23 @@ func checkExplorer(client *sshClient, network string) (*explorerInfos, error) {
|
|||||||
host = client.server
|
host = client.server
|
||||||
}
|
}
|
||||||
// Run a sanity check to see if the devp2p is reachable
|
// Run a sanity check to see if the devp2p is reachable
|
||||||
nodePort := infos.portmap[infos.envvars["NODE_PORT"]]
|
p2pPort := infos.portmap[infos.envvars["ETH_PORT"]+"/tcp"]
|
||||||
if err = checkPort(client.server, nodePort); err != nil {
|
if err = checkPort(host, p2pPort); err != nil {
|
||||||
log.Warn(fmt.Sprintf("Explorer devp2p port seems unreachable"), "server", client.server, "port", nodePort, "err", err)
|
log.Warn("Explorer node seems unreachable", "server", host, "port", p2pPort, "err", err)
|
||||||
|
}
|
||||||
|
if err = checkPort(host, port); err != nil {
|
||||||
|
log.Warn("Explorer service seems unreachable", "server", host, "port", port, "err", err)
|
||||||
}
|
}
|
||||||
// Assemble and return the useful infos
|
// Assemble and return the useful infos
|
||||||
stats := &explorerInfos{
|
stats := &explorerInfos{
|
||||||
datadir: infos.volumes["/root/.local/share/io.parity.ethereum"],
|
node: &nodeInfos{
|
||||||
nodePort: nodePort,
|
datadir: infos.volumes["/opt/app/.ethereum"],
|
||||||
webHost: host,
|
port: infos.portmap[infos.envvars["ETH_PORT"]+"/tcp"],
|
||||||
webPort: webPort,
|
ethstats: infos.envvars["ETH_NAME"],
|
||||||
ethstats: infos.envvars["STATS"],
|
},
|
||||||
|
dbdir: infos.volumes["/var/lib/postgresql/data"],
|
||||||
|
host: host,
|
||||||
|
port: port,
|
||||||
}
|
}
|
||||||
return stats, nil
|
return stats, nil
|
||||||
}
|
}
|
||||||
|
@ -77,7 +77,7 @@ func (w *wizard) deployDashboard() {
|
|||||||
}
|
}
|
||||||
case "explorer":
|
case "explorer":
|
||||||
if infos, err := checkExplorer(client, w.network); err == nil {
|
if infos, err := checkExplorer(client, w.network); err == nil {
|
||||||
port = infos.webPort
|
port = infos.port
|
||||||
}
|
}
|
||||||
case "wallet":
|
case "wallet":
|
||||||
if infos, err := checkWallet(client, w.network); err == nil {
|
if infos, err := checkWallet(client, w.network); err == nil {
|
||||||
|
@ -35,10 +35,6 @@ func (w *wizard) deployExplorer() {
|
|||||||
log.Error("No ethstats server configured")
|
log.Error("No ethstats server configured")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if w.conf.Genesis.Config.Ethash == nil {
|
|
||||||
log.Error("Only ethash network supported")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
// Select the server to interact with
|
// Select the server to interact with
|
||||||
server := w.selectServer()
|
server := w.selectServer()
|
||||||
if server == "" {
|
if server == "" {
|
||||||
@ -50,50 +46,57 @@ func (w *wizard) deployExplorer() {
|
|||||||
infos, err := checkExplorer(client, w.network)
|
infos, err := checkExplorer(client, w.network)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
infos = &explorerInfos{
|
infos = &explorerInfos{
|
||||||
nodePort: 30303, webPort: 80, webHost: client.server,
|
node: &nodeInfos{port: 30303},
|
||||||
|
port: 80,
|
||||||
|
host: client.server,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
existed := err == nil
|
existed := err == nil
|
||||||
|
|
||||||
chainspec, err := newParityChainSpec(w.network, w.conf.Genesis, w.conf.bootnodes)
|
infos.node.genesis, _ = json.MarshalIndent(w.conf.Genesis, "", " ")
|
||||||
if err != nil {
|
infos.node.network = w.conf.Genesis.Config.ChainID.Int64()
|
||||||
log.Error("Failed to create chain spec for explorer", "err", err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
chain, _ := json.MarshalIndent(chainspec, "", " ")
|
|
||||||
|
|
||||||
// Figure out which port to listen on
|
// Figure out which port to listen on
|
||||||
fmt.Println()
|
fmt.Println()
|
||||||
fmt.Printf("Which port should the explorer listen on? (default = %d)\n", infos.webPort)
|
fmt.Printf("Which port should the explorer listen on? (default = %d)\n", infos.port)
|
||||||
infos.webPort = w.readDefaultInt(infos.webPort)
|
infos.port = w.readDefaultInt(infos.port)
|
||||||
|
|
||||||
// Figure which virtual-host to deploy ethstats on
|
// Figure which virtual-host to deploy ethstats on
|
||||||
if infos.webHost, err = w.ensureVirtualHost(client, infos.webPort, infos.webHost); err != nil {
|
if infos.host, err = w.ensureVirtualHost(client, infos.port, infos.host); err != nil {
|
||||||
log.Error("Failed to decide on explorer host", "err", err)
|
log.Error("Failed to decide on explorer host", "err", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
// Figure out where the user wants to store the persistent data
|
// Figure out where the user wants to store the persistent data
|
||||||
fmt.Println()
|
fmt.Println()
|
||||||
if infos.datadir == "" {
|
if infos.node.datadir == "" {
|
||||||
fmt.Printf("Where should data be stored on the remote machine?\n")
|
fmt.Printf("Where should node data be stored on the remote machine?\n")
|
||||||
infos.datadir = w.readString()
|
infos.node.datadir = w.readString()
|
||||||
} else {
|
} else {
|
||||||
fmt.Printf("Where should data be stored on the remote machine? (default = %s)\n", infos.datadir)
|
fmt.Printf("Where should node data be stored on the remote machine? (default = %s)\n", infos.node.datadir)
|
||||||
infos.datadir = w.readDefaultString(infos.datadir)
|
infos.node.datadir = w.readDefaultString(infos.node.datadir)
|
||||||
|
}
|
||||||
|
// Figure out where the user wants to store the persistent data for backend database
|
||||||
|
fmt.Println()
|
||||||
|
if infos.dbdir == "" {
|
||||||
|
fmt.Printf("Where should postgres data be stored on the remote machine?\n")
|
||||||
|
infos.dbdir = w.readString()
|
||||||
|
} else {
|
||||||
|
fmt.Printf("Where should postgres data be stored on the remote machine? (default = %s)\n", infos.dbdir)
|
||||||
|
infos.dbdir = w.readDefaultString(infos.dbdir)
|
||||||
}
|
}
|
||||||
// Figure out which port to listen on
|
// Figure out which port to listen on
|
||||||
fmt.Println()
|
fmt.Println()
|
||||||
fmt.Printf("Which TCP/UDP port should the archive node listen on? (default = %d)\n", infos.nodePort)
|
fmt.Printf("Which TCP/UDP port should the archive node listen on? (default = %d)\n", infos.node.port)
|
||||||
infos.nodePort = w.readDefaultInt(infos.nodePort)
|
infos.node.port = w.readDefaultInt(infos.node.port)
|
||||||
|
|
||||||
// Set a proper name to report on the stats page
|
// Set a proper name to report on the stats page
|
||||||
fmt.Println()
|
fmt.Println()
|
||||||
if infos.ethstats == "" {
|
if infos.node.ethstats == "" {
|
||||||
fmt.Printf("What should the explorer be called on the stats page?\n")
|
fmt.Printf("What should the explorer be called on the stats page?\n")
|
||||||
infos.ethstats = w.readString() + ":" + w.conf.ethstats
|
infos.node.ethstats = w.readString() + ":" + w.conf.ethstats
|
||||||
} else {
|
} else {
|
||||||
fmt.Printf("What should the explorer be called on the stats page? (default = %s)\n", infos.ethstats)
|
fmt.Printf("What should the explorer be called on the stats page? (default = %s)\n", infos.node.ethstats)
|
||||||
infos.ethstats = w.readDefaultString(infos.ethstats) + ":" + w.conf.ethstats
|
infos.node.ethstats = w.readDefaultString(infos.node.ethstats) + ":" + w.conf.ethstats
|
||||||
}
|
}
|
||||||
// Try to deploy the explorer on the host
|
// Try to deploy the explorer on the host
|
||||||
nocache := false
|
nocache := false
|
||||||
@ -102,7 +105,7 @@ func (w *wizard) deployExplorer() {
|
|||||||
fmt.Printf("Should the explorer be built from scratch (y/n)? (default = no)\n")
|
fmt.Printf("Should the explorer be built from scratch (y/n)? (default = no)\n")
|
||||||
nocache = w.readDefaultYesNo(false)
|
nocache = w.readDefaultYesNo(false)
|
||||||
}
|
}
|
||||||
if out, err := deployExplorer(client, w.network, chain, infos, nocache); err != nil {
|
if out, err := deployExplorer(client, w.network, w.conf.bootnodes, infos, nocache, w.conf.Genesis.Config.Clique != nil); err != nil {
|
||||||
log.Error("Failed to deploy explorer container", "err", err)
|
log.Error("Failed to deploy explorer container", "err", err)
|
||||||
if len(out) > 0 {
|
if len(out) > 0 {
|
||||||
fmt.Printf("%s\n", out)
|
fmt.Printf("%s\n", out)
|
||||||
|
@ -174,7 +174,7 @@ func (w *wizard) deployComponent() {
|
|||||||
fmt.Println(" 1. Ethstats - Network monitoring tool")
|
fmt.Println(" 1. Ethstats - Network monitoring tool")
|
||||||
fmt.Println(" 2. Bootnode - Entry point of the network")
|
fmt.Println(" 2. Bootnode - Entry point of the network")
|
||||||
fmt.Println(" 3. Sealer - Full node minting new blocks")
|
fmt.Println(" 3. Sealer - Full node minting new blocks")
|
||||||
fmt.Println(" 4. Explorer - Chain analysis webservice (ethash only)")
|
fmt.Println(" 4. Explorer - Chain analysis webservice")
|
||||||
fmt.Println(" 5. Wallet - Browser wallet for quick sends")
|
fmt.Println(" 5. Wallet - Browser wallet for quick sends")
|
||||||
fmt.Println(" 6. Faucet - Crypto faucet to give away funds")
|
fmt.Println(" 6. Faucet - Crypto faucet to give away funds")
|
||||||
fmt.Println(" 7. Dashboard - Website listing above web-services")
|
fmt.Println(" 7. Dashboard - Website listing above web-services")
|
||||||
|
Loading…
Reference in New Issue
Block a user