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 |
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 |
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 |
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 |
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.