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

Commit 06eb5ca

Browse files
authored
Remove special auth and redaction rules for aliases events in experimental room ver. (#7037)
1 parent 66315d8 commit 06eb5ca

File tree

8 files changed

+148
-22
lines changed

8 files changed

+148
-22
lines changed

changelog.d/7037.feature

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
Implement updated authorization rules and redaction rules for aliases events, from [MSC2261](https://github.com/matrix-org/matrix-doc/pull/2261) and [MSC2432](https://github.com/matrix-org/matrix-doc/pull/2432).

synapse/api/room_versions.py

Lines changed: 5 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -57,7 +57,7 @@ class RoomVersion(object):
5757
state_res = attr.ib() # int; one of the StateResolutionVersions
5858
enforce_key_validity = attr.ib() # bool
5959

60-
# bool: before MSC2260, anyone was allowed to send an aliases event
60+
# bool: before MSC2261/MSC2432, m.room.aliases had special auth rules and redaction rules
6161
special_case_aliases_auth = attr.ib(type=bool, default=False)
6262

6363

@@ -102,12 +102,13 @@ class RoomVersions(object):
102102
enforce_key_validity=True,
103103
special_case_aliases_auth=True,
104104
)
105-
MSC2260_DEV = RoomVersion(
106-
"org.matrix.msc2260",
105+
MSC2432_DEV = RoomVersion(
106+
"org.matrix.msc2432",
107107
RoomDisposition.UNSTABLE,
108108
EventFormatVersions.V3,
109109
StateResolutionVersions.V2,
110110
enforce_key_validity=True,
111+
special_case_aliases_auth=False,
111112
)
112113

113114

@@ -119,6 +120,6 @@ class RoomVersions(object):
119120
RoomVersions.V3,
120121
RoomVersions.V4,
121122
RoomVersions.V5,
122-
RoomVersions.MSC2260_DEV,
123+
RoomVersions.MSC2432_DEV,
123124
)
124125
} # type: Dict[str, RoomVersion]

synapse/crypto/event_signing.py

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -140,7 +140,7 @@ def compute_event_signature(
140140
Returns:
141141
a dictionary in the same format of an event's signatures field.
142142
"""
143-
redact_json = prune_event_dict(event_dict)
143+
redact_json = prune_event_dict(room_version, event_dict)
144144
redact_json.pop("age_ts", None)
145145
redact_json.pop("unsigned", None)
146146
if logger.isEnabledFor(logging.DEBUG):

synapse/event_auth.py

Lines changed: 3 additions & 5 deletions
Original file line numberDiff line numberDiff line change
@@ -137,7 +137,7 @@ def check(
137137
raise AuthError(403, "This room has been marked as unfederatable.")
138138

139139
# 4. If type is m.room.aliases
140-
if event.type == EventTypes.Aliases:
140+
if event.type == EventTypes.Aliases and room_version_obj.special_case_aliases_auth:
141141
# 4a. If event has no state_key, reject
142142
if not event.is_state():
143143
raise AuthError(403, "Alias event must be a state event")
@@ -152,10 +152,8 @@ def check(
152152
)
153153

154154
# 4c. Otherwise, allow.
155-
# This is removed by https://github.com/matrix-org/matrix-doc/pull/2260
156-
if room_version_obj.special_case_aliases_auth:
157-
logger.debug("Allowing! %s", event)
158-
return
155+
logger.debug("Allowing! %s", event)
156+
return
159157

160158
if logger.isEnabledFor(logging.DEBUG):
161159
logger.debug("Auth events: %s", [a.event_id for a in auth_events.values()])

synapse/events/utils.py

Lines changed: 5 additions & 7 deletions
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,7 @@
2323
from twisted.internet import defer
2424

2525
from synapse.api.constants import EventTypes, RelationTypes
26+
from synapse.api.room_versions import RoomVersion
2627
from synapse.util.async_helpers import yieldable_gather_results
2728

2829
from . import EventBase
@@ -43,7 +44,7 @@ def prune_event(event: EventBase) -> EventBase:
4344
the user has specified, but we do want to keep necessary information like
4445
type, state_key etc.
4546
"""
46-
pruned_event_dict = prune_event_dict(event.get_dict())
47+
pruned_event_dict = prune_event_dict(event.room_version, event.get_dict())
4748

4849
from . import make_event_from_dict
4950

@@ -57,15 +58,12 @@ def prune_event(event: EventBase) -> EventBase:
5758
return pruned_event
5859

5960

60-
def prune_event_dict(event_dict):
61+
def prune_event_dict(room_version: RoomVersion, event_dict: dict) -> dict:
6162
"""Redacts the event_dict in the same way as `prune_event`, except it
6263
operates on dicts rather than event objects
6364
64-
Args:
65-
event_dict (dict)
66-
6765
Returns:
68-
dict: A copy of the pruned event dict
66+
A copy of the pruned event dict
6967
"""
7068

7169
allowed_keys = [
@@ -112,7 +110,7 @@ def add_fields(*fields):
112110
"kick",
113111
"redact",
114112
)
115-
elif event_type == EventTypes.Aliases:
113+
elif event_type == EventTypes.Aliases and room_version.special_case_aliases_auth:
116114
add_fields("aliases")
117115
elif event_type == EventTypes.RoomHistoryVisibility:
118116
add_fields("history_visibility")

synapse/storage/data_stores/main/events.py

Lines changed: 8 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -1168,7 +1168,11 @@ async def _censor_redactions(self):
11681168
and original_event.internal_metadata.is_redacted()
11691169
):
11701170
# Redaction was allowed
1171-
pruned_json = encode_json(prune_event_dict(original_event.get_dict()))
1171+
pruned_json = encode_json(
1172+
prune_event_dict(
1173+
original_event.room_version, original_event.get_dict()
1174+
)
1175+
)
11721176
else:
11731177
# Redaction wasn't allowed
11741178
pruned_json = None
@@ -1929,7 +1933,9 @@ def delete_expired_event_txn(txn):
19291933
return
19301934

19311935
# Prune the event's dict then convert it to JSON.
1932-
pruned_json = encode_json(prune_event_dict(event.get_dict()))
1936+
pruned_json = encode_json(
1937+
prune_event_dict(event.room_version, event.get_dict())
1938+
)
19331939

19341940
# Update the event_json table to replace the event's JSON with the pruned
19351941
# JSON.

tests/events/test_utils.py

Lines changed: 33 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@
1313
# See the License for the specific language governing permissions and
1414
# limitations under the License.
1515

16+
from synapse.api.room_versions import RoomVersions
1617
from synapse.events import make_event_from_dict
1718
from synapse.events.utils import (
1819
copy_power_levels_contents,
@@ -36,9 +37,9 @@ class PruneEventTestCase(unittest.TestCase):
3637
""" Asserts that a new event constructed with `evdict` will look like
3738
`matchdict` when it is redacted. """
3839

39-
def run_test(self, evdict, matchdict):
40+
def run_test(self, evdict, matchdict, **kwargs):
4041
self.assertEquals(
41-
prune_event(make_event_from_dict(evdict)).get_dict(), matchdict
42+
prune_event(make_event_from_dict(evdict, **kwargs)).get_dict(), matchdict
4243
)
4344

4445
def test_minimal(self):
@@ -128,6 +129,36 @@ def test_content(self):
128129
},
129130
)
130131

132+
def test_alias_event(self):
133+
"""Alias events have special behavior up through room version 6."""
134+
self.run_test(
135+
{
136+
"type": "m.room.aliases",
137+
"event_id": "$test:domain",
138+
"content": {"aliases": ["test"]},
139+
},
140+
{
141+
"type": "m.room.aliases",
142+
"event_id": "$test:domain",
143+
"content": {"aliases": ["test"]},
144+
"signatures": {},
145+
"unsigned": {},
146+
},
147+
)
148+
149+
def test_msc2432_alias_event(self):
150+
"""After MSC2432, alias events have no special behavior."""
151+
self.run_test(
152+
{"type": "m.room.aliases", "content": {"aliases": ["test"]}},
153+
{
154+
"type": "m.room.aliases",
155+
"content": {},
156+
"signatures": {},
157+
"unsigned": {},
158+
},
159+
room_version=RoomVersions.MSC2432_DEV,
160+
)
161+
131162

132163
class SerializeEventTestCase(unittest.TestCase):
133164
def serialize(self, ev, fields):

tests/test_event_auth.py

Lines changed: 92 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@
1919
from synapse.api.errors import AuthError
2020
from synapse.api.room_versions import RoomVersions
2121
from synapse.events import make_event_from_dict
22+
from synapse.types import get_domain_from_id
2223

2324

2425
class EventAuthTestCase(unittest.TestCase):
@@ -51,7 +52,7 @@ def test_random_users_cannot_send_state_before_first_pl(self):
5152
_random_state_event(joiner),
5253
auth_events,
5354
do_sig_check=False,
54-
),
55+
)
5556

5657
def test_state_default_level(self):
5758
"""
@@ -87,6 +88,83 @@ def test_state_default_level(self):
8788
RoomVersions.V1, _random_state_event(king), auth_events, do_sig_check=False,
8889
)
8990

91+
def test_alias_event(self):
92+
"""Alias events have special behavior up through room version 6."""
93+
creator = "@creator:example.com"
94+
other = "@other:example.com"
95+
auth_events = {
96+
("m.room.create", ""): _create_event(creator),
97+
("m.room.member", creator): _join_event(creator),
98+
}
99+
100+
# creator should be able to send aliases
101+
event_auth.check(
102+
RoomVersions.V1, _alias_event(creator), auth_events, do_sig_check=False,
103+
)
104+
105+
# Reject an event with no state key.
106+
with self.assertRaises(AuthError):
107+
event_auth.check(
108+
RoomVersions.V1,
109+
_alias_event(creator, state_key=""),
110+
auth_events,
111+
do_sig_check=False,
112+
)
113+
114+
# If the domain of the sender does not match the state key, reject.
115+
with self.assertRaises(AuthError):
116+
event_auth.check(
117+
RoomVersions.V1,
118+
_alias_event(creator, state_key="test.com"),
119+
auth_events,
120+
do_sig_check=False,
121+
)
122+
123+
# Note that the member does *not* need to be in the room.
124+
event_auth.check(
125+
RoomVersions.V1, _alias_event(other), auth_events, do_sig_check=False,
126+
)
127+
128+
def test_msc2432_alias_event(self):
129+
"""After MSC2432, alias events have no special behavior."""
130+
creator = "@creator:example.com"
131+
other = "@other:example.com"
132+
auth_events = {
133+
("m.room.create", ""): _create_event(creator),
134+
("m.room.member", creator): _join_event(creator),
135+
}
136+
137+
# creator should be able to send aliases
138+
event_auth.check(
139+
RoomVersions.MSC2432_DEV,
140+
_alias_event(creator),
141+
auth_events,
142+
do_sig_check=False,
143+
)
144+
145+
# No particular checks are done on the state key.
146+
event_auth.check(
147+
RoomVersions.MSC2432_DEV,
148+
_alias_event(creator, state_key=""),
149+
auth_events,
150+
do_sig_check=False,
151+
)
152+
event_auth.check(
153+
RoomVersions.MSC2432_DEV,
154+
_alias_event(creator, state_key="test.com"),
155+
auth_events,
156+
do_sig_check=False,
157+
)
158+
159+
# Per standard auth rules, the member must be in the room.
160+
with self.assertRaises(AuthError):
161+
event_auth.check(
162+
RoomVersions.MSC2432_DEV,
163+
_alias_event(other),
164+
auth_events,
165+
do_sig_check=False,
166+
)
167+
90168

91169
# helpers for making events
92170

@@ -131,6 +209,19 @@ def _power_levels_event(sender, content):
131209
)
132210

133211

212+
def _alias_event(sender, **kwargs):
213+
data = {
214+
"room_id": TEST_ROOM_ID,
215+
"event_id": _get_event_id(),
216+
"type": "m.room.aliases",
217+
"sender": sender,
218+
"state_key": get_domain_from_id(sender),
219+
"content": {"aliases": []},
220+
}
221+
data.update(**kwargs)
222+
return make_event_from_dict(data)
223+
224+
134225
def _random_state_event(sender):
135226
return make_event_from_dict(
136227
{

0 commit comments

Comments
 (0)