RESTEasy Tutorial Series
RESTEasy Tutorial Part-1: Basics
RESTEasy Tutorial Part-2: Spring Integration
RESTEasy Tutorial Part 3 - Exception Handling
Exception Handling is an obvious requirement while developing software application. If any error occured while processing user request we should show the user an error page with details like brief exception message, error code(optional), hints to correct the input and retry(optional) and actual root cause(optional). This is applicable to RESTful web services also.But putting try-catch-finally blocks all around the code is not a good practice. We should design/code in such a way that if there is any unrecoverable error occured then the code should throw that exception and there should an exception handler to catch those exceptions and extract the error details and give a proper error response to the client with all the error details.
RESTEasy provides such ExceptionHandler mechanism which simplifies the ExceptionHandling process.
In this part I will show you how we can use RESTEasy's ExceptionHandlers to handle Exceptions.
Step#1: Create Application Specific Exceptions.
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/** | |
* ResourceNotFoundException.java | |
*/ | |
package com.sivalabs.resteasydemo; | |
public class ResourceNotFoundException extends RuntimeException | |
{ | |
private static final long serialVersionUID = 1L; | |
public ResourceNotFoundException(String msg) | |
{ | |
super(msg); | |
} | |
} | |
/** | |
* ApplicationException.java | |
*/ | |
package com.sivalabs.resteasydemo; | |
import java.io.PrintWriter; | |
import java.io.StringWriter; | |
public class ApplicationException extends RuntimeException | |
{ | |
private static final long serialVersionUID = 1L; | |
public ApplicationException() | |
{ | |
super(); | |
} | |
public ApplicationException(String message, Throwable cause) | |
{ | |
super(message, cause); | |
} | |
public ApplicationException(Throwable cause) | |
{ | |
super(cause); | |
} | |
public ApplicationException(String msg) | |
{ | |
super(msg); | |
} | |
public String getInternalErrorMessage() | |
{ | |
Throwable cause = this.getCause(); | |
if(cause != null) | |
{ | |
StringWriter sw = new StringWriter(); | |
PrintWriter pw = new PrintWriter(sw); | |
cause.printStackTrace(pw); | |
return sw.toString(); | |
} | |
return null; | |
} | |
} |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/** | |
* ResourceNotFoundExceptionHandler.java | |
*/ | |
package com.sivalabs.resteasydemo; | |
import javax.ws.rs.core.Response; | |
import javax.ws.rs.ext.ExceptionMapper; | |
import javax.ws.rs.ext.Provider; | |
import org.springframework.stereotype.Component; | |
@Provider | |
@Component | |
public class ResourceNotFoundExceptionHandler implements ExceptionMapper<ResourceNotFoundException> | |
{ | |
@Override | |
public Response toResponse(ResourceNotFoundException ex) | |
//For simplicity I am preparing error xml by hand. | |
//Ideally we should create an ErrorResponse class to hold the error info. | |
String msg = ex.getMessage(); | |
StringBuilder response = new StringBuilder("<response>"); | |
response.append("<status>failed</status>"); | |
response.append("<message>"+msg+"</message>"); | |
response.append("</response>"); | |
return Response.serverError().entity(response.toString()).build(); | |
} | |
} | |
/** | |
* ApplicationExceptionHandler.java | |
*/ | |
package com.sivalabs.resteasydemo; | |
import javax.ws.rs.core.Response; | |
import javax.ws.rs.ext.ExceptionMapper; | |
import javax.ws.rs.ext.Provider; | |
import org.springframework.stereotype.Component; | |
@Provider | |
@Component | |
public class ApplicationExceptionHandler implements ExceptionMapper<ApplicationException> | |
{ | |
@Override | |
public Response toResponse(ApplicationException ex) | |
{ | |
//For simplicity I am preparing error xml by hand. | |
//Ideally we should create an ErrorResponse class to hold the error info. | |
String msg = ex.getMessage(); | |
String internalError = ex.getInternalErrorMessage(); | |
StringBuilder response = new StringBuilder("<response>"); | |
response.append("<status>failed</status>"); | |
response.append("<message>"+msg+"</message>"); | |
response.append("<internalError>"+internalError+"</internalError>"); | |
response.append("</response>"); | |
return Response.serverError().entity(response.toString()).build(); | |
} | |
} |
Step#3: Update UserResource.getUserXMLById() method to validate user input and throw respective exceptions.
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
@Path("/{id}") | |
@GET | |
public Response getUserXMLById(@PathParam("id") Integer id) | |
{ | |
if(id==null || id < 1 ){ | |
throw new ApplicationException("User Id["+id+"] should not be less than 1."); | |
} | |
User user = userService.getById(id); | |
if(user==null ){ | |
throw new ResourceNotFoundException("No User found with Id :["+id+"]"); | |
} | |
return Response.ok(user).build(); | |
} |
Step#4: Test the UserResource.getUserXMLById() service method by issueing following requests.
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
case 1 : GET http://localhost:8080/resteasy-demo/rest/users/0 | |
Response : | |
<response> | |
<status>failed</status> | |
<message>User Id[0] should not be less than 1.</message> | |
<internalError>null</internalError> | |
</response> | |
case 2: GET http://localhost:8080/resteasy-demo/rest/users/100 | |
Response : | |
<response> | |
<status>failed</status> | |
<message>No User found with Id :[100]</message> | |
</response> |
Important things to note:
As Spring is creating the necessary objects we should let Spring know about @Provider classes to get them registered with RESTEasy. We can do this in two ways.
a)Annotate Provider classes with @Component
b)Using component-scan's include-filter.
<context:component-scan base-package="com.sivalabs.springdemo">
<context:include-filter expression="javax.ws.rs.ext.Provider" type="annotation"/>
</context:component-scan>
Hi Siva,
ReplyDeleteNice article. I need to contact you regarding a suggestion. Please drop me a mail back here http://techsamosa.com/blogs/about-us/
Hi Siva.
ReplyDeleteIts a nice article.
I am throw ApplicationException from a normal java class. Then the ApplicationExceptionHandler is not handling the exception. The ApplicationExceptionHandler handling the exceptions only when they are thrown from jersey resources(Eg : UserResource).
How can we make the ApplicaionExceptionHandler to handle exceptions which are thrown by normal java classes.
Thanks and Regards,
Siva.