2019-04-11 10:08:13 -06:00
|
|
|
# Copyright 2018 New Vector Ltd
|
2018-12-05 06:38:58 -07: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.
|
|
|
|
import logging
|
2021-09-15 06:45:32 -06:00
|
|
|
from typing import TYPE_CHECKING, Optional
|
2018-12-05 06:38:58 -07:00
|
|
|
|
|
|
|
from twisted.web.resource import Resource
|
2021-09-15 06:45:32 -06:00
|
|
|
from twisted.web.server import Request
|
2019-02-18 08:52:26 -07:00
|
|
|
|
2019-02-18 07:59:23 -07:00
|
|
|
from synapse.http.server import set_cors_headers
|
2022-10-18 09:52:25 -06:00
|
|
|
from synapse.http.site import SynapseRequest
|
2021-09-15 06:45:32 -06:00
|
|
|
from synapse.types import JsonDict
|
2020-08-20 08:32:33 -06:00
|
|
|
from synapse.util import json_encoder
|
2021-11-01 09:10:16 -06:00
|
|
|
from synapse.util.stringutils import parse_server_name
|
2018-12-05 06:38:58 -07:00
|
|
|
|
2021-09-15 06:45:32 -06:00
|
|
|
if TYPE_CHECKING:
|
|
|
|
from synapse.server import HomeServer
|
|
|
|
|
2018-12-05 06:38:58 -07:00
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
2020-09-04 04:54:56 -06:00
|
|
|
class WellKnownBuilder:
|
2021-09-15 06:45:32 -06:00
|
|
|
def __init__(self, hs: "HomeServer"):
|
2018-12-05 06:38:58 -07:00
|
|
|
self._config = hs.config
|
|
|
|
|
2021-09-15 06:45:32 -06:00
|
|
|
def get_well_known(self) -> Optional[JsonDict]:
|
2021-11-08 07:13:10 -07:00
|
|
|
if not self._config.server.serve_client_wellknown:
|
2021-02-11 09:16:54 -07:00
|
|
|
return None
|
|
|
|
|
2021-09-13 11:07:12 -06:00
|
|
|
result = {"m.homeserver": {"base_url": self._config.server.public_baseurl}}
|
2018-12-05 06:38:58 -07:00
|
|
|
|
2021-10-04 05:18:54 -06:00
|
|
|
if self._config.registration.default_identity_server:
|
2018-12-05 06:38:58 -07:00
|
|
|
result["m.identity_server"] = {
|
2021-10-04 05:18:54 -06:00
|
|
|
"base_url": self._config.registration.default_identity_server
|
2018-12-05 06:38:58 -07:00
|
|
|
}
|
|
|
|
|
2023-05-09 08:20:04 -06:00
|
|
|
# We use the MSC3861 values as they are used by multiple MSCs
|
|
|
|
if self._config.experimental.msc3861.enabled:
|
2021-11-18 07:21:00 -07:00
|
|
|
result["org.matrix.msc2965.authentication"] = {
|
2023-05-09 08:20:04 -06:00
|
|
|
"issuer": self._config.experimental.msc3861.issuer
|
2021-11-18 07:21:00 -07:00
|
|
|
}
|
2023-05-09 08:20:04 -06:00
|
|
|
if self._config.experimental.msc3861.account_management_url is not None:
|
2021-11-18 07:21:00 -07:00
|
|
|
result["org.matrix.msc2965.authentication"][
|
|
|
|
"account"
|
2023-05-09 08:20:04 -06:00
|
|
|
] = self._config.experimental.msc3861.account_management_url
|
2021-11-18 07:21:00 -07:00
|
|
|
|
2022-06-16 04:48:18 -06:00
|
|
|
if self._config.server.extra_well_known_client_content:
|
|
|
|
for (
|
|
|
|
key,
|
|
|
|
value,
|
|
|
|
) in self._config.server.extra_well_known_client_content.items():
|
|
|
|
if key not in result:
|
|
|
|
result[key] = value
|
|
|
|
|
2018-12-05 06:38:58 -07:00
|
|
|
return result
|
|
|
|
|
|
|
|
|
2021-11-01 09:10:16 -06:00
|
|
|
class ClientWellKnownResource(Resource):
|
|
|
|
"""A Twisted web resource which renders the .well-known/matrix/client file"""
|
2018-12-05 06:38:58 -07:00
|
|
|
|
|
|
|
isLeaf = 1
|
|
|
|
|
2021-09-15 06:45:32 -06:00
|
|
|
def __init__(self, hs: "HomeServer"):
|
2018-12-05 06:38:58 -07:00
|
|
|
Resource.__init__(self)
|
|
|
|
self._well_known_builder = WellKnownBuilder(hs)
|
|
|
|
|
2022-10-18 09:52:25 -06:00
|
|
|
def render_GET(self, request: SynapseRequest) -> bytes:
|
2019-02-14 12:53:12 -07:00
|
|
|
set_cors_headers(request)
|
2018-12-05 06:38:58 -07:00
|
|
|
r = self._well_known_builder.get_well_known()
|
|
|
|
if not r:
|
|
|
|
request.setResponseCode(404)
|
|
|
|
request.setHeader(b"Content-Type", b"text/plain")
|
|
|
|
return b".well-known not available"
|
|
|
|
|
2019-04-24 10:44:06 -06:00
|
|
|
logger.debug("returning: %s", r)
|
2018-12-05 06:38:58 -07:00
|
|
|
request.setHeader(b"Content-Type", b"application/json")
|
2020-08-20 08:32:33 -06:00
|
|
|
return json_encoder.encode(r).encode("utf-8")
|
2021-11-01 09:10:16 -06:00
|
|
|
|
|
|
|
|
|
|
|
class ServerWellKnownResource(Resource):
|
|
|
|
"""Resource for .well-known/matrix/server, redirecting to port 443"""
|
|
|
|
|
|
|
|
isLeaf = 1
|
|
|
|
|
|
|
|
def __init__(self, hs: "HomeServer"):
|
|
|
|
super().__init__()
|
|
|
|
self._serve_server_wellknown = hs.config.server.serve_server_wellknown
|
|
|
|
|
|
|
|
host, port = parse_server_name(hs.config.server.server_name)
|
|
|
|
|
|
|
|
# If we've got this far, then https://<server_name>/ must route to us, so
|
|
|
|
# we just redirect the traffic to port 443 instead of 8448.
|
|
|
|
if port is None:
|
|
|
|
port = 443
|
|
|
|
|
|
|
|
self._response = json_encoder.encode({"m.server": f"{host}:{port}"}).encode(
|
|
|
|
"utf-8"
|
|
|
|
)
|
|
|
|
|
|
|
|
def render_GET(self, request: Request) -> bytes:
|
|
|
|
if not self._serve_server_wellknown:
|
|
|
|
request.setResponseCode(404)
|
|
|
|
request.setHeader(b"Content-Type", b"text/plain")
|
|
|
|
return b"404. Is anything ever truly *well* known?\n"
|
|
|
|
|
|
|
|
request.setHeader(b"Content-Type", b"application/json")
|
|
|
|
return self._response
|
|
|
|
|
|
|
|
|
|
|
|
def well_known_resource(hs: "HomeServer") -> Resource:
|
|
|
|
"""Returns a Twisted web resource which handles '.well-known' requests"""
|
|
|
|
res = Resource()
|
|
|
|
matrix_resource = Resource()
|
|
|
|
res.putChild(b"matrix", matrix_resource)
|
|
|
|
|
|
|
|
matrix_resource.putChild(b"server", ServerWellKnownResource(hs))
|
|
|
|
matrix_resource.putChild(b"client", ClientWellKnownResource(hs))
|
|
|
|
|
|
|
|
return res
|