require 'asciidoctor'
require 'asciidoctor/cli'
require 'asciidoctor-diagram'
require 'find'
require 'git'
require 'logger'
require 'pandoc-ruby'
require 'pathname'
require 'sitemap_generator'
require 'yaml'
require 'forwardable'
module AsciiBinder
module Helpers
extend Forwardable
def self.source_dir
@source_dir ||= `git rev-parse --show-toplevel`.chomp
end
def self.template_dir
@template_dir ||= File.join(source_dir,'_templates')
end
def self.preview_dir
@preview_dir ||= begin
lpreview_dir = File.join(source_dir,PREVIEW_DIRNAME)
if not File.exists?(lpreview_dir)
Dir.mkdir(lpreview_dir)
end
lpreview_dir
end
end
def self.package_dir
@package_dir ||= begin
lpackage_dir = File.join(source_dir,PACKAGE_DIRNAME)
if not File.exists?(lpackage_dir)
Dir.mkdir(lpackage_dir)
end
lpackage_dir
end
end
def_delegators self, :source_dir, :template_dir, :preview_dir, :package_dir
TemplateRenderer.initialize_cache(template_dir)
BUILD_FILENAME = '_build_cfg.yml'
DISTRO_MAP_FILENAME = '_distro_map.yml'
BUILDER_DIRNAME = '_build_system'
PREVIEW_DIRNAME = '_preview'
PACKAGE_DIRNAME = '_package'
BLANK_STRING_RE = Regexp.new('^\s*$')
def build_date
Time.now.utc
end
def git
@git ||= Git.open(source_dir)
end
def git_checkout branch_name
target_branch = git.branches.local.select{ |b| b.name == branch_name }[0]
if not target_branch.nil? and not target_branch.current
target_branch.checkout
end
end
def git_stash_all
# See if there are any changes in need of stashing
@stash_needed = `git status --porcelain` !~ /^\s*$/
if @stash_needed
puts "\nNOTICE: Stashing uncommited changes and files in working branch."
`git stash -u`
end
end
def git_apply_and_drop
return unless @stash_needed
puts "\nNOTE: Re-applying uncommitted changes and files to working branch."
if system("git stash pop")
puts "NOTE: Stash application successful."
else
puts "ERROR: Could not apply stashed code. Run `git stash apply` manually."
end
@stash_needed = false
end
# Returns the local git branches; current branch is always first
def local_branches
@local_branches ||= begin
branches = []
branches << git.branches.local.select{ |b| b.current }[0].name
branches << git.branches.local.select{ |b| not b.current }.map{ |b| b.name }
branches.flatten
end
end
def working_branch
@working_branch ||= local_branches[0]
end
def build_config_file
@build_config_file ||= File.join(source_dir,BUILD_FILENAME)
end
def distro_map_file
@distro_map_file ||= File.join(source_dir, DISTRO_MAP_FILENAME)
end
# Protip: Don't cache this! It needs to be reread every time we change branches.
def build_config
validate_config(YAML.load_stream(open(build_config_file)))
end
def find_topic_files
file_list = Find.find('.').select{ |path| not path.nil? and path =~ /.*\.adoc$/ and not path =~ /README/ and not path =~ /\/old\// and not path.split('/').length < 3 }
file_list.map{ |path|
parts = path.split('/').slice(1..-1);
parts.slice(0..-2).join('/') + '/' + parts[-1].split('.')[0]
}
end
def remove_found_config_files(branch,branch_build_config,branch_topic_files)
nonexistent_topics = []
branch_build_config.each do |topic_group|
tg_dir = topic_group['Dir']
topic_group['Topics'].each do |topic|
if topic.has_key?('File')
topic_path = tg_dir + '/' + topic['File']
result = branch_topic_files.delete(topic_path)
if result.nil?
nonexistent_topics << topic_path
end
elsif topic.has_key?('Dir')
topic_path = tg_dir + '/' + topic['Dir'] + '/'
topic['Topics'].each do |subtopic|
result = branch_topic_files.delete(topic_path + subtopic['File'])
if result.nil?
nonexistent_topics << topic_path + subtopic['File']
end
end
end
end
end
if nonexistent_topics.length > 0
puts "\nWARNING: The _build_cfg.yml file on branch '#{branch}' references nonexistant topics:\n" + nonexistent_topics.map{ |topic| "- #{topic}" }.join("\n")
end
end
def distro_map
@distro_map ||= YAML.load_file(distro_map_file)
end
def site_map
site_map = {}
distro_map.each do |distro,distro_config|
if not site_map.has_key?(distro_config["site"])
site_map[distro_config["site"]] = { :distros => {}, :name => distro_config['site_name'], :url => distro_config['site_url'] }
end
site_map[distro_config["site"]][:distros][distro] = distro_config["branches"]
end
site_map
end
def distro_branches(use_distro='')
@distro_branches ||= begin
use_distro_list = use_distro == '' ? distro_map.keys : [use_distro]
distro_map.select{ |dkey,dval| use_distro_list.include?(dkey) }.map{ |distro,dconfig| dconfig["branches"].keys }.flatten
end
end
def page(args)
# TODO: This process of rebuilding the entire nav for every page will not scale well.
# As the doc set increases, we will need to think about refactoring this.
args[:breadcrumb_root], args[:breadcrumb_group], args[:breadcrumb_subgroup], args[:breadcrumb_topic] = extract_breadcrumbs(args)
args[:breadcrumb_subgroup_block] = ''
args[:subtopic_shim] = ''
if args[:breadcrumb_subgroup]
args[:breadcrumb_subgroup_block] = "
#{args[:breadcrumb_subgroup]}"
args[:subtopic_shim] = '../'
end
TemplateRenderer.new.render("_templates/page.html.erb", args)
end
def extract_breadcrumbs(args)
breadcrumb_root = breadcrumb_group = breadcrumb_subgroup = breadcrumb_topic = nil
root_group = args[:navigation].first
selected_group = args[:navigation].detect { |group| group[:id] == args[:group_id] }
selected_subgroup = selected_group[:topics].detect { |subgroup| subgroup[:id] == args[:subgroup_id] }
current_is_subtopic = selected_subgroup ? true : false
if root_group
root_topic = root_group[:topics].first
breadcrumb_root = linkify_breadcrumb(root_topic[:path], "#{args[:distro]} #{args[:version]}", current_is_subtopic) if root_topic
end
if selected_group
group_topic = selected_group[:topics].first
breadcrumb_group = linkify_breadcrumb(group_topic[:path], selected_group[:name], current_is_subtopic) if group_topic
if selected_subgroup
subgroup_topic = selected_subgroup[:topics].first
breadcrumb_subgroup = linkify_breadcrumb(subgroup_topic[:path], selected_subgroup[:name], current_is_subtopic) if subgroup_topic
selected_topic = selected_subgroup[:topics].detect { |topic| topic[:id] == args[:topic_id] }
breadcrumb_topic = linkify_breadcrumb(nil, selected_topic[:name], current_is_subtopic) if selected_topic
else
selected_topic = selected_group[:topics].detect { |topic| topic[:id] == args[:topic_id] }
breadcrumb_topic = linkify_breadcrumb(nil, selected_topic[:name], current_is_subtopic) if selected_topic
end
end
return breadcrumb_root, breadcrumb_group, breadcrumb_subgroup, breadcrumb_topic
end
def linkify_breadcrumb(href, text, extra_level)
addl_level = extra_level ? '../' : ''
href ? "#{text}" : text
end
def parse_distros distros_string, for_validation=false
values = distros_string.split(',').map{ |v| v.strip }
return values if for_validation
return distro_map.keys if values.include?('all')
return values.uniq
end
def validate_distros distros_string
return false if not distros_string.is_a?(String)
values = parse_distros(distros_string, true)
values.each do |v|
return false if not v == 'all' and not distro_map.keys.include?(v)
end
return true
end
def validate_topic_group group, info
# Check for presence of topic group keys
['Name','Dir','Topics'].each do |group_key|
if not group.has_key?(group_key)
raise "One of the topic groups in #{build_config_file} is missing the '#{group_key}' key."
end
end
# Check for right format of topic group values
['Name','Dir'].each do |group_key|
if not group[group_key].is_a?(String)
raise "One of the topic groups in #{build_config_file} is not using a string for the #{group_key} setting; current value is #{group[group_key].inspect}"
end
if group[group_key].empty? or group[group_key].match BLANK_STRING_RE
raise "One of the topic groups in #{build_config_file} is using a blank value for the #{group_key} setting."
end
end
if not File.exists?(File.join(source_dir,info[:path]))
raise "In #{build_config_file}, the directory path '#{info[:path]}' for topic group #{group['Name']} does not exist under #{source_dir}"
end
# Validate the Distros setting
if group.has_key?('Distros')
if not validate_distros(group['Distros'])
key_list = distro_map.keys.map{ |k| "'#{k}'" }.sort.join(', ')
raise "In #{build_config_file}, the Distros value #{group['Distros'].inspect} for topic group #{group['Name']} is not valid. Legal values are 'all', #{key_list}, or a comma-separated list of legal values."
end
group['Distros'] = parse_distros(group['Distros'])
else
group['Distros'] = parse_distros('all')
end
if not group['Topics'].is_a?(Array)
raise "The #{group['Name']} topic group in #{build_config_file} is malformed; the build system is expecting an array of 'Topic' definitions."
end
# Generate an ID for this topic group
group['ID'] = camelize group['Name']
if info.has_key?(:parent_id)
group['ID'] = "#{info[:parent_id]}::#{group['ID']}"
end
end
def validate_topic_item item, info
['Name','File'].each do |topic_key|
if not item[topic_key].is_a?(String)
raise "In #{build_config_file}, topic group #{info[:group]}, one of the topics is not using a string for the '#{topic_key}' setting; current value is #{item[topic_key].inspect}"
end
if item[topic_key].empty? or item[topic_key].match BLANK_STRING_RE
raise "In #{build_config_file}, topic group #{topic_group['Name']}, one of the topics is using a blank value for the '#{topic_key}' setting"
end
end
# Normalize the filenames
if item['File'].end_with?('.adoc')
item['File'] = item['File'][0..-6]
end
if not File.exists?(File.join(source_dir,info[:path],"#{item['File']}.adoc"))
raise "In #{build_config_file}, could not find file #{item['File']} under directory #{info[:path]} for topic #{item['Name']} in topic group #{info[:group]}."
end
if item.has_key?('Distros')
if not validate_distros(item['Distros'])
key_list = distro_map.keys.map{ |k| "'#{k}'" }.sort.join(', ')
raise "In #{build_config_file}, the Distros value #{item['Distros'].inspect} for topic item #{item['Name']} in topic group #{info[:group]} is not valid. Legal values are 'all', #{key_list}, or a comma-separated list of legal values."
end
item['Distros'] = parse_distros(item['Distros'])
else
item['Distros'] = parse_distros('all')
end
# Generate an ID for this topic
item['ID'] = "#{info[:group_id]}::#{camelize(item['Name'])}"
end
def validate_config config_data
# Validate/normalize the config file straight away
if not config_data.is_a?(Array)
raise "The configuration in #{build_config_file} is malformed; the build system is expecting an array of topic groups."
end
config_data.each do |topic_group|
validate_topic_group(topic_group, { :path => topic_group['Dir'] })
# Now buzz through the topics
topic_group['Topics'].each do |topic|
# Is this an actual topic or a subtopic group?
is_subtopic_group = topic.has_key?('Dir') and topic.has_key?('Topics') and not topic.has_key?('File')
is_topic_item = topic.has_key?('File') and not topic.has_key?('Dir') and not topic.has_key?('Topics')
if not is_subtopic_group and not is_topic_item
raise "This topic could not definitively be determined to be a topic item or a subtopic group:\n#{topic.inspect}"
end
if is_topic_item
validate_topic_item(topic, { :group => topic_group['Name'], :group_id => topic_group['ID'], :path => topic_group['Dir'] })
elsif is_subtopic_group
topic_path = "#{topic_group['Dir']}/#{topic['Dir']}"
validate_topic_group(topic, { :path => topic_path, :parent_id => topic_group['ID'] })
topic['Topics'].each do |subtopic|
validate_topic_item(subtopic, { :group => "#{topic_group['Name']}/#{topic['Name']}", :group_id => topic['ID'], :path => topic_path })
end
end
end
end
config_data
end
def camelize text
text.gsub(/[^0-9a-zA-Z ]/i, '').split(' ').map{ |t| t.capitalize }.join
end
def nav_tree distro, branch_build_config
navigation = []
branch_build_config.each do |topic_group|
next if not topic_group['Distros'].include?(distro)
next if topic_group['Topics'].select{ |t| t['Distros'].include?(distro) }.length == 0
topic_list = []
topic_group['Topics'].each do |topic|
next if not topic['Distros'].include?(distro)
if topic.has_key?('File')
topic_list << {
:path => "../#{topic_group['Dir']}/#{topic['File']}.html",
:name => topic['Name'],
:id => topic['ID'],
}
elsif topic.has_key?('Dir')
next if topic['Topics'].select{ |t| t['Distros'].include?(distro) }.length == 0
subtopic_list = []
topic['Topics'].each do |subtopic|
next if not subtopic['Distros'].include?(distro)
subtopic_list << {
:path => "../#{topic_group['Dir']}/#{topic['Dir']}/#{subtopic['File']}.html",
:name => subtopic['Name'],
:id => subtopic['ID'],
}
end
topic_list << { :name => topic['Name'], :id => topic['ID'], :topics => subtopic_list }
end
end
navigation << { :name => topic_group['Name'], :id => topic_group['ID'], :topics => topic_list }
end
navigation
end
def asciidoctor_page_attrs(more_attrs=[])
[
'source-highlighter=coderay',
'coderay-css=style',
'linkcss!',
'icons=font',
'idprefix=',
'idseparator=-',
'sectanchors',
].concat(more_attrs)
end
def generate_docs(build_distro,single_page=nil)
single_page_dir = []
single_page_file = nil
if not single_page.nil?
single_page_dir = single_page.split(':')[0].split('/')
single_page_file = single_page.split(':')[1]
puts "Rebuilding '#{single_page_dir.join('/')}/#{single_page_file}' on branch '#{working_branch}'."
end
if not build_distro == ''
if not distro_map.has_key?(build_distro)
exit
else
puts "Building only the #{distro_map[build_distro]["name"]} distribution."
end
elsif single_page.nil?
puts "Building all distributions."
end
# First, notify the user of missing local branches
missing_branches = []
distro_branches(build_distro).sort.each do |dbranch|
next if local_branches.include?(dbranch)
missing_branches << dbranch
end
if missing_branches.length > 0 and single_page.nil?
puts "\nNOTE: The following branches do not exist in your local git repo:"
missing_branches.each do |mbranch|
puts "- #{mbranch}"
end
puts "The build will proceed but these branches will not be generated."
end
# Generate all distros for every local branch
local_branches.each do |local_branch|
# Single-page regen only occurs for the working branch
if not local_branch == working_branch
if single_page.nil?
# Checkout the branch
puts "\nCHANGING TO BRANCH '#{local_branch}'"
git_checkout(local_branch)
else
next
end
end
first_branch = single_page.nil?
if local_branch =~ /^\(detached from .*\)/
local_branch = 'detached'
end
# The branch_orphan_files list starts with the set of all
# .adoc files found in the repo, and will be whittled
# down from there.
branch_orphan_files = find_topic_files
branch_build_config = build_config
remove_found_config_files(local_branch,branch_build_config,branch_orphan_files)
if branch_orphan_files.length > 0 and single_page.nil?
puts "\nWARNING: Branch '#{local_branch}' includes the following .adoc files that are not referenced in the _build_cfg.yml file:\n" + branch_orphan_files.map{ |file| "- #{file}" }.join("\n")
end
# Run all distros.
distro_map.each do |distro,distro_config|
# Only building a single distro; skip the others.
if not build_distro == '' and not build_distro == distro
next
end
site_name = distro_config["site_name"]
branch_config = { "name" => "Branch Build", "dir" => local_branch }
dev_branch = true
if distro_config["branches"].has_key?(local_branch)
branch_config = distro_config["branches"][local_branch]
dev_branch = false
end
if first_branch
puts "\nBuilding #{distro_config["name"]} for branch '#{local_branch}'"
first_branch = false
end
# Create the target dir
branch_path = File.join(preview_dir,distro,branch_config["dir"])
system("mkdir -p #{branch_path}/stylesheets")
system("mkdir -p #{branch_path}/javascripts")
system("mkdir -p #{branch_path}/images")
# Copy stylesheets into preview area
system("cp -r _stylesheets/*css #{branch_path}/stylesheets")
# Copy javascripts into preview area
system("cp -r _javascripts/*js #{branch_path}/javascripts")
# Copy images into preview area
system("cp -r _images/* #{branch_path}/images")
# Build the landing page
navigation = nav_tree(distro,branch_build_config)
# Build the topic files for this branch & distro
branch_build_config.each do |topic_group|
next if not topic_group['Distros'].include?(distro)
next if topic_group['Topics'].select{ |t| t['Distros'].include?(distro) }.length == 0
next if not single_page.nil? and not single_page_dir[0] == topic_group['Dir']
topic_group['Topics'].each do |topic|
src_group_path = File.join(source_dir,topic_group['Dir'])
tgt_group_path = File.join(branch_path,topic_group['Dir'])
if not File.exists?(tgt_group_path)
Dir.mkdir(tgt_group_path)
end
next if not topic['Distros'].include?(distro)
if topic.has_key?('File')
next if not single_page.nil? and not topic['File'] == single_page_file
topic_path = File.join(topic_group['Dir'],topic['File'])
configure_and_generate_page({
:distro => distro,
:distro_config => distro_config,
:branch_config => branch_config,
:navigation => navigation,
:topic => topic,
:topic_group => topic_group,
:topic_path => topic_path,
:src_group_path => src_group_path,
:tgt_group_path => tgt_group_path,
:single_page => single_page,
:site_name => site_name,
})
elsif topic.has_key?('Dir')
next if not single_page.nil? and not single_page_dir.join('/') == topic_group['Dir'] + '/' + topic['Dir']
topic['Topics'].each do |subtopic|
next if not subtopic['Distros'].include?(distro)
next if not single_page.nil? and not subtopic['File'] == single_page_file
src_group_path = File.join(source_dir,topic_group['Dir'],topic['Dir'])
tgt_group_path = File.join(branch_path,topic_group['Dir'],topic['Dir'])
if not File.exists?(tgt_group_path)
Dir.mkdir(tgt_group_path)
end
topic_path = File.join(topic_group['Dir'],topic['Dir'],subtopic['File'])
configure_and_generate_page({
:distro => distro,
:distro_config => distro_config,
:branch_config => branch_config,
:navigation => navigation,
:topic => subtopic,
:topic_group => topic_group,
:topic_subgroup => topic,
:topic_path => topic_path,
:src_group_path => src_group_path,
:tgt_group_path => tgt_group_path,
:single_page => single_page,
:site_name => site_name,
})
end
end
end
end
if not single_page.nil?
next
end
# Create a distro landing page
# This is backwards compatible code. We can remove it when no
# official repo uses index.adoc. We are moving to flat HTML
# files for index.html
src_file_path = File.join(source_dir,'index.adoc')
if File.exists?(src_file_path)
topic_adoc = File.open(src_file_path,'r').read
page_attrs = asciidoctor_page_attrs([
"imagesdir=#{File.join(source_dir,'_site_images')}",
distro,
"product-title=#{distro_config["name"]}",
"product-version=Updated #{build_date}",
"product-author=#{distro_config["author"]}"
])
topic_html = Asciidoctor.render topic_adoc, :header_footer => true, :safe => :unsafe, :attributes => page_attrs
File.write(File.join(preview_dir,distro,'index.html'),topic_html)
end
end
if not single_page.nil?
return
end
if local_branch == working_branch
# We're moving away from the working branch, so save off changed files
git_stash_all
end
end
# Return to the original branch
git_checkout(working_branch)
# If necessary, restore temporarily stashed files
git_apply_and_drop
puts "\nAll builds completed."
end
def configure_and_generate_page options
distro = options[:distro]
distro_config = options[:distro_config]
branch_config = options[:branch_config]
navigation = options[:navigation]
topic = options[:topic]
topic_group = options[:topic_group]
topic_subgroup = options[:topic_subgroup]
topic_path = options[:topic_path]
src_group_path = options[:src_group_path]
tgt_group_path = options[:tgt_group_path]
single_page = options[:single_page]
site_name = options[:site_name]
src_file_path = File.join(src_group_path,"#{topic['File']}.adoc")
tgt_file_path = File.join(tgt_group_path,"#{topic['File']}.html")
if single_page.nil?
puts " - #{topic_path}"
end
topic_adoc = File.open(src_file_path,'r').read
page_attrs = asciidoctor_page_attrs([
"imagesdir=#{src_group_path}/images",
distro,
"product-title=#{distro_config["name"]}",
"product-version=#{branch_config["name"]}",
"product-author=#{distro_config["author"]}"
])
# Because we render only the body of the article with AsciiDoctor, the full article title
# would be lost in conversion. So, use the _build_cfg.yml 'Name' as a fallback but try
# to read the full article title out of the file itself.
file_lines = topic_adoc.split("\n")
article_title = topic['Name']
if file_lines.length > 0
article_title = file_lines[0].gsub(/^\=\s+/, '').gsub(/\s+$/, '').gsub(/\{product-title\}/, distro_config["name"]).gsub(/\{product-version\}/, branch_config["name"])
end
topic_html = Asciidoctor.render topic_adoc, :header_footer => false, :safe => :unsafe, :attributes => page_attrs
dir_depth = ''
if branch_config['dir'].split('/').length > 1
dir_depth = '../' * (branch_config['dir'].split('/').length - 1)
end
if not topic_subgroup.nil?
dir_depth = '../' + dir_depth
end
page_args = {
:distro_key => distro,
:distro => distro_config["name"],
:site_name => site_name,
:version => branch_config["name"],
:group_title => topic_group['Name'],
:subgroup_title => topic_subgroup && topic_subgroup['Name'],
:topic_title => topic['Name'],
:article_title => article_title,
:content => topic_html,
:navigation => navigation,
:group_id => topic_group['ID'],
:subgroup_id => topic_subgroup && topic_subgroup['ID'],
:topic_id => topic['ID'],
:css_path => "../../#{dir_depth}#{branch_config["dir"]}/stylesheets/",
:javascripts_path => "../../#{dir_depth}#{branch_config["dir"]}/javascripts/",
:images_path => "../../#{dir_depth}#{branch_config["dir"]}/images/",
:site_home_path => "../../#{dir_depth}index.html",
:css => ['docs.css'],
}
full_file_text = page(page_args)
File.write(tgt_file_path,full_file_text)
end
# package_docs
# This method generates the docs and then organizes them the way they will be arranged
# for the production websites.
def package_docs(package_site)
site_map.each do |site,site_config|
next if not package_site == '' and not package_site == site
site_config[:distros].each do |distro,branches|
branches.each do |branch,branch_config|
src_dir = File.join(preview_dir,distro,branch_config["dir"])
tgt_tdir = branch_config["dir"].split('/')
tgt_tdir.pop
tgt_dir = ''
if tgt_tdir.length > 0
tgt_dir = File.join(package_dir,site,tgt_tdir.join('/'))
else
tgt_dir = File.join(package_dir,site)
end
next if not File.directory?(src_dir)
FileUtils.mkdir_p(tgt_dir)
FileUtils.cp_r(src_dir,tgt_dir)
end
site_dir = File.join(package_dir,site)
if File.directory?(site_dir)
puts "\nBuilding #{site} site."
# With this update, site index files will always come from the master branch
working_branch_site_index = File.join(source_dir,'index-' + site + '.html')
if File.exists?(working_branch_site_index)
FileUtils.cp(working_branch_site_index,File.join(package_dir,site,'index.html'))
['_images','_stylesheets'].each do |support_dir|
FileUtils.cp_r(File.join(source_dir,support_dir),File.join(package_dir,site,support_dir))
end
else
FileUtils.cp(File.join(preview_dir,distro,'index.html'),File.join(package_dir,site,'index.html'))
end
# Now build a sitemap
site_dir_path = Pathname.new(site_dir)
SitemapGenerator::Sitemap.default_host = site_config[:url]
SitemapGenerator::Sitemap.create( :compress => false, :filename => File.join(site_dir,'sitemap') ) do
file_list = Find.find(site_dir).select{ |path| not path.nil? and path =~ /.*\.html$/ }.map{ |path| '/' + Pathname.new(path).relative_path_from(site_dir_path).to_s }
file_list.each do |file|
add(file, :changefreq => 'daily')
end
end
end
end
end
end
end
end