mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-19 09:04:13 +00:00
217 lines
7.2 KiB
Go
217 lines
7.2 KiB
Go
package services
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"math/big"
|
|
"os/exec"
|
|
"time"
|
|
|
|
"github.com/ledgerwatch/erigon/cmd/devnettest/utils"
|
|
|
|
"github.com/holiman/uint256"
|
|
"github.com/ledgerwatch/erigon/accounts/abi/bind"
|
|
"github.com/ledgerwatch/erigon/accounts/abi/bind/backends"
|
|
"github.com/ledgerwatch/erigon/cmd/devnettest/contracts"
|
|
"github.com/ledgerwatch/erigon/cmd/devnettest/requests"
|
|
"github.com/ledgerwatch/erigon/common"
|
|
"github.com/ledgerwatch/erigon/common/hexutil"
|
|
"github.com/ledgerwatch/erigon/core"
|
|
"github.com/ledgerwatch/erigon/core/types"
|
|
"github.com/ledgerwatch/erigon/crypto"
|
|
"github.com/ledgerwatch/erigon/params"
|
|
"github.com/ledgerwatch/erigon/rpc"
|
|
)
|
|
|
|
const (
|
|
gasPrice = 912345678
|
|
)
|
|
|
|
var (
|
|
devnetSignPrivateKey, _ = crypto.HexToECDSA("26e86e45f6fc45ec6e2ecd128cec80fa1d1505e5507dcd2ae58c3130a7a97b48")
|
|
signer = types.LatestSigner(params.AllCliqueProtocolChanges)
|
|
DevAddress = "67b1d87101671b127f5f8714789C7192f7ad340e"
|
|
gspec = core.DeveloperGenesisBlock(uint64(0), common.HexToAddress(DevAddress))
|
|
contractBackend = backends.NewSimulatedBackendWithConfig(gspec.Alloc, gspec.Config, 1_000_000)
|
|
)
|
|
|
|
type Block struct {
|
|
Number *hexutil.Big
|
|
Transactions []common.Hash
|
|
}
|
|
|
|
// CreateTransaction returns transaction details depending on what transaction type is given
|
|
func CreateTransaction(transactionType, addr string, value, nonce uint64) (*types.Transaction, common.Address, *contracts.Subscription, *bind.TransactOpts, error) {
|
|
if transactionType == "regular" {
|
|
tx, address, err := createNonContractTx(addr, value, nonce)
|
|
if err != nil {
|
|
return nil, common.Address{}, nil, nil, fmt.Errorf("failed to create non-contract transaction: %v", err)
|
|
}
|
|
return tx, address, nil, nil, nil
|
|
}
|
|
return createContractTx(nonce)
|
|
}
|
|
|
|
// createNonContractTx takes in a signer and returns the signed transaction and the address receiving the sent value
|
|
func createNonContractTx(addr string, value, nonce uint64) (*types.Transaction, common.Address, error) {
|
|
toAddress := common.HexToAddress(addr)
|
|
signedTx, err := types.SignTx(types.NewTransaction(nonce, toAddress, uint256.NewInt(value),
|
|
params.TxGas, uint256.NewInt(gasPrice), nil), *signer, devnetSignPrivateKey)
|
|
if err != nil {
|
|
return nil, toAddress, fmt.Errorf("failed to sign transaction: %v", err)
|
|
}
|
|
return &signedTx, toAddress, nil
|
|
}
|
|
|
|
// createContractTx creates and signs a transaction using the developer address, returns the contract and the signed transaction
|
|
func createContractTx(nonce uint64) (*types.Transaction, common.Address, *contracts.Subscription, *bind.TransactOpts, error) {
|
|
// initialize transactOpts
|
|
transactOpts, err := initializeTransactOps(nonce)
|
|
if err != nil {
|
|
return nil, common.Address{}, nil, nil, fmt.Errorf("failed to initialize transactOpts: %v", err)
|
|
}
|
|
|
|
// deploy the contract and get the contract handler
|
|
address, txToSign, subscriptionContract, err := contracts.DeploySubscription(transactOpts, contractBackend)
|
|
if err != nil {
|
|
return nil, common.Address{}, nil, nil, fmt.Errorf("failed to deploy subscription: %v", err)
|
|
}
|
|
|
|
// sign the transaction with the private key
|
|
signedTx, err := types.SignTx(txToSign, *signer, devnetSignPrivateKey)
|
|
if err != nil {
|
|
return nil, common.Address{}, nil, nil, fmt.Errorf("failed to sign tx: %v", err)
|
|
}
|
|
|
|
return &signedTx, address, subscriptionContract, transactOpts, nil
|
|
}
|
|
|
|
func initializeTransactOps(nonce uint64) (*bind.TransactOpts, error) {
|
|
const txGas uint64 = 200_000
|
|
var chainID = big.NewInt(1337)
|
|
|
|
transactOpts, err := bind.NewKeyedTransactorWithChainID(devnetSignPrivateKey, chainID)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("cannot create transactor with chainID %s, error: %v", chainID, err)
|
|
}
|
|
|
|
transactOpts.GasLimit = txGas
|
|
transactOpts.GasPrice = big.NewInt(880_000_000)
|
|
// TODO: Get Nonce from account automatically
|
|
transactOpts.Nonce = big.NewInt(int64(nonce))
|
|
|
|
return transactOpts, nil
|
|
}
|
|
|
|
// SearchBlockForTx connects the client to a websocket and listens for new heads to search the blocks for a tx hash
|
|
func SearchBlockForTx(txnHash common.Hash) (uint64, error) {
|
|
client, clientErr := rpc.DialWebsocket(context.Background(), "ws://127.0.0.1:8545", "")
|
|
if clientErr != nil {
|
|
return 0, fmt.Errorf("failed to dial websocket: %v", clientErr)
|
|
}
|
|
|
|
fmt.Printf("\nSearching for tx %q in new blocks...\n", txnHash)
|
|
blockN, err := subscribeToNewHeadsAndSearch(client, "eth_newHeads", txnHash)
|
|
if err != nil {
|
|
return 0, fmt.Errorf("failed to subscribe to ws: %v", err)
|
|
}
|
|
|
|
// var count int
|
|
|
|
// ForLoop:
|
|
// for {
|
|
// select {
|
|
// case v := <-ch:
|
|
// count++
|
|
// _map := v.(map[string]interface{})
|
|
// for k, val := range _map {
|
|
// fmt.Printf("%s: %+v, ", k, val)
|
|
// }
|
|
// fmt.Println()
|
|
// fmt.Println()
|
|
// if count == numberOfIterations {
|
|
// break ForLoop
|
|
// }
|
|
// }
|
|
// }
|
|
|
|
return blockN, nil
|
|
}
|
|
|
|
// blockHasHash checks if the current block has the transaction hash in its list of transactions
|
|
func blockHasHash(client *rpc.Client, hash common.Hash, blockNumber string) (uint64, bool, error) {
|
|
ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
|
|
defer cancel()
|
|
|
|
var currentBlock Block
|
|
err := client.CallContext(ctx, ¤tBlock, "eth_getBlockByNumber", blockNumber, false)
|
|
if err != nil {
|
|
return uint64(0), false, fmt.Errorf("failed to get block by number: %v", err)
|
|
}
|
|
|
|
for _, txnHash := range currentBlock.Transactions {
|
|
if txnHash == hash {
|
|
fmt.Printf("SUCCESS => Tx with hash %q is in mined block with number %q\n", hash, blockNumber)
|
|
return utils.HexToInt(blockNumber), true, nil
|
|
}
|
|
}
|
|
|
|
return uint64(0), false, nil
|
|
}
|
|
|
|
// EmitEventAndGetLogs emits an event from the contract using the fallback method
|
|
func EmitEventAndGetLogs(reqId int, subContract *contracts.Subscription, opts *bind.TransactOpts, address common.Address) error {
|
|
opts.Nonce.Add(opts.Nonce, big.NewInt(1))
|
|
|
|
tx, err := subContract.Fallback(opts, []byte{})
|
|
if err != nil {
|
|
return fmt.Errorf("failed to emit event from fallback: %v", err)
|
|
}
|
|
|
|
signedTx, err := types.SignTx(tx, *signer, devnetSignPrivateKey)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to sign transaction: %v", err)
|
|
}
|
|
|
|
fmt.Printf("\nSending Fallback tx from contract...\n")
|
|
hash, err := requests.SendTx(reqId, &signedTx)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to send transaction: %v", err)
|
|
}
|
|
fmt.Printf("SUCCESS => Tx submitted, adding tx with hash %q to txpool\n", hash)
|
|
|
|
blockN, err := SearchBlockForTx(*hash)
|
|
if err != nil {
|
|
return fmt.Errorf("error searching block for tx: %v", err)
|
|
}
|
|
|
|
if err = requests.GetLogs(reqId, blockN, blockN, address, false); err != nil {
|
|
return fmt.Errorf("failed to get logs: %v", err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func ApplyTransaction(ctx context.Context, tx types.Transaction) error {
|
|
err := contractBackend.SendTransaction(ctx, tx)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to send transaction: %v", err)
|
|
}
|
|
contractBackend.Commit()
|
|
return nil
|
|
}
|
|
|
|
// ClearDevDB cleans up the dev folder used for the operations
|
|
func ClearDevDB() {
|
|
fmt.Printf("\nDeleting ./dev folder\n")
|
|
|
|
cmd := exec.Command("rm", "-rf", "./dev")
|
|
err := cmd.Run()
|
|
if err != nil {
|
|
fmt.Println("Error occurred clearing Dev DB")
|
|
panic("could not clear dev DB")
|
|
}
|
|
|
|
fmt.Printf("SUCCESS => Deleted ./dev\n")
|
|
}
|