kernel.js
484 lines
| 16.5 KiB
| application/javascript
|
JavascriptLexer
Brian E. Granger
|
r4609 | //---------------------------------------------------------------------------- | ||
// Copyright (C) 2008-2011 The IPython Development Team | ||||
// | ||||
// Distributed under the terms of the BSD License. The full license is in | ||||
// the file COPYING, distributed as part of this software. | ||||
//---------------------------------------------------------------------------- | ||||
Brian E. Granger
|
r4349 | |||
//============================================================================ | ||||
// Kernel | ||||
//============================================================================ | ||||
Matthias BUSSONNIER
|
r8768 | /** | ||
* @module IPython | ||||
* @namespace IPython | ||||
* @submodule Kernel | ||||
*/ | ||||
Brian E. Granger
|
r4352 | var IPython = (function (IPython) { | ||
Brian E. Granger
|
r4349 | |||
Brian E. Granger
|
r4352 | var utils = IPython.utils; | ||
Brian Granger
|
r7168 | // Initialization and connection. | ||
Matthias BUSSONNIER
|
r8768 | /** | ||
* A Kernel Class to communicate with the Python kernel | ||||
* @Class Kernel | ||||
*/ | ||||
Brian Granger
|
r7168 | var Kernel = function (base_url) { | ||
Brian E. Granger
|
r4352 | this.kernel_id = null; | ||
Brian E. Granger
|
r4545 | this.shell_channel = null; | ||
this.iopub_channel = null; | ||||
MinRK
|
r10366 | this.stdin_channel = null; | ||
Brian Granger
|
r7168 | this.base_url = base_url; | ||
Brian E. Granger
|
r4545 | this.running = false; | ||
MinRK
|
r4694 | this.username = "username"; | ||
this.session_id = utils.uuid(); | ||||
Brian Granger
|
r7168 | this._msg_callbacks = {}; | ||
Brian E. Granger
|
r4612 | if (typeof(WebSocket) !== 'undefined') { | ||
Stefan van der Walt
|
r5479 | this.WebSocket = WebSocket; | ||
Brian E. Granger
|
r4612 | } else if (typeof(MozWebSocket) !== 'undefined') { | ||
Stefan van der Walt
|
r5479 | this.WebSocket = MozWebSocket; | ||
Brian E. Granger
|
r4611 | } else { | ||
MinRK
|
r5253 | alert('Your browser does not have WebSocket support, please try Chrome, Safari or Firefox ≥ 6. Firefox 4 and 5 are also supported by you have to enable WebSockets in about:config.'); | ||
Brian E. Granger
|
r4611 | }; | ||
Brian E. Granger
|
r4352 | }; | ||
Brian Granger
|
r7168 | Kernel.prototype._get_msg = function (msg_type, content) { | ||
Brian E. Granger
|
r4352 | var msg = { | ||
header : { | ||||
msg_id : utils.uuid(), | ||||
MinRK
|
r4694 | username : this.username, | ||
session : this.session_id, | ||||
Brian E. Granger
|
r4352 | msg_type : msg_type | ||
}, | ||||
MinRK
|
r7958 | metadata : {}, | ||
Brian E. Granger
|
r4352 | content : content, | ||
parent_header : {} | ||||
}; | ||||
return msg; | ||||
Stefan van der Walt
|
r5479 | }; | ||
Brian E. Granger
|
r4352 | |||
Matthias BUSSONNIER
|
r8768 | /** | ||
* Start the Python kernel | ||||
* @method start | ||||
*/ | ||||
Brian Granger
|
r7168 | Kernel.prototype.start = function (notebook_id) { | ||
Brian E. Granger
|
r4352 | var that = this; | ||
Brian E. Granger
|
r4545 | if (!this.running) { | ||
var qs = $.param({notebook:notebook_id}); | ||||
Stefan van der Walt
|
r5479 | var url = this.base_url + '?' + qs; | ||
Brian E. Granger
|
r5106 | $.post(url, | ||
Brian Granger
|
r7168 | $.proxy(that._kernel_started,that), | ||
Brian E. Granger
|
r4545 | 'json' | ||
); | ||||
}; | ||||
}; | ||||
Matthias BUSSONNIER
|
r8768 | /** | ||
* Restart the python kernel. | ||||
* | ||||
* Emit a 'status_restarting.Kernel' event with | ||||
* the current object as parameter | ||||
* | ||||
* @method restart | ||||
*/ | ||||
Brian Granger
|
r7168 | Kernel.prototype.restart = function () { | ||
Matthias BUSSONNIER
|
r8677 | $([IPython.events]).trigger('status_restarting.Kernel', {kernel: this}); | ||
Brian E. Granger
|
r4545 | var that = this; | ||
if (this.running) { | ||||
this.stop_channels(); | ||||
Brian Granger
|
r7168 | var url = this.kernel_url + "/restart"; | ||
Brian E. Granger
|
r4545 | $.post(url, | ||
Brian Granger
|
r7168 | $.proxy(that._kernel_started, that), | ||
Brian E. Granger
|
r4545 | 'json' | ||
); | ||||
}; | ||||
Brian E. Granger
|
r4352 | }; | ||
Brian Granger
|
r7168 | Kernel.prototype._kernel_started = function (json) { | ||
console.log("Kernel started: ", json.kernel_id); | ||||
Brian E. Granger
|
r4545 | this.running = true; | ||
Brian E. Granger
|
r4572 | this.kernel_id = json.kernel_id; | ||
this.ws_url = json.ws_url; | ||||
Brian E. Granger
|
r4352 | this.kernel_url = this.base_url + "/" + this.kernel_id; | ||
Brian E. Granger
|
r4545 | this.start_channels(); | ||
Matthias BUSSONNIER
|
r8690 | $([IPython.events]).trigger('status_started.Kernel', {kernel: this}); | ||
Brian E. Granger
|
r4352 | }; | ||
Brian Granger
|
r7168 | |||
Brian E. Granger
|
r9222 | Kernel.prototype._websocket_closed = function(ws_url, early) { | ||
this.stop_channels(); | ||||
$([IPython.events]).trigger('websocket_closed.Kernel', | ||||
{ws_url: ws_url, kernel: this, early: early} | ||||
); | ||||
Stefan van der Walt
|
r5479 | }; | ||
Brian E. Granger
|
r4352 | |||
Matthias BUSSONNIER
|
r8768 | /** | ||
* Start the `shell`and `iopub` channels. | ||||
* Will stop and restart them if they already exist. | ||||
* | ||||
* @method start_channels | ||||
*/ | ||||
Brian E. Granger
|
r4545 | Kernel.prototype.start_channels = function () { | ||
MinRK
|
r5253 | var that = this; | ||
Brian E. Granger
|
r4545 | this.stop_channels(); | ||
Brian E. Granger
|
r4572 | var ws_url = this.ws_url + this.kernel_url; | ||
Brian E. Granger
|
r9222 | console.log("Starting WebSockets:", ws_url); | ||
Brian E. Granger
|
r4611 | this.shell_channel = new this.WebSocket(ws_url + "/shell"); | ||
MinRK
|
r10366 | this.stdin_channel = new this.WebSocket(ws_url + "/stdin"); | ||
Brian E. Granger
|
r4611 | this.iopub_channel = new this.WebSocket(ws_url + "/iopub"); | ||
MinRK
|
r4707 | send_cookie = function(){ | ||
MinRK
|
r10379 | // send the session id so the Session object Python-side | ||
// has the same identity | ||||
MinRK
|
r10365 | this.send(that.session_id + ':' + document.cookie); | ||
Stefan van der Walt
|
r5479 | }; | ||
MinRK
|
r5253 | var already_called_onclose = false; // only alert once | ||
Mikhail Korobov
|
r8839 | var ws_closed_early = function(evt){ | ||
MinRK
|
r5253 | if (already_called_onclose){ | ||
return; | ||||
} | ||||
already_called_onclose = true; | ||||
if ( ! evt.wasClean ){ | ||||
MinRK
|
r5255 | that._websocket_closed(ws_url, true); | ||
MinRK
|
r5253 | } | ||
Stefan van der Walt
|
r5479 | }; | ||
Mikhail Korobov
|
r8839 | var ws_closed_late = function(evt){ | ||
MinRK
|
r5253 | if (already_called_onclose){ | ||
return; | ||||
} | ||||
already_called_onclose = true; | ||||
if ( ! evt.wasClean ){ | ||||
MinRK
|
r5255 | that._websocket_closed(ws_url, false); | ||
MinRK
|
r5253 | } | ||
Stefan van der Walt
|
r5479 | }; | ||
MinRK
|
r10366 | var channels = [this.shell_channel, this.iopub_channel, this.stdin_channel]; | ||
for (var i=0; i < channels.length; i++) { | ||||
channels[i].onopen = send_cookie; | ||||
channels[i].onclose = ws_closed_early; | ||||
} | ||||
MinRK
|
r5253 | // switch from early-close to late-close message after 1s | ||
Brian E. Granger
|
r9222 | setTimeout(function() { | ||
MinRK
|
r10366 | for (var i=0; i < channels.length; i++) { | ||
if (channels[i] !== null) { | ||||
channels[i].onclose = ws_closed_late; | ||||
} | ||||
Brian E. Granger
|
r9222 | } | ||
MinRK
|
r5253 | }, 1000); | ||
MinRK
|
r10366 | this.shell_channel.onmessage = $.proxy(this._handle_shell_reply, this); | ||
this.iopub_channel.onmessage = $.proxy(this._handle_iopub_reply, this); | ||||
this.stdin_channel.onmessage = $.proxy(this._handle_input_request, this); | ||||
MinRK
|
r10368 | |||
$([IPython.events]).on('send_input_reply.Kernel', function(evt, data) { | ||||
that.send_input_reply(data); | ||||
}); | ||||
Brian E. Granger
|
r4545 | }; | ||
Brian E. Granger
|
r4352 | |||
Matthias BUSSONNIER
|
r8768 | /** | ||
* Start the `shell`and `iopub` channels. | ||||
* @method stop_channels | ||||
*/ | ||||
Brian E. Granger
|
r4545 | Kernel.prototype.stop_channels = function () { | ||
MinRK
|
r10366 | var channels = [this.shell_channel, this.iopub_channel, this.stdin_channel]; | ||
for (var i=0; i < channels.length; i++) { | ||||
if ( channels[i] !== null ) { | ||||
channels[i].onclose = function (evt) {}; | ||||
channels[i].close(); | ||||
} | ||||
Brian E. Granger
|
r4545 | }; | ||
MinRK
|
r10366 | this.shell_channel = this.iopub_channel = this.stdin_channel = null; | ||
Brian E. Granger
|
r4545 | }; | ||
Brian Granger
|
r7168 | // Main public methods. | ||
Matthias BUSSONNIER
|
r8768 | /** | ||
* Get info on object asynchronoulsy | ||||
* | ||||
* @async | ||||
* @param objname {string} | ||||
* @param callback {dict} | ||||
* @method object_info_request | ||||
* | ||||
* @example | ||||
* | ||||
* When calling this method pass a callbacks structure of the form: | ||||
* | ||||
* callbacks = { | ||||
* 'object_info_reply': object_info_reply_callback | ||||
* } | ||||
* | ||||
* The `object_info_reply_callback` will be passed the content object of the | ||||
* | ||||
* `object_into_reply` message documented in | ||||
* [IPython dev documentation](http://ipython.org/ipython-doc/dev/development/messaging.html#object-information) | ||||
*/ | ||||
Brian Granger
|
r7168 | Kernel.prototype.object_info_request = function (objname, callbacks) { | ||
Matthias BUSSONNIER
|
r5522 | if(typeof(objname)!=null && objname!=null) | ||
Matthias BUSSONNIER
|
r5409 | { | ||
var content = { | ||||
oname : objname.toString(), | ||||
}; | ||||
Brian Granger
|
r7168 | var msg = this._get_msg("object_info_request", content); | ||
Matthias BUSSONNIER
|
r5409 | this.shell_channel.send(JSON.stringify(msg)); | ||
Brian Granger
|
r7168 | this.set_callbacks_for_msg(msg.header.msg_id, callbacks); | ||
Matthias BUSSONNIER
|
r5409 | return msg.header.msg_id; | ||
} | ||||
return; | ||||
Matthias BUSSONNIER
|
r5397 | } | ||
Matthias BUSSONNIER
|
r8768 | /** | ||
* Execute given code into kernel, and pass result to callback. | ||||
* | ||||
Matthias BUSSONNIER
|
r10594 | * TODO: document input_request in callbacks | ||
* | ||||
Matthias BUSSONNIER
|
r8768 | * @async | ||
* @method execute | ||||
* @param {string} code | ||||
Matthias Bussonnier
|
r10615 | * @param [callbacks] {Object} With the optional following keys | ||
Matthias BUSSONNIER
|
r10594 | * @param callbacks.'execute_reply' {function} | ||
* @param callbacks.'output' {function} | ||||
* @param callbacks.'clear_output' {function} | ||||
* @param callbacks.'set_next_input' {function} | ||||
Matthias BUSSONNIER
|
r8768 | * @param {object} [options] | ||
* @param [options.silent=false] {Boolean} | ||||
* @param [options.user_expressions=empty_dict] {Dict} | ||||
* @param [options.user_variables=empty_list] {List od Strings} | ||||
* @param [options.allow_stdin=false] {Boolean} true|false | ||||
* | ||||
* @example | ||||
* | ||||
* The options object should contain the options for the execute call. Its default | ||||
* values are: | ||||
* | ||||
* options = { | ||||
* silent : true, | ||||
* user_variables : [], | ||||
* user_expressions : {}, | ||||
* allow_stdin : false | ||||
* } | ||||
* | ||||
* When calling this method pass a callbacks structure of the form: | ||||
* | ||||
* callbacks = { | ||||
* 'execute_reply': execute_reply_callback, | ||||
* 'output': output_callback, | ||||
* 'clear_output': clear_output_callback, | ||||
* 'set_next_input': set_next_input_callback | ||||
* } | ||||
* | ||||
* The `execute_reply_callback` will be passed the content and metadata | ||||
* objects of the `execute_reply` message documented | ||||
* [here](http://ipython.org/ipython-doc/dev/development/messaging.html#execute) | ||||
* | ||||
* The `output_callback` will be passed `msg_type` ('stream','display_data','pyout','pyerr') | ||||
* of the output and the content and metadata objects of the PUB/SUB channel that contains the | ||||
* output: | ||||
* | ||||
* http://ipython.org/ipython-doc/dev/development/messaging.html#messages-on-the-pub-sub-socket | ||||
* | ||||
* The `clear_output_callback` will be passed a content object that contains | ||||
* stdout, stderr and other fields that are booleans, as well as the metadata object. | ||||
* | ||||
* The `set_next_input_callback` will be passed the text that should become the next | ||||
* input cell. | ||||
*/ | ||||
Brian Granger
|
r7176 | Kernel.prototype.execute = function (code, callbacks, options) { | ||
Brian E. Granger
|
r4352 | var content = { | ||
code : code, | ||||
Brian Granger
|
r7176 | silent : true, | ||
Brian E. Granger
|
r4352 | user_variables : [], | ||
MinRK
|
r4975 | user_expressions : {}, | ||
MinRK
|
r10368 | allow_stdin : false | ||
Brian E. Granger
|
r4352 | }; | ||
Matthias BUSSONNIER
|
r10594 | callbacks = callbacks || {}; | ||
MinRK
|
r10368 | if (callbacks.input_request !== undefined) { | ||
content.allow_stdin = true; | ||||
} | ||||
Matthias BUSSONNIER
|
r8605 | $.extend(true, content, options) | ||
Matthias BUSSONNIER
|
r8677 | $([IPython.events]).trigger('execution_request.Kernel', {kernel: this, content:content}); | ||
Brian Granger
|
r7168 | var msg = this._get_msg("execute_request", content); | ||
Brian E. Granger
|
r4352 | this.shell_channel.send(JSON.stringify(msg)); | ||
Brian Granger
|
r7168 | this.set_callbacks_for_msg(msg.header.msg_id, callbacks); | ||
Brian E. Granger
|
r4352 | return msg.header.msg_id; | ||
Stefan van der Walt
|
r5479 | }; | ||
Brian E. Granger
|
r4352 | |||
Matthias BUSSONNIER
|
r8768 | /** | ||
* When calling this method pass a callbacks structure of the form: | ||||
* | ||||
* callbacks = { | ||||
* 'complete_reply': complete_reply_callback | ||||
* } | ||||
* | ||||
* The `complete_reply_callback` will be passed the content object of the | ||||
* `complete_reply` message documented | ||||
* [here](http://ipython.org/ipython-doc/dev/development/messaging.html#complete) | ||||
* | ||||
* @method complete | ||||
* @param line {integer} | ||||
* @param cursor_pos {integer} | ||||
* @param {dict} callbacks | ||||
* @param callbacks.complete_reply {function} `complete_reply_callback` | ||||
* | ||||
*/ | ||||
Brian Granger
|
r7168 | Kernel.prototype.complete = function (line, cursor_pos, callbacks) { | ||
Brian Granger
|
r7212 | callbacks = callbacks || {}; | ||
Brian Granger
|
r4388 | var content = { | ||
text : '', | ||||
line : line, | ||||
cursor_pos : cursor_pos | ||||
}; | ||||
Brian Granger
|
r7168 | var msg = this._get_msg("complete_request", content); | ||
Brian Granger
|
r4388 | this.shell_channel.send(JSON.stringify(msg)); | ||
Brian Granger
|
r7168 | this.set_callbacks_for_msg(msg.header.msg_id, callbacks); | ||
Brian Granger
|
r4388 | return msg.header.msg_id; | ||
Stefan van der Walt
|
r5479 | }; | ||
Brian Granger
|
r4388 | |||
Brian E. Granger
|
r4352 | Kernel.prototype.interrupt = function () { | ||
Brian E. Granger
|
r4545 | if (this.running) { | ||
Matthias BUSSONNIER
|
r8677 | $([IPython.events]).trigger('status_interrupting.Kernel', {kernel: this}); | ||
Brian E. Granger
|
r4545 | $.post(this.kernel_url + "/interrupt"); | ||
}; | ||||
Brian E. Granger
|
r4349 | }; | ||
Brian E. Granger
|
r4496 | Kernel.prototype.kill = function () { | ||
Brian E. Granger
|
r4545 | if (this.running) { | ||
this.running = false; | ||||
var settings = { | ||||
cache : false, | ||||
Stefan van der Walt
|
r5479 | type : "DELETE" | ||
Brian E. Granger
|
r4545 | }; | ||
$.ajax(this.kernel_url, settings); | ||||
Brian E. Granger
|
r4496 | }; | ||
}; | ||||
MinRK
|
r10368 | Kernel.prototype.send_input_reply = function (input) { | ||
MinRK
|
r10366 | var content = { | ||
value : input, | ||||
}; | ||||
$([IPython.events]).trigger('input_reply.Kernel', {kernel: this, content:content}); | ||||
var msg = this._get_msg("input_reply", content); | ||||
this.stdin_channel.send(JSON.stringify(msg)); | ||||
return msg.header.msg_id; | ||||
}; | ||||
// Reply handlers | ||||
Brian Granger
|
r7168 | |||
Kernel.prototype.get_callbacks_for_msg = function (msg_id) { | ||||
var callbacks = this._msg_callbacks[msg_id]; | ||||
return callbacks; | ||||
}; | ||||
Kernel.prototype.set_callbacks_for_msg = function (msg_id, callbacks) { | ||||
Brian Granger
|
r7212 | this._msg_callbacks[msg_id] = callbacks || {}; | ||
Brian Granger
|
r7168 | } | ||
Kernel.prototype._handle_shell_reply = function (e) { | ||||
Mikhail Korobov
|
r8839 | var reply = $.parseJSON(e.data); | ||
Matthias BUSSONNIER
|
r8677 | $([IPython.events]).trigger('shell_reply.Kernel', {kernel: this, reply:reply}); | ||
Brian Granger
|
r7168 | var header = reply.header; | ||
var content = reply.content; | ||||
Jason Grout
|
r7952 | var metadata = reply.metadata; | ||
Brian Granger
|
r7168 | var msg_type = header.msg_type; | ||
var callbacks = this.get_callbacks_for_msg(reply.parent_header.msg_id); | ||||
if (callbacks !== undefined) { | ||||
var cb = callbacks[msg_type]; | ||||
if (cb !== undefined) { | ||||
Jason Grout
|
r7952 | cb(content, metadata); | ||
Brian Granger
|
r7168 | } | ||
}; | ||||
Brian Granger
|
r7223 | if (content.payload !== undefined) { | ||
Brian Granger
|
r7168 | var payload = content.payload || []; | ||
Brian Granger
|
r7223 | this._handle_payload(callbacks, payload); | ||
Brian Granger
|
r7168 | } | ||
}; | ||||
Brian Granger
|
r7223 | Kernel.prototype._handle_payload = function (callbacks, payload) { | ||
Brian Granger
|
r7168 | var l = payload.length; | ||
// Payloads are handled by triggering events because we don't want the Kernel | ||||
// to depend on the Notebook or Pager classes. | ||||
for (var i=0; i<l; i++) { | ||||
MinRK
|
r9412 | if (payload[i].source === 'IPython.kernel.zmq.page.page') { | ||
Brian Granger
|
r7168 | var data = {'text':payload[i].text} | ||
$([IPython.events]).trigger('open_with_text.Pager', data); | ||||
MinRK
|
r9412 | } else if (payload[i].source === 'IPython.kernel.zmq.zmqshell.ZMQInteractiveShell.set_next_input') { | ||
Brian Granger
|
r7223 | if (callbacks.set_next_input !== undefined) { | ||
callbacks.set_next_input(payload[i].text) | ||||
} | ||||
Brian Granger
|
r7168 | } | ||
}; | ||||
}; | ||||
Kernel.prototype._handle_iopub_reply = function (e) { | ||||
Jason Grout
|
r7952 | var reply = $.parseJSON(e.data); | ||
Brian Granger
|
r7168 | var content = reply.content; | ||
var msg_type = reply.header.msg_type; | ||||
Jason Grout
|
r7955 | var metadata = reply.metadata; | ||
Brian Granger
|
r7168 | var callbacks = this.get_callbacks_for_msg(reply.parent_header.msg_id); | ||
if (msg_type !== 'status' && callbacks === undefined) { | ||||
// Message not from one of this notebook's cells and there are no | ||||
// callbacks to handle it. | ||||
return; | ||||
} | ||||
var output_types = ['stream','display_data','pyout','pyerr']; | ||||
if (output_types.indexOf(msg_type) >= 0) { | ||||
var cb = callbacks['output']; | ||||
if (cb !== undefined) { | ||||
Jason Grout
|
r7952 | cb(msg_type, content, metadata); | ||
Brian Granger
|
r7168 | } | ||
} else if (msg_type === 'status') { | ||||
if (content.execution_state === 'busy') { | ||||
Matthias BUSSONNIER
|
r8677 | $([IPython.events]).trigger('status_busy.Kernel', {kernel: this}); | ||
Brian Granger
|
r7168 | } else if (content.execution_state === 'idle') { | ||
Matthias BUSSONNIER
|
r8677 | $([IPython.events]).trigger('status_idle.Kernel', {kernel: this}); | ||
MinRK
|
r10316 | } else if (content.execution_state === 'restarting') { | ||
$([IPython.events]).trigger('status_restarting.Kernel', {kernel: this}); | ||||
Brian Granger
|
r7168 | } else if (content.execution_state === 'dead') { | ||
this.stop_channels(); | ||||
Matthias BUSSONNIER
|
r8677 | $([IPython.events]).trigger('status_dead.Kernel', {kernel: this}); | ||
Brian Granger
|
r7168 | }; | ||
} else if (msg_type === 'clear_output') { | ||||
var cb = callbacks['clear_output']; | ||||
if (cb !== undefined) { | ||||
Jason Grout
|
r7952 | cb(content, metadata); | ||
Brian Granger
|
r7168 | } | ||
}; | ||||
}; | ||||
MinRK
|
r10366 | Kernel.prototype._handle_input_request = function (e) { | ||
var request = $.parseJSON(e.data); | ||||
var header = request.header; | ||||
var content = request.content; | ||||
var metadata = request.metadata; | ||||
var msg_type = header.msg_type; | ||||
if (msg_type !== 'input_request') { | ||||
console.log("Invalid input request!", request); | ||||
return; | ||||
} | ||||
MinRK
|
r10368 | var callbacks = this.get_callbacks_for_msg(request.parent_header.msg_id); | ||
if (callbacks !== undefined) { | ||||
var cb = callbacks[msg_type]; | ||||
if (cb !== undefined) { | ||||
cb(content, metadata); | ||||
} | ||||
}; | ||||
MinRK
|
r10366 | }; | ||
Brian E. Granger
|
r4352 | IPython.Kernel = Kernel; | ||
return IPython; | ||||
}(IPython)); | ||||