2 * Copyright © 2017-2018 AT&T Intellectual Property.
\r
3 * Modifications Copyright © 2018 IBM.
\r
5 * Licensed under the Apache License, Version 2.0 (the "License");
\r
6 * you may not use this file except in compliance with the License.
\r
7 * You may obtain a copy of the License at
\r
9 * http://www.apache.org/licenses/LICENSE-2.0
\r
11 * Unless required by applicable law or agreed to in writing, software
\r
12 * distributed under the License is distributed on an "AS IS" BASIS,
\r
13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
\r
14 * See the License for the specific language governing permissions and
\r
15 * limitations under the License.
\r
18 package org.onap.ccsdk.features.model.validator;
\r
20 import java.util.HashMap;
\r
21 import java.util.HashSet;
\r
22 import java.util.List;
\r
23 import java.util.Map;
\r
24 import java.util.Set;
\r
25 import org.apache.commons.lang3.StringUtils;
\r
26 import org.onap.ccsdk.features.model.ConfigModelConstant;
\r
27 import org.onap.ccsdk.features.model.ConfigModelException;
\r
28 import org.onap.ccsdk.features.model.data.CapabilityAssignment;
\r
29 import org.onap.ccsdk.features.model.data.NodeTemplate;
\r
30 import org.onap.ccsdk.features.model.data.ResourceAssignment;
\r
31 import org.onap.ccsdk.features.model.utils.TopologicalSortingUtils;
\r
32 import org.onap.ccsdk.features.model.utils.TransformationUtils;
\r
34 import com.att.eelf.configuration.EELFLogger;
\r
35 import com.att.eelf.configuration.EELFManager;
\r
37 public class ResourceAssignmentValidator {
\r
38 private static EELFLogger logger = EELFManager.getInstance().getLogger(ResourceAssignmentValidator.class);
\r
39 private List<ResourceAssignment> assignments;
\r
40 private Map<String, ResourceAssignment> resourceAssignmentMap = new HashMap<>();
\r
41 private StringBuilder validationMessage = new StringBuilder();
\r
43 public ResourceAssignmentValidator(NodeTemplate nodeTemplate) throws ConfigModelException {
\r
45 if (nodeTemplate != null && nodeTemplate.getCapabilities() != null) {
\r
46 CapabilityAssignment capabilityAssignment =
\r
47 nodeTemplate.getCapabilities().get(ConfigModelConstant.CAPABILITY_PROPERTY_MAPPING);
\r
48 if (capabilityAssignment != null && capabilityAssignment.getProperties() != null) {
\r
49 Object mappingObject =
\r
50 capabilityAssignment.getProperties().get(ConfigModelConstant.CAPABILITY_PROPERTY_MAPPING);
\r
51 if (mappingObject != null) {
\r
52 String mappingContent = TransformationUtils.getJson(mappingObject);
\r
53 if (StringUtils.isNotBlank(mappingContent)) {
\r
55 TransformationUtils.getListfromJson(mappingContent, ResourceAssignment.class);
\r
58 .append(String.format("Failed to transform Mapping Content (%s) ", mappingContent));
\r
59 throw new ConfigModelException(
\r
60 String.format("Failed to transform Mapping Content (%s) ", mappingContent));
\r
67 public ResourceAssignmentValidator(List<ResourceAssignment> assignments) {
\r
68 this.assignments = assignments;
\r
72 * This is a validateResourceAssignment to validate the Topology Template
\r
75 * @throws ConfigModelException
\r
77 public boolean validateResourceAssignment() throws ConfigModelException {
\r
78 if (assignments != null && !assignments.isEmpty()) {
\r
79 validateDuplicateDictionaryKeys();
\r
80 validateCyclicDependencty();
\r
81 if (validationMessage.length() > 0) {
\r
82 logger.error("Resourece Assignment Validation : {}", validationMessage);
\r
83 throw new ConfigModelException("Resourece Assignment Validation :" + validationMessage.toString());
\r
89 @SuppressWarnings("squid:S3776")
\r
90 private void validateDuplicateDictionaryKeys() {
\r
91 this.assignments.forEach(resourceMapping -> {
\r
92 if (resourceMapping != null) {
\r
93 if (!resourceAssignmentMap.containsKey(resourceMapping.getName())) {
\r
94 resourceAssignmentMap.put(resourceMapping.getName(), resourceMapping);
\r
96 validationMessage.append(String.format("Duplicate Assignment Template Key (%s) is Present",
\r
97 resourceMapping.getName()));
\r
102 if (!assignments.isEmpty()) {
\r
103 Set<String> uniqueSet = new HashSet<>();
\r
104 for (ResourceAssignment resourceAssignment : assignments) {
\r
105 if (resourceAssignment != null) {
\r
106 boolean added = uniqueSet.add(resourceAssignment.getDictionaryName());
\r
108 validationMessage.append(
\r
109 String.format("Duplicate Assignment Dictionary Key (%s) present with Template Key (%s)",
\r
110 resourceAssignment.getDictionaryName(), resourceAssignment.getName()));
\r
117 private void validateCyclicDependencty() {
\r
118 TopologicalSortingUtils<ResourceAssignment> topologySorting = new TopologicalSortingUtils<>();
\r
119 this.resourceAssignmentMap.forEach((mappingKey, mapping) -> {
\r
120 if (mapping != null) {
\r
121 if (mapping.getDependencies() != null && !mapping.getDependencies().isEmpty()) {
\r
122 for (String dependency : mapping.getDependencies()) {
\r
123 topologySorting.add(resourceAssignmentMap.get(dependency), mapping);
\r
126 topologySorting.add(null, mapping);
\r
131 if (!topologySorting.isDag()) {
\r
132 String graph = getTopologicalGraph(topologySorting);
\r
133 validationMessage.append("Cyclic Dependency :" + graph);
\r
137 public String getTopologicalGraph(TopologicalSortingUtils<ResourceAssignment> topologySorting) {
\r
138 StringBuilder s = new StringBuilder();
\r
139 if (topologySorting != null) {
\r
140 Map<ResourceAssignment, List<ResourceAssignment>> neighbors = topologySorting.getNeighbors();
\r
142 neighbors.forEach((v, vs) -> {
\r
144 s.append("\n * -> [");
\r
145 List<ResourceAssignment> links = vs;
\r
146 for (ResourceAssignment resourceAssignment : links) {
\r
147 s.append("(" + resourceAssignment.getDictionaryName() + ":" + resourceAssignment.getName()
\r
152 s.append("\n (" + v.getDictionaryName() + ":" + v.getName() + ") -> [");
\r
153 List<ResourceAssignment> links = vs;
\r
154 for (ResourceAssignment resourceAssignment : links) {
\r
155 s.append("(" + resourceAssignment.getDictionaryName() + ":" + resourceAssignment.getName()
\r
162 return s.toString();
\r