在Python中将Mock用于嵌套对象(DynamoDB和Table)

时间:2018-08-13 21:13:06

标签: python mocking python-unittest

我想用DynamoDb的不同输入来测试函数的行为。主要有两种行为:在表中找到搜索键时和在表中找不到搜索键时。这是该函数的最小代码:

import boto3
from boto3.dynamodb.conditions import Key

def main(symbol):
   dynamo = boto3.resource("dynamodb")
   table = dynamo.Table("mytable")
   data = table.query(KeyConditionExpression=Key("symbol").eq(symbol))

   if data.count > 0:
      # result = some_output
   else:
      result = {'status': '404'}
   return result

我想通过发送空结果和非空项目列表来进行单元测试来测试此代码,如下所示:

import boto3
import unittest
from unittest.mock import Mock, patch

class TestMainHandler(unittest.TestCase): 
   ...
   def test_main_fails_on_wrong_symbol(self):
       with patch.object(main_handler, 'table') as get_mock:
          get_mock.return_value = []
          result = main('dummy_symbol')
          expect_result = {'status': '404'}
   self.assertEqual(result, expect_result)

但是我不能运行模拟部分。我想知道您是否可以指导我如何模拟嵌套表和发电机变量。非常感谢您的帮助。

1 个答案:

答案 0 :(得分:1)

我建议您可以模拟第一个“集成点”,然后从中构建模拟结果。在这种情况下,它将是boto3.resource。从那里,您可以将boto3.resource的返回值修改为模拟表。然后,您可以将模拟表上任何调用的返回值更改为预期的结果。

import boto3
import unittest
from unittest.mock import Mock, patch

class TestMainHandler(unittest.TestCase): 
    @patch('boto3.resource')
    def test_main_fails_on_wrong_symbol(self, mock_dynamo):
        mock_table = Mock()
        mock_table.query.return_value = []
        mock_dynamo.Table.return_value = mock_table

        result = main('dummy_symbol')
        expected_result = {'status': '404'}
        self.assertEqual(expected_result, result)

还要注意,我在测试用例而不是上下文管理器上使用了补丁装饰器。这是我的偏爱,但我认为它看起来更干净。

编辑:为mock_dynamo函数调用分配返回值时出现错误。我还填写了您的if,以测试查询数据的长度并返回一些内容。但是,此处的测试未达到该分支。这是最终产品:

"""boto_main.py"""
import boto3
from boto3.dynamodb.conditions import Key

def main(symbol):
   dynamo = boto3.resource("dynamodb")
   table = dynamo.Table("mytable")
   data = table.query(KeyConditionExpression=Key("symbol").eq(symbol))
   if len(data) > 0:
      result = {'status': '200'}
   else:
      result = {'status': '404'}
   return result

"""boto_test.py"""
import unittest
from unittest.mock import Mock, patch
from boto_main import main

class TestMainHandler(unittest.TestCase): 
    @patch('boto3.resource')
    def test_main_fails_on_wrong_symbol(self, mock_dynamo):
        mock_table = Mock()
        mock_table.query.return_value = []
        mock_dynamo.return_value.Table.return_value = mock_table
        result = main('dummy_symbol')
        expected_result = {'status': '404'}
        self.assertEqual(expected_result, result)