]> git.proxmox.com Git - pve-manager.git/blob - www/manager6/lxc/Network.js
ui: lxc/CreateWizard: add a 'nesting' checkbox and enable it by default
[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: 'numberfield',
127 name: 'rate',
128 fieldLabel: gettext('Rate limit') + ' (MB/s)',
129 minValue: 0,
130 maxValue: 10*1024,
131 value: cdata.rate,
132 emptyText: 'unlimited',
133 allowBlank: true,
134 },
135 {
136 xtype: 'proxmoxcheckbox',
137 fieldLabel: gettext('Firewall'),
138 name: 'firewall',
139 value: cdata.firewall,
140 },
141 ];
142
143 let dhcp4 = cdata.ip === 'dhcp';
144 if (dhcp4) {
145 cdata.ip = '';
146 cdata.gw = '';
147 }
148
149 let auto6 = cdata.ip6 === 'auto';
150 let dhcp6 = cdata.ip6 === 'dhcp';
151 if (auto6 || dhcp6) {
152 cdata.ip6 = '';
153 cdata.gw6 = '';
154 }
155
156 me.column2 = [
157 {
158 layout: {
159 type: 'hbox',
160 align: 'middle',
161 },
162 border: false,
163 margin: '0 0 5 0',
164 items: [
165 {
166 xtype: 'label',
167 text: 'IPv4:', // do not localize
168 },
169 {
170 xtype: 'radiofield',
171 boxLabel: gettext('Static'),
172 name: 'ipv4mode',
173 inputValue: 'static',
174 checked: !dhcp4,
175 margin: '0 0 0 10',
176 listeners: {
177 change: function(cb, value) {
178 me.down('field[name=ip]').setEmptyText(
179 value ? Proxmox.Utils.NoneText : "",
180 );
181 me.down('field[name=ip]').setDisabled(!value);
182 me.down('field[name=gw]').setDisabled(!value);
183 },
184 },
185 },
186 {
187 xtype: 'radiofield',
188 boxLabel: 'DHCP', // do not localize
189 name: 'ipv4mode',
190 inputValue: 'dhcp',
191 checked: dhcp4,
192 margin: '0 0 0 10',
193 },
194 ],
195 },
196 {
197 xtype: 'textfield',
198 name: 'ip',
199 vtype: 'IPCIDRAddress',
200 value: cdata.ip,
201 emptyText: dhcp4 ? '' : Proxmox.Utils.NoneText,
202 disabled: dhcp4,
203 fieldLabel: 'IPv4/CIDR', // do not localize
204 },
205 {
206 xtype: 'textfield',
207 name: 'gw',
208 value: cdata.gw,
209 vtype: 'IPAddress',
210 disabled: dhcp4,
211 fieldLabel: gettext('Gateway') + ' (IPv4)',
212 margin: '0 0 3 0', // override bottom margin to account for the menuseparator
213 },
214 {
215 xtype: 'menuseparator',
216 height: '3',
217 margin: '0',
218 },
219 {
220 layout: {
221 type: 'hbox',
222 align: 'middle',
223 },
224 border: false,
225 margin: '0 0 5 0',
226 items: [
227 {
228 xtype: 'label',
229 text: 'IPv6:', // do not localize
230 },
231 {
232 xtype: 'radiofield',
233 boxLabel: gettext('Static'),
234 name: 'ipv6mode',
235 inputValue: 'static',
236 checked: !(auto6 || dhcp6),
237 margin: '0 0 0 10',
238 listeners: {
239 change: function(cb, value) {
240 me.down('field[name=ip6]').setEmptyText(
241 value ? Proxmox.Utils.NoneText : "",
242 );
243 me.down('field[name=ip6]').setDisabled(!value);
244 me.down('field[name=gw6]').setDisabled(!value);
245 },
246 },
247 },
248 {
249 xtype: 'radiofield',
250 boxLabel: 'DHCP', // do not localize
251 name: 'ipv6mode',
252 inputValue: 'dhcp',
253 checked: dhcp6,
254 margin: '0 0 0 10',
255 },
256 {
257 xtype: 'radiofield',
258 boxLabel: 'SLAAC', // do not localize
259 name: 'ipv6mode',
260 inputValue: 'auto',
261 checked: auto6,
262 margin: '0 0 0 10',
263 },
264 ],
265 },
266 {
267 xtype: 'textfield',
268 name: 'ip6',
269 value: cdata.ip6,
270 emptyText: dhcp6 || auto6 ? '' : Proxmox.Utils.NoneText,
271 vtype: 'IP6CIDRAddress',
272 disabled: dhcp6 || auto6,
273 fieldLabel: 'IPv6/CIDR', // do not localize
274 },
275 {
276 xtype: 'textfield',
277 name: 'gw6',
278 vtype: 'IP6Address',
279 value: cdata.gw6,
280 disabled: dhcp6 || auto6,
281 fieldLabel: gettext('Gateway') + ' (IPv6)',
282 },
283 ];
284
285 me.callParent();
286 },
287 });
288
289 Ext.define('PVE.lxc.NetworkEdit', {
290 extend: 'Proxmox.window.Edit',
291
292 isAdd: true,
293
294 initComponent: function() {
295 let me = this;
296
297 if (!me.dataCache) {
298 throw "no dataCache specified";
299 }
300 if (!me.nodename) {
301 throw "no node name specified";
302 }
303
304 Ext.apply(me, {
305 subject: gettext('Network Device') + ' (veth)',
306 digest: me.dataCache.digest,
307 items: [
308 {
309 xtype: 'pveLxcNetworkInputPanel',
310 ifname: me.ifname,
311 nodename: me.nodename,
312 dataCache: me.dataCache,
313 isCreate: me.isCreate,
314 },
315 ],
316 });
317
318 me.callParent();
319 },
320 });
321
322 Ext.define('PVE.lxc.NetworkView', {
323 extend: 'Ext.grid.GridPanel',
324 alias: 'widget.pveLxcNetworkView',
325
326 onlineHelp: 'pct_container_network',
327
328 dataCache: {}, // used to store result of last load
329
330 stateful: true,
331 stateId: 'grid-lxc-network',
332
333 load: function() {
334 let me = this;
335
336 Proxmox.Utils.setErrorMask(me, true);
337
338 Proxmox.Utils.API2Request({
339 url: me.url,
340 failure: function(response, opts) {
341 Proxmox.Utils.setErrorMask(me, gettext('Error') + ': ' + response.htmlStatus);
342 },
343 success: function(response, opts) {
344 Proxmox.Utils.setErrorMask(me, false);
345 let result = Ext.decode(response.responseText);
346 me.dataCache = result.data || {};
347 let records = [];
348 for (const [key, value] of Object.entries(me.dataCache)) {
349 if (key.match(/^net\d+/)) {
350 let net = PVE.Parser.parseLxcNetwork(value);
351 net.id = key;
352 records.push(net);
353 }
354 }
355 me.store.loadData(records);
356 me.down('button[name=addButton]').setDisabled(records.length >= 32);
357 },
358 });
359 },
360
361 initComponent: function() {
362 let me = this;
363
364 let nodename = me.pveSelNode.data.node;
365 if (!nodename) {
366 throw "no node name specified";
367 }
368
369 let vmid = me.pveSelNode.data.vmid;
370 if (!vmid) {
371 throw "no VM ID specified";
372 }
373
374 let caps = Ext.state.Manager.get('GuiCap');
375
376 me.url = `/nodes/${nodename}/lxc/${vmid}/config`;
377
378 let store = new Ext.data.Store({
379 model: 'pve-lxc-network',
380 sorters: [
381 {
382 property: 'id',
383 direction: 'ASC',
384 },
385 ],
386 });
387
388 let sm = Ext.create('Ext.selection.RowModel', {});
389
390 let run_editor = function() {
391 let rec = sm.getSelection()[0];
392 if (!rec || !caps.vms['VM.Config.Network']) {
393 return false; // disable default-propagation when triggered by grid dblclick
394 }
395 Ext.create('PVE.lxc.NetworkEdit', {
396 url: me.url,
397 nodename: nodename,
398 dataCache: me.dataCache,
399 ifname: rec.data.id,
400 listeners: {
401 destroy: () => me.load(),
402 },
403 autoShow: true,
404 });
405 return undefined; // make eslint happier
406 };
407
408 Ext.apply(me, {
409 store: store,
410 selModel: sm,
411 tbar: [
412 {
413 text: gettext('Add'),
414 name: 'addButton',
415 disabled: !caps.vms['VM.Config.Network'],
416 handler: function() {
417 Ext.create('PVE.lxc.NetworkEdit', {
418 url: me.url,
419 nodename: nodename,
420 isCreate: true,
421 dataCache: me.dataCache,
422 listeners: {
423 destroy: () => me.load(),
424 },
425 autoShow: true,
426 });
427 },
428 },
429 {
430 xtype: 'proxmoxButton',
431 text: gettext('Remove'),
432 disabled: true,
433 selModel: sm,
434 enableFn: function(rec) {
435 return !!caps.vms['VM.Config.Network'];
436 },
437 confirmMsg: ({ data }) =>
438 Ext.String.format(gettext('Are you sure you want to remove entry {0}'), `'${data.id}'`),
439 handler: function(btn, e, rec) {
440 Proxmox.Utils.API2Request({
441 url: me.url,
442 waitMsgTarget: me,
443 method: 'PUT',
444 params: {
445 'delete': rec.data.id,
446 digest: me.dataCache.digest,
447 },
448 callback: () => me.load(),
449 failure: (response, opts) => Ext.Msg.alert(gettext('Error'), response.htmlStatus),
450 });
451 },
452 },
453 {
454 xtype: 'proxmoxButton',
455 text: gettext('Edit'),
456 selModel: sm,
457 disabled: true,
458 enableFn: rec => !!caps.vms['VM.Config.Network'],
459 handler: run_editor,
460 },
461 ],
462 columns: [
463 {
464 header: 'ID',
465 width: 50,
466 dataIndex: 'id',
467 },
468 {
469 header: gettext('Name'),
470 width: 80,
471 dataIndex: 'name',
472 },
473 {
474 header: gettext('Bridge'),
475 width: 80,
476 dataIndex: 'bridge',
477 },
478 {
479 header: gettext('Firewall'),
480 width: 80,
481 dataIndex: 'firewall',
482 renderer: Proxmox.Utils.format_boolean,
483 },
484 {
485 header: gettext('VLAN Tag'),
486 width: 80,
487 dataIndex: 'tag',
488 },
489 {
490 header: gettext('MAC address'),
491 width: 110,
492 dataIndex: 'hwaddr',
493 },
494 {
495 header: gettext('IP address'),
496 width: 150,
497 dataIndex: 'ip',
498 renderer: function(value, metaData, rec) {
499 if (rec.data.ip && rec.data.ip6) {
500 return rec.data.ip + "<br>" + rec.data.ip6;
501 } else if (rec.data.ip6) {
502 return rec.data.ip6;
503 } else {
504 return rec.data.ip;
505 }
506 },
507 },
508 {
509 header: gettext('Gateway'),
510 width: 150,
511 dataIndex: 'gw',
512 renderer: function(value, metaData, rec) {
513 if (rec.data.gw && rec.data.gw6) {
514 return rec.data.gw + "<br>" + rec.data.gw6;
515 } else if (rec.data.gw6) {
516 return rec.data.gw6;
517 } else {
518 return rec.data.gw;
519 }
520 },
521 },
522 ],
523 listeners: {
524 activate: me.load,
525 itemdblclick: run_editor,
526 },
527 });
528
529 me.callParent();
530 },
531 }, function() {
532 Ext.define('pve-lxc-network', {
533 extend: "Ext.data.Model",
534 proxy: { type: 'memory' },
535 fields: [
536 'id',
537 'name',
538 'hwaddr',
539 'bridge',
540 'ip',
541 'gw',
542 'ip6',
543 'gw6',
544 'tag',
545 'firewall',
546 ],
547 });
548 });
549