From c4ce612bde36c9bb8bfa318df00c0a96bd763ec6 Mon Sep 17 00:00:00 2001 From: monocle Date: Sun, 10 Apr 2011 16:08:43 -0700 Subject: [PATCH 1/8] Added hook/callback feature. --- lib/guard/hook.rb | 61 ++++++++++++++++++++++++++++++ spec/guard/hook_spec.rb | 82 +++++++++++++++++++++++++++++++++++++++++ 2 files changed, 143 insertions(+) create mode 100644 lib/guard/hook.rb create mode 100644 spec/guard/hook_spec.rb diff --git a/lib/guard/hook.rb b/lib/guard/hook.rb new file mode 100644 index 0000000..0fadc12 --- /dev/null +++ b/lib/guard/hook.rb @@ -0,0 +1,61 @@ +module Guard + module Hook + def self.included(base) + base.send :include, InstanceMethods + end + + module InstanceMethods + + # When passed a sybmol, #hook will generate a hook name + # from the symbol and calling method name. When passed + # a string, #hook will turn the string into a symbol + # directly. + def hook(event) + if event.class == Symbol + calling_method = caller[0][/`([^']*)'/, 1] + hook_name = "#{calling_method}_#{event}".to_sym + else + hook_name = event.to_sym + end + + UI.info "\nHook :#{hook_name} executed for #{self.class}" + Hook.notify(self.class, hook_name) + end + end + + class << self + def callbacks + @callbacks ||= Hash.new { |hash, key| hash[key] = [] } + end + + def add(listener, guard_class, events) + _events = events.class == Array ? events : [events] + _events.each do |event| + callbacks[[guard_class, event]] << listener + end + end + + def has_callback?(listener, guard_class, event) + listeners(guard_class, event).include? listener + end + + def notify(guard_class, event) + callbacks[[guard_class, event]].each do |listener| + listener.call(guard_class, event) + end + end + + def reset! + @callbacks = nil + end + + private + + def listeners(guard_class, event) + callbacks[[guard_class, event]].inject([]) do |all, arr| + all << arr + end + end + end + end +end diff --git a/spec/guard/hook_spec.rb b/spec/guard/hook_spec.rb new file mode 100644 index 0000000..f9b86ae --- /dev/null +++ b/spec/guard/hook_spec.rb @@ -0,0 +1,82 @@ +require "spec_helper" +require 'guard/guard' +require "guard/hook" + +describe Guard::Hook do + class Guard::Dummy < Guard::Guard + include Guard::Hook + + def run_all + hook :begin + hook :end + end + end + + let(:guard_class) { ::Guard::Dummy } + let(:listener) { double('listener').as_null_object } + + context "--module methods--" do + before { subject.add(listener, guard_class, :start_begin) } + + after { subject.reset! } + + describe ".add " do + it "can add a single callback" do + subject.has_callback?(listener, guard_class, :start_begin).should be_true + end + + it "can add multiple callbacks" do + subject.add(listener, guard_class, [:event1, :event2]) + subject.has_callback?(listener, guard_class, :event1).should be_true + subject.has_callback?(listener, guard_class, :event2).should be_true + end + end + + describe ".notify " do + it "sends :call to the given Guard class's callbacks" do + listener.should_receive(:call).with(guard_class, :start_begin) + subject.notify(guard_class, :start_begin) + end + + it "doesn't run callbacks of different types" do + listener2 = double('listener2') + subject.add(listener2, guard_class, :start_end) + listener2.should_not_receive(:call).with(guard_class, :start_end) + subject.notify(guard_class, :start_begin) + end + + it "doesn't run callbacks of the wrong class" do + guard2_class = double('Guard::Dummy2').class + subject.add(listener, guard2_class, :start_begin) + listener.should_not_receive(:call).with(guard2_class, :start_begin) + subject.notify(guard_class, :start_begin) + end + end + end + + describe "#hook " do + it "calls Guard::Hook.notify" do + guard = guard_class.new + Guard::Hook.should_receive(:notify). + with(guard_class, :run_all_begin) + + Guard::Hook.should_receive(:notify). + with(guard_class, :run_all_end) + + guard.run_all + end + + it "if passed a string parameter, will use that for the hook name" do + guard_class.class_eval do + def start + hook "my_hook" + end + end + + guard = guard_class.new + Guard::Hook.should_receive(:notify). + with(guard_class, :my_hook) + guard.start + end + end +end From 7b559ce255ac16590e473b1835afcf4649c2e057 Mon Sep 17 00:00:00 2001 From: monocle Date: Thu, 14 Apr 2011 13:31:34 -0700 Subject: [PATCH 2/8] Hooks - Add :begin and :end hook to all guard actions via Guard.supervised_task --- lib/guard.rb | 3 +++ lib/guard/hook.rb | 10 +--------- 2 files changed, 4 insertions(+), 9 deletions(-) diff --git a/lib/guard.rb b/lib/guard.rb index 38d90cc..56bdf5c 100644 --- a/lib/guard.rb +++ b/lib/guard.rb @@ -8,6 +8,7 @@ module Guard autoload :Listener, 'guard/listener' autoload :Watcher, 'guard/watcher' autoload :Notifier, 'guard/notifier' + autoload :Hook, 'guard/hook' class << self attr_accessor :options, :guards, :listener @@ -60,7 +61,9 @@ module Guard # Let a guard execute its task but # fire it if his work leads to a system failure def supervised_task(guard, task_to_supervise, *args) + guard.hook "#{task_to_supervise.to_s}_begin" guard.send(task_to_supervise, *args) + guard.hook "#{task_to_supervise.to_s}_end" rescue Exception UI.error("#{guard.class.name} guard failed to achieve its <#{task_to_supervise.to_s}> command: #{$!}") ::Guard.guards.delete guard diff --git a/lib/guard/hook.rb b/lib/guard/hook.rb index 0fadc12..6c62e5f 100644 --- a/lib/guard/hook.rb +++ b/lib/guard/hook.rb @@ -36,7 +36,7 @@ module Guard end def has_callback?(listener, guard_class, event) - listeners(guard_class, event).include? listener + callbacks[[guard_class, event]].include? listener end def notify(guard_class, event) @@ -48,14 +48,6 @@ module Guard def reset! @callbacks = nil end - - private - - def listeners(guard_class, event) - callbacks[[guard_class, event]].inject([]) do |all, arr| - all << arr - end - end end end end From 154ef207edae1a5060642c933bc333b47fbde1fb Mon Sep 17 00:00:00 2001 From: Nico Rieck Date: Thu, 24 Mar 2011 21:36:24 +0100 Subject: [PATCH 3/8] Use the correct ANSI escape code to reset SGR parameters. --- lib/guard/ui.rb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/guard/ui.rb b/lib/guard/ui.rb index 62c8de4..94953db 100644 --- a/lib/guard/ui.rb +++ b/lib/guard/ui.rb @@ -24,7 +24,7 @@ module Guard end def reset_line - print "\r\e " + print "\r\e[0m" end def clear From da0d059a43361cba2f558ab30be868af454628d4 Mon Sep 17 00:00:00 2001 From: Thibaud Guillaume-Gentil Date: Thu, 14 Apr 2011 21:38:30 +0200 Subject: [PATCH 4/8] Updated CHANGELOG --- CHANGELOG.rdoc | 3 +++ 1 file changed, 3 insertions(+) diff --git a/CHANGELOG.rdoc b/CHANGELOG.rdoc index 48844f7..a49ce21 100644 --- a/CHANGELOG.rdoc +++ b/CHANGELOG.rdoc @@ -1,8 +1,11 @@ +Features: + - Added a command line option (-n false) to disable notifications (growl/libnotify) Bugs fixes: - Return unique filenames from Linux listener (Marian Schubert) - Guard.get_guard_class return wrong class when loaded nested class. (koshigoe) - Fixed open-gem/gem_open dependency problem by using `gem which` to locate guards gem path + - Fixed an invalid ANSI escape code in UI.reset_line (gix) == 0.3.0 (Jan 19, 2011) From 1027e4b6b396f1b82b5ece000c385dee0cd3c333 Mon Sep 17 00:00:00 2001 From: Thibaud Guillaume-Gentil Date: Thu, 14 Apr 2011 21:39:29 +0200 Subject: [PATCH 5/8] Version 0.3.1 --- lib/guard/version.rb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/guard/version.rb b/lib/guard/version.rb index a3c09dc..2e9b5fb 100644 --- a/lib/guard/version.rb +++ b/lib/guard/version.rb @@ -1,3 +1,3 @@ module Guard - VERSION = "0.3.0" + VERSION = "0.3.1" end \ No newline at end of file From bbc63abd5e8475b08b549aebfbe9ec0080a43363 Mon Sep 17 00:00:00 2001 From: Thibaud Guillaume-Gentil Date: Thu, 14 Apr 2011 21:40:46 +0200 Subject: [PATCH 6/8] Updated CHANGELOG (0.3.1 tag) --- CHANGELOG.rdoc | 2 ++ 1 file changed, 2 insertions(+) diff --git a/CHANGELOG.rdoc b/CHANGELOG.rdoc index a49ce21..676fc87 100644 --- a/CHANGELOG.rdoc +++ b/CHANGELOG.rdoc @@ -1,3 +1,5 @@ +== 0.3.1 (Avril 14, 2011) + Features: - Added a command line option (-n false) to disable notifications (growl/libnotify) From 1d38c59200e5c93fb6d997228542335c5d894229 Mon Sep 17 00:00:00 2001 From: monocle Date: Thu, 14 Apr 2011 13:53:53 -0700 Subject: [PATCH 7/8] Fix preexisting tests after adding default hooks --- lib/guard.rb | 3 ++- lib/guard/guard.rb | 2 ++ spec/guard_spec.rb | 4 +++- 3 files changed, 7 insertions(+), 2 deletions(-) diff --git a/lib/guard.rb b/lib/guard.rb index 56bdf5c..29e235f 100644 --- a/lib/guard.rb +++ b/lib/guard.rb @@ -62,8 +62,9 @@ module Guard # fire it if his work leads to a system failure def supervised_task(guard, task_to_supervise, *args) guard.hook "#{task_to_supervise.to_s}_begin" - guard.send(task_to_supervise, *args) + result = guard.send(task_to_supervise, *args) guard.hook "#{task_to_supervise.to_s}_end" + result rescue Exception UI.error("#{guard.class.name} guard failed to achieve its <#{task_to_supervise.to_s}> command: #{$!}") ::Guard.guards.delete guard diff --git a/lib/guard/guard.rb b/lib/guard/guard.rb index a6b546c..e952fe3 100644 --- a/lib/guard/guard.rb +++ b/lib/guard/guard.rb @@ -1,5 +1,7 @@ module Guard class Guard + include Hook + attr_accessor :watchers, :options def initialize(watchers = [], options = {}) diff --git a/spec/guard_spec.rb b/spec/guard_spec.rb index dbd1c14..e1ec612 100644 --- a/spec/guard_spec.rb +++ b/spec/guard_spec.rb @@ -54,7 +54,7 @@ describe Guard do describe ".supervised_task" do subject { ::Guard.setup } before(:each) do - @g = mock(Guard::Guard) + @g = mock(Guard::Guard).as_null_object subject.guards.push(@g) end @@ -92,6 +92,8 @@ describe Guard do failing_result.message.should == 'I break your system' end end + + it "calls the default hooks" end end From 44aed3264c04478d9c1e03e1bf39d7bd4f93523c Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?R=C3=A9my=20Coutable?= Date: Sat, 16 Apr 2011 23:02:13 +0200 Subject: [PATCH 8/8] Added #callback DSL, modified Guard and Guard::Hook a bit in consequence. MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Signed-off-by: Rémy Coutable --- lib/guard.rb | 10 ++++--- lib/guard/dsl.rb | 24 +++++++++++----- lib/guard/hook.rb | 15 +++++----- spec/guard/dsl_spec.rb | 63 +++++++++++++++++++++++++++-------------- spec/guard/hook_spec.rb | 38 +++++++++++-------------- spec/guard_spec.rb | 15 ++++++++-- 6 files changed, 101 insertions(+), 64 deletions(-) diff --git a/lib/guard.rb b/lib/guard.rb index 29e235f..20687e3 100644 --- a/lib/guard.rb +++ b/lib/guard.rb @@ -61,12 +61,12 @@ module Guard # Let a guard execute its task but # fire it if his work leads to a system failure def supervised_task(guard, task_to_supervise, *args) - guard.hook "#{task_to_supervise.to_s}_begin" + guard.hook "#{task_to_supervise}_begin" result = guard.send(task_to_supervise, *args) - guard.hook "#{task_to_supervise.to_s}_end" + guard.hook "#{task_to_supervise}_end" result rescue Exception - UI.error("#{guard.class.name} guard failed to achieve its <#{task_to_supervise.to_s}> command: #{$!}") + UI.error("#{guard.class.name} guard failed to achieve its <#{task_to_supervise}> command: #{$!}") ::Guard.guards.delete guard UI.info("Guard #{guard.class.name} has just been fired") return $! @@ -82,9 +82,11 @@ module Guard listener.start end - def add_guard(name, watchers = [], options = {}) + def add_guard(name, watchers = [], callbacks = [], options = {}) guard_class = get_guard_class(name) + watchers = watchers.map { |watcher| ::Guard::Watcher.new(watcher[:pattern], watcher[:action]) } @guards << guard_class.new(watchers, options) + callbacks.each { |callback| ::Guard::Hook.add_callback(callback[:listener], guard_class, callback[:events]) } end def get_guard_class(name) diff --git a/lib/guard/dsl.rb b/lib/guard/dsl.rb index fbbb2ec..aee322d 100644 --- a/lib/guard/dsl.rb +++ b/lib/guard/dsl.rb @@ -4,7 +4,7 @@ module Guard class << self def evaluate_guardfile(options = {}) @@options = options - + if File.exists?(guardfile_path) begin new.instance_eval(File.read(guardfile_path), guardfile_path, 1) @@ -21,7 +21,7 @@ module Guard def guardfile_include?(guard_name) File.read(guardfile_path).match(/^guard\s*\(?\s*['":]#{guard_name}['"]?/) end - + def guardfile_path File.join(Dir.pwd, 'Guardfile') end @@ -31,14 +31,24 @@ module Guard guard_definition.call if guard_definition && (@@options[:group].empty? || @@options[:group].include?(name)) end - def guard(name, options = {}, &watch_definition) - @watchers = [] - watch_definition.call if watch_definition - ::Guard.add_guard(name, @watchers, options) + def guard(name, options = {}, &watch_and_callback_definition) + @watchers = [] + @callbacks = [] + watch_and_callback_definition.call if watch_and_callback_definition + ::Guard.add_guard(name, @watchers, @callbacks, options) end def watch(pattern, &action) - @watchers << ::Guard::Watcher.new(pattern, action) + @watchers << { :pattern => pattern, :action => action } + end + + def callback(*args, &listener) + listener, events = if args.size > 1 + args + else + [listener, args[0]] + end + @callbacks << { :events => events, :listener => listener } end end diff --git a/lib/guard/hook.rb b/lib/guard/hook.rb index 6c62e5f..99657bc 100644 --- a/lib/guard/hook.rb +++ b/lib/guard/hook.rb @@ -5,17 +5,16 @@ module Guard end module InstanceMethods - # When passed a sybmol, #hook will generate a hook name # from the symbol and calling method name. When passed # a string, #hook will turn the string into a symbol # directly. def hook(event) - if event.class == Symbol + hook_name = if event.is_a? Symbol calling_method = caller[0][/`([^']*)'/, 1] - hook_name = "#{calling_method}_#{event}".to_sym + "#{calling_method}_#{event}".to_sym else - hook_name = event.to_sym + event.to_sym end UI.info "\nHook :#{hook_name} executed for #{self.class}" @@ -28,15 +27,15 @@ module Guard @callbacks ||= Hash.new { |hash, key| hash[key] = [] } end - def add(listener, guard_class, events) - _events = events.class == Array ? events : [events] + def add_callback(listener, guard_class, events) + _events = events.is_a?(Array) ? events : [events] _events.each do |event| callbacks[[guard_class, event]] << listener end end def has_callback?(listener, guard_class, event) - callbacks[[guard_class, event]].include? listener + callbacks[[guard_class, event]].include?(listener) end def notify(guard_class, event) @@ -45,7 +44,7 @@ module Guard end end - def reset! + def reset_callbacks! @callbacks = nil end end diff --git a/spec/guard/dsl_spec.rb b/spec/guard/dsl_spec.rb index 097ff23..b1f6135 100644 --- a/spec/guard/dsl_spec.rb +++ b/spec/guard/dsl_spec.rb @@ -52,56 +52,75 @@ describe Guard::Dsl do end") end - it "should evaluates only the specified group" do - ::Guard.should_receive(:add_guard).with('test', anything, {}) - ::Guard.should_not_receive(:add_guard).with('another', anything, {}) + it "evaluates only the specified group" do + ::Guard.should_receive(:add_guard).with('test', anything, anything, {}) + ::Guard.should_not_receive(:add_guard).with('another', anything, anything, {}) subject.evaluate_guardfile(:group => ['x']) end - it "should evaluates only the specified groups" do - ::Guard.should_receive(:add_guard).with('test', anything, {}) - ::Guard.should_receive(:add_guard).with('another', anything, {}) + it "evaluates only the specified groups" do + ::Guard.should_receive(:add_guard).with('test', anything, anything, {}) + ::Guard.should_receive(:add_guard).with('another', anything, anything, {}) subject.evaluate_guardfile(:group => ['x', 'y']) end end describe "#guard" do - it "should load a guard specified as a string from the DSL" do + it "loads a guard specified by a string" do mock_guardfile_content("guard 'test'") - - ::Guard.should_receive(:add_guard).with('test', [], {}) + ::Guard.should_receive(:add_guard).with('test', [], [], {}) subject.evaluate_guardfile end - it "should load a guard specified as a symbol from the DSL" do + it "loads a guard specified as a symbol from the DSL" do mock_guardfile_content("guard :test") - - ::Guard.should_receive(:add_guard).with(:test, [], {}) + ::Guard.should_receive(:add_guard).with(:test, [], [], {}) subject.evaluate_guardfile end - it "should receive options when specified" do + it "accepts options" do mock_guardfile_content("guard 'test', :opt_a => 1, :opt_b => 'fancy'") - - ::Guard.should_receive(:add_guard).with('test', anything, { :opt_a => 1, :opt_b => 'fancy' }) + ::Guard.should_receive(:add_guard).with('test', anything, anything, { :opt_a => 1, :opt_b => 'fancy' }) subject.evaluate_guardfile end end describe "#watch" do - it "should receive watchers when specified" do + it "creates watchers for the guard" do mock_guardfile_content(" guard 'test' do watch('a') { 'b' } watch('c') end") - ::Guard.should_receive(:add_guard).with('test', anything, {}) do |name, watchers, options| - watchers.size.should == 2 - watchers[0].pattern.should == 'a' - watchers[0].action.call.should == proc { 'b' }.call - watchers[1].pattern.should == 'c' - watchers[1].action.should be_nil + ::Guard.should_receive(:add_guard).with('test', anything, anything, {}) do |name, watchers, callbacks, options| + watchers.should have(2).items + watchers[0][:pattern].should == 'a' + watchers[0][:action].call.should == proc { 'b' }.call + watchers[1][:pattern].should == 'c' + watchers[1][:action].should be_nil + end + subject.evaluate_guardfile + end + end + + describe "#callback" do + it "creates callbacks for the guard" do + class MyCustomCallback + end + + mock_guardfile_content(" + guard 'test' do + callback(:start_end) { 'Guard::Test started!' } + callback(MyCustomCallback, [:start_begin, :run_all_begin]) + end") + + ::Guard.should_receive(:add_guard).with('test', anything, anything, {}) do |name, watchers, callbacks, options| + callbacks.should have(2).items + callbacks[0][:events].should == :start_end + callbacks[0][:listener].call.should == proc { 'Guard::Test started!' }.call + callbacks[1][:events].should == [:start_begin, :run_all_begin] + callbacks[1][:listener].should == MyCustomCallback end subject.evaluate_guardfile end diff --git a/spec/guard/hook_spec.rb b/spec/guard/hook_spec.rb index f9b86ae..901057b 100644 --- a/spec/guard/hook_spec.rb +++ b/spec/guard/hook_spec.rb @@ -1,8 +1,8 @@ -require "spec_helper" -require 'guard/guard' -require "guard/hook" +require 'spec_helper' describe Guard::Hook do + subject { Guard::Hook } + class Guard::Dummy < Guard::Guard include Guard::Hook @@ -16,53 +16,49 @@ describe Guard::Hook do let(:listener) { double('listener').as_null_object } context "--module methods--" do - before { subject.add(listener, guard_class, :start_begin) } + before { subject.add_callback(listener, guard_class, :start_begin) } - after { subject.reset! } + after { subject.reset_callbacks! } - describe ".add " do + describe ".add_callback" do it "can add a single callback" do subject.has_callback?(listener, guard_class, :start_begin).should be_true end it "can add multiple callbacks" do - subject.add(listener, guard_class, [:event1, :event2]) + subject.add_callback(listener, guard_class, [:event1, :event2]) subject.has_callback?(listener, guard_class, :event1).should be_true subject.has_callback?(listener, guard_class, :event2).should be_true end end - describe ".notify " do + describe ".notify" do it "sends :call to the given Guard class's callbacks" do listener.should_receive(:call).with(guard_class, :start_begin) subject.notify(guard_class, :start_begin) end - it "doesn't run callbacks of different types" do + it "runs only the given callbacks" do listener2 = double('listener2') - subject.add(listener2, guard_class, :start_end) + subject.add_callback(listener2, guard_class, :start_end) listener2.should_not_receive(:call).with(guard_class, :start_end) subject.notify(guard_class, :start_begin) end - it "doesn't run callbacks of the wrong class" do + it "runs callbacks only for the guard given" do guard2_class = double('Guard::Dummy2').class - subject.add(listener, guard2_class, :start_begin) + subject.add_callback(listener, guard2_class, :start_begin) listener.should_not_receive(:call).with(guard2_class, :start_begin) subject.notify(guard_class, :start_begin) end end end - describe "#hook " do + describe "#hook" do it "calls Guard::Hook.notify" do guard = guard_class.new - Guard::Hook.should_receive(:notify). - with(guard_class, :run_all_begin) - - Guard::Hook.should_receive(:notify). - with(guard_class, :run_all_end) - + Guard::Hook.should_receive(:notify).with(guard_class, :run_all_begin) + Guard::Hook.should_receive(:notify).with(guard_class, :run_all_end) guard.run_all end @@ -74,9 +70,9 @@ describe Guard::Hook do end guard = guard_class.new - Guard::Hook.should_receive(:notify). - with(guard_class, :my_hook) + Guard::Hook.should_receive(:notify).with(guard_class, :my_hook) guard.start end end + end diff --git a/spec/guard_spec.rb b/spec/guard_spec.rb index e1ec612..88db168 100644 --- a/spec/guard_spec.rb +++ b/spec/guard_spec.rb @@ -76,6 +76,12 @@ describe Guard do ::Guard.supervised_task(@g, :regular).should be_true ::Guard.supervised_task(@g, :regular_with_arg, "given_path").should == "i'm a success" end + + it "calls the default hooks" do + @g.should_receive(:hook).with("regular_begin") + @g.should_receive(:hook).with("regular_end") + ::Guard.supervised_task(@g, :regular) + end end describe "tasks that raise an exception" do @@ -91,10 +97,15 @@ describe Guard do failing_result.should be_kind_of(Exception) failing_result.message.should == 'I break your system' end - end - it "calls the default hooks" + it "calls the default begin hook but not the default end hook" do + @g.should_receive(:hook).with("failing_begin") + @g.should_not_receive(:hook).with("failing_end") + ::Guard.supervised_task(@g, :failing) + end + end end + end end