Bird
Raised Fist0
Djangoframework~5 mins

Factory Boy for test data in Django

Choose your learning style10 modes available

Start learning this pattern below

Jump into concepts and practice - no test required

or
Recommended
Test this pattern10 questions across easy, medium, and hard to know if this pattern is strong
Introduction

Factory Boy helps you create fake data easily for testing your Django apps. It saves time and keeps tests clean.

When you need to create many test users with different details.
When you want to test how your app works with various data without typing it all manually.
When you want to avoid repeating the same setup code in multiple tests.
When you want your tests to be easy to read and maintain.
When you want to quickly generate related objects, like a blog post with an author.
Syntax
Django
import factory
from myapp.models import MyModel

class MyModelFactory(factory.django.DjangoModelFactory):
    class Meta:
        model = MyModel

    field1 = 'default value'
    field2 = factory.Faker('name')

Define a factory class inheriting from DjangoModelFactory.

Set the Meta.model to the Django model you want to create data for.

Examples
This factory creates fake users with random usernames and emails.
Django
class UserFactory(factory.django.DjangoModelFactory):
    class Meta:
        model = User

    username = factory.Faker('user_name')
    email = factory.Faker('email')
    password = 'testpass'
This factory creates articles with a linked author created by UserFactory.
Django
class ArticleFactory(factory.django.DjangoModelFactory):
    class Meta:
        model = Article

    title = factory.Faker('sentence')
    content = factory.Faker('text')
    author = factory.SubFactory(UserFactory)
Sample Program

This test uses Factory Boy to create a user with fake username and email. It checks the user has these fields and prints them.

Django
import factory
from django.test import TestCase
from django.contrib.auth.models import User

class UserFactory(factory.django.DjangoModelFactory):
    class Meta:
        model = User

    username = factory.Faker('user_name')
    email = factory.Faker('email')
    password = 'testpass'

class UserTest(TestCase):
    def test_user_creation(self):
        user = UserFactory()
        self.assertTrue(user.username)
        self.assertTrue(user.email)
        print(f"Created user: {user.username}, email: {user.email}")
OutputSuccess
Important Notes

You can customize fields with Faker or fixed values.

Use SubFactory to create related objects automatically.

Factories help keep tests fast and easy to write.

Summary

Factory Boy creates fake test data easily for Django models.

It helps avoid repetitive setup code in tests.

Use DjangoModelFactory and Faker to define factories.

Practice

(1/5)
1. What is the main purpose of using Factory Boy in Django testing?
easy
A. To create reusable fake data for tests easily
B. To speed up the Django server
C. To replace Django's ORM
D. To deploy Django applications automatically

Solution

  1. Step 1: Understand Factory Boy's role

    Factory Boy is designed to generate fake data for tests, making test setup easier and less repetitive.
  2. Step 2: Eliminate unrelated options

    Speeding up the server, replacing ORM, or deployment are unrelated to test data creation.
  3. Final Answer:

    To create reusable fake data for tests easily -> Option A
  4. Quick Check:

    Factory Boy = reusable fake test data [OK]
Hint: Factory Boy = fake test data creator [OK]
Common Mistakes:
  • Thinking Factory Boy speeds up the server
  • Confusing Factory Boy with deployment tools
  • Assuming Factory Boy replaces Django ORM
2. Which of the following is the correct way to define a basic factory for a Django model Book using Factory Boy?
easy
A. class BookFactory(factory.DjangoModelFactory): class Meta: model = Book
B. class BookFactory(factory.Factory): model = Book
C. class BookFactory(factory.ModelFactory): model = Book
D. class BookFactory(factory.DjangoFactory): class Meta: model = Book

Solution

  1. Step 1: Identify correct base class

    Factory Boy uses DjangoModelFactory as the base class for Django models.
  2. Step 2: Check Meta class usage

    The model must be specified inside a nested Meta class with attribute model.
  3. Final Answer:

    class BookFactory(factory.DjangoModelFactory): class Meta: model = Book -> Option A
  4. Quick Check:

    DjangoModelFactory + Meta.model = correct syntax [OK]
Hint: Use DjangoModelFactory with Meta.model for Django models [OK]
Common Mistakes:
  • Using factory.Factory instead of DjangoModelFactory
  • Not using a Meta class for model assignment
  • Using incorrect base class names
3. Given this factory definition:
class UserFactory(factory.DjangoModelFactory):
    class Meta:
        model = User
    username = factory.Faker('user_name')
    email = factory.Faker('email')

What will UserFactory().username return?
medium
A. None, because username is not set
B. The literal string 'user_name'
C. An error because Faker is not imported
D. A random username string generated by Faker

Solution

  1. Step 1: Understand Faker usage in Factory Boy

    Using factory.Faker('user_name') generates a random username string each time the factory is called.
  2. Step 2: Evaluate the expression UserFactory().username

    Calling UserFactory() creates a User instance with a random username, so .username returns that random string.
  3. Final Answer:

    A random username string generated by Faker -> Option D
  4. Quick Check:

    Faker('user_name') = random username string [OK]
Hint: Faker fields produce random data, not literals [OK]
Common Mistakes:
  • Thinking Faker returns the field name as string
  • Assuming missing imports cause runtime error here
  • Expecting None if not explicitly set
4. What is wrong with this factory code?
class ProductFactory(factory.DjangoModelFactory):
    class Meta:
        model = Product
    name = factory.Faker('product_name')
    price = factory.Faker('float')
medium
A. Faker does not have a 'product_name' provider
B. The 'float' provider requires arguments to specify range
C. Missing import of factory module
D. Meta class should be outside the factory class

Solution

  1. Step 1: Check Faker providers used

    Faker has no built-in 'product_name' provider, but this is a common custom name; however, 'float' requires arguments like min and max.
  2. Step 2: Identify the error cause

    Using factory.Faker('float') without arguments causes an error because Faker's float provider needs parameters.
  3. Final Answer:

    The 'float' provider requires arguments to specify range -> Option B
  4. Quick Check:

    Faker float needs min/max args [OK]
Hint: Faker float needs range arguments to work [OK]
Common Mistakes:
  • Assuming 'product_name' is always valid
  • Ignoring required arguments for Faker float
  • Thinking Meta class placement is wrong
5. You want to create a factory for a Django model Order that has a foreign key to User. How do you correctly define the user field in OrderFactory to use UserFactory?
hard
A. user = UserFactory()
B. user = factory.RelatedFactory(UserFactory)
C. user = factory.SubFactory(UserFactory)
D. user = factory.Faker('user')

Solution

  1. Step 1: Understand foreign key factory usage

    To link a foreign key to another factory, use factory.SubFactory with the related factory class.
  2. Step 2: Evaluate options

    Directly calling UserFactory() assigns an instance at class load time, not per object. RelatedFactory is for reverse relations. Faker does not create model instances.
  3. Final Answer:

    user = factory.SubFactory(UserFactory) -> Option C
  4. Quick Check:

    Foreign key uses SubFactory [OK]
Hint: Use SubFactory for foreign key relations [OK]
Common Mistakes:
  • Calling UserFactory() directly in factory field
  • Using RelatedFactory for foreign keys
  • Using Faker for model relations