prysm-pulse/contracts/deposit-contract/sendDepositTx/sendDeposits.go

234 lines
6.3 KiB
Go
Raw Normal View History

package main
import (
"bufio"
"bytes"
"crypto/rand"
"io/ioutil"
"math"
"math/big"
"os"
"time"
"github.com/ethereum/go-ethereum/accounts/abi/bind"
"github.com/ethereum/go-ethereum/accounts/keystore"
"github.com/ethereum/go-ethereum/common"
"github.com/ethereum/go-ethereum/crypto"
"github.com/ethereum/go-ethereum/ethclient"
"github.com/ethereum/go-ethereum/rpc"
contracts "github.com/prysmaticlabs/prysm/contracts/deposit-contract"
pb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1"
prysmKeyStore "github.com/prysmaticlabs/prysm/shared/keystore"
"github.com/prysmaticlabs/prysm/shared/ssz"
"github.com/prysmaticlabs/prysm/shared/version"
"github.com/sirupsen/logrus"
"github.com/urfave/cli"
prefixed "github.com/x-cray/logrus-prefixed-formatter"
rand2 "golang.org/x/exp/rand"
"gonum.org/v1/gonum/stat/distuv"
)
func main() {
var keystoreUTCPath string
var ipcPath string
var passwordFile string
var httpPath string
var privKeyString string
var depositContractAddr string
var numberOfDeposits int64
var depositAmount int64
var depositDelay int64
var variableTx bool
var txDeviation int64
customFormatter := new(prefixed.TextFormatter)
customFormatter.TimestampFormat = "2006-01-02 15:04:05"
customFormatter.FullTimestamp = true
logrus.SetFormatter(customFormatter)
log := logrus.WithField("prefix", "main")
app := cli.NewApp()
app.Name = "sendDepositTx"
app.Usage = "this is a util to send deposit transactions"
app.Version = version.GetVersion()
app.Flags = []cli.Flag{
cli.StringFlag{
Name: "keystoreUTCPath",
Usage: "Location of keystore",
Destination: &keystoreUTCPath,
},
cli.StringFlag{
Name: "ipcPath",
Usage: "Filename for IPC socket/pipe within the datadir",
Destination: &ipcPath,
},
cli.StringFlag{
Name: "httpPath",
Value: "http://localhost:8545/",
Usage: "HTTP-RPC server listening interface",
Destination: &httpPath,
},
cli.StringFlag{
Name: "passwordFile",
Value: "./password.txt",
Usage: "Password file for unlock account",
Destination: &passwordFile,
},
cli.StringFlag{
Name: "privKey",
Usage: "Private key to unlock account",
Destination: &privKeyString,
},
cli.StringFlag{
Name: "depositContract",
Usage: "Address of the deposit contract",
Destination: &depositContractAddr,
},
cli.Int64Flag{
Name: "numberOfDeposits",
Value: 8,
Usage: "number of deposits to send to the contract",
Destination: &numberOfDeposits,
},
cli.Int64Flag{
Name: "depositAmount",
Value: 3200,
Usage: "Maximum deposit value allowed in contract(in gwei)",
Destination: &depositAmount,
},
cli.Int64Flag{
Name: "depositDelay",
Value: 5,
Usage: "The time delay between sending the deposits to the contract(in seconds)",
Destination: &depositDelay,
},
cli.BoolFlag{
Name: "variableTx",
Usage: "This enables variable transaction latencies to simulate real-world transactions",
Destination: &variableTx,
},
cli.Int64Flag{
Name: "txDeviation",
Usage: "The standard deviation between transaction times",
Value: 2,
Destination: &txDeviation,
},
}
app.Action = func(c *cli.Context) {
// Set up RPC client
var rpcClient *rpc.Client
var err error
var txOps *bind.TransactOpts
// Uses HTTP-RPC if IPC is not set
if ipcPath == "" {
rpcClient, err = rpc.Dial(httpPath)
} else {
rpcClient, err = rpc.Dial(ipcPath)
}
if err != nil {
log.Fatal(err)
}
client := ethclient.NewClient(rpcClient)
depositAmount = depositAmount * 1e9
// User inputs private key, sign tx with private key
if privKeyString != "" {
privKey, err := crypto.HexToECDSA(privKeyString)
if err != nil {
log.Fatal(err)
}
txOps = bind.NewKeyedTransactor(privKey)
txOps.Value = big.NewInt(depositAmount)
txOps.GasLimit = 4000000
// User inputs keystore json file, sign tx with keystore json
} else {
// #nosec - Inclusion of file via variable is OK for this tool.
file, err := os.Open(passwordFile)
if err != nil {
log.Fatal(err)
}
scanner := bufio.NewScanner(file)
scanner.Split(bufio.ScanWords)
scanner.Scan()
password := scanner.Text()
// #nosec - Inclusion of file via variable is OK for this tool.
keyJSON, err := ioutil.ReadFile(keystoreUTCPath)
if err != nil {
log.Fatal(err)
}
privKey, err := keystore.DecryptKey(keyJSON, password)
if err != nil {
log.Fatal(err)
}
txOps = bind.NewKeyedTransactor(privKey.PrivateKey)
txOps.Value = big.NewInt(depositAmount)
txOps.GasLimit = 4000000
}
depositContract, err := contracts.NewDepositContract(common.HexToAddress(depositContractAddr), client)
if err != nil {
log.Fatal(err)
}
statDist := buildStatisticalDist(depositDelay, numberOfDeposits, txDeviation)
for i := int64(0); i < numberOfDeposits; i++ {
validatorKey, err := prysmKeyStore.NewKey(rand.Reader)
data := &pb.DepositInput{
Pubkey: validatorKey.PublicKey.Marshal(),
ProofOfPossession: []byte("pop"),
WithdrawalCredentialsHash32: []byte("withdraw"),
}
serializedData := new(bytes.Buffer)
if err := ssz.Encode(serializedData, data); err != nil {
log.Errorf("could not serialize deposit data: %v", err)
}
tx, err := depositContract.Deposit(txOps, serializedData.Bytes())
if err != nil {
log.Error("unable to send transaction to contract")
}
log.WithFields(logrus.Fields{
"Transaction Hash": tx.Hash(),
}).Infof("Deposit %d sent to contract for validator with a public key %#x", i, validatorKey.PublicKey.Marshal())
// If flag is enabled make transaction times variable
if variableTx {
time.Sleep(time.Duration(math.Abs(statDist.Rand())) * time.Second)
continue
}
time.Sleep(time.Duration(depositDelay) * time.Second)
}
}
err := app.Run(os.Args)
if err != nil {
log.Fatal(err)
}
}
func buildStatisticalDist(depositDelay int64, numberOfDeposits int64, txDeviation int64) *distuv.StudentsT {
src := rand2.NewSource(uint64(time.Now().Unix()))
dist := &distuv.StudentsT{
Mu: float64(depositDelay),
Sigma: float64(txDeviation),
Nu: float64(numberOfDeposits - 1),
Src: src,
}
return dist
}