Dependency Injection
(If you have already signed up) Read the moodle first
The Moodle pages of the course contain important information which you should read before studying these pages, including information about the aims of the project and its grading. The moodle pages also link to the most relevant parts of these pages. These pages expect you to be familiar with the contents of the moodle and can feel confusing if you do not.
These instructions have been translated from the material of the course Software Production
First, read http://jamesshore.com/Blog/Dependency-Injection-Demystified.html
Below is a simple calculator:
class Calculator:
def execute(self):
while True:
number1 = int(input("Number 1:"))
if number1 == -9999:
return
number2 = int(input("Number 2:"))
if number2 == -9999:
return
result = self._calculate_sum(number1, number2)
print(f"Sum: {result}")
def _calculate_sum(self, number1, number2):
return number1 + number2
def main():
calculator = Calculator()
calculator.execute()
if __name__ == "__main__":
main()
The downside of the program is that the Calculator
class has a concrete dependency on the functions handling output and input: print
and input
.
Concrete dependencies make testing more difficult and complicate program expansion.
Eliminating Direct Dependency
Let’s isolate the output and input handling into a separate ConsoleIO
object:
class ConsoleIO:
def read(self, text):
return input(text)
def write(self, text):
print(text)
Let’s modify the Laskin
class so that it receives an object as a constructor parameter, which will handle communication with the user:
class Calculator:
def __init__(self, io):
self._io = io
def execute(self):
while True:
number1 = int(self._io.read("Number 1:"))
if number1 == -9999:
return
number2 = int(self._io.read("Number 2:"))
if number2 == -9999:
return
result = self._calculate_sum(number1, number2)
self._io.write(f"Sum: {result}")
def _calculate_sum(self, number1, number2):
return number1 + number2
The application is now started in such a way that the communication-handling object is injected as a constructor parameter:
def main():
io = ConsoleIO()
calculator = Calculator(io)
calculator.execute()
main()
Testing
Now, unit tests can be easily written for the program. For testing purposes, we create a fake class, a stub, which externally behaves the same way as ConsoleIO
objects:
class StubIO:
def __init__(self, inputs):
self.inputs = inputs
self.outputs = []
def read(self, text):
return self.inputs.pop(0)
def write(self, text):
self.outputs.append(text)
The stub can be given “user inputs” as a constructor parameter. After execution, the program’s outputs can be retrieved from the stub. Here is the test:
class TestCalculator(unittest.TestCase):
def test_one_sum_correct(self):
io = StubIO(["1", "3", "-9999"])
calculator = Calculator(io)
calculator.execute()
self.assertEqual(io.outputs[0], "Sum: 4")
Summary
Dependency injection is actually an extremely simple technique, and many have likely used it already in basic programming courses.
Consider, for example, computer games, which often rely on random numbers. If a game is coded as follows, automated testing becomes very difficult:
class Game:
def moving_player(self):
direction = random.randint(0, 8)
If, on the other hand, the random number generator is injected into the game as follows
class Game:
def __init__(self, generator):
self._generator = generator
def moving_player(self):
direction = self._generator.randint(0, 8)
we can inject a version of the random number generator during testing that allows us to control the numbers it generates. For example, here is a version of the random number generator that always returns the number 1 when the randint method is called:
class Generator:
def randint(self, a, b):
return 1
Fix this page
Make an suggestion for an improvement by editing this file in GitHub.