load_tests() not called in unittests

1.7k Views Asked by At

I am trying to use the load_tests protocol to parameterize some test cases. Here is my MCVE:

# mytests.py
import unittest

class MyTests(unittest.TestCase):
  def __init__(self, x, y):
    super(MyTests, self).__init__("testMyMethod")
    self.x = x
    self. y = y

  def testMyMethod(self):
    self.assertEqual(self.y, my_method(self.x))

def load_tests(loader, tests, pattern):
  print("load_tests()")
  foobar = zip(range(5), range(5))
  test_suite = unittest.TestSuite()
  for x, y in foobar:
    test_suite.addTest(MyTests(x, y))
  return test_suite

def my_method(x):
  return x

if __name__ == "__main__":
  unittest.main()

When I run this I get the following output:

myrdraal% python mytests.py 
Traceback (most recent call last):
  File "mytests.py", line 24, in <module>
    unittest.main()
  File "/usr/lib/python3.5/unittest/main.py", line 93, in __init__
    self.parseArgs(argv)
  File "/usr/lib/python3.5/unittest/main.py", line 140, in parseArgs
    self.createTests()
  File "/usr/lib/python3.5/unittest/main.py", line 144, in createTests
    self.test = self.testLoader.loadTestsFromModule(self.module)
  File "/usr/lib/python3.5/unittest/loader.py", line 123, in loadTestsFromModule
    tests.append(self.loadTestsFromTestCase(obj))
  File "/usr/lib/python3.5/unittest/loader.py", line 92, in loadTestsFromTestCase
    loaded_suite = self.suiteClass(map(testCaseClass, testCaseNames))
  File "/usr/lib/python3.5/unittest/suite.py", line 24, in __init__
    self.addTests(tests)
  File "/usr/lib/python3.5/unittest/suite.py", line 57, in addTests
    for test in tests:
TypeError: __init__() missing 1 required positional argument: 'y'

It appears that my load_tests() is not even being called. What am I doing wrong?

A similar MCVE runs to just fine:

# parameterized.py
import unittest

class ParameterizedTest(unittest.TestCase):
  def __init__(self, value):
    super(ParameterizedTest, self).__init__("test")
    self.value = value

  def test(self):
    self.assertEqual(self.value, self.value)

def load_tests(loader, tests, pattern):
  test_suite = unittest.TestSuite()
  for i in range(5):
    test_suite.addTest(ParameterizedTest(i))
  return test_suite

if __name__ == "__main__":
  unittest.main()

Output 2:

myrdraal% python parameterized.py
.....
----------------------------------------------------------------------
Ran 5 tests in 0.001s

OK

Apparently I am missing something because these look similar to me.

1

There are 1 best solutions below

0
On

The problem appears to be that the unittest framework calls __init__() on each subclass of unittest.TestCase for each method which starts with "test". It passes a single parameter which is a string containing the test method's name. This all occurs before calling load_tests(). One solution is to modify __init__() to allow this mechanism to run:

class MyTests(unittest.TestCase):
  def __init__(self, testName, x=None, y=None):
    super(MyTests, self).__init__(testName)
    self.x = x
    self. y = y

  def testMyMethod(self):
    self.assertEqual(self.y, my_method(self.x))

def load_tests(loader, tests, pattern):
  print("load_tests()")
  foobar = zip(range(5), range(5))
  test_suite = unittest.TestSuite()
  for x, y in foobar:
    test_suite.addTest(MyTests('testMyMethod', x, y))
  return test_suite