/dart_katas

(My) Katas in Dart. Or maybe a simple lab for experimentation.

Primary LanguageDart

Dart Katas

Build Status

Quite frankly, I think that this should actually better be named something along the lines of Dart Lab, the examples/exercises below are much more suited to expose experimentation than demonstration. But I won't change anything, if this is shameful, so be it!

These are coding exercises meant to sharpen your programming abilities, analogous to martial arts'.

All of the katas were done using TDD, meaning they are not only tested but were tailored for (driven by) the tests.

Featured Katas

  1. Genetic Algorithm
    • Is done with OOP, in a more maintainable way than the original article.
    • The object that manages the evolution (GeneticEvolutionSimulator) outputs to a stream, making further reactive client-side programming much easier.
    • The core objects — Individual and Population — are immutable.
    • The hierarchical parameters are organized into objects.
  2. FizzBuzz
    • Also features a minimal server.
  3. Prime Factors
    • By Object Mentor (probably, hopefully Robert C. Martin, aka Uncle Bob).
  4. Bowling Game
    • By Object Mentor (probably, hopefully Robert C. Martin, aka Uncle Bob).
  5. Conway's Game of Life
    • By me, without any prior knowledge, except for the rules themselves.
    • Since I struggle a bit when learning some of the more advanced Dart features — and because I created considerable overhead with the GridParser class —, this particular kata took me about 2 days to complete.
    • The tests only test for high-level behavior — we should not couple the tests with the production code by testing implementation —; there should have been more tests, strictly speaking.
    • The GridParser class was not necessary. I created it so string grids could be converted into objects in the background and, so, we had truly OOP software — but is it really?
    • When going through the grid, on the borders, I avoided the typical null error of having invalid indices by shielding the logic with a try-catch. A more ideal OOP solution would be to create an outer buffer layer of Null Cells (Null-Object Pattern), which would represent nothingness in terms of the rules.
  6. Sieve of Eratosthenes (Prime Generator)
    • By me, with some prior knowledge. I had already created a simple version of the algorithm when I was learning Python 1.5 years ago.
    • This is only one version of the sieve, there are many others much more optimized. This version features two optimizations:
      • Only using bases up to sqrt(N).
      • Starting the sieve on the base from base^2.
  7. Tic-Tac-Toe
    • The overall program is a bit (cough) overengineered. But it is good for practice, and, in the end, it even works for other board sizes than the typical 3x3 (there's already a test in place proving that it works for 4x4 boards).
      • Using asymmetrical boards will need some adaptation and refactoring, which I'm not interested in at the moment. The optional boardSize parameter for the game would have to be modified.
    • I completely dismissed the fact that this game is played by two people. What I mean is that I didn't create a player class to interact with the game itself. All of the plays are inserted directly into the game class. But fixing this to make things friendlier shouldn't be that hard.
      • It would be best if changes to the board were notified to the players' instances, with the game board being an observable.
      • The player class could be some sort of façade between the game class and the end-user.
    • The board property itself could be separated into another class and done a State pattern.
  8. Dungeon Problem
    • Based on William Fiset's formulation and solution. It's a Breadth-First Search (BFS) algorithm for finding the shortest path to the exit of a dungeon.
    • I added a way of extracting the complete shortest path also, not only the number of steps.
    • My implementation assumes there are a start, an end and it's possible to reach the end. It's not a huge leap to make it worth with dungeons without a reachable end, but I didn't want to unnecessarily complicate this more.
  9. Design Patterns
    • This is only at its beginning stages right now.
    1. Observer
      • So far, only the Observer pattern is featured. To achieve it I could have implemented the vanilla interfaces in the GoF book (I still intend on doing it). However, I was interested in knowing how to do it in a way that's more in tune with Dart. So, naturally, I chose to try doing it Streams. It isn't very trivial, native support isn't very straightforward at all and I needed Rémi Rousselet's help in order to make it work. You can find out more about it in this answer.

I've taken the liberty (good practice?) of refactoring the repetitive tests into for's with organized maps of input-outputs inside the mocks folder. If you want to check out how they looked like before this refactoring, look for the tag refactored_generalized_tests.

For more info about them, go to the docs or docs/presentations folder.

How do I run these tests?

There are two options:

  • pub run test <path_to_file>
    • pub run test . at the root of the repo will run all tests.
  • dart <path_to_file>