Merge branch 'development' into LTS_development
# Conflicts: # lsp/minified.js
This commit is contained in:
		
						commit
						5505b68bc2
					
				
					 2 changed files with 339 additions and 610 deletions
				
			
		
							
								
								
									
										271
									
								
								lsp/minified.js
									
										
									
									
									
								
							
							
						
						
									
										271
									
								
								lsp/minified.js
									
										
									
									
									
								
							|  | @ -1,11 +1,11 @@ | |||
| var MD5=function(a){function c(a,c){var b,d,g,i,e;g=a&2147483648;i=c&2147483648;b=a&1073741824;d=c&1073741824;e=(a&1073741823)+(c&1073741823);return b&d?e^2147483648^g^i:b|d?e&1073741824?e^3221225472^g^i:e^1073741824^g^i:e^g^i}function b(a,b,d,g,i,e,f){a=c(a,c(c(b&d|~b&g,i),f));return c(a<<e|a>>>32-e,b)}function d(a,b,d,g,i,e,f){a=c(a,c(c(b&g|d&~g,i),f));return c(a<<e|a>>>32-e,b)}function f(a,b,d,g,i,e,f){a=c(a,c(c(b^d^g,i),f));return c(a<<e|a>>>32-e,b)}function k(a,b,d,g,i,e,f){a=c(a,c(c(d^(b|~g), | ||||
| i),f));return c(a<<e|a>>>32-e,b)}function m(a){var b="",c="",d;for(d=0;3>=d;d++)c=a>>>8*d&255,c="0"+c.toString(16),b+=c.substr(c.length-2,2);return b}var e=[],p,o,i,v,g,l,j,h,e=a.replace(/\r\n/g,"\n"),a="";for(p=0;p<e.length;p++)o=e.charCodeAt(p),128>o?a+=String.fromCharCode(o):(127<o&&2048>o?a+=String.fromCharCode(o>>6|192):(a+=String.fromCharCode(o>>12|224),a+=String.fromCharCode(o>>6&63|128)),a+=String.fromCharCode(o&63|128));e=a;a=e.length;p=a+8;o=16*((p-p%64)/64+1);i=Array(o-1);for(g=v=0;g<a;)p= | ||||
| (g-g%4)/4,v=8*(g%4),i[p]|=e.charCodeAt(g)<<v,g++;p=(g-g%4)/4;i[p]|=128<<8*(g%4);i[o-2]=a<<3;i[o-1]=a>>>29;e=i;g=1732584193;l=4023233417;j=2562383102;h=271733878;for(a=0;a<e.length;a+=16)p=g,o=l,i=j,v=h,g=b(g,l,j,h,e[a+0],7,3614090360),h=b(h,g,l,j,e[a+1],12,3905402710),j=b(j,h,g,l,e[a+2],17,606105819),l=b(l,j,h,g,e[a+3],22,3250441966),g=b(g,l,j,h,e[a+4],7,4118548399),h=b(h,g,l,j,e[a+5],12,1200080426),j=b(j,h,g,l,e[a+6],17,2821735955),l=b(l,j,h,g,e[a+7],22,4249261313),g=b(g,l,j,h,e[a+8],7,1770035416), | ||||
| h=b(h,g,l,j,e[a+9],12,2336552879),j=b(j,h,g,l,e[a+10],17,4294925233),l=b(l,j,h,g,e[a+11],22,2304563134),g=b(g,l,j,h,e[a+12],7,1804603682),h=b(h,g,l,j,e[a+13],12,4254626195),j=b(j,h,g,l,e[a+14],17,2792965006),l=b(l,j,h,g,e[a+15],22,1236535329),g=d(g,l,j,h,e[a+1],5,4129170786),h=d(h,g,l,j,e[a+6],9,3225465664),j=d(j,h,g,l,e[a+11],14,643717713),l=d(l,j,h,g,e[a+0],20,3921069994),g=d(g,l,j,h,e[a+5],5,3593408605),h=d(h,g,l,j,e[a+10],9,38016083),j=d(j,h,g,l,e[a+15],14,3634488961),l=d(l,j,h,g,e[a+4],20,3889429448), | ||||
| g=d(g,l,j,h,e[a+9],5,568446438),h=d(h,g,l,j,e[a+14],9,3275163606),j=d(j,h,g,l,e[a+3],14,4107603335),l=d(l,j,h,g,e[a+8],20,1163531501),g=d(g,l,j,h,e[a+13],5,2850285829),h=d(h,g,l,j,e[a+2],9,4243563512),j=d(j,h,g,l,e[a+7],14,1735328473),l=d(l,j,h,g,e[a+12],20,2368359562),g=f(g,l,j,h,e[a+5],4,4294588738),h=f(h,g,l,j,e[a+8],11,2272392833),j=f(j,h,g,l,e[a+11],16,1839030562),l=f(l,j,h,g,e[a+14],23,4259657740),g=f(g,l,j,h,e[a+1],4,2763975236),h=f(h,g,l,j,e[a+4],11,1272893353),j=f(j,h,g,l,e[a+7],16,4139469664), | ||||
| l=f(l,j,h,g,e[a+10],23,3200236656),g=f(g,l,j,h,e[a+13],4,681279174),h=f(h,g,l,j,e[a+0],11,3936430074),j=f(j,h,g,l,e[a+3],16,3572445317),l=f(l,j,h,g,e[a+6],23,76029189),g=f(g,l,j,h,e[a+9],4,3654602809),h=f(h,g,l,j,e[a+12],11,3873151461),j=f(j,h,g,l,e[a+15],16,530742520),l=f(l,j,h,g,e[a+2],23,3299628645),g=k(g,l,j,h,e[a+0],6,4096336452),h=k(h,g,l,j,e[a+7],10,1126891415),j=k(j,h,g,l,e[a+14],15,2878612391),l=k(l,j,h,g,e[a+5],21,4237533241),g=k(g,l,j,h,e[a+12],6,1700485571),h=k(h,g,l,j,e[a+3],10,2399980690), | ||||
| j=k(j,h,g,l,e[a+10],15,4293915773),l=k(l,j,h,g,e[a+1],21,2240044497),g=k(g,l,j,h,e[a+8],6,1873313359),h=k(h,g,l,j,e[a+15],10,4264355552),j=k(j,h,g,l,e[a+6],15,2734768916),l=k(l,j,h,g,e[a+13],21,1309151649),g=k(g,l,j,h,e[a+4],6,4149444226),h=k(h,g,l,j,e[a+11],10,3174756917),j=k(j,h,g,l,e[a+2],15,718787259),l=k(l,j,h,g,e[a+9],21,3951481745),g=c(g,p),l=c(l,o),j=c(j,i),h=c(h,v);return(m(g)+m(l)+m(j)+m(h)).toLowerCase()};(function(a){a.fn.stupidtable=function(){a(this).on("click","thead th",function(){a(this).stupidsort()})};a.fn.stupidsort=function(){function c(b){var c=0,d;a(b).children("td,th").each(function(){if(c==p)return d=a(this),!1;var b=a(this).attr("colspan");c+=b?Number(b):1});b="undefined"!=typeof d.data("sort-value")?d.data("sort-value"):"undefined"!=typeof d.attr("data-sort-value")?d.attr("data-sort-value"):d.text();switch(m){case "string":case "string-ins":b=String(b).toLowerCase();break;case "int":b= | ||||
| parseInt(Number(b));break;case "float":b=Number(b)}return b}var b=a(this),d=b.closest("table"),f=d.children("tbody"),k=f.children("tr"),m=b.attr("data-sort-type");if(m){var e=!0;b.hasClass("sorting-asc")&&(e=!1);var p=0;b.prevAll().each(function(){var b=a(this).attr("colspan");p+=b?Number(b):1});k.sort(function(a,b){var d=e?1:-1,a=c(a),b=c(b);return a>b?1*d:a<b?-1*d:0});f.append(k);d.find("thead th").removeClass("sorting-asc").removeClass("sorting-desc");b.addClass(e?"sorting-asc":"sorting-desc")}}})(jQuery);$(function(){UI.elements={menu:$("nav > .menu"),secondary_menu:$("nav.secondary_menu"),main:$("main"),connection:{status:$("#connection"),user_and_host:$("#user_and_host"),msg:$("#message")}};UI.buildMenu();UI.stored.getOpts();if(location.hash){var a=decodeURIComponent(location.hash).substring(1).split("@")[0].split("&");mist.user.name=a[0];a[1]&&(mist.user.host=a[1])}mist.send(function(){$(window).trigger("hashchange")},{},{timeout:5,hide:!0})}); | ||||
| var MD5=function(a){function c(a,c){var b,d,f,i,h;f=a&2147483648;i=c&2147483648;b=a&1073741824;d=c&1073741824;h=(a&1073741823)+(c&1073741823);return b&d?h^2147483648^f^i:b|d?h&1073741824?h^3221225472^f^i:h^1073741824^f^i:h^f^i}function b(a,b,d,f,i,h,g){a=c(a,c(c(b&d|~b&f,i),g));return c(a<<h|a>>>32-h,b)}function d(a,b,d,f,i,h,g){a=c(a,c(c(b&f|d&~f,i),g));return c(a<<h|a>>>32-h,b)}function e(a,b,d,f,i,h,g){a=c(a,c(c(b^d^f,i),g));return c(a<<h|a>>>32-h,b)}function j(a,b,d,f,i,h,g){a=c(a,c(c(d^(b|~f), | ||||
| i),g));return c(a<<h|a>>>32-h,b)}function m(a){var b="",c="",d;for(d=0;3>=d;d++)c=a>>>8*d&255,c="0"+c.toString(16),b+=c.substr(c.length-2,2);return b}var g=[],p,n,h,s,f,i,l,k,g=a.replace(/\r\n/g,"\n"),a="";for(p=0;p<g.length;p++)n=g.charCodeAt(p),128>n?a+=String.fromCharCode(n):(127<n&&2048>n?a+=String.fromCharCode(n>>6|192):(a+=String.fromCharCode(n>>12|224),a+=String.fromCharCode(n>>6&63|128)),a+=String.fromCharCode(n&63|128));g=a;a=g.length;p=a+8;n=16*((p-p%64)/64+1);h=Array(n-1);for(f=s=0;f<a;)p= | ||||
| (f-f%4)/4,s=8*(f%4),h[p]|=g.charCodeAt(f)<<s,f++;p=(f-f%4)/4;h[p]|=128<<8*(f%4);h[n-2]=a<<3;h[n-1]=a>>>29;g=h;f=1732584193;i=4023233417;l=2562383102;k=271733878;for(a=0;a<g.length;a+=16)p=f,n=i,h=l,s=k,f=b(f,i,l,k,g[a+0],7,3614090360),k=b(k,f,i,l,g[a+1],12,3905402710),l=b(l,k,f,i,g[a+2],17,606105819),i=b(i,l,k,f,g[a+3],22,3250441966),f=b(f,i,l,k,g[a+4],7,4118548399),k=b(k,f,i,l,g[a+5],12,1200080426),l=b(l,k,f,i,g[a+6],17,2821735955),i=b(i,l,k,f,g[a+7],22,4249261313),f=b(f,i,l,k,g[a+8],7,1770035416), | ||||
| k=b(k,f,i,l,g[a+9],12,2336552879),l=b(l,k,f,i,g[a+10],17,4294925233),i=b(i,l,k,f,g[a+11],22,2304563134),f=b(f,i,l,k,g[a+12],7,1804603682),k=b(k,f,i,l,g[a+13],12,4254626195),l=b(l,k,f,i,g[a+14],17,2792965006),i=b(i,l,k,f,g[a+15],22,1236535329),f=d(f,i,l,k,g[a+1],5,4129170786),k=d(k,f,i,l,g[a+6],9,3225465664),l=d(l,k,f,i,g[a+11],14,643717713),i=d(i,l,k,f,g[a+0],20,3921069994),f=d(f,i,l,k,g[a+5],5,3593408605),k=d(k,f,i,l,g[a+10],9,38016083),l=d(l,k,f,i,g[a+15],14,3634488961),i=d(i,l,k,f,g[a+4],20,3889429448), | ||||
| f=d(f,i,l,k,g[a+9],5,568446438),k=d(k,f,i,l,g[a+14],9,3275163606),l=d(l,k,f,i,g[a+3],14,4107603335),i=d(i,l,k,f,g[a+8],20,1163531501),f=d(f,i,l,k,g[a+13],5,2850285829),k=d(k,f,i,l,g[a+2],9,4243563512),l=d(l,k,f,i,g[a+7],14,1735328473),i=d(i,l,k,f,g[a+12],20,2368359562),f=e(f,i,l,k,g[a+5],4,4294588738),k=e(k,f,i,l,g[a+8],11,2272392833),l=e(l,k,f,i,g[a+11],16,1839030562),i=e(i,l,k,f,g[a+14],23,4259657740),f=e(f,i,l,k,g[a+1],4,2763975236),k=e(k,f,i,l,g[a+4],11,1272893353),l=e(l,k,f,i,g[a+7],16,4139469664), | ||||
| i=e(i,l,k,f,g[a+10],23,3200236656),f=e(f,i,l,k,g[a+13],4,681279174),k=e(k,f,i,l,g[a+0],11,3936430074),l=e(l,k,f,i,g[a+3],16,3572445317),i=e(i,l,k,f,g[a+6],23,76029189),f=e(f,i,l,k,g[a+9],4,3654602809),k=e(k,f,i,l,g[a+12],11,3873151461),l=e(l,k,f,i,g[a+15],16,530742520),i=e(i,l,k,f,g[a+2],23,3299628645),f=j(f,i,l,k,g[a+0],6,4096336452),k=j(k,f,i,l,g[a+7],10,1126891415),l=j(l,k,f,i,g[a+14],15,2878612391),i=j(i,l,k,f,g[a+5],21,4237533241),f=j(f,i,l,k,g[a+12],6,1700485571),k=j(k,f,i,l,g[a+3],10,2399980690), | ||||
| l=j(l,k,f,i,g[a+10],15,4293915773),i=j(i,l,k,f,g[a+1],21,2240044497),f=j(f,i,l,k,g[a+8],6,1873313359),k=j(k,f,i,l,g[a+15],10,4264355552),l=j(l,k,f,i,g[a+6],15,2734768916),i=j(i,l,k,f,g[a+13],21,1309151649),f=j(f,i,l,k,g[a+4],6,4149444226),k=j(k,f,i,l,g[a+11],10,3174756917),l=j(l,k,f,i,g[a+2],15,718787259),i=j(i,l,k,f,g[a+9],21,3951481745),f=c(f,p),i=c(i,n),l=c(l,h),k=c(k,s);return(m(f)+m(i)+m(l)+m(k)).toLowerCase()};(function(a){a.fn.stupidtable=function(){a(this).on("click","thead th",function(){a(this).stupidsort()})};a.fn.stupidsort=function(){function c(c){var b=0,d;a(c).children("td,th").each(function(){if(b==p)return d=a(this),!1;var c=a(this).attr("colspan");b+=c?Number(c):1});c="undefined"!=typeof d.data("sort-value")?d.data("sort-value"):"undefined"!=typeof d.attr("data-sort-value")?d.attr("data-sort-value"):d.text();switch(m){case "string":case "string-ins":c=String(c).toLowerCase();break;case "int":c= | ||||
| parseInt(Number(c));break;case "float":c=Number(c)}return c}var b=a(this),d=b.closest("table"),e=d.children("tbody"),j=e.children("tr"),m=b.attr("data-sort-type");if(m){var g=!0;b.hasClass("sorting-asc")&&(g=!1);var p=0;b.prevAll().each(function(){var c=a(this).attr("colspan");p+=c?Number(c):1});j.sort(function(a,b){var d=g?1:-1,a=c(a),b=c(b);return a>b?1*d:a<b?-1*d:0});e.append(j);d.find("thead th").removeClass("sorting-asc").removeClass("sorting-desc");b.addClass(g?"sorting-asc":"sorting-desc")}}})(jQuery);$(function(){UI.elements={menu:$("nav > .menu"),secondary_menu:$("nav.secondary_menu"),main:$("main"),connection:{status:$("#connection"),user_and_host:$("#user_and_host"),msg:$("#message")}};UI.buildMenu();UI.stored.getOpts();if(location.hash){var a=decodeURIComponent(location.hash).substring(1).split("@")[0].split("&");mist.user.name=a[0];a[1]&&(mist.user.host=a[1])}mist.send(function(){$(window).trigger("hashchange")},{},{timeout:5,hide:!0})}); | ||||
| $(window).on("hashchange",function(){var a=decodeURIComponent(location.hash).substring(1).split("@");a[1]||(a[1]="");a=a[1].split("&");""==a[0]&&(a[0]="Overview");UI.showTab(a[0],a[1])}); | ||||
| var UI={debug:!1,elements:{},stored:{getOpts:function(){var a=localStorage.stored;a&&(a=JSON.parse(a));$.extend(!0,this.vars,a);return this.vars},saveOpt:function(a,c){this.vars[a]=c;localStorage.stored=JSON.stringify(this.vars);return this.vars},vars:{helpme:!0}},interval:{clear:function(){"undefined"!=typeof this.opts&&(clearInterval(this.opts.id),delete this.opts)},set:function(a,c){this.opts&&log("[interval]","Set called on interval, but an interval is already active.");this.opts={delay:c,callback:a}; | ||||
| this.opts.id=setInterval(a,c)}},returnTab:["Overview"],countrylist:{AF:"Afghanistan",AX:"Åland Islands",AL:"Albania",DZ:"Algeria",AS:"American Samoa",AD:"Andorra",AO:"Angola",AI:"Anguilla",AQ:"Antarctica",AG:"Antigua and Barbuda",AR:"Argentina",AM:"Armenia",AW:"Aruba",AU:"Australia",AT:"Austria",AZ:"Azerbaijan",BS:"Bahamas",BH:"Bahrain",BD:"Bangladesh",BB:"Barbados",BY:"Belarus",BE:"Belgium",BZ:"Belize",BJ:"Benin",BM:"Bermuda",BT:"Bhutan",BO:"Bolivia, Plurinational State of",BQ:"Bonaire, Sint Eustatius and Saba", | ||||
|  | @ -18,144 +18,137 @@ RO:"Romania",RU:"Russian Federation",RW:"Rwanda",BL:"Saint Barthélemy",SH | |||
| ZA:"South Africa",GS:"South Georgia and the South Sandwich Islands",SS:"South Sudan",ES:"Spain",LK:"Sri Lanka",SD:"Sudan",SR:"Suriname",SJ:"Svalbard and Jan Mayen",SZ:"Swaziland",SE:"Sweden",CH:"Switzerland",SY:"Syrian Arab Republic",TW:"Taiwan, Province of China",TJ:"Tajikistan",TZ:"Tanzania, United Republic of",TH:"Thailand",TL:"Timor-Leste",TG:"Togo",TK:"Tokelau",TO:"Tonga",TT:"Trinidad and Tobago",TN:"Tunisia",TR:"Turkey",TM:"Turkmenistan",TC:"Turks and Caicos Islands",TV:"Tuvalu",UG:"Uganda", | ||||
| UA:"Ukraine",AE:"United Arab Emirates",GB:"United Kingdom",US:"United States",UM:"United States Minor Outlying Islands",UY:"Uruguay",UZ:"Uzbekistan",VU:"Vanuatu",VE:"Venezuela, Bolivarian Republic of",VN:"Viet Nam",VG:"Virgin Islands, British",VI:"Virgin Islands, U.S.",WF:"Wallis and Futuna",EH:"Western Sahara",YE:"Yemen",ZM:"Zambia",ZW:"Zimbabwe"},tooltip:{show:function(a,c){$tooltip=this.element;$.contains(document.body,$tooltip[0])||$("body").append($tooltip);$tooltip.html(c);clearTimeout(this.hiding); | ||||
| delete this.hiding;var b=$(document).height()-$tooltip.outerHeight(),d=$(document).width()-$tooltip.outerWidth();$tooltip.css("left",Math.min(a.pageX+10,d-10));$tooltip.css("top",Math.min(a.pageY+25,b-10));$tooltip.show().addClass("show")},hide:function(){$tooltip=this.element;$tooltip.removeClass("show");this.hiding=setTimeout(function(){$tooltip.hide()},500)},element:$("<div>").attr("id","tooltip")},popup:{element:null,show:function(a){this.element=$("<div>").attr("id","popup").append($("<button>").text("Close").addClass("close").click(function(){UI.popup.element.fadeOut("fast", | ||||
| function(){UI.popup.element.remove();UI.popup.element=null})})).append(a);$("body").append(this.element)}},menu:[{Overview:{},Protocols:{},Streams:{},Preview:{},Limits:{LTSonly:!0},Triggers:{LTSonly:!1},Logs:{},Statistics:{},"Server Stats":{}},{Disconnect:{classes:["red"]}},{Guides:{link:"http://mistserver.org/wiki/Category:Guides"},Tools:{submenu:{"Release notes":{link:"http://mistserver.org/wiki/Mistserver_Changelog"},"Mist Shop":{link:"http://mistserver.org/products"},"Email for Help":{},"Terms & Conditions":{link:"http://mistserver.org/wiki/Mistserver_license"}}}}], | ||||
| buildMenu:function(){function a(a,b){var c=$("<a>").addClass("button");c.html($("<span>").addClass("plain").text(a)).append($("<span>").addClass("highlighted").text(a));for(var d in b.classes)c.addClass(b.classes[d]);"LTSonly"in b&&c.addClass("LTSonly");"link"in b?c.attr("href",b.link).attr("target","_blank"):"submenu"in b||c.click(function(b){UI.navto(a);b.stopPropagation()});return c}var c=UI.elements.menu,b;for(b in UI.menu){0<b&&c.append($("<br>"));for(var d in UI.menu[b]){var f=UI.menu[b][d], | ||||
| k=a(d,f);c.append(k);if("submenu"in f){var m=$("<span>").addClass("submenu");k.addClass("arrowdown").append(m);for(var e in f.submenu)m.append(a(e,f.submenu[e]))}}}b=$("<div>").attr("id","ih_button").text("?").click(function(){$("body").toggleClass("helpme");UI.stored.saveOpt("helpme",$("body").hasClass("helpme"))}).attr("title","Click to toggle the display of integrated help");UI.stored.getOpts().helpme&&$("body").addClass("helpme");c.after(b).after($("<div>").addClass("separator"))},buildUI:function(a){var c= | ||||
| $("<div>").addClass("input_container"),b;for(b in a){var d=a[b];if(d instanceof jQuery)c.append(d);else if("help"==d.type){var f=$("<span>").addClass("text_container").append($("<span>").addClass("description").append(d.help));c.append(f);if("classes"in d)for(var k in d.classes)f.addClass(d.classes[k])}else if("text"==d.type)c.append($("<span>").addClass("text_container").append($("<span>").addClass("text").append(d.text)));else if("custom"==d.type)c.append(d.custom);else if("buttons"==d.type)for(k in f= | ||||
| $("<span>").addClass("button_container"),"css"in d&&f.css(d.css),c.append(f),d.buttons){var m=d.buttons[k],e=$("<button>").text(m.label).data("opts",m);"css"in m&&e.css(m.css);f.append(e);switch(m.type){case "cancel":e.addClass("cancel").click(m["function"]);break;case "save":e.addClass("save").click(function(){var a=$(this).closest(".input_container"),b=!1;a.find(".hasValidate").each(function(){if(b=$(this).data("validate")(this,!0))return!1});b||(a.find(".isSetting").each(function(){var a=$(this).getval(), | ||||
| b=$(this).data("pointer");if(""==a)if("default"in $(this).data("opts"))a=$(this).data("opts")["default"];else return delete b.main[b.index],!0;b.main[b.index]=a}),(a=$(this).data("opts")["function"])&&a(this))});break;default:e.click(m["function"])}}else{m=$("<label>").addClass("UIelement");c.append(m);"css"in d&&m.css(d.css);m.append($("<span>").addClass("label").html(d.label+":"));e=$("<span>").addClass("field_container");m.append(e);switch(d.type){case "password":f=$("<input>").attr("type","password"); | ||||
| break;case "int":f=$("<input>").attr("type","number");"min"in d&&f.attr("min",d.min);"max"in d&&f.attr("max",d.min);"validate"in d?d.validate.push("int"):d.validate=["int"];break;case "span":f=$("<span>");break;case "debug":d.select=[["","Default"],[0,"0 - All debugging messages disabled"],[1,"1 - Messages about failed operations"],[2,"2 - Previous level, and error messages"],[3,"3 - Previous level, and warning messages"],[4,"4 - Previous level, and status messages for development"],[5,"5 - Previous level, and more status messages for development"], | ||||
| [6,"6 - Previous level, and verbose debugging messages"],[7,"7 - Previous level, and very verbose debugging messages"],[8,"8 - Report everything in extreme detail"],[9,"9 - Report everything in insane detail"],[10,"10 - All messages enabled"]];case "select":f=$("<select>");for(k in d.select){var p=$("<option>");"string"==typeof d.select[k]?p.text(d.select[k]):p.val(d.select[k][0]).text(d.select[k][1]);f.append(p)}break;case "textarea":f=$("<textarea>").on("keydown",function(a){a.stopPropagation()}); | ||||
| break;case "checkbox":f=$("<input>").attr("type","checkbox");break;case "hidden":f=$("<input>").attr("type","hidden");m.hide();break;case "email":f=$("<input>").attr("type","email").attr("autocomplete","on").attr("required","");break;case "browse":f=$("<input>").attr("type","text");"filetypes"in d&&f.data("filetypes",d.filetypes);break;case "geolimited":case "hostlimited":f=$("<input>").attr("type","hidden");break;case "radioselect":f=$("<div>").addClass("radioselect");for(b in d.radioselect){var o= | ||||
| $("<input>").attr("type","radio").val(d.radioselect[b][0]).attr("name",d.label);("LTSonly"in d&&!mist.data.LTS||d.readonly)&&o.prop("disabled",!0);p=$("<label>").append(o).append($("<span>").html(d.radioselect[b][1]));f.append(p);if(2<d.radioselect[b].length)for(k in o=$("<select>").change(function(){$(this).parent().find("input[type=radio]:enabled").prop("checked","true")}),p.append(o),("LTSonly"in d&&!mist.data.LTS||d.readonly)&&o.prop("disabled",!0),d.radioselect[b][2])p=$("<option>"),o.append(p), | ||||
| d.radioselect[b][2][k]instanceof Array?p.val(d.radioselect[b][2][k][0]).html(d.radioselect[b][2][k][1]):p.html(d.radioselect[b][2][k])}break;case "checklist":f=$("<div>").addClass("checkcontainer");$controls=$("<div>").addClass("controls");$checklist=$("<div>").addClass("checklist");f.append($controls).append($checklist);$controls.append($("<label>").text("All").prepend($("<input>").attr("type","checkbox").click(function(){$(this).is(":checked")?$(this).closest(".checkcontainer").find("input[type=checkbox]").prop("checked", | ||||
| !0):$(this).closest(".checkcontainer").find("input[type=checkbox]").prop("checked",!1)})));for(b in d.checklist)"string"==typeof d.checklist[b]&&(d.checklist[b]=[d.checklist[b],d.checklist[b]]),$checklist.append($("<label>").text(d.checklist[b][1]).prepend($("<input>").attr("type","checkbox").attr("name",d.checklist[b][0])));break;default:f=$("<input>").attr("type","text")}f.addClass("field").data("opts",d);"pointer"in d&&f.attr("name",d.pointer.index);e.append(f);if("classes"in d)for(k in d.classes)f.addClass(d.classes[k]); | ||||
| "placeholder"in d&&f.attr("placeholder",d.placeholder);"default"in d&&f.attr("placeholder",d["default"]);"unit"in d&&e.append($("<span>").addClass("unit").html(d.unit));"readonly"in d&&(f.attr("readonly","readonly"),f.click(function(){$(this).select()}));"qrcode"in d&&e.append($("<span>").addClass("unit").html($("<button>").text("QR").click(function(){var a=String($(this).closest(".field_container").find(".field").getval()),b=$("<div>").addClass("qrcode");UI.popup.show($("<span>").addClass("qr_container").append($("<p>").text(a)).append(b)); | ||||
| b.qrcode({text:a,size:Math.min(b.width(),b.height())})})));"rows"in d&&f.attr("rows",d.rows);"LTSonly"in d&&!mist.data.LTS&&(e.addClass("LTSonly"),f.prop("disabled",!0));switch(d.type){case "browse":o=$("<div>").addClass("grouper").append(m);c.append(o);o=$("<button>").text("Browse");e.append(o);o.click(function(){function a(b){k.text("Loading..");mist.send(function(a){f.text(a.browse.path[0]);k.html(o.clone(true).text("..").attr("title","Folder up"));if(a.browse.subdirectories){a.browse.subdirectories.sort(); | ||||
| for(var b in a.browse.subdirectories){var i=a.browse.subdirectories[b];k.append(o.clone(true).attr("title",f.text()+p+i).text(i))}}if(a.browse.files){a.browse.files.sort();for(b in a.browse.files){var i=a.browse.files[b],m=f.text()+p+i,i=$("<a>").text(i).addClass("file").attr("title",m);k.append(i);if(n){var v=true,r;for(r in n)if(typeof n[r]!="undefined"&&mist.inputMatch(n[r],m)){v=false;break}v&&i.hide()}i.click(function(){var a=$(this).attr("title");d.setval(a);e.show();c.remove()})}}},{browse:b})} | ||||
| var b=$(this).closest(".grouper"),c=$("<div>").addClass("browse_container"),d=b.find(".field"),e=$(this),f=$("<span>").addClass("field"),m=$("<button>").text("Select this folder"),k=$("<div>").addClass("browse_contents"),o=$("<a>").addClass("folder"),n=d.data("filetypes");b.append(c);c.append($("<label>").addClass("UIelement").append($("<span>").addClass("label").text("Current folder:")).append($("<span>").addClass("field_container").append(f).append(m))).append(k);m.click(function(){var a=f.text()+ | ||||
| "/";d.setval(a);e.show();c.remove()});var p="/";mist.data.config.version.indexOf("indows")>-1&&(p="\\");o.click(function(){var b=f.text()+p+$(this).text();a(b)});b=d.getval();b=b.split(p);b.pop();b=b.join(p);e.hide();a(b)});break;case "geolimited":case "hostlimited":o={field:f};o.blackwhite=$("<select>").append($("<option>").val("-").text("Blacklist")).append($("<option>").val("+").text("Whitelist"));o.values=$("<span>").addClass("limit_value_list");switch(d.type){case "geolimited":o.prototype=$("<select>").append($("<option>").val("").text("[Select a country]")); | ||||
| for(b in UI.countrylist)o.prototype.append($("<option>").val(b).html(UI.countrylist[b]));break;case "hostlimited":o.prototype=$("<input>").attr("type","text").attr("placeholder","type a host")}o.prototype.on("change keyup",function(){$(this).closest(".field_container").data("subUI").blackwhite.trigger("change")});o.blackwhite.change(function(){var a=$(this).closest(".field_container").data("subUI"),b=[],c=false;a.values.children().each(function(){c=$(this).val();c!=""?b.push(c):$(this).remove()}); | ||||
| a.values.append(a.prototype.clone(true));b.length>0?a.field.val($(this).val()+b.join(" ")):a.field.val("");a.field.trigger("change")});"LTSonly"in d&&!mist.data.LTS&&(o.blackwhite.prop("disabled",!0),o.prototype.prop("disabled",!0));o.values.append(o.prototype.clone(!0));e.data("subUI",o).addClass("limit_list").append(o.blackwhite).append(o.values)}"pointer"in d&&(f.data("pointer",d.pointer).addClass("isSetting"),e=d.pointer.main[d.pointer.index],"undefined"!=e&&f.setval(e));"value"in d&&f.setval(d.value); | ||||
| e=$("<span>").addClass("help_container");m.append(e);"help"in d&&(e.append($("<span>").addClass("ih_balloon").html(d.help)),f.on("focus mouseover",function(){$(this).closest("label").addClass("active")}).on("blur mouseout",function(){$(this).closest("label").removeClass("active")}));if("validate"in d){m=[];for(k in d.validate){o=d.validate[k];if("function"!=typeof o)switch(o){case "required":o=function(a){return a==""?{msg:"This is a required field.",classes:["red"]}:false};break;case "int":o=function(a, | ||||
| b){var c=$(b).data("opts");if(!$(b)[0].validity.valid){var d=[];"min"in c&&d.push(" greater than or equal to "+c.min);"max"in c&&d.push(" smaller than or equal to "+c.max);return{msg:"Please enter an integer"+d.join(" and")+".",classes:["red"]}}if(parseInt(Number(a))!=a)return{msg:"Please enter an integer.",classes:["red"]}};break;case "streamname":o=function(a,b){if(!isNaN(a.charAt(0)))return{msg:"The first character may not be a number.",classes:["red"]};if(a.toLowerCase()!=a)return{msg:"Uppercase letters are not allowed.", | ||||
| classes:["red"]};if(a.replace(/[^\da-z_]/g,"")!=a)return{msg:"Special characters (except for underscores) are not allowed.",classes:["red"]};if("streams"in mist.data&&a in mist.data.streams&&$(b).data("pointer").main.name!=a)return{msg:"This streamname already exists.<br>If you want to edit an existing stream, please click edit on the the streams tab.",classes:["red"]}};break;default:o=function(){}}m.push(o)}f.data("validate_functions",m).data("help_container",e).data("validate",function(a,b){var c= | ||||
| $(a).getval(),d=$(a).data("validate_functions"),e=$(a).data("help_container");e.find(".err_balloon").remove();for(var f in d){var m=d[f](c,a);if(m){$err=$("<span>").addClass("err_balloon").html(m.msg);for(var k in m.classes)$err.addClass(m.classes[k]);e.prepend($err);b&&$(a).focus();return true}}return false}).addClass("hasValidate").on("change keyup",function(){$(this).data("validate")($(this))});""!=f.getval()&&f.trigger("change")}"function"in d&&(f.on("change keyup",d["function"]),f.trigger("change"))}}c.on("keydown", | ||||
| function(a){switch(a.which){case 13:$(this).find("button.save").trigger("click");break;case 27:$(this).find("button.cancel").trigger("click")}});return c},buildVheaderTable:function(a){var c=$("<table>").css("margin","0.2em"),b=$("<tr>").addClass("header").append($("<td>").addClass("vheader").attr("rowspan",a.labels.length+1).append($("<span>").text(a.vheader))),d=[];b.append($("<td>"));for(var f in a.labels)d.push($("<tr>").append($("<td>").html(""==a.labels[f]?" ":a.labels[f]+":")));for(var k in a.content)for(f in b.append($("<td>").html(a.content[k].header)), | ||||
| a.content[k].body)d[f].append($("<td>").html(a.content[k].body[f]));c.append($("<tbody>").append(b).append(d));return c},buildLimits:function(a,c,b){function d(a,d){var e=$("<tbody>"),g;for(g in a){var f=$("<tr>");f.data("pointer",d.concat(g));var j="";switch(d[0]){case "server":j="The entire server";break;case "stream":j='The stream "'+d[1]+'"'}f.append($("<td>").addClass("applies_to").text(j));var j=a[g].name,h=a[g].value;switch(a[g].name){case "kbps_max":j="Maximum bandwidth";h=UI.format.bytes(h, | ||||
| !0);break;case "users":j="Maximum connected users";h=UI.format.number(h);break;case "geo":var j="Geolimited",m=h,h="<span class=unit>["+("-"==h.charAt(0)?"Blacklist":"Whitelist")+"]</span> ",m=m.substr(1).split(" "),k=[],p;for(p in m){var n=UI.countrylist[m[p]];k.push("undefined"==typeof n?m[p]:n)}h+=k.join(", ");break;case "host":j="Hostlimited",m=h,h="<span class=unit>["+("-"==h.charAt(0)?"Blacklist":"Whitelist")+"]</span> ",m=m.substr(1).split(" "),h+=m.join(", ")}f.append($("<td>").text(a[g].type).addClass("kind")).append($("<td>").text(j).addClass("type")).append($("<td>").html(h).addClass("value")).append($("<td>").css("text-align", | ||||
| "right").html($("<button>").text("Edit").click(function(){UI.navto("Edit Limit",$(this).closest("tr").data("pointer").join("^"));UI.returnTab=[c,b]})).append($("<button>").text("Delete").click(function(){var a=$(this).closest("tr"),d="Are you sure you want to delete the "+a.find(".kind").text()+" "+a.find(".type").text()+" limit for "+a.find(".applies_to").text()+"?";if(confirm(d)){var a=a.data("pointer"),d=a[a.length-1],e={};switch(a[0]){case "server":mist.data.config.limits.splice(d,1);e={config:mist.data.config}; | ||||
| break;case "stream":mist.data.streams[a[1]].limits.splice(d,1),1==mist.data.LTS?(d={},d[a[1]]=mist.data.streams[a[1]],e={addstream:d}):e={streams:mist.data.streams}}mist.send(function(){UI.navto(c,b)},e)}})));e.append(f)}return e.children()}var f=$("<div>"),k=$("<table>");f.append(k);var m=$("<tr>").append($("<th>").text("Kind").attr("data-sort-type","string")).append($("<th>").text("Type").attr("data-sort-type","string")).append($("<th>").text("Value")).append($("<th>"));a instanceof Array?a={noapply:a}: | ||||
| m.prepend($("<th>").text("Applies to").attr("data-sort-type","string").addClass("sorting-asc").addClass("applies_to"));k.append($("<thead>").append(m));m=$("<tbody>");k.append(m);for(var e in a)switch(e){case "server":m.append(d(a[e],["server"]));break;case "streams":for(var p in a[e])m.append(d(a[e][p],["stream",p]))}f.prepend($("<button>").text("New limit").click(function(){UI.navto("Edit Limit");UI.returnTab=[c,b]}));0<m.children().length?k.stupidtable():(k.remove(),f.append($("<span>").text("No limits set."))); | ||||
| return f.children()},plot:{addGraph:function(a,c){var b={id:a.id,xaxis:a.xaxis,datasets:[],elements:{cont:$("<div>").addClass("graph"),plot:$("<div>").addClass("plot"),legend:$("<div>").addClass("legend")}};b.elements.cont.append(b.elements.plot).append(b.elements.legend);c.append(b.elements.cont);return b},go:function(a){if(!(1>Object.keys(a).length)){var c={totals:[],clients:[]},b;for(b in a)for(var d in a[b].datasets){var f=a[b].datasets[d];switch(f.datatype){case "clients":case "upbps":case "downbps":switch(f.origin[0]){case "total":c.totals.push({fields:[f.datatype]}); | ||||
| break;case "stream":c.totals.push({fields:[f.datatype],streams:[f.origin[1]]});break;case "protocol":c.totals.push({fields:[f.datatype],protocols:[f.origin[1]]})}break;case "cpuload":case "memload":c.capabilities={}}}0==c.totals.length&&delete c.totals;0==c.clients.length&&delete c.clients;mist.send(function(){for(var b in a){var c=a[b];if(1>c.datasets.length){c.elements.plot.html("");c.elements.legend.html("");break}switch(c.xaxis){case "time":var d=[];c.yaxes={};var f=[],o;for(o in c.datasets){var i= | ||||
| c.datasets[o];i.display&&(i.getdata(),i.yaxistype in c.yaxes||(d.push(UI.plot.yaxes[i.yaxistype]),c.yaxes[i.yaxistype]=d.length),i.yaxis=c.yaxes[i.yaxistype],f.push(i))}d[0]&&(d[0].color=0);c.plot=$.plot(c.elements.plot,f,{legend:{show:!1},xaxis:UI.plot.xaxes[c.xaxis],yaxes:d,grid:{hoverable:!0,borderWidth:{top:0,right:0,bottom:1,left:1},color:"black",backgroundColor:{colors:["rgba(0,0,0,0)","rgba(0,0,0,0.025)"]}}});d=$("<div>").addClass("legend-list").addClass("checklist");c.elements.legend.html($("<h3>").text(c.id)).append($("<button>").data("opts", | ||||
| c).text("X").addClass("close").click(function(){var b=$(this).data("opts");if(confirm("Are you sure you want to remove "+b.id+"?")){b.elements.cont.remove();var c=$(".graph_ids option:contains("+b.id+")"),d=c.parent();c.remove();UI.plot.del(b.id);delete a[b.id];d.trigger("change");UI.plot.go(a)}})).append(d);c.plot.getOptions();for(o in c.datasets)f=$("<input>").attr("type","checkbox").data("index",o).data("graph",c).click(function(){var a=$(this).data("graph");$(this).is(":checked")?a.datasets[$(this).data("index")].display= | ||||
| true:a.datasets[$(this).data("index")].display=false;var b={};b[a.id]=a;UI.plot.go(b)}),c.datasets[o].display&&f.attr("checked","checked"),d.append($("<label>").html(f).append($("<div>").addClass("series-color").css("background-color",c.datasets[o].color)).append(c.datasets[o].label).append($("<button>").text("X").addClass("close").data("index",o).data("graph",c).click(function(){var b=$(this).data("index"),c=$(this).data("graph");if(confirm("Are you sure you want to remove "+c.datasets[b].label+ | ||||
| " from "+c.id+"?")){c.datasets.splice(b,1);if(c.datasets.length==0){c.elements.cont.remove();var b=$(".graph_ids option:contains("+c.id+")"),d=b.parent();b.remove();d.trigger("change");UI.plot.del(c.id);delete a[c.id];UI.plot.go(a)}else{UI.plot.save(c);b={};b[c.id]=c;UI.plot.go(b)}}})));c.elements.plot.on("plothover",function(a,b,c){if(c){a=$("<span>").append($("<h3>").text(c.series.label).prepend($("<div>").addClass("series-color").css("background-color",c.series.color))).append($("<table>").addClass("nolay").html($("<tr>").html($("<td>").text("Time:")).append($("<td>").html(UI.format.dateTime(c.datapoint[0]/ | ||||
| 1E3,"long")))).append($("<tr>").html($("<td>").text("Value:")).append($("<td>").html(c.series.yaxis.tickFormatter(c.datapoint[1],c.series.yaxis)))));UI.tooltip.show(b,a.children())}else UI.tooltip.hide()})}}},c)}},save:function(a){var c={id:a.id,xaxis:a.xaxis,datasets:[]},b;for(b in a.datasets)c.datasets.push({origin:a.datasets[b].origin,datatype:a.datasets[b].datatype});a=mist.stored.get().graphs||{};a[c.id]=c;mist.stored.set("graphs",a)},del:function(a){var c=mist.stored.get().graphs||{};delete c[a]; | ||||
| mist.stored.set("graphs",c)},datatype:{getOptions:function(a){var c=$.extend(!0,{},UI.plot.datatype.templates.general),b=$.extend(!0,{},UI.plot.datatype.templates[a.datatype]),a=$.extend(!0,b,a),a=$.extend(!0,c,a);switch(a.origin[0]){case "total":switch(a.datatype){case "cpuload":case "memload":break;default:a.label+=" (total)"}break;case "stream":case "protocol":a.label+=" ("+a.origin[1]+")"}var c=[],d;for(d in a.basecolor)b=a.basecolor[d],b+=50*(0.5-Math.random()),b=Math.round(b),b=Math.min(255, | ||||
| Math.max(0,b)),c.push(b);a.color="rgb("+c.join(",")+")";return a},templates:{general:{display:!0,datatype:"general",label:"",yaxistype:"amount",data:[],lines:{show:!0},points:{show:!1},getdata:function(){var a=mist.data.totals["stream"==this.origin[0]?this.origin[1]:"all_streams"]["protocol"==this.origin[0]?this.origin[1]:"all_protocols"][this.datatype];return this.data=a}},cpuload:{label:"CPU load",yaxistype:"percentage",basecolor:[237,194,64],cores:1,getdata:function(){var a=!1,c;for(c in this.data)this.data[c][0]< | ||||
| 1E3*(mist.data.config.time-600)&&(a=c);!1!==a&&this.data.splice(0,Number(a)+1);this.data.push([1E3*mist.data.config.time,Math.min(100,mist.data.capabilities.load.one/this.cores)]);return this.data}},memload:{label:"Memory load",yaxistype:"percentage",basecolor:[175,216,248],getdata:function(){var a=!1,c;for(c in this.data)this.data[c][0]<1E3*(mist.data.config.time-600)&&(a=c);!1!==a&&this.data.splice(0,Number(a)+1);this.data.push([1E3*mist.data.config.time,mist.data.capabilities.load.memory]);return this.data}}, | ||||
| clients:{label:"Connections",basecolor:[203,75,75]},upbps:{label:"Bandwidth up",yaxistype:"bytespersec",basecolor:[77,167,77]},downbps:{label:"Bandwidth down",yaxistype:"bytespersec",basecolor:[148,64,237]}}},yaxes:{percentage:{name:"percentage",color:"black",tickColor:0,tickDecimals:0,tickFormatter:function(a){return UI.format.addUnit(UI.format.number(a),"%")},tickLength:0,min:0,max:100},amount:{name:"amount",color:"black",tickColor:0,tickDecimals:0,tickFormatter:function(a){return UI.format.number(a)}, | ||||
| tickLength:0,min:0},bytespersec:{name:"bytespersec",color:"black",tickColor:0,tickDecimals:1,tickFormatter:function(a){return UI.format.bytes(a,!0)},tickLength:0,ticks:function(a){var c=0.3*Math.sqrt($(".graph").first().height()),c=(a.max-a.min)/c,b=Math.floor(Math.log(Math.abs(c))/Math.log(1024)),d=c/Math.pow(1024,b),f=-Math.floor(Math.log(d)/Math.LN10),k=a.tickDecimals;null!=k&&f>k&&(f=k);var m=Math.pow(10,-f),d=d/m,e;if(1.5>d)e=1;else if(3>d){if(e=2,2.25<d&&(null==k||f+1<=k))e=2.5,++f}else e=7.5> | ||||
| d?5:10;e=e*m*Math.pow(1024,b);null!=a.minTickSize&&e<a.minTickSize&&(e=a.minTickSize);a.delta=c;a.tickDecimals=Math.max(0,null!=k?k:f);a.tickSize=e;c=[];b=a.tickSize*Math.floor(a.min/a.tickSize);f=0;k=Number.NaN;do m=k,k=b+f*a.tickSize,c.push(k),++f;while(k<a.max&&k!=m);return c},min:0}},xaxes:{time:{name:"time",mode:"time",timezone:"browser",ticks:5}}},format:{time:function(a,c){var b=new Date(1E3*a),d=[];d.push(("0"+b.getHours()).slice(-2));d.push(("0"+b.getMinutes()).slice(-2));"short"!=c&&d.push(("0"+ | ||||
| b.getSeconds()).slice(-2));return d.join(":")},date:function(a,c){var b=new Date(1E3*a),d="Sun Mon Tue Wed Thu Fri Sat".split(" "),f=[];"long"==c&&f.push(d[b.getDay()]);f.push(("0"+b.getDate()).slice(-2));f.push("Jan Feb Mar Apr May Jun Jul Aug Sep Oct Nov Dec".split(" ")[b.getMonth()]);"short"!=c&&f.push(b.getFullYear());return f.join(" ")},dateTime:function(a,c){return UI.format.date(a,c)+", "+UI.format.time(a,c)},duration:function(a){var c=[0.001,1E3,60,60,24,7,1E9],b="ms sec min hr day week".split(" "), | ||||
| d={},f;for(f in b){var a=a/c[f],k=Math.round(a%c[Number(f)+1]);d[b[f]]=k;a-=k}var m;for(f=b.length-1;0<=f;f--)if(0<d[b[f]]){m=b[f];break}c=$("<span>");switch(m){case "week":c.append(UI.format.addUnit(d.week,"wks, ")).append(UI.format.addUnit(d.day,"days"));break;case "day":c.append(UI.format.addUnit(d.day,"days, ")).append(UI.format.addUnit(d.hr,"hrs"));break;default:c.append([("0"+d.hr).slice(-2),("0"+d.min).slice(-2),("0"+d.sec).slice(-2)+(d.ms?"."+d.ms:"")].join(":"))}return c[0].innerHTML},number:function(a){if(isNaN(Number(a))|| | ||||
| 0==a)return a;var c=Math.pow(10,3-Math.floor(Math.log(a)/Math.LN10)-1),a=Math.round(a*c)/c;if(1E4<a){number=a.toString().split(".");for(a=/(\d+)(\d{3})/;a.test(number[0]);)number[0]=number[0].replace(a,"$1 $2");a=number.join(".")}return a},status:function(a){var c=$("<span>");if("undefined"==typeof a.online)return c.text("Unknown, checking.."),"undefined"!=typeof a.error&&c.text(a.error),c;switch(a.online){case -1:c.text("Enabling");break;case 0:c.text("Unavailable").addClass("red");break;case 1:c.text("Active").addClass("green"); | ||||
| break;case 2:c.text("Standby").addClass("orange");break;default:c.text(a.online)}"error"in a&&c.text(a.error);return c},capital:function(a){return a.charAt(0).toUpperCase()+a.substring(1)},addUnit:function(a,c){var b=$("<span>").html(a);b.append($("<span>").addClass("unit").html(c));return b[0].innerHTML},bytes:function(a,c){var b="bytes KiB MiB GiB TiB PiB".split(" ");if(0==a)unit=b[0];else{var d=Math.floor(Math.log(Math.abs(a))/Math.log(1024));0>d?unit=b[0]:(a/=Math.pow(1024,d),unit=b[d])}return UI.format.addUnit(UI.format.number(a), | ||||
| unit+(c?"/s":""))}},navto:function(a,c){var b=location.hash,d=b.split("@");d[0]=[mist.user.name,mist.user.host].join("&");d[1]=[a,c].join("&");"undefined"!=typeof screenlog&&screenlog.navto(d[1]);location.hash=d.join("@");location.hash==b&&$(window).trigger("hashchange")},showTab:function(a,c){var b=UI.elements.main;if(mist.user.loggedin&&!("ui_settings"in mist.data))b.html("Loading.."),mist.send(function(){UI.showTab(a,c)},{ui_settings:!0});else{var d=UI.elements.menu.css("visibility","visible").find(".button").filter(function(){if($(this).find(".plain").text()== | ||||
| a)return!0});0<d.length&&(UI.elements.menu.find(".button.active").removeClass("active"),d.addClass("active"));UI.elements.secondary_menu.html("");UI.interval.clear();b.html($("<h2>").text(a));switch(a){case "Login":if(mist.user.loggedin){UI.navto("Overview");break}UI.elements.menu.css("visibility","hidden");UI.elements.connection.status.text("Disconnected").removeClass("green").addClass("red");b.append(UI.buildUI([{type:"help",help:"Please provide your account details.<br>You were asked to set these when MistController was started for the first time. If you did not yet set any account details, log in with your desired credentials to create a new account."}, | ||||
| {label:"Host",help:"Url location of the MistServer API. Generally located at http://MistServerIP:4242/api","default":"http://localhost:4242/api",pointer:{main:mist.user,index:"host"}},{label:"Username",help:"Please enter your username here.",validate:["required"],pointer:{main:mist.user,index:"name"}},{label:"Password",type:"password",help:"Please enter your password here.",validate:["required"],pointer:{main:mist.user,index:"password"}},{type:"buttons",buttons:[{label:"Login",type:"save","function":function(){mist.send(function(){UI.navto("Overview")})}}]}])); | ||||
| break;case "Create a new account":UI.elements.menu.css("visibility","hidden");b.append($("<p>").text("No account has been created yet in the MistServer at ").append($("<i>").text(mist.user.host)).append("."));b.append(UI.buildUI([{type:"buttons",buttons:[{label:"Select other host",type:"cancel",css:{"float":"left"},"function":function(){UI.navto("Login")}}]},{type:"custom",custom:$("<br>")},{label:"Desired username",type:"str",validate:["required"],help:"Enter your desired username. In the future, you will need this to access the Management Interface.", | ||||
| pointer:{main:mist.user,index:"name"}},{label:"Desired password",type:"password",validate:["required",function(a,b){$(".match_password").not($(b)).trigger("change");return false}],help:"Enter your desired password. In the future, you will need this to access the Management Interface.",pointer:{main:mist.user,index:"password"},classes:["match_password"]},{label:"Repeat password",type:"password",validate:["required",function(a,b){return a!=$(".match_password").not($(b)).val()?{msg:'The fields "Desired password" and "Repeat password" do not match.', | ||||
| function(){UI.popup.element.remove();UI.popup.element=null})})).append(a);$("body").append(this.element)}},menu:[{Overview:{},Protocols:{},Streams:{},Preview:{},Triggers:{LTSonly:!1},Logs:{},Statistics:{},"Server Stats":{}},{Disconnect:{classes:["red"]}},{Guides:{link:"http://mistserver.org/documentation#Userdocs"},Tools:{submenu:{"Release notes":{link:"http://mistserver.org/documentation#Devdocs"},"Mist Shop":{link:"http://mistserver.org/products"},"Email for Help":{},"Terms & Conditions":{link:"http://mistserver.org/documentation#Legal"}}}}], | ||||
| buildMenu:function(){function a(a,c){var b=$("<a>").addClass("button");b.html($("<span>").addClass("plain").text(a)).append($("<span>").addClass("highlighted").text(a));for(var d in c.classes)b.addClass(c.classes[d]);"LTSonly"in c&&b.addClass("LTSonly");"link"in c?b.attr("href",c.link).attr("target","_blank"):"submenu"in c||b.click(function(c){UI.navto(a);c.stopPropagation()});return b}var c=UI.elements.menu,b;for(b in UI.menu){0<b&&c.append($("<br>"));for(var d in UI.menu[b]){var e=UI.menu[b][d], | ||||
| j=a(d,e);c.append(j);if("submenu"in e){var m=$("<span>").addClass("submenu");j.addClass("arrowdown").append(m);for(var g in e.submenu)m.append(a(g,e.submenu[g]))}}}b=$("<div>").attr("id","ih_button").text("?").click(function(){$("body").toggleClass("helpme");UI.stored.saveOpt("helpme",$("body").hasClass("helpme"))}).attr("title","Click to toggle the display of integrated help");UI.stored.getOpts().helpme&&$("body").addClass("helpme");c.after(b).after($("<div>").addClass("separator"))},buildUI:function(a){var c= | ||||
| $("<div>").addClass("input_container"),b;for(b in a){var d=a[b];if(d instanceof jQuery)c.append(d);else if("help"==d.type){var e=$("<span>").addClass("text_container").append($("<span>").addClass("description").append(d.help));c.append(e);if("classes"in d)for(var j in d.classes)e.addClass(d.classes[j])}else if("text"==d.type)c.append($("<span>").addClass("text_container").append($("<span>").addClass("text").append(d.text)));else if("custom"==d.type)c.append(d.custom);else if("buttons"==d.type)for(j in e= | ||||
| $("<span>").addClass("button_container"),"css"in d&&e.css(d.css),c.append(e),d.buttons){var m=d.buttons[j],g=$("<button>").text(m.label).data("opts",m);"css"in m&&g.css(m.css);e.append(g);switch(m.type){case "cancel":g.addClass("cancel").click(m["function"]);break;case "save":g.addClass("save").click(function(){var a=$(this).closest(".input_container"),c=!1;a.find(".hasValidate").each(function(){if(c=$(this).data("validate")(this,!0))return!1});c||(a.find(".isSetting").each(function(){var a=$(this).getval(), | ||||
| c=$(this).data("pointer");if(""==a)if("default"in $(this).data("opts"))a=$(this).data("opts")["default"];else return delete c.main[c.index],!0;c.main[c.index]=a}),(a=$(this).data("opts")["function"])&&a(this))});break;default:g.click(m["function"])}}else{m=$("<label>").addClass("UIelement");c.append(m);"css"in d&&m.css(d.css);m.append($("<span>").addClass("label").html(d.label+":"));g=$("<span>").addClass("field_container");m.append(g);switch(d.type){case "password":e=$("<input>").attr("type","password"); | ||||
| break;case "int":e=$("<input>").attr("type","number");"min"in d&&e.attr("min",d.min);"max"in d&&e.attr("max",d.min);"validate"in d?d.validate.push("int"):d.validate=["int"];break;case "span":e=$("<span>");break;case "debug":d.select=[["","Default"],[0,"0 - All debugging messages disabled"],[1,"1 - Messages about failed operations"],[2,"2 - Previous level, and error messages"],[3,"3 - Previous level, and warning messages"],[4,"4 - Previous level, and status messages for development"],[5,"5 - Previous level, and more status messages for development"], | ||||
| [6,"6 - Previous level, and verbose debugging messages"],[7,"7 - Previous level, and very verbose debugging messages"],[8,"8 - Report everything in extreme detail"],[9,"9 - Report everything in insane detail"],[10,"10 - All messages enabled"]];case "select":e=$("<select>");for(j in d.select){var p=$("<option>");"string"==typeof d.select[j]?p.text(d.select[j]):p.val(d.select[j][0]).text(d.select[j][1]);e.append(p)}break;case "textarea":e=$("<textarea>").on("keydown",function(a){a.stopPropagation()}); | ||||
| break;case "checkbox":e=$("<input>").attr("type","checkbox");break;case "hidden":e=$("<input>").attr("type","hidden");m.hide();break;case "email":e=$("<input>").attr("type","email").attr("autocomplete","on").attr("required","");break;case "browse":e=$("<input>").attr("type","text");"filetypes"in d&&e.data("filetypes",d.filetypes);break;case "geolimited":case "hostlimited":e=$("<input>").attr("type","hidden");break;case "radioselect":e=$("<div>").addClass("radioselect");for(b in d.radioselect){var n= | ||||
| $("<input>").attr("type","radio").val(d.radioselect[b][0]).attr("name",d.label);("LTSonly"in d&&!mist.data.LTS||d.readonly)&&n.prop("disabled",!0);p=$("<label>").append(n).append($("<span>").html(d.radioselect[b][1]));e.append(p);if(2<d.radioselect[b].length)for(j in n=$("<select>").change(function(){$(this).parent().find("input[type=radio]:enabled").prop("checked","true")}),p.append(n),("LTSonly"in d&&!mist.data.LTS||d.readonly)&&n.prop("disabled",!0),d.radioselect[b][2])p=$("<option>"),n.append(p), | ||||
| d.radioselect[b][2][j]instanceof Array?p.val(d.radioselect[b][2][j][0]).html(d.radioselect[b][2][j][1]):p.html(d.radioselect[b][2][j])}break;case "checklist":e=$("<div>").addClass("checkcontainer");$controls=$("<div>").addClass("controls");$checklist=$("<div>").addClass("checklist");e.append($controls).append($checklist);$controls.append($("<label>").text("All").prepend($("<input>").attr("type","checkbox").click(function(){$(this).is(":checked")?$(this).closest(".checkcontainer").find("input[type=checkbox]").prop("checked", | ||||
| !0):$(this).closest(".checkcontainer").find("input[type=checkbox]").prop("checked",!1)})));for(b in d.checklist)"string"==typeof d.checklist[b]&&(d.checklist[b]=[d.checklist[b],d.checklist[b]]),$checklist.append($("<label>").text(d.checklist[b][1]).prepend($("<input>").attr("type","checkbox").attr("name",d.checklist[b][0])));break;case "DOMfield":e=d.DOMfield;break;default:e=$("<input>").attr("type","text")}e.addClass("field").data("opts",d);"pointer"in d&&e.attr("name",d.pointer.index);g.append(e); | ||||
| if("classes"in d)for(j in d.classes)e.addClass(d.classes[j]);"placeholder"in d&&e.attr("placeholder",d.placeholder);"default"in d&&e.attr("placeholder",d["default"]);"unit"in d&&g.append($("<span>").addClass("unit").html(d.unit));"readonly"in d&&(e.attr("readonly","readonly"),e.click(function(){$(this).select()}));"qrcode"in d&&g.append($("<span>").addClass("unit").html($("<button>").text("QR").click(function(){var a=String($(this).closest(".field_container").find(".field").getval()),c=$("<div>").addClass("qrcode"); | ||||
| UI.popup.show($("<span>").addClass("qr_container").append($("<p>").text(a)).append(c));c.qrcode({text:a,size:Math.min(c.width(),c.height())})})));"rows"in d&&e.attr("rows",d.rows);"LTSonly"in d&&!mist.data.LTS&&(g.addClass("LTSonly"),e.prop("disabled",!0));switch(d.type){case "browse":n=$("<div>").addClass("grouper").append(m);c.append(n);n=$("<button>").text("Browse");g.append(n);n.click(function(){function a(c){j.text("Loading..");mist.send(function(a){e.text(a.browse.path[0]);j.html(n.clone(true).text("..").attr("title", | ||||
| "Folder up"));if(a.browse.subdirectories){a.browse.subdirectories.sort();for(var c in a.browse.subdirectories){var h=a.browse.subdirectories[c];j.append(n.clone(true).attr("title",e.text()+p+h).text(h))}}if(a.browse.files){a.browse.files.sort();for(c in a.browse.files){var h=a.browse.files[c],m=e.text()+p+h,h=$("<a>").text(h).addClass("file").attr("title",m);j.append(h);if(o){var q=true,s;for(s in o)if(typeof o[s]!="undefined"&&mist.inputMatch(o[s],m)){q=false;break}q&&h.hide()}h.click(function(){var a= | ||||
| $(this).attr("title");d.setval(a);g.show();b.remove()})}}},{browse:c})}var c=$(this).closest(".grouper"),b=$("<div>").addClass("browse_container"),d=c.find(".field"),g=$(this),e=$("<span>").addClass("field"),m=$("<button>").text("Select this folder"),j=$("<div>").addClass("browse_contents"),n=$("<a>").addClass("folder"),o=d.data("filetypes");c.append(b);b.append($("<label>").addClass("UIelement").append($("<span>").addClass("label").text("Current folder:")).append($("<span>").addClass("field_container").append(e).append(m))).append(j); | ||||
| m.click(function(){var a=e.text()+"/";d.setval(a);g.show();b.remove()});var p="/";mist.data.config.version.indexOf("indows")>-1&&(p="\\");n.click(function(){var c=e.text()+p+$(this).text();a(c)});c=d.getval();c=c.split(p);c.pop();c=c.join(p);g.hide();a(c)});break;case "geolimited":case "hostlimited":n={field:e};n.blackwhite=$("<select>").append($("<option>").val("-").text("Blacklist")).append($("<option>").val("+").text("Whitelist"));n.values=$("<span>").addClass("limit_value_list");switch(d.type){case "geolimited":n.prototype= | ||||
| $("<select>").append($("<option>").val("").text("[Select a country]"));for(b in UI.countrylist)n.prototype.append($("<option>").val(b).html(UI.countrylist[b]));break;case "hostlimited":n.prototype=$("<input>").attr("type","text").attr("placeholder","type a host")}n.prototype.on("change keyup",function(){$(this).closest(".field_container").data("subUI").blackwhite.trigger("change")});n.blackwhite.change(function(){var a=$(this).closest(".field_container").data("subUI"),c=[],b=false;a.values.children().each(function(){b= | ||||
| $(this).val();b!=""?c.push(b):$(this).remove()});a.values.append(a.prototype.clone(true));c.length>0?a.field.val($(this).val()+c.join(" ")):a.field.val("");a.field.trigger("change")});"LTSonly"in d&&!mist.data.LTS&&(n.blackwhite.prop("disabled",!0),n.prototype.prop("disabled",!0));n.values.append(n.prototype.clone(!0));g.data("subUI",n).addClass("limit_list").append(n.blackwhite).append(n.values)}"pointer"in d&&(e.data("pointer",d.pointer).addClass("isSetting"),g=d.pointer.main[d.pointer.index],"undefined"!= | ||||
| g&&e.setval(g));"value"in d&&e.setval(d.value);g=$("<span>").addClass("help_container");m.append(g);"help"in d&&(g.append($("<span>").addClass("ih_balloon").html(d.help)),e.on("focus mouseover",function(){$(this).closest("label").addClass("active")}).on("blur mouseout",function(){$(this).closest("label").removeClass("active")}));if("validate"in d){m=[];for(j in d.validate){n=d.validate[j];if("function"!=typeof n)switch(n){case "required":n=function(a){return a==""?{msg:"This is a required field.", | ||||
| classes:["red"]}:false};break;case "int":n=function(a,c){var b=$(c).data("opts");if(!$(c)[0].validity.valid){var d=[];"min"in b&&d.push(" greater than or equal to "+b.min);"max"in b&&d.push(" smaller than or equal to "+b.max);return{msg:"Please enter an integer"+d.join(" and")+".",classes:["red"]}}if(parseInt(Number(a))!=a)return{msg:"Please enter an integer.",classes:["red"]}};break;case "streamname":n=function(a,c){if(!isNaN(a.charAt(0)))return{msg:"The first character may not be a number.",classes:["red"]}; | ||||
| if(a.toLowerCase()!=a)return{msg:"Uppercase letters are not allowed.",classes:["red"]};if(a.replace(/[^\da-z_]/g,"")!=a)return{msg:"Special characters (except for underscores) are not allowed.",classes:["red"]};if("streams"in mist.data&&a in mist.data.streams&&$(c).data("pointer").main.name!=a)return{msg:"This streamname already exists.<br>If you want to edit an existing stream, please click edit on the the streams tab.",classes:["red"]}};break;default:n=function(){}}m.push(n)}e.data("validate_functions", | ||||
| m).data("help_container",g).data("validate",function(a,c){var b=$(a).getval(),d=$(a).data("validate_functions"),g=$(a).data("help_container");g.find(".err_balloon").remove();for(var e in d){var m=d[e](b,a);if(m){$err=$("<span>").addClass("err_balloon").html(m.msg);for(var j in m.classes)$err.addClass(m.classes[j]);g.prepend($err);c&&$(a).focus();return true}}return false}).addClass("hasValidate").on("change keyup",function(){$(this).data("validate")($(this))});""!=e.getval()&&e.trigger("change")}"function"in | ||||
| d&&(e.on("change keyup",d["function"]),e.trigger("change"))}}c.on("keydown",function(a){switch(a.which){case 13:$(this).find("button.save").trigger("click");break;case 27:$(this).find("button.cancel").trigger("click")}});return c},buildVheaderTable:function(a){var c=$("<table>").css("margin","0.2em"),b=$("<tr>").addClass("header").append($("<td>").addClass("vheader").attr("rowspan",a.labels.length+1).append($("<span>").text(a.vheader))),d=[];b.append($("<td>"));for(var e in a.labels)d.push($("<tr>").append($("<td>").html(""== | ||||
| a.labels[e]?" ":a.labels[e]+":")));for(var j in a.content)for(e in b.append($("<td>").html(a.content[j].header)),a.content[j].body)d[e].append($("<td>").html(a.content[j].body[e]));c.append($("<tbody>").append(b).append(d));return c},plot:{addGraph:function(a,c){var b={id:a.id,xaxis:a.xaxis,datasets:[],elements:{cont:$("<div>").addClass("graph"),plot:$("<div>").addClass("plot"),legend:$("<div>").addClass("legend").attr("draggable","true")}};UI.draggable(b.elements.legend);b.elements.cont.append(b.elements.plot).append(b.elements.legend); | ||||
| c.append(b.elements.cont);return b},go:function(a){if(!(1>Object.keys(a).length)){var c={totals:[],clients:[]},b;for(b in a)for(var d in a[b].datasets){var e=a[b].datasets[d];switch(e.datatype){case "clients":case "upbps":case "downbps":switch(e.origin[0]){case "total":c.totals.push({fields:[e.datatype],end:-15});break;case "stream":c.totals.push({fields:[e.datatype],streams:[e.origin[1]],end:-15});break;case "protocol":c.totals.push({fields:[e.datatype],protocols:[e.origin[1]],end:-15})}break;case "cpuload":case "memload":c.capabilities= | ||||
| {}}}0==c.totals.length&&delete c.totals;0==c.clients.length&&delete c.clients;mist.send(function(){for(var c in a){var b=a[c];if(1>b.datasets.length){b.elements.plot.html("");b.elements.legend.html("");break}switch(b.xaxis){case "time":var d=[];b.yaxes={};var e=[],n;for(n in b.datasets){var h=b.datasets[n];h.display&&(h.getdata(),h.yaxistype in b.yaxes||(d.push(UI.plot.yaxes[h.yaxistype]),b.yaxes[h.yaxistype]=d.length),h.yaxis=b.yaxes[h.yaxistype],e.push(h))}d[0]&&(d[0].color=0);b.plot=$.plot(b.elements.plot, | ||||
| e,{legend:{show:!1},xaxis:UI.plot.xaxes[b.xaxis],yaxes:d,grid:{hoverable:!0,borderWidth:{top:0,right:0,bottom:1,left:1},color:"black",backgroundColor:{colors:["rgba(0,0,0,0)","rgba(0,0,0,0.025)"]}},crosshair:{mode:"x"}});d=$("<table>").addClass("legend-list").addClass("nolay").html($("<tr>").html($("<td>").html($("<h3>").text(b.id))).append($("<td>").css("padding-right","2em").css("text-align","right").html($("<span>").addClass("value")).append($("<button>").data("opts",b).text("X").addClass("close").click(function(){var c= | ||||
| $(this).data("opts");if(confirm("Are you sure you want to remove "+c.id+"?")){c.elements.cont.remove();var b=$(".graph_ids option:contains("+c.id+")"),d=b.parent();b.remove();UI.plot.del(c.id);delete a[c.id];d.trigger("change");UI.plot.go(a)}}))));b.elements.legend.html(d);var s=function(a){var c=b.elements.legend.find(".value"),d=1;if(typeof a=="undefined")c.eq(0).html("Latest:");else{var g=b.plot.getXAxes()[0],a=Math.min(g.max,a),a=Math.max(g.min,a);c.eq(0).html(UI.format.time(a/1E3))}for(var e in b.datasets){var f= | ||||
| " ";if(b.datasets[e].display){var g=UI.plot.yaxes[b.datasets[e].yaxistype].tickFormatter,h=b.datasets[e].data;if(a)for(var j in h){if(h[j][0]==a){f=g(h[j][1]);break}if(h[j][0]>a){if(j!=0){f=h[j];h=h[j-1];f=g(f[1]+(a-f[0])*(h[1]-f[1])/(h[0]-f[0]))}break}}else f=g(b.datasets[e].data[b.datasets[e].data.length-1][1])}c.eq(d).html(f);d++}};b.plot.getOptions();for(n in b.datasets)e=$("<input>").attr("type","checkbox").data("index",n).data("graph",b).click(function(){var a=$(this).data("graph");$(this).is(":checked")? | ||||
| a.datasets[$(this).data("index")].display=true:a.datasets[$(this).data("index")].display=false;var c={};c[a.id]=a;UI.plot.go(c)}),b.datasets[n].display&&e.attr("checked","checked"),d.append($("<tr>").html($("<td>").html($("<label>").html(e).append($("<div>").addClass("series-color").css("background-color",b.datasets[n].color)).append(b.datasets[n].label))).append($("<td>").css("padding-right","2em").css("text-align","right").html($("<span>").addClass("value")).append($("<button>").text("X").addClass("close").data("index", | ||||
| n).data("graph",b).click(function(){var c=$(this).data("index"),b=$(this).data("graph");if(confirm("Are you sure you want to remove "+b.datasets[c].label+" from "+b.id+"?")){b.datasets.splice(c,1);if(b.datasets.length==0){b.elements.cont.remove();var c=$(".graph_ids option:contains("+b.id+")"),d=c.parent();c.remove();d.trigger("change");UI.plot.del(b.id);delete a[b.id];UI.plot.go(a)}else{UI.plot.save(b);c={};c[b.id]=b;UI.plot.go(c)}}}))));s();var f=!1;b.elements.plot.on("plothover",function(a,c,b){if(c.x!= | ||||
| f){s(c.x);f=c.x}if(b){a=$("<span>").append($("<h3>").text(b.series.label).prepend($("<div>").addClass("series-color").css("background-color",b.series.color))).append($("<table>").addClass("nolay").html($("<tr>").html($("<td>").text("Time:")).append($("<td>").html(UI.format.dateTime(b.datapoint[0]/1E3,"long")))).append($("<tr>").html($("<td>").text("Value:")).append($("<td>").html(b.series.yaxis.tickFormatter(b.datapoint[1],b.series.yaxis)))));UI.tooltip.show(c,a.children())}else UI.tooltip.hide()}).on("mouseout", | ||||
| function(){s()})}}},c)}},save:function(a){var c={id:a.id,xaxis:a.xaxis,datasets:[]},b;for(b in a.datasets)c.datasets.push({origin:a.datasets[b].origin,datatype:a.datasets[b].datatype});a=mist.stored.get().graphs||{};a[c.id]=c;mist.stored.set("graphs",a)},del:function(a){var c=mist.stored.get().graphs||{};delete c[a];mist.stored.set("graphs",c)},datatype:{getOptions:function(a){var c=$.extend(!0,{},UI.plot.datatype.templates.general),b=$.extend(!0,{},UI.plot.datatype.templates[a.datatype]),a=$.extend(!0, | ||||
| b,a),a=$.extend(!0,c,a);switch(a.origin[0]){case "total":switch(a.datatype){case "cpuload":case "memload":break;default:a.label+=" (total)"}break;case "stream":case "protocol":a.label+=" ("+a.origin[1]+")"}var c=[],d;for(d in a.basecolor)b=a.basecolor[d],b+=50*(0.5-Math.random()),b=Math.round(b),b=Math.min(255,Math.max(0,b)),c.push(b);a.color="rgb("+c.join(",")+")";return a},templates:{general:{display:!0,datatype:"general",label:"",yaxistype:"amount",data:[],lines:{show:!0},points:{show:!1},getdata:function(){var a= | ||||
| mist.data.totals["stream"==this.origin[0]?this.origin[1]:"all_streams"]["protocol"==this.origin[0]?this.origin[1]:"all_protocols"][this.datatype];return this.data=a}},cpuload:{label:"CPU use",yaxistype:"percentage",basecolor:[237,194,64],cores:1,getdata:function(){var a=!1,c;for(c in this.data)this.data[c][0]<1E3*(mist.data.config.time-600)&&(a=c);!1!==a&&this.data.splice(0,Number(a)+1);this.data.push([1E3*mist.data.config.time,mist.data.capabilities.cpu_use/10]);return this.data}},memload:{label:"Memory load", | ||||
| yaxistype:"percentage",basecolor:[175,216,248],getdata:function(){var a=!1,c;for(c in this.data)this.data[c][0]<1E3*(mist.data.config.time-600)&&(a=c);!1!==a&&this.data.splice(0,Number(a)+1);this.data.push([1E3*mist.data.config.time,mist.data.capabilities.load.memory]);return this.data}},clients:{label:"Connections",basecolor:[203,75,75]},upbps:{label:"Bandwidth up",yaxistype:"bytespersec",basecolor:[77,167,77]},downbps:{label:"Bandwidth down",yaxistype:"bytespersec",basecolor:[148,64,237]}}},yaxes:{percentage:{name:"percentage", | ||||
| color:"black",tickColor:0,tickDecimals:0,tickFormatter:function(a){return UI.format.addUnit(UI.format.number(a),"%")},tickLength:0,min:0,max:100},amount:{name:"amount",color:"black",tickColor:0,tickDecimals:0,tickFormatter:function(a){return UI.format.number(a)},tickLength:0,min:0},bytespersec:{name:"bytespersec",color:"black",tickColor:0,tickDecimals:1,tickFormatter:function(a){return UI.format.bytes(a,!0)},tickLength:0,ticks:function(a){var c=0.3*Math.sqrt($(".graph").first().height()),c=(a.max- | ||||
| a.min)/c,b=Math.floor(Math.log(Math.abs(c))/Math.log(1024)),d=c/Math.pow(1024,b),e=-Math.floor(Math.log(d)/Math.LN10),j=a.tickDecimals;null!=j&&e>j&&(e=j);var m=Math.pow(10,-e),d=d/m,g;if(1.5>d)g=1;else if(3>d){if(g=2,2.25<d&&(null==j||e+1<=j))g=2.5,++e}else g=7.5>d?5:10;g=g*m*Math.pow(1024,b);null!=a.minTickSize&&g<a.minTickSize&&(g=a.minTickSize);a.delta=c;a.tickDecimals=Math.max(0,null!=j?j:e);a.tickSize=g;c=[];b=a.tickSize*Math.floor(a.min/a.tickSize);e=0;j=Number.NaN;do m=j,j=b+e*a.tickSize, | ||||
| c.push(j),++e;while(j<a.max&&j!=m);return c},min:0}},xaxes:{time:{name:"time",mode:"time",timezone:"browser",ticks:5}}},draggable:function(a){a.attr("draggable",!0);a.on("dragstart",function(a){$(this).css("opacity",0.4).data("dragstart",{click:{x:a.originalEvent.pageX,y:a.originalEvent.pageY},ele:{x:this.offsetLeft,y:this.offsetTop}})}).on("dragend",function(a){var b=$(this).data("dragstart"),d=b.ele.x-b.click.x+a.originalEvent.pageX,a=b.ele.y-b.click.y+a.originalEvent.pageY;$(this).css({opacity:1, | ||||
| top:a,left:d,right:"auto",bottom:"auto"})});a.parent().on("dragleave",function(){})},format:{time:function(a,c){var b=new Date(1E3*a),d=[];d.push(("0"+b.getHours()).slice(-2));d.push(("0"+b.getMinutes()).slice(-2));"short"!=c&&d.push(("0"+b.getSeconds()).slice(-2));return d.join(":")},date:function(a,c){var b=new Date(1E3*a),d="Sun Mon Tue Wed Thu Fri Sat".split(" "),e=[];"long"==c&&e.push(d[b.getDay()]);e.push(("0"+b.getDate()).slice(-2));e.push("Jan Feb Mar Apr May Jun Jul Aug Sep Oct Nov Dec".split(" ")[b.getMonth()]); | ||||
| "short"!=c&&e.push(b.getFullYear());return e.join(" ")},dateTime:function(a,c){return UI.format.date(a,c)+", "+UI.format.time(a,c)},duration:function(a){var c=[0.001,1E3,60,60,24,7,1E9],b="ms sec min hr day week".split(" "),d={},e;for(e in b){var a=a/c[e],j=Math.round(a%c[Number(e)+1]);d[b[e]]=j;a-=j}var m;for(e=b.length-1;0<=e;e--)if(0<d[b[e]]){m=b[e];break}c=$("<span>");switch(m){case "week":c.append(UI.format.addUnit(d.week,"wks, ")).append(UI.format.addUnit(d.day,"days"));break;case "day":c.append(UI.format.addUnit(d.day, | ||||
| "days, ")).append(UI.format.addUnit(d.hr,"hrs"));break;default:c.append([("0"+d.hr).slice(-2),("0"+d.min).slice(-2),("0"+d.sec).slice(-2)+(d.ms?"."+d.ms:"")].join(":"))}return c[0].innerHTML},number:function(a){if(isNaN(Number(a))||0==a)return a;var c=Math.pow(10,3-Math.floor(Math.log(a)/Math.LN10)-1),a=Math.round(a*c)/c;if(1E4<a){number=a.toString().split(".");for(a=/(\d+)(\d{3})/;a.test(number[0]);)number[0]=number[0].replace(a,"$1 $2");a=number.join(".")}return a},status:function(a){var c=$("<span>"); | ||||
| if("undefined"==typeof a.online)return c.text("Unknown, checking.."),"undefined"!=typeof a.error&&c.text(a.error),c;switch(a.online){case -1:c.text("Enabling");break;case 0:c.text("Unavailable").addClass("red");break;case 1:c.text("Active").addClass("green");break;case 2:c.text("Standby").addClass("orange");break;default:c.text(a.online)}"error"in a&&c.text(a.error);return c},capital:function(a){return a.charAt(0).toUpperCase()+a.substring(1)},addUnit:function(a,c){var b=$("<span>").html(a);b.append($("<span>").addClass("unit").html(c)); | ||||
| return b[0].innerHTML},bytes:function(a,c){var b="bytes KiB MiB GiB TiB PiB".split(" ");if(0==a)unit=b[0];else{var d=Math.floor(Math.log(Math.abs(a))/Math.log(1024));0>d?unit=b[0]:(a/=Math.pow(1024,d),unit=b[d])}return UI.format.addUnit(UI.format.number(a),unit+(c?"/s":""))}},navto:function(a,c){var b=location.hash,d=b.split("@");d[0]=[mist.user.name,mist.user.host].join("&");d[1]=[a,c].join("&");"undefined"!=typeof screenlog&&screenlog.navto(d[1]);location.hash=d.join("@");location.hash==b&&$(window).trigger("hashchange")}, | ||||
| showTab:function(a,c){var b=UI.elements.main;if(mist.user.loggedin&&!("ui_settings"in mist.data))b.html("Loading.."),mist.send(function(){UI.showTab(a,c)},{ui_settings:!0});else{var d=UI.elements.menu.css("opacity","1").find(".button").filter(function(){if($(this).find(".plain").text()==a)return!0});0<d.length&&(UI.elements.menu.find(".button.active").removeClass("active"),d.addClass("active"));UI.elements.secondary_menu.html("");UI.interval.clear();b.html($("<h2>").text(a));switch(a){case "Login":if(mist.user.loggedin){UI.navto("Overview"); | ||||
| break}UI.elements.menu.css("opacity","0");UI.elements.connection.status.text("Disconnected").removeClass("green").addClass("red");b.append(UI.buildUI([{type:"help",help:"Please provide your account details.<br>You were asked to set these when MistController was started for the first time. If you did not yet set any account details, log in with your desired credentials to create a new account."},{label:"Host",help:"Url location of the MistServer API. Generally located at http://MistServerIP:4242/api", | ||||
| "default":"http://localhost:4242/api",pointer:{main:mist.user,index:"host"}},{label:"Username",help:"Please enter your username here.",validate:["required"],pointer:{main:mist.user,index:"name"}},{label:"Password",type:"password",help:"Please enter your password here.",validate:["required"],pointer:{main:mist.user,index:"password"}},{type:"buttons",buttons:[{label:"Login",type:"save","function":function(){mist.send(function(){UI.navto("Overview")})}}]}]));break;case "Create a new account":UI.elements.menu.css("visibility", | ||||
| "hidden");b.append($("<p>").text("No account has been created yet in the MistServer at ").append($("<i>").text(mist.user.host)).append("."));b.append(UI.buildUI([{type:"buttons",buttons:[{label:"Select other host",type:"cancel",css:{"float":"left"},"function":function(){UI.navto("Login")}}]},{type:"custom",custom:$("<br>")},{label:"Desired username",type:"str",validate:["required"],help:"Enter your desired username. In the future, you will need this to access the Management Interface.",pointer:{main:mist.user, | ||||
| index:"name"}},{label:"Desired password",type:"password",validate:["required",function(a,c){$(".match_password").not($(c)).trigger("change");return false}],help:"Enter your desired password. In the future, you will need this to access the Management Interface.",pointer:{main:mist.user,index:"password"},classes:["match_password"]},{label:"Repeat password",type:"password",validate:["required",function(a,c){return a!=$(".match_password").not($(c)).val()?{msg:'The fields "Desired password" and "Repeat password" do not match.', | ||||
| classes:["red"]}:false}],help:"Repeat your desired password.",classes:["match_password"]},{type:"buttons",buttons:[{type:"save",label:"Create new account","function":function(){mist.send(function(){UI.navto("Account created")},{authorize:{new_username:mist.user.name,new_password:mist.user.password}})}}]}]));break;case "Account created":UI.elements.menu.css("visibility","hidden");b.append($("<p>").text("Your account has been created succesfully.")).append(UI.buildUI([{type:"text",text:"Would you like to enable all (currently) available protocols with their default settings?"}, | ||||
| {type:"buttons",buttons:[{label:"Enable protocols",type:"save","function":function(){if(mist.data.config.protocols)b.append("Unable to enable all protocols as protocol settings already exist.<br>");else{b.append("Retrieving available protocols..<br>");mist.send(function(a){var c=[],d;for(d in a.capabilities.connectors)if(a.capabilities.connectors[d].required)b.append('Could not enable protocol "'+d+'" because it has required settings.<br>');else{c.push({connector:d});b.append('Enabled protocol "'+ | ||||
| d+'".<br>')}b.append("Saving protocol settings..<br>");mist.send(function(){b.append("Protocols enabled. Redirecting..");setTimeout(function(){UI.navto("Overview")},5E3)},{config:{protocols:c}})},{capabilities:true})}}},{label:"Skip",type:"cancel","function":function(){UI.navto("Overview")}}]}]));break;case "Overview":var f=$("<span>").text("Loading.."),k=$("<span>"),m=$("<span>"),e=$("<span>");b.append(UI.buildUI([{type:"help",help:"You can find most basic information about your MistServer here.<br>You can also set the debug level and force a save to the config.json file that MistServer uses to save your settings. "}, | ||||
| {type:"span",label:"Version",pointer:{main:mist.data.config,index:"version"}},{type:"span",label:"Version check",value:f,LTSonly:!0},{type:"span",label:"Server time",value:e},{type:"span",label:"Current streams",value:k},{type:"span",label:"Current connections",value:m},$("<br>"),{type:"str",label:"Human readable name",pointer:{main:mist.data.config,index:"name"},help:"You can name your MistServer here for personal use. You???ll still need to set host name within your network yourself."},{type:"debug", | ||||
| d+'".<br>')}b.append("Saving protocol settings..<br>");mist.send(function(){b.append("Protocols enabled. Redirecting..");setTimeout(function(){UI.navto("Overview")},5E3)},{config:{protocols:c}})},{capabilities:true})}}},{label:"Skip",type:"cancel","function":function(){UI.navto("Overview")}}]}]));break;case "Overview":var e=$("<span>").text("Loading.."),j=$("<span>"),m=$("<span>"),g=$("<span>");b.append(UI.buildUI([{type:"help",help:"You can find most basic information about your MistServer here.<br>You can also set the debug level and force a save to the config.json file that MistServer uses to save your settings. "}, | ||||
| {type:"span",label:"Version",pointer:{main:mist.data.config,index:"version"}},{type:"span",label:"Version check",value:e,LTSonly:!0},{type:"span",label:"Server time",value:g},{type:"span",label:"Current streams",value:j},{type:"span",label:"Current connections",value:m},$("<br>"),{type:"str",label:"Human readable name",pointer:{main:mist.data.config,index:"name"},help:"You can name your MistServer here for personal use. You???ll still need to set host name within your network yourself."},{type:"debug", | ||||
| label:"Debug level",pointer:{main:mist.data.config,index:"debug"},help:"You can set the amount of debug information MistServer saves in the log. A full reboot of MistServer is required before some components of MistServer can post debug information."},{type:"checkbox",label:"Force JSON file save",pointer:{main:mist.data,index:"save"},help:"Tick the box in order to force an immediate save to the config.json MistServer uses to save your settings. Saving will otherwise happen upon closing MistServer. Don???t forget to press save after ticking the box."}, | ||||
| {type:"buttons",buttons:[{type:"save",label:"Save","function":function(){var a={config:mist.data.config};if(mist.data.save)a.save=mist.data.save;mist.send(function(){UI.navto("Overview")},a)}}]}]));if(mist.data.LTS){var p=function(){var a=mist.stored.get().update||{};"uptodate"in a?a.error?f.addClass("red").text(a.error):a.uptodate?f.text("Your version is up to date.").addClass("green"):f.addClass("red").text("Version outdated!").append($("<button>").text("Update").css({"font-size":"1em","margin-left":"1em"}).click(function(){mist.send(function(){UI.navto("Overview")}, | ||||
| {autoupdate:true})})):f.text("Unknown")};if(!mist.stored.get().update||864E5<(new Date).getTime()-mist.stored.get().update.lastchecked){var o=mist.stored.get().update||{};o.lastchecked=(new Date).getTime();mist.send(function(a){mist.stored.set("update",$.extend(true,o,a.update));p()},{checkupdate:!0})}else p()}else f.text("");var i=function(){mist.send(function(){v()},{totals:{fields:["clients"],start:-10},active_streams:true})},v=function(){k.text(("active_streams"in mist.data?mist.data.active_streams? | ||||
| mist.data.active_streams.length:0:"?")+" active, "+(mist.data.streams?Object.keys(mist.data.streams).length:0)+" configured");if("totals"in mist.data&&"all_streams"in mist.data.totals)var a=mist.data.totals.all_streams.all_protocols.clients,a=a.length?UI.format.number(a[a.length-1][1]):0;else a="Loading..";m.text(a);e.text(UI.format.dateTime(mist.data.config.time,"long"))};i();v();UI.interval.set(i,3E4);break;case "Protocols":if("undefined"==typeof mist.data.capabilities){mist.send(function(){UI.navto(a)}, | ||||
| {capabilities:!0});b.append("Loading..");break}var g=$("<tbody>");b.append(UI.buildUI([{type:"help",help:"You can find an overview of all the protocols and their relevant information here. You can add, edit or delete protocols."}])).append($("<button>").text("New protocol").click(function(){UI.navto("Edit Protocol")})).append($("<table>").html($("<thead>").html($("<tr>").html($("<th>").text("Protocol")).append($("<th>").text("Status")).append($("<th>").text("Settings")).append($("<th>")))).append(g)); | ||||
| var l=function(){function a(b){var c=mist.data.capabilities.connectors[b.connector];if(!c)return"";var d=[],e=["required","optional"],f;for(f in e)for(var g in c[e[f]])b[g]&&b[g]!=""?d.push(g+": "+b[g]):c[e[f]][g]["default"]&&d.push(g+": "+c[e[f]][g]["default"]);return $("<span>").addClass("description").text(d.join(", "))}g.html("");for(var b in mist.data.config.protocols){var c=mist.data.config.protocols[b];g.append($("<tr>").data("index",b).append($("<td>").text(c.connector)).append($("<td>").html(UI.format.status(c))).append($("<td>").html(a(c))).append($("<td>").css("text-align", | ||||
| "right").html($("<button>").text("Edit").click(function(){UI.navto("Edit Protocol",$(this).closest("tr").data("index"))})).append($("<button>").text("Delete").click(function(){var a=$(this).closest("tr").data("index");if(confirm('Are you sure you want to delete the protocol "'+mist.data.config.protocols[a].connector+'"?')){mist.data.config.protocols.splice(a,1);mist.send(function(){UI.navto("Protocols")},{config:mist.data.config})}}))))}};l();UI.interval.set(function(){mist.send(function(){l()})}, | ||||
| 3E4);break;case "Edit Protocol":if("undefined"==typeof mist.data.capabilities){mist.send(function(){UI.navto(a,c)},{capabilities:!0});b.append("Loading..");break}var j=!1;""!=c&&0<=c&&(j=!0);var h={},r;for(r in mist.data.config.protocols)h[mist.data.config.protocols[r].connector]=1;var P=function(a){var b=mist.data.capabilities.connectors[a],d=mist.convertBuildOptions(b,n);d.push({type:"hidden",pointer:{main:n,index:"connector"},value:a});d.push({type:"buttons",buttons:[{type:"save",label:"Save", | ||||
| "function":function(){if(j)mist.data.config.protocols[c]=n;else{if(!mist.data.config.protocols)mist.data.config.protocols=[];mist.data.config.protocols.push(n)}mist.send(function(){UI.navto("Protocols")},{config:mist.data.config})}},{type:"cancel",label:"Cancel","function":function(){UI.navto("Protocols")}}]});if("deps"in b&&b.deps!=""){$t=$("<span>").text("Dependencies:");$ul=$("<ul>");$t.append($ul);if(typeof b.deps=="string")b.deps=b.deps.split(", ");for(var e in b.deps){a=$("<li>").text(b.deps[e]+ | ||||
| " ");$ul.append(a);typeof h[b.deps[e]]!="undefined"||typeof h[b.deps[e]+".exe"]!="undefined"?a.append($("<span>").addClass("green").text("(Configured)")):a.append($("<span>").addClass("red").text("(Not yet configured)"))}d.unshift({type:"text",text:$t[0].innerHTML})}return UI.buildUI(d)},h={};for(r in mist.data.config.protocols)h[mist.data.config.protocols[r].connector]=1;if(j){var q=mist.data.config.protocols[c],n=q;b.find("h2").append(' "'+q.connector+'"');b.append(P(q.connector))}else{b.html($("<h2>").text("New Protocol")); | ||||
| var n={},t=[];for(r in mist.data.capabilities.connectors)t.push([r,r]);var Q=$("<span>");b.append(UI.buildUI([{label:"Protocol",type:"select",select:t,"function":function(){Q.html(P($(this).getval()))}}])).append(Q)}break;case "Streams":if(!("capabilities"in mist.data)){b.html("Loading..");mist.send(function(){UI.navto(a)},{capabilities:!0});break}g=$("<tbody>").append($("<tr>").append("<td>").attr("colspan",6).text("Loading.."));q=$("<table>").html($("<thead>").html($("<tr>").html($("<th>").text("Stream name").attr("data-sort-type", | ||||
| "string").addClass("sorting-asc")).append($("<th>").text("Source").attr("data-sort-type","string")).append($("<th>").text("Status").attr("data-sort-type","int")).append($("<th>").css("text-align","right").text("Connections").attr("data-sort-type","int")).append($("<th>")).append($("<th>")))).append(g);b.append(UI.buildUI([{type:"help",help:"Here you can create, edit or delete new and existing streams. Immidiately go to the stream preview or view the information available about the stream with the info button."}])).append($("<button>").text("New stream").click(function(){UI.navto("Edit Stream")})).append(q); | ||||
| q.stupidtable();var D=function(){var a=[],b;for(b in mist.data.active_streams)a.push({streams:[mist.data.active_streams[b]],fields:["clients"],start:-2});mist.send(function(){$.extend(true,y,mist.data.streams);var a=0;g.html("");if(mist.data.LTS)for(a in mist.data.active_streams){var b=mist.data.active_streams[a].split("+");if(!(b.length<2)&&b[0]in mist.data.streams){b=R(mist.data.active_streams[a],mist.data.streams[b[0]]);b.online=1;y[mist.data.active_streams[a]]=b}}b=Object.keys(y);b.sort();for(var c in b){var d= | ||||
| b[c],e;e=d in mist.data.streams?mist.data.streams[d]:y[d];var f=$("<td>").css("text-align","right").html($("<span>").addClass("description").text("Loading..")),h=0;if(typeof mist.data.totals!="undefined"&&typeof mist.data.totals[d]!="undefined"){var i=mist.data.totals[d].all_protocols.clients,h=0;if(i.length){for(a in i)h=h+i[a][1];h=Math.round(h/i.length)}}f.html(UI.format.number(h));if(h==0&&e.online==1)e.online=2;h=$("<td>").css("text-align","right").css("white-space","nowrap");(!("ischild"in e)|| | ||||
| {type:"buttons",buttons:[{type:"save",label:"Save","function":function(){var a={config:mist.data.config};if(mist.data.save)a.save=mist.data.save;mist.send(function(){UI.navto("Overview")},a)}}]}]));if(mist.data.LTS){var p=function(){var a=mist.stored.get().update||{};"uptodate"in a?a.error?e.addClass("red").text(a.error):a.uptodate?e.text("Your version is up to date.").addClass("green"):e.addClass("red").text("Version outdated!").append($("<button>").text("Update").css({"font-size":"1em","margin-left":"1em"}).click(function(){confirm("Are you sure you want to execute a rolling update?")&& | ||||
| mist.send(function(){UI.navto("Overview")},{autoupdate:true})})):e.text("Unknown")};if(!mist.stored.get().update||864E5<(new Date).getTime()-mist.stored.get().update.lastchecked){var n=mist.stored.get().update||{};n.lastchecked=(new Date).getTime();mist.send(function(a){mist.stored.set("update",$.extend(true,n,a.update));p()},{checkupdate:!0})}else p()}else e.text("");var h=function(){mist.send(function(){s()},{totals:{fields:["clients"],start:-10},active_streams:true})},s=function(){j.text(("active_streams"in | ||||
| mist.data?mist.data.active_streams?mist.data.active_streams.length:0:"?")+" active, "+(mist.data.streams?Object.keys(mist.data.streams).length:0)+" configured");if("totals"in mist.data&&"all_streams"in mist.data.totals)var a=mist.data.totals.all_streams.all_protocols.clients,a=a.length?UI.format.number(a[a.length-1][1]):0;else a="Loading..";m.text(a);g.text(UI.format.dateTime(mist.data.config.time,"long"))};h();s();UI.interval.set(h,3E4);break;case "Protocols":if("undefined"==typeof mist.data.capabilities){mist.send(function(){UI.navto(a)}, | ||||
| {capabilities:!0});b.append("Loading..");break}var f=$("<tbody>");b.append(UI.buildUI([{type:"help",help:"You can find an overview of all the protocols and their relevant information here. You can add, edit or delete protocols."}])).append($("<button>").text("New protocol").click(function(){UI.navto("Edit Protocol")})).append($("<table>").html($("<thead>").html($("<tr>").html($("<th>").text("Protocol")).append($("<th>").text("Status")).append($("<th>").text("Settings")).append($("<th>")))).append(f)); | ||||
| var i=function(){function a(c){var b=mist.data.capabilities.connectors[c.connector];if(!b)return"";var d=[],g=["required","optional"],f;for(f in g)for(var e in b[g[f]])c[e]&&c[e]!=""?d.push(e+": "+c[e]):b[g[f]][e]["default"]&&d.push(e+": "+b[g[f]][e]["default"]);return $("<span>").addClass("description").text(d.join(", "))}f.html("");for(var c in mist.data.config.protocols){var b=mist.data.config.protocols[c];f.append($("<tr>").data("index",c).append($("<td>").text(b.connector)).append($("<td>").html(UI.format.status(b))).append($("<td>").html(a(b))).append($("<td>").css("text-align", | ||||
| "right").html($("<button>").text("Edit").click(function(){UI.navto("Edit Protocol",$(this).closest("tr").data("index"))})).append($("<button>").text("Delete").click(function(){var a=$(this).closest("tr").data("index");if(confirm('Are you sure you want to delete the protocol "'+mist.data.config.protocols[a].connector+'"?')){mist.data.config.protocols.splice(a,1);mist.send(function(){UI.navto("Protocols")},{config:mist.data.config})}}))))}};i();UI.interval.set(function(){mist.send(function(){i()})}, | ||||
| 3E4);break;case "Edit Protocol":if("undefined"==typeof mist.data.capabilities){mist.send(function(){UI.navto(a,c)},{capabilities:!0});b.append("Loading..");break}var l=!1;""!=c&&0<=c&&(l=!0);var k={},q;for(q in mist.data.config.protocols)k[mist.data.config.protocols[q].connector]=1;var P=function(a){var b=mist.data.capabilities.connectors[a],d=mist.convertBuildOptions(b,o);d.push({type:"hidden",pointer:{main:o,index:"connector"},value:a});d.push({type:"buttons",buttons:[{type:"save",label:"Save", | ||||
| "function":function(){if(l)mist.data.config.protocols[c]=o;else{if(!mist.data.config.protocols)mist.data.config.protocols=[];mist.data.config.protocols.push(o)}mist.send(function(){UI.navto("Protocols")},{config:mist.data.config})}},{type:"cancel",label:"Cancel","function":function(){UI.navto("Protocols")}}]});if("deps"in b&&b.deps!=""){$t=$("<span>").text("Dependencies:");$ul=$("<ul>");$t.append($ul);if(typeof b.deps=="string")b.deps=b.deps.split(", ");for(var e in b.deps){a=$("<li>").text(b.deps[e]+ | ||||
| " ");$ul.append(a);typeof k[b.deps[e]]!="undefined"||typeof k[b.deps[e]+".exe"]!="undefined"?a.append($("<span>").addClass("green").text("(Configured)")):a.append($("<span>").addClass("red").text("(Not yet configured)"))}d.unshift({type:"text",text:$t[0].innerHTML})}return UI.buildUI(d)},k={};for(q in mist.data.config.protocols)k[mist.data.config.protocols[q].connector]=1;if(l){var r=mist.data.config.protocols[c],o=r;b.find("h2").append(' "'+r.connector+'"');b.append(P(r.connector))}else{b.html($("<h2>").text("New Protocol")); | ||||
| var o={},u=[];for(q in mist.data.capabilities.connectors)u.push([q,q]);var Q=$("<span>");b.append(UI.buildUI([{label:"Protocol",type:"select",select:u,"function":function(){Q.html(P($(this).getval()))}}])).append(Q)}break;case "Streams":if(!("capabilities"in mist.data)){b.html("Loading..");mist.send(function(){UI.navto(a)},{capabilities:!0});break}f=$("<tbody>").append($("<tr>").append("<td>").attr("colspan",6).text("Loading.."));d=$("<table>").html($("<thead>").html($("<tr>").html($("<th>").text("Stream name").attr("data-sort-type", | ||||
| "string").addClass("sorting-asc")).append($("<th>").text("Source").attr("data-sort-type","string")).append($("<th>").text("Status").attr("data-sort-type","int")).append($("<th>").css("text-align","right").text("Connections").attr("data-sort-type","int")).append($("<th>")).append($("<th>")))).append(f);b.append(UI.buildUI([{type:"help",help:"Here you can create, edit or delete new and existing streams. Immidiately go to the stream preview or view the information available about the stream with the info button."}])).append($("<button>").text("New stream").click(function(){UI.navto("Edit Stream")})).append(d); | ||||
| d.stupidtable();var B=function(){var a=[],b;for(b in mist.data.active_streams)a.push({streams:[mist.data.active_streams[b]],fields:["clients"],start:-2});mist.send(function(){$.extend(true,w,mist.data.streams);var a=0;f.html("");if(mist.data.LTS)for(a in mist.data.active_streams){var b=mist.data.active_streams[a].split("+");if(!(b.length<2)&&b[0]in mist.data.streams){b=R(mist.data.active_streams[a],mist.data.streams[b[0]]);b.online=1;w[mist.data.active_streams[a]]=b}}b=Object.keys(w);b.sort();for(var c in b){var d= | ||||
| b[c],e;e=d in mist.data.streams?mist.data.streams[d]:w[d];var g=$("<td>").css("text-align","right").html($("<span>").addClass("description").text("Loading..")),h=0;if(typeof mist.data.totals!="undefined"&&typeof mist.data.totals[d]!="undefined"){var i=mist.data.totals[d].all_protocols.clients,h=0;if(i.length){for(a in i)h=h+i[a][1];h=Math.round(h/i.length)}}g.html(UI.format.number(h));if(h==0&&e.online==1)e.online=2;h=$("<td>").css("text-align","right").css("white-space","nowrap");(!("ischild"in e)|| | ||||
| !e.ischild)&&h.html($("<button>").text("Edit").click(function(){UI.navto("Edit Stream",$(this).closest("tr").data("index"))})).append($("<button>").text("Delete").click(function(){var a=$(this).closest("tr").data("index");if(confirm('Are you sure you want to delete the stream "'+a+'"?')){delete mist.data.streams[a];var b={};mist.data.LTS?b.deletestream=[a]:b.streams=mist.data.streams;mist.send(function(){UI.navto("Streams")},b)}}));i=$("<span>").text(d);e.ischild&&i.css("padding-left","1em");var j= | ||||
| UI.format.status(e),S=$("<button>").text("Preview").click(function(){UI.navto("Preview",$(this).closest("tr").data("index"))});if(e.filesfound){j.html("");S="";f.html("")}g.append($("<tr>").data("index",d).html($("<td>").html(i).attr("title",d).addClass("overflow_ellipsis")).append($("<td>").text(e.source).attr("title",e.source).addClass("description").addClass("overflow_ellipsis").css("max-width","20em")).append($("<td>").data("sort-value",e.online).html(j)).append(f).append($("<td>").html(S)).append(h)); | ||||
| a++}},{totals:a,active_streams:true})},y=$.extend(!0,{},mist.data.streams),R=function(a,b){var c=$.extend({},b);delete c.meta;delete c.error;c.online=2;c.name=a;c.ischild=true;return c};if(mist.data.LTS){var B=0,E=0;for(i in mist.data.streams)q=mist.data.capabilities.inputs.Folder||mist.data.capabilities.inputs["Folder.exe"],mist.inputMatch(q.source_match,mist.data.streams[i].source)&&(y[i].source+="*",mist.send(function(a,b){var c=b.stream,d;for(d in a.browse.files)for(var e in mist.data.capabilities.inputs)if(!(e.indexOf("Buffer")>= | ||||
| 0||e.indexOf("Folder")>=0)&&mist.inputMatch(mist.data.capabilities.inputs[e].source_match,"/"+a.browse.files[d])){var f=c+"+"+a.browse.files[d];y[f]=R(f,mist.data.streams[c]);y[f].source=mist.data.streams[c].source+a.browse.files[d]}a.browse.files.length?y[c].filesfound=true:mist.data.streams[c].filesfound=false;E++;if(B==E){mist.send(function(){D()},{active_streams:true});UI.interval.set(function(){D()},1E4)}},{browse:mist.data.streams[i].source},{stream:i}),B++);0==B&&(mist.send(function(){D()}, | ||||
| {active_streams:!0}),UI.interval.set(function(){D()},3E4))}else mist.send(function(){D()},{active_streams:!0}),UI.interval.set(function(){D()},1E4);break;case "Edit Stream":if("undefined"==typeof mist.data.capabilities){mist.send(function(){UI.navto(a,c)},{capabilities:!0});b.append("Loading..");break}j=!1;""!=c&&(j=!0);j?(q=c,n=mist.data.streams[q],b.find("h2").append(' "'+q+'"')):(b.html($("<h2>").text("New Stream")),n={});i=[];for(r in mist.data.capabilities.inputs)i.push(mist.data.capabilities.inputs[r].source_match); | ||||
| var G=$("<div>");b.append(UI.buildUI([{label:"Stream name",type:"str",validate:["required","streamname"],pointer:{main:n,index:"name"},help:"Set the name this stream will be recognised by for players and/or stream pushing."},{label:"Source",type:"browse",validate:["required"],filetypes:i,pointer:{main:n,index:"source"},help:'Set the stream source.<table><tr><td>VoD:</td><td>You can browse to the file or folder as a source or simply enter the path to the file.</td></tr><tr><td>Live:</td><td>You\'ll need to enter "push://IP" with the IP of the machine pushing towards MistServer.<br>You can use "push://" to accept any source.</td></tr><tr><td>(Pro only)</td><td>Use "push://(IP)@password" to set a password protection for pushes.</td></tr></table>If you\'re unsure how to set the source properly, please view our Live pushing guide at the tools section.', | ||||
| "function":function(){var a=$(this).val();if(a!=""){var b=null,c;for(c in mist.data.capabilities.inputs)if(typeof mist.data.capabilities.inputs[c].source_match!="undefined"&&mist.inputMatch(mist.data.capabilities.inputs[c].source_match,a)){b=c;break}if(b===null)G.html($("<h3>").text("Unrecognized input").addClass("red")).append($("<span>").text("Please edit the stream source.").addClass("red"));else{a=mist.data.capabilities.inputs[b];G.html($("<h3>").text(a.name+" Input options"));a=mist.convertBuildOptions(a, | ||||
| n);G.append(UI.buildUI(a))}}}},$("<br>"),{type:"custom",custom:G},$("<br>"),$("<h3>").text("Encryption"),{type:"help",help:"To enable encryption, the licence acquisition url must be entered, as well as either the content key or the key ID and seed.<br>Unsure how you should fill in your encryption or missing your preferred encryption? Please contact us."},{label:"License acquisition url",type:"str",LTSonly:!0,pointer:{main:n,index:"la_url"}},$("<br>"),{label:"Content key",type:"str",LTSonly:!0,pointer:{main:n, | ||||
| index:"contentkey"}},{type:"text",text:" - or - "},{label:"Key ID",type:"str",LTSonly:!0,pointer:{main:n,index:"keyid"}},{label:"Key seed",type:"str",LTSonly:!0,pointer:{main:n,index:"keyseed"}},{type:"buttons",buttons:[{type:"cancel",label:"Cancel","function":function(){UI.navto("Streams")}},{type:"save",label:"Save","function":function(){if(!mist.data.streams)mist.data.streams={};mist.data.streams[n.name]=n;c!=n.name&&delete mist.data.streams[c];var a={};if(mist.data.LTS){a.addstream={};a.addstream[n.name]= | ||||
| n;if(c!=n.name)a.deletestream=[c]}else a.streams=mist.data.streams;mist.send(function(){delete mist.data.streams[n.name].online;delete mist.data.streams[n.name].error;UI.navto("Streams")},a)}}]}]));j&&(i={streams:{}},i.streams[q]=mist.data.streams[q].limits,b.append($("<h3>").text("Limits")).append(UI.buildLimits(i,a,c)));break;case "Preview":if(""==c){b.append("Loading..");var J=function(c){var d={};c.sort();b.html($("<h2>").text(a)).append(UI.buildUI([{label:"Select a stream",type:"select",select:c, | ||||
| pointer:{main:d,index:"stream"}},{type:"buttons",buttons:[{type:"save",label:"Go","function":function(){UI.navto(a,d.stream)}}]}]));UI.elements.secondary_menu.html("").append($("<a>").addClass("button").addClass("active").text("Choose stream").click(function(){UI.navto("Preview")}));var e=$("<div>").addClass("preview_icons");b.append($("<span>").addClass("description").text("Or, click a stream from the list below.")).append(e);for(var f in c){var g=c[f],h="";if(g.indexOf("+")>-1){h=g.split("+");h= | ||||
| mist.data.streams[h[0]].source+h[1]}else h=mist.data.streams[g].source;e.append($("<button>").append($("<span>").text(g)).append($("<span>").addClass("description").text(h)).attr("title",g).attr("data-stream",g).click(function(){UI.navto("Preview",$(this).attr("data-stream"))}))}};if(mist.data.LTS){if("undefined"==typeof mist.data.capabilities){mist.send(function(){UI.navto(a,c)},{capabilities:!0});b.append("Loading..");break}E=B=0;t={};for(i in mist.data.streams)q=mist.data.capabilities.inputs.Folder|| | ||||
| mist.data.capabilities.inputs["Folder.exe"],mist.inputMatch(q.source_match,mist.data.streams[i].source)&&(mist.send(function(a,b){var c=b.stream,d;for(d in mist.data.browse.files)for(var e in mist.data.capabilities.inputs)e.indexOf("Buffer")>=0||e.indexOf("Folder")>=0||mist.inputMatch(mist.data.capabilities.inputs[e].source_match,"/"+mist.data.browse.files[d])&&(t[c+"+"+mist.data.browse.files[d]]=true);E++;B==E&&mist.send(function(){for(var a in mist.data.active_streams){var b=mist.data.active_streams[a].split("+"); | ||||
| b.length>1&&b[0]in mist.data.streams&&(t[mist.data.active_streams[a]]=true)}t=Object.keys(t);t=t.concat(Object.keys(mist.data.streams));t.sort();J(t)},{active_streams:true})},{browse:mist.data.streams[i].source},{stream:i}),B++);0==B&&mist.send(function(){var c=[],d;for(d in mist.data.active_streams){var e=mist.data.active_streams[d].split("+");e.length>1&&e[0]in mist.data.streams&&(c[mist.data.active_streams[d]]=true)}mist.data.streams&&(c=c.concat(Object.keys(mist.data.streams)));if(c.length==0)b.html($("<h2>").text(a)).append("Please set up a stream first."); | ||||
| else{c.sort();J(c)}},{active_streams:!0})}else J(Object.keys(mist.data.streams));break}b.find("h2").append(' of "'+c+'"');i=":8080";for(r in mist.data.config.protocols)if(q=mist.data.config.protocols[r],"HTTP"==q.connector||"HTTP.exe"==q.connector)i=q.port?":"+q.port:":8080";var C={},q=$("<span>").hide();C["Embed urls"]=q;b.append(q);var w="",d=mist.user.host.match(/(https?)\:\/\/([^:\/]+)\:(\d+)?/i);null!=d&&(w=d[2]);var F="http://"+w+i+"/",x={},H=function(a){var b=["div"],d='\n   <script src="'+ | ||||
| F+"embed_"+c+'.js"><\/script>\n';a.autoplay&&b.push("data-autoplay");a.forceprotocol&&a.forceprotocol!=""&&b.push('data-forcetype="'+a.forceprotocol+'"');a.urlappend&&a.urlappend!=""&&b.push('data-urlappend="'+a.urlappend.replace(/\"/g,'\\"')+'"');return"<"+b.join(" ")+">"+d+"</div>"},T=$("<span>");q.append($("<h3>").text("Embed urls")).append(UI.buildUI([{label:"Embedable script",type:"str",value:F+"embed_"+c+".js",readonly:!0},{label:"Stream info script",type:"str",value:F+"info_"+c+".js",readonly:!0}, | ||||
| {label:"Autodetect player",type:"str",value:F+c+".html",readonly:!0,qrcode:!0},$("<h3>").text("Embed code"),{label:"Embed code",type:"textarea",value:H(x),rows:4,readonly:!0,classes:["embed_code"]},$("<h4>").text("Embed code options").css("margin-top",0),{label:"Autoplay",type:"checkbox",pointer:{main:x,index:"autoplay"},"function":function(){x.autoplay=$(this).getval();$(".embed_code").setval(H(x))}},{label:"Force protocol",type:"select",select:[["","Automatic"]],pointer:{main:x,index:"protocol"}, | ||||
| classes:["embed_code_forceprotocol"],"function":function(){x.forceprotocol=$(this).getval();$(".embed_code").setval(H(x))}},{label:"Video URL addition",type:"str",pointer:{main:x,index:"urlappend"},help:"The embed script will append this string to the video url, useful for sending through params.",classes:["embed_code_forceprotocol"],"function":function(){x.urlappend=$(this).getval();$(".embed_code").setval(H(x))}},$("<h3>").text("Protocol stream urls"),T]));i=$("<span>").append($("<h3>").text("Meta information")).hide(); | ||||
| C["Meta information"]=i;var K=$("<span>");i.append(K);b.append(i);i=$("<span>").hide();C.Preview=i;b.append(i);var z=$("<div>").css("float","left").css("margin-right","1em").attr("data-forcesupportcheck",""),U=$("<div>").css("float","left");i.append(z).append(U);mist.stored.get().autoplay&&z.attr("data-autoplay","");var I=function(){z.text("Loading..");var a=document.createElement("script");a.src=F+"embed_"+c+".js";a.onerror=function(){z.html('Error loading "'+a.src+'".<br>').append($("<button>").text("Try again").click(function(){I()}))}; | ||||
| a.onload=function(){if(typeof mistvideo[c].error!="undefined")z.html(mistvideo[c].error+"<br>").append($("<button>").text("Try again").click(function(){I()}));else{var a=mistvideo[c],b=UI.buildUI([{label:"Protocol stream url",type:"str",readonly:true,value:a.embedded?a.embedded.url:"",qrcode:true},{label:"Autoplay (from now on)",type:"checkbox",value:mist.stored.get().autoplay,"function":function(){mist.stored.set("autoplay",$(this).getval()?1:0)}}]);b.find(".help_container").remove();z.append(b); | ||||
| var d=$("<table>").css("font-size","0.9em").html($("<thead>").html($("<tr>").html($("<th>")).append($("<th>").text("Type")).append($("<th>").text("Priority")).append($("<th>").text("Simul. tracks")).append($("<th>").html("Your browser<br>support"))));U.html(d);b=$("<tbody>");d.append(b);var d=$(".embed_code_forceprotocol"),e=[];d.find(".clear").remove();for(var f in a.source){var g=a.source[f],h=g.type.split("/"),i=h[0];i.length<6&&(i=i.toUpperCase());switch(h.length){case 1:break;case 2:i=UI.format.capital(h[0])+ | ||||
| " v"+h[1];if(h[0]=="flash")switch(h[1]){case "7":i="Progressive ("+i+")";break;case "10":i="RTMP ("+i+")";break;case "11":i="HDS ("+i+")"}break;case 3:switch(h[2]){case "vnd.apple.mpegurl":i=i+" HLS";break;case "vnd.ms-ss":i=i+" Smooth";break;case "mp2t":i=i+" TS";break;default:h[2].length<6&&(h[2]=h[2].toUpperCase());i=i+(" "+h[2]);h[1]!="video"&&(i=i+(" ("+h[1]+")"))}break;default:i=g.type}i=UI.format.capital(i);d.append($("<option>").text(i).val(g.type).addClass("clear"));e.push({label:i,type:"str", | ||||
| value:g.url,readonly:true,qrcode:true});h=$("<tr>");b.append(h);h.html($("<td>").html($("<input>").attr("type","radio").attr("name","protocolforce").change(function(){z.attr("data-forcetype",$(this).val()).html("Loading embed..");I()}).val(g.type))).append($("<td>").text(i)).append($("<td>").text(g.priority)).append($("<td>").text(g.simul_tracks+"/"+g.total_matches)).append($("<td>").text(g.browser_support?"yes":"no"));if(a.embedded&&a.embedded.type==g.type){h.css("outline","1px solid rgba(0,0,0,0.5)"); | ||||
| h.find("input[type=radio]").prop("checked",true)}}T.html(UI.buildUI(e));var j;if(c in mistvideo)j=mistvideo[c].meta;if(j){a=[];a.push({label:"Type",type:"span",value:j.live?"Live":"Pre-recorded (VoD)"});"format"in j&&a.push({label:"Format",type:"span",value:j.format});j.live&&a.push({label:"Buffer window",type:"span",value:UI.format.addUnit(j.buffer_window,"ms")});b={vheader:"Audio",labels:["Codec","Duration","Peak bitrate","Channels","Samplerate"],content:[]};f={vheader:"Video",labels:["Codec","Duration", | ||||
| "Peak bitrate","Size","Framerate"],content:[]};d=Object.keys(j.tracks);d.sort(function(a,b){a=a.split("_").pop();b=b.split("_").pop();return a-b});for(var k in d){e=d[k];g=j.tracks[e];switch(g.type){case "audio":b.content.push({header:"Track "+e.split("_").pop(),body:[g.codec,UI.format.duration((g.lastms-g.firstms)/1E3)+"<br><span class=description>"+UI.format.duration(g.firstms/1E3)+" to "+UI.format.duration(g.lastms/1E3)+"</span>",UI.format.bytes(g.bps,1),g.channels,UI.format.addUnit(UI.format.number(g.rate), | ||||
| "Hz")]});break;case "video":f.content.push({header:"Track "+e.split("_").pop(),body:[g.codec,UI.format.duration((g.lastms-g.firstms)/1E3)+"<br><span class=description>"+UI.format.duration(g.firstms/1E3)+" to "+UI.format.duration(g.lastms/1E3)+"</span>",UI.format.bytes(g.bps,1),UI.format.addUnit(g.width,"x ")+UI.format.addUnit(g.height,"px"),UI.format.addUnit(UI.format.number(g.fpks/1E3),"fps")]})}}j=UI.buildVheaderTable(b).css("width","auto");k=UI.buildVheaderTable(f).css("width","auto");a.push($("<span>").text("Tracks:")); | ||||
| a.push($("<div>").css({display:"flex","flex-flow":"row wrap","justify-content":"center","font-size":"0.9em"}).append(j).append(k));K.html(UI.buildUI(a))}else K.html("No meta information available.")}};z.html("")[0].appendChild(a)};I();var L=UI.elements.secondary_menu;L.html("").append($("<a>").addClass("button").text("Choose stream").click(function(){UI.navto("Preview")})).append($("<span>").addClass("separator"));i=["Preview","Embed urls","Meta information"];h=i[0];for(r in i)q=$("<a>").addClass("button").text(i[r]).click(function(){L.find(".active").removeClass("active"); | ||||
| $(this).addClass("active");for(r in C)C[r].hide();C[$(this).text()].show()}),L.append(q),i[r]==h&&(q.addClass("active"),C[h].show());break;case "Limits":b.append(UI.buildUI([{type:"help",help:"Here you can see an overview of all the limits you currently have. Limits are an LTS only feature and you can simply add new limits by selecting new, by selecting edit you can edit the existing limit, by selecting delete you can delete the existing limit."}]));i={server:mist.data.config.limits,streams:{}};for(r in mist.data.streams)"limits"in | ||||
| mist.data.streams[r]&&(i.streams[r]=mist.data.streams[r].limits);b.append(UI.buildLimits(i,a,c));break;case "Edit Limit":j=!1;""!=c&&(j=!0);n={};i=[{type:"help",help:"Here you can set the limit specifications, soft limits only warn while hard limits restrict access. Please feel free to set up limits according to your preferences."}];"Overview"==UI.returnTab[0]&&(UI.returnTab=["Limits"]);if(j){q=c.split("^");w=a;switch(q[0]){case "server":n=mist.data.config.limits[q[1]];w="For the entire server";break; | ||||
| case "stream":n=mist.data.streams[q[1]].limits[q[2]],w='For the stream "'+q[1]+'"'}i.push({type:"text",text:w})}else{b.html($("<h2>").text("New limit"));var A=[];for(r in mist.data.streams)A.push(r);t=[["server","The entire server"],["stream","The stream:",A]];q={label:"Applies to",type:"radioselect",radioselect:t,pointer:{main:n,index:"applies_to"},LTSonly:!0,validate:["required"]};"Edit Stream"==UI.returnTab[0]&&UI.returnTab[1]&&(q.value=["stream",UI.returnTab[1]],q.readonly=!0);i.push(q)}var i= | ||||
| i.concat([$("<br>"),{label:"Kind",type:"select",select:[["soft","Soft"],["hard","Hard"]],pointer:{main:n,index:"type"},help:"The server will not allow a hard limit to be passed. A soft limit can be used to set alerts.",LTSonly:!0,validate:["required"]},{label:"Value",pointer:{main:n,index:"value"},classes:["limit_value"],LTSonly:!0},{label:"Type",type:"select",select:[["kbps_max","Maximum bandwidth"],["users","Maximum connected users"],["geo","Geo limited"],["host","Host limited"]],pointer:{main:n, | ||||
| index:"name"},LTSonly:!0,validate:["required"],classes:["limit_type"],"function":function(){var a=$(this).getval(),b=$(this).closest(".input_container").find(".limit_value"),c=b.closest("label"),b=b.data("opts");delete b.type;delete b.min;delete b.unit;b.validate=["required"];switch(a){case "kbps_max":b.type="int";b.min=1;b.unit="bytes/s";break;case "users":b.type="int";b.min=1;break;case "geo":b.type="geolimited";break;case "host":b.type="hostlimited"}a=UI.buildUI([b]);c.replaceWith(a.children())}}, | ||||
| {type:"buttons",buttons:[{type:"cancel",label:"Cancel","function":function(){UI.navto(UI.returnTab[0],UI.returnTab[1])}},{type:"save",label:"Save","function":function(){var a={};if(j){var b=c.split("^");switch(b[0]){case "server":a={config:{limits:mist.data.config.limits}};break;case "stream":if(mist.data.LTS){a={addstream:{}};a.addstream[b[1]]=mist.data.streams[b[1]]}else a={streams:mist.data.streams}}}else{b=n.applies_to;delete n.applies_to;switch(b[0]){case "server":if(!mist.data.config.limits)mist.data.config.limits= | ||||
| [];mist.data.config.limits.push(n);a={config:{limits:mist.data.config.limits}};break;case "stream":if(typeof mist.data.streams[b[1]].limits=="undefined")mist.data.streams[b[1]].limits=[];mist.data.streams[b[1]].limits.push(n);if(mist.data.LTS){a={addstream:{}};a.addstream[b[1]]=mist.data.streams[b[1]]}else a={streams:mist.data.streams}}}mist.send(function(){UI.navto(UI.returnTab[0],UI.returnTab[1])},a)}}]}]),u=UI.buildUI(i);b.append(u);u.find(".limit_type").closest("label").after(u.find(".limit_value").closest("label")); | ||||
| u.find(".limit_type").trigger("change");break;case "Triggers":"triggers"in mist.data.config||(mist.data.config.triggers={});g=$("<tbody>");q=$("<table>").html($("<thead>").html($("<tr>").html($("<th>").text("Trigger on").attr("data-sort-type","string").addClass("sorting-asc")).append($("<th>").text("Applies to").attr("data-sort-type","string")).append($("<th>").text("Handler").attr("data-sort-type","string")).append($("<th>")))).append(g);b.append(UI.buildUI([{type:"help",help:"Triggers are the system you can use to react to events that occur inside MistServer. These allow you to block specific users, redirect streams, keep tabs on what is being pushed where, etcetera. For full documentation, please refer to the developer documentation section on the MistServer website."}])).append($("<button>").text("New trigger").click(function(){UI.navto("Edit Trigger")})).append(q); | ||||
| q.stupidtable();i=mist.data.config.triggers;for(r in i)for(w in i[r])g.append($("<tr>").attr("data-index",r+","+w).append($("<td>").text(r)).append($("<td>").text(i[r][w][2].join(", "))).append($("<td>").text(i[r][w][0])).append($("<td>").html($("<button>").text("Edit").click(function(){UI.navto("Edit Trigger",$(this).closest("tr").attr("data-index"))})).append($("<button>").text("Delete").click(function(){var a=$(this).closest("tr").attr("data-index").split(",");if(confirm("Are you sure you want to delete this "+ | ||||
| a[0]+" trigger?")){mist.data.config.triggers[a[0]].splice(a[1],1);mist.data.config.triggers[a[0]].length==0&&delete mist.data.config.triggers[a[0]];mist.send(function(){UI.navto("Triggers")},{config:mist.data.config})}}))));break;case "Edit Trigger":"triggers"in mist.data.config||(mist.data.config.triggers={});c?(c=c.split(","),i=mist.data.config.triggers[c[0]][c[1]],n={triggeron:c[0],appliesto:i[2],url:i[0],async:i[1],"default":i[3]}):(b.html($("<h2>").text("New Trigger")),n={});b.append(UI.buildUI([{label:"Trigger on", | ||||
| pointer:{main:n,index:"triggeron"},help:"For what event this trigger should activate.",type:"select",select:[["SYSTEM_START","SYSTEM_START: after MistServer boot"],["SYSTEM_STOP","SYSTEM_STOP: right before MistServer shutdown"],["SYSTEM_CONFIG","SYSTEM_CONFIG: after MistServer configurations have changed"],["OUTPUT_START","OUTPUT_START: right after the start command has been send to a protocol"],["OUTPUT_STOP","OUTPUT_STOP: right after the close command has been send to a protocol "],["STREAM_ADD", | ||||
| UI.format.status(e),k=$("<button>").text("Preview").click(function(){UI.navto("Preview",$(this).closest("tr").data("index"))});if(e.filesfound){j.html("");k="";g.html("")}f.append($("<tr>").data("index",d).html($("<td>").html(i).attr("title",d).addClass("overflow_ellipsis")).append($("<td>").text(e.source).attr("title",e.source).addClass("description").addClass("overflow_ellipsis").css("max-width","20em")).append($("<td>").data("sort-value",e.online).html(j)).append(g).append($("<td>").html(k)).append(h)); | ||||
| a++}},{totals:a,active_streams:true})},w=$.extend(!0,{},mist.data.streams),R=function(a,b){var c=$.extend({},b);delete c.meta;delete c.error;c.online=2;c.name=a;c.ischild=true;return c};if(mist.data.LTS){var z=0,C=0;for(h in mist.data.streams)r=mist.data.capabilities.inputs.Folder||mist.data.capabilities.inputs["Folder.exe"],mist.inputMatch(r.source_match,mist.data.streams[h].source)&&(w[h].source+="*",mist.send(function(a,b){var c=b.stream,d;for(d in a.browse.files)for(var e in mist.data.capabilities.inputs)if(!(e.indexOf("Buffer")>= | ||||
| 0||e.indexOf("Folder")>=0)&&mist.inputMatch(mist.data.capabilities.inputs[e].source_match,"/"+a.browse.files[d])){var g=c+"+"+a.browse.files[d];w[g]=R(g,mist.data.streams[c]);w[g].source=mist.data.streams[c].source+a.browse.files[d]}"files"in a.browse&&a.browse.files.length?w[c].filesfound=true:mist.data.streams[c].filesfound=false;C++;if(z==C){mist.send(function(){B()},{active_streams:true});UI.interval.set(function(){B()},1E4)}},{browse:mist.data.streams[h].source},{stream:h}),z++);0==z&&(mist.send(function(){B()}, | ||||
| {active_streams:!0}),UI.interval.set(function(){B()},3E4))}else mist.send(function(){B()},{active_streams:!0}),UI.interval.set(function(){B()},1E4);break;case "Edit Stream":if("undefined"==typeof mist.data.capabilities){mist.send(function(){UI.navto(a,c)},{capabilities:!0});b.append("Loading..");break}l=!1;""!=c&&(l=!0);l?(h=c,o=mist.data.streams[h],b.find("h2").append(' "'+h+'"')):(b.html($("<h2>").text("New Stream")),o={});h=[];for(q in mist.data.capabilities.inputs)h.push(mist.data.capabilities.inputs[q].source_match); | ||||
| var F=$("<div>");b.append(UI.buildUI([{label:"Stream name",type:"str",validate:["required","streamname"],pointer:{main:o,index:"name"},help:"Set the name this stream will be recognised by for players and/or stream pushing."},{label:"Source",type:"browse",validate:["required"],filetypes:h,pointer:{main:o,index:"source"},help:'Set the stream source.<table><tr><td>VoD:</td><td>You can browse to the file or folder as a source or simply enter the path to the file.</td></tr><tr><td>Live:</td><td>You\'ll need to enter "push://IP" with the IP of the machine pushing towards MistServer.<br>You can use "push://" to accept any source.</td></tr><tr><td>(Pro only)</td><td>Use "push://(IP)@password" to set a password protection for pushes.</td></tr></table>If you\'re unsure how to set the source properly, please view our Live pushing guide at the tools section.', | ||||
| "function":function(){var a=$(this).val();if(a!=""){var b=null,c;for(c in mist.data.capabilities.inputs)if(typeof mist.data.capabilities.inputs[c].source_match!="undefined"&&mist.inputMatch(mist.data.capabilities.inputs[c].source_match,a)){b=c;break}if(b===null)F.html($("<h3>").text("Unrecognized input").addClass("red")).append($("<span>").text("Please edit the stream source.").addClass("red"));else{a=mist.data.capabilities.inputs[b];F.html($("<h3>").text(a.name+" Input options"));a=mist.convertBuildOptions(a, | ||||
| o);F.append(UI.buildUI(a))}}}},$("<br>"),{type:"custom",custom:F},$("<br>"),$("<h3>").text("Encryption"),{type:"help",help:"To enable encryption, the licence acquisition url must be entered, as well as either the content key or the key ID and seed.<br>Unsure how you should fill in your encryption or missing your preferred encryption? Please contact us."},{label:"License acquisition url",type:"str",LTSonly:!0,pointer:{main:o,index:"la_url"}},$("<br>"),{label:"Content key",type:"str",LTSonly:!0,pointer:{main:o, | ||||
| index:"contentkey"}},{type:"text",text:" - or - "},{label:"Key ID",type:"str",LTSonly:!0,pointer:{main:o,index:"keyid"}},{label:"Key seed",type:"str",LTSonly:!0,pointer:{main:o,index:"keyseed"}},{type:"buttons",buttons:[{type:"cancel",label:"Cancel","function":function(){UI.navto("Streams")}},{type:"save",label:"Save","function":function(){if(!mist.data.streams)mist.data.streams={};mist.data.streams[o.name]=o;c!=o.name&&delete mist.data.streams[c];var a={};if(mist.data.LTS){a.addstream={};a.addstream[o.name]= | ||||
| o;if(c!=o.name)a.deletestream=[c]}else a.streams=mist.data.streams;mist.send(function(){delete mist.data.streams[o.name].online;delete mist.data.streams[o.name].error;UI.navto("Streams")},a)}}]}]));break;case "Preview":if(""==c){b.append("Loading..");var I=function(c){var d={};c.sort();b.html($("<h2>").text(a)).append(UI.buildUI([{label:"Select a stream",type:"select",select:c,pointer:{main:d,index:"stream"}},{type:"buttons",buttons:[{type:"save",label:"Go","function":function(){UI.navto(a,d.stream)}}]}])); | ||||
| UI.elements.secondary_menu.html("").append($("<a>").addClass("button").addClass("active").text("Choose stream").click(function(){UI.navto("Preview")}));var e=$("<div>").addClass("preview_icons");b.append($("<span>").addClass("description").text("Or, click a stream from the list below.")).append(e);for(var g in c){var f=c[g],h="";if(f.indexOf("+")>-1){h=f.split("+");h=mist.data.streams[h[0]].source+h[1]}else h=mist.data.streams[f].source;e.append($("<button>").append($("<span>").text(f)).append($("<span>").addClass("description").text(h)).attr("title", | ||||
| f).attr("data-stream",f).click(function(){UI.navto("Preview",$(this).attr("data-stream"))}))}};if(mist.data.LTS){if("undefined"==typeof mist.data.capabilities){mist.send(function(){UI.navto(a,c)},{capabilities:!0});b.append("Loading..");break}C=z=0;u={};for(h in mist.data.streams)r=mist.data.capabilities.inputs.Folder||mist.data.capabilities.inputs["Folder.exe"],mist.inputMatch(r.source_match,mist.data.streams[h].source)&&(mist.send(function(a,b){var c=b.stream,d;for(d in mist.data.browse.files)for(var e in mist.data.capabilities.inputs)e.indexOf("Buffer")>= | ||||
| 0||e.indexOf("Folder")>=0||mist.inputMatch(mist.data.capabilities.inputs[e].source_match,"/"+mist.data.browse.files[d])&&(u[c+"+"+mist.data.browse.files[d]]=true);C++;z==C&&mist.send(function(){for(var a in mist.data.active_streams){var b=mist.data.active_streams[a].split("+");b.length>1&&b[0]in mist.data.streams&&(u[mist.data.active_streams[a]]=true)}u=Object.keys(u);u=u.concat(Object.keys(mist.data.streams));u.sort();I(u)},{active_streams:true})},{browse:mist.data.streams[h].source},{stream:h}), | ||||
| z++);0==z&&mist.send(function(){var c=[],d;for(d in mist.data.active_streams){var e=mist.data.active_streams[d].split("+");e.length>1&&e[0]in mist.data.streams&&(c[mist.data.active_streams[d]]=true)}mist.data.streams&&(c=c.concat(Object.keys(mist.data.streams)));if(c.length==0)b.html($("<h2>").text(a)).append("Please set up a stream first.");else{c.sort();I(c)}},{active_streams:!0})}else I(Object.keys(mist.data.streams));break}b.find("h2").append(' of "'+c+'"');h=":8080";for(q in mist.data.config.protocols)if(r= | ||||
| mist.data.config.protocols[q],"HTTP"==r.connector||"HTTP.exe"==r.connector)h=r.port?":"+r.port:":8080";var A={},r=$("<span>").hide();A["Embed urls"]=r;b.append(r);var d="",S=mist.user.host.match(/(https?)\:\/\/([^:\/]+)\:(\d+)?/i);null!=S&&(d=S[2]);var D="http://"+d+h+"/",v={},G=function(a){var b=["div"],d='\n   <script src="'+D+"embed_"+c+'.js"><\/script>\n';a.autoplay&&b.push("data-autoplay");a.forceprotocol&&a.forceprotocol!=""&&b.push('data-forcetype="'+a.forceprotocol+'"');a.urlappend&&a.urlappend!= | ||||
| ""&&b.push('data-urlappend="'+a.urlappend.replace(/\"/g,'\\"')+'"');return"<"+b.join(" ")+">"+d+"</div>"},T=$("<span>");r.append($("<h3>").text("Embed urls")).append(UI.buildUI([{label:"Embedable script",type:"str",value:D+"embed_"+c+".js",readonly:!0},{label:"Stream info script",type:"str",value:D+"info_"+c+".js",readonly:!0},{label:"Autodetect player",type:"str",value:D+c+".html",readonly:!0,qrcode:!0},$("<h3>").text("Embed code"),{label:"Embed code",type:"textarea",value:G(v),rows:4,readonly:!0, | ||||
| classes:["embed_code"]},$("<h4>").text("Embed code options").css("margin-top",0),{label:"Autoplay",type:"checkbox",pointer:{main:v,index:"autoplay"},"function":function(){v.autoplay=$(this).getval();$(".embed_code").setval(G(v))}},{label:"Force protocol",type:"select",select:[["","Automatic"]],pointer:{main:v,index:"protocol"},classes:["embed_code_forceprotocol"],"function":function(){v.forceprotocol=$(this).getval();$(".embed_code").setval(G(v))}},{label:"Video URL addition",type:"str",pointer:{main:v, | ||||
| index:"urlappend"},help:"The embed script will append this string to the video url, useful for sending through params.",classes:["embed_code_forceprotocol"],"function":function(){v.urlappend=$(this).getval();$(".embed_code").setval(G(v))}},$("<h3>").text("Protocol stream urls"),T]));h=$("<span>").append($("<h3>").text("Meta information")).hide();A["Meta information"]=h;var J=$("<span>");h.append(J);b.append(h);h=$("<span>").hide();A.Preview=h;b.append(h);var x=$("<div>").css("float","left").css("margin-right", | ||||
| "1em").attr("data-forcesupportcheck",""),U=$("<div>").css("float","left");h.append(x).append(U);mist.stored.get().autoplay&&x.attr("data-autoplay","");var H=function(){x.text("Loading..");var a=document.createElement("script");a.src=D+"embed_"+c+".js";a.onerror=function(){x.html('Error loading "'+a.src+'".<br>').append($("<button>").text("Try again").click(function(){H()}))};a.onload=function(){if(typeof mistvideo[c].error!="undefined")x.html(mistvideo[c].error+"<br>").append($("<button>").text("Try again").click(function(){H()})); | ||||
| else{var a=mistvideo[c],b=UI.buildUI([{label:"Protocol stream url",type:"str",readonly:true,value:a.embedded?a.embedded.url:"",qrcode:true},{label:"Autoplay (from now on)",type:"checkbox",value:mist.stored.get().autoplay,"function":function(){mist.stored.set("autoplay",$(this).getval()?1:0)}}]);b.find(".help_container").remove();x.append(b);var d=$("<table>").css("font-size","0.9em").html($("<thead>").html($("<tr>").html($("<th>")).append($("<th>").text("Type")).append($("<th>").text("Priority")).append($("<th>").text("Simul. tracks")).append($("<th>").html("Your browser<br>support")))); | ||||
| U.html(d);b=$("<tbody>");d.append(b);var d=$(".embed_code_forceprotocol"),e=[];d.find(".clear").remove();for(var g in a.source){var f=a.source[g],h=f.type.split("/"),i=h[0];i.length<6&&(i=i.toUpperCase());switch(h.length){case 1:break;case 2:i=UI.format.capital(h[0])+" v"+h[1];if(h[0]=="flash")switch(h[1]){case "7":i="Progressive ("+i+")";break;case "10":i="RTMP ("+i+")";break;case "11":i="HDS ("+i+")"}break;case 3:switch(h[2]){case "vnd.apple.mpegurl":i=i+" HLS";break;case "vnd.ms-ss":i=i+" Smooth"; | ||||
| break;case "mp2t":i=i+" TS";break;default:h[2].length<6&&(h[2]=h[2].toUpperCase());i=i+(" "+h[2]);h[1]!="video"&&(i=i+(" ("+h[1]+")"))}break;default:i=f.type}i=UI.format.capital(i);d.append($("<option>").text(i).val(f.type).addClass("clear"));e.push({label:i,type:"str",value:f.url,readonly:true,qrcode:true});h=$("<tr>");b.append(h);h.html($("<td>").html($("<input>").attr("type","radio").attr("name","protocolforce").change(function(){x.attr("data-forcetype",$(this).val()).html("Loading embed..");H()}).val(f.type))).append($("<td>").text(i)).append($("<td>").text(f.priority)).append($("<td>").text(f.simul_tracks+ | ||||
| "/"+f.total_matches)).append($("<td>").text(f.browser_support?"yes":"no"));if(a.embedded&&a.embedded.type==f.type){h.css("outline","1px solid rgba(0,0,0,0.5)");h.find("input[type=radio]").prop("checked",true)}}T.html(UI.buildUI(e));var j;if(c in mistvideo)j=mistvideo[c].meta;if(j){a=[];a.push({label:"Type",type:"span",value:j.live?"Live":"Pre-recorded (VoD)"});"format"in j&&a.push({label:"Format",type:"span",value:j.format});j.live&&a.push({label:"Buffer window",type:"span",value:UI.format.addUnit(j.buffer_window, | ||||
| "ms")});b={vheader:"Audio",labels:["Codec","Duration","Peak bitrate","Channels","Samplerate"],content:[]};g={vheader:"Video",labels:["Codec","Duration","Peak bitrate","Size","Framerate"],content:[]};d=Object.keys(j.tracks);d.sort(function(a,b){a=a.split("_").pop();b=b.split("_").pop();return a-b});for(var k in d){e=d[k];f=j.tracks[e];switch(f.type){case "audio":b.content.push({header:"Track "+e.split("_").pop(),body:[f.codec,UI.format.duration((f.lastms-f.firstms)/1E3)+"<br><span class=description>"+ | ||||
| UI.format.duration(f.firstms/1E3)+" to "+UI.format.duration(f.lastms/1E3)+"</span>",UI.format.bytes(f.bps,1),f.channels,UI.format.addUnit(UI.format.number(f.rate),"Hz")]});break;case "video":g.content.push({header:"Track "+e.split("_").pop(),body:[f.codec,UI.format.duration((f.lastms-f.firstms)/1E3)+"<br><span class=description>"+UI.format.duration(f.firstms/1E3)+" to "+UI.format.duration(f.lastms/1E3)+"</span>",UI.format.bytes(f.bps,1),UI.format.addUnit(f.width,"x ")+UI.format.addUnit(f.height,"px"), | ||||
| UI.format.addUnit(UI.format.number(f.fpks/1E3),"fps")]})}}j=UI.buildVheaderTable(b).css("width","auto");k=UI.buildVheaderTable(g).css("width","auto");a.push($("<span>").text("Tracks:"));a.push($("<div>").css({display:"flex","flex-flow":"row wrap","justify-content":"center","font-size":"0.9em"}).append(j).append(k));J.html(UI.buildUI(a))}else J.html("No meta information available.")}};x.html("")[0].appendChild(a)};H();var K=UI.elements.secondary_menu;K.html("").append($("<a>").addClass("button").text("Choose stream").click(function(){UI.navto("Preview")})).append($("<span>").addClass("separator")); | ||||
| h=["Preview","Embed urls","Meta information"];k=h[0];for(q in h)r=$("<a>").addClass("button").text(h[q]).click(function(){K.find(".active").removeClass("active");$(this).addClass("active");for(q in A)A[q].hide();A[$(this).text()].show()}),K.append(r),h[q]==k&&(r.addClass("active"),A[k].show());break;case "Triggers":"triggers"in mist.data.config||(mist.data.config.triggers={});f=$("<tbody>");d=$("<table>").html($("<thead>").html($("<tr>").html($("<th>").text("Trigger on").attr("data-sort-type","string").addClass("sorting-asc")).append($("<th>").text("Applies to").attr("data-sort-type", | ||||
| "string")).append($("<th>").text("Handler").attr("data-sort-type","string")).append($("<th>")))).append(f);b.append(UI.buildUI([{type:"help",help:"Triggers are the system you can use to react to events that occur inside MistServer. These allow you to block specific users, redirect streams, keep tabs on what is being pushed where, etcetera. For full documentation, please refer to the developer documentation section on the MistServer website."}])).append($("<button>").text("New trigger").click(function(){UI.navto("Edit Trigger")})).append(d); | ||||
| d.stupidtable();h=mist.data.config.triggers;for(q in h)for(r in h[q])f.append($("<tr>").attr("data-index",q+","+r).append($("<td>").text(q)).append($("<td>").text(h[q][r][2].join(", "))).append($("<td>").text(h[q][r][0])).append($("<td>").html($("<button>").text("Edit").click(function(){UI.navto("Edit Trigger",$(this).closest("tr").attr("data-index"))})).append($("<button>").text("Delete").click(function(){var a=$(this).closest("tr").attr("data-index").split(",");if(confirm("Are you sure you want to delete this "+ | ||||
| a[0]+" trigger?")){mist.data.config.triggers[a[0]].splice(a[1],1);mist.data.config.triggers[a[0]].length==0&&delete mist.data.config.triggers[a[0]];mist.send(function(){UI.navto("Triggers")},{config:mist.data.config})}}))));break;case "Edit Trigger":"triggers"in mist.data.config||(mist.data.config.triggers={});c?(c=c.split(","),h=mist.data.config.triggers[c[0]][c[1]],o={triggeron:c[0],appliesto:h[2],url:h[0],async:h[1],"default":h[3]}):(b.html($("<h2>").text("New Trigger")),o={});b.append(UI.buildUI([{label:"Trigger on", | ||||
| pointer:{main:o,index:"triggeron"},help:"For what event this trigger should activate.",type:"select",select:[["SYSTEM_START","SYSTEM_START: after MistServer boot"],["SYSTEM_STOP","SYSTEM_STOP: right before MistServer shutdown"],["SYSTEM_CONFIG","SYSTEM_CONFIG: after MistServer configurations have changed"],["OUTPUT_START","OUTPUT_START: right after the start command has been send to a protocol"],["OUTPUT_STOP","OUTPUT_STOP: right after the close command has been send to a protocol "],["STREAM_ADD", | ||||
| "STREAM_ADD: right before new stream configured"],["STREAM_CONFIG","STREAM_CONFIG: right before a stream configuration has changed"],["STREAM_REMOVE","STREAM_REMOVE: right before a stream has been deleted"],["STREAM_SOURCE","STREAM_SOURCE: right before stream source is loaded"],["STREAM_LOAD","STREAM_LOAD: right before stream input is loaded in memory"],["STREAM_READY","STREAM_READY: when the stream input is loaded and ready for playback"],["STREAM_UNLOAD","STREAM_UNLOAD: right before the stream input is removed from memory"], | ||||
| ["STREAM_PUSH","STREAM_PUSH: right before an incoming push is accepted"],["STREAM_TRACK_ADD","STREAM_TRACK_ADD: right before a track will be added to a stream; e.g.: additional push received"],["STREAM_TRACK_REMOVE","STREAM_TRACK_REMOVE: right before a track will be removed track from a stream; e.g.: push timeout"],["STREAM_BUFFER","STREAM_BUFFER: when a buffer changes between mostly full or mostly empty"],["RTMP_PUSH_REWRITE","RTMP_PUSH_REWRITE: allows rewriting of RTMP push URLs from external to internal representation before further parsing"], | ||||
| ["RECORDING_VALIDATE","RECORDING_VALIDATE: before recording, check if we this stream is allowed to record."],["RECORDING_FILEPATH","RECORDING_FILEPATH: before recording, hook can return filename to be used. "],["RECORDING_START","RECORDING_START: started a recording"],["RECORDING_STOP","RECORDING_STOP: stopped a recording"],["CONN_OPEN","CONN_OPEN: right after a new incoming connection has been received"],["CONN_CLOSE","CONN_CLOSE: right after a connection has been closed"],["CONN_PLAY","CONN_PLAY: right before a stream playback of a connection"], | ||||
| ["USER_NEW","USER_NEW: A new user connects that hasn't been allowed or denied access before"]],LTSonly:!0,"function":function(){switch($(this).getval()){case "SYSTEM_START":case "SYSTEM_STOP":case "SYSTEM_CONFIG":case "OUTPUT_START":case "OUTPUT_STOP":case "RTMP_PUSH_REWRITE":$("[name=appliesto]").setval([]).closest(".UIelement").hide();break;default:$("[name=appliesto]").closest(".UIelement").show()}}},{label:"Applies to",pointer:{main:n,index:"appliesto"},help:"For triggers that can apply to specific streams, this value decides what streams they are triggered for. (none checked = always triggered)", | ||||
| type:"checklist",checklist:Object.keys(mist.data.streams),LTSonly:!0},$("<br>"),{label:"Handler (URL or executable)",help:"This can be either an HTTP URL or a full path to an executable.",pointer:{main:n,index:"url"},validate:["required"],type:"str",LTSonly:!0},{label:"Blocking",type:"checkbox",help:"If checked, pauses processing and uses the response of the handler. If the response does not start with 1, true, yes or cont, further processing is aborted. If unchecked, processing is never paused and the response is not checked.", | ||||
| pointer:{main:n,index:"async"},LTSonly:!0},{label:"Default response",type:"str",help:"For blocking requests, the default response in case the handler cannot be executed for any reason.",pointer:{main:n,index:"default"},LTSonly:!0},{type:"buttons",buttons:[{type:"cancel",label:"Cancel","function":function(){UI.navto("Triggers")}},{type:"save",label:"Save","function":function(){c&&mist.data.config.triggers[c[0]].splice(c[1],1);var a=[n.url,n.async?true:false,typeof n.appliesto!="undefined"?n.appliesto: | ||||
| []];typeof n["default"]!="undefined"&&a.push(n["default"]);n.triggeron in mist.data.config.triggers||(mist.data.config.triggers[n.triggeron]=[]);mist.data.config.triggers[n.triggeron].push(a);mist.send(function(){UI.navto("Triggers")},{config:mist.data.config})}}]}]));$("[name=triggeron]").trigger("change");break;case "Logs":b.append(UI.buildUI([{type:"help",help:"Here you have an overview of all edited settings within MistServer and possible warnings or errors MistServer has encountered. MistServer stores up to 100 logs at a time."}, | ||||
| {label:"Refresh every",type:"select",select:[[10,"10 seconds"],[30,"30 seconds"],[60,"minute"],[300,"5 minutes"]],value:30,"function":function(){clearInterval(UI.interval);UI.interval.set(function(){mist.send(function(){V()})},$(this).val()*1E3)}}]));b.append($("<button>").text("Purge logs").click(function(){mist.send(function(){mist.data.log=[];UI.navto("Logs")},{clearstatlogs:true})}));g=$("<tbody>").css("font-size","0.9em");b.append($("<table>").append(g));var X=function(a){var b=$("<span>").text(a); | ||||
| switch(a){case "WARN":b.addClass("orange");break;case "ERROR":case "FAIL":b.addClass("red")}return b},V=function(){var a=mist.data.log;if(a){a.length>=2&&a[0][0]<a[a.length-1][0]&&a.reverse();g.html("");for(var b in a)g.append($("<tr>").html($("<td>").text(UI.format.dateTime(a[b][0],"long")).css("white-space","nowrap")).append($("<td>").html(X(a[b][1])).css("text-align","center")).append($("<td>").text(a[b][2]).css("text-align","left")))}};V();break;case "Statistics":u=$("<span>").text("Loading.."); | ||||
| b.append(u);var n={},s=mist.stored.get().graphs?$.extend(!0,{},mist.stored.get().graphs):{},A={};for(r in mist.data.streams)A[r]=!0;for(r in mist.data.active_streams)A[mist.data.active_streams[r]]=!0;var A=Object.keys(A).sort(),M=[];for(r in mist.data.config.protocols)M.push(mist.data.config.protocols[r].connector);M.sort();mist.send(function(){UI.plot.datatype.templates.cpuload.cores=0;for(var a in mist.data.capabilities.cpu)UI.plot.datatype.templates.cpuload.cores=UI.plot.datatype.templates.cpuload.cores+ | ||||
| mist.data.capabilities.cpu[a].cores;u.html(UI.buildUI([{type:"help",help:"Here you will find the MistServer stream statistics, you can select various categories yourself. All statistics are live: up to five minutes are saved."},$("<h3>").text("Select the data to display"),{label:"Add to",type:"select",select:[["new","New graph"]],pointer:{main:n,index:"graph"},classes:["graph_ids"],"function":function(){if($(this).val()){var a=u.find(".graph_xaxis"),b=u.find(".graph_id");if($(this).val()=="new"){a.children("option").prop("disabled", | ||||
| false);b.setval("Graph "+(Object.keys(s).length+1)).closest("label").show()}else{var c=s[$(this).val()].xaxis;a.children("option").prop("disabled",true).filter('[value="'+c+'"]').prop("disabled",false);b.closest("label").hide()}a.children('option[value="'+a.val()+'"]:disabled').length&&a.val(a.children("option:enabled").first().val());a.trigger("change")}}},{label:"Graph id",type:"str",pointer:{main:n,index:"id"},classes:["graph_id"],validate:[function(a){return a in s?{msg:"This graph id has already been used. Please enter something else.", | ||||
| classes:["red"]}:false}]},{label:"Axis type",type:"select",select:[["time","Time line"],["coords","Geographical"]],pointer:{main:n,index:"xaxis"},classes:["graph_xaxis"],"function":function(){$s=u.find(".graph_datatype");switch($(this).getval()){case "coords":$s.children("option").prop("disabled",true).filter('[value="coords"]').prop("disabled",false);break;case "time":$s.children("option").prop("disabled",false).filter('[value="coords"]').prop("disabled",true)}if(!$s.val()||$s.children('option[value="'+ | ||||
| $s.val()+'"]:disabled').length){$s.val($s.children("option:enabled").first().val());$s.trigger("change")}}},{label:"Data type",type:"select",select:[["clients","Connections"],["upbps","Bandwidth (up)"],["downbps","Bandwidth (down)"],["cpuload","CPU load"],["memload","Memory load"],["coords","Client location"]],pointer:{main:n,index:"datatype"},classes:["graph_datatype"],"function":function(){$s=u.find(".graph_origin");switch($(this).getval()){case "cpuload":case "memload":$s.find("input[type=radio]").not('[value="total"]').prop("disabled", | ||||
| true);$s.find('input[type=radio][value="total"]').prop("checked",true);break;default:$s.find("input[type=radio]").prop("disabled",false)}}},{label:"Data origin",type:"radioselect",radioselect:[["total","All"],["stream","The stream:",A],["protocol","The protocol:",M]],pointer:{main:n,index:"origin"},value:["total"],classes:["graph_origin"]},{type:"buttons",buttons:[{label:"Add data set",type:"save","function":function(){var a;if(n.graph=="new"){a=UI.plot.addGraph(n,c);s[a.id]=a;u.find("select.graph_ids").append($("<option>").text(a.id)).val(a.id).trigger("change")}else a= | ||||
| s[n.graph];var b=UI.plot.datatype.getOptions({datatype:n.datatype,origin:n.origin});a.datasets.push(b);UI.plot.save(a);UI.plot.go(s)}}]}]));var c=$("<div>").addClass("graph_container");b.append(c);var d=u.find("select.graph_ids");for(a in s){var e=UI.plot.addGraph(s[a],c);d.append($("<option>").text(e.id)).val(e.id);var g=[],f;for(f in s[a].datasets){var h=UI.plot.datatype.getOptions({datatype:s[a].datasets[f].datatype,origin:s[a].datasets[f].origin});g.push(h)}e.datasets=g;s[e.id]=e}d.trigger("change"); | ||||
| UI.plot.go(s);UI.interval.set(function(){UI.plot.go(s)},1E4)},{active_streams:!0,capabilities:!0});break;case "Server Stats":if("undefined"==typeof mist.data.capabilities){mist.send(function(){UI.navto(a)},{capabilities:!0});b.append("Loading..");break}var N=$("<table>"),O=$("<table>"),i={vheader:"CPUs",labels:["Model","Processor speed","Amount of cores","Amount of threads"],content:[]};for(r in mist.data.capabilities.cpu)q=mist.data.capabilities.cpu[r],i.content.push({header:"CPU #"+(Number(r)+1), | ||||
| body:[q.model,UI.format.addUnit(UI.format.number(q.mhz),"MHz"),q.cores,q.threads]});var i=UI.buildVheaderTable(i),W=function(){var a=mist.data.capabilities.mem,b=mist.data.capabilities.load,a={vheader:"Memory",labels:["Used","Cached","Available","Total"],content:[{header:"Physical memory",body:[UI.format.bytes(a.used*1048576)+" ("+UI.format.addUnit(b.memory,"%")+")",UI.format.bytes(a.cached*1048576),UI.format.bytes(a.free*1048576),UI.format.bytes(a.total*1048576)]},{header:"Swap memory",body:[UI.format.bytes((a.swaptotal- | ||||
| a.swapfree)*1048576),UI.format.addUnit("","N/A"),UI.format.bytes(a.swapfree*1048576),UI.format.bytes(a.swaptotal*1048576)]}]},a=UI.buildVheaderTable(a);N.replaceWith(a);N=a;b={vheader:"Load average",labels:["1 minute","5 minutes","15 minutes",""],content:[{header:" ",body:[UI.format.number(b.one/100),UI.format.number(b.five/100),UI.format.number(b.fifteen/100),""]}]};b=UI.buildVheaderTable(b);O.replaceWith(b);O=b};W();b.append(UI.buildUI([{type:"help",help:"You can find general server statistics here. Note that memory and CPU usage is for your entire machine, not just MistServer."}])).append($("<table>").css("width", | ||||
| "auto").addClass("nolay").append($("<tr>").append($("<td>").append(N)).append($("<td>").append(O))).append($("<tr>").append($("<td>").append(i).attr("colspan",2))));UI.interval.set(function(){mist.send(function(){W()},{capabilities:true})},3E4);break;case "Email for Help":i=$.extend({},mist.data);delete i.statistics;delete i.totals;delete i.clients;delete i.capabilities;i=JSON.stringify(i);i="Version: "+mist.data.config.version+"\n\nConfig:\n"+i;n={};b.append(UI.buildUI([{type:"help",help:"You can use this form to email MistServer support if you're having difficulties.<br>A copy of your server config file will automatically be included."}, | ||||
| {type:"str",label:"Your name",validate:["required"],pointer:{main:n,index:"name"},value:mist.user.name},{type:"email",label:"Your email address",validate:["required"],pointer:{main:n,index:"email"}},{type:"hidden",value:"Integrated Help",pointer:{main:n,index:"subject"}},{type:"hidden",value:"-",pointer:{main:n,index:"company"}},{type:"textarea",rows:20,label:"Your message",validate:["required"],pointer:{main:n,index:"message"}},{type:"textarea",rows:20,label:"Your config file",readonly:!0,value:i, | ||||
| pointer:{main:n,index:"configfile"}},{type:"buttons",buttons:[{type:"save",label:"Send","function":function(a){$(a).text("Sending..");$.ajax({type:"POST",url:"http://mistserver.org/contact_us?skin=plain",data:n,success:function(a){a=$("<span>").html(a);a.find("script").remove();b.html(a[0].innerHTML)}})}}]}]));break;case "Disconnect":mist.user.password="";delete mist.user.authstring;delete mist.user.loggedin;UI.navto("Login");break;default:b.append($("<p>").text("This tab does not exist."))}}}},mist= | ||||
| {data:{},user:{name:"",password:"",host:"http://"+(location.hostname?location.hostname:"localhost")+":4242/api"},send:function(a,c,b){var c=c||{},b=b||{},b=$.extend(!0,{timeout:30,sendData:c},b),d={authorize:{password:mist.user.authstring?MD5(MD5(mist.user.password)+mist.user.authstring):"",username:mist.user.name}};$.extend(!0,d,c);log("Send",$.extend(!0,{},c));d={url:mist.user.host,type:"POST",data:{command:JSON.stringify(d)},dataType:"jsonp",crossDomain:!0,timeout:1E3*b.timeout,async:!0,error:function(d, | ||||
| f){delete mist.user.loggedin;if(!b.hide){switch(f){case "timeout":f=$("<i>").text("The connection timed out. ");break;case "abort":f=$("<i>").text("The connection was aborted. ");break;default:f=$("<i>").text(f+". ").css("text-transform","capitalize")}$("#message").addClass("red").text("An error occurred while attempting to communicate with MistServer:").append($("<br>")).append(f).append($("<a>").text("Send server request again").click(function(){mist.send(a,c,b)}))}UI.navto("Login")},success:function(d){log("Receive", | ||||
| $.extend(!0,{},d),"as reply to",b.sendData);delete mist.user.loggedin;switch(d.authorize.status){case "OK":"streams"in d&&(d.streams?"incomplete list"in d.streams?(delete d.streams["incomplete list"],$.extend(mist.data.streams,d.streams)):mist.data.streams=d.streams:mist.data.streams={});var f=$.extend({},d),e="config capabilities ui_settings LTS active_streams browse log totals".split(" "),p;for(p in f)-1==e.indexOf(p)&&delete f[p];$.extend(!0,mist.data,f);mist.user.loggedin=!0;UI.elements.connection.status.text("Connected").removeClass("red").addClass("green"); | ||||
| UI.elements.connection.user_and_host.text(mist.user.name+" @ "+mist.user.host);UI.elements.connection.msg.removeClass("red").text("Last communication with the server at "+UI.format.time((new Date).getTime()/1E3));d.LTS&&UI.elements.menu.find(".LTSonly").removeClass("LTSonly");d.log&&(f=d.log[d.log.length-1],UI.elements.connection.msg.append($("<br>")).append("Last log entry: "+UI.format.time(f[0])+" ["+f[1]+"] "+f[2]));if("totals"in d)if(f=function(a,b,c){var d;d=function(){for(var a in c.fields)e[c.fields[a]].push([k, | ||||
| 0])};var e={},f;for(f in c.fields)e[c.fields[f]]=[];var h=0,k;if(c.data){if(c.end-c.start<600){k=(c.end-600)*1E3;d();k=c.start*1E3;d()}else k=c.start*1E3;for(f in c.data){if(f==0){k=c.start*1E3;var m=0}else{k=k+c.interval[m][1]*1E3;c.interval[m][0]--;if(c.interval[m][0]<=0){m++;m<c.interval.length-1&&(h=2)}}if(h==1){d();h--}for(var p in c.data[f])e[c.fields[p]].push([k,c.data[f][p]]);if(h){d();h--}}if(mist.data.config.time-c.end>5){d();k=mist.data.config.time*1E3;d()}}else{k=(mist.data.config.time- | ||||
| 600)*1E3;d();k=mist.data.config.time*1E3;d()}d=e;stream=a?a.join(" "):"all_streams";protocol=b?b.join("_"):"all_protocols";stream in mist.data.totals||(mist.data.totals[stream]={});protocol in mist.data.totals[stream]||(mist.data.totals[stream][protocol]={});$.extend(mist.data.totals[stream][protocol],d)},mist.data.totals={},"fields"in d.totals)f(c.totals.streams,c.totals.protocols,d.totals);else for(p in d.totals)f(c.totals[p].streams,c.totals[p].protocols,d.totals[p]);a&&a(d,b);break;case "CHALL":d.authorize.challenge== | ||||
| mist.user.authstring?(""!=mist.user.password&&UI.elements.connection.msg.text("The credentials you provided are incorrect.").addClass("red"),UI.navto("Login")):""==mist.user.password?UI.navto("Login"):(mist.user.authstring=d.authorize.challenge,mist.send(a,c,b));break;case "NOACC":UI.navto("Create a new account");break;case "ACC_MADE":delete c.authorize;mist.send(a,c,b);break;default:UI.navto("Login")}}};b.hide||UI.elements.connection.msg.removeClass("red").text("Data sent, waiting for a reply..").append($("<br>")).append($("<a>").text("Cancel request").click(function(){f.abort()})); | ||||
| var f=$.ajax(d)},inputMatch:function(a,c){if("undefined"==typeof a)return!1;var b=a.replace(/[^\w\s]/g,"\\$&"),b=b.replace(/\\\?/g,".").replace(/\\\*/g,"(?:.)*");return RegExp("^"+b+"$","i").test(c)},convertBuildOptions:function(a,c){var b=[],d=["required","optional"];"desc"in a&&b.push({type:"help",help:a.desc});for(var f in d)if(a[d[f]]){b.push($("<h4>").text(UI.format.capital(d[f])+" parameters"));for(var k in a[d[f]]){var m=a[d[f]][k],e={label:UI.format.capital(m.name),pointer:{main:c,index:k}, | ||||
| validate:[]};"required"==d[f]&&!("default"in m)&&e.validate.push("required");"default"in m&&(e.placeholder=m["default"]);"help"in m&&(e.help=m.help);"unit"in m&&(e.unit=m.unit);switch(m.type){case "int":e.type="int";break;case "uint":e.type="int";e.min=0;break;case "debug":e.type="debug";break;case "select":e.type="select";e.select=m.select;break;default:e.type="str"}b.push(e)}}return b},stored:{get:function(){return mist.data.ui_settings||{}},set:function(a,c){var b=this.get();b[a]=c;mist.send(function(){}, | ||||
| ["USER_NEW","USER_NEW: A new user connects that hasn't been allowed or denied access before"]],LTSonly:!0,"function":function(){switch($(this).getval()){case "SYSTEM_START":case "SYSTEM_STOP":case "SYSTEM_CONFIG":case "OUTPUT_START":case "OUTPUT_STOP":case "RTMP_PUSH_REWRITE":$("[name=appliesto]").setval([]).closest(".UIelement").hide();break;default:$("[name=appliesto]").closest(".UIelement").show()}}},{label:"Applies to",pointer:{main:o,index:"appliesto"},help:"For triggers that can apply to specific streams, this value decides what streams they are triggered for. (none checked = always triggered)", | ||||
| type:"checklist",checklist:Object.keys(mist.data.streams),LTSonly:!0},$("<br>"),{label:"Handler (URL or executable)",help:"This can be either an HTTP URL or a full path to an executable.",pointer:{main:o,index:"url"},validate:["required"],type:"str",LTSonly:!0},{label:"Blocking",type:"checkbox",help:"If checked, pauses processing and uses the response of the handler. If the response does not start with 1, true, yes or cont, further processing is aborted. If unchecked, processing is never paused and the response is not checked.", | ||||
| pointer:{main:o,index:"async"},LTSonly:!0},{label:"Default response",type:"str",help:"For blocking requests, the default response in case the handler cannot be executed for any reason.",pointer:{main:o,index:"default"},LTSonly:!0},{type:"buttons",buttons:[{type:"cancel",label:"Cancel","function":function(){UI.navto("Triggers")}},{type:"save",label:"Save","function":function(){c&&mist.data.config.triggers[c[0]].splice(c[1],1);var a=[o.url,o.async?true:false,typeof o.appliesto!="undefined"?o.appliesto: | ||||
| []];typeof o["default"]!="undefined"&&a.push(o["default"]);o.triggeron in mist.data.config.triggers||(mist.data.config.triggers[o.triggeron]=[]);mist.data.config.triggers[o.triggeron].push(a);mist.send(function(){UI.navto("Triggers")},{config:mist.data.config})}}]}]));$("[name=triggeron]").trigger("change");break;case "Logs":var V=$("<button>").text("Refresh now").click(function(){$(this).text("Loading..");mist.send(function(){L();V.text("Refresh now")})}).css("padding","0.2em 0.5em").css("flex-grow", | ||||
| 0);b.append(UI.buildUI([{type:"help",help:"Here you have an overview of all edited settings within MistServer and possible warnings or errors MistServer has encountered. MistServer stores up to 100 logs at a time."},{label:"Refresh every",type:"select",select:[[10,"10 seconds"],[30,"30 seconds"],[60,"minute"],[300,"5 minutes"]],value:30,"function":function(){UI.interval.clear();UI.interval.set(function(){mist.send(function(){L()})},$(this).val()*1E3)},help:"How often the table below should be updated."}, | ||||
| {label:"..or",type:"DOMfield",DOMfield:V,help:"Instantly refresh the table below."}]));b.append($("<button>").text("Purge logs").click(function(){mist.send(function(){mist.data.log=[];UI.navto("Logs")},{clearstatlogs:true})}));f=$("<tbody>").css("font-size","0.9em");b.append($("<table>").append(f));var X=function(a){var b=$("<span>").text(a);switch(a){case "WARN":b.addClass("orange");break;case "ERROR":case "FAIL":b.addClass("red")}return b},L=function(){var a=mist.data.log;if(a){a.length>=2&&a[0][0]< | ||||
| a[a.length-1][0]&&a.reverse();f.html("");for(var b in a)f.append($("<tr>").html($("<td>").text(UI.format.dateTime(a[b][0],"long")).css("white-space","nowrap")).append($("<td>").html(X(a[b][1])).css("text-align","center")).append($("<td>").text(a[b][2]).css("text-align","left")))}};L();break;case "Statistics":var y=$("<span>").text("Loading..");b.append(y);var o={},t=mist.stored.get().graphs?$.extend(!0,{},mist.stored.get().graphs):{},E={};for(q in mist.data.streams)E[q]=!0;for(q in mist.data.active_streams)E[mist.data.active_streams[q]]= | ||||
| !0;var E=Object.keys(E).sort(),M=[];for(q in mist.data.config.protocols)M.push(mist.data.config.protocols[q].connector);M.sort();mist.send(function(){UI.plot.datatype.templates.cpuload.cores=0;for(var a in mist.data.capabilities.cpu)UI.plot.datatype.templates.cpuload.cores=UI.plot.datatype.templates.cpuload.cores+mist.data.capabilities.cpu[a].cores;y.html(UI.buildUI([{type:"help",help:"Here you will find the MistServer stream statistics, you can select various categories yourself. All statistics are live: up to five minutes are saved."}, | ||||
| $("<h3>").text("Select the data to display"),{label:"Add to",type:"select",select:[["new","New graph"]],pointer:{main:o,index:"graph"},classes:["graph_ids"],"function":function(){if($(this).val()){var a=y.find(".graph_xaxis"),b=y.find(".graph_id");if($(this).val()=="new"){a.children("option").prop("disabled",false);b.setval("Graph "+(Object.keys(t).length+1)).closest("label").show()}else{var c=t[$(this).val()].xaxis;a.children("option").prop("disabled",true).filter('[value="'+c+'"]').prop("disabled", | ||||
| false);b.closest("label").hide()}a.children('option[value="'+a.val()+'"]:disabled').length&&a.val(a.children("option:enabled").first().val());a.trigger("change")}}},{label:"Graph id",type:"str",pointer:{main:o,index:"id"},classes:["graph_id"],validate:[function(a){return a in t?{msg:"This graph id has already been used. Please enter something else.",classes:["red"]}:false}]},{label:"Axis type",type:"select",select:[["time","Time line"],["coords","Geographical"]],pointer:{main:o,index:"xaxis"},classes:["graph_xaxis"], | ||||
| "function":function(){$s=y.find(".graph_datatype");switch($(this).getval()){case "coords":$s.children("option").prop("disabled",true).filter('[value="coords"]').prop("disabled",false);break;case "time":$s.children("option").prop("disabled",false).filter('[value="coords"]').prop("disabled",true)}if(!$s.val()||$s.children('option[value="'+$s.val()+'"]:disabled').length){$s.val($s.children("option:enabled").first().val());$s.trigger("change")}}},{label:"Data type",type:"select",select:[["clients","Connections"], | ||||
| ["upbps","Bandwidth (up)"],["downbps","Bandwidth (down)"],["cpuload","CPU use"],["memload","Memory load"],["coords","Client location"]],pointer:{main:o,index:"datatype"},classes:["graph_datatype"],"function":function(){$s=y.find(".graph_origin");switch($(this).getval()){case "cpuload":case "memload":$s.find("input[type=radio]").not('[value="total"]').prop("disabled",true);$s.find('input[type=radio][value="total"]').prop("checked",true);break;default:$s.find("input[type=radio]").prop("disabled",false)}}}, | ||||
| {label:"Data origin",type:"radioselect",radioselect:[["total","All"],["stream","The stream:",E],["protocol","The protocol:",M]],pointer:{main:o,index:"origin"},value:["total"],classes:["graph_origin"]},{type:"buttons",buttons:[{label:"Add data set",type:"save","function":function(){var a;if(o.graph=="new"){a=UI.plot.addGraph(o,c);t[a.id]=a;y.find("select.graph_ids").append($("<option>").text(a.id)).val(a.id).trigger("change")}else a=t[o.graph];var b=UI.plot.datatype.getOptions({datatype:o.datatype, | ||||
| origin:o.origin});a.datasets.push(b);UI.plot.save(a);UI.plot.go(t)}}]}]));var c=$("<div>").addClass("graph_container");b.append(c);var d=y.find("select.graph_ids");for(a in t){var e=UI.plot.addGraph(t[a],c);d.append($("<option>").text(e.id)).val(e.id);var f=[],g;for(g in t[a].datasets){var h=UI.plot.datatype.getOptions({datatype:t[a].datasets[g].datatype,origin:t[a].datasets[g].origin});f.push(h)}e.datasets=f;t[e.id]=e}d.trigger("change");UI.plot.go(t);UI.interval.set(function(){UI.plot.go(t)},1E4)}, | ||||
| {active_streams:!0,capabilities:!0});break;case "Server Stats":if("undefined"==typeof mist.data.capabilities){mist.send(function(){UI.navto(a)},{capabilities:!0});b.append("Loading..");break}var N=$("<table>"),O=$("<table>"),h={vheader:"CPUs",labels:["Model","Processor speed","Amount of cores","Amount of threads"],content:[]};for(q in mist.data.capabilities.cpu)r=mist.data.capabilities.cpu[q],h.content.push({header:"CPU #"+(Number(q)+1),body:[r.model,UI.format.addUnit(UI.format.number(r.mhz),"MHz"), | ||||
| r.cores,r.threads]});var h=UI.buildVheaderTable(h),W=function(){var a=mist.data.capabilities.mem,b=mist.data.capabilities.load,a={vheader:"Memory",labels:["Used","Cached","Available","Total"],content:[{header:"Physical memory",body:[UI.format.bytes(a.used*1048576)+" ("+UI.format.addUnit(b.memory,"%")+")",UI.format.bytes(a.cached*1048576),UI.format.bytes(a.free*1048576),UI.format.bytes(a.total*1048576)]},{header:"Swap memory",body:[UI.format.bytes((a.swaptotal-a.swapfree)*1048576),UI.format.addUnit("", | ||||
| "N/A"),UI.format.bytes(a.swapfree*1048576),UI.format.bytes(a.swaptotal*1048576)]}]},a=UI.buildVheaderTable(a);N.replaceWith(a);N=a;b={vheader:"Load average",labels:["CPU use","1 minute","5 minutes","15 minutes"],content:[{header:" ",body:[UI.format.addUnit(UI.format.number(mist.data.capabilities.cpu_use/10),"%"),UI.format.number(b.one/100),UI.format.number(b.five/100),UI.format.number(b.fifteen/100)]}]};b=UI.buildVheaderTable(b);O.replaceWith(b);O=b};W();b.append(UI.buildUI([{type:"help",help:"You can find general server statistics here. Note that memory and CPU usage is for your entire machine, not just MistServer."}])).append($("<table>").css("width", | ||||
| "auto").addClass("nolay").append($("<tr>").append($("<td>").append(N)).append($("<td>").append(O))).append($("<tr>").append($("<td>").append(h).attr("colspan",2))));UI.interval.set(function(){mist.send(function(){W()},{capabilities:true})},3E4);break;case "Email for Help":h=$.extend({},mist.data);delete h.statistics;delete h.totals;delete h.clients;delete h.capabilities;h=JSON.stringify(h);h="Version: "+mist.data.config.version+"\n\nConfig:\n"+h;o={};b.append(UI.buildUI([{type:"help",help:"You can use this form to email MistServer support if you're having difficulties.<br>A copy of your server config file will automatically be included."}, | ||||
| {type:"str",label:"Your name",validate:["required"],pointer:{main:o,index:"name"},value:mist.user.name},{type:"email",label:"Your email address",validate:["required"],pointer:{main:o,index:"email"}},{type:"hidden",value:"Integrated Help",pointer:{main:o,index:"subject"}},{type:"hidden",value:"-",pointer:{main:o,index:"company"}},{type:"textarea",rows:20,label:"Your message",validate:["required"],pointer:{main:o,index:"message"}},{type:"textarea",rows:20,label:"Your config file",readonly:!0,value:h, | ||||
| pointer:{main:o,index:"configfile"}},{type:"buttons",buttons:[{type:"save",label:"Send","function":function(a){$(a).text("Sending..");$.ajax({type:"POST",url:"http://mistserver.org/contact?skin=plain",data:o,success:function(a){a=$("<span>").html(a);a.find("script").remove();b.html(a[0].innerHTML)}})}}]}]));break;case "Disconnect":mist.user.password="";delete mist.user.authstring;delete mist.user.loggedin;UI.navto("Login");break;default:b.append($("<p>").text("This tab does not exist."))}}}},mist= | ||||
| {data:{},user:{name:"",password:"",host:location.origin+location.pathname.replace(/\/+$/,"")+"/api"},send:function(a,c,b){var c=c||{},b=b||{},b=$.extend(!0,{timeout:30,sendData:c},b),d={authorize:{password:mist.user.authstring?MD5(MD5(mist.user.password)+mist.user.authstring):"",username:mist.user.name}};$.extend(!0,d,c);log("Send",$.extend(!0,{},c));d={url:mist.user.host,type:"POST",data:{command:JSON.stringify(d)},dataType:"jsonp",crossDomain:!0,timeout:1E3*b.timeout,async:!0,error:function(d,e){delete mist.user.loggedin; | ||||
| if(!b.hide){switch(e){case "timeout":e=$("<i>").text("The connection timed out. ");break;case "abort":e=$("<i>").text("The connection was aborted. ");break;default:e=$("<i>").text(e+". ").css("text-transform","capitalize")}$("#message").addClass("red").text("An error occurred while attempting to communicate with MistServer:").append($("<br>")).append(e).append($("<a>").text("Send server request again").click(function(){mist.send(a,c,b)}))}UI.navto("Login")},success:function(d){log("Receive",$.extend(!0, | ||||
| {},d),"as reply to",b.sendData);delete mist.user.loggedin;switch(d.authorize.status){case "OK":"streams"in d&&(d.streams?"incomplete list"in d.streams?(delete d.streams["incomplete list"],$.extend(mist.data.streams,d.streams)):mist.data.streams=d.streams:mist.data.streams={});var e=$.extend({},d),g="config capabilities ui_settings LTS active_streams browse log totals".split(" "),p;for(p in e)-1==g.indexOf(p)&&delete e[p];$.extend(!0,mist.data,e);mist.user.loggedin=!0;UI.elements.connection.status.text("Connected").removeClass("red").addClass("green"); | ||||
| UI.elements.connection.user_and_host.text(mist.user.name+" @ "+mist.user.host);UI.elements.connection.msg.removeClass("red").text("Last communication with the server at "+UI.format.time((new Date).getTime()/1E3));d.LTS&&UI.elements.menu.find(".LTSonly").removeClass("LTSonly");d.log&&(e=d.log[d.log.length-1],UI.elements.connection.msg.append($("<br>")).append("Last log entry: "+UI.format.time(e[0])+" ["+e[1]+"] "+e[2]));if("totals"in d)if(e=function(a,b,c){var d;d=function(){for(var a in c.fields)e[c.fields[a]].push([j, | ||||
| 0])};var e={},g;for(g in c.fields)e[c.fields[g]]=[];var k=0,j;if(c.data){if(c.start>mist.data.config.time-600){j=(mist.data.config.time-600)*1E3;d();j=c.start*1E3;d()}else j=c.start*1E3;for(g in c.data){if(g==0){j=c.start*1E3;var m=0}else{j=j+c.interval[m][1]*1E3;c.interval[m][0]--;if(c.interval[m][0]<=0){m++;m<c.interval.length-1&&(k=k+2)}}if(k%2==1){d();k--}for(var p in c.data[g])e[c.fields[p]].push([j,c.data[g][p]]);if(k){d();k--}}if(mist.data.config.time-c.end>20){d();j=(mist.data.config.time- | ||||
| 15)*1E3;d()}}else{j=(mist.data.config.time-600)*1E3;d();j=(mist.data.config.time-15)*1E3;d()}d=e;stream=a?a.join(" "):"all_streams";protocol=b?b.join("_"):"all_protocols";stream in mist.data.totals||(mist.data.totals[stream]={});protocol in mist.data.totals[stream]||(mist.data.totals[stream][protocol]={});$.extend(mist.data.totals[stream][protocol],d)},mist.data.totals={},"fields"in d.totals)e(c.totals.streams,c.totals.protocols,d.totals);else for(p in d.totals)e(c.totals[p].streams,c.totals[p].protocols, | ||||
| d.totals[p]);a&&a(d,b);break;case "CHALL":d.authorize.challenge==mist.user.authstring?(""!=mist.user.password&&UI.elements.connection.msg.text("The credentials you provided are incorrect.").addClass("red"),UI.navto("Login")):""==mist.user.password?UI.navto("Login"):(mist.user.authstring=d.authorize.challenge,mist.send(a,c,b));break;case "NOACC":UI.navto("Create a new account");break;case "ACC_MADE":delete c.authorize;mist.send(a,c,b);break;default:UI.navto("Login")}}};b.hide||UI.elements.connection.msg.removeClass("red").text("Data sent, waiting for a reply..").append($("<br>")).append($("<a>").text("Cancel request").click(function(){e.abort()})); | ||||
| var e=$.ajax(d)},inputMatch:function(a,c){if("undefined"==typeof a)return!1;var b=a.replace(/[^\w\s]/g,"\\$&"),b=b.replace(/\\\?/g,".").replace(/\\\*/g,"(?:.)*");return RegExp("^"+b+"$","i").test(c)},convertBuildOptions:function(a,c){var b=[],d=["required","optional"];"desc"in a&&b.push({type:"help",help:a.desc});for(var e in d)if(a[d[e]]){b.push($("<h4>").text(UI.format.capital(d[e])+" parameters"));for(var j in a[d[e]]){var m=a[d[e]][j],g={label:UI.format.capital(m.name),pointer:{main:c,index:j}, | ||||
| validate:[]};"required"==d[e]&&!("default"in m)&&g.validate.push("required");"default"in m&&(g.placeholder=m["default"]);"help"in m&&(g.help=m.help);"unit"in m&&(g.unit=m.unit);switch(m.type){case "int":g.type="int";break;case "uint":g.type="int";g.min=0;break;case "debug":g.type="debug";break;case "select":g.type="select";g.select=m.select;break;default:g.type="str"}b.push(g)}}return b},stored:{get:function(){return mist.data.ui_settings||{}},set:function(a,c){var b=this.get();b[a]=c;mist.send(function(){}, | ||||
| {ui_settings:b})},del:function(a){delete mist.data.ui_settings[a];mist.send(function(){},{ui_settings:mist.data.ui_settings})}}};function log(){try{UI.debug&&[].push.call(arguments,Error().stack),[].unshift.call(arguments,"["+UI.format.time((new Date).getTime()/1E3)+"]"),console.log.apply(console,arguments)}catch(a){}} | ||||
| $.fn.getval=function(){var a=$(this).data("opts"),c=$(this).val();if(a&&"type"in a)switch(a.type){case "span":c=$(this).html();break;case "checkbox":c=$(this).prop("checked");break;case "radioselect":a=$(this).find("label > input[type=radio]:checked").parent();a.length?(c=[],c.push(a.children("input[type=radio]").val()),a=a.children("select"),a.length&&c.push(a.val())):c="";break;case "checklist":c=[],$(this).find(".checklist input[type=checkbox]:checked").each(function(){c.push($(this).attr("name"))})}return c}; | ||||
| $.fn.setval=function(a){var c=$(this).data("opts");$(this).val(a);if(c&&"type"in c)switch(c.type){case "span":$(this).html(a);break;case "checkbox":$(this).prop("checked",a);break;case "geolimited":case "hostlimited":c=$(this).closest(".field_container").data("subUI");if("undefined"==typeof a||0==a.length)a="-";c.blackwhite.val(a.charAt(0));var a=a.substr(1).split(" "),b;for(b in a)c.values.append(c.prototype.clone(!0).val(a[b]));c.blackwhite.trigger("change");break;case "radioselect":if("undefined"== | ||||
|  |  | |||
							
								
								
									
										586
									
								
								lsp/mist.js
									
										
									
									
									
								
							
							
						
						
									
										586
									
								
								lsp/mist.js
									
										
									
									
									
								
							|  | @ -170,9 +170,6 @@ var UI = { | |||
|       Protocols: {}, | ||||
|       Streams: {}, | ||||
|       Preview: {}, | ||||
|       Limits: { | ||||
|         LTSonly: true | ||||
|       }, | ||||
|       'Triggers': { | ||||
|         LTSonly: false | ||||
|       }, | ||||
|  | @ -187,19 +184,19 @@ var UI = { | |||
|     }, | ||||
|     { | ||||
|       Guides: { | ||||
|         link: 'http://mistserver.org/wiki/Category:Guides' | ||||
|         link: 'http://mistserver.org/documentation#Userdocs' | ||||
|       }, | ||||
|       Tools: { | ||||
|         submenu: { | ||||
|           'Release notes': { | ||||
|             link: 'http://mistserver.org/wiki/Mistserver_Changelog' | ||||
|             link: 'http://mistserver.org/documentation#Devdocs' | ||||
|           }, | ||||
|           'Mist Shop': { | ||||
|             link: 'http://mistserver.org/products' | ||||
|           }, | ||||
|           'Email for Help': {}, | ||||
|           'Terms & Conditions': { | ||||
|             link: 'http://mistserver.org/wiki/Mistserver_license' | ||||
|             link: 'http://mistserver.org/documentation#Legal' | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|  | @ -523,6 +520,9 @@ var UI = { | |||
|             ); | ||||
|           } | ||||
|           break; | ||||
|         case 'DOMfield': | ||||
|           $field = e.DOMfield; | ||||
|           break; | ||||
|         default: | ||||
|           $field = $('<input>').attr('type','text'); | ||||
|       } | ||||
|  | @ -946,169 +946,6 @@ var UI = { | |||
|      | ||||
|     return $table; | ||||
|   }, | ||||
|   buildLimits: function(limits,tab,other){ | ||||
|     var $c = $('<div>'); | ||||
|     var $table = $('<table>'); | ||||
|     $c.append($table); | ||||
|     var $thead = $('<tr>').append( | ||||
|       $('<th>').text('Kind').attr('data-sort-type','string') | ||||
|     ).append( | ||||
|       $('<th>').text('Type').attr('data-sort-type','string') | ||||
|     ).append( | ||||
|       $('<th>').text('Value') | ||||
|     ).append( | ||||
|       $('<th>') | ||||
|     ); | ||||
|     if (!(limits instanceof Array)) { | ||||
|       $thead.prepend( | ||||
|         $('<th>').text('Applies to').attr('data-sort-type','string').addClass('sorting-asc').addClass('applies_to') | ||||
|       ); | ||||
|     } | ||||
|     else { | ||||
|       limits = { | ||||
|         'noapply': limits | ||||
|       }; | ||||
|     } | ||||
|     $table.append( | ||||
|       $('<thead>').append($thead) | ||||
|     ); | ||||
|     var $tbody = $('<tbody>'); | ||||
|     $table.append($tbody); | ||||
|      | ||||
|     function buildrows(limit,apply) { | ||||
|       var $tb = $('<tbody>'); | ||||
|       for (var i in limit) { | ||||
|         var $tr = $('<tr>'); | ||||
|          | ||||
|         $tr.data('pointer',apply.concat(i)); | ||||
|          | ||||
|         var text = ''; | ||||
|         switch (apply[0]) { | ||||
|           case 'server': | ||||
|             text = 'The entire server'; | ||||
|             break; | ||||
|           case 'stream': | ||||
|             text = 'The stream "'+apply[1]+'"'; | ||||
|             break; | ||||
|         } | ||||
|         $tr.append( | ||||
|           $('<td>').addClass('applies_to').text(text) | ||||
|         ); | ||||
|          | ||||
|         var name = limit[i].name; | ||||
|         var value = limit[i].value; | ||||
|         switch (limit[i].name) { | ||||
|           case 'kbps_max': | ||||
|             name = 'Maximum bandwidth'; | ||||
|             value = UI.format.bytes(value,true); | ||||
|             break; | ||||
|           case 'users': | ||||
|             name = 'Maximum connected users'; | ||||
|             value = UI.format.number(value); | ||||
|             break; | ||||
|           case 'geo': | ||||
|             name = 'Geolimited'; | ||||
|             var vals = value; | ||||
|             var value = '<span class=unit>['+(value.charAt(0) == '-' ? 'Blacklist' : 'Whitelist')+']</span> '; | ||||
|             vals = vals.substr(1).split(' '); | ||||
|             var cs = []; | ||||
|             for (var j in vals) { | ||||
|               var c = UI.countrylist[vals[j]]; | ||||
|               cs.push(typeof c == 'undefined' ? vals[j] : c); | ||||
|             } | ||||
|             value += cs.join(', '); | ||||
|             break; | ||||
|           case 'host': | ||||
|             name = 'Hostlimited'; | ||||
|             var vals = value; | ||||
|             var value = '<span class=unit>['+(value.charAt(0) == '-' ? 'Blacklist' : 'Whitelist')+']</span> '; | ||||
|             vals = vals.substr(1).split(' '); | ||||
|             value += vals.join(', '); | ||||
|             break; | ||||
|         } | ||||
|          | ||||
|         $tr.append( | ||||
|           $('<td>').text(limit[i].type).addClass('kind') | ||||
|         ).append( | ||||
|           $('<td>').text(name).addClass('type') | ||||
|         ).append( | ||||
|           $('<td>').html(value).addClass('value') | ||||
|         ).append( | ||||
|           $('<td>').css('text-align','right').html( | ||||
|             $('<button>').text('Edit').click(function(){ | ||||
|               UI.navto('Edit Limit',$(this).closest('tr').data('pointer').join('^')); | ||||
|               UI.returnTab = [tab,other]; | ||||
|             }) | ||||
|           ).append( | ||||
|             $('<button>').text('Delete').click(function(){ | ||||
|               var $tr = $(this).closest('tr'); | ||||
|               var q = 'Are you sure you want to delete the '+$tr.find('.kind').text()+' '+$tr.find('.type').text()+' limit for '+$tr.find('.applies_to').text()+'?'; | ||||
|               if (confirm(q)) { | ||||
|                 var pointer = $tr.data('pointer'); | ||||
|                 var index = pointer[pointer.length-1]; | ||||
|                 var sendData = {}; | ||||
|                 switch (pointer[0]) { | ||||
|                   case 'server': | ||||
|                     mist.data.config.limits.splice(index,1); | ||||
|                     sendData = {config: mist.data.config}; | ||||
|                     break; | ||||
|                   case 'stream': | ||||
|                     mist.data.streams[pointer[1]].limits.splice(index,1); | ||||
|                     if (mist.data.LTS == 1) { | ||||
|                       var addstream = {}; | ||||
|                       addstream[pointer[1]] = mist.data.streams[pointer[1]]; | ||||
|                       sendData = {addstream: addstream}; | ||||
|                     } | ||||
|                     else { | ||||
|                       sendData = {streams: mist.data.streams}; | ||||
|                     } | ||||
|                     break; | ||||
|                 } | ||||
|                 mist.send(function(d){ | ||||
|                   UI.navto(tab,other); | ||||
|                 },sendData); | ||||
|               } | ||||
|             }) | ||||
|           ) | ||||
|         ); | ||||
|         $tb.append($tr); | ||||
|       } | ||||
|       return $tb.children(); | ||||
|     } | ||||
|      | ||||
|     for (var i in limits) { | ||||
|       switch (i) { | ||||
|         case 'server': | ||||
|           $tbody.append( | ||||
|             buildrows(limits[i],['server']) | ||||
|           ); | ||||
|           break; | ||||
|         case 'streams': | ||||
|           for (var j in limits[i]) { | ||||
|             $tbody.append( | ||||
|               buildrows(limits[i][j],['stream',j]) | ||||
|             ); | ||||
|           } | ||||
|           break; | ||||
|       } | ||||
|     } | ||||
|      | ||||
|     $c.prepend( | ||||
|       $('<button>').text('New limit').click(function(){ | ||||
|         UI.navto('Edit Limit'); | ||||
|         UI.returnTab = [tab,other]; | ||||
|       }) | ||||
|     ); | ||||
|      | ||||
|     if ($tbody.children().length > 0) { | ||||
|       $table.stupidtable(); | ||||
|     } | ||||
|     else { | ||||
|       $table.remove(); | ||||
|       $c.append($('<span>').text('No limits set.')); | ||||
|     } | ||||
|     return $c.children(); | ||||
|   }, | ||||
|   plot: { | ||||
|     addGraph: function(saveas,$graph_c){ | ||||
|       var graph = { | ||||
|  | @ -1118,9 +955,10 @@ var UI = { | |||
|         elements: { | ||||
|           cont: $('<div>').addClass('graph'), | ||||
|           plot: $('<div>').addClass('plot'), | ||||
|           legend: $('<div>').addClass('legend') | ||||
|           legend: $('<div>').addClass('legend').attr('draggable','true') | ||||
|         } | ||||
|       } | ||||
|       UI.draggable(graph.elements.legend); | ||||
|       graph.elements.cont.append( | ||||
|         graph.elements.plot | ||||
|       ).append( | ||||
|  | @ -1134,6 +972,7 @@ var UI = { | |||
|        | ||||
|       //get plotdata
 | ||||
|       //build request object
 | ||||
|       //changed data to show up in graphs to -15 sec to match API calls.
 | ||||
|       var reqobj = { | ||||
|         totals: [], | ||||
|         clients: [] | ||||
|  | @ -1146,9 +985,9 @@ var UI = { | |||
|             case 'upbps': | ||||
|             case 'downbps': | ||||
|               switch (set.origin[0]) { | ||||
|                 case 'total':     reqobj['totals'].push({fields: [set.datatype]});                               break; | ||||
|                 case 'stream':    reqobj['totals'].push({fields: [set.datatype], streams: [set.origin[1]]});     break; | ||||
|                 case 'protocol':  reqobj['totals'].push({fields: [set.datatype], protocols: [set.origin[1]]});   break; | ||||
|                 case 'total':     reqobj['totals'].push({fields: [set.datatype], end: -15});                               break; | ||||
|                 case 'stream':    reqobj['totals'].push({fields: [set.datatype], streams: [set.origin[1]], end: -15});     break; | ||||
|                 case 'protocol':  reqobj['totals'].push({fields: [set.datatype], protocols: [set.origin[1]], end: -15});   break; | ||||
|               } | ||||
|               break; | ||||
|             case 'cpuload': | ||||
|  | @ -1202,14 +1041,22 @@ var UI = { | |||
|                     borderWidth: {top: 0, right: 0, bottom: 1, left: 1}, | ||||
|                     color: 'black', | ||||
|                     backgroundColor: {colors: ['rgba(0,0,0,0)','rgba(0,0,0,0.025)']} | ||||
|                   }, | ||||
|                   crosshair: { | ||||
|                     mode: 'x' | ||||
|                   } | ||||
|                 } | ||||
|               ); | ||||
|                | ||||
|               //now the legend
 | ||||
|               var $list = $('<div>').addClass('legend-list').addClass('checklist'); | ||||
|               graph.elements.legend.html( | ||||
|               var $list = $('<table>').addClass('legend-list').addClass('nolay').html( | ||||
|                 $('<tr>').html( | ||||
|                   $('<td>').html( | ||||
|                     $('<h3>').text(graph.id) | ||||
|                   ) | ||||
|                 ).append( | ||||
|                   $('<td>').css('padding-right','2em').css('text-align','right').html( | ||||
|                     $('<span>').addClass('value') | ||||
|                   ).append( | ||||
|                     $('<button>').data('opts',graph).text('X').addClass('close').click(function(){ | ||||
|                       var graph = $(this).data('opts'); | ||||
|  | @ -1224,7 +1071,57 @@ var UI = { | |||
|                         UI.plot.go(graphs); | ||||
|                       } | ||||
|                     }) | ||||
|               ).append($list); | ||||
|                   ) | ||||
|                 ) | ||||
|               ); | ||||
|               graph.elements.legend.html($list); | ||||
|                | ||||
|               function updateLegendValues(x) { | ||||
|                 var $spans = graph.elements.legend.find('.value'); | ||||
|                 var n = 1; | ||||
|                  | ||||
|                 if (typeof x == 'undefined') { | ||||
|                   $spans.eq(0).html('Latest:'); | ||||
|                 } | ||||
|                 else { | ||||
|                   var axis = graph.plot.getXAxes()[0]; | ||||
|                   x = Math.min(axis.max,x); | ||||
|                   x = Math.max(axis.min,x); | ||||
|                   $spans.eq(0).html(UI.format.time(x/1e3)); | ||||
|                 } | ||||
|                  | ||||
|                  | ||||
|                 for (var i in graph.datasets) { | ||||
|                   var label = ' '; | ||||
|                   if (graph.datasets[i].display) { | ||||
|                     var tickformatter = UI.plot.yaxes[graph.datasets[i].yaxistype].tickFormatter; | ||||
|                     var data = graph.datasets[i].data; | ||||
|                     if (!x) { | ||||
|                       label = tickformatter(graph.datasets[i].data[graph.datasets[i].data.length-1][1]); | ||||
|                     } | ||||
|                     else { | ||||
|                       for (var j in data) { | ||||
|                         if (data[j][0] == x) { | ||||
|                           label = tickformatter(data[j][1]); | ||||
|                           break; | ||||
|                         } | ||||
|                         if (data[j][0] > x) { | ||||
|                           if (j != 0) { | ||||
|                             var p1 = data[j]; | ||||
|                             var p2 = data[j-1]; | ||||
|                             var y = p1[1] + (x - p1[0]) * (p2[1] - p1[1]) / (p2[0] - p1[0]); | ||||
|                             label = tickformatter(y); | ||||
|                           } | ||||
|                           break; | ||||
|                         } | ||||
|                       } | ||||
|                     } | ||||
|                   } | ||||
|                   $spans.eq(n).html(label); | ||||
|                   n++; | ||||
|                 } | ||||
|               } | ||||
|                | ||||
|               var plotdata = graph.plot.getOptions(); | ||||
|               for (var i in graph.datasets) { | ||||
|                 var $checkbox = $('<input>').attr('type','checkbox').data('index',i).data('graph',graph).click(function(){ | ||||
|  | @ -1243,12 +1140,19 @@ var UI = { | |||
|                   $checkbox.attr('checked','checked'); | ||||
|                 } | ||||
|                 $list.append( | ||||
|                   $('<tr>').html( | ||||
|                     $('<td>').html( | ||||
|                       $('<label>').html( | ||||
|                         $checkbox | ||||
|                       ).append( | ||||
|                         $('<div>').addClass('series-color').css('background-color',graph.datasets[i].color) | ||||
|                       ).append( | ||||
|                         graph.datasets[i].label | ||||
|                       ) | ||||
|                     ) | ||||
|                   ).append( | ||||
|                     $('<td>').css('padding-right','2em').css('text-align','right').html( | ||||
|                       $('<span>').addClass('value') | ||||
|                     ).append( | ||||
|                       $('<button>').text('X').addClass('close').data('index',i).data('graph',graph).click(function(){ | ||||
|                         var i = $(this).data('index'); | ||||
|  | @ -1276,11 +1180,18 @@ var UI = { | |||
|                         } | ||||
|                       }) | ||||
|                     ) | ||||
|                   ) | ||||
|                 ); | ||||
|               } | ||||
|               updateLegendValues(); | ||||
|                | ||||
|               //and the tooltip
 | ||||
|               var lastval = false; | ||||
|               graph.elements.plot.on('plothover',function(e,pos,item){ | ||||
|                 if (pos.x != lastval) { | ||||
|                   updateLegendValues(pos.x); | ||||
|                   lastval = pos.x; | ||||
|                 } | ||||
|                 if (item) { | ||||
|                   var $t = $('<span>').append( | ||||
|                     $('<h3>').text(item.series.label).prepend( | ||||
|  | @ -1307,6 +1218,8 @@ var UI = { | |||
|                 else { | ||||
|                   UI.tooltip.hide(); | ||||
|                 } | ||||
|               }).on('mouseout',function(){ | ||||
|                 updateLegendValues(); | ||||
|               }); | ||||
|                | ||||
|               break; | ||||
|  | @ -1396,7 +1309,7 @@ var UI = { | |||
|           } | ||||
|         }, | ||||
|         cpuload: { | ||||
|           label: 'CPU load', | ||||
|           label: 'CPU use', | ||||
|           yaxistype: 'percentage', | ||||
|           basecolor: [237,194,64], | ||||
|           cores: 1, | ||||
|  | @ -1411,7 +1324,7 @@ var UI = { | |||
|             if (removebefore !== false) { | ||||
|               this.data.splice(0,Number(removebefore)+1); | ||||
|             } | ||||
|             this.data.push([mist.data.config.time*1000,Math.min(100,mist.data.capabilities.load.one/this.cores)]); | ||||
|             this.data.push([mist.data.config.time*1000,mist.data.capabilities.cpu_use/10]); | ||||
|             return this.data; | ||||
|           } | ||||
|         }, | ||||
|  | @ -1557,6 +1470,35 @@ var UI = { | |||
|       } | ||||
|     } | ||||
|   }, | ||||
|   draggable: function(ele){ | ||||
|     ele.attr('draggable',true); | ||||
|     ele.on('dragstart',function(e){ | ||||
|       $(this).css('opacity',0.4).data('dragstart',{ | ||||
|         click: { | ||||
|           x: e.originalEvent.pageX, | ||||
|           y: e.originalEvent.pageY | ||||
|         }, | ||||
|         ele: { | ||||
|           x: this.offsetLeft, | ||||
|           y: this.offsetTop | ||||
|         } | ||||
|       }); | ||||
|     }).on('dragend',function(e){ | ||||
|       var old = $(this).data('dragstart'); | ||||
|       var x = old.ele.x - old.click.x + e.originalEvent.pageX; | ||||
|       var y = old.ele.y - old.click.y + e.originalEvent.pageY; | ||||
|       $(this).css({ | ||||
|         'opacity': 1, | ||||
|         'top': y, | ||||
|         'left': x, | ||||
|         'right' : 'auto', | ||||
|         'bottom' : 'auto' | ||||
|       }); | ||||
|     }); | ||||
|     ele.parent().on('dragleave',function(){ | ||||
|       //end the drag ?
 | ||||
|     }); | ||||
|   }, | ||||
|   format: { | ||||
|     time: function(secs,type){ | ||||
|       var d = new Date(secs * 1000); | ||||
|  | @ -1717,7 +1659,7 @@ var UI = { | |||
|       return; | ||||
|     } | ||||
|      | ||||
|     var $currbut = UI.elements.menu.css('visibility','visible').find('.button').filter(function(){ | ||||
|     var $currbut = UI.elements.menu.css('opacity','1').find('.button').filter(function(){ | ||||
|       if ($(this).find('.plain').text() == tab) { return true; } | ||||
|     }); | ||||
|     if ($currbut.length > 0) { | ||||
|  | @ -1738,7 +1680,7 @@ var UI = { | |||
|           UI.navto('Overview'); | ||||
|           return; | ||||
|         } | ||||
|         UI.elements.menu.css('visibility','hidden'); | ||||
|         UI.elements.menu.css('opacity','0'); | ||||
|         UI.elements.connection.status.text('Disconnected').removeClass('green').addClass('red'); | ||||
|         $main.append(UI.buildUI([ | ||||
|           { | ||||
|  | @ -2009,9 +1951,11 @@ var UI = { | |||
|             else { | ||||
|               $versioncheck.addClass('red').text('Version outdated!').append( | ||||
|                 $('<button>').text('Update').css({'font-size':'1em','margin-left':'1em'}).click(function(){ | ||||
|                   if (confirm('Are you sure you want to execute a rolling update?')) { | ||||
|                     mist.send(function(d){ | ||||
|                       UI.navto('Overview'); | ||||
|                     },{autoupdate: true}); | ||||
|                   } | ||||
|                 }) | ||||
|               ); | ||||
|             } | ||||
|  | @ -2472,7 +2416,7 @@ var UI = { | |||
|                     } | ||||
|                   } | ||||
|                 } | ||||
|                 if (d.browse.files.length) { | ||||
|                 if (('files' in d.browse) && (d.browse.files.length)) { | ||||
|                   allstreams[s].filesfound = true; | ||||
|                 } | ||||
|                 else { | ||||
|  | @ -2677,16 +2621,6 @@ var UI = { | |||
|           } | ||||
|         ])); | ||||
|          | ||||
|         if (editing) { | ||||
|           var limits = {streams: {}}; | ||||
|           limits.streams[streamname] = mist.data.streams[streamname].limits; | ||||
|           $main.append( | ||||
|             $('<h3>').text('Limits') | ||||
|           ).append( | ||||
|             UI.buildLimits(limits,tab,other) | ||||
|           ); | ||||
|         } | ||||
|          | ||||
|         break; | ||||
|       case 'Preview': | ||||
|         if (other == '') { | ||||
|  | @ -3233,224 +3167,6 @@ var UI = { | |||
|           } | ||||
|         } | ||||
|          | ||||
|         break; | ||||
|       case 'Limits': | ||||
|         $main.append(UI.buildUI([{ | ||||
|           type: 'help', | ||||
|           help: 'Here you can see an overview of all the limits you currently have. Limits are an LTS only feature and you can simply add new limits by selecting new, by selecting edit you can edit the existing limit, by selecting delete you can delete the existing limit.' | ||||
|         }])); | ||||
|         var limits = { | ||||
|           'server': mist.data.config.limits, | ||||
|           'streams': {} | ||||
|         }; | ||||
|         for (var i in mist.data.streams) { | ||||
|           if ('limits' in mist.data.streams[i]) { | ||||
|             limits.streams[i] = mist.data.streams[i].limits; | ||||
|           } | ||||
|         } | ||||
|         $main.append(UI.buildLimits(limits,tab,other)); | ||||
|         break; | ||||
|       case 'Edit Limit': | ||||
|         var editing = false; | ||||
|         if (other != '') { editing = true; } | ||||
|         var saveas = {}; | ||||
|         var build = [{ | ||||
|           type: 'help', | ||||
|           help: 'Here you can set the limit specifications, soft limits only warn while hard limits restrict access. Please feel free to set up limits according to your preferences.' | ||||
|         }]; | ||||
|         if (UI.returnTab[0] == 'Overview') { UI.returnTab =  ['Limits']; } | ||||
|          | ||||
|         if (!editing) { | ||||
|           $main.html( | ||||
|             $('<h2>').text('New limit') | ||||
|           ); | ||||
|            | ||||
|           var thestreams = []; | ||||
|           for (var i in mist.data.streams) { | ||||
|             thestreams.push(i); | ||||
|           } | ||||
|           var select = [ | ||||
|             ['server','The entire server'], | ||||
|             ['stream','The stream:',thestreams] | ||||
|           ]; | ||||
|           var appliesto = { | ||||
|             label: 'Applies to', | ||||
|             type: 'radioselect', | ||||
|             radioselect: select, | ||||
|             pointer: { | ||||
|               main: saveas, | ||||
|               index: 'applies_to' | ||||
|             }, | ||||
|             LTSonly: true, | ||||
|             validate: ['required'] | ||||
|           }; | ||||
|           if ((UI.returnTab[0] == 'Edit Stream') && (UI.returnTab[1])) { | ||||
|             appliesto.value = ['stream',UI.returnTab[1]]; | ||||
|             appliesto.readonly = true; | ||||
|           } | ||||
|           build.push(appliesto); | ||||
|         } | ||||
|         else { | ||||
|           var pointer = other.split('^'); | ||||
|           var text = tab; | ||||
|           switch (pointer[0]) { | ||||
|             case 'server': | ||||
|               saveas = mist.data.config.limits[pointer[1]]; | ||||
|               text = 'For the entire server'; | ||||
|               break; | ||||
|             case 'stream': | ||||
|               saveas = mist.data.streams[pointer[1]].limits[pointer[2]]; | ||||
|               text = 'For the stream "'+pointer[1]+'"'; | ||||
|               break; | ||||
|           } | ||||
|           build.push({ | ||||
|             type: 'text', | ||||
|             text: text | ||||
|           }); | ||||
|         } | ||||
|          | ||||
|         build = build.concat([$('<br>'),{ | ||||
|           label: 'Kind', | ||||
|           type: 'select', | ||||
|           select: [ | ||||
|             ['soft','Soft'], | ||||
|             ['hard','Hard'] | ||||
|           ], | ||||
|           pointer: { | ||||
|             main: saveas, | ||||
|             index: 'type' | ||||
|           }, | ||||
|           help: 'The server will not allow a hard limit to be passed. A soft limit can be used to set alerts.', | ||||
|           LTSonly: true, | ||||
|           validate: ['required'] | ||||
|         },{ | ||||
|           label: 'Value', | ||||
|           pointer: { | ||||
|             main: saveas, | ||||
|             index: 'value' | ||||
|           }, | ||||
|           classes: ['limit_value'], | ||||
|           LTSonly: true | ||||
|         },{ | ||||
|           label: 'Type', | ||||
|           type: 'select', | ||||
|           select: [ | ||||
|             ['kbps_max','Maximum bandwidth'], | ||||
|             ['users','Maximum connected users'], | ||||
|             ['geo','Geo limited'], | ||||
|             ['host','Host limited'] | ||||
|           ], | ||||
|           pointer: { | ||||
|             main: saveas, | ||||
|             index: 'name' | ||||
|           }, | ||||
|           LTSonly: true, | ||||
|           validate: ['required'], | ||||
|           classes: ['limit_type'], | ||||
|           'function': function(){ | ||||
|             var type = $(this).getval(); | ||||
|             var $lvalue = $(this).closest('.input_container').find('.limit_value'); | ||||
|             var $c = $lvalue.closest('label'); | ||||
|             var opts = $lvalue.data('opts'); | ||||
|              | ||||
|             var nopts = opts; | ||||
|             delete nopts.type; | ||||
|             delete nopts.min; | ||||
|             delete nopts.unit; | ||||
|             nopts.validate = ['required']; | ||||
|              | ||||
|             switch (type) { | ||||
|               case 'kbps_max': | ||||
|                 nopts.type = 'int'; | ||||
|                 nopts.min = 1; | ||||
|                 nopts.unit = 'bytes/s' | ||||
|                 break; | ||||
|               case 'users': | ||||
|                 nopts.type = 'int'; | ||||
|                 nopts.min = 1; | ||||
|                 break; | ||||
|               case 'geo': | ||||
|                 nopts.type = 'geolimited'; | ||||
|                 break; | ||||
|               case 'host': | ||||
|                 nopts.type = 'hostlimited'; | ||||
|                 break; | ||||
|             } | ||||
|             var $nc = UI.buildUI([nopts]); | ||||
|             $c.replaceWith($nc.children()); | ||||
|           } | ||||
|         },{ | ||||
|           type: 'buttons', | ||||
|           buttons: [ | ||||
|             { | ||||
|               type: 'cancel', | ||||
|               label: 'Cancel', | ||||
|               'function': function(){ | ||||
|                 UI.navto(UI.returnTab[0],UI.returnTab[1]); | ||||
|               } | ||||
|             },{ | ||||
|               type: 'save', | ||||
|               label: 'Save', | ||||
|               'function': function(){ | ||||
|                 var send = {}; | ||||
|                 if (editing) { | ||||
|                   var pointer = other.split('^'); | ||||
|                   switch (pointer[0]) { | ||||
|                     case 'server': | ||||
|                       send = {config: {limits: mist.data.config.limits}}; | ||||
|                       break; | ||||
|                     case 'stream': | ||||
|                       if (mist.data.LTS) { | ||||
|                         send = {addstream: {}}; | ||||
|                         send.addstream[pointer[1]] = mist.data.streams[pointer[1]]; | ||||
|                       } | ||||
|                       else { | ||||
|                         send = {streams: mist.data.streams}; | ||||
|                       } | ||||
|                       break; | ||||
|                   } | ||||
|                 } | ||||
|                 else { | ||||
|                   var pointer = saveas.applies_to; | ||||
|                   delete saveas.applies_to; | ||||
|                   switch (pointer[0]) { | ||||
|                     case 'server': | ||||
|                       if (!mist.data.config.limits) { | ||||
|                         mist.data.config.limits = []; | ||||
|                       } | ||||
|                       mist.data.config.limits.push(saveas); | ||||
|                       send = {config: {limits: mist.data.config.limits}}; | ||||
|                       break; | ||||
|                     case 'stream': | ||||
|                       if (typeof mist.data.streams[pointer[1]].limits == 'undefined') { | ||||
|                         mist.data.streams[pointer[1]].limits = []; | ||||
|                       } | ||||
|                       mist.data.streams[pointer[1]].limits.push(saveas); | ||||
|                       if (mist.data.LTS) { | ||||
|                         send = {addstream: {}}; | ||||
|                         send.addstream[pointer[1]] = mist.data.streams[pointer[1]]; | ||||
|                       } | ||||
|                       else { | ||||
|                         send = {streams: mist.data.streams}; | ||||
|                       } | ||||
|                       break; | ||||
|                   } | ||||
|                 } | ||||
|                 mist.send(function(){ | ||||
|                   UI.navto(UI.returnTab[0],UI.returnTab[1]); | ||||
|                 },send) | ||||
|               } | ||||
|             } | ||||
|           ] | ||||
|         }]); | ||||
|         var $UI = UI.buildUI(build); | ||||
|         $main.append($UI); | ||||
|         //draw the type input after the value input, but show the value input last
 | ||||
|         $UI.find('.limit_type').closest('label').after( | ||||
|           $UI.find('.limit_value').closest('label') | ||||
|         ); | ||||
|         $UI.find('.limit_type').trigger('change'); | ||||
|          | ||||
|         break; | ||||
|       case 'Triggers': | ||||
|         if (!('triggers' in mist.data.config)) { | ||||
|  | @ -3675,6 +3391,13 @@ var UI = { | |||
|          | ||||
|         break; | ||||
|       case 'Logs': | ||||
|         var $refreshbutton = $('<button>').text('Refresh now').click(function(){ | ||||
|           $(this).text('Loading..'); | ||||
|           mist.send(function(){ | ||||
|             buildLogsTable(); | ||||
|             $refreshbutton.text('Refresh now'); | ||||
|           }); | ||||
|         }).css('padding','0.2em 0.5em').css('flex-grow',0); | ||||
|          | ||||
|         $main.append(UI.buildUI([{ | ||||
|           type: 'help', | ||||
|  | @ -3690,13 +3413,19 @@ var UI = { | |||
|           ], | ||||
|           value: 30, | ||||
|           'function': function(){ | ||||
|             clearInterval(UI.interval); | ||||
|             UI.interval.clear(); | ||||
|             UI.interval.set(function(){ | ||||
|               mist.send(function(){ | ||||
|                 buildLogsTable(); | ||||
|               }); | ||||
|             },$(this).val()*1e3); | ||||
|           } | ||||
|           }, | ||||
|           help: 'How often the table below should be updated.' | ||||
|         },{ | ||||
|           label: '..or', | ||||
|           type: 'DOMfield', | ||||
|           DOMfield: $refreshbutton, | ||||
|           help: 'Instantly refresh the table below.' | ||||
|         }])); | ||||
|          | ||||
|         $main.append( | ||||
|  | @ -3859,7 +3588,7 @@ var UI = { | |||
|               ['clients','Connections'], | ||||
|               ['upbps','Bandwidth (up)'], | ||||
|               ['downbps','Bandwidth (down)'], | ||||
|               ['cpuload','CPU load'], | ||||
|               ['cpuload','CPU use'], | ||||
|               ['memload','Memory load'], | ||||
|               ['coords','Client location'] | ||||
|             ], | ||||
|  | @ -4025,14 +3754,14 @@ var UI = { | |||
|           //CPU loading/total amount of cores is a percentage, over 1 means there are tasks waiting.
 | ||||
|           var loading = { | ||||
|             vheader: 'Load average', | ||||
|             labels: ['1 minute','5 minutes','15 minutes',''], | ||||
|             labels: ['CPU use','1 minute','5 minutes','15 minutes'], | ||||
|             content: [{ | ||||
|               header: ' ', | ||||
|               body: [ | ||||
|                 UI.format.addUnit(UI.format.number(mist.data.capabilities.cpu_use/10),'%'), | ||||
|                 UI.format.number(load.one/100), | ||||
|                 UI.format.number(load.five/100), | ||||
|                 UI.format.number(load.fifteen/100), | ||||
|                 '' | ||||
|                 UI.format.number(load.fifteen/100) | ||||
|               ] | ||||
|             }] | ||||
|           }; | ||||
|  | @ -4143,7 +3872,7 @@ var UI = { | |||
|                 $(me).text('Sending..'); | ||||
|                 $.ajax({ | ||||
|                   type: 'POST', | ||||
|                   url: 'http://mistserver.org/contact_us?skin=plain', | ||||
|                   url: 'http://mistserver.org/contact?skin=plain', | ||||
|                   data: saveas, | ||||
|                   success: function(d) { | ||||
|                     var $s = $('<span>').html(d); | ||||
|  | @ -4175,7 +3904,7 @@ var mist = { | |||
|   user: { | ||||
|     name: '', | ||||
|     password: '', | ||||
|     host: 'http://'+(location.hostname ? location.hostname : 'localhost')+':4242/api' | ||||
|     host: location.origin+location.pathname.replace(/\/+$/, "")+'/api' | ||||
|   }, | ||||
|   send: function(callback,sendData,opts){ | ||||
|     sendData = sendData || {}; | ||||
|  | @ -4277,9 +4006,14 @@ var mist = { | |||
|               ); | ||||
|             } | ||||
|             if ('totals' in d) { | ||||
|                | ||||
|               //reformat to something more readable
 | ||||
|               function reformat(main) { | ||||
|                 function insertZero() { | ||||
|                 function insertZero(overridetime) { | ||||
|                   if (typeof overridetime == 'undefined') { | ||||
|                     overridetime = time; | ||||
|                   } | ||||
|                    | ||||
|                   for (var j in main.fields) { | ||||
|                     obj[main.fields[j]].push([time,0]); | ||||
|                   } | ||||
|  | @ -4296,13 +4030,13 @@ var mist = { | |||
|                   //no data
 | ||||
|                   time = (mist.data.config.time - 600)*1e3; | ||||
|                   insertZero(); | ||||
|                   time = mist.data.config.time*1e3; | ||||
|                   time = (mist.data.config.time - 15)*1e3; | ||||
|                   insertZero(); | ||||
|                 } | ||||
|                 else { | ||||
|                   //leading 0?
 | ||||
|                   if ((main.end - main.start) < 600) { | ||||
|                     time = (main.end - 600)*1e3; | ||||
|                   if (main.start > (mist.data.config.time - 600)) { | ||||
|                     time = (mist.data.config.time - 600)*1e3; | ||||
|                     insertZero(); | ||||
|                     time = main.start*1e3; | ||||
|                     insertZero(); | ||||
|  | @ -4324,11 +4058,13 @@ var mist = { | |||
|                         interval_n++; //go to next interval
 | ||||
|                          | ||||
|                         //insert zeros between the intervals
 | ||||
|                         if (interval_n < main.interval.length-1) { insert = 2; }  | ||||
|                         //+= 2 in case the interval is only 1 long
 | ||||
|                         if (interval_n < main.interval.length-1) { insert += 2; } | ||||
|                       } | ||||
|                     } | ||||
|                      | ||||
|                     if (insert == 1) { | ||||
|                     if (insert % 2 == 1) { | ||||
|                       //modulus in case the interval was only 1 long; prevents diagonal lines
 | ||||
|                       insertZero(); | ||||
|                       insert--; | ||||
|                     } | ||||
|  | @ -4345,9 +4081,9 @@ var mist = { | |||
|                   } | ||||
|                    | ||||
|                   //trailing 0?
 | ||||
|                   if ((mist.data.config.time - main.end) > 5) { | ||||
|                   if ((mist.data.config.time - main.end) > 20) { | ||||
|                     insertZero(); | ||||
|                     time = mist.data.config.time * 1e3; | ||||
|                     time = (mist.data.config.time -15) * 1e3; | ||||
|                     insertZero(); | ||||
|                   } | ||||
|                 } | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Thulinma
						Thulinma