Monday, 26 August 2013

Do you adhere to the Single-Responsibility Principle?

The single-responsibility principle is one of the fundamental principles of object orientated design.  It is the idea that any object, method, variable etc has one purpose and one purpose only.

'Yes', I hear you say, 'I'm a super-duper expert in object-orientated design. I follow this principle and I defy you to find a single part of my code where I haven't followed SRP!'

Well, dear reader, how are your tests?

In my opinion, the SRP should apply to tests as well, meaning that every test should have a point, and one point only. Here is a little example showing a common way to structure tests:

Here is a little MessageListener.java class.  When the a message is received, the onMessage method is called and the messageListener both stores the message and increments a counter of the total number of messages received.

public class MessageListener {
private int messageCount = 0;
private List<Message> receivedMessages = new ArrayList<>();
public void onMessage(Message message) {
System.out.println(message.getContent());
messageCount++;
receivedMessages.add(message);
}
public List<Message> getReceivedMessages() {
return receivedMessages;
}
public int getMessageCount(){
return this.messageCount;
}
}
view raw gistfile1.java hosted with ❤ by GitHub

All is beautiful so far.

However here is our MessageListenerTest.java:

@Test
public void testOnMessageMethod() {
MessageListener messageListener = new MessageListener();
Message message = new Message("hi there!");
messageListener.onMessage(message);
Assert.assertEquals(1, messageListener.getMessageCount());
Assert.assertEquals(1, messageListener.getReceivedMessages().size());
Assert.assertEquals(message, messageListener.getReceivedMessages().get(0));
Message anotherMessage = new Message("yo yo");
messageListener.onMessage(anotherMessage);
Assert.assertEquals(2, messageListener.getMessageCount());
Assert.assertEquals(2, messageListener.getReceivedMessages().size());
Assert.assertEquals(anotherMessage, messageListener.getReceivedMessages().get(1));
}
view raw gistfile1.java hosted with ❤ by GitHub

What on earth is going on here? This test is not adhering to the SRP as it is clearly testing multiple things.  There are 6 asserts in one test, which should set off alarm bells.  If the messageCount is broken in the message listener, or the message isn't added to the receivedMessages collection etc, then this test will fail and the developer will have to waste time figuring out exactly what went on.  Okay, in this example it's not exactly difficult to understand what is happening, but in more complicated examples this problem quickly becomes apparent.

How about writing the test like this:

@Test
public void testThatWhenOnMessageMethodIsCalledWithOneValidMessageTheMessageCounterIsIncremented() {
MessageListener messageListener = new MessageListener();
Message message = new Message("hi there!");
messageListener.onMessage(message);
Assert.assertEquals(1, messageListener.getMessageCount());
}
@Test
public void testThatWhenOnMessageMethodIsCalledWithTwoValidMessageTheMessageCounterIsIncrementedTwice() {
MessageListener messageListener = new MessageListener();
Message message = new Message("hi there!");
Message anotherMessage = new Message("yo yo");
messageListener.onMessage(message);
messageListener.onMessage(anotherMessage);
Assert.assertEquals(2, messageListener.getMessageCount());
}
@Test
public void testThatWhenOnMessageMethodIsCalledWithOneValidMessageTheMessageIsAddedToTheReceivedMessageList() {
MessageListener messageListener = new MessageListener();
Message message = new Message("hi there!");
messageListener.onMessage(message);
Assert.assertEquals(1, messageListener.getReceivedMessages().size());
Assert.assertTrue(messageListener.getReceivedMessages().contains(message));
}
@Test
public void testThatWhenOnMessageMethodIsCalledWithTwoValidMessagesBothMessageAreAddedToTheReceivedMessagesList() {
MessageListener messageListener = new MessageListener();
Message message = new Message("hi there!");
Message anotherMessage = new Message("yo yo");
messageListener.onMessage(message);
messageListener.onMessage(anotherMessage);
Assert.assertEquals(2, messageListener.getReceivedMessages().size());
Assert.assertTrue(messageListener.getReceivedMessages().contains(anotherMessage));
}
view raw gistfile1.java hosted with ❤ by GitHub
Every test method has one distinct purpose which is summed up in the method name.  If a test fails, you can immediately understand what has failed and which part of the code is incorrect.


So, things to look out for in your own code:

- a test method with some code, then asserts, then more code, then more asserts.
- if you break one tiny bit of your application, how many tests fail? If we change one tiny bit and suddenly 20 tests break all over the shop then what is going on? 

No comments:

Post a Comment

Scala with Cats: Answers to revision questions

I'm studying the 'Scala with Cats' book. I want the information to stick so I am applying a technique from 'Ultralearning...