/src/main/java/org/jrack/RackFilter.java

http://github.com/florinpatrascu/jrack · Java · 200 lines · 129 code · 28 blank · 43 comment · 31 complexity · 50a62f79b988176ef7b62f499e1aa8e7 MD5 · raw file

  1. package org.jrack;
  2. import org.jrack.utils.ClassUtilities;
  3. import javax.servlet.*;
  4. import javax.servlet.http.HttpServletRequest;
  5. import javax.servlet.http.HttpServletResponse;
  6. import java.io.File;
  7. import java.io.FileInputStream;
  8. import java.io.IOException;
  9. import java.nio.channels.Channels;
  10. import java.util.HashSet;
  11. import java.util.Map;
  12. import java.util.Set;
  13. /**
  14. * @author <a href="mailto:florin.patrascu@gmail.com">Florin T.PATRASCU</a>
  15. * @since $Revision$ (created: 2012-12-17 7:19 PM)
  16. */
  17. public class RackFilter implements Filter {
  18. private FilterConfig filterConfig;
  19. private ServletContext servletContext;
  20. private JRack rack;
  21. String skipWord = null;
  22. /**
  23. * set of paths that should be ignored
  24. */
  25. private final Set<String> ignorePaths = new HashSet<String>();
  26. /**
  27. * Filter initialization
  28. *
  29. * @param filterConfig
  30. * @throws ServletException
  31. */
  32. public void init(FilterConfig filterConfig) throws ServletException {
  33. this.filterConfig = filterConfig;
  34. servletContext = filterConfig.getServletContext();
  35. String rackClass = filterConfig.getInitParameter("rack");
  36. String paths = filterConfig.getInitParameter("ignore");
  37. skipWord = filterConfig.getInitParameter("skipWord");
  38. if (paths != null) {
  39. String[] parts = paths.replaceAll(" ", Rack.EMPTY_STRING).split(",");
  40. for (String path : parts) {
  41. path = path.trim();
  42. if (path.startsWith("/")) {
  43. path = path.substring(1);
  44. }
  45. ignorePaths.add(path);
  46. }
  47. }
  48. try {
  49. setRack((JRack) ClassUtilities.loadClass(rackClass).newInstance());
  50. rack.init(filterConfig);
  51. } catch (Exception e) {
  52. e.printStackTrace();
  53. throw new ServletException("Cannot load: " + rackClass);
  54. }
  55. }
  56. /**
  57. * @param request
  58. * @param response
  59. * @param chain
  60. * @throws IOException
  61. * @throws ServletException
  62. * @see Filter#doFilter(ServletRequest, ServletResponse, FilterChain)
  63. */
  64. public final void doFilter(
  65. ServletRequest request, ServletResponse response,
  66. FilterChain chain) throws IOException, ServletException {
  67. HttpServletRequest httpRequest = (HttpServletRequest) request;
  68. HttpServletResponse httpResponse = (HttpServletResponse) response;
  69. if (shouldIgnorePath(httpRequest)) {
  70. if (chain != null) {
  71. chain.doFilter(request, response);
  72. }
  73. } else {
  74. // see Rack specs
  75. // - http://rack.rubyforge.org/doc/SPEC.html
  76. try {
  77. Context<String> resp = rack.call(rack.getEnvironment(httpRequest).with(Rack.FILTER_CHAIN, chain));
  78. if (skipWord == null || httpRequest.getAttribute(skipWord) == null) {
  79. httpResponse.setStatus((Integer) resp.getObject(Rack.MESSAGE_STATUS));
  80. for (Map.Entry<String, Object> entry : resp) {
  81. if (entry.getKey().startsWith(Rack.HTTP_)) {
  82. httpResponse.setHeader(entry.getKey().substring(Rack.HTTP_.length()),
  83. (String) entry.getValue());
  84. }
  85. }
  86. RackBody body = (RackBody) resp.getObject(Rack.MESSAGE_BODY);
  87. //httpResponse.getWriter().print(rackResponse.getResponse());
  88. if (body != null) {
  89. if (body.getType() == RackBody.Type.file) {
  90. // Copy.copy(new FileInputStream(body.getBodyAsFile()), httpResponse.getOutputStream());
  91. // or use NIO?
  92. final File file = body.getBodyAsFile();
  93. if (file != null) {
  94. try (FileInputStream inputStream = new FileInputStream(file)) {
  95. inputStream.getChannel()
  96. .transferTo(0, file.length(),
  97. Channels.newChannel(httpResponse.getOutputStream()));
  98. }
  99. }
  100. } else {
  101. response.setCharacterEncoding(RackResponse.DEFAULT_ENCODING);
  102. httpResponse.getOutputStream().write(body.getBytes(RackResponse.DEFAULT_ENCODING));
  103. }
  104. } else {
  105. httpResponse.setStatus(HttpServletResponse.SC_NO_CONTENT);
  106. }
  107. }else{
  108. chain.doFilter(request, response);
  109. }
  110. } catch (Exception e) {
  111. RackServlet.throwAsError(e);
  112. }
  113. }
  114. }
  115. /**
  116. * Filter destroy event interceptor
  117. */
  118. public void destroy() {
  119. this.filterConfig = null;
  120. }
  121. /**
  122. * Filter configuration getter
  123. *
  124. * @return FilterConfiguration
  125. */
  126. public FilterConfig getFilterConfig() {
  127. return filterConfig;
  128. }
  129. public ServletContext getServletContext() {
  130. return servletContext;
  131. }
  132. public JRack getRack() {
  133. return rack;
  134. }
  135. public void setRack(JRack rack) {
  136. this.rack = rack;
  137. }
  138. /**
  139. * Returns a relative path to the filter path and context root from an HttpServletRequest
  140. */
  141. public String getRelativePath(HttpServletRequest request) {
  142. String path = request.getPathInfo();
  143. if (path == null) {
  144. path = request.getServletPath();
  145. }
  146. if (path.length() > 0) {
  147. path = path.substring(1);
  148. }
  149. return path;
  150. }
  151. private boolean shouldIgnorePath(HttpServletRequest request) {
  152. boolean ignore = false;
  153. if (ignorePaths.size() > 0) {
  154. String relativePath = getRelativePath(request);
  155. if (relativePath != null && relativePath.length() > 0) {
  156. for (String path : ignorePaths) {
  157. if (relativePath.startsWith(path)) {
  158. ignore = true;
  159. break;
  160. }
  161. }
  162. }
  163. }
  164. return ignore;
  165. }
  166. /**
  167. * todo: refactor
  168. *
  169. * @param req the HttpServletRequest
  170. * @return
  171. */
  172. }