github-changelog-generator/lib/github_changelog_generator/parser.rb

237 lines
11 KiB
Ruby
Raw Normal View History

#!/usr/bin/env ruby
2016-09-22 17:16:29 +00:00
# frozen_string_literal: true
require "optparse"
require "pp"
2017-10-14 19:52:26 +00:00
require "github_changelog_generator/version"
require "github_changelog_generator/helper"
2014-11-17 15:54:13 +00:00
module GitHubChangelogGenerator
class Parser
2015-05-25 14:16:35 +00:00
# parse options with optparse
2014-11-17 15:54:13 +00:00
def self.parse_options
options = default_options
2015-03-26 23:21:42 +00:00
ParserFile.new(options).parse!
2015-09-15 18:38:41 +00:00
2015-05-25 14:16:35 +00:00
parser = setup_parser(options)
begin parser.parse!
rescue OptionParser::InvalidOption => e
abort [e, parser].join("\n")
end
2015-05-25 14:16:35 +00:00
unless options[:user] && options[:project]
2017-10-26 12:46:20 +00:00
warn "Configure which user and project to work on."
warn "Options --user and --project, or settings to that effect. See --help for more."
2017-10-27 07:29:17 +00:00
abort(parser.banner)
end
2015-05-25 14:16:35 +00:00
2017-10-14 19:52:26 +00:00
options.print_options
2015-09-01 06:11:52 +00:00
options
end
# Setup parsing options
#
# @param options [Options]
2017-10-14 19:52:26 +00:00
# @return [OptionParser]
2015-05-25 14:16:35 +00:00
def self.setup_parser(options)
2017-10-14 19:52:26 +00:00
OptionParser.new do |opts| # rubocop:disable Metrics/BlockLength
2017-10-15 20:33:26 +00:00
opts.banner = "Usage: github_changelog_generator --user USER --project PROJECT [options]"
opts.on("-u", "--user USER", "Username of the owner of target GitHub repo") do |last|
2014-11-17 15:54:13 +00:00
options[:user] = last
end
2017-10-15 20:33:26 +00:00
opts.on("-p", "--project PROJECT", "Name of project on GitHub") do |last|
2014-11-17 15:54:13 +00:00
options[:project] = last
end
opts.on("-t", "--token [TOKEN]", "To make more than 50 requests per hour your GitHub token is required. You can generate it at: https://github.com/settings/tokens/new") do |last|
2014-11-17 15:54:13 +00:00
options[:token] = last
end
2017-10-15 20:33:26 +00:00
opts.on("-f", "--date-format FORMAT", "Date format. Default is %Y-%m-%d") do |last|
2015-05-25 06:56:59 +00:00
options[:date_format] = last
2014-11-24 19:38:06 +00:00
end
opts.on("-o", "--output [NAME]", "Output file. Default is CHANGELOG.md") do |last|
2014-11-24 19:38:06 +00:00
options[:output] = last
end
2015-08-04 14:03:13 +00:00
opts.on("-b", "--base [NAME]", "Optional base file to append generated changes to.") do |last|
options[:base] = last
end
2016-07-22 06:28:06 +00:00
opts.on("--bugs-label [LABEL]", "Setup custom label for bug-fixes section. Default is \"**Fixed bugs:**\"") do |v|
2015-06-12 08:23:32 +00:00
options[:bug_prefix] = v
end
opts.on("--enhancement-label [LABEL]", "Setup custom label for enhancements section. Default is \"**Implemented enhancements:**\"") do |v|
options[:enhancement_prefix] = v
end
opts.on("--breaking-label [LABEL]", "Setup custom label for the breaking changes section. Default is \"**Breaking changes:**\"") do |v|
options[:breaking_prefix] = v
end
2015-06-12 08:23:32 +00:00
opts.on("--issues-label [LABEL]", "Setup custom label for closed-issues section. Default is \"**Closed issues:**\"") do |v|
options[:issue_prefix] = v
end
2017-12-13 21:06:00 +00:00
opts.on("--header-label [LABEL]", "Setup custom header label. Default is \"# Changelog\"") do |v|
2015-07-16 11:47:01 +00:00
options[:header] = v
end
Refactor generation code and allow custom sections There's a lot in this PR. - Added a Section class to more easily make the other changes and hopefully add flexibility for the future - Added an option called `configure_sections` that allows you create your own custom sections. It blows away all other sections and uses only the ones you give it. - Added an option called `add_sections` that allows you to add_sections to the default section set - Added an option called `include_merged` that can be used when configure_sections is defined. Configure sections blows away any and all default sections so to get this one back, you have to set this option. - Added tests for this stuff @HAIL9000 was a co-author. Because of a little git snafu, I accidentally squashed all of our work into one so it looks like it was just me. --- Refactor details: Before this change, the code in generator.rb and generator_generation.rb was conflated and method call flow went back and forth between the two files seemingly randomly. They also both defined the exact same class, which is un-ruby-ish. I tried to separate methods used for the whole changelog generation from methods used for specific parts of the changelog and move them into specific classes. I reasoned that a changelog is a series of "entries" of all tagged releases plus an extra entry for the unreleased entry. Each entry is comprised of a header and a series of "sections" for that entry. Each section is comprized of a list of issues and/or pull requests for that entry. So the log contains entries, entries contain sections, and sections contain issues & prs. I have structured the classes around this idea. - lib/github_changelog_generator/generator/generator.rb is for code related to generating the entire changelog. - lib/github_changelog_generator/generator/entry.rb is for code related to generating entries. - lib/github_changelog_generator/generator/section.rb is for code relating to geneating entry sections. Issues and PRs are already special objects, so it doesn't make sense to break those out into their own class.
2017-11-07 05:19:49 +00:00
opts.on("--configure-sections [Hash, String]", "Define your own set of sections which overrides all default sections") do |v|
options[:configure_sections] = v
end
opts.on("--add-sections [Hash, String]", "Add new sections but keep the default sections") do |v|
options[:add_sections] = v
end
2016-02-03 12:56:44 +00:00
opts.on("--front-matter [JSON]", "Add YAML front matter. Formatted as JSON because it's easier to add on the command line") do |v|
options[:frontmatter] = JSON.parse(v).to_yaml + "---\n"
end
2015-06-12 08:23:32 +00:00
opts.on("--pr-label [LABEL]", "Setup custom label for pull requests section. Default is \"**Merged pull requests:**\"") do |v|
options[:merge_prefix] = v
end
opts.on("--[no-]issues", "Include closed issues in changelog. Default is true") do |v|
2014-11-17 15:54:13 +00:00
options[:issues] = v
end
opts.on("--[no-]issues-wo-labels", "Include closed issues without labels in changelog. Default is true") do |v|
2014-11-17 15:54:13 +00:00
options[:add_issues_wo_labels] = v
end
opts.on("--[no-]pr-wo-labels", "Include pull requests without labels in changelog. Default is true") do |v|
2015-02-18 20:21:00 +00:00
options[:add_pr_wo_labels] = v
end
opts.on("--[no-]pull-requests", "Include pull-requests in changelog. Default is true") do |v|
2014-11-17 15:54:13 +00:00
options[:pulls] = v
end
opts.on("--[no-]filter-by-milestone", "Use milestone to detect when issue was resolved. Default is true") do |last|
options[:filter_issues_by_milestone] = last
end
opts.on("--[no-]author", "Add author of pull-request in the end. Default is true") do |author|
2015-02-18 16:27:57 +00:00
options[:author] = author
end
opts.on("--usernames-as-github-logins", "Use GitHub tags instead of Markdown links for the author of an issue or pull-request.") do |v|
options[:usernames_as_github_logins] = v
end
opts.on("--unreleased-only", "Generate log from unreleased closed issues only.") do |v|
options[:unreleased_only] = v
end
opts.on("--[no-]unreleased", "Add to log unreleased closed issues. Default is true") do |v|
2015-02-18 16:27:57 +00:00
options[:unreleased] = v
end
2016-11-04 21:53:49 +00:00
opts.on("--unreleased-label [label]", "Setup custom label for unreleased closed issues section. Default is \"**Unreleased:**\"") do |v|
options[:unreleased_label] = v
end
opts.on("--[no-]compare-link", "Include compare link (Full Changelog) between older version and newer version. Default is true") do |v|
2015-01-26 12:30:27 +00:00
options[:compare_link] = v
end
2015-07-16 12:29:00 +00:00
opts.on("--include-labels x,y,z", Array, "Only issues with the specified labels will be included in the changelog.") do |list|
2015-02-18 19:50:33 +00:00
options[:include_labels] = list
end
2017-10-27 06:49:52 +00:00
opts.on("--exclude-labels x,y,z", Array, "Issues with the specified labels will be always excluded from changelog. Default is 'duplicate,question,invalid,wontfix'") do |list|
options[:exclude_labels] = list
end
2015-06-11 13:35:38 +00:00
opts.on("--bug-labels x,y,z", Array, 'Issues with the specified labels will be always added to "Fixed bugs" section. Default is \'bug,Bug\'') do |list|
options[:bug_labels] = list
end
opts.on("--enhancement-labels x,y,z", Array, 'Issues with the specified labels will be always added to "Implemented enhancements" section. Default is \'enhancement,Enhancement\'') do |list|
options[:enhancement_labels] = list
end
opts.on("--breaking-labels x,y,z", Array, 'Issues with these labels will be added to a new section, called "Breaking Changes". Default is \'backwards-incompatible\'') do |list|
options[:breaking_labels] = list
end
opts.on("--issue-line-labels x,y,z", Array, 'The specified labels will be shown in brackets next to each matching issue. Use "ALL" to show all labels. Default is [].') do |list|
options[:issue_line_labels] = list
end
2017-12-13 21:06:00 +00:00
opts.on("--exclude-tags x,y,z", Array, "Changelog will exclude specified tags") do |list|
options[:exclude_tags] = list
end
2016-03-24 12:22:33 +00:00
opts.on("--exclude-tags-regex [REGEX]", "Apply a regular expression on tag names so that they can be excluded, for example: --exclude-tags-regex \".*\+\d{1,}\" ") do |last|
options[:exclude_tags_regex] = last
end
2017-12-13 21:06:00 +00:00
opts.on("--since-tag x", "Changelog will start after specified tag") do |v|
2015-08-04 12:58:25 +00:00
options[:since_tag] = v
end
2017-12-13 21:06:00 +00:00
opts.on("--due-tag x", "Changelog will end before specified tag") do |v|
2015-08-24 14:29:07 +00:00
options[:due_tag] = v
end
opts.on("--max-issues [NUMBER]", Integer, "Max number of issues to fetch from GitHub. Default is unlimited") do |max|
options[:max_issues] = max
end
2015-08-05 07:35:03 +00:00
opts.on("--release-url [URL]", "The URL to point to for release links, in printf format (with the tag as variable).") do |url|
options[:release_url] = url
end
opts.on("--github-site [URL]", "The Enterprise Github site on which your project is hosted.") do |last|
options[:github_site] = last
end
opts.on("--github-api [URL]", "The enterprise endpoint to use for your Github API.") do |last|
options[:github_endpoint] = last
end
opts.on("--simple-list", "Create simple list from issues and pull requests. Default is false.") do |v|
2015-02-25 14:37:34 +00:00
options[:simple_list] = v
end
2015-04-30 23:23:33 +00:00
opts.on("--future-release [RELEASE-VERSION]", "Put the unreleased changes in the specified release number.") do |future_release|
options[:future_release] = future_release
end
opts.on("--release-branch [RELEASE-BRANCH]", "Limit pull requests to the release branch, such as master or release") do |release_branch|
options[:release_branch] = release_branch
end
opts.on("--[no-]http-cache", "Use HTTP Cache to cache Github API requests (useful for large repos) Default is true.") do |http_cache|
options[:http_cache] = http_cache
end
opts.on("--cache-file [CACHE-FILE]", "Filename to use for cache. Default is github-changelog-http-cache in a temporary directory.") do |cache_file|
options[:cache_file] = cache_file
end
opts.on("--cache-log [CACHE-LOG]", "Filename to use for cache log. Default is github-changelog-logger.log in a temporary directory.") do |cache_log|
options[:cache_log] = cache_log
end
opts.on("--ssl-ca-file [PATH]", "Path to cacert.pem file. Default is a bundled lib/github_changelog_generator/ssl_certs/cacert.pem. Respects SSL_CA_PATH.") do |ssl_ca_file|
options[:ssl_ca_file] = ssl_ca_file
end
opts.on("--require x,y,z", Array, "Path to Ruby file(s) to require.") do |paths|
options[:require] = paths
end
opts.on("--[no-]verbose", "Run verbosely. Default is true") do |v|
2015-02-25 12:18:19 +00:00
options[:verbose] = v
end
opts.on("-v", "--version", "Print version number") do |_v|
2014-11-17 15:54:13 +00:00
puts "Version: #{GitHubChangelogGenerator::VERSION}"
exit
end
opts.on("-h", "--help", "Displays Help") do
2014-11-24 19:33:03 +00:00
puts opts
exit
end
2015-03-26 13:43:47 +00:00
end
2015-05-25 14:16:35 +00:00
end
2014-11-17 15:54:13 +00:00
# @return [Options] Default options
def self.default_options
Options.new(
2015-05-25 14:16:35 +00:00
date_format: "%Y-%m-%d",
output: "CHANGELOG.md",
2015-08-04 14:03:13 +00:00
base: "HISTORY.md",
2015-05-25 14:16:35 +00:00
issues: true,
add_issues_wo_labels: true,
add_pr_wo_labels: true,
pulls: true,
filter_issues_by_milestone: true,
author: true,
unreleased: true,
unreleased_label: "Unreleased",
compare_link: true,
enhancement_labels: ["enhancement", "Enhancement", "Type: Enhancement"],
bug_labels: ["bug", "Bug", "Type: Bug"],
exclude_labels: ["duplicate", "question", "invalid", "wontfix", "Duplicate", "Question", "Invalid", "Wontfix", "Meta: Exclude From Changelog"],
breaking_labels: %w[backwards-incompatible breaking],
Refactor generation code and allow custom sections There's a lot in this PR. - Added a Section class to more easily make the other changes and hopefully add flexibility for the future - Added an option called `configure_sections` that allows you create your own custom sections. It blows away all other sections and uses only the ones you give it. - Added an option called `add_sections` that allows you to add_sections to the default section set - Added an option called `include_merged` that can be used when configure_sections is defined. Configure sections blows away any and all default sections so to get this one back, you have to set this option. - Added tests for this stuff @HAIL9000 was a co-author. Because of a little git snafu, I accidentally squashed all of our work into one so it looks like it was just me. --- Refactor details: Before this change, the code in generator.rb and generator_generation.rb was conflated and method call flow went back and forth between the two files seemingly randomly. They also both defined the exact same class, which is un-ruby-ish. I tried to separate methods used for the whole changelog generation from methods used for specific parts of the changelog and move them into specific classes. I reasoned that a changelog is a series of "entries" of all tagged releases plus an extra entry for the unreleased entry. Each entry is comprised of a header and a series of "sections" for that entry. Each section is comprized of a list of issues and/or pull requests for that entry. So the log contains entries, entries contain sections, and sections contain issues & prs. I have structured the classes around this idea. - lib/github_changelog_generator/generator/generator.rb is for code related to generating the entire changelog. - lib/github_changelog_generator/generator/entry.rb is for code related to generating entries. - lib/github_changelog_generator/generator/section.rb is for code relating to geneating entry sections. Issues and PRs are already special objects, so it doesn't make sense to break those out into their own class.
2017-11-07 05:19:49 +00:00
configure_sections: {},
add_sections: {},
issue_line_labels: [],
2015-05-25 14:16:35 +00:00
max_issues: nil,
simple_list: false,
ssl_ca_file: nil,
2015-05-25 14:16:35 +00:00
verbose: true,
2017-12-13 21:06:00 +00:00
header: "# Changelog",
2015-05-25 14:16:35 +00:00
merge_prefix: "**Merged pull requests:**",
issue_prefix: "**Closed issues:**",
bug_prefix: "**Fixed bugs:**",
enhancement_prefix: "**Implemented enhancements:**",
breaking_prefix: "**Breaking changes:**",
http_cache: true,
require: []
2016-10-05 21:31:19 +00:00
)
2015-04-03 23:30:02 +00:00
end
end
end