Ответ 1
Используйте assertQuerysetEqual, который построен для сравнения двух запросов для вас. Вам потребуется подкласс Django django.test.TestCase
чтобы он был доступен в ваших тестах.
Я пытаюсь проверить свои взгляды на Джанго. Это представление передает QuerySet в шаблон:
def merchant_home(request, slug):
merchant = Merchant.objects.get(slug=slug)
product_list = merchant.products.all()
return render_to_response('merchant_home.html',
{'merchant': merchant,
'product_list': product_list},
context_instance=RequestContext(request))
и проверить:
def test(self):
"Merchant home view should send merchant and merchant products to the template"
merchant = Merchant.objects.create(name='test merchant')
product = Product.objects.create(name='test product', price=100.00)
merchant.products.add(product)
test_client = Client()
response = test_client.get('/' + merchant.slug)
# self.assertListEqual(response.context['product_list'], merchant.products.all())
self.assertQuerysetEqual(response.context['product_list'], merchant.products.all())
РЕДАКТИРОВАТЬ Я использую self.assertQuerysetEqua
l вместо self.assertListEqual
. К сожалению, это все еще не работает, и терминал отображает это: ['<Product: Product object>'] != [<Product: Product object>]
assertListEqual
повышает: 'QuerySet' object has no attribute 'difference'
и assertEqual
также не работает, хотя self.assertSetEqual(response.context['product_list'][0], merchant.products.all()[0])
проходит,
Я предполагаю, что это потому, что QuerySets - это разные объекты, даже если они содержат одинаковые экземпляры модели.
Как проверить, что два QuerySet содержат одинаковые данные? Я даже проверяю это правильно? Это мой 4-й день обучения Джанго, поэтому я хотел бы узнать лучшие практики, если это возможно. Благодарю.
Используйте assertQuerysetEqual, который построен для сравнения двух запросов для вас. Вам потребуется подкласс Django django.test.TestCase
чтобы он был доступен в ваших тестах.
По умолчанию assertQuerysetEqual
использует repr()
для первого аргумента. Вот почему у вас возникли проблемы со строками в сравнении наборов запросов.
Чтобы обойти это, вы можете переопределить аргумент transform
с помощью lambda
функции, которая не использует repr()
:
self.assertQuerysetEqual(queryset_1, queryset_2, transform=lambda x: x)
У меня была такая же проблема. Второй аргумент assertQuerysetEqual
должен быть списком ожидаемых строк repr() s как строк. Вот пример из набора тестов Django:
self.assertQuerysetEqual(c1.tags.all(), ["<Tag: t1>", "<Tag: t2>"], ordered=False)
Я self.assertQuerysetEqual
решить эту проблему, используя map
для repr()
каждой записи в self.assertQuerysetEqual
внутри вызова self.assertQuerysetEqual
, например
self.assertQuerysetEqual(queryset_1, map(repr, queryset_2))
Альтернативный метод, но не обязательно лучший, может выглядеть так (контекст тестирования в представлении, например) при использовании pytest:
all_the_things = Things.objects.all()
assert set(list(response.context_data['all_the_things'])) == set(list(all_the_things))
Это преобразует его в список, а затем в набор, который напрямую сопоставим с другим набором. Будьте осторожны с поведением set
хотя это может быть не совсем то, что вы хотите, так как оно удалит дубликаты.
Я обнаружил, что использование self.assertCountEqual(queryset1, queryset2)
также решает проблему.