Overview
이 프로젝트는 tokyo-tyrant의 java client를 만드는 시작 부분을 TDD로 진행하는 과정을 보여줍니다
Kent Beck의 원본 video를 저자의 양해를 구해서 약간의 각색, 부연 설명을 추가했습니다
이 예제는 mac, home brew, IntelliJ 를 사용하는 것을 가정하고 진행됩니다.
1. Create TODO List
Kent Beck은 TDD의 첫번째 단계는 해야 할 목록 즉 TODO 목록을 작성하는 것이라고 합니다.
아래와 같이 TODO를 작성.
TODO
Function List
--------------------------------
* put(0x10)
* get(0x30)
* remove(0x20)
* vanish(0x72)
* iterator
* size(0x80)
* reset(0x50)
* get next key(0x51)
* not found(1)
* success(0)
port 1978
--------------------------------
packet structure
--------------------------------
1 OPERATION_PREFIX(0xC8)
1 OPERATION_CODE
4 key length
4 value length
n key
n value
TODO에는 구현할 기능 목록, tyrant 접속 포트번호, TCP Packet 구조 등을 기록.
1.5 tutorial branch 만들기
git checkout -b diy diy
2. Start with High Level Test
원하는 것이 무엇인지를 표현하는 상위 레벨의 테스트로 시작한다.
이때 assert 부터 반대 순으로 테스트를 작성한다.
그리고 intellij의 show intention actions(opt+enter)를 이용해서 진행한다.
public class TyrantMapTest {
@Test
public void get_retrives_what_was_put() {
TyrantMap map = new TyrantMap(); // step 2.1
byte[] key = "key".getBytes(); // step 2.2
byte[] value = "value".getBytes(); // step 2.3
assertThat(map.get(key), is(value)); // step 1
}
private class TyrantMap { // step 2.4
public byte[] get(byte[] key) { // step 2.5
return new byte[0];
}
}
}
3. Commnet Out
이 테스트가 우리가 궁극적으로 원하는 바이지만,
이걸 구현하려면 수십분이 걸리고, 테스트가 실패한다면 어디에 오류가 있는지 계속 따라 들어가면서 찾아야 한다.
그럴려면 수시간이 걸릴 수 있다. ㅠㅠ
제일 먼저 실행해 볼 수 있는 쉬운 일로 시작해야 한다.
원하는 바를 잘 기억하기 위해 지우지 않고 comment out만 하고,
문제를 작게 나눠서 짧은 싸이클로 개발하고 피드백을 얻을 수 있도록 한다.
4. 제일 쉽고, 바로 실행해 볼 수 있는 일로 시작
무엇을 제일 먼저 해 봐야 할까 ?
socket 접속이 되는지 부터 확인한다.
new Socket("localhost", 1978);
그리고 실행해 본다.
tyrant 서버가 기동되어 있지 않아 java.net.ConnectException: Connection refused
이 발생한다.
4.1 tokyo-tyrant 설치 및 기동
brew install homebrew/boneyard/tokyo-tyrant
https://github.com/Homebrew/homebrew-boneyard 를 참고해서 home brew repo에서 제거된 패키지 설치
아래와 같이 서버 기동
/usr/local/Cellar/tokyo-tyrant/1.1.41_1/bin $ ./ttserver
이제 테스트는 성공한다.
5. put해 본다.
소켓 접속이 되면 put을 해 보고 제대로 put되었는지 확인해 보고 싶을 것이다.
코드를 작성하고 실행해 본다
Socket s = new Socket("localhost", 1978);
OutputStream writer = s.getOutputStream();
writer.write(0xC8); // operation prefix
writer.write(0x10); // put operation
writer.write(0);
writer.write(0);
writer.write(0);
writer.write(3); // 4 byte
writer.write(0);
writer.write(0);
writer.write(0);
writer.write(5); // 4 byte
writer.write(new byte [] {'k', 'e', 'y'}); // key
writer.write(new byte [] {'v', 'a', 'l', 'u', 'e'}); // value
InputStream reader = s.getInputStream();
int status = reader.read();
assertThat(status, is(0));
6. Refactoring
모든 코드는 이야기 하듯이 작성되어야 한다(잘 쓰여진 산문처럼 읽을 수 있어야 한다).
그런데 이 코드는 테스트가 무엇을 하는지 이야기하고 있지 않고 어떻게 구현되었는지를 드러내고 있다(구현 디테일을 드러내고 있다).
커멘트 처리한 처음 3줄이 테스트가 무엇을 하는지에 대한 이야기이다.
6.1 Extract Method Object - TyrantMap
Detail 코드를 객체로 추출한다
6.2 more refactoring
- rename invoke to put
- extract field(OPERATION_PREFIX, PUT_OPERATION)
- remove unnecessary comments(duplicated)
- extract parameter(key, value)
6.3 Introduce DataInputStream and DataOutputStream
6.4 extract method open(bottom up)
- move line up(reader)
- extract fields
6.5 implement method close(top down)
6.6 call open/close
- extract variable
- call open from test not in put
- extract variable(key, value)
7. implement get
- add assert
- implement by copying from put
8. refactoring
- extract field(map)
- extract methods(setUp, tearDown)
이제 최초 작성했던 High Level Test와 동일한 모습을 갖게 되었다.
8.1 extract method writeOperation
8.2 Move Inner to upper level
- move-inner-to-upper-level
- fix compile errors
- move down private methods
- remove dependency to junit, hamcrest
8.3 extract more methods
- extract method validateStatus and move down
- extract method readBytes
9. 나머지들
TyrantMapFinalTest.java에 나머지 모든 테스트가 순차적으로 들어있고, 테스트를 추가하면서 구현한 기능이 있다.
나머지 기능들은 한번씩 스스로 해 보면 재밌을 것 같다 ^^