package sonia.scm.plugin; import java.util.Arrays; import java.util.Collection; import java.util.List; import java.util.Set; import java.util.stream.Collectors; class SmpNodeBuilder { List buildNodeTree(ExplodedSmp[] smps) { return buildNodeTree(Arrays.asList(smps)); } List buildNodeTree(Collection smps) { Set availablePlugins = getAvailablePluginNames(smps); smps.forEach(smp -> this.assertDependenciesFulfilled(availablePlugins, smp)); List nodes = createNodes(smps); nodes.forEach(node -> nodes.forEach(otherNode -> appendIfDependsOnOtherNode(node, otherNode)) ); nodes.forEach(this::checkForCircle); return nodes; } private Set getAvailablePluginNames(Collection smps) { return smps.stream() .map(ExplodedSmp::getPlugin) .map(InstalledPluginDescriptor::getInformation) .map(PluginInformation::getName) .collect(Collectors.toSet()); } private void appendIfDependsOnOtherNode(PluginNode node, PluginNode otherNode) { if (node.getPlugin().getPlugin().getDependenciesInclusiveOptionals().contains(otherNode.getId()) && !otherNode.getChildren().contains(node)) { otherNode.addChild(node); } } private List createNodes(Collection smps) { return smps.stream().map(PluginNode::new).collect(Collectors.toList()); } private void assertDependenciesFulfilled(Set availablePlugins, ExplodedSmp smp) { smp.getPlugin().getDependencies().forEach(dependency -> { if (!availablePlugins.contains(dependency)) { throw new PluginNotInstalledException( String.format( "dependency %s of %s is not installed", dependency, smp.getPlugin().getInformation().getName() ) ); } }); } private void checkForCircle(PluginNode pluginNode) { pluginNode.getChildren().forEach(child -> assertDoesNotContainsDependency(pluginNode, child)); } private void assertDoesNotContainsDependency(PluginNode pluginNode, PluginNode childNode) { if (childNode == pluginNode) { throw new PluginCircularDependencyException("circular dependency detected: " + childNode.getId() + " depends on " + pluginNode.getId() + " and " + pluginNode.getId() + " depends on " + childNode.getId() ); } childNode.getChildren().forEach(child -> assertDoesNotContainsDependency(pluginNode, child)); } }