2024-06-06 20:08:25 +00:00
|
|
|
package pathprocessor
|
2022-09-13 07:10:59 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2024-05-14 19:11:16 +00:00
|
|
|
"encoding/json"
|
2023-11-02 11:35:28 +00:00
|
|
|
"errors"
|
2024-04-25 14:43:58 +00:00
|
|
|
"fmt"
|
2022-09-13 07:10:59 +00:00
|
|
|
"math/big"
|
2024-05-14 19:11:16 +00:00
|
|
|
netUrl "net/url"
|
2023-11-02 11:35:28 +00:00
|
|
|
"strings"
|
2022-09-13 07:10:59 +00:00
|
|
|
|
2023-11-02 11:35:28 +00:00
|
|
|
"github.com/ethereum/go-ethereum"
|
|
|
|
"github.com/ethereum/go-ethereum/accounts/abi"
|
2022-09-13 07:10:59 +00:00
|
|
|
"github.com/ethereum/go-ethereum/accounts/abi/bind"
|
|
|
|
"github.com/ethereum/go-ethereum/common"
|
|
|
|
"github.com/ethereum/go-ethereum/common/hexutil"
|
2023-09-29 17:56:27 +00:00
|
|
|
ethTypes "github.com/ethereum/go-ethereum/core/types"
|
2022-09-13 07:10:59 +00:00
|
|
|
"github.com/status-im/status-go/account"
|
|
|
|
"github.com/status-im/status-go/contracts"
|
2022-12-19 12:37:37 +00:00
|
|
|
"github.com/status-im/status-go/contracts/hop"
|
2024-05-23 12:38:39 +00:00
|
|
|
hopL1CctpImplementation "github.com/status-im/status-go/contracts/hop/l1Contracts/l1CctpImplementation"
|
|
|
|
hopL1Erc20Bridge "github.com/status-im/status-go/contracts/hop/l1Contracts/l1Erc20Bridge"
|
|
|
|
hopL1EthBridge "github.com/status-im/status-go/contracts/hop/l1Contracts/l1EthBridge"
|
|
|
|
hopL1HopBridge "github.com/status-im/status-go/contracts/hop/l1Contracts/l1HopBridge"
|
|
|
|
hopL2AmmWrapper "github.com/status-im/status-go/contracts/hop/l2Contracts/l2AmmWrapper"
|
|
|
|
hopL2ArbitrumBridge "github.com/status-im/status-go/contracts/hop/l2Contracts/l2ArbitrumBridge"
|
|
|
|
hopL2CctpImplementation "github.com/status-im/status-go/contracts/hop/l2Contracts/l2CctpImplementation"
|
|
|
|
hopL2OptimismBridge "github.com/status-im/status-go/contracts/hop/l2Contracts/l2OptimismBridge"
|
2022-09-13 07:10:59 +00:00
|
|
|
"github.com/status-im/status-go/eth-node/types"
|
|
|
|
"github.com/status-im/status-go/rpc"
|
2024-05-23 12:38:39 +00:00
|
|
|
"github.com/status-im/status-go/rpc/chain"
|
2024-05-21 14:33:36 +00:00
|
|
|
"github.com/status-im/status-go/services/wallet/bigint"
|
2024-05-14 19:11:16 +00:00
|
|
|
walletCommon "github.com/status-im/status-go/services/wallet/common"
|
|
|
|
"github.com/status-im/status-go/services/wallet/thirdparty"
|
2022-09-13 07:10:59 +00:00
|
|
|
"github.com/status-im/status-go/services/wallet/token"
|
|
|
|
"github.com/status-im/status-go/transactions"
|
|
|
|
)
|
|
|
|
|
2024-05-23 12:38:39 +00:00
|
|
|
const (
|
|
|
|
SevenDaysInSeconds = 604800
|
|
|
|
hopSymbol = "HOP"
|
|
|
|
)
|
2024-05-21 14:33:36 +00:00
|
|
|
|
2024-06-06 20:08:25 +00:00
|
|
|
type HopBridgeTxArgs struct {
|
2022-09-13 07:10:59 +00:00
|
|
|
transactions.SendTxArgs
|
|
|
|
ChainID uint64 `json:"chainId"`
|
|
|
|
Symbol string `json:"symbol"`
|
|
|
|
Recipient common.Address `json:"recipient"`
|
|
|
|
Amount *hexutil.Big `json:"amount"`
|
|
|
|
BonderFee *hexutil.Big `json:"bonderFee"`
|
|
|
|
}
|
|
|
|
|
2024-05-14 19:11:16 +00:00
|
|
|
type BonderFee struct {
|
2024-05-21 14:33:36 +00:00
|
|
|
AmountIn *bigint.BigInt `json:"amountIn"`
|
|
|
|
Slippage float32 `json:"slippage"`
|
|
|
|
AmountOutMin *bigint.BigInt `json:"amountOutMin"`
|
|
|
|
DestinationAmountOutMin *bigint.BigInt `json:"destinationAmountOutMin"`
|
|
|
|
BonderFee *bigint.BigInt `json:"bonderFee"`
|
|
|
|
EstimatedRecieved *bigint.BigInt `json:"estimatedRecieved"`
|
|
|
|
Deadline int64 `json:"deadline"`
|
|
|
|
DestinationDeadline int64 `json:"destinationDeadline"`
|
2024-05-14 19:11:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (bf *BonderFee) UnmarshalJSON(data []byte) error {
|
|
|
|
type Alias BonderFee
|
|
|
|
aux := &struct {
|
|
|
|
AmountIn string `json:"amountIn"`
|
|
|
|
Slippage float32 `json:"slippage"`
|
|
|
|
AmountOutMin string `json:"amountOutMin"`
|
|
|
|
DestinationAmountOutMin string `json:"destinationAmountOutMin"`
|
|
|
|
BonderFee string `json:"bonderFee"`
|
|
|
|
EstimatedRecieved string `json:"estimatedRecieved"`
|
|
|
|
Deadline int64 `json:"deadline"`
|
|
|
|
DestinationDeadline *int64 `json:"destinationDeadline"`
|
|
|
|
*Alias
|
|
|
|
}{
|
|
|
|
Alias: (*Alias)(bf),
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := json.Unmarshal(data, aux); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2024-05-21 14:33:36 +00:00
|
|
|
bf.AmountIn = &bigint.BigInt{Int: new(big.Int)}
|
2024-05-14 19:11:16 +00:00
|
|
|
bf.AmountIn.SetString(aux.AmountIn, 10)
|
|
|
|
|
2024-05-21 14:33:36 +00:00
|
|
|
bf.AmountOutMin = &bigint.BigInt{Int: new(big.Int)}
|
2024-05-14 19:11:16 +00:00
|
|
|
bf.AmountOutMin.SetString(aux.AmountOutMin, 10)
|
|
|
|
|
2024-05-21 14:33:36 +00:00
|
|
|
bf.DestinationAmountOutMin = &bigint.BigInt{Int: new(big.Int)}
|
2024-05-14 19:11:16 +00:00
|
|
|
bf.DestinationAmountOutMin.SetString(aux.DestinationAmountOutMin, 10)
|
|
|
|
|
2024-05-21 14:33:36 +00:00
|
|
|
bf.BonderFee = &bigint.BigInt{Int: new(big.Int)}
|
2024-05-14 19:11:16 +00:00
|
|
|
bf.BonderFee.SetString(aux.BonderFee, 10)
|
|
|
|
|
2024-05-21 14:33:36 +00:00
|
|
|
bf.EstimatedRecieved = &bigint.BigInt{Int: new(big.Int)}
|
2024-05-14 19:11:16 +00:00
|
|
|
bf.EstimatedRecieved.SetString(aux.EstimatedRecieved, 10)
|
|
|
|
|
2024-05-21 14:33:36 +00:00
|
|
|
bf.Deadline = aux.Deadline
|
|
|
|
|
2024-05-14 19:11:16 +00:00
|
|
|
if aux.DestinationDeadline != nil {
|
|
|
|
bf.DestinationDeadline = *aux.DestinationDeadline
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-06-06 20:08:25 +00:00
|
|
|
type HopBridgeProcessor struct {
|
2024-05-26 08:31:13 +00:00
|
|
|
transactor transactions.TransactorIface
|
2024-05-14 19:11:16 +00:00
|
|
|
httpClient *thirdparty.HTTPClient
|
2022-12-19 12:37:37 +00:00
|
|
|
tokenManager *token.Manager
|
2022-09-13 07:10:59 +00:00
|
|
|
contractMaker *contracts.ContractMaker
|
2024-05-14 19:11:16 +00:00
|
|
|
bonderFee *BonderFee
|
2022-09-13 07:10:59 +00:00
|
|
|
}
|
|
|
|
|
2024-06-06 20:08:25 +00:00
|
|
|
func NewHopBridgeProcessor(rpcClient *rpc.Client, transactor transactions.TransactorIface, tokenManager *token.Manager) *HopBridgeProcessor {
|
|
|
|
return &HopBridgeProcessor{
|
2022-09-13 07:10:59 +00:00
|
|
|
contractMaker: &contracts.ContractMaker{RPCClient: rpcClient},
|
2024-05-14 19:11:16 +00:00
|
|
|
httpClient: thirdparty.NewHTTPClient(),
|
2022-12-19 12:37:37 +00:00
|
|
|
transactor: transactor,
|
|
|
|
tokenManager: tokenManager,
|
2022-09-13 07:10:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-06-06 20:08:25 +00:00
|
|
|
func (h *HopBridgeProcessor) Name() string {
|
|
|
|
return ProcessorBridgeHopName
|
2022-09-13 07:10:59 +00:00
|
|
|
}
|
|
|
|
|
2024-06-06 20:08:25 +00:00
|
|
|
func (h *HopBridgeProcessor) AvailableFor(params ProcessorInputParams) (bool, error) {
|
2024-05-23 12:38:39 +00:00
|
|
|
// We chcek if the contract is available on the network for the token
|
2024-06-05 07:56:02 +00:00
|
|
|
_, err := h.GetContractAddress(params)
|
2024-05-23 12:38:39 +00:00
|
|
|
// toToken is not nil only if the send type is Swap
|
2024-06-05 07:56:02 +00:00
|
|
|
return err == nil && params.ToToken == nil, nil
|
2024-05-23 12:38:39 +00:00
|
|
|
}
|
2022-09-13 07:10:59 +00:00
|
|
|
|
2024-06-06 20:08:25 +00:00
|
|
|
func (c *HopBridgeProcessor) getAppropriateABI(contractType string, chainID uint64, token *token.Token) (abi.ABI, error) {
|
2024-05-23 12:38:39 +00:00
|
|
|
switch contractType {
|
|
|
|
case hop.CctpL1Bridge:
|
|
|
|
return abi.JSON(strings.NewReader(hopL1CctpImplementation.HopL1CctpImplementationABI))
|
|
|
|
case hop.L1Bridge:
|
|
|
|
if token.IsNative() {
|
|
|
|
return abi.JSON(strings.NewReader(hopL1EthBridge.HopL1EthBridgeABI))
|
|
|
|
}
|
|
|
|
if token.Symbol == hopSymbol {
|
|
|
|
return abi.JSON(strings.NewReader(hopL1HopBridge.HopL1HopBridgeABI))
|
|
|
|
}
|
|
|
|
return abi.JSON(strings.NewReader(hopL1Erc20Bridge.HopL1Erc20BridgeABI))
|
|
|
|
case hop.L2AmmWrapper:
|
|
|
|
return abi.JSON(strings.NewReader(hopL2AmmWrapper.HopL2AmmWrapperABI))
|
|
|
|
case hop.CctpL2Bridge:
|
|
|
|
return abi.JSON(strings.NewReader(hopL2CctpImplementation.HopL2CctpImplementationABI))
|
|
|
|
case hop.L2Bridge:
|
|
|
|
if chainID == walletCommon.OptimismMainnet ||
|
|
|
|
chainID == walletCommon.OptimismSepolia {
|
|
|
|
return abi.JSON(strings.NewReader(hopL2OptimismBridge.HopL2OptimismBridgeABI))
|
|
|
|
}
|
|
|
|
if chainID == walletCommon.ArbitrumMainnet ||
|
|
|
|
chainID == walletCommon.ArbitrumSepolia {
|
|
|
|
return abi.JSON(strings.NewReader(hopL2ArbitrumBridge.HopL2ArbitrumBridgeABI))
|
|
|
|
}
|
2022-09-13 07:10:59 +00:00
|
|
|
}
|
|
|
|
|
2024-05-23 12:38:39 +00:00
|
|
|
return abi.ABI{}, errors.New("not available for contract type")
|
2022-09-13 07:10:59 +00:00
|
|
|
}
|
|
|
|
|
2024-06-10 12:52:47 +00:00
|
|
|
func (h *HopBridgeProcessor) PackTxInputData(params ProcessorInputParams) ([]byte, error) {
|
|
|
|
_, contractType, err := hop.GetContractAddress(params.FromChain.ChainID, params.FromToken.Symbol)
|
|
|
|
if err != nil {
|
|
|
|
return []byte{}, err
|
2024-06-10 12:39:49 +00:00
|
|
|
}
|
|
|
|
|
2024-06-10 12:52:47 +00:00
|
|
|
return h.packTxInputDataInternally(params, contractType)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (h *HopBridgeProcessor) packTxInputDataInternally(params ProcessorInputParams, contractType string) ([]byte, error) {
|
2024-06-05 07:56:02 +00:00
|
|
|
abi, err := h.getAppropriateABI(contractType, params.FromChain.ChainID, params.FromToken)
|
2024-05-23 12:38:39 +00:00
|
|
|
if err != nil {
|
|
|
|
return []byte{}, err
|
|
|
|
}
|
2023-11-02 11:35:28 +00:00
|
|
|
|
2024-05-23 12:38:39 +00:00
|
|
|
switch contractType {
|
|
|
|
case hop.CctpL1Bridge:
|
2024-06-05 07:56:02 +00:00
|
|
|
return h.packCctpL1BridgeTx(abi, params.ToChain.ChainID, params.ToAddr)
|
2024-05-23 12:38:39 +00:00
|
|
|
case hop.L1Bridge:
|
2024-06-05 07:56:02 +00:00
|
|
|
return h.packL1BridgeTx(abi, params.ToChain.ChainID, params.ToAddr)
|
2024-05-23 12:38:39 +00:00
|
|
|
case hop.L2AmmWrapper:
|
2024-06-05 07:56:02 +00:00
|
|
|
return h.packL2AmmWrapperTx(abi, params.ToChain.ChainID, params.ToAddr)
|
2024-05-23 12:38:39 +00:00
|
|
|
case hop.CctpL2Bridge:
|
2024-06-05 07:56:02 +00:00
|
|
|
return h.packCctpL2BridgeTx(abi, params.ToChain.ChainID, params.ToAddr)
|
2024-05-23 12:38:39 +00:00
|
|
|
case hop.L2Bridge:
|
2024-06-05 07:56:02 +00:00
|
|
|
return h.packL2BridgeTx(abi, params.ToChain.ChainID, params.ToAddr)
|
2024-05-21 14:33:36 +00:00
|
|
|
}
|
2024-05-23 12:38:39 +00:00
|
|
|
|
|
|
|
return []byte{}, errors.New("contract type not supported yet")
|
2024-05-21 14:33:36 +00:00
|
|
|
}
|
2023-11-02 11:35:28 +00:00
|
|
|
|
2024-06-06 20:08:25 +00:00
|
|
|
func (h *HopBridgeProcessor) EstimateGas(params ProcessorInputParams) (uint64, error) {
|
2024-05-23 12:38:39 +00:00
|
|
|
value := big.NewInt(0)
|
2024-06-05 07:56:02 +00:00
|
|
|
if params.FromToken.IsNative() {
|
|
|
|
value = params.AmountIn
|
2024-05-21 14:33:36 +00:00
|
|
|
}
|
|
|
|
|
2024-06-05 07:56:02 +00:00
|
|
|
contractAddress, contractType, err := hop.GetContractAddress(params.FromChain.ChainID, params.FromToken.Symbol)
|
2024-05-21 14:33:36 +00:00
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
2023-11-02 11:35:28 +00:00
|
|
|
}
|
|
|
|
|
2024-06-10 12:52:47 +00:00
|
|
|
input, err := h.packTxInputDataInternally(params, contractType)
|
2023-11-02 11:35:28 +00:00
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
2024-06-05 07:56:02 +00:00
|
|
|
ethClient, err := h.contractMaker.RPCClient.EthClient(params.FromChain.ChainID)
|
2024-05-23 12:38:39 +00:00
|
|
|
if err != nil {
|
2023-11-02 11:35:28 +00:00
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
msg := ethereum.CallMsg{
|
2024-06-05 07:56:02 +00:00
|
|
|
From: params.FromAddr,
|
2024-05-23 12:38:39 +00:00
|
|
|
To: &contractAddress,
|
2023-11-02 11:35:28 +00:00
|
|
|
Value: value,
|
|
|
|
Data: input,
|
|
|
|
}
|
|
|
|
|
2024-05-23 12:38:39 +00:00
|
|
|
estimation, err := ethClient.EstimateGas(context.Background(), msg)
|
2023-11-02 11:35:28 +00:00
|
|
|
if err != nil {
|
2024-06-05 07:56:02 +00:00
|
|
|
if !params.FromToken.IsNative() {
|
2024-05-28 10:29:02 +00:00
|
|
|
// TODO: this is a temporary solution until we find a better way to estimate the gas
|
|
|
|
// hardcoding the estimation for other than ETH, cause we cannot get a proper estimation without having an approval placed first
|
|
|
|
// this is an error we're facing otherwise: `execution reverted: ERC20: transfer amount exceeds allowance`
|
|
|
|
estimation = 350000
|
|
|
|
} else {
|
|
|
|
return 0, err
|
|
|
|
}
|
2023-11-02 11:35:28 +00:00
|
|
|
}
|
2024-05-23 12:38:39 +00:00
|
|
|
|
2023-11-02 11:35:28 +00:00
|
|
|
increasedEstimation := float64(estimation) * IncreaseEstimatedGasFactor
|
|
|
|
return uint64(increasedEstimation), nil
|
2022-09-13 07:10:59 +00:00
|
|
|
}
|
|
|
|
|
2024-06-06 20:08:25 +00:00
|
|
|
func (h *HopBridgeProcessor) BuildTx(params ProcessorInputParams) (*ethTypes.Transaction, error) {
|
2024-06-05 07:56:02 +00:00
|
|
|
toAddr := types.Address(params.ToAddr)
|
2024-06-06 20:08:25 +00:00
|
|
|
sendArgs := &MultipathProcessorTxArgs{
|
|
|
|
HopTx: &HopBridgeTxArgs{
|
2024-03-25 12:40:00 +00:00
|
|
|
SendTxArgs: transactions.SendTxArgs{
|
2024-06-05 07:56:02 +00:00
|
|
|
From: types.Address(params.FromAddr),
|
2024-03-25 12:40:00 +00:00
|
|
|
To: &toAddr,
|
2024-06-05 07:56:02 +00:00
|
|
|
Value: (*hexutil.Big)(params.AmountIn),
|
2024-03-25 12:40:00 +00:00
|
|
|
Data: types.HexBytes("0x0"),
|
|
|
|
},
|
2024-06-05 07:56:02 +00:00
|
|
|
Symbol: params.FromToken.Symbol,
|
|
|
|
Recipient: params.ToAddr,
|
|
|
|
Amount: (*hexutil.Big)(params.AmountIn),
|
|
|
|
BonderFee: (*hexutil.Big)(params.BonderFee),
|
|
|
|
ChainID: params.ToChain.ChainID,
|
2024-03-25 12:40:00 +00:00
|
|
|
},
|
2024-06-05 07:56:02 +00:00
|
|
|
ChainID: params.FromChain.ChainID,
|
2024-03-25 12:40:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return h.BuildTransaction(sendArgs)
|
|
|
|
}
|
|
|
|
|
2024-06-06 20:08:25 +00:00
|
|
|
func (h *HopBridgeProcessor) GetContractAddress(params ProcessorInputParams) (common.Address, error) {
|
2024-06-05 07:56:02 +00:00
|
|
|
address, _, err := hop.GetContractAddress(params.FromChain.ChainID, params.FromToken.Symbol)
|
2024-05-23 12:38:39 +00:00
|
|
|
return address, err
|
2022-12-19 12:37:37 +00:00
|
|
|
}
|
|
|
|
|
2024-06-06 20:08:25 +00:00
|
|
|
func (h *HopBridgeProcessor) sendOrBuild(sendArgs *MultipathProcessorTxArgs, signerFn bind.SignerFn) (tx *ethTypes.Transaction, err error) {
|
2024-06-05 07:56:02 +00:00
|
|
|
fromChain := h.contractMaker.RPCClient.NetworkManager.Find(sendArgs.ChainID)
|
|
|
|
if fromChain == nil {
|
2024-04-25 14:43:58 +00:00
|
|
|
return tx, fmt.Errorf("ChainID not supported %d", sendArgs.ChainID)
|
2022-09-13 07:10:59 +00:00
|
|
|
}
|
2022-12-19 12:37:37 +00:00
|
|
|
|
2024-06-05 07:56:02 +00:00
|
|
|
token := h.tokenManager.FindToken(fromChain, sendArgs.HopTx.Symbol)
|
2024-05-23 12:38:39 +00:00
|
|
|
|
2024-06-05 07:56:02 +00:00
|
|
|
nonce, err := h.transactor.NextNonce(h.contractMaker.RPCClient, fromChain.ChainID, sendArgs.HopTx.From)
|
2022-12-19 12:37:37 +00:00
|
|
|
if err != nil {
|
2023-12-19 13:38:01 +00:00
|
|
|
return tx, err
|
2022-09-13 07:10:59 +00:00
|
|
|
}
|
2023-11-06 09:26:02 +00:00
|
|
|
|
2022-12-19 12:37:37 +00:00
|
|
|
argNonce := hexutil.Uint64(nonce)
|
|
|
|
sendArgs.HopTx.Nonce = &argNonce
|
2022-09-13 07:10:59 +00:00
|
|
|
|
2024-05-23 12:38:39 +00:00
|
|
|
txOpts := sendArgs.HopTx.ToTransactOpts(signerFn)
|
|
|
|
if token.IsNative() {
|
|
|
|
txOpts.Value = (*big.Int)(sendArgs.HopTx.Amount)
|
2023-09-29 17:56:27 +00:00
|
|
|
}
|
|
|
|
|
2024-06-05 07:56:02 +00:00
|
|
|
ethClient, err := h.contractMaker.RPCClient.EthClient(fromChain.ChainID)
|
2023-09-29 17:56:27 +00:00
|
|
|
if err != nil {
|
2024-05-23 12:38:39 +00:00
|
|
|
return tx, err
|
2022-09-13 07:10:59 +00:00
|
|
|
}
|
|
|
|
|
2024-06-05 07:56:02 +00:00
|
|
|
contractAddress, contractType, err := hop.GetContractAddress(fromChain.ChainID, sendArgs.HopTx.Symbol)
|
2022-09-13 07:10:59 +00:00
|
|
|
if err != nil {
|
2023-09-29 17:56:27 +00:00
|
|
|
return tx, err
|
2022-09-13 07:10:59 +00:00
|
|
|
}
|
2024-05-21 14:33:36 +00:00
|
|
|
|
2024-05-23 12:38:39 +00:00
|
|
|
switch contractType {
|
|
|
|
case hop.CctpL1Bridge:
|
|
|
|
return h.sendCctpL1BridgeTx(contractAddress, ethClient, sendArgs.HopTx.ChainID, sendArgs.HopTx.Recipient, txOpts)
|
|
|
|
case hop.L1Bridge:
|
|
|
|
return h.sendL1BridgeTx(contractAddress, ethClient, sendArgs.HopTx.ChainID, sendArgs.HopTx.Recipient, txOpts, token)
|
|
|
|
case hop.L2AmmWrapper:
|
|
|
|
return h.sendL2AmmWrapperTx(contractAddress, ethClient, sendArgs.HopTx.ChainID, sendArgs.HopTx.Recipient, txOpts)
|
|
|
|
case hop.CctpL2Bridge:
|
|
|
|
return h.sendCctpL2BridgeTx(contractAddress, ethClient, sendArgs.HopTx.ChainID, sendArgs.HopTx.Recipient, txOpts)
|
|
|
|
case hop.L2Bridge:
|
|
|
|
return h.sendL2BridgeTx(contractAddress, ethClient, sendArgs.HopTx.ChainID, sendArgs.HopTx.Recipient, txOpts)
|
2024-05-21 14:33:36 +00:00
|
|
|
}
|
|
|
|
|
2023-09-29 17:56:27 +00:00
|
|
|
return tx, err
|
2022-09-13 07:10:59 +00:00
|
|
|
}
|
|
|
|
|
2024-06-06 20:08:25 +00:00
|
|
|
func (h *HopBridgeProcessor) Send(sendArgs *MultipathProcessorTxArgs, verifiedAccount *account.SelectedExtKey) (hash types.Hash, err error) {
|
2024-05-23 12:38:39 +00:00
|
|
|
tx, err := h.sendOrBuild(sendArgs, getSigner(sendArgs.ChainID, sendArgs.HopTx.From, verifiedAccount))
|
2022-09-13 07:10:59 +00:00
|
|
|
if err != nil {
|
2024-05-23 12:38:39 +00:00
|
|
|
return types.Hash{}, err
|
2024-02-28 13:51:52 +00:00
|
|
|
}
|
2024-05-23 12:38:39 +00:00
|
|
|
return types.Hash(tx.Hash()), nil
|
|
|
|
}
|
2024-02-28 13:51:52 +00:00
|
|
|
|
2024-06-06 20:08:25 +00:00
|
|
|
func (h *HopBridgeProcessor) BuildTransaction(sendArgs *MultipathProcessorTxArgs) (*ethTypes.Transaction, error) {
|
2024-05-23 12:38:39 +00:00
|
|
|
return h.sendOrBuild(sendArgs, nil)
|
2022-09-13 07:10:59 +00:00
|
|
|
}
|
|
|
|
|
2024-06-06 20:08:25 +00:00
|
|
|
func (h *HopBridgeProcessor) CalculateFees(params ProcessorInputParams) (*big.Int, *big.Int, error) {
|
2024-05-16 07:37:36 +00:00
|
|
|
hopChainsMap := map[uint64]string{
|
|
|
|
walletCommon.EthereumMainnet: "ethereum",
|
|
|
|
walletCommon.OptimismMainnet: "optimism",
|
|
|
|
walletCommon.ArbitrumMainnet: "arbitrum",
|
|
|
|
}
|
2022-09-13 07:10:59 +00:00
|
|
|
|
2024-06-05 07:56:02 +00:00
|
|
|
fromChainName, ok := hopChainsMap[params.FromChain.ChainID]
|
2024-05-16 07:37:36 +00:00
|
|
|
if !ok {
|
|
|
|
return nil, nil, errors.New("from chain not supported")
|
2022-09-13 07:10:59 +00:00
|
|
|
}
|
|
|
|
|
2024-06-05 07:56:02 +00:00
|
|
|
toChainName, ok := hopChainsMap[params.ToChain.ChainID]
|
2024-05-16 07:37:36 +00:00
|
|
|
if !ok {
|
|
|
|
return nil, nil, errors.New("to chain not supported")
|
2022-09-13 07:10:59 +00:00
|
|
|
}
|
|
|
|
|
2024-06-05 07:56:02 +00:00
|
|
|
reqParams := netUrl.Values{}
|
|
|
|
reqParams.Add("amount", params.AmountIn.String())
|
|
|
|
reqParams.Add("token", params.FromToken.Symbol)
|
|
|
|
reqParams.Add("fromChain", fromChainName)
|
|
|
|
reqParams.Add("toChain", toChainName)
|
|
|
|
reqParams.Add("slippage", "0.5") // menas 0.5%
|
2024-05-14 19:11:16 +00:00
|
|
|
|
|
|
|
url := "https://api.hop.exchange/v1/quote"
|
2024-06-05 07:56:02 +00:00
|
|
|
response, err := h.httpClient.DoGetRequest(context.Background(), url, reqParams)
|
2022-09-13 07:10:59 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
|
2024-05-21 14:33:36 +00:00
|
|
|
h.bonderFee = &BonderFee{}
|
2024-05-14 19:11:16 +00:00
|
|
|
err = json.Unmarshal(response, h.bonderFee)
|
2022-09-13 07:10:59 +00:00
|
|
|
if err != nil {
|
2024-05-14 19:11:16 +00:00
|
|
|
return nil, nil, err
|
2022-09-13 07:10:59 +00:00
|
|
|
}
|
|
|
|
|
2024-05-23 12:38:39 +00:00
|
|
|
// Remove token fee from bonder fee as said here:
|
|
|
|
// https://docs.hop.exchange/v/developer-docs/api/api#get-v1-quote
|
|
|
|
// `bonderFee` - The suggested bonder fee for the amount in. The bonder fee also includes the cost of the destination transaction fee.
|
2024-06-06 20:08:25 +00:00
|
|
|
tokenFee := ZeroBigIntValue //new(big.Int).Sub(h.bonderFee.AmountIn.Int, h.bonderFee.EstimatedRecieved.Int)
|
2022-09-13 07:10:59 +00:00
|
|
|
|
2024-05-21 14:33:36 +00:00
|
|
|
return h.bonderFee.BonderFee.Int, tokenFee, nil
|
2022-09-13 07:10:59 +00:00
|
|
|
}
|
|
|
|
|
2024-06-06 20:08:25 +00:00
|
|
|
func (h *HopBridgeProcessor) CalculateAmountOut(params ProcessorInputParams) (*big.Int, error) {
|
2024-05-21 14:33:36 +00:00
|
|
|
return h.bonderFee.EstimatedRecieved.Int, nil
|
2022-09-13 07:10:59 +00:00
|
|
|
}
|
2024-05-23 12:38:39 +00:00
|
|
|
|
2024-06-06 20:08:25 +00:00
|
|
|
func (h *HopBridgeProcessor) packCctpL1BridgeTx(abi abi.ABI, toChainID uint64, to common.Address) ([]byte, error) {
|
2024-05-23 12:38:39 +00:00
|
|
|
return abi.Pack("send",
|
|
|
|
big.NewInt(int64(toChainID)),
|
|
|
|
to,
|
|
|
|
h.bonderFee.AmountIn.Int,
|
|
|
|
h.bonderFee.BonderFee.Int)
|
|
|
|
}
|
|
|
|
|
2024-06-06 20:08:25 +00:00
|
|
|
func (h *HopBridgeProcessor) sendCctpL1BridgeTx(contractAddress common.Address, ethClient chain.ClientInterface, toChainID uint64, to common.Address, txOpts *bind.TransactOpts) (tx *ethTypes.Transaction, err error) {
|
2024-05-23 12:38:39 +00:00
|
|
|
contractInstance, err := hopL1CctpImplementation.NewHopL1CctpImplementation(
|
|
|
|
contractAddress,
|
|
|
|
ethClient,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return tx, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return contractInstance.Send(
|
|
|
|
txOpts,
|
|
|
|
big.NewInt(int64(toChainID)),
|
|
|
|
to,
|
|
|
|
h.bonderFee.AmountIn.Int,
|
|
|
|
h.bonderFee.BonderFee.Int)
|
|
|
|
}
|
|
|
|
|
2024-06-06 20:08:25 +00:00
|
|
|
func (h *HopBridgeProcessor) packL1BridgeTx(abi abi.ABI, toChainID uint64, to common.Address) ([]byte, error) {
|
2024-05-23 12:38:39 +00:00
|
|
|
return abi.Pack("sendToL2",
|
|
|
|
big.NewInt(int64(toChainID)),
|
|
|
|
to,
|
|
|
|
h.bonderFee.AmountIn.Int,
|
|
|
|
h.bonderFee.AmountOutMin.Int,
|
|
|
|
big.NewInt(h.bonderFee.Deadline),
|
|
|
|
common.Address{},
|
2024-06-06 20:08:25 +00:00
|
|
|
ZeroBigIntValue)
|
2024-05-23 12:38:39 +00:00
|
|
|
}
|
|
|
|
|
2024-06-06 20:08:25 +00:00
|
|
|
func (h *HopBridgeProcessor) sendL1BridgeTx(contractAddress common.Address, ethClient chain.ClientInterface, toChainID uint64, to common.Address, txOpts *bind.TransactOpts, token *token.Token) (tx *ethTypes.Transaction, err error) {
|
2024-05-23 12:38:39 +00:00
|
|
|
if token.IsNative() {
|
|
|
|
contractInstance, err := hopL1EthBridge.NewHopL1EthBridge(
|
|
|
|
contractAddress,
|
|
|
|
ethClient,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return tx, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return contractInstance.SendToL2(
|
|
|
|
txOpts,
|
|
|
|
big.NewInt(int64(toChainID)),
|
|
|
|
to,
|
|
|
|
h.bonderFee.AmountIn.Int,
|
|
|
|
h.bonderFee.AmountOutMin.Int,
|
|
|
|
big.NewInt(h.bonderFee.Deadline),
|
|
|
|
common.Address{},
|
2024-06-06 20:08:25 +00:00
|
|
|
ZeroBigIntValue)
|
2024-05-23 12:38:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if token.Symbol == hopSymbol {
|
|
|
|
contractInstance, err := hopL1HopBridge.NewHopL1HopBridge(
|
|
|
|
contractAddress,
|
|
|
|
ethClient,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return tx, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return contractInstance.SendToL2(
|
|
|
|
txOpts,
|
|
|
|
big.NewInt(int64(toChainID)),
|
|
|
|
to,
|
|
|
|
h.bonderFee.AmountIn.Int,
|
|
|
|
h.bonderFee.AmountOutMin.Int,
|
|
|
|
big.NewInt(h.bonderFee.Deadline),
|
|
|
|
common.Address{},
|
2024-06-06 20:08:25 +00:00
|
|
|
ZeroBigIntValue)
|
2024-05-23 12:38:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
contractInstance, err := hopL1Erc20Bridge.NewHopL1Erc20Bridge(
|
|
|
|
contractAddress,
|
|
|
|
ethClient,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return tx, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return contractInstance.SendToL2(
|
|
|
|
txOpts,
|
|
|
|
big.NewInt(int64(toChainID)),
|
|
|
|
to,
|
|
|
|
h.bonderFee.AmountIn.Int,
|
|
|
|
h.bonderFee.AmountOutMin.Int,
|
|
|
|
big.NewInt(h.bonderFee.Deadline),
|
|
|
|
common.Address{},
|
2024-06-06 20:08:25 +00:00
|
|
|
ZeroBigIntValue)
|
2024-05-23 12:38:39 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2024-06-06 20:08:25 +00:00
|
|
|
func (h *HopBridgeProcessor) packCctpL2BridgeTx(abi abi.ABI, toChainID uint64, to common.Address) ([]byte, error) {
|
2024-05-23 12:38:39 +00:00
|
|
|
return abi.Pack("send",
|
|
|
|
big.NewInt(int64(toChainID)),
|
|
|
|
to,
|
|
|
|
h.bonderFee.AmountIn.Int,
|
|
|
|
h.bonderFee.BonderFee.Int)
|
|
|
|
}
|
|
|
|
|
2024-06-06 20:08:25 +00:00
|
|
|
func (h *HopBridgeProcessor) sendCctpL2BridgeTx(contractAddress common.Address, ethClient chain.ClientInterface, toChainID uint64, to common.Address, txOpts *bind.TransactOpts) (tx *ethTypes.Transaction, err error) {
|
2024-05-23 12:38:39 +00:00
|
|
|
contractInstance, err := hopL2CctpImplementation.NewHopL2CctpImplementation(
|
|
|
|
contractAddress,
|
|
|
|
ethClient,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return tx, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return contractInstance.Send(
|
|
|
|
txOpts,
|
|
|
|
big.NewInt(int64(toChainID)),
|
|
|
|
to,
|
|
|
|
h.bonderFee.AmountIn.Int,
|
|
|
|
h.bonderFee.BonderFee.Int,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2024-06-06 20:08:25 +00:00
|
|
|
func (h *HopBridgeProcessor) packL2AmmWrapperTx(abi abi.ABI, toChainID uint64, to common.Address) ([]byte, error) {
|
2024-05-23 12:38:39 +00:00
|
|
|
return abi.Pack("swapAndSend",
|
|
|
|
big.NewInt(int64(toChainID)),
|
|
|
|
to,
|
|
|
|
h.bonderFee.AmountIn.Int,
|
|
|
|
h.bonderFee.BonderFee.Int,
|
|
|
|
h.bonderFee.AmountOutMin.Int,
|
|
|
|
big.NewInt(h.bonderFee.Deadline),
|
|
|
|
h.bonderFee.DestinationAmountOutMin.Int,
|
|
|
|
big.NewInt(h.bonderFee.DestinationDeadline))
|
|
|
|
}
|
|
|
|
|
2024-06-06 20:08:25 +00:00
|
|
|
func (h *HopBridgeProcessor) sendL2AmmWrapperTx(contractAddress common.Address, ethClient chain.ClientInterface, toChainID uint64, to common.Address, txOpts *bind.TransactOpts) (tx *ethTypes.Transaction, err error) {
|
2024-05-23 12:38:39 +00:00
|
|
|
contractInstance, err := hopL2AmmWrapper.NewHopL2AmmWrapper(
|
|
|
|
contractAddress,
|
|
|
|
ethClient,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return tx, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return contractInstance.SwapAndSend(
|
|
|
|
txOpts,
|
|
|
|
big.NewInt(int64(toChainID)),
|
|
|
|
to,
|
|
|
|
h.bonderFee.AmountIn.Int,
|
|
|
|
h.bonderFee.BonderFee.Int,
|
|
|
|
h.bonderFee.AmountOutMin.Int,
|
|
|
|
big.NewInt(h.bonderFee.Deadline),
|
|
|
|
h.bonderFee.DestinationAmountOutMin.Int,
|
|
|
|
big.NewInt(h.bonderFee.DestinationDeadline))
|
|
|
|
}
|
|
|
|
|
2024-06-06 20:08:25 +00:00
|
|
|
func (h *HopBridgeProcessor) packL2BridgeTx(abi abi.ABI, toChainID uint64, to common.Address) ([]byte, error) {
|
2024-05-23 12:38:39 +00:00
|
|
|
return abi.Pack("send",
|
|
|
|
big.NewInt(int64(toChainID)),
|
|
|
|
to,
|
|
|
|
h.bonderFee.AmountIn.Int,
|
|
|
|
h.bonderFee.BonderFee.Int,
|
|
|
|
h.bonderFee.AmountOutMin.Int,
|
|
|
|
big.NewInt(h.bonderFee.Deadline))
|
|
|
|
}
|
|
|
|
|
2024-06-06 20:08:25 +00:00
|
|
|
func (h *HopBridgeProcessor) sendL2BridgeTx(contractAddress common.Address, ethClient chain.ClientInterface, toChainID uint64, to common.Address, txOpts *bind.TransactOpts) (tx *ethTypes.Transaction, err error) {
|
2024-05-23 12:38:39 +00:00
|
|
|
fromChainID := ethClient.NetworkID()
|
|
|
|
if fromChainID == walletCommon.OptimismMainnet ||
|
|
|
|
fromChainID == walletCommon.OptimismSepolia {
|
|
|
|
contractInstance, err := hopL2OptimismBridge.NewHopL2OptimismBridge(
|
|
|
|
contractAddress,
|
|
|
|
ethClient,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return tx, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return contractInstance.Send(
|
|
|
|
txOpts,
|
|
|
|
big.NewInt(int64(toChainID)),
|
|
|
|
to,
|
|
|
|
h.bonderFee.AmountIn.Int,
|
|
|
|
h.bonderFee.BonderFee.Int,
|
|
|
|
h.bonderFee.AmountOutMin.Int,
|
|
|
|
big.NewInt(h.bonderFee.Deadline))
|
|
|
|
}
|
|
|
|
if fromChainID == walletCommon.ArbitrumMainnet ||
|
|
|
|
fromChainID == walletCommon.ArbitrumSepolia {
|
|
|
|
contractInstance, err := hopL2ArbitrumBridge.NewHopL2ArbitrumBridge(
|
|
|
|
contractAddress,
|
|
|
|
ethClient,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return tx, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return contractInstance.Send(
|
|
|
|
txOpts,
|
|
|
|
big.NewInt(int64(toChainID)),
|
|
|
|
to,
|
|
|
|
h.bonderFee.AmountIn.Int,
|
|
|
|
h.bonderFee.BonderFee.Int,
|
|
|
|
h.bonderFee.AmountOutMin.Int,
|
|
|
|
big.NewInt(h.bonderFee.Deadline))
|
|
|
|
}
|
|
|
|
|
|
|
|
return tx, errors.New("tx for chain not supported yet")
|
|
|
|
}
|