2020-07-22 07:41:40 +00:00
|
|
|
package pushnotificationclient
|
2020-06-30 07:50:59 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"crypto/ecdsa"
|
2020-07-10 13:26:06 +00:00
|
|
|
"io/ioutil"
|
2020-06-30 07:50:59 +00:00
|
|
|
"math/rand"
|
2020-07-10 13:26:06 +00:00
|
|
|
"os"
|
2020-06-30 07:50:59 +00:00
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/google/uuid"
|
2020-07-22 07:41:40 +00:00
|
|
|
"github.com/stretchr/testify/suite"
|
|
|
|
|
2020-06-30 07:50:59 +00:00
|
|
|
"github.com/status-im/status-go/eth-node/crypto"
|
|
|
|
"github.com/status-im/status-go/eth-node/crypto/ecies"
|
2020-07-22 07:41:40 +00:00
|
|
|
"github.com/status-im/status-go/eth-node/types"
|
2020-07-10 07:45:40 +00:00
|
|
|
"github.com/status-im/status-go/protocol/common"
|
2020-06-30 07:50:59 +00:00
|
|
|
"github.com/status-im/status-go/protocol/protobuf"
|
2020-07-10 13:26:06 +00:00
|
|
|
"github.com/status-im/status-go/protocol/sqlite"
|
|
|
|
"github.com/status-im/status-go/protocol/tt"
|
2020-06-30 07:50:59 +00:00
|
|
|
)
|
|
|
|
|
2020-07-22 07:41:40 +00:00
|
|
|
const testDeviceToken = "test-token"
|
|
|
|
|
2020-07-10 13:26:06 +00:00
|
|
|
type ClientSuite struct {
|
|
|
|
suite.Suite
|
|
|
|
tmpFile *os.File
|
|
|
|
persistence *Persistence
|
|
|
|
identity *ecdsa.PrivateKey
|
|
|
|
installationID string
|
|
|
|
client *Client
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestClientSuite(t *testing.T) {
|
|
|
|
s := new(ClientSuite)
|
|
|
|
s.installationID = "c6ae4fde-bb65-11ea-b3de-0242ac130004"
|
|
|
|
|
|
|
|
suite.Run(t, s)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *ClientSuite) SetupTest() {
|
|
|
|
tmpFile, err := ioutil.TempFile("", "")
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.tmpFile = tmpFile
|
|
|
|
|
|
|
|
database, err := sqlite.Open(s.tmpFile.Name(), "")
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.persistence = NewPersistence(database)
|
|
|
|
|
|
|
|
identity, err := crypto.GenerateKey()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.identity = identity
|
|
|
|
|
|
|
|
config := &Config{
|
|
|
|
Identity: identity,
|
|
|
|
Logger: tt.MustCreateTestLogger(),
|
|
|
|
RemoteNotificationsEnabled: true,
|
|
|
|
InstallationID: s.installationID,
|
|
|
|
}
|
|
|
|
|
|
|
|
s.client = New(s.persistence, config, nil)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *ClientSuite) TestBuildPushNotificationRegisterMessage() {
|
2020-06-30 07:50:59 +00:00
|
|
|
mutedChatList := []string{"a", "b"}
|
|
|
|
|
|
|
|
// build chat lish hashes
|
|
|
|
var mutedChatListHashes [][]byte
|
|
|
|
for _, chatID := range mutedChatList {
|
2020-07-10 07:45:40 +00:00
|
|
|
mutedChatListHashes = append(mutedChatListHashes, common.Shake256([]byte(chatID)))
|
2020-06-30 07:50:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
contactKey, err := crypto.GenerateKey()
|
2020-07-10 13:26:06 +00:00
|
|
|
s.Require().NoError(err)
|
2020-06-30 07:50:59 +00:00
|
|
|
contactIDs := []*ecdsa.PublicKey{&contactKey.PublicKey}
|
|
|
|
|
|
|
|
// Set random generator for uuid
|
|
|
|
var seed int64 = 1
|
|
|
|
uuid.SetRand(rand.New(rand.NewSource(seed)))
|
|
|
|
|
|
|
|
// Get token
|
|
|
|
expectedUUID := uuid.New().String()
|
|
|
|
|
2020-07-17 12:29:51 +00:00
|
|
|
// Reset random generator
|
|
|
|
uuid.SetRand(rand.New(rand.NewSource(seed)))
|
|
|
|
|
2020-07-22 07:41:40 +00:00
|
|
|
s.client.deviceToken = testDeviceToken
|
2020-07-17 12:29:51 +00:00
|
|
|
// Set reader
|
|
|
|
s.client.reader = bytes.NewReader([]byte(expectedUUID))
|
|
|
|
|
|
|
|
options := &protobuf.PushNotificationRegistration{
|
|
|
|
Version: 1,
|
|
|
|
AccessToken: expectedUUID,
|
2020-07-22 07:41:40 +00:00
|
|
|
DeviceToken: testDeviceToken,
|
2020-07-17 12:29:51 +00:00
|
|
|
InstallationId: s.installationID,
|
|
|
|
Enabled: true,
|
|
|
|
BlockedChatList: mutedChatListHashes,
|
|
|
|
}
|
|
|
|
|
|
|
|
actualMessage, err := s.client.buildPushNotificationRegistrationMessage(contactIDs, mutedChatList)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
s.Require().Equal(options, actualMessage)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *ClientSuite) TestBuildPushNotificationRegisterMessageAllowFromContactsOnly() {
|
|
|
|
mutedChatList := []string{"a", "b"}
|
|
|
|
|
|
|
|
// build chat lish hashes
|
|
|
|
var mutedChatListHashes [][]byte
|
|
|
|
for _, chatID := range mutedChatList {
|
|
|
|
mutedChatListHashes = append(mutedChatListHashes, common.Shake256([]byte(chatID)))
|
|
|
|
}
|
|
|
|
|
|
|
|
contactKey, err := crypto.GenerateKey()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
contactIDs := []*ecdsa.PublicKey{&contactKey.PublicKey}
|
|
|
|
|
|
|
|
// Set random generator for uuid
|
|
|
|
var seed int64 = 1
|
|
|
|
uuid.SetRand(rand.New(rand.NewSource(seed)))
|
|
|
|
|
|
|
|
// Get token
|
|
|
|
expectedUUID := uuid.New().String()
|
|
|
|
|
2020-06-30 07:50:59 +00:00
|
|
|
// set up reader
|
|
|
|
reader := bytes.NewReader([]byte(expectedUUID))
|
|
|
|
|
2020-07-10 13:26:06 +00:00
|
|
|
sharedKey, err := ecies.ImportECDSA(s.identity).GenerateShared(
|
2020-06-30 07:50:59 +00:00
|
|
|
ecies.ImportECDSAPublic(&contactKey.PublicKey),
|
|
|
|
accessTokenKeyLength,
|
|
|
|
accessTokenKeyLength,
|
|
|
|
)
|
2020-07-10 13:26:06 +00:00
|
|
|
s.Require().NoError(err)
|
2020-06-30 07:50:59 +00:00
|
|
|
// build encrypted token
|
|
|
|
encryptedToken, err := encryptAccessToken([]byte(expectedUUID), sharedKey, reader)
|
2020-07-10 13:26:06 +00:00
|
|
|
s.Require().NoError(err)
|
2020-06-30 07:50:59 +00:00
|
|
|
|
|
|
|
// Reset random generator
|
|
|
|
uuid.SetRand(rand.New(rand.NewSource(seed)))
|
|
|
|
|
2020-07-22 07:41:40 +00:00
|
|
|
s.client.config.AllowFromContactsOnly = true
|
|
|
|
s.client.deviceToken = testDeviceToken
|
2020-06-30 07:50:59 +00:00
|
|
|
// Set reader
|
2020-07-10 13:26:06 +00:00
|
|
|
s.client.reader = bytes.NewReader([]byte(expectedUUID))
|
2020-06-30 07:50:59 +00:00
|
|
|
|
2020-07-02 08:08:19 +00:00
|
|
|
options := &protobuf.PushNotificationRegistration{
|
2020-07-20 10:01:42 +00:00
|
|
|
Version: 1,
|
|
|
|
AccessToken: expectedUUID,
|
2020-07-22 07:41:40 +00:00
|
|
|
DeviceToken: testDeviceToken,
|
2020-07-20 10:01:42 +00:00
|
|
|
InstallationId: s.installationID,
|
|
|
|
AllowFromContactsOnly: true,
|
|
|
|
Enabled: true,
|
|
|
|
BlockedChatList: mutedChatListHashes,
|
2020-07-22 07:41:40 +00:00
|
|
|
AllowedKeyList: [][]byte{encryptedToken},
|
2020-06-30 07:50:59 +00:00
|
|
|
}
|
|
|
|
|
2020-07-10 13:26:06 +00:00
|
|
|
actualMessage, err := s.client.buildPushNotificationRegistrationMessage(contactIDs, mutedChatList)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
s.Require().Equal(options, actualMessage)
|
|
|
|
}
|
2020-06-30 07:50:59 +00:00
|
|
|
|
2020-07-22 07:41:40 +00:00
|
|
|
func (s *ClientSuite) TestHandleMessageScheduled() {
|
2020-07-10 13:26:06 +00:00
|
|
|
messageID := []byte("message-id")
|
|
|
|
chatID := "chat-id"
|
|
|
|
installationID1 := "1"
|
|
|
|
installationID2 := "2"
|
2020-07-22 07:41:40 +00:00
|
|
|
rawMessage := &common.RawMessage{
|
|
|
|
ID: types.EncodeHex(messageID),
|
|
|
|
SendPushNotification: true,
|
|
|
|
LocalChatID: chatID,
|
|
|
|
}
|
2020-07-10 13:26:06 +00:00
|
|
|
|
2020-07-22 07:41:40 +00:00
|
|
|
s.Require().NoError(s.client.handleMessageScheduled(rawMessage))
|
2020-07-10 13:26:06 +00:00
|
|
|
|
|
|
|
key1, err := crypto.GenerateKey()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
// First time, should notify
|
|
|
|
response, err := s.client.shouldNotifyOn(&key1.PublicKey, installationID1, messageID)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().True(response)
|
|
|
|
|
|
|
|
// Save notification
|
|
|
|
s.Require().NoError(s.client.notifiedOn(&key1.PublicKey, installationID1, messageID))
|
|
|
|
|
|
|
|
// Second time, should not notify
|
|
|
|
response, err = s.client.shouldNotifyOn(&key1.PublicKey, installationID1, messageID)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().False(response)
|
|
|
|
|
|
|
|
// Different installationID
|
|
|
|
response, err = s.client.shouldNotifyOn(&key1.PublicKey, installationID2, messageID)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().True(response)
|
|
|
|
|
|
|
|
key2, err := crypto.GenerateKey()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
// different key, should notify
|
|
|
|
response, err = s.client.shouldNotifyOn(&key2.PublicKey, installationID1, messageID)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().True(response)
|
|
|
|
|
|
|
|
// non tracked message id
|
|
|
|
response, err = s.client.shouldNotifyOn(&key1.PublicKey, installationID1, []byte("not-existing"))
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().False(response)
|
2020-06-30 07:50:59 +00:00
|
|
|
}
|
2020-07-15 07:23:31 +00:00
|
|
|
|
|
|
|
func (s *ClientSuite) TestShouldRefreshToken() {
|
|
|
|
key1, err := crypto.GenerateKey()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
key2, err := crypto.GenerateKey()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
key3, err := crypto.GenerateKey()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
key4, err := crypto.GenerateKey()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
// Contacts are added
|
|
|
|
s.Require().False(s.client.shouldRefreshToken([]*ecdsa.PublicKey{&key1.PublicKey, &key2.PublicKey}, []*ecdsa.PublicKey{&key1.PublicKey, &key2.PublicKey, &key3.PublicKey, &key4.PublicKey}))
|
|
|
|
|
|
|
|
// everything the same
|
|
|
|
s.Require().False(s.client.shouldRefreshToken([]*ecdsa.PublicKey{&key1.PublicKey, &key2.PublicKey}, []*ecdsa.PublicKey{&key2.PublicKey, &key1.PublicKey}))
|
|
|
|
|
|
|
|
// A contact is removed
|
|
|
|
s.Require().True(s.client.shouldRefreshToken([]*ecdsa.PublicKey{&key1.PublicKey, &key2.PublicKey}, []*ecdsa.PublicKey{&key2.PublicKey}))
|
|
|
|
|
|
|
|
}
|