mdi*_*kin 2 python unit-testing mocking
我是 Python 新手,但我已经用 C# 和 JavaScript 进行了大量单元测试。我在弄清楚 Python 中的模拟框架时遇到了麻烦。这是我所拥有的(精简):
发票业务.py
import ims.repository.invoice_repository as invoiceRepository
import logging
logger = logging.getLogger(__name__)
def update_invoice_statuses(invoices):
for invoice in invoices:
dbInvoice = invoiceRepository.get(invoice.invoice_id)
print("dbInvoice is %s" % dbInvoice) #prints <MagicMock etc.>
if dbInvoice is None:
logger.error("Unable to update status for invoice %d" % invoice.invoice_id)
continue;
Run Code Online (Sandbox Code Playgroud)
test_invoice_business.py
from unittest import TestCase, mock
import logging
import ims.business.invoice_business as business
class UpdateInvoiceTests(TestCase):
@mock.patch("ims.business.invoice_business.invoiceRepository")
@mock.patch("ims.business.invoice_business.logger")
def test_invoiceDoesNotExist_logsErrorAndContinues(self, invoiceRepoMock, loggerMock):
#Arrange
invoice = Invoice(123)
invoice.set_status(InvoiceStatus.Filed, None)
invoiceRepoMock.get.return_value(33)
#Act
business.update_invoice_statuses([invoice])
#Assert
invoiceRepoMock.get.assert_called_once_with(123)
loggerMock.error.assert_called_once_with("Unable to update status for invoice 123")
Run Code Online (Sandbox Code Playgroud)
测试失败
断言错误:预期调用一次“get”。调用了 0 次。
但是,中的print语句update_invoice_statuses被击中,因为我看到了
dbInvoice is <MagicMock name='invoiceRepository.get()' id='xxxx'>
Run Code Online (Sandbox Code Playgroud)
知道我在这里做错了什么吗?
编辑:在@chepner 的帮助下,我遇到了另一个断言错误并意识到这是因为我应该使用invoiceRepoMock.get.return_value = None而不是.return_value(None)
测试函数的模拟参数被交换。首先应用内部装饰器(用于记录器),因此模拟记录器应该是您方法的第一个参数。
@mock.patch("ims.business.invoice_business.invoiceRepository")
@mock.patch("ims.business.invoice_business.logger")
def test_invoiceDoesNotExist_logsErrorAndContinues(self, loggerMock, invoiceRepoMock):
...
Run Code Online (Sandbox Code Playgroud)