2  *  ============LICENSE_START=======================================================
 
   3  *  Copyright (C) 2022-2024 Nordix Foundation
 
   4  *  Modifications Copyright (C) 2022 Bell Canada
 
   5  *  ================================================================================
 
   6  *  Licensed under the Apache License, Version 2.0 (the "License");
 
   7  *  you may not use this file except in compliance with the License.
 
   8  *  You may obtain a copy of the License at
 
  10  *        http://www.apache.org/licenses/LICENSE-2.0
 
  12  *  Unless required by applicable law or agreed to in writing, software
 
  13  *  distributed under the License is distributed on an "AS IS" BASIS,
 
  14  *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 
  15  *  See the License for the specific language governing permissions and
 
  16  *  limitations under the License.
 
  18  *  SPDX-License-Identifier: Apache-2.0
 
  19  *  ============LICENSE_END=========================================================
 
  22 package org.onap.cps.ncmp.impl.inventory.sync
 
  24 import ch.qos.logback.classic.Level
 
  25 import ch.qos.logback.classic.Logger
 
  26 import ch.qos.logback.core.read.ListAppender
 
  27 import com.fasterxml.jackson.databind.JsonNode
 
  28 import com.fasterxml.jackson.databind.ObjectMapper
 
  29 import org.onap.cps.ncmp.api.inventory.models.CompositeState
 
  30 import org.onap.cps.ncmp.api.inventory.models.CompositeStateBuilder
 
  31 import org.onap.cps.ncmp.impl.data.DmiDataOperations
 
  32 import org.onap.cps.ncmp.impl.inventory.CmHandleQueryService
 
  33 import org.onap.cps.ncmp.impl.inventory.DataStoreSyncState
 
  34 import org.onap.cps.ncmp.impl.inventory.models.CmHandleState
 
  35 import org.onap.cps.spi.FetchDescendantsOption
 
  36 import org.onap.cps.spi.model.DataNode
 
  37 import org.onap.cps.utils.JsonObjectMapper
 
  38 import org.slf4j.LoggerFactory
 
  39 import org.springframework.context.annotation.AnnotationConfigApplicationContext
 
  40 import org.springframework.http.HttpStatus
 
  41 import org.springframework.http.ResponseEntity
 
  42 import spock.lang.Specification
 
  44 import java.time.OffsetDateTime
 
  45 import java.time.format.DateTimeFormatter
 
  46 import java.util.stream.Collectors
 
  48 import static org.onap.cps.ncmp.impl.inventory.models.LockReasonCategory.LOCKED_MISBEHAVING
 
  49 import static org.onap.cps.ncmp.impl.inventory.models.LockReasonCategory.MODULE_SYNC_FAILED
 
  50 import static org.onap.cps.ncmp.impl.inventory.models.LockReasonCategory.MODULE_UPGRADE
 
  51 import static org.onap.cps.ncmp.impl.inventory.models.LockReasonCategory.MODULE_UPGRADE_FAILED
 
  53 class ModuleOperationsUtilsSpec extends Specification{
 
  55     def mockCmHandleQueries = Mock(CmHandleQueryService)
 
  57     def mockDmiDataOperations = Mock(DmiDataOperations)
 
  59     def jsonObjectMapper = new JsonObjectMapper(new ObjectMapper())
 
  61     def objectUnderTest = new ModuleOperationsUtils(mockCmHandleQueries, mockDmiDataOperations, jsonObjectMapper)
 
  63     def static neverUpdatedBefore = '1900-01-01T00:00:00.000+0100'
 
  65     def static now = OffsetDateTime.now()
 
  67     def static nowAsString = DateTimeFormatter.ofPattern("yyyy-MM-dd'T'HH:mm:ss.SSSZ").format(now)
 
  69     def static dataNode = new DataNode(leaves: ['id': 'cm-handle-123'])
 
  71     def applicationContext = new AnnotationConfigApplicationContext()
 
  73     def logger = (Logger) LoggerFactory.getLogger(ModuleOperationsUtils)
 
  74     def loggingListAppender
 
  77         logger.setLevel(Level.DEBUG)
 
  78         loggingListAppender = new ListAppender()
 
  79         logger.addAppender(loggingListAppender)
 
  80         loggingListAppender.start()
 
  81         applicationContext.refresh()
 
  85         ((Logger) LoggerFactory.getLogger(ModuleOperationsUtils.class)).detachAndStopAllAppenders()
 
  86         applicationContext.close()
 
  89     def 'Get an advised Cm-Handle where ADVISED cm handle #scenario'() {
 
  90         given: 'the inventory persistence service returns a collection of data nodes'
 
  91             mockCmHandleQueries.queryCmHandlesByState(CmHandleState.ADVISED) >> dataNodeCollection
 
  92         when: 'get advised cm handles are fetched'
 
  93             def yangModelCmHandles = objectUnderTest.getAdvisedCmHandles()
 
  94         then: 'the returned data node collection is the correct size'
 
  95             yangModelCmHandles.size() == expectedDataNodeSize
 
  96         where: 'the following scenarios are used'
 
  97             scenario         | dataNodeCollection || expectedCallsToGetYangModelCmHandle | expectedDataNodeSize
 
  98             'exists'         | [dataNode]         || 1                                   | 1
 
  99             'does not exist' | []                 || 0                                   | 0
 
 102     def 'Update Lock Reason, Details and Attempts where lock reason #scenario'() {
 
 103         given: 'A locked state'
 
 104             def compositeState = new CompositeState(lockReason: lockReason)
 
 105         when: 'update cm handle details and attempts is called'
 
 106             objectUnderTest.updateLockReasonDetailsAndAttempts(compositeState, MODULE_SYNC_FAILED, 'new error message')
 
 107         then: 'the composite state lock reason and details are updated'
 
 108             assert compositeState.lockReason.lockReasonCategory == MODULE_SYNC_FAILED
 
 109             assert compositeState.lockReason.details.contains(expectedDetails)
 
 111             scenario                           | lockReason                                                                                       || expectedDetails
 
 112             'does not exist'                   | null                                                                                             || 'Attempt #1 failed: new error message'
 
 113             'exists'                           | CompositeState.LockReason.builder().details("Attempt #2 failed: some error message").build()     || 'Attempt #3 failed: new error message'
 
 116     def 'Update lock reason details that contains #scenario'() {
 
 117         given: 'A locked state'
 
 118             def compositeState = new CompositeStateBuilder().withCmHandleState(CmHandleState.LOCKED)
 
 119                 .withLockReason(MODULE_UPGRADE, "Upgrade to ModuleSetTag: " + moduleSetTag).build()
 
 120         when: 'update cm handle details'
 
 121             objectUnderTest.updateLockReasonDetailsAndAttempts(compositeState, MODULE_UPGRADE_FAILED, 'new error message')
 
 122         then: 'the composite state lock reason and details are updated'
 
 123             assert compositeState.lockReason.lockReasonCategory == MODULE_UPGRADE_FAILED
 
 124             assert compositeState.lockReason.details.contains("Upgrade to ModuleSetTag: " + expectedDetails)
 
 126             scenario               | moduleSetTag       || expectedDetails
 
 127             'a module set tag'     | 'someModuleSetTag' || 'someModuleSetTag'
 
 128             'empty module set tag' | ''                 || ''
 
 131     def 'Get all locked cm-Handles where lock reasons are model sync failed or upgrade'() {
 
 132         given: 'the cps (persistence service) returns a collection of data nodes'
 
 133             mockCmHandleQueries.queryCmHandleAncestorsByCpsPath(ModuleOperationsUtils.CPS_PATH_CM_HANDLES_MODEL_SYNC_FAILED_OR_UPGRADE,
 
 134                 FetchDescendantsOption.INCLUDE_ALL_DESCENDANTS) >> [dataNode]
 
 135         when: 'get locked Misbehaving cm handle is called'
 
 136             def result = objectUnderTest.getCmHandlesThatFailedModelSyncOrUpgrade()
 
 137         then: 'the returned cm handle collection is the correct size'
 
 139         and: 'the correct cm handle is returned'
 
 140             result[0].id == 'cm-handle-123'
 
 143     def 'Retry Locked Cm-Handle where the last update time is #scenario'() {
 
 144         given: 'Last update was #lastUpdateMinutesAgo minutes ago (-1 means never)'
 
 145             def lastUpdatedTime = DateTimeFormatter.ofPattern("yyyy-MM-dd'T'HH:mm:ss.SSSZ").format(now.minusMinutes(lastUpdateMinutesAgo))
 
 146             if (lastUpdateMinutesAgo < 0 ) {
 
 147                 lastUpdatedTime = neverUpdatedBefore
 
 149         when: 'checking to see if cm handle is ready for retry'
 
 150          def result = objectUnderTest.needsModuleSyncRetryOrUpgrade(new CompositeStateBuilder()
 
 151                 .withLockReason(MODULE_SYNC_FAILED, lockDetails)
 
 152                 .withLastUpdatedTime(lastUpdatedTime).build())
 
 153         then: 'retry is only attempted when expected'
 
 154             assert result == retryExpected
 
 155         and: 'logs contain related information'
 
 156             def logs = loggingListAppender.list.toString()
 
 157             assert logs.contains(logReason)
 
 158         where: 'the following parameters are used'
 
 159             scenario                                    | lastUpdateMinutesAgo | lockDetails                     | logReason                               || retryExpected
 
 160             'never attempted before'                    | -1                   | 'Fist attempt:'                 | 'First Attempt:'                        || true
 
 161             '1st attempt, last attempt > 2 minute ago'  | 3                    | 'Attempt #1 failed: some error' | 'Retry due now'                         || true
 
 162             '2nd attempt, last attempt < 4 minutes ago' | 1                    | 'Attempt #2 failed: some error' | 'Time until next attempt is 3 minutes:' || false
 
 163             '2nd attempt, last attempt > 4 minutes ago' | 5                    | 'Attempt #2 failed: some error' | 'Retry due now'                         || true
 
 166     def 'Retry Locked Cm-Handle with lock reasons (category) #lockReasonCategory'() {
 
 167         when: 'checking to see if cm handle is ready for retry'
 
 168             def result = objectUnderTest.needsModuleSyncRetryOrUpgrade(new CompositeStateBuilder()
 
 169                 .withLockReason(lockReasonCategory, 'some details')
 
 170                 .withLastUpdatedTime(nowAsString).build())
 
 171         then: 'verify retry attempts'
 
 173         and: 'logs contain related information'
 
 174             def logs = loggingListAppender.list.toString()
 
 175             assert logs.contains(logReason)
 
 176         where: 'the following lock reasons occurred'
 
 177             scenario             | lockReasonCategory    || logReason
 
 178             'module upgrade'     | MODULE_UPGRADE_FAILED || 'First Attempt:'
 
 179             'module sync failed' | MODULE_SYNC_FAILED    || 'First Attempt:'
 
 180             'lock misbehaving'   | LOCKED_MISBEHAVING    || 'Locked for other reason'
 
 183     def 'Get a Cm-Handle where #scenario'() {
 
 184         given: 'the inventory persistence service returns a collection of data nodes'
 
 185             mockCmHandleQueries.queryCmHandlesByOperationalSyncState(DataStoreSyncState.UNSYNCHRONIZED) >> unSynchronizedDataNodes
 
 186             mockCmHandleQueries.cmHandleHasState('cm-handle-123', CmHandleState.READY) >> cmHandleHasState
 
 187         when: 'get advised cm handles are fetched'
 
 188             def yangModelCollection = objectUnderTest.getUnsynchronizedReadyCmHandles()
 
 189         then: 'the returned data node collection is the correct size'
 
 190             yangModelCollection.size() == expectedDataNodeSize
 
 191         and: 'the result contains the correct data'
 
 192             yangModelCollection.stream().map(yangModel -> yangModel.id).collect(Collectors.toSet()) == expectedYangModelCollectionIds
 
 193         where: 'the following scenarios are used'
 
 194             scenario                                   | unSynchronizedDataNodes | cmHandleHasState || expectedDataNodeSize | expectedYangModelCollectionIds
 
 195             'a Cm-Handle unsynchronized and ready'     | [dataNode]              | true             || 1                    | ['cm-handle-123'] as Set
 
 196             'a Cm-Handle unsynchronized but not ready' | [dataNode]              | false            || 0                    | [] as Set
 
 197             'all Cm-Handle synchronized'               | []                      | false            || 0                    | [] as Set
 
 200     def 'Get resource data through DMI Operations #scenario'() {
 
 201         given: 'the inventory persistence service returns a collection of data nodes'
 
 202             def jsonString = '{"stores:bookstore":{"categories":[{"code":"01"}]}}'
 
 203             JsonNode jsonNode = jsonObjectMapper.convertToJsonNode(jsonString);
 
 204             def responseEntity = new ResponseEntity<>(jsonNode, HttpStatus.OK)
 
 205             mockDmiDataOperations.getAllResourceDataFromDmi('cm-handle-123', _) >> responseEntity
 
 206         when: 'get resource data is called'
 
 207             def result = objectUnderTest.getResourceData('cm-handle-123')
 
 208         then: 'the returned data is correct'
 
 212     def 'Extract module set tag and number of attempt when lock reason contains #scenario'() {
 
 213         expect: 'lock reason details are extracted correctly'
 
 214             def result = objectUnderTest.getLockedCompositeStateDetails(new CompositeStateBuilder().withLockReason(MODULE_UPGRADE, lockReasonDetails).build().lockReason)
 
 215         and: 'the result contains the correct moduleSetTag'
 
 216             assert result['moduleSetTag'] == expectedModuleSetTag
 
 217         and: 'the result contains the correct number of attempts'
 
 218             assert result['attempt'] == expectedNumberOfAttempts
 
 219         where: 'the following scenarios are used'
 
 220             scenario                                     | lockReasonDetails                                                           || expectedModuleSetTag | expectedNumberOfAttempts
 
 221             'module set tag only'                        | 'Upgrade to ModuleSetTag: targetModuleSetTag'                               || 'targetModuleSetTag' | null
 
 222             'number of attempts only'                    | 'Attempt #1 failed: some error'                                             || null                 | '1'
 
 223             'number of attempts and module set tag both' | 'Upgrade to ModuleSetTag: targetModuleSetTag Attempt #1 failed: some error' || 'targetModuleSetTag' | '1'