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 +7 -0
- data/.gitignore +11 -0
- data/.travis.yml +9 -0
- data/Gemfile +8 -0
- data/LICENSE +201 -0
- data/OWNERS +2 -0
- data/README.md +93 -0
- data/Rakefile +2 -0
- data/bin/console +14 -0
- data/bin/setup +8 -0
- data/examples/custom_config.yaml +23 -0
- data/examples/custom_template.md +15 -0
- data/examples/default_config.yaml +71 -0
- data/exe/pr_releasenotes +11 -0
- data/lib/pr_releasenotes/configuration.rb +144 -0
- data/lib/pr_releasenotes/version.rb +3 -0
- data/lib/pr_releasenotes.rb +220 -0
- data/pr_releasenotes.gemspec +40 -0
- metadata +119 -0
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
data/.travis.yml
ADDED
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
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
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,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
|
+
|
data/exe/pr_releasenotes
ADDED
@@ -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,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: []
|