forked from MirrorHub/mautrix-whatsapp
f2e762680c
This adds a new backfill type for media that sends a request to the phone for every media that is not available on the WA servers. WA deletes media from their servers after about two weeks, so you have to ask the phone to re-upload it. In order to use this, you need to enable bridge.history_sync.backfill_media and configure the requests that will be made per portal using bridge.history_sync.media (which is similar to the deferred backfill config). If you already have backfilled portals, but want to do a one-off media backfill for all existing portals, you can set bridge.history_sync.enqueue_backfill_media_next_start to true.
69 lines
2.2 KiB
Go
69 lines
2.2 KiB
Go
// mautrix-whatsapp - A Matrix-WhatsApp puppeting bridge.
|
|
// Copyright (C) 2021 Tulir Asokan, Sumner Evans
|
|
//
|
|
// 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 (
|
|
"time"
|
|
|
|
log "maunium.net/go/maulogger/v2"
|
|
"maunium.net/go/mautrix-whatsapp/database"
|
|
)
|
|
|
|
type BackfillQueue struct {
|
|
BackfillQuery *database.BackfillQuery
|
|
ImmediateBackfillRequests chan *database.Backfill
|
|
DeferredBackfillRequests chan *database.Backfill
|
|
ReCheckQueue chan bool
|
|
|
|
log log.Logger
|
|
}
|
|
|
|
func (bq *BackfillQueue) RunLoops(user *User) {
|
|
go bq.immediateBackfillLoop(user)
|
|
bq.deferredBackfillLoop(user)
|
|
}
|
|
|
|
func (bq *BackfillQueue) immediateBackfillLoop(user *User) {
|
|
for {
|
|
if backfill := bq.BackfillQuery.GetNext(user.MXID, database.BackfillImmediate); backfill != nil {
|
|
bq.ImmediateBackfillRequests <- backfill
|
|
backfill.MarkDone()
|
|
} else {
|
|
select {
|
|
case <-bq.ReCheckQueue:
|
|
case <-time.After(10 * time.Second):
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func (bq *BackfillQueue) deferredBackfillLoop(user *User) {
|
|
for {
|
|
// Finish all immediate backfills before doing the deferred ones.
|
|
if immediate := bq.BackfillQuery.GetNext(user.MXID, database.BackfillImmediate); immediate != nil {
|
|
time.Sleep(10 * time.Second)
|
|
} else if backfill := bq.BackfillQuery.GetNext(user.MXID, database.BackfillDeferred); backfill != nil {
|
|
bq.DeferredBackfillRequests <- backfill
|
|
backfill.MarkDone()
|
|
} else if mediaBackfill := bq.BackfillQuery.GetNext(user.MXID, database.BackfillMedia); mediaBackfill != nil {
|
|
bq.DeferredBackfillRequests <- mediaBackfill
|
|
mediaBackfill.MarkDone()
|
|
} else {
|
|
time.Sleep(10 * time.Second)
|
|
}
|
|
}
|
|
}
|