From 0b945f10ec7013b8a857c06d1a9f451cc21b8dd7 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Re=CC=81my=20Coutable?= Date: Wed, 17 Aug 2011 00:26:03 +0200 Subject: [PATCH 01/13] Ensure that scoped groups and group name are symbolized before checking for inclusion. This is useful since Dsl.evaluate_guardfile can be called directly (hence, options are note passed by Thor, which stringify options). --- lib/guard/dsl.rb | 4 +- spec/guard/dsl_spec.rb | 117 ++++++++++++++++++++++++----------------- 2 files changed, 70 insertions(+), 51 deletions(-) diff --git a/lib/guard/dsl.rb b/lib/guard/dsl.rb index 1a6c739..45e51b2 100644 --- a/lib/guard/dsl.rb +++ b/lib/guard/dsl.rb @@ -106,13 +106,13 @@ module Guard def group(name, &guard_definition) @groups = @@options[:group] || [] - guard_definition.call if guard_definition && (@groups.empty? || @groups.include?(name.to_s)) + guard_definition.call if guard_definition && (@groups.empty? || @groups.map(&:to_sym).include?(name.to_sym)) end def guard(name, options = {}, &watch_definition) @watchers = [] watch_definition.call if watch_definition - ::Guard.add_guard(name, @watchers, options) + ::Guard.add_guard(name.to_sym, @watchers, @guard_options) end def watch(pattern, &action) diff --git a/spec/guard/dsl_spec.rb b/spec/guard/dsl_spec.rb index 740d8ed..86afac9 100644 --- a/spec/guard/dsl_spec.rb +++ b/spec/guard/dsl_spec.rb @@ -18,7 +18,7 @@ describe Guard::Dsl do subject.guardfile_contents.should == valid_guardfile_string end - it "should use a -command file over the default loc" do + it "should use a given file over the default loc" do fake_guardfile('/abc/Guardfile', "guard :foo") Guard::UI.should_not_receive(:error) @@ -66,22 +66,22 @@ describe Guard::Dsl do subject.evaluate_guardfile(:guardfile_contents => valid_guardfile_string) end - describe "it should correctly read data from its valid data source" do + describe "correctly reads data from its valid data source" do before(:each) { ::Guard::Dsl.stub!(:instance_eval_guardfile) } - it "should read correctly from a string" do + it "reads correctly from a string" do lambda { subject.evaluate_guardfile(:guardfile_contents => valid_guardfile_string) }.should_not raise_error subject.guardfile_contents.should == valid_guardfile_string end - it "should read correctly from a Guardfile" do + it "reads correctly from a Guardfile" do fake_guardfile('/abc/Guardfile', "guard :foo" ) lambda { subject.evaluate_guardfile(:guardfile => '/abc/Guardfile') }.should_not raise_error subject.guardfile_contents.should == "guard :foo" end - it "should read correctly from a Guardfile" do + it "reads correctly from a Guardfile" do fake_guardfile(File.join(Dir.pwd, 'Guardfile'), valid_guardfile_string) lambda { subject.evaluate_guardfile }.should_not raise_error @@ -89,10 +89,10 @@ describe Guard::Dsl do end end - describe "It should correctly throw errors when initializing with invalid data" do + describe "correctly throws errors when initializing with invalid data" do before(:each) { ::Guard::Dsl.stub!(:instance_eval_guardfile) } - it "should raise error when there's a problem reading a file" do + it "raises error when there's a problem reading a file" do File.stub!(:exist?).with('/def/Guardfile') { true } File.stub!(:read).with('/def/Guardfile') { raise Errno::EACCES.new("permission error") } @@ -100,14 +100,14 @@ describe Guard::Dsl do lambda { subject.evaluate_guardfile(:guardfile => '/def/Guardfile') }.should raise_error end - it "should raise error when -guardfile doesn't exist" do + it "raises error when given Guardfile doesn't exist" do File.stub!(:exist?).with('/def/Guardfile') { false } Guard::UI.should_receive(:error).with(/No Guardfile exists at/) lambda { subject.evaluate_guardfile(:guardfile => '/def/Guardfile') }.should raise_error end - it "should raise error when resorting to use default, finds no default" do + it "raises error when resorting to use default, finds no default" do File.stub!(:exist?).with(@local_guardfile_path) { false } File.stub!(:exist?).with(@home_guardfile_path) { false } @@ -115,12 +115,12 @@ describe Guard::Dsl do lambda { subject.evaluate_guardfile }.should raise_error end - it "should raise error when guardfile_content ends up empty or nil" do + it "raises error when guardfile_content ends up empty or nil" do Guard::UI.should_receive(:error).with(/The command file/) lambda { subject.evaluate_guardfile(:guardfile_contents => "") }.should raise_error end - it "should not raise error when guardfile_content is nil (skipped)" do + it "doesn't raise error when guardfile_content is nil (skipped)" do Guard::UI.should_not_receive(:error) lambda { subject.evaluate_guardfile(:guardfile_contents => nil) }.should_not raise_error end @@ -200,48 +200,74 @@ describe Guard::Dsl do end describe "#group" do - it "should evaluates only the specified string group" do - ::Guard.should_receive(:add_guard).with('test', anything, {}) - lambda { subject.evaluate_guardfile(:guardfile_contents => valid_guardfile_string, :group => ['w']) }.should_not raise_error + it "evaluates only the specified string group" do + ::Guard.should_receive(:add_guard).with(:pow, [], {}) + ::Guard.should_receive(:add_guard).with(:test, [], {}) + + subject.evaluate_guardfile(:guardfile_contents => valid_guardfile_string, :group => ['w']) end - it "should evaluates only the specified symbol group" do - ::Guard.should_receive(:add_guard).with('test', anything, {}) - lambda { subject.evaluate_guardfile(:guardfile_contents => valid_guardfile_string, :group => ['x']) }.should_not raise_error + + it "evaluates only the specified symbol group" do + ::Guard.should_receive(:add_guard).with(:pow, [], {}) + ::Guard.should_receive(:add_guard).with(:test, [], {}) + + subject.evaluate_guardfile(:guardfile_contents => valid_guardfile_string, :group => [:w]) 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, {}) - lambda { subject.evaluate_guardfile(:guardfile_contents => valid_guardfile_string, :group => ['x','y']) }.should_not raise_error + + it "evaluates only the specified groups" do + ::Guard.should_receive(:add_guard).with(:pow, [], {}) + ::Guard.should_receive(:add_guard).with(:rspec, [], {}) + ::Guard.should_receive(:add_guard).with(:ronn, [], {}) + ::Guard.should_receive(:add_guard).with(:less, [], {}) + + subject.evaluate_guardfile(:guardfile_contents => valid_guardfile_string, :group => [:x, :y]) end - it "should evaluate all groups when no group option is specified" do - ::Guard.should_receive(:add_guard).with('test', anything, {}).twice - ::Guard.should_receive(:add_guard).with('another', anything, {}).twice - lambda { subject.evaluate_guardfile(:guardfile_contents => valid_guardfile_string) }.should_not raise_error + + it "evaluates always guard outside any group (even when a group is given)" do + ::Guard.should_receive(:add_guard).with(:pow, [], {}) + ::Guard.should_receive(:add_guard).with(:test, [], {}) + + subject.evaluate_guardfile(:guardfile_contents => valid_guardfile_string, :group => [:w]) + end + + it "evaluates all groups when no group option is specified" do + ::Guard.should_receive(:add_guard).with(:pow, [], {}) + ::Guard.should_receive(:add_guard).with(:test, [], {}) + ::Guard.should_receive(:add_guard).with(:rspec, [], {}) + ::Guard.should_receive(:add_guard).with(:ronn, [], {}) + ::Guard.should_receive(:add_guard).with(:less, [], {}) + + subject.evaluate_guardfile(:guardfile_contents => valid_guardfile_string) end end - # TODO: not sure if each seperate quoting/call type needs its own test describe "#guard" do - it "should load a guard specified as a quoted string from the DSL" do - ::Guard.should_receive(:add_guard).with('test', [], {}) + it "loads a guard specified as a quoted string from the DSL" do + ::Guard.should_receive(:add_guard).with(:test, [], {}) subject.evaluate_guardfile(:guardfile_contents => "guard 'test'") end - it "should load a guard specified as a symbol from the DSL" do + it "loads a guard specified as a double quoted string from the DSL" do + ::Guard.should_receive(:add_guard).with(:test, [], {}) + + subject.evaluate_guardfile(:guardfile_contents => 'guard "test"') + end + + it "loads a guard specified as a symbol from the DSL" do ::Guard.should_receive(:add_guard).with(:test, [], {}) subject.evaluate_guardfile(:guardfile_contents => "guard :test") end - it "should load a guard specified as a symbol and called with parens from the DSL" do + it "loads a guard specified as a symbol and called with parens from the DSL" do ::Guard.should_receive(:add_guard).with(:test, [], {}) subject.evaluate_guardfile(:guardfile_contents => "guard(:test)") end - it "should receive options when specified" do - ::Guard.should_receive(:add_guard).with('test', anything, { :opt_a => 1, :opt_b => 'fancy' }) + it "receives options when specified, from normal arg" do + ::Guard.should_receive(:add_guard).with(:test, [], { :opt_a => 1, :opt_b => 'fancy' }) subject.evaluate_guardfile(:guardfile_contents => "guard 'test', :opt_a => 1, :opt_b => 'fancy'") end @@ -254,7 +280,7 @@ describe Guard::Dsl do watch('c') end" - ::Guard.should_receive(:add_guard).with('test', anything, {}) do |name, watchers, options| + ::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 @@ -273,29 +299,22 @@ private end def valid_guardfile_string - "group 'w' do - guard 'test' do - watch('c') - end + " + guard :pow + + group 'w' do + guard 'test' end group :x do - guard 'test' do - watch('c') - end + guard 'rspec' + guard :ronn end group 'y' do - guard 'another' do - watch('c') - end + guard 'less' end - - group 'z' do - guard 'another' do - watch('c') - end - end" + " end def mock_guardfile_content(content) From 24f0d21f71ca6869643ed4e5b6c92dc6995e6abe Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Re=CC=81my=20Coutable?= Date: Wed, 17 Aug 2011 00:27:02 +0200 Subject: [PATCH 02/13] Try to reduce specs execution time by providing an option for the "rest delay" before & after starting / stopping the listener. --- spec/guard/listeners/darwin_spec.rb | 4 ++-- spec/support/listener_helper.rb | 14 ++++++++------ 2 files changed, 10 insertions(+), 8 deletions(-) diff --git a/spec/guard/listeners/darwin_spec.rb b/spec/guard/listeners/darwin_spec.rb index 1b679bb..2c31edb 100644 --- a/spec/guard/listeners/darwin_spec.rb +++ b/spec/guard/listeners/darwin_spec.rb @@ -21,7 +21,7 @@ describe Guard::Darwin do subject.should be_usable end - it_should_behave_like "a listener that reacts to #on_change" - it_should_behave_like "a listener scoped to a specific directory" + it_should_behave_like "a listener that reacts to #on_change", 0.4 + it_should_behave_like "a listener scoped to a specific directory", 0.4 end end diff --git a/spec/support/listener_helper.rb b/spec/support/listener_helper.rb index 85c9d78..38388e5 100644 --- a/spec/support/listener_helper.rb +++ b/spec/support/listener_helper.rb @@ -1,10 +1,10 @@ private def start - sleep 1 + sleep(@rest_delay || 1) @listener.update_last_event Thread.new { @listener.start } - sleep 1 + sleep(@rest_delay || 1) end def record_results @@ -17,17 +17,18 @@ private end def stop - sleep 1 + sleep(@rest_delay || 1) @listener.stop - sleep 1 + sleep(@rest_delay || 1) end def results @results.flatten end -shared_examples_for 'a listener that reacts to #on_change' do +shared_examples_for 'a listener that reacts to #on_change' do |rest_delay| before(:each) do + @rest_delay = rest_delay @listener = described_class.new record_results end @@ -105,8 +106,9 @@ shared_examples_for 'a listener that reacts to #on_change' do end -shared_examples_for "a listener scoped to a specific directory" do +shared_examples_for "a listener scoped to a specific directory" do |rest_delay| before :each do + @rest_delay = rest_delay @wd = @fixture_path.join("folder1") @listener = described_class.new @wd end From e6dc507de763139e104b02cb4c38ed91811364a2 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Re=CC=81my=20Coutable?= Date: Wed, 17 Aug 2011 00:27:22 +0200 Subject: [PATCH 03/13] Rename manual source file --- man/{guard.ronn => guard.1.ronn} | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) rename man/{guard.ronn => guard.1.ronn} (99%) diff --git a/man/guard.ronn b/man/guard.1.ronn similarity index 99% rename from man/guard.ronn rename to man/guard.1.ronn index 412284a..af33537 100644 --- a/man/guard.ronn +++ b/man/guard.1.ronn @@ -59,7 +59,7 @@ List defined groups and guards for the current Guardfile. ### -h, help [COMMAND] List all of Guard's available commands. - + If is given, displays a specific help for . ## EXAMPLES From 6f7ce6feb78b535a3d87c84fcd2b5e0415367878 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Re=CC=81my=20Coutable?= Date: Wed, 17 Aug 2011 00:36:02 +0200 Subject: [PATCH 04/13] On a guard's initialization, pass the group in which it's been declared in the Guardfile, pass group's name on Guard.add_guard and thus in the Dsl --- lib/guard/dsl.rb | 10 ++++++++-- lib/guard/guard.rb | 5 +++-- spec/guard/dsl_spec.rb | 42 +++++++++++++++++++++--------------------- 3 files changed, 32 insertions(+), 25 deletions(-) diff --git a/lib/guard/dsl.rb b/lib/guard/dsl.rb index 45e51b2..57faefe 100644 --- a/lib/guard/dsl.rb +++ b/lib/guard/dsl.rb @@ -106,13 +106,19 @@ module Guard def group(name, &guard_definition) @groups = @@options[:group] || [] - guard_definition.call if guard_definition && (@groups.empty? || @groups.map(&:to_sym).include?(name.to_sym)) + name = name.to_sym + + if guard_definition && (@groups.empty? || @groups.map(&:to_sym).include?(name)) + @current_group = name + guard_definition.call + @current_group = nil + end end def guard(name, options = {}, &watch_definition) @watchers = [] watch_definition.call if watch_definition - ::Guard.add_guard(name.to_sym, @watchers, @guard_options) + ::Guard.add_guard(name.to_s.downcase.to_sym, @watchers, options, @current_group || :default) end def watch(pattern, &action) diff --git a/lib/guard/guard.rb b/lib/guard/guard.rb index a4d1488..b6b081a 100644 --- a/lib/guard/guard.rb +++ b/lib/guard/guard.rb @@ -1,10 +1,11 @@ module Guard class Guard - attr_accessor :watchers, :options + attr_accessor :watchers, :options, :group - def initialize(watchers = [], options = {}) + def initialize(watchers = [], options = {}, group = nil) @watchers, @options = watchers, options + @group ||= :default end # Guardfile template needed inside guard gem diff --git a/spec/guard/dsl_spec.rb b/spec/guard/dsl_spec.rb index 86afac9..6bbb166 100644 --- a/spec/guard/dsl_spec.rb +++ b/spec/guard/dsl_spec.rb @@ -201,41 +201,41 @@ describe Guard::Dsl do describe "#group" do it "evaluates only the specified string group" do - ::Guard.should_receive(:add_guard).with(:pow, [], {}) - ::Guard.should_receive(:add_guard).with(:test, [], {}) + ::Guard.should_receive(:add_guard).with(:pow, [], {}, :default) + ::Guard.should_receive(:add_guard).with(:test, [], {}, :w) subject.evaluate_guardfile(:guardfile_contents => valid_guardfile_string, :group => ['w']) end it "evaluates only the specified symbol group" do - ::Guard.should_receive(:add_guard).with(:pow, [], {}) - ::Guard.should_receive(:add_guard).with(:test, [], {}) + ::Guard.should_receive(:add_guard).with(:pow, [], {}, :default) + ::Guard.should_receive(:add_guard).with(:test, [], {}, :w) subject.evaluate_guardfile(:guardfile_contents => valid_guardfile_string, :group => [:w]) end it "evaluates only the specified groups" do - ::Guard.should_receive(:add_guard).with(:pow, [], {}) - ::Guard.should_receive(:add_guard).with(:rspec, [], {}) - ::Guard.should_receive(:add_guard).with(:ronn, [], {}) - ::Guard.should_receive(:add_guard).with(:less, [], {}) + ::Guard.should_receive(:add_guard).with(:pow, [], {}, :default) + ::Guard.should_receive(:add_guard).with(:rspec, [], {}, :x) + ::Guard.should_receive(:add_guard).with(:ronn, [], {}, :x) + ::Guard.should_receive(:add_guard).with(:less, [], {}, :y) subject.evaluate_guardfile(:guardfile_contents => valid_guardfile_string, :group => [:x, :y]) end it "evaluates always guard outside any group (even when a group is given)" do - ::Guard.should_receive(:add_guard).with(:pow, [], {}) - ::Guard.should_receive(:add_guard).with(:test, [], {}) + ::Guard.should_receive(:add_guard).with(:pow, [], {}, :default) + ::Guard.should_receive(:add_guard).with(:test, [], {}, :w) subject.evaluate_guardfile(:guardfile_contents => valid_guardfile_string, :group => [:w]) end it "evaluates all groups when no group option is specified" do - ::Guard.should_receive(:add_guard).with(:pow, [], {}) - ::Guard.should_receive(:add_guard).with(:test, [], {}) - ::Guard.should_receive(:add_guard).with(:rspec, [], {}) - ::Guard.should_receive(:add_guard).with(:ronn, [], {}) - ::Guard.should_receive(:add_guard).with(:less, [], {}) + ::Guard.should_receive(:add_guard).with(:pow, [], {}, :default) + ::Guard.should_receive(:add_guard).with(:test, [], {}, :w) + ::Guard.should_receive(:add_guard).with(:rspec, [], {}, :x) + ::Guard.should_receive(:add_guard).with(:ronn, [], {}, :x) + ::Guard.should_receive(:add_guard).with(:less, [], {}, :y) subject.evaluate_guardfile(:guardfile_contents => valid_guardfile_string) end @@ -243,31 +243,31 @@ describe Guard::Dsl do describe "#guard" do it "loads a guard specified as a quoted string from the DSL" do - ::Guard.should_receive(:add_guard).with(:test, [], {}) + ::Guard.should_receive(:add_guard).with(:test, [], {}, :default) subject.evaluate_guardfile(:guardfile_contents => "guard 'test'") end it "loads a guard specified as a double quoted string from the DSL" do - ::Guard.should_receive(:add_guard).with(:test, [], {}) + ::Guard.should_receive(:add_guard).with(:test, [], {}, :default) subject.evaluate_guardfile(:guardfile_contents => 'guard "test"') end it "loads a guard specified as a symbol from the DSL" do - ::Guard.should_receive(:add_guard).with(:test, [], {}) + ::Guard.should_receive(:add_guard).with(:test, [], {}, :default) subject.evaluate_guardfile(:guardfile_contents => "guard :test") end it "loads a guard specified as a symbol and called with parens from the DSL" do - ::Guard.should_receive(:add_guard).with(:test, [], {}) + ::Guard.should_receive(:add_guard).with(:test, [], {}, :default) subject.evaluate_guardfile(:guardfile_contents => "guard(:test)") end it "receives options when specified, from normal arg" do - ::Guard.should_receive(:add_guard).with(:test, [], { :opt_a => 1, :opt_b => 'fancy' }) + ::Guard.should_receive(:add_guard).with(:test, [], { :opt_a => 1, :opt_b => 'fancy' }, :default) subject.evaluate_guardfile(:guardfile_contents => "guard 'test', :opt_a => 1, :opt_b => 'fancy'") end @@ -280,7 +280,7 @@ describe Guard::Dsl do watch('c') end" - ::Guard.should_receive(:add_guard).with(:test, anything, {}) do |name, watchers, options| + ::Guard.should_receive(:add_guard).with(:test, anything, {}, :default) do |name, watchers, options| watchers.size.should == 2 watchers[0].pattern.should == 'a' watchers[0].action.call.should == proc { 'b' }.call From dd864021090a3e779cb12ed37cce60f94c153bd9 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Re=CC=81my=20Coutable?= Date: Wed, 17 Aug 2011 00:39:45 +0200 Subject: [PATCH 05/13] Add @groups attributes to keep track of the groups, ordered as in the Guardfile --- lib/guard.rb | 19 +++++---- spec/guard_spec.rb | 100 +++++++++++++++++++++++++++++++++++++++++++-- 2 files changed, 109 insertions(+), 10 deletions(-) diff --git a/lib/guard.rb b/lib/guard.rb index 72a8518..4fc8407 100644 --- a/lib/guard.rb +++ b/lib/guard.rb @@ -9,12 +9,13 @@ module Guard autoload :Notifier, 'guard/notifier' class << self - attr_accessor :options, :guards, :listener + attr_accessor :options, :guards, :groups, :listener # initialize this singleton def setup(options = {}) @options = options @listener = Listener.select_and_init(@options[:watchdir] ? File.expand_path(@options[:watchdir]) : Dir.pwd) + @groups = [:default] @guards = [] @options[:notify] && ENV["GUARD_NOTIFY"] != 'false' ? Notifier.turn_on : Notifier.turn_off @@ -81,21 +82,25 @@ module Guard listener.start end - def add_guard(name, watchers = [], options = {}) - if name.downcase == 'ego' + def add_guard(name, watchers = [], options = {}, group = nil) + if name.to_sym == :ego UI.deprecation("Guard::Ego is now part of Guard. You can remove it from your Guardfile.") else - guard_class = get_guard_class(name) - @guards << guard_class.new(watchers, options) + guard = get_guard_class(name).new(watchers, options, group) + @guards << guard end end + def add_group(name) + @groups << name.to_sym unless name.nil? + end + def get_guard_class(name) try_require = false const_name = name.to_s.downcase.gsub('-', '') begin - require "guard/#{name.to_s.downcase}" if try_require - self.const_get(self.constants.find {|c| c.to_s.downcase == const_name }) + require "guard/#{name.downcase}" if try_require + self.const_get(self.constants.find { |c| c.to_s.downcase == const_name }) rescue TypeError unless try_require try_require = true diff --git a/spec/guard_spec.rb b/spec/guard_spec.rb index d85b8fb..461c157 100644 --- a/spec/guard_spec.rb +++ b/spec/guard_spec.rb @@ -10,13 +10,17 @@ describe Guard do subject.should be ::Guard end - it "initializes the Guards" do - ::Guard.guards.should be_kind_of(Array) + it "initializes @guards" do + subject.guards.should eql [] + end + + it "initializes @groups" do + Guard.groups.should eql [:default] end it "initializes the options" do opts = { :my_opts => true } - ::Guard.setup(opts).options.should include(:my_opts) + Guard.setup(opts).options.should include(:my_opts) end it "initializes the listener" do @@ -62,6 +66,96 @@ describe Guard do end end + describe ".add_guard" do + before(:each) do + @guard_rspec_class = double('Guard::RSpec') + @guard_rspec = double('Guard::RSpec') + + Guard.stub!(:get_guard_class) { @guard_rspec_class } + + Guard.setup + end + + it "accepts guard name as string" do + @guard_rspec_class.should_receive(:new).and_return(@guard_rspec) + + Guard.add_guard('rspec') + end + + it "accepts guard name as symbol" do + @guard_rspec_class.should_receive(:new).and_return(@guard_rspec) + + Guard.add_guard(:rspec) + end + + it "adds guard to the @guards array" do + @guard_rspec_class.should_receive(:new).and_return(@guard_rspec) + + Guard.add_guard(:rspec) + + Guard.guards.should eql [@guard_rspec] + end + + context "with no watchers given" do + it "gives an empty array of watchers" do + @guard_rspec_class.should_receive(:new).with([], {}, nil).and_return(@guard_rspec) + + Guard.add_guard(:rspec, []) + end + end + + context "with watchers given" do + it "give the watchers array" do + @guard_rspec_class.should_receive(:new).with([:foo], {}, nil).and_return(@guard_rspec) + + Guard.add_guard(:rspec, [:foo]) + end + end + + context "with no options given" do + it "gives an empty hash of options" do + @guard_rspec_class.should_receive(:new).with([], {}, nil).and_return(@guard_rspec) + + Guard.add_guard(:rspec, [], {}) + end + end + + context "with options given" do + it "give the options hash" do + @guard_rspec_class.should_receive(:new).with([], { :foo => true }, nil).and_return(@guard_rspec) + + Guard.add_guard(:rspec, [], { :foo => true }) + end + end + + context "with the group :backend given" do + it "initialize the guard and pass it its group" do + @guard_rspec_class.should_receive(:new).with([], {}, :backend).and_return(@guard_rspec) + + Guard.add_guard(:rspec, [], {}, :backend) + end + end + end + + + describe ".add_group" do + before(:each) do + Guard.setup + end + + it "accepts group name as string" do + Guard.add_group('backend') + + Guard.groups.should eql [:default, :backend] + end + + it "accepts group name as symbol" do + Guard.add_group(:backend) + + Guard.groups.should eql [:default, :backend] + end + end + describe ".get_guard_class" do after do [:Classname, :DashedClassName, :Inline].each do |const| From 563b020d4f143bb3139f57455b9fbed49c82708b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Re=CC=81my=20Coutable?= Date: Wed, 17 Aug 2011 01:36:44 +0200 Subject: [PATCH 06/13] Oops (worked on 1.9.2 only) :) --- lib/guard.rb | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/lib/guard.rb b/lib/guard.rb index 4fc8407..761bc67 100644 --- a/lib/guard.rb +++ b/lib/guard.rb @@ -96,8 +96,9 @@ module Guard end def get_guard_class(name) + name = name.to_s try_require = false - const_name = name.to_s.downcase.gsub('-', '') + const_name = name.downcase.gsub('-', '') begin require "guard/#{name.downcase}" if try_require self.const_get(self.constants.find { |c| c.to_s.downcase == const_name }) From dc2ab97d234a5cf85a8cdcdd61614e645addc2ca Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Re=CC=81my=20Coutable?= Date: Wed, 17 Aug 2011 10:04:42 +0200 Subject: [PATCH 07/13] Pass the group in the options hash instead of a new parameter in the Guard::Guard.initialize method! --- lib/guard.rb | 4 ++-- lib/guard/dsl.rb | 3 ++- lib/guard/guard.rb | 4 ++-- spec/guard/dsl_spec.rb | 42 +++++++++++++++++++++--------------------- spec/guard_spec.rb | 24 ++++++++---------------- 5 files changed, 35 insertions(+), 42 deletions(-) diff --git a/lib/guard.rb b/lib/guard.rb index 761bc67..701d428 100644 --- a/lib/guard.rb +++ b/lib/guard.rb @@ -82,11 +82,11 @@ module Guard listener.start end - def add_guard(name, watchers = [], options = {}, group = nil) + def add_guard(name, watchers = [], options = {}) if name.to_sym == :ego UI.deprecation("Guard::Ego is now part of Guard. You can remove it from your Guardfile.") else - guard = get_guard_class(name).new(watchers, options, group) + guard = get_guard_class(name).new(watchers, options) @guards << guard end end diff --git a/lib/guard/dsl.rb b/lib/guard/dsl.rb index 57faefe..cba1b20 100644 --- a/lib/guard/dsl.rb +++ b/lib/guard/dsl.rb @@ -118,7 +118,8 @@ module Guard def guard(name, options = {}, &watch_definition) @watchers = [] watch_definition.call if watch_definition - ::Guard.add_guard(name.to_s.downcase.to_sym, @watchers, options, @current_group || :default) + options.update(:group => (@current_group || :default)) + ::Guard.add_guard(name.to_s.downcase.to_sym, @watchers, options) end def watch(pattern, &action) diff --git a/lib/guard/guard.rb b/lib/guard/guard.rb index b6b081a..8009903 100644 --- a/lib/guard/guard.rb +++ b/lib/guard/guard.rb @@ -3,9 +3,9 @@ module Guard attr_accessor :watchers, :options, :group - def initialize(watchers = [], options = {}, group = nil) + def initialize(watchers = [], options = {}) + @group = options.delete(:group) || :default @watchers, @options = watchers, options - @group ||= :default end # Guardfile template needed inside guard gem diff --git a/spec/guard/dsl_spec.rb b/spec/guard/dsl_spec.rb index 6bbb166..1de89aa 100644 --- a/spec/guard/dsl_spec.rb +++ b/spec/guard/dsl_spec.rb @@ -201,41 +201,41 @@ describe Guard::Dsl do describe "#group" do it "evaluates only the specified string group" do - ::Guard.should_receive(:add_guard).with(:pow, [], {}, :default) - ::Guard.should_receive(:add_guard).with(:test, [], {}, :w) + ::Guard.should_receive(:add_guard).with(:pow, [], { :group => :default }) + ::Guard.should_receive(:add_guard).with(:test, [], { :group => :w }) subject.evaluate_guardfile(:guardfile_contents => valid_guardfile_string, :group => ['w']) end it "evaluates only the specified symbol group" do - ::Guard.should_receive(:add_guard).with(:pow, [], {}, :default) - ::Guard.should_receive(:add_guard).with(:test, [], {}, :w) + ::Guard.should_receive(:add_guard).with(:pow, [], { :group => :default }) + ::Guard.should_receive(:add_guard).with(:test, [], { :group => :w }) subject.evaluate_guardfile(:guardfile_contents => valid_guardfile_string, :group => [:w]) end it "evaluates only the specified groups" do - ::Guard.should_receive(:add_guard).with(:pow, [], {}, :default) - ::Guard.should_receive(:add_guard).with(:rspec, [], {}, :x) - ::Guard.should_receive(:add_guard).with(:ronn, [], {}, :x) - ::Guard.should_receive(:add_guard).with(:less, [], {}, :y) + ::Guard.should_receive(:add_guard).with(:pow, [], { :group => :default }) + ::Guard.should_receive(:add_guard).with(:rspec, [], { :group => :x }) + ::Guard.should_receive(:add_guard).with(:ronn, [], { :group => :x }) + ::Guard.should_receive(:add_guard).with(:less, [], { :group => :y }) subject.evaluate_guardfile(:guardfile_contents => valid_guardfile_string, :group => [:x, :y]) end it "evaluates always guard outside any group (even when a group is given)" do - ::Guard.should_receive(:add_guard).with(:pow, [], {}, :default) - ::Guard.should_receive(:add_guard).with(:test, [], {}, :w) + ::Guard.should_receive(:add_guard).with(:pow, [], { :group => :default }) + ::Guard.should_receive(:add_guard).with(:test, [], { :group => :w }) subject.evaluate_guardfile(:guardfile_contents => valid_guardfile_string, :group => [:w]) end it "evaluates all groups when no group option is specified" do - ::Guard.should_receive(:add_guard).with(:pow, [], {}, :default) - ::Guard.should_receive(:add_guard).with(:test, [], {}, :w) - ::Guard.should_receive(:add_guard).with(:rspec, [], {}, :x) - ::Guard.should_receive(:add_guard).with(:ronn, [], {}, :x) - ::Guard.should_receive(:add_guard).with(:less, [], {}, :y) + ::Guard.should_receive(:add_guard).with(:pow, [], { :group => :default }) + ::Guard.should_receive(:add_guard).with(:test, [], { :group => :w }) + ::Guard.should_receive(:add_guard).with(:rspec, [], { :group => :x }) + ::Guard.should_receive(:add_guard).with(:ronn, [], { :group => :x }) + ::Guard.should_receive(:add_guard).with(:less, [], { :group => :y }) subject.evaluate_guardfile(:guardfile_contents => valid_guardfile_string) end @@ -243,31 +243,31 @@ describe Guard::Dsl do describe "#guard" do it "loads a guard specified as a quoted string from the DSL" do - ::Guard.should_receive(:add_guard).with(:test, [], {}, :default) + ::Guard.should_receive(:add_guard).with(:test, [], { :group => :default }) subject.evaluate_guardfile(:guardfile_contents => "guard 'test'") end it "loads a guard specified as a double quoted string from the DSL" do - ::Guard.should_receive(:add_guard).with(:test, [], {}, :default) + ::Guard.should_receive(:add_guard).with(:test, [], { :group => :default }) subject.evaluate_guardfile(:guardfile_contents => 'guard "test"') end it "loads a guard specified as a symbol from the DSL" do - ::Guard.should_receive(:add_guard).with(:test, [], {}, :default) + ::Guard.should_receive(:add_guard).with(:test, [], { :group => :default }) subject.evaluate_guardfile(:guardfile_contents => "guard :test") end it "loads a guard specified as a symbol and called with parens from the DSL" do - ::Guard.should_receive(:add_guard).with(:test, [], {}, :default) + ::Guard.should_receive(:add_guard).with(:test, [], { :group => :default }) subject.evaluate_guardfile(:guardfile_contents => "guard(:test)") end it "receives options when specified, from normal arg" do - ::Guard.should_receive(:add_guard).with(:test, [], { :opt_a => 1, :opt_b => 'fancy' }, :default) + ::Guard.should_receive(:add_guard).with(:test, [], { :opt_a => 1, :opt_b => 'fancy', :group => :default }) subject.evaluate_guardfile(:guardfile_contents => "guard 'test', :opt_a => 1, :opt_b => 'fancy'") end @@ -280,7 +280,7 @@ describe Guard::Dsl do watch('c') end" - ::Guard.should_receive(:add_guard).with(:test, anything, {}, :default) do |name, watchers, options| + ::Guard.should_receive(:add_guard).with(:test, anything, { :group => :default }) do |name, watchers, options| watchers.size.should == 2 watchers[0].pattern.should == 'a' watchers[0].action.call.should == proc { 'b' }.call diff --git a/spec/guard_spec.rb b/spec/guard_spec.rb index 461c157..bc62280 100644 --- a/spec/guard_spec.rb +++ b/spec/guard_spec.rb @@ -98,7 +98,7 @@ describe Guard do context "with no watchers given" do it "gives an empty array of watchers" do - @guard_rspec_class.should_receive(:new).with([], {}, nil).and_return(@guard_rspec) + @guard_rspec_class.should_receive(:new).with([], {}).and_return(@guard_rspec) Guard.add_guard(:rspec, []) end @@ -106,7 +106,7 @@ describe Guard do context "with watchers given" do it "give the watchers array" do - @guard_rspec_class.should_receive(:new).with([:foo], {}, nil).and_return(@guard_rspec) + @guard_rspec_class.should_receive(:new).with([:foo], {}).and_return(@guard_rspec) Guard.add_guard(:rspec, [:foo]) end @@ -114,7 +114,7 @@ describe Guard do context "with no options given" do it "gives an empty hash of options" do - @guard_rspec_class.should_receive(:new).with([], {}, nil).and_return(@guard_rspec) + @guard_rspec_class.should_receive(:new).with([], {}).and_return(@guard_rspec) Guard.add_guard(:rspec, [], {}) end @@ -122,17 +122,9 @@ describe Guard do context "with options given" do it "give the options hash" do - @guard_rspec_class.should_receive(:new).with([], { :foo => true }, nil).and_return(@guard_rspec) + @guard_rspec_class.should_receive(:new).with([], { :foo => true, :group => :backend }).and_return(@guard_rspec) - Guard.add_guard(:rspec, [], { :foo => true }) - end - end - - context "with the group :backend given" do - it "initialize the guard and pass it its group" do - @guard_rspec_class.should_receive(:new).with([], {}, :backend).and_return(@guard_rspec) - - Guard.add_guard(:rspec, [], {}, :backend) + Guard.add_guard(:rspec, [], { :foo => true, :group => :backend }) end end end @@ -145,17 +137,17 @@ describe Guard do it "accepts group name as string" do Guard.add_group('backend') - + Guard.groups.should eql [:default, :backend] end it "accepts group name as symbol" do Guard.add_group(:backend) - + Guard.groups.should eql [:default, :backend] end end - + describe ".get_guard_class" do after do [:Classname, :DashedClassName, :Inline].each do |const| From 9710229f7cc36c1f6db585635156344241db1f3b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Re=CC=81my=20Coutable?= Date: Wed, 17 Aug 2011 10:45:54 +0200 Subject: [PATCH 08/13] Use doc formatter again (instead of Fuubar) --- Gemfile | 4 ---- Guardfile | 2 +- 2 files changed, 1 insertion(+), 5 deletions(-) diff --git a/Gemfile b/Gemfile index fe4b33b..37eab73 100644 --- a/Gemfile +++ b/Gemfile @@ -8,10 +8,6 @@ group :guard do gem 'guard-ronn' end -group :test do - gem 'fuubar' -end - require 'rbconfig' if RbConfig::CONFIG['target_os'] =~ /darwin/i diff --git a/Guardfile b/Guardfile index 106cadf..39cf104 100644 --- a/Guardfile +++ b/Guardfile @@ -1,4 +1,4 @@ -guard :rspec, :version => 2, :keep_failed => false, :cli => '--format Fuubar' do +guard :rspec, :version => 2, :keep_failed => false, :cli => '-f doc' do watch(%r{^spec/.+_spec\.rb$}) watch(%r{^lib/(.+)\.rb$}) { |m| "spec/#{m[1]}_spec.rb" } watch('spec/spec_helper.rb') { "spec" } From 0e1564ef3b5e976ecdb8c2533896720bd1377abe Mon Sep 17 00:00:00 2001 From: Michael Kessler Date: Wed, 17 Aug 2011 14:27:40 +0200 Subject: [PATCH 09/13] Revert "remove growl support completely" This reverts commit 7f87411189ff2c3fac869a4d3ff1987e6e4e8b26. --- Gemfile | 2 +- README.md | 11 +++++-- lib/guard/notifier.rb | 30 +++++++++++++------ spec/guard/notifier_spec.rb | 57 ++++++++++++++++++++++++++++++++++++- 4 files changed, 86 insertions(+), 14 deletions(-) diff --git a/Gemfile b/Gemfile index 37eab73..5aa0145 100644 --- a/Gemfile +++ b/Gemfile @@ -12,7 +12,7 @@ require 'rbconfig' if RbConfig::CONFIG['target_os'] =~ /darwin/i gem 'rb-fsevent', '>= 0.4.0', :require => false - gem 'growl_notify', :require => false + gem 'growl', '~> 1.0.3', :require => false end if RbConfig::CONFIG['target_os'] =~ /linux/i gem 'rb-inotify', '>= 0.8.5', :require => false diff --git a/README.md b/README.md index 06bd4a0..5b2b622 100644 --- a/README.md +++ b/README.md @@ -55,19 +55,24 @@ Install the rb-fsevent gem for [FSEvent](http://en.wikipedia.org/wiki/FSEvents) $ gem install rb-fsevent ``` -Install the growl_notify gem if you want notification support: +Install either the growl_notify or the Growl gem if you want notification support: ``` bash $ gem install growl_notify +$ # or +$ gem install growl ``` -And add it to your Gemfile: +And add them to your Gemfile: ``` ruby gem 'rb-fsevent' -gem 'growl_notify' +gem 'growl' ``` +growl_notify uses AppleScript, the suggested method for interfacing with Growl, +rather than the `growlnotify` command to display messages. + ### On Linux Install the rb-inotify gem for [inotify](http://en.wikipedia.org/wiki/Inotify) support: diff --git a/lib/guard/notifier.rb b/lib/guard/notifier.rb index b3ae3b2..e960301 100644 --- a/lib/guard/notifier.rb +++ b/lib/guard/notifier.rb @@ -47,10 +47,18 @@ module Guard def self.notify_mac(title, message, image, options) require_growl # need for guard-rspec formatter that is called out of guard scope - options = { :description => message, :title => title, :icon => image_path(image), :application_name => APPLICATION_NAME }.merge(options) - options.delete(:name) + default_options = { :title => title, :icon => image_path(image), :name => APPLICATION_NAME } + default_options.merge!(options) - GrowlNotify.send_notification(options) if enabled? + if defined?(GrowlNotify) + default_options[:description] = message + default_options[:application_name] = APPLICATION_NAME + default_options.delete(:name) + + GrowlNotify.send_notification(default_options) if enabled? + else + Growl.notify message, default_options.merge(options) if enabled? + end end def self.notify_linux(title, message, image, options) @@ -94,17 +102,21 @@ module Guard end def self.require_growl - require 'growl_notify' + begin + require 'growl_notify' - if GrowlNotify.application_name != APPLICATION_NAME - GrowlNotify.config do |c| - c.notifications = c.default_notifications = [ APPLICATION_NAME ] - c.application_name = c.notifications.first + if GrowlNotify.application_name != APPLICATION_NAME + GrowlNotify.config do |c| + c.notifications = c.default_notifications = [ APPLICATION_NAME ] + c.application_name = c.notifications.first + end end + rescue LoadError + require 'growl' end rescue LoadError turn_off - UI.info "Please install growl_notify gem for Mac OS X notification support and add it to your Gemfile" + UI.info "Please install growl or growl_notify gem for Mac OS X notification support and add it to your Gemfile" end def self.require_libnotify diff --git a/spec/guard/notifier_spec.rb b/spec/guard/notifier_spec.rb index 2d4858f..fba4145 100644 --- a/spec/guard/notifier_spec.rb +++ b/spec/guard/notifier_spec.rb @@ -39,9 +39,19 @@ describe Guard::Notifier do end end - context "without the GrowlNofity library available" do + context "with the Growl library available" do + it "loads the library and enables the notifications" do + subject.should_receive(:require).with('growl_notify').and_raise LoadError + subject.should_receive(:require).with('growl').and_return true + subject.turn_on + subject.should be_enabled + end + end + + context "without the Growl library available" do it "disables the notifications" do subject.should_receive(:require).with('growl_notify').and_raise LoadError + subject.should_receive(:require).with('growl').and_raise LoadError subject.turn_on subject.should_not be_enabled end @@ -102,6 +112,51 @@ describe Guard::Notifier do subject.stub(:require_growl) end + context 'with growl gem' do + before do + Object.send(:remove_const, :Growl) if defined?(Growl) + Growl = Object.new + end + + after do + Object.send(:remove_const, :Growl) + end + + it "passes the notification to Growl" do + Growl.should_receive(:notify).with("great", + :title => "Guard", + :icon => Pathname.new(File.dirname(__FILE__)).join('../../images/success.png').to_s, + :name => "Guard" + ) + subject.notify 'great', :title => 'Guard' + end + + it "don't passes the notification to Growl if library is not available" do + Growl.should_not_receive(:notify) + subject.should_receive(:enabled?).and_return(true, false) + subject.notify 'great', :title => 'Guard' + end + + it "allows additional notification options" do + Growl.should_receive(:notify).with("great", + :title => "Guard", + :icon => Pathname.new(File.dirname(__FILE__)).join('../../images/success.png').to_s, + :name => "Guard", + :priority => 1 + ) + subject.notify 'great', :title => 'Guard', :priority => 1 + end + + it "allows to overwrite a default notification option" do + Growl.should_receive(:notify).with("great", + :title => "Guard", + :icon => Pathname.new(File.dirname(__FILE__)).join('../../images/success.png').to_s, + :name => "Guard-Cucumber" + ) + subject.notify 'great', :title => 'Guard', :name => "Guard-Cucumber" + end + end + context 'with growl_notify gem' do before do Object.send(:remove_const, :GrowlNotify) if defined?(GrowlNotify) From 1fd6f2d7beddc0084a9d3a7eebd275e70dbd4e61 Mon Sep 17 00:00:00 2001 From: Michael Kessler Date: Wed, 17 Aug 2011 14:57:20 +0200 Subject: [PATCH 10/13] Explain the growl/growl_notify differences. --- README.md | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/README.md b/README.md index 5b2b622..5b3b05a 100644 --- a/README.md +++ b/README.md @@ -13,7 +13,7 @@ Features * [Directory Change Notification](http://msdn.microsoft.com/en-us/library/aa365261\(VS.85\).aspx) support on Windows ([rb-fchange, >= 0.0.2](https://rubygems.org/gems/rb-fchange) required). * Polling on the other operating systems (help us to support more OS). * Automatic & Super fast (when polling is not used) files modifications detection (even new files are detected). -* Growl notifications ([growlnotify](http://growl.info/documentation/growlnotify.php) & [growl gem](https://rubygems.org/gems/growl) required). +* Growl notifications ([growl_notify gem](https://rubygems.org/gems/growl_notify) or [growlnotify](http://growl.info/documentation/growlnotify.php) & [growl gem](https://rubygems.org/gems/growl) required). * Libnotify notifications ([libnotify gem](https://rubygems.org/gems/libnotify) required). * Tested against Ruby 1.8.7, 1.9.2 and REE. @@ -55,7 +55,7 @@ Install the rb-fsevent gem for [FSEvent](http://en.wikipedia.org/wiki/FSEvents) $ gem install rb-fsevent ``` -Install either the growl_notify or the Growl gem if you want notification support: +Install either the growl_notify or the growl gem if you want notification support: ``` bash $ gem install growl_notify @@ -70,8 +70,9 @@ gem 'rb-fsevent' gem 'growl' ``` -growl_notify uses AppleScript, the suggested method for interfacing with Growl, -rather than the `growlnotify` command to display messages. +The difference between growl and growl_notify is that growl_notify uses AppleScript to +display a message, whereas growl uses the `growlnotify` command. In general the AppleScript +approach is preferred, but this is currently known to not work in conjunction with Spork. ### On Linux From 537f5cb00ab45d91913f1888a2d6678b6cc91a4c Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?R=C3=A9my=20Coutable?= Date: Wed, 17 Aug 2011 17:05:11 +0300 Subject: [PATCH 11/13] Update CHANGELOG for 0.6.2! [ci skip] --- CHANGELOG.md | 16 ++++++++++++++++ 1 file changed, 16 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 2a14a6c..e05d04d 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,3 +1,19 @@ +## 0.6.2 - August 17, 2011 + +### Bugs fixes: + +- Re-add the possibility to use the `growl` gem since the `growl_notify` gem this is currently known to not work in conjunction with Spork. ([@netzpirat][]) +- Ensure that scoped groups and group name are symbolized before checking for inclusion. ([@rymai][]) + +### New features + +- Groups are now stored in a @groups variable (will be used for future features). ([@rymai][]) +- Guards will now receive their group in the options hash at initialization (will be used for future features). ([@rymai][]) + +### Improvement + +- Explain the growl/growl_notify differences in the README. ([@netzpirat][]) + ## 0.6.1 - August 15, 2011 ### Bugs fixes: From 8ff259b249680433f5fab2a9ddd0e66b7ce4fbf4 Mon Sep 17 00:00:00 2001 From: Thibaud Guillaume-Gentil Date: Wed, 17 Aug 2011 16:07:23 +0200 Subject: [PATCH 12/13] Version 0.6.2 --- 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 6b73f67..8daad1b 100644 --- a/lib/guard/version.rb +++ b/lib/guard/version.rb @@ -1,3 +1,3 @@ module Guard - VERSION = "0.6.1" unless defined? Guard::VERSION + VERSION = "0.6.2" unless defined? Guard::VERSION end From b144514b06357a890b19d29b6001eaf2d686800b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?R=C3=A9my=20Coutable?= Date: Wed, 17 Aug 2011 17:10:14 +0300 Subject: [PATCH 13/13] Formatting [ci skip] --- CHANGELOG.md | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index e05d04d..9ce3cff 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -5,12 +5,12 @@ - Re-add the possibility to use the `growl` gem since the `growl_notify` gem this is currently known to not work in conjunction with Spork. ([@netzpirat][]) - Ensure that scoped groups and group name are symbolized before checking for inclusion. ([@rymai][]) -### New features +### New features: - Groups are now stored in a @groups variable (will be used for future features). ([@rymai][]) - Guards will now receive their group in the options hash at initialization (will be used for future features). ([@rymai][]) -### Improvement +### Improvement: - Explain the growl/growl_notify differences in the README. ([@netzpirat][]) @@ -28,11 +28,11 @@ - Pull request [#107](https://github.com/guard/guard/pull/107): Small spelling fix. ([@dnagir][]) - Dir.glob now ignores files that don't need to be watched. ([@rymai][]) -### New features +### New features: - Pull request [#112](https://github.com/guard/guard/pull/112): Add `list` command to CLI. ([@docwhat][]) -### Improvements +### Improvements: - Pull request [#99](https://github.com/guard/guard/pull/99): [OS X] Switch from growl gem to growl_notify gem. ([@johnbintz][]) - Pull request [#115](https://github.com/guard/guard/pull/115): [Linux] Add ':transient => true' to default libnotify options. ([@zonque][]) @@ -51,12 +51,12 @@ ## 0.5.0 - July 2, 2011 -### New features +### New features: - Guard::Ego is now part of Guard, so Guardfile is automagically re-evaluated when modified. ([@thibaudgg][]) - Pull request [#91](https://github.com/guard/guard/pull/91): Show Guards in Guardfile with the `guard -T`. ([@johnbintz][]) -### Improvements +### Improvements: - Issue [#98](https://github.com/guard/guard/issues/98): Multiple calls per watch event on linux with rb-inotify. ([@jeffutter][] & [@netzpirat][]) - Pull request [#94](https://github.com/guard/guard/pull/94): Show backtrace in terminal when a problem with a watch action occurs. ([@capotej][]) @@ -78,7 +78,7 @@ ## 0.4.1 - June 7, 2011 -### Improvements +### Improvements: - Pull request [#77](https://github.com/guard/guard/pull/77): Refactor `get_guard_class` to first try the constant and fallback to require + various tweaks. ([@mislav][]) - Notifier improvement, don't use system notification library if could not be required. ([@yannlugrin][]) @@ -94,7 +94,7 @@ - Pull request [#73](https://github.com/guard/guard/pull/73): Allow DSL's `group` method to accept a Symbol as group name. ([@johnbintz][]) - Pull request [#51](https://github.com/guard/guard/pull/51): Allow options (like `:priority`) to be passed through to the Notifier. ([@indirect][] & [@netzpirat][]) -### Improvements +### Improvements: - Pull request [#74](https://github.com/guard/guard/pull/74): Added link definitions to make the CHANGELOG more DRY! That's for sure now, we have the cleanest CHANGELOG ever! (even the link definitions are sorted alphabetically!) ([@pcreux][])