# encoding: utf-8 usage 'compile [options]' summary 'compile items of this site' description <<-EOS Compile all items of the current site. The compile command will show all items of the site as they are processed. The time spent compiling the item will be printed, as well as a status message, which can be one of the following: CREATED - The compiled item did not yet exist and has been created UPDATED - The compiled item did already exist and has been modified IDENTICAL - The item was deemed outdated and has been recompiled, but the compiled version turned out to be identical to the already existing version SKIP - The item was deemed not outdated and was therefore not recompiled EOS option :a, :all, '(ignored)' option :f, :force, '(ignored)' run do |opts, args, cmd| Nanoc3::CLI::Commands::Compile.call(opts, args, cmd) end module Nanoc3::CLI::Commands class Compile < ::Nanoc3::CLI::Command def run # Make sure we are in a nanoc site directory puts "Loading site data..." self.require_site # Check presence of --all option if options.has_key?(:all) || options.has_key?(:force) $stderr.puts "Warning: the --force option (and its deprecated --all alias) are, as of nanoc 3.2, no longer supported and have no effect." end # Warn if trying to compile a single item if arguments.size == 1 $stderr.puts '-' * 80 $stderr.puts 'Note: As of nanoc 3.2, it is no longer possible to compile a single item. When invoking the “compile” command, all items in the site will be compiled.'.make_compatible_with_env $stderr.puts '-' * 80 end # Give feedback puts "Compiling site..." # Initialize profiling stuff time_before = Time.now @rep_times = {} @filter_times = {} setup_notifications # Prepare for generating diffs setup_diffs # Compile self.site.compile # Find reps reps = self.site.items.map { |i| i.reps }.flatten # Show skipped reps reps.select { |r| !r.compiled? }.each do |rep| rep.raw_paths.each do |snapshot_name, filename| next if filename.nil? duration = @rep_times[filename] Nanoc3::CLI::Logger.instance.file(:high, :skip, filename, duration) end end # Stop diffing teardown_diffs # Give general feedback puts puts "Site compiled in #{format('%.2f', Time.now - time_before)}s." # Give detailed feedback if options.has_key?(:verbose) print_profiling_feedback(reps) end end def setup_notifications # File notifications Nanoc3::NotificationCenter.on(:will_write_rep) do |rep, snapshot| generate_diff_for(rep, snapshot) end Nanoc3::NotificationCenter.on(:rep_written) do |rep, path, is_created, is_modified| action = (is_created ? :create : (is_modified ? :update : :identical)) duration = Time.now - @rep_times[rep.raw_path] if @rep_times[rep.raw_path] Nanoc3::CLI::Logger.instance.file(:high, action, path, duration) end # Debug notifications if self.debug? Nanoc3::NotificationCenter.on(:compilation_started) do |rep| puts "*** Started compilation of #{rep.inspect}" end Nanoc3::NotificationCenter.on(:compilation_ended) do |rep| puts "*** Ended compilation of #{rep.inspect}" end Nanoc3::NotificationCenter.on(:compilation_failed) do |rep| puts "*** Suspended compilation of #{rep.inspect} due to unmet dependencies" end Nanoc3::NotificationCenter.on(:cached_content_used) do |rep| puts "*** Used cached compiled content for #{rep.inspect} instead of recompiling" end Nanoc3::NotificationCenter.on(:filtering_started) do |rep, filter_name| puts "*** Started filtering #{rep.inspect} with #{filter_name}" end Nanoc3::NotificationCenter.on(:filtering_ended) do |rep, filter_name| puts "*** Ended filtering #{rep.inspect} with #{filter_name}" end end # Timing notifications Nanoc3::NotificationCenter.on(:compilation_started) do |rep| @rep_times[rep.raw_path] = Time.now end Nanoc3::NotificationCenter.on(:compilation_ended) do |rep| @rep_times[rep.raw_path] = Time.now - @rep_times[rep.raw_path] end Nanoc3::NotificationCenter.on(:filtering_started) do |rep, filter_name| @filter_times[filter_name] ||= [] @filter_times[filter_name] << Time.now start_filter_progress(rep, filter_name) end Nanoc3::NotificationCenter.on(:filtering_ended) do |rep, filter_name| @filter_times[filter_name] << Time.now - @filter_times[filter_name].pop stop_filter_progress(rep, filter_name) end end def setup_diffs @diff_lock = Mutex.new @diff_threads = [] FileUtils.rm('output.diff') if File.file?('output.diff') end def teardown_diffs @diff_threads.each { |t| t.join } end def generate_diff_for(rep, snapshot) return if !self.site.config[:enable_output_diff] return if !File.file?(rep.raw_path(:snapshot => snapshot)) return if rep.binary? # Get old and new content old_content = File.read(rep.raw_path(:snapshot => snapshot)) new_content = rep.compiled_content(:snapshot => snapshot, :force => true) # Check whether there’s a different return if old_content == new_content @diff_threads << Thread.new do # Generate diff diff = diff_strings(old_content, new_content) diff.sub!(/^--- .*/, '--- ' + rep.raw_path) diff.sub!(/^\+\+\+ .*/, '+++ ' + rep.raw_path) # Write diff @diff_lock.synchronize do File.open('output.diff', 'a') { |io| io.write(diff) } end end end # TODO move this elsewhere def diff_strings(a, b) require 'open3' # Create files Tempfile.open('old') do |old_file| Tempfile.open('new') do |new_file| # Write files old_file.write(a) old_file.flush new_file.write(b) new_file.flush # Diff cmd = [ 'diff', '-u', old_file.path, new_file.path ] Open3.popen3(*cmd) do |stdin, stdout, stderr| result = stdout.read return (result == '' ? nil : result) end end end rescue Errno::ENOENT warn 'Failed to run `diff`, so no diff with the previously compiled ' \ 'content will be available.' nil end def start_filter_progress(rep, filter_name) # Only show progress on terminals return if !$stdout.tty? @progress_thread = Thread.new do delay = 1.0 step = 0 text = "Running #{filter_name} filter… ".make_compatible_with_env while !Thread.current[:stopped] sleep 0.1 # Wait for a while before showing text delay -= 0.1 next if delay > 0.05 # Print progress $stdout.print text + %w( | / - \\ )[step] + "\r" step = (step + 1) % 4 end # Clear text if delay < 0.05 $stdout.print ' ' * (text.length + 1 + 1) + "\r" end end end def stop_filter_progress(rep, filter_name) # Only show progress on terminals return if !$stdout.tty? @progress_thread[:stopped] = true end def print_profiling_feedback(reps) # Get max filter length max_filter_name_length = @filter_times.keys.map { |k| k.to_s.size }.max return if max_filter_name_length.nil? # Print warning if necessary if reps.any? { |r| !r.compiled? } $stderr.puts $stderr.puts "Warning: profiling information may not be accurate because " + "some items were not compiled." end # Print header puts puts ' ' * max_filter_name_length + ' | count min avg max tot' puts '-' * max_filter_name_length + '-+-----------------------------------' @filter_times.to_a.sort_by { |r| r[1] }.each do |row| # Extract data filter_name, samples = *row # Calculate stats count = samples.size min = samples.min tot = samples.inject { |memo, i| memo + i} avg = tot/count max = samples.max # Format stats count = format('%4d', count) min = format('%4.2f', min) avg = format('%4.2f', avg) max = format('%4.2f', max) tot = format('%5.2f', tot) # Output stats filter_name = format("%#{max_filter_name_length}s", filter_name) puts "#{filter_name} | #{count} #{min}s #{avg}s #{max}s #{tot}s" end end end end