package group_test import ( "testing" "time" "github.com/stretchr/testify/require" sdk "github.com/cosmos/cosmos-sdk/types" "github.com/cosmos/cosmos-sdk/x/group" ) var ( admin = sdk.AccAddress("admin") member1 = sdk.AccAddress("member1") member2 = sdk.AccAddress("member2") ) func TestMsgCreateGroup(t *testing.T) { testCases := []struct { name string msg *group.MsgCreateGroup expErr bool errMsg string }{ { "invalid admin address", &group.MsgCreateGroup{ Admin: "admin", }, true, "admin: decoding bech32 failed", }, { "invalid member address", &group.MsgCreateGroup{ Admin: admin.String(), Members: []group.Member{ group.Member{ Address: "invalid address", }, }, }, true, "address: decoding bech32 failed", }, { "negitive member's weight not allowed", &group.MsgCreateGroup{ Admin: admin.String(), Members: []group.Member{ group.Member{ Address: member1.String(), Weight: "-1", }, }, }, true, "expected a positive decimal", }, { "zero member's weight not allowed", &group.MsgCreateGroup{ Admin: admin.String(), Members: []group.Member{ group.Member{ Address: member1.String(), Weight: "0", }, }, }, true, "expected a positive decimal", }, { "duplicate member not allowed", &group.MsgCreateGroup{ Admin: admin.String(), Members: []group.Member{ group.Member{ Address: member1.String(), Weight: "1", Metadata: []byte("metadata"), }, group.Member{ Address: member1.String(), Weight: "1", Metadata: []byte("metadata"), }, }, }, true, "duplicate value", }, { "valid test case with single member", &group.MsgCreateGroup{ Admin: admin.String(), Members: []group.Member{ group.Member{ Address: member1.String(), Weight: "1", Metadata: []byte("metadata"), }, }, }, false, "", }, { "minimum fields", &group.MsgCreateGroup{ Admin: admin.String(), Members: []group.Member{}, }, false, "", }, { "valid test case with multiple members", &group.MsgCreateGroup{ Admin: admin.String(), Members: []group.Member{ group.Member{ Address: member1.String(), Weight: "1", Metadata: []byte("metadata"), }, group.Member{ Address: member2.String(), Weight: "1", Metadata: []byte("metadata"), }, }, }, false, "", }, } for _, tc := range testCases { t.Run(tc.name, func(t *testing.T) { err := tc.msg.ValidateBasic() if tc.expErr { require.Error(t, err) require.Contains(t, err.Error(), tc.errMsg) } else { require.NoError(t, err) require.Equal(t, tc.msg.Type(), group.TypeMsgCreateGroup) } }) } } func TestMsgUpdateGroupAdmin(t *testing.T) { testCases := []struct { name string msg *group.MsgUpdateGroupAdmin expErr bool errMsg string }{ { "empty group id", &group.MsgUpdateGroupAdmin{ Admin: admin.String(), NewAdmin: member1.String(), }, true, "group id: value is empty", }, { "admin: invalid bech32 address", &group.MsgUpdateGroupAdmin{ GroupId: 1, Admin: "admin", }, true, "admin: decoding bech32 failed", }, { "new admin: invalid bech32 address", &group.MsgUpdateGroupAdmin{ GroupId: 1, Admin: admin.String(), NewAdmin: "new-admin", }, true, "new admin: decoding bech32 failed", }, { "admin & new admin is same", &group.MsgUpdateGroupAdmin{ GroupId: 1, Admin: admin.String(), NewAdmin: admin.String(), }, true, "new and old admin are the same", }, { "valid case", &group.MsgUpdateGroupAdmin{ GroupId: 1, Admin: admin.String(), NewAdmin: member1.String(), }, false, "", }, } for _, tc := range testCases { t.Run(tc.name, func(t *testing.T) { err := tc.msg.ValidateBasic() if tc.expErr { require.Error(t, err) require.Contains(t, err.Error(), tc.errMsg) } else { require.NoError(t, err) require.Equal(t, tc.msg.Type(), group.TypeMsgUpdateGroupAdmin) } }) } } func TestMsgUpdateGroupMetadata(t *testing.T) { testCases := []struct { name string msg *group.MsgUpdateGroupMetadata expErr bool errMsg string }{ { "empty group id", &group.MsgUpdateGroupMetadata{ Admin: admin.String(), }, true, "group id: value is empty", }, { "admin: invalid bech32 address", &group.MsgUpdateGroupMetadata{ GroupId: 1, Admin: "admin", }, true, "admin: decoding bech32 failed", }, { "valid test", &group.MsgUpdateGroupMetadata{ GroupId: 1, Admin: admin.String(), Metadata: []byte("metadata"), }, false, "", }, } for _, tc := range testCases { t.Run(tc.name, func(t *testing.T) { err := tc.msg.ValidateBasic() if tc.expErr { require.Error(t, err) require.Contains(t, err.Error(), tc.errMsg) } else { require.NoError(t, err) require.Equal(t, tc.msg.Type(), group.TypeMsgUpdateGroupMetadata) } }) } } func TestMsgUpdateGroupMembers(t *testing.T) { testCases := []struct { name string msg *group.MsgUpdateGroupMembers expErr bool errMsg string }{ { "empty group id", &group.MsgUpdateGroupMembers{}, true, "group id: value is empty", }, { "admin: invalid bech32 address", &group.MsgUpdateGroupMembers{ GroupId: 1, Admin: "admin", }, true, "admin: decoding bech32 failed", }, { "empty member list", &group.MsgUpdateGroupMembers{ GroupId: 1, Admin: admin.String(), MemberUpdates: []group.Member{}, }, true, "member updates: value is empty", }, { "valid test", &group.MsgUpdateGroupMembers{ GroupId: 1, Admin: admin.String(), MemberUpdates: []group.Member{ group.Member{ Address: member1.String(), Weight: "1", Metadata: []byte("metadata"), }, }, }, false, "", }, { "valid test with zero weight", &group.MsgUpdateGroupMembers{ GroupId: 1, Admin: admin.String(), MemberUpdates: []group.Member{ group.Member{ Address: member1.String(), Weight: "0", Metadata: []byte("metadata"), }, }, }, false, "", }, } for _, tc := range testCases { t.Run(tc.name, func(t *testing.T) { err := tc.msg.ValidateBasic() if tc.expErr { require.Error(t, err) require.Contains(t, err.Error(), tc.errMsg) } else { require.NoError(t, err) require.Equal(t, tc.msg.Type(), group.TypeMsgUpdateGroupMembers) } }) } } func TestMsgCreateGroupPolicy(t *testing.T) { testCases := []struct { name string msg func() *group.MsgCreateGroupPolicy expErr bool errMsg string }{ { "empty group id", func() *group.MsgCreateGroupPolicy { return &group.MsgCreateGroupPolicy{ Admin: admin.String(), } }, true, "group id: value is empty", }, { "admin: invalid bech32 address", func() *group.MsgCreateGroupPolicy { return &group.MsgCreateGroupPolicy{ Admin: "admin", GroupId: 1, } }, true, "admin: decoding bech32 failed", }, { "invalid threshold policy", func() *group.MsgCreateGroupPolicy { policy := group.NewThresholdDecisionPolicy("-1", time.Second) req, err := group.NewMsgCreateGroupPolicy(admin, 1, []byte("metadata"), policy) require.NoError(t, err) return req }, true, "expected a positive decimal", }, { "valid test case", func() *group.MsgCreateGroupPolicy { policy := group.NewThresholdDecisionPolicy("1", time.Second) req, err := group.NewMsgCreateGroupPolicy(admin, 1, []byte("metadata"), policy) require.NoError(t, err) return req }, false, "", }, } for _, tc := range testCases { t.Run(tc.name, func(t *testing.T) { msg := tc.msg() err := msg.ValidateBasic() if tc.expErr { require.Error(t, err) require.Contains(t, err.Error(), tc.errMsg) } else { require.NoError(t, err) require.Equal(t, msg.Type(), group.TypeMsgCreateGroupPolicy) } }) } } func TestMsgUpdateGroupPolicyDecisionPolicy(t *testing.T) { validPolicy := group.NewThresholdDecisionPolicy("1", time.Second) msg1, err := group.NewMsgUpdateGroupPolicyDecisionPolicyRequest(admin, member1, validPolicy) require.NoError(t, err) invalidPolicy := group.NewThresholdDecisionPolicy("-1", time.Second) msg2, err := group.NewMsgUpdateGroupPolicyDecisionPolicyRequest(admin, member2, invalidPolicy) require.NoError(t, err) testCases := []struct { name string msg *group.MsgUpdateGroupPolicyDecisionPolicy expErr bool errMsg string }{ { "admin: invalid bech32 address", &group.MsgUpdateGroupPolicyDecisionPolicy{ Admin: "admin", }, true, "admin: decoding bech32 failed", }, { "group policy: invalid bech32 address", &group.MsgUpdateGroupPolicyDecisionPolicy{ Admin: admin.String(), Address: "address", }, true, "group policy: decoding bech32 failed", }, { "group policy: invalid bech32 address", &group.MsgUpdateGroupPolicyDecisionPolicy{ Admin: admin.String(), Address: "address", }, true, "group policy: decoding bech32 failed", }, { "invalid decision policy", msg2, true, "decision policy: threshold: expected a positive decimal", }, { "invalid decision policy", msg1, false, "", }, } for _, tc := range testCases { t.Run(tc.name, func(t *testing.T) { msg := tc.msg err := msg.ValidateBasic() if tc.expErr { require.Error(t, err) require.Contains(t, err.Error(), tc.errMsg) } else { require.NoError(t, err) require.Equal(t, msg.Type(), group.TypeMsgUpdateGroupPolicyDecisionPolicy) } }) } } func TestMsgUpdateGroupPolicyAdmin(t *testing.T) { testCases := []struct { name string msg *group.MsgUpdateGroupPolicyAdmin expErr bool errMsg string }{ { "admin: invalid bech32 address", &group.MsgUpdateGroupPolicyAdmin{ Admin: "admin", }, true, "admin: decoding bech32 failed", }, { "policy address: invalid bech32 address", &group.MsgUpdateGroupPolicyAdmin{ Admin: admin.String(), NewAdmin: member1.String(), Address: "address", }, true, "group policy: decoding bech32 failed", }, { "new admin: invalid bech32 address", &group.MsgUpdateGroupPolicyAdmin{ Admin: admin.String(), Address: admin.String(), NewAdmin: "new-admin", }, true, "new admin: decoding bech32 failed", }, { "same old and new admin", &group.MsgUpdateGroupPolicyAdmin{ Admin: admin.String(), Address: admin.String(), NewAdmin: admin.String(), }, true, "new and old admin are same", }, { "valid test", &group.MsgUpdateGroupPolicyAdmin{ Admin: admin.String(), Address: admin.String(), NewAdmin: member1.String(), }, false, "", }, } for _, tc := range testCases { t.Run(tc.name, func(t *testing.T) { msg := tc.msg err := msg.ValidateBasic() if tc.expErr { require.Error(t, err) require.Contains(t, err.Error(), tc.errMsg) } else { require.NoError(t, err) require.Equal(t, msg.Type(), group.TypeMsgUpdateGroupPolicyAdmin) } }) } } func TestMsgUpdateGroupPolicyMetadata(t *testing.T) { testCases := []struct { name string msg *group.MsgUpdateGroupPolicyMetadata expErr bool errMsg string }{ { "admin: invalid bech32 address", &group.MsgUpdateGroupPolicyMetadata{ Admin: "admin", }, true, "admin: decoding bech32 failed", }, { "group policy address: invalid bech32 address", &group.MsgUpdateGroupPolicyMetadata{ Admin: admin.String(), Address: "address", }, true, "group policy: decoding bech32 failed", }, { "valid testcase", &group.MsgUpdateGroupPolicyMetadata{ Admin: admin.String(), Address: member1.String(), Metadata: []byte("metadata"), }, false, "", }, } for _, tc := range testCases { t.Run(tc.name, func(t *testing.T) { msg := tc.msg err := msg.ValidateBasic() if tc.expErr { require.Error(t, err) require.Contains(t, err.Error(), tc.errMsg) } else { require.NoError(t, err) require.Equal(t, msg.Type(), group.TypeMsgUpdateGroupPolicyMetadata) } }) } } func TestMsgCreateProposal(t *testing.T) { testCases := []struct { name string msg *group.MsgCreateProposal expErr bool errMsg string }{ { "invalid group policy address", &group.MsgCreateProposal{ Address: "address", }, true, "group policy: decoding bech32 failed", }, { "proposers required", &group.MsgCreateProposal{ Address: admin.String(), }, true, "proposers: value is empty", }, { "valid testcase", &group.MsgCreateProposal{ Address: admin.String(), Proposers: []string{member1.String(), member2.String()}, }, false, "", }, } for _, tc := range testCases { t.Run(tc.name, func(t *testing.T) { msg := tc.msg err := msg.ValidateBasic() if tc.expErr { require.Error(t, err) require.Contains(t, err.Error(), tc.errMsg) } else { require.NoError(t, err) require.Equal(t, msg.Type(), group.TypeMsgCreateProposal) } }) } } func TestMsgVote(t *testing.T) { testCases := []struct { name string msg *group.MsgVote expErr bool errMsg string }{ { "invalid voter address", &group.MsgVote{ Voter: "voter", }, true, "voter: decoding bech32 failed", }, { "proposal id is required", &group.MsgVote{ Voter: member1.String(), }, true, "proposal id: value is empty", }, { "unspecified vote choice", &group.MsgVote{ Voter: member1.String(), ProposalId: 1, }, true, "choice: value is empty", }, { "valid test case", &group.MsgVote{ Voter: member1.String(), ProposalId: 1, Choice: group.Choice_CHOICE_YES, }, false, "", }, } for _, tc := range testCases { t.Run(tc.name, func(t *testing.T) { msg := tc.msg err := msg.ValidateBasic() if tc.expErr { require.Error(t, err) require.Contains(t, err.Error(), tc.errMsg) } else { require.NoError(t, err) require.Equal(t, msg.Type(), group.TypeMsgVote) } }) } } func TestMsgExec(t *testing.T) { testCases := []struct { name string msg *group.MsgExec expErr bool errMsg string }{ { "invalid signer address", &group.MsgExec{ Signer: "signer", }, true, "signer: decoding bech32 failed", }, { "proposal is required", &group.MsgExec{ Signer: admin.String(), }, true, "proposal id: value is empty", }, { "valid testcase", &group.MsgExec{ Signer: admin.String(), ProposalId: 1, }, false, "", }, } for _, tc := range testCases { t.Run(tc.name, func(t *testing.T) { msg := tc.msg err := msg.ValidateBasic() if tc.expErr { require.Error(t, err) require.Contains(t, err.Error(), tc.errMsg) } else { require.NoError(t, err) require.Equal(t, msg.Type(), group.TypeMsgExec) } }) } }