[tor-commits] [onionoo/master] Add version_status field to details documents.

karsten at torproject.org karsten at torproject.org
Wed Apr 4 19:22:06 UTC 2018


commit 76dec0651148e99d069ab798693b22a33162b6b8
Author: Karsten Loesing <karsten.loesing at gmx.net>
Date:   Thu Feb 22 20:59:31 2018 +0100

    Add version_status field to details documents.
    
    Implements #24256.
---
 CHANGELOG.md                                       |   3 +
 .../torproject/onionoo/docs/DetailsDocument.java   |  10 ++
 .../org/torproject/onionoo/docs/DetailsStatus.java |  10 ++
 .../org/torproject/onionoo/docs/NodeStatus.java    |  17 +++
 .../torproject/onionoo/server/ResponseBuilder.java |   2 +
 .../onionoo/updater/NodeDetailsStatusUpdater.java  |  23 +++-
 .../org/torproject/onionoo/updater/TorVersion.java | 150 +++++++++++++++++++++
 .../onionoo/updater/TorVersionStatus.java          |  47 +++++++
 .../onionoo/writer/DetailsDocumentWriter.java      |   1 +
 .../torproject/onionoo/updater/TorVersionTest.java | 127 +++++++++++++++++
 10 files changed, 384 insertions(+), 6 deletions(-)

diff --git a/CHANGELOG.md b/CHANGELOG.md
index 3a3c468..36d6471 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -1,5 +1,8 @@
 # Changes in version 5.1-1.12.0 - 2018-??-??
 
+ * Medium changes
+   - Add version_status field to details documents.
+
  * Minor changes
    - Don't attempt to un-escape character sequences in contact lines
      (like "\uk") that only happen to start like escaped utf-8 characters
diff --git a/src/main/java/org/torproject/onionoo/docs/DetailsDocument.java b/src/main/java/org/torproject/onionoo/docs/DetailsDocument.java
index c749fba..e112efe 100644
--- a/src/main/java/org/torproject/onionoo/docs/DetailsDocument.java
+++ b/src/main/java/org/torproject/onionoo/docs/DetailsDocument.java
@@ -356,6 +356,16 @@ public class DetailsDocument extends Document {
     return this.version;
   }
 
+  private String version_status;
+
+  public void setVersionStatus(String versionStatus) {
+    this.version_status = versionStatus;
+  }
+
+  public String getVersionStatus() {
+    return this.version_status;
+  }
+
   private SortedSet<String> alleged_family;
 
   public void setAllegedFamily(SortedSet<String> allegedFamily) {
diff --git a/src/main/java/org/torproject/onionoo/docs/DetailsStatus.java b/src/main/java/org/torproject/onionoo/docs/DetailsStatus.java
index c3cdc28..f838ec0 100644
--- a/src/main/java/org/torproject/onionoo/docs/DetailsStatus.java
+++ b/src/main/java/org/torproject/onionoo/docs/DetailsStatus.java
@@ -550,5 +550,15 @@ public class DetailsStatus extends Document {
   public String getVersion() {
     return this.version;
   }
+
+  private String version_status;
+
+  public void setVersionStatus(String versionStatus) {
+    this.version_status = versionStatus;
+  }
+
+  public String getVersionStatus() {
+    return this.version_status;
+  }
 }
 
diff --git a/src/main/java/org/torproject/onionoo/docs/NodeStatus.java b/src/main/java/org/torproject/onionoo/docs/NodeStatus.java
index 94da8c8..1ea8b98 100644
--- a/src/main/java/org/torproject/onionoo/docs/NodeStatus.java
+++ b/src/main/java/org/torproject/onionoo/docs/NodeStatus.java
@@ -3,6 +3,8 @@
 
 package org.torproject.onionoo.docs;
 
+import org.torproject.onionoo.updater.TorVersionStatus;
+
 import org.apache.commons.lang3.StringUtils;
 import org.slf4j.Logger;
 import org.slf4j.LoggerFactory;
@@ -330,6 +332,16 @@ public class NodeStatus extends Document {
     return this.version;
   }
 
+  private TorVersionStatus versionStatus;
+
+  public void setVersionStatus(TorVersionStatus versionStatus) {
+    this.versionStatus = versionStatus;
+  }
+
+  public TorVersionStatus getVersionStatus() {
+    return this.versionStatus;
+  }
+
   /* From exit lists: */
 
   private SortedSet<String> exitAddresses;
@@ -572,6 +584,9 @@ public class NodeStatus extends Document {
       if (parts.length >= 25 && !parts[24].isEmpty()) {
         nodeStatus.setHostName(parts[24]);
       }
+      if (parts.length >= 26) {
+        nodeStatus.setVersionStatus(TorVersionStatus.ofAbbreviation(parts[25]));
+      }
       return nodeStatus;
     } catch (NumberFormatException e) {
       log.error("Number format exception while parsing node "
@@ -640,6 +655,8 @@ public class NodeStatus extends Document {
         .append((this.getVersion() != null ? this.getVersion() : ""));
     sb.append("\t")
         .append((this.getHostName() != null ? this.getHostName() : ""));
+    sb.append("\t").append(null != this.getVersionStatus()
+        ? this.getVersionStatus().getAbbreviation() : "");
     return sb.toString();
   }
 }
diff --git a/src/main/java/org/torproject/onionoo/server/ResponseBuilder.java b/src/main/java/org/torproject/onionoo/server/ResponseBuilder.java
index e2bdf82..fcf3c8f 100644
--- a/src/main/java/org/torproject/onionoo/server/ResponseBuilder.java
+++ b/src/main/java/org/torproject/onionoo/server/ResponseBuilder.java
@@ -339,6 +339,8 @@ public class ResponseBuilder {
                 detailsDocument.getUnreachableOrAddresses());
           } else if (field.equals("version")) {
             dd.setVersion(detailsDocument.getVersion());
+          } else if (field.equals("version_status")) {
+            dd.setVersionStatus(detailsDocument.getVersionStatus());
           }
         }
         /* Don't escape HTML characters, like < and >, contained in
diff --git a/src/main/java/org/torproject/onionoo/updater/NodeDetailsStatusUpdater.java b/src/main/java/org/torproject/onionoo/updater/NodeDetailsStatusUpdater.java
index 4fdf98b..b8bd4f6 100644
--- a/src/main/java/org/torproject/onionoo/updater/NodeDetailsStatusUpdater.java
+++ b/src/main/java/org/torproject/onionoo/updater/NodeDetailsStatusUpdater.java
@@ -22,10 +22,8 @@ import org.slf4j.LoggerFactory;
 
 import java.util.Arrays;
 import java.util.HashMap;
-import java.util.HashSet;
 import java.util.List;
 import java.util.Map;
-import java.util.Set;
 import java.util.SortedMap;
 import java.util.SortedSet;
 import java.util.TreeMap;
@@ -89,7 +87,7 @@ public class NodeDetailsStatusUpdater implements DescriptorListener,
 
   private SortedMap<String, Integer> lastBandwidthWeights = null;
 
-  private Set<String> lastRecommendedServerVersions = null;
+  private SortedSet<TorVersion> lastRecommendedServerVersions = null;
 
   private int relayConsensusesProcessed = 0;
 
@@ -306,8 +304,15 @@ public class NodeDetailsStatusUpdater implements DescriptorListener,
     this.relayConsensusesProcessed++;
     if (this.relaysLastValidAfterMillis == validAfterMillis) {
       this.lastBandwidthWeights = consensus.getBandwidthWeights();
-      this.lastRecommendedServerVersions
-          = new HashSet<>(consensus.getRecommendedServerVersions());
+      this.lastRecommendedServerVersions = new TreeSet<>();
+      for (String recommendedServerVersion :
+          consensus.getRecommendedServerVersions()) {
+        TorVersion recommendedTorServerVersion
+            = TorVersion.of(recommendedServerVersion);
+        if (null != recommendedTorServerVersion) {
+          this.lastRecommendedServerVersions.add(recommendedTorServerVersion);
+        }
+      }
     }
   }
 
@@ -798,8 +803,13 @@ public class NodeDetailsStatusUpdater implements DescriptorListener,
        * the recommended_version field accordingly. */
       if (null != this.lastRecommendedServerVersions
           && null != nodeStatus.getVersion()) {
+        TorVersion torVersion = TorVersion.of(nodeStatus.getVersion());
         nodeStatus.setRecommendedVersion(this.lastRecommendedServerVersions
-            .contains(nodeStatus.getVersion()));
+            .contains(torVersion));
+        nodeStatus.setVersionStatus(null != torVersion
+            ? torVersion.determineVersionStatus(
+                this.lastRecommendedServerVersions)
+            : TorVersionStatus.UNRECOMMENDED);
       }
 
       Map<String, Long> exitAddresses = new HashMap<>();
@@ -909,6 +919,7 @@ public class NodeDetailsStatusUpdater implements DescriptorListener,
       detailsStatus.setLastChangedOrAddressOrPort(
           nodeStatus.getLastChangedOrAddressOrPort());
       detailsStatus.setVersion(nodeStatus.getVersion());
+      detailsStatus.setVersionStatus(nodeStatus.getVersionStatus().toString());
 
       this.documentStore.store(detailsStatus, fingerprint);
       this.documentStore.store(nodeStatus, fingerprint);
diff --git a/src/main/java/org/torproject/onionoo/updater/TorVersion.java b/src/main/java/org/torproject/onionoo/updater/TorVersion.java
new file mode 100644
index 0000000..d8e1683
--- /dev/null
+++ b/src/main/java/org/torproject/onionoo/updater/TorVersion.java
@@ -0,0 +1,150 @@
+/* Copyright 2018 The Tor Project
+ * See LICENSE for licensing information */
+
+package org.torproject.onionoo.updater;
+
+import java.util.HashMap;
+import java.util.Map;
+import java.util.SortedSet;
+
+/**
+ * Helper class to compare Tor versions.
+ *
+ * <p>Based on "How Tor Version Numbers Work", available at
+ * https://gitweb.torproject.org/torspec.git/tree/version-spec.txt</p>
+ */
+public class TorVersion implements Comparable<TorVersion> {
+
+  private int majorVersion;
+
+  private int minorVersion;
+
+  private int microVersion;
+
+  private String releaseSeries;
+
+  private Integer patchLevel = null;
+
+  private String statusTag = null;
+
+  private static Map<String, TorVersion> knownVersions = new HashMap<>();
+
+  private TorVersion() {
+  }
+
+  /** Return a TorVersion instance from the given tor version string that can be
+   * compared to other tor version strings, or null if the given string is not a
+   * valid tor version. */
+  public static TorVersion of(String versionString) {
+    if (null == versionString) {
+      return null;
+    }
+    if (!knownVersions.containsKey(versionString)) {
+      TorVersion result = new TorVersion();
+      String[] components = versionString.split("-")[0].split("\\.");
+      try {
+        result.majorVersion = Integer.parseInt(components[0]);
+        result.minorVersion = Integer.parseInt(components[1]);
+        result.microVersion = Integer.parseInt(components[2]);
+        result.releaseSeries = String.format("%d.%d.%d",
+            result.majorVersion, result.minorVersion, result.microVersion);
+        if (components.length == 4) {
+          result.patchLevel = Integer.parseInt(components[3]);
+          if (versionString.contains("-")) {
+            result.statusTag = versionString.split("-", 2)[1].split(" ")[0];
+          }
+        }
+      } catch (ArrayIndexOutOfBoundsException
+          | NumberFormatException exception) {
+        result = null;
+      }
+      knownVersions.put(versionString, result);
+    }
+    return knownVersions.get(versionString);
+  }
+
+  @Override
+  public int compareTo(TorVersion other) {
+    if (null == other) {
+      throw new NullPointerException();
+    }
+    int result;
+    if ((result = Integer.compare(this.majorVersion,
+        other.majorVersion)) != 0) {
+      return result;
+    }
+    if ((result = Integer.compare(this.minorVersion,
+        other.minorVersion)) != 0) {
+      return result;
+    }
+    if ((result = Integer.compare(this.microVersion,
+        other.microVersion)) != 0) {
+      return result;
+    }
+    if (null == this.patchLevel && null == other.patchLevel) {
+      return 0;
+    } else if (null == patchLevel) {
+      return -1;
+    } else if (null == other.patchLevel) {
+      return 1;
+    } else if ((result = Integer.compare(this.patchLevel,
+        other.patchLevel)) != 0) {
+      return result;
+    }
+    if (null == this.statusTag && null == other.statusTag) {
+      return 0;
+    } else if (null == this.statusTag) {
+      return -1;
+    } else if (null == other.statusTag) {
+      return 1;
+    } else {
+      return this.statusTag.compareTo(other.statusTag);
+    }
+  }
+
+  @Override
+  public boolean equals(Object other) {
+    return null != other && other instanceof TorVersion
+        && this.compareTo((TorVersion) other) == 0;
+  }
+
+  @Override
+  public int hashCode() {
+    return 2 * Integer.hashCode(this.majorVersion)
+        + 3 * Integer.hashCode(this.minorVersion)
+        + 5 * Integer.hashCode(this.microVersion)
+        + 7 * (null == this.patchLevel ? 0 : this.patchLevel)
+        + 11 * (null == this.statusTag ? 0 : this.statusTag.hashCode());
+  }
+
+  /** Determine the version status of this tor version in the context of the
+   * given recommended tor versions. */
+  public TorVersionStatus determineVersionStatus(
+      SortedSet<TorVersion> recommendedVersions) {
+    if (recommendedVersions.contains(this)) {
+      return TorVersionStatus.RECOMMENDED;
+    } else if (this.compareTo(recommendedVersions.last()) > 0) {
+      return TorVersionStatus.EXPERIMENTAL;
+    } else if (this.compareTo(recommendedVersions.first()) < 0) {
+      return TorVersionStatus.OBSOLETE;
+    } else {
+      boolean seriesHasRecommendedVersions = false;
+      boolean notNewInSeries = false;
+      for (TorVersion recommendedVersion : recommendedVersions) {
+        if (this.releaseSeries.equals(
+            recommendedVersion.releaseSeries)) {
+          seriesHasRecommendedVersions = true;
+          if (this.compareTo(recommendedVersion) < 0) {
+            notNewInSeries = true;
+          }
+        }
+      }
+      if (seriesHasRecommendedVersions && !notNewInSeries) {
+        return TorVersionStatus.NEW_IN_SERIES;
+      } else {
+        return TorVersionStatus.UNRECOMMENDED;
+      }
+    }
+  }
+}
+
diff --git a/src/main/java/org/torproject/onionoo/updater/TorVersionStatus.java b/src/main/java/org/torproject/onionoo/updater/TorVersionStatus.java
new file mode 100644
index 0000000..5825064
--- /dev/null
+++ b/src/main/java/org/torproject/onionoo/updater/TorVersionStatus.java
@@ -0,0 +1,47 @@
+/* Copyright 2018 The Tor Project
+ * See LICENSE for licensing information */
+
+package org.torproject.onionoo.updater;
+
+import java.util.HashMap;
+import java.util.Map;
+
+public enum TorVersionStatus {
+
+  RECOMMENDED("recommended", "r"),
+  EXPERIMENTAL("experimental", "e"),
+  OBSOLETE("obsolete", "o"),
+  NEW_IN_SERIES("new in series", "n"),
+  UNRECOMMENDED("unrecommended", "u");
+
+  private final String statusString;
+
+  private final String abbreviation;
+
+  TorVersionStatus(String statusString, String abbreviation) {
+    this.statusString = statusString;
+    this.abbreviation = abbreviation;
+  }
+
+  public String getAbbreviation() {
+    return this.abbreviation;
+  }
+
+  private static Map<String, TorVersionStatus> byAbbreviation = new HashMap<>();
+
+  static {
+    for (TorVersionStatus status : TorVersionStatus.values()) {
+      byAbbreviation.put(status.abbreviation, status);
+    }
+  }
+
+  public static TorVersionStatus ofAbbreviation(String abbrevation) {
+    return byAbbreviation.get(abbrevation);
+  }
+
+  @Override
+  public String toString() {
+    return this.statusString;
+  }
+}
+
diff --git a/src/main/java/org/torproject/onionoo/writer/DetailsDocumentWriter.java b/src/main/java/org/torproject/onionoo/writer/DetailsDocumentWriter.java
index 8490f05..eca7874 100644
--- a/src/main/java/org/torproject/onionoo/writer/DetailsDocumentWriter.java
+++ b/src/main/java/org/torproject/onionoo/writer/DetailsDocumentWriter.java
@@ -162,6 +162,7 @@ public class DetailsDocumentWriter implements DocumentWriter {
       detailsDocument.setUnreachableOrAddresses(unreachableOrAddresses);
     }
     detailsDocument.setVersion(detailsStatus.getVersion());
+    detailsDocument.setVersionStatus(detailsStatus.getVersionStatus());
     this.documentStore.store(detailsDocument, fingerprint);
   }
 
diff --git a/src/test/java/org/torproject/onionoo/updater/TorVersionTest.java b/src/test/java/org/torproject/onionoo/updater/TorVersionTest.java
new file mode 100644
index 0000000..b6a9764
--- /dev/null
+++ b/src/test/java/org/torproject/onionoo/updater/TorVersionTest.java
@@ -0,0 +1,127 @@
+/* Copyright 2018 The Tor Project
+ * See LICENSE for licensing information */
+
+package org.torproject.onionoo.updater;
+
+import static org.junit.Assert.assertEquals;
+import static org.junit.Assert.assertTrue;
+
+import org.junit.Test;
+import org.junit.experimental.runners.Enclosed;
+import org.junit.runner.RunWith;
+import org.junit.runners.Parameterized;
+import org.junit.runners.Parameterized.Parameter;
+import org.junit.runners.Parameterized.Parameters;
+
+import java.util.Arrays;
+import java.util.Collection;
+import java.util.SortedSet;
+import java.util.TreeSet;
+
+ at RunWith(Enclosed.class)
+public class TorVersionTest {
+
+  @RunWith(Parameterized.class)
+  public static class TorVersionStatusTest {
+
+    private static String[] recommendedVersionStrings = new String[] {
+        "0.2.5.16", "0.2.5.17", "0.2.9.14", "0.2.9.15", "0.3.1.9", "0.3.1.10",
+        "0.3.2.8-rc", "0.3.2.9", "0.3.3.1-alpha", "0.3.3.2-alpha" };
+
+    /** Provide test data. */
+    @Parameters
+    public static Collection<Object[]> data() {
+      return Arrays.asList(new Object[][] {
+          { "Recommended version", "0.2.5.16", "recommended" },
+          { "Recommended version", "0.3.2.8-rc", "recommended" },
+          { "Recommended version", "0.3.3.2-alpha", "recommended" },
+          { "Experimental version", "0.3.3.2-alpha-dev", "experimental" },
+          { "Experimental version", "0.3.3.3-alpha",  "experimental" },
+          { "Experimental version", "0.3.4.0-alpha-dev", "experimental" },
+          { "Obsolete version", "0.2.5.15", "obsolete" },
+          { "Obsolete version", "0.1.0.1-rc", "obsolete" },
+          { "New-in-series version", "0.2.5.18", "new in series" },
+          { "New-in-series version", "0.3.2.9-dev", "new in series" },
+          { "New-in-series version", "0.3.2.10", "new in series" },
+          { "Unrecommended version", "0.2.9.13", "unrecommended" },
+          { "Unrecommended version", "0.3.1.8-dev", "unrecommended" },
+          { "Unrecommended version", "0.3.3.0-alpha-dev", "unrecommended" },
+          { "Unrecognized (experimental) version", "1.0-final",
+              "unrecommended" },
+          { "Unrecognized (obsolete) version", "0.0.2pre13", "unrecommended" }
+      });
+    }
+
+    @Parameter
+    public String testDescription;
+
+    @Parameter(1)
+    public String versionString;
+
+    @Parameter(2)
+    public String expectedVersionStatus;
+
+    @Test
+    public void test() {
+      SortedSet<TorVersion> recommendedTorVersions = new TreeSet<>();
+      for (String recommendedVersionString : recommendedVersionStrings) {
+        recommendedTorVersions.add(TorVersion.of(recommendedVersionString));
+      }
+      TorVersion torVersion = TorVersion.of(this.versionString);
+      String determinedVersionStatus = "unrecommended";
+      if (null != torVersion) {
+        determinedVersionStatus = torVersion
+            .determineVersionStatus(recommendedTorVersions).toString();
+      }
+      assertEquals(this.testDescription, this.expectedVersionStatus,
+          determinedVersionStatus);
+    }
+  }
+
+  @RunWith(Parameterized.class)
+  public static class TorVersionEqualsHashCodeCompareToTest {
+
+    /** Provide test data. */
+    @Parameters
+    public static Collection<Object[]> data() {
+      return Arrays.asList(new Object[][] {
+          { "0.2.5.16", "0.2.5.16", true, true, 0 },
+          { "0.2.5.16", "0.2.5.17", false, false, -1 },
+          { "0.3.3.1-alpha", "0.3.3.1-alpha", true, true, 0 },
+          { "0.1.2.3", "00.01.02.03", true, true, 0 },
+          { "0.1.2.3-alpha", "00.01.02.03-aallpphhaa", false, false, 1 }
+      });
+    }
+
+    @Parameter
+    public String firstVersionString;
+
+    @Parameter(1)
+    public String secondVersionString;
+
+    @Parameter(2)
+    public boolean expectedEqualsResult;
+
+    @Parameter(3)
+    public boolean expectedSameHashCodes;
+
+    @Parameter(4)
+    public int expectedCompareToResult;
+
+    @Test
+    public void test() {
+      TorVersion firstVersion = TorVersion.of(this.firstVersionString);
+      TorVersion secondVersion = TorVersion.of(this.secondVersionString);
+      assertEquals(this.expectedEqualsResult,
+          firstVersion.equals(secondVersion));
+      if (this.expectedSameHashCodes) {
+        assertEquals(firstVersion.hashCode(), secondVersion.hashCode());
+      }
+      int actualCompareToResult = firstVersion.compareTo(secondVersion);
+      assertTrue(this.expectedCompareToResult < 0 && actualCompareToResult < 0
+          || this.expectedCompareToResult == 0 && actualCompareToResult == 0
+          || this.expectedCompareToResult > 0 && actualCompareToResult > 0);
+    }
+  }
+}
+



More information about the tor-commits mailing list