mukan-ibc/e2e/tests/interchain_accounts/groups_test.go
Mukan Erkin Törük 88dd97a9f8
Some checks failed
CodeQL / Analyze (push) Waiting to run
golangci-lint / lint (push) Waiting to run
Tests / Code Coverage / build (amd64) (push) Waiting to run
Tests / Code Coverage / build (arm64) (push) Waiting to run
Tests / Code Coverage / unit-tests (map[additional-args:-tags="test_e2e" name:e2e path:./e2e]) (push) Waiting to run
Tests / Code Coverage / unit-tests (map[name:08-wasm path:./modules/light-clients/08-wasm]) (push) Waiting to run
Tests / Code Coverage / unit-tests (map[name:ibc-go path:.]) (push) Waiting to run
Docker Build & Push Simapp (main) / docker-build (push) Has been cancelled
refactor: replace all github.com upstream refs with git.cw.tr/mukan-network
2026-05-11 03:36:22 +03:00

221 lines
8.2 KiB
Go

//go:build !test_e2e
package interchainaccounts
import (
"context"
"testing"
"time"
"github.com/cosmos/gogoproto/proto"
interchaintest "github.com/cosmos/interchaintest/v10"
"github.com/cosmos/interchaintest/v10/ibc"
test "github.com/cosmos/interchaintest/v10/testutil"
testifysuite "github.com/stretchr/testify/suite"
sdkmath "cosmossdk.io/math"
sdk "git.cw.tr/mukan-network/mukan-sdk/types"
banktypes "git.cw.tr/mukan-network/mukan-sdk/x/bank/types"
grouptypes "git.cw.tr/mukan-network/mukan-sdk/x/group"
"github.com/cosmos/ibc-go/e2e/testsuite"
"github.com/cosmos/ibc-go/e2e/testsuite/query"
"github.com/cosmos/ibc-go/e2e/testvalues"
controllertypes "git.cw.tr/mukan-network/mukan-ibc/modules/apps/27-interchain-accounts/controller/types"
icatypes "git.cw.tr/mukan-network/mukan-ibc/modules/apps/27-interchain-accounts/types"
channeltypes "git.cw.tr/mukan-network/mukan-ibc/modules/core/04-channel/types"
ibctesting "git.cw.tr/mukan-network/mukan-ibc/testing"
)
const (
// DefaultGroupMemberWeight is the members voting weight.
// A group members weight is used in the sum of `YES` votes required to meet a decision policy threshold.
DefaultGroupMemberWeight = "1"
// DefaultGroupThreshold is the minimum weighted sum of `YES` votes that must be met or
// exceeded for a proposal to succeed.
DefaultGroupThreshold = "1"
// DefaultMetadata defines a reusable metadata string for testing purposes
DefaultMetadata = "custom metadata"
// DefaultMinExecutionPeriod is the minimum duration after the proposal submission
// where members can start sending MsgExec. This means that the window for
// sending a MsgExec transaction is:
// `[ submission + min_execution_period ; submission + voting_period + max_execution_period]`
// where max_execution_period is an app-specific config, defined in the keeper.
// If not set, min_execution_period will default to 0.
DefaultMinExecutionPeriod = time.Duration(0)
// DefaultVotingPeriod is the duration from submission of a proposal to the end of voting period
// Within this times votes can be submitted with MsgVote.
DefaultVotingPeriod = time.Minute
// InitialGroupID is the first group ID generated by x/group
InitialGroupID = 1
// InitialProposalID is the first group proposal ID generated by x/group
InitialProposalID = 1
)
// compatibility:from_version: v7.10.0
func TestInterchainAccountsGroupsTestSuite(t *testing.T) {
testifysuite.Run(t, new(InterchainAccountsGroupsTestSuite))
}
type InterchainAccountsGroupsTestSuite struct {
testsuite.E2ETestSuite
}
// SetupSuite sets up chains for the current test suite
func (s *InterchainAccountsGroupsTestSuite) SetupSuite() {
s.SetupChains(context.TODO(), 2, nil)
}
func (s *InterchainAccountsGroupsTestSuite) QueryGroupPolicyAddress(ctx context.Context, chain ibc.Chain) string {
res, err := query.GRPCQuery[grouptypes.QueryGroupPoliciesByGroupResponse](ctx, chain, &grouptypes.QueryGroupPoliciesByGroupRequest{
GroupId: InitialGroupID, // always use the initial group id
})
s.Require().NoError(err)
return res.GroupPolicies[0].Address
}
func (s *InterchainAccountsGroupsTestSuite) TestInterchainAccountsGroupsIntegration() {
t := s.T()
ctx := context.TODO()
var (
groupPolicyAddr string
interchainAccAddr string
err error
)
testName := t.Name()
relayer := s.CreateDefaultPaths(testName)
chainA, chainB := s.GetChains()
chainAWallet := s.CreateUserOnChainA(ctx, testvalues.StartingTokenAmount)
chainAAddress := chainAWallet.FormattedAddress()
chainBWallet := s.CreateUserOnChainB(ctx, testvalues.StartingTokenAmount)
chainBAddress := chainBWallet.FormattedAddress()
t.Run("create group with new threshold decision policy", func(t *testing.T) {
members := []grouptypes.MemberRequest{
{
Address: chainAAddress,
Weight: DefaultGroupMemberWeight,
},
}
decisionPolicy := grouptypes.NewThresholdDecisionPolicy(DefaultGroupThreshold, DefaultVotingPeriod, DefaultMinExecutionPeriod)
msgCreateGroupWithPolicy, err := grouptypes.NewMsgCreateGroupWithPolicy(chainAAddress, members, DefaultMetadata, DefaultMetadata, true, decisionPolicy)
s.Require().NoError(err)
txResp := s.BroadcastMessages(ctx, chainA, chainAWallet, msgCreateGroupWithPolicy)
s.AssertTxSuccess(txResp)
})
t.Run("submit proposal for MsgRegisterInterchainAccount", func(t *testing.T) {
groupPolicyAddr = s.QueryGroupPolicyAddress(ctx, chainA)
msgRegisterAccount := controllertypes.NewMsgRegisterInterchainAccount(ibctesting.FirstConnectionID, groupPolicyAddr, icatypes.NewDefaultMetadataString(ibctesting.FirstConnectionID, ibctesting.FirstConnectionID), channeltypes.ORDERED)
msgSubmitProposal, err := grouptypes.NewMsgSubmitProposal(groupPolicyAddr, []string{chainAAddress}, []sdk.Msg{msgRegisterAccount}, DefaultMetadata, grouptypes.Exec_EXEC_UNSPECIFIED, "e2e groups proposal: for MsgRegisterInterchainAccount", "e2e groups proposal: for MsgRegisterInterchainAccount")
s.Require().NoError(err)
txResp := s.BroadcastMessages(ctx, chainA, chainAWallet, msgSubmitProposal)
s.AssertTxSuccess(txResp)
})
t.Run("vote and exec proposal", func(t *testing.T) {
msgVote := &grouptypes.MsgVote{
ProposalId: InitialProposalID,
Voter: chainAAddress,
Option: grouptypes.VOTE_OPTION_YES,
Exec: grouptypes.Exec_EXEC_TRY,
}
txResp := s.BroadcastMessages(ctx, chainA, chainAWallet, msgVote)
s.AssertTxSuccess(txResp)
})
t.Run("start relayer", func(t *testing.T) {
s.StartRelayer(relayer, testName)
})
t.Run("verify interchain account registration success", func(t *testing.T) {
interchainAccAddr, err = query.InterchainAccount(ctx, chainA, groupPolicyAddr, ibctesting.FirstConnectionID)
s.Require().NotEmpty(interchainAccAddr)
s.Require().NoError(err)
channels, err := relayer.GetChannels(ctx, s.GetRelayerExecReporter(), chainA.Config().ChainID)
s.Require().NoError(err)
s.Require().Equal(len(channels), 2) // 1 transfer (created by default), 1 interchain-accounts
})
t.Run("fund interchain account wallet", func(t *testing.T) {
err := chainB.SendFunds(ctx, interchaintest.FaucetAccountKeyName, ibc.WalletAmount{
Address: interchainAccAddr,
Amount: sdkmath.NewInt(testvalues.StartingTokenAmount),
Denom: chainB.Config().Denom,
})
s.Require().NoError(err)
})
t.Run("submit proposal for MsgSendTx", func(t *testing.T) {
msgBankSend := &banktypes.MsgSend{
FromAddress: interchainAccAddr,
ToAddress: chainBAddress,
Amount: sdk.NewCoins(testvalues.DefaultTransferAmount(chainB.Config().Denom)),
}
cdc := testsuite.Codec()
bz, err := icatypes.SerializeCosmosTx(cdc, []proto.Message{msgBankSend}, icatypes.EncodingProtobuf)
s.Require().NoError(err)
packetData := icatypes.InterchainAccountPacketData{
Type: icatypes.EXECUTE_TX,
Data: bz,
Memo: "e2e",
}
msgSubmitTx := controllertypes.NewMsgSendTx(groupPolicyAddr, ibctesting.FirstConnectionID, uint64(time.Hour.Nanoseconds()), packetData)
msgSubmitProposal, err := grouptypes.NewMsgSubmitProposal(groupPolicyAddr, []string{chainAAddress}, []sdk.Msg{msgSubmitTx}, DefaultMetadata, grouptypes.Exec_EXEC_UNSPECIFIED, "e2e groups proposal: for MsgRegisterInterchainAccount", "e2e groups proposal: for MsgRegisterInterchainAccount")
s.Require().NoError(err)
txResp := s.BroadcastMessages(ctx, chainA, chainAWallet, msgSubmitProposal)
s.AssertTxSuccess(txResp)
})
t.Run("vote and exec proposal", func(t *testing.T) {
msgVote := &grouptypes.MsgVote{
ProposalId: InitialProposalID + 1,
Voter: chainAAddress,
Option: grouptypes.VOTE_OPTION_YES,
Exec: grouptypes.Exec_EXEC_TRY,
}
txResp := s.BroadcastMessages(ctx, chainA, chainAWallet, msgVote)
s.AssertTxSuccess(txResp)
})
t.Run("verify tokens transferred", func(t *testing.T) {
s.Require().NoError(test.WaitForBlocks(ctx, 10, chainA, chainB), "failed to wait for blocks")
balance, err := query.Balance(ctx, chainB, chainBAddress, chainB.Config().Denom)
s.Require().NoError(err)
expected := testvalues.IBCTransferAmount + testvalues.StartingTokenAmount
s.Require().Equal(expected, balance.Int64())
balance, err = query.Balance(ctx, chainB, interchainAccAddr, chainB.Config().Denom)
s.Require().NoError(err)
expected = testvalues.StartingTokenAmount - testvalues.IBCTransferAmount
s.Require().Equal(expected, balance.Int64())
})
}