5 changed files with 177 additions and 0 deletions
@ -0,0 +1,9 @@
@@ -0,0 +1,9 @@
|
||||
<?xml version="1.0"?> |
||||
<!DOCTYPE module PUBLIC |
||||
"-//Checkstyle//DTD Checkstyle Configuration 1.3//EN" |
||||
"https://checkstyle.org/dtds/configuration_1_3.dtd"> |
||||
<module name="com.puppycrawl.tools.checkstyle.Checker"> |
||||
<module name="io.spring.javaformat.checkstyle.SpringChecks"> |
||||
<property name="excludes" value="com.puppycrawl.tools.checkstyle.checks.javadoc.JavadocPackageCheck" /> |
||||
</module> |
||||
</module> |
||||
@ -0,0 +1,47 @@
@@ -0,0 +1,47 @@
|
||||
/* |
||||
* Copyright 2012-present the original author or authors. |
||||
* |
||||
* 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 |
||||
* |
||||
* https://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. |
||||
*/ |
||||
|
||||
|
||||
plugins { |
||||
id 'java-gradle-plugin' |
||||
id "checkstyle" |
||||
id "io.spring.javaformat" version "$javaFormatVersion" |
||||
|
||||
} |
||||
|
||||
repositories { |
||||
mavenCentral() |
||||
} |
||||
|
||||
checkstyle { |
||||
toolVersion = "${checkstyleToolVersion}" |
||||
} |
||||
|
||||
dependencies { |
||||
checkstyle("com.puppycrawl.tools:checkstyle:${checkstyle.toolVersion}") |
||||
checkstyle("io.spring.javaformat:spring-javaformat-checkstyle:${javaFormatVersion}") |
||||
|
||||
implementation("org.jgrapht:jgrapht-core:1.5.2") |
||||
} |
||||
|
||||
gradlePlugin { |
||||
plugins { |
||||
cycleDetectionPlugin { |
||||
id = "org.springframework.boot.cycle-detection" |
||||
implementationClass = "org.springframework.boot.build.cycledetection.CycleDetectionPlugin" |
||||
} |
||||
} |
||||
} |
||||
@ -0,0 +1,90 @@
@@ -0,0 +1,90 @@
|
||||
/* |
||||
* Copyright 2012-present the original author or authors. |
||||
* |
||||
* 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 |
||||
* |
||||
* https://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. |
||||
*/ |
||||
|
||||
package org.springframework.boot.build.cycledetection; |
||||
|
||||
import java.util.HashMap; |
||||
import java.util.LinkedHashSet; |
||||
import java.util.List; |
||||
import java.util.Map; |
||||
import java.util.Set; |
||||
|
||||
import org.gradle.api.GradleException; |
||||
import org.gradle.api.Plugin; |
||||
import org.gradle.api.Project; |
||||
import org.gradle.api.Task; |
||||
import org.gradle.api.execution.TaskExecutionGraph; |
||||
import org.gradle.api.initialization.Settings; |
||||
import org.jgrapht.Graph; |
||||
import org.jgrapht.alg.cycle.TarjanSimpleCycles; |
||||
import org.jgrapht.graph.DefaultDirectedGraph; |
||||
import org.jgrapht.graph.DefaultEdge; |
||||
|
||||
/** |
||||
* A {@link Settings} {@link Plugin plugin} to detect cycles between a build's projects. |
||||
* |
||||
* @author Andy Wilkinson |
||||
*/ |
||||
public class CycleDetectionPlugin implements Plugin<Settings> { |
||||
|
||||
@Override |
||||
public void apply(Settings settings) { |
||||
settings.getGradle().getTaskGraph().whenReady(this::detectCycles); |
||||
} |
||||
|
||||
private void detectCycles(TaskExecutionGraph taskGraph) { |
||||
Map<Project, Set<Project>> dependenciesByProject = getProjectsAndDependencies(taskGraph); |
||||
Graph<String, DefaultEdge> graph = createGraph(dependenciesByProject); |
||||
List<List<String>> cycles = findCycles(graph); |
||||
if (!cycles.isEmpty()) { |
||||
StringBuilder message = new StringBuilder("Cycles detected:\n"); |
||||
for (List<String> cycle : cycles) { |
||||
cycle.add(cycle.get(0)); |
||||
message.append(" " + String.join(" -> ", cycle) + "\n"); |
||||
} |
||||
throw new GradleException(message.toString()); |
||||
} |
||||
} |
||||
|
||||
private Map<Project, Set<Project>> getProjectsAndDependencies(TaskExecutionGraph taskGraph) { |
||||
Map<Project, Set<Project>> dependenciesByProject = new HashMap<>(); |
||||
for (Task task : taskGraph.getAllTasks()) { |
||||
Project project = task.getProject(); |
||||
Set<Project> dependencies = dependenciesByProject.computeIfAbsent(project, (p) -> new LinkedHashSet<>()); |
||||
taskGraph.getDependencies(task) |
||||
.stream() |
||||
.map(Task::getProject) |
||||
.filter((taskProject) -> !taskProject.equals(project)) |
||||
.forEach(dependencies::add); |
||||
} |
||||
return dependenciesByProject; |
||||
} |
||||
|
||||
private Graph<String, DefaultEdge> createGraph(Map<Project, Set<Project>> dependenciesByProject) { |
||||
Graph<String, DefaultEdge> graph = new DefaultDirectedGraph<>(DefaultEdge.class); |
||||
dependenciesByProject.keySet().forEach((project) -> graph.addVertex(project.getName())); |
||||
dependenciesByProject.forEach((project, dependencies) -> dependencies |
||||
.forEach((dependency) -> graph.addEdge(project.getName(), dependency.getName()))); |
||||
return graph; |
||||
} |
||||
|
||||
private List<List<String>> findCycles(Graph<String, DefaultEdge> graph) { |
||||
TarjanSimpleCycles<String, DefaultEdge> simpleCycles = new TarjanSimpleCycles<>(graph); |
||||
List<List<String>> cycles = simpleCycles.findSimpleCycles(); |
||||
return cycles; |
||||
} |
||||
|
||||
} |
||||
@ -0,0 +1,29 @@
@@ -0,0 +1,29 @@
|
||||
/* |
||||
* Copyright 2012-present the original author or authors. |
||||
* |
||||
* 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 |
||||
* |
||||
* https://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. |
||||
*/ |
||||
|
||||
|
||||
pluginManagement { |
||||
new File(rootDir.parentFile.parentFile, "gradle.properties").withInputStream { |
||||
def properties = new Properties() |
||||
properties.load(it) |
||||
properties.forEach(settings.ext::set) |
||||
gradle.rootProject { |
||||
properties.forEach(project.ext::set) |
||||
} |
||||
} |
||||
} |
||||
|
||||
include 'cycle-detection-plugin' |
||||
Loading…
Reference in new issue