Readable Specs == Business Value

Consider the situation of a product manager or software architect overseeing the work of several dozen if not hundreds of programmers. If you have been in this situation you know what the problems are

  • Ensuring delivery quality is a nightmare - you are spending as much time verifying the work of the junior programmers as you are solving problems.
  • It is very difficult to maintain a clear separation between the architect and the implementation team. Architects are expensive and should not be concerning themselves with low level implementation details. Also, architects need to think at a different level of abstraction than the implementers. However, without this clear separation, you find that a lot of the senior programmers time is wasted on low level issues
  • Levels of supervision being high, code reviews become time consuming and expensive. Bugs slip through regardless and such bugs become expensive in terms of client satisfaction.
  • Programmers leave frequently and new programmers need to come up the curve quickly in order to avoid interruptions to service delivery.
  • Good programmers are hard to find. One has to make do with locally available talent which most likely cannot deliver working code without supervision. Because good programmers are hard to find, they are expensive. Also, they bring about “key-man” risk in your organisation.

If your organisation is not serious about writing tests for your code, it is obvious that the product delivery manager is going to have a hard time supervising the output. It is very time consuming to read through source code and one has to verify the correctness of the behaviour by hand. This is clearly not scalable.

If you are writing specs, you’re in a much better position. Reading the specs is much easier than reading code and passing specs mean that the code is performing as desired. However, it is possible to turn the humble spec into a powerful force multiplier and generate huge amounts of business value through the simple expedient of making the spec readable.

Consider the following three examples. Which one would you rather work with?

First, the actual implementation itself.

def scheduled_dates_must_be_center_meeting_days #this function is only for repayment dates
return [false, "Not client defined"] unless client
center = client.center
failed = []
correct_weekday = nil
["scheduled_first_payment_date"].each do |d|
# if the loan disbursal date is set and it is not being set right now, no need to check as the loan has been already disbursed
# hence we need not check it again
if self.disbursal_date and not self.dirty_attributes.keys.find{|da| da.name == :disbursal_date}
return true
end
if date = instance_eval(d) and not date.weekday == center.meeting_day_for(date).to_sym
failed << d.humanize
correct_weekday = center.meeting_day_for(date)
end
end
return [false, "#{failed.join(",")} must be #{correct_weekday}"] unless failed.blank?
return true
end
view raw loan.rb hosted with ❤ by GitHub

This is very cumbersome to work with. The reader has to try and follow along the logic in their mind, an impossible task. Failing that, they must test the functionality by hand which makes the whole process very slow and time-consuming. Clearly not a scalable process. This is why the software industry developed automated testing.

Then, a naively written spec of questionable readability

it "should not be valid if scheduled disbursal date and scheduled first payment date are not center meeting dates" do
@loan.scheduled_disbursal_date = @center.meeting_day
@loan.should be_valid
@loan.scheduled_first_payment_date = @center.meeting_day
@loan.should be_valid
@loan.scheduled_disbursal_date = @center.meeting_day + 1
@loan.should_not be_valid
@loan.scheduled_disbursal_date = @center.meeting_day - 1
@loan.should_not be_valid
@loan.scheduled_first_payment_date = @center.meeting_day + 1
@loan.should_not be_valid
@loan.scheduled_first_payment_date = @center.meeting_day - 1
@loan.should_not be_valid
end
view raw loan_spec.rb hosted with ❤ by GitHub
This is much better. It tests the behaviour of the program automatically and so this means that simply verifying that the spec is properly written means that passing tests give you a lot of confidence in the behaviour of the software. However, the spec is very long and verbose and it is possible that the spec itself has bugs in it. Secondly, lazy programmers can slip in some tests that do not actually test anything.

And lastly, a readable loan spec

describe "scheduled first payment date must be a center meeting date" do
let(:center) { FactoryGirl.create(:center) }
subject { FactoryGirl.build(:loan,
:center => center,
:scheduled_first_payment_date => Date.new(2012,2,2) }
before { center.stub(:center_meeting_dates).and_return([Date.new(2012,2,1)])
it { should_not be_valid }
its("errors") { should have_key(:schedled_first_payment_date) }
end
view raw loan_spec.rb hosted with ❤ by GitHub

Ah! So much easier to work with. A casual glance at the spec reveals the intention of the spec and passing specs give confidence as to behaviour. A delivery manager working with the latter is going to be orders of magnitude more productive than with the first. i.e. if you are not writing specs, you are living in the dark ages. If you are, then making them more readable is perhaps the single most effective thing you could do to improve productivity.

Last weekend we were doing a TDD course with a client and here’s the spec file that emerged from it. I consider this to be a very readable spec file and my thesis is that were I to be the delivery manager on this project, I have achieved complete decoupling between the architecture and the implementation of the product. Here’s the spec for a simple tic-tac-toe game.

require 'spec_helper'
describe Game do
let!(:player1) { FactoryGirl.create(:user) }
let!(:player2) { FactoryGirl.create(:user) }
let(:started_game) { FactoryGirl.build(:game).tap { |g| g.users << player1 ; g.users << player2 ; g.start! }}
it { should have_many :moves }
it { should have_and_belong_to_many :users}
context "with zero players" do
subject { FactoryGirl.build(:game) }
it { should_not be_valid }
end
context "with one player" do
subject { FactoryGirl.build(:game).tap { |g| g.users << player1 } }
it { should be_valid }
its(:first_player) { should be player1 }
its(:second_player) { should be_nil}
it { should_not be_startable }
context "when trying to start" do
before(:each) { subject.start! }
it { should_not be_started }
it { should_not be_startable }
context "when trying to make a move" do
context "with first move" do
let!(:first_move) { FactoryGirl.create(:move, :user => player1, :x=> 0, :y => 0) }
specify { subject.add_move(first_move).should be_false }
it { should be_valid }
its("moves.size") { should be 0 }
its(:current_player) { should be player1}
end
end
end
end
context "with two players" do
subject { FactoryGirl.build(:game).tap { |g| g.users << player1 ; g.users << player2 }}
it { should be_valid }
its(:first_player) { should be player1 }
its(:second_player) { should be player2 }
it { should be_startable }
context "when it has started" do
subject { started_game }
it { should be_valid }
it { should be_started }
it { should_not be_startable }
its(:current_player) { should be player1 }
context "when moves are made" do
context "with first valid move" do
let!(:first_move) { FactoryGirl.create(:move, :user => player1, :x=> 0, :y => 0) }
specify { subject.add_move(first_move).should be_true }
context "when added" do
before(:each) { subject.add_move(first_move) }
it { should be_valid }
its("moves.size") { should be 1 }
its(:current_player) { should be player2 }
it { should_not be_complete }
its(:winner) { should be_nil }
its(:result) { should be_nil }
end
end
context "with first invalid move" do
let!(:first_move) { FactoryGirl.create(:move, :user => player1, :x=> 6, :y => 6) }
specify { subject.add_move(first_move).should be_false }
context "when added" do
before(:each) { subject.add_move(first_move) }
it { should be_valid }
its("moves.size") { should be 0 }
its(:current_player) { should be player1 }
end
end
context "with wrong players move" do
let!(:first_move) { FactoryGirl.create(:move, :user => player2, :x=> 0, :y => 0) }
specify { subject.add_move(first_move).should be_false }
its(:current_player) { should be player1 }
end
end
end
context "completed game" do
subject { FactoryGirl.build(:game).tap { |g| g.users << player1 ; g.users << player2 ; g.start! }}
before(:all) do
subject.add_move(FactoryGirl.create(:move, :user => player1, :x => 0, :y => 0))
subject.add_move(FactoryGirl.create(:move, :user => player2, :x => 1, :y => 1))
subject.add_move(FactoryGirl.create(:move, :user => player1, :x => 1, :y => 0))
subject.add_move(FactoryGirl.create(:move, :user => player2, :x => 2, :y => 2))
subject.add_move(FactoryGirl.create(:move, :user => player1, :x => 2, :y => 0))
end
it { should be_complete }
its(:winner) { should be player1 }
end
end
context "with three players" do
subject { FactoryGirl.build(:game).tap { |g| 3.times { g.users << FactoryGirl.create(:user) } } }
it { should_not be_valid }
end
context "when dealing with moves" do
context "with reference to bounds" do
let!(:move) { FactoryGirl.build(:move, :x => 0, :y => 0) }
let!(:good_move) { FactoryGirl.build(:move, :x => 2, :y => 2) }
let!(:bad_move) { FactoryGirl.build(:move, :x => 3, :y => 3) }
let!(:another_bad_move) { FactoryGirl.build(:move, :x => -1, :y => -1) }
specify { subject.move_within_bounds?(move).should be_true }
specify { subject.move_within_bounds?(good_move).should be_true }
specify { subject.move_within_bounds?(bad_move).should be_false}
specify { subject.move_within_bounds?(another_bad_move).should be_false}
end
context "with reference to players" do
before(:each) { subject.stub(:current_player).and_return(player1) }
let!(:good_move) { FactoryGirl.build(:move, :user => player1, :x => 2, :y => 2) }
let!(:bad_move) { FactoryGirl.build(:move, :user => player2, :x => 2, :y => 2) }
specify { subject.valid_move?(good_move).should be_true }
specify { subject.valid_move?(bad_move).should be_false}
end
context "with reference to duplicate moves" do
subject { started_game }
let!(:good_move) { FactoryGirl.build(:move, :user => player1, :x => 2, :y => 2) }
let!(:bad_move) { FactoryGirl.build(:move, :user => player2, :x => 2, :y => 2) }
before :all do
subject.add_move(good_move)
end
specify { subject.valid_move?(bad_move).should be_false}
end
end
end
view raw game_spec.rb hosted with ❤ by GitHub

Readable specs enable the product manager or architect to simply forget about implementation issues and concern himself only with the API and behaviour issues. Specs like these will enable you to manage many more projects than you currently do. With specs like these, you can add more people to the team seamlessly since these specs act as a very readable form of documentation. Readable specs completely decouple your architecture from it implementation. This is a huge win considering the manpower churn that goes on in the industry.

The entire RSpec team must be saluted for creating this amazing piece of software. Whenever I look at a Ruby codebase these days and I don’t see a spec/ directory, I do a little head-shake inside. There is just so much value hidden inside this gem! David Chelimsky and team, do take a bow!

If you’d like your team to write readable specs, give me a shout.