mirror of
https://github.com/certbot/certbot.git
synced 2026-01-26 07:41:33 +03:00
126 lines
5.0 KiB
Python
126 lines
5.0 KiB
Python
# type: ignore
|
|
"""
|
|
General conftest for pytest execution of all integration tests lying
|
|
in the certbot_integration tests package.
|
|
As stated by pytest documentation, conftest module is used to set on
|
|
for a directory a specific configuration using built-in pytest hooks.
|
|
|
|
See https://docs.pytest.org/en/latest/reference.html#hook-reference
|
|
"""
|
|
import contextlib
|
|
import subprocess
|
|
import sys
|
|
|
|
from certbot_integration_tests.utils import acme_server as acme_lib
|
|
from certbot_integration_tests.utils import dns_server as dns_lib
|
|
|
|
|
|
def pytest_addoption(parser):
|
|
"""
|
|
Standard pytest hook to add options to the pytest parser.
|
|
:param parser: current pytest parser that will be used on the CLI
|
|
"""
|
|
parser.addoption('--acme-server', default='pebble',
|
|
choices=['boulder-v2', 'pebble'],
|
|
help='select the ACME server to use (boulder-v2, pebble), '
|
|
'defaulting to pebble')
|
|
parser.addoption('--dns-server', default='challtestsrv',
|
|
choices=['bind', 'challtestsrv'],
|
|
help='select the DNS server to use (bind, challtestsrv), '
|
|
'defaulting to challtestsrv')
|
|
|
|
|
|
def pytest_configure(config):
|
|
"""
|
|
Standard pytest hook used to add a configuration logic for each node of a pytest run.
|
|
:param config: the current pytest configuration
|
|
"""
|
|
if not hasattr(config, 'workerinput'): # If true, this is the primary node
|
|
with _print_on_err():
|
|
_setup_primary_node(config)
|
|
|
|
|
|
def pytest_configure_node(node):
|
|
"""
|
|
Standard pytest-xdist hook used to configure a worker node.
|
|
:param node: current worker node
|
|
"""
|
|
node.workerinput['acme_xdist'] = node.config.acme_xdist
|
|
node.workerinput['dns_xdist'] = node.config.dns_xdist
|
|
|
|
|
|
@contextlib.contextmanager
|
|
def _print_on_err():
|
|
"""
|
|
During pytest-xdist setup, stdout is used for nodes communication, so print is useless.
|
|
However, stderr is still available. This context manager transfers stdout to stderr
|
|
for the duration of the context, allowing to display prints to the user.
|
|
"""
|
|
old_stdout = sys.stdout
|
|
sys.stdout = sys.stderr
|
|
try:
|
|
yield
|
|
finally:
|
|
sys.stdout = old_stdout
|
|
|
|
|
|
def _setup_primary_node(config):
|
|
"""
|
|
Setup the environment for integration tests.
|
|
|
|
This function will:
|
|
- check runtime compatibility (Docker, docker-compose, Nginx)
|
|
- create a temporary workspace and the persistent GIT repositories space
|
|
- configure and start a DNS server using Docker, if configured
|
|
- configure and start paralleled ACME CA servers using Docker
|
|
- transfer ACME CA and DNS servers configurations to pytest nodes using env variables
|
|
|
|
This function modifies `config` by injecting the ACME CA and DNS server configurations,
|
|
in addition to cleanup functions for those servers.
|
|
|
|
:param config: Configuration of the pytest primary node. Is modified by this function.
|
|
"""
|
|
# Check for runtime compatibility: some tools are required to be available in PATH
|
|
if 'boulder' in config.option.acme_server:
|
|
try:
|
|
subprocess.check_output(['docker', '-v'], stderr=subprocess.STDOUT)
|
|
except (subprocess.CalledProcessError, OSError):
|
|
raise ValueError('Error: docker is required in PATH to launch the integration tests on'
|
|
'boulder, but is not installed or not available for current user.')
|
|
|
|
try:
|
|
subprocess.check_output(['docker-compose', '-v'], stderr=subprocess.STDOUT)
|
|
except (subprocess.CalledProcessError, OSError):
|
|
raise ValueError(
|
|
'Error: docker-compose is required in PATH to launch the integration tests, '
|
|
'but is not installed or not available for current user.'
|
|
)
|
|
|
|
# Parameter numprocesses is added to option by pytest-xdist
|
|
workers = ['primary'] if not config.option.numprocesses\
|
|
else ['gw{0}'.format(i) for i in range(config.option.numprocesses)]
|
|
|
|
# If a non-default DNS server is configured, start it and feed it to the ACME server
|
|
dns_server = None
|
|
acme_dns_server = None
|
|
if config.option.dns_server == 'bind':
|
|
dns_server = dns_lib.DNSServer(workers)
|
|
config.add_cleanup(dns_server.stop)
|
|
print('DNS xdist config:\n{0}'.format(dns_server.dns_xdist))
|
|
dns_server.start()
|
|
acme_dns_server = '{}:{}'.format(
|
|
dns_server.dns_xdist['address'],
|
|
dns_server.dns_xdist['port']
|
|
)
|
|
|
|
# By calling setup_acme_server we ensure that all necessary acme server instances will be
|
|
# fully started. This runtime is reflected by the acme_xdist returned.
|
|
acme_server = acme_lib.ACMEServer(config.option.acme_server, workers,
|
|
dns_server=acme_dns_server)
|
|
config.add_cleanup(acme_server.stop)
|
|
print('ACME xdist config:\n{0}'.format(acme_server.acme_xdist))
|
|
acme_server.start()
|
|
|
|
config.acme_xdist = acme_server.acme_xdist
|
|
config.dns_xdist = dns_server.dns_xdist if dns_server else None
|