diff --git a/IPython/html/static/notebook/js/menubar.js b/IPython/html/static/notebook/js/menubar.js index eaf7584..04779b7 100644 --- a/IPython/html/static/notebook/js/menubar.js +++ b/IPython/html/static/notebook/js/menubar.js @@ -295,6 +295,9 @@ define([ this.element.find('#restart_kernel').click(function () { that.notebook.restart_kernel(); }); + this.element.find('#reconnect_kernel').click(function () { + that.notebook.kernel.reconnect(); + }); // Help if (this.tour) { this.element.find('#notebook_tour').click(function () { diff --git a/IPython/html/static/notebook/js/notificationarea.js b/IPython/html/static/notebook/js/notificationarea.js index 1b50a27..ca71b13 100644 --- a/IPython/html/static/notebook/js/notificationarea.js +++ b/IPython/html/static/notebook/js/notificationarea.js @@ -132,6 +132,13 @@ define([ knw.warning("Connecting to kernel"); }); + this.events.on('kernel_connection_dead.Kernel', function (evt, info) { + knw.danger("Not Connected", undefined, function () { + // schedule reconnect a short time in the future, don't reconnect immediately + setTimeout($.proxy(info.kernel.reconnect, info.kernel), 500); + }, {title: 'click to reconnect'}); + }); + this.events.on('kernel_connected.Kernel', function () { knw.info("Connected", 500); }); diff --git a/IPython/html/static/services/kernels/kernel.js b/IPython/html/static/services/kernels/kernel.js index 38ac473..77cf4b1 100644 --- a/IPython/html/static/services/kernels/kernel.js +++ b/IPython/html/static/services/kernels/kernel.js @@ -66,6 +66,7 @@ define([ this._autorestart_attempt = 0; this._reconnect_attempt = 0; + this.reconnect_limit = 7; }; /** @@ -332,8 +333,15 @@ define([ * @function reconnect */ Kernel.prototype.reconnect = function () { - this.events.trigger('kernel_reconnecting.Kernel', {kernel: this}); - setTimeout($.proxy(this.start_channels, this), 3000); + if (this.is_connected()) { + return; + } + this._reconnect_attempt = this._reconnect_attempt + 1; + this.events.trigger('kernel_reconnecting.Kernel', { + kernel: this, + attempt: this._reconnect_attempt, + }); + this.start_channels(); }; /** @@ -524,12 +532,27 @@ define([ this.events.trigger('kernel_disconnected.Kernel', {kernel: this}); if (error) { console.log('WebSocket connection failed: ', ws_url); - this._reconnect_attempt = this._reconnect_attempt + 1; this.events.trigger('kernel_connection_failed.Kernel', {kernel: this, ws_url: ws_url, attempt: this._reconnect_attempt}); } - this.reconnect(); + this._schedule_reconnect(); }; - + + Kernel.prototype._schedule_reconnect = function () { + // function to call when kernel connection is lost + // schedules reconnect, or fires 'connection_dead' if reconnect limit is hit + if (this._reconnect_attempt < this.reconnect_limit) { + var timeout = Math.pow(2, this._reconnect_attempt); + console.log("Connection lost, reconnecting in " + timeout + " seconds."); + setTimeout($.proxy(this.reconnect, this), 1e3 * timeout); + } else { + this.events.trigger('kernel_connection_dead.Kernel', { + kernel: this, + reconnect_attempt: this._reconnect_attempt, + }); + console.log("Failed to reconnect, giving up."); + } + }; + /** * Close the websocket channels. After successful close, the value * in `this.channels[channel_name]` will be null. diff --git a/IPython/html/templates/notebook.html b/IPython/html/templates/notebook.html index 958374e..4a9a890 100644 --- a/IPython/html/templates/notebook.html +++ b/IPython/html/templates/notebook.html @@ -230,10 +230,16 @@ class="notebook_app"