Fix target_helper config for >= Wallaby

Starting with the release of Wallaby, the 'tgtadm' package is
now the default target helper, which doesn't work for cinder-lvm.
As such, we need to explicitly set the new package to 'lioadm'.

Closes-Bug: #1949074
Change-Id: I4d88d1e30d8b2b133bd575a76d0dc5e566415cbc
This commit is contained in:
Luciano Lo Giudice 2022-02-09 12:18:01 -03:00 committed by Erlon R. Cruz
parent bd2d061f69
commit 3144ea032b
3 changed files with 115 additions and 32 deletions

View File

@ -17,7 +17,7 @@ import subprocess
import socket import socket
import charms_openstack.charm import charms_openstack.charm
import charmhelpers.core.hookenv as ch_hookenv # noqa import charmhelpers.core.hookenv as ch_hookenv
from charmhelpers.core.strutils import ( from charmhelpers.core.strutils import (
bytes_from_string, bytes_from_string,
@ -115,10 +115,10 @@ def configure_lvm_storage(block_devices, volume_group, overwrite=False,
volume group even if logical volumes are allocated volume group even if logical volumes are allocated
on them. Overrides 'remove_missing' if set. on them. Overrides 'remove_missing' if set.
''' '''
juju_log('LVM info before preparation') ch_hookenv.log('LVM info before preparation')
log_lvm_info() log_lvm_info()
juju_log('block_devices: {}'.format(','.join(block_devices))) ch_hookenv.log('block_devices: {}'.format(','.join(block_devices)))
devices = [] devices = []
for block_device in block_devices: for block_device in block_devices:
@ -131,7 +131,7 @@ def configure_lvm_storage(block_devices, volume_group, overwrite=False,
lo_device = ensure_loopback_device(block_device, str(size)) lo_device = ensure_loopback_device(block_device, str(size))
devices.append(lo_device) devices.append(lo_device)
juju_log('devices: {}'.format(','.join(devices))) ch_hookenv.log('devices: {}'.format(','.join(devices)))
vg_found = False vg_found = False
new_devices = [] new_devices = []
@ -148,13 +148,13 @@ def configure_lvm_storage(block_devices, volume_group, overwrite=False,
new_devices.append(device) new_devices.append(device)
else: else:
# Mark vg as found # Mark vg as found
juju_log('Found volume-group already created on {}'.format( ch_hookenv.log('Found volume-group already created on {}'.format(
device)) device))
vg_found = True vg_found = True
juju_log('new_devices: {}'.format(','.join(new_devices))) ch_hookenv.log('new_devices: {}'.format(','.join(new_devices)))
juju_log('LVM info mid preparation') ch_hookenv.log('LVM info mid preparation')
log_lvm_info() log_lvm_info()
if not vg_found and new_devices: if not vg_found and new_devices:
@ -173,8 +173,8 @@ def configure_lvm_storage(block_devices, volume_group, overwrite=False,
elif remove_missing: elif remove_missing:
reduce_lvm_volume_group_missing(volume_group) reduce_lvm_volume_group_missing(volume_group)
except subprocess.CalledProcessError as e: except subprocess.CalledProcessError as e:
juju_log("reduce_lvm_volume_group_missing() didn't complete." ch_hookenv.log("reduce_lvm_volume_group_missing() didn't complete."
" LVM may not be fully configured yet. Error was: '{}'." " LVM may not be fully configured yet. Error was: '{}'"
.format(str(e))) .format(str(e)))
if new_devices: if new_devices:
@ -183,18 +183,18 @@ def configure_lvm_storage(block_devices, volume_group, overwrite=False,
extend_lvm_volume_group(volume_group, new_device) extend_lvm_volume_group(volume_group, new_device)
thin_pools = list_thin_logical_volume_pools(path_mode=True) thin_pools = list_thin_logical_volume_pools(path_mode=True)
if not thin_pools: if not thin_pools:
juju_log("No thin pools found") ch_hookenv.log("No thin pools found")
elif len(thin_pools) == 1: elif len(thin_pools) == 1:
juju_log("Thin pool {} found, extending with {}".format( ch_hookenv.log("Thin pool {} found, extending with {}".format(
thin_pools[0], thin_pools[0],
new_device)) new_device))
extend_logical_volume_by_device(thin_pools[0], new_device) extend_logical_volume_by_device(thin_pools[0], new_device)
else: else:
juju_log("Multiple thin pools ({}) found, " msg = ("Multiple thin pools ({}) found, skipping auto "
"skipping auto extending with {}".format( "extending with {}").format(','.join(thin_pools),
','.join(thin_pools), new_device)
new_device)) ch_hookenv.log(msg)
juju_log('LVM info after preparation') ch_hookenv.log('LVM info after preparation')
log_lvm_info() log_lvm_info()
@ -261,26 +261,23 @@ def log_lvm_info():
"""Log some useful information about how LVM is setup.""" """Log some useful information about how LVM is setup."""
try: try:
pvscan_output = subprocess.check_output(['pvscan']).decode('UTF-8') pvscan_output = subprocess.check_output(['pvscan']).decode('UTF-8')
juju_log('pvscan:\n{}'.format(pvscan_output)) ch_hookenv.log('pvscan:\n{}'.format(pvscan_output))
except subprocess.CalledProcessError: except subprocess.CalledProcessError:
juju_log('pvscan did not complete successfully; may not be setup yet') ch_hookenv.log('pvscan did not complete successfully; may not be '
'setup yet')
try: try:
vgscan_output = subprocess.check_output(['vgscan']).decode('UTF-8') vgscan_output = subprocess.check_output(['vgscan']).decode('UTF-8')
juju_log('vgscan:\n{}'.format(vgscan_output)) ch_hookenv.log('vgscan:\n{}'.format(vgscan_output))
except subprocess.CalledProcessError: except subprocess.CalledProcessError:
juju_log('vgscan did not complete successfully') ch_hookenv.log('vgscan did not complete successfully')
def juju_log(msg):
ch_hookenv.log(msg, ch_hookenv.INFO)
def prepare_volume(device): def prepare_volume(device):
juju_log("prepare_volume: {}".format(device)) ch_hookenv.log("prepare_volume: {}".format(device))
clean_storage(device) clean_storage(device)
create_lvm_physical_volume(device) create_lvm_physical_volume(device)
juju_log("prepared volume: {}".format(device)) ch_hookenv.log("prepared volume: {}".format(device))
def has_partition_table(block_device): def has_partition_table(block_device):
@ -300,8 +297,8 @@ def clean_storage(block_device):
''' '''
for mp, d in mounts(): for mp, d in mounts():
if d == block_device: if d == block_device:
juju_log('clean_storage(): Found %s mounted @ %s, unmounting.' % ch_hookenv.log('clean_storage(): Found %s mounted @ %s, '
(d, mp)) 'unmounting.' % (d, mp))
umount(mp, persist=True) umount(mp, persist=True)
if is_lvm_physical_volume(block_device): if is_lvm_physical_volume(block_device):
@ -381,3 +378,37 @@ class CinderLVMCharm(
driver_options.append(tup) driver_options.append(tup)
return driver_options return driver_options
class CinderLVMCharmWallaby(CinderLVMCharm):
release = 'wallaby'
def cinder_configuration(self):
"""Add some extra configuration for Wallaby and on.
:return: list of tuples with the configuration options for this driver.
"""
opts = super(CinderLVMCharmWallaby, self).cinder_configuration()
# From Wallaby and on, the default target_helper that comes with the
# Ubuntu cinder package is not tgtadm (default for Cinder), and
# therefore, not installed as dependency anymore. Users should either
# install it by hand or set the config to use lioadm.
dict_opts = dict(opts)
helper = dict_opts.get('target_helper')
if helper:
if helper != 'lioadm':
ch_hookenv.log("Configured target helper '{}' might not be "
"installed as the default target_helper is "
"lioadm.".format(helper),
level=ch_hookenv.WARNING)
return opts
else:
tgt_helper = ch_hookenv.leader_get('target-helper')
tgt_port = ch_hookenv.leader_get('target-port')
if tgt_helper:
opts.append(('target_helper', tgt_helper))
if tgt_port:
opts.append(('target_port', tgt_port))
return opts

View File

@ -15,6 +15,14 @@
import charms_openstack.charm import charms_openstack.charm
import charms.reactive import charms.reactive
from charmhelpers.contrib.openstack.utils import os_release
from charmhelpers.core.hookenv import (
leader_get,
leader_set,
log,
)
from charmhelpers.fetch.ubuntu import (get_installed_version,
apt_mark)
# This charm's library contains all of the handler code associated with # This charm's library contains all of the handler code associated with
# this charm -- we will use the auto-discovery feature of charms.openstack # this charm -- we will use the auto-discovery feature of charms.openstack
# to get the definitions for the charm. # to get the definitions for the charm.
@ -33,3 +41,31 @@ charms_openstack.charm.use_defaults(
def reinstall(): def reinstall():
with charms_openstack.charm.provide_charm_instance() as charm: with charms_openstack.charm.provide_charm_instance() as charm:
charm.install() charm.install()
@charms.reactive.when('leadership.is_leader')
@charms.reactive.when_any('charm.installed', 'upgrade-charm',
'storage-backend.connected')
def set_target_helper():
log("Setting target-helper: {}".format(leader_get('target-helper')),
"DEBUG")
if leader_get('target-helper') is None:
# For deployments upgrading from Victoria, we set the target helper
# to the legacy default of tgtadm. For new deployments, lioadm. See
# LP#1949074 for more details.
if os_release('cinder-common') <= 'victoria':
leader_set(settings={"target-helper": "tgtadm",
"target-port": 3260})
# Mark tgt as manual to prevent it from being removed. Since
# Wallaby tgt is no longer a dependency of cinder-volume package
# but some backends will still use it.
if get_installed_version("tgt"):
apt_mark("tgt", "manual")
else:
leader_set(settings={"target-helper": "lioadm",
"target-port": 3260})
log("Setting target-helper/port: {}/{}".format(
leader_get('target-helper'), leader_get('target-port')),
"DEBUG")

View File

@ -141,6 +141,8 @@ class TestCinderLVMCharm(test_utils.PatchHelper):
return self._config.get(key) return self._config.get(key)
self.patch_object(charmhelpers.core.hookenv, 'config') self.patch_object(charmhelpers.core.hookenv, 'config')
self.patch_object(charmhelpers.core.hookenv, 'leader_get',
return_value='lioadm')
self.patch_object(cinder_lvm, 'mounts') self.patch_object(cinder_lvm, 'mounts')
self.patch_object(cinder_lvm, 'umount') self.patch_object(cinder_lvm, 'umount')
self.patch_object(cinder_lvm, 'is_block_device') self.patch_object(cinder_lvm, 'is_block_device')
@ -187,7 +189,7 @@ class TestCinderLVMCharm(test_utils.PatchHelper):
def _patch_config_and_charm(self, config): def _patch_config_and_charm(self, config):
self._config.update(config) self._config.update(config)
return cinder_lvm.CinderLVMCharm() return cinder_lvm.CinderLVMCharmWallaby()
def test_cinder_base(self): def test_cinder_base(self):
charm = self._patch_config_and_charm({}) charm = self._patch_config_and_charm({})
@ -203,8 +205,10 @@ class TestCinderLVMCharm(test_utils.PatchHelper):
charm = self._patch_config_and_charm( charm = self._patch_config_and_charm(
{'a': 'b', 'config-flags': 'val=3'}) {'a': 'b', 'config-flags': 'val=3'})
config = charm.cinder_configuration() config = charm.cinder_configuration()
self.assertEqual(config[-1][1], '3') cfg_dict = dict()
self.assertNotIn('a', list(x[0] for x in config)) cfg_dict.update(dict(config))
self.assertEqual('3', cfg_dict.get('val'))
self.assertIsNone(cfg_dict.get('a'))
def test_cinder_vg_and_backend(self): def test_cinder_vg_and_backend(self):
base = {'volume-group': 'test-vg', 'volume-backend-name': 'test-bn'} base = {'volume-group': 'test-vg', 'volume-backend-name': 'test-bn'}
@ -246,3 +250,15 @@ class TestCinderLVMCharm(test_utils.PatchHelper):
dev = self.LVM.find_device(loop_dev) dev = self.LVM.find_device(loop_dev)
self.assertTrue(dev) self.assertTrue(dev)
self.assertEqual(dev.size, '100') self.assertEqual(dev.size, '100')
def test_cinder_wallaby_upgrade(self):
charm = self._patch_config_and_charm({})
charm.release = "wallaby"
config2 = charm.cinder_configuration()
self.assertIn(('target_helper', 'lioadm'), config2)
charm = self._patch_config_and_charm({
'config-flags': 'target_helper=tgtadm'})
config3 = charm.cinder_configuration()
self.assertIn(('target_helper', 'tgtadm'), config3)