Introduce `Clock.call_when_running(...)` to wrap startup code in a logcontext, ensuring we can identify which server generated the logs. Background: > Ideally, nothing from the Synapse homeserver would be logged against the `sentinel` > logcontext as we want to know which server the logs came from. In practice, this is not > always the case yet especially outside of request handling. > > Global things outside of Synapse (e.g. Twisted reactor code) should run in the > `sentinel` logcontext. It's only when it calls into application code that a logcontext > gets activated. This means the reactor should be started in the `sentinel` logcontext, > and any time an awaitable yields control back to the reactor, it should reset the > logcontext to be the `sentinel` logcontext. This is important to avoid leaking the > current logcontext to the reactor (which would then get picked up and associated with > the next thing the reactor does). > > *-- `docs/log_contexts.md` Also adds a lint to prefer `Clock.call_when_running(...)` over `reactor.callWhenRunning(...)` Part of https://github.com/element-hq/synapse/issues/18905
309 lines
9.3 KiB
Python
309 lines
9.3 KiB
Python
from http import HTTPStatus
|
|
|
|
from parameterized import parameterized_class
|
|
|
|
from twisted.internet.testing import MemoryReactor
|
|
|
|
from synapse.api.errors import Codes
|
|
from synapse.api.room_versions import KNOWN_ROOM_VERSIONS, RoomVersions
|
|
from synapse.rest import admin
|
|
from synapse.rest.client import login, room
|
|
from synapse.server import HomeServer
|
|
from synapse.types import JsonDict
|
|
from synapse.util.clock import Clock
|
|
|
|
from tests.unittest import HomeserverTestCase
|
|
|
|
_STATE_EVENT_TEST_TYPE = "com.example.test"
|
|
|
|
# To stress-test parsing, include separator & sigil characters
|
|
_STATE_KEY_SUFFIX = "_state_key_suffix:!@#$123"
|
|
|
|
|
|
class OwnedStateBase(HomeserverTestCase):
|
|
servlets = [
|
|
admin.register_servlets,
|
|
room.register_servlets,
|
|
login.register_servlets,
|
|
]
|
|
|
|
def prepare(self, reactor: MemoryReactor, clock: Clock, hs: HomeServer) -> None:
|
|
self.creator_user_id = self.register_user("creator", "pass")
|
|
self.creator_access_token = self.login("creator", "pass")
|
|
self.user1_user_id = self.register_user("user1", "pass")
|
|
self.user1_access_token = self.login("user1", "pass")
|
|
|
|
self.room_id = self.helper.create_room_as(
|
|
self.creator_user_id,
|
|
tok=self.creator_access_token,
|
|
is_public=True,
|
|
extra_content={
|
|
"power_level_content_override": {
|
|
"events": {
|
|
_STATE_EVENT_TEST_TYPE: 0,
|
|
},
|
|
},
|
|
},
|
|
)
|
|
|
|
self.helper.join(
|
|
room=self.room_id, user=self.user1_user_id, tok=self.user1_access_token
|
|
)
|
|
|
|
|
|
class WithoutOwnedStateTestCase(OwnedStateBase):
|
|
def default_config(self) -> JsonDict:
|
|
config = super().default_config()
|
|
config["default_room_version"] = RoomVersions.V10.identifier
|
|
return config
|
|
|
|
def test_user_can_set_state_with_own_userid_key(self) -> None:
|
|
self.helper.send_state(
|
|
self.room_id,
|
|
_STATE_EVENT_TEST_TYPE,
|
|
{},
|
|
state_key=f"{self.user1_user_id}",
|
|
tok=self.user1_access_token,
|
|
expect_code=HTTPStatus.OK,
|
|
)
|
|
|
|
def test_room_creator_cannot_set_state_with_own_suffixed_key(self) -> None:
|
|
body = self.helper.send_state(
|
|
self.room_id,
|
|
_STATE_EVENT_TEST_TYPE,
|
|
{},
|
|
state_key=f"{self.creator_user_id}{_STATE_KEY_SUFFIX}",
|
|
tok=self.creator_access_token,
|
|
expect_code=HTTPStatus.FORBIDDEN,
|
|
)
|
|
|
|
self.assertEqual(
|
|
body["errcode"],
|
|
Codes.FORBIDDEN,
|
|
body,
|
|
)
|
|
|
|
def test_room_creator_cannot_set_state_with_other_userid_key(self) -> None:
|
|
body = self.helper.send_state(
|
|
self.room_id,
|
|
_STATE_EVENT_TEST_TYPE,
|
|
{},
|
|
state_key=f"{self.user1_user_id}",
|
|
tok=self.creator_access_token,
|
|
expect_code=HTTPStatus.FORBIDDEN,
|
|
)
|
|
|
|
self.assertEqual(
|
|
body["errcode"],
|
|
Codes.FORBIDDEN,
|
|
body,
|
|
)
|
|
|
|
def test_room_creator_cannot_set_state_with_other_suffixed_key(self) -> None:
|
|
body = self.helper.send_state(
|
|
self.room_id,
|
|
_STATE_EVENT_TEST_TYPE,
|
|
{},
|
|
state_key=f"{self.user1_user_id}{_STATE_KEY_SUFFIX}",
|
|
tok=self.creator_access_token,
|
|
expect_code=HTTPStatus.FORBIDDEN,
|
|
)
|
|
|
|
self.assertEqual(
|
|
body["errcode"],
|
|
Codes.FORBIDDEN,
|
|
body,
|
|
)
|
|
|
|
def test_room_creator_cannot_set_state_with_nonmember_userid_key(self) -> None:
|
|
body = self.helper.send_state(
|
|
self.room_id,
|
|
_STATE_EVENT_TEST_TYPE,
|
|
{},
|
|
state_key="@notinroom:hs2",
|
|
tok=self.creator_access_token,
|
|
expect_code=HTTPStatus.FORBIDDEN,
|
|
)
|
|
|
|
self.assertEqual(
|
|
body["errcode"],
|
|
Codes.FORBIDDEN,
|
|
body,
|
|
)
|
|
|
|
def test_room_creator_cannot_set_state_with_malformed_userid_key(self) -> None:
|
|
body = self.helper.send_state(
|
|
self.room_id,
|
|
_STATE_EVENT_TEST_TYPE,
|
|
{},
|
|
state_key="@oops",
|
|
tok=self.creator_access_token,
|
|
expect_code=HTTPStatus.FORBIDDEN,
|
|
)
|
|
|
|
self.assertEqual(
|
|
body["errcode"],
|
|
Codes.FORBIDDEN,
|
|
body,
|
|
)
|
|
|
|
|
|
@parameterized_class(
|
|
("room_version",),
|
|
[(i,) for i, v in KNOWN_ROOM_VERSIONS.items() if v.msc3757_enabled],
|
|
)
|
|
class MSC3757OwnedStateTestCase(OwnedStateBase):
|
|
room_version: str
|
|
|
|
def default_config(self) -> JsonDict:
|
|
config = super().default_config()
|
|
config["default_room_version"] = self.room_version
|
|
return config
|
|
|
|
def prepare(self, reactor: MemoryReactor, clock: Clock, hs: HomeServer) -> None:
|
|
super().prepare(reactor, clock, hs)
|
|
|
|
self.user2_user_id = self.register_user("user2", "pass")
|
|
self.user2_access_token = self.login("user2", "pass")
|
|
|
|
self.helper.join(
|
|
room=self.room_id, user=self.user2_user_id, tok=self.user2_access_token
|
|
)
|
|
|
|
def test_user_can_set_state_with_own_suffixed_key(self) -> None:
|
|
self.helper.send_state(
|
|
self.room_id,
|
|
_STATE_EVENT_TEST_TYPE,
|
|
{},
|
|
state_key=f"{self.user1_user_id}{_STATE_KEY_SUFFIX}",
|
|
tok=self.user1_access_token,
|
|
expect_code=HTTPStatus.OK,
|
|
)
|
|
|
|
def test_room_creator_can_set_state_with_other_userid_key(self) -> None:
|
|
self.helper.send_state(
|
|
self.room_id,
|
|
_STATE_EVENT_TEST_TYPE,
|
|
{},
|
|
state_key=f"{self.user1_user_id}",
|
|
tok=self.creator_access_token,
|
|
expect_code=HTTPStatus.OK,
|
|
)
|
|
|
|
def test_room_creator_can_set_state_with_other_suffixed_key(self) -> None:
|
|
self.helper.send_state(
|
|
self.room_id,
|
|
_STATE_EVENT_TEST_TYPE,
|
|
{},
|
|
state_key=f"{self.user1_user_id}{_STATE_KEY_SUFFIX}",
|
|
tok=self.creator_access_token,
|
|
expect_code=HTTPStatus.OK,
|
|
)
|
|
|
|
def test_user_cannot_set_state_with_other_userid_key(self) -> None:
|
|
body = self.helper.send_state(
|
|
self.room_id,
|
|
_STATE_EVENT_TEST_TYPE,
|
|
{},
|
|
state_key=f"{self.user2_user_id}",
|
|
tok=self.user1_access_token,
|
|
expect_code=HTTPStatus.FORBIDDEN,
|
|
)
|
|
|
|
self.assertEqual(
|
|
body["errcode"],
|
|
Codes.FORBIDDEN,
|
|
body,
|
|
)
|
|
|
|
def test_user_cannot_set_state_with_other_suffixed_key(self) -> None:
|
|
body = self.helper.send_state(
|
|
self.room_id,
|
|
_STATE_EVENT_TEST_TYPE,
|
|
{},
|
|
state_key=f"{self.user2_user_id}{_STATE_KEY_SUFFIX}",
|
|
tok=self.user1_access_token,
|
|
expect_code=HTTPStatus.FORBIDDEN,
|
|
)
|
|
|
|
self.assertEqual(
|
|
body["errcode"],
|
|
Codes.FORBIDDEN,
|
|
body,
|
|
)
|
|
|
|
def test_user_cannot_set_state_with_unseparated_suffixed_key(self) -> None:
|
|
body = self.helper.send_state(
|
|
self.room_id,
|
|
_STATE_EVENT_TEST_TYPE,
|
|
{},
|
|
state_key=f"{self.user1_user_id}{_STATE_KEY_SUFFIX[1:]}",
|
|
tok=self.user1_access_token,
|
|
expect_code=HTTPStatus.FORBIDDEN,
|
|
)
|
|
|
|
self.assertEqual(
|
|
body["errcode"],
|
|
Codes.FORBIDDEN,
|
|
body,
|
|
)
|
|
|
|
def test_user_cannot_set_state_with_misplaced_userid_in_key(self) -> None:
|
|
body = self.helper.send_state(
|
|
self.room_id,
|
|
_STATE_EVENT_TEST_TYPE,
|
|
{},
|
|
# Still put @ at start of state key, because without it, there is no write protection at all
|
|
state_key=f"@prefix_{self.user1_user_id}{_STATE_KEY_SUFFIX}",
|
|
tok=self.user1_access_token,
|
|
expect_code=HTTPStatus.FORBIDDEN,
|
|
)
|
|
|
|
self.assertEqual(
|
|
body["errcode"],
|
|
Codes.FORBIDDEN,
|
|
body,
|
|
)
|
|
|
|
def test_room_creator_can_set_state_with_nonmember_userid_key(self) -> None:
|
|
self.helper.send_state(
|
|
self.room_id,
|
|
_STATE_EVENT_TEST_TYPE,
|
|
{},
|
|
state_key="@notinroom:hs2",
|
|
tok=self.creator_access_token,
|
|
expect_code=HTTPStatus.OK,
|
|
)
|
|
|
|
def test_room_creator_cannot_set_state_with_malformed_userid_key(self) -> None:
|
|
body = self.helper.send_state(
|
|
self.room_id,
|
|
_STATE_EVENT_TEST_TYPE,
|
|
{},
|
|
state_key="@oops",
|
|
tok=self.creator_access_token,
|
|
expect_code=HTTPStatus.BAD_REQUEST,
|
|
)
|
|
|
|
self.assertEqual(
|
|
body["errcode"],
|
|
Codes.BAD_JSON,
|
|
body,
|
|
)
|
|
|
|
def test_room_creator_cannot_set_state_with_improperly_suffixed_key(self) -> None:
|
|
body = self.helper.send_state(
|
|
self.room_id,
|
|
_STATE_EVENT_TEST_TYPE,
|
|
{},
|
|
state_key=f"{self.creator_user_id}@{_STATE_KEY_SUFFIX[1:]}",
|
|
tok=self.creator_access_token,
|
|
expect_code=HTTPStatus.BAD_REQUEST,
|
|
)
|
|
|
|
self.assertEqual(
|
|
body["errcode"],
|
|
Codes.BAD_JSON,
|
|
body,
|
|
)
|