22
22
import org .slf4j .LoggerFactory ;
23
23
24
24
import java .io .File ;
25
+ import java .nio .file .Files ;
26
+ import java .nio .file .Path ;
25
27
import java .time .ZonedDateTime ;
26
28
import java .util .HashSet ;
27
29
import java .util .List ;
28
30
import java .util .Set ;
29
31
import java .util .concurrent .TimeUnit ;
32
+ import java .util .stream .Collectors ;
30
33
31
34
import static java .time .format .DateTimeFormatter .RFC_1123_DATE_TIME ;
32
35
@@ -63,19 +66,25 @@ class DependencyVerifier {
63
66
}
64
67
65
68
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 );
68
71
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 );
71
74
pullTheLatestRepoChanges ();
72
- Set <Dependency > dependenciesAfterDependabot = gradleParser .fetchAllDependencies ();
75
+ Set <Dependency > dependenciesAfterDependabot = micrometerOnly ( gradleParser .fetchAllDependencies () );
73
76
Set <Dependency > diff = new HashSet <>(dependenciesAfterDependabot );
74
77
diff .removeAll (dependenciesBeforeDependabot );
75
78
log .info ("Dependency diff {}" , diff );
76
79
assertDependencyDiff (status , diff );
77
80
}
78
81
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
+
79
88
private void assertDependencyDiff (Status status , Set <Dependency > diff ) {
80
89
if (status == Status .NO_PRS ) {
81
90
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) {
94
103
}
95
104
}
96
105
97
- private Status dependabotUpdateStatus (String orgRepository ) {
106
+ private Status dependabotUpdateStatus (File clonedRepo , String orgRepository ) {
98
107
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 );
103
111
}
104
112
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 );
107
115
return gradleParser (branchProcessRunner );
108
116
}
109
117
110
118
GradleParser gradleParser (ProcessRunner branchProcessRunner ) {
111
119
return new GradleParser (branchProcessRunner );
112
120
}
113
121
114
- private void cloneRepo (String branch , String orgRepository ) {
122
+ private File cloneRepo (String branch , String orgRepository ) {
115
123
log .info ("Cloning out {} branch to folder {}" , branch , branch );
116
124
processRunner .run ("git" , "clone" , "-b" , branch , "--single-branch" ,
117
125
"https://github.com/" + orgRepository + ".git" , branch );
126
+ return clonedDir (branch );
127
+ }
128
+
129
+ File clonedDir (String branch ) {
130
+ return new File (branch );
118
131
}
119
132
120
133
private void pullTheLatestRepoChanges () {
@@ -150,14 +163,71 @@ private String getGitHubServerTime() {
150
163
return serverTime ;
151
164
}
152
165
153
- private void triggerDependabotCheck (String orgRepository ) {
166
+ private void triggerDependabotCheck (File clonedRepo ) {
154
167
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
+ }
157
192
log .info ("Triggered Dependabot check" );
158
193
}
159
194
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 );
161
231
long startTime = System .currentTimeMillis ();
162
232
long timeoutMillis = timeUnit .toMillis (timeout );
163
233
while (System .currentTimeMillis () - startTime < timeoutMillis ) {
0 commit comments