| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166 |
- // Copyright 2018 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 (
- "io/ioutil"
- "strconv"
- "strings"
- "github.com/ethereum/go-ethereum/accounts/keystore"
- "github.com/ethereum/go-ethereum/cmd/utils"
- "github.com/ethereum/go-ethereum/common"
- "github.com/ethereum/go-ethereum/console"
- "github.com/ethereum/go-ethereum/contracts/checkpointoracle"
- "github.com/ethereum/go-ethereum/ethclient"
- "github.com/ethereum/go-ethereum/params"
- "github.com/ethereum/go-ethereum/rpc"
- "gopkg.in/urfave/cli.v1"
- )
- // newClient creates a client with specified remote URL.
- func newClient(ctx *cli.Context) *ethclient.Client {
- client, err := ethclient.Dial(ctx.GlobalString(nodeURLFlag.Name))
- if err != nil {
- utils.Fatalf("Failed to connect to Ethereum node: %v", err)
- }
- return client
- }
- // newRPCClient creates a rpc client with specified node URL.
- func newRPCClient(url string) *rpc.Client {
- client, err := rpc.Dial(url)
- if err != nil {
- utils.Fatalf("Failed to connect to Ethereum node: %v", err)
- }
- return client
- }
- // getContractAddr retrieves the register contract address through
- // rpc request.
- func getContractAddr(client *rpc.Client) common.Address {
- var addr string
- if err := client.Call(&addr, "les_getCheckpointContractAddress"); err != nil {
- utils.Fatalf("Failed to fetch checkpoint oracle address: %v", err)
- }
- return common.HexToAddress(addr)
- }
- // getCheckpoint retrieves the specified checkpoint or the latest one
- // through rpc request.
- func getCheckpoint(ctx *cli.Context, client *rpc.Client) *params.TrustedCheckpoint {
- var checkpoint *params.TrustedCheckpoint
- if ctx.GlobalIsSet(indexFlag.Name) {
- var result [3]string
- index := uint64(ctx.GlobalInt64(indexFlag.Name))
- if err := client.Call(&result, "les_getCheckpoint", index); err != nil {
- utils.Fatalf("Failed to get local checkpoint %v, please ensure the les API is exposed", err)
- }
- checkpoint = ¶ms.TrustedCheckpoint{
- SectionIndex: index,
- SectionHead: common.HexToHash(result[0]),
- CHTRoot: common.HexToHash(result[1]),
- BloomRoot: common.HexToHash(result[2]),
- }
- } else {
- var result [4]string
- err := client.Call(&result, "les_latestCheckpoint")
- if err != nil {
- utils.Fatalf("Failed to get local checkpoint %v, please ensure the les API is exposed", err)
- }
- index, err := strconv.ParseUint(result[0], 0, 64)
- if err != nil {
- utils.Fatalf("Failed to parse checkpoint index %v", err)
- }
- checkpoint = ¶ms.TrustedCheckpoint{
- SectionIndex: index,
- SectionHead: common.HexToHash(result[1]),
- CHTRoot: common.HexToHash(result[2]),
- BloomRoot: common.HexToHash(result[3]),
- }
- }
- return checkpoint
- }
- // newContract creates a registrar contract instance with specified
- // contract address or the default contracts for mainnet or testnet.
- func newContract(client *rpc.Client) (common.Address, *checkpointoracle.CheckpointOracle) {
- addr := getContractAddr(client)
- if addr == (common.Address{}) {
- utils.Fatalf("No specified registrar contract address")
- }
- contract, err := checkpointoracle.NewCheckpointOracle(addr, ethclient.NewClient(client))
- if err != nil {
- utils.Fatalf("Failed to setup registrar contract %s: %v", addr, err)
- }
- return addr, contract
- }
- // promptPassphrase prompts the user for a passphrase.
- // Set confirmation to true to require the user to confirm the passphrase.
- func promptPassphrase(confirmation bool) string {
- passphrase, err := console.Stdin.PromptPassword("Passphrase: ")
- if err != nil {
- utils.Fatalf("Failed to read passphrase: %v", err)
- }
- if confirmation {
- confirm, err := console.Stdin.PromptPassword("Repeat passphrase: ")
- if err != nil {
- utils.Fatalf("Failed to read passphrase confirmation: %v", err)
- }
- if passphrase != confirm {
- utils.Fatalf("Passphrases do not match")
- }
- }
- return passphrase
- }
- // getPassphrase obtains a passphrase given by the user. It first checks the
- // --password command line flag and ultimately prompts the user for a
- // passphrase.
- func getPassphrase(ctx *cli.Context) string {
- passphraseFile := ctx.String(utils.PasswordFileFlag.Name)
- if passphraseFile != "" {
- content, err := ioutil.ReadFile(passphraseFile)
- if err != nil {
- utils.Fatalf("Failed to read passphrase file '%s': %v",
- passphraseFile, err)
- }
- return strings.TrimRight(string(content), "\r\n")
- }
- // Otherwise prompt the user for the passphrase.
- return promptPassphrase(false)
- }
- // getKey retrieves the user key through specified key file.
- func getKey(ctx *cli.Context) *keystore.Key {
- // Read key from file.
- keyFile := ctx.GlobalString(keyFileFlag.Name)
- keyJson, err := ioutil.ReadFile(keyFile)
- if err != nil {
- utils.Fatalf("Failed to read the keyfile at '%s': %v", keyFile, err)
- }
- // Decrypt key with passphrase.
- passphrase := getPassphrase(ctx)
- key, err := keystore.DecryptKey(keyJson, passphrase)
- if err != nil {
- utils.Fatalf("Failed to decrypt user key '%s': %v", keyFile, err)
- }
- return key
- }
|