[Author Prev][Author Next][Thread Prev][Thread Next][Author Index][Thread Index]

[tor-commits] [metrics-web/master] Add new module to aggregate conn-bi-direct stats.



commit 2f9099c12b6d3b91c9e2609e1ce9bc137085aba9
Author: Karsten Loesing <karsten.loesing@xxxxxxx>
Date:   Wed Aug 19 19:28:52 2015 +0200

    Add new module to aggregate conn-bi-direct stats.
---
 modules/connbidirect/.gitignore                    |    3 +
 modules/connbidirect/build.xml                     |   68 +++
 .../org/torproject/metrics/connbidirect/Main.java  |  478 ++++++++++++++++++++
 .../torproject/metrics/connbidirect/MainTest.java  |  261 +++++++++++
 shared/bin/10-run-connbidirect-stats.sh            |    5 +
 shared/bin/99-copy-stats-files.sh                  |    1 +
 website/etc/web.xml                                |    1 +
 website/rserve/graphs.R                            |   31 +-
 .../org/torproject/metrics/web/IndexServlet.java   |    3 +
 .../metrics/web/graphs/GraphsSubpagesServlet.java  |    2 +
 .../metrics/web/research/ResearchStatsServlet.java |    1 +
 website/web/WEB-INF/connbidirect-data.jsp          |   10 +-
 website/web/WEB-INF/connbidirect.jsp               |    2 +-
 website/web/WEB-INF/connbidirect2-data.jsp         |   67 +++
 website/web/WEB-INF/performance.jsp                |    2 +-
 website/web/WEB-INF/stats.jsp                      |    2 +-
 16 files changed, 912 insertions(+), 25 deletions(-)

diff --git a/modules/connbidirect/.gitignore b/modules/connbidirect/.gitignore
new file mode 100644
index 0000000..b359b59
--- /dev/null
+++ b/modules/connbidirect/.gitignore
@@ -0,0 +1,3 @@
+classes/
+stats/
+
diff --git a/modules/connbidirect/build.xml b/modules/connbidirect/build.xml
new file mode 100644
index 0000000..29629e8
--- /dev/null
+++ b/modules/connbidirect/build.xml
@@ -0,0 +1,68 @@
+<project default="run" name="connbidirect" basedir=".">
+
+  <property name="connbidirect-sources" value="src/main/java"/>
+  <property name="connbidirect-tests" value="src/test/java"/>
+  <property name="connbidirect-classes" value="classes"/>
+  <path id="classpath">
+    <pathelement path="${connbidirect-classes}"/>
+    <fileset dir="/usr/share/java">
+      <include name="commons-codec-1.6.jar"/>
+      <include name="commons-compress-1.4.1.jar"/>
+      <include name="commons-lang-2.6.jar"/>
+      <include name="junit4.jar"/>
+    </fileset>
+    <fileset dir="../../deps/metrics-lib">
+      <include name="descriptor.jar"/>
+    </fileset>
+  </path>
+
+  <target name="metrics-lib">
+    <ant dir="../../deps/metrics-lib"/>
+  </target>
+
+  <target name="compile" depends="metrics-lib">
+    <mkdir dir="${connbidirect-classes}"/>
+    <javac destdir="${connbidirect-classes}"
+           srcdir="${connbidirect-sources}"
+           source="1.6"
+           target="1.6"
+           debug="true"
+           deprecation="true"
+           optimize="false"
+           failonerror="true"
+           includeantruntime="false">
+      <classpath refid="classpath"/>
+    </javac>
+  </target>
+
+  <target name="test" depends="compile">
+    <javac destdir="${connbidirect-classes}"
+           srcdir="${connbidirect-tests}"
+           source="1.6"
+           target="1.6"
+           debug="true"
+           deprecation="true"
+           optimize="false"
+           failonerror="true"
+           includeantruntime="false">
+      <classpath refid="classpath"/>
+    </javac>
+    <junit fork="true" haltonfailure="true" printsummary="off">
+      <classpath refid="classpath"/>
+      <formatter type="plain" usefile="false"/>
+      <batchtest>
+        <fileset dir="${connbidirect-classes}"
+                 includes="**/*Test.class"/>
+      </batchtest>
+    </junit>
+  </target>
+
+  <target name="run" depends="compile">
+    <java fork="true"
+          maxmemory="2g"
+          classname="org.torproject.metrics.connbidirect.Main">
+      <classpath refid="classpath"/>
+    </java>
+  </target>
+</project>
+
diff --git a/modules/connbidirect/src/main/java/org/torproject/metrics/connbidirect/Main.java b/modules/connbidirect/src/main/java/org/torproject/metrics/connbidirect/Main.java
new file mode 100644
index 0000000..190b3df
--- /dev/null
+++ b/modules/connbidirect/src/main/java/org/torproject/metrics/connbidirect/Main.java
@@ -0,0 +1,478 @@
+/* Copyright 2015 The Tor Project
+ * See LICENSE for licensing information */
+package org.torproject.metrics.connbidirect;
+
+import java.io.BufferedReader;
+import java.io.BufferedWriter;
+import java.io.File;
+import java.io.FileReader;
+import java.io.FileWriter;
+import java.io.IOException;
+import java.io.LineNumberReader;
+import java.io.StringReader;
+import java.text.DateFormat;
+import java.text.SimpleDateFormat;
+import java.util.ArrayList;
+import java.util.Collections;
+import java.util.Iterator;
+import java.util.List;
+import java.util.Map;
+import java.util.SortedMap;
+import java.util.SortedSet;
+import java.util.TimeZone;
+import java.util.TreeMap;
+import java.util.TreeSet;
+
+import org.torproject.descriptor.Descriptor;
+import org.torproject.descriptor.DescriptorFile;
+import org.torproject.descriptor.DescriptorReader;
+import org.torproject.descriptor.DescriptorSourceFactory;
+import org.torproject.descriptor.ExtraInfoDescriptor;
+
+public class Main {
+
+  static class RawStat implements Comparable<RawStat> {
+
+    /* Date when the statistics interval ended in days since the epoch. */
+    long dateDays;
+
+    /* Relay fingerprint, or <code>null</code> if this entry only
+     * indicates that outdated raw statistics have been discarded and
+     * hence new raw statistics for this date should not be aggregated. */
+    String fingerprint;
+
+    /* Fraction of mostly reading connections as a value between 0 and
+     * 100. */
+    short fractionRead;
+
+    /* Fraction of mostly writing connections as a value between 0 and
+     * 100. */
+    short fractionWrite;
+
+    /* Fraction of both reading and writing connections as a value between
+     * 0 and 100. */
+    short fractionBoth;
+
+    RawStat(long dateDays, String fingerprint, short fractionRead,
+        short fractionWrite, short fractionBoth) {
+      this.dateDays = dateDays;
+      this.fingerprint = fingerprint;
+      this.fractionRead = fractionRead;
+      this.fractionWrite = fractionWrite;
+      this.fractionBoth = fractionBoth;
+    }
+
+    static RawStat fromString(String string) {
+      try {
+        String[] parts = string.split(",");
+        if (parts.length == 5) {
+          long dateDays = Long.parseLong(parts[0]);
+          String fingerprint = parts[1];
+          short fractionRead = Short.parseShort(parts[2]);
+          short fractionWrite = Short.parseShort(parts[3]);
+          short fractionBoth = Short.parseShort(parts[4]);
+          return new RawStat(dateDays, fingerprint, fractionRead,
+              fractionWrite, fractionBoth);
+        } else {
+          System.err.println("Could not deserialize raw statistic from "
+              + "string '" + string + "'.");
+          return null;
+        }
+      } catch (NumberFormatException e) {
+        System.err.println("Could not deserialize raw statistic from "
+            + "string '" + string + "'.");
+        return null;
+      }
+    }
+
+    @Override
+    public String toString() {
+      if (this.fingerprint == null) {
+        return String.valueOf(this.dateDays);
+      } else {
+        return String.format("%d,%s,%d,%d,%d", this.dateDays,
+            this.fingerprint, this.fractionRead, this.fractionWrite,
+            this.fractionBoth);
+      }
+    }
+
+    @Override
+    public int compareTo(RawStat other) {
+      if (this.dateDays != other.dateDays) {
+        return this.dateDays < other.dateDays ? -1 : 1;
+      } else if (this.fingerprint != null && other.fingerprint != null) {
+        return this.fingerprint.compareTo(other.fingerprint);
+      } else if (this.fingerprint != null) {
+        return -1;
+      } else if (other.fingerprint != null) {
+        return 1;
+      } else {
+        return 0;
+      }
+    }
+
+    @Override
+    public boolean equals(Object otherObject) {
+      if (!(otherObject instanceof RawStat)) {
+        return false;
+      }
+      RawStat other = (RawStat) otherObject;
+      return this.dateDays == other.dateDays &&
+          this.fingerprint.equals(other.fingerprint);
+    }
+  }
+
+  static final long ONE_DAY_IN_MILLIS = 86400000L;
+
+  public static void main(String[] args) throws IOException {
+    File parseHistoryFile = new File("stats/parse-history");
+    File aggregateStatsFile = new File("stats/connbidirect2.csv");
+    File rawStatsFile = new File("stats/raw-stats");
+    File[] descriptorsDirectories = new File[] {
+        new File("../../shared/in/archive/relay-descriptors/extra-infos"),
+        new File("../../shared/in/recent/relay-descriptors/extra-infos")};
+    SortedMap<String, Long> parseHistory = parseParseHistory(
+        readStringFromFile(parseHistoryFile));
+    if (parseHistory == null) {
+      System.err.println("Could not parse "
+          + parseHistoryFile.getAbsolutePath() + ".  Proceeding without "
+          + "parse history.");
+    }
+    SortedMap<String, Short> aggregateStats = parseAggregateStats(
+        readStringFromFile(aggregateStatsFile));
+    if (aggregateStats == null) {
+      System.err.println("Could not parse previously aggregated "
+          + "statistics.  Not proceeding, because we would otherwise "
+          + "lose previously aggregated values for which we don't have "
+          + "raw statistics anymore.");
+      return;
+    }
+    SortedSet<RawStat> newRawStats = new TreeSet<RawStat>();
+    parseHistory = addRawStatsFromDescriptors(newRawStats,
+        descriptorsDirectories, parseHistory);
+    if (parseHistory == null) {
+      System.err.println("Could not parse raw statistics from "
+          + "descriptors.  Not proceeding, because we would otherwise "
+          + "leave out those descriptors in future runs.");
+      return;
+    }
+    SortedSet<RawStat> rawStats = parseRawStats(
+        readStringFromFile(rawStatsFile));
+    if (rawStats == null) {
+      System.err.println("Could not parse previously parsed raw "
+          + "statistics.  Not proceeding, because we might otherwise "
+          + "leave out previously parsed statistics in the aggregates.");
+      return;
+    }
+    SortedSet<Long> conflictingDates = mergeRawStats(rawStats,
+        newRawStats);
+    if (!conflictingDates.isEmpty()) {
+      System.err.print("Could not update aggregate statistics, because "
+          + "we already aggregated statistics for at least one contained "
+          + "date and discarded the underlying raw statistics.  Not "
+          + "proceeding.  To fix this, you'll have to re-import "
+          + "statistics for the following dates:");
+      DateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd");
+      dateFormat.setTimeZone(TimeZone.getTimeZone("UTC"));
+      for (long conflictingDate : conflictingDates) {
+        System.err.print(" " + dateFormat.format(conflictingDate
+            * ONE_DAY_IN_MILLIS));
+      }
+      System.err.println();
+      return;
+    }
+    updateAggregateStats(aggregateStats, rawStats);
+    writeStringToFile(aggregateStatsFile, formatAggregateStats(
+        aggregateStats));
+    writeStringToFile(rawStatsFile, formatRawStats(rawStats));
+    writeStringToFile(parseHistoryFile, formatParseHistory(parseHistory));
+  }
+
+  /* Read the string contained in <code>file</code> and return it. */
+  private static String readStringFromFile(File file) throws IOException {
+    StringBuilder sb = new StringBuilder();
+    if (file.exists()) {
+      BufferedReader br = new BufferedReader(new FileReader(file));
+      String line;
+      while ((line = br.readLine()) != null) {
+        sb.append(line + "\n");
+      }
+      br.close();
+    }
+    return sb.toString();
+  }
+
+  /* Write <code>string</code> to <code>file</code> by first creating its
+   * parent directory if that doesn't exist yet, then writing to a
+   * temporary file, and finally renaming that temporary file to the file
+   * to be written. */
+  private static void writeStringToFile(File file, String string)
+      throws IOException {
+    file.getParentFile().mkdirs();
+    File tempFile = new File(file.getParentFile(), file.getName()
+        + ".tmp");
+    BufferedWriter bw = new BufferedWriter(new FileWriter(tempFile));
+    bw.write(string);
+    bw.close();
+    tempFile.renameTo(file);
+  }
+
+  /* Format a parse history containing paths and last-modified times. */
+  static String formatParseHistory(SortedMap<String, Long> parseHistory) {
+    StringBuilder sb = new StringBuilder();
+    for (Map.Entry<String, Long> e : parseHistory.entrySet()) {
+      sb.append(e.getKey() + "," + e.getValue() + "\n");
+    }
+    return sb.toString();
+  }
+
+  /* Parse a parse history. */
+  static SortedMap<String, Long> parseParseHistory(
+      String formattedParseHistory) {
+    SortedMap<String, Long> parsedParseHistory =
+        new TreeMap<String, Long>();
+    LineNumberReader lnr = new LineNumberReader(new StringReader(
+        formattedParseHistory));
+    String line = "";
+    try {
+      while ((line = lnr.readLine()) != null) {
+        String[] parts = line.split(",");
+        if (parts.length < 2) {
+          System.err.println("Invalid line " + lnr.getLineNumber()
+              + " in parse history: '" + line + "'.");
+          return null;
+        }
+        parsedParseHistory.put(parts[0], Long.parseLong(parts[1]));
+      }
+    } catch (IOException e) {
+      System.err.println("Unexpected I/O exception while reading line "
+          + lnr.getLineNumber() + " from parse history.");
+      e.printStackTrace();
+      return null;
+    } catch (NumberFormatException e) {
+      System.err.println("Invalid line " + lnr.getLineNumber()
+          + " in parse history: '" + line + "'.");
+      return null;
+    }
+    return parsedParseHistory;
+  }
+
+  private static final String AGGREGATE_STATS_HEADER =
+      "date,direction,quantile,fraction";
+
+  /* Format aggregate connbidirect stats containing a combined identifier
+   * consisting of date (e.g., 2015-08-18), direction (both, read, write),
+   * and quantile (0.25, 0.5, 0.75) and a fraction value (between 0.0 and
+   * 1.0). */
+  static String formatAggregateStats(
+      SortedMap<String, Short> aggregateStats) {
+    StringBuilder sb = new StringBuilder();
+    sb.append(AGGREGATE_STATS_HEADER + "\n");
+    for (Map.Entry<String, Short> e : aggregateStats.entrySet()) {
+      sb.append(e.getKey() + "," + e.getValue() + "\n");
+    }
+    return sb.toString();
+  }
+
+  /* Parse aggregate connbidirect stats. */
+  static SortedMap<String, Short> parseAggregateStats(
+      String formattedAggregatedStats) {
+    SortedMap<String, Short> parsedAggregateStats =
+        new TreeMap<String, Short>();
+    if (formattedAggregatedStats.length() < 1) {
+      return parsedAggregateStats; /* Empty file. */
+    }
+    LineNumberReader lnr = new LineNumberReader(new StringReader(
+        formattedAggregatedStats));
+    String line = "";
+    try {
+      if (!AGGREGATE_STATS_HEADER.equals(lnr.readLine())) {
+        System.err.println("First line of aggregate statistics does not "
+            + "contain the header line.  Is this the correct file?");
+        return null;
+      }
+      while ((line = lnr.readLine()) != null) {
+        String[] parts = line.split(",");
+        if (parts.length != 4) {
+          System.err.println("Invalid line " + lnr.getLineNumber()
+              + " in aggregate statistics: '" + line + "'.");
+          return null;
+        }
+        parsedAggregateStats.put(parts[0] + "," + parts[1] + ","
+            + parts[2], Short.parseShort(parts[3]));
+      }
+    } catch (IOException e) {
+      System.err.println("Unexpected I/O exception while reading line "
+          + lnr.getLineNumber() + " from aggregate statistics.");
+      e.printStackTrace();
+      return null;
+    } catch (NumberFormatException e) {
+      System.err.println("Invalid line " + lnr.getLineNumber()
+          + " in aggregate statistics: '" + line + "'.");
+      return null;
+    }
+    return parsedAggregateStats;
+  }
+
+  /* Format raw statistics separated by newlines using the formatter in
+   * RawStats. */
+  static String formatRawStats(SortedSet<RawStat> rawStats) {
+    StringBuilder sb = new StringBuilder();
+    for (RawStat rawStat : rawStats) {
+      sb.append(rawStat.toString() + "\n");
+    }
+    return sb.toString();
+  }
+
+  /* Parse raw statistics. */
+  static SortedSet<RawStat> parseRawStats(String formattedRawStats) {
+    SortedSet<RawStat> parsedRawStats = new TreeSet<RawStat>();
+    LineNumberReader lnr = new LineNumberReader(new StringReader(
+        formattedRawStats));
+    String line = "";
+    try {
+      while ((line = lnr.readLine()) != null) {
+        RawStat rawStat = RawStat.fromString(line);
+        if (rawStat == null) {
+          System.err.println("Invalid line " + lnr.getLineNumber()
+              + " in raw statistics: '" + line + "'.");
+          return null;
+        }
+        parsedRawStats.add(rawStat);
+      }
+    } catch (IOException e) {
+      System.err.println("Unexpected I/O exception while reading line "
+          + lnr.getLineNumber() + " from raw statistics.");
+      e.printStackTrace();
+      return null;
+    } catch (NumberFormatException e) {
+      System.err.println("Invalid line " + lnr.getLineNumber()
+          + " in raw statistics: '" + line + "'.");
+      return null;
+    }
+    return parsedRawStats;
+  }
+
+  private static SortedMap<String, Long> addRawStatsFromDescriptors(
+      SortedSet<RawStat> rawStats, File[] descriptorsDirectories,
+      SortedMap<String, Long> parseHistory) {
+    DescriptorReader descriptorReader =
+        DescriptorSourceFactory.createDescriptorReader();
+    for (File descriptorsDirectory : descriptorsDirectories) {
+      descriptorReader.addDirectory(descriptorsDirectory);
+    }
+    descriptorReader.setExcludedFiles(parseHistory);
+    Iterator<DescriptorFile> descriptorFiles =
+        descriptorReader.readDescriptors();
+    while (descriptorFiles.hasNext()) {
+      DescriptorFile descriptorFile = descriptorFiles.next();
+      for (Descriptor descriptor : descriptorFile.getDescriptors()) {
+        if (!(descriptor instanceof ExtraInfoDescriptor)) {
+          continue;
+        }
+        RawStat rawStat = parseRawStatFromDescriptor(
+            (ExtraInfoDescriptor) descriptor);
+        if (rawStat != null) {
+          rawStats.add(rawStat);
+        }
+      }
+    }
+    parseHistory.clear();
+    parseHistory.putAll(descriptorReader.getExcludedFiles());
+    parseHistory.putAll(descriptorReader.getParsedFiles());
+    return parseHistory;
+  }
+
+  private static RawStat parseRawStatFromDescriptor(
+      ExtraInfoDescriptor extraInfo) {
+    if (extraInfo.getConnBiDirectStatsEndMillis() <= 0L) {
+      return null;
+    }
+    int below = extraInfo.getConnBiDirectBelow(),
+        read = extraInfo.getConnBiDirectRead(),
+        write = extraInfo.getConnBiDirectWrite(),
+        both = extraInfo.getConnBiDirectBoth();
+    if (below < 0 || read < 0 || write < 0 || both < 0) {
+      System.err.println("Could not parse incomplete conn-bi-direct "
+          + "statistics.  Skipping descriptor.");
+      return null;
+    }
+    long statsEndMillis = extraInfo.getConnBiDirectStatsEndMillis();
+    String fingerprint = extraInfo.getFingerprint();
+    return parseRawStatFromDescriptorContents(statsEndMillis, fingerprint,
+        below, read, write, both);
+  }
+
+  static RawStat parseRawStatFromDescriptorContents(long statsEndMillis,
+      String fingerprint, int below, int read, int write, int both) {
+    int total = read + write + both;
+    if (below < 0 || read < 0 || write < 0 || both < 0 || total <= 0) {
+      return null;
+    }
+    long dateDays = statsEndMillis / ONE_DAY_IN_MILLIS;
+    short fractionRead = (short) ((read * 100) / total);
+    short fractionWrite = (short) ((write * 100) / total);
+    short fractionBoth = (short) ((both * 100) / total);
+    return new RawStat(dateDays, fingerprint, fractionRead, fractionWrite,
+        fractionBoth);
+  }
+
+  static SortedSet<Long> mergeRawStats(
+      SortedSet<RawStat> rawStats, SortedSet<RawStat> newRawStats) {
+    rawStats.addAll(newRawStats);
+    SortedSet<Long> discardedRawStats = new TreeSet<Long>(),
+        availableRawStats = new TreeSet<Long>();
+    for (RawStat rawStat : rawStats) {
+      if (rawStat.fingerprint != null) {
+        availableRawStats.add(rawStat.dateDays);
+      } else {
+        discardedRawStats.add(rawStat.dateDays);
+      }
+    }
+    discardedRawStats.retainAll(availableRawStats);
+    return discardedRawStats;
+  }
+
+  static void updateAggregateStats(
+      SortedMap<String, Short> aggregateStats,
+      SortedSet<RawStat> rawStats) {
+    DateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd");
+    dateFormat.setTimeZone(TimeZone.getTimeZone("UTC"));
+    SortedMap<String, List<Short>> fractionsByDateAndDirection =
+        new TreeMap<String, List<Short>>();
+    final String[] directions = new String[] { "read", "write", "both" };
+    for (RawStat rawStat : rawStats) {
+      if (rawStat.fingerprint != null) {
+        String date = dateFormat.format(rawStat.dateDays
+            * ONE_DAY_IN_MILLIS);
+        short[] fractions = new short[] { rawStat.fractionRead,
+            rawStat.fractionWrite, rawStat.fractionBoth };
+        for (int i = 0; i < directions.length; i++) {
+          String dateAndDirection = date + "," + directions[i];
+          if (!fractionsByDateAndDirection.containsKey(
+              dateAndDirection)) {
+            fractionsByDateAndDirection.put(dateAndDirection,
+                new ArrayList<Short>());
+          }
+          fractionsByDateAndDirection.get(dateAndDirection).add(
+              fractions[i]);
+        }
+      }
+    }
+    final String[] quantiles = new String[] { "0.25", "0.5", "0.75" };
+    final int[] centiles = new int[] { 25, 50, 75 };
+    for (Map.Entry<String, List<Short>> e :
+        fractionsByDateAndDirection.entrySet()) {
+      String dateAndDirection = e.getKey();
+      List<Short> fractions = e.getValue();
+      Collections.sort(fractions);
+      for (int i = 0; i < quantiles.length; i++) {
+        String dateDirectionAndQuantile = dateAndDirection + ","
+            + quantiles[i];
+        short fraction = fractions.get((centiles[i] * fractions.size())
+            / 100);
+        aggregateStats.put(dateDirectionAndQuantile, fraction);
+      }
+    }
+  }
+}
diff --git a/modules/connbidirect/src/test/java/org/torproject/metrics/connbidirect/MainTest.java b/modules/connbidirect/src/test/java/org/torproject/metrics/connbidirect/MainTest.java
new file mode 100644
index 0000000..a490dd2
--- /dev/null
+++ b/modules/connbidirect/src/test/java/org/torproject/metrics/connbidirect/MainTest.java
@@ -0,0 +1,261 @@
+package org.torproject.metrics.connbidirect;
+
+import static org.junit.Assert.assertEquals;
+import static org.junit.Assert.assertNull;
+import static org.junit.Assert.assertSame;
+
+import java.io.ByteArrayOutputStream;
+import java.io.PrintStream;
+import java.util.SortedMap;
+import java.util.SortedSet;
+import java.util.TreeMap;
+import java.util.TreeSet;
+
+import org.junit.Test;
+
+public class MainTest {
+
+  private void assertParseHistoryCanBeSerializedAndDeserialized(
+      SortedMap<String, Long> parseHistory) {
+    String formattedParseHistory = Main.formatParseHistory(parseHistory);
+    SortedMap<String, Long> parsedParseHistory = Main.parseParseHistory(
+        formattedParseHistory);
+    assertEquals("Parse histories are not equal", parseHistory,
+        parsedParseHistory);
+  }
+
+  @Test
+  public void testParseHistoryEmpty() {
+    assertParseHistoryCanBeSerializedAndDeserialized(
+        new TreeMap<String, Long>());
+  }
+
+  private final String PATH_A = "a", PATH_B = "/b";
+
+  private final long LASTMOD_A = 1L, LASTMOD_B = 2L;
+
+  @Test
+  public void testParseHistoryOneEntry() {
+    SortedMap<String, Long> parseHistory = new TreeMap<String, Long>();
+    parseHistory.put(PATH_A, LASTMOD_A);
+    assertParseHistoryCanBeSerializedAndDeserialized(parseHistory);
+  }
+
+  @Test
+  public void testParseHistoryTwoEntries() {
+    SortedMap<String, Long> parseHistory = new TreeMap<String, Long>();
+    parseHistory.put(PATH_A, LASTMOD_A);
+    parseHistory.put(PATH_B, LASTMOD_B);
+    assertParseHistoryCanBeSerializedAndDeserialized(parseHistory);
+  }
+
+  private void assertParseHistoryCannotBeDeserialized(
+      String brokenParseHistory) {
+    final ByteArrayOutputStream baos = new ByteArrayOutputStream();
+    System.setErr(new PrintStream(baos));
+    SortedMap<String, Long> parsedParseHistory =
+        Main.parseParseHistory(brokenParseHistory);
+    assertNull("Parsed parse history is supposed to be null",
+        parsedParseHistory);
+  }
+
+  @Test
+  public void testParseHistoryNoLastModifiedTime() {
+    assertParseHistoryCannotBeDeserialized(String.format("%s%n", PATH_A));
+  }
+
+  @Test
+  public void testParseHistoryLastModifiedTimeNoNumber() {
+    assertParseHistoryCannotBeDeserialized(String.format("%s%s%n",
+        PATH_A, PATH_B));
+  }
+
+  private void assertAggregateStatsCanBeSerializedAndDeserialized(
+      SortedMap<String, Short> aggregateStats) {
+    String formattedAggregateStats = Main.formatAggregateStats(
+        aggregateStats);
+    SortedMap<String, Short> parsedParseHistory =
+        Main.parseAggregateStats(formattedAggregateStats);
+    assertEquals("Aggregate statistics are not equal", aggregateStats,
+        parsedParseHistory);
+  }
+
+  @Test
+  public void testAggregateStatsEmpty() {
+    assertAggregateStatsCanBeSerializedAndDeserialized(
+        new TreeMap<String, Short>());
+  }
+
+  @Test
+  public void testAggregateStatsOneEntry() {
+    SortedMap<String, Short> aggregateStats =
+        new TreeMap<String, Short>();
+    aggregateStats.put("2015-08-18,read,0.25", (short) 42);
+    assertAggregateStatsCanBeSerializedAndDeserialized(aggregateStats);
+  }
+
+  @Test
+  public void testAggregateStatsThreeEntries() {
+    SortedMap<String, Short> aggregateStats =
+        new TreeMap<String, Short>();
+    aggregateStats.put("2015-08-18,read,0.25", (short) 12);
+    aggregateStats.put("2015-08-18,read,0.5", (short) 24);
+    aggregateStats.put("2015-08-18,read,0.75", (short) 42);
+    assertAggregateStatsCanBeSerializedAndDeserialized(aggregateStats);
+  }
+
+  private void assertRawStatsCanBeSerializedAndDeserialized(
+      SortedSet<Main.RawStat> rawStats) {
+    String formattedRawStats = Main.formatRawStats(rawStats);
+    SortedSet<Main.RawStat> parsedRawStats = Main.parseRawStats(
+        formattedRawStats);
+    assertEquals("Raw statistics are not equal", rawStats,
+        parsedRawStats);
+  }
+
+  @Test
+  public void testRawStatsEmpty() {
+    assertRawStatsCanBeSerializedAndDeserialized(
+        new TreeSet<Main.RawStat>());
+  }
+
+  private static final long DATE_A = 16665, /* 2015-08-18 */
+      DATE_B = 16680; /* 2015-09-02 */
+
+  private static final String
+      FPR_A = "1234567890123456789012345678901234567890",
+      FPR_B = "2345678901234567890123456789012345678901";
+
+  @Test
+  public void testRawStatsOneEntry() {
+    SortedSet<Main.RawStat> rawStats = new TreeSet<Main.RawStat>();
+    rawStats.add(new Main.RawStat(DATE_A, FPR_A, (short) 40, (short) 30,
+        (short) 50));
+    assertRawStatsCanBeSerializedAndDeserialized(rawStats);
+  }
+
+  private void assertRawStatsCanBeMerged(SortedSet<Main.RawStat> rawStats,
+      SortedSet<Main.RawStat> newRawStats, boolean expectConflicts) {
+    SortedSet<Long> conflictingDays = Main.mergeRawStats(rawStats,
+        newRawStats);
+    assertSame("Expected merge conflicts differ from observed conflicts",
+        expectConflicts, !conflictingDays.isEmpty());
+  }
+
+  @Test
+  public void testMergeRawStatsAddNothing() {
+    SortedSet<Main.RawStat> rawStats = new TreeSet<Main.RawStat>();
+    rawStats.add(new Main.RawStat(DATE_A, FPR_A, (short) 40, (short) 30,
+        (short) 50));
+    assertRawStatsCanBeMerged(rawStats, new TreeSet<Main.RawStat>(),
+        false);
+  }
+
+  @Test
+  public void testMergeRawStatsAddSame() {
+    SortedSet<Main.RawStat> rawStats = new TreeSet<Main.RawStat>();
+    rawStats.add(new Main.RawStat(DATE_A, FPR_A, (short) 40, (short) 30,
+        (short) 50));
+    SortedSet<Main.RawStat> newRawStats = new TreeSet<Main.RawStat>();
+    newRawStats.add(new Main.RawStat(DATE_A, FPR_A, (short) 40,
+        (short) 30, (short) 50));
+    assertRawStatsCanBeMerged(rawStats, newRawStats, false);
+  }
+
+  @Test
+  public void testMergeRawStatsAddOther() {
+    SortedSet<Main.RawStat> rawStats = new TreeSet<Main.RawStat>();
+    rawStats.add(new Main.RawStat(DATE_A, FPR_A, (short) 40, (short) 30,
+        (short) 50));
+    SortedSet<Main.RawStat> newRawStats = new TreeSet<Main.RawStat>();
+    newRawStats.add(new Main.RawStat(DATE_B, FPR_B, (short) 40,
+        (short) 30, (short) 50));
+    assertRawStatsCanBeMerged(rawStats, newRawStats, false);
+  }
+
+  @Test
+  public void testParseRawStatAllNegative() {
+    Main.RawStat rawStat = Main.parseRawStatFromDescriptorContents(DATE_A,
+        FPR_A, -1, -1, -1, -1);
+    assertNull(rawStat);
+  }
+
+  @Test
+  public void testParseRawStatOneNegative() {
+    Main.RawStat rawStat = Main.parseRawStatFromDescriptorContents(DATE_A,
+        FPR_A, -1, 1, 1, 1);
+    assertNull(rawStat);
+  }
+
+  @Test
+  public void testParseRawStatTotalZero() {
+    Main.RawStat rawStat = Main.parseRawStatFromDescriptorContents(DATE_A,
+        FPR_A, 0, 0, 0, 0);
+    assertNull(rawStat);
+  }
+
+  @Test
+  public void testParseRawStatOneOfEach() {
+    Main.RawStat rawStat = Main.parseRawStatFromDescriptorContents(DATE_A,
+        FPR_A, 1, 1, 1, 2);
+    assertSame("Read fraction", (short) 25, rawStat.fractionRead);
+    assertSame("Write fraction", (short) 25, rawStat.fractionWrite);
+    assertSame("Both fraction", (short) 50, rawStat.fractionBoth);
+  }
+
+  private void assertStatsCanBeAggregated(
+      SortedMap<String, Short> expectedAggregateStats,
+      SortedSet<Main.RawStat> rawStats) {
+    SortedMap<String, Short> updatedAggregateStats =
+        new TreeMap<String, Short>();
+    Main.updateAggregateStats(updatedAggregateStats, rawStats);
+    assertEquals("Updated aggregate statistics don't match",
+        expectedAggregateStats, updatedAggregateStats);
+  }
+
+  @Test
+  public void testUpdateAggregateStatsEmpty() {
+    assertStatsCanBeAggregated(new TreeMap<String, Short>(),
+        new TreeSet<Main.RawStat>());
+  }
+
+  @Test
+  public void testUpdateAggregateStatsSingleRawStat() {
+    SortedMap<String, Short> expectedAggregateStats =
+        new TreeMap<String, Short>();
+    expectedAggregateStats.put("2015-08-18,read,0.25", (short) 42);
+    expectedAggregateStats.put("2015-08-18,read,0.5", (short) 42);
+    expectedAggregateStats.put("2015-08-18,read,0.75", (short) 42);
+    expectedAggregateStats.put("2015-08-18,write,0.25", (short) 32);
+    expectedAggregateStats.put("2015-08-18,write,0.5", (short) 32);
+    expectedAggregateStats.put("2015-08-18,write,0.75", (short) 32);
+    expectedAggregateStats.put("2015-08-18,both,0.25", (short) 22);
+    expectedAggregateStats.put("2015-08-18,both,0.5", (short) 22);
+    expectedAggregateStats.put("2015-08-18,both,0.75", (short) 22);
+    SortedSet<Main.RawStat> rawStats = new TreeSet<Main.RawStat>();
+    rawStats.add(new Main.RawStat(DATE_A, FPR_A, (short) 42, (short) 32,
+        (short) 22));
+    assertStatsCanBeAggregated(expectedAggregateStats, rawStats);
+  }
+
+  @Test
+  public void testUpdateAggregateStatsTwoRawStat() {
+    SortedMap<String, Short> expectedAggregateStats =
+        new TreeMap<String, Short>();
+    expectedAggregateStats.put("2015-08-18,read,0.25", (short) 32);
+    expectedAggregateStats.put("2015-08-18,read,0.5", (short) 42);
+    expectedAggregateStats.put("2015-08-18,read,0.75", (short) 42);
+    expectedAggregateStats.put("2015-08-18,write,0.25", (short) 22);
+    expectedAggregateStats.put("2015-08-18,write,0.5", (short) 32);
+    expectedAggregateStats.put("2015-08-18,write,0.75", (short) 32);
+    expectedAggregateStats.put("2015-08-18,both,0.25", (short) 12);
+    expectedAggregateStats.put("2015-08-18,both,0.5", (short) 22);
+    expectedAggregateStats.put("2015-08-18,both,0.75", (short) 22);
+    SortedSet<Main.RawStat> rawStats = new TreeSet<Main.RawStat>();
+    rawStats.add(new Main.RawStat(DATE_A, FPR_A, (short) 32, (short) 22,
+        (short) 12));
+    rawStats.add(new Main.RawStat(DATE_A, FPR_B, (short) 42, (short) 32,
+        (short) 22));
+    assertStatsCanBeAggregated(expectedAggregateStats, rawStats);
+  }
+}
diff --git a/shared/bin/10-run-connbidirect-stats.sh b/shared/bin/10-run-connbidirect-stats.sh
new file mode 100755
index 0000000..d59c315
--- /dev/null
+++ b/shared/bin/10-run-connbidirect-stats.sh
@@ -0,0 +1,5 @@
+#!/bin/sh
+cd modules/connbidirect/
+ant | grep "\[java\]"
+cd ../../
+
diff --git a/shared/bin/99-copy-stats-files.sh b/shared/bin/99-copy-stats-files.sh
index 4a30f24..22018ba 100755
--- a/shared/bin/99-copy-stats-files.sh
+++ b/shared/bin/99-copy-stats-files.sh
@@ -1,6 +1,7 @@
 #!/bin/sh
 mkdir -p shared/stats
 cp -a modules/legacy/stats/*.csv shared/stats/
+cp -a modules/connbidirect/stats/connbidirect2.csv shared/stats/
 cp -a modules/advbwdist/stats/advbwdist.csv shared/stats/
 cp -a modules/hidserv/stats/hidserv.csv shared/stats/
 cp -a modules/clients/stats/clients.csv shared/stats/
diff --git a/website/etc/web.xml b/website/etc/web.xml
index e32f26b..87dec8f 100644
--- a/website/etc/web.xml
+++ b/website/etc/web.xml
@@ -64,6 +64,7 @@
     <url-pattern>/clients-data.html</url-pattern>
     <url-pattern>/torperf-data.html</url-pattern>
     <url-pattern>/connbidirect-data.html</url-pattern>
+    <url-pattern>/connbidirect2-data.html</url-pattern>
     <url-pattern>/hidserv-data.html</url-pattern>
     <url-pattern>/hidserv-dir-onions-seen.html</url-pattern>
     <url-pattern>/hidserv-rend-relayed-cells.html</url-pattern>
diff --git a/website/rserve/graphs.R b/website/rserve/graphs.R
index 0e8e0a3..c35f1ec 100644
--- a/website/rserve/graphs.R
+++ b/website/rserve/graphs.R
@@ -674,30 +674,21 @@ plot_torperf_failures <- function(start, end, source, filesize, path) {
 plot_connbidirect <- function(start, end, path) {
   end <- min(end, as.character(Sys.Date() - 2))
   c <- read.csv(paste("/srv/metrics.torproject.org/metrics/shared/stats/",
-                "connbidirect.csv", sep = ""), stringsAsFactors = FALSE)
-  c <- c[c$date >= start & c$date <= end &
-         c$read + c$write + c$both > 0, ]
-  c <- data.frame(date = as.Date(c$date, "%Y-%m-%d"),
-                  both = c$both / (c$read + c$write + c$both),
-                  read = c$read / (c$read + c$write + c$both),
-                  write = c$write / (c$read + c$write + c$both))
-  c <- aggregate(list(both = c$both, read = c$read, write = c$write),
-                 by = list(date = c$date), quantile,
-                 probs = c(0.25, 0.5, 0.75))
-  c <- rbind(
-    data.frame(date = as.Date(c$date), data.frame(c$both),
-               variable = "both"),
-    data.frame(date = as.Date(c$date), data.frame(c$write),
-               variable = "write"),
-    data.frame(date = as.Date(c$date), data.frame(c$read),
-               variable = "read"))
+                "connbidirect2.csv", sep = ""), stringsAsFactors = FALSE)
+  c <- c[c$date >= start & c$date <= end, ]
+  c <- data.frame(date = as.Date(c$date),
+                  direction = factor(c$direction,
+                              levels = c("both", "write", "read")),
+                  quantile = paste("X", c$quantile, sep = ""),
+                  fraction = c$fraction / 100)
+  c <- cast(c, date + direction ~ quantile, value = "fraction")
   date_breaks <- date_breaks(
     as.numeric(max(as.Date(c$date, "%Y-%m-%d")) -
     min(as.Date(c$date, "%Y-%m-%d"))))
-  ggplot(c, aes(x = date, y = X50., colour = variable)) +
+  ggplot(c, aes(x = date, y = X0.5, colour = direction)) +
     geom_line(size = 0.75) +
-    geom_ribbon(aes(x = date, ymin = X25., ymax = X75., fill = variable),
-                alpha = 0.5, legend = FALSE) +
+    geom_ribbon(aes(x = date, ymin = X0.25, ymax = X0.75,
+                fill = direction), alpha = 0.5, legend = FALSE) +
     scale_x_date(name = paste("\nThe Tor Project - ",
         "https://metrics.torproject.org/";, sep = ""),
         format = date_breaks$format, major = date_breaks$major,
diff --git a/website/src/org/torproject/metrics/web/IndexServlet.java b/website/src/org/torproject/metrics/web/IndexServlet.java
index 63b0da6..47cb4cd 100644
--- a/website/src/org/torproject/metrics/web/IndexServlet.java
+++ b/website/src/org/torproject/metrics/web/IndexServlet.java
@@ -149,6 +149,9 @@ public class IndexServlet extends HttpServlet {
         "Performance of downloading static files over Tor",
         new String[] { "Performance" }, "Data", "Advanced"));
     availableMetrics.add(new Metric("connbidirect-data.html",
+        "Fraction of connections used uni-/bidirectionally (deprecated)",
+        new String[] { "Performance" }, "Data", "Advanced"));
+    availableMetrics.add(new Metric("connbidirect2-data.html",
         "Fraction of connections used uni-/bidirectionally",
         new String[] { "Performance" }, "Data", "Advanced"));
     availableMetrics.add(new Metric("hidserv-dir-onions-seen.html",
diff --git a/website/src/org/torproject/metrics/web/graphs/GraphsSubpagesServlet.java b/website/src/org/torproject/metrics/web/graphs/GraphsSubpagesServlet.java
index 0192836..f484d8f 100644
--- a/website/src/org/torproject/metrics/web/graphs/GraphsSubpagesServlet.java
+++ b/website/src/org/torproject/metrics/web/graphs/GraphsSubpagesServlet.java
@@ -90,6 +90,8 @@ public class GraphsSubpagesServlet extends HttpServlet {
         "WEB-INF/torperf-data.jsp");
     this.availableGraphsSubpages.put("connbidirect-data.html",
         "WEB-INF/connbidirect-data.jsp");
+    this.availableGraphsSubpages.put("connbidirect2-data.html",
+        "WEB-INF/connbidirect2-data.jsp");
     this.availableGraphsSubpages.put("hidserv-data.html",
         "WEB-INF/hidserv-data.jsp");
     this.availableGraphsSubpages.put("hidserv-dir-onions-seen.html",
diff --git a/website/src/org/torproject/metrics/web/research/ResearchStatsServlet.java b/website/src/org/torproject/metrics/web/research/ResearchStatsServlet.java
index ea30c60..618d614 100644
--- a/website/src/org/torproject/metrics/web/research/ResearchStatsServlet.java
+++ b/website/src/org/torproject/metrics/web/research/ResearchStatsServlet.java
@@ -33,6 +33,7 @@ public class ResearchStatsServlet extends HttpServlet {
     this.availableStatisticsFiles.add("clients");
     this.availableStatisticsFiles.add("torperf");
     this.availableStatisticsFiles.add("connbidirect");
+    this.availableStatisticsFiles.add("connbidirect2");
     this.availableStatisticsFiles.add("advbwdist");
     this.availableStatisticsFiles.add("hidserv");
   }
diff --git a/website/web/WEB-INF/connbidirect-data.jsp b/website/web/WEB-INF/connbidirect-data.jsp
index 51c78a7..8b7d9ed 100644
--- a/website/web/WEB-INF/connbidirect-data.jsp
+++ b/website/web/WEB-INF/connbidirect-data.jsp
@@ -3,7 +3,7 @@
 <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN">
 <html>
 <head>
-  <title>Tor Metrics &mdash; Fraction of connections used uni-/bidirectionally</title>
+  <title>Tor Metrics &mdash; Fraction of connections used uni-/bidirectionally (deprecated)</title>
   <meta http-equiv="content-type" content="text/html; charset=ISO-8859-1">
   <link href="/css/stylesheet-ltr.css" type="text/css" rel="stylesheet">
   <link href="/images/favicon.ico" type="image/x-icon" rel="shortcut icon">
@@ -13,8 +13,14 @@
     <%@ include file="banner.jsp"%>
     <div class="main-column">
 
-<h2><a href="/">Tor Metrics</a> &mdash; Fraction of connections used uni-/bidirectionally</h2>
+<h2><a href="/">Tor Metrics</a> &mdash; Fraction of connections used uni-/bidirectionally (deprecated)</h2>
 <br>
+<p><font color="red">As of August 25, 2015, this page and the linked data
+file have been replaced by <a href="connbidirect2-data.html">this page and
+the data file linked from there</a>.
+This page and the linked data file will be removed in the
+future.</font></p>
+
 <p>The following data file contains statistics on the fraction of direct
 connections between a <a href="about.html#relay">relay</a> and other nodes
 in the network that are used uni- or bidirectionally.
diff --git a/website/web/WEB-INF/connbidirect.jsp b/website/web/WEB-INF/connbidirect.jsp
index affa329..6ed512b 100644
--- a/website/web/WEB-INF/connbidirect.jsp
+++ b/website/web/WEB-INF/connbidirect.jsp
@@ -55,7 +55,7 @@ fractions.</p>
 
 <h4>Related metrics</h4>
 <ul>
-<li><a href="connbidirect-data.html">Data: Fraction of connections used uni-/bidirectionally</a></li>
+<li><a href="connbidirect2-data.html">Data: Fraction of connections used uni-/bidirectionally</a></li>
 </ul>
 
     </div>
diff --git a/website/web/WEB-INF/connbidirect2-data.jsp b/website/web/WEB-INF/connbidirect2-data.jsp
new file mode 100644
index 0000000..f29373a
--- /dev/null
+++ b/website/web/WEB-INF/connbidirect2-data.jsp
@@ -0,0 +1,67 @@
+<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core"; %>
+<%@ taglib prefix="fn" uri="http://java.sun.com/jsp/jstl/functions"; %>
+<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN">
+<html>
+<head>
+  <title>Tor Metrics &mdash; Fraction of connections used uni-/bidirectionally</title>
+  <meta http-equiv="content-type" content="text/html; charset=ISO-8859-1">
+  <link href="/css/stylesheet-ltr.css" type="text/css" rel="stylesheet">
+  <link href="/images/favicon.ico" type="image/x-icon" rel="shortcut icon">
+</head>
+<body>
+  <div class="center">
+    <%@ include file="banner.jsp"%>
+    <div class="main-column">
+
+<h2><a href="/">Tor Metrics</a> &mdash; Fraction of connections used uni-/bidirectionally</h2>
+<br>
+<p>The following data file contains statistics on the fraction of direct
+connections between a <a href="about.html#relay">relay</a> and other nodes
+in the network that are used uni- or bidirectionally.
+Every 10 seconds, relays determine for every direct connection whether
+they read and wrote less than a threshold of 20 KiB.
+Connections below this threshold are excluded from the statistics file.
+For the remaining connections, relays determine whether they read/wrote at
+least 10 times as many bytes as they wrote/read.
+If so, they classify a connection as "mostly reading" or "mostly writing",
+respectively.
+All other connections are classified as "both reading and writing".
+After classifying connections, read and write counters are reset for the
+next 10-second interval.
+The data file contains daily medians and quartiles of reported
+fractions.</p>
+
+<p><b>Download as <a href="stats/connbidirect2.csv">CSV file</a>.</b></p>
+
+<p>The statistics file contains the following columns:</p>
+<ul>
+<li><b>date:</b> UTC date (YYYY-MM-DD) for which statistics on
+uni-/bidirectional connection usage were reported.</li>
+<li><b>direction:</b> Direction of reported fraction, which can be
+<b>"read"</b>, <b>"write"</b>, or <b>"both"</b> for connections classified
+as "mostly reading", "mostly writing", or "both reading as writing".
+Connections below the threshold have been removed from this statistics
+file entirely.</li>
+<li><b>quantile:</b> Quantile of the reported fraction when considering
+all statistics reported for this date.
+Examples are <b>"0.5"</b> for the median and <b>"0.25"</b> and
+<b>"0.75"</b> for the lower and upper quartile.</li>
+<li><b>fraction:</b> Fraction of connections in percent for the given
+date, direction, and quantile.
+For each daily statistic reported by a relay, fractions for the three
+directions "read", "write", and "both" sum up to exactly 100.</li>
+</ul>
+
+<h4>Related metrics</h4>
+<ul>
+<li><a href="connbidirect.html">Graph: Fraction of connections used uni-/bidirectionally</a></li>
+</ul>
+
+    </div>
+  </div>
+  <div class="bottom" id="bottom">
+    <%@ include file="footer.jsp"%>
+  </div>
+</body>
+</html>
+
diff --git a/website/web/WEB-INF/performance.jsp b/website/web/WEB-INF/performance.jsp
index 04231d3..37e05a1 100644
--- a/website/web/WEB-INF/performance.jsp
+++ b/website/web/WEB-INF/performance.jsp
@@ -25,7 +25,7 @@ Sorry for any inconvenience caused by this.</p>
 <li><a href="torperf-failures.html">Graph: Timeouts and failures of downloading files over Tor</a></li>
 <li><a href="connbidirect.html">Graph: Fraction of connections used uni-/bidirectionally</a></li>
 <li><a href="torperf-data.html">Data: Performance of downloading static files over Tor</a></li>
-<li><a href="connbidirect-data.html">Data: Fraction of connections used uni-/bidirectionally</a></li>
+<li><a href="connbidirect2-data.html">Data: Fraction of connections used uni-/bidirectionally</a></li>
 </ul>
 
     </div>
diff --git a/website/web/WEB-INF/stats.jsp b/website/web/WEB-INF/stats.jsp
index 6a807ac..ef8319b 100644
--- a/website/web/WEB-INF/stats.jsp
+++ b/website/web/WEB-INF/stats.jsp
@@ -24,7 +24,7 @@ Sorry for any inconvenience caused by this.</p>
 <li><a href="advbwdist-data.html">Data: Advertised bandwidth distribution and n-th fastest relays</a></li>
 <li><a href="clients-data.html">Data: Estimated number of clients in the Tor network</a></li>
 <li><a href="torperf-data.html">Data: Performance of downloading static files over Tor</a></li>
-<li><a href="connbidirect-data.html">Data: Fraction of connections used uni-/bidirectionally</a></li>
+<li><a href="connbidirect2-data.html">Data: Fraction of connections used uni-/bidirectionally</a></li>
 </ul>
 
     </div>

_______________________________________________
tor-commits mailing list
tor-commits@xxxxxxxxxxxxxxxxxxxx
https://lists.torproject.org/cgi-bin/mailman/listinfo/tor-commits