Merge branch 'feature/audio-messages' into develop
This commit is contained in:
commit
4574ab4c22
Binary file not shown.
|
@ -0,0 +1,29 @@
|
||||||
|
package audio
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/status-im/status-go/protocol/protobuf"
|
||||||
|
)
|
||||||
|
|
||||||
|
func aac(buf []byte) bool {
|
||||||
|
return len(buf) > 1 &&
|
||||||
|
((buf[0] == 0xFF && buf[1] == 0xF1) ||
|
||||||
|
(buf[0] == 0xFF && buf[1] == 0xF9))
|
||||||
|
}
|
||||||
|
|
||||||
|
func amr(buf []byte) bool {
|
||||||
|
return len(buf) > 11 &&
|
||||||
|
buf[0] == 0x23 && buf[1] == 0x21 &&
|
||||||
|
buf[2] == 0x41 && buf[3] == 0x4D &&
|
||||||
|
buf[4] == 0x52 && buf[5] == 0x0A
|
||||||
|
}
|
||||||
|
|
||||||
|
func Type(buf []byte) protobuf.AudioMessage_AudioType {
|
||||||
|
switch {
|
||||||
|
case aac(buf):
|
||||||
|
return protobuf.AudioMessage_AAC
|
||||||
|
case amr(buf):
|
||||||
|
return protobuf.AudioMessage_AMR
|
||||||
|
default:
|
||||||
|
return protobuf.AudioMessage_UNKNOWN_AUDIO_TYPE
|
||||||
|
}
|
||||||
|
}
|
|
@ -5,4 +5,7 @@ type FeatureFlags struct {
|
||||||
// using datasync, breaking change for non-v1 clients. Public messages
|
// using datasync, breaking change for non-v1 clients. Public messages
|
||||||
// are not impacted
|
// are not impacted
|
||||||
Datasync bool
|
Datasync bool
|
||||||
|
|
||||||
|
// PushNotification indicates whether we should be enabling the push notification feature
|
||||||
|
PushNotifications bool
|
||||||
}
|
}
|
||||||
|
|
|
@ -22,6 +22,10 @@ type QuotedMessage struct {
|
||||||
Text string `json:"text"`
|
Text string `json:"text"`
|
||||||
// Base64Image is the converted base64 image
|
// Base64Image is the converted base64 image
|
||||||
Base64Image string `json:"image,omitempty"`
|
Base64Image string `json:"image,omitempty"`
|
||||||
|
// Base64Audio is the converted base64 audio
|
||||||
|
Base64Audio string `json:"audio,omitempty"`
|
||||||
|
// AudioDurationMs is the audio duration in milliseconds
|
||||||
|
AudioDurationMs uint64 `json:"audioDurationMs,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type CommandState int
|
type CommandState int
|
||||||
|
@ -102,6 +106,10 @@ type Message struct {
|
||||||
Base64Image string `json:"image,omitempty"`
|
Base64Image string `json:"image,omitempty"`
|
||||||
// ImagePath is the path of the image to be sent
|
// ImagePath is the path of the image to be sent
|
||||||
ImagePath string `json:"imagePath,omitempty"`
|
ImagePath string `json:"imagePath,omitempty"`
|
||||||
|
// Base64Audio is the converted base64 audio
|
||||||
|
Base64Audio string `json:"audio,omitempty"`
|
||||||
|
// AudioPath is the path of the audio to be sent
|
||||||
|
AudioPath string `json:"audioPath,omitempty"`
|
||||||
|
|
||||||
// Replace indicates that this is a replacement of a message
|
// Replace indicates that this is a replacement of a message
|
||||||
// that has been updated
|
// that has been updated
|
||||||
|
@ -134,6 +142,8 @@ func (m *Message) MarshalJSON() ([]byte, error) {
|
||||||
ResponseTo string `json:"responseTo"`
|
ResponseTo string `json:"responseTo"`
|
||||||
EnsName string `json:"ensName"`
|
EnsName string `json:"ensName"`
|
||||||
Image string `json:"image,omitempty"`
|
Image string `json:"image,omitempty"`
|
||||||
|
Audio string `json:"audio,omitempty"`
|
||||||
|
AudioDurationMs uint64 `json:"audioDurationMs,omitempty"`
|
||||||
Sticker *StickerAlias `json:"sticker"`
|
Sticker *StickerAlias `json:"sticker"`
|
||||||
CommandParameters *CommandParameters `json:"commandParameters"`
|
CommandParameters *CommandParameters `json:"commandParameters"`
|
||||||
Timestamp uint64 `json:"timestamp"`
|
Timestamp uint64 `json:"timestamp"`
|
||||||
|
@ -159,6 +169,7 @@ func (m *Message) MarshalJSON() ([]byte, error) {
|
||||||
ResponseTo: m.ResponseTo,
|
ResponseTo: m.ResponseTo,
|
||||||
EnsName: m.EnsName,
|
EnsName: m.EnsName,
|
||||||
Image: m.Base64Image,
|
Image: m.Base64Image,
|
||||||
|
Audio: m.Base64Audio,
|
||||||
Timestamp: m.Timestamp,
|
Timestamp: m.Timestamp,
|
||||||
ContentType: m.ContentType,
|
ContentType: m.ContentType,
|
||||||
MessageType: m.MessageType,
|
MessageType: m.MessageType,
|
||||||
|
@ -171,6 +182,11 @@ func (m *Message) MarshalJSON() ([]byte, error) {
|
||||||
Hash: sticker.Hash,
|
Hash: sticker.Hash,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if audio := m.GetAudio(); audio != nil {
|
||||||
|
item.AudioDurationMs = audio.DurationMs
|
||||||
|
}
|
||||||
|
|
||||||
return json.Marshal(item)
|
return json.Marshal(item)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -178,11 +194,12 @@ func (m *Message) UnmarshalJSON(data []byte) error {
|
||||||
type Alias Message
|
type Alias Message
|
||||||
aux := struct {
|
aux := struct {
|
||||||
*Alias
|
*Alias
|
||||||
ResponseTo string `json:"responseTo"`
|
ResponseTo string `json:"responseTo"`
|
||||||
EnsName string `json:"ensName"`
|
EnsName string `json:"ensName"`
|
||||||
ChatID string `json:"chatId"`
|
ChatID string `json:"chatId"`
|
||||||
Sticker *protobuf.StickerMessage `json:"sticker"`
|
Sticker *protobuf.StickerMessage `json:"sticker"`
|
||||||
ContentType protobuf.ChatMessage_ContentType `json:"contentType"`
|
AudioDurationMs uint64 `json:"audioDurationMs"`
|
||||||
|
ContentType protobuf.ChatMessage_ContentType `json:"contentType"`
|
||||||
}{
|
}{
|
||||||
Alias: (*Alias)(m),
|
Alias: (*Alias)(m),
|
||||||
}
|
}
|
||||||
|
@ -192,6 +209,11 @@ func (m *Message) UnmarshalJSON(data []byte) error {
|
||||||
if aux.ContentType == protobuf.ChatMessage_STICKER {
|
if aux.ContentType == protobuf.ChatMessage_STICKER {
|
||||||
m.Payload = &protobuf.ChatMessage_Sticker{Sticker: aux.Sticker}
|
m.Payload = &protobuf.ChatMessage_Sticker{Sticker: aux.Sticker}
|
||||||
}
|
}
|
||||||
|
if aux.ContentType == protobuf.ChatMessage_AUDIO {
|
||||||
|
m.Payload = &protobuf.ChatMessage_Audio{
|
||||||
|
Audio: &protobuf.AudioMessage{DurationMs: aux.AudioDurationMs},
|
||||||
|
}
|
||||||
|
}
|
||||||
m.ResponseTo = aux.ResponseTo
|
m.ResponseTo = aux.ResponseTo
|
||||||
m.EnsName = aux.EnsName
|
m.EnsName = aux.EnsName
|
||||||
m.ChatId = aux.ChatID
|
m.ChatId = aux.ChatID
|
||||||
|
@ -239,6 +261,37 @@ func (m *Message) parseImage() error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// parseAudio check the message contains an audio, and if so
|
||||||
|
// it creates a base64 encoded version of it.
|
||||||
|
func (m *Message) parseAudio() error {
|
||||||
|
if m.ContentType != protobuf.ChatMessage_AUDIO {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
audio := m.GetAudio()
|
||||||
|
if audio == nil {
|
||||||
|
return errors.New("audio empty")
|
||||||
|
}
|
||||||
|
|
||||||
|
payload := audio.Payload
|
||||||
|
|
||||||
|
e64 := base64.StdEncoding
|
||||||
|
|
||||||
|
maxEncLen := e64.EncodedLen(len(payload))
|
||||||
|
encBuf := make([]byte, maxEncLen)
|
||||||
|
|
||||||
|
e64.Encode(encBuf, payload)
|
||||||
|
|
||||||
|
mime, err := getAudioMessageMIME(audio)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
m.Base64Audio = fmt.Sprintf("data:audio/%s;base64,%s", mime, encBuf)
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
// PrepareContent return the parsed content of the message, the line-count and whether
|
// PrepareContent return the parsed content of the message, the line-count and whether
|
||||||
// is a right-to-left message
|
// is a right-to-left message
|
||||||
func (m *Message) PrepareContent() error {
|
func (m *Message) PrepareContent() error {
|
||||||
|
@ -250,7 +303,10 @@ func (m *Message) PrepareContent() error {
|
||||||
m.ParsedText = jsonParsedText
|
m.ParsedText = jsonParsedText
|
||||||
m.LineCount = strings.Count(m.Text, "\n")
|
m.LineCount = strings.Count(m.Text, "\n")
|
||||||
m.RTL = isRTL(m.Text)
|
m.RTL = isRTL(m.Text)
|
||||||
return m.parseImage()
|
if err := m.parseImage(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return m.parseAudio()
|
||||||
}
|
}
|
||||||
|
|
||||||
func getImageMessageMIME(i *protobuf.ImageMessage) (string, error) {
|
func getImageMessageMIME(i *protobuf.ImageMessage) (string, error) {
|
||||||
|
@ -266,3 +322,14 @@ func getImageMessageMIME(i *protobuf.ImageMessage) (string, error) {
|
||||||
}
|
}
|
||||||
return "", errors.New("image format not supported")
|
return "", errors.New("image format not supported")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func getAudioMessageMIME(i *protobuf.AudioMessage) (string, error) {
|
||||||
|
switch i.Type {
|
||||||
|
case protobuf.AudioMessage_AAC:
|
||||||
|
return "aac", nil
|
||||||
|
case protobuf.AudioMessage_AMR:
|
||||||
|
return "amr", nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return "", errors.New("audio format not supported")
|
||||||
|
}
|
||||||
|
|
|
@ -36,6 +36,10 @@ func (db sqlitePersistence) tableUserMessagesAllFields() string {
|
||||||
image_payload,
|
image_payload,
|
||||||
image_type,
|
image_type,
|
||||||
image_base64,
|
image_base64,
|
||||||
|
audio_payload,
|
||||||
|
audio_type,
|
||||||
|
audio_duration_ms,
|
||||||
|
audio_base64,
|
||||||
command_id,
|
command_id,
|
||||||
command_value,
|
command_value,
|
||||||
command_from,
|
command_from,
|
||||||
|
@ -68,6 +72,8 @@ func (db sqlitePersistence) tableUserMessagesAllFieldsJoin() string {
|
||||||
m1.sticker_pack,
|
m1.sticker_pack,
|
||||||
m1.sticker_hash,
|
m1.sticker_hash,
|
||||||
m1.image_base64,
|
m1.image_base64,
|
||||||
|
m1.audio_duration_ms,
|
||||||
|
m1.audio_base64,
|
||||||
m1.command_id,
|
m1.command_id,
|
||||||
m1.command_value,
|
m1.command_value,
|
||||||
m1.command_from,
|
m1.command_from,
|
||||||
|
@ -83,6 +89,8 @@ func (db sqlitePersistence) tableUserMessagesAllFieldsJoin() string {
|
||||||
m2.source,
|
m2.source,
|
||||||
m2.text,
|
m2.text,
|
||||||
m2.image_base64,
|
m2.image_base64,
|
||||||
|
m2.audio_duration_ms,
|
||||||
|
m2.audio_base64,
|
||||||
c.alias,
|
c.alias,
|
||||||
c.identicon`
|
c.identicon`
|
||||||
}
|
}
|
||||||
|
@ -99,11 +107,14 @@ func (db sqlitePersistence) tableUserMessagesScanAllFields(row scanner, message
|
||||||
var quotedText sql.NullString
|
var quotedText sql.NullString
|
||||||
var quotedFrom sql.NullString
|
var quotedFrom sql.NullString
|
||||||
var quotedImage sql.NullString
|
var quotedImage sql.NullString
|
||||||
|
var quotedAudio sql.NullString
|
||||||
|
var quotedAudioDuration sql.NullInt64
|
||||||
var alias sql.NullString
|
var alias sql.NullString
|
||||||
var identicon sql.NullString
|
var identicon sql.NullString
|
||||||
|
|
||||||
sticker := &protobuf.StickerMessage{}
|
sticker := &protobuf.StickerMessage{}
|
||||||
command := &CommandParameters{}
|
command := &CommandParameters{}
|
||||||
|
audio := &protobuf.AudioMessage{}
|
||||||
|
|
||||||
args := []interface{}{
|
args := []interface{}{
|
||||||
&message.ID,
|
&message.ID,
|
||||||
|
@ -123,6 +134,8 @@ func (db sqlitePersistence) tableUserMessagesScanAllFields(row scanner, message
|
||||||
&sticker.Pack,
|
&sticker.Pack,
|
||||||
&sticker.Hash,
|
&sticker.Hash,
|
||||||
&message.Base64Image,
|
&message.Base64Image,
|
||||||
|
&audio.DurationMs,
|
||||||
|
&message.Base64Audio,
|
||||||
&command.ID,
|
&command.ID,
|
||||||
&command.Value,
|
&command.Value,
|
||||||
&command.From,
|
&command.From,
|
||||||
|
@ -138,6 +151,8 @@ func (db sqlitePersistence) tableUserMessagesScanAllFields(row scanner, message
|
||||||
"edFrom,
|
"edFrom,
|
||||||
"edText,
|
"edText,
|
||||||
"edImage,
|
"edImage,
|
||||||
|
"edAudioDuration,
|
||||||
|
"edAudio,
|
||||||
&alias,
|
&alias,
|
||||||
&identicon,
|
&identicon,
|
||||||
}
|
}
|
||||||
|
@ -148,9 +163,11 @@ func (db sqlitePersistence) tableUserMessagesScanAllFields(row scanner, message
|
||||||
|
|
||||||
if quotedText.Valid {
|
if quotedText.Valid {
|
||||||
message.QuotedMessage = &QuotedMessage{
|
message.QuotedMessage = &QuotedMessage{
|
||||||
From: quotedFrom.String,
|
From: quotedFrom.String,
|
||||||
Text: quotedText.String,
|
Text: quotedText.String,
|
||||||
Base64Image: quotedImage.String,
|
Base64Image: quotedImage.String,
|
||||||
|
AudioDurationMs: uint64(quotedAudioDuration.Int64),
|
||||||
|
Base64Audio: quotedAudio.String,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
message.Alias = alias.String
|
message.Alias = alias.String
|
||||||
|
@ -159,6 +176,10 @@ func (db sqlitePersistence) tableUserMessagesScanAllFields(row scanner, message
|
||||||
message.Payload = &protobuf.ChatMessage_Sticker{Sticker: sticker}
|
message.Payload = &protobuf.ChatMessage_Sticker{Sticker: sticker}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if message.ContentType == protobuf.ChatMessage_AUDIO {
|
||||||
|
message.Payload = &protobuf.ChatMessage_Audio{Audio: audio}
|
||||||
|
}
|
||||||
|
|
||||||
if message.ContentType == protobuf.ChatMessage_TRANSACTION_COMMAND {
|
if message.ContentType == protobuf.ChatMessage_TRANSACTION_COMMAND {
|
||||||
message.CommandParameters = command
|
message.CommandParameters = command
|
||||||
}
|
}
|
||||||
|
@ -177,6 +198,11 @@ func (db sqlitePersistence) tableUserMessagesAllValues(message *Message) ([]inte
|
||||||
image = &protobuf.ImageMessage{}
|
image = &protobuf.ImageMessage{}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
audio := message.GetAudio()
|
||||||
|
if audio == nil {
|
||||||
|
audio = &protobuf.AudioMessage{}
|
||||||
|
}
|
||||||
|
|
||||||
command := message.CommandParameters
|
command := message.CommandParameters
|
||||||
if command == nil {
|
if command == nil {
|
||||||
command = &CommandParameters{}
|
command = &CommandParameters{}
|
||||||
|
@ -201,6 +227,10 @@ func (db sqlitePersistence) tableUserMessagesAllValues(message *Message) ([]inte
|
||||||
image.Payload,
|
image.Payload,
|
||||||
image.Type,
|
image.Type,
|
||||||
message.Base64Image,
|
message.Base64Image,
|
||||||
|
audio.Payload,
|
||||||
|
audio.Type,
|
||||||
|
audio.DurationMs,
|
||||||
|
message.Base64Audio,
|
||||||
command.ID,
|
command.ID,
|
||||||
command.Value,
|
command.Value,
|
||||||
command.From,
|
command.From,
|
||||||
|
|
|
@ -11,6 +11,7 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
const expectedJPEG = ""
|
const expectedJPEG = ""
|
||||||
|
const expectedAAC = "data:audio/aac;base64,//FQgBw//NoATGF2YzUyLjcwLjAAQniptokphEFCg5qs1v9fn48+qz1rfWNhwvz+CqB5dipmq3T2PlT1Ld6sPj+19fUt1C3NKV0KowiqohZVCrdf19WMatvV3YbIvAuy/q2RafA8UiZPmZY7DdmHZtP9ri25kedWSiMKQRt79ttlod55LkuX7/f7/f7/f7/YGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYHNqo8g5qs1v9fn48+qz1rfWNhwvz+CqAAAAAAAAAAAAAAAAAAAAAAABw//FQgCNf/CFXbUZfDKFRgsYlKDegtXJH9eLkT54uRM1ckDYDcXRzZGF6Kz5Yps5fTeLY6w7gclwly+0PJL3udY3PyekTFI65bdniF3OjvHeafzZfWTs0qRMSkdll1sbb4SNT5e8vX98ytot6jEZ0NhJi2pBVP/tKV2JMyo36n9uxR2tKR+FoLCsP4SVi49kmvaSCWm5bQD96OmVQA9Q40bqnOa7rT8j9N0TlK991XdcenGTLbyS6eUnN2U1ckf14uRPni5EzVyQAAAAAAAAAAx6Q1flBp+KH2LhgH2Xx+14QB2/jcizm6ngck4vB9DoH9/Vcb7E8Dy+D/1ii1pSPwsUUUXCSsXHsk17SBfKwn2uHr6QAAAAAAAHA//FQgBt//CF3VO1KFCFWcd/r04m+O0758/tXHUlvaqEK9lvhUZXEZMXKMV/LQ6B3/mOl/Mrfs6jpD2b7f+n4yt+tm2x5ZmnpD++dZo/V9VgblI3OW/s1b8qt0h1RBiIRIIYIYQIBeCM8yy7etkwt1JAajRSoZGwwNZ07TTFTyMR1mTUVVUTW97vaDaHU5DV1snBf0mN4fraa+rf/vpdZ8FxqatGjNxPh35UuVfpNqc48W4nZ6rOO/16cTfHad8+f2rjqS3tVAAAAAAAAAAAAAAAAAAAAAAAAAAAO//FQgBm//CEXVPU+GiFsPr7x6+N6v+m+q511I4SgtYVyoyWjcMWMxkaxxDGSx1qVcarjDESt8zLQehx/lkil/GrHBy/NfJcHek0XtfanZJLHNXO2rUnFklPAlQSBS4l0pIoXIfORcXx0UYj1nTsSe1/0wXDkkFCfxWHtqRayOmWm3oS6JGdnZdtjesjByefiS8dLW1tVVVC58ijoxN3gmGFYj07+YJ6eth9fePXxvV/031XOupHCUAAAAAAAAAAAAAAAAAAAAAAAAAAA4P/xUIAcf/whN1T9NsMOEK5rxxxxXnid+f0/Ia195vi6oGH1ZVr6kjqScdSF9lt3qXH+Lxf0fo/Oe53r99IUPzybv/YWGZ7Vgk31MGw+DMp05+3y9fPERUTHlt1c9sUyoqCaD5bdXVz2wkG0hnpDmFy8r0fr3VBn/C7Rmg+L0/45EWfdocGq3HQ1uRro0GJK+vsvo837NR82s01l/n97rsWn7RYNBM3WRcDY3cJKosqMJhgdHtj9yflthd65rxxxxXnid+f0/Ia195vi6oAAAAAAAAAAAAAAAAAAAAAAAAAAAABw"
|
||||||
|
|
||||||
func TestPrepareContentImage(t *testing.T) {
|
func TestPrepareContentImage(t *testing.T) {
|
||||||
file, err := os.Open("../_assets/tests/test.jpg")
|
file, err := os.Open("../_assets/tests/test.jpg")
|
||||||
|
@ -29,7 +30,7 @@ func TestPrepareContentImage(t *testing.T) {
|
||||||
message.Payload = &protobuf.ChatMessage_Image{Image: &image}
|
message.Payload = &protobuf.ChatMessage_Image{Image: &image}
|
||||||
|
|
||||||
require.NoError(t, message.PrepareContent())
|
require.NoError(t, message.PrepareContent())
|
||||||
require.Equal(t, message.Base64Image, expectedJPEG)
|
require.Equal(t, expectedJPEG, message.Base64Image)
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestGetImageMessageMIME(t *testing.T) {
|
func TestGetImageMessageMIME(t *testing.T) {
|
||||||
|
@ -57,3 +58,39 @@ func TestGetImageMessageMIME(t *testing.T) {
|
||||||
_, err = getImageMessageMIME(unknown)
|
_, err = getImageMessageMIME(unknown)
|
||||||
require.Error(t, err)
|
require.Error(t, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestPrepareContentAudio(t *testing.T) {
|
||||||
|
file, err := os.Open("../_assets/tests/test.aac")
|
||||||
|
require.NoError(t, err)
|
||||||
|
defer file.Close()
|
||||||
|
|
||||||
|
payload, err := ioutil.ReadAll(file)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
message := &Message{}
|
||||||
|
message.ContentType = protobuf.ChatMessage_AUDIO
|
||||||
|
audio := protobuf.AudioMessage{
|
||||||
|
Payload: payload,
|
||||||
|
Type: protobuf.AudioMessage_AAC,
|
||||||
|
}
|
||||||
|
message.Payload = &protobuf.ChatMessage_Audio{Audio: &audio}
|
||||||
|
|
||||||
|
require.NoError(t, message.PrepareContent())
|
||||||
|
require.Equal(t, expectedAAC, message.Base64Audio)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestGetAudioMessageMIME(t *testing.T) {
|
||||||
|
aac := &protobuf.AudioMessage{Type: protobuf.AudioMessage_AAC}
|
||||||
|
mime, err := getAudioMessageMIME(aac)
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, "aac", mime)
|
||||||
|
|
||||||
|
amr := &protobuf.AudioMessage{Type: protobuf.AudioMessage_AMR}
|
||||||
|
mime, err = getAudioMessageMIME(amr)
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, "amr", mime)
|
||||||
|
|
||||||
|
unknown := &protobuf.ImageMessage{Type: protobuf.ImageMessage_UNKNOWN_IMAGE_TYPE}
|
||||||
|
_, err = getImageMessageMIME(unknown)
|
||||||
|
require.Error(t, err)
|
||||||
|
}
|
||||||
|
|
|
@ -209,5 +209,22 @@ func ValidateReceivedChatMessage(message *protobuf.ChatMessage, whisperTimestamp
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
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")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -402,6 +402,68 @@ func (s *MessageValidatorSuite) TestValidatePlainTextMessage() {
|
||||||
ContentType: protobuf.ChatMessage_IMAGE,
|
ContentType: protobuf.ChatMessage_IMAGE,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
Name: "Valid audio message",
|
||||||
|
WhisperTimestamp: 2,
|
||||||
|
Valid: true,
|
||||||
|
Message: protobuf.ChatMessage{
|
||||||
|
ChatId: "a",
|
||||||
|
Text: "valid",
|
||||||
|
Clock: 2,
|
||||||
|
Timestamp: 3,
|
||||||
|
ResponseTo: "",
|
||||||
|
EnsName: "",
|
||||||
|
Payload: &protobuf.ChatMessage_Audio{
|
||||||
|
Audio: &protobuf.AudioMessage{
|
||||||
|
Type: 1,
|
||||||
|
Payload: []byte("some-payload"),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
MessageType: protobuf.ChatMessage_ONE_TO_ONE,
|
||||||
|
ContentType: protobuf.ChatMessage_AUDIO,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "Invalid audio message, type unknown",
|
||||||
|
WhisperTimestamp: 2,
|
||||||
|
Valid: false,
|
||||||
|
Message: protobuf.ChatMessage{
|
||||||
|
ChatId: "a",
|
||||||
|
Text: "valid",
|
||||||
|
Clock: 2,
|
||||||
|
Timestamp: 3,
|
||||||
|
ResponseTo: "",
|
||||||
|
EnsName: "",
|
||||||
|
Payload: &protobuf.ChatMessage_Audio{
|
||||||
|
Audio: &protobuf.AudioMessage{
|
||||||
|
Type: protobuf.AudioMessage_UNKNOWN_AUDIO_TYPE,
|
||||||
|
Payload: []byte("some-payload"),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
MessageType: protobuf.ChatMessage_ONE_TO_ONE,
|
||||||
|
ContentType: protobuf.ChatMessage_STICKER,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Name: "Invalid audio message, missing payload",
|
||||||
|
WhisperTimestamp: 2,
|
||||||
|
Valid: false,
|
||||||
|
Message: protobuf.ChatMessage{
|
||||||
|
ChatId: "a",
|
||||||
|
Text: "valid",
|
||||||
|
Clock: 2,
|
||||||
|
Timestamp: 3,
|
||||||
|
ResponseTo: "",
|
||||||
|
EnsName: "",
|
||||||
|
Payload: &protobuf.ChatMessage_Audio{
|
||||||
|
Audio: &protobuf.AudioMessage{
|
||||||
|
Type: 1,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
MessageType: protobuf.ChatMessage_ONE_TO_ONE,
|
||||||
|
ContentType: protobuf.ChatMessage_AUDIO,
|
||||||
|
},
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, tc := range testCases {
|
for _, tc := range testCases {
|
||||||
|
|
|
@ -19,6 +19,7 @@ import (
|
||||||
"github.com/status-im/status-go/eth-node/crypto"
|
"github.com/status-im/status-go/eth-node/crypto"
|
||||||
"github.com/status-im/status-go/eth-node/types"
|
"github.com/status-im/status-go/eth-node/types"
|
||||||
enstypes "github.com/status-im/status-go/eth-node/types/ens"
|
enstypes "github.com/status-im/status-go/eth-node/types/ens"
|
||||||
|
"github.com/status-im/status-go/protocol/audio"
|
||||||
"github.com/status-im/status-go/protocol/common"
|
"github.com/status-im/status-go/protocol/common"
|
||||||
"github.com/status-im/status-go/protocol/encryption"
|
"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/multidevice"
|
||||||
|
@ -1382,6 +1383,32 @@ func (m *Messenger) SendChatMessage(ctx context.Context, message *Message) (*Mes
|
||||||
message.Payload = &protobuf.ChatMessage_Image{Image: &image}
|
message.Payload = &protobuf.ChatMessage_Image{Image: &image}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if len(message.AudioPath) != 0 {
|
||||||
|
file, err := os.Open(message.AudioPath)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
defer file.Close()
|
||||||
|
|
||||||
|
payload, err := ioutil.ReadAll(file)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
|
||||||
|
}
|
||||||
|
audioMessage := message.GetAudio()
|
||||||
|
if audioMessage == nil {
|
||||||
|
return nil, errors.New("no audio has been passed")
|
||||||
|
}
|
||||||
|
audioMessage.Payload = payload
|
||||||
|
audioMessage.Type = audio.Type(payload)
|
||||||
|
message.Payload = &protobuf.ChatMessage_Audio{Audio: audioMessage}
|
||||||
|
err = os.Remove(message.AudioPath)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
logger := m.logger.With(zap.String("site", "Send"), zap.String("chatID", message.ChatId))
|
logger := m.logger.With(zap.String("site", "Send"), zap.String("chatID", message.ChatId))
|
||||||
var response MessengerResponse
|
var response MessengerResponse
|
||||||
|
|
||||||
|
@ -1432,7 +1459,7 @@ func (m *Messenger) SendChatMessage(ctx context.Context, message *Message) (*Mes
|
||||||
|
|
||||||
id, err := m.dispatchMessage(ctx, &common.RawMessage{
|
id, err := m.dispatchMessage(ctx, &common.RawMessage{
|
||||||
LocalChatID: chat.ID,
|
LocalChatID: chat.ID,
|
||||||
SendPushNotification: !chat.Public(),
|
SendPushNotification: m.featureFlags.PushNotifications && !chat.Public(),
|
||||||
Payload: encodedMessage,
|
Payload: encodedMessage,
|
||||||
MessageType: protobuf.ApplicationMetadataMessage_CHAT_MESSAGE,
|
MessageType: protobuf.ApplicationMetadataMessage_CHAT_MESSAGE,
|
||||||
ResendAutomatically: true,
|
ResendAutomatically: true,
|
||||||
|
|
|
@ -116,6 +116,13 @@ func WithDatasync() func(c *config) error {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func WithPushNotifications() func(c *config) error {
|
||||||
|
return func(c *config) error {
|
||||||
|
c.featureFlags.PushNotifications = true
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func WithEnvelopesMonitorConfig(emc *transport.EnvelopesMonitorConfig) Option {
|
func WithEnvelopesMonitorConfig(emc *transport.EnvelopesMonitorConfig) Option {
|
||||||
return func(c *config) error {
|
return func(c *config) error {
|
||||||
c.envelopesMonitorConfig = emc
|
c.envelopesMonitorConfig = emc
|
||||||
|
|
|
@ -14,6 +14,8 @@
|
||||||
// 1591277220_add_index_messages.up.sql (240B)
|
// 1591277220_add_index_messages.up.sql (240B)
|
||||||
// 1593087212_add_mute_chat_and_raw_message_fields.down.sql (0)
|
// 1593087212_add_mute_chat_and_raw_message_fields.down.sql (0)
|
||||||
// 1593087212_add_mute_chat_and_raw_message_fields.up.sql (215B)
|
// 1593087212_add_mute_chat_and_raw_message_fields.up.sql (215B)
|
||||||
|
// 1595862781_add_audio_data.down.sql (0)
|
||||||
|
// 1595862781_add_audio_data.up.sql (246B)
|
||||||
// doc.go (850B)
|
// doc.go (850B)
|
||||||
|
|
||||||
package migrations
|
package migrations
|
||||||
|
@ -338,7 +340,7 @@ func _1593087212_add_mute_chat_and_raw_message_fieldsDownSql() (*asset, error) {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
info := bindataFileInfo{name: "1593087212_add_mute_chat_and_raw_message_fields.down.sql", size: 0, mode: os.FileMode(0644), modTime: time.Unix(1595832279, 0)}
|
info := bindataFileInfo{name: "1593087212_add_mute_chat_and_raw_message_fields.down.sql", size: 0, mode: os.FileMode(0644), modTime: time.Unix(1595862768, 0)}
|
||||||
a := &asset{bytes: bytes, info: info, digest: [32]uint8{0xe3, 0xb0, 0xc4, 0x42, 0x98, 0xfc, 0x1c, 0x14, 0x9a, 0xfb, 0xf4, 0xc8, 0x99, 0x6f, 0xb9, 0x24, 0x27, 0xae, 0x41, 0xe4, 0x64, 0x9b, 0x93, 0x4c, 0xa4, 0x95, 0x99, 0x1b, 0x78, 0x52, 0xb8, 0x55}}
|
a := &asset{bytes: bytes, info: info, digest: [32]uint8{0xe3, 0xb0, 0xc4, 0x42, 0x98, 0xfc, 0x1c, 0x14, 0x9a, 0xfb, 0xf4, 0xc8, 0x99, 0x6f, 0xb9, 0x24, 0x27, 0xae, 0x41, 0xe4, 0x64, 0x9b, 0x93, 0x4c, 0xa4, 0x95, 0x99, 0x1b, 0x78, 0x52, 0xb8, 0x55}}
|
||||||
return a, nil
|
return a, nil
|
||||||
}
|
}
|
||||||
|
@ -358,11 +360,51 @@ func _1593087212_add_mute_chat_and_raw_message_fieldsUpSql() (*asset, error) {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
info := bindataFileInfo{name: "1593087212_add_mute_chat_and_raw_message_fields.up.sql", size: 215, mode: os.FileMode(0644), modTime: time.Unix(1595832279, 0)}
|
info := bindataFileInfo{name: "1593087212_add_mute_chat_and_raw_message_fields.up.sql", size: 215, mode: os.FileMode(0644), modTime: time.Unix(1595862768, 0)}
|
||||||
a := &asset{bytes: bytes, info: info, digest: [32]uint8{0x73, 0x99, 0x61, 0xd1, 0xaa, 0xb4, 0xbf, 0xaf, 0xd7, 0x20, 0x17, 0x40, 0xf9, 0x2, 0xfb, 0xcc, 0x40, 0x2a, 0xd, 0x86, 0x36, 0x30, 0x88, 0x89, 0x25, 0x80, 0x42, 0xb0, 0x5b, 0xe9, 0x73, 0x78}}
|
a := &asset{bytes: bytes, info: info, digest: [32]uint8{0x73, 0x99, 0x61, 0xd1, 0xaa, 0xb4, 0xbf, 0xaf, 0xd7, 0x20, 0x17, 0x40, 0xf9, 0x2, 0xfb, 0xcc, 0x40, 0x2a, 0xd, 0x86, 0x36, 0x30, 0x88, 0x89, 0x25, 0x80, 0x42, 0xb0, 0x5b, 0xe9, 0x73, 0x78}}
|
||||||
return a, nil
|
return a, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var __1595862781_add_audio_dataDownSql = []byte("\x1f\x8b\x08\x00\x00\x00\x00\x00\x00\xff\x01\x00\x00\xff\xff\x00\x00\x00\x00\x00\x00\x00\x00")
|
||||||
|
|
||||||
|
func _1595862781_add_audio_dataDownSqlBytes() ([]byte, error) {
|
||||||
|
return bindataRead(
|
||||||
|
__1595862781_add_audio_dataDownSql,
|
||||||
|
"1595862781_add_audio_data.down.sql",
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
func _1595862781_add_audio_dataDownSql() (*asset, error) {
|
||||||
|
bytes, err := _1595862781_add_audio_dataDownSqlBytes()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
info := bindataFileInfo{name: "1595862781_add_audio_data.down.sql", size: 0, mode: os.FileMode(0644), modTime: time.Unix(1595862768, 0)}
|
||||||
|
a := &asset{bytes: bytes, info: info, digest: [32]uint8{0xe3, 0xb0, 0xc4, 0x42, 0x98, 0xfc, 0x1c, 0x14, 0x9a, 0xfb, 0xf4, 0xc8, 0x99, 0x6f, 0xb9, 0x24, 0x27, 0xae, 0x41, 0xe4, 0x64, 0x9b, 0x93, 0x4c, 0xa4, 0x95, 0x99, 0x1b, 0x78, 0x52, 0xb8, 0x55}}
|
||||||
|
return a, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var __1595862781_add_audio_dataUpSql = []byte("\x1f\x8b\x08\x00\x00\x00\x00\x00\x00\xff\x94\xcb\x41\xaa\xc3\x20\x10\x06\xe0\x7d\x4e\xf1\x93\x33\x3c\xde\x26\x2b\x53\x2d\x14\xa6\x0a\x65\x84\xee\x64\x8a\x52\x02\x4d\x0d\x99\xb8\xc8\xed\x7b\x06\x0f\xf0\x19\x62\xf7\x00\x9b\x99\x1c\x9a\x96\x3d\xad\x45\x55\xde\x45\x61\xac\xc5\x25\x50\xbc\x7b\x48\xcb\x4b\x4d\x9b\x9c\x9f\x2a\x19\x33\x85\x79\x1a\x3a\xe0\x71\x6e\x05\x37\xcf\x5d\x28\xb7\x5d\x8e\xa5\x7e\xd3\xaa\xdd\xf6\x25\x5a\xfe\xff\xc0\xee\xc9\xf0\x81\xe1\x23\x11\xac\xbb\x9a\x48\x8c\x71\x9c\x86\x5f\x00\x00\x00\xff\xff\xeb\x6f\xa0\x62\xf6\x00\x00\x00")
|
||||||
|
|
||||||
|
func _1595862781_add_audio_dataUpSqlBytes() ([]byte, error) {
|
||||||
|
return bindataRead(
|
||||||
|
__1595862781_add_audio_dataUpSql,
|
||||||
|
"1595862781_add_audio_data.up.sql",
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
func _1595862781_add_audio_dataUpSql() (*asset, error) {
|
||||||
|
bytes, err := _1595862781_add_audio_dataUpSqlBytes()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
info := bindataFileInfo{name: "1595862781_add_audio_data.up.sql", size: 246, mode: os.FileMode(0644), modTime: time.Unix(1595862893, 0)}
|
||||||
|
a := &asset{bytes: bytes, info: info, digest: [32]uint8{0xae, 0xd2, 0xee, 0x55, 0xfb, 0x36, 0xa4, 0x92, 0x66, 0xe, 0x81, 0x62, 0x1e, 0x7a, 0x69, 0xa, 0xd5, 0x4b, 0xa5, 0x6a, 0x8d, 0x1d, 0xce, 0xf3, 0x3e, 0xc0, 0x5f, 0x9c, 0x66, 0x1b, 0xb4, 0xed}}
|
||||||
|
return a, nil
|
||||||
|
}
|
||||||
|
|
||||||
var _docGo = []byte("\x1f\x8b\x08\x00\x00\x00\x00\x00\x00\xff\x84\x52\x3f\x8f\xdb\x3e\x0c\xdd\xf3\x29\x1e\x6e\xb9\xe5\x22\x07\xf8\xfd\xa6\xdb\x3a\x74\xe8\xd2\x2e\xd9\x0b\x46\xa6\x6d\x22\x32\xe5\x8a\xf4\x39\xf9\xf6\x85\x74\x17\x9c\x51\x14\xe8\x4a\x89\x8f\xef\x5f\xd7\xe1\x3c\x89\x61\x90\xc4\x10\x83\x72\x64\x33\x2a\x77\x5c\x38\xd2\x6a\x8c\xa7\x51\x7c\x5a\x2f\x21\xe6\xb9\x33\x27\x5f\xed\x28\x73\x37\xcb\x58\xc8\xb9\x7b\xfb\xff\xe9\xd0\x75\x88\xa4\xcf\x8e\x89\xb4\x4f\xdc\xb0\x0c\xe6\x54\x5c\x74\xc4\x26\x3e\x81\xb0\x14\x1e\xe4\x16\xf0\xc5\x91\x98\xcc\xe1\x13\xf9\xb3\xc1\x27\x46\x24\xe3\x0a\x33\xe4\x82\x31\x1f\x2f\xa2\x3d\x39\x85\x3a\xfa\x36\xec\x26\x95\x61\xa4\x94\xb8\xc7\x50\xf2\xdc\x76\x8d\x66\x46\x2f\x85\xa3\xe7\x72\x7f\x01\x99\xb1\x43\x69\x66\xab\xfb\x13\xbd\x31\x34\x7f\x9c\x07\x69\xff\x6f\x45\xd8\x72\xb9\x1a\xc8\xc0\xb7\x85\xa3\x73\x1f\x0e\x15\xeb\xfb\x8f\xf3\xd7\x57\x9c\x27\xae\xf0\x55\x5a\x1e\x1a\x85\x66\x9e\x32\xf7\x06\xcf\x18\x72\x4a\x79\x6b\x0f\xab\xca\x0d\x2e\x33\x9b\xd3\xbc\x20\x66\x7d\x63\x75\xc9\x5a\xd1\x56\x4d\x72\xe5\xf6\xcf\xb7\x0c\x51\x71\xa1\xf4\xee\x5e\x93\x7e\x7e\x37\xe8\x11\x44\x5c\x4b\x61\xf5\x74\x6f\x2b\xac\xb1\xdc\x97\x8a\x85\x77\xe6\x92\xd5\x9a\xbc\xa5\x64\xcf\x31\xa7\xdd\xbc\xa2\xd9\x44\x85\x3f\x1d\x73\xba\x24\x7e\xc1\x36\x49\x9c\x30\x33\xa9\xb5\x40\xda\x87\x44\xce\xe6\x9f\xfb\x10\x85\x73\x99\xad\x0a\xae\xfc\xaa\xbb\x15\xb3\x16\xe7\x91\xc3\x8e\x50\x33\x7f\xa1\xf8\x51\x85\xc7\x95\xd5\xd8\x40\x7f\x98\xf2\x08\x79\x63\x50\xdf\xe3\x74\x3a\x9d\xfe\xfb\x19\x42\x68\x5d\xe0\x1b\xcd\x4b\xa5\xe9\xb5\xa3\x9b\xa4\x84\x0b\x43\x46\xcd\x85\xfb\xca\x8a\x6f\x62\xad\x64\x31\x09\xab\xd7\xcc\x2a\x5e\x4e\x3d\x97\xaa\x47\xf7\x7a\xfe\x66\x59\x38\x1c\x16\x8a\x57\x1a\x19\xf6\x2b\x89\x73\x0d\x7a\xcc\xaf\x23\x2b\xd7\x3a\xec\xcb\x77\x5c\xae\xe3\xde\xec\x63\x46\x08\xdd\xe7\x20\x8c\x19\xe1\xf0\x3b\x00\x00\xff\xff\x12\xcd\x7f\xc4\x52\x03\x00\x00")
|
var _docGo = []byte("\x1f\x8b\x08\x00\x00\x00\x00\x00\x00\xff\x84\x52\x3f\x8f\xdb\x3e\x0c\xdd\xf3\x29\x1e\x6e\xb9\xe5\x22\x07\xf8\xfd\xa6\xdb\x3a\x74\xe8\xd2\x2e\xd9\x0b\x46\xa6\x6d\x22\x32\xe5\x8a\xf4\x39\xf9\xf6\x85\x74\x17\x9c\x51\x14\xe8\x4a\x89\x8f\xef\x5f\xd7\xe1\x3c\x89\x61\x90\xc4\x10\x83\x72\x64\x33\x2a\x77\x5c\x38\xd2\x6a\x8c\xa7\x51\x7c\x5a\x2f\x21\xe6\xb9\x33\x27\x5f\xed\x28\x73\x37\xcb\x58\xc8\xb9\x7b\xfb\xff\xe9\xd0\x75\x88\xa4\xcf\x8e\x89\xb4\x4f\xdc\xb0\x0c\xe6\x54\x5c\x74\xc4\x26\x3e\x81\xb0\x14\x1e\xe4\x16\xf0\xc5\x91\x98\xcc\xe1\x13\xf9\xb3\xc1\x27\x46\x24\xe3\x0a\x33\xe4\x82\x31\x1f\x2f\xa2\x3d\x39\x85\x3a\xfa\x36\xec\x26\x95\x61\xa4\x94\xb8\xc7\x50\xf2\xdc\x76\x8d\x66\x46\x2f\x85\xa3\xe7\x72\x7f\x01\x99\xb1\x43\x69\x66\xab\xfb\x13\xbd\x31\x34\x7f\x9c\x07\x69\xff\x6f\x45\xd8\x72\xb9\x1a\xc8\xc0\xb7\x85\xa3\x73\x1f\x0e\x15\xeb\xfb\x8f\xf3\xd7\x57\x9c\x27\xae\xf0\x55\x5a\x1e\x1a\x85\x66\x9e\x32\xf7\x06\xcf\x18\x72\x4a\x79\x6b\x0f\xab\xca\x0d\x2e\x33\x9b\xd3\xbc\x20\x66\x7d\x63\x75\xc9\x5a\xd1\x56\x4d\x72\xe5\xf6\xcf\xb7\x0c\x51\x71\xa1\xf4\xee\x5e\x93\x7e\x7e\x37\xe8\x11\x44\x5c\x4b\x61\xf5\x74\x6f\x2b\xac\xb1\xdc\x97\x8a\x85\x77\xe6\x92\xd5\x9a\xbc\xa5\x64\xcf\x31\xa7\xdd\xbc\xa2\xd9\x44\x85\x3f\x1d\x73\xba\x24\x7e\xc1\x36\x49\x9c\x30\x33\xa9\xb5\x40\xda\x87\x44\xce\xe6\x9f\xfb\x10\x85\x73\x99\xad\x0a\xae\xfc\xaa\xbb\x15\xb3\x16\xe7\x91\xc3\x8e\x50\x33\x7f\xa1\xf8\x51\x85\xc7\x95\xd5\xd8\x40\x7f\x98\xf2\x08\x79\x63\x50\xdf\xe3\x74\x3a\x9d\xfe\xfb\x19\x42\x68\x5d\xe0\x1b\xcd\x4b\xa5\xe9\xb5\xa3\x9b\xa4\x84\x0b\x43\x46\xcd\x85\xfb\xca\x8a\x6f\x62\xad\x64\x31\x09\xab\xd7\xcc\x2a\x5e\x4e\x3d\x97\xaa\x47\xf7\x7a\xfe\x66\x59\x38\x1c\x16\x8a\x57\x1a\x19\xf6\x2b\x89\x73\x0d\x7a\xcc\xaf\x23\x2b\xd7\x3a\xec\xcb\x77\x5c\xae\xe3\xde\xec\x63\x46\x08\xdd\xe7\x20\x8c\x19\xe1\xf0\x3b\x00\x00\xff\xff\x12\xcd\x7f\xc4\x52\x03\x00\x00")
|
||||||
|
|
||||||
func docGoBytes() ([]byte, error) {
|
func docGoBytes() ([]byte, error) {
|
||||||
|
@ -502,6 +544,10 @@ var _bindata = map[string]func() (*asset, error){
|
||||||
|
|
||||||
"1593087212_add_mute_chat_and_raw_message_fields.up.sql": _1593087212_add_mute_chat_and_raw_message_fieldsUpSql,
|
"1593087212_add_mute_chat_and_raw_message_fields.up.sql": _1593087212_add_mute_chat_and_raw_message_fieldsUpSql,
|
||||||
|
|
||||||
|
"1595862781_add_audio_data.down.sql": _1595862781_add_audio_dataDownSql,
|
||||||
|
|
||||||
|
"1595862781_add_audio_data.up.sql": _1595862781_add_audio_dataUpSql,
|
||||||
|
|
||||||
"doc.go": docGo,
|
"doc.go": docGo,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -560,7 +606,9 @@ var _bintree = &bintree{nil, map[string]*bintree{
|
||||||
"1591277220_add_index_messages.up.sql": &bintree{_1591277220_add_index_messagesUpSql, map[string]*bintree{}},
|
"1591277220_add_index_messages.up.sql": &bintree{_1591277220_add_index_messagesUpSql, map[string]*bintree{}},
|
||||||
"1593087212_add_mute_chat_and_raw_message_fields.down.sql": &bintree{_1593087212_add_mute_chat_and_raw_message_fieldsDownSql, map[string]*bintree{}},
|
"1593087212_add_mute_chat_and_raw_message_fields.down.sql": &bintree{_1593087212_add_mute_chat_and_raw_message_fieldsDownSql, map[string]*bintree{}},
|
||||||
"1593087212_add_mute_chat_and_raw_message_fields.up.sql": &bintree{_1593087212_add_mute_chat_and_raw_message_fieldsUpSql, map[string]*bintree{}},
|
"1593087212_add_mute_chat_and_raw_message_fields.up.sql": &bintree{_1593087212_add_mute_chat_and_raw_message_fieldsUpSql, map[string]*bintree{}},
|
||||||
"doc.go": &bintree{docGo, map[string]*bintree{}},
|
"1595862781_add_audio_data.down.sql": &bintree{_1595862781_add_audio_dataDownSql, map[string]*bintree{}},
|
||||||
|
"1595862781_add_audio_data.up.sql": &bintree{_1595862781_add_audio_dataUpSql, map[string]*bintree{}},
|
||||||
|
"doc.go": &bintree{docGo, map[string]*bintree{}},
|
||||||
}}
|
}}
|
||||||
|
|
||||||
// RestoreAsset restores an asset under the given directory.
|
// RestoreAsset restores an asset under the given directory.
|
||||||
|
|
|
@ -0,0 +1,4 @@
|
||||||
|
ALTER TABLE user_messages ADD COLUMN audio_payload BLOB;
|
||||||
|
ALTER TABLE user_messages ADD COLUMN audio_type INT;
|
||||||
|
ALTER TABLE user_messages ADD COLUMN audio_duration_ms INT;
|
||||||
|
ALTER TABLE user_messages ADD COLUMN audio_base64 TEXT NOT NULL DEFAULT "";
|
|
@ -32,6 +32,10 @@ func (db sqlitePersistence) SaveChat(chat Chat) error {
|
||||||
|
|
||||||
func (db sqlitePersistence) SaveChats(chats []*Chat) error {
|
func (db sqlitePersistence) SaveChats(chats []*Chat) error {
|
||||||
tx, err := db.db.BeginTx(context.Background(), &sql.TxOptions{})
|
tx, err := db.db.BeginTx(context.Background(), &sql.TxOptions{})
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
defer func() {
|
defer func() {
|
||||||
if err == nil {
|
if err == nil {
|
||||||
err = tx.Commit()
|
err = tx.Commit()
|
||||||
|
@ -52,6 +56,10 @@ func (db sqlitePersistence) SaveChats(chats []*Chat) error {
|
||||||
|
|
||||||
func (db sqlitePersistence) SaveContacts(contacts []*Contact) error {
|
func (db sqlitePersistence) SaveContacts(contacts []*Contact) error {
|
||||||
tx, err := db.db.BeginTx(context.Background(), &sql.TxOptions{})
|
tx, err := db.db.BeginTx(context.Background(), &sql.TxOptions{})
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
defer func() {
|
defer func() {
|
||||||
if err == nil {
|
if err == nil {
|
||||||
err = tx.Commit()
|
err = tx.Commit()
|
||||||
|
|
|
@ -54,6 +54,34 @@ func (ImageMessage_ImageType) EnumDescriptor() ([]byte, []int) {
|
||||||
return fileDescriptor_263952f55fd35689, []int{1, 0}
|
return fileDescriptor_263952f55fd35689, []int{1, 0}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type AudioMessage_AudioType int32
|
||||||
|
|
||||||
|
const (
|
||||||
|
AudioMessage_UNKNOWN_AUDIO_TYPE AudioMessage_AudioType = 0
|
||||||
|
AudioMessage_AAC AudioMessage_AudioType = 1
|
||||||
|
AudioMessage_AMR AudioMessage_AudioType = 2
|
||||||
|
)
|
||||||
|
|
||||||
|
var AudioMessage_AudioType_name = map[int32]string{
|
||||||
|
0: "UNKNOWN_AUDIO_TYPE",
|
||||||
|
1: "AAC",
|
||||||
|
2: "AMR",
|
||||||
|
}
|
||||||
|
|
||||||
|
var AudioMessage_AudioType_value = map[string]int32{
|
||||||
|
"UNKNOWN_AUDIO_TYPE": 0,
|
||||||
|
"AAC": 1,
|
||||||
|
"AMR": 2,
|
||||||
|
}
|
||||||
|
|
||||||
|
func (x AudioMessage_AudioType) String() string {
|
||||||
|
return proto.EnumName(AudioMessage_AudioType_name, int32(x))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (AudioMessage_AudioType) EnumDescriptor() ([]byte, []int) {
|
||||||
|
return fileDescriptor_263952f55fd35689, []int{2, 0}
|
||||||
|
}
|
||||||
|
|
||||||
type ChatMessage_MessageType int32
|
type ChatMessage_MessageType int32
|
||||||
|
|
||||||
const (
|
const (
|
||||||
|
@ -86,7 +114,7 @@ func (x ChatMessage_MessageType) String() string {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ChatMessage_MessageType) EnumDescriptor() ([]byte, []int) {
|
func (ChatMessage_MessageType) EnumDescriptor() ([]byte, []int) {
|
||||||
return fileDescriptor_263952f55fd35689, []int{2, 0}
|
return fileDescriptor_263952f55fd35689, []int{3, 0}
|
||||||
}
|
}
|
||||||
|
|
||||||
type ChatMessage_ContentType int32
|
type ChatMessage_ContentType int32
|
||||||
|
@ -101,6 +129,7 @@ const (
|
||||||
// Only local
|
// Only local
|
||||||
ChatMessage_SYSTEM_MESSAGE_CONTENT_PRIVATE_GROUP ChatMessage_ContentType = 6
|
ChatMessage_SYSTEM_MESSAGE_CONTENT_PRIVATE_GROUP ChatMessage_ContentType = 6
|
||||||
ChatMessage_IMAGE ChatMessage_ContentType = 7
|
ChatMessage_IMAGE ChatMessage_ContentType = 7
|
||||||
|
ChatMessage_AUDIO ChatMessage_ContentType = 8
|
||||||
)
|
)
|
||||||
|
|
||||||
var ChatMessage_ContentType_name = map[int32]string{
|
var ChatMessage_ContentType_name = map[int32]string{
|
||||||
|
@ -112,6 +141,7 @@ var ChatMessage_ContentType_name = map[int32]string{
|
||||||
5: "TRANSACTION_COMMAND",
|
5: "TRANSACTION_COMMAND",
|
||||||
6: "SYSTEM_MESSAGE_CONTENT_PRIVATE_GROUP",
|
6: "SYSTEM_MESSAGE_CONTENT_PRIVATE_GROUP",
|
||||||
7: "IMAGE",
|
7: "IMAGE",
|
||||||
|
8: "AUDIO",
|
||||||
}
|
}
|
||||||
|
|
||||||
var ChatMessage_ContentType_value = map[string]int32{
|
var ChatMessage_ContentType_value = map[string]int32{
|
||||||
|
@ -123,6 +153,7 @@ var ChatMessage_ContentType_value = map[string]int32{
|
||||||
"TRANSACTION_COMMAND": 5,
|
"TRANSACTION_COMMAND": 5,
|
||||||
"SYSTEM_MESSAGE_CONTENT_PRIVATE_GROUP": 6,
|
"SYSTEM_MESSAGE_CONTENT_PRIVATE_GROUP": 6,
|
||||||
"IMAGE": 7,
|
"IMAGE": 7,
|
||||||
|
"AUDIO": 8,
|
||||||
}
|
}
|
||||||
|
|
||||||
func (x ChatMessage_ContentType) String() string {
|
func (x ChatMessage_ContentType) String() string {
|
||||||
|
@ -130,7 +161,7 @@ func (x ChatMessage_ContentType) String() string {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ChatMessage_ContentType) EnumDescriptor() ([]byte, []int) {
|
func (ChatMessage_ContentType) EnumDescriptor() ([]byte, []int) {
|
||||||
return fileDescriptor_263952f55fd35689, []int{2, 1}
|
return fileDescriptor_263952f55fd35689, []int{3, 1}
|
||||||
}
|
}
|
||||||
|
|
||||||
type StickerMessage struct {
|
type StickerMessage struct {
|
||||||
|
@ -227,6 +258,61 @@ func (m *ImageMessage) GetType() ImageMessage_ImageType {
|
||||||
return ImageMessage_UNKNOWN_IMAGE_TYPE
|
return ImageMessage_UNKNOWN_IMAGE_TYPE
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type AudioMessage struct {
|
||||||
|
Payload []byte `protobuf:"bytes,1,opt,name=payload,proto3" json:"payload,omitempty"`
|
||||||
|
Type AudioMessage_AudioType `protobuf:"varint,2,opt,name=type,proto3,enum=protobuf.AudioMessage_AudioType" json:"type,omitempty"`
|
||||||
|
DurationMs uint64 `protobuf:"varint,3,opt,name=duration_ms,json=durationMs,proto3" json:"duration_ms,omitempty"`
|
||||||
|
XXX_NoUnkeyedLiteral struct{} `json:"-"`
|
||||||
|
XXX_unrecognized []byte `json:"-"`
|
||||||
|
XXX_sizecache int32 `json:"-"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *AudioMessage) Reset() { *m = AudioMessage{} }
|
||||||
|
func (m *AudioMessage) String() string { return proto.CompactTextString(m) }
|
||||||
|
func (*AudioMessage) ProtoMessage() {}
|
||||||
|
func (*AudioMessage) Descriptor() ([]byte, []int) {
|
||||||
|
return fileDescriptor_263952f55fd35689, []int{2}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *AudioMessage) XXX_Unmarshal(b []byte) error {
|
||||||
|
return xxx_messageInfo_AudioMessage.Unmarshal(m, b)
|
||||||
|
}
|
||||||
|
func (m *AudioMessage) XXX_Marshal(b []byte, deterministic bool) ([]byte, error) {
|
||||||
|
return xxx_messageInfo_AudioMessage.Marshal(b, m, deterministic)
|
||||||
|
}
|
||||||
|
func (m *AudioMessage) XXX_Merge(src proto.Message) {
|
||||||
|
xxx_messageInfo_AudioMessage.Merge(m, src)
|
||||||
|
}
|
||||||
|
func (m *AudioMessage) XXX_Size() int {
|
||||||
|
return xxx_messageInfo_AudioMessage.Size(m)
|
||||||
|
}
|
||||||
|
func (m *AudioMessage) XXX_DiscardUnknown() {
|
||||||
|
xxx_messageInfo_AudioMessage.DiscardUnknown(m)
|
||||||
|
}
|
||||||
|
|
||||||
|
var xxx_messageInfo_AudioMessage proto.InternalMessageInfo
|
||||||
|
|
||||||
|
func (m *AudioMessage) GetPayload() []byte {
|
||||||
|
if m != nil {
|
||||||
|
return m.Payload
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *AudioMessage) GetType() AudioMessage_AudioType {
|
||||||
|
if m != nil {
|
||||||
|
return m.Type
|
||||||
|
}
|
||||||
|
return AudioMessage_UNKNOWN_AUDIO_TYPE
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *AudioMessage) GetDurationMs() uint64 {
|
||||||
|
if m != nil {
|
||||||
|
return m.DurationMs
|
||||||
|
}
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
type ChatMessage struct {
|
type ChatMessage struct {
|
||||||
// Lamport timestamp of the chat message
|
// Lamport timestamp of the chat message
|
||||||
Clock uint64 `protobuf:"varint,1,opt,name=clock,proto3" json:"clock,omitempty"`
|
Clock uint64 `protobuf:"varint,1,opt,name=clock,proto3" json:"clock,omitempty"`
|
||||||
|
@ -251,6 +337,7 @@ type ChatMessage struct {
|
||||||
// Types that are valid to be assigned to Payload:
|
// Types that are valid to be assigned to Payload:
|
||||||
// *ChatMessage_Sticker
|
// *ChatMessage_Sticker
|
||||||
// *ChatMessage_Image
|
// *ChatMessage_Image
|
||||||
|
// *ChatMessage_Audio
|
||||||
Payload isChatMessage_Payload `protobuf_oneof:"payload"`
|
Payload isChatMessage_Payload `protobuf_oneof:"payload"`
|
||||||
XXX_NoUnkeyedLiteral struct{} `json:"-"`
|
XXX_NoUnkeyedLiteral struct{} `json:"-"`
|
||||||
XXX_unrecognized []byte `json:"-"`
|
XXX_unrecognized []byte `json:"-"`
|
||||||
|
@ -261,7 +348,7 @@ func (m *ChatMessage) Reset() { *m = ChatMessage{} }
|
||||||
func (m *ChatMessage) String() string { return proto.CompactTextString(m) }
|
func (m *ChatMessage) String() string { return proto.CompactTextString(m) }
|
||||||
func (*ChatMessage) ProtoMessage() {}
|
func (*ChatMessage) ProtoMessage() {}
|
||||||
func (*ChatMessage) Descriptor() ([]byte, []int) {
|
func (*ChatMessage) Descriptor() ([]byte, []int) {
|
||||||
return fileDescriptor_263952f55fd35689, []int{2}
|
return fileDescriptor_263952f55fd35689, []int{3}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *ChatMessage) XXX_Unmarshal(b []byte) error {
|
func (m *ChatMessage) XXX_Unmarshal(b []byte) error {
|
||||||
|
@ -350,10 +437,16 @@ type ChatMessage_Image struct {
|
||||||
Image *ImageMessage `protobuf:"bytes,10,opt,name=image,proto3,oneof"`
|
Image *ImageMessage `protobuf:"bytes,10,opt,name=image,proto3,oneof"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type ChatMessage_Audio struct {
|
||||||
|
Audio *AudioMessage `protobuf:"bytes,11,opt,name=audio,proto3,oneof"`
|
||||||
|
}
|
||||||
|
|
||||||
func (*ChatMessage_Sticker) isChatMessage_Payload() {}
|
func (*ChatMessage_Sticker) isChatMessage_Payload() {}
|
||||||
|
|
||||||
func (*ChatMessage_Image) isChatMessage_Payload() {}
|
func (*ChatMessage_Image) isChatMessage_Payload() {}
|
||||||
|
|
||||||
|
func (*ChatMessage_Audio) isChatMessage_Payload() {}
|
||||||
|
|
||||||
func (m *ChatMessage) GetPayload() isChatMessage_Payload {
|
func (m *ChatMessage) GetPayload() isChatMessage_Payload {
|
||||||
if m != nil {
|
if m != nil {
|
||||||
return m.Payload
|
return m.Payload
|
||||||
|
@ -375,61 +468,76 @@ func (m *ChatMessage) GetImage() *ImageMessage {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (m *ChatMessage) GetAudio() *AudioMessage {
|
||||||
|
if x, ok := m.GetPayload().(*ChatMessage_Audio); ok {
|
||||||
|
return x.Audio
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
// XXX_OneofWrappers is for the internal use of the proto package.
|
// XXX_OneofWrappers is for the internal use of the proto package.
|
||||||
func (*ChatMessage) XXX_OneofWrappers() []interface{} {
|
func (*ChatMessage) XXX_OneofWrappers() []interface{} {
|
||||||
return []interface{}{
|
return []interface{}{
|
||||||
(*ChatMessage_Sticker)(nil),
|
(*ChatMessage_Sticker)(nil),
|
||||||
(*ChatMessage_Image)(nil),
|
(*ChatMessage_Image)(nil),
|
||||||
|
(*ChatMessage_Audio)(nil),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
proto.RegisterEnum("protobuf.ImageMessage_ImageType", ImageMessage_ImageType_name, ImageMessage_ImageType_value)
|
proto.RegisterEnum("protobuf.ImageMessage_ImageType", ImageMessage_ImageType_name, ImageMessage_ImageType_value)
|
||||||
|
proto.RegisterEnum("protobuf.AudioMessage_AudioType", AudioMessage_AudioType_name, AudioMessage_AudioType_value)
|
||||||
proto.RegisterEnum("protobuf.ChatMessage_MessageType", ChatMessage_MessageType_name, ChatMessage_MessageType_value)
|
proto.RegisterEnum("protobuf.ChatMessage_MessageType", ChatMessage_MessageType_name, ChatMessage_MessageType_value)
|
||||||
proto.RegisterEnum("protobuf.ChatMessage_ContentType", ChatMessage_ContentType_name, ChatMessage_ContentType_value)
|
proto.RegisterEnum("protobuf.ChatMessage_ContentType", ChatMessage_ContentType_name, ChatMessage_ContentType_value)
|
||||||
proto.RegisterType((*StickerMessage)(nil), "protobuf.StickerMessage")
|
proto.RegisterType((*StickerMessage)(nil), "protobuf.StickerMessage")
|
||||||
proto.RegisterType((*ImageMessage)(nil), "protobuf.ImageMessage")
|
proto.RegisterType((*ImageMessage)(nil), "protobuf.ImageMessage")
|
||||||
|
proto.RegisterType((*AudioMessage)(nil), "protobuf.AudioMessage")
|
||||||
proto.RegisterType((*ChatMessage)(nil), "protobuf.ChatMessage")
|
proto.RegisterType((*ChatMessage)(nil), "protobuf.ChatMessage")
|
||||||
}
|
}
|
||||||
|
|
||||||
func init() { proto.RegisterFile("chat_message.proto", fileDescriptor_263952f55fd35689) }
|
func init() { proto.RegisterFile("chat_message.proto", fileDescriptor_263952f55fd35689) }
|
||||||
|
|
||||||
var fileDescriptor_263952f55fd35689 = []byte{
|
var fileDescriptor_263952f55fd35689 = []byte{
|
||||||
// 569 bytes of a gzipped FileDescriptorProto
|
// 652 bytes of a gzipped FileDescriptorProto
|
||||||
0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0x7c, 0x53, 0xcd, 0x4e, 0xdb, 0x40,
|
0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0xa4, 0x53, 0x4d, 0x6f, 0xd3, 0x40,
|
||||||
0x10, 0xc6, 0x89, 0x1d, 0x27, 0xe3, 0x14, 0x6d, 0xa7, 0x08, 0x5c, 0x09, 0xa9, 0x69, 0xd4, 0x43,
|
0x10, 0xad, 0x13, 0x27, 0x8e, 0xc7, 0xa1, 0x5a, 0x96, 0xaa, 0x35, 0x52, 0x25, 0x82, 0xc5, 0x21,
|
||||||
0x4e, 0x39, 0x50, 0x0e, 0xbd, 0x9a, 0xb0, 0x0d, 0x06, 0xfc, 0xa3, 0xf5, 0xa6, 0x94, 0x93, 0x65,
|
0xa7, 0x1c, 0x4a, 0x91, 0xb8, 0xba, 0xae, 0x49, 0xdd, 0xd6, 0x1f, 0x5a, 0x6f, 0x28, 0x3d, 0x59,
|
||||||
0xcc, 0x96, 0x44, 0xe0, 0xd8, 0xc2, 0x5b, 0xa9, 0x5c, 0xfa, 0x3e, 0x7d, 0x9e, 0xbe, 0x50, 0xe5,
|
0xae, 0xb3, 0x34, 0x51, 0xeb, 0x0f, 0xc5, 0xae, 0x44, 0x2f, 0xfc, 0x04, 0xfe, 0x07, 0x77, 0xc4,
|
||||||
0x35, 0x26, 0x4e, 0x55, 0xf5, 0xb4, 0xf3, 0xfb, 0xed, 0xb7, 0xdf, 0xcc, 0x02, 0xa6, 0xcb, 0x44,
|
0xef, 0x43, 0xbb, 0xae, 0x13, 0xa7, 0x42, 0x5c, 0x38, 0x79, 0x66, 0xf6, 0xcd, 0xf3, 0x9b, 0xd9,
|
||||||
0xc6, 0x99, 0x28, 0xcb, 0xe4, 0x4e, 0x4c, 0x8b, 0xc7, 0x5c, 0xe6, 0xd8, 0x57, 0xc7, 0xcd, 0xf7,
|
0xb7, 0x80, 0x93, 0x45, 0x5c, 0x45, 0x29, 0x2b, 0xcb, 0xf8, 0x96, 0x4d, 0x8a, 0x55, 0x5e, 0xe5,
|
||||||
0x6f, 0xe3, 0x4f, 0xb0, 0x1b, 0xc9, 0x55, 0x7a, 0x2f, 0x1e, 0xbd, 0xba, 0x02, 0x11, 0xf4, 0x65,
|
0x78, 0x20, 0x3e, 0x37, 0x0f, 0x5f, 0x8d, 0x8f, 0xb0, 0x1b, 0x56, 0xcb, 0xe4, 0x8e, 0xad, 0xdc,
|
||||||
0x52, 0x2e, 0x6d, 0x6d, 0xa4, 0x4d, 0x06, 0x4c, 0xd9, 0x55, 0xac, 0x48, 0xd2, 0x7b, 0xbb, 0x33,
|
0x1a, 0x81, 0x31, 0xc8, 0x8b, 0xb8, 0x5c, 0xe8, 0xd2, 0x48, 0x1a, 0xab, 0x44, 0xc4, 0xbc, 0x56,
|
||||||
0xd2, 0x26, 0x06, 0x53, 0xf6, 0xf8, 0x97, 0x06, 0x43, 0x37, 0x4b, 0xee, 0x44, 0xd3, 0x68, 0x83,
|
0xc4, 0xc9, 0x9d, 0xde, 0x19, 0x49, 0xe3, 0x1e, 0x11, 0xb1, 0xf1, 0x53, 0x82, 0xa1, 0x93, 0xc6,
|
||||||
0x59, 0x24, 0x4f, 0x0f, 0x79, 0x72, 0xab, 0x7a, 0x87, 0xac, 0x71, 0xf1, 0x18, 0x74, 0xf9, 0x54,
|
0xb7, 0xac, 0x69, 0xd4, 0x41, 0x29, 0xe2, 0xc7, 0xfb, 0x3c, 0x9e, 0x8b, 0xde, 0x21, 0x69, 0x52,
|
||||||
0x08, 0xd5, 0xbe, 0x7b, 0x34, 0x9a, 0x36, 0xb7, 0x4f, 0xdb, 0xfd, 0xb5, 0xc3, 0x9f, 0x0a, 0xc1,
|
0x7c, 0x0c, 0x72, 0xf5, 0x58, 0x30, 0xd1, 0xbe, 0x7b, 0x34, 0x9a, 0x34, 0x7f, 0x9f, 0xb4, 0xfb,
|
||||||
0x54, 0xf5, 0xd8, 0x85, 0xc1, 0x4b, 0x08, 0xf7, 0x01, 0x17, 0xfe, 0x85, 0x1f, 0x5c, 0xf9, 0xb1,
|
0xeb, 0x84, 0x3e, 0x16, 0x8c, 0x08, 0xb4, 0xe1, 0x80, 0xba, 0x2e, 0xe1, 0x7d, 0xc0, 0x33, 0xef,
|
||||||
0xeb, 0x39, 0x73, 0x1a, 0xf3, 0xeb, 0x90, 0x92, 0x1d, 0x34, 0xa1, 0x1b, 0xfa, 0x73, 0xa2, 0x61,
|
0xc2, 0xf3, 0xaf, 0xbc, 0xc8, 0x71, 0xcd, 0xa9, 0x1d, 0xd1, 0xeb, 0xc0, 0x46, 0x3b, 0x58, 0x81,
|
||||||
0x1f, 0xf4, 0xf3, 0x90, 0xce, 0x49, 0xa7, 0xb2, 0xae, 0xe8, 0x49, 0x48, 0xba, 0x55, 0x72, 0xee,
|
0x6e, 0xe0, 0x4d, 0x91, 0x84, 0x07, 0x20, 0x9f, 0x07, 0xf6, 0x14, 0x75, 0x78, 0x74, 0x65, 0x9f,
|
||||||
0x7e, 0x26, 0xfa, 0xf8, 0xb7, 0x01, 0xd6, 0x6c, 0x99, 0xc8, 0x86, 0xea, 0x1e, 0x18, 0xe9, 0x43,
|
0x04, 0xa8, 0xcb, 0x0f, 0xa7, 0xce, 0x27, 0x24, 0x1b, 0xbf, 0x25, 0x18, 0x9a, 0x0f, 0xf3, 0x65,
|
||||||
0x9e, 0xde, 0x2b, 0xa2, 0x3a, 0xab, 0x1d, 0x3c, 0x84, 0x81, 0x5c, 0x65, 0xa2, 0x94, 0x49, 0x56,
|
0xfe, 0x1f, 0x5a, 0xdb, 0xfd, 0x75, 0xb2, 0xd1, 0x8a, 0xdf, 0x80, 0x36, 0x7f, 0x58, 0xc5, 0xd5,
|
||||||
0x28, 0xae, 0x3a, 0xdb, 0x04, 0x2a, 0x0d, 0xa4, 0xf8, 0x21, 0xed, 0x6e, 0xad, 0x4b, 0x65, 0xe3,
|
0x32, 0xcf, 0xa2, 0xb4, 0xd4, 0xbb, 0x23, 0x69, 0x2c, 0x13, 0x68, 0x4a, 0x6e, 0x69, 0x7c, 0x00,
|
||||||
0x3b, 0xb0, 0x1e, 0x45, 0x59, 0xe4, 0xeb, 0x52, 0xc4, 0x32, 0xb7, 0x75, 0x95, 0x82, 0x26, 0xc4,
|
0x75, 0xdd, 0xd3, 0x1e, 0xc6, 0x9c, 0x9d, 0x3a, 0x7e, 0x6b, 0x18, 0xd3, 0xb4, 0x90, 0x24, 0x02,
|
||||||
0x73, 0x7c, 0x0b, 0x7d, 0xb1, 0x2e, 0xe3, 0x75, 0x92, 0x09, 0xdb, 0x50, 0x59, 0x53, 0xac, 0x4b,
|
0x97, 0xa0, 0x8e, 0xf1, 0xa3, 0x0f, 0x9a, 0xb5, 0x88, 0xab, 0x46, 0xf7, 0x1e, 0xf4, 0x92, 0xfb,
|
||||||
0x3f, 0xc9, 0x04, 0x1e, 0x80, 0xa9, 0x26, 0xb3, 0xba, 0xb5, 0x7b, 0x2a, 0xd3, 0xab, 0x5c, 0xf7,
|
0x3c, 0xb9, 0x13, 0xaa, 0x65, 0x52, 0x27, 0xf8, 0x10, 0xd4, 0x6a, 0x99, 0xb2, 0xb2, 0x8a, 0xd3,
|
||||||
0x16, 0x4f, 0x61, 0xf8, 0x3c, 0xad, 0x58, 0xa9, 0x66, 0x2a, 0xd5, 0xde, 0x6f, 0x54, 0x6b, 0xbd,
|
0x42, 0x08, 0x97, 0xc9, 0xa6, 0xc0, 0x2f, 0xaf, 0x62, 0xdf, 0x2a, 0x21, 0x4a, 0x25, 0x22, 0xe6,
|
||||||
0x64, 0xfa, 0x7c, 0x2a, 0xd9, 0xac, 0x6c, 0xe3, 0x54, 0x28, 0x69, 0xbe, 0x96, 0x62, 0x2d, 0x6b,
|
0x7a, 0x57, 0xac, 0x2c, 0xf2, 0xac, 0x64, 0x51, 0x95, 0xeb, 0xb2, 0x38, 0x82, 0xa6, 0x44, 0x73,
|
||||||
0x94, 0xfe, 0xff, 0x50, 0x66, 0x75, 0x65, 0x8d, 0x92, 0x6e, 0x1c, 0x3c, 0x06, 0xb3, 0xac, 0xd7,
|
0xfc, 0x1a, 0x06, 0x2c, 0x2b, 0xa3, 0x2c, 0x4e, 0x99, 0xde, 0x13, 0xa7, 0x0a, 0xcb, 0x4a, 0x2f,
|
||||||
0xc3, 0x1e, 0x8c, 0xb4, 0x89, 0x75, 0x64, 0x6f, 0x00, 0xb6, 0xf7, 0xe6, 0x6c, 0x87, 0x35, 0xa5,
|
0x4e, 0x19, 0x3e, 0x00, 0x45, 0x58, 0x6a, 0x39, 0xd7, 0xfb, 0xe2, 0xa4, 0xcf, 0x53, 0x67, 0x8e,
|
||||||
0x38, 0x05, 0x63, 0x55, 0x4d, 0xce, 0x06, 0xd5, 0xb3, 0xff, 0xef, 0x81, 0x9f, 0xed, 0xb0, 0xba,
|
0x4f, 0x61, 0xf8, 0x64, 0xb3, 0x48, 0xac, 0x50, 0x11, 0x2b, 0x7c, 0xbb, 0x59, 0x61, 0x6b, 0x92,
|
||||||
0x6c, 0xfc, 0x13, 0xac, 0xd6, 0x3b, 0xd0, 0x86, 0xbd, 0x66, 0xd6, 0x1e, 0x8d, 0xa2, 0xd6, 0xb4,
|
0xc9, 0xd3, 0x57, 0xec, 0x50, 0x4b, 0x37, 0x09, 0x67, 0x49, 0xf2, 0xac, 0x62, 0x59, 0x55, 0xb3,
|
||||||
0x77, 0x01, 0x02, 0x9f, 0xc6, 0x3c, 0x88, 0x03, 0x9f, 0x12, 0x0d, 0x09, 0x0c, 0xc3, 0xc5, 0xc9,
|
0x0c, 0xfe, 0xc5, 0x62, 0xd5, 0xc8, 0x9a, 0x25, 0xd9, 0x24, 0xf8, 0x18, 0x94, 0xb2, 0xf6, 0xb5,
|
||||||
0xa5, 0x3b, 0x8b, 0xe7, 0x2c, 0x58, 0x84, 0xa4, 0x83, 0xaf, 0xe1, 0x55, 0xc8, 0xdc, 0x2f, 0x0e,
|
0xae, 0x8e, 0xa4, 0xb1, 0x76, 0xa4, 0x6f, 0x08, 0xb6, 0x0d, 0x7f, 0xb6, 0x43, 0x1a, 0x28, 0x9e,
|
||||||
0xa7, 0xcf, 0xa1, 0x2e, 0x8e, 0xe0, 0x30, 0xba, 0x8e, 0x38, 0xf5, 0x5e, 0xd0, 0xb6, 0x2b, 0xf4,
|
0x40, 0x6f, 0xc9, 0x2d, 0xa7, 0x83, 0xe8, 0xd9, 0xff, 0xbb, 0x53, 0xcf, 0x76, 0x48, 0x0d, 0xe3,
|
||||||
0x6a, 0x95, 0xad, 0x96, 0x04, 0x6d, 0x02, 0xb3, 0xc0, 0xe7, 0xd4, 0xe7, 0x2d, 0x02, 0x9c, 0x7e,
|
0xf8, 0x98, 0xdf, 0xaa, 0xae, 0x3d, 0xc7, 0xb7, 0xdd, 0xc2, 0xf1, 0x02, 0x66, 0x7c, 0x07, 0xad,
|
||||||
0xe5, 0x71, 0x78, 0xe9, 0xb8, 0x3e, 0xd1, 0xd0, 0x02, 0x33, 0xe2, 0xee, 0xec, 0x82, 0x32, 0xd2,
|
0x35, 0x37, 0xd6, 0x61, 0xaf, 0xf1, 0x81, 0x6b, 0x87, 0x61, 0xcb, 0xd6, 0xbb, 0x00, 0xbe, 0x67,
|
||||||
0x41, 0x80, 0x5e, 0xc4, 0x1d, 0xbe, 0x88, 0x48, 0x17, 0x07, 0x60, 0x50, 0x2f, 0x38, 0x77, 0x89,
|
0x47, 0xd4, 0x8f, 0x7c, 0xcf, 0x46, 0x12, 0x46, 0x30, 0x0c, 0x66, 0x27, 0x97, 0x8e, 0x15, 0x4d,
|
||||||
0x8e, 0x07, 0xf0, 0x86, 0x33, 0xc7, 0x8f, 0x9c, 0x19, 0x77, 0x83, 0x0a, 0xd1, 0xf3, 0x1c, 0xff,
|
0x89, 0x3f, 0x0b, 0x50, 0x07, 0xbf, 0x84, 0x17, 0x01, 0x71, 0x3e, 0x9b, 0xd4, 0x7e, 0x2a, 0x75,
|
||||||
0x94, 0x18, 0x38, 0x81, 0x0f, 0x7f, 0x11, 0x6b, 0x6e, 0xdb, 0x26, 0xd8, 0xab, 0xd0, 0xd4, 0xd6,
|
0xf1, 0x08, 0x0e, 0xc3, 0xeb, 0x90, 0xda, 0xee, 0x9a, 0x6d, 0x1b, 0x21, 0x1b, 0xbf, 0x24, 0xd0,
|
||||||
0x13, 0xf3, 0x64, 0xf0, 0xf2, 0xcb, 0x6e, 0x7a, 0x4a, 0xd6, 0x8f, 0x7f, 0x02, 0x00, 0x00, 0xff,
|
0x5a, 0x2b, 0x6b, 0x0b, 0xb0, 0x7c, 0x8f, 0xda, 0x1e, 0x6d, 0x09, 0xa0, 0xf6, 0x17, 0x1a, 0x05,
|
||||||
0xff, 0x1f, 0xf2, 0x87, 0xb6, 0xe2, 0x03, 0x00, 0x00,
|
0x97, 0xa6, 0xe3, 0x21, 0x09, 0x6b, 0xa0, 0x84, 0xd4, 0xb1, 0x2e, 0x6c, 0x82, 0x3a, 0x18, 0xa0,
|
||||||
|
0x1f, 0x52, 0x93, 0xce, 0x42, 0xd4, 0xc5, 0x2a, 0xf4, 0x6c, 0xd7, 0x3f, 0x77, 0x90, 0x8c, 0x0f,
|
||||||
|
0xe0, 0x15, 0x25, 0xa6, 0x17, 0x9a, 0x16, 0x75, 0x7c, 0xce, 0xe8, 0xba, 0xa6, 0x77, 0x8a, 0x7a,
|
||||||
|
0x78, 0x0c, 0xef, 0x9e, 0x09, 0x6b, 0xfe, 0xb6, 0x2d, 0xb0, 0xcf, 0xd9, 0xc4, 0xf3, 0x46, 0x0a,
|
||||||
|
0x0f, 0xc5, 0xe3, 0x40, 0x83, 0x13, 0x75, 0xfd, 0x5a, 0x6f, 0xfa, 0x62, 0xc3, 0xef, 0xff, 0x04,
|
||||||
|
0x00, 0x00, 0xff, 0xff, 0x5b, 0xc2, 0xb0, 0xfc, 0xd6, 0x04, 0x00, 0x00,
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,6 +19,17 @@ message ImageMessage {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
message AudioMessage {
|
||||||
|
bytes payload = 1;
|
||||||
|
AudioType type = 2;
|
||||||
|
uint64 duration_ms = 3;
|
||||||
|
enum AudioType {
|
||||||
|
UNKNOWN_AUDIO_TYPE = 0;
|
||||||
|
AAC = 1;
|
||||||
|
AMR = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
message ChatMessage {
|
message ChatMessage {
|
||||||
// Lamport timestamp of the chat message
|
// Lamport timestamp of the chat message
|
||||||
uint64 clock = 1;
|
uint64 clock = 1;
|
||||||
|
@ -45,6 +56,7 @@ message ChatMessage {
|
||||||
oneof payload {
|
oneof payload {
|
||||||
StickerMessage sticker = 9;
|
StickerMessage sticker = 9;
|
||||||
ImageMessage image = 10;
|
ImageMessage image = 10;
|
||||||
|
AudioMessage audio = 11;
|
||||||
}
|
}
|
||||||
|
|
||||||
enum MessageType {
|
enum MessageType {
|
||||||
|
@ -64,5 +76,6 @@ message ChatMessage {
|
||||||
// Only local
|
// Only local
|
||||||
SYSTEM_MESSAGE_CONTENT_PRIVATE_GROUP = 6;
|
SYSTEM_MESSAGE_CONTENT_PRIVATE_GROUP = 6;
|
||||||
IMAGE = 7;
|
IMAGE = 7;
|
||||||
|
AUDIO = 8;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -88,6 +88,7 @@ func (s *MessengerPushNotificationSuite) newMessengerWithKey(shh types.Waku, pri
|
||||||
WithMessagesPersistenceEnabled(),
|
WithMessagesPersistenceEnabled(),
|
||||||
WithDatabaseConfig(tmpFile.Name(), ""),
|
WithDatabaseConfig(tmpFile.Name(), ""),
|
||||||
WithDatasync(),
|
WithDatasync(),
|
||||||
|
WithPushNotifications(),
|
||||||
}
|
}
|
||||||
return s.newMessengerWithOptions(shh, privateKey, options)
|
return s.newMessengerWithOptions(shh, privateKey, options)
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue