2014-09-29 07:59:52 -06:00
|
|
|
# -*- coding: utf-8 -*-
|
2016-01-06 21:26:29 -07:00
|
|
|
# Copyright 2014-2016 OpenMarket Ltd
|
2014-09-29 07:59:52 -06:00
|
|
|
#
|
|
|
|
# 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.
|
|
|
|
|
2018-07-09 00:09:20 -06:00
|
|
|
import logging
|
|
|
|
|
2014-09-29 07:59:52 -06:00
|
|
|
from twisted.internet import defer
|
|
|
|
|
2019-07-01 10:55:11 -06:00
|
|
|
from synapse.api.constants import Membership
|
|
|
|
from synapse.types import RoomStreamToken
|
|
|
|
from synapse.visibility import filter_events_for_client
|
|
|
|
|
2014-09-29 07:59:52 -06:00
|
|
|
from ._base import BaseHandler
|
|
|
|
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
class AdminHandler(BaseHandler):
|
|
|
|
def __init__(self, hs):
|
|
|
|
super(AdminHandler, self).__init__(hs)
|
|
|
|
|
2019-10-23 10:25:54 -06:00
|
|
|
self.storage = hs.get_storage()
|
|
|
|
self.state_store = self.storage.state
|
|
|
|
|
2014-09-29 07:59:52 -06:00
|
|
|
@defer.inlineCallbacks
|
|
|
|
def get_whois(self, user):
|
2015-12-02 10:27:49 -07:00
|
|
|
connections = []
|
|
|
|
|
|
|
|
sessions = yield self.store.get_user_ip_and_agents(user)
|
|
|
|
for session in sessions:
|
2019-06-20 03:32:02 -06:00
|
|
|
connections.append(
|
|
|
|
{
|
|
|
|
"ip": session["ip"],
|
|
|
|
"last_seen": session["last_seen"],
|
|
|
|
"user_agent": session["user_agent"],
|
|
|
|
}
|
|
|
|
)
|
2014-09-29 07:59:52 -06:00
|
|
|
|
|
|
|
ret = {
|
|
|
|
"user_id": user.to_string(),
|
2019-06-20 03:32:02 -06:00
|
|
|
"devices": {"": {"sessions": [{"connections": connections}]}},
|
2014-09-29 07:59:52 -06:00
|
|
|
}
|
|
|
|
|
2019-07-23 07:00:55 -06:00
|
|
|
return ret
|
2017-02-02 06:02:26 -07:00
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
|
|
|
def get_users(self):
|
2019-12-05 11:12:23 -07:00
|
|
|
"""Function to retrieve a list of users in users table.
|
2017-02-02 06:02:26 -07:00
|
|
|
|
|
|
|
Args:
|
|
|
|
Returns:
|
|
|
|
defer.Deferred: resolves to list[dict[str, Any]]
|
|
|
|
"""
|
|
|
|
ret = yield self.store.get_users()
|
|
|
|
|
2019-07-23 07:00:55 -06:00
|
|
|
return ret
|
2017-02-02 06:02:26 -07:00
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
2019-12-05 11:12:23 -07:00
|
|
|
def get_users_paginate(self, start, limit, name, guests, deactivated):
|
|
|
|
"""Function to retrieve a paginated list of users from
|
|
|
|
users list. This will return a json list of users.
|
2017-02-02 06:02:26 -07:00
|
|
|
|
|
|
|
Args:
|
|
|
|
start (int): start number to begin the query from
|
2019-12-05 11:12:23 -07:00
|
|
|
limit (int): number of rows to retrieve
|
|
|
|
name (string): filter for user names
|
|
|
|
guests (bool): whether to in include guest users
|
|
|
|
deactivated (bool): whether to include deactivated users
|
2017-02-02 06:02:26 -07:00
|
|
|
Returns:
|
2019-12-05 11:12:23 -07:00
|
|
|
defer.Deferred: resolves to json list[dict[str, Any]]
|
2017-02-02 06:02:26 -07:00
|
|
|
"""
|
2019-12-05 11:12:23 -07:00
|
|
|
ret = yield self.store.get_users_paginate(
|
|
|
|
start, limit, name, guests, deactivated
|
|
|
|
)
|
2017-02-02 06:02:26 -07:00
|
|
|
|
2019-07-23 07:00:55 -06:00
|
|
|
return ret
|
2017-02-02 06:02:26 -07:00
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
|
|
|
def search_users(self, term):
|
|
|
|
"""Function to search users list for one or more users with
|
|
|
|
the matched term.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
term (str): search term
|
|
|
|
Returns:
|
|
|
|
defer.Deferred: resolves to list[dict[str, Any]]
|
|
|
|
"""
|
|
|
|
ret = yield self.store.search_users(term)
|
|
|
|
|
2019-07-23 07:00:55 -06:00
|
|
|
return ret
|
2019-07-01 10:55:11 -06:00
|
|
|
|
2019-08-27 06:12:27 -06:00
|
|
|
def get_user_server_admin(self, user):
|
|
|
|
"""
|
|
|
|
Get the admin bit on a user.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
user_id (UserID): the (necessarily local) user to manipulate
|
|
|
|
"""
|
|
|
|
return self.store.is_server_admin(user)
|
|
|
|
|
2019-08-27 03:14:00 -06:00
|
|
|
def set_user_server_admin(self, user, admin):
|
|
|
|
"""
|
|
|
|
Set the admin bit on a user.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
user_id (UserID): the (necessarily local) user to manipulate
|
|
|
|
admin (bool): whether or not the user should be an admin of this server
|
|
|
|
"""
|
|
|
|
return self.store.set_server_admin(user, admin)
|
|
|
|
|
2019-07-01 10:55:11 -06:00
|
|
|
@defer.inlineCallbacks
|
2019-07-04 04:07:09 -06:00
|
|
|
def export_user_data(self, user_id, writer):
|
2019-07-03 08:03:38 -06:00
|
|
|
"""Write all data we have on the user to the given writer.
|
2019-07-01 10:55:11 -06:00
|
|
|
|
|
|
|
Args:
|
|
|
|
user_id (str)
|
|
|
|
writer (ExfiltrationWriter)
|
|
|
|
|
|
|
|
Returns:
|
2019-07-04 04:07:09 -06:00
|
|
|
defer.Deferred: Resolves when all data for a user has been written.
|
|
|
|
The returned value is that returned by `writer.finished()`.
|
2019-07-01 10:55:11 -06:00
|
|
|
"""
|
|
|
|
# Get all rooms the user is in or has been in
|
|
|
|
rooms = yield self.store.get_rooms_for_user_where_membership_is(
|
|
|
|
user_id,
|
|
|
|
membership_list=(
|
|
|
|
Membership.JOIN,
|
|
|
|
Membership.LEAVE,
|
|
|
|
Membership.BAN,
|
|
|
|
Membership.INVITE,
|
|
|
|
),
|
|
|
|
)
|
|
|
|
|
|
|
|
# We only try and fetch events for rooms the user has been in. If
|
|
|
|
# they've been e.g. invited to a room without joining then we handle
|
|
|
|
# those seperately.
|
|
|
|
rooms_user_has_been_in = yield self.store.get_rooms_user_has_been_in(user_id)
|
|
|
|
|
|
|
|
for index, room in enumerate(rooms):
|
|
|
|
room_id = room.room_id
|
|
|
|
|
|
|
|
logger.info(
|
|
|
|
"[%s] Handling room %s, %d/%d", user_id, room_id, index + 1, len(rooms)
|
|
|
|
)
|
|
|
|
|
|
|
|
forgotten = yield self.store.did_forget(user_id, room_id)
|
|
|
|
if forgotten:
|
2019-07-04 04:07:09 -06:00
|
|
|
logger.info("[%s] User forgot room %d, ignoring", user_id, room_id)
|
2019-07-01 10:55:11 -06:00
|
|
|
continue
|
|
|
|
|
|
|
|
if room_id not in rooms_user_has_been_in:
|
|
|
|
# If we haven't been in the rooms then the filtering code below
|
|
|
|
# won't return anything, so we need to handle these cases
|
|
|
|
# explicitly.
|
|
|
|
|
|
|
|
if room.membership == Membership.INVITE:
|
|
|
|
event_id = room.event_id
|
|
|
|
invite = yield self.store.get_event(event_id, allow_none=True)
|
|
|
|
if invite:
|
|
|
|
invited_state = invite.unsigned["invite_room_state"]
|
|
|
|
writer.write_invite(room_id, invite, invited_state)
|
|
|
|
|
|
|
|
continue
|
|
|
|
|
|
|
|
# We only want to bother fetching events up to the last time they
|
|
|
|
# were joined. We estimate that point by looking at the
|
|
|
|
# stream_ordering of the last membership if it wasn't a join.
|
|
|
|
if room.membership == Membership.JOIN:
|
|
|
|
stream_ordering = yield self.store.get_room_max_stream_ordering()
|
|
|
|
else:
|
|
|
|
stream_ordering = room.stream_ordering
|
|
|
|
|
|
|
|
from_key = str(RoomStreamToken(0, 0))
|
|
|
|
to_key = str(RoomStreamToken(None, stream_ordering))
|
|
|
|
|
|
|
|
written_events = set() # Events that we've processed in this room
|
|
|
|
|
|
|
|
# We need to track gaps in the events stream so that we can then
|
|
|
|
# write out the state at those events. We do this by keeping track
|
|
|
|
# of events whose prev events we haven't seen.
|
|
|
|
|
|
|
|
# Map from event ID to prev events that haven't been processed,
|
|
|
|
# dict[str, set[str]].
|
|
|
|
event_to_unseen_prevs = {}
|
|
|
|
|
2019-07-04 04:07:09 -06:00
|
|
|
# The reverse mapping to above, i.e. map from unseen event to events
|
|
|
|
# that have the unseen event in their prev_events, i.e. the unseen
|
|
|
|
# events "children". dict[str, set[str]]
|
|
|
|
unseen_to_child_events = {}
|
2019-07-01 10:55:11 -06:00
|
|
|
|
|
|
|
# We fetch events in the room the user could see by fetching *all*
|
|
|
|
# events that we have and then filtering, this isn't the most
|
2019-07-03 08:03:38 -06:00
|
|
|
# efficient method perhaps but it does guarantee we get everything.
|
2019-07-01 10:55:11 -06:00
|
|
|
while True:
|
|
|
|
events, _ = yield self.store.paginate_room_events(
|
|
|
|
room_id, from_key, to_key, limit=100, direction="f"
|
|
|
|
)
|
|
|
|
if not events:
|
|
|
|
break
|
|
|
|
|
|
|
|
from_key = events[-1].internal_metadata.after
|
|
|
|
|
2019-10-23 10:25:54 -06:00
|
|
|
events = yield filter_events_for_client(self.storage, user_id, events)
|
2019-07-01 10:55:11 -06:00
|
|
|
|
|
|
|
writer.write_events(room_id, events)
|
|
|
|
|
|
|
|
# Update the extremity tracking dicts
|
|
|
|
for event in events:
|
|
|
|
# Check if we have any prev events that haven't been
|
|
|
|
# processed yet, and add those to the appropriate dicts.
|
|
|
|
unseen_events = set(event.prev_event_ids()) - written_events
|
|
|
|
if unseen_events:
|
|
|
|
event_to_unseen_prevs[event.event_id] = unseen_events
|
|
|
|
for unseen in unseen_events:
|
2019-07-04 04:07:09 -06:00
|
|
|
unseen_to_child_events.setdefault(unseen, set()).add(
|
2019-07-01 10:55:11 -06:00
|
|
|
event.event_id
|
|
|
|
)
|
|
|
|
|
|
|
|
# Now check if this event is an unseen prev event, if so
|
|
|
|
# then we remove this event from the appropriate dicts.
|
2019-07-04 04:07:09 -06:00
|
|
|
for child_id in unseen_to_child_events.pop(event.event_id, []):
|
2019-07-05 07:07:56 -06:00
|
|
|
event_to_unseen_prevs[child_id].discard(event.event_id)
|
2019-07-01 10:55:11 -06:00
|
|
|
|
|
|
|
written_events.add(event.event_id)
|
|
|
|
|
|
|
|
logger.info(
|
|
|
|
"Written %d events in room %s", len(written_events), room_id
|
|
|
|
)
|
|
|
|
|
|
|
|
# Extremities are the events who have at least one unseen prev event.
|
|
|
|
extremities = (
|
|
|
|
event_id
|
|
|
|
for event_id, unseen_prevs in event_to_unseen_prevs.items()
|
|
|
|
if unseen_prevs
|
|
|
|
)
|
|
|
|
for event_id in extremities:
|
|
|
|
if not event_to_unseen_prevs[event_id]:
|
|
|
|
continue
|
2019-10-23 10:25:54 -06:00
|
|
|
state = yield self.state_store.get_state_for_event(event_id)
|
2019-07-01 10:55:11 -06:00
|
|
|
writer.write_state(room_id, event_id, state)
|
|
|
|
|
2019-07-23 07:00:55 -06:00
|
|
|
return writer.finished()
|
2019-07-01 10:55:11 -06:00
|
|
|
|
|
|
|
|
|
|
|
class ExfiltrationWriter(object):
|
2019-07-04 04:07:09 -06:00
|
|
|
"""Interface used to specify how to write exported data.
|
2019-07-01 10:55:11 -06:00
|
|
|
"""
|
|
|
|
|
|
|
|
def write_events(self, room_id, events):
|
|
|
|
"""Write a batch of events for a room.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
room_id (str)
|
|
|
|
events (list[FrozenEvent])
|
|
|
|
"""
|
|
|
|
pass
|
|
|
|
|
|
|
|
def write_state(self, room_id, event_id, state):
|
|
|
|
"""Write the state at the given event in the room.
|
|
|
|
|
|
|
|
This only gets called for backward extremities rather than for each
|
|
|
|
event.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
room_id (str)
|
|
|
|
event_id (str)
|
2019-07-04 04:07:09 -06:00
|
|
|
state (dict[tuple[str, str], FrozenEvent])
|
2019-07-01 10:55:11 -06:00
|
|
|
"""
|
|
|
|
pass
|
|
|
|
|
|
|
|
def write_invite(self, room_id, event, state):
|
|
|
|
"""Write an invite for the room, with associated invite state.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
room_id (str)
|
2019-07-03 08:03:38 -06:00
|
|
|
event (FrozenEvent)
|
2019-07-04 04:07:09 -06:00
|
|
|
state (dict[tuple[str, str], dict]): A subset of the state at the
|
|
|
|
invite, with a subset of the event keys (type, state_key
|
|
|
|
content and sender)
|
2019-07-01 10:55:11 -06:00
|
|
|
"""
|
|
|
|
|
|
|
|
def finished(self):
|
2019-07-04 04:07:09 -06:00
|
|
|
"""Called when all data has succesfully been exported and written.
|
|
|
|
|
|
|
|
This functions return value is passed to the caller of
|
|
|
|
`export_user_data`.
|
2019-07-01 10:55:11 -06:00
|
|
|
"""
|
|
|
|
pass
|