Merge "Graph cluster model instead of mapping one"
This commit is contained in:
commit
e3c6db11c6
@ -41,3 +41,5 @@ stevedore>=1.17.1 # Apache-2.0
|
|||||||
taskflow>=2.7.0 # Apache-2.0
|
taskflow>=2.7.0 # Apache-2.0
|
||||||
WebOb>=1.6.0 # MIT
|
WebOb>=1.6.0 # MIT
|
||||||
WSME>=0.8 # MIT
|
WSME>=0.8 # MIT
|
||||||
|
networkx>=1.10 # BSD
|
||||||
|
|
||||||
|
@ -64,6 +64,15 @@ class NovaHelper(object):
|
|||||||
LOG.exception(exc)
|
LOG.exception(exc)
|
||||||
raise exception.ComputeNodeNotFound(name=node_hostname)
|
raise exception.ComputeNodeNotFound(name=node_hostname)
|
||||||
|
|
||||||
|
def get_instance_list(self):
|
||||||
|
return self.nova.servers.list(search_opts={'all_tenants': True})
|
||||||
|
|
||||||
|
def get_service(self, service_id):
|
||||||
|
return self.nova.services.find(id=service_id)
|
||||||
|
|
||||||
|
def get_flavor(self, flavor_id):
|
||||||
|
return self.nova.flavors.get(flavor_id)
|
||||||
|
|
||||||
def get_aggregate_list(self):
|
def get_aggregate_list(self):
|
||||||
return self.nova.aggregates.list()
|
return self.nova.aggregates.list()
|
||||||
|
|
||||||
|
@ -34,3 +34,7 @@ class Model(object):
|
|||||||
@abc.abstractmethod
|
@abc.abstractmethod
|
||||||
def to_string(self):
|
def to_string(self):
|
||||||
raise NotImplementedError()
|
raise NotImplementedError()
|
||||||
|
|
||||||
|
@abc.abstractmethod
|
||||||
|
def to_xml(self):
|
||||||
|
raise NotImplementedError()
|
||||||
|
@ -1,7 +1,5 @@
|
|||||||
# -*- encoding: utf-8 -*-
|
# -*- encoding: utf-8 -*-
|
||||||
# Copyright (c) 2015 b<>com
|
# Copyright (c) 2017 Intel Innovation and Research Ireland Ltd.
|
||||||
#
|
|
||||||
# Authors: Jean-Emile DARTOIS <jean-emile.dartois@b-com.com>
|
|
||||||
#
|
#
|
||||||
# Licensed under the Apache License, Version 2.0 (the "License");
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
# you may not use this file except in compliance with the License.
|
# you may not use this file except in compliance with the License.
|
||||||
@ -11,13 +9,13 @@
|
|||||||
#
|
#
|
||||||
# Unless required by applicable law or agreed to in writing, software
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
# distributed under the License is distributed on an "AS IS" BASIS,
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
# implied.
|
|
||||||
# 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.
|
||||||
|
|
||||||
from oslo_log import log
|
from oslo_log import log
|
||||||
|
|
||||||
|
from watcher.common import exception
|
||||||
from watcher.common import nova_helper
|
from watcher.common import nova_helper
|
||||||
from watcher.decision_engine.model.collector import base
|
from watcher.decision_engine.model.collector import base
|
||||||
from watcher.decision_engine.model import element
|
from watcher.decision_engine.model import element
|
||||||
@ -36,7 +34,6 @@ class NovaClusterDataModelCollector(base.BaseClusterDataModelCollector):
|
|||||||
|
|
||||||
def __init__(self, config, osc=None):
|
def __init__(self, config, osc=None):
|
||||||
super(NovaClusterDataModelCollector, self).__init__(config, osc)
|
super(NovaClusterDataModelCollector, self).__init__(config, osc)
|
||||||
self.wrapper = nova_helper.NovaHelper(osc=self.osc)
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def notification_endpoints(self):
|
def notification_endpoints(self):
|
||||||
@ -62,49 +59,313 @@ class NovaClusterDataModelCollector(base.BaseClusterDataModelCollector):
|
|||||||
"""Build the compute cluster data model"""
|
"""Build the compute cluster data model"""
|
||||||
LOG.debug("Building latest Nova cluster data model")
|
LOG.debug("Building latest Nova cluster data model")
|
||||||
|
|
||||||
model = model_root.ModelRoot()
|
builder = ModelBuilder(self.osc)
|
||||||
mem = element.Resource(element.ResourceType.memory)
|
return builder.execute()
|
||||||
num_cores = element.Resource(element.ResourceType.cpu_cores)
|
|
||||||
disk = element.Resource(element.ResourceType.disk)
|
|
||||||
disk_capacity = element.Resource(element.ResourceType.disk_capacity)
|
|
||||||
model.create_resource(mem)
|
|
||||||
model.create_resource(num_cores)
|
|
||||||
model.create_resource(disk)
|
|
||||||
model.create_resource(disk_capacity)
|
|
||||||
|
|
||||||
flavor_cache = {}
|
|
||||||
nodes = self.wrapper.get_compute_node_list()
|
|
||||||
for n in nodes:
|
|
||||||
service = self.wrapper.nova.services.find(id=n.service['id'])
|
|
||||||
# create node in cluster_model_collector
|
|
||||||
node = element.ComputeNode(n.id)
|
|
||||||
node.uuid = service.host
|
|
||||||
node.hostname = n.hypervisor_hostname
|
|
||||||
# set capacity
|
|
||||||
mem.set_capacity(node, n.memory_mb)
|
|
||||||
disk.set_capacity(node, n.free_disk_gb)
|
|
||||||
disk_capacity.set_capacity(node, n.local_gb)
|
|
||||||
num_cores.set_capacity(node, n.vcpus)
|
|
||||||
node.state = n.state
|
|
||||||
node.status = n.status
|
|
||||||
model.add_node(node)
|
|
||||||
instances = self.wrapper.get_instances_by_node(str(service.host))
|
|
||||||
for v in instances:
|
|
||||||
# create VM in cluster_model_collector
|
|
||||||
instance = element.Instance()
|
|
||||||
instance.uuid = v.id
|
|
||||||
# nova/nova/compute/instance_states.py
|
|
||||||
instance.state = getattr(v, 'OS-EXT-STS:vm_state')
|
|
||||||
|
|
||||||
# set capacity
|
class ModelBuilder(object):
|
||||||
self.wrapper.get_flavor_instance(v, flavor_cache)
|
"""Build the graph-based model
|
||||||
mem.set_capacity(instance, v.flavor['ram'])
|
|
||||||
# FIXME: update all strategies to use disk_capacity
|
|
||||||
# for instances instead of disk
|
|
||||||
disk.set_capacity(instance, v.flavor['disk'])
|
|
||||||
disk_capacity.set_capacity(instance, v.flavor['disk'])
|
|
||||||
num_cores.set_capacity(instance, v.flavor['vcpus'])
|
|
||||||
|
|
||||||
model.map_instance(instance, node)
|
This model builder adds the following data"
|
||||||
|
|
||||||
return model
|
- Compute-related knowledge (Nova)
|
||||||
|
- TODO(v-francoise): Storage-related knowledge (Cinder)
|
||||||
|
- TODO(v-francoise): Network-related knowledge (Neutron)
|
||||||
|
|
||||||
|
NOTE(v-francoise): This model builder is meant to be extended in the future
|
||||||
|
to also include both storage and network information respectively coming
|
||||||
|
from Cinder and Neutron. Some prelimary work has been done in this
|
||||||
|
direction in https://review.openstack.org/#/c/362730 but since we cannot
|
||||||
|
guarantee a sufficient level of consistency for neither the storage nor the
|
||||||
|
network part before the end of the Ocata cycle, this work has been
|
||||||
|
re-scheduled for Pike. In the meantime, all the associated code has been
|
||||||
|
commented out.
|
||||||
|
"""
|
||||||
|
def __init__(self, osc):
|
||||||
|
self.osc = osc
|
||||||
|
self.model = model_root.ModelRoot()
|
||||||
|
self.nova = osc.nova()
|
||||||
|
self.nova_helper = nova_helper.NovaHelper(osc=self.osc)
|
||||||
|
# self.neutron = osc.neutron()
|
||||||
|
# self.cinder = osc.cinder()
|
||||||
|
|
||||||
|
def _add_physical_layer(self):
|
||||||
|
"""Add the physical layer of the graph.
|
||||||
|
|
||||||
|
This includes components which represent actual infrastructure
|
||||||
|
hardware.
|
||||||
|
"""
|
||||||
|
for cnode in self.nova_helper.get_compute_node_list():
|
||||||
|
self.add_compute_node(cnode)
|
||||||
|
|
||||||
|
def add_compute_node(self, node):
|
||||||
|
# Build and add base node.
|
||||||
|
compute_node = self.build_compute_node(node)
|
||||||
|
self.model.add_node(compute_node)
|
||||||
|
|
||||||
|
# NOTE(v-francoise): we can encapsulate capabilities of the node
|
||||||
|
# (special instruction sets of CPUs) in the attributes; as well as
|
||||||
|
# sub-nodes can be added re-presenting e.g. GPUs/Accelerators etc.
|
||||||
|
|
||||||
|
# # Build & add disk, memory, network and cpu nodes.
|
||||||
|
# disk_id, disk_node = self.build_disk_compute_node(base_id, node)
|
||||||
|
# self.add_node(disk_id, disk_node)
|
||||||
|
# mem_id, mem_node = self.build_memory_compute_node(base_id, node)
|
||||||
|
# self.add_node(mem_id, mem_node)
|
||||||
|
# net_id, net_node = self._build_network_compute_node(base_id)
|
||||||
|
# self.add_node(net_id, net_node)
|
||||||
|
# cpu_id, cpu_node = self.build_cpu_compute_node(base_id, node)
|
||||||
|
# self.add_node(cpu_id, cpu_node)
|
||||||
|
|
||||||
|
# # Connect the base compute node to the dependant nodes.
|
||||||
|
# self.add_edges_from([(base_id, disk_id), (base_id, mem_id),
|
||||||
|
# (base_id, cpu_id), (base_id, net_id)],
|
||||||
|
# label="contains")
|
||||||
|
|
||||||
|
def build_compute_node(self, node):
|
||||||
|
"""Build a compute node from a Nova compute node
|
||||||
|
|
||||||
|
:param node: A node hypervisor instance
|
||||||
|
:type node: :py:class:`~novaclient.v2.hypervisors.Hypervisor`
|
||||||
|
"""
|
||||||
|
# build up the compute node.
|
||||||
|
compute_service = self.nova_helper.get_service(node.service["id"])
|
||||||
|
node_attributes = {
|
||||||
|
"id": node.id,
|
||||||
|
"human_id": None, # TODO(v-francoise): get rid of it
|
||||||
|
"uuid": compute_service.host,
|
||||||
|
"hostname": node.hypervisor_hostname,
|
||||||
|
"memory": node.memory_mb,
|
||||||
|
"disk": node.free_disk_gb,
|
||||||
|
"disk_capacity": node.local_gb,
|
||||||
|
"vcpus": node.vcpus,
|
||||||
|
"state": node.state,
|
||||||
|
"status": node.status}
|
||||||
|
|
||||||
|
compute_node = element.ComputeNode(**node_attributes)
|
||||||
|
# compute_node = self._build_node("physical", "compute", "hypervisor",
|
||||||
|
# node_attributes)
|
||||||
|
return compute_node
|
||||||
|
|
||||||
|
# def _build_network_compute_node(self, base_node):
|
||||||
|
# attributes = {}
|
||||||
|
# net_node = self._build_node("physical", "network", "NIC", attributes)
|
||||||
|
# net_id = "{}_network".format(base_node)
|
||||||
|
# return net_id, net_node
|
||||||
|
|
||||||
|
# def build_disk_compute_node(self, base_node, compute):
|
||||||
|
# # Build disk node attributes.
|
||||||
|
# disk_attributes = {
|
||||||
|
# "size_gb": compute.local_gb,
|
||||||
|
# "used_gb": compute.local_gb_used,
|
||||||
|
# "available_gb": compute.free_disk_gb}
|
||||||
|
# disk_node = self._build_node("physical", "storage", "disk",
|
||||||
|
# disk_attributes)
|
||||||
|
# disk_id = "{}_disk".format(base_node)
|
||||||
|
# return disk_id, disk_node
|
||||||
|
|
||||||
|
# def build_memory_compute_node(self, base_node, compute):
|
||||||
|
# # Build memory node attributes.
|
||||||
|
# memory_attrs = {"size_mb": compute.memory_mb,
|
||||||
|
# "used_mb": compute.memory_mb_used,
|
||||||
|
# "available_mb": compute.free_ram_mb}
|
||||||
|
# memory_node = self._build_node("physical", "memory", "memory",
|
||||||
|
# memory_attrs)
|
||||||
|
# memory_id = "{}_memory".format(base_node)
|
||||||
|
# return memory_id, memory_node
|
||||||
|
|
||||||
|
# def build_cpu_compute_node(self, base_node, compute):
|
||||||
|
# # Build memory node attributes.
|
||||||
|
# cpu_attributes = {"vcpus": compute.vcpus,
|
||||||
|
# "vcpus_used": compute.vcpus_used,
|
||||||
|
# "info": jsonutils.loads(compute.cpu_info)}
|
||||||
|
# cpu_node = self._build_node("physical", "cpu", "cpu", cpu_attributes)
|
||||||
|
# cpu_id = "{}_cpu".format(base_node)
|
||||||
|
# return cpu_id, cpu_node
|
||||||
|
|
||||||
|
# @staticmethod
|
||||||
|
# def _build_node(layer, category, node_type, attributes):
|
||||||
|
# return {"layer": layer, "category": category, "type": node_type,
|
||||||
|
# "attributes": attributes}
|
||||||
|
|
||||||
|
def _add_virtual_layer(self):
|
||||||
|
"""Add the virtual layer to the graph.
|
||||||
|
|
||||||
|
This layer is the virtual components of the infrastructure,
|
||||||
|
such as vms.
|
||||||
|
"""
|
||||||
|
self._add_virtual_servers()
|
||||||
|
# self._add_virtual_network()
|
||||||
|
# self._add_virtual_storage()
|
||||||
|
|
||||||
|
def _add_virtual_servers(self):
|
||||||
|
all_instances = self.nova_helper.get_instance_list()
|
||||||
|
for inst in all_instances:
|
||||||
|
# Add Node
|
||||||
|
instance = self._build_instance_node(inst)
|
||||||
|
self.model.add_instance(instance)
|
||||||
|
# Get the cnode_name uuid.
|
||||||
|
cnode_uuid = getattr(inst, "OS-EXT-SRV-ATTR:host")
|
||||||
|
if cnode_uuid is None:
|
||||||
|
# The instance is not attached to any Compute node
|
||||||
|
continue
|
||||||
|
try:
|
||||||
|
# Nova compute node
|
||||||
|
# cnode = self.nova_helper.get_compute_node_by_hostname(
|
||||||
|
# cnode_uuid)
|
||||||
|
compute_node = self.model.get_node_by_uuid(
|
||||||
|
cnode_uuid)
|
||||||
|
# Connect the instance to its compute node
|
||||||
|
self.model.add_edge(
|
||||||
|
instance, compute_node, label='RUNS_ON')
|
||||||
|
except exception.ComputeNodeNotFound:
|
||||||
|
continue
|
||||||
|
|
||||||
|
def _build_instance_node(self, instance):
|
||||||
|
"""Build an instance node
|
||||||
|
|
||||||
|
Create an instance node for the graph using nova and the
|
||||||
|
`server` nova object.
|
||||||
|
:param instance: Nova VM object.
|
||||||
|
:return: A instance node for the graph.
|
||||||
|
"""
|
||||||
|
flavor = self.nova_helper.get_flavor(instance.flavor["id"])
|
||||||
|
instance_attributes = {
|
||||||
|
"uuid": instance.id,
|
||||||
|
"human_id": instance.human_id,
|
||||||
|
"memory": flavor.ram,
|
||||||
|
"disk": flavor.disk,
|
||||||
|
"disk_capacity": flavor.disk,
|
||||||
|
"vcpus": flavor.vcpus,
|
||||||
|
"state": getattr(instance, "OS-EXT-STS:vm_state")}
|
||||||
|
|
||||||
|
# node_attributes = dict()
|
||||||
|
# node_attributes["layer"] = "virtual"
|
||||||
|
# node_attributes["category"] = "compute"
|
||||||
|
# node_attributes["type"] = "compute"
|
||||||
|
# node_attributes["attributes"] = instance_attributes
|
||||||
|
return element.Instance(**instance_attributes)
|
||||||
|
|
||||||
|
# def _add_virtual_storage(self):
|
||||||
|
# try:
|
||||||
|
# volumes = self.cinder.volumes.list()
|
||||||
|
# except Exception:
|
||||||
|
# return
|
||||||
|
# for volume in volumes:
|
||||||
|
# volume_id, volume_node = self._build_storage_node(volume)
|
||||||
|
# self.add_node(volume_id, volume_node)
|
||||||
|
# host = self._get_volume_host_id(volume_node)
|
||||||
|
# self.add_edge(volume_id, host)
|
||||||
|
# # Add connections to an instance.
|
||||||
|
# if volume_node['attributes']['attachments']:
|
||||||
|
# for attachment in volume_node['attributes']['attachments']:
|
||||||
|
# self.add_edge(volume_id, attachment['server_id'],
|
||||||
|
# label='ATTACHED_TO')
|
||||||
|
# volume_node['attributes'].pop('attachments')
|
||||||
|
|
||||||
|
# def _add_virtual_network(self):
|
||||||
|
# try:
|
||||||
|
# routers = self.neutron.list_routers()
|
||||||
|
# except Exception:
|
||||||
|
# return
|
||||||
|
|
||||||
|
# for network in self.neutron.list_networks()['networks']:
|
||||||
|
# self.add_node(*self._build_network(network))
|
||||||
|
|
||||||
|
# for router in routers['routers']:
|
||||||
|
# self.add_node(*self._build_router(router))
|
||||||
|
|
||||||
|
# router_interfaces, _, compute_ports = self._group_ports()
|
||||||
|
# for router_interface in router_interfaces:
|
||||||
|
# interface = self._build_router_interface(router_interface)
|
||||||
|
# router_interface_id = interface[0]
|
||||||
|
# router_interface_node = interface[1]
|
||||||
|
# router_id = interface[2]
|
||||||
|
# self.add_node(router_interface_id, router_interface_node)
|
||||||
|
# self.add_edge(router_id, router_interface_id)
|
||||||
|
# network_id = router_interface_node['attributes']['network_id']
|
||||||
|
# self.add_edge(router_interface_id, network_id)
|
||||||
|
|
||||||
|
# for compute_port in compute_ports:
|
||||||
|
# cp_id, cp_node, instance_id = self._build_compute_port_node(
|
||||||
|
# compute_port)
|
||||||
|
# self.add_node(cp_id, cp_node)
|
||||||
|
# self.add_edge(cp_id, vm_id)
|
||||||
|
# net_id = cp_node['attributes']['network_id']
|
||||||
|
# self.add_edge(net_id, cp_id)
|
||||||
|
# # Connect port to physical node
|
||||||
|
# phys_net_node = "{}_network".format(cp_node['attributes']
|
||||||
|
# ['binding:host_id'])
|
||||||
|
# self.add_edge(cp_id, phys_net_node)
|
||||||
|
|
||||||
|
# def _get_volume_host_id(self, volume_node):
|
||||||
|
# host = volume_node['attributes']['os-vol-host-attr:host']
|
||||||
|
# if host.find('@') != -1:
|
||||||
|
# host = host.split('@')[0]
|
||||||
|
# elif host.find('#') != -1:
|
||||||
|
# host = host.split('#')[0]
|
||||||
|
# return "{}_disk".format(host)
|
||||||
|
|
||||||
|
# def _build_storage_node(self, volume_obj):
|
||||||
|
# volume = volume_obj.__dict__
|
||||||
|
# volume["name"] = volume["id"]
|
||||||
|
# volume.pop("id")
|
||||||
|
# volume.pop("manager")
|
||||||
|
# node = self._build_node("virtual", "storage", 'volume', volume)
|
||||||
|
# return volume["name"], node
|
||||||
|
|
||||||
|
# def _build_compute_port_node(self, compute_port):
|
||||||
|
# compute_port["name"] = compute_port["id"]
|
||||||
|
# compute_port.pop("id")
|
||||||
|
# nde_type = "{}_port".format(
|
||||||
|
# compute_port["device_owner"].split(":")[0])
|
||||||
|
# compute_port.pop("device_owner")
|
||||||
|
# device_id = compute_port["device_id"]
|
||||||
|
# compute_port.pop("device_id")
|
||||||
|
# node = self._build_node("virtual", "network", nde_type, compute_port)
|
||||||
|
# return compute_port["name"], node, device_id
|
||||||
|
|
||||||
|
# def _group_ports(self):
|
||||||
|
# router_interfaces = []
|
||||||
|
# floating_ips = []
|
||||||
|
# compute_ports = []
|
||||||
|
# interface_types = ["network:router_interface",
|
||||||
|
# 'network:router_gateway']
|
||||||
|
|
||||||
|
# for port in self.neutron.list_ports()['ports']:
|
||||||
|
# if port['device_owner'] in interface_types:
|
||||||
|
# router_interfaces.append(port)
|
||||||
|
# elif port['device_owner'].startswith('compute:'):
|
||||||
|
# compute_ports.append(port)
|
||||||
|
# elif port['device_owner'] == 'network:floatingip':
|
||||||
|
# floating_ips.append(port)
|
||||||
|
|
||||||
|
# return router_interfaces, floating_ips, compute_ports
|
||||||
|
|
||||||
|
# def _build_router_interface(self, interface):
|
||||||
|
# interface["name"] = interface["id"]
|
||||||
|
# interface.pop("id")
|
||||||
|
# node_type = interface["device_owner"].split(":")[1]
|
||||||
|
# node = self._build_node("virtual", "network", node_type, interface)
|
||||||
|
# return interface["name"], node, interface["device_id"]
|
||||||
|
|
||||||
|
# def _build_router(self, router):
|
||||||
|
# router_attrs = {"uuid": router['id'],
|
||||||
|
# "name": router['name'],
|
||||||
|
# "state": router['status']}
|
||||||
|
# node = self._build_node('virtual', 'network', 'router', router_attrs)
|
||||||
|
# return str(router['id']), node
|
||||||
|
|
||||||
|
# def _build_network(self, network):
|
||||||
|
# node = self._build_node('virtual', 'network', 'network', network)
|
||||||
|
# return network['id'], node
|
||||||
|
|
||||||
|
def execute(self):
|
||||||
|
"""Instantiates the graph with the openstack cluster data.
|
||||||
|
|
||||||
|
The graph is populated along 2 layers: virtual and physical. As each
|
||||||
|
new layer is built connections are made back to previous layers.
|
||||||
|
"""
|
||||||
|
self._add_physical_layer()
|
||||||
|
self._add_virtual_layer()
|
||||||
|
return self.model
|
||||||
|
@ -17,13 +17,51 @@
|
|||||||
# limitations under the License.
|
# limitations under the License.
|
||||||
|
|
||||||
import abc
|
import abc
|
||||||
|
import collections
|
||||||
|
|
||||||
|
from lxml import etree
|
||||||
|
from oslo_log import log
|
||||||
import six
|
import six
|
||||||
|
|
||||||
|
from watcher.objects import base
|
||||||
|
from watcher.objects import fields as wfields
|
||||||
|
|
||||||
|
LOG = log.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
@six.add_metaclass(abc.ABCMeta)
|
@six.add_metaclass(abc.ABCMeta)
|
||||||
class Element(object):
|
class Element(base.WatcherObject, base.WatcherObjectDictCompat):
|
||||||
|
|
||||||
|
# Initial version
|
||||||
|
VERSION = '1.0'
|
||||||
|
|
||||||
|
fields = {}
|
||||||
|
|
||||||
|
def __init__(self, context=None, **kwargs):
|
||||||
|
for name, field in self.fields.items():
|
||||||
|
# The idea here is to force the initialization of unspecified
|
||||||
|
# fields that have a default value
|
||||||
|
if (name not in kwargs and not field.nullable and
|
||||||
|
field.default != wfields.UnspecifiedDefault):
|
||||||
|
kwargs[name] = field.default
|
||||||
|
super(Element, self).__init__(context, **kwargs)
|
||||||
|
|
||||||
@abc.abstractmethod
|
@abc.abstractmethod
|
||||||
def accept(self, visitor):
|
def accept(self, visitor):
|
||||||
raise NotImplementedError()
|
raise NotImplementedError()
|
||||||
|
|
||||||
|
def as_xml_element(self):
|
||||||
|
sorted_fieldmap = []
|
||||||
|
for field in self.fields:
|
||||||
|
try:
|
||||||
|
value = str(self[field])
|
||||||
|
sorted_fieldmap.append((field, value))
|
||||||
|
except Exception as exc:
|
||||||
|
LOG.exception(exc)
|
||||||
|
|
||||||
|
attrib = collections.OrderedDict(sorted_fieldmap)
|
||||||
|
|
||||||
|
element_name = self.__class__.__name__
|
||||||
|
instance_el = etree.Element(element_name, attrib=attrib)
|
||||||
|
|
||||||
|
return instance_el
|
||||||
|
@ -19,39 +19,15 @@ import abc
|
|||||||
import six
|
import six
|
||||||
|
|
||||||
from watcher.decision_engine.model.element import base
|
from watcher.decision_engine.model.element import base
|
||||||
|
from watcher.objects import fields as wfields
|
||||||
|
|
||||||
|
|
||||||
@six.add_metaclass(abc.ABCMeta)
|
@six.add_metaclass(abc.ABCMeta)
|
||||||
class ComputeResource(base.Element):
|
class ComputeResource(base.Element):
|
||||||
|
|
||||||
def __init__(self):
|
VERSION = '1.0'
|
||||||
self._uuid = ""
|
|
||||||
self._human_id = ""
|
|
||||||
self._hostname = ""
|
|
||||||
|
|
||||||
@property
|
fields = {
|
||||||
def uuid(self):
|
"uuid": wfields.StringField(),
|
||||||
return self._uuid
|
"human_id": wfields.StringField(default=""),
|
||||||
|
}
|
||||||
@uuid.setter
|
|
||||||
def uuid(self, u):
|
|
||||||
self._uuid = u
|
|
||||||
|
|
||||||
@property
|
|
||||||
def hostname(self):
|
|
||||||
return self._hostname
|
|
||||||
|
|
||||||
@hostname.setter
|
|
||||||
def hostname(self, h):
|
|
||||||
self._hostname = h
|
|
||||||
|
|
||||||
@property
|
|
||||||
def human_id(self):
|
|
||||||
return self._human_id
|
|
||||||
|
|
||||||
@human_id.setter
|
|
||||||
def human_id(self, h):
|
|
||||||
self._human_id = h
|
|
||||||
|
|
||||||
def __str__(self):
|
|
||||||
return "[{0}]".format(self.uuid)
|
|
||||||
|
@ -17,6 +17,8 @@
|
|||||||
import enum
|
import enum
|
||||||
|
|
||||||
from watcher.decision_engine.model.element import compute_resource
|
from watcher.decision_engine.model.element import compute_resource
|
||||||
|
from watcher.objects import base
|
||||||
|
from watcher.objects import fields as wfields
|
||||||
|
|
||||||
|
|
||||||
class InstanceState(enum.Enum):
|
class InstanceState(enum.Enum):
|
||||||
@ -36,19 +38,17 @@ class InstanceState(enum.Enum):
|
|||||||
ERROR = 'error'
|
ERROR = 'error'
|
||||||
|
|
||||||
|
|
||||||
|
@base.WatcherObjectRegistry.register_if(False)
|
||||||
class Instance(compute_resource.ComputeResource):
|
class Instance(compute_resource.ComputeResource):
|
||||||
|
|
||||||
def __init__(self):
|
fields = {
|
||||||
super(Instance, self).__init__()
|
"state": wfields.StringField(default=InstanceState.ACTIVE.value),
|
||||||
self._state = InstanceState.ACTIVE.value
|
|
||||||
|
"memory": wfields.NonNegativeIntegerField(),
|
||||||
|
"disk": wfields.IntegerField(),
|
||||||
|
"disk_capacity": wfields.NonNegativeIntegerField(),
|
||||||
|
"vcpus": wfields.NonNegativeIntegerField(),
|
||||||
|
}
|
||||||
|
|
||||||
def accept(self, visitor):
|
def accept(self, visitor):
|
||||||
raise NotImplementedError()
|
raise NotImplementedError()
|
||||||
|
|
||||||
@property
|
|
||||||
def state(self):
|
|
||||||
return self._state
|
|
||||||
|
|
||||||
@state.setter
|
|
||||||
def state(self, state):
|
|
||||||
self._state = state
|
|
||||||
|
@ -17,6 +17,8 @@
|
|||||||
import enum
|
import enum
|
||||||
|
|
||||||
from watcher.decision_engine.model.element import compute_resource
|
from watcher.decision_engine.model.element import compute_resource
|
||||||
|
from watcher.objects import base
|
||||||
|
from watcher.objects import fields as wfields
|
||||||
|
|
||||||
|
|
||||||
class ServiceState(enum.Enum):
|
class ServiceState(enum.Enum):
|
||||||
@ -26,29 +28,20 @@ class ServiceState(enum.Enum):
|
|||||||
DISABLED = 'disabled'
|
DISABLED = 'disabled'
|
||||||
|
|
||||||
|
|
||||||
|
@base.WatcherObjectRegistry.register_if(False)
|
||||||
class ComputeNode(compute_resource.ComputeResource):
|
class ComputeNode(compute_resource.ComputeResource):
|
||||||
|
|
||||||
def __init__(self, id):
|
fields = {
|
||||||
super(ComputeNode, self).__init__()
|
"id": wfields.NonNegativeIntegerField(),
|
||||||
self.id = id
|
"hostname": wfields.StringField(),
|
||||||
self._state = ServiceState.ONLINE.value
|
"status": wfields.StringField(default=ServiceState.ENABLED.value),
|
||||||
self._status = ServiceState.ENABLED.value
|
"state": wfields.StringField(default=ServiceState.ONLINE.value),
|
||||||
|
|
||||||
|
"memory": wfields.NonNegativeIntegerField(),
|
||||||
|
"disk": wfields.IntegerField(),
|
||||||
|
"disk_capacity": wfields.NonNegativeIntegerField(),
|
||||||
|
"vcpus": wfields.NonNegativeIntegerField(),
|
||||||
|
}
|
||||||
|
|
||||||
def accept(self, visitor):
|
def accept(self, visitor):
|
||||||
raise NotImplementedError()
|
raise NotImplementedError()
|
||||||
|
|
||||||
@property
|
|
||||||
def state(self):
|
|
||||||
return self._state
|
|
||||||
|
|
||||||
@state.setter
|
|
||||||
def state(self, state):
|
|
||||||
self._state = state
|
|
||||||
|
|
||||||
@property
|
|
||||||
def status(self):
|
|
||||||
return self._status
|
|
||||||
|
|
||||||
@status.setter
|
|
||||||
def status(self, s):
|
|
||||||
self._status = s
|
|
||||||
|
@ -20,7 +20,8 @@ from watcher.common import exception
|
|||||||
|
|
||||||
|
|
||||||
class ResourceType(enum.Enum):
|
class ResourceType(enum.Enum):
|
||||||
cpu_cores = 'num_cores'
|
cpu_cores = 'vcpus'
|
||||||
|
vcpus = 'vcpus'
|
||||||
memory = 'memory'
|
memory = 'memory'
|
||||||
disk = 'disk'
|
disk = 'disk'
|
||||||
disk_capacity = 'disk_capacity'
|
disk_capacity = 'disk_capacity'
|
||||||
|
@ -1,101 +0,0 @@
|
|||||||
# -*- encoding: utf-8 -*-
|
|
||||||
# Copyright (c) 2015 b<>com
|
|
||||||
#
|
|
||||||
# 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_concurrency import lockutils
|
|
||||||
from oslo_log import log
|
|
||||||
|
|
||||||
from watcher._i18n import _LW
|
|
||||||
|
|
||||||
LOG = log.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
class Mapping(object):
|
|
||||||
def __init__(self, model):
|
|
||||||
self.model = model
|
|
||||||
self.compute_node_mapping = {}
|
|
||||||
self.instance_mapping = {}
|
|
||||||
|
|
||||||
def map(self, node, instance):
|
|
||||||
"""Select the node where the instance is launched
|
|
||||||
|
|
||||||
:param node: the node
|
|
||||||
:param instance: the virtual machine or instance
|
|
||||||
"""
|
|
||||||
with lockutils.lock(__name__):
|
|
||||||
# init first
|
|
||||||
if node.uuid not in self.compute_node_mapping.keys():
|
|
||||||
self.compute_node_mapping[node.uuid] = set()
|
|
||||||
|
|
||||||
# map node => instances
|
|
||||||
self.compute_node_mapping[node.uuid].add(instance.uuid)
|
|
||||||
|
|
||||||
# map instance => node
|
|
||||||
self.instance_mapping[instance.uuid] = node.uuid
|
|
||||||
|
|
||||||
def unmap(self, node, instance):
|
|
||||||
"""Remove the instance from the node
|
|
||||||
|
|
||||||
:param node: the node
|
|
||||||
:param instance: the virtual machine or instance
|
|
||||||
"""
|
|
||||||
self.unmap_by_uuid(node.uuid, instance.uuid)
|
|
||||||
|
|
||||||
def unmap_by_uuid(self, node_uuid, instance_uuid):
|
|
||||||
"""Remove the instance (by id) from the node (by id)
|
|
||||||
|
|
||||||
:rtype : object
|
|
||||||
"""
|
|
||||||
with lockutils.lock(__name__):
|
|
||||||
if str(node_uuid) in self.compute_node_mapping:
|
|
||||||
self.compute_node_mapping[str(node_uuid)].remove(
|
|
||||||
str(instance_uuid))
|
|
||||||
# remove instance
|
|
||||||
self.instance_mapping.pop(instance_uuid)
|
|
||||||
else:
|
|
||||||
LOG.warning(
|
|
||||||
_LW("Trying to delete the instance %(instance)s but it "
|
|
||||||
"was not found on node %(node)s") %
|
|
||||||
{'instance': instance_uuid, 'node': node_uuid})
|
|
||||||
|
|
||||||
def get_mapping(self):
|
|
||||||
return self.compute_node_mapping
|
|
||||||
|
|
||||||
def get_node_from_instance(self, instance):
|
|
||||||
return self.get_node_by_instance_uuid(instance.uuid)
|
|
||||||
|
|
||||||
def get_node_by_instance_uuid(self, instance_uuid):
|
|
||||||
"""Getting host information from the guest instance
|
|
||||||
|
|
||||||
:param instance: the uuid of the instance
|
|
||||||
:return: node
|
|
||||||
"""
|
|
||||||
return self.model.get_node_by_uuid(
|
|
||||||
self.instance_mapping[str(instance_uuid)])
|
|
||||||
|
|
||||||
def get_node_instances(self, node):
|
|
||||||
"""Get the list of instances running on the node
|
|
||||||
|
|
||||||
:param node:
|
|
||||||
:return:
|
|
||||||
"""
|
|
||||||
return self.get_node_instances_by_uuid(node.uuid)
|
|
||||||
|
|
||||||
def get_node_instances_by_uuid(self, node_uuid):
|
|
||||||
if str(node_uuid) in self.compute_node_mapping.keys():
|
|
||||||
return self.compute_node_mapping[str(node_uuid)]
|
|
||||||
else:
|
|
||||||
# empty
|
|
||||||
return set()
|
|
@ -1,5 +1,5 @@
|
|||||||
# -*- encoding: utf-8 -*-
|
# -*- encoding: utf-8 -*-
|
||||||
# Copyright (c) 2015 b<>com
|
# Copyright (c) 2016 Intel Innovation and Research Ireland Ltd.
|
||||||
#
|
#
|
||||||
# Licensed under the Apache License, Version 2.0 (the "License");
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
# you may not use this file except in compliance with the License.
|
# you may not use this file except in compliance with the License.
|
||||||
@ -9,31 +9,32 @@
|
|||||||
#
|
#
|
||||||
# Unless required by applicable law or agreed to in writing, software
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
# distributed under the License is distributed on an "AS IS" BASIS,
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
# implied.
|
|
||||||
# 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
|
Openstack implementation of the cluster graph.
|
||||||
|
"""
|
||||||
|
|
||||||
from lxml import etree
|
from lxml import etree
|
||||||
|
import networkx as nx
|
||||||
|
from oslo_concurrency import lockutils
|
||||||
|
from oslo_log import log
|
||||||
import six
|
import six
|
||||||
|
|
||||||
from watcher._i18n import _
|
from watcher._i18n import _
|
||||||
from watcher.common import exception
|
from watcher.common import exception
|
||||||
from watcher.common import utils
|
|
||||||
from watcher.decision_engine.model import base
|
from watcher.decision_engine.model import base
|
||||||
from watcher.decision_engine.model import element
|
from watcher.decision_engine.model import element
|
||||||
from watcher.decision_engine.model import mapping
|
|
||||||
|
LOG = log.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
class ModelRoot(base.Model):
|
class ModelRoot(nx.DiGraph, base.Model):
|
||||||
|
"""Cluster graph for an Openstack cluster."""
|
||||||
|
|
||||||
def __init__(self, stale=False):
|
def __init__(self, stale=False):
|
||||||
self._nodes = utils.Struct()
|
super(ModelRoot, self).__init__()
|
||||||
self._instances = utils.Struct()
|
|
||||||
self.mapping = mapping.Mapping(self)
|
|
||||||
self.resource = utils.Struct()
|
|
||||||
self.stale = stale
|
self.stale = stale
|
||||||
|
|
||||||
def __nonzero__(self):
|
def __nonzero__(self):
|
||||||
@ -41,35 +42,47 @@ class ModelRoot(base.Model):
|
|||||||
|
|
||||||
__bool__ = __nonzero__
|
__bool__ = __nonzero__
|
||||||
|
|
||||||
def assert_node(self, obj):
|
@staticmethod
|
||||||
|
def assert_node(obj):
|
||||||
if not isinstance(obj, element.ComputeNode):
|
if not isinstance(obj, element.ComputeNode):
|
||||||
raise exception.IllegalArgumentException(
|
raise exception.IllegalArgumentException(
|
||||||
message=_("'obj' argument type is not valid"))
|
message=_("'obj' argument type is not valid: %s") % type(obj))
|
||||||
|
|
||||||
def assert_instance(self, obj):
|
@staticmethod
|
||||||
|
def assert_instance(obj):
|
||||||
if not isinstance(obj, element.Instance):
|
if not isinstance(obj, element.Instance):
|
||||||
raise exception.IllegalArgumentException(
|
raise exception.IllegalArgumentException(
|
||||||
message=_("'obj' argument type is not valid"))
|
message=_("'obj' argument type is not valid"))
|
||||||
|
|
||||||
|
@lockutils.synchronized("model_root")
|
||||||
def add_node(self, node):
|
def add_node(self, node):
|
||||||
self.assert_node(node)
|
self.assert_node(node)
|
||||||
self._nodes[node.uuid] = node
|
super(ModelRoot, self).add_node(node)
|
||||||
|
|
||||||
|
@lockutils.synchronized("model_root")
|
||||||
def remove_node(self, node):
|
def remove_node(self, node):
|
||||||
self.assert_node(node)
|
self.assert_node(node)
|
||||||
if str(node.uuid) not in self._nodes:
|
try:
|
||||||
|
super(ModelRoot, self).remove_node(node)
|
||||||
|
except nx.NetworkXError as exc:
|
||||||
|
LOG.exception(exc)
|
||||||
raise exception.ComputeNodeNotFound(name=node.uuid)
|
raise exception.ComputeNodeNotFound(name=node.uuid)
|
||||||
else:
|
|
||||||
del self._nodes[node.uuid]
|
|
||||||
|
|
||||||
|
@lockutils.synchronized("model_root")
|
||||||
def add_instance(self, instance):
|
def add_instance(self, instance):
|
||||||
self.assert_instance(instance)
|
self.assert_instance(instance)
|
||||||
self._instances[instance.uuid] = instance
|
try:
|
||||||
|
super(ModelRoot, self).add_node(instance)
|
||||||
|
except nx.NetworkXError as exc:
|
||||||
|
LOG.exception(exc)
|
||||||
|
raise exception.InstanceNotFound(name=instance.uuid)
|
||||||
|
|
||||||
|
@lockutils.synchronized("model_root")
|
||||||
def remove_instance(self, instance):
|
def remove_instance(self, instance):
|
||||||
self.assert_instance(instance)
|
self.assert_instance(instance)
|
||||||
del self._instances[instance.uuid]
|
super(ModelRoot, self).remove_node(instance)
|
||||||
|
|
||||||
|
@lockutils.synchronized("model_root")
|
||||||
def map_instance(self, instance, node):
|
def map_instance(self, instance, node):
|
||||||
"""Map a newly created instance to a node
|
"""Map a newly created instance to a node
|
||||||
|
|
||||||
@ -82,38 +95,25 @@ class ModelRoot(base.Model):
|
|||||||
instance = self.get_instance_by_uuid(instance)
|
instance = self.get_instance_by_uuid(instance)
|
||||||
if isinstance(node, six.string_types):
|
if isinstance(node, six.string_types):
|
||||||
node = self.get_node_by_uuid(node)
|
node = self.get_node_by_uuid(node)
|
||||||
|
self.assert_node(node)
|
||||||
|
self.assert_instance(instance)
|
||||||
|
|
||||||
self.add_instance(instance)
|
self.add_edge(instance, node)
|
||||||
self.mapping.map(node, instance)
|
|
||||||
|
|
||||||
|
@lockutils.synchronized("model_root")
|
||||||
def unmap_instance(self, instance, node):
|
def unmap_instance(self, instance, node):
|
||||||
"""Unmap an instance from a node
|
|
||||||
|
|
||||||
:param instance: :py:class:`~.Instance` object or instance UUID
|
|
||||||
:type instance: str or :py:class:`~.Instance`
|
|
||||||
:param node: :py:class:`~.ComputeNode` object or node UUID
|
|
||||||
:type node: str or :py:class:`~.Instance`
|
|
||||||
"""
|
|
||||||
if isinstance(instance, six.string_types):
|
if isinstance(instance, six.string_types):
|
||||||
instance = self.get_instance_by_uuid(instance)
|
instance = self.get_instance_by_uuid(instance)
|
||||||
if isinstance(node, six.string_types):
|
if isinstance(node, six.string_types):
|
||||||
node = self.get_node_by_uuid(node)
|
node = self.get_node_by_uuid(node)
|
||||||
|
|
||||||
self.add_instance(instance)
|
self.remove_edge(instance, node)
|
||||||
self.mapping.unmap(node, instance)
|
|
||||||
|
|
||||||
def delete_instance(self, instance, node=None):
|
def delete_instance(self, instance, node=None):
|
||||||
if node is not None:
|
self.assert_instance(instance)
|
||||||
self.mapping.unmap(node, instance)
|
|
||||||
|
|
||||||
self.remove_instance(instance)
|
self.remove_instance(instance)
|
||||||
|
|
||||||
for resource in self.resource.values():
|
@lockutils.synchronized("model_root")
|
||||||
try:
|
|
||||||
resource.unset_capacity(instance)
|
|
||||||
except KeyError:
|
|
||||||
pass
|
|
||||||
|
|
||||||
def migrate_instance(self, instance, source_node, destination_node):
|
def migrate_instance(self, instance, source_node, destination_node):
|
||||||
"""Migrate single instance from source_node to destination_node
|
"""Migrate single instance from source_node to destination_node
|
||||||
|
|
||||||
@ -122,96 +122,96 @@ class ModelRoot(base.Model):
|
|||||||
:param destination_node:
|
:param destination_node:
|
||||||
:return:
|
:return:
|
||||||
"""
|
"""
|
||||||
|
self.assert_instance(instance)
|
||||||
|
self.assert_node(source_node)
|
||||||
|
self.assert_node(destination_node)
|
||||||
|
|
||||||
if source_node == destination_node:
|
if source_node == destination_node:
|
||||||
return False
|
return False
|
||||||
|
|
||||||
# unmap
|
# unmap
|
||||||
self.mapping.unmap(source_node, instance)
|
self.remove_edge(instance, source_node)
|
||||||
# map
|
# map
|
||||||
self.mapping.map(destination_node, instance)
|
self.add_edge(instance, destination_node)
|
||||||
return True
|
return True
|
||||||
|
|
||||||
|
@lockutils.synchronized("model_root")
|
||||||
def get_all_compute_nodes(self):
|
def get_all_compute_nodes(self):
|
||||||
return self._nodes
|
return {cn.uuid: cn for cn in self.nodes()
|
||||||
|
if isinstance(cn, element.ComputeNode)}
|
||||||
|
|
||||||
def get_node_by_uuid(self, node_uuid):
|
@lockutils.synchronized("model_root")
|
||||||
if str(node_uuid) not in self._nodes:
|
def get_node_by_uuid(self, uuid):
|
||||||
raise exception.ComputeNodeNotFound(name=node_uuid)
|
for graph_node in self.nodes():
|
||||||
return self._nodes[str(node_uuid)]
|
if (isinstance(graph_node, element.ComputeNode) and
|
||||||
|
graph_node.uuid == uuid):
|
||||||
|
return graph_node
|
||||||
|
raise exception.ComputeNodeNotFound(name=uuid)
|
||||||
|
|
||||||
|
@lockutils.synchronized("model_root")
|
||||||
def get_instance_by_uuid(self, uuid):
|
def get_instance_by_uuid(self, uuid):
|
||||||
if str(uuid) not in self._instances:
|
return self._get_instance_by_uuid(uuid)
|
||||||
|
|
||||||
|
def _get_instance_by_uuid(self, uuid):
|
||||||
|
for graph_node in self.nodes():
|
||||||
|
if (isinstance(graph_node, element.Instance) and
|
||||||
|
graph_node.uuid == str(uuid)):
|
||||||
|
return graph_node
|
||||||
raise exception.InstanceNotFound(name=uuid)
|
raise exception.InstanceNotFound(name=uuid)
|
||||||
return self._instances[str(uuid)]
|
|
||||||
|
|
||||||
|
@lockutils.synchronized("model_root")
|
||||||
def get_node_by_instance_uuid(self, instance_uuid):
|
def get_node_by_instance_uuid(self, instance_uuid):
|
||||||
"""Getting host information from the guest instance
|
instance = self._get_instance_by_uuid(instance_uuid)
|
||||||
|
for node in self.neighbors(instance):
|
||||||
:param instance_uuid: the uuid of the instance
|
if isinstance(node, element.ComputeNode):
|
||||||
:return: node
|
return node
|
||||||
"""
|
raise exception.ComputeNodeNotFound(name=instance_uuid)
|
||||||
if str(instance_uuid) not in self.mapping.instance_mapping:
|
|
||||||
raise exception.InstanceNotFound(name=instance_uuid)
|
|
||||||
return self.get_node_by_uuid(
|
|
||||||
self.mapping.instance_mapping[str(instance_uuid)])
|
|
||||||
|
|
||||||
|
@lockutils.synchronized("model_root")
|
||||||
def get_all_instances(self):
|
def get_all_instances(self):
|
||||||
return self._instances
|
return {inst.uuid: inst for inst in self.nodes()
|
||||||
|
if isinstance(inst, element.Instance)}
|
||||||
def get_mapping(self):
|
|
||||||
return self.mapping
|
|
||||||
|
|
||||||
def create_resource(self, r):
|
|
||||||
self.resource[str(r.name)] = r
|
|
||||||
|
|
||||||
|
@lockutils.synchronized("model_root")
|
||||||
def get_resource_by_uuid(self, resource_id):
|
def get_resource_by_uuid(self, resource_id):
|
||||||
return self.resource[str(resource_id)]
|
# TODO(v-francoise): deprecate this method
|
||||||
|
# This is a trick to keep the compatibility with the old model root
|
||||||
|
class Resource(object):
|
||||||
|
def __init__(self, resource_id):
|
||||||
|
if isinstance(resource_id, element.ResourceType):
|
||||||
|
resource_id = resource_id.value
|
||||||
|
self.resource_id = resource_id
|
||||||
|
|
||||||
|
def get_capacity(self, element):
|
||||||
|
# We ignore element because value already contains the value
|
||||||
|
return getattr(element, self.resource_id)
|
||||||
|
|
||||||
|
return Resource(resource_id)
|
||||||
|
|
||||||
|
@lockutils.synchronized("model_root")
|
||||||
def get_node_instances(self, node):
|
def get_node_instances(self, node):
|
||||||
return self.mapping.get_node_instances(node)
|
self.assert_node(node)
|
||||||
|
node_instances = []
|
||||||
|
for neighbor in self.predecessors(node):
|
||||||
|
if isinstance(neighbor, element.Instance):
|
||||||
|
node_instances.append(neighbor)
|
||||||
|
|
||||||
def _build_compute_node_element(self, compute_node):
|
return node_instances
|
||||||
attrib = collections.OrderedDict(
|
|
||||||
id=six.text_type(compute_node.id), uuid=compute_node.uuid,
|
|
||||||
human_id=compute_node.human_id, hostname=compute_node.hostname,
|
|
||||||
state=compute_node.state, status=compute_node.status)
|
|
||||||
|
|
||||||
for resource_name, resource in sorted(
|
|
||||||
self.resource.items(), key=lambda x: x[0]):
|
|
||||||
res_value = resource.get_capacity(compute_node)
|
|
||||||
if res_value is not None:
|
|
||||||
attrib[resource_name] = six.text_type(res_value)
|
|
||||||
|
|
||||||
compute_node_el = etree.Element("ComputeNode", attrib=attrib)
|
|
||||||
|
|
||||||
return compute_node_el
|
|
||||||
|
|
||||||
def _build_instance_element(self, instance):
|
|
||||||
attrib = collections.OrderedDict(
|
|
||||||
uuid=instance.uuid, human_id=instance.human_id,
|
|
||||||
hostname=instance.hostname, state=instance.state)
|
|
||||||
|
|
||||||
for resource_name, resource in sorted(
|
|
||||||
self.resource.items(), key=lambda x: x[0]):
|
|
||||||
res_value = resource.get_capacity(instance)
|
|
||||||
if res_value is not None:
|
|
||||||
attrib[resource_name] = six.text_type(res_value)
|
|
||||||
|
|
||||||
instance_el = etree.Element("Instance", attrib=attrib)
|
|
||||||
|
|
||||||
return instance_el
|
|
||||||
|
|
||||||
def to_string(self):
|
def to_string(self):
|
||||||
|
return self.to_xml()
|
||||||
|
|
||||||
|
def to_xml(self):
|
||||||
root = etree.Element("ModelRoot")
|
root = etree.Element("ModelRoot")
|
||||||
# Build compute node tree
|
# Build compute node tree
|
||||||
for cn in sorted(self.get_all_compute_nodes().values(),
|
for cn in sorted(self.get_all_compute_nodes().values(),
|
||||||
key=lambda cn: cn.uuid):
|
key=lambda cn: cn.uuid):
|
||||||
compute_node_el = self._build_compute_node_element(cn)
|
compute_node_el = cn.as_xml_element()
|
||||||
|
|
||||||
# Build mapped instance tree
|
# Build mapped instance tree
|
||||||
node_instance_uuids = self.get_node_instances(cn)
|
node_instances = self.get_node_instances(cn)
|
||||||
for instance_uuid in sorted(node_instance_uuids):
|
for instance in sorted(node_instances, key=lambda x: x.uuid):
|
||||||
instance = self.get_instance_by_uuid(instance_uuid)
|
instance_el = instance.as_xml_element()
|
||||||
instance_el = self._build_instance_element(instance)
|
|
||||||
compute_node_el.append(instance_el)
|
compute_node_el.append(instance_el)
|
||||||
|
|
||||||
root.append(compute_node_el)
|
root.append(compute_node_el)
|
||||||
@ -221,51 +221,23 @@ class ModelRoot(base.Model):
|
|||||||
key=lambda inst: inst.uuid):
|
key=lambda inst: inst.uuid):
|
||||||
try:
|
try:
|
||||||
self.get_node_by_instance_uuid(instance.uuid)
|
self.get_node_by_instance_uuid(instance.uuid)
|
||||||
except exception.InstanceNotFound:
|
except (exception.InstanceNotFound, exception.ComputeNodeNotFound):
|
||||||
root.append(self._build_instance_element(instance))
|
root.append(instance.as_xml_element())
|
||||||
|
|
||||||
return etree.tostring(root, pretty_print=True).decode('utf-8')
|
return etree.tostring(root, pretty_print=True).decode('utf-8')
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def from_xml(cls, data):
|
def from_xml(cls, data):
|
||||||
model = cls()
|
model = cls()
|
||||||
|
|
||||||
root = etree.fromstring(data)
|
root = etree.fromstring(data)
|
||||||
|
|
||||||
mem = element.Resource(element.ResourceType.memory)
|
|
||||||
num_cores = element.Resource(element.ResourceType.cpu_cores)
|
|
||||||
disk = element.Resource(element.ResourceType.disk)
|
|
||||||
disk_capacity = element.Resource(element.ResourceType.disk_capacity)
|
|
||||||
model.create_resource(mem)
|
|
||||||
model.create_resource(num_cores)
|
|
||||||
model.create_resource(disk)
|
|
||||||
model.create_resource(disk_capacity)
|
|
||||||
|
|
||||||
for cn in root.findall('.//ComputeNode'):
|
for cn in root.findall('.//ComputeNode'):
|
||||||
node = element.ComputeNode(cn.get('id'))
|
node = element.ComputeNode(**cn.attrib)
|
||||||
node.uuid = cn.get('uuid')
|
|
||||||
node.hostname = cn.get('hostname')
|
|
||||||
# set capacity
|
|
||||||
mem.set_capacity(node, int(cn.get(str(mem.name))))
|
|
||||||
disk.set_capacity(node, int(cn.get(str(disk.name))))
|
|
||||||
disk_capacity.set_capacity(
|
|
||||||
node, int(cn.get(str(disk_capacity.name))))
|
|
||||||
num_cores.set_capacity(node, int(cn.get(str(num_cores.name))))
|
|
||||||
node.state = cn.get('state')
|
|
||||||
node.status = cn.get('status')
|
|
||||||
|
|
||||||
model.add_node(node)
|
model.add_node(node)
|
||||||
|
|
||||||
for inst in root.findall('.//Instance'):
|
for inst in root.findall('.//Instance'):
|
||||||
instance = element.Instance()
|
instance = element.Instance(**inst.attrib)
|
||||||
instance.uuid = inst.get('uuid')
|
model.add_instance(instance)
|
||||||
instance.state = inst.get('state')
|
|
||||||
|
|
||||||
mem.set_capacity(instance, int(inst.get(str(mem.name))))
|
|
||||||
disk.set_capacity(instance, int(inst.get(str(disk.name))))
|
|
||||||
disk_capacity.set_capacity(
|
|
||||||
instance, int(inst.get(str(disk_capacity.name))))
|
|
||||||
num_cores.set_capacity(
|
|
||||||
instance, int(inst.get(str(num_cores.name))))
|
|
||||||
|
|
||||||
parent = inst.getparent()
|
parent = inst.getparent()
|
||||||
if parent.tag == 'ComputeNode':
|
if parent.tag == 'ComputeNode':
|
||||||
|
@ -18,7 +18,7 @@
|
|||||||
|
|
||||||
from oslo_log import log
|
from oslo_log import log
|
||||||
|
|
||||||
from watcher._i18n import _LI
|
from watcher._i18n import _LI, _LW
|
||||||
from watcher.common import exception
|
from watcher.common import exception
|
||||||
from watcher.common import nova_helper
|
from watcher.common import nova_helper
|
||||||
from watcher.decision_engine.model import element
|
from watcher.decision_engine.model import element
|
||||||
@ -40,14 +40,21 @@ class NovaNotification(base.NotificationEndpoint):
|
|||||||
self._nova = nova_helper.NovaHelper()
|
self._nova = nova_helper.NovaHelper()
|
||||||
return self._nova
|
return self._nova
|
||||||
|
|
||||||
def get_or_create_instance(self, uuid):
|
def get_or_create_instance(self, instance_uuid, node_uuid=None):
|
||||||
try:
|
try:
|
||||||
instance = self.cluster_data_model.get_instance_by_uuid(uuid)
|
if node_uuid:
|
||||||
|
self.get_or_create_node(node_uuid)
|
||||||
|
except exception.ComputeNodeNotFound:
|
||||||
|
LOG.warning(_LW("Could not find compute node %(node)s for "
|
||||||
|
"instance %(instance)s"),
|
||||||
|
dict(node=node_uuid, instance=instance_uuid))
|
||||||
|
try:
|
||||||
|
instance = self.cluster_data_model.get_instance_by_uuid(
|
||||||
|
instance_uuid)
|
||||||
except exception.InstanceNotFound:
|
except exception.InstanceNotFound:
|
||||||
# The instance didn't exist yet so we create a new instance object
|
# The instance didn't exist yet so we create a new instance object
|
||||||
LOG.debug("New instance created: %s", uuid)
|
LOG.debug("New instance created: %s", instance_uuid)
|
||||||
instance = element.Instance()
|
instance = element.Instance(uuid=instance_uuid)
|
||||||
instance.uuid = uuid
|
|
||||||
|
|
||||||
self.cluster_data_model.add_instance(instance)
|
self.cluster_data_model.add_instance(instance)
|
||||||
|
|
||||||
@ -57,9 +64,11 @@ class NovaNotification(base.NotificationEndpoint):
|
|||||||
instance_data = data['nova_object.data']
|
instance_data = data['nova_object.data']
|
||||||
instance_flavor_data = instance_data['flavor']['nova_object.data']
|
instance_flavor_data = instance_data['flavor']['nova_object.data']
|
||||||
|
|
||||||
instance.state = instance_data['state']
|
instance.update({
|
||||||
instance.hostname = instance_data['host_name']
|
'state': instance_data['state'],
|
||||||
instance.human_id = instance_data['display_name']
|
'hostname': instance_data['host_name'],
|
||||||
|
'human_id': instance_data['display_name'],
|
||||||
|
})
|
||||||
|
|
||||||
memory_mb = instance_flavor_data['memory_mb']
|
memory_mb = instance_flavor_data['memory_mb']
|
||||||
num_cores = instance_flavor_data['vcpus']
|
num_cores = instance_flavor_data['vcpus']
|
||||||
@ -67,7 +76,7 @@ class NovaNotification(base.NotificationEndpoint):
|
|||||||
|
|
||||||
self.update_capacity(element.ResourceType.memory, instance, memory_mb)
|
self.update_capacity(element.ResourceType.memory, instance, memory_mb)
|
||||||
self.update_capacity(
|
self.update_capacity(
|
||||||
element.ResourceType.cpu_cores, instance, num_cores)
|
element.ResourceType.vcpus, instance, num_cores)
|
||||||
self.update_capacity(
|
self.update_capacity(
|
||||||
element.ResourceType.disk, instance, disk_gb)
|
element.ResourceType.disk, instance, disk_gb)
|
||||||
self.update_capacity(
|
self.update_capacity(
|
||||||
@ -83,13 +92,14 @@ class NovaNotification(base.NotificationEndpoint):
|
|||||||
self.update_instance_mapping(instance, node)
|
self.update_instance_mapping(instance, node)
|
||||||
|
|
||||||
def update_capacity(self, resource_id, obj, value):
|
def update_capacity(self, resource_id, obj, value):
|
||||||
resource = self.cluster_data_model.get_resource_by_uuid(resource_id)
|
setattr(obj, resource_id.value, value)
|
||||||
resource.set_capacity(obj, value)
|
|
||||||
|
|
||||||
def legacy_update_instance(self, instance, data):
|
def legacy_update_instance(self, instance, data):
|
||||||
instance.state = data['state']
|
instance.update({
|
||||||
instance.hostname = data['hostname']
|
'state': data['state'],
|
||||||
instance.human_id = data['display_name']
|
'hostname': data['hostname'],
|
||||||
|
'human_id': data['display_name'],
|
||||||
|
})
|
||||||
|
|
||||||
memory_mb = data['memory_mb']
|
memory_mb = data['memory_mb']
|
||||||
num_cores = data['vcpus']
|
num_cores = data['vcpus']
|
||||||
@ -97,7 +107,7 @@ class NovaNotification(base.NotificationEndpoint):
|
|||||||
|
|
||||||
self.update_capacity(element.ResourceType.memory, instance, memory_mb)
|
self.update_capacity(element.ResourceType.memory, instance, memory_mb)
|
||||||
self.update_capacity(
|
self.update_capacity(
|
||||||
element.ResourceType.cpu_cores, instance, num_cores)
|
element.ResourceType.vcpus, instance, num_cores)
|
||||||
self.update_capacity(
|
self.update_capacity(
|
||||||
element.ResourceType.disk, instance, disk_gb)
|
element.ResourceType.disk, instance, disk_gb)
|
||||||
self.update_capacity(
|
self.update_capacity(
|
||||||
@ -115,28 +125,34 @@ class NovaNotification(base.NotificationEndpoint):
|
|||||||
def update_compute_node(self, node, data):
|
def update_compute_node(self, node, data):
|
||||||
"""Update the compute node using the notification data."""
|
"""Update the compute node using the notification data."""
|
||||||
node_data = data['nova_object.data']
|
node_data = data['nova_object.data']
|
||||||
node.hostname = node_data['host']
|
node_state = (
|
||||||
node.state = (
|
|
||||||
element.ServiceState.OFFLINE.value
|
element.ServiceState.OFFLINE.value
|
||||||
if node_data['forced_down'] else element.ServiceState.ONLINE.value)
|
if node_data['forced_down'] else element.ServiceState.ONLINE.value)
|
||||||
node.status = (
|
node_status = (
|
||||||
element.ServiceState.DISABLED.value
|
element.ServiceState.DISABLED.value
|
||||||
if node_data['disabled'] else element.ServiceState.ENABLED.value)
|
if node_data['disabled'] else element.ServiceState.ENABLED.value)
|
||||||
|
|
||||||
|
node.update({
|
||||||
|
'hostname': node_data['host'],
|
||||||
|
'state': node_state,
|
||||||
|
'status': node_status,
|
||||||
|
})
|
||||||
|
|
||||||
def create_compute_node(self, node_hostname):
|
def create_compute_node(self, node_hostname):
|
||||||
"""Update the compute node by querying the Nova API."""
|
"""Update the compute node by querying the Nova API."""
|
||||||
try:
|
try:
|
||||||
_node = self.nova.get_compute_node_by_hostname(node_hostname)
|
_node = self.nova.get_compute_node_by_hostname(node_hostname)
|
||||||
node = element.ComputeNode(_node.id)
|
node = element.ComputeNode(
|
||||||
node.uuid = node_hostname
|
id=_node.id,
|
||||||
node.hostname = _node.hypervisor_hostname
|
uuid=node_hostname,
|
||||||
node.state = _node.state
|
hostname=_node.hypervisor_hostname,
|
||||||
node.status = _node.status
|
state=_node.state,
|
||||||
|
status=_node.status)
|
||||||
|
|
||||||
self.update_capacity(
|
self.update_capacity(
|
||||||
element.ResourceType.memory, node, _node.memory_mb)
|
element.ResourceType.memory, node, _node.memory_mb)
|
||||||
self.update_capacity(
|
self.update_capacity(
|
||||||
element.ResourceType.cpu_cores, node, _node.vcpus)
|
element.ResourceType.vcpus, node, _node.vcpus)
|
||||||
self.update_capacity(
|
self.update_capacity(
|
||||||
element.ResourceType.disk, node, _node.free_disk_gb)
|
element.ResourceType.disk, node, _node.free_disk_gb)
|
||||||
self.update_capacity(
|
self.update_capacity(
|
||||||
@ -170,18 +186,20 @@ class NovaNotification(base.NotificationEndpoint):
|
|||||||
return
|
return
|
||||||
try:
|
try:
|
||||||
try:
|
try:
|
||||||
old_node = self.get_or_create_node(node.uuid)
|
current_node = (
|
||||||
|
self.cluster_data_model.get_node_by_instance_uuid(
|
||||||
|
instance.uuid) or self.get_or_create_node(node.uuid))
|
||||||
except exception.ComputeNodeNotFound as exc:
|
except exception.ComputeNodeNotFound as exc:
|
||||||
LOG.exception(exc)
|
LOG.exception(exc)
|
||||||
# If we can't create the node,
|
# If we can't create the node,
|
||||||
# we consider the instance as unmapped
|
# we consider the instance as unmapped
|
||||||
old_node = None
|
current_node = None
|
||||||
|
|
||||||
LOG.debug("Mapped node %s found", node.uuid)
|
LOG.debug("Mapped node %s found", node.uuid)
|
||||||
if node and node != old_node:
|
if current_node and node != current_node:
|
||||||
LOG.debug("Unmapping instance %s from %s",
|
LOG.debug("Unmapping instance %s from %s",
|
||||||
instance.uuid, node.uuid)
|
instance.uuid, node.uuid)
|
||||||
self.cluster_data_model.unmap_instance(instance, old_node)
|
self.cluster_data_model.unmap_instance(instance, current_node)
|
||||||
except exception.InstanceNotFound:
|
except exception.InstanceNotFound:
|
||||||
# The instance didn't exist yet so we map it for the first time
|
# The instance didn't exist yet so we map it for the first time
|
||||||
LOG.debug("New instance: mapping it to %s", node.uuid)
|
LOG.debug("New instance: mapping it to %s", node.uuid)
|
||||||
@ -221,6 +239,7 @@ class ServiceUpdated(VersionnedNotificationEndpoint):
|
|||||||
dict(event=event_type,
|
dict(event=event_type,
|
||||||
publisher=publisher_id,
|
publisher=publisher_id,
|
||||||
metadata=metadata))
|
metadata=metadata))
|
||||||
|
LOG.debug(payload)
|
||||||
node_data = payload['nova_object.data']
|
node_data = payload['nova_object.data']
|
||||||
node_uuid = node_data['host']
|
node_uuid = node_data['host']
|
||||||
try:
|
try:
|
||||||
@ -262,10 +281,12 @@ class InstanceCreated(VersionnedNotificationEndpoint):
|
|||||||
dict(event=event_type,
|
dict(event=event_type,
|
||||||
publisher=publisher_id,
|
publisher=publisher_id,
|
||||||
metadata=metadata))
|
metadata=metadata))
|
||||||
|
LOG.debug(payload)
|
||||||
instance_data = payload['nova_object.data']
|
instance_data = payload['nova_object.data']
|
||||||
|
|
||||||
instance_uuid = instance_data['uuid']
|
instance_uuid = instance_data['uuid']
|
||||||
instance = self.get_or_create_instance(instance_uuid)
|
node_uuid = instance_data.get('host')
|
||||||
|
instance = self.get_or_create_instance(instance_uuid, node_uuid)
|
||||||
|
|
||||||
self.update_instance(instance, payload)
|
self.update_instance(instance, payload)
|
||||||
|
|
||||||
@ -294,9 +315,11 @@ class InstanceUpdated(VersionnedNotificationEndpoint):
|
|||||||
dict(event=event_type,
|
dict(event=event_type,
|
||||||
publisher=publisher_id,
|
publisher=publisher_id,
|
||||||
metadata=metadata))
|
metadata=metadata))
|
||||||
|
LOG.debug(payload)
|
||||||
instance_data = payload['nova_object.data']
|
instance_data = payload['nova_object.data']
|
||||||
instance_uuid = instance_data['uuid']
|
instance_uuid = instance_data['uuid']
|
||||||
instance = self.get_or_create_instance(instance_uuid)
|
node_uuid = instance_data.get('host')
|
||||||
|
instance = self.get_or_create_instance(instance_uuid, node_uuid)
|
||||||
|
|
||||||
self.update_instance(instance, payload)
|
self.update_instance(instance, payload)
|
||||||
|
|
||||||
@ -317,10 +340,12 @@ class InstanceDeletedEnd(VersionnedNotificationEndpoint):
|
|||||||
dict(event=event_type,
|
dict(event=event_type,
|
||||||
publisher=publisher_id,
|
publisher=publisher_id,
|
||||||
metadata=metadata))
|
metadata=metadata))
|
||||||
|
LOG.debug(payload)
|
||||||
|
|
||||||
instance_data = payload['nova_object.data']
|
instance_data = payload['nova_object.data']
|
||||||
instance_uuid = instance_data['uuid']
|
instance_uuid = instance_data['uuid']
|
||||||
instance = self.get_or_create_instance(instance_uuid)
|
node_uuid = instance_data.get('host')
|
||||||
|
instance = self.get_or_create_instance(instance_uuid, node_uuid)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
node = self.get_or_create_node(instance_data['host'])
|
node = self.get_or_create_node(instance_data['host'])
|
||||||
@ -348,9 +373,11 @@ class LegacyInstanceUpdated(UnversionnedNotificationEndpoint):
|
|||||||
dict(event=event_type,
|
dict(event=event_type,
|
||||||
publisher=publisher_id,
|
publisher=publisher_id,
|
||||||
metadata=metadata))
|
metadata=metadata))
|
||||||
|
LOG.debug(payload)
|
||||||
|
|
||||||
instance_uuid = payload['instance_id']
|
instance_uuid = payload['instance_id']
|
||||||
instance = self.get_or_create_instance(instance_uuid)
|
node_uuid = payload.get('node')
|
||||||
|
instance = self.get_or_create_instance(instance_uuid, node_uuid)
|
||||||
|
|
||||||
self.legacy_update_instance(instance, payload)
|
self.legacy_update_instance(instance, payload)
|
||||||
|
|
||||||
@ -371,9 +398,11 @@ class LegacyInstanceCreatedEnd(UnversionnedNotificationEndpoint):
|
|||||||
dict(event=event_type,
|
dict(event=event_type,
|
||||||
publisher=publisher_id,
|
publisher=publisher_id,
|
||||||
metadata=metadata))
|
metadata=metadata))
|
||||||
|
LOG.debug(payload)
|
||||||
|
|
||||||
instance_uuid = payload['instance_id']
|
instance_uuid = payload['instance_id']
|
||||||
instance = self.get_or_create_instance(instance_uuid)
|
node_uuid = payload.get('node')
|
||||||
|
instance = self.get_or_create_instance(instance_uuid, node_uuid)
|
||||||
|
|
||||||
self.legacy_update_instance(instance, payload)
|
self.legacy_update_instance(instance, payload)
|
||||||
|
|
||||||
@ -394,8 +423,10 @@ class LegacyInstanceDeletedEnd(UnversionnedNotificationEndpoint):
|
|||||||
dict(event=event_type,
|
dict(event=event_type,
|
||||||
publisher=publisher_id,
|
publisher=publisher_id,
|
||||||
metadata=metadata))
|
metadata=metadata))
|
||||||
|
LOG.debug(payload)
|
||||||
instance_uuid = payload['instance_id']
|
instance_uuid = payload['instance_id']
|
||||||
instance = self.get_or_create_instance(instance_uuid)
|
node_uuid = payload.get('node')
|
||||||
|
instance = self.get_or_create_instance(instance_uuid, node_uuid)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
node = self.get_or_create_node(payload['host'])
|
node = self.get_or_create_node(payload['host'])
|
||||||
@ -423,8 +454,10 @@ class LegacyLiveMigratedEnd(UnversionnedNotificationEndpoint):
|
|||||||
dict(event=event_type,
|
dict(event=event_type,
|
||||||
publisher=publisher_id,
|
publisher=publisher_id,
|
||||||
metadata=metadata))
|
metadata=metadata))
|
||||||
|
LOG.debug(payload)
|
||||||
|
|
||||||
instance_uuid = payload['instance_id']
|
instance_uuid = payload['instance_id']
|
||||||
instance = self.get_or_create_instance(instance_uuid)
|
node_uuid = payload.get('node')
|
||||||
|
instance = self.get_or_create_instance(instance_uuid, node_uuid)
|
||||||
|
|
||||||
self.legacy_update_instance(instance, payload)
|
self.legacy_update_instance(instance, payload)
|
||||||
|
@ -13,9 +13,6 @@
|
|||||||
# implied.
|
# implied.
|
||||||
# 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 copy
|
|
||||||
|
|
||||||
from oslo_log import log
|
from oslo_log import log
|
||||||
|
|
||||||
@ -101,9 +98,8 @@ class DefaultScope(base.BaseScope):
|
|||||||
self._osc = osc
|
self._osc = osc
|
||||||
self.wrapper = nova_helper.NovaHelper(osc=self._osc)
|
self.wrapper = nova_helper.NovaHelper(osc=self._osc)
|
||||||
|
|
||||||
def _remove_instance(self, cluster_model, instance_uuid, node_name):
|
def remove_instance(self, cluster_model, instance, node_name):
|
||||||
node = cluster_model.get_node_by_uuid(node_name)
|
node = cluster_model.get_node_by_uuid(node_name)
|
||||||
instance = cluster_model.get_instance_by_uuid(instance_uuid)
|
|
||||||
cluster_model.delete_instance(instance, node)
|
cluster_model.delete_instance(instance, node)
|
||||||
|
|
||||||
def _check_wildcard(self, aggregate_list):
|
def _check_wildcard(self, aggregate_list):
|
||||||
@ -147,7 +143,7 @@ class DefaultScope(base.BaseScope):
|
|||||||
if zone.zoneName in zone_names or include_all_nodes:
|
if zone.zoneName in zone_names or include_all_nodes:
|
||||||
allowed_nodes.extend(zone.hosts.keys())
|
allowed_nodes.extend(zone.hosts.keys())
|
||||||
|
|
||||||
def _exclude_resources(self, resources, **kwargs):
|
def exclude_resources(self, resources, **kwargs):
|
||||||
instances_to_exclude = kwargs.get('instances')
|
instances_to_exclude = kwargs.get('instances')
|
||||||
nodes_to_exclude = kwargs.get('nodes')
|
nodes_to_exclude = kwargs.get('nodes')
|
||||||
for resource in resources:
|
for resource in resources:
|
||||||
@ -160,32 +156,32 @@ class DefaultScope(base.BaseScope):
|
|||||||
[host['name'] for host
|
[host['name'] for host
|
||||||
in resource['compute_nodes']])
|
in resource['compute_nodes']])
|
||||||
|
|
||||||
def _remove_node_from_model(self, nodes_to_remove, cluster_model):
|
def remove_nodes_from_model(self, nodes_to_remove, cluster_model):
|
||||||
for node_name in nodes_to_remove:
|
for node_uuid in nodes_to_remove:
|
||||||
instances = copy.copy(
|
node = cluster_model.get_node_by_uuid(node_uuid)
|
||||||
cluster_model.get_mapping().get_node_instances_by_uuid(
|
instances = cluster_model.get_node_instances(node)
|
||||||
node_name))
|
for instance in instances:
|
||||||
for instance_uuid in instances:
|
self.remove_instance(cluster_model, instance, node_uuid)
|
||||||
self._remove_instance(cluster_model, instance_uuid, node_name)
|
|
||||||
node = cluster_model.get_node_by_uuid(node_name)
|
|
||||||
cluster_model.remove_node(node)
|
cluster_model.remove_node(node)
|
||||||
|
|
||||||
def _remove_instances_from_model(self, instances_to_remove, cluster_model):
|
def remove_instances_from_model(self, instances_to_remove, cluster_model):
|
||||||
for instance_uuid in instances_to_remove:
|
for instance_uuid in instances_to_remove:
|
||||||
try:
|
try:
|
||||||
node_name = (cluster_model.get_mapping()
|
node_name = cluster_model.get_node_by_instance_uuid(
|
||||||
.get_node_by_instance_uuid(instance_uuid).uuid)
|
instance_uuid).uuid
|
||||||
except KeyError:
|
except exception.InstanceNotFound:
|
||||||
LOG.warning(_LW("The following instance %s cannot be found. "
|
LOG.warning(_LW("The following instance %s cannot be found. "
|
||||||
"It might be deleted from CDM along with node"
|
"It might be deleted from CDM along with node"
|
||||||
" instance was hosted on."),
|
" instance was hosted on."),
|
||||||
instance_uuid)
|
instance_uuid)
|
||||||
continue
|
continue
|
||||||
self._remove_instance(cluster_model, instance_uuid, node_name)
|
self.remove_instance(
|
||||||
|
cluster_model,
|
||||||
|
cluster_model.get_instance_by_uuid(instance_uuid),
|
||||||
|
node_name)
|
||||||
|
|
||||||
def get_scoped_model(self, cluster_model):
|
def get_scoped_model(self, cluster_model):
|
||||||
"""Leave only nodes and instances proposed in the audit scope"""
|
"""Leave only nodes and instances proposed in the audit scope"""
|
||||||
|
|
||||||
if not cluster_model:
|
if not cluster_model:
|
||||||
return None
|
return None
|
||||||
|
|
||||||
@ -206,7 +202,7 @@ class DefaultScope(base.BaseScope):
|
|||||||
self._collect_zones(rule['availability_zones'],
|
self._collect_zones(rule['availability_zones'],
|
||||||
allowed_nodes)
|
allowed_nodes)
|
||||||
elif 'exclude' in rule:
|
elif 'exclude' in rule:
|
||||||
self._exclude_resources(
|
self.exclude_resources(
|
||||||
rule['exclude'], instances=instances_to_exclude,
|
rule['exclude'], instances=instances_to_exclude,
|
||||||
nodes=nodes_to_exclude)
|
nodes=nodes_to_exclude)
|
||||||
|
|
||||||
@ -215,7 +211,7 @@ class DefaultScope(base.BaseScope):
|
|||||||
nodes_to_remove = set(model_hosts) - set(allowed_nodes)
|
nodes_to_remove = set(model_hosts) - set(allowed_nodes)
|
||||||
nodes_to_remove.update(nodes_to_exclude)
|
nodes_to_remove.update(nodes_to_exclude)
|
||||||
|
|
||||||
self._remove_node_from_model(nodes_to_remove, cluster_model)
|
self.remove_nodes_from_model(nodes_to_remove, cluster_model)
|
||||||
self._remove_instances_from_model(instances_to_remove, cluster_model)
|
self.remove_instances_from_model(instances_to_remove, cluster_model)
|
||||||
|
|
||||||
return cluster_model
|
return cluster_model
|
||||||
|
@ -174,15 +174,14 @@ class BasicConsolidation(base.ServerConsolidationBaseStrategy):
|
|||||||
total_disk = 0
|
total_disk = 0
|
||||||
total_mem = 0
|
total_mem = 0
|
||||||
cpu_capacity = self.compute_model.get_resource_by_uuid(
|
cpu_capacity = self.compute_model.get_resource_by_uuid(
|
||||||
element.ResourceType.cpu_cores)
|
element.ResourceType.vcpus)
|
||||||
disk_capacity = self.compute_model.get_resource_by_uuid(
|
disk_capacity = self.compute_model.get_resource_by_uuid(
|
||||||
element.ResourceType.disk)
|
element.ResourceType.disk)
|
||||||
memory_capacity = self.compute_model.get_resource_by_uuid(
|
memory_capacity = self.compute_model.get_resource_by_uuid(
|
||||||
element.ResourceType.memory)
|
element.ResourceType.memory)
|
||||||
|
|
||||||
for instance_id in self.compute_model.mapping.get_node_instances(
|
for instance in self.compute_model.get_node_instances(
|
||||||
destination_node):
|
destination_node):
|
||||||
instance = self.compute_model.get_instance_by_uuid(instance_id)
|
|
||||||
total_cores += cpu_capacity.get_capacity(instance)
|
total_cores += cpu_capacity.get_capacity(instance)
|
||||||
total_disk += disk_capacity.get_capacity(instance)
|
total_disk += disk_capacity.get_capacity(instance)
|
||||||
total_mem += memory_capacity.get_capacity(instance)
|
total_mem += memory_capacity.get_capacity(instance)
|
||||||
@ -210,7 +209,7 @@ class BasicConsolidation(base.ServerConsolidationBaseStrategy):
|
|||||||
:return: True if the threshold is not exceed
|
:return: True if the threshold is not exceed
|
||||||
"""
|
"""
|
||||||
cpu_capacity = self.compute_model.get_resource_by_uuid(
|
cpu_capacity = self.compute_model.get_resource_by_uuid(
|
||||||
element.ResourceType.cpu_cores).get_capacity(destination_node)
|
element.ResourceType.vcpus).get_capacity(destination_node)
|
||||||
disk_capacity = self.compute_model.get_resource_by_uuid(
|
disk_capacity = self.compute_model.get_resource_by_uuid(
|
||||||
element.ResourceType.disk).get_capacity(destination_node)
|
element.ResourceType.disk).get_capacity(destination_node)
|
||||||
memory_capacity = self.compute_model.get_resource_by_uuid(
|
memory_capacity = self.compute_model.get_resource_by_uuid(
|
||||||
@ -231,7 +230,7 @@ class BasicConsolidation(base.ServerConsolidationBaseStrategy):
|
|||||||
:return:
|
:return:
|
||||||
"""
|
"""
|
||||||
cpu_capacity = self.compute_model.get_resource_by_uuid(
|
cpu_capacity = self.compute_model.get_resource_by_uuid(
|
||||||
element.ResourceType.cpu_cores).get_capacity(compute_resource)
|
element.ResourceType.vcpus).get_capacity(compute_resource)
|
||||||
|
|
||||||
disk_capacity = self.compute_model.get_resource_by_uuid(
|
disk_capacity = self.compute_model.get_resource_by_uuid(
|
||||||
element.ResourceType.disk).get_capacity(compute_resource)
|
element.ResourceType.disk).get_capacity(compute_resource)
|
||||||
@ -333,7 +332,7 @@ class BasicConsolidation(base.ServerConsolidationBaseStrategy):
|
|||||||
host_avg_cpu_util = 100
|
host_avg_cpu_util = 100
|
||||||
|
|
||||||
cpu_capacity = self.compute_model.get_resource_by_uuid(
|
cpu_capacity = self.compute_model.get_resource_by_uuid(
|
||||||
element.ResourceType.cpu_cores).get_capacity(node)
|
element.ResourceType.vcpus).get_capacity(node)
|
||||||
|
|
||||||
total_cores_used = cpu_capacity * (host_avg_cpu_util / 100.0)
|
total_cores_used = cpu_capacity * (host_avg_cpu_util / 100.0)
|
||||||
|
|
||||||
@ -356,7 +355,7 @@ class BasicConsolidation(base.ServerConsolidationBaseStrategy):
|
|||||||
instance_cpu_utilization = 100
|
instance_cpu_utilization = 100
|
||||||
|
|
||||||
cpu_capacity = self.compute_model.get_resource_by_uuid(
|
cpu_capacity = self.compute_model.get_resource_by_uuid(
|
||||||
element.ResourceType.cpu_cores).get_capacity(instance)
|
element.ResourceType.vcpus).get_capacity(instance)
|
||||||
|
|
||||||
total_cores_used = cpu_capacity * (instance_cpu_utilization / 100.0)
|
total_cores_used = cpu_capacity * (instance_cpu_utilization / 100.0)
|
||||||
|
|
||||||
@ -387,28 +386,24 @@ class BasicConsolidation(base.ServerConsolidationBaseStrategy):
|
|||||||
if node.status == element.ServiceState.ENABLED.value:
|
if node.status == element.ServiceState.ENABLED.value:
|
||||||
self.number_of_enabled_nodes += 1
|
self.number_of_enabled_nodes += 1
|
||||||
|
|
||||||
count = self.compute_model.mapping.get_node_instances(node)
|
instances = self.compute_model.get_node_instances(node)
|
||||||
if len(count) > 0:
|
if len(instances) > 0:
|
||||||
result = self.calculate_score_node(node)
|
result = self.calculate_score_node(node)
|
||||||
else:
|
|
||||||
# The node has not VMs
|
|
||||||
result = 0
|
|
||||||
if len(count) > 0:
|
|
||||||
score.append((node.uuid, result))
|
score.append((node.uuid, result))
|
||||||
|
|
||||||
return score
|
return score
|
||||||
|
|
||||||
def node_and_instance_score(self, sorted_scores):
|
def node_and_instance_score(self, sorted_scores):
|
||||||
"""Get List of VMs from node"""
|
"""Get List of VMs from node"""
|
||||||
node_to_release = sorted_scores[len(sorted_scores) - 1][0]
|
node_to_release = sorted_scores[len(sorted_scores) - 1][0]
|
||||||
instances_to_migrate = self.compute_model.mapping.get_node_instances(
|
instances_to_migrate = self.compute_model.get_node_instances(
|
||||||
self.compute_model.get_node_by_uuid(node_to_release))
|
self.compute_model.get_node_by_uuid(node_to_release))
|
||||||
|
|
||||||
instance_score = []
|
instance_score = []
|
||||||
for instance_id in instances_to_migrate:
|
for instance in instances_to_migrate:
|
||||||
instance = self.compute_model.get_instance_by_uuid(instance_id)
|
|
||||||
if instance.state == element.InstanceState.ACTIVE.value:
|
if instance.state == element.InstanceState.ACTIVE.value:
|
||||||
instance_score.append(
|
instance_score.append(
|
||||||
(instance_id, self.calculate_score_instance(instance)))
|
(instance, self.calculate_score_instance(instance)))
|
||||||
|
|
||||||
return node_to_release, instance_score
|
return node_to_release, instance_score
|
||||||
|
|
||||||
@ -421,8 +416,7 @@ class BasicConsolidation(base.ServerConsolidationBaseStrategy):
|
|||||||
mig_source_node.uuid,
|
mig_source_node.uuid,
|
||||||
mig_destination_node.uuid)
|
mig_destination_node.uuid)
|
||||||
|
|
||||||
if len(self.compute_model.mapping.get_node_instances(
|
if len(self.compute_model.get_node_instances(mig_source_node)) == 0:
|
||||||
mig_source_node)) == 0:
|
|
||||||
self.add_change_service_state(mig_source_node.
|
self.add_change_service_state(mig_source_node.
|
||||||
uuid,
|
uuid,
|
||||||
element.ServiceState.DISABLED.value)
|
element.ServiceState.DISABLED.value)
|
||||||
@ -431,10 +425,8 @@ class BasicConsolidation(base.ServerConsolidationBaseStrategy):
|
|||||||
def calculate_num_migrations(self, sorted_instances, node_to_release,
|
def calculate_num_migrations(self, sorted_instances, node_to_release,
|
||||||
sorted_score):
|
sorted_score):
|
||||||
number_migrations = 0
|
number_migrations = 0
|
||||||
for instance in sorted_instances:
|
for mig_instance, __ in sorted_instances:
|
||||||
for j in range(0, len(sorted_score)):
|
for j in range(0, len(sorted_score)):
|
||||||
mig_instance = self.compute_model.get_instance_by_uuid(
|
|
||||||
instance[0])
|
|
||||||
mig_source_node = self.compute_model.get_node_by_uuid(
|
mig_source_node = self.compute_model.get_node_by_uuid(
|
||||||
node_to_release)
|
node_to_release)
|
||||||
mig_destination_node = self.compute_model.get_node_by_uuid(
|
mig_destination_node = self.compute_model.get_node_by_uuid(
|
||||||
|
@ -124,13 +124,11 @@ class OutletTempControl(base.ThermalOptimizationBaseStrategy):
|
|||||||
def calc_used_res(self, node, cpu_capacity,
|
def calc_used_res(self, node, cpu_capacity,
|
||||||
memory_capacity, disk_capacity):
|
memory_capacity, disk_capacity):
|
||||||
"""Calculate the used vcpus, memory and disk based on VM flavors"""
|
"""Calculate the used vcpus, memory and disk based on VM flavors"""
|
||||||
instances = self.compute_model.mapping.get_node_instances(node)
|
instances = self.compute_model.get_node_instances(node)
|
||||||
vcpus_used = 0
|
vcpus_used = 0
|
||||||
memory_mb_used = 0
|
memory_mb_used = 0
|
||||||
disk_gb_used = 0
|
disk_gb_used = 0
|
||||||
if len(instances) > 0:
|
for instance in instances:
|
||||||
for instance_id in instances:
|
|
||||||
instance = self.compute_model.get_instance_by_uuid(instance_id)
|
|
||||||
vcpus_used += cpu_capacity.get_capacity(instance)
|
vcpus_used += cpu_capacity.get_capacity(instance)
|
||||||
memory_mb_used += memory_capacity.get_capacity(instance)
|
memory_mb_used += memory_capacity.get_capacity(instance)
|
||||||
disk_gb_used += disk_capacity.get_capacity(instance)
|
disk_gb_used += disk_capacity.get_capacity(instance)
|
||||||
@ -146,9 +144,7 @@ class OutletTempControl(base.ThermalOptimizationBaseStrategy):
|
|||||||
|
|
||||||
hosts_need_release = []
|
hosts_need_release = []
|
||||||
hosts_target = []
|
hosts_target = []
|
||||||
for node_id in nodes:
|
for node in nodes.values():
|
||||||
node = self.compute_model.get_node_by_uuid(
|
|
||||||
node_id)
|
|
||||||
resource_id = node.uuid
|
resource_id = node.uuid
|
||||||
|
|
||||||
outlet_temp = self.ceilometer.statistic_aggregation(
|
outlet_temp = self.ceilometer.statistic_aggregation(
|
||||||
@ -174,14 +170,11 @@ class OutletTempControl(base.ThermalOptimizationBaseStrategy):
|
|||||||
"""Pick up an active instance to migrate from provided hosts"""
|
"""Pick up an active instance to migrate from provided hosts"""
|
||||||
for instance_data in hosts:
|
for instance_data in hosts:
|
||||||
mig_source_node = instance_data['node']
|
mig_source_node = instance_data['node']
|
||||||
instances_of_src = self.compute_model.mapping.get_node_instances(
|
instances_of_src = self.compute_model.get_node_instances(
|
||||||
mig_source_node)
|
mig_source_node)
|
||||||
if len(instances_of_src) > 0:
|
for instance in instances_of_src:
|
||||||
for instance_id in instances_of_src:
|
|
||||||
try:
|
try:
|
||||||
# select the first active instance to migrate
|
# select the first active instance to migrate
|
||||||
instance = self.compute_model.get_instance_by_uuid(
|
|
||||||
instance_id)
|
|
||||||
if (instance.state !=
|
if (instance.state !=
|
||||||
element.InstanceState.ACTIVE.value):
|
element.InstanceState.ACTIVE.value):
|
||||||
LOG.info(_LI("Instance not active, skipped: %s"),
|
LOG.info(_LI("Instance not active, skipped: %s"),
|
||||||
@ -197,7 +190,7 @@ class OutletTempControl(base.ThermalOptimizationBaseStrategy):
|
|||||||
def filter_dest_servers(self, hosts, instance_to_migrate):
|
def filter_dest_servers(self, hosts, instance_to_migrate):
|
||||||
"""Only return hosts with sufficient available resources"""
|
"""Only return hosts with sufficient available resources"""
|
||||||
cpu_capacity = self.compute_model.get_resource_by_uuid(
|
cpu_capacity = self.compute_model.get_resource_by_uuid(
|
||||||
element.ResourceType.cpu_cores)
|
element.ResourceType.vcpus)
|
||||||
disk_capacity = self.compute_model.get_resource_by_uuid(
|
disk_capacity = self.compute_model.get_resource_by_uuid(
|
||||||
element.ResourceType.disk)
|
element.ResourceType.disk)
|
||||||
memory_capacity = self.compute_model.get_resource_by_uuid(
|
memory_capacity = self.compute_model.get_resource_by_uuid(
|
||||||
|
@ -45,7 +45,7 @@ airflow is higher than the specified threshold.
|
|||||||
|
|
||||||
from oslo_log import log
|
from oslo_log import log
|
||||||
|
|
||||||
from watcher._i18n import _, _LE, _LI, _LW
|
from watcher._i18n import _, _LI, _LW
|
||||||
from watcher.common import exception as wexc
|
from watcher.common import exception as wexc
|
||||||
from watcher.datasource import ceilometer as ceil
|
from watcher.datasource import ceilometer as ceil
|
||||||
from watcher.decision_engine.model import element
|
from watcher.decision_engine.model import element
|
||||||
@ -166,13 +166,11 @@ class UniformAirflow(base.BaseStrategy):
|
|||||||
|
|
||||||
def calculate_used_resource(self, node, cap_cores, cap_mem, cap_disk):
|
def calculate_used_resource(self, node, cap_cores, cap_mem, cap_disk):
|
||||||
"""Compute the used vcpus, memory and disk based on instance flavors"""
|
"""Compute the used vcpus, memory and disk based on instance flavors"""
|
||||||
instances = self.compute_model.mapping.get_node_instances(node)
|
instances = self.compute_model.get_node_instances(node)
|
||||||
vcpus_used = 0
|
vcpus_used = 0
|
||||||
memory_mb_used = 0
|
memory_mb_used = 0
|
||||||
disk_gb_used = 0
|
disk_gb_used = 0
|
||||||
for instance_id in instances:
|
for instance in instances:
|
||||||
instance = self.compute_model.get_instance_by_uuid(
|
|
||||||
instance_id)
|
|
||||||
vcpus_used += cap_cores.get_capacity(instance)
|
vcpus_used += cap_cores.get_capacity(instance)
|
||||||
memory_mb_used += cap_mem.get_capacity(instance)
|
memory_mb_used += cap_mem.get_capacity(instance)
|
||||||
disk_gb_used += cap_disk.get_capacity(instance)
|
disk_gb_used += cap_disk.get_capacity(instance)
|
||||||
@ -187,7 +185,7 @@ class UniformAirflow(base.BaseStrategy):
|
|||||||
instances_tobe_migrate = []
|
instances_tobe_migrate = []
|
||||||
for nodemap in hosts:
|
for nodemap in hosts:
|
||||||
source_node = nodemap['node']
|
source_node = nodemap['node']
|
||||||
source_instances = self.compute_model.mapping.get_node_instances(
|
source_instances = self.compute_model.get_node_instances(
|
||||||
source_node)
|
source_node)
|
||||||
if source_instances:
|
if source_instances:
|
||||||
inlet_t = self.ceilometer.statistic_aggregation(
|
inlet_t = self.ceilometer.statistic_aggregation(
|
||||||
@ -203,21 +201,12 @@ class UniformAirflow(base.BaseStrategy):
|
|||||||
if (power < self.threshold_power and
|
if (power < self.threshold_power and
|
||||||
inlet_t < self.threshold_inlet_t):
|
inlet_t < self.threshold_inlet_t):
|
||||||
# hardware issue, migrate all instances from this node
|
# hardware issue, migrate all instances from this node
|
||||||
for instance_id in source_instances:
|
for instance in source_instances:
|
||||||
try:
|
|
||||||
instance = (self.compute_model.
|
|
||||||
get_instance_by_uuid(instance_id))
|
|
||||||
instances_tobe_migrate.append(instance)
|
instances_tobe_migrate.append(instance)
|
||||||
except wexc.InstanceNotFound:
|
|
||||||
LOG.error(_LE("Instance not found; error: %s"),
|
|
||||||
instance_id)
|
|
||||||
return source_node, instances_tobe_migrate
|
return source_node, instances_tobe_migrate
|
||||||
else:
|
else:
|
||||||
# migrate the first active instance
|
# migrate the first active instance
|
||||||
for instance_id in source_instances:
|
for instance in source_instances:
|
||||||
try:
|
|
||||||
instance = (self.compute_model.
|
|
||||||
get_instance_by_uuid(instance_id))
|
|
||||||
if (instance.state !=
|
if (instance.state !=
|
||||||
element.InstanceState.ACTIVE.value):
|
element.InstanceState.ACTIVE.value):
|
||||||
LOG.info(
|
LOG.info(
|
||||||
@ -226,9 +215,6 @@ class UniformAirflow(base.BaseStrategy):
|
|||||||
continue
|
continue
|
||||||
instances_tobe_migrate.append(instance)
|
instances_tobe_migrate.append(instance)
|
||||||
return source_node, instances_tobe_migrate
|
return source_node, instances_tobe_migrate
|
||||||
except wexc.InstanceNotFound:
|
|
||||||
LOG.error(_LE("Instance not found; error: %s"),
|
|
||||||
instance_id)
|
|
||||||
else:
|
else:
|
||||||
LOG.info(_LI("Instance not found on node: %s"),
|
LOG.info(_LI("Instance not found on node: %s"),
|
||||||
source_node.uuid)
|
source_node.uuid)
|
||||||
|
@ -139,18 +139,16 @@ class VMWorkloadConsolidation(base.ServerConsolidationBaseStrategy):
|
|||||||
input_parameters=params)
|
input_parameters=params)
|
||||||
self.number_of_released_nodes += 1
|
self.number_of_released_nodes += 1
|
||||||
|
|
||||||
def add_migration(self, instance_uuid, source_node,
|
def add_migration(self, instance, source_node,
|
||||||
destination_node, model):
|
destination_node, model):
|
||||||
"""Add an action for VM migration into the solution.
|
"""Add an action for VM migration into the solution.
|
||||||
|
|
||||||
:param instance_uuid: instance uuid
|
:param instance: instance object
|
||||||
:param source_node: node object
|
:param source_node: node object
|
||||||
:param destination_node: node object
|
:param destination_node: node object
|
||||||
:param model: model_root object
|
:param model: model_root object
|
||||||
:return: None
|
:return: None
|
||||||
"""
|
"""
|
||||||
instance = model.get_instance_by_uuid(instance_uuid)
|
|
||||||
|
|
||||||
instance_state_str = self.get_state_str(instance.state)
|
instance_state_str = self.get_state_str(instance.state)
|
||||||
if instance_state_str != element.InstanceState.ACTIVE.value:
|
if instance_state_str != element.InstanceState.ACTIVE.value:
|
||||||
# Watcher curently only supports live VM migration and block live
|
# Watcher curently only supports live VM migration and block live
|
||||||
@ -160,7 +158,7 @@ class VMWorkloadConsolidation(base.ServerConsolidationBaseStrategy):
|
|||||||
LOG.error(
|
LOG.error(
|
||||||
_LE('Cannot live migrate: instance_uuid=%(instance_uuid)s, '
|
_LE('Cannot live migrate: instance_uuid=%(instance_uuid)s, '
|
||||||
'state=%(instance_state)s.') % dict(
|
'state=%(instance_state)s.') % dict(
|
||||||
instance_uuid=instance_uuid,
|
instance_uuid=instance.uuid,
|
||||||
instance_state=instance_state_str))
|
instance_state=instance_state_str))
|
||||||
return
|
return
|
||||||
|
|
||||||
@ -169,6 +167,7 @@ class VMWorkloadConsolidation(base.ServerConsolidationBaseStrategy):
|
|||||||
destination_node_state_str = self.get_state_str(destination_node.state)
|
destination_node_state_str = self.get_state_str(destination_node.state)
|
||||||
if destination_node_state_str == element.ServiceState.DISABLED.value:
|
if destination_node_state_str == element.ServiceState.DISABLED.value:
|
||||||
self.add_action_enable_compute_node(destination_node)
|
self.add_action_enable_compute_node(destination_node)
|
||||||
|
|
||||||
if model.migrate_instance(instance, source_node, destination_node):
|
if model.migrate_instance(instance, source_node, destination_node):
|
||||||
params = {'migration_type': migration_type,
|
params = {'migration_type': migration_type,
|
||||||
'source_node': source_node.uuid,
|
'source_node': source_node.uuid,
|
||||||
@ -185,7 +184,7 @@ class VMWorkloadConsolidation(base.ServerConsolidationBaseStrategy):
|
|||||||
:return: None
|
:return: None
|
||||||
"""
|
"""
|
||||||
for node in model.get_all_compute_nodes().values():
|
for node in model.get_all_compute_nodes().values():
|
||||||
if (len(model.mapping.get_node_instances(node)) == 0 and
|
if (len(model.get_node_instances(node)) == 0 and
|
||||||
node.status !=
|
node.status !=
|
||||||
element.ServiceState.DISABLED.value):
|
element.ServiceState.DISABLED.value):
|
||||||
self.add_action_disable_node(node)
|
self.add_action_disable_node(node)
|
||||||
@ -254,14 +253,13 @@ class VMWorkloadConsolidation(base.ServerConsolidationBaseStrategy):
|
|||||||
:param aggr: string
|
:param aggr: string
|
||||||
:return: dict(cpu(number of cores used), ram(MB used), disk(B used))
|
:return: dict(cpu(number of cores used), ram(MB used), disk(B used))
|
||||||
"""
|
"""
|
||||||
node_instances = model.mapping.get_node_instances_by_uuid(
|
node_instances = model.get_node_instances(node)
|
||||||
node.uuid)
|
|
||||||
node_ram_util = 0
|
node_ram_util = 0
|
||||||
node_disk_util = 0
|
node_disk_util = 0
|
||||||
node_cpu_util = 0
|
node_cpu_util = 0
|
||||||
for instance_uuid in node_instances:
|
for instance in node_instances:
|
||||||
instance_util = self.get_instance_utilization(
|
instance_util = self.get_instance_utilization(
|
||||||
instance_uuid, model, period, aggr)
|
instance.uuid, model, period, aggr)
|
||||||
node_cpu_util += instance_util['cpu']
|
node_cpu_util += instance_util['cpu']
|
||||||
node_ram_util += instance_util['ram']
|
node_ram_util += instance_util['ram']
|
||||||
node_disk_util += instance_util['disk']
|
node_disk_util += instance_util['disk']
|
||||||
@ -402,9 +400,9 @@ class VMWorkloadConsolidation(base.ServerConsolidationBaseStrategy):
|
|||||||
self.number_of_migrations -= 1
|
self.number_of_migrations -= 1
|
||||||
src_node = model.get_node_by_uuid(src_uuid)
|
src_node = model.get_node_by_uuid(src_uuid)
|
||||||
dst_node = model.get_node_by_uuid(dst_uuid)
|
dst_node = model.get_node_by_uuid(dst_uuid)
|
||||||
if model.migrate_instance(instance_uuid, dst_node, src_node):
|
instance = model.get_instance_by_uuid(instance_uuid)
|
||||||
self.add_migration(
|
if model.migrate_instance(instance, dst_node, src_node):
|
||||||
instance_uuid, src_node, dst_node, model)
|
self.add_migration(instance, src_node, dst_node, model)
|
||||||
|
|
||||||
def offload_phase(self, model, cc):
|
def offload_phase(self, model, cc):
|
||||||
"""Perform offloading phase.
|
"""Perform offloading phase.
|
||||||
@ -431,13 +429,13 @@ class VMWorkloadConsolidation(base.ServerConsolidationBaseStrategy):
|
|||||||
for node in reversed(sorted_nodes):
|
for node in reversed(sorted_nodes):
|
||||||
if self.is_overloaded(node, model, cc):
|
if self.is_overloaded(node, model, cc):
|
||||||
for instance in sorted(
|
for instance in sorted(
|
||||||
model.mapping.get_node_instances(node),
|
model.get_node_instances(node),
|
||||||
key=lambda x: self.get_instance_utilization(
|
key=lambda x: self.get_instance_utilization(
|
||||||
x, model)['cpu']
|
x.uuid, model)['cpu']
|
||||||
):
|
):
|
||||||
for destination_node in reversed(sorted_nodes):
|
for destination_node in reversed(sorted_nodes):
|
||||||
if self.instance_fits(
|
if self.instance_fits(
|
||||||
instance, destination_node, model, cc):
|
instance.uuid, destination_node, model, cc):
|
||||||
self.add_migration(instance, node,
|
self.add_migration(instance, node,
|
||||||
destination_node, model)
|
destination_node, model)
|
||||||
break
|
break
|
||||||
@ -465,15 +463,16 @@ class VMWorkloadConsolidation(base.ServerConsolidationBaseStrategy):
|
|||||||
asc = 0
|
asc = 0
|
||||||
for node in sorted_nodes:
|
for node in sorted_nodes:
|
||||||
instances = sorted(
|
instances = sorted(
|
||||||
model.mapping.get_node_instances(node),
|
model.get_node_instances(node),
|
||||||
key=lambda x: self.get_instance_utilization(x, model)['cpu'])
|
key=lambda x: self.get_instance_utilization(
|
||||||
|
x.uuid, model)['cpu'])
|
||||||
for instance in reversed(instances):
|
for instance in reversed(instances):
|
||||||
dsc = len(sorted_nodes) - 1
|
dsc = len(sorted_nodes) - 1
|
||||||
for destination_node in reversed(sorted_nodes):
|
for destination_node in reversed(sorted_nodes):
|
||||||
if asc >= dsc:
|
if asc >= dsc:
|
||||||
break
|
break
|
||||||
if self.instance_fits(
|
if self.instance_fits(
|
||||||
instance, destination_node, model, cc):
|
instance.uuid, destination_node, model, cc):
|
||||||
self.add_migration(instance, node,
|
self.add_migration(instance, node,
|
||||||
destination_node, model)
|
destination_node, model)
|
||||||
break
|
break
|
||||||
|
@ -148,12 +148,11 @@ class WorkloadBalance(base.WorkloadStabilizationBaseStrategy):
|
|||||||
def calculate_used_resource(self, node, cap_cores, cap_mem,
|
def calculate_used_resource(self, node, cap_cores, cap_mem,
|
||||||
cap_disk):
|
cap_disk):
|
||||||
"""Calculate the used vcpus, memory and disk based on VM flavors"""
|
"""Calculate the used vcpus, memory and disk based on VM flavors"""
|
||||||
instances = self.compute_model.mapping.get_node_instances(node)
|
instances = self.compute_model.get_node_instances(node)
|
||||||
vcpus_used = 0
|
vcpus_used = 0
|
||||||
memory_mb_used = 0
|
memory_mb_used = 0
|
||||||
disk_gb_used = 0
|
disk_gb_used = 0
|
||||||
for instance_id in instances:
|
for instance in instances:
|
||||||
instance = self.compute_model.get_instance_by_uuid(instance_id)
|
|
||||||
vcpus_used += cap_cores.get_capacity(instance)
|
vcpus_used += cap_cores.get_capacity(instance)
|
||||||
memory_mb_used += cap_mem.get_capacity(instance)
|
memory_mb_used += cap_mem.get_capacity(instance)
|
||||||
disk_gb_used += cap_disk.get_capacity(instance)
|
disk_gb_used += cap_disk.get_capacity(instance)
|
||||||
@ -169,27 +168,25 @@ class WorkloadBalance(base.WorkloadStabilizationBaseStrategy):
|
|||||||
"""
|
"""
|
||||||
for instance_data in hosts:
|
for instance_data in hosts:
|
||||||
source_node = instance_data['node']
|
source_node = instance_data['node']
|
||||||
source_instances = self.compute_model.mapping.get_node_instances(
|
source_instances = self.compute_model.get_node_instances(
|
||||||
source_node)
|
source_node)
|
||||||
if source_instances:
|
if source_instances:
|
||||||
delta_workload = instance_data['workload'] - avg_workload
|
delta_workload = instance_data['workload'] - avg_workload
|
||||||
min_delta = 1000000
|
min_delta = 1000000
|
||||||
instance_id = None
|
instance_id = None
|
||||||
for inst_id in source_instances:
|
for instance in source_instances:
|
||||||
try:
|
try:
|
||||||
# select the first active VM to migrate
|
# select the first active VM to migrate
|
||||||
instance = self.compute_model.get_instance_by_uuid(
|
|
||||||
inst_id)
|
|
||||||
if (instance.state !=
|
if (instance.state !=
|
||||||
element.InstanceState.ACTIVE.value):
|
element.InstanceState.ACTIVE.value):
|
||||||
LOG.debug("Instance not active, skipped: %s",
|
LOG.debug("Instance not active, skipped: %s",
|
||||||
instance.uuid)
|
instance.uuid)
|
||||||
continue
|
continue
|
||||||
current_delta = (
|
current_delta = (
|
||||||
delta_workload - workload_cache[inst_id])
|
delta_workload - workload_cache[instance.uuid])
|
||||||
if 0 <= current_delta < min_delta:
|
if 0 <= current_delta < min_delta:
|
||||||
min_delta = current_delta
|
min_delta = current_delta
|
||||||
instance_id = inst_id
|
instance_id = instance.uuid
|
||||||
except wexc.InstanceNotFound:
|
except wexc.InstanceNotFound:
|
||||||
LOG.error(_LE("Instance not found; error: %s"),
|
LOG.error(_LE("Instance not found; error: %s"),
|
||||||
instance_id)
|
instance_id)
|
||||||
@ -254,7 +251,7 @@ class WorkloadBalance(base.WorkloadStabilizationBaseStrategy):
|
|||||||
raise wexc.ClusterEmpty()
|
raise wexc.ClusterEmpty()
|
||||||
# get cpu cores capacity of nodes and instances
|
# get cpu cores capacity of nodes and instances
|
||||||
cap_cores = self.compute_model.get_resource_by_uuid(
|
cap_cores = self.compute_model.get_resource_by_uuid(
|
||||||
element.ResourceType.cpu_cores)
|
element.ResourceType.vcpus)
|
||||||
overload_hosts = []
|
overload_hosts = []
|
||||||
nonoverload_hosts = []
|
nonoverload_hosts = []
|
||||||
# total workload of cluster
|
# total workload of cluster
|
||||||
@ -264,13 +261,12 @@ class WorkloadBalance(base.WorkloadStabilizationBaseStrategy):
|
|||||||
for node_id in nodes:
|
for node_id in nodes:
|
||||||
node = self.compute_model.get_node_by_uuid(
|
node = self.compute_model.get_node_by_uuid(
|
||||||
node_id)
|
node_id)
|
||||||
instances = self.compute_model.mapping.get_node_instances(node)
|
instances = self.compute_model.get_node_instances(node)
|
||||||
node_workload = 0.0
|
node_workload = 0.0
|
||||||
for instance_id in instances:
|
for instance in instances:
|
||||||
instance = self.compute_model.get_instance_by_uuid(instance_id)
|
|
||||||
try:
|
try:
|
||||||
cpu_util = self.ceilometer.statistic_aggregation(
|
cpu_util = self.ceilometer.statistic_aggregation(
|
||||||
resource_id=instance_id,
|
resource_id=instance.uuid,
|
||||||
meter_name=self._meter,
|
meter_name=self._meter,
|
||||||
period=self._period,
|
period=self._period,
|
||||||
aggregate='avg')
|
aggregate='avg')
|
||||||
@ -279,12 +275,12 @@ class WorkloadBalance(base.WorkloadStabilizationBaseStrategy):
|
|||||||
LOG.error(_LE("Can not get cpu_util from Ceilometer"))
|
LOG.error(_LE("Can not get cpu_util from Ceilometer"))
|
||||||
continue
|
continue
|
||||||
if cpu_util is None:
|
if cpu_util is None:
|
||||||
LOG.debug("Instance (%s): cpu_util is None", instance_id)
|
LOG.debug("Instance (%s): cpu_util is None", instance.uuid)
|
||||||
continue
|
continue
|
||||||
instance_cores = cap_cores.get_capacity(instance)
|
instance_cores = cap_cores.get_capacity(instance)
|
||||||
workload_cache[instance_id] = cpu_util * instance_cores / 100
|
workload_cache[instance.uuid] = cpu_util * instance_cores / 100
|
||||||
node_workload += workload_cache[instance_id]
|
node_workload += workload_cache[instance.uuid]
|
||||||
LOG.debug("VM (%s): cpu_util %f", instance_id, cpu_util)
|
LOG.debug("VM (%s): cpu_util %f", instance.uuid, cpu_util)
|
||||||
node_cores = cap_cores.get_capacity(node)
|
node_cores = cap_cores.get_capacity(node)
|
||||||
hy_cpu_util = node_workload / node_cores * 100
|
hy_cpu_util = node_workload / node_cores * 100
|
||||||
|
|
||||||
|
@ -343,29 +343,26 @@ class WorkloadStabilization(base.WorkloadStabilizationBaseStrategy):
|
|||||||
|
|
||||||
instance_host_map = []
|
instance_host_map = []
|
||||||
nodes = list(self.get_available_nodes())
|
nodes = list(self.get_available_nodes())
|
||||||
for source_hp_id in nodes:
|
for src_node_id in nodes:
|
||||||
|
src_node = self.compute_model.get_node_by_uuid(src_node_id)
|
||||||
c_nodes = copy.copy(nodes)
|
c_nodes = copy.copy(nodes)
|
||||||
c_nodes.remove(source_hp_id)
|
c_nodes.remove(src_node_id)
|
||||||
node_list = yield_nodes(c_nodes)
|
node_list = yield_nodes(c_nodes)
|
||||||
instances_id = self.compute_model.get_mapping(). \
|
for instance in self.compute_model.get_node_instances(src_node):
|
||||||
get_node_instances_by_uuid(source_hp_id)
|
|
||||||
for instance_id in instances_id:
|
|
||||||
min_sd_case = {'value': len(self.metrics)}
|
min_sd_case = {'value': len(self.metrics)}
|
||||||
instance = self.compute_model.get_instance_by_uuid(instance_id)
|
|
||||||
if instance.state not in [element.InstanceState.ACTIVE.value,
|
if instance.state not in [element.InstanceState.ACTIVE.value,
|
||||||
element.InstanceState.PAUSED.value]:
|
element.InstanceState.PAUSED.value]:
|
||||||
continue
|
continue
|
||||||
for dst_node_id in next(node_list):
|
for dst_node_id in next(node_list):
|
||||||
sd_case = self.calculate_migration_case(hosts, instance_id,
|
sd_case = self.calculate_migration_case(
|
||||||
source_hp_id,
|
hosts, instance.uuid, src_node_id, dst_node_id)
|
||||||
dst_node_id)
|
|
||||||
|
|
||||||
weighted_sd = self.calculate_weighted_sd(sd_case[:-1])
|
weighted_sd = self.calculate_weighted_sd(sd_case[:-1])
|
||||||
|
|
||||||
if weighted_sd < min_sd_case['value']:
|
if weighted_sd < min_sd_case['value']:
|
||||||
min_sd_case = {
|
min_sd_case = {
|
||||||
'host': dst_node_id, 'value': weighted_sd,
|
'host': dst_node_id, 'value': weighted_sd,
|
||||||
's_host': source_hp_id, 'instance': instance_id}
|
's_host': src_node_id, 'instance': instance.uuid}
|
||||||
instance_host_map.append(min_sd_case)
|
instance_host_map.append(min_sd_case)
|
||||||
return sorted(instance_host_map, key=lambda x: x['value'])
|
return sorted(instance_host_map, key=lambda x: x['value'])
|
||||||
|
|
||||||
|
@ -24,10 +24,14 @@ BaseEnumField = fields.BaseEnumField
|
|||||||
BooleanField = fields.BooleanField
|
BooleanField = fields.BooleanField
|
||||||
DateTimeField = fields.DateTimeField
|
DateTimeField = fields.DateTimeField
|
||||||
Enum = fields.Enum
|
Enum = fields.Enum
|
||||||
|
FloatField = fields.FloatField
|
||||||
IntegerField = fields.IntegerField
|
IntegerField = fields.IntegerField
|
||||||
ListOfStringsField = fields.ListOfStringsField
|
ListOfStringsField = fields.ListOfStringsField
|
||||||
|
NonNegativeFloatField = fields.NonNegativeFloatField
|
||||||
|
NonNegativeIntegerField = fields.NonNegativeIntegerField
|
||||||
ObjectField = fields.ObjectField
|
ObjectField = fields.ObjectField
|
||||||
StringField = fields.StringField
|
StringField = fields.StringField
|
||||||
|
UnspecifiedDefault = fields.UnspecifiedDefault
|
||||||
UUIDField = fields.UUIDField
|
UUIDField = fields.UUIDField
|
||||||
|
|
||||||
|
|
||||||
|
@ -19,6 +19,7 @@
|
|||||||
import mock
|
import mock
|
||||||
|
|
||||||
from watcher.common import nova_helper
|
from watcher.common import nova_helper
|
||||||
|
from watcher.common import utils
|
||||||
from watcher.decision_engine.model.collector import nova
|
from watcher.decision_engine.model.collector import nova
|
||||||
from watcher.tests import base
|
from watcher.tests import base
|
||||||
from watcher.tests import conf_fixture
|
from watcher.tests import conf_fixture
|
||||||
@ -33,9 +34,13 @@ class TestNovaClusterDataModelCollector(base.TestCase):
|
|||||||
@mock.patch('keystoneclient.v3.client.Client', mock.Mock())
|
@mock.patch('keystoneclient.v3.client.Client', mock.Mock())
|
||||||
@mock.patch.object(nova_helper, 'NovaHelper')
|
@mock.patch.object(nova_helper, 'NovaHelper')
|
||||||
def test_nova_cdmc_execute(self, m_nova_helper_cls):
|
def test_nova_cdmc_execute(self, m_nova_helper_cls):
|
||||||
m_nova_helper = mock.Mock()
|
m_nova_helper = mock.Mock(name="nova_helper")
|
||||||
m_nova_helper_cls.return_value = m_nova_helper
|
m_nova_helper_cls.return_value = m_nova_helper
|
||||||
|
m_nova_helper.get_service.return_value = mock.Mock(
|
||||||
|
host="test_hostname")
|
||||||
|
|
||||||
fake_compute_node = mock.Mock(
|
fake_compute_node = mock.Mock(
|
||||||
|
id=1337,
|
||||||
service={'id': 123},
|
service={'id': 123},
|
||||||
hypervisor_hostname='test_hostname',
|
hypervisor_hostname='test_hostname',
|
||||||
memory_mb=333,
|
memory_mb=333,
|
||||||
@ -47,19 +52,16 @@ class TestNovaClusterDataModelCollector(base.TestCase):
|
|||||||
)
|
)
|
||||||
fake_instance = mock.Mock(
|
fake_instance = mock.Mock(
|
||||||
id='ef500f7e-dac8-470f-960c-169486fce71b',
|
id='ef500f7e-dac8-470f-960c-169486fce71b',
|
||||||
state=mock.Mock(**{'OS-EXT-STS:vm_state': 'VM_STATE'}),
|
human_id='fake_instance',
|
||||||
flavor={'ram': 333, 'disk': 222, 'vcpus': 4},
|
flavor={'ram': 333, 'disk': 222, 'vcpus': 4, 'id': 1},
|
||||||
)
|
)
|
||||||
|
setattr(fake_instance, 'OS-EXT-STS:vm_state', 'VM_STATE')
|
||||||
m_nova_helper.get_compute_node_list.return_value = [fake_compute_node]
|
m_nova_helper.get_compute_node_list.return_value = [fake_compute_node]
|
||||||
m_nova_helper.get_instances_by_node.return_value = [fake_instance]
|
# m_nova_helper.get_instances_by_node.return_value = [fake_instance]
|
||||||
m_nova_helper.nova.services.find.return_value = mock.Mock(
|
m_nova_helper.get_instance_list.return_value = [fake_instance]
|
||||||
host='test_hostname')
|
|
||||||
|
|
||||||
def m_get_flavor_instance(instance, cache):
|
m_nova_helper.get_flavor.return_value = utils.Struct(**{
|
||||||
instance.flavor = {'ram': 333, 'disk': 222, 'vcpus': 4}
|
'ram': 333, 'disk': 222, 'vcpus': 4})
|
||||||
return instance
|
|
||||||
|
|
||||||
m_nova_helper.get_flavor_instance.side_effect = m_get_flavor_instance
|
|
||||||
|
|
||||||
m_config = mock.Mock()
|
m_config = mock.Mock()
|
||||||
m_osc = mock.Mock()
|
m_osc = mock.Mock()
|
||||||
|
@ -1,47 +1,47 @@
|
|||||||
<ModelRoot>
|
<ModelRoot>
|
||||||
<ComputeNode human_id="" uuid="Node_0" status="enabled" state="up" id="0" hostname="hostname_0" ResourceType.cpu_cores="40" ResourceType.disk="250" ResourceType.disk_capacity="250" ResourceType.memory="132">
|
<ComputeNode human_id="" uuid="Node_0" status="enabled" state="up" id="0" hostname="hostname_0" vcpus="40" disk="250" disk_capacity="250" memory="132">
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_0" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_0" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_1" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_1" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
</ComputeNode>
|
</ComputeNode>
|
||||||
<ComputeNode human_id="" uuid="Node_1" status="enabled" state="up" id="1" hostname="hostname_1" ResourceType.cpu_cores="40" ResourceType.disk="250" ResourceType.disk_capacity="250" ResourceType.memory="132">
|
<ComputeNode human_id="" uuid="Node_1" status="enabled" state="up" id="1" hostname="hostname_1" vcpus="40" disk="250" disk_capacity="250" memory="132">
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_2" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_2" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
</ComputeNode>
|
</ComputeNode>
|
||||||
<ComputeNode human_id="" uuid="Node_2" status="enabled" state="up" id="2" hostname="hostname_2" ResourceType.cpu_cores="40" ResourceType.disk="250" ResourceType.disk_capacity="250" ResourceType.memory="132">
|
<ComputeNode human_id="" uuid="Node_2" status="enabled" state="up" id="2" hostname="hostname_2" vcpus="40" disk="250" disk_capacity="250" memory="132">
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_3" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_3" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_4" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_4" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_5" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_5" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
</ComputeNode>
|
</ComputeNode>
|
||||||
<ComputeNode human_id="" uuid="Node_3" status="enabled" state="up" id="3" hostname="hostname_3" ResourceType.cpu_cores="40" ResourceType.disk="250" ResourceType.disk_capacity="250" ResourceType.memory="132">
|
<ComputeNode human_id="" uuid="Node_3" status="enabled" state="up" id="3" hostname="hostname_3" vcpus="40" disk="250" disk_capacity="250" memory="132">
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_6" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_6" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
</ComputeNode>
|
</ComputeNode>
|
||||||
<ComputeNode human_id="" uuid="Node_4" status="enabled" state="up" id="4" hostname="hostname_4" ResourceType.cpu_cores="40" ResourceType.disk="250" ResourceType.disk_capacity="250" ResourceType.memory="132">
|
<ComputeNode human_id="" uuid="Node_4" status="enabled" state="up" id="4" hostname="hostname_4" vcpus="40" disk="250" disk_capacity="250" memory="132">
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_7" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_7" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
</ComputeNode>
|
</ComputeNode>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_10" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_10" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_11" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_11" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_12" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_12" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_13" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_13" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_14" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_14" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_15" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_15" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_16" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_16" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_17" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_17" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_18" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_18" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_19" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_19" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_20" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_20" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_21" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_21" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_22" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_22" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_23" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_23" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_24" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_24" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_25" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_25" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_26" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_26" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_27" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_27" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_28" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_28" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_29" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_29" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_30" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_30" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_31" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_31" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_32" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_32" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_33" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_33" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_34" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_34" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_8" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_8" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_9" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_9" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
</ModelRoot>
|
</ModelRoot>
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
<ModelRoot>
|
<ModelRoot>
|
||||||
<ComputeNode hostname="hostname_0" uuid="Node_0" id="0" state="enabled" human_id="" status="enabled" ResourceType.cpu_cores="40" ResourceType.disk="250" ResourceType.disk_capacity="250" ResourceType.memory="64">
|
<ComputeNode hostname="hostname_0" uuid="Node_0" id="0" state="enabled" human_id="" status="enabled" vcpus="40" disk="250" disk_capacity="250" memory="64">
|
||||||
<Instance hostname="" human_id="" state="active" uuid="INSTANCE_0" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance human_id="" state="active" uuid="INSTANCE_0" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
</ComputeNode>
|
</ComputeNode>
|
||||||
<ComputeNode hostname="hostname_1" uuid="Node_1" id="1" state="enabled" human_id="" status="enabled" ResourceType.cpu_cores="40" ResourceType.disk="250" ResourceType.disk_capacity="250" ResourceType.memory="64">
|
<ComputeNode hostname="hostname_1" uuid="Node_1" id="1" state="enabled" human_id="" status="enabled" vcpus="40" disk="250" disk_capacity="250" memory="64">
|
||||||
<Instance hostname="" human_id="" state="active" uuid="INSTANCE_1" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance human_id="" state="active" uuid="INSTANCE_1" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
</ComputeNode>
|
</ComputeNode>
|
||||||
</ModelRoot>
|
</ModelRoot>
|
||||||
|
@ -1,13 +1,13 @@
|
|||||||
<ModelRoot>
|
<ModelRoot>
|
||||||
<ComputeNode hostname="hostname_0" uuid="Node_0" id="0" state="up" human_id="" status="enabled" ResourceType.cpu_cores="16" ResourceType.disk="250" ResourceType.disk_capacity="250" ResourceType.memory="64">
|
<ComputeNode hostname="hostname_0" uuid="Node_0" id="0" state="up" human_id="" status="enabled" vcpus="16" disk="250" disk_capacity="250" memory="64">
|
||||||
<Instance hostname="" human_id="" state="active" uuid="INSTANCE_0" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance human_id="" state="active" uuid="INSTANCE_0" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance hostname="" human_id="" state="active" uuid="INSTANCE_1" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance human_id="" state="active" uuid="INSTANCE_1" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance hostname="" human_id="" state="active" uuid="INSTANCE_2" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance human_id="" state="active" uuid="INSTANCE_2" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance hostname="" human_id="" state="active" uuid="INSTANCE_3" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance human_id="" state="active" uuid="INSTANCE_3" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance hostname="" human_id="" state="active" uuid="INSTANCE_4" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance human_id="" state="active" uuid="INSTANCE_4" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance hostname="" human_id="" state="active" uuid="INSTANCE_5" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance human_id="" state="active" uuid="INSTANCE_5" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
</ComputeNode>
|
</ComputeNode>
|
||||||
<ComputeNode hostname="hostname_1" uuid="Node_1" id="1" state="up" human_id="" status="enabled" ResourceType.cpu_cores="16" ResourceType.disk="250" ResourceType.disk_capacity="250" ResourceType.memory="64"/>
|
<ComputeNode hostname="hostname_1" uuid="Node_1" id="1" state="up" human_id="" status="enabled" vcpus="16" disk="250" disk_capacity="250" memory="64"/>
|
||||||
<ComputeNode hostname="hostname_2" uuid="Node_2" id="2" state="up" human_id="" status="enabled" ResourceType.cpu_cores="16" ResourceType.disk="250" ResourceType.disk_capacity="250" ResourceType.memory="64"/>
|
<ComputeNode hostname="hostname_2" uuid="Node_2" id="2" state="up" human_id="" status="enabled" vcpus="16" disk="250" disk_capacity="250" memory="64"/>
|
||||||
<ComputeNode hostname="hostname_3" uuid="Node_3" id="3" state="up" human_id="" status="enabled" ResourceType.cpu_cores="16" ResourceType.disk="250" ResourceType.disk_capacity="250" ResourceType.memory="64"/>
|
<ComputeNode hostname="hostname_3" uuid="Node_3" id="3" state="up" human_id="" status="enabled" vcpus="16" disk="250" disk_capacity="250" memory="64"/>
|
||||||
</ModelRoot>
|
</ModelRoot>
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
<ModelRoot>
|
<ModelRoot>
|
||||||
<ComputeNode human_id="" uuid="Node_0" status="enabled" state="up" id="0" hostname="hostname_0" ResourceType.cpu_cores="40" ResourceType.disk="250" ResourceType.disk_capacity="250" ResourceType.memory="132">
|
<ComputeNode human_id="" uuid="Node_0" status="enabled" state="up" id="0" hostname="hostname_0" vcpus="40" disk="250" disk_capacity="250" memory="132">
|
||||||
<Instance state="active" human_id="" uuid="73b09e16-35b7-4922-804e-e8f5d9b740fc" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="73b09e16-35b7-4922-804e-e8f5d9b740fc" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
</ComputeNode>
|
</ComputeNode>
|
||||||
<ComputeNode human_id="" uuid="Node_1" status="enabled" state="up" id="1" hostname="hostname_1" ResourceType.cpu_cores="40" ResourceType.disk="250" ResourceType.disk_capacity="250" ResourceType.memory="132">
|
<ComputeNode human_id="" uuid="Node_1" status="enabled" state="up" id="1" hostname="hostname_1" vcpus="40" disk="250" disk_capacity="250" memory="132">
|
||||||
<Instance state="active" human_id="" uuid="a4cab39b-9828-413a-bf88-f76921bf1517" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="a4cab39b-9828-413a-bf88-f76921bf1517" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
</ComputeNode>
|
</ComputeNode>
|
||||||
</ModelRoot>
|
</ModelRoot>
|
||||||
|
@ -1,9 +1,9 @@
|
|||||||
<ModelRoot>
|
<ModelRoot>
|
||||||
<ComputeNode hostname="hostname_0" uuid="Node_0" id="0" state="up" human_id="" status="enabled" ResourceType.cpu_cores="10" ResourceType.disk="250" ResourceType.disk_capacity="250" ResourceType.memory="64">
|
<ComputeNode hostname="hostname_0" uuid="Node_0" id="0" state="up" human_id="" status="enabled" vcpus="10" disk="250" disk_capacity="250" memory="64">
|
||||||
<Instance hostname="" human_id="" state="active" uuid="INSTANCE_6" ResourceType.cpu_cores="1" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance human_id="" state="active" uuid="INSTANCE_6" vcpus="1" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance hostname="" human_id="" state="active" uuid="INSTANCE_7" ResourceType.cpu_cores="2" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance human_id="" state="active" uuid="INSTANCE_7" vcpus="2" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance hostname="" human_id="" state="active" uuid="INSTANCE_8" ResourceType.cpu_cores="4" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance human_id="" state="active" uuid="INSTANCE_8" vcpus="4" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance hostname="" human_id="" state="active" uuid="INSTANCE_9" ResourceType.cpu_cores="8" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance human_id="" state="active" uuid="INSTANCE_9" vcpus="8" disk="20" disk_capacity="20" memory="2"/>
|
||||||
</ComputeNode>
|
</ComputeNode>
|
||||||
<ComputeNode hostname="hostname_1" uuid="Node_1" id="1" state="up" human_id="" status="enabled" ResourceType.cpu_cores="10" ResourceType.disk="250" ResourceType.disk_capacity="250" ResourceType.memory="64"/>
|
<ComputeNode hostname="hostname_1" uuid="Node_1" id="1" state="up" human_id="" status="enabled" vcpus="10" disk="250" disk_capacity="250" memory="64"/>
|
||||||
</ModelRoot>
|
</ModelRoot>
|
||||||
|
@ -1,3 +1,3 @@
|
|||||||
<ModelRoot>
|
<ModelRoot>
|
||||||
<ComputeNode human_id="" uuid="Node_0" status="enabled" state="up" id="0" hostname="hostname_0" ResourceType.cpu_cores="1" ResourceType.disk="1" ResourceType.disk_capacity="1" ResourceType.memory="1"/>
|
<ComputeNode human_id="" uuid="Node_0" status="enabled" state="up" id="0" hostname="hostname_0" vcpus="1" disk="1" disk_capacity="1" memory="1"/>
|
||||||
</ModelRoot>
|
</ModelRoot>
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
<ModelRoot>
|
<ModelRoot>
|
||||||
<ComputeNode human_id="" uuid="Node_0" status="enabled" state="up" id="0" hostname="hostname_0" ResourceType.cpu_cores="4" ResourceType.disk="4" ResourceType.disk_capacity="4" ResourceType.memory="4">
|
<ComputeNode human_id="" uuid="Node_0" status="enabled" state="up" id="0" hostname="hostname_0" vcpus="4" disk="4" disk_capacity="4" memory="4">
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_0" hostname="" ResourceType.cpu_cores="4" ResourceType.disk="0" ResourceType.disk_capacity="0" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_0" vcpus="4" disk="0" disk_capacity="0" memory="2"/>
|
||||||
</ComputeNode>
|
</ComputeNode>
|
||||||
</ModelRoot>
|
</ModelRoot>
|
||||||
|
@ -1,10 +1,10 @@
|
|||||||
<ModelRoot>
|
<ModelRoot>
|
||||||
<ComputeNode human_id="" uuid="Node_0" status="enabled" state="up" id="0" hostname="hostname_0" ResourceType.cpu_cores="40" ResourceType.disk="250" ResourceType.disk_capacity="250" ResourceType.memory="132">
|
<ComputeNode human_id="" uuid="Node_0" status="enabled" state="up" id="0" hostname="hostname_0" vcpus="40" disk="250" disk_capacity="250" memory="132">
|
||||||
<Instance state="active" human_id="" uuid="73b09e16-35b7-4922-804e-e8f5d9b740fc" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="73b09e16-35b7-4922-804e-e8f5d9b740fc" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_1" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_1" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
</ComputeNode>
|
</ComputeNode>
|
||||||
<ComputeNode human_id="" uuid="Node_1" status="enabled" state="up" id="1" hostname="hostname_1" ResourceType.cpu_cores="40" ResourceType.disk="250" ResourceType.disk_capacity="250" ResourceType.memory="132">
|
<ComputeNode human_id="" uuid="Node_1" status="enabled" state="up" id="1" hostname="hostname_1" vcpus="40" disk="250" disk_capacity="250" memory="132">
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_3" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_3" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_4" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_4" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
</ComputeNode>
|
</ComputeNode>
|
||||||
</ModelRoot>
|
</ModelRoot>
|
||||||
|
@ -1,10 +1,10 @@
|
|||||||
<ModelRoot>
|
<ModelRoot>
|
||||||
<ComputeNode human_id="" uuid="Node_0" status="enabled" state="up" id="0" hostname="hostname_0" ResourceType.cpu_cores="50" ResourceType.disk="250" ResourceType.disk_capacity="250" ResourceType.memory="132">
|
<ComputeNode human_id="" uuid="Node_0" status="enabled" state="up" id="0" hostname="hostname_0" vcpus="50" disk="250" disk_capacity="250" memory="132">
|
||||||
<Instance state="active" human_id="" uuid="73b09e16-35b7-4922-804e-e8f5d9b740fc" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="73b09e16-35b7-4922-804e-e8f5d9b740fc" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="cae81432-1631-4d4e-b29c-6f3acdcde906" hostname="" ResourceType.cpu_cores="15" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="cae81432-1631-4d4e-b29c-6f3acdcde906" vcpus="15" disk="20" disk_capacity="20" memory="2"/>
|
||||||
</ComputeNode>
|
</ComputeNode>
|
||||||
<ComputeNode human_id="" uuid="Node_1" status="enabled" state="up" id="1" hostname="hostname_1" ResourceType.cpu_cores="50" ResourceType.disk="250" ResourceType.disk_capacity="250" ResourceType.memory="132">
|
<ComputeNode human_id="" uuid="Node_1" status="enabled" state="up" id="1" hostname="hostname_1" vcpus="50" disk="250" disk_capacity="250" memory="132">
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_3" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_3" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance state="active" human_id="" uuid="INSTANCE_4" hostname="" ResourceType.cpu_cores="10" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance state="active" human_id="" uuid="INSTANCE_4" vcpus="10" disk="20" disk_capacity="20" memory="2"/>
|
||||||
</ComputeNode>
|
</ComputeNode>
|
||||||
</ModelRoot>
|
</ModelRoot>
|
||||||
|
@ -1,16 +1,16 @@
|
|||||||
<ModelRoot>
|
<ModelRoot>
|
||||||
<ComputeNode hostname="hostname_0" uuid="Node_0" id="0" state="up" human_id="" status="enabled" ResourceType.cpu_cores="16" ResourceType.disk="250" ResourceType.disk_capacity="250" ResourceType.memory="64">
|
<ComputeNode hostname="hostname_0" uuid="Node_0" id="0" state="up" human_id="" status="enabled" vcpus="16" disk="250" disk_capacity="250" memory="64">
|
||||||
<Instance hostname="" human_id="" state="active" uuid="INSTANCE_0" ResourceType.cpu_cores="2" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance human_id="" state="active" uuid="INSTANCE_0" vcpus="2" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance hostname="" human_id="" state="active" uuid="INSTANCE_1" ResourceType.cpu_cores="2" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance human_id="" state="active" uuid="INSTANCE_1" vcpus="2" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance hostname="" human_id="" state="active" uuid="INSTANCE_2" ResourceType.cpu_cores="2" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance human_id="" state="active" uuid="INSTANCE_2" vcpus="2" disk="20" disk_capacity="20" memory="2"/>
|
||||||
</ComputeNode>
|
</ComputeNode>
|
||||||
<ComputeNode hostname="hostname_1" uuid="Node_1" id="1" state="up" human_id="" status="enabled" ResourceType.cpu_cores="16" ResourceType.disk="250" ResourceType.disk_capacity="250" ResourceType.memory="64">
|
<ComputeNode hostname="hostname_1" uuid="Node_1" id="1" state="up" human_id="" status="enabled" vcpus="16" disk="250" disk_capacity="250" memory="64">
|
||||||
<Instance hostname="" human_id="" state="active" uuid="INSTANCE_3" ResourceType.cpu_cores="2" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance human_id="" state="active" uuid="INSTANCE_3" vcpus="2" disk="20" disk_capacity="20" memory="2"/>
|
||||||
</ComputeNode>
|
</ComputeNode>
|
||||||
<ComputeNode hostname="hostname_2" uuid="Node_2" id="2" state="up" human_id="" status="enabled" ResourceType.cpu_cores="16" ResourceType.disk="250" ResourceType.disk_capacity="250" ResourceType.memory="64">
|
<ComputeNode hostname="hostname_2" uuid="Node_2" id="2" state="up" human_id="" status="enabled" vcpus="16" disk="250" disk_capacity="250" memory="64">
|
||||||
<Instance hostname="" human_id="" state="active" uuid="INSTANCE_4" ResourceType.cpu_cores="2" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance human_id="" state="active" uuid="INSTANCE_4" vcpus="2" disk="20" disk_capacity="20" memory="2"/>
|
||||||
</ComputeNode>
|
</ComputeNode>
|
||||||
<ComputeNode hostname="hostname_3" uuid="Node_3" id="3" state="up" human_id="" status="enabled" ResourceType.cpu_cores="16" ResourceType.disk="250" ResourceType.disk_capacity="250" ResourceType.memory="64">
|
<ComputeNode hostname="hostname_3" uuid="Node_3" id="3" state="up" human_id="" status="enabled" vcpus="16" disk="250" disk_capacity="250" memory="64">
|
||||||
<Instance hostname="" human_id="" state="active" uuid="INSTANCE_5" ResourceType.cpu_cores="2" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance human_id="" state="active" uuid="INSTANCE_5" vcpus="2" disk="20" disk_capacity="20" memory="2"/>
|
||||||
</ComputeNode>
|
</ComputeNode>
|
||||||
</ModelRoot>
|
</ModelRoot>
|
||||||
|
@ -1,16 +1,16 @@
|
|||||||
<ModelRoot>
|
<ModelRoot>
|
||||||
<ComputeNode hostname="hostname_0" uuid="Node_0" id="0" state="up" human_id="" status="enabled" ResourceType.cpu_cores="16" ResourceType.disk="250" ResourceType.disk_capacity="250" ResourceType.memory="64">
|
<ComputeNode hostname="hostname_0" uuid="Node_0" id="0" state="up" human_id="" status="enabled" vcpus="16" disk="250" disk_capacity="250" memory="64">
|
||||||
<Instance hostname="" human_id="" state="active" uuid="INSTANCE_0" ResourceType.cpu_cores="2" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance human_id="" state="active" uuid="INSTANCE_0" vcpus="2" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance hostname="" human_id="" state="active" uuid="INSTANCE_1" ResourceType.cpu_cores="2" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance human_id="" state="active" uuid="INSTANCE_1" vcpus="2" disk="20" disk_capacity="20" memory="2"/>
|
||||||
<Instance hostname="" human_id="" state="active" uuid="INSTANCE_2" ResourceType.cpu_cores="2" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance human_id="" state="active" uuid="INSTANCE_2" vcpus="2" disk="20" disk_capacity="20" memory="2"/>
|
||||||
</ComputeNode>
|
</ComputeNode>
|
||||||
<ComputeNode hostname="hostname_1" uuid="Node_1" id="1" state="up" human_id="" status="enabled" ResourceType.cpu_cores="16" ResourceType.disk="250" ResourceType.disk_capacity="250" ResourceType.memory="64">
|
<ComputeNode hostname="hostname_1" uuid="Node_1" id="1" state="up" human_id="" status="enabled" vcpus="16" disk="250" disk_capacity="250" memory="64">
|
||||||
<Instance hostname="" human_id="" state="active" uuid="INSTANCE_3" ResourceType.cpu_cores="2" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance human_id="" state="active" uuid="INSTANCE_3" vcpus="2" disk="20" disk_capacity="20" memory="2"/>
|
||||||
</ComputeNode>
|
</ComputeNode>
|
||||||
<ComputeNode hostname="hostname_2" uuid="Node_2" id="2" state="up" human_id="" status="enabled" ResourceType.cpu_cores="16" ResourceType.disk="250" ResourceType.disk_capacity="250" ResourceType.memory="64">
|
<ComputeNode hostname="hostname_2" uuid="Node_2" id="2" state="up" human_id="" status="enabled" vcpus="16" disk="250" disk_capacity="250" memory="64">
|
||||||
<Instance hostname="" human_id="" state="active" uuid="INSTANCE_4" ResourceType.cpu_cores="2" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance human_id="" state="active" uuid="INSTANCE_4" vcpus="2" disk="20" disk_capacity="20" memory="2"/>
|
||||||
</ComputeNode>
|
</ComputeNode>
|
||||||
<ComputeNode hostname="hostname_3" uuid="Node_3" id="3" state="up" human_id="" status="disabled" ResourceType.cpu_cores="16" ResourceType.disk="250" ResourceType.disk_capacity="250" ResourceType.memory="64">
|
<ComputeNode hostname="hostname_3" uuid="Node_3" id="3" state="up" human_id="" status="disabled" vcpus="16" disk="250" disk_capacity="250" memory="64">
|
||||||
<Instance hostname="" human_id="" state="active" uuid="INSTANCE_5" ResourceType.cpu_cores="2" ResourceType.disk="20" ResourceType.disk_capacity="20" ResourceType.memory="2"/>
|
<Instance human_id="" state="active" uuid="INSTANCE_5" vcpus="2" disk="20" disk_capacity="20" memory="2"/>
|
||||||
</ComputeNode>
|
</ComputeNode>
|
||||||
</ModelRoot>
|
</ModelRoot>
|
||||||
|
@ -102,11 +102,12 @@ class FakeCeilometerMetrics(object):
|
|||||||
Returns relative node CPU utilization <0, 100>.
|
Returns relative node CPU utilization <0, 100>.
|
||||||
:param r_id: resource id
|
:param r_id: resource id
|
||||||
"""
|
"""
|
||||||
uuid = '%s_%s' % (r_id.split('_')[0], r_id.split('_')[1])
|
node_uuid = '%s_%s' % (r_id.split('_')[0], r_id.split('_')[1])
|
||||||
instances = self.model.get_mapping().get_node_instances_by_uuid(uuid)
|
node = self.model.get_node_by_uuid(node_uuid)
|
||||||
|
instances = self.model.get_node_instances(node)
|
||||||
util_sum = 0.0
|
util_sum = 0.0
|
||||||
node_cpu_cores = self.model.get_resource_by_uuid(
|
node_cpu_cores = self.model.get_resource_by_uuid(
|
||||||
element.ResourceType.cpu_cores).get_capacity_by_uuid(uuid)
|
element.ResourceType.cpu_cores).get_capacity_by_uuid(node.uuid)
|
||||||
for instance_uuid in instances:
|
for instance_uuid in instances:
|
||||||
instance_cpu_cores = self.model.get_resource_by_uuid(
|
instance_cpu_cores = self.model.get_resource_by_uuid(
|
||||||
element.ResourceType.cpu_cores).\
|
element.ResourceType.cpu_cores).\
|
||||||
|
@ -54,7 +54,7 @@ class FakerModelCollector(base.BaseClusterDataModelCollector):
|
|||||||
def build_scenario_1(self):
|
def build_scenario_1(self):
|
||||||
instances = []
|
instances = []
|
||||||
|
|
||||||
current_state_cluster = modelroot.ModelRoot()
|
model = modelroot.ModelRoot()
|
||||||
# number of nodes
|
# number of nodes
|
||||||
node_count = 5
|
node_count = 5
|
||||||
# number max of instance per node
|
# number max of instance per node
|
||||||
@ -62,74 +62,52 @@ class FakerModelCollector(base.BaseClusterDataModelCollector):
|
|||||||
# total number of virtual machine
|
# total number of virtual machine
|
||||||
instance_count = (node_count * node_instance_count)
|
instance_count = (node_count * node_instance_count)
|
||||||
|
|
||||||
# define ressouce ( CPU, MEM disk, ... )
|
|
||||||
mem = element.Resource(element.ResourceType.memory)
|
|
||||||
# 2199.954 Mhz
|
|
||||||
num_cores = element.Resource(element.ResourceType.cpu_cores)
|
|
||||||
disk = element.Resource(element.ResourceType.disk)
|
|
||||||
disk_capacity = element.Resource(element.ResourceType.disk_capacity)
|
|
||||||
|
|
||||||
current_state_cluster.create_resource(mem)
|
|
||||||
current_state_cluster.create_resource(num_cores)
|
|
||||||
current_state_cluster.create_resource(disk)
|
|
||||||
current_state_cluster.create_resource(disk_capacity)
|
|
||||||
|
|
||||||
for id_ in range(0, node_count):
|
for id_ in range(0, node_count):
|
||||||
node_uuid = "Node_{0}".format(id_)
|
node_uuid = "Node_{0}".format(id_)
|
||||||
node = element.ComputeNode(id_)
|
hostname = "hostname_{0}".format(id_)
|
||||||
node.uuid = node_uuid
|
node_attributes = {
|
||||||
node.hostname = "hostname_{0}".format(id_)
|
"id": id_,
|
||||||
|
"uuid": node_uuid,
|
||||||
mem.set_capacity(node, 132)
|
"hostname": hostname,
|
||||||
disk.set_capacity(node, 250)
|
"memory": 132,
|
||||||
disk_capacity.set_capacity(node, 250)
|
"disk": 250,
|
||||||
num_cores.set_capacity(node, 40)
|
"disk_capacity": 250,
|
||||||
current_state_cluster.add_node(node)
|
"vcpus": 40,
|
||||||
|
}
|
||||||
|
node = element.ComputeNode(**node_attributes)
|
||||||
|
model.add_node(node)
|
||||||
|
|
||||||
for i in range(0, instance_count):
|
for i in range(0, instance_count):
|
||||||
instance_uuid = "INSTANCE_{0}".format(i)
|
instance_uuid = "INSTANCE_{0}".format(i)
|
||||||
instance = element.Instance()
|
instance_attributes = {
|
||||||
instance.uuid = instance_uuid
|
"uuid": instance_uuid,
|
||||||
mem.set_capacity(instance, 2)
|
"memory": 2,
|
||||||
disk.set_capacity(instance, 20)
|
"disk": 20,
|
||||||
disk_capacity.set_capacity(instance, 20)
|
"disk_capacity": 20,
|
||||||
num_cores.set_capacity(instance, 10)
|
"vcpus": 10,
|
||||||
|
}
|
||||||
|
|
||||||
|
instance = element.Instance(**instance_attributes)
|
||||||
instances.append(instance)
|
instances.append(instance)
|
||||||
current_state_cluster.add_instance(instance)
|
model.add_instance(instance)
|
||||||
|
|
||||||
current_state_cluster.mapping.map(
|
mappings = [
|
||||||
current_state_cluster.get_node_by_uuid("Node_0"),
|
("INSTANCE_0", "Node_0"),
|
||||||
current_state_cluster.get_instance_by_uuid("INSTANCE_0"))
|
("INSTANCE_1", "Node_0"),
|
||||||
|
("INSTANCE_2", "Node_1"),
|
||||||
|
("INSTANCE_3", "Node_2"),
|
||||||
|
("INSTANCE_4", "Node_2"),
|
||||||
|
("INSTANCE_5", "Node_2"),
|
||||||
|
("INSTANCE_6", "Node_3"),
|
||||||
|
("INSTANCE_7", "Node_4"),
|
||||||
|
]
|
||||||
|
for instance_uuid, node_uuid in mappings:
|
||||||
|
model.map_instance(
|
||||||
|
model.get_instance_by_uuid(instance_uuid),
|
||||||
|
model.get_node_by_uuid(node_uuid),
|
||||||
|
)
|
||||||
|
|
||||||
current_state_cluster.mapping.map(
|
return model
|
||||||
current_state_cluster.get_node_by_uuid("Node_0"),
|
|
||||||
current_state_cluster.get_instance_by_uuid("INSTANCE_1"))
|
|
||||||
|
|
||||||
current_state_cluster.mapping.map(
|
|
||||||
current_state_cluster.get_node_by_uuid("Node_1"),
|
|
||||||
current_state_cluster.get_instance_by_uuid("INSTANCE_2"))
|
|
||||||
|
|
||||||
current_state_cluster.mapping.map(
|
|
||||||
current_state_cluster.get_node_by_uuid("Node_2"),
|
|
||||||
current_state_cluster.get_instance_by_uuid("INSTANCE_3"))
|
|
||||||
|
|
||||||
current_state_cluster.mapping.map(
|
|
||||||
current_state_cluster.get_node_by_uuid("Node_2"),
|
|
||||||
current_state_cluster.get_instance_by_uuid("INSTANCE_4"))
|
|
||||||
|
|
||||||
current_state_cluster.mapping.map(
|
|
||||||
current_state_cluster.get_node_by_uuid("Node_2"),
|
|
||||||
current_state_cluster.get_instance_by_uuid("INSTANCE_5"))
|
|
||||||
|
|
||||||
current_state_cluster.mapping.map(
|
|
||||||
current_state_cluster.get_node_by_uuid("Node_3"),
|
|
||||||
current_state_cluster.get_instance_by_uuid("INSTANCE_6"))
|
|
||||||
|
|
||||||
current_state_cluster.mapping.map(
|
|
||||||
current_state_cluster.get_node_by_uuid("Node_4"),
|
|
||||||
current_state_cluster.get_instance_by_uuid("INSTANCE_7"))
|
|
||||||
|
|
||||||
return current_state_cluster
|
|
||||||
|
|
||||||
def generate_scenario_1(self):
|
def generate_scenario_1(self):
|
||||||
return self.load_model('scenario_1.xml')
|
return self.load_model('scenario_1.xml')
|
||||||
|
@ -26,7 +26,6 @@ from watcher.common import exception
|
|||||||
from watcher.common import nova_helper
|
from watcher.common import nova_helper
|
||||||
from watcher.common import service as watcher_service
|
from watcher.common import service as watcher_service
|
||||||
from watcher.decision_engine.model import element
|
from watcher.decision_engine.model import element
|
||||||
from watcher.decision_engine.model import model_root
|
|
||||||
from watcher.decision_engine.model.notification import nova as novanotification
|
from watcher.decision_engine.model.notification import nova as novanotification
|
||||||
from watcher.tests import base as base_test
|
from watcher.tests import base as base_test
|
||||||
from watcher.tests.decision_engine.model import faker_cluster_state
|
from watcher.tests.decision_engine.model import faker_cluster_state
|
||||||
@ -35,7 +34,8 @@ from watcher.tests.decision_engine.model.notification import fake_managers
|
|||||||
|
|
||||||
class NotificationTestCase(base_test.TestCase):
|
class NotificationTestCase(base_test.TestCase):
|
||||||
|
|
||||||
def load_message(self, filename):
|
@staticmethod
|
||||||
|
def load_message(filename):
|
||||||
cwd = os.path.abspath(os.path.dirname(__file__))
|
cwd = os.path.abspath(os.path.dirname(__file__))
|
||||||
data_folder = os.path.join(cwd, "data")
|
data_folder = os.path.join(cwd, "data")
|
||||||
|
|
||||||
@ -188,6 +188,9 @@ class TestNovaNotifications(NotificationTestCase):
|
|||||||
side_effect=lambda uuid: mock.Mock(
|
side_effect=lambda uuid: mock.Mock(
|
||||||
name='m_get_compute_node_by_hostname',
|
name='m_get_compute_node_by_hostname',
|
||||||
id=3,
|
id=3,
|
||||||
|
hypervisor_hostname="Node_2",
|
||||||
|
state='up',
|
||||||
|
status='enabled',
|
||||||
uuid=uuid,
|
uuid=uuid,
|
||||||
memory_mb=7777,
|
memory_mb=7777,
|
||||||
vcpus=42,
|
vcpus=42,
|
||||||
@ -215,7 +218,7 @@ class TestNovaNotifications(NotificationTestCase):
|
|||||||
|
|
||||||
instance0 = compute_model.get_instance_by_uuid(instance0_uuid)
|
instance0 = compute_model.get_instance_by_uuid(instance0_uuid)
|
||||||
cpu_capacity = compute_model.get_resource_by_uuid(
|
cpu_capacity = compute_model.get_resource_by_uuid(
|
||||||
element.ResourceType.cpu_cores)
|
element.ResourceType.vcpus)
|
||||||
disk = compute_model.get_resource_by_uuid(
|
disk = compute_model.get_resource_by_uuid(
|
||||||
element.ResourceType.disk)
|
element.ResourceType.disk)
|
||||||
disk_capacity = compute_model.get_resource_by_uuid(
|
disk_capacity = compute_model.get_resource_by_uuid(
|
||||||
@ -239,7 +242,7 @@ class TestNovaNotifications(NotificationTestCase):
|
|||||||
def test_instance_update_node_notfound_set_unmapped(
|
def test_instance_update_node_notfound_set_unmapped(
|
||||||
self, m_nova_helper_cls):
|
self, m_nova_helper_cls):
|
||||||
m_get_compute_node_by_hostname = mock.Mock(
|
m_get_compute_node_by_hostname = mock.Mock(
|
||||||
side_effect=exception.ComputeNodeNotFound)
|
side_effect=exception.ComputeNodeNotFound(name="TEST"))
|
||||||
m_nova_helper_cls.return_value = mock.Mock(
|
m_nova_helper_cls.return_value = mock.Mock(
|
||||||
get_compute_node_by_hostname=m_get_compute_node_by_hostname,
|
get_compute_node_by_hostname=m_get_compute_node_by_hostname,
|
||||||
name='m_nova_helper')
|
name='m_nova_helper')
|
||||||
@ -263,7 +266,7 @@ class TestNovaNotifications(NotificationTestCase):
|
|||||||
|
|
||||||
instance0 = compute_model.get_instance_by_uuid(instance0_uuid)
|
instance0 = compute_model.get_instance_by_uuid(instance0_uuid)
|
||||||
cpu_capacity = compute_model.get_resource_by_uuid(
|
cpu_capacity = compute_model.get_resource_by_uuid(
|
||||||
element.ResourceType.cpu_cores)
|
element.ResourceType.vcpus)
|
||||||
disk = compute_model.get_resource_by_uuid(
|
disk = compute_model.get_resource_by_uuid(
|
||||||
element.ResourceType.disk)
|
element.ResourceType.disk)
|
||||||
disk_capacity = compute_model.get_resource_by_uuid(
|
disk_capacity = compute_model.get_resource_by_uuid(
|
||||||
@ -304,7 +307,7 @@ class TestNovaNotifications(NotificationTestCase):
|
|||||||
|
|
||||||
instance0 = compute_model.get_instance_by_uuid(instance0_uuid)
|
instance0 = compute_model.get_instance_by_uuid(instance0_uuid)
|
||||||
cpu_capacity = compute_model.get_resource_by_uuid(
|
cpu_capacity = compute_model.get_resource_by_uuid(
|
||||||
element.ResourceType.cpu_cores)
|
element.ResourceType.vcpus)
|
||||||
disk_capacity = compute_model.get_resource_by_uuid(
|
disk_capacity = compute_model.get_resource_by_uuid(
|
||||||
element.ResourceType.disk)
|
element.ResourceType.disk)
|
||||||
memory_capacity = compute_model.get_resource_by_uuid(
|
memory_capacity = compute_model.get_resource_by_uuid(
|
||||||
@ -324,8 +327,6 @@ class TestNovaNotifications(NotificationTestCase):
|
|||||||
|
|
||||||
# Before
|
# Before
|
||||||
self.assertTrue(compute_model.get_instance_by_uuid(instance0_uuid))
|
self.assertTrue(compute_model.get_instance_by_uuid(instance0_uuid))
|
||||||
for resource in compute_model.resource.values():
|
|
||||||
self.assertIn(instance0_uuid, resource.mapping)
|
|
||||||
|
|
||||||
message = self.load_message('scenario3_instance-delete-end.json')
|
message = self.load_message('scenario3_instance-delete-end.json')
|
||||||
handler.info(
|
handler.info(
|
||||||
@ -341,9 +342,6 @@ class TestNovaNotifications(NotificationTestCase):
|
|||||||
exception.InstanceNotFound,
|
exception.InstanceNotFound,
|
||||||
compute_model.get_instance_by_uuid, instance0_uuid)
|
compute_model.get_instance_by_uuid, instance0_uuid)
|
||||||
|
|
||||||
for resource in compute_model.resource.values():
|
|
||||||
self.assertNotIn(instance0_uuid, resource.mapping)
|
|
||||||
|
|
||||||
|
|
||||||
class TestLegacyNovaNotifications(NotificationTestCase):
|
class TestLegacyNovaNotifications(NotificationTestCase):
|
||||||
|
|
||||||
@ -377,7 +375,7 @@ class TestLegacyNovaNotifications(NotificationTestCase):
|
|||||||
|
|
||||||
instance0 = compute_model.get_instance_by_uuid(instance0_uuid)
|
instance0 = compute_model.get_instance_by_uuid(instance0_uuid)
|
||||||
cpu_capacity = compute_model.get_resource_by_uuid(
|
cpu_capacity = compute_model.get_resource_by_uuid(
|
||||||
element.ResourceType.cpu_cores)
|
element.ResourceType.vcpus)
|
||||||
disk_capacity = compute_model.get_resource_by_uuid(
|
disk_capacity = compute_model.get_resource_by_uuid(
|
||||||
element.ResourceType.disk)
|
element.ResourceType.disk)
|
||||||
memory_capacity = compute_model.get_resource_by_uuid(
|
memory_capacity = compute_model.get_resource_by_uuid(
|
||||||
@ -410,31 +408,6 @@ class TestLegacyNovaNotifications(NotificationTestCase):
|
|||||||
|
|
||||||
self.assertEqual(element.InstanceState.PAUSED.value, instance0.state)
|
self.assertEqual(element.InstanceState.PAUSED.value, instance0.state)
|
||||||
|
|
||||||
def test_legacy_instance_update_instance_notfound_creates(self):
|
|
||||||
compute_model = self.fake_cdmc.generate_scenario_3_with_2_nodes()
|
|
||||||
self.fake_cdmc.cluster_data_model = compute_model
|
|
||||||
handler = novanotification.LegacyInstanceUpdated(self.fake_cdmc)
|
|
||||||
|
|
||||||
instance0_uuid = '73b09e16-35b7-4922-804e-e8f5d9b740fc'
|
|
||||||
|
|
||||||
message = self.load_message('scenario3_legacy_instance-update.json')
|
|
||||||
|
|
||||||
with mock.patch.object(
|
|
||||||
model_root.ModelRoot, 'get_instance_by_uuid'
|
|
||||||
) as m_get_instance_by_uuid:
|
|
||||||
m_get_instance_by_uuid.side_effect = exception.InstanceNotFound(
|
|
||||||
name='TEST')
|
|
||||||
handler.info(
|
|
||||||
ctxt=self.context,
|
|
||||||
publisher_id=message['publisher_id'],
|
|
||||||
event_type=message['event_type'],
|
|
||||||
payload=message['payload'],
|
|
||||||
metadata=self.FAKE_METADATA,
|
|
||||||
)
|
|
||||||
|
|
||||||
instance0 = compute_model.get_instance_by_uuid(instance0_uuid)
|
|
||||||
self.assertEqual(element.InstanceState.PAUSED.value, instance0.state)
|
|
||||||
|
|
||||||
@mock.patch.object(nova_helper, "NovaHelper")
|
@mock.patch.object(nova_helper, "NovaHelper")
|
||||||
def test_legacy_instance_update_node_notfound_still_creates(
|
def test_legacy_instance_update_node_notfound_still_creates(
|
||||||
self, m_nova_helper_cls):
|
self, m_nova_helper_cls):
|
||||||
@ -443,6 +416,9 @@ class TestLegacyNovaNotifications(NotificationTestCase):
|
|||||||
name='m_get_compute_node_by_hostname',
|
name='m_get_compute_node_by_hostname',
|
||||||
id=3,
|
id=3,
|
||||||
uuid=uuid,
|
uuid=uuid,
|
||||||
|
hypervisor_hostname="Node_2",
|
||||||
|
state='up',
|
||||||
|
status='enabled',
|
||||||
memory_mb=7777,
|
memory_mb=7777,
|
||||||
vcpus=42,
|
vcpus=42,
|
||||||
free_disk_gb=974,
|
free_disk_gb=974,
|
||||||
@ -470,7 +446,7 @@ class TestLegacyNovaNotifications(NotificationTestCase):
|
|||||||
|
|
||||||
instance0 = compute_model.get_instance_by_uuid(instance0_uuid)
|
instance0 = compute_model.get_instance_by_uuid(instance0_uuid)
|
||||||
cpu_capacity = compute_model.get_resource_by_uuid(
|
cpu_capacity = compute_model.get_resource_by_uuid(
|
||||||
element.ResourceType.cpu_cores)
|
element.ResourceType.vcpus)
|
||||||
disk = compute_model.get_resource_by_uuid(
|
disk = compute_model.get_resource_by_uuid(
|
||||||
element.ResourceType.disk)
|
element.ResourceType.disk)
|
||||||
disk_capacity = compute_model.get_resource_by_uuid(
|
disk_capacity = compute_model.get_resource_by_uuid(
|
||||||
@ -519,7 +495,7 @@ class TestLegacyNovaNotifications(NotificationTestCase):
|
|||||||
|
|
||||||
instance0 = compute_model.get_instance_by_uuid(instance0_uuid)
|
instance0 = compute_model.get_instance_by_uuid(instance0_uuid)
|
||||||
cpu_capacity = compute_model.get_resource_by_uuid(
|
cpu_capacity = compute_model.get_resource_by_uuid(
|
||||||
element.ResourceType.cpu_cores)
|
element.ResourceType.vcpus)
|
||||||
disk = compute_model.get_resource_by_uuid(
|
disk = compute_model.get_resource_by_uuid(
|
||||||
element.ResourceType.disk)
|
element.ResourceType.disk)
|
||||||
disk_capacity = compute_model.get_resource_by_uuid(
|
disk_capacity = compute_model.get_resource_by_uuid(
|
||||||
@ -571,8 +547,6 @@ class TestLegacyNovaNotifications(NotificationTestCase):
|
|||||||
|
|
||||||
# Before
|
# Before
|
||||||
self.assertTrue(compute_model.get_instance_by_uuid(instance0_uuid))
|
self.assertTrue(compute_model.get_instance_by_uuid(instance0_uuid))
|
||||||
for resource in compute_model.resource.values():
|
|
||||||
self.assertIn(instance0_uuid, resource.mapping)
|
|
||||||
|
|
||||||
message = self.load_message(
|
message = self.load_message(
|
||||||
'scenario3_legacy_instance-delete-end.json')
|
'scenario3_legacy_instance-delete-end.json')
|
||||||
@ -588,6 +562,3 @@ class TestLegacyNovaNotifications(NotificationTestCase):
|
|||||||
self.assertRaises(
|
self.assertRaises(
|
||||||
exception.InstanceNotFound,
|
exception.InstanceNotFound,
|
||||||
compute_model.get_instance_by_uuid, instance0_uuid)
|
compute_model.get_instance_by_uuid, instance0_uuid)
|
||||||
|
|
||||||
for resource in compute_model.resource.values():
|
|
||||||
self.assertNotIn(instance0_uuid, resource.mapping)
|
|
||||||
|
@ -1,33 +0,0 @@
|
|||||||
# -*- encoding: utf-8 -*-
|
|
||||||
# Copyright (c) 2015 b<>com
|
|
||||||
#
|
|
||||||
# Authors: Jean-Emile DARTOIS <jean-emile.dartois@b-com.com>
|
|
||||||
#
|
|
||||||
# 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 watcher.decision_engine.model import element
|
|
||||||
from watcher.tests import base
|
|
||||||
|
|
||||||
|
|
||||||
class TestDiskInfo(base.TestCase):
|
|
||||||
def test_all(self):
|
|
||||||
disk_information = element.DiskInfo()
|
|
||||||
disk_information.set_size(1024)
|
|
||||||
self.assertEqual(1024, disk_information.get_size())
|
|
||||||
|
|
||||||
disk_information.set_scheduler = "scheduler_qcq"
|
|
||||||
|
|
||||||
disk_information.set_device_name("nom_qcq")
|
|
||||||
self.assertEqual("nom_qcq", disk_information.get_device_name())
|
|
72
watcher/tests/decision_engine/model/test_element.py
Normal file
72
watcher/tests/decision_engine/model/test_element.py
Normal file
@ -0,0 +1,72 @@
|
|||||||
|
# -*- encoding: utf-8 -*-
|
||||||
|
# Copyright (c) 2017 b<>com
|
||||||
|
#
|
||||||
|
# Authors: Vincent FRANCOISE <Vincent.FRANCOISE@b-com.com>
|
||||||
|
#
|
||||||
|
# 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 watcher.decision_engine.model import element
|
||||||
|
from watcher.tests import base
|
||||||
|
|
||||||
|
|
||||||
|
class TestElement(base.TestCase):
|
||||||
|
|
||||||
|
scenarios = [
|
||||||
|
("ComputeNode_with_all_fields", dict(
|
||||||
|
cls=element.Instance,
|
||||||
|
data={
|
||||||
|
'uuid': 'FAKE_UUID',
|
||||||
|
'state': 'state',
|
||||||
|
'hostname': 'hostname',
|
||||||
|
'human_id': 'human_id',
|
||||||
|
'memory': 111,
|
||||||
|
'vcpus': 222,
|
||||||
|
'disk': 333,
|
||||||
|
'disk_capacity': 444,
|
||||||
|
})),
|
||||||
|
("ComputeNode_with_some_fields", dict(
|
||||||
|
cls=element.Instance,
|
||||||
|
data={
|
||||||
|
'uuid': 'FAKE_UUID',
|
||||||
|
'state': 'state',
|
||||||
|
'vcpus': 222,
|
||||||
|
'disk': 333,
|
||||||
|
'disk_capacity': 444,
|
||||||
|
})),
|
||||||
|
("Instance_with_all_fields", dict(
|
||||||
|
cls=element.Instance,
|
||||||
|
data={
|
||||||
|
'uuid': 'FAKE_UUID',
|
||||||
|
'state': 'state',
|
||||||
|
'hostname': 'hostname',
|
||||||
|
'human_id': 'human_id',
|
||||||
|
'memory': 111,
|
||||||
|
'vcpus': 222,
|
||||||
|
'disk': 333,
|
||||||
|
'disk_capacity': 444,
|
||||||
|
})),
|
||||||
|
("Instance_with_some_fields", dict(
|
||||||
|
cls=element.Instance,
|
||||||
|
data={
|
||||||
|
'uuid': 'FAKE_UUID',
|
||||||
|
'state': 'state',
|
||||||
|
'vcpus': 222,
|
||||||
|
'disk': 333,
|
||||||
|
'disk_capacity': 444,
|
||||||
|
})),
|
||||||
|
]
|
||||||
|
|
||||||
|
def test_as_xml_element(self):
|
||||||
|
el = self.cls(**self.data)
|
||||||
|
el.as_xml_element()
|
@ -1,30 +0,0 @@
|
|||||||
# -*- encoding: utf-8 -*-
|
|
||||||
# Copyright (c) 2015 b<>com
|
|
||||||
#
|
|
||||||
# Authors: Jean-Emile DARTOIS <jean-emile.dartois@b-com.com>
|
|
||||||
#
|
|
||||||
# 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 watcher.decision_engine.model import element
|
|
||||||
from watcher.tests import base
|
|
||||||
|
|
||||||
|
|
||||||
class TestInstance(base.TestCase):
|
|
||||||
|
|
||||||
def test_namedelement(self):
|
|
||||||
instance = element.Instance()
|
|
||||||
instance.state = element.InstanceState.ACTIVE
|
|
||||||
self.assertEqual(element.InstanceState.ACTIVE, instance.state)
|
|
||||||
instance.human_id = "human_05"
|
|
||||||
self.assertEqual("human_05", instance.human_id)
|
|
@ -1,113 +0,0 @@
|
|||||||
# -*- encoding: utf-8 -*-
|
|
||||||
# Copyright (c) 2015 b<>com
|
|
||||||
#
|
|
||||||
# Authors: Jean-Emile DARTOIS <jean-emile.dartois@b-com.com>
|
|
||||||
#
|
|
||||||
# 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_utils import uuidutils
|
|
||||||
|
|
||||||
from watcher.decision_engine.model import element
|
|
||||||
from watcher.tests import base
|
|
||||||
from watcher.tests.decision_engine.model import faker_cluster_state
|
|
||||||
|
|
||||||
|
|
||||||
class TestMapping(base.TestCase):
|
|
||||||
|
|
||||||
INST1_UUID = "73b09e16-35b7-4922-804e-e8f5d9b740fc"
|
|
||||||
INST2_UUID = "a4cab39b-9828-413a-bf88-f76921bf1517"
|
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
super(TestMapping, self).setUp()
|
|
||||||
self.fake_cluster = faker_cluster_state.FakerModelCollector()
|
|
||||||
|
|
||||||
def test_get_node_from_instance(self):
|
|
||||||
model = self.fake_cluster.generate_scenario_3_with_2_nodes()
|
|
||||||
|
|
||||||
instances = model.get_all_instances()
|
|
||||||
keys = list(instances.keys())
|
|
||||||
instance = instances[keys[0]]
|
|
||||||
if instance.uuid != self.INST1_UUID:
|
|
||||||
instance = instances[keys[1]]
|
|
||||||
node = model.mapping.get_node_from_instance(instance)
|
|
||||||
self.assertEqual('Node_0', node.uuid)
|
|
||||||
|
|
||||||
def test_get_node_by_instance_uuid(self):
|
|
||||||
model = self.fake_cluster.generate_scenario_3_with_2_nodes()
|
|
||||||
|
|
||||||
nodes = model.mapping.get_node_instances_by_uuid("BLABLABLA")
|
|
||||||
self.assertEqual(0, len(nodes))
|
|
||||||
|
|
||||||
def test_get_all_instances(self):
|
|
||||||
model = self.fake_cluster.generate_scenario_3_with_2_nodes()
|
|
||||||
|
|
||||||
instances = model.get_all_instances()
|
|
||||||
self.assertEqual(2, len(instances))
|
|
||||||
self.assertEqual(element.InstanceState.ACTIVE.value,
|
|
||||||
instances[self.INST1_UUID].state)
|
|
||||||
self.assertEqual(self.INST1_UUID, instances[self.INST1_UUID].uuid)
|
|
||||||
self.assertEqual(element.InstanceState.ACTIVE.value,
|
|
||||||
instances[self.INST2_UUID].state)
|
|
||||||
self.assertEqual(self.INST2_UUID, instances[self.INST2_UUID].uuid)
|
|
||||||
|
|
||||||
def test_get_mapping(self):
|
|
||||||
model = self.fake_cluster.generate_scenario_3_with_2_nodes()
|
|
||||||
instance_mapping = model.mapping.instance_mapping
|
|
||||||
self.assertEqual(2, len(instance_mapping))
|
|
||||||
self.assertEqual('Node_0', instance_mapping[self.INST1_UUID])
|
|
||||||
self.assertEqual('Node_1', instance_mapping[self.INST2_UUID])
|
|
||||||
|
|
||||||
def test_migrate_instance(self):
|
|
||||||
model = self.fake_cluster.generate_scenario_3_with_2_nodes()
|
|
||||||
instances = model.get_all_instances()
|
|
||||||
keys = list(instances.keys())
|
|
||||||
instance0 = instances[keys[0]]
|
|
||||||
node0 = model.mapping.get_node_by_instance_uuid(instance0.uuid)
|
|
||||||
instance1 = instances[keys[1]]
|
|
||||||
node1 = model.mapping.get_node_by_instance_uuid(instance1.uuid)
|
|
||||||
|
|
||||||
self.assertEqual(
|
|
||||||
False,
|
|
||||||
model.migrate_instance(instance1, node1, node1))
|
|
||||||
self.assertEqual(
|
|
||||||
False,
|
|
||||||
model.migrate_instance(instance1, node0, node0))
|
|
||||||
self.assertEqual(
|
|
||||||
True,
|
|
||||||
model.migrate_instance(instance1, node1, node0))
|
|
||||||
self.assertEqual(
|
|
||||||
True,
|
|
||||||
model.migrate_instance(instance1, node0, node1))
|
|
||||||
|
|
||||||
def test_unmap_by_uuid_log_warning(self):
|
|
||||||
model = self.fake_cluster.generate_scenario_3_with_2_nodes()
|
|
||||||
instances = model.get_all_instances()
|
|
||||||
keys = list(instances.keys())
|
|
||||||
instance0 = instances[keys[0]]
|
|
||||||
uuid_ = uuidutils.generate_uuid()
|
|
||||||
node = element.ComputeNode(id=1)
|
|
||||||
node.uuid = uuid_
|
|
||||||
|
|
||||||
model.mapping.unmap_by_uuid(node.uuid, instance0.uuid)
|
|
||||||
|
|
||||||
def test_unmap_by_uuid(self):
|
|
||||||
model = self.fake_cluster.generate_scenario_3_with_2_nodes()
|
|
||||||
instances = model.get_all_instances()
|
|
||||||
keys = list(instances.keys())
|
|
||||||
instance0 = instances[keys[0]]
|
|
||||||
node0 = model.mapping.get_node_by_instance_uuid(instance0.uuid)
|
|
||||||
|
|
||||||
model.mapping.unmap_by_uuid(node0.uuid, instance0.uuid)
|
|
||||||
self.assertEqual(0, len(model.mapping.get_node_instances_by_uuid(
|
|
||||||
node0.uuid)))
|
|
@ -47,9 +47,9 @@ class TestModel(base.TestCase):
|
|||||||
fake_cluster = faker_cluster_state.FakerModelCollector()
|
fake_cluster = faker_cluster_state.FakerModelCollector()
|
||||||
model = fake_cluster.build_scenario_1()
|
model = fake_cluster.build_scenario_1()
|
||||||
|
|
||||||
self.assertEqual(5, len(model._nodes))
|
self.assertEqual(5, len(model.get_all_compute_nodes()))
|
||||||
self.assertEqual(35, len(model._instances))
|
self.assertEqual(35, len(model.get_all_instances()))
|
||||||
self.assertEqual(5, len(model.mapping.get_mapping()))
|
self.assertEqual(8, len(model.edges()))
|
||||||
|
|
||||||
expected_struct_str = self.load_data('scenario_1.xml')
|
expected_struct_str = self.load_data('scenario_1.xml')
|
||||||
parser = etree.XMLParser(remove_blank_text=True)
|
parser = etree.XMLParser(remove_blank_text=True)
|
||||||
|
@ -42,34 +42,37 @@ class TestDefaultScope(base.TestCase):
|
|||||||
for i in range(2)]
|
for i in range(2)]
|
||||||
model = default.DefaultScope(audit_scope,
|
model = default.DefaultScope(audit_scope,
|
||||||
osc=mock.Mock()).get_scoped_model(cluster)
|
osc=mock.Mock()).get_scoped_model(cluster)
|
||||||
nodes = {'Node_4': set([]), 'Node_0': set([]), 'Node_3': set([]),
|
|
||||||
'Node_1': set(['INSTANCE_2']), 'Node_2': set([])}
|
expected_edges = [('INSTANCE_2', 'Node_1')]
|
||||||
self.assertEqual(nodes, model.get_mapping().get_mapping())
|
edges = [(src.uuid, dst.uuid) for src, dst in model.edges()]
|
||||||
|
self.assertEqual(sorted(expected_edges), sorted(edges))
|
||||||
|
|
||||||
@mock.patch.object(nova_helper.NovaHelper, 'get_availability_zone_list')
|
@mock.patch.object(nova_helper.NovaHelper, 'get_availability_zone_list')
|
||||||
def test_get_scoped_model_without_scope(self, mock_zone_list):
|
def test_get_scoped_model_without_scope(self, mock_zone_list):
|
||||||
cluster = self.fake_cluster.generate_scenario_1()
|
model = self.fake_cluster.generate_scenario_1()
|
||||||
default.DefaultScope([],
|
default.DefaultScope([],
|
||||||
osc=mock.Mock()).get_scoped_model(cluster)
|
osc=mock.Mock()).get_scoped_model(model)
|
||||||
assert not mock_zone_list.called
|
assert not mock_zone_list.called
|
||||||
|
|
||||||
def test__remove_instance(self):
|
def test_remove_instance(self):
|
||||||
cluster = self.fake_cluster.generate_scenario_1()
|
model = self.fake_cluster.generate_scenario_1()
|
||||||
default.DefaultScope([],
|
default.DefaultScope([], osc=mock.Mock()).remove_instance(
|
||||||
osc=mock.Mock())._remove_instance(cluster,
|
model, model.get_instance_by_uuid('INSTANCE_2'), 'Node_1')
|
||||||
'INSTANCE_2',
|
expected_edges = [
|
||||||
'Node_1')
|
('INSTANCE_0', 'Node_0'),
|
||||||
expected_map = {'Node_4': set(['INSTANCE_7']), 'Node_1': set([]),
|
('INSTANCE_1', 'Node_0'),
|
||||||
'Node_0': set(['INSTANCE_0', 'INSTANCE_1']),
|
('INSTANCE_3', 'Node_2'),
|
||||||
'Node_3': set(['INSTANCE_6']),
|
('INSTANCE_4', 'Node_2'),
|
||||||
'Node_2': set(['INSTANCE_4', 'INSTANCE_5',
|
('INSTANCE_5', 'Node_2'),
|
||||||
'INSTANCE_3'])}
|
('INSTANCE_6', 'Node_3'),
|
||||||
self.assertEqual(expected_map, cluster.get_mapping().get_mapping())
|
('INSTANCE_7', 'Node_4'),
|
||||||
|
]
|
||||||
|
edges = [(src.uuid, dst.uuid) for src, dst in model.edges()]
|
||||||
|
self.assertEqual(sorted(expected_edges), sorted(edges))
|
||||||
|
|
||||||
@mock.patch.object(nova_helper.NovaHelper, 'get_aggregate_detail')
|
@mock.patch.object(nova_helper.NovaHelper, 'get_aggregate_detail')
|
||||||
@mock.patch.object(nova_helper.NovaHelper, 'get_aggregate_list')
|
@mock.patch.object(nova_helper.NovaHelper, 'get_aggregate_list')
|
||||||
def test__collect_aggregates(self, mock_aggregate,
|
def test_collect_aggregates(self, mock_aggregate, mock_detailed_aggregate):
|
||||||
mock_detailed_aggregate):
|
|
||||||
allowed_nodes = []
|
allowed_nodes = []
|
||||||
mock_aggregate.return_value = [mock.Mock(id=i) for i in range(2)]
|
mock_aggregate.return_value = [mock.Mock(id=i) for i in range(2)]
|
||||||
mock_detailed_aggregate.side_effect = [
|
mock_detailed_aggregate.side_effect = [
|
||||||
@ -126,12 +129,12 @@ class TestDefaultScope(base.TestCase):
|
|||||||
self.assertEqual(['Node_0', 'Node_1'], allowed_nodes)
|
self.assertEqual(['Node_0', 'Node_1'], allowed_nodes)
|
||||||
|
|
||||||
@mock.patch.object(nova_helper.NovaHelper, 'get_availability_zone_list')
|
@mock.patch.object(nova_helper.NovaHelper, 'get_availability_zone_list')
|
||||||
def test__collect_zones(self, mock_zone_list):
|
def test_collect_zones(self, mock_zone_list):
|
||||||
allowed_nodes = []
|
allowed_nodes = []
|
||||||
mock_zone_list.return_value = [
|
mock_zone_list.return_value = [
|
||||||
mock.Mock(zoneName="AZ{0}".format(i+1),
|
mock.Mock(zoneName="AZ{0}".format(i + 1),
|
||||||
hosts={'Node_{0}'.format(2*i): 1,
|
hosts={'Node_{0}'.format(2 * i): 1,
|
||||||
'Node_{0}'.format(2*i+1): 2})
|
'Node_{0}'.format(2 * i + 1): 2})
|
||||||
for i in range(2)]
|
for i in range(2)]
|
||||||
default.DefaultScope([{'availability_zones': [{'name': "AZ1"}]}],
|
default.DefaultScope([{'availability_zones': [{'name': "AZ1"}]}],
|
||||||
osc=mock.Mock())._collect_zones(
|
osc=mock.Mock())._collect_zones(
|
||||||
@ -142,9 +145,9 @@ class TestDefaultScope(base.TestCase):
|
|||||||
def test_zones_wildcard_is_used(self, mock_zone_list):
|
def test_zones_wildcard_is_used(self, mock_zone_list):
|
||||||
allowed_nodes = []
|
allowed_nodes = []
|
||||||
mock_zone_list.return_value = [
|
mock_zone_list.return_value = [
|
||||||
mock.Mock(zoneName="AZ{0}".format(i+1),
|
mock.Mock(zoneName="AZ{0}".format(i + 1),
|
||||||
hosts={'Node_{0}'.format(2*i): 1,
|
hosts={'Node_{0}'.format(2 * i): 1,
|
||||||
'Node_{0}'.format(2*i+1): 2})
|
'Node_{0}'.format(2 * i + 1): 2})
|
||||||
for i in range(2)]
|
for i in range(2)]
|
||||||
default.DefaultScope([{'availability_zones': [{'name': "*"}]}],
|
default.DefaultScope([{'availability_zones': [{'name': "*"}]}],
|
||||||
osc=mock.Mock())._collect_zones(
|
osc=mock.Mock())._collect_zones(
|
||||||
@ -156,9 +159,9 @@ class TestDefaultScope(base.TestCase):
|
|||||||
def test_zones_wildcard_with_other_ids(self, mock_zone_list):
|
def test_zones_wildcard_with_other_ids(self, mock_zone_list):
|
||||||
allowed_nodes = []
|
allowed_nodes = []
|
||||||
mock_zone_list.return_value = [
|
mock_zone_list.return_value = [
|
||||||
mock.Mock(zoneName="AZ{0}".format(i+1),
|
mock.Mock(zoneName="AZ{0}".format(i + 1),
|
||||||
hosts={'Node_{0}'.format(2*i): 1,
|
hosts={'Node_{0}'.format(2 * i): 1,
|
||||||
'Node_{0}'.format(2*i+1): 2})
|
'Node_{0}'.format(2 * i + 1): 2})
|
||||||
for i in range(2)]
|
for i in range(2)]
|
||||||
scope_handler = default.DefaultScope(
|
scope_handler = default.DefaultScope(
|
||||||
[{'availability_zones': [{'name': "*"}, {'name': 'AZ1'}]}],
|
[{'availability_zones': [{'name': "*"}, {'name': 'AZ1'}]}],
|
||||||
@ -173,38 +176,42 @@ class TestDefaultScope(base.TestCase):
|
|||||||
validators.Draft4Validator(
|
validators.Draft4Validator(
|
||||||
default.DefaultScope.DEFAULT_SCHEMA).validate(test_scope)
|
default.DefaultScope.DEFAULT_SCHEMA).validate(test_scope)
|
||||||
|
|
||||||
def test__exclude_resources(self):
|
def test_exclude_resources(self):
|
||||||
resources_to_exclude = [{'instances': [{'uuid': 'INSTANCE_1'},
|
resources_to_exclude = [{'instances': [{'uuid': 'INSTANCE_1'},
|
||||||
{'uuid': 'INSTANCE_2'}]},
|
{'uuid': 'INSTANCE_2'}]},
|
||||||
{'compute_nodes': [{'name': 'Node_1'},
|
{'compute_nodes': [{'name': 'Node_1'},
|
||||||
{'name': 'Node_2'}]}
|
{'name': 'Node_2'}]}]
|
||||||
]
|
|
||||||
instances_to_exclude = []
|
instances_to_exclude = []
|
||||||
nodes_to_exclude = []
|
nodes_to_exclude = []
|
||||||
default.DefaultScope([], osc=mock.Mock())._exclude_resources(
|
default.DefaultScope([], osc=mock.Mock()).exclude_resources(
|
||||||
resources_to_exclude, instances=instances_to_exclude,
|
resources_to_exclude, instances=instances_to_exclude,
|
||||||
nodes=nodes_to_exclude)
|
nodes=nodes_to_exclude)
|
||||||
self.assertEqual(['Node_1', 'Node_2'], sorted(nodes_to_exclude))
|
self.assertEqual(['Node_1', 'Node_2'], sorted(nodes_to_exclude))
|
||||||
self.assertEqual(['INSTANCE_1', 'INSTANCE_2'],
|
self.assertEqual(['INSTANCE_1', 'INSTANCE_2'],
|
||||||
sorted(instances_to_exclude))
|
sorted(instances_to_exclude))
|
||||||
|
|
||||||
def test__remove_node_from_model(self):
|
def test_remove_nodes_from_model(self):
|
||||||
cluster = self.fake_cluster.generate_scenario_1()
|
model = self.fake_cluster.generate_scenario_1()
|
||||||
default.DefaultScope([], osc=mock.Mock())._remove_node_from_model(
|
default.DefaultScope([], osc=mock.Mock()).remove_nodes_from_model(
|
||||||
['Node_1', 'Node_2'], cluster)
|
['Node_1', 'Node_2'], model)
|
||||||
expected_cluster = {'Node_0': set(['INSTANCE_0', 'INSTANCE_1']),
|
expected_edges = [
|
||||||
'Node_1': set([]), 'Node_2': set([]),
|
('INSTANCE_0', 'Node_0'),
|
||||||
'Node_3': set(['INSTANCE_6']),
|
('INSTANCE_1', 'Node_0'),
|
||||||
'Node_4': set(['INSTANCE_7'])}
|
('INSTANCE_6', 'Node_3'),
|
||||||
self.assertEqual(expected_cluster, cluster.get_mapping().get_mapping())
|
('INSTANCE_7', 'Node_4')]
|
||||||
|
edges = [(src.uuid, dst.uuid) for src, dst in model.edges()]
|
||||||
|
self.assertEqual(sorted(expected_edges), sorted(edges))
|
||||||
|
|
||||||
def test__remove_instances_from_model(self):
|
def test_remove_instances_from_model(self):
|
||||||
cluster = self.fake_cluster.generate_scenario_1()
|
model = self.fake_cluster.generate_scenario_1()
|
||||||
default.DefaultScope([], osc=mock.Mock())._remove_instances_from_model(
|
default.DefaultScope([], osc=mock.Mock()).remove_instances_from_model(
|
||||||
['INSTANCE_1', 'INSTANCE_2'], cluster)
|
['INSTANCE_1', 'INSTANCE_2'], model)
|
||||||
expected_cluster = {'Node_0': set(['INSTANCE_0']), 'Node_1': set([]),
|
expected_edges = [
|
||||||
'Node_2': set(['INSTANCE_3', 'INSTANCE_4',
|
('INSTANCE_0', 'Node_0'),
|
||||||
'INSTANCE_5']),
|
('INSTANCE_3', 'Node_2'),
|
||||||
'Node_3': set(['INSTANCE_6']),
|
('INSTANCE_4', 'Node_2'),
|
||||||
'Node_4': set(['INSTANCE_7'])}
|
('INSTANCE_5', 'Node_2'),
|
||||||
self.assertEqual(expected_cluster, cluster.get_mapping().get_mapping())
|
('INSTANCE_6', 'Node_3'),
|
||||||
|
('INSTANCE_7', 'Node_4')]
|
||||||
|
edges = [(src.uuid, dst.uuid) for src, dst in model.edges()]
|
||||||
|
self.assertEqual(sorted(expected_edges), sorted(edges))
|
||||||
|
@ -130,7 +130,7 @@ class TestUniformAirflow(base.TestCase):
|
|||||||
self.strategy.threshold_inlet_t = 22
|
self.strategy.threshold_inlet_t = 22
|
||||||
n1, n2 = self.strategy.group_hosts_by_airflow()
|
n1, n2 = self.strategy.group_hosts_by_airflow()
|
||||||
instances = model.get_all_instances()
|
instances = model.get_all_instances()
|
||||||
instances.clear()
|
[model.remove_instance(inst) for inst in instances.values()]
|
||||||
instance_to_mig = self.strategy.choose_instance_to_migrate(n1)
|
instance_to_mig = self.strategy.choose_instance_to_migrate(n1)
|
||||||
self.assertIsNone(instance_to_mig)
|
self.assertIsNone(instance_to_mig)
|
||||||
|
|
||||||
|
@ -127,7 +127,8 @@ class TestVMWorkloadConsolidation(base.TestCase):
|
|||||||
n1 = model.get_node_by_uuid('Node_0')
|
n1 = model.get_node_by_uuid('Node_0')
|
||||||
n2 = model.get_node_by_uuid('Node_1')
|
n2 = model.get_node_by_uuid('Node_1')
|
||||||
instance_uuid = 'INSTANCE_0'
|
instance_uuid = 'INSTANCE_0'
|
||||||
self.strategy.add_migration(instance_uuid, n1, n2, model)
|
instance = model.get_instance_by_uuid(instance_uuid)
|
||||||
|
self.strategy.add_migration(instance, n1, n2, model)
|
||||||
self.assertEqual(1, len(self.strategy.solution.actions))
|
self.assertEqual(1, len(self.strategy.solution.actions))
|
||||||
expected = {'action_type': 'migrate',
|
expected = {'action_type': 'migrate',
|
||||||
'input_parameters': {'destination_node': n2.uuid,
|
'input_parameters': {'destination_node': n2.uuid,
|
||||||
@ -196,11 +197,12 @@ class TestVMWorkloadConsolidation(base.TestCase):
|
|||||||
n1 = model.get_node_by_uuid('Node_0')
|
n1 = model.get_node_by_uuid('Node_0')
|
||||||
n2 = model.get_node_by_uuid('Node_1')
|
n2 = model.get_node_by_uuid('Node_1')
|
||||||
instance_uuid = 'INSTANCE_0'
|
instance_uuid = 'INSTANCE_0'
|
||||||
|
instance = model.get_instance_by_uuid(instance_uuid)
|
||||||
self.strategy.disable_unused_nodes(model)
|
self.strategy.disable_unused_nodes(model)
|
||||||
self.assertEqual(0, len(self.strategy.solution.actions))
|
self.assertEqual(0, len(self.strategy.solution.actions))
|
||||||
|
|
||||||
# Migrate VM to free the node
|
# Migrate VM to free the node
|
||||||
self.strategy.add_migration(instance_uuid, n1, n2, model)
|
self.strategy.add_migration(instance, n1, n2, model)
|
||||||
|
|
||||||
self.strategy.disable_unused_nodes(model)
|
self.strategy.disable_unused_nodes(model)
|
||||||
expected = {'action_type': 'change_nova_service_state',
|
expected = {'action_type': 'change_nova_service_state',
|
||||||
|
@ -74,7 +74,7 @@ class TestWorkloadBalance(base.TestCase):
|
|||||||
model = self.fake_cluster.generate_scenario_6_with_2_nodes()
|
model = self.fake_cluster.generate_scenario_6_with_2_nodes()
|
||||||
self.m_model.return_value = model
|
self.m_model.return_value = model
|
||||||
node = model.get_node_by_uuid('Node_0')
|
node = model.get_node_by_uuid('Node_0')
|
||||||
cap_cores = model.get_resource_by_uuid(element.ResourceType.cpu_cores)
|
cap_cores = model.get_resource_by_uuid(element.ResourceType.vcpus)
|
||||||
cap_mem = model.get_resource_by_uuid(element.ResourceType.memory)
|
cap_mem = model.get_resource_by_uuid(element.ResourceType.memory)
|
||||||
cap_disk = model.get_resource_by_uuid(element.ResourceType.disk)
|
cap_disk = model.get_resource_by_uuid(element.ResourceType.disk)
|
||||||
cores_used, mem_used, disk_used = (
|
cores_used, mem_used, disk_used = (
|
||||||
@ -107,7 +107,7 @@ class TestWorkloadBalance(base.TestCase):
|
|||||||
self.m_model.return_value = model
|
self.m_model.return_value = model
|
||||||
n1, n2, avg, w_map = self.strategy.group_hosts_by_cpu_util()
|
n1, n2, avg, w_map = self.strategy.group_hosts_by_cpu_util()
|
||||||
instances = model.get_all_instances()
|
instances = model.get_all_instances()
|
||||||
instances.clear()
|
[model.remove_instance(inst) for inst in instances.values()]
|
||||||
instance_to_mig = self.strategy.choose_instance_to_migrate(
|
instance_to_mig = self.strategy.choose_instance_to_migrate(
|
||||||
n1, avg, w_map)
|
n1, avg, w_map)
|
||||||
self.assertIsNone(instance_to_mig)
|
self.assertIsNone(instance_to_mig)
|
||||||
|
Loading…
Reference in New Issue
Block a user