##// END OF EJS Templates
add Kernel.clear_callbacks_for_msg
MinRK -
Show More
@@ -1,517 +1,524 b''
1 1 //----------------------------------------------------------------------------
2 2 // Copyright (C) 2008-2011 The IPython Development Team
3 3 //
4 4 // Distributed under the terms of the BSD License. The full license is in
5 5 // the file COPYING, distributed as part of this software.
6 6 //----------------------------------------------------------------------------
7 7
8 8 //============================================================================
9 9 // Kernel
10 10 //============================================================================
11 11
12 12 /**
13 13 * @module IPython
14 14 * @namespace IPython
15 15 * @submodule Kernel
16 16 */
17 17
18 18 var IPython = (function (IPython) {
19 19
20 20 var utils = IPython.utils;
21 21
22 22 // Initialization and connection.
23 23 /**
24 24 * A Kernel Class to communicate with the Python kernel
25 25 * @Class Kernel
26 26 */
27 27 var Kernel = function (base_url) {
28 28 this.kernel_id = null;
29 29 this.shell_channel = null;
30 30 this.iopub_channel = null;
31 31 this.stdin_channel = null;
32 32 this.base_url = base_url;
33 33 this.running = false;
34 34 this.username = "username";
35 35 this.session_id = utils.uuid();
36 36 this._msg_callbacks = {};
37 37
38 38 if (typeof(WebSocket) !== 'undefined') {
39 39 this.WebSocket = WebSocket;
40 40 } else if (typeof(MozWebSocket) !== 'undefined') {
41 41 this.WebSocket = MozWebSocket;
42 42 } else {
43 43 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.');
44 44 };
45 45 this.bind_events();
46 46 };
47 47
48 48
49 49 Kernel.prototype._get_msg = function (msg_type, content) {
50 50 var msg = {
51 51 header : {
52 52 msg_id : utils.uuid(),
53 53 username : this.username,
54 54 session : this.session_id,
55 55 msg_type : msg_type
56 56 },
57 57 metadata : {},
58 58 content : content,
59 59 parent_header : {}
60 60 };
61 61 return msg;
62 62 };
63 63
64 64 Kernel.prototype.bind_events = function() {
65 65 var that = this;
66 66 $([IPython.events]).on('send_input_reply.Kernel', function(evt, data) {
67 67 that.send_input_reply(data);
68 68 });
69 69 }
70 70
71 71 /**
72 72 * Start the Python kernel
73 73 * @method start
74 74 */
75 75 Kernel.prototype.start = function (notebook_id) {
76 76 var that = this;
77 77 if (!this.running) {
78 78 var qs = $.param({notebook:notebook_id});
79 79 var url = this.base_url + '?' + qs;
80 80 $.post(url,
81 81 $.proxy(that._kernel_started,that),
82 82 'json'
83 83 );
84 84 };
85 85 };
86 86
87 87 /**
88 88 * Restart the python kernel.
89 89 *
90 90 * Emit a 'status_restarting.Kernel' event with
91 91 * the current object as parameter
92 92 *
93 93 * @method restart
94 94 */
95 95 Kernel.prototype.restart = function () {
96 96 $([IPython.events]).trigger('status_restarting.Kernel', {kernel: this});
97 97 var that = this;
98 98 if (this.running) {
99 99 this.stop_channels();
100 100 var url = this.kernel_url + "/restart";
101 101 $.post(url,
102 102 $.proxy(that._kernel_started, that),
103 103 'json'
104 104 );
105 105 };
106 106 };
107 107
108 108
109 109 Kernel.prototype._kernel_started = function (json) {
110 110 console.log("Kernel started: ", json.kernel_id);
111 111 this.running = true;
112 112 this.kernel_id = json.kernel_id;
113 113 var ws_url = json.ws_url;
114 114 if (ws_url.match(/wss?:\/\//) == null) {
115 115 // trailing 's' in https will become wss for secure web sockets
116 116 prot = location.protocol.replace('http', 'ws') + "//";
117 117 ws_url = prot + location.host + ws_url;
118 118 };
119 119 this.ws_url = ws_url;
120 120 this.kernel_url = this.base_url + "/" + this.kernel_id;
121 121 this.start_channels();
122 122 };
123 123
124 124
125 125 Kernel.prototype._websocket_closed = function(ws_url, early) {
126 126 this.stop_channels();
127 127 $([IPython.events]).trigger('websocket_closed.Kernel',
128 128 {ws_url: ws_url, kernel: this, early: early}
129 129 );
130 130 };
131 131
132 132 /**
133 133 * Start the `shell`and `iopub` channels.
134 134 * Will stop and restart them if they already exist.
135 135 *
136 136 * @method start_channels
137 137 */
138 138 Kernel.prototype.start_channels = function () {
139 139 var that = this;
140 140 this.stop_channels();
141 141 var ws_url = this.ws_url + this.kernel_url;
142 142 console.log("Starting WebSockets:", ws_url);
143 143 this.shell_channel = new this.WebSocket(ws_url + "/shell");
144 144 this.stdin_channel = new this.WebSocket(ws_url + "/stdin");
145 145 this.iopub_channel = new this.WebSocket(ws_url + "/iopub");
146 146
147 147 var already_called_onclose = false; // only alert once
148 148 var ws_closed_early = function(evt){
149 149 if (already_called_onclose){
150 150 return;
151 151 }
152 152 already_called_onclose = true;
153 153 if ( ! evt.wasClean ){
154 154 that._websocket_closed(ws_url, true);
155 155 }
156 156 };
157 157 var ws_closed_late = function(evt){
158 158 if (already_called_onclose){
159 159 return;
160 160 }
161 161 already_called_onclose = true;
162 162 if ( ! evt.wasClean ){
163 163 that._websocket_closed(ws_url, false);
164 164 }
165 165 };
166 166 var channels = [this.shell_channel, this.iopub_channel, this.stdin_channel];
167 167 for (var i=0; i < channels.length; i++) {
168 168 channels[i].onopen = $.proxy(this._ws_opened, this);
169 169 channels[i].onclose = ws_closed_early;
170 170 }
171 171 // switch from early-close to late-close message after 1s
172 172 setTimeout(function() {
173 173 for (var i=0; i < channels.length; i++) {
174 174 if (channels[i] !== null) {
175 175 channels[i].onclose = ws_closed_late;
176 176 }
177 177 }
178 178 }, 1000);
179 179 this.shell_channel.onmessage = $.proxy(this._handle_shell_reply, this);
180 180 this.iopub_channel.onmessage = $.proxy(this._handle_iopub_reply, this);
181 181 this.stdin_channel.onmessage = $.proxy(this._handle_input_request, this);
182 182 };
183 183
184 184 /**
185 185 * Handle a websocket entering the open state
186 186 * sends session and cookie authentication info as first message.
187 187 * Once all sockets are open, signal the Kernel.status_started event.
188 188 * @method _ws_opened
189 189 */
190 190 Kernel.prototype._ws_opened = function (evt) {
191 191 // send the session id so the Session object Python-side
192 192 // has the same identity
193 193 evt.target.send(this.session_id + ':' + document.cookie);
194 194
195 195 var channels = [this.shell_channel, this.iopub_channel, this.stdin_channel];
196 196 for (var i=0; i < channels.length; i++) {
197 197 // if any channel is not ready, don't trigger event.
198 198 if ( !channels[i].readyState ) return;
199 199 }
200 200 // all events ready, trigger started event.
201 201 $([IPython.events]).trigger('status_started.Kernel', {kernel: this});
202 202 };
203 203
204 204 /**
205 205 * Stop the websocket channels.
206 206 * @method stop_channels
207 207 */
208 208 Kernel.prototype.stop_channels = function () {
209 209 var channels = [this.shell_channel, this.iopub_channel, this.stdin_channel];
210 210 for (var i=0; i < channels.length; i++) {
211 211 if ( channels[i] !== null ) {
212 212 channels[i].onclose = function (evt) {};
213 213 channels[i].close();
214 214 }
215 215 };
216 216 this.shell_channel = this.iopub_channel = this.stdin_channel = null;
217 217 };
218 218
219 219 // Main public methods.
220 220
221 221 /**
222 222 * Get info on object asynchronoulsy
223 223 *
224 224 * @async
225 225 * @param objname {string}
226 226 * @param callback {dict}
227 227 * @method object_info_request
228 228 *
229 229 * @example
230 230 *
231 231 * When calling this method pass a callbacks structure of the form:
232 232 *
233 233 * callbacks = {
234 234 * 'object_info_reply': object_info_reply_callback
235 235 * }
236 236 *
237 237 * The `object_info_reply_callback` will be passed the content object of the
238 238 *
239 239 * `object_into_reply` message documented in
240 240 * [IPython dev documentation](http://ipython.org/ipython-doc/dev/development/messaging.html#object-information)
241 241 */
242 242 Kernel.prototype.object_info_request = function (objname, callbacks) {
243 243 if(typeof(objname)!=null && objname!=null)
244 244 {
245 245 var content = {
246 246 oname : objname.toString(),
247 247 detail_level : 0,
248 248 };
249 249 var msg = this._get_msg("object_info_request", content);
250 250 this.shell_channel.send(JSON.stringify(msg));
251 251 this.set_callbacks_for_msg(msg.header.msg_id, callbacks);
252 252 return msg.header.msg_id;
253 253 }
254 254 return;
255 255 }
256 256
257 257 /**
258 258 * Execute given code into kernel, and pass result to callback.
259 259 *
260 260 * TODO: document input_request in callbacks
261 261 *
262 262 * @async
263 263 * @method execute
264 264 * @param {string} code
265 265 * @param [callbacks] {Object} With the optional following keys
266 266 * @param callbacks.'execute_reply' {function}
267 267 * @param callbacks.'output' {function}
268 268 * @param callbacks.'clear_output' {function}
269 269 * @param callbacks.'set_next_input' {function}
270 270 * @param {object} [options]
271 271 * @param [options.silent=false] {Boolean}
272 272 * @param [options.user_expressions=empty_dict] {Dict}
273 273 * @param [options.user_variables=empty_list] {List od Strings}
274 274 * @param [options.allow_stdin=false] {Boolean} true|false
275 275 *
276 276 * @example
277 277 *
278 278 * The options object should contain the options for the execute call. Its default
279 279 * values are:
280 280 *
281 281 * options = {
282 282 * silent : true,
283 283 * user_variables : [],
284 284 * user_expressions : {},
285 285 * allow_stdin : false
286 286 * }
287 287 *
288 288 * When calling this method pass a callbacks structure of the form:
289 289 *
290 290 * callbacks = {
291 291 * 'execute_reply': execute_reply_callback,
292 292 * 'output': output_callback,
293 293 * 'clear_output': clear_output_callback,
294 294 * 'set_next_input': set_next_input_callback
295 295 * }
296 296 *
297 297 * The `execute_reply_callback` will be passed the content and metadata
298 298 * objects of the `execute_reply` message documented
299 299 * [here](http://ipython.org/ipython-doc/dev/development/messaging.html#execute)
300 300 *
301 301 * The `output_callback` will be passed `msg_type` ('stream','display_data','pyout','pyerr')
302 302 * of the output and the content and metadata objects of the PUB/SUB channel that contains the
303 303 * output:
304 304 *
305 305 * http://ipython.org/ipython-doc/dev/development/messaging.html#messages-on-the-pub-sub-socket
306 306 *
307 307 * The `clear_output_callback` will be passed a content object that contains
308 308 * stdout, stderr and other fields that are booleans, as well as the metadata object.
309 309 *
310 310 * The `set_next_input_callback` will be passed the text that should become the next
311 311 * input cell.
312 312 */
313 313 Kernel.prototype.execute = function (code, callbacks, options) {
314 314
315 315 var content = {
316 316 code : code,
317 317 silent : true,
318 318 store_history : false,
319 319 user_variables : [],
320 320 user_expressions : {},
321 321 allow_stdin : false
322 322 };
323 323 callbacks = callbacks || {};
324 324 if (callbacks.input_request !== undefined) {
325 325 content.allow_stdin = true;
326 326 }
327 327 $.extend(true, content, options)
328 328 $([IPython.events]).trigger('execution_request.Kernel', {kernel: this, content:content});
329 329 var msg = this._get_msg("execute_request", content);
330 330 this.shell_channel.send(JSON.stringify(msg));
331 331 this.set_callbacks_for_msg(msg.header.msg_id, callbacks);
332 332 return msg.header.msg_id;
333 333 };
334 334
335 335 /**
336 336 * When calling this method pass a callbacks structure of the form:
337 337 *
338 338 * callbacks = {
339 339 * 'complete_reply': complete_reply_callback
340 340 * }
341 341 *
342 342 * The `complete_reply_callback` will be passed the content object of the
343 343 * `complete_reply` message documented
344 344 * [here](http://ipython.org/ipython-doc/dev/development/messaging.html#complete)
345 345 *
346 346 * @method complete
347 347 * @param line {integer}
348 348 * @param cursor_pos {integer}
349 349 * @param {dict} callbacks
350 350 * @param callbacks.complete_reply {function} `complete_reply_callback`
351 351 *
352 352 */
353 353 Kernel.prototype.complete = function (line, cursor_pos, callbacks) {
354 354 callbacks = callbacks || {};
355 355 var content = {
356 356 text : '',
357 357 line : line,
358 358 block : null,
359 359 cursor_pos : cursor_pos
360 360 };
361 361 var msg = this._get_msg("complete_request", content);
362 362 this.shell_channel.send(JSON.stringify(msg));
363 363 this.set_callbacks_for_msg(msg.header.msg_id, callbacks);
364 364 return msg.header.msg_id;
365 365 };
366 366
367 367
368 368 Kernel.prototype.interrupt = function () {
369 369 if (this.running) {
370 370 $([IPython.events]).trigger('status_interrupting.Kernel', {kernel: this});
371 371 $.post(this.kernel_url + "/interrupt");
372 372 };
373 373 };
374 374
375 375
376 376 Kernel.prototype.kill = function () {
377 377 if (this.running) {
378 378 this.running = false;
379 379 var settings = {
380 380 cache : false,
381 381 type : "DELETE"
382 382 };
383 383 $.ajax(this.kernel_url, settings);
384 384 };
385 385 };
386 386
387 387 Kernel.prototype.send_input_reply = function (input) {
388 388 var content = {
389 389 value : input,
390 390 };
391 391 $([IPython.events]).trigger('input_reply.Kernel', {kernel: this, content:content});
392 392 var msg = this._get_msg("input_reply", content);
393 393 this.stdin_channel.send(JSON.stringify(msg));
394 394 return msg.header.msg_id;
395 395 };
396 396
397 397
398 398 // Reply handlers
399 399
400 400 Kernel.prototype.get_callbacks_for_msg = function (msg_id) {
401 401 var callbacks = this._msg_callbacks[msg_id];
402 402 return callbacks;
403 403 };
404 404
405 405
406 Kernel.prototype.clear_callbacks_for_msg = function (msg_id) {
407 if (this._msg_callbacks[msg_id] !== undefined ) {
408 delete this._msg_callbacks[msg_id];
409 }
410 };
411
412
406 413 Kernel.prototype.set_callbacks_for_msg = function (msg_id, callbacks) {
407 414 this._msg_callbacks[msg_id] = callbacks || {};
408 }
415 };
409 416
410 417
411 418 Kernel.prototype._handle_shell_reply = function (e) {
412 419 var reply = $.parseJSON(e.data);
413 420 $([IPython.events]).trigger('shell_reply.Kernel', {kernel: this, reply:reply});
414 421 var header = reply.header;
415 422 var content = reply.content;
416 423 var metadata = reply.metadata;
417 424 var msg_type = header.msg_type;
418 425 var callbacks = this.get_callbacks_for_msg(reply.parent_header.msg_id);
419 426 if (callbacks !== undefined) {
420 427 var cb = callbacks[msg_type];
421 428 if (cb !== undefined) {
422 429 cb(content, metadata);
423 430 }
424 431 };
425 432
426 433 if (content.payload !== undefined) {
427 434 var payload = content.payload || [];
428 435 this._handle_payload(callbacks, payload);
429 436 }
430 437 };
431 438
432 439
433 440 Kernel.prototype._handle_payload = function (callbacks, payload) {
434 441 var l = payload.length;
435 442 // Payloads are handled by triggering events because we don't want the Kernel
436 443 // to depend on the Notebook or Pager classes.
437 444 for (var i=0; i<l; i++) {
438 445 if (payload[i].source === 'page') {
439 446 var data = {'text':payload[i].text}
440 447 $([IPython.events]).trigger('open_with_text.Pager', data);
441 448 } else if (payload[i].source === 'set_next_input') {
442 449 if (callbacks.set_next_input !== undefined) {
443 450 callbacks.set_next_input(payload[i].text)
444 451 }
445 452 }
446 453 };
447 454 };
448 455
449 456
450 457 Kernel.prototype._handle_iopub_reply = function (e) {
451 458 var reply = $.parseJSON(e.data);
452 459 var content = reply.content;
453 460 var msg_type = reply.header.msg_type;
454 461 var metadata = reply.metadata;
455 462 var callbacks = this.get_callbacks_for_msg(reply.parent_header.msg_id);
456 463 if (msg_type !== 'status' && callbacks === undefined) {
457 464 // Message not from one of this notebook's cells and there are no
458 465 // callbacks to handle it.
459 466 return;
460 467 }
461 468 var output_types = ['stream','display_data','pyout','pyerr'];
462 469 if (output_types.indexOf(msg_type) >= 0) {
463 470 var cb = callbacks['output'];
464 471 if (cb !== undefined) {
465 472 cb(msg_type, content, metadata);
466 473 }
467 474 } else if (msg_type === 'status') {
468 475 if (content.execution_state === 'busy') {
469 476 $([IPython.events]).trigger('status_busy.Kernel', {kernel: this});
470 477 } else if (content.execution_state === 'idle') {
471 478 $([IPython.events]).trigger('status_idle.Kernel', {kernel: this});
472 479 } else if (content.execution_state === 'restarting') {
473 480 // autorestarting is distinct from restarting,
474 481 // in that it means the kernel died and the server is restarting it.
475 482 // status_restarting sets the notification widget,
476 483 // autorestart shows the more prominent dialog.
477 484 $([IPython.events]).trigger('status_autorestarting.Kernel', {kernel: this});
478 485 $([IPython.events]).trigger('status_restarting.Kernel', {kernel: this});
479 486 } else if (content.execution_state === 'dead') {
480 487 this.stop_channels();
481 488 $([IPython.events]).trigger('status_dead.Kernel', {kernel: this});
482 489 };
483 490 } else if (msg_type === 'clear_output') {
484 491 var cb = callbacks['clear_output'];
485 492 if (cb !== undefined) {
486 493 cb(content, metadata);
487 494 }
488 495 };
489 496 };
490 497
491 498
492 499 Kernel.prototype._handle_input_request = function (e) {
493 500 var request = $.parseJSON(e.data);
494 501 var header = request.header;
495 502 var content = request.content;
496 503 var metadata = request.metadata;
497 504 var msg_type = header.msg_type;
498 505 if (msg_type !== 'input_request') {
499 506 console.log("Invalid input request!", request);
500 507 return;
501 508 }
502 509 var callbacks = this.get_callbacks_for_msg(request.parent_header.msg_id);
503 510 if (callbacks !== undefined) {
504 511 var cb = callbacks[msg_type];
505 512 if (cb !== undefined) {
506 513 cb(content, metadata);
507 514 }
508 515 };
509 516 };
510 517
511 518
512 519 IPython.Kernel = Kernel;
513 520
514 521 return IPython;
515 522
516 523 }(IPython));
517 524
General Comments 0
You need to be logged in to leave comments. Login now