]> git.proxmox.com Git - pve-manager.git/blob - www/manager6/lxc/Network.js
ui: {lxc, qemu}/Config: show Tags and make them editable
[pve-manager.git] / www / manager6 / lxc / Network.js
1 Ext.define('PVE.lxc.NetworkInputPanel', {
2 extend: 'Proxmox.panel.InputPanel',
3 alias: 'widget.pveLxcNetworkInputPanel',
4
5 insideWizard: false,
6
7 onlineHelp: 'pct_container_network',
8
9 setNodename: function(nodename) {
10 let me = this;
11
12 if (!nodename || me.nodename === nodename) {
13 return;
14 }
15 me.nodename = nodename;
16
17 let bridgeSelector = me.query("[isFormField][name=bridge]")[0];
18 bridgeSelector.setNodename(nodename);
19 },
20
21 onGetValues: function(values) {
22 let me = this;
23
24 let id;
25 if (me.isCreate) {
26 id = values.id;
27 delete values.id;
28 } else {
29 id = me.ifname;
30 }
31 let newdata = {};
32 if (id) {
33 if (values.ipv6mode !== 'static') {
34 values.ip6 = values.ipv6mode;
35 }
36 if (values.ipv4mode !== 'static') {
37 values.ip = values.ipv4mode;
38 }
39 newdata[id] = PVE.Parser.printLxcNetwork(values);
40 }
41 return newdata;
42 },
43
44 initComponent: function() {
45 let me = this;
46
47 let cdata = {};
48 if (me.insideWizard) {
49 me.ifname = 'net0';
50 cdata.name = 'eth0';
51 me.dataCache = {};
52 }
53 cdata.firewall = me.insideWizard || me.isCreate;
54
55 if (!me.dataCache) {
56 throw "no dataCache specified";
57 }
58
59 if (!me.isCreate) {
60 if (!me.ifname) {
61 throw "no interface name specified";
62 }
63 if (!me.dataCache[me.ifname]) {
64 throw "no such interface '" + me.ifname + "'";
65 }
66 cdata = PVE.Parser.parseLxcNetwork(me.dataCache[me.ifname]);
67 }
68
69 for (let i = 0; i < 32; i++) {
70 let ifname = 'net' + i.toString();
71 if (me.isCreate && !me.dataCache[ifname]) {
72 me.ifname = ifname;
73 break;
74 }
75 }
76
77 me.column1 = [
78 {
79 xtype: 'hidden',
80 name: 'id',
81 value: me.ifname,
82 },
83 {
84 xtype: 'textfield',
85 name: 'name',
86 fieldLabel: gettext('Name'),
87 emptyText: '(e.g., eth0)',
88 allowBlank: false,
89 value: cdata.name,
90 validator: function(value) {
91 for (const [key, netRaw] of Object.entries(me.dataCache)) {
92 if (!key.match(/^net\d+/) || key === me.ifname) {
93 continue;
94 }
95 let net = PVE.Parser.parseLxcNetwork(netRaw);
96 if (net.name === value) {
97 return "interface name already in use";
98 }
99 }
100 return true;
101 },
102 },
103 {
104 xtype: 'textfield',
105 name: 'hwaddr',
106 fieldLabel: gettext('MAC address'),
107 vtype: 'MacAddress',
108 value: cdata.hwaddr,
109 allowBlank: true,
110 emptyText: 'auto',
111 },
112 {
113 xtype: 'PVE.form.BridgeSelector',
114 name: 'bridge',
115 nodename: me.nodename,
116 fieldLabel: gettext('Bridge'),
117 value: cdata.bridge,
118 allowBlank: false,
119 },
120 {
121 xtype: 'pveVlanField',
122 name: 'tag',
123 value: cdata.tag,
124 },
125 {
126 xtype: 'proxmoxcheckbox',
127 fieldLabel: gettext('Firewall'),
128 name: 'firewall',
129 value: cdata.firewall,
130 },
131 ];
132
133 let dhcp4 = cdata.ip === 'dhcp';
134 if (dhcp4) {
135 cdata.ip = '';
136 cdata.gw = '';
137 }
138
139 let auto6 = cdata.ip6 === 'auto';
140 let dhcp6 = cdata.ip6 === 'dhcp';
141 if (auto6 || dhcp6) {
142 cdata.ip6 = '';
143 cdata.gw6 = '';
144 }
145
146 me.column2 = [
147 {
148 layout: {
149 type: 'hbox',
150 align: 'middle',
151 },
152 border: false,
153 margin: '0 0 5 0',
154 items: [
155 {
156 xtype: 'label',
157 text: 'IPv4:', // do not localize
158 },
159 {
160 xtype: 'radiofield',
161 boxLabel: gettext('Static'),
162 name: 'ipv4mode',
163 inputValue: 'static',
164 checked: !dhcp4,
165 margin: '0 0 0 10',
166 listeners: {
167 change: function(cb, value) {
168 me.down('field[name=ip]').setEmptyText(
169 value ? Proxmox.Utils.NoneText : "",
170 );
171 me.down('field[name=ip]').setDisabled(!value);
172 me.down('field[name=gw]').setDisabled(!value);
173 },
174 },
175 },
176 {
177 xtype: 'radiofield',
178 boxLabel: 'DHCP', // do not localize
179 name: 'ipv4mode',
180 inputValue: 'dhcp',
181 checked: dhcp4,
182 margin: '0 0 0 10',
183 },
184 ],
185 },
186 {
187 xtype: 'textfield',
188 name: 'ip',
189 vtype: 'IPCIDRAddress',
190 value: cdata.ip,
191 emptyText: dhcp4 ? '' : Proxmox.Utils.NoneText,
192 disabled: dhcp4,
193 fieldLabel: 'IPv4/CIDR', // do not localize
194 },
195 {
196 xtype: 'textfield',
197 name: 'gw',
198 value: cdata.gw,
199 vtype: 'IPAddress',
200 disabled: dhcp4,
201 fieldLabel: gettext('Gateway') + ' (IPv4)',
202 margin: '0 0 3 0', // override bottom margin to account for the menuseparator
203 },
204 {
205 xtype: 'menuseparator',
206 height: '3',
207 margin: '0',
208 },
209 {
210 layout: {
211 type: 'hbox',
212 align: 'middle',
213 },
214 border: false,
215 margin: '0 0 5 0',
216 items: [
217 {
218 xtype: 'label',
219 text: 'IPv6:', // do not localize
220 },
221 {
222 xtype: 'radiofield',
223 boxLabel: gettext('Static'),
224 name: 'ipv6mode',
225 inputValue: 'static',
226 checked: !(auto6 || dhcp6),
227 margin: '0 0 0 10',
228 listeners: {
229 change: function(cb, value) {
230 me.down('field[name=ip6]').setEmptyText(
231 value ? Proxmox.Utils.NoneText : "",
232 );
233 me.down('field[name=ip6]').setDisabled(!value);
234 me.down('field[name=gw6]').setDisabled(!value);
235 },
236 },
237 },
238 {
239 xtype: 'radiofield',
240 boxLabel: 'DHCP', // do not localize
241 name: 'ipv6mode',
242 inputValue: 'dhcp',
243 checked: dhcp6,
244 margin: '0 0 0 10',
245 },
246 {
247 xtype: 'radiofield',
248 boxLabel: 'SLAAC', // do not localize
249 name: 'ipv6mode',
250 inputValue: 'auto',
251 checked: auto6,
252 margin: '0 0 0 10',
253 },
254 ],
255 },
256 {
257 xtype: 'textfield',
258 name: 'ip6',
259 value: cdata.ip6,
260 emptyText: dhcp6 || auto6 ? '' : Proxmox.Utils.NoneText,
261 vtype: 'IP6CIDRAddress',
262 disabled: dhcp6 || auto6,
263 fieldLabel: 'IPv6/CIDR', // do not localize
264 },
265 {
266 xtype: 'textfield',
267 name: 'gw6',
268 vtype: 'IP6Address',
269 value: cdata.gw6,
270 disabled: dhcp6 || auto6,
271 fieldLabel: gettext('Gateway') + ' (IPv6)',
272 },
273 ];
274
275 me.advancedColumn1 = [
276 {
277 xtype: 'proxmoxintegerfield',
278 fieldLabel: 'MTU',
279 emptyText: gettext('Same as bridge'),
280 name: 'mtu',
281 value: cdata.mtu,
282 minValue: 576,
283 maxValue: 65535,
284 },
285 ];
286
287 me.advancedColumn2 = [
288 {
289 xtype: 'numberfield',
290 name: 'rate',
291 fieldLabel: gettext('Rate limit') + ' (MB/s)',
292 minValue: 0,
293 maxValue: 10*1024,
294 value: cdata.rate,
295 emptyText: 'unlimited',
296 allowBlank: true,
297 },
298 ];
299
300 me.callParent();
301 },
302 });
303
304 Ext.define('PVE.lxc.NetworkEdit', {
305 extend: 'Proxmox.window.Edit',
306
307 isAdd: true,
308
309 initComponent: function() {
310 let me = this;
311
312 if (!me.dataCache) {
313 throw "no dataCache specified";
314 }
315 if (!me.nodename) {
316 throw "no node name specified";
317 }
318
319 Ext.apply(me, {
320 subject: gettext('Network Device') + ' (veth)',
321 digest: me.dataCache.digest,
322 items: [
323 {
324 xtype: 'pveLxcNetworkInputPanel',
325 ifname: me.ifname,
326 nodename: me.nodename,
327 dataCache: me.dataCache,
328 isCreate: me.isCreate,
329 },
330 ],
331 });
332
333 me.callParent();
334 },
335 });
336
337 Ext.define('PVE.lxc.NetworkView', {
338 extend: 'Ext.grid.GridPanel',
339 alias: 'widget.pveLxcNetworkView',
340
341 onlineHelp: 'pct_container_network',
342
343 dataCache: {}, // used to store result of last load
344
345 stateful: true,
346 stateId: 'grid-lxc-network',
347
348 load: function() {
349 let me = this;
350
351 Proxmox.Utils.setErrorMask(me, true);
352
353 Proxmox.Utils.API2Request({
354 url: me.url,
355 failure: function(response, opts) {
356 Proxmox.Utils.setErrorMask(me, gettext('Error') + ': ' + response.htmlStatus);
357 },
358 success: function(response, opts) {
359 Proxmox.Utils.setErrorMask(me, false);
360 let result = Ext.decode(response.responseText);
361 me.dataCache = result.data || {};
362 let records = [];
363 for (const [key, value] of Object.entries(me.dataCache)) {
364 if (key.match(/^net\d+/)) {
365 let net = PVE.Parser.parseLxcNetwork(value);
366 net.id = key;
367 records.push(net);
368 }
369 }
370 me.store.loadData(records);
371 me.down('button[name=addButton]').setDisabled(records.length >= 32);
372 },
373 });
374 },
375
376 initComponent: function() {
377 let me = this;
378
379 let nodename = me.pveSelNode.data.node;
380 if (!nodename) {
381 throw "no node name specified";
382 }
383
384 let vmid = me.pveSelNode.data.vmid;
385 if (!vmid) {
386 throw "no VM ID specified";
387 }
388
389 let caps = Ext.state.Manager.get('GuiCap');
390
391 me.url = `/nodes/${nodename}/lxc/${vmid}/config`;
392
393 let store = new Ext.data.Store({
394 model: 'pve-lxc-network',
395 sorters: [
396 {
397 property: 'id',
398 direction: 'ASC',
399 },
400 ],
401 });
402
403 let sm = Ext.create('Ext.selection.RowModel', {});
404
405 let run_editor = function() {
406 let rec = sm.getSelection()[0];
407 if (!rec || !caps.vms['VM.Config.Network']) {
408 return false; // disable default-propagation when triggered by grid dblclick
409 }
410 Ext.create('PVE.lxc.NetworkEdit', {
411 url: me.url,
412 nodename: nodename,
413 dataCache: me.dataCache,
414 ifname: rec.data.id,
415 listeners: {
416 destroy: () => me.load(),
417 },
418 autoShow: true,
419 });
420 return undefined; // make eslint happier
421 };
422
423 Ext.apply(me, {
424 store: store,
425 selModel: sm,
426 tbar: [
427 {
428 text: gettext('Add'),
429 name: 'addButton',
430 disabled: !caps.vms['VM.Config.Network'],
431 handler: function() {
432 Ext.create('PVE.lxc.NetworkEdit', {
433 url: me.url,
434 nodename: nodename,
435 isCreate: true,
436 dataCache: me.dataCache,
437 listeners: {
438 destroy: () => me.load(),
439 },
440 autoShow: true,
441 });
442 },
443 },
444 {
445 xtype: 'proxmoxButton',
446 text: gettext('Remove'),
447 disabled: true,
448 selModel: sm,
449 enableFn: function(rec) {
450 return !!caps.vms['VM.Config.Network'];
451 },
452 confirmMsg: ({ data }) =>
453 Ext.String.format(gettext('Are you sure you want to remove entry {0}'), `'${data.id}'`),
454 handler: function(btn, e, rec) {
455 Proxmox.Utils.API2Request({
456 url: me.url,
457 waitMsgTarget: me,
458 method: 'PUT',
459 params: {
460 'delete': rec.data.id,
461 digest: me.dataCache.digest,
462 },
463 callback: () => me.load(),
464 failure: (response, opts) => Ext.Msg.alert(gettext('Error'), response.htmlStatus),
465 });
466 },
467 },
468 {
469 xtype: 'proxmoxButton',
470 text: gettext('Edit'),
471 selModel: sm,
472 disabled: true,
473 enableFn: rec => !!caps.vms['VM.Config.Network'],
474 handler: run_editor,
475 },
476 ],
477 columns: [
478 {
479 header: 'ID',
480 width: 50,
481 dataIndex: 'id',
482 },
483 {
484 header: gettext('Name'),
485 width: 80,
486 dataIndex: 'name',
487 },
488 {
489 header: gettext('Bridge'),
490 width: 80,
491 dataIndex: 'bridge',
492 },
493 {
494 header: gettext('Firewall'),
495 width: 80,
496 dataIndex: 'firewall',
497 renderer: Proxmox.Utils.format_boolean,
498 },
499 {
500 header: gettext('VLAN Tag'),
501 width: 80,
502 dataIndex: 'tag',
503 },
504 {
505 header: gettext('MAC address'),
506 width: 110,
507 dataIndex: 'hwaddr',
508 },
509 {
510 header: gettext('IP address'),
511 width: 150,
512 dataIndex: 'ip',
513 renderer: function(value, metaData, rec) {
514 if (rec.data.ip && rec.data.ip6) {
515 return rec.data.ip + "<br>" + rec.data.ip6;
516 } else if (rec.data.ip6) {
517 return rec.data.ip6;
518 } else {
519 return rec.data.ip;
520 }
521 },
522 },
523 {
524 header: gettext('Gateway'),
525 width: 150,
526 dataIndex: 'gw',
527 renderer: function(value, metaData, rec) {
528 if (rec.data.gw && rec.data.gw6) {
529 return rec.data.gw + "<br>" + rec.data.gw6;
530 } else if (rec.data.gw6) {
531 return rec.data.gw6;
532 } else {
533 return rec.data.gw;
534 }
535 },
536 },
537 {
538 header: gettext('MTU'),
539 width: 80,
540 dataIndex: 'mtu',
541 },
542 ],
543 listeners: {
544 activate: me.load,
545 itemdblclick: run_editor,
546 },
547 });
548
549 me.callParent();
550 },
551 }, function() {
552 Ext.define('pve-lxc-network', {
553 extend: "Ext.data.Model",
554 proxy: { type: 'memory' },
555 fields: [
556 'id',
557 'name',
558 'hwaddr',
559 'bridge',
560 'ip',
561 'gw',
562 'ip6',
563 'gw6',
564 'tag',
565 'firewall',
566 'mtu',
567 ],
568 });
569 });
570