Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add test to rotate mysql service user password [Yoga backport] #1134

Merged
merged 7 commits into from
Sep 26, 2023
111 changes: 111 additions & 0 deletions zaza/openstack/charm_tests/mysql/tests.py
Original file line number Diff line number Diff line change
Expand Up @@ -14,12 +14,14 @@

"""MySQL/Percona Cluster Testing."""

import configparser
import json
import logging
import os
import re
import tempfile
import tenacity
import yaml

import zaza.charm_lifecycle.utils as lifecycle_utils
import zaza.model
Expand Down Expand Up @@ -549,6 +551,115 @@ def test_120_set_cluster_option(self):
logging.info("Passed set cluster option action test.")


class MySQLInnoDBClusterRotatePasswordTests(MySQLCommonTests):
"""Mysql-innodb-cluster charm tests.

Note: The restart on changed and pause/resume tests also validate the
changing of the R/W primary. On each mysqld shutodown a new R/W primary is
elected automatically by MySQL.
"""

@classmethod
def setUpClass(cls):
"""Run class setup for running mysql-innodb-cluster tests."""
super().setUpClass()
cls.application = "mysql-innodb-cluster"

def test_rotate_keystone_service_user_password(self):
"""Verify action used to rotate a service user (keystone) password."""
KEYSTONE_APP = 'keystone'
KEYSTONE_PASSWD_KEY = "mysql-keystone.passwd"
KEYSTONE_CONF_FILE = '/etc/keystone/keystone.conf'

def _get_password_from_keystone_leader():
conf = zaza.model.file_contents(
'keystone/leader', KEYSTONE_CONF_FILE)
config = configparser.ConfigParser()
config.read_string(conf)
connection_info = config['database']['connection']
match = re.match(r"^mysql.*keystone:(.+)@", connection_info)
if match:
return match[1]
self.fail("Couldn't find mysql password in {}"
.format(connection_info))

# only do the test if keystone is in the model
applications = zaza.model.sync_deployed(self.model_name)
if KEYSTONE_APP not in applications:
self.skipTest(
'{} is not deployed, so not doing password rotation'
.format(KEYSTONE_APP))

# get the users via the 'list-service-usernames' action.
logging.info(
"Getting usernames from mysql that can have password rotated.")
action = zaza.model.run_action_on_leader(
self.application,
'list-service-usernames',
action_params={}
)
usernames = yaml.safe_load(action.data['results']['usernames'])
logging.info("... usernames: %s", usernames)
self.assertIn('keystone', usernames)

# grab the password for keystone from the leader / to verify the change
old_keystone_passwd_on_mysql = juju_utils.leader_get(
self.application_name, KEYSTONE_PASSWD_KEY).strip()
old_keystone_passwd_conf = _get_password_from_keystone_leader()

# verify that keystone is working.
admin_keystone_session = (
openstack_utils.get_overcloud_keystone_session())
keystone_client = openstack_utils.get_keystone_session_client(
admin_keystone_session)
keystone_client.users.list()

# now rotate the password for keystone
# run the action to rotate the password.
logging.info("Rotating password for keystone in mysql.")
zaza.model.run_action_on_leader(
self.application_name,
'rotate-service-user-password',
action_params={'service-user': 'keystone'},
)

# let everything settle.
logging.info("Waiting for model to settle.")
zaza.model.wait_for_agent_status()
zaza.model.block_until_all_units_idle()

# verify that the password has changed.
# Due to the async-ness of the whole model and when the various hooks
# will fire between mysql-innodb-cluster, the mysql-router and
# keystone, so we retry a reasonable time to wait for everything to
# propagate through.
for attempt in tenacity.Retrying(
reraise=True,
wait=tenacity.wait_fixed(30),
stop=tenacity.stop_after_attempt(20), # wait for max 10m
):
with attempt:
new_keystone_passwd_on_mysql = juju_utils.leader_get(
self.application_name, KEYSTONE_PASSWD_KEY).strip()
new_keystone_passwd_conf = _get_password_from_keystone_leader()
self.assertNotEqual(old_keystone_passwd_on_mysql,
new_keystone_passwd_on_mysql)
self.assertNotEqual(old_keystone_passwd_conf,
new_keystone_passwd_conf)
self.assertEqual(new_keystone_passwd_on_mysql,
new_keystone_passwd_conf)

# really wait for keystone to finish it's thing
zaza.model.block_until_all_units_idle()

# finally, verify that keystone is still working.
admin_keystone_session = (
openstack_utils.get_overcloud_keystone_session())
keystone_client = openstack_utils.get_keystone_session_client(
admin_keystone_session)
keystone_client.users.list()


class MySQLInnoDBClusterColdStartTest(MySQLBaseTest):
"""Percona Cluster cold start tests."""

Expand Down
Loading