forked from puppetlabs/puppet-docs
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathRakefile
351 lines (293 loc) · 12.6 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
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
# encoding: UTF-8
require 'rubygems'
require 'bundler/setup'
require 'rake'
require 'pathname'
require 'fileutils'
require 'yaml'
require 'puppet_docs/config'
begin
require "vlad"
Vlad.load :scm => :git
rescue LoadError
# do nothing
end
top_dir = Dir.pwd
SOURCE_DIR = "#{top_dir}/source"
OUTPUT_DIR = "#{top_dir}/output"
STASH_DIR = "#{top_dir}/_stash"
PREVIEW_DIR = "#{top_dir}/_preview"
VERSION_FILE = "#{OUTPUT_DIR}/VERSION.txt"
@config_data = PuppetDocs::Config.new("#{SOURCE_DIR}/_config.yml")
def jekyll(command = 'build', source = SOURCE_DIR, destination = OUTPUT_DIR, *args)
amended_config = "#{SOURCE_DIR}/_config_amended.yml"
File.write(amended_config, YAML.dump(@config_data))
system("bundle exec jekyll #{command} --source #{source} --destination #{destination} #{args.join(' ')} --config #{amended_config}")
FileUtils.rm(amended_config)
end
desc "Stash all directories but one in a temporary location. Run a preview server on localhost:4000."
task :preview, :filename do |t, args|
if ["marionette-collective", "puppetdb_master", "puppetdb_1.1", "puppetdb", "mcollective"].include?(args.filename)
abort("\n\n*** External documentation sources aren't supported right now.\n\n")
end
# Make sure we have a stash_directory
FileUtils.mkdir(STASH_DIR) unless File.exist?(STASH_DIR)
# Directories and files we have to have for a good live preview
required_dirs = ["_config.yml", "_includes", "_plugins", "files", "favicon.ico", "_layouts","images"]
# Move the things we don't need into the _stash
Dir.glob("#{SOURCE_DIR}/*") do |directory|
FileUtils.mv directory, STASH_DIR unless directory.include?(args.filename) || required_dirs.include?(File.basename(directory))
end
# Get all the files we'd like to see in a temporary preview index (so we don't have to hunt for files by name)
Dir.chdir("#{SOURCE_DIR}/#{args.filename}")
file_list = Dir.glob("**/*.markdown")
preview_index_files = []
file_list.each do |f|
html_name = f.gsub(/\.markdown/,'.html')
preview_index_files << "* [#{args.filename}/#{html_name}](#{args.filename}/#{html_name})\n"
end
preview_index=<<PREVIEW_INDEX
---
layout: frontpage
title: Files Available for Live Preview
canonical: "/"
---
#{preview_index_files}
PREVIEW_INDEX
Dir.chdir(SOURCE_DIR)
# put our file list index in place
File.open("index.markdown", 'w') {|f| f.write(preview_index) }
# Run our preview server, watching ... watching ...
jekyll('serve', SOURCE_DIR, PREVIEW_DIR)
# When we kill it with a ctl-c ...
puts "\n\n*** Shut down the server."
# Clean up after ourselves (in a separate task in case something goes wrong and we need to do it manually)
Rake::Task['unpreview'].invoke
end
desc "Move all stashed directories back into the source directory, ready for site generation. "
task :unpreview do
puts "\n*** Putting back the stashed files, removing the preview directory."
FileUtils.mv Dir.glob("#{STASH_DIR}/*"), "#{SOURCE_DIR}"
FileUtils.rm_rf(PREVIEW_DIR)
puts "\n*** Done.\n\n"
end
namespace :externalsources do
unless File.exists?("externalsources") && File.directory?("externalsources")
Dir.mkdir("externalsources")
end
# Returns the short name of a repo, which would be the directory name if you did a `git clone` without specifying a directory name. This isn't used anymore, but I left it around in case it's useful later.
def repo_name(repo_url)
repo_url.split('/')[-1].sub(/\.git$/, '')
end
# Returns something like git_github_com_puppetlabs_marionette-collective_git. We use this as the name of the main repo directory, because we may need to disambiguate between two repos with the same short name but a different user account on github.
def safe_dirname(name)
name.sub(/^[:\/\.]+/, '').gsub(/[:\/\.]+/, '_')
end
# "Update all working copies defined in source/_config.yml"
task :update do
Rake::Task['externalsources:clone'].invoke
Dir.chdir('externalsources') do
@config_data['externalsources'].each do |url, info|
workdir = safe_dirname(url)
local_repo = safe_dirname(info['repo'])
unless File.directory?(workdir)
puts "Making new working directory for #{url}"
system ("\"#{top_dir}/vendor/bin/git-new-workdir\" '#{local_repo}' '#{workdir}' '#{info['commit']}'")
end
Dir.chdir(workdir) do
puts "Updating #{url}"
system ("git checkout --force #{info['commit']} && git clean --force .")
end
end
end
end
# "Fetch all external doc repos (from externalsources in source/_config.yml), cloning any that don't yet exist"
task :clone do
repos = @config_data['externalsources'].values.map {|info| info['repo']}.uniq
Dir.chdir("externalsources") do
repos.each do |repo|
puts "Fetching #{repo}"
repo_dir = safe_dirname(repo)
system ("git clone #{repo} #{repo_dir}") unless File.directory?(repo_dir)
Dir.chdir(repo_dir) do
system("git fetch origin")
end
end
end
end
# "Symlink external documentation into place in the source directory"
task :link do
Rake::Task['externalsources:clean'].invoke # Bad things happen if any of these symlinks already exist, and Jekyll will run FOREVER
Rake::Task['externalsources:clean'].reenable
@config_data['externalsources'].each do |url, info|
workdir = safe_dirname(url)
# Have to use absolute paths for the source, since we have no idea how deep in the hierarchy the url is (and thus how many ../..s it would need).
FileUtils.ln_sf "#{top_dir}/externalsources/#{workdir}/#{info['subdirectory']}", "#{SOURCE_DIR}#{url}"
end
end
# "Clean up any external source symlinks from the source directory" # In the current implementation, all external sources are symlinks and there are no other symlinks in the source. This means we can naively kill all symlinks in ./source.
task :clean do
allsymlinks = FileList.new("#{SOURCE_DIR}/**/*").select{|f| File.symlink?(f)}
allsymlinks.each do |f|
File.delete(f)
end
end
end
desc "Clean up any crap left over from failed docs site builds"
task :clean do
# Get rid of external sources symlinks
Rake::Task['externalsources:clean'].invoke
Rake::Task['externalsources:clean'].reenable
# Get rid of the amended config file we write for Jekyll
FileUtils.rm("#{SOURCE_DIR}/_config_amended.yml")
end
desc "Generate the documentation"
task :generate do
Rake::Task['externalsources:update'].invoke # Create external sources if necessary, and check out the required working directories
Rake::Task['externalsources:link'].invoke # Link docs folders from external sources into the source at the appropriate places.
system("mkdir -p #{OUTPUT_DIR}")
system("rm -rf #{OUTPUT_DIR}/*")
jekyll()
Rake::Task['symlink_latest_versions'].invoke
Rake::Task['externalsources:clean'].invoke # The opposite of externalsources:link. Delete all symlinks in the source.
Rake::Task['externalsources:clean'].reenable
end
desc "Symlink latest versions of several projects; see symlink_latest and lock_latest lists in _config.yml"
task :symlink_latest_versions do
require 'puppet_docs/versions'
# Auto-link the latest version of every project in symlink_latest
@config_data['symlink_latest'].each do |project|
project_dir = "#{OUTPUT_DIR}/#{project}"
versions = Pathname.glob("#{project_dir}/*").select {|f|
f.directory? && !f.symlink?
}.map {|d| d.basename.to_s}
latest = @config_data['lock_latest'][project] || PuppetDocs::Versions.latest(versions)
Dir.chdir project_dir do
FileUtils.ln_sf latest, 'latest'
end
end
end
desc "Generate bidirectional redirects using the source/_redirects.yaml file, and write them to the Nginx config file in the output dir"
task :generate_redirects do
require 'puppet_docs/auto_redirects'
nginx_config = "#{OUTPUT_DIR}/nginx_rewrite.conf"
redirects_yaml = "#{SOURCE_DIR}/_redirects.yaml"
generated_lines = PuppetDocs::AutoRedirects.generate(@config_data, redirects_yaml)
File.open(nginx_config, 'a') {|f| f.write(generated_lines)}
end
desc "Serve generated output on port 9292"
task :serve do
system("rackup")
end
desc "Generate docs and serve locally"
task :run => [:generate, :serve]
task :write_version do
if File.directory?('.git')
current_commit = `git rev-parse HEAD`.strip
File.open(VERSION_FILE, 'w') do |f|
f.print(current_commit)
end
end
end
task :check_git_dirty_status do
if File.directory?('.git')
Rake::Task['externalsources:clean'].invoke # Don't let leftover symlinks from a cancelled build mark the tree as dirty.
Rake::Task['externalsources:clean'].reenable
if `git status --porcelain 2> /dev/null | tail -n1` != ''
STDOUT.puts "The working directory has uncommitted changes. They're probably either \n incomplete changes you don't want to release, or important changes you \n don't want lost; in either case, you might want to deal with them before \n you build and deploy the site. Continue anyway? (y/n)"
abort "Aborting." unless STDIN.gets.strip.downcase =~ /^y/
end
end
end
task :check_build_version do
abort "No site build found! Run 'rake build' before releasing." unless File.directory?(OUTPUT_DIR)
abort "Site build is empty! Run 'rake build' before releasing." if (Dir.entries(OUTPUT_DIR) - %w{ . .. }).empty?
if File.directory?('.git')
if File.exists?(VERSION_FILE)
head = `git rev-parse HEAD`.strip
build_version = File.read(VERSION_FILE)
if head != build_version
STDOUT.puts "This build wasn't built from HEAD and may be outdated. Continue anyway? (y/n)"
abort "Aborting." unless STDIN.gets.strip.downcase =~ /^y/
end
else
STDOUT.puts "Can't tell age of site build; it's probably outdated. Continue anyway? (y/n)"
abort "Aborting." unless STDIN.gets.strip.downcase =~ /^y/
end
end
end
desc "Build the documentation site and prepare it for deployment"
task :build do
Rake::Task['check_git_dirty_status'].invoke
Rake::Task['generate'].invoke
Rake::Task['generate_redirects'].invoke
Rake::Task['write_version'].invoke
end
desc 'Check all internal links in the site. This also results in a deployable site build.'
task :build_and_check_links do
@config_data['check_links'] = true
Rake::Task['build'].invoke
end
namespace :links do
desc 'Format a link report for latest versions of all projects'
task :report_latest_all do
prefixes = ['pe', 'puppet', 'puppetdb', 'puppetserver', 'facter', 'hiera'].map {|proj|
@config_data['document_version_index'][proj]['latest']
}.join(' ')
puts `#{top_dir}/util/link_report.rb #{prefixes}`
end
desc 'Format a link report for latest version of just PE'
task :report_latest_pe do
puts `#{top_dir}/util/link_report.rb #{@config_data['document_version_index']['pe']['latest']}`
end
end
desc "Instead of building real pages, build naked HTML fragments (with no nav, etc.)"
task :build_html_fragments do
Rake::Task['check_git_dirty_status'].invoke
Dir.chdir("#{SOURCE_DIR}/_layouts") do
FileUtils.mv("default.html", "real_default.html")
FileUtils.mv("fragment.html", "default.html")
end
Rake::Task['generate'].invoke
# don't write version, so we'll give a noisy error if you try to deploy fragments!
Dir.chdir("#{SOURCE_DIR}/_layouts") do
FileUtils.mv("default.html", "fragment.html")
FileUtils.mv("real_default.html", "default.html")
end
end
desc "List the available groups of references. Run `rake references:<GROUP>` to build."
task :references do
puts 'The following references are available:'
puts 'bundle exec rake references:puppet VERSION=<GIT TAG OR COMMIT>'
puts 'bundle exec rake references:facter VERSION=<GIT TAG OR COMMIT>'
puts 'bundle exec rake references:version_tables'
end
namespace :references do
task :puppet => 'references:check_version' do
require 'puppet_references'
PuppetReferences.build_puppet_references(ENV['VERSION'])
end
task :facter => 'references:check_version' do
require 'puppet_references'
PuppetReferences.build_facter_references(ENV['VERSION'])
end
task :version_tables do
require 'puppet_references'
PuppetReferences.build_version_tables
end
task :check_version do
abort "No VERSION given to build references for" unless ENV['VERSION']
end
end
desc "Deploy the site to the production servers"
task :deploy do
mirrors = ['mirror0','mirror1']
Rake::Task['build'].invoke
mirrors.each do |mirror|
sh "rake #{mirror} vlad:release"
# Note that the below will always fail on the second mirror, even though it should totally work. Life is filled with mystery. >:|
# Rake::Task[mirror].invoke
# Rake::Task['vlad:release'].reenable # so we can invoke it again if this isn't the last mirror
# Rake::Task['vlad:release'].invoke
end
end