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