Add TLS checks
Add checks that TLS data has been supplied and refactor the charms checks to make them more easy to extend. Also a drive by fix to minimum_supported to fix the comparison of ceph versions. Change-Id: I2f27529da53c7d482d64dff4c9aaf3b0a08369b4
This commit is contained in:
parent
56b48dcd37
commit
ee0e4e8c49
@ -1,4 +1,5 @@
|
|||||||
ops >= 1.2.0
|
ops >= 1.2.0
|
||||||
|
tenacity
|
||||||
git+https://github.com/openstack/charms.ceph#egg=charms_ceph
|
git+https://github.com/openstack/charms.ceph#egg=charms_ceph
|
||||||
git+https://opendev.org/openstack/charm-ops-openstack#egg=ops_openstack
|
git+https://opendev.org/openstack/charm-ops-openstack#egg=ops_openstack
|
||||||
#git+https://opendev.org/openstack/charm-ops-interface-tls-certificates#egg=interface_tls_certificates
|
#git+https://opendev.org/openstack/charm-ops-interface-tls-certificates#egg=interface_tls_certificates
|
||||||
|
54
src/charm.py
54
src/charm.py
@ -22,6 +22,7 @@ import secrets
|
|||||||
import socket
|
import socket
|
||||||
import string
|
import string
|
||||||
import subprocess
|
import subprocess
|
||||||
|
import tenacity
|
||||||
import ops_openstack.plugins.classes
|
import ops_openstack.plugins.classes
|
||||||
import interface_dashboard
|
import interface_dashboard
|
||||||
import interface_api_endpoints
|
import interface_api_endpoints
|
||||||
@ -67,7 +68,7 @@ class CephDashboardCharm(ops_openstack.core.OSBaseCharm):
|
|||||||
|
|
||||||
def minimum_supported(self, supported_version: str) -> bool:
|
def minimum_supported(self, supported_version: str) -> bool:
|
||||||
"""Check if installed Ceph release is >= to supported_version"""
|
"""Check if installed Ceph release is >= to supported_version"""
|
||||||
return ch_host.cmp_pkgrevno('ceph-common', supported_version) < 1
|
return ch_host.cmp_pkgrevno('ceph-common', supported_version) >= 0
|
||||||
|
|
||||||
def convert_option(self, value: Union[bool, str, int]) -> List[str]:
|
def convert_option(self, value: Union[bool, str, int]) -> List[str]:
|
||||||
"""Convert a value to the corresponding value part of the ceph
|
"""Convert a value to the corresponding value part of the ceph
|
||||||
@ -190,20 +191,46 @@ class CephDashboardCharm(ops_openstack.core.OSBaseCharm):
|
|||||||
sans.append(self.config.get('public-hostname'))
|
sans.append(self.config.get('public-hostname'))
|
||||||
self.ca_client.request_server_certificate(socket.getfqdn(), sans)
|
self.ca_client.request_server_certificate(socket.getfqdn(), sans)
|
||||||
|
|
||||||
|
def _check_for_certs(self) -> bool:
|
||||||
|
"""Check that charm has TLS data it needs"""
|
||||||
|
# Check charm config for TLS data
|
||||||
|
key, cert, _ = self._get_tls_from_config()
|
||||||
|
if key and cert:
|
||||||
|
return True
|
||||||
|
# Check relation for TLS data
|
||||||
|
try:
|
||||||
|
self.ca_client.server_key
|
||||||
|
return True
|
||||||
|
except ca_client.CAClientError:
|
||||||
|
return False
|
||||||
|
|
||||||
|
def _check_dashboard_responding(self) -> bool:
|
||||||
|
"""Check the dashboard port is open"""
|
||||||
|
|
||||||
|
@tenacity.retry(wait=tenacity.wait_fixed(2),
|
||||||
|
stop=tenacity.stop_after_attempt(30), reraise=True)
|
||||||
|
def _check_port(ip, port):
|
||||||
|
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
|
||||||
|
result = sock.connect_ex((ip, port))
|
||||||
|
assert result == 0
|
||||||
|
|
||||||
|
try:
|
||||||
|
_check_port(self._get_bind_ip(), self.TLS_PORT)
|
||||||
|
return True
|
||||||
|
except AssertionError:
|
||||||
|
return False
|
||||||
|
|
||||||
def check_dashboard(self) -> StatusBase:
|
def check_dashboard(self) -> StatusBase:
|
||||||
"""Check status of dashboard"""
|
"""Check status of dashboard"""
|
||||||
self._stored.is_started = ceph_utils.is_dashboard_enabled()
|
checks = [
|
||||||
if self._stored.is_started:
|
(ceph_utils.is_dashboard_enabled, 'Dashboard is not enabled'),
|
||||||
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
|
(self._check_for_certs, ('No certificates found. Please add a '
|
||||||
result = sock.connect_ex((self._get_bind_ip(), self.TLS_PORT))
|
'certifcates relation or provide via '
|
||||||
if result == 0:
|
'charm config')),
|
||||||
return ActiveStatus()
|
(self._check_dashboard_responding, 'Dashboard not responding')]
|
||||||
else:
|
for check_f, msg in checks:
|
||||||
return BlockedStatus(
|
if not check_f():
|
||||||
'Dashboard not responding')
|
return BlockedStatus(msg)
|
||||||
else:
|
|
||||||
return BlockedStatus(
|
|
||||||
'Dashboard is not enabled')
|
|
||||||
return ActiveStatus()
|
return ActiveStatus()
|
||||||
|
|
||||||
def kick_dashboard(self) -> None:
|
def kick_dashboard(self) -> None:
|
||||||
@ -251,6 +278,7 @@ class CephDashboardCharm(ops_openstack.core.OSBaseCharm):
|
|||||||
'mgr/dashboard/{hostname}/server_addr'.format(
|
'mgr/dashboard/{hostname}/server_addr'.format(
|
||||||
hostname=socket.gethostname()),
|
hostname=socket.gethostname()),
|
||||||
str(self._get_bind_ip()))
|
str(self._get_bind_ip()))
|
||||||
|
self._stored.is_started = True
|
||||||
self.update_status()
|
self.update_status()
|
||||||
|
|
||||||
def _get_bind_ip(self) -> str:
|
def _get_bind_ip(self) -> str:
|
||||||
|
@ -13,4 +13,4 @@ target_deploy_status:
|
|||||||
workload-status-message: Vault needs to be initialized
|
workload-status-message: Vault needs to be initialized
|
||||||
ceph-dashboard:
|
ceph-dashboard:
|
||||||
workload-status: blocked
|
workload-status: blocked
|
||||||
workload-status-message-prefix: Dashboard not responding
|
workload-status-message-prefix: No certificates found
|
||||||
|
@ -17,3 +17,8 @@ import mock
|
|||||||
|
|
||||||
# Mock out secrets to make py35 happy.
|
# Mock out secrets to make py35 happy.
|
||||||
sys.modules['secrets'] = mock.MagicMock()
|
sys.modules['secrets'] = mock.MagicMock()
|
||||||
|
|
||||||
|
# Tenacity decorators need to be mocked before import
|
||||||
|
tenacity = mock.MagicMock()
|
||||||
|
tenacity.retry.side_effect = lambda *args, **kwargs: lambda x: x
|
||||||
|
sys.modules['tenacity'] = tenacity
|
||||||
|
@ -279,6 +279,15 @@ class TestCephDashboardCharmBase(CharmTestCase):
|
|||||||
socket_mock.connect_ex.return_value = 0
|
socket_mock.connect_ex.return_value = 0
|
||||||
self.ceph_utils.is_dashboard_enabled.return_value = True
|
self.ceph_utils.is_dashboard_enabled.return_value = True
|
||||||
self.harness.begin()
|
self.harness.begin()
|
||||||
|
self.assertEqual(
|
||||||
|
self.harness.charm.check_dashboard(),
|
||||||
|
BlockedStatus('No certificates found. Please add a certifcates '
|
||||||
|
'relation or provide via charm config'))
|
||||||
|
self.harness.update_config(
|
||||||
|
key_values={
|
||||||
|
'ssl_key': base64.b64encode(TEST_KEY.encode("utf-8")),
|
||||||
|
'ssl_cert': base64.b64encode(TEST_CERT.encode("utf-8")),
|
||||||
|
'ssl_ca': base64.b64encode(TEST_CA.encode("utf-8"))})
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
self.harness.charm.check_dashboard(),
|
self.harness.charm.check_dashboard(),
|
||||||
ActiveStatus())
|
ActiveStatus())
|
||||||
|
Loading…
Reference in New Issue
Block a user