089fce0b1c35b657c38b828355c091e3d5067080
[ccsdk/cds.git] /
1 /*
2  *  Copyright © 2017-2018 AT&T Intellectual Property.
3  *  Modifications Copyright © 2018 IBM.
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
18 package org.onap.ccsdk.apps.controllerblueprints.resource.dict.service
19
20 import com.att.eelf.configuration.EELFLogger
21 import org.apache.commons.collections.CollectionUtils
22 import org.apache.commons.lang3.StringUtils
23 import org.apache.commons.lang3.text.StrBuilder
24 import org.onap.ccsdk.apps.controllerblueprints.core.BluePrintException
25 import org.onap.ccsdk.apps.controllerblueprints.core.utils.TopologicalSortingUtils
26 import org.onap.ccsdk.apps.controllerblueprints.resource.dict.ResourceAssignment
27 import com.att.eelf.configuration.EELFManager
28 import org.onap.ccsdk.apps.controllerblueprints.core.format
29 import org.onap.ccsdk.apps.controllerblueprints.resource.dict.factory.ResourceSourceMappingFactory
30 import java.io.Serializable
31
32 /**
33  * ResourceAssignmentValidationService.
34  *
35  * @author Brinda Santh
36  */
37 interface ResourceAssignmentValidationService : Serializable {
38
39     @Throws(BluePrintException::class)
40     fun validate(resourceAssignments: List<ResourceAssignment>): Boolean
41 }
42
43 /**
44  * ResourceAssignmentValidationDefaultService.
45  *
46  * @author Brinda Santh
47  */
48 open class ResourceAssignmentValidationDefaultService : ResourceAssignmentValidationService {
49     private val log: EELFLogger = EELFManager.getInstance().getLogger(ResourceAssignmentValidationDefaultService::class.java)
50
51     open var resourceAssignmentMap: Map<String, ResourceAssignment> = hashMapOf()
52     open val validationMessage = StrBuilder()
53
54     override fun validate(resourceAssignments: List<ResourceAssignment>): Boolean {
55         try {
56             validateSources(resourceAssignments)
57             validateTemplateNDictionaryKeys(resourceAssignments)
58             validateCyclicDependency(resourceAssignments)
59             if (StringUtils.isNotBlank(validationMessage)) {
60                 throw BluePrintException("Resource Assignment Validation Failure")
61             }
62         } catch (e: Exception) {
63             throw BluePrintException("Resource Assignment Validation :" + validationMessage.toString(), e)
64         }
65         return true
66     }
67
68     open fun validateSources(resourceAssignments: List<ResourceAssignment>) {
69         log.info("validating resource assignment sources")
70         resourceAssignments.forEach { resourceAssignment ->
71             try {
72                 ResourceSourceMappingFactory.getRegisterSourceMapping(resourceAssignment.dictionarySource!!)
73             } catch (e: BluePrintException) {
74                 validationMessage.appendln(e.message + format(" for resource assignment({})", resourceAssignment.name))
75             }
76         }
77     }
78
79     open fun validateTemplateNDictionaryKeys(resourceAssignments: List<ResourceAssignment>) {
80
81         resourceAssignmentMap = resourceAssignments.map { it.name to it }.toMap()
82
83         val duplicateKeyNames = resourceAssignments.groupBy { it.name }
84                 .filter { it.value.size > 1 }
85                 .map { it.key }
86
87         if (duplicateKeyNames.isNotEmpty()) {
88             validationMessage.appendln(String.format("Duplicate Assignment Template Keys (%s) is Present", duplicateKeyNames))
89         }
90
91         val duplicateDictionaryKeyNames = resourceAssignments.groupBy { it.dictionaryName }
92                 .filter { it.value.size > 1 }
93                 .map { it.key }
94         if (duplicateDictionaryKeyNames.isNotEmpty()) {
95             validationMessage.appendln(String.format("Duplicate Assignment Dictionary Keys (%s) is Present", duplicateDictionaryKeyNames))
96         }
97
98         val dependenciesNames = resourceAssignments.mapNotNull { it.dependencies }.flatten()
99
100         val notPresentDictionaries = dependenciesNames.filter { !resourceAssignmentMap.containsKey(it) }.distinct()
101         if (notPresentDictionaries.isNotEmpty()) {
102             validationMessage.appendln(String.format("No assignments for Dictionary Keys (%s)", notPresentDictionaries))
103         }
104
105         if (StringUtils.isNotBlank(validationMessage)) {
106             throw BluePrintException("Resource Assignment Validation Failure")
107         }
108     }
109
110     open fun validateCyclicDependency(resourceAssignments: List<ResourceAssignment>) {
111         val startResourceAssignment = ResourceAssignment()
112         startResourceAssignment.name = "*"
113
114         val topologySorting = TopologicalSortingUtils<ResourceAssignment>()
115
116         resourceAssignmentMap.map { it.value }.map { resourceAssignment ->
117             if (CollectionUtils.isNotEmpty(resourceAssignment.dependencies)) {
118                 resourceAssignment.dependencies!!.map {
119                     log.info("Topological Graph link from {} to {}", it, resourceAssignment.name)
120                     topologySorting.add(resourceAssignmentMap[it]!!, resourceAssignment)
121                 }
122             } else {
123                 topologySorting.add(startResourceAssignment, resourceAssignment)
124             }
125         }
126
127         if (!topologySorting.isDag) {
128             val graph = getTopologicalGraph(topologySorting)
129             validationMessage.appendln("Cyclic Dependency :$graph")
130         }
131     }
132
133     open fun getTopologicalGraph(topologySorting: TopologicalSortingUtils<ResourceAssignment>): String {
134         val s = StringBuilder()
135         val neighbors = topologySorting.getNeighbors()
136
137         neighbors.forEach { v, vs ->
138             if (v.name == "*") {
139                 s.append("\n    * -> [")
140                 for (resourceAssignment in vs) {
141                     s.append("(" + resourceAssignment.dictionaryName + ":" + resourceAssignment.name
142                             + "),")
143                 }
144                 s.append("]")
145             } else {
146                 s.append("\n    (" + v.dictionaryName + ":" + v.name + ") -> [")
147                 for (resourceAssignment in vs) {
148                     s.append("(" + resourceAssignment.dictionaryName + ":" + resourceAssignment.name
149                             + "),")
150                 }
151                 s.append("]")
152             }
153         }
154         return s.toString()
155     }
156
157
158 }