Merge "Move libvirt NFS volume driver impl into volume.py"

This commit is contained in:
Jenkins 2013-02-06 02:39:05 +00:00 committed by Gerrit Code Review
commit 4c61423b91
4 changed files with 86 additions and 80 deletions

View File

@ -61,7 +61,6 @@ from nova.virt.libvirt import firewall
from nova.virt.libvirt import imagebackend from nova.virt.libvirt import imagebackend
from nova.virt.libvirt import utils as libvirt_utils from nova.virt.libvirt import utils as libvirt_utils
from nova.virt.libvirt import volume from nova.virt.libvirt import volume
from nova.virt.libvirt import volume_nfs
try: try:
@ -412,7 +411,7 @@ class LibvirtVolumeTestCase(test.TestCase):
mnt_base = '/mnt' mnt_base = '/mnt'
self.flags(nfs_mount_point_base=mnt_base) self.flags(nfs_mount_point_base=mnt_base)
libvirt_driver = volume_nfs.NfsVolumeDriver(self.fake_conn) libvirt_driver = volume.LibvirtNFSVolumeDriver(self.fake_conn)
export_string = '192.168.1.1:/nfs/share1' export_string = '192.168.1.1:/nfs/share1'
name = 'volume-00001' name = 'volume-00001'
export_mnt_base = os.path.join(mnt_base, export_mnt_base = os.path.join(mnt_base,

View File

@ -150,7 +150,7 @@ libvirt_opts = [
'fake=nova.virt.libvirt.volume.LibvirtFakeVolumeDriver', 'fake=nova.virt.libvirt.volume.LibvirtFakeVolumeDriver',
'rbd=nova.virt.libvirt.volume.LibvirtNetVolumeDriver', 'rbd=nova.virt.libvirt.volume.LibvirtNetVolumeDriver',
'sheepdog=nova.virt.libvirt.volume.LibvirtNetVolumeDriver', 'sheepdog=nova.virt.libvirt.volume.LibvirtNetVolumeDriver',
'nfs=nova.virt.libvirt.volume_nfs.NfsVolumeDriver' 'nfs=nova.virt.libvirt.volume.LibvirtNFSVolumeDriver'
], ],
help='Libvirt handlers for remote volumes.'), help='Libvirt handlers for remote volumes.'),
cfg.StrOpt('libvirt_disk_prefix', cfg.StrOpt('libvirt_disk_prefix',

View File

@ -17,6 +17,7 @@
"""Volume drivers for libvirt.""" """Volume drivers for libvirt."""
import hashlib
import os import os
import time import time
@ -24,6 +25,7 @@ from nova import exception
from nova.openstack.common import cfg from nova.openstack.common import cfg
from nova.openstack.common import lockutils from nova.openstack.common import lockutils
from nova.openstack.common import log as logging from nova.openstack.common import log as logging
from nova import paths
from nova import utils from nova import utils
from nova.virt.libvirt import config as vconfig from nova.virt.libvirt import config as vconfig
from nova.virt.libvirt import utils as virtutils from nova.virt.libvirt import utils as virtutils
@ -40,7 +42,14 @@ volume_opts = [
cfg.StrOpt('rbd_secret_uuid', cfg.StrOpt('rbd_secret_uuid',
default=None, default=None,
help='the libvirt uuid of the secret for the rbd_user' help='the libvirt uuid of the secret for the rbd_user'
'volumes') 'volumes'),
cfg.StrOpt('nfs_mount_point_base',
default=paths.state_path_def('mnt'),
help='Dir where the nfs volume is mounted on the compute node'),
cfg.StrOpt('nfs_mount_options',
default=None,
help='Mount options passed to the nfs client. See section '
'of the nfs man page for details'),
] ]
CONF = cfg.CONF CONF = cfg.CONF
@ -246,3 +255,63 @@ class LibvirtISCSIVolumeDriver(LibvirtBaseVolumeDriver):
check_exit_code=[0, 21, 255]) check_exit_code=[0, 21, 255])
self._run_iscsiadm(iscsi_properties, ('--op', 'delete'), self._run_iscsiadm(iscsi_properties, ('--op', 'delete'),
check_exit_code=[0, 21, 255]) check_exit_code=[0, 21, 255])
class LibvirtNFSVolumeDriver(LibvirtBaseVolumeDriver):
"""Class implements libvirt part of volume driver for NFS."""
def __init__(self, connection):
"""Create back-end to nfs."""
super(LibvirtNFSVolumeDriver,
self).__init__(connection, is_block_dev=False)
def connect_volume(self, connection_info, mount_device):
"""Connect the volume. Returns xml for libvirt."""
conf = super(LibvirtNFSVolumeDriver,
self).connect_volume(connection_info, mount_device)
path = self._ensure_mounted(connection_info['data']['export'])
path = os.path.join(path, connection_info['data']['name'])
conf.source_type = 'file'
conf.source_path = path
return conf
def _ensure_mounted(self, nfs_export):
"""
@type nfs_export: string
"""
mount_path = os.path.join(CONF.nfs_mount_point_base,
self.get_hash_str(nfs_export))
self._mount_nfs(mount_path, nfs_export, ensure=True)
return mount_path
def _mount_nfs(self, mount_path, nfs_share, ensure=False):
"""Mount nfs export to mount path."""
if not self._path_exists(mount_path):
utils.execute('mkdir', '-p', mount_path)
# Construct the NFS mount command.
nfs_cmd = ['mount', '-t', 'nfs']
if CONF.nfs_mount_options is not None:
nfs_cmd.extend(['-o', CONF.nfs_mount_options])
nfs_cmd.extend([nfs_share, mount_path])
try:
utils.execute(*nfs_cmd, run_as_root=True)
except exception.ProcessExecutionError as exc:
if ensure and 'already mounted' in exc.message:
LOG.warn(_("%s is already mounted"), nfs_share)
else:
raise
@staticmethod
def get_hash_str(base_str):
"""returns string that represents hash of base_str (in hex format)."""
return hashlib.md5(base_str).hexdigest()
@staticmethod
def _path_exists(path):
"""Check path."""
try:
return utils.execute('stat', path, run_as_root=True)
except exception.ProcessExecutionError:
return False

View File

@ -15,88 +15,26 @@
# License for the specific language governing permissions and limitations # License for the specific language governing permissions and limitations
# under the License. # under the License.
"""Volume driver for using NFS as volumes storage. Nova compute part.""" """Deprecated file, kept for back-compat only. To be removed in Hxxxx."""
import hashlib
import os
from nova import exception
from nova.openstack.common import cfg
from nova.openstack.common import log as logging from nova.openstack.common import log as logging
from nova import paths
from nova import utils
from nova.virt.libvirt import volume from nova.virt.libvirt import volume
LOG = logging.getLogger(__name__) LOG = logging.getLogger(__name__)
volume_opts = [
cfg.StrOpt('nfs_mount_point_base',
default=paths.state_path_def('mnt'),
help='Base dir where nfs expected to be mounted on compute'),
cfg.StrOpt('nfs_mount_options',
default=None,
help='Mount options passed to the nfs client. See section '
'of the nfs man page for details'),
]
CONF = cfg.CONF
CONF.register_opts(volume_opts)
class NfsVolumeDriver(volume.LibvirtNFSVolumeDriver):
"""Deprecated driver for NFS, renamed to LibvirtNFSVolumeDriver
and moved into the main volume.py module. Kept for backwards
compatibility in the Grizzly cycle to give users opportunity
to configure before its removal in the Hxxxx cycle."""
class NfsVolumeDriver(volume.LibvirtBaseVolumeDriver): def __init__(self, *args, **kwargs):
"""Class implements libvirt part of volume driver for NFS."""
def __init__(self, connection):
"""Create back-end to nfs."""
super(NfsVolumeDriver, super(NfsVolumeDriver,
self).__init__(connection, is_block_dev=False) self).__init__(*args, **kwargs)
LOG.deprecated(
def connect_volume(self, connection_info, mount_device): _("The nova.virt.libvirt.volume_nfs.NfsVolumeDriver "
"""Connect the volume. Returns xml for libvirt.""" "class is deprecated and will be removed in the "
conf = super(NfsVolumeDriver, "Hxxxx release. Please update nova.conf so that "
self).connect_volume(connection_info, mount_device) "the 'libvirt_volume_drivers' parameter refers to "
path = self._ensure_mounted(connection_info['data']['export']) "nova.virt.libvirt.volume.LibvirtNFSVolumeDriver."))
path = os.path.join(path, connection_info['data']['name'])
conf.source_type = 'file'
conf.source_path = path
return conf
def _ensure_mounted(self, nfs_export):
"""
@type nfs_export: string
"""
mount_path = os.path.join(CONF.nfs_mount_point_base,
self.get_hash_str(nfs_export))
self._mount_nfs(mount_path, nfs_export, ensure=True)
return mount_path
def _mount_nfs(self, mount_path, nfs_share, ensure=False):
"""Mount nfs export to mount path."""
if not self._path_exists(mount_path):
utils.execute('mkdir', '-p', mount_path)
# Construct the NFS mount command.
nfs_cmd = ['mount', '-t', 'nfs']
if CONF.nfs_mount_options is not None:
nfs_cmd.extend(['-o', CONF.nfs_mount_options])
nfs_cmd.extend([nfs_share, mount_path])
try:
utils.execute(*nfs_cmd, run_as_root=True)
except exception.ProcessExecutionError as exc:
if ensure and 'already mounted' in exc.message:
LOG.warn(_("%s is already mounted"), nfs_share)
else:
raise
@staticmethod
def get_hash_str(base_str):
"""returns string that represents hash of base_str (in hex format)."""
return hashlib.md5(base_str).hexdigest()
@staticmethod
def _path_exists(path):
"""Check path."""
try:
return utils.execute('stat', path, run_as_root=True)
except exception.ProcessExecutionError:
return False