-
Notifications
You must be signed in to change notification settings - Fork 11.1k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[11.x] Add assertThrown()
#54046
base: 11.x
Are you sure you want to change the base?
[11.x] Add assertThrown()
#54046
Conversation
Seems very good method to have :) |
If it's of any help, PHPUnit has assertions for exceptions:
|
That's a good point! I think what this approach still lacks is being able to perform additional assertions against the exception. For instance: class PaymentFailed extends RuntimeException
{
public PaymentMethodEnum $method;
public User $receivingUser;
public function setPaymentMethod(PaymentMethodEnum $method): void
{
$this->method = $method;
}
public function setReceivingUser(User $user): void
{
$this->receivingUser = $user;
}
} I may not render both of those pieces of information in the exception message, but I may want to perform assertions against them. |
Doesn't this already solved with #50704 |
I did not know this existed. Thanks for mentioning it and nice work @nunomaduro!
public function test_uniqueEmail() {
Exceptions::fake();
// Given
Org::factory()->create(['id' => 12]);
$userFactory = fn () => User::factory()->create(['email' => '[email protected]', 'org_id' => 12]);
// And the user exists
$userFactory();
// When we attempt to create a user with the same email address
$userFactory();
// Then
Exceptions::assertReported(function(UniqueConstraintViolationException $e) {
self::assertStringContainsString(
"Duplicate entry '[email protected]' for key 'users.users_email_org_id_unique'",
$exception->getMessage()
);
return true;
});
} ☝️ this could be solved with PHPUnit's expectException/expectExceptionMessage of course. I'm not sure if I'm doing something wrong, but it doesn't seem to work here either? public function test_ModelNotFound()
{
// Given a route
Route::get('/some-example', function () {
return User::findOrFail(1);
});
// And we are faking exception handling
Exceptions::fake([ModelNotFoundException::class]);
// $this->withoutExceptionHandling(); <-- also tried this but it didn't seem to change anything
// When we request the endpoint
$response = $this->get('/some-example');
// Then the response is not found
$response->assertNotFound();
// And the exception looks like this
Exceptions::assertReported(function (ModelNotFoundException $e) {
self::assertStringContainsString('No query results for model [App\Models\User] 1', $e->getMessage());
return true;
});
}
Obviously anyone is free to add their own |
As I was working with this method today, I realized that this causes a failing test due to no assertions: #[Test]
public function throws_an_exception(): void
{
$myFunction = static fn () => throw new \RuntimeException();
self::assertThrown($myFunction, \RuntimeException::class);
} Just making mention of it. |
This adds a testing helper for asserting a closure throws an exception/error, ensures the type, and then returns the exception. This is similar to how JUnit's assertThrows works.
The big difference from Laravel's
assertThrows()
is that you can make expectations against the exception.Code examples
This is taken from
tests/Auth/AuthAccessResponseTest.php
.The way its written currently
Rewritten to use the
assertThrows
Using the newly added method
My belief is:
The first way (try-catch and perform assertions in the catch block) is most common and works fine.
The second way (
assertThrows()
) is cumbersome to write, requires you to remember to addreturn true
to the end, and the nested closure makes it difficult to parse. If you don't addreturn true
to the end, you can just convert the second closure into a series of boolean statements, but then you can't really leverage the PHPUnit assertion methods which is a big loss.The third way (
assertThrown()
) breaks down a layer of indentation, doesn't require adding anyuse ($otherVariablesFromOuterScope)
, nor does it have the gotcha of remembering to return true.I do not love the name of this method, but couldn't think of anything better. I have added this as a trait in a work project, and figured I would see if it was of interest to the Laravel community at large.