Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
Avatar for KuCalc : devops.
Download
50654 views
1
// Copyright (c) IPython Development Team.
2
// Distributed under the terms of the Modified BSD License.
3
4
define([
5
'jquery',
6
'base/js/namespace',
7
'base/js/dialog',
8
'base/js/utils',
9
], function($, IPython, dialog, utils) {
10
"use strict";
11
12
var KernelSelector = function(selector, notebook) {
13
var that = this;
14
this.selector = selector;
15
this.notebook = notebook;
16
this.notebook.set_kernelselector(this);
17
this.events = notebook.events;
18
this.current_selection = null;
19
this.kernelspecs = {};
20
if (this.selector !== undefined) {
21
this.element = $(selector);
22
this.request_kernelspecs();
23
}
24
this.bind_events();
25
// Make the object globally available for user convenience & inspection
26
IPython.kernelselector = this;
27
this._finish_load = null;
28
this._loaded = false;
29
this.loaded = new Promise(function(resolve) {
30
that._finish_load = resolve;
31
});
32
33
Object.seal(this);
34
};
35
36
KernelSelector.prototype.request_kernelspecs = function() {
37
var url = utils.url_join_encode(this.notebook.base_url, 'api/kernelspecs');
38
utils.promising_ajax(url).then($.proxy(this._got_kernelspecs, this));
39
};
40
41
var _sorted_names = function(kernelspecs) {
42
// sort kernel names
43
return Object.keys(kernelspecs).sort(function (a, b) {
44
// sort by display_name
45
var da = kernelspecs[a].spec.display_name;
46
var db = kernelspecs[b].spec.display_name;
47
if (da === db) {
48
return 0;
49
} else if (da > db) {
50
return 1;
51
} else {
52
return -1;
53
}
54
});
55
};
56
57
KernelSelector.prototype._got_kernelspecs = function(data) {
58
var that = this;
59
this.kernelspecs = data.kernelspecs;
60
var change_kernel_submenu = $("#menu-change-kernel-submenu");
61
var new_notebook_submenu = $("#menu-new-notebook-submenu");
62
var keys = _sorted_names(data.kernelspecs);
63
64
keys.map(function (key) {
65
// Create the Kernel > Change kernel submenu
66
var ks = data.kernelspecs[key];
67
change_kernel_submenu.append(
68
$("<li>").attr("id", "kernel-submenu-"+ks.name).append(
69
$('<a>')
70
.attr('href', '#')
71
.click( function () {
72
that.set_kernel(ks.name);
73
})
74
.text(ks.spec.display_name)
75
)
76
);
77
// Create the File > New Notebook submenu
78
new_notebook_submenu.append(
79
$("<li>").attr("id", "new-notebook-submenu-"+ks.name).append(
80
$('<a>')
81
.attr('href', '#')
82
.click( function () {
83
that.new_notebook(ks.name);
84
})
85
.text(ks.spec.display_name)
86
)
87
);
88
89
});
90
// trigger loaded promise
91
this._loaded = true;
92
this._finish_load();
93
};
94
95
KernelSelector.prototype._spec_changed = function (event, ks) {
96
/** event handler for spec_changed */
97
var that = this;
98
99
// update selection
100
this.current_selection = ks.name;
101
102
// put the current kernel at the top of File > New Notebook
103
var cur_kernel_entry = $("#new-notebook-submenu-" + ks.name);
104
var parent = cur_kernel_entry.parent();
105
// do something only if there is more than one kernel
106
if (parent.children().length > 1) {
107
// first, sort back the submenu
108
parent.append(
109
parent.children("li[class!='divider']").sort(
110
function (a,b) {
111
var da = $("a",a).text();
112
var db = $("a",b).text();
113
if (da === db) {
114
return 0;
115
} else if (da > db) {
116
return 1;
117
} else {
118
return -1;
119
}}));
120
// then, if there is no divider yet, add one
121
if (!parent.children("li[class='divider']").length) {
122
parent.prepend($("<li>").attr("class","divider"));
123
}
124
// finally, put the current kernel at the top
125
parent.prepend(cur_kernel_entry);
126
}
127
128
// load logo
129
var logo_img = this.element.find("img.current_kernel_logo");
130
$("#kernel_indicator").find('.kernel_indicator_name').text(ks.spec.display_name);
131
if (ks.resources['logo-64x64']) {
132
logo_img.attr("src", ks.resources['logo-64x64']);
133
logo_img.show();
134
} else {
135
logo_img.hide();
136
}
137
138
// load kernel css
139
var css_url = ks.resources['kernel.css'];
140
if (css_url) {
141
$('#kernel-css').attr('href', css_url);
142
} else {
143
$('#kernel-css').attr('href', '');
144
}
145
146
// load kernel js
147
if (ks.resources['kernel.js']) {
148
require([ks.resources['kernel.js']],
149
function (kernel_mod) {
150
if (kernel_mod && kernel_mod.onload) {
151
kernel_mod.onload();
152
} else {
153
console.warn("Kernel " + ks.name + " has a kernel.js file that does not contain "+
154
"any asynchronous module definition. This is undefined behavior "+
155
"and not recommended.");
156
}
157
}, function (err) {
158
console.warn("Failed to load kernel.js from ", ks.resources['kernel.js'], err);
159
}
160
);
161
this.events.on('spec_changed.Kernel', function (evt, new_ks) {
162
if (ks.name != new_ks.name) {
163
console.warn("kernelspec %s had custom kernel.js. Forcing page reload for %s.",
164
ks.name, new_ks.name);
165
that.notebook.save_notebook().then(function () {
166
window.location.reload();
167
});
168
}
169
});
170
}
171
};
172
173
KernelSelector.prototype.set_kernel = function (selected) {
174
/** set the kernel by name, ensuring kernelspecs have been loaded, first
175
176
kernel can be just a kernel name, or a notebook kernelspec metadata
177
(name, language, display_name).
178
*/
179
var that = this;
180
if (typeof selected === 'string') {
181
selected = {
182
name: selected
183
};
184
}
185
if (this._loaded) {
186
this._set_kernel(selected);
187
} else {
188
return this.loaded.then(function () {
189
that._set_kernel(selected);
190
});
191
}
192
};
193
194
KernelSelector.prototype._set_kernel = function (selected) {
195
/** Actually set the kernel (kernelspecs have been loaded) */
196
if (selected.name === this.current_selection) {
197
// only trigger event if value changed
198
return;
199
}
200
var kernelspecs = this.kernelspecs;
201
var ks = kernelspecs[selected.name];
202
if (ks === undefined) {
203
var available = _sorted_names(kernelspecs);
204
var matches = [];
205
if (selected.language && selected.language.length > 0) {
206
available.map(function (name) {
207
if (kernelspecs[name].spec.language.toLowerCase() === selected.language.toLowerCase()) {
208
matches.push(name);
209
}
210
});
211
}
212
if (matches.length === 1) {
213
ks = kernelspecs[matches[0]];
214
console.log("No exact match found for " + selected.name +
215
", using only kernel that matches language=" + selected.language, ks);
216
this.events.trigger("spec_match_found.Kernel", {
217
selected: selected,
218
found: ks,
219
});
220
}
221
// if still undefined, trigger failure event
222
if (ks === undefined) {
223
this.events.trigger("spec_not_found.Kernel", {
224
selected: selected,
225
matches: matches,
226
available: available,
227
});
228
return;
229
}
230
}
231
if (this.notebook._session_starting &&
232
this.notebook.session.kernel.name !== ks.name) {
233
console.error("Cannot change kernel while waiting for pending session start.");
234
return;
235
}
236
this.current_selection = ks.name;
237
this.events.trigger('spec_changed.Kernel', ks);
238
};
239
240
KernelSelector.prototype._spec_not_found = function (event, data) {
241
var that = this;
242
var select = $("<select>").addClass('form-control');
243
console.warn("Kernelspec not found:", data);
244
var names;
245
if (data.matches.length > 1) {
246
names = data.matches;
247
} else {
248
names = data.available;
249
}
250
names.map(function (name) {
251
var ks = that.kernelspecs[name];
252
select.append(
253
$('<option/>').attr('value', ks.name).text(ks.spec.display_name || ks.name)
254
);
255
});
256
257
var body = $("<form>").addClass("form-inline").append(
258
$("<span>").text(
259
"I couldn't find a kernel matching " + (data.selected.display_name || data.selected.name) + "." +
260
" Please select a kernel:"
261
)
262
).append(select);
263
264
dialog.modal({
265
title : 'Kernel not found',
266
body : body,
267
buttons : {
268
'Continue without kernel' : {
269
class : 'btn-danger',
270
click : function () {
271
that.events.trigger('no_kernel.Kernel');
272
}
273
},
274
OK : {
275
class : 'btn-primary',
276
click : function () {
277
that.set_kernel(select.val());
278
}
279
}
280
}
281
});
282
};
283
284
KernelSelector.prototype.new_notebook = function (kernel_name) {
285
286
var w = window.open('', IPython._target);
287
// Create a new notebook in the same path as the current
288
// notebook's path.
289
var that = this;
290
var parent = utils.url_path_split(that.notebook.notebook_path)[0];
291
that.notebook.contents.new_untitled(parent, {type: "notebook"}).then(
292
function (data) {
293
var url = utils.url_join_encode(
294
that.notebook.base_url, 'notebooks', data.path
295
);
296
url += "?kernel_name=" + kernel_name;
297
w.location = url;
298
},
299
function(error) {
300
w.close();
301
dialog.modal({
302
title : 'Creating Notebook Failed',
303
body : "The error was: " + error.message,
304
buttons : {'OK' : {'class' : 'btn-primary'}}
305
});
306
}
307
);
308
};
309
310
KernelSelector.prototype.lock_switch = function() {
311
// should set a flag and display warning+reload if user want to
312
// re-change kernel. As UI discussion never finish
313
// making that a separate PR.
314
console.warn('switching kernel is not guaranteed to work !');
315
};
316
317
KernelSelector.prototype.bind_events = function() {
318
var that = this;
319
this.events.on('spec_changed.Kernel', $.proxy(this._spec_changed, this));
320
this.events.on('spec_not_found.Kernel', $.proxy(this._spec_not_found, this));
321
this.events.on('kernel_created.Session', function (event, data) {
322
that.set_kernel(data.kernel.name);
323
});
324
325
var logo_img = this.element.find("img.current_kernel_logo");
326
logo_img.on("load", function() {
327
logo_img.show();
328
});
329
logo_img.on("error", function() {
330
logo_img.hide();
331
});
332
};
333
334
return {'KernelSelector': KernelSelector};
335
});
336
337