2014-08-12 15:10:52 +01:00
|
|
|
# -*- coding: utf-8 -*-
|
2016-01-07 04:26:29 +00:00
|
|
|
# Copyright 2014-2016 OpenMarket Ltd
|
2014-08-12 15:10:52 +01: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-13 03:14:34 +01:00
|
|
|
|
2014-08-12 15:10:52 +01: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
|
2016-08-01 18:02:07 +01:00
|
|
|
import logging
|
|
|
|
|
2016-01-26 13:52:29 +00:00
|
|
|
from twisted.enterprise import adbapi
|
2016-08-01 18:02:07 +01:00
|
|
|
from twisted.web.client import BrowserLikePolicyForHTTPS
|
2016-01-26 13:52:29 +00:00
|
|
|
|
2016-08-01 18:02:07 +01:00
|
|
|
from synapse.api.auth import Auth
|
|
|
|
from synapse.api.filtering import Filtering
|
|
|
|
from synapse.api.ratelimiting import Ratelimiter
|
2016-05-31 13:53:48 +01:00
|
|
|
from synapse.appservice.api import ApplicationServiceApi
|
2016-08-01 18:02:07 +01:00
|
|
|
from synapse.appservice.scheduler import ApplicationServiceScheduler
|
|
|
|
from synapse.crypto.keyring import Keyring
|
|
|
|
from synapse.events.builder import EventBuilderFactory
|
2017-09-26 19:20:23 +01:00
|
|
|
from synapse.events.spamcheck import SpamChecker
|
2014-08-12 15:10:52 +01:00
|
|
|
from synapse.federation import initialize_http_replication
|
2016-11-16 17:34:44 +00:00
|
|
|
from synapse.federation.send_queue import FederationRemoteSendQueue
|
2016-11-16 14:15:50 +00:00
|
|
|
from synapse.federation.transport.client import TransportLayerClient
|
|
|
|
from synapse.federation.transaction_queue import TransactionQueue
|
2014-08-12 15:10:52 +01:00
|
|
|
from synapse.handlers import Handlers
|
2016-08-01 18:02:07 +01:00
|
|
|
from synapse.handlers.appservice import ApplicationServicesHandler
|
2017-02-02 10:53:36 +00:00
|
|
|
from synapse.handlers.auth import AuthHandler, MacaroonGeneartor
|
2017-11-29 11:48:43 +00:00
|
|
|
from synapse.handlers.deactivate_account import DeactivateAccountHandler
|
2016-09-06 18:16:20 +01:00
|
|
|
from synapse.handlers.devicemessage import DeviceMessageHandler
|
2016-08-01 18:02:07 +01:00
|
|
|
from synapse.handlers.device import DeviceHandler
|
|
|
|
from synapse.handlers.e2e_keys import E2eKeysHandler
|
2016-05-16 18:56:37 +01:00
|
|
|
from synapse.handlers.presence import PresenceHandler
|
2016-09-14 14:07:37 +01:00
|
|
|
from synapse.handlers.room_list import RoomListHandler
|
2017-11-29 14:10:46 +00:00
|
|
|
from synapse.handlers.set_password import SetPasswordHandler
|
2016-05-16 20:19:26 +01:00
|
|
|
from synapse.handlers.sync import SyncHandler
|
2016-05-17 15:58:46 +01:00
|
|
|
from synapse.handlers.typing import TypingHandler
|
2016-08-12 15:31:44 +01:00
|
|
|
from synapse.handlers.events import EventHandler, EventStreamHandler
|
2016-09-21 11:46:28 +01:00
|
|
|
from synapse.handlers.initial_sync import InitialSyncHandler
|
2016-11-23 15:14:24 +00:00
|
|
|
from synapse.handlers.receipts import ReceiptsHandler
|
2017-04-11 15:01:39 +01:00
|
|
|
from synapse.handlers.read_marker import ReadMarkerHandler
|
2017-11-29 16:46:45 +00:00
|
|
|
from synapse.handlers.user_directory import UserDirectoryHandler
|
2017-07-10 14:52:27 +01:00
|
|
|
from synapse.handlers.groups_local import GroupsLocalHandler
|
2017-08-25 14:34:56 +01:00
|
|
|
from synapse.handlers.profile import ProfileHandler
|
2017-07-10 15:44:15 +01:00
|
|
|
from synapse.groups.groups_server import GroupsServerHandler
|
|
|
|
from synapse.groups.attestations import GroupAttestionRenewer, GroupAttestationSigning
|
2016-08-01 18:02:07 +01:00
|
|
|
from synapse.http.client import SimpleHttpClient, InsecureInterceptableContextFactory
|
|
|
|
from synapse.http.matrixfederationclient import MatrixFederationHttpClient
|
|
|
|
from synapse.notifier import Notifier
|
2017-05-18 18:17:40 +01:00
|
|
|
from synapse.push.action_generator import ActionGenerator
|
2016-08-01 18:02:07 +01:00
|
|
|
from synapse.push.pusherpool import PusherPool
|
2017-11-21 11:08:08 +00:00
|
|
|
from synapse.rest.media.v1.media_repository import (
|
|
|
|
MediaRepository,
|
|
|
|
MediaRepositoryResource,
|
|
|
|
)
|
2014-08-12 15:10:52 +01:00
|
|
|
from synapse.state import StateHandler
|
2016-01-28 14:32:05 +00:00
|
|
|
from synapse.storage import DataStore
|
2016-08-01 18:02:07 +01:00
|
|
|
from synapse.streams.events import EventSources
|
2014-08-12 15:10:52 +01:00
|
|
|
from synapse.util import Clock
|
|
|
|
from synapse.util.distributor import Distributor
|
2016-01-26 15:51:06 +00:00
|
|
|
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
2014-08-12 15:10:52 +01:00
|
|
|
|
2016-01-26 13:52:29 +00:00
|
|
|
class HomeServer(object):
|
2014-08-12 15:10:52 +01:00
|
|
|
"""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 = [
|
|
|
|
'http_client',
|
|
|
|
'db_pool',
|
|
|
|
'replication_layer',
|
|
|
|
'handlers',
|
2015-03-24 17:24:15 +00:00
|
|
|
'v1auth',
|
2014-08-12 15:10:52 +01:00
|
|
|
'auth',
|
|
|
|
'state_handler',
|
2016-05-16 18:56:37 +01:00
|
|
|
'presence_handler',
|
2016-05-16 20:19:26 +01:00
|
|
|
'sync_handler',
|
2016-05-17 15:58:46 +01:00
|
|
|
'typing_handler',
|
2016-05-31 11:05:16 +01:00
|
|
|
'room_list_handler',
|
2016-06-02 13:31:45 +01:00
|
|
|
'auth_handler',
|
2016-07-15 13:19:07 +01:00
|
|
|
'device_handler',
|
2016-08-01 18:02:07 +01:00
|
|
|
'e2e_keys_handler',
|
2016-08-12 15:31:44 +01:00
|
|
|
'event_handler',
|
|
|
|
'event_stream_handler',
|
2016-09-21 11:46:28 +01:00
|
|
|
'initial_sync_handler',
|
2016-05-31 13:53:48 +01:00
|
|
|
'application_service_api',
|
|
|
|
'application_service_scheduler',
|
|
|
|
'application_service_handler',
|
2016-09-06 18:16:20 +01:00
|
|
|
'device_message_handler',
|
2017-08-25 14:34:56 +01:00
|
|
|
'profile_handler',
|
2017-11-29 11:48:43 +00:00
|
|
|
'deactivate_account_handler',
|
2017-11-29 14:10:46 +00:00
|
|
|
'set_password_handler',
|
2014-08-12 15:10:52 +01:00
|
|
|
'notifier',
|
2014-08-26 18:57:46 +01:00
|
|
|
'event_sources',
|
2014-09-30 15:15:10 +01:00
|
|
|
'keyring',
|
2014-12-18 15:15:22 +00:00
|
|
|
'pusherpool',
|
2014-12-04 15:50:01 +00:00
|
|
|
'event_builder_factory',
|
2015-01-27 14:28:56 +00:00
|
|
|
'filtering',
|
2015-09-09 12:02:07 +01:00
|
|
|
'http_client_context_factory',
|
|
|
|
'simple_http_client',
|
2016-06-29 14:57:59 +01:00
|
|
|
'media_repository',
|
2017-11-21 11:08:08 +00:00
|
|
|
'media_repository_resource',
|
2016-11-16 14:15:50 +00:00
|
|
|
'federation_transport_client',
|
|
|
|
'federation_sender',
|
2016-11-23 15:14:24 +00:00
|
|
|
'receipts_handler',
|
2017-02-02 10:53:36 +00:00
|
|
|
'macaroon_generator',
|
2017-03-27 16:33:44 +01:00
|
|
|
'tcp_replication',
|
2017-04-11 15:01:39 +01:00
|
|
|
'read_marker_handler',
|
2017-05-18 18:17:40 +01:00
|
|
|
'action_generator',
|
2017-05-31 11:51:01 +01:00
|
|
|
'user_directory_handler',
|
2017-07-10 14:52:27 +01:00
|
|
|
'groups_local_handler',
|
2017-07-10 15:44:15 +01:00
|
|
|
'groups_server_handler',
|
|
|
|
'groups_attestation_signing',
|
|
|
|
'groups_attestation_renewer',
|
2017-09-26 19:20:23 +01:00
|
|
|
'spam_checker',
|
2014-08-12 15:10:52 +01:00
|
|
|
]
|
|
|
|
|
|
|
|
def __init__(self, hostname, **kwargs):
|
|
|
|
"""
|
|
|
|
Args:
|
|
|
|
hostname : The hostname for the server.
|
|
|
|
"""
|
|
|
|
self.hostname = hostname
|
|
|
|
self._building = {}
|
|
|
|
|
2016-01-26 15:51:06 +00:00
|
|
|
self.clock = Clock()
|
|
|
|
self.distributor = Distributor()
|
|
|
|
self.ratelimiter = Ratelimiter()
|
|
|
|
|
2014-08-12 15:10:52 +01:00
|
|
|
# Other kwargs are explicit dependencies
|
|
|
|
for depname in kwargs:
|
|
|
|
setattr(self, depname, kwargs[depname])
|
|
|
|
|
2016-01-26 15:51:06 +00:00
|
|
|
def setup(self):
|
|
|
|
logger.info("Setting up.")
|
2016-01-28 14:32:05 +00:00
|
|
|
self.datastore = DataStore(self.get_db_conn(), self)
|
2016-01-26 15:51:06 +00:00
|
|
|
logger.info("Finished setting up.")
|
|
|
|
|
2014-09-26 16:36:24 +01:00
|
|
|
def get_ip_from_request(self, request):
|
2015-06-12 17:13:23 +01:00
|
|
|
# X-Forwarded-For is handled by our custom request type.
|
|
|
|
return request.getClientIP()
|
2014-09-26 16:36:24 +01:00
|
|
|
|
2014-12-02 10:42:28 +00:00
|
|
|
def is_mine(self, domain_specific_string):
|
|
|
|
return domain_specific_string.domain == self.hostname
|
|
|
|
|
2016-01-19 16:01:05 +00:00
|
|
|
def is_mine_id(self, string):
|
2016-01-19 16:11:39 +00:00
|
|
|
return string.split(":", 1)[1] == self.hostname
|
2016-01-19 16:01:05 +00:00
|
|
|
|
2017-11-21 10:50:23 +00:00
|
|
|
def get_clock(self):
|
|
|
|
return self.clock
|
|
|
|
|
|
|
|
def get_datastore(self):
|
|
|
|
return self.datastore
|
|
|
|
|
|
|
|
def get_config(self):
|
|
|
|
return self.config
|
|
|
|
|
|
|
|
def get_distributor(self):
|
|
|
|
return self.distributor
|
|
|
|
|
|
|
|
def get_ratelimiter(self):
|
|
|
|
return self.ratelimiter
|
|
|
|
|
2014-08-12 15:10:52 +01:00
|
|
|
def build_replication_layer(self):
|
|
|
|
return initialize_http_replication(self)
|
|
|
|
|
|
|
|
def build_handlers(self):
|
|
|
|
return Handlers(self)
|
|
|
|
|
|
|
|
def build_notifier(self):
|
|
|
|
return Notifier(self)
|
|
|
|
|
|
|
|
def build_auth(self):
|
|
|
|
return Auth(self)
|
|
|
|
|
2015-09-09 12:02:07 +01:00
|
|
|
def build_http_client_context_factory(self):
|
|
|
|
return (
|
2015-09-15 15:50:13 +01:00
|
|
|
InsecureInterceptableContextFactory()
|
2016-01-26 15:51:06 +00:00
|
|
|
if self.config.use_insecure_ssl_client_just_for_testing_do_not_use
|
2015-09-09 12:02:07 +01:00
|
|
|
else BrowserLikePolicyForHTTPS()
|
|
|
|
)
|
|
|
|
|
|
|
|
def build_simple_http_client(self):
|
|
|
|
return SimpleHttpClient(self)
|
|
|
|
|
2015-03-24 17:24:15 +00:00
|
|
|
def build_v1auth(self):
|
|
|
|
orf = Auth(self)
|
|
|
|
# Matrix spec makes no reference to what HTTP status code is returned,
|
|
|
|
# but the V1 API uses 403 where it means 401, and the webclient
|
|
|
|
# relies on this behaviour, so V1 gets its own copy of the auth
|
|
|
|
# with backwards compat behaviour.
|
|
|
|
orf.TOKEN_NOT_FOUND_HTTP_STATUS = 403
|
|
|
|
return orf
|
|
|
|
|
2014-08-12 15:10:52 +01:00
|
|
|
def build_state_handler(self):
|
|
|
|
return StateHandler(self)
|
|
|
|
|
2016-05-16 18:56:37 +01:00
|
|
|
def build_presence_handler(self):
|
|
|
|
return PresenceHandler(self)
|
|
|
|
|
2016-05-17 15:58:46 +01:00
|
|
|
def build_typing_handler(self):
|
|
|
|
return TypingHandler(self)
|
|
|
|
|
2016-05-16 20:19:26 +01:00
|
|
|
def build_sync_handler(self):
|
|
|
|
return SyncHandler(self)
|
|
|
|
|
2016-05-31 11:05:16 +01:00
|
|
|
def build_room_list_handler(self):
|
|
|
|
return RoomListHandler(self)
|
|
|
|
|
2016-06-02 13:31:45 +01:00
|
|
|
def build_auth_handler(self):
|
|
|
|
return AuthHandler(self)
|
|
|
|
|
2017-02-02 10:53:36 +00:00
|
|
|
def build_macaroon_generator(self):
|
|
|
|
return MacaroonGeneartor(self)
|
|
|
|
|
2016-07-15 13:19:07 +01:00
|
|
|
def build_device_handler(self):
|
|
|
|
return DeviceHandler(self)
|
|
|
|
|
2016-09-06 18:16:20 +01:00
|
|
|
def build_device_message_handler(self):
|
|
|
|
return DeviceMessageHandler(self)
|
|
|
|
|
2016-08-01 18:02:07 +01:00
|
|
|
def build_e2e_keys_handler(self):
|
|
|
|
return E2eKeysHandler(self)
|
|
|
|
|
2016-05-31 13:53:48 +01:00
|
|
|
def build_application_service_api(self):
|
|
|
|
return ApplicationServiceApi(self)
|
|
|
|
|
|
|
|
def build_application_service_scheduler(self):
|
|
|
|
return ApplicationServiceScheduler(self)
|
|
|
|
|
|
|
|
def build_application_service_handler(self):
|
|
|
|
return ApplicationServicesHandler(self)
|
|
|
|
|
2016-08-12 15:31:44 +01:00
|
|
|
def build_event_handler(self):
|
|
|
|
return EventHandler(self)
|
|
|
|
|
|
|
|
def build_event_stream_handler(self):
|
|
|
|
return EventStreamHandler(self)
|
|
|
|
|
2016-09-21 11:46:28 +01:00
|
|
|
def build_initial_sync_handler(self):
|
|
|
|
return InitialSyncHandler(self)
|
|
|
|
|
2017-08-25 14:34:56 +01:00
|
|
|
def build_profile_handler(self):
|
|
|
|
return ProfileHandler(self)
|
|
|
|
|
2017-11-29 11:48:43 +00:00
|
|
|
def build_deactivate_account_handler(self):
|
|
|
|
return DeactivateAccountHandler(self)
|
|
|
|
|
2017-11-29 14:10:46 +00:00
|
|
|
def build_set_password_handler(self):
|
|
|
|
return SetPasswordHandler(self)
|
|
|
|
|
2014-08-26 18:57:46 +01:00
|
|
|
def build_event_sources(self):
|
|
|
|
return EventSources(self)
|
|
|
|
|
2014-09-30 15:15:10 +01:00
|
|
|
def build_keyring(self):
|
|
|
|
return Keyring(self)
|
|
|
|
|
2014-12-04 15:50:01 +00:00
|
|
|
def build_event_builder_factory(self):
|
|
|
|
return EventBuilderFactory(
|
|
|
|
clock=self.get_clock(),
|
|
|
|
hostname=self.hostname,
|
|
|
|
)
|
2015-01-27 14:28:56 +00:00
|
|
|
|
|
|
|
def build_filtering(self):
|
|
|
|
return Filtering(self)
|
2015-01-29 14:55:27 +00:00
|
|
|
|
2014-11-19 18:20:59 +00:00
|
|
|
def build_pusherpool(self):
|
|
|
|
return PusherPool(self)
|
2016-01-26 13:52:29 +00:00
|
|
|
|
|
|
|
def build_http_client(self):
|
|
|
|
return MatrixFederationHttpClient(self)
|
|
|
|
|
|
|
|
def build_db_pool(self):
|
|
|
|
name = self.db_config["name"]
|
|
|
|
|
|
|
|
return adbapi.ConnectionPool(
|
|
|
|
name,
|
|
|
|
**self.db_config.get("args", {})
|
|
|
|
)
|
|
|
|
|
2017-11-21 11:08:08 +00:00
|
|
|
def build_media_repository_resource(self):
|
|
|
|
# build the media repo resource. This indirects through the HomeServer
|
|
|
|
# to ensure that we only have a single instance of
|
|
|
|
return MediaRepositoryResource(self)
|
|
|
|
|
2016-06-29 14:57:59 +01:00
|
|
|
def build_media_repository(self):
|
|
|
|
return MediaRepository(self)
|
|
|
|
|
2016-11-16 14:15:50 +00:00
|
|
|
def build_federation_transport_client(self):
|
|
|
|
return TransportLayerClient(self)
|
|
|
|
|
|
|
|
def build_federation_sender(self):
|
2016-11-23 14:09:47 +00:00
|
|
|
if self.should_send_federation():
|
2016-11-16 17:34:44 +00:00
|
|
|
return TransactionQueue(self)
|
2016-11-23 14:09:47 +00:00
|
|
|
elif not self.config.worker_app:
|
2016-11-16 17:34:44 +00:00
|
|
|
return FederationRemoteSendQueue(self)
|
2016-11-23 14:09:47 +00:00
|
|
|
else:
|
|
|
|
raise Exception("Workers cannot send federation traffic")
|
2016-11-16 14:15:50 +00:00
|
|
|
|
2016-11-23 15:14:24 +00:00
|
|
|
def build_receipts_handler(self):
|
|
|
|
return ReceiptsHandler(self)
|
|
|
|
|
2017-04-11 15:01:39 +01:00
|
|
|
def build_read_marker_handler(self):
|
|
|
|
return ReadMarkerHandler(self)
|
|
|
|
|
2017-03-27 16:33:44 +01:00
|
|
|
def build_tcp_replication(self):
|
|
|
|
raise NotImplementedError()
|
|
|
|
|
2017-05-18 18:17:40 +01:00
|
|
|
def build_action_generator(self):
|
|
|
|
return ActionGenerator(self)
|
|
|
|
|
2017-05-31 11:51:01 +01:00
|
|
|
def build_user_directory_handler(self):
|
2017-11-29 16:46:45 +00:00
|
|
|
return UserDirectoryHandler(self)
|
2017-05-31 11:51:01 +01:00
|
|
|
|
2017-07-10 14:52:27 +01:00
|
|
|
def build_groups_local_handler(self):
|
|
|
|
return GroupsLocalHandler(self)
|
|
|
|
|
2017-07-10 15:44:15 +01:00
|
|
|
def build_groups_server_handler(self):
|
|
|
|
return GroupsServerHandler(self)
|
|
|
|
|
|
|
|
def build_groups_attestation_signing(self):
|
|
|
|
return GroupAttestationSigning(self)
|
|
|
|
|
|
|
|
def build_groups_attestation_renewer(self):
|
|
|
|
return GroupAttestionRenewer(self)
|
|
|
|
|
2017-09-26 19:20:23 +01:00
|
|
|
def build_spam_checker(self):
|
|
|
|
return SpamChecker(self)
|
|
|
|
|
2016-04-21 17:21:02 +01:00
|
|
|
def remove_pusher(self, app_id, push_key, user_id):
|
|
|
|
return self.get_pusherpool().remove_pusher(app_id, push_key, user_id)
|
|
|
|
|
2016-11-23 14:09:47 +00:00
|
|
|
def should_send_federation(self):
|
|
|
|
"Should this server be sending federation traffic directly?"
|
|
|
|
return self.config.send_federation and (
|
|
|
|
not self.config.worker_app
|
|
|
|
or self.config.worker_app == "synapse.app.federation_sender"
|
|
|
|
)
|
|
|
|
|
2016-01-26 13:52:29 +00:00
|
|
|
|
|
|
|
def _make_dependency_method(depname):
|
|
|
|
def _get(hs):
|
|
|
|
try:
|
|
|
|
return getattr(hs, depname)
|
|
|
|
except AttributeError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
try:
|
|
|
|
builder = getattr(hs, "build_%s" % (depname))
|
|
|
|
except AttributeError:
|
|
|
|
builder = None
|
|
|
|
|
|
|
|
if builder:
|
|
|
|
# Prevent cyclic dependencies from deadlocking
|
|
|
|
if depname in hs._building:
|
|
|
|
raise ValueError("Cyclic dependency while building %s" % (
|
|
|
|
depname,
|
|
|
|
))
|
|
|
|
hs._building[depname] = 1
|
|
|
|
|
|
|
|
dep = builder()
|
|
|
|
setattr(hs, depname, dep)
|
|
|
|
|
|
|
|
del hs._building[depname]
|
|
|
|
|
|
|
|
return dep
|
|
|
|
|
|
|
|
raise NotImplementedError(
|
|
|
|
"%s has no %s nor a builder for it" % (
|
|
|
|
type(hs).__name__, depname,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
setattr(HomeServer, "get_%s" % (depname), _get)
|
|
|
|
|
|
|
|
|
|
|
|
# Build magic accessors for every dependency
|
|
|
|
for depname in HomeServer.DEPENDENCIES:
|
|
|
|
_make_dependency_method(depname)
|