Refactor the Build CLI code to be easier to read and use Pathname throughout.
This also throws an error if :build_dir is set to a parent directory of :source_dir, which resolves #991.
This commit is contained in:
parent
ae2b6b34f1
commit
d3e4882b93
1 changed files with 95 additions and 118 deletions
|
@ -1,14 +1,18 @@
|
||||||
require "middleman-core"
|
require "middleman-core"
|
||||||
require "fileutils"
|
require "fileutils"
|
||||||
|
require 'set'
|
||||||
|
|
||||||
# CLI Module
|
# CLI Module
|
||||||
module Middleman::Cli
|
module Middleman::Cli
|
||||||
|
# Alias "b" to "build"
|
||||||
|
Base.map({ "b" => "build" })
|
||||||
|
|
||||||
# The CLI Build class
|
# The CLI Build class
|
||||||
class Build < Thor
|
class Build < Thor
|
||||||
include Thor::Actions
|
include Thor::Actions
|
||||||
|
|
||||||
attr_reader :debugging
|
attr_reader :debugging
|
||||||
|
attr_accessor :had_errors
|
||||||
|
|
||||||
check_unknown_options!
|
check_unknown_options!
|
||||||
|
|
||||||
|
@ -51,21 +55,19 @@ module Middleman::Cli
|
||||||
require 'find'
|
require 'find'
|
||||||
|
|
||||||
@debugging = Middleman::Cli::Base.respond_to?(:debugging) && Middleman::Cli::Base.debugging
|
@debugging = Middleman::Cli::Base.respond_to?(:debugging) && Middleman::Cli::Base.debugging
|
||||||
@had_errors = false
|
self.had_errors = false
|
||||||
|
|
||||||
self.class.shared_instance(options["verbose"], options["instrument"])
|
self.class.shared_instance(options["verbose"], options["instrument"])
|
||||||
|
|
||||||
self.class.shared_rack
|
|
||||||
|
|
||||||
opts = {}
|
opts = {}
|
||||||
opts[:glob] = options["glob"] if options.has_key?("glob")
|
opts[:glob] = options["glob"] if options.has_key?("glob")
|
||||||
opts[:clean] = options["clean"] if options.has_key?("clean")
|
opts[:clean] = options["clean"]
|
||||||
|
|
||||||
action GlobAction.new(self, opts)
|
action BuildAction.new(self, opts)
|
||||||
|
|
||||||
self.class.shared_instance.run_hook :after_build, self
|
self.class.shared_instance.run_hook :after_build, self
|
||||||
|
|
||||||
if @had_errors && !@debugging
|
if self.had_errors && !self.debugging
|
||||||
msg = "There were errors during this build"
|
msg = "There were errors during this build"
|
||||||
unless options["verbose"]
|
unless options["verbose"]
|
||||||
msg << ", re-run with `middleman build --verbose` to see the full exception."
|
msg << ", re-run with `middleman build --verbose` to see the full exception."
|
||||||
|
@ -73,7 +75,7 @@ module Middleman::Cli
|
||||||
self.shell.say msg, :red
|
self.shell.say msg, :red
|
||||||
end
|
end
|
||||||
|
|
||||||
exit(1) if @had_errors
|
exit(1) if self.had_errors
|
||||||
end
|
end
|
||||||
|
|
||||||
# Static methods
|
# Static methods
|
||||||
|
@ -91,88 +93,11 @@ module Middleman::Cli
|
||||||
logger(verbose ? 0 : 1, instrument)
|
logger(verbose ? 0 : 1, instrument)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
# Middleman::Application class singleton
|
|
||||||
#
|
|
||||||
# @return [Middleman::Application]
|
|
||||||
def shared_server
|
|
||||||
@_shared_server ||= shared_instance.class
|
|
||||||
end
|
end
|
||||||
|
|
||||||
# Rack::Test::Session singleton
|
|
||||||
#
|
|
||||||
# @return [Rack::Test::Session]
|
|
||||||
def shared_rack
|
|
||||||
@_shared_rack ||= ::Rack::Test::Session.new(shared_server.to_rack_app)
|
|
||||||
end
|
|
||||||
|
|
||||||
# Set the root path to the Middleman::Application's root
|
|
||||||
def source_root
|
|
||||||
shared_instance.root
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
no_tasks {
|
|
||||||
# Render a resource to a file.
|
|
||||||
#
|
|
||||||
# @param [Middleman::Sitemap::Resource] resource
|
|
||||||
# @return [String] The full path of the file that was written
|
|
||||||
def render_to_file(resource)
|
|
||||||
build_dir = self.class.shared_instance.config[:build_dir]
|
|
||||||
output_file = File.join(build_dir, resource.destination_path.gsub('%20', ' '))
|
|
||||||
|
|
||||||
if resource.binary?
|
|
||||||
if !File.exists?(output_file)
|
|
||||||
say_status :create, output_file, :green
|
|
||||||
elsif FileUtils.compare_file(resource.source_file, output_file)
|
|
||||||
say_status :identical, output_file, :blue
|
|
||||||
return output_file
|
|
||||||
else
|
|
||||||
say_status :update, output_file, :yellow
|
|
||||||
end
|
|
||||||
|
|
||||||
FileUtils.mkdir_p(File.dirname(output_file))
|
|
||||||
FileUtils.cp(resource.source_file, output_file)
|
|
||||||
else
|
|
||||||
begin
|
|
||||||
response = self.class.shared_rack.get(URI.escape(resource.request_path))
|
|
||||||
|
|
||||||
if response.status == 200
|
|
||||||
create_file(output_file, binary_encode(response.body))
|
|
||||||
else
|
|
||||||
handle_error(output_file, response.body)
|
|
||||||
end
|
|
||||||
rescue => e
|
|
||||||
handle_error(output_file, "#{e}\n#{e.backtrace.join("\n")}", e)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
output_file
|
|
||||||
end
|
|
||||||
|
|
||||||
def handle_error(file_name, response, e=Thor::Error.new(response))
|
|
||||||
@had_errors = true
|
|
||||||
|
|
||||||
say_status :error, file_name, :red
|
|
||||||
if self.debugging
|
|
||||||
raise e
|
|
||||||
exit(1)
|
|
||||||
elsif options["verbose"]
|
|
||||||
self.shell.say response, :red
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def binary_encode(string)
|
|
||||||
if string.respond_to?(:force_encoding)
|
|
||||||
string.force_encoding("ascii-8bit")
|
|
||||||
end
|
|
||||||
string
|
|
||||||
end
|
|
||||||
}
|
|
||||||
end
|
end
|
||||||
|
|
||||||
# A Thor Action, modular code, which does the majority of the work.
|
# A Thor Action, modular code, which does the majority of the work.
|
||||||
class GlobAction < ::Thor::Actions::EmptyDirectory
|
class BuildAction < ::Thor::Actions::EmptyDirectory
|
||||||
attr_reader :source
|
attr_reader :source
|
||||||
attr_reader :logger
|
attr_reader :logger
|
||||||
|
|
||||||
|
@ -182,22 +107,22 @@ module Middleman::Cli
|
||||||
# @param [Hash] config
|
# @param [Hash] config
|
||||||
def initialize(base, config={})
|
def initialize(base, config={})
|
||||||
@app = base.class.shared_instance
|
@app = base.class.shared_instance
|
||||||
source = @app.source
|
@source_dir = Pathname(@app.source_dir)
|
||||||
@destination = @app.build_dir
|
@build_dir = Pathname(@app.build_dir)
|
||||||
|
@to_clean = Set.new
|
||||||
|
|
||||||
@source = File.expand_path(base.find_in_source_paths(source.to_s))
|
@logger = @app.logger
|
||||||
|
@rack = ::Rack::Test::Session.new(@app.class.to_rack_app)
|
||||||
|
|
||||||
@logger = Middleman::Cli::Build.shared_instance.logger
|
super(base, @build_dir, config)
|
||||||
|
|
||||||
super(base, @destination, config)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
# Execute the action
|
# Execute the action
|
||||||
# @return [void]
|
# @return [void]
|
||||||
def invoke!
|
def invoke!
|
||||||
queue_current_paths if cleaning?
|
queue_current_paths if should_clean?
|
||||||
execute!
|
execute!
|
||||||
clean! if cleaning?
|
clean! if should_clean?
|
||||||
end
|
end
|
||||||
|
|
||||||
protected
|
protected
|
||||||
|
@ -205,39 +130,36 @@ module Middleman::Cli
|
||||||
# Remove files which were not built in this cycle
|
# Remove files which were not built in this cycle
|
||||||
# @return [void]
|
# @return [void]
|
||||||
def clean!
|
def clean!
|
||||||
@cleaning_queue.select { |q| q.file? }.each do |f|
|
@to_clean.each do |f|
|
||||||
base.remove_file f, :force => true
|
base.remove_file f, :force => true
|
||||||
end
|
end
|
||||||
|
|
||||||
Dir[File.join(@destination, "**", "*")].select { |d|
|
Dir[@build_dir.join("**", "*")].select {|d| File.directory?(d) }.each do |d|
|
||||||
File.directory?(d)
|
base.remove_file d, :force => true if directory_empty? d
|
||||||
}.each do |d|
|
|
||||||
base.remove_file d, :force => true if directory_empty? Pathname(d)
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
# Whether we should clean the build
|
# Whether we should clean the build
|
||||||
# @return [Boolean]
|
# @return [Boolean]
|
||||||
def cleaning?
|
def should_clean?
|
||||||
@config.has_key?(:clean) && @config[:clean]
|
@config[:clean]
|
||||||
end
|
end
|
||||||
|
|
||||||
# Whether the given directory is empty
|
# Whether the given directory is empty
|
||||||
# @param [String] directory
|
# @param [String, Pathname] directory
|
||||||
# @return [Boolean]
|
# @return [Boolean]
|
||||||
def directory_empty?(directory)
|
def directory_empty?(directory)
|
||||||
directory.children.empty?
|
Pathname(directory).children.empty?
|
||||||
end
|
end
|
||||||
|
|
||||||
# Get a list of all the paths in the destination folder and save them
|
# Get a list of all the file paths in the destination folder and save them
|
||||||
# for comparison against the files we build in this cycle
|
# for comparison against the files we build in this cycle
|
||||||
# @return [void]
|
# @return [void]
|
||||||
def queue_current_paths
|
def queue_current_paths
|
||||||
@cleaning_queue = []
|
return unless File.exist?(@build_dir)
|
||||||
return unless File.exist?(@destination)
|
|
||||||
|
|
||||||
paths = ::Middleman::Util.all_files_under(@destination).map(&:realpath)
|
paths = ::Middleman::Util.all_files_under(@build_dir).map(&:realpath).select(&:file?)
|
||||||
@cleaning_queue += paths.select do |path|
|
@to_clean += paths.select do |path|
|
||||||
path.to_s !~ /\/\./ || path.to_s =~ /\.(htaccess|htpasswd)/
|
path.to_s !~ /\/\./ || path.to_s =~ /\.(htaccess|htpasswd)/
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
@ -254,13 +176,13 @@ module Middleman::Cli
|
||||||
@app.sitemap.resources.select do |resource|
|
@app.sitemap.resources.select do |resource|
|
||||||
resource.ext == ".css"
|
resource.ext == ".css"
|
||||||
end.each do |resource|
|
end.each do |resource|
|
||||||
Middleman::Cli::Build.shared_rack.get(URI.escape(resource.destination_path))
|
@rack.get(URI.escape(resource.destination_path))
|
||||||
end
|
end
|
||||||
|
|
||||||
logger.debug "== Checking for Compass sprites"
|
logger.debug "== Checking for Compass sprites"
|
||||||
|
|
||||||
# Double-check for compass sprites
|
# Double-check for compass sprites
|
||||||
@app.files.find_new_files((Pathname(@app.source_dir) + @app.images_dir).relative_path_from(@app.root_path))
|
@app.files.find_new_files((@source_dir + @app.images_dir).relative_path_from(@app.root_path))
|
||||||
@app.sitemap.ensure_resource_list_updated!
|
@app.sitemap.ensure_resource_list_updated!
|
||||||
|
|
||||||
# Sort paths to be built by the above order. This is primarily so Compass can
|
# Sort paths to be built by the above order. This is primarily so Compass can
|
||||||
|
@ -273,24 +195,79 @@ module Middleman::Cli
|
||||||
sort_order.index(r.ext) || 100
|
sort_order.index(r.ext) || 100
|
||||||
end
|
end
|
||||||
|
|
||||||
|
if @build_dir.expand_path.relative_path_from(@source_dir).to_s =~ /\A[.\/]+\Z/
|
||||||
|
raise ":build_dir (#{@build_dir}) cannot be a parent of :source_dir (#{@source_dir})"
|
||||||
|
end
|
||||||
|
|
||||||
# Loop over all the paths and build them.
|
# Loop over all the paths and build them.
|
||||||
resources.each do |resource|
|
resources.each do |resource|
|
||||||
next if @config[:glob] && !File.fnmatch(@config[:glob], resource.destination_path)
|
next if @config[:glob] && !File.fnmatch(@config[:glob], resource.destination_path)
|
||||||
|
|
||||||
output_path = base.render_to_file(resource)
|
output_path = render_to_file(resource)
|
||||||
|
|
||||||
if cleaning?
|
if should_clean?
|
||||||
pn = Pathname(output_path)
|
@to_clean.delete(output_path.realpath) if output_path.exist?
|
||||||
@cleaning_queue.delete(pn.realpath) if pn.exist?
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
::Middleman::Profiling.report("build")
|
::Middleman::Profiling.report("build")
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# Render a resource to a file.
|
||||||
|
#
|
||||||
|
# @param [Middleman::Sitemap::Resource] resource
|
||||||
|
# @return [Pathname] The full path of the file that was written
|
||||||
|
def render_to_file(resource)
|
||||||
|
output_file = @build_dir + resource.destination_path.gsub('%20', ' ')
|
||||||
|
|
||||||
|
if resource.binary?
|
||||||
|
if !output_file.exist?
|
||||||
|
base.say_status :create, output_file, :green
|
||||||
|
elsif FileUtils.compare_file(resource.source_file, output_file)
|
||||||
|
base.say_status :identical, output_file, :blue
|
||||||
|
return output_file
|
||||||
|
else
|
||||||
|
base.say_status :update, output_file, :yellow
|
||||||
end
|
end
|
||||||
|
|
||||||
# Alias "b" to "build"
|
output_file.dirname.mkpath
|
||||||
Base.map({ "b" => "build" })
|
FileUtils.cp(resource.source_file, output_file)
|
||||||
|
else
|
||||||
|
begin
|
||||||
|
response = @rack.get(URI.escape(resource.request_path))
|
||||||
|
|
||||||
|
if response.status == 200
|
||||||
|
base.create_file(output_file, binary_encode(response.body))
|
||||||
|
else
|
||||||
|
handle_error(output_file, response.body)
|
||||||
|
end
|
||||||
|
rescue => e
|
||||||
|
handle_error(output_file, "#{e}\n#{e.backtrace.join("\n")}", e)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
output_file
|
||||||
|
end
|
||||||
|
|
||||||
|
def handle_error(file_name, response, e=Thor::Error.new(response))
|
||||||
|
base.had_errors = true
|
||||||
|
|
||||||
|
base.say_status :error, file_name, :red
|
||||||
|
if base.debugging
|
||||||
|
raise e
|
||||||
|
exit(1)
|
||||||
|
elsif base.options["verbose"]
|
||||||
|
base.shell.say response, :red
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def binary_encode(string)
|
||||||
|
if string.respond_to?(:force_encoding)
|
||||||
|
string.force_encoding("ascii-8bit")
|
||||||
|
end
|
||||||
|
string
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
# Quiet down create file
|
# Quiet down create file
|
||||||
|
|
Loading…
Add table
Reference in a new issue