]> git.proxmox.com Git - pve-manager.git/blame - www/manager6/ceph/OSD.js
ui: ceph/osd: reintroduce Used column
[pve-manager.git] / www / manager6 / ceph / OSD.js
CommitLineData
4ea09218 1Ext.define('PVE.CephCreateOsd', {
9fccc702 2 extend: 'Proxmox.window.Edit',
4d3e918a 3 xtype: 'pveCephCreateOsd',
4ea09218
DM
4
5 subject: 'Ceph OSD',
6
7 showProgress: true,
8
35085f4a
TL
9 onlineHelp: 'pve_ceph_osds',
10
4ea09218 11 initComponent : function() {
4ea09218
DM
12 var me = this;
13
14 if (!me.nodename) {
15 throw "no node name specified";
16 }
17
d5e771ce 18 me.isCreate = true;
c474314e 19
4ea09218
DM
20 Ext.applyIf(me, {
21 url: "/nodes/" + me.nodename + "/ceph/osd",
9ad28182
DC
22 method: 'POST',
23 items: [
24 {
4d3e918a 25 xtype: 'inputpanel',
4c94e9de
DC
26 onGetValues: function(values) {
27 Object.keys(values || {}).forEach(function(name) {
28 if (values[name] === '') {
29 delete values[name];
30 }
31 });
32
33 return values;
34 },
4d3e918a
DC
35 column1: [
36 {
37 xtype: 'pveDiskSelector',
38 name: 'dev',
39 nodename: me.nodename,
40 diskType: 'unused',
41 fieldLabel: gettext('Disk'),
42 allowBlank: false
43 }
44 ],
45 column2: [
46 {
47 xtype: 'pveDiskSelector',
48 name: 'db_dev',
49 nodename: me.nodename,
50 diskType: 'journal_disks',
51 fieldLabel: gettext('DB Disk'),
52 value: '',
53 autoSelect: false,
54 allowBlank: true,
55 emptyText: 'use OSD disk',
56 listeners: {
57 change: function(field, val) {
58 me.down('field[name=db_size]').setDisabled(!val);
59 }
60 }
61 },
62 {
63 xtype: 'numberfield',
64 name: 'db_size',
65 fieldLabel: gettext('DB size') + ' (GiB)',
66 minValue: 1,
67 maxValue: 128*1024,
68 decimalPrecision: 2,
69 allowBlank: true,
70 disabled: true,
71 emptyText: gettext('Automatic')
72 }
73 ],
74 advancedColumn1: [
6c311f2c
DC
75 {
76 xtype: 'proxmoxcheckbox',
77 name: 'encrypted',
78 fieldLabel: gettext('Encrypt OSD')
79 },
80 ],
81 advancedColumn2: [
4d3e918a
DC
82 {
83 xtype: 'pveDiskSelector',
84 name: 'wal_dev',
85 nodename: me.nodename,
86 diskType: 'journal_disks',
87 fieldLabel: gettext('WAL Disk'),
88 value: '',
89 autoSelect: false,
90 allowBlank: true,
91 emptyText: 'use OSD/DB disk',
92 listeners: {
93 change: function(field, val) {
94 me.down('field[name=wal_size]').setDisabled(!val);
95 }
96 }
97 },
98 {
99 xtype: 'numberfield',
100 name: 'wal_size',
101 fieldLabel: gettext('WAL size') + ' (GiB)',
102 minValue: 0.5,
103 maxValue: 128*1024,
104 decimalPrecision: 2,
105 allowBlank: true,
106 disabled: true,
107 emptyText: gettext('Automatic')
108 }
109 ]
deca45c1
TL
110 },
111 {
112 xtype: 'displayfield',
113 padding: '5 0 0 0',
114 userCls: 'pve-hint',
115 value: 'Note: Ceph is not compatible with disks backed by a hardware ' +
116 'RAID controller. For details see ' +
117 '<a target="_blank" href="' + Proxmox.Utils.get_help_link('chapter_pveceph') + '">the reference documentation</a>.',
fddeb25b 118 }
4d3e918a
DC
119 ]
120 });
4ea09218 121
4d3e918a 122 me.callParent();
4ea09218
DM
123 }
124});
125
126Ext.define('PVE.CephRemoveOsd', {
9fccc702 127 extend: 'Proxmox.window.Edit',
4ea09218
DM
128 alias: ['widget.pveCephRemoveOsd'],
129
130 isRemove: true,
131
132 showProgress: true,
89cd5a3f
DC
133 method: 'DELETE',
134 items: [
135 {
896c0d50 136 xtype: 'proxmoxcheckbox',
89cd5a3f
DC
137 name: 'cleanup',
138 checked: true,
139 labelWidth: 130,
c56634f1 140 fieldLabel: gettext('Cleanup Disks')
89cd5a3f
DC
141 }
142 ],
4ea09218 143 initComponent : function() {
d5e771ce 144
4ea09218
DM
145 var me = this;
146
147 if (!me.nodename) {
148 throw "no node name specified";
149 }
150 if (me.osdid === undefined || me.osdid < 0) {
151 throw "no osdid specified";
152 }
153
d5e771ce 154 me.isCreate = true;
4ea09218 155
fb0e1813 156 me.title = gettext('Destroy') + ': Ceph OSD osd.' + me.osdid.toString();
c474314e 157
4ea09218 158 Ext.applyIf(me, {
d5e771ce 159 url: "/nodes/" + me.nodename + "/ceph/osd/" + me.osdid.toString()
4ea09218
DM
160 });
161
162 me.callParent();
163 }
164});
165
166Ext.define('PVE.node.CephOsdTree', {
167 extend: 'Ext.tree.Panel',
168 alias: ['widget.pveNodeCephOsdTree'],
ba93a9c6 169 onlineHelp: 'chapter_pveceph',
3982a214
DC
170
171 viewModel: {
172 data: {
173 nodename: '',
174 flags: [],
175 maxversion: '0',
176 versions: {},
177 isOsd: false,
178 downOsd: false,
179 upOsd: false,
180 inOsd: false,
181 outOsd: false,
182 osdid: '',
183 osdhost: '',
184 }
185 },
186
187 controller: {
188 xclass: 'Ext.app.ViewController',
189
190 reload: function() {
191 var me = this.getView();
192 var vm = this.getViewModel();
193 var nodename = vm.get('nodename');
194 var sm = me.getSelectionModel();
195 Proxmox.Utils.API2Request({
196 url: "/nodes/" + nodename + "/ceph/osd",
197 waitMsgTarget: me,
198 method: 'GET',
199 failure: function(response, opts) {
200 var msg = response.htmlStatus;
201 PVE.Utils.showCephInstallOrMask(me, msg, nodename,
202 function(win){
203 me.mon(win, 'cephInstallWindowClosed', this.reload);
204 }
205 );
206 },
207 success: function(response, opts) {
208 var data = response.result.data;
209 var selected = me.getSelection();
210 var name;
211 if (selected.length) {
212 name = selected[0].data.name;
213 }
214 vm.set('versions', data.versions);
215 // extract max version
216 var maxversion = vm.get('maxversion');
217 Object.values(data.versions || {}).forEach(function(version) {
218 if (PVE.Utils.compare_ceph_versions(version, maxversion) > 0) {
219 maxversion = version;
220 }
221 });
222 vm.set('maxversion', maxversion);
223 sm.deselectAll();
224 me.setRootNode(data.root);
225 me.expandAll();
226 if (name) {
227 var node = me.getRootNode().findChild('name', name, true);
228 if (node) {
229 me.setSelection([node]);
230 }
231 }
232
233 var flags = data.flags.split(',');
234 vm.set('flags', flags);
235 var noout = flags.includes('noout');
236 me.down('#nooutBtn').setText(noout ? gettext("Unset noout") : gettext("Set noout"));
237 }
238 });
239 },
240
241 osd_cmd: function(comp) {
242 var me = this;
243 var vm = this.getViewModel();
244 var cmd = comp.cmd;
245 var params = comp.params || {};
246 var osdid = vm.get('osdid');
247
248 var doRequest = function() {
249 Proxmox.Utils.API2Request({
250 url: "/nodes/" + vm.get('osdhost') + "/ceph/osd/" + osdid + '/' + cmd,
251 waitMsgTarget: me.getView(),
252 method: 'POST',
253 params: params,
254 success: () => { me.reload(); },
255 failure: function(response, opts) {
256 Ext.Msg.alert(gettext('Error'), response.htmlStatus);
257 }
258 });
259 };
260
261 if (cmd === 'scrub') {
262 Ext.MessageBox.defaultButton = params.deep === 1 ? 2 : 1;
263 Ext.Msg.show({
264 title: gettext('Confirm'),
265 icon: params.deep === 1 ? Ext.Msg.WARNING : Ext.Msg.QUESTION,
266 msg: params.deep !== 1 ?
267 Ext.String.format(gettext("Scrub OSD.{0}"), osdid) :
268 Ext.String.format(gettext("Deep Scrub OSD.{0}"), osdid) +
269 "<br>Caution: This can reduce performance while it is running.",
270 buttons: Ext.Msg.YESNO,
271 callback: function(btn) {
272 if (btn !== 'yes') {
273 return;
274 }
275 doRequest();
276 }
277 });
278 } else {
279 doRequest();
280 }
281 },
282
283 create_osd: function() {
284 var me = this;
285 var vm = this.getViewModel();
286 Ext.create('PVE.CephCreateOsd', {
287 nodename: vm.get('nodename'),
288 taskDone: () => { me.reload(); }
289 }).show();
290 },
291
292 destroy_osd: function() {
293 var me = this;
294 var vm = this.getViewModel();
295 Ext.create('PVE.CephRemoveOsd', {
296 nodename: vm.get('osdhost'),
297 osdid: vm.get('osdid'),
298 taskDone: () => { me.reload(); }
299 }).show();
300 },
301
302 set_flag: function() {
303 var me = this;
304 var vm = this.getViewModel();
305 var flags = vm.get('flags');
306 Proxmox.Utils.API2Request({
307 url: "/nodes/" + vm.get('nodename') + "/ceph/flags/noout",
308 waitMsgTarget: me.getView(),
309 method: flags.includes('noout') ? 'DELETE' : 'POST',
310 failure: function(response, opts) {
311 Ext.Msg.alert(gettext('Error'), response.htmlStatus);
312 },
313 success: () => { me.reload(); }
314 });
315 },
316
317 service_cmd: function(comp) {
318 var me = this;
319 var vm = this.getViewModel();
320 var cmd = comp.cmd || comp;
321 Proxmox.Utils.API2Request({
322 url: "/nodes/" + vm.get('osdhost') + "/ceph/" + cmd,
323 params: { service: "osd." + vm.get('osdid') },
324 waitMsgTarget: me.getView(),
325 method: 'POST',
326 success: function(response, options) {
327 var upid = response.result.data;
328 var win = Ext.create('Proxmox.window.TaskProgress', {
329 upid: upid,
330 taskDone: () => { me.reload(); }
331 });
332 win.show();
333 },
334 failure: function(response, opts) {
335 Ext.Msg.alert(gettext('Error'), response.htmlStatus);
336 }
337 });
338 },
339
340 set_selection_status: function(tp, selection) {
341 if (selection.length < 1) {
342 return;
343 }
344 var rec = selection[0];
345 var vm = this.getViewModel();
346
347 var isOsd = (rec.data.host && (rec.data.type === 'osd') && (rec.data.id >= 0));
348
349 vm.set('isOsd', isOsd);
350 vm.set('downOsd', isOsd && rec.data.status === 'down');
351 vm.set('upOsd', isOsd && rec.data.status !== 'down');
352 vm.set('inOsd', isOsd && rec.data.in);
353 vm.set('outOsd', isOsd && !rec.data.in);
354 vm.set('osdid', isOsd ? rec.data.id : undefined);
355 vm.set('osdhost', isOsd ? rec.data.host : undefined);
356 vm.notify();
357 },
358
359 render_status: function(value, metaData, rec) {
360 if (!value) {
361 return value;
362 }
363 var inout = rec.data['in'] ? 'in' : 'out';
364 var updownicon = value === 'up' ? 'good fa-arrow-circle-up' :
365 'critical fa-arrow-circle-down';
366
367 var inouticon = rec.data['in'] ? 'good fa-circle' :
368 'warning fa-circle-o';
369
370 var text = value + ' <i class="fa ' + updownicon + '"></i> / ' +
371 inout + ' <i class="fa ' + inouticon + '"></i>';
372
373 return text;
374 },
375
376 render_wal: function(value, metaData, rec) {
377 if (!value &&
378 rec.data.osdtype === 'bluestore' &&
379 rec.data.type === 'osd') {
380 return 'N/A';
381 }
382 return value;
383 },
384
385 render_version: function(value, metadata, rec) {
386 var vm = this.getViewModel();
387 var versions = vm.get('versions');
388 var icon = "";
389 var version = value || "";
390 if (value && value != vm.get('maxversion')) {
391 icon = PVE.Utils.get_ceph_icon_html('HEALTH_OLD');
392 }
393
394 if (!value && rec.data.type == 'host') {
395 version = versions[rec.data.name] || Proxmox.Utils.unknownText;
396 }
397
398 return icon + version;
399 },
400
401 render_osd_val: function(value, metaData, rec) {
402 return (rec.data.type === 'osd') ? value : '';
403 },
404
405 render_osd_size: function(value, metaData, rec) {
406 return this.render_osd_val(PVE.Utils.render_size(value), metaData, rec);
407 },
408
409 control: {
410 '#': {
411 selectionchange: 'set_selection_status'
412 }
413 },
414
415 init: function(view) {
416 var me = this;
417 var vm = this.getViewModel();
418
419 if (!view.pveSelNode.data.node) {
420 throw "no node name specified";
421 }
422
423 vm.set('nodename', view.pveSelNode.data.node);
424
425 me.callParent();
426 me.reload();
427 }
428 },
429
361aafd0
DC
430 stateful: true,
431 stateId: 'grid-ceph-osd',
3982a214
DC
432 rootVisible: false,
433 useArrows: true,
434
89cd5a3f
DC
435 columns: [
436 {
437 xtype: 'treecolumn',
438 text: 'Name',
439 dataIndex: 'name',
440 width: 150
441 },
442 {
443 text: 'Type',
444 dataIndex: 'type',
445 align: 'right',
446 width: 60
447 },
33a7e157
DC
448 {
449 text: gettext("Class"),
450 dataIndex: 'device_class',
451 align: 'right',
452 width: 40
453 },
7d406f18
DC
454 {
455 text: "OSD Type",
456 dataIndex: 'osdtype',
457 align: 'right',
458 width: 40
459 },
460 {
461 text: "Bluestore Device",
462 dataIndex: 'blfsdev',
463 align: 'right',
464 width: 40,
465 hidden: true
466 },
467 {
468 text: "DB Device",
469 dataIndex: 'dbdev',
470 align: 'right',
471 width: 40,
472 hidden: true
473 },
474 {
475 text: "WAL Device",
476 dataIndex: 'waldev',
477 align: 'right',
3982a214 478 renderer: 'render_wal',
7d406f18
DC
479 width: 40,
480 hidden: true
481 },
89cd5a3f
DC
482 {
483 text: 'Status',
484 dataIndex: 'status',
485 align: 'right',
3982a214 486 renderer: 'render_status',
89cd5a3f
DC
487 width: 80
488 },
e0297023
DC
489 {
490 text: gettext('Version'),
491 dataIndex: 'version',
3982a214 492 renderer: 'render_version'
e0297023 493 },
89cd5a3f
DC
494 {
495 text: 'weight',
496 dataIndex: 'crush_weight',
497 align: 'right',
3982a214 498 renderer: 'render_osd_val',
89cd5a3f
DC
499 width: 80
500 },
501 {
502 text: 'reweight',
503 dataIndex: 'reweight',
504 align: 'right',
3982a214 505 renderer: 'render_osd_val',
89cd5a3f
DC
506 width: 90
507 },
508 {
35c9c3d6
TL
509 header: gettext('Used'),
510 columns: [
511 {
512 text: '%',
513 dataIndex: 'percent_used',
514 align: 'right',
515 renderer: function(value, metaData, rec) {
516 if (rec.data.type !== 'osd') {
517 return '';
518 }
519 return Ext.util.Format.number(value, '0.00');
520 },
521 width: 80
522 },
523 {
524 text: gettext('Total'),
525 dataIndex: 'total_space',
526 align: 'right',
527 renderer: 'render_osd_size',
528 width: 100
529 }
530 ]
89cd5a3f
DC
531 },
532 {
533 header: gettext('Latency (ms)'),
534 columns: [
535 {
536 text: 'Apply',
537 dataIndex: 'apply_latency_ms',
538 align: 'right',
3982a214 539 renderer: 'render_osd_val',
89cd5a3f
DC
540 width: 60
541 },
542 {
543 text: 'Commit',
544 dataIndex: 'commit_latency_ms',
545 align: 'right',
3982a214 546 renderer: 'render_osd_val',
89cd5a3f
DC
547 width: 60
548 }
549 ]
550 }
551 ],
4ea09218 552
4ea09218 553
3982a214
DC
554 tbar: {
555 items: [
556 {
557 text: gettext('Reload'),
558 iconCls: 'fa fa-refresh',
559 handler: 'reload'
560 },
561 {
562 text: gettext('Create') + ': OSD',
563 handler: 'create_osd',
564 },
565 {
566 text: gettext('Set noout'),
567 itemId: 'nooutBtn',
568 handler: 'set_flag',
569 },
570 '->',
571 {
572 xtype: 'tbtext',
573 data: {
574 osd: undefined
4ea09218 575 },
3982a214
DC
576 bind: {
577 data: {
578 osd: "{osdid}"
e2924302 579 }
4ea09218 580 },
3982a214
DC
581 tpl: [
582 '<tpl if="osd">',
583 'osd.{osd}:',
584 '<tpl else>',
585 gettext('No OSD selected'),
586 '</tpl>'
587 ]
6f82e1b6 588 },
3982a214
DC
589 {
590 text: gettext('Start'),
591 iconCls: 'fa fa-play',
592 disabled: true,
593 bind: {
594 disabled: '{!downOsd}'
595 },
596 cmd: 'start',
597 handler: 'service_cmd'
598 },
599 {
600 text: gettext('Stop'),
601 iconCls: 'fa fa-stop',
602 disabled: true,
603 bind: {
604 disabled: '{!upOsd}'
605 },
606 cmd: 'stop',
607 handler: 'service_cmd'
608 },
609 {
610 text: gettext('Restart'),
611 iconCls: 'fa fa-refresh',
612 disabled: true,
613 bind: {
614 disabled: '{!upOsd}'
615 },
616 cmd: 'restart',
617 handler: 'service_cmd'
618 },
619 '-',
620 {
621 text: 'Out',
622 iconCls: 'fa fa-circle-o',
623 disabled: true,
624 bind: {
625 disabled: '{!inOsd}'
626 },
627 cmd: 'out',
628 handler: 'osd_cmd'
629 },
630 {
631 text: 'In',
632 iconCls: 'fa fa-circle',
633 disabled: true,
634 bind: {
635 disabled: '{!outOsd}'
636 },
637 cmd: 'in',
638 handler: 'osd_cmd'
639 },
640 '-',
641 {
642 text: gettext('Actions'),
643 iconCls: 'fa fa-bars',
644 disabled: true,
645 bind: {
646 disabled: '{!isOsd}'
647 },
648 menu: [
649 {
650 text: gettext('Scrub'),
651 iconCls: 'fa fa-shower',
652 cmd: 'scrub',
653 handler: 'osd_cmd'
654 },
655 {
656 text: gettext('Deep Scrub'),
657 iconCls: 'fa fa-bath',
658 cmd: 'scrub',
659 params: {
660 deep: 1,
661 },
662 handler: 'osd_cmd'
663 },
664 {
665 text: gettext('Destroy'),
666 itemId: 'remove',
667 iconCls: 'fa fa-fw fa-trash-o',
668 bind: {
669 disabled: '{!downOsd}'
670 },
671 handler: 'destroy_osd'
672 }
673 ],
4ea09218 674 }
3982a214
DC
675 ]
676 },
677
678 fields: [
679 'name', 'type', 'status', 'host', 'in', 'id' ,
680 { type: 'number', name: 'reweight' },
681 { type: 'number', name: 'percent_used' },
682 { type: 'integer', name: 'bytes_used' },
683 { type: 'integer', name: 'total_space' },
684 { type: 'integer', name: 'apply_latency_ms' },
685 { type: 'integer', name: 'commit_latency_ms' },
686 { type: 'string', name: 'device_class' },
687 { type: 'string', name: 'osdtype' },
688 { type: 'string', name: 'blfsdev' },
689 { type: 'string', name: 'dbdev' },
690 { type: 'string', name: 'waldev' },
691 { type: 'string', name: 'version', calculate: function(data) {
692 return PVE.Utils.parse_ceph_version(data);
693 } },
694 { type: 'string', name: 'iconCls', calculate: function(data) {
695 var iconMap = {
696 host: 'fa-building',
697 osd: 'fa-hdd-o',
698 root: 'fa-server',
699 };
700 return 'fa x-fa-tree ' + iconMap[data.type];
701 } },
702 { type: 'number', name: 'crush_weight' }
703 ],
4ea09218 704});