2016-01-05 19:12:37 +01:00
|
|
|
# Copyright 2014 - 2016 OpenMarket Ltd
|
2014-08-12 16:10:52 +02:00
|
|
|
#
|
|
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
# you may not use this file except in compliance with the License.
|
|
|
|
# You may obtain a copy of the License at
|
|
|
|
#
|
|
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
#
|
|
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
# See the License for the specific language governing permissions and
|
|
|
|
# limitations under the License.
|
|
|
|
|
2016-07-26 17:46:53 +02:00
|
|
|
import logging
|
2020-10-21 12:44:31 +02:00
|
|
|
from typing import TYPE_CHECKING, Optional
|
2016-07-26 17:46:53 +02:00
|
|
|
|
2020-06-05 11:47:20 +02:00
|
|
|
from synapse.api.ratelimiting import Ratelimiter
|
2014-11-05 12:07:54 +01:00
|
|
|
|
2020-10-21 12:44:31 +02:00
|
|
|
if TYPE_CHECKING:
|
2021-03-23 12:12:48 +01:00
|
|
|
from synapse.server import HomeServer
|
2020-10-21 12:44:31 +02:00
|
|
|
|
2014-11-05 12:07:54 +01:00
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
2020-09-04 12:54:56 +02:00
|
|
|
class BaseHandler:
|
2015-11-13 11:31:15 +01:00
|
|
|
"""
|
|
|
|
Common base class for the event handlers.
|
2020-12-08 15:03:38 +01:00
|
|
|
|
|
|
|
Deprecated: new code should not use this. Instead, Handler classes should define the
|
|
|
|
fields they actually need. The utility methods should either be factored out to
|
|
|
|
standalone helper functions, or to different Handler classes.
|
2015-11-13 11:31:15 +01:00
|
|
|
"""
|
2014-08-12 16:10:52 +02:00
|
|
|
|
2020-10-21 12:44:31 +02:00
|
|
|
def __init__(self, hs: "HomeServer"):
|
2021-07-16 19:22:36 +02:00
|
|
|
self.store = hs.get_datastore()
|
2014-08-12 16:10:52 +02:00
|
|
|
self.auth = hs.get_auth()
|
|
|
|
self.notifier = hs.get_notifier()
|
2021-07-16 19:22:36 +02:00
|
|
|
self.state_handler = hs.get_state_handler()
|
2014-08-21 15:38:22 +02:00
|
|
|
self.distributor = hs.get_distributor()
|
2014-09-02 18:57:04 +02:00
|
|
|
self.clock = hs.get_clock()
|
2014-08-12 16:10:52 +02:00
|
|
|
self.hs = hs
|
2014-08-26 19:49:51 +02:00
|
|
|
|
2020-06-05 11:47:20 +02:00
|
|
|
# The rate_hz and burst_count are overridden on a per-user basis
|
|
|
|
self.request_ratelimiter = Ratelimiter(
|
2021-03-30 13:06:09 +02:00
|
|
|
store=self.store, clock=self.clock, rate_hz=0, burst_count=0
|
2020-06-05 11:47:20 +02:00
|
|
|
)
|
|
|
|
self._rc_message = self.hs.config.rc_message
|
|
|
|
|
|
|
|
# Check whether ratelimiting room admin message redaction is enabled
|
|
|
|
# by the presence of rate limits in the config
|
|
|
|
if self.hs.config.rc_admin_redaction:
|
2021-07-16 19:22:36 +02:00
|
|
|
self.admin_redaction_ratelimiter: Optional[Ratelimiter] = Ratelimiter(
|
2021-03-30 13:06:09 +02:00
|
|
|
store=self.store,
|
2020-06-05 11:47:20 +02:00
|
|
|
clock=self.clock,
|
|
|
|
rate_hz=self.hs.config.rc_admin_redaction.per_second,
|
|
|
|
burst_count=self.hs.config.rc_admin_redaction.burst_count,
|
2021-07-16 19:22:36 +02:00
|
|
|
)
|
2020-06-05 11:47:20 +02:00
|
|
|
else:
|
|
|
|
self.admin_redaction_ratelimiter = None
|
|
|
|
|
2014-11-03 12:33:28 +01:00
|
|
|
self.server_name = hs.hostname
|
|
|
|
|
2014-12-04 16:50:01 +01:00
|
|
|
self.event_builder_factory = hs.get_event_builder_factory()
|
|
|
|
|
2020-07-17 13:08:30 +02:00
|
|
|
async def ratelimit(self, requester, update=True, is_admin_redaction=False):
|
2017-05-10 12:05:43 +02:00
|
|
|
"""Ratelimits requests.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
requester (Requester)
|
|
|
|
update (bool): Whether to record that a request is being processed.
|
|
|
|
Set to False when doing multiple checks for one request (e.g.
|
|
|
|
to check up front if we would reject the request), and set to
|
|
|
|
True for the last call for a given request.
|
2019-09-11 11:46:38 +02:00
|
|
|
is_admin_redaction (bool): Whether this is a room admin/moderator
|
|
|
|
redacting an event. If so then we may apply different
|
|
|
|
ratelimits depending on config.
|
2017-05-10 12:05:43 +02:00
|
|
|
|
|
|
|
Raises:
|
|
|
|
LimitExceededError if the request should be ratelimited
|
|
|
|
"""
|
2016-10-04 21:53:35 +02:00
|
|
|
user_id = requester.user.to_string()
|
|
|
|
|
2016-10-18 18:04:09 +02:00
|
|
|
# The AS user itself is never rate limited.
|
2016-10-04 21:53:35 +02:00
|
|
|
app_service = self.store.get_app_service_by_user_id(user_id)
|
|
|
|
if app_service is not None:
|
|
|
|
return # do not ratelimit app service senders
|
|
|
|
|
2020-06-05 11:47:20 +02:00
|
|
|
messages_per_second = self._rc_message.per_second
|
|
|
|
burst_count = self._rc_message.burst_count
|
|
|
|
|
2017-05-10 12:05:43 +02:00
|
|
|
# Check if there is a per user override in the DB.
|
2020-07-17 13:08:30 +02:00
|
|
|
override = await self.store.get_ratelimit_for_user(user_id)
|
2017-05-10 12:05:43 +02:00
|
|
|
if override:
|
2020-06-05 11:47:20 +02:00
|
|
|
# If overridden with a null Hz then ratelimiting has been entirely
|
2017-05-10 12:05:43 +02:00
|
|
|
# disabled for the user
|
|
|
|
if not override.messages_per_second:
|
|
|
|
return
|
|
|
|
|
|
|
|
messages_per_second = override.messages_per_second
|
|
|
|
burst_count = override.burst_count
|
2020-06-05 11:47:20 +02:00
|
|
|
|
|
|
|
if is_admin_redaction and self.admin_redaction_ratelimiter:
|
|
|
|
# If we have separate config for admin redactions, use a separate
|
|
|
|
# ratelimiter as to not have user_ids clash
|
2021-03-30 13:06:09 +02:00
|
|
|
await self.admin_redaction_ratelimiter.ratelimit(requester, update=update)
|
2017-05-10 12:05:43 +02:00
|
|
|
else:
|
2020-06-05 11:47:20 +02:00
|
|
|
# Override rate and burst count per-user
|
2021-03-30 13:06:09 +02:00
|
|
|
await self.request_ratelimiter.ratelimit(
|
|
|
|
requester,
|
2019-09-11 11:46:38 +02:00
|
|
|
rate_hz=messages_per_second,
|
|
|
|
burst_count=burst_count,
|
|
|
|
update=update,
|
|
|
|
)
|