package husacct.analyse.task.reconstruct.algorithms.hu.layers.goldstein;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Set;
import java.util.TreeMap;
import org.apache.log4j.Logger;
import husacct.analyse.domain.IModelQueryService;
import husacct.analyse.task.reconstruct.AnalyseReconstructConstants;
import husacct.analyse.task.reconstruct.ReconstructArchitecture;
import husacct.analyse.task.reconstruct.algorithms.GraphOfSuClusters;
import husacct.analyse.task.reconstruct.algorithms.Algorithm_SuperClass;
import husacct.analyse.task.reconstruct.dto.ReconstructArchitectureDTO;
import husacct.analyse.task.reconstruct.parameters.ReconstructArchitectureParameterDTO;
import husacct.common.dto.ModuleDTO;
import husacct.common.dto.SoftwareUnitDTO;
import husacct.common.enums.ModuleTypes;
public class Layers_Goldstein_HUSACCT_Algorithm_SelectedModule extends Algorithm_SuperClass{
private ModuleDTO selectedModule;
private final Logger logger = Logger.getLogger(ReconstructArchitecture.class);
private ArrayList<SoftwareUnitDTO> softwareUnitsToIncludeInAlgorithm = new ArrayList<SoftwareUnitDTO>();
private HashMap<String, SoftwareUnitDTO> softwareUnitsToExclude = new HashMap<String, SoftwareUnitDTO>();
private GraphOfSuClusters graphOfSuClusters; // Each node in the graph represents 1-n SoftwareUnits. If *, it is a cohesive cluster of SUs.
private TreeMap<Integer, Set<Integer>> layersWithNodesMap = new TreeMap<Integer, Set<Integer>>();
private HashSet<Integer> notAssignedNodes;
public Layers_Goldstein_HUSACCT_Algorithm_SelectedModule (IModelQueryService queryService) {
super(queryService);
graphOfSuClusters = new GraphOfSuClusters(this);
}
@Override
public void executeAlgorithm(ReconstructArchitectureDTO dto, IModelQueryService queryService) {
try {
selectedModule = dto.getSelectedModule();
if (selectedModule.logicalPath.equals("")) {
selectedModule.logicalPath = "**"; // Root of intended software architecture
selectedModule.type = "Root"; // Root of intended software architecture
}
// If the selectedModule is of type Facade or ExternalLibrary, nothing is done.
if ((selectedModule == null) || selectedModule.type.equals(ModuleTypes.EXTERNAL_LIBRARY.toString()) || selectedModule.type.equals(ModuleTypes.FACADE.toString())) {
return;
}
// Select the set of SUs to be used, and activate the component-identifying algorithm
if (selectedModule.logicalPath.equals("**")) {
SoftwareUnitDTO[] softwareUnitsInRoot = queryService.getSoftwareUnitsInRoot();
for (SoftwareUnitDTO rootUnit : softwareUnitsInRoot) {
if (!rootUnit.uniqueName.equals("xLibraries")) {
softwareUnitsToIncludeInAlgorithm.add(rootUnit);
}
}
if (softwareUnitsToIncludeInAlgorithm.size() == 1) {
SoftwareUnitDTO selectedSU = softwareUnitsToIncludeInAlgorithm.get(0);
softwareUnitsToIncludeInAlgorithm = getSetOfChildSoftwareUnits(selectedSU);
}
} else {
softwareUnitsToIncludeInAlgorithm = getRelevantSoftwareUnits();
}
graphOfSuClusters.initializeGraph(softwareUnitsToIncludeInAlgorithm, dto);
Set<Integer> allNodes = graphOfSuClusters.getNodes();
identifyLayers(allNodes);
addIdentifiedLayersToIntendedArchitecture();
} catch (Exception e) {
logger.warn(" Exception: " + e );
}
}
private void identifyLayers(Set<Integer> allNodes) {
notAssignedNodes = new HashSet<Integer>(allNodes);
// 1) Assign to bottom layer all nodes without dependencies on other nodes.
int layerId = 0;
assignNodesToBottomLayer(layerId, allNodes);
if (!layersWithNodesMap.isEmpty()) {
// 2) Iteratively create a new layer with dependencies on the lower layer and no dependencies on other
// nodes in notAssignedNodes (or, in case of cyclic dependencies not above the backCallThreshold)..
while (layersWithNodesMap.lastKey() > layerId) {
layerId ++;
identifyNewTopLayer(layerId);
}
}
}
private void assignNodesToBottomLayer(int layerId, Set<Integer> allNodes) {
HashSet<Integer> assignedNodesToBottomLayer = new HashSet<Integer>();
for (int fromNodeId : allNodes) {
boolean fromNodeUsesAnotherNode = false;
for (int toNodeId : allNodes) {
if (fromNodeId != toNodeId) {
int nrOfDependenciesFromNodeToToNode = graphOfSuClusters.getNumberOfDependencies(fromNodeId, toNodeId);
int nrOfDependenciesFromToNodeToNode = graphOfSuClusters.getNumberOfDependencies(toNodeId, fromNodeId);
if (nrOfDependenciesFromNodeToToNode > 0) {
if (nrOfDependenciesFromNodeToToNode > nrOfDependenciesFromToNodeToNode) { // Filters out dependencies below backCallThreshold.
fromNodeUsesAnotherNode = true;
}
}
}
}
if (!fromNodeUsesAnotherNode) {
// Assign unit to the bottom layer and remove it from notAssignedNodes
assignedNodesToBottomLayer.add(fromNodeId);
notAssignedNodes.remove(fromNodeId);
}
}
if (!assignedNodesToBottomLayer.isEmpty()) {
layerId ++;
layersWithNodesMap.put(layerId, assignedNodesToBottomLayer);
}
}
private void identifyNewTopLayer(int bottomLayerId) {
HashSet<Integer> assignedNodesNewTopLayer = new HashSet<Integer>();
for (int fromNodeId : notAssignedNodes) {
// Determine if fromNode uses a node in the bottom layer.
boolean fromNodeUsesNodeInBottomLayer = false;
for (int bottomLayerNodeId : layersWithNodesMap.get(bottomLayerId)) {
int nrOfDependenciesFromNodeToBootomLayerNode = graphOfSuClusters.getNumberOfDependencies(fromNodeId, bottomLayerNodeId);
if (nrOfDependenciesFromNodeToBootomLayerNode > 0) {
fromNodeUsesNodeInBottomLayer = true;
}
}
// Determine if fromNode uses a another node in notAssignedNodes.
boolean fromNodeUsesAnotherNotAssignedNode = false;
for (int toNodeId : notAssignedNodes) {
if (fromNodeId != toNodeId) {
int nrOfDependenciesFromNodeToToNode = graphOfSuClusters.getNumberOfDependencies(fromNodeId, toNodeId);
int nrOfDependenciesFromToNodeToNode = graphOfSuClusters.getNumberOfDependencies(toNodeId, fromNodeId);
if (nrOfDependenciesFromNodeToToNode > 0) {
if (nrOfDependenciesFromNodeToToNode > nrOfDependenciesFromToNodeToNode) { // Filters out dependencies below backCallThreshold.
fromNodeUsesAnotherNotAssignedNode = true;
}
}
}
}
if (fromNodeUsesNodeInBottomLayer && !fromNodeUsesAnotherNotAssignedNode) {
// Assign unit to the new top layer
assignedNodesNewTopLayer.add(fromNodeId);
}
}
// Create a new top layer only if the top layer contain at least one node.
if (!assignedNodesNewTopLayer.isEmpty()) {
int newTopLayerId = bottomLayerId + 1;
layersWithNodesMap.put(newTopLayerId, assignedNodesNewTopLayer);
// Remove added nodes from
for (int nodeId : assignedNodesNewTopLayer) {
notAssignedNodes.remove(nodeId);
}
}
}
// Returns the SUs assigned to selectedModule or, if only one SU is assigned, the children of this SU.
// In case the selectedModule is a Component, the SUs assigned to the interface should not be returned. Prepare.
private ArrayList<SoftwareUnitDTO> getRelevantSoftwareUnits() {
ArrayList<SoftwareUnitDTO> softwareUnitsToReturn = new ArrayList<SoftwareUnitDTO>();
addSoftwareUnitsAssignedToComponentInterface_To_softwareUnitsToExcludeMap();
int numberOfAssignedSoftwareUnits = defineService.getAssignedSoftwareUnitsOfModule(selectedModule.logicalPath).size();
if (numberOfAssignedSoftwareUnits > 1) {
for(String logicalSoftwarePathSelectedModule : defineService.getAssignedSoftwareUnitsOfModule(selectedModule.logicalPath)){
SoftwareUnitDTO suDTO = queryService.getSoftwareUnitByUniqueName(logicalSoftwarePathSelectedModule);
if (!softwareUnitsToExclude.containsKey(suDTO.uniqueName)) {
softwareUnitsToReturn.add(suDTO);
}
}
} else if (numberOfAssignedSoftwareUnits == 1){
SoftwareUnitDTO assignedSU = new SoftwareUnitDTO("", "", "", "");
for(String uniqueNameAssignedSU : defineService.getAssignedSoftwareUnitsOfModule(selectedModule.logicalPath)){
assignedSU = queryService.getSoftwareUnitByUniqueName(uniqueNameAssignedSU);
}
for (SoftwareUnitDTO subModule : getSetOfChildSoftwareUnits(assignedSU)){
if (!softwareUnitsToExclude.containsKey(subModule.uniqueName)) {
softwareUnitsToReturn.add(subModule);
}
}
}
return softwareUnitsToReturn;
}
private void addSoftwareUnitsAssignedToComponentInterface_To_softwareUnitsToExcludeMap() {
if (selectedModule.type.equals(ModuleTypes.COMPONENT.toString())) {
for (ModuleDTO subModule : selectedModule.subModules) {
if (subModule.type.equals(ModuleTypes.FACADE.toString())) {
defineService.getAssignedSoftwareUnitsOfModule(subModule.logicalPath);
for (String assignedUnitUniqueName : defineService.getAssignedSoftwareUnitsOfModule(subModule.logicalPath)) {
SoftwareUnitDTO assignedUnit = queryService.getSoftwareUnitByUniqueName(assignedUnitUniqueName);
if (!assignedUnit.name.isEmpty()) {
softwareUnitsToExclude.put(assignedUnit.uniqueName, assignedUnit);
}
}
}
}
}
}
/** Returns the first set of children (number of children >= 2) in the decomposition hierarchy of the parent SoftwareUnit
* @param parentSoftwareUnit (of a SoftwareUnit)
* @return ArrayList<String> with unique names of children, or an empty list, if no child SoftwareUnits are existing.
*/
private ArrayList<SoftwareUnitDTO> getSetOfChildSoftwareUnits(SoftwareUnitDTO parentSoftwareUnit) {
ArrayList<SoftwareUnitDTO> childSoftwareUnits = new ArrayList<SoftwareUnitDTO>();
SoftwareUnitDTO softwareUnit = parentSoftwareUnit;
while (childSoftwareUnits.size() < 2) {
SoftwareUnitDTO[] childUnits = (queryService.getChildUnitsOfSoftwareUnit(softwareUnit.uniqueName));
if (childUnits.length == 0) {
if (!softwareUnit.equals(parentSoftwareUnit)) {
childSoftwareUnits.add(softwareUnit);
}
break;
} else if ((childUnits.length == 1)) {
softwareUnit = childUnits[0];
} else if ((childUnits.length >= 2)) {
for (SoftwareUnitDTO childUnit : childUnits) {
childSoftwareUnits.add(childUnit);
}
}
}
return childSoftwareUnits;
}
private void addIdentifiedLayersToIntendedArchitecture() {
int highestLevelLayer = layersWithNodesMap.size();
if (highestLevelLayer >= 2) {
// Reverse the layer levels. The numbering of the layers within the
// intended architecture is different: the highest level layer has hierarchicalLevel = 1
int lowestLevelLayer = 1;
int raise = highestLevelLayer - lowestLevelLayer;
TreeMap<Integer, ArrayList<SoftwareUnitDTO>> layersWithSoftwareUnitsMap = new TreeMap<Integer, ArrayList<SoftwareUnitDTO>>();
for (int i = lowestLevelLayer; i <= highestLevelLayer; i++) {
Set<Integer> nodesOfLayer = layersWithNodesMap.get(i);
ArrayList<SoftwareUnitDTO> unitsOfLayer = new ArrayList<SoftwareUnitDTO>();
for (int nodeId : nodesOfLayer) {
unitsOfLayer.addAll(graphOfSuClusters.getSoftwareUnitsOfNode(nodeId));
}
int level = lowestLevelLayer + raise;
layersWithSoftwareUnitsMap.put(level, unitsOfLayer);
raise--;
}
int numberOfAddedLayers = 0;
for (int level : layersWithSoftwareUnitsMap.keySet()) {
ModuleDTO newModule = createModule_andAddItToReverseList("Layer" + level, selectedModule.logicalPath, "Layer", level, layersWithSoftwareUnitsMap.get(level));
if (!newModule.logicalPath.equals("")) {
numberOfAddedLayers ++;
}
}
logger.info(" Number of added Layers: " + numberOfAddedLayers);
}
}
@Override
public ReconstructArchitectureDTO getAlgorithmParameterSettings() {
ReconstructArchitectureDTO reconstructArchitecture = new ReconstructArchitectureDTO();
reconstructArchitecture.approachId = AnalyseReconstructConstants.Algorithms.Layers_Goldstein_HUSACCT_SelectedModule;
reconstructArchitecture.threshold = 10;
reconstructArchitecture.relationType = AnalyseReconstructConstants.RelationTypes.allDependencies;
reconstructArchitecture.granularity = AnalyseReconstructConstants.Granularities.PackagesAndClasses;
reconstructArchitecture.parameterDTOs = createParameterPanels();
return reconstructArchitecture;
}
private ArrayList<ReconstructArchitectureParameterDTO> createParameterPanels(){
ArrayList<ReconstructArchitectureParameterDTO> parameterDTOs = new ArrayList<>();
parameterDTOs.add(ReconstructArchitectureParameterDTO.DefaultParameterDTOs.createThresholdParameter(10));
parameterDTOs.add(ReconstructArchitectureParameterDTO.DefaultParameterDTOs.createRelationTypeParameter(AnalyseReconstructConstants.RelationTypes.allDependencies));
parameterDTOs.add(ReconstructArchitectureParameterDTO.DefaultParameterDTOs.createGranularityPanel(AnalyseReconstructConstants.Granularities.PackagesAndClasses));
return parameterDTOs;
}
}