Refactor to allow passing actual filter

This commit is contained in:
Eric Eastwood 2020-12-24 23:39:46 -06:00
parent 82afb32464
commit 4cf73a5dc2
4 changed files with 40 additions and 51 deletions

View file

@ -64,10 +64,10 @@ type Database interface {
// from when the device sent the event via an API that included a transaction // from when the device sent the event via an API that included a transaction
// ID. A response object must be provided for IncrementaSync to populate - it // ID. A response object must be provided for IncrementaSync to populate - it
// will not create one. // will not create one.
IncrementalSync(ctx context.Context, res *types.Response, device userapi.Device, fromPos, toPos types.StreamingToken, numRecentEventsPerRoom int, wantFullState bool) (*types.Response, error) IncrementalSync(ctx context.Context, res *types.Response, device userapi.Device, fromPos, toPos types.StreamingToken, filter *gomatrixserverlib.Filter, wantFullState bool) (*types.Response, error)
// CompleteSync returns a complete /sync API response for the given user. A response object // CompleteSync returns a complete /sync API response for the given user. A response object
// must be provided for CompleteSync to populate - it will not create one. // must be provided for CompleteSync to populate - it will not create one.
CompleteSync(ctx context.Context, res *types.Response, device userapi.Device, numRecentEventsPerRoom int) (*types.Response, error) CompleteSync(ctx context.Context, res *types.Response, device userapi.Device, filter *gomatrixserverlib.Filter) (*types.Response, error)
// GetAccountDataInRange returns all account data for a given user inserted or // GetAccountDataInRange returns all account data for a given user inserted or
// updated between two given positions // updated between two given positions
// Returns a map following the format data[roomID] = []dataTypes // Returns a map following the format data[roomID] = []dataTypes

View file

@ -685,7 +685,7 @@ func (d *Database) IncrementalSync(
ctx context.Context, res *types.Response, ctx context.Context, res *types.Response,
device userapi.Device, device userapi.Device,
fromPos, toPos types.StreamingToken, fromPos, toPos types.StreamingToken,
numRecentEventsPerRoom int, filter *gomatrixserverlib.Filter,
wantFullState bool, wantFullState bool,
) (*types.Response, error) { ) (*types.Response, error) {
res.NextBatch = fromPos.WithUpdates(toPos) res.NextBatch = fromPos.WithUpdates(toPos)
@ -697,6 +697,9 @@ func (d *Database) IncrementalSync(
From: fromPos.PDUPosition, From: fromPos.PDUPosition,
To: toPos.PDUPosition, To: toPos.PDUPosition,
} }
numRecentEventsPerRoom := filter.Room.Timeline.Limit
joinedRoomIDs, err = d.addPDUDeltaToResponse( joinedRoomIDs, err = d.addPDUDeltaToResponse(
ctx, device, r, numRecentEventsPerRoom, wantFullState, res, ctx, device, r, numRecentEventsPerRoom, wantFullState, res,
) )
@ -761,8 +764,7 @@ func (d *Database) RedactEvent(ctx context.Context, redactedEventID string, reda
func (d *Database) getResponseWithPDUsForCompleteSync( func (d *Database) getResponseWithPDUsForCompleteSync(
ctx context.Context, res *types.Response, ctx context.Context, res *types.Response,
userID string, device userapi.Device, userID string, device userapi.Device,
numRecentEventsPerRoom int, filter *gomatrixserverlib.Filter,
includeLeave bool,
) ( ) (
toPos types.StreamingToken, toPos types.StreamingToken,
joinedRoomIDs []string, joinedRoomIDs []string,
@ -801,13 +803,11 @@ func (d *Database) getResponseWithPDUsForCompleteSync(
return return
} }
stateFilter := gomatrixserverlib.DefaultStateFilter() // TODO: use filter provided in request
// Build up a /sync response. Add joined rooms. // Build up a /sync response. Add joined rooms.
for _, roomID := range joinedRoomIDs { for _, roomID := range joinedRoomIDs {
var jr *types.JoinResponse var jr *types.JoinResponse
jr, err = d.getJoinResponseForCompleteSync( jr, err = d.getJoinResponseForCompleteSync(
ctx, txn, roomID, r, &stateFilter, numRecentEventsPerRoom, device, ctx, txn, roomID, r, filter, device,
) )
if err != nil { if err != nil {
return return
@ -827,7 +827,7 @@ func (d *Database) getResponseWithPDUsForCompleteSync(
if !peek.Deleted { if !peek.Deleted {
var jr *types.JoinResponse var jr *types.JoinResponse
jr, err = d.getJoinResponseForCompleteSync( jr, err = d.getJoinResponseForCompleteSync(
ctx, txn, peek.RoomID, r, &stateFilter, numRecentEventsPerRoom, device, ctx, txn, peek.RoomID, r, filter, device,
) )
if err != nil { if err != nil {
return return
@ -848,14 +848,17 @@ func (d *Database) getJoinResponseForCompleteSync(
ctx context.Context, txn *sql.Tx, ctx context.Context, txn *sql.Tx,
roomID string, roomID string,
r types.Range, r types.Range,
stateFilter *gomatrixserverlib.StateFilter, filter *gomatrixserverlib.Filter,
numRecentEventsPerRoom int, device userapi.Device, device userapi.Device,
) (jr *types.JoinResponse, err error) { ) (jr *types.JoinResponse, err error) {
var stateEvents []*gomatrixserverlib.HeaderedEvent var stateEvents []*gomatrixserverlib.HeaderedEvent
stateEvents, err = d.CurrentRoomState.SelectCurrentState(ctx, txn, roomID, stateFilter) stateEvents, err = d.CurrentRoomState.SelectCurrentState(ctx, txn, roomID, &filter.Room.State)
if err != nil { if err != nil {
return return
} }
numRecentEventsPerRoom := filter.Room.Timeline.Limit
// TODO: When filters are added, we may need to call this multiple times to get enough events. // TODO: When filters are added, we may need to call this multiple times to get enough events.
// See: https://github.com/matrix-org/synapse/blob/v0.19.3/synapse/handlers/sync.py#L316 // See: https://github.com/matrix-org/synapse/blob/v0.19.3/synapse/handlers/sync.py#L316
var recentStreamEvents []types.StreamEvent var recentStreamEvents []types.StreamEvent
@ -924,11 +927,13 @@ func (d *Database) getJoinResponseForCompleteSync(
} }
func (d *Database) CompleteSync( func (d *Database) CompleteSync(
ctx context.Context, res *types.Response, ctx context.Context,
device userapi.Device, numRecentEventsPerRoom int, res *types.Response,
device userapi.Device,
filter *gomatrixserverlib.Filter,
) (*types.Response, error) { ) (*types.Response, error) {
toPos, joinedRoomIDs, err := d.getResponseWithPDUsForCompleteSync( toPos, joinedRoomIDs, err := d.getResponseWithPDUsForCompleteSync(
ctx, res, device.UserID, device, numRecentEventsPerRoom, ctx, res, device.UserID, device, filter,
) )
if err != nil { if err != nil {
return nil, fmt.Errorf("d.getResponseWithPDUsForCompleteSync: %w", err) return nil, fmt.Errorf("d.getResponseWithPDUsForCompleteSync: %w", err)

View file

@ -33,21 +33,11 @@ const defaultSyncTimeout = time.Duration(0)
const defaultIncludeLeave = false const defaultIncludeLeave = false
const DefaultTimelineLimit = 20 const DefaultTimelineLimit = 20
type filter struct {
Room struct {
IncludeLeave *bool `json:"include_leave"`
Timeline struct {
Limit *int `json:"limit"`
} `json:"timeline"`
} `json:"room"`
}
// syncRequest represents a /sync request, with sensible defaults/sanity checks applied. // syncRequest represents a /sync request, with sensible defaults/sanity checks applied.
type syncRequest struct { type syncRequest struct {
ctx context.Context ctx context.Context
device userapi.Device device userapi.Device
limit int filter *gomatrixserverlib.Filter
includeLeave bool
timeout time.Duration timeout time.Duration
since types.StreamingToken // nil means that no since token was supplied since types.StreamingToken // nil means that no since token was supplied
wantFullState bool wantFullState bool
@ -66,21 +56,14 @@ func newSyncRequest(req *http.Request, device userapi.Device, syncDB storage.Dat
return nil, err return nil, err
} }
} }
timelineLimit := DefaultTimelineLimit
includeLeave := defaultIncludeLeave var f *gomatrixserverlib.Filter
// TODO: read from stored filters too // TODO: read from stored filters too
filterQuery := req.URL.Query().Get("filter") filterQuery := req.URL.Query().Get("filter")
if filterQuery != "" { if filterQuery != "" {
if filterQuery[0] == '{' { if filterQuery[0] == '{' {
// attempt to parse the timeline limit at least // attempt to parse the timeline limit at least
var f filter json.Unmarshal([]byte(filterQuery), &f)
err := json.Unmarshal([]byte(filterQuery), &f)
if err == nil && f.Room.Timeline.Limit != nil {
timelineLimit = *f.Room.Timeline.Limit
}
if err == nil && f.Room.IncludeLeave != nil {
includeLeave = *f.Room.IncludeLeave
}
} else { } else {
// attempt to load the filter ID // attempt to load the filter ID
localpart, _, err := gomatrixserverlib.SplitID('@', device.UserID) localpart, _, err := gomatrixserverlib.SplitID('@', device.UserID)
@ -88,10 +71,10 @@ func newSyncRequest(req *http.Request, device userapi.Device, syncDB storage.Dat
util.GetLogger(req.Context()).WithError(err).Error("gomatrixserverlib.SplitID failed") util.GetLogger(req.Context()).WithError(err).Error("gomatrixserverlib.SplitID failed")
return nil, err return nil, err
} }
f, err := syncDB.GetFilter(req.Context(), localpart, filterQuery) f, err = syncDB.GetFilter(req.Context(), localpart, filterQuery)
if err == nil { if err != nil {
timelineLimit = f.Room.Timeline.Limit util.GetLogger(req.Context()).WithError(err).Error("syncDB.GetFilter failed")
includeLeave = f.Room.IncludeLeave return nil, err
} }
} }
} }
@ -102,8 +85,7 @@ func newSyncRequest(req *http.Request, device userapi.Device, syncDB storage.Dat
timeout: timeout, timeout: timeout,
since: since, since: since,
wantFullState: wantFullState, wantFullState: wantFullState,
limit: timelineLimit, filter: f,
includeLeave: includeLeave,
log: util.GetLogger(req.Context()), log: util.GetLogger(req.Context()),
}, nil }, nil
} }

View file

@ -140,12 +140,11 @@ func (rp *RequestPool) OnIncomingSyncRequest(req *http.Request, device *userapi.
} }
logger := util.GetLogger(req.Context()).WithFields(log.Fields{ logger := util.GetLogger(req.Context()).WithFields(log.Fields{
"user_id": device.UserID, "user_id": device.UserID,
"device_id": device.ID, "device_id": device.ID,
"since": syncReq.since, "since": syncReq.since,
"timeout": syncReq.timeout, "timeout": syncReq.timeout,
"limit": syncReq.limit, "filter": syncReq.filter,
"include_leave": syncReq.includeLeave,
}) })
activeSyncRequests.Inc() activeSyncRequests.Inc()
@ -248,9 +247,12 @@ func (rp *RequestPool) OnIncomingKeyChangeRequest(req *http.Request, device *use
JSON: jsonerror.InvalidArgumentValue("bad 'to' value"), JSON: jsonerror.InvalidArgumentValue("bad 'to' value"),
} }
} }
// work out room joins/leaves // work out room joins/leaves
var f gomatrixserverlib.Filter
f.Room.Timeline.Limit = 10
res, err := rp.db.IncrementalSync( res, err := rp.db.IncrementalSync(
req.Context(), types.NewResponse(), *device, fromToken, toToken, 10, false, req.Context(), types.NewResponse(), *device, fromToken, toToken, &f, false,
) )
if err != nil { if err != nil {
util.GetLogger(req.Context()).WithError(err).Error("Failed to IncrementalSync") util.GetLogger(req.Context()).WithError(err).Error("Failed to IncrementalSync")
@ -286,12 +288,12 @@ func (rp *RequestPool) currentSyncForUser(req syncRequest, latestPos types.Strea
// TODO: handle ignored users // TODO: handle ignored users
if req.since.IsEmpty() { if req.since.IsEmpty() {
res, err = rp.db.CompleteSync(req.ctx, res, req.device, req.limit, req.includeLeave) res, err = rp.db.CompleteSync(req.ctx, res, req.device, req.filter)
if err != nil { if err != nil {
return res, fmt.Errorf("rp.db.CompleteSync: %w", err) return res, fmt.Errorf("rp.db.CompleteSync: %w", err)
} }
} else { } else {
res, err = rp.db.IncrementalSync(req.ctx, res, req.device, req.since, latestPos, req.limit, req.wantFullState) res, err = rp.db.IncrementalSync(req.ctx, res, req.device, req.since, latestPos, req.filter, req.wantFullState)
if err != nil { if err != nil {
return res, fmt.Errorf("rp.db.IncrementalSync: %w", err) return res, fmt.Errorf("rp.db.IncrementalSync: %w", err)
} }