2016-09-14 09:07:37 -04:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
# Copyright 2014 - 2016 OpenMarket Ltd
|
|
|
|
#
|
|
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
# you may not use this file except in compliance with the License.
|
|
|
|
# You may obtain a copy of the License at
|
|
|
|
#
|
|
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
#
|
|
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
# See the License for the specific language governing permissions and
|
|
|
|
# limitations under the License.
|
|
|
|
|
|
|
|
from twisted.internet import defer
|
|
|
|
|
|
|
|
from ._base import BaseHandler
|
|
|
|
|
|
|
|
from synapse.api.constants import (
|
2016-09-14 12:17:07 -04:00
|
|
|
EventTypes, JoinRules,
|
2016-09-14 09:07:37 -04:00
|
|
|
)
|
|
|
|
from synapse.api.errors import SynapseError
|
|
|
|
from synapse.util.async import concurrently_execute
|
|
|
|
from synapse.util.caches.response_cache import ResponseCache
|
|
|
|
|
2016-09-15 04:08:57 -04:00
|
|
|
from collections import namedtuple
|
|
|
|
from unpaddedbase64 import encode_base64, decode_base64
|
|
|
|
|
2016-09-14 09:07:37 -04:00
|
|
|
import logging
|
2016-09-15 04:08:57 -04:00
|
|
|
import msgpack
|
2016-09-14 09:07:37 -04:00
|
|
|
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
REMOTE_ROOM_LIST_POLL_INTERVAL = 60 * 1000
|
|
|
|
|
|
|
|
|
|
|
|
class RoomListHandler(BaseHandler):
|
|
|
|
def __init__(self, hs):
|
|
|
|
super(RoomListHandler, self).__init__(hs)
|
|
|
|
self.response_cache = ResponseCache(hs)
|
|
|
|
self.remote_list_request_cache = ResponseCache(hs)
|
|
|
|
self.remote_list_cache = {}
|
|
|
|
self.fetch_looping_call = hs.get_clock().looping_call(
|
|
|
|
self.fetch_all_remote_lists, REMOTE_ROOM_LIST_POLL_INTERVAL
|
|
|
|
)
|
|
|
|
self.fetch_all_remote_lists()
|
|
|
|
|
2016-09-15 04:08:57 -04:00
|
|
|
def get_local_public_room_list(self, limit=None, next_batch=None):
|
|
|
|
result = self.response_cache.get((limit, next_batch))
|
2016-09-14 09:07:37 -04:00
|
|
|
if not result:
|
2016-09-15 04:08:57 -04:00
|
|
|
result = self.response_cache.set(
|
|
|
|
(limit, next_batch),
|
|
|
|
self._get_public_room_list(limit, next_batch)
|
|
|
|
)
|
2016-09-14 09:07:37 -04:00
|
|
|
return result
|
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
2016-09-15 04:08:57 -04:00
|
|
|
def _get_public_room_list(self, limit=None, next_batch=None):
|
|
|
|
if next_batch and next_batch != "END":
|
|
|
|
next_batch = RoomListNextBatch.from_token(next_batch)
|
|
|
|
else:
|
|
|
|
next_batch = None
|
|
|
|
|
2016-09-14 09:07:37 -04:00
|
|
|
room_ids = yield self.store.get_public_room_ids()
|
|
|
|
|
2016-09-14 12:28:52 -04:00
|
|
|
rooms_to_order_value = {}
|
|
|
|
rooms_to_num_joined = {}
|
|
|
|
rooms_to_latest_event_ids = {}
|
|
|
|
|
2016-09-15 04:08:57 -04:00
|
|
|
if next_batch:
|
|
|
|
current_stream_token = next_batch.sstream_ordering
|
|
|
|
else:
|
|
|
|
current_stream_token = yield self.store.get_room_max_stream_ordering()
|
2016-09-14 12:28:52 -04:00
|
|
|
|
|
|
|
# We want to return rooms in a particular order: the number of joined
|
|
|
|
# users. We then arbitrarily use the room_id as a tie breaker.
|
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
|
|
|
def get_order_for_room(room_id):
|
|
|
|
latest_event_ids = rooms_to_latest_event_ids.get(room_id, None)
|
|
|
|
if not latest_event_ids:
|
|
|
|
latest_event_ids = yield self.store.get_forward_extremeties_for_room(
|
|
|
|
room_id, current_stream_token
|
|
|
|
)
|
|
|
|
rooms_to_latest_event_ids[room_id] = latest_event_ids
|
|
|
|
|
|
|
|
if not latest_event_ids:
|
|
|
|
return
|
|
|
|
|
|
|
|
joined_users = yield self.state_handler.get_current_user_in_room(
|
|
|
|
room_id, latest_event_ids,
|
|
|
|
)
|
|
|
|
num_joined_users = len(joined_users)
|
|
|
|
rooms_to_num_joined[room_id] = num_joined_users
|
|
|
|
|
|
|
|
if num_joined_users == 0:
|
|
|
|
return
|
|
|
|
|
|
|
|
# We want larger rooms to be first, hence negating num_joined_users
|
|
|
|
rooms_to_order_value[room_id] = (-num_joined_users, room_id)
|
|
|
|
|
|
|
|
yield concurrently_execute(get_order_for_room, room_ids, 10)
|
|
|
|
|
|
|
|
sorted_entries = sorted(rooms_to_order_value.items(), key=lambda e: e[1])
|
|
|
|
sorted_rooms = [room_id for room_id, _ in sorted_entries]
|
|
|
|
|
2016-09-15 04:08:57 -04:00
|
|
|
if next_batch:
|
|
|
|
sorted_rooms = sorted_rooms[next_batch.current_limit:]
|
|
|
|
|
|
|
|
new_limit = None
|
|
|
|
if limit:
|
|
|
|
if sorted_rooms[limit:]:
|
|
|
|
new_limit = limit
|
|
|
|
if next_batch:
|
|
|
|
new_limit += next_batch.current_limit
|
|
|
|
sorted_rooms = sorted_rooms[:limit]
|
|
|
|
|
2016-09-14 09:07:37 -04:00
|
|
|
results = []
|
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
|
|
|
def handle_room(room_id):
|
2016-09-14 12:28:52 -04:00
|
|
|
num_joined_users = rooms_to_num_joined[room_id]
|
2016-09-14 12:17:07 -04:00
|
|
|
if num_joined_users == 0:
|
|
|
|
return
|
|
|
|
|
|
|
|
result = {
|
|
|
|
"room_id": room_id,
|
|
|
|
"num_joined_members": num_joined_users,
|
|
|
|
}
|
|
|
|
|
|
|
|
current_state_ids = yield self.state_handler.get_current_state_ids(room_id)
|
|
|
|
|
|
|
|
event_map = yield self.store.get_events([
|
|
|
|
event_id for key, event_id in current_state_ids.items()
|
|
|
|
if key[0] in (
|
|
|
|
EventTypes.JoinRules,
|
|
|
|
EventTypes.Name,
|
|
|
|
EventTypes.Topic,
|
|
|
|
EventTypes.CanonicalAlias,
|
|
|
|
EventTypes.RoomHistoryVisibility,
|
|
|
|
EventTypes.GuestAccess,
|
|
|
|
"m.room.avatar",
|
|
|
|
)
|
|
|
|
])
|
|
|
|
|
|
|
|
current_state = {
|
|
|
|
(ev.type, ev.state_key): ev
|
|
|
|
for ev in event_map.values()
|
|
|
|
}
|
2016-09-14 09:07:37 -04:00
|
|
|
|
|
|
|
# Double check that this is actually a public room.
|
|
|
|
join_rules_event = current_state.get((EventTypes.JoinRules, ""))
|
|
|
|
if join_rules_event:
|
|
|
|
join_rule = join_rules_event.content.get("join_rule", None)
|
|
|
|
if join_rule and join_rule != JoinRules.PUBLIC:
|
|
|
|
defer.returnValue(None)
|
|
|
|
|
|
|
|
aliases = yield self.store.get_aliases_for_room(room_id)
|
|
|
|
if aliases:
|
|
|
|
result["aliases"] = aliases
|
|
|
|
|
|
|
|
name_event = yield current_state.get((EventTypes.Name, ""))
|
|
|
|
if name_event:
|
|
|
|
name = name_event.content.get("name", None)
|
|
|
|
if name:
|
|
|
|
result["name"] = name
|
|
|
|
|
|
|
|
topic_event = current_state.get((EventTypes.Topic, ""))
|
|
|
|
if topic_event:
|
|
|
|
topic = topic_event.content.get("topic", None)
|
|
|
|
if topic:
|
|
|
|
result["topic"] = topic
|
|
|
|
|
|
|
|
canonical_event = current_state.get((EventTypes.CanonicalAlias, ""))
|
|
|
|
if canonical_event:
|
|
|
|
canonical_alias = canonical_event.content.get("alias", None)
|
|
|
|
if canonical_alias:
|
|
|
|
result["canonical_alias"] = canonical_alias
|
|
|
|
|
|
|
|
visibility_event = current_state.get((EventTypes.RoomHistoryVisibility, ""))
|
|
|
|
visibility = None
|
|
|
|
if visibility_event:
|
|
|
|
visibility = visibility_event.content.get("history_visibility", None)
|
|
|
|
result["world_readable"] = visibility == "world_readable"
|
|
|
|
|
|
|
|
guest_event = current_state.get((EventTypes.GuestAccess, ""))
|
|
|
|
guest = None
|
|
|
|
if guest_event:
|
|
|
|
guest = guest_event.content.get("guest_access", None)
|
|
|
|
result["guest_can_join"] = guest == "can_join"
|
|
|
|
|
|
|
|
avatar_event = current_state.get(("m.room.avatar", ""))
|
|
|
|
if avatar_event:
|
|
|
|
avatar_url = avatar_event.content.get("url", None)
|
|
|
|
if avatar_url:
|
|
|
|
result["avatar_url"] = avatar_url
|
|
|
|
|
|
|
|
results.append(result)
|
|
|
|
|
2016-09-14 12:28:52 -04:00
|
|
|
yield concurrently_execute(handle_room, sorted_rooms, 10)
|
2016-09-14 09:07:37 -04:00
|
|
|
|
2016-09-15 04:08:57 -04:00
|
|
|
if new_limit:
|
|
|
|
end_token = RoomListNextBatch(
|
|
|
|
stream_ordering=current_stream_token,
|
|
|
|
current_limit=new_limit,
|
|
|
|
).to_token()
|
|
|
|
else:
|
|
|
|
end_token = "END"
|
|
|
|
|
|
|
|
if next_batch:
|
|
|
|
start_token = next_batch.to_token()
|
|
|
|
else:
|
|
|
|
start_token = "START"
|
|
|
|
|
|
|
|
defer.returnValue({
|
|
|
|
"start": start_token,
|
|
|
|
"end": end_token,
|
|
|
|
"chunk": results,
|
|
|
|
})
|
2016-09-14 09:07:37 -04:00
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
|
|
|
def fetch_all_remote_lists(self):
|
|
|
|
deferred = self.hs.get_replication_layer().get_public_rooms(
|
|
|
|
self.hs.config.secondary_directory_servers
|
|
|
|
)
|
|
|
|
self.remote_list_request_cache.set((), deferred)
|
|
|
|
self.remote_list_cache = yield deferred
|
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
|
|
|
def get_remote_public_room_list(self, server_name):
|
|
|
|
res = yield self.hs.get_replication_layer().get_public_rooms(
|
|
|
|
[server_name]
|
|
|
|
)
|
|
|
|
|
|
|
|
if server_name not in res:
|
|
|
|
raise SynapseError(404, "Server not found")
|
|
|
|
defer.returnValue(res[server_name])
|
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
|
|
|
def get_aggregated_public_room_list(self):
|
|
|
|
"""
|
|
|
|
Get the public room list from this server and the servers
|
|
|
|
specified in the secondary_directory_servers config option.
|
|
|
|
XXX: Pagination...
|
|
|
|
"""
|
|
|
|
# We return the results from out cache which is updated by a looping call,
|
|
|
|
# unless we're missing a cache entry, in which case wait for the result
|
|
|
|
# of the fetch if there's one in progress. If not, omit that server.
|
|
|
|
wait = False
|
|
|
|
for s in self.hs.config.secondary_directory_servers:
|
|
|
|
if s not in self.remote_list_cache:
|
|
|
|
logger.warn("No cached room list from %s: waiting for fetch", s)
|
|
|
|
wait = True
|
|
|
|
break
|
|
|
|
|
|
|
|
if wait and self.remote_list_request_cache.get(()):
|
|
|
|
yield self.remote_list_request_cache.get(())
|
|
|
|
|
|
|
|
public_rooms = yield self.get_local_public_room_list()
|
|
|
|
|
|
|
|
# keep track of which room IDs we've seen so we can de-dup
|
|
|
|
room_ids = set()
|
|
|
|
|
|
|
|
# tag all the ones in our list with our server name.
|
|
|
|
# Also add the them to the de-deping set
|
|
|
|
for room in public_rooms['chunk']:
|
|
|
|
room["server_name"] = self.hs.hostname
|
|
|
|
room_ids.add(room["room_id"])
|
|
|
|
|
|
|
|
# Now add the results from federation
|
|
|
|
for server_name, server_result in self.remote_list_cache.items():
|
|
|
|
for room in server_result["chunk"]:
|
|
|
|
if room["room_id"] not in room_ids:
|
|
|
|
room["server_name"] = server_name
|
|
|
|
public_rooms["chunk"].append(room)
|
|
|
|
room_ids.add(room["room_id"])
|
|
|
|
|
|
|
|
defer.returnValue(public_rooms)
|
2016-09-15 04:08:57 -04:00
|
|
|
|
|
|
|
|
|
|
|
class RoomListNextBatch(namedtuple("RoomListNextBatch", (
|
|
|
|
"stream_ordering", # stream_ordering of the first public room list
|
|
|
|
"current_limit", # The number of previous rooms returned
|
|
|
|
))):
|
|
|
|
|
|
|
|
KEY_DICT = {
|
|
|
|
"stream_ordering": "s",
|
|
|
|
"current_limit": "n",
|
|
|
|
}
|
|
|
|
|
|
|
|
REVERSE_KEY_DICT = {v: k for k, v in KEY_DICT.items()}
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def from_token(cls, token):
|
|
|
|
return RoomListNextBatch(**{
|
|
|
|
cls.REVERSE_KEY_DICT[key]: val
|
|
|
|
for key, val in msgpack.loads(decode_base64(token)).items()
|
|
|
|
})
|
|
|
|
|
|
|
|
def to_token(self):
|
|
|
|
return encode_base64(msgpack.dumps({
|
|
|
|
self.KEY_DICT[key]: val
|
|
|
|
for key, val in self._asdict().items()
|
|
|
|
}))
|