In the second part we are working on the game, a player has 12 chances to break the code.
We are going to change the game engine and also add random drawing of the key code..
Here again we first show all the tests before implementing them, they are our todo list.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 | class PlayerGuessedServiceTest < Test::Unit::TestCase def setup @engine = GameEngine.new('GRWR') end def test_11_wins_immediately assert_equal :win, @engine.player_guessed('GRWR') end def test_12_wins_after_some_guesses assert_equal [], @engine.player_guessed('BBBB') assert_equal [:black], @engine.player_guessed('GBBB') assert_equal :win, @engine.player_guessed('GRWR') end def test_13_defeats number_of_guesses_before_last = GameEngine::MAX_GUESSES - 1 number_of_guesses_before_last.times do @engine.player_guessed('BBBB') end assert_equal :defeat, @engine.player_guessed('BBBB') end def test_14_wins_on_last_chance number_of_guesses_before_last = GameEngine::MAX_GUESSES - 1 number_of_guesses_before_last.times do @engine.player_guessed('BBBB') end assert_equal :win, @engine.player_guessed('GRWR') end end |
The tests expect a new API from the game engine, a player_guessed
method. The method either returns an array of white
and black
pegs (symbols), either returns symbols named defeat
or win
.
Tests also expect MAX_GUESSES
to provide how many guesses a player can make before losing.
Notice how the code simulates invalid guesses to reach the edge cases, on line 23, for instance, the code repeats an invalid guess MAX_GUESSES - 1
times so that the guess on line 27 is the last one.
We build on the current engine implementation presented in the first part of the series. We are not going to show the whole code, we focus on the player_guessed
method.
Our first test is about a player winning at the very first guess.
1 2 3 4 5 6 7 8 9 10 | # returns an array of :white and :black values, :win or :defeat def player_guessed guess result = check(guess) return :win if result == [:black, :black, :black, :black] result end |
On line 4 we reuse the check
method. On line 6, we check if it returns four black
symbols, meaning the guess was successful.
$> ruby game_engine_test.rb -n /11/ Loaded suite /tmp/release/mastermind/game_engine_test Started . Finished in 0.000397777 seconds. ------ 1 tests, 1 assertions, 0 failures, 0 errors, 0 pendings, 0 omissions, 0 notifications 100% passed ------ 2513.97 tests/s, 2513.97 assertions/s
As long as we don’t add code that implements the defeat aspects, the current version is fine…
$> ruby game_engine_test.rb -n /1[12]/ Loaded suite /tmp/release/mastermind/game_engine_test Started .. Finished in 0.000522613 seconds. ------ 2 tests, 4 assertions, 0 failures, 0 errors, 0 pendings, 0 omissions, 0 notifications 100% passed ------ 3826.92 tests/s, 7653.85 assertions/s
A player defeats if he/she does not break the key code after 12 tries. We set up a constant equal to 12.
1
| MAX_GUESSES = 12 |
We need to count the number of guesses, therefore we add the count_guesses
attribute initialized to 0 (see line 3).
1 2 3 4 | def initialize key_code @key_code = key_code.upcase @count_guesses = 0 end |
In the player_guessed
method we start checking if the player didn’t exceed the allowed number of guesses.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 | # returns an array of :white and :black values, :win or :defeat def player_guessed guess return :defeat if @count_guesses > MAX_GUESSES @count_guesses = @count_guesses + 1 result = check(guess) return :win if result == [:black, :black, :black, :black] result end |
Line 4 returns the defeat
symbol if the counter is greater than the maximum allowed so that once the guesses exceed the maximum it will always fail.
Line 6 increments the counter, as we are now evaluating a new guess.
We call the check
method, on line 8, and if the guess was successful the code returns win
(line 10). Otherwise before returning the feedback, we need to check if the current guess is not the last one.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 | # returns an array of :white and :black values, :win or :defeat def player_guessed guess return :defeat if @count_guesses > MAX_GUESSES @count_guesses = @count_guesses + 1 result = check(guess) return :win if result == [:black, :black, :black, :black] return :defeat if @count_guesses == MAX_GUESSES result end |
Line 11, checks if the player did reach the last guess.
$> ruby game_engine_test.rb -n /1[123]/ Loaded suite /tmp/release/mastermind/game_engine_test Started ... Finished in 0.00076802 seconds. ------ 3 tests, 5 assertions, 0 failures, 0 errors, 0 pendings, 0 omissions, 0 notifications 100% passed ------ 3906.15 tests/s, 6510.25 assertions/s
The last test and all the test do pass…
$> ruby game_engine_test.rb Loaded suite /tmp/release/mastermind/game_engine_test Started ............... Finished in 0.002383086 seconds. ------ 15 tests, 17 assertions, 0 failures, 0 errors, 0 pendings, 0 omissions, 0 notifications 100% passed ------ 6294.36 tests/s, 7133.61 assertions/s
Now that the engine is ready, we must make it usable by a real user.
A very simple implementation is going to run in the terminal.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 | require 'game_engine' def play_game engine = GameEngine.new loop do print "Make a guess (or exit): " guess = gets.chomp exit if guess == 'exit' result = engine.player_guessed(guess) if result == :win puts "You win!" break elsif result == :loose puts "You loose, the code was '#{engine.key_code}'" break end p result end end loop do play_game print "Do you want to play again (yes or no)? " answer = gets.chomp break if answer == 'no' end |
We first require the game engine, then define a method named play_game
.
At line 32 we start a loop that first calls play_game
(ensuring to play one game). Then (line 34), the code asks the user if he/she wants to play again.
The play_game
method is pretty simple, it prompts the user for a guess. Uses the engine to check the guess (line 14) and give a feedback to the user depending on the check result.
The problem with our implementation is that the game engine uses only one code. We must introduce the random drawing of the code (without breaking the tests).
Because the tests need the hardcoded code while the interactive game needs a random code, we introduce the code drawer concept.
1 2 3 4 5 6 7 8 9 10 11 12 13 | class CodeDrawer COLORS = ['W', 'R', 'B', 'G', 'Y', 'P', 'O', 'G'] # returns a random 4-color code def draw COLORS[rand(COLORS.length)] + COLORS[rand(COLORS.length)] + COLORS[rand(COLORS.length)] + COLORS[rand(COLORS.length)] end end |
The creation of a GameEngine
requires a code drawer which by default is the real CodeDrawer
.
In the tests we create a mock version of the code drawer like this.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 | class CodeDrawerMocker def initialize code @code = code end # returns a random 4-color code def draw @code end end class PlayerGuessedServiceTest < Test::Unit::TestCase def setup @engine = GameEngine.new(CodeDrawerMocker.new('GRWR')) end def test_11_wins_immediately assert_equal :win, @engine.player_guessed('GRWR') end def test_12_wins_after_some_guesses assert_equal [], @engine.player_guessed('BBBB') assert_equal [:black], @engine.player_guessed('GBBB') assert_equal :win, @engine.player_guessed('GRWR') end def test_13_defeats number_of_guesses_before_last = GameEngine::MAX_GUESSES - 1 number_of_guesses_before_last.times do @engine.player_guessed('BBBB') end assert_equal :defeat, @engine.player_guessed('BBBB') end def test_14_wins_on_last_chance number_of_guesses_before_last = GameEngine::MAX_GUESSES - 1 number_of_guesses_before_last.times do @engine.player_guessed('BBBB') end assert_equal :win, @engine.player_guessed('GRWR') end end |
We define the mock class (line 1-12) and use it at line 17 (we apply the same change to the previous test cases setup).
And we also need to change GameEngine
initialization.
1 2 3 4 5 6 7 8 | def initialize drawer = CodeDrawer.new @drawer = drawer @count_guesses = 0 @key_code = @drawer.draw.upcase end |
Let’s run the test a last time.
$> ruby game_engine_test.rb Loaded suite /tmp/release/mastermind/game_engine_test Started ............... Finished in 0.002187674 seconds. ------ 15 tests, 17 assertions, 0 failures, 0 errors, 0 pendings, 0 omissions, 0 notifications 100% passed ------ 6856.60 tests/s, 7770.81 assertions/s