require 'rbconfig' require 'digest/sha1' module Guard autoload :Darwin, 'guard/listeners/darwin' autoload :Linux, 'guard/listeners/linux' autoload :Windows, 'guard/listeners/windows' autoload :Polling, 'guard/listeners/polling' class Listener attr_reader :directory def self.select_and_init(*a) if mac? && Darwin.usable? Darwin.new(*a) elsif linux? && Linux.usable? Linux.new(*a) elsif windows? && Windows.usable? Windows.new(*a) else UI.info "Using polling (Please help us to support your system better than that.)" Polling.new(*a) end end def initialize(directory=Dir.pwd, options={}) @directory = directory.to_s @sha1_checksums_hash = {} @file_timestamp_hash = {} @relativize_paths = options.fetch(:relativize_paths, true) @watch_deletions = options.fetch(:deletions, false) update_last_event end def start watch(@directory) # populate initial sha1 hash to watch for deleted or moved files all_files.each {|path| set_file_timestamp_hash(path, file_timestamp(path)) } if @watch_deletions end def stop end def on_change(&callback) @callback = callback end def update_last_event @last_event = Time.now end def modified_files(dirs, options={}) files = [] if @watch_deletions deleted_files = @file_timestamp_hash.collect do |path, ts| unless File.exists?(path) @sha1_checksums_hash.delete(path) @file_timestamp_hash.delete(path) "!#{path}" end end files.concat(deleted_files.compact) end files.concat(potentially_modified_files(dirs, options).select { |path| file_modified?(path) }) update_last_event relativize_paths(files) end def worker raise NotImplementedError, "should respond to #watch" end # register a directory to watch. must be implemented by the subclasses def watch(directory) raise NotImplementedError, "do whatever you want here, given the directory as only argument" end def all_files potentially_modified_files([@directory], :all => true) end # scopes all given paths to the current #directory def relativize_paths(paths) return paths unless relativize_paths? paths.map do |path| path.gsub(%r{#{@directory}/}, '') end end def relativize_paths? !!@relativize_paths end private def potentially_modified_files(dirs, options={}) paths = Dir.glob(dirs.map { |d| "#{d.sub(%r{/+$}, '')}/*" }, File::FNM_DOTMATCH).reject do |path| %w[. .. .bundle .git log tmp vendor].include?(File.basename(path)) end if options[:all] paths.inject([]) do |array, path| if File.file?(path) array << path else array += Dir.glob("#{path}/**/*", File::FNM_DOTMATCH).select { |p| File.file?(p) } end array end else paths.select { |path| File.file?(path) } end end # Depending on the filesystem, mtime is probably only precise to the second, so round # both values down to the second for the comparison. def file_modified?(path) if File.mtime(path).to_i == @last_event.to_i file_content_modified?(path, sha1_checksum(path)) elsif File.mtime(path).to_i > @last_event.to_i set_sha1_checksums_hash(path, sha1_checksum(path)) true elsif @watch_deletions ts = file_timestamp(path) if ts != @file_timestamp_hash[path] set_file_timestamp_hash(path, ts) true end end rescue false end def file_content_modified?(path, sha1_checksum) if @sha1_checksums_hash[path] != sha1_checksum set_sha1_checksums_hash(path, sha1_checksum) true else false end end def set_file_timestamp_hash(path, file_timestamp) @file_timestamp_hash[path] = file_timestamp end def set_sha1_checksums_hash(path, sha1_checksum) @sha1_checksums_hash[path] = sha1_checksum end def file_timestamp(path) File.mtime(path).to_i end def sha1_checksum(path) Digest::SHA1.file(path).to_s end def self.mac? RbConfig::CONFIG['target_os'] =~ /darwin/i end def self.linux? RbConfig::CONFIG['target_os'] =~ /linux/i end def self.windows? RbConfig::CONFIG['target_os'] =~ /mswin|mingw/i end end end