Deprecate returning patches from plugins

This approach was an optimization, but it proved to bring more
troubles than benefits - see LP issue for details.

Precautions are made to avoid breakage. Still, some 3rdparty
hooks might get broken by this change.

Change-Id: I5a84512758e92c87091d6306c8d5baa944955e55
Closes-Bug: #1492946
This commit is contained in:
Dmitry Tantsur 2015-09-07 14:43:23 +02:00
parent f21eb0ab8c
commit 105aa64f60
14 changed files with 126 additions and 175 deletions

View File

@ -154,11 +154,14 @@ Writing a Plugin
called before any data processing, providing the raw data. Each plugin in
the chain can modify the data, so order in which plugins are loaded
matters here. Returns nothing.
``before_update(introspection_data,node_info,node_patches,ports_patches,**)``
``before_update(introspection_data,node_info,**)``
called after node is found and ports are created, but before data is
updated on a node. ``node_patches`` and ``ports_patches`` are JSON
patches for node and ports to apply.
Please refer to the docstring for details and examples.
updated on a node. Please refer to the docstring for details
and examples.
.. note::
Keyword arguments node_patches and port_patches are also provided, but
should not be used in new plugins.
Make your plugin a setuptools entry point under
``ironic_inspector.hooks.processing`` namespace and enable it in the

View File

@ -39,32 +39,17 @@ class ProcessingHook(object): # pragma: no cover
:returns: nothing.
"""
def before_update(self, introspection_data, node_info, node_patches,
ports_patches, **kwargs):
def before_update(self, introspection_data, node_info, **kwargs):
"""Hook to run before Ironic node update.
This hook is run after node is found and ports are created,
just before the node is updated with the data.
To update node and/or ports use *node_patches* and *ports_patches*
arguments.
:param introspection_data: processed data from the ramdisk.
:param node_info: NodeInfo instance.
:param node_patches: list of JSON patches [RFC 6902] to apply
to the node, e.g.
::
[{'op': 'add',
'path': '/extra/foo',
'value': 'bar'}]
:param ports_patches: dict where keys are port MAC's,
values are lists of JSON patches, e.g.
::
{'11:22:33:44:55:55': [
{'op': 'add', 'path': '/extra/foo',
'value': 'bar'}
]}
:param kwargs: used for extensibility without breaking existing hooks
:param kwargs: used for extensibility without breaking existing hooks,
currently deprecated arguments node_patches and
ports_patches are provided here.
:returns: nothing.
[RFC 6902] - http://tools.ietf.org/html/rfc6902

View File

@ -25,8 +25,7 @@ class ExampleProcessingHook(base.ProcessingHook): # pragma: no cover
def before_processing(self, introspection_data, **kwargs):
LOG.debug('before_processing: %s', introspection_data)
def before_update(self, introspection_data, node_info, node_patches,
ports_patches, **kwargs):
def before_update(self, introspection_data, node_info, **kwargs):
LOG.debug('before_update: %s (node %s)', introspection_data,
node_info.uuid)

View File

@ -41,8 +41,7 @@ class ExtraHardwareHook(base.ProcessingHook):
swift_api = swift.SwiftAPI()
swift_api.create_object(name, data)
def before_update(self, introspection_data, node_info, node_patches,
ports_patches, **kwargs):
def before_update(self, introspection_data, node_info, **kwargs):
"""Stores the 'data' key from introspection_data in Swift.
If the 'data' key exists, updates Ironic extra column
@ -60,6 +59,5 @@ class ExtraHardwareHook(base.ProcessingHook):
self._store_extra_hardware(name,
json.dumps(introspection_data['data']))
node_patches.append({'op': 'add',
'path': '/extra/hardware_swift_object',
'value': name})
node_info.patch([{'op': 'add', 'path': '/extra/hardware_swift_object',
'value': name}])

View File

@ -51,8 +51,7 @@ class RaidDeviceDetection(base.ProcessingHook):
'value for "local_gb"'))
introspection_data['local_gb'] = 1
def before_update(self, introspection_data, node_info, node_patches,
ports_patches, **kwargs):
def before_update(self, introspection_data, node_info, **kwargs):
if 'block_devices' not in introspection_data:
LOG.warning(_LW('No block device was received from ramdisk'))
return
@ -78,7 +77,7 @@ class RaidDeviceDetection(base.ProcessingHook):
LOG.warning(_LW('No new devices were found'))
return
node_patches.extend([
node_info.patch([
{'op': 'remove',
'path': '/extra/block_devices'},
{'op': 'add',
@ -89,6 +88,6 @@ class RaidDeviceDetection(base.ProcessingHook):
else:
# No previously discovered devices - save the inspector block
# devices in node.extra
node_patches.append({'op': 'add',
'path': '/extra/block_devices',
'value': introspection_data['block_devices']})
node_info.patch([{'op': 'add',
'path': '/extra/block_devices',
'value': introspection_data['block_devices']}])

View File

@ -49,15 +49,13 @@ class SchedulerHook(base.ProcessingHook):
'memory %(memory_mb)s MiB, disk %(local_gb)s GiB'),
{key: introspection_data.get(key) for key in self.KEYS})
def before_update(self, introspection_data, node_info, node_patches,
ports_patches, **kwargs):
def before_update(self, introspection_data, node_info, **kwargs):
"""Update node with scheduler properties."""
overwrite = CONF.processing.overwrite_existing
patches = [{'op': 'add', 'path': '/properties/%s' % key,
'value': str(introspection_data[key])}
for key in self.KEYS
if overwrite or not node_info.node().properties.get(key)]
node_patches.extend(patches)
properties = {key: str(introspection_data[key])
for key in self.KEYS if overwrite or
not node_info.node().properties.get(key)}
node_info.update_properties(**properties)
class ValidateInterfacesHook(base.ProcessingHook):
@ -129,8 +127,7 @@ class ValidateInterfacesHook(base.ProcessingHook):
valid_macs = [iface['mac'] for iface in valid_interfaces.values()]
introspection_data['macs'] = valid_macs
def before_update(self, introspection_data, node_info, node_patches,
ports_patches, **kwargs):
def before_update(self, introspection_data, node_info, **kwargs):
"""Drop ports that are not present in the data."""
if CONF.processing.keep_ports == 'present':
expected_macs = {

View File

@ -18,7 +18,7 @@ from ironicclient import exceptions
from oslo_config import cfg
from oslo_log import log
from ironic_inspector.common.i18n import _, _LE, _LI
from ironic_inspector.common.i18n import _, _LE, _LI, _LW
from ironic_inspector.common import swift
from ironic_inspector import firewall
from ironic_inspector import node_cache
@ -123,16 +123,20 @@ def process(introspection_data):
def _run_post_hooks(node_info, introspection_data):
hooks = plugins_base.processing_hooks_manager()
node_patches = []
port_patches = {}
for hook_ext in hooks:
node_patches = []
ports_patches = {}
hook_ext.obj.before_update(introspection_data, node_info,
node_patches, port_patches)
node_patches=node_patches,
ports_patches=ports_patches)
if node_patches:
LOG.warn(_LW('Using node_patches is deprecated'))
node_info.patch(node_patches)
node_patches = [p for p in node_patches if p]
port_patches = {mac: patch for (mac, patch) in port_patches.items()
if patch and mac in node_info.ports()}
return node_patches, port_patches
if ports_patches:
LOG.warn(_LW('Using ports_patches is deprecated'))
for mac, patches in ports_patches.items():
node_info.patch_port(mac, patches)
def _process_node(node, introspection_data, node_info):
@ -141,8 +145,7 @@ def _process_node(node, introspection_data, node_info):
node_info.create_ports(introspection_data.get('macs') or ())
node_patches, port_patches = _run_post_hooks(node_info,
introspection_data)
_run_post_hooks(node_info, introspection_data)
if CONF.processing.store_data == 'swift':
swift_object_name = swift.store_introspection_data(introspection_data,
@ -151,18 +154,13 @@ def _process_node(node, introspection_data, node_info):
'Swift in object %(obj)s'),
{'node': node_info.uuid, 'obj': swift_object_name})
if CONF.processing.store_data_location:
node_patches.append({'op': 'add',
'path': '/extra/%s' %
CONF.processing.store_data_location,
'value': swift_object_name})
node_info.patch([{'op': 'add', 'path': '/extra/%s' %
CONF.processing.store_data_location,
'value': swift_object_name}])
else:
LOG.debug('Swift support is disabled, introspection data for node %s '
'won\'t be stored', node_info.uuid)
node_info.patch(node_patches)
for mac, patches in port_patches.items():
node_info.patch_port(mac, patches)
ironic = utils.get_client()
firewall.update_filters(ironic)
@ -193,7 +191,7 @@ def _finish_set_ipmi_credentials(ironic, node, node_info, introspection_data,
introspection_data.get('ipmi_address')):
patch.append({'op': 'add', 'path': '/driver_info/ipmi_address',
'value': introspection_data['ipmi_address']})
ironic.node.update(node_info.uuid, patch)
node_info.patch(patch)
for attempt in range(_CREDENTIALS_WAIT_RETRIES):
try:

View File

@ -73,6 +73,16 @@ class BaseTest(unittest.TestCase):
actual = sorted(actual, key=lambda p: p['path'])
self.assertEqual(expected, actual)
def assertCalledWithPatch(self, expected, mock_call):
def _get_patch_param(call):
try:
return call[0][1]
except IndexError:
return call[0][0]
actual = sum(map(_get_patch_param, mock_call.call_args_list), [])
self.assertPatchEqual(actual, expected)
class NodeTest(BaseTest):
def setUp(self):

View File

@ -126,7 +126,8 @@ class Test(Base):
self.assertEqual({'uuid': self.uuid}, res)
eventlet.greenthread.sleep(DEFAULT_SLEEP)
self.cli.node.update.assert_any_call(self.uuid, self.patch)
self.cli.node.update.assert_called_once_with(self.uuid, mock.ANY)
self.assertCalledWithPatch(self.patch, self.cli.node.update)
self.cli.port.create.assert_called_once_with(
node_uuid=self.uuid, address='11:22:33:44:55:66')
@ -155,8 +156,8 @@ class Test(Base):
self.assertTrue(res['ipmi_setup_credentials'])
eventlet.greenthread.sleep(DEFAULT_SLEEP)
self.cli.node.update.assert_any_call(self.uuid, self.patch)
self.cli.node.update.assert_any_call(self.uuid, patch_credentials)
self.assertCalledWithPatch(self.patch + patch_credentials,
self.cli.node.update)
self.cli.port.create.assert_called_once_with(
node_uuid=self.uuid, address='11:22:33:44:55:66')

View File

@ -271,8 +271,8 @@ class TestPlugins(unittest.TestCase):
mgr = plugins_base.processing_hooks_manager()
mgr.map_method('before_processing', 'introspection_data')
mock_pre.assert_called_once_with(mock.ANY, 'introspection_data')
mgr.map_method('before_update', 'node_info', {}, [], {})
mock_post.assert_called_once_with(mock.ANY, 'node_info', {}, [], {})
mgr.map_method('before_update', 'node_info', {})
mock_post.assert_called_once_with(mock.ANY, 'node_info', {})
def test_manager_is_cached(self):
self.assertIs(plugins_base.processing_hooks_manager(),

View File

@ -12,52 +12,39 @@
# limitations under the License.
import json
try:
from unittest import mock
except ImportError:
import mock
import mock
from ironic_inspector import node_cache
from ironic_inspector.plugins import extra_hardware
from ironic_inspector.test import base as test_base
@mock.patch.object(extra_hardware.swift, 'SwiftAPI', autospec=True)
@mock.patch.object(node_cache.NodeInfo, 'patch')
class TestExtraHardware(test_base.NodeTest):
def setUp(self):
super(TestExtraHardware, self).setUp()
self.hook = extra_hardware.ExtraHardwareHook()
hook = extra_hardware.ExtraHardwareHook()
def _before_update(self, introspection_data):
node_patches = []
ports_patches = {}
self.hook.before_update(introspection_data, self.node_info,
node_patches, ports_patches)
self.assertFalse(ports_patches)
return node_patches
def test_data_recieved(self, swift_mock):
def test_data_recieved(self, patch_mock, swift_mock):
introspection_data = {
'data': [['memory', 'total', 'size', '4294967296'],
['cpu', 'physical', 'number', '1'],
['cpu', 'logical', 'number', '1']]}
self.hook.before_processing(introspection_data)
node_patches = self._before_update(introspection_data)
self.hook.before_update(introspection_data, self.node_info)
swift_conn = swift_mock.return_value
name = 'extra_hardware-%s' % self.uuid
data = json.dumps(introspection_data['data'])
swift_conn.create_object.assert_called_once_with(name, data)
self.assertEqual('add',
node_patches[0]['op'])
self.assertEqual('/extra/hardware_swift_object',
node_patches[0]['path'])
self.assertEqual(name,
node_patches[0]['value'])
patch_mock.assert_called_once_with(
[{'op': 'add', 'path': '/extra/hardware_swift_object',
'value': name}])
def test_no_data_recieved(self, swift_mock):
def test_no_data_recieved(self, patch_mock, swift_mock):
introspection_data = {'cats': 'meow'}
swift_conn = swift_mock.return_value
self.hook.before_processing(introspection_data)
node_patches = self._before_update(introspection_data)
self.assertFalse(node_patches)
self.hook.before_update(introspection_data, self.node_info)
self.assertFalse(patch_mock.called)
self.assertFalse(swift_conn.create_object.called)

View File

@ -11,24 +11,16 @@
# See the License for the specific language governing permissions and
# limitations under the License.
import mock
from ironic_inspector import node_cache
from ironic_inspector.plugins import base
from ironic_inspector.plugins import raid_device
from ironic_inspector.test import base as test_base
class TestRaidDeviceDetection(test_base.NodeTest):
def setUp(self):
super(TestRaidDeviceDetection, self).setUp()
self.hook = raid_device.RaidDeviceDetection()
def _before_update(self, introspection_data):
node_patches = []
ports_patches = {}
self.hook.before_update(introspection_data, self.node_info,
node_patches, ports_patches)
self.assertFalse(ports_patches)
return node_patches
hook = raid_device.RaidDeviceDetection()
def test_loadable_by_name(self):
names = ('raid_device', 'root_device_hint')
@ -50,63 +42,52 @@ class TestRaidDeviceDetection(test_base.NodeTest):
self.assertEqual(42, introspection_data['local_gb'])
class TestRaidDeviceDetectionUpdate(test_base.NodeTest):
hook = raid_device.RaidDeviceDetection()
@mock.patch.object(node_cache.NodeInfo, 'patch')
def _check(self, data, patch, mock_patch):
self.hook.before_processing(data)
self.hook.before_update(data, self.node_info)
self.assertCalledWithPatch(patch, mock_patch)
def test_no_previous_block_devices(self):
introspection_data = {'block_devices': {'serials': ['foo', 'bar']}}
node_patches = self._before_update(introspection_data)
self.assertEqual('add',
node_patches[0]['op'])
self.assertEqual('/extra/block_devices',
node_patches[0]['path'])
self.assertEqual(introspection_data['block_devices'],
node_patches[0]['value'])
expected = [{'op': 'add', 'path': '/extra/block_devices',
'value': introspection_data['block_devices']}]
self._check(introspection_data, expected)
def test_root_device_found(self):
self.node.extra['block_devices'] = {'serials': ['foo', 'bar']}
introspection_data = {'block_devices': {'serials': ['foo', 'baz']}}
self.hook.before_processing(introspection_data)
node_patches = self._before_update(introspection_data)
expected = [{'op': 'remove', 'path': '/extra/block_devices'},
{'op': 'add', 'path': '/properties/root_device',
'value': {'serial': 'baz'}}]
self.assertEqual('remove',
node_patches[0]['op'])
self.assertEqual('/extra/block_devices',
node_patches[0]['path'])
self.assertEqual('add',
node_patches[1]['op'])
self.assertEqual('/properties/root_device',
node_patches[1]['path'])
self.assertEqual({'serial': 'baz'},
node_patches[1]['value'])
self._check(introspection_data, expected)
def test_root_device_already_exposed(self):
self.node.properties['root_device'] = {'serial': 'foo'}
introspection_data = {'block_devices': {'serials': ['foo', 'baz']}}
self.hook.before_processing(introspection_data)
node_patches = self._before_update(introspection_data)
self.assertFalse(node_patches)
self._check(introspection_data, [])
def test_multiple_new_devices(self):
self.node.extra['block_devices'] = {'serials': ['foo', 'bar']}
introspection_data = {
'block_devices': {'serials': ['foo', 'baz', 'qux']}
}
self.hook.before_processing(introspection_data)
node_patches = self._before_update(introspection_data)
self.assertFalse(node_patches)
self._check(introspection_data, [])
def test_no_new_devices(self):
self.node.extra['block_devices'] = {'serials': ['foo', 'bar']}
introspection_data = {'block_devices': {'serials': ['foo', 'bar']}}
self.hook.before_processing(introspection_data)
node_patches = self._before_update(introspection_data)
self.assertFalse(node_patches)
self._check(introspection_data, [])
def test_no_block_devices_from_ramdisk(self):
introspection_data = {}
self.hook.before_processing(introspection_data)
node_patches = self._before_update(introspection_data)
self.assertFalse(node_patches)
self._check(introspection_data, [])

View File

@ -59,7 +59,8 @@ class TestSchedulerHook(test_base.NodeTest):
self.assertRaisesRegexp(utils.Error, key,
self.hook.before_processing, new_data)
def test_before_update(self):
@mock.patch.object(node_cache.NodeInfo, 'patch')
def test_before_update(self, mock_patch):
patch = [
{'path': '/properties/cpus', 'value': '2', 'op': 'add'},
{'path': '/properties/cpu_arch', 'value': 'x86_64', 'op': 'add'},
@ -67,12 +68,11 @@ class TestSchedulerHook(test_base.NodeTest):
{'path': '/properties/local_gb', 'value': '20', 'op': 'add'}
]
self.hook.before_update(self.data, self.node_info,
self.node_patches, self.ports_patches)
self.assertPatchEqual(patch, self.node_patches)
self.assertFalse(self.ports_patches)
self.hook.before_update(self.data, self.node_info)
self.assertCalledWithPatch(patch, mock_patch)
def test_before_update_no_overwrite(self):
@mock.patch.object(node_cache.NodeInfo, 'patch')
def test_before_update_no_overwrite(self, mock_patch):
CONF.set_override('overwrite_existing', False, 'processing')
self.node.properties = {
'memory_mb': '4096',
@ -83,10 +83,8 @@ class TestSchedulerHook(test_base.NodeTest):
{'path': '/properties/local_gb', 'value': '20', 'op': 'add'}
]
self.hook.before_update(self.data, self.node_info,
self.node_patches, self.ports_patches)
self.assertPatchEqual(patch, self.node_patches)
self.assertFalse(self.ports_patches)
self.hook.before_update(self.data, self.node_info)
self.assertCalledWithPatch(patch, mock_patch)
class TestValidateInterfacesHook(test_base.NodeTest):
@ -183,14 +181,14 @@ class TestValidateInterfacesHook(test_base.NodeTest):
@mock.patch.object(node_cache.NodeInfo, 'delete_port', autospec=True)
def test_keep_all(self, mock_delete_port):
self.hook.before_update(self.data, self.node_info, None, None)
self.hook.before_update(self.data, self.node_info)
self.assertFalse(mock_delete_port.called)
@mock.patch.object(node_cache.NodeInfo, 'delete_port')
def test_keep_present(self, mock_delete_port):
CONF.set_override('keep_ports', 'present', 'processing')
self.data['all_interfaces'] = self.orig_interfaces
self.hook.before_update(self.data, self.node_info, None, None)
self.hook.before_update(self.data, self.node_info)
mock_delete_port.assert_called_once_with(self.existing_ports[1])
@ -198,7 +196,7 @@ class TestValidateInterfacesHook(test_base.NodeTest):
def test_keep_added(self, mock_delete_port):
CONF.set_override('keep_ports', 'added', 'processing')
self.data['macs'] = [self.pxe_interface['mac']]
self.hook.before_update(self.data, self.node_info, None, None)
self.hook.before_update(self.data, self.node_info)
mock_delete_port.assert_any_call(self.existing_ports[0])
mock_delete_port.assert_any_call(self.existing_ports[1])

View File

@ -294,13 +294,13 @@ class TestProcessNode(BaseTest):
address=self.macs[0])
self.cli.port.create.assert_any_call(node_uuid=self.uuid,
address=self.macs[1])
self.cli.node.update.assert_called_once_with(self.uuid,
self.patch_props)
self.assertCalledWithPatch(self.patch_props, self.cli.node.update)
self.cli.node.set_power_state.assert_called_once_with(self.uuid, 'off')
self.assertFalse(self.cli.node.validate.called)
post_hook_mock.assert_called_once_with(self.data, self.node_info,
self.patch_props, {})
node_patches=mock.ANY,
ports_patches=mock.ANY)
finished_mock.assert_called_once_with(mock.ANY)
def test_overwrite_disabled(self, filters_mock, post_hook_mock):
@ -312,7 +312,7 @@ class TestProcessNode(BaseTest):
self.call()
self.cli.node.update.assert_called_once_with(self.uuid, patch)
self.assertCalledWithPatch(patch, self.cli.node.update)
def test_port_failed(self, filters_mock, post_hook_mock):
self.cli.port.create.side_effect = (
@ -324,26 +324,25 @@ class TestProcessNode(BaseTest):
address=self.macs[0])
self.cli.port.create.assert_any_call(node_uuid=self.uuid,
address=self.macs[1])
self.cli.node.update.assert_called_once_with(self.uuid,
self.patch_props)
self.assertCalledWithPatch(self.patch_props, self.cli.node.update)
def test_hook_patches(self, filters_mock, post_hook_mock):
node_patches = ['node patch1', 'node patch2']
port_patch = ['port patch']
expected_node_patches = [{'path': 'foo', 'op': 'bar'}]
expected_port_patch = [{'path': 'foo', 'op': 'baz'}]
def fake_hook(data, node_info, node_p, ports_p):
node_p.extend(node_patches)
ports_p.setdefault(self.macs[1], []).extend(port_patch)
def fake_hook(data, node_info, node_patches, ports_patches):
node_patches.extend(expected_node_patches)
ports_patches.setdefault(self.macs[1],
[]).extend(expected_port_patch)
post_hook_mock.side_effect = fake_hook
self.call()
self.cli.node.update.assert_called_once_with(self.uuid,
self.patch_props
+ node_patches)
self.cli.port.update.assert_called_once_with(self.ports[1].uuid,
port_patch)
self.assertCalledWithPatch(self.patch_props + expected_node_patches,
self.cli.node.update)
self.assertCalledWithPatch(expected_port_patch,
self.cli.port.update)
def test_set_ipmi_credentials(self, filters_mock, post_hook_mock):
self.node_info.set_option('new_ipmi_credentials', self.new_creds)
@ -381,7 +380,6 @@ class TestProcessNode(BaseTest):
self.assertRaisesRegexp(utils.Error, 'Failed to validate',
self.call)
self.cli.node.update.assert_any_call(self.uuid, self.patch_props)
self.cli.node.update.assert_any_call(self.uuid, self.patch_credentials)
self.assertEqual(2, self.cli.node.update.call_count)
self.assertEqual(process._CREDENTIALS_WAIT_RETRIES,
@ -401,8 +399,7 @@ class TestProcessNode(BaseTest):
self.call)
self.cli.node.set_power_state.assert_called_once_with(self.uuid, 'off')
self.cli.node.update.assert_called_once_with(self.uuid,
self.patch_props)
self.assertCalledWithPatch(self.patch_props, self.cli.node.update)
finished_mock.assert_called_once_with(
mock.ANY,
error='Failed to power off node %s, check it\'s power management'
@ -418,8 +415,7 @@ class TestProcessNode(BaseTest):
self.call()
swift_conn.create_object.assert_called_once_with(name, expected)
self.cli.node.update.assert_called_once_with(self.uuid,
self.patch_props)
self.assertCalledWithPatch(self.patch_props, self.cli.node.update)
@mock.patch.object(process.swift, 'SwiftAPI', autospec=True)
def test_store_data_location(self, swift_mock, filters_mock,
@ -439,5 +435,4 @@ class TestProcessNode(BaseTest):
self.call()
swift_conn.create_object.assert_called_once_with(name, expected)
self.cli.node.update.assert_called_once_with(self.uuid,
self.patch_props)
self.assertCalledWithPatch(self.patch_props, self.cli.node.update)