python - 当两个外键指向同一个表时如何使用工厂男孩

标签 python django python-3.x django-models factory-boy

我正在尝试使用 Factory Boy 链接两个 Django 模型,但我找不到针对此问题的简单解决方案。这些是具有相应工厂的模型:

class Currency(models.Model):
    id = models.CharField(max_length=3, primary_key=True)


class ConversionRate(models.Model):
    currency = models.ForeignKey(Currency, null=False, on_delete=models.CASCADE)
    quote = models.ForeignKey(Currency, null=False, on_delete=models.CASCADE)
    rate = models.DecimalField(max_digits=6, decimal_places=2)


class CurrencyFactory(factory.django.DjangoModelFactory):
    class Meta:
        model = Currency

    id = factory.Sequence(lambda n: ['EUR', 'USD'][n%2])
    conversion_rate = factory.RelatedFactory('my_app.factories.ConversionRateFactory', 'currency')


class ConversionRateFactory(factory.django.DjangoModelFactory):
    class Meta:
        model = ConversionRate

    currency = factory.SubFactory(CurrencyFactory)
    quote = factory.SubFactory(CurrencyFactory, id='EUR')
    rate = 1.2

这是用于测试的表格的默认内容:

+--------+   +--------------------------+ 
|Currency|   |       ConversionRate     |
+--------+   +----------+--------+------+ 
|   id   |   | currency |  quote | rate |
+--------+   +----------+--------+------+ 
|  EUR   |   |    USD   |  EUR   |  1.2 |
+--------+   +----------+--------+------+  
|  USD   |   |    EUR   |  EUR   |   1  |
+--------+   +----------+--------+------+  

当我尝试构建工厂时会抛出完整性错误:

CurrencyFactory.create()
# Error:  UNIQUE constraint failed: Currency.id

我还尝试在 CurrencyFactory“Meta”部分添加 django_get_or_create = ('id',),但这会造成无限循环。

过去有人遇到过这样的问题吗?有什么建议吗?

这是使用 django_get_or_create = ('id',) 时的回溯:

    env/lib/python3.6/site-packages/factory/builder.py:272: in build
        step.resolve(pre)
    env/lib/python3.6/site-packages/factory/builder.py:221: in resolve
        self.attributes[field_name] = getattr(self.stub, field_name)
    env/lib/python3.6/site-packages/factory/builder.py:375: in __getattr__
        extra=context,
    env/lib/python3.6/site-packages/factory/declarations.py:324: in evaluate
        return self.generate(step, defaults)
    env/lib/python3.6/site-packages/factory/declarations.py:414: in generate
        return step.recurse(subfactory, params, force_sequence=force_sequence)
    env/lib/python3.6/site-packages/factory/builder.py:233: in recurse
        return builder.build(parent_step=self, force_sequence=force_sequence)
    env/lib/python3.6/site-packages/factory/builder.py:299: in build
        context=postgen_context,
    env/lib/python3.6/site-packages/factory/declarations.py:675: in call
        return step.recurse(factory, passed_kwargs)
    env/lib/python3.6/site-packages/factory/builder.py:233: in recurse
        return builder.build(parent_step=self, force_sequence=force_sequence)
    env/lib/python3.6/site-packages/factory/builder.py:272: in build
        step.resolve(pre)
    env/lib/python3.6/site-packages/factory/builder.py:221: in resolve
        self.attributes[field_name] = getattr(self.stub, field_name)
    env/lib/python3.6/site-packages/factory/builder.py:375: in __getattr__
        extra=context,
    env/lib/python3.6/site-packages/factory/declarations.py:324: in evaluate
        return self.generate(step, defaults)
    env/lib/python3.6/site-packages/factory/declarations.py:414: in generate
        return step.recurse(subfactory, params, force_sequence=force_sequence)
    env/lib/python3.6/site-packages/factory/builder.py:233: in recurse
        return builder.build(parent_step=self, force_sequence=force_sequence)
    env/lib/python3.6/site-packages/factory/builder.py:279: in build
        kwargs=kwargs,
    env/lib/python3.6/site-packages/factory/base.py:314: in instantiate
        return self.factory._create(model, *args, **kwargs)
    env/lib/python3.6/site-packages/factory/django.py:163: in _create
        return cls._get_or_create(model_class, *args, **kwargs)
    env/lib/python3.6/site-packages/factory/django.py:154: in _get_or_create
        instance, _created = manager.get_or_create(*args, **key_fields)
    env/lib/python3.6/site-packages/django/db/models/manager.py:82: in manager_method
        return getattr(self.get_queryset(), name)(*args, **kwargs)
    env/lib/python3.6/site-packages/django/db/models/query.py:487: in get_or_create
        return self.get(**lookup), False
    env/lib/python3.6/site-packages/django/db/models/query.py:394: in get
        clone = self.filter(*args, **kwargs)
    env/lib/python3.6/site-packages/django/db/models/query.py:836: in filter
        return self._filter_or_exclude(False, *args, **kwargs)
    env/lib/python3.6/site-packages/django/db/models/query.py:850: in _filter_or_exclude
        clone = self._chain()
    env/lib/python3.6/site-packages/django/db/models/query.py:1156: in _chain
        obj = self._clone()
    env/lib/python3.6/site-packages/django/db/models/query.py:1168: in _clone
        c = self.__class__(model=self.model, query=self.query.chain(), using=self._db, hints=self._hints)
    env/lib/python3.6/site-packages/django/db/models/sql/query.py:337: in chain
        obj = self.clone()
    env/lib/python3.6/site-packages/django/db/models/sql/query.py:300: in clone
        obj.where = self.where.clone()
    env/lib/python3.6/site-packages/django/db/models/sql/where.py:148: in clone
        children=[], connector=self.connector, negated=self.negated)

    _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _

    cls = <class 'django.db.models.sql.where.WhereNode'>, children = [], connector = 'AND', negated = False

        @classmethod
        def _new_instance(cls, children=None, connector=None, negated=False):
            """
                Create a new instance of this class when new Nodes (or subclasses) are
                needed in the internal code in this class. Normally, it just shadows
                __init__(). However, subclasses with an __init__ signature that aren't
                an extension of Node.__init__ might need to implement this method to
                allow a Node to create a new instance of them (if they have any extra
                setting up to do).
                """
    >       obj = Node(children, connector, negated)
    E       RecursionError: maximum recursion depth exceeded while calling a Python object

最佳答案

如您所述,问题出在 CurrencyFactory 创建一个 ConversionRateFactory,而后者又创建了 2 CurrencyFactory.

我建议使用 factory.Trait 通过递归禁用它:

  1. 在工厂 class Params 部分定义特征:启用时(通过设置 bool 标志),它将添加附加声明
  2. 为该特征设置默认值 True:直接调用 CurrencyFactory 将添加一个 ConversionRate
  3. ConversionRateFactory 中,禁用特征 - 防止循环触发。

请看下面的代码:

class CurrencyFactory(factory.django.DjangoModelFactory):
    class Meta:
        model = models.Currency
        django_get_or_create = ['id']

    class Params:
        with_conversion_rate = factory.Trait(
            conversion_rate=factory.RelatedFactory('my_app.factories.ConversionRateFactory', 'currency'),
        )

    # Small improvement: use a `factory.Iterator` to cycle between value
    id = factory.Iterator(['EUR', 'USD'])
    # By default, force each CurrencyFactory to create a ConversionRate.
    with_conversion_rate = True

class ConversionRateFactory(factory.django.DjangoModelFactory):
    class Meta:
        model = models.ConversionRate

    rate = 1.2
    currency = factory.SubFactory(
        CurrencyFactory,
        with_conversion_rate=False,
    )
    quote = factory.SubFactory(
        CurrencyFactory,
        id='EUR',
        with_conversion_rate=False,
    )

关于python - 当两个外键指向同一个表时如何使用工厂男孩,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/50782258/

相关文章:

python - 复制pandas数据框中的数据

python - 如果在 Django 中输入任何无效数据,则在输入字段下方引发表单错误

python - 如何在虚拟环境中运行 Spyder?

python魔杖错误 "wand.resource.DestroyedResourceError: <wand.image.Image: (closed)> is destroyed already"

python - 异步抓取并使用django celery和redis存储结果并存储我的正确方法是什么?

python - TfIdfVectorizer 未正确标记

python - 值错误 : Input 0 is incompatible with layer lstm_14: expected ndim=3, 发现 ndim=2

database - 没有数据库的django模型

python - Django - 错误 : No module named grappelli

python - 按多个元素按升序和降序对元组列表进行排序