Merge "LVM-nvmet: Use nvmetcli as library instead of CLI"
This commit is contained in:
commit
bc9910f040
cinder
privsep
tests/unit
volume/targets
doc/source
releasenotes/notes
@ -1,41 +0,0 @@
|
||||
# Copyright 2018 Red Hat, Inc
|
||||
# Copyright 2017 Rackspace Australia
|
||||
# Copyright 2018 Michael Still and Aptira
|
||||
#
|
||||
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||
# not use this file except in compliance with the License. You may obtain
|
||||
# a copy of the License at
|
||||
#
|
||||
# http://www.apache.org/licenses/LICENSE-2.0
|
||||
#
|
||||
# Unless required by applicable law or agreed to in writing, software
|
||||
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||
# License for the specific language governing permissions and limitations
|
||||
# under the License.
|
||||
|
||||
"""
|
||||
Helpers for nvmetcli related routines.
|
||||
"""
|
||||
|
||||
from oslo_concurrency import processutils
|
||||
|
||||
import cinder.privsep
|
||||
|
||||
|
||||
@cinder.privsep.sys_admin_pctxt.entrypoint
|
||||
def restore(path):
|
||||
cmd = [
|
||||
'nvmetcli',
|
||||
'restore',
|
||||
path]
|
||||
return processutils.execute(*cmd)
|
||||
|
||||
|
||||
@cinder.privsep.sys_admin_pctxt.entrypoint
|
||||
def save(path):
|
||||
cmd = [
|
||||
'nvmetcli',
|
||||
'save',
|
||||
path]
|
||||
return processutils.execute(*cmd)
|
@ -24,21 +24,6 @@ from cinder import exception
|
||||
import cinder.privsep
|
||||
|
||||
|
||||
@cinder.privsep.sys_admin_pctxt.entrypoint
|
||||
def readfile(path):
|
||||
if not os.path.exists(path):
|
||||
raise exception.FileNotFound(file_path=path)
|
||||
with open(path, 'r') as f:
|
||||
return f.read()
|
||||
|
||||
|
||||
@cinder.privsep.sys_admin_pctxt.entrypoint
|
||||
def removefile(path):
|
||||
if not os.path.exists(path):
|
||||
raise exception.FileNotFound(file_path=path)
|
||||
os.unlink(path)
|
||||
|
||||
|
||||
@cinder.privsep.sys_admin_pctxt.entrypoint
|
||||
def touch(path):
|
||||
if os.path.exists(path):
|
||||
|
209
cinder/privsep/targets/nvmet.py
Normal file
209
cinder/privsep/targets/nvmet.py
Normal file
@ -0,0 +1,209 @@
|
||||
# Copyright 2022 Red Hat, Inc
|
||||
#
|
||||
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||
# not use this file except in compliance with the License. You may obtain
|
||||
# a copy of the License at
|
||||
#
|
||||
# http://www.apache.org/licenses/LICENSE-2.0
|
||||
#
|
||||
# Unless required by applicable law or agreed to in writing, software
|
||||
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||
# License for the specific language governing permissions and limitations
|
||||
# under the License.
|
||||
"""NVMet Python Interface using privsep
|
||||
|
||||
This file adds the privsep support to the nvmet package so it can be easily
|
||||
consumed by Cinder nvmet target.
|
||||
|
||||
It also:
|
||||
|
||||
- Adds some methods to the Root class to be able to get a specific subsystem or
|
||||
port directly without having to go through all the existing ones.
|
||||
|
||||
- Presents the CFSNotFound exception as a NotFound exception which is easier to
|
||||
consume.
|
||||
"""
|
||||
import os
|
||||
|
||||
import nvmet
|
||||
from oslo_log import log as logging
|
||||
|
||||
from cinder import exception
|
||||
from cinder import privsep
|
||||
|
||||
|
||||
LOG = logging.getLogger(__name__)
|
||||
|
||||
|
||||
###################
|
||||
# Helper methods to serialize/deserialize parameters to be sent through privsep
|
||||
# and to do the instance/class calls on the privsep side.
|
||||
|
||||
def serialize(instance):
|
||||
"""Serialize parameters, specially nvmet instances.
|
||||
|
||||
The idea is to be able to pass an nvmet instance to privsep methods, since
|
||||
they are sometimes required as parameters (ie: port.setup) and also to pass
|
||||
the instance where do_privsep_call has to call a specific method.
|
||||
|
||||
Instances are passed as a tuple, with the name of the class as the first
|
||||
element, and in the second element the kwargs necessary to instantiate the
|
||||
instance of that class.
|
||||
|
||||
To differentiate nvmet instances from tuples there is a 'tuple' value that
|
||||
can be passed in the first element of the tuple to differentiate them.
|
||||
|
||||
All other instances as passed unaltered.
|
||||
"""
|
||||
if isinstance(instance, nvmet.Root):
|
||||
return ('Root', {})
|
||||
|
||||
if isinstance(instance, (nvmet.Subsystem, nvmet.Host)):
|
||||
return (type(instance).__name__, {'nqn': instance.nqn,
|
||||
'mode': 'lookup'})
|
||||
|
||||
if isinstance(instance, nvmet.Namespace):
|
||||
return ('Namespace', {'nsid': instance.nsid,
|
||||
'subsystem': serialize(instance.subsystem),
|
||||
'mode': 'lookup'})
|
||||
|
||||
if isinstance(instance, nvmet.Port):
|
||||
return ('Port', {'portid': instance.portid, 'mode': 'lookup'})
|
||||
|
||||
if isinstance(instance, nvmet.Referral):
|
||||
return ('Referral', {'name': instance.name,
|
||||
'port': serialize(instance.port),
|
||||
'mode': 'lookup'})
|
||||
|
||||
if isinstance(instance, nvmet.ANAGroup):
|
||||
return ('ANAGroup', {'grpid': instance.grpid,
|
||||
'port': serialize(instance.port),
|
||||
'mode': 'lookup'})
|
||||
|
||||
if isinstance(instance, tuple):
|
||||
return ('tuple', instance)
|
||||
|
||||
return instance
|
||||
|
||||
|
||||
def deserialize(data):
|
||||
"""Deserialize an instance, specially nvmet instances.
|
||||
|
||||
Reverse operation of the serialize method. Converts an nvmet instance
|
||||
serialized in a tuple into an actual nvmet instance.
|
||||
"""
|
||||
if not isinstance(data, tuple):
|
||||
return data
|
||||
|
||||
cls_name, cls_params = data
|
||||
if cls_name == 'tuple':
|
||||
return cls_params
|
||||
|
||||
# Parameters for the instantiation of the class can be nvmet objects
|
||||
# themselves.
|
||||
params = {name: deserialize(value) for name, value in cls_params.items()}
|
||||
# We don't want the classes from the nvmet method but ours instead
|
||||
instance = getattr(nvmet, cls_name)(**params)
|
||||
return instance
|
||||
|
||||
|
||||
def deserialize_params(args, kwargs):
|
||||
"""Deserialize function arguments using deserialize method."""
|
||||
args = [deserialize(arg) for arg in args]
|
||||
kwargs = {key: deserialize(value) for key, value in kwargs.items()}
|
||||
return args, kwargs
|
||||
|
||||
|
||||
def _nvmet_setup_failure(message):
|
||||
"""Simple error method to use when calling nvmet setup methods."""
|
||||
LOG.error(message)
|
||||
raise exception.CinderException(message)
|
||||
|
||||
|
||||
@privsep.sys_admin_pctxt.entrypoint
|
||||
def do_privsep_call(instance, method_name, *args, **kwargs):
|
||||
"""General privsep method for instance calls.
|
||||
|
||||
Handle privsep method calls by deserializing the instance where we want to
|
||||
call a given method with the deserialized parameters.
|
||||
"""
|
||||
LOG.debug('Calling %s on %s with %s - %s',
|
||||
method_name, instance, args, kwargs)
|
||||
instance = deserialize(instance)
|
||||
method = getattr(instance, method_name)
|
||||
args, kwargs = deserialize_params(args, kwargs)
|
||||
# NOTE: No returning nvmet objects support. If needed add serialization on
|
||||
# the result and deserialization decorator before the entrypoint.
|
||||
return method(*args, **kwargs)
|
||||
|
||||
|
||||
@privsep.sys_admin_pctxt.entrypoint
|
||||
def privsep_setup(cls_name, *args, **kwargs):
|
||||
"""Special privsep method for nvmet setup method calls.
|
||||
|
||||
The setup method is a special case because it's a class method (which
|
||||
privsep cannot handle) and also requires a function for the error handling.
|
||||
|
||||
This method accepts a class name and reconstructs it, then calls the
|
||||
class' setup method passing our own error function.
|
||||
"""
|
||||
LOG.debug('Setup %s with %s - %s', cls_name, args, kwargs)
|
||||
cls = getattr(nvmet, cls_name)
|
||||
args, kwargs = deserialize_params(args, kwargs)
|
||||
kwargs['err_func'] = _nvmet_setup_failure
|
||||
cls.setup(*args, **kwargs)
|
||||
|
||||
|
||||
###################
|
||||
# Classes that don't currently have privsep support
|
||||
|
||||
Namespace = nvmet.Namespace
|
||||
Host = nvmet.Host
|
||||
Referral = nvmet.Referral
|
||||
ANAGroup = nvmet.ANAGroup
|
||||
|
||||
|
||||
###################
|
||||
# Custom classes that divert privileges calls to privsep
|
||||
# Support in these classes is limited to what's needed by the nvmet target.
|
||||
|
||||
# Convenience error class link to nvmet's
|
||||
NotFound = nvmet.nvme.CFSNotFound
|
||||
|
||||
|
||||
class Subsystem(nvmet.Subsystem):
|
||||
def __init__(self, nqn=None, mode='lookup'):
|
||||
super().__init__(nqn=nqn, mode=mode)
|
||||
|
||||
@classmethod
|
||||
def setup(cls, t):
|
||||
privsep_setup(cls.__name__, t)
|
||||
|
||||
def delete(self):
|
||||
do_privsep_call(serialize(self), 'delete')
|
||||
|
||||
|
||||
class Port(nvmet.Port):
|
||||
def __init__(self, portid, mode='lookup'):
|
||||
super().__init__(portid=portid, mode=mode)
|
||||
|
||||
@classmethod
|
||||
def setup(cls, root, n):
|
||||
privsep_setup(cls.__name__, serialize(root), n)
|
||||
|
||||
def add_subsystem(self, nqn):
|
||||
do_privsep_call(serialize(self), 'add_subsystem', nqn)
|
||||
|
||||
def remove_subsystem(self, nqn):
|
||||
do_privsep_call(serialize(self), 'remove_subsystem', nqn)
|
||||
|
||||
def delete(self):
|
||||
do_privsep_call(serialize(self), 'delete')
|
||||
|
||||
|
||||
class Root(nvmet.Root):
|
||||
@property
|
||||
def ports(self):
|
||||
for d in os.listdir(self.path + '/ports/'):
|
||||
yield Port(os.path.basename(d))
|
0
cinder/tests/unit/privsep/__init__.py
Normal file
0
cinder/tests/unit/privsep/__init__.py
Normal file
0
cinder/tests/unit/privsep/targets/__init__.py
Normal file
0
cinder/tests/unit/privsep/targets/__init__.py
Normal file
40
cinder/tests/unit/privsep/targets/fake_nvmet_lib.py
Normal file
40
cinder/tests/unit/privsep/targets/fake_nvmet_lib.py
Normal file
@ -0,0 +1,40 @@
|
||||
# Copyright 2022 Red Hat, Inc
|
||||
#
|
||||
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||
# not use this file except in compliance with the License. You may obtain
|
||||
# a copy of the License at
|
||||
#
|
||||
# http://www.apache.org/licenses/LICENSE-2.0
|
||||
#
|
||||
# Unless required by applicable law or agreed to in writing, software
|
||||
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||
# License for the specific language governing permissions and limitations
|
||||
# under the License.
|
||||
"""
|
||||
FAKE the nvmet library if it's not installed.
|
||||
This must be imported before cinder/volume/targets/nvmet.py and
|
||||
cinder/privsep/targets/nvmet.py
|
||||
"""
|
||||
|
||||
import sys
|
||||
from unittest import mock
|
||||
|
||||
from cinder import exception
|
||||
|
||||
try:
|
||||
import nvmet # noqa
|
||||
reset_mock = lambda: None # noqa
|
||||
except ImportError:
|
||||
mock_nvmet_lib = mock.Mock(name='nvmet',
|
||||
Root=type('Root', (mock.Mock, ), {}),
|
||||
Subsystem=type('Subsystem', (mock.Mock, ), {}),
|
||||
Port=type('Port', (mock.Mock, ), {}),
|
||||
Namespace=type('Namespace', (mock.Mock, ), {}),
|
||||
Host=type('Host', (mock.Mock, ), {}),
|
||||
ANAGroup=type('ANAGroup', (mock.Mock, ), {}),
|
||||
Referral=type('Referral', (mock.Mock, ), {}),
|
||||
nvme=mock.Mock(CFSNotFound=exception.NotFound))
|
||||
|
||||
sys.modules['nvmet'] = mock_nvmet_lib
|
||||
reset_mock = mock_nvmet_lib.reset_mock
|
342
cinder/tests/unit/privsep/targets/test_nvmet.py
Normal file
342
cinder/tests/unit/privsep/targets/test_nvmet.py
Normal file
@ -0,0 +1,342 @@
|
||||
# Copyright 2022 Red Hat, Inc
|
||||
#
|
||||
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||
# not use this file except in compliance with the License. You may obtain
|
||||
# a copy of the License at
|
||||
#
|
||||
# http://www.apache.org/licenses/LICENSE-2.0
|
||||
#
|
||||
# Unless required by applicable law or agreed to in writing, software
|
||||
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||
# License for the specific language governing permissions and limitations
|
||||
# under the License.
|
||||
|
||||
from unittest import mock
|
||||
|
||||
import ddt
|
||||
|
||||
from cinder import exception
|
||||
from cinder.tests.unit.privsep.targets import fake_nvmet_lib
|
||||
from cinder.tests.unit import test
|
||||
# This must go after fake_nvmet_lib has been imported (thus the noqa)
|
||||
from cinder.privsep.targets import nvmet # noqa
|
||||
|
||||
|
||||
@ddt.ddt
|
||||
class TestSerialize(test.TestCase):
|
||||
def setUp(self):
|
||||
super().setUp()
|
||||
fake_nvmet_lib.reset_mock()
|
||||
|
||||
def test_tuple(self):
|
||||
"""Test serialization of a tuple."""
|
||||
instance = (1, 'string')
|
||||
res = nvmet.serialize(instance)
|
||||
self.assertEqual(('tuple', instance), res)
|
||||
|
||||
@ddt.data(1, 1.1, 'string', None, [1, 2, 'string'])
|
||||
def test_others(self, instance):
|
||||
"""Test normal Python instances that should not be modified."""
|
||||
res = nvmet.serialize(instance)
|
||||
self.assertEqual(instance, res)
|
||||
|
||||
def test_root(self):
|
||||
instance = nvmet.Root()
|
||||
res = nvmet.serialize(instance)
|
||||
self.assertEqual(('Root', {}), res)
|
||||
|
||||
def test_host(self):
|
||||
instance = nvmet.Host(nqn='_nqn')
|
||||
res = nvmet.serialize(instance)
|
||||
self.assertEqual(('Host', {'nqn': '_nqn', 'mode': 'lookup'}), res)
|
||||
|
||||
def test_subsystem(self):
|
||||
instance = nvmet.Subsystem(nqn='_nqn')
|
||||
res = nvmet.serialize(instance)
|
||||
self.assertEqual(('Subsystem', {'nqn': '_nqn', 'mode': 'lookup'}), res)
|
||||
|
||||
def test_namespace(self):
|
||||
subsys = nvmet.Subsystem(nqn='_nqn')
|
||||
instance = nvmet.Namespace(subsystem=subsys, nsid='_nsid')
|
||||
res = nvmet.serialize(instance)
|
||||
# Subsystem is a recursive serialization
|
||||
expected = (
|
||||
'Namespace', {'subsystem': ('Subsystem', {'nqn': '_nqn',
|
||||
'mode': 'lookup'}),
|
||||
'nsid': '_nsid',
|
||||
'mode': 'lookup'})
|
||||
self.assertEqual(expected, res)
|
||||
|
||||
def test_port(self):
|
||||
instance = nvmet.Port(portid='_portid')
|
||||
res = nvmet.serialize(instance)
|
||||
expected = ('Port', {'portid': '_portid', 'mode': 'lookup'})
|
||||
self.assertEqual(expected, res)
|
||||
|
||||
def test_Referral(self):
|
||||
port = nvmet.Port(portid='_portid')
|
||||
# name is a Mock attribute, so we'll use it as instance.name
|
||||
instance = nvmet.Referral(port=port, name='_name')
|
||||
res = nvmet.serialize(instance)
|
||||
# Port is a recursive serialization
|
||||
expected = (
|
||||
'Referral', {'port': ('Port', {'portid': '_portid',
|
||||
'mode': 'lookup'}),
|
||||
'name': instance.name,
|
||||
'mode': 'lookup'})
|
||||
self.assertEqual(expected, res)
|
||||
|
||||
def test_ANAGroup(self):
|
||||
port = nvmet.Port(portid='_portid')
|
||||
instance = nvmet.ANAGroup(port=port, grpid='_grpid')
|
||||
res = nvmet.serialize(instance)
|
||||
expected = (
|
||||
'ANAGroup', {'port': ('Port', {'portid': '_portid',
|
||||
'mode': 'lookup'}),
|
||||
'grpid': '_grpid',
|
||||
'mode': 'lookup'})
|
||||
self.assertEqual(expected, res)
|
||||
|
||||
|
||||
@ddt.ddt
|
||||
class TestDeserialize(test.TestCase):
|
||||
def test_deserialize_tuple(self):
|
||||
"""Test serialization of a tuple."""
|
||||
expected = (1, 'string')
|
||||
data = ('tuple', expected)
|
||||
res = nvmet.deserialize(data)
|
||||
self.assertEqual(expected, res)
|
||||
|
||||
@ddt.data(1, 1.1, 'string', None, [1, 2, 'string'])
|
||||
def test_deserialize_others(self, data):
|
||||
"""Test normal Python instances that should not be modified."""
|
||||
res = nvmet.deserialize(data)
|
||||
self.assertEqual(data, res)
|
||||
|
||||
def test_deserialize_root(self):
|
||||
data = ('Root', {})
|
||||
res = nvmet.deserialize(data)
|
||||
self.assertIsInstance(res, nvmet.nvmet.Root)
|
||||
|
||||
def test_deserialize_host(self):
|
||||
data = ('Host', {'nqn': '_nqn', 'mode': 'lookup'})
|
||||
host = nvmet.deserialize(data)
|
||||
self.assertIsInstance(host, nvmet.nvmet.Host)
|
||||
self.assertEqual('_nqn', host.nqn)
|
||||
self.assertEqual('lookup', host.mode)
|
||||
|
||||
def test_deserialize_subsystem(self):
|
||||
data = ('Subsystem', {'nqn': '_nqn', 'mode': 'lookup'})
|
||||
subsys = nvmet.deserialize(data)
|
||||
self.assertIsInstance(subsys, nvmet.nvmet.Subsystem)
|
||||
self.assertEqual('_nqn', subsys.nqn)
|
||||
self.assertEqual('lookup', subsys.mode)
|
||||
|
||||
def test_deserialize_namespace(self):
|
||||
data = ('Namespace', {'subsystem': ('Subsystem', {'nqn': '_nqn',
|
||||
'mode': 'lookup'}),
|
||||
'nsid': '_nsid',
|
||||
'mode': 'lookup'})
|
||||
|
||||
ns = nvmet.deserialize(data)
|
||||
self.assertIsInstance(ns, nvmet.nvmet.Namespace)
|
||||
self.assertEqual('_nsid', ns.nsid)
|
||||
self.assertEqual('lookup', ns.mode)
|
||||
self.assertIsInstance(ns.subsystem, nvmet.nvmet.Subsystem)
|
||||
self.assertEqual('_nqn', ns.subsystem.nqn)
|
||||
self.assertEqual('lookup', ns.subsystem.mode)
|
||||
|
||||
def test_deserialize_port(self):
|
||||
data = ('Port', {'portid': '_portid', 'mode': 'lookup'})
|
||||
port = nvmet.deserialize(data)
|
||||
self.assertIsInstance(port, nvmet.nvmet.Port)
|
||||
self.assertEqual('_portid', port.portid)
|
||||
self.assertEqual('lookup', port.mode)
|
||||
|
||||
def test_deserialize_Referral(self):
|
||||
data = ('Referral', {'port': ('Port', {'portid': '_portid',
|
||||
'mode': 'lookup'}),
|
||||
'name': '1',
|
||||
'mode': 'lookup'})
|
||||
ref = nvmet.deserialize(data)
|
||||
|
||||
self.assertIsInstance(ref, nvmet.nvmet.Referral)
|
||||
self.assertEqual('1', ref._mock_name) # Because name is used by Mock
|
||||
self.assertEqual('lookup', ref.mode)
|
||||
self.assertIsInstance(ref.port, nvmet.nvmet.Port)
|
||||
self.assertEqual('_portid', ref.port.portid)
|
||||
self.assertEqual('lookup', ref.port.mode)
|
||||
|
||||
def test_deserialize_ANAGroup(self):
|
||||
data = ('ANAGroup', {'port': ('Port', {'portid': '_portid',
|
||||
'mode': 'lookup'}),
|
||||
'grpid': '_grpid',
|
||||
'mode': 'lookup'})
|
||||
ana = nvmet.deserialize(data)
|
||||
|
||||
self.assertIsInstance(ana, nvmet.nvmet.ANAGroup)
|
||||
self.assertEqual('_grpid', ana.grpid)
|
||||
self.assertEqual('lookup', ana.mode)
|
||||
self.assertIsInstance(ana.port, nvmet.nvmet.Port)
|
||||
self.assertEqual('_portid', ana.port.portid)
|
||||
self.assertEqual('lookup', ana.port.mode)
|
||||
|
||||
@mock.patch.object(nvmet, 'deserialize')
|
||||
def test_deserialize_params(self, mock_deserialize):
|
||||
mock_deserialize.side_effect = [11, 22, 33, 55, 77]
|
||||
args = [1, 2, 3]
|
||||
kwargs = {'4': 5, '6': 7}
|
||||
|
||||
res_args, res_kwargs = nvmet.deserialize_params(args, kwargs)
|
||||
|
||||
self.assertEqual(5, mock_deserialize.call_count)
|
||||
mock_deserialize.assert_has_calls((mock.call(1),
|
||||
mock.call(2),
|
||||
mock.call(3),
|
||||
mock.call(5),
|
||||
mock.call(7)))
|
||||
self.assertEqual([11, 22, 33], res_args)
|
||||
self.assertEqual({'4': 55, '6': 77}, res_kwargs)
|
||||
|
||||
|
||||
class TestPrivsep(test.TestCase):
|
||||
@mock.patch.object(nvmet.LOG, 'error')
|
||||
def test__nvmet_setup_failure(self, mock_log):
|
||||
self.assertRaises(exception.CinderException,
|
||||
nvmet._nvmet_setup_failure, mock.sentinel.message)
|
||||
mock_log.assert_called_once_with(mock.sentinel.message)
|
||||
|
||||
# We mock the privsep context mode to fake that we are not the client
|
||||
@mock.patch('cinder.privsep.sys_admin_pctxt.client_mode', False)
|
||||
@mock.patch.object(nvmet, 'deserialize_params')
|
||||
@mock.patch.object(nvmet.nvmet, 'MyClass')
|
||||
def test_privsep_setup(self, mock_class, mock_deserialize):
|
||||
args = (1, 2, 3)
|
||||
kwargs = {'4': 5, '6': 7}
|
||||
deserialized_args = (11, 22, 33)
|
||||
deserialized_kwargs = {'4': 55, '6': 77}
|
||||
|
||||
expected_args = deserialized_args[:]
|
||||
expected_kwargs = deserialized_kwargs.copy()
|
||||
expected_kwargs['err_func'] = nvmet._nvmet_setup_failure
|
||||
|
||||
mock_deserialize.return_value = (deserialized_args,
|
||||
deserialized_kwargs)
|
||||
|
||||
nvmet.privsep_setup('MyClass', *args, **kwargs)
|
||||
|
||||
mock_deserialize.assert_called_once_with(args, kwargs)
|
||||
mock_class.setup.assert_called_once_with(*expected_args,
|
||||
**expected_kwargs)
|
||||
|
||||
# We mock the privsep context mode to fake that we are not the client
|
||||
@mock.patch('cinder.privsep.sys_admin_pctxt.client_mode', False)
|
||||
@mock.patch.object(nvmet, 'deserialize')
|
||||
@mock.patch.object(nvmet, 'deserialize_params')
|
||||
def test_do_privsep_call(self, mock_deserialize_params, mock_deserialize):
|
||||
args = (1, 2, 3)
|
||||
kwargs = {'4': 5, '6': 7}
|
||||
deserialized_args = (11, 22, 33)
|
||||
deserialized_kwargs = {'4': 55, '6': 77}
|
||||
|
||||
mock_deserialize_params.return_value = (deserialized_args,
|
||||
deserialized_kwargs)
|
||||
|
||||
res = nvmet.do_privsep_call(mock.sentinel.instance,
|
||||
'method_name',
|
||||
*args, **kwargs)
|
||||
mock_deserialize.assert_called_once_with(mock.sentinel.instance)
|
||||
mock_deserialize_params.assert_called_once_with(args, kwargs)
|
||||
|
||||
mock_method = mock_deserialize.return_value.method_name
|
||||
mock_method.assert_called_once_with(*deserialized_args,
|
||||
**deserialized_kwargs)
|
||||
self.assertEqual(mock_method.return_value, res)
|
||||
|
||||
|
||||
@ddt.ddt
|
||||
class TestNvmetClasses(test.TestCase):
|
||||
@ddt.data('Namespace', 'Host', 'Referral', 'ANAGroup')
|
||||
def test_same_classes(self, cls_name):
|
||||
self.assertEqual(getattr(nvmet, cls_name),
|
||||
getattr(nvmet.nvmet, cls_name))
|
||||
|
||||
def test_subsystem_init(self):
|
||||
subsys = nvmet.Subsystem('nqn')
|
||||
self.assertIsInstance(subsys, nvmet.nvmet.Subsystem)
|
||||
self.assertIsInstance(subsys, nvmet.Subsystem)
|
||||
self.assertEqual('nqn', subsys.nqn)
|
||||
self.assertEqual('lookup', subsys.mode)
|
||||
|
||||
@mock.patch.object(nvmet, 'privsep_setup')
|
||||
def test_subsystem_setup(self, mock_setup):
|
||||
nvmet.Subsystem.setup(mock.sentinel.t)
|
||||
mock_setup.assert_called_once_with('Subsystem', mock.sentinel.t)
|
||||
|
||||
@mock.patch.object(nvmet, 'serialize')
|
||||
@mock.patch.object(nvmet, 'do_privsep_call')
|
||||
def test_subsystem_delete(self, mock_privsep, mock_serialize):
|
||||
subsys = nvmet.Subsystem('nqn')
|
||||
subsys.delete()
|
||||
mock_serialize.assert_called_once_with(subsys)
|
||||
mock_privsep.assert_called_once_with(mock_serialize.return_value,
|
||||
'delete')
|
||||
|
||||
def test_port_init(self):
|
||||
port = nvmet.Port('portid')
|
||||
self.assertIsInstance(port, nvmet.nvmet.Port)
|
||||
self.assertIsInstance(port, nvmet.Port)
|
||||
self.assertEqual('portid', port.portid)
|
||||
self.assertEqual('lookup', port.mode)
|
||||
|
||||
@mock.patch.object(nvmet, 'serialize')
|
||||
@mock.patch.object(nvmet, 'privsep_setup')
|
||||
def test_port_setup(self, mock_setup, mock_serialize):
|
||||
nvmet.Port.setup(mock.sentinel.root, mock.sentinel.n)
|
||||
mock_serialize.assert_called_once_with(mock.sentinel.root)
|
||||
mock_setup.assert_called_once_with('Port', mock_serialize.return_value,
|
||||
mock.sentinel.n)
|
||||
|
||||
@mock.patch.object(nvmet, 'serialize')
|
||||
@mock.patch.object(nvmet, 'do_privsep_call')
|
||||
def test_port_add_subsystem(self, mock_privsep, mock_serialize):
|
||||
port = nvmet.Port('portid')
|
||||
port.add_subsystem(mock.sentinel.nqn)
|
||||
mock_serialize.assert_called_once_with(port)
|
||||
mock_privsep.assert_called_once_with(mock_serialize.return_value,
|
||||
'add_subsystem',
|
||||
mock.sentinel.nqn)
|
||||
|
||||
@mock.patch.object(nvmet, 'serialize')
|
||||
@mock.patch.object(nvmet, 'do_privsep_call')
|
||||
def test_port_remove_subsystem(self, mock_privsep, mock_serialize):
|
||||
port = nvmet.Port('portid')
|
||||
port.remove_subsystem(mock.sentinel.nqn)
|
||||
mock_serialize.assert_called_once_with(port)
|
||||
mock_privsep.assert_called_once_with(mock_serialize.return_value,
|
||||
'remove_subsystem',
|
||||
mock.sentinel.nqn)
|
||||
|
||||
@mock.patch.object(nvmet, 'serialize')
|
||||
@mock.patch.object(nvmet, 'do_privsep_call')
|
||||
def test_port_delete(self, mock_privsep, mock_serialize):
|
||||
port = nvmet.Port('portid')
|
||||
port.delete()
|
||||
mock_serialize.assert_called_once_with(port)
|
||||
mock_privsep.assert_called_once_with(mock_serialize.return_value,
|
||||
'delete')
|
||||
|
||||
@mock.patch('os.listdir', return_value=['/path/ports/1', '/path/ports/2'])
|
||||
@mock.patch.object(nvmet, 'Port')
|
||||
def test_root_ports(self, mock_port, mock_listdir):
|
||||
r = nvmet.Root()
|
||||
r.path = '/path' # This is set by the parent nvmet library Root class
|
||||
|
||||
res = list(r.ports)
|
||||
|
||||
self.assertEqual([mock_port.return_value, mock_port.return_value], res)
|
||||
|
||||
mock_listdir.assert_called_once_with('/path/ports/')
|
||||
self.assertEqual(2, mock_port.call_count)
|
||||
mock_port.assert_has_calls((mock.call('1'), mock.call('2')))
|
@ -13,12 +13,13 @@
|
||||
from unittest import mock
|
||||
|
||||
import ddt
|
||||
from oslo_utils import timeutils
|
||||
|
||||
from cinder import context
|
||||
from cinder.tests.unit.privsep.targets import fake_nvmet_lib
|
||||
from cinder.tests.unit.targets import targets_fixture as tf
|
||||
from cinder import utils
|
||||
from cinder.volume.targets import nvmet
|
||||
# This must go after fake_nvmet_lib has been imported (thus the noqa)
|
||||
from cinder.privsep.targets import nvmet as priv_nvmet # noqa
|
||||
|
||||
|
||||
@ddt.ddt
|
||||
@ -26,248 +27,241 @@ class TestNVMETDriver(tf.TargetDriverFixture):
|
||||
|
||||
def setUp(self):
|
||||
super(TestNVMETDriver, self).setUp()
|
||||
self.initialize('nvmet_rdma', 'rdma')
|
||||
|
||||
def initialize(self, target_protocol, transport_type):
|
||||
self.configuration.target_protocol = target_protocol
|
||||
self.configuration.target_prefix = 'nvme-subsystem-1'
|
||||
self.configuration.target_protocol = 'nvmet_rdma'
|
||||
self.target = nvmet.NVMET(root_helper=utils.get_root_helper(),
|
||||
configuration=self.configuration)
|
||||
fake_nvmet_lib.reset_mock()
|
||||
|
||||
self.target_ip = '192.168.0.1'
|
||||
self.target_port = '1234'
|
||||
self.nvmet_subsystem_name = self.configuration.target_prefix
|
||||
self.nvmet_ns_id = self.configuration.nvmet_ns_id
|
||||
self.nvmet_port_id = self.configuration.nvmet_port_id
|
||||
self.nvme_transport_type = transport_type
|
||||
@mock.patch.object(nvmet.NVMET, 'create_nvmeof_target')
|
||||
def test_create_export(self, mock_create_target):
|
||||
"""Test that the nvmeof class calls the nvmet method."""
|
||||
res = self.target.create_export(mock.sentinel.ctxt,
|
||||
self.testvol,
|
||||
mock.sentinel.volume_path)
|
||||
self.assertEqual(mock_create_target.return_value, res)
|
||||
mock_create_target.assert_called_once_with(
|
||||
self.testvol['id'],
|
||||
self.target.configuration.target_prefix,
|
||||
self.target.target_ip,
|
||||
self.target.target_port,
|
||||
self.target.nvme_transport_type,
|
||||
self.target.nvmet_port_id,
|
||||
self.target.nvmet_ns_id,
|
||||
mock.sentinel.volume_path)
|
||||
|
||||
self.fake_volume_id = 'c446b9a2-c968-4260-b95f-a18a7b41c004'
|
||||
self.testvol_path = (
|
||||
'/dev/stack-volumes-lvmdriver-1/volume-%s' % self.fake_volume_id)
|
||||
self.fake_project_id = 'ed2c1fd4-5555-1111-aa15-123b93f75cba'
|
||||
self.testvol = (
|
||||
{'project_id': self.fake_project_id,
|
||||
'name': 'testvol',
|
||||
'size': 1,
|
||||
'id': self.fake_volume_id,
|
||||
'volume_type_id': None,
|
||||
'provider_location': self.target.get_nvmeof_location(
|
||||
"nqn.%s-%s" % (self.nvmet_subsystem_name,
|
||||
self.fake_volume_id),
|
||||
self.target_ip, self.target_port, self.nvme_transport_type,
|
||||
self.nvmet_ns_id),
|
||||
'provider_auth': None,
|
||||
'provider_geometry': None,
|
||||
'created_at': timeutils.utcnow(),
|
||||
'host': 'fake_host@lvm#lvm'})
|
||||
@mock.patch.object(nvmet.NVMET, 'get_nvmeof_location')
|
||||
@mock.patch.object(nvmet.NVMET, '_ensure_port_exports')
|
||||
@mock.patch.object(nvmet.NVMET, '_ensure_subsystem_exists')
|
||||
@mock.patch.object(nvmet.NVMET, '_get_target_nqn')
|
||||
def test_create_nvmeof_target(self, mock_nqn, mock_subsys, mock_port,
|
||||
mock_location):
|
||||
"""Normal create target execution."""
|
||||
mock_nqn.return_value = mock.sentinel.nqn
|
||||
|
||||
@ddt.data(('nvmet_rdma', 'rdma'), ('nvmet_tcp', 'tcp'))
|
||||
res = self.target.create_nvmeof_target(mock.sentinel.vol_id,
|
||||
mock.sentinel.target_prefix,
|
||||
mock.sentinel.target_ip,
|
||||
mock.sentinel.target_port,
|
||||
mock.sentinel.transport_type,
|
||||
mock.sentinel.port_id,
|
||||
mock.sentinel.ns_id,
|
||||
mock.sentinel.volume_path)
|
||||
|
||||
self.assertEqual({'location': mock_location.return_value, 'auth': ''},
|
||||
res)
|
||||
mock_nqn.assert_called_once_with(mock.sentinel.vol_id)
|
||||
mock_subsys.assert_called_once_with(mock.sentinel.nqn,
|
||||
mock.sentinel.ns_id,
|
||||
mock.sentinel.volume_path)
|
||||
mock_port.assert_called_once_with(mock.sentinel.nqn,
|
||||
mock.sentinel.target_ip,
|
||||
mock.sentinel.target_port,
|
||||
mock.sentinel.transport_type,
|
||||
mock.sentinel.port_id)
|
||||
|
||||
mock_location.assert_called_once_with(mock.sentinel.nqn,
|
||||
mock.sentinel.target_ip,
|
||||
mock.sentinel.target_port,
|
||||
mock.sentinel.transport_type,
|
||||
mock.sentinel.ns_id)
|
||||
|
||||
@ddt.data((ValueError, None), (None, IndexError))
|
||||
@ddt.unpack
|
||||
@mock.patch.object(nvmet.NVMET, '_get_nvmf_subsystem')
|
||||
@mock.patch.object(nvmet.NVMET, '_get_available_nvmf_subsystems')
|
||||
@mock.patch.object(nvmet.NVMET, '_add_nvmf_subsystem')
|
||||
def test_create_export(self, target_protocol, transport_type,
|
||||
mock_add_nvmf_subsystem,
|
||||
mock_get_available_nvmf_subsystems,
|
||||
mock_get_nvmf_subsystem):
|
||||
self.initialize(target_protocol, transport_type)
|
||||
@mock.patch.object(nvmet.NVMET, 'get_nvmeof_location')
|
||||
@mock.patch.object(nvmet.NVMET, '_ensure_port_exports')
|
||||
@mock.patch.object(nvmet.NVMET, '_ensure_subsystem_exists')
|
||||
@mock.patch.object(nvmet.NVMET, '_get_target_nqn')
|
||||
def test_create_nvmeof_target_error(self, subsys_effect, port_effect,
|
||||
mock_nqn, mock_subsys, mock_port,
|
||||
mock_location):
|
||||
"""Failing create target executing subsystem or port creation."""
|
||||
mock_subsys.side_effect = subsys_effect
|
||||
mock_port.side_effect = port_effect
|
||||
mock_nqn.return_value = mock.sentinel.nqn
|
||||
|
||||
mock_testvol = self.testvol
|
||||
mock_testvol_path = self.testvol_path
|
||||
ctxt = context.get_admin_context()
|
||||
mock_get_available_nvmf_subsystems.return_value = {
|
||||
"subsystems": [],
|
||||
"hosts": [],
|
||||
"ports": [
|
||||
{"subsystems": [],
|
||||
"referrals": [],
|
||||
"portid": 1,
|
||||
"addr":
|
||||
{"treq": "not specified",
|
||||
"trtype": self.nvme_transport_type,
|
||||
"adrfam": "ipv4",
|
||||
"trsvcid": self.target_port,
|
||||
"traddr":
|
||||
self.target_ip
|
||||
}
|
||||
}]
|
||||
}
|
||||
mock_get_nvmf_subsystem.return_value = (
|
||||
"nqn.%s-%s" % (self.nvmet_subsystem_name,
|
||||
mock_testvol['id']))
|
||||
|
||||
mock_add_nvmf_subsystem.return_value = (
|
||||
"nqn.%s-%s" % (self.nvmet_subsystem_name,
|
||||
mock_testvol['id']))
|
||||
|
||||
expected_return = {
|
||||
'location': self.target.get_nvmeof_location(
|
||||
mock_add_nvmf_subsystem.return_value, self.target_ip,
|
||||
self.target_port, self.nvme_transport_type, self.nvmet_ns_id),
|
||||
'auth': ''
|
||||
}
|
||||
|
||||
self.target.target_ip = self.target_ip
|
||||
self.target.target_port = self.target_port
|
||||
self.assertEqual(expected_return,
|
||||
self.target.create_export(
|
||||
ctxt, mock_testvol,
|
||||
mock_testvol_path))
|
||||
|
||||
@mock.patch.object(nvmet.NVMET, '_get_nvmf_subsystem')
|
||||
@mock.patch.object(nvmet.NVMET, '_get_available_nvmf_subsystems')
|
||||
@mock.patch.object(nvmet.NVMET, '_add_nvmf_subsystem')
|
||||
def test_create_export_with_error_add_nvmf_subsystem(
|
||||
self,
|
||||
mock_add_nvmf_subsystem,
|
||||
mock_get_available_nvmf_subsystems,
|
||||
mock_get_nvmf_subsystem):
|
||||
|
||||
mock_testvol = self.testvol
|
||||
mock_testvol_path = self.testvol_path
|
||||
ctxt = context.get_admin_context()
|
||||
mock_get_available_nvmf_subsystems.return_value = {
|
||||
"subsystems": [],
|
||||
"hosts": [],
|
||||
"ports": [
|
||||
{"subsystems": [],
|
||||
"referrals": [],
|
||||
"portid": 1,
|
||||
"addr":
|
||||
{"treq": "not specified",
|
||||
"trtype": self.nvme_transport_type,
|
||||
"adrfam": "ipv4",
|
||||
"trsvcid": self.target_port,
|
||||
"traddr":
|
||||
self.target_ip
|
||||
}
|
||||
}]
|
||||
}
|
||||
mock_get_nvmf_subsystem.return_value = None
|
||||
|
||||
mock_add_nvmf_subsystem.return_value = None
|
||||
|
||||
self.target.target_ip = self.target_ip
|
||||
self.target.target_port = self.target_port
|
||||
self.assertRaises(nvmet.NVMETTargetAddError,
|
||||
self.target.create_export,
|
||||
ctxt,
|
||||
mock_testvol,
|
||||
mock_testvol_path)
|
||||
self.target.create_nvmeof_target,
|
||||
mock.sentinel.vol_id,
|
||||
mock.sentinel.target_prefix,
|
||||
mock.sentinel.target_ip,
|
||||
mock.sentinel.target_port,
|
||||
mock.sentinel.transport_type,
|
||||
mock.sentinel.port_id,
|
||||
mock.sentinel.ns_id,
|
||||
mock.sentinel.volume_path)
|
||||
|
||||
@mock.patch.object(nvmet.NVMET, '_get_nvmf_subsystem')
|
||||
@mock.patch.object(nvmet.NVMET, '_get_available_nvmf_subsystems')
|
||||
@mock.patch.object(nvmet.NVMET, '_delete_nvmf_subsystem')
|
||||
def test_remove_export(self, mock_delete_nvmf_subsystem,
|
||||
mock_get_available_nvmf_subsystems,
|
||||
mock_get_nvmf_subsystem):
|
||||
mock_testvol = self.testvol
|
||||
mock_testvol_path = self.testvol_path
|
||||
ctxt = context.get_admin_context()
|
||||
mock_get_available_nvmf_subsystems.return_value = {
|
||||
"subsystems": [
|
||||
{"allowed_hosts": [],
|
||||
"nqn": "nqn.%s-%s" % (
|
||||
self.nvmet_subsystem_name,
|
||||
mock_testvol['id']),
|
||||
"attr": {"allow_any_host": "1"},
|
||||
"namespaces": [
|
||||
{"device":
|
||||
{"path": mock_testvol_path,
|
||||
"nguid":
|
||||
"86fab0e0-825d-4f25-a449-28b93c5e8dd6"
|
||||
},
|
||||
"enable": 1, "nsid":
|
||||
self.nvmet_ns_id,
|
||||
}]}],
|
||||
"hosts": [],
|
||||
"ports": [
|
||||
{"subsystems": [
|
||||
"nqn.%s-%s" % (self.nvmet_subsystem_name,
|
||||
mock_testvol['id'])],
|
||||
"referrals": [],
|
||||
"portid": self.nvmet_port_id,
|
||||
"addr":
|
||||
{"treq": "not specified",
|
||||
"trtype": self.nvme_transport_type,
|
||||
"adrfam": "ipv4",
|
||||
"trsvcid": self.target_port,
|
||||
"traddr": self.target_ip}}
|
||||
]
|
||||
mock_nqn.assert_called_once_with(mock.sentinel.vol_id)
|
||||
mock_subsys.assert_called_once_with(mock.sentinel.nqn,
|
||||
mock.sentinel.ns_id,
|
||||
mock.sentinel.volume_path)
|
||||
if subsys_effect:
|
||||
mock_port.assert_not_called()
|
||||
else:
|
||||
mock_port.assert_called_once_with(mock.sentinel.nqn,
|
||||
mock.sentinel.target_ip,
|
||||
mock.sentinel.target_port,
|
||||
mock.sentinel.transport_type,
|
||||
mock.sentinel.port_id)
|
||||
mock_location.assert_not_called()
|
||||
|
||||
@mock.patch.object(priv_nvmet, 'Subsystem')
|
||||
def test__ensure_subsystem_exists_already_exists(self, mock_subsys):
|
||||
"""Skip subsystem creation if already exists."""
|
||||
nqn = 'nqn.nvme-subsystem-1-uuid'
|
||||
self.target._ensure_subsystem_exists(nqn, mock.sentinel.ns_id,
|
||||
mock.sentinel.vol_path)
|
||||
mock_subsys.assert_called_once_with(nqn)
|
||||
mock_subsys.setup.assert_not_called()
|
||||
|
||||
@mock.patch('oslo_utils.uuidutils.generate_uuid')
|
||||
@mock.patch.object(priv_nvmet, 'Subsystem')
|
||||
def test__ensure_subsystem_exists(self, mock_subsys, mock_uuid):
|
||||
"""Create subsystem when it doesn't exist."""
|
||||
mock_subsys.side_effect = priv_nvmet.NotFound
|
||||
mock_uuid.return_value = 'uuid'
|
||||
nqn = 'nqn.nvme-subsystem-1-uuid'
|
||||
self.target._ensure_subsystem_exists(nqn, mock.sentinel.ns_id,
|
||||
mock.sentinel.vol_path)
|
||||
mock_subsys.assert_called_once_with(nqn)
|
||||
expected_section = {
|
||||
'allowed_hosts': [],
|
||||
'attr': {'allow_any_host': '1'},
|
||||
'namespaces': [{'device': {'nguid': 'uuid',
|
||||
'path': mock.sentinel.vol_path},
|
||||
'enable': 1,
|
||||
'nsid': mock.sentinel.ns_id}],
|
||||
'nqn': nqn
|
||||
}
|
||||
mock_subsys.setup.assert_called_once_with(expected_section)
|
||||
|
||||
mock_get_nvmf_subsystem.return_value = (
|
||||
"nqn.%s-%s" % (self.nvmet_subsystem_name,
|
||||
mock_testvol['id']))
|
||||
mock_delete_nvmf_subsystem.return_value = (
|
||||
"nqn.%s-%s" % (self.nvmet_subsystem_name,
|
||||
mock_testvol['id']))
|
||||
expected_return = mock_delete_nvmf_subsystem.return_value
|
||||
self.assertEqual(expected_return,
|
||||
self.target.remove_export(ctxt, mock_testvol))
|
||||
@mock.patch.object(priv_nvmet, 'Port')
|
||||
def test__ensure_port_exports_already_does(self, mock_port):
|
||||
"""Skips port creation and subsystem export since they both exist."""
|
||||
nqn = 'nqn.nvme-subsystem-1-uuid'
|
||||
mock_port.return_value.subsystems = [nqn]
|
||||
self.target._ensure_port_exports(nqn,
|
||||
mock.sentinel.addr,
|
||||
mock.sentinel.port,
|
||||
mock.sentinel.transport,
|
||||
mock.sentinel.port_id)
|
||||
mock_port.assert_called_once_with(mock.sentinel.port_id)
|
||||
mock_port.setup.assert_not_called()
|
||||
mock_port.return_value.add_subsystem.assert_not_called()
|
||||
|
||||
@mock.patch.object(nvmet.NVMET, '_get_nvmf_subsystem')
|
||||
@mock.patch.object(nvmet.NVMET, '_get_available_nvmf_subsystems')
|
||||
def test_remove_export_with_empty_subsystems(
|
||||
self,
|
||||
mock_get_available_nvmf_subsystems,
|
||||
mock_get_nvmf_subsystem):
|
||||
mock_testvol = self.testvol
|
||||
ctxt = context.get_admin_context()
|
||||
mock_get_available_nvmf_subsystems.return_value = {
|
||||
"subsystems": [],
|
||||
"hosts": [],
|
||||
"ports": []
|
||||
}
|
||||
mock_get_nvmf_subsystem.return_value = None
|
||||
self.assertIsNone(self.target.remove_export(ctxt, mock_testvol))
|
||||
@mock.patch.object(priv_nvmet, 'Port')
|
||||
def test__ensure_port_exports_port_exists_not_exported(self, mock_port):
|
||||
"""Skips port creation if exists but exports subsystem."""
|
||||
nqn = 'nqn.nvme-subsystem-1-vol-2-uuid'
|
||||
mock_port.return_value.subsystems = ['nqn.nvme-subsystem-1-vol-1-uuid']
|
||||
self.target._ensure_port_exports(nqn,
|
||||
mock.sentinel.addr,
|
||||
mock.sentinel.port,
|
||||
mock.sentinel.transport,
|
||||
mock.sentinel.port_id)
|
||||
mock_port.assert_called_once_with(mock.sentinel.port_id)
|
||||
mock_port.setup.assert_not_called()
|
||||
mock_port.return_value.add_subsystem.assert_called_once_with(nqn)
|
||||
|
||||
@mock.patch.object(nvmet.NVMET, '_get_nvmf_subsystem')
|
||||
@mock.patch.object(nvmet.NVMET, '_get_available_nvmf_subsystems')
|
||||
@mock.patch.object(nvmet.NVMET, '_delete_nvmf_subsystem')
|
||||
def test_remove_export_with_delete_nvmf_subsystem_fails(
|
||||
self,
|
||||
moc_delete_nvmf_subsystem,
|
||||
mock_get_available_nvmf_subsystems,
|
||||
mock_get_nvmf_subsystem):
|
||||
mock_testvol = self.testvol
|
||||
mock_testvol_path = self.testvol_path
|
||||
ctxt = context.get_admin_context()
|
||||
mock_get_available_nvmf_subsystems.return_value = {
|
||||
"subsystems": [
|
||||
{"allowed_hosts": [],
|
||||
"nqn": "nqn.%s-%s" % (
|
||||
self.nvmet_subsystem_name,
|
||||
mock_testvol['id']),
|
||||
"attr": {"allow_any_host": "1"},
|
||||
"namespaces": [
|
||||
{"device":
|
||||
{"path": mock_testvol_path,
|
||||
"nguid":
|
||||
"86fab0e0-825d-4f25-a449-28b93c5e8dd6"
|
||||
},
|
||||
"enable": 1, "nsid":
|
||||
self.nvmet_ns_id,
|
||||
}]}],
|
||||
"hosts": [],
|
||||
"ports": [
|
||||
{"subsystems": [
|
||||
"nqn.%s-%s" % (self.nvmet_subsystem_name,
|
||||
mock_testvol['id'])],
|
||||
"referrals": [],
|
||||
"portid": self.nvmet_port_id,
|
||||
"addr":
|
||||
{"treq": "not specified",
|
||||
"trtype": self.nvme_transport_type,
|
||||
"adrfam": "ipv4",
|
||||
"trsvcid": self.target_port,
|
||||
"traddr": self.target_ip}}
|
||||
]
|
||||
}
|
||||
mock_get_nvmf_subsystem.return_value = (
|
||||
"nqn.%s-%s" % (self.nvmet_subsystem_name,
|
||||
mock_testvol['id']))
|
||||
moc_delete_nvmf_subsystem.return_value = None
|
||||
self.assertRaises(nvmet.NVMETTargetDeleteError,
|
||||
self.target.remove_export,
|
||||
ctxt,
|
||||
mock_testvol)
|
||||
@mock.patch.object(priv_nvmet, 'Port')
|
||||
def test__ensure_port_exports_port(self, mock_port):
|
||||
"""Creates the port and export the subsystem when they don't exist."""
|
||||
nqn = 'nqn.nvme-subsystem-1-vol-2-uuid'
|
||||
mock_port.side_effect = priv_nvmet.NotFound
|
||||
self.target._ensure_port_exports(nqn,
|
||||
mock.sentinel.addr,
|
||||
mock.sentinel.port,
|
||||
mock.sentinel.transport,
|
||||
mock.sentinel.port_id)
|
||||
mock_port.assert_called_once_with(mock.sentinel.port_id)
|
||||
new_port = {'addr': {'adrfam': 'ipv4',
|
||||
'traddr': mock.sentinel.addr,
|
||||
'treq': 'not specified',
|
||||
'trsvcid': mock.sentinel.port,
|
||||
'trtype': mock.sentinel.transport},
|
||||
'portid': mock.sentinel.port_id,
|
||||
'referrals': [],
|
||||
'subsystems': [nqn]}
|
||||
mock_port.setup.assert_called_once_with(self.target._nvmet_root,
|
||||
new_port)
|
||||
mock_port.return_value.assert_not_called()
|
||||
|
||||
@mock.patch.object(nvmet.NVMET, 'delete_nvmeof_target')
|
||||
def test_remove_export(self, mock_delete_target):
|
||||
"""Test that the nvmeof class calls the nvmet method."""
|
||||
res = self.target.remove_export(mock.sentinel.ctxt,
|
||||
mock.sentinel.volume)
|
||||
self.assertEqual(mock_delete_target.return_value, res)
|
||||
mock_delete_target.assert_called_once_with(mock.sentinel.volume)
|
||||
|
||||
@mock.patch.object(priv_nvmet, 'Subsystem')
|
||||
@mock.patch.object(nvmet.NVMET, '_get_target_nqn')
|
||||
def test_delete_nvmeof_target_nothing_present(self, mock_nqn, mock_subsys):
|
||||
"""Delete doesn't do anything because there is nothing to do."""
|
||||
mock_nqn.return_value = mock.sentinel.nqn
|
||||
mock_subsys.side_effect = priv_nvmet.NotFound
|
||||
|
||||
port1 = mock.Mock(subsystems=[])
|
||||
port2 = mock.Mock(subsystems=['subs1'])
|
||||
self.mock_object(priv_nvmet.Root, 'ports', [port1, port2])
|
||||
|
||||
volume = mock.Mock(id='vol-uuid')
|
||||
self.target.delete_nvmeof_target(volume)
|
||||
|
||||
mock_nqn.assert_called_once_with(volume.id)
|
||||
port1.remove_subsystem.assert_not_called()
|
||||
port2.remove_subsystem.assert_not_called()
|
||||
mock_subsys.delete.assert_not_called()
|
||||
|
||||
@mock.patch.object(priv_nvmet, 'Subsystem')
|
||||
@mock.patch.object(nvmet.NVMET, '_get_target_nqn')
|
||||
def test_delete_nvmeof_target(self, mock_nqn, mock_subsys):
|
||||
"""Delete removes subsystems from port and the subsystem."""
|
||||
mock_nqn.return_value = mock.sentinel.nqn
|
||||
|
||||
port1 = mock.Mock(subsystems=[])
|
||||
port2 = mock.Mock(subsystems=[mock.sentinel.nqn])
|
||||
port3 = mock.Mock(subsystems=['subs1'])
|
||||
self.mock_object(priv_nvmet.Root, 'ports', [port1, port2, port3])
|
||||
|
||||
volume = mock.Mock(id='vol-uuid')
|
||||
self.target.delete_nvmeof_target(volume)
|
||||
|
||||
mock_nqn.assert_called_once_with(volume.id)
|
||||
port1.remove_subsystem.assert_not_called()
|
||||
port2.remove_subsystem.assert_called_once_with(mock.sentinel.nqn)
|
||||
port3.remove_subsystem.assert_not_called()
|
||||
mock_subsys.assert_called_once_with(mock.sentinel.nqn)
|
||||
mock_subsys.return_value.delete.assert_called_once_with()
|
||||
|
||||
@mock.patch.object(priv_nvmet, 'Root')
|
||||
def test__get_available_nvmf_subsystems(self, mock_root):
|
||||
res = self.target._get_available_nvmf_subsystems()
|
||||
mock_dump = mock_root.return_value.dump
|
||||
self.assertEqual(mock_dump.return_value, res)
|
||||
mock_dump.assert_called_once_with()
|
||||
|
||||
def test__get_target_nqn(self):
|
||||
res = self.target._get_target_nqn('volume_id')
|
||||
self.assertEqual('nqn.nvme-subsystem-1-volume_id', res)
|
||||
|
@ -10,17 +10,11 @@
|
||||
# License for the specific language governing permissions and limitations
|
||||
# under the License.
|
||||
|
||||
import tempfile
|
||||
|
||||
from oslo_concurrency import processutils as putils
|
||||
from oslo_log import log as logging
|
||||
from oslo_serialization import jsonutils as json
|
||||
from oslo_utils import excutils
|
||||
from oslo_utils import uuidutils
|
||||
|
||||
from cinder import exception
|
||||
from cinder.privsep import nvmcli
|
||||
import cinder.privsep.path
|
||||
from cinder.privsep.targets import nvmet
|
||||
from cinder import utils
|
||||
from cinder.volume.targets import nvmeof
|
||||
|
||||
@ -38,81 +32,49 @@ class NVMETTargetDeleteError(exception.CinderException):
|
||||
|
||||
class NVMET(nvmeof.NVMeOF):
|
||||
|
||||
def __init__(self, *args, **kwargs):
|
||||
super().__init__(*args, **kwargs)
|
||||
self._nvmet_root = nvmet.Root()
|
||||
|
||||
@utils.synchronized('nvmetcli', external=True)
|
||||
def create_nvmeof_target(self,
|
||||
volume_id,
|
||||
subsystem_name,
|
||||
subsystem_name, # Ignoring this, using config
|
||||
target_ip,
|
||||
target_port,
|
||||
transport_type,
|
||||
nvmet_port_id,
|
||||
ns_id,
|
||||
volume_path):
|
||||
|
||||
# Create NVME subsystem for previously created LV
|
||||
nvmf_subsystems = self._get_available_nvmf_subsystems()
|
||||
nqn = self._get_target_nqn(volume_id)
|
||||
try:
|
||||
self._ensure_subsystem_exists(nqn, ns_id, volume_path)
|
||||
self._ensure_port_exports(nqn, target_ip, target_port,
|
||||
transport_type, nvmet_port_id)
|
||||
except Exception:
|
||||
LOG.error('Failed to add subsystem: %s', nqn)
|
||||
raise NVMETTargetAddError(subsystem=nqn)
|
||||
|
||||
# Check if subsystem already exists
|
||||
search_for_subsystem = self._get_nvmf_subsystem(
|
||||
nvmf_subsystems, volume_id)
|
||||
if search_for_subsystem is None:
|
||||
newly_added_subsystem = self._add_nvmf_subsystem(
|
||||
nvmf_subsystems,
|
||||
target_ip,
|
||||
target_port,
|
||||
nvmet_port_id,
|
||||
subsystem_name,
|
||||
ns_id, volume_id, volume_path, transport_type)
|
||||
if newly_added_subsystem is None:
|
||||
LOG.error('Failed to add subsystem: %s', subsystem_name)
|
||||
raise NVMETTargetAddError(subsystem=subsystem_name)
|
||||
LOG.info('Added subsystem: %s', newly_added_subsystem)
|
||||
search_for_subsystem = newly_added_subsystem
|
||||
else:
|
||||
LOG.debug('Skip creating subsystem %s as '
|
||||
'it already exists.', search_for_subsystem)
|
||||
LOG.info('Subsystem %s now exported on port %s', nqn, target_port)
|
||||
return {
|
||||
'location': self.get_nvmeof_location(
|
||||
search_for_subsystem,
|
||||
nqn,
|
||||
target_ip,
|
||||
target_port,
|
||||
transport_type,
|
||||
ns_id),
|
||||
'auth': ''}
|
||||
|
||||
def _restore(self, nvmf_subsystems):
|
||||
# Dump updated JSON dict to append new subsystem
|
||||
with tempfile.NamedTemporaryFile(mode='w') as tmp_fd:
|
||||
tmp_fd.write(json.dumps(nvmf_subsystems))
|
||||
tmp_fd.flush()
|
||||
def _ensure_subsystem_exists(self, nqn, nvmet_ns_id, volume_path):
|
||||
# Assume if subsystem exists, it has the right configuration
|
||||
try:
|
||||
out, err = nvmcli.restore(tmp_fd.name)
|
||||
except putils.ProcessExecutionError:
|
||||
with excutils.save_and_reraise_exception():
|
||||
LOG.exception('Error from nvmetcli restore')
|
||||
nvmet.Subsystem(nqn)
|
||||
LOG.debug('Skip creating subsystem %s as it already exists.', nqn)
|
||||
return
|
||||
except nvmet.NotFound:
|
||||
LOG.debug('Creating subsystem %s.', nqn)
|
||||
|
||||
def _add_nvmf_subsystem(self, nvmf_subsystems, target_ip, target_port,
|
||||
nvmet_port_id, nvmet_subsystem_name, nvmet_ns_id,
|
||||
volume_id, volume_path, transport_type):
|
||||
|
||||
subsystem_name = self._get_target_info(nvmet_subsystem_name, volume_id)
|
||||
# Create JSON sections for the new subsystem to be created
|
||||
# Port section
|
||||
port_section = {
|
||||
"addr": {
|
||||
"adrfam": "ipv4",
|
||||
"traddr": target_ip,
|
||||
"treq": "not specified",
|
||||
"trsvcid": target_port,
|
||||
"trtype": transport_type,
|
||||
},
|
||||
"portid": nvmet_port_id,
|
||||
"referrals": [],
|
||||
"subsystems": [subsystem_name]
|
||||
}
|
||||
nvmf_subsystems['ports'].append(port_section)
|
||||
|
||||
# Subsystem section
|
||||
subsystem_section = {
|
||||
"allowed_hosts": [],
|
||||
"attr": {
|
||||
@ -128,92 +90,70 @@ class NVMET(nvmeof.NVMeOF):
|
||||
"nsid": nvmet_ns_id
|
||||
}
|
||||
],
|
||||
"nqn": subsystem_name}
|
||||
nvmf_subsystems['subsystems'].append(subsystem_section)
|
||||
"nqn": nqn}
|
||||
|
||||
LOG.info(
|
||||
'Trying to load the following subsystems: %s', nvmf_subsystems)
|
||||
nvmet.Subsystem.setup(subsystem_section) # privsep
|
||||
LOG.debug('Added subsystem: %s', nqn)
|
||||
|
||||
self._restore(nvmf_subsystems)
|
||||
def _ensure_port_exports(self, nqn, addr, port, transport_type, port_id):
|
||||
# Assume if port exists, it has the right configuration
|
||||
try:
|
||||
port = nvmet.Port(port_id)
|
||||
LOG.debug('Skip creating port %s as it already exists.', port_id)
|
||||
except nvmet.NotFound:
|
||||
LOG.debug('Creating port %s.', port_id)
|
||||
|
||||
return subsystem_name
|
||||
# Port section
|
||||
port_section = {
|
||||
"addr": {
|
||||
"adrfam": "ipv4",
|
||||
"traddr": addr,
|
||||
"treq": "not specified",
|
||||
"trsvcid": port,
|
||||
"trtype": transport_type,
|
||||
},
|
||||
"portid": port_id,
|
||||
"referrals": [],
|
||||
"subsystems": [nqn]
|
||||
}
|
||||
nvmet.Port.setup(self._nvmet_root, port_section) # privsep
|
||||
LOG.debug('Added port: %s', port_id)
|
||||
|
||||
else:
|
||||
if nqn in port.subsystems:
|
||||
LOG.debug('%s already exported on port %s', nqn, port_id)
|
||||
else:
|
||||
port.add_subsystem(nqn) # privsep
|
||||
LOG.debug('Exported %s on port %s', nqn, port_id)
|
||||
|
||||
@utils.synchronized('nvmetcli', external=True)
|
||||
def delete_nvmeof_target(self, volume):
|
||||
nvmf_subsystems = self._get_available_nvmf_subsystems()
|
||||
subsystem_name = self._get_nvmf_subsystem(
|
||||
nvmf_subsystems, volume['id'])
|
||||
if subsystem_name:
|
||||
removed_subsystem = self._delete_nvmf_subsystem(
|
||||
nvmf_subsystems, subsystem_name)
|
||||
if removed_subsystem is None:
|
||||
LOG.error(
|
||||
'Failed to delete subsystem: %s', subsystem_name)
|
||||
subsystem_name = self._get_target_nqn(volume.id)
|
||||
LOG.debug('Removing subsystem: %s', subsystem_name)
|
||||
|
||||
for port in self._nvmet_root.ports:
|
||||
if subsystem_name in port.subsystems:
|
||||
LOG.debug('Removing %s from port %s',
|
||||
subsystem_name, port.portid)
|
||||
port.remove_subsystem(subsystem_name)
|
||||
|
||||
try:
|
||||
subsys = nvmet.Subsystem(subsystem_name)
|
||||
LOG.debug('Deleting %s', subsystem_name)
|
||||
subsys.delete() # privsep call
|
||||
LOG.info('Subsystem %s removed', subsystem_name)
|
||||
except nvmet.NotFound:
|
||||
LOG.info('Skipping remove_export. No NVMe subsystem for volume: '
|
||||
'%s', volume.id)
|
||||
except Exception:
|
||||
LOG.error('Failed to delete subsystem: %s', subsystem_name)
|
||||
raise NVMETTargetDeleteError(subsystem=subsystem_name)
|
||||
elif removed_subsystem == subsystem_name:
|
||||
LOG.info(
|
||||
'Managed to delete subsystem: %s', subsystem_name)
|
||||
return removed_subsystem
|
||||
else:
|
||||
LOG.info("Skipping remove_export. No NVMe subsystem "
|
||||
"for volume: %s", volume['id'])
|
||||
|
||||
def _delete_nvmf_subsystem(self, nvmf_subsystems, subsystem_name):
|
||||
LOG.debug(
|
||||
'Removing this subsystem: %s', subsystem_name)
|
||||
|
||||
for port in nvmf_subsystems['ports']:
|
||||
if subsystem_name in port['subsystems']:
|
||||
port['subsystems'].remove(subsystem_name)
|
||||
break
|
||||
for subsys in nvmf_subsystems['subsystems']:
|
||||
if subsys['nqn'] == subsystem_name:
|
||||
nvmf_subsystems['subsystems'].remove(subsys)
|
||||
break
|
||||
|
||||
LOG.debug(
|
||||
'Newly loaded subsystems will be: %s', nvmf_subsystems)
|
||||
self._restore(nvmf_subsystems)
|
||||
return subsystem_name
|
||||
|
||||
def _get_nvmf_subsystem(self, nvmf_subsystems, volume_id):
|
||||
subsystem_name = self._get_target_info(
|
||||
self.nvmet_subsystem_name, volume_id)
|
||||
for subsys in nvmf_subsystems['subsystems']:
|
||||
if subsys['nqn'] == subsystem_name:
|
||||
return subsystem_name
|
||||
LOG.info('Volume %s is no longer exported', volume.id)
|
||||
|
||||
def _get_available_nvmf_subsystems(self):
|
||||
__, tmp_file_path = tempfile.mkstemp(prefix='nvmet')
|
||||
nvme_root = nvmet.Root()
|
||||
subsystems = nvme_root.dump()
|
||||
return subsystems
|
||||
|
||||
# nvmetcli doesn't support printing to stdout yet,
|
||||
try:
|
||||
out, err = nvmcli.save(tmp_file_path)
|
||||
except putils.ProcessExecutionError:
|
||||
with excutils.save_and_reraise_exception():
|
||||
LOG.exception('Error from nvmetcli save')
|
||||
self._delete_file(tmp_file_path)
|
||||
|
||||
# temp file must be readable by this process user
|
||||
# in order to avoid executing cat as root
|
||||
with utils.temporary_chown(tmp_file_path):
|
||||
try:
|
||||
out = cinder.privsep.path.readfile(tmp_file_path)
|
||||
except putils.ProcessExecutionError:
|
||||
with excutils.save_and_reraise_exception():
|
||||
LOG.exception('Failed to read: %s', tmp_file_path)
|
||||
self._delete_file(tmp_file_path)
|
||||
nvmf_subsystems = json.loads(out)
|
||||
|
||||
self._delete_file(tmp_file_path)
|
||||
|
||||
return nvmf_subsystems
|
||||
|
||||
def _get_target_info(self, subsystem, volume_id):
|
||||
return "nqn.%s-%s" % (subsystem, volume_id)
|
||||
|
||||
def _delete_file(self, file_path):
|
||||
try:
|
||||
cinder.privsep.path.removefile(file_path)
|
||||
except putils.ProcessExecutionError:
|
||||
LOG.exception('Failed to delete file: %s', file_path)
|
||||
def _get_target_nqn(self, volume_id):
|
||||
return "nqn.%s-%s" % (self.nvmet_subsystem_name, volume_id)
|
||||
|
@ -73,6 +73,9 @@ config_generator_config_file = (
|
||||
'../../tools/config/cinder-config-generator.conf')
|
||||
sample_config_basename = '_static/cinder'
|
||||
|
||||
# These are driver specific libraries that are not always present
|
||||
autodoc_mock_imports = ['nvmet']
|
||||
|
||||
policy_generator_config_file = (
|
||||
'../../tools/config/cinder-policy-generator.conf')
|
||||
sample_policy_basename = '_static/cinder'
|
||||
|
12
releasenotes/notes/lvm-nvmet-fixes-fc5e867abc699633.yaml
Normal file
12
releasenotes/notes/lvm-nvmet-fixes-fc5e867abc699633.yaml
Normal file
@ -0,0 +1,12 @@
|
||||
---
|
||||
fixes:
|
||||
- |
|
||||
LVM nvmet target `bug #1964391
|
||||
<https://bugs.launchpad.net/cinder/+bug/1964391>`_: Fixed temporary
|
||||
disconnection of all volumes from all hosts when creating and removing
|
||||
volume exports.
|
||||
|
||||
- |
|
||||
LVM nvmet target `bug #1964394
|
||||
<https://bugs.launchpad.net/cinder/+bug/1964394>`_: Fixed annoying kernel
|
||||
log message when exporting a volume.
|
Loading…
Reference in New Issue
Block a user