commit ee7c3eb73824a84e33b6c44b60fa8a32f37ab71e
Author: Karsten Loesing <karsten.loesing(a)gmx.net>
Date: Tue Dec 8 10:40:19 2020 +0100
Parse version 3 onion service statistics lines.
Implements the library part of tpo/metrics/statistics#40002.
---
CHANGELOG.md | 6 +-
.../torproject/descriptor/ExtraInfoDescriptor.java | 55 +++++++++++++
.../descriptor/impl/ExtraInfoDescriptorImpl.java | 93 ++++++++++++++++++++++
.../java/org/torproject/descriptor/impl/Key.java | 3 +
.../impl/ExtraInfoDescriptorImplTest.java | 46 +++++++++++
5 files changed, 202 insertions(+), 1 deletion(-)
diff --git a/CHANGELOG.md b/CHANGELOG.md
index c6886e8..8ff5723 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -1,4 +1,8 @@
-# Changes in version 2.??.? - 2020-??-??
+# Changes in version 2.15.0 - 2020-??-??
+
+ * Medium changes
+ - Parse version 3 onion service statistics contained in extra-info
+ descriptors.
# Changes in version 2.14.0 - 2020-08-07
diff --git a/src/main/java/org/torproject/descriptor/ExtraInfoDescriptor.java b/src/main/java/org/torproject/descriptor/ExtraInfoDescriptor.java
index e094fed..b553a22 100644
--- a/src/main/java/org/torproject/descriptor/ExtraInfoDescriptor.java
+++ b/src/main/java/org/torproject/descriptor/ExtraInfoDescriptor.java
@@ -674,6 +674,61 @@ public interface ExtraInfoDescriptor extends Descriptor {
*/
Map<String, Double> getHidservDirOnionsSeenParameters();
+ /**
+ * Return the time in milliseconds since the epoch when the included version 3
+ * onion service statistics interval ended, or -1 if no such statistics are
+ * included.
+ *
+ * @since 2.15.0
+ */
+ long getHidservV3StatsEndMillis();
+
+ /**
+ * Return the interval length of the included version 3 onion service
+ * statistics in seconds, or -1 if no such statistics are included.
+ *
+ * @since 2.15.0
+ */
+ long getHidservV3StatsIntervalLength();
+
+ /**
+ * Return the approximate number of RELAY cells seen in either direction on a
+ * version 3 onion service circuit after receiving and successfully processing
+ * a RENDEZVOUS1 cell, or null if no such statistics are included.
+ *
+ * @since 2.15.0
+ */
+ Double getHidservRendV3RelayedCells();
+
+ /**
+ * Return the obfuscation parameters applied to the original measurement value
+ * of RELAY cells seen in either direction on a version 3 onion service
+ * circuit after receiving and successfully processing a RENDEZVOUS1 cell, or
+ * null if no such statistics are included.
+ *
+ * @since 2.15.0
+ */
+ Map<String, Double> getHidservRendV3RelayedCellsParameters();
+
+ /**
+ * Return the approximate number of unique version 3 onion service identities
+ * seen in descriptors published to and accepted by this onion service
+ * directory, or null if no such statistics are included.
+ *
+ * @since 2.15.0
+ */
+ Double getHidservDirV3OnionsSeen();
+
+ /**
+ * Return the obfuscation parameters applied to the original measurement value
+ * of unique version 3 onion service identities seen in descriptors published
+ * to and accepted by this onion service directory, or null if no such
+ * statistics are included.
+ *
+ * @since 2.15.0
+ */
+ Map<String, Double> getHidservDirV3OnionsSeenParameters();
+
/**
* Return the time in milliseconds since the epoch when the included
* padding-counts statistics ended, or -1 if no such statistics are included.
diff --git a/src/main/java/org/torproject/descriptor/impl/ExtraInfoDescriptorImpl.java b/src/main/java/org/torproject/descriptor/impl/ExtraInfoDescriptorImpl.java
index 5cab6ab..4f87237 100644
--- a/src/main/java/org/torproject/descriptor/impl/ExtraInfoDescriptorImpl.java
+++ b/src/main/java/org/torproject/descriptor/impl/ExtraInfoDescriptorImpl.java
@@ -225,6 +225,15 @@ public abstract class ExtraInfoDescriptorImpl extends DescriptorImpl
case HIDSERV_DIR_ONIONS_SEEN:
this.parseHidservDirOnionsSeenLine(line, partsNoOpt);
break;
+ case HIDSERV_V3_STATS_END:
+ this.parseHidservV3StatsEndLine(line, partsNoOpt);
+ break;
+ case HIDSERV_REND_V3_RELAYED_CELLS:
+ this.parseHidservRendV3RelayedCellsLine(line, partsNoOpt);
+ break;
+ case HIDSERV_DIR_V3_ONIONS_SEEN:
+ this.parseHidservDirV3OnionsSeenLine(line, partsNoOpt);
+ break;
case PADDING_COUNTS:
this.parsePaddingCountsLine(line, partsNoOpt);
break;
@@ -764,6 +773,46 @@ public abstract class ExtraInfoDescriptorImpl extends DescriptorImpl
partsNoOpt, 2);
}
+ private void parseHidservV3StatsEndLine(String line,
+ String[] partsNoOpt) throws DescriptorParseException {
+ long[] parsedStatsEndData = this.parseStatsEndLine(line, partsNoOpt,
+ 5);
+ this.hidservV3StatsEndMillis = parsedStatsEndData[0];
+ this.hidservV3StatsIntervalLength = parsedStatsEndData[1];
+ }
+
+ private void parseHidservRendV3RelayedCellsLine(String line,
+ String[] partsNoOpt)
+ throws DescriptorParseException {
+ if (partsNoOpt.length < 2) {
+ throw new DescriptorParseException("Illegal line '" + line + "'.");
+ }
+ try {
+ this.hidservRendV3RelayedCells = Double.parseDouble(partsNoOpt[1]);
+ } catch (NumberFormatException e) {
+ throw new DescriptorParseException("Illegal line '" + line + "'.");
+ }
+ this.hidservRendV3RelayedCellsParameters =
+ ParseHelper.parseSpaceSeparatedStringKeyDoubleValueMap(line,
+ partsNoOpt, 2);
+ }
+
+ private void parseHidservDirV3OnionsSeenLine(String line,
+ String[] partsNoOpt)
+ throws DescriptorParseException {
+ if (partsNoOpt.length < 2) {
+ throw new DescriptorParseException("Illegal line '" + line + "'.");
+ }
+ try {
+ this.hidservDirV3OnionsSeen = Double.parseDouble(partsNoOpt[1]);
+ } catch (NumberFormatException e) {
+ throw new DescriptorParseException("Illegal line '" + line + "'.");
+ }
+ this.hidservDirV3OnionsSeenParameters =
+ ParseHelper.parseSpaceSeparatedStringKeyDoubleValueMap(line,
+ partsNoOpt, 2);
+ }
+
private void parsePaddingCountsLine(String line,
String[] partsNoOpt) throws DescriptorParseException {
long[] parsedStatsEndData = this.parseStatsEndLine(line, partsNoOpt,
@@ -1319,6 +1368,50 @@ public abstract class ExtraInfoDescriptorImpl extends DescriptorImpl
: new HashMap<>(this.hidservDirOnionsSeenParameters);
}
+ private long hidservV3StatsEndMillis = -1L;
+
+ @Override
+ public long getHidservV3StatsEndMillis() {
+ return this.hidservV3StatsEndMillis;
+ }
+
+ private long hidservV3StatsIntervalLength = -1L;
+
+ @Override
+ public long getHidservV3StatsIntervalLength() {
+ return this.hidservV3StatsIntervalLength;
+ }
+
+ private Double hidservRendV3RelayedCells;
+
+ @Override
+ public Double getHidservRendV3RelayedCells() {
+ return this.hidservRendV3RelayedCells;
+ }
+
+ private Map<String, Double> hidservRendV3RelayedCellsParameters;
+
+ @Override
+ public Map<String, Double> getHidservRendV3RelayedCellsParameters() {
+ return this.hidservRendV3RelayedCellsParameters == null ? null
+ : new HashMap<>(this.hidservRendV3RelayedCellsParameters);
+ }
+
+ private Double hidservDirV3OnionsSeen;
+
+ @Override
+ public Double getHidservDirV3OnionsSeen() {
+ return this.hidservDirV3OnionsSeen;
+ }
+
+ private Map<String, Double> hidservDirV3OnionsSeenParameters;
+
+ @Override
+ public Map<String, Double> getHidservDirV3OnionsSeenParameters() {
+ return this.hidservDirV3OnionsSeenParameters == null ? null
+ : new HashMap<>(this.hidservDirV3OnionsSeenParameters);
+ }
+
private long paddingCountsStatsEndMillis = -1L;
@Override
diff --git a/src/main/java/org/torproject/descriptor/impl/Key.java b/src/main/java/org/torproject/descriptor/impl/Key.java
index 70db350..b02d96e 100644
--- a/src/main/java/org/torproject/descriptor/impl/Key.java
+++ b/src/main/java/org/torproject/descriptor/impl/Key.java
@@ -90,8 +90,11 @@ public enum Key {
HIBERNATING("hibernating"),
HIDDEN_SERVICE_DIR("hidden-service-dir"),
HIDSERV_DIR_ONIONS_SEEN("hidserv-dir-onions-seen"),
+ HIDSERV_DIR_V3_ONIONS_SEEN("hidserv-dir-v3-onions-seen"),
HIDSERV_REND_RELAYED_CELLS("hidserv-rend-relayed-cells"),
+ HIDSERV_REND_V3_RELAYED_CELLS("hidserv-rend-v3-relayed-cells"),
HIDSERV_STATS_END("hidserv-stats-end"),
+ HIDSERV_V3_STATS_END("hidserv-v3-stats-end"),
ID("id"),
IDENTITY_ED25519("identity-ed25519"),
IPV6_CONN_BI_DIRECT("ipv6-conn-bi-direct"),
diff --git a/src/test/java/org/torproject/descriptor/impl/ExtraInfoDescriptorImplTest.java b/src/test/java/org/torproject/descriptor/impl/ExtraInfoDescriptorImplTest.java
index 1233bd5..08a5bc9 100644
--- a/src/test/java/org/torproject/descriptor/impl/ExtraInfoDescriptorImplTest.java
+++ b/src/test/java/org/torproject/descriptor/impl/ExtraInfoDescriptorImplTest.java
@@ -960,6 +960,16 @@ public class ExtraInfoDescriptorImplTest {
hsb.buildHidservStatsLines());
}
+ private String hidservV3StatsEndLine = "hidserv-v3-stats-end 2020-11-30 "
+ + "12:00:00 (86400 s)";
+
+ private String hidservRendV3RelayedCellsLine
+ = "hidserv-rend-v3-relayed-cells 6920802 delta_f=2048 epsilon=0.30 "
+ + "bin_size=1024";
+
+ private String hidservDirV3OnionsSeenLine = "hidserv-dir-v3-onions-seen 28 "
+ + "delta_f=8 epsilon=0.30 bin_size=8";
+
private static ExtraInfoDescriptor createWithDefaultLines()
throws DescriptorParseException {
return DescriptorBuilder.createWithHidservStatsLines(
@@ -977,6 +987,15 @@ public class ExtraInfoDescriptorImplTest {
if (this.hidservDirOnionsSeenLine != null) {
sb.append(this.hidservDirOnionsSeenLine).append("\n");
}
+ if (this.hidservV3StatsEndLine != null) {
+ sb.append(this.hidservV3StatsEndLine).append("\n");
+ }
+ if (this.hidservRendV3RelayedCellsLine != null) {
+ sb.append(this.hidservRendV3RelayedCellsLine).append("\n");
+ }
+ if (this.hidservDirV3OnionsSeenLine != null) {
+ sb.append(this.hidservDirV3OnionsSeenLine).append("\n");
+ }
String lines = sb.toString();
if (lines.endsWith("\n")) {
lines = lines.substring(0, lines.length() - 1);
@@ -2170,6 +2189,33 @@ public class ExtraInfoDescriptorImplTest {
assertTrue(params.isEmpty());
}
+ @Test
+ public void testHidservV3StatsValid() throws DescriptorParseException {
+ ExtraInfoDescriptor descriptor = HidservStatsBuilder
+ .createWithDefaultLines();
+ assertEquals(1606737600000L, descriptor.getHidservV3StatsEndMillis());
+ assertEquals(86400L, descriptor.getHidservV3StatsIntervalLength());
+ assertEquals(6920802.0, descriptor.getHidservRendV3RelayedCells(), 0.0001);
+ Map<String, Double> params =
+ descriptor.getHidservRendV3RelayedCellsParameters();
+ assertTrue(params.containsKey("delta_f"));
+ assertEquals(2048.0, params.remove("delta_f"), 0.0001);
+ assertTrue(params.containsKey("epsilon"));
+ assertEquals(0.3, params.remove("epsilon"), 0.0001);
+ assertTrue(params.containsKey("bin_size"));
+ assertEquals(1024.0, params.remove("bin_size"), 0.0001);
+ assertTrue(params.isEmpty());
+ assertEquals(28.0, descriptor.getHidservDirV3OnionsSeen(), 0.0001);
+ params = descriptor.getHidservDirV3OnionsSeenParameters();
+ assertTrue(params.containsKey("delta_f"));
+ assertEquals(8.0, params.remove("delta_f"), 0.0001);
+ assertTrue(params.containsKey("epsilon"));
+ assertEquals(0.3, params.remove("epsilon"), 0.0001);
+ assertTrue(params.containsKey("bin_size"));
+ assertEquals(8.0, params.remove("bin_size"), 0.0001);
+ assertTrue(params.isEmpty());
+ }
+
@Test
public void testHidservStatsEndLineMissing()
throws DescriptorParseException {