2014-12-03 11:07:21 -05: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 06:26:48 -05:00
|
|
|
# Copyright 2014-2016 OpenMarket Ltd
|
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]
|
2014-12-03 11:07:21 -05:00
|
|
|
#
|
|
|
|
#
|
2021-08-26 12:07:58 -04:00
|
|
|
import collections.abc
|
2023-09-25 11:19:08 -04:00
|
|
|
from typing import TYPE_CHECKING, List, Type, Union, cast
|
2020-11-03 07:13:48 -05:00
|
|
|
|
2021-08-26 12:07:58 -04:00
|
|
|
import jsonschema
|
2023-09-25 11:19:08 -04:00
|
|
|
|
|
|
|
from synapse._pydantic_compat import HAS_PYDANTIC_V2
|
|
|
|
|
|
|
|
if TYPE_CHECKING or HAS_PYDANTIC_V2:
|
|
|
|
from pydantic.v1 import Field, StrictBool, StrictStr
|
|
|
|
else:
|
|
|
|
from pydantic import Field, StrictBool, StrictStr
|
2021-08-26 12:07:58 -04:00
|
|
|
|
2023-03-24 08:31:14 -04:00
|
|
|
from synapse.api.constants import (
|
|
|
|
MAX_ALIAS_LENGTH,
|
|
|
|
EventContentFields,
|
|
|
|
EventTypes,
|
|
|
|
Membership,
|
|
|
|
)
|
2019-05-08 12:01:30 -04:00
|
|
|
from synapse.api.errors import Codes, SynapseError
|
2019-04-01 05:24:38 -04:00
|
|
|
from synapse.api.room_versions import EventFormatVersions
|
2020-11-03 07:13:48 -05:00
|
|
|
from synapse.config.homeserver import HomeServerConfig
|
|
|
|
from synapse.events import EventBase
|
|
|
|
from synapse.events.builder import EventBuilder
|
2021-08-26 12:07:58 -04:00
|
|
|
from synapse.events.utils import (
|
|
|
|
CANONICALJSON_MAX_INT,
|
|
|
|
CANONICALJSON_MIN_INT,
|
|
|
|
validate_canonicaljson,
|
|
|
|
)
|
2023-03-24 08:31:14 -04:00
|
|
|
from synapse.http.servlet import validate_json_object
|
2023-09-26 11:57:50 -04:00
|
|
|
from synapse.storage.controllers.state import server_acl_evaluator_from_event
|
2023-09-13 07:57:19 -04:00
|
|
|
from synapse.types import EventID, JsonDict, RoomID, StrCollection, UserID
|
2024-06-10 16:03:50 -04:00
|
|
|
from synapse.types.rest import RequestBodyModel
|
2018-07-09 02:09:20 -04:00
|
|
|
|
2014-12-03 11:07:21 -05:00
|
|
|
|
2020-09-04 06:54:56 -04:00
|
|
|
class EventValidator:
|
2021-10-13 07:24:07 -04:00
|
|
|
def validate_new(self, event: EventBase, config: HomeServerConfig) -> None:
|
2019-01-29 05:34:49 -05:00
|
|
|
"""Validates the event has roughly the right format
|
|
|
|
|
2022-06-09 10:51:34 -04:00
|
|
|
Suitable for checking a locally-created event. It has stricter checks than
|
|
|
|
is appropriate for an event received over federation (for which, see
|
|
|
|
event_auth.validate_event_for_room_version)
|
|
|
|
|
2019-01-29 05:34:49 -05:00
|
|
|
Args:
|
2020-11-03 07:13:48 -05:00
|
|
|
event: The event to validate.
|
|
|
|
config: The homeserver's configuration.
|
2019-01-29 05:34:49 -05:00
|
|
|
"""
|
|
|
|
self.validate_builder(event)
|
2014-12-03 11:07:21 -05:00
|
|
|
|
2022-09-07 06:08:20 -04:00
|
|
|
if event.format_version == EventFormatVersions.ROOM_V1_V2:
|
2019-01-29 12:23:47 -05:00
|
|
|
EventID.from_string(event.event_id)
|
2014-12-03 11:07:21 -05:00
|
|
|
|
2014-12-10 12:59:47 -05:00
|
|
|
required = [
|
2019-01-28 12:00:14 -05:00
|
|
|
"auth_events",
|
2014-12-10 12:59:47 -05:00
|
|
|
"content",
|
2019-01-28 12:00:14 -05:00
|
|
|
"hashes",
|
2014-12-10 12:59:47 -05:00
|
|
|
"origin",
|
2019-01-28 12:00:14 -05:00
|
|
|
"prev_events",
|
2014-12-10 12:59:47 -05:00
|
|
|
"sender",
|
|
|
|
"type",
|
|
|
|
]
|
|
|
|
|
|
|
|
for k in required:
|
2021-11-02 09:55:52 -04:00
|
|
|
if k not in event:
|
2014-12-10 12:59:47 -05:00
|
|
|
raise SynapseError(400, "Event does not have key %s" % (k,))
|
2014-12-03 11:07:21 -05:00
|
|
|
|
|
|
|
# Check that the following keys have string values
|
2019-01-29 05:34:49 -05:00
|
|
|
event_strings = ["origin"]
|
2014-12-03 11:07:21 -05:00
|
|
|
|
2019-01-29 05:34:49 -05:00
|
|
|
for s in event_strings:
|
2020-06-16 08:51:47 -04:00
|
|
|
if not isinstance(getattr(event, s), str):
|
2019-01-29 05:36:46 -05:00
|
|
|
raise SynapseError(400, "'%s' not a string type" % (s,))
|
2014-12-03 11:07:21 -05:00
|
|
|
|
2020-05-14 13:24:01 -04:00
|
|
|
# Depending on the room version, ensure the data is spec compliant JSON.
|
|
|
|
if event.room_version.strict_canonicaljson:
|
|
|
|
# Note that only the client controlled portion of the event is
|
|
|
|
# checked, since we trust the portions of the event we created.
|
|
|
|
validate_canonicaljson(event.content)
|
|
|
|
|
2019-05-08 12:01:30 -04:00
|
|
|
if event.type == EventTypes.Aliases:
|
|
|
|
if "aliases" in event.content:
|
|
|
|
for alias in event.content["aliases"]:
|
|
|
|
if len(alias) > MAX_ALIAS_LENGTH:
|
|
|
|
raise SynapseError(
|
|
|
|
400,
|
|
|
|
(
|
|
|
|
"Can't create aliases longer than"
|
|
|
|
" %d characters" % (MAX_ALIAS_LENGTH,)
|
|
|
|
),
|
|
|
|
Codes.INVALID_PARAM,
|
|
|
|
)
|
|
|
|
|
2023-03-24 08:31:14 -04:00
|
|
|
elif event.type == EventTypes.Retention:
|
2020-08-24 13:21:04 -04:00
|
|
|
self._validate_retention(event)
|
2019-11-04 12:09:22 -05:00
|
|
|
|
2023-03-24 08:31:14 -04:00
|
|
|
elif event.type == EventTypes.ServerACL:
|
2023-09-26 11:57:50 -04:00
|
|
|
server_acl_evaluator = server_acl_evaluator_from_event(event)
|
|
|
|
if not server_acl_evaluator.server_matches_acl_event(
|
|
|
|
config.server.server_name
|
|
|
|
):
|
2020-11-03 07:13:48 -05:00
|
|
|
raise SynapseError(
|
|
|
|
400, "Can't create an ACL event that denies the local server"
|
|
|
|
)
|
|
|
|
|
2023-03-24 08:31:14 -04:00
|
|
|
elif event.type == EventTypes.PowerLevels:
|
2021-08-26 12:07:58 -04:00
|
|
|
try:
|
|
|
|
jsonschema.validate(
|
|
|
|
instance=event.content,
|
|
|
|
schema=POWER_LEVELS_SCHEMA,
|
2023-03-24 08:31:14 -04:00
|
|
|
cls=POWER_LEVELS_VALIDATOR,
|
2021-08-26 12:07:58 -04:00
|
|
|
)
|
|
|
|
except jsonschema.ValidationError as e:
|
|
|
|
if e.path:
|
|
|
|
# example: "users_default": '0' is not of type 'integer'
|
2022-05-30 05:47:09 -04:00
|
|
|
# cast safety: path entries can be integers, if we fail to validate
|
2023-03-24 08:31:14 -04:00
|
|
|
# items in an array. However, the POWER_LEVELS_SCHEMA doesn't expect
|
2022-05-30 05:47:09 -04:00
|
|
|
# to see any arrays.
|
|
|
|
message = (
|
|
|
|
'"' + cast(str, e.path[-1]) + '": ' + e.message # noqa: B306
|
|
|
|
)
|
2021-08-26 12:07:58 -04:00
|
|
|
# jsonschema.ValidationError.message is a valid attribute
|
|
|
|
else:
|
|
|
|
# example: '0' is not of type 'integer'
|
|
|
|
message = e.message # noqa: B306
|
|
|
|
# jsonschema.ValidationError.message is a valid attribute
|
|
|
|
|
|
|
|
raise SynapseError(
|
|
|
|
code=400,
|
|
|
|
msg=message,
|
|
|
|
errcode=Codes.BAD_JSON,
|
|
|
|
)
|
|
|
|
|
2023-03-24 08:31:14 -04:00
|
|
|
# If the event contains a mentions key, validate it.
|
2023-06-06 04:11:07 -04:00
|
|
|
if EventContentFields.MENTIONS in event.content:
|
|
|
|
validate_json_object(event.content[EventContentFields.MENTIONS], Mentions)
|
2023-03-24 08:31:14 -04:00
|
|
|
|
2021-10-13 07:24:07 -04:00
|
|
|
def _validate_retention(self, event: EventBase) -> None:
|
2019-11-04 12:09:22 -05:00
|
|
|
"""Checks that an event that defines the retention policy for a room respects the
|
2020-08-24 13:21:04 -04:00
|
|
|
format enforced by the spec.
|
2019-11-04 12:09:22 -05:00
|
|
|
|
|
|
|
Args:
|
2020-11-03 07:13:48 -05:00
|
|
|
event: The event to validate.
|
2019-11-04 12:09:22 -05:00
|
|
|
"""
|
2020-10-14 07:00:52 -04:00
|
|
|
if not event.is_state():
|
|
|
|
raise SynapseError(code=400, msg="must be a state event")
|
|
|
|
|
2019-11-04 12:09:22 -05:00
|
|
|
min_lifetime = event.content.get("min_lifetime")
|
|
|
|
max_lifetime = event.content.get("max_lifetime")
|
|
|
|
|
|
|
|
if min_lifetime is not None:
|
2023-08-29 09:41:43 -04:00
|
|
|
if type(min_lifetime) is not int: # noqa: E721
|
2019-11-04 12:09:22 -05:00
|
|
|
raise SynapseError(
|
|
|
|
code=400,
|
|
|
|
msg="'min_lifetime' must be an integer",
|
|
|
|
errcode=Codes.BAD_JSON,
|
|
|
|
)
|
|
|
|
|
|
|
|
if max_lifetime is not None:
|
2023-08-29 09:41:43 -04:00
|
|
|
if type(max_lifetime) is not int: # noqa: E721
|
2019-11-04 12:09:22 -05:00
|
|
|
raise SynapseError(
|
|
|
|
code=400,
|
|
|
|
msg="'max_lifetime' must be an integer",
|
|
|
|
errcode=Codes.BAD_JSON,
|
|
|
|
)
|
|
|
|
|
|
|
|
if (
|
|
|
|
min_lifetime is not None
|
|
|
|
and max_lifetime is not None
|
|
|
|
and min_lifetime > max_lifetime
|
|
|
|
):
|
|
|
|
raise SynapseError(
|
|
|
|
code=400,
|
|
|
|
msg="'min_lifetime' can't be greater than 'max_lifetime",
|
|
|
|
errcode=Codes.BAD_JSON,
|
|
|
|
)
|
|
|
|
|
2021-10-13 07:24:07 -04:00
|
|
|
def validate_builder(self, event: Union[EventBase, EventBuilder]) -> None:
|
2019-01-28 12:00:14 -05:00
|
|
|
"""Validates that the builder/event has roughly the right format. Only
|
|
|
|
checks values that we expect a proto event to have, rather than all the
|
|
|
|
fields an event would have
|
|
|
|
"""
|
|
|
|
|
|
|
|
strings = ["room_id", "sender", "type"]
|
|
|
|
|
|
|
|
if hasattr(event, "state_key"):
|
|
|
|
strings.append("state_key")
|
|
|
|
|
|
|
|
for s in strings:
|
2020-06-16 08:51:47 -04:00
|
|
|
if not isinstance(getattr(event, s), str):
|
2019-01-28 12:00:14 -05:00
|
|
|
raise SynapseError(400, "Not '%s' a string type" % (s,))
|
|
|
|
|
|
|
|
RoomID.from_string(event.room_id)
|
2014-12-09 05:58:31 -05:00
|
|
|
UserID.from_string(event.sender)
|
2014-12-12 05:56:14 -05:00
|
|
|
|
|
|
|
if event.type == EventTypes.Message:
|
|
|
|
strings = ["body", "msgtype"]
|
|
|
|
|
|
|
|
self._ensure_strings(event.content, strings)
|
|
|
|
|
|
|
|
elif event.type == EventTypes.Topic:
|
|
|
|
self._ensure_strings(event.content, ["topic"])
|
2019-07-31 11:36:20 -04:00
|
|
|
self._ensure_state_event(event)
|
2014-12-12 05:56:14 -05:00
|
|
|
elif event.type == EventTypes.Name:
|
|
|
|
self._ensure_strings(event.content, ["name"])
|
2019-07-31 11:36:20 -04:00
|
|
|
self._ensure_state_event(event)
|
2019-01-28 12:00:14 -05:00
|
|
|
elif event.type == EventTypes.Member:
|
|
|
|
if "membership" not in event.content:
|
|
|
|
raise SynapseError(400, "Content has not membership key")
|
|
|
|
|
|
|
|
if event.content["membership"] not in Membership.LIST:
|
|
|
|
raise SynapseError(400, "Invalid membership key")
|
|
|
|
|
2019-07-31 11:36:20 -04:00
|
|
|
self._ensure_state_event(event)
|
2019-07-31 10:52:27 -04:00
|
|
|
elif event.type == EventTypes.Tombstone:
|
|
|
|
if "replacement_room" not in event.content:
|
|
|
|
raise SynapseError(400, "Content has no replacement_room key")
|
|
|
|
|
|
|
|
if event.content["replacement_room"] == event.room_id:
|
2019-08-01 08:14:25 -04:00
|
|
|
raise SynapseError(
|
|
|
|
400, "Tombstone cannot reference the room it was sent in"
|
|
|
|
)
|
2019-07-31 10:52:27 -04:00
|
|
|
|
2019-07-31 11:36:20 -04:00
|
|
|
self._ensure_state_event(event)
|
|
|
|
|
2023-09-13 07:57:19 -04:00
|
|
|
def _ensure_strings(self, d: JsonDict, keys: StrCollection) -> None:
|
2014-12-12 05:56:14 -05:00
|
|
|
for s in keys:
|
|
|
|
if s not in d:
|
|
|
|
raise SynapseError(400, "'%s' not in content" % (s,))
|
2020-06-16 08:51:47 -04:00
|
|
|
if not isinstance(d[s], str):
|
2019-01-29 05:36:46 -05:00
|
|
|
raise SynapseError(400, "'%s' not a string type" % (s,))
|
2019-07-31 11:36:20 -04:00
|
|
|
|
2021-10-13 07:24:07 -04:00
|
|
|
def _ensure_state_event(self, event: Union[EventBase, EventBuilder]) -> None:
|
2019-07-31 11:36:20 -04:00
|
|
|
if not event.is_state():
|
|
|
|
raise SynapseError(400, "'%s' must be state events" % (event.type,))
|
2021-08-26 12:07:58 -04:00
|
|
|
|
|
|
|
|
|
|
|
POWER_LEVELS_SCHEMA = {
|
|
|
|
"type": "object",
|
|
|
|
"properties": {
|
|
|
|
"ban": {"$ref": "#/definitions/int"},
|
|
|
|
"events": {"$ref": "#/definitions/objectOfInts"},
|
|
|
|
"events_default": {"$ref": "#/definitions/int"},
|
|
|
|
"invite": {"$ref": "#/definitions/int"},
|
|
|
|
"kick": {"$ref": "#/definitions/int"},
|
|
|
|
"notifications": {"$ref": "#/definitions/objectOfInts"},
|
|
|
|
"redact": {"$ref": "#/definitions/int"},
|
|
|
|
"state_default": {"$ref": "#/definitions/int"},
|
|
|
|
"users": {"$ref": "#/definitions/objectOfInts"},
|
|
|
|
"users_default": {"$ref": "#/definitions/int"},
|
|
|
|
},
|
|
|
|
"definitions": {
|
|
|
|
"int": {
|
|
|
|
"type": "integer",
|
|
|
|
"minimum": CANONICALJSON_MIN_INT,
|
|
|
|
"maximum": CANONICALJSON_MAX_INT,
|
|
|
|
},
|
|
|
|
"objectOfInts": {
|
|
|
|
"type": "object",
|
|
|
|
"additionalProperties": {"$ref": "#/definitions/int"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-03-24 08:31:14 -04:00
|
|
|
class Mentions(RequestBodyModel):
|
|
|
|
user_ids: List[StrictStr] = Field(default_factory=list)
|
|
|
|
room: StrictBool = False
|
|
|
|
|
|
|
|
|
2021-10-13 07:24:07 -04:00
|
|
|
# This could return something newer than Draft 7, but that's the current "latest"
|
|
|
|
# validator.
|
2023-03-24 08:31:14 -04:00
|
|
|
def _create_validator(schema: JsonDict) -> Type[jsonschema.Draft7Validator]:
|
|
|
|
validator = jsonschema.validators.validator_for(schema)
|
2021-08-26 12:07:58 -04:00
|
|
|
|
2023-03-22 13:15:34 -04:00
|
|
|
# by default jsonschema does not consider a immutabledict to be an object so
|
2021-08-26 12:07:58 -04:00
|
|
|
# we need to use a custom type checker
|
|
|
|
# https://python-jsonschema.readthedocs.io/en/stable/validate/?highlight=object#validating-with-additional-types
|
|
|
|
type_checker = validator.TYPE_CHECKER.redefine(
|
|
|
|
"object", lambda checker, thing: isinstance(thing, collections.abc.Mapping)
|
|
|
|
)
|
|
|
|
|
|
|
|
return jsonschema.validators.extend(validator, type_checker=type_checker)
|
|
|
|
|
|
|
|
|
2023-03-24 08:31:14 -04:00
|
|
|
POWER_LEVELS_VALIDATOR = _create_validator(POWER_LEVELS_SCHEMA)
|