Allow attaching existing ports
Also fix cleaning up node.extra and undeployment. Change-Id: Ic63b3663caea4eb9acd9d8f48008785dec2a62d3
This commit is contained in:
parent
a065777baa
commit
1037276d61
@ -25,6 +25,17 @@ from metalsmith import _provisioner
|
|||||||
LOG = logging.getLogger(__name__)
|
LOG = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class NICAction(argparse.Action):
|
||||||
|
def __call__(self, parser, namespace, values, option_string=None):
|
||||||
|
assert option_string in ('--port', '--network')
|
||||||
|
nics = getattr(namespace, self.dest, None) or []
|
||||||
|
if option_string == '--network':
|
||||||
|
nics.append({'network': values})
|
||||||
|
else:
|
||||||
|
nics.append({'port': values})
|
||||||
|
setattr(namespace, self.dest, nics)
|
||||||
|
|
||||||
|
|
||||||
def _do_deploy(api, args, wait=None):
|
def _do_deploy(api, args, wait=None):
|
||||||
capabilities = dict(item.split('=', 1) for item in args.capability)
|
capabilities = dict(item.split('=', 1) for item in args.capability)
|
||||||
if args.ssh_public_key:
|
if args.ssh_public_key:
|
||||||
@ -36,7 +47,7 @@ def _do_deploy(api, args, wait=None):
|
|||||||
node = api.reserve_node(args.resource_class, capabilities=capabilities)
|
node = api.reserve_node(args.resource_class, capabilities=capabilities)
|
||||||
api.provision_node(node,
|
api.provision_node(node,
|
||||||
image_ref=args.image,
|
image_ref=args.image,
|
||||||
network_refs=[args.network],
|
nics=args.nics,
|
||||||
root_disk_size=args.root_disk_size,
|
root_disk_size=args.root_disk_size,
|
||||||
ssh_keys=ssh_keys,
|
ssh_keys=ssh_keys,
|
||||||
netboot=args.netboot,
|
netboot=args.netboot,
|
||||||
@ -72,7 +83,9 @@ def _parse_args(args, config):
|
|||||||
deploy.add_argument('--image', help='image to use (name or UUID)',
|
deploy.add_argument('--image', help='image to use (name or UUID)',
|
||||||
required=True)
|
required=True)
|
||||||
deploy.add_argument('--network', help='network to use (name or UUID)',
|
deploy.add_argument('--network', help='network to use (name or UUID)',
|
||||||
required=True),
|
dest='nics', action=NICAction)
|
||||||
|
deploy.add_argument('--port', help='port to attach (name or UUID)',
|
||||||
|
dest='nics', action=NICAction)
|
||||||
deploy.add_argument('--netboot', action='store_true',
|
deploy.add_argument('--netboot', action='store_true',
|
||||||
help='boot from network instead of local disk')
|
help='boot from network instead of local disk')
|
||||||
deploy.add_argument('--root-disk-size', type=int,
|
deploy.add_argument('--root-disk-size', type=int,
|
||||||
|
@ -61,8 +61,8 @@ class InvalidImage(Error):
|
|||||||
"""Requested image is invalid and cannot be used."""
|
"""Requested image is invalid and cannot be used."""
|
||||||
|
|
||||||
|
|
||||||
class InvalidNetwork(Error):
|
class InvalidNIC(Error):
|
||||||
"""Requested network is invalid and cannot be used."""
|
"""Requested NIC is invalid and cannot be used."""
|
||||||
|
|
||||||
|
|
||||||
class UnknownRootDiskSize(Error):
|
class UnknownRootDiskSize(Error):
|
||||||
|
@ -89,7 +89,8 @@ class API(object):
|
|||||||
return node
|
return node
|
||||||
|
|
||||||
def get_port(self, port_id):
|
def get_port(self, port_id):
|
||||||
return self.connection.network.get_port(port_id)
|
return self.connection.network.find_port(port_id,
|
||||||
|
ignore_missing=False)
|
||||||
|
|
||||||
def list_node_attached_ports(self, node):
|
def list_node_attached_ports(self, node):
|
||||||
return self.ironic.node.vif_list(_node_id(node))
|
return self.ironic.node.vif_list(_node_id(node))
|
||||||
|
@ -13,6 +13,7 @@
|
|||||||
# See the License for the specific language governing permissions and
|
# See the License for the specific language governing permissions and
|
||||||
# limitations under the License.
|
# limitations under the License.
|
||||||
|
|
||||||
|
import collections
|
||||||
import logging
|
import logging
|
||||||
import random
|
import random
|
||||||
|
|
||||||
@ -27,6 +28,7 @@ from metalsmith import _utils
|
|||||||
LOG = logging.getLogger(__name__)
|
LOG = logging.getLogger(__name__)
|
||||||
|
|
||||||
_CREATED_PORTS = 'metalsmith_created_ports'
|
_CREATED_PORTS = 'metalsmith_created_ports'
|
||||||
|
_ATTACHED_PORTS = 'metalsmith_attached_ports'
|
||||||
|
|
||||||
|
|
||||||
class Provisioner(object):
|
class Provisioner(object):
|
||||||
@ -63,14 +65,16 @@ class Provisioner(object):
|
|||||||
return _scheduler.schedule_node(nodes, filters, reserver,
|
return _scheduler.schedule_node(nodes, filters, reserver,
|
||||||
dry_run=self._dry_run)
|
dry_run=self._dry_run)
|
||||||
|
|
||||||
def provision_node(self, node, image_ref, network_refs,
|
def provision_node(self, node, image_ref, nics=None, root_disk_size=None,
|
||||||
root_disk_size=None, ssh_keys=None, netboot=False,
|
ssh_keys=None, netboot=False, wait=None):
|
||||||
wait=None):
|
|
||||||
"""Provision the node with the given image.
|
"""Provision the node with the given image.
|
||||||
|
|
||||||
:param node: Node object, UUID or name.
|
:param node: Node object, UUID or name.
|
||||||
:param image_ref: Image name or UUID to provision.
|
:param image_ref: Image name or UUID to provision.
|
||||||
:param network_refs: List of network names or UUIDs to use.
|
:param nics: List of virtual NICs to attach to physical ports.
|
||||||
|
Each item is a dict with a key describing the type of the NIC:
|
||||||
|
either a port (``{"port": "<port name or ID>"}``) or a network
|
||||||
|
to create a port on (``{"network": "<network name or ID>"}``).
|
||||||
:param root_disk_size: The size of the root partition. By default
|
:param root_disk_size: The size of the root partition. By default
|
||||||
the value of the local_gb property is used.
|
the value of the local_gb property is used.
|
||||||
:param ssh_keys: list of public parts of the SSH keys to upload
|
:param ssh_keys: list of public parts of the SSH keys to upload
|
||||||
@ -81,6 +85,7 @@ class Provisioner(object):
|
|||||||
:return: Reservation
|
:return: Reservation
|
||||||
"""
|
"""
|
||||||
created_ports = []
|
created_ports = []
|
||||||
|
attached_ports = []
|
||||||
|
|
||||||
try:
|
try:
|
||||||
node = self._api.get_node(node)
|
node = self._api.get_node(node)
|
||||||
@ -96,21 +101,23 @@ class Provisioner(object):
|
|||||||
|
|
||||||
LOG.debug('Image: %s', image)
|
LOG.debug('Image: %s', image)
|
||||||
|
|
||||||
networks = self._get_networks(network_refs)
|
nics = self._get_nics(nics or [])
|
||||||
|
|
||||||
if self._dry_run:
|
if self._dry_run:
|
||||||
LOG.warning('Dry run, not provisioning node %s',
|
LOG.warning('Dry run, not provisioning node %s',
|
||||||
_utils.log_node(node))
|
_utils.log_node(node))
|
||||||
return node
|
return node
|
||||||
|
|
||||||
self._create_ports(node, networks, created_ports)
|
self._create_and_attach_ports(node, nics,
|
||||||
|
created_ports, attached_ports)
|
||||||
|
|
||||||
target_caps = {'boot_option': 'netboot' if netboot else 'local'}
|
target_caps = {'boot_option': 'netboot' if netboot else 'local'}
|
||||||
|
|
||||||
updates = {'/instance_info/image_source': image.id,
|
updates = {'/instance_info/image_source': image.id,
|
||||||
'/instance_info/root_gb': root_disk_size,
|
'/instance_info/root_gb': root_disk_size,
|
||||||
'/instance_info/capabilities': target_caps,
|
'/instance_info/capabilities': target_caps,
|
||||||
'/extra/%s' % _CREATED_PORTS: created_ports}
|
'/extra/%s' % _CREATED_PORTS: created_ports,
|
||||||
|
'/extra/%s' % _ATTACHED_PORTS: attached_ports}
|
||||||
|
|
||||||
for prop in ('kernel', 'ramdisk'):
|
for prop in ('kernel', 'ramdisk'):
|
||||||
value = getattr(image, '%s_id' % prop, None)
|
value = getattr(image, '%s_id' % prop, None)
|
||||||
@ -134,7 +141,7 @@ class Provisioner(object):
|
|||||||
with excutils.save_and_reraise_exception():
|
with excutils.save_and_reraise_exception():
|
||||||
LOG.error('Deploy attempt failed on node %s, cleaning up',
|
LOG.error('Deploy attempt failed on node %s, cleaning up',
|
||||||
_utils.log_node(node))
|
_utils.log_node(node))
|
||||||
self._clean_up(node, created_ports)
|
self._clean_up(node, created_ports, attached_ports)
|
||||||
|
|
||||||
if wait is not None:
|
if wait is not None:
|
||||||
LOG.info('Deploy succeeded on node %s', _utils.log_node(node))
|
LOG.info('Deploy succeeded on node %s', _utils.log_node(node))
|
||||||
@ -157,45 +164,73 @@ class Provisioner(object):
|
|||||||
else:
|
else:
|
||||||
LOG.warning('No IPs for node %s', _utils.log_node(node))
|
LOG.warning('No IPs for node %s', _utils.log_node(node))
|
||||||
|
|
||||||
def _clean_up(self, node, created_ports):
|
def _clean_up(self, node, created_ports, attached_ports):
|
||||||
try:
|
try:
|
||||||
self._delete_ports(node, created_ports)
|
self._delete_ports(node, created_ports, attached_ports)
|
||||||
self._api.release_node(node)
|
self._api.release_node(node)
|
||||||
except Exception:
|
except Exception:
|
||||||
LOG.exception('Clean up failed')
|
LOG.exception('Clean up failed')
|
||||||
|
|
||||||
def _get_networks(self, network_refs):
|
def _get_nics(self, nics):
|
||||||
"""Validate and get the networks."""
|
"""Validate and get the NICs."""
|
||||||
networks = []
|
result = []
|
||||||
for network_ref in network_refs:
|
if not isinstance(nics, collections.Sequence):
|
||||||
|
raise TypeError("NICs must be a list of dicts")
|
||||||
|
|
||||||
|
for nic in nics:
|
||||||
|
if not isinstance(nic, collections.Mapping) or len(nic) != 1:
|
||||||
|
raise TypeError("Each NIC must be a dict with one item, "
|
||||||
|
"got %s" % nic)
|
||||||
|
|
||||||
|
nic_type, nic_id = next(iter(nic.items()))
|
||||||
|
if nic_type == 'network':
|
||||||
try:
|
try:
|
||||||
network = self._api.get_network(network_ref)
|
network = self._api.get_network(nic_id)
|
||||||
except Exception as exc:
|
except Exception as exc:
|
||||||
raise _exceptions.InvalidNetwork(
|
raise _exceptions.InvalidNIC(
|
||||||
'Cannot find network %(net)s: %(error)s' %
|
'Cannot find network %(net)s: %(error)s' %
|
||||||
{'net': network_ref, 'error': exc})
|
{'net': nic_id, 'error': exc})
|
||||||
|
else:
|
||||||
|
result.append((nic_type, network))
|
||||||
|
elif nic_type == 'port':
|
||||||
|
try:
|
||||||
|
port = self._api.get_port(nic_id)
|
||||||
|
except Exception as exc:
|
||||||
|
raise _exceptions.InvalidNIC(
|
||||||
|
'Cannot find port %(port)s: %(error)s' %
|
||||||
|
{'port': nic_id, 'error': exc})
|
||||||
|
else:
|
||||||
|
result.append((nic_type, port))
|
||||||
|
else:
|
||||||
|
raise ValueError("Unexpected NIC type %s, supported values: "
|
||||||
|
"'port', 'network'" % nic_type)
|
||||||
|
|
||||||
LOG.debug('Network: %s', network)
|
return result
|
||||||
networks.append(network)
|
|
||||||
return networks
|
|
||||||
|
|
||||||
def _create_ports(self, node, networks, created_ports):
|
def _create_and_attach_ports(self, node, nics, created_ports,
|
||||||
|
attached_ports):
|
||||||
"""Create and attach ports on given networks."""
|
"""Create and attach ports on given networks."""
|
||||||
for network in networks:
|
for nic_type, nic in nics:
|
||||||
port = self._api.create_port(network_id=network.id)
|
if nic_type == 'network':
|
||||||
|
port = self._api.create_port(network_id=nic.id)
|
||||||
created_ports.append(port.id)
|
created_ports.append(port.id)
|
||||||
LOG.debug('Created Neutron port %s', port)
|
LOG.debug('Created Neutron port %s', port)
|
||||||
|
else:
|
||||||
|
port = nic
|
||||||
|
|
||||||
self._api.attach_port_to_node(node.uuid, port.id)
|
self._api.attach_port_to_node(node.uuid, port.id)
|
||||||
LOG.info('Attached port %(port)s to node %(node)s',
|
LOG.info('Attached port %(port)s to node %(node)s',
|
||||||
{'port': port.id,
|
{'port': port.id,
|
||||||
'node': _utils.log_node(node)})
|
'node': _utils.log_node(node)})
|
||||||
|
attached_ports.append(port.id)
|
||||||
|
|
||||||
def _delete_ports(self, node, created_ports=None):
|
def _delete_ports(self, node, created_ports=None, attached_ports=None):
|
||||||
if created_ports is None:
|
if created_ports is None:
|
||||||
created_ports = node.extra.get(_CREATED_PORTS, [])
|
created_ports = node.extra.get(_CREATED_PORTS, [])
|
||||||
|
if attached_ports is None:
|
||||||
|
attached_ports = node.extra.get(_ATTACHED_PORTS, [])
|
||||||
|
|
||||||
for port_id in created_ports:
|
for port_id in set(attached_ports + created_ports):
|
||||||
LOG.debug('Detaching port %(port)s from node %(node)s',
|
LOG.debug('Detaching port %(port)s from node %(node)s',
|
||||||
{'port': port_id, 'node': node.uuid})
|
{'port': port_id, 'node': node.uuid})
|
||||||
try:
|
try:
|
||||||
@ -206,12 +241,21 @@ class Provisioner(object):
|
|||||||
{'vif': port_id, 'node': _utils.log_node(node),
|
{'vif': port_id, 'node': _utils.log_node(node),
|
||||||
'exc': exc})
|
'exc': exc})
|
||||||
|
|
||||||
|
for port_id in created_ports:
|
||||||
LOG.debug('Deleting port %s', port_id)
|
LOG.debug('Deleting port %s', port_id)
|
||||||
try:
|
try:
|
||||||
self._api.delete_port(port_id)
|
self._api.delete_port(port_id)
|
||||||
except Exception:
|
except Exception:
|
||||||
LOG.warning('Failed to delete neutron port %s', port_id)
|
LOG.warning('Failed to delete neutron port %s', port_id)
|
||||||
|
|
||||||
|
update = {'/extra/%s' % item: _os_api.REMOVE
|
||||||
|
for item in (_CREATED_PORTS, _ATTACHED_PORTS)}
|
||||||
|
try:
|
||||||
|
self._api.update_node(node, update)
|
||||||
|
except Exception as exc:
|
||||||
|
LOG.warning('Failed to clear node %(node)s extra: %(exc)s',
|
||||||
|
{'node': _utils.log_node(node), 'exc': exc})
|
||||||
|
|
||||||
def unprovision_node(self, node, wait=None):
|
def unprovision_node(self, node, wait=None):
|
||||||
"""Unprovision a previously provisioned node.
|
"""Unprovision a previously provisioned node.
|
||||||
|
|
||||||
|
@ -38,7 +38,7 @@ class TestMain(testtools.TestCase):
|
|||||||
mock_pr.return_value.provision_node.assert_called_once_with(
|
mock_pr.return_value.provision_node.assert_called_once_with(
|
||||||
mock_pr.return_value.reserve_node.return_value,
|
mock_pr.return_value.reserve_node.return_value,
|
||||||
image_ref='myimg',
|
image_ref='myimg',
|
||||||
network_refs=['mynet'],
|
nics=[{'network': 'mynet'}],
|
||||||
root_disk_size=None,
|
root_disk_size=None,
|
||||||
ssh_keys=[],
|
ssh_keys=[],
|
||||||
netboot=False,
|
netboot=False,
|
||||||
@ -58,7 +58,7 @@ class TestMain(testtools.TestCase):
|
|||||||
mock_pr.return_value.provision_node.assert_called_once_with(
|
mock_pr.return_value.provision_node.assert_called_once_with(
|
||||||
mock_pr.return_value.reserve_node.return_value,
|
mock_pr.return_value.reserve_node.return_value,
|
||||||
image_ref='myimg',
|
image_ref='myimg',
|
||||||
network_refs=['mynet'],
|
nics=[{'network': 'mynet'}],
|
||||||
root_disk_size=None,
|
root_disk_size=None,
|
||||||
ssh_keys=[],
|
ssh_keys=[],
|
||||||
netboot=False,
|
netboot=False,
|
||||||
@ -78,7 +78,7 @@ class TestMain(testtools.TestCase):
|
|||||||
mock_pr.return_value.provision_node.assert_called_once_with(
|
mock_pr.return_value.provision_node.assert_called_once_with(
|
||||||
mock_pr.return_value.reserve_node.return_value,
|
mock_pr.return_value.reserve_node.return_value,
|
||||||
image_ref='myimg',
|
image_ref='myimg',
|
||||||
network_refs=['mynet'],
|
nics=[{'network': 'mynet'}],
|
||||||
root_disk_size=None,
|
root_disk_size=None,
|
||||||
ssh_keys=[],
|
ssh_keys=[],
|
||||||
netboot=False,
|
netboot=False,
|
||||||
@ -98,7 +98,7 @@ class TestMain(testtools.TestCase):
|
|||||||
mock_pr.return_value.provision_node.assert_called_once_with(
|
mock_pr.return_value.provision_node.assert_called_once_with(
|
||||||
mock_pr.return_value.reserve_node.return_value,
|
mock_pr.return_value.reserve_node.return_value,
|
||||||
image_ref='myimg',
|
image_ref='myimg',
|
||||||
network_refs=['mynet'],
|
nics=[{'network': 'mynet'}],
|
||||||
root_disk_size=None,
|
root_disk_size=None,
|
||||||
ssh_keys=[],
|
ssh_keys=[],
|
||||||
netboot=False,
|
netboot=False,
|
||||||
@ -135,7 +135,7 @@ class TestMain(testtools.TestCase):
|
|||||||
mock_pr.return_value.provision_node.assert_called_once_with(
|
mock_pr.return_value.provision_node.assert_called_once_with(
|
||||||
mock_pr.return_value.reserve_node.return_value,
|
mock_pr.return_value.reserve_node.return_value,
|
||||||
image_ref='myimg',
|
image_ref='myimg',
|
||||||
network_refs=['mynet'],
|
nics=[{'network': 'mynet'}],
|
||||||
root_disk_size=None,
|
root_disk_size=None,
|
||||||
ssh_keys=[],
|
ssh_keys=[],
|
||||||
netboot=False,
|
netboot=False,
|
||||||
@ -159,8 +159,68 @@ class TestMain(testtools.TestCase):
|
|||||||
mock_pr.return_value.provision_node.assert_called_once_with(
|
mock_pr.return_value.provision_node.assert_called_once_with(
|
||||||
mock_pr.return_value.reserve_node.return_value,
|
mock_pr.return_value.reserve_node.return_value,
|
||||||
image_ref='myimg',
|
image_ref='myimg',
|
||||||
network_refs=['mynet'],
|
nics=[{'network': 'mynet'}],
|
||||||
root_disk_size=None,
|
root_disk_size=None,
|
||||||
ssh_keys=['foo'],
|
ssh_keys=['foo'],
|
||||||
netboot=False,
|
netboot=False,
|
||||||
wait=1800)
|
wait=1800)
|
||||||
|
|
||||||
|
def test_args_port(self, mock_os_conf, mock_pr):
|
||||||
|
args = ['deploy', '--port', 'myport', '--image', 'myimg', 'compute']
|
||||||
|
_cmd.main(args)
|
||||||
|
mock_pr.assert_called_once_with(
|
||||||
|
cloud_region=mock_os_conf.return_value.get_one.return_value,
|
||||||
|
dry_run=False)
|
||||||
|
mock_pr.return_value.reserve_node.assert_called_once_with(
|
||||||
|
resource_class='compute',
|
||||||
|
capabilities={}
|
||||||
|
)
|
||||||
|
mock_pr.return_value.provision_node.assert_called_once_with(
|
||||||
|
mock_pr.return_value.reserve_node.return_value,
|
||||||
|
image_ref='myimg',
|
||||||
|
nics=[{'port': 'myport'}],
|
||||||
|
root_disk_size=None,
|
||||||
|
ssh_keys=[],
|
||||||
|
netboot=False,
|
||||||
|
wait=1800)
|
||||||
|
|
||||||
|
def test_args_no_nics(self, mock_os_conf, mock_pr):
|
||||||
|
args = ['deploy', '--image', 'myimg', 'compute']
|
||||||
|
_cmd.main(args)
|
||||||
|
mock_pr.assert_called_once_with(
|
||||||
|
cloud_region=mock_os_conf.return_value.get_one.return_value,
|
||||||
|
dry_run=False)
|
||||||
|
mock_pr.return_value.reserve_node.assert_called_once_with(
|
||||||
|
resource_class='compute',
|
||||||
|
capabilities={}
|
||||||
|
)
|
||||||
|
mock_pr.return_value.provision_node.assert_called_once_with(
|
||||||
|
mock_pr.return_value.reserve_node.return_value,
|
||||||
|
image_ref='myimg',
|
||||||
|
nics=None,
|
||||||
|
root_disk_size=None,
|
||||||
|
ssh_keys=[],
|
||||||
|
netboot=False,
|
||||||
|
wait=1800)
|
||||||
|
|
||||||
|
def test_args_networks_and_ports(self, mock_os_conf, mock_pr):
|
||||||
|
args = ['deploy', '--network', 'net1', '--port', 'port1',
|
||||||
|
'--port', 'port2', '--network', 'net2',
|
||||||
|
'--image', 'myimg', 'compute']
|
||||||
|
_cmd.main(args)
|
||||||
|
mock_pr.assert_called_once_with(
|
||||||
|
cloud_region=mock_os_conf.return_value.get_one.return_value,
|
||||||
|
dry_run=False)
|
||||||
|
mock_pr.return_value.reserve_node.assert_called_once_with(
|
||||||
|
resource_class='compute',
|
||||||
|
capabilities={}
|
||||||
|
)
|
||||||
|
mock_pr.return_value.provision_node.assert_called_once_with(
|
||||||
|
mock_pr.return_value.reserve_node.return_value,
|
||||||
|
image_ref='myimg',
|
||||||
|
nics=[{'network': 'net1'}, {'port': 'port1'},
|
||||||
|
{'port': 'port2'}, {'network': 'net2'}],
|
||||||
|
root_disk_size=None,
|
||||||
|
ssh_keys=[],
|
||||||
|
netboot=False,
|
||||||
|
wait=1800)
|
||||||
|
@ -75,25 +75,63 @@ class TestReserveNode(Base):
|
|||||||
self.assertIs(node, expected)
|
self.assertIs(node, expected)
|
||||||
|
|
||||||
|
|
||||||
|
CLEAN_UP = {
|
||||||
|
'/extra/metalsmith_created_ports': _os_api.REMOVE,
|
||||||
|
'/extra/metalsmith_attached_ports': _os_api.REMOVE
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
class TestProvisionNode(Base):
|
class TestProvisionNode(Base):
|
||||||
|
|
||||||
def test_ok(self):
|
def setUp(self):
|
||||||
self.pr.provision_node(self.node, 'image', ['network'])
|
super(TestProvisionNode, self).setUp()
|
||||||
|
|
||||||
self.api.create_port.assert_called_once_with(
|
|
||||||
network_id=self.api.get_network.return_value.id)
|
|
||||||
self.api.attach_port_to_node.assert_called_once_with(
|
|
||||||
self.node.uuid, self.api.create_port.return_value.id)
|
|
||||||
image = self.api.get_image_info.return_value
|
image = self.api.get_image_info.return_value
|
||||||
updates = {'/instance_info/ramdisk': image.ramdisk_id,
|
self.updates = {
|
||||||
|
'/instance_info/ramdisk': image.ramdisk_id,
|
||||||
'/instance_info/kernel': image.kernel_id,
|
'/instance_info/kernel': image.kernel_id,
|
||||||
'/instance_info/image_source': image.id,
|
'/instance_info/image_source': image.id,
|
||||||
'/instance_info/root_gb': 99, # 100 - 1
|
'/instance_info/root_gb': 99, # 100 - 1
|
||||||
'/instance_info/capabilities': {'boot_option': 'local'},
|
'/instance_info/capabilities': {'boot_option': 'local'},
|
||||||
'/extra/metalsmith_created_ports': [
|
'/extra/metalsmith_created_ports': [
|
||||||
self.api.create_port.return_value.id
|
self.api.create_port.return_value.id
|
||||||
]}
|
],
|
||||||
self.api.update_node.assert_called_once_with(self.node, updates)
|
'/extra/metalsmith_attached_ports': [
|
||||||
|
self.api.create_port.return_value.id
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
def test_ok(self):
|
||||||
|
self.pr.provision_node(self.node, 'image', [{'network': 'network'}])
|
||||||
|
|
||||||
|
self.api.create_port.assert_called_once_with(
|
||||||
|
network_id=self.api.get_network.return_value.id)
|
||||||
|
self.api.attach_port_to_node.assert_called_once_with(
|
||||||
|
self.node.uuid, self.api.create_port.return_value.id)
|
||||||
|
self.api.update_node.assert_called_once_with(self.node, self.updates)
|
||||||
|
self.api.validate_node.assert_called_once_with(self.node,
|
||||||
|
validate_deploy=True)
|
||||||
|
self.api.node_action.assert_called_once_with(self.node, 'active',
|
||||||
|
configdrive=mock.ANY)
|
||||||
|
self.assertFalse(self.api.wait_for_node_state.called)
|
||||||
|
self.assertFalse(self.api.release_node.called)
|
||||||
|
self.assertFalse(self.api.delete_port.called)
|
||||||
|
|
||||||
|
def test_with_ports(self):
|
||||||
|
self.updates['/extra/metalsmith_created_ports'] = []
|
||||||
|
self.updates['/extra/metalsmith_attached_ports'] = [
|
||||||
|
self.api.get_port.return_value.id
|
||||||
|
] * 2
|
||||||
|
|
||||||
|
self.pr.provision_node(self.node, 'image',
|
||||||
|
[{'port': 'port1'}, {'port': 'port2'}])
|
||||||
|
|
||||||
|
self.assertFalse(self.api.create_port.called)
|
||||||
|
self.api.attach_port_to_node.assert_called_with(
|
||||||
|
self.node.uuid, self.api.get_port.return_value.id)
|
||||||
|
self.assertEqual(2, self.api.attach_port_to_node.call_count)
|
||||||
|
self.assertEqual([mock.call('port1'), mock.call('port2')],
|
||||||
|
self.api.get_port.call_args_list)
|
||||||
|
self.api.update_node.assert_called_once_with(self.node, self.updates)
|
||||||
self.api.validate_node.assert_called_once_with(self.node,
|
self.api.validate_node.assert_called_once_with(self.node,
|
||||||
validate_deploy=True)
|
validate_deploy=True)
|
||||||
self.api.node_action.assert_called_once_with(self.node, 'active',
|
self.api.node_action.assert_called_once_with(self.node, 'active',
|
||||||
@ -106,20 +144,16 @@ class TestProvisionNode(Base):
|
|||||||
image = self.api.get_image_info.return_value
|
image = self.api.get_image_info.return_value
|
||||||
image.kernel_id = None
|
image.kernel_id = None
|
||||||
image.ramdisk_id = None
|
image.ramdisk_id = None
|
||||||
|
del self.updates['/instance_info/kernel']
|
||||||
|
del self.updates['/instance_info/ramdisk']
|
||||||
|
|
||||||
self.pr.provision_node(self.node, 'image', ['network'])
|
self.pr.provision_node(self.node, 'image', [{'network': 'network'}])
|
||||||
|
|
||||||
self.api.create_port.assert_called_once_with(
|
self.api.create_port.assert_called_once_with(
|
||||||
network_id=self.api.get_network.return_value.id)
|
network_id=self.api.get_network.return_value.id)
|
||||||
self.api.attach_port_to_node.assert_called_once_with(
|
self.api.attach_port_to_node.assert_called_once_with(
|
||||||
self.node.uuid, self.api.create_port.return_value.id)
|
self.node.uuid, self.api.create_port.return_value.id)
|
||||||
updates = {'/instance_info/image_source': image.id,
|
self.api.update_node.assert_called_once_with(self.node, self.updates)
|
||||||
'/instance_info/root_gb': 99, # 100 - 1
|
|
||||||
'/instance_info/capabilities': {'boot_option': 'local'},
|
|
||||||
'/extra/metalsmith_created_ports': [
|
|
||||||
self.api.create_port.return_value.id
|
|
||||||
]}
|
|
||||||
self.api.update_node.assert_called_once_with(self.node, updates)
|
|
||||||
self.api.validate_node.assert_called_once_with(self.node,
|
self.api.validate_node.assert_called_once_with(self.node,
|
||||||
validate_deploy=True)
|
validate_deploy=True)
|
||||||
self.api.node_action.assert_called_once_with(self.node, 'active',
|
self.api.node_action.assert_called_once_with(self.node, 'active',
|
||||||
@ -129,23 +163,16 @@ class TestProvisionNode(Base):
|
|||||||
self.assertFalse(self.api.delete_port.called)
|
self.assertFalse(self.api.delete_port.called)
|
||||||
|
|
||||||
def test_with_root_disk_size(self):
|
def test_with_root_disk_size(self):
|
||||||
self.pr.provision_node(self.node, 'image', ['network'],
|
self.updates['/instance_info/root_gb'] = 50
|
||||||
|
|
||||||
|
self.pr.provision_node(self.node, 'image', [{'network': 'network'}],
|
||||||
root_disk_size=50)
|
root_disk_size=50)
|
||||||
|
|
||||||
self.api.create_port.assert_called_once_with(
|
self.api.create_port.assert_called_once_with(
|
||||||
network_id=self.api.get_network.return_value.id)
|
network_id=self.api.get_network.return_value.id)
|
||||||
self.api.attach_port_to_node.assert_called_once_with(
|
self.api.attach_port_to_node.assert_called_once_with(
|
||||||
self.node.uuid, self.api.create_port.return_value.id)
|
self.node.uuid, self.api.create_port.return_value.id)
|
||||||
image = self.api.get_image_info.return_value
|
self.api.update_node.assert_called_once_with(self.node, self.updates)
|
||||||
updates = {'/instance_info/ramdisk': image.ramdisk_id,
|
|
||||||
'/instance_info/kernel': image.kernel_id,
|
|
||||||
'/instance_info/image_source': image.id,
|
|
||||||
'/instance_info/root_gb': 50,
|
|
||||||
'/instance_info/capabilities': {'boot_option': 'local'},
|
|
||||||
'/extra/metalsmith_created_ports': [
|
|
||||||
self.api.create_port.return_value.id
|
|
||||||
]}
|
|
||||||
self.api.update_node.assert_called_once_with(self.node, updates)
|
|
||||||
self.api.validate_node.assert_called_once_with(self.node,
|
self.api.validate_node.assert_called_once_with(self.node,
|
||||||
validate_deploy=True)
|
validate_deploy=True)
|
||||||
self.api.node_action.assert_called_once_with(self.node, 'active',
|
self.api.node_action.assert_called_once_with(self.node, 'active',
|
||||||
@ -159,22 +186,14 @@ class TestProvisionNode(Base):
|
|||||||
spec=['fixed_ips'],
|
spec=['fixed_ips'],
|
||||||
fixed_ips=[{'ip_address': '192.168.1.5'}, {}]
|
fixed_ips=[{'ip_address': '192.168.1.5'}, {}]
|
||||||
)
|
)
|
||||||
self.pr.provision_node(self.node, 'image', ['network'], wait=3600)
|
self.pr.provision_node(self.node, 'image', [{'network': 'network'}],
|
||||||
|
wait=3600)
|
||||||
|
|
||||||
self.api.create_port.assert_called_once_with(
|
self.api.create_port.assert_called_once_with(
|
||||||
network_id=self.api.get_network.return_value.id)
|
network_id=self.api.get_network.return_value.id)
|
||||||
self.api.attach_port_to_node.assert_called_once_with(
|
self.api.attach_port_to_node.assert_called_once_with(
|
||||||
self.node.uuid, self.api.create_port.return_value.id)
|
self.node.uuid, self.api.create_port.return_value.id)
|
||||||
image = self.api.get_image_info.return_value
|
self.api.update_node.assert_called_once_with(self.node, self.updates)
|
||||||
updates = {'/instance_info/ramdisk': image.ramdisk_id,
|
|
||||||
'/instance_info/kernel': image.kernel_id,
|
|
||||||
'/instance_info/image_source': image.id,
|
|
||||||
'/instance_info/root_gb': 99, # 100 - 1
|
|
||||||
'/instance_info/capabilities': {'boot_option': 'local'},
|
|
||||||
'/extra/metalsmith_created_ports': [
|
|
||||||
self.api.create_port.return_value.id
|
|
||||||
]}
|
|
||||||
self.api.update_node.assert_called_once_with(self.node, updates)
|
|
||||||
self.api.validate_node.assert_called_once_with(self.node,
|
self.api.validate_node.assert_called_once_with(self.node,
|
||||||
validate_deploy=True)
|
validate_deploy=True)
|
||||||
self.api.node_action.assert_called_once_with(self.node, 'active',
|
self.api.node_action.assert_called_once_with(self.node, 'active',
|
||||||
@ -192,7 +211,8 @@ class TestProvisionNode(Base):
|
|||||||
self.api.get_port.return_value = mock.Mock(
|
self.api.get_port.return_value = mock.Mock(
|
||||||
spec=['fixed_ips'], fixed_ips=[]
|
spec=['fixed_ips'], fixed_ips=[]
|
||||||
)
|
)
|
||||||
self.pr.provision_node(self.node, 'image', ['network'], wait=3600)
|
self.pr.provision_node(self.node, 'image', [{'network': 'network'}],
|
||||||
|
wait=3600)
|
||||||
|
|
||||||
self.api.node_action.assert_called_once_with(self.node, 'active',
|
self.api.node_action.assert_called_once_with(self.node, 'active',
|
||||||
configdrive=mock.ANY)
|
configdrive=mock.ANY)
|
||||||
@ -203,7 +223,7 @@ class TestProvisionNode(Base):
|
|||||||
|
|
||||||
def test_dry_run(self):
|
def test_dry_run(self):
|
||||||
self.pr._dry_run = True
|
self.pr._dry_run = True
|
||||||
self.pr.provision_node(self.node, 'image', ['network'])
|
self.pr.provision_node(self.node, 'image', [{'network': 'network'}])
|
||||||
|
|
||||||
self.assertFalse(self.api.create_port.called)
|
self.assertFalse(self.api.create_port.called)
|
||||||
self.assertFalse(self.api.attach_port_to_node.called)
|
self.assertFalse(self.api.attach_port_to_node.called)
|
||||||
@ -217,21 +237,27 @@ class TestProvisionNode(Base):
|
|||||||
self.api.node_action.side_effect = RuntimeError('boom')
|
self.api.node_action.side_effect = RuntimeError('boom')
|
||||||
self.assertRaisesRegex(RuntimeError, 'boom',
|
self.assertRaisesRegex(RuntimeError, 'boom',
|
||||||
self.pr.provision_node, self.node,
|
self.pr.provision_node, self.node,
|
||||||
'image', ['network'], wait=3600)
|
'image', [{'network': 'n1'}, {'port': 'p1'}],
|
||||||
|
wait=3600)
|
||||||
|
|
||||||
|
self.api.update_node.assert_any_call(self.node, CLEAN_UP)
|
||||||
self.assertFalse(self.api.wait_for_node_state.called)
|
self.assertFalse(self.api.wait_for_node_state.called)
|
||||||
self.api.release_node.assert_called_once_with(self.node)
|
self.api.release_node.assert_called_once_with(self.node)
|
||||||
self.api.delete_port.assert_called_once_with(
|
self.api.delete_port.assert_called_once_with(
|
||||||
self.api.create_port.return_value.id)
|
self.api.create_port.return_value.id)
|
||||||
self.api.detach_port_from_node.assert_called_once_with(
|
calls = [
|
||||||
self.node, self.api.create_port.return_value.id)
|
mock.call(self.node, self.api.create_port.return_value.id),
|
||||||
|
mock.call(self.node, self.api.get_port.return_value.id)
|
||||||
|
]
|
||||||
|
self.api.detach_port_from_node.assert_has_calls(calls, any_order=True)
|
||||||
|
|
||||||
def test_port_creation_failure(self):
|
def test_port_creation_failure(self):
|
||||||
self.api.create_port.side_effect = RuntimeError('boom')
|
self.api.create_port.side_effect = RuntimeError('boom')
|
||||||
self.assertRaisesRegex(RuntimeError, 'boom',
|
self.assertRaisesRegex(RuntimeError, 'boom',
|
||||||
self.pr.provision_node, self.node,
|
self.pr.provision_node, self.node,
|
||||||
'image', ['network'], wait=3600)
|
'image', [{'network': 'network'}], wait=3600)
|
||||||
|
|
||||||
|
self.api.update_node.assert_called_once_with(self.node, CLEAN_UP)
|
||||||
self.assertFalse(self.api.node_action.called)
|
self.assertFalse(self.api.node_action.called)
|
||||||
self.api.release_node.assert_called_once_with(self.node)
|
self.api.release_node.assert_called_once_with(self.node)
|
||||||
self.assertFalse(self.api.delete_port.called)
|
self.assertFalse(self.api.delete_port.called)
|
||||||
@ -241,8 +267,9 @@ class TestProvisionNode(Base):
|
|||||||
self.api.attach_port_to_node.side_effect = RuntimeError('boom')
|
self.api.attach_port_to_node.side_effect = RuntimeError('boom')
|
||||||
self.assertRaisesRegex(RuntimeError, 'boom',
|
self.assertRaisesRegex(RuntimeError, 'boom',
|
||||||
self.pr.provision_node, self.node,
|
self.pr.provision_node, self.node,
|
||||||
'image', ['network'], wait=3600)
|
'image', [{'network': 'network'}], wait=3600)
|
||||||
|
|
||||||
|
self.api.update_node.assert_called_once_with(self.node, CLEAN_UP)
|
||||||
self.assertFalse(self.api.node_action.called)
|
self.assertFalse(self.api.node_action.called)
|
||||||
self.api.release_node.assert_called_once_with(self.node)
|
self.api.release_node.assert_called_once_with(self.node)
|
||||||
self.api.delete_port.assert_called_once_with(
|
self.api.delete_port.assert_called_once_with(
|
||||||
@ -259,7 +286,8 @@ class TestProvisionNode(Base):
|
|||||||
self.api.node_action.side_effect = RuntimeError('boom')
|
self.api.node_action.side_effect = RuntimeError('boom')
|
||||||
self.assertRaisesRegex(RuntimeError, 'boom',
|
self.assertRaisesRegex(RuntimeError, 'boom',
|
||||||
self.pr.provision_node, self.node,
|
self.pr.provision_node, self.node,
|
||||||
'image', ['network'], wait=3600)
|
'image', [{'network': 'network'}],
|
||||||
|
wait=3600)
|
||||||
|
|
||||||
self.assertFalse(self.api.wait_for_node_state.called)
|
self.assertFalse(self.api.wait_for_node_state.called)
|
||||||
self.api.release_node.assert_called_once_with(self.node)
|
self.api.release_node.assert_called_once_with(self.node)
|
||||||
@ -270,20 +298,46 @@ class TestProvisionNode(Base):
|
|||||||
self.assertEqual(mock_log_exc.called,
|
self.assertEqual(mock_log_exc.called,
|
||||||
failed_call == 'release_node')
|
failed_call == 'release_node')
|
||||||
|
|
||||||
|
def test_failure_during_extra_update_on_deploy_failure(self):
|
||||||
|
self.api.update_node.side_effect = [self.node, AssertionError()]
|
||||||
|
self.api.node_action.side_effect = RuntimeError('boom')
|
||||||
|
self.assertRaisesRegex(RuntimeError, 'boom',
|
||||||
|
self.pr.provision_node, self.node,
|
||||||
|
'image', [{'network': 'network'}],
|
||||||
|
wait=3600)
|
||||||
|
|
||||||
|
self.assertFalse(self.api.wait_for_node_state.called)
|
||||||
|
self.api.release_node.assert_called_once_with(self.node)
|
||||||
|
self.api.delete_port.assert_called_once_with(
|
||||||
|
self.api.create_port.return_value.id)
|
||||||
|
self.api.detach_port_from_node.assert_called_once_with(
|
||||||
|
self.node, self.api.create_port.return_value.id)
|
||||||
|
|
||||||
def test_missing_image(self):
|
def test_missing_image(self):
|
||||||
self.api.get_image_info.side_effect = RuntimeError('Not found')
|
self.api.get_image_info.side_effect = RuntimeError('Not found')
|
||||||
self.assertRaisesRegex(_exceptions.InvalidImage, 'Not found',
|
self.assertRaisesRegex(_exceptions.InvalidImage, 'Not found',
|
||||||
self.pr.provision_node,
|
self.pr.provision_node,
|
||||||
self.node, 'image', ['network'])
|
self.node, 'image', [{'network': 'network'}])
|
||||||
self.assertFalse(self.api.update_node.called)
|
self.api.update_node.assert_called_once_with(self.node, CLEAN_UP)
|
||||||
self.assertFalse(self.api.node_action.called)
|
self.assertFalse(self.api.node_action.called)
|
||||||
self.api.release_node.assert_called_once_with(self.node)
|
self.api.release_node.assert_called_once_with(self.node)
|
||||||
|
|
||||||
def test_invalid_network(self):
|
def test_invalid_network(self):
|
||||||
self.api.get_network.side_effect = RuntimeError('Not found')
|
self.api.get_network.side_effect = RuntimeError('Not found')
|
||||||
self.assertRaisesRegex(_exceptions.InvalidNetwork, 'Not found',
|
self.assertRaisesRegex(_exceptions.InvalidNIC, 'Not found',
|
||||||
self.pr.provision_node,
|
self.pr.provision_node,
|
||||||
self.node, 'image', ['network'])
|
self.node, 'image', [{'network': 'network'}])
|
||||||
|
self.api.update_node.assert_called_once_with(self.node, CLEAN_UP)
|
||||||
|
self.assertFalse(self.api.create_port.called)
|
||||||
|
self.assertFalse(self.api.node_action.called)
|
||||||
|
self.api.release_node.assert_called_once_with(self.node)
|
||||||
|
|
||||||
|
def test_invalid_port(self):
|
||||||
|
self.api.get_port.side_effect = RuntimeError('Not found')
|
||||||
|
self.assertRaisesRegex(_exceptions.InvalidNIC, 'Not found',
|
||||||
|
self.pr.provision_node,
|
||||||
|
self.node, 'image', [{'port': 'port1'}])
|
||||||
|
self.api.update_node.assert_called_once_with(self.node, CLEAN_UP)
|
||||||
self.assertFalse(self.api.create_port.called)
|
self.assertFalse(self.api.create_port.called)
|
||||||
self.assertFalse(self.api.node_action.called)
|
self.assertFalse(self.api.node_action.called)
|
||||||
self.api.release_node.assert_called_once_with(self.node)
|
self.api.release_node.assert_called_once_with(self.node)
|
||||||
@ -292,7 +346,7 @@ class TestProvisionNode(Base):
|
|||||||
self.node.properties = {}
|
self.node.properties = {}
|
||||||
self.assertRaises(_exceptions.UnknownRootDiskSize,
|
self.assertRaises(_exceptions.UnknownRootDiskSize,
|
||||||
self.pr.provision_node,
|
self.pr.provision_node,
|
||||||
self.node, 'image', ['network'])
|
self.node, 'image', [{'network': 'network'}])
|
||||||
self.assertFalse(self.api.create_port.called)
|
self.assertFalse(self.api.create_port.called)
|
||||||
self.assertFalse(self.api.node_action.called)
|
self.assertFalse(self.api.node_action.called)
|
||||||
self.api.release_node.assert_called_once_with(self.node)
|
self.api.release_node.assert_called_once_with(self.node)
|
||||||
@ -302,7 +356,7 @@ class TestProvisionNode(Base):
|
|||||||
self.node.properties = {'local_gb': value}
|
self.node.properties = {'local_gb': value}
|
||||||
self.assertRaises(_exceptions.UnknownRootDiskSize,
|
self.assertRaises(_exceptions.UnknownRootDiskSize,
|
||||||
self.pr.provision_node,
|
self.pr.provision_node,
|
||||||
self.node, 'image', ['network'])
|
self.node, 'image', [{'network': 'network'}])
|
||||||
self.assertFalse(self.api.create_port.called)
|
self.assertFalse(self.api.create_port.called)
|
||||||
self.assertFalse(self.api.node_action.called)
|
self.assertFalse(self.api.node_action.called)
|
||||||
self.api.release_node.assert_called_with(self.node)
|
self.api.release_node.assert_called_with(self.node)
|
||||||
@ -310,16 +364,44 @@ class TestProvisionNode(Base):
|
|||||||
def test_invalid_root_disk_size(self):
|
def test_invalid_root_disk_size(self):
|
||||||
self.assertRaises(TypeError,
|
self.assertRaises(TypeError,
|
||||||
self.pr.provision_node,
|
self.pr.provision_node,
|
||||||
self.node, 'image', ['network'],
|
self.node, 'image', [{'network': 'network'}],
|
||||||
root_disk_size={})
|
root_disk_size={})
|
||||||
self.assertRaises(ValueError,
|
self.assertRaises(ValueError,
|
||||||
self.pr.provision_node,
|
self.pr.provision_node,
|
||||||
self.node, 'image', ['network'],
|
self.node, 'image', [{'network': 'network'}],
|
||||||
root_disk_size=0)
|
root_disk_size=0)
|
||||||
self.assertFalse(self.api.create_port.called)
|
self.assertFalse(self.api.create_port.called)
|
||||||
self.assertFalse(self.api.node_action.called)
|
self.assertFalse(self.api.node_action.called)
|
||||||
self.api.release_node.assert_called_with(self.node)
|
self.api.release_node.assert_called_with(self.node)
|
||||||
|
|
||||||
|
def test_invalid_nics(self):
|
||||||
|
self.assertRaisesRegex(TypeError, 'must be a list',
|
||||||
|
self.pr.provision_node,
|
||||||
|
self.node, 'image', 42)
|
||||||
|
self.assertFalse(self.api.create_port.called)
|
||||||
|
self.assertFalse(self.api.attach_port_to_node.called)
|
||||||
|
self.assertFalse(self.api.node_action.called)
|
||||||
|
self.api.release_node.assert_called_once_with(self.node)
|
||||||
|
|
||||||
|
def test_invalid_nic(self):
|
||||||
|
for item in ('string', ['string'], [{1: 2, 3: 4}]):
|
||||||
|
self.assertRaisesRegex(TypeError, 'must be a dict',
|
||||||
|
self.pr.provision_node,
|
||||||
|
self.node, 'image', item)
|
||||||
|
self.assertFalse(self.api.create_port.called)
|
||||||
|
self.assertFalse(self.api.attach_port_to_node.called)
|
||||||
|
self.assertFalse(self.api.node_action.called)
|
||||||
|
self.api.release_node.assert_called_with(self.node)
|
||||||
|
|
||||||
|
def test_invalid_nic_type(self):
|
||||||
|
self.assertRaisesRegex(ValueError, 'Unexpected NIC type foo',
|
||||||
|
self.pr.provision_node,
|
||||||
|
self.node, 'image', [{'foo': 'bar'}])
|
||||||
|
self.assertFalse(self.api.create_port.called)
|
||||||
|
self.assertFalse(self.api.attach_port_to_node.called)
|
||||||
|
self.assertFalse(self.api.node_action.called)
|
||||||
|
self.api.release_node.assert_called_once_with(self.node)
|
||||||
|
|
||||||
|
|
||||||
class TestUnprovisionNode(Base):
|
class TestUnprovisionNode(Base):
|
||||||
|
|
||||||
@ -333,6 +415,20 @@ class TestUnprovisionNode(Base):
|
|||||||
self.api.node_action.assert_called_once_with(self.node, 'deleted')
|
self.api.node_action.assert_called_once_with(self.node, 'deleted')
|
||||||
self.api.release_node.assert_called_once_with(self.node)
|
self.api.release_node.assert_called_once_with(self.node)
|
||||||
self.assertFalse(self.api.wait_for_node_state.called)
|
self.assertFalse(self.api.wait_for_node_state.called)
|
||||||
|
self.api.update_node.assert_called_once_with(self.node, CLEAN_UP)
|
||||||
|
|
||||||
|
def test_with_attached(self):
|
||||||
|
self.node.extra['metalsmith_created_ports'] = ['port1']
|
||||||
|
self.node.extra['metalsmith_attached_ports'] = ['port1', 'port2']
|
||||||
|
self.pr.unprovision_node(self.node)
|
||||||
|
|
||||||
|
self.api.delete_port.assert_called_once_with('port1')
|
||||||
|
calls = [mock.call(self.node, 'port1'), mock.call(self.node, 'port2')]
|
||||||
|
self.api.detach_port_from_node.assert_has_calls(calls, any_order=True)
|
||||||
|
self.api.node_action.assert_called_once_with(self.node, 'deleted')
|
||||||
|
self.api.release_node.assert_called_once_with(self.node)
|
||||||
|
self.assertFalse(self.api.wait_for_node_state.called)
|
||||||
|
self.api.update_node.assert_called_once_with(self.node, CLEAN_UP)
|
||||||
|
|
||||||
def test_with_wait(self):
|
def test_with_wait(self):
|
||||||
self.node.extra['metalsmith_created_ports'] = ['port1']
|
self.node.extra['metalsmith_created_ports'] = ['port1']
|
||||||
@ -357,3 +453,4 @@ class TestUnprovisionNode(Base):
|
|||||||
self.assertFalse(self.api.delete_port.called)
|
self.assertFalse(self.api.delete_port.called)
|
||||||
self.assertFalse(self.api.detach_port_from_node.called)
|
self.assertFalse(self.api.detach_port_from_node.called)
|
||||||
self.assertFalse(self.api.wait_for_node_state.called)
|
self.assertFalse(self.api.wait_for_node_state.called)
|
||||||
|
self.assertFalse(self.api.update_node.called)
|
||||||
|
@ -1,7 +1,21 @@
|
|||||||
|
- name: Create a port
|
||||||
|
command: openstack port create --network private test-port
|
||||||
|
when: precreate_port
|
||||||
|
|
||||||
|
- name: Set port argument
|
||||||
|
set_fact:
|
||||||
|
nic: --port test-port
|
||||||
|
when: precreate_port
|
||||||
|
|
||||||
|
- name: Set network argument
|
||||||
|
set_fact:
|
||||||
|
nic: --network private
|
||||||
|
when: not precreate_port
|
||||||
|
|
||||||
- name: Deploy a node
|
- name: Deploy a node
|
||||||
command: >
|
command: >
|
||||||
metalsmith --debug deploy
|
metalsmith --debug deploy
|
||||||
--network private
|
{{ nic }}
|
||||||
--image {{ image }}
|
--image {{ image }}
|
||||||
--ssh-public-key {{ ssh_key_file }}
|
--ssh-public-key {{ ssh_key_file }}
|
||||||
--root-disk-size 9
|
--root-disk-size 9
|
||||||
@ -28,10 +42,37 @@
|
|||||||
command: metalsmith --debug undeploy {{ active_node }}
|
command: metalsmith --debug undeploy {{ active_node }}
|
||||||
|
|
||||||
- name: Get the current status of the deployed node
|
- name: Get the current status of the deployed node
|
||||||
command: openstack baremetal node show {{ active_node }} -f value -c provision_state
|
command: openstack baremetal node show {{ active_node }} -f json
|
||||||
register: undeployed_node_result
|
register: undeployed_node_result
|
||||||
|
|
||||||
|
- name: Parse node state
|
||||||
|
set_fact:
|
||||||
|
undeployed_node: "{{ undeployed_node_result.stdout | from_json }}"
|
||||||
|
|
||||||
- name: Check that the node was undeployed
|
- name: Check that the node was undeployed
|
||||||
fail:
|
fail:
|
||||||
msg: The node is in unexpected status {{ undeployed_node_result.stdout }}
|
msg: The node is in unexpected status {{ undeployed_node }}
|
||||||
when: undeployed_node_result.stdout != "available"
|
when: undeployed_node.provision_state != "available"
|
||||||
|
|
||||||
|
- name: Check that the node extra was cleared
|
||||||
|
fail:
|
||||||
|
msg: The node still has extra {{ undeployed_node }}
|
||||||
|
when: undeployed_node.extra != {}
|
||||||
|
|
||||||
|
- name: Get attached VIFs for the node
|
||||||
|
command: openstack baremetal node vif list {{ active_node }} -f value -c ID
|
||||||
|
register: vif_list_output
|
||||||
|
|
||||||
|
- name: Check that no VIFs are still attached
|
||||||
|
fail:
|
||||||
|
msg: Some VIFs are still attached
|
||||||
|
when: vif_list_output.stdout != ""
|
||||||
|
|
||||||
|
- name: Show remaining ports
|
||||||
|
command: openstack port list
|
||||||
|
|
||||||
|
- name: Delete created port
|
||||||
|
command: openstack port delete test-port
|
||||||
|
when: precreate_port
|
||||||
|
# FIXME(dtantsur): fails because of ironic mis-behavior
|
||||||
|
ignore_errors: true
|
||||||
|
@ -51,7 +51,9 @@
|
|||||||
- include: exercise.yaml
|
- include: exercise.yaml
|
||||||
vars:
|
vars:
|
||||||
image: "{{ cirros_uec_image }}"
|
image: "{{ cirros_uec_image }}"
|
||||||
|
precreate_port: false
|
||||||
|
|
||||||
- include: exercise.yaml
|
- include: exercise.yaml
|
||||||
vars:
|
vars:
|
||||||
image: "{{ cirros_disk_image }}"
|
image: "{{ cirros_disk_image }}"
|
||||||
|
precreate_port: true
|
||||||
|
Loading…
x
Reference in New Issue
Block a user