Expose Allocation objects on Instance

Change-Id: If62a6829478c532428aff2f5f51edc35df8ee2e1
This commit is contained in:
Dmitry Tantsur 2019-05-16 11:49:38 +02:00
parent ce77bf8e1e
commit 1afa4ac4ed
7 changed files with 74 additions and 17 deletions

View File

@ -25,6 +25,9 @@ Installation
pip install --user metalsmith pip install --user metalsmith
.. note::
The current versions of *metalsmith* require Bare Metal API from the Stein
release or newer. Use the 0.11 release series for older versions.
Contributing Contributing
------------ ------------

View File

@ -5,7 +5,7 @@ fixtures==3.0.0
flake8-import-order==0.13 flake8-import-order==0.13
hacking==1.0.0 hacking==1.0.0
mock==2.0 mock==2.0
openstacksdk==0.22.0 openstacksdk==0.25.0
pbr==2.0.0 pbr==2.0.0
Pygments==2.2.0 Pygments==2.2.0
requests==2.18.4 requests==2.18.4

View File

@ -86,10 +86,16 @@ _DEPLOYED_STATES = frozenset([InstanceState.ACTIVE, InstanceState.MAINTENANCE])
class Instance(object): class Instance(object):
"""Instance status in metalsmith.""" """Instance status in metalsmith."""
def __init__(self, connection, node): def __init__(self, connection, node, allocation=None):
self._connection = connection self._connection = connection
self._uuid = node.id self._uuid = node.id
self._node = node self._node = node
self._allocation = allocation
@property
def allocation(self):
"""Allocation object associated with the node (if any)."""
return self._allocation
@property @property
def hostname(self): def hostname(self):
@ -163,6 +169,8 @@ class Instance(object):
def to_dict(self): def to_dict(self):
"""Convert instance to a dict.""" """Convert instance to a dict."""
return { return {
'allocation': (self._allocation.to_dict()
if self._allocation is not None else None),
'hostname': self.hostname, 'hostname': self.hostname,
'ip_addresses': self.ip_addresses(), 'ip_addresses': self.ip_addresses(),
'node': self._node.to_dict(), 'node': self._node.to_dict(),

View File

@ -332,6 +332,7 @@ class Provisioner(_utils.GetNodeMixin):
else: else:
# Update the node to return it's latest state # Update the node to return it's latest state
node = self._get_node(node, refresh=True) node = self._get_node(node, refresh=True)
# We don't create allocations yet, so don't use _get_instance.
instance = _instance.Instance(self.connection, node) instance = _instance.Instance(self.connection, node)
return instance return instance
@ -358,7 +359,9 @@ class Provisioner(_utils.GetNodeMixin):
nodes = [self._get_node(n, accept_hostname=True) for n in nodes] nodes = [self._get_node(n, accept_hostname=True) for n in nodes]
nodes = self.connection.baremetal.wait_for_nodes_provision_state( nodes = self.connection.baremetal.wait_for_nodes_provision_state(
nodes, 'active', timeout=timeout) nodes, 'active', timeout=timeout)
return [_instance.Instance(self.connection, node) for node in nodes] # Using _get_instance in case the deployment started by something
# external that uses allocations.
return [self._get_instance(node) for node in nodes]
def _clean_instance_info(self, instance_info): def _clean_instance_info(self, instance_info):
return {key: value return {key: value
@ -426,6 +429,16 @@ class Provisioner(_utils.GetNodeMixin):
""" """
return self.show_instances([instance_id])[0] return self.show_instances([instance_id])[0]
def _get_instance(self, ident):
node = self._get_node(ident, accept_hostname=True)
if node.allocation_id:
allocation = self.connection.baremetal.get_allocation(
node.allocation_id)
else:
allocation = None
return _instance.Instance(self.connection, node,
allocation=allocation)
def show_instances(self, instances): def show_instances(self, instances):
"""Show information about instance. """Show information about instance.
@ -439,12 +452,7 @@ class Provisioner(_utils.GetNodeMixin):
if one of the instances are not valid instances. if one of the instances are not valid instances.
""" """
with self._cache_node_list_for_lookup(): with self._cache_node_list_for_lookup():
result = [ result = [self._get_instance(inst) for inst in instances]
_instance.Instance(
self.connection,
self._get_node(inst, accept_hostname=True))
for inst in instances
]
# NOTE(dtantsur): do not accept node names as valid instances if they # NOTE(dtantsur): do not accept node names as valid instances if they
# are not deployed or being deployed. # are not deployed or being deployed.
missing = [inst for (res, inst) in zip(result, instances) missing = [inst for (res, inst) in zip(result, instances)
@ -461,8 +469,6 @@ class Provisioner(_utils.GetNodeMixin):
:return: list of :py:class:`metalsmith.Instance` objects. :return: list of :py:class:`metalsmith.Instance` objects.
""" """
nodes = self.connection.baremetal.nodes(associated=True, details=True) nodes = self.connection.baremetal.nodes(associated=True, details=True)
instances = [i for i in instances = [i for i in map(self._get_instance, nodes)
(_instance.Instance(self.connection, node)
for node in nodes)
if i.state != _instance.InstanceState.UNKNOWN] if i.state != _instance.InstanceState.UNKNOWN]
return instances return instances

View File

@ -129,7 +129,28 @@ class TestInstanceStates(test_provisioner.Base):
mock_ips.return_value = {'private': ['1.2.3.4']} mock_ips.return_value = {'private': ['1.2.3.4']}
to_dict = self.instance.to_dict() to_dict = self.instance.to_dict()
self.assertEqual({'hostname': 'host', self.assertEqual({'allocation': None,
'hostname': 'host',
'ip_addresses': {'private': ['1.2.3.4']},
'node': {'node': 'dict'},
'state': 'deploying',
'uuid': self.node.id},
to_dict)
# States are converted to strings
self.assertIsInstance(to_dict['state'], six.string_types)
@mock.patch.object(_instance.Instance, 'ip_addresses', autospec=True)
def test_to_dict_with_allocation(self, mock_ips):
self.node.provision_state = 'wait call-back'
self.node.to_dict.return_value = {'node': 'dict'}
self.node.instance_info = {'metalsmith_hostname': 'host'}
mock_ips.return_value = {'private': ['1.2.3.4']}
self.instance._allocation = mock.Mock()
self.instance._allocation.to_dict.return_value = {'alloc': 'dict'}
to_dict = self.instance.to_dict()
self.assertEqual({'allocation': {'alloc': 'dict'},
'hostname': 'host',
'ip_addresses': {'private': ['1.2.3.4']}, 'ip_addresses': {'private': ['1.2.3.4']},
'node': {'node': 'dict'}, 'node': {'node': 'dict'},
'state': 'deploying', 'state': 'deploying',

View File

@ -29,7 +29,8 @@ from metalsmith import sources
NODE_FIELDS = ['name', 'id', 'instance_info', 'instance_id', 'is_maintenance', NODE_FIELDS = ['name', 'id', 'instance_info', 'instance_id', 'is_maintenance',
'maintenance_reason', 'properties', 'provision_state', 'extra', 'maintenance_reason', 'properties', 'provision_state', 'extra',
'last_error', 'traits', 'resource_class', 'conductor_group'] 'last_error', 'traits', 'resource_class', 'conductor_group',
'allocation_id']
class TestInit(testtools.TestCase): class TestInit(testtools.TestCase):
@ -66,7 +67,8 @@ class Base(testtools.TestCase):
id='000', instance_id=None, id='000', instance_id=None,
properties={'local_gb': 100}, properties={'local_gb': 100},
instance_info={}, instance_info={},
is_maintenance=False, extra={}) is_maintenance=False, extra={},
allocation_id=None)
self.node.name = 'control-0' self.node.name = 'control-0'
def _reset_api_mock(self): def _reset_api_mock(self):
@ -1386,6 +1388,19 @@ class TestShowInstance(Base):
self.assertIs(inst.node, self.node) self.assertIs(inst.node, self.node)
self.assertIs(inst.uuid, self.node.id) self.assertIs(inst.uuid, self.node.id)
def test_show_instance_with_allocation(self):
self.node.allocation_id = 'id2'
self.mock_get_node.side_effect = lambda n, *a, **kw: self.node
inst = self.pr.show_instance('id1')
self.mock_get_node.assert_called_once_with(self.pr, 'id1',
accept_hostname=True)
self.api.baremetal.get_allocation.assert_called_once_with('id2')
self.assertIsInstance(inst, _instance.Instance)
self.assertIs(inst.allocation,
self.api.baremetal.get_allocation.return_value)
self.assertIs(inst.node, self.node)
self.assertIs(inst.uuid, self.node.id)
def test_show_instances(self): def test_show_instances(self):
self.mock_get_node.side_effect = [self.node, self.node] self.mock_get_node.side_effect = [self.node, self.node]
result = self.pr.show_instances(['1', '2']) result = self.pr.show_instances(['1', '2'])
@ -1415,10 +1430,11 @@ class TestListInstances(Base):
super(TestListInstances, self).setUp() super(TestListInstances, self).setUp()
self.nodes = [ self.nodes = [
mock.Mock(spec=NODE_FIELDS, provision_state=state, mock.Mock(spec=NODE_FIELDS, provision_state=state,
instance_id='1234') instance_id='1234', allocation_id=None)
for state in ('active', 'active', 'deploying', 'wait call-back', for state in ('active', 'active', 'deploying', 'wait call-back',
'deploy failed', 'available', 'available', 'enroll') 'deploy failed', 'available', 'available', 'enroll')
] ]
self.nodes[0].allocation_id = 'id2'
self.nodes[6].instance_id = None self.nodes[6].instance_id = None
self.api.baremetal.nodes.return_value = self.nodes self.api.baremetal.nodes.return_value = self.nodes
@ -1426,5 +1442,8 @@ class TestListInstances(Base):
instances = self.pr.list_instances() instances = self.pr.list_instances()
self.assertTrue(isinstance(i, _instance.Instance) for i in instances) self.assertTrue(isinstance(i, _instance.Instance) for i in instances)
self.assertEqual(self.nodes[:6], [i.node for i in instances]) self.assertEqual(self.nodes[:6], [i.node for i in instances])
self.assertEqual([self.api.baremetal.get_allocation.return_value]
+ [None] * 5,
[i.allocation for i in instances])
self.api.baremetal.nodes.assert_called_once_with(associated=True, self.api.baremetal.nodes.assert_called_once_with(associated=True,
details=True) details=True)

View File

@ -2,7 +2,7 @@
# of appearance. Changing the order has an impact on the overall integration # of appearance. Changing the order has an impact on the overall integration
# process, which may cause wedges in the gate later. # process, which may cause wedges in the gate later.
pbr!=2.1.0,>=2.0.0 # Apache-2.0 pbr!=2.1.0,>=2.0.0 # Apache-2.0
openstacksdk>=0.22.0 # Apache-2.0 openstacksdk>=0.25.0 # Apache-2.0
requests>=2.18.4 # Apache-2.0 requests>=2.18.4 # Apache-2.0
six>=1.10.0 # MIT six>=1.10.0 # MIT
enum34>=1.0.4;python_version=='2.7' or python_version=='2.6' or python_version=='3.3' # BSD enum34>=1.0.4;python_version=='2.7' or python_version=='2.6' or python_version=='3.3' # BSD