Add gauge metric for NCMP "cmhandle states" 14/139714/11
authoremaclee <lee.anjella.macabuhay@est.tech>
Wed, 11 Dec 2024 14:58:25 +0000 (14:58 +0000)
committerLee Anjella Macabuhay <lee.anjella.macabuhay@est.tech>
Tue, 7 Jan 2025 12:02:29 +0000 (12:02 +0000)
Issue-ID: CPS-2456
Change-Id: I1aebcc68dfdc9c48c230c74376742d67b05c0615
Signed-off-by: emaclee <lee.anjella.macabuhay@est.tech>
cps-application/src/main/java/org/onap/cps/config/MicroMeterConfig.java
cps-application/src/main/resources/application.yml
cps-application/src/test/groovy/org/onap/cps/config/MicroMeterConfigSpec.groovy
cps-ncmp-service/src/main/java/org/onap/cps/ncmp/impl/cache/AdminCacheConfig.java [new file with mode: 0644]
cps-ncmp-service/src/main/java/org/onap/cps/ncmp/impl/inventory/sync/lcm/CmHandleStateMonitor.java [new file with mode: 0644]
cps-ncmp-service/src/main/java/org/onap/cps/ncmp/impl/inventory/sync/lcm/LcmEventsCmHandleStateHandlerImpl.java
cps-ncmp-service/src/test/groovy/org/onap/cps/ncmp/impl/cache/AdminCacheConfigSpec.groovy [new file with mode: 0644]
cps-ncmp-service/src/test/groovy/org/onap/cps/ncmp/impl/inventory/sync/lcm/CmHandleStateMonitorSpec.groovy [new file with mode: 0644]
cps-ncmp-service/src/test/groovy/org/onap/cps/ncmp/impl/inventory/sync/lcm/LcmEventsCmHandleStateHandlerImplSpec.groovy

index 22194f3..39ed6ef 100644 (file)
@@ -1,6 +1,6 @@
 /*
  * ============LICENSE_START=======================================================
- * Copyright (C) 2023 Nordix Foundation.
+ * Copyright (C) 2023-2025 Nordix Foundation.
  * ================================================================================
  * Licensed under the Apache License, Version 2.0 (the "License");
  * you may not use this file except in compliance with the License.
 
 package org.onap.cps.config;
 
+import com.hazelcast.map.IMap;
 import io.micrometer.core.aop.TimedAspect;
+import io.micrometer.core.instrument.Gauge;
 import io.micrometer.core.instrument.MeterRegistry;
+import lombok.RequiredArgsConstructor;
 import org.springframework.context.annotation.Bean;
 import org.springframework.context.annotation.Configuration;
 
 @Configuration
+@RequiredArgsConstructor
 public class MicroMeterConfig {
 
+    private static final String TAG = "state";
+    private static final String CMHANDLE_STATE_GAUGE = "cmHandlesByState";
+    final IMap<String, Integer> cmHandlesByState;
+
+    @Bean
+    public TimedAspect timedAspect(final MeterRegistry meterRegistry) {
+        return new TimedAspect(meterRegistry);
+    }
+
+    /**
+     * Register gauge metric for cm handles with state 'advised'.
+     *
+     * @param meterRegistry meter registry
+     * @return cm handle state gauge
+     */
+    @Bean
+    public Gauge advisedCmHandles(final MeterRegistry meterRegistry) {
+        return Gauge.builder(CMHANDLE_STATE_GAUGE, cmHandlesByState,
+                        value -> cmHandlesByState.get("advisedCmHandlesCount"))
+                .tag(TAG, "ADVISED")
+                .description("Current number of cmhandles in advised state")
+                .register(meterRegistry);
+    }
+
+    /**
+     * Register gauge metric for cm handles with state 'ready'.
+     *
+     * @param meterRegistry meter registry
+     * @return cm handle state gauge
+     */
+    @Bean
+    public Gauge readyCmHandles(final MeterRegistry meterRegistry) {
+        return Gauge.builder(CMHANDLE_STATE_GAUGE, cmHandlesByState,
+                        value -> cmHandlesByState.get("readyCmHandlesCount"))
+                .tag(TAG, "READY")
+                .description("Current number of cmhandles in ready state")
+                .register(meterRegistry);
+    }
+
+    /**
+     * Register gauge metric for cm handles with state 'locked'.
+     *
+     * @param meterRegistry meter registry
+     * @return cm handle state gauge
+     */
+    @Bean
+    public Gauge lockedCmHandles(final MeterRegistry meterRegistry) {
+        return Gauge.builder(CMHANDLE_STATE_GAUGE, cmHandlesByState,
+                value -> cmHandlesByState.get("lockedCmHandlesCount"))
+                .tag(TAG, "LOCKED")
+                .description("Current number of cmhandles in locked state")
+                .register(meterRegistry);
+    }
+
+    /**
+     * Register gauge metric for cm handles with state 'deleting'.
+     *
+     * @param meterRegistry meter registry
+     * @return cm handle state gauge
+     */
+    @Bean
+    public Gauge deletingCmHandles(final MeterRegistry meterRegistry) {
+        return Gauge.builder(CMHANDLE_STATE_GAUGE, cmHandlesByState,
+                        value -> cmHandlesByState.get("deletingCmHandlesCount"))
+                .tag(TAG, "DELETING")
+                .description("Current number of cmhandles in deleting state")
+                .register(meterRegistry);
+    }
+
+    /**
+     * Register gauge metric for cm handles with state 'deleted'.
+     *
+     * @param meterRegistry meter registry
+     * @return cm handle state gauge
+     */
     @Bean
-    public TimedAspect timedAspect(final MeterRegistry registry) {
-        return new TimedAspect(registry);
+    public Gauge deletedCmHandles(final MeterRegistry meterRegistry) {
+        return Gauge.builder(CMHANDLE_STATE_GAUGE, cmHandlesByState,
+                        value -> cmHandlesByState.get("deletedCmHandlesCount"))
+                .tag(TAG, "DELETED")
+                .description("Current number of cmhandles in deleted state")
+                .register(meterRegistry);
     }
 
 }
index d7e39f7..573db1f 100644 (file)
@@ -1,8 +1,8 @@
 #  ============LICENSE_START=======================================================
 #  Copyright (C) 2021 Pantheon.tech
 #  Modifications Copyright (C) 2021-2022 Bell Canada
-#  Modifications Copyright (C) 2021-2024 Nordix Foundation
 #  Modifications Copyright (C) 2024 TechMahindra Ltd
+#  Modifications Copyright (C) 2021-2025 Nordix Foundation
 #  ================================================================================
 #  Licensed under the Apache License, Version 2.0 (the "License");
 #  you may not use this file except in compliance with the License.
@@ -172,7 +172,7 @@ management:
     endpoints:
         web:
             exposure:
-                include: info,health,loggers,prometheus
+                include: info,health,loggers,prometheus,metrics
     endpoint:
         health:
             show-details: always
index 61bc2cf..67ca646 100644 (file)
@@ -1,6 +1,6 @@
 /*
  * ============LICENSE_START=======================================================
- * Copyright (C) 2023 Nordix Foundation.
+ * Copyright (C) 2023-2025 Nordix Foundation.
  * ================================================================================
  * Licensed under the Apache License, Version 2.0 (the "License");
  * you may not use this file except in compliance with the License.
 
 package org.onap.cps.config
 
+import com.hazelcast.map.IMap
 import io.micrometer.core.instrument.simple.SimpleMeterRegistry
 import spock.lang.Specification
 
 class MicroMeterConfigSpec extends Specification {
 
-    def objectUnderTest = new MicroMeterConfig()
+    def cmHandlesByState = Mock(IMap)
+    def objectUnderTest = new MicroMeterConfig(cmHandlesByState)
+    def simpleMeterRegistry = new SimpleMeterRegistry()
 
-    def 'Creating a tined aspect.'() {
+    def 'Creating a timed aspect.'() {
         expect: ' a timed aspect can be created'
-            assert objectUnderTest.timedAspect(new SimpleMeterRegistry()) != null
+            assert objectUnderTest.timedAspect(simpleMeterRegistry) != null
+    }
+
+    def 'Creating gauges for cm handle states.'() {
+        given: 'cache returns value for each state'
+            cmHandlesByState.get(_) >> 1
+        when: 'gauges for each state are created'
+             objectUnderTest.advisedCmHandles(simpleMeterRegistry)
+             objectUnderTest.readyCmHandles(simpleMeterRegistry)
+             objectUnderTest.lockedCmHandles(simpleMeterRegistry)
+             objectUnderTest.deletingCmHandles(simpleMeterRegistry)
+             objectUnderTest.deletedCmHandles(simpleMeterRegistry)
+        then: 'each state has the correct value when queried'
+            def states = ["ADVISED", "READY", "LOCKED", "DELETING", "DELETED"]
+            states.each { state ->
+                def gaugeValue = simpleMeterRegistry.get("cmHandlesByState").tag("state",state).gauge().value()
+                assert gaugeValue == 1
+            }
     }
 
 }
diff --git a/cps-ncmp-service/src/main/java/org/onap/cps/ncmp/impl/cache/AdminCacheConfig.java b/cps-ncmp-service/src/main/java/org/onap/cps/ncmp/impl/cache/AdminCacheConfig.java
new file mode 100644 (file)
index 0000000..a29799b
--- /dev/null
@@ -0,0 +1,49 @@
+/*
+ * ============LICENSE_START=======================================================
+ *  Copyright (C) 2025 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.impl.cache;
+
+import com.hazelcast.config.MapConfig;
+import com.hazelcast.map.IMap;
+import org.springframework.context.annotation.Bean;
+import org.springframework.context.annotation.Configuration;
+
+@Configuration
+public class AdminCacheConfig extends HazelcastCacheConfig {
+
+    private static final MapConfig adminCacheMapConfig = createMapConfig("adminCacheMapConfig");
+
+    /**
+     * Distributed instance admin cache map for cm handles by state for use of gauge metrics.
+     *
+     * @return configured map of cm handles by state.
+     */
+    @Bean
+    public IMap<String, Integer> cmHandlesByState() {
+        final IMap<String, Integer> cmHandlesByState = getOrCreateHazelcastInstance(adminCacheMapConfig).getMap(
+                "cmHandlesByState");
+        cmHandlesByState.putIfAbsent("advisedCmHandlesCount", 0);
+        cmHandlesByState.putIfAbsent("readyCmHandlesCount", 0);
+        cmHandlesByState.putIfAbsent("lockedCmHandlesCount", 0);
+        cmHandlesByState.putIfAbsent("deletingCmHandlesCount", 0);
+        cmHandlesByState.putIfAbsent("deletedCmHandlesCount", 0);
+        return cmHandlesByState;
+    }
+}
diff --git a/cps-ncmp-service/src/main/java/org/onap/cps/ncmp/impl/inventory/sync/lcm/CmHandleStateMonitor.java b/cps-ncmp-service/src/main/java/org/onap/cps/ncmp/impl/inventory/sync/lcm/CmHandleStateMonitor.java
new file mode 100644 (file)
index 0000000..4fd752c
--- /dev/null
@@ -0,0 +1,100 @@
+/*
+ * ============LICENSE_START=======================================================
+ * Copyright (C) 2025 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.impl.inventory.sync.lcm;
+
+import com.hazelcast.map.EntryProcessor;
+import com.hazelcast.map.IMap;
+import java.util.Collection;
+import java.util.Map;
+import lombok.RequiredArgsConstructor;
+import org.onap.cps.ncmp.api.inventory.models.CompositeState;
+import org.onap.cps.ncmp.impl.inventory.models.CmHandleState;
+import org.onap.cps.ncmp.impl.inventory.sync.lcm.LcmEventsCmHandleStateHandlerImpl.CmHandleTransitionPair;
+import org.springframework.scheduling.annotation.Async;
+import org.springframework.stereotype.Component;
+
+@Component
+@RequiredArgsConstructor
+public class CmHandleStateMonitor {
+
+    private static final String METRIC_POSTFIX = "CmHandlesCount";
+    final IMap<String, Integer> cmHandlesByState;
+
+    /**
+     * Asynchronously update the cm handle state metrics.
+     *
+     * @param cmHandleTransitionPairs cm handle transition pairs
+     */
+    @Async
+    public void updateCmHandleStateMetrics(final Collection<CmHandleTransitionPair>
+                                                       cmHandleTransitionPairs) {
+        cmHandleTransitionPairs.forEach(this::updateMetricWithStateChange);
+    }
+
+    private void updateMetricWithStateChange(final CmHandleTransitionPair cmHandleTransitionPair) {
+        final CmHandleState targetCmHandleState = cmHandleTransitionPair.getTargetYangModelCmHandle()
+                .getCompositeState().getCmHandleState();
+        if (isNew(cmHandleTransitionPair.getCurrentYangModelCmHandle().getCompositeState())) {
+            updateTargetStateCount(targetCmHandleState);
+        } else {
+            final CmHandleState previousCmHandleState = cmHandleTransitionPair.getCurrentYangModelCmHandle()
+                    .getCompositeState().getCmHandleState();
+            updatePreviousStateCount(previousCmHandleState);
+            updateTargetStateCount(targetCmHandleState);
+        }
+    }
+
+    private void updatePreviousStateCount(final CmHandleState previousCmHandleState) {
+        final String keyName = previousCmHandleState.name().toLowerCase() + METRIC_POSTFIX;
+        cmHandlesByState.executeOnKey(keyName, new DecreasingEntryProcessor());
+    }
+
+    private void updateTargetStateCount(final CmHandleState targetCmHandleState) {
+        final String keyName = targetCmHandleState.name().toLowerCase() + METRIC_POSTFIX;
+        cmHandlesByState.executeOnKey(keyName, new IncreasingEntryProcessor());
+    }
+
+    private boolean isNew(final CompositeState existingCompositeState) {
+        return (existingCompositeState == null);
+    }
+
+    static class DecreasingEntryProcessor implements EntryProcessor<String, Integer, Void> {
+        @Override
+        public Void process(final Map.Entry<String, Integer> entry) {
+            final int currentValue = entry.getValue();
+            if (currentValue > 0) {
+                entry.setValue(currentValue - 1);
+            }
+            return null;
+        }
+    }
+
+    static class IncreasingEntryProcessor implements EntryProcessor<String, Integer, Void> {
+        @Override
+        public Void process(final Map.Entry<String, Integer> entry) {
+            final int currentValue = entry.getValue();
+            entry.setValue(currentValue + 1);
+            return null;
+        }
+    }
+
+
+}
\ No newline at end of file
index e9bd372..3e6597e 100644 (file)
@@ -1,6 +1,6 @@
 /*
  * ============LICENSE_START=======================================================
- * Copyright (C) 2022-2024 Nordix Foundation
+ * Copyright (C) 2022-2025 Nordix Foundation
  * ================================================================================
  * Licensed under the Apache License, Version 2.0 (the "License");
  * you may not use this file except in compliance with the License.
@@ -51,6 +51,7 @@ public class LcmEventsCmHandleStateHandlerImpl implements LcmEventsCmHandleState
 
     private final InventoryPersistence inventoryPersistence;
     private final LcmEventsCmHandleStateHandlerAsyncHelper lcmEventsCmHandleStateHandlerAsyncHelper;
+    private final CmHandleStateMonitor cmHandleStateMonitor;
 
     @Override
     @Timed(value = "cps.ncmp.cmhandle.state.update.batch",
@@ -60,6 +61,7 @@ public class LcmEventsCmHandleStateHandlerImpl implements LcmEventsCmHandleState
                 prepareCmHandleTransitionBatch(cmHandleStatePerCmHandle);
         persistCmHandleBatch(cmHandleTransitionPairs);
         lcmEventsCmHandleStateHandlerAsyncHelper.publishLcmEventBatchAsynchronously(cmHandleTransitionPairs);
+        cmHandleStateMonitor.updateCmHandleStateMetrics(cmHandleTransitionPairs);
     }
 
     @Override
@@ -168,7 +170,7 @@ public class LcmEventsCmHandleStateHandlerImpl implements LcmEventsCmHandleState
     @Getter
     @Setter
     @NoArgsConstructor
-    static class CmHandleTransitionPair {
+    public static class CmHandleTransitionPair {
 
         private YangModelCmHandle currentYangModelCmHandle;
         private YangModelCmHandle targetYangModelCmHandle;
diff --git a/cps-ncmp-service/src/test/groovy/org/onap/cps/ncmp/impl/cache/AdminCacheConfigSpec.groovy b/cps-ncmp-service/src/test/groovy/org/onap/cps/ncmp/impl/cache/AdminCacheConfigSpec.groovy
new file mode 100644 (file)
index 0000000..9b96033
--- /dev/null
@@ -0,0 +1,57 @@
+/*
+ * ============LICENSE_START=======================================================
+ * Copyright (C) 2025 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.impl.cache
+
+import com.hazelcast.core.Hazelcast
+import com.hazelcast.map.IMap
+import org.springframework.beans.factory.annotation.Autowired
+import org.springframework.boot.test.context.SpringBootTest
+import org.springframework.test.context.ContextConfiguration;
+import spock.lang.Specification;
+
+@SpringBootTest
+@ContextConfiguration(classes = [AdminCacheConfig])
+class AdminCacheConfigSpec extends Specification {
+
+    @Autowired
+    IMap<String, Integer> cmHandlesByState
+
+    def cleanupSpec() {
+        Hazelcast.getHazelcastInstanceByName('cps-and-ncmp-hazelcast-instance-test-config').shutdown()
+    }
+
+    def 'Hazelcast cache for cm handle by state gauge'() {
+        expect: 'system is able to create an instance of the cm handle by state cache'
+            assert null != cmHandlesByState
+        and: 'there is at least 1 instance'
+            assert Hazelcast.allHazelcastInstances.size() > 0
+        and: 'Hazelcast cache instance for cm handle by state is present'
+            assert Hazelcast.getHazelcastInstanceByName('cps-and-ncmp-hazelcast-instance-test-config').getMap('cmHandlesByState') != null
+        and: 'the cache already contains 5 entries, an entry for each state'
+            def cmHandleByState =  Hazelcast.getHazelcastInstanceByName('cps-and-ncmp-hazelcast-instance-test-config').getMap('cmHandlesByState')
+            assert cmHandleByState.size() == 5
+            assert cmHandleByState.containsKey('advisedCmHandlesCount')
+            assert cmHandleByState.containsKey('lockedCmHandlesCount')
+            assert cmHandleByState.containsKey('readyCmHandlesCount')
+            assert cmHandleByState.containsKey('deletingCmHandlesCount')
+            assert cmHandleByState.containsKey('deletedCmHandlesCount')
+    }
+}
diff --git a/cps-ncmp-service/src/test/groovy/org/onap/cps/ncmp/impl/inventory/sync/lcm/CmHandleStateMonitorSpec.groovy b/cps-ncmp-service/src/test/groovy/org/onap/cps/ncmp/impl/inventory/sync/lcm/CmHandleStateMonitorSpec.groovy
new file mode 100644 (file)
index 0000000..2833ca8
--- /dev/null
@@ -0,0 +1,92 @@
+/*
+ * ============LICENSE_START=======================================================
+ * Copyright (C) 2025 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.impl.inventory.sync.lcm
+
+import static org.onap.cps.ncmp.impl.inventory.models.CmHandleState.ADVISED
+import static org.onap.cps.ncmp.impl.inventory.models.CmHandleState.READY
+
+import com.hazelcast.core.Hazelcast
+import com.hazelcast.map.IMap
+import org.onap.cps.ncmp.api.inventory.models.CompositeState
+import org.onap.cps.ncmp.impl.inventory.models.YangModelCmHandle
+import org.onap.cps.ncmp.impl.inventory.sync.lcm.CmHandleStateMonitor.DecreasingEntryProcessor
+import org.onap.cps.ncmp.impl.inventory.sync.lcm.CmHandleStateMonitor.IncreasingEntryProcessor
+import spock.lang.Shared
+import spock.lang.Specification;
+
+class CmHandleStateMonitorSpec extends Specification {
+
+    def cmHandlesByState = Mock(IMap)
+    def objectUnderTest = new CmHandleStateMonitor(cmHandlesByState)
+
+    @Shared
+    def entryProcessingMap =  Hazelcast.newHazelcastInstance().getMap('entryProcessingMap')
+
+    def setup() {
+        entryProcessingMap.put('zeroCmHandlesCount', 0)
+        entryProcessingMap.put('tenCmHandlesCount', 10)
+    }
+
+    def cleanupSpec() {
+        Hazelcast.shutdownAll()
+    }
+
+    def 'Update cm handle state metric'() {
+        given: 'a collection of cm handle state pair'
+            def cmHandleTransitionPair = new LcmEventsCmHandleStateHandlerImpl.CmHandleTransitionPair()
+            cmHandleTransitionPair.currentYangModelCmHandle = new YangModelCmHandle(compositeState: new CompositeState(cmHandleState: ADVISED))
+            cmHandleTransitionPair.targetYangModelCmHandle =  new YangModelCmHandle(compositeState: new CompositeState(cmHandleState: READY))
+        when: 'method to update cm handle state metrics is called'
+            objectUnderTest.updateCmHandleStateMetrics([cmHandleTransitionPair])
+        then: 'cm handle by state cache map is called once for current and target state for entry processing'
+            1 * cmHandlesByState.executeOnKey('advisedCmHandlesCount', _)
+            1 * cmHandlesByState.executeOnKey('readyCmHandlesCount', _)
+    }
+
+    def 'Updating cm handle state metric with no previous state'() {
+        given: 'a collection of cm handle state pair wherein current state is null'
+            def cmHandleTransitionPair = new LcmEventsCmHandleStateHandlerImpl.CmHandleTransitionPair()
+            cmHandleTransitionPair.currentYangModelCmHandle = new YangModelCmHandle(compositeState: null)
+            cmHandleTransitionPair.targetYangModelCmHandle =  new YangModelCmHandle(compositeState: new CompositeState(cmHandleState: ADVISED))
+        when: 'method to update cm handle state metrics is called'
+            objectUnderTest.updateCmHandleStateMetrics([cmHandleTransitionPair])
+        then: 'cm handle by state cache map is called only once'
+            1 * cmHandlesByState.executeOnKey(_, _)
+    }
+
+    def 'Applying decreasing entry processor to a key on map where #scenario'() {
+        when: 'decreasing entry processor is applied to subtract 1 to the value'
+            entryProcessingMap.executeOnKey(key, new DecreasingEntryProcessor())
+        then: 'the new value is as expected'
+            assert entryProcessingMap.get(key) == expectedValue
+        where: 'the following data is used'
+            scenario                        | key                 || expectedValue
+            'current value of count is zero'| 'zeroCmHandlesCount'|| 0
+            'current value of count is >0'  | 'tenCmHandlesCount' || 9
+    }
+
+    def 'Applying increasing entry processor to a key on map'() {
+        when: 'increasing entry processor is applied to add 1 to the value'
+            entryProcessingMap.executeOnKey('tenCmHandlesCount', new IncreasingEntryProcessor())
+        then: 'the new value is as expected'
+            assert entryProcessingMap.get('tenCmHandlesCount') == 11
+    }
+}
index 4b676e1..5e614a6 100644 (file)
@@ -1,6 +1,6 @@
 /*
  * ============LICENSE_START=======================================================
- * Copyright (C) 2022-2024 Nordix Foundation
+ * Copyright (C) 2022-2025 Nordix Foundation
  * ================================================================================
  * Licensed under the Apache License, Version 2.0 (the "License");
  * you may not use this file except in compliance with the License.
@@ -56,9 +56,10 @@ class LcmEventsCmHandleStateHandlerImplSpec extends Specification {
     def mockInventoryPersistence = Mock(InventoryPersistence)
     def mockLcmEventsCreator = Mock(LcmEventsCreator)
     def mockLcmEventsService = Mock(LcmEventsService)
+    def mockCmHandleStateMonitor = Mock(CmHandleStateMonitor)
 
     def lcmEventsCmHandleStateHandlerAsyncHelper = new LcmEventsCmHandleStateHandlerAsyncHelper(mockLcmEventsCreator, mockLcmEventsService)
-    def objectUnderTest = new LcmEventsCmHandleStateHandlerImpl(mockInventoryPersistence, lcmEventsCmHandleStateHandlerAsyncHelper)
+    def objectUnderTest = new LcmEventsCmHandleStateHandlerImpl(mockInventoryPersistence, lcmEventsCmHandleStateHandlerAsyncHelper, mockCmHandleStateMonitor)
 
     def cmHandleId = 'cmhandle-id-1'
     def compositeState