mirror of
https://github.com/matrix-org/dendrite.git
synced 2025-12-26 08:13:09 -06:00
More lightweight notifications
This commit is contained in:
parent
875cb2c379
commit
a6549669ca
|
|
@ -6,18 +6,25 @@ import (
|
||||||
|
|
||||||
"github.com/matrix-org/dendrite/syncapi/storage"
|
"github.com/matrix-org/dendrite/syncapi/storage"
|
||||||
"github.com/matrix-org/dendrite/syncapi/types"
|
"github.com/matrix-org/dendrite/syncapi/types"
|
||||||
|
userapi "github.com/matrix-org/dendrite/userapi/api"
|
||||||
)
|
)
|
||||||
|
|
||||||
type PartitionedStreamProvider struct {
|
type PartitionedStreamProvider struct {
|
||||||
DB storage.Database
|
DB storage.Database
|
||||||
latest types.LogPosition
|
latest types.LogPosition
|
||||||
latestMutex sync.RWMutex
|
latestMutex sync.RWMutex
|
||||||
update *sync.Cond
|
subscriptions map[string]*partitionedStreamSubscription // userid+deviceid
|
||||||
|
subscriptionsMutex sync.Mutex
|
||||||
|
}
|
||||||
|
|
||||||
|
type partitionedStreamSubscription struct {
|
||||||
|
ctx context.Context
|
||||||
|
from types.LogPosition
|
||||||
|
ch chan struct{}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *PartitionedStreamProvider) Setup() {
|
func (p *PartitionedStreamProvider) Setup() {
|
||||||
locker := &sync.Mutex{}
|
p.subscriptions = make(map[string]*partitionedStreamSubscription)
|
||||||
p.update = sync.NewCond(locker)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *PartitionedStreamProvider) Advance(
|
func (p *PartitionedStreamProvider) Advance(
|
||||||
|
|
@ -28,7 +35,22 @@ func (p *PartitionedStreamProvider) Advance(
|
||||||
|
|
||||||
if latest.IsAfter(&p.latest) {
|
if latest.IsAfter(&p.latest) {
|
||||||
p.latest = latest
|
p.latest = latest
|
||||||
p.update.Broadcast()
|
|
||||||
|
p.subscriptionsMutex.Lock()
|
||||||
|
for id, s := range p.subscriptions {
|
||||||
|
select {
|
||||||
|
case <-s.ctx.Done():
|
||||||
|
close(s.ch)
|
||||||
|
delete(p.subscriptions, id)
|
||||||
|
continue
|
||||||
|
default:
|
||||||
|
if latest.IsAfter(&s.from) {
|
||||||
|
close(s.ch)
|
||||||
|
delete(p.subscriptions, id)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
p.subscriptionsMutex.Unlock()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -43,6 +65,7 @@ func (p *PartitionedStreamProvider) LatestPosition(
|
||||||
|
|
||||||
func (p *PartitionedStreamProvider) NotifyAfter(
|
func (p *PartitionedStreamProvider) NotifyAfter(
|
||||||
ctx context.Context,
|
ctx context.Context,
|
||||||
|
device *userapi.Device,
|
||||||
from types.LogPosition,
|
from types.LogPosition,
|
||||||
) chan struct{} {
|
) chan struct{} {
|
||||||
ch := make(chan struct{})
|
ch := make(chan struct{})
|
||||||
|
|
@ -63,32 +86,13 @@ func (p *PartitionedStreamProvider) NotifyAfter(
|
||||||
return ch
|
return ch
|
||||||
}
|
}
|
||||||
|
|
||||||
// If we haven't, then we'll subscribe to updates. The
|
id := device.UserID + device.ID
|
||||||
// sync.Cond will fire every time the latest position
|
p.subscriptionsMutex.Lock()
|
||||||
// updates, so we can check and see if we've advanced
|
if s, ok := p.subscriptions[id]; ok {
|
||||||
// past it.
|
close(s.ch)
|
||||||
go func(p *PartitionedStreamProvider) {
|
|
||||||
p.update.L.Lock()
|
|
||||||
defer p.update.L.Unlock()
|
|
||||||
|
|
||||||
for {
|
|
||||||
select {
|
|
||||||
case <-ctx.Done():
|
|
||||||
// The context has expired, so there's no point
|
|
||||||
// in continuing to wait for the update.
|
|
||||||
close(ch)
|
|
||||||
return
|
|
||||||
default:
|
|
||||||
// The latest position has been advanced. Let's
|
|
||||||
// see if it's advanced to the position we care
|
|
||||||
// about. If it has then we'll return.
|
|
||||||
p.update.Wait()
|
|
||||||
if check() {
|
|
||||||
return
|
|
||||||
}
|
}
|
||||||
}
|
p.subscriptions[id] = &partitionedStreamSubscription{ctx, from, ch}
|
||||||
}
|
p.subscriptionsMutex.Unlock()
|
||||||
}(p)
|
|
||||||
|
|
||||||
return ch
|
return ch
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -6,18 +6,25 @@ import (
|
||||||
|
|
||||||
"github.com/matrix-org/dendrite/syncapi/storage"
|
"github.com/matrix-org/dendrite/syncapi/storage"
|
||||||
"github.com/matrix-org/dendrite/syncapi/types"
|
"github.com/matrix-org/dendrite/syncapi/types"
|
||||||
|
userapi "github.com/matrix-org/dendrite/userapi/api"
|
||||||
)
|
)
|
||||||
|
|
||||||
type StreamProvider struct {
|
type StreamProvider struct {
|
||||||
DB storage.Database
|
DB storage.Database
|
||||||
latest types.StreamPosition
|
latest types.StreamPosition
|
||||||
latestMutex sync.RWMutex
|
latestMutex sync.RWMutex
|
||||||
update *sync.Cond
|
subscriptions map[string]*streamSubscription // userid+deviceid
|
||||||
|
subscriptionsMutex sync.Mutex
|
||||||
|
}
|
||||||
|
|
||||||
|
type streamSubscription struct {
|
||||||
|
ctx context.Context
|
||||||
|
from types.StreamPosition
|
||||||
|
ch chan struct{}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *StreamProvider) Setup() {
|
func (p *StreamProvider) Setup() {
|
||||||
locker := &sync.Mutex{}
|
p.subscriptions = make(map[string]*streamSubscription)
|
||||||
p.update = sync.NewCond(locker)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *StreamProvider) Advance(
|
func (p *StreamProvider) Advance(
|
||||||
|
|
@ -28,7 +35,22 @@ func (p *StreamProvider) Advance(
|
||||||
|
|
||||||
if latest > p.latest {
|
if latest > p.latest {
|
||||||
p.latest = latest
|
p.latest = latest
|
||||||
p.update.Broadcast()
|
|
||||||
|
p.subscriptionsMutex.Lock()
|
||||||
|
for id, s := range p.subscriptions {
|
||||||
|
select {
|
||||||
|
case <-s.ctx.Done():
|
||||||
|
close(s.ch)
|
||||||
|
delete(p.subscriptions, id)
|
||||||
|
continue
|
||||||
|
default:
|
||||||
|
if latest > s.from {
|
||||||
|
close(s.ch)
|
||||||
|
delete(p.subscriptions, id)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
p.subscriptionsMutex.Unlock()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -43,6 +65,7 @@ func (p *StreamProvider) LatestPosition(
|
||||||
|
|
||||||
func (p *StreamProvider) NotifyAfter(
|
func (p *StreamProvider) NotifyAfter(
|
||||||
ctx context.Context,
|
ctx context.Context,
|
||||||
|
device *userapi.Device,
|
||||||
from types.StreamPosition,
|
from types.StreamPosition,
|
||||||
) chan struct{} {
|
) chan struct{} {
|
||||||
ch := make(chan struct{})
|
ch := make(chan struct{})
|
||||||
|
|
@ -63,32 +86,13 @@ func (p *StreamProvider) NotifyAfter(
|
||||||
return ch
|
return ch
|
||||||
}
|
}
|
||||||
|
|
||||||
// If we haven't, then we'll subscribe to updates. The
|
id := device.UserID + device.ID
|
||||||
// sync.Cond will fire every time the latest position
|
p.subscriptionsMutex.Lock()
|
||||||
// updates, so we can check and see if we've advanced
|
if s, ok := p.subscriptions[id]; ok {
|
||||||
// past it.
|
close(s.ch)
|
||||||
go func(p *StreamProvider) {
|
|
||||||
p.update.L.Lock()
|
|
||||||
defer p.update.L.Unlock()
|
|
||||||
|
|
||||||
for {
|
|
||||||
select {
|
|
||||||
case <-ctx.Done():
|
|
||||||
// The context has expired, so there's no point
|
|
||||||
// in continuing to wait for the update.
|
|
||||||
close(ch)
|
|
||||||
return
|
|
||||||
default:
|
|
||||||
// The latest position has been advanced. Let's
|
|
||||||
// see if it's advanced to the position we care
|
|
||||||
// about. If it has then we'll return.
|
|
||||||
p.update.Wait()
|
|
||||||
if check() {
|
|
||||||
return
|
|
||||||
}
|
}
|
||||||
}
|
p.subscriptions[id] = &streamSubscription{ctx, from, ch}
|
||||||
}
|
p.subscriptionsMutex.Unlock()
|
||||||
}(p)
|
|
||||||
|
|
||||||
return ch
|
return ch
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -176,13 +176,13 @@ func (rp *RequestPool) OnIncomingSyncRequest(req *http.Request, device *userapi.
|
||||||
case <-timer.C: // Timeout reached
|
case <-timer.C: // Timeout reached
|
||||||
return giveup()
|
return giveup()
|
||||||
|
|
||||||
case <-rp.streams.PDUStreamProvider.NotifyAfter(waitctx, syncReq.Since.PDUPosition):
|
case <-rp.streams.PDUStreamProvider.NotifyAfter(waitctx, device, syncReq.Since.PDUPosition):
|
||||||
case <-rp.streams.TypingStreamProvider.NotifyAfter(waitctx, syncReq.Since.TypingPosition):
|
case <-rp.streams.TypingStreamProvider.NotifyAfter(waitctx, device, syncReq.Since.TypingPosition):
|
||||||
case <-rp.streams.ReceiptStreamProvider.NotifyAfter(waitctx, syncReq.Since.ReceiptPosition):
|
case <-rp.streams.ReceiptStreamProvider.NotifyAfter(waitctx, device, syncReq.Since.ReceiptPosition):
|
||||||
case <-rp.streams.InviteStreamProvider.NotifyAfter(waitctx, syncReq.Since.InvitePosition):
|
case <-rp.streams.InviteStreamProvider.NotifyAfter(waitctx, device, syncReq.Since.InvitePosition):
|
||||||
case <-rp.streams.SendToDeviceStreamProvider.NotifyAfter(waitctx, syncReq.Since.SendToDevicePosition):
|
case <-rp.streams.SendToDeviceStreamProvider.NotifyAfter(waitctx, device, syncReq.Since.SendToDevicePosition):
|
||||||
case <-rp.streams.AccountDataStreamProvider.NotifyAfter(waitctx, syncReq.Since.AccountDataPosition):
|
case <-rp.streams.AccountDataStreamProvider.NotifyAfter(waitctx, device, syncReq.Since.AccountDataPosition):
|
||||||
case <-rp.streams.DeviceListStreamProvider.NotifyAfter(waitctx, syncReq.Since.DeviceListPosition):
|
case <-rp.streams.DeviceListStreamProvider.NotifyAfter(waitctx, device, syncReq.Since.DeviceListPosition):
|
||||||
}
|
}
|
||||||
|
|
||||||
syncReq.Log.Println("Responding to sync after wakeup")
|
syncReq.Log.Println("Responding to sync after wakeup")
|
||||||
|
|
|
||||||
|
|
@ -42,7 +42,7 @@ type StreamProvider interface {
|
||||||
|
|
||||||
// NotifyAfter returns a channel which will be closed once the
|
// NotifyAfter returns a channel which will be closed once the
|
||||||
// stream advances past the "from" position.
|
// stream advances past the "from" position.
|
||||||
NotifyAfter(ctx context.Context, from StreamPosition) chan struct{}
|
NotifyAfter(ctx context.Context, device *userapi.Device, from StreamPosition) chan struct{}
|
||||||
|
|
||||||
// LatestPosition returns the latest stream position for this stream.
|
// LatestPosition returns the latest stream position for this stream.
|
||||||
LatestPosition(ctx context.Context) StreamPosition
|
LatestPosition(ctx context.Context) StreamPosition
|
||||||
|
|
@ -53,6 +53,6 @@ type PartitionedStreamProvider interface {
|
||||||
Advance(latest LogPosition)
|
Advance(latest LogPosition)
|
||||||
CompleteSync(ctx context.Context, req *SyncRequest) LogPosition
|
CompleteSync(ctx context.Context, req *SyncRequest) LogPosition
|
||||||
IncrementalSync(ctx context.Context, req *SyncRequest, from, to LogPosition) LogPosition
|
IncrementalSync(ctx context.Context, req *SyncRequest, from, to LogPosition) LogPosition
|
||||||
NotifyAfter(ctx context.Context, from LogPosition) chan struct{}
|
NotifyAfter(ctx context.Context, device *userapi.Device, from LogPosition) chan struct{}
|
||||||
LatestPosition(ctx context.Context) LogPosition
|
LatestPosition(ctx context.Context) LogPosition
|
||||||
}
|
}
|
||||||
|
|
|
||||||
Loading…
Reference in a new issue