2020-11-18 10:16:51 +01:00
|
|
|
package protocol
|
|
|
|
|
|
|
|
import (
|
2021-01-11 11:32:51 +01:00
|
|
|
"bytes"
|
2020-11-18 10:16:51 +01:00
|
|
|
"context"
|
|
|
|
"crypto/ecdsa"
|
2021-07-22 13:41:49 -04:00
|
|
|
"encoding/json"
|
2020-11-18 10:16:51 +01:00
|
|
|
"errors"
|
2021-08-06 16:40:23 +01:00
|
|
|
"fmt"
|
2020-11-18 10:16:51 +01:00
|
|
|
"io/ioutil"
|
feat: introduce messenger APIs to extract discord channels
As part of the new Discord <-> Status Community Import functionality,
we're adding an API that extracts all discord categories and channels
from a previously exported discord export file.
These APIs can be used in clients to show the user what categories and
channels will be imported later on.
There are two APIs:
1. `Messenger.ExtractDiscordCategoriesAndChannels(filesToimport
[]string) (*MessengerResponse, map[string]*discord.ImportError)`
This takes a list of exported discord export (JSON) files (typically one per
channel), reads them, and extracts the categories and channels into
dedicated data structures (`[]DiscordChannel` and `[]DiscordCategory`)
It also returns the oldest message timestamp found in all extracted
channels.
The API is synchronous and returns the extracted data as
a `*MessengerResponse`. This allows to make the API available
status-go's RPC interface.
The error case is a `map[string]*discord.ImportError` where each key
is a file path of a JSON file that we tried to extract data from, and
the value a `discord.ImportError` which holds an error message and an
error code, allowing for distinguishing between "critical" errors and
"non-critical" errors.
2. `Messenger.RequestExtractDiscordCategoriesAndChannels(filesToImport
[]string)`
This is the asynchronous counterpart to
`ExtractDiscordCategoriesAndChannels`. The reason this API has been
added is because discord servers can have a lot of message and
channel data, which causes `ExtractDiscordCategoriesAndChannels` to
block the thread for too long, making apps potentially feel like they
are stuck.
This API runs inside a go routine, eventually calls
`ExtractDiscordCategoriesAndChannels`, and then emits a newly
introduced `DiscordCategoriesAndChannelsExtractedSignal` that clients
can react to.
Failure of extraction has to be determined by the
`discord.ImportErrors` emitted by the signal.
**A note about exported discord history files**
We expect users to export their discord histories via the
[DiscordChatExporter](https://github.com/Tyrrrz/DiscordChatExporter/wiki/GUI%2C-CLI-and-Formats-explained#exportguild)
tool. The tool allows to export the data in different formats, such as
JSON, HTML and CSV.
We expect users to have their data exported as JSON.
Closes: https://github.com/status-im/status-desktop/issues/6690
2022-07-13 11:33:53 +02:00
|
|
|
"os"
|
2020-11-18 10:16:51 +01:00
|
|
|
"strings"
|
|
|
|
"testing"
|
2022-06-01 09:55:48 +02:00
|
|
|
"time"
|
2020-11-18 10:16:51 +01:00
|
|
|
|
2023-05-29 13:57:05 -04:00
|
|
|
"github.com/golang/protobuf/proto"
|
2020-11-18 10:16:51 +01:00
|
|
|
"github.com/stretchr/testify/suite"
|
|
|
|
"go.uber.org/zap"
|
|
|
|
|
|
|
|
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"
|
2023-05-24 16:42:31 +02:00
|
|
|
"github.com/status-im/status-go/multiaccounts/accounts"
|
2020-11-18 10:16:51 +01:00
|
|
|
"github.com/status-im/status-go/protocol/common"
|
2021-01-11 11:32:51 +01:00
|
|
|
"github.com/status-im/status-go/protocol/communities"
|
feat: introduce messenger APIs to extract discord channels
As part of the new Discord <-> Status Community Import functionality,
we're adding an API that extracts all discord categories and channels
from a previously exported discord export file.
These APIs can be used in clients to show the user what categories and
channels will be imported later on.
There are two APIs:
1. `Messenger.ExtractDiscordCategoriesAndChannels(filesToimport
[]string) (*MessengerResponse, map[string]*discord.ImportError)`
This takes a list of exported discord export (JSON) files (typically one per
channel), reads them, and extracts the categories and channels into
dedicated data structures (`[]DiscordChannel` and `[]DiscordCategory`)
It also returns the oldest message timestamp found in all extracted
channels.
The API is synchronous and returns the extracted data as
a `*MessengerResponse`. This allows to make the API available
status-go's RPC interface.
The error case is a `map[string]*discord.ImportError` where each key
is a file path of a JSON file that we tried to extract data from, and
the value a `discord.ImportError` which holds an error message and an
error code, allowing for distinguishing between "critical" errors and
"non-critical" errors.
2. `Messenger.RequestExtractDiscordCategoriesAndChannels(filesToImport
[]string)`
This is the asynchronous counterpart to
`ExtractDiscordCategoriesAndChannels`. The reason this API has been
added is because discord servers can have a lot of message and
channel data, which causes `ExtractDiscordCategoriesAndChannels` to
block the thread for too long, making apps potentially feel like they
are stuck.
This API runs inside a go routine, eventually calls
`ExtractDiscordCategoriesAndChannels`, and then emits a newly
introduced `DiscordCategoriesAndChannelsExtractedSignal` that clients
can react to.
Failure of extraction has to be determined by the
`discord.ImportErrors` emitted by the signal.
**A note about exported discord history files**
We expect users to export their discord histories via the
[DiscordChatExporter](https://github.com/Tyrrrz/DiscordChatExporter/wiki/GUI%2C-CLI-and-Formats-explained#exportguild)
tool. The tool allows to export the data in different formats, such as
JSON, HTML and CSV.
We expect users to have their data exported as JSON.
Closes: https://github.com/status-im/status-desktop/issues/6690
2022-07-13 11:33:53 +02:00
|
|
|
"github.com/status-im/status-go/protocol/discord"
|
2021-08-06 16:40:23 +01:00
|
|
|
"github.com/status-im/status-go/protocol/encryption/multidevice"
|
2020-11-18 10:16:51 +01:00
|
|
|
"github.com/status-im/status-go/protocol/protobuf"
|
2021-01-11 11:32:51 +01:00
|
|
|
"github.com/status-im/status-go/protocol/requests"
|
2023-05-29 13:57:05 -04:00
|
|
|
"github.com/status-im/status-go/protocol/transport"
|
2020-11-18 10:16:51 +01:00
|
|
|
"github.com/status-im/status-go/protocol/tt"
|
2023-05-29 13:57:05 -04:00
|
|
|
v1protocol "github.com/status-im/status-go/protocol/v1"
|
2020-11-18 10:16:51 +01:00
|
|
|
"github.com/status-im/status-go/waku"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestMessengerCommunitiesSuite(t *testing.T) {
|
|
|
|
suite.Run(t, new(MessengerCommunitiesSuite))
|
|
|
|
}
|
|
|
|
|
|
|
|
type MessengerCommunitiesSuite struct {
|
|
|
|
suite.Suite
|
2022-08-23 18:30:19 +02:00
|
|
|
admin *Messenger
|
2020-11-18 10:16:51 +01:00
|
|
|
bob *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 *MessengerCommunitiesSuite) SetupTest() {
|
|
|
|
s.logger = tt.MustCreateTestLogger()
|
|
|
|
|
|
|
|
config := waku.DefaultConfig
|
|
|
|
config.MinimumAcceptedPoW = 0
|
|
|
|
shh := waku.New(&config, s.logger)
|
|
|
|
s.shh = gethbridge.NewGethWakuWrapper(shh)
|
2021-07-09 15:19:33 +02:00
|
|
|
s.Require().NoError(shh.Start())
|
2020-11-18 10:16:51 +01:00
|
|
|
|
2023-06-27 14:20:20 +02:00
|
|
|
s.admin = s.newMessenger()
|
|
|
|
s.bob = s.newMessenger()
|
|
|
|
s.alice = s.newMessenger()
|
2022-08-23 18:30:19 +02:00
|
|
|
_, err := s.admin.Start()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
_, err = s.bob.Start()
|
2021-01-14 23:15:13 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
_, err = s.alice.Start()
|
|
|
|
s.Require().NoError(err)
|
2020-11-18 10:16:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *MessengerCommunitiesSuite) TearDownTest() {
|
2022-08-23 18:30:19 +02:00
|
|
|
s.Require().NoError(s.admin.Shutdown())
|
2020-11-18 10:16:51 +01:00
|
|
|
s.Require().NoError(s.bob.Shutdown())
|
|
|
|
s.Require().NoError(s.alice.Shutdown())
|
|
|
|
_ = s.logger.Sync()
|
|
|
|
}
|
|
|
|
|
2023-06-27 14:20:20 +02:00
|
|
|
func (s *MessengerCommunitiesSuite) newMessengerWithKey(privateKey *ecdsa.PrivateKey) *Messenger {
|
|
|
|
messenger, err := newCommunitiesTestMessenger(s.shh, privateKey, s.logger, nil, nil)
|
2020-11-18 10:16:51 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
|
2023-06-27 14:20:20 +02:00
|
|
|
return messenger
|
2020-11-18 10:16:51 +01:00
|
|
|
}
|
|
|
|
|
2023-06-27 14:20:20 +02:00
|
|
|
func (s *MessengerCommunitiesSuite) newMessenger() *Messenger {
|
2020-11-18 10:16:51 +01:00
|
|
|
privateKey, err := crypto.GenerateKey()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
2023-06-27 14:20:20 +02:00
|
|
|
return s.newMessengerWithKey(privateKey)
|
2020-11-18 10:16:51 +01:00
|
|
|
}
|
|
|
|
|
2022-07-19 11:31:52 +02:00
|
|
|
func (s *MessengerCommunitiesSuite) TestCreateCommunity() {
|
|
|
|
description := &requests.CreateCommunity{
|
|
|
|
Membership: protobuf.CommunityPermissions_NO_MEMBERSHIP,
|
|
|
|
Name: "status",
|
|
|
|
Color: "#ffffff",
|
|
|
|
Description: "status community description",
|
|
|
|
}
|
|
|
|
response, err := s.bob.CreateCommunity(description, true)
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
s.Require().Len(response.Chats(), 1)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *MessengerCommunitiesSuite) TestCreateCommunity_WithoutDefaultChannel() {
|
|
|
|
|
|
|
|
description := &requests.CreateCommunity{
|
|
|
|
Membership: protobuf.CommunityPermissions_NO_MEMBERSHIP,
|
|
|
|
Name: "status",
|
|
|
|
Color: "#ffffff",
|
|
|
|
Description: "status community description",
|
|
|
|
}
|
|
|
|
response, err := s.bob.CreateCommunity(description, false)
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
s.Require().Len(response.Chats(), 0)
|
|
|
|
}
|
|
|
|
|
2020-11-18 10:16:51 +01:00
|
|
|
func (s *MessengerCommunitiesSuite) TestRetrieveCommunity() {
|
2023-06-27 14:20:20 +02:00
|
|
|
alice := s.newMessenger()
|
2020-11-18 10:16:51 +01:00
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
description := &requests.CreateCommunity{
|
|
|
|
Membership: protobuf.CommunityPermissions_NO_MEMBERSHIP,
|
|
|
|
Name: "status",
|
|
|
|
Color: "#ffffff",
|
|
|
|
Description: "status community description",
|
2020-11-18 10:16:51 +01:00
|
|
|
}
|
|
|
|
|
2022-07-19 11:31:52 +02:00
|
|
|
response, err := s.bob.CreateCommunity(description, true)
|
2020-11-18 10:16:51 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
2021-01-11 11:32:51 +01:00
|
|
|
s.Require().Len(response.Communities(), 1)
|
2022-03-08 16:25:00 +01:00
|
|
|
s.Require().Len(response.CommunitiesSettings(), 1)
|
2022-05-10 15:25:08 +02:00
|
|
|
s.Require().Len(response.Chats(), 1)
|
2022-03-08 16:25:00 +01:00
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
community := response.Communities()[0]
|
2022-03-08 16:25:00 +01:00
|
|
|
communitySettings := response.CommunitiesSettings()[0]
|
|
|
|
|
|
|
|
s.Require().Equal(communitySettings.CommunityID, community.IDString())
|
|
|
|
s.Require().Equal(communitySettings.HistoryArchiveSupportEnabled, false)
|
2020-11-18 10:16:51 +01:00
|
|
|
|
|
|
|
// Send an community message
|
|
|
|
chat := CreateOneToOneChat(common.PubkeyToHex(&alice.identity.PublicKey), &alice.identity.PublicKey, s.alice.transport)
|
|
|
|
|
|
|
|
inputMessage := &common.Message{}
|
|
|
|
inputMessage.ChatId = chat.ID
|
|
|
|
inputMessage.Text = "some text"
|
|
|
|
inputMessage.CommunityID = community.IDString()
|
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
err = s.bob.SaveChat(chat)
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err)
|
2020-11-18 10:16:51 +01:00
|
|
|
_, err = s.bob.SendChatMessage(context.Background(), inputMessage)
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err)
|
2020-11-18 10:16:51 +01:00
|
|
|
|
|
|
|
// Pull message and make sure org is received
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = alice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-01-11 11:32:51 +01:00
|
|
|
if len(response.Communities()) == 0 {
|
2020-11-18 10:16:51 +01:00
|
|
|
return errors.New("community not received")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
communities, err := alice.Communities()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(communities, 2)
|
2021-01-11 11:32:51 +01:00
|
|
|
s.Require().Len(response.Communities(), 1)
|
2021-06-03 15:11:55 +02:00
|
|
|
s.Require().Len(response.Messages(), 1)
|
|
|
|
s.Require().Equal(community.IDString(), response.Messages()[0].CommunityID)
|
2020-11-18 10:16:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *MessengerCommunitiesSuite) TestJoinCommunity() {
|
2021-07-22 13:41:49 -04:00
|
|
|
ctx := context.Background()
|
2021-01-11 11:32:51 +01:00
|
|
|
|
|
|
|
description := &requests.CreateCommunity{
|
|
|
|
Membership: protobuf.CommunityPermissions_NO_MEMBERSHIP,
|
|
|
|
Name: "status",
|
|
|
|
Color: "#ffffff",
|
|
|
|
Description: "status community description",
|
2020-11-18 10:16:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Create an community chat
|
2022-07-19 11:31:52 +02:00
|
|
|
response, err := s.bob.CreateCommunity(description, true)
|
2020-11-18 10:16:51 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
2021-01-11 11:32:51 +01:00
|
|
|
s.Require().Len(response.Communities(), 1)
|
2022-03-08 16:25:00 +01:00
|
|
|
s.Require().Len(response.CommunitiesSettings(), 1)
|
2020-11-18 10:16:51 +01:00
|
|
|
|
2022-03-08 16:25:00 +01:00
|
|
|
communitySettings := response.CommunitiesSettings()[0]
|
2021-01-11 11:32:51 +01:00
|
|
|
community := response.Communities()[0]
|
2020-11-18 10:16:51 +01:00
|
|
|
|
2022-03-08 16:25:00 +01:00
|
|
|
s.Require().Equal(communitySettings.CommunityID, community.IDString())
|
|
|
|
s.Require().Equal(communitySettings.HistoryArchiveSupportEnabled, false)
|
|
|
|
|
2020-11-18 10:16:51 +01:00
|
|
|
orgChat := &protobuf.CommunityChat{
|
|
|
|
Permissions: &protobuf.CommunityPermissions{
|
|
|
|
Access: protobuf.CommunityPermissions_NO_MEMBERSHIP,
|
|
|
|
},
|
|
|
|
Identity: &protobuf.ChatIdentity{
|
|
|
|
DisplayName: "status-core",
|
2021-10-04 18:32:25 +05:30
|
|
|
Emoji: "😎",
|
2020-11-18 10:16:51 +01:00
|
|
|
Description: "status-core community chat",
|
|
|
|
},
|
|
|
|
}
|
2021-01-11 11:32:51 +01:00
|
|
|
response, err = s.bob.CreateCommunityChat(community.ID(), orgChat)
|
2020-11-18 10:16:51 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
2021-01-11 11:32:51 +01:00
|
|
|
s.Require().Len(response.Communities(), 1)
|
2021-10-06 11:08:17 +01:00
|
|
|
s.Require().Len(response.Chats(), 1)
|
2020-11-18 10:16:51 +01:00
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
createdChat := response.Chats()[0]
|
2020-11-18 10:16:51 +01:00
|
|
|
s.Require().Equal(community.IDString(), createdChat.CommunityID)
|
|
|
|
s.Require().Equal(orgChat.Identity.DisplayName, createdChat.Name)
|
2021-10-04 18:32:25 +05:30
|
|
|
s.Require().Equal(orgChat.Identity.Emoji, createdChat.Emoji)
|
2020-11-18 10:16:51 +01:00
|
|
|
s.Require().NotEmpty(createdChat.ID)
|
|
|
|
s.Require().Equal(ChatTypeCommunityChat, createdChat.ChatType)
|
|
|
|
s.Require().True(createdChat.Active)
|
|
|
|
s.Require().NotEmpty(createdChat.Timestamp)
|
|
|
|
s.Require().True(strings.HasPrefix(createdChat.ID, community.IDString()))
|
|
|
|
|
2021-05-23 09:34:17 -04:00
|
|
|
// Make sure the changes are reflect in the community
|
|
|
|
community = response.Communities()[0]
|
|
|
|
|
|
|
|
var chatIds []string
|
|
|
|
for k := range community.Chats() {
|
|
|
|
chatIds = append(chatIds, k)
|
|
|
|
}
|
|
|
|
category := &requests.CreateCommunityCategory{
|
|
|
|
CommunityID: community.ID(),
|
|
|
|
CategoryName: "category-name",
|
|
|
|
ChatIDs: chatIds,
|
|
|
|
}
|
|
|
|
|
|
|
|
response, err = s.bob.CreateCommunityCategory(category)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
s.Require().Len(response.Communities()[0].Categories(), 1)
|
|
|
|
|
2020-11-18 10:16:51 +01:00
|
|
|
// Make sure the changes are reflect in the community
|
2021-01-11 11:32:51 +01:00
|
|
|
community = response.Communities()[0]
|
2020-11-18 10:16:51 +01:00
|
|
|
chats := community.Chats()
|
2022-05-10 15:25:08 +02:00
|
|
|
s.Require().Len(chats, 2)
|
2020-11-18 10:16:51 +01:00
|
|
|
|
|
|
|
// Send an community message
|
|
|
|
chat := CreateOneToOneChat(common.PubkeyToHex(&s.alice.identity.PublicKey), &s.alice.identity.PublicKey, s.bob.transport)
|
|
|
|
|
|
|
|
inputMessage := &common.Message{}
|
|
|
|
inputMessage.ChatId = chat.ID
|
|
|
|
inputMessage.Text = "some text"
|
|
|
|
inputMessage.CommunityID = community.IDString()
|
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
err = s.bob.SaveChat(chat)
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err)
|
2020-11-18 10:16:51 +01:00
|
|
|
_, err = s.bob.SendChatMessage(context.Background(), inputMessage)
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err)
|
2020-11-18 10:16:51 +01:00
|
|
|
|
|
|
|
// Pull message and make sure org is received
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = s.alice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-01-11 11:32:51 +01:00
|
|
|
if len(response.Communities()) == 0 {
|
2020-11-18 10:16:51 +01:00
|
|
|
return errors.New("community not received")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
communities, err := s.alice.Communities()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(communities, 2)
|
2021-01-11 11:32:51 +01:00
|
|
|
s.Require().Len(response.Communities(), 1)
|
2021-06-03 15:11:55 +02:00
|
|
|
s.Require().Len(response.Messages(), 1)
|
|
|
|
s.Require().Equal(community.IDString(), response.Messages()[0].CommunityID)
|
2020-11-18 10:16:51 +01:00
|
|
|
|
|
|
|
// We join the org
|
2023-05-29 13:57:05 -04:00
|
|
|
response, err = s.alice.JoinCommunity(ctx, community.ID(), false)
|
2020-11-18 10:16:51 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
2021-01-11 11:32:51 +01:00
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
s.Require().True(response.Communities()[0].Joined())
|
2022-05-10 15:25:08 +02:00
|
|
|
s.Require().Len(response.Chats(), 2)
|
2021-05-23 09:34:17 -04:00
|
|
|
s.Require().Len(response.Communities()[0].Categories(), 1)
|
|
|
|
|
|
|
|
var categoryID string
|
|
|
|
for k := range response.Communities()[0].Categories() {
|
|
|
|
categoryID = k
|
|
|
|
}
|
2020-11-18 10:16:51 +01:00
|
|
|
|
|
|
|
// The chat should be created
|
2022-05-13 16:02:38 +02:00
|
|
|
|
|
|
|
found := false
|
|
|
|
for _, createdChat := range response.Chats() {
|
|
|
|
if orgChat.Identity.DisplayName == createdChat.Name {
|
|
|
|
found = true
|
|
|
|
s.Require().Equal(community.IDString(), createdChat.CommunityID)
|
|
|
|
s.Require().Equal(orgChat.Identity.DisplayName, createdChat.Name)
|
|
|
|
s.Require().Equal(orgChat.Identity.Emoji, createdChat.Emoji)
|
|
|
|
s.Require().NotEmpty(createdChat.ID)
|
|
|
|
s.Require().Equal(ChatTypeCommunityChat, createdChat.ChatType)
|
|
|
|
s.Require().Equal(categoryID, createdChat.CategoryID)
|
|
|
|
s.Require().True(createdChat.Active)
|
|
|
|
s.Require().NotEmpty(createdChat.Timestamp)
|
|
|
|
s.Require().True(strings.HasPrefix(createdChat.ID, community.IDString()))
|
|
|
|
}
|
2022-05-13 08:54:55 +02:00
|
|
|
}
|
2022-05-13 16:02:38 +02:00
|
|
|
s.Require().True(found)
|
2020-11-18 10:16:51 +01:00
|
|
|
|
|
|
|
// Create another org chat
|
|
|
|
orgChat = &protobuf.CommunityChat{
|
|
|
|
Permissions: &protobuf.CommunityPermissions{
|
|
|
|
Access: protobuf.CommunityPermissions_NO_MEMBERSHIP,
|
|
|
|
},
|
|
|
|
Identity: &protobuf.ChatIdentity{
|
|
|
|
DisplayName: "status-core-ui",
|
2021-10-04 18:32:25 +05:30
|
|
|
Emoji: "👍",
|
2020-11-18 10:16:51 +01:00
|
|
|
Description: "status-core-ui community chat",
|
|
|
|
},
|
|
|
|
}
|
2021-01-11 11:32:51 +01:00
|
|
|
response, err = s.bob.CreateCommunityChat(community.ID(), orgChat)
|
2020-11-18 10:16:51 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
2021-01-11 11:32:51 +01:00
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
s.Require().Len(response.Chats(), 1)
|
2020-11-18 10:16:51 +01:00
|
|
|
|
2021-10-06 12:44:24 +01:00
|
|
|
var actualChat *Chat
|
2020-11-18 10:16:51 +01:00
|
|
|
// Pull message, this time it should be received as advertised automatically
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = s.alice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-10-06 11:08:35 +01:00
|
|
|
if len(response.Communities()) != 1 {
|
2020-11-18 10:16:51 +01:00
|
|
|
return errors.New("community not received")
|
|
|
|
}
|
2021-10-06 12:44:24 +01:00
|
|
|
|
|
|
|
for _, c := range response.Chats() {
|
|
|
|
if c.Name == orgChat.Identity.DisplayName {
|
|
|
|
actualChat = c
|
|
|
|
return nil
|
|
|
|
}
|
2021-10-06 11:08:35 +01:00
|
|
|
}
|
2021-10-06 12:44:24 +01:00
|
|
|
return errors.New("chat not found")
|
2020-11-18 10:16:51 +01:00
|
|
|
})
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
communities, err = s.alice.Communities()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(communities, 2)
|
2021-01-11 11:32:51 +01:00
|
|
|
s.Require().Len(response.Communities(), 1)
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NotNil(actualChat)
|
2021-10-06 11:08:35 +01:00
|
|
|
s.Require().Equal(community.IDString(), actualChat.CommunityID)
|
|
|
|
s.Require().Equal(orgChat.Identity.DisplayName, actualChat.Name)
|
|
|
|
s.Require().Equal(orgChat.Identity.Emoji, actualChat.Emoji)
|
|
|
|
s.Require().NotEmpty(actualChat.ID)
|
|
|
|
s.Require().Equal(ChatTypeCommunityChat, actualChat.ChatType)
|
|
|
|
s.Require().True(actualChat.Active)
|
|
|
|
s.Require().NotEmpty(actualChat.Timestamp)
|
|
|
|
s.Require().True(strings.HasPrefix(actualChat.ID, community.IDString()))
|
2020-11-18 10:16:51 +01:00
|
|
|
|
|
|
|
// We leave the org
|
2021-01-11 11:32:51 +01:00
|
|
|
response, err = s.alice.LeaveCommunity(community.ID())
|
2020-11-18 10:16:51 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
2021-01-11 11:32:51 +01:00
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
s.Require().False(response.Communities()[0].Joined())
|
2022-05-10 15:25:08 +02:00
|
|
|
s.Require().Len(response.RemovedChats(), 3)
|
2020-11-18 10:16:51 +01:00
|
|
|
}
|
|
|
|
|
2022-08-23 18:30:19 +02:00
|
|
|
func (s *MessengerCommunitiesSuite) createCommunity() *communities.Community {
|
2023-06-27 14:20:20 +02:00
|
|
|
community, _ := createCommunity(&s.Suite, s.admin)
|
2022-08-23 18:30:19 +02:00
|
|
|
return community
|
|
|
|
}
|
2022-07-08 12:25:46 +02:00
|
|
|
|
2022-08-23 18:30:19 +02:00
|
|
|
func (s *MessengerCommunitiesSuite) advertiseCommunityTo(community *communities.Community, user *Messenger) {
|
2023-06-27 14:20:20 +02:00
|
|
|
advertiseCommunityTo(&s.Suite, community, s.admin, user)
|
2022-08-23 18:30:19 +02:00
|
|
|
}
|
|
|
|
|
2023-06-27 14:20:20 +02:00
|
|
|
func (s *MessengerCommunitiesSuite) joinCommunity(community *communities.Community, user *Messenger) {
|
|
|
|
request := &requests.RequestToJoinCommunity{CommunityID: community.ID()}
|
|
|
|
joinCommunity(&s.Suite, community, s.admin, user, request)
|
2023-06-22 14:59:07 -04:00
|
|
|
}
|
|
|
|
|
2022-08-23 18:30:19 +02:00
|
|
|
func (s *MessengerCommunitiesSuite) TestCommunityContactCodeAdvertisement() {
|
2023-05-24 16:42:31 +02:00
|
|
|
// add bob's profile keypair
|
|
|
|
bobProfileKp := accounts.GetProfileKeypairForTest(true, false, false)
|
|
|
|
bobProfileKp.KeyUID = s.bob.account.KeyUID
|
|
|
|
bobProfileKp.Accounts[0].KeyUID = s.bob.account.KeyUID
|
|
|
|
|
|
|
|
err := s.bob.settings.SaveOrUpdateKeypair(bobProfileKp)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
// create community and make bob and alice join to it
|
2022-08-23 18:30:19 +02:00
|
|
|
community := s.createCommunity()
|
|
|
|
s.advertiseCommunityTo(community, s.bob)
|
|
|
|
s.advertiseCommunityTo(community, s.alice)
|
|
|
|
|
2023-06-27 14:20:20 +02:00
|
|
|
s.joinCommunity(community, s.bob)
|
|
|
|
s.joinCommunity(community, s.alice)
|
2022-07-08 12:25:46 +02:00
|
|
|
|
|
|
|
// Trigger ContactCodeAdvertisement
|
2023-05-24 16:42:31 +02:00
|
|
|
err = s.bob.SetDisplayName("bobby")
|
2022-07-08 12:25:46 +02:00
|
|
|
s.Require().NoError(err)
|
2022-08-23 18:30:19 +02:00
|
|
|
err = s.bob.SetBio("I like P2P chats")
|
2022-08-05 13:22:35 +02:00
|
|
|
s.Require().NoError(err)
|
2022-07-08 12:25:46 +02:00
|
|
|
|
2022-08-23 18:30:19 +02:00
|
|
|
// Ensure alice receives bob's ContactCodeAdvertisement
|
2022-07-08 12:25:46 +02:00
|
|
|
err = tt.RetryWithBackOff(func() error {
|
2022-08-23 18:30:19 +02:00
|
|
|
response, err := s.alice.RetrieveAll()
|
2022-07-08 12:25:46 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(response.Contacts) == 0 {
|
|
|
|
return errors.New("no contacts in response")
|
|
|
|
}
|
2022-08-23 18:30:19 +02:00
|
|
|
if response.Contacts[0].DisplayName != "bobby" {
|
2022-07-08 12:25:46 +02:00
|
|
|
return errors.New("display name was not updated")
|
|
|
|
}
|
2022-08-05 13:22:35 +02:00
|
|
|
if response.Contacts[0].Bio != "I like P2P chats" {
|
|
|
|
return errors.New("bio was not updated")
|
|
|
|
}
|
2022-07-08 12:25:46 +02:00
|
|
|
return nil
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
}
|
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
func (s *MessengerCommunitiesSuite) TestInviteUsersToCommunity() {
|
|
|
|
description := &requests.CreateCommunity{
|
|
|
|
Membership: protobuf.CommunityPermissions_NO_MEMBERSHIP,
|
|
|
|
Name: "status",
|
|
|
|
Color: "#ffffff",
|
|
|
|
Description: "status community description",
|
2020-11-18 10:16:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Create an community chat
|
2022-07-19 11:31:52 +02:00
|
|
|
response, err := s.bob.CreateCommunity(description, true)
|
2020-11-18 10:16:51 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
2021-01-11 11:32:51 +01:00
|
|
|
s.Require().Len(response.Communities(), 1)
|
2021-03-31 18:23:45 +02:00
|
|
|
s.Require().True(response.Communities()[0].HasMember(&s.bob.identity.PublicKey))
|
2023-06-14 16:15:46 +02:00
|
|
|
s.Require().True(response.Communities()[0].IsMemberOwner(&s.bob.identity.PublicKey))
|
2020-11-18 10:16:51 +01:00
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
community := response.Communities()[0]
|
2020-11-18 10:16:51 +01:00
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
response, err = s.bob.InviteUsersToCommunity(
|
|
|
|
&requests.InviteUsersToCommunity{
|
|
|
|
CommunityID: community.ID(),
|
|
|
|
Users: []types.HexBytes{common.PubkeyToHexBytes(&s.alice.identity.PublicKey)},
|
|
|
|
},
|
|
|
|
)
|
2020-11-18 10:16:51 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
2021-01-11 11:32:51 +01:00
|
|
|
s.Require().Len(response.Communities(), 1)
|
2020-11-18 10:16:51 +01:00
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
community = response.Communities()[0]
|
2020-11-18 10:16:51 +01:00
|
|
|
s.Require().True(community.HasMember(&s.alice.identity.PublicKey))
|
|
|
|
|
|
|
|
// Pull message and make sure org is received
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = s.alice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-01-11 11:32:51 +01:00
|
|
|
if len(response.Communities()) == 0 {
|
2020-11-18 10:16:51 +01:00
|
|
|
return errors.New("community not received")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
communities, err := s.alice.Communities()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(communities, 2)
|
2021-01-11 11:32:51 +01:00
|
|
|
s.Require().Len(response.Communities(), 1)
|
2020-11-18 10:16:51 +01:00
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
community = response.Communities()[0]
|
2020-11-18 10:16:51 +01:00
|
|
|
s.Require().True(community.HasMember(&s.alice.identity.PublicKey))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *MessengerCommunitiesSuite) TestPostToCommunityChat() {
|
2021-01-11 11:32:51 +01:00
|
|
|
description := &requests.CreateCommunity{
|
|
|
|
Membership: protobuf.CommunityPermissions_INVITATION_ONLY,
|
|
|
|
Name: "status",
|
|
|
|
Color: "#ffffff",
|
|
|
|
Description: "status community description",
|
2020-11-18 10:16:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Create an community chat
|
2022-07-19 11:31:52 +02:00
|
|
|
response, err := s.bob.CreateCommunity(description, true)
|
2020-11-18 10:16:51 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
2021-01-11 11:32:51 +01:00
|
|
|
s.Require().Len(response.Communities(), 1)
|
2022-05-13 09:44:26 -04:00
|
|
|
s.Require().Len(response.Communities()[0].Chats(), 1)
|
|
|
|
s.Require().Len(response.Chats(), 1)
|
2020-11-18 10:16:51 +01:00
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
community := response.Communities()[0]
|
2020-11-18 10:16:51 +01:00
|
|
|
|
|
|
|
// Create chat
|
|
|
|
orgChat := &protobuf.CommunityChat{
|
|
|
|
Permissions: &protobuf.CommunityPermissions{
|
|
|
|
Access: protobuf.CommunityPermissions_NO_MEMBERSHIP,
|
|
|
|
},
|
|
|
|
Identity: &protobuf.ChatIdentity{
|
|
|
|
DisplayName: "status-core",
|
|
|
|
Description: "status-core community chat",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
response, err = s.bob.CreateCommunityChat(community.ID(), orgChat)
|
2020-11-18 10:16:51 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
2021-01-11 11:32:51 +01:00
|
|
|
s.Require().Len(response.Communities(), 1)
|
2022-05-13 09:44:26 -04:00
|
|
|
s.Require().Len(response.Communities()[0].Chats(), 2)
|
2021-01-11 11:32:51 +01:00
|
|
|
s.Require().Len(response.Chats(), 1)
|
2020-11-18 10:16:51 +01:00
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
response, err = s.bob.InviteUsersToCommunity(
|
|
|
|
&requests.InviteUsersToCommunity{
|
|
|
|
CommunityID: community.ID(),
|
|
|
|
Users: []types.HexBytes{common.PubkeyToHexBytes(&s.alice.identity.PublicKey)},
|
|
|
|
},
|
|
|
|
)
|
2020-11-18 10:16:51 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
2021-01-11 11:32:51 +01:00
|
|
|
s.Require().Len(response.Communities(), 1)
|
2020-11-18 10:16:51 +01:00
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
community = response.Communities()[0]
|
2020-11-18 10:16:51 +01:00
|
|
|
s.Require().True(community.HasMember(&s.alice.identity.PublicKey))
|
|
|
|
|
|
|
|
// Pull message and make sure org is received
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = s.alice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-01-11 11:32:51 +01:00
|
|
|
if len(response.Communities()) == 0 {
|
2020-11-18 10:16:51 +01:00
|
|
|
return errors.New("community not received")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
communities, err := s.alice.Communities()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(communities, 2)
|
2021-01-11 11:32:51 +01:00
|
|
|
s.Require().Len(response.Communities(), 1)
|
2020-11-18 10:16:51 +01:00
|
|
|
|
2022-05-27 12:14:40 +03:00
|
|
|
communityID := response.Communities()[0].ID()
|
|
|
|
s.Require().Equal(communityID, community.ID())
|
|
|
|
|
2021-07-22 13:41:49 -04:00
|
|
|
ctx := context.Background()
|
|
|
|
|
2020-11-18 10:16:51 +01:00
|
|
|
// We join the org
|
2023-05-29 13:57:05 -04:00
|
|
|
response, err = s.alice.JoinCommunity(ctx, community.ID(), false)
|
2020-11-18 10:16:51 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
2021-01-11 11:32:51 +01:00
|
|
|
s.Require().Len(response.Communities(), 1)
|
2022-05-13 09:44:26 -04:00
|
|
|
s.Require().Len(response.Communities()[0].Chats(), 2)
|
2021-01-11 11:32:51 +01:00
|
|
|
s.Require().True(response.Communities()[0].Joined())
|
2022-05-10 15:25:08 +02:00
|
|
|
s.Require().Len(response.Chats(), 2)
|
2020-11-18 10:16:51 +01:00
|
|
|
|
2022-05-10 15:25:08 +02:00
|
|
|
chatID := response.Chats()[1].ID
|
2020-11-18 10:16:51 +01:00
|
|
|
inputMessage := &common.Message{}
|
|
|
|
inputMessage.ChatId = chatID
|
|
|
|
inputMessage.ContentType = protobuf.ChatMessage_TEXT_PLAIN
|
|
|
|
inputMessage.Text = "some text"
|
|
|
|
|
2021-07-22 13:41:49 -04:00
|
|
|
_, err = s.alice.SendChatMessage(ctx, inputMessage)
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err)
|
2020-11-18 10:16:51 +01:00
|
|
|
|
|
|
|
// Pull message and make sure org is received
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = s.bob.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-06-03 15:11:55 +02:00
|
|
|
if len(response.messages) == 0 {
|
2020-11-18 10:16:51 +01:00
|
|
|
return errors.New("message not received")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
2021-06-03 15:11:55 +02:00
|
|
|
s.Require().Len(response.Messages(), 1)
|
2022-09-05 13:08:44 +01:00
|
|
|
s.Require().Equal(inputMessage.Text, response.Messages()[0].Text)
|
2022-05-13 09:44:26 -04:00
|
|
|
|
|
|
|
// check if response contains the chat we're interested in
|
|
|
|
// we use this instead of checking just the length of the chat because
|
|
|
|
// a CommunityDescription message might be received in the meantime due to syncing
|
|
|
|
// hence response.Chats() might contain the general chat, and the new chat;
|
|
|
|
// or only the new chat if the CommunityDescription message has not arrived
|
|
|
|
found := false
|
|
|
|
for _, chat := range response.Chats() {
|
|
|
|
if chat.ID == chatID {
|
|
|
|
found = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
s.Require().True(found)
|
2020-11-18 10:16:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *MessengerCommunitiesSuite) TestImportCommunity() {
|
2021-07-22 13:41:49 -04:00
|
|
|
ctx := context.Background()
|
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
description := &requests.CreateCommunity{
|
|
|
|
Membership: protobuf.CommunityPermissions_NO_MEMBERSHIP,
|
|
|
|
Name: "status",
|
|
|
|
Color: "#ffffff",
|
|
|
|
Description: "status community description",
|
2020-11-18 10:16:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Create an community chat
|
2022-07-19 11:31:52 +02:00
|
|
|
response, err := s.bob.CreateCommunity(description, true)
|
2020-11-18 10:16:51 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
2021-01-11 11:32:51 +01:00
|
|
|
s.Require().Len(response.Communities(), 1)
|
2022-03-08 16:25:00 +01:00
|
|
|
s.Require().Len(response.CommunitiesSettings(), 1)
|
2022-07-01 15:54:02 +02:00
|
|
|
s.Require().True(response.Communities()[0].Joined())
|
2023-06-14 16:15:46 +02:00
|
|
|
s.Require().True(response.Communities()[0].IsOwner())
|
2023-07-21 13:41:26 +04:00
|
|
|
s.Require().True(response.Communities()[0].IsControlNode())
|
2020-11-18 10:16:51 +01:00
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
community := response.Communities()[0]
|
2022-03-08 16:25:00 +01:00
|
|
|
communitySettings := response.CommunitiesSettings()[0]
|
|
|
|
|
|
|
|
s.Require().Equal(communitySettings.CommunityID, community.IDString())
|
|
|
|
s.Require().Equal(communitySettings.HistoryArchiveSupportEnabled, false)
|
2020-11-18 10:16:51 +01:00
|
|
|
|
2021-05-23 09:34:17 -04:00
|
|
|
category := &requests.CreateCommunityCategory{
|
|
|
|
CommunityID: community.ID(),
|
|
|
|
CategoryName: "category-name",
|
|
|
|
ChatIDs: []string{},
|
|
|
|
}
|
|
|
|
|
|
|
|
response, err = s.bob.CreateCommunityCategory(category)
|
2022-03-28 11:10:40 +01:00
|
|
|
s.Require().NoError(err)
|
2021-05-23 09:34:17 -04:00
|
|
|
community = response.Communities()[0]
|
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
privateKey, err := s.bob.ExportCommunity(community.ID())
|
2020-11-18 10:16:51 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
|
2021-07-22 13:41:49 -04:00
|
|
|
_, err = s.alice.ImportCommunity(ctx, privateKey)
|
2020-11-18 10:16:51 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
// Invite user on bob side
|
|
|
|
newUser, err := crypto.GenerateKey()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
_, err = s.bob.InviteUsersToCommunity(
|
|
|
|
&requests.InviteUsersToCommunity{
|
|
|
|
CommunityID: community.ID(),
|
|
|
|
Users: []types.HexBytes{common.PubkeyToHexBytes(&newUser.PublicKey)},
|
|
|
|
},
|
|
|
|
)
|
2020-11-18 10:16:51 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
// Pull message and make sure org is received
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = s.alice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-01-11 11:32:51 +01:00
|
|
|
if len(response.Communities()) == 0 {
|
2020-11-18 10:16:51 +01:00
|
|
|
return errors.New("community not received")
|
|
|
|
}
|
2023-07-21 13:41:26 +04:00
|
|
|
if !response.Communities()[0].IsControlNode() {
|
2022-07-01 15:54:02 +02:00
|
|
|
return errors.New("isn't admin despite import")
|
|
|
|
}
|
2020-11-18 10:16:51 +01:00
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
}
|
2021-01-11 11:32:51 +01:00
|
|
|
|
2023-07-21 13:41:26 +04:00
|
|
|
func (s *MessengerCommunitiesSuite) TestRemovePrivateKey() {
|
|
|
|
description := &requests.CreateCommunity{
|
|
|
|
Membership: protobuf.CommunityPermissions_NO_MEMBERSHIP,
|
|
|
|
Name: "status",
|
|
|
|
Color: "#ffffff",
|
|
|
|
Description: "status community description",
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create an community chat
|
|
|
|
response, err := s.bob.CreateCommunity(description, true)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
|
|
|
|
community := response.Communities()[0]
|
|
|
|
s.Require().True(community.IsOwner())
|
|
|
|
s.Require().True(community.IsControlNode())
|
|
|
|
|
|
|
|
response, err = s.bob.RemovePrivateKey(community.ID())
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
|
|
|
|
community = response.Communities()[0]
|
|
|
|
s.Require().True(community.IsOwner())
|
|
|
|
s.Require().False(community.IsControlNode())
|
|
|
|
}
|
|
|
|
|
2022-09-21 12:50:56 +02:00
|
|
|
func (s *MessengerCommunitiesSuite) TestRolesAfterImportCommunity() {
|
|
|
|
ctx := context.Background()
|
|
|
|
|
|
|
|
description := &requests.CreateCommunity{
|
|
|
|
Membership: protobuf.CommunityPermissions_NO_MEMBERSHIP,
|
|
|
|
Name: "status",
|
|
|
|
Color: "#ffffff",
|
|
|
|
Description: "status community description",
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create a community chat
|
|
|
|
response, err := s.bob.CreateCommunity(description, true)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
s.Require().Len(response.CommunitiesSettings(), 1)
|
|
|
|
s.Require().True(response.Communities()[0].Joined())
|
2023-06-14 16:15:46 +02:00
|
|
|
s.Require().True(response.Communities()[0].IsOwner())
|
|
|
|
s.Require().True(response.Communities()[0].IsMemberOwner(&s.bob.identity.PublicKey))
|
|
|
|
s.Require().False(response.Communities()[0].IsMemberOwner(&s.alice.identity.PublicKey))
|
2022-09-21 12:50:56 +02:00
|
|
|
|
|
|
|
community := response.Communities()[0]
|
|
|
|
communitySettings := response.CommunitiesSettings()[0]
|
|
|
|
|
|
|
|
s.Require().Equal(communitySettings.CommunityID, community.IDString())
|
|
|
|
s.Require().Equal(communitySettings.HistoryArchiveSupportEnabled, false)
|
|
|
|
|
|
|
|
category := &requests.CreateCommunityCategory{
|
|
|
|
CommunityID: community.ID(),
|
|
|
|
CategoryName: "category-name",
|
|
|
|
ChatIDs: []string{},
|
|
|
|
}
|
|
|
|
|
|
|
|
response, err = s.bob.CreateCommunityCategory(category)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
community = response.Communities()[0]
|
|
|
|
|
|
|
|
privateKey, err := s.bob.ExportCommunity(community.ID())
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
response, err = s.alice.ImportCommunity(ctx, privateKey)
|
|
|
|
s.Require().NoError(err)
|
2023-06-14 16:15:46 +02:00
|
|
|
s.Require().True(response.Communities()[0].IsMemberOwner(&s.alice.identity.PublicKey))
|
2022-09-21 12:50:56 +02:00
|
|
|
}
|
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
func (s *MessengerCommunitiesSuite) TestRequestAccess() {
|
2021-07-22 13:41:49 -04:00
|
|
|
ctx := context.Background()
|
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
description := &requests.CreateCommunity{
|
|
|
|
Membership: protobuf.CommunityPermissions_ON_REQUEST,
|
|
|
|
Name: "status",
|
|
|
|
Color: "#ffffff",
|
|
|
|
Description: "status community description",
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create an community chat
|
2022-07-19 11:31:52 +02:00
|
|
|
response, err := s.bob.CreateCommunity(description, true)
|
2021-01-11 11:32:51 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
|
|
|
|
community := response.Communities()[0]
|
|
|
|
|
|
|
|
chat := CreateOneToOneChat(common.PubkeyToHex(&s.alice.identity.PublicKey), &s.alice.identity.PublicKey, s.alice.transport)
|
|
|
|
|
|
|
|
s.Require().NoError(s.bob.SaveChat(chat))
|
|
|
|
|
|
|
|
message := buildTestMessage(*chat)
|
|
|
|
message.CommunityID = community.IDString()
|
|
|
|
|
|
|
|
// We send a community link to alice
|
2021-07-22 13:41:49 -04:00
|
|
|
response, err = s.bob.SendChatMessage(ctx, message)
|
2021-01-11 11:32:51 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
|
|
|
|
// Retrieve community link & community
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = s.alice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(response.Communities()) == 0 {
|
|
|
|
return errors.New("message not received")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
request := &requests.RequestToJoinCommunity{CommunityID: community.ID()}
|
|
|
|
// We try to join the org
|
|
|
|
response, err = s.alice.RequestToJoinCommunity(request)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.RequestsToJoinCommunity, 1)
|
|
|
|
|
2022-10-26 01:06:20 +03:00
|
|
|
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)
|
2023-01-28 13:52:53 +04:00
|
|
|
s.Require().Equal(notification.Read, true)
|
|
|
|
s.Require().Equal(notification.Accepted, false)
|
|
|
|
s.Require().Equal(notification.Dismissed, false)
|
2022-10-26 01:06:20 +03:00
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
requestToJoin1 := response.RequestsToJoinCommunity[0]
|
|
|
|
s.Require().NotNil(requestToJoin1)
|
|
|
|
s.Require().Equal(community.ID(), requestToJoin1.CommunityID)
|
|
|
|
s.Require().True(requestToJoin1.Our)
|
|
|
|
s.Require().NotEmpty(requestToJoin1.ID)
|
|
|
|
s.Require().NotEmpty(requestToJoin1.Clock)
|
|
|
|
s.Require().Equal(requestToJoin1.PublicKey, common.PubkeyToHex(&s.alice.identity.PublicKey))
|
|
|
|
s.Require().Equal(communities.RequestToJoinStatePending, requestToJoin1.State)
|
|
|
|
|
|
|
|
// Make sure clock is not empty
|
|
|
|
s.Require().NotEmpty(requestToJoin1.Clock)
|
|
|
|
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
s.Require().Equal(response.Communities()[0].RequestedToJoinAt(), requestToJoin1.Clock)
|
|
|
|
|
|
|
|
// pull all communities to make sure we set RequestedToJoinAt
|
|
|
|
|
|
|
|
allCommunities, err := s.alice.Communities()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(allCommunities, 2)
|
|
|
|
|
|
|
|
if bytes.Equal(allCommunities[0].ID(), community.ID()) {
|
|
|
|
s.Require().Equal(allCommunities[0].RequestedToJoinAt(), requestToJoin1.Clock)
|
|
|
|
} else {
|
|
|
|
s.Require().Equal(allCommunities[1].RequestedToJoinAt(), requestToJoin1.Clock)
|
|
|
|
}
|
|
|
|
|
|
|
|
// pull to make sure it has been saved
|
|
|
|
requestsToJoin, err := s.alice.MyPendingRequestsToJoin()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(requestsToJoin, 1)
|
|
|
|
|
|
|
|
// Make sure the requests are fetched also by community
|
|
|
|
requestsToJoin, err = s.alice.PendingRequestsToJoinForCommunity(community.ID())
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(requestsToJoin, 1)
|
|
|
|
|
|
|
|
// Retrieve request to join
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = s.bob.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(response.RequestsToJoinCommunity) == 0 {
|
|
|
|
return errors.New("request to join community not received")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(response.RequestsToJoinCommunity, 1)
|
|
|
|
|
|
|
|
requestToJoin2 := response.RequestsToJoinCommunity[0]
|
|
|
|
|
|
|
|
s.Require().NotNil(requestToJoin2)
|
|
|
|
s.Require().Equal(community.ID(), requestToJoin2.CommunityID)
|
|
|
|
s.Require().False(requestToJoin2.Our)
|
|
|
|
s.Require().NotEmpty(requestToJoin2.ID)
|
|
|
|
s.Require().NotEmpty(requestToJoin2.Clock)
|
|
|
|
s.Require().Equal(requestToJoin2.PublicKey, common.PubkeyToHex(&s.alice.identity.PublicKey))
|
|
|
|
s.Require().Equal(communities.RequestToJoinStatePending, requestToJoin2.State)
|
|
|
|
|
|
|
|
s.Require().Equal(requestToJoin1.ID, requestToJoin2.ID)
|
|
|
|
|
2023-04-14 13:02:28 +04:00
|
|
|
s.Require().Len(response.ActivityCenterNotifications(), 1)
|
|
|
|
|
|
|
|
notification = response.ActivityCenterNotifications()[0]
|
|
|
|
s.Require().NotNil(notification)
|
|
|
|
s.Require().Equal(notification.Type, ActivityCenterNotificationTypeCommunityMembershipRequest)
|
|
|
|
s.Require().Equal(notification.MembershipStatus, ActivityCenterMembershipStatusPending)
|
|
|
|
s.Require().Equal(notification.Read, false)
|
|
|
|
s.Require().Equal(notification.Accepted, false)
|
|
|
|
s.Require().Equal(notification.Dismissed, false)
|
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
// Accept request
|
|
|
|
|
|
|
|
acceptRequestToJoin := &requests.AcceptRequestToJoinCommunity{ID: requestToJoin1.ID}
|
|
|
|
|
|
|
|
response, err = s.bob.AcceptRequestToJoinCommunity(acceptRequestToJoin)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
|
|
|
|
updatedCommunity := response.Communities()[0]
|
|
|
|
|
|
|
|
s.Require().NotNil(updatedCommunity)
|
|
|
|
s.Require().True(updatedCommunity.HasMember(&s.alice.identity.PublicKey))
|
|
|
|
|
2022-10-26 01:06:20 +03:00
|
|
|
s.Require().Len(response.ActivityCenterNotifications(), 1)
|
|
|
|
|
|
|
|
notification = response.ActivityCenterNotifications()[0]
|
|
|
|
s.Require().NotNil(notification)
|
|
|
|
s.Require().Equal(notification.Type, ActivityCenterNotificationTypeCommunityMembershipRequest)
|
|
|
|
s.Require().Equal(notification.MembershipStatus, ActivityCenterMembershipStatusAccepted)
|
2023-01-28 13:52:53 +04:00
|
|
|
s.Require().Equal(notification.Read, true)
|
|
|
|
s.Require().Equal(notification.Accepted, true)
|
|
|
|
s.Require().Equal(notification.Dismissed, false)
|
2022-10-26 01:06:20 +03:00
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
// Pull message and make sure org is received
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = s.alice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(response.Communities()) == 0 {
|
|
|
|
return errors.New("community not received")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
|
2023-02-03 16:33:16 +00:00
|
|
|
s.Require().Len(response.RequestsToJoinCommunity, 1)
|
|
|
|
s.Require().Equal(communities.RequestToJoinStateAccepted, response.RequestsToJoinCommunity[0].State)
|
2021-01-11 11:32:51 +01:00
|
|
|
|
2023-04-14 13:02:28 +04:00
|
|
|
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, ActivityCenterMembershipStatusAccepted)
|
2023-04-21 17:18:47 +08:00
|
|
|
s.Require().Equal(notification.Read, false)
|
2023-04-14 13:02:28 +04:00
|
|
|
s.Require().Equal(notification.Accepted, false)
|
|
|
|
s.Require().Equal(notification.Dismissed, false)
|
|
|
|
|
2023-02-03 16:33:16 +00:00
|
|
|
s.Require().Len(response.Communities(), 1)
|
2021-01-11 11:32:51 +01:00
|
|
|
aliceCommunity := response.Communities()[0]
|
|
|
|
|
|
|
|
s.Require().Equal(community.ID(), aliceCommunity.ID())
|
|
|
|
s.Require().True(aliceCommunity.HasMember(&s.alice.identity.PublicKey))
|
|
|
|
|
|
|
|
// Community should be joined at this point
|
|
|
|
s.Require().True(aliceCommunity.Joined())
|
|
|
|
|
|
|
|
// Make sure the requests are not pending on either sides
|
|
|
|
requestsToJoin, err = s.bob.PendingRequestsToJoinForCommunity(community.ID())
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(requestsToJoin, 0)
|
|
|
|
|
|
|
|
requestsToJoin, err = s.alice.MyPendingRequestsToJoin()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(requestsToJoin, 0)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2023-04-21 17:18:47 +08:00
|
|
|
func (s *MessengerCommunitiesSuite) TestDeletePendingRequestAccess() {
|
|
|
|
ctx := context.Background()
|
|
|
|
|
|
|
|
description := &requests.CreateCommunity{
|
|
|
|
Membership: protobuf.CommunityPermissions_ON_REQUEST,
|
|
|
|
Name: "status",
|
|
|
|
Color: "#ffffff",
|
|
|
|
Description: "status community description",
|
|
|
|
}
|
|
|
|
|
|
|
|
// Bob creates a community
|
|
|
|
response, err := s.bob.CreateCommunity(description, true)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
|
|
|
|
community := response.Communities()[0]
|
|
|
|
|
|
|
|
chat := CreateOneToOneChat(common.PubkeyToHex(&s.alice.identity.PublicKey), &s.alice.identity.PublicKey, s.alice.transport)
|
|
|
|
|
|
|
|
s.Require().NoError(s.bob.SaveChat(chat))
|
|
|
|
|
|
|
|
message := buildTestMessage(*chat)
|
|
|
|
message.CommunityID = community.IDString()
|
|
|
|
|
|
|
|
// Bob sends the community link to Alice
|
|
|
|
response, err = s.bob.SendChatMessage(ctx, message)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
|
|
|
|
// Retrieve community link & community for Alice
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = s.alice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(response.Communities()) == 0 {
|
|
|
|
return errors.New("message not received")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
// Alice request to join community
|
|
|
|
request := &requests.RequestToJoinCommunity{CommunityID: community.ID()}
|
|
|
|
response, err = s.alice.RequestToJoinCommunity(request)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.RequestsToJoinCommunity, 1)
|
|
|
|
|
|
|
|
requestToJoin := response.RequestsToJoinCommunity[0]
|
|
|
|
s.Require().NotNil(requestToJoin)
|
|
|
|
s.Require().Equal(community.ID(), requestToJoin.CommunityID)
|
|
|
|
s.Require().NotEmpty(requestToJoin.ID)
|
|
|
|
s.Require().NotEmpty(requestToJoin.Clock)
|
|
|
|
s.Require().Equal(requestToJoin.PublicKey, common.PubkeyToHex(&s.alice.identity.PublicKey))
|
|
|
|
s.Require().Equal(communities.RequestToJoinStatePending, requestToJoin.State)
|
|
|
|
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
s.Require().Equal(response.Communities()[0].RequestedToJoinAt(), requestToJoin.Clock)
|
|
|
|
|
|
|
|
// updating request clock by 8 days back
|
|
|
|
requestTime := uint64(time.Now().AddDate(0, 0, -8).Unix())
|
|
|
|
err = s.alice.communitiesManager.UpdateClockInRequestToJoin(requestToJoin.ID, requestTime)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
// pull to make sure it has been saved
|
|
|
|
requestsToJoin, err := s.alice.MyPendingRequestsToJoin()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(requestsToJoin, 1)
|
|
|
|
|
|
|
|
requestToJoin = requestsToJoin[0]
|
|
|
|
s.Require().Equal(requestToJoin.Clock, requestTime)
|
|
|
|
|
|
|
|
// Make sure the requests are fetched also by community
|
|
|
|
requestsToJoin, err = s.alice.PendingRequestsToJoinForCommunity(community.ID())
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(requestsToJoin, 1)
|
|
|
|
|
|
|
|
// Retrieve request to join
|
|
|
|
bobRetrieveAll := func() (*MessengerResponse, error) {
|
|
|
|
return s.bob.RetrieveAll()
|
|
|
|
}
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = bobRetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(response.RequestsToJoinCommunity) == 0 {
|
|
|
|
return errors.New("request to join community not received")
|
|
|
|
}
|
|
|
|
|
|
|
|
// updating request clock by 8 days back
|
|
|
|
requestToJoin := response.RequestsToJoinCommunity[0]
|
|
|
|
err = s.bob.communitiesManager.UpdateClockInRequestToJoin(requestToJoin.ID, requestTime)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(response.ActivityCenterNotifications()) == 0 {
|
|
|
|
return errors.New("request to join community notification not added in activity center")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(response.RequestsToJoinCommunity, 1)
|
|
|
|
|
|
|
|
// Check activity center notification for Bob
|
|
|
|
fetchActivityCenterNotificationsForAdmin := func() (*ActivityCenterPaginationResponse, error) {
|
|
|
|
return s.bob.ActivityCenterNotifications(ActivityCenterNotificationsRequest{
|
|
|
|
Cursor: "",
|
|
|
|
Limit: 10,
|
|
|
|
ActivityTypes: []ActivityCenterType{},
|
|
|
|
ReadType: ActivityCenterQueryParamsReadUnread,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
notifications, err := fetchActivityCenterNotificationsForAdmin()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(notifications.Notifications, 1)
|
|
|
|
|
|
|
|
notification := notifications.Notifications[0]
|
|
|
|
s.Require().Equal(notification.Type, ActivityCenterNotificationTypeCommunityMembershipRequest)
|
|
|
|
s.Require().Equal(notification.MembershipStatus, ActivityCenterMembershipStatusPending)
|
|
|
|
|
|
|
|
// Delete pending request to join
|
|
|
|
response, err = s.alice.CheckAndDeletePendingRequestToJoinCommunity(true)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(response.RequestsToJoinCommunity, 1)
|
|
|
|
s.Require().Len(response.ActivityCenterNotifications(), 1)
|
|
|
|
|
|
|
|
requestToJoin = response.RequestsToJoinCommunity[0]
|
|
|
|
s.Require().True(requestToJoin.Deleted)
|
|
|
|
|
|
|
|
notification = response.ActivityCenterNotifications()[0]
|
|
|
|
s.Require().Equal(notification.Type, ActivityCenterNotificationTypeCommunityRequest)
|
|
|
|
s.Require().Equal(notification.MembershipStatus, ActivityCenterMembershipStatusIdle)
|
|
|
|
|
|
|
|
response, err = s.bob.CheckAndDeletePendingRequestToJoinCommunity(true)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(response.RequestsToJoinCommunity, 1)
|
|
|
|
s.Require().Len(response.ActivityCenterNotifications(), 1)
|
|
|
|
|
|
|
|
requestToJoin = response.RequestsToJoinCommunity[0]
|
|
|
|
s.Require().True(requestToJoin.Deleted)
|
|
|
|
|
|
|
|
notification = response.ActivityCenterNotifications()[0]
|
|
|
|
s.Require().Equal(notification.Type, ActivityCenterNotificationTypeCommunityMembershipRequest)
|
|
|
|
s.Require().True(notification.Deleted)
|
|
|
|
|
|
|
|
// Alice request to join community
|
|
|
|
request = &requests.RequestToJoinCommunity{CommunityID: community.ID()}
|
|
|
|
response, err = s.alice.RequestToJoinCommunity(request)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.RequestsToJoinCommunity, 1)
|
|
|
|
|
|
|
|
// Retrieve request to join and Check activity center notification for Bob
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = bobRetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(response.RequestsToJoinCommunity) == 0 {
|
|
|
|
return errors.New("request to join community not received")
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(response.ActivityCenterNotifications()) == 0 {
|
|
|
|
return errors.New("request to join community notification not added in activity center")
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(response.RequestsToJoinCommunity, 1)
|
|
|
|
|
|
|
|
// Check activity center notification for Bob
|
|
|
|
notifications, err = fetchActivityCenterNotificationsForAdmin()
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(notifications.Notifications, 1)
|
|
|
|
|
|
|
|
notification = notifications.Notifications[0]
|
|
|
|
s.Require().Equal(notification.Type, ActivityCenterNotificationTypeCommunityMembershipRequest)
|
|
|
|
s.Require().Equal(notification.MembershipStatus, ActivityCenterMembershipStatusPending)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *MessengerCommunitiesSuite) TestDeletePendingRequestAccessWithDeclinedState() {
|
|
|
|
ctx := context.Background()
|
|
|
|
|
|
|
|
description := &requests.CreateCommunity{
|
|
|
|
Membership: protobuf.CommunityPermissions_ON_REQUEST,
|
|
|
|
Name: "status",
|
|
|
|
Color: "#ffffff",
|
|
|
|
Description: "status community description",
|
|
|
|
}
|
|
|
|
|
|
|
|
// Bob creates a community
|
|
|
|
response, err := s.bob.CreateCommunity(description, true)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
|
|
|
|
community := response.Communities()[0]
|
|
|
|
|
|
|
|
chat := CreateOneToOneChat(common.PubkeyToHex(&s.alice.identity.PublicKey), &s.alice.identity.PublicKey, s.alice.transport)
|
|
|
|
|
|
|
|
s.Require().NoError(s.bob.SaveChat(chat))
|
|
|
|
|
|
|
|
message := buildTestMessage(*chat)
|
|
|
|
message.CommunityID = community.IDString()
|
|
|
|
|
|
|
|
// Bob sends the community link to Alice
|
|
|
|
response, err = s.bob.SendChatMessage(ctx, message)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
|
|
|
|
// Retrieve community link & community for Alice
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = s.alice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(response.Communities()) == 0 {
|
|
|
|
return errors.New("message not received")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
// Alice request to join community
|
|
|
|
request := &requests.RequestToJoinCommunity{CommunityID: community.ID()}
|
|
|
|
response, err = s.alice.RequestToJoinCommunity(request)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.RequestsToJoinCommunity, 1)
|
|
|
|
|
|
|
|
notification := response.ActivityCenterNotifications()[0]
|
|
|
|
s.Require().NotNil(notification)
|
|
|
|
s.Require().NotEmpty(notification.ID)
|
|
|
|
s.Require().Equal(notification.Type, ActivityCenterNotificationTypeCommunityRequest)
|
|
|
|
s.Require().Equal(notification.MembershipStatus, ActivityCenterMembershipStatusPending)
|
|
|
|
s.Require().Equal(notification.Deleted, false)
|
|
|
|
s.Require().Equal(notification.Read, true)
|
|
|
|
|
|
|
|
requestToJoin := response.RequestsToJoinCommunity[0]
|
|
|
|
s.Require().NotNil(requestToJoin)
|
|
|
|
s.Require().Equal(community.ID(), requestToJoin.CommunityID)
|
|
|
|
s.Require().NotEmpty(requestToJoin.ID)
|
|
|
|
s.Require().NotEmpty(requestToJoin.Clock)
|
|
|
|
s.Require().Equal(requestToJoin.PublicKey, common.PubkeyToHex(&s.alice.identity.PublicKey))
|
|
|
|
s.Require().Equal(communities.RequestToJoinStatePending, requestToJoin.State)
|
|
|
|
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
s.Require().Equal(response.Communities()[0].RequestedToJoinAt(), requestToJoin.Clock)
|
|
|
|
|
|
|
|
// Alice deletes activity center notification
|
|
|
|
err = s.alice.DeleteActivityCenterNotifications(ctx, []types.HexBytes{notification.ID}, false)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
// Check activity center notification for Bob after deleting
|
|
|
|
notifications, err := s.alice.ActivityCenterNotifications(ActivityCenterNotificationsRequest{
|
|
|
|
Cursor: "",
|
|
|
|
Limit: 10,
|
|
|
|
ActivityTypes: []ActivityCenterType{},
|
|
|
|
ReadType: ActivityCenterQueryParamsReadUnread,
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(notifications.Notifications, 0)
|
|
|
|
|
|
|
|
// updating request clock by 8 days back
|
|
|
|
requestTime := uint64(time.Now().AddDate(0, 0, -8).Unix())
|
|
|
|
err = s.alice.communitiesManager.UpdateClockInRequestToJoin(requestToJoin.ID, requestTime)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
// pull to make sure it has been saved
|
|
|
|
requestsToJoin, err := s.alice.MyPendingRequestsToJoin()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(requestsToJoin, 1)
|
|
|
|
|
|
|
|
requestToJoin = requestsToJoin[0]
|
|
|
|
s.Require().Equal(requestToJoin.Clock, requestTime)
|
|
|
|
|
|
|
|
// Make sure the requests are fetched also by community
|
|
|
|
requestsToJoin, err = s.alice.PendingRequestsToJoinForCommunity(community.ID())
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(requestsToJoin, 1)
|
|
|
|
|
|
|
|
bobRetrieveAll := func() (*MessengerResponse, error) {
|
|
|
|
return s.bob.RetrieveAll()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Retrieve request to join
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = bobRetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(response.RequestsToJoinCommunity) == 0 {
|
|
|
|
return errors.New("request to join community not received")
|
|
|
|
}
|
|
|
|
|
|
|
|
// updating request clock by 8 days back
|
|
|
|
requestToJoin := response.RequestsToJoinCommunity[0]
|
|
|
|
err = s.bob.communitiesManager.UpdateClockInRequestToJoin(requestToJoin.ID, requestTime)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(response.ActivityCenterNotifications()) == 0 {
|
|
|
|
return errors.New("request to join community notification not added in activity center")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(response.RequestsToJoinCommunity, 1)
|
|
|
|
|
|
|
|
// Check activity center notification for Bob
|
|
|
|
fetchActivityCenterNotificationsForAdmin := func() (*ActivityCenterPaginationResponse, error) {
|
|
|
|
return s.bob.ActivityCenterNotifications(ActivityCenterNotificationsRequest{
|
|
|
|
Cursor: "",
|
|
|
|
Limit: 10,
|
|
|
|
ActivityTypes: []ActivityCenterType{},
|
|
|
|
ReadType: ActivityCenterQueryParamsReadUnread,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
notifications, err = fetchActivityCenterNotificationsForAdmin()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(notifications.Notifications, 1)
|
|
|
|
|
|
|
|
notification = notifications.Notifications[0]
|
|
|
|
s.Require().Equal(notification.Type, ActivityCenterNotificationTypeCommunityMembershipRequest)
|
|
|
|
s.Require().Equal(notification.MembershipStatus, ActivityCenterMembershipStatusPending)
|
|
|
|
|
|
|
|
// Check if admin sees requests correctly
|
|
|
|
requestsToJoin, err = s.bob.PendingRequestsToJoinForCommunity(community.ID())
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(requestsToJoin, 1)
|
|
|
|
|
|
|
|
requestsToJoin, err = s.bob.DeclinedRequestsToJoinForCommunity(community.ID())
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(requestsToJoin, 0)
|
|
|
|
|
|
|
|
// Decline request
|
|
|
|
declinedRequestToJoin := &requests.DeclineRequestToJoinCommunity{ID: requestToJoin.ID}
|
|
|
|
response, err = s.bob.DeclineRequestToJoinCommunity(declinedRequestToJoin)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.ActivityCenterNotifications(), 1)
|
|
|
|
|
|
|
|
notification = response.ActivityCenterNotifications()[0]
|
|
|
|
s.Require().NotNil(notification)
|
|
|
|
s.Require().Equal(notification.Type, ActivityCenterNotificationTypeCommunityMembershipRequest)
|
|
|
|
s.Require().Equal(notification.MembershipStatus, ActivityCenterMembershipStatusDeclined)
|
|
|
|
s.Require().Equal(notification.Read, true)
|
|
|
|
s.Require().Equal(notification.Accepted, false)
|
|
|
|
s.Require().Equal(notification.Dismissed, true)
|
|
|
|
|
|
|
|
// Check if admin sees requests correctly
|
|
|
|
requestsToJoin, err = s.bob.PendingRequestsToJoinForCommunity(community.ID())
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(requestsToJoin, 0)
|
|
|
|
|
|
|
|
requestsToJoin, err = s.bob.DeclinedRequestsToJoinForCommunity(community.ID())
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(requestsToJoin, 1)
|
|
|
|
|
|
|
|
// Bob deletes activity center notification
|
|
|
|
err = s.bob.DeleteActivityCenterNotifications(ctx, []types.HexBytes{notification.ID}, false)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
// Check activity center notification for Bob after deleting
|
|
|
|
notifications, err = fetchActivityCenterNotificationsForAdmin()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(notifications.Notifications, 0)
|
|
|
|
|
|
|
|
// Delete pending request to join
|
|
|
|
response, err = s.alice.CheckAndDeletePendingRequestToJoinCommunity(true)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(response.RequestsToJoinCommunity, 1)
|
|
|
|
|
|
|
|
requestToJoin = response.RequestsToJoinCommunity[0]
|
|
|
|
s.Require().True(requestToJoin.Deleted)
|
|
|
|
|
|
|
|
notification = response.ActivityCenterNotifications()[0]
|
|
|
|
s.Require().NotNil(notification)
|
|
|
|
s.Require().Equal(notification.Type, ActivityCenterNotificationTypeCommunityRequest)
|
|
|
|
s.Require().Equal(notification.MembershipStatus, ActivityCenterMembershipStatusIdle)
|
|
|
|
s.Require().Equal(notification.Read, false)
|
|
|
|
s.Require().Equal(notification.Deleted, false)
|
|
|
|
|
|
|
|
notificationState := response.ActivityCenterState()
|
|
|
|
s.Require().False(notificationState.HasSeen)
|
|
|
|
|
|
|
|
// Alice request to join community
|
|
|
|
request = &requests.RequestToJoinCommunity{CommunityID: community.ID()}
|
|
|
|
response, err = s.alice.RequestToJoinCommunity(request)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.RequestsToJoinCommunity, 1)
|
|
|
|
|
|
|
|
// Retrieve request to join and Check activity center notification for Bob
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = bobRetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(response.RequestsToJoinCommunity) == 0 {
|
|
|
|
return errors.New("request to join community not received")
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(response.ActivityCenterNotifications()) == 0 {
|
|
|
|
return errors.New("request to join community notification not added in activity center")
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(response.RequestsToJoinCommunity, 1)
|
|
|
|
|
|
|
|
// Check activity center notification for Bob
|
|
|
|
notifications, err = fetchActivityCenterNotificationsForAdmin()
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(notifications.Notifications, 1)
|
|
|
|
|
|
|
|
notification = notifications.Notifications[0]
|
|
|
|
s.Require().Equal(notification.Type, ActivityCenterNotificationTypeCommunityMembershipRequest)
|
|
|
|
s.Require().Equal(notification.MembershipStatus, ActivityCenterMembershipStatusPending)
|
|
|
|
s.Require().False(notification.Deleted)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2022-10-28 11:41:20 +03:00
|
|
|
func (s *MessengerCommunitiesSuite) TestCancelRequestAccess() {
|
|
|
|
ctx := context.Background()
|
|
|
|
|
|
|
|
description := &requests.CreateCommunity{
|
|
|
|
Membership: protobuf.CommunityPermissions_ON_REQUEST,
|
|
|
|
Name: "status",
|
|
|
|
Color: "#ffffff",
|
|
|
|
Description: "status community description",
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create an community chat
|
|
|
|
response, err := s.bob.CreateCommunity(description, true)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
|
|
|
|
community := response.Communities()[0]
|
|
|
|
|
|
|
|
chat := CreateOneToOneChat(common.PubkeyToHex(&s.alice.identity.PublicKey), &s.alice.identity.PublicKey, s.alice.transport)
|
|
|
|
|
|
|
|
s.Require().NoError(s.bob.SaveChat(chat))
|
|
|
|
|
|
|
|
message := buildTestMessage(*chat)
|
|
|
|
message.CommunityID = community.IDString()
|
|
|
|
|
|
|
|
// We send a community link to alice
|
|
|
|
response, err = s.bob.SendChatMessage(ctx, message)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
|
|
|
|
// Retrieve community link & community
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = s.alice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(response.Communities()) == 0 {
|
|
|
|
return errors.New("message not received")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
request := &requests.RequestToJoinCommunity{CommunityID: community.ID()}
|
|
|
|
// We try to join the org
|
|
|
|
response, err = s.alice.RequestToJoinCommunity(request)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.RequestsToJoinCommunity, 1)
|
|
|
|
|
|
|
|
requestToJoin1 := response.RequestsToJoinCommunity[0]
|
|
|
|
s.Require().NotNil(requestToJoin1)
|
|
|
|
s.Require().Equal(community.ID(), requestToJoin1.CommunityID)
|
|
|
|
s.Require().True(requestToJoin1.Our)
|
|
|
|
s.Require().NotEmpty(requestToJoin1.ID)
|
|
|
|
s.Require().NotEmpty(requestToJoin1.Clock)
|
|
|
|
s.Require().Equal(requestToJoin1.PublicKey, common.PubkeyToHex(&s.alice.identity.PublicKey))
|
|
|
|
s.Require().Equal(communities.RequestToJoinStatePending, requestToJoin1.State)
|
|
|
|
|
|
|
|
// Make sure clock is not empty
|
|
|
|
s.Require().NotEmpty(requestToJoin1.Clock)
|
|
|
|
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
s.Require().Equal(response.Communities()[0].RequestedToJoinAt(), requestToJoin1.Clock)
|
|
|
|
|
|
|
|
// pull all communities to make sure we set RequestedToJoinAt
|
|
|
|
|
|
|
|
allCommunities, err := s.alice.Communities()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(allCommunities, 2)
|
|
|
|
|
|
|
|
if bytes.Equal(allCommunities[0].ID(), community.ID()) {
|
|
|
|
s.Require().Equal(allCommunities[0].RequestedToJoinAt(), requestToJoin1.Clock)
|
|
|
|
} else {
|
|
|
|
s.Require().Equal(allCommunities[1].RequestedToJoinAt(), requestToJoin1.Clock)
|
|
|
|
}
|
|
|
|
|
|
|
|
// pull to make sure it has been saved
|
|
|
|
requestsToJoin, err := s.alice.MyPendingRequestsToJoin()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(requestsToJoin, 1)
|
|
|
|
|
|
|
|
// Make sure the requests are fetched also by community
|
|
|
|
requestsToJoin, err = s.alice.PendingRequestsToJoinForCommunity(community.ID())
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(requestsToJoin, 1)
|
|
|
|
|
|
|
|
// Retrieve request to join
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = s.bob.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(response.RequestsToJoinCommunity) == 0 {
|
|
|
|
return errors.New("request to join community not received")
|
|
|
|
}
|
2023-04-06 22:06:24 +08:00
|
|
|
if len(response.ActivityCenterNotifications()) == 0 {
|
|
|
|
return errors.New("request to join community notification not added in activity center")
|
|
|
|
}
|
2022-10-28 11:41:20 +03:00
|
|
|
return nil
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(response.RequestsToJoinCommunity, 1)
|
|
|
|
|
|
|
|
requestToJoin2 := response.RequestsToJoinCommunity[0]
|
|
|
|
|
|
|
|
s.Require().NotNil(requestToJoin2)
|
|
|
|
s.Require().Equal(community.ID(), requestToJoin2.CommunityID)
|
|
|
|
s.Require().False(requestToJoin2.Our)
|
|
|
|
s.Require().NotEmpty(requestToJoin2.ID)
|
|
|
|
s.Require().NotEmpty(requestToJoin2.Clock)
|
|
|
|
s.Require().Equal(requestToJoin2.PublicKey, common.PubkeyToHex(&s.alice.identity.PublicKey))
|
|
|
|
s.Require().Equal(communities.RequestToJoinStatePending, requestToJoin2.State)
|
|
|
|
|
|
|
|
s.Require().Equal(requestToJoin1.ID, requestToJoin2.ID)
|
|
|
|
|
|
|
|
// Cancel request to join community
|
|
|
|
requestsToJoin, err = s.alice.MyPendingRequestsToJoin()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(requestsToJoin, 1)
|
|
|
|
|
|
|
|
requestToJoin := requestsToJoin[0]
|
|
|
|
|
|
|
|
requestToCancel := &requests.CancelRequestToJoinCommunity{ID: requestToJoin.ID}
|
|
|
|
response, err = s.alice.CancelRequestToJoinCommunity(requestToCancel)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
2023-02-07 18:22:49 +00:00
|
|
|
s.Require().Len(response.RequestsToJoinCommunity, 1)
|
|
|
|
s.Require().Equal(communities.RequestToJoinStateCanceled, response.RequestsToJoinCommunity[0].State)
|
2022-10-28 11:41:20 +03:00
|
|
|
|
|
|
|
// pull to make sure it has been saved
|
|
|
|
cancelRequestsToJoin, err := s.alice.MyCanceledRequestsToJoin()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(cancelRequestsToJoin, 1)
|
|
|
|
s.Require().Equal(cancelRequestsToJoin[0].State, communities.RequestToJoinStateCanceled)
|
|
|
|
|
|
|
|
// Retrieve cancel request to join
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = s.bob.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(response.RequestsToJoinCommunity) == 0 {
|
|
|
|
return errors.New("request to join community not received")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(response.RequestsToJoinCommunity, 1)
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(response.RequestsToJoinCommunity, 1)
|
|
|
|
|
2023-04-06 22:06:24 +08:00
|
|
|
// Retrieve activity center notifications for admin to make sure the request notification is deleted
|
|
|
|
notifications, err := s.bob.ActivityCenterNotifications(ActivityCenterNotificationsRequest{
|
|
|
|
Cursor: "",
|
|
|
|
Limit: 10,
|
|
|
|
ActivityTypes: []ActivityCenterType{},
|
|
|
|
ReadType: ActivityCenterQueryParamsReadUnread,
|
|
|
|
})
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(notifications.Notifications, 0)
|
|
|
|
|
2022-10-28 11:41:20 +03:00
|
|
|
cancelRequestToJoin2 := response.RequestsToJoinCommunity[0]
|
|
|
|
|
|
|
|
s.Require().NotNil(cancelRequestToJoin2)
|
|
|
|
s.Require().Equal(community.ID(), cancelRequestToJoin2.CommunityID)
|
|
|
|
s.Require().False(cancelRequestToJoin2.Our)
|
|
|
|
s.Require().NotEmpty(cancelRequestToJoin2.ID)
|
|
|
|
s.Require().NotEmpty(cancelRequestToJoin2.Clock)
|
|
|
|
s.Require().Equal(cancelRequestToJoin2.PublicKey, common.PubkeyToHex(&s.alice.identity.PublicKey))
|
|
|
|
s.Require().Equal(communities.RequestToJoinStateCanceled, cancelRequestToJoin2.State)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
func (s *MessengerCommunitiesSuite) TestRequestAccessAgain() {
|
|
|
|
description := &requests.CreateCommunity{
|
|
|
|
Membership: protobuf.CommunityPermissions_ON_REQUEST,
|
|
|
|
Name: "status",
|
|
|
|
Color: "#ffffff",
|
|
|
|
Description: "status community description",
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create an community chat
|
2022-07-19 11:31:52 +02:00
|
|
|
response, err := s.bob.CreateCommunity(description, true)
|
2021-01-11 11:32:51 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
|
|
|
|
community := response.Communities()[0]
|
|
|
|
|
|
|
|
chat := CreateOneToOneChat(common.PubkeyToHex(&s.alice.identity.PublicKey), &s.alice.identity.PublicKey, s.alice.transport)
|
|
|
|
|
|
|
|
s.Require().NoError(s.bob.SaveChat(chat))
|
|
|
|
|
|
|
|
message := buildTestMessage(*chat)
|
|
|
|
message.CommunityID = community.IDString()
|
|
|
|
|
|
|
|
// We send a community link to alice
|
|
|
|
response, err = s.bob.SendChatMessage(context.Background(), message)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
|
|
|
|
// Retrieve community link & community
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = s.alice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(response.Communities()) == 0 {
|
|
|
|
return errors.New("message not received")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
request := &requests.RequestToJoinCommunity{CommunityID: community.ID()}
|
|
|
|
// We try to join the org
|
|
|
|
response, err = s.alice.RequestToJoinCommunity(request)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.RequestsToJoinCommunity, 1)
|
|
|
|
|
2022-10-26 01:06:20 +03:00
|
|
|
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)
|
2023-01-28 13:52:53 +04:00
|
|
|
s.Require().Equal(notification.Read, true)
|
|
|
|
s.Require().Equal(notification.Accepted, false)
|
|
|
|
s.Require().Equal(notification.Dismissed, false)
|
2022-10-26 01:06:20 +03:00
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
requestToJoin1 := response.RequestsToJoinCommunity[0]
|
|
|
|
s.Require().NotNil(requestToJoin1)
|
|
|
|
s.Require().Equal(community.ID(), requestToJoin1.CommunityID)
|
|
|
|
s.Require().True(requestToJoin1.Our)
|
|
|
|
s.Require().NotEmpty(requestToJoin1.ID)
|
|
|
|
s.Require().NotEmpty(requestToJoin1.Clock)
|
|
|
|
s.Require().Equal(requestToJoin1.PublicKey, common.PubkeyToHex(&s.alice.identity.PublicKey))
|
|
|
|
s.Require().Equal(communities.RequestToJoinStatePending, requestToJoin1.State)
|
|
|
|
|
|
|
|
// Make sure clock is not empty
|
|
|
|
s.Require().NotEmpty(requestToJoin1.Clock)
|
|
|
|
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
s.Require().Equal(response.Communities()[0].RequestedToJoinAt(), requestToJoin1.Clock)
|
|
|
|
|
|
|
|
// pull all communities to make sure we set RequestedToJoinAt
|
|
|
|
|
|
|
|
allCommunities, err := s.alice.Communities()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(allCommunities, 2)
|
|
|
|
|
|
|
|
if bytes.Equal(allCommunities[0].ID(), community.ID()) {
|
|
|
|
s.Require().Equal(allCommunities[0].RequestedToJoinAt(), requestToJoin1.Clock)
|
|
|
|
} else {
|
|
|
|
s.Require().Equal(allCommunities[1].RequestedToJoinAt(), requestToJoin1.Clock)
|
|
|
|
}
|
|
|
|
|
|
|
|
// pull to make sure it has been saved
|
|
|
|
requestsToJoin, err := s.alice.MyPendingRequestsToJoin()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(requestsToJoin, 1)
|
|
|
|
|
|
|
|
// Make sure the requests are fetched also by community
|
|
|
|
requestsToJoin, err = s.alice.PendingRequestsToJoinForCommunity(community.ID())
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(requestsToJoin, 1)
|
|
|
|
|
|
|
|
// Retrieve request to join
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = s.bob.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(response.RequestsToJoinCommunity) == 0 {
|
|
|
|
return errors.New("request to join community not received")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(response.RequestsToJoinCommunity, 1)
|
|
|
|
|
|
|
|
requestToJoin2 := response.RequestsToJoinCommunity[0]
|
|
|
|
|
|
|
|
s.Require().NotNil(requestToJoin2)
|
|
|
|
s.Require().Equal(community.ID(), requestToJoin2.CommunityID)
|
|
|
|
s.Require().False(requestToJoin2.Our)
|
|
|
|
s.Require().NotEmpty(requestToJoin2.ID)
|
|
|
|
s.Require().NotEmpty(requestToJoin2.Clock)
|
|
|
|
s.Require().Equal(requestToJoin2.PublicKey, common.PubkeyToHex(&s.alice.identity.PublicKey))
|
|
|
|
s.Require().Equal(communities.RequestToJoinStatePending, requestToJoin2.State)
|
|
|
|
|
|
|
|
s.Require().Equal(requestToJoin1.ID, requestToJoin2.ID)
|
|
|
|
|
2021-03-31 18:23:45 +02:00
|
|
|
// Check that a notification is been added to messenger
|
|
|
|
|
|
|
|
notifications := response.Notifications()
|
|
|
|
s.Require().Len(notifications, 1)
|
|
|
|
s.Require().NotEqual(notifications[0].ID.Hex(), "0x0000000000000000000000000000000000000000000000000000000000000000")
|
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
// Accept request
|
|
|
|
|
|
|
|
acceptRequestToJoin := &requests.AcceptRequestToJoinCommunity{ID: requestToJoin1.ID}
|
|
|
|
|
|
|
|
response, err = s.bob.AcceptRequestToJoinCommunity(acceptRequestToJoin)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
|
2022-10-26 01:06:20 +03:00
|
|
|
s.Require().Len(response.ActivityCenterNotifications(), 1)
|
|
|
|
|
|
|
|
notification = response.ActivityCenterNotifications()[0]
|
|
|
|
s.Require().NotNil(notification)
|
|
|
|
s.Require().Equal(notification.Type, ActivityCenterNotificationTypeCommunityMembershipRequest)
|
|
|
|
s.Require().Equal(notification.MembershipStatus, ActivityCenterMembershipStatusAccepted)
|
2023-01-28 13:52:53 +04:00
|
|
|
s.Require().Equal(notification.Read, true)
|
|
|
|
s.Require().Equal(notification.Accepted, true)
|
|
|
|
s.Require().Equal(notification.Dismissed, false)
|
2022-10-26 01:06:20 +03:00
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
|
|
|
|
updatedCommunity := response.Communities()[0]
|
|
|
|
|
|
|
|
s.Require().NotNil(updatedCommunity)
|
|
|
|
s.Require().True(updatedCommunity.HasMember(&s.alice.identity.PublicKey))
|
|
|
|
|
|
|
|
// Pull message and make sure org is received
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = s.alice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(response.Communities()) == 0 {
|
|
|
|
return errors.New("community not received")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
|
|
|
|
aliceCommunity := response.Communities()[0]
|
|
|
|
|
|
|
|
s.Require().Equal(community.ID(), aliceCommunity.ID())
|
|
|
|
s.Require().True(aliceCommunity.HasMember(&s.alice.identity.PublicKey))
|
|
|
|
|
|
|
|
// Community should be joined at this point
|
|
|
|
s.Require().True(aliceCommunity.Joined())
|
|
|
|
|
|
|
|
// Make sure the requests are not pending on either sides
|
|
|
|
requestsToJoin, err = s.bob.PendingRequestsToJoinForCommunity(community.ID())
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(requestsToJoin, 0)
|
|
|
|
|
|
|
|
requestsToJoin, err = s.alice.MyPendingRequestsToJoin()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(requestsToJoin, 0)
|
|
|
|
|
|
|
|
// We request again
|
|
|
|
request2 := &requests.RequestToJoinCommunity{CommunityID: community.ID()}
|
|
|
|
// We try to join the org, it should error as we are already a member
|
|
|
|
response, err = s.alice.RequestToJoinCommunity(request2)
|
|
|
|
s.Require().Error(err)
|
|
|
|
|
|
|
|
// We kick the member
|
|
|
|
response, err = s.bob.RemoveUserFromCommunity(
|
|
|
|
community.ID(),
|
|
|
|
common.PubkeyToHex(&s.alice.identity.PublicKey),
|
|
|
|
)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
|
|
|
|
community = response.Communities()[0]
|
|
|
|
s.Require().False(community.HasMember(&s.alice.identity.PublicKey))
|
|
|
|
|
|
|
|
// Alice should then be removed
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = s.alice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(response.Communities()) == 0 {
|
|
|
|
return errors.New("community not received")
|
|
|
|
}
|
2023-04-26 22:00:54 +08:00
|
|
|
if len(response.ActivityCenterNotifications()) == 0 {
|
|
|
|
return errors.New("activity center notification not received")
|
|
|
|
}
|
|
|
|
if response.ActivityCenterState().HasSeen {
|
|
|
|
return errors.New("activity center seen state is incorrect")
|
|
|
|
}
|
2021-01-11 11:32:51 +01:00
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
2023-04-14 13:02:28 +04:00
|
|
|
// Check we got AC notification for Alice
|
|
|
|
aliceNotifications, err := s.alice.ActivityCenterNotifications(ActivityCenterNotificationsRequest{
|
|
|
|
Cursor: "",
|
|
|
|
Limit: 10,
|
|
|
|
ActivityTypes: []ActivityCenterType{ActivityCenterNotificationTypeCommunityKicked},
|
|
|
|
ReadType: ActivityCenterQueryParamsReadUnread,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(aliceNotifications.Notifications, 1)
|
|
|
|
s.Require().Equal(community.IDString(), aliceNotifications.Notifications[0].CommunityID)
|
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
|
|
|
|
aliceCommunity = response.Communities()[0]
|
|
|
|
|
|
|
|
s.Require().Equal(community.ID(), aliceCommunity.ID())
|
|
|
|
s.Require().False(aliceCommunity.HasMember(&s.alice.identity.PublicKey))
|
|
|
|
|
|
|
|
// Alice can request access again
|
|
|
|
request3 := &requests.RequestToJoinCommunity{CommunityID: community.ID()}
|
|
|
|
response, err = s.alice.RequestToJoinCommunity(request3)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.RequestsToJoinCommunity, 1)
|
|
|
|
|
|
|
|
requestToJoin3 := response.RequestsToJoinCommunity[0]
|
|
|
|
s.Require().NotNil(requestToJoin3)
|
|
|
|
s.Require().Equal(community.ID(), requestToJoin3.CommunityID)
|
|
|
|
s.Require().True(requestToJoin3.Our)
|
|
|
|
s.Require().NotEmpty(requestToJoin3.ID)
|
|
|
|
s.Require().NotEmpty(requestToJoin3.Clock)
|
|
|
|
s.Require().Equal(requestToJoin3.PublicKey, common.PubkeyToHex(&s.alice.identity.PublicKey))
|
|
|
|
s.Require().Equal(communities.RequestToJoinStatePending, requestToJoin3.State)
|
|
|
|
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
s.Require().Equal(response.Communities()[0].RequestedToJoinAt(), requestToJoin3.Clock)
|
|
|
|
|
2022-10-26 01:06:20 +03:00
|
|
|
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)
|
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
// Retrieve request to join
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = s.bob.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(response.RequestsToJoinCommunity) == 0 {
|
|
|
|
return errors.New("request to join community not received")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(response.RequestsToJoinCommunity, 1)
|
|
|
|
|
|
|
|
requestToJoin4 := response.RequestsToJoinCommunity[0]
|
|
|
|
|
|
|
|
s.Require().NotNil(requestToJoin4)
|
|
|
|
s.Require().Equal(community.ID(), requestToJoin4.CommunityID)
|
|
|
|
s.Require().False(requestToJoin4.Our)
|
|
|
|
s.Require().NotEmpty(requestToJoin4.ID)
|
|
|
|
s.Require().NotEmpty(requestToJoin4.Clock)
|
|
|
|
s.Require().Equal(requestToJoin4.PublicKey, common.PubkeyToHex(&s.alice.identity.PublicKey))
|
|
|
|
s.Require().Equal(communities.RequestToJoinStatePending, requestToJoin4.State)
|
|
|
|
|
|
|
|
s.Require().Equal(requestToJoin3.ID, requestToJoin4.ID)
|
|
|
|
}
|
|
|
|
|
2022-08-04 09:44:35 +02:00
|
|
|
func (s *MessengerCommunitiesSuite) TestDeclineAccess() {
|
|
|
|
ctx := context.Background()
|
|
|
|
|
|
|
|
description := &requests.CreateCommunity{
|
|
|
|
Membership: protobuf.CommunityPermissions_ON_REQUEST,
|
|
|
|
Name: "status",
|
|
|
|
Color: "#ffffff",
|
|
|
|
Description: "status community description",
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create an community chat
|
|
|
|
response, err := s.bob.CreateCommunity(description, true)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
|
|
|
|
community := response.Communities()[0]
|
|
|
|
|
|
|
|
chat := CreateOneToOneChat(common.PubkeyToHex(&s.alice.identity.PublicKey), &s.alice.identity.PublicKey, s.alice.transport)
|
|
|
|
|
|
|
|
s.Require().NoError(s.bob.SaveChat(chat))
|
|
|
|
|
|
|
|
message := buildTestMessage(*chat)
|
|
|
|
message.CommunityID = community.IDString()
|
|
|
|
|
|
|
|
// We send a community link to alice
|
|
|
|
response, err = s.bob.SendChatMessage(ctx, message)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
|
|
|
|
// Retrieve community link & community
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = s.alice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(response.Communities()) == 0 {
|
|
|
|
return errors.New("message not received")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
request := &requests.RequestToJoinCommunity{CommunityID: community.ID()}
|
|
|
|
// We try to join the org
|
|
|
|
response, err = s.alice.RequestToJoinCommunity(request)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.RequestsToJoinCommunity, 1)
|
|
|
|
|
|
|
|
requestToJoin1 := response.RequestsToJoinCommunity[0]
|
|
|
|
s.Require().NotNil(requestToJoin1)
|
|
|
|
s.Require().Equal(community.ID(), requestToJoin1.CommunityID)
|
|
|
|
s.Require().True(requestToJoin1.Our)
|
|
|
|
s.Require().NotEmpty(requestToJoin1.ID)
|
|
|
|
s.Require().NotEmpty(requestToJoin1.Clock)
|
|
|
|
s.Require().Equal(requestToJoin1.PublicKey, common.PubkeyToHex(&s.alice.identity.PublicKey))
|
|
|
|
s.Require().Equal(communities.RequestToJoinStatePending, requestToJoin1.State)
|
|
|
|
|
2022-10-26 01:06:20 +03:00
|
|
|
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)
|
2023-01-28 13:52:53 +04:00
|
|
|
s.Require().Equal(notification.Read, true)
|
|
|
|
s.Require().Equal(notification.Dismissed, false)
|
|
|
|
s.Require().Equal(notification.Accepted, false)
|
2022-10-26 01:06:20 +03:00
|
|
|
|
2022-08-04 09:44:35 +02:00
|
|
|
// Make sure clock is not empty
|
|
|
|
s.Require().NotEmpty(requestToJoin1.Clock)
|
|
|
|
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
s.Require().Equal(response.Communities()[0].RequestedToJoinAt(), requestToJoin1.Clock)
|
|
|
|
|
|
|
|
// pull all communities to make sure we set RequestedToJoinAt
|
|
|
|
|
|
|
|
allCommunities, err := s.alice.Communities()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(allCommunities, 2)
|
|
|
|
|
|
|
|
if bytes.Equal(allCommunities[0].ID(), community.ID()) {
|
|
|
|
s.Require().Equal(allCommunities[0].RequestedToJoinAt(), requestToJoin1.Clock)
|
|
|
|
} else {
|
|
|
|
s.Require().Equal(allCommunities[1].RequestedToJoinAt(), requestToJoin1.Clock)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Retrieve request to join
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = s.bob.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(response.RequestsToJoinCommunity) == 0 {
|
|
|
|
return errors.New("request to join community not received")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(response.RequestsToJoinCommunity, 1)
|
|
|
|
|
|
|
|
// Check if admin sees requests correctly
|
|
|
|
requestsToJoin, err := s.bob.PendingRequestsToJoinForCommunity(community.ID())
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(requestsToJoin, 1)
|
|
|
|
|
|
|
|
requestsToJoin, err = s.bob.DeclinedRequestsToJoinForCommunity(community.ID())
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(requestsToJoin, 0)
|
|
|
|
|
|
|
|
requestToJoin2 := response.RequestsToJoinCommunity[0]
|
|
|
|
|
|
|
|
s.Require().NotNil(requestToJoin2)
|
|
|
|
s.Require().Equal(community.ID(), requestToJoin2.CommunityID)
|
|
|
|
s.Require().False(requestToJoin2.Our)
|
|
|
|
s.Require().NotEmpty(requestToJoin2.ID)
|
|
|
|
s.Require().NotEmpty(requestToJoin2.Clock)
|
|
|
|
s.Require().Equal(requestToJoin2.PublicKey, common.PubkeyToHex(&s.alice.identity.PublicKey))
|
|
|
|
s.Require().Equal(communities.RequestToJoinStatePending, requestToJoin2.State)
|
|
|
|
|
|
|
|
s.Require().Equal(requestToJoin1.ID, requestToJoin2.ID)
|
|
|
|
|
|
|
|
// Decline request
|
|
|
|
declinedRequestToJoin := &requests.DeclineRequestToJoinCommunity{ID: requestToJoin1.ID}
|
2022-10-26 01:06:20 +03:00
|
|
|
response, err = s.bob.DeclineRequestToJoinCommunity(declinedRequestToJoin)
|
2022-08-04 09:44:35 +02:00
|
|
|
s.Require().NoError(err)
|
2022-10-26 01:06:20 +03:00
|
|
|
s.Require().NotNil(response)
|
|
|
|
|
|
|
|
s.Require().Len(response.ActivityCenterNotifications(), 1)
|
|
|
|
|
|
|
|
notification = response.ActivityCenterNotifications()[0]
|
|
|
|
s.Require().NotNil(notification)
|
|
|
|
s.Require().Equal(notification.Type, ActivityCenterNotificationTypeCommunityMembershipRequest)
|
|
|
|
s.Require().Equal(notification.MembershipStatus, ActivityCenterMembershipStatusDeclined)
|
2023-01-28 13:52:53 +04:00
|
|
|
s.Require().Equal(notification.Read, true)
|
|
|
|
s.Require().Equal(notification.Accepted, false)
|
|
|
|
s.Require().Equal(notification.Dismissed, true)
|
2022-08-04 09:44:35 +02:00
|
|
|
|
|
|
|
// Check if admin sees requests correctly
|
|
|
|
requestsToJoin, err = s.bob.PendingRequestsToJoinForCommunity(community.ID())
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(requestsToJoin, 0)
|
|
|
|
|
|
|
|
requestsToJoin, err = s.bob.DeclinedRequestsToJoinForCommunity(community.ID())
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(requestsToJoin, 1)
|
|
|
|
|
|
|
|
// Accept declined request
|
|
|
|
acceptRequestToJoin := &requests.AcceptRequestToJoinCommunity{ID: requestToJoin1.ID}
|
|
|
|
response, err = s.bob.AcceptRequestToJoinCommunity(acceptRequestToJoin)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
|
|
|
|
updatedCommunity := response.Communities()[0]
|
|
|
|
|
|
|
|
s.Require().NotNil(updatedCommunity)
|
|
|
|
s.Require().True(updatedCommunity.HasMember(&s.alice.identity.PublicKey))
|
|
|
|
|
2022-10-26 01:06:20 +03:00
|
|
|
s.Require().Len(response.ActivityCenterNotifications(), 1)
|
|
|
|
|
|
|
|
notification = response.ActivityCenterNotifications()[0]
|
|
|
|
s.Require().NotNil(notification)
|
|
|
|
s.Require().Equal(notification.Type, ActivityCenterNotificationTypeCommunityMembershipRequest)
|
|
|
|
s.Require().Equal(notification.MembershipStatus, ActivityCenterMembershipStatusAccepted)
|
|
|
|
|
2022-08-04 09:44:35 +02:00
|
|
|
// Pull message and make sure org is received
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = s.alice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(response.Communities()) == 0 {
|
|
|
|
return errors.New("community not received")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
|
|
|
|
aliceCommunity := response.Communities()[0]
|
|
|
|
|
|
|
|
s.Require().Equal(community.ID(), aliceCommunity.ID())
|
|
|
|
s.Require().True(aliceCommunity.HasMember(&s.alice.identity.PublicKey))
|
|
|
|
|
|
|
|
// Community should be joined at this point
|
|
|
|
s.Require().True(aliceCommunity.Joined())
|
|
|
|
|
|
|
|
// Make sure the requests are not pending on either sides
|
|
|
|
requestsToJoin, err = s.bob.PendingRequestsToJoinForCommunity(community.ID())
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(requestsToJoin, 0)
|
|
|
|
|
|
|
|
requestsToJoin, err = s.bob.DeclinedRequestsToJoinForCommunity(community.ID())
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(requestsToJoin, 0)
|
|
|
|
|
|
|
|
requestsToJoin, err = s.alice.MyPendingRequestsToJoin()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(requestsToJoin, 0)
|
|
|
|
}
|
|
|
|
|
2022-08-23 18:30:19 +02:00
|
|
|
func (s *MessengerCommunitiesSuite) TestLeaveAndRejoinCommunity() {
|
|
|
|
community := s.createCommunity()
|
|
|
|
s.advertiseCommunityTo(community, s.alice)
|
|
|
|
s.advertiseCommunityTo(community, s.bob)
|
|
|
|
|
2023-06-27 14:20:20 +02:00
|
|
|
s.joinCommunity(community, s.alice)
|
|
|
|
s.joinCommunity(community, s.bob)
|
2022-08-23 18:30:19 +02:00
|
|
|
|
|
|
|
joinedCommunities, err := s.admin.communitiesManager.Joined()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Equal(3, joinedCommunities[0].MembersCount())
|
|
|
|
|
|
|
|
response, err := s.alice.LeaveCommunity(community.ID())
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
s.Require().False(response.Communities()[0].Joined())
|
|
|
|
|
|
|
|
// admin should receive alice's request to leave
|
|
|
|
// and then update and advertise community members list accordingly
|
|
|
|
|
|
|
|
verifyCommunityMembers := func(user *Messenger) error {
|
|
|
|
response, err := user.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(response.Communities()) == 0 {
|
|
|
|
return errors.New("no communities in response")
|
|
|
|
}
|
|
|
|
|
|
|
|
var communityMembersError error = nil
|
|
|
|
|
|
|
|
if response.Communities()[0].MembersCount() != 2 {
|
|
|
|
communityMembersError = fmt.Errorf("invalid number of members: %d", response.Communities()[0].MembersCount())
|
|
|
|
} else if !response.Communities()[0].HasMember(&s.admin.identity.PublicKey) {
|
|
|
|
communityMembersError = errors.New("admin removed from community")
|
|
|
|
} else if !response.Communities()[0].HasMember(&s.bob.identity.PublicKey) {
|
|
|
|
communityMembersError = errors.New("bob removed from community")
|
|
|
|
} else if response.Communities()[0].HasMember(&s.alice.identity.PublicKey) {
|
|
|
|
communityMembersError = errors.New("alice not removed from community")
|
|
|
|
}
|
|
|
|
|
2023-04-17 15:44:48 +01:00
|
|
|
return communityMembersError
|
2022-08-23 18:30:19 +02:00
|
|
|
}
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
return verifyCommunityMembers(s.admin)
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
return verifyCommunityMembers(s.bob)
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
joinedCommunities, err = s.admin.communitiesManager.Joined()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Equal(2, joinedCommunities[0].MembersCount())
|
|
|
|
|
2022-12-07 14:34:48 -05:00
|
|
|
chats, err := s.alice.persistence.Chats()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
var numberInactiveChats = 0
|
|
|
|
for i := 0; i < len(chats); i++ {
|
|
|
|
if !chats[i].Active {
|
|
|
|
numberInactiveChats++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
s.Require().Equal(3, numberInactiveChats)
|
|
|
|
|
2022-08-23 18:30:19 +02:00
|
|
|
// alice can rejoin
|
2023-06-27 14:20:20 +02:00
|
|
|
s.joinCommunity(community, s.alice)
|
2022-08-23 18:30:19 +02:00
|
|
|
|
|
|
|
joinedCommunities, err = s.admin.communitiesManager.Joined()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Equal(3, joinedCommunities[0].MembersCount())
|
2022-12-07 14:34:48 -05:00
|
|
|
|
|
|
|
chats, err = s.alice.persistence.Chats()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
numberInactiveChats = 0
|
|
|
|
for i := 0; i < len(chats); i++ {
|
|
|
|
if !chats[i].Active {
|
|
|
|
numberInactiveChats++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
s.Require().Equal(1, numberInactiveChats)
|
2022-08-23 18:30:19 +02:00
|
|
|
}
|
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
func (s *MessengerCommunitiesSuite) TestShareCommunity() {
|
|
|
|
description := &requests.CreateCommunity{
|
|
|
|
Membership: protobuf.CommunityPermissions_NO_MEMBERSHIP,
|
|
|
|
Name: "status",
|
|
|
|
Color: "#ffffff",
|
|
|
|
Description: "status community description",
|
|
|
|
}
|
|
|
|
|
2022-08-08 13:49:39 +03:00
|
|
|
inviteMessage := "invite to community testing message"
|
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
// Create an community chat
|
2022-07-19 11:31:52 +02:00
|
|
|
response, err := s.bob.CreateCommunity(description, true)
|
2021-01-11 11:32:51 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
|
|
|
|
community := response.Communities()[0]
|
|
|
|
|
|
|
|
response, err = s.bob.ShareCommunity(
|
|
|
|
&requests.ShareCommunity{
|
2022-08-08 13:49:39 +03:00
|
|
|
CommunityID: community.ID(),
|
|
|
|
Users: []types.HexBytes{common.PubkeyToHexBytes(&s.alice.identity.PublicKey)},
|
|
|
|
InviteMessage: inviteMessage,
|
2021-01-11 11:32:51 +01:00
|
|
|
},
|
|
|
|
)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
2021-06-03 15:11:55 +02:00
|
|
|
s.Require().Len(response.Messages(), 1)
|
2021-01-11 11:32:51 +01:00
|
|
|
|
2021-04-07 14:57:14 +02:00
|
|
|
// Add bob to contacts so it does not go on activity center
|
2023-04-25 15:27:15 +04:00
|
|
|
bobPk := common.PubkeyToHex(&s.bob.identity.PublicKey)
|
2022-10-14 09:50:36 +01:00
|
|
|
request := &requests.AddContact{ID: bobPk}
|
2021-10-26 11:48:02 +01:00
|
|
|
_, err = s.alice.AddContact(context.Background(), request)
|
2021-04-07 14:57:14 +02:00
|
|
|
s.Require().NoError(err)
|
|
|
|
|
2021-01-11 11:32:51 +01:00
|
|
|
// Pull message and make sure org is received
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = s.alice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-06-03 15:11:55 +02:00
|
|
|
if len(response.messages) == 0 {
|
2021-01-11 11:32:51 +01:00
|
|
|
return errors.New("community link not received")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
2021-06-03 15:11:55 +02:00
|
|
|
s.Require().Len(response.Messages(), 1)
|
2022-08-08 13:49:39 +03:00
|
|
|
|
|
|
|
message := response.Messages()[0]
|
|
|
|
s.Require().Equal(community.IDString(), message.CommunityID)
|
|
|
|
s.Require().Equal(inviteMessage, message.Text)
|
2021-01-11 11:32:51 +01:00
|
|
|
}
|
2021-03-19 10:15:45 +01:00
|
|
|
|
|
|
|
func (s *MessengerCommunitiesSuite) TestBanUser() {
|
|
|
|
description := &requests.CreateCommunity{
|
|
|
|
Membership: protobuf.CommunityPermissions_NO_MEMBERSHIP,
|
|
|
|
Name: "status",
|
|
|
|
Color: "#ffffff",
|
|
|
|
Description: "status community description",
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create an community chat
|
2022-07-19 11:31:52 +02:00
|
|
|
response, err := s.bob.CreateCommunity(description, true)
|
2021-03-19 10:15:45 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
|
|
|
|
community := response.Communities()[0]
|
|
|
|
|
|
|
|
response, err = s.bob.InviteUsersToCommunity(
|
|
|
|
&requests.InviteUsersToCommunity{
|
|
|
|
CommunityID: community.ID(),
|
|
|
|
Users: []types.HexBytes{common.PubkeyToHexBytes(&s.alice.identity.PublicKey)},
|
|
|
|
},
|
|
|
|
)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
|
|
|
|
community = response.Communities()[0]
|
|
|
|
s.Require().True(community.HasMember(&s.alice.identity.PublicKey))
|
|
|
|
|
|
|
|
response, err = s.bob.BanUserFromCommunity(
|
|
|
|
&requests.BanUserFromCommunity{
|
|
|
|
CommunityID: community.ID(),
|
|
|
|
User: common.PubkeyToHexBytes(&s.alice.identity.PublicKey),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
|
|
|
|
community = response.Communities()[0]
|
|
|
|
s.Require().False(community.HasMember(&s.alice.identity.PublicKey))
|
|
|
|
s.Require().True(community.IsBanned(&s.alice.identity.PublicKey))
|
|
|
|
|
2022-06-23 09:12:15 +02:00
|
|
|
response, err = s.bob.UnbanUserFromCommunity(
|
|
|
|
&requests.UnbanUserFromCommunity{
|
|
|
|
CommunityID: community.ID(),
|
|
|
|
User: common.PubkeyToHexBytes(&s.alice.identity.PublicKey),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
|
|
|
|
community = response.Communities()[0]
|
|
|
|
s.Require().False(community.IsBanned(&s.alice.identity.PublicKey))
|
2021-03-19 10:15:45 +01:00
|
|
|
}
|
2021-08-06 16:40:23 +01:00
|
|
|
|
2022-06-01 09:55:48 +02:00
|
|
|
func (s *MessengerCommunitiesSuite) TestSyncCommunitySettings() {
|
|
|
|
// Create new device
|
2023-06-27 14:20:20 +02:00
|
|
|
alicesOtherDevice := s.newMessengerWithKey(s.alice.identity)
|
2022-06-01 09:55:48 +02:00
|
|
|
|
|
|
|
// Pair devices
|
2023-06-27 14:20:20 +02:00
|
|
|
err := alicesOtherDevice.SetInstallationMetadata(alicesOtherDevice.installationID, &multidevice.InstallationMetadata{
|
2022-06-01 09:55:48 +02:00
|
|
|
Name: "their-name",
|
|
|
|
DeviceType: "their-device-type",
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
s.pairTwoDevices(alicesOtherDevice, s.alice, "their-name", "their-device-type")
|
|
|
|
|
|
|
|
// Create a community
|
|
|
|
createCommunityReq := &requests.CreateCommunity{
|
|
|
|
Membership: protobuf.CommunityPermissions_ON_REQUEST,
|
|
|
|
Name: "new community",
|
|
|
|
Color: "#000000",
|
|
|
|
Description: "new community description",
|
|
|
|
}
|
|
|
|
|
2022-07-19 11:31:52 +02:00
|
|
|
mr, err := s.alice.CreateCommunity(createCommunityReq, true)
|
2022-06-01 09:55:48 +02:00
|
|
|
s.Require().NoError(err, "s.alice.CreateCommunity")
|
|
|
|
var newCommunity *communities.Community
|
|
|
|
for _, com := range mr.Communities() {
|
|
|
|
if com.Name() == createCommunityReq.Name {
|
|
|
|
newCommunity = com
|
|
|
|
}
|
|
|
|
}
|
|
|
|
s.Require().NotNil(newCommunity)
|
|
|
|
|
|
|
|
// Check that Alice has community settings
|
|
|
|
cs, err := s.alice.communitiesManager.GetCommunitySettingsByID(newCommunity.ID())
|
|
|
|
s.Require().NoError(err, "communitiesManager.GetCommunitySettingsByID")
|
|
|
|
s.NotNil(cs, "Must have community settings")
|
|
|
|
|
|
|
|
// Wait for the message to reach its destination
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
_, err = alicesOtherDevice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Do we have new synced community settings?
|
|
|
|
syncedSettings, err := alicesOtherDevice.communitiesManager.GetCommunitySettingsByID(newCommunity.ID())
|
|
|
|
if err != nil || syncedSettings == nil {
|
|
|
|
return fmt.Errorf("community with sync not received %w", err)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
tcs, err := alicesOtherDevice.communitiesManager.GetCommunitySettingsByID(newCommunity.ID())
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
// Check the community settings on their device matched the community settings on Alice's device
|
|
|
|
s.Equal(cs.CommunityID, tcs.CommunityID)
|
|
|
|
s.Equal(cs.HistoryArchiveSupportEnabled, tcs.HistoryArchiveSupportEnabled)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *MessengerCommunitiesSuite) TestSyncCommunitySettings_EditCommunity() {
|
|
|
|
// Create new device
|
2023-06-27 14:20:20 +02:00
|
|
|
alicesOtherDevice := s.newMessengerWithKey(s.alice.identity)
|
2022-06-01 09:55:48 +02:00
|
|
|
|
|
|
|
// Pair devices
|
2023-06-27 14:20:20 +02:00
|
|
|
err := alicesOtherDevice.SetInstallationMetadata(alicesOtherDevice.installationID, &multidevice.InstallationMetadata{
|
2022-06-01 09:55:48 +02:00
|
|
|
Name: "their-name",
|
|
|
|
DeviceType: "their-device-type",
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
s.pairTwoDevices(alicesOtherDevice, s.alice, "their-name", "their-device-type")
|
|
|
|
|
|
|
|
// Create a community
|
|
|
|
createCommunityReq := &requests.CreateCommunity{
|
|
|
|
Membership: protobuf.CommunityPermissions_ON_REQUEST,
|
|
|
|
Name: "new community",
|
|
|
|
Color: "#000000",
|
|
|
|
Description: "new community description",
|
|
|
|
}
|
|
|
|
|
2022-07-19 11:31:52 +02:00
|
|
|
mr, err := s.alice.CreateCommunity(createCommunityReq, true)
|
2022-06-01 09:55:48 +02:00
|
|
|
s.Require().NoError(err, "s.alice.CreateCommunity")
|
|
|
|
var newCommunity *communities.Community
|
|
|
|
for _, com := range mr.Communities() {
|
|
|
|
if com.Name() == createCommunityReq.Name {
|
|
|
|
newCommunity = com
|
|
|
|
}
|
|
|
|
}
|
|
|
|
s.Require().NotNil(newCommunity)
|
|
|
|
|
|
|
|
// Check that Alice has community settings
|
|
|
|
cs, err := s.alice.communitiesManager.GetCommunitySettingsByID(newCommunity.ID())
|
|
|
|
s.Require().NoError(err, "communitiesManager.GetCommunitySettingsByID")
|
|
|
|
s.NotNil(cs, "Must have community settings")
|
|
|
|
|
|
|
|
// Wait for the message to reach its destination
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
_, err = alicesOtherDevice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Do we have new synced community settings?
|
|
|
|
syncedSettings, err := alicesOtherDevice.communitiesManager.GetCommunitySettingsByID(newCommunity.ID())
|
|
|
|
if err != nil || syncedSettings == nil {
|
|
|
|
return fmt.Errorf("community settings with sync not received %w", err)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
tcs, err := alicesOtherDevice.communitiesManager.GetCommunitySettingsByID(newCommunity.ID())
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
// Check the community settings on their device matched the community settings on Alice's device
|
|
|
|
s.Equal(cs.CommunityID, tcs.CommunityID)
|
|
|
|
s.Equal(cs.HistoryArchiveSupportEnabled, tcs.HistoryArchiveSupportEnabled)
|
|
|
|
|
|
|
|
req := createCommunityReq
|
|
|
|
req.HistoryArchiveSupportEnabled = true
|
|
|
|
editCommunityReq := &requests.EditCommunity{
|
|
|
|
CommunityID: newCommunity.ID(),
|
|
|
|
CreateCommunity: *req,
|
|
|
|
}
|
|
|
|
|
|
|
|
mr, err = s.alice.EditCommunity(editCommunityReq)
|
|
|
|
s.Require().NoError(err, "s.alice.EditCommunity")
|
|
|
|
var editedCommunity *communities.Community
|
|
|
|
for _, com := range mr.Communities() {
|
|
|
|
if com.Name() == createCommunityReq.Name {
|
|
|
|
editedCommunity = com
|
|
|
|
}
|
|
|
|
}
|
|
|
|
s.Require().NotNil(editedCommunity)
|
|
|
|
|
|
|
|
// Wait a bit for sync messages to reach destination
|
|
|
|
time.Sleep(1 * time.Second)
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
_, err = alicesOtherDevice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
tcs, err = alicesOtherDevice.communitiesManager.GetCommunitySettingsByID(newCommunity.ID())
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
// Check the community settings on their device matched the community settings on Alice's device
|
|
|
|
s.Equal(cs.CommunityID, tcs.CommunityID)
|
|
|
|
s.Equal(req.HistoryArchiveSupportEnabled, tcs.HistoryArchiveSupportEnabled)
|
|
|
|
}
|
|
|
|
|
2021-08-06 16:40:23 +01:00
|
|
|
// TestSyncCommunity tests basic sync functionality between 2 Messengers
|
|
|
|
func (s *MessengerCommunitiesSuite) TestSyncCommunity() {
|
|
|
|
// Create new device
|
2023-06-27 14:20:20 +02:00
|
|
|
alicesOtherDevice := s.newMessengerWithKey(s.alice.identity)
|
2021-08-06 16:40:23 +01:00
|
|
|
|
|
|
|
tcs, err := alicesOtherDevice.communitiesManager.All()
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err, "alicesOtherDevice.communitiesManager.All")
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Len(tcs, 1, "Must have 1 communities")
|
|
|
|
|
|
|
|
// Pair devices
|
|
|
|
err = alicesOtherDevice.SetInstallationMetadata(alicesOtherDevice.installationID, &multidevice.InstallationMetadata{
|
|
|
|
Name: "their-name",
|
|
|
|
DeviceType: "their-device-type",
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
s.pairTwoDevices(alicesOtherDevice, s.alice, "their-name", "their-device-type")
|
|
|
|
|
|
|
|
// Create a community
|
|
|
|
createCommunityReq := &requests.CreateCommunity{
|
|
|
|
Membership: protobuf.CommunityPermissions_ON_REQUEST,
|
|
|
|
Name: "new community",
|
|
|
|
Color: "#000000",
|
|
|
|
Description: "new community description",
|
|
|
|
}
|
|
|
|
|
2022-07-19 11:31:52 +02:00
|
|
|
mr, err := s.alice.CreateCommunity(createCommunityReq, true)
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err, "s.alice.CreateCommunity")
|
2021-08-06 16:40:23 +01:00
|
|
|
var newCommunity *communities.Community
|
|
|
|
for _, com := range mr.Communities() {
|
|
|
|
if com.Name() == createCommunityReq.Name {
|
|
|
|
newCommunity = com
|
|
|
|
}
|
|
|
|
}
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NotNil(newCommunity)
|
2021-08-06 16:40:23 +01:00
|
|
|
|
|
|
|
// Check that Alice has 2 communities
|
|
|
|
cs, err := s.alice.communitiesManager.All()
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err, "communitiesManager.All")
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Len(cs, 2, "Must have 2 communities")
|
|
|
|
|
|
|
|
// Wait for the message to reach its destination
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
_, err = alicesOtherDevice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Do we have a new synced community?
|
|
|
|
_, err = alicesOtherDevice.communitiesManager.GetSyncedRawCommunity(newCommunity.ID())
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("community with sync not received %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
})
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err)
|
2021-08-06 16:40:23 +01:00
|
|
|
|
|
|
|
// Count the number of communities in their device
|
|
|
|
tcs, err = alicesOtherDevice.communitiesManager.All()
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err)
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Len(tcs, 2, "There must be 2 communities")
|
2022-11-07 17:30:00 +00:00
|
|
|
|
2021-08-06 16:40:23 +01:00
|
|
|
s.logger.Debug("", zap.Any("tcs", tcs))
|
|
|
|
|
|
|
|
// Get the new community from their db
|
|
|
|
tnc, err := alicesOtherDevice.communitiesManager.GetByID(newCommunity.ID())
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err)
|
2021-08-06 16:40:23 +01:00
|
|
|
|
|
|
|
// Check the community on their device matched the new community on Alice's device
|
|
|
|
s.Equal(newCommunity.ID(), tnc.ID())
|
|
|
|
s.Equal(newCommunity.Name(), tnc.Name())
|
|
|
|
s.Equal(newCommunity.DescriptionText(), tnc.DescriptionText())
|
|
|
|
s.Equal(newCommunity.IDString(), tnc.IDString())
|
2023-07-21 13:41:26 +04:00
|
|
|
|
|
|
|
// Private Key for synced community should be null
|
|
|
|
s.Require().NotNil(newCommunity.PrivateKey())
|
|
|
|
s.Require().Nil(tnc.PrivateKey())
|
|
|
|
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Equal(newCommunity.PublicKey(), tnc.PublicKey())
|
|
|
|
s.Equal(newCommunity.Verified(), tnc.Verified())
|
|
|
|
s.Equal(newCommunity.Muted(), tnc.Muted())
|
|
|
|
s.Equal(newCommunity.Joined(), tnc.Joined())
|
2022-09-20 21:57:39 +02:00
|
|
|
s.Equal(newCommunity.Spectated(), tnc.Spectated())
|
2023-06-14 16:15:46 +02:00
|
|
|
s.Equal(newCommunity.IsOwner(), tnc.IsOwner())
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Equal(newCommunity.InvitationOnly(), tnc.InvitationOnly())
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestSyncCommunity_RequestToJoin tests more complex pairing and syncing scenario where one paired device
|
|
|
|
// makes a request to join a community
|
|
|
|
func (s *MessengerCommunitiesSuite) TestSyncCommunity_RequestToJoin() {
|
|
|
|
// Set Alice's installation metadata
|
|
|
|
aim := &multidevice.InstallationMetadata{
|
|
|
|
Name: "alice's-device",
|
|
|
|
DeviceType: "alice's-device-type",
|
|
|
|
}
|
|
|
|
err := s.alice.SetInstallationMetadata(s.alice.installationID, aim)
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err)
|
2021-08-06 16:40:23 +01:00
|
|
|
|
|
|
|
// Create Alice's other device
|
2023-06-27 14:20:20 +02:00
|
|
|
alicesOtherDevice := s.newMessengerWithKey(s.alice.identity)
|
2021-08-06 16:40:23 +01:00
|
|
|
|
|
|
|
im1 := &multidevice.InstallationMetadata{
|
|
|
|
Name: "alice's-other-device",
|
|
|
|
DeviceType: "alice's-other-device-type",
|
|
|
|
}
|
|
|
|
err = alicesOtherDevice.SetInstallationMetadata(alicesOtherDevice.installationID, im1)
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err)
|
2021-08-06 16:40:23 +01:00
|
|
|
|
|
|
|
// Pair alice's two devices
|
|
|
|
s.pairTwoDevices(alicesOtherDevice, s.alice, im1.Name, im1.DeviceType)
|
|
|
|
s.pairTwoDevices(s.alice, alicesOtherDevice, aim.Name, aim.DeviceType)
|
|
|
|
|
|
|
|
// Check bob the admin has only one community
|
|
|
|
tcs2, err := s.bob.communitiesManager.All()
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err, "admin.communitiesManager.All")
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Len(tcs2, 1, "Must have 1 communities")
|
|
|
|
|
|
|
|
// Bob the admin creates a community
|
|
|
|
createCommunityReq := &requests.CreateCommunity{
|
|
|
|
Membership: protobuf.CommunityPermissions_ON_REQUEST,
|
|
|
|
Name: "new community",
|
|
|
|
Color: "#000000",
|
|
|
|
Description: "new community description",
|
|
|
|
}
|
2022-07-19 11:31:52 +02:00
|
|
|
mr, err := s.bob.CreateCommunity(createCommunityReq, true)
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err, "CreateCommunity")
|
|
|
|
s.Require().NotNil(mr)
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Len(mr.Communities(), 1)
|
|
|
|
|
|
|
|
community := mr.Communities()[0]
|
|
|
|
|
|
|
|
// Check that admin has 2 communities
|
|
|
|
acs, err := s.bob.communitiesManager.All()
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err, "communitiesManager.All")
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Len(acs, 2, "Must have 2 communities")
|
|
|
|
|
|
|
|
// Check that Alice has only 1 community on either device
|
|
|
|
cs, err := s.alice.communitiesManager.All()
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err, "communitiesManager.All")
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Len(cs, 1, "Must have 1 communities")
|
|
|
|
|
|
|
|
tcs1, err := alicesOtherDevice.communitiesManager.All()
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err, "alicesOtherDevice.communitiesManager.All")
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Len(tcs1, 1, "Must have 1 communities")
|
|
|
|
|
|
|
|
// Bob the admin opens up a 1-1 chat with alice
|
|
|
|
chat := CreateOneToOneChat(common.PubkeyToHex(&s.alice.identity.PublicKey), &s.alice.identity.PublicKey, s.alice.transport)
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(s.bob.SaveChat(chat))
|
2021-08-06 16:40:23 +01:00
|
|
|
|
|
|
|
// Bob the admin shares with Alice, via public chat, an invite link to the new community
|
|
|
|
message := buildTestMessage(*chat)
|
|
|
|
message.CommunityID = community.IDString()
|
|
|
|
response, err := s.bob.SendChatMessage(context.Background(), message)
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
2021-08-06 16:40:23 +01:00
|
|
|
|
|
|
|
// Retrieve community link & community
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = s.alice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(response.Communities()) == 0 {
|
|
|
|
return errors.New("no communities received from 1-1")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err)
|
2021-08-06 16:40:23 +01:00
|
|
|
|
|
|
|
// Check that alice now has 2 communities
|
|
|
|
cs, err = s.alice.communitiesManager.All()
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err, "communitiesManager.All")
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Len(cs, 2, "Must have 2 communities")
|
|
|
|
for _, c := range cs {
|
|
|
|
s.False(c.Joined(), "Must not have joined the community")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Alice requests to join the new community
|
|
|
|
response, err = s.alice.RequestToJoinCommunity(&requests.RequestToJoinCommunity{CommunityID: community.ID()})
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
2023-06-16 11:10:32 +04:00
|
|
|
s.Require().Len(response.RequestsToJoinCommunity, 1)
|
2021-08-06 16:40:23 +01:00
|
|
|
|
2022-10-26 01:06:20 +03:00
|
|
|
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)
|
|
|
|
|
2021-08-06 16:40:23 +01:00
|
|
|
aRtj := response.RequestsToJoinCommunity[0]
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NotNil(aRtj)
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Equal(community.ID(), aRtj.CommunityID)
|
|
|
|
s.True(aRtj.Our)
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NotEmpty(aRtj.ID)
|
|
|
|
s.Require().NotEmpty(aRtj.Clock)
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Equal(aRtj.PublicKey, common.PubkeyToHex(&s.alice.identity.PublicKey))
|
|
|
|
s.Equal(communities.RequestToJoinStatePending, aRtj.State)
|
|
|
|
|
|
|
|
// Make sure clock is not empty
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NotEmpty(aRtj.Clock)
|
2021-08-06 16:40:23 +01:00
|
|
|
|
|
|
|
s.Len(response.Communities(), 1)
|
|
|
|
s.Equal(response.Communities()[0].RequestedToJoinAt(), aRtj.Clock)
|
|
|
|
|
|
|
|
// pull all communities to make sure we set RequestedToJoinAt
|
|
|
|
allCommunities, err := s.alice.Communities()
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err)
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Len(allCommunities, 2)
|
|
|
|
|
|
|
|
if bytes.Equal(allCommunities[0].ID(), community.ID()) {
|
|
|
|
s.Equal(allCommunities[0].RequestedToJoinAt(), aRtj.Clock)
|
|
|
|
} else {
|
|
|
|
s.Equal(allCommunities[1].RequestedToJoinAt(), aRtj.Clock)
|
|
|
|
}
|
|
|
|
|
|
|
|
// pull to make sure it has been saved
|
|
|
|
requestsToJoin, err := s.alice.MyPendingRequestsToJoin()
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err)
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Len(requestsToJoin, 1)
|
|
|
|
|
|
|
|
// Make sure the requests are fetched also by community
|
|
|
|
requestsToJoin, err = s.alice.PendingRequestsToJoinForCommunity(community.ID())
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err)
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Len(requestsToJoin, 1)
|
|
|
|
|
|
|
|
// Alice's other device retrieves sync message from the join
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = alicesOtherDevice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Do we have a new synced community?
|
|
|
|
_, err = alicesOtherDevice.communitiesManager.GetSyncedRawCommunity(community.ID())
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("community with sync not received %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Do we have a new pending request to join for the new community
|
|
|
|
requestsToJoin, err = alicesOtherDevice.PendingRequestsToJoinForCommunity(community.ID())
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(requestsToJoin) == 0 {
|
|
|
|
return errors.New("no requests to join")
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
})
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err)
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Len(response.Communities(), 1)
|
|
|
|
|
|
|
|
// Get the pending requests to join for the new community on alicesOtherDevice
|
|
|
|
requestsToJoin, err = alicesOtherDevice.PendingRequestsToJoinForCommunity(community.ID())
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err)
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Len(requestsToJoin, 1)
|
|
|
|
|
|
|
|
// Check request to join on alicesOtherDevice matches the RTJ on alice
|
|
|
|
aodRtj := requestsToJoin[0]
|
|
|
|
s.Equal(aRtj.PublicKey, aodRtj.PublicKey)
|
|
|
|
s.Equal(aRtj.ID, aodRtj.ID)
|
|
|
|
s.Equal(aRtj.CommunityID, aodRtj.CommunityID)
|
|
|
|
s.Equal(aRtj.Clock, aodRtj.Clock)
|
|
|
|
s.Equal(aRtj.ENSName, aodRtj.ENSName)
|
|
|
|
s.Equal(aRtj.ChatID, aodRtj.ChatID)
|
|
|
|
s.Equal(aRtj.State, aodRtj.State)
|
|
|
|
|
|
|
|
// Bob the admin retrieves request to join
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = s.bob.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(response.RequestsToJoinCommunity) == 0 {
|
|
|
|
return errors.New("request to join community not received")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err)
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Len(response.RequestsToJoinCommunity, 1)
|
|
|
|
|
|
|
|
// Check that bob the admin's newly received request to join matches what we expect
|
|
|
|
bobRtj := response.RequestsToJoinCommunity[0]
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NotNil(bobRtj)
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Equal(community.ID(), bobRtj.CommunityID)
|
|
|
|
s.False(bobRtj.Our)
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NotEmpty(bobRtj.ID)
|
|
|
|
s.Require().NotEmpty(bobRtj.Clock)
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Equal(bobRtj.PublicKey, common.PubkeyToHex(&s.alice.identity.PublicKey))
|
|
|
|
s.Equal(communities.RequestToJoinStatePending, bobRtj.State)
|
|
|
|
|
|
|
|
s.Equal(aRtj.PublicKey, bobRtj.PublicKey)
|
|
|
|
s.Equal(aRtj.ID, bobRtj.ID)
|
|
|
|
s.Equal(aRtj.CommunityID, bobRtj.CommunityID)
|
|
|
|
s.Equal(aRtj.Clock, bobRtj.Clock)
|
|
|
|
s.Equal(aRtj.ENSName, bobRtj.ENSName)
|
|
|
|
s.Equal(aRtj.ChatID, bobRtj.ChatID)
|
|
|
|
s.Equal(aRtj.State, bobRtj.State)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *MessengerCommunitiesSuite) pairTwoDevices(device1, device2 *Messenger, deviceName, deviceType string) {
|
|
|
|
// Send pairing data
|
2023-02-28 20:32:45 +08:00
|
|
|
response, err := device1.SendPairInstallation(context.Background(), nil)
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Len(response.Chats(), 1)
|
|
|
|
s.False(response.Chats()[0].Active)
|
|
|
|
|
|
|
|
// Wait for the message to reach its destination
|
|
|
|
response, err = WaitOnMessengerResponse(
|
|
|
|
device2,
|
2021-10-06 12:44:24 +01:00
|
|
|
func(r *MessengerResponse) bool {
|
|
|
|
for _, installation := range r.Installations {
|
|
|
|
if installation.ID == device1.installationID {
|
|
|
|
return installation.InstallationMetadata != nil && deviceName == installation.InstallationMetadata.Name && deviceType == installation.InstallationMetadata.DeviceType
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
|
|
|
|
},
|
2021-08-06 16:40:23 +01:00
|
|
|
"installation not received",
|
|
|
|
)
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
2021-08-06 16:40:23 +01:00
|
|
|
|
|
|
|
// Ensure installation is enabled
|
|
|
|
err = device2.EnableInstallation(device1.installationID)
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err)
|
2021-08-06 16:40:23 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *MessengerCommunitiesSuite) TestSyncCommunity_Leave() {
|
|
|
|
// Set Alice's installation metadata
|
|
|
|
aim := &multidevice.InstallationMetadata{
|
|
|
|
Name: "alice's-device",
|
|
|
|
DeviceType: "alice's-device-type",
|
|
|
|
}
|
|
|
|
err := s.alice.SetInstallationMetadata(s.alice.installationID, aim)
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err)
|
2021-08-06 16:40:23 +01:00
|
|
|
|
|
|
|
// Create Alice's other device
|
2023-06-27 14:20:20 +02:00
|
|
|
alicesOtherDevice := s.newMessengerWithKey(s.alice.identity)
|
2021-08-06 16:40:23 +01:00
|
|
|
|
|
|
|
im1 := &multidevice.InstallationMetadata{
|
|
|
|
Name: "alice's-other-device",
|
|
|
|
DeviceType: "alice's-other-device-type",
|
|
|
|
}
|
|
|
|
err = alicesOtherDevice.SetInstallationMetadata(alicesOtherDevice.installationID, im1)
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err)
|
2021-08-06 16:40:23 +01:00
|
|
|
|
|
|
|
// Pair alice's two devices
|
|
|
|
s.pairTwoDevices(alicesOtherDevice, s.alice, im1.Name, im1.DeviceType)
|
|
|
|
s.pairTwoDevices(s.alice, alicesOtherDevice, aim.Name, aim.DeviceType)
|
|
|
|
|
|
|
|
// Check bob the admin has only one community
|
|
|
|
tcs2, err := s.bob.communitiesManager.All()
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err, "admin.communitiesManager.All")
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Len(tcs2, 1, "Must have 1 communities")
|
|
|
|
|
|
|
|
// Bob the admin creates a community
|
|
|
|
createCommunityReq := &requests.CreateCommunity{
|
|
|
|
Membership: protobuf.CommunityPermissions_NO_MEMBERSHIP,
|
|
|
|
Name: "new community",
|
|
|
|
Color: "#000000",
|
|
|
|
Description: "new community description",
|
|
|
|
}
|
2022-07-19 11:31:52 +02:00
|
|
|
mr, err := s.bob.CreateCommunity(createCommunityReq, true)
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err, "CreateCommunity")
|
|
|
|
s.Require().NotNil(mr)
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Len(mr.Communities(), 1)
|
|
|
|
|
|
|
|
community := mr.Communities()[0]
|
|
|
|
|
|
|
|
// Check that admin has 2 communities
|
|
|
|
acs, err := s.bob.communitiesManager.All()
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err, "communitiesManager.All")
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Len(acs, 2, "Must have 2 communities")
|
|
|
|
|
|
|
|
// Check that Alice has only 1 community on either device
|
|
|
|
cs, err := s.alice.communitiesManager.All()
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err, "communitiesManager.All")
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Len(cs, 1, "Must have 1 communities")
|
|
|
|
|
|
|
|
tcs1, err := alicesOtherDevice.communitiesManager.All()
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err, "alicesOtherDevice.communitiesManager.All")
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Len(tcs1, 1, "Must have 1 communities")
|
|
|
|
|
|
|
|
// Bob the admin opens up a 1-1 chat with alice
|
|
|
|
chat := CreateOneToOneChat(common.PubkeyToHex(&s.alice.identity.PublicKey), &s.alice.identity.PublicKey, s.alice.transport)
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(s.bob.SaveChat(chat))
|
2021-08-06 16:40:23 +01:00
|
|
|
|
|
|
|
// Bob the admin shares with Alice, via public chat, an invite link to the new community
|
|
|
|
message := buildTestMessage(*chat)
|
|
|
|
message.CommunityID = community.IDString()
|
|
|
|
response, err := s.bob.SendChatMessage(context.Background(), message)
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
2021-08-06 16:40:23 +01:00
|
|
|
|
|
|
|
// Retrieve community link & community
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = s.alice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(response.Communities()) == 0 {
|
|
|
|
return errors.New("no communities received from 1-1")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err)
|
2021-08-06 16:40:23 +01:00
|
|
|
|
|
|
|
// Check that alice now has 2 communities
|
|
|
|
cs, err = s.alice.communitiesManager.All()
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err, "communitiesManager.All")
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Len(cs, 2, "Must have 2 communities")
|
|
|
|
for _, c := range cs {
|
|
|
|
s.False(c.Joined(), "Must not have joined the community")
|
|
|
|
}
|
|
|
|
|
|
|
|
// alice joins the community
|
2023-05-29 13:57:05 -04:00
|
|
|
mr, err = s.alice.JoinCommunity(context.Background(), community.ID(), false)
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err, "s.alice.JoinCommunity")
|
|
|
|
s.Require().NotNil(mr)
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Len(mr.Communities(), 1)
|
|
|
|
aCom := mr.Communities()[0]
|
|
|
|
|
|
|
|
// Check that the joined community has the correct values
|
|
|
|
s.Equal(community.ID(), aCom.ID())
|
2022-05-10 15:25:08 +02:00
|
|
|
s.Equal(uint64(0x2), aCom.Clock())
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Equal(community.PublicKey(), aCom.PublicKey())
|
|
|
|
|
|
|
|
// Check alicesOtherDevice receives the sync join message
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = alicesOtherDevice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Do we have a new synced community?
|
|
|
|
_, err = alicesOtherDevice.communitiesManager.GetSyncedRawCommunity(community.ID())
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("community with sync not received %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
})
|
2021-10-06 12:44:24 +01:00
|
|
|
s.Require().NoError(err)
|
2021-08-06 16:40:23 +01:00
|
|
|
s.Len(response.Communities(), 1, "")
|
|
|
|
|
|
|
|
aoCom := mr.Communities()[0]
|
|
|
|
s.Equal(aCom, aoCom)
|
|
|
|
}
|
2022-07-06 19:16:19 +03:00
|
|
|
|
|
|
|
func (s *MessengerCommunitiesSuite) TestSetMutePropertyOnChatsByCategory() {
|
|
|
|
// Create a community
|
|
|
|
createCommunityReq := &requests.CreateCommunity{
|
|
|
|
Membership: protobuf.CommunityPermissions_ON_REQUEST,
|
|
|
|
Name: "new community",
|
|
|
|
Color: "#000000",
|
|
|
|
Description: "new community description",
|
|
|
|
}
|
|
|
|
|
2022-07-19 11:31:52 +02:00
|
|
|
mr, err := s.alice.CreateCommunity(createCommunityReq, true)
|
2022-07-06 19:16:19 +03:00
|
|
|
s.Require().NoError(err, "s.alice.CreateCommunity")
|
|
|
|
var newCommunity *communities.Community
|
|
|
|
for _, com := range mr.Communities() {
|
|
|
|
if com.Name() == createCommunityReq.Name {
|
|
|
|
newCommunity = com
|
|
|
|
}
|
|
|
|
}
|
|
|
|
s.Require().NotNil(newCommunity)
|
|
|
|
|
|
|
|
orgChat1 := &protobuf.CommunityChat{
|
|
|
|
Permissions: &protobuf.CommunityPermissions{
|
|
|
|
Access: protobuf.CommunityPermissions_NO_MEMBERSHIP,
|
|
|
|
},
|
|
|
|
Identity: &protobuf.ChatIdentity{
|
|
|
|
DisplayName: "status-core",
|
|
|
|
Emoji: "😎",
|
|
|
|
Description: "status-core community chat",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
orgChat2 := &protobuf.CommunityChat{
|
|
|
|
Permissions: &protobuf.CommunityPermissions{
|
|
|
|
Access: protobuf.CommunityPermissions_NO_MEMBERSHIP,
|
|
|
|
},
|
|
|
|
Identity: &protobuf.ChatIdentity{
|
|
|
|
DisplayName: "status-core2",
|
|
|
|
Emoji: "😎",
|
|
|
|
Description: "status-core community chat2",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
mr, err = s.alice.CreateCommunityChat(newCommunity.ID(), orgChat1)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(mr)
|
|
|
|
s.Require().Len(mr.Communities(), 1)
|
|
|
|
s.Require().Len(mr.Chats(), 1)
|
|
|
|
|
|
|
|
mr, err = s.alice.CreateCommunityChat(newCommunity.ID(), orgChat2)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(mr)
|
|
|
|
s.Require().Len(mr.Communities(), 1)
|
|
|
|
s.Require().Len(mr.Chats(), 1)
|
|
|
|
|
|
|
|
var chatIds []string
|
|
|
|
for k := range newCommunity.Chats() {
|
|
|
|
chatIds = append(chatIds, k)
|
|
|
|
}
|
|
|
|
category := &requests.CreateCommunityCategory{
|
|
|
|
CommunityID: newCommunity.ID(),
|
|
|
|
CategoryName: "category-name",
|
|
|
|
ChatIDs: chatIds,
|
|
|
|
}
|
|
|
|
|
|
|
|
mr, err = s.alice.CreateCommunityCategory(category)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(mr)
|
|
|
|
s.Require().Len(mr.Communities(), 1)
|
|
|
|
s.Require().Len(mr.Communities()[0].Categories(), 1)
|
|
|
|
|
|
|
|
var categoryID string
|
|
|
|
for k := range mr.Communities()[0].Categories() {
|
|
|
|
categoryID = k
|
|
|
|
}
|
|
|
|
|
2023-06-02 14:25:32 +03:00
|
|
|
err = s.alice.SetMutePropertyOnChatsByCategory(&requests.MuteCategory{
|
|
|
|
CommunityID: newCommunity.IDString(),
|
|
|
|
CategoryID: categoryID,
|
|
|
|
MutedType: MuteTillUnmuted,
|
|
|
|
}, true)
|
2022-07-06 19:16:19 +03:00
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
for _, chat := range s.alice.Chats() {
|
|
|
|
if chat.CategoryID == categoryID {
|
|
|
|
s.Require().True(chat.Muted)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-02 14:25:32 +03:00
|
|
|
err = s.alice.SetMutePropertyOnChatsByCategory(&requests.MuteCategory{
|
|
|
|
CommunityID: newCommunity.IDString(),
|
|
|
|
CategoryID: categoryID,
|
|
|
|
MutedType: Unmuted,
|
|
|
|
}, false)
|
2022-07-06 19:16:19 +03:00
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
for _, chat := range s.alice.Chats() {
|
|
|
|
s.Require().False(chat.Muted)
|
|
|
|
}
|
|
|
|
}
|
feat: introduce messenger APIs to extract discord channels
As part of the new Discord <-> Status Community Import functionality,
we're adding an API that extracts all discord categories and channels
from a previously exported discord export file.
These APIs can be used in clients to show the user what categories and
channels will be imported later on.
There are two APIs:
1. `Messenger.ExtractDiscordCategoriesAndChannels(filesToimport
[]string) (*MessengerResponse, map[string]*discord.ImportError)`
This takes a list of exported discord export (JSON) files (typically one per
channel), reads them, and extracts the categories and channels into
dedicated data structures (`[]DiscordChannel` and `[]DiscordCategory`)
It also returns the oldest message timestamp found in all extracted
channels.
The API is synchronous and returns the extracted data as
a `*MessengerResponse`. This allows to make the API available
status-go's RPC interface.
The error case is a `map[string]*discord.ImportError` where each key
is a file path of a JSON file that we tried to extract data from, and
the value a `discord.ImportError` which holds an error message and an
error code, allowing for distinguishing between "critical" errors and
"non-critical" errors.
2. `Messenger.RequestExtractDiscordCategoriesAndChannels(filesToImport
[]string)`
This is the asynchronous counterpart to
`ExtractDiscordCategoriesAndChannels`. The reason this API has been
added is because discord servers can have a lot of message and
channel data, which causes `ExtractDiscordCategoriesAndChannels` to
block the thread for too long, making apps potentially feel like they
are stuck.
This API runs inside a go routine, eventually calls
`ExtractDiscordCategoriesAndChannels`, and then emits a newly
introduced `DiscordCategoriesAndChannelsExtractedSignal` that clients
can react to.
Failure of extraction has to be determined by the
`discord.ImportErrors` emitted by the signal.
**A note about exported discord history files**
We expect users to export their discord histories via the
[DiscordChatExporter](https://github.com/Tyrrrz/DiscordChatExporter/wiki/GUI%2C-CLI-and-Formats-explained#exportguild)
tool. The tool allows to export the data in different formats, such as
JSON, HTML and CSV.
We expect users to have their data exported as JSON.
Closes: https://github.com/status-im/status-desktop/issues/6690
2022-07-13 11:33:53 +02:00
|
|
|
|
2023-06-17 11:19:05 +03:00
|
|
|
func (s *MessengerCommunitiesSuite) TestCheckCommunitiesToUnmute() {
|
|
|
|
// Create a community
|
|
|
|
createCommunityReq := &requests.CreateCommunity{
|
|
|
|
Membership: protobuf.CommunityPermissions_ON_REQUEST,
|
|
|
|
Name: "new community",
|
|
|
|
Color: "#000000",
|
|
|
|
Description: "new community description",
|
|
|
|
}
|
|
|
|
|
|
|
|
mr, err := s.alice.CreateCommunity(createCommunityReq, true)
|
|
|
|
s.Require().NoError(err, "s.alice.CreateCommunity")
|
|
|
|
var newCommunity *communities.Community
|
|
|
|
for _, com := range mr.Communities() {
|
|
|
|
if com.Name() == createCommunityReq.Name {
|
|
|
|
newCommunity = com
|
|
|
|
}
|
|
|
|
}
|
|
|
|
s.Require().NotNil(newCommunity)
|
|
|
|
|
2023-07-19 15:14:42 +03:00
|
|
|
currTime, err := time.Parse(time.RFC3339, time.Now().Add(-time.Hour).Format(time.RFC3339))
|
|
|
|
s.Require().NoError(err)
|
2023-06-17 11:19:05 +03:00
|
|
|
|
2023-07-19 15:14:42 +03:00
|
|
|
err = s.alice.communitiesManager.SetMuted(newCommunity.ID(), true)
|
2023-06-17 11:19:05 +03:00
|
|
|
s.Require().NoError(err, "SetMuted to community")
|
|
|
|
|
2023-07-19 15:14:42 +03:00
|
|
|
err = s.alice.communitiesManager.MuteCommunityTill(newCommunity.ID(), currTime)
|
|
|
|
s.Require().NoError(err, "SetMuteTill to community")
|
2023-06-17 11:19:05 +03:00
|
|
|
|
2023-07-19 15:14:42 +03:00
|
|
|
response, err := s.alice.CheckCommunitiesToUnmute()
|
2023-06-17 11:19:05 +03:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(response.Communities(), 1, "CheckCommunitiesToUnmute should unmute the community")
|
2023-07-19 15:14:42 +03:00
|
|
|
|
|
|
|
community, err := s.alice.communitiesManager.GetByID(newCommunity.ID())
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().False(community.Muted())
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *MessengerCommunitiesSuite) TestMuteAllCommunityChats() {
|
|
|
|
// Create a community
|
|
|
|
createCommunityReq := &requests.CreateCommunity{
|
|
|
|
Membership: protobuf.CommunityPermissions_ON_REQUEST,
|
|
|
|
Name: "new community",
|
|
|
|
Color: "#000000",
|
|
|
|
Description: "new community description",
|
|
|
|
}
|
|
|
|
|
|
|
|
mr, err := s.alice.CreateCommunity(createCommunityReq, true)
|
|
|
|
s.Require().NoError(err, "s.alice.CreateCommunity")
|
|
|
|
var newCommunity *communities.Community
|
|
|
|
for _, com := range mr.Communities() {
|
|
|
|
if com.Name() == createCommunityReq.Name {
|
|
|
|
newCommunity = com
|
|
|
|
}
|
|
|
|
}
|
|
|
|
s.Require().NotNil(newCommunity)
|
|
|
|
|
|
|
|
orgChat1 := &protobuf.CommunityChat{
|
|
|
|
Permissions: &protobuf.CommunityPermissions{
|
|
|
|
Access: protobuf.CommunityPermissions_NO_MEMBERSHIP,
|
|
|
|
},
|
|
|
|
Identity: &protobuf.ChatIdentity{
|
|
|
|
DisplayName: "status-core",
|
|
|
|
Emoji: "😎",
|
|
|
|
Description: "status-core community chat",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
orgChat2 := &protobuf.CommunityChat{
|
|
|
|
Permissions: &protobuf.CommunityPermissions{
|
|
|
|
Access: protobuf.CommunityPermissions_NO_MEMBERSHIP,
|
|
|
|
},
|
|
|
|
Identity: &protobuf.ChatIdentity{
|
|
|
|
DisplayName: "status-core2",
|
|
|
|
Emoji: "😎",
|
|
|
|
Description: "status-core community chat2",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
mr, err = s.alice.CreateCommunityChat(newCommunity.ID(), orgChat1)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(mr)
|
|
|
|
s.Require().Len(mr.Communities(), 1)
|
|
|
|
s.Require().Len(mr.Chats(), 1)
|
|
|
|
|
|
|
|
mr, err = s.alice.CreateCommunityChat(newCommunity.ID(), orgChat2)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(mr)
|
|
|
|
s.Require().Len(mr.Communities(), 1)
|
|
|
|
s.Require().Len(mr.Chats(), 1)
|
|
|
|
|
|
|
|
muteDuration, err := s.alice.MuteDuration(MuteFor15Min)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
time, err := s.alice.MuteAllCommunityChats(&requests.MuteCommunity{
|
|
|
|
CommunityID: newCommunity.ID(),
|
|
|
|
MutedType: MuteFor15Min,
|
|
|
|
})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(time)
|
|
|
|
|
|
|
|
aliceCommunity, err := s.alice.GetCommunityByID(newCommunity.ID())
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().True(aliceCommunity.Muted())
|
|
|
|
|
|
|
|
for _, chat := range s.alice.Chats() {
|
|
|
|
if chat.CommunityID == newCommunity.IDString() {
|
|
|
|
s.Require().True(chat.Muted)
|
|
|
|
s.Require().Equal(chat.MuteTill, muteDuration)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, chat := range s.alice.Chats() {
|
|
|
|
if chat.CommunityID == newCommunity.IDString() {
|
|
|
|
err = s.alice.UnmuteChat(chat.ID)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().False(chat.Muted)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
aliceCommunity, err = s.alice.GetCommunityByID(newCommunity.ID())
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().False(aliceCommunity.Muted())
|
|
|
|
|
|
|
|
time, err = s.alice.UnMuteAllCommunityChats(newCommunity.IDString())
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(time)
|
|
|
|
s.Require().False(newCommunity.Muted())
|
|
|
|
|
|
|
|
for _, chat := range s.alice.Chats() {
|
|
|
|
s.Require().False(chat.Muted)
|
|
|
|
}
|
|
|
|
|
2023-06-17 11:19:05 +03:00
|
|
|
}
|
|
|
|
|
feat: introduce messenger APIs to extract discord channels
As part of the new Discord <-> Status Community Import functionality,
we're adding an API that extracts all discord categories and channels
from a previously exported discord export file.
These APIs can be used in clients to show the user what categories and
channels will be imported later on.
There are two APIs:
1. `Messenger.ExtractDiscordCategoriesAndChannels(filesToimport
[]string) (*MessengerResponse, map[string]*discord.ImportError)`
This takes a list of exported discord export (JSON) files (typically one per
channel), reads them, and extracts the categories and channels into
dedicated data structures (`[]DiscordChannel` and `[]DiscordCategory`)
It also returns the oldest message timestamp found in all extracted
channels.
The API is synchronous and returns the extracted data as
a `*MessengerResponse`. This allows to make the API available
status-go's RPC interface.
The error case is a `map[string]*discord.ImportError` where each key
is a file path of a JSON file that we tried to extract data from, and
the value a `discord.ImportError` which holds an error message and an
error code, allowing for distinguishing between "critical" errors and
"non-critical" errors.
2. `Messenger.RequestExtractDiscordCategoriesAndChannels(filesToImport
[]string)`
This is the asynchronous counterpart to
`ExtractDiscordCategoriesAndChannels`. The reason this API has been
added is because discord servers can have a lot of message and
channel data, which causes `ExtractDiscordCategoriesAndChannels` to
block the thread for too long, making apps potentially feel like they
are stuck.
This API runs inside a go routine, eventually calls
`ExtractDiscordCategoriesAndChannels`, and then emits a newly
introduced `DiscordCategoriesAndChannelsExtractedSignal` that clients
can react to.
Failure of extraction has to be determined by the
`discord.ImportErrors` emitted by the signal.
**A note about exported discord history files**
We expect users to export their discord histories via the
[DiscordChatExporter](https://github.com/Tyrrrz/DiscordChatExporter/wiki/GUI%2C-CLI-and-Formats-explained#exportguild)
tool. The tool allows to export the data in different formats, such as
JSON, HTML and CSV.
We expect users to have their data exported as JSON.
Closes: https://github.com/status-im/status-desktop/issues/6690
2022-07-13 11:33:53 +02:00
|
|
|
func (s *MessengerCommunitiesSuite) TestExtractDiscordChannelsAndCategories() {
|
|
|
|
|
|
|
|
tmpFile, err := ioutil.TempFile(os.TempDir(), "discord-channel-")
|
|
|
|
s.Require().NoError(err)
|
|
|
|
defer os.Remove(tmpFile.Name())
|
|
|
|
|
|
|
|
discordMessage := &protobuf.DiscordMessage{
|
|
|
|
Id: "1234",
|
|
|
|
Type: "Default",
|
|
|
|
Timestamp: "2022-07-26T14:20:17.305+00:00",
|
|
|
|
TimestampEdited: "",
|
|
|
|
Content: "Some discord message",
|
|
|
|
Author: &protobuf.DiscordMessageAuthor{
|
|
|
|
Id: "123",
|
|
|
|
Name: "TestAuthor",
|
|
|
|
Discriminator: "456",
|
|
|
|
Nickname: "",
|
|
|
|
AvatarUrl: "",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
messages := make([]*protobuf.DiscordMessage, 0)
|
|
|
|
messages = append(messages, discordMessage)
|
|
|
|
|
|
|
|
exportedDiscordData := &discord.ExportedData{
|
|
|
|
Channel: discord.Channel{
|
|
|
|
ID: "12345",
|
|
|
|
CategoryName: "test-category",
|
|
|
|
CategoryID: "6789",
|
|
|
|
Name: "test-channel",
|
|
|
|
Description: "This is a channel topic",
|
|
|
|
FilePath: tmpFile.Name(),
|
|
|
|
},
|
|
|
|
Messages: messages,
|
|
|
|
}
|
|
|
|
|
|
|
|
data, err := json.Marshal(exportedDiscordData)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
err = os.WriteFile(tmpFile.Name(), data, 0666) // nolint: gosec
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
files := make([]string, 0)
|
|
|
|
files = append(files, tmpFile.Name())
|
|
|
|
mr, errs := s.bob.ExtractDiscordChannelsAndCategories(files)
|
|
|
|
s.Require().Len(errs, 0)
|
|
|
|
|
|
|
|
s.Require().Len(mr.DiscordCategories, 1)
|
|
|
|
s.Require().Len(mr.DiscordChannels, 1)
|
|
|
|
s.Require().Equal(mr.DiscordOldestMessageTimestamp, int(1658845217))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *MessengerCommunitiesSuite) TestExtractDiscordChannelsAndCategories_WithErrors() {
|
|
|
|
|
|
|
|
tmpFile, err := ioutil.TempFile(os.TempDir(), "discord-channel-2")
|
|
|
|
s.Require().NoError(err)
|
|
|
|
defer os.Remove(tmpFile.Name())
|
|
|
|
|
|
|
|
exportedDiscordData := &discord.ExportedData{
|
|
|
|
Channel: discord.Channel{
|
|
|
|
ID: "12345",
|
|
|
|
CategoryName: "test-category",
|
|
|
|
CategoryID: "6789",
|
|
|
|
Name: "test-channel",
|
|
|
|
Description: "This is a channel topic",
|
|
|
|
FilePath: tmpFile.Name(),
|
|
|
|
},
|
|
|
|
Messages: make([]*protobuf.DiscordMessage, 0),
|
|
|
|
}
|
|
|
|
|
|
|
|
data, err := json.Marshal(exportedDiscordData)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
err = os.WriteFile(tmpFile.Name(), data, 0666) // nolint: gosec
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
files := make([]string, 0)
|
|
|
|
files = append(files, tmpFile.Name())
|
|
|
|
_, errs := s.bob.ExtractDiscordChannelsAndCategories(files)
|
|
|
|
// Expecting 1 errors since there are no messages to be extracted
|
|
|
|
s.Require().Len(errs, 1)
|
|
|
|
}
|
2023-02-20 12:57:33 +01:00
|
|
|
|
2023-05-03 18:01:55 +03:00
|
|
|
func (s *MessengerCommunitiesSuite) TestCommunityBanUserRequesToJoin() {
|
|
|
|
description := &requests.CreateCommunity{
|
|
|
|
Membership: protobuf.CommunityPermissions_NO_MEMBERSHIP,
|
|
|
|
Name: "status",
|
|
|
|
Color: "#ffffff",
|
|
|
|
Description: "status community description",
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create an community chat
|
|
|
|
response, err := s.bob.CreateCommunity(description, true)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
|
|
|
|
community := response.Communities()[0]
|
|
|
|
|
|
|
|
response, err = s.bob.InviteUsersToCommunity(
|
|
|
|
&requests.InviteUsersToCommunity{
|
|
|
|
CommunityID: community.ID(),
|
|
|
|
Users: []types.HexBytes{common.PubkeyToHexBytes(&s.alice.identity.PublicKey)},
|
|
|
|
},
|
|
|
|
)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
|
|
|
|
community = response.Communities()[0]
|
|
|
|
s.Require().True(community.HasMember(&s.alice.identity.PublicKey))
|
|
|
|
|
|
|
|
response, err = s.bob.BanUserFromCommunity(
|
|
|
|
&requests.BanUserFromCommunity{
|
|
|
|
CommunityID: community.ID(),
|
|
|
|
User: common.PubkeyToHexBytes(&s.alice.identity.PublicKey),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
|
|
|
|
community = response.Communities()[0]
|
|
|
|
s.Require().False(community.HasMember(&s.alice.identity.PublicKey))
|
|
|
|
s.Require().True(community.IsBanned(&s.alice.identity.PublicKey))
|
|
|
|
|
|
|
|
response, err = WaitOnMessengerResponse(
|
|
|
|
s.alice,
|
|
|
|
func(r *MessengerResponse) bool { return len(r.communities) > 0 },
|
|
|
|
"no communities",
|
|
|
|
)
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
|
2023-06-27 14:20:20 +02:00
|
|
|
request := &requests.RequestToJoinCommunity{CommunityID: community.ID()}
|
2023-05-03 18:01:55 +03:00
|
|
|
// We try to join the org
|
2023-07-14 13:06:37 -04:00
|
|
|
_, rtj, err := s.alice.communitiesManager.CreateRequestToJoin(&s.alice.identity.PublicKey, request)
|
2023-05-03 18:01:55 +03:00
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
displayName, err := s.alice.settings.DisplayName()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
requestToJoinProto := &protobuf.CommunityRequestToJoin{
|
2023-06-06 20:33:09 +02:00
|
|
|
Clock: rtj.Clock,
|
|
|
|
EnsName: rtj.ENSName,
|
|
|
|
DisplayName: displayName,
|
|
|
|
CommunityId: community.ID(),
|
|
|
|
RevealedAccounts: make([]*protobuf.RevealedAccount, 0),
|
2023-05-03 18:01:55 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
messageState := s.bob.buildMessageState()
|
|
|
|
|
|
|
|
err = s.bob.HandleCommunityRequestToJoin(messageState, &s.alice.identity.PublicKey, *requestToJoinProto)
|
|
|
|
|
|
|
|
s.Require().ErrorContains(err, "can't request access")
|
|
|
|
}
|
2023-05-29 13:57:05 -04:00
|
|
|
|
|
|
|
func (s *MessengerCommunitiesSuite) TestHandleImport() {
|
|
|
|
description := &requests.CreateCommunity{
|
|
|
|
Membership: protobuf.CommunityPermissions_INVITATION_ONLY,
|
|
|
|
Name: "status",
|
|
|
|
Color: "#ffffff",
|
|
|
|
Description: "status community description",
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create a community
|
|
|
|
response, err := s.bob.CreateCommunity(description, true)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
s.Require().Len(response.Communities()[0].Chats(), 1)
|
|
|
|
s.Require().Len(response.Chats(), 1)
|
|
|
|
|
|
|
|
community := response.Communities()[0]
|
|
|
|
|
|
|
|
// Create chat
|
|
|
|
orgChat := &protobuf.CommunityChat{
|
|
|
|
Permissions: &protobuf.CommunityPermissions{
|
|
|
|
Access: protobuf.CommunityPermissions_NO_MEMBERSHIP,
|
|
|
|
},
|
|
|
|
Identity: &protobuf.ChatIdentity{
|
|
|
|
DisplayName: "status-core",
|
|
|
|
Description: "status-core community chat",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
response, err = s.bob.CreateCommunityChat(community.ID(), orgChat)
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
s.Require().Len(response.Communities()[0].Chats(), 2)
|
|
|
|
s.Require().Len(response.Chats(), 1)
|
|
|
|
|
|
|
|
response, err = s.bob.InviteUsersToCommunity(
|
|
|
|
&requests.InviteUsersToCommunity{
|
|
|
|
CommunityID: community.ID(),
|
|
|
|
Users: []types.HexBytes{common.PubkeyToHexBytes(&s.alice.identity.PublicKey)},
|
|
|
|
},
|
|
|
|
)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
|
|
|
|
community = response.Communities()[0]
|
|
|
|
s.Require().True(community.HasMember(&s.alice.identity.PublicKey))
|
|
|
|
|
|
|
|
// Pull message and make sure org is received
|
|
|
|
err = tt.RetryWithBackOff(func() error {
|
|
|
|
response, err = s.alice.RetrieveAll()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(response.Communities()) == 0 {
|
|
|
|
return errors.New("community not received")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
s.Require().NoError(err)
|
|
|
|
communities, err := s.alice.Communities()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(communities, 2)
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
|
|
|
|
communityID := response.Communities()[0].ID()
|
|
|
|
s.Require().Equal(communityID, community.ID())
|
|
|
|
|
|
|
|
ctx := context.Background()
|
|
|
|
|
|
|
|
// We join the org
|
|
|
|
response, err = s.alice.JoinCommunity(ctx, community.ID(), false)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(response)
|
|
|
|
s.Require().Len(response.Communities(), 1)
|
|
|
|
s.Require().Len(response.Communities()[0].Chats(), 2)
|
|
|
|
s.Require().True(response.Communities()[0].Joined())
|
|
|
|
s.Require().Len(response.Chats(), 2)
|
|
|
|
|
|
|
|
chatID := response.Chats()[1].ID
|
|
|
|
|
|
|
|
// Check that there are no messages in the chat at first
|
|
|
|
chat, err := s.alice.persistence.Chat(chatID)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(chat)
|
|
|
|
s.Require().Equal(0, int(chat.UnviewedMessagesCount))
|
|
|
|
|
|
|
|
// Create an message that will be imported
|
|
|
|
testMessage := protobuf.ChatMessage{
|
|
|
|
Text: "abc123",
|
|
|
|
ChatId: chatID,
|
|
|
|
ContentType: protobuf.ChatMessage_TEXT_PLAIN,
|
|
|
|
MessageType: protobuf.MessageType_COMMUNITY_CHAT,
|
|
|
|
Clock: 1,
|
|
|
|
Timestamp: 1,
|
|
|
|
}
|
|
|
|
encodedPayload, err := proto.Marshal(&testMessage)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
wrappedPayload, err := v1protocol.WrapMessageV1(
|
|
|
|
encodedPayload,
|
|
|
|
protobuf.ApplicationMetadataMessage_CHAT_MESSAGE,
|
|
|
|
s.bob.identity,
|
|
|
|
)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
message := &types.Message{}
|
|
|
|
message.Sig = crypto.FromECDSAPub(&s.bob.identity.PublicKey)
|
|
|
|
message.Payload = wrappedPayload
|
|
|
|
|
|
|
|
filter := s.alice.transport.FilterByChatID(chatID)
|
|
|
|
importedMessages := make(map[transport.Filter][]*types.Message, 0)
|
|
|
|
|
|
|
|
importedMessages[*filter] = append(importedMessages[*filter], message)
|
|
|
|
|
|
|
|
// Import that message
|
|
|
|
err = s.alice.handleImportedMessages(importedMessages)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
// Get the chat again and see that there is still no unread message because we don't count import messages
|
|
|
|
chat, err = s.alice.persistence.Chat(chatID)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().NotNil(chat)
|
|
|
|
s.Require().Equal(0, int(chat.UnviewedMessagesCount))
|
|
|
|
}
|
2023-07-07 14:26:38 -04:00
|
|
|
|
|
|
|
func (s *MessengerCommunitiesSuite) TestGetCommunityIdFromKey() {
|
|
|
|
publicKey := "0x029e4777ce55f20373db33546c8681a082bd181d665c87e18d4306766de9302b53"
|
|
|
|
privateKey := "0x3f932031cb5f94ba7eb8ab4c824c3677973ab01fde65d1b89e0b3f470003a2cd"
|
|
|
|
|
|
|
|
// Public key returns the same
|
|
|
|
communityID := s.bob.GetCommunityIDFromKey(publicKey)
|
|
|
|
s.Require().Equal(communityID, publicKey)
|
|
|
|
|
|
|
|
// Private key returns the public key
|
|
|
|
communityID = s.bob.GetCommunityIDFromKey(privateKey)
|
|
|
|
s.Require().Equal(communityID, publicKey)
|
|
|
|
}
|