/* (c) 2014 Open Source Geospatial Foundation - all rights reserved
* (c) 2001 - 2013 OpenPlans
* This code is licensed under the GPL 2.0 license, available at the root
* application directory.
*/
package org.geoserver.filters;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.logging.Logger;
import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServletRequest;
import org.apache.commons.io.IOUtils;
/**
* Filter to log requests for debugging or statistics-gathering purposes.
*
* @author David Winslow <dwinslow@openplans.org>
*/
public class LoggingFilter implements Filter {
protected Logger logger =
org.geotools.util.logging.Logging.getLogger("org.geoserver.filters");
protected boolean enabled = true;
protected boolean logBodies = true;
public void doFilter(ServletRequest req, ServletResponse res,
FilterChain chain) throws IOException, ServletException {
String message = "";
String body = null;
String path = "";
if (enabled){
if (req instanceof HttpServletRequest){
HttpServletRequest hreq = (HttpServletRequest) req;
path = hreq.getRemoteHost() + " \"" + hreq.getMethod() + " " + hreq.getRequestURI();
if (hreq.getQueryString() != null){
path += "?" + hreq.getQueryString();
}
path += "\"";
message = "" + path;
message += " \"" + noNull(hreq.getHeader("User-Agent"));
message += "\" \"" + noNull(hreq.getHeader("Referer")) + "\" ";
message += "\" \"" + noNull(hreq.getHeader("Content-type")) + "\" ";
if (logBodies && (hreq.getMethod().equals("PUT") || hreq.getMethod().equals("POST"))){
message += " request-size: " + hreq.getContentLength();
message += " body: ";
String encoding = hreq.getCharacterEncoding();
if (encoding == null) {
// the default encoding for HTTP 1.1
encoding = "ISO-8859-1";
}
ByteArrayOutputStream bos = new ByteArrayOutputStream();
byte[] bytes;
try (InputStream is = hreq.getInputStream()) {
IOUtils.copy(is, bos);
bytes = bos.toByteArray();
body = new String(bytes, encoding);
}
req = new BufferedRequestWrapper(hreq, encoding, bytes);
}
} else {
message = "" + req.getRemoteHost() + " made a non-HTTP request";
}
logger.info(message + (body == null? "" : "\n" + body + "\n"));
long startTime = System.currentTimeMillis();
chain.doFilter(req, res);
long requestTime = System.currentTimeMillis() - startTime;
logger.info(path + " took " + requestTime + "ms");
} else {
chain.doFilter(req, res);
}
}
public void init(FilterConfig filterConfig) {
enabled = getConfigBool("enabled", filterConfig);
logBodies = getConfigBool("log-request-bodies", filterConfig);
}
protected boolean getConfigBool(String name, FilterConfig conf){
try {
String value = conf.getInitParameter(name);
return Boolean.valueOf(value).booleanValue();
} catch (Exception e){
return false;
}
}
protected String noNull(String s){
if (s == null) return "";
return s;
}
public void destroy() {
}
/**
* @return the enabled
*/
public boolean isEnabled() {
return enabled;
}
/**
* @param enabled the enabled to set
*/
public void setEnabled(boolean enabled) {
this.enabled = enabled;
}
/**
* @return the logBodies
*/
public boolean isLogBodies() {
return logBodies;
}
/**
* @param logBodies the logBodies to set
*/
public void setLogBodies(boolean logBodies) {
this.logBodies = logBodies;
}
}