SOLR-267: Logging updates, now logs how many hits, plus some other logging changes to reduce double logging

git-svn-id: https://svn.apache.org/repos/asf/lucene/solr/trunk@649066 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Grant Ingersoll 2008-04-17 12:37:38 +00:00
parent e2583ca6dd
commit f0a4b30cdc
6 changed files with 117 additions and 57 deletions

View File

@ -355,8 +355,11 @@ Bug Fixes
when no RequestHandler is mapped to "/update") now logs error correctly. when no RequestHandler is mapped to "/update") now logs error correctly.
(hossman) (hossman)
26. SOLR-509: Moved firstSearcher event notification to the end of the SolrCore constructor (Koji Sekiguchi via gsingers) 27. SOLR-267: Changed logging to report number of hits, and also provide a mechanism to add log messages to be output by the SolrCore via a NamedList toLog
member variable. (Will Johnson, yseeley, gsingers)
28. SOLR-509: Moved firstSearcher event notification to the end of the SolrCore constructor (Koji Sekiguchi via gsingers)
Other Changes Other Changes
1. SOLR-135: Moved common classes to org.apache.solr.common and altered the 1. SOLR-135: Moved common classes to org.apache.solr.common and altered the
build scripts to make two jars: apache-solr-1.3.jar and build scripts to make two jars: apache-solr-1.3.jar and

View File

@ -128,7 +128,7 @@ public final class SolrCore {
/** /**
* Gets the configuration resource name used by this core instance. * Gets the configuration resource name used by this core instance.
* @see getConfigResource * @see #getConfigResource()
*/ */
@Deprecated @Deprecated
public String getConfigFile() { public String getConfigFile() {
@ -151,7 +151,7 @@ public final class SolrCore {
/** /**
* Gets the schema resource name used by this core instance. * Gets the schema resource name used by this core instance.
* @see getSchemaResource * @see #getSchemaResource()
*/ */
@Deprecated @Deprecated
public String getSchemaFile() { public String getSchemaFile() {
@ -396,7 +396,7 @@ public final class SolrCore {
// Finally tell anyone who wants to know // Finally tell anyone who wants to know
loader.inform( loader ); loader.inform( loader );
loader.inform( this ); loader.inform( this );
// execute firstSearcher event // execute firstSearcher event
//TODO: It may not always be the case that this is the only time the first searcher event needs to fire. //TODO: It may not always be the case that this is the only time the first searcher event needs to fire.
doFirstSearcherEvent(getSearcher().get()); doFirstSearcherEvent(getSearcher().get());
@ -498,7 +498,7 @@ public final class SolrCore {
* Returns a Request object based on the admin/pingQuery section * Returns a Request object based on the admin/pingQuery section
* of the Solr config file. * of the Solr config file.
* *
* @use {@link PingRequestHandler} instead * @use {@link org.apache.solr.handler.PingRequestHandler} instead
*/ */
@Deprecated @Deprecated
public SolrQueryRequest getPingQueryRequest() { public SolrQueryRequest getPingQueryRequest() {
@ -948,12 +948,23 @@ public final class SolrCore {
// setup response header and handle request // setup response header and handle request
final NamedList<Object> responseHeader = new SimpleOrderedMap<Object>(); final NamedList<Object> responseHeader = new SimpleOrderedMap<Object>();
rsp.add("responseHeader", responseHeader); rsp.add("responseHeader", responseHeader);
NamedList toLog = rsp.getToLog();
//toLog.add("core", getName());
toLog.add("webapp", req.getContext().get("webapp"));
toLog.add("path", req.getContext().get("path"));
toLog.add("params", "{" + req.getParamString() + "}");
handler.handleRequest(req,rsp); handler.handleRequest(req,rsp);
setResponseHeaderValues(handler,responseHeader,req,rsp); setResponseHeaderValues(handler,req,rsp);
StringBuilder sb = new StringBuilder();
log.info(logid+"" + req.getContext().get("path") + " " for (int i=0; i<toLog.size(); i++) {
String name = toLog.getName(i);
Object val = toLog.getVal(i);
sb.append(name).append("=").append(val).append(" ");
}
log.info(logid + sb.toString());
/*log.info(logid+"" + req.getContext().get("path") + " "
+ req.getParamString()+ " 0 "+ + req.getParamString()+ " 0 "+
(int)(rsp.getEndTime() - req.getStartTime())); (int)(rsp.getEndTime() - req.getStartTime()));*/
} }
@Deprecated @Deprecated
@ -966,13 +977,15 @@ public final class SolrCore {
execute(handler, req, rsp); execute(handler, req, rsp);
} }
protected void setResponseHeaderValues(SolrRequestHandler handler, NamedList<Object> responseHeader,SolrQueryRequest req, SolrQueryResponse rsp) { protected void setResponseHeaderValues(SolrRequestHandler handler, SolrQueryRequest req, SolrQueryResponse rsp) {
// TODO should check that responseHeader has not been replaced by handler // TODO should check that responseHeader has not been replaced by handler
NamedList responseHeader = rsp.getResponseHeader();
final int qtime=(int)(rsp.getEndTime() - req.getStartTime()); final int qtime=(int)(rsp.getEndTime() - req.getStartTime());
responseHeader.add("status",rsp.getException()==null ? 0 : 500); responseHeader.add("status",rsp.getException()==null ? 0 : 500);
responseHeader.add("QTime",qtime); responseHeader.add("QTime",qtime);
rsp.getToLog().add("status",rsp.getException()==null ? 0 : 500);
rsp.getToLog().add("QTime",qtime);
SolrParams params = req.getParams(); SolrParams params = req.getParams();
if( params.getBool(CommonParams.HEADER_ECHO_HANDLER, false) ) { if( params.getBool(CommonParams.HEADER_ECHO_HANDLER, false) ) {
responseHeader.add("handler", handler.getName() ); responseHeader.add("handler", handler.getName() );

View File

@ -165,7 +165,7 @@ public class QueryComponent extends SearchComponent
} }
rsp.add("response",rb.getResults().docList); rsp.add("response",rb.getResults().docList);
rsp.getToLog().add("hits", rb.getResults().docList.size());
boolean fsv = req.getParams().getBool(ResponseBuilder.FIELD_SORT_VALUES,false); boolean fsv = req.getParams().getBool(ResponseBuilder.FIELD_SORT_VALUES,false);
if(fsv){ if(fsv){

View File

@ -60,7 +60,12 @@ public class SolrQueryResponse {
* @see #setAllValues * @see #setAllValues
* @see <a href="#returnable_data">Note on Returnable Data</a> * @see <a href="#returnable_data">Note on Returnable Data</a>
*/ */
protected NamedList values = new SimpleOrderedMap(); protected NamedList values = new SimpleOrderedMap();
/**
* Container for storing information that should be logged by Solr before returning.
*/
protected NamedList toLog = new SimpleOrderedMap();
protected Set<String> defaultReturnFields; protected Set<String> defaultReturnFields;
@ -73,6 +78,11 @@ public class SolrQueryResponse {
String errMsg; String errMsg;
***/ ***/
public SolrQueryResponse() {
add("responseHeader", new SimpleOrderedMap<Object>());
}
/** /**
* Gets data to be returned in this response * Gets data to be returned in this response
* @see <a href="#returnable_data">Note on Returnable Data</a> * @see <a href="#returnable_data">Note on Returnable Data</a>
@ -167,6 +177,28 @@ public class SolrQueryResponse {
} }
return this.endtime; return this.endtime;
} }
/** Repsonse header to be logged */
public NamedList getResponseHeader() {
SimpleOrderedMap<Object> header = (SimpleOrderedMap<Object>) values.get("responseHeader");
return header;
}
/** Add a value to be logged.
*
* @param name name of the thing to log
* @param val value of the thing to log
*/
public void addToLog(String name, Object val) {
toLog.add(name, val);
}
/** Get loggable items.
*
* @return things to log
*/
public NamedList getToLog() {
return toLog;
}
} }

View File

@ -42,12 +42,12 @@ public class JSONWriterTest extends AbstractSolrTestCase {
rsp.add("data2", Double.NEGATIVE_INFINITY); rsp.add("data2", Double.NEGATIVE_INFINITY);
rsp.add("data3", Float.POSITIVE_INFINITY); rsp.add("data3", Float.POSITIVE_INFINITY);
w.write(buf, req, rsp); w.write(buf, req, rsp);
assertEquals(buf.toString(), "{'data1':float('NaN'),'data2':-float('Inf'),'data3':float('Inf')}"); assertEquals(buf.toString(), "{'responseHeader':{},'data1':float('NaN'),'data2':-float('Inf'),'data3':float('Inf')}");
w = new RubyResponseWriter(); w = new RubyResponseWriter();
buf = new StringWriter(); buf = new StringWriter();
w.write(buf, req, rsp); w.write(buf, req, rsp);
assertEquals(buf.toString(), "{'data1'=>(0.0/0.0),'data2'=>-(1.0/0.0),'data3'=>(1.0/0.0)}"); assertEquals(buf.toString(), "{'responseHeader'=>{},'data1'=>(0.0/0.0),'data2'=>-(1.0/0.0),'data3'=>(1.0/0.0)}");
} }

View File

@ -35,6 +35,7 @@ import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse; import javax.servlet.http.HttpServletResponse;
import org.apache.solr.common.SolrException; import org.apache.solr.common.SolrException;
import org.apache.solr.common.util.NamedList;
import org.apache.solr.common.params.CommonParams; import org.apache.solr.common.params.CommonParams;
import org.apache.solr.core.MultiCore; import org.apache.solr.core.MultiCore;
import org.apache.solr.core.SolrConfig; import org.apache.solr.core.SolrConfig;
@ -46,21 +47,21 @@ import org.apache.solr.servlet.cache.Method;
/** /**
* This filter looks at the incoming URL maps them to handlers defined in solrconfig.xml * This filter looks at the incoming URL maps them to handlers defined in solrconfig.xml
* *
* @since solr 1.2 * @since solr 1.2
*/ */
public class SolrDispatchFilter implements Filter public class SolrDispatchFilter implements Filter
{ {
final Logger log = Logger.getLogger(SolrDispatchFilter.class.getName()); final Logger log = Logger.getLogger(SolrDispatchFilter.class.getName());
protected SolrCore singlecore; protected SolrCore singlecore;
protected MultiCore multicore; protected MultiCore multicore;
protected String pathPrefix = null; // strip this from the beginning of a path protected String pathPrefix = null; // strip this from the beginning of a path
protected String abortErrorMessage = null; protected String abortErrorMessage = null;
protected final WeakHashMap<SolrCore, SolrRequestParsers> parsers = new WeakHashMap<SolrCore, SolrRequestParsers>(); protected final WeakHashMap<SolrCore, SolrRequestParsers> parsers = new WeakHashMap<SolrCore, SolrRequestParsers>();
protected String solrConfigFilename = null; protected String solrConfigFilename = null;
public void init(FilterConfig config) throws ServletException public void init(FilterConfig config) throws ServletException
{ {
log.info("SolrDispatchFilter.init()"); log.info("SolrDispatchFilter.init()");
@ -69,10 +70,10 @@ public class SolrDispatchFilter implements Filter
// web.xml configuration // web.xml configuration
this.pathPrefix = config.getInitParameter( "path-prefix" ); this.pathPrefix = config.getInitParameter( "path-prefix" );
this.solrConfigFilename = config.getInitParameter("solrconfig-filename"); this.solrConfigFilename = config.getInitParameter("solrconfig-filename");
// multicore instantiation // multicore instantiation
this.multicore = initMultiCore(config); this.multicore = initMultiCore(config);
if(multicore != null && multicore.isEnabled() ) { if(multicore != null && multicore.isEnabled() ) {
abortOnConfigurationError = false; abortOnConfigurationError = false;
singlecore = null; singlecore = null;
@ -98,7 +99,7 @@ public class SolrDispatchFilter implements Filter
SolrConfig.severeErrors.add( t ); SolrConfig.severeErrors.add( t );
SolrCore.log( t ); SolrCore.log( t );
} }
// Optionally abort if we found a sever error // Optionally abort if we found a sever error
if( abortOnConfigurationError && SolrConfig.severeErrors.size() > 0 ) { if( abortOnConfigurationError && SolrConfig.severeErrors.size() > 0 ) {
StringWriter sw = new StringWriter(); StringWriter sw = new StringWriter();
@ -112,13 +113,13 @@ public class SolrDispatchFilter implements Filter
} else { } else {
out.println( "in solrconfig.xml\n" ); out.println( "in solrconfig.xml\n" );
} }
for( Throwable t : SolrConfig.severeErrors ) { for( Throwable t : SolrConfig.severeErrors ) {
out.println( "-------------------------------------------------------------" ); out.println( "-------------------------------------------------------------" );
t.printStackTrace( out ); t.printStackTrace( out );
} }
out.flush(); out.flush();
// Servlet containers behave slightly differently if you throw an exception during // Servlet containers behave slightly differently if you throw an exception during
// initialization. Resin will display that error for every page, jetty prints it in // initialization. Resin will display that error for every page, jetty prints it in
// the logs, but continues normally. (We will see a 404 rather then the real error) // the logs, but continues normally. (We will see a 404 rather then the real error)
@ -127,7 +128,7 @@ public class SolrDispatchFilter implements Filter
abortErrorMessage = sw.toString(); abortErrorMessage = sw.toString();
//throw new ServletException( abortErrorMessage ); //throw new ServletException( abortErrorMessage );
} }
log.info("SolrDispatchFilter.init() done"); log.info("SolrDispatchFilter.init() done");
} }
@ -159,13 +160,13 @@ public class SolrDispatchFilter implements Filter
singlecore = null; singlecore = null;
} }
} }
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException { public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
if( abortErrorMessage != null ) { if( abortErrorMessage != null ) {
((HttpServletResponse)response).sendError( 500, abortErrorMessage ); ((HttpServletResponse)response).sendError( 500, abortErrorMessage );
return; return;
} }
if( request instanceof HttpServletRequest) { if( request instanceof HttpServletRequest) {
HttpServletRequest req = (HttpServletRequest)request; HttpServletRequest req = (HttpServletRequest)request;
HttpServletResponse resp = (HttpServletResponse)response; HttpServletResponse resp = (HttpServletResponse)response;
@ -173,26 +174,26 @@ public class SolrDispatchFilter implements Filter
SolrQueryRequest solrReq = null; SolrQueryRequest solrReq = null;
try { try {
String path = req.getServletPath(); String path = req.getServletPath();
if( req.getPathInfo() != null ) { if( req.getPathInfo() != null ) {
// this lets you handle /update/commit when /update is a servlet // this lets you handle /update/commit when /update is a servlet
path += req.getPathInfo(); path += req.getPathInfo();
} }
if( pathPrefix != null && path.startsWith( pathPrefix ) ) { if( pathPrefix != null && path.startsWith( pathPrefix ) ) {
path = path.substring( pathPrefix.length() ); path = path.substring( pathPrefix.length() );
} }
int idx = path.indexOf( ':' ); int idx = path.indexOf( ':' );
if( idx > 0 ) { if( idx > 0 ) {
// save the portion after the ':' for a 'handler' path parameter // save the portion after the ':' for a 'handler' path parameter
path = path.substring( 0, idx ); path = path.substring( 0, idx );
} }
// By default use the single core. If multicore is enabled, look for one. // By default use the single core. If multicore is enabled, look for one.
final SolrCore core; final SolrCore core;
if (multicore != null && multicore.isEnabled()) { if (multicore != null && multicore.isEnabled()) {
req.setAttribute("org.apache.solr.MultiCore", multicore); req.setAttribute("org.apache.solr.MultiCore", multicore);
// if this is the multi-core admin page, it will handle it // if this is the multi-core admin page, it will handle it
if( path.equals( multicore.getAdminPath() ) ) { if( path.equals( multicore.getAdminPath() ) ) {
handler = multicore.getMultiCoreHandler(); handler = multicore.getMultiCoreHandler();
@ -217,7 +218,7 @@ public class SolrDispatchFilter implements Filter
else { else {
core = singlecore; core = singlecore;
} }
// With a valid core... // With a valid core...
if( core != null ) { if( core != null ) {
final SolrConfig config = core.getSolrConfig(); final SolrConfig config = core.getSolrConfig();
@ -228,7 +229,7 @@ public class SolrDispatchFilter implements Filter
parser = new SolrRequestParsers(config); parser = new SolrRequestParsers(config);
parsers.put( core, parser ); parsers.put( core, parser );
} }
// Determine the handler from the url path if not set // Determine the handler from the url path if not set
// (we might already have selected the multicore handler) // (we might already have selected the multicore handler)
if( handler == null && path.length() > 1 ) { // don't match "" or "/" as valid path if( handler == null && path.length() > 1 ) { // don't match "" or "/" as valid path
@ -244,18 +245,18 @@ public class SolrDispatchFilter implements Filter
handler = core.getRequestHandler( qt ); handler = core.getRequestHandler( qt );
if( handler == null ) { if( handler == null ) {
throw new SolrException( SolrException.ErrorCode.BAD_REQUEST, "unknown handler: "+qt); throw new SolrException( SolrException.ErrorCode.BAD_REQUEST, "unknown handler: "+qt);
} }
} }
} }
} }
// With a valid handler and a valid core... // With a valid handler and a valid core...
if( handler != null ) { if( handler != null ) {
// if not a /select, create the request // if not a /select, create the request
if( solrReq == null ) { if( solrReq == null ) {
solrReq = parser.parse( core, path, req ); solrReq = parser.parse( core, path, req );
} }
final Method reqMethod = Method.getMethod(req.getMethod()); final Method reqMethod = Method.getMethod(req.getMethod());
if (Method.POST != reqMethod) { if (Method.POST != reqMethod) {
HttpCacheHeaderUtil.setCacheControlHeader(config, resp); HttpCacheHeaderUtil.setCacheControlHeader(config, resp);
@ -266,11 +267,20 @@ public class SolrDispatchFilter implements Filter
!HttpCacheHeaderUtil.doCacheHeaderValidation(solrReq, req, resp)) { !HttpCacheHeaderUtil.doCacheHeaderValidation(solrReq, req, resp)) {
SolrQueryResponse solrRsp = new SolrQueryResponse(); SolrQueryResponse solrRsp = new SolrQueryResponse();
/* even for HEAD requests, we need to execute the handler to /* even for HEAD requests, we need to execute the handler to
* ensure we don't get an error (and to make sure the correct * ensure we don't get an error (and to make sure the correct
* QueryResponseWriter is selectedand we get the correct * QueryResponseWriter is selectedand we get the correct
* Content-Type) * Content-Type)
*/ */
this.execute( req, handler, solrReq, solrRsp ); this.execute( req, handler, solrReq, solrRsp );
// add info to http headers
try {
NamedList solrRspHeader = solrRsp.getResponseHeader();
for (int i=0; i<solrRspHeader.size(); i++) {
((javax.servlet.http.HttpServletResponse) response).addHeader(("Solr-" + solrRspHeader.getName(i)), String.valueOf(solrRspHeader.getVal(i)));
}
} catch (ClassCastException cce) {
log.log(Level.WARNING, "exception adding response header log information", cce);
}
if( solrRsp.getException() != null ) { if( solrRsp.getException() != null ) {
sendError( (HttpServletResponse)response, solrRsp.getException() ); sendError( (HttpServletResponse)response, solrRsp.getException() );
} }
@ -301,7 +311,7 @@ public class SolrDispatchFilter implements Filter
// Modify the request so each core gets its own /admin // Modify the request so each core gets its own /admin
if( singlecore == null && path.startsWith( "/admin" ) ) { if( singlecore == null && path.startsWith( "/admin" ) ) {
req.getRequestDispatcher( pathPrefix == null ? path : pathPrefix + path ).forward( request, response ); req.getRequestDispatcher( pathPrefix == null ? path : pathPrefix + path ).forward( request, response );
return; return;
} }
} }
} }
@ -315,7 +325,7 @@ public class SolrDispatchFilter implements Filter
} }
} }
} }
// Otherwise let the webapp handle the request // Otherwise let the webapp handle the request
chain.doFilter(request, response); chain.doFilter(request, response);
} }
@ -323,24 +333,26 @@ public class SolrDispatchFilter implements Filter
protected void execute( HttpServletRequest req, SolrRequestHandler handler, SolrQueryRequest sreq, SolrQueryResponse rsp) { protected void execute( HttpServletRequest req, SolrRequestHandler handler, SolrQueryRequest sreq, SolrQueryResponse rsp) {
// a custom filter could add more stuff to the request before passing it on. // a custom filter could add more stuff to the request before passing it on.
// for example: sreq.getContext().put( "HttpServletRequest", req ); // for example: sreq.getContext().put( "HttpServletRequest", req );
// used for logging query stats in SolrCore.execute()
sreq.getContext().put( "webapp", req.getContextPath() );
sreq.getCore().execute( handler, sreq, rsp ); sreq.getCore().execute( handler, sreq, rsp );
} }
protected void sendError(HttpServletResponse res, Throwable ex) throws IOException { protected void sendError(HttpServletResponse res, Throwable ex) throws IOException {
int code=500; int code=500;
String trace = ""; String trace = "";
if( ex instanceof SolrException ) { if( ex instanceof SolrException ) {
code = ((SolrException)ex).code(); code = ((SolrException)ex).code();
} }
// For any regular code, don't include the stack trace // For any regular code, don't include the stack trace
if( code == 500 || code < 100 ) { if( code == 500 || code < 100 ) {
StringWriter sw = new StringWriter(); StringWriter sw = new StringWriter();
ex.printStackTrace(new PrintWriter(sw)); ex.printStackTrace(new PrintWriter(sw));
trace = "\n\n"+sw.toString(); trace = "\n\n"+sw.toString();
SolrException.logOnce(log,null,ex ); SolrException.logOnce(log,null,ex );
// non standard codes have undefined results with various servers // non standard codes have undefined results with various servers
if( code < 100 ) { if( code < 100 ) {
log.warning( "invalid return code: "+code ); log.warning( "invalid return code: "+code );
@ -348,7 +360,7 @@ public class SolrDispatchFilter implements Filter
} }
} }
res.sendError( code, ex.getMessage() + trace ); res.sendError( code, ex.getMessage() + trace );
} }
//--------------------------------------------------------------------- //---------------------------------------------------------------------
//--------------------------------------------------------------------- //---------------------------------------------------------------------
@ -357,22 +369,22 @@ public class SolrDispatchFilter implements Filter
* Set the prefix for all paths. This is useful if you want to apply the * Set the prefix for all paths. This is useful if you want to apply the
* filter to something other then /*, perhaps because you are merging this * filter to something other then /*, perhaps because you are merging this
* filter into a larger web application. * filter into a larger web application.
* *
* For example, if web.xml specifies: * For example, if web.xml specifies:
* *
* <filter-mapping> * <filter-mapping>
* <filter-name>SolrRequestFilter</filter-name> * <filter-name>SolrRequestFilter</filter-name>
* <url-pattern>/xxx/*</url-pattern> * <url-pattern>/xxx/*</url-pattern>
* </filter-mapping> * </filter-mapping>
* *
* Make sure to set the PathPrefix to "/xxx" either with this function * Make sure to set the PathPrefix to "/xxx" either with this function
* or in web.xml. * or in web.xml.
* *
* <init-param> * <init-param>
* <param-name>path-prefix</param-name> * <param-name>path-prefix</param-name>
* <param-value>/xxx</param-value> * <param-value>/xxx</param-value>
* </init-param> * </init-param>
* *
*/ */
public void setPathPrefix(String pathPrefix) { public void setPathPrefix(String pathPrefix) {
this.pathPrefix = pathPrefix; this.pathPrefix = pathPrefix;