factory_boy中的多对多关系?

Ray*_*Ray 11 unit-testing django-testing factory-boy

我正在尝试使用factory_boy测试两个Django模型之间的多对多关系.factory_boy文档似乎没有讨论这个问题,我无法弄清楚我做错了什么.当我运行第一个测试时,我得到错误"AttributeError:'Pizza'对象没有属性'顶部'".第二次测试我得到了类似的错误.

当我在调试器中运行测试时,我可以看到"浇头"对象,但它不知道如何从中获取名称.我是否正确定义了PizzaFactory的_prepare方法?当你有多对多的关系时,如何从另一个表中访问一个表中的名称?

谢谢.

models.py:

from django.db import models

class Topping(models.Model):
    name = models.CharField(max_length=50)

    def __unicode__(self):
        return self.name

class Pizza(models.Model):
    name = models.CharField(max_length=100)
    toppings = models.ManyToManyField(Topping)

    def __unicode__(self):
        return self.name
Run Code Online (Sandbox Code Playgroud)

factories.py:

import factory
from models import Topping, Pizza

class ToppingFactory(factory.Factory):
    name = 'mushrooms'

class PizzaFactory(factory.Factory):
    name = 'Vegetarian'

    @classmethod
    def _prepare(cls, create, **kwargs):
        topping = ToppingFactory()
        pizza = super(PizzaFactory, cls)._prepare(create, **kwargs)
        pizza.toppings.add(topping)
        return pizza
Run Code Online (Sandbox Code Playgroud)

tests.py

from django.test import TestCase
import factory
from app.models import Topping, Pizza
from app.factories import ToppingFactory, PizzaFactory

class FactoryTestCase(TestCase):

    def test_pizza_has_mushrooms(self):
        pizza = PizzaFactory()
        self.assertTrue(pizza.topping.name, 'mushrooms')

    def test_mushrooms_on_pizza(self):
        topping = ToppingFactory()
        self.assertTrue(topping.pizza.name, 'Vegetarian')
Run Code Online (Sandbox Code Playgroud)

Bee*_*ees 19

我相信你需要使用@factory.post_generation装饰器:

class PizzaFactory(factory.Factory):
    name = 'Vegetarian'

    @factory.post_generation
    def toppings(self, create, extracted, **kwargs):
        if not create:
            # Simple build, do nothing.
            return

        if extracted:
            # A list of groups were passed in, use them
            for topping in extracted:
                self.toppings.add(topping)
Run Code Online (Sandbox Code Playgroud)

然后你会在tests.py中调用它 pizza = PizzaFactory.create(toppings=(topping1, topping2, tooping3))

来自https://factoryboy.readthedocs.org/en/latest/recipes.html.


kle*_*len 6

只需使用混合器:

from mixer.backend.django import mixer

# Generate toppings randomly
pizza = mixer.blend(Pizza, toppings=mixer.RANDOM)

# Set toppings
toppings = mixer.cycle(3).blend(Topping)
pizza = mixer.blend(Pizza, toppings=toppings)

# Generate toppings with name=tomato
pizze = mixer.blend(Pizza, toppings__name='tomato')
Run Code Online (Sandbox Code Playgroud)

简单、可配置、更快、无模式、声明式,并且您在一些测试中得到了您想要的。

  • 支持轶事——为什么我选择了 `mixer`...我最初被 `factory_boy` 所吸引。我喜欢在 `factory_boy` 中,对于给定的 `Factory`,你可以调用一个方法 `attributes()` 并返回一个 `dict`,非常适合与 **[`dataset`](http://dataset. readthedocs.org/en/latest/)**。但是`mixer` 的优点在于它可以帮助您保持干燥,因为它可以即时完成如此多的工作。这意味着如果您的被测代码将来发生更改,则需要更改的测试代码更少! (2认同)
  • 你们还支持吗? (2认同)