]>
git.proxmox.com Git - pve-manager.git/blob - www/manager6/Utils.js
3 // avoid errors related to Accessible Rich Internet Applications
4 // (access for people with disabilities)
5 // TODO reenable after all components are upgraded
6 Ext
. enableAria
= false ;
7 Ext
. enableAriaButtons
= false ;
8 Ext
. enableAriaPanels
= false ;
10 // avoid errors when running without development tools
11 if (! Ext
. isDefined ( Ext
. global
. console
)) {
17 console
. log ( "Starting PVE Manager" );
19 Ext
. Ajax
. defaultHeaders
= {
20 'Accept' : 'application/json'
23 Ext
. Ajax
. on ( 'beforerequest' , function ( conn
, options
) {
24 if ( PVE
. CSRFPreventionToken
) {
25 if (! options
. headers
) {
28 options
. headers
. CSRFPreventionToken
= PVE
. CSRFPreventionToken
;
32 var IPV4_OCTET
= "(?:25[0-5]|(?:[1-9]|1[0-9]|2[0-4])?[0-9])" ;
33 var IPV4_REGEXP
= "(?:(?:" + IPV4_OCTET
+ " \\ .){3}" + IPV4_OCTET
+ ")" ;
34 var IPV6_H16
= "(?:[0-9a-fA-F]{1,4})" ;
35 var IPV6_LS32
= "(?:(?:" + IPV6_H16
+ ":" + IPV6_H16
+ ")|" + IPV4_REGEXP
+ ")" ;
38 var IP4_match
= new RegExp ( "^(?:" + IPV4_REGEXP
+ ")$" );
39 var IP4_cidr_match
= new RegExp ( "^(?:" + IPV4_REGEXP
+ ")\/[1-3][0-9]?$" );
41 var IPV6_REGEXP
= "(?:" +
42 "(?:(?:" + "(?:" + IPV6_H16
+ ":){6})" + IPV6_LS32
+ ")|" +
43 "(?:(?:" + "::" + "(?:" + IPV6_H16
+ ":){5})" + IPV6_LS32
+ ")|" +
44 "(?:(?:(?:" + IPV6_H16
+ ")?::" + "(?:" + IPV6_H16
+ ":){4})" + IPV6_LS32
+ ")|" +
45 "(?:(?:(?:(?:" + IPV6_H16
+ ":){0,1}" + IPV6_H16
+ ")?::" + "(?:" + IPV6_H16
+ ":){3})" + IPV6_LS32
+ ")|" +
46 "(?:(?:(?:(?:" + IPV6_H16
+ ":){0,2}" + IPV6_H16
+ ")?::" + "(?:" + IPV6_H16
+ ":){2})" + IPV6_LS32
+ ")|" +
47 "(?:(?:(?:(?:" + IPV6_H16
+ ":){0,3}" + IPV6_H16
+ ")?::" + "(?:" + IPV6_H16
+ ":){1})" + IPV6_LS32
+ ")|" +
48 "(?:(?:(?:(?:" + IPV6_H16
+ ":){0,4}" + IPV6_H16
+ ")?::" + ")" + IPV6_LS32
+ ")|" +
49 "(?:(?:(?:(?:" + IPV6_H16
+ ":){0,5}" + IPV6_H16
+ ")?::" + ")" + IPV6_H16
+ ")|" +
50 "(?:(?:(?:(?:" + IPV6_H16
+ ":){0,7}" + IPV6_H16
+ ")?::" + ")" + ")" +
53 var IP6_match
= new RegExp ( "^(?:" + IPV6_REGEXP
+ ")$" );
54 var IP6_cidr_match
= new RegExp ( "^(?:" + IPV6_REGEXP
+ ")\/[0-9]{1,3}?$" );
55 var IP6_bracket_match
= new RegExp ( "^ \\ [(" + IPV6_REGEXP
+ ") \\ ]" );
57 var IP64_match
= new RegExp ( "^(?:" + IPV6_REGEXP
+ "|" + IPV4_REGEXP
+ ")$" );
59 Ext
. define ( 'PVE.Utils' , { statics
: {
61 // this class only contains static functions
63 toolkit
: undefined , // (extjs|touch), set inside Toolkit.js
77 'c' : gettext ( 'Community' ),
78 'b' : gettext ( 'Basic' ),
79 's' : gettext ( 'Standard' ),
80 'p' : gettext ( 'Premium' )
83 noSubKeyHtml
: 'You do not have a valid subscription for this server. Please visit <a target="_blank" href="http://www.proxmox.com/products/proxmox-ve/subscription-service-plans">www.proxmox.com</a> to get a list of available options.' ,
86 other
: gettext ( 'Other OS types' ),
87 wxp
: 'Microsoft Windows XP/2003' ,
88 w2k
: 'Microsoft Windows 2000' ,
89 w2k8
: 'Microsoft Windows Vista/2008' ,
90 win7
: 'Microsoft Windows 7/2008r2' ,
91 win8
: 'Microsoft Windows 8/2012' ,
92 l24
: 'Linux 2.4 Kernel' ,
93 l26
: 'Linux 3.X/2.6 Kernel' ,
94 solaris
: 'Solaris Kernel'
97 render_kvm_ostype : function ( value
) {
99 return gettext ( 'Other OS types' );
101 var text
= PVE
. Utils
. kvm_ostypes
[ value
];
103 return text
+ ' (' + value
+ ')' ;
108 render_hotplug_features : function ( value
) {
111 if (! value
|| ( value
=== '0' )) {
112 return gettext ( 'disabled' );
115 Ext
. each ( value
. split ( ',' ), function ( el
) {
117 fa
. push ( gettext ( 'Disk' ));
118 } else if ( el
=== 'network' ) {
119 fa
. push ( gettext ( 'Network' ));
120 } else if ( el
=== 'usb' ) {
121 fa
. push ( gettext ( 'USB' ));
122 } else if ( el
=== 'memory' ) {
123 fa
. push ( gettext ( 'Memory' ));
124 } else if ( el
=== 'cpu' ) {
125 fa
. push ( gettext ( 'CPU' ));
131 return fa
. join ( ', ' );
134 network_iface_types
: {
135 eth
: gettext ( "Network Device" ),
136 bridge
: 'Linux Bridge' ,
138 OVSBridge
: 'OVS Bridge' ,
141 OVSIntPort
: 'OVS IntPort'
144 render_network_iface_type : function ( value
) {
145 return PVE
. Utils
. network_iface_types
[ value
] ||
146 PVE
. Utils
. unknownText
;
149 render_qemu_bios : function ( value
) {
151 return PVE
. Utils
. defaultText
+ ' (SeaBIOS)' ;
152 } else if ( value
=== 'seabios' ) {
154 } else if ( value
=== 'ovmf' ) {
155 return "OVMF (UEFI)" ;
161 render_scsihw : function ( value
) {
163 return PVE
. Utils
. defaultText
+ ' (LSI 53C895A)' ;
164 } else if ( value
=== 'lsi' ) {
165 return 'LSI 53C895A' ;
166 } else if ( value
=== 'lsi53c810' ) {
168 } else if ( value
=== 'megasas' ) {
169 return 'MegaRAID SAS 8708EM2' ;
170 } else if ( value
=== 'virtio-scsi-pci' ) {
172 } else if ( value
=== 'pvscsi' ) {
173 return 'VMware PVSCSI' ;
179 // fixme: auto-generate this
180 // for now, please keep in sync with PVE::Tools::kvmkeymaps
185 'de-ch' : 'German (Swiss)' ,
186 'en-gb' : 'English (UK)' ,
187 'en-us' : 'English (USA)' ,
191 //fo: 'Faroe Islands',
193 'fr-be' : 'French (Belgium)' ,
194 'fr-ca' : 'French (Canada)' ,
195 'fr-ch' : 'French (Swiss)' ,
205 //'nl-be': 'Dutch (Belgium)',
209 'pt-br' : 'Portuguese (Brazil)' ,
218 std
: gettext ( 'Standard VGA' ),
219 vmware
: gettext ( 'VMWare compatible' ),
220 cirrus
: 'Cirrus Logic GD5446' ,
222 qxl2
: 'SPICE dual monitor' ,
223 qxl3
: 'SPICE three monitors' ,
224 qxl4
: 'SPICE four monitors' ,
225 serial0
: gettext ( 'Serial terminal' ) + ' 0' ,
226 serial1
: gettext ( 'Serial terminal' ) + ' 1' ,
227 serial2
: gettext ( 'Serial terminal' ) + ' 2' ,
228 serial3
: gettext ( 'Serial terminal' ) + ' 3'
231 render_kvm_language : function ( value
) {
233 return PVE
. Utils
. defaultText
;
235 var text
= PVE
. Utils
. kvm_keymaps
[ value
];
237 return text
+ ' (' + value
+ ')' ;
242 kvm_keymap_array : function () {
243 var data
= [[ '__default__' , PVE
. Utils
. render_kvm_language ( '' )]];
244 Ext
. Object
. each ( PVE
. Utils
. kvm_keymaps
, function ( key
, value
) {
245 data
. push ([ key
, PVE
. Utils
. render_kvm_language ( value
)]);
251 render_console_viewer : function ( value
) {
252 if (! value
|| value
=== '__default__' ) {
253 return PVE
. Utils
. defaultText
+ ' (HTML5)' ;
254 } else if ( value
=== 'vv' ) {
255 return 'SPICE (remote-viewer)' ;
256 } else if ( value
=== 'html5' ) {
257 return 'HTML5 (noVNC)' ;
268 eu
: 'Euskera (Basque)' ,
273 nb
: 'Norwegian (Bokmal)' ,
274 nn
: 'Norwegian (Nynorsk)' ,
275 fa
: 'Persian (Farsi)' ,
277 pt_BR
: 'Portuguese (Brazil)' ,
285 render_language : function ( value
) {
287 return PVE
. Utils
. defaultText
+ ' (English)' ;
289 var text
= PVE
. Utils
. language_map
[ value
];
291 return text
+ ' (' + value
+ ')' ;
296 language_array : function () {
297 var data
= [[ '__default__' , PVE
. Utils
. render_language ( '' )]];
298 Ext
. Object
. each ( PVE
. Utils
. language_map
, function ( key
, value
) {
299 data
. push ([ key
, PVE
. Utils
. render_language ( value
)]);
305 render_kvm_vga_driver : function ( value
) {
307 return PVE
. Utils
. defaultText
;
309 var text
= PVE
. Utils
. kvm_vga_drivers
[ value
];
311 return text
+ ' (' + value
+ ')' ;
316 kvm_vga_driver_array : function () {
317 var data
= [[ '__default__' , PVE
. Utils
. render_kvm_vga_driver ( '' )]];
318 Ext
. Object
. each ( PVE
. Utils
. kvm_vga_drivers
, function ( key
, value
) {
319 data
. push ([ key
, PVE
. Utils
. render_kvm_vga_driver ( value
)]);
325 render_kvm_startup : function ( value
) {
326 var startup
= PVE
. Parser
. parseStartup ( value
);
329 if ( startup
. order
=== undefined ) {
332 res
+= startup
. order
;
334 if ( startup
. up
!== undefined ) {
335 res
+= ',up=' + startup
. up
;
337 if ( startup
. down
!== undefined ) {
338 res
+= ',down=' + startup
. down
;
345 return Ext
. util
. Cookies
. get ( 'PVEAuthCookie' );
348 authClear : function () {
349 Ext
. util
. Cookies
. clear ( "PVEAuthCookie" );
352 // fixme: remove - not needed?
353 gridLineHeigh : function () {
361 extractRequestError : function ( result
, verbose
) {
362 var msg
= gettext ( 'Successful' );
364 if (! result
. success
) {
365 msg
= gettext ( "Unknown error" );
366 if ( result
. message
) {
367 msg
= result
. message
;
369 msg
+= ' (' + result
. status
+ ')' ;
372 if ( verbose
&& Ext
. isObject ( result
. errors
)) {
374 Ext
. Object
. each ( result
. errors
, function ( prop
, desc
) {
375 msg
+= "<br><b>" + Ext
. htmlEncode ( prop
) + "</b>: " +
376 Ext
. htmlEncode ( desc
);
384 extractFormActionError : function ( action
) {
386 switch ( action
. failureType
) {
387 case Ext
. form
. action
. Action
. CLIENT_INVALID
:
388 msg
= gettext ( 'Form fields may not be submitted with invalid values' );
390 case Ext
. form
. action
. Action
. CONNECT_FAILURE
:
391 msg
= gettext ( 'Connection error' );
392 var resp
= action
. response
;
393 if ( resp
. status
&& resp
. statusText
) {
394 msg
+= " " + resp
. status
+ ": " + resp
. statusText
;
397 case Ext
. form
. action
. Action
. LOAD_FAILURE
:
398 case Ext
. form
. action
. Action
. SERVER_INVALID
:
399 msg
= PVE
. Utils
. extractRequestError ( action
. result
, true );
406 API2Request : function ( reqOpts
) {
408 var newopts
= Ext
. apply ({
409 waitMsg
: gettext ( 'Please wait...' )
412 if (! newopts
. url
. match ( /^\/api2/ )) {
413 newopts
. url
= '/api2/extjs' + newopts
. url
;
415 delete newopts
. callback
;
417 var createWrapper = function ( successFn
, callbackFn
, failureFn
) {
419 success : function ( response
, options
) {
420 if ( options
. waitMsgTarget
) {
421 if ( PVE
. Utils
. toolkit
=== 'touch' ) {
422 options
. waitMsgTarget
. setMasked ( false );
424 options
. waitMsgTarget
. setLoading ( false );
427 var result
= Ext
. decode ( response
. responseText
);
428 response
. result
= result
;
429 if (! result
. success
) {
430 response
. htmlStatus
= PVE
. Utils
. extractRequestError ( result
, true );
431 Ext
. callback ( callbackFn
, options
. scope
, [ options
, false , response
]);
432 Ext
. callback ( failureFn
, options
. scope
, [ response
, options
]);
435 Ext
. callback ( callbackFn
, options
. scope
, [ options
, true , response
]);
436 Ext
. callback ( successFn
, options
. scope
, [ response
, options
]);
438 failure : function ( response
, options
) {
439 if ( options
. waitMsgTarget
) {
440 if ( PVE
. Utils
. toolkit
=== 'touch' ) {
441 options
. waitMsgTarget
. setMasked ( false );
443 options
. waitMsgTarget
. setLoading ( false );
446 response
. result
= {};
448 response
. result
= Ext
. decode ( response
. responseText
);
450 var msg
= gettext ( 'Connection error' ) + ' - server offline?' ;
451 if ( response
. aborted
) {
452 msg
= gettext ( 'Connection error' ) + ' - aborted.' ;
453 } else if ( response
. timedout
) {
454 msg
= gettext ( 'Connection error' ) + ' - Timeout.' ;
455 } else if ( response
. status
&& response
. statusText
) {
456 msg
= gettext ( 'Connection error' ) + ' ' + response
. status
+ ': ' + response
. statusText
;
458 response
. htmlStatus
= msg
;
459 Ext
. callback ( callbackFn
, options
. scope
, [ options
, false , response
]);
460 Ext
. callback ( failureFn
, options
. scope
, [ response
, options
]);
465 createWrapper ( reqOpts
. success
, reqOpts
. callback
, reqOpts
. failure
);
467 var target
= newopts
. waitMsgTarget
;
469 if ( PVE
. Utils
. toolkit
=== 'touch' ) {
470 target
. setMasked ({ xtype
: 'loadmask' , message
: newopts
. waitMsg
} );
472 // Note: ExtJS bug - this does not work when component is not rendered
473 target
. setLoading ( newopts
. waitMsg
);
476 Ext
. Ajax
. request ( newopts
);
479 assemble_field_data : function ( values
, data
) {
480 if ( Ext
. isObject ( data
)) {
481 Ext
. Object
. each ( data
, function ( name
, val
) {
482 if ( values
. hasOwnProperty ( name
)) {
483 var bucket
= values
[ name
];
484 if (! Ext
. isArray ( bucket
)) {
485 bucket
= values
[ name
] = [ bucket
];
487 if ( Ext
. isArray ( val
)) {
488 values
[ name
] = bucket
. concat ( val
);
499 checked_command : function ( orig_cmd
) {
500 PVE
. Utils
. API2Request ({
501 url
: '/nodes/localhost/subscription' ,
504 failure : function ( response
, opts
) {
505 Ext
. Msg
. alert ( gettext ( 'Error' ), response
. htmlStatus
);
507 success : function ( response
, opts
) {
508 var data
= response
. result
. data
;
510 if ( data
. status
!== 'Active' ) {
512 title
: gettext ( 'No valid subscription' ),
513 icon
: Ext
. Msg
. WARNING
,
514 msg
: PVE
. Utils
. noSubKeyHtml
,
516 callback : function ( btn
) {
531 vncproxy
: [ 'VM/CT' , gettext ( 'Console' ) ],
532 spiceproxy
: [ 'VM/CT' , gettext ( 'Console' ) + ' (Spice)' ],
533 vncshell
: [ '' , gettext ( 'Shell' ) ],
534 spiceshell
: [ '' , gettext ( 'Shell' ) + ' (Spice)' ],
535 qmsnapshot
: [ 'VM' , gettext ( 'Snapshot' ) ],
536 qmrollback
: [ 'VM' , gettext ( 'Rollback' ) ],
537 qmdelsnapshot
: [ 'VM' , gettext ( 'Delete Snapshot' ) ],
538 qmcreate
: [ 'VM' , gettext ( 'Create' ) ],
539 qmrestore
: [ 'VM' , gettext ( 'Restore' ) ],
540 qmdestroy
: [ 'VM' , gettext ( 'Destroy' ) ],
541 qmigrate
: [ 'VM' , gettext ( 'Migrate' ) ],
542 qmclone
: [ 'VM' , gettext ( 'Clone' ) ],
543 qmmove
: [ 'VM' , gettext ( 'Move disk' ) ],
544 qmtemplate
: [ 'VM' , gettext ( 'Convert to template' ) ],
545 qmstart
: [ 'VM' , gettext ( 'Start' ) ],
546 qmstop
: [ 'VM' , gettext ( 'Stop' ) ],
547 qmreset
: [ 'VM' , gettext ( 'Reset' ) ],
548 qmshutdown
: [ 'VM' , gettext ( 'Shutdown' ) ],
549 qmsuspend
: [ 'VM' , gettext ( 'Suspend' ) ],
550 qmresume
: [ 'VM' , gettext ( 'Resume' ) ],
551 qmconfig
: [ 'VM' , gettext ( 'Configure' ) ],
552 vzcreate
: [ 'CT' , gettext ( 'Create' ) ],
553 vzrestore
: [ 'CT' , gettext ( 'Restore' ) ],
554 vzdestroy
: [ 'CT' , gettext ( 'Destroy' ) ],
555 vzmigrate
: [ 'CT' , gettext ( 'Migrate' ) ],
556 vzclone
: [ 'CT' , gettext ( 'Clone' ) ],
557 vztemplate
: [ 'CT' , gettext ( 'Convert to template' ) ],
558 vzstart
: [ 'CT' , gettext ( 'Start' ) ],
559 vzstop
: [ 'CT' , gettext ( 'Stop' ) ],
560 vzmount
: [ 'CT' , gettext ( 'Mount' ) ],
561 vzumount
: [ 'CT' , gettext ( 'Unmount' ) ],
562 vzshutdown
: [ 'CT' , gettext ( 'Shutdown' ) ],
563 vzsuspend
: [ 'CT' , gettext ( 'Suspend' ) ],
564 vzresume
: [ 'CT' , gettext ( 'Resume' ) ],
565 hamigrate
: [ 'HA' , gettext ( 'Migrate' ) ],
566 hastart
: [ 'HA' , gettext ( 'Start' ) ],
567 hastop
: [ 'HA' , gettext ( 'Stop' ) ],
568 srvstart
: [ 'SRV' , gettext ( 'Start' ) ],
569 srvstop
: [ 'SRV' , gettext ( 'Stop' ) ],
570 srvrestart
: [ 'SRV' , gettext ( 'Restart' ) ],
571 srvreload
: [ 'SRV' , gettext ( 'Reload' ) ],
572 cephcreatemon
: [ 'Ceph Monitor' , gettext ( 'Create' ) ],
573 cephdestroymon
: [ 'Ceph Monitor' , gettext ( 'Destroy' ) ],
574 cephcreateosd
: [ 'Ceph OSD' , gettext ( 'Create' ) ],
575 cephdestroyosd
: [ 'Ceph OSD' , gettext ( 'Destroy' ) ],
576 imgcopy
: [ '' , gettext ( 'Copy data' ) ],
577 imgdel
: [ '' , gettext ( 'Erase data' ) ],
578 download
: [ '' , gettext ( 'Download' ) ],
579 vzdump
: [ '' , gettext ( 'Backup' ) ],
580 aptupdate
: [ '' , gettext ( 'Update package database' ) ],
581 startall
: [ '' , gettext ( 'Start all VMs and Containers' ) ],
582 stopall
: [ '' , gettext ( 'Stop all VMs and Containers' ) ],
583 migrateall
: [ '' , gettext ( 'Migrate all VMs and Containers' ) ]
586 format_task_description : function ( type
, id
) {
587 var farray
= PVE
. Utils
. task_desc_table
[ type
];
591 var prefix
= farray
[ 0 ];
592 var text
= farray
[ 1 ];
594 return prefix
+ ' ' + id
+ ' - ' + text
;
599 parse_task_upid : function ( upid
) {
602 var res
= upid
. match ( /^UPID:(\S+):([0-9A-Fa-f]{8}):([0-9A-Fa-f]{8,9}):([0-9A-Fa-f]{8}):([^:\s]+):([^:\s]*):([^:\s]+):$/ );
604 throw "unable to parse upid '" + upid
+ "'" ;
607 task
. pid
= parseInt ( res
[ 2 ], 16 );
608 task
. pstart
= parseInt ( res
[ 3 ], 16 );
609 task
. starttime
= parseInt ( res
[ 4 ], 16 );
614 task
. desc
= PVE
. Utils
. format_task_description ( task
. type
, task
. id
);
619 format_size : function ( size
) {
620 /*jslint confusion: true */
626 var kb
= size
/ 1024 ;
629 return kb
. toFixed ( 0 ) + "KiB" ;
632 var mb
= size
/ ( 1024 * 1024 );
635 return mb
. toFixed ( 0 ) + "MiB" ;
641 return gb
. toFixed ( 2 ) + "GiB" ;
646 return tb
. toFixed ( 2 ) + "TiB" ;
650 format_html_bar : function ( per
, text
) {
652 return "<div class='pve-bar-wrap'>" + text
+ "<div class='pve-bar-border'>" +
653 "<div class='pve-bar-inner' style='width:" + per
+ "%;'></div>" +
658 format_cpu_bar : function ( per1
, per2
, text
) {
660 return "<div class='pve-bar-border'>" +
661 "<div class='pve-bar-inner' style='width:" + per1
+ "%;'></div>" +
662 "<div class='pve-bar-inner2' style='width:" + per2
+ "%;'></div>" +
663 "<div class='pve-bar-text'>" + text
+ "</div>" +
667 format_large_bar : function ( per
, text
) {
670 text
= per
. toFixed ( 1 ) + "%" ;
673 return "<div class='pve-largebar-border'>" +
674 "<div class='pve-largebar-inner' style='width:" + per
+ "%;'></div>" +
675 "<div class='pve-largebar-text'>" + text
+ "</div>" +
679 format_duration_long : function ( ut
) {
681 var days
= Math
. floor ( ut
/ 86400 );
683 var hours
= Math
. floor ( ut
/ 3600 );
685 var mins
= Math
. floor ( ut
/ 60 );
688 var hours_str
= '00' + hours
. toString ();
689 hours_str
= hours_str
. substr ( hours_str
. length
- 2 );
690 var mins_str
= "00" + mins
. toString ();
691 mins_str
= mins_str
. substr ( mins_str
. length
- 2 );
692 var ut_str
= "00" + ut
. toString ();
693 ut_str
= ut_str
. substr ( ut_str
. length
- 2 );
696 var ds
= days
> 1 ? PVE
. Utils
. daysText
: PVE
. Utils
. dayText
;
697 return days
. toString () + ' ' + ds
+ ' ' +
698 hours_str
+ ':' + mins_str
+ ':' + ut_str
;
700 return hours_str
+ ':' + mins_str
+ ':' + ut_str
;
704 format_duration_short : function ( ut
) {
707 return ut
. toString () + 's' ;
712 return mins
. toFixed ( 0 ) + 'm' ;
716 var hours
= ut
/ 3600 ;
717 return hours
. toFixed ( 0 ) + 'h' ;
720 var days
= ut
/ 86400 ;
721 return days
. toFixed ( 0 ) + 'd' ;
724 yesText
: gettext ( 'Yes' ),
725 noText
: gettext ( 'No' ),
726 noneText
: gettext ( 'none' ),
727 errorText
: gettext ( 'Error' ),
728 unknownText
: gettext ( 'Unknown' ),
729 defaultText
: gettext ( 'Default' ),
730 daysText
: gettext ( 'days' ),
731 dayText
: gettext ( 'day' ),
732 runningText
: gettext ( 'running' ),
733 stoppedText
: gettext ( 'stopped' ),
734 neverText
: gettext ( 'never' ),
735 totalText
: gettext ( 'Total' ),
736 usedText
: gettext ( 'Used' ),
737 directoryText
: gettext ( 'Directory' ),
738 imagesText
: gettext ( 'Disk image' ),
739 backupFileText
: gettext ( 'VZDump backup file' ),
740 vztmplText
: gettext ( 'Container template' ),
741 isoImageText
: gettext ( 'ISO image' ),
742 containersText
: gettext ( 'Container' ),
744 format_expire : function ( date
) {
746 return PVE
. Utils
. neverText
;
748 return Ext
. Date
. format ( date
, "Y-m-d" );
751 format_storage_type : function ( value
) {
752 if ( value
=== 'dir' ) {
753 return PVE
. Utils
. directoryText
;
754 } else if ( value
=== 'nfs' ) {
756 } else if ( value
=== 'glusterfs' ) {
758 } else if ( value
=== 'lvm' ) {
760 } else if ( value
=== 'lvmthin' ) {
762 } else if ( value
=== 'iscsi' ) {
764 } else if ( value
=== 'rbd' ) {
766 } else if ( value
=== 'sheepdog' ) {
768 } else if ( value
=== 'zfs' ) {
769 return 'ZFS over iSCSI' ;
770 } else if ( value
=== 'zfspool' ) {
772 } else if ( value
=== 'iscsidirect' ) {
773 return 'iSCSIDirect' ;
774 } else if ( value
=== 'drbd' ) {
777 return PVE
. Utils
. unknownText
;
781 format_boolean_with_default : function ( value
) {
782 if ( Ext
. isDefined ( value
) && value
!== '__default__' ) {
783 return value
? PVE
. Utils
. yesText
: PVE
. Utils
. noText
;
785 return PVE
. Utils
. defaultText
;
788 format_boolean : function ( value
) {
789 return value
? PVE
. Utils
. yesText
: PVE
. Utils
. noText
;
792 format_neg_boolean : function ( value
) {
793 return ! value
? PVE
. Utils
. yesText
: PVE
. Utils
. noText
;
796 format_content_types : function ( value
) {
799 Ext
. each ( value
. split ( ',' ). sort (), function ( ct
) {
800 if ( ct
=== 'images' ) {
801 cta
. push ( PVE
. Utils
. imagesText
);
802 } else if ( ct
=== 'backup' ) {
803 cta
. push ( PVE
. Utils
. backupFileText
);
804 } else if ( ct
=== 'vztmpl' ) {
805 cta
. push ( PVE
. Utils
. vztmplText
);
806 } else if ( ct
=== 'iso' ) {
807 cta
. push ( PVE
. Utils
. isoImageText
);
808 } else if ( ct
=== 'rootdir' ) {
809 cta
. push ( PVE
. Utils
. containersText
);
813 return cta
. join ( ', ' );
816 render_storage_content : function ( value
, metaData
, record
) {
817 var data
= record
. data
;
818 if ( Ext
. isNumber ( data
. channel
) &&
819 Ext
. isNumber ( data
. id
) &&
820 Ext
. isNumber ( data
. lun
)) {
822 Ext
. String
. leftPad ( data
. channel
, 2 , '0' ) +
823 " ID " + data
. id
+ " LUN " + data
. lun
;
825 return data
. volid
. replace ( /^.*:(.*\/)?/ , '' );
828 render_serverity : function ( value
) {
829 return PVE
. Utils
. log_severity_hash
[ value
] || value
;
832 render_cpu : function ( value
, metaData
, record
, rowIndex
, colIndex
, store
) {
834 if (!( record
. data
. uptime
&& Ext
. isNumeric ( value
))) {
838 var maxcpu
= record
. data
. maxcpu
|| 1 ;
840 if (! Ext
. isNumeric ( maxcpu
) && ( maxcpu
>= 1 )) {
844 var per
= value
* 100 ;
846 return per
. toFixed ( 1 ) + '% of ' + maxcpu
. toString () + ( maxcpu
> 1 ? 'CPUs' : 'CPU' );
849 render_size : function ( value
, metaData
, record
, rowIndex
, colIndex
, store
) {
850 /*jslint confusion: true */
852 if (! Ext
. isNumeric ( value
)) {
856 return PVE
. Utils
. format_size ( value
);
859 render_timestamp : function ( value
, metaData
, record
, rowIndex
, colIndex
, store
) {
860 var servertime
= new Date ( value
* 1000 );
861 return Ext
. Date
. format ( servertime
, 'Y-m-d H:i:s' );
864 render_mem_usage : function ( value
, metaData
, record
, rowIndex
, colIndex
, store
) {
867 var maxmem
= record
. data
. maxmem
;
869 if (! record
. data
. uptime
) {
873 if (!( Ext
. isNumeric ( mem
) && maxmem
)) {
877 var per
= ( mem
* 100 ) / maxmem
;
879 return per
. toFixed ( 1 ) + '%' ;
882 render_disk_usage : function ( value
, metaData
, record
, rowIndex
, colIndex
, store
) {
885 var maxdisk
= record
. data
. maxdisk
;
887 if (!( Ext
. isNumeric ( disk
) && maxdisk
)) {
891 var per
= ( disk
* 100 ) / maxdisk
;
893 return per
. toFixed ( 1 ) + '%' ;
896 render_resource_type : function ( value
, metaData
, record
, rowIndex
, colIndex
, store
) {
898 var cls
= 'pve-itype-icon-' + value
;
900 if ( record
. data
. running
) {
901 metaData
. tdCls
= cls
+ "-running" ;
902 } else if ( record
. data
. template
) {
903 metaData
. tdCls
= cls
+ "-template" ;
905 metaData
. tdCls
= cls
;
911 render_uptime : function ( value
, metaData
, record
, rowIndex
, colIndex
, store
) {
915 if ( uptime
=== undefined ) {
923 return PVE
. Utils
. format_duration_long ( uptime
);
926 render_support_level : function ( value
, metaData
, record
) {
927 return PVE
. Utils
. support_level_hash
[ value
] || '-' ;
930 render_upid : function ( value
, metaData
, record
) {
931 var type
= record
. data
. type
;
932 var id
= record
. data
. id
;
934 return PVE
. Utils
. format_task_description ( type
, id
);
937 dialog_title : function ( subject
, create
, isAdd
) {
940 return gettext ( 'Add' ) + ': ' + subject
;
942 return gettext ( 'Create' ) + ': ' + subject
;
945 return gettext ( 'Edit' ) + ': ' + subject
;
949 windowHostname : function () {
950 return window
. location
. hostname
. replace ( IP6_bracket_match
,
951 function ( m
, addr
, offset
, original
) { return addr
; });
954 openDefaultConsoleWindow : function ( allowSpice
, vmtype
, vmid
, nodename
, vmname
) {
955 var dv
= PVE
. Utils
. defaultViewer ( allowSpice
);
956 PVE
. Utils
. openConsoleWindow ( dv
, vmtype
, vmid
, nodename
, vmname
);
959 openConsoleWindow : function ( viewer
, vmtype
, vmid
, nodename
, vmname
) {
960 // kvm, lxc, shell, upgrade
962 if ( vmid
== undefined && ( vmtype
=== 'kvm' || vmtype
=== 'lxc' )) {
963 throw "missing vmid" ;
967 throw "no nodename specified" ;
970 if ( viewer
=== 'html5' ) {
971 PVE
. Utils
. openVNCViewer ( vmtype
, vmid
, nodename
, vmname
);
972 } else if ( viewer
=== 'vv' ) {
974 var params
= { proxy
: PVE
. Utils
. windowHostname () };
975 if ( vmtype
=== 'kvm' ) {
976 url
= '/nodes/' + nodename
+ '/qemu/' + vmid
. toString () + '/spiceproxy' ;
977 PVE
. Utils
. openSpiceViewer ( url
, params
);
978 } else if ( vmtype
=== 'lxc' ) {
979 url
= '/nodes/' + nodename
+ '/lxc/' + vmid
. toString () + '/spiceproxy' ;
980 PVE
. Utils
. openSpiceViewer ( url
, params
);
981 } else if ( vmtype
=== 'shell' ) {
982 url
= '/nodes/' + nodename
+ '/spiceshell' ;
983 PVE
. Utils
. openSpiceViewer ( url
, params
);
984 } else if ( vmtype
=== 'upgrade' ) {
985 url
= '/nodes/' + nodename
+ '/spiceshell' ;
987 PVE
. Utils
. openSpiceViewer ( url
, params
);
990 throw "unknown viewer type" ;
994 defaultViewer : function ( allowSpice
) {
995 var vncdefault
= 'html5' ;
996 var dv
= PVE
. VersionInfo
. console
|| vncdefault
;
997 if ( dv
=== 'vv' && ! allowSpice
) {
1004 openVNCViewer : function ( vmtype
, vmid
, nodename
, vmname
) {
1005 var url
= Ext
. urlEncode ({
1006 console
: vmtype
, // kvm, lxc, upgrade or shell
1012 var nw
= window
. open ( "?" + url
, '_blank' , "innerWidth=745,innerheight=427" );
1016 openSpiceViewer : function ( url
, params
){
1018 var downloadWithName = function ( uri
, name
) {
1019 var link
= Ext
. DomHelper
. append ( document
. body
, {
1022 css
: 'display:none;visibility:hidden;height:0px;'
1025 // Note: we need to tell android the correct file name extension
1026 // but we do not set 'download' tag for other environments, because
1027 // It can have strange side effects (additional user prompt on firefox)
1028 var andriod
= navigator
. userAgent
. match ( /Android/i ) ? true : false ;
1030 link
. download
= name
;
1033 if ( link
. fireEvent
) {
1034 link
. fireEvent ( 'onclick' );
1036 var evt
= document
. createEvent ( "MouseEvents" );
1037 evt
. initMouseEvent ( 'click' , true , true , window
, 1 , 0 , 0 , 0 , 0 , false , false , false , false , 0 , null );
1038 link
. dispatchEvent ( evt
);
1042 PVE
. Utils
. API2Request ({
1046 failure : function ( response
, opts
){
1047 Ext
. Msg
. alert ( 'Error' , response
. htmlStatus
);
1049 success : function ( response
, opts
){
1050 var raw
= "[virt-viewer] \n " ;
1051 Ext
. Object
. each ( response
. result
. data
, function ( k
, v
) {
1052 raw
+= k
+ "=" + v
+ " \n " ;
1054 var url
= 'data:application/x-virt-viewer;charset=UTF-8,' +
1055 encodeURIComponent ( raw
);
1057 downloadWithName ( url
, "pve-spice.vv" );
1062 // comp.setLoading() is buggy in ExtJS 4.0.7, so we
1063 // use el.mask() instead
1064 setErrorMask : function ( comp
, msg
) {
1073 el
. mask ( gettext ( "Loading..." ));
1080 monStoreErrors : function ( me
, store
) {
1081 me
. mon ( store
, 'beforeload' , function ( s
, operation
, eOpts
) {
1082 if (! me
. loadCount
) {
1083 me
. loadCount
= 0 ; // make sure it is numeric
1084 PVE
. Utils
. setErrorMask ( me
, true );
1088 // only works with 'pve' proxy
1089 me
. mon ( store
. proxy
, 'afterload' , function ( proxy
, request
, success
) {
1093 PVE
. Utils
. setErrorMask ( me
, false );
1098 /*jslint nomen: true */
1099 var operation
= request
. _operation
;
1100 var error
= operation
. getError ();
1101 if ( error
. statusText
) {
1102 msg
= error
. statusText
+ ' (' + error
. status
+ ')' ;
1104 msg
= gettext ( 'Connection error' );
1106 PVE
. Utils
. setErrorMask ( me
, msg
);