This package will generate class-based factories
which you can use instead of the default Laravel factory files.
Why Class-Based Factories?:
- You have a dedicated class for every factory, and the default model data can be defined inside this class. This is a cleaner solution than the default factory files.
- If creating test data gets more complicated than creating just one model, you hide this inside the factory class so that your tests stay clean.
- The generated factory classes use return types so that your IDE knows what gets returned. (This is something you do not have with the default Laravel factories)
I already know a lot of people using factory classes. So why not just create your own classes when you need them?
- Automate everything! Even just calling an artisan command that creates a class is much faster than you doing it yourself.
- This package will create classes that already provide factory features, you know, like creating a new model instance, multiple ones and more.
You can install the package via composer:
composer require christophrumpel/laravel-factories-reloaded
To publish the config file run:
php artisan vendor:publish --provider="Christophrumpel\LaravelFactoriesReloaded\LaravelFactoriesReloadedServiceProvider"
It will provide the package's config file where you can define the paths of your models
, the path of the generated factories
, as well as the generated factories namespace
.
<?php
/*
* You can place your custom package configuration in here.
*/
return [
'models_path' => [
base_path('app')
],
'factories_path' => base_path('tests/Factories'),
'factories_namespace' => 'Tests\Factories',
];
First, you need to create a new factory class. This is done via a new command this package comes with. In this example, we want to create a new user factory.
php artisan make:factory-reloaded
After running this command, you have to select one of your models. Here you decide for which model you are creating a factory for. I will choose the user model. (Through a config you can define where your models live)
This will give you a new UserFactory
under the Tests\Factories
namespace. Here is your new basic factory class:
class UserFactory extends BaseFactory
{
protected string $modelClass = User::class;
public function create(array $extra = []): User
{
return parent::build($extra);
}
public function make(array $extra = []): User
{
return parent::build($extra, 'make');
}
public function getDefaults(Generator $faker): array
{
return [
'name' => $faker->name,
'email' => $faker->unique()->safeEmail,
'email_verified_at' => now(),
'password' => '$2y$10$92IXUNpkjO0rOQ5byMi.Ye4oKoEa3Ro9llC/.og/at2.uheWG/igi', // password
'remember_token' => Str::random(10),
];
}
}
Inside this class, you can define the properties of the model with the getDefaults
method. It is very similar to what you would do with a Laravel default factory, and you can make use of Faker as well. Actually, if you have already a Laravel factory for a specific model, this package will copy the default data to the new class factory while generating.
The create
method is only a copy of the one in the parent class BaseFactory
. Still, we need it in our dedicated factory class so that we can define what gets returned. In our case, it is a user model. Other methods like new
or times
are hidden in the parent class.
If you don't want to select your model from a list, you can pass the class name of a model in your model path as an argument and your factory will immediately be created for you:
php artisan make:factory-reloaded Ingredient
By default, this command will ask you if you want to overwrite the class factory, if it already exists. You can pass the --force
option in order to skip this question.
php artisan make:factory-reloaded Ingredient --force
You can publish a configuration file that lets you set the path to your models and factories as well as the namespace of your factories. Alternatively, you can set the configurations as options when you run the command:
--models_path=app/models
--factories_path=path/to/your/factories
--factories_namespace=Your\Factories\Namespace
Most people won't need to override these configuration on the fly, but if you splitting your app up into domains you might be keeping your factories closer to their models. In this case you could do something like this:
php artisan make:factory-reloaded --models_path=app/domains/customers/models
--factories_path=app/domains/customers/factories --factories_namespace=App\Domains\Customers\Factories
Now you can start using your new user factory class in your tests. The static new
method gives you a new instance of the factory. This is useful to chain other methods, like create
for example.
$user = UserFactory::new()->create();
This will give you back a newly created user instance from the database. If you want to create multiple instances, you can use the times
method, which will use the create
method behind the scenes and will return you a collection of the new model instances.
$user = UserFactory::new()
->times(4)
->create();
Like with Laravel factories you can also make
a new model which gets not
stored to the database yet.
$user = UserFactory::new()->make();
You probably have used states
with Laravel factories and that is possible with factory classes as well of course. Since you own your factory classes there are different ways to implement state-like functionality. If you have already defined states in your Laravel factory file, this package can convert them into state methods
in the new factory class. When you run the command to create a new class factory, you will be asked about that.
Besides, the easiest approach to create a new state is by calling a method which sets a property in the class.
$recipe = RecipeFactory::new()
->published()
->create();
And to make this work, we need to add the method to the factory class, as well as the property. In the getDefaults
method we then use the property to fill the model.
<?php
namespace Tests\Factories;
use App\Recipe;
use Christophrumpel\LaravelFactoriesReloaded\BaseFactory;
use Faker\Generator;
class RecipeFactory extends BaseFactory
{
protected string $modelClass = Recipe::class;
private bool $isPublished = false;
public function create(array $extra = []): Recipe
{
return parent::build($extra);
}
public function make(array $extra = []): Recipe
{
return parent::build($extra, 'make');
}
public function getData(Generator $faker): array
{
return [
'category_id' => 1,
'name' => $faker->name,
'is_published' => $this->isPublished,
];
}
public function published(): self
{
$clone = clone $this;
$clone->isPublished = true;
return $clone;
}
}
⚠️ Note: We clone the factory class to make it immuteable when using the published method. You can read more about it in Brent's factory article.
There will be situations when you need to add related models to your test data. This is already pretty easy with using multiple factory classes.
$user = UserFactory::new()->create();
$user->recipes()->saveMany(RecipeFactory::new()->times(4)->make());
Of course, the relations need to be set up before. Besides this, there is also an in-built solution.
$user = UserFactory::new()
->with(Recipe::class, 'recipes')
->create();
With the with
method, you can easily add relations in a more fluently way. The first parameter defines the mode and the second one the name of the relationship. If you need to add more than one related model, you can add a third argument to define the count.
$user = UserFactory::new()
->with(Recipe::class, 'recipes', 4)
->create();
⚠️ Note: For this to work, you need to have created a RecipeFactory before.
But there is one more way to add related data. Since you own your factory classes, you can add a method on the class itself. This way you can pick a much more expressive name like withRecipes
.
$user = UserFactory::new()
->withRecipes(4)
->create();
This way you can define yourself how to set up recipes and you are more flexible doing it. Here is an example of how your UserFactory
could look like with a custom relation method.
class UserFactory extends BaseFactory
{
protected string $modelClass = User::class;
/** @var Collection */
private $recipes;
public function create(array $extra = []): User
{
$user = parent::build($extra);
if ($this->recipes) {
$user->recipes()->saveMany($this->recipes);
}
return $user;
}
public function withRecipes(int $times = 1)
{
$clone = clone $this;
$clone->recipes = RecipeFactory::new()
->times($times)->make();
return $clone;
}
public function getData(Generator $faker): array
{
return [
'name' => $faker->name,
'email' => 'test@email.at',
'password' => bcrypt('test'),
];
}
}
⚠️ Note: Whenever you return the factory itself from a method likewithRecipes
, you should use aclone
of the instance like in the example above to prevent modifications.
composer test
Please see CHANGELOG for more information about what has changed recently.
Please see CONTRIBUTING for details.
If you discover any security-related issues, please email christoph@christoph-rumpel.com instead of using the issue tracker.
The current implementation was improved with help from Brent's article about how they deal with factories at Spatie.
The MIT License (MIT). Please see License File for more information.
This package was generated using the Laravel Package Boilerplate.