And*_*ges 5 python django factory-boy
我正在使用factory_boy替换Django应用中的装置。我有一个产品模型,其中应该包含许多优惠和商家。
#models.py
class Product(models.Model):
name = models.CharField()
class Merchant(models.Model):
product = models.ForeignKey(Product)
name = models.CharField()
class Offer(models.Model):
product = models.ForeignKey(Product)
price = models.DecimalField(max_digits=10, decimal_places=2)
Run Code Online (Sandbox Code Playgroud)
我想要一个工厂来创建具有多个商家和多个要约的产品。
#factories.py
import random
from models import Offer, Merchant, Product
class OfferFactory(factory.django.DjangoModelFactory):
FACTORY_FOR = Offer
product = factory.SubFactory(ProductFactory)
price = random.randrange(0, 50000, 1)/100.0
class MerchantFactory(factory.django.DjangoModelFactory):
FACTORY_FOR = Merchant
product = factory.SubFactory(ProductFactory)
name = factory.Sequence(lambda n: 'Merchant %s' % n)
url = factory.sequence(lambda n: 'www.merchant{n}.com'.format(n=n))
class ProductFactory(factory.django.DjangoModelFactory):
FACTORY_FOR = Product
name = "test product"
offer = factory.RelatedFactory(OfferFactory, 'product')
offer = factory.RelatedFactory(OfferFactory, 'product') # add a second offer
offer = factory.RelatedFactory(OfferFactory, 'product') # add a third offer
merchant = factory.RelatedFactory(MerchantFactory, 'product')
merchant = factory.RelatedFactory(MerchantFactory, 'product') # add a second merchant
merchant = factory.RelatedFactory(MerchantFactory, 'product') # add a third merchant
Run Code Online (Sandbox Code Playgroud)
但是,当我使用ProductFactory创建产品时,它只有一个要约和一个商人。
In [1]: from myapp.products.factories import ProductFactory
In [2]: p = ProductFactory()
In [3]: p.offer_set.all()
Out[3]: [<Offer: $39.11>]
Run Code Online (Sandbox Code Playgroud)
如何将一个ProductFactory设置为具有多个特定类型的依赖项?
为了能够指定父工厂中相关对象的数量:
class Company(models.Model):
name = models.CharField(max_length=255)
class ContactPerson(models.Model):
name = models.CharField(max_length=255)
company = models.ForeignKey(Company, on_delete=CASCADE, related_name='contacts')
Run Code Online (Sandbox Code Playgroud)
class CompanyFactory(factory.django.DjangoModelFactory):
name = factory.Faker('company')
class Meta:
model = Company
@factory.post_generation
def add_contacts(self, create, how_many, **kwargs):
# this method will be called twice, first time how_many will take the value passed
# in factory call (e.g., add_contacts=3), second time it will be None
# (see factory.declarations.PostGeneration#call to understand how how_many is populated)
# ContactPersonFactory is therefore called +1 times but somehow we get right amount of objs
at_least = 1
if not create:
return
for n in range(how_many or at_least):
ContactPersonFactory(contact=self)
class ContactPersonFactory(factory.django.DjangoModelFactory):
name = factory.Faker('first_name')
class Meta:
model = ContactPerson
Run Code Online (Sandbox Code Playgroud)
company = CompanyFactory(company_name='ACME ltd', add_contacts=4)
print(repr(company.name), len(company.contacts.all()))
company = CompanyFactory(company_name='ACME ltd')
print(repr(company.name), len(company.contacts.all()))
---
'ACME ltd' 4
'ACME ltd' 1
Run Code Online (Sandbox Code Playgroud)
如果您始终只有一个孩子,那么 docs 解决方案效果很好:
class CompanyFactory(factory.django.DjangoModelFactory):
name = factory.Faker('company')
whatever_really = factory.RelatedFactory('my_app.factories.ContactPersonFactory', 'contact')
class Meta:
model = Company
Run Code Online (Sandbox Code Playgroud)
注意相关工厂的完整路径。
company = CompanyFactory(company_name='ACME ltd')
print(repr(company.name), len(company.contacts.all()))
---
'ACME ltd' 1
Run Code Online (Sandbox Code Playgroud)
$ pip freeze | egrep 'factory|Faker|Django'
Django==2.0.4
factory-boy==2.10.0
Faker==0.8.13
$ python -V
Python 3.6.5
Run Code Online (Sandbox Code Playgroud)