APITimer: Non pull based Timer

The yammer Timer object calculate rate based on a timer, which causes
periodic calls to the API.

This replaces the implementation so that a timer would get all its
values from the API.

For object registration the APITimer still inherit from Timer but
override all its functionality.

Signed-off-by: Amnon Heiman <amnon@scylladb.com>
This commit is contained in:
Amnon Heiman 2016-05-17 11:12:01 +03:00
parent eca6451832
commit d3b8ef1ae5

View File

@ -4,41 +4,127 @@
*/ */
package com.yammer.metrics.core; package com.yammer.metrics.core;
import java.lang.reflect.Field;
import java.util.concurrent.ScheduledExecutorService; import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit; import java.util.concurrent.TimeUnit;
import javax.json.JsonObject;
import com.scylladb.jmx.api.APIClient;
import com.yammer.metrics.core.Histogram.SampleType; import com.yammer.metrics.core.Histogram.SampleType;
import com.yammer.metrics.stats.Snapshot;
/** /**
* A timer metric which aggregates timing durations and provides duration * A timer metric which aggregates timing durations and provides duration
* statistics, plus throughput statistics via {@link Meter}. * statistics, plus throughput statistics via {@link Meter}.
*/ */
public class APITimer extends Timer { public class APITimer extends Timer {
public final static long CACHE_DURATION = 1000;
final TimeUnit durationUnit, rateUnit;
final APIMeter meter;
final APIHistogram histogram;
APIClient c = new APIClient();
String url;
public APITimer(String url, ScheduledExecutorService tickThread, public APITimer(String url, ScheduledExecutorService tickThread,
TimeUnit durationUnit, TimeUnit rateUnit) { TimeUnit durationUnit, TimeUnit rateUnit) {
super(tickThread, durationUnit, rateUnit); super(tickThread, durationUnit, rateUnit);
setHistogram(url); super.stop();
this.url = url;
this.durationUnit = durationUnit;
this.rateUnit = rateUnit;
meter = new APIMeter(null, tickThread, "calls", rateUnit);
histogram = new APIHistogram(null, SampleType.BIASED);
} }
public APITimer(String url, ScheduledExecutorService tickThread, public void fromJson(JsonObject obj) {
TimeUnit durationUnit, TimeUnit rateUnit, Clock clock) { meter.fromJson(obj.getJsonObject("meter"));
super(tickThread, durationUnit, rateUnit, clock); histogram.updateValue(APIClient.json2histogram(obj.getJsonObject("hist")));
setHistogram(url);
} }
private void setHistogram(String url) { public void update_fields() {
Field histogram; if (url != null) {
try { fromJson(c.getJsonObj(url, null, CACHE_DURATION));
histogram = Timer.class.getDeclaredField("histogram");
histogram.setAccessible(true);
histogram.set(this, new APIHistogram(url, SampleType.BIASED));
} catch (NoSuchFieldException | SecurityException
| IllegalArgumentException | IllegalAccessException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} }
} }
@Override
public double max() {
update_fields();
return histogram.max();
}
@Override
public double min() {
update_fields();
return histogram.min();
}
@Override
public double mean() {
update_fields();
return histogram.mean();
}
@Override
public double stdDev() {
update_fields();
return histogram.stdDev();
}
@Override
public double sum() {
update_fields();
return 0;
}
@Override
public Snapshot getSnapshot() {
update_fields();
return histogram.getSnapshot();
}
@Override
public TimeUnit rateUnit() {
update_fields();
return meter.rateUnit();
}
@Override
public String eventType() {
update_fields();
return meter.eventType();
}
@Override
public long count() {
update_fields();
return meter.count();
}
@Override
public double fifteenMinuteRate() {
update_fields();
return meter.fifteenMinuteRate();
}
@Override
public double fiveMinuteRate() {
update_fields();
return meter.fiveMinuteRate();
}
@Override
public double meanRate() {
update_fields();
return meter.meanRate();
}
@Override
public double oneMinuteRate() {
update_fields();
return meter.oneMinuteRate();
}
} }