Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

(#495) Allow milestones without issues #615

Merged
merged 4 commits into from
Apr 3, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 6 additions & 0 deletions docs/input/docs/configuration/default-configuration.md
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ create:
sha-section-line-format: "- `{1}\t{0}`"
allow-update-to-published: false
include-contributors: false
allow-milestone-without-issues: false
export:
include-created-date-in-title: false
created-date-string-format: ''
Expand Down Expand Up @@ -142,6 +143,11 @@ control the look and feel of the generated release notes.
in the release notes. A contributor is defined as someone who opened an issue
or submitted a PR. **NOTE:** This configuration option was added in version
0.19.0 of GitReleaseManager.
- **allow-milestone-without-issues**
- A boolean value which indicates whether an empty release will be created, when
no issues are found to be associated with a milestone. The contents of the
empty release can be controlled via the associated Scriban template.
**NOTE:** This configuration option was added in version 0.20.0 of GitReleaseManager.

See the [example create configuration section](create-configuration) to see an
example of how a footer can be configured.
Expand Down
17 changes: 8 additions & 9 deletions src/GitReleaseManager.Core.Tests/VcsServiceTests.cs
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,6 @@
using GitReleaseManager.Core.Model;
using GitReleaseManager.Core.Provider;
using GitReleaseManager.Core.ReleaseNotes;
using GitReleaseManager.Core.Templates;
using NSubstitute;
using NUnit.Framework;
using Serilog;
Expand Down Expand Up @@ -303,7 +302,7 @@ public async Task Should_Create_Release_From_Milestone()
{
var release = new Release();

_releaseNotesBuilder.BuildReleaseNotesAsync(OWNER, REPOSITORY, MILESTONE_TITLE, ReleaseTemplates.DEFAULT_NAME)
_releaseNotesBuilder.BuildReleaseNotesAsync(OWNER, REPOSITORY, MILESTONE_TITLE, null)
.Returns(Task.FromResult(RELEASE_NOTES));

_vcsProvider.GetReleaseAsync(OWNER, REPOSITORY, MILESTONE_TITLE)
Expand All @@ -315,7 +314,7 @@ public async Task Should_Create_Release_From_Milestone()
var result = await _vcsService.CreateReleaseFromMilestoneAsync(OWNER, REPOSITORY, MILESTONE_TITLE, MILESTONE_TITLE, null, null, false, null).ConfigureAwait(false);
result.ShouldBeSameAs(release);

await _releaseNotesBuilder.Received(1).BuildReleaseNotesAsync(OWNER, REPOSITORY, MILESTONE_TITLE, ReleaseTemplates.DEFAULT_NAME).ConfigureAwait(false);
await _releaseNotesBuilder.Received(1).BuildReleaseNotesAsync(OWNER, REPOSITORY, MILESTONE_TITLE, null).ConfigureAwait(false);
await _vcsProvider.Received(1).GetReleaseAsync(OWNER, REPOSITORY, MILESTONE_TITLE).ConfigureAwait(false);
await _vcsProvider.Received(1).CreateReleaseAsync(OWNER, REPOSITORY, Arg.Is<Release>(o =>
o.Body == RELEASE_NOTES &&
Expand All @@ -333,7 +332,7 @@ public async Task Should_Create_Release_From_Milestone_With_Assets()

var assetsCount = _assets.Count;

_releaseNotesBuilder.BuildReleaseNotesAsync(OWNER, REPOSITORY, MILESTONE_TITLE, ReleaseTemplates.DEFAULT_NAME)
_releaseNotesBuilder.BuildReleaseNotesAsync(OWNER, REPOSITORY, MILESTONE_TITLE, null)
.Returns(Task.FromResult(RELEASE_NOTES));

_vcsProvider.GetReleaseAsync(OWNER, REPOSITORY, MILESTONE_TITLE)
Expand All @@ -353,7 +352,7 @@ public async Task Should_Create_Release_From_Milestone_With_Assets()
null).ConfigureAwait(false);
result.ShouldBeSameAs(release);

await _releaseNotesBuilder.Received(1).BuildReleaseNotesAsync(OWNER, REPOSITORY, MILESTONE_TITLE, ReleaseTemplates.DEFAULT_NAME).ConfigureAwait(false);
await _releaseNotesBuilder.Received(1).BuildReleaseNotesAsync(OWNER, REPOSITORY, MILESTONE_TITLE, null).ConfigureAwait(false);
await _vcsProvider.Received(1).GetReleaseAsync(OWNER, REPOSITORY, MILESTONE_TITLE).ConfigureAwait(false);
await _vcsProvider.Received(1).CreateReleaseAsync(OWNER, REPOSITORY, Arg.Is<Release>(o =>
o.Body == RELEASE_NOTES &&
Expand Down Expand Up @@ -430,7 +429,7 @@ public async Task Should_Update_Published_Release_On_Creating_Release_From_Miles

_configuration.Create.AllowUpdateToPublishedRelease = updatePublishedRelease;

_releaseNotesBuilder.BuildReleaseNotesAsync(OWNER, REPOSITORY, MILESTONE_TITLE, ReleaseTemplates.DEFAULT_NAME)
_releaseNotesBuilder.BuildReleaseNotesAsync(OWNER, REPOSITORY, MILESTONE_TITLE, null)
.Returns(Task.FromResult(RELEASE_NOTES));

_vcsProvider.GetReleaseAsync(OWNER, REPOSITORY, MILESTONE_TITLE)
Expand All @@ -442,7 +441,7 @@ public async Task Should_Update_Published_Release_On_Creating_Release_From_Miles
var result = await _vcsService.CreateReleaseFromMilestoneAsync(OWNER, REPOSITORY, MILESTONE_TITLE, MILESTONE_TITLE, null, null, false, null).ConfigureAwait(false);
result.ShouldBeSameAs(release);

await _releaseNotesBuilder.Received(1).BuildReleaseNotesAsync(OWNER, REPOSITORY, MILESTONE_TITLE, ReleaseTemplates.DEFAULT_NAME).ConfigureAwait(false);
await _releaseNotesBuilder.Received(1).BuildReleaseNotesAsync(OWNER, REPOSITORY, MILESTONE_TITLE, null).ConfigureAwait(false);
await _vcsProvider.Received(1).GetReleaseAsync(OWNER, REPOSITORY, MILESTONE_TITLE).ConfigureAwait(false);
await _vcsProvider.Received(1).UpdateReleaseAsync(OWNER, REPOSITORY, release).ConfigureAwait(false);

Expand All @@ -458,7 +457,7 @@ public async Task Should_Throw_Exception_While_Updating_Published_Release_On_Cre

_configuration.Create.AllowUpdateToPublishedRelease = false;

_releaseNotesBuilder.BuildReleaseNotesAsync(OWNER, REPOSITORY, MILESTONE_TITLE, ReleaseTemplates.DEFAULT_NAME)
_releaseNotesBuilder.BuildReleaseNotesAsync(OWNER, REPOSITORY, MILESTONE_TITLE, null)
.Returns(Task.FromResult(RELEASE_NOTES));

_vcsProvider.GetReleaseAsync(OWNER, REPOSITORY, MILESTONE_TITLE)
Expand All @@ -467,7 +466,7 @@ public async Task Should_Throw_Exception_While_Updating_Published_Release_On_Cre
var ex = await Should.ThrowAsync<InvalidOperationException>(() => _vcsService.CreateReleaseFromMilestoneAsync(OWNER, REPOSITORY, MILESTONE_TITLE, MILESTONE_TITLE, null, null, false, null)).ConfigureAwait(false);
ex.Message.ShouldBe($"Release with tag '{MILESTONE_TITLE}' not in draft state, so not updating");

await _releaseNotesBuilder.Received(1).BuildReleaseNotesAsync(OWNER, REPOSITORY, MILESTONE_TITLE, ReleaseTemplates.DEFAULT_NAME).ConfigureAwait(false);
await _releaseNotesBuilder.Received(1).BuildReleaseNotesAsync(OWNER, REPOSITORY, MILESTONE_TITLE, null).ConfigureAwait(false);
await _vcsProvider.Received(1).GetReleaseAsync(OWNER, REPOSITORY, MILESTONE_TITLE).ConfigureAwait(false);
}

Expand Down
1 change: 1 addition & 0 deletions src/GitReleaseManager.Core/Configuration/Config.cs
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,7 @@ public Config()
ShaSectionHeading = "SHA256 Hashes of the release artifacts",
ShaSectionLineFormat = "- `{1}\t{0}`",
AllowUpdateToPublishedRelease = false,
AllowMilestonesWithoutIssues = false,
IncludeContributors = false,
};

Expand Down
3 changes: 3 additions & 0 deletions src/GitReleaseManager.Core/Configuration/CreateConfig.cs
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,9 @@ public class CreateConfig
[YamlMember(Alias = "allow-update-to-published")]
public bool AllowUpdateToPublishedRelease { get; set; }

[YamlMember(Alias = "allow-milestone-without-issues")]
public bool AllowMilestonesWithoutIssues { get; set; }

[YamlMember(Alias = "include-contributors")]
public bool IncludeContributors { get; set; }
}
Expand Down
23 changes: 21 additions & 2 deletions src/GitReleaseManager.Core/ReleaseNotes/ReleaseNotesBuilder.cs
Original file line number Diff line number Diff line change
Expand Up @@ -35,7 +35,7 @@ public ReleaseNotesBuilder(IVcsProvider vcsProvider, ILogger logger, IFileSystem
_templateFactory = templateFactory;
}

public async Task<string> BuildReleaseNotesAsync(string user, string repository, string milestoneTitle, string template)
public async Task<string> BuildReleaseNotesAsync(string user, string repository, string milestoneTitle, string customTemplate)
{
_user = user;
_repository = repository;
Expand All @@ -58,12 +58,31 @@ public async Task<string> BuildReleaseNotesAsync(string user, string repository,

var numberOfCommits = await _vcsProvider.GetCommitsCountAsync(_user, _repository, @base, head).ConfigureAwait(false);

if (issues.Count == 0)
if (issues.Count == 0 && !_configuration.Create.AllowMilestonesWithoutIssues)
{
var logMessage = string.Format(CultureInfo.CurrentCulture, "No closed issues have been found for milestone {0}, or all assigned issues are meant to be excluded from release notes, aborting release creation.", _milestoneTitle);
throw new InvalidOperationException(logMessage);
}

// By default we use the custom template, if it was provided.
// Otherwise, we determine which template we should use.
var template = customTemplate;
if (string.IsNullOrWhiteSpace(template))
{
if (issues.Count == 0)
{
template = ReleaseTemplates.NO_ISSUES_NAME;
}
else if (_configuration.Create.IncludeContributors)
{
template = ReleaseTemplates.CONTRIBUTORS_NAME;
}
else
{
template = ReleaseTemplates.DEFAULT_NAME;
}
}

var commitsLink = _vcsProvider.GetCommitsUrl(_user, _repository, _targetMilestone?.Title, previousMilestone?.Title);

var issuesDict = GetIssuesDict(issues);
Expand Down
10 changes: 10 additions & 0 deletions src/GitReleaseManager.Core/Templates/no_issues/create/footer.sbn
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
{{ if config.create.include_footer }}

### {{ config.create.footer_heading }}

{{ if config.create.milestone_replace_text
replace_milestone_title config.create.footer_content config.create.milestone_replace_text milestone.target.title
else
config.create.footer_content
end
end }}
10 changes: 10 additions & 0 deletions src/GitReleaseManager.Core/Templates/no_issues/index.sbn
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
{{-
include 'release-info'
if milestone.target.description
include 'milestone'
end
include 'issues' | string.rstrip
if template_kind == "CREATE"
include 'create/footer'
end
~}}
2 changes: 2 additions & 0 deletions src/GitReleaseManager.Core/Templates/no_issues/issues.sbn
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@

This release had no issues associated with it.
2 changes: 2 additions & 0 deletions src/GitReleaseManager.Core/Templates/no_issues/milestone.sbn
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@

{{ milestone.target.description }}
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
{{
if commits.count > 0
}}As part of this release we had [{{ commits.count }} {{ commits.count | string.pluralize "commit" "commits" }}]({{ commits.html_url }}).
{{ end -}}
14 changes: 2 additions & 12 deletions src/GitReleaseManager.Core/VcsService.cs
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,6 @@
using GitReleaseManager.Core.Model;
using GitReleaseManager.Core.Provider;
using GitReleaseManager.Core.ReleaseNotes;
using GitReleaseManager.Core.Templates;
using Serilog;

namespace GitReleaseManager.Core
Expand Down Expand Up @@ -46,16 +45,7 @@ public async Task<Release> CreateEmptyReleaseAsync(string owner, string reposito

public async Task<Release> CreateReleaseFromMilestoneAsync(string owner, string repository, string milestone, string releaseName, string targetCommitish, IList<string> assets, bool prerelease, string templateFilePath)
{
var templatePath = _configuration.Create.IncludeContributors
? ReleaseTemplates.CONTRIBUTORS_NAME
: ReleaseTemplates.DEFAULT_NAME;

if (!string.IsNullOrWhiteSpace(templateFilePath))
{
templatePath = templateFilePath;
}

var releaseNotes = await _releaseNotesBuilder.BuildReleaseNotesAsync(owner, repository, milestone, templatePath).ConfigureAwait(false);
var releaseNotes = await _releaseNotesBuilder.BuildReleaseNotesAsync(owner, repository, milestone, templateFilePath).ConfigureAwait(false);
var release = await CreateReleaseAsync(owner, repository, releaseName, milestone, releaseNotes, prerelease, targetCommitish, assets).ConfigureAwait(false);

return release;
Expand All @@ -67,7 +57,7 @@ public async Task<Release> CreateReleaseFromInputFileAsync(string owner, string

_logger.Verbose("Reading release notes from: '{FilePath}'", inputFilePath);

var releaseNotes = File.ReadAllText(inputFilePath);
var releaseNotes = await File.ReadAllTextAsync(inputFilePath).ConfigureAwait(false);
var release = await CreateReleaseAsync(owner, repository, name, name, releaseNotes, prerelease, targetCommitish, assets).ConfigureAwait(false);

return release;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -74,17 +74,38 @@ public async Task SingleMilestone()
var configuration = ConfigurationProvider.Provide(currentDirectory, fileSystem);
configuration.IssueLabelsExclude.Add("Internal Refactoring"); // This is necessary to generate the release notes for GitReleaseManager version 0.12.0

// Indicate whether you want to include the 'Contributors' section in the release notes
// Indicate that we want to include the 'Contributors' section in the release notes
configuration.Create.IncludeContributors = true;

// Pick the template based on whether you want to include the 'Contributors' section in the release notes
var templatePath = configuration.Create.IncludeContributors
? ReleaseTemplates.CONTRIBUTORS_NAME
: ReleaseTemplates.DEFAULT_NAME;
var vcsProvider = new GitHubProvider(_gitHubClient, _mapper, _graphQlClient);
var releaseNotesBuilder = new ReleaseNotesBuilder(vcsProvider, _logger, fileSystem, configuration, new TemplateFactory(fileSystem, configuration, TemplateKind.Create));
var result = await releaseNotesBuilder.BuildReleaseNotesAsync("GitTools", "GitReleaseManager", "0.12.0", string.Empty).ConfigureAwait(false); // 0.12.0 contains a mix of issues and PRs
Debug.WriteLine(result);
ClipBoardHelper.SetClipboard(result);
}
}

[Test]
[Explicit]
public async Task MilestoneWithoutIssues()
{
if (string.IsNullOrWhiteSpace(_token))
{
Assert.Inconclusive("Unable to locate credentials for accessing GitHub API");
}
else
{
var fileSystem = new FileSystem(new CreateSubOptions());
var currentDirectory = Environment.CurrentDirectory;

var configuration = ConfigurationProvider.Provide(currentDirectory, fileSystem);

// Indicate that we allow milestones without issues
configuration.Create.AllowMilestonesWithoutIssues = true;

var vcsProvider = new GitHubProvider(_gitHubClient, _mapper, _graphQlClient);
var releaseNotesBuilder = new ReleaseNotesBuilder(vcsProvider, _logger, fileSystem, configuration, new TemplateFactory(fileSystem, configuration, TemplateKind.Create));
var result = await releaseNotesBuilder.BuildReleaseNotesAsync("GitTools", "GitReleaseManager", "0.12.0", templatePath).ConfigureAwait(false); // 0.12.0 contains a mix of issues and PRs
var result = await releaseNotesBuilder.BuildReleaseNotesAsync("jericho", "_testing", "0.1.0", string.Empty).ConfigureAwait(false); // There are no issues associated with milestone 0.1.0 in my testing repo.
Debug.WriteLine(result);
ClipBoardHelper.SetClipboard(result);
}
Expand Down
Loading