/* * ============LICENSE_START======================================================= * Copyright (C) 2023 TechMahindra Ltd. * ================================================================================ * 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.api.impl; import java.io.Serializable; import java.util.ArrayList; import java.util.Collection; import java.util.Collections; import java.util.LinkedHashMap; import java.util.List; import java.util.Map; import java.util.Objects; import lombok.extern.slf4j.Slf4j; import org.onap.cps.api.CpsDeltaService; import org.onap.cps.spi.model.DataNode; import org.onap.cps.spi.model.DeltaReport; import org.onap.cps.spi.model.DeltaReportBuilder; import org.springframework.stereotype.Service; @Slf4j @Service public class CpsDeltaServiceImpl implements CpsDeltaService { @Override public List getDeltaReports(final Collection sourceDataNodes, final Collection targetDataNodes) { final List deltaReport = new ArrayList<>(); final Map xpathToSourceDataNodes = convertToXPathToDataNodesMap(sourceDataNodes); final Map xpathToTargetDataNodes = convertToXPathToDataNodesMap(targetDataNodes); deltaReport.addAll(getRemovedAndUpdatedDeltaReports(xpathToSourceDataNodes, xpathToTargetDataNodes)); deltaReport.addAll(getAddedDeltaReports(xpathToSourceDataNodes, xpathToTargetDataNodes)); return Collections.unmodifiableList(deltaReport); } private static Map convertToXPathToDataNodesMap( final Collection dataNodes) { final Map xpathToDataNode = new LinkedHashMap<>(); for (final DataNode dataNode : dataNodes) { xpathToDataNode.put(dataNode.getXpath(), dataNode); final Collection childDataNodes = dataNode.getChildDataNodes(); if (!childDataNodes.isEmpty()) { xpathToDataNode.putAll(convertToXPathToDataNodesMap(childDataNodes)); } } return xpathToDataNode; } private static List getRemovedAndUpdatedDeltaReports( final Map xpathToSourceDataNodes, final Map xpathToTargetDataNodes) { final List removedAndUpdatedDeltaReportEntries = new ArrayList<>(); for (final Map.Entry entry: xpathToSourceDataNodes.entrySet()) { final String xpath = entry.getKey(); final DataNode sourceDataNode = entry.getValue(); final DataNode targetDataNode = xpathToTargetDataNodes.get(xpath); final List deltaReports; if (targetDataNode == null) { deltaReports = getRemovedDeltaReports(xpath, sourceDataNode); } else { deltaReports = getUpdatedDeltaReports(xpath, sourceDataNode, targetDataNode); } removedAndUpdatedDeltaReportEntries.addAll(deltaReports); } return removedAndUpdatedDeltaReportEntries; } private static List getRemovedDeltaReports(final String xpath, final DataNode sourceDataNode) { final List removedDeltaReportEntries = new ArrayList<>(); final Map sourceDataNodeLeaves = sourceDataNode.getLeaves(); final DeltaReport removedDeltaReportEntry = new DeltaReportBuilder().actionRemove().withXpath(xpath) .withSourceData(sourceDataNodeLeaves).build(); removedDeltaReportEntries.add(removedDeltaReportEntry); return removedDeltaReportEntries; } private static List getUpdatedDeltaReports(final String xpath, final DataNode sourceDataNode, final DataNode targetDataNode) { final List updatedDeltaReportEntries = new ArrayList<>(); final Map, Map> updatedLeavesAsSourceDataToTargetData = getUpdatedLeavesBetweenSourceAndTargetDataNode(sourceDataNode.getLeaves(), targetDataNode.getLeaves()); addUpdatedLeavesToDeltaReport(xpath, updatedLeavesAsSourceDataToTargetData, updatedDeltaReportEntries); return updatedDeltaReportEntries; } private static Map, Map> getUpdatedLeavesBetweenSourceAndTargetDataNode( final Map leavesOfSourceDataNode, final Map leavesOfTargetDataNode) { final Map, Map> updatedLeavesAsSourceDataToTargetData = new LinkedHashMap<>(); final Map sourceDataInDeltaReport = new LinkedHashMap<>(); final Map targetDataInDeltaReport = new LinkedHashMap<>(); processLeavesPresentInSourceAndTargetDataNode(leavesOfSourceDataNode, leavesOfTargetDataNode, sourceDataInDeltaReport, targetDataInDeltaReport); processLeavesUniqueInTargetDataNode(leavesOfSourceDataNode, leavesOfTargetDataNode, sourceDataInDeltaReport, targetDataInDeltaReport); final boolean isUpdatedDataInDeltaReport = !sourceDataInDeltaReport.isEmpty() || !targetDataInDeltaReport.isEmpty(); if (isUpdatedDataInDeltaReport) { updatedLeavesAsSourceDataToTargetData.put(sourceDataInDeltaReport, targetDataInDeltaReport); } return updatedLeavesAsSourceDataToTargetData; } private static void processLeavesPresentInSourceAndTargetDataNode( final Map leavesOfSourceDataNode, final Map leavesOfTargetDataNode, final Map sourceDataInDeltaReport, final Map targetDataInDeltaReport) { for (final Map.Entry entry: leavesOfSourceDataNode.entrySet()) { final String key = entry.getKey(); final Serializable sourceLeaf = entry.getValue(); final Serializable targetLeaf = leavesOfTargetDataNode.get(key); compareLeaves(key, sourceLeaf, targetLeaf, sourceDataInDeltaReport, targetDataInDeltaReport); } } private static void processLeavesUniqueInTargetDataNode( final Map leavesOfSourceDataNode, final Map leavesOfTargetDataNode, final Map sourceDataInDeltaReport, final Map targetDataInDeltaReport) { final Map uniqueLeavesOfTargetDataNode = new LinkedHashMap<>(leavesOfTargetDataNode); uniqueLeavesOfTargetDataNode.keySet().removeAll(leavesOfSourceDataNode.keySet()); for (final Map.Entry entry: uniqueLeavesOfTargetDataNode.entrySet()) { final String key = entry.getKey(); final Serializable targetLeaf = entry.getValue(); final Serializable sourceLeaf = leavesOfSourceDataNode.get(key); compareLeaves(key, sourceLeaf, targetLeaf, sourceDataInDeltaReport, targetDataInDeltaReport); } } private static void compareLeaves(final String key, final Serializable sourceLeaf, final Serializable targetLeaf, final Map sourceDataInDeltaReport, final Map targetDataInDeltaReport) { if (sourceLeaf != null && targetLeaf != null) { if (!Objects.equals(sourceLeaf, targetLeaf)) { sourceDataInDeltaReport.put(key, sourceLeaf); targetDataInDeltaReport.put(key, targetLeaf); } } else if (sourceLeaf != null) { sourceDataInDeltaReport.put(key, sourceLeaf); } else if (targetLeaf != null) { targetDataInDeltaReport.put(key, targetLeaf); } } private static void addUpdatedLeavesToDeltaReport(final String xpath, final Map, Map> updatedLeavesAsSourceDataToTargetData, final List updatedDeltaReportEntries) { for (final Map.Entry, Map> entry: updatedLeavesAsSourceDataToTargetData.entrySet()) { final DeltaReport updatedDataForDeltaReport = new DeltaReportBuilder().actionUpdate() .withXpath(xpath).withSourceData(entry.getKey()).withTargetData(entry.getValue()).build(); updatedDeltaReportEntries.add(updatedDataForDeltaReport); } } private static List getAddedDeltaReports(final Map xpathToSourceDataNodes, final Map xpathToTargetDataNodes) { final List addedDeltaReportEntries = new ArrayList<>(); final Map xpathToAddedNodes = new LinkedHashMap<>(xpathToTargetDataNodes); xpathToAddedNodes.keySet().removeAll(xpathToSourceDataNodes.keySet()); for (final Map.Entry entry: xpathToAddedNodes.entrySet()) { final String xpath = entry.getKey(); final DataNode dataNode = entry.getValue(); final DeltaReport addedDataForDeltaReport = new DeltaReportBuilder().actionAdd().withXpath(xpath) .withTargetData(dataNode.getLeaves()).build(); addedDeltaReportEntries.add(addedDataForDeltaReport); } return addedDeltaReportEntries; } }