erigon-pulse/cmd/devnet/services/block.go
Leonard Chinonso 805230ba63
Setup web socket endpoints to listen for new transactions after sending (#5940)
- Added listening methods for WebSocket subscriptions 
- Listened for new blocks using the newHeads method to determine when to
look for a transaction
- Added new util methods and tests for them
- Simplified communication to the user upon initiating the devnet tool
2022-11-03 09:45:36 +07:00

71 lines
2.5 KiB
Go

package services
import (
"context"
"fmt"
"time"
"github.com/holiman/uint256"
"github.com/ledgerwatch/erigon/cmd/devnet/devnetutils"
"github.com/ledgerwatch/erigon/cmd/devnet/models"
"github.com/ledgerwatch/erigon/common"
"github.com/ledgerwatch/erigon/core/types"
"github.com/ledgerwatch/erigon/params"
"github.com/ledgerwatch/erigon/rpc"
)
const gasPrice = 912345678
var (
signer = types.LatestSigner(params.AllCliqueProtocolChanges)
)
// CreateTransaction creates a transaction depending on the type of transaction being passed
func CreateTransaction(txType models.TransactionType, addr string, value, nonce uint64) (*types.Transaction, common.Address, error) {
if txType == models.NonContractTx {
tx, address, err := createNonContractTx(addr, value, nonce)
if err != nil {
return nil, common.Address{}, fmt.Errorf("failed to create non-contract transaction: %v", err)
}
return tx, address, nil
}
return nil, common.Address{}, models.ErrInvalidTransactionType
}
// createNonContractTx returns a signed transaction and the recipient address
func createNonContractTx(addr string, value, nonce uint64) (*types.Transaction, common.Address, error) {
toAddress := common.HexToAddress(addr)
// create a new transaction using the parameters to send
transaction := types.NewTransaction(nonce, toAddress, uint256.NewInt(value), params.TxGas, uint256.NewInt(gasPrice), nil)
// sign the transaction using the developer 0signed private key
signedTx, err := types.SignTx(transaction, *signer, models.DevSignedPrivateKey)
if err != nil {
return nil, common.Address{}, fmt.Errorf("failed to sign non-contract transaction: %v", err)
}
return &signedTx, toAddress, nil
}
// txHashInBlock checks if the block with block number has the transaction hash in its list of transactions
func txHashInBlock(client *rpc.Client, hash common.Hash, blockNumber string) (uint64, bool, error) {
ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
defer cancel() // releases the resources held by the context
var currBlock models.Block
err := client.CallContext(ctx, &currBlock, string(models.ETHGetBlockByNumber), blockNumber, false)
if err != nil {
return uint64(0), false, fmt.Errorf("failed to get block by number: %v", err)
}
for _, txnHash := range currBlock.Transactions {
if txnHash == hash {
fmt.Printf("SUCCESS => Tx with hash %q is in mined block with number %q\n", hash, blockNumber)
return devnetutils.HexToInt(blockNumber), true, nil
}
}
return uint64(0), false, nil
}