nova-powervm/nova_powervm/virt/powervm/vm.py

782 lines
27 KiB
Python

# Copyright 2014, 2015, 2016 IBM Corp.
#
# All Rights Reserved.
#
# Licensed under the Apache License, Version 2.0 (the "License"); you may
# not use this file except in compliance with the License. You may obtain
# a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
# License for the specific language governing permissions and limitations
# under the License.
from oslo_log import log as logging
from oslo_serialization import jsonutils
import re
import six
from nova.compute import power_state
from nova import exception
from nova import objects
from nova.virt import event
from nova.virt import hardware
from pypowervm import exceptions as pvm_exc
from pypowervm.helpers import log_helper as pvm_log
from pypowervm.tasks import cna
from pypowervm.tasks import ibmi
from pypowervm.tasks import power
from pypowervm.tasks import vterm
from pypowervm.utils import lpar_builder as lpar_bldr
from pypowervm.utils import uuid as pvm_uuid
from pypowervm.utils import validation as vldn
from pypowervm.wrappers import base_partition as pvm_bp
from pypowervm.wrappers import logical_partition as pvm_lpar
from pypowervm.wrappers import managed_system as pvm_ms
from pypowervm.wrappers import network as pvm_net
from pypowervm.wrappers import shared_proc_pool as pvm_spp
from nova_powervm import conf as cfg
from nova_powervm.virt.powervm.i18n import _
from nova_powervm.virt.powervm.i18n import _LE
from nova_powervm.virt.powervm.i18n import _LI
LOG = logging.getLogger(__name__)
CONF = cfg.CONF
POWERVM_TO_NOVA_STATE = {
pvm_bp.LPARState.MIGRATING_RUNNING: power_state.RUNNING,
pvm_bp.LPARState.RUNNING: power_state.RUNNING,
pvm_bp.LPARState.STARTING: power_state.RUNNING,
pvm_bp.LPARState.MIGRATING_NOT_ACTIVE: power_state.SHUTDOWN,
pvm_bp.LPARState.NOT_ACTIVATED: power_state.SHUTDOWN,
pvm_bp.LPARState.HARDWARE_DISCOVERY: power_state.NOSTATE,
pvm_bp.LPARState.NOT_AVAILBLE: power_state.NOSTATE,
# map open firmware state to active since it can be shut down
pvm_bp.LPARState.OPEN_FIRMWARE: power_state.RUNNING,
pvm_bp.LPARState.RESUMING: power_state.NOSTATE,
pvm_bp.LPARState.SHUTTING_DOWN: power_state.NOSTATE,
pvm_bp.LPARState.SUSPENDING: power_state.NOSTATE,
pvm_bp.LPARState.UNKNOWN: power_state.NOSTATE,
pvm_bp.LPARState.SUSPENDED: power_state.SUSPENDED,
pvm_bp.LPARState.ERROR: power_state.CRASHED
}
# Groupings of PowerVM events used when considering if a state transition
# has taken place.
RUNNING_EVENTS = [
pvm_bp.LPARState.MIGRATING_RUNNING,
pvm_bp.LPARState.RUNNING,
pvm_bp.LPARState.STARTING,
pvm_bp.LPARState.OPEN_FIRMWARE,
]
STOPPED_EVENTS = [
pvm_bp.LPARState.NOT_ACTIVATED,
pvm_bp.LPARState.ERROR,
pvm_bp.LPARState.UNKNOWN,
]
SUSPENDED_EVENTS = [
pvm_bp.LPARState.SUSPENDING,
]
RESUMING_EVENTS = [
pvm_bp.LPARState.RESUMING,
]
POWERVM_STARTABLE_STATE = (pvm_bp.LPARState.NOT_ACTIVATED)
POWERVM_STOPABLE_STATE = (pvm_bp.LPARState.RUNNING, pvm_bp.LPARState.STARTING,
pvm_bp.LPARState.OPEN_FIRMWARE,
pvm_bp.LPARState.ERROR, pvm_bp.LPARState.RESUMING)
# Attributes for secure RMC
# TODO(thorst) The name of the secure RMC vswitch will change.
SECURE_RMC_VSWITCH = 'MGMTSWITCH'
SECURE_RMC_VLAN = 4094
def translate_event(pvm_state, pwr_state):
"""Translate the PowerVM state and see if it has changed.
Compare the state from PowerVM to the state from OpenStack and see if
a life cycle event should be sent to up to OpenStack.
:param pvm_state: VM state from PowerVM
:param pwr_state: Instance power state from OpenStack
:returns: life cycle event to send.
"""
trans = None
if pvm_state in RUNNING_EVENTS and pwr_state != power_state.RUNNING:
trans = event.EVENT_LIFECYCLE_STARTED
elif pvm_state in STOPPED_EVENTS and pwr_state != power_state.SHUTDOWN:
trans = event.EVENT_LIFECYCLE_STOPPED
elif (pvm_state in SUSPENDED_EVENTS and
pwr_state != power_state.SUSPENDED):
trans = event.EVENT_LIFECYCLE_SUSPENDED
elif pvm_state in RESUMING_EVENTS and pwr_state != power_state.RUNNING:
trans = event.EVENT_LIFECYCLE_RESUMED
LOG.debug('Transistion to %s' % trans)
return trans
def _translate_vm_state(pvm_state):
"""Find the current state of the lpar and convert it to
the appropriate nova.compute.power_state
:return: The appropriate integer state value from power_state
"""
if pvm_state is None:
return power_state.NOSTATE
try:
nova_state = POWERVM_TO_NOVA_STATE[pvm_state.lower()]
except KeyError:
nova_state = power_state.NOSTATE
return nova_state
class InstanceInfo(hardware.InstanceInfo):
"""Instance Information
This object tries to lazy load the attributes since the compute
manager retrieves it a lot just to check the status and doesn't need
all the attributes.
:param adapter: pypowervm adapter
:param name: instance name
:param uuid: powervm uuid
"""
def __init__(self, adapter, name, uuid):
self._adapter = adapter
self._name = name
self._uuid = uuid
self._state = None
self._mem_kb = None
self._max_mem_kb = None
self._num_cpu = None
# Non-lazy loaded properties
self.cpu_time_ns = 0
# Todo(IBM) Should this be the instance UUID vs PowerVM?
self.id = uuid
def _get_property(self, q_prop):
return get_vm_qp(self._adapter, self._uuid, q_prop)
@property
def state(self):
# return the state if we previously loaded it
if self._state is not None:
return self._state
# otherwise, fetch the value now
pvm_state = self._get_property('PartitionState')
self._state = _translate_vm_state(pvm_state)
return self._state
@property
def mem_kb(self):
# return the memory if we previously loaded it
if self._mem_kb is not None:
return self._mem_kb
# otherwise, fetch the value now
pvm_mem_kb = self._get_property('CurrentMemory')
self._mem_kb = pvm_mem_kb
return self._mem_kb
@property
def max_mem_kb(self):
# return the max memory if we previously loaded it
if self._max_mem_kb is not None:
return self._max_mem_kb
# TODO(IBM) max isn't a quick property. We need the wrapper
return self.mem_kb()
@property
def num_cpu(self):
# return the number of cpus if we previously loaded it
if self._num_cpu is not None:
return self._num_cpu
# otherwise, fetch the value now
pvm_num_cpu = self._get_property('AllocatedVirtualProcessors')
self._num_cpu = pvm_num_cpu
return self._num_cpu
def __eq__(self, other):
return (self.__class__ == other.__class__ and
self._uuid == other._uuid)
class VMBuilder(object):
"""Converts a Nova Instance/Flavor into a pypowervm LPARBuilder."""
_PVM_PROC_COMPAT = 'powervm:processor_compatibility'
_PVM_UNCAPPED = 'powervm:uncapped'
_PVM_DED_SHAR_MODE = 'powervm:dedicated_sharing_mode'
_PVM_SHAR_PROC_POOL = 'powervm:shared_proc_pool_name'
# Map of PowerVM extra specs to the lpar builder attributes.
# '' is used for attributes that are not implemented yet.
# None means there is no direct attribute mapping and must
# be handled individually
_ATTRS_MAP = {
'powervm:min_mem': lpar_bldr.MIN_MEM,
'powervm:max_mem': lpar_bldr.MAX_MEM,
'powervm:min_vcpu': lpar_bldr.MIN_VCPU,
'powervm:max_vcpu': lpar_bldr.MAX_VCPU,
'powervm:proc_units': lpar_bldr.PROC_UNITS,
'powervm:min_proc_units': lpar_bldr.MIN_PROC_U,
'powervm:max_proc_units': lpar_bldr.MAX_PROC_U,
'powervm:dedicated_proc': lpar_bldr.DED_PROCS,
'powervm:shared_weight': lpar_bldr.UNCAPPED_WEIGHT,
'powervm:availability_priority': lpar_bldr.AVAIL_PRIORITY,
_PVM_UNCAPPED: None,
_PVM_DED_SHAR_MODE: None,
_PVM_PROC_COMPAT: None,
_PVM_SHAR_PROC_POOL: None,
'powervm:srr_capability': '',
}
_DED_SHARING_MODES_MAP = {
'share_idle_procs': pvm_bp.DedicatedSharingMode.SHARE_IDLE_PROCS,
'keep_idle_procs': pvm_bp.DedicatedSharingMode.KEEP_IDLE_PROCS,
'share_idle_procs_active':
pvm_bp.DedicatedSharingMode.SHARE_IDLE_PROCS_ACTIVE,
'share_idle_procs_always':
pvm_bp.DedicatedSharingMode.SHARE_IDLE_PROCS_ALWAYS
}
def __init__(self, host_w, adapter):
"""Initialize the converter.
:param host_w: The host system wrapper.
"""
self.adapter = adapter
self.host_w = host_w
self.stdz = lpar_bldr.DefaultStandardize(
self.host_w, uncapped_weight=CONF.powervm.uncapped_proc_weight,
proc_units_factor=CONF.powervm.proc_units_factor)
def lpar_builder(self, instance, flavor):
"""Returns the pypowervm LPARBuilder for a given Nova flavor.
:param instance: the VM instance
:param flavor: The Nova instance flavor.
"""
attrs = self._format_flavor(instance, flavor)
self._add_IBMi_attrs(instance, attrs)
return lpar_bldr.LPARBuilder(self.adapter, attrs, self.stdz)
def _add_IBMi_attrs(self, instance, attrs):
distro = instance.system_metadata.get('image_os_distro', '')
if distro.lower() == 'ibmi':
attrs[lpar_bldr.ENV] = pvm_bp.LPARType.OS400
# Add other attributes in the future
def _format_flavor(self, instance, flavor):
"""Returns the pypowervm format of the flavor.
:param instance: the VM instance
:param flavor: The Nova instance flavor.
:return: a dict that can be used by the LPAR builder
"""
# The attrs are what is sent to pypowervm to convert the lpar.
attrs = {}
attrs[lpar_bldr.NAME] = instance.name
# The uuid is only actually set on a create of an LPAR
attrs[lpar_bldr.UUID] = pvm_uuid.convert_uuid_to_pvm(instance.uuid)
attrs[lpar_bldr.MEM] = flavor.memory_mb
attrs[lpar_bldr.VCPU] = flavor.vcpus
# Loop through the extra specs and process powervm keys
for key in flavor.extra_specs.keys():
# If it is not a valid key, then can skip.
if not self._is_pvm_valid_key(key):
continue
# Look for the mapping to the lpar builder
bldr_key = self._ATTRS_MAP.get(key)
# Check for no direct mapping, if the value is none, need to
# derive the complex type
if bldr_key is None:
self._build_complex_type(key, attrs, flavor)
else:
# We found a direct mapping
attrs[bldr_key] = flavor.extra_specs[key]
return attrs
def _is_pvm_valid_key(self, key):
"""Will return if this is a valid PowerVM key.
:param key: The powervm key.
:return: True if valid key. False if non-powervm key and should be
skipped. Raises an InvalidAttribute exception if is an
unknown PowerVM key.
"""
# If not a powervm key, then it is not 'pvm_valid'
if not key.startswith('powervm:'):
return False
# Check if this is a valid attribute
if key not in self._ATTRS_MAP.keys():
exc = exception.InvalidAttribute(attr=key)
raise exc
return True
def _build_complex_type(self, key, attrs, flavor):
"""If a key does not directly map, this method derives the right value.
Some types are complex, in that the flavor may have one key that maps
to several different attributes in the lpar builder. This method
handles the complex types.
:param key: The flavor's key.
:param attrs: The attribute map to put the value into.
:param flavor: The Nova instance flavor.
:return: The value to put in for the key.
"""
# Map uncapped to sharing mode
if key == self._PVM_UNCAPPED:
is_uncapped = self._flavor_bool(flavor.extra_specs[key], key)
shar_mode = (pvm_bp.SharingMode.UNCAPPED if is_uncapped
else pvm_bp.SharingMode.CAPPED)
attrs[lpar_bldr.SHARING_MODE] = shar_mode
elif key == self._PVM_DED_SHAR_MODE:
# Dedicated sharing modes...map directly
mode = self._DED_SHARING_MODES_MAP.get(
flavor.extra_specs[key])
if mode is not None:
attrs[lpar_bldr.SHARING_MODE] = mode
else:
attr = key + '=' + flavor.extra_specs[key]
exc = exception.InvalidAttribute(attr=attr)
LOG.exception(exc)
raise exc
elif key == self._PVM_SHAR_PROC_POOL:
pool_name = flavor.extra_specs[key]
attrs[lpar_bldr.SPP] = self._spp_pool_id(pool_name)
elif key == self._PVM_PROC_COMPAT:
# Handle variants of the supported values
attrs[lpar_bldr.PROC_COMPAT] = re.sub(
r'\+', '_Plus', flavor.extra_specs[key])
# TODO(IBM): Handle other attributes
else:
# There was no mapping or we didn't handle it.
exc = exception.InvalidAttribute(attr=key)
LOG.exception(exc)
raise exc
def _spp_pool_id(self, pool_name):
"""Returns the shared proc pool id for a given pool name.
:param pool_name: The shared proc pool name.
:return: The internal API id for the shared proc pool.
"""
if (pool_name is None or
pool_name == pvm_spp.DEFAULT_POOL_DISPLAY_NAME):
# The default pool is 0
return 0
# Search for the pool with this name
pool_wraps = pvm_spp.SharedProcPool.search(
self.adapter, name=pool_name,
parent_type=pvm_ms.System.schema_type,
parent_uuid=self.host_w.uuid)
# Check to make sure there is a pool with the name, and only one pool.
if len(pool_wraps) > 1:
msg = (_('Multiple Shared Processing Pools with name %(pool)s.') %
{'pool': pool_name})
raise exception.ValidationError(msg)
elif len(pool_wraps) == 0:
msg = (_('Unable to find Shared Processing Pool %(pool)s') %
{'pool': pool_name})
raise exception.ValidationError(msg)
# Return the singular pool id.
return pool_wraps[0].id
def _flavor_bool(self, val, key):
"""Will validate and return the boolean for a given value.
:param val: The value to parse into a boolean.
:param key: The flavor key.
:return: The boolean value for the attribute. If is not well formed
will raise an ValidationError.
"""
trues = ['true', 't', 'yes', 'y']
falses = ['false', 'f', 'no', 'n']
if val.lower() in trues:
return True
elif val.lower() in falses:
return False
else:
msg = (_('Flavor attribute %(attr)s must be either True or '
'False. Current value %(val)s is not allowed.') %
{'attr': key, 'val': val})
raise exception.ValidationError(msg)
def get_lpars(adapter):
"""Get a list of the LPAR wrappers."""
return pvm_lpar.LPAR.search(adapter, is_mgmt_partition=False)
def get_lpar_names(adapter):
"""Get a list of the LPAR names."""
return [x.name for x in get_lpars(adapter)]
def get_instance_wrapper(adapter, instance, host_uuid):
"""Get the LPAR wrapper for a given Nova instance.
:param adapter: The adapter for the pypowervm API
:param instance: The nova instance.
:param host_uuid: (TEMPORARY) The host UUID
:return: The pypowervm logical_partition wrapper.
"""
pvm_inst_uuid = get_pvm_uuid(instance)
resp = adapter.read(pvm_ms.System.schema_type, root_id=host_uuid,
child_type=pvm_lpar.LPAR.schema_type,
child_id=pvm_inst_uuid)
return pvm_lpar.LPAR.wrap(resp)
def instance_exists(adapter, instance, host_uuid, log_errors=False):
"""Determine if an instance exists on the host.
:param adapter: The adapter for the pypowervm API
:param instance: The nova instance.
:param host_uuid: The host UUID
:param log_errors: Indicator whether to log REST data after an exception
:return: boolean, whether the instance exists.
"""
try:
# If we're able to get the property, then it exists.
get_vm_id(adapter, get_pvm_uuid(instance), log_errors=log_errors)
return True
except exception.InstanceNotFound:
return False
def get_vm_id(adapter, lpar_uuid, log_errors=True):
"""Returns the client LPAR ID for a given UUID.
:param adapter: The pypowervm adapter.
:param lpar_uuid: The UUID for the LPAR.
:param log_errors: Indicator whether to log REST data after an exception
:return: The system id (an integer value).
"""
return get_vm_qp(adapter, lpar_uuid, qprop='PartitionID',
log_errors=log_errors)
def get_vm_qp(adapter, lpar_uuid, qprop=None, log_errors=True):
"""Returns one or all quick properties of an LPAR.
:param adapter: The pypowervm adapter.
:param lpar_uuid: The (powervm) UUID for the LPAR.
:param qprop: The quick property key to return. If specified, that single
property value is returned. If None/unspecified, all quick
properties are returned in a dictionary.
:param log_errors: Indicator whether to log REST data after an exception
:return: Either a single quick property value or a dictionary of all quick
properties.
"""
try:
kwds = dict(root_id=lpar_uuid, suffix_type='quick', suffix_parm=qprop)
if not log_errors:
# Remove the log helper from the list of helpers
helpers = adapter.helpers
try:
helpers.remove(pvm_log.log_helper)
except ValueError:
# It's not an error if we didn't find it.
pass
kwds['helpers'] = helpers
resp = adapter.read(pvm_lpar.LPAR.schema_type, **kwds)
except pvm_exc.Error as e:
if e.response.status == 404:
raise exception.InstanceNotFound(instance_id=lpar_uuid)
else:
LOG.exception(e)
raise
return jsonutils.loads(resp.body)
def crt_lpar(adapter, host_wrapper, instance, flavor):
"""Create an LPAR based on the host based on the instance
:param adapter: The adapter for the pypowervm API
:param host_wrapper: The host wrapper
:param instance: The nova instance.
:param flavor: The nova flavor.
:return: The LPAR response from the API.
"""
lpar_b = VMBuilder(host_wrapper, adapter).lpar_builder(instance, flavor)
pending_lpar_w = lpar_b.build()
vldn.LPARWrapperValidator(pending_lpar_w, host_wrapper).validate_all()
lpar_w = pending_lpar_w.create(parent_type=pvm_ms.System,
parent_uuid=host_wrapper.uuid)
return lpar_w
def update(adapter, host_wrapper, instance, flavor, entry=None, name=None):
"""Update an LPAR based on the host based on the instance
:param adapter: The adapter for the pypowervm API
:param host_wrapper: The host wrapper
:param instance: The nova instance.
:param flavor: The nova flavor.
:param entry: The instance pvm entry, if available, otherwise it will
be fetched.
:param name: VM name to use for the update. Used on resize when we want
to rename it but not use the instance name.
:returns: The updated LPAR wrapper.
"""
if not entry:
entry = get_instance_wrapper(adapter, instance, host_wrapper.uuid)
lpar_b = VMBuilder(host_wrapper, adapter).lpar_builder(instance, flavor)
lpar_b.rebuild(entry)
# Set the new name if the instance name is not desired.
if name:
entry.name = name
# Write out the new specs, return the updated version
return entry.update()
def rename(adapter, host_uuid, instance, name, entry=None):
"""Rename a VM.
:param adapter: The adapter for the pypowervm API
:param host_uuid: The host UUID.
:param instance: The nova instance.
:param name: The new name.
:param entry: The instance pvm entry, if available, otherwise it will
be fetched.
:returns: The updated LPAR wrapper.
"""
if not entry:
entry = get_instance_wrapper(adapter, instance, host_uuid)
entry.name = name
return entry.update()
def dlt_lpar(adapter, lpar_uuid):
"""Delete an LPAR
:param adapter: The adapter for the pypowervm API
:param lpar_uuid: The lpar to delete
"""
# Attempt to delete the VM. If delete fails because of vterm
# we will close the vterm and try the delete again
try:
LOG.info(_LI('Deleting virtual machine. LPARID: %s'), lpar_uuid)
# Ensure any vterms are closed. Will no-op otherwise.
vterm.close_vterm(adapter, lpar_uuid)
# Run the LPAR delete
resp = adapter.delete(pvm_lpar.LPAR.schema_type, root_id=lpar_uuid)
LOG.info(_LI('Virtual machine delete status: %d'), resp.status)
return resp
except pvm_exc.Error:
# Attempting to close vterm did not help so raise exception
LOG.error(_LE('Virtual machine delete failed: LPARID=%s'),
lpar_uuid)
raise
def power_on(adapter, instance, host_uuid, entry=None):
if entry is None:
entry = get_instance_wrapper(adapter, instance, host_uuid)
# Get the current state and see if we can start the VM
if entry.state in POWERVM_STARTABLE_STATE:
# Now start the lpar
power.power_on(entry, host_uuid)
return True
return False
def power_off(adapter, instance, host_uuid, entry=None, add_parms=None,
force_immediate=False):
if entry is None:
entry = get_instance_wrapper(adapter, instance, host_uuid)
# Get the current state and see if we can stop the VM
if entry.state in POWERVM_STOPABLE_STATE:
# Now stop the lpar
try:
power.power_off(entry, host_uuid, force_immediate=force_immediate,
add_parms=add_parms)
except Exception as e:
LOG.exception(e)
raise exception.InstancePowerOffFailure(reason=six.text_type(e))
return True
return False
def get_pvm_uuid(instance):
"""Get the corresponding PowerVM VM uuid of an instance uuid
Maps a OpenStack instance uuid to a PowerVM uuid. The UUID between the
Nova instance and PowerVM will be 1 to 1 mapped. This method runs the
algorithm against the instance's uuid to convert it to the PowerVM
UUID.
:param instance: nova.objects.instance.Instance
:return: pvm_uuid.
"""
return pvm_uuid.convert_uuid_to_pvm(instance.uuid).upper()
def _uuid_set_high_bit(pvm_uuid):
"""Turns on the high bit of a uuid
PowerVM uuids always set the byte 0, bit 0 to 0.
So to convert it to an OpenStack uuid we may have to set the high bit.
:param uuid: A PowerVM compliant uuid
:returns: A standard format uuid string
"""
return "%x%s" % (int(pvm_uuid[0], 16) | 8, pvm_uuid[1:])
def get_instance(context, pvm_uuid):
"""Get an instance, if there is one, that corresponds to the PVM UUID
Not finding the instance can be a pretty normal case when handling events.
Don't log exceptions for those cases.
:param pvm_uuid: PowerVM UUID
:return: OpenStack instance or None
"""
uuid = pvm_uuid.lower()
def get_inst():
try:
return objects.Instance.get_by_uuid(context, uuid)
except exception.InstanceNotFound:
return objects.Instance.get_by_uuid(context,
_uuid_set_high_bit(uuid))
try:
return get_inst()
except exception.InstanceNotFound:
pass
except Exception as e:
LOG.debug('PowerVM UUID not found. %s', e)
return None
def get_cnas(adapter, instance, host_uuid):
"""Returns the current CNAs on the instance.
The Client Network Adapters are the Ethernet adapters for a VM.
:param adapter: The pypowervm adapter.
:param instance: The nova instance.
:param host_uuid: The host system UUID.
:return The CNA wrappers that represent the ClientNetworkAdapters on the VM
"""
cna_resp = adapter.read(pvm_lpar.LPAR.schema_type,
root_id=get_pvm_uuid(instance),
child_type=pvm_net.CNA.schema_type)
return pvm_net.CNA.wrap(cna_resp)
def crt_vif(adapter, instance, host_uuid, vif):
"""Will create a Client Network Adapter on the system.
:param adapter: The pypowervm adapter API interface.
:param instance: The nova instance to create the VIF against.
:param host_uuid: The host system UUID.
:param vif: The nova VIF that describes the ethernet interface.
:return: The created network adapter wrapper.
"""
lpar_uuid = get_pvm_uuid(instance)
# CNA's require a VLAN. If the network doesn't provide, default to 1
vlan = vif['network']['meta'].get('vlan', 1)
return cna.crt_cna(adapter, host_uuid, lpar_uuid, vlan,
mac_addr=vif['address'])
def crt_secure_rmc_vif(adapter, instance, host_uuid):
"""Creates the Secure RMC Network Adapter on the VM.
:param adapter: The pypowervm adapter API interface.
:param instance: The nova instance to create the VIF against.
:param host_uuid: The host system UUID.
:return: The created network adapter wrapper.
"""
lpar_uuid = get_pvm_uuid(instance)
return cna.crt_cna(adapter, host_uuid, lpar_uuid, SECURE_RMC_VLAN,
vswitch=SECURE_RMC_VSWITCH, crt_vswitch=True)
def get_secure_rmc_vswitch(adapter, host_uuid):
"""Returns the vSwitch that is used for secure RMC.
:param adapter: The pypowervm adapter API interface.
:param host_uuid: The host system UUID.
:return: The wrapper for the secure RMC vSwitch. If it does not exist
on the system, None is returned.
"""
resp = adapter.read(pvm_ms.System.schema_type, root_id=host_uuid,
child_type=pvm_net.VSwitch.schema_type)
vswitches = pvm_net.VSwitch.wrap(resp)
for vswitch in vswitches:
if vswitch.name == SECURE_RMC_VSWITCH:
return vswitch
return None
def norm_mac(mac):
"""Normalizes a MAC address from pypowervm format to OpenStack.
That means that the format will be converted to lower case and will
have colons added.
:param mac: A pypowervm mac address. Ex. 1234567890AB
:return: A mac that matches the standard neutron format.
Ex. 12:34:56:78:90:ab
"""
mac = mac.lower().replace(':', '')
return ':'.join(mac[i:i + 2] for i in range(0, len(mac), 2))
def update_ibmi_settings(adapter, instance, host_uuid, boot_type):
"""Update settings of IBMi VMs on the instance.
:param adapter: The pypowervm adapter.
:param instance: The nova instance.
:param host_uuid: The host system UUID.
:param boot_type: The boot connectivity type of the instance.
"""
lpar_wrap = get_instance_wrapper(adapter, instance, host_uuid)
entry = ibmi.update_ibmi_settings(adapter, lpar_wrap, boot_type)
entry.update()