master/lib/guard/watcher.rb

111 lines
3.5 KiB
Ruby
Raw Permalink Normal View History

2010-10-03 21:00:33 +00:00
module Guard
2011-09-20 10:54:28 +00:00
# The watcher defines a RegExp that will be matched against file system modifications.
2011-09-20 10:54:28 +00:00
# When a watcher matches a change, an optional action block is executed to enable
# processing the file system change result.
#
2010-10-03 21:00:33 +00:00
class Watcher
2011-09-20 10:54:28 +00:00
2010-10-03 21:00:33 +00:00
attr_accessor :pattern, :action
2011-09-20 10:54:28 +00:00
# Initialize a file watcher.
#
# @param [String, Regexp] pattern the pattern to be watched by the guard
# @param [Block] action the action to execute before passing the result to the Guard
#
2010-10-03 21:00:33 +00:00
def initialize(pattern, action = nil)
@pattern, @action = pattern, action
@@warning_printed ||= false
# deprecation warning
if @pattern.is_a?(String) && @pattern =~ /(^(\^))|(>?(\\\.)|(\.\*))|(\(.*\))|(\[.*\])|(\$$)/
unless @@warning_printed
2010-12-16 08:15:14 +00:00
UI.info "*"*20 + "\nDEPRECATION WARNING!\n" + "*"*20
2011-09-20 10:54:28 +00:00
UI.info <<-MSG
You have a string in your Guardfile watch patterns that seem to represent a Regexp.
Guard matches String with == and Regexp with Regexp#match.
You should either use plain String (without Regexp special characters) or real Regexp.
MSG
@@warning_printed = true
end
2011-09-20 10:54:28 +00:00
UI.info "\"#{@pattern}\" has been converted to #{ Regexp.new(@pattern).inspect }\n"
@pattern = Regexp.new(@pattern)
end
2010-10-03 21:00:33 +00:00
end
2011-09-20 10:54:28 +00:00
# Finds the files that matches a Guard.
#
# @param [Guard::Guard] guard the guard which watchers are used
# @param [Array<String>] files the changed files
# @return [Array<String>] the matched files
#
2010-10-03 21:00:33 +00:00
def self.match_files(guard, files)
guard.watchers.inject([]) do |paths, watcher|
files.each do |file|
if matches = watcher.match_file?(file)
2010-10-03 21:00:33 +00:00
if watcher.action
result = watcher.call_action(matches)
paths << Array(result) if result.respond_to?(:empty?) && !result.empty?
2010-10-03 21:00:33 +00:00
else
paths << matches[0]
end
end
end
2011-09-20 10:54:28 +00:00
paths.flatten.map { |p| p.to_s }
2010-10-03 21:00:33 +00:00
end
end
2011-09-20 10:54:28 +00:00
# Test if a file would be matched by any of the Guards watchers.
#
# @param [Array<Guard::Guard>] guards the guards to use the watchers from
# @param [Array<String>] files the files to test
# @return [Boolean] Whether a file matches
#
def self.match_files?(guards, files)
guards.any? do |guard|
guard.watchers.any? do |watcher|
files.any? { |file| watcher.match_file?(file) }
end
end
end
2011-09-20 10:54:28 +00:00
# Test the watchers pattern against a file.
#
# @param [String] file the file to test
# @return [Boolean] whether the given file is matched
#
def match_file?(file)
if @pattern.is_a?(Regexp)
file.match(@pattern)
else
file == @pattern ? [file] : nil
end
end
2011-09-20 10:54:28 +00:00
# Test if any of the files is the Guardfile.
#
# @param [Array<String>] the files to test
# @return [Boolean] whether one of these files is the Guardfile
#
def self.match_guardfile?(files)
2011-09-20 10:54:28 +00:00
files.any? { |file| "#{ Dir.pwd }/#{ file }" == Dsl.guardfile_path }
end
2011-09-20 10:54:28 +00:00
# Executes a watcher action.
#
# @param [String, MatchData] the matched path or the match from the Regex
# @return [String] the final paths
#
def call_action(matches)
begin
@action.arity > 0 ? @action.call(matches) : @action.call
rescue Exception => e
2011-09-20 10:54:28 +00:00
UI.error "Problem with watch action!\n#{ e.message }\n\n#{ e.backtrace.join("\n") }"
end
end
2010-10-03 21:00:33 +00:00
end
2011-05-24 01:36:51 +00:00
end