16

我正在使用Factory Boy为我的 django 应用程序创建测试工厂。我遇到问题的模型是一个非常基本的帐户模型,它与 django 用户身份验证模型具有 OneToOne 关系(使用 django < 1.5):

# models.py
from django.contrib.auth.models import User
from django.db import models

class Account(models.Model):
    user = models.OneToOneField(User)
    currency = models.CharField(max_length=3, default='USD')
    balance = models.CharField(max_length="5", default='0.00') 

这是我的工厂:

# factories.py
from django.db.models.signals import post_save
from django.contrib.auth.models import User

import factory

from models import Account


class AccountFactory(factory.django.DjangoModelFactory):
    FACTORY_FOR = Account

    user = factory.SubFactory('app.factories.UserFactory')
    currency             = 'USD'
    balance              = '50.00'

class UserFactory(factory.django.DjangoModelFactory):
    FACTORY_FOR = User

    username = 'bob'
    account = factory.RelatedFactory(AccountFactory)

所以我希望工厂男孩在调用 AccountFactory 时创建一个相关的 UserFactory:

# tests.py 
from django.test import TestCase

from factories import AccountFactory

class AccountTest(TestCase):

    def setUp(self):
        self.factory = AccountFactory()

    def test_factory_boy(self):
        print self.factory.id

但是,在运行测试时,似乎正在创建多个用户模型,并且我看到了一个完整性错误:

IntegrityError: column username is not unique

文档确实提到在处理循环导入时要注意循环,但我不确定这是否发生了,也不知道我将如何解决它。文档

如果任何熟悉 Factory Boy 的人都可以插话或提供一些关于可能导致此完整性错误的原因的见解,我们将不胜感激!

4

2 回答 2

13

我相信这是因为您的工厂定义中有循环引用。尝试account = factory.RelatedFactory(AccountFactory)UserFactory定义中删除该行。如果您总是要通过 AccountFactory 调用帐户创建,那么您不需要此行。

此外,您可以考虑将序列附加到名称字段,这样如果您确实需要多个帐户,它会自动生成它们。

更改:username = "bob"tousername = factory.Sequence(lambda n : "bob {}".format(n))并且您的用户将被命名为“bob 1”、“bob 2”等。

于 2013-07-16T02:49:53.913 回答
2

要将调用结果传递UserFactoryAccountFactory您应该使用factory_related_name文档

上面的代码接下来的工作方式:

  • AccountFactory用于实例化需求SubFactory(UserFactory)
  • UserFactory实例化用户。
  • UserFactory实例化调用后RelatedFactory(AccountFactory)
  • 递归,.. 由于唯一用户名约束而被破坏(您可能希望通过FuzzyTextor生成用户名Sequence

所以你需要UserFactory这样写:

class UserFactory(factory.django.DjangoModelFactory):
    account = factory.RelatedFactory(AccountFactory, factory_related_name='user')
    username = factory.Sequence(lambda a: 'email%04d@somedomain.com' % a)
    # rest of code

但是您仍然会遇到已经编写的测试的问题。想象一下,您有如下测试场所:

user = UserFactory()
account = Account(user=user)

然后添加RelatedFactory将破坏测试。如果你的项目中没有很多测试和贡献者,你可以重写它们。但如果不是,它不是一个选择。以下是如何处理它:

class UserFactory(factory.django.DjangoModelFactory):
    class Params:
        generate_account = factory.Trait(
            account=factory.RelatedFactory(AccountFactory, factory_related_name='user')
        )

那么上面的代码就不会被破坏,因为默认调用UserFactory不会实例化AccountFactory。使用帐户实例化用户:

user_with_account = UserFactory(generate_account=True)
于 2021-05-13T08:58:04.933 回答