-
Notifications
You must be signed in to change notification settings - Fork 221
/
Copy pathgenerate_changelog.rb
executable file
·152 lines (120 loc) · 3.65 KB
/
generate_changelog.rb
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
#!/usr/bin/env ruby
# frozen_string_literal: true
require 'octokit'
class ChangelogGenerator
attr_reader :version, :entries
def initialize(version = nil)
@version = version
@entries = {
'feature' => { name: 'New features', entries: [] },
'bug' => { name: 'Bug fixes', entries: [] },
'deprecation' => { name: 'Deprecations', entries: [] },
'removal' => { name: 'Removals', entries: [] }
}
# Setting the changelog path early lets us check that it exists
# before we spend time making API calls
changelog
end
def labels
@entries.keys
end
def client
unless @client
unless ENV['GITHUB_TOKEN']
warn "Missing GitHub personal access token. Set $GITHUB_TOKEN with a "\
"personal access token to use this script."
exit 1
end
Octokit.configure do |c|
c.auto_paginate = true
end
@client = Octokit::Client.new(access_token: ENV['GITHUB_TOKEN'])
end
@client
end
def latest
@latest ||= client.tags('puppetlabs/bolt').first.name
end
def commits
@commits ||= client.compare('puppetlabs/bolt', latest, 'main').commits
end
def org_members
@org_members ||= Set.new(client.org_members('puppetlabs').map(&:login))
end
def changelog
unless @changelog
@changelog = File.expand_path('CHANGELOG.md', Dir.pwd)
unless File.file?(@changelog)
warn "Unable to find changelog at #{@changelog}"
exit 1
end
end
@changelog
end
# Parses individual commits by scanning the commit message for valid release notes
# and adding them to the list of entries. Entries include extra information about
# the author and whether it was an internal or external contribution so we can give
# kudos.
def parse_commit(commit)
regex = /!(#{labels.join('|')})(.+?)(?=(?:!(?:#{labels.join('|')})|\z))/m
commit.commit.message.scan(regex) do |match|
label, entry = match
entries[label][:entries].push(
body: entry,
author: commit.commit.author.name || commit.author.login,
profile: commit.author.html_url,
internal: org_members.include?(commit.author.login)
)
end
end
def compute_version
x, y, z = latest.split('.').map(&:to_i)
if %w[feature deprecation removal].any? { |type| entries[type][:entries].any? }
y += 1
z = 0
else
z += 1
end
[x, y, z].join('.')
end
def update_changelog
old_lines = File.read(changelog).split("\n")[2..-1]
new_lines = [
"# Changelog\n",
"## Bolt #{version} (#{Time.now.strftime '%Y-%m-%d'})\n"
]
entries.each_value do |type|
next unless type[:entries].any?
new_lines << "### #{type[:name]}\n"
type[:entries].each do |entry|
new_lines << "#{entry[:body].strip}\n"
unless entry[:internal]
new_lines << " _Contributed by [#{entry[:author]}](#{entry[:profile]})_\n"
end
end
end
content = (new_lines + old_lines).join("\n")
if File.write(changelog, content)
puts "Successfully wrote entries to #{changelog}"
else
warn "Unable to write entries to #{changelog}"
exit 1
end
end
def generate
puts "Loading and parsing commits for #{latest}..main"
commits.each do |commit|
parse_commit(commit)
end
if entries.each_value.all? { |type| type[:entries].empty? }
warn "No release notes for #{latest}..main"
exit 0
end
@version ||= compute_version
puts "The next release will be version #{version}"
update_changelog
end
end
if $PROGRAM_NAME == __FILE__
ChangelogGenerator.new(ARGV.first).generate
end