2019-01-29 13:53:02 +00:00
|
|
|
#
|
2023-11-21 15:29:58 -05:00
|
|
|
# This file is licensed under the Affero General Public License (AGPL) version 3.
|
|
|
|
#
|
|
|
|
# Copyright (C) 2023 New Vector, Ltd
|
|
|
|
#
|
|
|
|
# This program is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU Affero General Public License as
|
|
|
|
# published by the Free Software Foundation, either version 3 of the
|
|
|
|
# License, or (at your option) any later version.
|
|
|
|
#
|
|
|
|
# See the GNU Affero General Public License for more details:
|
|
|
|
# <https://www.gnu.org/licenses/agpl-3.0.html>.
|
|
|
|
#
|
|
|
|
# Originally licensed under the Apache License, Version 2.0:
|
|
|
|
# <http://www.apache.org/licenses/LICENSE-2.0>.
|
|
|
|
#
|
|
|
|
# [This file includes modifications made by New Vector Limited]
|
2019-01-29 13:53:02 +00:00
|
|
|
#
|
|
|
|
#
|
|
|
|
import os.path
|
2019-06-10 16:06:25 +01:00
|
|
|
import subprocess
|
2022-11-16 10:25:24 -05:00
|
|
|
from typing import List
|
2019-06-10 16:06:25 +01:00
|
|
|
|
2023-10-25 07:39:45 -04:00
|
|
|
from incremental import Version
|
2019-06-10 16:06:25 +01:00
|
|
|
from zope.interface import implementer
|
2019-01-29 13:53:02 +00:00
|
|
|
|
2023-10-25 07:39:45 -04:00
|
|
|
import twisted
|
2019-01-29 13:53:02 +00:00
|
|
|
from OpenSSL import SSL
|
2019-06-10 16:06:25 +01:00
|
|
|
from OpenSSL.SSL import Connection
|
2023-02-07 00:20:04 +00:00
|
|
|
from twisted.internet.address import IPv4Address
|
2023-10-25 07:39:45 -04:00
|
|
|
from twisted.internet.interfaces import (
|
|
|
|
IOpenSSLServerConnectionCreator,
|
|
|
|
IProtocolFactory,
|
|
|
|
IReactorTime,
|
|
|
|
)
|
2019-11-01 14:07:44 +00:00
|
|
|
from twisted.internet.ssl import Certificate, trustRootFromCertificates
|
2023-10-25 07:39:45 -04:00
|
|
|
from twisted.protocols.tls import TLSMemoryBIOFactory, TLSMemoryBIOProtocol
|
2019-11-01 14:07:44 +00:00
|
|
|
from twisted.web.client import BrowserLikePolicyForHTTPS # noqa: F401
|
|
|
|
from twisted.web.iweb import IPolicyForHTTPS # noqa: F401
|
|
|
|
|
|
|
|
|
2023-02-07 00:20:04 +00:00
|
|
|
def get_test_https_policy() -> BrowserLikePolicyForHTTPS:
|
2019-11-01 14:07:44 +00:00
|
|
|
"""Get a test IPolicyForHTTPS which trusts the test CA cert
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
IPolicyForHTTPS
|
|
|
|
"""
|
|
|
|
ca_file = get_test_ca_cert_file()
|
|
|
|
with open(ca_file) as stream:
|
|
|
|
content = stream.read()
|
|
|
|
cert = Certificate.loadPEM(content)
|
|
|
|
trust_root = trustRootFromCertificates([cert])
|
|
|
|
return BrowserLikePolicyForHTTPS(trustRoot=trust_root)
|
2019-06-10 16:06:25 +01:00
|
|
|
|
|
|
|
|
2023-02-07 00:20:04 +00:00
|
|
|
def get_test_ca_cert_file() -> str:
|
2019-06-10 16:06:25 +01:00
|
|
|
"""Get the path to the test CA cert
|
|
|
|
|
|
|
|
The keypair is generated with:
|
|
|
|
|
|
|
|
openssl genrsa -out ca.key 2048
|
|
|
|
openssl req -new -x509 -key ca.key -days 3650 -out ca.crt \
|
|
|
|
-subj '/CN=synapse test CA'
|
|
|
|
"""
|
|
|
|
return os.path.join(os.path.dirname(__file__), "ca.crt")
|
|
|
|
|
|
|
|
|
2023-02-07 00:20:04 +00:00
|
|
|
def get_test_key_file() -> str:
|
2019-06-10 16:06:25 +01:00
|
|
|
"""get the path to the test key
|
|
|
|
|
|
|
|
The key file is made with:
|
|
|
|
|
|
|
|
openssl genrsa -out server.key 2048
|
|
|
|
"""
|
|
|
|
return os.path.join(os.path.dirname(__file__), "server.key")
|
|
|
|
|
|
|
|
|
|
|
|
cert_file_count = 0
|
|
|
|
|
|
|
|
CONFIG_TEMPLATE = b"""\
|
|
|
|
[default]
|
|
|
|
basicConstraints = CA:FALSE
|
|
|
|
keyUsage=nonRepudiation, digitalSignature, keyEncipherment
|
2019-06-10 17:55:01 +01:00
|
|
|
subjectAltName = %(sanentries)s
|
2019-06-10 16:06:25 +01:00
|
|
|
"""
|
|
|
|
|
|
|
|
|
2022-11-16 10:25:24 -05:00
|
|
|
def create_test_cert_file(sanlist: List[bytes]) -> str:
|
2019-06-10 16:06:25 +01:00
|
|
|
"""build an x509 certificate file
|
|
|
|
|
|
|
|
Args:
|
2022-11-16 10:25:24 -05:00
|
|
|
sanlist: a list of subjectAltName values for the cert
|
2019-06-10 16:06:25 +01:00
|
|
|
|
|
|
|
Returns:
|
2022-11-16 10:25:24 -05:00
|
|
|
The path to the file
|
2019-06-10 16:06:25 +01:00
|
|
|
"""
|
|
|
|
global cert_file_count
|
|
|
|
csr_filename = "server.csr"
|
|
|
|
cnf_filename = "server.%i.cnf" % (cert_file_count,)
|
|
|
|
cert_filename = "server.%i.crt" % (cert_file_count,)
|
|
|
|
cert_file_count += 1
|
|
|
|
|
|
|
|
# first build a CSR
|
2019-06-10 17:41:10 +01:00
|
|
|
subprocess.check_call(
|
2019-06-10 16:06:25 +01:00
|
|
|
[
|
|
|
|
"openssl",
|
|
|
|
"req",
|
|
|
|
"-new",
|
|
|
|
"-key",
|
|
|
|
get_test_key_file(),
|
|
|
|
"-subj",
|
|
|
|
"/",
|
|
|
|
"-out",
|
|
|
|
csr_filename,
|
2019-06-10 17:41:10 +01:00
|
|
|
]
|
2019-06-10 16:06:25 +01:00
|
|
|
)
|
2019-01-29 13:53:02 +00:00
|
|
|
|
2019-06-10 16:06:25 +01:00
|
|
|
# now a config file describing the right SAN entries
|
|
|
|
sanentries = b",".join(sanlist)
|
|
|
|
with open(cnf_filename, "wb") as f:
|
|
|
|
f.write(CONFIG_TEMPLATE % {b"sanentries": sanentries})
|
2019-01-29 13:53:02 +00:00
|
|
|
|
2019-06-10 16:06:25 +01:00
|
|
|
# finally the cert
|
|
|
|
ca_key_filename = os.path.join(os.path.dirname(__file__), "ca.key")
|
|
|
|
ca_cert_filename = get_test_ca_cert_file()
|
2019-06-10 17:41:10 +01:00
|
|
|
subprocess.check_call(
|
2019-06-10 16:06:25 +01:00
|
|
|
[
|
|
|
|
"openssl",
|
|
|
|
"x509",
|
|
|
|
"-req",
|
|
|
|
"-in",
|
|
|
|
csr_filename,
|
|
|
|
"-CA",
|
|
|
|
ca_cert_filename,
|
|
|
|
"-CAkey",
|
|
|
|
ca_key_filename,
|
|
|
|
"-set_serial",
|
|
|
|
"1",
|
|
|
|
"-extfile",
|
|
|
|
cnf_filename,
|
|
|
|
"-out",
|
|
|
|
cert_filename,
|
2019-06-10 17:41:10 +01:00
|
|
|
]
|
2019-06-10 16:06:25 +01:00
|
|
|
)
|
2019-01-29 13:53:02 +00:00
|
|
|
|
2019-06-10 16:06:25 +01:00
|
|
|
return cert_filename
|
2019-01-29 13:53:02 +00:00
|
|
|
|
|
|
|
|
2019-06-10 16:06:25 +01:00
|
|
|
@implementer(IOpenSSLServerConnectionCreator)
|
2020-09-04 06:54:56 -04:00
|
|
|
class TestServerTLSConnectionFactory:
|
2019-06-10 16:06:25 +01:00
|
|
|
"""An SSL connection creator which returns connections which present a certificate
|
|
|
|
signed by our test CA."""
|
2019-05-10 00:12:11 -05:00
|
|
|
|
2023-02-07 00:20:04 +00:00
|
|
|
def __init__(self, sanlist: List[bytes]):
|
2019-06-10 16:06:25 +01:00
|
|
|
"""
|
|
|
|
Args:
|
2023-02-07 00:20:04 +00:00
|
|
|
sanlist: a list of subjectAltName values for the cert
|
2019-06-10 16:06:25 +01:00
|
|
|
"""
|
|
|
|
self._cert_file = create_test_cert_file(sanlist)
|
2019-01-29 13:53:02 +00:00
|
|
|
|
2023-02-07 00:20:04 +00:00
|
|
|
def serverConnectionForTLS(self, tlsProtocol: TLSMemoryBIOProtocol) -> Connection:
|
2020-09-11 04:49:08 +10:00
|
|
|
ctx = SSL.Context(SSL.SSLv23_METHOD)
|
2019-06-10 16:06:25 +01:00
|
|
|
ctx.use_certificate_file(self._cert_file)
|
|
|
|
ctx.use_privatekey_file(get_test_key_file())
|
|
|
|
return Connection(ctx, None)
|
2023-02-07 00:20:04 +00:00
|
|
|
|
|
|
|
|
2023-10-25 07:39:45 -04:00
|
|
|
def wrap_server_factory_for_tls(
|
|
|
|
factory: IProtocolFactory, clock: IReactorTime, sanlist: List[bytes]
|
|
|
|
) -> TLSMemoryBIOFactory:
|
|
|
|
"""Wrap an existing Protocol Factory with a test TLSMemoryBIOFactory
|
|
|
|
|
|
|
|
The resultant factory will create a TLS server which presents a certificate
|
|
|
|
signed by our test CA, valid for the domains in `sanlist`
|
|
|
|
|
|
|
|
Args:
|
|
|
|
factory: protocol factory to wrap
|
|
|
|
sanlist: list of domains the cert should be valid for
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
interfaces.IProtocolFactory
|
|
|
|
"""
|
|
|
|
connection_creator = TestServerTLSConnectionFactory(sanlist=sanlist)
|
|
|
|
# Twisted > 23.8.0 has a different API that accepts a clock.
|
|
|
|
if twisted.version <= Version("Twisted", 23, 8, 0):
|
|
|
|
return TLSMemoryBIOFactory(
|
|
|
|
connection_creator, isClient=False, wrappedFactory=factory
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
return TLSMemoryBIOFactory(
|
2023-11-01 10:23:13 +00:00
|
|
|
connection_creator, isClient=False, wrappedFactory=factory, clock=clock
|
2023-10-25 07:39:45 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
|
2023-02-07 00:20:04 +00:00
|
|
|
# A dummy address, useful for tests that use FakeTransport and don't care about where
|
|
|
|
# packets are going to/coming from.
|
|
|
|
dummy_address = IPv4Address("TCP", "127.0.0.1", 80)
|