2023-07-18 08:47:04 +00:00
|
|
|
package contracts_steps
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"math/big"
|
|
|
|
|
2023-11-11 12:04:18 +00:00
|
|
|
"github.com/ledgerwatch/erigon-lib/common/hexutil"
|
|
|
|
|
2023-07-18 08:47:04 +00:00
|
|
|
ethereum "github.com/ledgerwatch/erigon"
|
|
|
|
libcommon "github.com/ledgerwatch/erigon-lib/common"
|
|
|
|
"github.com/ledgerwatch/erigon-lib/common/hexutility"
|
|
|
|
"github.com/ledgerwatch/log/v3"
|
|
|
|
|
|
|
|
"github.com/ledgerwatch/erigon/accounts/abi/bind"
|
|
|
|
"github.com/ledgerwatch/erigon/cmd/devnet/accounts"
|
|
|
|
"github.com/ledgerwatch/erigon/cmd/devnet/contracts"
|
|
|
|
"github.com/ledgerwatch/erigon/cmd/devnet/devnet"
|
|
|
|
"github.com/ledgerwatch/erigon/cmd/devnet/devnetutils"
|
|
|
|
"github.com/ledgerwatch/erigon/cmd/devnet/requests"
|
|
|
|
"github.com/ledgerwatch/erigon/cmd/devnet/scenarios"
|
|
|
|
"github.com/ledgerwatch/erigon/cmd/devnet/transactions"
|
|
|
|
"github.com/ledgerwatch/erigon/core/types"
|
2023-08-25 11:19:39 +00:00
|
|
|
"github.com/ledgerwatch/erigon/rpc"
|
2023-07-18 08:47:04 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
scenarios.MustRegisterStepHandlers(
|
|
|
|
scenarios.StepHandler(DeployAndCallLogSubscriber),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
func DeployAndCallLogSubscriber(ctx context.Context, deployer string) (*libcommon.Hash, error) {
|
|
|
|
logger := devnet.Logger(ctx)
|
|
|
|
|
|
|
|
node := devnet.SelectNode(ctx)
|
|
|
|
|
|
|
|
deployerAddress := libcommon.HexToAddress(deployer)
|
|
|
|
|
|
|
|
// subscriptionContract is the handler to the contract for further operations
|
|
|
|
tx, address, subscriptionContract, transactOpts, err := DeploySubsriptionContract(node, deployerAddress)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
logger.Error("failed to create transaction", "error", err)
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
hash := tx.Hash()
|
|
|
|
|
|
|
|
eventHash, err := EmitFallbackEvent(node, subscriptionContract, transactOpts, logger)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
logger.Error("failed to emit events", "error", err)
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2023-07-20 22:10:18 +00:00
|
|
|
txToBlockMap, err := transactions.AwaitTransactions(ctx, hash, eventHash)
|
2023-07-18 08:47:04 +00:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to call contract tx: %v", err)
|
|
|
|
}
|
|
|
|
|
2023-07-28 13:03:32 +00:00
|
|
|
blockNum := txToBlockMap[eventHash]
|
2023-07-18 08:47:04 +00:00
|
|
|
|
2023-11-17 10:41:45 +00:00
|
|
|
block, err := node.GetBlockByNumber(ctx, rpc.AsBlockNumber(blockNum), true)
|
2023-07-18 08:47:04 +00:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
logs, err := node.FilterLogs(ctx, ethereum.FilterQuery{
|
|
|
|
FromBlock: big.NewInt(0),
|
2023-07-28 13:03:32 +00:00
|
|
|
ToBlock: big.NewInt(int64(blockNum)),
|
2023-07-18 08:47:04 +00:00
|
|
|
Addresses: []libcommon.Address{address}})
|
|
|
|
|
|
|
|
if err != nil || len(logs) == 0 {
|
|
|
|
return nil, fmt.Errorf("failed to get logs: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// compare the log events
|
2023-07-28 13:03:32 +00:00
|
|
|
errs, ok := requests.Compare(requests.NewLog(eventHash, blockNum, address,
|
2023-07-18 08:47:04 +00:00
|
|
|
devnetutils.GenerateTopic("SubscriptionEvent()"), hexutility.Bytes{}, 1,
|
2023-07-28 13:03:32 +00:00
|
|
|
block.Hash, hexutil.Uint(0), false), logs[0])
|
2023-07-18 08:47:04 +00:00
|
|
|
|
|
|
|
if !ok {
|
|
|
|
logger.Error("Log result is incorrect", "errors", errs)
|
|
|
|
return nil, fmt.Errorf("incorrect logs: %v", errs)
|
|
|
|
}
|
|
|
|
|
|
|
|
logger.Info("SUCCESS => Logs compared successfully, no discrepancies")
|
|
|
|
|
|
|
|
return &hash, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// DeploySubsriptionContract creates and signs a transaction using the developer address, returns the contract and the signed transaction
|
|
|
|
func DeploySubsriptionContract(node devnet.Node, deployer libcommon.Address) (types.Transaction, libcommon.Address, *contracts.Subscription, *bind.TransactOpts, error) {
|
|
|
|
// initialize transactOpts
|
|
|
|
transactOpts, err := initializeTransactOps(node, deployer)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, libcommon.Address{}, nil, nil, fmt.Errorf("failed to initialize transactOpts: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// deploy the contract and get the contract handler
|
|
|
|
address, tx, subscriptionContract, err := contracts.DeploySubscription(transactOpts, contracts.NewBackend(node))
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, libcommon.Address{}, nil, nil, fmt.Errorf("failed to deploy subscription: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return tx, address, subscriptionContract, transactOpts, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// EmitFallbackEvent emits an event from the contract using the fallback method
|
|
|
|
func EmitFallbackEvent(node devnet.Node, subContract *contracts.Subscription, opts *bind.TransactOpts, logger log.Logger) (libcommon.Hash, error) {
|
|
|
|
logger.Info("EMITTING EVENT FROM FALLBACK...")
|
|
|
|
|
|
|
|
// adding one to the nonce before initiating another transaction
|
|
|
|
opts.Nonce.Add(opts.Nonce, big.NewInt(1))
|
|
|
|
|
|
|
|
tx, err := subContract.Fallback(opts, []byte{})
|
|
|
|
if err != nil {
|
|
|
|
return libcommon.Hash{}, fmt.Errorf("failed to emit event from fallback: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return tx.Hash(), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// initializeTransactOps initializes the transactOpts object for a contract transaction
|
|
|
|
func initializeTransactOps(node devnet.Node, transactor libcommon.Address) (*bind.TransactOpts, error) {
|
2023-08-25 11:19:39 +00:00
|
|
|
count, err := node.GetTransactionCount(transactor, rpc.LatestBlock)
|
2023-07-18 08:47:04 +00:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to get transaction count for address 0x%x: %v", transactor, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
transactOpts, err := bind.NewKeyedTransactorWithChainID(accounts.SigKey(transactor), node.ChainID())
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("cannot create transactor with chainID %d, error: %v", node.ChainID(), err)
|
|
|
|
}
|
|
|
|
|
|
|
|
transactOpts.GasLimit = uint64(200_000)
|
|
|
|
transactOpts.GasPrice = big.NewInt(880_000_000)
|
|
|
|
transactOpts.Nonce = count
|
|
|
|
|
|
|
|
return transactOpts, nil
|
|
|
|
}
|