Skip to content
This repository has been archived by the owner on Apr 26, 2024. It is now read-only.

Admin API to list, filter and sort rooms #6720

Merged
merged 23 commits into from
Jan 22, 2020
Merged
Show file tree
Hide file tree
Changes from 11 commits
Commits
Show all changes
23 commits
Select commit Hold shift + click to select a range
4f685c6
Admin API to list, filter and sort rooms
anoadragon453 Jan 5, 2020
6761961
Merge branch 'develop' of github.com:matrix-org/synapse into anoa/adm…
anoadragon453 Jan 16, 2020
908775c
Add changelog
anoadragon453 Jan 16, 2020
df32276
Add alias to subquery for postgres
anoadragon453 Jan 17, 2020
5aa30d8
Add documentation
anoadragon453 Jan 17, 2020
5552f97
Remove potentially confusing quotation marks
anoadragon453 Jan 17, 2020
a5a7e2e
Minor fixups
anoadragon453 Jan 17, 2020
9d30f4a
Fix LIMIT differences in Postgres vs. SQLite
anoadragon453 Jan 17, 2020
ad5edaa
debug
anoadragon453 Jan 17, 2020
973651d
more debugging
anoadragon453 Jan 17, 2020
ed5457c
Fix postgres-specific %-related bug
anoadragon453 Jan 17, 2020
9ca3bd0
Address review comments
anoadragon453 Jan 20, 2020
5205820
Address review comments. Return total rooms
anoadragon453 Jan 20, 2020
b449efd
Small docs clarification
anoadragon453 Jan 20, 2020
980cbd3
Indentation
anoadragon453 Jan 20, 2020
1cfd5e1
admin_api creation method refactor
anoadragon453 Jan 20, 2020
f91e692
Fix docstring
anoadragon453 Jan 20, 2020
93b2212
Add 'total_rooms' key to tests
anoadragon453 Jan 20, 2020
1c5ff45
Ensure pagination test returns rooms in a predicatable order
anoadragon453 Jan 20, 2020
6bebc12
Can't use regex pattern in method type definition
anoadragon453 Jan 20, 2020
8374cb1
Add offset parameter to the response
anoadragon453 Jan 21, 2020
7be87aa
Add prev_batch. next_token -> next_batch
anoadragon453 Jan 21, 2020
8c7f668
Don't raise exception on incorrect order_by value
anoadragon453 Jan 21, 2020
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions changelog.d/6720.feature
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
Add a new admin API to list and filter rooms on the server.
155 changes: 155 additions & 0 deletions docs/admin_api/rooms.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,155 @@
# List Room API

The List Room admin API allows server admins to get a list of rooms on their
server. There are various parameters available that allow for filtering and
sorting the returned list. This API supports pagination.

## Parameters

The following query parameters are available:

* `from` - Offset in the returned list. Defaults to `0`.
* `limit` - Maximum amount of rooms to return. Defaults to `100`.
* `order_by` - The method in which to sort the returned list of rooms. Valid values are:
- `"alphabetical"` - Rooms are ordered alphabetically by room name. This is the default.
- `"size"` - Rooms are ordered by the number of members. Largest to smallest.
* `dir` - Direction of room order. Either `"f"` for forwards or `b` for backwards. Setting
this value to `b` will reverse the above sort order. Defaults to `f`.
* `search_term` - Filter rooms by their room name. Search term can be contained in any
part of the room name. Defaults to no filtering.

The following fields are possible in the JSON response body:

* `rooms` - An array of objects, each containing information about a room.
- Room objects contain the following fields:
- `room_id` - The ID of the room.
- `name` - The name of the room.
- `canonical_alias` - The canonical (main) alias address of the room.
- `joined_members` - How many users are currently in the room.
* `next_token` - If this field is present, we know that there are potentially
more rooms on the server that did not all fit into this response.
We can use `next_token` to get the "next page" of results. To do
so, simply repeat your request, setting the `from` parameter to
the value of `next_token`.

## Usage

A standard request with no filtering:

```
GET /_synapse/admin/rooms

{}
```

Response:

```
{
"rooms": [
{
"room_id": "!OGEhHVWSdvArJzumhm:matrix.org",
"name": "Matrix HQ",
"canonical_alias": "#matrix:matrix.org",
"joined_members": 8326
},
...
{
"room_id": "!xYvNcQPhnkrdUmYczI:matrix.org",
"name": "This Week In Matrix (TWIM)",
"canonical_alias": "#twim:matrix.org",
"joined_members": 314
}
]
}
```

Filtering by room name:

```
GET /_synapse/admin/rooms?search_term=TWIM

{}
```

Response:

```
{
"rooms": [
{
"room_id": "!xYvNcQPhnkrdUmYczI:matrix.org",
"name": "This Week In Matrix (TWIM)",
"canonical_alias": "#twim:matrix.org",
"joined_members": 314
}
]
}
```

Paginating through a list of rooms:

```
GET /_synapse/admin/rooms?order_by=size

{}
```

Response:

```
{
"rooms": [
{
"room_id": "!OGEhHVWSdvArJzumhm:matrix.org",
"name": "Matrix HQ",
"canonical_alias": "#matrix:matrix.org",
"joined_members": 8326
},
... (98 hidden items) ...
{
"room_id": "!xYvNcQPhnkrdUmYczI:matrix.org",
"name": "This Week In Matrix (TWIM)",
"canonical_alias": "#twim:matrix.org",
"joined_members": 314
}
],
"next_token": 100
anoadragon453 marked this conversation as resolved.
Show resolved Hide resolved
}
```

The presence of the `next_token` parameter tells us that there are more rooms
than returned in this request, and we need to make another request to get them.
To get the next batch of room results, we repeat our request, setting the `from`
parameter to the value of `next_token`.

```
GET /_synapse/admin/rooms?order_by=size&from=100

{}
```

Response:

```
{
"rooms": [
{
"room_id": "!mscvqgqpHYjBGDxNym:matrix.org",
"name": "Music Theory",
"canonical_alias": "#musictheory:matrix.org",
"joined_members": 127
},
... (65 hidden items) ...
{
"room_id": "!twcBhHVdZlQWuuxBhN:termina.org.uk",
"name": "weechat-matrix",
"canonical_alias": "#weechat-matrix:termina.org.uk",
"joined_members": 137
}
]
}
```

Once the `next_token` parameter is not present, we know we've reached the end of
the list.
30 changes: 29 additions & 1 deletion synapse/handlers/admin.py
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@
# limitations under the License.

import logging
from typing import List
from typing import List, Optional

from synapse.api.constants import Membership
from synapse.events import FrozenEvent
Expand Down Expand Up @@ -92,6 +92,34 @@ async def get_users_paginate(self, start, limit, name, guests, deactivated):

return ret

async def get_rooms_paginate(
self,
start: int,
limit: int,
order_by: str,
reverse_order: bool,
search_term: Optional[str],
):
"""Function to retrieve a paginated list of rooms as json.

Args:
start: offset in the list
limit: maximum amount of rooms to retrieve
order_by: the sort order of the returned list. valid values:
* alphabetical (default)
* size
reverse_order: whether to reverse the room list
search_term: a string to filter room names by
Returns:
defer.Deferred: a tuple of list[dict[str, Any]], int|None. A list of
anoadragon453 marked this conversation as resolved.
Show resolved Hide resolved
room dicts and an integer, if not None, signifies more results can
be returned if the same call if repeated, substituting the value of
`start` for that of the returned int.
"""
return await self.store.get_rooms_paginate(
start, limit, order_by, reverse_order, search_term
)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

TBH I think we should probably just remove this method and call the store directly since we're not doing anything here.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The same should possibly be done for the other methods in synapse/handlers/admin.py?


async def search_users(self, term):
"""Function to search users list for one or more users with
the matched term.
Expand Down
3 changes: 2 additions & 1 deletion synapse/rest/admin/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -29,7 +29,7 @@
from synapse.rest.admin.groups import DeleteGroupAdminRestServlet
from synapse.rest.admin.media import ListMediaInRoom, register_servlets_for_media_repo
from synapse.rest.admin.purge_room_servlet import PurgeRoomServlet
from synapse.rest.admin.rooms import ShutdownRoomRestServlet
from synapse.rest.admin.rooms import ListRoomRestServlet, ShutdownRoomRestServlet
from synapse.rest.admin.server_notice_servlet import SendServerNoticeServlet
from synapse.rest.admin.users import (
AccountValidityRenewServlet,
Expand Down Expand Up @@ -188,6 +188,7 @@ def register_servlets(hs, http_server):
Register all the admin servlets.
"""
register_servlets_for_client_rest_resource(hs, http_server)
ListRoomRestServlet(hs).register(http_server)
PurgeRoomServlet(hs).register(http_server)
SendServerNoticeServlet(hs).register(http_server)
VersionServlet(hs).register(http_server)
Expand Down
60 changes: 60 additions & 0 deletions synapse/rest/admin/rooms.py
Original file line number Diff line number Diff line change
Expand Up @@ -15,10 +15,13 @@
import logging

from synapse.api.constants import Membership
from synapse.api.errors import Codes, SynapseError
from synapse.http.servlet import (
RestServlet,
assert_params_in_dict,
parse_integer,
parse_json_object_from_request,
parse_string,
)
from synapse.rest.admin._base import (
assert_user_is_admin,
Expand Down Expand Up @@ -155,3 +158,60 @@ async def on_POST(self, request, room_id):
"new_room_id": new_room_id,
},
)


class ListRoomRestServlet(RestServlet):
"""
List all rooms that are known to the homeserver. Results are returned
in a dictionary containing room information. Supports pagination.
"""

PATTERNS = historical_admin_path_patterns("/rooms")
anoadragon453 marked this conversation as resolved.
Show resolved Hide resolved

def __init__(self, hs):
self.hs = hs
anoadragon453 marked this conversation as resolved.
Show resolved Hide resolved
self.store = hs.get_datastore()
anoadragon453 marked this conversation as resolved.
Show resolved Hide resolved
self.auth = hs.get_auth()
self.admin_handler = hs.get_handlers().admin_handler

async def on_GET(self, request):
requester = await self.auth.get_user_by_req(request)
await assert_user_is_admin(self.auth, requester.user)

# Extract query parameters
start = parse_integer(request, "from", default=0)
limit = parse_integer(request, "limit", default=100)
order_by = parse_string(request, "order_by", default="alphabetical")
if order_by != "alphabetical" and order_by != "size":
anoadragon453 marked this conversation as resolved.
Show resolved Hide resolved
raise SynapseError(
400,
"Unknown value for order_by: %s" % (order_by,),
errcode=Codes.INVALID_PARAM,
)

search_term = parse_string(request, "search_term")
if search_term == "":
raise SynapseError(
400,
"search_term cannot be an empty string",
errcode=Codes.INVALID_PARAM,
)

direction = parse_string(request, "dir", default="f")
if direction != "f" and direction != "b":
anoadragon453 marked this conversation as resolved.
Show resolved Hide resolved
raise SynapseError(
400, "Unknown direction: %s" % (direction,), errcode=Codes.INVALID_PARAM
)

reverse_order = True if direction == "b" else False

# Return list of rooms according to parameters
rooms, next_token = await self.admin_handler.get_rooms_paginate(
start, limit, order_by, reverse_order, search_term
)
response = {"rooms": rooms}

if next_token:
response["next_token"] = next_token

return 200, response
Loading