diff --git a/bin/browserid b/bin/browserid
index 32cca02f7f9656f22e46ae55c3ad5d3cb9913121..75b8e889c0755165b5c41352c6c1e31c285ed450 100755
--- a/bin/browserid
+++ b/bin/browserid
@@ -40,7 +40,8 @@ fs = require('fs'),
 path = require('path'),
 url = require('url'),
 http = require('http');
-sessions = require('connect-cookie-session');
+sessions = require('connect-cookie-session'),
+urlparse = require('urlparse');
 
 // add lib/ to the require path
 require.paths.unshift(path.join(__dirname, '..', 'lib'));
@@ -55,9 +56,9 @@ db = require('db.js'),
 config = require('configuration.js'),
 heartbeat = require('heartbeat.js'),
 metrics = require("metrics.js"),
-logger = require("logging.js").logger,
+logger = require("logging.js").logger
 forward = require('browserid/http_forward'),
-urlparse = require('urlparse');
+shutdown = require('shutdown');
 
 var app = undefined;
 
@@ -205,9 +206,9 @@ function router(app) {
     });
   });
 
-  app.get('/code_update', function(req, resp, next) {
-    logger.warn("code updated.  shutting down.");
-    process.exit();
+  shutdown.installUpdateHandler(app, function(readyForShutdown) {
+    logger.debug("closing database connection");
+    db.close(readyForShutdown)
   });
 };
 
@@ -364,6 +365,12 @@ app.use(express.static(path.join(__dirname, "..", "resources", "static")));
 
 // open the databse
 db.open(config.get('database'), function () {
+
+  // shut down express gracefully on SIGINT
+  shutdown.handleTerminationSignals(app, function(readyForShutdown) {
+    db.close(readyForShutdown)
+  });
+
   var bindTo = config.get('bind_to');
   app.listen(bindTo.port, bindTo.host, function() {
     logger.info("running on http://" + app.address().address + ":" + app.address().port);
diff --git a/bin/keysigner b/bin/keysigner
index 211cf1c19b1bc563e6f207cbd8d9acc675ca448a..90314d82a164d3d4ddd99853eb4309ee1a2976f6 100755
--- a/bin/keysigner
+++ b/bin/keysigner
@@ -50,7 +50,8 @@ validate = require('validate.js'),
 metrics = require("metrics.js"),
 logger = require("logging.js").logger,
 ca = require('keysigner/ca.js'),
-heartbeat = require('heartbeat');
+heartbeat = require('heartbeat'),
+shutdown = require('shutdown');
 
 // create an express server
 var app = express.createServer();
@@ -90,6 +91,12 @@ app.post('/wsapi/cert_key', validate(["email", "pubkey"]), function(req, resp) {
   resp.end();
 });
 
+// shutdown when code_update is invoked
+shutdown.installUpdateHandler(app);
+
+// shutdown nicely on signals
+shutdown.handleTerminationSignals(app);
+
 var bindTo = config.get('bind_to');
 app.listen(bindTo.port, bindTo.host, function() {
   logger.info("running on http://" + app.address().address + ":" + app.address().port);
diff --git a/bin/verifier b/bin/verifier
index de9740281dbafe04f826282e17e7411e1d1c2bcb..305642f54d79e519b9dd3cfc98b46b2796f506bc 100755
--- a/bin/verifier
+++ b/bin/verifier
@@ -46,7 +46,8 @@ certassertion = require('../lib/verifier/certassertion.js'),
 metrics = require('../lib/metrics'),
 heartbeat = require('../lib/heartbeat'),
 logger = require('../lib/logging').logger,
-config = require('../lib/configuration');
+config = require('../lib/configuration'),
+shutdown = require('../lib/shutdown');
 
 logger.info("verifier server starting up");
 
@@ -64,13 +65,8 @@ app.use(express.logger({
 
 app.use(express.bodyParser());
 
-// code_update is an internal api that causes the node server to
-// shut down.  This should never be externally accessible and
-// is used during the dead simple deployment procedure.
-app.get("/code_update", function (req, resp) {
-  logger.warn("code updated.  shutting down.");
-  process.exit();
-});
+// shutdown when /code_update is invoked
+shutdown.installUpdateHandler(app);
 
 // setup health check / heartbeat
 heartbeat.setup(app);
@@ -119,6 +115,9 @@ app.post('/verify', function(req, resp, next) {
 
 });
 
+// shutdown nicely on signals
+shutdown.handleTerminationSignals(app);
+
 var bindTo = config.get('bind_to');
 app.listen(bindTo.port, bindTo.host, function(conn) {
   logger.info("running on http://" + app.address().address + ":" + app.address().port);
diff --git a/lib/logging.js b/lib/logging.js
index 6382a9030096f2e1515d48e4122f7fef0e8ed132..6d22f08cbc43adf7b1e3dbd78ae5405ccab63469 100644
--- a/lib/logging.js
+++ b/lib/logging.js
@@ -70,14 +70,18 @@ var filename = path.join(log_path, configuration.get('process_type') + ".log");
 exports.logger = new (winston.Logger)({
   transports: [new (winston.transports.File)({
     filename: filename,
-    colorize: true
+    colorize: true,
+    handleExceptions: true
   })]
 });
 
-exports.logger.emitErrs = false;
-
 exports.enableConsoleLogging = function() {
-  exports.logger.add(winston.transports.Console, { colorize: true });
+  exports.logger.add(winston.transports.Console, {
+    colorize: true,
+    handleExceptions: true
+  });
 };
 
+winston.handleExceptions();
+
 if (process.env['LOG_TO_CONSOLE']) exports.enableConsoleLogging();
diff --git a/lib/shutdown.js b/lib/shutdown.js
new file mode 100644
index 0000000000000000000000000000000000000000..308349ca6f652b045db2f293a6d669d74f69cabb
--- /dev/null
+++ b/lib/shutdown.js
@@ -0,0 +1,120 @@
+/* ***** BEGIN LICENSE BLOCK *****
+ * Version: MPL 1.1/GPL 2.0/LGPL 2.1
+ *
+ * The contents of this file are subject to the Mozilla Public License Version
+ * 1.1 (the "License"); you may not use this file except in compliance with
+ * the License. You may obtain a copy of the License at
+ * http://www.mozilla.org/MPL/
+ *
+ * Software distributed under the License is distributed on an "AS IS" basis,
+ * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
+ * for the specific language governing rights and limitations under the
+ * License.
+ *
+ * The Original Code is Mozilla BrowserID.
+ *
+ * The Initial Developer of the Original Code is Mozilla.
+ * Portions created by the Initial Developer are Copyright (C) 2011
+ * the Initial Developer. All Rights Reserved.
+ *
+ * Contributor(s):
+ *   Lloyd Hilaiel <lloyd@hilaiel.com>
+ *
+ * Alternatively, the contents of this file may be used under the terms of
+ * either the GNU General Public License Version 2 or later (the "GPL"), or
+ * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
+ * in which case the provisions of the GPL or the LGPL are applicable instead
+ * of those above. If you wish to allow use of your version of this file only
+ * under the terms of either the GPL or the LGPL, and not to allow others to
+ * use your version of this file under the terms of the MPL, indicate your
+ * decision by deleting the provisions above and replace them with the notice
+ * and other provisions required by the GPL or the LGPL. If you do not delete
+ * the provisions above, a recipient may use your version of this file under
+ * the terms of any one of the MPL, the GPL or the LGPL.
+ *
+ * ***** END LICENSE BLOCK ***** */
+
+/* code_update is a tiny abstraction of a handler that can be
+ * used to shutdown gracefully upon signals, and can be used
+ * to install a 'code_update' hook into a running express
+ * server.
+ */
+
+const logger = require("./logging.js").logger;
+
+const MAX_WAIT_MS = 10000;
+const MAX_NICE_END_MS = 5000;
+
+function connectionListener(app) {
+  var connections = [];
+
+  app.on('connection', function(c) {
+    connections.push(c);
+    c.on('close', function() {
+      var where = connections.indexOf(c);
+      if (where >= 0) connections.splice(where, 1);
+    });
+  });
+
+  return function(callback) {
+    if (!callback) callback = function(cli) { cli(); };
+
+    var total_timeout = setTimeout(function() {
+      logger.warn(MAX_WAIT_MS + "ms exceeded, going down forcefully...");
+      setTimeout(function() { process.exit(1); }, 0);
+    }, MAX_WAIT_MS);
+
+    var nice_timeout = setTimeout(function() {
+      logger.warn("forcefully closing " + connections.length + " remaining connections...");
+      connections.forEach(function(c) { c.destroy() });
+    }, MAX_NICE_END_MS);
+
+    app.on('close', function() {
+      function clearTimeoutsAndCallClient() {
+        clearTimeout(nice_timeout);
+        clearTimeout(total_timeout);
+        callback(function() {
+          logger.info("graceful shutdown complete...");
+        });
+      }
+
+      // if there aren't any open connections, we're done!
+      if (connections.length === 0) clearTimeoutsAndCallClient();
+
+      connections.forEach(function(c) {
+        c.on('close', function() {
+          if (!app.connections && connections.length === 0) {
+            // once all connections are shutdown, let's call the client
+            // to let him shutdown all his open connections
+            clearTimeoutsAndCallClient();
+          }
+        });
+        c.end();
+      });
+    });
+    app.close();
+  }
+};
+
+exports.handleTerminationSignals = function(app, callback) {
+  var gotSignal = false;
+  var terminate = connectionListener(app);
+  function endIt(signame) {
+    return function() {
+      if (gotSignal) return;
+      gotSignal = true;
+      logger.warn("SIG" + signame + " received.  closing " + app.connections + " connections and shutting down.");
+      terminate(callback);
+    };
+  }
+
+  process.on('SIGINT', endIt('INT')).on('SIGTERM', endIt('TERM')).on('SIGQUIT', endIt('QUIT'));
+};
+
+exports.installUpdateHandler = function(app, callback) {
+  var terminate = connectionListener(app);
+  app.get('/code_update', function(req, resp, next) {
+    logger.warn("code updated.  closing " + app.connections + " connections and shutting down.");
+    terminate(callback);
+  });
+};
diff --git a/package.json b/package.json
index b56ddad2a88986421673c40b17bb08fd0fbcda8c..8499c3c578ecae58a683c228fe33119eb4987618 100644
--- a/package.json
+++ b/package.json
@@ -15,7 +15,7 @@
     , "express-csrf": "0.3.2"
     , "uglify-js": "1.0.6"
     , "JSONSelect": "0.2.1"
-    , "winston" : "0.3.3"
+    , "winston" : "0.5.6"
     , "connect-cookie-session" : "0.0.2"
     , "mysql" : "0.9.2"
     , "optimist" : "0.2.6"
diff --git a/scripts/run_locally.js b/scripts/run_locally.js
index 25ded3dcce690b0fae6302c17b56af024eb81917..6c48078ec856a57b424559dc710ef94bbeb0930a 100755
--- a/scripts/run_locally.js
+++ b/scripts/run_locally.js
@@ -61,7 +61,7 @@ Object.keys(daemonsToRun).forEach(function(k) {
   daemons[k] = p;
 
   p.on('exit', function (code, signal) {
-    console.log(k, 'exited with code', code, (signal ? 'on signal ' + signal : ""));
+    console.log(k, 'exited(' + code + ') ', (signal ? 'on signal ' + signal : ""));
     delete daemons[k];
     Object.keys(daemons).forEach(function (k) { daemons[k].kill(); });
     if (Object.keys(daemons).length === 0) {