121 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			121 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2019 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 (
 | |
| 	"strconv"
 | |
| 
 | |
| 	"github.com/ethereum/go-ethereum/accounts"
 | |
| 	"github.com/ethereum/go-ethereum/accounts/abi/bind"
 | |
| 	"github.com/ethereum/go-ethereum/accounts/external"
 | |
| 	"github.com/ethereum/go-ethereum/cmd/utils"
 | |
| 	"github.com/ethereum/go-ethereum/common"
 | |
| 	"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
 | |
| }
 | |
| 
 | |
| // newClefSigner sets up a clef backend and returns a clef transaction signer.
 | |
| func newClefSigner(ctx *cli.Context) *bind.TransactOpts {
 | |
| 	clef, err := external.NewExternalSigner(ctx.String(clefURLFlag.Name))
 | |
| 	if err != nil {
 | |
| 		utils.Fatalf("Failed to create clef signer %v", err)
 | |
| 	}
 | |
| 	return bind.NewClefTransactor(clef, accounts.Account{Address: common.HexToAddress(ctx.String(signerFlag.Name))})
 | |
| }
 |