-
Notifications
You must be signed in to change notification settings - Fork 38
/
Copy pathRakefile
262 lines (199 loc) · 7.52 KB
/
Rakefile
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
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
require 'rspec/core/rake_task'
require 'chef/cookbook/metadata'
require 'yaml'
environment = ENV['ENVIRONMENT'] || ''
# this module groups useful general commands together
module GeneralCommands
@logging = ENV['LOGGING'] || true
def self.run(cmd, *expected_exitstatuses)
puts "+ #{cmd}" if @logging
output = `#{cmd} 2>&1`
puts output.gsub(/^/, '- ') if @logging
expected_exitstatuses << 0 if expected_exitstatuses.empty?
error_message = "ERROR: '#{cmd}' failed with exit status #{$CHILD_STATUS.exitstatus}"
raise StandardError, error_message unless [expected_exitstatuses].flatten.include?($CHILD_STATUS.exitstatus)
output
end
def self.parse_metadata(metadata_file = 'metadata.rb')
metadata = Chef::Cookbook::Metadata.new
metadata.from_file(metadata_file)
metadata
end
end
# this module groups useful release commands together
module ReleaseCommands
def self.ensure_working_directory_state(allowed_modified_files = [])
modified_files = GeneralCommands.run('git diff --name-only', 0, 1)
modified_files = modified_files.split(/\n+/)
return if allowed_modified_files.uniq.sort == modified_files.uniq.sort
raise '[RELEASE] Working directory is in an unexpected state'
end
def self.verify_tag(should_exist = false)
return if GeneralCommands.run("git rev-parse -q --verify \"refs/tags/#{version}\"", 0, 1).empty? != should_exist
suffix = if should_exist == true
'does not exist'
else
'already exists'
end
raise "[RELEASE] Tag (#{version}) #{suffix}"
end
def self.version
GeneralCommands.parse_metadata.version
end
def self.name
GeneralCommands.parse_metadata.name
end
def self.category
file_name = '.category'
if File.exist?(file_name)
File.read(file_name).strip
else
''
end
end
def self.pre_publish
GitCommands.ensure_branch('master')
ReleaseCommands.ensure_working_directory_state
end
def self.publish_to_scm
ReleaseCommands.pre_publish
ReleaseCommands.verify_tag(false)
puts 'Creating an annotated tag...'
GitCommands.tag(ReleaseCommands.version)
puts 'Push tag to remote repository...'
GitCommands.push_tags('origin', 'master', ReleaseCommands.version)
end
def self.publish_to_chef_server
ReleaseCommands.pre_publish
ReleaseCommands.verify_tag(true)
puts 'Updating Berkshelf...'
BerkshelfCommands.update
puts 'Packaging cookbooks (creating a single archive containing all of your required cookbooks)'
BerkshelfCommands.package('cookbooks.tar.gz')
puts 'Installing packaged cookbooks into Chef Server'
BerkflowCommands.install('cookbooks.tar.gz')
end
def self.publish_to_chef_supermarket
ReleaseCommands.pre_publish
ReleaseCommands.verify_tag(true)
cookbook_name = ReleaseCommands.name
cookbook_category = ReleaseCommands.category
if cookbook_name.blank? || cookbook_category.blank?
puts 'Skipped sharing to the Chef Supermarket - cookbook name and/or category not set. Set it and run \'publish:chef:supermarket\' separately if needed.'
else
puts 'Sharing to the Chef Supermarket...'
KnifeCommands.share(cookbook_name, cookbook_category)
end
end
def self.release(environment = '')
raise 'You must specify an environment name.' if environment.blank?
ReleaseCommands.ensure_working_directory_state
ReleaseCommands.verify_tag(true)
puts "Applying the cookbook version locks to the '#{environment}'-Chef environment"
BerkshelfCommands.apply(environment)
end
end
# this module groups useful berkshelf commands together
module BerkshelfCommands
def self.update
return if GeneralCommands.run('chef exec berks update', 0, 1)
raise '[BERKSHELF] Failed to update'
end
def self.package(archive_filename = 'cookbooks.tar.gz')
raise '[BERKSHELF] You must specify an archive filename.' if archive_filename.blank?
return if GeneralCommands.run("chef exec berks package #{archive_filename}", 0, 1)
raise "[BERKSHELF] Failed to create #{archive_filename}-archive"
end
def self.apply(environment = '')
raise '[BERKSHELF] You must specify an environment.' if environment.blank?
return unless GeneralCommands.run("chef exec berks apply #{environment}", 0, 1).empty?
raise "[BERKSHELF] Failed to apply the cookbook version locks to the '#{environment}'-Chef environment"
end
end
# this module groups useful berkflow commands together
module BerkflowCommands
def self.install(archive_filename = 'cookbooks.tar.gz')
raise '[BERKFLOW] You must specify an archive filename.' if archive_filename.blank?
return if /Done./ =~ GeneralCommands.run("chef exec blo in #{archive_filename}", 0, 1)
raise "[BERKFLOW] Failed to install packaged cookbooks #{archive_filename} into Chef Server"
end
end
# this module groups useful git commands together
module GitCommands
def self.ensure_branch(branch = 'master')
raise '[GIT] You must specify a branch.' if branch.blank?
return if /#{branch}/ =~ GeneralCommands.run('git rev-parse --abbrev-ref HEAD', 0, 1)
raise '[GIT] Currently working on unexpected branch'
end
def self.tag(tag = '')
raise '[GIT] You must specify a tag.' if tag.blank?
return if GeneralCommands.run("git tag -a #{tag} -m '#{tag}'", 0, 1)
raise '[GIT] Failed to tag'
end
def self.push_tags(remote = 'origin', branch = 'master', tag = '')
raise '[GIT] You must specify a remote.' if remote.blank?
raise '[GIT] You must specify a branch.' if branch.blank?
raise '[GIT] You must specify a tag.' if tag.blank?
return if GeneralCommands.run("git push #{remote} #{branch} --tags", 0, 1)
raise '[GIT] Failed to push to remote'
end
end
# this module groups useful knife commands together
module KnifeCommands
def self.share(cookbook_name = '', cookbook_category = '')
raise '[KNIFE] Missing cookbook name.' if cookbook_name.blank?
raise '[KNIFE] Missing cookbook category.' if cookbook_category.blank?
return if /Upload complete/ =~ GeneralCommands.run("chef exec knife supermarket share '#{cookbook_name}' '#{cookbook_category}' --cookbook-path ../", 0, 1)
raise "[KNIFE] Failed to publish the #{cookbook_name}-cookbook on the Chef Supermarket"
end
end
require 'cookstyle'
require 'rubocop/rake_task'
desc 'Run Linter (cookstyle) tests'
RuboCop::RakeTask.new(:style) do |task|
task.options << '--display-cop-names'
end
desc 'Run ChefSpec tests'
RSpec::Core::RakeTask.new(:spec)
namespace :integration do
require 'kitchen'
desc 'Run integration tests (Test Kitchen & Vagrant)'
task :vagrant do
Kitchen.logger = Kitchen.default_file_logger
Kitchen::Config.new.instances.each do |instance|
instance.test(:always)
end
end
end
namespace :publish do
desc 'Publish to SCM'
task :scm do
ReleaseCommands.publish_to_scm
end
namespace :chef do
desc 'Publish to Chef Server'
task :server do
ReleaseCommands.publish_to_chef_server
end
desc 'Publish to Chef Supermarket'
task :supermarket do
ReleaseCommands.publish_to_chef_supermarket
end
end
task all: ['scm', 'chef:supermarket', 'chef:server']
end
desc 'Run lint checks'
task lint: %w(style)
desc 'Run unit tests'
task unit: %w(spec)
desc 'Run Travis CI tests'
task travis: %w(lint unit)
desc 'Run all integration tests'
task integration: %w(integration:vagrant)
desc 'Publish'
task publish: %w(publish:scm publish:chef:supermarket publish:chef:server)
desc 'Release'
task :release do
ReleaseCommands.release(environment)
end
task default: %w(lint unit integration)