2014-11-06 16:11:45 +02:00
#!/usr/bin/env ruby
2015-04-02 12:05:26 +03:00
require " optparse "
require " pp "
require_relative " version "
2015-06-10 13:27:27 +03:00
require_relative " helper "
2014-11-17 17:54:13 +02:00
module GitHubChangelogGenerator
class Parser
2015-05-25 17:16:35 +03:00
# parse options with optparse
2014-11-17 17:54:13 +02:00
def self . parse_options
2015-05-25 17:16:35 +03:00
options = get_default_options
2015-03-27 00:21:42 +01:00
2015-05-25 17:16:35 +03:00
parser = setup_parser ( options )
parser . parse!
2015-06-11 16:10:13 +03:00
if options [ :user ] . nil? || options [ :project ] . nil?
2015-08-05 09:50:06 +02:00
detect_user_and_project ( options , ARGV [ 0 ] , ARGV [ 1 ] )
2015-06-11 16:10:13 +03:00
end
2015-05-25 17:16:35 +03:00
if ! options [ :user ] || ! options [ :project ]
puts parser . banner
exit
end
if options [ :verbose ]
2015-06-10 13:27:27 +03:00
Helper . log . info " Performing task with options: "
2015-05-25 17:16:35 +03:00
pp options
puts " "
end
options
end
2014-11-06 16:11:45 +02:00
2015-05-25 17:16:35 +03:00
# setup parsing options
def self . setup_parser ( options )
2015-03-26 15:43:47 +02:00
parser = OptionParser . new do | opts |
2015-04-02 12:05:26 +03:00
opts . banner = " Usage: github_changelog_generator [options] "
opts . on ( " -u " , " --user [USER] " , " Username of the owner of target GitHub repo " ) do | last |
2014-11-17 17:54:13 +02:00
options [ :user ] = last
end
2015-04-02 12:05:26 +03:00
opts . on ( " -p " , " --project [PROJECT] " , " Name of project on GitHub " ) do | last |
2014-11-17 17:54:13 +02:00
options [ :project ] = last
end
2015-04-02 12:05:26 +03:00
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 17:54:13 +02:00
options [ :token ] = last
end
2015-04-02 12:05:26 +03:00
opts . on ( " -f " , " --date-format [FORMAT] " , " Date format. Default is %Y-%m-%d " ) do | last |
2015-05-25 09:56:59 +03:00
options [ :date_format ] = last
2014-11-24 21:38:06 +02:00
end
2015-04-02 12:05:26 +03:00
opts . on ( " -o " , " --output [NAME] " , " Output file. Default is CHANGELOG.md " ) do | last |
2014-11-24 21:38:06 +02:00
options [ :output ] = last
end
2015-08-04 16:03:13 +02:00
opts . on ( " -b " , " --base [NAME] " , " Optional base file to append generated changes to. " ) do | last |
options [ :base ] = last
end
2015-06-12 11:23:32 +03:00
opts . on ( " --bugs-label [LABEL] " , " Setup custom label for bug-fixes section. Default is \" **Fixed bugs:** " " " ) do | v |
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 ( " --issues-label [LABEL] " , " Setup custom label for closed-issues section. Default is \" **Closed issues:** \" " ) do | v |
options [ :issue_prefix ] = v
end
2015-07-16 14:47:01 +03:00
opts . on ( " --header-label [LABEL] " , " Setup custom header label. Default is \" # Change Log \" " ) do | v |
options [ :header ] = v
end
2015-06-12 11:23:32 +03: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
2015-04-02 12:05:26 +03:00
opts . on ( " --[no-]issues " , " Include closed issues in changelog. Default is true " ) do | v |
2014-11-17 17:54:13 +02:00
options [ :issues ] = v
end
2015-04-02 12:05:26 +03:00
opts . on ( " --[no-]issues-wo-labels " , " Include closed issues without labels in changelog. Default is true " ) do | v |
2014-11-17 17:54:13 +02:00
options [ :add_issues_wo_labels ] = v
end
2015-04-02 12:05:26 +03:00
opts . on ( " --[no-]pr-wo-labels " , " Include pull requests without labels in changelog. Default is true " ) do | v |
2015-02-18 22:21:00 +02:00
options [ :add_pr_wo_labels ] = v
end
2015-04-02 12:05:26 +03:00
opts . on ( " --[no-]pull-requests " , " Include pull-requests in changelog. Default is true " ) do | v |
2014-11-17 17:54:13 +02:00
options [ :pulls ] = v
end
2015-04-02 12:05:26 +03:00
opts . on ( " --[no-]filter-by-milestone " , " Use milestone to detect when issue was resolved. Default is true " ) do | last |
2014-12-22 15:41:20 +02:00
options [ :filter_issues_by_milestone ] = last
end
2015-04-02 12:05:26 +03:00
opts . on ( " --[no-]author " , " Add author of pull-request in the end. Default is true " ) do | author |
2015-02-18 18:27:57 +02:00
options [ :author ] = author
end
2015-04-02 12:05:26 +03:00
opts . on ( " --unreleased-only " , " Generate log from unreleased closed issues only. " ) do | v |
2015-02-18 20:14:39 +02:00
options [ :unreleased_only ] = v
end
2015-04-02 12:05:26 +03:00
opts . on ( " --[no-]unreleased " , " Add to log unreleased closed issues. Default is true " ) do | v |
2015-02-18 18:27:57 +02:00
options [ :unreleased ] = v
2014-11-19 13:53:36 +02:00
end
2015-04-02 12:05:26 +03:00
opts . on ( " --unreleased-label [label] " , " Add to log unreleased closed issues. Default is true " ) do | v |
2015-03-03 17:43:22 +02:00
options [ :unreleased_label ] = v
end
2015-04-02 12:05:26 +03:00
opts . on ( " --[no-]compare-link " , " Include compare link (Full Changelog) between older version and newer version. Default is true " ) do | v |
2015-01-26 21:30:27 +09:00
options [ :compare_link ] = v
end
2015-07-16 15:29:00 +03: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 21:50:33 +02:00
options [ :include_labels ] = list
2014-11-19 15:45:24 +02:00
end
2015-04-02 12:05:26 +03: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 |
2015-02-18 22:08:10 +02:00
options [ :exclude_labels ] = list
end
2015-06-11 16:35:38 +03: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
2015-05-26 16:01:23 +03:00
opts . on ( " --between-tags x,y,z " , Array , " Change log will be filled only between specified tags " ) do | list |
2015-05-25 17:16:35 +03:00
options [ :between_tags ] = list
end
2015-08-04 14:56:39 +02:00
opts . on ( " --exclude-tags x,y,z " , Array , " Change log will exclude specified tags " ) do | list |
2015-05-26 16:01:23 +03:00
options [ :exclude_tags ] = list
end
2015-08-04 14:58:25 +02:00
opts . on ( " --since-tag x " , " Change log will start after specified tag " ) do | v |
options [ :since_tag ] = v
end
2015-08-24 17:29:07 +03:00
opts . on ( " --due-tag x " , " Change log will end before specified tag " ) do | v |
options [ :due_tag ] = v
end
2015-04-02 12:05:26 +03:00
opts . on ( " --max-issues [NUMBER] " , Integer , " Max number of issues to fetch from GitHub. Default is unlimited " ) do | max |
2015-03-20 18:41:15 -07:00
options [ :max_issues ] = max
end
2015-08-05 09:35:03 +02: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
2015-04-02 12:05:26 +03:00
opts . on ( " --github-site [URL] " , " The Enterprise Github site on which your project is hosted. " ) do | last |
2014-12-19 16:55:42 -05:00
options [ :github_site ] = last
end
2015-04-02 12:05:26 +03:00
opts . on ( " --github-api [URL] " , " The enterprise endpoint to use for your Github API. " ) do | last |
2014-12-19 16:55:42 -05:00
options [ :github_endpoint ] = last
end
2015-04-02 12:05:26 +03:00
opts . on ( " --simple-list " , " Create simple list from issues and pull requests. Default is false. " ) do | v |
2015-02-25 16:37:34 +02:00
options [ :simple_list ] = v
end
2015-04-30 20:23:33 -03: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
2015-04-02 12:05:26 +03:00
opts . on ( " --[no-]verbose " , " Run verbosely. Default is true " ) do | v |
2015-02-25 14:18:19 +02:00
options [ :verbose ] = v
end
2015-04-02 12:05:26 +03:00
opts . on ( " -v " , " --version " , " Print version number " ) do | _v |
2014-11-17 17:54:13 +02:00
puts " Version: #{ GitHubChangelogGenerator :: VERSION } "
exit
end
2015-04-02 12:05:26 +03:00
opts . on ( " -h " , " --help " , " Displays Help " ) do
2014-11-24 21:33:03 +02:00
puts opts
exit
end
2015-03-26 15:43:47 +02:00
end
2015-05-25 17:16:35 +03:00
parser
end
2014-11-17 17:54:13 +02:00
2015-05-25 17:16:35 +03:00
# just get default options
def self . get_default_options
options = {
tag1 : nil ,
tag2 : nil ,
date_format : " %Y-%m-%d " ,
output : " CHANGELOG.md " ,
2015-08-04 16:03:13 +02:00
base : " HISTORY.md " ,
2015-05-25 17:16:35 +03: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 ,
2015-06-11 16:35:38 +03:00
enhancement_labels : %w( enhancement Enhancement ) ,
bug_labels : %w( bug Bug ) ,
2015-06-11 16:43:45 +03:00
exclude_labels : %w( duplicate question invalid wontfix Duplicate Question Invalid Wontfix ) ,
2015-05-25 17:16:35 +03:00
max_issues : nil ,
simple_list : false ,
verbose : true ,
2015-07-16 14:47:01 +03:00
header : " # Change Log " ,
2015-05-25 17:16:35 +03:00
merge_prefix : " **Merged pull requests:** " ,
issue_prefix : " **Closed issues:** " ,
bug_prefix : " **Fixed bugs:** " ,
enhancement_prefix : " **Implemented enhancements:** " ,
git_remote : " origin "
}
2015-04-04 02:30:02 +03:00
options
end
2015-05-25 17:16:35 +03:00
# Detects user and project from git
2015-08-05 09:50:06 +02:00
def self . detect_user_and_project ( options , arg0 = nil , arg1 = nil )
options [ :user ] , options [ :project ] = user_project_from_option ( arg0 , arg1 , options [ :github_site ] )
2015-05-25 13:02:10 +03:00
if ! options [ :user ] || ! options [ :project ]
if ENV [ " RUBYLIB " ] =~ / ruby-debug-ide /
options [ :user ] = " skywinder "
options [ :project ] = " changelog_test "
else
2015-05-25 13:34:37 +03:00
remote = ` git config --get remote. #{ options [ :git_remote ] } .url `
options [ :user ] , options [ :project ] = user_project_from_remote ( remote )
2015-05-25 13:02:10 +03:00
end
end
end
2015-05-25 14:21:23 +03:00
# Try to find user and project name from git remote output
#
# @param [String] output of git remote command
# @return [Array] user and project
2015-05-25 15:39:24 +03:00
def self . user_project_from_option ( arg0 , arg1 , github_site = nil )
2015-05-25 14:21:23 +03:00
user = nil
project = nil
2015-05-25 15:39:24 +03:00
github_site || = " github.com "
if arg0 && ! arg1
2015-02-06 16:36:52 +02:00
# this match should parse strings such "https://github.com/skywinder/Github-Changelog-Generator" or "skywinder/Github-Changelog-Generator" to user and name
2015-05-25 15:39:24 +03:00
puts arg0
2015-05-25 14:21:23 +03:00
match = / (?:.+ #{ Regexp . escape ( github_site ) } \/ )?(.+) \/ (.+) / . match ( arg0 )
2014-12-11 18:04:11 +02:00
2015-02-06 16:36:52 +02:00
begin
2015-02-25 19:02:41 +02:00
param = match [ 2 ] . nil?
2015-02-06 16:36:52 +02:00
rescue
2015-05-25 14:21:23 +03:00
puts " Can't detect user and name from first parameter: ' #{ arg0 } ' -> exit' "
2015-02-06 16:36:52 +02:00
exit
end
if param
2014-12-11 18:04:11 +02:00
exit
else
2015-05-25 14:21:23 +03:00
user = match [ 1 ]
project = match [ 2 ]
2014-12-11 18:04:11 +02:00
end
2014-11-06 16:11:45 +02:00
end
2015-05-25 14:21:23 +03:00
[ user , project ]
2015-05-25 13:02:10 +03:00
end
2014-11-06 16:11:45 +02:00
2015-05-25 13:34:37 +03:00
# Try to find user and project name from git remote output
2015-05-25 13:02:10 +03:00
#
2015-05-25 13:34:37 +03:00
# @param [String] output of git remote command
2015-05-25 14:21:23 +03:00
# @return [Array] user and project
2015-05-25 13:34:37 +03:00
def self . user_project_from_remote ( remote )
2015-05-25 13:02:10 +03:00
# try to find repo in format:
# origin git@github.com:skywinder/Github-Changelog-Generator.git (fetch)
# git@github.com:skywinder/Github-Changelog-Generator.git
regex1 = / .*(?:[: \/ ])((?:-| \ w| \ .)*) \/ ((?:-| \ w| \ .)*)(?: \ .git).* /
# try to find repo in format:
# origin https://github.com/skywinder/ChangelogMerger (fetch)
# https://github.com/skywinder/ChangelogMerger
regex2 = / .* \/ ((?:-| \ w| \ .)*) \/ ((?:-| \ w| \ .)*).* /
remote_structures = [ regex1 , regex2 ]
user = nil
project = nil
remote_structures . each do | regex |
matches = Regexp . new ( regex ) . match ( remote )
if matches && matches [ 1 ] && matches [ 2 ]
puts " Detected user: #{ matches [ 1 ] } , project: #{ matches [ 2 ] } "
user = matches [ 1 ]
project = matches [ 2 ]
2014-11-17 17:54:13 +02:00
end
2015-05-25 13:02:10 +03:00
break unless matches . nil?
2014-11-17 17:54:13 +02:00
end
2015-05-25 13:02:10 +03:00
[ user , project ]
2014-11-06 16:11:45 +02:00
end
end
2015-05-25 13:34:37 +03:00
if __FILE__ == $PROGRAM_NAME
2015-05-25 14:21:23 +03:00
remote = " invalid reference to project "
p user_project_from_option ( ARGV [ 0 ] , ARGV [ 1 ] , remote )
2015-05-25 13:34:37 +03:00
end
2015-01-14 13:56:44 -06:00
end