2020-06-04 00:59:27 -05:00
|
|
|
#
|
|
|
|
# Copyright (C) 2020 FreeIPA Contributors see COPYING for license
|
|
|
|
#
|
|
|
|
|
|
|
|
import time
|
|
|
|
|
2020-06-24 23:51:32 -05:00
|
|
|
from cryptography.hazmat.backends import default_backend
|
|
|
|
from cryptography import x509
|
2020-06-04 00:59:27 -05:00
|
|
|
import pytest
|
|
|
|
|
|
|
|
from ipalib.constants import IPA_CA_RECORD
|
|
|
|
from ipatests.test_integration.base import IntegrationTest
|
|
|
|
from ipatests.pytest_ipa.integration import tasks
|
|
|
|
from ipaplatform.osinfo import osinfo
|
|
|
|
|
|
|
|
|
|
|
|
# RHEL does not have certbot. EPEL's version is broken with
|
|
|
|
# python-cryptography-2.3; likewise recent PyPI releases.
|
|
|
|
# So for now, on RHEL we suppress tests that use certbot.
|
|
|
|
skip_certbot_tests = osinfo.id not in ['fedora',]
|
|
|
|
|
2020-06-10 23:33:08 -05:00
|
|
|
# Fedora mod_md package needs some patches before it will work.
|
|
|
|
# RHEL version has the patches.
|
|
|
|
skip_mod_md_tests = osinfo.id not in ['rhel',]
|
|
|
|
|
2020-06-04 00:59:27 -05:00
|
|
|
|
|
|
|
class TestACME(IntegrationTest):
|
|
|
|
"""
|
|
|
|
Test the FreeIPA ACME service by using ACME clients on a FreeIPA client.
|
|
|
|
|
2020-06-10 23:33:08 -05:00
|
|
|
We currently test:
|
2020-06-04 00:59:27 -05:00
|
|
|
|
2020-06-10 23:33:08 -05:00
|
|
|
* service enable/disable (using Curl)
|
|
|
|
* http-01 challenge with Certbot's standalone HTTP server
|
2020-06-24 23:51:32 -05:00
|
|
|
* revocation with Certbot
|
2020-06-10 23:33:08 -05:00
|
|
|
* http-01 challenge with mod_md
|
|
|
|
|
|
|
|
Tests we should add:
|
|
|
|
|
|
|
|
* dns-01 challenge with Certbot and FreeIPA DNS
|
|
|
|
(see https://frasertweedale.github.io
|
|
|
|
/blog-redhat/posts/2020-05-13-ipa-acme-dns.html for details.)
|
|
|
|
* dns-01 challenge with mod_md (see
|
|
|
|
https://httpd.apache.org/docs/current/mod/mod_md.html#mdchallengedns01)
|
|
|
|
|
|
|
|
Things that are not implmented/supported yet, but may be in future:
|
|
|
|
|
|
|
|
* IP address SAN
|
|
|
|
* tls-alpn-01 challenge
|
|
|
|
* Other clients or service scenarios
|
2020-06-04 00:59:27 -05:00
|
|
|
|
2020-06-10 23:33:08 -05:00
|
|
|
"""
|
2020-06-04 00:59:27 -05:00
|
|
|
num_replicas = 0
|
|
|
|
num_clients = 1
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def install(cls, mh):
|
|
|
|
# cache the acme service uri
|
|
|
|
acme_host = f'{IPA_CA_RECORD}.{cls.master.domain.name}'
|
|
|
|
cls.acme_server = f'https://{acme_host}/acme/directory'
|
|
|
|
|
|
|
|
# install packages before client install in case of IPA DNS problems
|
|
|
|
if not skip_certbot_tests:
|
|
|
|
cls.clients[0].run_command(['dnf', 'install', '-y', 'certbot'])
|
2020-06-10 23:33:08 -05:00
|
|
|
if not skip_mod_md_tests:
|
|
|
|
cls.clients[0].run_command(['dnf', 'install', '-y', 'mod_md'])
|
2020-06-04 00:59:27 -05:00
|
|
|
|
|
|
|
tasks.install_master(cls.master, setup_dns=True)
|
|
|
|
|
|
|
|
tasks.install_client(cls.master, cls.clients[0])
|
|
|
|
tasks.config_host_resolvconf_with_master_data(
|
|
|
|
cls.master, cls.clients[0]
|
|
|
|
)
|
|
|
|
|
2020-06-24 23:51:32 -05:00
|
|
|
#######
|
|
|
|
# kinit
|
|
|
|
#######
|
|
|
|
|
|
|
|
def test_kinit_master(self):
|
|
|
|
# Some tests require executing ipa commands, e.g. to
|
|
|
|
# check revocation status or add/remove DNS entries.
|
|
|
|
# Preemptively kinit as admin on the master.
|
|
|
|
tasks.kinit_admin(self.master)
|
|
|
|
|
2020-06-04 00:59:27 -05:00
|
|
|
#####################
|
|
|
|
# Enable ACME service
|
|
|
|
#####################
|
|
|
|
|
|
|
|
def test_acme_service_not_yet_enabled(self):
|
|
|
|
# --fail makes curl exit code 22 when response status >= 400.
|
|
|
|
# ACME service should return 503 because it was not enabled yet.
|
|
|
|
self.clients[0].run_command(
|
|
|
|
['curl', '--fail', self.acme_server],
|
|
|
|
ok_returncode=22,
|
|
|
|
)
|
|
|
|
|
|
|
|
def test_enable_acme_service(self):
|
|
|
|
self.master.run_command(['ipa-acme-manage', 'enable'])
|
|
|
|
|
|
|
|
# wait a short time for Dogtag ACME service to observe config
|
|
|
|
# change and reconfigure itself to service requests
|
|
|
|
exc = None
|
|
|
|
for _i in range(5):
|
|
|
|
time.sleep(2)
|
|
|
|
try:
|
|
|
|
self.clients[0].run_command(
|
|
|
|
['curl', '--fail', self.acme_server])
|
|
|
|
break
|
|
|
|
except Exception as e:
|
|
|
|
exc = e
|
|
|
|
else:
|
|
|
|
raise exc
|
|
|
|
|
|
|
|
###############
|
|
|
|
# Certbot tests
|
|
|
|
###############
|
|
|
|
|
|
|
|
@pytest.mark.skipif(skip_certbot_tests, reason='certbot not available')
|
|
|
|
def test_certbot_register(self):
|
|
|
|
# service is enabled; registration should succeed
|
|
|
|
self.clients[0].run_command(
|
|
|
|
[
|
|
|
|
'certbot',
|
|
|
|
'--server', self.acme_server,
|
|
|
|
'register',
|
|
|
|
'-m', 'nobody@example.test',
|
|
|
|
'--agree-tos',
|
|
|
|
'--no-eff-email',
|
|
|
|
],
|
|
|
|
)
|
|
|
|
|
|
|
|
@pytest.mark.skipif(skip_certbot_tests, reason='certbot not available')
|
|
|
|
def test_certbot_certonly_standalone(self):
|
|
|
|
# Get a cert from ACME service using HTTP challenge and Certbot's
|
|
|
|
# standalone HTTP server mode
|
|
|
|
self.clients[0].run_command(
|
|
|
|
[
|
|
|
|
'certbot',
|
|
|
|
'--server', self.acme_server,
|
|
|
|
'certonly',
|
|
|
|
'--domain', self.clients[0].hostname,
|
|
|
|
'--standalone',
|
|
|
|
],
|
|
|
|
)
|
|
|
|
|
2020-06-24 23:51:32 -05:00
|
|
|
@pytest.mark.skipif(skip_certbot_tests, reason='certbot not available')
|
|
|
|
def test_certbot_revoke(self):
|
|
|
|
# Assume previous certonly operation succeeded.
|
|
|
|
# Read certificate to learn serial number.
|
|
|
|
cert_path = \
|
|
|
|
f'/etc/letsencrypt/live/{self.clients[0].hostname}/cert.pem'
|
|
|
|
data = self.clients[0].get_file_contents(cert_path)
|
|
|
|
cert = x509.load_pem_x509_certificate(data, backend=default_backend())
|
|
|
|
|
|
|
|
# revoke cert via ACME
|
|
|
|
self.clients[0].run_command(
|
|
|
|
[
|
|
|
|
'certbot',
|
|
|
|
'--server', self.acme_server,
|
|
|
|
'revoke',
|
|
|
|
'--cert-name', self.clients[0].hostname,
|
|
|
|
'--delete-after-revoke',
|
|
|
|
],
|
|
|
|
)
|
|
|
|
|
|
|
|
# check cert is revoked (kinit already performed)
|
|
|
|
result = self.master.run_command(
|
|
|
|
['ipa', 'cert-show', str(cert.serial_number), '--raw']
|
|
|
|
)
|
|
|
|
assert 'revocation_reason:' in result.stdout_text
|
|
|
|
|
2020-06-04 00:59:27 -05:00
|
|
|
##############
|
|
|
|
# mod_md tests
|
|
|
|
##############
|
|
|
|
|
2020-06-10 23:33:08 -05:00
|
|
|
@pytest.mark.skipif(skip_mod_md_tests, reason='mod_md too old')
|
|
|
|
def test_mod_md(self):
|
|
|
|
# write config
|
|
|
|
self.clients[0].run_command(['mkdir', '-p', '/etc/httpd/conf.d'])
|
|
|
|
self.clients[0].put_file_contents(
|
|
|
|
'/etc/httpd/conf.d/md.conf',
|
|
|
|
'\n'.join([
|
|
|
|
f'MDCertificateAuthority {self.acme_server}',
|
|
|
|
'MDCertificateAgreement accepted',
|
|
|
|
f'MDomain {self.clients[0].hostname}',
|
|
|
|
'<VirtualHost *:443>',
|
|
|
|
f' ServerName {self.clients[0].hostname}',
|
|
|
|
' SSLEngine on',
|
|
|
|
'</VirtualHost>\n',
|
|
|
|
]),
|
|
|
|
)
|
|
|
|
|
|
|
|
# To check for successful cert issuance means knowing how mod_md
|
|
|
|
# stores certificates, or looking for specific log messages.
|
|
|
|
# If the thing we are inspecting changes, the test will break.
|
|
|
|
# So I prefer a conservative sleep.
|
|
|
|
#
|
|
|
|
self.clients[0].run_command(['systemctl', 'restart', 'httpd'])
|
|
|
|
time.sleep(15)
|
|
|
|
|
|
|
|
# We expect mod_md has acquired the certificate by now.
|
|
|
|
# Perform a graceful restart to begin using the cert.
|
|
|
|
# (If mod_md ever learns to start using newly acquired
|
|
|
|
# certificates /without/ the second restart, then both
|
|
|
|
# of these sleeps can be replaced by "loop until good".)
|
|
|
|
#
|
|
|
|
self.clients[0].run_command(['systemctl', 'reload', 'httpd'])
|
|
|
|
time.sleep(3)
|
|
|
|
|
|
|
|
# HTTPS request from server to client (should succeed)
|
|
|
|
self.master.run_command(
|
|
|
|
['curl', f'https://{self.clients[0].hostname}'])
|
2020-06-04 00:59:27 -05:00
|
|
|
|
|
|
|
######################
|
|
|
|
# Disable ACME service
|
|
|
|
######################
|
|
|
|
|
|
|
|
def test_disable_acme_service(self):
|
|
|
|
"""
|
|
|
|
Disable ACME service again, and observe that it no longer services
|
|
|
|
requests.
|
|
|
|
|
|
|
|
"""
|
|
|
|
self.master.run_command(['ipa-acme-manage', 'disable'])
|
|
|
|
|
|
|
|
# wait a short time for Dogtag ACME service to observe config
|
|
|
|
# change and reconfigure itself to no longer service requests
|
|
|
|
time.sleep(3)
|
|
|
|
|
|
|
|
# should fail now
|
|
|
|
self.clients[0].run_command(
|
|
|
|
['curl', '--fail', self.acme_server],
|
|
|
|
ok_returncode=22,
|
|
|
|
)
|