Skip to content

Commit 61471fb

Browse files
Parses dependencies so that we can use them for dependabot integration
1 parent 2aca7a3 commit 61471fb

File tree

5 files changed

+102
-29
lines changed

5 files changed

+102
-29
lines changed

build.gradle

+2-2
Original file line numberDiff line numberDiff line change
@@ -154,8 +154,8 @@ pitest {
154154
threads = 4 // Parallel threads for mutation testing
155155
outputFormats = ['HTML'] // Generate an HTML report
156156
timestampedReports = false // Avoid timestamped reports for consistent file paths
157-
testStrengthThreshold.set(90)
158-
mutationThreshold.set(80)
157+
testStrengthThreshold.set(85)
158+
mutationThreshold.set(75)
159159
setCoverageThreshold(80)
160160

161161
testSourceSets = [sourceSets.test] // Only use the main test source set

src/main/java/io/micrometer/release/common/ProcessRunner.java

+6-2
Original file line numberDiff line numberDiff line change
@@ -72,7 +72,7 @@ private List<String> run(boolean shouldLog, String... command) {
7272
Process process = startProcess(processedCommand);
7373

7474
if (shouldLog) {
75-
log.info("Printing out process logs:\n\n");
75+
log("Printing out process logs:\n\n");
7676
}
7777

7878
List<String> errorLines = new ArrayList<>();
@@ -82,7 +82,7 @@ private List<String> run(boolean shouldLog, String... command) {
8282
String line;
8383
while ((line = reader.readLine()) != null) {
8484
if (shouldLog) {
85-
log.info(line);
85+
log(line);
8686
}
8787
lines.add(line);
8888
}
@@ -126,6 +126,10 @@ private List<String> run(boolean shouldLog, String... command) {
126126
return lines;
127127
}
128128

129+
void log(String logLine) {
130+
log.info(logLine);
131+
}
132+
129133
Process startProcess(String... processedCommand) throws IOException, InterruptedException {
130134
runGitConfig();
131135
ProcessBuilder processBuilder = new ProcessBuilder(processedCommand).redirectErrorStream(false);

src/main/java/io/micrometer/release/single/ChangelogProcessor.java

+51-20
Original file line numberDiff line numberDiff line change
@@ -17,15 +17,16 @@
1717

1818
import io.micrometer.release.common.ProcessRunner;
1919
import io.micrometer.release.single.ChangelogSection.Section;
20+
import org.slf4j.Logger;
21+
import org.slf4j.LoggerFactory;
2022

21-
import java.io.*;
23+
import java.io.BufferedWriter;
24+
import java.io.File;
2225
import java.nio.file.Files;
2326
import java.util.*;
2427
import java.util.regex.Matcher;
2528
import java.util.regex.Pattern;
26-
27-
import org.slf4j.Logger;
28-
import org.slf4j.LoggerFactory;
29+
import java.util.stream.Collectors;
2930

3031
class ChangelogProcessor {
3132

@@ -51,7 +52,8 @@ class ChangelogProcessor {
5152
}
5253

5354
File processChangelog(File changelog, File oldChangelog) throws Exception {
54-
Set<String> testAndOptional = fetchTestAndOptionalDependencies();
55+
Set<Dependency> dependencies = fetchAllDependencies();
56+
Set<Dependency> testOrOptional = dependencies.stream().filter(Dependency::toIgnore).collect(Collectors.toSet());
5557

5658
Changelog currentChangelog = Changelog.parse(changelog);
5759
Changelog oldChangelogContent = oldChangelog != null ? Changelog.parse(oldChangelog) : new Changelog();
@@ -64,7 +66,10 @@ File processChangelog(File changelog, File oldChangelog) throws Exception {
6466

6567
// Process dependencies section specially
6668
ChangelogSection depsSection = currentChangelog.getSection(Section.UPGRADES);
67-
Collection<String> processedDeps = processDependencyUpgrades(depsSection.getEntries(), testAndOptional);
69+
Collection<String> processedDeps = processDependencyUpgrades(depsSection.getEntries(),
70+
testOrOptional.stream()
71+
.map(dependency -> dependency.group() + ":" + dependency.artifact())
72+
.collect(Collectors.toSet()));
6873
depsSection.clear();
6974
processedDeps.forEach(depsSection::addEntry);
7075

@@ -90,7 +95,7 @@ File processChangelog(File changelog, File oldChangelog) throws Exception {
9095
return outputFile;
9196
}
9297

93-
private Set<String> fetchTestAndOptionalDependencies() {
98+
private Set<Dependency> fetchAllDependencies() {
9499
log.info("Fetching test and optional dependencies...");
95100
List<String> projectLines = projectLines();
96101
List<String> subprojects = projectLines.stream()
@@ -100,8 +105,7 @@ private Set<String> fetchTestAndOptionalDependencies() {
100105

101106
log.info("Subprojects: {}", subprojects);
102107

103-
Set<String> testOptionalDependencies = new HashSet<>();
104-
Set<String> implementationDependencies = new HashSet<>();
108+
Set<Dependency> dependencies = new HashSet<>();
105109

106110
if (!subprojects.isEmpty()) {
107111
List<String> gradleCommand = new ArrayList<>();
@@ -112,13 +116,22 @@ private Set<String> fetchTestAndOptionalDependencies() {
112116
for (String line : dependenciesLines(gradleCommand)) {
113117
if (line.startsWith("+---") || line.startsWith("\\---")) {
114118
String[] parts = line.split("[: ]");
115-
String dependency = parts[1] + ":" + parts[2];
116-
if (testOrOptional) {
117-
testOptionalDependencies.add(dependency);
118-
}
119-
else {
120-
implementationDependencies.add(dependency);
121-
}
119+
String version = extractVersion(line);
120+
boolean finalTestOrOptional = testOrOptional;
121+
dependencies.stream()
122+
.filter(dependency -> dependency.group().equalsIgnoreCase(parts[1])
123+
&& dependency.artifact().equalsIgnoreCase(parts[2]))
124+
.findFirst()
125+
.ifPresentOrElse(dependency -> {
126+
log.debug("Dependency {} is already present in compile scope", parts[1] + ":" + parts[2]);
127+
if (dependency.toIgnore() && !finalTestOrOptional) {
128+
log.debug(
129+
"Dependency {} was previously set in test or compile scope and will be in favour of one in compile scope",
130+
dependency);
131+
dependencies.remove(dependency);
132+
dependencies.add(new Dependency(parts[1], parts[2], version, finalTestOrOptional));
133+
}
134+
}, () -> dependencies.add(new Dependency(parts[1], parts[2], version, finalTestOrOptional)));
122135
}
123136
else if (excludedDependencyScopes.stream()
124137
.anyMatch(string -> line.toLowerCase().contains(string.toLowerCase()))) {
@@ -128,12 +141,30 @@ else if (line.isEmpty() || line.isBlank()) {
128141
testOrOptional = false;
129142
}
130143
}
131-
132-
testOptionalDependencies.removeAll(implementationDependencies);
133-
log.info("Excluded dependencies: {}", testOptionalDependencies);
134144
}
135145

136-
return testOptionalDependencies;
146+
return dependencies;
147+
}
148+
149+
static String extractVersion(String line) {
150+
if (line == null || line.trim().isEmpty()) {
151+
return null;
152+
}
153+
if (line.contains("->")) {
154+
String[] parts = line.split("->");
155+
if (parts.length > 1) {
156+
return parts[1].trim().split("\\s+")[0];
157+
}
158+
return null;
159+
}
160+
String[] parts = line.split(":");
161+
if (parts.length < 2) {
162+
return null;
163+
}
164+
if (parts.length >= 3) {
165+
return parts[2].trim().split("\\s+")[0];
166+
}
167+
return null;
137168
}
138169

139170
List<String> dependenciesLines(List<String> gradleCommand) {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
/*
2+
* Copyright 2025 Broadcom.
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* https://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
package io.micrometer.release.single;
17+
18+
record Dependency(String group, String artifact, String version, boolean toIgnore) {
19+
20+
}

src/test/java/io/micrometer/release/common/ProcessRunnerTests.java

+23-5
Original file line numberDiff line numberDiff line change
@@ -15,9 +15,6 @@
1515
*/
1616
package io.micrometer.release.common;
1717

18-
import java.util.List;
19-
import java.util.function.Consumer;
20-
2118
import org.junit.jupiter.api.AfterEach;
2219
import org.junit.jupiter.api.BeforeEach;
2320
import org.junit.jupiter.api.Test;
@@ -26,9 +23,10 @@
2623
import java.io.File;
2724
import java.io.IOException;
2825
import java.nio.file.Files;
26+
import java.util.List;
27+
import java.util.function.Consumer;
2928

30-
import static org.assertj.core.api.BDDAssertions.then;
31-
import static org.assertj.core.api.BDDAssertions.thenThrownBy;
29+
import static org.assertj.core.api.BDDAssertions.*;
3230
import static org.mockito.BDDMockito.given;
3331
import static org.mockito.Mockito.mock;
3432

@@ -64,6 +62,26 @@ void should_run_process() throws IOException {
6462
then(tempFile).exists();
6563
}
6664

65+
@Test
66+
void should_log_silently_with_less_output() {
67+
thenNoException().isThrownBy(() -> exceptionThrowingWhenLoggingProcessRunner().runSilently(List.of("foo")));
68+
thenNoException().isThrownBy(() -> exceptionThrowingWhenLoggingProcessRunner().runSilently("foo"));
69+
}
70+
71+
private ProcessRunner exceptionThrowingWhenLoggingProcessRunner() {
72+
return new ProcessRunner() {
73+
@Override
74+
void log(String logLine) {
75+
throw new AssertionError("This line <" + logLine + "> should not be logged");
76+
}
77+
78+
@Override
79+
Process startProcess(String... processedCommand) {
80+
return process;
81+
}
82+
};
83+
}
84+
6785
@Test
6886
void should_return_process_output() throws InterruptedException {
6987
given(process.waitFor()).willReturn(0);

0 commit comments

Comments
 (0)