package protocol import ( "context" "crypto/ecdsa" "encoding/json" "errors" "io/ioutil" "testing" "github.com/google/uuid" "github.com/stretchr/testify/suite" "go.uber.org/zap" "github.com/status-im/status-go/account/generator" gethbridge "github.com/status-im/status-go/eth-node/bridge/geth" "github.com/status-im/status-go/eth-node/crypto" "github.com/status-im/status-go/eth-node/types" "github.com/status-im/status-go/multiaccounts" "github.com/status-im/status-go/multiaccounts/settings" "github.com/status-im/status-go/params" "github.com/status-im/status-go/protocol/common" "github.com/status-im/status-go/protocol/communities" "github.com/status-im/status-go/protocol/protobuf" "github.com/status-im/status-go/protocol/requests" "github.com/status-im/status-go/protocol/sqlite" "github.com/status-im/status-go/protocol/tt" "github.com/status-im/status-go/waku" ) func TestAdminMessengerCommunitiesSuite(t *testing.T) { suite.Run(t, new(AdminMessengerCommunitiesSuite)) } type AdminMessengerCommunitiesSuite struct { suite.Suite owner *Messenger admin *Messenger alice *Messenger // If one wants to send messages between different instances of Messenger, // a single Waku service should be shared. shh types.Waku logger *zap.Logger } func (s *AdminMessengerCommunitiesSuite) SetupTest() { s.logger = tt.MustCreateTestLogger() config := waku.DefaultConfig config.MinimumAcceptedPoW = 0 shh := waku.New(&config, s.logger) s.shh = gethbridge.NewGethWakuWrapper(shh) s.Require().NoError(shh.Start()) s.owner = s.newMessenger() s.admin = s.newMessenger() s.alice = s.newMessenger() _, err := s.owner.Start() s.Require().NoError(err) _, err = s.admin.Start() s.Require().NoError(err) _, err = s.alice.Start() s.Require().NoError(err) } func (s *AdminMessengerCommunitiesSuite) TearDownTest() { s.Require().NoError(s.owner.Shutdown()) s.Require().NoError(s.admin.Shutdown()) s.Require().NoError(s.alice.Shutdown()) _ = s.logger.Sync() } func (s *AdminMessengerCommunitiesSuite) newMessengerWithOptions(shh types.Waku, privateKey *ecdsa.PrivateKey, options []Option) *Messenger { m, err := NewMessenger( "Test", privateKey, &testNode{shh: shh}, uuid.New().String(), nil, nil, options..., ) s.Require().NoError(err) err = m.Init() s.Require().NoError(err) config := params.NodeConfig{ NetworkID: 10, DataDir: "test", } networks := json.RawMessage("{}") setting := settings.Settings{ Address: types.HexToAddress("0x1122334455667788990011223344556677889900"), AnonMetricsShouldSend: false, CurrentNetwork: "mainnet_rpc", DappsAddress: types.HexToAddress("0x1122334455667788990011223344556677889900"), InstallationID: "d3efcff6-cffa-560e-a547-21d3858cbc51", KeyUID: "0x1122334455667788990011223344556677889900", Name: "Test", Networks: &networks, PhotoPath: "", PreviewPrivacy: false, PublicKey: "0x04112233445566778899001122334455667788990011223344556677889900112233445566778899001122334455667788990011223344556677889900", SigningPhrase: "yurt joey vibe", SendPushNotifications: true, ProfilePicturesVisibility: 1, DefaultSyncPeriod: 777600, UseMailservers: true, LinkPreviewRequestEnabled: true, SendStatusUpdates: true, WalletRootAddress: types.HexToAddress("0x1122334455667788990011223344556677889900")} _ = m.settings.CreateSettings(setting, config) return m } func (s *AdminMessengerCommunitiesSuite) newMessengerWithKey(shh types.Waku, privateKey *ecdsa.PrivateKey) *Messenger { tmpfile, err := ioutil.TempFile("", "accounts-tests-") s.Require().NoError(err) madb, err := multiaccounts.InitializeDB(tmpfile.Name()) s.Require().NoError(err) acc := generator.NewAccount(privateKey, nil) iai := acc.ToIdentifiedAccountInfo("") options := []Option{ WithCustomLogger(s.logger), WithDatabaseConfig(":memory:", "somekey", sqlite.ReducedKDFIterationsNumber), WithMultiAccounts(madb), WithAccount(iai.ToMultiAccount()), WithDatasync(), } return s.newMessengerWithOptions(shh, privateKey, options) } func (s *AdminMessengerCommunitiesSuite) newMessenger() *Messenger { privateKey, err := crypto.GenerateKey() s.Require().NoError(err) return s.newMessengerWithKey(s.shh, privateKey) } func (s *AdminMessengerCommunitiesSuite) TestAdminEditCommunityDescription() { // TODO admin test: update to include edit tags, logo, banner, request to join required setting, pin setting, etc... community := s.setUpCommunityAndRoles() s.adminEditsCommunityDescription(community) } func (s *AdminMessengerCommunitiesSuite) TestAdminCreateEditDeleteChannels() { community := s.setUpCommunityAndRoles() newAdminChat := &protobuf.CommunityChat{ Permissions: &protobuf.CommunityPermissions{ Access: protobuf.CommunityPermissions_NO_MEMBERSHIP, }, Identity: &protobuf.ChatIdentity{ DisplayName: "chat from admin", Emoji: "", Description: "chat created by an admin", }, } newChatID := s.adminCreateCommunityChannel(community, newAdminChat) newAdminChat.Identity.DisplayName = "modified chat from admin" s.adminEditCommunityChannel(community, newAdminChat, newChatID) s.adminDeleteCommunityChannel(community, newChatID) } func (s *AdminMessengerCommunitiesSuite) TestAdminCreateEditDeleteBecomeMemberPermission() { community := s.setUpCommunityAndRoles() // first, create token permission tokenPermissionID, createTokenPermission := s.adminCreateTestTokenPermission(community) createTokenPermission.TokenCriteria[0].Symbol = "UPDATED" createTokenPermission.TokenCriteria[0].Amount = "200" editTokenPermission := &requests.EditCommunityTokenPermission{ PermissionID: tokenPermissionID, CreateCommunityTokenPermission: *createTokenPermission, } // then, admin edits the permission s.adminEditTokenPermission(community, editTokenPermission) deleteTokenPermission := &requests.DeleteCommunityTokenPermission{ CommunityID: community.ID(), PermissionID: tokenPermissionID, } // then, admin deletes previously created token permission s.adminDeleteTokenPermission(community, deleteTokenPermission) } func (s *AdminMessengerCommunitiesSuite) TestAdminCannotCreateBecomeAdminPermission() { community := s.setUpCommunityAndRoles() permissionRequest := createTestPermissionRequest(community) permissionRequest.Type = protobuf.CommunityTokenPermission_BECOME_ADMIN response, err := s.admin.CreateCommunityTokenPermission(permissionRequest) s.Require().Nil(response) s.Require().Error(err) } func (s *AdminMessengerCommunitiesSuite) TestAdminCannotEditBecomeAdminPermission() { community := s.setUpCommunityAndRoles() permissionRequest := createTestPermissionRequest(community) permissionRequest.Type = protobuf.CommunityTokenPermission_BECOME_ADMIN // owner creates BECOME_ADMIN permission response, err := s.owner.CreateCommunityTokenPermission(permissionRequest) s.Require().NoError(err) var tokenPermissionID string for id := range response.CommunityChanges[0].TokenPermissionsAdded { tokenPermissionID = id } s.Require().NotEqual(tokenPermissionID, "") ownerCommunity, err := s.owner.communitiesManager.GetByID(community.ID()) s.Require().NoError(err) s.assertAdminTokenPermissionCreated(ownerCommunity) // then, ensure admin receives updated community _, err = WaitOnMessengerResponse( s.admin, func(r *MessengerResponse) bool { return len(r.Communities()) > 0 }, "admin did not receive updated community", ) s.Require().NoError(err) adminCommunity, err := s.admin.communitiesManager.GetByID(community.ID()) s.Require().NoError(err) s.assertAdminTokenPermissionCreated(adminCommunity) permissionRequest.TokenCriteria[0].Symbol = "UPDATED" permissionRequest.TokenCriteria[0].Amount = "200" permissionEditRequest := &requests.EditCommunityTokenPermission{ PermissionID: tokenPermissionID, CreateCommunityTokenPermission: *permissionRequest, } // then, admin tries to edit permission response, err = s.admin.EditCommunityTokenPermission(permissionEditRequest) s.Require().Error(err) s.Require().Nil(response) } func (s *AdminMessengerCommunitiesSuite) TestAdminCannotDeleteBecomeAdminPermission() { community := s.setUpCommunityAndRoles() permissionRequest := createTestPermissionRequest(community) permissionRequest.Type = protobuf.CommunityTokenPermission_BECOME_ADMIN // owner creates BECOME_ADMIN permission response, err := s.owner.CreateCommunityTokenPermission(permissionRequest) s.Require().NoError(err) var tokenPermissionID string for id := range response.CommunityChanges[0].TokenPermissionsAdded { tokenPermissionID = id } s.Require().NotEqual(tokenPermissionID, "") // then, ensure admin receives updated community _, err = WaitOnMessengerResponse( s.admin, func(r *MessengerResponse) bool { return len(r.Communities()) > 0 }, "admin did not receive updated community", ) s.Require().NoError(err) adminCommunity, err := s.admin.communitiesManager.GetByID(community.ID()) s.Require().NoError(err) s.assertAdminTokenPermissionCreated(adminCommunity) deleteTokenPermission := &requests.DeleteCommunityTokenPermission{ CommunityID: community.ID(), PermissionID: tokenPermissionID, } // then admin tries to delete BECOME_ADMIN permission which should fail response, err = s.admin.DeleteCommunityTokenPermission(deleteTokenPermission) s.Require().Error(err) s.Require().Nil(response) } func (s *AdminMessengerCommunitiesSuite) TestAdminAcceptMemberRequestToJoin() { community := s.setUpOnRequestCommunityAndRoles() // set up additional user that will send request to join user := s.newMessenger() _, err := user.Start() s.Require().NoError(err) defer user.Shutdown() // nolint: errcheck s.advertiseCommunityTo(community, user) // user sends request to join requestToJoin := &requests.RequestToJoinCommunity{CommunityID: community.ID()} response, err := user.RequestToJoinCommunity(requestToJoin) s.Require().NoError(err) s.Require().NotNil(response) s.Require().Len(response.RequestsToJoinCommunity, 1) _ = response.RequestsToJoinCommunity[0] // admin receives request to join response, err = WaitOnMessengerResponse( s.admin, func(r *MessengerResponse) bool { return len(r.RequestsToJoinCommunity) > 0 }, "admin did not receive community request to join", ) s.Require().NoError(err) s.Require().Len(response.RequestsToJoinCommunity, 1) receivedRequest := response.RequestsToJoinCommunity[0] // admin has not accepted request yet adminCommunity, err := s.admin.GetCommunityByID(community.ID()) s.Require().NoError(err) s.Require().False(adminCommunity.HasMember(&user.identity.PublicKey)) acceptRequestToJoin := &requests.AcceptRequestToJoinCommunity{ID: receivedRequest.ID} response, err = s.admin.AcceptRequestToJoinCommunity(acceptRequestToJoin) s.Require().NoError(err) s.Require().NotNil(response) s.Require().Len(response.Communities(), 1) s.Require().True(response.Communities()[0].HasMember(&user.identity.PublicKey)) // user receives request to join response response, err = WaitOnMessengerResponse( user, func(r *MessengerResponse) bool { return len(r.Communities()) > 0 }, "user did not receive community request to join response", ) s.Require().NoError(err) s.Require().Len(response.Communities(), 1) s.Require().True(response.Communities()[0].HasMember(&user.identity.PublicKey)) // owner receives updated community response, err = WaitOnMessengerResponse( s.owner, func(r *MessengerResponse) bool { return len(r.Communities()) > 0 }, "owner did not receive community request to join response", ) s.Require().NoError(err) s.Require().Len(response.Communities(), 1) requests, err := s.owner.AcceptedRequestsToJoinForCommunity(community.ID()) // there's now two requests to join (admin and alice) + 1 from user s.Require().NoError(err) s.Require().Len(requests, 3) s.Require().True(response.Communities()[0].HasMember(&user.identity.PublicKey)) // alice receives updated community response, err = WaitOnMessengerResponse( s.alice, func(r *MessengerResponse) bool { return len(r.Communities()) > 0 }, "alice did not receive community request to join response", ) s.Require().NoError(err) s.Require().Len(response.Communities(), 1) s.Require().True(response.Communities()[0].HasMember(&user.identity.PublicKey)) } func (s *AdminMessengerCommunitiesSuite) TestAdminRejectMemberRequestToJoin() { community := s.setUpOnRequestCommunityAndRoles() // set up additional user that will send request to join user := s.newMessenger() _, err := user.Start() s.Require().NoError(err) defer user.Shutdown() // nolint: errcheck s.advertiseCommunityTo(community, user) // user sends request to join requestToJoin := &requests.RequestToJoinCommunity{CommunityID: community.ID()} response, err := user.RequestToJoinCommunity(requestToJoin) s.Require().NoError(err) s.Require().NotNil(response) s.Require().Len(response.RequestsToJoinCommunity, 1) // admin receives request to join response, err = WaitOnMessengerResponse( s.admin, func(r *MessengerResponse) bool { return len(r.RequestsToJoinCommunity) > 0 }, "admin did not receive community request to join", ) s.Require().NoError(err) s.Require().Len(response.RequestsToJoinCommunity, 1) receivedRequest := response.RequestsToJoinCommunity[0] // admin has not accepted request yet adminCommunity, err := s.admin.GetCommunityByID(community.ID()) s.Require().NoError(err) s.Require().False(adminCommunity.HasMember(&user.identity.PublicKey)) // admin rejects request to join rejectRequestToJoin := &requests.DeclineRequestToJoinCommunity{ID: receivedRequest.ID} _, err = s.admin.DeclineRequestToJoinCommunity(rejectRequestToJoin) s.Require().NoError(err) adminCommunity, err = s.admin.GetCommunityByID(community.ID()) s.Require().NoError(err) s.Require().False(adminCommunity.HasMember(&user.identity.PublicKey)) // owner receives admin event and stores rejected request to join response, err = WaitOnMessengerResponse( s.owner, func(r *MessengerResponse) bool { return len(r.Communities()) > 0 }, "owner did not receive community request to join update from admin", ) s.Require().NoError(err) s.Require().False(response.Communities()[0].HasMember(&user.identity.PublicKey)) requests, err := s.owner.DeclinedRequestsToJoinForCommunity(community.ID()) s.Require().Len(requests, 1) s.Require().NoError(err) } func (s *AdminMessengerCommunitiesSuite) TestAdminCreateEditDeleteCategories() { community := s.setUpCommunityAndRoles() newCategory := &requests.CreateCommunityCategory{ CommunityID: community.ID(), CategoryName: "admin-category-name", } categoryID := s.adminCreateCommunityCategory(community, newCategory) editCategory := &requests.EditCommunityCategory{ CommunityID: community.ID(), CategoryID: categoryID, CategoryName: "edited-admin-category-name", } s.adminEditCommunityCategory(community.IDString(), editCategory) deleteCategory := &requests.DeleteCommunityCategory{ CommunityID: community.ID(), CategoryID: categoryID, } s.adminDeleteCommunityCategory(community.IDString(), deleteCategory) } func (s *AdminMessengerCommunitiesSuite) TestAdminReorderChannelsAndCategories() { community := s.setUpCommunityAndRoles() newCategory := &requests.CreateCommunityCategory{ CommunityID: community.ID(), CategoryName: "admin-category-name", } _ = s.adminCreateCommunityCategory(community, newCategory) newCategory.CategoryName = "admin-category-name2" categoryID2 := s.adminCreateCommunityCategory(community, newCategory) chat := &protobuf.CommunityChat{ Permissions: &protobuf.CommunityPermissions{ Access: protobuf.CommunityPermissions_NO_MEMBERSHIP, }, Identity: &protobuf.ChatIdentity{ DisplayName: "chat from admin", Emoji: "", Description: "chat created by an admin", }, } chatID := s.adminCreateCommunityChannel(community, chat) reorderCommunityRequest := requests.ReorderCommunityCategories{ CommunityID: community.ID(), CategoryID: categoryID2, Position: 0, } s.adminReorderCategory(&reorderCommunityRequest) reorderChatRequest := requests.ReorderCommunityChat{ CommunityID: community.ID(), CategoryID: categoryID2, ChatID: chatID, Position: 0, } s.adminReorderChannel(&reorderChatRequest) } func (s *AdminMessengerCommunitiesSuite) TestAdminKickAdmin() { community := s.setUpCommunityAndRoles() // admin tries to kick the owner _, err := s.admin.RemoveUserFromCommunity( community.ID(), common.PubkeyToHex(&s.admin.identity.PublicKey), ) s.Require().Error(err) s.Require().EqualError(err, "not allowed to remove admin or owner") } func (s *AdminMessengerCommunitiesSuite) TestAdminKickMember() { community := s.setUpCommunityAndRoles() // admin tries to kick the owner _, err := s.admin.RemoveUserFromCommunity( community.ID(), common.PubkeyToHex(&s.owner.identity.PublicKey), ) s.Require().Error(err) s.adminKickAlice(community.ID(), common.PubkeyToHex(&s.alice.identity.PublicKey)) } func (s *AdminMessengerCommunitiesSuite) TestAdminBanAdmin() { community := s.setUpCommunityAndRoles() // verify that admin can't ban an admin _, err := s.admin.BanUserFromCommunity( &requests.BanUserFromCommunity{ CommunityID: community.ID(), User: common.PubkeyToHexBytes(&s.admin.identity.PublicKey), }, ) s.Require().Error(err) s.Require().EqualError(err, "not allowed to ban admin or owner") } func (s *AdminMessengerCommunitiesSuite) TestAdminBanUnbanMember() { community := s.setUpCommunityAndRoles() // verify that admin can't ban an owner _, err := s.admin.BanUserFromCommunity( &requests.BanUserFromCommunity{ CommunityID: community.ID(), User: common.PubkeyToHexBytes(&s.owner.identity.PublicKey), }, ) s.Require().Error(err) banRequest := &requests.BanUserFromCommunity{ CommunityID: community.ID(), User: common.PubkeyToHexBytes(&s.alice.identity.PublicKey), } s.adminBanAlice(banRequest) unbanRequest := &requests.UnbanUserFromCommunity{ CommunityID: community.ID(), User: common.PubkeyToHexBytes(&s.alice.identity.PublicKey), } s.adminUnbanAlice(unbanRequest) } func (s *AdminMessengerCommunitiesSuite) TestAdminDeleteAnyMessageInTheCommunity() { community := s.setUpCommunityAndRoles() chatID := community.ChatIDs()[0] inputMessage := common.Message{} inputMessage.ChatId = chatID inputMessage.ContentType = protobuf.ChatMessage_TEXT_PLAIN inputMessage.Text = "owner text" messageID := s.ownerSendMessage(&inputMessage) s.adminDeleteMessage(messageID) } func (s *AdminMessengerCommunitiesSuite) TestAdminPinMessage() { community := s.setUpCommunityAndRoles() s.Require().False(community.AllowsAllMembersToPinMessage()) chatID := community.ChatIDs()[0] inputMessage := common.Message{} inputMessage.ChatId = chatID inputMessage.ContentType = protobuf.ChatMessage_TEXT_PLAIN inputMessage.Text = "owner text" messageID := s.ownerSendMessage(&inputMessage) pinnedMessage := common.PinMessage{} pinnedMessage.MessageId = messageID pinnedMessage.ChatId = chatID pinnedMessage.Pinned = true s.adminPinMessage(&pinnedMessage) } func (s *AdminMessengerCommunitiesSuite) TestAdminMintToken() { s.setUpCommunityAndRoles() // TODO admin test: Mint Tokens (rescticted) } func (s *AdminMessengerCommunitiesSuite) TestAdminAirdropTokens() { s.setUpCommunityAndRoles() // TODO admin test: Airdrop Tokens (restricted) } func (s *AdminMessengerCommunitiesSuite) TestMemberReceiveAdminEventsWhenOwnerOffline() { community := s.setUpCommunityAndRoles() // To simulate behavior when owner is offline, we will not use owner for listening new events // In this scenario member will reveive list of events newAdminChat := &protobuf.CommunityChat{ Permissions: &protobuf.CommunityPermissions{ Access: protobuf.CommunityPermissions_NO_MEMBERSHIP, }, Identity: &protobuf.ChatIdentity{ DisplayName: "chat from admin", Emoji: "", Description: "chat created by an admin", }, } checkChannelCreated := func(response *MessengerResponse) error { modifiedCommmunity, err := getModifiedCommunity(response, community.IDString()) if err != nil { return err } for _, chat := range modifiedCommmunity.Chats() { if chat.GetIdentity().GetDisplayName() == newAdminChat.GetIdentity().GetDisplayName() { return nil } } return errors.New("couldn't find created chat in response") } response, err := s.admin.CreateCommunityChat(community.ID(), newAdminChat) s.Require().NoError(err) s.Require().NoError(checkChannelCreated(response)) s.Require().Len(response.CommunityChanges, 1) s.Require().Len(response.CommunityChanges[0].ChatsAdded, 1) var addedChatID string for addedChatID = range response.CommunityChanges[0].ChatsAdded { break } s.waitOnMessengerResponse(WaitCommunityCondition, checkChannelCreated, s.alice) s.waitOnMessengerResponse(WaitCommunityCondition, checkChannelCreated, s.admin) newAdminChat.Identity.DisplayName = "modified chat from admin" checkChannelEdited := func(response *MessengerResponse) error { modifiedCommmunity, err := getModifiedCommunity(response, community.IDString()) if err != nil { return err } for _, chat := range modifiedCommmunity.Chats() { if chat.GetIdentity().GetDisplayName() == newAdminChat.GetIdentity().GetDisplayName() { return nil } } return errors.New("couldn't find modified chat in response") } response, err = s.admin.EditCommunityChat(community.ID(), addedChatID, newAdminChat) s.Require().NoError(err) s.Require().NoError(checkChannelEdited(response)) s.waitOnMessengerResponse(WaitCommunityCondition, checkChannelEdited, s.alice) s.waitOnMessengerResponse(WaitCommunityCondition, checkChannelEdited, s.admin) checkChannelDeleted := func(response *MessengerResponse) error { modifiedCommmunity, err := getModifiedCommunity(response, community.IDString()) if err != nil { return err } if _, exists := modifiedCommmunity.Chats()[addedChatID]; exists { return errors.New("channel was not deleted") } return nil } response, err = s.admin.DeleteCommunityChat(community.ID(), addedChatID) s.Require().NoError(err) s.Require().NoError(checkChannelDeleted(response)) s.waitOnMessengerResponse(WaitCommunityCondition, checkChannelDeleted, s.alice) s.waitOnMessengerResponse(WaitCommunityCondition, checkChannelDeleted, s.admin) } func (s *AdminMessengerCommunitiesSuite) setUpOnRequestCommunityAndRoles() *communities.Community { tcs2, err := s.owner.communitiesManager.All() s.Require().NoError(err, "admin.communitiesManager.All") s.Len(tcs2, 1, "Must have 1 community") // owner creates a community and chat community := s.createCommunity(protobuf.CommunityPermissions_ON_REQUEST) s.advertiseCommunityTo(community, s.admin) s.advertiseCommunityTo(community, s.alice) s.refreshMessengerResponses() s.joinOnRequestCommunity(community, s.admin) s.joinOnRequestCommunity(community, s.alice) s.refreshMessengerResponses() // grant admin permissions to the admin s.grantAdminPermissions(community, s.admin) return community } func (s *AdminMessengerCommunitiesSuite) setUpCommunityAndRoles() *communities.Community { tcs2, err := s.owner.communitiesManager.All() s.Require().NoError(err, "admin.communitiesManager.All") s.Len(tcs2, 1, "Must have 1 community") // owner creates a community and chat community := s.createCommunity(protobuf.CommunityPermissions_NO_MEMBERSHIP) //_ = s.createCommunityChat(community) s.refreshMessengerResponses() // add admin and alice to the community s.advertiseCommunityTo(community, s.admin) s.advertiseCommunityTo(community, s.alice) s.joinCommunity(community, s.admin) s.joinCommunity(community, s.alice) s.refreshMessengerResponses() // grant admin permissions to the admin s.grantAdminPermissions(community, s.admin) return community } func (s *AdminMessengerCommunitiesSuite) advertiseCommunityTo(community *communities.Community, user *Messenger) { chat := CreateOneToOneChat(common.PubkeyToHex(&user.identity.PublicKey), &user.identity.PublicKey, user.transport) inputMessage := &common.Message{} inputMessage.ChatId = chat.ID inputMessage.Text = "some text" inputMessage.CommunityID = community.IDString() err := s.owner.SaveChat(chat) s.Require().NoError(err) _, err = s.owner.SendChatMessage(context.Background(), inputMessage) s.Require().NoError(err) // Ensure community is received err = tt.RetryWithBackOff(func() error { response, err := user.RetrieveAll() if err != nil { return err } if len(response.Communities()) == 0 { return errors.New("community not received") } return nil }) s.Require().NoError(err) } func (s *AdminMessengerCommunitiesSuite) joinOnRequestCommunity(community *communities.Community, user *Messenger) { // Request to join the community request := &requests.RequestToJoinCommunity{CommunityID: community.ID()} response, err := user.RequestToJoinCommunity(request) s.Require().NoError(err) s.Require().NotNil(response) s.Require().Len(response.RequestsToJoinCommunity, 1) requestToJoin := response.RequestsToJoinCommunity[0] s.Require().Equal(requestToJoin.PublicKey, common.PubkeyToHex(&user.identity.PublicKey)) response, err = WaitOnMessengerResponse( s.owner, func(r *MessengerResponse) bool { return len(r.RequestsToJoinCommunity) > 0 }, "owner did not receive community request to join", ) s.Require().NoError(err) userRequestToJoin := response.RequestsToJoinCommunity[0] s.Require().Equal(userRequestToJoin.PublicKey, common.PubkeyToHex(&user.identity.PublicKey)) // accept join request acceptRequestToJoin := &requests.AcceptRequestToJoinCommunity{ID: requestToJoin.ID} response, err = s.owner.AcceptRequestToJoinCommunity(acceptRequestToJoin) s.Require().NoError(err) s.Require().NotNil(response) updatedCommunity := response.Communities()[0] s.Require().NotNil(updatedCommunity) s.Require().True(updatedCommunity.HasMember(&user.identity.PublicKey)) // receive request to join response _, err = WaitOnMessengerResponse( user, func(r *MessengerResponse) bool { return len(r.Communities()) > 0 }, "user did not receive request to join response", ) s.Require().NoError(err) userCommunity, err := user.GetCommunityByID(community.ID()) s.Require().NoError(err) s.Require().True(userCommunity.HasMember(&user.identity.PublicKey)) } func (s *AdminMessengerCommunitiesSuite) joinCommunity(community *communities.Community, user *Messenger) { // Request to join the community request := &requests.RequestToJoinCommunity{CommunityID: community.ID()} response, err := user.RequestToJoinCommunity(request) s.Require().NoError(err) s.Require().NotNil(response) s.Require().Len(response.RequestsToJoinCommunity, 1) s.Require().Len(response.ActivityCenterNotifications(), 1) notification := response.ActivityCenterNotifications()[0] s.Require().NotNil(notification) s.Require().Equal(notification.Type, ActivityCenterNotificationTypeCommunityRequest) s.Require().Equal(notification.MembershipStatus, ActivityCenterMembershipStatusPending) // Retrieve and accept join request err = tt.RetryWithBackOff(func() error { response, err := s.owner.RetrieveAll() if err != nil { return err } if len(response.Communities()) == 0 { return errors.New("no communities in response (accept join request)") } if !response.Communities()[0].HasMember(&user.identity.PublicKey) { return errors.New("user not accepted") } return nil }) s.Require().NoError(err) // Retrieve join request response err = tt.RetryWithBackOff(func() error { response, err := user.RetrieveAll() if err != nil { return err } if len(response.Communities()) == 0 { return errors.New("no communities in response (join request response)") } if !response.Communities()[0].HasMember(&user.identity.PublicKey) { return errors.New("user not a member") } return nil }) s.Require().NoError(err) } func (s *AdminMessengerCommunitiesSuite) createCommunity(membershipType protobuf.CommunityPermissions_Access) *communities.Community { description := &requests.CreateCommunity{ Membership: membershipType, Name: "status", Color: "#ffffff", Description: "status community description", PinMessageAllMembersEnabled: false, } response, err := s.owner.CreateCommunity(description, true) s.Require().NoError(err) s.Require().NotNil(response) s.Require().Len(response.Communities(), 1) s.Require().Len(response.Chats(), 1) return response.Communities()[0] } func (s *AdminMessengerCommunitiesSuite) grantAdminPermissions(community *communities.Community, target *Messenger) { responseAddRole, err := s.owner.AddRoleToMember(&requests.AddRoleToMember{ CommunityID: community.ID(), User: common.PubkeyToHexBytes(target.IdentityPublicKey()), Role: protobuf.CommunityMember_ROLE_ADMIN, }) s.Require().NoError(err) checkAdminRole := func(response *MessengerResponse) bool { if len(response.Communities()) == 0 { return false } rCommunities := response.Communities() s.Require().Len(rCommunities, 1) s.Require().True(rCommunities[0].IsMemberAdmin(target.IdentityPublicKey())) return true } checkAdminRole(responseAddRole) _, err = WaitOnMessengerResponse(s.admin, func(response *MessengerResponse) bool { return checkAdminRole(response) }, "community description changed message not received") s.Require().NoError(err) _, err = WaitOnMessengerResponse(s.alice, func(response *MessengerResponse) bool { return checkAdminRole(response) }, "community description changed message not received") s.Require().NoError(err) s.refreshMessengerResponses() } func (s *AdminMessengerCommunitiesSuite) adminEditsCommunityDescription(community *communities.Community) { expectedName := "edited community name" expectedColor := "#000000" expectedDescr := "edited community description" response, err := s.admin.EditCommunity(&requests.EditCommunity{ CommunityID: community.ID(), CreateCommunity: requests.CreateCommunity{ Membership: protobuf.CommunityPermissions_ON_REQUEST, Name: expectedName, Color: expectedColor, Description: expectedDescr, }, }) checkCommunityEdit := func(response *MessengerResponse) error { if len(response.Communities()) == 0 { return errors.New("community not received") } rCommunities := response.Communities() if expectedName != rCommunities[0].Name() { return errors.New("incorrect community name") } if expectedColor != rCommunities[0].Color() { return errors.New("incorrect community color") } if expectedDescr != rCommunities[0].DescriptionText() { return errors.New("incorrect community description") } return nil } s.Require().NoError(err) s.Require().Nil(checkCommunityEdit(response)) s.checkClientsReceivedAdminEvent(WaitCommunityCondition, checkCommunityEdit) } func (s *AdminMessengerCommunitiesSuite) refreshMessengerResponses() { _, err := WaitOnMessengerResponse(s.owner, func(response *MessengerResponse) bool { return true }, "community description changed message not received") s.Require().NoError(err) _, err = WaitOnMessengerResponse(s.admin, func(response *MessengerResponse) bool { return true }, "community description changed message not received") s.Require().NoError(err) _, err = WaitOnMessengerResponse(s.alice, func(response *MessengerResponse) bool { return true }, "community description changed message not received") s.Require().NoError(err) } type MessageResponseValidator func(*MessengerResponse) error type WaitResponseValidator func(*MessengerResponse) bool func WaitCommunityCondition(r *MessengerResponse) bool { return len(r.Communities()) > 0 } func WaitMessageCondition(response *MessengerResponse) bool { return len(response.Messages()) > 0 } func (s *AdminMessengerCommunitiesSuite) waitOnMessengerResponse(fnWait WaitResponseValidator, fn MessageResponseValidator, user *Messenger) { response, err := WaitOnMessengerResponse( user, fnWait, "MessengerResponse data not received", ) s.Require().NoError(err) s.Require().NoError(fn(response)) } func (s *AdminMessengerCommunitiesSuite) checkClientsReceivedAdminEvent(fnWait WaitResponseValidator, fn MessageResponseValidator) { // Wait and verify Alice received admin event s.waitOnMessengerResponse(fnWait, fn, s.alice) // Wait and verify Admin received his own admin event s.waitOnMessengerResponse(fnWait, fn, s.admin) // Wait and verify Owner received admin event // Owner will publish CommunityDescription update s.waitOnMessengerResponse(fnWait, fn, s.owner) // Wait and verify Alice received the Owner CommunityDescription update s.waitOnMessengerResponse(fnWait, fn, s.alice) // Wait and verify Admin received the Owner CommunityDescription update s.waitOnMessengerResponse(fnWait, fn, s.admin) // Wait and verify Owner received his own CommunityDescription update s.waitOnMessengerResponse(fnWait, fn, s.owner) } func (s *AdminMessengerCommunitiesSuite) adminCreateTokenPermission(community *communities.Community, request *requests.CreateCommunityTokenPermission, assertFn func(*communities.Community)) (string, *requests.CreateCommunityTokenPermission) { checkTokenPermissionCreation := func(response *MessengerResponse) error { modifiedCommmunity, err := getModifiedCommunity(response, community.IDString()) if err != nil { return err } if !modifiedCommmunity.HasTokenPermissions() { return errors.New("new token permission was not found") } return nil } response, err := s.admin.CreateCommunityTokenPermission(request) s.Require().NoError(err) s.Require().Nil(checkTokenPermissionCreation(response)) s.checkClientsReceivedAdminEvent(WaitCommunityCondition, checkTokenPermissionCreation) var tokenPermissionID string for tokenPermissionID = range response.CommunityChanges[0].TokenPermissionsAdded { break } s.Require().NotEqual(tokenPermissionID, "") return tokenPermissionID, request } func (s *AdminMessengerCommunitiesSuite) adminEditTokenPermission(community *communities.Community, request *requests.EditCommunityTokenPermission) { checkTokenPermissionEdit := func(response *MessengerResponse) error { modifiedCommmunity, err := getModifiedCommunity(response, community.IDString()) if err != nil { return err } s.assertAdminTokenPermissionEdited(modifiedCommmunity) return nil } response, err := s.admin.EditCommunityTokenPermission(request) s.Require().NoError(err) s.Require().Nil(checkTokenPermissionEdit(response)) s.checkClientsReceivedAdminEvent(WaitCommunityCondition, checkTokenPermissionEdit) } func (s *AdminMessengerCommunitiesSuite) adminDeleteTokenPermission(community *communities.Community, request *requests.DeleteCommunityTokenPermission) { checkTokenPermissionDeleted := func(response *MessengerResponse) error { modifiedCommmunity, err := getModifiedCommunity(response, community.IDString()) if err != nil { return err } if modifiedCommmunity.HasTokenPermissions() { return errors.New("token permission was not deleted") } return nil } response, err := s.admin.DeleteCommunityTokenPermission(request) s.Require().NoError(err) s.Require().Nil(checkTokenPermissionDeleted(response)) s.checkClientsReceivedAdminEvent(WaitCommunityCondition, checkTokenPermissionDeleted) } func createTestPermissionRequest(community *communities.Community) *requests.CreateCommunityTokenPermission { return &requests.CreateCommunityTokenPermission{ CommunityID: community.ID(), Type: protobuf.CommunityTokenPermission_BECOME_MEMBER, TokenCriteria: []*protobuf.TokenCriteria{ { Type: protobuf.CommunityTokenType_ERC20, ContractAddresses: map[uint64]string{uint64(1): "0x123"}, Symbol: "TEST", Amount: "100", Decimals: uint64(18), }, }, } } func (s *AdminMessengerCommunitiesSuite) adminCreateTestTokenPermission(community *communities.Community) (string, *requests.CreateCommunityTokenPermission) { createTokenPermission := createTestPermissionRequest(community) return s.adminCreateTokenPermission(community, createTokenPermission, s.assertAdminTokenPermissionCreated) } func (s *AdminMessengerCommunitiesSuite) assertAdminTokenPermissionCreated(community *communities.Community) { permissions := make([]*protobuf.CommunityTokenPermission, 0) tokenPermissions := community.TokenPermissions() for _, p := range tokenPermissions { permissions = append(permissions, p) } s.Require().Len(permissions, 1) s.Require().Len(permissions[0].TokenCriteria, 1) s.Require().Equal(permissions[0].TokenCriteria[0].Type, protobuf.CommunityTokenType_ERC20) s.Require().Equal(permissions[0].TokenCriteria[0].Symbol, "TEST") s.Require().Equal(permissions[0].TokenCriteria[0].Amount, "100") s.Require().Equal(permissions[0].TokenCriteria[0].Decimals, uint64(18)) } func (s *AdminMessengerCommunitiesSuite) assertAdminTokenPermissionEdited(community *communities.Community) { permissions := community.TokenPermissionsByType(protobuf.CommunityTokenPermission_BECOME_MEMBER) s.Require().Len(permissions, 1) s.Require().Len(permissions[0].TokenCriteria, 1) s.Require().Equal(permissions[0].TokenCriteria[0].Type, protobuf.CommunityTokenType_ERC20) s.Require().Equal(permissions[0].TokenCriteria[0].Symbol, "UPDATED") s.Require().Equal(permissions[0].TokenCriteria[0].Amount, "200") s.Require().Equal(permissions[0].TokenCriteria[0].Decimals, uint64(18)) } func (s *AdminMessengerCommunitiesSuite) adminCreateCommunityChannel(community *communities.Community, newChannel *protobuf.CommunityChat) string { checkChannelCreated := func(response *MessengerResponse) error { modifiedCommmunity, err := getModifiedCommunity(response, community.IDString()) if err != nil { return err } for _, chat := range modifiedCommmunity.Chats() { if chat.GetIdentity().GetDisplayName() == newChannel.GetIdentity().GetDisplayName() { return nil } } return errors.New("couldn't find created chat in response") } response, err := s.admin.CreateCommunityChat(community.ID(), newChannel) s.Require().NoError(err) s.Require().NoError(checkChannelCreated(response)) s.Require().Len(response.CommunityChanges, 1) s.Require().Len(response.CommunityChanges[0].ChatsAdded, 1) var addedChatID string for addedChatID = range response.CommunityChanges[0].ChatsAdded { break } s.checkClientsReceivedAdminEvent(WaitCommunityCondition, checkChannelCreated) return addedChatID } func (s *AdminMessengerCommunitiesSuite) adminEditCommunityChannel(community *communities.Community, editChannel *protobuf.CommunityChat, channelID string) { checkChannelEdited := func(response *MessengerResponse) error { modifiedCommmunity, err := getModifiedCommunity(response, community.IDString()) if err != nil { return err } for _, chat := range modifiedCommmunity.Chats() { if chat.GetIdentity().GetDisplayName() == editChannel.GetIdentity().GetDisplayName() { return nil } } return errors.New("couldn't find modified chat in response") } response, err := s.admin.EditCommunityChat(community.ID(), channelID, editChannel) s.Require().NoError(err) s.Require().NoError(checkChannelEdited(response)) s.checkClientsReceivedAdminEvent(WaitCommunityCondition, checkChannelEdited) } func (s *AdminMessengerCommunitiesSuite) adminDeleteCommunityChannel(community *communities.Community, channelID string) { checkChannelDeleted := func(response *MessengerResponse) error { modifiedCommmunity, err := getModifiedCommunity(response, community.IDString()) if err != nil { return err } if _, exists := modifiedCommmunity.Chats()[channelID]; exists { return errors.New("channel was not deleted") } return nil } response, err := s.admin.DeleteCommunityChat(community.ID(), channelID) s.Require().NoError(err) s.Require().NoError(checkChannelDeleted(response)) s.checkClientsReceivedAdminEvent(WaitCommunityCondition, checkChannelDeleted) } func (s *AdminMessengerCommunitiesSuite) adminCreateCommunityCategory(community *communities.Community, newCategory *requests.CreateCommunityCategory) string { checkCategoryCreated := func(response *MessengerResponse) error { modifiedCommmunity, err := getModifiedCommunity(response, community.IDString()) if err != nil { return err } for _, category := range modifiedCommmunity.Categories() { if category.GetName() == newCategory.CategoryName { return nil } } return errors.New("couldn't find created Category in the response") } response, err := s.admin.CreateCommunityCategory(newCategory) s.Require().NoError(err) s.Require().NoError(checkCategoryCreated(response)) s.Require().Len(response.Communities(), 1) s.Require().Len(response.CommunityChanges[0].CategoriesAdded, 1) var categoryID string for categoryID = range response.CommunityChanges[0].CategoriesAdded { break } s.checkClientsReceivedAdminEvent(WaitCommunityCondition, checkCategoryCreated) return categoryID } func (s *AdminMessengerCommunitiesSuite) adminEditCommunityCategory(communityID string, editCategory *requests.EditCommunityCategory) { checkCategoryEdited := func(response *MessengerResponse) error { modifiedCommmunity, err := getModifiedCommunity(response, communityID) if err != nil { return err } for _, category := range modifiedCommmunity.Categories() { if category.GetName() == editCategory.CategoryName { return nil } } return errors.New("couldn't find edited Category in the response") } response, err := s.admin.EditCommunityCategory(editCategory) s.Require().NoError(err) s.Require().NoError(checkCategoryEdited(response)) s.checkClientsReceivedAdminEvent(WaitCommunityCondition, checkCategoryEdited) } func (s *AdminMessengerCommunitiesSuite) adminDeleteCommunityCategory(communityID string, deleteCategory *requests.DeleteCommunityCategory) { checkCategoryDeleted := func(response *MessengerResponse) error { modifiedCommmunity, err := getModifiedCommunity(response, communityID) if err != nil { return err } if _, exists := modifiedCommmunity.Chats()[deleteCategory.CategoryID]; exists { return errors.New("community was not deleted") } return nil } response, err := s.admin.DeleteCommunityCategory(deleteCategory) s.Require().NoError(err) s.Require().NoError(checkCategoryDeleted(response)) s.checkClientsReceivedAdminEvent(WaitCommunityCondition, checkCategoryDeleted) } func (s *AdminMessengerCommunitiesSuite) ownerSendMessage(inputMessage *common.Message) string { response, err := s.owner.SendChatMessage(context.Background(), inputMessage) s.Require().NoError(err) message := response.Messages()[0] s.Require().Equal(inputMessage.Text, message.Text) messageID := message.ID response, err = WaitOnMessengerResponse(s.admin, WaitMessageCondition, "messages not received") s.Require().NoError(err) message = response.Messages()[0] s.Require().Equal(inputMessage.Text, message.Text) response, err = WaitOnMessengerResponse(s.alice, WaitMessageCondition, "messages not received") s.Require().NoError(err) message = response.Messages()[0] s.Require().Equal(inputMessage.Text, message.Text) s.refreshMessengerResponses() return messageID } func (s *AdminMessengerCommunitiesSuite) adminDeleteMessage(messageID string) { checkMessageDeleted := func(response *MessengerResponse) error { if len(response.RemovedMessages()) > 0 { return nil } return errors.New("message was not deleted") } response, err := s.admin.DeleteMessageAndSend(context.Background(), messageID) s.Require().NoError(err) s.Require().NoError(checkMessageDeleted(response)) waitMessageCondition := func(response *MessengerResponse) bool { return len(response.RemovedMessages()) > 0 } s.waitOnMessengerResponse(waitMessageCondition, checkMessageDeleted, s.alice) s.waitOnMessengerResponse(waitMessageCondition, checkMessageDeleted, s.owner) } func (s *AdminMessengerCommunitiesSuite) adminPinMessage(pinnedMessage *common.PinMessage) { checkPinned := func(response *MessengerResponse) error { if len(response.PinMessages()) > 0 { return nil } return errors.New("pin messages was not added") } response, err := s.admin.SendPinMessage(context.Background(), pinnedMessage) s.Require().NoError(err) s.Require().NoError(checkPinned(response)) s.waitOnMessengerResponse(WaitMessageCondition, checkPinned, s.alice) s.waitOnMessengerResponse(WaitMessageCondition, checkPinned, s.owner) } func (s *AdminMessengerCommunitiesSuite) adminBanAlice(banRequest *requests.BanUserFromCommunity) { checkBanned := func(response *MessengerResponse) error { modifiedCommmunity, err := getModifiedCommunity(response, types.EncodeHex(banRequest.CommunityID)) if err != nil { return err } if modifiedCommmunity.HasMember(&s.alice.identity.PublicKey) { return errors.New("alice was not removed from the member list") } if !modifiedCommmunity.IsBanned(&s.alice.identity.PublicKey) { return errors.New("alice was not added to the banned list") } return nil } response, err := s.admin.BanUserFromCommunity(banRequest) s.Require().NoError(err) s.Require().Nil(checkBanned(response)) s.checkClientsReceivedAdminEvent(WaitCommunityCondition, checkBanned) } func (s *AdminMessengerCommunitiesSuite) adminUnbanAlice(unbanRequest *requests.UnbanUserFromCommunity) { checkUnbanned := func(response *MessengerResponse) error { modifiedCommmunity, err := getModifiedCommunity(response, types.EncodeHex(unbanRequest.CommunityID)) if err != nil { return err } if modifiedCommmunity.IsBanned(&s.alice.identity.PublicKey) { return errors.New("alice was not unbanned") } return nil } response, err := s.admin.UnbanUserFromCommunity(unbanRequest) s.Require().NoError(err) s.Require().Nil(checkUnbanned(response)) response, err = WaitOnMessengerResponse( s.owner, WaitCommunityCondition, "MessengerResponse data not received", ) s.Require().NoError(err) s.Require().NoError(checkUnbanned(response)) } func (s *AdminMessengerCommunitiesSuite) adminKickAlice(communityID types.HexBytes, pubkey string) { checkKicked := func(response *MessengerResponse) error { modifiedCommmunity, err := getModifiedCommunity(response, types.EncodeHex(communityID)) if err != nil { return err } if modifiedCommmunity.HasMember(&s.alice.identity.PublicKey) { return errors.New("alice was not kicked") } return nil } response, err := s.admin.RemoveUserFromCommunity( communityID, pubkey, ) s.Require().NoError(err) s.Require().Nil(checkKicked(response)) s.checkClientsReceivedAdminEvent(WaitCommunityCondition, checkKicked) } func (s *AdminMessengerCommunitiesSuite) adminReorderCategory(reorderRequest *requests.ReorderCommunityCategories) { checkCategoryReorder := func(response *MessengerResponse) error { modifiedCommmunity, err := getModifiedCommunity(response, types.EncodeHex(reorderRequest.CommunityID)) if err != nil { return err } category, exist := modifiedCommmunity.Categories()[reorderRequest.CategoryID] if !exist { return errors.New("couldn't find community category") } if int(category.Position) != reorderRequest.Position { return errors.New("category was not reordered") } return nil } response, err := s.admin.ReorderCommunityCategories(reorderRequest) s.Require().NoError(err) s.Require().NoError(checkCategoryReorder(response)) s.checkClientsReceivedAdminEvent(WaitCommunityCondition, checkCategoryReorder) } func (s *AdminMessengerCommunitiesSuite) adminReorderChannel(reorderRequest *requests.ReorderCommunityChat) { checkChannelReorder := func(response *MessengerResponse) error { modifiedCommmunity, err := getModifiedCommunity(response, types.EncodeHex(reorderRequest.CommunityID)) if err != nil { return err } chat, exist := modifiedCommmunity.Chats()[reorderRequest.ChatID] if !exist { return errors.New("couldn't find community chat") } if int(chat.Position) != reorderRequest.Position { return errors.New("chat position was not reordered") } if chat.CategoryId != reorderRequest.CategoryID { return errors.New("chat category was not reordered") } return nil } response, err := s.admin.ReorderCommunityChat(reorderRequest) s.Require().NoError(err) s.Require().NoError(checkChannelReorder(response)) s.checkClientsReceivedAdminEvent(WaitCommunityCondition, checkChannelReorder) } func getModifiedCommunity(response *MessengerResponse, communityID string) (*communities.Community, error) { if len(response.Communities()) == 0 { return nil, errors.New("community not received") } var modifiedCommmunity *communities.Community = nil for _, c := range response.Communities() { if c.IDString() == communityID { modifiedCommmunity = c } } if modifiedCommmunity == nil { return nil, errors.New("couldn't find community in response") } return modifiedCommmunity, nil }