2018-01-25 21:25:03 +00:00
|
|
|
|
#
|
2023-11-21 15:29:58 -05:00
|
|
|
|
# This file is licensed under the Affero General Public License (AGPL) version 3.
|
|
|
|
|
#
|
2024-01-23 11:26:48 +00:00
|
|
|
|
# Copyright 2018-2021 The Matrix.org Foundation C.I.C.
|
2023-11-21 15:29:58 -05:00
|
|
|
|
# Copyright (C) 2023 New Vector, Ltd
|
|
|
|
|
#
|
|
|
|
|
# This program is free software: you can redistribute it and/or modify
|
|
|
|
|
# it under the terms of the GNU Affero General Public License as
|
|
|
|
|
# published by the Free Software Foundation, either version 3 of the
|
|
|
|
|
# License, or (at your option) any later version.
|
|
|
|
|
#
|
|
|
|
|
# See the GNU Affero General Public License for more details:
|
|
|
|
|
# <https://www.gnu.org/licenses/agpl-3.0.html>.
|
|
|
|
|
#
|
|
|
|
|
# Originally licensed under the Apache License, Version 2.0:
|
|
|
|
|
# <http://www.apache.org/licenses/LICENSE-2.0>.
|
|
|
|
|
#
|
|
|
|
|
# [This file includes modifications made by New Vector Limited]
|
2018-01-25 21:25:03 +00:00
|
|
|
|
#
|
|
|
|
|
#
|
2022-12-12 13:21:17 +01:00
|
|
|
|
import re
|
2023-10-26 13:01:36 -04:00
|
|
|
|
from typing import Any, Dict, List, Optional, Set, Tuple, cast
|
2021-10-05 18:35:25 +01:00
|
|
|
|
from unittest import mock
|
2021-10-04 12:45:51 +01:00
|
|
|
|
from unittest.mock import Mock, patch
|
2021-09-30 11:04:40 +01:00
|
|
|
|
|
|
|
|
|
from twisted.test.proto_helpers import MemoryReactor
|
|
|
|
|
|
2021-10-05 18:35:25 +01:00
|
|
|
|
from synapse.api.constants import EventTypes, Membership, UserTypes
|
2021-10-04 12:45:51 +01:00
|
|
|
|
from synapse.appservice import ApplicationService
|
2021-09-30 11:04:40 +01:00
|
|
|
|
from synapse.rest import admin
|
2021-10-04 12:45:51 +01:00
|
|
|
|
from synapse.rest.client import login, register, room
|
2021-09-30 11:04:40 +01:00
|
|
|
|
from synapse.server import HomeServer
|
|
|
|
|
from synapse.storage import DataStore
|
2021-11-29 16:57:06 +00:00
|
|
|
|
from synapse.storage.background_updates import _BackgroundUpdateHandler
|
2023-02-20 12:00:18 +00:00
|
|
|
|
from synapse.storage.databases.main import user_directory
|
|
|
|
|
from synapse.storage.databases.main.user_directory import (
|
|
|
|
|
_parse_words_with_icu,
|
|
|
|
|
_parse_words_with_regex,
|
|
|
|
|
)
|
2021-10-05 18:35:25 +01:00
|
|
|
|
from synapse.storage.roommember import ProfileInfo
|
2021-09-30 11:04:40 +01:00
|
|
|
|
from synapse.util import Clock
|
2018-01-25 21:25:03 +00:00
|
|
|
|
|
2023-01-26 11:15:50 +01:00
|
|
|
|
from tests.server import ThreadedMemoryReactorClock
|
2021-10-04 12:45:51 +01:00
|
|
|
|
from tests.test_utils.event_injection import inject_member_event
|
2021-04-06 07:21:02 -04:00
|
|
|
|
from tests.unittest import HomeserverTestCase, override_config
|
2018-01-25 21:25:03 +00:00
|
|
|
|
|
2022-12-12 13:21:17 +01:00
|
|
|
|
try:
|
|
|
|
|
import icu
|
|
|
|
|
except ImportError:
|
|
|
|
|
icu = None # type: ignore
|
|
|
|
|
|
|
|
|
|
|
2018-01-25 21:25:03 +00:00
|
|
|
|
ALICE = "@alice:a"
|
|
|
|
|
BOB = "@bob:b"
|
|
|
|
|
BOBBY = "@bobby:a"
|
2020-12-17 14:42:30 +01:00
|
|
|
|
# The localpart isn't 'Bela' on purpose so we can test looking up display names.
|
2023-02-20 12:00:18 +00:00
|
|
|
|
BELA = "@somenickname:example.org"
|
2018-01-25 21:25:03 +00:00
|
|
|
|
|
|
|
|
|
|
2021-09-30 11:04:40 +01:00
|
|
|
|
class GetUserDirectoryTables:
|
|
|
|
|
"""Helper functions that we want to reuse in tests/handlers/test_user_directory.py"""
|
|
|
|
|
|
|
|
|
|
def __init__(self, store: DataStore):
|
|
|
|
|
self.store = store
|
|
|
|
|
|
2021-10-15 15:53:05 +01:00
|
|
|
|
async def get_users_in_public_rooms(self) -> Set[Tuple[str, str]]:
|
2021-10-05 18:35:25 +01:00
|
|
|
|
"""Fetch the entire `users_in_public_rooms` table.
|
|
|
|
|
|
|
|
|
|
Returns a list of tuples (user_id, room_id) where room_id is public and
|
|
|
|
|
contains the user with the given id.
|
|
|
|
|
"""
|
2023-10-26 13:01:36 -04:00
|
|
|
|
r = cast(
|
|
|
|
|
List[Tuple[str, str]],
|
|
|
|
|
await self.store.db_pool.simple_select_list(
|
|
|
|
|
"users_in_public_rooms", None, ("user_id", "room_id")
|
|
|
|
|
),
|
2021-09-30 11:04:40 +01:00
|
|
|
|
)
|
2023-10-26 13:01:36 -04:00
|
|
|
|
return set(r)
|
2021-09-30 11:04:40 +01:00
|
|
|
|
|
2021-10-15 15:53:05 +01:00
|
|
|
|
async def get_users_who_share_private_rooms(self) -> Set[Tuple[str, str, str]]:
|
2021-10-05 18:35:25 +01:00
|
|
|
|
"""Fetch the entire `users_who_share_private_rooms` table.
|
|
|
|
|
|
2021-10-15 15:53:05 +01:00
|
|
|
|
Returns a set of tuples (user_id, other_user_id, room_id) corresponding
|
|
|
|
|
to the rows of `users_who_share_private_rooms`.
|
2021-10-05 18:35:25 +01:00
|
|
|
|
"""
|
|
|
|
|
|
2023-10-26 13:01:36 -04:00
|
|
|
|
rows = cast(
|
|
|
|
|
List[Tuple[str, str, str]],
|
|
|
|
|
await self.store.db_pool.simple_select_list(
|
|
|
|
|
"users_who_share_private_rooms",
|
|
|
|
|
None,
|
|
|
|
|
["user_id", "other_user_id", "room_id"],
|
|
|
|
|
),
|
2021-09-30 11:04:40 +01:00
|
|
|
|
)
|
2023-10-26 13:01:36 -04:00
|
|
|
|
return set(rows)
|
2021-09-30 11:04:40 +01:00
|
|
|
|
|
2021-10-04 12:45:51 +01:00
|
|
|
|
async def get_users_in_user_directory(self) -> Set[str]:
|
2021-10-05 18:35:25 +01:00
|
|
|
|
"""Fetch the set of users in the `user_directory` table.
|
|
|
|
|
|
|
|
|
|
This is useful when checking we've correctly excluded users from the directory.
|
|
|
|
|
"""
|
2023-10-26 13:01:36 -04:00
|
|
|
|
result = cast(
|
|
|
|
|
List[Tuple[str]],
|
|
|
|
|
await self.store.db_pool.simple_select_list(
|
|
|
|
|
"user_directory",
|
|
|
|
|
None,
|
|
|
|
|
["user_id"],
|
|
|
|
|
),
|
2021-10-04 12:45:51 +01:00
|
|
|
|
)
|
2023-10-26 13:01:36 -04:00
|
|
|
|
return {row[0] for row in result}
|
2021-10-04 12:45:51 +01:00
|
|
|
|
|
2021-10-05 18:35:25 +01:00
|
|
|
|
async def get_profiles_in_user_directory(self) -> Dict[str, ProfileInfo]:
|
|
|
|
|
"""Fetch users and their profiles from the `user_directory` table.
|
|
|
|
|
|
|
|
|
|
This is useful when we want to inspect display names and avatars.
|
|
|
|
|
It's almost the entire contents of the `user_directory` table: the only
|
|
|
|
|
thing missing is an unused room_id column.
|
|
|
|
|
"""
|
2023-10-26 13:01:36 -04:00
|
|
|
|
rows = cast(
|
|
|
|
|
List[Tuple[str, Optional[str], Optional[str]]],
|
|
|
|
|
await self.store.db_pool.simple_select_list(
|
|
|
|
|
"user_directory",
|
|
|
|
|
None,
|
|
|
|
|
("user_id", "display_name", "avatar_url"),
|
|
|
|
|
),
|
2021-10-05 18:35:25 +01:00
|
|
|
|
)
|
|
|
|
|
return {
|
2023-10-26 13:01:36 -04:00
|
|
|
|
user_id: ProfileInfo(display_name=display_name, avatar_url=avatar_url)
|
|
|
|
|
for user_id, display_name, avatar_url in rows
|
2021-10-05 18:35:25 +01:00
|
|
|
|
}
|
|
|
|
|
|
2021-10-15 15:53:05 +01:00
|
|
|
|
async def get_tables(
|
|
|
|
|
self,
|
|
|
|
|
) -> Tuple[Set[str], Set[Tuple[str, str]], Set[Tuple[str, str, str]]]:
|
|
|
|
|
"""Multiple tests want to inspect these tables, so expose them together."""
|
|
|
|
|
return (
|
|
|
|
|
await self.get_users_in_user_directory(),
|
|
|
|
|
await self.get_users_in_public_rooms(),
|
|
|
|
|
await self.get_users_who_share_private_rooms(),
|
|
|
|
|
)
|
|
|
|
|
|
2021-09-30 11:04:40 +01:00
|
|
|
|
|
|
|
|
|
class UserDirectoryInitialPopulationTestcase(HomeserverTestCase):
|
|
|
|
|
"""Ensure that rebuilding the directory writes the correct data to the DB.
|
|
|
|
|
|
|
|
|
|
See also tests/handlers/test_user_directory.py for similar checks. They
|
|
|
|
|
test the incremental updates, rather than the big rebuild.
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
servlets = [
|
|
|
|
|
login.register_servlets,
|
2021-10-04 12:45:51 +01:00
|
|
|
|
admin.register_servlets,
|
2021-09-30 11:04:40 +01:00
|
|
|
|
room.register_servlets,
|
2021-10-04 12:45:51 +01:00
|
|
|
|
register.register_servlets,
|
2021-09-30 11:04:40 +01:00
|
|
|
|
]
|
|
|
|
|
|
2023-01-26 11:15:50 +01:00
|
|
|
|
def make_homeserver(
|
|
|
|
|
self, reactor: ThreadedMemoryReactorClock, clock: Clock
|
|
|
|
|
) -> HomeServer:
|
2021-10-04 12:45:51 +01:00
|
|
|
|
self.appservice = ApplicationService(
|
|
|
|
|
token="i_am_an_app_service",
|
|
|
|
|
id="1234",
|
|
|
|
|
namespaces={"users": [{"regex": r"@as_user.*", "exclusive": True}]},
|
|
|
|
|
sender="@as:test",
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
mock_load_appservices = Mock(return_value=[self.appservice])
|
|
|
|
|
with patch(
|
|
|
|
|
"synapse.storage.databases.main.appservice.load_appservices",
|
|
|
|
|
mock_load_appservices,
|
|
|
|
|
):
|
|
|
|
|
hs = super().make_homeserver(reactor, clock)
|
|
|
|
|
return hs
|
|
|
|
|
|
2021-09-30 11:04:40 +01:00
|
|
|
|
def prepare(self, reactor: MemoryReactor, clock: Clock, hs: HomeServer) -> None:
|
2022-02-23 11:04:02 +00:00
|
|
|
|
self.store = hs.get_datastores().main
|
2021-09-30 11:04:40 +01:00
|
|
|
|
self.user_dir_helper = GetUserDirectoryTables(self.store)
|
|
|
|
|
|
|
|
|
|
def _purge_and_rebuild_user_dir(self) -> None:
|
|
|
|
|
"""Nuke the user directory tables, start the background process to
|
|
|
|
|
repopulate them, and wait for the process to complete. This allows us
|
|
|
|
|
to inspect the outcome of the background process alone, without any of
|
|
|
|
|
the other incremental updates.
|
|
|
|
|
"""
|
|
|
|
|
self.get_success(self.store.update_user_directory_stream_pos(None))
|
|
|
|
|
self.get_success(self.store.delete_all_from_user_dir())
|
|
|
|
|
|
|
|
|
|
shares_private = self.get_success(
|
|
|
|
|
self.user_dir_helper.get_users_who_share_private_rooms()
|
|
|
|
|
)
|
|
|
|
|
public_users = self.get_success(
|
|
|
|
|
self.user_dir_helper.get_users_in_public_rooms()
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
# Nothing updated yet
|
2021-10-15 15:53:05 +01:00
|
|
|
|
self.assertEqual(shares_private, set())
|
|
|
|
|
self.assertEqual(public_users, set())
|
2021-09-30 11:04:40 +01:00
|
|
|
|
|
|
|
|
|
# Ugh, have to reset this flag
|
|
|
|
|
self.store.db_pool.updates._all_done = False
|
|
|
|
|
|
|
|
|
|
self.get_success(
|
|
|
|
|
self.store.db_pool.simple_insert(
|
|
|
|
|
"background_updates",
|
|
|
|
|
{
|
|
|
|
|
"update_name": "populate_user_directory_createtables",
|
|
|
|
|
"progress_json": "{}",
|
|
|
|
|
},
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
self.get_success(
|
|
|
|
|
self.store.db_pool.simple_insert(
|
|
|
|
|
"background_updates",
|
|
|
|
|
{
|
|
|
|
|
"update_name": "populate_user_directory_process_rooms",
|
|
|
|
|
"progress_json": "{}",
|
|
|
|
|
"depends_on": "populate_user_directory_createtables",
|
|
|
|
|
},
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
self.get_success(
|
|
|
|
|
self.store.db_pool.simple_insert(
|
|
|
|
|
"background_updates",
|
|
|
|
|
{
|
|
|
|
|
"update_name": "populate_user_directory_process_users",
|
|
|
|
|
"progress_json": "{}",
|
|
|
|
|
"depends_on": "populate_user_directory_process_rooms",
|
|
|
|
|
},
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
self.get_success(
|
|
|
|
|
self.store.db_pool.simple_insert(
|
|
|
|
|
"background_updates",
|
|
|
|
|
{
|
|
|
|
|
"update_name": "populate_user_directory_cleanup",
|
|
|
|
|
"progress_json": "{}",
|
|
|
|
|
"depends_on": "populate_user_directory_process_users",
|
|
|
|
|
},
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
|
2021-10-06 13:56:45 +01:00
|
|
|
|
self.wait_for_background_updates()
|
2021-09-30 11:04:40 +01:00
|
|
|
|
|
|
|
|
|
def test_initial(self) -> None:
|
|
|
|
|
"""
|
|
|
|
|
The user directory's initial handler correctly updates the search tables.
|
|
|
|
|
"""
|
|
|
|
|
u1 = self.register_user("user1", "pass")
|
|
|
|
|
u1_token = self.login(u1, "pass")
|
|
|
|
|
u2 = self.register_user("user2", "pass")
|
|
|
|
|
u2_token = self.login(u2, "pass")
|
|
|
|
|
u3 = self.register_user("user3", "pass")
|
|
|
|
|
u3_token = self.login(u3, "pass")
|
|
|
|
|
|
|
|
|
|
room = self.helper.create_room_as(u1, is_public=True, tok=u1_token)
|
|
|
|
|
self.helper.invite(room, src=u1, targ=u2, tok=u1_token)
|
|
|
|
|
self.helper.join(room, user=u2, tok=u2_token)
|
|
|
|
|
|
|
|
|
|
private_room = self.helper.create_room_as(u1, is_public=False, tok=u1_token)
|
|
|
|
|
self.helper.invite(private_room, src=u1, targ=u3, tok=u1_token)
|
|
|
|
|
self.helper.join(private_room, user=u3, tok=u3_token)
|
|
|
|
|
|
|
|
|
|
# Do the initial population of the user directory via the background update
|
|
|
|
|
self._purge_and_rebuild_user_dir()
|
|
|
|
|
|
2021-10-15 15:53:05 +01:00
|
|
|
|
users, in_public, in_private = self.get_success(
|
|
|
|
|
self.user_dir_helper.get_tables()
|
2021-09-30 11:04:40 +01:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
# User 1 and User 2 are in the same public room
|
2021-10-15 15:53:05 +01:00
|
|
|
|
self.assertEqual(in_public, {(u1, room), (u2, room)})
|
2021-09-30 11:04:40 +01:00
|
|
|
|
# User 1 and User 3 share private rooms
|
2021-10-15 15:53:05 +01:00
|
|
|
|
self.assertEqual(in_private, {(u1, u3, private_room), (u3, u1, private_room)})
|
2021-10-04 12:45:51 +01:00
|
|
|
|
# All three should have entries in the directory
|
|
|
|
|
self.assertEqual(users, {u1, u2, u3})
|
|
|
|
|
|
2021-10-13 10:38:22 +01:00
|
|
|
|
# The next four tests (test_population_excludes_*) all set up
|
2021-10-04 12:45:51 +01:00
|
|
|
|
# - A normal user included in the user dir
|
|
|
|
|
# - A public and private room created by that user
|
|
|
|
|
# - A user excluded from the room dir, belonging to both rooms
|
|
|
|
|
|
|
|
|
|
# They match similar logic in handlers/test_user_directory.py But that tests
|
|
|
|
|
# updating the directory; this tests rebuilding it from scratch.
|
|
|
|
|
|
|
|
|
|
def _create_rooms_and_inject_memberships(
|
|
|
|
|
self, creator: str, token: str, joiner: str
|
|
|
|
|
) -> Tuple[str, str]:
|
|
|
|
|
"""Create a public and private room as a normal user.
|
|
|
|
|
Then get the `joiner` into those rooms.
|
|
|
|
|
"""
|
|
|
|
|
public_room = self.helper.create_room_as(
|
|
|
|
|
creator,
|
|
|
|
|
is_public=True,
|
|
|
|
|
# See https://github.com/matrix-org/synapse/issues/10951
|
|
|
|
|
extra_content={"visibility": "public"},
|
|
|
|
|
tok=token,
|
|
|
|
|
)
|
|
|
|
|
private_room = self.helper.create_room_as(creator, is_public=False, tok=token)
|
|
|
|
|
|
|
|
|
|
# HACK: get the user into these rooms
|
|
|
|
|
self.get_success(inject_member_event(self.hs, public_room, joiner, "join"))
|
|
|
|
|
self.get_success(inject_member_event(self.hs, private_room, joiner, "join"))
|
|
|
|
|
|
|
|
|
|
return public_room, private_room
|
|
|
|
|
|
|
|
|
|
def _check_room_sharing_tables(
|
|
|
|
|
self, normal_user: str, public_room: str, private_room: str
|
|
|
|
|
) -> None:
|
|
|
|
|
# After rebuilding the directory, we should only see the normal user.
|
2021-10-15 15:53:05 +01:00
|
|
|
|
users, in_public, in_private = self.get_success(
|
|
|
|
|
self.user_dir_helper.get_tables()
|
2021-10-04 12:45:51 +01:00
|
|
|
|
)
|
2021-10-15 15:53:05 +01:00
|
|
|
|
self.assertEqual(users, {normal_user})
|
|
|
|
|
self.assertEqual(in_public, {(normal_user, public_room)})
|
|
|
|
|
self.assertEqual(in_private, set())
|
2021-10-04 12:45:51 +01:00
|
|
|
|
|
|
|
|
|
def test_population_excludes_support_user(self) -> None:
|
|
|
|
|
# Create a normal and support user.
|
|
|
|
|
user = self.register_user("user", "pass")
|
|
|
|
|
token = self.login(user, "pass")
|
|
|
|
|
support = "@support1:test"
|
|
|
|
|
self.get_success(
|
|
|
|
|
self.store.register_user(
|
|
|
|
|
user_id=support, password_hash=None, user_type=UserTypes.SUPPORT
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
# Join the support user to rooms owned by the normal user.
|
|
|
|
|
public, private = self._create_rooms_and_inject_memberships(
|
|
|
|
|
user, token, support
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
# Rebuild the directory.
|
|
|
|
|
self._purge_and_rebuild_user_dir()
|
|
|
|
|
|
|
|
|
|
# Check the support user is not in the directory.
|
|
|
|
|
self._check_room_sharing_tables(user, public, private)
|
|
|
|
|
|
|
|
|
|
def test_population_excludes_deactivated_user(self) -> None:
|
|
|
|
|
user = self.register_user("naughty", "pass")
|
|
|
|
|
admin = self.register_user("admin", "pass", admin=True)
|
|
|
|
|
admin_token = self.login(admin, "pass")
|
|
|
|
|
|
|
|
|
|
# Deactivate the user.
|
|
|
|
|
channel = self.make_request(
|
|
|
|
|
"PUT",
|
|
|
|
|
f"/_synapse/admin/v2/users/{user}",
|
|
|
|
|
access_token=admin_token,
|
|
|
|
|
content={"deactivated": True},
|
|
|
|
|
)
|
|
|
|
|
self.assertEqual(channel.code, 200)
|
|
|
|
|
self.assertEqual(channel.json_body["deactivated"], True)
|
|
|
|
|
|
|
|
|
|
# Join the deactivated user to rooms owned by the admin.
|
|
|
|
|
# Is this something that could actually happen outside of a test?
|
|
|
|
|
public, private = self._create_rooms_and_inject_memberships(
|
|
|
|
|
admin, admin_token, user
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
# Rebuild the user dir. The deactivated user should be missing.
|
|
|
|
|
self._purge_and_rebuild_user_dir()
|
|
|
|
|
self._check_room_sharing_tables(admin, public, private)
|
|
|
|
|
|
|
|
|
|
def test_population_excludes_appservice_user(self) -> None:
|
|
|
|
|
# Register an AS user.
|
|
|
|
|
user = self.register_user("user", "pass")
|
|
|
|
|
token = self.login(user, "pass")
|
2022-02-02 09:59:55 +00:00
|
|
|
|
as_user, _ = self.register_appservice_user(
|
|
|
|
|
"as_user_potato", self.appservice.token
|
|
|
|
|
)
|
2021-10-04 12:45:51 +01:00
|
|
|
|
|
|
|
|
|
# Join the AS user to rooms owned by the normal user.
|
|
|
|
|
public, private = self._create_rooms_and_inject_memberships(
|
|
|
|
|
user, token, as_user
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
# Rebuild the directory.
|
|
|
|
|
self._purge_and_rebuild_user_dir()
|
|
|
|
|
|
|
|
|
|
# Check the AS user is not in the directory.
|
|
|
|
|
self._check_room_sharing_tables(user, public, private)
|
|
|
|
|
|
2021-10-13 10:38:22 +01:00
|
|
|
|
def test_population_excludes_appservice_sender(self) -> None:
|
|
|
|
|
user = self.register_user("user", "pass")
|
|
|
|
|
token = self.login(user, "pass")
|
|
|
|
|
|
|
|
|
|
# Join the AS sender to rooms owned by the normal user.
|
|
|
|
|
public, private = self._create_rooms_and_inject_memberships(
|
|
|
|
|
user, token, self.appservice.sender
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
# Rebuild the directory.
|
|
|
|
|
self._purge_and_rebuild_user_dir()
|
|
|
|
|
|
|
|
|
|
# Check the AS sender is not in the directory.
|
|
|
|
|
self._check_room_sharing_tables(user, public, private)
|
|
|
|
|
|
2021-10-05 18:35:25 +01:00
|
|
|
|
def test_population_conceals_private_nickname(self) -> None:
|
|
|
|
|
# Make a private room, and set a nickname within
|
|
|
|
|
user = self.register_user("aaaa", "pass")
|
|
|
|
|
user_token = self.login(user, "pass")
|
|
|
|
|
private_room = self.helper.create_room_as(user, is_public=False, tok=user_token)
|
|
|
|
|
self.helper.send_state(
|
|
|
|
|
private_room,
|
|
|
|
|
EventTypes.Member,
|
|
|
|
|
state_key=user,
|
|
|
|
|
body={"membership": Membership.JOIN, "displayname": "BBBB"},
|
|
|
|
|
tok=user_token,
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
# Rebuild the user directory. Make the rescan of the `users` table a no-op
|
|
|
|
|
# so we only see the effect of scanning the `room_memberships` table.
|
|
|
|
|
async def mocked_process_users(*args: Any, **kwargs: Any) -> int:
|
|
|
|
|
await self.store.db_pool.updates._end_background_update(
|
|
|
|
|
"populate_user_directory_process_users"
|
|
|
|
|
)
|
|
|
|
|
return 1
|
|
|
|
|
|
|
|
|
|
with mock.patch.dict(
|
|
|
|
|
self.store.db_pool.updates._background_update_handlers,
|
2021-11-29 16:57:06 +00:00
|
|
|
|
populate_user_directory_process_users=_BackgroundUpdateHandler(
|
|
|
|
|
mocked_process_users,
|
|
|
|
|
),
|
2021-10-05 18:35:25 +01:00
|
|
|
|
):
|
|
|
|
|
self._purge_and_rebuild_user_dir()
|
|
|
|
|
|
|
|
|
|
# Local users are ignored by the scan over rooms
|
|
|
|
|
users = self.get_success(self.user_dir_helper.get_profiles_in_user_directory())
|
|
|
|
|
self.assertEqual(users, {})
|
|
|
|
|
|
|
|
|
|
# Do a full rebuild including the scan over the `users` table. The local
|
|
|
|
|
# user should appear with their profile name.
|
|
|
|
|
self._purge_and_rebuild_user_dir()
|
|
|
|
|
users = self.get_success(self.user_dir_helper.get_profiles_in_user_directory())
|
|
|
|
|
self.assertEqual(
|
|
|
|
|
users, {user: ProfileInfo(display_name="aaaa", avatar_url=None)}
|
|
|
|
|
)
|
|
|
|
|
|
2021-09-30 11:04:40 +01:00
|
|
|
|
|
2021-04-06 07:21:02 -04:00
|
|
|
|
class UserDirectoryStoreTestCase(HomeserverTestCase):
|
2023-02-20 12:00:18 +00:00
|
|
|
|
use_icu = False
|
|
|
|
|
|
2021-09-30 11:04:40 +01:00
|
|
|
|
def prepare(self, reactor: MemoryReactor, clock: Clock, hs: HomeServer) -> None:
|
2022-02-23 11:04:02 +00:00
|
|
|
|
self.store = hs.get_datastores().main
|
2018-01-25 21:25:03 +00:00
|
|
|
|
|
|
|
|
|
# alice and bob are both in !room_id. bobby is not but shares
|
|
|
|
|
# a homeserver with alice.
|
2021-04-06 07:21:02 -04:00
|
|
|
|
self.get_success(self.store.update_profile_in_user_dir(ALICE, "alice", None))
|
|
|
|
|
self.get_success(self.store.update_profile_in_user_dir(BOB, "bob", None))
|
|
|
|
|
self.get_success(self.store.update_profile_in_user_dir(BOBBY, "bobby", None))
|
|
|
|
|
self.get_success(self.store.update_profile_in_user_dir(BELA, "Bela", None))
|
|
|
|
|
self.get_success(self.store.add_users_in_public_rooms("!room:id", (ALICE, BOB)))
|
2018-01-25 21:25:03 +00:00
|
|
|
|
|
2023-02-20 12:00:18 +00:00
|
|
|
|
self._restore_use_icu = user_directory.USE_ICU
|
|
|
|
|
user_directory.USE_ICU = self.use_icu
|
|
|
|
|
|
|
|
|
|
def tearDown(self) -> None:
|
|
|
|
|
user_directory.USE_ICU = self._restore_use_icu
|
|
|
|
|
|
2021-09-30 11:04:40 +01:00
|
|
|
|
def test_search_user_dir(self) -> None:
|
2018-01-25 21:25:03 +00:00
|
|
|
|
# normally when alice searches the directory she should just find
|
|
|
|
|
# bob because bobby doesn't share a room with her.
|
2021-04-06 07:21:02 -04:00
|
|
|
|
r = self.get_success(self.store.search_user_dir(ALICE, "bob", 10))
|
2018-01-25 21:25:03 +00:00
|
|
|
|
self.assertFalse(r["limited"])
|
|
|
|
|
self.assertEqual(1, len(r["results"]))
|
|
|
|
|
self.assertDictEqual(
|
|
|
|
|
r["results"][0], {"user_id": BOB, "display_name": "bob", "avatar_url": None}
|
|
|
|
|
)
|
|
|
|
|
|
2021-04-06 07:21:02 -04:00
|
|
|
|
@override_config({"user_directory": {"search_all_users": True}})
|
2021-09-30 11:04:40 +01:00
|
|
|
|
def test_search_user_dir_all_users(self) -> None:
|
2021-04-06 07:21:02 -04:00
|
|
|
|
r = self.get_success(self.store.search_user_dir(ALICE, "bob", 10))
|
|
|
|
|
self.assertFalse(r["limited"])
|
|
|
|
|
self.assertEqual(2, len(r["results"]))
|
|
|
|
|
self.assertDictEqual(
|
|
|
|
|
r["results"][0],
|
|
|
|
|
{"user_id": BOB, "display_name": "bob", "avatar_url": None},
|
|
|
|
|
)
|
|
|
|
|
self.assertDictEqual(
|
|
|
|
|
r["results"][1],
|
|
|
|
|
{"user_id": BOBBY, "display_name": "bobby", "avatar_url": None},
|
|
|
|
|
)
|
2020-12-17 14:42:30 +01:00
|
|
|
|
|
2022-12-07 11:19:43 +00:00
|
|
|
|
@override_config({"user_directory": {"search_all_users": True}})
|
|
|
|
|
def test_search_user_limit_correct(self) -> None:
|
|
|
|
|
r = self.get_success(self.store.search_user_dir(ALICE, "bob", 1))
|
|
|
|
|
self.assertTrue(r["limited"])
|
|
|
|
|
self.assertEqual(1, len(r["results"]))
|
|
|
|
|
|
2021-04-06 07:21:02 -04:00
|
|
|
|
@override_config({"user_directory": {"search_all_users": True}})
|
2021-09-30 11:04:40 +01:00
|
|
|
|
def test_search_user_dir_stop_words(self) -> None:
|
2020-12-17 14:42:30 +01:00
|
|
|
|
"""Tests that a user can look up another user by searching for the start if its
|
|
|
|
|
display name even if that name happens to be a common English word that would
|
|
|
|
|
usually be ignored in full text searches.
|
|
|
|
|
"""
|
2021-04-06 07:21:02 -04:00
|
|
|
|
r = self.get_success(self.store.search_user_dir(ALICE, "be", 10))
|
|
|
|
|
self.assertFalse(r["limited"])
|
|
|
|
|
self.assertEqual(1, len(r["results"]))
|
|
|
|
|
self.assertDictEqual(
|
|
|
|
|
r["results"][0],
|
|
|
|
|
{"user_id": BELA, "display_name": "Bela", "avatar_url": None},
|
|
|
|
|
)
|
2022-12-12 13:21:17 +01:00
|
|
|
|
|
2023-02-20 12:00:18 +00:00
|
|
|
|
@override_config({"user_directory": {"search_all_users": True}})
|
|
|
|
|
def test_search_user_dir_start_of_user_id(self) -> None:
|
|
|
|
|
"""Tests that a user can look up another user by searching for the start
|
|
|
|
|
of their user ID.
|
|
|
|
|
"""
|
|
|
|
|
r = self.get_success(self.store.search_user_dir(ALICE, "somenickname:exa", 10))
|
|
|
|
|
self.assertFalse(r["limited"])
|
|
|
|
|
self.assertEqual(1, len(r["results"]))
|
|
|
|
|
self.assertDictEqual(
|
|
|
|
|
r["results"][0],
|
|
|
|
|
{"user_id": BELA, "display_name": "Bela", "avatar_url": None},
|
|
|
|
|
)
|
|
|
|
|
|
2023-02-24 13:39:45 +00:00
|
|
|
|
@override_config({"user_directory": {"search_all_users": True}})
|
|
|
|
|
def test_search_user_dir_ascii_case_insensitivity(self) -> None:
|
|
|
|
|
"""Tests that a user can look up another user by searching for their name in a
|
|
|
|
|
different case.
|
|
|
|
|
"""
|
|
|
|
|
CHARLIE = "@someuser:example.org"
|
|
|
|
|
self.get_success(
|
|
|
|
|
self.store.update_profile_in_user_dir(CHARLIE, "Charlie", None)
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
r = self.get_success(self.store.search_user_dir(ALICE, "cHARLIE", 10))
|
|
|
|
|
self.assertFalse(r["limited"])
|
|
|
|
|
self.assertEqual(1, len(r["results"]))
|
|
|
|
|
self.assertDictEqual(
|
|
|
|
|
r["results"][0],
|
|
|
|
|
{"user_id": CHARLIE, "display_name": "Charlie", "avatar_url": None},
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
@override_config({"user_directory": {"search_all_users": True}})
|
|
|
|
|
def test_search_user_dir_unicode_case_insensitivity(self) -> None:
|
|
|
|
|
"""Tests that a user can look up another user by searching for their name in a
|
|
|
|
|
different case.
|
|
|
|
|
"""
|
|
|
|
|
IVAN = "@someuser:example.org"
|
|
|
|
|
self.get_success(self.store.update_profile_in_user_dir(IVAN, "Иван", None))
|
|
|
|
|
|
|
|
|
|
r = self.get_success(self.store.search_user_dir(ALICE, "иВАН", 10))
|
|
|
|
|
self.assertFalse(r["limited"])
|
|
|
|
|
self.assertEqual(1, len(r["results"]))
|
|
|
|
|
self.assertDictEqual(
|
|
|
|
|
r["results"][0],
|
|
|
|
|
{"user_id": IVAN, "display_name": "Иван", "avatar_url": None},
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
@override_config({"user_directory": {"search_all_users": True}})
|
|
|
|
|
def test_search_user_dir_dotted_dotless_i_case_insensitivity(self) -> None:
|
|
|
|
|
"""Tests that a user can look up another user by searching for their name in a
|
|
|
|
|
different case, when their name contains dotted or dotless "i"s.
|
|
|
|
|
|
|
|
|
|
Some languages have dotted and dotless versions of "i", which are considered to
|
|
|
|
|
be different letters: i <-> İ, ı <-> I. To make things difficult, they reuse the
|
|
|
|
|
ASCII "i" and "I" code points, despite having different lowercase / uppercase
|
|
|
|
|
forms.
|
|
|
|
|
"""
|
|
|
|
|
USER = "@someuser:example.org"
|
|
|
|
|
|
|
|
|
|
expected_matches = [
|
|
|
|
|
# (search_term, display_name)
|
|
|
|
|
# A search for "i" should match "İ".
|
|
|
|
|
("iiiii", "İİİİİ"),
|
|
|
|
|
# A search for "I" should match "ı".
|
|
|
|
|
("IIIII", "ııııı"),
|
|
|
|
|
# A search for "ı" should match "I".
|
|
|
|
|
("ııııı", "IIIII"),
|
|
|
|
|
# A search for "İ" should match "i".
|
|
|
|
|
("İİİİİ", "iiiii"),
|
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
for search_term, display_name in expected_matches:
|
|
|
|
|
self.get_success(
|
|
|
|
|
self.store.update_profile_in_user_dir(USER, display_name, None)
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
r = self.get_success(self.store.search_user_dir(ALICE, search_term, 10))
|
|
|
|
|
self.assertFalse(r["limited"])
|
|
|
|
|
self.assertEqual(
|
|
|
|
|
1,
|
|
|
|
|
len(r["results"]),
|
|
|
|
|
f"searching for {search_term!r} did not match {display_name!r}",
|
|
|
|
|
)
|
|
|
|
|
self.assertDictEqual(
|
|
|
|
|
r["results"][0],
|
|
|
|
|
{"user_id": USER, "display_name": display_name, "avatar_url": None},
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
# We don't test for negative matches, to allow implementations that consider all
|
|
|
|
|
# the i variants to be the same.
|
|
|
|
|
|
|
|
|
|
test_search_user_dir_dotted_dotless_i_case_insensitivity.skip = "not supported" # type: ignore
|
|
|
|
|
|
|
|
|
|
@override_config({"user_directory": {"search_all_users": True}})
|
|
|
|
|
def test_search_user_dir_unicode_normalization(self) -> None:
|
|
|
|
|
"""Tests that a user can look up another user by searching for their name with
|
|
|
|
|
either composed or decomposed accents.
|
|
|
|
|
"""
|
|
|
|
|
AMELIE = "@someuser:example.org"
|
|
|
|
|
|
|
|
|
|
expected_matches = [
|
|
|
|
|
# (search_term, display_name)
|
|
|
|
|
("Ame\u0301lie", "Amélie"),
|
|
|
|
|
("Amélie", "Ame\u0301lie"),
|
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
for search_term, display_name in expected_matches:
|
|
|
|
|
self.get_success(
|
|
|
|
|
self.store.update_profile_in_user_dir(AMELIE, display_name, None)
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
r = self.get_success(self.store.search_user_dir(ALICE, search_term, 10))
|
|
|
|
|
self.assertFalse(r["limited"])
|
|
|
|
|
self.assertEqual(
|
|
|
|
|
1,
|
|
|
|
|
len(r["results"]),
|
|
|
|
|
f"searching for {search_term!r} did not match {display_name!r}",
|
|
|
|
|
)
|
|
|
|
|
self.assertDictEqual(
|
|
|
|
|
r["results"][0],
|
|
|
|
|
{"user_id": AMELIE, "display_name": display_name, "avatar_url": None},
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
@override_config({"user_directory": {"search_all_users": True}})
|
|
|
|
|
def test_search_user_dir_accent_insensitivity(self) -> None:
|
|
|
|
|
"""Tests that a user can look up another user by searching for their name
|
|
|
|
|
without any accents.
|
|
|
|
|
"""
|
|
|
|
|
AMELIE = "@someuser:example.org"
|
|
|
|
|
self.get_success(self.store.update_profile_in_user_dir(AMELIE, "Amélie", None))
|
|
|
|
|
|
|
|
|
|
r = self.get_success(self.store.search_user_dir(ALICE, "amelie", 10))
|
|
|
|
|
self.assertFalse(r["limited"])
|
|
|
|
|
self.assertEqual(1, len(r["results"]))
|
|
|
|
|
self.assertDictEqual(
|
|
|
|
|
r["results"][0],
|
|
|
|
|
{"user_id": AMELIE, "display_name": "Amélie", "avatar_url": None},
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
# It may be desirable for "é"s in search terms to not match plain "e"s and we
|
|
|
|
|
# really don't want "é"s in search terms to match "e"s with different accents.
|
|
|
|
|
# But we don't test for this to allow implementations that consider all
|
|
|
|
|
# "e"-lookalikes to be the same.
|
|
|
|
|
|
|
|
|
|
test_search_user_dir_accent_insensitivity.skip = "not supported yet" # type: ignore
|
|
|
|
|
|
2023-02-20 12:00:18 +00:00
|
|
|
|
|
|
|
|
|
class UserDirectoryStoreTestCaseWithIcu(UserDirectoryStoreTestCase):
|
|
|
|
|
use_icu = True
|
|
|
|
|
|
|
|
|
|
if not icu:
|
|
|
|
|
skip = "Requires PyICU"
|
|
|
|
|
|
2022-12-12 13:21:17 +01:00
|
|
|
|
|
|
|
|
|
class UserDirectoryICUTestCase(HomeserverTestCase):
|
|
|
|
|
if not icu:
|
|
|
|
|
skip = "Requires PyICU"
|
|
|
|
|
|
|
|
|
|
def prepare(self, reactor: MemoryReactor, clock: Clock, hs: HomeServer) -> None:
|
|
|
|
|
self.store = hs.get_datastores().main
|
|
|
|
|
self.user_dir_helper = GetUserDirectoryTables(self.store)
|
|
|
|
|
|
|
|
|
|
def test_icu_word_boundary(self) -> None:
|
|
|
|
|
"""Tests that we correctly detect word boundaries when ICU (International
|
|
|
|
|
Components for Unicode) support is available.
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
display_name = "Gáo"
|
|
|
|
|
|
|
|
|
|
# This word is not broken down correctly by Python's regular expressions,
|
|
|
|
|
# likely because á is actually a lowercase a followed by a U+0301 combining
|
|
|
|
|
# acute accent. This is specifically something that ICU support fixes.
|
|
|
|
|
matches = re.findall(r"([\w\-]+)", display_name, re.UNICODE)
|
|
|
|
|
self.assertEqual(len(matches), 2)
|
|
|
|
|
|
|
|
|
|
self.get_success(
|
|
|
|
|
self.store.update_profile_in_user_dir(ALICE, display_name, None)
|
|
|
|
|
)
|
|
|
|
|
self.get_success(self.store.add_users_in_public_rooms("!room:id", (ALICE,)))
|
|
|
|
|
|
|
|
|
|
# Check that searching for this user yields the correct result.
|
|
|
|
|
r = self.get_success(self.store.search_user_dir(BOB, display_name, 10))
|
|
|
|
|
self.assertFalse(r["limited"])
|
|
|
|
|
self.assertEqual(len(r["results"]), 1)
|
|
|
|
|
self.assertDictEqual(
|
|
|
|
|
r["results"][0],
|
|
|
|
|
{"user_id": ALICE, "display_name": display_name, "avatar_url": None},
|
|
|
|
|
)
|
2023-02-20 12:00:18 +00:00
|
|
|
|
|
|
|
|
|
def test_icu_word_boundary_punctuation(self) -> None:
|
|
|
|
|
"""
|
|
|
|
|
Tests the behaviour of punctuation with the ICU tokeniser.
|
|
|
|
|
|
|
|
|
|
Seems to depend on underlying version of ICU.
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
# Note: either tokenisation is fine, because Postgres actually splits
|
|
|
|
|
# words itself afterwards.
|
|
|
|
|
self.assertIn(
|
|
|
|
|
_parse_words_with_icu("lazy'fox jumped:over the.dog"),
|
|
|
|
|
(
|
|
|
|
|
# ICU 66 on Ubuntu 20.04
|
|
|
|
|
["lazy'fox", "jumped", "over", "the", "dog"],
|
|
|
|
|
# ICU 70 on Ubuntu 22.04
|
|
|
|
|
["lazy'fox", "jumped:over", "the.dog"],
|
2023-03-03 15:22:06 +01:00
|
|
|
|
# pyicu 2.10.2 on Alpine edge / macOS
|
|
|
|
|
["lazy'fox", "jumped", "over", "the.dog"],
|
2023-02-20 12:00:18 +00:00
|
|
|
|
),
|
|
|
|
|
)
|
|
|
|
|
|
2024-06-05 10:40:34 +01:00
|
|
|
|
self.assertEqual(_parse_words_with_icu("user-1"), ["user-1"])
|
|
|
|
|
self.assertEqual(_parse_words_with_icu("user-ab"), ["user-ab"])
|
|
|
|
|
self.assertEqual(_parse_words_with_icu("user.--1"), ["user", "-1"])
|
|
|
|
|
|
2023-02-20 12:00:18 +00:00
|
|
|
|
def test_regex_word_boundary_punctuation(self) -> None:
|
|
|
|
|
"""
|
|
|
|
|
Tests the behaviour of punctuation with the non-ICU tokeniser
|
|
|
|
|
"""
|
|
|
|
|
self.assertEqual(
|
|
|
|
|
_parse_words_with_regex("lazy'fox jumped:over the.dog"),
|
|
|
|
|
["lazy", "fox", "jumped", "over", "the", "dog"],
|
|
|
|
|
)
|