On Wed, May 25, 2011 at 9:30 PM, Gustavo Delfino <gdelf...@umich.edu> wrote:

> Thank you Justin & David.
>
> I was able to to use Justin solution. I also tried hard to understand and
> use David's solution but I could not.
>
> Now I am just curious. What is &HaveXXXHelpers.on_column? A reference to a
> class method?
>
> Regards,
>
> Gustavo Delfino
>
> On May 25, 2011, at 2:04 PM, Justin Ko wrote:
>
> >
> >
> > On Wed, May 25, 2011 at 11:20 AM, David Chelimsky <dchelim...@gmail.com>
> wrote:
> > On May 25, 2011, at 10:07 AM, Gustavo Delfino wrote:
> >
> > > Hello all. Thanks to subjects, custom matchers and fluent chaining I
> was able to greatly simplify my spec. But now I want to DRY my custom
> matchers.
> > >
> > > I have two custom matchers: 'have_text' and 'have_number' and both
> contain exactly the same chains 'on_column' and 'on_columns'.
> > >
> > > Is there a way to DRY this up?
> > >
> > >
> > > module CustomColumnsMatchers
> > >
> > >  RSpec::Matchers.define(:have_text) do |expected_string|
> > >    chain(:on_column) do |colnumber|
> > >      @index = colnumber - 1
> > >      @width = 1
> > >    end
> > >    chain(:on_columns) do |range|
> > >      a = range.to_a
> > >      @index = Range.new( a[0]-1, a[-1]-1 )
> > >      @width = a[-1] - a[0] + 1
> > >    end
> > >    match do |line|
> > >      line[@index] == expected_string.ljust(@width)
> > >    end
> > >  end
> > >
> > >  RSpec::Matchers.define(:have_number) do |expected_number_string|
> > >    chain(:on_column) do |colnumber|
> > >      @index = colnumber - 1
> > >      @width = 1
> > >    end
> > >    chain(:on_columns) do |range|
> > >      a = range.to_a
> > >      @index = Range.new( a[0]-1, a[-1]-1 )
> > >      @width = a[-1] - a[0] + 1
> > >    end
> > >    match do |line|
> > >      line[@index] == expected_number_string.rjust(@width,'0')
> > >    end
> > >  end
> > >
> > > end
> > >
> > >
> > > Regards,
> > >
> > > Gustavo Delfino
> >
> > I'd probably do something like:
> >
> >  RSpec::Matchers.define(:have_text) do |expected_string|
> >   chain(:on_column, &HaveXXXHelpers.on_column)
> >   chain(:on_columns, &HaveXXXHelpers.on_columns)
> >   match do |line|
> >     line[@index] == expected_string.ljust(@width)
> >   end
> >  end
> >
> > HTH,
> > David
> > _______________________________________________
> > rspec-users mailing list
> > rspec-users@rubyforge.org
> > http://rubyforge.org/mailman/listinfo/rspec-users
> >
> > Ahh David beat me to the lambda approach! Here is another way:
> >
> >
> > module CustomColumnsMatchers
> >
> >  RSpec::Matchers.define(:have_text) do |expected_string|
> >    extend ChainMethods
> >
> >    match do |line|
> >      line[@index] == expected_string.ljust(@width)
> >    end
> >  end
> >
> >  RSpec::Matchers.define(:have_number) do |expected_number_string|
> >    extend ChainMethods
> >
> >    match do |line|
> >      line[@index] == expected_number_string.rjust(@width,'0')
> >    end
> >  end
> >
> >  module ChainMethods
> >    def self.extended(matcher)
> >      matcher.instance_eval do
> >       chain(:on_column) do |colnumber|
> >         @index = colnumber - 1
> >         @width = 1
> >       end
> >       chain(:on_columns) do |range|
> >          a = range.to_a
> >          @index = Range.new( a[0]-1, a[-1]-1 )
> >          @width = a[-1] - a[0] + 1
> >       end
> >     end
> >   end
> > end
> >
> > end
> > _______________________________________________
> > rspec-users mailing list
> > rspec-users@rubyforge.org
> > http://rubyforge.org/mailman/listinfo/rspec-users
>
> _______________________________________________
> rspec-users mailing list
> rspec-users@rubyforge.org
> http://rubyforge.org/mailman/listinfo/rspec-users
>

Actually, the lambda approach won't work, because you're carrying state
(@index & @width) that is outside of scope in the matcher. This is what it
would look like though:

module CustomColumnsMatchers

 RSpec::Matchers.define(:have_text) do |expected_string|
   chain(:on_column, &ChainHelpers.on_column)
   chain(:on_columns, &ChainHelpers.on_columns)

   match do |line|
     line[@index] == expected_string.ljust(@width)
   end
 end

 RSpec::Matchers.define(:have_number) do |expected_number_string|
   chain(:on_column, &ChainHelpers.on_column)
   chain(:on_columns, &ChainHelpers.on_columns)

   match do |line|
     line[@index] == expected_number_string.rjust(@width,'0')
   end
 end

 module ChainHelpers
   def self.on_column
     lambda do |colnumber|
        @index = colnumber - 1
        @width = 1
      end
    end

    def self.on_columns
      lambda do |range|
         a = range.to_a
         @index = Range.new( a[0]-1, a[-1]-1 )
         @width = a[-1] - a[0] + 1
      end
    end
  end

end


You could keep all of the state in the ChainHelpers module:

module CustomColumnsMatchers

 RSpec::Matchers.define(:have_text) do |expected_string|
   chain(:on_column, &ChainHelpers.on_column)
   chain(:on_columns, &ChainHelpers.on_columns)

   match do |line|
     line[ChainHelpers.index] == expected_string.ljust(ChainHelpers.width)
   end
 end

 RSpec::Matchers.define(:have_number) do |expected_number_string|
   chain(:on_column, &ChainHelpers.on_column)
   chain(:on_columns, &ChainHelpers.on_columns)

   match do |line|
     line[ChainHelpers.index] ==
expected_number_string.rjust(ChainHelpers.width,'0')
   end
 end

 module ChainHelpers
   class << self
     attr_accessor :index, :width
   end

   def self.on_column
     lambda do |colnumber|
        self.index = colnumber - 1
        self.width = 1
      end
    end

    def self.on_columns
      lambda do |range|
         a = range.to_a
         self.index = Range.new( a[0]-1, a[-1]-1 )
         self.width = a[-1] - a[0] + 1
      end
    end
  end

end


That is kinda dirty and ugly - I'd stick with the "extend" solution.
_______________________________________________
rspec-users mailing list
rspec-users@rubyforge.org
http://rubyforge.org/mailman/listinfo/rspec-users

Reply via email to