require 'find'
require 'fileutils'
require 'erb'
module Webby
# The Builder class performs the work of scanning the content folder,
# creating Resource objects, and converting / copying the contents to the
# output folder as needed.
#
class Builder
class << self
# call-seq:
# Builder.run( :rebuild => false )
#
# Create a new instance of the Builder class and invoke the run method.
# If the :rebuild
option is given as +true+, then all pages
# will be recreated / copied.
#
def run( opts = {} )
self.new.run opts
end
# call-seq:
# Builder.create( page, :from => template, :locals => {} )
#
# This mehod is used to create a new _page_ in the content folder based
# on the specified template. _page_ is the relative path to the new page
# from the content/
folder. The _template_ is the name of
# the template to use from the templates/
folder.
#
def create( page, opts = {} )
tmpl = opts[:from]
raise Error, "template not given" unless tmpl
name = ::Webby::Resources::File.basename(page)
ext = ::Webby::Resources::File.extname(page)
dir = ::File.dirname(page)
dir = '' if dir == '.'
if tmpl.pathmap('%n') =~ %r/^_/
page = ::File.join(::Webby.site.content_dir, dir, '_'+name)
page << '.' << (ext.empty? ? 'txt' : ext)
elsif ::Webby.site.create_mode == 'directory' and name != 'index'
page = ::File.join(::Webby.site.content_dir, dir, name, 'index')
page << '.' << (ext.empty? ? 'txt' : ext)
else
page = ::File.join(::Webby.site.content_dir, page)
page << '.txt' if ext.empty?
end
raise Error, "#{page} already exists" if test ?e, page
Logging::Logger[self].info "creating #{page}"
FileUtils.mkdir_p ::File.dirname(page)
context = scope
opts[:locals].each do |k,v|
Thread.current[:value] = v
definition = "#{k} = Thread.current[:value]"
eval(definition, context)
end if opts.has_key?(:locals)
str = ERB.new(::File.read(tmpl), nil, '-').result(context)
::File.open(page, 'w') {|fd| fd.write str}
page
end
# call-seq:
# Builder.new_page_info( task ) => [page, title, directory]
#
def new_page_info( task )
return @new_page_info if defined? @new_page_info
raise "Usage: rake #{ARGV.first} path" unless ARGV.length > 1
page = task.application.top_level_tasks.slice!(1..-1).join('-')
title = ::Webby::Resources::File.basename(page).
split('-').map {|w| w.capitalize}.join(' ')
dir = ::File.dirname(page)
dir = '' if dir == '.'
@new_page_info = [page, title, dir]
end
private
# Returns the binding in the scope of the Builder class object.
#
def scope() binding end
end # class << self
# call-seq:
# Builder.new
#
# Creates a new Builder object for creating pages from the content and
# layout directories.
#
def initialize
@log = Logging::Logger[self]
end
# call-seq:
# run( :rebuild => false, :load_files => true )
#
# Runs the Webby builder by loading in the layout files from the
# layouts/
folder and the content from the
# contents/
folder. Content is analyzed, and those that need
# to be copied or compiled (filtered using ERB, Texttile, Markdown, etc.)
# are handled. The results are placed in the output/
folder.
#
# If the :rebuild
flag is set to +true+, then all content is
# copied and/or compiled to the output folder.
#
# A content file can mark itself as dirty by setting the +dirty+ flag to
# +true+ in the meta-data of the file. This will cause the contenet to
# always be compiled when the builder is run. Conversely, setting the
# dirty flag to +false+ will cause the content to never be compiled or
# copied to the output folder.
#
# A content file needs to be built if the age of the file is less then the
# age of the output product -- i.e. the content file has been modified
# more recently than the output file.
#
def run( opts = {} )
opts[:load_files] = true unless opts.has_key?(:load_files)
unless test(?d, output_dir)
@log.info "creating #{output_dir}"
FileUtils.mkdir output_dir
end
load_files if opts[:load_files]
Resources.pages.each do |page|
next unless page.dirty? or opts[:rebuild]
@log.info "creating #{page.destination}"
# make sure the directory exists
FileUtils.mkdir_p ::File.dirname(page.destination)
# copy the resource to the output directory if it is static
if page.instance_of? Resources::Static
FileUtils.cp page.path, page.destination
FileUtils.chmod 0644, page.destination
# otherwise, layout the resource and write the results to
# the output directory
else Renderer.write(page) end
end
# touch the cairn so we know when the website was last generated
FileUtils.touch ::Webby.cairn
nil
end
private
# Scan the layouts/
folder and the content/
# folder and create a new Resource object for each file found there.
#
def load_files
::Find.find(layout_dir, content_dir) do |path|
next unless test ?f, path
next if path =~ ::Webby.exclude
Resources.new path
end
end
%w(output_dir layout_dir content_dir).each do |key|
self.class_eval <<-CODE
def #{key}( ) ::Webby.site.#{key} end
CODE
end
end # class Builder
end # module Webby
# EOF