API Problem implements application/problem+json responses. It comes with convenient factories and builders for most common use cases.
There are cases in which an HTTP status code is enough to convey the necessary information.
ApiProblem.from(Status.NOT_FOUND);
The above call will produce the following response:
{
"title": "Not Found",
"status": 404,
"type": "https://www.w3.org/Protocols/rfc2616/rfc2616-sec10.html#sec10.4.5"
}
In some scenarios a type
might not be necessary.
As specified by the Predefined Problem Types section:
The "about:blank" URI, when used as a problem type, indicates that the problem has no additional semantics beyond that of the HTTP status code.
If you want to omit the type
, please use the following factory.
ApiProblem.witDefaultType(Status.NOT_FOUND);
It will produce the following response:
{
"title": "Not Found",
"status": 404
}
Most of the time you'll need to define problem types that are unique to your application.
The ApiProblemBuilder
class offers a fluent API to create problem instances without the need to create custom classes:
import static com.montealegreluis.apiproblem.ApiProblemBuilder.aProblem;
// ...
aProblem()
.withType(URI.create("https://example.org/not-found"))
.withTitle("Concert not found")
.withStatus(Status.NOT_FOUND.code())
.withDetail("Concert with ID 2ca332b9-3f69-4882-abc9-58f534217bdd cannot be found")
.with("concertId", "2ca332b9-3f69-4882-abc9-58f534217bdd")
.build();
Producing the following response.
{
"type": "https://example.org/not-found",
"title": "Concert not found",
"status": 404,
"details": "Concert with ID 2ca332b9-3f69-4882-abc9-58f534217bdd cannot be found",
"concertId": "2ca332b9-3f69-4882-abc9-58f534217bdd"
}
The final stage of customization is achieved by sub-classing ApiProblem
.
public final class ConcertNotFoundProblem extends ApiProblem {
private static final URI TYPE = URI.create("https://example.com/not-found");
private final String concertId;
public ConcertNotFoundProblem(final String concertId) {
super("Concert not found", 404, TYPE, format("Concert with ID %s cannot be found", concertId), null, null);
this.concertId = concertId;
}
public String getConcertId() {
return concertId;
}
}
// ...
new ConcertNotFoundProblem("e2e864c7-9646-4219-ab96-6f9eafdcf7a5");
It is also possible to include exception information in an API Problem response.
aProblem()
.from(INTERNAL_SERVER_ERROR)
.withDetails(exception.getMessage())
.withException(exception)
.build());
The example above would be represented as a JSON response as follows.
{
"type": "https://www.w3.org/Protocols/rfc2616/rfc2616-sec10.html#sec10.5.1",
"title": "Internal Server Error",
"status": 500,
"details": "For input string \"two\"",
"exception": {
"message": "For input string \"two\"",
"class": "java.lang.NumberFormatException",
"line": 65,
"file": "NumberFormatException.java",
"trace": [
"java.base/java.lang.NumberFormatException.forInputString(NumberFormatException.java:65)",
"java.base/java.lang.Integer.parseInt(Integer.java:652)",
"java.base/java.lang.Integer.parseInt(Integer.java:770)",
"com.montealegreluis.activityfeed.Application.main(Application.java:10)"
]
}
}
The response will include previous exceptions if available.
Please refer to CONTRIBUTING for information on how to contribute to API Problem.
Released under the BSD-3-Clause.