单元测试Django查询集

Vau*_*han 8 python django django-unittest python-unittest

我正在尝试用Django/unittest学习单元测试.

这些是我的模型的简单版本:

class Device(models.Model):
    name = models.CharField(max_length=100)

    def get_ips(self):
        return DeviceIP.objects.filter(device=self.id)


class DeviceIP(models.Model):
    ip = models.GenericIPAddressField()
    device = models.ForeignKey(Device)
Run Code Online (Sandbox Code Playgroud)

这是我提出的测试代码:

from django.test import TestCase

class DeviceTest(TestCase):

    def test_get_ips(self):
        device = Device()
        device.name = 'My Device'

        ip1 = DeviceIP()
        ip1.ip = '127.0.0.1'
        ip1.device = device
        ip1.save()

        ip2 = DeviceIP()
        ip2.ip = '127.0.0.2'
        ip2.device = device
        ip2.save()

        ip3 = DeviceIP()
        ip3.ip = '127.0.0.3'
        ip3.device = device
        ip3.save()

        self.assertEqual(device.get_ips(), [ip1, ip2, ip3])
Run Code Online (Sandbox Code Playgroud)

测试结果失败,因为AssertionError即使字符串表示device.get_ips()[ip1, ip2, ip3]相同.

如果我尝试使用self.assertListEqual我得到一个错误,因为device.get_ips()是一个QuerySet而不是一个列表.

如果我尝试self.assertQuerySetEqual我得到一个错误说" DeviceTest object has no attribute assertQuerySetEqual"但我不知道为什么因为DeviceTest扩展django.test的TestCase.

我该怎么做这样的测试?

此外,在"真实"项目中进行这样一个简单的测试是否有意义?

Pio*_*zek 8

实际上正确的方式,并由djangoproject推荐是:

    self.assertEqual(list(device.get_ips()), [ip1, ip2, ip3])
Run Code Online (Sandbox Code Playgroud)

强制查询集sorted和列表将改变您的测试场景,而您不希望它.


Sim*_*ser 6

该调用device.get_ips()返回一个QuerySet而是[ip1, ip2, ip3]一个列表.因此他们目前不平等.

鉴于您不想测试可能无关紧要的事情(.filter()从数据库返回哪些行的顺序),我建议测试如下:

results = device.get_ips()
result_ips = [ip.ip for ip in results]
self.assertEqual(len(results), 3)
self.assertTrue(ip1.ip in result_ips)
self.assertTrue(ip2.ip in result_ips)
self.assertTrue(ip3.ip in result_ips)
Run Code Online (Sandbox Code Playgroud)

这个测试:三个结果和IP是相同的.这应该让您有合理的信心,即获得相同的对象(尽管您可以根据需要添加更多断言).

  • 或者由于输入是唯一的,只需在断言中应用排序,它将处理列表长度和内容.`self.assertEqual(sorted(result_ips),sorted([ip1,ip,ip2.ip,ip3.ip]))` (2认同)