package protocol import ( "crypto/ecdsa" "encoding/json" "reflect" "github.com/golang/protobuf/proto" "github.com/jinzhu/copier" "github.com/pkg/errors" "github.com/ethereum/go-ethereum/common/hexutil" "github.com/ethereum/go-ethereum/log" "github.com/status-im/status-go/eth-node/crypto" "github.com/status-im/status-go/eth-node/types" "github.com/status-im/status-go/protocol/encryption" "github.com/status-im/status-go/protocol/encryption/multidevice" "github.com/status-im/status-go/protocol/encryption/sharedsecret" "github.com/status-im/status-go/protocol/protobuf" ) type StatusMessageT int // StatusMessage is any Status Protocol message. type StatusMessage struct { // TransportMessage is the parsed message received from the transport layer, i.e the input TransportMessage *types.Message `json:"transportMessage"` // Type is the type of application message contained Type protobuf.ApplicationMetadataMessage_Type `json:"-"` // ParsedMessage is the parsed message by the application layer, i.e the output ParsedMessage *reflect.Value `json:"-"` // TransportPayload is the payload as received from the transport layer TransportPayload []byte `json:"-"` // DecryptedPayload is the payload after having been processed by the encryption layer DecryptedPayload []byte `json:"decryptedPayload"` // UnwrappedPayload is the payload after having been unwrapped from the applicaition metadata layer UnwrappedPayload []byte `json:"unwrappedPayload"` // ID is the canonical ID of the message ID types.HexBytes `json:"id"` // Hash is the transport layer hash Hash []byte `json:"-"` // Dst is the targeted public key Dst *ecdsa.PublicKey // TransportLayerSigPubKey contains the public key provided by the transport layer TransportLayerSigPubKey *ecdsa.PublicKey `json:"-"` // ApplicationMetadataLayerPubKey contains the public key provided by the application metadata layer ApplicationMetadataLayerSigPubKey *ecdsa.PublicKey `json:"-"` // Installations is the new installations returned by the encryption layer Installations []*multidevice.Installation // SharedSecret is the shared secret returned by the encryption layer SharedSecrets []*sharedsecret.Secret // HashRatchetInfo is the information about a new hash ratchet group/key pair HashRatchetInfo []*encryption.HashRatchetInfo } // Temporary JSON marshaling for those messages that are not yet processed // by the go code func (m *StatusMessage) MarshalJSON() ([]byte, error) { item := struct { ID types.HexBytes `json:"id"` Payload string `json:"payload"` From types.HexBytes `json:"from"` Timestamp uint32 `json:"timestamp"` }{ ID: m.ID, Payload: string(m.UnwrappedPayload), Timestamp: m.TransportMessage.Timestamp, From: m.TransportMessage.Sig, } return json.Marshal(item) } // SigPubKey returns the most important signature, from the application layer to transport func (m *StatusMessage) SigPubKey() *ecdsa.PublicKey { if m.ApplicationMetadataLayerSigPubKey != nil { return m.ApplicationMetadataLayerSigPubKey } return m.TransportLayerSigPubKey } func (m *StatusMessage) Clone() (*StatusMessage, error) { copy := &StatusMessage{} err := copier.Copy(©, m) return copy, err } func (m *StatusMessage) HandleTransport(shhMessage *types.Message) error { publicKey, err := crypto.UnmarshalPubkey(shhMessage.Sig) if err != nil { return errors.Wrap(err, "failed to get signature") } m.TransportMessage = shhMessage m.Hash = shhMessage.Hash m.TransportLayerSigPubKey = publicKey m.TransportPayload = shhMessage.Payload if shhMessage.Dst != nil { publicKey, err := crypto.UnmarshalPubkey(shhMessage.Dst) if err != nil { return err } m.Dst = publicKey } return nil } func (m *StatusMessage) HandleEncryption(myKey *ecdsa.PrivateKey, senderKey *ecdsa.PublicKey, enc *encryption.Protocol, skipNegotiation bool) error { // As we handle non-encrypted messages, we make sure that DecryptPayload // is set regardless of whether this step is successful m.DecryptedPayload = m.TransportPayload // Nothing to do if skipNegotiation { return nil } var protocolMessage encryption.ProtocolMessage err := proto.Unmarshal(m.TransportPayload, &protocolMessage) if err != nil { return errors.Wrap(err, "failed to unmarshal ProtocolMessage") } response, err := enc.HandleMessage( myKey, senderKey, &protocolMessage, m.Hash, ) if err == encryption.ErrHashRatchetGroupIDNotFound { if response != nil { m.HashRatchetInfo = response.HashRatchetInfo } return err } if err != nil { return errors.Wrap(err, "failed to handle Encryption message") } m.DecryptedPayload = response.DecryptedMessage m.Installations = response.Installations m.SharedSecrets = response.SharedSecrets m.HashRatchetInfo = response.HashRatchetInfo return nil } func (m *StatusMessage) HandleApplicationMetadata() error { message, err := protobuf.Unmarshal(m.DecryptedPayload) if err != nil { return err } recoveredKey, err := message.RecoverKey() if err != nil { return err } m.ApplicationMetadataLayerSigPubKey = recoveredKey // Calculate ID using the wrapped record m.ID = MessageID(recoveredKey, m.DecryptedPayload) log.Debug("calculated ID for envelope", "envelopeHash", hexutil.Encode(m.Hash), "messageId", hexutil.Encode(m.ID)) m.UnwrappedPayload = message.Payload m.Type = message.Type return nil } func (m *StatusMessage) HandleApplication() error { switch m.Type { case protobuf.ApplicationMetadataMessage_CHAT_MESSAGE: return m.unmarshalProtobufData(new(protobuf.ChatMessage)) case protobuf.ApplicationMetadataMessage_MEMBERSHIP_UPDATE_MESSAGE: return m.unmarshalProtobufData(new(protobuf.MembershipUpdateMessage)) case protobuf.ApplicationMetadataMessage_ACCEPT_REQUEST_ADDRESS_FOR_TRANSACTION: return m.unmarshalProtobufData(new(protobuf.AcceptRequestAddressForTransaction)) case protobuf.ApplicationMetadataMessage_SEND_TRANSACTION: return m.unmarshalProtobufData(new(protobuf.SendTransaction)) case protobuf.ApplicationMetadataMessage_REQUEST_TRANSACTION: return m.unmarshalProtobufData(new(protobuf.RequestTransaction)) case protobuf.ApplicationMetadataMessage_DECLINE_REQUEST_ADDRESS_FOR_TRANSACTION: return m.unmarshalProtobufData(new(protobuf.DeclineRequestAddressForTransaction)) case protobuf.ApplicationMetadataMessage_DECLINE_REQUEST_TRANSACTION: return m.unmarshalProtobufData(new(protobuf.DeclineRequestTransaction)) case protobuf.ApplicationMetadataMessage_REQUEST_ADDRESS_FOR_TRANSACTION: return m.unmarshalProtobufData(new(protobuf.RequestAddressForTransaction)) case protobuf.ApplicationMetadataMessage_CONTACT_UPDATE: return m.unmarshalProtobufData(new(protobuf.ContactUpdate)) case protobuf.ApplicationMetadataMessage_PIN_MESSAGE: return m.unmarshalProtobufData(new(protobuf.PinMessage)) case protobuf.ApplicationMetadataMessage_SYNC_INSTALLATION: return m.unmarshalProtobufData(new(protobuf.SyncInstallation)) case protobuf.ApplicationMetadataMessage_SYNC_INSTALLATION_CONTACT: log.Debug("Sync installation contact") return m.unmarshalProtobufData(new(protobuf.SyncInstallationContactV2)) case protobuf.ApplicationMetadataMessage_SYNC_INSTALLATION_PUBLIC_CHAT: return m.unmarshalProtobufData(new(protobuf.SyncInstallationPublicChat)) case protobuf.ApplicationMetadataMessage_SYNC_INSTALLATION_ACCOUNT: return m.unmarshalProtobufData(new(protobuf.SyncInstallationAccount)) case protobuf.ApplicationMetadataMessage_SYNC_PROFILE_PICTURE: return m.unmarshalProtobufData(new(protobuf.SyncProfilePictures)) case protobuf.ApplicationMetadataMessage_PAIR_INSTALLATION: return m.unmarshalProtobufData(new(protobuf.PairInstallation)) case protobuf.ApplicationMetadataMessage_SYNC_INSTALLATION_COMMUNITY: return m.unmarshalProtobufData(new(protobuf.SyncCommunity)) case protobuf.ApplicationMetadataMessage_CONTACT_CODE_ADVERTISEMENT: return m.unmarshalProtobufData(new(protobuf.ContactCodeAdvertisement)) case protobuf.ApplicationMetadataMessage_PUSH_NOTIFICATION_REQUEST: return m.unmarshalProtobufData(new(protobuf.PushNotificationRequest)) case protobuf.ApplicationMetadataMessage_PUSH_NOTIFICATION_REGISTRATION_RESPONSE: return m.unmarshalProtobufData(new(protobuf.PushNotificationRegistrationResponse)) case protobuf.ApplicationMetadataMessage_PUSH_NOTIFICATION_QUERY: return m.unmarshalProtobufData(new(protobuf.PushNotificationQuery)) case protobuf.ApplicationMetadataMessage_PUSH_NOTIFICATION_QUERY_RESPONSE: return m.unmarshalProtobufData(new(protobuf.PushNotificationQueryResponse)) case protobuf.ApplicationMetadataMessage_PUSH_NOTIFICATION_RESPONSE: return m.unmarshalProtobufData(new(protobuf.PushNotificationResponse)) case protobuf.ApplicationMetadataMessage_EMOJI_REACTION: return m.unmarshalProtobufData(new(protobuf.EmojiReaction)) case protobuf.ApplicationMetadataMessage_GROUP_CHAT_INVITATION: return m.unmarshalProtobufData(new(protobuf.GroupChatInvitation)) case protobuf.ApplicationMetadataMessage_COMMUNITY_DESCRIPTION: return m.unmarshalProtobufData(new(protobuf.CommunityDescription)) case protobuf.ApplicationMetadataMessage_COMMUNITY_REQUEST_TO_JOIN: return m.unmarshalProtobufData(new(protobuf.CommunityRequestToJoin)) case protobuf.ApplicationMetadataMessage_COMMUNITY_EDIT_SHARED_ADDRESSES: return m.unmarshalProtobufData(new(protobuf.CommunityEditRevealedAccounts)) case protobuf.ApplicationMetadataMessage_COMMUNITY_REQUEST_TO_JOIN_RESPONSE: return m.unmarshalProtobufData(new(protobuf.CommunityRequestToJoinResponse)) case protobuf.ApplicationMetadataMessage_COMMUNITY_CANCEL_REQUEST_TO_JOIN: return m.unmarshalProtobufData(new(protobuf.CommunityCancelRequestToJoin)) case protobuf.ApplicationMetadataMessage_COMMUNITY_REQUEST_TO_LEAVE: return m.unmarshalProtobufData(new(protobuf.CommunityRequestToLeave)) case protobuf.ApplicationMetadataMessage_EDIT_MESSAGE: return m.unmarshalProtobufData(new(protobuf.EditMessage)) case protobuf.ApplicationMetadataMessage_DELETE_MESSAGE: return m.unmarshalProtobufData(new(protobuf.DeleteMessage)) case protobuf.ApplicationMetadataMessage_SYNC_DELETE_FOR_ME_MESSAGE: return m.unmarshalProtobufData(new(protobuf.DeleteForMeMessage)) case protobuf.ApplicationMetadataMessage_STATUS_UPDATE: return m.unmarshalProtobufData(new(protobuf.StatusUpdate)) case protobuf.ApplicationMetadataMessage_PUSH_NOTIFICATION_REGISTRATION: // This message is a bit different as it's encrypted, so we pass it straight through v := reflect.ValueOf(m.UnwrappedPayload) m.ParsedMessage = &v return nil case protobuf.ApplicationMetadataMessage_CHAT_IDENTITY: return m.unmarshalProtobufData(new(protobuf.ChatIdentity)) case protobuf.ApplicationMetadataMessage_ANONYMOUS_METRIC_BATCH: return m.unmarshalProtobufData(new(protobuf.AnonymousMetricBatch)) case protobuf.ApplicationMetadataMessage_SYNC_CHAT_REMOVED: return m.unmarshalProtobufData(new(protobuf.SyncChatRemoved)) case protobuf.ApplicationMetadataMessage_SYNC_CHAT_MESSAGES_READ: return m.unmarshalProtobufData(new(protobuf.SyncChatMessagesRead)) case protobuf.ApplicationMetadataMessage_BACKUP: return m.unmarshalProtobufData(new(protobuf.Backup)) case protobuf.ApplicationMetadataMessage_SYNC_ACTIVITY_CENTER_READ: return m.unmarshalProtobufData(new(protobuf.SyncActivityCenterRead)) case protobuf.ApplicationMetadataMessage_SYNC_ACTIVITY_CENTER_ACCEPTED: return m.unmarshalProtobufData(new(protobuf.SyncActivityCenterAccepted)) case protobuf.ApplicationMetadataMessage_SYNC_ACTIVITY_CENTER_DISMISSED: return m.unmarshalProtobufData(new(protobuf.SyncActivityCenterDismissed)) case protobuf.ApplicationMetadataMessage_SYNC_BOOKMARK: return m.unmarshalProtobufData(new(protobuf.SyncBookmark)) case protobuf.ApplicationMetadataMessage_SYNC_CLEAR_HISTORY: return m.unmarshalProtobufData(new(protobuf.SyncClearHistory)) case protobuf.ApplicationMetadataMessage_SYNC_SETTING: return m.unmarshalProtobufData(new(protobuf.SyncSetting)) case protobuf.ApplicationMetadataMessage_COMMUNITY_ARCHIVE_MAGNETLINK: return m.unmarshalProtobufData(new(protobuf.CommunityMessageArchiveMagnetlink)) case protobuf.ApplicationMetadataMessage_SYNC_ACCOUNT: return m.unmarshalProtobufData(new(protobuf.SyncAccount)) case protobuf.ApplicationMetadataMessage_ACCEPT_CONTACT_REQUEST: return m.unmarshalProtobufData(new(protobuf.AcceptContactRequest)) case protobuf.ApplicationMetadataMessage_RETRACT_CONTACT_REQUEST: return m.unmarshalProtobufData(new(protobuf.RetractContactRequest)) case protobuf.ApplicationMetadataMessage_SYNC_COMMUNITY_SETTINGS: return m.unmarshalProtobufData(new(protobuf.SyncCommunitySettings)) case protobuf.ApplicationMetadataMessage_REQUEST_CONTACT_VERIFICATION: return m.unmarshalProtobufData(new(protobuf.RequestContactVerification)) case protobuf.ApplicationMetadataMessage_ACCEPT_CONTACT_VERIFICATION: return m.unmarshalProtobufData(new(protobuf.AcceptContactVerification)) case protobuf.ApplicationMetadataMessage_CANCEL_CONTACT_VERIFICATION: return m.unmarshalProtobufData(new(protobuf.CancelContactVerification)) case protobuf.ApplicationMetadataMessage_DECLINE_CONTACT_VERIFICATION: return m.unmarshalProtobufData(new(protobuf.DeclineContactVerification)) case protobuf.ApplicationMetadataMessage_SYNC_TRUSTED_USER: return m.unmarshalProtobufData((new(protobuf.SyncTrustedUser))) case protobuf.ApplicationMetadataMessage_SYNC_VERIFICATION_REQUEST: return m.unmarshalProtobufData((new(protobuf.SyncVerificationRequest))) case protobuf.ApplicationMetadataMessage_SYNC_CONTACT_REQUEST_DECISION: return m.unmarshalProtobufData((new(protobuf.SyncContactRequestDecision))) case protobuf.ApplicationMetadataMessage_SYNC_SAVED_ADDRESS: return m.unmarshalProtobufData(new(protobuf.SyncSavedAddress)) case protobuf.ApplicationMetadataMessage_SYNC_SOCIAL_LINKS: return m.unmarshalProtobufData(new(protobuf.SyncSocialLinks)) case protobuf.ApplicationMetadataMessage_SYNC_ENS_USERNAME_DETAIL: return m.unmarshalProtobufData(new(protobuf.SyncEnsUsernameDetail)) case protobuf.ApplicationMetadataMessage_SYNC_KEYPAIR: return m.unmarshalProtobufData(new(protobuf.SyncKeypair)) case protobuf.ApplicationMetadataMessage_SYNC_ACTIVITY_CENTER_NOTIFICATION: return m.unmarshalProtobufData(new(protobuf.SyncActivityCenterNotifications)) case protobuf.ApplicationMetadataMessage_SYNC_ACTIVITY_CENTER_NOTIFICATION_STATE: return m.unmarshalProtobufData(new(protobuf.SyncActivityCenterNotificationState)) case protobuf.ApplicationMetadataMessage_COMMUNITY_EVENTS_MESSAGE: return m.unmarshalProtobufData(new(protobuf.CommunityEventsMessage)) case protobuf.ApplicationMetadataMessage_COMMUNITY_EVENTS_MESSAGE_REJECTED: return m.unmarshalProtobufData(new(protobuf.CommunityEventsMessageRejected)) case protobuf.ApplicationMetadataMessage_SYNC_ACCOUNT_CUSTOMIZATION_COLOR: return m.unmarshalProtobufData(new(protobuf.SyncAccountCustomizationColor)) case protobuf.ApplicationMetadataMessage_SYNC_ACCOUNTS_POSITIONS: return m.unmarshalProtobufData(new(protobuf.SyncAccountsPositions)) case protobuf.ApplicationMetadataMessage_COMMUNITY_PRIVILEGED_USER_SYNC_MESSAGE: return m.unmarshalProtobufData(new(protobuf.CommunityPrivilegedUserSyncMessage)) } return nil } func (m *StatusMessage) unmarshalProtobufData(pb proto.Message) error { var ptr proto.Message rv := reflect.ValueOf(pb) if rv.Kind() == reflect.Ptr { ptr = pb } else { ptr = rv.Addr().Interface().(proto.Message) } err := proto.Unmarshal(m.UnwrappedPayload, ptr) if err != nil { m.ParsedMessage = nil log.Error("[message::DecodeMessage] could not decode %T: %#x, err: %v", pb, m.Hash, err.Error()) } else { rv = reflect.ValueOf(ptr) elem := rv.Elem() m.ParsedMessage = &elem return nil } return nil }