This bundle provides various tools to handle errors in a consistent way in your RESTful APIs based on Symfony.
Because you are smart and you don't want to reinvent the wheel. This bundle provides you a framework to build your delightful error messages in a structured way. You could accomplish the same result without this bundle but you should write some boilerplate code that this bundle is already includes. Don't waste your time!
A good error response consists of three basic criteria in order to be truly helpful:
- An HTTP Status Code, so that the source and realm of the problem can be ascertained with ease.
- An Internal Reference ID for documentation-specific notation of errors.
- Human readable messages that summarize the context, cause, and general solution for the error at hand.
The RFC 7807 aims to define a standard way to carry machine- readable details of errors in a HTTP response to avoid the need to define new error response formats for HTTP APIs.
This solution, inspired by the RFC 7807, is fully customizable and extensible. I could say you can use this bundle in every API project to render your error messages in the format you like.
Add the following dependency in the require section of your composer.json:
"ped/api-error-bundle": "dev-master"
Add the repository as well in the repositories section of the composer.json:
"repositories": [
{
"type": "vcs",
"url": "[email protected]:paneedesign/api-error-bundle.git"
}
],
Enable the bundle by adding the following line in the bundles.php
:
<?php
return [
...
PED\ApiErrorBundle\PedApiErrorBundle::class => ['all' => true],
];
Create the config file config/packages/ped_api_error.yml
and add your settings like in the following example:
ped_api_error:
mapping:
fqcn:
Symfony\Component\HttpKernel\Exception\NotFoundHttpException: NOT_FOUND
Symfony\Component\HttpKernel\Exception\MethodNotAllowedHttpException: METHOD_NOT_ALLOWED
Assert\InvalidArgumentException:
type: BAD_REQUEST
forwardMessage: true
InvalidArgumentException:
type: BAD_REQUEST
forwardMessage: true
ErrorException:
type: INTERNAL_SERVER_ERROR
forwardMessage: false
Symfony\Component\Debug\Exception\FatalThrowableError:
type: INTERNAL_SERVER_ERROR
forwardMessage: false
Symfony\Component\HttpKernel\Exception\BadRequestHttpException:
type: BAD_REQUEST
forwardMessage: true
errors:
NOT_FOUND:
title: Not found
statusCode: 404
METHOD_NOT_ALLOWED:
title: Method not allowed
statusCode: 405
BAD_REQUEST:
title: Bad request
statusCode: 404
INTERNAL_SERVER_ERROR:
title: Internal server error
statusCode: 500
At the moment the bundle allows to map the Fully-Qualified Class Name of the exceptions to error codes.
This mapping is defined in the ped_api_error.mapping.fqcn
section of the config file.
For instance look at the following snippet extracted from the previous example:
ped_api_error:
mapping:
fqcn:
Symfony\Component\HttpKernel\Exception\NotFoundHttpException: NOT_FOUND
This entry defines the association between the NotFoundHttpException
and the NOT_FOUND
error type.
According to the RFC 7807, each "problem detail" must have a type and a title. We also need to map it to a HTTP status
code. This mapping is defined in the ped_api_error.mapping.errors
section of the config file.
For instance look at the following snippet extracted from the previous example:
errors:
NOT_FOUND:
title: Not found
statusCode: 404
This entry defines the association between the NOT_FOUND
error type and the the related title and status code.
So every time the NotFoundHttpException
will be trown a response like the following will be displayed:
{
"type": "NOT_FOUND",
"title": "Not found"
}
Note that the type can be whatever you want, and you can also customize the way it is rendered simply extending the default behaviour of this bundle. For instance you could render the type as an URL like in the following example:
{
"type": "https://example.com/probs/not-found",
"title": "Not found"
}
Sometime you would also display some details. The bundle offers you a basic way to do it. You can set the
forwardMessage
to true
to tell the bundle that the exception message must be displayed to the client:
ped_api_error:
mapping:
fqcn:
App\Domain\OutOfCreditException:
type: OUT_OF_CREDIT
forwardMessage: true # Enables the exception message to be displayed
errors:
OUT_OF_CREDIT:
title: You do not have enough credit
statusCode: 400
The resulting response will be something like this:
{
"type": "OUT_OF_CREDIT",
"title": "You do not have enough credit",
"detail": "Your current balance is 30, but that costs 50."
}