33e83f1013ae789b87599be36544fa4b28ed274c
[cps.git] / cps-ri / src / test / groovy / org / onap / cps / spi / performance / CpsToDataNodePerfTest.groovy
1 /*
2  *  ============LICENSE_START=======================================================
3  *  Copyright (C) 2022 Nordix Foundation
4  *  ================================================================================
5  *  Licensed under the Apache License, Version 2.0 (the "License");
6  *  you may not use this file except in compliance with the License.
7  *  You may obtain a copy of the License at
8  *
9  *        http://www.apache.org/licenses/LICENSE-2.0
10  *
11  *  Unless required by applicable law or agreed to in writing, software
12  *  distributed under the License is distributed on an "AS IS" BASIS,
13  *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14  *  See the License for the specific language governing permissions and
15  *  limitations under the License.
16  *
17  *  SPDX-License-Identifier: Apache-2.0
18  *  ============LICENSE_END=========================================================
19  */
20
21 package org.onap.cps.spi.performance
22
23 import org.springframework.util.StopWatch
24 import org.onap.cps.spi.CpsDataPersistenceService
25 import org.onap.cps.spi.impl.CpsPersistenceSpecBase
26 import org.onap.cps.spi.model.DataNode
27 import org.onap.cps.spi.model.DataNodeBuilder
28 import org.springframework.beans.factory.annotation.Autowired
29 import org.springframework.test.context.jdbc.Sql
30
31 import java.util.concurrent.TimeUnit
32
33 import static org.onap.cps.spi.FetchDescendantsOption.INCLUDE_ALL_DESCENDANTS
34 import static org.onap.cps.spi.FetchDescendantsOption.OMIT_DESCENDANTS
35
36 class CpsToDataNodePerfTest extends CpsPersistenceSpecBase {
37
38     static final String PERF_TEST_DATA = '/data/perf-test.sql'
39
40     @Autowired
41     CpsDataPersistenceService objectUnderTest
42
43     static def PERF_TEST_PARENT = '/perf-parent-1'
44     static def NUMBER_OF_CHILDREN = 200
45     static def NUMBER_OF_GRAND_CHILDREN = 50
46     static def TOTAL_NUMBER_OF_NODES = 1 + NUMBER_OF_CHILDREN + (NUMBER_OF_CHILDREN * NUMBER_OF_GRAND_CHILDREN)  //  Parent + Children +  Grand-children
47     static def ALLOWED_SETUP_TIME_MS = TimeUnit.SECONDS.toMillis(10)
48     static def ALLOWED_READ_TIME_AL_NODES_MS = 500
49
50     def readStopWatch = new StopWatch()
51
52     @Sql([CLEAR_DATA, PERF_TEST_DATA])
53     def 'Create a node with many descendants (please note, subsequent tests depend on this running first).'() {
54         given: 'a node with a large number of descendants is created'
55             def setupStopWatch = new StopWatch()
56             setupStopWatch.start()
57             createLineage()
58             setupStopWatch.stop()
59             def setupDurationInMillis = setupStopWatch.getTotalTimeMillis()
60         and: 'setup duration is under #ALLOWED_SETUP_TIME_MS milliseconds'
61             assert setupDurationInMillis < ALLOWED_SETUP_TIME_MS
62     }
63
64     def 'Get data node with many descendants by xpath #scenario'() {
65         when: 'get parent is executed with all descendants'
66             readStopWatch.start()
67             def result = objectUnderTest.getDataNode('PERF-DATASPACE', 'PERF-ANCHOR', xpath, INCLUDE_ALL_DESCENDANTS)
68             readStopWatch.stop()
69             def readDurationInMillis = readStopWatch.getTotalTimeMillis()
70         then: 'read duration is under 500 milliseconds'
71             assert readDurationInMillis < ALLOWED_READ_TIME_AL_NODES_MS
72         and: 'data node is returned with all the descendants populated'
73             assert countDataNodes(result) == TOTAL_NUMBER_OF_NODES
74         where: 'the following xPaths are used'
75             scenario || xpath
76             'parent' || PERF_TEST_PARENT
77             'root'   || ''
78     }
79
80     def 'Query parent data node with many descendants by cps-path'() {
81         when: 'query is executed with all descendants'
82             readStopWatch.start()
83             def result = objectUnderTest.queryDataNodes('PERF-DATASPACE', 'PERF-ANCHOR', '//perf-parent-1' , INCLUDE_ALL_DESCENDANTS)
84             readStopWatch.stop()
85             def readDurationInMillis = readStopWatch.getTotalTimeMillis()
86         then: 'read duration is under 500 milliseconds'
87             assert readDurationInMillis < ALLOWED_READ_TIME_AL_NODES_MS
88         and: 'data node is returned with all the descendants populated'
89             assert countDataNodes(result) == TOTAL_NUMBER_OF_NODES
90     }
91
92     def 'Query many descendants by cps-path with #scenario'() {
93         when: 'query is executed with all descendants'
94             readStopWatch.start()
95             def result = objectUnderTest.queryDataNodes('PERF-DATASPACE', 'PERF-ANCHOR',  '//perf-test-grand-child-1', descendantsOption)
96             readStopWatch.stop()
97             def readDurationInMillis = readStopWatch.getTotalTimeMillis()
98         then: 'read duration is under 500 milliseconds'
99             assert readDurationInMillis < alowedDuration
100         and: 'data node is returned with all the descendants populated'
101             assert result.size() == NUMBER_OF_CHILDREN
102         where: 'the following options are used'
103             scenario                                        | descendantsOption        || alowedDuration
104             'omit descendants                             ' | OMIT_DESCENDANTS         || 150
105             'include descendants (although there are none)' | INCLUDE_ALL_DESCENDANTS  || 150
106     }
107
108     def createLineage() {
109         (1..NUMBER_OF_CHILDREN).each {
110             def childName = "perf-test-child-${it}".toString()
111             def child = goForthAndMultiply(PERF_TEST_PARENT, childName)
112             objectUnderTest.addChildDataNode('PERF-DATASPACE', 'PERF-ANCHOR', PERF_TEST_PARENT, child)
113         }
114     }
115
116     def goForthAndMultiply(parentXpath, childName) {
117         def grandChildren = []
118         (1..NUMBER_OF_GRAND_CHILDREN).each {
119             def grandChild = new DataNodeBuilder().withXpath("${parentXpath}/${childName}/perf-test-grand-child-${it}").build()
120             grandChildren.add(grandChild)
121         }
122         return new DataNodeBuilder().withXpath("${parentXpath}/${childName}").withChildDataNodes(grandChildren).build()
123     }
124
125     def countDataNodes(dataNodes) {
126         int nodeCount = 1
127         for (DataNode parent : dataNodes) {
128             for (DataNode child : parent.childDataNodes) {
129                 nodeCount = nodeCount + (countDataNodes(child))
130             }
131         }
132         return nodeCount
133     }
134 }