2021-11-03 21:34:06 +02:00
// mautrix-whatsapp - A Matrix-WhatsApp puppeting bridge.
// Copyright (C) 2021 Tulir Asokan
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.
package main
import (
2022-03-14 23:53:50 +02:00
"fmt"
2021-11-03 21:34:06 +02:00
"sort"
"sync"
"time"
waProto "go.mau.fi/whatsmeow/binary/proto"
"go.mau.fi/whatsmeow/types"
2022-01-03 16:11:39 +02:00
2021-11-03 21:34:06 +02:00
"maunium.net/go/mautrix"
"maunium.net/go/mautrix/appservice"
"maunium.net/go/mautrix/event"
"maunium.net/go/mautrix/id"
2022-02-10 19:18:49 +02:00
"maunium.net/go/mautrix-whatsapp/database"
2021-11-03 21:34:06 +02:00
)
// region User history sync handling
type portalToBackfill struct {
portal * Portal
conv * waProto . Conversation
msgs [ ] * waProto . WebMessageInfo
}
2022-02-10 19:18:49 +02:00
type wrappedInfo struct {
* types . MessageInfo
2022-03-05 21:22:31 +02:00
Type database . MessageType
2022-02-10 19:18:49 +02:00
Error database . MessageErrorType
2022-03-14 23:48:41 +02:00
ExpirationStart uint64
ExpiresIn uint32
2022-02-10 19:18:49 +02:00
}
2021-11-03 21:34:06 +02:00
type conversationList [ ] * waProto . Conversation
var _ sort . Interface = ( conversationList ) ( nil )
func ( c conversationList ) Len ( ) int {
return len ( c )
}
func ( c conversationList ) Less ( i , j int ) bool {
2021-11-04 00:09:23 +02:00
return getConversationTimestamp ( c [ i ] ) < getConversationTimestamp ( c [ j ] )
2021-11-03 21:34:06 +02:00
}
func ( c conversationList ) Swap ( i , j int ) {
c [ i ] , c [ j ] = c [ j ] , c [ i ]
}
func ( user * User ) handleHistorySyncsLoop ( ) {
for evt := range user . historySyncs {
2021-11-08 22:56:34 +02:00
go user . sendBridgeState ( BridgeState { StateEvent : StateBackfilling } )
2021-11-03 21:34:06 +02:00
user . handleHistorySync ( evt . Data )
2021-11-08 22:56:34 +02:00
if len ( user . historySyncs ) == 0 && user . IsConnected ( ) {
go user . sendBridgeState ( BridgeState { StateEvent : StateConnected } )
}
2021-11-03 21:34:06 +02:00
}
}
func ( user * User ) handleHistorySync ( evt * waProto . HistorySync ) {
2021-11-04 01:35:17 +02:00
if evt == nil || evt . SyncType == nil || evt . GetSyncType ( ) == waProto . HistorySync_INITIAL_STATUS_V3 || evt . GetSyncType ( ) == waProto . HistorySync_PUSH_NAME {
2021-11-03 21:34:06 +02:00
return
}
2022-03-14 23:53:50 +02:00
description := fmt . Sprintf ( "type %s, %d conversations, chunk order %d, progress %d%%" , evt . GetSyncType ( ) , len ( evt . GetConversations ( ) ) , evt . GetChunkOrder ( ) , evt . GetProgress ( ) )
user . log . Infoln ( "Handling history sync with" , description )
2021-11-03 21:34:06 +02:00
conversations := conversationList ( evt . GetConversations ( ) )
// We want to handle recent conversations first
sort . Sort ( sort . Reverse ( conversations ) )
portalsToBackfill := make ( chan portalToBackfill , len ( conversations ) )
2022-03-14 23:53:50 +02:00
var backfillWait sync . WaitGroup
2021-11-03 21:34:06 +02:00
backfillWait . Add ( 1 )
2022-03-14 23:53:50 +02:00
go user . backfillLoop ( portalsToBackfill , backfillWait . Done )
for _ , conv := range conversations {
user . handleHistorySyncConversation ( conv , portalsToBackfill )
2021-11-03 21:34:06 +02:00
}
close ( portalsToBackfill )
backfillWait . Wait ( )
2022-03-14 23:53:50 +02:00
user . log . Infoln ( "Finished handling history sync with" , description )
2021-11-03 21:34:06 +02:00
}
2022-03-14 23:53:50 +02:00
func ( user * User ) backfillLoop ( ch chan portalToBackfill , done func ( ) ) {
2021-11-03 21:34:06 +02:00
defer done ( )
for ptb := range ch {
2021-11-03 22:04:03 +02:00
if len ( ptb . msgs ) > 0 {
user . log . Debugln ( "Bridging history sync payload for" , ptb . portal . Key . JID )
ptb . portal . backfill ( user , ptb . msgs )
2021-11-04 00:09:23 +02:00
} else {
user . log . Debugfln ( "Not backfilling %s: no bridgeable messages found" , ptb . portal . Key . JID )
2021-11-03 22:04:03 +02:00
}
2021-11-03 21:34:06 +02:00
if ! ptb . conv . GetMarkedAsUnread ( ) && ptb . conv . GetUnreadCount ( ) == 0 {
user . markSelfReadFull ( ptb . portal )
}
}
}
2022-03-14 23:53:50 +02:00
func ( user * User ) handleHistorySyncConversation ( conv * waProto . Conversation , portalsToBackfill chan portalToBackfill ) {
2021-11-03 21:34:06 +02:00
jid , err := types . ParseJID ( conv . GetId ( ) )
if err != nil {
user . log . Warnfln ( "Failed to parse chat JID '%s' in history sync: %v" , conv . GetId ( ) , err )
return
}
// Update the client store with basic chat settings.
muteEnd := time . Unix ( int64 ( conv . GetMuteEndTime ( ) ) , 0 )
if muteEnd . After ( time . Now ( ) ) {
_ = user . Client . Store . ChatSettings . PutMutedUntil ( jid , muteEnd )
}
if conv . GetArchived ( ) {
_ = user . Client . Store . ChatSettings . PutArchived ( jid , true )
}
if conv . GetPinned ( ) > 0 {
_ = user . Client . Store . ChatSettings . PutPinned ( jid , true )
}
portal := user . GetPortalByJID ( jid )
2022-01-19 16:23:25 +02:00
if conv . EphemeralExpiration != nil && portal . ExpirationTime != conv . GetEphemeralExpiration ( ) {
portal . ExpirationTime = conv . GetEphemeralExpiration ( )
portal . Update ( )
}
2021-11-03 21:34:06 +02:00
// Check if portal is too old or doesn't contain anything we can bridge.
if ! user . shouldCreatePortalForHistorySync ( conv , portal ) {
return
}
var msgs [ ] * waProto . WebMessageInfo
if user . bridge . Config . Bridge . HistorySync . Backfill {
msgs = filterMessagesToBackfill ( conv . GetMessages ( ) )
}
ptb := portalToBackfill { portal : portal , conv : conv , msgs : msgs }
if len ( portal . MXID ) == 0 {
user . log . Debugln ( "Creating portal for" , portal . Key . JID , "as part of history sync handling" )
err = portal . CreateMatrixRoom ( user , getPartialInfoFromConversation ( jid , conv ) , false )
if err != nil {
user . log . Warnfln ( "Failed to create room for %s during backfill: %v" , portal . Key . JID , err )
return
}
2021-12-29 21:40:08 +02:00
} else {
portal . UpdateMatrixRoom ( user , nil )
2021-11-03 21:34:06 +02:00
}
if ! user . bridge . Config . Bridge . HistorySync . Backfill {
user . log . Debugln ( "Backfill is disabled, not bridging history sync payload for" , portal . Key . JID )
} else {
portalsToBackfill <- ptb
}
}
2021-11-04 00:09:23 +02:00
func getConversationTimestamp ( conv * waProto . Conversation ) uint64 {
convTs := conv . GetConversationTimestamp ( )
if convTs == 0 && len ( conv . GetMessages ( ) ) > 0 {
convTs = conv . Messages [ 0 ] . GetMessage ( ) . GetMessageTimestamp ( )
}
return convTs
}
2021-11-03 21:34:06 +02:00
func ( user * User ) shouldCreatePortalForHistorySync ( conv * waProto . Conversation , portal * Portal ) bool {
maxAge := user . bridge . Config . Bridge . HistorySync . MaxAge
minLastMsgToCreate := time . Now ( ) . Add ( - time . Duration ( maxAge ) * time . Second )
2021-11-04 00:09:23 +02:00
lastMsg := time . Unix ( int64 ( getConversationTimestamp ( conv ) ) , 0 )
2021-11-03 21:34:06 +02:00
if len ( portal . MXID ) > 0 {
user . log . Debugfln ( "Portal for %s already exists, ensuring user is invited" , portal . Key . JID )
portal . ensureUserInvited ( user )
// Portal exists, let backfill continue
return true
} else if ! user . bridge . Config . Bridge . HistorySync . CreatePortals {
user . log . Debugfln ( "Not creating portal for %s: creating rooms from history sync is disabled" , portal . Key . JID )
} else if ! containsSupportedMessages ( conv ) {
user . log . Debugfln ( "Not creating portal for %s: no interesting messages found" , portal . Key . JID )
} else if maxAge > 0 && ! lastMsg . After ( minLastMsgToCreate ) {
user . log . Debugfln ( "Not creating portal for %s: last message older than limit (%s)" , portal . Key . JID , lastMsg )
} else {
// Portal doesn't exist, but should be created
return true
}
// Portal shouldn't be created, reason logged above
return false
}
func filterMessagesToBackfill ( messages [ ] * waProto . HistorySyncMsg ) [ ] * waProto . WebMessageInfo {
filtered := make ( [ ] * waProto . WebMessageInfo , 0 , len ( messages ) )
for _ , msg := range messages {
wmi := msg . GetMessage ( )
msgType := getMessageType ( wmi . GetMessage ( ) )
if msgType == "unknown" || msgType == "ignore" || msgType == "unknown_protocol" {
continue
} else {
filtered = append ( filtered , wmi )
}
}
return filtered
}
func containsSupportedMessages ( conv * waProto . Conversation ) bool {
for _ , msg := range conv . GetMessages ( ) {
if containsSupportedMessage ( msg . GetMessage ( ) . GetMessage ( ) ) {
return true
}
}
return false
}
func getPartialInfoFromConversation ( jid types . JID , conv * waProto . Conversation ) * types . GroupInfo {
// TODO broadcast list info?
if jid . Server != types . GroupServer {
return nil
}
participants := make ( [ ] types . GroupParticipant , len ( conv . GetParticipant ( ) ) )
for i , pcp := range conv . GetParticipant ( ) {
participantJID , _ := types . ParseJID ( pcp . GetUserJid ( ) )
participants [ i ] = types . GroupParticipant {
JID : participantJID ,
IsAdmin : pcp . GetRank ( ) == waProto . GroupParticipant_ADMIN ,
IsSuperAdmin : pcp . GetRank ( ) == waProto . GroupParticipant_SUPERADMIN ,
}
}
return & types . GroupInfo {
JID : jid ,
GroupName : types . GroupName { Name : conv . GetName ( ) } ,
Participants : participants ,
}
}
// endregion
// region Portal backfilling
var (
PortalCreationDummyEvent = event . Type { Type : "fi.mau.dummy.portal_created" , Class : event . MessageEventType }
BackfillEndDummyEvent = event . Type { Type : "fi.mau.dummy.backfill_end" , Class : event . MessageEventType }
PreBackfillDummyEvent = event . Type { Type : "fi.mau.dummy.pre_backfill" , Class : event . MessageEventType }
)
func ( portal * Portal ) backfill ( source * User , messages [ ] * waProto . WebMessageInfo ) {
portal . backfillLock . Lock ( )
defer portal . backfillLock . Unlock ( )
var historyBatch , newBatch mautrix . ReqBatchSend
2022-02-10 19:18:49 +02:00
var historyBatchInfos , newBatchInfos [ ] * wrappedInfo
2021-11-03 21:34:06 +02:00
firstMsgTimestamp := time . Unix ( int64 ( messages [ len ( messages ) - 1 ] . GetMessageTimestamp ( ) ) , 0 )
2022-02-28 23:59:05 +02:00
historyBatch . StateEventsAtStart = make ( [ ] * event . Event , 0 )
newBatch . StateEventsAtStart = make ( [ ] * event . Event , 0 )
2021-11-03 21:34:06 +02:00
addedMembers := make ( map [ id . UserID ] * event . MemberEventContent )
addMember := func ( puppet * Puppet ) {
if _ , alreadyAdded := addedMembers [ puppet . MXID ] ; alreadyAdded {
return
}
mxid := puppet . MXID . String ( )
content := event . MemberEventContent {
Membership : event . MembershipJoin ,
Displayname : puppet . Displayname ,
AvatarURL : puppet . AvatarURL . CUString ( ) ,
}
inviteContent := content
inviteContent . Membership = event . MembershipInvite
historyBatch . StateEventsAtStart = append ( historyBatch . StateEventsAtStart , & event . Event {
Type : event . StateMember ,
Sender : portal . MainIntent ( ) . UserID ,
StateKey : & mxid ,
Timestamp : firstMsgTimestamp . UnixMilli ( ) ,
Content : event . Content { Parsed : & inviteContent } ,
} , & event . Event {
Type : event . StateMember ,
Sender : puppet . MXID ,
StateKey : & mxid ,
Timestamp : firstMsgTimestamp . UnixMilli ( ) ,
Content : event . Content { Parsed : & content } ,
} )
addedMembers [ puppet . MXID ] = & content
}
firstMessage := portal . bridge . DB . Message . GetFirstInChat ( portal . Key )
lastMessage := portal . bridge . DB . Message . GetLastInChat ( portal . Key )
var historyMaxTs , newMinTs time . Time
if portal . FirstEventID != "" || portal . NextBatchID != "" {
historyBatch . PrevEventID = portal . FirstEventID
historyBatch . BatchID = portal . NextBatchID
if firstMessage == nil && lastMessage == nil {
historyMaxTs = time . Now ( )
} else {
historyMaxTs = firstMessage . Timestamp
}
}
if lastMessage != nil {
newBatch . PrevEventID = lastMessage . MXID
newMinTs = lastMessage . Timestamp
}
portal . log . Infofln ( "Processing history sync with %d messages" , len ( messages ) )
// The messages are ordered newest to oldest, so iterate them in reverse order.
for i := len ( messages ) - 1 ; i >= 0 ; i -- {
webMsg := messages [ i ]
msgType := getMessageType ( webMsg . GetMessage ( ) )
if msgType == "unknown" || msgType == "ignore" || msgType == "unknown_protocol" {
if msgType != "ignore" {
portal . log . Debugfln ( "Skipping message %s with unknown type in backfill" , webMsg . GetKey ( ) . GetId ( ) )
}
continue
}
2022-02-10 19:41:34 +02:00
info := portal . parseWebMessageInfo ( source , webMsg )
2021-11-03 21:34:06 +02:00
if info == nil {
continue
}
var batch * mautrix . ReqBatchSend
2022-02-10 19:18:49 +02:00
var infos * [ ] * wrappedInfo
2021-11-03 21:34:06 +02:00
if ! historyMaxTs . IsZero ( ) && info . Timestamp . Before ( historyMaxTs ) {
2021-11-06 21:30:27 +02:00
batch , infos = & historyBatch , & historyBatchInfos
2021-11-03 21:34:06 +02:00
} else if ! newMinTs . IsZero ( ) && info . Timestamp . After ( newMinTs ) {
batch , infos = & newBatch , & newBatchInfos
} else {
continue
}
2021-11-09 23:44:35 +02:00
if webMsg . GetPushName ( ) != "" && webMsg . GetPushName ( ) != "-" {
2021-11-09 23:59:28 +02:00
existingContact , _ := source . Client . Store . Contacts . GetContact ( info . Sender )
if ! existingContact . Found || existingContact . PushName == "" {
changed , _ , err := source . Client . Store . Contacts . PutPushName ( info . Sender , webMsg . GetPushName ( ) )
if err != nil {
source . log . Errorfln ( "Failed to save push name of %s from historical message in device store: %v" , info . Sender , err )
} else if changed {
source . log . Debugfln ( "Got push name %s for %s from historical message" , webMsg . GetPushName ( ) , info . Sender )
}
2021-11-09 23:44:35 +02:00
}
}
2021-11-03 21:34:06 +02:00
puppet := portal . getMessagePuppet ( source , info )
2021-11-06 14:20:56 +02:00
intent := puppet . IntentFor ( portal )
2021-11-06 15:33:27 +02:00
if intent . IsCustomPuppet && ! portal . bridge . Config . CanDoublePuppetBackfill ( puppet . CustomMXID ) {
2021-11-03 21:34:06 +02:00
intent = puppet . DefaultIntent ( )
}
converted := portal . convertMessage ( intent , source , info , webMsg . GetMessage ( ) )
if converted == nil {
portal . log . Debugfln ( "Skipping unsupported message %s in backfill" , info . ID )
continue
}
2021-11-06 21:28:42 +02:00
if ! intent . IsCustomPuppet && ! portal . bridge . StateStore . IsInRoom ( portal . MXID , puppet . MXID ) {
2021-11-03 21:34:06 +02:00
addMember ( puppet )
}
2021-11-05 11:47:51 +02:00
// TODO this won't work for history
if len ( converted . ReplyTo ) > 0 {
portal . SetReply ( converted . Content , converted . ReplyTo )
}
2022-03-14 23:48:41 +02:00
err := portal . appendBatchEvents ( converted , info , webMsg . GetEphemeralStartTimestamp ( ) , & batch . Events , infos )
2021-11-03 21:34:06 +02:00
if err != nil {
portal . log . Errorfln ( "Error handling message %s during backfill: %v" , info . ID , err )
}
}
if ( len ( historyBatch . Events ) > 0 && len ( historyBatch . BatchID ) == 0 ) || len ( newBatch . Events ) > 0 {
portal . log . Debugln ( "Sending a dummy event to avoid forward extremity errors with backfill" )
_ , err := portal . MainIntent ( ) . SendMessageEvent ( portal . MXID , PreBackfillDummyEvent , struct { } { } )
if err != nil {
portal . log . Warnln ( "Error sending pre-backfill dummy event:" , err )
}
}
if len ( historyBatch . Events ) > 0 && len ( historyBatch . PrevEventID ) > 0 {
portal . log . Infofln ( "Sending %d historical messages..." , len ( historyBatch . Events ) )
historyResp , err := portal . MainIntent ( ) . BatchSend ( portal . MXID , & historyBatch )
if err != nil {
portal . log . Errorln ( "Error sending batch of historical messages:" , err )
} else {
portal . finishBatch ( historyResp . EventIDs , historyBatchInfos )
portal . NextBatchID = historyResp . NextBatchID
portal . Update ( )
// If batchID is non-empty, it means this is backfilling very old messages, and we don't need a post-backfill dummy.
if historyBatch . BatchID == "" {
portal . sendPostBackfillDummy ( time . UnixMilli ( historyBatch . Events [ len ( historyBatch . Events ) - 1 ] . Timestamp ) )
}
}
}
if len ( newBatch . Events ) > 0 && len ( newBatch . PrevEventID ) > 0 {
portal . log . Infofln ( "Sending %d new messages..." , len ( newBatch . Events ) )
newResp , err := portal . MainIntent ( ) . BatchSend ( portal . MXID , & newBatch )
if err != nil {
portal . log . Errorln ( "Error sending batch of new messages:" , err )
} else {
portal . finishBatch ( newResp . EventIDs , newBatchInfos )
portal . sendPostBackfillDummy ( time . UnixMilli ( newBatch . Events [ len ( newBatch . Events ) - 1 ] . Timestamp ) )
}
}
}
2022-02-10 19:41:34 +02:00
func ( portal * Portal ) parseWebMessageInfo ( source * User , webMsg * waProto . WebMessageInfo ) * types . MessageInfo {
2021-11-03 21:34:06 +02:00
info := types . MessageInfo {
MessageSource : types . MessageSource {
Chat : portal . Key . JID ,
IsFromMe : webMsg . GetKey ( ) . GetFromMe ( ) ,
2021-11-09 23:44:35 +02:00
IsGroup : portal . Key . JID . Server == types . GroupServer ,
2021-11-03 21:34:06 +02:00
} ,
ID : webMsg . GetKey ( ) . GetId ( ) ,
PushName : webMsg . GetPushName ( ) ,
Timestamp : time . Unix ( int64 ( webMsg . GetMessageTimestamp ( ) ) , 0 ) ,
}
var err error
if info . IsFromMe {
2022-02-10 19:41:34 +02:00
info . Sender = source . JID . ToNonAD ( )
2021-11-03 21:34:06 +02:00
} else if portal . IsPrivateChat ( ) {
info . Sender = portal . Key . JID
} else if webMsg . GetParticipant ( ) != "" {
info . Sender , err = types . ParseJID ( webMsg . GetParticipant ( ) )
} else if webMsg . GetKey ( ) . GetParticipant ( ) != "" {
info . Sender , err = types . ParseJID ( webMsg . GetKey ( ) . GetParticipant ( ) )
}
if info . Sender . IsEmpty ( ) {
portal . log . Warnfln ( "Failed to get sender of message %s (parse error: %v)" , info . ID , err )
return nil
}
return & info
}
2022-03-14 23:48:41 +02:00
func ( portal * Portal ) appendBatchEvents ( converted * ConvertedMessage , info * types . MessageInfo , expirationStart uint64 , eventsArray * [ ] * event . Event , infoArray * [ ] * wrappedInfo ) error {
2021-11-03 21:34:06 +02:00
mainEvt , err := portal . wrapBatchEvent ( info , converted . Intent , converted . Type , converted . Content , converted . Extra )
if err != nil {
return err
}
if converted . Caption != nil {
captionEvt , err := portal . wrapBatchEvent ( info , converted . Intent , converted . Type , converted . Caption , nil )
if err != nil {
return err
}
* eventsArray = append ( * eventsArray , mainEvt , captionEvt )
2022-03-14 23:48:41 +02:00
* infoArray = append ( * infoArray , & wrappedInfo { info , database . MsgNormal , converted . Error , expirationStart , converted . ExpiresIn } , nil )
2021-11-03 21:34:06 +02:00
} else {
* eventsArray = append ( * eventsArray , mainEvt )
2022-03-14 23:48:41 +02:00
* infoArray = append ( * infoArray , & wrappedInfo { info , database . MsgNormal , converted . Error , expirationStart , converted . ExpiresIn } )
2021-11-03 21:34:06 +02:00
}
2022-01-03 16:11:39 +02:00
if converted . MultiEvent != nil {
for _ , subEvtContent := range converted . MultiEvent {
subEvt , err := portal . wrapBatchEvent ( info , converted . Intent , converted . Type , subEvtContent , nil )
if err != nil {
return err
}
* eventsArray = append ( * eventsArray , subEvt )
* infoArray = append ( * infoArray , nil )
}
}
2021-11-03 21:34:06 +02:00
return nil
}
const backfillIDField = "fi.mau.whatsapp.backfill_msg_id"
func ( portal * Portal ) wrapBatchEvent ( info * types . MessageInfo , intent * appservice . IntentAPI , eventType event . Type , content * event . MessageEventContent , extraContent map [ string ] interface { } ) ( * event . Event , error ) {
if extraContent == nil {
extraContent = map [ string ] interface { } { }
}
extraContent [ backfillIDField ] = info . ID
if intent . IsCustomPuppet {
2021-12-15 13:51:20 +02:00
extraContent [ doublePuppetKey ] = doublePuppetValue
2021-11-03 21:34:06 +02:00
}
wrappedContent := event . Content {
Parsed : content ,
Raw : extraContent ,
}
newEventType , err := portal . encrypt ( & wrappedContent , eventType )
if err != nil {
return nil , err
}
return & event . Event {
Sender : intent . UserID ,
Type : newEventType ,
Timestamp : info . Timestamp . UnixMilli ( ) ,
Content : wrappedContent ,
} , nil
}
2022-02-10 19:18:49 +02:00
func ( portal * Portal ) finishBatch ( eventIDs [ ] id . EventID , infos [ ] * wrappedInfo ) {
2021-11-03 21:34:06 +02:00
if len ( eventIDs ) != len ( infos ) {
portal . log . Errorfln ( "Length of event IDs (%d) and message infos (%d) doesn't match! Using slow path for mapping event IDs" , len ( eventIDs ) , len ( infos ) )
2022-02-10 19:18:49 +02:00
infoMap := make ( map [ types . MessageID ] * wrappedInfo , len ( infos ) )
2021-11-03 21:34:06 +02:00
for _ , info := range infos {
infoMap [ info . ID ] = info
}
for _ , eventID := range eventIDs {
if evt , err := portal . MainIntent ( ) . GetEvent ( portal . MXID , eventID ) ; err != nil {
portal . log . Warnfln ( "Failed to get event %s to register it in the database: %v" , eventID , err )
} else if msgID , ok := evt . Content . Raw [ backfillIDField ] . ( string ) ; ! ok {
portal . log . Warnfln ( "Event %s doesn't include the WhatsApp message ID" , eventID )
} else if info , ok := infoMap [ types . MessageID ( msgID ) ] ; ! ok {
portal . log . Warnfln ( "Didn't find info of message %s (event %s) to register it in the database" , msgID , eventID )
} else {
2022-03-14 23:48:41 +02:00
portal . finishBatchEvt ( info , eventID )
2021-11-03 21:34:06 +02:00
}
}
} else {
for i := 0 ; i < len ( infos ) ; i ++ {
2022-03-14 23:48:41 +02:00
portal . finishBatchEvt ( infos [ i ] , eventIDs [ i ] )
2021-11-03 21:34:06 +02:00
}
portal . log . Infofln ( "Successfully sent %d events" , len ( eventIDs ) )
}
}
2022-03-14 23:48:41 +02:00
func ( portal * Portal ) finishBatchEvt ( info * wrappedInfo , eventID id . EventID ) {
if info == nil {
return
}
portal . markHandled ( nil , info . MessageInfo , eventID , true , false , info . Type , info . Error )
if info . ExpiresIn > 0 {
if info . ExpirationStart > 0 {
remainingSeconds := time . Unix ( int64 ( info . ExpirationStart ) , 0 ) . Add ( time . Duration ( info . ExpiresIn ) * time . Second ) . Sub ( time . Now ( ) ) . Seconds ( )
portal . log . Debugfln ( "Disappearing history sync message: expires in %d, started at %d, remaining %d" , info . ExpiresIn , info . ExpirationStart , int ( remainingSeconds ) )
portal . MarkDisappearing ( eventID , uint32 ( remainingSeconds ) , true )
} else {
portal . log . Debugfln ( "Disappearing history sync message: expires in %d (not started)" , info . ExpiresIn )
portal . MarkDisappearing ( eventID , info . ExpiresIn , false )
}
}
}
2021-11-03 21:34:06 +02:00
func ( portal * Portal ) sendPostBackfillDummy ( lastTimestamp time . Time ) {
resp , err := portal . MainIntent ( ) . SendMessageEvent ( portal . MXID , BackfillEndDummyEvent , struct { } { } )
if err != nil {
portal . log . Errorln ( "Error sending post-backfill dummy event:" , err )
return
}
msg := portal . bridge . DB . Message . New ( )
msg . Chat = portal . Key
msg . MXID = resp . EventID
msg . JID = types . MessageID ( resp . EventID )
msg . Timestamp = lastTimestamp . Add ( 1 * time . Second )
msg . Sent = true
msg . Insert ( )
}
// endregion