diff --git a/apps/BOB/nbproject/project.properties b/apps/BOB/nbproject/project.properties
index ee493f7643478691570bad67cc658046b11b3c15..9eeffd29f1470219c12f805941a06cf5f7347ad8 100644
--- a/apps/BOB/nbproject/project.properties
+++ b/apps/BOB/nbproject/project.properties
@@ -27,6 +27,7 @@ excludes=
 file.reference.build-javadoc=../../i2p.i2p/build/javadoc
 file.reference.i2p.jar=../../core/java/build/i2p.jar
 file.reference.i2ptunnel.jar=../i2ptunnel/java/build/i2ptunnel.jar
+file.reference.jbigi.jar=../../installer/lib/jbigi/jbigi.jar
 file.reference.mstreaming.jar=../ministreaming/java/build/mstreaming.jar
 file.reference.router.jar=../../router/java/build/router.jar
 file.reference.streaming.jar=../streaming/java/build/streaming.jar
diff --git a/apps/BOB/src/net/i2p/BOB/BOB.java b/apps/BOB/src/net/i2p/BOB/BOB.java
index 59b46b8d7475f10632e37b422d250b41c713a5bb..48a0e067c6efc25d99cd80ba7724bc68c3755d2d 100644
--- a/apps/BOB/src/net/i2p/BOB/BOB.java
+++ b/apps/BOB/src/net/i2p/BOB/BOB.java
@@ -34,6 +34,7 @@ import java.util.Properties;
 import net.i2p.client.I2PClient;
 import net.i2p.client.streaming.RetransmissionTimer;
 import net.i2p.util.Log;
+
 /**
  * <span style="font-size:8px;font-family:courier;color:#EEEEEE;background-color:#000000">
  * ################################################################################<br>
@@ -157,12 +158,12 @@ public class BOB {
 		boolean save = false;
 		// Set up all defaults to be passed forward to other threads.
 		// Re-reading the config file in each thread is pretty damn stupid.
-		// I2PClient client = I2PClientFactory.createClient();
 		String configLocation = System.getProperty(PROP_CONFIG_LOCATION, "bob.config");
 
 		// This is here just to ensure there is no interference with our threadgroups.
 		RetransmissionTimer Y = RetransmissionTimer.getInstance();
 		i = Y.hashCode();
+
 		{
 			try {
 				FileInputStream fi = new FileInputStream(configLocation);
diff --git a/apps/BOB/src/net/i2p/BOB/DoCMDS.java b/apps/BOB/src/net/i2p/BOB/DoCMDS.java
index 099d69fecac4e6d6a6378aac9be51a6216891b14..12d8d615607697286bbcadd649bb5fe1814f5565 100644
--- a/apps/BOB/src/net/i2p/BOB/DoCMDS.java
+++ b/apps/BOB/src/net/i2p/BOB/DoCMDS.java
@@ -46,7 +46,7 @@ public class DoCMDS implements Runnable {
 
 	// FIX ME
 	// I need a better way to do versioning, but this will do for now.
-	public static final String BMAJ = "00",  BMIN = "00",  BREV = "04",  BEXT = "";
+	public static final String BMAJ = "00",  BMIN = "00",  BREV = "05",  BEXT = "";
 	public static final String BOBversion = BMAJ + "." + BMIN + "." + BREV + BEXT;
 	private Socket server;
 	private Properties props;
diff --git a/apps/BOB/src/net/i2p/BOB/I2Plistener.java b/apps/BOB/src/net/i2p/BOB/I2Plistener.java
index c59683270e585b5127ba8020ee4768a49fec6a02..813bb5d34387b7d63b6ce985d16d2c77229c2a4e 100644
--- a/apps/BOB/src/net/i2p/BOB/I2Plistener.java
+++ b/apps/BOB/src/net/i2p/BOB/I2Plistener.java
@@ -62,6 +62,26 @@ public class I2Plistener implements Runnable {
 		tgwatch = 1;
 	}
 
+	private void rlock() throws Exception {
+		database.getReadLock();
+		info.getReadLock();
+	}
+
+	private void runlock() throws Exception {
+		database.releaseReadLock();
+		info.releaseReadLock();
+	}
+
+	private void wlock() throws Exception {
+		database.getWriteLock();
+		info.getWriteLock();
+	}
+
+	private void wunlock() throws Exception {
+		info.releaseWriteLock();
+		database.releaseWriteLock();
+	}
+
 	/**
 	 * Simply listen on I2P port, and thread connections
 	 *
@@ -70,68 +90,86 @@ public class I2Plistener implements Runnable {
 		boolean g = false;
 		I2PSocket sessSocket = null;
 
-		serverSocket.setSoTimeout(50);
-		database.getReadLock();
-		info.getReadLock();
-		if(info.exists("INPORT")) {
-			tgwatch = 2;
-		}
-		info.releaseReadLock();
-		database.releaseReadLock();
-		boolean spin = true;
-		while(spin) {
+die:		{
 
-			database.getReadLock();
-			info.getReadLock();
-			spin = info.get("RUNNING").equals(Boolean.TRUE);
-			info.releaseReadLock();
-			database.releaseReadLock();
+			serverSocket.setSoTimeout(50);
 			try {
+				if (info.exists("INPORT")) {
+					tgwatch = 2;
+				}
+			} catch (Exception e) {
+				try {
+					runlock();
+				} catch (Exception e2) {
+					break die;
+				}
+				break die;
+			}
+			boolean spin = true;
+			while (spin) {
+
 				try {
-					sessSocket = serverSocket.accept();
-					g = true;
-				} catch(ConnectException ce) {
-					g = false;
-				} catch(SocketTimeoutException ste) {
-					g = false;
+					rlock();
+				} catch (Exception e) {
+					break die;
 				}
-				if(g) {
-					g = false;
-					// toss the connection to a new thread.
-					I2PtoTCP conn_c = new I2PtoTCP(sessSocket, info, database);
-					Thread t = new Thread(conn_c, "BOBI2PtoTCP");
-					t.start();
+				try {
+					spin = info.get("RUNNING").equals(Boolean.TRUE);
+				} catch (Exception e) {
+					try {
+						runlock();
+					} catch (Exception e2) {
+						break die;
+					}
+					break die;
 				}
+				try {
+					try {
+						sessSocket = serverSocket.accept();
+						g = true;
+					} catch (ConnectException ce) {
+						g = false;
+					} catch (SocketTimeoutException ste) {
+						g = false;
+					}
+					if (g) {
+						g = false;
+						// toss the connection to a new thread.
+						I2PtoTCP conn_c = new I2PtoTCP(sessSocket, info, database);
+						Thread t = new Thread(conn_c, "BOBI2PtoTCP");
+						t.start();
+					}
 
-			} catch(I2PException e) {
-				//	System.out.println("Exception " + e);
+				} catch (I2PException e) {
+					//	System.out.println("Exception " + e);
+				}
 			}
 		}
 		// System.out.println("I2Plistener: Close");
 		try {
 			serverSocket.close();
-		} catch(I2PException e) {
+		} catch (I2PException e) {
 			// nop
 		}
 		// need to kill off the socket manager too.
 		I2PSession session = socketManager.getSession();
-		if(session != null) {
+		if (session != null) {
 			// System.out.println("I2Plistener: destroySession");
 			try {
 				session.destroySession();
-			} catch(I2PSessionException ex) {
+			} catch (I2PSessionException ex) {
 				// nop
 			}
 		}
 		// System.out.println("I2Plistener: Waiting for children");
-		while(Thread.activeCount() > tgwatch) { // wait for all threads in our threadgroup to finish
+		while (Thread.activeCount() > tgwatch) { // wait for all threads in our threadgroup to finish
 			try {
 				Thread.sleep(100); //sleep for 100 ms (One tenth second)
-			} catch(Exception e) {
+			} catch (Exception e) {
 				// nop
 			}
 		}
 
-		// System.out.println("I2Plistener: Done.");
+	// System.out.println("I2Plistener: Done.");
 	}
 }
diff --git a/apps/BOB/src/net/i2p/BOB/I2PtoTCP.java b/apps/BOB/src/net/i2p/BOB/I2PtoTCP.java
index 5d24e19d36bcf629d2c24f2dd3156db0d0181900..06c3131fee8ba62dfd7172ab5ff5a92aa15810e1 100644
--- a/apps/BOB/src/net/i2p/BOB/I2PtoTCP.java
+++ b/apps/BOB/src/net/i2p/BOB/I2PtoTCP.java
@@ -105,8 +105,8 @@ die:            {
 					out.flush(); // not really needed, but...
 				}
 				// setup to cross the streams
-				TCPio conn_c = new TCPio(in, Iout, info, database); // app -> I2P
-				TCPio conn_a = new TCPio(Iin, out, info, database); // I2P -> app
+				TCPio conn_c = new TCPio(in, Iout /*, info, database */ ); // app -> I2P
+				TCPio conn_a = new TCPio(Iin, out /* , info, database */); // I2P -> app
 				Thread t = new Thread(conn_c, "TCPioA");
 				Thread q = new Thread(conn_a, "TCPioB");
 				// Fire!
diff --git a/apps/BOB/src/net/i2p/BOB/MUXlisten.java b/apps/BOB/src/net/i2p/BOB/MUXlisten.java
index 89ab53fe62f9f8fad5afee5c08d1a511b2a735c9..879cf9a64a1e4bad3f93d41e33971e780120df75 100644
--- a/apps/BOB/src/net/i2p/BOB/MUXlisten.java
+++ b/apps/BOB/src/net/i2p/BOB/MUXlisten.java
@@ -100,7 +100,7 @@ public class MUXlisten implements Runnable {
 		// Everything is OK as far as we can tell.
 		this.database.getWriteLock();
 		this.info.getWriteLock();
-		this.info.add("STARTING", Boolean.TRUE);
+		this.info.add("STARTING", new Boolean(true));
 		this.info.releaseWriteLock();
 		this.database.releaseWriteLock();
 	}
@@ -134,8 +134,8 @@ public class MUXlisten implements Runnable {
 		try {
 			wlock();
 			try {
-				info.add("RUNNING", Boolean.TRUE);
-				info.add("STARTING", Boolean.FALSE);
+				info.add("RUNNING", new Boolean(true));
+				info.add("STARTING", new Boolean(false));
 			} catch(Exception e) {
 				wunlock();
 				return;
@@ -198,7 +198,7 @@ die:                            {
 					try {
 						wlock();
 						try {
-							info.add("RUNNING", Boolean.FALSE);
+							info.add("RUNNING", new Boolean(false));
 						} catch(Exception e) {
 							wunlock();
 							break die;
@@ -255,9 +255,9 @@ die:                            {
 		try {
 			wlock();
 			try {
-				info.add("STARTING", Boolean.FALSE);
-				info.add("STOPPING", Boolean.FALSE);
-				info.add("RUNNING", Boolean.FALSE);
+				info.add("STARTING", new Boolean(false));
+				info.add("STOPPING", new Boolean(false));
+				info.add("RUNNING", new Boolean(false));
 			} catch(Exception e) {
 				wunlock();
 				return;
diff --git a/apps/BOB/src/net/i2p/BOB/Main.java b/apps/BOB/src/net/i2p/BOB/Main.java
index 2d81fb30eda99fe34341d11f953f72cb45e18fde..b32e4758efbf6952b0f549ced189be5198a947cc 100644
--- a/apps/BOB/src/net/i2p/BOB/Main.java
+++ b/apps/BOB/src/net/i2p/BOB/Main.java
@@ -24,7 +24,7 @@
 package net.i2p.BOB;
 
 import net.i2p.client.streaming.RetransmissionTimer;
-
+import net.i2p.util.SimpleScheduler;
 /**
  * Start from command line
  *
@@ -39,6 +39,9 @@ public class Main {
 	public static void main(String[] args) {
 		// THINK THINK THINK THINK THINK THINK
 		RetransmissionTimer Y = RetransmissionTimer.getInstance();
+		// needs SimpleScheduler
+		// no way to stop the scheduler?!?
+		SimpleScheduler.getInstance();
 		BOB.main(args);
 		Y.stop();
 	}
diff --git a/apps/BOB/src/net/i2p/BOB/TCPio.java b/apps/BOB/src/net/i2p/BOB/TCPio.java
index c9f4ab64cd069c561fb4f55ff11a4f2ca5d85aeb..109d8e8cb56f8b66454009b0188a5d00cf6e5f58 100644
--- a/apps/BOB/src/net/i2p/BOB/TCPio.java
+++ b/apps/BOB/src/net/i2p/BOB/TCPio.java
@@ -35,7 +35,7 @@ public class TCPio implements Runnable {
 
 	private InputStream Ain;
 	private OutputStream Aout;
-	private NamedDB info,  database;
+	// private NamedDB info,  database;
 
 	/**
 	 * Constructor
@@ -43,13 +43,14 @@ public class TCPio implements Runnable {
 	 * @param Ain
 	 * @param Aout
 	 * @param info
-	 * @param database
+	 *
+	 * param database
 	 */
-	TCPio(InputStream Ain, OutputStream Aout, NamedDB info, NamedDB database) {
+	TCPio(InputStream Ain, OutputStream Aout /*, NamedDB info , NamedDB database */) {
 		this.Ain = Ain;
 		this.Aout = Aout;
-		this.info = info;
-		this.database = database;
+		// this.info = info;
+		// this.database = database;
 	}
 
 	/**
@@ -86,11 +87,11 @@ public class TCPio implements Runnable {
 		boolean spin = true;
 		try {
 			while(spin) {
-				database.getReadLock();
-				info.getReadLock();
-				spin = info.get("RUNNING").equals(Boolean.TRUE);
-				info.releaseReadLock();
-				database.releaseReadLock();
+				// database.getReadLock();
+				// info.getReadLock();
+				// spin = info.get("RUNNING").equals(Boolean.TRUE);
+				// info.releaseReadLock();
+				// database.releaseReadLock();
 				b = Ain.read(a, 0, 1);
 				// System.out.println(info.get("NICKNAME").toString() + " " + b);
 				if(b > 0) {
diff --git a/apps/BOB/src/net/i2p/BOB/TCPlistener.java b/apps/BOB/src/net/i2p/BOB/TCPlistener.java
index 30380a55dd0241fb104fb36a71dfbfaa4c6959de..7e931768c90692b2c16c6b87cb453455903c7f7a 100644
--- a/apps/BOB/src/net/i2p/BOB/TCPlistener.java
+++ b/apps/BOB/src/net/i2p/BOB/TCPlistener.java
@@ -63,6 +63,26 @@ public class TCPlistener implements Runnable {
 		tgwatch = 1;
 	}
 
+	private void rlock() throws Exception {
+		database.getReadLock();
+		info.getReadLock();
+	}
+
+	private void runlock() throws Exception {
+		database.releaseReadLock();
+		info.releaseReadLock();
+	}
+
+	private void wlock() throws Exception {
+		database.getWriteLock();
+		info.getWriteLock();
+	}
+
+	private void wunlock() throws Exception {
+		info.releaseWriteLock();
+		database.releaseWriteLock();
+	}
+
 	/**
 	 * Simply listen on TCP port, and thread connections
 	 *
@@ -70,77 +90,121 @@ public class TCPlistener implements Runnable {
 	public void run() {
 		boolean g = false;
 		boolean spin = true;
-		database.getReadLock();
-		info.getReadLock();
-		if(info.exists("OUTPORT")) {
-			tgwatch = 2;
-		}
-		try {
-			Socket server = new Socket();
-			listener.setSoTimeout(50); // Half of the expected time from MUXlisten
-			info.releaseReadLock();
-			database.releaseReadLock();
-			while(spin) {
-				database.getReadLock();
-				info.getReadLock();
-				spin = info.get("RUNNING").equals(Boolean.TRUE);
-				info.releaseReadLock();
-				database.releaseReadLock();
-				try {
-					server = listener.accept();
-					g = true;
-				} catch(SocketTimeoutException ste) {
-					g = false;
+
+die:		{
+			try {
+				rlock();
+			} catch (Exception e) {
+				break die;
+			}
+			try {
+				if (info.exists("OUTPORT")) {
+					tgwatch = 2;
 				}
-				if(g) {
-					// toss the connection to a new thread.
-					TCPtoI2P conn_c = new TCPtoI2P(socketManager, server, info, database);
-					Thread t = new Thread(conn_c, "BOBTCPtoI2P");
-					t.start();
-					g = false;
+			} catch (Exception e) {
+				try {
+					runlock();
+				} catch (Exception e2) {
+					break die;
 				}
+				break die;
 			}
-			//System.out.println("TCPlistener: destroySession");
-			listener.close();
-		} catch(IOException ioe) {
 			try {
-				listener.close();
-			} catch(IOException e) {
+				runlock();
+			} catch (Exception e) {
+				break die;
 			}
-			// Fatal failure, cause a stop event
-			database.getReadLock();
-			info.getReadLock();
-			spin = info.get("RUNNING").equals(Boolean.TRUE);
-			info.releaseReadLock();
-			database.releaseReadLock();
-			if(spin) {
-				database.getWriteLock();
-				info.getWriteLock();
-				info.add("STOPPING", new Boolean(true));
-				info.add("RUNNING", new Boolean(false));
-				info.releaseWriteLock();
-				database.releaseWriteLock();
+			try {
+				Socket server = new Socket();
+				listener.setSoTimeout(50); // Half of the expected time from MUXlisten
+				while (spin) {
+					try {
+						rlock();
+					} catch (Exception e) {
+						break die;
+					}
+					try {
+						spin = info.get("RUNNING").equals(Boolean.TRUE);
+					} catch (Exception e) {
+						try {
+							runlock();
+						} catch (Exception e2) {
+							break die;
+						}
+						break die;
+					}
+					try {
+						server = listener.accept();
+						g = true;
+					} catch (SocketTimeoutException ste) {
+						g = false;
+					}
+					if (g) {
+						// toss the connection to a new thread.
+						TCPtoI2P conn_c = new TCPtoI2P(socketManager, server /* , info, database */);
+						Thread t = new Thread(conn_c, "BOBTCPtoI2P");
+						t.start();
+						g = false;
+					}
+				}
+				//System.out.println("TCPlistener: destroySession");
+				listener.close();
+			} catch (IOException ioe) {
+				try {
+					listener.close();
+				} catch (IOException e) {
+				}
+				// Fatal failure, cause a stop event
+				try {
+					rlock();
+					try {
+						spin = info.get("RUNNING").equals(Boolean.TRUE);
+					} catch (Exception e) {
+						runlock();
+						break die;
+					}
+				} catch (Exception e) {
+					break die;
+				}
+				if (spin) {
+					try {
+						wlock();
+						try {
+							info.add("STOPPING", new Boolean(true));
+							info.add("RUNNING", new Boolean(false));
+						} catch (Exception e) {
+							wunlock();
+							break die;
+						}
+					} catch (Exception e) {
+						break die;
+					}
+					try {
+						wunlock();
+					} catch (Exception e) {
+						break die;
+					}
+				}
 			}
 		}
-
 		// need to kill off the socket manager too.
 		I2PSession session = socketManager.getSession();
-		if(session != null) {
+		if (session != null) {
 			try {
 				session.destroySession();
-			} catch(I2PSessionException ex) {
+			} catch (I2PSessionException ex) {
 				// nop
 			}
 		}
 		//System.out.println("TCPlistener: Waiting for children");
-		while(Thread.activeCount() > tgwatch) { // wait for all threads in our threadgroup to finish
+		while (Thread.activeCount() > tgwatch) { // wait for all threads in our threadgroup to finish
 			try {
 				Thread.sleep(100); //sleep for 100 ms (One tenth second)
-			} catch(Exception e) {
+			} catch (Exception e) {
 				// nop
-				}
+			}
 		}
-		//System.out.println("TCPlistener: Done.");
+	//System.out.println("TCPlistener: Done.");
 	}
 }
 
diff --git a/apps/BOB/src/net/i2p/BOB/TCPtoI2P.java b/apps/BOB/src/net/i2p/BOB/TCPtoI2P.java
index df61e78e1b1fc7700d6cc7cd7c5b16d2ef4e446b..fe1ca327881d051c64691ceeb0504749f61bd0bd 100644
--- a/apps/BOB/src/net/i2p/BOB/TCPtoI2P.java
+++ b/apps/BOB/src/net/i2p/BOB/TCPtoI2P.java
@@ -45,7 +45,7 @@ import net.i2p.i2ptunnel.I2PTunnel;
 public class TCPtoI2P implements Runnable {
 
 	private I2PSocket I2P;
-	private NamedDB info,  database;
+	// private NamedDB info,  database;
 	private Socket sock;
 	private I2PSocketManager socketManager;
 
@@ -84,13 +84,13 @@ public class TCPtoI2P implements Runnable {
 	 * Constructor
 	 * @param i2p
 	 * @param socket
-	 * @param info
-	 * @param database
+	 * param info
+	 * param database
 	 */
-	TCPtoI2P(I2PSocketManager i2p, Socket socket, NamedDB info, NamedDB database) {
+	TCPtoI2P(I2PSocketManager i2p, Socket socket /*, NamedDB info, NamedDB database */) {
 		this.sock = socket;
-		this.info = info;
-		this.database = database;
+		// this.info = info;
+		// this.database = database;
 		this.socketManager = i2p;
 	}
 
@@ -110,6 +110,7 @@ public class TCPtoI2P implements Runnable {
 
 	/**
 	 * TCP stream to I2P stream thread starter
+	 *
 	 */
 	public void run() {
 		String line, input;
@@ -138,8 +139,8 @@ public class TCPtoI2P implements Runnable {
 					InputStream Iin = I2P.getInputStream();
 					OutputStream Iout = I2P.getOutputStream();
 					// setup to cross the streams
-					TCPio conn_c = new TCPio(in, Iout, info, database); // app -> I2P
-					TCPio conn_a = new TCPio(Iin, out, info, database); // I2P -> app
+					TCPio conn_c = new TCPio(in, Iout /*, info, database */); // app -> I2P
+					TCPio conn_a = new TCPio(Iin, out /*, info, database */); // I2P -> app
 					Thread t = new Thread(conn_c, "TCPioA");
 					Thread q = new Thread(conn_a, "TCPioB");
 					// Fire!
@@ -167,7 +168,8 @@ public class TCPtoI2P implements Runnable {
 			} catch(Exception e) {
 				Emsg("ERROR " + e.toString(), out);
 			}
-		} catch(IOException ioe) {
+		} catch(Exception e) {
+			// bail on anything else
 		}
 		try {
 			// System.out.println("TCPtoI2P: Close I2P");
@@ -181,6 +183,5 @@ public class TCPtoI2P implements Runnable {
 		} catch(Exception e) {
 		}
 		// System.out.println("TCPtoI2P: Done.");
-
 	}
 }
diff --git a/router/java/src/net/i2p/router/RouterVersion.java b/router/java/src/net/i2p/router/RouterVersion.java
index d1e87948b170206a143fd46b908dbf9a47f0171f..eeea3faa97532098603c3aeaabb131ed6d53b50c 100644
--- a/router/java/src/net/i2p/router/RouterVersion.java
+++ b/router/java/src/net/i2p/router/RouterVersion.java
@@ -17,7 +17,7 @@ import net.i2p.CoreVersion;
 public class RouterVersion {
     public final static String ID = "$Revision: 1.548 $ $Date: 2008-06-07 23:00:00 $";
     public final static String VERSION = CoreVersion.VERSION;
-    public final static long BUILD = 7;
+    public final static long BUILD = 8;
     public static void main(String args[]) {
         System.out.println("I2P Router version: " + VERSION + "-" + BUILD);
         System.out.println("Router ID: " + RouterVersion.ID);