synapse-product/synapse/app/media_repository.py

176 lines
6.2 KiB
Python
Raw Normal View History

2016-08-18 09:38:42 +01:00
#!/usr/bin/env python
# -*- coding: utf-8 -*-
# Copyright 2016 OpenMarket Ltd
#
# 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.
import logging
import sys
2016-08-18 09:38:42 +01:00
2018-07-09 16:09:20 +10:00
from twisted.internet import reactor
from twisted.web.resource import NoResource
2016-08-18 09:38:42 +01:00
import synapse
from synapse import events
from synapse.api.urls import LEGACY_MEDIA_PREFIX, MEDIA_PREFIX
from synapse.app import _base
2016-08-18 09:38:42 +01:00
from synapse.config._base import ConfigError
from synapse.config.homeserver import HomeServerConfig
from synapse.config.logger import setup_logging
from synapse.http.server import JsonResource
2016-08-18 09:38:42 +01:00
from synapse.http.site import SynapseSite
from synapse.logging.context import LoggingContext
from synapse.metrics import METRICS_PREFIX, MetricsResource, RegistryProxy
2016-08-18 09:38:42 +01:00
from synapse.replication.slave.storage._base import BaseSlavedStore
from synapse.replication.slave.storage.appservice import SlavedApplicationServiceStore
from synapse.replication.slave.storage.client_ips import SlavedClientIpStore
2016-08-18 09:38:42 +01:00
from synapse.replication.slave.storage.registration import SlavedRegistrationStore
from synapse.replication.slave.storage.room import RoomStore
from synapse.replication.slave.storage.transactions import SlavedTransactionStore
from synapse.replication.tcp.client import ReplicationClientHandler
from synapse.rest.admin import register_servlets_for_media_repo
2016-08-18 09:38:42 +01:00
from synapse.server import HomeServer
from synapse.storage.data_stores.main.media_repository import MediaRepositoryStore
2016-08-18 09:38:42 +01:00
from synapse.util.httpresourcetree import create_resource_tree
from synapse.util.manhole import manhole
from synapse.util.versionstring import get_version_string
logger = logging.getLogger("synapse.app.media_repository")
class MediaRepositorySlavedStore(
RoomStore,
2016-08-18 09:38:42 +01:00
SlavedApplicationServiceStore,
SlavedRegistrationStore,
SlavedClientIpStore,
SlavedTransactionStore,
2016-08-18 09:38:42 +01:00
BaseSlavedStore,
MediaRepositoryStore,
):
pass
class MediaRepositoryServer(HomeServer):
DATASTORE_CLASS = MediaRepositorySlavedStore
2016-08-18 09:38:42 +01:00
def _listen_http(self, listener_config):
port = listener_config["port"]
2017-01-10 17:21:41 +00:00
bind_addresses = listener_config["bind_addresses"]
2016-08-18 09:38:42 +01:00
site_tag = listener_config.get("tag", port)
resources = {}
for res in listener_config["resources"]:
for name in res["names"]:
if name == "metrics":
resources[METRICS_PREFIX] = MetricsResource(RegistryProxy)
2016-08-18 09:38:42 +01:00
elif name == "media":
media_repo = self.get_media_repository_resource()
# We need to serve the admin servlets for media on the
# worker.
admin_resource = JsonResource(self, canonical_json=False)
register_servlets_for_media_repo(self, admin_resource)
2019-06-20 19:32:02 +10:00
resources.update(
{
MEDIA_PREFIX: media_repo,
LEGACY_MEDIA_PREFIX: media_repo,
"/_synapse/admin": admin_resource,
2019-06-20 19:32:02 +10:00
}
)
2016-08-18 09:38:42 +01:00
root_resource = create_resource_tree(resources, NoResource())
_base.listen_tcp(
bind_addresses,
port,
SynapseSite(
"synapse.access.http.%s" % (site_tag,),
site_tag,
listener_config,
root_resource,
self.version_string,
2019-06-20 19:32:02 +10:00
),
)
logger.info("Synapse media repository now listening on port %d", port)
2016-08-18 09:38:42 +01:00
def start_listening(self, listeners):
for listener in listeners:
if listener["type"] == "http":
self._listen_http(listener)
elif listener["type"] == "manhole":
_base.listen_tcp(
listener["bind_addresses"],
listener["port"],
manhole(
2019-06-20 19:32:02 +10:00
username="matrix", password="rabbithole", globals={"hs": self}
),
)
elif listener["type"] == "metrics":
if not self.get_config().enable_metrics:
logger.warning(
2019-06-20 19:32:02 +10:00
(
"Metrics listener configured, but "
"enable_metrics is not True!"
)
)
else:
2019-06-20 19:32:02 +10:00
_base.listen_metrics(listener["bind_addresses"], listener["port"])
2016-08-18 09:38:42 +01:00
else:
logger.warning("Unrecognized listener type: %s", listener["type"])
2016-08-18 09:38:42 +01:00
self.get_tcp_replication().start_replication(self)
2016-08-18 09:38:42 +01:00
def build_tcp_replication(self):
return ReplicationClientHandler(self.get_datastore())
2016-08-18 09:38:42 +01:00
def start(config_options):
try:
config = HomeServerConfig.load_config(
"Synapse media repository", config_options
)
except ConfigError as e:
sys.stderr.write("\n" + str(e) + "\n")
2016-08-18 09:38:42 +01:00
sys.exit(1)
assert config.worker_app == "synapse.app.media_repository"
if config.enable_media_repo:
_base.quit_with_error(
"enable_media_repo must be disabled in the main synapse process\n"
"before the media repo can be run in a separate worker.\n"
"Please add ``enable_media_repo: false`` to the main config\n"
)
events.USE_FROZEN_DICTS = config.use_frozen_dicts
2016-08-18 09:38:42 +01:00
ss = MediaRepositoryServer(
config.server_name,
config=config,
version_string="Synapse/" + get_version_string(synapse),
)
setup_logging(ss, config, use_worker_options=True)
2016-08-18 09:38:42 +01:00
ss.setup()
reactor.addSystemEventTrigger(
"before", "startup", _base.start, ss, config.worker_listeners
)
2016-08-18 09:38:42 +01:00
_base.start_worker_reactor("synapse-media-repository", config)
2016-08-18 09:38:42 +01:00
2019-06-20 19:32:02 +10:00
if __name__ == "__main__":
2016-08-18 09:38:42 +01:00
with LoggingContext("main"):
start(sys.argv[1:])