Python으로 코딩을 시작했고, 알고리즘도 Python으로 풀기 때문에 언어에 대한 자신감은 있었는데 생각해보니 프로그램을 구현한 경험은 거의 없었다. Class를 만드는 것도 어색하고 실습을 따라가기가 어색해서 Python 공부도 병행하면서 읽어야겠다.
Todo
테스트 메서드 호출하기
먼저 setUp 호출하기
나중에 tearDown 호출하기
테스트 메서드가 실패하더라도 tearDown 호출하기
여러 개의 테스트 실행하기
수집된 결과를 출력하기
테스트 메서드가 실행이 되었는지 알아보기 위해서 wasRun 클래스를 생성했다.
체크포인트를 만들어서 테스트하기전에는 'None' , 테스트를 마치면 '1'을 출력해주자
class wasRun:
def __init__(self,name):
self.wasRun= None
def testMethod(self):
self.wasRun = 1
test = wasRun("testMethod")
print(test.wasRun)
test.testMethod()
print(test.wasRun)
다음은 testMethod를 직접 호출하는 대신 진짜 인터페이스인 run()을 사용한다.
class wasRun(TestCase):
def __init__(self, name):
self.wasRun= None
self.name = name
def run(self):
method = getattr(self, self.name) #메서드 가져오기
method() #메서드 실행
TestCase 클래스는 부모클래스로,
WasRun은 작동했는지 여부를,
TestCaseTest는 참, 거짓 여부를 알려준다.
class TestCase:
def __init__(self, name):
self.name = name
def run(self):
method = getattr(self, self.name)
method()
class wasRun(TestCase):
def __init__(self, name):
self.wasRun= None
TestCase.__init__(self, name)
def testMethod(self):
self.wasRun = 1
class TestCaseTest(TestCase):
def testRunning(self):
test = wasRun("testMethod")
assert (not test.wasRun)
test.run()
assert (test.wasRun)
TestCaseTest("testRunning").run()
테스트를 하다보면 발생하는 3가지 패턴 (Bill Wake의 3A)
2,3 단계는 다르지만 1단계는 공통으로 가져가는 경우가 있다.
ex) 객체 7,3을 통한 덧셈과 뺄셈
객체를 생성할 때 두 가지를 고려한다.
테스트 사이의 커플링을 만들면 안된다. 객체가 충분히 빨리 생성된다는 가정하에 코드를 수정한다.
class TestCase:
def __init__(self, name):
print("name= ", name)
self.name = name
def setUp(self):
pass
def run(self):
self.setUp()
method = getattr(self, self.name)
method()
class WasRun(TestCase):
def __init__(self, name):
self.wasRun= None
TestCase.__init__(self, name)
def testMethod(self):
self.wasRun = 1
def setUp(self):
self.wasRun = None
self.wasSetUp = 1
class TestCaseTest(TestCase):
def setUp(self):
self.test = WasRun("testMethod")
def testRunning(self):
self.test.run()
assert (self.test.wasRun)
def testSetUp(self):
self.test.run()
assert (self.test.wasSetUp)
TestCaseTest("testRunning").run()
TestCaseTest("testSetUp").run()
TestCaseTest("testRunnint").run()
의 작동원리
setUp()을 외부 자원을 할당하는 경우가 있다. 각 테스트가 독립적이길 바란다면 tearDown() 메서드 같은 곳에서 자원을 다시 반환할 필요가 있다.
class TestCase:
def __init__(self, name):
self.name = name
def setUp(self):
pass
def tearDown(self):
pass
def run(self):
self.setUp()
method = getattr(self, self.name)
method()
self.tearDown()
class WasRun(TestCase):
def testMethod(self):
self.wasRun = 1
self.log = self.log + "testMethod "
def setUp(self):
self.wasRun = None
self.log= "setUp "
def tearDown(self):
self.log = self.log + "tearDown "
class TestCaseTest(TestCase):
def testTemplateMethod(self):
test = WasRun("testMethod")
test.run()
assert ("setUp testMethod tearDown " == test.log)
TestCaseTest("testTemplateMethod").run()
목표 : 테스트 실행 갯수와 실패 여부 보여주기
5 run, 2 failed
와 같은 형식으로
따로 실행 결과를 기록하는 TestResult 객체를 반환하게 만든다.
class TestCase:
def __init__(self, name):
self.name = name
def setUp(self):
pass
def tearDown(self):
pass
def run(self):
result = TestResult()
result.testStarted()
self.setUp()
method = getattr(self, self.name)
method()
self.tearDown()
return result
class WasRun(TestCase):
def testMethod(self):
self.wasRun = 1
self.log = self.log + "testMethod "
def setUp(self):
self.wasRun = None
self.log= "setUp "
def tearDown(self):
self.log = self.log + "tearDown "
def testBrokenMethod(self):
raise Exception
class TestCaseTest(TestCase):
def testResult(self):
test = WasRun("testMethod")
result = test.run()
assert ("1 run, 0 failed" == result.summary())
def testFailedResult(self):
test = WasRun("testBrokenMethod")
result = test.run()
assert ("1 run, 1 failed" == result.summary())
class TestResult:
def __init__(self):
self.runCount = 0
def testStarted(self):
self.runCount = self.runCount + 1
def summary(self):
return "%d run, 0 failed" % self.runCount
TestCaseTest("testResult").run()
TestCaseTest("testFailedResult").run()
테스트가 실패한 경우 Count가 있어야하기 때문에 속성을 추가한다.
테스트를 run하는 중에 fail이 발생한다면 늘어날 수 있도록 설정한다.
class TestCase:
def __init__(self, name):
self.name = name
def setUp(self):
pass
def tearDown(self):
pass
def run(self):
result = TestResult()
result.testStarted()
self.setUp()
try:
method = getattr(self, self.name)
method()
except:
result.testFailed()
self.tearDown()
return result
class WasRun(TestCase):
def testMethod(self):
self.wasRun = 1
self.log = self.log + "testMethod "
def setUp(self):
self.wasRun = None
self.log= "setUp "
def tearDown(self):
self.log = self.log + "tearDown "
def testBrokenMethod(self):
raise Exception
class TestCaseTest(TestCase):
def testResult(self):
test = WasRun("testMethod")
result = test.run()
assert ("1 run, 0 failed" == result.summary())
def testFailedResult(self):
test = WasRun("testBrokenMethod")
result = test.run()
assert ("1 run, 1 failed" == result.summary())
def testFailedResultFormatting(self):
result = TestResult()
result.testStarted()
result.testFailed()
assert ("1 run, 1 failed" == result.summary())
class TestResult:
def __init__(self):
self.runCount = 0
self.failureCount = 0
def testStarted(self):
self.runCount = self.runCount + 1
def testFailed(self):
self.failureCount = self.failureCount + 1
def summary(self):
return "%d run, 0 failed" % (self.runCount, self.failureCount)
TestCaseTest("testResult").run()
TestCaseTest("testFailedResult").run()
test를 여러번 해야하는 번거로움이 있다. 한번에 테스트를 하나만 할 수 있다면 독립적으로 돌아가는 의미가 없다. TestSuite를 구현하여 모두 실행할 수 있도록 한다.
class TestSuite:
def __init__(self):
self.tests = []
def add(self, test):
self.tests.append(test)
def run(self, result):
for test in self.tests:
test.run(result)
suite = TestSuite()
suite.add(TestCaseTest("testTemplateMethod"))
suite.add(TestCaseTest("testResult"))
suite.add(TestCaseTest("testFailedResultFormatting"))
suite.add(TestCaseTest("testFailedResult"))
suite.add(TestCaseTest("testSuite"))
result = TestResult()
suite.run(result)
print(result.summary())
TestCaseTest를 하는 중에 발생하는 중복들을 정리했다.
최종코드
class TestCase:
def __init__(self, name):
self.name = name
def setUp(self):
pass
def tearDown(self):
pass
def run(self, result):
result.testStarted()
self.setUp()
try:
method = getattr(self, self.name)
method()
except:
result.testFailed()
self.tearDown()
class WasRun(TestCase):
def testMethod(self):
self.wasRun = 1
self.log = self.log + "testMethod "
def setUp(self):
self.wasRun = None
self.log= "setUp "
def tearDown(self):
self.log = self.log + "tearDown "
def testBrokenMethod(self):
raise Exception
class TestCaseTest(TestCase):
def setUp(self):
self.result = TestResult()
def testTemplateMethod(self):
test = WasRun("testMethod")
test.run(self.result)
assert ("setUp testMethod tearDown " == test.log)
def testResult(self):
test = WasRun("testMethod")
test.run(self.result)
assert ("1 run, 0 failed" == self.result.summary())
def testFailedResult(self):
test = WasRun("testBrokenMethod")
test.run(self.result)
assert ("1 run, 1 failed" == self.result.summary())
def testFailedResultFormatting(self):
self.result.testStarted()
self.result.testFailed()
assert ("1 run, 1 failed" == self.result.summary())
def testSuite(self):
suite = TestSuite()
suite.add(WasRun("testMethod"))
suite.add(WasRun("testBrokenMethod"))
suite.run(self.result)
assert ("2 run, 1 failed" == self.result.summary())
class TestResult:
def __init__(self):
self.runCount = 0
self.failureCount = 0
def testStarted(self):
self.runCount = self.runCount + 1
def testFailed(self):
self.failureCount = self.failureCount + 1
def summary(self):
return "%d run, %d failed" %(self.runCount, self.failureCount)
class TestSuite:
def __init__(self):
self.tests = []
def add(self, test):
self.tests.append(test)
def run(self, result):
for test in self.tests:
test.run(result)
suite = TestSuite()
suite.add(TestCaseTest("testTemplateMethod"))
suite.add(TestCaseTest("testResult"))
suite.add(TestCaseTest("testFailedResultFormatting"))
suite.add(TestCaseTest("testFailedResult"))
suite.add(TestCaseTest("testSuite"))
result = TestResult()
suite.run(result)
print(result.summary())
각 언어별로 xUnit은 이미 잘 구현되어있다. 하지만 직접 구현해봐야하는 이유는 두가지이다.