Convert classic drivers to hardware types on enrollment
Change-Id: I42983a9a7129fcb859fba6cef97405635152563e
This commit is contained in:
parent
61b501151d
commit
274b2d3253
@ -0,0 +1,7 @@
|
|||||||
|
---
|
||||||
|
features:
|
||||||
|
- |
|
||||||
|
On enrollment, all classic drivers are replaced with their hardware type
|
||||||
|
equivalents (e.g. ``pxe_ipmitool`` is replaced with ``ipmi``).
|
||||||
|
The ``fake_pxe`` classic driver is replaced with the ``manual-management``
|
||||||
|
hardware type (which must be enabled in the undercloud).
|
@ -667,12 +667,29 @@ class NodesTest(base.TestCase):
|
|||||||
node['pm_type'] = 'fake_pxe'
|
node['pm_type'] = 'fake_pxe'
|
||||||
client = mock.MagicMock()
|
client = mock.MagicMock()
|
||||||
nodes.register_ironic_node(node, client=client)
|
nodes.register_ironic_node(node, client=client)
|
||||||
client.node.create.assert_called_once_with(driver='fake_pxe',
|
client.node.create.assert_called_once_with(driver='manual-management',
|
||||||
name='node1',
|
name='node1',
|
||||||
properties=node_properties,
|
properties=node_properties,
|
||||||
resource_class='baremetal',
|
resource_class='baremetal',
|
||||||
driver_info={})
|
driver_info={})
|
||||||
|
|
||||||
|
def test_register_ironic_node_ucs(self):
|
||||||
|
node_properties = {"cpus": "1",
|
||||||
|
"memory_mb": "2048",
|
||||||
|
"local_gb": "30",
|
||||||
|
"cpu_arch": "amd64",
|
||||||
|
"capabilities": "num_nics:6"}
|
||||||
|
node = self._get_node()
|
||||||
|
node['pm_type'] = 'cisco-ucs-managed'
|
||||||
|
client = mock.MagicMock()
|
||||||
|
nodes.register_ironic_node(node, client=client)
|
||||||
|
client.node.create.assert_called_once_with(
|
||||||
|
driver='cisco-ucs-managed', name='node1',
|
||||||
|
properties=node_properties,
|
||||||
|
resource_class='baremetal',
|
||||||
|
driver_info={'ucs_password': 'random', 'ucs_address': 'foo.bar',
|
||||||
|
'ucs_username': 'test'})
|
||||||
|
|
||||||
def test_register_ironic_node_pxe_ucs(self):
|
def test_register_ironic_node_pxe_ucs(self):
|
||||||
node_properties = {"cpus": "1",
|
node_properties = {"cpus": "1",
|
||||||
"memory_mb": "2048",
|
"memory_mb": "2048",
|
||||||
@ -684,7 +701,8 @@ class NodesTest(base.TestCase):
|
|||||||
client = mock.MagicMock()
|
client = mock.MagicMock()
|
||||||
nodes.register_ironic_node(node, client=client)
|
nodes.register_ironic_node(node, client=client)
|
||||||
client.node.create.assert_called_once_with(
|
client.node.create.assert_called_once_with(
|
||||||
driver='pxe_ucs', name='node1', properties=node_properties,
|
driver='cisco-ucs-managed', name='node1',
|
||||||
|
properties=node_properties,
|
||||||
resource_class='baremetal',
|
resource_class='baremetal',
|
||||||
driver_info={'ucs_password': 'random', 'ucs_address': 'foo.bar',
|
driver_info={'ucs_password': 'random', 'ucs_address': 'foo.bar',
|
||||||
'ucs_username': 'test'})
|
'ucs_username': 'test'})
|
||||||
@ -718,7 +736,7 @@ class NodesTest(base.TestCase):
|
|||||||
client = mock.MagicMock()
|
client = mock.MagicMock()
|
||||||
nodes.register_ironic_node(node, client=client)
|
nodes.register_ironic_node(node, client=client)
|
||||||
client.node.create.assert_called_once_with(
|
client.node.create.assert_called_once_with(
|
||||||
driver='pxe_ipmitool', name='node1', properties=node_properties,
|
driver='ipmi', name='node1', properties=node_properties,
|
||||||
resource_class='baremetal',
|
resource_class='baremetal',
|
||||||
driver_info={'ipmi_password': 'random', 'ipmi_address': 'foo.bar',
|
driver_info={'ipmi_password': 'random', 'ipmi_address': 'foo.bar',
|
||||||
'ipmi_username': 'test', 'ipmi_port': '6230'})
|
'ipmi_username': 'test', 'ipmi_port': '6230'})
|
||||||
@ -769,11 +787,28 @@ class NodesTest(base.TestCase):
|
|||||||
client = mock.MagicMock()
|
client = mock.MagicMock()
|
||||||
nodes.register_ironic_node(node, client=client)
|
nodes.register_ironic_node(node, client=client)
|
||||||
client.node.create.assert_called_once_with(
|
client.node.create.assert_called_once_with(
|
||||||
driver='pxe_drac', name='node1', properties=node_properties,
|
driver='idrac', name='node1', properties=node_properties,
|
||||||
resource_class='baremetal',
|
resource_class='baremetal',
|
||||||
driver_info={'drac_password': 'random', 'drac_address': 'foo.bar',
|
driver_info={'drac_password': 'random', 'drac_address': 'foo.bar',
|
||||||
'drac_username': 'test', 'drac_port': '6230'})
|
'drac_username': 'test', 'drac_port': '6230'})
|
||||||
|
|
||||||
|
def test_register_ironic_node_pxe_ilo(self):
|
||||||
|
node_properties = {"cpus": "1",
|
||||||
|
"memory_mb": "2048",
|
||||||
|
"local_gb": "30",
|
||||||
|
"cpu_arch": "amd64",
|
||||||
|
"capabilities": "num_nics:6"}
|
||||||
|
node = self._get_node()
|
||||||
|
node['pm_type'] = 'pxe_ilo'
|
||||||
|
node['pm_port'] = '1234'
|
||||||
|
client = mock.MagicMock()
|
||||||
|
nodes.register_ironic_node(node, client=client)
|
||||||
|
client.node.create.assert_called_once_with(
|
||||||
|
driver='ilo', name='node1', properties=node_properties,
|
||||||
|
resource_class='baremetal',
|
||||||
|
driver_info={'ilo_password': 'random', 'ilo_address': 'foo.bar',
|
||||||
|
'ilo_username': 'test', 'ilo_port': '1234'})
|
||||||
|
|
||||||
def test_register_ironic_node_redfish(self):
|
def test_register_ironic_node_redfish(self):
|
||||||
node_properties = {"cpus": "1",
|
node_properties = {"cpus": "1",
|
||||||
"memory_mb": "2048",
|
"memory_mb": "2048",
|
||||||
@ -833,16 +868,16 @@ class NodesTest(base.TestCase):
|
|||||||
nodes._clean_up_extra_nodes(seen, client, remove=True)
|
nodes._clean_up_extra_nodes(seen, client, remove=True)
|
||||||
client.node.delete.assert_called_once_with('foobar')
|
client.node.delete.assert_called_once_with('foobar')
|
||||||
|
|
||||||
def test__get_node_id_fake_pxe(self):
|
def test__get_node_id_manual_management(self):
|
||||||
node = self._get_node()
|
node = self._get_node()
|
||||||
node['pm_type'] = 'fake_pxe'
|
node['pm_type'] = 'manual-management'
|
||||||
handler = nodes.find_driver_handler('fake_pxe')
|
handler = nodes.find_driver_handler('manual-management')
|
||||||
node_map = {'mac': {'aaa': 'abcdef'}, 'pm_addr': {}}
|
node_map = {'mac': {'aaa': 'abcdef'}, 'pm_addr': {}}
|
||||||
self.assertEqual('abcdef', nodes._get_node_id(node, handler, node_map))
|
self.assertEqual('abcdef', nodes._get_node_id(node, handler, node_map))
|
||||||
|
|
||||||
def test__get_node_id_conflict(self):
|
def test__get_node_id_conflict(self):
|
||||||
node = self._get_node()
|
node = self._get_node()
|
||||||
handler = nodes.find_driver_handler('pxe_ipmitool')
|
handler = nodes.find_driver_handler('ipmi')
|
||||||
node_map = {'mac': {'aaa': 'abcdef'},
|
node_map = {'mac': {'aaa': 'abcdef'},
|
||||||
'pm_addr': {'foo.bar': 'defabc'}}
|
'pm_addr': {'foo.bar': 'defabc'}}
|
||||||
self.assertRaises(exception.InvalidNode,
|
self.assertRaises(exception.InvalidNode,
|
||||||
@ -851,7 +886,7 @@ class NodesTest(base.TestCase):
|
|||||||
|
|
||||||
def test_get_node_id_valid_duplicate(self):
|
def test_get_node_id_valid_duplicate(self):
|
||||||
node = self._get_node()
|
node = self._get_node()
|
||||||
handler = nodes.find_driver_handler('pxe_ipmitool')
|
handler = nodes.find_driver_handler('ipmi')
|
||||||
node_map = {'mac': {'aaa': 'id'},
|
node_map = {'mac': {'aaa': 'id'},
|
||||||
'pm_addr': {'foo.bar': 'id'}}
|
'pm_addr': {'foo.bar': 'id'}}
|
||||||
self.assertEqual('id', nodes._get_node_id(node, handler, node_map))
|
self.assertEqual('id', nodes._get_node_id(node, handler, node_map))
|
||||||
@ -874,12 +909,12 @@ class TestPopulateNodeMapping(base.TestCase):
|
|||||||
'uuids': {'abcdef', 'fedcba', 'xyz'}}
|
'uuids': {'abcdef', 'fedcba', 'xyz'}}
|
||||||
self.assertEqual(expected, nodes._populate_node_mapping(client))
|
self.assertEqual(expected, nodes._populate_node_mapping(client))
|
||||||
|
|
||||||
def test_populate_node_mapping_ironic_fake_pxe(self):
|
def test_populate_node_mapping_ironic_manual_management(self):
|
||||||
client = mock.MagicMock()
|
client = mock.MagicMock()
|
||||||
ironic_node = collections.namedtuple('node', ['uuid', 'driver',
|
ironic_node = collections.namedtuple('node', ['uuid', 'driver',
|
||||||
'driver_info'])
|
'driver_info'])
|
||||||
ironic_port = collections.namedtuple('port', ['address'])
|
ironic_port = collections.namedtuple('port', ['address'])
|
||||||
node = ironic_node('abcdef', 'fake_pxe', None)
|
node = ironic_node('abcdef', 'manual-management', None)
|
||||||
client.node.list_ports.return_value = [ironic_port('aaa')]
|
client.node.list_ports.return_value = [ironic_port('aaa')]
|
||||||
client.node.list.return_value = [node]
|
client.node.list.return_value = [node]
|
||||||
expected = {'mac': {'aaa': 'abcdef'}, 'pm_addr': {},
|
expected = {'mac': {'aaa': 'abcdef'}, 'pm_addr': {},
|
||||||
|
@ -54,17 +54,22 @@ class DriverInfo(object):
|
|||||||
DEFAULTS = {}
|
DEFAULTS = {}
|
||||||
|
|
||||||
def __init__(self, prefix, mapping, deprecated_mapping=None,
|
def __init__(self, prefix, mapping, deprecated_mapping=None,
|
||||||
mandatory_fields=(), default_port=None):
|
mandatory_fields=(), default_port=None, hardware_type=None):
|
||||||
self._prefix = prefix
|
self._prefix = prefix
|
||||||
self._mapping = mapping
|
self._mapping = mapping
|
||||||
self._deprecated_mapping = deprecated_mapping or {}
|
self._deprecated_mapping = deprecated_mapping or {}
|
||||||
self._mandatory_fields = mandatory_fields
|
self._mandatory_fields = mandatory_fields
|
||||||
self._default_port = default_port
|
self._default_port = default_port
|
||||||
|
self._hardware_type = hardware_type
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def default_port(self):
|
def default_port(self):
|
||||||
return self._default_port
|
return self._default_port
|
||||||
|
|
||||||
|
@property
|
||||||
|
def hardware_type(self):
|
||||||
|
return self._hardware_type
|
||||||
|
|
||||||
def convert_key(self, key):
|
def convert_key(self, key):
|
||||||
if key in self._mapping:
|
if key in self._mapping:
|
||||||
return self._mapping[key]
|
return self._mapping[key]
|
||||||
@ -116,7 +121,7 @@ class DriverInfo(object):
|
|||||||
class PrefixedDriverInfo(DriverInfo):
|
class PrefixedDriverInfo(DriverInfo):
|
||||||
def __init__(self, prefix, deprecated_mapping=None,
|
def __init__(self, prefix, deprecated_mapping=None,
|
||||||
has_port=False, address_field='address',
|
has_port=False, address_field='address',
|
||||||
default_port=None):
|
default_port=None, hardware_type=None):
|
||||||
mapping = {
|
mapping = {
|
||||||
'pm_addr': '%s_%s' % (prefix, address_field),
|
'pm_addr': '%s_%s' % (prefix, address_field),
|
||||||
'pm_user': '%s_username' % prefix,
|
'pm_user': '%s_username' % prefix,
|
||||||
@ -133,6 +138,7 @@ class PrefixedDriverInfo(DriverInfo):
|
|||||||
deprecated_mapping=deprecated_mapping,
|
deprecated_mapping=deprecated_mapping,
|
||||||
mandatory_fields=mandatory_fields,
|
mandatory_fields=mandatory_fields,
|
||||||
default_port=default_port,
|
default_port=default_port,
|
||||||
|
hardware_type=hardware_type,
|
||||||
)
|
)
|
||||||
|
|
||||||
def unique_id_from_fields(self, fields):
|
def unique_id_from_fields(self, fields):
|
||||||
@ -179,6 +185,7 @@ class RedfishDriverInfo(DriverInfo):
|
|||||||
'redfish', mapping,
|
'redfish', mapping,
|
||||||
deprecated_mapping=None,
|
deprecated_mapping=None,
|
||||||
mandatory_fields=mandatory_fields,
|
mandatory_fields=mandatory_fields,
|
||||||
|
hardware_type='redfish',
|
||||||
)
|
)
|
||||||
|
|
||||||
def _build_id(self, address, system):
|
def _build_id(self, address, system):
|
||||||
@ -211,6 +218,7 @@ class oVirtDriverInfo(DriverInfo):
|
|||||||
super(oVirtDriverInfo, self).__init__(
|
super(oVirtDriverInfo, self).__init__(
|
||||||
'ovirt', mapping,
|
'ovirt', mapping,
|
||||||
mandatory_fields=list(mapping),
|
mandatory_fields=list(mapping),
|
||||||
|
hardware_type='staging-ovirt',
|
||||||
)
|
)
|
||||||
|
|
||||||
def unique_id_from_fields(self, fields):
|
def unique_id_from_fields(self, fields):
|
||||||
@ -227,38 +235,14 @@ class oVirtDriverInfo(DriverInfo):
|
|||||||
return
|
return
|
||||||
|
|
||||||
|
|
||||||
class SshDriverInfo(DriverInfo):
|
|
||||||
DEFAULTS = {'ssh_virt_type': 'virsh'}
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
super(SshDriverInfo, self).__init__(
|
|
||||||
'ssh',
|
|
||||||
{
|
|
||||||
'pm_addr': 'ssh_address',
|
|
||||||
'pm_user': 'ssh_username',
|
|
||||||
# TODO(dtantsur): support ssh_key_filename as well
|
|
||||||
'pm_password': 'ssh_key_contents',
|
|
||||||
},
|
|
||||||
deprecated_mapping={
|
|
||||||
'pm_virt_type': 'ssh_virt_type',
|
|
||||||
},
|
|
||||||
mandatory_fields=['pm_addr', 'pm_user', 'pm_password'],
|
|
||||||
)
|
|
||||||
|
|
||||||
def validate(self, node):
|
|
||||||
super(SshDriverInfo, self).validate(node)
|
|
||||||
if not node.get('ports')[0]['address']:
|
|
||||||
raise exception.InvalidNode(
|
|
||||||
'Nodes with SSH drivers require at least one PORT')
|
|
||||||
|
|
||||||
|
|
||||||
class iBootDriverInfo(PrefixedDriverInfo):
|
class iBootDriverInfo(PrefixedDriverInfo):
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
super(iBootDriverInfo, self).__init__(
|
super(iBootDriverInfo, self).__init__(
|
||||||
'iboot', has_port=True,
|
'iboot', has_port=True,
|
||||||
deprecated_mapping={
|
deprecated_mapping={
|
||||||
'pm_relay_id': 'iboot_relay_id',
|
'pm_relay_id': 'iboot_relay_id',
|
||||||
}
|
},
|
||||||
|
hardware_type='staging-iboot',
|
||||||
)
|
)
|
||||||
|
|
||||||
def unique_id_from_fields(self, fields):
|
def unique_id_from_fields(self, fields):
|
||||||
@ -282,23 +266,35 @@ class iBootDriverInfo(PrefixedDriverInfo):
|
|||||||
DRIVER_INFO = {
|
DRIVER_INFO = {
|
||||||
# production drivers
|
# production drivers
|
||||||
'(ipmi|.*_ipmitool)': PrefixedDriverInfo('ipmi', has_port=True,
|
'(ipmi|.*_ipmitool)': PrefixedDriverInfo('ipmi', has_port=True,
|
||||||
default_port=623),
|
default_port=623,
|
||||||
'(idrac|.*_drac)': PrefixedDriverInfo('drac', has_port=True),
|
hardware_type='ipmi'),
|
||||||
'(ilo|.*_ilo)': PrefixedDriverInfo('ilo', has_port=True),
|
'(idrac|.*_drac)': PrefixedDriverInfo('drac', has_port=True,
|
||||||
'(cisco\-ucs\-managed|.*_ucs)': PrefixedDriverInfo('ucs'),
|
hardware_type='idrac'),
|
||||||
'(irmc|.*_irmc)': PrefixedDriverInfo('irmc', has_port=True),
|
'(ilo|.*_ilo)': PrefixedDriverInfo('ilo', has_port=True,
|
||||||
|
hardware_type='ilo'),
|
||||||
|
'(cisco\-ucs\-managed|.*_ucs)': PrefixedDriverInfo(
|
||||||
|
'ucs', hardware_type='cisco-ucs-managed'),
|
||||||
|
'(irmc|.*_irmc)': PrefixedDriverInfo('irmc', has_port=True,
|
||||||
|
hardware_type='irmc'),
|
||||||
'redfish': RedfishDriverInfo(),
|
'redfish': RedfishDriverInfo(),
|
||||||
# test drivers
|
# test drivers
|
||||||
'staging\-ovirt': oVirtDriverInfo(),
|
'staging\-ovirt': oVirtDriverInfo(),
|
||||||
'.*_iboot': iBootDriverInfo(),
|
'(staging\-iboot|.*_iboot)': iBootDriverInfo(),
|
||||||
'.*_wol': DriverInfo(
|
'(staging\-wol|.*wol)': DriverInfo(
|
||||||
'wol',
|
'wol',
|
||||||
mapping={
|
mapping={
|
||||||
'pm_addr': 'wol_host',
|
'pm_addr': 'wol_host',
|
||||||
'pm_port': 'wol_port',
|
'pm_port': 'wol_port',
|
||||||
}),
|
},
|
||||||
'.*_amt': PrefixedDriverInfo('amt'),
|
hardware_type='staging-wol'),
|
||||||
'fake(|_pxe|_agent)': DriverInfo('fake', mapping={}),
|
'(staging\-amt|.*_amt)': PrefixedDriverInfo('amt',
|
||||||
|
hardware_type='staging-amt'),
|
||||||
|
# fake_pxe was used when no management interface was supported, now
|
||||||
|
# manual-management is used for the same purpose
|
||||||
|
'(manual\-management|fake_pxe|fake_agent)': DriverInfo(
|
||||||
|
'fake', mapping={}, hardware_type='manual-management'),
|
||||||
|
'^fake(|\-hardware)$': DriverInfo('fake', mapping={},
|
||||||
|
hardware_type='fake-hardware'),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -359,7 +355,14 @@ def register_ironic_node(node, client):
|
|||||||
caps = dict_to_capabilities(caps)
|
caps = dict_to_capabilities(caps)
|
||||||
properties.update({"capabilities": six.text_type(caps)})
|
properties.update({"capabilities": six.text_type(caps)})
|
||||||
|
|
||||||
create_map = {"driver": node["pm_type"],
|
driver = node['pm_type']
|
||||||
|
if handler.hardware_type and handler.hardware_type != driver:
|
||||||
|
LOG.warning('Replacing deprecated driver %(old)s with the '
|
||||||
|
'hardware type %(new)s, please update your inventory',
|
||||||
|
{'old': driver, 'new': handler.hardware_type})
|
||||||
|
driver = handler.hardware_type
|
||||||
|
|
||||||
|
create_map = {"driver": driver,
|
||||||
"properties": properties,
|
"properties": properties,
|
||||||
"driver_info": driver_info,
|
"driver_info": driver_info,
|
||||||
"resource_class": resource_class}
|
"resource_class": resource_class}
|
||||||
|
Loading…
Reference in New Issue
Block a user