TDD Tic Tac Toe with Rust
I wanted to learn about Test Driven Design in Rust. While I’ve followed a couple tutorials in Python, this did not feel like I was really learning how to apply TDD, because all of the design decisions were already made.
To find out how this would work, I decided to try it for a small project. For this, I decided to make a command-line Tic Tac Toe application, with the option in mind, to add a GUI later.
This is how it looks:
Welcome to Tic-Tac-Toe!
| |
-+-+-
| |
-+-+-
| |
Where would you like to place your X :
2
|X|
-+-+-
| |
-+-+-
| |
Where would you like to place your O :
3
|X|O
-+-+-
| |
-+-+-
| |
Where would you like to place your X :
2
The last input was not valid. Please try again!
|X|O
-+-+-
| |
-+-+-
| |
Where would you like to place your X :
8
|X|O
-+-+-
| |
-+-+-
|X|
I also learned about rusts macro_rules
so that I could make defining test-cases more intuitive. And there are a lot of tests necessary for the TDD approach, even for such a small application. With the simple new_board!()
macro, I could now create a board state by writing
let board = new_board!(
X,X,O;
O,X,O;
X,O,None;
)
Instead of having to write the following:
let board = Board {
board: [
Symbol::X, Symbol::X, Symbol::O,
Symbol::O, Symbol::X, Symbol:O,
Symbol::X, Symbol::O, Symbol::None,
]
}
While this could still be improved, It already helped a lot with defining new tests and making them readable.
While I was able to finish the project with the TDD approach, it took a lot of energy to do it this way. The program is nicely tested, but TDD falls a bit short for Rust, because a lot of work is already done by the strong type System. The Idea, that every change should be motivated by a test falls apart, when you write a single test and basically have to implement the whole fuction you are trying to test to make the compiler happy about the types. In my opinion, the TDD approach makes a lot more sense in a language like Python, where you actually benefit more from the tests you are writing.
The code for the game is available over on my github