aboutsummaryrefslogtreecommitdiffstats
path: root/ms/blueprintsprocessor/modules/services/execution-service/src/test/kotlin/org/onap/ccsdk/cds/blueprintsprocessor/services/execution/scripts/AbstractComponentFunctionTest.kt
blob: b5757812cda3a2e8a330abf441d82a6ef0dc4235 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
/*-
 * ============LICENSE_START=======================================================
 * ONAP - CDS
 * ================================================================================
 * Copyright (C) 2019 Huawei Technologies Co., Ltd. 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.
 * ============LICENSE_END=========================================================
 */

package org.onap.ccsdk.cds.blueprintsprocessor.services.execution.scripts

import com.fasterxml.jackson.databind.JsonNode
import com.fasterxml.jackson.databind.ObjectMapper
import com.fasterxml.jackson.databind.node.ObjectNode
import io.mockk.every
import io.mockk.mockk
import io.mockk.spyk
import io.mockk.verify
import kotlinx.coroutines.runBlocking
import org.junit.Test
import org.junit.runner.RunWith
import org.onap.ccsdk.cds.blueprintsprocessor.core.api.data.ActionIdentifiers
import org.onap.ccsdk.cds.blueprintsprocessor.core.api.data.CommonHeader
import org.onap.ccsdk.cds.blueprintsprocessor.core.api.data.ExecutionServiceInput
import org.onap.ccsdk.cds.blueprintsprocessor.core.api.data.StepData
import org.onap.ccsdk.cds.blueprintsprocessor.core.service.BluePrintClusterService
import org.onap.ccsdk.cds.blueprintsprocessor.core.service.CDS_LOCK_GROUP
import org.onap.ccsdk.cds.blueprintsprocessor.core.service.ClusterLock
import org.onap.ccsdk.cds.blueprintsprocessor.services.execution.AbstractComponentFunction
import org.onap.ccsdk.cds.blueprintsprocessor.services.execution.ComponentFunctionScriptingService
import org.onap.ccsdk.cds.blueprintsprocessor.services.execution.nodeTypeComponentScriptExecutor
import org.onap.ccsdk.cds.controllerblueprints.core.BluePrintConstants
import org.onap.ccsdk.cds.controllerblueprints.core.BluePrintError
import org.onap.ccsdk.cds.controllerblueprints.core.BluePrintTypes
import org.onap.ccsdk.cds.controllerblueprints.core.asJsonPrimitive
import org.onap.ccsdk.cds.controllerblueprints.core.asJsonType
import org.onap.ccsdk.cds.controllerblueprints.core.data.Implementation
import org.onap.ccsdk.cds.controllerblueprints.core.data.LockAssignment
import org.onap.ccsdk.cds.controllerblueprints.core.normalizedPathName
import org.onap.ccsdk.cds.controllerblueprints.core.scripts.BluePrintScriptsServiceImpl
import org.onap.ccsdk.cds.controllerblueprints.core.service.BluePrintContext
import org.onap.ccsdk.cds.controllerblueprints.core.service.DefaultBluePrintRuntimeService
import org.onap.ccsdk.cds.controllerblueprints.core.utils.JacksonUtils
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.test.context.ContextConfiguration
import org.springframework.test.context.junit4.SpringRunner
import kotlin.test.BeforeTest
import kotlin.test.assertEquals
import kotlin.test.assertNotNull

/**
 * Unit test cases for abstract component function.
 */
@RunWith(SpringRunner::class)
@ContextConfiguration(
    classes = [
        ComponentFunctionScriptingService::class,
        BluePrintScriptsServiceImpl::class, DeprecatedBlueprintJythonService::class
    ]
)
class AbstractComponentFunctionTest {

    lateinit var bluePrintRuntimeService: DefaultBluePrintRuntimeService
    lateinit var blueprintContext: BluePrintContext
    lateinit var blueprintClusterService: BluePrintClusterService

    @Autowired
    lateinit var compSvc: ComponentFunctionScriptingService

    @BeforeTest
    fun init() {
        bluePrintRuntimeService = mockk()
        blueprintContext = mockk()
        blueprintClusterService = mockk()
        every { bluePrintRuntimeService.bluePrintContext() } returns blueprintContext

        every { blueprintContext.rootPath } returns normalizedPathName("target")
        every {
            blueprintContext.nodeTemplateOperationImplementation(
                any(), any(), any()
            )
        } returns Implementation()

        every { bluePrintRuntimeService.getBluePrintError() } returns BluePrintError()
    }

    @Test
    fun testAbstractComponent() {
        runBlocking {
            val samp = SampleComponent()
            val comp = samp as AbstractComponentFunction

            comp.bluePrintRuntimeService = bluePrintRuntimeService
            comp.stepName = "sample-step"
            assertNotNull(comp, "failed to get kotlin instance")

            val input = getMockedInput(bluePrintRuntimeService)

            val output = comp.applyNB(input)

            assertEquals(output.actionIdentifiers.actionName, "activate")
            assertEquals(output.commonHeader.requestId, "1234")
            assertEquals(output.stepData!!.name, "activate-restconf")
            assertEquals(output.status.message, "success")
        }
    }

    @Test
    fun testComponentFunctionPayload() {
        val sampleComponent = SampleComponent()
        sampleComponent.workflowName = "sample-action"
        sampleComponent.executionServiceInput = JacksonUtils.readValueFromClassPathFile(
            "payload/requests/sample-execution-request.json", ExecutionServiceInput::class.java
        )!!
        val payload = sampleComponent.requestPayload()
        assertNotNull(payload, "failed to get payload")
        val data = sampleComponent.requestPayloadActionProperty("data")?.first()
        assertNotNull(data, "failed to get payload request action data")
    }

    @Test
    fun testAbstractScriptComponent() {
        runBlocking {
            val samp = SampleRestconfComponent(compSvc)
            val comp = samp as AbstractComponentFunction

            comp.bluePrintRuntimeService = bluePrintRuntimeService
            comp.stepName = "sample-step"
            assertNotNull(comp, "failed to get kotlin instance")

            val input = getMockedInput(bluePrintRuntimeService)

            val output = comp.applyNB(input)

            assertEquals(output.actionIdentifiers.actionName, "activate")
            assertEquals(output.commonHeader.requestId, "1234")
            assertEquals(output.stepData!!.name, "activate-restconf")
            assertEquals(output.status.message, "success")
        }
    }

    @Test
    fun testComponentScriptExecutorNodeType() {
        val componentScriptExecutor = BluePrintTypes.nodeTypeComponentScriptExecutor()
        assertNotNull(componentScriptExecutor.interfaces, "failed to get interface operations")
    }

    @Test
    fun `prepareRequestNB should resolve lock properties`() {
        val implementation = Implementation().apply {
            this.lock = LockAssignment().apply {
                this.key = """ {"get_input": "lock-key"} """.asJsonPrimitive()
            }
        }
        every {
            blueprintContext.nodeTemplateOperationImplementation(any(), any(), any())
        } returns implementation

        every {
            bluePrintRuntimeService.resolvePropertyAssignments(any(), any(), any())
        } returns mutableMapOf(
            "key" to "abc-123-def-456".asJsonType(),
            "acquireTimeout" to implementation.lock!!.acquireTimeout
        )

        val component: AbstractComponentFunction = SampleComponent()
        component.bluePrintRuntimeService = bluePrintRuntimeService
        component.bluePrintClusterService = blueprintClusterService

        runBlocking {
            component.prepareRequestNB(getMockedInput(bluePrintRuntimeService))
        }

        val resolvedLock = component.implementation.lock!!

        assertEquals("abc-123-def-456", resolvedLock.key.textValue())
        // default value
        assertEquals(180, resolvedLock.acquireTimeout.intValue())
    }

    @Test(expected = Exception::class)
    fun `prepareRequestNB should throw exception if it fails to resolve lock key`() {
        every {
            blueprintContext.nodeTemplateOperationImplementation(any(), any(), any())
        } returns Implementation().apply { this.lock = LockAssignment() }

        every {
            bluePrintRuntimeService.resolvePropertyAssignments(any(), any(), any())
        } returns mutableMapOf(
            "key" to "".asJsonType(),
            "acquireTimeout" to Integer(360).asJsonType()
        )

        val component: AbstractComponentFunction = SampleComponent()
        component.bluePrintRuntimeService = bluePrintRuntimeService
        component.bluePrintClusterService = blueprintClusterService

        runBlocking {
            component.prepareRequestNB(getMockedInput(bluePrintRuntimeService))
        }
    }

    @Test
    fun `applyNB should catch exceptions and call recoverNB`() {
        val exception = RuntimeException("Intentional test exception")
        every {
            bluePrintRuntimeService.resolvePropertyAssignments(any(), any(), any())
        } throws exception
        every {
            blueprintContext.nodeTemplateOperationImplementation(any(), any(), any())
        } returns Implementation().apply {
            this.lock = LockAssignment().apply { this.key = "testing-lock".asJsonType() }
        }

        val component: AbstractComponentFunction = spyk(SampleComponent())
        component.bluePrintRuntimeService = bluePrintRuntimeService
        component.bluePrintClusterService = blueprintClusterService
        val input = getMockedInput(bluePrintRuntimeService)

        runBlocking { component.applyNB(input) }
        verify { runBlocking { component.recoverNB(exception, input) } }
    }

    @Test
    fun `applyNB - when lock is present use ClusterLock`() {

        val lockName = "testing-lock"

        every {
            blueprintContext.nodeTemplateOperationImplementation(any(), any(), any())
        } returns Implementation().apply {
            this.lock = LockAssignment().apply { this.key = lockName.asJsonType() }
        }

        every {
            bluePrintRuntimeService.resolvePropertyAssignments(any(), any(), any())
        } returns mutableMapOf(
            "key" to lockName.asJsonType(),
            "acquireTimeout" to Integer(180).asJsonType()
        )

        val clusterLock: ClusterLock = mockk()

        every { clusterLock.name() } returns lockName
        every { runBlocking { clusterLock.tryLock(any()) } } returns true
        every { runBlocking { clusterLock.unLock() } } returns Unit

        every {
            runBlocking { blueprintClusterService.clusterLock(any()) }
        } returns clusterLock

        val component: AbstractComponentFunction = SampleComponent()
        component.bluePrintRuntimeService = bluePrintRuntimeService
        component.bluePrintClusterService = blueprintClusterService

        runBlocking {
            component.applyNB(getMockedInput(bluePrintRuntimeService))
        }

        verify {
            runBlocking { blueprintClusterService.clusterLock("$lockName@$CDS_LOCK_GROUP") }
        }
        verify { runBlocking { clusterLock.unLock() } }
    }

    /**
     * Mocked input for abstract function test.
     */
    private fun getMockedInput(bluePrintRuntime: DefaultBluePrintRuntimeService):
        ExecutionServiceInput {

            val mapper = ObjectMapper()
            val rootNode = mapper.createObjectNode()
            rootNode.put("ip-address", "0.0.0.0")
            rootNode.put("type", "rest")

            val operationInputs = hashMapOf<String, JsonNode>()
            operationInputs[BluePrintConstants.PROPERTY_CURRENT_NODE_TEMPLATE] =
                "activate-restconf".asJsonPrimitive()
            operationInputs[BluePrintConstants.PROPERTY_CURRENT_INTERFACE] =
                "interfaceName".asJsonPrimitive()
            operationInputs[BluePrintConstants.PROPERTY_CURRENT_OPERATION] =
                "operationName".asJsonPrimitive()
            operationInputs["dynamic-properties"] = rootNode

            val stepInputData = StepData().apply {
                name = "activate-restconf"
                properties = operationInputs
            }
            val executionServiceInput = ExecutionServiceInput().apply {
                commonHeader = CommonHeader().apply {
                    requestId = "1234"
                }
                actionIdentifiers = ActionIdentifiers().apply {
                    actionName = "activate"
                }
                payload = JacksonUtils.jsonNode("{}") as ObjectNode
            }
            executionServiceInput.stepData = stepInputData

            every {
                bluePrintRuntime.resolveNodeTemplateInterfaceOperationInputs(
                    "activate-restconf", "interfaceName", "operationName"
                )
            } returns operationInputs

            val operationOutputs = hashMapOf<String, JsonNode>()
            every {
                bluePrintRuntime.resolveNodeTemplateInterfaceOperationOutputs(
                    "activate-restconf", "interfaceName", "operationName"
                )
            } returns operationOutputs
            every { bluePrintRuntime.bluePrintContext() } returns blueprintContext

            return executionServiceInput
        }
}