Model Factories for Custom Namespace Models

(Updated on )

Update:

You can now use the #[UseFactory(YourFactory::class)] attribute to tell Laravel to use a factory for a model. This is a much cleaner solution than the one below, and it was written by yours truly :)


If you like to follow certain patterns, such as DDD, you may find models in different directories than the typical App\Models namespace in your Laravel app.

Laravel can work with this no problem, except …for Factories. These normally rely on convention to find your model, but in the case of using custom namespaces outside of the App\Models namespace, they won’t be able to find your model for a given Factory.

Step 1: Set Factory Model Property

Let’s say we have this structure below:

app/
  Content/
    Models/
      Post.php

This would make our namespace for our Post model App\Content\Models\Post.

Next, we’ll create our factory:

php artisan make:factory PostFactory

We can set the model to be used by this factory by setting the $model property:

class PostFactory extends Factory {
   protected $model = \App\Content\Models\Post::class;
// other code
}

Step 2: Service Provider

The previous step is not enough on its own, we will also want to tell Laravel how to resolve the factories so that it can find our factories all over the app (in PHPUnit setup() method, database seeding, etc).

Update your AppProvider in app/Providers/AppServiceProvider.php

use Illuminate\Database\Eloquent\Factories\Factory;

class AppServiceProvider extends ServiceProvider
{
  public function boot(): void {
     // ...  
    Factory::guessFactoryNamesUsing(function(string $modelName) {
      return 'Database\\Factories\\' . class_basename($modelName) . 'Factory';
    });
}

(Tip provided by @ejunker, thank you!)

This will tell Laravel where the factory name is for our custom namespaced models. It’s important that we follow the convention its looking for. So in our example, our Post model would have a factory named PostFactory. If you have other models like Comment, then the factory would need to be called CommentFactory.

Conclusion

Hope this little tip helps. It’s a slight annoyance that I think would make a good candidate for a PR to introduce a more elegant way to define the factory for a model outside the App\Models namespace, but this workaround will get you going for now 😊