The error reporting endpoint was designed to allow the frontend developers of your API to report any bugs they encounter in a secure way that is fully under your control. To prevent unauthorized usage, the endpoint is protected by a token in the request's header.
Error reporting is done by sending a POST request to the /error-report
endpoint, together with a token in the header.
In the sections below we will detail how to configure error reporting in your API and how the endpoint is used by the frontend developers.
First you need to generate a token for your request. This is done by using the command
php ./bin/cli.php token:generate error-reporting
The resulting token has this format 0123456789abcdef0123456789abcdef01234567
.
Note: this example is not a valid token, it just lets you know what to look for.
Copy the generated token in your config/autoload/error-handling.global.php
file.
It should look similar to the example below.
Your API can have multiple tokens, if needed.
return [
...
ErrorReportServiceInterface::class => [
...
'tokens' => [
'0123456789abcdef0123456789abcdef01234567',
],
...
]
]
Behind the scenes, the API validates your configuration and lets you know if any config items prevent the submission of the error report. Below are the requirements for an application to be able to send error messages to Dotkernel API.
config/autoload/error-handling.global.php
(these can be set/overwritten in config/autoload/local.php
):enabled
, path
, tokens
, domain_whitelist
and ip_whitelist
) must exist under ErrorReportServiceInterface::class
.ErrorReportServiceInterface::class
. enabled
to true
.ErrorReportServiceInterface::class
. path
must have a value; if the destination file does not exist, it will be created automatically.ErrorReportServiceInterface::class
. tokens
must contain at least one token.ErrorReportServiceInterface::class
. domain_whitelist
/ip_whitelist
must have at least one value.Note: In src/App/src/Service/ErrorReportService.php
, the method checkRequest()
tries to validate the request by checking matches for domain_whitelist
with isMatchingDomain()
and for ip_whitelist
with isMatchingIpAddress()
.
If both return false
, a ForbiddenException
is thrown and the error message does not get stored.
Error-Reporting-Token
header with a valid token previously stored in config/autoload/error-handling.global.php
in the ErrorReportServiceInterface::class
. tokens
array.Origin
header set to the application's URL; this is the application that sends the error message.Note:
ErrorReportServiceInterface::class
. tokens
do not expire.If your request passes all the checks, the message is saved in the log file specified in ErrorReportServiceInterface::class
. path
.
If there are multiple applications that report errors to your API, you can assign a different error reporting token for each. The tokens support key-value pairs where:
Example:
// ...
return [
...
ErrorReportServiceInterface::class => [
// ...
'tokens' => [
'frontend' => '0123456789abcdef0123456789abcdef01234567',
'admin' => '9876543210abcdef0123456789abcdef7654321',
// other tokens
],
],
];
The log file will have entries similar to the below:
[2024-08-29 12:47:00] [0123456789abcdef0123456789abcdef01234567] Demo error message
The inclusion of the token helps you identify the source of the error message.
In our example, it's the application that uses the 0123456789abcdef0123456789abcdef01234567
token, which is assigned to the application frontend
.
The API developer sends a generated token to the frontend developer who will save it in their environment.staging.ts
and/or environment.prod.ts
.
From then on, it's the frontend developer's job to set up an error reporting function similar to the one below.
postError(body: object): Promise<any> {
return new Promise((resolve, reject) => {
return this.http.post(API_ENDPOINT + 'error-report', body , {headers: new HttpHeaders({'Error-Reporting-Token': 'TOKEN', 'Origin': 'https://example.com'})})).subscribe({
next: (response: any) => {
resolve(response);
},
error: (e: HttpErrorResponse) => reject(e),
complete: () => console.info('Error on sending error'),
});
});
}
Whenever an error is found, the frontend will call postError()
with a relevant description under message
.
apiService.postError({message: 'ERROR MESSAGE'})