1482 lines
48 KiB
Go
1482 lines
48 KiB
Go
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: "data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAADIAAAAyCAIAAACRXR/mAAAAjklEQVR4nOzXwQmFMBAAUZXUYh32ZB32ZB02sxYQQSZGsod55/91WFgSS0RM+SyjA56ZRZhFmEWYRRT6h+M6G16zrxv6fdJpmUWYRbxsYr13dKfanpN0WmYRZhGzXz6AWYRZRIfbaX26fT9Jk07LLMIsosPt9I/dTDotswizCG+nhFmEWYRZhFnEHQAA///z1CFkYamgfQAAAABJRU5ErkJggg==",
|
|
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.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
|
|
}
|