pr_releasenotes 0.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
checksums.yaml ADDED
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA256:
3
+ metadata.gz: 950d6b6e778e173335598ed4574806b6bbd3db28575eccf5a3c40d33fac844c7
4
+ data.tar.gz: 9cd54bffe0f6d5ea60c3058f7e3cba1e7b09f29fe14da8f532398f24937f5ce0
5
+ SHA512:
6
+ metadata.gz: fb18ff14b3a0a87bbe379d4a8c4b5abedd6ece5d660d147af49b8caddbd3ee18c81a6d0dad566111a71a5939086ae313bdd215da48c13cb3ceff19799bc076a4
7
+ data.tar.gz: 2f12d003a12abdb59025ea5d1e200b16625548d85c141d5ba18f7d804df47e7a2c96316e63e7e2b9c028550949d8dc883532e5ac7ce498b36dfe2763040757c0
data/.gitignore ADDED
@@ -0,0 +1,11 @@
1
+ /.bundle/
2
+ /.yardoc
3
+ /Gemfile.lock
4
+ /_yardoc/
5
+ /coverage/
6
+ /doc/
7
+ /pkg/
8
+ /spec/reports/
9
+ /tmp/
10
+ /*.gem
11
+ .rakeTasks
data/.travis.yml ADDED
@@ -0,0 +1,9 @@
1
+ language: ruby
2
+ rvm:
3
+ - 2.0
4
+ - 2.1
5
+ - 2.2
6
+ - 2.3
7
+ script:
8
+ - rake clean
9
+ - rake build
data/Gemfile ADDED
@@ -0,0 +1,8 @@
1
+ source 'https://rubygems.org'
2
+
3
+ # For additional security under Bundler 1.x versions
4
+ # https://bundler.io/guides/git.html#security
5
+ git_source(:github) {|repo_name| "https://github.com/#{repo_name}.git" }
6
+
7
+ # Specify your gem's dependencies in pr_releasenotes.gemspec
8
+ gemspec
data/LICENSE ADDED
@@ -0,0 +1,201 @@
1
+ Apache License
2
+ Version 2.0, January 2004
3
+ http://www.apache.org/licenses/
4
+
5
+ TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
6
+
7
+ 1. Definitions.
8
+
9
+ "License" shall mean the terms and conditions for use, reproduction,
10
+ and distribution as defined by Sections 1 through 9 of this document.
11
+
12
+ "Licensor" shall mean the copyright owner or entity authorized by
13
+ the copyright owner that is granting the License.
14
+
15
+ "Legal Entity" shall mean the union of the acting entity and all
16
+ other entities that control, are controlled by, or are under common
17
+ control with that entity. For the purposes of this definition,
18
+ "control" means (i) the power, direct or indirect, to cause the
19
+ direction or management of such entity, whether by contract or
20
+ otherwise, or (ii) ownership of fifty percent (50%) or more of the
21
+ outstanding shares, or (iii) beneficial ownership of such entity.
22
+
23
+ "You" (or "Your") shall mean an individual or Legal Entity
24
+ exercising permissions granted by this License.
25
+
26
+ "Source" form shall mean the preferred form for making modifications,
27
+ including but not limited to software source code, documentation
28
+ source, and configuration files.
29
+
30
+ "Object" form shall mean any form resulting from mechanical
31
+ transformation or translation of a Source form, including but
32
+ not limited to compiled object code, generated documentation,
33
+ and conversions to other media types.
34
+
35
+ "Work" shall mean the work of authorship, whether in Source or
36
+ Object form, made available under the License, as indicated by a
37
+ copyright notice that is included in or attached to the work
38
+ (an example is provided in the Appendix below).
39
+
40
+ "Derivative Works" shall mean any work, whether in Source or Object
41
+ form, that is based on (or derived from) the Work and for which the
42
+ editorial revisions, annotations, elaborations, or other modifications
43
+ represent, as a whole, an original work of authorship. For the purposes
44
+ of this License, Derivative Works shall not include works that remain
45
+ separable from, or merely link (or bind by name) to the interfaces of,
46
+ the Work and Derivative Works thereof.
47
+
48
+ "Contribution" shall mean any work of authorship, including
49
+ the original version of the Work and any modifications or additions
50
+ to that Work or Derivative Works thereof, that is intentionally
51
+ submitted to Licensor for inclusion in the Work by the copyright owner
52
+ or by an individual or Legal Entity authorized to submit on behalf of
53
+ the copyright owner. For the purposes of this definition, "submitted"
54
+ means any form of electronic, verbal, or written communication sent
55
+ to the Licensor or its representatives, including but not limited to
56
+ communication on electronic mailing lists, source code control systems,
57
+ and issue tracking systems that are managed by, or on behalf of, the
58
+ Licensor for the purpose of discussing and improving the Work, but
59
+ excluding communication that is conspicuously marked or otherwise
60
+ designated in writing by the copyright owner as "Not a Contribution."
61
+
62
+ "Contributor" shall mean Licensor and any individual or Legal Entity
63
+ on behalf of whom a Contribution has been received by Licensor and
64
+ subsequently incorporated within the Work.
65
+
66
+ 2. Grant of Copyright License. Subject to the terms and conditions of
67
+ this License, each Contributor hereby grants to You a perpetual,
68
+ worldwide, non-exclusive, no-charge, royalty-free, irrevocable
69
+ copyright license to reproduce, prepare Derivative Works of,
70
+ publicly display, publicly perform, sublicense, and distribute the
71
+ Work and such Derivative Works in Source or Object form.
72
+
73
+ 3. Grant of Patent License. Subject to the terms and conditions of
74
+ this License, each Contributor hereby grants to You a perpetual,
75
+ worldwide, non-exclusive, no-charge, royalty-free, irrevocable
76
+ (except as stated in this section) patent license to make, have made,
77
+ use, offer to sell, sell, import, and otherwise transfer the Work,
78
+ where such license applies only to those patent claims licensable
79
+ by such Contributor that are necessarily infringed by their
80
+ Contribution(s) alone or by combination of their Contribution(s)
81
+ with the Work to which such Contribution(s) was submitted. If You
82
+ institute patent litigation against any entity (including a
83
+ cross-claim or counterclaim in a lawsuit) alleging that the Work
84
+ or a Contribution incorporated within the Work constitutes direct
85
+ or contributory patent infringement, then any patent licenses
86
+ granted to You under this License for that Work shall terminate
87
+ as of the date such litigation is filed.
88
+
89
+ 4. Redistribution. You may reproduce and distribute copies of the
90
+ Work or Derivative Works thereof in any medium, with or without
91
+ modifications, and in Source or Object form, provided that You
92
+ meet the following conditions:
93
+
94
+ (a) You must give any other recipients of the Work or
95
+ Derivative Works a copy of this License; and
96
+
97
+ (b) You must cause any modified files to carry prominent notices
98
+ stating that You changed the files; and
99
+
100
+ (c) You must retain, in the Source form of any Derivative Works
101
+ that You distribute, all copyright, patent, trademark, and
102
+ attribution notices from the Source form of the Work,
103
+ excluding those notices that do not pertain to any part of
104
+ the Derivative Works; and
105
+
106
+ (d) If the Work includes a "NOTICE" text file as part of its
107
+ distribution, then any Derivative Works that You distribute must
108
+ include a readable copy of the attribution notices contained
109
+ within such NOTICE file, excluding those notices that do not
110
+ pertain to any part of the Derivative Works, in at least one
111
+ of the following places: within a NOTICE text file distributed
112
+ as part of the Derivative Works; within the Source form or
113
+ documentation, if provided along with the Derivative Works; or,
114
+ within a display generated by the Derivative Works, if and
115
+ wherever such third-party notices normally appear. The contents
116
+ of the NOTICE file are for informational purposes only and
117
+ do not modify the License. You may add Your own attribution
118
+ notices within Derivative Works that You distribute, alongside
119
+ or as an addendum to the NOTICE text from the Work, provided
120
+ that such additional attribution notices cannot be construed
121
+ as modifying the License.
122
+
123
+ You may add Your own copyright statement to Your modifications and
124
+ may provide additional or different license terms and conditions
125
+ for use, reproduction, or distribution of Your modifications, or
126
+ for any such Derivative Works as a whole, provided Your use,
127
+ reproduction, and distribution of the Work otherwise complies with
128
+ the conditions stated in this License.
129
+
130
+ 5. Submission of Contributions. Unless You explicitly state otherwise,
131
+ any Contribution intentionally submitted for inclusion in the Work
132
+ by You to the Licensor shall be under the terms and conditions of
133
+ this License, without any additional terms or conditions.
134
+ Notwithstanding the above, nothing herein shall supersede or modify
135
+ the terms of any separate license agreement you may have executed
136
+ with Licensor regarding such Contributions.
137
+
138
+ 6. Trademarks. This License does not grant permission to use the trade
139
+ names, trademarks, service marks, or product names of the Licensor,
140
+ except as required for reasonable and customary use in describing the
141
+ origin of the Work and reproducing the content of the NOTICE file.
142
+
143
+ 7. Disclaimer of Warranty. Unless required by applicable law or
144
+ agreed to in writing, Licensor provides the Work (and each
145
+ Contributor provides its Contributions) on an "AS IS" BASIS,
146
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
147
+ implied, including, without limitation, any warranties or conditions
148
+ of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
149
+ PARTICULAR PURPOSE. You are solely responsible for determining the
150
+ appropriateness of using or redistributing the Work and assume any
151
+ risks associated with Your exercise of permissions under this License.
152
+
153
+ 8. Limitation of Liability. In no event and under no legal theory,
154
+ whether in tort (including negligence), contract, or otherwise,
155
+ unless required by applicable law (such as deliberate and grossly
156
+ negligent acts) or agreed to in writing, shall any Contributor be
157
+ liable to You for damages, including any direct, indirect, special,
158
+ incidental, or consequential damages of any character arising as a
159
+ result of this License or out of the use or inability to use the
160
+ Work (including but not limited to damages for loss of goodwill,
161
+ work stoppage, computer failure or malfunction, or any and all
162
+ other commercial damages or losses), even if such Contributor
163
+ has been advised of the possibility of such damages.
164
+
165
+ 9. Accepting Warranty or Additional Liability. While redistributing
166
+ the Work or Derivative Works thereof, You may choose to offer,
167
+ and charge a fee for, acceptance of support, warranty, indemnity,
168
+ or other liability obligations and/or rights consistent with this
169
+ License. However, in accepting such obligations, You may act only
170
+ on Your own behalf and on Your sole responsibility, not on behalf
171
+ of any other Contributor, and only if You agree to indemnify,
172
+ defend, and hold each Contributor harmless for any liability
173
+ incurred by, or claims asserted against, such Contributor by reason
174
+ of your accepting any such warranty or additional liability.
175
+
176
+ END OF TERMS AND CONDITIONS
177
+
178
+ APPENDIX: How to apply the Apache License to your work.
179
+
180
+ To apply the Apache License to your work, attach the following
181
+ boilerplate notice, with the fields enclosed by brackets "[]"
182
+ replaced with your own identifying information. (Don't include
183
+ the brackets!) The text should be enclosed in the appropriate
184
+ comment syntax for the file format. We also recommend that a
185
+ file or class name and description of purpose be included on the
186
+ same "printed page" as the copyright notice for easier
187
+ identification within third-party archives.
188
+
189
+ Copyright 2019 Bazaarvoice, Inc.
190
+
191
+ Licensed under the Apache License, Version 2.0 (the "License");
192
+ you may not use this file except in compliance with the License.
193
+ You may obtain a copy of the License at
194
+
195
+ http://www.apache.org/licenses/LICENSE-2.0
196
+
197
+ Unless required by applicable law or agreed to in writing, software
198
+ distributed under the License is distributed on an "AS IS" BASIS,
199
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
200
+ See the License for the specific language governing permissions and
201
+ limitations under the License.
data/OWNERS ADDED
@@ -0,0 +1,2 @@
1
+ # Owning team.
2
+ bulk-team@bazaarvoice.com
data/README.md ADDED
@@ -0,0 +1,93 @@
1
+ # pr_releasenotes
2
+
3
+ ## Overview
4
+
5
+ Create release notes for your github repo using the descriptions of pull requests in each release. Simply invoking:
6
+
7
+ $ pr_releasenotes -r <user/repo> -t <token> -s <latest_tagged_release>
8
+
9
+ will generate release notes from your latest release tag to the tip of the master branch.
10
+
11
+ ## Why this tool vs others
12
+
13
+ There are quite a few online tools for generating release notes based on github history, but all the ones I've found rely on commit messages. This makes them inflexible, since it would require rewriting commit history in order to make any changes to the generated notes.
14
+
15
+ This tool uses the pull request descriptions, so the release notes for any version can be updated at any time by simply updating the corresponding pull request's description and rerunning this tool.
16
+
17
+ Secondly, this tool provides the additional option to post the release notes back to the github releases page.
18
+
19
+ ## Installation
20
+
21
+ Add this line to your application's Gemfile:
22
+
23
+ ```ruby
24
+ gem 'pr_releasenotes'
25
+ ```
26
+
27
+ And then execute:
28
+
29
+ $ bundle
30
+
31
+ Or install it yourself as:
32
+
33
+ $ gem install pr_releasenotes
34
+
35
+ ## Usage
36
+
37
+ The minimum configuration necessary to run the tool is the repo, github token and a start release tag:
38
+
39
+ $ pr_releasenotes --repo <user/repo> --token <token> --start <latest_tagged_release>
40
+
41
+ Additional options can be specified either by passing a yaml configuration to the included executable:
42
+
43
+ $ pr_releasenotes --config <config.yaml> --token <token> --start <start_version> --end <end_version> --branch <non-default branch>
44
+
45
+ or by invoking the gem directly from ruby code:
46
+
47
+ ```ruby
48
+ require 'pr_releasenotes'
49
+
50
+ include PrReleasenotes
51
+
52
+ PrReleasenotes.configure do |config|
53
+ config.repo('user/repo')
54
+ config.token('github_token')
55
+ config.parse_args(ARGV)
56
+ end
57
+
58
+ ReleaseNotes.new.run
59
+ ```
60
+
61
+ Get a brief usage summary by running
62
+
63
+ $ pr_releasenotes --help
64
+
65
+
66
+ See the [examples folder](examples) for some sample yaml configuration files.
67
+
68
+ ### Github token permissions
69
+
70
+ This tool does require a github token since it accesses the github api. However any token created should have only as many, or rather as few, permissions as needed.
71
+
72
+ A suitable token can be generated using the Github [Personal Access Tokens](https://github.com/settings/tokens) page. Depending on the use, the following permissions are required on the token:
73
+
74
+ * Print out the release notes for a public repo on stdout:
75
+ * no permissions selected, i.e. public access
76
+ * Update releases on github for a public repo:
77
+ * public_repo scope, i.e. write access to the user's public repos.
78
+ * Print release notes or update releases on private repos:
79
+ * repo scope, i.e. read/write access to all private repos.
80
+
81
+ **Add permissions with caution, and only if necessary!**
82
+
83
+ Unfortunately, github currently does not provide separate write access for releases from write access for code, so these permissions are required for the corresponding use. Although this gem was written to make the minimal necessary use of the token, both the public_repo and repo scope tokens are nearly as powerful as your password, and should be well protected.
84
+
85
+ ## Development
86
+
87
+ After checking out the repo, run `bin/setup` to install dependencies. You can also run `bin/console` for an interactive prompt that will allow you to experiment.
88
+
89
+ To install this gem onto your local machine, run `bundle exec rake install`. To release a new version, update the version number in `version.rb`, and then run `bundle exec rake release`, which will create a git tag for the version, push git commits and tags, and push the `.gem` file to [rubygems.org](https://rubygems.org).
90
+
91
+ ## Contributing
92
+
93
+ Bug reports and pull requests are welcome on GitHub at https://github.com/bazaarvoice/pr_releasenotes.
data/Rakefile ADDED
@@ -0,0 +1,2 @@
1
+ require "bundler/gem_tasks"
2
+ task :default => :spec
data/bin/console ADDED
@@ -0,0 +1,14 @@
1
+ #!/usr/bin/env ruby
2
+
3
+ require "bundler/setup"
4
+ require "pr_releasenotes"
5
+
6
+ # You can add fixtures and/or initialization code here to make experimenting
7
+ # with your gem easier. You can also use a different console, if you like.
8
+
9
+ # (If you use this, don't forget to add pry to your Gemfile!)
10
+ # require "pry"
11
+ # Pry.start
12
+
13
+ require "irb"
14
+ IRB.start(__FILE__)
data/bin/setup ADDED
@@ -0,0 +1,8 @@
1
+ #!/usr/bin/env bash
2
+ set -euo pipefail
3
+ IFS=$'\n\t'
4
+ set -vx
5
+
6
+ bundle install
7
+
8
+ # Do any other automated setup that you need to do here
@@ -0,0 +1,23 @@
1
+ #
2
+ # Sample configuration containing custom release notes, categorization and jira support. Assumes
3
+ # a PR template that looks like custom_template.md
4
+ #
5
+
6
+ # Name of your repo in user/repo format
7
+ repo:
8
+
9
+ # Release notes parsing options. Note that comments will always get stripped
10
+
11
+ # Assumes a separate Release Notes section in the PR description
12
+ relnotes_regex: /.*### Release Notes(.*)/m
13
+ relnotes_group: 1 # group to capture from the regex
14
+
15
+ # Enable categorization
16
+ categorize: true
17
+
18
+ # Update to your own jira server
19
+ jira_baseurl: 'https://issues.apache.org/jira/browse/'
20
+
21
+ # Enable pagination
22
+ auto_paginate: true
23
+
@@ -0,0 +1,15 @@
1
+ <!--
2
+ Custom template that can be used with custom_config.yaml. Save this file under .github/PULL_REQUEST_TEMPLATE.md to ensure all pull requests follow this template.
3
+ -->
4
+
5
+ ### Description
6
+ <!-- Internal changes section. If this PR has only external facing changes, remove this section, including heading. -->
7
+
8
+ ### Release Notes
9
+ <!-- External facing changes section. If this PR has only internal changes, remove this entire Release Notes section, including heading. Otherwise, remove any categories below that are not relevant to these changes. Be sure to maintain the 3# headings above and bullet format below, so that the release notes can get parsed/rolled up properly later. The leading numerical comments on the category names allow sorting the categories, so do not strip them off.-->
10
+ * <!--01-->Manual Deploy Steps
11
+ * <!-- Add note here to provide more detail about the steps needed prior to or during deployment of a version containing this PR's changes. -->
12
+ * <!--02-->Category A
13
+ * <!-- Add note here to provide more detail, or remove the sub-bullets so that only the PR title is included in the release notes under this category. -->
14
+ * <!--03-->Category B
15
+ * <!--04-->Category C
@@ -0,0 +1,71 @@
1
+ #
2
+ # Sample configuration containing default configuration values
3
+ #
4
+
5
+ # Name of your repo in user/repo format. Defaults to nil and must be overridden in the yaml or on the cmd line.
6
+ # repo:
7
+
8
+ # Release notes will be pulled from PRs merged to this branch. Defaults to master.
9
+ # branch: master
10
+
11
+ # Version range from which release notes should be gathered
12
+
13
+ # Extract pull requests starting from this version. Defaults to nil for latest published release. Override if
14
+ # there are no published releases yet.
15
+ # Note: Although this parameter is listed here for completeness, this will likely be specified on the cmd line
16
+ # since its value will vary with each release.
17
+ # start_version:
18
+
19
+ # Defaults to nil for latest commit. nil will also skip the creation of a github release
20
+ # end_version:
21
+
22
+ # If all your tags have a common prefix, add that here, and use just the varying suffix as the
23
+ # start/end version arguments. Defaults to blank.
24
+ # tag_prefix:
25
+
26
+ # Short sha uniqueness threshold. Shas shorter than this may collide with other commits. When searching for
27
+ # PRs related to a set of commits, shorter shas means more of them can be added into each query, allowing for
28
+ # fewer calls to github search.
29
+ # min_sha_size: 7
30
+
31
+ # Whether to include even PRs not matching the relnotes_regex
32
+ # include_all_prs: true
33
+
34
+ # Finish by posting release notes to github
35
+ # github_release: false
36
+
37
+
38
+ # Release notes parsing options. Note that comments will always get stripped
39
+
40
+ # By default, the entire PR description will be used.
41
+ # relnotes_regex: '/.*/m'
42
+
43
+ # group to capture from the above regex
44
+ # relnotes_group: 0
45
+
46
+
47
+ # Set categorization options
48
+
49
+ # Set to true to enable categorization
50
+ # categorize: false
51
+
52
+ # Prefix to use to detect categories in the PR description
53
+ # category_prefix: '^\* '
54
+
55
+ # PRs without categories will get included into this default category
56
+ # category_default: '<!--99-->Other'
57
+
58
+ # Add a prefix to the release notes items
59
+ # relnotes_hdr_prefix: '### '
60
+
61
+
62
+ # Optional jira url to auto link all strings matching a jira ticket pattern
63
+ # Default to nil to skip auto link
64
+ # jira_baseurl:
65
+
66
+
67
+ # !!!CAUTION!!! this will auto paginate github api calls, and potentially overrun your rate limits! Do not try
68
+ # to get release notes for all history for your project. Limit to no more than 2-3 pages worth of commits per
69
+ # the UI.
70
+ # auto_paginate: false
71
+
@@ -0,0 +1,11 @@
1
+ #!/usr/bin/env ruby
2
+
3
+ require 'pr_releasenotes'
4
+
5
+ include PrReleasenotes
6
+
7
+ PrReleasenotes.configure do |config|
8
+ config.parse_args(ARGV)
9
+ end
10
+
11
+ ReleaseNotes.new.run
@@ -0,0 +1,144 @@
1
+ module PrReleasenotes
2
+
3
+ class << self
4
+ attr_accessor :config
5
+ end
6
+
7
+ def self.configuration
8
+ self.config ||= Configuration.new
9
+ end
10
+
11
+ def self.configure
12
+ configuration
13
+ yield(config)
14
+ end
15
+
16
+ class Configuration
17
+
18
+ require 'optparse'
19
+ require 'yaml'
20
+ require 'logger'
21
+ require 'to_regexp'
22
+ require "pr_releasenotes/version"
23
+
24
+ attr_accessor :repo, :token, :tag_prefix, :min_sha_size, :start_version, :end_version,
25
+ :branch, :include_all_prs, :github_release, :relnotes_group,
26
+ :categorize, :category_prefix, :category_default, :relnotes_hdr_prefix,
27
+ :jira_baseurl, :auto_paginate, :log
28
+
29
+ attr_reader :relnotes_regex
30
+
31
+
32
+ def initialize
33
+ @log = Logger.new(STDOUT)
34
+ @log.level = Logger::INFO
35
+
36
+ # Name of your repo in user/repo format
37
+ @repo = nil
38
+ # If all your tags have a common prefix, add that here, and use just the varying suffix as the argument
39
+ @tag_prefix = ''
40
+ # Short sha uniqueness threshold. Shas shorter than this may collide with other commits
41
+ @min_sha_size = 7
42
+ # Version range from which release notes should be gathered
43
+ @start_version = nil # nil for latest released version
44
+ @end_version = nil # nil for latest commit. nil will skip the creation of a github release
45
+ # Release notes will be pulled from PRs merged to this branch
46
+ @branch = 'master'
47
+ # Whether to include even PRs without explicit release notes
48
+ @include_all_prs = true
49
+ # Finish by posting release notes to github
50
+ @github_release = false
51
+
52
+ # Release notes parsing options. Note that comments will always get stripped
53
+ # Use .* with group 0 to use the entire PR description as the notes
54
+ @relnotes_regex = /.*/m
55
+ @relnotes_group = 0 # group to capture from the regex
56
+
57
+ # Set categorization options
58
+ @categorize = false # false to disable categorization
59
+ @category_prefix = '^\* '
60
+ # PRs without categories will get included into this default category
61
+ @category_default = '<!--99-->Other'
62
+ # Add a prefix to the release notes items
63
+ @relnotes_hdr_prefix = '### '
64
+
65
+ # Optional jira url to auto link all strings matching a jira ticket pattern
66
+ # Set to nil to skip auto link
67
+ @jira_baseurl = nil
68
+
69
+ # !!!CAUTION!!! this will auto paginate, and potentially overrun your rate limits! Do not try to get
70
+ # release notes for all history for your project. Limit to no more than 2-3 pages worth of commits per
71
+ # the UI.
72
+ @auto_paginate = false
73
+ end
74
+
75
+ def dump
76
+ instance_variables.map do |var|
77
+ "#{var}: #{instance_variable_get(var)}"
78
+ end
79
+ end
80
+
81
+ def relnotes_regex= (regex_str)
82
+ @relnotes_regex = regex_str.to_regexp
83
+ end
84
+
85
+ def parse_args (args)
86
+ opt_parser = OptionParser.new do |opts|
87
+ opts.banner = "Usage: pr_releasenotes [options]"
88
+
89
+ opts.separator ''
90
+ opts.on('-c', '--config <config.yaml>', 'Yaml configuration') do |config|
91
+ YAML.load_file(config).each do |k, v|
92
+ begin
93
+ # Try calling public setter for this option
94
+ public_send("#{k}=", v)
95
+ rescue NoMethodError => e
96
+ raise "#{e.message} caused by (#{config}) Invalid key '#{k}'"
97
+ end
98
+ end
99
+ end
100
+
101
+ opts.separator ''
102
+ opts.on('-r', '--repo <user/repo>', 'Repo to scan for pull requests') do |repo|
103
+ @repo = repo
104
+ end
105
+ opts.on('-t', '--token <token>', 'Github token with repo scope') do |token|
106
+ @token = token
107
+ end
108
+
109
+ opts.separator ''
110
+ opts.on('-s', '--start <version>', 'Get release notes from this version tag. (Default: latest release)') do |version|
111
+ @start_version = version
112
+ end
113
+ opts.on('-e', '--end <version>', 'Get release notes till this version tag. (Default: latest commit, skips creating github release)') do |version|
114
+ @end_version = version
115
+ end
116
+ opts.on('-b', '--branch <branchname>', 'Use pull requests to this branch. (Default: master)') do |branch|
117
+ @branch = branch
118
+ end
119
+
120
+ opts.separator ''
121
+ opts.on('-p', '--post-to-github', 'Create/update release on github') do
122
+ @github_release = true
123
+ end
124
+
125
+ opts.separator ''
126
+ opts.on('-d', '--debug', 'Enable debug logging') do
127
+ @log.level = Logger::DEBUG
128
+ @log.debug 'Debug mode enabled'
129
+ end
130
+ opts.on_tail('-h', '--help', 'Prints this help') do
131
+ puts opts
132
+ exit
133
+ end
134
+ opts.on_tail('-v', '--version', 'Show version') do
135
+ puts ::VERSION
136
+ exit
137
+ end
138
+ end
139
+
140
+ opt_parser.parse!(args)
141
+ end
142
+
143
+ end
144
+ end
@@ -0,0 +1,3 @@
1
+ module PrReleasenotes
2
+ VERSION = "0.0.1"
3
+ end
@@ -0,0 +1,220 @@
1
+ require "pr_releasenotes/configuration"
2
+
3
+ # Generate release notes from github pull requests, and optionally
4
+ # post to github releases
5
+ module PrReleasenotes
6
+
7
+ class ReleaseNotes
8
+
9
+ require 'octokit'
10
+
11
+ attr_reader :config, :git_client, :log
12
+
13
+ def initialize
14
+
15
+ @config = PrReleasenotes.configuration
16
+ @log = config.log
17
+
18
+ raise "No repo defined! Run with -h for usage." if config.repo.nil?
19
+ raise "Missing github token! Run with -h for usage." if config.token.nil?
20
+
21
+ Octokit.auto_paginate = config.auto_paginate
22
+ @git_client ||= Octokit::Client.new(:access_token => config.token)
23
+ log.info 'Created github client.'
24
+
25
+ end
26
+
27
+ def run
28
+ # Infer start version if necessary
29
+ set_start_version
30
+ log.info "Retrieving release notes between #{config.start_version} and #{config.end_version.nil? ? 'now' : config.end_version} on branch #{config.branch}"
31
+ # Get commits between versions
32
+ commits_for_versions = get_commits_for_versions
33
+ # Get merged PRs for those commits
34
+ prs = get_prs_for_commits(commits_for_versions)
35
+ # Get release notes from those PRs
36
+ notes_by_pr = get_releasenotes_for_prs(prs)
37
+ # Convert to a single string
38
+ notes_str = get_notes_str(notes_by_pr)
39
+ log.info "Release Notes:\n\n#{notes_str}"
40
+ # Optionally post to github as a new or updated release
41
+ post_to_github(notes_str) if config.github_release
42
+ end
43
+
44
+ def set_start_version
45
+ if config.start_version.nil?
46
+ # If start version isn't set, try to infer from the latest github release
47
+ begin
48
+ releases = git_client.releases(config.repo)
49
+ .sort_by { |release| release[:created_at]}.reverse # order by create date, newest first
50
+ release = releases.find { |release|
51
+ unless release[:draft]
52
+ # is a pre-release or published release, so check if this release is an ancestor of the current branch
53
+ # "diverged" indicates it was on a different branch & "ahead" indicates it's after the
54
+ # specified end_version, so neither can be used as a start_version
55
+ # "behind" indicates it's an ancestor and can be used as a start_version,
56
+ git_client.compare(config.repo, config.branch, release[:tag_name])[:status] == 'behind'
57
+ end
58
+ }
59
+ config.start_version = release[:tag_name].sub /#{config.tag_prefix}/, ''
60
+ rescue StandardError
61
+ log.error "No published releases found in #{config.repo} for branch #{config.branch}. Specify a start version explicitly."
62
+ exit 1
63
+ end
64
+ end
65
+ end
66
+
67
+ def get_date_for_version(tags, version)
68
+ # From the tags, find the tag for the specified version and get its sha
69
+ tagname = "#{config.tag_prefix}#{version}"
70
+ tag = tags.select {|tag| tag.name == tagname }.shift
71
+ tag.nil? and log.error "No commit found with tag name #{tagname}\n" and exit 1
72
+ tag_sha = tag[:commit][:sha]
73
+
74
+ # get the commit for that sha, and the date for that commit.
75
+ commit = git_client.commit( config.repo, tag_sha )
76
+ commit[:commit][:author][:date]
77
+ end
78
+
79
+ def get_commits_for_versions
80
+
81
+ # The github api does not directly support getting a list of commits since a tag was applied.
82
+ # To work around this, we instead:
83
+
84
+ # get a list of tags
85
+ tags = git_client.tags(config.repo)
86
+ # get the corresponding date for the tag corresponding to the start version
87
+ start_date = get_date_for_version tags, config.start_version
88
+ if config.end_version.nil?
89
+ # and get a list of commits since the start_date on the configured branch
90
+ commits = git_client.commits_since( config.repo, start_date, config.branch)
91
+ log.info "Got #{commits.length} commits on #{config.branch} between #{config.start_version}(#{start_date}) and now"
92
+ else
93
+ # and for the end version if not nil
94
+ end_date = get_date_for_version tags, config.end_version
95
+ # and get a list of commits between the start/end dates on the configured branch
96
+ commits = git_client.commits_between( config.repo, start_date, end_date, config.branch)
97
+ log.info "Got #{commits.length} commits on #{config.branch} between #{config.start_version}(#{start_date}) and #{config.end_version}(#{end_date})"
98
+ end
99
+ log.debug "Commits: " + commits.map(&:sha).join(',')
100
+ commits
101
+ end
102
+
103
+ def get_prs_for_commits(commits)
104
+ # To optimize the number of calls to github, we'll combine commits into fewer search
105
+ # calls, taking care not to exceed maximum query size of 256 chars
106
+
107
+ # Searches need to restrict to merged pull requests in the specified repo
108
+ search_suffix = " is:merged repo:#{config.repo}"
109
+ # Max commits that can fit into a single query, including ',' join delimiter
110
+ max_commits_per_search = 256/(config.min_sha_size + 1)
111
+
112
+ shas = commits.map { |commit| commit[:sha][0..(config.min_sha_size-1)] }
113
+ prs = shas.each_slice(max_commits_per_search).reduce([]) { |prs, sha_slice|
114
+ # We slice the list of commits into sublists that'll fit into the github query size limits
115
+ query = sha_slice.join(',') + search_suffix
116
+ results = git_client.search_issues( query )
117
+ log.info "Query '#{query}' matched #{results[:total_count]} PRs"
118
+ # the reduce method allows us to accummulate PRs, skipping any
119
+ prs.concat(results[:items])
120
+ }
121
+ # Dedupe prs
122
+ prs.uniq! { |pr| pr[:number] }
123
+ log.info "Retrieved #{prs.size} unique PRs"
124
+ prs.each {|pr| log.debug pr[:title]}
125
+ prs
126
+ end
127
+
128
+ def get_releasenotes_for_prs(prs)
129
+ notes = prs.reduce([]) { |notes, pr|
130
+ # Get release notes section
131
+ body = pr[:body].strip.slice(config.relnotes_regex, config.relnotes_group)
132
+ unless body.nil?
133
+ # Release notes exist for this PR, so do some cleanup
134
+ body.gsub! /^<!--.+?(?=-->)-->/m, '' # strip off (multiline) comments
135
+ body.gsub! /^\r?\n/, '' # and empty lines
136
+ end
137
+ # For PRs without release notes, add just the title
138
+ notes << {:title => pr[:title], :date => pr[:closed_at], :prnum => pr[:number], :body => body}
139
+ }
140
+ notes.sort_by! {|note| note[:date]}.reverse! # order by PR merge date, newest first
141
+ end
142
+
143
+ def jiraize(notes_str)
144
+ unless config.jira_baseurl.nil?
145
+ # jira url is defined, so linkify any jira tickets in the notes
146
+ # Regex: negative lookbehind(?<!...) for jira_prefix or beginning of link markdown,
147
+ # followed by jira ticket pattern, followed by negative lookahead(?!...) for end of
148
+ # link markdown
149
+ # | neg lookbehind | | tkt pattern | |neg lookahead|
150
+ notes_str.gsub!(/(?<!#{config.jira_baseurl}|\[)\b([A-Z]{2,}-\d+)\b(?![^\[\]]*\]\()/, "[\\1](#{config.jira_baseurl}\\1)")
151
+ end
152
+ notes_str
153
+ end
154
+
155
+ def get_notes_str(notes_by_pr)
156
+ notes_str = "Changes since #{config.tag_prefix}#{config.start_version}:\r\n"
157
+ if config.categorize
158
+ # Categorize the notes, using the regex as the category names
159
+ notes = notes_by_pr.reduce({}) { | notes, pr_note |
160
+ if pr_note[:body].nil?
161
+ # No explicit release notes
162
+ if config.include_all_prs
163
+ # Still want to include PR title under default category
164
+ note = "* #{pr_note[:title]} ([##{pr_note[:prnum]}](https://github.com/#{config.repo}/pull/#{pr_note[:prnum]} \"Merged #{pr_note[:date]}\"))\r\n"
165
+ # Initialize collapsible notes for category if it doesn't exist, and append note
166
+ (notes[config.category_default] ||= "<details><summary>Show/Hide</summary><p>\r\n\r\n") << note
167
+ end
168
+ else
169
+ # For each PR's notes, split by the category prefix
170
+ # The lookahead (?=) regex allows including the category prefix delimiter at the beginning of each split record
171
+ pr_note[:body].split(/(?=#{config.category_prefix})/).each { |pr_category_note|
172
+ # Assume all of the line after the category_prefix is the category name. Slice off the category prefix and extract the category name
173
+ category = pr_category_note.slice!(/#{config.category_prefix}[^\r]*/)
174
+ # Remove trailing newlines
175
+ pr_category_note.sub!(/^(\r\n)+|(\r\n)+$/, '')
176
+ # Generate section header using PR title
177
+ note = "* #{pr_note[:title]} ([##{pr_note[:prnum]}](https://github.com/#{config.repo}/pull/#{pr_note[:prnum]} \"Merged #{pr_note[:date]}\"))\r\n#{pr_category_note}\r\n"
178
+ # Use default category if no category found
179
+ category = category.nil? ? config.category_default : category.sub(/#{config.category_prefix}(.*)/,'\1')
180
+ # Initialize notes for category if it doesn't exist, and append note
181
+ (notes[category] ||= '') << note
182
+ }
183
+ end
184
+ notes
185
+ }
186
+ notes[config.category_default] << "</p></details>" unless notes[config.category_default].nil? # Close the collapsible section
187
+ # Accumulate each category's notes
188
+ notes_str << notes.sort.map { |category, note|
189
+ note_str = category.nil? ? "#{config.relnotes_hdr_prefix}#{config.category_default}\r\n" : "#{config.relnotes_hdr_prefix}#{category}\r\n"
190
+ note_str << note
191
+ }.join("\r\n")
192
+ else
193
+ # No categorization required, use notes as is
194
+ notes_str << notes_by_pr.reduce("") { | str, note |
195
+ str << "#{config.relnotes_hdr_prefix}#{note[:title]} [##{note[:prnum]}](https://github.com/#{config.repo}/pull/#{note[:prnum]} \"Merged #{note[:date]}\")\r\n#{note[:body]}\r\n"
196
+ }
197
+ end
198
+ jiraize notes_str
199
+ end
200
+
201
+ def post_to_github(notes_str)
202
+ unless config.end_version.nil?
203
+ tag_name = "#{config.tag_prefix}#{config.end_version}"
204
+ begin
205
+ release = git_client.release_for_tag(config.repo, tag_name)
206
+ # Found existing release, so update it
207
+ log.info "Found existing #{release.draft? ? 'draft ' : ''}#{release.prerelease? ? 'pre-' : ''}release with tag #{tag_name} at #{release.html_url} with body: #{release.body}"
208
+ release = git_client.update_release(release.url, { :name => tag_name, :body => notes_str, :draft => false, :prerelease => true, :target_commitish => config.branch })
209
+ log.info "Updated pre-release #{release.id} at #{release.html_url} with body #{notes_str}"
210
+ rescue Octokit::NotFound
211
+ # no existing release, so create a new one
212
+ release = git_client.create_release(config.repo, tag_name, { :name => tag_name, :body => notes_str, :draft => false, :prerelease => true, :target_commitish => config.branch })
213
+ log.info "Created pre-release #{release.id} at #{release.html_url} with body #{notes_str}"
214
+ end
215
+ end
216
+ end
217
+
218
+ end
219
+
220
+ end
@@ -0,0 +1,40 @@
1
+ # coding: utf-8
2
+ lib = File.expand_path('../lib', __FILE__)
3
+ $LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
4
+ require 'pr_releasenotes/version'
5
+
6
+ Gem::Specification.new do |spec|
7
+ spec.name = 'pr_releasenotes'
8
+ spec.version = PrReleasenotes::VERSION
9
+ spec.version = "#{spec.version}.#{ENV['TRAVIS_BUILD_NUMBER']}" \
10
+ if ENV['TRAVIS'] && PrReleasenotes::VERSION.include?('.pre')
11
+ spec.authors = ['arvindth']
12
+ spec.email = ['arvind.thirunarayanan@bazaarvoice.com']
13
+
14
+ spec.summary = %q{Post pull request based release notes to github releases}
15
+ spec.description = %q{Generate release notes from pull request descriptions, and optionally post them to the github releases page}
16
+ spec.homepage = 'https://github.com/bazaarvoice/pr_releasenotes'
17
+
18
+ # Prevent pushing this gem to RubyGems.org. To allow pushes either set the 'allowed_push_host'
19
+ # to allow pushing to a single host or delete this section to allow pushing to any host.
20
+ if spec.respond_to?(:metadata)
21
+ spec.metadata['allowed_push_host'] = "https://rubygems.org"
22
+ else
23
+ raise 'RubyGems 2.0 or newer is required to protect against ' \
24
+ 'public gem pushes.'
25
+ end
26
+
27
+ # Specify which files should be added to the gem when it is released.
28
+ # The `git ls-files -z` loads the files in the RubyGem that have been added into git.
29
+ spec.files = Dir.chdir(File.expand_path('..', __FILE__)) do
30
+ `git ls-files -z`.split("\x0").reject { |f| f.match(%r{^(test|spec|features)/}) } end
31
+ spec.bindir = 'exe'
32
+ spec.executables = spec.files.grep(%r{^exe/}) { |f| File.basename(f) }
33
+ spec.require_paths = ['lib']
34
+
35
+ spec.add_development_dependency 'bundler', '~> 1.16'
36
+ spec.add_development_dependency 'rake', '~> 10.0'
37
+
38
+ spec.add_dependency 'octokit', '~> 4.14'
39
+ spec.add_dependency 'to_regexp', '~> 0.2.1'
40
+ end
metadata ADDED
@@ -0,0 +1,119 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: pr_releasenotes
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.0.1
5
+ platform: ruby
6
+ authors:
7
+ - arvindth
8
+ autorequire:
9
+ bindir: exe
10
+ cert_chain: []
11
+ date: 2019-04-28 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: bundler
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - "~>"
18
+ - !ruby/object:Gem::Version
19
+ version: '1.16'
20
+ type: :development
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - "~>"
25
+ - !ruby/object:Gem::Version
26
+ version: '1.16'
27
+ - !ruby/object:Gem::Dependency
28
+ name: rake
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - "~>"
32
+ - !ruby/object:Gem::Version
33
+ version: '10.0'
34
+ type: :development
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - "~>"
39
+ - !ruby/object:Gem::Version
40
+ version: '10.0'
41
+ - !ruby/object:Gem::Dependency
42
+ name: octokit
43
+ requirement: !ruby/object:Gem::Requirement
44
+ requirements:
45
+ - - "~>"
46
+ - !ruby/object:Gem::Version
47
+ version: '4.14'
48
+ type: :runtime
49
+ prerelease: false
50
+ version_requirements: !ruby/object:Gem::Requirement
51
+ requirements:
52
+ - - "~>"
53
+ - !ruby/object:Gem::Version
54
+ version: '4.14'
55
+ - !ruby/object:Gem::Dependency
56
+ name: to_regexp
57
+ requirement: !ruby/object:Gem::Requirement
58
+ requirements:
59
+ - - "~>"
60
+ - !ruby/object:Gem::Version
61
+ version: 0.2.1
62
+ type: :runtime
63
+ prerelease: false
64
+ version_requirements: !ruby/object:Gem::Requirement
65
+ requirements:
66
+ - - "~>"
67
+ - !ruby/object:Gem::Version
68
+ version: 0.2.1
69
+ description: Generate release notes from pull request descriptions, and optionally
70
+ post them to the github releases page
71
+ email:
72
+ - arvind.thirunarayanan@bazaarvoice.com
73
+ executables:
74
+ - pr_releasenotes
75
+ extensions: []
76
+ extra_rdoc_files: []
77
+ files:
78
+ - ".gitignore"
79
+ - ".travis.yml"
80
+ - Gemfile
81
+ - LICENSE
82
+ - OWNERS
83
+ - README.md
84
+ - Rakefile
85
+ - bin/console
86
+ - bin/setup
87
+ - examples/custom_config.yaml
88
+ - examples/custom_template.md
89
+ - examples/default_config.yaml
90
+ - exe/pr_releasenotes
91
+ - lib/pr_releasenotes.rb
92
+ - lib/pr_releasenotes/configuration.rb
93
+ - lib/pr_releasenotes/version.rb
94
+ - pr_releasenotes.gemspec
95
+ homepage: https://github.com/bazaarvoice/pr_releasenotes
96
+ licenses: []
97
+ metadata:
98
+ allowed_push_host: https://rubygems.org
99
+ post_install_message:
100
+ rdoc_options: []
101
+ require_paths:
102
+ - lib
103
+ required_ruby_version: !ruby/object:Gem::Requirement
104
+ requirements:
105
+ - - ">="
106
+ - !ruby/object:Gem::Version
107
+ version: '0'
108
+ required_rubygems_version: !ruby/object:Gem::Requirement
109
+ requirements:
110
+ - - ">="
111
+ - !ruby/object:Gem::Version
112
+ version: '0'
113
+ requirements: []
114
+ rubyforge_project:
115
+ rubygems_version: 2.7.7
116
+ signing_key:
117
+ specification_version: 4
118
+ summary: Post pull request based release notes to github releases
119
+ test_files: []