问题:
在以下情况下使用 Factory Boy 有什么优势?我真的不明白为什么我不应该只提供我自己的自定义对象。如果我错了,请告诉我为什么。
我在测试期间使用 Factory Boy 来创建用户实例,这会动态创建一个 UserProfile 对象(来自 Factory_Boy文档的标准配方)。
Data 类创建将在发布期间传递到表单的数据(我正在使用的其他方法为 self.client.post 登录、注册和激活用户的方法传递数据。除非我遗漏了什么,否则我会有为每种情况构建一个单独的 DjangoModelFactory 子类,以便在数据要求不同的地方使用 ClassName.attributes()。我朝这个方向走的另一个原因是 UserProfile 有一个 User 外键,所以我无法调用 UserProfileFactory .attributes() 直接,只有 UserFactory.attributes()。为什么不像我一样自己做呢?
#Factories.py
IMAGE_PATH = os.path.join(os.path.dirname(__file__),
'../../test_files/test_images/image.jpeg')
class UserProfileFactory(DjangoModelFactory):
FACTORY_FOR = UserProfile
user = factory.SubFactory('portal.factories.UserFactory', profile=None)
first_name = factory.Sequence(lambda n: "Joe_%d" % n)
last_name = factory.Sequence(lambda n: "Schmoe_%d" % n)
nickname = factory.Sequence(lambda n: "JoeBlow_%d" % n)
profile_image = factory.LazyAttribute(lambda t: File(open(IMAGE_PATH)))
class UserFactory(DjangoModelFactory):
FACTORY_FOR = User
username = factory.Sequence(lambda n: "user_%d" % n)
password = make_password("password")
email = factory.Sequence(lambda n: "user_%d@gmail.com" % n)
profile = factory.RelatedFactory(UserProfileFactory, 'user')
@classmethod
def _generate(cls, create, attrs):
models.signals.post_save.disconnect(user_post_save, sender=User)
user = super(UserFactory, cls)._generate(create, attrs)
models.signals.post_save.connect(user_post_save, sender=User)
return user
class Data(object):
def __init__(self):
self.IMAGE_PATH = os.path.join(os.path.dirname(__file__),
'../../test_files/test_images/image.jpeg')
self.profile_image = File(open(IMAGE_PATH))
def get_profile_update(self, user):
return {'first_name': 'Jeff',
'last_name': 'Lebowski',
'nickname': 'The Dude',
'profile_image': self.profile_image,
'user': user.pk,}
def and_so_on(self):
continues...
然后我在集成测试期间在以下上下文中使用这样的数据:
class PortalTestCase(TestCase):
"""Shortened and simplified"""
def test_edit_profile_post(self):
user = UserFactory.create()
login_bool = self.client.login(username=user.username,
password=self.data.get_password())
data = self.data.get_profile_update(user)
response = self.client.post(reverse(self.get_edit_profile()),
data=data,
follow=True)
success_url = 'http://testserver%s' % reverse(self.get_portal())
template_name = self.get_portal_template()
content_text_img = 'src="/' + user.get_profile().profile_image.url + '"'
self.assertRedirects(response, success_url)
self.assertTemplateUsed(response, template_name)
self.assertContains(response, content_text_img)