Package Data | |
---|---|
Maintainer Username: | 96downlu |
Maintainer Contact: | lukeraymonddowning@gmail.com (luke) |
Package Create Date: | 2022-04-13 |
Package Last Update: | 2024-02-08 |
Home Page: | |
Language: | PHP |
License: | MIT |
Last Refreshed: | 2024-12-24 15:08:49 |
Package Statistics | |
---|---|
Total Downloads: | 923,086 |
Monthly Downloads: | 38,014 |
Daily Downloads: | 666 |
Total Stars: | 608 |
Total Watchers: | 12 |
Total Forks: | 11 |
Total Open Issues: | 4 |
Test requests in Laravel without all the boilerplate.
💡 Psst. Although our examples use Pest PHP, this works just as well in PHPUnit.
Take a look at the following test:
it('can sign up a user with an international phone number', function () {
$this->put('/users', [
'phone' => '+375 154 767 1088',
'email' => 'foo@bar.com', 🙄
'name' => 'Luke Downing', 😛
'company' => 'Worksome', 😒
'bio' => 'Blah blah blah', 😫
'profile_picture' => UploadedFile::fake()->image('luke.png', 200, 200), ðŸ˜
'accepts_terms_and_conditions' => true, 🤬
]);
expect(User::latest()->first()->phone)->toBe('+375 154 767 1088');
});
Oof. See, all we wanted to test was the phone number, but because our route's FormRequest has validation rules, we have to send all of these additional fields at the same time. This approach has a few downsides:
We think this experience can be vastly improved. Take a look:
it('can sign up a user with an international phone number', function () {
SignupRequest::fake();
$this->put('/users', ['phone' => '+375 154 767 1088']);
expect(User::latest()->first()->phone)->toBe('+375 154 767 1088');
});
Soooooo much nicer. And all thanks to Request Factories. Let's dive in...
You can install the package as a developer dependency via Composer:
composer require --dev worksome/request-factories
First, let's create a new RequestFactory
. A RequestFactory
usually complements a FormRequest
in your application (request factories work with standard requests too!). You can create a RequestFactory
using the make:request-factory
Artisan command:
php artisan make:request-factory "App\Http\Requests\SignupRequest"
Note that we've passed the SignupRequest
FQCN as an argument. This will create a new request factory
at tests/RequestFactories/SignupRequestFactory.php
.
You can also pass your desired request factory name as an argument instead:
php artisan make:request-factory SignupRequestFactory
Whilst you're free to name your request factories as you please, we recommend two defaults for a seamless experience:
tests/RequestFactories
. The Artisan command will do this for you.Factory
suffix. So SignupRequest
becomes SignupRequestFactory
.Let's take a look at our newly created SignupRequestFactory
. You'll see something like this:
namespace Tests\RequestFactories;
use Worksome\RequestFactories;
class SignupRequestFactory extends RequestFactory
{
public function definition(): array
{
return [
// 'email' => $this->faker->email,
];
}
}
If you've used Laravel's model factories before, this will look pretty familiar. That's because the basic concept is the same: a model factory is designed to generate data for eloquent models, a request factory is designed to generate data for form requests.
The definition
method should return an array of valid data that can be used when submitting your form. Let's fill it out for
our example SignupRequestFactory
:
namespace Tests\RequestFactories;
use Worksome\RequestFactories;
class SignupRequestFactory extends RequestFactory
{
public function definition(): array
{
return [
'phone' => '01234567890',
'email' => 'foo@bar.com',
'name' => 'Luke Downing',
'company' => 'Worksome',
'bio' => $this->faker->words(300, true),
'accepts_terms_and_conditions' => true,
];
}
public function files(): array
{
return [
'profile_picture' => $this->file()->image('luke.png', 200, 200),
];
}
}
Note that we have access to a faker
property for easily generating fake content, such as a paragraph
for our bio, along with a files
method we can declare to segregate files from other request data.
So how do we use this factory in our tests? There are a few options, depending on your preferred style.
create
on the factoryThis method is most similar to Laravel's model factories. The create
method returns an array,
which you can then pass as data to put
, post
or any other request testing method.
it('can sign up a user with an international phone number', function () {
$data = SignupRequest::factory()->create(['phone' => '+44 1234 567890']);
$this->put('/users', $data)->assertValid();
});
fake
on the request factorySeeing as you only normally make a single request per test, we support registering your factory globally with fake
. If you're using this approach,
make sure that it's the last method you call on the factory, and that you call it before making a request
to the relevant endpoint.
it('can sign up a user with an international phone number', function () {
SignupRequestFactory::new()->fake();
$this->put('/users')->assertValid();
});
fake
on the form requestIf you've used Laravel model factories, you'll likely be used to calling ::factory()
on eloquent models to get a
new factory instance. Request factories have similar functionality available. First, add the Worksome\RequestFactories\Concerns\HasFactory
trait to the relevant FormRequest:
class SignupRequest extends \Illuminate\Foundation\Http\FormRequest
{
use \Worksome\RequestFactories\Concerns\HasFactory;
}
This provides access to 2 new static methods on the form request: ::factory()
and ::fake()
. You can use these methods
in your tests instead of instantiating the request factory directly:
it('can sign up a user with an international phone number', function () {
// Using the factory method...
SignupRequest::factory()->fake();
// ...or using the fake method
SignupRequest::fake();
$this->put('/users')->assertValid();
});
fakeRequest
in Pest PHPIf you're using Pest, we provide a higher order method that you can chain onto your tests:
// You can provide the form request FQCN...
it('can sign up a user with an international phone number', function () {
$this->put('/users')->assertValid();
})->fakeRequest(SignupRequest::class);
// Or the request factory FQCN...
it('can sign up a user with an international phone number', function () {
$this->put('/users')->assertValid();
})->fakeRequest(SignupRequestFactory::class);
// Or even a closure that returns a request factory...
it('can sign up a user with an international phone number', function () {
$this->put('/users')->assertValid();
})->fakeRequest(fn () => SignupRequest::factory());
You can even chain factory methods onto the end of the fakeRequest
method:
it('can sign up a user with an international phone number', function () {
$this->put('/users')->assertValid();
})
->fakeRequest(SignupRequest::class)
->state(['name' => 'Jane Bloggs']);
It's important to note the order of importance request factories take when injecting data into your request.
get
, post
, put
, patch
, delete
or similar methods will always take precedence.state
, or methods called on a factory that alter state will be next in line.definition
and files
methods come last, only filling out missing properties from the request.Let's take a look at an example to illustrate this order of importance:
it('can sign up a user with an international phone number', function () {
SignupRequest::factory()->state(['name' => 'Oliver Nybroe', 'email' => 'oliver@worksome.com'])->fake();
$this->put('/users', ['email' => 'luke@worksome.com'])->assertValid();
});
The default email defined in SignupRequestFactory
is foo@bar.com
. The default name is Luke Downing
.
Because we override the name
property using the state
method before calling fake
, the name used in
the form request will actually be Oliver Nybroe
, not Luke Downing
.
However, because we pass luke@worksome.com
as data to the put
method, that will take priority over
all other defined data, both foo@bar.com
and oliver@worksome.com
.
Factories are really cool, because they allow us to create a domain-specific-language for our feature tests. Because factories are classes, we can add declarative methods that serve as state transformers.
// In our factory...
class SignupRequestFactory extends RequestFactory
{
// After the definition...
public function withOversizedProfilePicture(): static
{
return $this->state(['profile_picture' => $this->file()->image('profile.png', 2001, 2001)])
}
}
// In our test...
it('does not allow profile pictures larger than 2000 pixels', function () {
SignupRequest::factory()->withOversizedProfilePicture()->fake();
$this->put('/users')->assertInvalid(['profile_picture' => 'size']);
});
The state
method is your friend for any data you want to add or change on your factory. What about if you'd like to omit a property
from the request? Try the without
method!
it('requires an email address', function () {
SignupRequest::factory()->without(['email'])->fake();
$this->put('/users')->assertInvalid(['email' => 'required']);
});
💡 You can use dot syntax in the
without
method to unset deeply nested keys
Sometimes, you'll have a property that you want to be based on the value of other properties. In that case, you can provide a closure as the property value, which receives an array of all other parameters:
class SignupRequestFactory extends RequestFactory
{
public function definition(): array
{
return [
'name' => 'Luke Downing',
'company' => 'Worksome',
'email' => fn ($properties) => Str::of($properties['name'])
->replace(' ', '.')
->append("@{$properties['company']}.com")
->lower()
->__toString(), // luke.downing@worksome.com
];
}
}
Occasionally, you'll notice that multiple requests across your application share a similar subset of fields. For example, a signup form and a payment form might both contain an address array. Rather than duplicating these fields in your factory, you can nest factories inside factories:
class SignupRequestFactory extends RequestFactory
{
public function definition(): array
{
return [
'name' => 'Luke Downing',
'company' => 'Worksome',
'address' => AddressRequestFactory::new(),
];
}
}
Now, when the SignupRequestFactory
is created, it will resolve the AddressRequestFactory
for you
and fill the address
property with all fields contained in the AddressRequestFactory
definition.
Pretty cool hey?
Not every controller in your app requires a backing form request. Thankfully, we also support faking a generic request:
it('lets a guest sign up to the newsletter', function () {
NewsletterSignupFactory::new()->fake();
post('/newsletter', ['email' => 'foo@bar.com'])->assertRedirect('/thanks');
});
CouldNotLocateRequestFactoryException
When using the HasFactory
trait on a FormRequest
, we attempt to auto-locate the relevant
request factory for you. If your directory structure doesn't match for whatever reason, this exception
will be thrown.
It can easily be resolved by adding a public static $factory
property to your form request:
class SignupRequest extends FormRequest
{
use HasFactory;
public static $factory = SignupRequestFactory::class;
}
No sweat. Just place a call to fake
on the relevant request factory before making each request:
it('allows a user to sign up and update their profile', function () {
SignupRequest::fake();
post('/signup');
ProfileRequest::fake();
post('/profile')->assertValid();
});
Not a problem. We provide a config file you may publish where you can edit the path and namespace of request factories. First, publish the config file:
php artisan vendor:publish --tag=request-factories
Now, in the newly created config/request-factories.php
, alter the path
and namespace
keys to suit your requirements:
return [
'path' => base_path('request_factories'),
'namespace' => 'App\\RequestFactories',
];
We pride ourselves on a thorough test suite and strict static analysis. You can run all of our checks via a composer script:
composer test
Please see CHANGELOG for more information on what has changed recently.
The MIT License (MIT). Please see License File for more information.