4. Introducing a test harness.
To be able to refactor Alarm, we first need to protect its current behavior (its check method) from regressions by writing tests for it.The implicit dependency of Alarm on Sensor makes Alarm difficult to test. However, it's the fact that Sensor returns random values that makes Alarm impossible to test because the measured pressure values it gets are not deterministic.
It seems we're trapped in a vicious circle: in order to refactor the code (improving its design without altering its behavior) we must test it first, but in order to test it, we must change it first.
We can get out of this problem by applying a dependency-breaking technique called Extract and Override call.
4.1. Extract and Override call.
This is a dependency-breaking technique from Michael Feather's Working Effectively with Legacy code book. These techniques consist of carefully making a very small modification in the production code in order to create a seam:A seam is a place where you can alter behavior in your program without editing in that place.The behavior we want to test is the logic in Alarm's check method. This logic is very simple, just an if condition and a mutation of a property, but as we saw its dependence on Sensor makes it untestable.
To test it, we need to alter the collaboration between Alarm and Sensor so that it becomes deterministic. That would make Alarm testable. For that we have to create a seam first.
4.1.1. Extract call to create a seam.
First, we create a seam by extracting the collaboration with Sensor to a protected method, probeValue. This step must be made with a lot of caution because we have no tests yet.
Thankfully in Java, we can rely on the IDE to do it automatically.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
package tddmicroexercises.tirepressuremonitoringsystem; | |
public class Alarm { | |
private final double LowPressureThreshold = 17; | |
private final double HighPressureThreshold = 21; | |
private Sensor sensor = new Sensor(); | |
private boolean alarmOn = false; | |
public void check() { | |
double psiPressureValue = probeValue(); | |
if (psiPressureValue < LowPressureThreshold || HighPressureThreshold < psiPressureValue) { | |
alarmOn = true; | |
} | |
} | |
protected double probeValue() { | |
return sensor.popNextPressurePsiValue(); | |
} | |
public boolean isAlarmOn() { | |
return alarmOn; | |
} | |
} |
Next, we take advantage of the new seam, to alter the behavior of Alarm without affecting the logic we want to test.
To do it, we create a FakeAlarm class inside the AlarmShould tests that inherits from Alarm and overrides the call to the protected probeValue method:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
import org.junit.Test; | |
import tddmicroexercises.tirepressuremonitoringsystem.Alarm; | |
import static org.hamcrest.MatcherAssert.assertThat; | |
import static org.hamcrest.Matchers.is; | |
public class AlarmShould { | |
private class FakeAlarm extends Alarm { | |
private final double samplePressure; | |
public FakeAlarm(double samplePressure) { | |
super(); | |
this.samplePressure = samplePressure; | |
} | |
@Override | |
protected double probeValue() { | |
return samplePressure; | |
} | |
} | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
import org.junit.Test; | |
import tddmicroexercises.tirepressuremonitoringsystem.Alarm; | |
import static org.hamcrest.MatcherAssert.assertThat; | |
import static org.hamcrest.Matchers.is; | |
public class AlarmShould { | |
@Test | |
public void be_on_when_pressure_value_is_too_low() { | |
Alarm alarm = new FakeAlarm(5.0); | |
alarm.check(); | |
assertThat(alarm.isAlarmOn(), is(true)); | |
} | |
@Test | |
public void be_on_when_pressure_value_is_too_high() { | |
Alarm alarm = new FakeAlarm(25.0); | |
alarm.check(); | |
assertThat(alarm.isAlarmOn(), is(true)); | |
} | |
@Test | |
public void be_off_when_pressure_value_is_within_safety_range() { | |
Alarm alarm = new FakeAlarm(20.0); | |
alarm.check(); | |
assertThat(alarm.isAlarmOn(), is(false)); | |
} | |
private class FakeAlarm extends Alarm { | |
/// ... | |
} | |
} |
5. Making the dependency on Sensor explicit.
Now our goal is to inject the dependency on Sensor into Alarm through its constructor.To remain in the green all the time, we use the Parallel Change technique.
With TDD we drive a new constructor without touching the one already in place by writing a new behavior test with the help of a mocking library (mockito):
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
import org.junit.Test; | |
import tddmicroexercises.tirepressuremonitoringsystem.Alarm; | |
import tddmicroexercises.tirepressuremonitoringsystem.Sensor; | |
import static org.hamcrest.MatcherAssert.assertThat; | |
import static org.hamcrest.Matchers.is; | |
import static org.mockito.Mockito.doReturn; | |
import static org.mockito.Mockito.mock; | |
import static org.mockito.Mockito.verify; | |
public class AlarmShould { | |
// Tests using FakeAlarm | |
// ... | |
@Test | |
public void collaborate_with_an_injected_sensor() { | |
Sensor sensor = mock(Sensor.class); | |
Alarm alarm = new Alarm(sensor); | |
alarm.check(); | |
verify(sensor).popNextPressurePsiValue(); | |
} | |
private class FakeAlarm extends Alarm { | |
// has not changed | |
// ... | |
} | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
package tddmicroexercises.tirepressuremonitoringsystem; | |
public class Alarm { | |
private final double LowPressureThreshold = 17; | |
private final double HighPressureThreshold = 21; | |
private Sensor sensor; | |
private boolean alarmOn; | |
public Alarm(Sensor sensor) { | |
this.sensor = sensor; | |
this.alarmOn = false; | |
} | |
public Alarm() { | |
this(new Sensor()); | |
} | |
// code that has not changed | |
// ... | |
} |
Then we use the new constructor in the rest of the tests one by one in order to stop using FakeAlarm.
Once there are no test using FakeAlarm, we can delete it. This makes the default constructor become obsolete, so we delete it too.
Finally, we also inline the previously extracted probeValue method.
This is the resulting test code after introducing dependency injection in which we have also deleted the test used to drive the new constructor because we think it was redundant:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
import org.junit.Test; | |
import tddmicroexercises.tirepressuremonitoringsystem.Alarm; | |
import tddmicroexercises.tirepressuremonitoringsystem.Sensor; | |
import static org.hamcrest.MatcherAssert.assertThat; | |
import static org.hamcrest.Matchers.is; | |
import static org.mockito.Mockito.doReturn; | |
import static org.mockito.Mockito.mock; | |
public class AlarmShould { | |
@Test | |
public void be_on_when_pressure_value_is_too_low() { | |
Alarm alarm = new Alarm(sensorThatProbes(5.0)); | |
alarm.check(); | |
assertThat(alarm.isAlarmOn(), is(true)); | |
} | |
@Test | |
public void be_on_when_pressure_value_is_too_high() { | |
Alarm alarm = new Alarm(sensorThatProbes(25.0)); | |
alarm.check(); | |
assertThat(alarm.isAlarmOn(), is(true)); | |
} | |
@Test | |
public void be_off_when_pressure_value_is_within_safety_range() { | |
Alarm alarm = new Alarm(sensorThatProbes(20.0)); | |
alarm.check(); | |
assertThat(alarm.isAlarmOn(), is(false)); | |
} | |
protected Sensor sensorThatProbes(double value) { | |
Sensor sensor = mock(Sensor.class); | |
doReturn(value).when(sensor).popNextPressurePsiValue(); | |
return sensor; | |
} | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
package tddmicroexercises.tirepressuremonitoringsystem; | |
public class Alarm { | |
private final double LowPressureThreshold = 17; | |
private final double HighPressureThreshold = 21; | |
private Sensor sensor; | |
private boolean alarmOn; | |
public Alarm(Sensor sensor) { | |
this.sensor = sensor; | |
this.alarmOn = false; | |
} | |
public void check() { | |
double psiPressureValue = sensor.popNextPressurePsiValue(); | |
if (psiPressureValue < LowPressureThreshold || HighPressureThreshold < psiPressureValue) { | |
alarmOn = true; | |
} | |
} | |
public boolean isAlarmOn() { | |
return alarmOn; | |
} | |
} |
This is the second post in a series of posts about solving the Tire Pressure Monitoring System exercise in Java:
- Solving the Tire Pressure Monitoring System exercise (I)
- Solving the Tire Pressure Monitoring System exercise (II)
- Solving the Tire Pressure Monitoring System exercise (III)
- Solving the Tire Pressure Monitoring System exercise (IV)
No comments:
Post a Comment