2023-01-06 20:21:14 +08:00
package protocol
import (
"context"
2023-02-28 20:32:45 +08:00
"errors"
2023-01-06 20:21:14 +08:00
"github.com/golang/protobuf/proto"
"go.uber.org/zap"
"github.com/status-im/status-go/protocol/common"
2023-02-28 20:32:45 +08:00
"github.com/status-im/status-go/protocol/encryption/multidevice"
"github.com/status-im/status-go/protocol/protobuf"
2023-01-06 20:21:14 +08:00
)
type RawMessageHandler func ( ctx context . Context , rawMessage common . RawMessage ) ( common . RawMessage , error )
func ( m * Messenger ) HandleSyncRawMessages ( rawMessages [ ] * protobuf . RawMessage ) error {
state := m . buildMessageState ( )
for _ , rawMessage := range rawMessages {
switch rawMessage . GetMessageType ( ) {
case protobuf . ApplicationMetadataMessage_CONTACT_UPDATE :
var message protobuf . ContactUpdate
err := proto . Unmarshal ( rawMessage . GetPayload ( ) , & message )
if err != nil {
return err
}
2023-03-28 11:45:54 +08:00
publicKey , err := common . HexToPubkey ( message . PublicKey )
if err != nil {
return err
}
var contact * Contact
if c , ok := state . AllContacts . Load ( message . PublicKey ) ; ok {
contact = c
} else {
c , err := buildContact ( message . PublicKey , publicKey )
if err != nil {
m . logger . Info ( "failed to build contact" , zap . Error ( err ) )
continue
}
contact = c
state . AllContacts . Store ( message . PublicKey , contact )
}
currentMessageState := & CurrentMessageState {
2023-08-18 12:39:59 +01:00
Message : & protobuf . ChatMessage {
2023-03-28 11:45:54 +08:00
Clock : message . Clock ,
} ,
MessageID : " " , // make it not empty to bypass this validation: https://github.com/status-im/status-go/blob/7cd7430d3141b08f7c455d7918f4160ea8fd0559/protocol/messenger_handler.go#L325
WhisperTimestamp : message . Clock ,
PublicKey : publicKey ,
Contact : contact ,
}
state . CurrentMessageState = currentMessageState
2023-08-18 12:39:59 +01:00
err = m . HandleContactUpdate ( state , & message , nil )
2023-01-06 20:21:14 +08:00
if err != nil {
m . logger . Warn ( "failed to HandleContactUpdate when HandleSyncRawMessages" , zap . Error ( err ) )
continue
}
2023-10-17 09:24:15 -04:00
case protobuf . ApplicationMetadataMessage_SYNC_CHAT :
var message protobuf . SyncChat
2023-01-06 20:21:14 +08:00
err := proto . Unmarshal ( rawMessage . GetPayload ( ) , & message )
if err != nil {
return err
}
2023-10-17 09:24:15 -04:00
err = m . HandleSyncChat ( state , & message , nil )
2023-08-18 12:39:59 +01:00
if err != nil {
2023-10-17 09:24:15 -04:00
m . logger . Error ( "error createChat when HandleSyncRawMessages" , zap . Error ( err ) )
2023-08-18 12:39:59 +01:00
continue
2023-01-06 20:21:14 +08:00
}
case protobuf . ApplicationMetadataMessage_SYNC_CHAT_REMOVED :
var message protobuf . SyncChatRemoved
err := proto . Unmarshal ( rawMessage . GetPayload ( ) , & message )
if err != nil {
return err
}
2023-08-18 12:39:59 +01:00
err = m . HandleSyncChatRemoved ( state , & message , nil )
2023-01-06 20:21:14 +08:00
if err != nil {
m . logger . Error ( "failed to HandleSyncChatRemoved when HandleSyncRawMessages" , zap . Error ( err ) )
continue
}
case protobuf . ApplicationMetadataMessage_SYNC_CHAT_MESSAGES_READ :
var message protobuf . SyncChatMessagesRead
err := proto . Unmarshal ( rawMessage . GetPayload ( ) , & message )
if err != nil {
return err
}
2023-08-18 12:39:59 +01:00
err = m . HandleSyncChatMessagesRead ( state , & message , nil )
2023-01-06 20:21:14 +08:00
if err != nil {
m . logger . Error ( "failed to HandleSyncChatMessagesRead when HandleSyncRawMessages" , zap . Error ( err ) )
continue
}
case protobuf . ApplicationMetadataMessage_SYNC_CLEAR_HISTORY :
var message protobuf . SyncClearHistory
err := proto . Unmarshal ( rawMessage . GetPayload ( ) , & message )
if err != nil {
return err
}
2023-08-18 12:39:59 +01:00
err = m . HandleSyncClearHistory ( state , & message , nil )
2023-01-06 20:21:14 +08:00
if err != nil {
m . logger . Error ( "failed to handleSyncClearHistory when HandleSyncRawMessages" , zap . Error ( err ) )
continue
}
2023-08-18 12:39:59 +01:00
case protobuf . ApplicationMetadataMessage_SYNC_INSTALLATION_CONTACT_V2 :
2023-01-06 20:21:14 +08:00
var message protobuf . SyncInstallationContactV2
err := proto . Unmarshal ( rawMessage . GetPayload ( ) , & message )
if err != nil {
return err
}
2023-08-18 12:39:59 +01:00
err = m . HandleSyncInstallationContactV2 ( state , & message , nil )
2023-01-06 20:21:14 +08:00
if err != nil {
m . logger . Error ( "failed to HandleSyncInstallationContact when HandleSyncRawMessages" , zap . Error ( err ) )
continue
}
case protobuf . ApplicationMetadataMessage_SYNC_INSTALLATION_COMMUNITY :
2023-08-18 12:39:59 +01:00
var message protobuf . SyncInstallationCommunity
2023-01-06 20:21:14 +08:00
err := proto . Unmarshal ( rawMessage . GetPayload ( ) , & message )
if err != nil {
return err
}
2024-06-18 18:18:21 +02:00
err = m . handleSyncInstallationCommunity ( state , & message )
2023-01-06 20:21:14 +08:00
if err != nil {
m . logger . Error ( "failed to handleSyncCommunity when HandleSyncRawMessages" , zap . Error ( err ) )
continue
}
case protobuf . ApplicationMetadataMessage_SYNC_BOOKMARK :
var message protobuf . SyncBookmark
err := proto . Unmarshal ( rawMessage . GetPayload ( ) , & message )
if err != nil {
return err
}
2023-08-18 12:39:59 +01:00
err = m . HandleSyncBookmark ( state , & message , nil )
2023-01-06 20:21:14 +08:00
if err != nil {
m . logger . Error ( "failed to handleSyncBookmark when HandleSyncRawMessages" , zap . Error ( err ) )
continue
}
case protobuf . ApplicationMetadataMessage_SYNC_TRUSTED_USER :
var message protobuf . SyncTrustedUser
err := proto . Unmarshal ( rawMessage . GetPayload ( ) , & message )
if err != nil {
return err
}
2023-08-18 12:39:59 +01:00
err = m . HandleSyncTrustedUser ( state , & message , nil )
2023-01-06 20:21:14 +08:00
if err != nil {
m . logger . Error ( "failed to handleSyncTrustedUser when HandleSyncRawMessages" , zap . Error ( err ) )
continue
}
case protobuf . ApplicationMetadataMessage_SYNC_VERIFICATION_REQUEST :
var message protobuf . SyncVerificationRequest
err := proto . Unmarshal ( rawMessage . GetPayload ( ) , & message )
if err != nil {
return err
}
2023-08-18 12:39:59 +01:00
err = m . HandleSyncVerificationRequest ( state , & message , nil )
2023-01-06 20:21:14 +08:00
if err != nil {
m . logger . Error ( "failed to handleSyncVerificationRequest when HandleSyncRawMessages" , zap . Error ( err ) )
continue
}
case protobuf . ApplicationMetadataMessage_SYNC_SETTING :
var message protobuf . SyncSetting
err := proto . Unmarshal ( rawMessage . GetPayload ( ) , & message )
if err != nil {
return err
}
2023-08-18 12:39:59 +01:00
err = m . HandleSyncSetting ( state , & message , nil )
2023-01-06 20:21:14 +08:00
if err != nil {
m . logger . Error ( "failed to handleSyncSetting when HandleSyncRawMessages" , zap . Error ( err ) )
continue
}
2023-08-18 12:39:59 +01:00
case protobuf . ApplicationMetadataMessage_SYNC_PROFILE_PICTURES :
2023-01-06 20:21:14 +08:00
var message protobuf . SyncProfilePictures
err := proto . Unmarshal ( rawMessage . GetPayload ( ) , & message )
if err != nil {
return err
}
2023-08-18 12:39:59 +01:00
err = m . HandleSyncProfilePictures ( state , & message , nil )
2023-01-06 20:21:14 +08:00
if err != nil {
m . logger . Error ( "failed to HandleSyncProfilePictures when HandleSyncRawMessages" , zap . Error ( err ) )
continue
}
case protobuf . ApplicationMetadataMessage_SYNC_CONTACT_REQUEST_DECISION :
var message protobuf . SyncContactRequestDecision
err := proto . Unmarshal ( rawMessage . GetPayload ( ) , & message )
if err != nil {
return err
}
2023-08-18 12:39:59 +01:00
err = m . HandleSyncContactRequestDecision ( state , & message , nil )
2023-01-06 20:21:14 +08:00
if err != nil {
m . logger . Error ( "failed to HandleSyncContactRequestDecision when HandleSyncRawMessages" , zap . Error ( err ) )
continue
}
2023-05-16 12:48:00 +02:00
case protobuf . ApplicationMetadataMessage_SYNC_ACCOUNT :
var message protobuf . SyncAccount
2023-01-06 20:21:14 +08:00
err := proto . Unmarshal ( rawMessage . GetPayload ( ) , & message )
if err != nil {
return err
}
2023-08-18 12:39:59 +01:00
err = m . HandleSyncAccount ( state , & message , nil )
2023-01-06 20:21:14 +08:00
if err != nil {
2023-06-28 21:45:36 +02:00
m . logger . Error ( "failed to HandleSyncWatchOnlyAccount when HandleSyncRawMessages" , zap . Error ( err ) )
2023-01-06 20:21:14 +08:00
continue
}
2023-06-28 21:45:36 +02:00
case protobuf . ApplicationMetadataMessage_SYNC_KEYPAIR :
var message protobuf . SyncKeypair
2023-01-06 20:21:14 +08:00
err := proto . Unmarshal ( rawMessage . GetPayload ( ) , & message )
if err != nil {
return err
}
2023-08-18 12:39:59 +01:00
err = m . handleSyncKeypairInternal ( state , & message , true )
2023-01-06 20:21:14 +08:00
if err != nil {
2023-06-28 21:45:36 +02:00
m . logger . Error ( "failed to HandleSyncKeypair when HandleSyncRawMessages" , zap . Error ( err ) )
2023-01-06 20:21:14 +08:00
continue
}
2023-07-16 13:11:48 +02:00
case protobuf . ApplicationMetadataMessage_SYNC_ACCOUNTS_POSITIONS :
var message protobuf . SyncAccountsPositions
err := proto . Unmarshal ( rawMessage . GetPayload ( ) , & message )
if err != nil {
return err
}
2023-08-18 12:39:59 +01:00
err = m . HandleSyncAccountsPositions ( state , & message , nil )
2023-07-16 13:11:48 +02:00
if err != nil {
m . logger . Error ( "failed to HandleSyncAccountsPositions when HandleSyncRawMessages" , zap . Error ( err ) )
continue
}
2023-12-04 11:18:05 +01:00
case protobuf . ApplicationMetadataMessage_SYNC_TOKEN_PREFERENCES :
var message protobuf . SyncTokenPreferences
err := proto . Unmarshal ( rawMessage . GetPayload ( ) , & message )
if err != nil {
return err
}
err = m . HandleSyncTokenPreferences ( state , & message , nil )
if err != nil {
m . logger . Error ( "failed to HandleSyncTokenPreferences when HandleSyncRawMessages" , zap . Error ( err ) )
continue
}
2024-01-17 10:12:49 -03:00
case protobuf . ApplicationMetadataMessage_SYNC_COLLECTIBLE_PREFERENCES :
var message protobuf . SyncCollectiblePreferences
err := proto . Unmarshal ( rawMessage . GetPayload ( ) , & message )
if err != nil {
return err
}
err = m . HandleSyncCollectiblePreferences ( state , & message , nil )
if err != nil {
m . logger . Error ( "failed to HandleSyncCollectiblePreferences when HandleSyncRawMessages" , zap . Error ( err ) )
continue
}
2023-05-16 12:48:00 +02:00
case protobuf . ApplicationMetadataMessage_SYNC_SAVED_ADDRESS :
var message protobuf . SyncSavedAddress
2023-02-21 13:35:26 +01:00
err := proto . Unmarshal ( rawMessage . GetPayload ( ) , & message )
if err != nil {
return err
}
2023-08-18 12:39:59 +01:00
err = m . HandleSyncSavedAddress ( state , & message , nil )
2023-02-21 13:35:26 +01:00
if err != nil {
2023-05-16 12:48:00 +02:00
m . logger . Error ( "failed to handleSyncSavedAddress when HandleSyncRawMessages" , zap . Error ( err ) )
2023-04-20 06:59:09 +08:00
continue
}
2023-04-26 23:37:18 +08:00
case protobuf . ApplicationMetadataMessage_SYNC_ENS_USERNAME_DETAIL :
var message protobuf . SyncEnsUsernameDetail
err := proto . Unmarshal ( rawMessage . GetPayload ( ) , & message )
if err != nil {
return err
}
2023-08-18 12:39:59 +01:00
err = m . HandleSyncEnsUsernameDetail ( state , & message , nil )
2023-04-26 23:37:18 +08:00
if err != nil {
m . logger . Error ( "failed to handleSyncEnsUsernameDetail when HandleSyncRawMessages" , zap . Error ( err ) )
2023-05-12 16:31:34 +08:00
continue
}
case protobuf . ApplicationMetadataMessage_SYNC_DELETE_FOR_ME_MESSAGE :
2023-08-18 12:39:59 +01:00
var message protobuf . SyncDeleteForMeMessage
2023-05-12 16:31:34 +08:00
err := proto . Unmarshal ( rawMessage . GetPayload ( ) , & message )
if err != nil {
return err
}
2023-08-18 12:39:59 +01:00
err = m . HandleSyncDeleteForMeMessage ( state , & message , nil )
2023-05-12 16:31:34 +08:00
if err != nil {
m . logger . Error ( "failed to HandleDeleteForMeMessage when HandleSyncRawMessages" , zap . Error ( err ) )
2023-02-21 13:35:26 +01:00
continue
}
2023-08-18 12:39:59 +01:00
case protobuf . ApplicationMetadataMessage_SYNC_PAIR_INSTALLATION :
var message protobuf . SyncPairInstallation
2023-02-28 20:32:45 +08:00
err := proto . Unmarshal ( rawMessage . GetPayload ( ) , & message )
if err != nil {
return err
}
identity := m . myHexIdentity ( )
installations := [ ] * multidevice . Installation {
{
Identity : identity ,
ID : message . InstallationId ,
Version : message . Version ,
Enabled : true ,
Timestamp : int64 ( message . Clock ) ,
InstallationMetadata : & multidevice . InstallationMetadata {
DeviceType : message . DeviceType ,
Name : message . Name ,
} ,
} }
m . handleInstallations ( installations )
2023-08-18 12:39:59 +01:00
// set WhisperTimestamp to pass the validation in HandleSyncPairInstallation
2023-02-28 20:32:45 +08:00
state . CurrentMessageState = & CurrentMessageState { WhisperTimestamp : message . Clock }
2023-08-18 12:39:59 +01:00
err = m . HandleSyncPairInstallation ( state , & message , nil )
2023-02-28 20:32:45 +08:00
if err != nil {
return err
}
multidevice := m . encryptor . GetMultiDevice ( )
if multidevice == nil {
return errors . New ( "multidevice is nil" )
}
_ , err = multidevice . AddInstallations ( m . IdentityPublicKeyCompressed ( ) , int64 ( message . GetClock ( ) ) , installations , true )
if err != nil {
return err
}
// if receiver already logged in before local pairing, we need force enable the installation,
// AddInstallations won't make sure enable it, e.g. installation maybe already exist in db but not enabled yet
err = m . EnableInstallation ( message . InstallationId )
if err != nil {
return err
}
2024-02-17 18:07:20 +00:00
case protobuf . ApplicationMetadataMessage_SYNC_PROFILE_SHOWCASE_PREFERENCES :
var message protobuf . SyncProfileShowcasePreferences
err := proto . Unmarshal ( rawMessage . GetPayload ( ) , & message )
if err != nil {
return err
}
2024-02-26 16:53:40 +03:00
_ , err = m . saveProfileShowcasePreferencesProto ( & message , false )
2024-02-17 18:07:20 +00:00
if err != nil {
return err
}
2023-01-06 20:21:14 +08:00
}
}
response , err := m . saveDataAndPrepareResponse ( state )
if err != nil {
return err
}
2023-11-25 23:24:20 +00:00
m . PublishMessengerResponse ( response )
2023-01-06 20:21:14 +08:00
return nil
}