tacker/tacker/tests/functional/sol_v2/test_server_notification.py

441 lines
17 KiB
Python

# 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_v2_common import base_v2
from tacker.tests.functional.sol_v2_common import paramgen
from tacker.tests.functional.sol_v2_common import test_vnflcm_basic_common
test_count = 0
def make_alarm_id(header, body):
from tacker.tests.functional.sol_v2 import test_server_notification
id = 'alarm_id_' + str(test_server_notification.test_count)
return {'alarm_id': id}
@ddt.ddt
class ServerNotificationTest(test_vnflcm_basic_common.CommonVnfLcmTest):
@classmethod
def setUpClass(cls):
super(ServerNotificationTest, cls).setUpClass()
cur_dir = os.path.dirname(__file__)
# tacker/tests/etc...
# /functional/sol_v2
image_dir = os.path.join(
cur_dir, "../../etc/samples/etsi/nfv/common/Files/images")
image_file = "cirros-0.5.2-x86_64-disk.img"
image_path = os.path.abspath(os.path.join(image_dir, image_file))
# for basic lcms tests max pattern
basic_lcms_max_path = os.path.join(cur_dir, "../sol_v2_common/"
"samples/basic_lcms_max")
cls.vnf_pkg_1, cls.vnfd_id_1 = cls.create_vnf_package(
basic_lcms_max_path, image_path=image_path)
# 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)
# for update vnf test
update_vnf_path = os.path.join(cur_dir, "../sol_v2_common/"
"samples/update_vnf")
# no image contained
cls.vnf_pkg_3, cls.vnfd_id_3 = cls.create_vnf_package(update_vnf_path)
# for server_notification test
server_notification_path = os.path.join(
cur_dir, "../sol_v2_common/samples/server_notification")
# no image contained
cls.svn_pkg, cls.svn_id = cls.create_vnf_package(
server_notification_path)
@classmethod
def tearDownClass(cls):
super(ServerNotificationTest, cls).tearDownClass()
cls.delete_vnf_package(cls.vnf_pkg_1)
cls.delete_vnf_package(cls.vnf_pkg_2)
cls.delete_vnf_package(cls.vnf_pkg_3)
cls.delete_vnf_package(cls.svn_pkg)
def setUp(self):
super().setUp()
def _check_package_usage(self, is_nfvo, package_id, state='NOT_IN_USE'):
if not is_nfvo:
usage_state = self.get_vnf_package(package_id)['usageState']
self.assertEqual(state, usage_state)
def fault_notification_queueing_test(self):
"""Test Fault Notification with queueing
* About Test operations:
This test includes the following operations.
- 1. LCM-Create
- 2. LCM-Instantiate
- 3. ServerNotifier-Notify (multiple times)
- 4. LCM-Heal
- 5. LCM-Scale (SCALE_OUT)
- 6. LCM-Scale (SCALE_IN)
- 7. LCM-Terminate
- 8. LCM-Delete
"""
self.fault_notification_basic_test(repeat=3)
def fault_notification_basic_test(self, repeat=1):
"""Test Fault Notification basic
* About Test operations:
This test includes the following operations.
- 1. LCM-Create
- 2. LCM-Instantiate
- 3. ServerNotifier-Notify
- 4. LCM-Heal
- 5. LCM-Scale (SCALE_OUT)
- 6. LCM-Scale (SCALE_IN)
- 7. LCM-Terminate
- 8. LCM-Delete
"""
is_nfvo = False
# 0. Pre setting
create_req = paramgen.create_vnf_min(self.svn_id)
# Create subscription
callback_url = os.path.join(base_v2.MOCK_NOTIFY_CALLBACK_URL,
self._testMethodName)
callback_uri = ('http://localhost:'
f'{base_v2.FAKE_SERVER_MANAGER.SERVER_PORT}'
f'{callback_url}')
server_notification_uri = ('http://localhost:'
f'{base_v2.FAKE_SERVER_MANAGER.SERVER_PORT}'
'/server_notification')
base_v2.FAKE_SERVER_MANAGER.set_callback(
'POST',
'/server_notification',
status_code=200,
response_headers={"Content-Type": "application/json"},
callback=make_alarm_id
)
sub_req = paramgen.sub_create_min(callback_uri)
resp, body = self.create_subscription(sub_req)
self.assertEqual(201, resp.status_code)
self.check_resp_headers_in_create(resp)
sub_id = body['id']
# Test notification
self.assert_notification_get(callback_url)
# 1. LCM-Create
# ETSI NFV SOL003 v3.3.1 5.5.2.2 VnfInstance
expected_inst_attrs = [
'id',
'vnfdId',
'vnfProvider',
'vnfProductName',
'vnfSoftwareVersion',
'vnfdVersion',
'instantiationState',
'_links'
]
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']
# check usageState of VNF Package
self._check_package_usage(is_nfvo, self.svn_pkg, 'IN_USE')
# 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'])
# 2. LCM-Instantiate
instantiate_req = paramgen.instantiate_vnf_min()
instantiate_req['additionalParams'] = {
'ServerNotifierUri': server_notification_uri,
'ServerNotifierFaultID': '1234'
}
resp, body = self.instantiate_vnf_instance(inst_id, instantiate_req)
self.assertEqual(202, resp.status_code)
self.check_resp_headers_in_operation_task(resp)
lcmocc_id = os.path.basename(resp.headers['Location'])
self.wait_lcmocc_complete(lcmocc_id)
# check creation of Heat-stack
stack_name = f'vnf-{inst_id}'
stack_status, _ = self.heat_client.get_status(stack_name)
self.assertEqual("CREATE_COMPLETE", stack_status)
# check that the servers set in "nfvi_node:Affinity" are
# deployed on the same host.
# NOTE: it's up to heat to decide which host to deploy to
vdu1_details = self.get_server_details('VDU1')
vdu2_details = self.get_server_details('VDU2')
vdu1_host = vdu1_details['hostId']
vdu2_host = vdu2_details['hostId']
self.assertEqual(vdu1_host, vdu2_host)
# Show VNF instance
additional_inst_attrs = [
'vimConnectionInfo',
'instantiatedVnfInfo'
]
expected_inst_attrs.extend(additional_inst_attrs)
resp, body = self.show_vnf_instance(inst_id)
self.assertEqual(200, resp.status_code)
self.check_resp_headers_in_get(resp)
self.check_resp_body(body, expected_inst_attrs)
# check instantiationState of VNF
self.assertEqual(fields.VnfInstanceState.INSTANTIATED,
body['instantiationState'])
# check vnfState of VNF
self.assertEqual(fields.VnfOperationalStateType.STARTED,
body['instantiatedVnfInfo']['vnfState'])
# 3. ServerNotifier-Notify
for i in range(repeat):
alarm_id = 'alarm_id_0'
fault_notification_param = paramgen.server_notification(alarm_id)
resp, body = self.server_notification(
inst_id, 'server_id', fault_notification_param)
self.assertTrue(resp.status_code == 204 or resp.status_code == 404)
time.sleep(1)
# waiting for auto healing process complete after packing timer.
time.sleep(60)
# 4. LCM-Heal
nested_stacks = self.heat_client.get_resources(stack_name)
temp_stacks = [stack for stack in nested_stacks if
(stack['resource_name'] in ['VDU1', 'VDU2'])]
vdu1_stack_before_heal = [stack for stack in temp_stacks if
(stack['resource_name'] == 'VDU1')][0]
vdu2_stack_before_heal = [stack for stack in temp_stacks if
(stack['resource_name'] == 'VDU2')][0]
heal_req = paramgen.heal_vnf_all_min()
resp, body = self.heal_vnf_instance(inst_id, heal_req)
self.assertEqual(202, resp.status_code)
self.check_resp_headers_in_operation_task(resp)
lcmocc_id = os.path.basename(resp.headers['Location'])
self.wait_lcmocc_complete(lcmocc_id)
# check stack info
stack_status, _ = self.heat_client.get_status(stack_name)
self.assertEqual("UPDATE_COMPLETE", stack_status)
nested_stacks = self.heat_client.get_resources(stack_name)
temp_stacks = [stack for stack in nested_stacks if
(stack['resource_name'] in ['VDU1', 'VDU2'])]
vdu1_stack_after_heal = [stack for stack in temp_stacks if
(stack['resource_name'] == 'VDU1')][0]
vdu2_stack_after_heal = [stack for stack in temp_stacks if
(stack['resource_name'] == 'VDU2')][0]
self.assertEqual("CREATE_COMPLETE",
vdu1_stack_after_heal['resource_status'])
self.assertEqual("CREATE_COMPLETE",
vdu2_stack_after_heal['resource_status'])
self.assertNotEqual(vdu1_stack_before_heal['physical_resource_id'],
vdu1_stack_after_heal['physical_resource_id'])
self.assertNotEqual(vdu2_stack_before_heal['physical_resource_id'],
vdu2_stack_after_heal['physical_resource_id'])
# Show VNF instance
additional_inst_attrs = [
'vimConnectionInfo',
'instantiatedVnfInfo'
]
expected_inst_attrs.extend(additional_inst_attrs)
resp, body = self.show_vnf_instance(inst_id)
self.assertEqual(200, resp.status_code)
self.check_resp_headers_in_get(resp)
self.check_resp_body(body, expected_inst_attrs)
# check instantiationState of VNF
self.assertEqual(fields.VnfInstanceState.INSTANTIATED,
body['instantiationState'])
# check vnfState of VNF
self.assertEqual(fields.VnfOperationalStateType.STARTED,
body['instantiatedVnfInfo']['vnfState'])
# check usageState of VNF Package 2
self._check_package_usage(is_nfvo, self.svn_pkg, 'IN_USE')
self.assertEqual(self.svn_id, body['vnfdId'])
# Heal VNF(vnfc)
nested_stacks = self.heat_client.get_resources(stack_name)
temp_stacks = [stack for stack in nested_stacks if
(stack['resource_name'] == 'VDU2')]
vdu2_stack_before_heal = temp_stacks[0]
resp, body = self.show_vnf_instance(inst_id)
self.assertEqual(200, resp.status_code)
vnfc_info = body['instantiatedVnfInfo']['vnfcInfo']
self.assertGreater(len(vnfc_info), 1)
vnfc_id = [vnfc['id'] for vnfc in vnfc_info if (
"VDU2" == vnfc['vduId'])][0]
self.assertIsNotNone(vnfc_id)
heal_req = paramgen.heal_vnf_vnfc_min(vnfc_id)
resp, body = self.heal_vnf_instance(inst_id, heal_req)
self.assertEqual(202, resp.status_code)
self.check_resp_headers_in_operation_task(resp)
lcmocc_id = os.path.basename(resp.headers['Location'])
self.wait_lcmocc_complete(lcmocc_id)
# check stack info
stack_status, _ = self.heat_client.get_status(stack_name)
self.assertEqual("UPDATE_COMPLETE", stack_status)
nested_stacks = self.heat_client.get_resources(stack_name)
temp_stacks = [stack for stack in nested_stacks if
(stack['resource_name'] == 'VDU2')]
vdu2_stack_after_heal = temp_stacks[0]
self.assertEqual("CREATE_COMPLETE",
vdu2_stack_after_heal['resource_status'])
self.assertNotEqual(vdu2_stack_before_heal['physical_resource_id'],
vdu2_stack_after_heal['physical_resource_id'])
# Show VNF instance
additional_inst_attrs = [
'vimConnectionInfo',
'instantiatedVnfInfo'
]
expected_inst_attrs.extend(additional_inst_attrs)
resp, body = self.show_vnf_instance(inst_id)
self.assertEqual(200, resp.status_code)
self.check_resp_headers_in_get(resp)
self.check_resp_body(body, expected_inst_attrs)
# check vnfState of VNF
self.assertEqual(fields.VnfOperationalStateType.STARTED,
body['instantiatedVnfInfo']['vnfState'])
# 5. LCM-Scale (SCALE_OUT)
# get nested stack count before scaleout
nested_stacks = self.heat_client.get_resources(stack_name)
count_before_scaleout = len(nested_stacks)
scaleout_req = paramgen.scaleout_vnf_min()
resp, body = self.scale_vnf_instance(inst_id, scaleout_req)
self.assertEqual(202, resp.status_code)
self.check_resp_headers_in_operation_task(resp)
lcmocc_id = os.path.basename(resp.headers['Location'])
self.wait_lcmocc_complete(lcmocc_id)
# Show VNF instance
additional_inst_attrs = [
'vimConnectionInfo',
'instantiatedVnfInfo'
]
expected_inst_attrs.extend(additional_inst_attrs)
resp, body = self.show_vnf_instance(inst_id)
self.assertEqual(200, resp.status_code)
self.check_resp_headers_in_get(resp)
self.check_resp_body(body, expected_inst_attrs)
# check vnfState of VNF
self.assertEqual(fields.VnfOperationalStateType.STARTED,
body['instantiatedVnfInfo']['vnfState'])
# get nested stack count after scale out
nested_stacks = self.heat_client.get_resources(stack_name)
count_after_scaleout = len(nested_stacks)
# check nested stack was created
# 3 was the sum of 1 VM, 1 CP, 1 stack(VDU1.yaml)
self.assertEqual(3, count_after_scaleout - count_before_scaleout)
# 6. LCM-Scale (SCALE_IN)
scalein_req = paramgen.scalein_vnf_min()
resp, body = self.scale_vnf_instance(inst_id, scalein_req)
self.assertEqual(202, resp.status_code)
self.check_resp_headers_in_operation_task(resp)
lcmocc_id = os.path.basename(resp.headers['Location'])
self.wait_lcmocc_complete(lcmocc_id)
# get nested stack count after scale in
nested_stacks = self.heat_client.get_resources(stack_name)
count_after_scalein = len(nested_stacks)
# check nested stack was deleted
# 3 was the sum of 1 VM, 1 CP, 1 stack(VDU1.yaml)
self.assertEqual(3, count_after_scaleout - count_after_scalein)
# 7. LCM-Terminate
terminate_req = paramgen.terminate_vnf_min()
resp, body = self.terminate_vnf_instance(inst_id, terminate_req)
self.assertEqual(202, resp.status_code)
self.check_resp_headers_in_operation_task(resp)
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(20)
# check deletion of Heat-stack
stack_status, _ = self.heat_client.get_status(stack_name)
self.assertIsNone(stack_status)
# 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'])
# 8. LCM-Delete
resp, body = self.delete_vnf_instance(inst_id)
self.assertEqual(204, resp.status_code)
self.check_resp_headers_in_delete(resp)
# check deletion of VNF instance
resp, body = self.show_vnf_instance(inst_id)
self.assertEqual(404, resp.status_code)
# Delete subscription
resp, body = self.delete_subscription(sub_id)
self.assertEqual(204, resp.status_code)
self.check_resp_headers_in_delete(resp)
# Show subscription
resp, body = self.show_subscription(sub_id)
self.assertEqual(404, resp.status_code)
def test_fault_notification_basic(self):
self.fault_notification_basic_test()
def test_fault_notification_queueing(self):
self.fault_notification_queueing_test()