diff --git a/apps/addressbook/build.xml b/apps/addressbook/build.xml
index ad27a009059cb83d09cdaebac3cab0e145e93160..beff280f8c33227ad719c922d6fa51c31c636ee0 100644
--- a/apps/addressbook/build.xml
+++ b/apps/addressbook/build.xml
@@ -56,6 +56,7 @@
 			<manifest>
 				<attribute name="Main-Class" value="addressbook.Daemon"/>
 				<attribute name="Implementation-Version" value="${full.version}" />
+				<attribute name="Built-By" value="${build.built-by}" />
 		                <attribute name="Build-Date" value="${build.timestamp}" />
 		                <attribute name="Base-Revision" value="${workspace.version}" />
 		                <attribute name="Workspace-Changes" value="${workspace.changes}" />
@@ -75,6 +76,7 @@
 		<war basedir="${dist}/tmp" webxml="web.xml" destfile="${dist}/${war}">
 			<manifest>
 				<attribute name="Implementation-Version" value="${full.version}" />
+				<attribute name="Built-By" value="${build.built-by}" />
 		                <attribute name="Build-Date" value="${build.timestamp}" />
 		                <attribute name="Base-Revision" value="${workspace.version}" />
 		                <attribute name="Workspace-Changes" value="${workspace.changes.tr}" />
diff --git a/apps/i2psnark/_icons/application.png b/apps/i2psnark/icons/application.png
similarity index 100%
rename from apps/i2psnark/_icons/application.png
rename to apps/i2psnark/icons/application.png
diff --git a/apps/i2psnark/_icons/cancel.png b/apps/i2psnark/icons/cancel.png
similarity index 100%
rename from apps/i2psnark/_icons/cancel.png
rename to apps/i2psnark/icons/cancel.png
diff --git a/apps/i2psnark/_icons/cd.png b/apps/i2psnark/icons/cd.png
similarity index 100%
rename from apps/i2psnark/_icons/cd.png
rename to apps/i2psnark/icons/cd.png
diff --git a/apps/i2psnark/_icons/clock.png b/apps/i2psnark/icons/clock.png
similarity index 100%
rename from apps/i2psnark/_icons/clock.png
rename to apps/i2psnark/icons/clock.png
diff --git a/apps/i2psnark/_icons/clock_red.png b/apps/i2psnark/icons/clock_red.png
similarity index 100%
rename from apps/i2psnark/_icons/clock_red.png
rename to apps/i2psnark/icons/clock_red.png
diff --git a/apps/i2psnark/_icons/compress.png b/apps/i2psnark/icons/compress.png
similarity index 100%
rename from apps/i2psnark/_icons/compress.png
rename to apps/i2psnark/icons/compress.png
diff --git a/apps/i2psnark/_icons/film.png b/apps/i2psnark/icons/film.png
similarity index 100%
rename from apps/i2psnark/_icons/film.png
rename to apps/i2psnark/icons/film.png
diff --git a/apps/i2psnark/_icons/folder.png b/apps/i2psnark/icons/folder.png
similarity index 100%
rename from apps/i2psnark/_icons/folder.png
rename to apps/i2psnark/icons/folder.png
diff --git a/apps/i2psnark/_icons/html.png b/apps/i2psnark/icons/html.png
similarity index 100%
rename from apps/i2psnark/_icons/html.png
rename to apps/i2psnark/icons/html.png
diff --git a/apps/i2psnark/_icons/magnet.png b/apps/i2psnark/icons/magnet.png
similarity index 100%
rename from apps/i2psnark/_icons/magnet.png
rename to apps/i2psnark/icons/magnet.png
diff --git a/apps/i2psnark/_icons/music.png b/apps/i2psnark/icons/music.png
similarity index 100%
rename from apps/i2psnark/_icons/music.png
rename to apps/i2psnark/icons/music.png
diff --git a/apps/i2psnark/_icons/package.png b/apps/i2psnark/icons/package.png
similarity index 100%
rename from apps/i2psnark/_icons/package.png
rename to apps/i2psnark/icons/package.png
diff --git a/apps/i2psnark/_icons/page.png b/apps/i2psnark/icons/page.png
similarity index 100%
rename from apps/i2psnark/_icons/page.png
rename to apps/i2psnark/icons/page.png
diff --git a/apps/i2psnark/_icons/page_white.png b/apps/i2psnark/icons/page_white.png
similarity index 100%
rename from apps/i2psnark/_icons/page_white.png
rename to apps/i2psnark/icons/page_white.png
diff --git a/apps/i2psnark/_icons/page_white_acrobat.png b/apps/i2psnark/icons/page_white_acrobat.png
similarity index 100%
rename from apps/i2psnark/_icons/page_white_acrobat.png
rename to apps/i2psnark/icons/page_white_acrobat.png
diff --git a/apps/i2psnark/_icons/photo.png b/apps/i2psnark/icons/photo.png
similarity index 100%
rename from apps/i2psnark/_icons/photo.png
rename to apps/i2psnark/icons/photo.png
diff --git a/apps/i2psnark/_icons/plugin.png b/apps/i2psnark/icons/plugin.png
similarity index 100%
rename from apps/i2psnark/_icons/plugin.png
rename to apps/i2psnark/icons/plugin.png
diff --git a/apps/i2psnark/_icons/tick.png b/apps/i2psnark/icons/tick.png
similarity index 100%
rename from apps/i2psnark/_icons/tick.png
rename to apps/i2psnark/icons/tick.png
diff --git a/apps/i2psnark/java/build.xml b/apps/i2psnark/java/build.xml
index c5a3b106fd7fd5e3d44efcb0184818a04652c005..94dd809b6e528dfab3eaa711e343e70e82b840d7 100644
--- a/apps/i2psnark/java/build.xml
+++ b/apps/i2psnark/java/build.xml
@@ -61,6 +61,7 @@
                 <attribute name="Main-Class" value="org.klomp.snark.Snark" />
                 <attribute name="Class-Path" value="i2p.jar mstreaming.jar streaming.jar" />
                 <attribute name="Implementation-Version" value="${full.version}" />
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.tr}" />
@@ -95,11 +96,16 @@
     <target name="war" depends="jar, bundle, warUpToDate, listChangedFiles" unless="war.uptodate" > 
         <!-- set if unset -->
         <property name="workspace.changes.tr" value="" />
-        <war destfile="../i2psnark.war" webxml="../web.xml" basedir="../" includes="_icons/*" >
+        <copy todir="build/icons/.icons" >
+            <fileset dir="../icons/" />
+        </copy>
+        <war destfile="../i2psnark.war" webxml="../web.xml" >
           <!-- include only the web stuff, as of 0.7.12 the router will add i2psnark.jar to the classpath for the war -->
           <classes dir="./build/obj" includes="**/web/*.class" />
+            <fileset dir="build/icons/" />
             <manifest>
                 <attribute name="Implementation-Version" value="${full.version}" />
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.tr}" />
@@ -109,7 +115,7 @@
     
     <target name="warUpToDate">
         <uptodate property="war.uptodate" targetfile="../i2psnark.war" >
-            <srcfiles dir= "." includes="build/obj/org/klomp/snark/web/*.class ../_icons/* ../web.xml" />
+            <srcfiles dir= "." includes="build/obj/org/klomp/snark/web/*.class ../icons/* ../web.xml" />
         </uptodate>
     </target>
     
diff --git a/apps/i2psnark/java/src/org/klomp/snark/ExtensionHandler.java b/apps/i2psnark/java/src/org/klomp/snark/ExtensionHandler.java
index b3770070f4e99436f747a1ca8aee559b64bc51f5..3d68677646b9c0ce54463cb32bbd172768ad7368 100644
--- a/apps/i2psnark/java/src/org/klomp/snark/ExtensionHandler.java
+++ b/apps/i2psnark/java/src/org/klomp/snark/ExtensionHandler.java
@@ -39,15 +39,19 @@ abstract class ExtensionHandler {
 
   /**
    *  @param metasize -1 if unknown
+   *  @param pexAndMetadata advertise these capabilities
    *  @return bencoded outgoing handshake message
    */
-    public static byte[] getHandshake(int metasize) {
+    public static byte[] getHandshake(int metasize, boolean pexAndMetadata) {
         Map<String, Object> handshake = new HashMap();
         Map<String, Integer> m = new HashMap();
-        m.put(TYPE_METADATA, Integer.valueOf(ID_METADATA));
-        m.put(TYPE_PEX, Integer.valueOf(ID_PEX));
-        if (metasize >= 0)
-            handshake.put("metadata_size", Integer.valueOf(metasize));
+        if (pexAndMetadata) {
+            m.put(TYPE_METADATA, Integer.valueOf(ID_METADATA));
+            m.put(TYPE_PEX, Integer.valueOf(ID_PEX));
+            if (metasize >= 0)
+                handshake.put("metadata_size", Integer.valueOf(metasize));
+        }
+        // include the map even if empty so the far-end doesn't NPE
         handshake.put("m", m);
         handshake.put("p", Integer.valueOf(6881));
         handshake.put("v", "I2PSnark");
diff --git a/apps/i2psnark/java/src/org/klomp/snark/I2PSnarkUtil.java b/apps/i2psnark/java/src/org/klomp/snark/I2PSnarkUtil.java
index 254f74c44b38de1510f458878f4c198fe409de48..81c56a1947d0b9a1214f1548d29aa6e4c613b742 100644
--- a/apps/i2psnark/java/src/org/klomp/snark/I2PSnarkUtil.java
+++ b/apps/i2psnark/java/src/org/klomp/snark/I2PSnarkUtil.java
@@ -219,6 +219,10 @@ public class I2PSnarkUtil {
             //    opts.setProperty("i2p.streaming.writeTimeout", "90000");
             //if (opts.getProperty("i2p.streaming.readTimeout") == null)
             //    opts.setProperty("i2p.streaming.readTimeout", "120000");
+            if (opts.getProperty("i2p.streaming.maxConnsPerMinute") == null)
+                opts.setProperty("i2p.streaming.maxConnsPerMinute", "2");
+            if (opts.getProperty("i2p.streaming.maxTotalConnsPerMinute") == null)
+                opts.setProperty("i2p.streaming.maxTotalConnsPerMinute", "6");
             _manager = I2PSocketManagerFactory.createManager(_i2cpHost, _i2cpPort, opts);
         }
         // FIXME this only instantiates krpc once, left stuck with old manager
diff --git a/apps/i2psnark/java/src/org/klomp/snark/MetaInfo.java b/apps/i2psnark/java/src/org/klomp/snark/MetaInfo.java
index 28677addaf2d361a291eee0590ef0187558feb39..648a055243f25dd3668421154382c7b362bbb78f 100644
--- a/apps/i2psnark/java/src/org/klomp/snark/MetaInfo.java
+++ b/apps/i2psnark/java/src/org/klomp/snark/MetaInfo.java
@@ -61,6 +61,7 @@ public class MetaInfo
   private final int piece_length;
   private final byte[] piece_hashes;
   private final long length;
+  private final boolean privateTorrent;
   private Map<String, BEValue> infoMap;
 
   /**
@@ -71,7 +72,7 @@ public class MetaInfo
    *  @param lengths null for single-file torrent
    */
   MetaInfo(String announce, String name, String name_utf8, List<List<String>> files, List<Long> lengths,
-           int piece_length, byte[] piece_hashes, long length)
+           int piece_length, byte[] piece_hashes, long length, boolean privateTorrent)
   {
     this.announce = announce;
     this.name = name;
@@ -82,6 +83,7 @@ public class MetaInfo
     this.piece_length = piece_length;
     this.piece_hashes = piece_hashes;
     this.length = length;
+    this.privateTorrent = privateTorrent;
 
     // TODO if we add a parameter for other keys
     //if (other != null) {
@@ -160,6 +162,10 @@ public class MetaInfo
     else
         name_utf8 = null;
 
+    // BEP 27
+    val = info.get("private");
+    privateTorrent = val != null && val.getString().equals("1");
+
     val = info.get("piece length");
     if (val == null)
         throw new InvalidBEncodingException("Missing piece length number");
@@ -318,6 +324,14 @@ public class MetaInfo
     return name;
   }
 
+  /**
+   * Is it a private torrent?
+   * @since 0.9
+   */
+  public boolean isPrivate() {
+    return privateTorrent;
+  }
+
   /**
    * Returns a list of lists of file name hierarchies or null if it is
    * a single name. It has the same size as the list returned by
@@ -439,7 +453,7 @@ public class MetaInfo
   {
     return new MetaInfo(announce, name, name_utf8, files,
                         lengths, piece_length,
-                        piece_hashes, length);
+                        piece_hashes, length, privateTorrent);
   }
 
   /**
@@ -475,6 +489,10 @@ public class MetaInfo
     info.put("name", name);
     if (name_utf8 != null)
         info.put("name.utf-8", name_utf8);
+    // BEP 27
+    if (privateTorrent)
+        info.put("private", "1");
+
     info.put("piece length", Integer.valueOf(piece_length));
     info.put("pieces", piece_hashes);
     if (files == null)
diff --git a/apps/i2psnark/java/src/org/klomp/snark/Peer.java b/apps/i2psnark/java/src/org/klomp/snark/Peer.java
index 5489148121228bab903e2c4ac79ef7c1c03b979b..1330365ce879d6aa28d4524001265b70b7c96897 100644
--- a/apps/i2psnark/java/src/org/klomp/snark/Peer.java
+++ b/apps/i2psnark/java/src/org/klomp/snark/Peer.java
@@ -268,7 +268,8 @@ public class Peer implements Comparable
             if (_log.shouldLog(Log.DEBUG))
                 _log.debug("Peer supports extensions, sending reply message");
             int metasize = metainfo != null ? metainfo.getInfoBytes().length : -1;
-            out.sendExtension(0, ExtensionHandler.getHandshake(metasize));
+            boolean pexAndMetadata = metainfo == null || !metainfo.isPrivate();
+            out.sendExtension(0, ExtensionHandler.getHandshake(metasize, pexAndMetadata));
         }
 
         if ((options & OPTION_I2P_DHT) != 0 && util.getDHT() != null) {
diff --git a/apps/i2psnark/java/src/org/klomp/snark/PeerCoordinator.java b/apps/i2psnark/java/src/org/klomp/snark/PeerCoordinator.java
index b87c7af3af4b0a69720354ec11988dc4309066c7..c9f70ba251777c2106d602b1735770d6f9fbce2b 100644
--- a/apps/i2psnark/java/src/org/klomp/snark/PeerCoordinator.java
+++ b/apps/i2psnark/java/src/org/klomp/snark/PeerCoordinator.java
@@ -1186,6 +1186,8 @@ public class PeerCoordinator implements PeerListener
    *  @since 0.8.4
    */
   void sendPeers(Peer peer) {
+      if (metainfo != null && metainfo.isPrivate())
+          return;
       Map<String, BEValue> handshake = peer.getHandshakeMap();
       if (handshake == null)
           return;
diff --git a/apps/i2psnark/java/src/org/klomp/snark/PeerState.java b/apps/i2psnark/java/src/org/klomp/snark/PeerState.java
index 111ddbbe61095716afb728fe84eea237e3cc685e..c83a43b19173786decf0ae70651ebdf69d825bea 100644
--- a/apps/i2psnark/java/src/org/klomp/snark/PeerState.java
+++ b/apps/i2psnark/java/src/org/klomp/snark/PeerState.java
@@ -489,6 +489,13 @@ class PeerState implements DataLoader
   /** @since 0.8.2 */
   void extensionMessage(int id, byte[] bs)
   {
+      if (metainfo != null && metainfo.isPrivate() &&
+          (id == ExtensionHandler.ID_METADATA || id == ExtensionHandler.ID_PEX)) {
+          // shouldn't get this since we didn't advertise it but they could send it anyway
+          if (_log.shouldLog(Log.WARN))
+              _log.warn("Private torrent, ignoring ext msg " + id);
+          return;
+      }
       ExtensionHandler.handleMessage(peer, listener, id, bs);
       // Peer coord will get metadata from MagnetState,
       // verify, and then call gotMetaInfo()
diff --git a/apps/i2psnark/java/src/org/klomp/snark/SnarkManager.java b/apps/i2psnark/java/src/org/klomp/snark/SnarkManager.java
index 0dd3e0731d5b9366266a78554b58cad2d878b6e8..4c46ec6f040291f2fc584dc10124e9fddcfbfbab 100644
--- a/apps/i2psnark/java/src/org/klomp/snark/SnarkManager.java
+++ b/apps/i2psnark/java/src/org/klomp/snark/SnarkManager.java
@@ -8,6 +8,8 @@ import java.io.FilenameFilter;
 import java.io.IOException;
 import java.io.OutputStream;
 import java.util.ArrayList;
+import java.util.Collections;
+import java.util.Comparator;
 import java.util.HashMap;
 import java.util.HashSet;
 import java.util.Iterator;
@@ -35,8 +37,6 @@ import net.i2p.util.SecureFileOutputStream;
  * Manage multiple snarks
  */
 public class SnarkManager implements Snark.CompleteListener {
-    private static SnarkManager _instance = new SnarkManager();
-    public static SnarkManager instance() { return _instance; }
     
     /**
      *  Map of (canonical) filename of the .torrent file to Snark instance.
@@ -57,6 +57,7 @@ public class SnarkManager implements Snark.CompleteListener {
     private ConnectionAcceptor _connectionAcceptor;
     private Thread _monitor;
     private volatile boolean _running;
+    private final Map<String, String> _trackerMap;
     
     public static final String PROP_I2CP_HOST = "i2psnark.i2cpHost";
     public static final String PROP_I2CP_PORT = "i2psnark.i2cpPort";
@@ -89,6 +90,34 @@ public class SnarkManager implements Snark.CompleteListener {
     public static final int DEFAULT_STARTUP_DELAY = 3; 
     public static final int DEFAULT_REFRESH_DELAY_SECS = 60;
 
+    /**
+     *  "name", "announceURL=websiteURL" pairs
+     *  '=' in announceURL must be escaped as &#44;
+     */
+    private static final String DEFAULT_TRACKERS[] = { 
+//       "Postman", "http://YRgrgTLGnbTq2aZOZDJQ~o6Uk5k6TK-OZtx0St9pb0G-5EGYURZioxqYG8AQt~LgyyI~NCj6aYWpPO-150RcEvsfgXLR~CxkkZcVpgt6pns8SRc3Bi-QSAkXpJtloapRGcQfzTtwllokbdC-aMGpeDOjYLd8b5V9Im8wdCHYy7LRFxhEtGb~RL55DA8aYOgEXcTpr6RPPywbV~Qf3q5UK55el6Kex-6VCxreUnPEe4hmTAbqZNR7Fm0hpCiHKGoToRcygafpFqDw5frLXToYiqs9d4liyVB-BcOb0ihORbo0nS3CLmAwZGvdAP8BZ7cIYE3Z9IU9D1G8JCMxWarfKX1pix~6pIA-sp1gKlL1HhYhPMxwyxvuSqx34o3BqU7vdTYwWiLpGM~zU1~j9rHL7x60pVuYaXcFQDR4-QVy26b6Pt6BlAZoFmHhPcAuWfu-SFhjyZYsqzmEmHeYdAwa~HojSbofg0TMUgESRXMw6YThK1KXWeeJVeztGTz25sL8AAAA.i2p/announce.php=http://tracker.postman.i2p/"
+//       , "eBook", "http://E71FRom6PZNEqTN2Lr8P-sr23b7HJVC32KoGnVQjaX6zJiXwhJy2HsXob36Qmj81TYFZdewFZa9mSJ533UZgGyQkXo2ahctg82JKYZfDe5uDxAn1E9YPjxZCWJaFJh0S~UwSs~9AZ7UcauSJIoNtpxrtbmRNVFLqnkEDdLZi26TeucfOmiFmIWnVblLniWv3tG1boE9Abd-6j3FmYVrRucYuepAILYt6katmVNOk6sXmno1Eynrp~~MBuFq0Ko6~jsc2E2CRVYXDhGHEMdt-j6JUz5D7S2RIVzDRqQyAZLKJ7OdQDmI31przzmne1vOqqqLC~1xUumZVIvF~yOeJUGNjJ1Vx0J8i2BQIusn1pQJ6UCB~ZtZZLQtEb8EPVCfpeRi2ri1M5CyOuxN0V5ekmPHrYIBNevuTCRC26NP7ZS5VDgx1~NaC3A-CzJAE6f1QXi0wMI9aywNG5KGzOPifcsih8eyGyytvgLtrZtV7ykzYpPCS-rDfITncpn5hliPUAAAA.i2p/pub/bt/announce.php=http://de-ebook-archiv.i2p/pub/bt/"
+//       , "Gaytorrents", "http://uxPWHbK1OIj9HxquaXuhMiIvi21iK0~ZiG9d8G0840ZXIg0r6CbiV71xlsqmdnU6wm0T2LySriM0doW2gUigo-5BNkUquHwOjLROiETnB3ZR0Ml4IGa6QBPn1aAq2d9~g1r1nVjLE~pcFnXB~cNNS7kIhX1d6nLgYVZf0C2cZopEow2iWVUggGGnAA9mHjE86zLEnTvAyhbAMTqDQJhEuLa0ZYSORqzJDMkQt90MV4YMjX1ICY6RfUSFmxEqu0yWTrkHsTtRw48l~dz9wpIgc0a0T9C~eeWvmBFTqlJPtQZwntpNeH~jF7nlYzB58olgV2HHFYpVYD87DYNzTnmNWxCJ5AfDorm6AIUCV2qaE7tZtI1h6fbmGpGlPyW~Kw5GXrRfJwNvr6ajwAVi~bPVnrBwDZezHkfW4slOO8FACPR28EQvaTu9nwhAbqESxV2hCTq6vQSGjuxHeOuzBOEvRWkLKOHWTC09t2DbJ94FSqETmZopTB1ukEmaxRWbKSIaAAAA.i2p/announce.php=http://gaytorrents.i2p/"
+//       , "NickyB", "http://9On6d3cZ27JjwYCtyJJbowe054d5tFnfMjv4PHsYs-EQn4Y4mk2zRixatvuAyXz2MmRfXG-NAUfhKr0KCxRNZbvHmlckYfT-WBzwwpiMAl0wDFY~Pl8cqXuhfikSG5WrqdPfDNNIBuuznS0dqaczf~OyVaoEOpvuP3qV6wKqbSSLpjOwwAaQPHjlRtNIW8-EtUZp-I0LT45HSoowp~6b7zYmpIyoATvIP~sT0g0MTrczWhbVTUZnEkZeLhOR0Duw1-IRXI2KHPbA24wLO9LdpKKUXed05RTz0QklW5ROgR6TYv7aXFufX8kC0-DaKvQ5JKG~h8lcoHvm1RCzNqVE-2aiZnO2xH08H-iCWoLNJE-Td2kT-Tsc~3QdQcnEUcL5BF-VT~QYRld2--9r0gfGl-yDrJZrlrihHGr5J7ImahelNn9PpkVp6eIyABRmJHf2iicrk3CtjeG1j9OgTSwaNmEpUpn4aN7Kx0zNLdH7z6uTgCGD9Kmh1MFYrsoNlTp4AAAA.i2p/bittorrent/announce.php=http://nickyb.i2p/bittorrent/"
+//       , "Orion", "http://gKik1lMlRmuroXVGTZ~7v4Vez3L3ZSpddrGZBrxVriosCQf7iHu6CIk8t15BKsj~P0JJpxrofeuxtm7SCUAJEr0AIYSYw8XOmp35UfcRPQWyb1LsxUkMT4WqxAT3s1ClIICWlBu5An~q-Mm0VFlrYLIPBWlUFnfPR7jZ9uP5ZMSzTKSMYUWao3ejiykr~mtEmyls6g-ZbgKZawa9II4zjOy-hdxHgP-eXMDseFsrym4Gpxvy~3Fv9TuiSqhpgm~UeTo5YBfxn6~TahKtE~~sdCiSydqmKBhxAQ7uT9lda7xt96SS09OYMsIWxLeQUWhns-C~FjJPp1D~IuTrUpAFcVEGVL-BRMmdWbfOJEcWPZ~CBCQSO~VkuN1ebvIOr9JBerFMZSxZtFl8JwcrjCIBxeKPBmfh~xYh16BJm1BBBmN1fp2DKmZ2jBNkAmnUbjQOqWvUcehrykWk5lZbE7bjJMDFH48v3SXwRuDBiHZmSbsTY6zhGY~GkMQHNGxPMMSIAAAA.i2p/bt/announce.php=http://orion.i2p/bt/"
+//       , "anonymity", "http://8EoJZIKrWgGuDrxA3nRJs1jsPfiGwmFWL91hBrf0HA7oKhEvAna4Ocx47VLUR9retVEYBAyWFK-eZTPcvhnz9XffBEiJQQ~kFSCqb1fV6IfPiV3HySqi9U5Caf6~hC46fRd~vYnxmaBLICT3N160cxBETqH3v2rdxdJpvYt8q4nMk9LUeVXq7zqCTFLLG5ig1uKgNzBGe58iNcsvTEYlnbYcE930ABmrzj8G1qQSgSwJ6wx3tUQNl1z~4wSOUMan~raZQD60lRK70GISjoX0-D0Po9WmPveN3ES3g72TIET3zc3WPdK2~lgmKGIs8GgNLES1cXTolvbPhdZK1gxddRMbJl6Y6IPFyQ9o4-6Rt3Lp-RMRWZ2TG7j2OMcNSiOmATUhKEFBDfv-~SODDyopGBmfeLw16F4NnYednvn4qP10dyMHcUASU6Zag4mfc2-WivrOqeWhD16fVAh8MoDpIIT~0r9XmwdaVFyLcjbXObabJczxCAW3fodQUnvuSkwzAAAA.i2p/anonymityTracker/announce.php=http://anonymityweb.i2p/anonymityTracker/"
+//       , "The freak's tracker", "http://mHKva9x24E5Ygfey2llR1KyQHv5f8hhMpDMwJDg1U-hABpJ2NrQJd6azirdfaR0OKt4jDlmP2o4Qx0H598~AteyD~RJU~xcWYdcOE0dmJ2e9Y8-HY51ie0B1yD9FtIV72ZI-V3TzFDcs6nkdX9b81DwrAwwFzx0EfNvK1GLVWl59Ow85muoRTBA1q8SsZImxdyZ-TApTVlMYIQbdI4iQRwU9OmmtefrCe~ZOf4UBS9-KvNIqUL0XeBSqm0OU1jq-D10Ykg6KfqvuPnBYT1BYHFDQJXW5DdPKwcaQE4MtAdSGmj1epDoaEBUa9btQlFsM2l9Cyn1hzxqNWXELmx8dRlomQLlV4b586dRzW~fLlOPIGC13ntPXogvYvHVyEyptXkv890jC7DZNHyxZd5cyrKC36r9huKvhQAmNABT2Y~pOGwVrb~RpPwT0tBuPZ3lHYhBFYmD8y~AOhhNHKMLzea1rfwTvovBMByDdFps54gMN1mX4MbCGT4w70vIopS9yAAAA.i2p/bytemonsoon/announce.php"
+//       , "mastertracker", "http://VzXD~stRKbL3MOmeTn1iaCQ0CFyTmuFHiKYyo0Rd~dFPZFCYH-22rT8JD7i-C2xzYFa4jT5U2aqHzHI-Jre4HL3Ri5hFtZrLk2ax3ji7Qfb6qPnuYkuiF2E2UDmKUOppI8d9Ye7tjdhQVCy0izn55tBaB-U7UWdcvSK2i85sauyw3G0Gfads1Rvy5-CAe2paqyYATcDmGjpUNLoxbfv9KH1KmwRTNH6k1v4PyWYYnhbT39WfKMbBjSxVQRdi19cyJrULSWhjxaQfJHeWx5Z8Ev4bSPByBeQBFl2~4vqy0S5RypINsRSa3MZdbiAAyn5tr5slWR6QdoqY3qBQgBJFZppy-3iWkFqqKgSxCPundF8gdDLC5ddizl~KYcYKl42y9SGFHIukH-TZs8~em0~iahzsqWVRks3zRG~tlBcX2U3M2~OJs~C33-NKhyfZT7-XFBREvb8Szmd~p66jDxrwOnKaku-G6DyoQipJqIz4VHmY9-y5T8RrUcJcM-5lVoMpAAAA.i2p/announce.php=http://tracker.mastertracker.i2p/"
+//       , "Galen", "http://5jpwQMI5FT303YwKa5Rd38PYSX04pbIKgTaKQsWbqoWjIfoancFdWCShXHLI5G5ofOb0Xu11vl2VEMyPsg1jUFYSVnu4-VfMe3y4TKTR6DTpetWrnmEK6m2UXh91J5DZJAKlgmO7UdsFlBkQfR2rY853-DfbJtQIFl91tbsmjcA5CGQi4VxMFyIkBzv-pCsuLQiZqOwWasTlnzey8GcDAPG1LDcvfflGV~6F5no9mnuisZPteZKlrv~~TDoXTj74QjByWc4EOYlwqK8sbU9aOvz~s31XzErbPTfwiawiaZ0RUI-IDrKgyvmj0neuFTWgjRGVTH8bz7cBZIc3viy6ioD-eMQOrXaQL0TCWZUelRwHRvgdPiQrxdYQs7ixkajeHzxi-Pq0EMm5Vbh3j3Q9kfUFW3JjFDA-MLB4g6XnjCbM5J1rC0oOBDCIEfhQkszru5cyLjHiZ5yeA0VThgu~c7xKHybv~OMXION7V8pBKOgET7ZgAkw1xgYe3Kkyq5syAAAA.i2p/tr/announce.php=http://galen.i2p/tr/"
+       "Postman", "http://tracker2.postman.i2p/announce.php=http://tracker2.postman.i2p/"
+       ,"Welterde", "http://tracker.welterde.i2p/a=http://tracker.welterde.i2p/stats?mode=top5"
+       ,"Diftracker", "http://n--XWjHjUPjnMNrSwXA2OYXpMIUL~u4FNXnrt2HtjK3y6j~4SOClyyeKzd0zRPlixxkCe2wfBIYye3bZsaqAD8bd0QMmowxbq91WpjsPfKMiphJbePKXtYAVARiy0cqyvh1d2LyDE-6wkvgaw45hknmS0U-Dg3YTJZbAQRU2SKXgIlAbWCv4R0kDFBLEVpReDiJef3rzAWHiW8yjmJuJilkYjMwlfRjw8xx1nl2s~yhlljk1pl13jGYb0nfawQnuOWeP-ASQWvAAyVgKvZRJE2O43S7iveu9piuv7plXWbt36ef7ndu2GNoNyPOBdpo9KUZ-NOXm4Kgh659YtEibL15dEPAOdxprY0sYUurVw8OIWqrpX7yn08nbi6qHVGqQwTpxH35vkL8qrCbm-ym7oQJQnNmSDrNTyWYRFSq5s5~7DAdFDzqRPW-pX~g0zEivWj5tzkhvG9rVFgFo0bpQX3X0PUAV9Xbyf8u~v8Zbr9K1pCPqBq9XEr4TqaLHw~bfAAAA.i2p/announce.php=http://diftracker.i2p/"
+//       , "CRSTRACK", "http://b4G9sCdtfvccMAXh~SaZrPqVQNyGQbhbYMbw6supq2XGzbjU4NcOmjFI0vxQ8w1L05twmkOvg5QERcX6Mi8NQrWnR0stLExu2LucUXg1aYjnggxIR8TIOGygZVIMV3STKH4UQXD--wz0BUrqaLxPhrm2Eh9Hwc8TdB6Na4ShQUq5Xm8D4elzNUVdpM~RtChEyJWuQvoGAHY3ppX-EJJLkiSr1t77neS4Lc-KofMVmgI9a2tSSpNAagBiNI6Ak9L1T0F9uxeDfEG9bBSQPNMOSUbAoEcNxtt7xOW~cNOAyMyGydwPMnrQ5kIYPY8Pd3XudEko970vE0D6gO19yoBMJpKx6Dh50DGgybLQ9CpRaynh2zPULTHxm8rneOGRcQo8D3mE7FQ92m54~SvfjXjD2TwAVGI~ae~n9HDxt8uxOecAAvjjJ3TD4XM63Q9TmB38RmGNzNLDBQMEmJFpqQU8YeuhnS54IVdUoVQFqui5SfDeLXlSkh4vYoMU66pvBfWbAAAA.i2p/tracker/announce.php=http://crstrack.i2p/tracker/"
+//       ,"Exotrack", "http://blbgywsjubw3d2zih2giokakhe3o2cko7jtte4risb3hohbcoyva.b32.i2p/announce.php=http://exotrack.i2p/"
+    };
+    
+    /** comma delimited list of name=announceURL=baseURL for the trackers to be displayed */
+    public static final String PROP_TRACKERS = "i2psnark.trackers";
+
+    private static final SnarkManager _instance = new SnarkManager();
+
+    public static SnarkManager instance() { return _instance; }
+
     private SnarkManager() {
         _snarks = new ConcurrentHashMap();
         _magnets = new ConcurrentHashSet();
@@ -100,6 +129,7 @@ public class SnarkManager implements Snark.CompleteListener {
         _configFile = new File(CONFIG_FILE);
         if (!_configFile.isAbsolute())
             _configFile = new File(_context.getConfigDir(), CONFIG_FILE);
+        _trackerMap = Collections.synchronizedMap(new TreeMap(new IgnoreCaseComparator()));
         loadConfig(null);
     }
 
@@ -312,6 +342,7 @@ public class SnarkManager implements Snark.CompleteListener {
         boolean bOT = useOT == null || Boolean.valueOf(useOT).booleanValue();
         _util.setUseOpenTrackers(bOT);
         getDataDir().mkdirs();
+        initTrackerMap();
     }
     
     private int getInt(String prop, int defaultVal) {
@@ -663,7 +694,9 @@ public class SnarkManager implements Snark.CompleteListener {
                     }
 
                     if (!TrackerClient.isValidAnnounce(info.getAnnounce())) {
-                        if (_util.shouldUseOpenTrackers() && _util.getOpenTrackers() != null) {
+                        if (info.isPrivate()) {
+                            addMessage(_("ERROR - No I2P trackers in private torrent \"{0}\"", info.getName()));
+                        } else if (_util.shouldUseOpenTrackers() && _util.getOpenTrackers() != null) {
                             //addMessage(_("Warning - No I2P trackers in \"{0}\", will announce to I2P open trackers and DHT only.", info.getName()));
                             addMessage(_("Warning - No I2P trackers in \"{0}\", will announce to I2P open trackers only.", info.getName()));
                         //} else if (_util.getDHT() != null) {
@@ -1356,57 +1389,59 @@ public class SnarkManager implements Snark.CompleteListener {
     }
 
     /**
-     *  "name", "announceURL=websiteURL" pairs
+     *  Sorted map of name to announceURL=baseURL
+     *  Modifiable, not a copy
      */
-    private static final String DEFAULT_TRACKERS[] = { 
-//       "Postman", "http://YRgrgTLGnbTq2aZOZDJQ~o6Uk5k6TK-OZtx0St9pb0G-5EGYURZioxqYG8AQt~LgyyI~NCj6aYWpPO-150RcEvsfgXLR~CxkkZcVpgt6pns8SRc3Bi-QSAkXpJtloapRGcQfzTtwllokbdC-aMGpeDOjYLd8b5V9Im8wdCHYy7LRFxhEtGb~RL55DA8aYOgEXcTpr6RPPywbV~Qf3q5UK55el6Kex-6VCxreUnPEe4hmTAbqZNR7Fm0hpCiHKGoToRcygafpFqDw5frLXToYiqs9d4liyVB-BcOb0ihORbo0nS3CLmAwZGvdAP8BZ7cIYE3Z9IU9D1G8JCMxWarfKX1pix~6pIA-sp1gKlL1HhYhPMxwyxvuSqx34o3BqU7vdTYwWiLpGM~zU1~j9rHL7x60pVuYaXcFQDR4-QVy26b6Pt6BlAZoFmHhPcAuWfu-SFhjyZYsqzmEmHeYdAwa~HojSbofg0TMUgESRXMw6YThK1KXWeeJVeztGTz25sL8AAAA.i2p/announce.php=http://tracker.postman.i2p/"
-//       , "eBook", "http://E71FRom6PZNEqTN2Lr8P-sr23b7HJVC32KoGnVQjaX6zJiXwhJy2HsXob36Qmj81TYFZdewFZa9mSJ533UZgGyQkXo2ahctg82JKYZfDe5uDxAn1E9YPjxZCWJaFJh0S~UwSs~9AZ7UcauSJIoNtpxrtbmRNVFLqnkEDdLZi26TeucfOmiFmIWnVblLniWv3tG1boE9Abd-6j3FmYVrRucYuepAILYt6katmVNOk6sXmno1Eynrp~~MBuFq0Ko6~jsc2E2CRVYXDhGHEMdt-j6JUz5D7S2RIVzDRqQyAZLKJ7OdQDmI31przzmne1vOqqqLC~1xUumZVIvF~yOeJUGNjJ1Vx0J8i2BQIusn1pQJ6UCB~ZtZZLQtEb8EPVCfpeRi2ri1M5CyOuxN0V5ekmPHrYIBNevuTCRC26NP7ZS5VDgx1~NaC3A-CzJAE6f1QXi0wMI9aywNG5KGzOPifcsih8eyGyytvgLtrZtV7ykzYpPCS-rDfITncpn5hliPUAAAA.i2p/pub/bt/announce.php=http://de-ebook-archiv.i2p/pub/bt/"
-//       , "Gaytorrents", "http://uxPWHbK1OIj9HxquaXuhMiIvi21iK0~ZiG9d8G0840ZXIg0r6CbiV71xlsqmdnU6wm0T2LySriM0doW2gUigo-5BNkUquHwOjLROiETnB3ZR0Ml4IGa6QBPn1aAq2d9~g1r1nVjLE~pcFnXB~cNNS7kIhX1d6nLgYVZf0C2cZopEow2iWVUggGGnAA9mHjE86zLEnTvAyhbAMTqDQJhEuLa0ZYSORqzJDMkQt90MV4YMjX1ICY6RfUSFmxEqu0yWTrkHsTtRw48l~dz9wpIgc0a0T9C~eeWvmBFTqlJPtQZwntpNeH~jF7nlYzB58olgV2HHFYpVYD87DYNzTnmNWxCJ5AfDorm6AIUCV2qaE7tZtI1h6fbmGpGlPyW~Kw5GXrRfJwNvr6ajwAVi~bPVnrBwDZezHkfW4slOO8FACPR28EQvaTu9nwhAbqESxV2hCTq6vQSGjuxHeOuzBOEvRWkLKOHWTC09t2DbJ94FSqETmZopTB1ukEmaxRWbKSIaAAAA.i2p/announce.php=http://gaytorrents.i2p/"
-//       , "NickyB", "http://9On6d3cZ27JjwYCtyJJbowe054d5tFnfMjv4PHsYs-EQn4Y4mk2zRixatvuAyXz2MmRfXG-NAUfhKr0KCxRNZbvHmlckYfT-WBzwwpiMAl0wDFY~Pl8cqXuhfikSG5WrqdPfDNNIBuuznS0dqaczf~OyVaoEOpvuP3qV6wKqbSSLpjOwwAaQPHjlRtNIW8-EtUZp-I0LT45HSoowp~6b7zYmpIyoATvIP~sT0g0MTrczWhbVTUZnEkZeLhOR0Duw1-IRXI2KHPbA24wLO9LdpKKUXed05RTz0QklW5ROgR6TYv7aXFufX8kC0-DaKvQ5JKG~h8lcoHvm1RCzNqVE-2aiZnO2xH08H-iCWoLNJE-Td2kT-Tsc~3QdQcnEUcL5BF-VT~QYRld2--9r0gfGl-yDrJZrlrihHGr5J7ImahelNn9PpkVp6eIyABRmJHf2iicrk3CtjeG1j9OgTSwaNmEpUpn4aN7Kx0zNLdH7z6uTgCGD9Kmh1MFYrsoNlTp4AAAA.i2p/bittorrent/announce.php=http://nickyb.i2p/bittorrent/"
-//       , "Orion", "http://gKik1lMlRmuroXVGTZ~7v4Vez3L3ZSpddrGZBrxVriosCQf7iHu6CIk8t15BKsj~P0JJpxrofeuxtm7SCUAJEr0AIYSYw8XOmp35UfcRPQWyb1LsxUkMT4WqxAT3s1ClIICWlBu5An~q-Mm0VFlrYLIPBWlUFnfPR7jZ9uP5ZMSzTKSMYUWao3ejiykr~mtEmyls6g-ZbgKZawa9II4zjOy-hdxHgP-eXMDseFsrym4Gpxvy~3Fv9TuiSqhpgm~UeTo5YBfxn6~TahKtE~~sdCiSydqmKBhxAQ7uT9lda7xt96SS09OYMsIWxLeQUWhns-C~FjJPp1D~IuTrUpAFcVEGVL-BRMmdWbfOJEcWPZ~CBCQSO~VkuN1ebvIOr9JBerFMZSxZtFl8JwcrjCIBxeKPBmfh~xYh16BJm1BBBmN1fp2DKmZ2jBNkAmnUbjQOqWvUcehrykWk5lZbE7bjJMDFH48v3SXwRuDBiHZmSbsTY6zhGY~GkMQHNGxPMMSIAAAA.i2p/bt/announce.php=http://orion.i2p/bt/"
-//       , "anonymity", "http://8EoJZIKrWgGuDrxA3nRJs1jsPfiGwmFWL91hBrf0HA7oKhEvAna4Ocx47VLUR9retVEYBAyWFK-eZTPcvhnz9XffBEiJQQ~kFSCqb1fV6IfPiV3HySqi9U5Caf6~hC46fRd~vYnxmaBLICT3N160cxBETqH3v2rdxdJpvYt8q4nMk9LUeVXq7zqCTFLLG5ig1uKgNzBGe58iNcsvTEYlnbYcE930ABmrzj8G1qQSgSwJ6wx3tUQNl1z~4wSOUMan~raZQD60lRK70GISjoX0-D0Po9WmPveN3ES3g72TIET3zc3WPdK2~lgmKGIs8GgNLES1cXTolvbPhdZK1gxddRMbJl6Y6IPFyQ9o4-6Rt3Lp-RMRWZ2TG7j2OMcNSiOmATUhKEFBDfv-~SODDyopGBmfeLw16F4NnYednvn4qP10dyMHcUASU6Zag4mfc2-WivrOqeWhD16fVAh8MoDpIIT~0r9XmwdaVFyLcjbXObabJczxCAW3fodQUnvuSkwzAAAA.i2p/anonymityTracker/announce.php=http://anonymityweb.i2p/anonymityTracker/"
-//       , "The freak's tracker", "http://mHKva9x24E5Ygfey2llR1KyQHv5f8hhMpDMwJDg1U-hABpJ2NrQJd6azirdfaR0OKt4jDlmP2o4Qx0H598~AteyD~RJU~xcWYdcOE0dmJ2e9Y8-HY51ie0B1yD9FtIV72ZI-V3TzFDcs6nkdX9b81DwrAwwFzx0EfNvK1GLVWl59Ow85muoRTBA1q8SsZImxdyZ-TApTVlMYIQbdI4iQRwU9OmmtefrCe~ZOf4UBS9-KvNIqUL0XeBSqm0OU1jq-D10Ykg6KfqvuPnBYT1BYHFDQJXW5DdPKwcaQE4MtAdSGmj1epDoaEBUa9btQlFsM2l9Cyn1hzxqNWXELmx8dRlomQLlV4b586dRzW~fLlOPIGC13ntPXogvYvHVyEyptXkv890jC7DZNHyxZd5cyrKC36r9huKvhQAmNABT2Y~pOGwVrb~RpPwT0tBuPZ3lHYhBFYmD8y~AOhhNHKMLzea1rfwTvovBMByDdFps54gMN1mX4MbCGT4w70vIopS9yAAAA.i2p/bytemonsoon/announce.php"
-//       , "mastertracker", "http://VzXD~stRKbL3MOmeTn1iaCQ0CFyTmuFHiKYyo0Rd~dFPZFCYH-22rT8JD7i-C2xzYFa4jT5U2aqHzHI-Jre4HL3Ri5hFtZrLk2ax3ji7Qfb6qPnuYkuiF2E2UDmKUOppI8d9Ye7tjdhQVCy0izn55tBaB-U7UWdcvSK2i85sauyw3G0Gfads1Rvy5-CAe2paqyYATcDmGjpUNLoxbfv9KH1KmwRTNH6k1v4PyWYYnhbT39WfKMbBjSxVQRdi19cyJrULSWhjxaQfJHeWx5Z8Ev4bSPByBeQBFl2~4vqy0S5RypINsRSa3MZdbiAAyn5tr5slWR6QdoqY3qBQgBJFZppy-3iWkFqqKgSxCPundF8gdDLC5ddizl~KYcYKl42y9SGFHIukH-TZs8~em0~iahzsqWVRks3zRG~tlBcX2U3M2~OJs~C33-NKhyfZT7-XFBREvb8Szmd~p66jDxrwOnKaku-G6DyoQipJqIz4VHmY9-y5T8RrUcJcM-5lVoMpAAAA.i2p/announce.php=http://tracker.mastertracker.i2p/"
-//       , "Galen", "http://5jpwQMI5FT303YwKa5Rd38PYSX04pbIKgTaKQsWbqoWjIfoancFdWCShXHLI5G5ofOb0Xu11vl2VEMyPsg1jUFYSVnu4-VfMe3y4TKTR6DTpetWrnmEK6m2UXh91J5DZJAKlgmO7UdsFlBkQfR2rY853-DfbJtQIFl91tbsmjcA5CGQi4VxMFyIkBzv-pCsuLQiZqOwWasTlnzey8GcDAPG1LDcvfflGV~6F5no9mnuisZPteZKlrv~~TDoXTj74QjByWc4EOYlwqK8sbU9aOvz~s31XzErbPTfwiawiaZ0RUI-IDrKgyvmj0neuFTWgjRGVTH8bz7cBZIc3viy6ioD-eMQOrXaQL0TCWZUelRwHRvgdPiQrxdYQs7ixkajeHzxi-Pq0EMm5Vbh3j3Q9kfUFW3JjFDA-MLB4g6XnjCbM5J1rC0oOBDCIEfhQkszru5cyLjHiZ5yeA0VThgu~c7xKHybv~OMXION7V8pBKOgET7ZgAkw1xgYe3Kkyq5syAAAA.i2p/tr/announce.php=http://galen.i2p/tr/"
-       "Postman", "http://tracker2.postman.i2p/announce.php=http://tracker2.postman.i2p/"
-       ,"Welterde", "http://tracker.welterde.i2p/a=http://tracker.welterde.i2p/stats?mode=top5"
-       ,"Diftracker", "http://n--XWjHjUPjnMNrSwXA2OYXpMIUL~u4FNXnrt2HtjK3y6j~4SOClyyeKzd0zRPlixxkCe2wfBIYye3bZsaqAD8bd0QMmowxbq91WpjsPfKMiphJbePKXtYAVARiy0cqyvh1d2LyDE-6wkvgaw45hknmS0U-Dg3YTJZbAQRU2SKXgIlAbWCv4R0kDFBLEVpReDiJef3rzAWHiW8yjmJuJilkYjMwlfRjw8xx1nl2s~yhlljk1pl13jGYb0nfawQnuOWeP-ASQWvAAyVgKvZRJE2O43S7iveu9piuv7plXWbt36ef7ndu2GNoNyPOBdpo9KUZ-NOXm4Kgh659YtEibL15dEPAOdxprY0sYUurVw8OIWqrpX7yn08nbi6qHVGqQwTpxH35vkL8qrCbm-ym7oQJQnNmSDrNTyWYRFSq5s5~7DAdFDzqRPW-pX~g0zEivWj5tzkhvG9rVFgFo0bpQX3X0PUAV9Xbyf8u~v8Zbr9K1pCPqBq9XEr4TqaLHw~bfAAAA.i2p/announce.php=http://diftracker.i2p/"
-//       , "CRSTRACK", "http://b4G9sCdtfvccMAXh~SaZrPqVQNyGQbhbYMbw6supq2XGzbjU4NcOmjFI0vxQ8w1L05twmkOvg5QERcX6Mi8NQrWnR0stLExu2LucUXg1aYjnggxIR8TIOGygZVIMV3STKH4UQXD--wz0BUrqaLxPhrm2Eh9Hwc8TdB6Na4ShQUq5Xm8D4elzNUVdpM~RtChEyJWuQvoGAHY3ppX-EJJLkiSr1t77neS4Lc-KofMVmgI9a2tSSpNAagBiNI6Ak9L1T0F9uxeDfEG9bBSQPNMOSUbAoEcNxtt7xOW~cNOAyMyGydwPMnrQ5kIYPY8Pd3XudEko970vE0D6gO19yoBMJpKx6Dh50DGgybLQ9CpRaynh2zPULTHxm8rneOGRcQo8D3mE7FQ92m54~SvfjXjD2TwAVGI~ae~n9HDxt8uxOecAAvjjJ3TD4XM63Q9TmB38RmGNzNLDBQMEmJFpqQU8YeuhnS54IVdUoVQFqui5SfDeLXlSkh4vYoMU66pvBfWbAAAA.i2p/tracker/announce.php=http://crstrack.i2p/tracker/"
-//       ,"Exotrack", "http://blbgywsjubw3d2zih2giokakhe3o2cko7jtte4risb3hohbcoyva.b32.i2p/announce.php=http://exotrack.i2p/"
-    };
-    
-    /** comma delimited list of name=announceURL=baseURL for the trackers to be displayed */
-    public static final String PROP_TRACKERS = "i2psnark.trackers";
-    private static Map<String, String> trackerMap = null;
-    /** sorted map of name to announceURL=baseURL */
     public Map<String, String> getTrackers() { 
-        if (trackerMap != null) // only do this once, can't be updated while running
-            return trackerMap;
-        Map<String, String> rv = new TreeMap();
+        return _trackerMap;
+    }
+
+    /** @since 0.9 */
+    private void initTrackerMap() {
         String trackers = _config.getProperty(PROP_TRACKERS);
         if ( (trackers == null) || (trackers.trim().length() <= 0) )
             trackers = _context.getProperty(PROP_TRACKERS);
+        _trackerMap.clear();
         if ( (trackers == null) || (trackers.trim().length() <= 0) ) {
             for (int i = 0; i < DEFAULT_TRACKERS.length; i += 2)
-                rv.put(DEFAULT_TRACKERS[i], DEFAULT_TRACKERS[i+1]);
+                _trackerMap.put(DEFAULT_TRACKERS[i], DEFAULT_TRACKERS[i+1]);
         } else {
-            StringTokenizer tok = new StringTokenizer(trackers, ",");
-            while (tok.hasMoreTokens()) {
-                String pair = tok.nextToken();
-                int split = pair.indexOf('=');
-                if (split <= 0)
-                    continue;
-                String name = pair.substring(0, split).trim();
-                String url = pair.substring(split+1).trim();
+            String[] toks = trackers.split(",");
+            for (int i = 0; i < toks.length; i += 2) {
+                String name = toks[i].trim().replace("&#44;", ",");
+                String url = toks[i+1].trim().replace("&#44;", ",");
                 if ( (name.length() > 0) && (url.length() > 0) )
-                    rv.put(name, url);
+                    _trackerMap.put(name, url);
             }
         }
-        
-        trackerMap = rv;
-        return trackerMap;
     }
-    
+
+    /** @since 0.9 */
+    public void setDefaultTrackerMap() {
+        _trackerMap.clear();
+        for (int i = 0; i < DEFAULT_TRACKERS.length; i += 2) {
+            _trackerMap.put(DEFAULT_TRACKERS[i], DEFAULT_TRACKERS[i+1]);
+        }
+        if (_config.remove(PROP_TRACKERS) != null) {
+            saveConfig();
+        }
+    }
+
+    /** @since 0.9 */
+    public void saveTrackerMap() {
+        StringBuilder buf = new StringBuilder(2048);
+        boolean comma = false;
+        for (Map.Entry<String, String> e : _trackerMap.entrySet()) {
+            if (comma)
+                buf.append(',');
+            else
+                comma = true;
+            buf.append(e.getKey().replace(",", "&#44;")).append(',').append(e.getValue().replace(",", "&#44;"));
+        }
+        _config.setProperty(PROP_TRACKERS, buf.toString());
+        saveConfig();
+    }
+
     private static class TorrentFilenameFilter implements FilenameFilter {
         private static final TorrentFilenameFilter _filter = new TorrentFilenameFilter();
         public static TorrentFilenameFilter instance() { return _filter; }
@@ -1426,4 +1461,14 @@ public class SnarkManager implements Snark.CompleteListener {
             }
         }
     }
+
+    /**
+     *  ignore case, current locale
+     *  @since 0.9
+     */
+    private static class IgnoreCaseComparator implements Comparator<String> {
+        public int compare(String l, String r) {
+            return l.toLowerCase().compareTo(r.toLowerCase());
+        }
+    }
 }
diff --git a/apps/i2psnark/java/src/org/klomp/snark/Storage.java b/apps/i2psnark/java/src/org/klomp/snark/Storage.java
index 804f4c304c0815a00d8a53cd0bdc4520603ec47a..36fdd748cdb5eabfbe9fc6c5345de8a82e8a4d22 100644
--- a/apps/i2psnark/java/src/org/klomp/snark/Storage.java
+++ b/apps/i2psnark/java/src/org/klomp/snark/Storage.java
@@ -101,7 +101,8 @@ public class Storage
    * @param announce may be null
    * @param listener may be null
    */
-  public Storage(I2PSnarkUtil util, File baseFile, String announce, StorageListener listener)
+  public Storage(I2PSnarkUtil util, File baseFile, String announce,
+                 boolean privateTorrent, StorageListener listener)
     throws IOException
   {
     _util = util;
@@ -157,7 +158,7 @@ public class Storage
 
     byte[] piece_hashes = fast_digestCreate();
     metainfo = new MetaInfo(announce, baseFile.getName(), null, files,
-                            lengthsList, piece_size, piece_hashes, total);
+                            lengthsList, piece_size, piece_hashes, total, privateTorrent);
 
   }
 
diff --git a/apps/i2psnark/java/src/org/klomp/snark/TrackerClient.java b/apps/i2psnark/java/src/org/klomp/snark/TrackerClient.java
index 3f7c6b8bd16965bbe7f0177cd28f13ce54c083bc..2007fb84f2812bfb3b6c93f899b4a555311930b4 100644
--- a/apps/i2psnark/java/src/org/klomp/snark/TrackerClient.java
+++ b/apps/i2psnark/java/src/org/klomp/snark/TrackerClient.java
@@ -156,7 +156,7 @@ public class TrackerClient extends I2PAppThread
         primary = "";
     }
     List tlist = _util.getOpenTrackers();
-    if (tlist != null) {
+    if (tlist != null && !meta.isPrivate()) {
         for (int i = 0; i < tlist.size(); i++) {
              String url = (String)tlist.get(i);
              if (!isValidAnnounce(url)) {
@@ -348,7 +348,7 @@ public class TrackerClient extends I2PAppThread
             }  // *** end of trackers loop here
 
             // Get peers from PEX
-            if (left > 0 && coordinator.needPeers() && !stop) {
+            if (left > 0 && coordinator.needPeers() && (!meta.isPrivate()) && !stop) {
                 Set<PeerID> pids = coordinator.getPEXPeers();
                 if (!pids.isEmpty()) {
                     _util.debug("Got " + pids.size() + " from PEX", Snark.INFO);
@@ -370,7 +370,7 @@ public class TrackerClient extends I2PAppThread
 
             // Get peers from DHT
             // FIXME this needs to be in its own thread
-            if (_util.getDHT() != null && !stop) {
+            if (_util.getDHT() != null && (!meta.isPrivate()) && !stop) {
                 int numwant;
                 if (left == 0 || event.equals(STOPPED_EVENT) || !coordinator.needPeers())
                     numwant = 1;
@@ -444,22 +444,33 @@ public class TrackerClient extends I2PAppThread
                                 long downloaded, long left, String event)
     throws IOException
   {
+    StringBuilder buf = new StringBuilder(512);
+    buf.append(tr.announce);
+    if (tr.announce.contains("?"))
+        buf.append('&');
+    else
+        buf.append('?');
+    buf.append("info_hash=").append(infoHash)
+       .append("&peer_id=").append(peerID)
+       .append("&port=").append(port)
+       .append("&ip=" ).append(_util.getOurIPString()).append(".i2p")
+       .append("&uploaded=").append(uploaded)
+       .append("&downloaded=").append(downloaded)
+       .append("&left=");
     // What do we send for left in magnet mode? Can we omit it?
-    long tleft = left >= 0 ? left : 1;
-    String s = tr.announce
-      + "?info_hash=" + infoHash
-      + "&peer_id=" + peerID
-      + "&port=" + port
-      + "&ip=" + _util.getOurIPString() + ".i2p"
-      + "&uploaded=" + uploaded
-      + "&downloaded=" + downloaded
-      + "&left=" + tleft
-      + "&compact=1"   // NOTE: opentracker will return 400 for &compact alone
-      + ((! event.equals(NO_EVENT)) ? ("&event=" + event) : "");
+    if (left >= 0)
+        buf.append(left);
+    else
+        buf.append('1');
+    buf.append("&compact=1");  // NOTE: opentracker will return 400 for &compact alone
+    if (! event.equals(NO_EVENT))
+        buf.append("&event=").append(event);
+    buf.append("&numwant=");
     if (left == 0 || event.equals(STOPPED_EVENT) || !coordinator.needPeers())
-        s += "&numwant=0";
+        buf.append('0');
     else
-        s += "&numwant=" + _util.getMaxConnections();
+        buf.append(_util.getMaxConnections());
+    String s = buf.toString();
     _util.debug("Sending TrackerClient request: " + s, Snark.INFO);
       
     tr.lastRequestTime = System.currentTimeMillis();
diff --git a/apps/i2psnark/java/src/org/klomp/snark/web/I2PSnarkServlet.java b/apps/i2psnark/java/src/org/klomp/snark/web/I2PSnarkServlet.java
index 7f3839ec943472a227612fe6bed573977a329500..24094515c5696cc322b3e8cf892cfc2e323b9320 100644
--- a/apps/i2psnark/java/src/org/klomp/snark/web/I2PSnarkServlet.java
+++ b/apps/i2psnark/java/src/org/klomp/snark/web/I2PSnarkServlet.java
@@ -102,7 +102,8 @@ public class I2PSnarkServlet extends Default {
     protected Resource getResource(String pathInContext) throws IOException
     {
         if (pathInContext == null || pathInContext.equals("/") || pathInContext.equals("/index.jsp") ||
-            pathInContext.equals("/index.html") || pathInContext.startsWith("/_icons/"))
+            pathInContext.equals("/index.html") || pathInContext.startsWith("/.icons/") ||
+            pathInContext.startsWith("/.js/") || pathInContext.startsWith("/.ajax/"))
             return super.getResource(pathInContext);
         // files in the i2psnark/ directory
         return _resourceBase.addPath(pathInContext);
@@ -151,6 +152,19 @@ public class I2PSnarkServlet extends Default {
         _imgPath = _themePath + "images/";
         // this is the part after /i2psnark
         String path = req.getServletPath();
+
+        // AJAX for mainsection
+        if ("/.ajax/xhr1.html".equals(path)) {
+            resp.setCharacterEncoding("UTF-8");
+            resp.setContentType("text/html; charset=UTF-8");
+            PrintWriter out = resp.getWriter();
+            //if (_log.shouldLog(Log.DEBUG))
+            //    _manager.addMessage((_context.clock().now() / 1000) + " xhr1 p=" + req.getParameter("p"));
+            writeMessages(out);
+            writeTorrents(out, req);
+            return;
+        }
+
         boolean isConfigure = "/configure".equals(path);
         // index.jsp doesn't work, it is grabbed by the war handler before here
         if (!(path == null || path.equals("/") || path.equals("/index.jsp") || path.equals("/index.html") || path.equals("/_post") || isConfigure)) {
@@ -192,7 +206,8 @@ public class I2PSnarkServlet extends Default {
         
         String peerParam = req.getParameter("p");
         String peerString;
-        if (peerParam == null || !_manager.util().connected()) {
+        if (peerParam == null || (!_manager.util().connected()) ||
+            peerParam.replaceAll("[a-zA-Z0-9~=-]", "").length() > 0) {  // XSS
             peerString = "";
         } else {
             peerString = "?p=" + peerParam;
@@ -208,13 +223,23 @@ public class I2PSnarkServlet extends Default {
         out.write("</title>\n");
                                          
         // we want it to go to the base URI so we don't refresh with some funky action= value
+        int delay = 0;
         if (!isConfigure) {
-            int delay = _manager.getRefreshDelaySeconds();
-            if (delay > 0)
-                out.write("<meta http-equiv=\"refresh\" content=\"" + delay + ";/i2psnark/" + peerString + "\">\n");
+            delay = _manager.getRefreshDelaySeconds();
+            if (delay > 0) {
+                //out.write("<meta http-equiv=\"refresh\" content=\"" + delay + ";/i2psnark/" + peerString + "\">\n");
+                out.write("<script src=\"/js/ajax.js\" type=\"text/javascript\"></script>\n" +
+                          "<script type=\"text/javascript\">\n"  +
+                          "function requestAjax1() { ajax(\"/i2psnark/.ajax/xhr1.html" + peerString + "\", \"mainsection\", " + (delay*1000) + "); }\n" +
+                          "function initAjax(delayMs) { setTimeout(requestAjax1, " + (delay*1000) +");  }\n"  +
+                          "</script>\n");
+            }
         }
-        out.write(HEADER_A + _themePath + HEADER_B);
-        out.write("</head><body>");
+        out.write(HEADER_A + _themePath + HEADER_B + "</head>\n");
+        if (isConfigure || delay <= 0)
+            out.write("<body>");
+        else
+            out.write("<body onload=\"initAjax()\">");
         out.write("<center>");
         if (isConfigure) {
             out.write("<div class=\"snarknavbar\"><a href=\"/i2psnark/\" title=\"");
@@ -249,34 +274,44 @@ public class I2PSnarkServlet extends Default {
         String newURL = req.getParameter("newURL");
         if (newURL != null && newURL.trim().length() > 0 && req.getMethod().equals("GET"))
             _manager.addMessage(_("Click \"Add torrent\" button to fetch torrent"));
-        out.write("<div class=\"page\"><div class=\"mainsection\"><div class=\"snarkMessages\"><table><tr><td align=\"left\"><pre>");
-        List msgs = _manager.getMessages();
-        for (int i = msgs.size()-1; i >= 0; i--) {
-            String msg = (String)msgs.get(i);
-            out.write(msg + "\n");
-        }
-        out.write("</pre></td></tr></table></div>");
+        out.write("<div class=\"page\"><div id=\"mainsection\" class=\"mainsection\">");
+
+        writeMessages(out);
 
         if (isConfigure) {
+            // end of mainsection div
             out.write("<div class=\"logshim\"></div></div>\n");
             writeConfigForm(out, req);
+            writeTrackerForm(out, req);
         } else {
             writeTorrents(out, req);
-            out.write("</div>\n");
+            // end of mainsection div
+            out.write("</div><div id=\"lowersection\">\n");
             writeAddForm(out, req);
             writeSeedForm(out, req);
             writeConfigLink(out);
+            // end of lowersection div
+            out.write("</div>\n");
         }
         out.write(FOOTER);
     }
 
+    private void writeMessages(PrintWriter out) throws IOException {
+        out.write("<div class=\"snarkMessages\"><table><tr><td align=\"left\"><pre>");
+        List msgs = _manager.getMessages();
+        for (int i = msgs.size()-1; i >= 0; i--) {
+            String msg = (String)msgs.get(i);
+            out.write(msg + "\n");
+        }
+        out.write("</pre></td></tr></table></div>");
+    }
+
     private void writeTorrents(PrintWriter out, HttpServletRequest req) throws IOException {
         /** dl, ul, down rate, up rate, peers, size */
         final long stats[] = {0,0,0,0,0,0};
         String peerParam = req.getParameter("p");
 
         List snarks = getSortedSnarks(req);
-        String uri = req.getRequestURI();
         boolean isForm = _manager.util().connected() || !snarks.isEmpty();
         if (isForm) {
             out.write("<form action=\"_post\" method=\"POST\">\n");
@@ -390,6 +425,7 @@ public class I2PSnarkServlet extends Default {
             out.write("&nbsp;");
         }
         out.write("</th></tr></thead>\n");
+        String uri = "/i2psnark/";
         for (int i = 0; i < snarks.size(); i++) {
             Snark snark = (Snark)snarks.get(i);
             boolean showDebug = "2".equals(peerParam);
@@ -650,14 +686,19 @@ public class I2PSnarkServlet extends Default {
             _manager.updateConfig(dataDir, filesPublic, autoStart, refreshDel, startupDel,
                                   seedPct, eepHost, eepPort, i2cpHost, i2cpPort, i2cpOpts,
                                   upLimit, upBW, useOpenTrackers, openTrackers, theme);
+        } else if ("Save2".equals(action)) {
+            String taction = req.getParameter("taction");
+            if (taction != null)
+                processTrackerForm(taction, req);
         } else if ("Create".equals(action)) {
             String baseData = req.getParameter("baseFile");
             if (baseData != null && baseData.trim().length() > 0) {
                 File baseFile = new File(_manager.getDataDir(), baseData);
                 String announceURL = req.getParameter("announceURL");
-                String announceURLOther = req.getParameter("announceURLOther");
-                if ( (announceURLOther != null) && (announceURLOther.trim().length() > "http://.i2p/announce".length()) )
-                    announceURL = announceURLOther;
+                // make the user add a tracker on the config form now
+                //String announceURLOther = req.getParameter("announceURLOther");
+                //if ( (announceURLOther != null) && (announceURLOther.trim().length() > "http://.i2p/announce".length()) )
+                //    announceURL = announceURLOther;
 
                 if (announceURL == null || announceURL.length() <= 0)
                     _manager.addMessage(_("Error creating torrent - you must select a tracker"));
@@ -668,7 +709,7 @@ public class I2PSnarkServlet extends Default {
                     try {
                         // This may take a long time to check the storage, but since it already exists,
                         // it shouldn't be THAT bad, so keep it in this thread.
-                        Storage s = new Storage(_manager.util(), baseFile, announceURL, null);
+                        Storage s = new Storage(_manager.util(), baseFile, announceURL, req.getParameter("private") != null, null);
                         s.close(); // close the files... maybe need a way to pass this Storage to addTorrent rather than starting over
                         MetaInfo info = s.getMetaInfo();
                         File torrentFile = new File(_manager.getDataDir(), s.getBaseName() + ".torrent");
@@ -713,6 +754,54 @@ public class I2PSnarkServlet extends Default {
             _manager.addMessage("Unknown POST action: \"" + action + '\"');
         }
     }
+
+    /** @since 0.9 */
+    private void processTrackerForm(String action, HttpServletRequest req) {
+        if (action.equals(_("Delete selected"))) {
+            boolean changed = false;
+            Map<String, String> trackers = _manager.getTrackers();
+            Enumeration e = req.getParameterNames();
+            while (e.hasMoreElements()) {
+                 Object o = e.nextElement();
+                 if (!(o instanceof String))
+                     continue;
+                 String k = (String) o;
+                 if (!k.startsWith("delete_"))
+                     continue;
+                 k = k.substring(7);
+                 if (trackers.remove(k) != null) {
+                    _manager.addMessage(_("Removed") + ": " + k);
+                    changed = true;
+                }
+            }
+            if (changed) {
+                _manager.saveTrackerMap();
+            }
+        } else if (action.equals(_("Add tracker"))) {
+            String name = req.getParameter("tname");
+            String hurl = req.getParameter("thurl");
+            String aurl = req.getParameter("taurl");
+            if (name != null && hurl != null && aurl != null) {
+                name = name.trim();
+                hurl = hurl.trim();
+                aurl = aurl.trim().replace("=", "&#61;");
+                if (name.length() > 0 && hurl.startsWith("http://") && aurl.startsWith("http://")) {
+                    Map<String, String> trackers = _manager.getTrackers();
+                    trackers.put(name, aurl + '=' + hurl);
+                    _manager.saveTrackerMap();
+                } else {
+                    _manager.addMessage(_("Enter valid tracker name and URLs"));
+                }
+            } else {
+                _manager.addMessage(_("Enter valid tracker name and URLs"));
+            }
+        } else if (action.equals(_("Restore defaults"))) {
+            _manager.setDefaultTrackerMap();
+            _manager.addMessage(_("Restored default trackers"));
+        } else {
+            _manager.addMessage("Unknown POST action: \"" + action + '\"');
+        }
+    }
     
     private static final String iopts[] = {"inbound.length", "inbound.quantity",
                                            "outbound.length", "outbound.quantity" };
@@ -1222,7 +1311,7 @@ public class I2PSnarkServlet extends Default {
         out.write("\"> \n");
         // not supporting from file at the moment, since the file name passed isn't always absolute (so it may not resolve)
         //out.write("From file: <input type=\"file\" name=\"newFile\" size=\"50\" value=\"" + newFile + "\" /><br>");
-        out.write("<input type=\"submit\" value=\"");
+        out.write("<input type=\"submit\" class=\"add\" value=\"");
         out.write(_("Add torrent"));
         out.write("\" name=\"foo\" ><br>\n");
         out.write("<tr><td>&nbsp;<td><span class=\"snarkAddInfo\">");
@@ -1269,31 +1358,40 @@ public class I2PSnarkServlet extends Default {
         //out.write(_("Open trackers and DHT only"));
         out.write(_("Open trackers only"));
         out.write("</option>\n");
-        Map trackers = _manager.getTrackers();
-        for (Iterator iter = trackers.entrySet().iterator(); iter.hasNext(); ) {
-            Map.Entry entry = (Map.Entry)iter.next();
-            String name = (String)entry.getKey();
-            String announceURL = (String)entry.getValue();
+        Map<String, String> trackers = _manager.getTrackers();
+        for (Map.Entry<String, String> entry : trackers.entrySet()) {
+            String name = entry.getKey();
+            String announceURL = entry.getValue();
             int e = announceURL.indexOf('=');
             if (e > 0)
-                announceURL = announceURL.substring(0, e);
+                announceURL = announceURL.substring(0, e).replace("&#61;", "=");
             if (announceURL.equals(_lastAnnounceURL))
                 announceURL += "\" selected=\"selected";
             out.write("\t<option value=\"" + announceURL + "\">" + name + "</option>\n");
         }
         out.write("</select>\n");
-        out.write(_("or"));
-        out.write("&nbsp;<input type=\"text\" name=\"announceURLOther\" size=\"57\" value=\"http://\" " +
-                  "title=\"");
-        out.write(_("Specify custom tracker announce URL"));
-        out.write("\" > " +
-                  "<input type=\"submit\" value=\"");
+        // make the user add a tracker on the config form now
+        //out.write(_("or"));
+        //out.write("&nbsp;<input type=\"text\" name=\"announceURLOther\" size=\"57\" value=\"http://\" " +
+        //          "title=\"");
+        //out.write(_("Specify custom tracker announce URL"));
+        //out.write("\" > " +
+        out.write(" <input type=\"submit\" class=\"create\" value=\"");
         out.write(_("Create torrent"));
-        out.write("\" name=\"foo\" ></table>\n" +
+        out.write("\" name=\"foo\" >\n" +
+                  "</td></tr><tr><td>");
+        out.write(_("Private?"));
+        out.write(" </td><td> <input type=\"checkbox\" class=\"optbox\" name=\"private\" value=\"true\" title=\"");
+            out.write(_("Use for private trackers"));
+        out.write("\"");
+        if (req.getParameter("private") != null)
+            out.write(" checked");
+        out.write("></td></tr>" +
+                  "</table>\n" +
                   "</form></div></div>");        
     }
     
-    private static final int[] times = { 30, 60, 2*60, 5*60, 10*60, 30*60, -1 };
+    private static final int[] times = { 5, 15, 30, 60, 2*60, 5*60, 10*60, 30*60, -1 };
 
     private void writeConfigForm(PrintWriter out, HttpServletRequest req) throws IOException {
         String dataDir = _manager.getDataDir().getAbsolutePath();
@@ -1356,7 +1454,7 @@ public class I2PSnarkServlet extends Default {
             out.write(Integer.toString(times[i]));
             out.write("\"");
             if (times[i] == delay)
-                out.write(" selected=\"true\"");
+                out.write(" selected=\"selected\"");
             out.write(">");
             if (times[i] > 0)
                 out.write(DataHelper.formatDuration2(times[i] * 1000));
@@ -1379,15 +1477,15 @@ public class I2PSnarkServlet extends Default {
 /*
         out.write("Seed percentage: <select name=\"seedPct\" disabled=\"true\" >\n\t");
         if (seedPct <= 0)
-            out.write("<option value=\"0\" selected=\"true\">Unlimited</option>\n\t");
+            out.write("<option value=\"0\" selected=\"selected\">Unlimited</option>\n\t");
         else
             out.write("<option value=\"0\">Unlimited</option>\n\t");
         if (seedPct == 100)
-            out.write("<option value=\"100\" selected=\"true\">100%</option>\n\t");
+            out.write("<option value=\"100\" selected=\"selected\">100%</option>\n\t");
         else
             out.write("<option value=\"100\">100%</option>\n\t");
         if (seedPct == 150)
-            out.write("<option value=\"150\" selected=\"true\">150%</option>\n\t");
+            out.write("<option value=\"150\" selected=\"selected\">150%</option>\n\t");
         else
             out.write("<option value=\"150\">150%</option>\n\t");
         out.write("</select><br>\n");
@@ -1473,6 +1571,57 @@ public class I2PSnarkServlet extends Default {
                   "</table></div></div></form>");
     }
     
+    /** @since 0.9 */
+    private void writeTrackerForm(PrintWriter out, HttpServletRequest req) throws IOException {
+        StringBuilder buf = new StringBuilder(1024);
+        buf.append("<form action=\"/i2psnark/configure\" method=\"POST\">\n" +
+                  "<div class=\"configsectionpanel\"><div class=\"snarkConfig\">\n" +
+                  "<input type=\"hidden\" name=\"nonce\" value=\"" + _nonce + "\" >\n" +
+                  "<input type=\"hidden\" name=\"action\" value=\"Save2\" >\n" +
+                  "<span class=\"snarkConfigTitle\">" +
+                  "<img alt=\"\" border=\"0\" src=\"" + _imgPath + "config.png\"> ");
+        buf.append(_("Trackers"));
+        buf.append("</span><hr>\n"   +
+                   "<table><tr><th>")
+           //.append(_("Remove"))
+           .append("</th><th>")
+           .append(_("Name"))
+           .append("</th><th>")
+           .append(_("Website URL"))
+           .append("</th><th>")
+           .append(_("Announce URL"))
+           .append("</th></tr>\n");
+        Map<String, String> trackers = _manager.getTrackers();
+        for (Map.Entry<String, String> entry : trackers.entrySet()) {
+            String name = entry.getKey();
+            String announceURL = entry.getValue();
+            int e = announceURL.indexOf('=');
+            if (e <= 0)
+                continue;
+            String homeURL = announceURL.substring(e + 1);
+            announceURL = announceURL.substring(0, e).replace("&#61;", "=");
+            buf.append("<tr><td align=\"center\"><input type=\"checkbox\" class=\"optbox\" name=\"delete_")
+               .append(name).append("\">" +
+                       "</td><td align=\"left\">").append(name)
+               .append("</td><td align=\"left\">").append(urlify(homeURL, 35))
+               .append("</td><td align=\"left\">").append(urlify(announceURL, 35))
+               .append("</td></tr>\n");
+        }
+        buf.append("<tr><td align=\"center\"><b>")
+           .append(_("Add")).append(":</b></td>" +
+                   "<td align=\"left\"><input type=\"text\" size=\"16\" name=\"tname\"></td>" +
+                   "<td align=\"left\"><input type=\"text\" size=\"40\" name=\"thurl\"></td>" +
+                   "<td align=\"left\"><input type=\"text\" size=\"40\" name=\"taurl\"></td></tr>\n" +
+                   "<tr><td colspan=\"2\"></td><td colspan=\"2\" align=\"left\">\n" +
+                   "<input type=\"submit\" name=\"taction\" class=\"default\" value=\"").append(_("Add tracker")).append("\">\n" +
+                   "<input type=\"submit\" name=\"taction\" class=\"delete\" value=\"").append(_("Delete selected")).append("\">\n" +
+                   // "<input type=\"reset\" class=\"cancel\" value=\"").append(_("Cancel")).append("\">\n" +
+                   "<input type=\"submit\" name=\"taction\" class=\"reload\" value=\"").append(_("Restore defaults")).append("\">\n" +
+                   "<input type=\"submit\" name=\"taction\" class=\"add\" value=\"").append(_("Add tracker")).append("\">\n" +
+                   "</td></tr></table></div></div></form>\n");
+        out.write(buf.toString());
+    }
+
     private void writeConfigLink(PrintWriter out) throws IOException {
         out.write("<div class=\"configsection\"><span class=\"snarkConfig\">\n" +
                   "<span class=\"snarkConfigTitle\"><a href=\"configure\">" +
@@ -1574,7 +1723,7 @@ public class I2PSnarkServlet extends Default {
         for (int i = min; i <= max; i++) {
             buf.append("<option value=\"").append(i).append("\" ");
             if (i == now)
-                buf.append("selected=\"true\" ");
+                buf.append("selected=\"selected\" ");
             // constants to prevent tagging
             buf.append(">").append(ngettext(DUMMY1 + name, DUMMY0 + name + 's', i));
             buf.append("</option>\n");
@@ -1622,10 +1771,20 @@ public class I2PSnarkServlet extends Default {
     
     /** @since 0.7.14 */
     private static String urlify(String s) {
+        return urlify(s, 100);
+    }
+    
+    /** @since 0.9 */
+    private static String urlify(String s, int max) {
         StringBuilder buf = new StringBuilder(256);
         // browsers seem to work without doing this but let's be strict
         String link = urlEncode(s);
-        buf.append("<a href=\"").append(link).append("\">").append(link).append("</a>");
+        String display;
+        if (s.length() <= max)
+            display = link;
+        else
+            display = urlEncode(s.substring(0, max)) + "&hellip;";
+        buf.append("<a href=\"").append(link).append("\">").append(display).append("</a>");
         return buf.toString();
     }
     
@@ -1991,12 +2150,12 @@ public class I2PSnarkServlet extends Default {
     
     /** @since 0.7.14 */
     private static String toImg(String icon) {
-        return "<img alt=\"\" height=\"16\" width=\"16\" src=\"/i2psnark/_icons/" + icon + ".png\">";
+        return "<img alt=\"\" height=\"16\" width=\"16\" src=\"/i2psnark/.icons/" + icon + ".png\">";
     }
 
     /** @since 0.8.2 */
     private static String toImg(String icon, String altText) {
-        return "<img alt=\"" + altText + "\" height=\"16\" width=\"16\" src=\"/i2psnark/_icons/" + icon + ".png\">";
+        return "<img alt=\"" + altText + "\" height=\"16\" width=\"16\" src=\"/i2psnark/.icons/" + icon + ".png\">";
     }
 
     /** @since 0.8.1 */
diff --git a/apps/i2ptunnel/java/build.xml b/apps/i2ptunnel/java/build.xml
index 1b62279145de25669f48b9c6356ee25069488ba1..df8387b6d2d9d0bda087c6559fc7ffc4c3bfc624 100644
--- a/apps/i2ptunnel/java/build.xml
+++ b/apps/i2ptunnel/java/build.xml
@@ -63,6 +63,7 @@
                 <attribute name="Main-Class" value="net.i2p.i2ptunnel.I2PTunnel" />
                 <attribute name="Class-Path" value="i2p.jar mstreaming.jar" />
                 <attribute name="Implementation-Version" value="${full.version}" />
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.j.tr}" />
@@ -136,6 +137,7 @@
              basedir="../jsp/" excludes="web.xml, web-fragment.xml, web-out.xml, **/*.java, *.jsp">
             <manifest>
                 <attribute name="Implementation-Version" value="${full.version}" />
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.w.tr}" />
diff --git a/apps/i2ptunnel/java/src/net/i2p/i2ptunnel/I2PTunnelHTTPClient.java b/apps/i2ptunnel/java/src/net/i2p/i2ptunnel/I2PTunnelHTTPClient.java
index b3bce9cbcdaa8da754503cc4635bba8adc14576d..ba47c5c37a14008a53945e83bbb9a01c00017662 100644
--- a/apps/i2ptunnel/java/src/net/i2p/i2ptunnel/I2PTunnelHTTPClient.java
+++ b/apps/i2ptunnel/java/src/net/i2p/i2ptunnel/I2PTunnelHTTPClient.java
@@ -13,6 +13,8 @@ import java.io.InputStreamReader;
 import java.io.OutputStream;
 import java.net.Socket;
 import java.net.SocketException;
+import java.net.URI;
+import java.net.URISyntaxException;
 import java.util.ArrayList;
 import java.util.Date;
 import java.util.HashMap;
@@ -36,8 +38,8 @@ import net.i2p.data.DataFormatException;
 import net.i2p.data.DataHelper;
 import net.i2p.data.Destination;
 import net.i2p.data.Hash;
+import net.i2p.i2ptunnel.localServer.LocalHTTPServer;
 import net.i2p.util.EventDispatcher;
-import net.i2p.util.FileUtil;
 import net.i2p.util.Log;
 import net.i2p.util.PortMapper;
 import net.i2p.util.Translate;
@@ -61,6 +63,10 @@ import net.i2p.util.Translate;
  * in browsers or other user-visible applications, as relative links will not
  * resolve correctly, cookies won't work, etc.
  *
+ * Note that http://$b64key/... and http://$b64key.i2p/... are NOT supported, as
+ * a b64 key may contain '=' and '~', both of which are illegal host name characters.
+ * Rewrite as http://i2p/$b64key/...
+ *
  * If the $site resolves with the I2P naming service, then it is directed towards
  * that eepsite, otherwise it is directed towards this client's outproxy (typically
  * "squid.i2p").  Only HTTP is supported (no HTTPS, ftp, mailto, etc).  Both GET
@@ -102,7 +108,7 @@ public class I2PTunnelHTTPClient extends I2PTunnelHTTPClientBase implements Runn
          "That I2P Destination was not found. Perhaps you pasted in the "+
          "wrong BASE64 I2P Destination or the link you are following is "+
          "bad. The host (or the WWW proxy, if you're using one) could also "+
-     "be temporarily offline.  You may want to <b>retry</b>.  "+
+         "be temporarily offline.  You may want to <b>retry</b>.  "+
          "Could not find the following Destination:<BR><BR><div>")
         .getBytes();
 
@@ -309,7 +315,8 @@ public class I2PTunnelHTTPClient extends I2PTunnelHTTPClientBase implements Runn
         return rv;
     }
 
-    private static final String LOCAL_SERVER = "proxy.i2p";
+    private static final String HELPER_PARAM = "i2paddresshelper";
+    public static final String LOCAL_SERVER = "proxy.i2p";
     private static final boolean DEFAULT_GZIP = true;
     /** all default to false */
     public static final String PROP_REFERER = "i2ptunnel.httpclient.sendReferer";
@@ -321,11 +328,19 @@ public class I2PTunnelHTTPClient extends I2PTunnelHTTPClientBase implements Runn
     protected void clientConnectionRun(Socket s) {
         InputStream in = null;
         OutputStream out = null;
+
+        /**
+         * The URL after fixup, always starting with http://
+         */
         String targetRequest = null;
+
         boolean usingWWWProxy = false;
         boolean usingInternalServer = false;
+        String internalPath = null;
+        String internalRawQuery = null;
         String currentProxy = null;
         long requestId = ++__requestId;
+
         try {
             out = s.getOutputStream();
             InputReader reader = new InputReader(s.getInputStream());
@@ -351,79 +366,84 @@ public class I2PTunnelHTTPClient extends I2PTunnelHTTPClientBase implements Runn
                     if (_log.shouldLog(Log.DEBUG))
                         _log.debug(getPrefix(requestId) + "First line [" + line + "]");
 
-                    int pos = line.indexOf(" ");
-                    if (pos == -1) break;
-                    method = line.substring(0, pos);
-                    // TODO use Java URL class to make all this simpler and more robust
-                    // That will also fix IPV6 [a:b:c]
-                    String request = line.substring(pos + 1);
+                    String[] params = line.split(" ", 3);
+                    if (params.length != 3)
+                        break;
+                    String request = params[1];
+
+                    // various obscure fixups
                     if (request.startsWith("/") && getTunnel().getClientOptions().getProperty("i2ptunnel.noproxy") != null) {
                         // what is this for ???
                         request = "http://i2p" + request;
                     } else if (request.startsWith("/eepproxy/")) {
-                        // /eepproxy/foo.i2p/bar/baz.html HTTP/1.0
+                        // Deprecated
+                        // /eepproxy/foo.i2p/bar/baz.html
                         String subRequest = request.substring("/eepproxy/".length());
-                        int protopos = subRequest.indexOf(" ");
-                        String uri = subRequest.substring(0, protopos);
-                        if (uri.indexOf("/") == -1) {
-                                uri = uri + "/";
-                        }
-                        // "http://" + "foo.i2p/bar/baz.html" + " HTTP/1.0"
-                        request = "http://" + uri + subRequest.substring(protopos);
+                        if (subRequest.indexOf("/") == -1)
+                                subRequest += "/";
+                        request = "http://" + subRequest;
+                 /****
                     } else if (request.toLowerCase(Locale.US).startsWith("http://i2p/")) {
-                        // http://i2p/b64key/bar/baz.html HTTP/1.0
+                        // http://i2p/b64key/bar/baz.html
+                        // we can't do this now by setting the URI host to the b64key, as
+                        // it probably contains '=' and '~' which are illegal,
+                        // and a host may not include escaped octets
+                        // This will get undone below.
                         String subRequest = request.substring("http://i2p/".length());
-                        int protopos = subRequest.indexOf(" ");
-                        String uri = subRequest.substring(0, protopos);
-                        if (uri.indexOf("/") == -1) {
-                                uri = uri + "/";
-                        }
-                        // "http://" + "b64key/bar/baz.html" + " HTTP/1.0"
-                        request = "http://" + uri + subRequest.substring(protopos);
+                        if (subRequest.indexOf("/") == -1)
+                                subRequest += "/";
+                         "http://" + "b64key/bar/baz.html"
+                        request = "http://" + subRequest;
+                    } else if (request.toLowerCase(Locale.US).startsWith("http://")) {
+                        // Unsupported
+                        // http://$b64key/...
+                        // This probably used to work, rewrite it so that
+                        // we can create a URI without illegal characters
+                        // This will get undone below.
+                        String  oldPath = request.substring(7);
+                        int slash = oldPath.indexOf("/");
+                        if (slash < 0)
+                            slash = oldPath.length();
+                        if (slash >= 516 && !oldPath.substring(0, slash).contains("."))
+                            request = "http://i2p/" + oldPath;
+                   ****/
                     }
 
-                    pos = request.indexOf("//");
-                    if (pos == -1) {
-                        method = null;
+                    // Now use the Java URI parser
+                    // This will be the incoming URI but will then get modified
+                    // to be the outgoing URI (with http:// if going to outproxy, otherwise without)
+                    URI requestURI;
+                    try {
+                        requestURI = new URI(request);
+                        if (requestURI.getRawUserInfo() != null || requestURI.getRawFragment() != null) {
+                            // these should never be sent to the proxy in the request line
+                            if (_log.shouldLog(Log.WARN))
+                                _log.warn(getPrefix(requestId) + "Removing userinfo or fragment [" + request + "]");
+                            requestURI = changeURI(requestURI, null, 0, null);
+                        }
+                        if (requestURI.getPath() == null || requestURI.getPath().length() <= 0) {
+                            // Add a path
+                            if (_log.shouldLog(Log.WARN))
+                                _log.warn(getPrefix(requestId) + "Adding / path to [" + request + "]");
+                            requestURI = changeURI(requestURI, null, 0, "/");
+                        }
+                    } catch (URISyntaxException use) {
+                        if (_log.shouldLog(Log.WARN))
+                            _log.warn(getPrefix(requestId) + "Bad request [" + request + "]", use);
                         break;
                     }
-                    protocol = request.substring(0, pos + 2);
-                    request = request.substring(pos + 2);
-
-                    // "foo.i2p/bar/baz HTTP/1.1", with any i2paddresshelper parameter removed
-                    targetRequest = request;
+                    method = params[0];
+                    String protocolVersion = params[2];
 
-                    // pos is the start of the path
-                    pos = request.indexOf("/");
-                    if (pos == -1) {
-                        //pos = request.length();
+                    protocol = requestURI.getScheme();
+                    host = requestURI.getHost();
+                    if (protocol == null || host == null) {
+                        _log.warn("Null protocol or host: " + request);
                         method = null;
                         break;
                     }
-                    host = request.substring(0, pos);
 
-                    // parse port
-                    int posPort = host.indexOf(":");
-                    int port = 80;
-                    if(posPort != -1) {
-                        String[] parts = host.split(":");
-                        try {
-                        host = parts[0];
-                        } catch (ArrayIndexOutOfBoundsException ex) {
-                        if (out != null) {
-                            out.write(getErrorPage("denied", ERR_REQUEST_DENIED));
-                            writeFooter(out);
-                        }
-                        s.close();
-                        return;
-
-                        }
-                        try {
-                            port = Integer.parseInt(parts[1]);
-                        } catch(Exception exc) {
-                            // TODO: log this
-                        }
-                    }
+                    int port = requestURI.getPort();
 
                     // Go through the various types of host names, set
                     // the host and destination variables accordingly,
@@ -433,115 +453,142 @@ public class I2PTunnelHTTPClient extends I2PTunnelHTTPClientBase implements Runn
                     // in our addressbook (all naming is local),
                     // and it is removed from the request line.
 
-                    if (host.length() >= 516 && host.indexOf(".") < 0) {
-                        // http://b64key/bar/baz.html
-                        destination = host;
-                        host = getHostName(destination);
-                        line = method + ' ' + request.substring(pos);
-                    } else if (host.toLowerCase(Locale.US).equals(LOCAL_SERVER)) {
+                    String hostLowerCase = host.toLowerCase(Locale.US);
+                    if (hostLowerCase.equals(LOCAL_SERVER)) {
                         // so we don't do any naming service lookups
                         destination = host;
                         usingInternalServer = true;
-                    } else if (host.toLowerCase(Locale.US).endsWith(".i2p")) {
+                        internalPath = requestURI.getPath();
+                        internalRawQuery = requestURI.getRawQuery();
+                    } else if (hostLowerCase.equals("i2p")) {
+                        // pull the b64 dest out of the first path element
+                        String oldPath = requestURI.getPath().substring(1);
+                        int slash = oldPath.indexOf("/");
+                        if (slash < 0) {
+                            slash = oldPath.length();
+                            oldPath += "/";
+                        }
+                        String dest = oldPath.substring(0, slash);
+                        if (slash >= 516 && !dest.contains(".")) {
+                            // possible alternative:
+                            // redirect to b32
+                            destination = dest;
+                            host = getHostName(destination);
+                            targetRequest = requestURI.toASCIIString();
+                            String newURI = oldPath.substring(slash);
+                            String query = requestURI.getRawQuery();
+                            if (query != null)
+                                newURI += '?' + query;
+                            try {
+                                requestURI = new URI(newURI);
+                            } catch (URISyntaxException use) {
+                                // shouldnt happen
+                                _log.warn(request, use);
+                                method = null;
+                                break;
+                            }
+                        } else {
+                            _log.warn("Bad http://i2p/b64dest " + request);
+                            host = null;
+                            break;
+                        }
+                    } else if (hostLowerCase.endsWith(".i2p")) {
                         // Destination gets the host name
                         destination = host;
                         // Host becomes the destination's "{b32}.b32.i2p" string, or "i2p" on lookup failure
                         host = getHostName(destination);
 
-                        int pos2;
-                        if ((pos2 = request.indexOf("?")) != -1) {
-                            // Try to find an address helper in the fragments
-                            // and split the request into it's component parts for rebuilding later
+                        if (requestURI.getPort() >= 0) {
+                            // TODO support I2P ports someday
+                            //if (port >= 0)
+                            //    host = host + ':' + port;
+                            if (_log.shouldLog(Log.WARN))
+                                _log.warn(getPrefix(requestId) + "Removing port from [" + request + "]");
+                            try {
+                                requestURI = changeURI(requestURI, null, -1, null);
+                            } catch (URISyntaxException use) {
+                                _log.warn(request, use);
+                                method = null;
+                                break;
+                            }
+                        }
+
+                        String query = requestURI.getRawQuery();
+                        if (query != null) {
                             boolean ahelperConflict = false;
 
-                            String fragments = request.substring(pos2 + 1);
-                            String uriPath = request.substring(0, pos2);
-                            pos2 = fragments.indexOf(" ");
-                            String protocolVersion = fragments.substring(pos2 + 1);
-                            String urlEncoding = "";
-                            fragments = fragments.substring(0, pos2);
-                            String initialFragments = fragments;
-                            // FIXME split on ';' also
-                            fragments = fragments + "&";
-                            String fragment;
-                            while(fragments.length() > 0) {
-                                pos2 = fragments.indexOf("&");
-                                fragment = fragments.substring(0, pos2);
-                                fragments = fragments.substring(pos2 + 1);
-
-                                // Fragment looks like addresshelper key
-                                if (fragment.startsWith("i2paddresshelper=") &&
-                                    !Boolean.valueOf(getTunnel().getClientOptions().getProperty(PROP_DISABLE_HELPER)).booleanValue()) {
-                                    pos2 = fragment.indexOf("=");
-                                    ahelperKey = fragment.substring(pos2 + 1);
-                                    // Key contains data, lets not ignore it
-                                    if (ahelperKey != null) {
-                                        if(ahelperKey.endsWith(".i2p")) {
-                                            // allow i2paddresshelper=<b32>.b32.i2p syntax.
-                                            /*
-                                              also i2paddresshelper=name.i2p for aliases
-                                              i.e. on your eepsite put 
-                                              <a href="?i2paddresshelper=name.i2p">This is the name I want to be called.</a>
-                                            */
-                                            Destination dest = _context.namingService().lookup(ahelperKey);
-                                            if(dest==null) {
-                                                if (_log.shouldLog(Log.WARN))
-                                                    _log.warn(getPrefix(requestId) + "Could not find destination for "+ahelperKey);
-                                                byte[] header = getErrorPage("ahelper-notfound", ERR_AHELPER_NOTFOUND);
-                                                out.write(header);
-                                                out.write(("<p>" + _("This seems to be a bad destination:") + " " + ahelperKey + " " + _("i2paddresshelper cannot help you with a destination like that!") + "</p>").getBytes("UTF-8"));
-                                                writeFooter(out);
-                                                // XXX: should closeSocket(s) be in a finally block?
-                                                closeSocket(s);
-                                                return;
-                                            }
-                                            ahelperKey = dest.toBase64();
-                                        } 
-
-                                        ahelperPresent = true;
-                                        // ahelperKey will be validated later
-                                        if (host == null || "i2p".equals(host)) {
-                                            // Host lookup failed - resolvable only with addresshelper
-                                            // Store in local HashMap unless there is conflict
-                                            String old = addressHelpers.putIfAbsent(destination.toLowerCase(Locale.US), ahelperKey);
-                                            ahelperNew = old == null;
-                                            if ((!ahelperNew) && !old.equals(ahelperKey)) {
+                            // Try to find an address helper in the query
+                            String[] helperStrings = removeHelper(query);
+                            if (helperStrings != null &&
+                                !Boolean.valueOf(getTunnel().getClientOptions().getProperty(PROP_DISABLE_HELPER)).booleanValue()) {
+                                query = helperStrings[0];
+                                if (query.equals(""))
+                                    query = null;
+                                try {
+                                    requestURI = replaceQuery(requestURI, query);
+                                } catch (URISyntaxException use) {
+                                    // shouldn't happen
+                                    _log.warn(request, use);
+                                    method = null;
+                                    break;
+                                }
+                                ahelperKey = helperStrings[1];
+                                // Key contains data, lets not ignore it
+                                if (ahelperKey.length() > 0) {
+                                    if(ahelperKey.endsWith(".i2p")) {
+                                        // allow i2paddresshelper=<b32>.b32.i2p syntax.
+                                        /*
+                                          also i2paddresshelper=name.i2p for aliases
+                                          i.e. on your eepsite put 
+                                          <a href="?i2paddresshelper=name.i2p">This is the name I want to be called.</a>
+                                        */
+                                        Destination dest = _context.namingService().lookup(ahelperKey);
+                                        if(dest==null) {
+                                            if (_log.shouldLog(Log.WARN))
+                                                _log.warn(getPrefix(requestId) + "Could not find destination for "+ahelperKey);
+                                            byte[] header = getErrorPage("ahelper-notfound", ERR_AHELPER_NOTFOUND);
+                                            out.write(header);
+                                            out.write(("<p>" + _("This seems to be a bad destination:") + " " + ahelperKey + " " + _("i2paddresshelper cannot help you with a destination like that!") + "</p>").getBytes("UTF-8"));
+                                            writeFooter(out);
+                                            // XXX: should closeSocket(s) be in a finally block?
+                                            closeSocket(s);
+                                            return;
+                                        }
+                                        ahelperKey = dest.toBase64();
+                                    } 
+
+                                    ahelperPresent = true;
+                                    // ahelperKey will be validated later
+                                    if (host == null || "i2p".equals(host)) {
+                                        // Host lookup failed - resolvable only with addresshelper
+                                        // Store in local HashMap unless there is conflict
+                                        String old = addressHelpers.putIfAbsent(destination.toLowerCase(Locale.US), ahelperKey);
+                                        ahelperNew = old == null;
+                                        if ((!ahelperNew) && !old.equals(ahelperKey)) {
+                                            // Conflict: handle when URL reconstruction done
+                                            ahelperConflict = true;
+                                            if (_log.shouldLog(Log.WARN))
+                                                _log.warn(getPrefix(requestId) + "Addresshelper key conflict for site [" + destination +
+                                                          "], trusted key [" + old + "], specified key [" + ahelperKey + "].");
+                                        }
+                                    } else {
+                                        // If the host is resolvable from database, verify addresshelper key
+                                        // Silently bypass correct keys, otherwise alert
+                                        Destination hostDest = _context.namingService().lookup(destination);
+                                        if (hostDest != null) {
+                                            String destB64 = hostDest.toBase64();
+                                            if (destB64 != null && !destB64.equals(ahelperKey)) {
                                                 // Conflict: handle when URL reconstruction done
                                                 ahelperConflict = true;
                                                 if (_log.shouldLog(Log.WARN))
                                                     _log.warn(getPrefix(requestId) + "Addresshelper key conflict for site [" + destination +
-                                                              "], trusted key [" + old + "], specified key [" + ahelperKey + "].");
-                                            }
-                                        } else {
-                                            // If the host is resolvable from database, verify addresshelper key
-                                            // Silently bypass correct keys, otherwise alert
-                                            Destination hostDest = _context.namingService().lookup(destination);
-                                            if (hostDest != null) {
-                                                String destB64 = hostDest.toBase64();
-                                                if (destB64 != null && !destB64.equals(ahelperKey)) {
-                                                    // Conflict: handle when URL reconstruction done
-                                                    ahelperConflict = true;
-                                                    if (_log.shouldLog(Log.WARN))
-                                                        _log.warn(getPrefix(requestId) + "Addresshelper key conflict for site [" + destination +
-                                                                  "], trusted key [" + destB64 + "], specified key [" + ahelperKey + "].");
-                                                    
-                                                }
+                                                              "], trusted key [" + destB64 + "], specified key [" + ahelperKey + "].");
+                                                
                                             }
                                         }
-                                    } // ahelperKey
-                                } else {
-                                    // Other fragments, just pass along
-                                    // Append each fragment to urlEncoding
-                                    if ("".equals(urlEncoding)) {
-                                        urlEncoding = "?" + fragment;
-                                    } else {
-                                        urlEncoding = urlEncoding + "&" + fragment;
                                     }
-                                }
-                            }
-                            // Reconstruct the request minus the i2paddresshelper GET var
-                            request = uriPath + urlEncoding + " " + protocolVersion;
-                            targetRequest = request;
+                                } // ahelperKey
+                            } // helperstrings
 
                             // Did addresshelper key conflict?
                             if (ahelperConflict) {
@@ -553,9 +600,17 @@ public class I2PTunnelHTTPClient extends I2PTunnelHTTPClientBase implements Runn
                                         byte[] header = getErrorPage("dnfb", ERR_DESTINATION_UNKNOWN);
                                         writeErrorMessage(header, out, targetRequest, false, destination, null);
                                     } else {
-                                        String trustedURL = protocol + uriPath + urlEncoding;
-                                        // Fixme - any path is lost
-                                        String conflictURL = protocol + alias + '/' + urlEncoding;
+                                        String trustedURL = requestURI.toASCIIString();
+                                        URI conflictURI;
+                                        try {
+                                            conflictURI = changeURI(requestURI, alias, 0, null);
+                                        } catch (URISyntaxException use) {
+                                            // shouldn't happen
+                                            _log.warn(request, use);
+                                            method = null;
+                                            break;
+                                        }
+                                        String conflictURL = conflictURI.toASCIIString();
                                         byte[] header = getErrorPage("ahelper-conflict", ERR_AHELPER_CONFLICT);
                                         out.write(header);
                                         out.write(_("To visit the destination in your host database, click <a href=\"{0}\">here</a>. To visit the conflicting addresshelper destination, click <a href=\"{1}\">here</a>.", trustedURL, conflictURL).getBytes("UTF-8"));
@@ -572,11 +627,24 @@ public class I2PTunnelHTTPClient extends I2PTunnelHTTPClientBase implements Runn
                         if (addressHelper != null)
                             host = getHostName(addressHelper);
 
-                        line = method + " " + request.substring(pos);
+                        // now strip everything but path and query from URI
+                        targetRequest = requestURI.toASCIIString();
+                        String newURI = requestURI.getRawPath();
+                        if (query != null)
+                            newURI += '?' + query;
+                        try {
+                            requestURI = new URI(newURI);
+                        } catch (URISyntaxException use) {
+                            // shouldnt happen
+                            _log.warn(request, use);
+                            method = null;
+                            break;
+                        }
+
                         // end of (host endsWith(".i2p"))
 
-                    } else if (host.toLowerCase(Locale.US).equals("localhost") || host.equals("127.0.0.1") ||
-                               host.startsWith("192.168.")) {
+                    } else if (hostLowerCase.equals("localhost") || host.equals("127.0.0.1") ||
+                               host.startsWith("192.168.") || host.equals("[::1]")) {
                         // if somebody is trying to get to 192.168.example.com, oh well
                         if (out != null) {
                             out.write(getErrorPage("localhost", ERR_LOCALHOST));
@@ -584,9 +652,9 @@ public class I2PTunnelHTTPClient extends I2PTunnelHTTPClientBase implements Runn
                         }
                         s.close();
                         return;
-                    } else if (host.indexOf(".") != -1) {
-                        // rebuild host
-                        host = host + ":" + port;
+                    } else if (host.contains(".") || host.startsWith("[")) {
+                        if (port >= 0)
+                            host = host + ':' + port;
                         // The request must be forwarded to a WWW proxy
                         if (_log.shouldLog(Log.DEBUG))
                             _log.debug("Before selecting outproxy for " + host);
@@ -606,36 +674,23 @@ public class I2PTunnelHTTPClient extends I2PTunnelHTTPClientBase implements Runn
                         }
                         destination = currentProxy;
                         usingWWWProxy = true;
+                        targetRequest = requestURI.toASCIIString();
                         if (_log.shouldLog(Log.DEBUG))
-                            _log.debug(getPrefix(requestId) + "Host doesnt end with .i2p and it contains a period [" + host + "]: wwwProxy!");
+                            _log.debug(getPrefix(requestId) +  " [" + host + "]: wwwProxy!");
                     } else {
                         // what is left for here? a hostname with no dots, and != "i2p"
                         // and not a destination ???
                         // Perhaps something in privatehosts.txt ...
-                        request = request.substring(pos + 1);
-                        pos = request.indexOf("/");
-                        if (pos < 0) {
-                            l.log("Invalid request url [" + request + "]");
-                            if (out != null) {
-                                out.write(getErrorPage("denied", ERR_REQUEST_DENIED));
-                                writeFooter(out);
-                            }
-                            s.close();
-                            return;
-                        }
-                        destination = request.substring(0, pos);
-                        host = getHostName(destination);
-                        line = method + " " + request.substring(pos);
-                    }   // end host name processing
-
-                    if (port != 80 && !usingWWWProxy) {
+                        // Rather than look it up, just bail out.
+                        if (_log.shouldLog(Log.WARN))
+                            _log.warn("NODOTS, NOI2P: " + request);
                         if (out != null) {
                             out.write(getErrorPage("denied", ERR_REQUEST_DENIED));
                             writeFooter(out);
                         }
                         s.close();
                         return;
-                    }
+                    }   // end host name processing
 
                     boolean isValid = usingWWWProxy || usingInternalServer || isSupportedAddress(host, protocol);
                     if (!isValid) {
@@ -645,18 +700,10 @@ public class I2PTunnelHTTPClient extends I2PTunnelHTTPClientBase implements Runn
                         break;
                     }
 
-                    // don't do this, it forces yet another hostname lookup,
-                    // and in all cases host was already set above
-                    //if ((!usingWWWProxy) && (!usingInternalServer)) {
-                    //    String oldhost = host;
-                    //    host = getHostName(destination); // hide original host
-                    //    if (_log.shouldLog(Log.INFO))
-                    //        _log.info(getPrefix(requestId) + " oldhost " + oldhost + " newhost " + host + " dest " + destination);
-                    //}
+                    line = method + ' ' + requestURI.toASCIIString() + ' ' + protocolVersion;
 
                     if (_log.shouldLog(Log.DEBUG)) {
-                        _log.debug(getPrefix(requestId) + "METHOD: \"" + method + "\"");
-                        _log.debug(getPrefix(requestId) + "PROTOC: \"" + protocol + "\"");
+                        _log.debug(getPrefix(requestId) + "NEWREQ: \"" + line + "\"");
                         _log.debug(getPrefix(requestId) + "HOST  : \"" + host + "\"");
                         _log.debug(getPrefix(requestId) + "DEST  : \"" + destination + "\"");
                     }
@@ -763,7 +810,7 @@ public class I2PTunnelHTTPClient extends I2PTunnelHTTPClientBase implements Runn
             if (method == null || destination == null) {
                 //l.log("No HTTP method found in the request.");
                 if (out != null) {
-                    if (protocol != null && "http://".equals(protocol.toLowerCase(Locale.US)))
+                    if (protocol != null && "http".equals(protocol.toLowerCase(Locale.US)))
                         out.write(getErrorPage("denied", ERR_REQUEST_DENIED));
                     else
                         out.write(getErrorPage("protocol", ERR_BAD_PROTOCOL));
@@ -794,11 +841,11 @@ public class I2PTunnelHTTPClient extends I2PTunnelHTTPClientBase implements Runn
             // Ignore all the headers
             if (usingInternalServer) {
                 // disable the add form if address helper is disabled
-                if (targetRequest.startsWith(LOCAL_SERVER + "/add?") &&
+                if (internalPath.equals("/add") &&
                     Boolean.valueOf(getTunnel().getClientOptions().getProperty(PROP_DISABLE_HELPER)).booleanValue()) {
                     out.write(ERR_HELPER_DISABLED);
                 } else {
-                    serveLocalFile(out, method, targetRequest, _proxyNonce);
+                    LocalHTTPServer.serveLocalFile(out, method, internalPath, internalRawQuery, _proxyNonce);
                 }
                 s.close();
                 return;
@@ -812,9 +859,16 @@ public class I2PTunnelHTTPClient extends I2PTunnelHTTPClientBase implements Runn
             String addressHelper = addressHelpers.get(destination.toLowerCase(Locale.US));
             if (addressHelper != null) {
                 clientDest = _context.namingService().lookup(addressHelper);
-                // remove bad entries
-                if (clientDest == null)
+                if (clientDest == null) {
+                    // remove bad entries
                     addressHelpers.remove(destination.toLowerCase(Locale.US));
+                    if (_log.shouldLog(Log.WARN))
+                        _log.warn(getPrefix(requestId) + "Could not find destination for " + addressHelper);
+                    byte[] header = getErrorPage("ahelper-notfound", ERR_AHELPER_NOTFOUND);
+                    writeErrorMessage(header, out, targetRequest, false, destination, null);
+                    s.close();
+                    return;
+                }
             } else if ("i2p".equals(host)) {
                 clientDest = null;
             } else if (destination.length() == 60 && destination.toLowerCase(Locale.US).endsWith(".b32.i2p")) {
@@ -865,7 +919,7 @@ public class I2PTunnelHTTPClient extends I2PTunnelHTTPClientBase implements Runn
             if (ahelperNew && "GET".equals(method) &&
                 (userAgent == null || !userAgent.startsWith("Wget")) &&
                 !Boolean.valueOf(getTunnel().getClientOptions().getProperty(PROP_DISABLE_HELPER)).booleanValue()) {
-                writeHelperSaveForm(out, destination, ahelperKey, protocol + targetRequest);
+                writeHelperSaveForm(out, destination, ahelperKey, targetRequest);
                 s.close();
                 return;
             }
@@ -875,10 +929,7 @@ public class I2PTunnelHTTPClient extends I2PTunnelHTTPClientBase implements Runn
             // This also prevents the not-found error page from looking bad
             // Syndie can't handle a redirect of a POST
             if (ahelperPresent && !"POST".equals(method)) {
-                String uri = protocol + targetRequest;
-                int spc = uri.indexOf(" ");
-                if (spc >= 0)
-                    uri = uri.substring(0, spc);
+                String uri = targetRequest;
                 if (_log.shouldLog(Log.DEBUG))
                     _log.debug("Auto redirecting to " + uri);
                 out.write(("HTTP/1.1 301 Address Helper Accepted\r\n"+
@@ -928,10 +979,6 @@ public class I2PTunnelHTTPClient extends I2PTunnelHTTPClientBase implements Runn
     private void writeHelperSaveForm(OutputStream out, String destination, String ahelperKey, String targetRequest) throws IOException {
         if (out == null)
             return;
-        // strip HTTP/1.1
-        int protopos = targetRequest.indexOf(" ");
-        if (protopos >= 0)
-            targetRequest = targetRequest.substring(0, protopos);
         byte[] header = getErrorPage("ahelper-new", ERR_AHELPER_NEW);
         out.write(header);
         out.write(("<table><tr><td class=\"mediumtags\" align=\"right\">" + _("Host") + "</td><td class=\"mediumtags\">" + destination + "</td></tr>\n" +
@@ -939,6 +986,7 @@ public class I2PTunnelHTTPClient extends I2PTunnelHTTPClientBase implements Runn
                    "<textarea rows=\"1\" style=\"height: 4em; min-width: 0; min-height: 0;\" cols=\"70\" wrap=\"off\" readonly=\"readonly\" >" +
                    ahelperKey + "</textarea></td></tr></table>\n" +
                    "<hr><div class=\"formaction\">"+
+                   // FIXME if there is a query remaining it is lost
                    "<form method=\"GET\" action=\"" + targetRequest + "\">" +
                    "<button type=\"submit\" class=\"go\">" + _("Continue to {0} without saving", destination) + "</button>" +
                    "</form>\n<form method=\"GET\" action=\"http://" + LOCAL_SERVER + "/add\">" +
@@ -1046,7 +1094,10 @@ public class I2PTunnelHTTPClient extends I2PTunnelHTTPClientBase implements Runn
          // we won't ever get here
     }
 
-    private static void writeFooter(OutputStream out) throws IOException {
+    /**
+     *  Public only for LocalHTTPServer, not for general use
+     */
+    public static void writeFooter(OutputStream out) throws IOException {
         // the css is hiding this div for now, but we'll keep it here anyway
         out.write("<div class=\"proxyfooter\"><p><i>I2P HTTP Proxy Server<br>Generated on: ".getBytes());
         out.write(new Date().toString().getBytes());
@@ -1091,15 +1142,10 @@ public class I2PTunnelHTTPClient extends I2PTunnelHTTPClientBase implements Runn
         if (out != null) {
             out.write(errMessage);
             if (targetRequest != null) {
-                int protopos = targetRequest.indexOf(" ");
-                String uri;
-                if (protopos >= 0)
-                    uri = targetRequest.substring(0, protopos);
-                else
-                    uri = targetRequest;
-                out.write("<a href=\"http://".getBytes());
+                String uri = targetRequest;
+                out.write("<a href=\"".getBytes());
                 out.write(uri.getBytes());
-                out.write("\">http://".getBytes());
+                out.write("\">".getBytes());
                 out.write(uri.getBytes());
                 out.write("</a>".getBytes());
                 if (usingWWWProxy) {
@@ -1193,23 +1239,9 @@ public class I2PTunnelHTTPClient extends I2PTunnelHTTPClientBase implements Runn
             }
         }
       ****/
-        return protocol.toLowerCase(Locale.US).equals("http://");
+        return protocol.toLowerCase(Locale.US).equals("http");
     }
 
-    private final static byte[] ERR_404 =
-        ("HTTP/1.1 404 Not Found\r\n"+
-         "Content-Type: text/plain\r\n"+
-         "\r\n"+
-         "HTTP Proxy local file not found")
-        .getBytes();
-
-    private final static byte[] ERR_ADD =
-        ("HTTP/1.1 409 Bad\r\n"+
-         "Content-Type: text/plain\r\n"+
-         "\r\n"+
-         "Add to addressbook failed - bad parameters")
-        .getBytes();
-
     private final static byte[] ERR_HELPER_DISABLED =
         ("HTTP/1.1 403 Disabled\r\n"+
          "Content-Type: text/plain\r\n"+
@@ -1218,185 +1250,127 @@ public class I2PTunnelHTTPClient extends I2PTunnelHTTPClientBase implements Runn
         .getBytes();
 
     /**
-     *  Very simple web server.
+     *  Change various parts of the URI.
+     *  String parameters are all non-encoded.
      *
-     *  Serve local files in the docs/ directory, for CSS and images in
-     *  error pages, using the reserved address proxy.i2p
-     *  (similar to p.p in privoxy).
-     *  This solves the problems with including links to the router console,
-     *  as assuming the router console is at 127.0.0.1 leads to broken
-     *  links if it isn't.
+     *  Scheme always preserved.
+     *  Userinfo always cleared.
+     *  Host changed if non-null.
+     *  Port changed if non-zero.
+     *  Path changed if non-null.
+     *  Query always preserved.
+     *  Fragment always cleared.
      *
-     *  Ignore all request headers (If-Modified-Since, etc.)
-     *
-     *  There is basic protection here -
-     *  FileUtil.readFile() prevents traversal above the base directory -
-     *  but inproxy/gateway ops would be wise to block proxy.i2p to prevent
-     *  exposing the docs/ directory or perhaps other issues through
-     *  uncaught vulnerabilities.
-     *  Restrict to the /themes/ directory for now.
+     *  @since 0.9
+     */
+    private static URI changeURI(URI uri, String host, int port, String path) throws URISyntaxException {
+        return new URI(uri.getScheme(),
+                       null,
+                       host != null ? host : uri.getHost(),
+                       port != 0 ? port : uri.getPort(),
+                       path != null ? path : uri.getPath(),
+                       // FIXME this breaks encoded =, &
+                       uri.getQuery(),
+                       null);
+    }
+
+    /**
+     *  Replace query in the URI.
+     *  Userinfo cleared if uri contained a query.
+     *  Fragment cleared if uri contained a query.
      *
-     *  @param targetRequest "proxy.i2p/themes/foo.png HTTP/1.1"
+     *  @param query an ENCODED query, removed if null
+     *  @since 0.9
      */
-    private static void serveLocalFile(OutputStream out, String method, String targetRequest, String proxyNonce) {
-        //System.err.println("targetRequest: \"" + targetRequest + "\"");
-        // a home page message for the curious...
-        if (targetRequest.startsWith(LOCAL_SERVER + "/ ")) {
-            try {
-                out.write(("HTTP/1.1 200 OK\r\nContent-Type: text/plain\r\nCache-Control: max-age=86400\r\n\r\nI2P HTTP proxy OK").getBytes());
-                out.flush();
-            } catch (IOException ioe) {}
-            return;
-        }
-        if ((method.equals("GET") || method.equals("HEAD")) &&
-            targetRequest.startsWith(LOCAL_SERVER + "/themes/") &&
-            !targetRequest.contains("..")) {
-            int space = targetRequest.indexOf(' ');
-            String filename = null;
-            try {
-                filename = targetRequest.substring(LOCAL_SERVER.length() + 8, space); // "/themes/".length
-            } catch (IndexOutOfBoundsException ioobe) {
-                 return;
-            }
-            // theme hack
-            if (filename.startsWith("console/default/"))
-                filename = filename.replaceFirst("default", I2PAppContext.getGlobalContext().getProperty("routerconsole.theme", "light"));
-            File themesDir = new File(_errorDir, "themes");
-            File file = new File(themesDir, filename);
-            if (file.exists() && !file.isDirectory()) {
-                String type;
-                if (filename.endsWith(".css"))
-                    type = "text/css";
-                else if (filename.endsWith(".ico"))
-                    type = "image/x-icon";
-                else if (filename.endsWith(".png"))
-                    type = "image/png";
-                else if (filename.endsWith(".jpg"))
-                    type = "image/jpeg";
-                else type = "text/html";
-                try {
-                    out.write("HTTP/1.1 200 OK\r\nContent-Type: ".getBytes());
-                    out.write(type.getBytes());
-                    out.write("\r\nCache-Control: max-age=86400\r\n\r\n".getBytes());
-                    FileUtil.readFile(filename, themesDir.getAbsolutePath(), out);
-                } catch (IOException ioe) {}
-                return;
-            }
+    private static URI replaceQuery(URI uri, String query) throws URISyntaxException {
+        URI rv = uri;
+        if (rv.getRawQuery() != null) {
+            rv = new URI(rv.getScheme(),
+                       null,
+                       uri.getHost(),
+                       uri.getPort(),
+                       uri.getPath(),
+                       null,
+                       null);
         }
-
-        // Add to addressbook (form submit)
-        // Parameters are url, host, dest, nonce, and master | router | private.
-        // Do the add and redirect.
-        if (targetRequest.startsWith(LOCAL_SERVER + "/add?")) {
-            int spc = targetRequest.indexOf(' ');
-            String query = targetRequest.substring(LOCAL_SERVER.length() + 5, spc);   // "/add?".length()
-            Map<String, String> opts = new HashMap(8);
-            StringTokenizer tok = new StringTokenizer(query, "=&;");
-            while (tok.hasMoreTokens()) {
-                String k = tok.nextToken();
-                if (!tok.hasMoreTokens())
-                    break;
-                String v = tok.nextToken();
-                opts.put(decode(k), decode(v));
-            }
-
-            String url = opts.get("url");
-            String host = opts.get("host");
-            String b64Dest = opts.get("dest");
-            String nonce = opts.get("nonce");
-            String book = "privatehosts.txt";
-            if (opts.get("master") != null)
-                book = "userhosts.txt";
-            else if (opts.get("router") != null)
-                book = "hosts.txt";
-            Destination dest = null;
-            if (b64Dest != null) {
-                try {
-                    dest = new Destination(b64Dest);
-                } catch (DataFormatException dfe) {
-                    System.err.println("Bad dest to save?" + b64Dest);
-                }
-            }
-            //System.err.println("url          : \"" + url           + "\"");
-            //System.err.println("host         : \"" + host          + "\"");
-            //System.err.println("b64dest      : \"" + b64Dest       + "\"");
-            //System.err.println("book         : \"" + book          + "\"");
-            //System.err.println("nonce        : \"" + nonce         + "\"");
-            if (proxyNonce.equals(nonce) && url != null && host != null && dest != null) {
-                try {
-                    NamingService ns = I2PAppContext.getGlobalContext().namingService();
-                    Properties nsOptions = new Properties();
-                    nsOptions.setProperty("list", book);
-                    nsOptions.setProperty("s", _("Added via address helper"));
-                    boolean success = ns.put(host, dest, nsOptions);
-                    writeRedirectPage(out, success, host, book, url);
-                    return;
-                } catch (IOException ioe) {}
-            }
-            try {
-                out.write(ERR_ADD);
-                out.flush();
-            } catch (IOException ioe) {}
-            return;
+        if (query != null) {
+            String newURI = rv.toASCIIString() + '?' + query;
+            rv = new URI(newURI);
         }
-        try {
-            out.write(ERR_404);
-            out.flush();
-        } catch (IOException ioe) {}
-    }
-
-    /** @since 0.8.7 */
-    private static void writeRedirectPage(OutputStream out, boolean success, String host, String book, String url) throws IOException {
-        out.write(("HTTP/1.1 200 OK\r\n"+
-                  "Content-Type: text/html; charset=UTF-8\r\n"+
-                  "\r\n"+
-                  "<html><head>"+
-                  "<title>" + _("Redirecting to {0}", host) + "</title>\n" +
-                  "<link rel=\"shortcut icon\" href=\"http://proxy.i2p/themes/console/images/favicon.ico\" >\n" +
-                  "<link href=\"http://proxy.i2p/themes/console/default/console.css\" rel=\"stylesheet\" type=\"text/css\" >\n" +
-                  "<meta http-equiv=\"Refresh\" content=\"1; url=" + url + "\">\n" +
-                  "</head><body>\n" +
-                  "<div class=logo>\n" +
-                  "<a href=\"http://127.0.0.1:7657/\" title=\"" + _("Router Console") + "\"><img src=\"http://proxy.i2p/themes/console/images/i2plogo.png\" alt=\"I2P Router Console\" border=\"0\"></a><hr>\n" +
-                  "<a href=\"http://127.0.0.1:7657/config\">" + _("Configuration") + "</a> <a href=\"http://127.0.0.1:7657/help.jsp\">" + _("Help") + "</a> <a href=\"http://127.0.0.1:7657/susidns/index\">" + _("Addressbook") + "</a>\n" +
-                  "</div>" +
-                  "<div class=warning id=warning>\n" +
-                  "<h3>" +
-                  (success ?
-                           _("Saved {0} to the {1} addressbook, redirecting now.", host, book) :
-                           _("Failed to save {0} to the {1} addressbook, redirecting now.", host, book)) +
-                  "</h3>\n<p><a href=\"" + url + "\">" +
-                  _("Click here if you are not redirected automatically.") +
-                  "</a></p></div>").getBytes("UTF-8"));
-        writeFooter(out);
-        out.flush();
+        return rv;
     }
 
     /**
-     *  Decode %xx encoding
-     *  @since 0.8.7
+     *  Remove the address helper from an encoded query.
+     *
+     *  @param query an ENCODED query, removed if null
+     *  @return rv[0] is ENCODED query with helper removed, non-null but possibly empty;
+     *          rv[1] is DECODED helper value, non-null but possibly empty;
+     *          rv null if no helper present
+     *  @since 0.9
      */
-    private static String decode(String s) {
-        if (!s.contains("%"))
-            return s;
-        StringBuilder buf = new StringBuilder(s.length());
-        for (int i = 0; i < s.length(); i++) {
-            char c = s.charAt(i);
-            if (c != '%') {
-                buf.append(c);
-            } else {
-                try {
-                    buf.append((char) Integer.parseInt(s.substring(++i, (++i) + 1), 16));
-                } catch (IndexOutOfBoundsException ioobe) {
-                    break;
-                } catch (NumberFormatException nfe) {
-                    break;
+    private static String[] removeHelper(String query) {
+        int keystart = 0;
+        int valstart = -1;
+        String key = null;
+        for (int i = 0; i <= query.length(); i++) {
+            char c = i < query.length() ? query.charAt(i) : '&';
+            if (c == ';' || c == '&') {
+                // end of key or value
+                if (valstart < 0)
+                    key = query.substring(keystart, i);
+                String decodedKey = LocalHTTPServer.decode(key);
+                if (decodedKey.equals(HELPER_PARAM)) {
+                    String newQuery = keystart > 0 ? query.substring(0, keystart - 1) : "";
+                    if (i < query.length() - 1) {
+                        if (keystart > 0)
+                            newQuery += query.substring(i);
+                        else
+                            newQuery += query.substring(i + 1);
+                    }
+                    String value = valstart >= 0 ? query.substring(valstart, i) : "";
+                    String helperValue = LocalHTTPServer.decode(value);
+                    return new String[] { newQuery, helperValue };
                 }
+                keystart = i + 1;
+                valstart = -1;
+            } else if (c == '=') {
+                // end of key
+                key = query.substring(keystart, i);
+                valstart = i + 1;
             }
         }
-        return buf.toString();
+        return null;
+    }
+
+/****
+    private static String[] tests = {
+        "", "foo", "foo=bar", "&", "&=&", "===", "&&",
+        "i2paddresshelper=foo",
+        "i2paddresshelpe=foo",
+        "2paddresshelper=foo",
+        "i2paddresshelper=%66oo",
+        "%692paddresshelper=foo",
+        "i2paddresshelper=foo&a=b",
+        "a=b&i2paddresshelper=foo",
+        "a=b&i2paddresshelper&c=d",
+        "a=b&i2paddresshelper=foo&c=d",
+        "a=b;i2paddresshelper=foo;c=d",
+        "a=b&i2paddresshelper=foo&c"
+    };
+
+    public static void main(String[] args) {
+        for (int i = 0; i < tests.length; i++) {
+            String[] s = removeHelper(tests[i]);
+            if (s != null)
+                System.out.println("Test \"" + tests[i] + "\" q=\"" + s[0] + "\" h=\"" + s[1] + "\"");
+            else
+                System.out.println("Test \"" + tests[i] + "\" no match");
+        }
     }
+****/
 
+    /** */
     private static final String BUNDLE_NAME = "net.i2p.i2ptunnel.web.messages";
 
     /** lang in routerconsole.lang property, else current locale */
diff --git a/apps/i2ptunnel/java/src/net/i2p/i2ptunnel/localServer/LocalHTTPServer.java b/apps/i2ptunnel/java/src/net/i2p/i2ptunnel/localServer/LocalHTTPServer.java
new file mode 100644
index 0000000000000000000000000000000000000000..7c68c4e4062525dbc55a5b1780943cf4a51b6c3d
--- /dev/null
+++ b/apps/i2ptunnel/java/src/net/i2p/i2ptunnel/localServer/LocalHTTPServer.java
@@ -0,0 +1,247 @@
+/* I2PTunnel is GPL'ed (with the exception mentioned in I2PTunnel.java)
+ * (c) 2003 - 2004 mihi
+ */
+package net.i2p.i2ptunnel.localServer;
+
+import java.io.File;
+import java.io.IOException;
+import java.io.OutputStream;
+import java.util.HashMap;
+import java.util.Map;
+import java.util.Properties;
+import java.util.StringTokenizer;
+
+import net.i2p.I2PAppContext;
+import net.i2p.client.naming.NamingService;
+import net.i2p.data.DataFormatException;
+import net.i2p.data.Destination;
+import net.i2p.i2ptunnel.I2PTunnelHTTPClient;
+import net.i2p.util.FileUtil;
+import net.i2p.util.Log;
+import net.i2p.util.Translate;
+
+/**
+ *  Very simple web server.
+ *
+ *  Serve local files in the docs/ directory, for CSS and images in
+ *  error pages, using the reserved address proxy.i2p
+ *  (similar to p.p in privoxy).
+ *  This solves the problems with including links to the router console,
+ *  as assuming the router console is at 127.0.0.1 leads to broken
+ *  links if it isn't.
+ *
+ *  @since 0.7.6, moved from I2PTunnelHTTPClient in 0.9
+ */
+public abstract class LocalHTTPServer {
+
+    private final static byte[] ERR_404 =
+        ("HTTP/1.1 404 Not Found\r\n"+
+         "Content-Type: text/plain\r\n"+
+         "\r\n"+
+         "HTTP Proxy local file not found")
+        .getBytes();
+
+    private final static byte[] ERR_ADD =
+        ("HTTP/1.1 409 Bad\r\n"+
+         "Content-Type: text/plain\r\n"+
+         "\r\n"+
+         "Add to addressbook failed - bad parameters")
+        .getBytes();
+
+    /**
+     *  Very simple web server.
+     *
+     *  Serve local files in the docs/ directory, for CSS and images in
+     *  error pages, using the reserved address proxy.i2p
+     *  (similar to p.p in privoxy).
+     *  This solves the problems with including links to the router console,
+     *  as assuming the router console is at 127.0.0.1 leads to broken
+     *  links if it isn't.
+     *
+     *  Ignore all request headers (If-Modified-Since, etc.)
+     *
+     *  There is basic protection here -
+     *  FileUtil.readFile() prevents traversal above the base directory -
+     *  but inproxy/gateway ops would be wise to block proxy.i2p to prevent
+     *  exposing the docs/ directory or perhaps other issues through
+     *  uncaught vulnerabilities.
+     *  Restrict to the /themes/ directory for now.
+     *
+     *  @param targetRequest decoded path only, non-null
+     *  @param query raw (encoded), may be null
+     */
+    public static void serveLocalFile(OutputStream out, String method, String targetRequest, String query, String proxyNonce) {
+        //System.err.println("targetRequest: \"" + targetRequest + "\"");
+        // a home page message for the curious...
+        if (targetRequest.equals("/")) {
+            try {
+                out.write(("HTTP/1.1 200 OK\r\nContent-Type: text/plain\r\nCache-Control: max-age=86400\r\n\r\nI2P HTTP proxy OK").getBytes());
+                out.flush();
+            } catch (IOException ioe) {}
+            return;
+        }
+        if ((method.equals("GET") || method.equals("HEAD")) &&
+            targetRequest.startsWith("/themes/") &&
+            !targetRequest.contains("..")) {
+            String filename = null;
+            try {
+                filename = targetRequest.substring(8); // "/themes/".length
+            } catch (IndexOutOfBoundsException ioobe) {
+                 return;
+            }
+            // theme hack
+            if (filename.startsWith("console/default/"))
+                filename = filename.replaceFirst("default", I2PAppContext.getGlobalContext().getProperty("routerconsole.theme", "light"));
+            File themesDir = new File(I2PAppContext.getGlobalContext().getBaseDir(), "docs/themes");
+            File file = new File(themesDir, filename);
+            if (file.exists() && !file.isDirectory()) {
+                String type;
+                if (filename.endsWith(".css"))
+                    type = "text/css";
+                else if (filename.endsWith(".ico"))
+                    type = "image/x-icon";
+                else if (filename.endsWith(".png"))
+                    type = "image/png";
+                else if (filename.endsWith(".jpg"))
+                    type = "image/jpeg";
+                else type = "text/html";
+                try {
+                    out.write("HTTP/1.1 200 OK\r\nContent-Type: ".getBytes());
+                    out.write(type.getBytes());
+                    out.write("\r\nCache-Control: max-age=86400\r\n\r\n".getBytes());
+                    FileUtil.readFile(filename, themesDir.getAbsolutePath(), out);
+                } catch (IOException ioe) {}
+                return;
+            }
+        }
+
+        // Add to addressbook (form submit)
+        // Parameters are url, host, dest, nonce, and master | router | private.
+        // Do the add and redirect.
+        if (targetRequest.equals("/add")) {
+            Map<String, String> opts = new HashMap(8);
+            // this only works if all keys are followed by =value
+            StringTokenizer tok = new StringTokenizer(query, "=&;");
+            while (tok.hasMoreTokens()) {
+                String k = tok.nextToken();
+                if (!tok.hasMoreTokens())
+                    break;
+                String v = tok.nextToken();
+                opts.put(decode(k), decode(v));
+            }
+
+            String url = opts.get("url");
+            String host = opts.get("host");
+            String b64Dest = opts.get("dest");
+            String nonce = opts.get("nonce");
+            String book = "privatehosts.txt";
+            if (opts.get("master") != null)
+                book = "userhosts.txt";
+            else if (opts.get("router") != null)
+                book = "hosts.txt";
+            Destination dest = null;
+            if (b64Dest != null) {
+                try {
+                    dest = new Destination(b64Dest);
+                } catch (DataFormatException dfe) {
+                    System.err.println("Bad dest to save?" + b64Dest);
+                }
+            }
+            //System.err.println("url          : \"" + url           + "\"");
+            //System.err.println("host         : \"" + host          + "\"");
+            //System.err.println("b64dest      : \"" + b64Dest       + "\"");
+            //System.err.println("book         : \"" + book          + "\"");
+            //System.err.println("nonce        : \"" + nonce         + "\"");
+            if (proxyNonce.equals(nonce) && url != null && host != null && dest != null) {
+                try {
+                    NamingService ns = I2PAppContext.getGlobalContext().namingService();
+                    Properties nsOptions = new Properties();
+                    nsOptions.setProperty("list", book);
+                    nsOptions.setProperty("s", _("Added via address helper"));
+                    boolean success = ns.put(host, dest, nsOptions);
+                    writeRedirectPage(out, success, host, book, url);
+                    return;
+                } catch (IOException ioe) {}
+            }
+            try {
+                out.write(ERR_ADD);
+                out.flush();
+            } catch (IOException ioe) {}
+            return;
+        }
+        try {
+            out.write(ERR_404);
+            out.flush();
+        } catch (IOException ioe) {}
+    }
+
+    /** @since 0.8.7 */
+    private static void writeRedirectPage(OutputStream out, boolean success, String host, String book, String url) throws IOException {
+        out.write(("HTTP/1.1 200 OK\r\n"+
+                  "Content-Type: text/html; charset=UTF-8\r\n"+
+                  "\r\n"+
+                  "<html><head>"+
+                  "<title>" + _("Redirecting to {0}", host) + "</title>\n" +
+                  "<link rel=\"shortcut icon\" href=\"http://proxy.i2p/themes/console/images/favicon.ico\" >\n" +
+                  "<link href=\"http://proxy.i2p/themes/console/default/console.css\" rel=\"stylesheet\" type=\"text/css\" >\n" +
+                  "<meta http-equiv=\"Refresh\" content=\"1; url=" + url + "\">\n" +
+                  "</head><body>\n" +
+                  "<div class=logo>\n" +
+                  "<a href=\"http://127.0.0.1:7657/\" title=\"" + _("Router Console") + "\"><img src=\"http://proxy.i2p/themes/console/images/i2plogo.png\" alt=\"I2P Router Console\" border=\"0\"></a><hr>\n" +
+                  "<a href=\"http://127.0.0.1:7657/config\">" + _("Configuration") + "</a> <a href=\"http://127.0.0.1:7657/help.jsp\">" + _("Help") + "</a> <a href=\"http://127.0.0.1:7657/susidns/index\">" + _("Addressbook") + "</a>\n" +
+                  "</div>" +
+                  "<div class=warning id=warning>\n" +
+                  "<h3>" +
+                  (success ?
+                           _("Saved {0} to the {1} addressbook, redirecting now.", host, book) :
+                           _("Failed to save {0} to the {1} addressbook, redirecting now.", host, book)) +
+                  "</h3>\n<p><a href=\"" + url + "\">" +
+                  _("Click here if you are not redirected automatically.") +
+                  "</a></p></div>").getBytes("UTF-8"));
+        I2PTunnelHTTPClient.writeFooter(out);
+        out.flush();
+    }
+
+    /**
+     *  Decode %xx encoding
+     *  @since 0.8.7
+     */
+    public static String decode(String s) {
+        if (!s.contains("%"))
+            return s;
+        StringBuilder buf = new StringBuilder(s.length());
+        for (int i = 0; i < s.length(); i++) {
+            char c = s.charAt(i);
+            if (c != '%') {
+                buf.append(c);
+            } else {
+                try {
+                    buf.append((char) Integer.parseInt(s.substring(++i, (++i) + 1), 16));
+                } catch (IndexOutOfBoundsException ioobe) {
+                    break;
+                } catch (NumberFormatException nfe) {
+                    break;
+                }
+            }
+        }
+        return buf.toString();
+    }
+
+    private static final String BUNDLE_NAME = "net.i2p.i2ptunnel.web.messages";
+
+    /** lang in routerconsole.lang property, else current locale */
+    protected static String _(String key) {
+        return Translate.getString(key, I2PAppContext.getGlobalContext(), BUNDLE_NAME);
+    }
+
+    /** {0} */
+    protected static String _(String key, Object o) {
+        return Translate.getString(key, o, I2PAppContext.getGlobalContext(), BUNDLE_NAME);
+    }
+
+    /** {0} and {1} */
+    protected static String _(String key, Object o, Object o2) {
+        return Translate.getString(key, o, o2, I2PAppContext.getGlobalContext(), BUNDLE_NAME);
+    }
+
+}
diff --git a/apps/i2ptunnel/jsp/wizard.jsp b/apps/i2ptunnel/jsp/wizard.jsp
index ddb6556991c90efeda21f47053f09536b894b1c5..74afb65a28790f56d2a470484ad441af1ac27647 100644
--- a/apps/i2ptunnel/jsp/wizard.jsp
+++ b/apps/i2ptunnel/jsp/wizard.jsp
@@ -1,6 +1,10 @@
 <%
     // NOTE: Do the header carefully so there is no whitespace before the <?xml... line
 
+    // http://www.crazysquirrel.com/computing/general/form-encoding.jspx
+    if (request.getCharacterEncoding() == null)
+        request.setCharacterEncoding("UTF-8");
+
 %><%@page pageEncoding="UTF-8"
 %><%@page contentType="text/html" import="net.i2p.i2ptunnel.web.EditBean"
 %><?xml version="1.0" encoding="UTF-8"?>
diff --git a/apps/jetty/build.xml b/apps/jetty/build.xml
index e686e364a73c17db730bbcf1959f1bc4f5f3ca05..de9534274b35867cc1e869c5c6cb8c77a5884c5a 100644
--- a/apps/jetty/build.xml
+++ b/apps/jetty/build.xml
@@ -117,6 +117,7 @@
         <property name="workspace.changes.tr" value="" />
         <jar destfile="./jettylib/org.mortbay.jetty.jar" basedir="./build/obj" includes="**/*.class" update="true" >
             <manifest>
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.tr}" />
diff --git a/apps/ministreaming/java/build.xml b/apps/ministreaming/java/build.xml
index 235088fc312852a00a4acd178198fc3b7267932d..451ecf0f4c031b27c42a136706163cad968a5227 100644
--- a/apps/ministreaming/java/build.xml
+++ b/apps/ministreaming/java/build.xml
@@ -51,6 +51,7 @@
         <jar destfile="./build/mstreaming.jar" basedir="./build/obj" includes="**/*.class" >
             <manifest>
                 <attribute name="Implementation-Version" value="${full.version}" />
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.tr}" />
diff --git a/apps/routerconsole/java/build.xml b/apps/routerconsole/java/build.xml
index 3dba335781c6f87708177aa15ce91c80d5017dba..714bd17db5b6fe4bccf9b5c858ffc822ad171344 100644
--- a/apps/routerconsole/java/build.xml
+++ b/apps/routerconsole/java/build.xml
@@ -92,6 +92,7 @@
                 <!-- DTG added in 0.8.4, not in the classpath for very old installs, before we changed wrapper.config to specify * -->
                 <attribute name="Class-Path" value="i2p.jar router.jar jrobin.jar desktopgui.jar" />
                 <attribute name="Implementation-Version" value="${full.version}" />
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.j.tr}" />
@@ -172,6 +173,7 @@
              basedir="../jsp/" excludes="web.xml, *.css, **/*.java, *.jsp, *.jsi, web-fragment.xml, web-out.xml">
             <manifest>
                 <attribute name="Implementation-Version" value="${full.version}" />
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.w.tr}" />
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/ConfigClientsHelper.java b/apps/routerconsole/java/src/net/i2p/router/web/ConfigClientsHelper.java
index 7f59670bc1c1ca5e069f1dc2caeba60dce054eb6..f94f4b9c77287ae366901f13925e9d13c4193875 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/ConfigClientsHelper.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/ConfigClientsHelper.java
@@ -52,7 +52,7 @@ public class ConfigClientsHelper extends HelperBase {
         if ((mode == 0 && disabled) ||
             (mode == 1 && (!disabled) && (!ssl)) ||
             (mode == 2 && (!disabled) && ssl))
-            return "checked=\"true\"";
+            return "checked=\"checked\"";
         return "";
     }
 
@@ -60,7 +60,7 @@ public class ConfigClientsHelper extends HelperBase {
     public String getAuth() {
         boolean enabled =  _context.getBooleanProperty(PROP_AUTH);
         if (enabled)
-            return "checked=\"true\"";
+            return "checked=\"checked\"";
         return "";
     }
 
@@ -253,9 +253,9 @@ public class ConfigClientsHelper extends HelperBase {
         }
         buf.append("</td><td align=\"center\" width=\"10%\"><input type=\"checkbox\" class=\"optbox\" name=\"").append(index).append(".enabled\" value=\"true\" ");
         if (enabled) {
-            buf.append("checked=\"true\" ");
+            buf.append("checked=\"checked\" ");
             if (ro)
-                buf.append("disabled=\"true\" ");
+                buf.append("disabled=\"disabled\" ");
         }
         buf.append("></td><td align=\"center\" width=\"15%\">");
         // The icons were way too much, so there's an X in each button class,
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/ConfigLoggingHelper.java b/apps/routerconsole/java/src/net/i2p/router/web/ConfigLoggingHelper.java
index 6b86caeb226f3eb62c64f5ae9f7edb4f26cca807..ea8ec0aedca99061ede2590b182da01274633171 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/ConfigLoggingHelper.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/ConfigLoggingHelper.java
@@ -80,7 +80,7 @@ public class ConfigLoggingHelper extends HelperBase {
             String l = levels[i];
             buf.append("<option value=\"").append(l).append('\"');
             if (l.equals(cur))
-                buf.append(" selected=\"true\"");
+                buf.append(" selected=\"selected\"");
             buf.append('>').append(_(l)).append("</option>\n");
         }        
         
@@ -121,7 +121,7 @@ public class ConfigLoggingHelper extends HelperBase {
 
         StringBuilder buf = new StringBuilder(65536);
         buf.append("<select name=\"newlogclass\">\n" +
-                   "<option value=\"\" selected=\"true\">")
+                   "<option value=\"\" selected=\"selected\">")
            .append(_("Select a class to add"))
            .append("</option>\n");
 
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/ConfigNetHelper.java b/apps/routerconsole/java/src/net/i2p/router/web/ConfigNetHelper.java
index e9d3b152992bedb6af17eef8937bf65024edf6fd..b08fb3b019bbdb1d4a96118d58f7c89090027eed 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/ConfigNetHelper.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/ConfigNetHelper.java
@@ -20,8 +20,8 @@ public class ConfigNetHelper extends HelperBase {
     public final static String PROP_I2NP_NTCP_PORT = "i2np.ntcp.port";
     public final static String PROP_I2NP_NTCP_AUTO_PORT = "i2np.ntcp.autoport";
     public final static String PROP_I2NP_NTCP_AUTO_IP = "i2np.ntcp.autoip";
-    private final static String CHECKED = " checked=\"true\" ";
-    private final static String DISABLED = " disabled=\"true\" ";
+    private final static String CHECKED = " checked=\"checked\" ";
+    private final static String DISABLED = " disabled=\"disabled\" ";
 
     public String getUdphostname() {
         return _context.getProperty(UDPTransport.PROP_EXTERNAL_HOST, ""); 
@@ -250,7 +250,7 @@ public class ConfigNetHelper extends HelperBase {
             }
             buf.append("<option style=\"text-align: right;\" value=\"").append(val).append("\" ");
             if (pct == val) {
-                buf.append("selected=\"true\" ");
+                buf.append("selected=\"selected\" ");
                 found = true;
             }
             buf.append(">").append(val).append("%</option>\n");
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/ConfigReseedHelper.java b/apps/routerconsole/java/src/net/i2p/router/web/ConfigReseedHelper.java
index 145bd49024b94c1995677126f6c305c4f4a2bcb5..5b5d747519629a0f7efe5dcba5b55476897832e3 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/ConfigReseedHelper.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/ConfigReseedHelper.java
@@ -56,7 +56,7 @@ public class ConfigReseedHelper extends HelperBase {
         if ((mode == 0 && (!disabled) && (!required)) ||
             (mode == 1 && (!disabled) && required) ||
             (mode == 2 && disabled))
-            return "checked=\"true\"";
+            return "checked=\"checked\"";
         return "";
     }
 
@@ -82,7 +82,7 @@ public class ConfigReseedHelper extends HelperBase {
     private String checked(String prop) {
         boolean enabled =  _context.getBooleanProperty(prop);
         if (enabled)
-            return "checked=\"true\"";
+            return "checked=\"checked\"";
         return "";
     }
 
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/ConfigStatsHelper.java b/apps/routerconsole/java/src/net/i2p/router/web/ConfigStatsHelper.java
index 254a20728d2ed733f65b2adf6545dbd5b8922b51..61d4969330f498b2607cb622c1deff980c04902f 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/ConfigStatsHelper.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/ConfigStatsHelper.java
@@ -59,8 +59,16 @@ public class ConfigStatsHelper extends HelperBase {
             _filters.add(tok.nextToken().trim());
     }
 
-    public ConfigStatsHelper() {}
-    
+    /**
+     *  Just hide for everybody unless already set.
+     *  To enable set advanced config stat.logFilters=foo before starting...
+     *  it has to be set at startup anyway for logging to be enabled at all
+     *  @since 0.9
+     */
+    public boolean shouldShowLog() {
+        return !_filters.isEmpty();
+    }
+
     public String getFilename() { return _context.statManager().getStatFile(); }
     
     /** 
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/ConfigTunnelsHelper.java b/apps/routerconsole/java/src/net/i2p/router/web/ConfigTunnelsHelper.java
index 6df070bd34ba698505fb8810a30a91d75ce66bb0..14b18c7d706df5ef3a27013af393545942596765 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/ConfigTunnelsHelper.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/ConfigTunnelsHelper.java
@@ -172,7 +172,7 @@ public class ConfigTunnelsHelper extends HelperBase {
         if (!props.isEmpty()) {
             buf.append("<tr><td align=\"right\" class=\"mediumtags\">" + _("Inbound options") + ":</td>\n" +
                        "<td colspan=\"2\" align=\"center\"><input name=\"").append(index);
-            buf.append(".inboundOptions\" type=\"text\" size=\"32\" disabled=\"true\" " +
+            buf.append(".inboundOptions\" type=\"text\" size=\"32\" disabled=\"disabled\" " +
                        "value=\"");
             for (Iterator iter = props.keySet().iterator(); iter.hasNext(); ) {
                 String prop = (String)iter.next();
@@ -185,7 +185,7 @@ public class ConfigTunnelsHelper extends HelperBase {
         if (!props.isEmpty()) {
             buf.append("<tr><td align=\"right\" class=\"mediumtags\">" + _("Outbound options") + ":</td>\n" +
                        "<td colspan=\"2\" align=\"center\"><input name=\"").append(index);
-            buf.append(".outboundOptions\" type=\"text\" size=\"32\" disabled=\"true\" " +
+            buf.append(".outboundOptions\" type=\"text\" size=\"32\" disabled=\"disabled\" " +
                        "value=\"");
             for (Iterator iter = props.keySet().iterator(); iter.hasNext(); ) {
                 String prop = (String)iter.next();
@@ -205,7 +205,7 @@ public class ConfigTunnelsHelper extends HelperBase {
         for (int i = min; i <= max; i++) {
             buf.append("<option value=\"").append(i).append("\" ");
             if (i == now)
-                buf.append("selected=\"true\" ");
+                buf.append("selected=\"selected\" ");
             buf.append(">").append(ngettext(DUMMY1 + name, DUMMY2 + name + 's', i));
             buf.append("</option>\n");
         }
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/ConfigUIHelper.java b/apps/routerconsole/java/src/net/i2p/router/web/ConfigUIHelper.java
index 7acae19630d5f3ddd811dd141039eedcef055170..f5f16aebd2af5bdfa6d777809bb8b006aa5f6364 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/ConfigUIHelper.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/ConfigUIHelper.java
@@ -14,7 +14,7 @@ public class ConfigUIHelper extends HelperBase {
         for (String theme : themes) {
             buf.append("<input type=\"radio\" class=\"optbox\" name=\"theme\" ");
             if (theme.equals(current))
-                buf.append("checked=\"true\" ");
+                buf.append("checked=\"checked\" ");
             buf.append("value=\"").append(theme).append("\">").append(_(theme)).append("<br>\n");
         }
         return buf.toString();
@@ -73,7 +73,7 @@ public class ConfigUIHelper extends HelperBase {
             // we use "lang" so it is set automagically in CSSHelper
             buf.append("<input type=\"radio\" class=\"optbox\" name=\"lang\" ");
             if (langs[i].equals(current))
-                buf.append("checked=\"true\" ");
+                buf.append("checked=\"checked\" ");
             buf.append("value=\"").append(langs[i]).append("\">")
                .append("<img height=\"11\" width=\"16\" alt=\"\" src=\"/flags.jsp?c=").append(flags[i]).append("\"> ")
                .append(_(xlangs[i])).append("<br>\n");
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/ConfigUpdateHelper.java b/apps/routerconsole/java/src/net/i2p/router/web/ConfigUpdateHelper.java
index 88aaa70b208c334130613e642837925104a29ef4..8ef110c396564c89989a09682f6b08e3249f5735 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/ConfigUpdateHelper.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/ConfigUpdateHelper.java
@@ -75,7 +75,7 @@ public class ConfigUpdateHelper extends HelperBase {
     public String getUpdateThroughProxy() {
         String proxy = _context.getProperty(ConfigUpdateHandler.PROP_SHOULD_PROXY, ConfigUpdateHandler.DEFAULT_SHOULD_PROXY);
         if (Boolean.valueOf(proxy).booleanValue()) 
-            return "<input type=\"checkbox\" class=\"optbox\" value=\"true\" name=\"updateThroughProxy\" checked=\"true\" >";
+            return "<input type=\"checkbox\" class=\"optbox\" value=\"true\" name=\"updateThroughProxy\" checked=\"checked\" >";
         else
             return "<input type=\"checkbox\" class=\"optbox\" value=\"true\" name=\"updateThroughProxy\" >";
     }
@@ -83,7 +83,7 @@ public class ConfigUpdateHelper extends HelperBase {
     public String getUpdateUnsigned() {
         String foo = _context.getProperty(ConfigUpdateHandler.PROP_UPDATE_UNSIGNED);
         if (Boolean.valueOf(foo).booleanValue()) 
-            return "<input type=\"checkbox\" class=\"optbox\" value=\"true\" name=\"updateUnsigned\" checked=\"true\" >";
+            return "<input type=\"checkbox\" class=\"optbox\" value=\"true\" name=\"updateUnsigned\" checked=\"checked\" >";
         else
             return "<input type=\"checkbox\" class=\"optbox\" value=\"true\" name=\"updateUnsigned\" >";
     }
@@ -106,7 +106,7 @@ public class ConfigUpdateHelper extends HelperBase {
         for (int i = 0; i < PERIODS.length; i++) {
             buf.append("<option value=\"").append(PERIODS[i]);
             if (PERIODS[i] == ms)
-                buf.append("\" selected=\"true");
+                buf.append("\" selected=\"selected");
             
             if (PERIODS[i] == -1)
                 buf.append("\">" + _("Never") + "</option>\n");
@@ -128,22 +128,22 @@ public class ConfigUpdateHelper extends HelperBase {
         
         buf.append("<option value=\"notify\"");
         if ("notify".equals(policy) || _dontInstall)
-            buf.append(" selected=\"true\"");
+            buf.append(" selected=\"selected\"");
         buf.append('>').append(_("Notify only")).append("</option>");
 
         buf.append("<option value=\"download\"");
         if (_dontInstall)
-            buf.append(" disabled=\"true\"");
+            buf.append(" disabled=\"disabled\"");
         else if ("download".equals(policy))
-            buf.append(" selected=\"true\"");
+            buf.append(" selected=\"selected\"");
         buf.append('>').append(_("Download and verify only")).append("</option>");
         
         if (_context.hasWrapper()) {
             buf.append("<option value=\"install\"");
             if (_dontInstall)
-                buf.append(" disabled=\"true\"");
+                buf.append(" disabled=\"disabled\"");
             else if ("install".equals(policy))
-                buf.append(" selected=\"true\"");
+                buf.append(" selected=\"selected\"");
             buf.append('>').append(_("Download, verify, and restart")).append("</option>");
         }
         
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/FileDumpHelper.java b/apps/routerconsole/java/src/net/i2p/router/web/FileDumpHelper.java
index eea306aa209f174fdb77cf725fbfced0306f8085..7bb1e99d05f2efe61713b419dbd7ccedb9c7a43d 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/FileDumpHelper.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/FileDumpHelper.java
@@ -18,6 +18,7 @@ import java.util.jar.Manifest;
 
 import net.i2p.crypto.SHA256Generator;
 import net.i2p.data.DataHelper;
+import net.i2p.util.FileUtil;
 
 /**
  *  Dump info on jars and wars
@@ -29,7 +30,7 @@ public class FileDumpHelper extends HelperBase {
     public String getFileSummary() {
         StringBuilder buf = new StringBuilder(16*1024);
         buf.append("<table><tr><th>File</th><th>Size</th><th>Date</th><th>SHA 256</th><th>Revision</th>" +
-                   "<th>JDK</th><th>Built</th><th>Mods</th></tr>");
+                   "<th>JDK</th><th>Built</th><th>By</th><th>Mods</th></tr>");
 
         // jars added in wrapper.config
         URLClassLoader urlClassLoader = (URLClassLoader) ClassLoader.getSystemClassLoader();
@@ -95,7 +96,9 @@ public class FileDumpHelper extends HelperBase {
             buf.append((new Date(mod)).toString());
         else
             buf.append("<font color=\"red\">Not found</font>");
-        buf.append("</td><td>");
+        buf.append("</td><td align=\"center\">");
+        if (mod > 0 && !FileUtil.verifyZip(f))
+            buf.append("<font color=\"red\">CORRUPT</font><br>");
         byte[] hash = sha256(f);
         if (hash != null) {
             byte[] hh = new byte[16];
@@ -132,13 +135,17 @@ public class FileDumpHelper extends HelperBase {
             buf.append(s);
         buf.append("</td><td>");
         s = getAtt(att, "Build-Date");
+        if (s != null)
+            buf.append(s);
+        buf.append("</td><td align=\"center\">");
+        s = getAtt(att, "Built-By");
         if (s != null)
             buf.append(s);
         buf.append("</td><td><font color=\"red\">");
         s = getAtt(att, "Workspace-Changes");
         if (s != null)
             buf.append(s.replace(",", "<br>"));
-        buf.append("</font></td>");
+        buf.append("</font></td></tr>\n");
     }
 
     private static byte[] sha256(File f) {
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/GraphHelper.java b/apps/routerconsole/java/src/net/i2p/router/web/GraphHelper.java
index b6a4ad283778f94332fbbf27ae5aea1cec96e8ac..b5d54b4112712405309d4d6b23c077a251b63791 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/GraphHelper.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/GraphHelper.java
@@ -20,6 +20,8 @@ public class GraphHelper extends FormHandler {
     private int _height;
     private int _refreshDelaySeconds;
     private boolean _persistent;
+    private String _stat;
+    private int _end;
 
     private static final String PROP_X = "routerconsole.graphX";
     private static final String PROP_Y = "routerconsole.graphY";
@@ -32,6 +34,10 @@ public class GraphHelper extends FormHandler {
     private static final int DEFAULT_PERIODS = 60;
     static final int MAX_X = 2048;
     static final int MAX_Y = 1024;
+    private static final int MIN_X = 200;
+    private static final int MIN_Y = 60;
+    private static final int MIN_C = 20;
+    private static final int MAX_C = SummaryListener.MAX_ROWS;
     private static final int MIN_REFRESH = 15;
     
     /** set the defaults after we have a context */
@@ -64,17 +70,45 @@ public class GraphHelper extends FormHandler {
     public void storeWriter(Writer out) { _out = out; }
 
     public void setPeriodCount(String str) { 
-        try { _periodCount = Integer.parseInt(str); } catch (NumberFormatException nfe) {}
+        setC(str);
     }
 
-    public void setShowEvents(boolean b) { _showEvents = b; }
+    /** @since 0.9 */
+    public void setE(String str) { 
+        try {
+            _end = Math.max(0, Integer.parseInt(str));
+        } catch (NumberFormatException nfe) {}
+    }
+
+    /** @since 0.9 shorter parameter */
+    public void setC(String str) { 
+        try {
+            _periodCount = Math.max(MIN_C, Math.min(Integer.parseInt(str), MAX_C));
+        } catch (NumberFormatException nfe) {}
+    }
+
+    public void setShowEvents(String b) { _showEvents = !"false".equals(b); }
 
     public void setHeight(String str) {
-        try { _height = Math.min(Integer.parseInt(str), MAX_Y); } catch (NumberFormatException nfe) {}
+        setH(str);
+    }
+
+    /** @since 0.9 shorter parameter */
+    public void setH(String str) { 
+        try {
+            _height = Math.max(MIN_Y, Math.min(Integer.parseInt(str), MAX_Y));
+        } catch (NumberFormatException nfe) {}
     }
 
     public void setWidth(String str) {
-        try { _width = Math.min(Integer.parseInt(str), MAX_X); } catch (NumberFormatException nfe) {}
+        setW(str);
+    }
+
+    /** @since 0.9 shorter parameter */
+    public void setW(String str) { 
+        try {
+            _width = Math.max(MIN_X, Math.min(Integer.parseInt(str), MAX_X));
+        } catch (NumberFormatException nfe) {}
     }
 
     public void setRefreshDelay(String str) {
@@ -89,6 +123,14 @@ public class GraphHelper extends FormHandler {
 
     /** @since 0.8.7 */
     public void setPersistent(String foo) { _persistent = true; }
+
+    /**
+     *  For single stat page
+     *  @since 0.9
+     */
+    public void setStat(String stat) {
+        _stat = stat;
+    }
     
     public String getImages() { 
         if (StatSummarizer.isDisabled())
@@ -109,11 +151,11 @@ public class GraphHelper extends FormHandler {
             }
 
             if (hasTx && hasRx && !_showEvents) {
-                _out.write("<a href=\"viewstat.jsp?stat=bw.combined"
+                _out.write("<a href=\"viewstat?stat=bw.combined"
                            + "&amp;periodCount=" + (3 * _periodCount )
                            + "&amp;width=" + (3 * _width)
                            + "&amp;height=" + (3 * _height)
-                           + "\" target=\"_blank\">");
+                           + "\">");
                 String title = _("Combined bandwidth graph");
                 _out.write("<img class=\"statimage\""
                            + " src=\"viewstat.jsp?stat=bw.combined"
@@ -128,14 +170,14 @@ public class GraphHelper extends FormHandler {
                 Rate r = lsnr.getRate();
                 // e.g. "statname for 60m"
                 String title = _("{0} for {1}", r.getRateStat().getName(), DataHelper.formatDuration2(_periodCount * r.getPeriod()));
-                _out.write("<a href=\"viewstat.jsp?stat="
+                _out.write("<a href=\"graph?stat="
                            + r.getRateStat().getName() 
-                           + "&amp;showEvents=" + _showEvents
-                           + "&amp;period=" + r.getPeriod() 
-                           + "&amp;periodCount=" + (3 * _periodCount)
-                           + "&amp;width=" + (3 * _width)
-                           + "&amp;height=" + (3 * _height)
-                           + "\" target=\"_blank\">");
+                           + '.' + r.getPeriod() 
+                           + "&amp;c=" + (3 * _periodCount)
+                           + "&amp;w=" + (3 * _width)
+                           + "&amp;h=" + (3 * _height)
+                           + (_showEvents ? "&amp;showEvents=1" : "")
+                           + "\">");
                 _out.write("<img class=\"statimage\" border=\"0\""
                            + " src=\"viewstat.jsp?stat="
                            + r.getRateStat().getName() 
@@ -156,6 +198,134 @@ public class GraphHelper extends FormHandler {
         return ""; 
     }
 
+    /**
+     *  For single stat page
+     *  @since 0.9
+     */
+    public String getSingleStat() {
+        try {
+            if (StatSummarizer.isDisabled())
+                return "";
+            if (_stat == null) {
+                _out.write("No stat");
+                return "";
+            }
+            List<Rate> rates = StatSummarizer.instance().parseSpecs(_stat);
+            if (rates.size() != 1) {
+                _out.write("Graphs not enabled for " + _stat);
+                return "";
+            }
+            Rate r = rates.get(0);
+            _out.write("<h3>");
+            _out.write(_("{0} for {1}", r.getRateStat().getName(), DataHelper.formatDuration2(_periodCount * r.getPeriod())));
+            if (_end > 0)
+                _out.write(' ' + _("ending {0} ago", DataHelper.formatDuration2(_end * r.getPeriod())));
+
+            _out.write("</h3><p><img class=\"statimage\" border=\"0\""
+                       + " src=\"viewstat.jsp?stat="
+                       + r.getRateStat().getName() 
+                       + "&amp;showEvents=" + _showEvents
+                       + "&amp;period=" + r.getPeriod() 
+                       + "&amp;periodCount=" + _periodCount 
+                       + "&amp;end=" + _end 
+                       + "&amp;width=" + _width
+                       + "&amp;height=" + _height
+                       + "\"></p><p>\n");
+
+            if (_width < MAX_X && _height < MAX_Y) {
+                _out.write(link(_stat, _showEvents, _periodCount, _end, _width * 3 / 2, _height * 3 / 2));
+                _out.write(_("Larger"));
+                _out.write("</a> - ");
+            }
+
+            if (_width > MIN_X && _height > MIN_Y) {
+                _out.write(link(_stat, _showEvents, _periodCount, _end, _width * 2 / 3, _height * 2 / 3));
+                _out.write(_("Smaller"));
+                _out.write("</a> - ");
+            }
+
+            if (_height < MAX_Y) {
+                _out.write(link(_stat, _showEvents, _periodCount, _end, _width, _height * 3 / 2));
+                _out.write(_("Taller"));
+                _out.write("</a> - ");
+            }
+
+            if (_height > MIN_Y) {
+                _out.write(link(_stat, _showEvents, _periodCount, _end, _width, _height * 2 / 3));
+                _out.write(_("Shorter"));
+                _out.write("</a> - ");
+            }
+
+            if (_width < MAX_X) {
+                _out.write(link(_stat, _showEvents, _periodCount, _end, _width * 3 / 2, _height));
+                _out.write(_("Wider"));
+                _out.write("</a> - ");
+            }
+
+            if (_width > MIN_X) {
+                _out.write(link(_stat, _showEvents, _periodCount, _end, _width * 2 / 3, _height));
+                _out.write(_("Narrower"));
+                _out.write("</a>");
+            }
+
+            _out.write("<br>");
+            if (_periodCount < MAX_C) {
+                _out.write(link(_stat, _showEvents, _periodCount * 2, _end, _width, _height));
+                _out.write(_("Larger interval"));
+                _out.write("</a> - ");
+            }
+
+            if (_periodCount > MIN_C) {
+                _out.write(link(_stat, _showEvents, _periodCount / 2, _end, _width, _height));
+                _out.write(_("Smaller interval"));
+                _out.write("</a>");
+            }
+
+            _out.write("<br>");
+            if (_periodCount < MAX_C) {
+                _out.write(link(_stat, _showEvents, _periodCount, _end + _periodCount, _width, _height));
+                _out.write(_("Previous interval"));
+                _out.write("</a>");
+            }
+
+            if (_end > 0) {
+                int end = _end - _periodCount;
+                if (end <= 0)
+                    end = 0;
+                if (_periodCount < MAX_C)
+                    _out.write(" - ");
+                _out.write(link(_stat, _showEvents, _periodCount, end, _width, _height));
+                _out.write(_("Next interval"));
+                _out.write("</a> ");
+            }
+
+            _out.write("<br>");
+            _out.write(link(_stat, !_showEvents, _periodCount, _end, _width, _height));
+            _out.write(_showEvents ? _("Plot averages") : _("plot events"));
+            _out.write("</a>");
+
+            _out.write("</p><p><i>" + _("All times are UTC.") + "</i></p>\n");
+        } catch (IOException ioe) {
+            ioe.printStackTrace();
+        }
+        return ""; 
+    }
+
+    /** @since 0.9 */
+    private static String link(String stat, boolean showEvents,
+                               int periodCount, int end,
+                               int width, int height) {
+        return
+               "<a href=\"graph?stat="
+               + stat
+               + "&amp;c=" + periodCount
+               + "&amp;w=" + width
+               + "&amp;h=" + height
+               + (end > 0 ? "&amp;e=" + end : "")
+               + (showEvents ? "&amp;showEvents=1" : "")
+               + "\">";
+    }
+
     private static final int[] times = { 60, 2*60, 5*60, 10*60, 30*60, 60*60, -1 };
 
     public String getForm() { 
@@ -171,8 +341,8 @@ public class GraphHelper extends FormHandler {
                        "<input type=\"hidden\" name=\"action\" value=\"foo\">\n" +
                        "<input type=\"hidden\" name=\"nonce\" value=\"" + nonce + "\" >\n");
             _out.write(_("Periods") + ": <input size=\"5\" style=\"text-align: right;\" type=\"text\" name=\"periodCount\" value=\"" + _periodCount + "\"><br>\n");
-            _out.write(_("Plot averages") + ": <input type=\"radio\" class=\"optbox\" name=\"showEvents\" value=\"false\" " + (_showEvents ? "" : "checked=\"true\" ") + "> ");
-            _out.write(_("or")+ " " +_("plot events") + ": <input type=\"radio\" class=\"optbox\" name=\"showEvents\" value=\"true\" "+ (_showEvents ? "checked=\"true\" " : "") + "><br>\n");
+            _out.write(_("Plot averages") + ": <input type=\"radio\" class=\"optbox\" name=\"showEvents\" value=\"false\" " + (_showEvents ? "" : "checked=\"checked\" ") + "> ");
+            _out.write(_("or")+ " " +_("plot events") + ": <input type=\"radio\" class=\"optbox\" name=\"showEvents\" value=\"true\" "+ (_showEvents ? "checked=\"checked\" " : "") + "><br>\n");
             _out.write(_("Image sizes") + ": " + _("width") + ": <input size=\"4\" style=\"text-align: right;\" type=\"text\" name=\"width\" value=\"" + _width 
                        + "\"> " + _("pixels") + ", " + _("height") + ": <input size=\"4\" style=\"text-align: right;\" type=\"text\" name=\"height\" value=\"" + _height  
                        + "\"> " + _("pixels") + "<br>\n");
@@ -182,7 +352,7 @@ public class GraphHelper extends FormHandler {
                 _out.write(Integer.toString(times[i]));
                 _out.write("\"");
                 if (times[i] == _refreshDelaySeconds)
-                    _out.write(" selected=\"true\"");
+                    _out.write(" selected=\"selected\"");
                 _out.write(">");
                 if (times[i] > 0)
                     _out.write(DataHelper.formatDuration2(times[i] * 1000));
@@ -195,7 +365,7 @@ public class GraphHelper extends FormHandler {
                        " <input type=\"checkbox\" class=\"optbox\" value=\"true\" name=\"persistent\"");
             boolean persistent = _context.getBooleanPropertyDefaultTrue(SummaryListener.PROP_PERSISTENT);
             if (persistent)
-                _out.write(" checked=\"true\"");
+                _out.write(" checked=\"checked\"");
             _out.write(">" +
                        "<hr><div class=\"formaction\"><input type=\"submit\" class=\"acceot\" value=\"" + _("Save settings and redraw graphs") + "\"></div></form>");
         } catch (IOException ioe) {
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 085ac317217dc75a9d7a30bfe8227114b6c107d5..ebdacbfd69f184b42f86628362a46a024ce45e61 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/NetDbRenderer.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/NetDbRenderer.java
@@ -33,6 +33,7 @@ import net.i2p.data.RouterInfo;
 import net.i2p.router.RouterContext;
 import net.i2p.router.TunnelPoolSettings;
 import net.i2p.router.networkdb.kademlia.HashDistance;   // debug
+import net.i2p.router.networkdb.kademlia.FloodfillNetworkDatabaseFacade;
 import net.i2p.util.HexDump;                             // debug
 import net.i2p.util.ObjectCounter;
 import net.i2p.util.OrderedProperties;
@@ -105,7 +106,10 @@ public class NetDbRenderer {
         StringBuilder buf = new StringBuilder(4*1024);
         buf.append("<h2>" + _("Network Database Contents") + "</h2>\n");
         buf.append("<a href=\"netdb\">" + _("View RouterInfo") + "</a>");
-        buf.append("<h3>").append(_("LeaseSets")).append("</h3>\n");
+        buf.append("<h3>").append(_("LeaseSets"));
+        if (debug)
+            buf.append(" - Debug mode - Sorted by hash distance, closest first");
+        buf.append("</h3>\n");
         Hash ourRKey;
         Set<LeaseSet> leases;
         DecimalFormat fmt;
@@ -120,15 +124,16 @@ public class NetDbRenderer {
         }
         leases.addAll(_context.netDb().getLeases());
         int medianCount = 0;
+        int rapCount = 0;
         BigInteger median = null;
         int c = 0;
         if (debug) {
             // Find the center of the RAP leasesets
             for (LeaseSet ls : leases) {
                 if (ls.getReceivedAsPublished())
-                    medianCount++;
+                    rapCount++;
             }
-            medianCount /= 2;
+            medianCount = rapCount / 2;
         }
         long now = _context.clock().now();
         for (LeaseSet ls : leases) {
@@ -167,9 +172,10 @@ public class NetDbRenderer {
                     if (c++ == medianCount)
                         median = dist;
                 }
-                buf.append(" Dist: <b>").append(fmt.format(biLog2(dist))).append("</b>");
-                buf.append(" RKey: ").append(ls.getRoutingKey().toBase64());
+                buf.append(" Dist: <b>").append(fmt.format(biLog2(dist))).append("</b><br>");
+                buf.append("Routing Key: ").append(ls.getRoutingKey().toBase64());
                 buf.append("<br>");
+                buf.append("Encryption Key: ").append(ls.getEncryptionKey().toBase64().substring(0, 20)).append("...<br>");
             }
             for (int i = 0; i < ls.getLeaseCount(); i++) {
                 buf.append(_("Lease")).append(' ').append(i + 1).append(": " + _("Gateway") + ' ');
@@ -181,18 +187,23 @@ public class NetDbRenderer {
             buf.setLength(0);
         }
         if (debug) {
-            buf.append("<p><b>Total Leasesets: " + leases.size());
-            buf.append("</b></p><p><b>Published (RAP) Leasesets: " + _context.netDb().getKnownLeaseSets());
+            FloodfillNetworkDatabaseFacade netdb = (FloodfillNetworkDatabaseFacade)_context.netDb();
+            buf.append("<p><b>Total Leasesets: ").append(leases.size());
+            buf.append("</b></p><p><b>Published (RAP) Leasesets: ").append(netdb.getKnownLeaseSets());
             //buf.append("</b></p><p><b>Mod Data: " + HexDump.dump(_context.routingKeyGenerator().getModData()));
+            int ff = _context.peerManager().getPeersByCapability(FloodfillNetworkDatabaseFacade.CAPABILITY_FLOODFILL).size();
+            buf.append("</b></p><p><b>Known Floodfills: ").append(ff);
+            buf.append("</b></p><p><b>Currently Floodfill? ");
+            buf.append(netdb.floodfillEnabled() ? "yes" : "no");
             buf.append("</b></p><p><b>Network data (only valid if floodfill):");
             //buf.append("</b></p><p><b>Center of Key Space (router hash): " + ourRKey.toBase64());
             if (median != null) {
                 double log2 = biLog2(median);
-                buf.append("</b></p><p><b>Median distance (bits): " + fmt.format(log2));
+                buf.append("</b></p><p><b>Median distance (bits): ").append(fmt.format(log2));
                 // 3 for 8 floodfills... -1 for median
                 int total = (int) Math.round(Math.pow(2, 3 + 256 - 1 - log2));
-                buf.append("</b></p><p><b>Estimated total floodfills: " + total);
-                buf.append("</b></p><p><b>Estimated network total leasesets: " + (total * leases.size() / 8));
+                buf.append("</b></p><p><b>Estimated total floodfills: ").append(total);
+                buf.append("</b></p><p><b>Estimated total leasesets: ").append(total * rapCount / 8);
             }
             buf.append("</b></p>");
         }
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/StatSummarizer.java b/apps/routerconsole/java/src/net/i2p/router/web/StatSummarizer.java
index d7b0077a9c01a5eb1feea57c43461db07a2f8598..4d740e4b5b02dc0d6c2ff06289e18273a70bb281 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/StatSummarizer.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/StatSummarizer.java
@@ -172,32 +172,37 @@ public class StatSummarizer implements Runnable {
 
     public boolean renderPng(Rate rate, OutputStream out) throws IOException { 
         return renderPng(rate, out, GraphHelper.DEFAULT_X, GraphHelper.DEFAULT_Y,
-                         false, false, false, false, -1, true); 
+                         false, false, false, false, -1, 0, true); 
     }
 
     /**
      *  This does the single data graphs.
      *  For the two-data bandwidth graph see renderRatePng().
      *  Synchronized to conserve memory.
+     *
+     *  @param end number of periods before now
      *  @return success
      */
     public boolean renderPng(Rate rate, OutputStream out, int width, int height, boolean hideLegend,
                                           boolean hideGrid, boolean hideTitle, boolean showEvents, int periodCount,
-                                          boolean showCredit) throws IOException {
+                                          int end, boolean showCredit) throws IOException {
         try {
             try {
                 _sem.acquire();
             } catch (InterruptedException ie) {}
             return locked_renderPng(rate, out, width, height, hideLegend, hideGrid, hideTitle, showEvents,
-                                    periodCount, showCredit);
+                                    periodCount, end, showCredit);
         } finally {
             _sem.release();
         }
     }
 
+    /**
+     *  @param end number of periods before now
+     */
     private boolean locked_renderPng(Rate rate, OutputStream out, int width, int height, boolean hideLegend,
                                           boolean hideGrid, boolean hideTitle, boolean showEvents, int periodCount,
-                                          boolean showCredit) throws IOException {
+                                          int end, boolean showCredit) throws IOException {
         if (width > GraphHelper.MAX_X)
             width = GraphHelper.MAX_X;
         else if (width <= 0)
@@ -206,9 +211,11 @@ public class StatSummarizer implements Runnable {
             height = GraphHelper.MAX_Y;
         else if (height <= 0)
             height = GraphHelper.DEFAULT_Y;
+        if (end < 0)
+            end = 0;
         for (SummaryListener lsnr : _listeners) {
             if (lsnr.getRate().equals(rate)) {
-                lsnr.renderPng(out, width, height, hideLegend, hideGrid, hideTitle, showEvents, periodCount, showCredit);
+                lsnr.renderPng(out, width, height, hideLegend, hideGrid, hideTitle, showEvents, periodCount, end, showCredit);
                 return true;
             }
         }
@@ -368,7 +375,7 @@ public class StatSummarizer implements Runnable {
      * @param specs statName.period,statName.period,statName.period
      * @return list of Rate objects
      */
-    private List<Rate> parseSpecs(String specs) {
+    List<Rate> parseSpecs(String specs) {
         StringTokenizer tok = new StringTokenizer(specs, ",");
         List<Rate> rv = new ArrayList();
         while (tok.hasMoreTokens()) {
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/SummaryListener.java b/apps/routerconsole/java/src/net/i2p/router/web/SummaryListener.java
index 1ea872a88066c813a39f7185b50f005f02da26d9..39290a04224836d38d1b87682effe531314b0129 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/SummaryListener.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/SummaryListener.java
@@ -55,7 +55,7 @@ class SummaryListener implements RateSummaryListener {
     
     static final int PERIODS = 60 * 24;  // 1440
     private static final int MIN_ROWS = PERIODS;
-    private static final int MAX_ROWS = 91 * MIN_ROWS;
+    static final int MAX_ROWS = 91 * MIN_ROWS;
     private static final long THREE_MONTHS = 91l * 24 * 60 * 60 * 1000;
     
     public SummaryListener(Rate r) {
@@ -191,10 +191,15 @@ class SummaryListener implements RateSummaryListener {
         _db = null;
     }
 
-    public void renderPng(OutputStream out, int width, int height, boolean hideLegend, boolean hideGrid, boolean hideTitle, boolean showEvents, int periodCount, boolean showCredit) throws IOException {
+    /**
+     *  @param end number of periods before now
+     */
+    public void renderPng(OutputStream out, int width, int height, boolean hideLegend, boolean hideGrid,
+                          boolean hideTitle, boolean showEvents, int periodCount,
+                          int end, boolean showCredit) throws IOException {
         if (_renderer == null || _db == null)
             throw new IOException("No RRD, check logs for previous errors");
-        _renderer.render(out, width, height, hideLegend, hideGrid, hideTitle, showEvents, periodCount, showCredit); 
+        _renderer.render(out, width, height, hideLegend, hideGrid, hideTitle, showEvents, periodCount, end, showCredit); 
     }
 
     public void renderPng(OutputStream out) throws IOException {
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/SummaryRenderer.java b/apps/routerconsole/java/src/net/i2p/router/web/SummaryRenderer.java
index 02b4e51d3843bd63e27c373c77c4c3a490a9d45f..35845f7c7c8c9b063b9ed46508e46a03ac20abf5 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/SummaryRenderer.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/SummaryRenderer.java
@@ -81,13 +81,21 @@ class SummaryRenderer {
     }
 
     public void render(OutputStream out) throws IOException { render(out, GraphHelper.DEFAULT_X, GraphHelper.DEFAULT_Y,
-                                                                     false, false, false, false, -1, false); }
+                                                                     false, false, false, false, -1, 0, false); }
 
-    public void render(OutputStream out, int width, int height, boolean hideLegend, boolean hideGrid, boolean hideTitle, boolean showEvents, int periodCount, boolean showCredit) throws IOException {
+    /**
+     *  @param endp number of periods before now
+     */
+    public void render(OutputStream out, int width, int height, boolean hideLegend, boolean hideGrid,
+                       boolean hideTitle, boolean showEvents, int periodCount,
+                       int endp, boolean showCredit) throws IOException {
         long end = _listener.now() - 75*1000;
+        long period = _listener.getRate().getPeriod();
+        if (endp > 0)
+            end -= period * endp;
         if (periodCount <= 0 || periodCount > _listener.getRows())
             periodCount = _listener.getRows();
-        long start = end - _listener.getRate().getPeriod()*periodCount;
+        long start = end - (period * periodCount);
         //long begin = System.currentTimeMillis();
         try {
             RrdGraphDef def = new RrdGraphDef();
@@ -103,9 +111,9 @@ class SummaryRenderer {
                 String p;
                 // we want the formatting and translation of formatDuration2(), except not zh, and not the &nbsp;
                 if (IS_WIN && "zh".equals(Messages.getLanguage(_context)))
-                    p = DataHelper.formatDuration(_listener.getRate().getPeriod());
+                    p = DataHelper.formatDuration(period);
                 else
-                    p = DataHelper.formatDuration2(_listener.getRate().getPeriod()).replace("&nbsp;", " ");
+                    p = DataHelper.formatDuration2(period).replace("&nbsp;", " ");
                 if (showEvents)
                     title = name + ' ' + _("events in {0}", p);
                 else
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/TunnelRenderer.java b/apps/routerconsole/java/src/net/i2p/router/web/TunnelRenderer.java
index 73efe19de47fe39e5c29e6e1b2c1f3c390ac1653..fdc670f53490d977e393925ab926221dad0519cd 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/TunnelRenderer.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/TunnelRenderer.java
@@ -134,6 +134,7 @@ public class TunnelRenderer {
         out.write("<div class=\"statusnotes\"><b>" + _("Inactive participating tunnels") + ": " + inactive + "</b></div>\n");
         out.write("<div class=\"statusnotes\"><b>" + _("Lifetime bandwidth usage") + ": " + DataHelper.formatSize2(processed*1024) + "B</b></div>\n");
         //renderPeers(out);
+        out.write("</div>");
     }
     
     private static class TunnelComparator implements Comparator<HopConfig> {
@@ -222,7 +223,7 @@ public class TunnelRenderer {
             }
         }
         if (live <= 0)
-            out.write("<div class=\"statusnotes\"><center><b>" + _("No tunnels; waiting for the grace period to end.") + "</center></b></div>\n");
+            out.write("<div class=\"statusnotes\"><center><b>" + _("No tunnels; waiting for the grace period to end.") + "</b></center></div>\n");
         out.write("<div class=\"statusnotes\"><center><b>" + _("Lifetime bandwidth usage") + ": " +
                   DataHelper.formatSize2(processedIn*1024) + "B " + _("in") + ", " +
                   DataHelper.formatSize2(processedOut*1024) + "B " + _("out") + "</b></center></div>");
diff --git a/apps/routerconsole/jsp/configreseed.jsp b/apps/routerconsole/jsp/configreseed.jsp
index 648a14ee953e7678e0d56d5c3a86c07b2bd92d99..2a08d1015fc78be61524aeffea2b785929c8ab30 100644
--- a/apps/routerconsole/jsp/configreseed.jsp
+++ b/apps/routerconsole/jsp/configreseed.jsp
@@ -41,7 +41,7 @@
 <input type="radio" class="optbox" name="mode" value="2" <%=reseedHelper.modeChecked(2) %> >
 <b><%=intl._("Use non-SSL only")%></b></td></tr>
 <tr><td class="mediumtags" align="right"><b><%=intl._("Reseed URLs")%>:</b></td>
-<td><textarea name="reseedURL" wrap="off" spellcheck="false"><jsp:getProperty name="reseedHelper" property="reseedURL" /></textarea></td></tr>
+<td><textarea wrap="off" name="reseedURL" cols="60" rows="7" spellcheck="false"><jsp:getProperty name="reseedHelper" property="reseedURL" /></textarea></td></tr>
 
 <tr><td class="mediumtags" align="right"><b><%=intl._("Enable HTTP Proxy?")%></b></td>
 <td><input type="checkbox" class="optbox" name="enable" value="true" <jsp:getProperty name="reseedHelper" property="enable" /> ></td></tr>
diff --git a/apps/routerconsole/jsp/configstats.jsp b/apps/routerconsole/jsp/configstats.jsp
index 7e548a11361c6323e5e15112855c8e03dd36fe13..f284badae45030608674f402edeecbca4e54c168 100644
--- a/apps/routerconsole/jsp/configstats.jsp
+++ b/apps/routerconsole/jsp/configstats.jsp
@@ -76,10 +76,21 @@ function toggleAll(category)
  <h3><%=intl._("Configure I2P Stat Collection")%></h3>
  <p><%=intl._("Enable full stats?")%>
  <input type="checkbox" class="optbox" name="isFull" value="true" <%
- if (statshelper.getIsFull()) { %>checked="true" <% } %> >
+ if (statshelper.getIsFull()) { %>checked="checked" <% } %> >
  (<%=intl._("change requires restart to take effect")%>)<br>
- <%=intl._("Stat file")%>: <input type="text" name="filename" value="<%=statshelper.getFilename()%>" ><br>
-<%=intl._("Filter")%>: (<a href="javascript:void(null);" onclick="toggleAll('*')"><%=intl._("toggle all")%></a>)<br></p>
+<%
+
+  // stats.log for devs only and grows without bounds, not recommended
+  boolean shouldShowLog = statshelper.shouldShowLog();
+  if (shouldShowLog) {
+
+%><%=intl._("Stat file")%>: <input type="text" name="filename" value="<%=statshelper.getFilename()%>" ><br>
+Warning - Log with care, stat file grows without limit.<br>
+<%
+
+  }  // shouldShowLog
+
+%><%=intl._("Filter")%>: (<a href="javascript:void(null);" onclick="toggleAll('*')"><%=intl._("toggle all")%></a>)<br></p>
  <div class="wideload">
  <table>
  <% while (statshelper.hasMoreStats()) {
@@ -90,27 +101,51 @@ function toggleAll(category)
      (<a href="javascript:void(null);" onclick="toggleAll('<%=statshelper.getCurrentGroupName()%>')"><%=intl._("toggle all")%></a>)
      </td></tr>
  <tr class="tablefooter">
-    <td align="center"><b><%=intl._("Log")%></b></td>
-    <td align="center"><b><%=intl._("Graph")%></b></td>
+<%
+
+  if (shouldShowLog) {
+
+%>  <td align="center"><b><%=intl._("Log")%></b></td>
+<%
+
+  }  // shouldShowLog
+
+%>    <td align="center"><b><%=intl._("Graph")%></b></td>
     <td></td></tr>
         <%
      } // end iterating over required groups for the current stat %>
- <tr><td align="center">
+ <tr>
+<%
+
+  if (shouldShowLog) {
+
+%>   <td align="center">
      <a name="<%=statshelper.getCurrentStatName()%>"></a>
      <input type="checkbox" class="optbox <%=statshelper.getCurrentGroupName()%>" name="statList" value="<%=statshelper.getCurrentStatName()%>" <%
-     if (statshelper.getCurrentIsLogged()) { %>checked="true" <% } %> ></td>
-     <td align="center">
+     if (statshelper.getCurrentIsLogged()) { %>checked="checked" <% } %> ></td>
+<%
+
+  }  // shouldShowLog
+
+%>   <td align="center">
      <% if (statshelper.getCurrentCanBeGraphed()) { %>
        <input type="checkbox" class="optbox <%=statshelper.getCurrentGroupName()%>" name="graphList" value="<%=statshelper.getCurrentGraphName()%>" <%
-       if (statshelper.getCurrentIsGraphed()) { %>checked="true" <% } %> ><% } %></td>
+       if (statshelper.getCurrentIsGraphed()) { %>checked="checked" <% } %> ><% } %></td>
      <td align="left"><b><%=statshelper.getCurrentStatName()%>:</b><br>
      <%=statshelper.getCurrentStatDescription()%></td></tr><%
-    } // end iterating over all stats %>
- <tr><td colspan="3"></td></tr>
+    } // end iterating over all stats
+
+  if (shouldShowLog) {
+
+%> <tr><td colspan="3"></td></tr>
  <tr><td align="center"><input type="checkbox" class="optbox" name="explicitFilter" ></td>
      <td colspan="2"><%=intl._("Advanced filter")%>:
      <input type="text" name="explicitFilterValue" value="<%=statshelper.getExplicitFilter()%>" size="40" ></td></tr>
-     <tr class="tablefooter"><td colspan="3" align="right">
+<%
+
+  }  // shouldShowLog
+
+%>   <tr class="tablefooter"><td colspan="3" align="right">
 <input type="reset" class="cancel" value="<%=intl._("Cancel")%>" >
 <input type="submit" name="shouldsave" class="accept" value="<%=intl._("Save changes")%>" >
 </td></tr>
diff --git a/apps/routerconsole/jsp/configupdate.jsp b/apps/routerconsole/jsp/configupdate.jsp
index c3066ade634da3c58e851ccca958709539aa5084..465f9b412e370617c2aefda6147b3d9a491675f1 100644
--- a/apps/routerconsole/jsp/configupdate.jsp
+++ b/apps/routerconsole/jsp/configupdate.jsp
@@ -56,9 +56,9 @@
           <td><input type="text" size="10" name="proxyPort" value="<jsp:getProperty name="updatehelper" property="proxyPort" />" /></td></tr>
     <% if (updatehelper.canInstall()) { %>
         <tr><td class= "mediumtags" align="right"><b><%=intl._("Update URLs")%>:</b></td>
-          <td><textarea name="updateURL" wrap="off" spellcheck="false"><jsp:getProperty name="updatehelper" property="updateURL" /></textarea></td>
+          <td><textarea cols="60" rows="6" name="updateURL" wrap="off" spellcheck="false"><jsp:getProperty name="updatehelper" property="updateURL" /></textarea></td>
         </tr><tr><td class= "mediumtags" align="right"><b><%=intl._("Trusted keys")%>:</b></td>
-          <td><textarea name="trustedKeys" wrap="off" spellcheck="false"><jsp:getProperty name="updatehelper" property="trustedKeys" /></textarea></td>
+          <td><textarea cols="60" rows="6" name="trustedKeys" wrap="off" spellcheck="false"><jsp:getProperty name="updatehelper" property="trustedKeys" /></textarea></td>
         </tr><tr><td class= "mediumtags" align="right"><b><%=intl._("Update with unsigned development builds?")%></b></td>
           <td><jsp:getProperty name="updatehelper" property="updateUnsigned" /></td>
         </tr><tr><td class= "mediumtags" align="right"><b><%=intl._("Unsigned Build URL")%>:</b></td>
diff --git a/apps/routerconsole/jsp/graph.jsp b/apps/routerconsole/jsp/graph.jsp
new file mode 100644
index 0000000000000000000000000000000000000000..6147fd96d0fb75325a45ce6d208bd751c1b80a27
--- /dev/null
+++ b/apps/routerconsole/jsp/graph.jsp
@@ -0,0 +1,22 @@
+<%@page contentType="text/html"%>
+<%@page pageEncoding="UTF-8"%>
+<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
+
+<html><head>
+<%@include file="css.jsi" %>
+<%=intl.title("graphs")%>
+ <jsp:useBean class="net.i2p.router.web.GraphHelper" id="graphHelper" scope="request" />
+ <jsp:setProperty name="graphHelper" property="contextId" value="<%=(String)session.getAttribute(\"i2p.contextId\")%>" />
+<% /* GraphHelper sets the defaults in setContextId, so setting the properties must be after the context */ %>
+ <jsp:setProperty name="graphHelper" property="*" />
+<%
+    graphHelper.storeWriter(out);
+%>
+</head><body>
+<%@include file="summary.jsi" %>
+<h1><%=intl._("I2P Performance Graphs")%></h1>
+<div class="main" id="main">
+ <div class="graphspanel">
+ <div class="widepanel">
+ <jsp:getProperty name="graphHelper" property="singleStat" />
+</div></div></div></body></html>
diff --git a/apps/routerconsole/jsp/graphs.jsp b/apps/routerconsole/jsp/graphs.jsp
index e54133250a7233399a9c3e696fae2a2daf3f3324..24dd94185df4c43f7996ec3077f4361fe8e2f9bf 100644
--- a/apps/routerconsole/jsp/graphs.jsp
+++ b/apps/routerconsole/jsp/graphs.jsp
@@ -6,7 +6,6 @@
 <%@include file="css.jsi" %>
 <%=intl.title("graphs")%>
  <jsp:useBean class="net.i2p.router.web.GraphHelper" id="graphHelper" scope="request" />
- <% graphHelper.storeMethod(request.getMethod()); %>
  <jsp:setProperty name="graphHelper" property="contextId" value="<%=(String)session.getAttribute("i2p.contextId")%>" />
 <% /* GraphHelper sets the defaults in setContextId, so setting the properties must be after the context */ %>
  <jsp:setProperty name="graphHelper" property="*" />
diff --git a/apps/routerconsole/jsp/js/ajax.js b/apps/routerconsole/jsp/js/ajax.js
new file mode 100644
index 0000000000000000000000000000000000000000..dab3164a098fd00dfcc3653fa9e4af40a5001dc8
--- /dev/null
+++ b/apps/routerconsole/jsp/js/ajax.js
@@ -0,0 +1,33 @@
+function ajax(url, target, refresh) {
+  // native XMLHttpRequest object
+  if (window.XMLHttpRequest) {
+    req = new XMLHttpRequest();
+    req.onreadystatechange = function() {ajaxDone(url, target, refresh);};
+    req.open("GET", url, true);
+    req.send(null);
+    // IE/Windows ActiveX version
+  } else if (window.ActiveXObject) {
+    req = new ActiveXObject("Microsoft.XMLDOM");
+    if (req) {
+      req.onreadystatechange = function() {ajaxDone(target);};
+      req.open("GET", url, true);
+      req.send(null);
+    }
+  }
+}
+
+function ajaxDone(url, target, refresh) {
+  // only if req is "loaded"
+  if (req.readyState == 4) {
+    // only if "OK"
+    if (req.status == 200) {
+      results = req.responseText;
+      document.getElementById(target).innerHTML = results;
+      document.getElementById("lowersection").style.display="block";
+    } else {
+      document.getElementById(target).innerHTML="<b>Router is down</b>";
+      document.getElementById("lowersection").style.display="none";
+    }
+    setTimeout(function() {ajax(url, target, refresh);}, refresh);
+  }
+}
diff --git a/apps/routerconsole/jsp/viewstat.jsp b/apps/routerconsole/jsp/viewstat.jsp
index ef5f1140fec0bc2c2c51baf6e70efac0f66644d3..af7b2944cd2fcc542b932135928bdf9e5c447352 100644
--- a/apps/routerconsole/jsp/viewstat.jsp
+++ b/apps/routerconsole/jsp/viewstat.jsp
@@ -47,12 +47,15 @@ if ( !rendered && ((rs != null) || fakeBw) ) {
         int width = -1;
         int height = -1;
         int periodCount = -1;
+        int end = 0;
         String str = request.getParameter("width");
         if (str != null) try { width = Integer.parseInt(str); } catch (NumberFormatException nfe) {}
         str = request.getParameter("height");
         if (str != null) try { height = Integer.parseInt(str); } catch (NumberFormatException nfe) {}
         str = request.getParameter("periodCount");
         if (str != null) try { periodCount = Integer.parseInt(str); } catch (NumberFormatException nfe) {}
+        str = request.getParameter("end");
+        if (str != null) try { end = Integer.parseInt(str); } catch (NumberFormatException nfe) {}
         boolean hideLegend = Boolean.valueOf(""+request.getParameter("hideLegend")).booleanValue();
         boolean hideGrid = Boolean.valueOf(""+request.getParameter("hideGrid")).booleanValue();
         boolean hideTitle = Boolean.valueOf(""+request.getParameter("hideTitle")).booleanValue();
@@ -63,7 +66,7 @@ if ( !rendered && ((rs != null) || fakeBw) ) {
         if (fakeBw)
             rendered = net.i2p.router.web.StatSummarizer.instance().renderRatePng(cout, width, height, hideLegend, hideGrid, hideTitle, showEvents, periodCount, showCredit);
         else
-            rendered = net.i2p.router.web.StatSummarizer.instance().renderPng(rate, cout, width, height, hideLegend, hideGrid, hideTitle, showEvents, periodCount, showCredit);
+            rendered = net.i2p.router.web.StatSummarizer.instance().renderPng(rate, cout, width, height, hideLegend, hideGrid, hideTitle, showEvents, periodCount, end, showCredit);
       }
       if (rendered)
         cout.close();
diff --git a/apps/sam/java/build.xml b/apps/sam/java/build.xml
index 849fbd55d74830eb90573e1f601af9c1b4bf79d5..076c2e4533bc301576880ff5c39824a11f55aecc 100644
--- a/apps/sam/java/build.xml
+++ b/apps/sam/java/build.xml
@@ -72,6 +72,7 @@
                 <attribute name="Main-Class" value="net.i2p.sam.SAMBridge" />
                 <attribute name="Class-Path" value="i2p.jar mstreaming.jar streaming.jar" />
                 <attribute name="Implementation-Version" value="${full.version}" />
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.tr}" />
@@ -87,6 +88,7 @@
                 <attribute name="Main-Class" value="net.i2p.sam.SAMBridge" />
                 <attribute name="Class-Path" value="i2p.jar mstreaming.jar streaming.jar" />
                 <attribute name="Implementation-Version" value="${full.version}" />
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.tr}" />
diff --git a/apps/streaming/java/build.xml b/apps/streaming/java/build.xml
index 5f278f307c67183780e29bb28cb3294c0721d6df..971ae21cf2df0b3e0db0a7f024478fdd79d83583 100644
--- a/apps/streaming/java/build.xml
+++ b/apps/streaming/java/build.xml
@@ -64,6 +64,7 @@
         <jar destfile="./build/streaming.jar" basedir="./build/obj" includes="**/*.class" >
             <manifest>
                 <attribute name="Implementation-Version" value="${full.version}" />
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.tr}" />
diff --git a/apps/streaming/java/src/net/i2p/client/streaming/ConnectionManager.java b/apps/streaming/java/src/net/i2p/client/streaming/ConnectionManager.java
index f50a4b5cc8115ab55e0e8acc7d69369406f4f3f5..6e8811a2fea1fa79f73be5f5b4694359f306125a 100644
--- a/apps/streaming/java/src/net/i2p/client/streaming/ConnectionManager.java
+++ b/apps/streaming/java/src/net/i2p/client/streaming/ConnectionManager.java
@@ -161,8 +161,7 @@ class ConnectionManager {
             //        active++;
             //}
             if (locked_tooManyStreams()) {
-                if (_log.shouldLog(Log.WARN))
-                    _log.warn("Refusing connection since we have exceeded our max of " 
+                _log.logAlways(Log.WARN, "Refusing connection since we have exceeded our max of " 
                               + _maxConcurrentStreams + " connections");
                 reject = true;
             } else {
@@ -233,8 +232,7 @@ class ConnectionManager {
         while (true) {
             long remaining = expiration - _context.clock().now();
             if (remaining <= 0) { 
-                if (_log.shouldLog(Log.WARN))
-                _log.warn("Refusing to connect since we have exceeded our max of " 
+                _log.logAlways(Log.WARN, "Refusing to connect since we have exceeded our max of " 
                           + _maxConcurrentStreams + " connections");
                 _numWaiting--;
                 return null;
@@ -243,8 +241,7 @@ class ConnectionManager {
                 if (locked_tooManyStreams()) {
                     // allow a full buffer of pending/waiting streams
                     if (_numWaiting > _maxConcurrentStreams) {
-                        if (_log.shouldLog(Log.WARN))
-                            _log.warn("Refusing connection since we have exceeded our max of "
+                        _log.logAlways(Log.WARN, "Refusing connection since we have exceeded our max of "
                                       + _maxConcurrentStreams + " and there are " + _numWaiting
                                       + " waiting already");
                         _numWaiting--;
diff --git a/apps/susidns/src/build.xml b/apps/susidns/src/build.xml
index 3fff55a34c9fce7f4e652c65027842ddd79d6a28..49a1629c35ec8869e0d3057844ed56679be79066 100644
--- a/apps/susidns/src/build.xml
+++ b/apps/susidns/src/build.xml
@@ -98,6 +98,7 @@
         	</fileset>
             <manifest>
                 <attribute name="Implementation-Version" value="${full.version}" />
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.tr}" />
diff --git a/apps/susidns/src/java/src/i2p/susi/dns/AddressBean.java b/apps/susidns/src/java/src/i2p/susi/dns/AddressBean.java
index 6f34af3e5b51a889705017d228cd3b1112712ea1..9c1c85863194cf9bf0163df6bdff55a0f0b6038d 100644
--- a/apps/susidns/src/java/src/i2p/susi/dns/AddressBean.java
+++ b/apps/susidns/src/java/src/i2p/susi/dns/AddressBean.java
@@ -170,6 +170,15 @@ public class AddressBean
 		return Base32.encode(hash) + ".b32.i2p";
 	}
 
+	/** @since 0.9 */
+	public String getB64() 
+	{
+		byte[] dest = Base64.decode(destination);
+		if (dest == null)
+			return "";
+		return I2PAppContext.getGlobalContext().sha().calculateHash(dest).toBase64();
+	}
+
 	/** @since 0.8.7 */
 	public void setProperties(Properties p) {
 		props = p;
diff --git a/apps/susidns/src/jsp/addressbook.jsp b/apps/susidns/src/jsp/addressbook.jsp
index 4736a4dc01c0bf78f99b4879cbd32a508e5cb14c..e0a5de3148438bc87452858ae7ddb6e078c4610d 100644
--- a/apps/susidns/src/jsp/addressbook.jsp
+++ b/apps/susidns/src/jsp/addressbook.jsp
@@ -42,7 +42,7 @@
 <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
 <html>
 <head>
-<meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />
+<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
 <title>${book.book} <%=intl._("address book")%> - susidns</title>
 <link rel="stylesheet" type="text/css" href="css.css">
 </head>
@@ -158,7 +158,7 @@ ${book.loadBookMessages}
 </td><td class="names">
 <span class="addrhlpr"><a href="details?h=${addr.name}" title="<%=intl._("More information on this entry")%>"><%=intl._("details")%></a></span>
 </td>
-<td class="destinations"><textarea rows="1" style="height: 3em;" cols="40" wrap="off" readonly="readonly" name="dest_${addr.name}" >${addr.destination}</textarea></td>
+<td class="destinations"><textarea rows="1" style="height:3em;" wrap="off" cols="40" readonly="readonly" name="dest_${addr.name}" >${addr.destination}</textarea></td>
 </tr>
 </c:forEach>
 </table>
@@ -190,7 +190,7 @@ ${book.loadBookMessages}
 <table><tr><td>
 <b><%=intl._("Host Name")%></b></td><td><input type="text" name="hostname" value="${book.hostname}" size="54">
 </td></tr><tr><td>
-<b><%=intl._("Destination")%></b></td><td><textarea name="destination" rows="1" style="height: 3em;" cols="70" wrap="off" spellcheck="false">${book.destination}</textarea>
+<b><%=intl._("Destination")%></b></td><td><textarea name="destination" rows="1" style="height:3em" wrap="off" cols="70" spellcheck="false">${book.destination}</textarea>
 </td></tr></table>
 <p class="buttons">
 <input class="cancel" type="reset" value="<%=intl._("Cancel")%>" >
diff --git a/apps/susidns/src/jsp/config.jsp b/apps/susidns/src/jsp/config.jsp
index 23cce5adadc4900af64b849084b49ce547f69427..a94e693e5420f5ff1199154014dd75161cc9872a 100644
--- a/apps/susidns/src/jsp/config.jsp
+++ b/apps/susidns/src/jsp/config.jsp
@@ -38,7 +38,7 @@
 <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
 <html>
 <head>
-<meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />
+<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
 <title><%=intl._("configuration")%> - susidns</title>
 <link rel="stylesheet" type="text/css" href="css.css">
 </head>
diff --git a/apps/susidns/src/jsp/details.jsp b/apps/susidns/src/jsp/details.jsp
index f9769ecac13f3b655cf730edcbed1a6de7b87552..49a4b7b6340050fec85a0573b82a838195e843f3 100644
--- a/apps/susidns/src/jsp/details.jsp
+++ b/apps/susidns/src/jsp/details.jsp
@@ -36,7 +36,7 @@
 <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
 <html>
 <head>
-<meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />
+<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
 <title>${book.book} <%=intl._("addressbook")%> - susidns</title>
 <link rel="stylesheet" type="text/css" href="css.css">
 </head>
@@ -94,6 +94,9 @@
 <td><%=intl._("Base 32 Address")%></td>
 <td><a href="http://<%=b32%>/"><%=b32%></a></td>
 </tr><tr class="list${book.trClass}">
+<td><%=intl._("Base 64 Hash")%></td>
+<td><%=addr.getB64()%></td>
+</tr><tr class="list${book.trClass}">
 <td><%=intl._("Address Helper")%></td>
 <td><a href="http://<%=addr.getName()%>/?i2paddresshelper=<%=addr.getDestination()%>"><%=intl._("link")%></a></td>
 </tr><tr class="list${book.trClass}">
@@ -119,7 +122,7 @@
 <td><%=addr.getNotes()%></td>
 </tr><tr class="list${book.trClass}">
 <td><%=intl._("Destination")%></td>
-<td class="destinations"><textarea rows="1" style="height: 3em;" cols="70" wrap="off" readonly="readonly" ><%=addr.getDestination()%></textarea></td>
+<td class="destinations"><textarea rows="1" style="height:3em;" wrap="off" cols="70" readonly="readonly" ><%=addr.getDestination()%></textarea></td>
 </tr></table>
 </div>
 <div id="buttons">
diff --git a/apps/susidns/src/jsp/subscriptions.jsp b/apps/susidns/src/jsp/subscriptions.jsp
index 8d1c6c24894a4fad5920c93a631687e5e5f8bf7c..c01e2ee08e9e194d060bb4a6b03f08093569a6f3 100644
--- a/apps/susidns/src/jsp/subscriptions.jsp
+++ b/apps/susidns/src/jsp/subscriptions.jsp
@@ -38,7 +38,7 @@
 <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
 <html>
 <head>
-<meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />
+<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
 <title><%=intl._("subscriptions")%> - susidns</title>
 <link rel="stylesheet" type="text/css" href="css.css">
 </head>
diff --git a/apps/susimail/build.xml b/apps/susimail/build.xml
index 3368283d115e8b65a72f790f99e4c982d7f4b75d..3e22444d892e06b92255376f76d105ec05d696a5 100644
--- a/apps/susimail/build.xml
+++ b/apps/susimail/build.xml
@@ -46,6 +46,7 @@
              basedir="src/" excludes="WEB-INF/web.xml LICENSE src/**/*">
             <manifest>
                 <attribute name="Implementation-Version" value="${full.version}" />
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.tr}" />
diff --git a/apps/systray/java/build.xml b/apps/systray/java/build.xml
index 5baf4ab6224a5b7183af97e6a43053b9b3e78180..eeec392303a9f3da648a15c98cd0d166953bdee9 100644
--- a/apps/systray/java/build.xml
+++ b/apps/systray/java/build.xml
@@ -45,6 +45,7 @@
                 <attribute name="Main-Class" value="net.i2p.apps.systray.SysTray" />
                 <attribute name="Class-Path" value="systray4j.jar" />
                 <attribute name="Implementation-Version" value="${full.version}" />
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.tr}" />
diff --git a/build.properties b/build.properties
index f3ead757658b05a21cdbe376cf30d28353927db9..ab2fa1b9916c9f85378fdaad6506dbcd6b6e4b60 100644
--- a/build.properties
+++ b/build.properties
@@ -16,3 +16,5 @@ wrapperdocs.url=http://wrapper.tanukisoftware.com/jdoc/
 # these are only for unit test javadocs
 i2pdocs.url=http://docs.i2p-projekt.de/javadoc/
 junitdocs.url=http://junit.org/apidocs/
+# This will go in the jar manifests
+build.built-by=unknown	
diff --git a/build.xml b/build.xml
index a0ffe8ab5486b11d18cf48d9a27f9a3372408e4f..150db9b8594a824d6af01c6cef84ca0409931594 100644
--- a/build.xml
+++ b/build.xml
@@ -166,6 +166,7 @@
             <zipfileset src="apps/jrobin/jrobin-1.5.9.1.jar"
                         excludes="org/jrobin/cmd/ org/jrobin/convertor/ org/jrobin/inspector/" />
             <manifest>
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
 	    </manifest>
@@ -206,6 +207,7 @@
         <copy file="apps/jetty/jettylib/javax.servlet.jar" todir="build/" />
     </target>
 
+    <!-- this makes an empty build/launchi2p.jar and the build/i2p.exe for the no-wrapper windows startup, if possible -->
     <target name="buildexe">
         <condition property="noExe">
 	    <os arch="x86_64" />
@@ -234,11 +236,13 @@
              classname="net.sf.launch4j.ant.Launch4jTask"
              classpath="${basedir}/installer/lib/launch4j/launch4j.jar:${basedir}/installer/lib/launch4j/lib/xstream.jar" />
 
+    <!-- this makes an empty build/launchi2p.jar and the build/i2p.exe for the no-wrapper windows startup -->
     <target name="doBuildEXE" depends="buildProperties" unless="noExe">
         <jar destfile="./build/launchi2p.jar">
             <manifest>
 	     <attribute name="Main-Class" value="net.i2p.router.RouterLaunch" />
 	     <attribute name="Class-Path" value="lib/i2p.jar lib/router.jar lib/jbigi.jar lib/BOB.jar lib/sam.jar lib/mstreaming.jar lib/streaming.jar lib/routerconsole.jar lib/i2ptunnel.jar lib/org.mortbay.jetty.jar lib/javax.servlet.jar lib/jasper-compiler.jar lib/jasper-runtime.jar lib/commons-logging.jar lib/commons-el.jar lib/wrapper.jar lib/systray.jar lib/systray4j.jar lib/desktopgui.jar" />
+             <attribute name="Built-By" value="${build.built-by}" />
              <attribute name="Build-Date" value="${build.timestamp}" />
              <attribute name="Base-Revision" value="${workspace.version}" />
 	    </manifest>
@@ -289,6 +293,7 @@
         <jar destfile="build/jbigi.jar" whenmanifestonly="fail" >
             <fileset dir="installer/lib/jbigi" includes="*.so *.dll *.jnilib" />
             <manifest>
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.jbigi.tr}" />
@@ -302,6 +307,7 @@
         <jar destfile="build/jbigi.jar" whenmanifestonly="fail" >
             <fileset dir="installer/lib/jbigi" includes="*.so *.jnilib" />
             <manifest>
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.jbigi.tr}" />
@@ -316,6 +322,7 @@
         <jar destfile="build/jbigi.jar"  whenmanifestonly="fail" >
             <fileset dir="installer/lib/jbigi" includes="*freebsd*.so" />
             <manifest>
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.jbigi.tr}" />
@@ -329,6 +336,7 @@
         <jar destfile="build/jbigi.jar"  whenmanifestonly="fail" >
             <fileset dir="installer/lib/jbigi" includes="*linux*.so" />
             <manifest>
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.jbigi.tr}" />
@@ -342,6 +350,7 @@
         <jar destfile="build/jbigi.jar"  whenmanifestonly="fail" >
             <fileset dir="installer/lib/jbigi" includes="*linux-arm*.so,*linux-ppc*.so" />
             <manifest>
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.jbigi.tr}" />
@@ -355,6 +364,7 @@
         <jar destfile="build/jbigi.jar"  whenmanifestonly="fail" >
             <fileset dir="installer/lib/jbigi" includes="*.jnilib" />
             <manifest>
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.jbigi.tr}" />
@@ -368,6 +378,7 @@
         <jar destfile="build/jbigi.jar" whenmanifestonly="fail" >
             <fileset dir="installer/lib/jbigi" includes="*windows*.dll" />
             <manifest>
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.jbigi.tr}" />
@@ -544,6 +555,7 @@
     </target>
 
     <target name="verifyReleaseBuildNumbers" depends="getReleaseNumber, getBuildNumber" >
+        <echo message="SDK: ${java.vendor} ${java.version} (${java.runtime.name} ${java.runtime.version})" />
         <fail message="Bad release number: ${release.number}" >
             <condition>
                 <or>
@@ -572,7 +584,6 @@
         <delete dir="./build" />
         <delete file="installer/lib/izpack/patches.jar" failonerror="false" quiet="true" />	
         <delete file="i2pinstall.exe" failonerror="false" quiet="true" />	
-        <delete file="i2p.exe" failonerror="false" quiet="true" />
         <delete file="syndie-standalone.zip" failonerror="false" quiet="true" />
         <delete>
             <fileset dir="." includes="i2pinstall*jar i2pinstall*bz2" />
@@ -754,7 +765,7 @@
     </target>
     
     <target name="preppkg-windows" depends="preppkg-base, buildexe">
-        <copy file="i2p.exe" todir="pkg-temp/" failonerror="false" />
+        <copy file="build/i2p.exe" todir="pkg-temp/" failonerror="false" />
         <copy file="apps/systray/java/lib/systray4j.dll" todir="pkg-temp/lib" />
         <copy file="apps/systray/java/resources/iggy.ico" todir="pkg-temp/icons" />
         <copy file="apps/systray/java/resources/iggy.xpm" todir="pkg-temp/icons" />
@@ -1016,7 +1027,9 @@
         <copy file="build/i2psnark.jar" todir="pkg-temp/lib" />
         <!-- include systray changes in 0.7.5 -->
         <copy file="build/systray.jar" todir="pkg-temp/lib/" />
+       <!-- removed from updater in 0.9
         <copy file="build/desktopgui.jar" todir="pkg-temp/lib/" />
+       -->
         <copy file="build/susimail.war" todir="pkg-temp/webapps/" />
         <copy file="build/susidns.war" todir="pkg-temp/webapps/" />
         <!-- as of 0.7.12; someday, we can remove these from the updater -->
@@ -1116,6 +1129,7 @@
         <jar destfile="./pkg-temp/installer/copy.jar" basedir="./core/java/build/obj" includes="net/i2p/util/Copy.class net/i2p/util/FileUtil.class">
             <manifest>
                 <attribute name="Main-Class" value="net.i2p.util.Copy" />
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.util.tr}" />
@@ -1124,6 +1138,7 @@
         <jar destfile="./pkg-temp/installer/delete.jar" basedir="./core/java/build/obj" includes="net/i2p/util/Delete.class net/i2p/util/FileUtil.class">
             <manifest>
                 <attribute name="Main-Class" value="net.i2p.util.Delete" />
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.util.tr}" />
@@ -1132,6 +1147,7 @@
         <jar destfile="./pkg-temp/installer/exec.jar" basedir="./core/java/build/obj" includes="net/i2p/util/Exec.class">
             <manifest>
                 <attribute name="Main-Class" value="net.i2p.util.Exec" />
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.util.tr}" />
@@ -1169,43 +1185,31 @@
 	<ant target="doInstallerEXE" />
     </target>
 
+    <!-- this makes i2pinstall.exe from install.jar -->
     <target name="doInstallerEXE" unless="noExe">
 	<!-- now the installer exe -->
         <launch4j configFile="./installer/i2pinstaller.xml" />
-        <launch4j configFile="./installer/i2pstandalone.xml" />
 	<!-- thazzit -->
     </target>
 
     <!-- Custom installers -->
-    <target name="installer-nowindows" depends="clean, preppkg-nowindows, getReleaseNumber, getBuildNumber, buildProperties, util-list-changes, izpack-patches" >
-        <mkdir dir="pkg-temp/installer" />
-        <!-- set if unset -->
-        <property name="workspace.changes.util.tr" value="" />
-	<izpack input="${basedir}/installer/install.xml" output="${basedir}/i2pinstall_${release.number}-${i2p.build.number}${build.extra}.jar" installerType="standard" basedir="${basedir}" />
+    <target name="installer-nowindows" depends="clean, preppkg-nowindows, izpack-patches" >
+	<izpack input="${basedir}/installer/install.xml" output="${basedir}/i2pinstall_${full.version}.jar" installerType="standard" basedir="${basedir}" />
     </target>
 
 
-    <target name="installer-freebsd" depends="clean, preppkg-freebsd-only, getReleaseNumber, getBuildNumber, buildProperties, util-list-changes, izpack-patches" >
-        <mkdir dir="pkg-temp/installer" />
-        <!-- set if unset -->
-        <property name="workspace.changes.util.tr" value="" />
-	<izpack input="${basedir}/installer/install.xml" output="${basedir}/i2pinstall_${release.number}-${i2p.build.number}${build.extra}_freebsd-only.jar" installerType="standard" basedir="${basedir}" />
+    <target name="installer-freebsd" depends="clean, preppkg-freebsd-only, izpack-patches" >
+	<izpack input="${basedir}/installer/install.xml" output="${basedir}/i2pinstall_${full.version}_freebsd-only.jar" installerType="standard" basedir="${basedir}" />
     </target>
 
-    <target name="installer-linux" depends="clean, preppkg-linux-only, getReleaseNumber, getBuildNumber, buildProperties, util-list-changes, izpack-patches" >
-        <mkdir dir="pkg-temp/installer" />
-        <!-- set if unset -->
-        <property name="workspace.changes.util.tr" value="" />
-	<izpack input="${basedir}/installer/install.xml" output="${basedir}/i2pinstall_${release.number}-${i2p.build.number}${build.extra}_linux-only.jar" installerType="standard" basedir="${basedir}" />
+    <target name="installer-linux" depends="clean, preppkg-linux-only, izpack-patches" >
+	<izpack input="${basedir}/installer/install.xml" output="${basedir}/i2pinstall_${full.version}_linux-only.jar" installerType="standard" basedir="${basedir}" />
     </target>
 
 
-    <target name="installer-osx" depends="clean, checkForIzpack2App, preppkg-osx-only, getReleaseNumber, getBuildNumber, buildProperties, util-list-changes, izpack-patches">
-        <mkdir dir="pkg-temp/installer" />
+    <target name="installer-osx" depends="clean, checkForIzpack2App, preppkg-osx-only, izpack-patches">
 	<mkdir dir="pkg-temp/osx" />
-        <!-- set if unset -->
-        <property name="workspace.changes.util.tr" value="" />
-	<izpack input="${basedir}/installer/install.xml" output="${basedir}/i2pinstall_${release.number}-${i2p.build.number}${build.extra}_osx-only.jar" installerType="standard" basedir="${basedir}" />
+	<izpack input="${basedir}/installer/install.xml" output="${basedir}/i2pinstall_${full.version}_osx-only.jar" installerType="standard" basedir="${basedir}" />
 	<ant target="installer2app" />
 	<delete dir="pkg-temp/osx" />
     </target>
@@ -1218,34 +1222,32 @@
 	<mkdir dir="pkg-temp/osx" />
 	<exec executable="python" failonerror="true">
 	    <arg value="${user.home}/IzPack/utils/wrappers/izpack2app/izpack2app.py" />
-	    <arg value="${basedir}/i2pinstall_${release.number}-${i2p.build.number}${build.extra}_osx-only.jar" />
-	    <arg value="${basedir}/pkg-temp/osx/i2p-${release.number}-${i2p.build.number}${build.extra}_osx-install.app" />
+	    <arg value="${basedir}/i2pinstall_${full.version}_osx-only.jar" />
+	    <arg value="${basedir}/pkg-temp/osx/i2p-${full.version}_osx-install.app" />
 	</exec>
 	<exec executable="chmod" failonerror="true" osfamily="unix">
 		<arg value="755" />
-		<arg value="${basedir}/pkg-temp/osx/i2p-${release.number}-${i2p.build.number}${build.extra}_osx-install.app/Contents/MacOS/JavaApplicationStub" />
+		<arg value="${basedir}/pkg-temp/osx/i2p-${full.version}_osx-install.app/Contents/MacOS/JavaApplicationStub" />
 	</exec>
 	<exec executable="tar" osfamily="unix" failonerror="true">
 		<arg value="--owner=root" />
 		<arg value="--group=root" />
 		<arg value="-cjvf" />
-		<arg value="${basedir}/i2pinstall_${release.number}-${i2p.build.number}${build.extra}_osx.tar.bz2" />
+		<arg value="${basedir}/i2pinstall_${full.version}_osx.tar.bz2" />
 		<arg value="-C" />
 		<arg value="${basedir}/pkg-temp/osx" />
-		<arg value="i2p-${release.number}-${i2p.build.number}${build.extra}_osx-install.app" />
+		<arg value="i2p-${full.version}_osx-install.app" />
 	</exec>
     </target>
 
-    <!-- Yes, even though this makes an installer for Windows, we still need to depend on preppkg-unix (because the installer attempts to delete these files
-     and will fail if they don't exist. Thankfully these files are so very small there's not much harm in including them.
-    -->
-    <target name="installer-windows" depends="clean, doBuildEXE, preppkg-windows-only, preppkg-unix, getReleaseNumber, getBuildNumber, buildProperties, util-list-changes, izpack-patches" >
+    <target name="installer-windows" depends="clean, preppkg-windows-only, util-list-changes, izpack-patches" >
         <mkdir dir="pkg-temp/installer" />
         <!-- set if unset -->
         <property name="workspace.changes.util.tr" value="" />
         <jar destfile="./pkg-temp/installer/copy.jar" basedir="./core/java/build/obj" includes="net/i2p/util/Copy.class net/i2p/util/FileUtil.class">
             <manifest>
                 <attribute name="Main-Class" value="net.i2p.util.Copy" />
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.util.tr}" />
@@ -1254,6 +1256,7 @@
         <jar destfile="./pkg-temp/installer/delete.jar" basedir="./core/java/build/obj" includes="net/i2p/util/Delete.class net/i2p/util/FileUtil.class">
             <manifest>
                 <attribute name="Main-Class" value="net.i2p.util.Delete" />
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.util.tr}" />
@@ -1262,13 +1265,16 @@
         <jar destfile="./pkg-temp/installer/exec.jar" basedir="./core/java/build/obj" includes="net/i2p/util/Exec.class">
             <manifest>
                 <attribute name="Main-Class" value="net.i2p.util.Exec" />
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.util.tr}" />
             </manifest>
         </jar>
-	<izpack input="${basedir}/installer/install.xml" output="${basedir}/i2pinstall_${release.number}-${i2p.build.number}${build.extra}_windows-only.jar" installerType="standard" basedir="${basedir}" />
-	<delete dir="pkg-temp/win" />
+	<izpack input="${basedir}/installer/install.xml" output="${basedir}/install.jar" installerType="standard" basedir="${basedir}" />
+        <ant target="installerexe" />
+        <delete file="${basedir}/install.jar" />
+        <move file="${basedir}/i2pinstall.exe" tofile="${basedir}/i2pinstall_${full.version}_windows-only.exe" />
     </target>
 
     <target name="installer-all" depends="installer-freebsd, installer-linux, installer-osx, installer-windows, installer" >
@@ -1383,8 +1389,9 @@
         <echo message="New version number is ${release.number}" />
         <copy file="i2pupdate.zip" tofile="i2pupdate_${release.number}.zip" />
         <copy file="i2pinstall.exe" tofile="i2pinstall_${release.number}.exe" />
-        <delete file="i2pupdate.sud" failonerror="false" />
-        <delete file="i2pupdate.su2" failonerror="false" />
+        <delete file="i2pupdate.sud" />
+        <delete file="i2pupdate.su2" />
+        <!-- make this a lot easier by putting release.privkey=/path/to/privkey in override.properties -->
         <input message="Enter private signing key file:" addproperty="release.privkey" />
         <fail message="You must enter a path." >
             <condition>
@@ -1494,6 +1501,17 @@
             <arg value="-b" />
             <arg value="i2pupdate_${release.number}.zip" />
         </exec>
+        <exec executable="chmod" failonerror="true">
+            <arg value="444" />
+            <arg value="i2pinstall_${release.number}.exe" />
+            <arg value="i2psource_${release.number}.tar.bz2" />
+            <arg value="i2pupdate_${release.number}.zip" />
+            <arg value="i2pupdate.su2" />
+            <arg value="i2pupdate.sud" />
+            <arg value="i2pinstall_${release.number}.exe.sig" />
+            <arg value="i2psource_${release.number}.tar.bz2.sig" />
+            <arg value="i2pupdate_${release.number}.zip.sig" />
+        </exec>
         <echo message="File sizes:" />
         <exec executable="ls" failonerror="true">
             <arg value="-l" />
diff --git a/core/java/build.xml b/core/java/build.xml
index c3eaf5c0780dd3d9f59692af034f100916ccf2c5..dad75fc9b5561fc002a172c3bcb650acc5b0eb37 100644
--- a/core/java/build.xml
+++ b/core/java/build.xml
@@ -59,6 +59,7 @@
         <jar destfile="./build/i2p.jar" basedir="./build/obj" includes="**/*.class" >
             <manifest>
                 <attribute name="Implementation-Version" value="${full.version}" />
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.tr}" />
diff --git a/core/java/src/net/i2p/client/RequestLeaseSetMessageHandler.java b/core/java/src/net/i2p/client/RequestLeaseSetMessageHandler.java
index 663c01b7bb3c204641f98c601f45f5747818ba32..6bacb95f2276093374d50631928b2e7f7a9da5f8 100644
--- a/core/java/src/net/i2p/client/RequestLeaseSetMessageHandler.java
+++ b/core/java/src/net/i2p/client/RequestLeaseSetMessageHandler.java
@@ -34,7 +34,7 @@ import net.i2p.util.Log;
  * @author jrandom
  */
 class RequestLeaseSetMessageHandler extends HandlerImpl {
-    private final Map _existingLeaseSets;
+    private final Map<Destination, LeaseInfo> _existingLeaseSets;
 
     public RequestLeaseSetMessageHandler(I2PAppContext context) {
         super(context, RequestLeaseSetMessage.MESSAGE_TYPE);
@@ -59,7 +59,7 @@ class RequestLeaseSetMessageHandler extends HandlerImpl {
         leaseSet.setDestination(session.getMyDestination());
 
         // reuse the old keys for the client
-        LeaseInfo li = (LeaseInfo) _existingLeaseSets.get(session.getMyDestination());
+        LeaseInfo li = _existingLeaseSets.get(session.getMyDestination());
         if (li == null) {
             li = new LeaseInfo(session.getMyDestination());
             _existingLeaseSets.put(session.getMyDestination(), li);
@@ -98,11 +98,11 @@ class RequestLeaseSetMessageHandler extends HandlerImpl {
     }
 
     private static class LeaseInfo {
-        private PublicKey _pubKey;
-        private PrivateKey _privKey;
-        private SigningPublicKey _signingPubKey;
-        private SigningPrivateKey _signingPrivKey;
-        private Destination _dest;
+        private final PublicKey _pubKey;
+        private final PrivateKey _privKey;
+        private final SigningPublicKey _signingPubKey;
+        private final SigningPrivateKey _signingPrivKey;
+        private final Destination _dest;
 
         public LeaseInfo(Destination dest) {
             _dest = dest;
diff --git a/core/java/src/net/i2p/client/naming/BlockfileNamingService.java b/core/java/src/net/i2p/client/naming/BlockfileNamingService.java
index 6d0e1bb6f3bd636e435c1f330fc4ea1efe09f8a9..8172ee7fd2b88b713342f1da52742bac6d6fe57f 100644
--- a/core/java/src/net/i2p/client/naming/BlockfileNamingService.java
+++ b/core/java/src/net/i2p/client/naming/BlockfileNamingService.java
@@ -92,6 +92,7 @@ public class BlockfileNamingService extends DummyNamingService {
     private final RAIFile _raf;
     private final List<String> _lists;
     private final List<InvalidEntry> _invalid;
+    private final Map<String, String> _negativeCache;
     private volatile boolean _isClosed;
     private final boolean _readOnly;
     private boolean _needsUpgrade;
@@ -118,6 +119,9 @@ public class BlockfileNamingService extends DummyNamingService {
     private static final String PROP_ADDED = "a";
     private static final String PROP_SOURCE = "s";
     
+    private static final String DUMMY = "";
+    private static final int NEGATIVE_CACHE_SIZE = 32;
+
     /**
      *  Opens the database at hostsdb.blockfile or creates a new
      *  one and imports entries from hosts.txt, userhosts.txt, and privatehosts.txt.
@@ -132,6 +136,7 @@ public class BlockfileNamingService extends DummyNamingService {
         super(context);
         _lists = new ArrayList();
         _invalid = new ArrayList();
+        _negativeCache = new LHM(NEGATIVE_CACHE_SIZE);
         BlockFile bf = null;
         RAIFile raf = null;
         boolean readOnly = false;
@@ -628,6 +633,10 @@ public class BlockfileNamingService extends DummyNamingService {
         }
 
         String key = hostname.toLowerCase(Locale.US);
+        synchronized(_negativeCache) {
+            if (_negativeCache.get(key) != null)
+                return null;
+        }
         synchronized(_bf) {
             if (_isClosed)
                 return null;
@@ -650,8 +659,13 @@ public class BlockfileNamingService extends DummyNamingService {
             }
             deleteInvalid();
         }
-        if (d != null)
+        if (d != null) {
             putCache(hostname, d);
+        } else {
+            synchronized(_negativeCache) {
+                _negativeCache.put(key, DUMMY);
+            }
+        }
         return d;
     }
 
@@ -683,6 +697,9 @@ public class BlockfileNamingService extends DummyNamingService {
             return false;
         }
         String key = hostname.toLowerCase(Locale.US);
+        synchronized(_negativeCache) {
+            _negativeCache.remove(key);
+        }
         String listname = FALLBACK_LIST;
         Properties props = new Properties();
         props.setProperty(PROP_ADDED, Long.toString(_context.clock().now()));
@@ -1031,6 +1048,9 @@ public class BlockfileNamingService extends DummyNamingService {
             }
             _isClosed = true;
         }
+        synchronized(_negativeCache) {
+            _negativeCache.clear();
+        }
         clearCache();
     }
 
diff --git a/core/java/src/net/i2p/client/naming/DummyNamingService.java b/core/java/src/net/i2p/client/naming/DummyNamingService.java
index 21f8a7c7cc1f0de892979ddd57431e84e6450198..63c6cf26484d67f9b94d1d5d1ae06da6a9651428 100644
--- a/core/java/src/net/i2p/client/naming/DummyNamingService.java
+++ b/core/java/src/net/i2p/client/naming/DummyNamingService.java
@@ -116,7 +116,7 @@ class DummyNamingService extends NamingService {
         }
     }
 
-    private static class LHM<K, V> extends LinkedHashMap<K, V> {
+    protected static class LHM<K, V> extends LinkedHashMap<K, V> {
         private final int _max;
 
         public LHM(int max) {
diff --git a/core/java/src/net/i2p/crypto/TransientSessionKeyManager.java b/core/java/src/net/i2p/crypto/TransientSessionKeyManager.java
index e17802042b22a3a09e8e774d9dc0ee98ab411bc6..7959c68e9315d1a99976875bcf0d12c6170192f6 100644
--- a/core/java/src/net/i2p/crypto/TransientSessionKeyManager.java
+++ b/core/java/src/net/i2p/crypto/TransientSessionKeyManager.java
@@ -625,7 +625,7 @@ public class TransientSessionKeyManager extends SessionKeyManager {
             OutboundSession sess = iter.next();
             Set<TagSet> sets = new TreeSet(new TagSetComparator());
             sets.addAll(sess.getTagSets());
-            buf.append("<tr><td><b>Target key:</b> ").append(sess.getTarget().toBase64().substring(0, 64)).append("<br>" +
+            buf.append("<tr><td><b>Target public key:</b> ").append(sess.getTarget().toBase64().substring(0, 20)).append("...<br>" +
                        "<b>Established:</b> ").append(DataHelper.formatDuration(now - sess.getEstablishedDate())).append(" ago<br>" +
                        "<b>Last Used:</b> ").append(DataHelper.formatDuration(now - sess.getLastUsedDate())).append(" ago<br>" +
                        "<b>Session key:</b> ").append(sess.getCurrentKey().toBase64()).append("</td>" +
diff --git a/core/java/src/net/i2p/data/Base32.java b/core/java/src/net/i2p/data/Base32.java
index fd6f81a02de28a5958cce041c3b78b8abc639ccb..d3d438d62e777dbf3dff51aa1def5a15b9c6bcd7 100644
--- a/core/java/src/net/i2p/data/Base32.java
+++ b/core/java/src/net/i2p/data/Base32.java
@@ -29,7 +29,7 @@ import net.i2p.util.Log;
  */
 public class Base32 {
 
-    private final static Log _log = new Log(Base32.class);
+    //private final static Log _log = new Log(Base32.class);
 
     /** The 32 valid Base32 values. */
     private final static char[] ALPHABET = {'a', 'b', 'c', 'd',
@@ -248,12 +248,12 @@ public class Base32 {
                          outBuff[outBuffPosn] = next;
                          usedbits -= 3;
                      } else if (next != 0) {
-                       _log.warn("Extra data at the end: " + next + "(decimal)");
+                       //_log.warn("Extra data at the end: " + next + "(decimal)");
                        return null;
                      }
                  }
             } else {
-                _log.warn("Bad Base32 input character at " + i + ": " + source[i] + "(decimal)");
+                //_log.warn("Bad Base32 input character at " + i + ": " + source[i] + "(decimal)");
                 return null;
             }
         }
diff --git a/core/java/src/net/i2p/data/Base64.java b/core/java/src/net/i2p/data/Base64.java
index 6f5fc34a57dd048497ee177d704714856b9496f7..9adb3e202e9e455851eaa0bd6e838df8b8e37ae4 100644
--- a/core/java/src/net/i2p/data/Base64.java
+++ b/core/java/src/net/i2p/data/Base64.java
@@ -41,7 +41,7 @@ import net.i2p.util.Log;
  */
 public class Base64 {
 
-    private final static Log _log = new Log(Base64.class);
+    //private final static Log _log = new Log(Base64.class);
 
     /**
      *  @param source if null will return ""
@@ -750,7 +750,7 @@ public class Base64 {
 
             } // end if: white space, equals sign or better
             else {
-                _log.warn("Bad Base64 input character at " + i + ": " + source[i] + "(decimal)");
+                //_log.warn("Bad Base64 input character at " + i + ": " + source[i] + "(decimal)");
                 return null;
             } // end else: 
         } // each input character
diff --git a/core/java/src/net/i2p/data/DataHelper.java b/core/java/src/net/i2p/data/DataHelper.java
index d809926098896ce073233a1fd82186b6a1266bec..ad6ac6ea9d0b10c536fe365fecf4954aad25f900 100644
--- a/core/java/src/net/i2p/data/DataHelper.java
+++ b/core/java/src/net/i2p/data/DataHelper.java
@@ -1299,10 +1299,20 @@ public class DataHelper {
         //    rv.add(struct);
         //}
         ArrayList<DataStructure> rv = new ArrayList(dataStructures);
-        Collections.sort(rv, new DataStructureComparator());
+        sortStructureList(rv);
         return rv;
     }
 
+    /**
+     *  See above.
+     *  DEPRECATED - Only used by RouterInfo.
+     *
+     *  @since 0.9
+     */
+    static void sortStructureList(List<? extends DataStructure> dataStructures) {
+        Collections.sort(dataStructures, new DataStructureComparator());
+    }
+
     /**
      * See sortStructures() comments.
      * @since 0.8.3
diff --git a/core/java/src/net/i2p/data/DataStructureImpl.java b/core/java/src/net/i2p/data/DataStructureImpl.java
index b4b43eb57ad2d835e4bde3abc12ce8d6914b5640..1ca043672ef2ae14f818bdc1921d113e0ae40b52 100644
--- a/core/java/src/net/i2p/data/DataStructureImpl.java
+++ b/core/java/src/net/i2p/data/DataStructureImpl.java
@@ -14,6 +14,7 @@ import java.io.ByteArrayOutputStream;
 import java.io.IOException;
 import java.io.InputStream;
 
+import net.i2p.I2PAppContext;
 import net.i2p.crypto.SHA256Generator;
 import net.i2p.util.Log;
 
@@ -23,7 +24,6 @@ import net.i2p.util.Log;
  * @author jrandom
  */
 public abstract class DataStructureImpl implements DataStructure {
-    private final static Log _log = new Log(DataStructureImpl.class);
     
     public String toBase64() {
         byte data[] = toByteArray();
@@ -48,10 +48,12 @@ public abstract class DataStructureImpl implements DataStructure {
             writeBytes(baos);
             return baos.toByteArray();
         } catch (IOException ioe) {
-            _log.error("Error writing out the byte array", ioe);
+            Log log = I2PAppContext.getGlobalContext().logManager().getLog(getClass());
+            log.error("Error writing out the byte array", ioe);
             return null;
         } catch (DataFormatException dfe) {
-            _log.error("Error writing out the byte array", dfe);
+            Log log = I2PAppContext.getGlobalContext().logManager().getLog(getClass());
+            log.error("Error writing out the byte array", dfe);
             return null;
         }
     }
@@ -73,4 +75,4 @@ public abstract class DataStructureImpl implements DataStructure {
     protected int read(InputStream in, byte target[]) throws IOException {
         return DataHelper.read(in, target);
     }
-}
\ No newline at end of file
+}
diff --git a/core/java/src/net/i2p/data/Lease.java b/core/java/src/net/i2p/data/Lease.java
index 283e2aa7c232e3462a4a4dbd170f5d9ce692030b..d6c97f1502f23402f64e0a755615d0b15dccaad4 100644
--- a/core/java/src/net/i2p/data/Lease.java
+++ b/core/java/src/net/i2p/data/Lease.java
@@ -26,8 +26,8 @@ public class Lease extends DataStructureImpl {
     private Hash _gateway;
     private TunnelId _tunnelId;
     private Date _end;
-    private int _numSuccess;
-    private int _numFailure;
+    //private int _numSuccess;
+    //private int _numFailure;
 
     public Lease() {
     }
@@ -74,14 +74,18 @@ public class Lease extends DataStructureImpl {
      *
      * @deprecated unused
      */
+/****
     public int getNumSuccess() {
         return _numSuccess;
     }
+****/
 
     /** @deprecated unused */
+/****
     public void setNumSuccess(int num) {
         _numSuccess = num;
     }
+****/
 
     /**
      * Transient attribute of the lease, used to note how many times messages sent
@@ -89,14 +93,18 @@ public class Lease extends DataStructureImpl {
      *
      * @deprecated unused
      */
+/****
     public int getNumFailure() {
         return _numFailure;
     }
+****/
 
     /** @deprecated unused */
+/****
     public void setNumFailure(int num) {
         _numFailure = num;
     }
+****/
 
     /** has this lease already expired? */
     public boolean isExpired() {
diff --git a/core/java/src/net/i2p/data/LeaseSet.java b/core/java/src/net/i2p/data/LeaseSet.java
index e372f401cd32172ebbbccdda2ba5749f890b0c77..9fe9ebb3044da1a19b1d370dcfd91a37d1d18a0d 100644
--- a/core/java/src/net/i2p/data/LeaseSet.java
+++ b/core/java/src/net/i2p/data/LeaseSet.java
@@ -56,7 +56,6 @@ import net.i2p.util.RandomSource;
  * @author jrandom
  */
 public class LeaseSet extends DatabaseEntry {
-    private final static Log _log = new Log(LeaseSet.class);
     private Destination _destination;
     private PublicKey _encryptionKey;
     private SigningPublicKey _signingKey;
@@ -71,11 +70,26 @@ public class LeaseSet extends DatabaseEntry {
     private boolean _decrypted;
     private boolean _checked;
 
-    /** This seems like plenty  */
-    public final static int MAX_LEASES = 6;
+    /**
+     *  Unlimited before 0.6.3;
+     *  6 as of 0.6.3;
+     *  Increased in version 0.9.
+     *
+     *  Leasesets larger than 6 should be used with caution,
+     *  as each lease adds 44 bytes, and routers older than version 0.9
+     *  will not be able to connect as they will throw an exception in
+     *  readBytes(). Also, the churn will be quite rapid, leading to
+     *  frequent netdb stores and transmission on existing connections.
+     *
+     *  However we increase it now in case some hugely popular eepsite arrives.
+     *  Strategies elsewhere in the router to efficiently handle
+     *  large leasesets are TBD.
+     */
+    public static final int MAX_LEASES = 16;
+    private static final int OLD_MAX_LEASES = 6;
 
     public LeaseSet() {
-        _leases = new ArrayList(MAX_LEASES);
+        _leases = new ArrayList(OLD_MAX_LEASES);
         _firstExpiration = Long.MAX_VALUE;
     }
 
@@ -354,14 +368,16 @@ public class LeaseSet extends DatabaseEntry {
      *  Must be called after all the leases are in place, but before sign().
      */
     public void encrypt(SessionKey key) {
-        if (_log.shouldLog(Log.WARN))
-            _log.warn("encrypting lease: " + _destination.calculateHash());
+        //if (_log.shouldLog(Log.WARN))
+        //    _log.warn("encrypting lease: " + _destination.calculateHash());
         try {
             encryp(key);
         } catch (DataFormatException dfe) {
-            _log.error("Error encrypting lease: " + _destination.calculateHash());
+            Log log = I2PAppContext.getGlobalContext().logManager().getLog(LeaseSet.class);
+            log.error("Error encrypting lease: " + _destination.calculateHash());
         } catch (IOException ioe) {
-            _log.error("Error encrypting lease: " + _destination.calculateHash());
+            Log log = I2PAppContext.getGlobalContext().logManager().getLog(LeaseSet.class);
+            log.error("Error encrypting lease: " + _destination.calculateHash());
         }
     }
 
@@ -420,8 +436,8 @@ public class LeaseSet extends DatabaseEntry {
      *  encrypted leaseset can be sent on to others (via writeBytes())
      */
     private void decrypt(SessionKey key) throws DataFormatException, IOException {
-        if (_log.shouldLog(Log.WARN))
-            _log.warn("decrypting lease: " + _destination.calculateHash());
+        //if (_log.shouldLog(Log.WARN))
+        //    _log.warn("decrypting lease: " + _destination.calculateHash());
         int size = _leases.size();
         if (size < 2)
             throw new DataFormatException("Bad number of leases for decryption");
@@ -468,9 +484,11 @@ public class LeaseSet extends DatabaseEntry {
                 decrypt(key);
                 _decrypted = true;
             } catch (DataFormatException dfe) {
-                _log.error("Error decrypting lease: " + _destination.calculateHash() + dfe);
+                Log log = I2PAppContext.getGlobalContext().logManager().getLog(LeaseSet.class);
+                log.error("Error decrypting lease: " + _destination.calculateHash() + dfe);
             } catch (IOException ioe) {
-                _log.error("Error decrypting lease: " + _destination.calculateHash() + ioe);
+                Log log = I2PAppContext.getGlobalContext().logManager().getLog(LeaseSet.class);
+                log.error("Error decrypting lease: " + _destination.calculateHash() + ioe);
             }
         }
         _checked = true;
diff --git a/core/java/src/net/i2p/data/Payload.java b/core/java/src/net/i2p/data/Payload.java
index bdaac7d4c50601bdba37df4cb5aa15e9c2c704d0..453fe80773eb66032932a0a88e1fefbde5ce125f 100644
--- a/core/java/src/net/i2p/data/Payload.java
+++ b/core/java/src/net/i2p/data/Payload.java
@@ -26,7 +26,7 @@ import net.i2p.util.Log;
  * @author jrandom
  */
 public class Payload extends DataStructureImpl {
-    private final static Log _log = new Log(Payload.class);
+    //private final static Log _log = new Log(Payload.class);
     private byte[] _encryptedData;
     private byte[] _unencryptedData;
 
@@ -82,16 +82,16 @@ public class Payload extends DataStructureImpl {
         _encryptedData = new byte[size];
         int read = read(in, _encryptedData);
         if (read != size) throw new DataFormatException("Incorrect number of bytes read in the payload structure");
-        if (_log.shouldLog(Log.DEBUG))
-            _log.debug("read payload: " + read + " bytes");
+        //if (_log.shouldLog(Log.DEBUG))
+        //    _log.debug("read payload: " + read + " bytes");
     }
     
     public void writeBytes(OutputStream out) throws DataFormatException, IOException {
         if (_encryptedData == null) throw new DataFormatException("Not yet encrypted.  Please set the encrypted data");
         DataHelper.writeLong(out, 4, _encryptedData.length);
         out.write(_encryptedData);
-        if (_log.shouldLog(Log.DEBUG))
-            _log.debug("wrote payload: " + _encryptedData.length);
+        //if (_log.shouldLog(Log.DEBUG))
+        //    _log.debug("wrote payload: " + _encryptedData.length);
     }
 
     /**
diff --git a/core/java/src/net/i2p/data/RouterInfo.java b/core/java/src/net/i2p/data/RouterInfo.java
index 6c34ad2023989aed04caff0b47d662e346296e0a..398b61e0c84c34d12b20d40b33692f9c6ebebbc0 100644
--- a/core/java/src/net/i2p/data/RouterInfo.java
+++ b/core/java/src/net/i2p/data/RouterInfo.java
@@ -13,6 +13,7 @@ import java.io.ByteArrayOutputStream;
 import java.io.IOException;
 import java.io.InputStream;
 import java.io.OutputStream;
+import java.util.ArrayList;
 import java.util.Collection;
 import java.util.Collections;
 import java.util.Date;
@@ -24,6 +25,7 @@ import java.util.Properties;
 import java.util.Set;
 import java.util.Vector;
 
+import net.i2p.I2PAppContext;
 import net.i2p.crypto.SHA256Generator;
 import net.i2p.util.Clock;
 import net.i2p.util.Log;
@@ -43,10 +45,14 @@ import net.i2p.util.OrderedProperties;
  * @author jrandom
  */
 public class RouterInfo extends DatabaseEntry {
-    private final static Log _log = new Log(RouterInfo.class);
     private RouterIdentity _identity;
     private volatile long _published;
-    private final Set<RouterAddress> _addresses;
+    /**
+     *  Addresses must be sorted by SHA256.
+     *  When an RI is created, they are sorted in setAddresses().
+     *  Save addresses in the order received so we need not resort.
+     */
+    private final List<RouterAddress> _addresses;
     /** may be null to save memory, no longer final */
     private Set<Hash> _peers;
     private final Properties _options;
@@ -71,7 +77,7 @@ public class RouterInfo extends DatabaseEntry {
     public static final String BW_CAPABILITY_CHARS = "KLMNO";
     
     public RouterInfo() {
-        _addresses = new HashSet(2);
+        _addresses = new ArrayList(2);
         _options = new OrderedProperties();
     }
 
@@ -156,21 +162,33 @@ public class RouterInfo extends DatabaseEntry {
      *
      * @return unmodifiable view, non-null
      */
-    public Set<RouterAddress> getAddresses() {
-            return Collections.unmodifiableSet(_addresses);
+    public Collection<RouterAddress> getAddresses() {
+            return Collections.unmodifiableCollection(_addresses);
     }
 
     /**
      * Specify a set of RouterAddress structures at which this router
      * can be contacted.
      *
-     * @throws IllegalStateException if RouterInfo is already signed
+     * Warning - Sorts the addresses here. Do not modify any address
+     *           after calling this, as the sort order is based on the
+     *           hash of the entire address structure.
+     *
+     * @param addresses may be null
+     * @throws IllegalStateException if RouterInfo is already signed or addresses previously set
      */
-    public void setAddresses(Set<RouterAddress> addresses) {
-        if (_signature != null)
+    public void setAddresses(Collection<RouterAddress> addresses) {
+        if (_signature != null || !_addresses.isEmpty())
             throw new IllegalStateException();
-        _addresses.clear();
-        if (addresses != null) _addresses.addAll(addresses);
+        if (addresses != null) {
+            _addresses.addAll(addresses);
+            if (_addresses.size() > 1) {
+                // WARNING this sort algorithm cannot be changed, as it must be consistent
+                // network-wide. The signature is not checked at readin time, but only
+                // later, and the addresses are stored in a Set, not a List.
+                DataHelper.sortStructureList(_addresses);
+            }
+        }
     }
 
     /**
@@ -270,14 +288,7 @@ public class RouterInfo extends DatabaseEntry {
                 DataHelper.writeLong(out, 1, 0);
             } else {
                 DataHelper.writeLong(out, 1, sz);
-                Collection<RouterAddress> addresses = _addresses;
-                if (sz > 1) {
-                    // WARNING this sort algorithm cannot be changed, as it must be consistent
-                    // network-wide. The signature is not checked at readin time, but only
-                    // later, and the addresses are stored in a Set, not a List.
-                    addresses = (Collection<RouterAddress>) DataHelper.sortStructures(addresses);
-                }
-                for (RouterAddress addr : addresses) {
+                for (RouterAddress addr : _addresses) {
                     addr.writeBytes(out);
                 }
             }
@@ -458,16 +469,16 @@ public class RouterInfo extends DatabaseEntry {
         _validated = true;
 
         if (!_isValid) {
+            Log log = I2PAppContext.getGlobalContext().logManager().getLog(RouterInfo.class);
             byte data[] = null;
             try {
                 data = getBytes();
             } catch (DataFormatException dfe) {
-                _log.error("Error validating", dfe);
+                log.error("Error validating", dfe);
                 return;
             }
-            if (_log.shouldLog(Log.ERROR))
-                _log.error("Invalid [" + SHA256Generator.getInstance().calculateHash(data).toBase64()
-                           + (_log.shouldLog(Log.WARN) ? ("]\n" + toString()) : ""),
+            log.error("Invalid [" + SHA256Generator.getInstance().calculateHash(data).toBase64()
+                           + (log.shouldLog(Log.WARN) ? ("]\n" + toString()) : ""),
                            new Exception("Signature failed"));
         }
     }
diff --git a/core/java/src/net/i2p/util/Log.java b/core/java/src/net/i2p/util/Log.java
index 0a03e19b5efee99099d1d86fc2cd6caa5d5c32e4..538e3986b98e2faefb6bb1262623e42daaa2662c 100644
--- a/core/java/src/net/i2p/util/Log.java
+++ b/core/java/src/net/i2p/util/Log.java
@@ -16,7 +16,7 @@ import net.i2p.I2PAppContext;
 /**
  * Wrapper class for whatever logging system I2P uses.  This class should be 
  * instantiated and kept as a variable for each class it is used by, ala:
- *  <code>private final static Log _log = new Log(MyClassName.class);</code>
+ *  <code>private final Log _log = context.logManager().getLog(MyClassName.class);</code>
  *
  * If there is anything in here that doesn't make sense, turn off your computer
  * and go fly a kite.
diff --git a/core/java/src/net/i2p/util/SimpleTimer2.java b/core/java/src/net/i2p/util/SimpleTimer2.java
index ab568d64466823225a04bc66ad2e1d32d8eda7c1..8e2aee8a746407871e0f9f1d49a66b3e5bce25f8 100644
--- a/core/java/src/net/i2p/util/SimpleTimer2.java
+++ b/core/java/src/net/i2p/util/SimpleTimer2.java
@@ -31,7 +31,6 @@ public class SimpleTimer2 {
     private static final int MIN_THREADS = 2;
     private static final int MAX_THREADS = 4;
     private final I2PAppContext _context;
-    private static Log _log; // static so TimedEvent can use it
     private final ScheduledThreadPoolExecutor _executor;
     private final String _name;
     private int _count;
@@ -40,7 +39,6 @@ public class SimpleTimer2 {
     protected SimpleTimer2() { this("SimpleTimer2"); }
     protected SimpleTimer2(String name) {
         _context = I2PAppContext.getGlobalContext();
-        _log = _context.logManager().getLog(SimpleTimer2.class);
         _name = name;
         _count = 0;
         long maxMemory = Runtime.getRuntime().maxMemory();
@@ -79,8 +77,10 @@ public class SimpleTimer2 {
         @Override
         protected void afterExecute(Runnable r, Throwable t) {
             super.afterExecute(r, t);
-            if (t != null) // shoudn't happen, caught in RunnableEvent.run()
-                _log.log(Log.CRIT, "wtf, event borked: " + r, t);
+            if (t != null) { // shoudn't happen, caught in RunnableEvent.run()
+                Log log = I2PAppContext.getGlobalContext().logManager().getLog(SimpleTimer2.class);
+                log.log(Log.CRIT, "wtf, event borked: " + r, t);
+            }
         }
     }
 
@@ -126,6 +126,7 @@ public class SimpleTimer2 {
      *
      */
     public static abstract class TimedEvent implements Runnable {
+        private final Log _log;
         private SimpleTimer2 _pool;
         private int _fuzz;
         protected static final int DEFAULT_FUZZ = 3;
@@ -136,7 +137,9 @@ public class SimpleTimer2 {
         public TimedEvent(SimpleTimer2 pool) {
             _pool = pool;
             _fuzz = DEFAULT_FUZZ;
+            _log = I2PAppContext.getGlobalContext().logManager().getLog(SimpleTimer2.class);
         }
+
         /** automatically schedules, don't use this one if you have other things to do first */
         public TimedEvent(SimpleTimer2 pool, long timeoutMs) {
             this(pool);
diff --git a/history.txt b/history.txt
index e3b1aae098cf2e07a74ff48f0fc753e05cd6c45a..c3290a434f70183e9e6df9bb40740953a7869c76 100644
--- a/history.txt
+++ b/history.txt
@@ -1,3 +1,36 @@
+2012-03-02 zzz
+  * BlockfileNamingService: Add negative cache
+  * Build: Add built-by to jars; check for corrupt jars on debug page
+  * configstats.jsp: Hide log settings unless already enabled
+  * DataStructures:
+    - Remove static logs
+    - Sort addresses in RouterInfo at initialization only;
+      change from Set to List to save space
+    - Remove unused counters in Lease to save space
+    - Increase max leases to 16
+  * Graphs:
+    - New single graph page with easy resizing
+    - Support graphing of previous intervals
+  * i2pinstall.exe: Add icon
+  * i2psnark:
+    - Add tracker configuration form
+    - Remove custom tracker option from create form
+    - Add private torrent option
+    - More icons in buttons
+    - Use js for refresh
+  * I2PTunnelHTTPClient:
+    - Refactoring to use Java URI parser to better handle
+      escapes, IPv6 addresses, ports
+    - Rewrite i2paddresshelper scanning/removal
+    - Refactor out local server code
+    - Nicer address helper error page
+  * NetDB:
+    - Reenable verify of RI stores, disabled in 0.7.9,
+      checkin comments claim reenabled in 0.7.10 but didn't happen.
+    - Synchronize StoreJob.sendNext() to avoid dups
+  * netdb.jsp: Fix debug leaseset count again
+  * susidns: Add b64 hash to details page
+
 * 2012-02-27 0.8.13 released
 
 2012-02-22 kytv
diff --git a/installer/i2pinstaller.xml b/installer/i2pinstaller.xml
index fd46517beb743296c00506693a62cc2d3428dd70..4751a856501e5ede9073e4c491b07c7b371dceae 100644
--- a/installer/i2pinstaller.xml
+++ b/installer/i2pinstaller.xml
@@ -5,7 +5,7 @@
   <errTitle>I2P Installer</errTitle>
   <chdir>.</chdir>
   <customProcName>false</customProcName>
-  <!--<icon>SimpleApp.ico</icon>-->
+  <icon>resources/console.ico</icon>
   <jre>
     <minVersion>1.5.0</minVersion>
   </jre>
diff --git a/installer/i2pstandalone.xml b/installer/i2pstandalone.xml
index 9a6f0a65955b863e9ea49c637853bdd96a232b97..60ba8163bbed6c4ba98ed7f0aba39755d5f68366 100644
--- a/installer/i2pstandalone.xml
+++ b/installer/i2pstandalone.xml
@@ -1,7 +1,7 @@
 <launch4jConfig>
   <headerType>0</headerType>
   <jar>../build/launchi2p.jar</jar>
-  <outfile>../i2p.exe</outfile>
+  <outfile>../build/i2p.exe</outfile>
   <errTitle>I2P</errTitle>
   <chdir>.</chdir>
   <customProcName>false</customProcName>
diff --git a/installer/resources/proxy/ahelper-notfound-header.ht b/installer/resources/proxy/ahelper-notfound-header.ht
new file mode 100644
index 0000000000000000000000000000000000000000..0cdc36fb6d08600fc49759c0912ec36b1657f261
--- /dev/null
+++ b/installer/resources/proxy/ahelper-notfound-header.ht
@@ -0,0 +1,23 @@
+HTTP/1.1 409 Bad Helper
+Content-Type: text/html; charset=UTF-8
+Cache-control: no-cache
+Connection: close
+Proxy-Connection: close
+
+<html><head>
+<title>I2P Warning: Bad Address Helper</title>
+<link rel="shortcut icon" href="http://proxy.i2p/themes/console/images/favicon.ico" >
+<link href="http://proxy.i2p/themes/console/default/console.css" rel="stylesheet" type="text/css" >
+</head>
+<body>
+<div class=logo>
+ <a href="http://127.0.0.1:7657/index.jsp" title="Router Console"><img src="http://proxy.i2p/themes/console/images/i2plogo.png" alt="I2P Router Console" border="0"></a><hr>
+ <a href="http://127.0.0.1:7657/config.jsp">Configuration</a> <a href="http://127.0.0.1:7657/help.jsp">Help</a> <a href="http://127.0.0.1:7657/susidns/">Addressbook</a>
+</div>
+<div class=warning id=warning>
+<h3>Warning: Bad Address Helper</h3>
+<p>
+The helper key you put for i2paddresshelper= is not resolvable.
+It seems to be garbage data, or a mistyped b32. Check your URL
+to try and fix the helper key to be a valid Base 32 hostname or Base 64 key.
+</p>
diff --git a/installer/resources/themes/snark/ubergine/snark.css b/installer/resources/themes/snark/ubergine/snark.css
index 01c1632e341d1a1a82d14e40d9c366229e8507a0..560a212209f1fe3ae0e015bf3618d6ef142c82f3 100644
--- a/installer/resources/themes/snark/ubergine/snark.css
+++ b/installer/resources/themes/snark/ubergine/snark.css
@@ -470,7 +470,7 @@ input {
      border: 1px inset #000;
      background: #212 url('/themes/snark/ubergine/images/graytile.png');	 
      color: #f60;
-     margin: 2px 0;
+     margin: 2px 4px;
 }
 
 input.r {
@@ -525,12 +525,50 @@ input[type=radio] {
      vertical-align: bottom;
 }
 
+input.default { width: 1px; height: 1px; visibility: hidden; }
+
 input.accept {
      background: #989 url('../../console/images/accept.png') no-repeat 2px center;
      padding: 2px 3px 2px 20px !important;
      min-height: 22px;
 }
 
+input.add {
+     background: #989 url('../../console/images/add.png') no-repeat 2px center;
+     padding: 2px 3px 2px 20px !important;
+     min-height: 22px;
+}
+
+input.create {
+     background: #989 url('images/create.png') no-repeat 2px center;
+     padding: 2px 3px 2px 20px !important;
+     min-height: 22px;
+}
+
+input.cancel {
+     background: #989 url('../../console/images/cancel.png') no-repeat 2px center;
+     padding: 2px 3px 2px 20px !important;
+     min-height: 22px;
+}
+
+input.create {
+     background: #989 url('images/create.png') no-repeat 2px center;
+     padding: 2px 3px 2px 20px !important;
+     min-height: 22px;
+}
+
+input.delete {
+     background: #989 url('../../console/images/delete.png') no-repeat 2px center;
+     padding: 2px 3px 2px 20px !important;
+     min-height: 22px;
+}
+
+input.reload {
+     background: #989 url('../../console/images/arrow_refresh.png') no-repeat 2px center;
+     padding: 2px 3px 2px 20px !important;
+     min-height: 22px;
+}
+
 select {
      background: #333;
      background: url('/themes/snark/ubergine/images/graytile.png') !important;
diff --git a/installer/resources/themes/snark/vanilla/snark.css b/installer/resources/themes/snark/vanilla/snark.css
index b0c0c1de5d432d00795b0d3b68141e7fe4269eea..efd061582de04be6360e211920bd57cfadbee5e6 100644
--- a/installer/resources/themes/snark/vanilla/snark.css
+++ b/installer/resources/themes/snark/vanilla/snark.css
@@ -493,7 +493,7 @@ input {
      border: 1px inset #000;
      background: #fff /*url('/themes/snark/ubergine/images/graytile.png')*/;	 
      color: #000;
-     margin: 2px 0;
+     margin: 2px 4px;
 }
 
 input.r {
@@ -549,12 +549,44 @@ input[type=radio] {
      vertical-align: bottom;
 }
 
+input.default { width: 1px; height: 1px; visibility: hidden; }
+
 input.accept {
      background: #f3efc7 url('../../console/images/accept.png') no-repeat 2px center;
      padding: 2px 3px 2px 20px !important;
      min-height: 22px;
 }
 
+input.add {
+     background: #f3efc7 url('../../console/images/add.png') no-repeat 2px center;
+     padding: 2px 3px 2px 20px !important;
+     min-height: 22px;
+}
+
+input.cancel {
+     background: #f3efc7 url('../../console/images/cancel.png') no-repeat 2px center;
+     padding: 2px 3px 2px 20px !important;
+     min-height: 22px;
+}
+
+input.create {
+     background: #f3efc7 url('images/create.png') no-repeat 2px center;
+     padding: 2px 3px 2px 20px !important;
+     min-height: 22px;
+}
+
+input.delete {
+     background: #f3efc7 url('../../console/images/delete.png') no-repeat 2px center;
+     padding: 2px 3px 2px 20px !important;
+     min-height: 22px;
+}
+
+input.reload {
+     background: #f3efc7 url('../../console/images/arrow_refresh.png') no-repeat 2px center;
+     padding: 2px 3px 2px 20px !important;
+     min-height: 22px;
+}
+
 select {
      background: #fff;
 /*     background: url('/themes/snark/ubergine/images/graytile.png') !important;*/
diff --git a/router/java/build.xml b/router/java/build.xml
index e933086cf805521f2df9b380d6623be6d1c5a74d..990b3e4732c824c9c7593d9a7089f98a9b492595 100644
--- a/router/java/build.xml
+++ b/router/java/build.xml
@@ -73,6 +73,7 @@
         <jar destfile="./build/router.jar" basedir="./build/obj" includes="**/*.class" >
             <manifest>
                 <attribute name="Implementation-Version" value="${full.version}" />
+                <attribute name="Built-By" value="${build.built-by}" />
                 <attribute name="Build-Date" value="${build.timestamp}" />
                 <attribute name="Base-Revision" value="${workspace.version}" />
                 <attribute name="Workspace-Changes" value="${workspace.changes.tr}" />
diff --git a/router/java/src/net/i2p/router/Blocklist.java b/router/java/src/net/i2p/router/Blocklist.java
index 153b14e7c928064182255abb0e51f209f9764946..10f9d51b3a6fe4695421c0446d7c7de63e02d9e6 100644
--- a/router/java/src/net/i2p/router/Blocklist.java
+++ b/router/java/src/net/i2p/router/Blocklist.java
@@ -479,15 +479,9 @@ public class Blocklist {
         List<byte[]> rv = new ArrayList(1);
         RouterInfo pinfo = _context.netDb().lookupRouterInfoLocally(peer);
         if (pinfo == null) return rv;
-        Set<RouterAddress> paddr = pinfo.getAddresses();
-        if (paddr == null || paddr.isEmpty())
-            return rv;
         String oldphost = null;
-        List<RouterAddress> pladdr = new ArrayList(paddr);
         // for each peer address
-        for (int j = 0; j < paddr.size(); j++) {
-            RouterAddress pa = (RouterAddress) pladdr.get(j);
-            if (pa == null) continue;
+        for (RouterAddress pa : pinfo.getAddresses()) {
             String phost = pa.getOption("host");
             if (phost == null) continue;
             if (oldphost != null && oldphost.equals(phost)) continue;
@@ -787,7 +781,7 @@ public class Blocklist {
         Set<Integer> singles = new TreeSet();
         singles.addAll(_singleIPBlocklist);
         if (!singles.isEmpty()) {
-            out.write("<table><tr><th align=center colspan=2><b>");
+            out.write("<table><tr><th align=\"center\" colspan=\"2\"><b>");
             out.write(_("IPs Banned Until Restart"));
             out.write("</b></td></tr>");
             // first 0 - 127
@@ -795,27 +789,27 @@ public class Blocklist {
                  int ip = ii.intValue();
                  if (ip < 0)
                      continue;
-                 out.write("<tr><td align=center width=50%>");
+                 out.write("<tr><td align=\"center\" width=\"50%\">");
                  out.write(toStr(ip));
-                 out.write("</td><td width=50%>&nbsp;</td></tr>\n");
+                 out.write("</td><td width=\"50%\">&nbsp;</td></tr>\n");
             }
             // then 128 - 255
             for (Integer ii : singles) {
                  int ip = ii.intValue();
                  if (ip >= 0)
                      break;
-                 out.write("<tr><td align=center width=50%>");
+                 out.write("<tr><td align=\"center\" width=\"50%\">");
                  out.write(toStr(ip));
-                 out.write("</td><td width=50%>&nbsp;</td></tr>\n");
+                 out.write("</td><td width=\"50%\">&nbsp;</td></tr>\n");
             }
             out.write("</table>");
         }
         if (_blocklistSize > 0) {
-            out.write("<table><tr><th align=center colspan=2><b>");
+            out.write("<table><tr><th align=\"center\" colspan=\"2\"><b>");
             out.write(_("IPs Permanently Banned"));
-            out.write("</b></th></tr><tr><td align=center width=50%><b>");
+            out.write("</b></th></tr><tr><td align=\"center\" width=\"50%\"><b>");
             out.write(_("From"));
-            out.write("</b></td><td align=center width=50%><b>");
+            out.write("</b></td><td align=\"center\" width=\"50%\"><b>");
             out.write(_("To"));
             out.write("</b></td></tr>");
             int max = Math.min(_blocklistSize, MAX_DISPLAY);
@@ -825,7 +819,7 @@ public class Blocklist {
                  int from = getFrom(_blocklist[i]);
                  if (from < 0)
                      continue;
-                 out.write("<tr><td align=center width=50%>"); out.write(toStr(from)); out.write("</td><td align=center width=50%>");
+                 out.write("<tr><td align=\"center\" width=\"50%\">"); out.write(toStr(from)); out.write("</td><td align=\"center\" width=\"50%\">");
                  int to = getTo(_blocklist[i]);
                  if (to != from) {
                      out.write(toStr(to)); out.write("</td></tr>\n");
@@ -838,7 +832,7 @@ public class Blocklist {
                  int from = getFrom(_blocklist[i]);
                  if (from >= 0)
                      break;
-                 out.write("<tr><td align=center width=50%>"); out.write(toStr(from)); out.write("</td><td align=center width=50%>");
+                 out.write("<tr><td align=\"center\" width=\"50%\">"); out.write(toStr(from)); out.write("</td><td align=\"center\" width=\"50%\">");
                  int to = getTo(_blocklist[i]);
                  if (to != from) {
                      out.write(toStr(to)); out.write("</td></tr>\n");
diff --git a/router/java/src/net/i2p/router/RouterVersion.java b/router/java/src/net/i2p/router/RouterVersion.java
index a411b009986dc5f1f6384384ad98aeea42885422..87e5bffca0684477c0ba8daee48da05fbea99f44 100644
--- a/router/java/src/net/i2p/router/RouterVersion.java
+++ b/router/java/src/net/i2p/router/RouterVersion.java
@@ -18,7 +18,7 @@ public class RouterVersion {
     /** deprecated */
     public final static String ID = "Monotone";
     public final static String VERSION = CoreVersion.VERSION;
-    public final static long BUILD = 0;
+    public final static long BUILD = 1;
 
     /** for example "-test" */
     public final static String EXTRA = "";
diff --git a/router/java/src/net/i2p/router/StatisticsManager.java b/router/java/src/net/i2p/router/StatisticsManager.java
index 75dcd5ecd3f0f598085daf452b8adcb07f3e74a0..c943401257035920f4c9e78c1d42598d8073a6ad 100644
--- a/router/java/src/net/i2p/router/StatisticsManager.java
+++ b/router/java/src/net/i2p/router/StatisticsManager.java
@@ -85,7 +85,7 @@ public class StatisticsManager implements Service {
         if (_context.getBooleanPropertyDefaultTrue(PROP_PUBLISH_RANKINGS)) {
             long publishedUptime = _context.router().getUptime();
             // Don't publish these for first hour
-            if (publishedUptime > 62*60*1000)
+            if (publishedUptime > 62*60*1000 && CoreVersion.VERSION.equals("0.8.13"))
                 includeAverageThroughput(stats);
             //includeRate("router.invalidMessageTime", stats, new long[] { 10*60*1000 });
             //includeRate("router.duplicateMessageId", stats, new long[] { 24*60*60*1000 });
diff --git a/router/java/src/net/i2p/router/client/ClientConnectionRunner.java b/router/java/src/net/i2p/router/client/ClientConnectionRunner.java
index a7a136d1ddc4865369551bb9b49bbab4a86fa983..8597c52b2b183da972d991ee76a838af90dc7cf3 100644
--- a/router/java/src/net/i2p/router/client/ClientConnectionRunner.java
+++ b/router/java/src/net/i2p/router/client/ClientConnectionRunner.java
@@ -239,13 +239,13 @@ class ClientConnectionRunner {
     }
     
     /**
-     *  Send a DisconnectMessage and log with level Log.CRIT.
+     *  Send a DisconnectMessage and log with level Log.ERROR.
      *  This is always bad.
      *  See ClientMessageEventListener.handleCreateSession()
      *  for why we don't send a SessionStatusMessage when we do this.
      */
     void disconnectClient(String reason) {
-        disconnectClient(reason, Log.CRIT);
+        disconnectClient(reason, Log.ERROR);
     }
 
     /**
@@ -254,9 +254,10 @@ class ClientConnectionRunner {
      */
     void disconnectClient(String reason, int logLevel) {
         if (_log.shouldLog(logLevel))
-            _log.log(logLevel, "Disconnecting the client (" 
-                     + _config
-                     + ") : " + reason);
+            _log.log(logLevel, "Disconnecting the client - " 
+                     + reason
+                     + " config: "
+                     + _config);
         DisconnectMessage msg = new DisconnectMessage();
         msg.setReason(reason);
         try {
diff --git a/router/java/src/net/i2p/router/client/ClientManagerFacadeImpl.java b/router/java/src/net/i2p/router/client/ClientManagerFacadeImpl.java
index 94e2c0cf357dba26e6cc4cf81a16e023ca8c0816..022326cb6d263b9e8be097bbc8ec1119cf31e819 100644
--- a/router/java/src/net/i2p/router/client/ClientManagerFacadeImpl.java
+++ b/router/java/src/net/i2p/router/client/ClientManagerFacadeImpl.java
@@ -36,7 +36,7 @@ import net.i2p.util.Log;
  * @author jrandom
  */
 public class ClientManagerFacadeImpl extends ClientManagerFacade implements InternalClientManager {
-    private final static Log _log = new Log(ClientManagerFacadeImpl.class);
+    private final Log _log;
     private ClientManager _manager; 
     private RouterContext _context;
     /** note that this is different than the property the client side uses, i2cp.tcp.port */
@@ -48,7 +48,8 @@ public class ClientManagerFacadeImpl extends ClientManagerFacade implements Inte
     
     public ClientManagerFacadeImpl(RouterContext context) {
         _context = context;
-        _log.debug("Client manager facade created");
+        _log = _context.logManager().getLog(ClientManagerFacadeImpl.class);
+        //_log.debug("Client manager facade created");
     }
     
     public void startup() {
diff --git a/router/java/src/net/i2p/router/client/RequestLeaseSetJob.java b/router/java/src/net/i2p/router/client/RequestLeaseSetJob.java
index 516a652d22b8b6381b5d94ec42c28193556649c1..1ac0b93d0400b7d071f853cd625ef4bd8d35b55c 100644
--- a/router/java/src/net/i2p/router/client/RequestLeaseSetJob.java
+++ b/router/java/src/net/i2p/router/client/RequestLeaseSetJob.java
@@ -27,13 +27,13 @@ import net.i2p.util.Log;
  *
  */
 class RequestLeaseSetJob extends JobImpl {
-    private Log _log;
-    private ClientConnectionRunner _runner;
-    private LeaseSet _ls;
-    private long _expiration;
-    private Job _onCreate;
-    private Job _onFail;
-    private LeaseRequestState _requestState;
+    private final Log _log;
+    private final ClientConnectionRunner _runner;
+    private final LeaseSet _ls;
+    private final long _expiration;
+    private final Job _onCreate;
+    private final Job _onFail;
+    private final LeaseRequestState _requestState;
     
     public RequestLeaseSetJob(RouterContext ctx, ClientConnectionRunner runner, LeaseSet set, long expiration, Job onCreate, Job onFail, LeaseRequestState state) {
         super(ctx);
@@ -92,8 +92,8 @@ class RequestLeaseSetJob extends JobImpl {
      *
      */
     private class CheckLeaseRequestStatus extends JobImpl {
-        private LeaseRequestState _req;
-        private long _start;
+        private final LeaseRequestState _req;
+        private final long _start;
         
         public CheckLeaseRequestStatus(RouterContext enclosingContext, LeaseRequestState state) {
             super(enclosingContext);
@@ -114,9 +114,9 @@ class RequestLeaseSetJob extends JobImpl {
                 return;
             } else {
                 RequestLeaseSetJob.CheckLeaseRequestStatus.this.getContext().statManager().addRateData("client.requestLeaseSetTimeout", 1, 0);
-                if (_log.shouldLog(Log.CRIT)) {
+                if (_log.shouldLog(Log.ERROR)) {
                     long waited = System.currentTimeMillis() - _start;
-                    _log.log(Log.CRIT, "Failed to receive a leaseSet in the time allotted (" + waited + "): " + _req + " for " 
+                    _log.error("Failed to receive a leaseSet in the time allotted (" + waited + "): " + _req + " for " 
                              + _runner.getConfig().getDestination().calculateHash().toBase64());
                 }
                 _runner.disconnectClient("Took too long to request leaseSet");
diff --git a/router/java/src/net/i2p/router/networkdb/kademlia/FloodfillStoreJob.java b/router/java/src/net/i2p/router/networkdb/kademlia/FloodfillStoreJob.java
index edc760c3d8c2709fe7fe43c2eb5db17178b8bba4..581566287dc96548dcb703864a327a4507ffdaa6 100644
--- a/router/java/src/net/i2p/router/networkdb/kademlia/FloodfillStoreJob.java
+++ b/router/java/src/net/i2p/router/networkdb/kademlia/FloodfillStoreJob.java
@@ -68,10 +68,7 @@ class FloodfillStoreJob extends StoreJob {
             DatabaseEntry data = _state.getData();
             boolean isRouterInfo = data.getType() == DatabaseEntry.KEY_TYPE_ROUTERINFO;
             long published = data.getDate();
-            if (isRouterInfo) {
-                // Temporarily disable
-                return;
-            }
+
             // we should always have exactly one successful entry
             Hash sentTo = null;
             try {
diff --git a/router/java/src/net/i2p/router/networkdb/kademlia/FloodfillVerifyStoreJob.java b/router/java/src/net/i2p/router/networkdb/kademlia/FloodfillVerifyStoreJob.java
index 9fe6e2522109abcd9fefcc539b0f62188453a212..7b567d6126c1dc574daa5e7c69681c1550191770 100644
--- a/router/java/src/net/i2p/router/networkdb/kademlia/FloodfillVerifyStoreJob.java
+++ b/router/java/src/net/i2p/router/networkdb/kademlia/FloodfillVerifyStoreJob.java
@@ -216,7 +216,7 @@ public class FloodfillVerifyStoreJob extends JobImpl {
                 if (_log.shouldLog(Log.WARN))
                     _log.warn("Verify failed (older) for " + _key);
                 if (_log.shouldLog(Log.INFO))
-                    _log.info("Rcvd older lease: " + dsm.getEntry());
+                    _log.info("Rcvd older data: " + dsm.getEntry());
             } else if (_message instanceof DatabaseSearchReplyMessage) {
                 // assume 0 old, all new, 0 invalid, 0 dup
                 getContext().profileManager().dbLookupReply(_target,  0,
diff --git a/router/java/src/net/i2p/router/networkdb/kademlia/HandleFloodfillDatabaseStoreMessageJob.java b/router/java/src/net/i2p/router/networkdb/kademlia/HandleFloodfillDatabaseStoreMessageJob.java
index 9774f65bd80cbd95de7f869dfec9cd7804a6e27e..21c6a5ccd5f46a5e7b72bd5cbdd0669b3a058ae6 100644
--- a/router/java/src/net/i2p/router/networkdb/kademlia/HandleFloodfillDatabaseStoreMessageJob.java
+++ b/router/java/src/net/i2p/router/networkdb/kademlia/HandleFloodfillDatabaseStoreMessageJob.java
@@ -8,12 +8,13 @@ package net.i2p.router.networkdb.kademlia;
  *
  */
 
+import java.util.Collection;
 import java.util.Date;
-import java.util.Set;
 
 import net.i2p.data.DatabaseEntry;
 import net.i2p.data.Hash;
 import net.i2p.data.LeaseSet;
+import net.i2p.data.RouterAddress;
 import net.i2p.data.RouterIdentity;
 import net.i2p.data.RouterInfo;
 import net.i2p.data.i2np.DatabaseStoreMessage;
@@ -145,9 +146,9 @@ public class HandleFloodfillDatabaseStoreMessageJob extends JobImpl {
                             _log.shouldLog(Log.WARN))
                                 _log.warn("Blocklisting new peer " + key + ' ' + ri);
                     } else {
-                        Set oldAddr = prevNetDb.getAddresses();
-                        Set newAddr = ri.getAddresses();
-                        if (newAddr != null && (!newAddr.equals(oldAddr)) &&
+                        Collection<RouterAddress> oldAddr = prevNetDb.getAddresses();
+                        Collection<RouterAddress> newAddr = ri.getAddresses();
+                        if ((!newAddr.equals(oldAddr)) &&
                             (!getContext().shitlist().isShitlistedForever(key)) &&
                             getContext().blocklist().isBlocklisted(key) &&
                             _log.shouldLog(Log.WARN))
diff --git a/router/java/src/net/i2p/router/networkdb/kademlia/LocalHash.java b/router/java/src/net/i2p/router/networkdb/kademlia/LocalHash.java
index ebf787ad09466b6ed755a638940fa0eb0f31d726..53a28019c86f2ccf7474ba936869f719173c43c6 100644
--- a/router/java/src/net/i2p/router/networkdb/kademlia/LocalHash.java
+++ b/router/java/src/net/i2p/router/networkdb/kademlia/LocalHash.java
@@ -27,7 +27,7 @@ import net.i2p.util.Log;
  * @author moved from Hash.java by zzz
  */
 class LocalHash extends Hash {
-    private final static Log _log = new Log(LocalHash.class);
+    //private final static Log _log = new Log(LocalHash.class);
     private /* FIXME final FIXME */ Map<Hash, byte[]> _xorCache;
 
     private static final int MAX_CACHED_XOR = 1024;
@@ -86,6 +86,7 @@ class LocalHash extends Hash {
                 _xorCache.put(key, distance);
                 cached = _xorCache.size();
             }
+          /****
             if (_log.shouldLog(Log.DEBUG)) {
                 // explicit buffer, since the compiler can't guess how long it'll be
                 StringBuilder buf = new StringBuilder(128);
@@ -94,7 +95,9 @@ class LocalHash extends Hash {
                 buf.append(DataHelper.toHexString(key.getData()));
                 _log.debug(buf.toString(), new Exception());
             }
+          ****/
         } else {
+          /****
             if (_log.shouldLog(Log.DEBUG)) {
                 // explicit buffer, since the compiler can't guess how long it'll be
                 StringBuilder buf = new StringBuilder(128);
@@ -103,6 +106,7 @@ class LocalHash extends Hash {
                 buf.append(DataHelper.toHexString(key.getData()));
                 _log.debug(buf.toString());
             }
+          ****/
         }
         return distance;
     }
diff --git a/router/java/src/net/i2p/router/networkdb/kademlia/MessageWrapper.java b/router/java/src/net/i2p/router/networkdb/kademlia/MessageWrapper.java
index b53269e52134eb2e38bd36461fdbbed29e40f338..d8550791642cf8215bf2fb2d06da55593a22be26 100644
--- a/router/java/src/net/i2p/router/networkdb/kademlia/MessageWrapper.java
+++ b/router/java/src/net/i2p/router/networkdb/kademlia/MessageWrapper.java
@@ -27,7 +27,7 @@ import net.i2p.util.Log;
  */
 class MessageWrapper {
 
-    private static final Log _log = RouterContext.getGlobalContext().logManager().getLog(MessageWrapper.class);
+    //private static final Log _log = RouterContext.getGlobalContext().logManager().getLog(MessageWrapper.class);
 
     private static final int NETDB_TAGS_TO_DELIVER = 6;
     private static final int NETDB_LOW_THRESHOLD = 3;
@@ -71,8 +71,8 @@ class MessageWrapper {
         PublicKey sentTo = to.getIdentity().getPublicKey();
         if (!sentTags.isEmpty())
             tsh = skm.tagsDelivered(sentTo, sentKey, sentTags);
-        if (_log.shouldLog(Log.DEBUG))
-            _log.debug("Sent to: " + to.getIdentity().getHash() + " with key: " + sentKey + " and tags: " + sentTags.size());
+        //if (_log.shouldLog(Log.DEBUG))
+        //    _log.debug("Sent to: " + to.getIdentity().getHash() + " with key: " + sentKey + " and tags: " + sentTags.size());
         return new WrappedMessage(msg, skm, sentTo, sentKey, tsh);
     }
 
@@ -103,8 +103,8 @@ class MessageWrapper {
         void acked() {
             if (this.tsh != null) {
                 this.skm.tagsAcked(this.sentTo, this.sessionKey, this.tsh);
-                if (_log.shouldLog(Log.DEBUG))
-                    _log.debug("Tags acked for key: " + this.sessionKey);
+                //if (_log.shouldLog(Log.DEBUG))
+                //    _log.debug("Tags acked for key: " + this.sessionKey);
             }
         }
 
@@ -112,8 +112,8 @@ class MessageWrapper {
         void fail() {
             if (this.tsh != null) {
                 this.skm.failTags(this.sentTo, this.sessionKey, this.tsh);
-                if (_log.shouldLog(Log.DEBUG))
-                    _log.debug("Tags NOT acked for key: " + this.sessionKey);
+                //if (_log.shouldLog(Log.DEBUG))
+                //    _log.debug("Tags NOT acked for key: " + this.sessionKey);
             }
         }
     }    
diff --git a/router/java/src/net/i2p/router/networkdb/kademlia/StoreJob.java b/router/java/src/net/i2p/router/networkdb/kademlia/StoreJob.java
index 4620e3cffa9b614a8054b54c735801fe4ace89c1..e86b9b680700e7c05eb2733e774eb1d919b24dce 100644
--- a/router/java/src/net/i2p/router/networkdb/kademlia/StoreJob.java
+++ b/router/java/src/net/i2p/router/networkdb/kademlia/StoreJob.java
@@ -98,6 +98,8 @@ class StoreJob extends JobImpl {
 
     /**
      * send the key to the next batch of peers
+     *
+     * Synchronized to enforce parallelization limits and prevent dups
      */
     private void sendNext() {
         if (_state.completed()) {
@@ -130,8 +132,9 @@ class StoreJob extends JobImpl {
      * the routing table, but making sure no more than PARALLELIZATION are outstanding
      * at any time
      *
+     * Caller should synchronize to enforce parallelization limits and prevent dups
      */
-    private void continueSending() { 
+    private synchronized void continueSending() { 
         if (_state.completed()) return;
         int toCheck = getParallelization() - _state.getPending().size();
         if (toCheck <= 0) {
diff --git a/router/java/src/net/i2p/router/networkdb/kademlia/StoreState.java b/router/java/src/net/i2p/router/networkdb/kademlia/StoreState.java
index 9666c09be520533526c44e46a9621b9b6dce8f37..876e2f4af8eaddeb7c64894bd1f4534e8f963f3d 100644
--- a/router/java/src/net/i2p/router/networkdb/kademlia/StoreState.java
+++ b/router/java/src/net/i2p/router/networkdb/kademlia/StoreState.java
@@ -14,15 +14,15 @@ import net.i2p.data.Hash;
 import net.i2p.router.RouterContext;
 
 /**
- *
+ *  Tracks the state of a StoreJob
  */
 class StoreState {
-    private RouterContext _context;
-    private Hash _key;
-    private DatabaseEntry _data;
+    private final RouterContext _context;
+    private final Hash _key;
+    private final DatabaseEntry _data;
     private final HashSet<Hash> _pendingPeers;
-    private Map<Hash, Long> _pendingPeerTimes;
-    private Map<Hash, MessageWrapper.WrappedMessage> _pendingMessages;
+    private final Map<Hash, Long> _pendingPeerTimes;
+    private final Map<Hash, MessageWrapper.WrappedMessage> _pendingMessages;
     private final HashSet<Hash> _successfulPeers;
     //private final HashSet<Hash> _successfulExploratoryPeers;
     private final HashSet<Hash> _failedPeers;
diff --git a/router/java/src/net/i2p/router/networkdb/reseed/ReseedChecker.java b/router/java/src/net/i2p/router/networkdb/reseed/ReseedChecker.java
index 6bb816985da20e829434de112d4e815c3b47fc43..ea3e0408d37d01ea9ad21e0b9fbe8e4c34b20077 100644
--- a/router/java/src/net/i2p/router/networkdb/reseed/ReseedChecker.java
+++ b/router/java/src/net/i2p/router/networkdb/reseed/ReseedChecker.java
@@ -32,14 +32,16 @@ public class ReseedChecker {
         File noReseedFileAlt1 = new File(new File(System.getProperty("user.home")), "noreseed.i2p");
         File noReseedFileAlt2 = new File(context.getConfigDir(), ".i2pnoreseed");
         File noReseedFileAlt3 = new File(context.getConfigDir(), "noreseed.i2p");
+        Log _log = context.logManager().getLog(ReseedChecker.class);
         if (!noReseedFile.exists() && !noReseedFileAlt1.exists() && !noReseedFileAlt2.exists() && !noReseedFileAlt3.exists()) {
-            Log _log = context.logManager().getLog(ReseedChecker.class);
             if (count <= 1)
                 _log.logAlways(Log.INFO, "Downloading peer router information for a new I2P installation");
             else
                 _log.logAlways(Log.WARN, "Very few known peers remaining - reseeding now");
             Reseeder reseeder = new Reseeder(context);
             reseeder.requestReseed();
+        } else {
+            _log.logAlways(Log.WARN, "Only " + count + " peers remaining but reseed disabled by config file");
         }
     }
 }
diff --git a/router/java/src/net/i2p/router/peermanager/ProfileOrganizer.java b/router/java/src/net/i2p/router/peermanager/ProfileOrganizer.java
index 8b14d22a0f5a8ce73d096b5a318d46508f56ebe9..d8459166a5be84506e92fde941c20e68468cc84a 100644
--- a/router/java/src/net/i2p/router/peermanager/ProfileOrganizer.java
+++ b/router/java/src/net/i2p/router/peermanager/ProfileOrganizer.java
@@ -7,6 +7,7 @@ import java.net.UnknownHostException;
 import java.text.DecimalFormat;
 import java.text.DecimalFormatSymbols;
 import java.util.ArrayList;
+import java.util.Collection;
 import java.util.Collections;
 import java.util.HashMap;
 import java.util.HashSet;
@@ -1258,7 +1259,7 @@ public class ProfileOrganizer {
         RouterInfo pinfo = _context.netDb().lookupRouterInfoLocally(peer);
         if (pinfo == null)
             return rv;
-        Set<RouterAddress> paddr = pinfo.getAddresses();
+        Collection<RouterAddress> paddr = pinfo.getAddresses();
         if (paddr == null)
             return rv;
         for (RouterAddress pa : paddr) {
diff --git a/router/java/src/net/i2p/router/transport/TransportManager.java b/router/java/src/net/i2p/router/transport/TransportManager.java
index 8b6b29713d63307d6445fe041acc94b98a3a526e..d0a10f31b5b7f7183644476ce52c6798cc3bd7d1 100644
--- a/router/java/src/net/i2p/router/transport/TransportManager.java
+++ b/router/java/src/net/i2p/router/transport/TransportManager.java
@@ -518,9 +518,9 @@ public class TransportManager implements TransportEventListener {
         buf.append("<h3>").append(_("Definitions")).append("</h3><div class=\"configure\">" +
                    "<p><b id=\"def.peer\">").append(_("Peer")).append("</b>: ").append(_("The remote peer, identified by router hash")).append("<br>\n" +
                    "<b id=\"def.dir\">").append(_("Dir")).append("</b>: " +
-                   "<img src=\"/themes/console/images/inbound.png\"> ").append(_("Inbound connection")).append("<br>\n" +
+                   "<img alt=\"Inbound\" src=\"/themes/console/images/inbound.png\"> ").append(_("Inbound connection")).append("<br>\n" +
                    "&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;" +
-                   "<img src=\"/themes/console/images/outbound.png\"> ").append(_("Outbound connection")).append("<br>\n" +
+                   "<img alt=\"Outbound\" src=\"/themes/console/images/outbound.png\"> ").append(_("Outbound connection")).append("<br>\n" +
                    "&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;" +
                    "<img src=\"/themes/console/images/inbound.png\" alt=\"V\" height=\"8\" width=\"12\"> ").append(_("They offered to introduce us (help other peers traverse our firewall)")).append("<br>\n" +
                    "&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;" +
@@ -531,7 +531,7 @@ public class TransportManager implements TransportEventListener {
                    "<b id=\"def.skew\">").append(_("Skew")).append("</b>: ").append(_("The difference between the peer's clock and your own")).append("<br>\n" +
                    "<b id=\"def.cwnd\">CWND</b>: ").append(_("The congestion window, which is how many bytes can be sent without an acknowledgement")).append(" / <br>\n" +
                    "&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; ").append(_("The number of sent messages awaiting acknowledgement")).append(" /<br>\n" +
-                   "&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; ").append(_("The maximum number of concurrent messages to send")).append(" /<br>\n"+ 
+                   "&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; ").append(_("The maximum number of concurrent messages to send")).append(" /<br>\n"+
                    "&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; ").append(_("The number of pending sends which exceed congestion window")).append("<br>\n" +
                    "<b id=\"def.ssthresh\">SST</b>: ").append(_("The slow start threshold")).append("<br>\n" +
                    "<b id=\"def.rtt\">RTT</b>: ").append(_("The round trip time in milliseconds")).append("<br>\n" +
diff --git a/router/java/src/net/i2p/router/transport/ntcp/NTCPAddress.java b/router/java/src/net/i2p/router/transport/ntcp/NTCPAddress.java
index acced89f433a2cbd9fbf5687ef5bba52d1021f5d..a51233ad631f53a104430cea2b9cfe7e512f9b71 100644
--- a/router/java/src/net/i2p/router/transport/ntcp/NTCPAddress.java
+++ b/router/java/src/net/i2p/router/transport/ntcp/NTCPAddress.java
@@ -11,6 +11,7 @@ package net.i2p.router.transport.ntcp;
 import java.net.InetAddress;
 import java.util.Properties;
 
+import net.i2p.I2PAppContext;
 import net.i2p.data.DataHelper;
 import net.i2p.data.RouterAddress;
 import net.i2p.router.transport.TransportImpl;
@@ -20,7 +21,6 @@ import net.i2p.util.Log;
  * Wrap up an address 
  */
 public class NTCPAddress {
-    private final static Log _log = new Log(NTCPAddress.class);
     private int _port;
     private String _host;
     //private InetAddress _addr;
@@ -68,7 +68,8 @@ public class NTCPAddress {
                 try {
                     _port = Integer.parseInt(port.trim());
                 } catch (NumberFormatException nfe) {
-                    _log.error("Invalid port [" + port + "]", nfe);
+                    Log log = I2PAppContext.getGlobalContext().logManager().getLog(NTCPAddress.class);
+                    log.error("Invalid port [" + port + "]", nfe);
                     _port = -1;
                 }
             } else {
@@ -119,8 +120,8 @@ public class NTCPAddress {
             //}
             return TransportImpl.isPubliclyRoutable(quad);
         } catch (Throwable t) {
-            if (_log.shouldLog(Log.WARN))
-                _log.warn("Error checking routability", t);
+            //if (_log.shouldLog(Log.WARN))
+            //    _log.warn("Error checking routability", t);
             return false;
         }
     }
diff --git a/router/java/src/net/i2p/router/transport/udp/UDPPacket.java b/router/java/src/net/i2p/router/transport/udp/UDPPacket.java
index c45a29bdb1b0ed8c5eda33c5d648a8e713ddbb42..eb4c3e23c50d28d2aa029c8c2a968d0c2a6ece18 100644
--- a/router/java/src/net/i2p/router/transport/udp/UDPPacket.java
+++ b/router/java/src/net/i2p/router/transport/udp/UDPPacket.java
@@ -18,7 +18,6 @@ import net.i2p.util.Log;
  */
 class UDPPacket {
     private I2PAppContext _context;
-    private static Log _log;
     private final DatagramPacket _packet;
     private volatile short _priority;
     private volatile long _initializeTime;
@@ -48,7 +47,6 @@ class UDPPacket {
             _packetCache = new LinkedBlockingQueue(CACHE_SIZE);
         else
             _packetCache = null;
-        _log = I2PAppContext.getGlobalContext().logManager().getLog(UDPPacket.class);
     }
     
     /**
@@ -214,8 +212,8 @@ class UDPPacket {
             eq = DataHelper.eq(hmac.getData(), 0, _data, _packet.getOffset(), MAC_SIZE);
              */
         } else {
-            if (_log.shouldLog(Log.WARN))
-                _log.warn("Payload length is " + payloadLength);
+            //if (_log.shouldLog(Log.WARN))
+            //    _log.warn("Payload length is " + payloadLength);
         }
         
         _afterValidate = _context.clock().now();
@@ -321,9 +319,10 @@ class UDPPacket {
     private void verifyNotReleased() {
         if (CACHE) return;
         if (_released) {
-            _log.log(Log.CRIT, "Already released.  current stack trace is:", new Exception());
-            _log.log(Log.CRIT, "Released by: ", _releasedBy);
-            _log.log(Log.CRIT, "Acquired by: ", _acquiredBy);
+            Log log = I2PAppContext.getGlobalContext().logManager().getLog(UDPPacket.class);
+            log.log(Log.CRIT, "Already released.  current stack trace is:", new Exception());
+            log.log(Log.CRIT, "Released by: ", _releasedBy);
+            log.log(Log.CRIT, "Acquired by: ", _acquiredBy);
         }
     }
 }