diff --git a/apps/routerconsole/java/src/net/i2p/router/web/NetDbHelper.java b/apps/routerconsole/java/src/net/i2p/router/web/NetDbHelper.java
index 2de33338130792f307134e2c48bfbb586f04f987..338eccfaa1f0c28389b1762b5caab3f1ee1329cc 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/NetDbHelper.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/NetDbHelper.java
@@ -7,7 +7,7 @@ public class NetDbHelper extends HelperBase {
     private String _routerPrefix;
     private String _version;
     private String _country;
-    private String _family, _caps, _ip;
+    private String _family, _caps, _ip, _sybil;
     private int _full;
     private boolean _lease;
     private boolean _debug;
@@ -68,6 +68,12 @@ public class NetDbHelper extends HelperBase {
             _ip = DataHelper.stripHTML(c);  // XSS
     }
 
+    /** @since 0.9.28 */
+    public void setSybil(String c) {
+        if (c != null)
+            _sybil = DataHelper.stripHTML(c);  // XSS
+    }
+
     public void setFull(String f) {
         try {
             _full = Integer.parseInt(f);
@@ -95,8 +101,9 @@ public class NetDbHelper extends HelperBase {
         try {
             renderNavBar();
             if (_routerPrefix != null || _version != null || _country != null ||
-                _family != null || _caps != null || _ip != null)
-                renderer.renderRouterInfoHTML(_out, _routerPrefix, _version, _country, _family, _caps, _ip);
+                _family != null || _caps != null || _ip != null || _sybil != null)
+                renderer.renderRouterInfoHTML(_out, _routerPrefix, _version, _country,
+                                              _family, _caps, _ip, _sybil);
             else if (_lease)
                 renderer.renderLeaseSetHTML(_out, _debug);
             else if (_full == 3)
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/NetDbRenderer.java b/apps/routerconsole/java/src/net/i2p/router/web/NetDbRenderer.java
index 97acfe9c7b8147b66ef0f1d86b2f403cc8ba4954..68f6064b4199311970a3935e1e6a847c9ce75006 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/NetDbRenderer.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/NetDbRenderer.java
@@ -88,8 +88,10 @@ class NetDbRenderer {
      *  @param family may be null
      */
     public void renderRouterInfoHTML(Writer out, String routerPrefix, String version,
-                                     String country, String family, String caps, String ip) throws IOException {
+                                     String country, String family, String caps,
+                                     String ip, String sybil) throws IOException {
         StringBuilder buf = new StringBuilder(4*1024);
+        List<Hash> sybils = sybil != null ? new ArrayList<Hash>(128) : null;
         if (".".equals(routerPrefix)) {
             renderRouterInfo(buf, _context.router().getRouterInfo(), true, true);
         } else {
@@ -116,14 +118,18 @@ class NetDbRenderer {
                     (version != null && version.equals(ri.getVersion())) ||
                     (country != null && country.equals(_context.commSystem().getCountry(key))) ||
                     (family != null && family.equals(ri.getOption("family"))) ||
-                    (caps != null && caps.equals(ri.getCapabilities()))) {
+                    (caps != null && ri.getCapabilities().contains(caps))) {
                     renderRouterInfo(buf, ri, false, true);
+                    if (sybil != null)
+                        sybils.add(key);
                     notFound = false;
                 } else if (ip != null) {
                     for (RouterAddress ra : ri.getAddresses()) {
                         if (ipMode == 0) {
                             if (ip.equals(ra.getHost())) {
                                 renderRouterInfo(buf, ri, false, true);
+                                if (sybil != null)
+                                    sybils.add(key);
                                 notFound = false;
                                 break;
                             }
@@ -131,6 +137,8 @@ class NetDbRenderer {
                             String host = ra.getHost();
                             if (host != null && host.startsWith(ip)) {
                                 renderRouterInfo(buf, ri, false, true);
+                                if (sybil != null)
+                                    sybils.add(key);
                                 notFound = false;
                                 break;
                             }
@@ -153,6 +161,8 @@ class NetDbRenderer {
         }
         out.write(buf.toString());
         out.flush();
+        if (sybil != null)
+            SybilRenderer.renderSybilHTML(out, _context, sybils, sybil);
     }
 
     /**
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/SybilRenderer.java b/apps/routerconsole/java/src/net/i2p/router/web/SybilRenderer.java
index 73d59432f739901c98c3721b8c5b9b56f0b52ad9..8293a4b07eb453fd6fe6244a6bbfe38a0b3fd807 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/SybilRenderer.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/SybilRenderer.java
@@ -16,6 +16,7 @@ import java.util.Locale;
 import java.util.Map;
 import java.util.Set;
 
+import net.i2p.data.Base64;
 import net.i2p.data.DataHelper;
 import net.i2p.data.Destination;
 import net.i2p.data.Hash;
@@ -23,6 +24,7 @@ import net.i2p.data.LeaseSet;
 import net.i2p.data.router.RouterAddress;
 import net.i2p.data.router.RouterInfo;
 import net.i2p.data.router.RouterKeyGenerator;
+import net.i2p.kademlia.XORComparator;
 import net.i2p.router.RouterContext;
 import net.i2p.router.TunnelPoolSettings;
 import net.i2p.router.crypto.FamilyKeyCrypto;
@@ -450,7 +452,7 @@ class SybilRenderer {
             int i3 = i & 0xff;
             String sip = i0 + "." + i1 + '.' + i2 + '.' + i3;
             buf.append("<p><b>").append(count).append(" floodfills with IP <a href=\"/netdb?ip=")
-               .append(sip).append("\">").append(sip)
+               .append(sip).append("&amp;sybil\">").append(sip)
                .append("</a>:</b></p>");
             for (RouterInfo info : ris) {
                 byte[] ip = getIP(info);
@@ -503,7 +505,7 @@ class SybilRenderer {
             int i2 = i & 0xff;
             String sip = i0 + "." + i1 + '.' + i2 + ".0/24";
             buf.append("<p><b>").append(count).append(" floodfills with IP <a href=\"/netdb?ip=")
-               .append(sip).append("\">").append(sip)
+               .append(sip).append("&amp;sybil\">").append(sip)
                .append("</a>:</b></p>");
             for (RouterInfo info : ris) {
                 byte[] ip = getIP(info);
@@ -553,7 +555,7 @@ class SybilRenderer {
             int i1 = i & 0xff;
             String sip = i0 + "." + i1 + ".0/16";
             buf.append("<p><b>").append(count).append(" floodfills with IP <a href=\"/netdb?ip=")
-               .append(sip).append("\">").append(sip)
+               .append(sip).append("&amp;sybil\">").append(sip)
                .append("</a></b></p>");
             for (RouterInfo info : ris) {
                 byte[] ip = getIP(info);
@@ -595,7 +597,7 @@ class SybilRenderer {
             int count = oc.count(s);
             String ss = DataHelper.escapeHTML(s);
             buf.append("<p><b>").append(count).append(" floodfills in declared family \"<a href=\"/netdb?fam=")
-               .append(ss).append("\">").append(ss).append("</a>\"</b></p>");
+               .append(ss).append("&amp;sybil\">").append(ss).append("</a>\"</b></p>");
             for (RouterInfo info : ris) {
                 String fam = info.getOption("family");
                 if (fam == null)
@@ -901,6 +903,77 @@ class SybilRenderer {
         return distance;
     }
 
+    /**
+     *  Called from NetDbRenderer
+     *
+     *  @since 0.9.28
+     */
+    public static void renderSybilHTML(Writer out, RouterContext ctx, List<Hash> sybils, String victim) throws IOException {
+        if (sybils.isEmpty())
+            return;
+        final DecimalFormat fmt = new DecimalFormat("#0.00");
+        XORComparator<Hash> xor = new XORComparator<Hash>(Hash.FAKE_HASH);
+        out.write("<h3>Group Distances</h3><table><tr><th>Hash<th>Distance from previous</tr>\n");
+        Collections.sort(sybils, xor);
+        Hash prev = null;
+        for (Hash h : sybils) {
+            out.write("<tr><td><tt>" + h.toBase64() + "</tt><td>");
+            if (prev != null) {
+                BigInteger dist = HashDistance.getDistance(prev, h);
+                writeDistance(out, fmt, dist);
+            }
+            prev = h;
+            out.write("</tr>\n");
+        }
+        out.write("</table>\n");
+        out.flush();
+
+        RouterKeyGenerator rkgen = ctx.routerKeyGenerator();
+        long now = ctx.clock().now();
+        final int days = 7;
+        Hash from = ctx.routerHash();
+        if (victim != null) {
+            byte[] b = Base64.decode(victim);
+            if (b != null && b.length == Hash.HASH_LENGTH)
+                from = Hash.create(b);
+        }
+        out.write("<h3>Distance to " + from.toBase64() + "</h3>");
+        prev = null;
+        for (int i = 0; i < days; i++) {
+            out.write("<h3>Distance for " + new Date(now) +
+                      "</h3><table><tr><th>Hash<th>Distance<th>Distance from previous</tr>\n");
+            Hash rkey = rkgen.getRoutingKey(from, now);
+            xor = new XORComparator<Hash>(rkey);
+            Collections.sort(sybils, xor);
+            for (Hash h : sybils) {
+                out.write("<tr><td><tt>" + h.toBase64() + "</tt><td>");
+                BigInteger dist = HashDistance.getDistance(rkey, h);
+                writeDistance(out, fmt, dist);
+                out.write("<td>");
+                if (prev != null) {
+                    dist = HashDistance.getDistance(prev, h);
+                    writeDistance(out, fmt, dist);
+                }
+                prev = h;
+                out.write("</tr>\n");
+            }
+            out.write("</table>\n");
+            out.flush();
+            now += 24*60*60*1000;
+            prev = null;
+        }
+    }
+
+    /** @since 0.9.28 */
+    private static void writeDistance(Writer out, DecimalFormat fmt, BigInteger dist) throws IOException {
+        double distance = biLog2(dist);
+        if (distance < MIN_CLOSE)
+            out.write("<font color=\"red\">");
+        out.write(fmt.format(distance));
+        if (distance < MIN_CLOSE)
+            out.write("</font>");
+    }
+
     /** translate a string */
     private String _t(String s) {
         return Messages.getString(s, _context);
diff --git a/apps/routerconsole/jsp/netdb.jsp b/apps/routerconsole/jsp/netdb.jsp
index f880bc390144a9fdc3cc1219d1cfac89f785c469..0a8f862c457bfaa298282e72afa4d375e2262a0a 100644
--- a/apps/routerconsole/jsp/netdb.jsp
+++ b/apps/routerconsole/jsp/netdb.jsp
@@ -28,5 +28,6 @@
  <jsp:setProperty name="netdbHelper" property="family" value="<%=request.getParameter(\"fam\")%>" />
  <jsp:setProperty name="netdbHelper" property="caps" value="<%=request.getParameter(\"caps\")%>" />
  <jsp:setProperty name="netdbHelper" property="ip" value="<%=request.getParameter(\"ip\")%>" />
+ <jsp:setProperty name="netdbHelper" property="sybil" value="<%=request.getParameter(\"sybil\")%>" />
  <jsp:getProperty name="netdbHelper" property="netDbSummary" />
 </div></div></body></html>
diff --git a/router/java/src/net/i2p/data/router/RouterKeyGenerator.java b/router/java/src/net/i2p/data/router/RouterKeyGenerator.java
index 02e49ede5aea720d8fffd0759f1272bdbe255222..0f71c1b513138ac5b54e57ab53b545524f416952 100644
--- a/router/java/src/net/i2p/data/router/RouterKeyGenerator.java
+++ b/router/java/src/net/i2p/data/router/RouterKeyGenerator.java
@@ -186,6 +186,26 @@ public class RouterKeyGenerator extends RoutingKeyGenerator {
         return getKey(origKey, _nextModData);
     }
     
+    /**
+     * Get the routing key for the specified date, not today's
+     *
+     * @param time Java time
+     * @since 0.9.28
+     */
+    public Hash getRoutingKey(Hash origKey, long time) {
+        String modVal;
+        synchronized(this) {
+            modVal = _fmt.format(time);
+        }
+        if (modVal.length() != LENGTH)
+            throw new IllegalStateException();
+        byte[] mod = new byte[LENGTH];
+        for (int i = 0; i < LENGTH; i++) {
+            mod[i] = (byte)(modVal.charAt(i) & 0xFF);
+        }
+        return getKey(origKey, mod);
+    }
+    
     /**
      * Generate a modified (yet consistent) hash from the origKey by generating the
      * SHA256 of the targetKey with the specified modData appended to it