309 lines
9.3 KiB
Python
309 lines
9.3 KiB
Python
|
from http import HTTPStatus
|
||
|
|
||
|
from parameterized import parameterized_class
|
||
|
|
||
|
from twisted.test.proto_helpers 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 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,
|
||
|
)
|