aboutsummaryrefslogtreecommitdiffstats
path: root/cmd/checkpoint-admin/common.go
blob: 107cd1de02a67d8238edd8ed79692e45fabf18d0 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
// 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 (
    "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 = &params.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 = &params.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))})
}