Fix a bug with single-room search searching all rooms

* Create a new method for getting predecessor rooms
* Remove formatting change
pull/4415/head
Andrew Morgan 2019-01-22 11:12:48 +00:00
parent cb80db8941
commit c9bfb058d8
3 changed files with 51 additions and 35 deletions

View File

@ -444,8 +444,19 @@ class Filter(object):
def include_redundant_members(self): def include_redundant_members(self):
return self.filter_json.get("include_redundant_members", False) return self.filter_json.get("include_redundant_members", False)
def add_room_ids(self, room_ids): def with_room_ids(self, room_ids):
self.rooms += room_ids """Returns a new filter with the given room IDs appended.
Args:
room_ids (list): A list of room_ids.
Returns:
filter: A new filter including the given rooms and the old
filter's rooms.
"""
newFilter = self
newFilter.rooms += room_ids
return newFilter
def _matches_wildcard(actual_value, filter_value): def _matches_wildcard(actual_value, filter_value):

View File

@ -49,39 +49,26 @@ class SearchHandler(BaseHandler):
The full list of all found rooms in then returned. The full list of all found rooms in then returned.
Args: Args:
room_id (str): The ID of the room to search through. room_id (str): id of the room to search through.
Returns: Returns:
dict of past room IDs as strings Deferred[iterable[str]]: predecessor room ids
""" """
historical_room_ids = [] historical_room_ids = []
while True: while True:
state_ids = yield self.store.get_current_state_ids(room_id) predecessor = yield self.store.get_room_predecessor(room_id)
create_id = state_ids.get((EventTypes.Create, ""))
# If we can't find the create event, assume we've hit a dead end # If no predecessor, assume we've hit a dead end
if not create_id:
break
# Retrieve the room's create event
create_event = yield self.store.get_event(create_id)
if not create_event:
break
# Check if a predecessor room is present
predecessor = create_event.content.get("predecessor", None)
if not predecessor: if not predecessor:
break break
# Add predecessor's room ID # Add predecessor's room ID
historical_room_id = predecessor["room_id"] historical_room_ids.append(predecessor["room_id"])
historical_room_ids.append(historical_room_id)
# Scan through the old room for further predecessors # Scan through the old room for further predecessors
room_id = historical_room_id room_id = predecessor["room_id"]
defer.returnValue(historical_room_ids) defer.returnValue(historical_room_ids)
@ -185,28 +172,19 @@ class SearchHandler(BaseHandler):
) )
room_ids = set(r.room_id for r in rooms) room_ids = set(r.room_id for r in rooms)
room_ids = search_filter.filter_rooms(room_ids)
# If doing a subset of all rooms seearch, check if any of the rooms # If doing a subset of all rooms seearch, check if any of the rooms
# are from an upgraded room, and search their contents as well # are from an upgraded room, and search their contents as well
# XXX: There is the possibility that we don't have a create event for
# the room in question, in which case we can't return all the results
# we want to.
# Ideally we would just return the results we can get now, and
# try to get more results from other servers in the background.
if search_filter.rooms: if search_filter.rooms:
historical_room_ids = [] historical_room_ids = []
for room_id in room_ids: for room_id in search_filter.rooms:
# Add any previous rooms to the search if they exist # Add any previous rooms to the search if they exist
ids = yield self.get_old_rooms_from_upgraded_room(room_id) ids = yield self.get_old_rooms_from_upgraded_room(room_id)
historical_room_ids += ids historical_room_ids += ids
# Add any found rooms to the list to search
for historical_room_id in historical_room_ids:
room_ids.add(historical_room_id)
# Prevent any historical events from being filtered # Prevent any historical events from being filtered
search_filter.add_room_ids(historical_room_ids) search_filter = search_filter.with_room_ids(historical_room_ids)
room_ids = search_filter.filter_rooms(room_ids)
if batch_group == "room_id": if batch_group == "room_id":
room_ids.intersection_update({batch_group_key}) room_ids.intersection_update({batch_group_key})

View File

@ -437,6 +437,34 @@ class StateGroupWorkerStore(EventsWorkerStore, SQLBaseStore):
create_event = yield self.get_event(create_id) create_event = yield self.get_event(create_id)
defer.returnValue(create_event.content.get("room_version", "1")) defer.returnValue(create_event.content.get("room_version", "1"))
@defer.inlineCallbacks
def get_room_predecessor(self, room_id):
"""Get the predecessor room of an upgraded room if one exists.
Otherwise return None.
Args:
room_id (str)
Returns:
Deferred[str]: predecessor room id
"""
state_ids = yield self.get_current_state_ids(room_id)
create_id = state_ids.get((EventTypes.Create, ""))
# If we can't find the create event, assume we've hit a dead end
if not create_id:
return None
# Retrieve the room's create event
create_event = yield self.get_event(create_id)
if not create_event:
return None
# Return predecessor if present
return create_event.content.get("predecessor", None)
@cached(max_entries=100000, iterable=True) @cached(max_entries=100000, iterable=True)
def get_current_state_ids(self, room_id): def get_current_state_ids(self, room_id):
"""Get the current state event ids for a room based on the """Get the current state event ids for a room based on the
@ -448,7 +476,6 @@ class StateGroupWorkerStore(EventsWorkerStore, SQLBaseStore):
Returns: Returns:
deferred: dict of (type, state_key) -> event_id deferred: dict of (type, state_key) -> event_id
""" """
def _get_current_state_ids_txn(txn): def _get_current_state_ids_txn(txn):
txn.execute( txn.execute(
"""SELECT type, state_key, event_id FROM current_state_events """SELECT type, state_key, event_id FROM current_state_events