在Python中对TestCase进行子类化:在Parent TestCase中覆盖字段

时间:2018-04-29 20:12:56

标签: python unit-testing testing python-unittest

我正在为Alexa应用编写集成测试。

我们的应用程序使用控制器请求 - 响应模式。控制器接收具有指定intent和会话变量的请求,将请求路由到使用会话变量进行某些计算的函数,并返回带有该计算结果的响应对象。

就test_for_smoke而言,我们从UnhandledIntentTestCase获得正确的行为。但是,test_returning_reprompt_text 永远不会触发,因为returns_reprompt_text永远不会被覆盖。

有人可以解释我如何在父类和/或中覆盖它 如何将正确的意图名称传递给setUpClass中的请求对象?

intent_base_case.py

import unittest

import mycity.intents.intent_constants as intent_constants
import mycity.mycity_controller as mcc
import mycity.mycity_request_data_model as req
import mycity.test.test_constants as test_constants




###############################################################################                                                                
# TestCase parent class for all intent TestCases, which are integration tests #                                                                
# to see if any changes in codebase have broken response-request model.       #                                                                
#                                                                             #                                                                
# NOTE: Assumes that address has already been set.                            #                                                                
###############################################################################                                                                

class IntentBaseCase(unittest.TestCase):

    __test__ = False

    intent_to_test = None
    returns_reprompt_text = False

    @classmethod
    def setUpClass(cls):
        cls.controller = mcc.MyCityController()
        cls.request = req.MyCityRequestDataModel()
        key = intent_constants.CURRENT_ADDRESS_KEY
        cls.request._session_attributes[key] = "46 Everdean St"
        cls.request.intent_name = cls.intent_to_test
        cls.response = cls.controller.on_intent(cls.request)

    @classmethod
    def tearDownClass(cls):
        cls.controller = None
        cls.request = None

    def test_for_smoke(self):
        self.assertNotIn("Uh oh", self.response.output_speech)
        self.assertNotIn("Error", self.response.output_speech)

    def test_correct_intent_card_title(self):
        self.assertEqual(self.intent_to_test, self.response.card_title)


    @unittest.skipIf(not returns_reprompt_text,
                     "{} shouldn't return a reprompt text".format(intent_to_test))
    def test_returning_reprompt_text(self):
        self.assertIsNotNone(self.response.reprompt_text)


    @unittest.skipIf(returns_reprompt_text,
                   "{} should return a reprompt text".format(intent_to_test))
    def test_returning_no_reprompt_text(self):
        self.assertIsNone(self.response.reprompt_text)

test_unhandled_intent.py

import mycity.test.intent_base_case as base_case


########################################                                                                                                       
# TestCase class for unhandled intents #                                                                                                       
########################################                                                                                                       


class UnhandledIntentTestCase(base_case.IntentBaseCase):

    __test__ = True

    intent_to_test = "UnhandledIntent"
    returns_reprompt_text = True

输出

======================================================================
FAIL: test_correct_intent_card_title (mycity.test.test_unhandled_intent.UnhandledIntentTestCase)
----------------------------------------------------------------------
Traceback (most recent call last):
  File "/Users/wdrew/projects/alexa_311/my_city/mycity/mycity/test/intent_base_case.py", line 44, in test_correct_intent_card_title
    self.assertEqual(self.intent_to_test, self.response.card_title)
AssertionError: 'UnhandledIntent' != 'Unhandled intent'
- UnhandledIntent
?          ^
+ Unhandled intent
?          ^^


======================================================================
FAIL: test_returning_no_reprompt_text (mycity.test.test_unhandled_intent.UnhandledIntentTestCase)
----------------------------------------------------------------------
Traceback (most recent call last):
  File "/Users/wdrew/projects/alexa_311/my_city/mycity/mycity/test/intent_base_case.py", line 56, in test_returning_no_reprompt_text
    self.assertIsNone(self.response.reprompt_text)
AssertionError: 'So, what can I help you with today?' is not None

----------------------------------------------------------------------

1 个答案:

答案 0 :(得分:0)

这是因为执行顺序。在解析SkipIf类期间,IntentBaseCase装饰器执行一次。它们不会针对每个类或每次调用测试函数而重新执行。

SkipIf的装饰器模式设计用于固定的全局变量,例如依赖模块的版本,操作系统或其他可在全局上下文中计算或知道其可用性的外部资源。

跳过测试也应该出于外部原因而不是内部因素,例如子类的需要。跳过仍然是报告中指出的一种失败测试,​​因此您可以看到您的测试套件没有执行项目的整个功能范围。

您应该重新设计基类结构,以便只有子类可以运行函数并跳过使用Skip。我的建议是:

class IntentBaseCase(unittest.TestCase):
    ...

class RepromptBaseCase(IntentBaseCase):
    def test_returning_reprompt_text(self):
        self.assertIsNotNone(self.response.reprompt_text)

class NoRepromptBaseCase(IntentBaseCase):
    def test_returning_no_reprompt_text(self):
        self.assertIsNone(self.response.reprompt_text)

您还应该考虑将响应部分移出setUp并将其放入自己的test_函数中,并将这些test_returning函数更改为更简单的assertRepromptassertNoReprompt函数。在setUp中设置测试是个好主意,但在那里运行实际代码并不是一个好主意。