Skip to content

Commit 880a924

Browse files
Added triggering of dependabot via changing the dependabot.yml change
1 parent f0eb81f commit 880a924

File tree

7 files changed

+343
-55
lines changed

7 files changed

+343
-55
lines changed

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

+2-2
Original file line numberDiff line numberDiff line change
@@ -86,7 +86,7 @@ else if (line.isEmpty() || line.isBlank()) {
8686
return dependencies;
8787
}
8888

89-
static String extractVersion(String line) {
89+
private static String extractVersion(String line) {
9090
if (line == null || line.trim().isEmpty()) {
9191
return null;
9292
}
@@ -112,7 +112,7 @@ public List<String> dependenciesLines(List<String> gradleCommand) {
112112
}
113113

114114
public List<String> projectLines() {
115-
return processRunner.runSilently("./gradlew", "projects");
115+
return processRunner.runSilently(List.of("./gradlew", "projects"));
116116
}
117117

118118
}

src/main/java/io/micrometer/release/train/DependencyVerifier.java

+87-17
Original file line numberDiff line numberDiff line change
@@ -22,11 +22,14 @@
2222
import org.slf4j.LoggerFactory;
2323

2424
import java.io.File;
25+
import java.nio.file.Files;
26+
import java.nio.file.Path;
2527
import java.time.ZonedDateTime;
2628
import java.util.HashSet;
2729
import java.util.List;
2830
import java.util.Set;
2931
import java.util.concurrent.TimeUnit;
32+
import java.util.stream.Collectors;
3033

3134
import static java.time.format.DateTimeFormatter.RFC_1123_DATE_TIME;
3235

@@ -63,19 +66,25 @@ class DependencyVerifier {
6366
}
6467

6568
void verifyDependencies(String branch, String orgRepository) {
66-
cloneRepo(branch, orgRepository);
67-
GradleParser gradleParser = getGradleParser(branch);
69+
File clonedRepo = cloneRepo(branch, orgRepository);
70+
GradleParser gradleParser = getGradleParser(clonedRepo);
6871
log.info("Fetching all dependencies before dependabot...");
69-
Set<Dependency> dependenciesBeforeDependabot = gradleParser.fetchAllDependencies();
70-
Status status = dependabotUpdateStatus(orgRepository);
72+
Set<Dependency> dependenciesBeforeDependabot = micrometerOnly(gradleParser.fetchAllDependencies());
73+
Status status = dependabotUpdateStatus(clonedRepo, orgRepository);
7174
pullTheLatestRepoChanges();
72-
Set<Dependency> dependenciesAfterDependabot = gradleParser.fetchAllDependencies();
75+
Set<Dependency> dependenciesAfterDependabot = micrometerOnly(gradleParser.fetchAllDependencies());
7376
Set<Dependency> diff = new HashSet<>(dependenciesAfterDependabot);
7477
diff.removeAll(dependenciesBeforeDependabot);
7578
log.info("Dependency diff {}", diff);
7679
assertDependencyDiff(status, diff);
7780
}
7881

82+
private Set<Dependency> micrometerOnly(Set<Dependency> dependencies) {
83+
return dependencies.stream()
84+
.filter(dependency -> dependency.group().equalsIgnoreCase("io.micrometer"))
85+
.collect(Collectors.toSet());
86+
}
87+
7988
private void assertDependencyDiff(Status status, Set<Dependency> diff) {
8089
if (status == Status.NO_PRS) {
8190
log.info("There were no dependabot PRs, the dependency diff should have no differences");
@@ -94,27 +103,31 @@ private void assertDependencyDiff(Status status, Set<Dependency> diff) {
94103
}
95104
}
96105

97-
private Status dependabotUpdateStatus(String orgRepository) {
106+
private Status dependabotUpdateStatus(File clonedRepo, String orgRepository) {
98107
String githubServerTime = getGitHubServerTime();
99-
triggerDependabotCheck(orgRepository);
100-
log.info("Waiting {} {} for PRs to be created...", initialWait, timeUnit);
101-
sleep(initialWait);
102-
return waitForDependabotUpdates(githubServerTime);
108+
triggerDependabotCheck(clonedRepo);
109+
waitForDependabotJobsToFinish(orgRepository, githubServerTime);
110+
return waitForDependabotPrsToFinish(githubServerTime);
103111
}
104112

105-
private GradleParser getGradleParser(String branch) {
106-
ProcessRunner branchProcessRunner = new ProcessRunner(null, new File(branch));
113+
private GradleParser getGradleParser(File branch) {
114+
ProcessRunner branchProcessRunner = new ProcessRunner(null, branch);
107115
return gradleParser(branchProcessRunner);
108116
}
109117

110118
GradleParser gradleParser(ProcessRunner branchProcessRunner) {
111119
return new GradleParser(branchProcessRunner);
112120
}
113121

114-
private void cloneRepo(String branch, String orgRepository) {
122+
private File cloneRepo(String branch, String orgRepository) {
115123
log.info("Cloning out {} branch to folder {}", branch, branch);
116124
processRunner.run("git", "clone", "-b", branch, "--single-branch",
117125
"https://github.com/" + orgRepository + ".git", branch);
126+
return clonedDir(branch);
127+
}
128+
129+
File clonedDir(String branch) {
130+
return new File(branch);
118131
}
119132

120133
private void pullTheLatestRepoChanges() {
@@ -150,14 +163,71 @@ private String getGitHubServerTime() {
150163
return serverTime;
151164
}
152165

153-
private void triggerDependabotCheck(String orgRepository) {
166+
private void triggerDependabotCheck(File clonedRepo) {
154167
log.info("Will trigger a Dependabot check...");
155-
processRunner.run("gh", "api", "/repos/" + orgRepository + "/dispatches", "-X", "POST", "-F",
156-
"event_type=check-dependencies");
168+
try {
169+
String filePath = ".github/dependabot.yml";
170+
Path path = new File(clonedRepo, filePath).toPath();
171+
String fileContent = Files.readString(path);
172+
String triggerComment = "# Triggering dependabot";
173+
boolean hasComment = fileContent.trim().endsWith(triggerComment);
174+
if (hasComment) {
175+
fileContent = fileContent.substring(0, fileContent.lastIndexOf(triggerComment)).trim() + "\n";
176+
log.info("Removed trigger comment from dependabot.yml");
177+
}
178+
else {
179+
fileContent = fileContent.trim() + "\n" + triggerComment + "\n";
180+
log.info("Added trigger comment to dependabot.yml");
181+
}
182+
Files.writeString(path, fileContent);
183+
processRunner.run("git", "add", filePath);
184+
processRunner.run("git", "commit", "-m",
185+
"ci: " + (hasComment ? "Remove" : "Add") + " dependabot trigger comment");
186+
processRunner.run("git", "push");
187+
}
188+
catch (Exception e) {
189+
log.error("Failed to modify dependabot.yml", e);
190+
throw new IllegalStateException("Failed to trigger Dependabot check", e);
191+
}
157192
log.info("Triggered Dependabot check");
158193
}
159194

160-
private Status waitForDependabotUpdates(String githubServerTime) {
195+
private void waitForDependabotJobsToFinish(String orgRepository, String githubServerTime) {
196+
log.info("Waiting {} {} for Dependabot jobs to be created...", initialWait, timeUnit);
197+
sleep(initialWait);
198+
log.info("Waiting for Dependabot jobs to finish...");
199+
List<String> ids = processRunner.run("gh", "workflow", "list", "-R", orgRepository, "--json", "id,name", "--jq",
200+
".[] | select(.name==\"Dependabot Updates\") | .id");
201+
if (ids.isEmpty()) {
202+
throw new IllegalStateException("Could not find dependabot updates");
203+
}
204+
String id = ids.get(0);
205+
long startTime = System.currentTimeMillis();
206+
long timeoutMillis = timeUnit.toMillis(timeout / 2);
207+
while (System.currentTimeMillis() - startTime < timeoutMillis) {
208+
List<String> statuses = processRunner.run("gh", "run", "list", "--workflow=" + id, "-R", orgRepository,
209+
"--created=\">" + githubServerTime + "\"", "--json", "--jq", "'.[].status");
210+
if (statuses.isEmpty()) {
211+
log.info("No dependabot jobs found");
212+
}
213+
else {
214+
log.info("Found {} Dependabot jobs", statuses.size());
215+
boolean allCompleted = statuses.stream().allMatch(s -> s.equalsIgnoreCase("completed"));
216+
if (allCompleted) {
217+
log.info("All dependabot jobs completed");
218+
return;
219+
}
220+
}
221+
log.info("Not all Dependabot jobs processed, will try again...");
222+
sleep(waitBetweenRuns);
223+
}
224+
log.error("Failed! Dependabot jobs not processed within the provided timeout");
225+
throw new IllegalStateException("Timeout waiting for Dependabot jobs to complete");
226+
}
227+
228+
private Status waitForDependabotPrsToFinish(String githubServerTime) {
229+
log.info("Waiting {} {} for Dependabot PRs to be created...", initialWait, timeUnit);
230+
sleep(initialWait);
161231
long startTime = System.currentTimeMillis();
162232
long timeoutMillis = timeUnit.toMillis(timeout);
163233
while (System.currentTimeMillis() - startTime < timeoutMillis) {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,66 @@
1+
/**
2+
* Copyright 2025 the original author or authors.
3+
* <p>
4+
* Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except
5+
* in compliance with the License. You may obtain a copy of the License at
6+
* <p>
7+
* https://www.apache.org/licenses/LICENSE-2.0
8+
* <p>
9+
* Unless required by applicable law or agreed to in writing, software distributed under the License
10+
* is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express
11+
* or implied. See the License for the specific language governing permissions and limitations under
12+
* the License.
13+
*/
14+
package io.micrometer.release.common;
15+
16+
import static org.assertj.core.api.BDDAssertions.then;
17+
import static org.mockito.Mockito.mock;
18+
import static org.mockito.Mockito.verify;
19+
20+
import java.util.List;
21+
import java.util.Set;
22+
import org.junit.jupiter.api.Test;
23+
24+
class GradleParserTests {
25+
26+
ProcessRunner runner = mock();
27+
28+
GradleParser gradleParser = new GradleParser(runner) {
29+
@Override
30+
public List<String> dependenciesLines(List<String> gradleCommand) {
31+
return TestGradleParser.dependenciesFromFile();
32+
}
33+
34+
@Override
35+
public List<String> projectLines() {
36+
return TestGradleParser.projectLinesFromFile();
37+
}
38+
};
39+
40+
@Test
41+
void should_fetch_dependencies() {
42+
Set<Dependency> dependencies = gradleParser.fetchAllDependencies();
43+
44+
then(dependencies).hasSize(176);
45+
List<Dependency> micrometerDeps = dependencies.stream()
46+
.filter(dependency -> dependency.group().equalsIgnoreCase("io.micrometer"))
47+
.toList();
48+
then(micrometerDeps).hasSize(1);
49+
then(micrometerDeps.get(0)).isEqualTo(new Dependency("io.micrometer", "context-propagation", "1.1.1", false));
50+
}
51+
52+
@Test
53+
void should_run_dependencies() {
54+
new GradleParser(runner).dependenciesLines(List.of("foo", "bar"));
55+
56+
verify(runner).runSilently(List.of("foo", "bar"));
57+
}
58+
59+
@Test
60+
void should_run_project_lines() {
61+
new GradleParser(runner).projectLines();
62+
63+
verify(runner).runSilently(List.of("./gradlew", "projects"));
64+
}
65+
66+
}

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

+24-20
Original file line numberDiff line numberDiff line change
@@ -1,17 +1,15 @@
11
/**
22
* Copyright 2025 the original author or authors.
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-
*
3+
* <p>
4+
* Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except
5+
* in compliance with the License. You may obtain a copy of the License at
6+
* <p>
87
* 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.
8+
* <p>
9+
* Unless required by applicable law or agreed to in writing, software distributed under the License
10+
* is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express
11+
* or implied. See the License for the specific language governing permissions and limitations under
12+
* the License.
1513
*/
1614
package io.micrometer.release.common;
1715

@@ -24,7 +22,7 @@
2422

2523
public class TestGradleParser extends GradleParser {
2624

27-
private static List<String> expectedGradleCommand = List.of("./gradlew", "concurrency-tests:dependencies",
25+
static List<String> expectedGradleCommand = List.of("./gradlew", "concurrency-tests:dependencies",
2826
"docs:dependencies", "micrometer-benchmarks-core:dependencies", "micrometer-bom:dependencies",
2927
"micrometer-commons:dependencies", "micrometer-core:dependencies", "micrometer-jakarta9:dependencies",
3028
"micrometer-java11:dependencies", "micrometer-jetty11:dependencies", "micrometer-jetty12:dependencies",
@@ -55,7 +53,15 @@ public TestGradleParser() {
5553

5654
@Override
5755
public List<String> projectLines() {
58-
URL resource = TestGradleParser.class.getResource("/gradle/projects_output.txt");
56+
return projectLinesFromFile();
57+
}
58+
59+
static List<String> projectLinesFromFile() {
60+
return textFromFile("/gradle/projects_output.txt");
61+
}
62+
63+
private static List<String> textFromFile(String name) {
64+
URL resource = TestGradleParser.class.getResource(name);
5965
try {
6066
return Files.readAllLines(new File(resource.toURI()).toPath());
6167
}
@@ -67,13 +73,11 @@ public List<String> projectLines() {
6773
@Override
6874
public List<String> dependenciesLines(List<String> gradleCommand) {
6975
then(gradleCommand).isEqualTo(expectedGradleCommand);
70-
URL resource = TestGradleParser.class.getResource("/gradle/dependencies_output.txt");
71-
try {
72-
return Files.readAllLines(new File(resource.toURI()).toPath());
73-
}
74-
catch (Exception e) {
75-
throw new RuntimeException(e);
76-
}
76+
return dependenciesFromFile();
77+
}
78+
79+
static List<String> dependenciesFromFile() {
80+
return textFromFile("/gradle/dependencies_output.txt");
7781
}
7882

7983
}

src/test/java/io/micrometer/release/single/MilestoneUpdaterTests.java

+14-5
Original file line numberDiff line numberDiff line change
@@ -15,15 +15,14 @@
1515
*/
1616
package io.micrometer.release.single;
1717

18-
import static org.mockito.BDDMockito.given;
19-
import static org.mockito.Mockito.mock;
20-
import static org.mockito.Mockito.verify;
21-
2218
import io.micrometer.release.common.ProcessRunner;
19+
import org.junit.jupiter.api.Test;
2320

2421
import java.util.List;
2522

26-
import org.junit.jupiter.api.Test;
23+
import static org.mockito.BDDMockito.given;
24+
import static org.mockito.Mockito.mock;
25+
import static org.mockito.Mockito.verify;
2726

2827
class MilestoneUpdaterTests {
2928

@@ -43,4 +42,14 @@ void should_call_gh_api_to_close_a_milestone() {
4342
"-f", "state=closed");
4443
}
4544

45+
@Test
46+
void should_update_milestones() {
47+
MilestoneMigrator migrator = mock();
48+
MilestoneUpdater milestoneUpdater = new MilestoneUpdater(null, null, migrator);
49+
50+
milestoneUpdater.updateMilestones("v1.2.3");
51+
52+
verify(migrator).migrateMilestones("v1.2.3");
53+
}
54+
4655
}

0 commit comments

Comments
 (0)