Async incremental sync

This commit is contained in:
Till Faelligen 2022-10-28 10:01:19 +02:00
parent 9d6c9e4cc3
commit 2999559982
No known key found for this signature in database
GPG key ID: 3DF82D8AB9211D4E
2 changed files with 78 additions and 93 deletions

View file

@ -130,3 +130,29 @@ func ToToken(provider StreamProvider, position types.StreamPosition) types.Strea
} }
return types.StreamingToken{} return types.StreamingToken{}
} }
func IncrementalPositions(provider StreamProvider, current, since types.StreamingToken) (types.StreamPosition, types.StreamPosition) {
switch t := provider.(type) {
case *PDUStreamProvider:
return current.PDUPosition, since.PDUPosition
case *TypingStreamProvider:
return current.TypingPosition, since.TypingPosition
case *ReceiptStreamProvider:
return current.ReceiptPosition, since.ReceiptPosition
case *SendToDeviceStreamProvider:
return current.SendToDevicePosition, since.SendToDevicePosition
case *InviteStreamProvider:
return current.InvitePosition, since.InvitePosition
case *AccountDataStreamProvider:
return current.AccountDataPosition, since.AccountDataPosition
case *DeviceListStreamProvider:
return current.DeviceListPosition, since.DeviceListPosition
case *NotificationDataStreamProvider:
return current.NotificationDataPosition, since.NotificationDataPosition
case *PresenceStreamProvider:
return current.PresencePosition, since.PresencePosition
default:
panic(fmt.Sprintf("unknown stream provider: %T", t))
}
return 0, 0
}

View file

@ -329,6 +329,17 @@ func (rp *RequestPool) OnIncomingSyncRequest(req *http.Request, device *userapi.
return f(snapshot) return f(snapshot)
} }
allStreams := []streams.StreamProvider{
rp.streams.DeviceListStreamProvider,
rp.streams.TypingStreamProvider,
rp.streams.ReceiptStreamProvider,
rp.streams.InviteStreamProvider,
rp.streams.SendToDeviceStreamProvider,
rp.streams.AccountDataStreamProvider,
rp.streams.NotificationDataStreamProvider,
rp.streams.PresenceStreamProvider,
}
if syncReq.Since.IsEmpty() { if syncReq.Since.IsEmpty() {
// Complete sync // Complete sync
// The PDU stream needs to be the very first stream to get the data, // The PDU stream needs to be the very first stream to get the data,
@ -342,16 +353,6 @@ func (rp *RequestPool) OnIncomingSyncRequest(req *http.Request, device *userapi.
}, },
) )
syncReq.Response.NextBatch.PDUPosition = pduPos syncReq.Response.NextBatch.PDUPosition = pduPos
allStreams := []streams.StreamProvider{
rp.streams.DeviceListStreamProvider,
rp.streams.TypingStreamProvider,
rp.streams.ReceiptStreamProvider,
rp.streams.InviteStreamProvider,
rp.streams.SendToDeviceStreamProvider,
rp.streams.AccountDataStreamProvider,
rp.streams.NotificationDataStreamProvider,
rp.streams.PresenceStreamProvider,
}
streamPosCh := make(chan streamPosResponse, len(allStreams)) streamPosCh := make(chan streamPosResponse, len(allStreams))
wg := sync.WaitGroup{} wg := sync.WaitGroup{}
@ -361,7 +362,7 @@ func (rp *RequestPool) OnIncomingSyncRequest(req *http.Request, device *userapi.
for _, s := range allStreams { for _, s := range allStreams {
go func(stream streams.StreamProvider) { go func(stream streams.StreamProvider) {
streamPos := withTransaction( streamPos := withTransaction(
0, 0, // we're doing an initial sync
func(txn storage.DatabaseTransaction) types.StreamPosition { func(txn storage.DatabaseTransaction) types.StreamPosition {
return stream.CompleteSync( return stream.CompleteSync(
syncReq.Context, txn, syncReq, syncReq.Context, txn, syncReq,
@ -380,88 +381,46 @@ func (rp *RequestPool) OnIncomingSyncRequest(req *http.Request, device *userapi.
} }
} else { } else {
// Incremental sync // Incremental sync
syncReq.Response.NextBatch = types.StreamingToken{ // The PDU stream needs to be the very first stream to get the data,
PDUPosition: withTransaction( // as it sets values the other streams need
syncReq.Since.PDUPosition, current, since := streams.IncrementalPositions(rp.streams.PDUStreamProvider, rp.Notifier.CurrentPosition(), syncReq.Since)
func(txn storage.DatabaseTransaction) types.StreamPosition { pduPos := withTransaction(
return rp.streams.PDUStreamProvider.IncrementalSync( since,
syncReq.Context, txn, syncReq, func(txn storage.DatabaseTransaction) types.StreamPosition {
syncReq.Since.PDUPosition, rp.Notifier.CurrentPosition().PDUPosition, return rp.streams.PDUStreamProvider.IncrementalSync(
) syncReq.Context, txn, syncReq,
}, since, current,
), )
TypingPosition: withTransaction( },
syncReq.Since.TypingPosition, )
func(txn storage.DatabaseTransaction) types.StreamPosition { syncReq.Response.NextBatch.PDUPosition = pduPos
return rp.streams.TypingStreamProvider.IncrementalSync(
syncReq.Context, txn, syncReq, streamPosCh := make(chan streamPosResponse, len(allStreams))
syncReq.Since.TypingPosition, rp.Notifier.CurrentPosition().TypingPosition, wg := sync.WaitGroup{}
) wg.Add(len(allStreams))
},
), // fan out stream calculations
ReceiptPosition: withTransaction( for _, s := range allStreams {
syncReq.Since.ReceiptPosition, go func(stream streams.StreamProvider) {
func(txn storage.DatabaseTransaction) types.StreamPosition { current, since := streams.IncrementalPositions(stream, rp.Notifier.CurrentPosition(), syncReq.Since)
return rp.streams.ReceiptStreamProvider.IncrementalSync( streamPos := withTransaction(
syncReq.Context, txn, syncReq, since,
syncReq.Since.ReceiptPosition, rp.Notifier.CurrentPosition().ReceiptPosition, func(txn storage.DatabaseTransaction) types.StreamPosition {
) return stream.IncrementalSync(
}, syncReq.Context, txn, syncReq,
), since, current,
InvitePosition: withTransaction( )
syncReq.Since.InvitePosition, },
func(txn storage.DatabaseTransaction) types.StreamPosition { )
return rp.streams.InviteStreamProvider.IncrementalSync( streamPosCh <- streamPosResponse{provider: stream, pos: streamPos}
syncReq.Context, txn, syncReq, wg.Done()
syncReq.Since.InvitePosition, rp.Notifier.CurrentPosition().InvitePosition, }(s)
) }
}, // Wait for all streams to finish their work
), wg.Wait()
SendToDevicePosition: withTransaction( close(streamPosCh)
syncReq.Since.SendToDevicePosition, for resp := range streamPosCh {
func(txn storage.DatabaseTransaction) types.StreamPosition { syncReq.Response.NextBatch.ApplyUpdates(streams.ToToken(resp.provider, resp.pos))
return rp.streams.SendToDeviceStreamProvider.IncrementalSync(
syncReq.Context, txn, syncReq,
syncReq.Since.SendToDevicePosition, rp.Notifier.CurrentPosition().SendToDevicePosition,
)
},
),
AccountDataPosition: withTransaction(
syncReq.Since.AccountDataPosition,
func(txn storage.DatabaseTransaction) types.StreamPosition {
return rp.streams.AccountDataStreamProvider.IncrementalSync(
syncReq.Context, txn, syncReq,
syncReq.Since.AccountDataPosition, rp.Notifier.CurrentPosition().AccountDataPosition,
)
},
),
NotificationDataPosition: withTransaction(
syncReq.Since.NotificationDataPosition,
func(txn storage.DatabaseTransaction) types.StreamPosition {
return rp.streams.NotificationDataStreamProvider.IncrementalSync(
syncReq.Context, txn, syncReq,
syncReq.Since.NotificationDataPosition, rp.Notifier.CurrentPosition().NotificationDataPosition,
)
},
),
DeviceListPosition: withTransaction(
syncReq.Since.DeviceListPosition,
func(txn storage.DatabaseTransaction) types.StreamPosition {
return rp.streams.DeviceListStreamProvider.IncrementalSync(
syncReq.Context, txn, syncReq,
syncReq.Since.DeviceListPosition, rp.Notifier.CurrentPosition().DeviceListPosition,
)
},
),
PresencePosition: withTransaction(
syncReq.Since.PresencePosition,
func(txn storage.DatabaseTransaction) types.StreamPosition {
return rp.streams.PresenceStreamProvider.IncrementalSync(
syncReq.Context, txn, syncReq,
syncReq.Since.PresencePosition, rp.Notifier.CurrentPosition().PresencePosition,
)
},
),
} }
// it's possible for there to be no updates for this user even though since < current pos, // it's possible for there to be no updates for this user even though since < current pos,
// e.g busy servers with a quiet user. In this scenario, we don't want to return a no-op // e.g busy servers with a quiet user. In this scenario, we don't want to return a no-op