Skip to content

Commit b5c5994

Browse files
Doesn't leave an empty dependency upgrade section
1 parent 9caeb01 commit b5c5994

File tree

5 files changed

+102
-47
lines changed

5 files changed

+102
-47
lines changed

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

+5-6
Original file line numberDiff line numberDiff line change
@@ -68,20 +68,19 @@ File processChangelog(File changelog, File oldChangelog) throws Exception {
6868
depsSection.clear();
6969
processedDeps.forEach(depsSection::addEntry);
7070

71-
// Write the result
7271
try (BufferedWriter writer = Files.newBufferedWriter(outputFile.toPath())) {
73-
// Write header
72+
7473
for (String line : currentChangelog.getHeader()) {
7574
writer.write(line + "\n");
7675
}
7776

78-
// Write sections
7977
for (ChangelogSection section : currentChangelog.getSections()) {
80-
writer.write("## " + section.getTitle() + "\n\n");
81-
8278
List<String> sortedEntries = new ArrayList<>(section.getEntries());
79+
if (sortedEntries.isEmpty()) {
80+
continue;
81+
}
8382
Collections.sort(sortedEntries);
84-
83+
writer.write("## " + section.getTitle() + "\n\n");
8584
for (String entry : sortedEntries) {
8685
writer.write(entry + "\n");
8786
}

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

+1-1
Original file line numberDiff line numberDiff line change
@@ -63,7 +63,7 @@ private MilestoneWithDeadline createMilestone(String version) {
6363

6464
// Using 17:00 UTC for consistency
6565
List<String> lines = processRunner.run("gh", "api", "/repos/" + ghRepo + "/milestones", "-X", "POST", "-f",
66-
"title=" + version, "-f", "due_on=" + dueDate.toString() + "T17:00:00Z");
66+
"title=" + version, "-f", "due_on=" + dueDate + "T17:00:00Z");
6767

6868
if (lines.isEmpty()) {
6969
throw new IllegalStateException("Could not create milestone " + version);

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

+41-32
Original file line numberDiff line numberDiff line change
@@ -15,24 +15,23 @@
1515
*/
1616
package io.micrometer.release.common;
1717

18-
import static org.assertj.core.api.Assertions.assertThat;
19-
2018
import com.fasterxml.jackson.core.JsonProcessingException;
2119
import com.fasterxml.jackson.databind.DeserializationFeature;
2220
import com.fasterxml.jackson.databind.JsonNode;
2321
import com.fasterxml.jackson.databind.ObjectMapper;
2422
import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule;
2523
import io.micrometer.release.JavaHomeFinder;
24+
import org.junit.jupiter.api.BeforeAll;
25+
import org.junit.jupiter.api.Tag;
26+
import org.slf4j.Logger;
27+
import org.slf4j.LoggerFactory;
2628

2729
import java.time.LocalDate;
2830
import java.util.ArrayList;
2931
import java.util.List;
3032
import java.util.Map;
3133

32-
import org.junit.jupiter.api.BeforeAll;
33-
import org.junit.jupiter.api.Tag;
34-
import org.slf4j.Logger;
35-
import org.slf4j.LoggerFactory;
34+
import static org.assertj.core.api.Assertions.assertThat;
3635

3736
@Tag("e2e")
3837
public interface GithubActions {
@@ -49,8 +48,8 @@ public interface GithubActions {
4948
static void resetsMilestones() throws InterruptedException {
5049
assertThat(System.getenv("GH_TOKEN")).as("GH_TOKEN env var must be set!").isNotBlank();
5150
log.info(
52-
"This test requires GH connection and will operate on [{}] repository. It's quite slow because it runs GH actions so please be patient...",
53-
REPO);
51+
"This test requires GH connection and will operate on [{}] repository. It's quite slow because it runs GH actions so please be patient...",
52+
REPO);
5453
resetMilestones();
5554
}
5655

@@ -65,8 +64,7 @@ static void runWorkflow(String workflowName, List<String> commands) {
6564
processRunner.run(commands);
6665
try {
6766
waitForWorkflowCompletion(workflowName);
68-
}
69-
catch (InterruptedException e) {
67+
} catch (InterruptedException e) {
7068
throw new IllegalStateException(e);
7169
}
7270
}
@@ -79,16 +77,19 @@ private static void waitForWorkflowCompletion(String workflowFile) throws Interr
7977
int maxAttempts = 30;
8078
int attempts = 0;
8179
while (!completed && attempts < maxAttempts) { // 5 minute timeout
82-
List<String> status = processRunner.run("gh", "run", "list", "--workflow", workflowFile, "--limit", "1");
83-
log.info("Workflow [{}] not completed yet - attempt [{}]/[{}]", workflowFile, attempts + 1, maxAttempts);
80+
List<String> status = processRunner.run("gh", "run", "list", "--workflow", workflowFile,
81+
"--limit", "1");
82+
log.info("Workflow [{}] not completed yet - attempt [{}]/[{}]", workflowFile,
83+
attempts + 1, maxAttempts);
8484
completed = status.stream().anyMatch(line -> line.contains("completed"));
8585
if (!completed) {
8686
Thread.sleep(10_000);
8787
attempts++;
8888
}
8989
}
9090
if (!completed) {
91-
throw new RuntimeException("Workflow " + workflowFile + " did not complete within timeout");
91+
throw new RuntimeException(
92+
"Workflow " + workflowFile + " did not complete within timeout");
9293
}
9394
log.info("Workflow [{}] completed successfully!", workflowFile);
9495
}
@@ -116,46 +117,54 @@ class GithubClient {
116117
GithubClient(String token, String repo) {
117118
this.repo = repo;
118119
this.processRunner = new ProcessRunner(repo).withEnvVars(
119-
Map.of("JAVA_HOME", JavaHomeFinder.findJavaHomePath(), "GH_TOKEN", token != null ? token : ""));
120+
Map.of("JAVA_HOME", JavaHomeFinder.findJavaHomePath(), "GH_TOKEN",
121+
token != null ? token : ""));
120122
this.objectMapper = new ObjectMapper().registerModule(new JavaTimeModule())
121123
.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
122124
}
123125

124126
public Release getRelease(String tag) throws JsonProcessingException {
125-
List<String> output = processRunner.run("gh", "api", "/repos/" + repo + "/releases/tags/" + tag);
126-
return parseReleaseFromJson(output.get(0));
127+
String output = String.join("\n",
128+
processRunner.run("gh", "api", "/repos/" + repo + "/releases/tags/" + tag));
129+
return parseReleaseFromJson(output);
127130
}
128131

129132
public Milestone getMilestoneByTitle(String title) throws JsonProcessingException {
130-
List<String> output = processRunner.run("gh", "api", "/repos/" + repo + "/milestones");
131-
return parseMilestoneFromJson(output.get(0), title);
133+
String output = String.join("\n",
134+
processRunner.run("gh", "api", "/repos/" + repo + "/milestones"));
135+
return parseMilestoneFromJson(output, title);
132136
}
133137

134-
public List<Issue> getIssuesForMilestone(int milestoneNumber) throws JsonProcessingException {
135-
List<String> output = processRunner.run("gh", "api",
136-
"/repos/" + repo + "/issues?milestone=" + milestoneNumber);
137-
return parseIssuesFromJson(output.get(0));
138+
public List<Issue> getIssuesForMilestone(int milestoneNumber)
139+
throws JsonProcessingException {
140+
String output = String.join("\n", processRunner.run("gh", "api",
141+
"/repos/" + repo + "/issues?milestone=" + milestoneNumber));
142+
return parseIssuesFromJson(output);
138143
}
139144

140-
public List<Issue> getClosedIssuesForMilestone(int milestoneNumber) throws JsonProcessingException {
141-
List<String> output = processRunner.run("gh", "api",
142-
"/repos/" + repo + "/issues?milestone=" + milestoneNumber + "&state=closed");
143-
return parseIssuesFromJson(output.get(0));
145+
public List<Issue> getClosedIssuesForMilestone(int milestoneNumber)
146+
throws JsonProcessingException {
147+
String output = String.join("\n", processRunner.run("gh", "api",
148+
"/repos/" + repo + "/issues?milestone=" + milestoneNumber + "&state=closed"));
149+
return parseIssuesFromJson(output);
144150
}
145151

146152
private Release parseReleaseFromJson(String json) throws JsonProcessingException {
147153
JsonNode root = objectMapper.readTree(json);
148154
return new Release(root.get("body").asText());
149155
}
150156

151-
private Milestone parseMilestoneFromJson(String json, String title) throws JsonProcessingException {
157+
private Milestone parseMilestoneFromJson(String json, String title)
158+
throws JsonProcessingException {
152159
JsonNode root = objectMapper.readTree(json);
153160
for (JsonNode milestone : root) {
154161
if (milestone.get("title").asText().equals(title)) {
155-
return new Milestone(milestone.get("number").asInt(), milestone.get("state").asText(),
156-
milestone.get("title").asText(),
157-
milestone.get("due_on") != null && !milestone.get("due_on").isNull()
158-
? LocalDate.parse(milestone.get("due_on").asText().substring(0, 10)) : null);
162+
return new Milestone(milestone.get("number").asInt(),
163+
milestone.get("state").asText(),
164+
milestone.get("title").asText(),
165+
milestone.get("due_on") != null && !milestone.get("due_on").isNull()
166+
? LocalDate.parse(milestone.get("due_on").asText().substring(0, 10))
167+
: null);
159168
}
160169
}
161170
throw new RuntimeException("Milestone with title " + title + " not found");
@@ -166,7 +175,7 @@ private List<Issue> parseIssuesFromJson(String json) throws JsonProcessingExcept
166175
List<Issue> issues = new ArrayList<>();
167176
for (JsonNode issue : root) {
168177
issues.add(new Issue(issue.get("number").asInt(), issue.get("state").asText(),
169-
issue.get("title").asText()));
178+
issue.get("title").asText()));
170179
}
171180
return issues;
172181
}

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

-4
Original file line numberDiff line numberDiff line change
@@ -62,10 +62,6 @@ class ChangelogProcessorTests {
6262

6363
ChangelogProcessor processor = testChangelogProcessor(output);
6464

65-
static ChangelogProcessor testChangelogProcessor() {
66-
return testChangelogProcessor(new File(ChangelogProcessor.OUTPUT_FILE));
67-
}
68-
6965
static ChangelogProcessor testChangelogProcessor(File output) {
7066
return new ChangelogProcessor(new ProcessRunner(), output) {
7167
@Override

src/test/java/io/micrometer/release/train/TrainGithubActionsE2eTests.java

+55-4
Original file line numberDiff line numberDiff line change
@@ -37,12 +37,63 @@ static void should_go_through_whole_flow() {
3737
}
3838

3939
@Test
40-
void should_verify_release_notes_content() throws JsonProcessingException {
40+
void should_verify_release_notes_content_for_ga() throws JsonProcessingException {
4141
Release release = githubClient.getRelease("v0.1.1");
4242

43-
assertThat(release.body()).isEqualToIgnoringWhitespace("""
44-
TODO
45-
""");
43+
assertThat(release.body()).isEqualToIgnoringWhitespace(
44+
"""
45+
## :star: New Features
46+
47+
- Closed enhancement in generic 0.1.x [#8](https://github.com/marcingrzejszczak/gh-actions-test/issues/8)
48+
49+
## :lady_beetle: Bug Fixes
50+
51+
- Closed bug in concrete 0.1.1 [#12](https://github.com/marcingrzejszczak/gh-actions-test/issues/12)
52+
- Closed bug in generic 0.1.x [#9](https://github.com/marcingrzejszczak/gh-actions-test/issues/9)
53+
""");
54+
}
55+
56+
@Test
57+
void should_verify_release_notes_content_for_M2() throws JsonProcessingException {
58+
Release release = githubClient.getRelease("v0.2.0-M2");
59+
60+
assertThat(release.body()).isEqualToIgnoringWhitespace(
61+
"""
62+
## :star: New Features
63+
64+
- Closed enhancement in generic 0.1.x [#8](https://github.com/marcingrzejszczak/gh-actions-test/issues/8)
65+
- Closed enhancement in generic 0.2.x [#14](https://github.com/marcingrzejszczak/gh-actions-test/issues/14)
66+
67+
## :lady_beetle: Bug Fixes
68+
69+
- Closed bug in concrete 0.1.1 [#12](https://github.com/marcingrzejszczak/gh-actions-test/issues/12)
70+
- Closed bug in concrete 0.2.0-M2 [#18](https://github.com/marcingrzejszczak/gh-actions-test/issues/18)
71+
- Closed bug in generic 0.1.x [#9](https://github.com/marcingrzejszczak/gh-actions-test/issues/9)
72+
- Closed bug in generic 0.2.x [#15](https://github.com/marcingrzejszczak/gh-actions-test/issues/15)
73+
""");
74+
}
75+
76+
@Test
77+
void should_verify_release_notes_content_for_RC1() throws JsonProcessingException {
78+
Release release = githubClient.getRelease("v1.0.0-RC1");
79+
80+
assertThat(release.body()).isEqualToIgnoringWhitespace(
81+
"""
82+
## :star: New Features
83+
84+
- Closed enhancement in generic 0.1.x [#8](https://github.com/marcingrzejszczak/gh-actions-test/issues/8)
85+
- Closed enhancement in generic 0.2.x [#14](https://github.com/marcingrzejszczak/gh-actions-test/issues/14)
86+
- Closed enhancement in generic 1.0.x [#5](https://github.com/marcingrzejszczak/gh-actions-test/issues/5)
87+
88+
## :lady_beetle: Bug Fixes
89+
90+
- Closed bug in concrete 0.1.1 [#12](https://github.com/marcingrzejszczak/gh-actions-test/issues/12)
91+
- Closed bug in concrete 0.2.0-M2 [#18](https://github.com/marcingrzejszczak/gh-actions-test/issues/18)
92+
- Closed bug in concrete 1.0.0-RC1 [#3](https://github.com/marcingrzejszczak/gh-actions-test/issues/3)
93+
- Closed bug in generic 0.1.x [#9](https://github.com/marcingrzejszczak/gh-actions-test/issues/9)
94+
- Closed bug in generic 0.2.x [#15](https://github.com/marcingrzejszczak/gh-actions-test/issues/15)
95+
- Closed bug in generic 1.0.x [#4](https://github.com/marcingrzejszczak/gh-actions-test/issues/4)
96+
""");
4697
}
4798

4899
@ParameterizedTest

0 commit comments

Comments
 (0)