2020-07-22 09:41:40 +02:00
|
|
|
package pushnotificationclient
|
2020-07-07 11:00:04 +02:00
|
|
|
|
|
|
|
import (
|
2020-07-15 10:22:43 +02:00
|
|
|
"crypto/ecdsa"
|
2020-07-07 11:00:04 +02:00
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
"testing"
|
2020-07-20 12:01:42 +02:00
|
|
|
"time"
|
2020-07-07 11:00:04 +02:00
|
|
|
|
2020-07-15 09:23:31 +02:00
|
|
|
"github.com/golang/protobuf/proto"
|
2020-07-07 11:00:04 +02:00
|
|
|
"github.com/stretchr/testify/suite"
|
|
|
|
|
|
|
|
"github.com/status-im/status-go/eth-node/crypto"
|
|
|
|
"github.com/status-im/status-go/protocol/common"
|
2020-07-15 09:23:31 +02:00
|
|
|
"github.com/status-im/status-go/protocol/protobuf"
|
2020-07-07 11:00:04 +02:00
|
|
|
"github.com/status-im/status-go/protocol/sqlite"
|
|
|
|
)
|
|
|
|
|
2020-07-22 09:41:40 +02:00
|
|
|
const (
|
|
|
|
testAccessToken = "token"
|
|
|
|
installationID1 = "installation-id-1"
|
|
|
|
installationID2 = "installation-id-2"
|
|
|
|
installationID3 = "installation-id-3"
|
|
|
|
)
|
|
|
|
|
2020-07-07 11:00:04 +02:00
|
|
|
func TestSQLitePersistenceSuite(t *testing.T) {
|
|
|
|
suite.Run(t, new(SQLitePersistenceSuite))
|
|
|
|
}
|
|
|
|
|
|
|
|
type SQLitePersistenceSuite struct {
|
|
|
|
suite.Suite
|
|
|
|
tmpFile *os.File
|
|
|
|
persistence *Persistence
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *SQLitePersistenceSuite) 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)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *SQLitePersistenceSuite) TearDownTest() {
|
|
|
|
_ = os.Remove(s.tmpFile.Name())
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *SQLitePersistenceSuite) TestSaveAndRetrieveServer() {
|
|
|
|
key, err := crypto.GenerateKey()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
server := &PushNotificationServer{
|
2020-07-10 09:45:40 +02:00
|
|
|
PublicKey: &key.PublicKey,
|
|
|
|
Registered: true,
|
|
|
|
RegisteredAt: 1,
|
2020-07-22 09:41:40 +02:00
|
|
|
AccessToken: testAccessToken,
|
2020-07-07 11:00:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
s.Require().NoError(s.persistence.UpsertServer(server))
|
|
|
|
|
|
|
|
retrievedServers, err := s.persistence.GetServers()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
s.Require().Len(retrievedServers, 1)
|
2020-07-10 09:45:40 +02:00
|
|
|
s.Require().True(retrievedServers[0].Registered)
|
|
|
|
s.Require().Equal(int64(1), retrievedServers[0].RegisteredAt)
|
|
|
|
s.Require().True(common.IsPubKeyEqual(retrievedServers[0].PublicKey, &key.PublicKey))
|
2020-07-22 09:41:40 +02:00
|
|
|
s.Require().Equal(testAccessToken, retrievedServers[0].AccessToken)
|
2020-07-07 11:00:04 +02:00
|
|
|
|
2020-07-10 09:45:40 +02:00
|
|
|
server.Registered = false
|
|
|
|
server.RegisteredAt = 2
|
2020-07-07 11:00:04 +02:00
|
|
|
|
|
|
|
s.Require().NoError(s.persistence.UpsertServer(server))
|
|
|
|
|
|
|
|
retrievedServers, err = s.persistence.GetServers()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
s.Require().Len(retrievedServers, 1)
|
2020-07-10 09:45:40 +02:00
|
|
|
s.Require().False(retrievedServers[0].Registered)
|
|
|
|
s.Require().Equal(int64(2), retrievedServers[0].RegisteredAt)
|
|
|
|
s.Require().True(common.IsPubKeyEqual(retrievedServers[0].PublicKey, &key.PublicKey))
|
2020-07-07 11:00:04 +02:00
|
|
|
}
|
2020-07-10 15:26:06 +02:00
|
|
|
|
|
|
|
func (s *SQLitePersistenceSuite) TestSaveAndRetrieveInfo() {
|
|
|
|
key1, err := crypto.GenerateKey()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
key2, err := crypto.GenerateKey()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
serverKey, err := crypto.GenerateKey()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
infos := []*PushNotificationInfo{
|
|
|
|
{
|
|
|
|
PublicKey: &key1.PublicKey,
|
|
|
|
ServerPublicKey: &serverKey.PublicKey,
|
|
|
|
RetrievedAt: 1,
|
2020-07-17 13:41:49 +02:00
|
|
|
Version: 1,
|
2020-07-22 09:41:40 +02:00
|
|
|
AccessToken: testAccessToken,
|
2020-07-10 15:26:06 +02:00
|
|
|
InstallationID: installationID1,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
PublicKey: &key1.PublicKey,
|
|
|
|
ServerPublicKey: &serverKey.PublicKey,
|
|
|
|
RetrievedAt: 1,
|
2020-07-17 13:41:49 +02:00
|
|
|
Version: 1,
|
2020-07-22 09:41:40 +02:00
|
|
|
AccessToken: testAccessToken,
|
2020-07-10 15:26:06 +02:00
|
|
|
InstallationID: installationID2,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
PublicKey: &key1.PublicKey,
|
|
|
|
ServerPublicKey: &serverKey.PublicKey,
|
|
|
|
RetrievedAt: 1,
|
2020-07-17 13:41:49 +02:00
|
|
|
Version: 1,
|
2020-07-22 09:41:40 +02:00
|
|
|
AccessToken: testAccessToken,
|
2020-07-10 15:26:06 +02:00
|
|
|
InstallationID: installationID3,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
PublicKey: &key2.PublicKey,
|
|
|
|
ServerPublicKey: &serverKey.PublicKey,
|
|
|
|
RetrievedAt: 1,
|
2020-07-17 13:41:49 +02:00
|
|
|
Version: 1,
|
2020-07-22 09:41:40 +02:00
|
|
|
AccessToken: testAccessToken,
|
2020-07-10 15:26:06 +02:00
|
|
|
InstallationID: installationID1,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
PublicKey: &key2.PublicKey,
|
|
|
|
ServerPublicKey: &serverKey.PublicKey,
|
|
|
|
RetrievedAt: 1,
|
2020-07-17 13:41:49 +02:00
|
|
|
Version: 1,
|
2020-07-22 09:41:40 +02:00
|
|
|
AccessToken: testAccessToken,
|
2020-07-10 15:26:06 +02:00
|
|
|
InstallationID: installationID2,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
PublicKey: &key2.PublicKey,
|
|
|
|
ServerPublicKey: &serverKey.PublicKey,
|
|
|
|
RetrievedAt: 1,
|
2020-07-17 13:41:49 +02:00
|
|
|
Version: 1,
|
2020-07-22 09:41:40 +02:00
|
|
|
AccessToken: testAccessToken,
|
2020-07-10 15:26:06 +02:00
|
|
|
InstallationID: installationID3,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
s.Require().NoError(s.persistence.SavePushNotificationInfo(infos))
|
|
|
|
|
|
|
|
retrievedInfos, err := s.persistence.GetPushNotificationInfo(&key1.PublicKey, []string{installationID1, installationID2})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
s.Require().Len(retrievedInfos, 2)
|
|
|
|
}
|
2020-07-15 09:23:31 +02:00
|
|
|
|
2020-07-17 13:41:49 +02:00
|
|
|
func (s *SQLitePersistenceSuite) TestSaveAndRetrieveInfoWithVersion() {
|
|
|
|
installationID := "installation-id-1"
|
|
|
|
key, err := crypto.GenerateKey()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
serverKey1, err := crypto.GenerateKey()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
serverKey2, err := crypto.GenerateKey()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
infos := []*PushNotificationInfo{
|
|
|
|
{
|
|
|
|
PublicKey: &key.PublicKey,
|
|
|
|
ServerPublicKey: &serverKey1.PublicKey,
|
|
|
|
RetrievedAt: 1,
|
|
|
|
Version: 1,
|
2020-07-22 09:41:40 +02:00
|
|
|
AccessToken: testAccessToken,
|
2020-07-17 13:41:49 +02:00
|
|
|
InstallationID: installationID,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
PublicKey: &key.PublicKey,
|
|
|
|
ServerPublicKey: &serverKey2.PublicKey,
|
|
|
|
RetrievedAt: 1,
|
|
|
|
Version: 1,
|
2020-07-22 09:41:40 +02:00
|
|
|
AccessToken: testAccessToken,
|
2020-07-17 13:41:49 +02:00
|
|
|
InstallationID: installationID,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
s.Require().NoError(s.persistence.SavePushNotificationInfo(infos))
|
|
|
|
|
|
|
|
retrievedInfos, err := s.persistence.GetPushNotificationInfo(&key.PublicKey, []string{installationID})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
// We should retrieve both
|
|
|
|
s.Require().Len(retrievedInfos, 2)
|
|
|
|
s.Require().Equal(uint64(1), retrievedInfos[0].Version)
|
|
|
|
|
|
|
|
// Bump version
|
|
|
|
infos[0].Version = 2
|
|
|
|
|
|
|
|
s.Require().NoError(s.persistence.SavePushNotificationInfo(infos))
|
|
|
|
|
|
|
|
retrievedInfos, err = s.persistence.GetPushNotificationInfo(&key.PublicKey, []string{installationID})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
// Only one should be retrieved now
|
|
|
|
s.Require().Len(retrievedInfos, 1)
|
|
|
|
s.Require().Equal(uint64(2), retrievedInfos[0].Version)
|
|
|
|
|
|
|
|
// Lower version
|
|
|
|
infos[0].Version = 1
|
|
|
|
|
|
|
|
s.Require().NoError(s.persistence.SavePushNotificationInfo(infos))
|
|
|
|
|
|
|
|
retrievedInfos, err = s.persistence.GetPushNotificationInfo(&key.PublicKey, []string{installationID})
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
s.Require().Len(retrievedInfos, 1)
|
|
|
|
s.Require().Equal(uint64(2), retrievedInfos[0].Version)
|
|
|
|
}
|
|
|
|
|
2020-07-20 12:01:42 +02:00
|
|
|
func (s *SQLitePersistenceSuite) TestNotifiedOnAndUpdateNotificationResponse() {
|
|
|
|
key, err := crypto.GenerateKey()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
installationID := "installation-id"
|
|
|
|
messageID := []byte("message-id")
|
|
|
|
|
|
|
|
sentNotification := &SentNotification{
|
|
|
|
PublicKey: &key.PublicKey,
|
|
|
|
InstallationID: installationID,
|
|
|
|
MessageID: messageID,
|
2020-07-20 14:46:15 +02:00
|
|
|
LastTriedAt: time.Now().Unix(),
|
2020-07-20 12:01:42 +02:00
|
|
|
}
|
|
|
|
|
2020-07-20 14:46:15 +02:00
|
|
|
s.Require().NoError(s.persistence.UpsertSentNotification(sentNotification))
|
2020-07-20 12:01:42 +02:00
|
|
|
|
|
|
|
retrievedNotification, err := s.persistence.GetSentNotification(sentNotification.HashedPublicKey(), installationID, messageID)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Equal(sentNotification, retrievedNotification)
|
|
|
|
|
2020-07-20 14:46:15 +02:00
|
|
|
retriableNotifications, err := s.persistence.GetRetriablePushNotifications()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(retriableNotifications, 0)
|
|
|
|
|
2020-07-20 12:01:42 +02:00
|
|
|
response := &protobuf.PushNotificationReport{
|
|
|
|
Success: false,
|
|
|
|
Error: protobuf.PushNotificationReport_WRONG_TOKEN,
|
|
|
|
PublicKey: sentNotification.HashedPublicKey(),
|
|
|
|
InstallationId: installationID,
|
|
|
|
}
|
|
|
|
|
|
|
|
s.Require().NoError(s.persistence.UpdateNotificationResponse(messageID, response))
|
2020-07-20 14:46:15 +02:00
|
|
|
// This notification should be retriable
|
|
|
|
retriableNotifications, err = s.persistence.GetRetriablePushNotifications()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(retriableNotifications, 1)
|
2020-07-20 12:01:42 +02:00
|
|
|
|
|
|
|
sentNotification.Error = protobuf.PushNotificationReport_WRONG_TOKEN
|
|
|
|
|
|
|
|
retrievedNotification, err = s.persistence.GetSentNotification(sentNotification.HashedPublicKey(), installationID, messageID)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Equal(sentNotification, retrievedNotification)
|
|
|
|
|
|
|
|
// Update with a successful notification
|
|
|
|
response = &protobuf.PushNotificationReport{
|
|
|
|
Success: true,
|
|
|
|
PublicKey: sentNotification.HashedPublicKey(),
|
|
|
|
InstallationId: installationID,
|
|
|
|
}
|
|
|
|
|
|
|
|
s.Require().NoError(s.persistence.UpdateNotificationResponse(messageID, response))
|
|
|
|
|
|
|
|
sentNotification.Success = true
|
|
|
|
sentNotification.Error = protobuf.PushNotificationReport_UNKNOWN_ERROR_TYPE
|
|
|
|
|
|
|
|
retrievedNotification, err = s.persistence.GetSentNotification(sentNotification.HashedPublicKey(), installationID, messageID)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Equal(sentNotification, retrievedNotification)
|
|
|
|
|
2020-07-20 14:46:15 +02:00
|
|
|
// This notification should not be retriable
|
|
|
|
retriableNotifications, err = s.persistence.GetRetriablePushNotifications()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Len(retriableNotifications, 0)
|
|
|
|
|
2020-07-20 12:01:42 +02:00
|
|
|
// Update with a unsuccessful notification, it should be ignored
|
|
|
|
response = &protobuf.PushNotificationReport{
|
|
|
|
Success: false,
|
|
|
|
Error: protobuf.PushNotificationReport_WRONG_TOKEN,
|
|
|
|
PublicKey: sentNotification.HashedPublicKey(),
|
|
|
|
InstallationId: installationID,
|
|
|
|
}
|
|
|
|
|
|
|
|
s.Require().NoError(s.persistence.UpdateNotificationResponse(messageID, response))
|
|
|
|
|
|
|
|
sentNotification.Success = true
|
|
|
|
sentNotification.Error = protobuf.PushNotificationReport_UNKNOWN_ERROR_TYPE
|
|
|
|
|
|
|
|
retrievedNotification, err = s.persistence.GetSentNotification(sentNotification.HashedPublicKey(), installationID, messageID)
|
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Equal(sentNotification, retrievedNotification)
|
|
|
|
}
|
|
|
|
|
2020-07-15 09:23:31 +02:00
|
|
|
func (s *SQLitePersistenceSuite) TestSaveAndRetrieveRegistration() {
|
|
|
|
// Try with nil first
|
2020-07-15 10:22:43 +02:00
|
|
|
retrievedRegistration, retrievedContactIDs, err := s.persistence.GetLastPushNotificationRegistration()
|
2020-07-15 09:23:31 +02:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().Nil(retrievedRegistration)
|
2020-07-15 10:22:43 +02:00
|
|
|
s.Require().Nil(retrievedContactIDs)
|
2020-07-15 09:23:31 +02:00
|
|
|
|
|
|
|
// Save & retrieve registration
|
|
|
|
registration := &protobuf.PushNotificationRegistration{
|
|
|
|
AccessToken: "test",
|
|
|
|
Version: 3,
|
|
|
|
}
|
|
|
|
|
2020-07-15 10:22:43 +02:00
|
|
|
key1, err := crypto.GenerateKey()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
key2, err := crypto.GenerateKey()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
key3, err := crypto.GenerateKey()
|
|
|
|
s.Require().NoError(err)
|
|
|
|
|
|
|
|
publicKeys := []*ecdsa.PublicKey{&key1.PublicKey, &key2.PublicKey}
|
|
|
|
|
|
|
|
s.Require().NoError(s.persistence.SaveLastPushNotificationRegistration(registration, publicKeys))
|
|
|
|
retrievedRegistration, retrievedContactIDs, err = s.persistence.GetLastPushNotificationRegistration()
|
2020-07-15 09:23:31 +02:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().True(proto.Equal(registration, retrievedRegistration))
|
2020-07-15 10:22:43 +02:00
|
|
|
s.Require().Equal(publicKeys, retrievedContactIDs)
|
2020-07-15 09:23:31 +02:00
|
|
|
|
|
|
|
// Override and retrieve
|
|
|
|
|
|
|
|
registration.Version = 5
|
2020-07-15 10:22:43 +02:00
|
|
|
publicKeys = append(publicKeys, &key3.PublicKey)
|
|
|
|
s.Require().NoError(s.persistence.SaveLastPushNotificationRegistration(registration, publicKeys))
|
|
|
|
retrievedRegistration, retrievedContactIDs, err = s.persistence.GetLastPushNotificationRegistration()
|
2020-07-15 09:23:31 +02:00
|
|
|
s.Require().NoError(err)
|
|
|
|
s.Require().True(proto.Equal(registration, retrievedRegistration))
|
2020-07-15 10:22:43 +02:00
|
|
|
s.Require().Equal(publicKeys, retrievedContactIDs)
|
2020-07-15 09:23:31 +02:00
|
|
|
}
|