diff --git a/apps/routerconsole/java/src/net/i2p/router/web/CSSHelper.java b/apps/routerconsole/java/src/net/i2p/router/web/CSSHelper.java
index cc5b415c32a0d197d7abf32b49cf4a1b271f011a..dc95bf6d17d12939ae88a97e94584446044a3a6d 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/CSSHelper.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/CSSHelper.java
@@ -38,8 +38,7 @@ public class CSSHelper extends HelperBase {
     public void setLang(String lang) {
         // Protected with nonce in css.jsi
         if (lang != null && lang.length() == 2 && !lang.equals(_context.getProperty(Messages.PROP_LANG))) {
-            _context.router().setConfigSetting(Messages.PROP_LANG, lang);
-            _context.router().saveConfig();
+            _context.router().saveConfig(Messages.PROP_LANG, lang);
         }
     }
 
@@ -61,8 +60,7 @@ public class CSSHelper extends HelperBase {
 
     /** change refresh and save it */
     public void setRefresh(String r) {
-        _context.router().setConfigSetting(PROP_REFRESH, r);
-        _context.router().saveConfig();
+        _context.router().saveConfig(PROP_REFRESH, r);
     }
 
     /** @return refresh time in seconds, as a string */
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/ConfigAdvancedHandler.java b/apps/routerconsole/java/src/net/i2p/router/web/ConfigAdvancedHandler.java
index 6b13e0d1107606074df634b6b47be5b280f0a9bb..3844881e3f1c4dc2ac9644ceb6da6a8d6cbdea01 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/ConfigAdvancedHandler.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/ConfigAdvancedHandler.java
@@ -53,18 +53,11 @@ public class ConfigAdvancedHandler extends FormHandler {
                 return;
             }
 
-            for (Map.Entry e : props.entrySet()) {
-                String key = (String) e.getKey();
-                String val = (String) e.getValue();
-                _context.router().setConfigSetting(key, val);
+            for (Object key : props.keySet()) {
                 unsetKeys.remove(key);
             }
 
-            for (String unsetKey : unsetKeys) {
-                _context.router().removeConfigSetting(unsetKey);
-            }
-
-            boolean saved = _context.router().saveConfig();
+            boolean saved = _context.router().saveConfig(props, unsetKeys);
             if (saved) 
                 addFormNotice(_("Configuration saved successfully"));
             else
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/ConfigNetHandler.java b/apps/routerconsole/java/src/net/i2p/router/web/ConfigNetHandler.java
index 8f9a20749c81d019764220e712a477da90fed9bf..cf1caf99b0b9171c28d06d680e12566cced25968 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/ConfigNetHandler.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/ConfigNetHandler.java
@@ -2,6 +2,10 @@ package net.i2p.router.web;
 
 import java.net.InetAddress;
 import java.net.UnknownHostException;
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
 
 import net.i2p.router.Router;
 import net.i2p.router.transport.FIFOBandwidthRefiller;
@@ -45,6 +49,7 @@ public class ConfigNetHandler extends FormHandler {
     private boolean _enableLoadTesting;
     private String _sharePct;
     private boolean _ratesOnly;
+    private final Map<String, String> changes = new HashMap();
     private static final String PROP_HIDDEN = Router.PROP_HIDDEN_HIDDEN; // see Router for other choice
     
     @Override
@@ -137,6 +142,7 @@ public class ConfigNetHandler extends FormHandler {
      */
     private void saveChanges() {
         boolean restartRequired = false;
+        List<String> removes = new ArrayList();
         
         if (!_ratesOnly) {
             // IP Settings
@@ -153,15 +159,15 @@ public class ConfigNetHandler extends FormHandler {
                     else
                         _udpAutoIP = UDPTransport.DEFAULT_SOURCES;
                 }
-                _context.router().setConfigSetting(UDPTransport.PROP_SOURCES, _udpAutoIP);
+                changes.put(UDPTransport.PROP_SOURCES, _udpAutoIP);
                 boolean valid = true;
                 if (uhost.length() > 0) {
                     valid = verifyAddress(uhost);
                     if (valid) {
-                        _context.router().setConfigSetting(UDPTransport.PROP_EXTERNAL_HOST, uhost);
+                        changes.put(UDPTransport.PROP_EXTERNAL_HOST, uhost);
                     }
                 } else {
-                    _context.router().removeConfigSetting(UDPTransport.PROP_EXTERNAL_HOST);
+                    removes.add(UDPTransport.PROP_EXTERNAL_HOST);
                 }
                 if (valid && ((!oldUdp.equals(_udpAutoIP)) || (!oldUHost.equals(uhost)))) {
                    addFormNotice(_("Updating IP address"));
@@ -187,31 +193,31 @@ public class ConfigNetHandler extends FormHandler {
                 } else if ("false".equals(_ntcpAutoIP) && _ntcpHostname.length() > 0) {
                     valid = verifyAddress(_ntcpHostname);
                     if (valid) {
-                        _context.router().setConfigSetting(ConfigNetHelper.PROP_I2NP_NTCP_HOSTNAME, _ntcpHostname);
+                        changes.put(ConfigNetHelper.PROP_I2NP_NTCP_HOSTNAME, _ntcpHostname);
                         addFormNotice(_("Updating inbound TCP address to") + " " + _ntcpHostname);
                     }
                 } else {
-                    _context.router().removeConfigSetting(ConfigNetHelper.PROP_I2NP_NTCP_HOSTNAME);
+                    removes.add(ConfigNetHelper.PROP_I2NP_NTCP_HOSTNAME);
                     if ("false".equals(_ntcpAutoIP))
                         addFormNotice(_("Disabling inbound TCP"));
                     else
                         addFormNotice(_("Updating inbound TCP address to auto")); // true or always
                 }
                 if (valid) {
-                    _context.router().setConfigSetting(ConfigNetHelper.PROP_I2NP_NTCP_AUTO_IP, _ntcpAutoIP);
-                    _context.router().setConfigSetting(TransportManager.PROP_ENABLE_NTCP, "" + !"disabled".equals(_ntcpAutoIP));
+                    changes.put(ConfigNetHelper.PROP_I2NP_NTCP_AUTO_IP, _ntcpAutoIP);
+                    changes.put(TransportManager.PROP_ENABLE_NTCP, "" + !"disabled".equals(_ntcpAutoIP));
                     restartRequired = true;
                 }
             }
             if (oldAutoPort != _ntcpAutoPort || ! oldNPort.equals(_ntcpPort)) {
                 if (_ntcpPort.length() > 0 && !_ntcpAutoPort) {
-                    _context.router().setConfigSetting(ConfigNetHelper.PROP_I2NP_NTCP_PORT, _ntcpPort);
+                    changes.put(ConfigNetHelper.PROP_I2NP_NTCP_PORT, _ntcpPort);
                     addFormNotice(_("Updating inbound TCP port to") + " " + _ntcpPort);
                 } else {
-                    _context.router().removeConfigSetting(ConfigNetHelper.PROP_I2NP_NTCP_PORT);
+                    removes.add(ConfigNetHelper.PROP_I2NP_NTCP_PORT);
                     addFormNotice(_("Updating inbound TCP port to auto"));
                 }
-                _context.router().setConfigSetting(ConfigNetHelper.PROP_I2NP_NTCP_AUTO_PORT, "" + _ntcpAutoPort);
+                changes.put(ConfigNetHelper.PROP_I2NP_NTCP_AUTO_PORT, "" + _ntcpAutoPort);
                 restartRequired = true;
             }
 
@@ -219,8 +225,8 @@ public class ConfigNetHandler extends FormHandler {
             if ( (_udpPort != null) && (_udpPort.length() > 0) ) {
                 String oldPort = "" + _context.getProperty(UDPTransport.PROP_INTERNAL_PORT, UDPTransport.DEFAULT_INTERNAL_PORT);
                 if (!oldPort.equals(_udpPort)) {
-                    _context.router().setConfigSetting(UDPTransport.PROP_INTERNAL_PORT, _udpPort);
-                    _context.router().setConfigSetting(UDPTransport.PROP_EXTERNAL_PORT, _udpPort);
+                    changes.put(UDPTransport.PROP_INTERNAL_PORT, _udpPort);
+                    changes.put(UDPTransport.PROP_EXTERNAL_PORT, _udpPort);
                     addFormNotice(_("Updating UDP port from") + " " + oldPort + " " + _("to") + " " + _udpPort);
                     restartRequired = true;
                 }
@@ -228,21 +234,21 @@ public class ConfigNetHandler extends FormHandler {
 
         }
         
-        updateRates();
+        boolean ratesUpdated = updateRates();
         
         boolean switchRequired = false;
         if (!_ratesOnly) {
             // If hidden mode value changes, restart is required
             switchRequired = _hiddenMode != _context.router().isHidden();
             if (switchRequired) {
-                _context.router().setConfigSetting(PROP_HIDDEN, "" + _hiddenMode);
+                changes.put(PROP_HIDDEN, "" + _hiddenMode);
                 if (_hiddenMode)
                     addFormError(_("Gracefully restarting into Hidden Router Mode"));
                 else
                     addFormError(_("Gracefully restarting to exit Hidden Router Mode"));
             }
 
-            _context.router().setConfigSetting(Router.PROP_DYNAMIC_KEYS, "" + _dynamicKeys);
+            changes.put(Router.PROP_DYNAMIC_KEYS, "" + _dynamicKeys);
 
             if (Boolean.valueOf(_context.getProperty(TransportManager.PROP_ENABLE_UPNP)).booleanValue() !=
                 _upnp) {
@@ -252,7 +258,7 @@ public class ConfigNetHandler extends FormHandler {
                 else
                     addFormNotice(_("Disabling UPnP, restart required to take effect"));
             }
-            _context.router().setConfigSetting(TransportManager.PROP_ENABLE_UPNP, "" + _upnp);
+            changes.put(TransportManager.PROP_ENABLE_UPNP, "" + _upnp);
 
             if (Boolean.valueOf(_context.getProperty(UDPTransport.PROP_LAPTOP_MODE)).booleanValue() !=
                 _laptop) {
@@ -262,28 +268,32 @@ public class ConfigNetHandler extends FormHandler {
                 else
                     addFormNotice(_("Disabling laptop mode"));
             }
-            _context.router().setConfigSetting(UDPTransport.PROP_LAPTOP_MODE, "" + _laptop);
+            changes.put(UDPTransport.PROP_LAPTOP_MODE, "" + _laptop);
 
             if (_requireIntroductions) {
-                _context.router().setConfigSetting(UDPTransport.PROP_FORCE_INTRODUCERS, "true");
+                changes.put(UDPTransport.PROP_FORCE_INTRODUCERS, "true");
                 addFormNotice(_("Requiring SSU introducers"));
             } else {
-                _context.router().removeConfigSetting(UDPTransport.PROP_FORCE_INTRODUCERS);
+                removes.add(UDPTransport.PROP_FORCE_INTRODUCERS);
             }
 
             // Time sync enable, means NOT disabled 
             // Hmm router sets this at startup, not required here
-            //_context.router().setConfigSetting(Timestamper.PROP_DISABLED, "false");
+            //changes.put(Timestamper.PROP_DISABLED, "false");
             
             // Hidden in the GUI
             //LoadTestManager.setEnableLoadTesting(_context, _enableLoadTesting);
         }
         
-        boolean saved = _context.router().saveConfig();
+        boolean saved = _context.router().saveConfig(changes, removes);
         if (saved) 
             addFormNotice(_("Configuration saved successfully"));
         else
             addFormError(_("Error saving the configuration (applied but not saved) - please see the error logs"));
+
+        // this has to be after the save
+        if (ratesUpdated)
+            _context.bandwidthLimiter().reinitialize();
         
         if (switchRequired) {
             hiddenSwitch();
@@ -343,14 +353,17 @@ public class ConfigNetHandler extends FormHandler {
     private static final int DEF_BURST_PCT = 10;
     private static final int DEF_BURST_TIME = 20;
 
-    private void updateRates() {
+    /**
+     *  @return changed
+     */
+    private boolean updateRates() {
         boolean updated = false;
         boolean bwUpdated = false;
 
         if (_sharePct != null) {
             String old = _context.router().getConfigSetting(Router.PROP_BANDWIDTH_SHARE_PERCENTAGE);
             if ( (old == null) || (!old.equals(_sharePct)) ) {
-                _context.router().setConfigSetting(Router.PROP_BANDWIDTH_SHARE_PERCENTAGE, _sharePct);
+                changes.put(Router.PROP_BANDWIDTH_SHARE_PERCENTAGE, _sharePct);
                 addFormNotice(_("Updating bandwidth share percentage"));
                 updated = true;
             }
@@ -359,23 +372,23 @@ public class ConfigNetHandler extends FormHandler {
         // Since burst is now hidden in the gui, set burst to +10% for 20 seconds
         if ( (_inboundRate != null) && (_inboundRate.length() > 0) &&
             !_inboundRate.equals(_context.getProperty(FIFOBandwidthRefiller.PROP_INBOUND_BANDWIDTH, "" + FIFOBandwidthRefiller.DEFAULT_INBOUND_BANDWIDTH))) {
-            _context.router().setConfigSetting(FIFOBandwidthRefiller.PROP_INBOUND_BANDWIDTH, _inboundRate);
+            changes.put(FIFOBandwidthRefiller.PROP_INBOUND_BANDWIDTH, _inboundRate);
             try {
                 int rate = Integer.parseInt(_inboundRate) * (100 + DEF_BURST_PCT) / 100;
                 int kb = DEF_BURST_TIME * rate;
-                _context.router().setConfigSetting(FIFOBandwidthRefiller.PROP_INBOUND_BURST_BANDWIDTH, "" + rate);
-                _context.router().setConfigSetting(FIFOBandwidthRefiller.PROP_INBOUND_BANDWIDTH_PEAK, "" + kb);
+                changes.put(FIFOBandwidthRefiller.PROP_INBOUND_BURST_BANDWIDTH, "" + rate);
+                changes.put(FIFOBandwidthRefiller.PROP_INBOUND_BANDWIDTH_PEAK, "" + kb);
             } catch (NumberFormatException nfe) {}
             bwUpdated = true;
         }
         if ( (_outboundRate != null) && (_outboundRate.length() > 0) &&
             !_outboundRate.equals(_context.getProperty(FIFOBandwidthRefiller.PROP_OUTBOUND_BANDWIDTH, "" + FIFOBandwidthRefiller.DEFAULT_OUTBOUND_BANDWIDTH))) {
-            _context.router().setConfigSetting(FIFOBandwidthRefiller.PROP_OUTBOUND_BANDWIDTH, _outboundRate);
+            changes.put(FIFOBandwidthRefiller.PROP_OUTBOUND_BANDWIDTH, _outboundRate);
             try {
                 int rate = Integer.parseInt(_outboundRate) * (100 + DEF_BURST_PCT) / 100;
                 int kb = DEF_BURST_TIME * rate;
-                _context.router().setConfigSetting(FIFOBandwidthRefiller.PROP_OUTBOUND_BURST_BANDWIDTH, "" + rate);
-                _context.router().setConfigSetting(FIFOBandwidthRefiller.PROP_OUTBOUND_BANDWIDTH_PEAK, "" + kb);
+                changes.put(FIFOBandwidthRefiller.PROP_OUTBOUND_BURST_BANDWIDTH, "" + rate);
+                changes.put(FIFOBandwidthRefiller.PROP_OUTBOUND_BANDWIDTH_PEAK, "" + kb);
             } catch (NumberFormatException nfe) {}
             bwUpdated = true;
         }
@@ -389,12 +402,12 @@ public class ConfigNetHandler extends FormHandler {
 
         if ( (_inboundBurstRate != null) && (_inboundBurstRate.length() > 0) &&
             !_inboundBurstRate.equals(_context.getProperty(FIFOBandwidthRefiller.PROP_INBOUND_BURST_BANDWIDTH, "" + FIFOBandwidthRefiller.DEFAULT_INBOUND_BURST_BANDWIDTH))) {
-            _context.router().setConfigSetting(FIFOBandwidthRefiller.PROP_INBOUND_BURST_BANDWIDTH, _inboundBurstRate);
+            changes.put(FIFOBandwidthRefiller.PROP_INBOUND_BURST_BANDWIDTH, _inboundBurstRate);
             updated = true;
         }
         if ( (_outboundBurstRate != null) && (_outboundBurstRate.length() > 0) &&
             !_outboundBurstRate.equals(_context.getProperty(FIFOBandwidthRefiller.PROP_OUTBOUND_BURST_BANDWIDTH, "" + FIFOBandwidthRefiller.DEFAULT_OUTBOUND_BURST_BANDWIDTH))) {
-            _context.router().setConfigSetting(FIFOBandwidthRefiller.PROP_OUTBOUND_BURST_BANDWIDTH, _outboundBurstRate);
+            changes.put(FIFOBandwidthRefiller.PROP_OUTBOUND_BURST_BANDWIDTH, _outboundBurstRate);
             updated = true;
         }
         
@@ -411,7 +424,7 @@ public class ConfigNetHandler extends FormHandler {
             }
             if ( (rateKBps > 0) && (burstSeconds > 0) ) {
                 int kb = rateKBps * burstSeconds;
-                _context.router().setConfigSetting(FIFOBandwidthRefiller.PROP_INBOUND_BANDWIDTH_PEAK, "" + kb);
+                changes.put(FIFOBandwidthRefiller.PROP_INBOUND_BANDWIDTH_PEAK, "" + kb);
                 updated = true;
             }
         }
@@ -429,15 +442,13 @@ public class ConfigNetHandler extends FormHandler {
             }
             if ( (rateKBps > 0) && (burstSeconds > 0) ) {
                 int kb = rateKBps * burstSeconds;
-                _context.router().setConfigSetting(FIFOBandwidthRefiller.PROP_OUTBOUND_BANDWIDTH_PEAK, "" + kb);
+                changes.put(FIFOBandwidthRefiller.PROP_OUTBOUND_BANDWIDTH_PEAK, "" + kb);
                 updated = true;
             }
         }
 
 ***********/
 
-        
-        if (updated)
-            _context.bandwidthLimiter().reinitialize();
+        return updated; 
     }
 }
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/ConfigReseedHandler.java b/apps/routerconsole/java/src/net/i2p/router/web/ConfigReseedHandler.java
index 7a1c84b9541554b3c5a8b3733889eae1b7109eb5..9d28994f39e6f36d0dc07fc3a541c7d7354c1722 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/ConfigReseedHandler.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/ConfigReseedHandler.java
@@ -1,6 +1,8 @@
 package net.i2p.router.web;
 
+import java.util.ArrayList;
 import java.util.HashMap;
+import java.util.List;
 import java.util.Map;
 
 import net.i2p.router.networkdb.reseed.Reseeder;
@@ -10,6 +12,8 @@ import net.i2p.router.networkdb.reseed.Reseeder;
  */
 public class ConfigReseedHandler extends FormHandler {
     private Map _settings;
+    private final Map<String, String> changes = new HashMap();
+    private final List<String> removes = new ArrayList();
     
     @Override
     protected void processForm() {
@@ -47,15 +51,15 @@ public class ConfigReseedHandler extends FormHandler {
     private void saveString(String config, String param) {
         String val = getJettyString(param);
         if (val != null && val.length() > 0)
-            _context.router().setConfigSetting(config, val);
+            changes.put(config, val);
         else
-            _context.router().removeConfigSetting(config);
+            removes.add(config);
     }
 
     /** @since 0.8.9 */
     private void saveBoolean(String config, String param) {
         boolean val = getJettyString(param) != null;
-        _context.router().setConfigSetting(config, Boolean.toString(val));
+        changes.put(config, Boolean.toString(val));
     }
 
     private void saveChanges() {
@@ -71,17 +75,19 @@ public class ConfigReseedHandler extends FormHandler {
         saveBoolean(Reseeder.PROP_SPROXY_AUTH_ENABLE, "sauth");
         String url = getJettyString("reseedURL");
         if (url != null)
-            _context.router().setConfigSetting(Reseeder.PROP_RESEED_URL, url.trim().replace("\r\n", ",").replace("\n", ","));
+            changes.put(Reseeder.PROP_RESEED_URL, url.trim().replace("\r\n", ",").replace("\n", ","));
         String mode = getJettyString("mode");
         boolean req = "1".equals(mode);
         boolean disabled = "2".equals(mode);
-        _context.router().setConfigSetting(Reseeder.PROP_SSL_REQUIRED,
+        changes.put(Reseeder.PROP_SSL_REQUIRED,
                                            Boolean.toString(req));
-        _context.router().setConfigSetting(Reseeder.PROP_SSL_DISABLE,
+        changes.put(Reseeder.PROP_SSL_DISABLE,
                                            Boolean.toString(disabled));
         saveBoolean(Reseeder.PROP_PROXY_ENABLE, "enable");
         saveBoolean(Reseeder.PROP_SPROXY_ENABLE, "senable");
-        _context.router().saveConfig();
-        addFormNotice(_("Configuration saved successfully."));
+        if (_context.router().saveConfig(changes, removes))
+            addFormNotice(_("Configuration saved successfully."));
+        else
+            addFormError(_("Error saving the configuration (applied but not saved) - please see the error logs"));
     }
 }
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/ConfigStatsHandler.java b/apps/routerconsole/java/src/net/i2p/router/web/ConfigStatsHandler.java
index 8d21fd63efb5f80101264388410bce70e24be67a..07f220f7a096268399173627aeb66170b7de1c7f 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/ConfigStatsHandler.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/ConfigStatsHandler.java
@@ -1,7 +1,9 @@
 package net.i2p.router.web;
 
 import java.util.ArrayList;
+import java.util.HashMap;
 import java.util.List;
+import java.util.Map;
 import java.util.StringTokenizer;
 
 import net.i2p.stat.StatManager;
@@ -73,9 +75,10 @@ public class ConfigStatsHandler extends FormHandler {
      *
      */
     private void saveChanges() {
+        Map<String, String> changes = new HashMap();
         if (_filename == null)
             _filename = StatManager.DEFAULT_STAT_FILE;
-        _context.router().setConfigSetting(StatManager.PROP_STAT_FILE, _filename);
+        changes.put(StatManager.PROP_STAT_FILE, _filename);
         
         if (_explicitFilter) {
             _stats.clear();
@@ -103,12 +106,12 @@ public class ConfigStatsHandler extends FormHandler {
                 stats.append(',');
         }
             
-        _context.router().setConfigSetting(StatManager.PROP_STAT_FILTER, stats.toString());
+        changes.put(StatManager.PROP_STAT_FILTER, stats.toString());
         boolean graphsChanged = !_graphs.equals(_context.getProperty("stat.summaries"));
-        _context.router().setConfigSetting("stat.summaries", _graphs);
+        changes.put("stat.summaries", _graphs);
         boolean fullChanged = _context.getBooleanProperty(StatManager.PROP_STAT_FULL) != _isFull;
-        _context.router().setConfigSetting(StatManager.PROP_STAT_FULL, "" + _isFull);
-        _context.router().saveConfig();
+        changes.put(StatManager.PROP_STAT_FULL, "" + _isFull);
+        _context.router().saveConfig(changes, null);
         if (!_stats.isEmpty())
             addFormNotice(_("Stat filter and location updated successfully to") + ": " + stats.toString());
         if (fullChanged) {
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/ConfigTunnelsHandler.java b/apps/routerconsole/java/src/net/i2p/router/web/ConfigTunnelsHandler.java
index fa2737c8e48cfe5561fe5817f40c3ad81703404c..0c0c502829c5ba65dd7bbb3e6eb8f341d1bb86ed 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/ConfigTunnelsHandler.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/ConfigTunnelsHandler.java
@@ -44,6 +44,7 @@ public class ConfigTunnelsHandler extends FormHandler {
      */
     private void saveChanges() {
         boolean saveRequired = false;
+        Map<String, String> changes = new HashMap();
         
         if (_log.shouldLog(Log.DEBUG))
             _log.debug("Saving changes, with props = " + _settings + ".");
@@ -90,21 +91,21 @@ public class ConfigTunnelsHandler extends FormHandler {
             out.setBackupQuantity(getInt(_settings.get(index + ".backupOutbound")));
             
             if ("exploratory".equals(poolName)) {
-                _context.router().setConfigSetting(TunnelPoolSettings.PREFIX_INBOUND_EXPLORATORY + 
+                changes.put(TunnelPoolSettings.PREFIX_INBOUND_EXPLORATORY + 
                                                    TunnelPoolSettings.PROP_LENGTH, in.getLength()+"");
-                _context.router().setConfigSetting(TunnelPoolSettings.PREFIX_OUTBOUND_EXPLORATORY + 
+                changes.put(TunnelPoolSettings.PREFIX_OUTBOUND_EXPLORATORY + 
                                                    TunnelPoolSettings.PROP_LENGTH, out.getLength()+"");
-                _context.router().setConfigSetting(TunnelPoolSettings.PREFIX_INBOUND_EXPLORATORY + 
+                changes.put(TunnelPoolSettings.PREFIX_INBOUND_EXPLORATORY + 
                                                    TunnelPoolSettings.PROP_LENGTH_VARIANCE, in.getLengthVariance()+"");
-                _context.router().setConfigSetting(TunnelPoolSettings.PREFIX_OUTBOUND_EXPLORATORY + 
+                changes.put(TunnelPoolSettings.PREFIX_OUTBOUND_EXPLORATORY + 
                                                    TunnelPoolSettings.PROP_LENGTH_VARIANCE, out.getLengthVariance()+"");
-                _context.router().setConfigSetting(TunnelPoolSettings.PREFIX_INBOUND_EXPLORATORY + 
+                changes.put(TunnelPoolSettings.PREFIX_INBOUND_EXPLORATORY + 
                                                    TunnelPoolSettings.PROP_QUANTITY, in.getQuantity()+"");
-                _context.router().setConfigSetting(TunnelPoolSettings.PREFIX_OUTBOUND_EXPLORATORY + 
+                changes.put(TunnelPoolSettings.PREFIX_OUTBOUND_EXPLORATORY + 
                                                    TunnelPoolSettings.PROP_QUANTITY, out.getQuantity()+"");
-                _context.router().setConfigSetting(TunnelPoolSettings.PREFIX_INBOUND_EXPLORATORY + 
+                changes.put(TunnelPoolSettings.PREFIX_INBOUND_EXPLORATORY + 
                                                    TunnelPoolSettings.PROP_BACKUP_QUANTITY, in.getBackupQuantity()+"");
-                _context.router().setConfigSetting(TunnelPoolSettings.PREFIX_OUTBOUND_EXPLORATORY + 
+                changes.put(TunnelPoolSettings.PREFIX_OUTBOUND_EXPLORATORY + 
                                                    TunnelPoolSettings.PROP_BACKUP_QUANTITY, out.getBackupQuantity()+"");
             }
             
@@ -135,7 +136,7 @@ public class ConfigTunnelsHandler extends FormHandler {
             addFormNotice(_("Updated settings for all pools."));
         
         if (saveRequired) {
-            boolean saved = _context.router().saveConfig();
+            boolean saved = _context.router().saveConfig(changes, null);
             if (saved) 
                 addFormNotice(_("Exploratory tunnel configuration saved successfully."));
             else
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/ConfigUIHandler.java b/apps/routerconsole/java/src/net/i2p/router/web/ConfigUIHandler.java
index 417fbaf244b3a3d3460ff96233eb674b69e507f3..514068ebfc5f2072f30ed5a1399b99d70c9cb340 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/ConfigUIHandler.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/ConfigUIHandler.java
@@ -22,11 +22,12 @@ public class ConfigUIHandler extends FormHandler {
         if (_config == null)
             return;
         String oldTheme = _context.getProperty(CSSHelper.PROP_THEME_NAME, CSSHelper.DEFAULT_THEME);
+        boolean ok;
         if (_config.equals("default")) // obsolete
-            _context.router().removeConfigSetting(CSSHelper.PROP_THEME_NAME);
+            ok = _context.router().saveConfig(CSSHelper.PROP_THEME_NAME, null);
         else
-            _context.router().setConfigSetting(CSSHelper.PROP_THEME_NAME, _config);
-        if (_context.router().saveConfig()) {
+            ok = _context.router().saveConfig(CSSHelper.PROP_THEME_NAME, _config);
+        if (ok) {
             if (!oldTheme.equals(_config))
                 addFormNotice(_("Theme change saved.") +
                               " <a href=\"configui\">" +
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/ConfigUpdateHandler.java b/apps/routerconsole/java/src/net/i2p/router/web/ConfigUpdateHandler.java
index 453c0ba3d15c02faba41f460af24e19bd3793fa6..a4d2d068e08abbf71d62985bdc65ff92cccb15b9 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/ConfigUpdateHandler.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/ConfigUpdateHandler.java
@@ -1,5 +1,8 @@
 package net.i2p.router.web;
 
+import java.util.HashMap;
+import java.util.Map;
+
 import net.i2p.crypto.TrustedUpdate;
 import net.i2p.data.DataHelper;
 import net.i2p.util.FileUtil;
@@ -98,10 +101,12 @@ public class ConfigUpdateHandler extends FormHandler {
             return;
         }
 
+        Map<String, String> changes = new HashMap();
+
         if ( (_newsURL != null) && (_newsURL.length() > 0) ) {
             String oldURL = ConfigUpdateHelper.getNewsURL(_context);
             if ( (oldURL == null) || (!_newsURL.equals(oldURL)) ) {
-                _context.router().setConfigSetting(PROP_NEWS_URL, _newsURL);
+                changes.put(PROP_NEWS_URL, _newsURL);
                 NewsFetcher.getInstance(_context).invalidateNews();
                 addFormNotice(_("Updating news URL to {0}", _newsURL));
             }
@@ -110,7 +115,7 @@ public class ConfigUpdateHandler extends FormHandler {
         if ( (_proxyHost != null) && (_proxyHost.length() > 0) ) {
             String oldHost = _context.router().getConfigSetting(PROP_PROXY_HOST);
             if ( (oldHost == null) || (!_proxyHost.equals(oldHost)) ) {
-                _context.router().setConfigSetting(PROP_PROXY_HOST, _proxyHost);
+                changes.put(PROP_PROXY_HOST, _proxyHost);
                 addFormNotice(_("Updating proxy host to {0}", _proxyHost));
             }
         }
@@ -118,19 +123,19 @@ public class ConfigUpdateHandler extends FormHandler {
         if ( (_proxyPort != null) && (_proxyPort.length() > 0) ) {
             String oldPort = _context.router().getConfigSetting(PROP_PROXY_PORT);
             if ( (oldPort == null) || (!_proxyPort.equals(oldPort)) ) {
-                _context.router().setConfigSetting(PROP_PROXY_PORT, _proxyPort);
+                changes.put(PROP_PROXY_PORT, _proxyPort);
                 addFormNotice(_("Updating proxy port to {0}", _proxyPort));
             }
         }
         
-        _context.router().setConfigSetting(PROP_SHOULD_PROXY, "" + _updateThroughProxy);
-        _context.router().setConfigSetting(PROP_UPDATE_UNSIGNED, "" + _updateUnsigned);
+        changes.put(PROP_SHOULD_PROXY, "" + _updateThroughProxy);
+        changes.put(PROP_UPDATE_UNSIGNED, "" + _updateUnsigned);
         
         String oldFreqStr = _context.getProperty(PROP_REFRESH_FREQUENCY, DEFAULT_REFRESH_FREQUENCY);
         long oldFreq = DEFAULT_REFRESH_FREQ;
         try { oldFreq = Long.parseLong(oldFreqStr); } catch (NumberFormatException nfe) {}
         if (_refreshFrequency != oldFreq) {
-            _context.router().setConfigSetting(PROP_REFRESH_FREQUENCY, ""+_refreshFrequency);
+            changes.put(PROP_REFRESH_FREQUENCY, ""+_refreshFrequency);
             addFormNotice(_("Updating refresh frequency to {0}",
                             _refreshFrequency <= 0 ? _("Never") : DataHelper.formatDuration2(_refreshFrequency)));
         }
@@ -138,7 +143,7 @@ public class ConfigUpdateHandler extends FormHandler {
         if ( (_updatePolicy != null) && (_updatePolicy.length() > 0) ) {
             String oldPolicy = _context.router().getConfigSetting(PROP_UPDATE_POLICY);
             if ( (oldPolicy == null) || (!_updatePolicy.equals(oldPolicy)) ) {
-                _context.router().setConfigSetting(PROP_UPDATE_POLICY, _updatePolicy);
+                changes.put(PROP_UPDATE_POLICY, _updatePolicy);
                 addFormNotice(_("Updating update policy to {0}", _updatePolicy));
             }
         }
@@ -147,7 +152,7 @@ public class ConfigUpdateHandler extends FormHandler {
             _updateURL = _updateURL.replace("\r\n", ",").replace("\n", ",");
             String oldURL = _context.router().getConfigSetting(PROP_UPDATE_URL);
             if ( (oldURL == null) || (!_updateURL.equals(oldURL)) ) {
-                _context.router().setConfigSetting(PROP_UPDATE_URL, _updateURL);
+                changes.put(PROP_UPDATE_URL, _updateURL);
                 addFormNotice(_("Updating update URLs."));
             }
         }
@@ -158,7 +163,7 @@ public class ConfigUpdateHandler extends FormHandler {
             oldKeys = oldKeys.replace("\r\n", ",");
             if (!_trustedKeys.equals(oldKeys)) {
                 // note that keys are not validated here and no console error message will be generated
-                _context.router().setConfigSetting(PROP_TRUSTED_KEYS, _trustedKeys);
+                changes.put(PROP_TRUSTED_KEYS, _trustedKeys);
                 addFormNotice(_("Updating trusted keys."));
             }
         }
@@ -166,12 +171,12 @@ public class ConfigUpdateHandler extends FormHandler {
         if ( (_zipURL != null) && (_zipURL.length() > 0) ) {
             String oldURL = _context.router().getConfigSetting(PROP_ZIP_URL);
             if ( (oldURL == null) || (!_zipURL.equals(oldURL)) ) {
-                _context.router().setConfigSetting(PROP_ZIP_URL, _zipURL);
+                changes.put(PROP_ZIP_URL, _zipURL);
                 addFormNotice(_("Updating unsigned update URL to {0}", _zipURL));
             }
         }
         
-        _context.router().saveConfig();
+        _context.router().saveConfig(changes, null);
     }
     
     public void setNewsURL(String url) { _newsURL = url; }
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 31b9433ac5c6e0273e15b1a9376a060bf40ebc05..b6a4ad283778f94332fbbf27ae5aea1cec96e8ac 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/GraphHelper.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/GraphHelper.java
@@ -3,8 +3,10 @@ package net.i2p.router.web;
 import java.io.IOException;
 import java.io.Writer;
 import java.util.Comparator;
+import java.util.HashMap;
 import java.util.Iterator;
 import java.util.List;
+import java.util.Map;
 import java.util.TreeSet;
 
 import net.i2p.data.DataHelper;
@@ -216,8 +218,7 @@ public class GraphHelper extends FormHandler {
                          System.getProperty("java.runtime.version") + ')');
             if (_context.getProperty(PROP_REFRESH, 0) >= 0) {
                 // force no refresh, save silently
-                _context.router().setConfigSetting(PROP_REFRESH, "-1");
-                _context.router().saveConfig();
+                _context.router().saveConfig(PROP_REFRESH, "-1");
             }
         }
         return super.getAllMessages();
@@ -243,13 +244,14 @@ public class GraphHelper extends FormHandler {
             _refreshDelaySeconds != _context.getProperty(PROP_REFRESH, DEFAULT_REFRESH) ||
             _showEvents != _context.getBooleanProperty(PROP_EVENTS) ||
             _persistent != _context.getBooleanPropertyDefaultTrue(SummaryListener.PROP_PERSISTENT)) {
-            _context.router().setConfigSetting(PROP_X, "" + _width);
-            _context.router().setConfigSetting(PROP_Y, "" + _height);
-            _context.router().setConfigSetting(PROP_PERIODS, "" + _periodCount);
-            _context.router().setConfigSetting(PROP_REFRESH, "" + _refreshDelaySeconds);
-            _context.router().setConfigSetting(PROP_EVENTS, "" + _showEvents);
-            _context.router().setConfigSetting(SummaryListener.PROP_PERSISTENT, "" + _persistent);
-            _context.router().saveConfig();
+            Map<String, String> changes = new HashMap();
+            changes.put(PROP_X, "" + _width);
+            changes.put(PROP_Y, "" + _height);
+            changes.put(PROP_PERIODS, "" + _periodCount);
+            changes.put(PROP_REFRESH, "" + _refreshDelaySeconds);
+            changes.put(PROP_EVENTS, "" + _showEvents);
+            changes.put(SummaryListener.PROP_PERSISTENT, "" + _persistent);
+            _context.router().saveConfig(changes, null);
             addFormNotice(_("Graph settings saved"));
         }
     }
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/NewsFetcher.java b/apps/routerconsole/java/src/net/i2p/router/web/NewsFetcher.java
index 253c36bd453f14d2c4f431c38a9d89f6d9292afa..6d7550dd26d0035da4bc1fdb134b9aacf8ac4288 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/NewsFetcher.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/NewsFetcher.java
@@ -119,8 +119,7 @@ public class NewsFetcher implements Runnable, EepGet.StatusListener {
      */
     public void showNews(boolean yes) {
         long stamp = yes ? 0 : _lastUpdated;
-        _context.router().setConfigSetting(PROP_LAST_HIDDEN, Long.toString(stamp));
-        _context.router().saveConfig();
+        _context.router().saveConfig(PROP_LAST_HIDDEN, Long.toString(stamp));
     }
 
     /**
@@ -283,9 +282,8 @@ public class NewsFetcher implements Runnable, EepGet.StatusListener {
                     if (lastUpdate == null) {
                         // we don't know what version you have, so stamp it with the current time,
                         // and we'll look for something newer next time around.
-                        _context.router().setConfigSetting(UpdateHandler.PROP_LAST_UPDATE_TIME,
+                        _context.router().saveConfig(UpdateHandler.PROP_LAST_UPDATE_TIME,
                                                            Long.toString(_context.clock().now()));
-                        _context.router().saveConfig();
                         return;
                     }
                     long ms = 0;
@@ -407,8 +405,7 @@ public class NewsFetcher implements Runnable, EepGet.StatusListener {
                 _log.warn("Transfer complete, but no file? - probably 304 Not Modified");
         }
         _lastFetch = now;
-        _context.router().setConfigSetting(PROP_LAST_CHECKED, Long.toString(now));
-        _context.router().saveConfig();
+        _context.router().saveConfig(PROP_LAST_CHECKED, Long.toString(now));
     }
     
     public void transferFailed(String url, long bytesTransferred, long bytesRemaining, int currentAttempt) {
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/PluginStarter.java b/apps/routerconsole/java/src/net/i2p/router/web/PluginStarter.java
index c4c567b3f9d7124a1a559dec2a34d0d1476544b3..7a3077ea5a2bca46c0d4231295781bf84ee1b4c1 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/PluginStarter.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/PluginStarter.java
@@ -416,15 +416,17 @@ public class PluginStarter implements Runnable {
         File[] tfiles = dir.listFiles();
         if (tfiles != null) {
             String current = ctx.getProperty(CSSHelper.PROP_THEME_NAME);
+            Map<String, String> changes = new HashMap();
+            List<String> removes = new ArrayList();
             for (int i = 0; i < tfiles.length; i++) {
                 String name = tfiles[i].getName();
                 if (tfiles[i].isDirectory() && (!Arrays.asList(STANDARD_THEMES).contains(tfiles[i]))) {
-                    ctx.router().removeConfigSetting(ConfigUIHelper.PROP_THEME_PFX + name);
+                    removes.add(ConfigUIHelper.PROP_THEME_PFX + name);
                     if (name.equals(current))
-                        ctx.router().setConfigSetting(CSSHelper.PROP_THEME_NAME, CSSHelper.DEFAULT_THEME);
+                        changes.put(CSSHelper.PROP_THEME_NAME, CSSHelper.DEFAULT_THEME);
                 }
             }
-            ctx.router().saveConfig();
+            ctx.router().saveConfig(changes, removes);
         }
 
         FileUtil.rmdir(pluginDir, false);
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/RouterConsoleRunner.java b/apps/routerconsole/java/src/net/i2p/router/web/RouterConsoleRunner.java
index ddcb98a0da57044531eb248da309d0ee251aefa7..3d624c64a2535d9dcf831dd873830ce2480ca50c 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/RouterConsoleRunner.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/RouterConsoleRunner.java
@@ -6,7 +6,9 @@ import java.io.File;
 import java.io.FilenameFilter;
 import java.io.IOException;
 import java.security.KeyStore;
+import java.util.HashMap;
 import java.util.List;
+import java.util.Map;
 import java.util.Properties;
 import java.util.StringTokenizer;
 
@@ -431,9 +433,10 @@ public class RouterConsoleRunner {
                 SecureFileOutputStream.setPerms(ks);
                 try {
                     RouterContext rctx = (RouterContext) ctx;
-                    rctx.router().setConfigSetting(PROP_KEYSTORE_PASSWORD, DEFAULT_KEYSTORE_PASSWORD);
-                    rctx.router().setConfigSetting(PROP_KEY_PASSWORD, keyPassword);
-                    rctx.router().saveConfig();
+                    Map<String, String> changes = new HashMap();
+                    changes.put(PROP_KEYSTORE_PASSWORD, DEFAULT_KEYSTORE_PASSWORD);
+                    changes.put(PROP_KEY_PASSWORD, keyPassword);
+                    rctx.router().saveConfig(changes, null);
                 } catch (Exception e) {}  // class cast exception
             }
         }
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/UnsignedUpdateHandler.java b/apps/routerconsole/java/src/net/i2p/router/web/UnsignedUpdateHandler.java
index 90e120af6e96e12cfcd38c6792ebf38c646979f0..27a66996b98a215f273a44392c661aeb18cf689f 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/UnsignedUpdateHandler.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/UnsignedUpdateHandler.java
@@ -105,8 +105,7 @@ public class UnsignedUpdateHandler extends UpdateHandler {
                     modtime = RFC822Date.parse822Date(lastmod);
                 if (modtime <= 0)
                     modtime = _context.clock().now();
-                _context.router().setConfigSetting(PROP_LAST_UPDATE_TIME, "" + modtime);
-                _context.router().saveConfig();
+                _context.router().saveConfig(PROP_LAST_UPDATE_TIME, "" + modtime);
                 if ("install".equals(policy)) {
                     _log.log(Log.CRIT, "Update was downloaded, restarting to install it");
                     updateStatus("<b>" + _("Update downloaded") + "</b><br>" + _("Restarting"));
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/UpdateHandler.java b/apps/routerconsole/java/src/net/i2p/router/web/UpdateHandler.java
index 5b7eaec03c2bfe90288ee8be0fa4c2c83160171d..2253fb75a60907dea742f2d3078c6695f40ae81e 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/UpdateHandler.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/UpdateHandler.java
@@ -275,8 +275,7 @@ public class UpdateHandler {
                     modtime = RFC822Date.parse822Date(lastmod);
                 if (modtime <= 0)
                     modtime = _context.clock().now();
-                _context.router().setConfigSetting(PROP_LAST_UPDATE_TIME, "" + modtime);
-                _context.router().saveConfig();
+                _context.router().saveConfig(PROP_LAST_UPDATE_TIME, "" + modtime);
                 if ("install".equals(policy)) {
                     _log.log(Log.CRIT, "Update was VERIFIED, restarting to install it");
                     updateStatus("<b>" + _("Update verified") + "</b><br>" + _("Restarting"));
diff --git a/router/java/src/net/i2p/router/PersistentKeyRing.java b/router/java/src/net/i2p/router/PersistentKeyRing.java
index af7c8c8dec6fa39b167c433a9c41f30a83c8d832..1a8ec464cfd0028a4f997ab2260edbcabce36e1d 100644
--- a/router/java/src/net/i2p/router/PersistentKeyRing.java
+++ b/router/java/src/net/i2p/router/PersistentKeyRing.java
@@ -31,16 +31,14 @@ public class PersistentKeyRing extends KeyRing {
     public SessionKey put(Hash h, SessionKey sk) {
         SessionKey old = super.put(h, sk);
         if (!sk.equals(old)) {
-            _ctx.router().setConfigSetting(PROP_PFX + h.toBase64().replace("=", "$"),
+            _ctx.router().saveConfig(PROP_PFX + h.toBase64().replace("=", "$"),
                                            sk.toBase64());
-            _ctx.router().saveConfig();
         }
         return old;
     }
 
     public SessionKey remove(Hash h) {
-        _ctx.router().removeConfigSetting(PROP_PFX + h.toBase64().replace("=", "$"));
-        _ctx.router().saveConfig();
+        _ctx.router().saveConfig(PROP_PFX + h.toBase64().replace("=", "$"), null);
         return super.remove(h);
     }
 
diff --git a/router/java/src/net/i2p/router/Router.java b/router/java/src/net/i2p/router/Router.java
index f97422d02e720848627d16af11ddfa1397f44570..473714c4ee53b0bf1e54e7e50dce723ca6d0cead 100644
--- a/router/java/src/net/i2p/router/Router.java
+++ b/router/java/src/net/i2p/router/Router.java
@@ -660,9 +660,11 @@ public class Router implements RouterClock.ClockShiftListener {
         }
 
         // now that we have random ports, keeping the same port would be bad
-        removeConfigSetting(UDPTransport.PROP_INTERNAL_PORT);
-        removeConfigSetting(UDPTransport.PROP_EXTERNAL_PORT);
-        saveConfig();
+        synchronized(this) {
+            removeConfigSetting(UDPTransport.PROP_INTERNAL_PORT);
+            removeConfigSetting(UDPTransport.PROP_EXTERNAL_PORT);
+            saveConfig();
+        }
 
         if (remCount > 0) {
             FileOutputStream log = null;
@@ -1061,8 +1063,7 @@ public class Router implements RouterClock.ClockShiftListener {
 
         // Set the last version to the current version, since 0.8.13
         if (!RouterVersion.VERSION.equals(_config.get("router.previousVersion"))) {
-            _config.put("router.previousVersion", RouterVersion.VERSION);
-            saveConfig();
+            saveConfig("router.previousVersion", RouterVersion.VERSION);
         }
 
         _context.removeShutdownTasks();
@@ -1301,7 +1302,7 @@ public class Router implements RouterClock.ClockShiftListener {
             _config.putAll(toAdd);
         if (toRemove != null) {
             for (String s : toRemove) {
-                _config.remove(toRemove);
+                _config.remove(s);
             }
         }
         return saveConfig();
diff --git a/router/java/src/net/i2p/router/client/SSLClientListenerRunner.java b/router/java/src/net/i2p/router/client/SSLClientListenerRunner.java
index 8d423ef0b31d2183f29bd0f830de84b8c54672fc..8e490c46a35ddab10b101241d5d85722fe8df7d2 100644
--- a/router/java/src/net/i2p/router/client/SSLClientListenerRunner.java
+++ b/router/java/src/net/i2p/router/client/SSLClientListenerRunner.java
@@ -15,6 +15,8 @@ import java.security.GeneralSecurityException;
 import java.security.cert.Certificate;
 import java.security.cert.CertificateEncodingException;
 import java.util.Arrays;
+import java.util.HashMap;
+import java.util.Map;
 
 import javax.net.ssl.KeyManagerFactory;
 import javax.net.ssl.SSLServerSocketFactory;
@@ -112,9 +114,10 @@ class SSLClientListenerRunner extends ClientListenerRunner {
             success = ks.exists();
             if (success) {
                 SecureFileOutputStream.setPerms(ks);
-                _context.router().setConfigSetting(PROP_KEYSTORE_PASSWORD, DEFAULT_KEYSTORE_PASSWORD);
-                _context.router().setConfigSetting(PROP_KEY_PASSWORD, keyPassword);
-                _context.router().saveConfig();
+                Map<String, String> changes = new HashMap();
+                changes.put(PROP_KEYSTORE_PASSWORD, DEFAULT_KEYSTORE_PASSWORD);
+                changes.put(PROP_KEY_PASSWORD, keyPassword);
+                _context.router().saveConfig(changes, null);
             }
         }
         if (success) {
diff --git a/router/java/src/net/i2p/router/transport/CommSystemFacadeImpl.java b/router/java/src/net/i2p/router/transport/CommSystemFacadeImpl.java
index 73fb5cb93016c5e3e00dc8fb74ba03d1b8534013..cb93d53337c60f38532090167058afc006626781 100644
--- a/router/java/src/net/i2p/router/transport/CommSystemFacadeImpl.java
+++ b/router/java/src/net/i2p/router/transport/CommSystemFacadeImpl.java
@@ -252,9 +252,10 @@ public class CommSystemFacadeImpl extends CommSystemFacade {
         addr.setTransportStyle(NTCPTransport.STYLE);
         //if (isNew) {
             // why save the same thing?
-            ctx.router().setConfigSetting(PROP_I2NP_NTCP_HOSTNAME, name);
-            ctx.router().setConfigSetting(PROP_I2NP_NTCP_PORT, port);
-            ctx.router().saveConfig();
+            Map<String, String> changes = new HashMap();
+            changes.put(PROP_I2NP_NTCP_HOSTNAME, name);
+            changes.put(PROP_I2NP_NTCP_PORT, port);
+            ctx.router().saveConfig(changes, null);
         //}
         return addr;
     }
diff --git a/router/java/src/net/i2p/router/transport/GeoIP.java b/router/java/src/net/i2p/router/transport/GeoIP.java
index 8a97a1f5f38cef353bbd15ace57645d75953418b..56d24a9cbd9c31d9996ea7cd5ce832574538b664 100644
--- a/router/java/src/net/i2p/router/transport/GeoIP.java
+++ b/router/java/src/net/i2p/router/transport/GeoIP.java
@@ -282,8 +282,7 @@ class GeoIP {
             return;
         String country = _context.commSystem().getCountry(ourHash);
         if (country != null && !country.equals(oldCountry)) {
-            _context.router().setConfigSetting(PROP_IP_COUNTRY, country);
-            _context.router().saveConfig();
+            _context.router().saveConfig(PROP_IP_COUNTRY, country);
             if (_context.commSystem().isInBadCountry() && _context.getProperty(Router.PROP_HIDDEN_HIDDEN) == null) {
                 String name = fullName(country);
                 if (name == null)
diff --git a/router/java/src/net/i2p/router/transport/udp/UDPTransport.java b/router/java/src/net/i2p/router/transport/udp/UDPTransport.java
index c642a68863c4a8e7eac21dd767b6d4a4b34ba56f..3f233c53cbc4431024ab00aa4eca9a8b38feba14 100644
--- a/router/java/src/net/i2p/router/transport/udp/UDPTransport.java
+++ b/router/java/src/net/i2p/router/transport/udp/UDPTransport.java
@@ -319,9 +319,10 @@ public class UDPTransport extends TransportImpl implements TimedWeightedPriority
         if (newPort != port || newPort != oldIPort || newPort != oldEPort) {
             // attempt to use it as our external port - this will be overridden by
             // externalAddressReceived(...)
-            _context.router().setConfigSetting(PROP_INTERNAL_PORT, newPort+"");
-            _context.router().setConfigSetting(PROP_EXTERNAL_PORT, newPort+"");
-            _context.router().saveConfig();
+            Map<String, String> changes = new HashMap();
+            changes.put(PROP_INTERNAL_PORT, newPort+"");
+            changes.put(PROP_EXTERNAL_PORT, newPort+"");
+            _context.router().saveConfig(changes, null);
         }
 
         _establisher.startup();
@@ -560,8 +561,9 @@ public class UDPTransport extends TransportImpl implements TimedWeightedPriority
             _context.statManager().addRateData("udp.addressTestInsteadOfUpdate", 1, 0);
         } else if (updated) {
             _context.statManager().addRateData("udp.addressUpdated", 1, 0);
+            Map<String, String> changes = new HashMap();
             if (!fixedPort)
-                _context.router().setConfigSetting(PROP_EXTERNAL_PORT, ourPort+"");
+                changes.put(PROP_EXTERNAL_PORT, ourPort+"");
             // queue a country code lookup of the new IP
             _context.commSystem().queueLookup(ourIP);
             // store these for laptop-mode (change ident on restart... or every time... when IP changes)
@@ -576,9 +578,9 @@ public class UDPTransport extends TransportImpl implements TimedWeightedPriority
                     } catch (NumberFormatException nfe) {}
                 }
 
-                _context.router().setConfigSetting(PROP_IP, _externalListenHost.getHostAddress());
-                _context.router().setConfigSetting(PROP_IP_CHANGE, "" + now);
-                _context.router().saveConfig();
+                changes.put(PROP_IP, _externalListenHost.getHostAddress());
+                changes.put(PROP_IP_CHANGE, "" + now);
+                _context.router().saveConfig(changes, null);
 
                 // laptop mode
                 // For now, only do this at startup
@@ -596,6 +598,9 @@ public class UDPTransport extends TransportImpl implements TimedWeightedPriority
                     _context.router().shutdown(Router.EXIT_HARD_RESTART);
                     // doesn't return
                 }
+            } else if (!fixedPort) {
+                // save PROP_EXTERNAL_PORT
+                _context.router().saveConfig(changes, null);
             }
             _context.router().rebuildRouterInfo();
         }