In the following video, you will see a graph and how in time we expand the code using the Outside-in TDD
Requirements
We have an web API where we should be able to write and read articles and their comments.
Acceptance Test
We save an article:
POST /api/1/save/Article
{
"title":"New article",
"content":"The content"
}
and we receive the article + id
{
id: 13, - can be any value
title:"New article",
content:"The content"
}
We then add 2 new comments:
POST /api/1/save/Article
{
id: 13, - can be any value
comments:[
{"comment":"This was awesome!"},
{"comment":"I loved it as well!"},
]
}
response:
{
id: 13, - can be any value
title:"New article",
content:"The content"
}
and now reading the articles and the comments:
POST api/1/query/Article
{
find: [id,title,content,{comments:[id,comment]}],
where:{id:13}
}
should return
{"13":{
id: 13, - can be any value
comments:{
"113":{"comment":"This was awesome!"},
"114":{"comment":"I loved it as well!"},
}
}
}
Outside-In TDD approach
Unlike classicist TDD, in Outside-In, you start the code from the outside, designing it along the way. You start with an acceptance test, then with the unit/integration tests and code needed until you make the acceptance test work:
data:image/s3,"s3://crabby-images/33e40/33e403b2d716567a65af4fbda30aae76181c80b6" alt=""
So let's begin:
Step 1: Acceptance test
data:image/s3,"s3://crabby-images/d0397/d0397270d65ec99062c580fb9da935e8e1f13346" alt=""
data:image/s3,"s3://crabby-images/f9560/f95603ebb2d7f8fcc29d0a4f678bfb16f9eaaea2" alt=""
When we run it:
data:image/s3,"s3://crabby-images/c44a8/c44a88a86f00ac251ad0ef6a24b395507b74f6fc" alt=""
2. Step 2 moving in,
From the outside, writing the first test for the service that will handle the url. A little bit of design, while writing the test: we consider, that we will use a function that will get the object from the database if we have an id in the json or will give us a brand new Article object if we have no id in the json.
data:image/s3,"s3://crabby-images/4aeb4/4aeb49c5513f4648fb697581e4ecdb06bcc17963" alt=""
data:image/s3,"s3://crabby-images/34b8a/34b8aae8050aca5985aad3d5c391a33997a93958" alt=""
data:image/s3,"s3://crabby-images/03d54/03d545ff1f07cc11c4afd30127c109b0f96164d2" alt=""
data:image/s3,"s3://crabby-images/978d2/978d2a2297d05eb59251b9c1607136d13700af36" alt=""
Now if we properly do the code, the test will pass:
data:image/s3,"s3://crabby-images/d313c/d313c7802c17ae751cef2e49f39961493749427f" alt=""
data:image/s3,"s3://crabby-images/093fd/093fd219875ac89c87dc2f9b2885e4418d99c947" alt=""
data:image/s3,"s3://crabby-images/4f87b/4f87b6c6651afa715eb8663dfc9586a0a1fd7328" alt=""
Step 3: moving furher in, database_services and get_database_object
However, we did not implement the get_database_object, which will work with the database. So for that we'll write the tests first:
data:image/s3,"s3://crabby-images/e4dd6/e4dd647fe00ff26eebbfaa53c489d321da63f181" alt=""
or:
data:image/s3,"s3://crabby-images/3e47f/3e47f2871c74291fbafa634fc8458e513448f348" alt=""
and
data:image/s3,"s3://crabby-images/b7d7f/b7d7f13fdb8cee7cb4057e8bb765d651af49183b" alt=""
data:image/s3,"s3://crabby-images/f87b2/f87b20a619c5b2b680247a0e49f931adaa645cbf" alt=""
Now let's write the code to make it pass:
data:image/s3,"s3://crabby-images/d3ec8/d3ec828ed240c256b470f10c69723eb5a0dc975f" alt=""
data:image/s3,"s3://crabby-images/59c51/59c51dc771ff43f5b78e0b309e658e51227e1033" alt=""
data:image/s3,"s3://crabby-images/9bf21/9bf21420e8c6e81a8c2af6c7c4e8c1c70488c36f" alt=""
We write in fact 2 tests to cover both scenarios: when editing or when adding a new article.
Step 4: now we move back and extend the api
We design it to read the data from json and transfer it to the database object. For that we'll use a function transfer_from_json
data:image/s3,"s3://crabby-images/59652/59652e0966807c8f1e8abe70c9eee7be9e743058" alt=""
data:image/s3,"s3://crabby-images/4b042/4b042af0b231a206250e5f2d5d7fb24f28ae2ed9" alt=""
Once it passes, we move further:
Step 5: writing the transfer service
data:image/s3,"s3://crabby-images/e654a/e654a99688d68488291ce92d8fc94fbb7ac01031" alt=""
data:image/s3,"s3://crabby-images/fa102/fa10294aada60cc292049910cd2d7534a0586b8d" alt=""
data:image/s3,"s3://crabby-images/c0607/c0607ea66f3e7f98078dfd1e920bc650f519515a" alt=""
data:image/s3,"s3://crabby-images/c4a62/c4a62f65da25b2e798c543ced114142d5f488248" alt=""
Now we make the test pass:
data:image/s3,"s3://crabby-images/ae0df/ae0df53e47afbbc256d51bc467591033d94f9001" alt=""
Step 6: back to extending the api
We now need to save the object with the data from the json into the db
data:image/s3,"s3://crabby-images/03d7b/03d7bc41b50685ba3a6067ec99cc2eab38b45a7e" alt=""
For this, we extend the test:
data:image/s3,"s3://crabby-images/ed58a/ed58a84ec9fb0d15ea4d0306399efa83c84bfc5b" alt=""
and make it pass, then we move back to implementing using TDD the save_database_object
Step 7 extending the database_service to also save
data:image/s3,"s3://crabby-images/8674f/8674fe4db72bb83547178e1613318d45a599c8e1" alt=""
Once we write the test, the code and make it pass:
data:image/s3,"s3://crabby-images/e356e/e356ec8b5f1a697607f0158fc6f54a62847fa6d9" alt=""
we move back to the api
Step 7 extend the api
data:image/s3,"s3://crabby-images/c9436/c9436db2a48aaa571c8c0868a4538dbe09454e57" alt=""
data:image/s3,"s3://crabby-images/ef22f/ef22fa1b629cd3fe5862091e181241ea978f3dc7" alt=""
Step 8: TDD the extension:
data:image/s3,"s3://crabby-images/5ba52/5ba522384787555196484a5d105b85e4772fc05b" alt=""
Until we make it pass:
data:image/s3,"s3://crabby-images/43fad/43fad52f52fd700219f6b4453a72b7efd3c8f22e" alt=""
After a few more intermediary steps:
...
We will TDD all the necessary code to make the acceptance test pass:
data:image/s3,"s3://crabby-images/329ed/329edc3c7fe24fd49c3abb0c6bc8b2f23b623f55" alt=""