2019-12-02 16:34:05 +01:00
|
|
|
package protocol
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
2020-01-10 19:59:01 +01:00
|
|
|
"strconv"
|
2019-12-02 16:34:05 +01:00
|
|
|
"strings"
|
2020-01-02 10:10:19 +01:00
|
|
|
|
|
|
|
"github.com/status-im/status-go/protocol/protobuf"
|
2020-01-20 17:44:32 +01:00
|
|
|
"github.com/status-im/status-go/protocol/v1"
|
2019-12-02 16:34:05 +01:00
|
|
|
)
|
|
|
|
|
2020-01-20 17:44:32 +01:00
|
|
|
// maxWhisperDrift is how many milliseconds we allow the clock value to differ
|
|
|
|
// from whisperTimestamp
|
|
|
|
const maxWhisperFutureDriftMs uint64 = 120000
|
|
|
|
|
|
|
|
func validateClockValue(clock uint64, whisperTimestamp uint64) error {
|
|
|
|
if clock == 0 {
|
2020-01-10 19:59:01 +01:00
|
|
|
return errors.New("clock can't be 0")
|
|
|
|
}
|
|
|
|
|
2020-01-20 17:44:32 +01:00
|
|
|
if clock > whisperTimestamp && clock-whisperTimestamp > maxWhisperFutureDriftMs {
|
|
|
|
return errors.New("clock value too high")
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func ValidateMembershipUpdateMessage(message *protocol.MembershipUpdateMessage, timeNowMs uint64) error {
|
|
|
|
|
|
|
|
for _, e := range message.Events {
|
|
|
|
if err := validateClockValue(e.ClockValue, timeNowMs); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func ValidateReceivedPairInstallation(message *protobuf.PairInstallation, whisperTimestamp uint64) error {
|
|
|
|
if err := validateClockValue(message.Clock, whisperTimestamp); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-01-10 19:59:01 +01:00
|
|
|
if len(strings.TrimSpace(message.Name)) == 0 {
|
|
|
|
return errors.New("name can't be empty")
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(strings.TrimSpace(message.DeviceType)) == 0 {
|
|
|
|
return errors.New("device type can't be empty")
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(strings.TrimSpace(message.InstallationId)) == 0 {
|
|
|
|
return errors.New("installationId can't be empty")
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-01-20 17:44:32 +01:00
|
|
|
func ValidateReceivedSendTransaction(message *protobuf.SendTransaction, whisperTimestamp uint64) error {
|
|
|
|
if err := validateClockValue(message.Clock, whisperTimestamp); err != nil {
|
|
|
|
return err
|
2020-01-10 19:59:01 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if len(strings.TrimSpace(message.TransactionHash)) == 0 {
|
|
|
|
return errors.New("transaction hash can't be empty")
|
|
|
|
}
|
|
|
|
|
|
|
|
if message.Signature == nil {
|
|
|
|
return errors.New("signature can't be nil")
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-01-20 17:44:32 +01:00
|
|
|
func ValidateReceivedRequestAddressForTransaction(message *protobuf.RequestAddressForTransaction, whisperTimestamp uint64) error {
|
|
|
|
if err := validateClockValue(message.Clock, whisperTimestamp); err != nil {
|
|
|
|
return err
|
2020-01-10 19:59:01 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if len(strings.TrimSpace(message.Value)) == 0 {
|
|
|
|
return errors.New("value can't be empty")
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err := strconv.ParseFloat(message.Value, 64)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-01-20 17:44:32 +01:00
|
|
|
func ValidateReceivedRequestTransaction(message *protobuf.RequestTransaction, whisperTimestamp uint64) error {
|
|
|
|
if err := validateClockValue(message.Clock, whisperTimestamp); err != nil {
|
|
|
|
return err
|
2020-01-10 19:59:01 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if len(strings.TrimSpace(message.Value)) == 0 {
|
|
|
|
return errors.New("value can't be empty")
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(strings.TrimSpace(message.Address)) == 0 {
|
|
|
|
return errors.New("address can't be empty")
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err := strconv.ParseFloat(message.Value, 64)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-01-20 17:44:32 +01:00
|
|
|
func ValidateReceivedAcceptRequestAddressForTransaction(message *protobuf.AcceptRequestAddressForTransaction, whisperTimestamp uint64) error {
|
|
|
|
if err := validateClockValue(message.Clock, whisperTimestamp); err != nil {
|
|
|
|
return err
|
2020-01-10 19:59:01 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if len(message.Id) == 0 {
|
|
|
|
return errors.New("messageID can't be empty")
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(strings.TrimSpace(message.Address)) == 0 {
|
|
|
|
return errors.New("address can't be empty")
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-01-20 17:44:32 +01:00
|
|
|
func ValidateReceivedDeclineRequestAddressForTransaction(message *protobuf.DeclineRequestAddressForTransaction, whisperTimestamp uint64) error {
|
|
|
|
if err := validateClockValue(message.Clock, whisperTimestamp); err != nil {
|
|
|
|
return err
|
2020-01-10 19:59:01 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if len(message.Id) == 0 {
|
|
|
|
return errors.New("messageID can't be empty")
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-01-20 17:44:32 +01:00
|
|
|
func ValidateReceivedDeclineRequestTransaction(message *protobuf.DeclineRequestTransaction, whisperTimestamp uint64) error {
|
|
|
|
if err := validateClockValue(message.Clock, whisperTimestamp); err != nil {
|
|
|
|
return err
|
2020-01-10 19:59:01 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if len(message.Id) == 0 {
|
|
|
|
return errors.New("messageID can't be empty")
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-01-20 17:44:32 +01:00
|
|
|
func ValidateReceivedChatMessage(message *protobuf.ChatMessage, whisperTimestamp uint64) error {
|
|
|
|
if err := validateClockValue(message.Clock, whisperTimestamp); err != nil {
|
|
|
|
return err
|
2019-12-02 16:34:05 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if message.Timestamp == 0 {
|
2020-01-10 19:59:01 +01:00
|
|
|
return errors.New("timestamp can't be 0")
|
2019-12-02 16:34:05 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if len(strings.TrimSpace(message.Text)) == 0 {
|
2020-01-10 19:59:01 +01:00
|
|
|
return errors.New("text can't be empty")
|
2019-12-02 16:34:05 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if len(message.ChatId) == 0 {
|
2020-01-10 19:59:01 +01:00
|
|
|
return errors.New("chatId can't be empty")
|
2019-12-02 16:34:05 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if message.MessageType == protobuf.ChatMessage_UNKNOWN_MESSAGE_TYPE || message.MessageType == protobuf.ChatMessage_SYSTEM_MESSAGE_PRIVATE_GROUP {
|
2020-01-10 19:59:01 +01:00
|
|
|
return errors.New("unknown message type")
|
2019-12-02 16:34:05 +01:00
|
|
|
}
|
|
|
|
|
2020-07-10 14:00:58 +01:00
|
|
|
switch message.ContentType {
|
|
|
|
case protobuf.ChatMessage_UNKNOWN_CONTENT_TYPE:
|
|
|
|
return errors.New("unknown content type")
|
|
|
|
|
|
|
|
case protobuf.ChatMessage_TRANSACTION_COMMAND:
|
|
|
|
return errors.New("can't receive request address for transaction from others")
|
|
|
|
|
|
|
|
case protobuf.ChatMessage_STICKER:
|
2019-12-02 16:34:05 +01:00
|
|
|
if message.Payload == nil {
|
2020-01-10 19:59:01 +01:00
|
|
|
return errors.New("no sticker content")
|
2019-12-02 16:34:05 +01:00
|
|
|
}
|
|
|
|
sticker := message.GetSticker()
|
|
|
|
if sticker == nil {
|
2020-01-10 19:59:01 +01:00
|
|
|
return errors.New("no sticker content")
|
2019-12-02 16:34:05 +01:00
|
|
|
}
|
|
|
|
if len(sticker.Hash) == 0 {
|
2020-01-10 19:59:01 +01:00
|
|
|
return errors.New("sticker hash not set")
|
2019-12-02 16:34:05 +01:00
|
|
|
}
|
2020-05-13 15:16:17 +02:00
|
|
|
|
2020-07-10 14:00:58 +01:00
|
|
|
case protobuf.ChatMessage_IMAGE:
|
2020-05-13 15:16:17 +02:00
|
|
|
if message.Payload == nil {
|
|
|
|
return errors.New("no image content")
|
|
|
|
}
|
|
|
|
image := message.GetImage()
|
|
|
|
if image == nil {
|
|
|
|
return errors.New("no image content")
|
|
|
|
}
|
|
|
|
if len(image.Payload) == 0 {
|
|
|
|
return errors.New("image payload empty")
|
|
|
|
}
|
|
|
|
if image.Type == protobuf.ImageMessage_UNKNOWN_IMAGE_TYPE {
|
|
|
|
return errors.New("image type unknown")
|
|
|
|
}
|
2020-07-10 14:07:26 +01:00
|
|
|
|
|
|
|
case protobuf.ChatMessage_EMOJI_REACTION:
|
|
|
|
if message.Payload == nil {
|
|
|
|
return errors.New("no emoji reaction data")
|
|
|
|
}
|
|
|
|
emojiR := message.GetEmojiReaction()
|
|
|
|
if emojiR == nil {
|
|
|
|
return errors.New("no emoji reaction data")
|
|
|
|
}
|
2020-07-10 14:36:31 +01:00
|
|
|
if len(emojiR.MessageId) == 0 {
|
|
|
|
return errors.New("emoji reaction no target message id")
|
|
|
|
}
|
2020-07-10 14:07:26 +01:00
|
|
|
if emojiR.Type == protobuf.EmojiReaction_UNKNOWN_EMOJI_REACTION_TYPE {
|
|
|
|
return errors.New("emoji reaction type unknown")
|
|
|
|
}
|
|
|
|
|
|
|
|
case protobuf.ChatMessage_EMOJI_REACTION_RETRACTION:
|
|
|
|
if message.Payload == nil {
|
|
|
|
return errors.New("no emoji reaction retraction data")
|
|
|
|
}
|
|
|
|
emojiRR := message.GetEmojiReactionRetraction()
|
|
|
|
if emojiRR == nil {
|
|
|
|
return errors.New("no emoji reaction retraction data")
|
|
|
|
}
|
2020-07-10 14:36:31 +01:00
|
|
|
if len(emojiRR.EmojiReactionId) == 0 {
|
|
|
|
return errors.New("emoji reaction retraction no target emoji reaction id")
|
|
|
|
}
|
2020-05-13 15:16:17 +02:00
|
|
|
}
|
|
|
|
|
2020-06-17 20:55:49 +02:00
|
|
|
if message.ContentType == protobuf.ChatMessage_AUDIO {
|
|
|
|
if message.Payload == nil {
|
|
|
|
return errors.New("no audio content")
|
|
|
|
}
|
|
|
|
audio := message.GetAudio()
|
|
|
|
if audio == nil {
|
|
|
|
return errors.New("no audio content")
|
|
|
|
}
|
|
|
|
if len(audio.Payload) == 0 {
|
|
|
|
return errors.New("audio payload empty")
|
|
|
|
}
|
|
|
|
|
|
|
|
if audio.Type == protobuf.AudioMessage_UNKNOWN_AUDIO_TYPE {
|
|
|
|
return errors.New("audio type unknown")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-02 16:34:05 +01:00
|
|
|
return nil
|
|
|
|
}
|