Merge "Add support VNFM auto heal and scale"

This commit is contained in:
Zuul 2023-03-03 00:20:12 +00:00 committed by Gerrit Code Review
commit 497586dffc
24 changed files with 1570 additions and 319 deletions

View File

@ -299,6 +299,9 @@
$TACKER_CONF: $TACKER_CONF:
server_notification: server_notification:
server_notification: true server_notification: true
prometheus_plugin:
auto_scaling: true
auto_healing: true
tox_envlist: dsvm-functional-sol-v2 tox_envlist: dsvm-functional-sol-v2
- job: - job:
@ -608,6 +611,7 @@
fault_management: true fault_management: true
performance_management: true performance_management: true
auto_scaling: true auto_scaling: true
auto_healing: true
test_rule_with_promtool: true test_rule_with_promtool: true
tox_envlist: dsvm-functional-sol-kubernetes-v2 tox_envlist: dsvm-functional-sol-kubernetes-v2
vars: vars:

View File

@ -8,7 +8,8 @@ use = egg:Paste#urlmap
/vnflcm/v2: vnflcm_v2 /vnflcm/v2: vnflcm_v2
/vnffm/v1: vnffm_v1 /vnffm/v1: vnffm_v1
/vnfpm/v2: vnfpm_v2 /vnfpm/v2: vnfpm_v2
/alert/vnf_instances: prometheus_auto_scaling /alert/auto_scaling: prometheus_auto_scaling
/alert/auto_healing: prometheus_auto_healing
/alert: prometheus_fm /alert: prometheus_fm
/pm_event: prometheus_pm /pm_event: prometheus_pm
/server_notification: server_notification /server_notification: server_notification
@ -93,6 +94,9 @@ paste.app_factory = tacker.sol_refactored.api.router:VnffmAPIRouterV1.factory
[app:prometheus_auto_scaling] [app:prometheus_auto_scaling]
paste.app_factory = tacker.sol_refactored.api.prometheus_plugin_router:AutoScalingRouter.factory paste.app_factory = tacker.sol_refactored.api.prometheus_plugin_router:AutoScalingRouter.factory
[app:prometheus_auto_healing]
paste.app_factory = tacker.sol_refactored.api.prometheus_plugin_router:AutoHealingRouter.factory
[app:prometheus_fm] [app:prometheus_fm]
paste.app_factory = tacker.sol_refactored.api.prometheus_plugin_router:FmAlertRouter.factory paste.app_factory = tacker.sol_refactored.api.prometheus_plugin_router:FmAlertRouter.factory

View File

@ -0,0 +1,9 @@
---
features:
- |
In Prometheus Plugin, two new interfaces are added to implement AutoHeal
and AutoScale for VNF and CNF.
As a VNFM, Tacker can decide whether to perform heal and scale operations
through the received alarms sent by External Monitoring Tools, without
NFVO.
Added new user guide to help users understand the function.

View File

@ -27,7 +27,8 @@ REPORT_GET = '/vnfpm/v2/pm_jobs/{id}/reports/{report_id}'
POLICY_NAME_PROM_PLUGIN = 'tacker_PROM_PLUGIN_api:PROM_PLUGIN:{}' POLICY_NAME_PROM_PLUGIN = 'tacker_PROM_PLUGIN_api:PROM_PLUGIN:{}'
PROM_PLUGIN_PM_PATH = '/pm_event' PROM_PLUGIN_PM_PATH = '/pm_event'
PROM_PLUGIN_AUTO_SCALING_PATH = '/alert/vnf_instances' PROM_PLUGIN_AUTO_HEALING_PATH = '/alert/auto_healing'
PROM_PLUGIN_AUTO_SCALING_PATH = '/alert/auto_scaling'
rules = [ rules = [
policy.DocumentedRuleDefault( policy.DocumentedRuleDefault(
@ -107,6 +108,15 @@ rules = [
'path': PROM_PLUGIN_PM_PATH} 'path': PROM_PLUGIN_PM_PATH}
] ]
), ),
policy.DocumentedRuleDefault(
name=POLICY_NAME_PROM_PLUGIN.format('auto_healing'),
check_str=RULE_ANY,
description="auto_healing",
operations=[
{'method': 'POST',
'path': PROM_PLUGIN_AUTO_HEALING_PATH}
]
),
policy.DocumentedRuleDefault( policy.DocumentedRuleDefault(
name=POLICY_NAME_PROM_PLUGIN.format('auto_scaling'), name=POLICY_NAME_PROM_PLUGIN.format('auto_scaling'),
check_str=RULE_ANY, check_str=RULE_ANY,
@ -115,15 +125,6 @@ rules = [
{'method': 'POST', {'method': 'POST',
'path': PROM_PLUGIN_AUTO_SCALING_PATH} 'path': PROM_PLUGIN_AUTO_SCALING_PATH}
] ]
),
policy.DocumentedRuleDefault(
name=POLICY_NAME_PROM_PLUGIN.format('auto_scaling_id'),
check_str=RULE_ANY,
description="auto_scaling_id",
operations=[
{'method': 'POST',
'path': PROM_PLUGIN_AUTO_SCALING_PATH + '/{vnfInstanceId}'}
]
) )
] ]

View File

@ -33,11 +33,15 @@ class FmAlertRouter(prom_wsgi.PrometheusPluginAPIRouter):
route_list = [("", {"POST": "alert"})] route_list = [("", {"POST": "alert"})]
class AutoHealingRouter(prom_wsgi.PrometheusPluginAPIRouter):
controller = prom_wsgi.PrometheusPluginResource(
prometheus_plugin_controller.AutoHealingController(),
policy_name=vnfpm_policy_v2.POLICY_NAME_PROM_PLUGIN)
route_list = [("", {"POST": "auto_healing"})]
class AutoScalingRouter(prom_wsgi.PrometheusPluginAPIRouter): class AutoScalingRouter(prom_wsgi.PrometheusPluginAPIRouter):
controller = prom_wsgi.PrometheusPluginResource( controller = prom_wsgi.PrometheusPluginResource(
prometheus_plugin_controller.AutoScalingController(), prometheus_plugin_controller.AutoScalingController(),
policy_name=vnfpm_policy_v2.POLICY_NAME_PROM_PLUGIN) policy_name=vnfpm_policy_v2.POLICY_NAME_PROM_PLUGIN)
route_list = [ route_list = [("", {"POST": "auto_scaling"})]
("", {"POST": "auto_scaling"}),
("/{id}", {"POST": "auto_scaling"})
]

View File

@ -34,11 +34,12 @@ Alert = {
}, },
'function_type': { 'function_type': {
'type': 'string', 'type': 'string',
'enum': ['vnffm', 'vnfpm', 'auto_scale'] 'enum': ['vnffm', 'vnfpm', 'auto_scale', 'auto_heal']
}, },
'job_id': {'type': 'string'}, 'job_id': {'type': 'string'},
'object_instance_id': {'type': 'string'}, 'object_instance_id': {'type': 'string'},
'vnf_instance_id': {'type': 'string'}, 'vnf_instance_id': {'type': 'string'},
'vnfc_info_id': {'type': 'string'},
'node': {'type': 'string'}, 'node': {'type': 'string'},
'perceived_severity': { 'perceived_severity': {
'type': 'string', 'type': 'string',

View File

@ -198,6 +198,9 @@ PROMETHEUS_PLUGIN_OPTS = [
cfg.BoolOpt('fault_management', cfg.BoolOpt('fault_management',
default=False, default=False,
help=_('Enable prometheus plugin fault management')), help=_('Enable prometheus plugin fault management')),
cfg.BoolOpt('auto_healing',
default=False,
help=_('Enable prometheus plugin autohealing')),
cfg.BoolOpt('auto_scaling', cfg.BoolOpt('auto_scaling',
default=False, default=False,
help=_('Enable prometheus plugin autoscaling')), help=_('Enable prometheus plugin autoscaling')),
@ -225,6 +228,22 @@ PROMETHEUS_PLUGIN_OPTS = [
'This configuration is changed in case of replacing ' 'This configuration is changed in case of replacing '
'the original function with a vendor specific ' 'the original function with a vendor specific '
'function.')), 'function.')),
cfg.StrOpt('auto_healing_package',
default='tacker.sol_refactored.common.prometheus_plugin',
help=_('Package name for auto healing. '
'This configuration is changed in case of replacing '
'the original function with a vendor specific '
'function.')),
cfg.StrOpt('auto_healing_class',
default='PrometheusPluginAutoHealing',
help=_('Class name for auto healing. '
'This configuration is changed in case of replacing '
'the original function with a vendor specific '
'function.')),
cfg.IntOpt('timer_interval',
default=20,
help=_('Timeout (second) of packing for multiple '
'auto healing.')),
cfg.StrOpt('auto_scaling_package', cfg.StrOpt('auto_scaling_package',
default='tacker.sol_refactored.common.prometheus_plugin', default='tacker.sol_refactored.common.prometheus_plugin',
help=_('Package name for auto scaling. ' help=_('Package name for auto scaling. '

View File

@ -370,7 +370,7 @@ class PrometheusPluginPm(PrometheusPluginPmBase, mon_base.MonitoringPlugin):
self._alert(kwargs['request'], body=kwargs['body']) self._alert(kwargs['request'], body=kwargs['body'])
except Exception as e: except Exception as e:
# All exceptions is ignored here and 204 response will always # All exceptions is ignored here and 204 response will always
# be returned. Because when tacker responds error to alertmanager, # be returned because when tacker responds error to alertmanager,
# alertmanager may repeat the same reports. # alertmanager may repeat the same reports.
LOG.error("%s: %s", e.__class__.__name__, e.args[0]) LOG.error("%s: %s", e.__class__.__name__, e.args[0])
@ -451,16 +451,16 @@ class PrometheusPluginPm(PrometheusPluginPmBase, mon_base.MonitoringPlugin):
result = [] result = []
context = request.context context = request.context
datetime_now = datetime.datetime.now(datetime.timezone.utc) datetime_now = datetime.datetime.now(datetime.timezone.utc)
for alt in body['alerts']: for alert in body['alerts']:
if alt['labels']['function_type'] != 'vnfpm': if alert['labels']['function_type'] != 'vnfpm':
continue continue
try: try:
pm_job_id = alt['labels']['job_id'] pm_job_id = alert['labels']['job_id']
object_instance_id = alt['labels']['object_instance_id'] object_instance_id = alert['labels']['object_instance_id']
metric = alt['labels']['metric'] metric = alert['labels']['metric']
sub_object_instance_id = alt['labels'].get( sub_object_instance_id = alert['labels'].get(
'sub_object_instance_id') 'sub_object_instance_id')
value = alt['annotations']['value'] value = alert['annotations']['value']
pm_job = pm_job_utils.get_pm_job(context, pm_job_id) pm_job = pm_job_utils.get_pm_job(context, pm_job_id)
self.filter_alert_by_time(context, pm_job, datetime_now, self.filter_alert_by_time(context, pm_job, datetime_now,
@ -692,7 +692,7 @@ class PrometheusPluginFm(PrometheusPlugin, mon_base.MonitoringPlugin):
self._alert(kwargs['request'], body=kwargs['body']) self._alert(kwargs['request'], body=kwargs['body'])
except Exception as e: except Exception as e:
# All exceptions is ignored here and 204 response will always # All exceptions is ignored here and 204 response will always
# be returned. Because when tacker responds error to alertmanager, # be returned because when tacker responds error to alertmanager,
# alertmanager may repeat the same reports. # alertmanager may repeat the same reports.
LOG.error("%s: %s", e.__class__.__name__, e.args[0]) LOG.error("%s: %s", e.__class__.__name__, e.args[0])
@ -823,18 +823,91 @@ class PrometheusPluginFm(PrometheusPlugin, mon_base.MonitoringPlugin):
def _alert(self, request, body): def _alert(self, request, body):
now = datetime.datetime.now(datetime.timezone.utc) now = datetime.datetime.now(datetime.timezone.utc)
result = [] result = []
for alt in body['alerts']: for alert in body['alerts']:
if alt['labels']['function_type'] != 'vnffm': if alert['labels']['function_type'] != 'vnffm':
continue continue
try: try:
alarms = self.create_or_update_alarm( alarms = self.create_or_update_alarm(
request.context, alt, now) request.context, alert, now)
result.extend(alarms) result.extend(alarms)
except sol_ex.PrometheusPluginSkipped: except sol_ex.PrometheusPluginSkipped:
pass pass
return result return result
class PrometheusPluginAutoHealing(PrometheusPlugin, mon_base.MonitoringPlugin):
_instance = None
@staticmethod
def instance():
if PrometheusPluginAutoHealing._instance is None:
if not CONF.prometheus_plugin.auto_healing:
stub = mon_base.MonitoringPluginStub.instance()
PrometheusPluginAutoHealing._instance = stub
else:
PrometheusPluginAutoHealing()
return PrometheusPluginAutoHealing._instance
def __init__(self):
if PrometheusPluginAutoHealing._instance:
raise SystemError(
"Not constructor but instance() should be used.")
super(PrometheusPluginAutoHealing, self).__init__()
self.set_callback(self.default_callback)
PrometheusPluginAutoHealing._instance = self
def set_callback(self, notification_callback):
self.notification_callback = notification_callback
def alert(self, **kwargs):
try:
self._alert(kwargs['request'], body=kwargs['body'])
except Exception as e:
# All exceptions is ignored here and 204 response will always
# be returned because when tacker responds error to alertmanager,
# alertmanager may repeat the same reports.
LOG.error("%s: %s", e.__class__.__name__, e.args[0])
def default_callback(self, context, vnf_instance_id, vnfc_info_id):
self.rpc.enqueue_auto_heal_instance(
context, vnf_instance_id, vnfc_info_id)
@validator.schema(prometheus_plugin_schemas.AlertMessage)
def _alert(self, request, body):
context = request.context
alerts = (alert for alert in body['alerts'] if
alert['status'] == 'firing' and
alert['labels']['receiver_type'] == 'tacker' and
alert['labels']['function_type'] == 'auto_heal')
for alert in alerts:
vnf_instance_id = alert['labels']['vnf_instance_id']
try:
inst = inst_utils.get_inst(context, vnf_instance_id)
except sol_ex.VnfInstanceNotFound:
continue
if inst.instantiationState != 'INSTANTIATED':
self.rpc.dequeue_auto_heal_instance(
None, vnf_instance_id)
if (not inst.obj_attr_is_set('vnfConfigurableProperties')
or not inst.vnfConfigurableProperties.get(
'isAutohealEnabled')):
continue
vnfc_info_id = alert['labels']['vnfc_info_id']
result = {
vnfcInfo for vnfcInfo in inst.instantiatedVnfInfo.vnfcInfo
if vnfcInfo.id == vnfc_info_id
}
if not result:
continue
if self.notification_callback:
self.notification_callback(
context, vnf_instance_id, vnfc_info_id)
class PrometheusPluginAutoScaling(PrometheusPlugin, mon_base.MonitoringPlugin): class PrometheusPluginAutoScaling(PrometheusPlugin, mon_base.MonitoringPlugin):
_instance = None _instance = None
@ -853,7 +926,7 @@ class PrometheusPluginAutoScaling(PrometheusPlugin, mon_base.MonitoringPlugin):
raise SystemError( raise SystemError(
"Not constructor but instance() should be used.") "Not constructor but instance() should be used.")
super(PrometheusPluginAutoScaling, self).__init__() super(PrometheusPluginAutoScaling, self).__init__()
self.notification_callback = self.default_callback self.set_callback(self.default_callback)
PrometheusPluginAutoScaling._instance = self PrometheusPluginAutoScaling._instance = self
def set_callback(self, notification_callback): def set_callback(self, notification_callback):
@ -864,46 +937,46 @@ class PrometheusPluginAutoScaling(PrometheusPlugin, mon_base.MonitoringPlugin):
self._alert(kwargs['request'], body=kwargs['body']) self._alert(kwargs['request'], body=kwargs['body'])
except Exception as e: except Exception as e:
# All exceptions is ignored here and 204 response will always # All exceptions is ignored here and 204 response will always
# be returned. Because when tacker responds error to alertmanager, # be returned because when tacker responds error to alertmanager,
# alertmanager may repeat the same reports. # alertmanager may repeat the same reports.
LOG.error("%s: %s", e.__class__.__name__, e.args[0]) LOG.error("%s: %s", e.__class__.__name__, e.args[0])
def default_callback(self, context, vnf_instance_id, scaling_param): def default_callback(self, context, vnf_instance_id, scaling_param):
self.rpc.request_scale(context, vnf_instance_id, scaling_param) self.rpc.trigger_scale(context, vnf_instance_id, scaling_param)
def skip_if_auto_scale_not_enabled(self, vnf_instance):
if (not vnf_instance.obj_attr_is_set('vnfConfigurableProperties') or
not vnf_instance.vnfConfigurableProperties.get(
'isAutoscaleEnabled')):
raise sol_ex.PrometheusPluginSkipped()
def process_auto_scale(self, request, vnf_instance_id, auto_scale_type,
aspect_id):
scaling_param = {
'type': auto_scale_type,
'aspectId': aspect_id,
}
context = request.context
if self.notification_callback:
self.notification_callback(context, vnf_instance_id, scaling_param)
@validator.schema(prometheus_plugin_schemas.AlertMessage) @validator.schema(prometheus_plugin_schemas.AlertMessage)
def _alert(self, request, body): def _alert(self, request, body):
result = []
for alt in body['alerts']:
if alt['labels']['function_type'] != 'auto_scale':
continue
try:
vnf_instance_id = alt['labels']['vnf_instance_id']
auto_scale_type = alt['labels']['auto_scale_type']
aspect_id = alt['labels']['aspect_id']
context = request.context context = request.context
alerts = (alert for alert in body['alerts'] if
alert['status'] == 'firing' and
alert['labels']['receiver_type'] == 'tacker' and
alert['labels']['function_type'] == 'auto_scale')
for alert in alerts:
vnf_instance_id = alert['labels']['vnf_instance_id']
try:
inst = inst_utils.get_inst(context, vnf_instance_id) inst = inst_utils.get_inst(context, vnf_instance_id)
self.skip_if_auto_scale_not_enabled(inst) except sol_ex.VnfInstanceNotFound:
self.process_auto_scale( continue
request, vnf_instance_id, auto_scale_type, aspect_id) if (inst.instantiationState != 'INSTANTIATED' or
result.append((vnf_instance_id, auto_scale_type, aspect_id)) not inst.obj_attr_is_set('vnfConfigurableProperties') or
except sol_ex.PrometheusPluginSkipped: not inst.vnfConfigurableProperties.get(
pass 'isAutoscaleEnabled')):
return result continue
aspect_id = alert['labels']['aspect_id']
result = {
scaleStatus for scaleStatus in
inst.instantiatedVnfInfo.scaleStatus
if scaleStatus.aspectId == aspect_id
}
if not result:
continue
scaling_param = {
'type': alert['labels']['auto_scale_type'],
'aspectId': aspect_id,
}
if self.notification_callback:
self.notification_callback(
context, vnf_instance_id, scaling_param)

View File

@ -0,0 +1,142 @@
# Copyright (C) 2023 Fujitsu
# All Rights Reserved.
#
# Licensed under the Apache License, Version 2.0 (the "License"); you may
# not use this file except in compliance with the License. You may obtain
# a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
# License for the specific language governing permissions and limitations
# under the License.
from datetime import datetime
from oslo_utils import uuidutils
from tacker.sol_refactored.common import coordinate
from tacker.sol_refactored.common import exceptions as sol_ex
from tacker.sol_refactored.common import lcm_op_occ_utils as lcmocc_utils
from tacker.sol_refactored.common import vnf_instance_utils as inst_utils
from tacker.sol_refactored.conductor import conductor_rpc_v2
from tacker.sol_refactored import objects
from tacker.sol_refactored.objects.v2 import fields as v2fields
# TODO(fengyi): The code of this function is all copied from the
# controller, which is not friendly to future development and
# maintenance, and may be refactored in the future. After
# refactoring, only the validation of the req body is left,
# and the creation of lcmocc and the call to start_lcm_op are
# all executed by the controller, notification driver, etc.
@coordinate.lock_vnf_instance('{vnf_instance_id}')
def heal(context, vnf_instance_id, body):
inst = inst_utils.get_inst(context, vnf_instance_id)
if inst.instantiationState != 'INSTANTIATED':
raise sol_ex.VnfInstanceIsNotInstantiated(inst_id=vnf_instance_id)
lcmocc_utils.check_lcmocc_in_progress(context, vnf_instance_id)
# check parameter for later use
is_all = body.get('additionalParams', {}).get('all', False)
if not isinstance(is_all, bool):
raise sol_ex.SolValidationError(
detail="additionalParams['all'] must be bool.")
if 'vnfcInstanceId' in body:
inst_info = inst.instantiatedVnfInfo
vnfc_id = []
if inst_info.obj_attr_is_set('vnfcInfo'):
vnfc_id = [vnfc.id for vnfc in inst_info.vnfcInfo]
for req_vnfc_id in body['vnfcInstanceId']:
if req_vnfc_id not in vnfc_id:
raise sol_ex.SolValidationError(
detail="vnfcInstanceId(%s) does not exist."
% req_vnfc_id)
lcmocc = new_lcmocc(vnf_instance_id, v2fields.LcmOperationType.HEAL, body)
lcmocc.create(context)
rpc = conductor_rpc_v2.VnfLcmRpcApiV2()
rpc.start_lcm_op(context, lcmocc.id)
return lcmocc
# TODO(fengyi): The code of this function is all copied from the
# controller, which is not friendly to future development and
# maintenance, and may be refactored in the future. After
# refactoring, only the validation of the req body is left,
# and the creation of lcmocc and the call to start_lcm_op are
# all executed by the controller, notification driver, etc.
@coordinate.lock_vnf_instance('{vnf_instance_id}')
def scale(context, vnf_instance_id, body):
inst = inst_utils.get_inst(context, vnf_instance_id)
if inst.instantiationState != 'INSTANTIATED':
raise sol_ex.VnfInstanceIsNotInstantiated(inst_id=vnf_instance_id)
lcmocc_utils.check_lcmocc_in_progress(context, vnf_instance_id)
# check parameters
aspect_id = body['aspectId']
if 'numberOfSteps' not in body:
# set default value (1) defined by SOL specification for
# the convenience of the following methods.
body['numberOfSteps'] = 1
scale_level = _get_current_scale_level(inst, aspect_id)
max_scale_level = _get_max_scale_level(inst, aspect_id)
if scale_level is None or max_scale_level is None:
raise sol_ex.InvalidScaleAspectId(aspect_id=aspect_id)
num_steps = body['numberOfSteps']
if body['type'] == 'SCALE_IN':
num_steps *= -1
scale_level += num_steps
if scale_level < 0 or scale_level > max_scale_level:
raise sol_ex.InvalidScaleNumberOfSteps(
num_steps=body['numberOfSteps'])
lcmocc = new_lcmocc(vnf_instance_id, v2fields.LcmOperationType.SCALE, body)
lcmocc.create(context)
rpc = conductor_rpc_v2.VnfLcmRpcApiV2()
rpc.start_lcm_op(context, lcmocc.id)
return lcmocc
def _get_current_scale_level(inst, aspect_id):
if (inst.obj_attr_is_set('instantiatedVnfInfo') and
inst.instantiatedVnfInfo.obj_attr_is_set('scaleStatus')):
for scale_info in inst.instantiatedVnfInfo.scaleStatus:
if scale_info.aspectId == aspect_id:
return scale_info.scaleLevel
def _get_max_scale_level(inst, aspect_id):
if (inst.obj_attr_is_set('instantiatedVnfInfo') and
inst.instantiatedVnfInfo.obj_attr_is_set('maxScaleLevels')):
for scale_info in inst.instantiatedVnfInfo.maxScaleLevels:
if scale_info.aspectId == aspect_id:
return scale_info.scaleLevel
def new_lcmocc(inst_id, operation, req_body,
op_state=v2fields.LcmOperationStateType.STARTING):
now = datetime.utcnow()
lcmocc = objects.VnfLcmOpOccV2(
id=uuidutils.generate_uuid(),
operationState=op_state,
stateEnteredTime=now,
startTime=now,
vnfInstanceId=inst_id,
operation=operation,
isAutomaticInvocation=False,
isCancelPending=False,
operationParams=req_body)
return lcmocc

View File

@ -95,5 +95,15 @@ class PrometheusPluginConductor(object):
def store_job_info(self, context, report): def store_job_info(self, context, report):
self.cast(context, 'store_job_info', report=report) self.cast(context, 'store_job_info', report=report)
def request_scale(self, context, id, scale_req): def trigger_scale(self, context, id, scale_req):
self.cast(context, 'request_scale', id=id, scale_req=scale_req) self.cast(context, 'trigger_scale', id=id, scale_req=scale_req)
def enqueue_auto_heal_instance(
self, context, vnf_instance_id, vnfc_info_id):
self.cast(context, 'enqueue_auto_heal_instance',
vnf_instance_id=vnf_instance_id,
vnfc_info_id=vnfc_info_id)
def dequeue_auto_heal_instance(self, context, vnf_instance_id):
self.cast(context, 'dequeue_auto_heal_instance',
vnf_instance_id=vnf_instance_id)

View File

@ -394,8 +394,17 @@ class ConductorV2(object):
self.vnfpm_driver.store_job_info(context, report) self.vnfpm_driver.store_job_info(context, report)
@log.log @log.log
def request_scale(self, context, id, scale_req): def trigger_scale(self, context, id, scale_req):
self.prom_driver.request_scale(context, id, scale_req) self.prom_driver.trigger_scale(context, id, scale_req)
@log.log
def enqueue_auto_heal_instance(
self, context, vnf_instance_id, vnfc_info_id):
self.prom_driver.enqueue_heal(context, vnf_instance_id, vnfc_info_id)
@log.log
def dequeue_auto_heal_instance(self, context, vnf_instance_id):
self.prom_driver.dequeue_heal(vnf_instance_id)
@log.log @log.log
def server_notification_notify( def server_notification_notify(

View File

@ -13,17 +13,66 @@
# License for the specific language governing permissions and limitations # License for the specific language governing permissions and limitations
# under the License. # under the License.
import threading
from oslo_log import log as logging from oslo_log import log as logging
from tacker.sol_refactored.common import config as cfg from tacker.sol_refactored.common import config as cfg
from tacker.sol_refactored.common import http_client from tacker.sol_refactored.common import vnflcm_utils
from tacker.sol_refactored import objects
LOG = logging.getLogger(__name__) LOG = logging.getLogger(__name__)
CONF = cfg.CONF CONF = cfg.CONF
class VnfmAutoHealTimer():
def __init__(self, context, vnf_instance_id,
expiration_time, expiration_handler):
self.lock = threading.Lock()
self.expired = False
self.queue = set()
self.context = context
self.vnf_instance_id = vnf_instance_id
self.expiration_handler = expiration_handler
self.timer = threading.Timer(expiration_time, self.expire)
self.timer.start()
def expire(self):
_expired = False
with self.lock:
if not self.expired:
self._cancel()
_expired = True
if _expired:
self.expiration_handler(
self.context, self.vnf_instance_id, list(self.queue))
def add_vnfc_info_id(self, vnfc_info_id):
with self.lock:
if not self.expired:
self.queue.add(vnfc_info_id)
def _cancel(self):
self.timer.cancel()
self.expired = True
def cancel(self):
with self.lock:
if not self.expired:
self._cancel()
def __del__(self):
self.cancel()
class PrometheusPluginDriverStub(): class PrometheusPluginDriverStub():
def request_scale(self, context, vnf_instance_id, scale_req): def trigger_scale(self, context, vnf_instance_id, scale_req):
pass
def enqueue_heal(self, context, vnf_instance_id, vnfc_info_id):
pass
def dequeue_heal(self, vnf_instance_id):
pass pass
@ -34,6 +83,7 @@ class PrometheusPluginDriver():
def instance(): def instance():
if PrometheusPluginDriver._instance is None: if PrometheusPluginDriver._instance is None:
if (CONF.prometheus_plugin.auto_scaling or if (CONF.prometheus_plugin.auto_scaling or
CONF.prometheus_plugin.auto_healing or
CONF.prometheus_plugin.fault_management or CONF.prometheus_plugin.fault_management or
CONF.prometheus_plugin.performance_management): CONF.prometheus_plugin.performance_management):
PrometheusPluginDriver() PrometheusPluginDriver()
@ -45,19 +95,35 @@ class PrometheusPluginDriver():
def __init__(self): def __init__(self):
if PrometheusPluginDriver._instance: if PrometheusPluginDriver._instance:
raise SystemError("Not constructor but instance() should be used.") raise SystemError("Not constructor but instance() should be used.")
auth_handle = http_client.KeystonePasswordAuthHandle(
auth_url=CONF.keystone_authtoken.auth_url,
username=CONF.keystone_authtoken.username,
password=CONF.keystone_authtoken.password,
project_name=CONF.keystone_authtoken.project_name,
user_domain_name=CONF.keystone_authtoken.user_domain_name,
project_domain_name=CONF.keystone_authtoken.project_domain_name)
self.client = http_client.HttpClient(auth_handle)
PrometheusPluginDriver._instance = self PrometheusPluginDriver._instance = self
self.timer_map = {}
self.expiration_time = CONF.prometheus_plugin.timer_interval
def request_scale(self, context, vnf_instance_id, scale_req): def enqueue_heal(self, context, vnf_instance_id, vnfc_info_id):
ep = CONF.v2_vnfm.endpoint if vnf_instance_id not in self.timer_map:
url = f'{ep}/vnflcm/v2/vnf_instances/{vnf_instance_id}/scale' self.timer_map[vnf_instance_id] = VnfmAutoHealTimer(
resp, _ = self.client.do_request( context, vnf_instance_id, self.expiration_time,
url, "POST", context=context, body=scale_req, version="2.0.0") self._timer_expired)
LOG.info("AutoScaling request is processed: %d.", resp.status_code) self.timer_map[vnf_instance_id].add_vnfc_info_id(vnfc_info_id)
def dequeue_heal(self, vnf_instance_id):
if vnf_instance_id in self.timer_map:
self.timer_map[vnf_instance_id].cancel()
del self.timer_map[vnf_instance_id]
def _trigger_heal(self, context, vnf_instance_id, vnfc_info_ids):
heal_req = objects.HealVnfRequest(vnfcInstanceId=vnfc_info_ids)
body = heal_req.to_dict()
LOG.info(f"VNFM AutoHealing is triggered. vnf: {vnf_instance_id}, "
f"vnfcInstanceId: {vnfc_info_ids}")
vnflcm_utils.heal(context, vnf_instance_id, body)
def _timer_expired(self, context, vnf_instance_id, vnfc_info_ids):
self.dequeue_heal(vnf_instance_id)
self._trigger_heal(context, vnf_instance_id, vnfc_info_ids)
def trigger_scale(self, context, vnf_instance_id, scale_req):
LOG.info(f"VNFM AutoScaling is triggered. vnf: {vnf_instance_id}, "
f"type: {scale_req['type']}, aspectId: "
f"{scale_req['aspectId']}")
vnflcm_utils.scale(context, vnf_instance_id, scale_req)

View File

@ -47,6 +47,19 @@ class FmAlertController(prom_wsgi.PrometheusPluginAPIController):
return prom_wsgi.PrometheusPluginResponse(204, None) return prom_wsgi.PrometheusPluginResponse(204, None)
class AutoHealingController(prom_wsgi.PrometheusPluginAPIController):
def auto_healing(self, request, body):
if not CONF.prometheus_plugin.auto_healing:
raise sol_ex.PrometheusPluginNotEnabled(
name='Auto healing')
cls = mon_base.get_class(
CONF.prometheus_plugin.auto_healing_package,
CONF.prometheus_plugin.auto_healing_class)
mon_base.MonitoringPlugin.get_instance(cls).alert(
request=request, body=body)
return prom_wsgi.PrometheusPluginResponse(204, None)
class AutoScalingController(prom_wsgi.PrometheusPluginAPIController): class AutoScalingController(prom_wsgi.PrometheusPluginAPIController):
def auto_scaling(self, request, body): def auto_scaling(self, request, body):
if not CONF.prometheus_plugin.auto_scaling: if not CONF.prometheus_plugin.auto_scaling:
@ -58,6 +71,3 @@ class AutoScalingController(prom_wsgi.PrometheusPluginAPIController):
mon_base.MonitoringPlugin.get_instance(cls).alert( mon_base.MonitoringPlugin.get_instance(cls).alert(
request=request, body=body) request=request, body=body)
return prom_wsgi.PrometheusPluginResponse(204, None) return prom_wsgi.PrometheusPluginResponse(204, None)
def auto_scaling_id(self, request, _, body):
return self.auto_scaling(request, body)

View File

@ -14,8 +14,6 @@
# under the License. # under the License.
from datetime import datetime
from oslo_log import log as logging from oslo_log import log as logging
from oslo_utils import uuidutils from oslo_utils import uuidutils
@ -30,6 +28,7 @@ from tacker.sol_refactored.common import lcm_op_occ_utils as lcmocc_utils
from tacker.sol_refactored.common import subscription_utils as subsc_utils from tacker.sol_refactored.common import subscription_utils as subsc_utils
from tacker.sol_refactored.common import vim_utils from tacker.sol_refactored.common import vim_utils
from tacker.sol_refactored.common import vnf_instance_utils as inst_utils from tacker.sol_refactored.common import vnf_instance_utils as inst_utils
from tacker.sol_refactored.common import vnflcm_utils
from tacker.sol_refactored.conductor import conductor_rpc_v2 from tacker.sol_refactored.conductor import conductor_rpc_v2
from tacker.sol_refactored.controller import vnflcm_view from tacker.sol_refactored.controller import vnflcm_view
from tacker.sol_refactored.nfvo import nfvo_client from tacker.sol_refactored.nfvo import nfvo_client
@ -153,22 +152,6 @@ class VnfLcmControllerV2(sol_wsgi.SolAPIController):
self.endpoint) self.endpoint)
return sol_wsgi.SolResponse(204, None) return sol_wsgi.SolResponse(204, None)
def _new_lcmocc(self, inst_id, operation, req_body,
op_state=v2fields.LcmOperationStateType.STARTING):
now = datetime.utcnow()
lcmocc = objects.VnfLcmOpOccV2(
id=uuidutils.generate_uuid(),
operationState=op_state,
stateEnteredTime=now,
startTime=now,
vnfInstanceId=inst_id,
operation=operation,
isAutomaticInvocation=False,
isCancelPending=False,
operationParams=req_body)
return lcmocc
@validator.schema(schema.VnfInfoModificationRequest_V200, '2.0.0') @validator.schema(schema.VnfInfoModificationRequest_V200, '2.0.0')
@coordinate.lock_vnf_instance('{id}') @coordinate.lock_vnf_instance('{id}')
def update(self, request, id, body): def update(self, request, id, body):
@ -205,7 +188,8 @@ class VnfLcmControllerV2(sol_wsgi.SolAPIController):
detail="vnfcInstanceId(%s) does not exist." detail="vnfcInstanceId(%s) does not exist."
% vnfc_mod['id']) % vnfc_mod['id'])
lcmocc = self._new_lcmocc(id, v2fields.LcmOperationType.MODIFY_INFO, lcmocc = vnflcm_utils.new_lcmocc(
id, v2fields.LcmOperationType.MODIFY_INFO,
body, v2fields.LcmOperationStateType.PROCESSING) body, v2fields.LcmOperationStateType.PROCESSING)
lcmocc.create(context) lcmocc.create(context)
@ -226,8 +210,8 @@ class VnfLcmControllerV2(sol_wsgi.SolAPIController):
lcmocc_utils.check_lcmocc_in_progress(context, id) lcmocc_utils.check_lcmocc_in_progress(context, id)
lcmocc = self._new_lcmocc(id, v2fields.LcmOperationType.INSTANTIATE, lcmocc = vnflcm_utils.new_lcmocc(
body) id, v2fields.LcmOperationType.INSTANTIATE, body)
req_param = lcmocc.operationParams req_param = lcmocc.operationParams
# if there is partial vimConnectionInfo check and fulfill here. # if there is partial vimConnectionInfo check and fulfill here.
@ -260,8 +244,8 @@ class VnfLcmControllerV2(sol_wsgi.SolAPIController):
lcmocc_utils.check_lcmocc_in_progress(context, id) lcmocc_utils.check_lcmocc_in_progress(context, id)
lcmocc = self._new_lcmocc(id, v2fields.LcmOperationType.TERMINATE, lcmocc = vnflcm_utils.new_lcmocc(
body) id, v2fields.LcmOperationType.TERMINATE, body)
lcmocc.create(context) lcmocc.create(context)
self.conductor_rpc.start_lcm_op(context, lcmocc.id) self.conductor_rpc.start_lcm_op(context, lcmocc.id)
@ -270,93 +254,19 @@ class VnfLcmControllerV2(sol_wsgi.SolAPIController):
return sol_wsgi.SolResponse(202, None, location=location) return sol_wsgi.SolResponse(202, None, location=location)
def _get_current_scale_level(self, inst, aspect_id):
if (inst.obj_attr_is_set('instantiatedVnfInfo') and
inst.instantiatedVnfInfo.obj_attr_is_set('scaleStatus')):
for scale_info in inst.instantiatedVnfInfo.scaleStatus:
if scale_info.aspectId == aspect_id:
return scale_info.scaleLevel
def _get_max_scale_level(self, inst, aspect_id):
if (inst.obj_attr_is_set('instantiatedVnfInfo') and
inst.instantiatedVnfInfo.obj_attr_is_set('maxScaleLevels')):
for scale_info in inst.instantiatedVnfInfo.maxScaleLevels:
if scale_info.aspectId == aspect_id:
return scale_info.scaleLevel
@validator.schema(schema.ScaleVnfRequest_V200, '2.0.0') @validator.schema(schema.ScaleVnfRequest_V200, '2.0.0')
@coordinate.lock_vnf_instance('{id}')
def scale(self, request, id, body): def scale(self, request, id, body):
context = request.context context = request.context
inst = inst_utils.get_inst(context, id) lcmocc = vnflcm_utils.scale(context, id, body)
if inst.instantiationState != 'INSTANTIATED':
raise sol_ex.VnfInstanceIsNotInstantiated(inst_id=id)
lcmocc_utils.check_lcmocc_in_progress(context, id)
# check parameters
aspect_id = body['aspectId']
if 'numberOfSteps' not in body:
# set default value (1) defined by SOL specification for
# the convenience of the following methods.
body['numberOfSteps'] = 1
scale_level = self._get_current_scale_level(inst, aspect_id)
max_scale_level = self._get_max_scale_level(inst, aspect_id)
if scale_level is None or max_scale_level is None:
raise sol_ex.InvalidScaleAspectId(aspect_id=aspect_id)
num_steps = body['numberOfSteps']
if body['type'] == 'SCALE_IN':
num_steps *= -1
scale_level += num_steps
if scale_level < 0 or scale_level > max_scale_level:
raise sol_ex.InvalidScaleNumberOfSteps(
num_steps=body['numberOfSteps'])
lcmocc = self._new_lcmocc(id, v2fields.LcmOperationType.SCALE,
body)
lcmocc.create(context)
self.conductor_rpc.start_lcm_op(context, lcmocc.id)
location = lcmocc_utils.lcmocc_href(lcmocc.id, self.endpoint) location = lcmocc_utils.lcmocc_href(lcmocc.id, self.endpoint)
return sol_wsgi.SolResponse(202, None, location=location) return sol_wsgi.SolResponse(202, None, location=location)
@validator.schema(schema.HealVnfRequest_V200, '2.0.0') @validator.schema(schema.HealVnfRequest_V200, '2.0.0')
@coordinate.lock_vnf_instance('{id}')
def heal(self, request, id, body): def heal(self, request, id, body):
context = request.context context = request.context
inst = inst_utils.get_inst(context, id) lcmocc = vnflcm_utils.heal(context, id, body)
if inst.instantiationState != 'INSTANTIATED':
raise sol_ex.VnfInstanceIsNotInstantiated(inst_id=id)
lcmocc_utils.check_lcmocc_in_progress(context, id)
# check parameter for later use
is_all = body.get('additionalParams', {}).get('all', False)
if not isinstance(is_all, bool):
raise sol_ex.SolValidationError(
detail="additionalParams['all'] must be bool.")
if 'vnfcInstanceId' in body:
inst_info = inst.instantiatedVnfInfo
vnfc_id = []
if inst_info.obj_attr_is_set('vnfcInfo'):
vnfc_id = [vnfc.id for vnfc in inst_info.vnfcInfo]
for req_vnfc_id in body['vnfcInstanceId']:
if req_vnfc_id not in vnfc_id:
raise sol_ex.SolValidationError(
detail="vnfcInstanceId(%s) does not exist."
% req_vnfc_id)
lcmocc = self._new_lcmocc(id, v2fields.LcmOperationType.HEAL, body)
lcmocc.create(context)
self.conductor_rpc.start_lcm_op(context, lcmocc.id)
location = lcmocc_utils.lcmocc_href(lcmocc.id, self.endpoint) location = lcmocc_utils.lcmocc_href(lcmocc.id, self.endpoint)
@ -373,7 +283,7 @@ class VnfLcmControllerV2(sol_wsgi.SolAPIController):
lcmocc_utils.check_lcmocc_in_progress(context, id) lcmocc_utils.check_lcmocc_in_progress(context, id)
lcmocc = self._new_lcmocc( lcmocc = vnflcm_utils.new_lcmocc(
id, v2fields.LcmOperationType.CHANGE_EXT_CONN, body) id, v2fields.LcmOperationType.CHANGE_EXT_CONN, body)
lcmocc.create(context) lcmocc.create(context)
@ -458,8 +368,8 @@ class VnfLcmControllerV2(sol_wsgi.SolAPIController):
raise sol_ex.SolValidationError( raise sol_ex.SolValidationError(
detail="'lcm-kubernetes-def-files' must be specified") detail="'lcm-kubernetes-def-files' must be specified")
lcmocc = self._new_lcmocc(id, v2fields.LcmOperationType.CHANGE_VNFPKG, lcmocc = vnflcm_utils.new_lcmocc(
body) id, v2fields.LcmOperationType.CHANGE_VNFPKG, body)
lcmocc.create(context) lcmocc.create(context)

View File

@ -288,6 +288,13 @@ class BaseVnfLcmKubernetesV2Test(base.BaseTestCase):
return self.tacker_client.do_request( return self.tacker_client.do_request(
path, "GET", version="2.0.0") path, "GET", version="2.0.0")
def list_lcmocc(self, filter_expr=None):
path = "/vnflcm/v2/vnf_lcm_op_occs"
if filter_expr:
path = "{}?{}".format(path, urllib.parse.urlencode(filter_expr))
return self.tacker_client.do_request(
path, "GET", version="2.0.0")
def create_subscription(self, req_body): def create_subscription(self, req_body):
path = "/vnffm/v1/subscriptions" path = "/vnffm/v1/subscriptions"
return self.tacker_client.do_request( return self.tacker_client.do_request(
@ -370,7 +377,12 @@ class BaseVnfLcmKubernetesV2Test(base.BaseTestCase):
path, "DELETE", version="2.1.0") path, "DELETE", version="2.1.0")
def prometheus_auto_scaling_alert(self, req_body): def prometheus_auto_scaling_alert(self, req_body):
path = "/alert/vnf_instances" path = "/alert/auto_scaling"
return self.tacker_client.do_request(
path, "POST", body=req_body)
def prometheus_auto_healing_alert(self, req_body):
path = "/alert/auto_healing"
return self.tacker_client.do_request( return self.tacker_client.do_request(
path, "POST", body=req_body) path, "POST", body=req_body)

View File

@ -829,3 +829,38 @@ def prometheus_auto_scaling_alert(inst_id):
"groupKey": "{}:{}", "groupKey": "{}:{}",
"truncatedAlerts": 0 "truncatedAlerts": 0
} }
def prometheus_auto_healing_alert(inst_id, vnfc_info_id):
return {
"receiver": "receiver",
"status": "firing",
"alerts": [{
"status": "firing",
"labels": {
"receiver_type": "tacker",
"function_type": "auto_heal",
"vnf_instance_id": inst_id,
"vnfc_info_id": vnfc_info_id
},
"annotations": {
},
"startsAt": "2022-06-21T23:47:36.453Z",
"endsAt": "0001-01-01T00:00:00Z",
"generatorURL": "http://controller147:9090/graph?g0.expr="
"up%7Bjob%3D%22node%22%7D+%3D%3D+0&g0.tab=1",
"fingerprint": "5ef77f1f8a3ecb8d"
}],
"groupLabels": {},
"commonLabels": {
"alertname": "NodeInstanceDown",
"job": "node"
},
"commonAnnotations": {
"description": "sample"
},
"externalURL": "http://controller147:9093",
"version": "4",
"groupKey": "{}:{}",
"truncatedAlerts": 0
}

View File

@ -0,0 +1,355 @@
# Copyright (C) 2023 Fujitsu
# All Rights Reserved.
#
# Licensed under the Apache License, Version 2.0 (the "License"); you may
# not use this file except in compliance with the License. You may obtain
# a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
# License for the specific language governing permissions and limitations
# under the License.
import ddt
import os
import time
from tacker.tests.functional.sol_kubernetes_v2 import base_v2
from tacker.tests.functional.sol_kubernetes_v2 import paramgen
# Waiting time to trigger autoheal (unit: second)
WAIT_AUTO_HEAL_TIME = 23
# Waiting time to lcmocc update in DB (unit: second)
WAIT_LCMOCC_UPDATE_TIME = 3
@ddt.ddt
class PromAutoScaleHealTest(base_v2.BaseVnfLcmKubernetesV2Test):
@classmethod
def setUpClass(cls):
super(PromAutoScaleHealTest, cls).setUpClass()
cur_dir = os.path.dirname(__file__)
test_instantiate_cnf_resources_path = os.path.join(
cur_dir, "samples/test_instantiate_cnf_resources")
cls.vnf_pkg_1, cls.vnfd_id_1 = cls.create_vnf_package(
test_instantiate_cnf_resources_path)
@classmethod
def tearDownClass(cls):
super(PromAutoScaleHealTest, cls).tearDownClass()
cls.delete_vnf_package(cls.vnf_pkg_1)
def setUp(self):
super(PromAutoScaleHealTest, self).setUp()
def test_vnfm_auto_heal_cnf(self):
"""Test Prometheus VNFM Auto Healing operations
* About LCM operations:
This test includes the following operations.
- 1. Create a new VNF instance resource
- 2. Instantiate a VNF instance
- 3. Show OpOcc
- 4-5. Receive Alert and Auto Heal
- 6. Show OpOcc
- 7. Receive Alert
- 8-9. Receive Alert and Auto Heal
- 10. Show OpOcc
- 11. Terminate a VNF instance
- 12. Show OpOcc
- 13. Delete a VNF instance
"""
# 1. LCM-Create: Create a new VNF instance resource
create_req = paramgen.instantiate_cnf_resources_create(self.vnfd_id_1)
resp, body = self.create_vnf_instance(create_req)
self.assertEqual(201, resp.status_code)
inst_id = body['id']
# 2. LCM-Instantiate: Instantiate a VNF instance
vim_id = self.get_k8s_vim_id()
instantiate_req = paramgen.min_sample_instantiate(vim_id)
instantiate_req['additionalParams'][
'lcm-kubernetes-def-files'] = ['Files/kubernetes/deployment.yaml']
instantiate_req['vnfConfigurableProperties'] = {
'isAutohealEnabled': True}
resp, body = self.instantiate_vnf_instance(inst_id, instantiate_req)
self.assertEqual(202, resp.status_code)
lcmocc_id = os.path.basename(resp.headers['Location'])
self.wait_lcmocc_complete(lcmocc_id)
# 3. LCM-Show-OpOccV2: Show OpOcc
resp, body = self.show_lcmocc(lcmocc_id)
self.assertEqual(200, resp.status_code)
self.assertEqual('COMPLETED', body['operationState'])
self.assertEqual('INSTANTIATE', body['operation'])
# 4-5. Send alert and auto heal
affected_vnfcs = body['resourceChanges']['affectedVnfcs']
vnfc_info_id = (affected_vnfcs[0]['vduId'] + '-'
+ affected_vnfcs[0]['id'])
alert = paramgen.prometheus_auto_healing_alert(inst_id, vnfc_info_id)
resp, body = self.prometheus_auto_healing_alert(alert)
self.assertEqual(204, resp.status_code)
# Since auto heal takes 20 seconds to trigger,
# wait 23 seconds here.
time.sleep(WAIT_AUTO_HEAL_TIME)
# 6. LCM-Show-OpOccV2: Show-OpOcc
filter_expr = {'filter': f'(eq,vnfInstanceId,{inst_id})'}
resp, body = self.list_lcmocc(filter_expr)
self.assertEqual(200, resp.status_code)
heal_lcmocc = [
heal_lcmocc for heal_lcmocc in body
if heal_lcmocc['startTime'] == max(
[lcmocc['startTime'] for lcmocc in body])][0]
lcmocc_id = heal_lcmocc['id']
self.wait_lcmocc_complete(lcmocc_id)
resp, body = self.show_lcmocc(lcmocc_id)
affected_vnfcs = body['resourceChanges']['affectedVnfcs']
self.assertEqual(200, resp.status_code)
self.assertEqual('COMPLETED', body['operationState'])
self.assertEqual('HEAL', body['operation'])
self.assertEqual(2, len(affected_vnfcs))
added_vnfcs = [
vnfc for vnfc in affected_vnfcs
if vnfc['changeType'] == 'ADDED']
self.assertEqual(1, len(added_vnfcs))
removed_vnfcs = [
vnfc for vnfc in affected_vnfcs
if vnfc['changeType'] == 'REMOVED']
self.assertEqual(1, len(removed_vnfcs))
removed_vnfc_info_id = (affected_vnfcs[0]['vduId'] + '-'
+ affected_vnfcs[0]['id'])
self.assertEqual(vnfc_info_id, removed_vnfc_info_id)
# 7. Send alert
resp, body = self.show_vnf_instance(inst_id)
self.assertEqual(200, resp.status_code)
vnfc_infos = body['instantiatedVnfInfo']['vnfcInfo']
vnfc_info_id_1 = vnfc_infos[0]['id']
alert = paramgen.prometheus_auto_healing_alert(
inst_id, vnfc_info_id_1)
resp, body = self.prometheus_auto_healing_alert(alert)
self.assertEqual(204, resp.status_code)
# 8-9. Send alert and auto heal
vnfc_info_id_2 = vnfc_infos[1]['id']
alert = paramgen.prometheus_auto_healing_alert(
inst_id, vnfc_info_id_2)
resp, body = self.prometheus_auto_healing_alert(alert)
self.assertEqual(204, resp.status_code)
# Since auto heal takes 20 seconds to trigger,
# wait 23 seconds here.
time.sleep(WAIT_AUTO_HEAL_TIME)
# 10. LCM-Show-OpOccV2: Show-OpOcc
filter_expr = {'filter': f'(eq,vnfInstanceId,{inst_id})'}
resp, body = self.list_lcmocc(filter_expr)
self.assertEqual(200, resp.status_code)
heal_lcmocc = [
heal_lcmocc for heal_lcmocc in body
if heal_lcmocc['startTime'] == max(
[lcmocc['startTime'] for lcmocc in body])][0]
lcmocc_id = heal_lcmocc['id']
self.wait_lcmocc_complete(lcmocc_id)
resp, body = self.show_lcmocc(lcmocc_id)
affected_vnfcs = body['resourceChanges']['affectedVnfcs']
self.assertEqual(200, resp.status_code)
self.assertEqual('COMPLETED', body['operationState'])
self.assertEqual('HEAL', body['operation'])
self.assertEqual(4, len(affected_vnfcs))
added_vnfcs = [
vnfc for vnfc in affected_vnfcs
if vnfc['changeType'] == 'ADDED']
self.assertEqual(2, len(added_vnfcs))
removed_vnfcs = [
vnfc for vnfc in affected_vnfcs
if vnfc['changeType'] == 'REMOVED']
self.assertEqual(2, len(removed_vnfcs))
removed_vnfc_info_ids = [
removed_vnfcs[0]['vduId'] + '-' + removed_vnfcs[0]['id'],
removed_vnfcs[1]['vduId'] + '-' + removed_vnfcs[1]['id']
]
self.assertCountEqual(
[vnfc_info_id_1, vnfc_info_id_2], removed_vnfc_info_ids)
# 11. LCM-Terminate: Terminate VNF
terminate_req = paramgen.terminate_vnf_min()
resp, body = self.terminate_vnf_instance(inst_id, terminate_req)
self.assertEqual(202, resp.status_code)
lcmocc_id = os.path.basename(resp.headers['Location'])
self.wait_lcmocc_complete(lcmocc_id)
# wait a bit because there is a bit time lag between lcmocc DB
# update and terminate completion.
time.sleep(WAIT_LCMOCC_UPDATE_TIME)
# 12. LCM-Show-OpOccV2: Show OpOcc
resp, body = self.show_lcmocc(lcmocc_id)
self.assertEqual(200, resp.status_code)
self.assertEqual('COMPLETED', body['operationState'])
self.assertEqual('TERMINATE', body['operation'])
# 13. LCM-Delete: Delete a VNF instance
resp, body = self.delete_vnf_instance(inst_id)
self.assertEqual(204, resp.status_code)
# check deletion of VNF instance
resp, body = self.show_vnf_instance(inst_id)
self.assertEqual(404, resp.status_code)
def test_vnfm_auto_scale_cnf(self):
"""Test Prometheus VNFM Auto Scaling operations
* About LCM operations:
This test includes the following operations.
- 1. Create a new VNF instance resource
- 2. Instantiate a VNF instance
- 3. Show OpOcc
- 4-5. Receive Alert and Auto Scale out
- 6. Show OpOcc
- 7-8. Receive Alert and Auto Scale in
- 9. Show OpOcc
- 10. Terminate a VNF instance
- 11. Show OpOcc
- 12. Delete a VNF instance
"""
# 1. LCM-Create: Create a new VNF instance resource
create_req = paramgen.instantiate_cnf_resources_create(self.vnfd_id_1)
resp, body = self.create_vnf_instance(create_req)
self.assertEqual(201, resp.status_code)
inst_id = body['id']
# 2. LCM-Instantiate: Instantiate a VNF instance
vim_id = self.get_k8s_vim_id()
instantiate_req = paramgen.min_sample_instantiate(vim_id)
instantiate_req['additionalParams'][
'lcm-kubernetes-def-files'] = ['Files/kubernetes/deployment.yaml']
instantiate_req['vnfConfigurableProperties'] = {
'isAutoscaleEnabled': True}
resp, body = self.instantiate_vnf_instance(inst_id, instantiate_req)
self.assertEqual(202, resp.status_code)
lcmocc_id = os.path.basename(resp.headers['Location'])
self.wait_lcmocc_complete(lcmocc_id)
# 3. LCM-Show-OpOccV2: Show OpOcc
resp, body = self.show_lcmocc(lcmocc_id)
self.assertEqual(200, resp.status_code)
self.assertEqual('COMPLETED', body['operationState'])
self.assertEqual('INSTANTIATE', body['operation'])
# 4-5. Send alert and auto-scale out
alert = paramgen.prometheus_auto_scaling_alert(inst_id)
resp, body = self.prometheus_auto_scaling_alert(alert)
self.assertEqual(204, resp.status_code)
# wait a bit because there is a bit time lag between lcmocc DB
# update and scale completion.
time.sleep(WAIT_LCMOCC_UPDATE_TIME)
# 6. LCM-Show-OpOccV2: Show-OpOcc
filter_expr = {'filter': f'(eq,vnfInstanceId,{inst_id})'}
resp, body = self.list_lcmocc(filter_expr)
self.assertEqual(200, resp.status_code)
scale_lcmocc = [
scale_lcmocc for scale_lcmocc in body
if scale_lcmocc['startTime'] == max(
[lcmocc['startTime'] for lcmocc in body])][0]
lcmocc_id = scale_lcmocc['id']
self.wait_lcmocc_complete(lcmocc_id)
resp, body = self.show_lcmocc(lcmocc_id)
affected_vnfcs = body['resourceChanges']['affectedVnfcs']
self.assertEqual(200, resp.status_code)
self.assertEqual('COMPLETED', body['operationState'])
self.assertEqual('SCALE', body['operation'])
self.assertEqual(1, len(affected_vnfcs))
self.assertEqual('ADDED', affected_vnfcs[0]['changeType'])
self.assertEqual(
alert['alerts'][0]['labels']['aspect_id'],
body['operationParams']['aspectId'])
# 7-8. Send alert and auto-scale in
alert['alerts'][0]['labels']['auto_scale_type'] = 'SCALE_IN'
resp, body = self.prometheus_auto_scaling_alert(alert)
self.assertEqual(204, resp.status_code)
# wait a bit because there is a bit time lag between lcmocc DB
# update and scale completion.
time.sleep(WAIT_LCMOCC_UPDATE_TIME)
# 9. LCM-Show-OpOccV2: Show-OpOcc
filter_expr = {'filter': f'(eq,vnfInstanceId,{inst_id})'}
resp, body = self.list_lcmocc(filter_expr)
self.assertEqual(200, resp.status_code)
scale_lcmocc = [
scale_lcmocc for scale_lcmocc in body
if scale_lcmocc['startTime'] == max(
[lcmocc['startTime'] for lcmocc in body])][0]
lcmocc_id = scale_lcmocc['id']
self.wait_lcmocc_complete(lcmocc_id)
resp, body = self.show_lcmocc(lcmocc_id)
affected_vnfcs = body['resourceChanges']['affectedVnfcs']
self.assertEqual(200, resp.status_code)
self.assertEqual('COMPLETED', body['operationState'])
self.assertEqual('SCALE', body['operation'])
self.assertEqual(1, len(affected_vnfcs))
self.assertEqual('REMOVED', affected_vnfcs[0]['changeType'])
self.assertEqual(
alert['alerts'][0]['labels']['aspect_id'],
body['operationParams']['aspectId'])
# 10. LCM-Terminate: Terminate VNF
terminate_req = paramgen.terminate_vnf_min()
resp, body = self.terminate_vnf_instance(inst_id, terminate_req)
self.assertEqual(202, resp.status_code)
lcmocc_id = os.path.basename(resp.headers['Location'])
self.wait_lcmocc_complete(lcmocc_id)
# wait a bit because there is a bit time lag between lcmocc DB
# update and terminate completion.
time.sleep(WAIT_LCMOCC_UPDATE_TIME)
# 11. LCM-Show-OpOccV2: Show OpOcc
resp, body = self.show_lcmocc(lcmocc_id)
self.assertEqual(200, resp.status_code)
self.assertEqual('COMPLETED', body['operationState'])
self.assertEqual('TERMINATE', body['operation'])
# 12. LCM-Delete: Delete a VNF instance
resp, body = self.delete_vnf_instance(inst_id)
self.assertEqual(204, resp.status_code)
# check deletion of VNF instance
resp, body = self.show_vnf_instance(inst_id)
self.assertEqual(404, resp.status_code)

View File

@ -1,115 +0,0 @@
# Copyright (C) 2022 Fujitsu
# All Rights Reserved.
#
# Licensed under the Apache License, Version 2.0 (the "License"); you may
# not use this file except in compliance with the License. You may obtain
# a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
# License for the specific language governing permissions and limitations
# under the License.
import ddt
import os
import time
from tacker.objects import fields
from tacker.tests.functional.sol_kubernetes_v2 import base_v2
from tacker.tests.functional.sol_kubernetes_v2 import paramgen
@ddt.ddt
class PrometheusAutoScalingTest(base_v2.BaseVnfLcmKubernetesV2Test):
@classmethod
def setUpClass(cls):
super(PrometheusAutoScalingTest, cls).setUpClass()
cur_dir = os.path.dirname(__file__)
test_instantiate_cnf_resources_path = os.path.join(
cur_dir, "samples/test_instantiate_cnf_resources")
cls.vnf_pkg_1, cls.vnfd_id_1 = cls.create_vnf_package(
test_instantiate_cnf_resources_path)
@classmethod
def tearDownClass(cls):
super(PrometheusAutoScalingTest, cls).tearDownClass()
cls.delete_vnf_package(cls.vnf_pkg_1)
def setUp(self):
super(PrometheusAutoScalingTest, self).setUp()
def test_prometheus_auto_scaling_basic(self):
"""Test Prometheus Auto Scaling operations with all attributes set
* About LCM operations:
This test includes the following operations.
- 1. Create a new VNF instance resource
- 2. Instantiate a VNF instance
- 3. Prometheus Auto Scaling alert.
- 4. Terminate a VNF instance
- 5. Delete a VNF instance
"""
# 1. LCM-Create: Create a new VNF instance resource
# NOTE: extensions and vnfConfigurableProperties are omitted
# because they are commented out in etsi_nfv_sol001.
create_req = paramgen.instantiate_cnf_resources_create(self.vnfd_id_1)
resp, body = self.create_vnf_instance(create_req)
self.assertEqual(201, resp.status_code)
inst_id = body['id']
# 2. LCM-Instantiate: Instantiate a VNF instance
vim_id = self.get_k8s_vim_id()
instantiate_req = paramgen.min_sample_instantiate(vim_id)
instantiate_req['additionalParams'][
'lcm-kubernetes-def-files'] = ['Files/kubernetes/deployment.yaml']
instantiate_req['vnfConfigurableProperties'] = {
'isAutoscaleEnabled': True}
resp, body = self.instantiate_vnf_instance(inst_id, instantiate_req)
self.assertEqual(202, resp.status_code)
lcmocc_id = os.path.basename(resp.headers['Location'])
self.wait_lcmocc_complete(lcmocc_id)
resp, body = self.show_vnf_instance(inst_id)
self.assertEqual(200, resp.status_code)
# 3. Send Auto-Healing alert
alert = paramgen.prometheus_auto_scaling_alert(inst_id)
# CNF scale is not integrated yet. use this value for now.
alert['alerts'][0]['labels']['aspect_id'] = 'invalid_id'
resp, body = self.prometheus_auto_scaling_alert(alert)
self.assertEqual(204, resp.status_code)
time.sleep(5)
# 4. LCM-Terminate: Terminate VNF
terminate_req = paramgen.terminate_vnf_min()
resp, body = self.terminate_vnf_instance(inst_id, terminate_req)
self.assertEqual(202, resp.status_code)
lcmocc_id = os.path.basename(resp.headers['Location'])
self.wait_lcmocc_complete(lcmocc_id)
# wait a bit because there is a bit time lag between lcmocc DB
# update and terminate completion.
time.sleep(10)
# check instantiationState of VNF
resp, body = self.show_vnf_instance(inst_id)
self.assertEqual(200, resp.status_code)
self.assertEqual(fields.VnfInstanceState.NOT_INSTANTIATED,
body['instantiationState'])
# 5. LCM-Delete: Delete a VNF instance
resp, body = self.delete_vnf_instance(inst_id)
self.assertEqual(204, resp.status_code)
# check deletion of VNF instance
resp, body = self.show_vnf_instance(inst_id)
self.assertEqual(404, resp.status_code)

View File

@ -0,0 +1,361 @@
# Copyright (C) 2023 Fujitsu
# All Rights Reserved.
#
# Licensed under the Apache License, Version 2.0 (the "License"); you may
# not use this file except in compliance with the License. You may obtain
# a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
# License for the specific language governing permissions and limitations
# under the License.
import ddt
import os
import time
from tacker.tests.functional.sol_v2_common import paramgen
from tacker.tests.functional.sol_v2_common import test_vnflcm_basic_common
# Waiting time to trigger autoheal (unit: second)
WAIT_AUTO_HEAL_TIME = 23
# Waiting time to lcmocc update in DB (unit: second)
WAIT_LCMOCC_UPDATE_TIME = 3
@ddt.ddt
class PromAutoScaleHealTest(test_vnflcm_basic_common.CommonVnfLcmTest):
@classmethod
def setUpClass(cls):
super(PromAutoScaleHealTest, cls).setUpClass()
cur_dir = os.path.dirname(__file__)
# for basic lcms tests min pattern
basic_lcms_min_path = os.path.join(
cur_dir, "../sol_v2_common/samples/basic_lcms_min")
# no image contained
cls.vnf_pkg_2, cls.vnfd_id_2 = cls.create_vnf_package(
basic_lcms_min_path)
@classmethod
def tearDownClass(cls):
super(PromAutoScaleHealTest, cls).tearDownClass()
cls.delete_vnf_package(cls.vnf_pkg_2)
def setUp(self):
super(PromAutoScaleHealTest, self).setUp()
def test_vnfm_auto_heal_vnf(self):
"""Test Prometheus VNFM Auto Healing operations
* About LCM operations:
This test includes the following operations.
- 1. Create a new VNF instance resource
- 2. Instantiate a VNF instance
- 3. Show OpOcc
- 4-5. Receive Alert and Auto Heal
- 6. Show OpOcc
- 7. Receive Alert
- 8-9. Receive Alert and Auto Heal
- 10. Show OpOcc
- 11. Terminate a VNF instance
- 12. Show OpOcc
- 13. Delete a VNF instance
"""
# 1. Create VNF instance
expected_inst_attrs = [
'id', 'vnfdId', 'vnfProductName', 'vnfSoftwareVersion',
'vnfdVersion', 'instantiationState', '_links'
]
create_req = paramgen.create_vnf_min(self.vnfd_id_2)
resp, body = self.create_vnf_instance(create_req)
self.assertEqual(201, resp.status_code)
self.check_resp_headers_in_create(resp)
self.check_resp_body(body, expected_inst_attrs)
inst_id = body['id']
# 2. Instantiate VNF
instantiate_req = paramgen.instantiate_vnf_min()
instantiate_req['vnfConfigurableProperties'] = {
'isAutohealEnabled': True}
resp, body = self.instantiate_vnf_instance(inst_id, instantiate_req)
self.assertEqual(202, resp.status_code)
lcmocc_id = os.path.basename(resp.headers['Location'])
self.wait_lcmocc_complete(lcmocc_id)
# 3. Show OpOcc
resp, body = self.show_lcmocc(lcmocc_id)
self.assertEqual(200, resp.status_code)
self.assertEqual('COMPLETED', body['operationState'])
self.assertEqual('INSTANTIATE', body['operation'])
# 4-5. Send alert and auto heal
affected_vnfcs = body['resourceChanges']['affectedVnfcs']
vnfc_info_id = (affected_vnfcs[0]['vduId'] + '-'
+ affected_vnfcs[0]['id'])
alert = paramgen.prometheus_auto_healing_alert(inst_id, vnfc_info_id)
resp, body = self.prometheus_auto_healing_alert(alert)
self.assertEqual(204, resp.status_code)
# Since auto heal takes 20 seconds to trigger,
# wait 23 seconds here.
time.sleep(WAIT_AUTO_HEAL_TIME)
# 6. Show-OpOcc
filter_expr = {'filter': f'(eq,vnfInstanceId,{inst_id})'}
resp, body = self.list_lcmocc(filter_expr)
self.assertEqual(200, resp.status_code)
heal_lcmocc = [
heal_lcmocc for heal_lcmocc in body
if heal_lcmocc['startTime'] == max(
[lcmocc['startTime'] for lcmocc in body])][0]
lcmocc_id = heal_lcmocc['id']
self.wait_lcmocc_complete(lcmocc_id)
resp, body = self.show_lcmocc(lcmocc_id)
affected_vnfcs = body['resourceChanges']['affectedVnfcs']
self.assertEqual(200, resp.status_code)
self.assertEqual('COMPLETED', body['operationState'])
self.assertEqual('HEAL', body['operation'])
self.assertEqual(2, len(affected_vnfcs))
added_vnfcs = [
vnfc for vnfc in affected_vnfcs
if vnfc['changeType'] == 'ADDED']
self.assertEqual(1, len(added_vnfcs))
removed_vnfcs = [
vnfc for vnfc in affected_vnfcs
if vnfc['changeType'] == 'REMOVED']
self.assertEqual(1, len(removed_vnfcs))
removed_vnfc_info_id = (affected_vnfcs[0]['vduId'] + '-'
+ affected_vnfcs[0]['id'])
self.assertEqual(vnfc_info_id, removed_vnfc_info_id)
# 7. Send alert
resp, body = self.show_vnf_instance(inst_id)
self.assertEqual(200, resp.status_code)
vnfc_infos = body['instantiatedVnfInfo']['vnfcInfo']
vnfc_info_id_1 = vnfc_infos[0]['id']
alert = paramgen.prometheus_auto_healing_alert(
inst_id, vnfc_info_id_1)
resp, body = self.prometheus_auto_healing_alert(alert)
self.assertEqual(204, resp.status_code)
# 8-9. Send alert and auto heal
vnfc_info_id_2 = vnfc_infos[1]['id']
alert = paramgen.prometheus_auto_healing_alert(
inst_id, vnfc_info_id_2)
resp, body = self.prometheus_auto_healing_alert(alert)
self.assertEqual(204, resp.status_code)
# Since auto heal takes 20 seconds to trigger,
# wait 23 seconds here.
time.sleep(WAIT_AUTO_HEAL_TIME)
# 10. Show-OpOcc
filter_expr = {'filter': f'(eq,vnfInstanceId,{inst_id})'}
resp, body = self.list_lcmocc(filter_expr)
self.assertEqual(200, resp.status_code)
heal_lcmocc = [
heal_lcmocc for heal_lcmocc in body
if heal_lcmocc['startTime'] == max(
[lcmocc['startTime'] for lcmocc in body])][0]
lcmocc_id = heal_lcmocc['id']
self.wait_lcmocc_complete(lcmocc_id)
resp, body = self.show_lcmocc(lcmocc_id)
affected_vnfcs = body['resourceChanges']['affectedVnfcs']
self.assertEqual(200, resp.status_code)
self.assertEqual('COMPLETED', body['operationState'])
self.assertEqual('HEAL', body['operation'])
self.assertEqual(4, len(affected_vnfcs))
added_vnfcs = [
vnfc for vnfc in affected_vnfcs
if vnfc['changeType'] == 'ADDED']
self.assertEqual(2, len(added_vnfcs))
removed_vnfcs = [
vnfc for vnfc in affected_vnfcs
if vnfc['changeType'] == 'REMOVED']
self.assertEqual(2, len(removed_vnfcs))
removed_vnfc_info_ids = [
removed_vnfcs[0]['vduId'] + '-' + removed_vnfcs[0]['id'],
removed_vnfcs[1]['vduId'] + '-' + removed_vnfcs[1]['id']
]
self.assertCountEqual(
[vnfc_info_id_1, vnfc_info_id_2], removed_vnfc_info_ids)
# 11. Terminate a VNF instance
terminate_req = paramgen.terminate_vnf_min()
resp, body = self.terminate_vnf_instance(inst_id, terminate_req)
self.assertEqual(202, resp.status_code)
lcmocc_id = os.path.basename(resp.headers['Location'])
self.wait_lcmocc_complete(lcmocc_id)
# wait a bit because there is a bit time lag between lcmocc DB
# update and terminate completion.
time.sleep(WAIT_LCMOCC_UPDATE_TIME)
# 12. Show OpOcc
resp, body = self.show_lcmocc(lcmocc_id)
self.assertEqual(200, resp.status_code)
self.assertEqual('COMPLETED', body['operationState'])
self.assertEqual('TERMINATE', body['operation'])
# 13. Delete a VNF instance
resp, body = self.delete_vnf_instance(inst_id)
self.assertEqual(204, resp.status_code)
# check deletion of VNF instance
resp, body = self.show_vnf_instance(inst_id)
self.assertEqual(404, resp.status_code)
def test_vnfm_auto_scale_vnf(self):
"""Test Prometheus VNFM Auto Scaling operations
* About LCM operations:
This test includes the following operations.
- 1. Create a new VNF instance resource
- 2. Instantiate a VNF instance
- 3. Show OpOcc
- 4-5. Receive Alert and Auto Scale out
- 6. Show OpOcc
- 7-8. Receive Alert and Auto Scale in
- 9. Show OpOcc
- 10. Terminate a VNF instance
- 11. Show OpOcc
- 12. Delete a VNF instance
"""
# 1. Create VNF instance
expected_inst_attrs = [
'id', 'vnfdId', 'vnfProductName', 'vnfSoftwareVersion',
'vnfdVersion', 'instantiationState', '_links'
]
create_req = paramgen.create_vnf_min(self.vnfd_id_2)
resp, body = self.create_vnf_instance(create_req)
self.assertEqual(201, resp.status_code)
self.check_resp_headers_in_create(resp)
self.check_resp_body(body, expected_inst_attrs)
inst_id = body['id']
# 2. Instantiate VNF
instantiate_req = paramgen.instantiate_vnf_min()
instantiate_req['vnfConfigurableProperties'] = {
'isAutoscaleEnabled': True}
resp, body = self.instantiate_vnf_instance(inst_id, instantiate_req)
self.assertEqual(202, resp.status_code)
lcmocc_id = os.path.basename(resp.headers['Location'])
self.wait_lcmocc_complete(lcmocc_id)
# 3. Show OpOcc
resp, body = self.show_lcmocc(lcmocc_id)
self.assertEqual(200, resp.status_code)
self.assertEqual('COMPLETED', body['operationState'])
self.assertEqual('INSTANTIATE', body['operation'])
# 4-5. Send alert and auto heal
alert = paramgen.prometheus_auto_scaling_alert(inst_id)
resp, body = self.prometheus_auto_scaling_alert(alert)
self.assertEqual(204, resp.status_code)
# wait a bit because there is a bit time lag between lcmocc DB
# update and scale completion.
time.sleep(WAIT_LCMOCC_UPDATE_TIME)
# 6. Show-OpOcc
filter_expr = {'filter': f'(eq,vnfInstanceId,{inst_id})'}
resp, body = self.list_lcmocc(filter_expr)
self.assertEqual(200, resp.status_code)
scale_lcmocc = [
scale_lcmocc for scale_lcmocc in body
if scale_lcmocc['startTime'] == max(
[lcmocc['startTime'] for lcmocc in body])][0]
lcmocc_id = scale_lcmocc['id']
self.wait_lcmocc_complete(lcmocc_id)
resp, body = self.show_lcmocc(lcmocc_id)
affected_vnfcs = body['resourceChanges']['affectedVnfcs']
self.assertEqual(200, resp.status_code)
self.assertEqual('COMPLETED', body['operationState'])
self.assertEqual('SCALE', body['operation'])
self.assertEqual(1, len(affected_vnfcs))
self.assertEqual('ADDED', affected_vnfcs[0]['changeType'])
self.assertEqual(
alert['alerts'][0]['labels']['aspect_id'],
body['operationParams']['aspectId'])
# 7-8. Send alert and auto-scale in
alert['alerts'][0]['labels']['auto_scale_type'] = 'SCALE_IN'
resp, body = self.prometheus_auto_scaling_alert(alert)
self.assertEqual(204, resp.status_code)
# wait a bit because there is a bit time lag between lcmocc DB
# update and scale completion.
time.sleep(WAIT_LCMOCC_UPDATE_TIME)
# 9. Show-OpOcc
filter_expr = {'filter': f'(eq,vnfInstanceId,{inst_id})'}
resp, body = self.list_lcmocc(filter_expr)
self.assertEqual(200, resp.status_code)
scale_lcmocc = [
scale_lcmocc for scale_lcmocc in body
if scale_lcmocc['startTime'] == max(
[lcmocc['startTime'] for lcmocc in body])][0]
lcmocc_id = scale_lcmocc['id']
self.wait_lcmocc_complete(lcmocc_id)
resp, body = self.show_lcmocc(lcmocc_id)
affected_vnfcs = body['resourceChanges']['affectedVnfcs']
self.assertEqual(200, resp.status_code)
self.assertEqual('COMPLETED', body['operationState'])
self.assertEqual('SCALE', body['operation'])
self.assertEqual(1, len(affected_vnfcs))
self.assertEqual('REMOVED', affected_vnfcs[0]['changeType'])
self.assertEqual(
alert['alerts'][0]['labels']['aspect_id'],
body['operationParams']['aspectId'])
# 10. Terminate a VNF instance
terminate_req = paramgen.terminate_vnf_min()
resp, body = self.terminate_vnf_instance(inst_id, terminate_req)
self.assertEqual(202, resp.status_code)
lcmocc_id = os.path.basename(resp.headers['Location'])
self.wait_lcmocc_complete(lcmocc_id)
# wait a bit because there is a bit time lag between lcmocc DB
# update and terminate completion.
time.sleep(WAIT_LCMOCC_UPDATE_TIME)
# 11. Show OpOcc
resp, body = self.show_lcmocc(lcmocc_id)
self.assertEqual(200, resp.status_code)
self.assertEqual('COMPLETED', body['operationState'])
self.assertEqual('TERMINATE', body['operation'])
# 12. Delete a VNF instance
resp, body = self.delete_vnf_instance(inst_id)
self.assertEqual(204, resp.status_code)
# check deletion of VNF instance
resp, body = self.show_vnf_instance(inst_id)
self.assertEqual(404, resp.status_code)

View File

@ -497,6 +497,16 @@ class BaseSolV2Test(base.BaseTestCase):
return self.tacker_client.do_request( return self.tacker_client.do_request(
path, "POST", version="2.0.0") path, "POST", version="2.0.0")
def prometheus_auto_healing_alert(self, req_body):
path = "/alert/auto_healing"
return self.tacker_client.do_request(
path, "POST", body=req_body)
def prometheus_auto_scaling_alert(self, req_body):
path = "/alert/auto_scaling"
return self.tacker_client.do_request(
path, "POST", body=req_body)
def create_subscription(self, req_body): def create_subscription(self, req_body):
path = "/vnflcm/v2/subscriptions" path = "/vnflcm/v2/subscriptions"
return self.tacker_client.do_request( return self.tacker_client.do_request(

View File

@ -1230,6 +1230,77 @@ def sample4_terminate():
} }
def prometheus_auto_healing_alert(inst_id, vnfc_info_id):
return {
"receiver": "receiver",
"status": "firing",
"alerts": [{
"status": "firing",
"labels": {
"receiver_type": "tacker",
"function_type": "auto_heal",
"vnf_instance_id": inst_id,
"vnfc_info_id": vnfc_info_id
},
"annotations": {
},
"startsAt": "2022-06-21T23:47:36.453Z",
"endsAt": "0001-01-01T00:00:00Z",
"generatorURL": "http://controller147:9090/graph?g0.expr="
"up%7Bjob%3D%22node%22%7D+%3D%3D+0&g0.tab=1",
"fingerprint": "5ef77f1f8a3ecb8d"
}],
"groupLabels": {},
"commonLabels": {
"alertname": "NodeInstanceDown",
"job": "node"
},
"commonAnnotations": {
"description": "sample"
},
"externalURL": "http://controller147:9093",
"version": "4",
"groupKey": "{}:{}",
"truncatedAlerts": 0
}
def prometheus_auto_scaling_alert(inst_id):
return {
"receiver": "receiver",
"status": "firing",
"alerts": [{
"status": "firing",
"labels": {
"receiver_type": "tacker",
"function_type": "auto_scale",
"vnf_instance_id": inst_id,
"auto_scale_type": "SCALE_OUT",
"aspect_id": "VDU1_scale"
},
"annotations": {
},
"startsAt": "2022-06-21T23:47:36.453Z",
"endsAt": "0001-01-01T00:00:00Z",
"generatorURL": "http://controller147:9090/graph?g0.expr="
"up%7Bjob%3D%22node%22%7D+%3D%3D+0&g0.tab=1",
"fingerprint": "5ef77f1f8a3ecb8d"
}],
"groupLabels": {},
"commonLabels": {
"alertname": "NodeInstanceDown",
"job": "node"
},
"commonAnnotations": {
"description": "sample"
},
"externalURL": "http://controller147:9093",
"version": "4",
"groupKey": "{}:{}",
"truncatedAlerts": 0
}
def server_notification(alarm_id): def server_notification(alarm_id):
return { return {
'notification': { 'notification': {

View File

@ -857,6 +857,40 @@ class TestPrometheusPluginFm(base.TestCase):
pp._alert, self.request) pp._alert, self.request)
class TestPrometheusPluginAutoHealing(base.TestCase):
def setUp(self):
super(TestPrometheusPluginAutoHealing, self).setUp()
objects.register_all()
self.context = context.get_admin_context()
self.request = mock.Mock()
self.request.context = self.context
prometheus_plugin.PrometheusPluginAutoHealing._instance = None
def tearDown(self):
super(TestPrometheusPluginAutoHealing, self).tearDown()
# delete singleton object
prometheus_plugin.PrometheusPluginAutoHealing._instance = None
def test_constructor_error(self):
self.config_fixture.config(
group='prometheus_plugin', auto_healing=False)
mon_base.MonitoringPlugin.get_instance(
prometheus_plugin.PrometheusPluginAutoHealing)
self.assertRaises(
SystemError,
prometheus_plugin.PrometheusPluginAutoHealing)
def test_constructor_stub(self):
self.config_fixture.config(
group='prometheus_plugin', auto_healing=False)
pp = mon_base.MonitoringPlugin.get_instance(
prometheus_plugin.PrometheusPluginAutoHealing)
self.assertIsInstance(pp._instance, mon_base.MonitoringPluginStub)
pp = mon_base.MonitoringPlugin.get_instance(
prometheus_plugin.PrometheusPluginAutoHealing)
self.assertIsInstance(pp._instance, mon_base.MonitoringPluginStub)
class TestPrometheusPluginAutoScaling(base.TestCase): class TestPrometheusPluginAutoScaling(base.TestCase):
def setUp(self): def setUp(self):
super(TestPrometheusPluginAutoScaling, self).setUp() super(TestPrometheusPluginAutoScaling, self).setUp()

View File

@ -13,10 +13,10 @@
# License for the specific language governing permissions and limitations # License for the specific language governing permissions and limitations
# under the License. # under the License.
import webob import time
from tacker import context from tacker import context
from tacker.sol_refactored.common import http_client from tacker.sol_refactored.common import vnflcm_utils
from tacker.sol_refactored.conductor import conductor_v2 from tacker.sol_refactored.conductor import conductor_v2
from tacker.sol_refactored.conductor import prometheus_plugin_driver as pp_drv from tacker.sol_refactored.conductor import prometheus_plugin_driver as pp_drv
from tacker.sol_refactored import objects from tacker.sol_refactored import objects
@ -119,6 +119,7 @@ class TestPrometheusPlugin(db_base.SqlTestCase):
self.context = context.get_admin_context() self.context = context.get_admin_context()
self.request = mock.Mock() self.request = mock.Mock()
self.request.context = self.context self.request.context = self.context
self.timer_test = (None, None)
self.config_fixture.config( self.config_fixture.config(
group='prometheus_plugin', performance_management=True) group='prometheus_plugin', performance_management=True)
self.conductor = conductor_v2.ConductorV2() self.conductor = conductor_v2.ConductorV2()
@ -129,16 +130,13 @@ class TestPrometheusPlugin(db_base.SqlTestCase):
# delete singleton object # delete singleton object
pp_drv.PrometheusPluginDriver._instance = None pp_drv.PrometheusPluginDriver._instance = None
@mock.patch.object(http_client.HttpClient, 'do_request') @mock.patch.object(vnflcm_utils, 'scale')
def test_request_scale(self, mock_do_request): def test_trigger_scale(self, mock_do_scale):
resp = webob.Response()
resp.status_code = 202
mock_do_request.return_value = resp, {}
scale_req = { scale_req = {
'type': 'SCALE_OUT', 'type': 'SCALE_OUT',
'aspect_id': 'vdu', 'aspectId': 'vdu',
} }
self.conductor.request_scale( self.conductor.trigger_scale(
self.context, 'vnf_instance_id', scale_req) self.context, 'vnf_instance_id', scale_req)
def test_constructor(self): def test_constructor(self):
@ -150,8 +148,7 @@ class TestPrometheusPlugin(db_base.SqlTestCase):
group='prometheus_plugin', performance_management=False) group='prometheus_plugin', performance_management=False)
pp_drv.PrometheusPluginDriver._instance = None pp_drv.PrometheusPluginDriver._instance = None
drv = pp_drv.PrometheusPluginDriver.instance() drv = pp_drv.PrometheusPluginDriver.instance()
drv = pp_drv.PrometheusPluginDriver.instance() drv.trigger_scale(None, None, None)
drv.request_scale(None, None, None)
self.config_fixture.config( self.config_fixture.config(
group='prometheus_plugin', performance_management=True) group='prometheus_plugin', performance_management=True)
drv = pp_drv.PrometheusPluginDriver.instance() drv = pp_drv.PrometheusPluginDriver.instance()
@ -163,3 +160,73 @@ class TestPrometheusPlugin(db_base.SqlTestCase):
self.assertRaises( self.assertRaises(
SystemError, SystemError,
pp_drv.PrometheusPluginDriver) pp_drv.PrometheusPluginDriver)
def test_conductor_vnfm_auto_heal_queue(self):
self.config_fixture.config(
group='prometheus_plugin', auto_healing=True)
pp_drv.PrometheusPluginDriver._instance = None
self.conductor.prom_driver = pp_drv.PrometheusPluginDriver.instance()
self.config_fixture.config(
group='prometheus_plugin', timer_interval=1)
# queueing test
id = 'test_id'
self.conductor.enqueue_auto_heal_instance(
self.context, id, 'id')
self.conductor.enqueue_auto_heal_instance(
self.context, id, 'id2')
self.assertEqual(
self.conductor.prom_driver.timer_map[id].queue,
{'id', 'id2'})
# Since the timeout period of `VnfmAutoHealTimer` is set to 1 second,
# it is also necessary to wait for 1 second before asserting.
time.sleep(1)
# remove_timer test
self.conductor.dequeue_auto_heal_instance(self.context, id)
self.assertNotIn(id, self.conductor.prom_driver.timer_map)
# remove_timer test: invalid_id
self.conductor.dequeue_auto_heal_instance(
self.context, 'invalid_id')
@mock.patch.object(vnflcm_utils, 'heal')
def test_conductor_timer_expired(self, mock_do_heal):
self.config_fixture.config(
group='prometheus_plugin', auto_healing=True)
pp_drv.PrometheusPluginDriver._instance = None
self.conductor.prom_driver = pp_drv.PrometheusPluginDriver.instance()
self.conductor.prom_driver._timer_expired(
self.context, 'test_id', ['id'])
def expired(self, context, id, queue):
queue.sort()
self.timer_test = (id, queue)
def test_timer(self):
# queueing test
timer = pp_drv.VnfmAutoHealTimer(self.context, 'id', 1, self.expired)
timer.add_vnfc_info_id('1')
timer.add_vnfc_info_id('3')
# Since the timeout period of `VnfmAutoHealTimer` is set to 1 second,
# it is also necessary to wait for 1 second before asserting.
time.sleep(1)
self.assertEqual(self.timer_test[0], 'id')
self.assertEqual(self.timer_test[1], ['1', '3'])
def test_timer_cancel(self):
# cancel test
timer = pp_drv.VnfmAutoHealTimer(self.context, 'id2', 1, self.expired)
timer.add_vnfc_info_id('5')
timer.cancel()
# Since the timeout period of `VnfmAutoHealTimer` is set to 1 second,
# it is also necessary to wait for 1 second before asserting.
time.sleep(1)
self.assertIsNone(self.timer_test[0])
self.assertIsNone(self.timer_test[1])
def test_timer_destructor(self):
# method call after cancel()
timer = pp_drv.VnfmAutoHealTimer(self.context, 'id', 1, self.expired)
timer.cancel()
timer.expire()
timer.add_vnfc_info_id(['4'])
timer.cancel()
timer.__del__()

View File

@ -145,15 +145,65 @@ _body_scale_alert1 = {
'fingerprint': '5ef77f1f8a3ecb8d' 'fingerprint': '5ef77f1f8a3ecb8d'
} }
_body_heal_alert1 = {
'status': 'firing',
'labels': {
'receiver_type': 'tacker',
'function_type': 'auto_heal',
'vnf_instance_id': 'vnf instance id',
'vnfc_info_id': 'vnfc info id'
},
'annotations': {
},
'startsAt': '2022-06-21T23:47:36.453Z',
'endsAt': '0001-01-01T00:00:00Z',
'generatorURL': 'http://controller147:9090/graph?g0.expr='
'up%7Bjob%3D%22node%22%7D+%3D%3D+0&g0.tab=1',
'fingerprint': '5ef77f1f8a3ecb8d'
}
# function_type mismatch # function_type mismatch
_body_scale_alert2 = copy.deepcopy(_body_scale_alert1) _body_scale_alert2 = copy.deepcopy(_body_scale_alert1)
_body_scale_alert2['labels']['function_type'] = 'vnffm' _body_scale_alert2['labels']['function_type'] = 'vnffm'
_body_scale_alert3 = copy.deepcopy(_body_scale_alert1)
_body_scale_alert3['status'] = 'resolved'
_body_scale_alert4 = copy.deepcopy(_body_scale_alert1)
_body_scale_alert4['labels']['function_type'] = 'auto_heal'
_body_scale_alert5 = copy.deepcopy(_body_scale_alert1)
_body_scale_alert5['labels']['aspect_id'] = 'aspect id'
_body_scale = copy.deepcopy(_body_base) _body_scale = copy.deepcopy(_body_base)
_body_scale.update({ _body_scale.update({
'alerts': [_body_scale_alert1, _body_scale_alert2] 'alerts': [_body_scale_alert1, _body_scale_alert2]
}) })
_body_scale_continue = copy.deepcopy(_body_base)
_body_scale_continue.update({
'alerts': [_body_scale_alert3, _body_scale_alert4, _body_scale_alert5]
})
_body_heal = copy.deepcopy(_body_base)
_body_heal.update({
'alerts': [_body_heal_alert1]
})
_body_heal_alert2 = copy.deepcopy(_body_heal_alert1)
_body_heal_alert2['status'] = 'resolved'
_body_heal_alert3 = copy.deepcopy(_body_heal_alert1)
_body_heal_alert3['labels']['function_type'] = 'auto_scale'
_body_heal_alert4 = copy.deepcopy(_body_heal_alert1)
_body_heal_alert4['labels']['vnfc_info_id'] = 'vnfcInfoId'
_body_heal_continue = copy.deepcopy(_body_base)
_body_heal_continue.update({
'alerts': [_body_heal_alert2, _body_heal_alert3, _body_heal_alert4]
})
_inst1 = { _inst1 = {
'id': 'test_id', 'id': 'test_id',
'vnfdId': 'vnfdId', 'vnfdId': 'vnfdId',
@ -187,6 +237,14 @@ _inst1 = {
'vduId': 'vdu_id', 'vduId': 'vdu_id',
'vnfcResourceInfoId': 'id2', 'vnfcResourceInfoId': 'id2',
'vnfcState': 'STARTED' 'vnfcState': 'STARTED'
}, {
'id': 'vnfc info id',
'vduId': 'vdu_id',
'vnfcResourceInfoId': 'id2',
'vnfcState': 'STARTED'
}],
'scaleStatus': [{
'aspectId': 'aspect'
}] }]
}, },
'metadata': { 'metadata': {
@ -198,6 +256,31 @@ _inst2.update({
'vnfConfigurableProperties': { 'vnfConfigurableProperties': {
'isAutoscaleEnabled': True 'isAutoscaleEnabled': True
}, },
'instantiationState': 'INSTANTIATED'
})
_inst3 = copy.deepcopy(_inst1)
_inst3.update({
'vnfConfigurableProperties': {
'isAutoscaleEnabled': False
},
'instantiationState': 'INSTANTIATED'
})
_inst4 = copy.deepcopy(_inst1)
_inst4.update({
'vnfConfigurableProperties': {
'isAutohealEnabled': False
},
'instantiationState': 'INSTANTIATED'
})
_inst5 = copy.deepcopy(_inst1)
_inst5.update({
'vnfConfigurableProperties': {
'isAutohealEnabled': True
},
'instantiationState': 'INSTANTIATED'
}) })
datetime_test = datetime.datetime.fromisoformat( datetime_test = datetime.datetime.fromisoformat(
@ -324,6 +407,64 @@ class TestPrometheusPluginFm(base.TestCase):
self.assertEqual(204, result.status) self.assertEqual(204, result.status)
class TestPrometheusPluginAutoHealing(base.TestCase):
def setUp(self):
super(TestPrometheusPluginAutoHealing, self).setUp()
objects.register_all()
self.context = context.get_admin_context()
self.request = mock.Mock()
self.request.context = self.context
self.controller = prometheus_plugin_controller.AutoHealingController()
plugin.PrometheusPluginAutoHealing._instance = None
def tearDown(self):
super(TestPrometheusPluginAutoHealing, self).tearDown()
# delete singleton object
plugin.PrometheusPluginAutoHealing._instance = None
def test_auto_healing_config_false(self):
self.config_fixture.config(
group='prometheus_plugin', auto_healing=False)
self.assertRaises(
sol_ex.PrometheusPluginNotEnabled,
self.controller.auto_healing, self.request, {})
@mock.patch.object(inst_utils, 'get_inst')
def test_auto_healing_no_autoheal_enabled(self, mock_inst):
self.config_fixture.config(
group='prometheus_plugin', auto_healing=True)
mock_inst.return_value = objects.VnfInstanceV2.from_dict(_inst4)
result = self.controller.auto_healing(
self.request, _body_heal)
self.assertEqual(204, result.status)
@mock.patch.object(inst_utils, 'get_inst')
def test_auto_healing_is_autoheal_enabled(self, mock_inst):
self.config_fixture.config(
group='prometheus_plugin', auto_healing=True)
mock_inst.return_value = objects.VnfInstanceV2.from_dict(_inst5)
result = self.controller.auto_healing(self.request, _body_heal)
self.assertEqual(204, result.status)
@mock.patch.object(inst_utils, 'get_inst')
def test_auto_healing_multiple_continue(self, mock_inst):
self.config_fixture.config(
group='prometheus_plugin', auto_healing=True)
mock_inst.return_value = objects.VnfInstanceV2.from_dict(_inst5)
result = self.controller.auto_healing(
self.request, _body_heal_continue)
self.assertEqual(204, result.status)
@mock.patch.object(inst_utils, 'get_inst')
def test_auto_healing_not_instantiated(self, mock_inst):
self.config_fixture.config(
group='prometheus_plugin', auto_healing=True)
mock_inst.return_value = objects.VnfInstanceV2.from_dict(_inst1)
result = self.controller.auto_healing(
self.request, _body_heal)
self.assertEqual(204, result.status)
class TestPrometheusPluginAutoScaling(base.TestCase): class TestPrometheusPluginAutoScaling(base.TestCase):
def setUp(self): def setUp(self):
super(TestPrometheusPluginAutoScaling, self).setUp() super(TestPrometheusPluginAutoScaling, self).setUp()
@ -344,15 +485,15 @@ class TestPrometheusPluginAutoScaling(base.TestCase):
group='prometheus_plugin', auto_scaling=False) group='prometheus_plugin', auto_scaling=False)
self.assertRaises( self.assertRaises(
sol_ex.PrometheusPluginNotEnabled, sol_ex.PrometheusPluginNotEnabled,
self.controller.auto_scaling_id, self.request, 'id', {}) self.controller.auto_scaling, self.request, {})
@mock.patch.object(inst_utils, 'get_inst') @mock.patch.object(inst_utils, 'get_inst')
def test_auto_scaling_no_autoscale_enabled(self, mock_inst): def test_auto_scaling_no_autoscale_enabled(self, mock_inst):
self.config_fixture.config( self.config_fixture.config(
group='prometheus_plugin', auto_scaling=True) group='prometheus_plugin', auto_scaling=True)
mock_inst.return_value = objects.VnfInstanceV2.from_dict(_inst1) mock_inst.return_value = objects.VnfInstanceV2.from_dict(_inst3)
result = self.controller.auto_scaling_id( result = self.controller.auto_scaling(
self.request, 'id', _body_scale) self.request, _body_scale)
self.assertEqual(204, result.status) self.assertEqual(204, result.status)
@mock.patch.object(inst_utils, 'get_inst') @mock.patch.object(inst_utils, 'get_inst')
@ -378,3 +519,21 @@ class TestPrometheusPluginAutoScaling(base.TestCase):
group='prometheus_plugin', auto_scaling=True) group='prometheus_plugin', auto_scaling=True)
result = self.controller.auto_scaling(self.request, {}) result = self.controller.auto_scaling(self.request, {})
self.assertEqual(204, result.status) self.assertEqual(204, result.status)
@mock.patch.object(inst_utils, 'get_inst')
def test_auto_scaling_multiple_continue(self, mock_inst):
self.config_fixture.config(
group='prometheus_plugin', auto_scaling=True)
mock_inst.return_value = objects.VnfInstanceV2.from_dict(_inst2)
result = self.controller.auto_scaling(
self.request, _body_scale_continue)
self.assertEqual(204, result.status)
@mock.patch.object(inst_utils, 'get_inst')
def test_auto_scaling_not_instantiated(self, mock_inst):
self.config_fixture.config(
group='prometheus_plugin', auto_scaling=True)
mock_inst.return_value = objects.VnfInstanceV2.from_dict(_inst1)
result = self.controller.auto_scaling(
self.request, _body_scale)
self.assertEqual(204, result.status)