Move port creation to validate_interfaces hook

Currently port logic is placed inconsistently: port creation is in core processing,
while port deletion is in validate_interfaces before_update. This changes moves
port creation there as well. This should only affect deployments that tamper with
validation_interfaces hook, as previously port creation was run just before running
before_update hooks.

This allows deployments to replace port creation logic by replacing the
validate_interfaces hook.

Change-Id: Idd8f748fdf31fc694bd7b554837e509024716c18
Partial-Bug: #1667472
This commit is contained in:
Dmitry Tantsur 2017-04-07 11:47:26 +02:00
parent 5c28ca6584
commit 9575d31920
4 changed files with 61 additions and 26 deletions

View File

@ -252,7 +252,10 @@ class ValidateInterfacesHook(base.ProcessingHook):
introspection_data['macs'] = valid_macs introspection_data['macs'] = valid_macs
def before_update(self, introspection_data, node_info, **kwargs): def before_update(self, introspection_data, node_info, **kwargs):
"""Drop ports that are not present in the data.""" """Create new ports and drop ports that are not present in the data."""
interfaces = introspection_data.get('interfaces')
node_info.create_ports(list(interfaces.values()))
if CONF.processing.keep_ports == 'present': if CONF.processing.keep_ports == 'present':
expected_macs = { expected_macs = {
iface['mac'] iface['mac']

View File

@ -268,8 +268,6 @@ def _run_post_hooks(node_info, introspection_data):
def _process_node(node_info, node, introspection_data): def _process_node(node_info, node, introspection_data):
# NOTE(dtantsur): repeat the check in case something changed # NOTE(dtantsur): repeat the check in case something changed
ir_utils.check_provision_state(node) ir_utils.check_provision_state(node)
interfaces = introspection_data.get('interfaces')
node_info.create_ports(list(interfaces.values()))
_run_post_hooks(node_info, introspection_data) _run_post_hooks(node_info, introspection_data)
_store_data(node_info, introspection_data) _store_data(node_info, introspection_data)
@ -435,8 +433,6 @@ def _reapply_with_data(node_info, introspection_data):
'introspection on stored data:\n%s') % 'introspection on stored data:\n%s') %
'\n'.join(failures), node_info=node_info) '\n'.join(failures), node_info=node_info)
interfaces = introspection_data.get('interfaces')
node_info.create_ports(list(interfaces.values()))
_run_post_hooks(node_info, introspection_data) _run_post_hooks(node_info, introspection_data)
_store_data(node_info, introspection_data) _store_data(node_info, introspection_data)
node_info.invalidate_cache() node_info.invalidate_cache()

View File

@ -96,18 +96,7 @@ class TestSchedulerHook(test_base.NodeTest):
self.assertCalledWithPatch(patch, mock_patch) self.assertCalledWithPatch(patch, mock_patch)
class TestValidateInterfacesHook(test_base.NodeTest): class TestValidateInterfacesHookLoad(test_base.NodeTest):
def setUp(self):
super(TestValidateInterfacesHook, self).setUp()
self.hook = std_plugins.ValidateInterfacesHook()
self.existing_ports = [mock.Mock(spec=['address', 'uuid'],
address=a)
for a in (self.macs[1],
'44:44:44:44:44:44')]
self.node_info = node_cache.NodeInfo(uuid=self.uuid, started_at=0,
node=self.node,
ports=self.existing_ports)
def test_hook_loadable_by_name(self): def test_hook_loadable_by_name(self):
CONF.set_override('processing_hooks', 'validate_interfaces', CONF.set_override('processing_hooks', 'validate_interfaces',
'processing') 'processing')
@ -122,6 +111,12 @@ class TestValidateInterfacesHook(test_base.NodeTest):
CONF.set_override('keep_ports', 'foobar', 'processing') CONF.set_override('keep_ports', 'foobar', 'processing')
self.assertRaises(SystemExit, std_plugins.ValidateInterfacesHook) self.assertRaises(SystemExit, std_plugins.ValidateInterfacesHook)
class TestValidateInterfacesHookBeforeProcessing(test_base.NodeTest):
def setUp(self):
super(TestValidateInterfacesHookBeforeProcessing, self).setUp()
self.hook = std_plugins.ValidateInterfacesHook()
def test_no_interfaces(self): def test_no_interfaces(self):
self.assertRaisesRegex(utils.Error, self.assertRaisesRegex(utils.Error,
'Hardware inventory is empty or missing', 'Hardware inventory is empty or missing',
@ -212,27 +207,61 @@ class TestValidateInterfacesHook(test_base.NodeTest):
self.assertRaisesRegex(utils.Error, 'No suitable interfaces found', self.assertRaisesRegex(utils.Error, 'No suitable interfaces found',
self.hook.before_processing, self.data) self.hook.before_processing, self.data)
@mock.patch.object(node_cache.NodeInfo, 'delete_port', autospec=True)
def test_keep_all(self, mock_delete_port): @mock.patch.object(node_cache.NodeInfo, 'delete_port', autospec=True)
@mock.patch.object(node_cache.NodeInfo, 'create_ports', autospec=True)
class TestValidateInterfacesHookBeforeUpdate(test_base.NodeTest):
def setUp(self):
super(TestValidateInterfacesHookBeforeUpdate, self).setUp()
self.hook = std_plugins.ValidateInterfacesHook()
self.interfaces_to_create = sorted(self.valid_interfaces.values(),
key=lambda i: i['mac'])
self.existing_ports = [mock.Mock(spec=['address', 'uuid'],
address=a)
for a in (self.macs[1],
'44:44:44:44:44:44')]
self.node_info = node_cache.NodeInfo(uuid=self.uuid, started_at=0,
node=self.node,
ports=self.existing_ports)
def test_keep_all(self, mock_create_ports, mock_delete_port):
self.hook.before_update(self.data, self.node_info) self.hook.before_update(self.data, self.node_info)
# NOTE(dtantsur): dictionary ordering is not defined
mock_create_ports.assert_called_once_with(self.node_info, mock.ANY)
self.assertEqual(self.interfaces_to_create,
sorted(mock_create_ports.call_args[0][1],
key=lambda i: i['mac']))
self.assertFalse(mock_delete_port.called) self.assertFalse(mock_delete_port.called)
@mock.patch.object(node_cache.NodeInfo, 'delete_port') def test_keep_present(self, mock_create_ports, mock_delete_port):
def test_keep_present(self, mock_delete_port):
CONF.set_override('keep_ports', 'present', 'processing') CONF.set_override('keep_ports', 'present', 'processing')
self.data['all_interfaces'] = self.all_interfaces self.data['all_interfaces'] = self.all_interfaces
self.hook.before_update(self.data, self.node_info) self.hook.before_update(self.data, self.node_info)
mock_delete_port.assert_called_once_with(self.existing_ports[1]) mock_create_ports.assert_called_once_with(self.node_info, mock.ANY)
self.assertEqual(self.interfaces_to_create,
sorted(mock_create_ports.call_args[0][1],
key=lambda i: i['mac']))
@mock.patch.object(node_cache.NodeInfo, 'delete_port') mock_delete_port.assert_called_once_with(self.node_info,
def test_keep_added(self, mock_delete_port): self.existing_ports[1])
def test_keep_added(self, mock_create_ports, mock_delete_port):
CONF.set_override('keep_ports', 'added', 'processing') CONF.set_override('keep_ports', 'added', 'processing')
self.data['macs'] = [self.pxe_mac] self.data['macs'] = [self.pxe_mac]
self.hook.before_update(self.data, self.node_info) self.hook.before_update(self.data, self.node_info)
mock_delete_port.assert_any_call(self.existing_ports[0]) mock_create_ports.assert_called_once_with(self.node_info, mock.ANY)
mock_delete_port.assert_any_call(self.existing_ports[1]) self.assertEqual(self.interfaces_to_create,
sorted(mock_create_ports.call_args[0][1],
key=lambda i: i['mac']))
mock_delete_port.assert_any_call(self.node_info,
self.existing_ports[0])
mock_delete_port.assert_any_call(self.node_info,
self.existing_ports[1])
class TestRootDiskSelection(test_base.NodeTest): class TestRootDiskSelection(test_base.NodeTest):

View File

@ -0,0 +1,7 @@
---
upgrade:
- |
Ports creating logic was moved from core processing code to the
``validate_interfaces`` processing hook. This may affect deployments
that disable this hook or replace it with something else. Also make
sure to place this hook before any hooks expecting ports to be created.