Patch for concurrent iterator & others (onto v1.11.6) #386
@ -80,12 +80,10 @@ var dashboardContent = `
|
|||||||
<ul class="nav side-menu">
|
<ul class="nav side-menu">
|
||||||
{{if .EthstatsPage}}<li id="stats_menu"><a onclick="load('#stats')"><i class="fa fa-tachometer"></i> Network Stats</a></li>{{end}}
|
{{if .EthstatsPage}}<li id="stats_menu"><a onclick="load('#stats')"><i class="fa fa-tachometer"></i> Network Stats</a></li>{{end}}
|
||||||
{{if .ExplorerPage}}<li id="explorer_menu"><a onclick="load('#explorer')"><i class="fa fa-database"></i> Block Explorer</a></li>{{end}}
|
{{if .ExplorerPage}}<li id="explorer_menu"><a onclick="load('#explorer')"><i class="fa fa-database"></i> Block Explorer</a></li>{{end}}
|
||||||
{{if .WalletPage}}<li id="wallet_menu"><a onclick="load('#wallet')"><i class="fa fa-address-book-o"></i> Browser Wallet</a></li>{{end}}
|
|
||||||
{{if .FaucetPage}}<li id="faucet_menu"><a onclick="load('#faucet')"><i class="fa fa-bath"></i> Crypto Faucet</a></li>{{end}}
|
{{if .FaucetPage}}<li id="faucet_menu"><a onclick="load('#faucet')"><i class="fa fa-bath"></i> Crypto Faucet</a></li>{{end}}
|
||||||
<li id="connect_menu"><a><i class="fa fa-plug"></i> Connect Yourself</a>
|
<li id="connect_menu"><a><i class="fa fa-plug"></i> Connect Yourself</a>
|
||||||
<ul id="connect_list" class="nav child_menu">
|
<ul id="connect_list" class="nav child_menu">
|
||||||
<li><a onclick="$('#connect_menu').removeClass('active'); $('#connect_list').toggle(); load('#geth')">Go Ethereum: Geth</a></li>
|
<li><a onclick="$('#connect_menu').removeClass('active'); $('#connect_list').toggle(); load('#geth')">Go Ethereum: Geth</a></li>
|
||||||
<li><a onclick="$('#connect_menu').removeClass('active'); $('#connect_list').toggle(); load('#mist')">Go Ethereum: Wallet & Mist</a></li>
|
|
||||||
<li><a onclick="$('#connect_menu').removeClass('active'); $('#connect_list').toggle(); load('#mobile')">Go Ethereum: Android & iOS</a></li>{{if .Ethash}}
|
<li><a onclick="$('#connect_menu').removeClass('active'); $('#connect_list').toggle(); load('#mobile')">Go Ethereum: Android & iOS</a></li>{{if .Ethash}}
|
||||||
<li><a onclick="$('#connect_menu').removeClass('active'); $('#connect_list').toggle(); load('#other')">Other Ethereum Clients</a></li>{{end}}
|
<li><a onclick="$('#connect_menu').removeClass('active'); $('#connect_list').toggle(); load('#other')">Other Ethereum Clients</a></li>{{end}}
|
||||||
</ul>
|
</ul>
|
||||||
@ -186,58 +184,6 @@ var dashboardContent = `
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div id="mist" hidden style="padding: 16px;">
|
|
||||||
<div class="page-title">
|
|
||||||
<div class="title_left">
|
|
||||||
<h3>Connect Yourself – Go Ethereum: Wallet & Mist</h3>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="clearfix"></div>
|
|
||||||
<div class="row">
|
|
||||||
<div class="col-md-6">
|
|
||||||
<div class="x_panel">
|
|
||||||
<div class="x_title">
|
|
||||||
<h2><i class="fa fa-credit-card" aria-hidden="true"></i> Desktop wallet <small>Interacts with accounts and contracts</small></h2>
|
|
||||||
<div class="clearfix"></div>
|
|
||||||
</div>
|
|
||||||
<div class="x_content">
|
|
||||||
<p>The Ethereum Wallet is an <a href="https://electron.atom.io/" target="about:blank">Electron</a> based desktop application to manage your Ethereum accounts and funds. Beside the usual account life-cycle operations you would expect to perform, the wallet also provides a means to send transactions from your accounts and to interact with smart contracts deployed on the network.</p>
|
|
||||||
<p>Under the hood the wallet is backed by a go-ethereum full node, meaning that a mid range machine is assumed. Similarly, synchronization is based on <strong>fast-sync</strong>, which will download all blockchain data from the network and make it available to the wallet. Light nodes cannot currently fully back the wallet, but it's a target actively pursued.</p>
|
|
||||||
<br/>
|
|
||||||
<p>To connect with the Ethereum Wallet, you'll need to initialize your private network first via Geth as the wallet does not currently support calling Geth directly. To initialize your local chain, download <a href="/{{.GethGenesis}}"><code>{{.GethGenesis}}</code></a> and run:
|
|
||||||
<pre>geth --datadir=$HOME/.{{.Network}} init {{.GethGenesis}}</pre>
|
|
||||||
</p>
|
|
||||||
<p>With your local chain initialized, you can start the Ethereum Wallet:
|
|
||||||
<pre>ethereumwallet --rpc $HOME/.{{.Network}}/geth.ipc --node-networkid={{.NetworkID}} --node-datadir=$HOME/.{{.Network}}{{if .Ethstats}} --node-ethstats='{{.Ethstats}}'{{end}} --node-bootnodes={{.BootnodesFlat}}</pre>
|
|
||||||
<p>
|
|
||||||
<br/>
|
|
||||||
<p>You can download the Ethereum Wallet from <a href="https://github.com/ethereum/mist/releases" target="about:blank">https://github.com/ethereum/mist/releases</a>.</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="col-md-6">
|
|
||||||
<div class="x_panel">
|
|
||||||
<div class="x_title">
|
|
||||||
<h2><i class="fa fa-picture-o" aria-hidden="true"></i> Mist browser <small>Interacts with third party DApps</small></h2>
|
|
||||||
<div class="clearfix"></div>
|
|
||||||
</div>
|
|
||||||
<div class="x_content">
|
|
||||||
<p>The Mist browser is an <a href="https://electron.atom.io/" target="about:blank">Electron</a> based desktop application to load and interact with Ethereum enabled third party web DApps. Beside all the functionality provided by the Ethereum Wallet, Mist is an extended web-browser where loaded pages have access to the Ethereum network via a web3.js provider, and may also interact with users' own accounts (given proper authorization and confirmation of course).</p>
|
|
||||||
<p>Under the hood the browser is backed by a go-ethereum full node, meaning that a mid range machine is assumed. Similarly, synchronization is based on <strong>fast-sync</strong>, which will download all blockchain data from the network and make it available to the wallet. Light nodes cannot currently fully back the wallet, but it's a target actively pursued.</p>
|
|
||||||
<br/>
|
|
||||||
<p>To connect with the Mist browser, you'll need to initialize your private network first via Geth as Mist does not currently support calling Geth directly. To initialize your local chain, download <a href="/{{.GethGenesis}}"><code>{{.GethGenesis}}</code></a> and run:
|
|
||||||
<pre>geth --datadir=$HOME/.{{.Network}} init {{.GethGenesis}}</pre>
|
|
||||||
</p>
|
|
||||||
<p>With your local chain initialized, you can start Mist:
|
|
||||||
<pre>mist --rpc $HOME/.{{.Network}}/geth.ipc --node-networkid={{.NetworkID}} --node-datadir=$HOME/.{{.Network}}{{if .Ethstats}} --node-ethstats='{{.Ethstats}}'{{end}} --node-bootnodes={{.BootnodesFlat}}</pre>
|
|
||||||
<p>
|
|
||||||
<br/>
|
|
||||||
<p>You can download the Mist browser from <a href="https://github.com/ethereum/mist/releases" target="about:blank">https://github.com/ethereum/mist/releases</a>.</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div id="mobile" hidden style="padding: 16px;">
|
<div id="mobile" hidden style="padding: 16px;">
|
||||||
<div class="page-title">
|
<div class="page-title">
|
||||||
<div class="title_left">
|
<div class="title_left">
|
||||||
@ -416,7 +362,7 @@ try! node?.start();
|
|||||||
<div class="clearfix"></div>
|
<div class="clearfix"></div>
|
||||||
</div>
|
</div>
|
||||||
<div style="display: inline-block; vertical-align: bottom; width: 623px; margin-top: 16px;">
|
<div style="display: inline-block; vertical-align: bottom; width: 623px; margin-top: 16px;">
|
||||||
<p>Puppeth is a tool to aid you in creating a new Ethereum network down to the genesis block, bootnodes, signers, ethstats server, crypto faucet, wallet browsers, block explorer, dashboard and more; without the hassle that it would normally entail to manually configure all these services one by one.</p>
|
<p>Puppeth is a tool to aid you in creating a new Ethereum network down to the genesis block, bootnodes, signers, ethstats server, crypto faucet, block explorer, dashboard and more; without the hassle that it would normally entail to manually configure all these services one by one.</p>
|
||||||
<p>Puppeth uses ssh to dial in to remote servers, and builds its network components out of docker containers using docker-compose. The user is guided through the process via a command line wizard that does the heavy lifting and topology configuration automatically behind the scenes.</p>
|
<p>Puppeth uses ssh to dial in to remote servers, and builds its network components out of docker containers using docker-compose. The user is guided through the process via a command line wizard that does the heavy lifting and topology configuration automatically behind the scenes.</p>
|
||||||
<br/>
|
<br/>
|
||||||
<p>Puppeth is distributed as part of the <a href="https://geth.ethereum.org/downloads/" target="about:blank">Geth & Tools</a> bundles, but can also be installed separately via:<pre>go get github.com/ethereum/go-ethereum/cmd/puppeth</pre></p>
|
<p>Puppeth is distributed as part of the <a href="https://geth.ethereum.org/downloads/" target="about:blank">Geth & Tools</a> bundles, but can also be installed separately via:<pre>go get github.com/ethereum/go-ethereum/cmd/puppeth</pre></p>
|
||||||
@ -461,9 +407,6 @@ try! node?.start();
|
|||||||
case "#explorer":
|
case "#explorer":
|
||||||
url = "//{{.ExplorerPage}}";
|
url = "//{{.ExplorerPage}}";
|
||||||
break;
|
break;
|
||||||
case "#wallet":
|
|
||||||
url = "//{{.WalletPage}}";
|
|
||||||
break;
|
|
||||||
case "#faucet":
|
case "#faucet":
|
||||||
url = "//{{.FaucetPage}}";
|
url = "//{{.FaucetPage}}";
|
||||||
break;
|
break;
|
||||||
@ -587,7 +530,6 @@ func deployDashboard(client *sshClient, network string, conf *config, config *da
|
|||||||
"VHost": config.host,
|
"VHost": config.host,
|
||||||
"EthstatsPage": config.ethstats,
|
"EthstatsPage": config.ethstats,
|
||||||
"ExplorerPage": config.explorer,
|
"ExplorerPage": config.explorer,
|
||||||
"WalletPage": config.wallet,
|
|
||||||
"FaucetPage": config.faucet,
|
"FaucetPage": config.faucet,
|
||||||
})
|
})
|
||||||
files[filepath.Join(workdir, "docker-compose.yaml")] = composefile.Bytes()
|
files[filepath.Join(workdir, "docker-compose.yaml")] = composefile.Bytes()
|
||||||
@ -615,7 +557,6 @@ func deployDashboard(client *sshClient, network string, conf *config, config *da
|
|||||||
"NetworkTitle": strings.Title(network),
|
"NetworkTitle": strings.Title(network),
|
||||||
"EthstatsPage": config.ethstats,
|
"EthstatsPage": config.ethstats,
|
||||||
"ExplorerPage": config.explorer,
|
"ExplorerPage": config.explorer,
|
||||||
"WalletPage": config.wallet,
|
|
||||||
"FaucetPage": config.faucet,
|
"FaucetPage": config.faucet,
|
||||||
"GethGenesis": network + ".json",
|
"GethGenesis": network + ".json",
|
||||||
"Bootnodes": conf.bootnodes,
|
"Bootnodes": conf.bootnodes,
|
||||||
@ -695,7 +636,6 @@ type dashboardInfos struct {
|
|||||||
|
|
||||||
ethstats string
|
ethstats string
|
||||||
explorer string
|
explorer string
|
||||||
wallet string
|
|
||||||
faucet string
|
faucet string
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -707,7 +647,6 @@ func (info *dashboardInfos) Report() map[string]string {
|
|||||||
"Website listener port": strconv.Itoa(info.port),
|
"Website listener port": strconv.Itoa(info.port),
|
||||||
"Ethstats service": info.ethstats,
|
"Ethstats service": info.ethstats,
|
||||||
"Explorer service": info.explorer,
|
"Explorer service": info.explorer,
|
||||||
"Wallet service": info.wallet,
|
|
||||||
"Faucet service": info.faucet,
|
"Faucet service": info.faucet,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -748,7 +687,6 @@ func checkDashboard(client *sshClient, network string) (*dashboardInfos, error)
|
|||||||
port: port,
|
port: port,
|
||||||
ethstats: infos.envvars["ETHSTATS_PAGE"],
|
ethstats: infos.envvars["ETHSTATS_PAGE"],
|
||||||
explorer: infos.envvars["EXPLORER_PAGE"],
|
explorer: infos.envvars["EXPLORER_PAGE"],
|
||||||
wallet: infos.envvars["WALLET_PAGE"],
|
|
||||||
faucet: infos.envvars["FAUCET_PAGE"],
|
faucet: infos.envvars["FAUCET_PAGE"],
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
@ -1,201 +0,0 @@
|
|||||||
// Copyright 2017 The go-ethereum Authors
|
|
||||||
// This file is part of go-ethereum.
|
|
||||||
//
|
|
||||||
// go-ethereum is free software: you can redistribute it and/or modify
|
|
||||||
// it under the terms of the GNU General Public License as published by
|
|
||||||
// the Free Software Foundation, either version 3 of the License, or
|
|
||||||
// (at your option) any later version.
|
|
||||||
//
|
|
||||||
// go-ethereum is distributed in the hope that it will be useful,
|
|
||||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
// GNU General Public License for more details.
|
|
||||||
//
|
|
||||||
// You should have received a copy of the GNU General Public License
|
|
||||||
// along with go-ethereum. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
|
|
||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"fmt"
|
|
||||||
"html/template"
|
|
||||||
"math/rand"
|
|
||||||
"path/filepath"
|
|
||||||
"strconv"
|
|
||||||
"strings"
|
|
||||||
|
|
||||||
"github.com/ethereum/go-ethereum/log"
|
|
||||||
)
|
|
||||||
|
|
||||||
// walletDockerfile is the Dockerfile required to run a web wallet.
|
|
||||||
var walletDockerfile = `
|
|
||||||
FROM puppeth/wallet:latest
|
|
||||||
|
|
||||||
ADD genesis.json /genesis.json
|
|
||||||
|
|
||||||
RUN \
|
|
||||||
echo 'node server.js &' > wallet.sh && \
|
|
||||||
echo 'geth --cache 512 init /genesis.json' >> wallet.sh && \
|
|
||||||
echo $'exec geth --networkid {{.NetworkID}} --port {{.NodePort}} --bootnodes {{.Bootnodes}} --ethstats \'{{.Ethstats}}\' --cache=512 --http --http.addr=0.0.0.0 --http.corsdomain "*" --http.vhosts "*"' >> wallet.sh
|
|
||||||
|
|
||||||
RUN \
|
|
||||||
sed -i 's/PuppethNetworkID/{{.NetworkID}}/g' dist/js/etherwallet-master.js && \
|
|
||||||
sed -i 's/PuppethNetwork/{{.Network}}/g' dist/js/etherwallet-master.js && \
|
|
||||||
sed -i 's/PuppethDenom/{{.Denom}}/g' dist/js/etherwallet-master.js && \
|
|
||||||
sed -i 's/PuppethHost/{{.Host}}/g' dist/js/etherwallet-master.js && \
|
|
||||||
sed -i 's/PuppethRPCPort/{{.RPCPort}}/g' dist/js/etherwallet-master.js
|
|
||||||
|
|
||||||
ENTRYPOINT ["/bin/sh", "wallet.sh"]
|
|
||||||
`
|
|
||||||
|
|
||||||
// walletComposefile is the docker-compose.yml file required to deploy and
|
|
||||||
// maintain a web wallet.
|
|
||||||
var walletComposefile = `
|
|
||||||
version: '2'
|
|
||||||
services:
|
|
||||||
wallet:
|
|
||||||
build: .
|
|
||||||
image: {{.Network}}/wallet
|
|
||||||
container_name: {{.Network}}_wallet_1
|
|
||||||
ports:
|
|
||||||
- "{{.NodePort}}:{{.NodePort}}"
|
|
||||||
- "{{.NodePort}}:{{.NodePort}}/udp"
|
|
||||||
- "{{.RPCPort}}:8545"{{if not .VHost}}
|
|
||||||
- "{{.WebPort}}:80"{{end}}
|
|
||||||
volumes:
|
|
||||||
- {{.Datadir}}:/root/.ethereum
|
|
||||||
environment:
|
|
||||||
- NODE_PORT={{.NodePort}}/tcp
|
|
||||||
- STATS={{.Ethstats}}{{if .VHost}}
|
|
||||||
- VIRTUAL_HOST={{.VHost}}
|
|
||||||
- VIRTUAL_PORT=80{{end}}
|
|
||||||
logging:
|
|
||||||
driver: "json-file"
|
|
||||||
options:
|
|
||||||
max-size: "1m"
|
|
||||||
max-file: "10"
|
|
||||||
restart: always
|
|
||||||
`
|
|
||||||
|
|
||||||
// deployWallet deploys a new web wallet container to a remote machine via SSH,
|
|
||||||
// docker and docker-compose. If an instance with the specified network name
|
|
||||||
// already exists there, it will be overwritten!
|
|
||||||
func deployWallet(client *sshClient, network string, bootnodes []string, config *walletInfos, nocache bool) ([]byte, error) {
|
|
||||||
// Generate the content to upload to the server
|
|
||||||
workdir := fmt.Sprintf("%d", rand.Int63())
|
|
||||||
files := make(map[string][]byte)
|
|
||||||
|
|
||||||
dockerfile := new(bytes.Buffer)
|
|
||||||
template.Must(template.New("").Parse(walletDockerfile)).Execute(dockerfile, map[string]interface{}{
|
|
||||||
"Network": strings.ToTitle(network),
|
|
||||||
"Denom": strings.ToUpper(network),
|
|
||||||
"NetworkID": config.network,
|
|
||||||
"NodePort": config.nodePort,
|
|
||||||
"RPCPort": config.rpcPort,
|
|
||||||
"Bootnodes": strings.Join(bootnodes, ","),
|
|
||||||
"Ethstats": config.ethstats,
|
|
||||||
"Host": client.address,
|
|
||||||
})
|
|
||||||
files[filepath.Join(workdir, "Dockerfile")] = dockerfile.Bytes()
|
|
||||||
|
|
||||||
composefile := new(bytes.Buffer)
|
|
||||||
template.Must(template.New("").Parse(walletComposefile)).Execute(composefile, map[string]interface{}{
|
|
||||||
"Datadir": config.datadir,
|
|
||||||
"Network": network,
|
|
||||||
"NodePort": config.nodePort,
|
|
||||||
"RPCPort": config.rpcPort,
|
|
||||||
"VHost": config.webHost,
|
|
||||||
"WebPort": config.webPort,
|
|
||||||
"Ethstats": config.ethstats[:strings.Index(config.ethstats, ":")],
|
|
||||||
})
|
|
||||||
files[filepath.Join(workdir, "docker-compose.yaml")] = composefile.Bytes()
|
|
||||||
|
|
||||||
files[filepath.Join(workdir, "genesis.json")] = config.genesis
|
|
||||||
|
|
||||||
// Upload the deployment files to the remote server (and clean up afterwards)
|
|
||||||
if out, err := client.Upload(files); err != nil {
|
|
||||||
return out, err
|
|
||||||
}
|
|
||||||
defer client.Run("rm -rf " + workdir)
|
|
||||||
|
|
||||||
// Build and deploy the boot or seal node service
|
|
||||||
if nocache {
|
|
||||||
return nil, client.Stream(fmt.Sprintf("cd %s && docker-compose -p %s build --pull --no-cache && docker-compose -p %s up -d --force-recreate --timeout 60", workdir, network, network))
|
|
||||||
}
|
|
||||||
return nil, client.Stream(fmt.Sprintf("cd %s && docker-compose -p %s up -d --build --force-recreate --timeout 60", workdir, network))
|
|
||||||
}
|
|
||||||
|
|
||||||
// walletInfos is returned from a web wallet status check to allow reporting
|
|
||||||
// various configuration parameters.
|
|
||||||
type walletInfos struct {
|
|
||||||
genesis []byte
|
|
||||||
network int64
|
|
||||||
datadir string
|
|
||||||
ethstats string
|
|
||||||
nodePort int
|
|
||||||
rpcPort int
|
|
||||||
webHost string
|
|
||||||
webPort int
|
|
||||||
}
|
|
||||||
|
|
||||||
// Report converts the typed struct into a plain string->string map, containing
|
|
||||||
// most - but not all - fields for reporting to the user.
|
|
||||||
func (info *walletInfos) Report() map[string]string {
|
|
||||||
report := map[string]string{
|
|
||||||
"Data directory": info.datadir,
|
|
||||||
"Ethstats username": info.ethstats,
|
|
||||||
"Node listener port ": strconv.Itoa(info.nodePort),
|
|
||||||
"RPC listener port ": strconv.Itoa(info.rpcPort),
|
|
||||||
"Website address ": info.webHost,
|
|
||||||
"Website listener port ": strconv.Itoa(info.webPort),
|
|
||||||
}
|
|
||||||
return report
|
|
||||||
}
|
|
||||||
|
|
||||||
// checkWallet does a health-check against web wallet server to verify whether
|
|
||||||
// it's running, and if yes, whether it's responsive.
|
|
||||||
func checkWallet(client *sshClient, network string) (*walletInfos, error) {
|
|
||||||
// Inspect a possible web wallet container on the host
|
|
||||||
infos, err := inspectContainer(client, fmt.Sprintf("%s_wallet_1", network))
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if !infos.running {
|
|
||||||
return nil, ErrServiceOffline
|
|
||||||
}
|
|
||||||
// Resolve the port from the host, or the reverse proxy
|
|
||||||
webPort := infos.portmap["80/tcp"]
|
|
||||||
if webPort == 0 {
|
|
||||||
if proxy, _ := checkNginx(client, network); proxy != nil {
|
|
||||||
webPort = proxy.port
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if webPort == 0 {
|
|
||||||
return nil, ErrNotExposed
|
|
||||||
}
|
|
||||||
// Resolve the host from the reverse-proxy and the config values
|
|
||||||
host := infos.envvars["VIRTUAL_HOST"]
|
|
||||||
if host == "" {
|
|
||||||
host = client.server
|
|
||||||
}
|
|
||||||
// Run a sanity check to see if the devp2p and RPC ports are reachable
|
|
||||||
nodePort := infos.portmap[infos.envvars["NODE_PORT"]]
|
|
||||||
if err = checkPort(client.server, nodePort); err != nil {
|
|
||||||
log.Warn("Wallet devp2p port seems unreachable", "server", client.server, "port", nodePort, "err", err)
|
|
||||||
}
|
|
||||||
rpcPort := infos.portmap["8545/tcp"]
|
|
||||||
if err = checkPort(client.server, rpcPort); err != nil {
|
|
||||||
log.Warn("Wallet RPC port seems unreachable", "server", client.server, "port", rpcPort, "err", err)
|
|
||||||
}
|
|
||||||
// Assemble and return the useful infos
|
|
||||||
stats := &walletInfos{
|
|
||||||
datadir: infos.volumes["/root/.ethereum"],
|
|
||||||
nodePort: nodePort,
|
|
||||||
rpcPort: rpcPort,
|
|
||||||
webHost: host,
|
|
||||||
webPort: webPort,
|
|
||||||
ethstats: infos.envvars["STATS"],
|
|
||||||
}
|
|
||||||
return stats, nil
|
|
||||||
}
|
|
@ -60,7 +60,7 @@ func (w *wizard) deployDashboard() {
|
|||||||
available[service] = append(available[service], server)
|
available[service] = append(available[service], server)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
for _, service := range []string{"ethstats", "explorer", "wallet", "faucet"} {
|
for _, service := range []string{"ethstats", "explorer", "faucet"} {
|
||||||
// Gather all the locally hosted pages of this type
|
// Gather all the locally hosted pages of this type
|
||||||
var pages []string
|
var pages []string
|
||||||
for _, server := range available[service] {
|
for _, server := range available[service] {
|
||||||
@ -79,10 +79,6 @@ func (w *wizard) deployDashboard() {
|
|||||||
if infos, err := checkExplorer(client, w.network); err == nil {
|
if infos, err := checkExplorer(client, w.network); err == nil {
|
||||||
port = infos.port
|
port = infos.port
|
||||||
}
|
}
|
||||||
case "wallet":
|
|
||||||
if infos, err := checkWallet(client, w.network); err == nil {
|
|
||||||
port = infos.webPort
|
|
||||||
}
|
|
||||||
case "faucet":
|
case "faucet":
|
||||||
if infos, err := checkFaucet(client, w.network); err == nil {
|
if infos, err := checkFaucet(client, w.network); err == nil {
|
||||||
port = infos.port
|
port = infos.port
|
||||||
@ -127,8 +123,6 @@ func (w *wizard) deployDashboard() {
|
|||||||
infos.ethstats = page
|
infos.ethstats = page
|
||||||
case "explorer":
|
case "explorer":
|
||||||
infos.explorer = page
|
infos.explorer = page
|
||||||
case "wallet":
|
|
||||||
infos.wallet = page
|
|
||||||
case "faucet":
|
case "faucet":
|
||||||
infos.faucet = page
|
infos.faucet = page
|
||||||
}
|
}
|
||||||
|
@ -141,14 +141,6 @@ func (w *wizard) gatherStats(server string, pubkey []byte, client *sshClient) *s
|
|||||||
} else {
|
} else {
|
||||||
stat.services["explorer"] = infos.Report()
|
stat.services["explorer"] = infos.Report()
|
||||||
}
|
}
|
||||||
logger.Debug("Checking for wallet availability")
|
|
||||||
if infos, err := checkWallet(client, w.network); err != nil {
|
|
||||||
if err != ErrServiceUnknown {
|
|
||||||
stat.services["wallet"] = map[string]string{"offline": err.Error()}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
stat.services["wallet"] = infos.Report()
|
|
||||||
}
|
|
||||||
logger.Debug("Checking for faucet availability")
|
logger.Debug("Checking for faucet availability")
|
||||||
if infos, err := checkFaucet(client, w.network); err != nil {
|
if infos, err := checkFaucet(client, w.network); err != nil {
|
||||||
if err != ErrServiceUnknown {
|
if err != ErrServiceUnknown {
|
||||||
|
@ -175,9 +175,8 @@ func (w *wizard) deployComponent() {
|
|||||||
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")
|
fmt.Println(" 4. Explorer - Chain analysis webservice")
|
||||||
fmt.Println(" 5. Wallet - Browser wallet for quick sends")
|
fmt.Println(" 5. Faucet - Crypto faucet to give away funds")
|
||||||
fmt.Println(" 6. Faucet - Crypto faucet to give away funds")
|
fmt.Println(" 6. Dashboard - Website listing above web-services")
|
||||||
fmt.Println(" 7. Dashboard - Website listing above web-services")
|
|
||||||
|
|
||||||
switch w.read() {
|
switch w.read() {
|
||||||
case "1":
|
case "1":
|
||||||
@ -189,10 +188,8 @@ func (w *wizard) deployComponent() {
|
|||||||
case "4":
|
case "4":
|
||||||
w.deployExplorer()
|
w.deployExplorer()
|
||||||
case "5":
|
case "5":
|
||||||
w.deployWallet()
|
|
||||||
case "6":
|
|
||||||
w.deployFaucet()
|
w.deployFaucet()
|
||||||
case "7":
|
case "6":
|
||||||
w.deployDashboard()
|
w.deployDashboard()
|
||||||
default:
|
default:
|
||||||
log.Error("That's not something I can do")
|
log.Error("That's not something I can do")
|
||||||
|
@ -1,113 +0,0 @@
|
|||||||
// Copyright 2017 The go-ethereum Authors
|
|
||||||
// This file is part of go-ethereum.
|
|
||||||
//
|
|
||||||
// go-ethereum is free software: you can redistribute it and/or modify
|
|
||||||
// it under the terms of the GNU General Public License as published by
|
|
||||||
// the Free Software Foundation, either version 3 of the License, or
|
|
||||||
// (at your option) any later version.
|
|
||||||
//
|
|
||||||
// go-ethereum is distributed in the hope that it will be useful,
|
|
||||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
// GNU General Public License for more details.
|
|
||||||
//
|
|
||||||
// You should have received a copy of the GNU General Public License
|
|
||||||
// along with go-ethereum. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
|
|
||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"encoding/json"
|
|
||||||
"fmt"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
"github.com/ethereum/go-ethereum/log"
|
|
||||||
)
|
|
||||||
|
|
||||||
// deployWallet creates a new web wallet based on some user input.
|
|
||||||
func (w *wizard) deployWallet() {
|
|
||||||
// Do some sanity check before the user wastes time on input
|
|
||||||
if w.conf.Genesis == nil {
|
|
||||||
log.Error("No genesis block configured")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if w.conf.ethstats == "" {
|
|
||||||
log.Error("No ethstats server configured")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
// Select the server to interact with
|
|
||||||
server := w.selectServer()
|
|
||||||
if server == "" {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
client := w.servers[server]
|
|
||||||
|
|
||||||
// Retrieve any active node configurations from the server
|
|
||||||
infos, err := checkWallet(client, w.network)
|
|
||||||
if err != nil {
|
|
||||||
infos = &walletInfos{
|
|
||||||
nodePort: 30303, rpcPort: 8545, webPort: 80, webHost: client.server,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
existed := err == nil
|
|
||||||
|
|
||||||
infos.genesis, _ = json.MarshalIndent(w.conf.Genesis, "", " ")
|
|
||||||
infos.network = w.conf.Genesis.Config.ChainID.Int64()
|
|
||||||
|
|
||||||
// Figure out which port to listen on
|
|
||||||
fmt.Println()
|
|
||||||
fmt.Printf("Which port should the wallet listen on? (default = %d)\n", infos.webPort)
|
|
||||||
infos.webPort = w.readDefaultInt(infos.webPort)
|
|
||||||
|
|
||||||
// Figure which virtual-host to deploy ethstats on
|
|
||||||
if infos.webHost, err = w.ensureVirtualHost(client, infos.webPort, infos.webHost); err != nil {
|
|
||||||
log.Error("Failed to decide on wallet host", "err", err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
// Figure out where the user wants to store the persistent data
|
|
||||||
fmt.Println()
|
|
||||||
if infos.datadir == "" {
|
|
||||||
fmt.Printf("Where should data be stored on the remote machine?\n")
|
|
||||||
infos.datadir = w.readString()
|
|
||||||
} else {
|
|
||||||
fmt.Printf("Where should data be stored on the remote machine? (default = %s)\n", infos.datadir)
|
|
||||||
infos.datadir = w.readDefaultString(infos.datadir)
|
|
||||||
}
|
|
||||||
// Figure out which port to listen on
|
|
||||||
fmt.Println()
|
|
||||||
fmt.Printf("Which TCP/UDP port should the backing node listen on? (default = %d)\n", infos.nodePort)
|
|
||||||
infos.nodePort = w.readDefaultInt(infos.nodePort)
|
|
||||||
|
|
||||||
fmt.Println()
|
|
||||||
fmt.Printf("Which port should the backing RPC API listen on? (default = %d)\n", infos.rpcPort)
|
|
||||||
infos.rpcPort = w.readDefaultInt(infos.rpcPort)
|
|
||||||
|
|
||||||
// Set a proper name to report on the stats page
|
|
||||||
fmt.Println()
|
|
||||||
if infos.ethstats == "" {
|
|
||||||
fmt.Printf("What should the wallet be called on the stats page?\n")
|
|
||||||
infos.ethstats = w.readString() + ":" + w.conf.ethstats
|
|
||||||
} else {
|
|
||||||
fmt.Printf("What should the wallet be called on the stats page? (default = %s)\n", infos.ethstats)
|
|
||||||
infos.ethstats = w.readDefaultString(infos.ethstats) + ":" + w.conf.ethstats
|
|
||||||
}
|
|
||||||
// Try to deploy the wallet on the host
|
|
||||||
nocache := false
|
|
||||||
if existed {
|
|
||||||
fmt.Println()
|
|
||||||
fmt.Printf("Should the wallet be built from scratch (y/n)? (default = no)\n")
|
|
||||||
nocache = w.readDefaultYesNo(false)
|
|
||||||
}
|
|
||||||
if out, err := deployWallet(client, w.network, w.conf.bootnodes, infos, nocache); err != nil {
|
|
||||||
log.Error("Failed to deploy wallet container", "err", err)
|
|
||||||
if len(out) > 0 {
|
|
||||||
fmt.Printf("%s\n", out)
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
// All ok, run a network scan to pick any changes up
|
|
||||||
log.Info("Waiting for node to finish booting")
|
|
||||||
time.Sleep(3 * time.Second)
|
|
||||||
|
|
||||||
w.networkStats()
|
|
||||||
}
|
|
Loading…
Reference in New Issue
Block a user