From dc5dd4c5d269fa5dc4518449be19ec848d833f25 Mon Sep 17 00:00:00 2001 From: Mark Haines Date: Thu, 14 Sep 2017 14:46:56 +0100 Subject: [PATCH] Add contexts to the public rooms database (#230) --- .../publicroomsapi/consumers/roomserver.go | 2 +- .../publicroomsapi/directory/directory.go | 4 +- .../publicroomsapi/directory/public_rooms.go | 6 +- .../storage/public_rooms_table.go | 80 ++++++++++--------- .../publicroomsapi/storage/storage.go | 79 ++++++++++-------- 5 files changed, 98 insertions(+), 73 deletions(-) diff --git a/src/github.com/matrix-org/dendrite/publicroomsapi/consumers/roomserver.go b/src/github.com/matrix-org/dendrite/publicroomsapi/consumers/roomserver.go index bb823e216..65209bf3d 100644 --- a/src/github.com/matrix-org/dendrite/publicroomsapi/consumers/roomserver.go +++ b/src/github.com/matrix-org/dendrite/publicroomsapi/consumers/roomserver.go @@ -98,5 +98,5 @@ func (s *OutputRoomEvent) onMessage(msg *sarama.ConsumerMessage) error { return err } - return s.db.UpdateRoomFromEvents(addQueryRes.Events, remQueryRes.Events) + return s.db.UpdateRoomFromEvents(context.TODO(), addQueryRes.Events, remQueryRes.Events) } diff --git a/src/github.com/matrix-org/dendrite/publicroomsapi/directory/directory.go b/src/github.com/matrix-org/dendrite/publicroomsapi/directory/directory.go index 1718a18af..8c6de95b3 100644 --- a/src/github.com/matrix-org/dendrite/publicroomsapi/directory/directory.go +++ b/src/github.com/matrix-org/dendrite/publicroomsapi/directory/directory.go @@ -32,7 +32,7 @@ func GetVisibility( req *http.Request, publicRoomsDatabase *storage.PublicRoomsServerDatabase, roomID string, ) util.JSONResponse { - isPublic, err := publicRoomsDatabase.GetRoomVisibility(roomID) + isPublic, err := publicRoomsDatabase.GetRoomVisibility(req.Context(), roomID) if err != nil { return httputil.LogThenError(req, err) } @@ -62,7 +62,7 @@ func SetVisibility( } isPublic := v.Visibility == "public" - if err := publicRoomsDatabase.SetRoomVisibility(isPublic, roomID); err != nil { + if err := publicRoomsDatabase.SetRoomVisibility(req.Context(), isPublic, roomID); err != nil { return httputil.LogThenError(req, err) } diff --git a/src/github.com/matrix-org/dendrite/publicroomsapi/directory/public_rooms.go b/src/github.com/matrix-org/dendrite/publicroomsapi/directory/public_rooms.go index 4566715c2..19ae90d8d 100644 --- a/src/github.com/matrix-org/dendrite/publicroomsapi/directory/public_rooms.go +++ b/src/github.com/matrix-org/dendrite/publicroomsapi/directory/public_rooms.go @@ -63,7 +63,7 @@ func GetPublicRooms( return httputil.LogThenError(req, err) } - if response.Estimate, err = publicRoomDatabase.CountPublicRooms(); err != nil { + if response.Estimate, err = publicRoomDatabase.CountPublicRooms(req.Context()); err != nil { return httputil.LogThenError(req, err) } @@ -75,7 +75,9 @@ func GetPublicRooms( response.NextBatch = strconv.Itoa(nextIndex) } - if response.Chunk, err = publicRoomDatabase.GetPublicRooms(offset, limit, request.Filter.SearchTerms); err != nil { + if response.Chunk, err = publicRoomDatabase.GetPublicRooms( + req.Context(), offset, limit, request.Filter.SearchTerms, + ); err != nil { return httputil.LogThenError(req, err) } diff --git a/src/github.com/matrix-org/dendrite/publicroomsapi/storage/public_rooms_table.go b/src/github.com/matrix-org/dendrite/publicroomsapi/storage/public_rooms_table.go index 5cef577e7..5e1eb3e12 100644 --- a/src/github.com/matrix-org/dendrite/publicroomsapi/storage/public_rooms_table.go +++ b/src/github.com/matrix-org/dendrite/publicroomsapi/storage/public_rooms_table.go @@ -15,6 +15,7 @@ package storage import ( + "context" "database/sql" "errors" "fmt" @@ -166,27 +167,35 @@ func (s *publicRoomsStatements) prepare(db *sql.DB) (err error) { return } -func (s *publicRoomsStatements) countPublicRooms() (nb int64, err error) { - err = s.countPublicRoomsStmt.QueryRow().Scan(&nb) +func (s *publicRoomsStatements) countPublicRooms(ctx context.Context) (nb int64, err error) { + err = s.countPublicRoomsStmt.QueryRowContext(ctx).Scan(&nb) return } -func (s *publicRoomsStatements) selectPublicRooms(offset int64, limit int16, filter string) ([]types.PublicRoom, error) { +func (s *publicRoomsStatements) selectPublicRooms( + ctx context.Context, offset int64, limit int16, filter string, +) ([]types.PublicRoom, error) { var rows *sql.Rows var err error if len(filter) > 0 { pattern := "%" + filter + "%" if limit == 0 { - rows, err = s.selectPublicRoomsWithFilterStmt.Query(pattern, offset) + rows, err = s.selectPublicRoomsWithFilterStmt.QueryContext( + ctx, pattern, offset, + ) } else { - rows, err = s.selectPublicRoomsWithLimitAndFilterStmt.Query(pattern, offset, limit) + rows, err = s.selectPublicRoomsWithLimitAndFilterStmt.QueryContext( + ctx, pattern, offset, limit, + ) } } else { if limit == 0 { - rows, err = s.selectPublicRoomsStmt.Query(offset) + rows, err = s.selectPublicRoomsStmt.QueryContext(ctx, offset) } else { - rows, err = s.selectPublicRoomsWithLimitStmt.Query(offset, limit) + rows, err = s.selectPublicRoomsWithLimitStmt.QueryContext( + ctx, offset, limit, + ) } } @@ -207,10 +216,7 @@ func (s *publicRoomsStatements) selectPublicRooms(offset int64, limit int16, fil return rooms, err } - r.Aliases = make([]string, len(aliases)) - for i := range aliases { - r.Aliases[i] = aliases[i] - } + r.Aliases = aliases rooms = append(rooms, r) } @@ -218,51 +224,53 @@ func (s *publicRoomsStatements) selectPublicRooms(offset int64, limit int16, fil return rooms, nil } -func (s *publicRoomsStatements) selectRoomVisibility(roomID string) (v bool, err error) { - err = s.selectRoomVisibilityStmt.QueryRow(roomID).Scan(&v) +func (s *publicRoomsStatements) selectRoomVisibility( + ctx context.Context, roomID string, +) (v bool, err error) { + err = s.selectRoomVisibilityStmt.QueryRowContext(ctx, roomID).Scan(&v) return } -func (s *publicRoomsStatements) insertNewRoom(roomID string) error { - _, err := s.insertNewRoomStmt.Exec(roomID) +func (s *publicRoomsStatements) insertNewRoom( + ctx context.Context, roomID string, +) error { + _, err := s.insertNewRoomStmt.ExecContext(ctx, roomID) return err } -func (s *publicRoomsStatements) incrementJoinedMembersInRoom(roomID string) error { - _, err := s.incrementJoinedMembersInRoomStmt.Exec(roomID) +func (s *publicRoomsStatements) incrementJoinedMembersInRoom( + ctx context.Context, roomID string, +) error { + _, err := s.incrementJoinedMembersInRoomStmt.ExecContext(ctx, roomID) return err } -func (s *publicRoomsStatements) decrementJoinedMembersInRoom(roomID string) error { - _, err := s.decrementJoinedMembersInRoomStmt.Exec(roomID) +func (s *publicRoomsStatements) decrementJoinedMembersInRoom( + ctx context.Context, roomID string, +) error { + _, err := s.decrementJoinedMembersInRoomStmt.ExecContext(ctx, roomID) return err } -func (s *publicRoomsStatements) updateRoomAttribute(attrName string, attrValue attributeValue, roomID string) error { - isEditable := false - for _, editable := range editableAttributes { - if editable == attrName { - isEditable = true - } - } +func (s *publicRoomsStatements) updateRoomAttribute( + ctx context.Context, attrName string, attrValue attributeValue, roomID string, +) error { + stmt, isEditable := s.updateRoomAttributeStmts[attrName] if !isEditable { return errors.New("Cannot edit " + attrName) } var value interface{} - if attrName == "aliases" { - // Aliases need a special conversion - valueAsSlice, isSlice := attrValue.([]string) - if !isSlice { - // attrValue isn't a slice of strings - return errors.New("New list of aliases is of the wrong type") - } - value = pq.StringArray(valueAsSlice) - } else { + switch v := attrValue.(type) { + case []string: + value = pq.StringArray(v) + case bool, string: value = attrValue + default: + return errors.New("Unsupported attribute type, must be bool, string or []string") } - _, err := s.updateRoomAttributeStmts[attrName].Exec(value, roomID) + _, err := stmt.ExecContext(ctx, value, roomID) return err } diff --git a/src/github.com/matrix-org/dendrite/publicroomsapi/storage/storage.go b/src/github.com/matrix-org/dendrite/publicroomsapi/storage/storage.go index 83861180f..c1482e334 100644 --- a/src/github.com/matrix-org/dendrite/publicroomsapi/storage/storage.go +++ b/src/github.com/matrix-org/dendrite/publicroomsapi/storage/storage.go @@ -15,6 +15,7 @@ package storage import ( + "context" "database/sql" "encoding/json" @@ -64,21 +65,25 @@ func (d *PublicRoomsServerDatabase) SetPartitionOffset(topic string, partition i // GetRoomVisibility returns the room visibility as a boolean: true if the room // is publicly visible, false if not. // Returns an error if the retrieval failed. -func (d *PublicRoomsServerDatabase) GetRoomVisibility(roomID string) (bool, error) { - return d.statements.selectRoomVisibility(roomID) +func (d *PublicRoomsServerDatabase) GetRoomVisibility( + ctx context.Context, roomID string, +) (bool, error) { + return d.statements.selectRoomVisibility(ctx, roomID) } // SetRoomVisibility updates the visibility attribute of a room. This attribute // must be set to true if the room is publicly visible, false if not. // Returns an error if the update failed. -func (d *PublicRoomsServerDatabase) SetRoomVisibility(visible bool, roomID string) error { - return d.statements.updateRoomAttribute("visibility", visible, roomID) +func (d *PublicRoomsServerDatabase) SetRoomVisibility( + ctx context.Context, visible bool, roomID string, +) error { + return d.statements.updateRoomAttribute(ctx, "visibility", visible, roomID) } // CountPublicRooms returns the number of room set as publicly visible on the server. // Returns an error if the retrieval failed. -func (d *PublicRoomsServerDatabase) CountPublicRooms() (int64, error) { - return d.statements.countPublicRooms() +func (d *PublicRoomsServerDatabase) CountPublicRooms(ctx context.Context) (int64, error) { + return d.statements.countPublicRooms(ctx) } // GetPublicRooms returns an array containing the local rooms set as publicly visible, ordered by their number @@ -86,8 +91,10 @@ func (d *PublicRoomsServerDatabase) CountPublicRooms() (int64, error) { // If the limit is 0, doesn't limit the number of results. If the offset is 0 too, the array contains all // the rooms set as publicly visible on the server. // Returns an error if the retrieval failed. -func (d *PublicRoomsServerDatabase) GetPublicRooms(offset int64, limit int16, filter string) ([]types.PublicRoom, error) { - return d.statements.selectPublicRooms(offset, limit, filter) +func (d *PublicRoomsServerDatabase) GetPublicRooms( + ctx context.Context, offset int64, limit int16, filter string, +) ([]types.PublicRoom, error) { + return d.statements.selectPublicRooms(ctx, offset, limit, filter) } // UpdateRoomFromEvents iterate over a slice of state events and call @@ -98,17 +105,19 @@ func (d *PublicRoomsServerDatabase) GetPublicRooms(offset int64, limit int16, fi // If the update triggered by one of the events failed, aborts the process and // returns an error. func (d *PublicRoomsServerDatabase) UpdateRoomFromEvents( - eventsToAdd []gomatrixserverlib.Event, eventsToRemove []gomatrixserverlib.Event, + ctx context.Context, + eventsToAdd []gomatrixserverlib.Event, + eventsToRemove []gomatrixserverlib.Event, ) error { for _, event := range eventsToAdd { - if err := d.UpdateRoomFromEvent(event); err != nil { + if err := d.UpdateRoomFromEvent(ctx, event); err != nil { return err } } for _, event := range eventsToRemove { if event.Type() == "m.room.member" { - if err := d.updateNumJoinedUsers(event, true); err != nil { + if err := d.updateNumJoinedUsers(ctx, event, true); err != nil { return err } } @@ -123,47 +132,49 @@ func (d *PublicRoomsServerDatabase) UpdateRoomFromEvents( // If the event doesn't match with any property used to compute the public room directory, // does nothing. // If something went wrong during the process, returns an error. -func (d *PublicRoomsServerDatabase) UpdateRoomFromEvent(event gomatrixserverlib.Event) error { +func (d *PublicRoomsServerDatabase) UpdateRoomFromEvent( + ctx context.Context, event gomatrixserverlib.Event, +) error { // Process the event according to its type switch event.Type() { case "m.room.create": - return d.statements.insertNewRoom(event.RoomID()) + return d.statements.insertNewRoom(ctx, event.RoomID()) case "m.room.member": - return d.updateNumJoinedUsers(event, false) + return d.updateNumJoinedUsers(ctx, event, false) case "m.room.aliases": - return d.updateRoomAliases(event) + return d.updateRoomAliases(ctx, event) case "m.room.canonical_alias": var content common.CanonicalAliasContent field := &(content.Alias) attrName := "canonical_alias" - return d.updateStringAttribute(attrName, event, &content, field) + return d.updateStringAttribute(ctx, attrName, event, &content, field) case "m.room.name": var content common.NameContent field := &(content.Name) attrName := "name" - return d.updateStringAttribute(attrName, event, &content, field) + return d.updateStringAttribute(ctx, attrName, event, &content, field) case "m.room.topic": var content common.TopicContent field := &(content.Topic) attrName := "topic" - return d.updateStringAttribute(attrName, event, &content, field) + return d.updateStringAttribute(ctx, attrName, event, &content, field) case "m.room.avatar": var content common.AvatarContent field := &(content.URL) attrName := "avatar_url" - return d.updateStringAttribute(attrName, event, &content, field) + return d.updateStringAttribute(ctx, attrName, event, &content, field) case "m.room.history_visibility": var content common.HistoryVisibilityContent field := &(content.HistoryVisibility) attrName := "world_readable" strForTrue := "world_readable" - return d.updateBooleanAttribute(attrName, event, &content, field, strForTrue) + return d.updateBooleanAttribute(ctx, attrName, event, &content, field, strForTrue) case "m.room.guest_access": var content common.GuestAccessContent field := &(content.GuestAccess) attrName := "guest_can_join" strForTrue := "can_join" - return d.updateBooleanAttribute(attrName, event, &content, field, strForTrue) + return d.updateBooleanAttribute(ctx, attrName, event, &content, field, strForTrue) } // If the event type didn't match, return with no error @@ -177,7 +188,7 @@ func (d *PublicRoomsServerDatabase) UpdateRoomFromEvent(event gomatrixserverlib. // database, if set to truem decrements it. // Returns an error if the update failed. func (d *PublicRoomsServerDatabase) updateNumJoinedUsers( - membershipEvent gomatrixserverlib.Event, remove bool, + ctx context.Context, membershipEvent gomatrixserverlib.Event, remove bool, ) error { membership, err := membershipEvent.Membership() if err != nil { @@ -189,9 +200,9 @@ func (d *PublicRoomsServerDatabase) updateNumJoinedUsers( } if remove { - return d.statements.decrementJoinedMembersInRoom(membershipEvent.RoomID()) + return d.statements.decrementJoinedMembersInRoom(ctx, membershipEvent.RoomID()) } - return d.statements.incrementJoinedMembersInRoom(membershipEvent.RoomID()) + return d.statements.incrementJoinedMembersInRoom(ctx, membershipEvent.RoomID()) } // updateStringAttribute updates a given string attribute in the database @@ -200,14 +211,14 @@ func (d *PublicRoomsServerDatabase) updateNumJoinedUsers( // Returns an error if decoding the Matrix event's content or updating the attribute // failed. func (d *PublicRoomsServerDatabase) updateStringAttribute( - attrName string, event gomatrixserverlib.Event, content interface{}, - field *string, + ctx context.Context, attrName string, event gomatrixserverlib.Event, + content interface{}, field *string, ) error { if err := json.Unmarshal(event.Content(), content); err != nil { return err } - return d.statements.updateRoomAttribute(attrName, *field, event.RoomID()) + return d.statements.updateRoomAttribute(ctx, attrName, *field, event.RoomID()) } // updateBooleanAttribute updates a given boolean attribute in the database @@ -217,8 +228,8 @@ func (d *PublicRoomsServerDatabase) updateStringAttribute( // Returns an error if decoding the Matrix event's content or updating the attribute // failed. func (d *PublicRoomsServerDatabase) updateBooleanAttribute( - attrName string, event gomatrixserverlib.Event, content interface{}, - field *string, strForTrue string, + ctx context.Context, attrName string, event gomatrixserverlib.Event, + content interface{}, field *string, strForTrue string, ) error { if err := json.Unmarshal(event.Content(), content); err != nil { return err @@ -231,17 +242,21 @@ func (d *PublicRoomsServerDatabase) updateBooleanAttribute( attrValue = false } - return d.statements.updateRoomAttribute(attrName, attrValue, event.RoomID()) + return d.statements.updateRoomAttribute(ctx, attrName, attrValue, event.RoomID()) } // updateRoomAliases decodes the content of a "m.room.aliases" Matrix event and update the list of aliases of // a given room with it. // Returns an error if decoding the Matrix event or updating the list failed. -func (d *PublicRoomsServerDatabase) updateRoomAliases(aliasesEvent gomatrixserverlib.Event) error { +func (d *PublicRoomsServerDatabase) updateRoomAliases( + ctx context.Context, aliasesEvent gomatrixserverlib.Event, +) error { var content common.AliasesContent if err := json.Unmarshal(aliasesEvent.Content(), &content); err != nil { return err } - return d.statements.updateRoomAttribute("aliases", content.Aliases, aliasesEvent.RoomID()) + return d.statements.updateRoomAttribute( + ctx, "aliases", content.Aliases, aliasesEvent.RoomID(), + ) }