SOLR-6275: Improve accuracy of QTime reporting

git-svn-id: https://svn.apache.org/repos/asf/lucene/dev/trunk@1663829 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Ramkumar Aiyengar 2015-03-03 23:24:50 +00:00
parent 8d6657a246
commit b29aef7954
12 changed files with 118 additions and 66 deletions

View File

@ -237,6 +237,8 @@ Other Changes
* SOLR-7151: SolrClient query methods throw IOException (Alan Woodward) * SOLR-7151: SolrClient query methods throw IOException (Alan Woodward)
* SOLR-6275: Improve accuracy of QTime reporting (Ramkumar Aiyengar)
================== 5.0.0 ================== ================== 5.0.0 ==================
Consult the LUCENE_CHANGES.txt file for additional, low level, changes in this release. Consult the LUCENE_CHANGES.txt file for additional, low level, changes in this release.

View File

@ -2031,9 +2031,9 @@ public final class SolrCore implements SolrInfoMBean, Closeable {
requestLog.info(rsp.getToLogAsString(logid)); requestLog.info(rsp.getToLogAsString(logid));
} }
if (log.isWarnEnabled()) { if (log.isWarnEnabled() && slowQueryThresholdMillis >= 0) {
final int qtime = (int)(rsp.getEndTime() - req.getStartTime()); final long qtime = (long) (req.getRequestTimer().getTime());
if (slowQueryThresholdMillis >= 0 && qtime >= slowQueryThresholdMillis) { if (qtime >= slowQueryThresholdMillis) {
log.warn("slow: " + rsp.getToLogAsString(logid)); log.warn("slow: " + rsp.getToLogAsString(logid));
} }
} }
@ -2067,7 +2067,7 @@ public final class SolrCore implements SolrInfoMBean, Closeable {
(SolrRequestHandler handler, SolrQueryRequest req, SolrQueryResponse rsp) { (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<Object> responseHeader = rsp.getResponseHeader(); NamedList<Object> responseHeader = rsp.getResponseHeader();
final int qtime=(int)(rsp.getEndTime() - req.getStartTime()); final int qtime=(int)(req.getRequestTimer().getTime());
int status = 0; int status = 0;
Exception exception = rsp.getException(); Exception exception = rsp.getException();
if( exception != null ){ if( exception != null ){

View File

@ -169,7 +169,6 @@ public class SearchHandler extends RequestHandlerBase implements SolrCoreAware ,
return components; return components;
} }
@Override @Override
public void handleRequestBody(SolrQueryRequest req, SolrQueryResponse rsp) throws Exception public void handleRequestBody(SolrQueryRequest req, SolrQueryResponse rsp) throws Exception
{ {
@ -190,8 +189,7 @@ public class SearchHandler extends RequestHandlerBase implements SolrCoreAware ,
SolrPluginUtils.getDebugInterests(req.getParams().getParams(CommonParams.DEBUG), rb); SolrPluginUtils.getDebugInterests(req.getParams().getParams(CommonParams.DEBUG), rb);
} }
final RTimer timer = rb.isDebug() ? new RTimer() : null; final RTimer timer = rb.isDebug() ? req.getRequestTimer() : null;
ShardHandler shardHandler1 = shardHandlerFactory.getShardHandler(); ShardHandler shardHandler1 = shardHandlerFactory.getShardHandler();
shardHandler1.checkDistributed(rb); shardHandler1.checkDistributed(rb);
@ -237,7 +235,6 @@ public class SearchHandler extends RequestHandlerBase implements SolrCoreAware ,
rb.getTimer().stop(); rb.getTimer().stop();
} }
subt.stop(); subt.stop();
timer.stop();
// add the timing info // add the timing info
if (rb.isDebugTimings()) { if (rb.isDebugTimings()) {
@ -395,7 +392,7 @@ public class SearchHandler extends RequestHandlerBase implements SolrCoreAware ,
nl.add("maxScore", rb.getResults().docList.maxScore()); nl.add("maxScore", rb.getResults().docList.maxScore());
} }
nl.add("shardAddress", rb.shortCircuitedURL); nl.add("shardAddress", rb.shortCircuitedURL);
nl.add("time", rsp.getEndTime()-req.getStartTime()); // elapsed time of this request so far nl.add("time", req.getRequestTimer().getTime()); // elapsed time of this request so far
int pos = rb.shortCircuitedURL.indexOf("://"); int pos = rb.shortCircuitedURL.indexOf("://");
String shardInfoName = pos != -1 ? rb.shortCircuitedURL.substring(pos+3) : rb.shortCircuitedURL; String shardInfoName = pos != -1 ? rb.shortCircuitedURL.substring(pos+3) : rb.shortCircuitedURL;

View File

@ -20,6 +20,7 @@ package org.apache.solr.request;
import org.apache.solr.response.SolrQueryResponse; import org.apache.solr.response.SolrQueryResponse;
import org.apache.solr.search.SolrIndexSearcher; import org.apache.solr.search.SolrIndexSearcher;
import org.apache.solr.schema.IndexSchema; import org.apache.solr.schema.IndexSchema;
import org.apache.solr.util.RTimer;
import org.apache.solr.common.params.SolrParams; import org.apache.solr.common.params.SolrParams;
import org.apache.solr.common.util.ContentStream; import org.apache.solr.common.util.ContentStream;
import org.apache.solr.core.SolrCore; import org.apache.solr.core.SolrCore;
@ -63,9 +64,15 @@ public interface SolrQueryRequest {
*/ */
public void close(); public void close();
/** The start time of this request in milliseconds */ /** The start time of this request in milliseconds.
* Use this only if you need the absolute system time at the start of the request,
* getRequestTimer() provides a more accurate mechanism for timing purposes.
*/
public long getStartTime(); public long getStartTime();
/** The timer for this request, created when the request started being processed */
public RTimer getRequestTimer();
/** The index searcher associated with this request */ /** The index searcher associated with this request */
public SolrIndexSearcher getSearcher(); public SolrIndexSearcher getSearcher();

View File

@ -19,6 +19,7 @@ package org.apache.solr.request;
import org.apache.solr.response.SolrQueryResponse; import org.apache.solr.response.SolrQueryResponse;
import org.apache.solr.search.SolrIndexSearcher; import org.apache.solr.search.SolrIndexSearcher;
import org.apache.solr.util.RTimer;
import org.apache.solr.util.RefCounted; import org.apache.solr.util.RefCounted;
import org.apache.solr.schema.IndexSchema; import org.apache.solr.schema.IndexSchema;
import org.apache.solr.common.params.SolrParams; import org.apache.solr.common.params.SolrParams;
@ -50,10 +51,17 @@ public abstract class SolrQueryRequestBase implements SolrQueryRequest, Closeabl
protected Map<Object,Object> context; protected Map<Object,Object> context;
protected Iterable<ContentStream> streams; protected Iterable<ContentStream> streams;
public SolrQueryRequestBase(SolrCore core, SolrParams params) { private final RTimer requestTimer;
public SolrQueryRequestBase(SolrCore core, SolrParams params, RTimer requestTimer) {
this.core = core; this.core = core;
this.schema = null == core ? null : core.getLatestSchema(); this.schema = null == core ? null : core.getLatestSchema();
this.params = this.origParams = params; this.params = this.origParams = params;
this.requestTimer = requestTimer;
}
public SolrQueryRequestBase(SolrCore core, SolrParams params) {
this(core, params, new RTimer());
} }
@Override @Override
@ -85,6 +93,10 @@ public abstract class SolrQueryRequestBase implements SolrQueryRequest, Closeabl
return startTime; return startTime;
} }
public RTimer getRequestTimer () {
return requestTimer;
}
// The index searcher associated with this request // The index searcher associated with this request
protected RefCounted<SolrIndexSearcher> searcherHolder; protected RefCounted<SolrIndexSearcher> searcherHolder;
@Override @Override

View File

@ -171,44 +171,7 @@ public class SolrQueryResponse {
return err; return err;
} }
/** /** Response header to be logged */
* The endtime of the request in milliseconds.
* Used to calculate query time.
* @see #setEndTime(long)
* @see #getEndTime()
*/
protected long endtime;
/**
* Get the time in milliseconds when the response officially finished.
*/
public long getEndTime() {
if (endtime==0) {
setEndTime();
}
return endtime;
}
/**
* Stop the timer for how long this query took.
* @see #setEndTime(long)
*/
public long setEndTime() {
return setEndTime(System.currentTimeMillis());
}
/**
* Set the in milliseconds when the response officially finished.
* @see #setEndTime()
*/
public long setEndTime(long endtime) {
if (endtime!=0) {
this.endtime=endtime;
}
return this.endtime;
}
/** Repsonse header to be logged */
public NamedList<Object> getResponseHeader() { public NamedList<Object> getResponseHeader() {
@SuppressWarnings("unchecked") @SuppressWarnings("unchecked")
SimpleOrderedMap<Object> header = (SimpleOrderedMap<Object>) values.get("responseHeader"); SimpleOrderedMap<Object> header = (SimpleOrderedMap<Object>) values.get("responseHeader");

View File

@ -70,6 +70,7 @@ import org.apache.solr.response.SolrQueryResponse;
import org.apache.solr.servlet.cache.HttpCacheHeaderUtil; import org.apache.solr.servlet.cache.HttpCacheHeaderUtil;
import org.apache.solr.servlet.cache.Method; import org.apache.solr.servlet.cache.Method;
import org.apache.solr.update.processor.DistributingUpdateProcessorFactory; import org.apache.solr.update.processor.DistributingUpdateProcessorFactory;
import org.apache.solr.util.RTimer;
import org.slf4j.Logger; import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
@ -233,6 +234,8 @@ public class SolrDispatchFilter extends BaseSolrFilter {
String corename = ""; String corename = "";
String origCorename = null; String origCorename = null;
try { try {
// set a request timer which can be reused by requests if needed
req.setAttribute(SolrRequestParsers.REQUEST_TIMER_SERVLET_ATTRIBUTE, new RTimer());
// put the core container in request attribute // put the core container in request attribute
req.setAttribute("org.apache.solr.CoreContainer", cores); req.setAttribute("org.apache.solr.CoreContainer", cores);
String path = req.getServletPath(); String path = req.getServletPath();

View File

@ -58,6 +58,7 @@ import org.apache.solr.core.SolrConfig;
import org.apache.solr.core.SolrCore; import org.apache.solr.core.SolrCore;
import org.apache.solr.request.SolrQueryRequest; import org.apache.solr.request.SolrQueryRequest;
import org.apache.solr.request.SolrQueryRequestBase; import org.apache.solr.request.SolrQueryRequestBase;
import org.apache.solr.util.RTimer;
import org.slf4j.Logger; import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
@ -78,6 +79,8 @@ public class SolrRequestParsers
public static final String INPUT_ENCODING_KEY = "ie"; public static final String INPUT_ENCODING_KEY = "ie";
private static final byte[] INPUT_ENCODING_BYTES = INPUT_ENCODING_KEY.getBytes(CHARSET_US_ASCII); private static final byte[] INPUT_ENCODING_BYTES = INPUT_ENCODING_KEY.getBytes(CHARSET_US_ASCII);
public static final String REQUEST_TIMER_SERVLET_ATTRIBUTE = "org.apache.solr.RequestTimer";
private final HashMap<String, SolrRequestParser> parsers = private final HashMap<String, SolrRequestParser> parsers =
new HashMap<>(); new HashMap<>();
private final boolean enableRemoteStreams; private final boolean enableRemoteStreams;
@ -137,6 +140,16 @@ public class SolrRequestParsers
parsers.put( "", standard ); parsers.put( "", standard );
} }
private static RTimer getRequestTimer(HttpServletRequest req)
{
final Object reqTimer = req.getAttribute(REQUEST_TIMER_SERVLET_ATTRIBUTE);
if (reqTimer != null && reqTimer instanceof RTimer) {
return ((RTimer) reqTimer);
}
return new RTimer();
}
public SolrQueryRequest parse( SolrCore core, String path, HttpServletRequest req ) throws Exception public SolrQueryRequest parse( SolrCore core, String path, HttpServletRequest req ) throws Exception
{ {
SolrRequestParser parser = standard; SolrRequestParser parser = standard;
@ -146,7 +159,7 @@ public class SolrRequestParsers
// Pick the parser from the request... // Pick the parser from the request...
ArrayList<ContentStream> streams = new ArrayList<>(1); ArrayList<ContentStream> streams = new ArrayList<>(1);
SolrParams params = parser.parseParamsAndFillStreams( req, streams ); SolrParams params = parser.parseParamsAndFillStreams( req, streams );
SolrQueryRequest sreq = buildRequestFrom( core, params, streams ); SolrQueryRequest sreq = buildRequestFrom( core, params, streams, getRequestTimer(req) );
// Handlers and login will want to know the path. If it contains a ':' // Handlers and login will want to know the path. If it contains a ':'
// the handler could use it for RESTful URLs // the handler could use it for RESTful URLs
@ -160,6 +173,11 @@ public class SolrRequestParsers
} }
public SolrQueryRequest buildRequestFrom( SolrCore core, SolrParams params, Collection<ContentStream> streams ) throws Exception public SolrQueryRequest buildRequestFrom( SolrCore core, SolrParams params, Collection<ContentStream> streams ) throws Exception
{
return buildRequestFrom( core, params, streams, new RTimer() );
}
private SolrQueryRequest buildRequestFrom( SolrCore core, SolrParams params, Collection<ContentStream> streams, RTimer requestTimer ) throws Exception
{ {
// The content type will be applied to all streaming content // The content type will be applied to all streaming content
String contentType = params.get( CommonParams.STREAM_CONTENTTYPE ); String contentType = params.get( CommonParams.STREAM_CONTENTTYPE );
@ -206,7 +224,7 @@ public class SolrRequestParsers
} }
} }
SolrQueryRequestBase q = new SolrQueryRequestBase( core, params ) { }; SolrQueryRequestBase q = new SolrQueryRequestBase( core, params, requestTimer ) { };
if( streams != null && streams.size() > 0 ) { if( streams != null && streams.size() > 0 ) {
q.setContentStreams( streams ); q.setContentStreams( streams );
} }

View File

@ -188,9 +188,9 @@ class LogUpdateProcessor extends UpdateRequestProcessor {
log.info(getLogStringAndClearRspToLog()); log.info(getLogStringAndClearRspToLog());
} }
if (log.isWarnEnabled()) { if (log.isWarnEnabled() && slowUpdateThresholdMillis >= 0) {
long elapsed = rsp.getEndTime() - req.getStartTime(); final long elapsed = (long) req.getRequestTimer().getTime();
if (slowUpdateThresholdMillis >= 0 && elapsed >= slowUpdateThresholdMillis) { if (elapsed >= slowUpdateThresholdMillis) {
log.warn("slow: " + getLogStringAndClearRspToLog()); log.warn("slow: " + getLogStringAndClearRspToLog());
} }
} }
@ -208,7 +208,7 @@ class LogUpdateProcessor extends UpdateRequestProcessor {
if (deletes != null && numDeletes > maxNumToLog) { if (deletes != null && numDeletes > maxNumToLog) {
deletes.add("... (" + numDeletes + " deletes)"); deletes.add("... (" + numDeletes + " deletes)");
} }
long elapsed = rsp.getEndTime() - req.getStartTime(); final long elapsed = (long) req.getRequestTimer().getTime();
sb.append(toLog).append(" 0 ").append(elapsed); sb.append(toLog).append(" 0 ").append(elapsed);
return sb.toString(); return sb.toString();

View File

@ -88,14 +88,15 @@ public class RTimer {
startTime = now(); startTime = now();
} }
/** Get total elapsed time for this timer. /** Get total elapsed time for this timer. */
*
* Timer must be STOPped.
*/
public double getTime() { public double getTime() {
assert state == STOPPED; if (state == STOPPED) return time;
return time; else if (state == PAUSED) return culmTime;
} else {
assert state == STARTED;
return culmTime + (now() - startTime);
}
}
/** Create new subtimer with given name /** Create new subtimer with given name
* *
@ -117,7 +118,7 @@ public class RTimer {
public NamedList asNamedList() { public NamedList asNamedList() {
NamedList<Object> m = new SimpleOrderedMap<>(); NamedList<Object> m = new SimpleOrderedMap<>();
m.add( "time", time ); m.add( "time", getTime() );
if( children.size() > 0 ) { if( children.size() > 0 ) {
for( Map.Entry<String, RTimer> entry : children ) { for( Map.Entry<String, RTimer> entry : children ) {
m.add( entry.getKey(), entry.getValue().asNamedList() ); m.add( entry.getKey(), entry.getValue().asNamedList() );

View File

@ -410,6 +410,7 @@ public class SolrRequestParserTest extends SolrTestCaseJ4 {
v.add(entry.getValue()); v.add(entry.getValue());
expect(request.getHeaders(entry.getKey())).andReturn(v.elements()).anyTimes(); expect(request.getHeaders(entry.getKey())).andReturn(v.elements()).anyTimes();
} }
expect(request.getAttribute(SolrRequestParsers.REQUEST_TIMER_SERVLET_ATTRIBUTE)).andReturn(null).anyTimes();
replay(request); replay(request);
SolrRequestParsers parsers = new SolrRequestParsers(h.getCore().getSolrConfig()); SolrRequestParsers parsers = new SolrRequestParsers(h.getCore().getSolrConfig());
@ -430,6 +431,7 @@ public class SolrRequestParserTest extends SolrTestCaseJ4 {
expect(request.getContentType()).andReturn(null).anyTimes(); expect(request.getContentType()).andReturn(null).anyTimes();
expect(request.getQueryString()).andReturn(null).anyTimes(); expect(request.getQueryString()).andReturn(null).anyTimes();
expect(request.getHeader(anyObject())).andReturn(null).anyTimes(); expect(request.getHeader(anyObject())).andReturn(null).anyTimes();
expect(request.getAttribute(SolrRequestParsers.REQUEST_TIMER_SERVLET_ATTRIBUTE)).andReturn(null).anyTimes();
replay(request); replay(request);
SolrRequestParsers parsers = new SolrRequestParsers(h.getCore().getSolrConfig()); SolrRequestParsers parsers = new SolrRequestParsers(h.getCore().getSolrConfig());

View File

@ -18,20 +18,28 @@
package org.apache.solr.util; package org.apache.solr.util;
import java.util.Arrays; import java.util.Arrays;
import java.util.ArrayList;
import java.util.HashMap; import java.util.HashMap;
import java.util.Iterator; import java.util.Iterator;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import org.apache.lucene.util.LuceneTestCase; import org.apache.solr.SolrTestCaseJ4;
import org.apache.solr.common.util.ExecutorUtil;
import org.apache.solr.common.util.SimpleOrderedMap; import org.apache.solr.common.util.SimpleOrderedMap;
import org.apache.solr.common.util.StrUtils; import org.apache.solr.common.util.StrUtils;
import org.junit.Assert; import org.junit.Assert;
import org.junit.Test;
/** /**
* *
*/ */
public class TestUtils extends LuceneTestCase { public class TestUtils extends SolrTestCaseJ4 {
public void testJoin() { public void testJoin() {
assertEquals("a|b|c", StrUtils.join(Arrays.asList("a","b","c"), '|')); assertEquals("a|b|c", StrUtils.join(Arrays.asList("a","b","c"), '|'));
@ -136,7 +144,7 @@ public class TestUtils extends LuceneTestCase {
try { try {
iter.remove(); iter.remove();
Assert.fail( "should be unsupported..." ); Assert.fail( "should be unsupported..." );
} catch( UnsupportedOperationException ex ) {} } catch( UnsupportedOperationException ignored) {}
} }
// the values should be bigger // the values should be bigger
assertEquals( new Integer(10), map.get( "one" ) ); assertEquals( new Integer(10), map.get( "one" ) );
@ -154,4 +162,43 @@ public class TestUtils extends LuceneTestCase {
assertEquals( num, NumberUtils.SortableStr2long(sortable, 0, sortable.length() ) ); assertEquals( num, NumberUtils.SortableStr2long(sortable, 0, sortable.length() ) );
assertEquals( Long.toString(num), NumberUtils.SortableStr2long(sortable) ); assertEquals( Long.toString(num), NumberUtils.SortableStr2long(sortable) );
} }
@Test
public void testNanoTimeSpeed()
{
final int maxNumThreads = 100;
final int numIters = 1000;
if (VERBOSE) log.info("testNanoTime: maxNumThreads = {}, numIters = {}", maxNumThreads, numIters);
final ExecutorService workers = Executors.newCachedThreadPool(new DefaultSolrThreadFactory("nanoTimeTestThread"));
for (int numThreads = 1; numThreads <= maxNumThreads; numThreads++) {
List<Callable<Long>> tasks = new ArrayList<> ();
for (int i = 0; i < numThreads; i ++) {
tasks.add(new Callable<Long>() {
@Override
public Long call() {
final long startTime = System.nanoTime();
for (int i = 0; i < numIters; i++) {
System.nanoTime();
}
return System.nanoTime() - startTime;
}
});
}
try {
List<Future<Long>> results = workers.invokeAll(tasks);
long totalTime = 0;
for (Future<Long> res : results) {
totalTime += res.get();
}
long timePerIter = totalTime / (numIters * numThreads);
assertTrue("Time taken for System.nanoTime is too high", timePerIter < 10000);
if (VERBOSE) log.info("numThreads = {}, time_per_call = {}ns", numThreads, timePerIter);
} catch (InterruptedException | ExecutionException ignored) {}
}
ExecutorUtil.shutdownAndAwaitTermination(workers);
}
} }