diff options
author | Priyank Maheshwari <priyank.maheshwari@est.tech> | 2023-08-25 13:54:17 +0000 |
---|---|---|
committer | Gerrit Code Review <gerrit@onap.org> | 2023-08-25 13:54:17 +0000 |
commit | 7dbd1856887a3e8a315752a10d7a0bf83d7da077 (patch) | |
tree | 5617da8492ae8f8f23991eca0b146f1b8730fb99 /cps-ncmp-service/src/test/groovy | |
parent | e8a7c4aac83b05ef3d19bf78174ee5f68e0c9c3e (diff) | |
parent | 4c5740094a9e7f84b72e79f691a34babbf6be344 (diff) |
Merge "Extend model loader to support model-upgrade (part 1)"
Diffstat (limited to 'cps-ncmp-service/src/test/groovy')
2 files changed, 207 insertions, 157 deletions
diff --git a/cps-ncmp-service/src/test/groovy/org/onap/cps/ncmp/init/AbstractModelLoaderSpec.groovy b/cps-ncmp-service/src/test/groovy/org/onap/cps/ncmp/init/AbstractModelLoaderSpec.groovy new file mode 100644 index 0000000000..a271ca4312 --- /dev/null +++ b/cps-ncmp-service/src/test/groovy/org/onap/cps/ncmp/init/AbstractModelLoaderSpec.groovy @@ -0,0 +1,191 @@ +/* + * ============LICENSE_START======================================================= + * Copyright (C) 2023 Nordix Foundation + * ================================================================================ + * 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. + * + * SPDX-License-Identifier: Apache-2.0 + * ============LICENSE_END========================================================= + */ + +package org.onap.cps.ncmp.init + +import ch.qos.logback.classic.Level +import ch.qos.logback.classic.Logger +import ch.qos.logback.core.read.ListAppender +import org.onap.cps.api.CpsAdminService +import org.onap.cps.api.CpsDataService +import org.onap.cps.api.CpsModuleService +import org.onap.cps.ncmp.api.impl.exception.NcmpStartUpException +import org.onap.cps.spi.exceptions.AlreadyDefinedException +import org.springframework.boot.SpringApplication +import org.slf4j.LoggerFactory +import org.springframework.boot.context.event.ApplicationReadyEvent +import org.springframework.context.annotation.AnnotationConfigApplicationContext +import spock.lang.Specification + +class AbstractModelLoaderSpec extends Specification { + + def mockCpsAdminService = Mock(CpsAdminService) + def mockCpsModuleService = Mock(CpsModuleService) + def mockCpsDataService = Mock(CpsDataService) + def objectUnderTest = Spy(new TestModelLoader(mockCpsAdminService, mockCpsModuleService, mockCpsDataService)) + + def applicationContext = new AnnotationConfigApplicationContext() + + def yangResourceToContentMap + def logger = (Logger) LoggerFactory.getLogger(AbstractModelLoader) + def loggingListAppender + + void setup() { + yangResourceToContentMap = objectUnderTest.createYangResourceToContentMap('subscription.yang') + logger.setLevel(Level.DEBUG) + loggingListAppender = new ListAppender() + logger.addAppender(loggingListAppender) + loggingListAppender.start() + applicationContext.refresh() + } + + void cleanup() { + ((Logger) LoggerFactory.getLogger(SubscriptionModelLoader.class)).detachAndStopAllAppenders() + applicationContext.close() + } + + def 'Application ready event'() { + when: 'Application (ready) event is triggered' + objectUnderTest.onApplicationEvent(Mock(ApplicationReadyEvent)) + then: 'the onboard/upgrade method is executed' + 1 * objectUnderTest.onboardOrUpgradeModel() + } + + def 'Application ready event with start up exception'() { + given: 'a start up exception is thrown doing model onboarding' + objectUnderTest.onboardOrUpgradeModel() >> { throw new NcmpStartUpException('test message','details are not logged') } + when: 'Application (ready) event is triggered' + objectUnderTest.onApplicationEvent(new ApplicationReadyEvent(new SpringApplication(), null, applicationContext, null)) + then: 'the exception message is logged' + def logs = loggingListAppender.list.toString() + assert logs.contains('test message') + } + + def 'Wait for non-existing dataspace'() { + when: 'wait for the dataspace' + objectUnderTest.waitUntilDataspaceIsAvailable('some dataspace') + then: 'a startup exception is thrown' + def thrown = thrown(NcmpStartUpException) + assert thrown.message.contains('Retrieval of NCMP dataspace failed') + } + + def 'Create schema set.'() { + when: 'creating a schema set' + objectUnderTest.createSchemaSet('some dataspace','new name','subscription.yang') + then: 'the operation is delegated to the admin service' + 1 * mockCpsModuleService.createSchemaSet('some dataspace','new name',_) + } + + def 'Create schema set with already defined exception.'() { + given: 'the module service throws an already defined exception' + mockCpsModuleService.createSchemaSet(*_) >> { throw AlreadyDefinedException.forSchemaSet('name','context',null) } + when: 'attempt to create a schema set' + objectUnderTest.createSchemaSet('some dataspace','new name','subscription.yang') + then: 'the exception is ignored i.e. no exception thrown up' + noExceptionThrown() + and: 'the exception message is logged' + def logs = loggingListAppender.list.toString() + assert logs.contains('Creating new schema set failed as schema set already exists') + } + + def 'Create schema set with non existing yang file.'() { + when: 'attempt to create a schema set from a non existing file' + objectUnderTest.createSchemaSet('some dataspace','some name','no such yang file') + then: 'a startup exception with correct message and details is thrown' + def thrown = thrown(NcmpStartUpException) + assert thrown.message.contains('Creating schema set failed') + assert thrown.details.contains('unable to read file') + } + + def 'Create anchor.'() { + when: 'creating an anchor' + objectUnderTest.createAnchor('some dataspace','some schema set','new name') + then: 'thr operation is delegated to the admin service' + 1 * mockCpsAdminService.createAnchor('some dataspace','some schema set', 'new name') + } + + def 'Create anchor with already defined exception.'() { + given: 'the admin service throws an already defined exception' + mockCpsAdminService.createAnchor(*_)>> { throw AlreadyDefinedException.forAnchor('name','context',null) } + when: 'attempt to create anchor' + objectUnderTest.createAnchor('some dataspace','some schema set','new name') + then: 'the exception is ignored i.e. no exception thrown up' + noExceptionThrown() + and: 'the exception message is logged' + def logs = loggingListAppender.list.toString() + assert logs.contains('Creating new anchor failed as anchor already exists') + } + + def 'Create anchor with any other exception.'() { + given: 'the admin service throws a exception' + mockCpsAdminService.createAnchor(*_)>> { throw new RuntimeException('test message') } + when: 'attempt to create anchor' + objectUnderTest.createAnchor('some dataspace','some schema set','new name') + then: 'a startup exception with correct message and details is thrown' + def thrown = thrown(NcmpStartUpException) + assert thrown.message.contains('Creating anchor failed') + assert thrown.details.contains('test message') + } + + def 'Create top level node.'() { + when: 'top level node is created' + objectUnderTest.createTopLevelDataNode('dataspace','anchor','new node') + then: 'the correct json is saved using the data service' + 1 * mockCpsDataService.saveData('dataspace','anchor', '{"new node":{}}',_) + } + + def 'Create top level node with already defined exception.'() { + given: 'the data service throws an Already Defined exception' + mockCpsDataService.saveData(*_) >> { throw AlreadyDefinedException.forDataNodes([], 'some context') } + when: 'attempt to create top level node' + objectUnderTest.createTopLevelDataNode('dataspace','anchor','new node') + then: 'the exception is ignored i.e. no exception thrown up' + noExceptionThrown() + and: 'the exception message is logged' + def logs = loggingListAppender.list.toString() + assert logs.contains('failed as data node already exists') + } + + def 'Create top level node with any other exception.'() { + given: 'the data service throws an exception' + mockCpsDataService.saveData(*_) >> { throw new RuntimeException('test message') } + when: 'attempt to create top level node' + objectUnderTest.createTopLevelDataNode('dataspace','anchor','new node') + then: 'a startup exception with correct message and details is thrown' + def thrown = thrown(NcmpStartUpException) + assert thrown.message.contains('Creating data node failed') + assert thrown.details.contains('test message') + } + + class TestModelLoader extends AbstractModelLoader { + + TestModelLoader(final CpsAdminService cpsAdminService, + final CpsModuleService cpsModuleService, + final CpsDataService cpsDataService) { + super(cpsAdminService, cpsModuleService, cpsDataService) + super.maximumAttemptCount = 2 + super.retryTimeMs = 1 + } + + @Override + void onboardOrUpgradeModel() { } + } + +} diff --git a/cps-ncmp-service/src/test/groovy/org/onap/cps/ncmp/init/SubscriptionModelLoaderSpec.groovy b/cps-ncmp-service/src/test/groovy/org/onap/cps/ncmp/init/SubscriptionModelLoaderSpec.groovy index 23fa357eed..305fe4c062 100644 --- a/cps-ncmp-service/src/test/groovy/org/onap/cps/ncmp/init/SubscriptionModelLoaderSpec.groovy +++ b/cps-ncmp-service/src/test/groovy/org/onap/cps/ncmp/init/SubscriptionModelLoaderSpec.groovy @@ -26,12 +26,7 @@ import ch.qos.logback.core.read.ListAppender import org.onap.cps.api.CpsAdminService import org.onap.cps.api.CpsDataService import org.onap.cps.api.CpsModuleService -import org.onap.cps.ncmp.api.impl.exception.NcmpStartUpException -import org.onap.cps.spi.exceptions.AlreadyDefinedException -import org.onap.cps.spi.exceptions.DataValidationException -import org.onap.cps.spi.exceptions.SchemaSetNotFoundException import org.onap.cps.spi.model.Dataspace -import org.springframework.boot.SpringApplication import org.slf4j.LoggerFactory import org.springframework.boot.context.event.ApplicationReadyEvent import org.springframework.context.annotation.AnnotationConfigApplicationContext @@ -44,18 +39,14 @@ class SubscriptionModelLoaderSpec extends Specification { def mockCpsDataService = Mock(CpsDataService) def objectUnderTest = new SubscriptionModelLoader(mockCpsAdminService, mockCpsModuleService, mockCpsDataService) - def sampleYangContentMap = ['subscription.yang':'module subscription { *sample content* }'] - def applicationContext = new AnnotationConfigApplicationContext() - def applicationReadyEvent = new ApplicationReadyEvent(new SpringApplication(), null, applicationContext, null) - def yangResourceToContentMap - def logger = (Logger) LoggerFactory.getLogger(objectUnderTest.getClass()) + def logger = (Logger) LoggerFactory.getLogger(objectUnderTest.class) def loggingListAppender void setup() { - yangResourceToContentMap = objectUnderTest.createYangResourceToContentMap() + yangResourceToContentMap = objectUnderTest.createYangResourceToContentMap('subscription.yang') logger.setLevel(Level.DEBUG) loggingListAppender = new ListAppender() logger.addAppender(loggingListAppender) @@ -68,163 +59,31 @@ class SubscriptionModelLoaderSpec extends Specification { applicationContext.close() } - def 'Onboard subscription model successfully via application ready event'() { - given: 'dataspace is ready for use' - mockCpsAdminService.getDataspace('NCMP-Admin') >> new Dataspace('NCMP-Admin') - and:'model loader is enabled' + def 'Onboard subscription model via application ready event.'() { + given:'model loader is enabled' objectUnderTest.subscriptionModelLoaderEnabled = true - and: 'maximum attempt count is set' - objectUnderTest.maximumAttemptCount = 20 - and: 'retry time is set' - objectUnderTest.retryTimeMs = 100 + and: 'dataspace is ready for use' + mockCpsAdminService.getDataspace('NCMP-Admin') >> new Dataspace('') when: 'the application is ready' - objectUnderTest.onApplicationEvent(applicationReadyEvent) + objectUnderTest.onApplicationEvent(Mock(ApplicationReadyEvent)) then: 'the module service to create schema set is called once' - 1 * mockCpsModuleService.createSchemaSet('NCMP-Admin', 'subscriptions',sampleYangContentMap) + 1 * mockCpsModuleService.createSchemaSet('NCMP-Admin', 'subscriptions', yangResourceToContentMap) and: 'the admin service to create an anchor set is called once' 1 * mockCpsAdminService.createAnchor('NCMP-Admin', 'subscriptions', 'AVC-Subscriptions') and: 'the data service to create a top level datanode is called once' 1 * mockCpsDataService.saveData('NCMP-Admin', 'AVC-Subscriptions', '{"subscription-registry":{}}', _) } - def 'No subscription model onboarding when subscription model loader is disabled' () { - when: 'model loader is disabled' + def 'Subscription model loader disabled.' () { + given: 'model loader is disabled' objectUnderTest.subscriptionModelLoaderEnabled = false - and: 'application is ready' - objectUnderTest.onApplicationEvent(applicationReadyEvent) - and: 'dataspace is ready for use' - mockCpsAdminService.getDataspace('NCMP-Admin') >> new Dataspace('NCMP-Admin') - then: 'the module service to create schema set was not called' - 0 * mockCpsModuleService.createSchemaSet(*_) - and: 'the admin service to create an anchor set was not called' - 0 * mockCpsAdminService.createAnchor(*_) - and: 'the data service to create a top level datanode was not called' - 0 * mockCpsDataService.saveData(*_) - } - - def 'Onboard subscription model fails as NCMP dataspace does not exist' () { - given: 'model loader is enabled' - objectUnderTest.subscriptionModelLoaderEnabled = true - and: 'maximum attempt count is set' - objectUnderTest.maximumAttemptCount = 20 - and: 'retry time is set' - objectUnderTest.retryTimeMs = 100 - when: 'the application is ready' - objectUnderTest.onApplicationEvent(applicationReadyEvent) - then: 'the module service to create schema set was not called' - 0 * mockCpsModuleService.createSchemaSet(*_) - and: 'the admin service to create an anchor set was not called' - 0 * mockCpsAdminService.createAnchor(*_) - and: 'the data service to create a top level datanode was not called' - 0 * mockCpsDataService.saveData(*_) - and: 'the log message contains the correct exception message' - def logs = loggingListAppender.list.toString() - assert logs.contains("Retrieval of NCMP dataspace fails") - } - - - def 'Exception occurred while schema set creation' () { - given: 'creating a schema set throws an exception' - mockCpsModuleService.createSchemaSet(*_) >> { throw new DataValidationException(*_) } - and: 'model loader is enabled' - objectUnderTest.subscriptionModelLoaderEnabled = true - and: 'dataspace is ready for use' - mockCpsAdminService.getDataspace('NCMP-Admin') >> new Dataspace('NCMP-Admin') when: 'application is ready' - objectUnderTest.onApplicationEvent(applicationReadyEvent) - then: 'the admin service to create an anchor set was not called' - 0 * mockCpsAdminService.createAnchor(*_) - and: 'the data service to create a top level datanode was not called' - 0 * mockCpsDataService.saveData(*_) - } - - def 'Create schema set from model file'() { - when: 'the method to create schema set is called with the following parameters' - objectUnderTest.createSchemaSet("myDataspace", "mySchemaSet", yangResourceToContentMap) - then: 'yang resource to content map is as expected' - assert sampleYangContentMap == yangResourceToContentMap - and: 'the module service to create schema set is called once with the correct map' - 1 * mockCpsModuleService.createSchemaSet(_, _, yangResourceToContentMap) - } - - def 'Create schema set fails due to AlreadyDefined exception'() { - given: 'creating a schema set throws an exception as it already exists' - mockCpsModuleService.createSchemaSet('NCMP-Admin', 'subscriptions', yangResourceToContentMap) >> - { throw AlreadyDefinedException.forSchemaSet('subscriptions', "sampleContextName", null) } - when: 'the method to onboard model is called' - objectUnderTest.onboardSubscriptionModel(yangResourceToContentMap) - then: 'the admin service to create an anchor set is then called once' - 1 * mockCpsAdminService.createAnchor('NCMP-Admin', 'subscriptions', 'AVC-Subscriptions') - } - - def 'Create schema set fails due to any other exception'() { - given: 'creating a schema set throws an exception' - mockCpsModuleService.createSchemaSet(*_) >> { throw new NcmpStartUpException("Creating schema set failed", "") } - when: 'the method to onboard model is called' - objectUnderTest.onboardSubscriptionModel(yangResourceToContentMap) - then: 'the log message contains the correct exception message' - def debugMessage = loggingListAppender.list[0].toString() - assert debugMessage.contains("Creating schema set failed") - and: 'exception is thrown' - thrown(NcmpStartUpException) - } - - def 'Create anchor fails due to AlreadyDefined exception'() { - given: 'creating anchor throws an exception as it already exists' - mockCpsAdminService.createAnchor(*_) >> - { throw AlreadyDefinedException.forSchemaSet('subscriptions', "sampleContextName", null) } - when: 'the method to onboard model is called' - objectUnderTest.onboardSubscriptionModel(yangResourceToContentMap) - then: 'no exception thrown' - noExceptionThrown() - and: 'the log message contains the correct exception message' - def infoMessage = loggingListAppender.list[0].toString() - assert infoMessage.contains("already exists") - } - - def 'Create anchor fails due to any other exception'() { - given: 'creating an anchor failed' - mockCpsAdminService.createAnchor(*_) >> - { throw new SchemaSetNotFoundException('NCMP-Admin', 'subscriptions') } - when: 'the method to onboard model is called' - objectUnderTest.onboardSubscriptionModel(yangResourceToContentMap) - then: 'the log message contains the correct exception message' - def debugMessage = loggingListAppender.list[0].toString() - assert debugMessage.contains("Schema Set not found") - and: 'exception is thrown' - thrown(NcmpStartUpException) - } - - def 'Create top level node fails due to an AlreadyDefined exception'() { - given: 'the saving of the node data will throw an Already Defined exception' - mockCpsDataService.saveData(*_) >> - { throw AlreadyDefinedException.forDataNodes(['/xpath'], "sampleContextName") } - when: 'the method to onboard model is called' - objectUnderTest.onboardSubscriptionModel(yangResourceToContentMap) - then: 'no exception thrown' - noExceptionThrown() - and: 'the log message contains the correct exception message' - def infoMessage = loggingListAppender.list[0].toString() - assert infoMessage.contains("already exists") - } - - def 'Create top level node fails due to any other exception'() { - given: 'the saving of the node data will throw an exception' - mockCpsDataService.saveData(*_) >> - { throw new DataValidationException("Invalid JSON", "JSON Data is invalid") } - when: 'the method to onboard model is called' - objectUnderTest.onboardSubscriptionModel(yangResourceToContentMap) - then: 'the log message contains the correct exception message' - def debugMessage = loggingListAppender.list[0].toString() - assert debugMessage.contains("Creating data node for subscription model failed: Invalid JSON") - and: 'exception is thrown' - thrown(NcmpStartUpException) + objectUnderTest.onApplicationEvent(Mock(ApplicationReadyEvent)) + then: 'no interaction with admin service' + 0 * mockCpsAdminService.getDataspace(_) + then: 'a message is logged that the function is disabled' + def logs = loggingListAppender.list.toString() + assert logs.contains('Subscription Model Loader is disabled') } - def 'Get file content as string'() { - when: 'the method to get yang content is called' - objectUnderTest.getFileContentAsString('NonExistingFile') - then: 'exception is thrown' - thrown(NcmpStartUpException) - } } |