commit 4ac8773be3c5d2296fd3ae293427a33f42d9fe7d
Author: Karsten Loesing <karsten.loesing(a)gmx.net>
Date: Tue Aug 12 09:41:47 2014 +0200
Add performance metrics to the servlet.
Implements #12655.
---
.../onionoo/server/PerformanceMetrics.java | 198 +++++++++++++++++++
.../torproject/onionoo/server/ResourceServlet.java | 12 ++
.../torproject/onionoo/server/ResponseBuilder.java | 27 ++-
.../onionoo/server/PerformanceMetricsTest.java | 199 ++++++++++++++++++++
4 files changed, 430 insertions(+), 6 deletions(-)
diff --git a/src/main/java/org/torproject/onionoo/server/PerformanceMetrics.java b/src/main/java/org/torproject/onionoo/server/PerformanceMetrics.java
new file mode 100644
index 0000000..5f780d9
--- /dev/null
+++ b/src/main/java/org/torproject/onionoo/server/PerformanceMetrics.java
@@ -0,0 +1,198 @@
+/* Copyright 2014 The Tor Project
+ * See LICENSE for licensing information */
+package org.torproject.onionoo.server;
+
+import java.util.Arrays;
+import java.util.Collection;
+import java.util.Collections;
+import java.util.HashMap;
+import java.util.Map;
+import java.util.SortedMap;
+import java.util.SortedSet;
+import java.util.TreeMap;
+import java.util.TreeSet;
+
+import org.torproject.onionoo.util.ApplicationFactory;
+import org.torproject.onionoo.util.DateTimeHelper;
+import org.torproject.onionoo.util.Time;
+
+class Counter {
+ int value = 0;
+ void increment() {
+ this.value++;
+ }
+ public String toString() {
+ return String.valueOf(this.value);
+ }
+ void clear() {
+ this.value = 0;
+ }
+}
+
+class MostFrequentString {
+ Map<String, Integer> stringFrequencies = new HashMap<String, Integer>();
+ void addString(String string) {
+ if (!this.stringFrequencies.containsKey(string)) {
+ this.stringFrequencies.put(string, 1);
+ } else {
+ this.stringFrequencies.put(string,
+ this.stringFrequencies.get(string) + 1);
+ }
+ }
+ public String toString() {
+ SortedMap<Integer, SortedSet<String>> sortedFrequencies =
+ new TreeMap<Integer, SortedSet<String>>(
+ Collections.reverseOrder());
+ if (this.stringFrequencies.isEmpty()) {
+ return "null (0)";
+ }
+ for (Map.Entry<String, Integer> e : stringFrequencies.entrySet()) {
+ if (!sortedFrequencies.containsKey(e.getValue())) {
+ sortedFrequencies.put(e.getValue(), new TreeSet<String>(
+ Arrays.asList(new String[] { e.getKey() } )));
+ } else {
+ sortedFrequencies.get(e.getValue()).add(e.getKey());
+ }
+ }
+ StringBuilder sb = new StringBuilder();
+ int stringsToAdd = 3, written = 0;
+ for (Map.Entry<Integer, SortedSet<String>> e :
+ sortedFrequencies.entrySet()) {
+ for (String string : e.getValue()) {
+ if (stringsToAdd-- > 0) {
+ sb.append((written++ > 0 ? ", " : "") + string + " ("
+ + e.getKey() + ")");
+ }
+ }
+ if (stringsToAdd == 0) {
+ break;
+ }
+ }
+ return sb.toString();
+ }
+ void clear() {
+ this.stringFrequencies.clear();
+ }
+}
+
+class IntegerDistribution {
+ int[] logValues = new int[64];
+ void addLong(long value) {
+ logValues[64 - Long.numberOfLeadingZeros(value)]++;
+ }
+ public String toString() {
+ StringBuilder sb = new StringBuilder();
+ int totalValues = 0;
+ for (int i = 0; i < logValues.length; i++) {
+ totalValues += logValues[i];
+ }
+ int[] permilles = new int[] { 500, 900, 990, 999 };
+ if (totalValues > 0) {
+ int seenValues = 0;
+ for (int i = 0, j = 0; i < logValues.length; i++) {
+ seenValues += logValues[i];
+ while (j < permilles.length &&
+ (seenValues * 1000 > totalValues * permilles[j])) {
+ sb.append((j > 0 ? ", " : "") + "." + permilles[j]
+ + (i < logValues.length - 1 ? "<" + (1L << i)
+ : ">=" + (1L << i - 1)));
+ j++;
+ }
+ if (j == permilles.length) {
+ break;
+ }
+ }
+ } else {
+ for (int j = 0; j < permilles.length; j++) {
+ sb.append((j > 0 ? ", " : "") + "." + permilles[j] + "<null");
+ }
+ }
+ return sb.toString();
+ }
+ void clear() {
+ Arrays.fill(logValues, 0, logValues.length - 1, 0);
+ }
+}
+
+public class PerformanceMetrics {
+
+ private static final Object lock = new Object();
+
+ private static Time time;
+
+ private static long lastLoggedMillis = -1L;
+
+ private static final long LOG_INTERVAL = DateTimeHelper.ONE_HOUR;
+
+ private static Counter totalProcessedRequests = new Counter();
+
+ private static MostFrequentString
+ requestsByResourceType = new MostFrequentString(),
+ requestsByParameters = new MostFrequentString();
+
+ private static IntegerDistribution
+ matchingRelayDocuments = new IntegerDistribution(),
+ matchingBridgeDocuments = new IntegerDistribution(),
+ writtenChars = new IntegerDistribution(),
+ handleRequestMillis = new IntegerDistribution(),
+ buildResponseMillis = new IntegerDistribution();
+
+ public static void logStatistics(long receivedRequestMillis,
+ String resourceType, Collection<String> parameterKeys,
+ long parsedRequestMillis, int relayDocumentsWritten,
+ int bridgeDocumentsWritten, int charsWritten,
+ long writtenResponseMillis) {
+ synchronized (lock) {
+ if (time == null) {
+ time = ApplicationFactory.getTime();
+ }
+ if (lastLoggedMillis < 0L) {
+ lastLoggedMillis = time.currentTimeMillis();
+ } else if (receivedRequestMillis - lastLoggedMillis >
+ LOG_INTERVAL) {
+ System.err.println("Request statistics ("
+ + DateTimeHelper.format(lastLoggedMillis + LOG_INTERVAL)
+ + ", " + LOG_INTERVAL + " s):");
+ System.err.println(" Total processed requests: "
+ + totalProcessedRequests);
+ System.err.println(" Most frequently requested resource: "
+ + requestsByResourceType);
+ System.err.println(" Most frequently requested parameter "
+ + "combinations: " + requestsByParameters);
+ System.err.println(" Matching relays per request: "
+ + matchingRelayDocuments);
+ System.err.println(" Matching bridges per request: "
+ + matchingBridgeDocuments);
+ System.err.println(" Written characters per response: "
+ + writtenChars);
+ System.err.println(" Milliseconds to handle request: "
+ + handleRequestMillis);
+ System.err.println(" Milliseconds to build response: "
+ + buildResponseMillis);
+ totalProcessedRequests.clear();
+ requestsByResourceType.clear();
+ requestsByParameters.clear();
+ matchingRelayDocuments.clear();
+ matchingBridgeDocuments.clear();
+ writtenChars.clear();
+ handleRequestMillis.clear();
+ buildResponseMillis.clear();
+ do {
+ lastLoggedMillis += LOG_INTERVAL;
+ } while (receivedRequestMillis - lastLoggedMillis > LOG_INTERVAL);
+ }
+ totalProcessedRequests.increment();
+ handleRequestMillis.addLong(parsedRequestMillis
+ - receivedRequestMillis);
+ requestsByResourceType.addString(resourceType);
+ requestsByParameters.addString(parameterKeys.toString());
+ matchingRelayDocuments.addLong(relayDocumentsWritten);
+ matchingBridgeDocuments.addLong(bridgeDocumentsWritten);
+ writtenChars.addLong(charsWritten);
+ buildResponseMillis.addLong(writtenResponseMillis
+ - parsedRequestMillis);
+ }
+ }
+
+}
+
diff --git a/src/main/java/org/torproject/onionoo/server/ResourceServlet.java b/src/main/java/org/torproject/onionoo/server/ResourceServlet.java
index 6f01448..481462e 100644
--- a/src/main/java/org/torproject/onionoo/server/ResourceServlet.java
+++ b/src/main/java/org/torproject/onionoo/server/ResourceServlet.java
@@ -19,6 +19,7 @@ import javax.servlet.http.HttpServletResponse;
import org.torproject.onionoo.util.ApplicationFactory;
import org.torproject.onionoo.util.DateTimeHelper;
+import org.torproject.onionoo.util.Time;
public class ResourceServlet extends HttpServlet {
@@ -74,6 +75,9 @@ public class ResourceServlet extends HttpServlet {
return;
}
+ Time time = ApplicationFactory.getTime();
+ long receivedRequestMillis = time.currentTimeMillis();
+
String uri = request.getRequestURI();
if (uri.startsWith("/onionoo/")) {
uri = uri.substring("/onionoo".length());
@@ -279,6 +283,7 @@ public class ResourceServlet extends HttpServlet {
rh.setFamily(family);
}
rh.handleRequest();
+ long parsedRequestMillis = time.currentTimeMillis();
ResponseBuilder rb = new ResponseBuilder();
rb.setResourceType(resourceType);
@@ -301,8 +306,15 @@ public class ResourceServlet extends HttpServlet {
response.setCharacterEncoding("utf-8");
PrintWriter pw = response.getWriter();
rb.buildResponse(pw);
+ int relayDocumentsWritten = rh.getOrderedRelays().size();
+ int bridgeDocumentsWritten = rh.getOrderedBridges().size();
+ int charsWritten = rb.getCharsWritten();
pw.flush();
pw.close();
+ long writtenResponseMillis = time.currentTimeMillis();
+ PerformanceMetrics.logStatistics(receivedRequestMillis, resourceType,
+ parameterMap.keySet(), parsedRequestMillis, relayDocumentsWritten,
+ bridgeDocumentsWritten, charsWritten, writtenResponseMillis);
}
private static Pattern searchParameterPattern =
diff --git a/src/main/java/org/torproject/onionoo/server/ResponseBuilder.java b/src/main/java/org/torproject/onionoo/server/ResponseBuilder.java
index 161692c..458f7db 100644
--- a/src/main/java/org/torproject/onionoo/server/ResponseBuilder.java
+++ b/src/main/java/org/torproject/onionoo/server/ResponseBuilder.java
@@ -64,31 +64,46 @@ public class ResponseBuilder {
writeBridges(orderedBridges, pw);
}
+ private int charsWritten = 0;
+ public int getCharsWritten() {
+ return this.charsWritten;
+ }
+
private void writeRelays(List<SummaryDocument> relays, PrintWriter pw) {
- pw.write("{\"relays_published\":\"" + relaysPublishedString
- + "\",\n\"relays\":[");
+ String header = "{\"relays_published\":\"" + relaysPublishedString
+ + "\",\n\"relays\":[";
+ this.charsWritten += header.length();
+ pw.write(header);
int written = 0;
for (SummaryDocument entry : relays) {
String lines = this.formatNodeStatus(entry);
if (lines.length() > 0) {
+ this.charsWritten += (written > 0 ? 2 : 1) + lines.length();
pw.print((written++ > 0 ? ",\n" : "\n") + lines);
}
}
- pw.print("\n],\n");
+ String footer = "\n],\n";
+ this.charsWritten += footer.length();
+ pw.print(footer);
}
private void writeBridges(List<SummaryDocument> bridges,
PrintWriter pw) {
- pw.write("\"bridges_published\":\"" + bridgesPublishedString
- + "\",\n\"bridges\":[");
+ String header = "\"bridges_published\":\"" + bridgesPublishedString
+ + "\",\n\"bridges\":[";
+ this.charsWritten += header.length();
+ pw.write(header);
int written = 0;
for (SummaryDocument entry : bridges) {
String lines = this.formatNodeStatus(entry);
if (lines.length() > 0) {
+ this.charsWritten += (written > 0 ? 2 : 1) + lines.length();
pw.print((written++ > 0 ? ",\n" : "\n") + lines);
}
}
- pw.print("\n]}\n");
+ String footer = "\n]}\n";
+ this.charsWritten += footer.length();
+ pw.print(footer);
}
private String formatNodeStatus(SummaryDocument entry) {
diff --git a/src/test/java/org/torproject/onionoo/server/PerformanceMetricsTest.java b/src/test/java/org/torproject/onionoo/server/PerformanceMetricsTest.java
new file mode 100644
index 0000000..7a6614e
--- /dev/null
+++ b/src/test/java/org/torproject/onionoo/server/PerformanceMetricsTest.java
@@ -0,0 +1,199 @@
+/* Copyright 2014 The Tor Project
+ * See LICENSE for licensing information */
+package org.torproject.onionoo.server;
+
+import static org.junit.Assert.assertEquals;
+
+import org.junit.Test;
+
+public class PerformanceMetricsTest {
+
+ @Test
+ public void testCounterZero() {
+ Counter c = new Counter();
+ assertEquals("0", c.toString());
+ }
+
+ @Test
+ public void testCounterClear() {
+ Counter c = new Counter();
+ c.increment();
+ c.clear();
+ assertEquals("0", c.toString());
+ }
+
+ @Test
+ public void testCounterOne() {
+ Counter c = new Counter();
+ c.increment();
+ assertEquals("1", c.toString());
+ }
+
+ @Test
+ public void testCounterTwo() {
+ Counter c = new Counter();
+ c.increment();
+ c.increment();
+ assertEquals("2", c.toString());
+ }
+
+ @Test
+ public void testCounterOverflow() {
+ Counter c = new Counter();
+ c.value = Integer.MAX_VALUE;
+ c.increment();
+ assertEquals(String.valueOf(Integer.MIN_VALUE), c.toString());
+ }
+
+ @Test
+ public void testMostFrequentStringNothing() {
+ MostFrequentString mfs = new MostFrequentString();
+ assertEquals("null (0)", mfs.toString());
+ }
+
+ @Test
+ public void testMostFrequentStringClear() {
+ MostFrequentString mfs = new MostFrequentString();
+ mfs.addString("foo");
+ mfs.clear();
+ assertEquals("null (0)", mfs.toString());
+ }
+
+ @Test
+ public void testMostFrequentStringOneFoo() {
+ MostFrequentString mfs = new MostFrequentString();
+ mfs.addString("foo");
+ assertEquals("foo (1)", mfs.toString());
+ }
+
+ @Test
+ public void testMostFrequentStringTwoFoos() {
+ MostFrequentString mfs = new MostFrequentString();
+ mfs.addString("foo");
+ mfs.addString("foo");
+ assertEquals("foo (2)", mfs.toString());
+ }
+
+ @Test
+ public void testMostFrequentStringTwoFoosOneBar() {
+ MostFrequentString mfs = new MostFrequentString();
+ mfs.addString("foo");
+ mfs.addString("foo");
+ mfs.addString("bar");
+ assertEquals("foo (2), bar (1)", mfs.toString());
+ }
+
+ @Test
+ public void testMostFrequentStringABCD() {
+ MostFrequentString mfs = new MostFrequentString();
+ mfs.addString("A");
+ mfs.addString("B");
+ mfs.addString("C");
+ mfs.addString("D");
+ assertEquals("A (1), B (1), C (1)", mfs.toString());
+ }
+
+ @Test
+ public void testIntegerDistributionNothing() {
+ IntegerDistribution id = new IntegerDistribution();
+ assertEquals(".500<null, .900<null, .990<null, .999<null",
+ id.toString());
+ }
+
+ @Test
+ public void testIntegerDistributionClear() {
+ IntegerDistribution id = new IntegerDistribution();
+ id.addLong(1);
+ id.clear();
+ assertEquals(".500<null, .900<null, .990<null, .999<null",
+ id.toString());
+ }
+
+ @Test
+ public void testIntegerDistributionZero() {
+ IntegerDistribution id = new IntegerDistribution();
+ id.addLong(0);
+ assertEquals(".500<1, .900<1, .990<1, .999<1", id.toString());
+ }
+
+ @Test
+ public void testIntegerDistributionOne() {
+ IntegerDistribution id = new IntegerDistribution();
+ id.addLong(1);
+ assertEquals(".500<2, .900<2, .990<2, .999<2", id.toString());
+ }
+
+ @Test
+ public void testIntegerDistributionTwo() {
+ IntegerDistribution id = new IntegerDistribution();
+ id.addLong(2);
+ assertEquals(".500<4, .900<4, .990<4, .999<4", id.toString());
+ }
+
+ @Test
+ public void testIntegerDistributionThree() {
+ IntegerDistribution id = new IntegerDistribution();
+ id.addLong(3);
+ assertEquals(".500<4, .900<4, .990<4, .999<4", id.toString());
+ }
+
+ @Test
+ public void testIntegerDistributionFour() {
+ IntegerDistribution id = new IntegerDistribution();
+ id.addLong(4);
+ assertEquals(".500<8, .900<8, .990<8, .999<8", id.toString());
+ }
+
+ @Test
+ public void testIntegerDistributionFive() {
+ IntegerDistribution id = new IntegerDistribution();
+ id.addLong(5);
+ assertEquals(".500<8, .900<8, .990<8, .999<8", id.toString());
+ }
+
+ @Test
+ public void testIntegerDistributionFifteen() {
+ IntegerDistribution id = new IntegerDistribution();
+ id.addLong(15);
+ assertEquals(".500<16, .900<16, .990<16, .999<16", id.toString());
+ }
+
+ @Test
+ public void testIntegerDistributionSixteen() {
+ IntegerDistribution id = new IntegerDistribution();
+ id.addLong(16);
+ assertEquals(".500<32, .900<32, .990<32, .999<32", id.toString());
+ }
+
+ @Test
+ public void testIntegerDistributionSeventeen() {
+ IntegerDistribution id = new IntegerDistribution();
+ id.addLong(17);
+ assertEquals(".500<32, .900<32, .990<32, .999<32", id.toString());
+ }
+
+ @Test
+ public void testIntegerDistributionMaxLong() {
+ IntegerDistribution id = new IntegerDistribution();
+ id.addLong(Long.MAX_VALUE);
+ long val = Long.highestOneBit(Long.MAX_VALUE);
+ assertEquals(".500>=" + val + ", .900>=" + val + ", .990>=" + val
+ + ", .999>=" + val, id.toString());
+ }
+
+ @Test
+ public void testIntegerDistributionToThirtyTwo() {
+ IntegerDistribution id = new IntegerDistribution();
+ for (int i = 13; i <= 32; i++) id.addLong(i);
+ assertEquals(".500<32, .900<32, .990<64, .999<64", id.toString());
+ }
+
+ @Test
+ public void testIntegerDistributionToOneHundredTwentyEight() {
+ IntegerDistribution id = new IntegerDistribution();
+ for (int i = 27; i <= 128; i++) id.addLong(i);
+ assertEquals(".500<128, .900<128, .990<128, .999<256",
+ id.toString());
+ }
+}
+