diff --git a/apps/addressbook/java/src/net/i2p/addressbook/AddressBook.java b/apps/addressbook/java/src/net/i2p/addressbook/AddressBook.java
index d1dc34034ec319c973a919867125b6ccd9a02d16..faa9979d652def125cf8b94a3bddc4acae066a41 100644
--- a/apps/addressbook/java/src/net/i2p/addressbook/AddressBook.java
+++ b/apps/addressbook/java/src/net/i2p/addressbook/AddressBook.java
@@ -177,6 +177,18 @@ class AddressBook implements Iterable<Map.Entry<String, HostTxtEntry>> {
         this.subFile = null;
     }
 
+    /**
+     * Test only.
+     * 
+     * @param testsubfile path to a file containing the simulated fetch of a subscription
+     * @since 0.9.26
+     */
+    public AddressBook(String testsubfile) {
+        this.location = testsubfile;
+        this.addresses = null;
+        this.subFile = new File(testsubfile);
+    }
+
     /**
      * Return an iterator over the addresses in the AddressBook.
      * @since 0.8.7
diff --git a/apps/addressbook/java/src/net/i2p/addressbook/Daemon.java b/apps/addressbook/java/src/net/i2p/addressbook/Daemon.java
index 297e6c0e1f73e841ac44571f53feae4a4493220c..243a61f2e8f4b8c7a3415680b02d0d53130cfcf4 100644
--- a/apps/addressbook/java/src/net/i2p/addressbook/Daemon.java
+++ b/apps/addressbook/java/src/net/i2p/addressbook/Daemon.java
@@ -737,7 +737,24 @@ public class Daemon {
      *            others are ignored.
      */
     public static void main(String[] args) {
-        _instance.run(args);
+        if (args != null && args.length > 0 && args[0].equals("test"))
+            _instance.test(args);
+        else
+            _instance.run(args);
+    }
+
+    /** @since 0.9.26 */
+    private static void test(String[] args) {
+        Properties ctxProps = new Properties();
+        String PROP_FORCE = "i2p.naming.blockfile.writeInAppContext";
+        ctxProps.setProperty(PROP_FORCE, "true");
+        I2PAppContext ctx = new I2PAppContext(ctxProps);
+        NamingService ns = getNamingService("hosts.txt");
+        File published = new File("test-published.txt");
+        Log log = new Log(new File("test-log.txt"));
+        SubscriptionList subscriptions = new SubscriptionList("test-sub.txt");
+        update(ns, published, subscriptions, log);
+        ctx.logManager().flush();
     }
     
     public void run(String[] args) {
diff --git a/apps/addressbook/java/src/net/i2p/addressbook/HostTxtEntry.java b/apps/addressbook/java/src/net/i2p/addressbook/HostTxtEntry.java
index 9f0e0a37d6e5a471f6b5ac66321fb8655690475a..0852c52884d65f46151919925b285dfd47520983 100644
--- a/apps/addressbook/java/src/net/i2p/addressbook/HostTxtEntry.java
+++ b/apps/addressbook/java/src/net/i2p/addressbook/HostTxtEntry.java
@@ -71,9 +71,16 @@ class HostTxtEntry {
      * @throws IllegalArgumentException on dup key in sprops and other errors
      */
     public HostTxtEntry(String name, String dest, String sprops) throws IllegalArgumentException {
-        this.name = name;
-        this.dest = dest;
-        this.props = parseProps(sprops);
+        this(name, dest, parseProps(sprops));
+    }
+
+    /**
+     * A 'remove' entry. Name and Dest will be null.
+     * @param sprops line part after the #!, non-null
+     * @throws IllegalArgumentException on dup key in sprops and other errors
+     */
+    public HostTxtEntry(String sprops) throws IllegalArgumentException {
+        this(null, null, parseProps(sprops));
     }
 
     /**
@@ -132,19 +139,19 @@ class HostTxtEntry {
     }
 
     /**
-     * Write as a "remove" line #!olddest=dest#oldname=name#k1=v1#k2=v2...]
+     * Write as a "remove" line #!dest=dest#name=name#k1=v1#sig=sig...]
      * Includes newline.
      * Must have been constructed with non-null properties.
      */
     public void writeRemove(BufferedWriter out) throws IOException {
         if (props == null)
             throw new IllegalStateException();
-        props.setProperty(PROP_OLDNAME, name);
-        props.setProperty(PROP_OLDDEST, dest);
+        props.setProperty(PROP_NAME, name);
+        props.setProperty(PROP_DEST, dest);
         writeProps(out, false, false);
         out.newLine();
-        props.remove(PROP_OLDNAME);
-        props.remove(PROP_OLDDEST);
+        props.remove(PROP_NAME);
+        props.remove(PROP_DEST);
     }
 
     /**
@@ -265,6 +272,50 @@ class HostTxtEntry {
         return rv;
     }
 
+    /**
+     * Verify with the "dest" property's public key using the "sig" property
+     */
+    public boolean hasValidRemoveSig() {
+        if (props == null)
+            return false;
+        boolean rv = false;
+        // don't cache result
+        if (true) {
+            StringWriter buf = new StringWriter(1024);
+            String sig = props.getProperty(PROP_SIG);
+            String olddest = props.getProperty(PROP_DEST);
+            if (sig == null || olddest == null)
+                return false;
+            try {
+                writeProps(buf, true, true);
+            } catch (IOException ioe) {
+                // won't happen
+                return false;
+            }
+            byte[] sdata = Base64.decode(sig);
+            if (sdata == null)
+                return false;
+            Destination d;
+            try {
+                d = new Destination(olddest);
+            } catch (DataFormatException dfe) {
+                return false;
+            }
+            SigningPublicKey spk = d.getSigningPublicKey();
+            SigType type = spk.getType();
+            if (type == null)
+                return false;
+            Signature s;
+            try {
+                s = new Signature(type, sdata);
+            } catch (IllegalArgumentException iae) {
+                return false;
+            }
+            rv = DSAEngine.getInstance().verifySignature(s, DataHelper.getUTF8(buf.toString()), spk);
+        }
+        return rv;
+    }
+
     @Override
     public int hashCode() {
         return dest.hashCode();
@@ -299,6 +350,30 @@ class HostTxtEntry {
         signIt(spk, PROP_OLDSIG);
     }
 
+    /**
+     * Sign as a "remove" line #!dest=dest#name=name#k1=v1#sig=sig...]
+     */
+    public void signRemove(SigningPrivateKey spk) {
+        if (props == null)
+            throw new IllegalStateException();
+        if (props.containsKey(PROP_SIG))
+            throw new IllegalStateException();
+        props.setProperty(PROP_NAME, name);
+        props.setProperty(PROP_DEST, dest);
+        StringWriter buf = new StringWriter(1024);
+        try {
+            writeProps(buf, false, false);
+        } catch (IOException ioe) {
+            throw new IllegalStateException(ioe);
+        }
+        props.remove(PROP_NAME);
+        props.remove(PROP_DEST);
+        Signature s = DSAEngine.getInstance().sign(DataHelper.getUTF8(buf.toString()), spk);
+        if (s == null)
+            throw new IllegalArgumentException("sig failed");
+        props.setProperty(PROP_SIG, s.toBase64());
+    }
+
     /**
      * for testing only
      * @param sigprop The signature property to set
@@ -384,6 +459,22 @@ class HostTxtEntry {
             throw new IllegalStateException("Inner fail 2");
         if (!he2.hasValidSig())
             throw new IllegalStateException("Outer fail 2");
+
+        // 'remove' tests (corrupts earlier sigs)
+        he.getProps().remove(PROP_SIG);
+        he.signRemove(priv);
+        //out.write("Remove entry:\n");
+        sw = new StringWriter(1024);
+        buf = new BufferedWriter(sw);
+        he.writeRemove(buf);
+        buf.flush();
+        out.write(sw.toString());
+        out.flush();
+        line = sw.toString().substring(2).trim();
+        HostTxtEntry he3 = new HostTxtEntry(line);
+        if (!he3.hasValidRemoveSig())
+            throw new IllegalStateException("Remove verify fail");
+
         //out.write("Test passed\n");
         //out.flush();
     }
diff --git a/apps/addressbook/java/src/net/i2p/addressbook/SubscriptionIterator.java b/apps/addressbook/java/src/net/i2p/addressbook/SubscriptionIterator.java
index 0472d2f266e4f6936f65cdcedf45f1f7bbaae859..5a880f2b14d3d8df8e8f215fe6dc213d5fd74add 100644
--- a/apps/addressbook/java/src/net/i2p/addressbook/SubscriptionIterator.java
+++ b/apps/addressbook/java/src/net/i2p/addressbook/SubscriptionIterator.java
@@ -75,7 +75,10 @@ class SubscriptionIterator implements Iterator<AddressBook> {
      */
     public AddressBook next() {
         Subscription sub = this.subIterator.next();
-        if (sub.getLastFetched() + this.delay < I2PAppContext.getGlobalContext().clock().now() &&
+        if (sub.getLocation().startsWith("file:")) {
+            // test only
+            return new AddressBook(sub.getLocation().substring(5));
+        } else if (sub.getLastFetched() + this.delay < I2PAppContext.getGlobalContext().clock().now() &&
             I2PAppContext.getGlobalContext().portMapper().getPort(PortMapper.SVC_HTTP_PROXY) >= 0 &&
             !I2PAppContext.getGlobalContext().getBooleanProperty("i2p.vmCommSystem")) {
             //System.err.println("Fetching addressbook from " + sub.getLocation());
diff --git a/apps/addressbook/java/src/net/i2p/addressbook/SubscriptionList.java b/apps/addressbook/java/src/net/i2p/addressbook/SubscriptionList.java
index 1bca24140304527c468ed5d22fc4b65ac2345b8e..d28637b405011771a014cc408e24ab509d32c1df 100644
--- a/apps/addressbook/java/src/net/i2p/addressbook/SubscriptionList.java
+++ b/apps/addressbook/java/src/net/i2p/addressbook/SubscriptionList.java
@@ -100,6 +100,24 @@ class SubscriptionList implements Iterable<AddressBook> {
         }
     }
     
+    /**
+     * Testing only.
+     * 
+     * @param hoststxt path to a local file used as the test 'subscription' input
+     * @since 0.9.26
+     */
+    public SubscriptionList(String hoststxt) {
+        File dummy = new File("/dev/null");
+        this.etagsFile = dummy;
+        this.lastModifiedFile = dummy;
+        this.lastFetchedFile = dummy;
+        this.delay = 0;
+        this.proxyHost = "127.0.0.1";
+        this.proxyPort = 4444;
+        Subscription sub = new Subscription("file:" + hoststxt, null, null, null);
+        this.subscriptions = Collections.singletonList(sub);
+    }
+    
     /**
      * Return an iterator over the AddressBooks represented by the Subscriptions
      * in this SubscriptionList.