Skip to content

Commit

Permalink
Add mantle multisig support (#11)
Browse files Browse the repository at this point in the history
  • Loading branch information
perrornet authored Aug 20, 2024
1 parent bea208d commit dbec732
Show file tree
Hide file tree
Showing 5 changed files with 710 additions and 3 deletions.
3 changes: 2 additions & 1 deletion utils/provider/bridge/routernitro/utils.go
Original file line number Diff line number Diff line change
Expand Up @@ -127,9 +127,10 @@ func (r Routernitro) GetBestQuote(ctx context.Context, args provider.SwapParams)
log.Debugf("#%d %s %s get quote error: %s", args.OrderId, msg, tokenIn.Name, err)
return
}

tokenAmount := quoteData.Get("destination").Get("tokenAmount").String()
if tokenAmount == "" {
log.Warnf("#%d %s %s get quote error: destination token amount is empty", args.OrderId, msg, tokenIn.Name)
log.Warnf("#%d %s %s get quote error: destination token amount is empty: %s", args.OrderId, msg, tokenIn.Name, quoteData.Raw)
return
}

Expand Down
4 changes: 2 additions & 2 deletions utils/wallets/dsafe/utils.go
Original file line number Diff line number Diff line change
Expand Up @@ -272,9 +272,9 @@ func (s *Dsafe) getOperatorAddress() common.Address {
}

func (s *Dsafe) nonce(ctx context.Context) (int64, error) {
u := fmt.Sprintf("https://dsafe.dcdao.box/cgw/v1/chains/46/safes/%s", s.getOperatorSafeAddress().Hex())
u := fmt.Sprintf("https://dsafe.dcdao.box/cgw/v1/chains/46/safes/%s/nonces", s.getOperatorSafeAddress().Hex())
var dest = struct {
RecommendedNonce int64 `json:"nonce"`
RecommendedNonce int64 `json:"recommendedNonce"`
}{}
return dest.RecommendedNonce, utils.Request(ctx, "GET", u, nil, &dest)
}
Expand Down
212 changes: 212 additions & 0 deletions utils/wallets/mantle_safe/safe.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,212 @@
package mantle_safe

import (
"context"
"encoding/json"
"fmt"
"omni-balance/utils"
"omni-balance/utils/chains"
"omni-balance/utils/constant"
"omni-balance/utils/error_types"
"omni-balance/utils/notice"
"omni-balance/utils/wallets"
"strings"
"time"

log "omni-balance/utils/logging"

"github.com/ethereum/go-ethereum/common"
"github.com/ethereum/go-ethereum/core/types"
"github.com/ethereum/go-ethereum/ethclient/simulated"
"github.com/pkg/errors"
"github.com/shopspring/decimal"
"github.com/spf13/cast"
"go.uber.org/zap/zapcore"
)

type MantleSafe struct {
conf wallets.WalletConfig
operatorSafe *MantleSafe
}

func NewMantleSafe(conf wallets.WalletConfig) *MantleSafe {
if conf.Operator.Address.Cmp(constant.ZeroAddress) == 0 {
panic("operator address is empty")
}
if conf.Operator.Address.Cmp(conf.Address) == 0 {
panic("operator address is same as safe address")
}
s := &MantleSafe{
conf: conf,
}
if conf.Operator.MultiSignType != "" {
operator := conf.Operator
s.operatorSafe = &MantleSafe{
conf: wallets.WalletConfig{
PrivateKey: operator.PrivateKey,
Address: operator.Address,
Operator: wallets.Operator{
Address: operator.Operator,
PrivateKey: operator.PrivateKey,
},
MultiSignType: operator.MultiSignType,
},
}
}
return s
}

func (*MantleSafe) IsSupportEip712() bool {
return false
}

func (s *MantleSafe) CheckFullAccess(ctx context.Context) error {
info, err := s.safeWalletInfo(ctx)
if err != nil {
return err
}
var owners []string
for _, v := range info.Owners {
owners = append(owners, v.Value)
}
if !utils.InArrayFold(s.conf.Operator.Address.Hex(), owners) {
return errors.New("operator is not in owners")
}
return nil
}

// GetAddress retrieves the address from the security configuration.
// If isReal is provided as true and the multisig type is set,
// it returns the operator's address; otherwise, it returns the default address.
// Parameters:
//
// isReal - An optional boolean indicating whether to return the real address.
//
// Returns:
//
// common.Address - The address from the configuration.
func (s *MantleSafe) GetAddress(isReal ...bool) common.Address {
if len(isReal) > 0 && isReal[0] {
if s.conf.Operator.MultiSignType == "" {
return s.conf.Address
}
return s.conf.Operator.Address
}
return s.conf.Address
}

func (s *MantleSafe) IsDifferentAddress() bool {
return true
}

func (s *MantleSafe) SignRawMessage(_ []byte) (sig []byte, err error) {
return nil, error_types.ErrUnsupportedWalletType
}

func (s *MantleSafe) GetExternalBalance(ctx context.Context, tokenAddress common.Address, decimals int32,
client simulated.Client) (decimal.Decimal, error) {
return s.GetBalance(ctx, tokenAddress, decimals, client)
}

func (s *MantleSafe) GetBalance(ctx context.Context, tokenAddress common.Address, decimals int32,
client simulated.Client) (decimal.Decimal, error) {
return chains.GetTokenBalance(ctx, client, tokenAddress.Hex(), s.GetAddress().Hex(), decimals)
}

func (s *MantleSafe) GetNonce(ctx context.Context, client simulated.Client) (uint64, error) {
if s.operatorSafe != nil {
nonce, err := s.operatorSafe.nonce(ctx)
log.Debugf("operator %s safe nonce: %d", s.conf.Operator.Address.Hex(), nonce)
return uint64(nonce), err
}
return client.NonceAt(ctx, s.GetAddress(), nil)
}

func (s *MantleSafe) SendTransaction(ctx context.Context, tx *types.LegacyTx, client simulated.Client) (common.Hash, error) {
return s.MultisigTransaction(ctx, tx, client)
}

func (s *MantleSafe) WaitTransaction(ctx context.Context, txHash common.Hash, _ simulated.Client) error {
var (
t = time.NewTicker(time.Second * 10)
count = 0
)
defer t.Stop()
for count < 60 {
select {
case <-ctx.Done():
return context.Canceled
case <-t.C:
tx, err := s.GetMultiSigTransaction(ctx, txHash)
if err != nil && strings.Contains(err.Error(), "No MultisigTransaction matches the given query") {
return errors.Wrap(err, "transaction not found")
}
if err != nil {
log.Debugf("wait transaction %s error: %s", txHash, err)
continue
}
if len(tx.DetailedExecutionInfo.Confirmations) < tx.DetailedExecutionInfo.ConfirmationsRequired {
count = 0
log.Infof("%s transaction %s confirmations: %d, required: %d,",
tx.SafeAddress, txHash, len(tx.DetailedExecutionInfo.Confirmations), tx.DetailedExecutionInfo.ConfirmationsRequired)
if err := notice.Send(ctx,
fmt.Sprintf("wait %s safeHash %s confirmations and execute.",
constant.GetChainName(s.GetChainIdByCtx(ctx)), txHash),
fmt.Sprintf("Please go to https://multisig.mantle.xyz/transactions/queue?safe=mantle:%s to confirm and execute #%d transaction.",
tx.SafeAddress, tx.DetailedExecutionInfo.Nonce),
zapcore.WarnLevel,
); err != nil {
log.Debugf("send notice error: %s", err)
}
continue
}
if tx.ExecutedAt == nil || *tx.ExecutedAt <= 0 {
count = 0
log.Debugf("transaction %s is not executed", txHash)
continue
}
if !strings.EqualFold(tx.TxStatus, "SUCCESS") {
log.Debugf("transaction %s status is %s", txHash, tx.TxStatus)
count++
continue
}
log.Debugf("transaction %s is successful", txHash)
return nil
}
}
return errors.Errorf("wait transaction %s timeout", txHash)
}

func (s *MantleSafe) MarshalJSON() ([]byte, error) {
return json.Marshal(map[string]interface{}{
"address": s.conf.Address.Hex(),
"operator": map[string]interface{}{
"address": s.conf.Operator.Address.Hex(),
"operator": s.conf.Operator.Operator.Hex(),
"multi_sign_type": s.conf.MultiSignType,
},
"multi_sign_type": s.conf.MultiSignType,
})
}

func (s *MantleSafe) GetRealHash(ctx context.Context, txHash common.Hash, client simulated.Client) (common.Hash, error) {
tx, err := s.GetMultiSigTransaction(ctx, txHash)
if err != nil {
return [32]byte{}, err
}
if len(tx.DetailedExecutionInfo.Confirmations) != tx.DetailedExecutionInfo.ConfirmationsRequired {
return [32]byte{}, errors.New("transaction not confirmed")
}
if tx.TxHash == "" {
return [32]byte{}, errors.New("transaction hash is empty")
}
txHashStr := cast.ToString(tx.TxHash)
if txHashStr == "" {
return [32]byte{}, errors.Errorf("transaction hash convert to hash error, the result is %+v", tx.TxHash)
}
return common.HexToHash(txHashStr), nil
}

func (s *MantleSafe) Name(_ context.Context) string {
return "dsafe"
}
Loading

0 comments on commit dbec732

Please sign in to comment.