2014-08-12 08:10:52 -06:00
|
|
|
# -*- coding: utf-8 -*-
|
2015-01-06 06:21:39 -07:00
|
|
|
# Copyright 2014, 2015 OpenMarket Ltd
|
2014-08-12 08:10:52 -06: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.
|
|
|
|
|
2014-08-12 20:14:34 -06:00
|
|
|
|
2014-08-12 08:10:52 -06:00
|
|
|
# This file provides some classes for setting up (partially-populated)
|
|
|
|
# homeservers; either as a full homeserver as a real application, or a small
|
|
|
|
# partial one for unit test mocking.
|
|
|
|
|
|
|
|
# Imports required for the default HomeServer() implementation
|
|
|
|
from synapse.federation import initialize_http_replication
|
2014-12-05 09:20:48 -07:00
|
|
|
from synapse.events.utils import serialize_event
|
2014-08-26 11:57:46 -06:00
|
|
|
from synapse.notifier import Notifier
|
2014-08-12 08:10:52 -06:00
|
|
|
from synapse.api.auth import Auth
|
|
|
|
from synapse.handlers import Handlers
|
|
|
|
from synapse.rest import RestServletFactory
|
|
|
|
from synapse.state import StateHandler
|
|
|
|
from synapse.storage import DataStore
|
2014-10-30 11:00:11 -06:00
|
|
|
from synapse.types import UserID, RoomAlias, RoomID, EventID
|
2014-08-12 08:10:52 -06:00
|
|
|
from synapse.util import Clock
|
|
|
|
from synapse.util.distributor import Distributor
|
|
|
|
from synapse.util.lockutils import LockManager
|
2014-08-26 11:57:46 -06:00
|
|
|
from synapse.streams.events import EventSources
|
2014-09-02 10:57:04 -06:00
|
|
|
from synapse.api.ratelimiting import Ratelimiter
|
2014-09-30 08:15:10 -06:00
|
|
|
from synapse.crypto.keyring import Keyring
|
2014-12-04 08:50:01 -07:00
|
|
|
from synapse.events.builder import EventBuilderFactory
|
2014-08-12 08:10:52 -06:00
|
|
|
|
|
|
|
|
|
|
|
class BaseHomeServer(object):
|
|
|
|
"""A basic homeserver object without lazy component builders.
|
|
|
|
|
|
|
|
This will need all of the components it requires to either be passed as
|
|
|
|
constructor arguments, or the relevant methods overriding to create them.
|
|
|
|
Typically this would only be used for unit tests.
|
|
|
|
|
|
|
|
For every dependency in the DEPENDENCIES list below, this class creates one
|
|
|
|
method,
|
|
|
|
def get_DEPENDENCY(self)
|
|
|
|
which returns the value of that dependency. If no value has yet been set
|
|
|
|
nor was provided to the constructor, it will attempt to call a lazy builder
|
|
|
|
method called
|
|
|
|
def build_DEPENDENCY(self)
|
|
|
|
which must be implemented by the subclass. This code may call any of the
|
|
|
|
required "get" methods on the instance to obtain the sub-dependencies that
|
|
|
|
one requires.
|
|
|
|
"""
|
|
|
|
|
|
|
|
DEPENDENCIES = [
|
|
|
|
'clock',
|
|
|
|
'http_client',
|
2014-09-10 08:42:15 -06:00
|
|
|
'db_name',
|
2014-08-12 08:10:52 -06:00
|
|
|
'db_pool',
|
|
|
|
'persistence_service',
|
|
|
|
'replication_layer',
|
|
|
|
'datastore',
|
|
|
|
'handlers',
|
|
|
|
'auth',
|
|
|
|
'rest_servlet_factory',
|
|
|
|
'state_handler',
|
|
|
|
'room_lock_manager',
|
|
|
|
'notifier',
|
|
|
|
'distributor',
|
2014-08-14 02:52:20 -06:00
|
|
|
'resource_for_client',
|
|
|
|
'resource_for_federation',
|
|
|
|
'resource_for_web_client',
|
2014-08-18 08:01:08 -06:00
|
|
|
'resource_for_content_repo',
|
2014-09-23 11:40:59 -06:00
|
|
|
'resource_for_server_key',
|
2014-12-02 12:51:47 -07:00
|
|
|
'resource_for_media_repository',
|
2014-08-26 11:57:46 -06:00
|
|
|
'event_sources',
|
2014-09-02 10:57:04 -06:00
|
|
|
'ratelimiter',
|
2014-09-30 08:15:10 -06:00
|
|
|
'keyring',
|
2014-12-04 08:50:01 -07:00
|
|
|
'event_builder_factory',
|
2014-08-12 08:10:52 -06:00
|
|
|
]
|
|
|
|
|
|
|
|
def __init__(self, hostname, **kwargs):
|
|
|
|
"""
|
|
|
|
Args:
|
|
|
|
hostname : The hostname for the server.
|
|
|
|
"""
|
|
|
|
self.hostname = hostname
|
|
|
|
self._building = {}
|
|
|
|
|
|
|
|
# Other kwargs are explicit dependencies
|
|
|
|
for depname in kwargs:
|
|
|
|
setattr(self, depname, kwargs[depname])
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def _make_dependency_method(cls, depname):
|
|
|
|
def _get(self):
|
|
|
|
if hasattr(self, depname):
|
|
|
|
return getattr(self, depname)
|
|
|
|
|
|
|
|
if hasattr(self, "build_%s" % (depname)):
|
|
|
|
# Prevent cyclic dependencies from deadlocking
|
|
|
|
if depname in self._building:
|
|
|
|
raise ValueError("Cyclic dependency while building %s" % (
|
|
|
|
depname,
|
|
|
|
))
|
|
|
|
self._building[depname] = 1
|
|
|
|
|
|
|
|
builder = getattr(self, "build_%s" % (depname))
|
|
|
|
dep = builder()
|
|
|
|
setattr(self, depname, dep)
|
|
|
|
|
|
|
|
del self._building[depname]
|
|
|
|
|
|
|
|
return dep
|
|
|
|
|
|
|
|
raise NotImplementedError(
|
|
|
|
"%s has no %s nor a builder for it" % (
|
|
|
|
type(self).__name__, depname,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
setattr(BaseHomeServer, "get_%s" % (depname), _get)
|
|
|
|
|
2014-08-27 02:43:42 -06:00
|
|
|
# TODO: Why are these parse_ methods so high up along with other globals?
|
|
|
|
# Surely these should be in a util package or in the api package?
|
|
|
|
|
2014-08-12 08:10:52 -06:00
|
|
|
# Other utility methods
|
|
|
|
def parse_userid(self, s):
|
|
|
|
"""Parse the string given by 's' as a User ID and return a UserID
|
|
|
|
object."""
|
2014-12-02 03:42:28 -07:00
|
|
|
return UserID.from_string(s)
|
2014-08-12 08:10:52 -06:00
|
|
|
|
2014-08-13 10:43:10 -06:00
|
|
|
def parse_roomalias(self, s):
|
|
|
|
"""Parse the string given by 's' as a Room Alias and return a RoomAlias
|
|
|
|
object."""
|
2014-12-02 03:42:28 -07:00
|
|
|
return RoomAlias.from_string(s)
|
2014-08-13 10:43:10 -06:00
|
|
|
|
2014-08-27 02:43:42 -06:00
|
|
|
def parse_roomid(self, s):
|
|
|
|
"""Parse the string given by 's' as a Room ID and return a RoomID
|
|
|
|
object."""
|
2014-12-02 03:42:28 -07:00
|
|
|
return RoomID.from_string(s)
|
2014-08-27 02:43:42 -06:00
|
|
|
|
2014-10-30 11:00:11 -06:00
|
|
|
def parse_eventid(self, s):
|
|
|
|
"""Parse the string given by 's' as a Event ID and return a EventID
|
|
|
|
object."""
|
2014-12-02 03:42:28 -07:00
|
|
|
return EventID.from_string(s)
|
2014-10-30 11:00:11 -06:00
|
|
|
|
2015-01-08 07:27:04 -07:00
|
|
|
def serialize_event(self, e, as_client_event=True):
|
|
|
|
return serialize_event(self, e, as_client_event)
|
2014-09-15 06:26:05 -06:00
|
|
|
|
2014-09-26 09:36:24 -06:00
|
|
|
def get_ip_from_request(self, request):
|
|
|
|
# May be an X-Forwarding-For header depending on config
|
|
|
|
ip_addr = request.getClientIP()
|
|
|
|
if self.config.captcha_ip_origin_is_x_forwarded:
|
|
|
|
# use the header
|
|
|
|
if request.requestHeaders.hasHeader("X-Forwarded-For"):
|
|
|
|
ip_addr = request.requestHeaders.getRawHeaders(
|
|
|
|
"X-Forwarded-For"
|
|
|
|
)[0]
|
|
|
|
|
|
|
|
return ip_addr
|
|
|
|
|
2014-12-02 03:42:28 -07:00
|
|
|
def is_mine(self, domain_specific_string):
|
|
|
|
return domain_specific_string.domain == self.hostname
|
|
|
|
|
2014-08-12 08:10:52 -06:00
|
|
|
# Build magic accessors for every dependency
|
|
|
|
for depname in BaseHomeServer.DEPENDENCIES:
|
|
|
|
BaseHomeServer._make_dependency_method(depname)
|
|
|
|
|
|
|
|
|
|
|
|
class HomeServer(BaseHomeServer):
|
|
|
|
"""A homeserver object that will construct most of its dependencies as
|
|
|
|
required.
|
|
|
|
|
|
|
|
It still requires the following to be specified by the caller:
|
2014-08-14 03:18:54 -06:00
|
|
|
resource_for_client
|
|
|
|
resource_for_web_client
|
|
|
|
resource_for_federation
|
2014-08-18 08:01:08 -06:00
|
|
|
resource_for_content_repo
|
2014-08-12 08:10:52 -06:00
|
|
|
http_client
|
|
|
|
db_pool
|
|
|
|
"""
|
|
|
|
|
|
|
|
def build_clock(self):
|
|
|
|
return Clock()
|
|
|
|
|
|
|
|
def build_replication_layer(self):
|
|
|
|
return initialize_http_replication(self)
|
|
|
|
|
|
|
|
def build_datastore(self):
|
|
|
|
return DataStore(self)
|
|
|
|
|
|
|
|
def build_handlers(self):
|
|
|
|
return Handlers(self)
|
|
|
|
|
|
|
|
def build_notifier(self):
|
|
|
|
return Notifier(self)
|
|
|
|
|
|
|
|
def build_auth(self):
|
|
|
|
return Auth(self)
|
|
|
|
|
|
|
|
def build_rest_servlet_factory(self):
|
|
|
|
return RestServletFactory(self)
|
|
|
|
|
|
|
|
def build_state_handler(self):
|
|
|
|
return StateHandler(self)
|
|
|
|
|
|
|
|
def build_room_lock_manager(self):
|
|
|
|
return LockManager()
|
|
|
|
|
|
|
|
def build_distributor(self):
|
|
|
|
return Distributor()
|
|
|
|
|
2014-08-26 11:57:46 -06:00
|
|
|
def build_event_sources(self):
|
|
|
|
return EventSources(self)
|
|
|
|
|
2014-09-02 10:57:04 -06:00
|
|
|
def build_ratelimiter(self):
|
|
|
|
return Ratelimiter()
|
|
|
|
|
2014-09-30 08:15:10 -06:00
|
|
|
def build_keyring(self):
|
|
|
|
return Keyring(self)
|
|
|
|
|
2014-12-04 08:50:01 -07:00
|
|
|
def build_event_builder_factory(self):
|
|
|
|
return EventBuilderFactory(
|
|
|
|
clock=self.get_clock(),
|
|
|
|
hostname=self.hostname,
|
|
|
|
)
|
|
|
|
|
2014-08-13 08:57:58 -06:00
|
|
|
def register_servlets(self):
|
2014-08-13 06:50:01 -06:00
|
|
|
""" Register all servlets associated with this HomeServer.
|
|
|
|
"""
|
|
|
|
# Simply building the ServletFactory is sufficient to have it register
|
2014-08-14 02:52:20 -06:00
|
|
|
self.get_rest_servlet_factory()
|