diff --git a/Doxyfile b/Doxyfile index be3218cc..794b6cbc 100644 --- a/Doxyfile +++ b/Doxyfile @@ -220,7 +220,7 @@ TAB_SIZE = 2 # "Side Effects:". You can put \n's in the value part of an alias to insert # newlines. -ALIASES = +ALIASES = "api=\xrefitem api \"API call\" \"API calls\"" # This tag can be used to specify a number of word-keyword mappings (TCL only). # A mapping has the form "name=value". For example adding "class=itcl::class" @@ -794,7 +794,7 @@ EXCLUDE_SYMLINKS = NO # Note that the wildcards are matched against the file with absolute path, so to # exclude all test directories for example use the pattern */test/* -EXCLUDE_PATTERNS = */.git/* +EXCLUDE_PATTERNS = */.git/* */tinythread.* # The EXCLUDE_SYMBOLS tag can be used to specify one or more symbol names # (namespaces, classes, functions, etc.) that should be excluded from the diff --git a/Makefile b/Makefile index 01845249..a3ae5abd 100644 --- a/Makefile +++ b/Makefile @@ -23,7 +23,7 @@ LDLIBS = -lmist -lrt .DEFAULT_GOAL := all -all: controller buffers connectors analysers converters +all: MistConnHTTP controller analysers inputs outputs DOXYGEN := $(shell doxygen -v 2> /dev/null) ifdef DOXYGEN @@ -33,71 +33,15 @@ $(warning Doxygen not installed - not building source documentation.) endif controller: MistController +MistController: override LDLIBS += $(THREADLIB) MistController: src/controller/server.html.h src/controller/* $(CXX) $(LDFLAGS) $(CPPFLAGS) src/controller/*.cpp $(LDLIBS) -o $@ -buffers: MistPlayer -MistPlayer: src/buffer/player.cpp - $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ - -buffers: MistBuffer -MistBuffer: override LDLIBS += $(THREADLIB) -MistBuffer: src/buffer/buffer.cpp src/buffer/buffer_stream.h src/buffer/buffer_stream.cpp - $(CXX) $(LDFLAGS) $(CPPFLAGS) src/buffer/buffer.cpp src/buffer/buffer_stream.cpp $(LDLIBS) -o $@ - -connectors: MistConnRaw -MistConnRaw: src/connectors/conn_raw.cpp - $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ - -connectors: MistConnRTMP -MistConnRTMP: src/connectors/conn_rtmp.cpp - $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ - connectors: MistConnHTTP MistConnHTTP: override LDLIBS += $(THREADLIB) MistConnHTTP: src/connectors/conn_http.cpp src/connectors/embed.js.h src/connectors/icon.h $(CXX) $(LDFLAGS) $(CPPFLAGS) $< $(LDLIBS) -o $@ -connectors: MistConnHTTPProgressiveFLV -MistConnHTTPProgressiveFLV: src/connectors/conn_http_progressive_flv.cpp - $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ - -connectors: MistConnHTTPProgressiveMP3 -MistConnHTTPProgressiveMP3: src/connectors/conn_http_progressive_mp3.cpp - $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ - -connectors: MistConnHTTPProgressiveMP4 -MistConnHTTPProgressiveMP4: src/connectors/conn_http_progressive_mp4.cpp - $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ - -connectors: MistConnHTTPProgressiveOGG -MistConnHTTPProgressiveOGG: src/connectors/conn_http_progressive_ogg.cpp src/converters/oggconv.cpp - $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ - -connectors: MistConnHTTPDynamic -MistConnHTTPDynamic: src/connectors/conn_http_dynamic.cpp - $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ - -connectors: MistConnHTTPSmooth -MistConnHTTPSmooth: src/connectors/conn_http_smooth.cpp src/connectors/xap.h - $(CXX) $(LDFLAGS) $(CPPFLAGS) $< $(LDLIBS) -o $@ - -connectors: MistConnHTTPLive -MistConnHTTPLive: src/connectors/conn_http_live.cpp - $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ - -connectors: MistConnHTTPSRT -MistConnHTTPSRT: src/connectors/conn_http_srt.cpp - $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ - -connectors: MistConnHTTPJSON -MistConnHTTPJSON: src/connectors/conn_http_json.cpp - $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ - -connectors: MistConnTS -MistConnTS: src/connectors/conn_ts.cpp - $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ - analysers: MistAnalyserRTMP MistAnalyserRTMP: src/analysers/rtmp_analyser.cpp $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ @@ -134,14 +78,6 @@ converters: MistFLV2DTSC MistFLV2DTSC: src/converters/flv2dtsc.cpp $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ -converters: MistOGG2DTSC -MistOGG2DTSC: src/converters/ogg2dtsc.cpp - $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ - -converters: MistDTSC2OGG -MistDTSC2OGG: src/converters/dtsc2ogg.cpp src/converters/oggconv.cpp - $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ - converters: MistDTSCFix MistDTSCFix: src/converters/dtscfix.cpp $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ @@ -162,8 +98,98 @@ converters: MistDTSC2SRT MistDTSC2SRT: src/converters/dtsc2srt.cpp $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ +inputs: MistInDTSC +MistInDTSC: override LDLIBS += $(THREADLIB) +MistInDTSC: override CPPFLAGS += "-DINPUTTYPE=\"input_dtsc.h\"" +MistInDTSC: src/input/mist_in.cpp src/input/input.cpp src/input/input_dtsc.cpp + $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ + +inputs: MistInFLV +MistInFLV: override LDLIBS += $(THREADLIB) +MistInFLV: override CPPFLAGS += "-DINPUTTYPE=\"input_flv.h\"" +MistInFLV: src/input/mist_in.cpp src/input/input.cpp src/input/input_flv.cpp + $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ + +inputs: MistInOGG +MistInOGG: override LDLIBS += $(THREADLIB) +MistInOGG: override CPPFLAGS += "-DINPUTTYPE=\"input_ogg.h\"" +MistInOGG: src/input/mist_in.cpp src/input/input.cpp src/input/input_ogg.cpp + $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ + +inputs: MistInBuffer +MistInBuffer: override LDLIBS += $(THREADLIB) +MistInBuffer: override CPPFLAGS += "-DINPUTTYPE=\"input_buffer.h\"" +MistInBuffer: src/input/mist_in.cpp src/input/input.cpp src/input/input_buffer.cpp + $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ + +outputs: MistOutFLV +MistOutFLV: override LDLIBS += $(THREADLIB) +MistOutFLV: override CPPFLAGS += "-DOUTPUTTYPE=\"output_progressive_flv.h\"" +MistOutFLV: src/output/mist_out.cpp src/output/output.cpp src/output/output_progressive_flv.cpp + $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ + +outputs: MistOutMP4 +MistOutMP4: override LDLIBS += $(THREADLIB) +MistOutMP4: override CPPFLAGS += "-DOUTPUTTYPE=\"output_progressive_mp4.h\"" +MistOutMP4: src/output/mist_out.cpp src/output/output.cpp src/output/output_progressive_mp4.cpp + $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ + +outputs: MistOutMP3 +MistOutMP3: override LDLIBS += $(THREADLIB) +MistOutMP3: override CPPFLAGS += "-DOUTPUTTYPE=\"output_progressive_mp3.h\"" +MistOutMP3: src/output/mist_out.cpp src/output/output.cpp src/output/output_progressive_mp3.cpp + $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ + +outputs: MistOutRTMP +MistOutRTMP: override LDLIBS += $(THREADLIB) +MistOutRTMP: override CPPFLAGS += "-DOUTPUTTYPE=\"output_rtmp.h\"" +MistOutRTMP: src/output/mist_out.cpp src/output/output.cpp src/output/output_rtmp.cpp + $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ + +outputs: MistOutRaw +MistOutRaw: override LDLIBS += $(THREADLIB) +MistOutRaw: override CPPFLAGS += "-DOUTPUTTYPE=\"output_raw.h\"" +MistOutRaw: src/output/mist_out.cpp src/output/output.cpp src/output/output_raw.cpp + $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ + +outputs: MistOutTS +MistOutTS: override LDLIBS += $(THREADLIB) +MistOutTS: override CPPFLAGS += "-DOUTPUTTYPE=\"output_ts.h\"" +MistOutTS: src/output/mist_out.cpp src/output/output.cpp src/output/output_ts.cpp + $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ + +outputs: MistOutHSS +MistOutHSS: override LDLIBS += $(THREADLIB) +MistOutHSS: override CPPFLAGS += "-DOUTPUTTYPE=\"output_hss.h\"" +MistOutHSS: src/output/mist_out.cpp src/output/output.cpp src/output/output_hss.cpp + $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ + +outputs: MistOutHLS +MistOutHLS: override LDLIBS += $(THREADLIB) +MistOutHLS: override CPPFLAGS += "-DOUTPUTTYPE=\"output_hls.h\"" +MistOutHLS: src/output/mist_out.cpp src/output/output.cpp src/output/output_hls.cpp + $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ + +outputs: MistOutHDS +MistOutHDS: override LDLIBS += $(THREADLIB) +MistOutHDS: override CPPFLAGS += "-DOUTPUTTYPE=\"output_hds.h\"" +MistOutHDS: src/output/mist_out.cpp src/output/output.cpp src/output/output_hds.cpp + $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ + +outputs: MistOutSRT +MistOutSRT: override LDLIBS += $(THREADLIB) +MistOutSRT: override CPPFLAGS += "-DOUTPUTTYPE=\"output_srt.h\"" +MistOutSRT: src/output/mist_out.cpp src/output/output.cpp src/output/output_srt.cpp + $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ + +outputs: MistOutJSON +MistOutJSON: override LDLIBS += $(THREADLIB) +MistOutJSON: override CPPFLAGS += "-DOUTPUTTYPE=\"output_json.h\"" +MistOutJSON: src/output/mist_out.cpp src/output/output.cpp src/output/output_json.cpp + $(CXX) $(LDFLAGS) $(CPPFLAGS) $^ $(LDLIBS) -o $@ + BUILT_SOURCES=controller/server.html.h connectors/embed.js.h -lspSOURCES=lsp/jquery.js lsp/placeholder.js lsp/md5.js lsp/main.js lsp/pages.js lsp/tablesort.js +lspSOURCES=lsp/plugins/jquery.js lsp/plugins/placeholder.js lsp/plugins/md5.js lsp/main.js lsp/pages.js lsp/plugins/tablesort.js lsp/plugins/jquery.flot.min.js lsp/plugins/jquery.flot.time.min.js lsp/plugins/jquery.flot.crosshair.min.js lspDATA=lsp/header.html lsp/main.css lsp/footer.html JAVA := $(shell which java 2> /dev/null) @@ -201,7 +227,7 @@ clean: rm -f *.o Mist* sourcery src/controller/server.html src/connectors/embed.js.h src/controller/server.html.h rm -rf ./docs -install: controller buffers connectors analysers converters +install: all install ./Mist* $(DESTDIR)$(bindir) uninstall: diff --git a/lsp/footer.html b/lsp/footer.html index b9209e1e..f2fe2ffe 100644 --- a/lsp/footer.html +++ b/lsp/footer.html @@ -1,3 +1,4 @@ + <link rel='icon' href=''> <script> //these are placed here because the compression compiler does not deal with the eval function properly. //enter the values of the settings object into their input fields @@ -45,8 +46,8 @@ <div id='header'> <div id='logo'> - <a href='http://mistserver.org' target='_blank'> - <span>Mist/</span> Server + <a> + <span>Mist</span>Server Management Interface </a> </div> <div id='status'> @@ -56,20 +57,31 @@ </div> </div> - <div id='menu'> + <div id='menu' class='menu'> <div class='button'>Overview</div> <div class='button'>Protocols</div> <div class='button'>Streams</div> + <div class='button'>Preview</div> <div class='button LTS-only'>Limits</div> - <div class='button'>Conversion</div> + <!--<div class='button'>Conversion</div>--> <div class='button'>Logs</div> + <!--<div class='button'>Statistics</div>--> <div class='button'>Server Stats</div> <br> <div class='button red'>Disconnect</div> - <a class='button' href='http://shop.mistserver.org' target='_blank'>Mist Shop</a> + <br> + <div class='expandbutton'> + Tools <span class=arrowdown></span> + <div class='expandcontainer'> + <a class='button' href='http://mistserver.org/products' target='_blank'>Mist Shop</a> + <a class='button' href='http://mistserver.org/streamtester' target='_blank'>Stream Tester</a> + <a class='button' href='http://mistserver.org/wiki/Category:Guides' target='_blank'>Guides</a> + <div class='button'>Email for Help</div> + </div> + </div> </div> <div id='page'></div> - + <div id='ih-button' title='Activate integrated help'>?</div> </body> </html> \ No newline at end of file diff --git a/lsp/header.html b/lsp/header.html index 50a544d9..5bb1b924 100644 --- a/lsp/header.html +++ b/lsp/header.html @@ -1,4 +1,4 @@ <html> <head> <meta http-equiv='content-type' content='text/html;charset=utf-8' /> - <title>MistServer Manager</title> \ No newline at end of file + <title>MistServer MI</title> \ No newline at end of file diff --git a/lsp/jquery.js b/lsp/jquery.js deleted file mode 100755 index 674c18e2..00000000 --- a/lsp/jquery.js +++ /dev/null @@ -1,5 +0,0 @@ - /*! jQuery v1.7.1 jquery.com | jquery.org/license */ - (function(a,b){function cy(a){return f.isWindow(a)?a:a.nodeType===9?a.defaultView||a.parentWindow:!1}function cv(a){if(!ck[a]){var b=c.body,d=f("<"+a+">").appendTo(b),e=d.css("display");d.remove();if(e==="none"||e===""){cl||(cl=c.createElement("iframe"),cl.frameBorder=cl.width=cl.height=0),b.appendChild(cl);if(!cm||!cl.createElement)cm=(cl.contentWindow||cl.contentDocument).document,cm.write((c.compatMode==="CSS1Compat"?"<!doctype html>":"")+"<html><body>"),cm.close();d=cm.createElement(a),cm.body.appendChild(d),e=f.css(d,"display"),b.removeChild(cl)}ck[a]=e}return ck[a]}function cu(a,b){var c={};f.each(cq.concat.apply([],cq.slice(0,b)),function(){c[this]=a});return c}function ct(){cr=b}function cs(){setTimeout(ct,0);return cr=f.now()}function cj(){try{return new a.ActiveXObject("Microsoft.XMLHTTP")}catch(b){}}function ci(){try{return new a.XMLHttpRequest}catch(b){}}function cc(a,c){a.dataFilter&&(c=a.dataFilter(c,a.dataType));var d=a.dataTypes,e={},g,h,i=d.length,j,k=d[0],l,m,n,o,p;for(g=1;g<i;g++){if(g===1)for(h in a.converters)typeof h=="string"&&(e[h.toLowerCase()]=a.converters[h]);l=k,k=d[g];if(k==="*")k=l;else if(l!=="*"&&l!==k){m=l+" "+k,n=e[m]||e["* "+k];if(!n){p=b;for(o in e){j=o.split(" ");if(j[0]===l||j[0]==="*"){p=e[j[1]+" "+k];if(p){o=e[o],o===!0?n=p:p===!0&&(n=o);break}}}}!n&&!p&&f.error("No conversion from "+m.replace(" "," to ")),n!==!0&&(c=n?n(c):p(o(c)))}}return c}function cb(a,c,d){var e=a.contents,f=a.dataTypes,g=a.responseFields,h,i,j,k;for(i in g)i in d&&(c[g[i]]=d[i]);while(f[0]==="*")f.shift(),h===b&&(h=a.mimeType||c.getResponseHeader("content-type"));if(h)for(i in e)if(e[i]&&e[i].test(h)){f.unshift(i);break}if(f[0]in d)j=f[0];else{for(i in d){if(!f[0]||a.converters[i+" "+f[0]]){j=i;break}k||(k=i)}j=j||k}if(j){j!==f[0]&&f.unshift(j);return d[j]}}function ca(a,b,c,d){if(f.isArray(b))f.each(b,function(b,e){c||bE.test(a)?d(a,e):ca(a+"["+(typeof e=="object"||f.isArray(e)?b:"")+"]",e,c,d)});else if(!c&&b!=null&&typeof b=="object")for(var e in b)ca(a+"["+e+"]",b[e],c,d);else d(a,b)}function b_(a,c){var d,e,g=f.ajaxSettings.flatOptions||{};for(d in c)c[d]!==b&&((g[d]?a:e||(e={}))[d]=c[d]);e&&f.extend(!0,a,e)}function b$(a,c,d,e,f,g){f=f||c.dataTypes[0],g=g||{},g[f]=!0;var h=a[f],i=0,j=h?h.length:0,k=a===bT,l;for(;i<j&&(k||!l);i++)l=h[i](c,d,e),typeof l=="string"&&(!k||g[l]?l=b:(c.dataTypes.unshift(l),l=b$(a,c,d,e,l,g)));(k||!l)&&!g["*"]&&(l=b$(a,c,d,e,"*",g));return l}function bZ(a){return function(b,c){typeof b!="string"&&(c=b,b="*");if(f.isFunction(c)){var d=b.toLowerCase().split(bP),e=0,g=d.length,h,i,j;for(;e<g;e++)h=d[e],j=/^\+/.test(h),j&&(h=h.substr(1)||"*"),i=a[h]=a[h]||[],i[j?"unshift":"push"](c)}}}function bC(a,b,c){var d=b==="width"?a.offsetWidth:a.offsetHeight,e=b==="width"?bx:by,g=0,h=e.length;if(d>0){if(c!=="border")for(;g<h;g++)c||(d-=parseFloat(f.css(a,"padding"+e[g]))||0),c==="margin"?d+=parseFloat(f.css(a,c+e[g]))||0:d-=parseFloat(f.css(a,"border"+e[g]+"Width"))||0;return d+"px"}d=bz(a,b,b);if(d<0||d==null)d=a.style[b]||0;d=parseFloat(d)||0;if(c)for(;g<h;g++)d+=parseFloat(f.css(a,"padding"+e[g]))||0,c!=="padding"&&(d+=parseFloat(f.css(a,"border"+e[g]+"Width"))||0),c==="margin"&&(d+=parseFloat(f.css(a,c+e[g]))||0);return d+"px"}function bp(a,b){b.src?f.ajax({url:b.src,async:!1,dataType:"script"}):f.globalEval((b.text||b.textContent||b.innerHTML||"").replace(bf,"/*$0*/")),b.parentNode&&b.parentNode.removeChild(b)}function bo(a){var b=c.createElement("div");bh.appendChild(b),b.innerHTML=a.outerHTML;return b.firstChild}function bn(a){var b=(a.nodeName||"").toLowerCase();b==="input"?bm(a):b!=="script"&&typeof a.getElementsByTagName!="undefined"&&f.grep(a.getElementsByTagName("input"),bm)}function bm(a){if(a.type==="checkbox"||a.type==="radio")a.defaultChecked=a.checked}function bl(a){return typeof a.getElementsByTagName!="undefined"?a.getElementsByTagName("*"):typeof a.querySelectorAll!="undefined"?a.querySelectorAll("*"):[]}function bk(a,b){var c;if(b.nodeType===1){b.clearAttributes&&b.clearAttributes(),b.mergeAttributes&&b.mergeAttributes(a),c=b.nodeName.toLowerCase();if(c==="object")b.outerHTML=a.outerHTML;else if(c!=="input"||a.type!=="checkbox"&&a.type!=="radio"){if(c==="option")b.selected=a.defaultSelected;else if(c==="input"||c==="textarea")b.defaultValue=a.defaultValue}else a.checked&&(b.defaultChecked=b.checked=a.checked),b.value!==a.value&&(b.value=a.value);b.removeAttribute(f.expando)}}function bj(a,b){if(b.nodeType===1&&!!f.hasData(a)){var c,d,e,g=f._data(a),h=f._data(b,g),i=g.events;if(i){delete h.handle,h.events={};for(c in i)for(d=0,e=i[c].length;d<e;d++)f.event.add(b,c+(i[c][d].namespace?".":"")+i[c][d].namespace,i[c][d],i[c][d].data)}h.data&&(h.data=f.extend({},h.data))}}function bi(a,b){return f.nodeName(a,"table")?a.getElementsByTagName("tbody")[0]||a.appendChild(a.ownerDocument.createElement("tbody")):a}function U(a){var b=V.split("|"),c=a.createDocumentFragment();if(c.createElement)while(b.length)c.createElement(b.pop());return c}function T(a,b,c){b=b||0;if(f.isFunction(b))return f.grep(a,function(a,d){var e=!!b.call(a,d,a);return e===c});if(b.nodeType)return f.grep(a,function(a,d){return a===b===c});if(typeof b=="string"){var d=f.grep(a,function(a){return a.nodeType===1});if(O.test(b))return f.filter(b,d,!c);b=f.filter(b,d)}return f.grep(a,function(a,d){return f.inArray(a,b)>=0===c})}function S(a){return!a||!a.parentNode||a.parentNode.nodeType===11}function K(){return!0}function J(){return!1}function n(a,b,c){var d=b+"defer",e=b+"queue",g=b+"mark",h=f._data(a,d);h&&(c==="queue"||!f._data(a,e))&&(c==="mark"||!f._data(a,g))&&setTimeout(function(){!f._data(a,e)&&!f._data(a,g)&&(f.removeData(a,d,!0),h.fire())},0)}function m(a){for(var b in a){if(b==="data"&&f.isEmptyObject(a[b]))continue;if(b!=="toJSON")return!1}return!0}function l(a,c,d){if(d===b&&a.nodeType===1){var e="data-"+c.replace(k,"-$1").toLowerCase();d=a.getAttribute(e);if(typeof d=="string"){try{d=d==="true"?!0:d==="false"?!1:d==="null"?null:f.isNumeric(d)?parseFloat(d):j.test(d)?f.parseJSON(d):d}catch(g){}f.data(a,c,d)}else d=b}return d}function h(a){var b=g[a]={},c,d;a=a.split(/\s+/);for(c=0,d=a.length;c<d;c++)b[a[c]]=!0;return b}var c=a.document,d=a.navigator,e=a.location,f=function(){function J(){if(!e.isReady){try{c.documentElement.doScroll("left")}catch(a){setTimeout(J,1);return}e.ready()}}var e=function(a,b){return new e.fn.init(a,b,h)},f=a.jQuery,g=a.$,h,i=/^(?:[^#<]*(<[\w\W]+>)[^>]*$|#([\w\-]*)$)/,j=/\S/,k=/^\s+/,l=/\s+$/,m=/^<(\w+)\s*\/?>(?:<\/\1>)?$/,n=/^[\],:{}\s]*$/,o=/\\(?:["\\\/bfnrt]|u[0-9a-fA-F]{4})/g,p=/"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g,q=/(?:^|:|,)(?:\s*\[)+/g,r=/(webkit)[ \/]([\w.]+)/,s=/(opera)(?:.*version)?[ \/]([\w.]+)/,t=/(msie) ([\w.]+)/,u=/(mozilla)(?:.*? rv:([\w.]+))?/,v=/-([a-z]|[0-9])/ig,w=/^-ms-/,x=function(a,b){return(b+"").toUpperCase()},y=d.userAgent,z,A,B,C=Object.prototype.toString,D=Object.prototype.hasOwnProperty,E=Array.prototype.push,F=Array.prototype.slice,G=String.prototype.trim,H=Array.prototype.indexOf,I={};e.fn=e.prototype={constructor:e,init:function(a,d,f){var g,h,j,k;if(!a)return this;if(a.nodeType){this.context=this[0]=a,this.length=1;return this}if(a==="body"&&!d&&c.body){this.context=c,this[0]=c.body,this.selector=a,this.length=1;return this}if(typeof a=="string"){a.charAt(0)!=="<"||a.charAt(a.length-1)!==">"||a.length<3?g=i.exec(a):g=[null,a,null];if(g&&(g[1]||!d)){if(g[1]){d=d instanceof e?d[0]:d,k=d?d.ownerDocument||d:c,j=m.exec(a),j?e.isPlainObject(d)?(a=[c.createElement(j[1])],e.fn.attr.call(a,d,!0)):a=[k.createElement(j[1])]:(j=e.buildFragment([g[1]],[k]),a=(j.cacheable?e.clone(j.fragment):j.fragment).childNodes);return e.merge(this,a)}h=c.getElementById(g[2]);if(h&&h.parentNode){if(h.id!==g[2])return f.find(a);this.length=1,this[0]=h}this.context=c,this.selector=a;return this}return!d||d.jquery?(d||f).find(a):this.constructor(d).find(a)}if(e.isFunction(a))return f.ready(a);a.selector!==b&&(this.selector=a.selector,this.context=a.context);return e.makeArray(a,this)},selector:"",jquery:"1.7.1",length:0,size:function(){return this.length},toArray:function(){return F.call(this,0)},get:function(a){return a==null?this.toArray():a<0?this[this.length+a]:this[a]},pushStack:function(a,b,c){var d=this.constructor();e.isArray(a)?E.apply(d,a):e.merge(d,a),d.prevObject=this,d.context=this.context,b==="find"?d.selector=this.selector+(this.selector?" ":"")+c:b&&(d.selector=this.selector+"."+b+"("+c+")");return d},each:function(a,b){return e.each(this,a,b)},ready:function(a){e.bindReady(),A.add(a);return this},eq:function(a){a=+a;return a===-1?this.slice(a):this.slice(a,a+1)},first:function(){return this.eq(0)},last:function(){return this.eq(-1)},slice:function(){return this.pushStack(F.apply(this,arguments),"slice",F.call(arguments).join(","))},map:function(a){return this.pushStack(e.map(this,function(b,c){return a.call(b,c,b)}))},end:function(){return this.prevObject||this.constructor(null)},push:E,sort:[].sort,splice:[].splice},e.fn.init.prototype=e.fn,e.extend=e.fn.extend=function(){var a,c,d,f,g,h,i=arguments[0]||{},j=1,k=arguments.length,l=!1;typeof i=="boolean"&&(l=i,i=arguments[1]||{},j=2),typeof i!="object"&&!e.isFunction(i)&&(i={}),k===j&&(i=this,--j);for(;j<k;j++)if((a=arguments[j])!=null)for(c in a){d=i[c],f=a[c];if(i===f)continue;l&&f&&(e.isPlainObject(f)||(g=e.isArray(f)))?(g?(g=!1,h=d&&e.isArray(d)?d:[]):h=d&&e.isPlainObject(d)?d:{},i[c]=e.extend(l,h,f)):f!==b&&(i[c]=f)}return i},e.extend({noConflict:function(b){a.$===e&&(a.$=g),b&&a.jQuery===e&&(a.jQuery=f);return e},isReady:!1,readyWait:1,holdReady:function(a){a?e.readyWait++:e.ready(!0)},ready:function(a){if(a===!0&&!--e.readyWait||a!==!0&&!e.isReady){if(!c.body)return setTimeout(e.ready,1);e.isReady=!0;if(a!==!0&&--e.readyWait>0)return;A.fireWith(c,[e]),e.fn.trigger&&e(c).trigger("ready").off("ready")}},bindReady:function(){if(!A){A=e.Callbacks("once memory");if(c.readyState==="complete")return setTimeout(e.ready,1);if(c.addEventListener)c.addEventListener("DOMContentLoaded",B,!1),a.addEventListener("load",e.ready,!1);else if(c.attachEvent){c.attachEvent("onreadystatechange",B),a.attachEvent("onload",e.ready);var b=!1;try{b=a.frameElement==null}catch(d){}c.documentElement.doScroll&&b&&J()}}},isFunction:function(a){return e.type(a)==="function"},isArray:Array.isArray||function(a){return e.type(a)==="array"},isWindow:function(a){return a&&typeof a=="object"&&"setInterval"in a},isNumeric:function(a){return!isNaN(parseFloat(a))&&isFinite(a)},type:function(a){return a==null?String(a):I[C.call(a)]||"object"},isPlainObject:function(a){if(!a||e.type(a)!=="object"||a.nodeType||e.isWindow(a))return!1;try{if(a.constructor&&!D.call(a,"constructor")&&!D.call(a.constructor.prototype,"isPrototypeOf"))return!1}catch(c){return!1}var d;for(d in a);return d===b||D.call(a,d)},isEmptyObject:function(a){for(var b in a)return!1;return!0},error:function(a){throw new Error(a)},parseJSON:function(b){if(typeof b!="string"||!b)return null;b=e.trim(b);if(a.JSON&&a.JSON.parse)return a.JSON.parse(b);if(n.test(b.replace(o,"@").replace(p,"]").replace(q,"")))return(new Function("return "+b))();e.error("Invalid JSON: "+b)},parseXML:function(c){var d,f;try{a.DOMParser?(f=new DOMParser,d=f.parseFromString(c,"text/xml")):(d=new ActiveXObject("Microsoft.XMLDOM"),d.async="false",d.loadXML(c))}catch(g){d=b}(!d||!d.documentElement||d.getElementsByTagName("parsererror").length)&&e.error("Invalid XML: "+c);return d},noop:function(){},globalEval:function(b){b&&j.test(b)&&(a.execScript||function(b){a.eval.call(a,b)})(b)},camelCase:function(a){return a.replace(w,"ms-").replace(v,x)},nodeName:function(a,b){return a.nodeName&&a.nodeName.toUpperCase()===b.toUpperCase()},each:function(a,c,d){var f,g=0,h=a.length,i=h===b||e.isFunction(a);if(d){if(i){for(f in a)if(c.apply(a[f],d)===!1)break}else for(;g<h;)if(c.apply(a[g++],d)===!1)break}else if(i){for(f in a)if(c.call(a[f],f,a[f])===!1)break}else for(;g<h;)if(c.call(a[g],g,a[g++])===!1)break;return a},trim:G?function(a){return a==null?"":G.call(a)}:function(a){return a==null?"":(a+"").replace(k,"").replace(l,"")},makeArray:function(a,b){var c=b||[];if(a!=null){var d=e.type(a);a.length==null||d==="string"||d==="function"||d==="regexp"||e.isWindow(a)?E.call(c,a):e.merge(c,a)}return c},inArray:function(a,b,c){var d;if(b){if(H)return H.call(b,a,c);d=b.length,c=c?c<0?Math.max(0,d+c):c:0;for(;c<d;c++)if(c in b&&b[c]===a)return c}return-1},merge:function(a,c){var d=a.length,e=0;if(typeof c.length=="number")for(var f=c.length;e<f;e++)a[d++]=c[e];else while(c[e]!==b)a[d++]=c[e++];a.length=d;return a},grep:function(a,b,c){var d=[],e;c=!!c;for(var f=0,g=a.length;f<g;f++)e=!!b(a[f],f),c!==e&&d.push(a[f]);return d},map:function(a,c,d){var f,g,h=[],i=0,j=a.length,k=a instanceof e||j!==b&&typeof j=="number"&&(j>0&&a[0]&&a[j-1]||j===0||e.isArray(a));if(k)for(;i<j;i++)f=c(a[i],i,d),f!=null&&(h[h.length]=f);else for(g in a)f=c(a[g],g,d),f!=null&&(h[h.length]=f);return h.concat.apply([],h)},guid:1,proxy:function(a,c){if(typeof c=="string"){var d=a[c];c=a,a=d}if(!e.isFunction(a))return b;var f=F.call(arguments,2),g=function(){return a.apply(c,f.concat(F.call(arguments)))};g.guid=a.guid=a.guid||g.guid||e.guid++;return g},access:function(a,c,d,f,g,h){var i=a.length;if(typeof c=="object"){for(var j in c)e.access(a,j,c[j],f,g,d);return a}if(d!==b){f=!h&&f&&e.isFunction(d);for(var k=0;k<i;k++)g(a[k],c,f?d.call(a[k],k,g(a[k],c)):d,h);return a}return i?g(a[0],c):b},now:function(){return(new Date).getTime()},uaMatch:function(a){a=a.toLowerCase();var b=r.exec(a)||s.exec(a)||t.exec(a)||a.indexOf("compatible")<0&&u.exec(a)||[];return{browser:b[1]||"",version:b[2]||"0"}},sub:function(){function a(b,c){return new a.fn.init(b,c)}e.extend(!0,a,this),a.superclass=this,a.fn=a.prototype=this(),a.fn.constructor=a,a.sub=this.sub,a.fn.init=function(d,f){f&&f instanceof e&&!(f instanceof a)&&(f=a(f));return e.fn.init.call(this,d,f,b)},a.fn.init.prototype=a.fn;var b=a(c);return a},browser:{}}),e.each("Boolean Number String Function Array Date RegExp Object".split(" "),function(a,b){I["[object "+b+"]"]=b.toLowerCase()}),z=e.uaMatch(y),z.browser&&(e.browser[z.browser]=!0,e.browser.version=z.version),e.browser.webkit&&(e.browser.safari=!0),j.test(" ")&&(k=/^[\s\xA0]+/,l=/[\s\xA0]+$/),h=e(c),c.addEventListener?B=function(){c.removeEventListener("DOMContentLoaded",B,!1),e.ready()}:c.attachEvent&&(B=function(){c.readyState==="complete"&&(c.detachEvent("onreadystatechange",B),e.ready())});return e}(),g={};f.Callbacks=function(a){a=a?g[a]||h(a):{};var c=[],d=[],e,i,j,k,l,m=function(b){var d,e,g,h,i;for(d=0,e=b.length;d<e;d++)g=b[d],h=f.type(g),h==="array"?m(g):h==="function"&&(!a.unique||!o.has(g))&&c.push(g)},n=function(b,f){f=f||[],e=!a.memory||[b,f],i=!0,l=j||0,j=0,k=c.length;for(;c&&l<k;l++)if(c[l].apply(b,f)===!1&&a.stopOnFalse){e=!0;break}i=!1,c&&(a.once?e===!0?o.disable():c=[]:d&&d.length&&(e=d.shift(),o.fireWith(e[0],e[1])))},o={add:function(){if(c){var a=c.length;m(arguments),i?k=c.length:e&&e!==!0&&(j=a,n(e[0],e[1]))}return this},remove:function(){if(c){var b=arguments,d=0,e=b.length;for(;d<e;d++)for(var f=0;f<c.length;f++)if(b[d]===c[f]){i&&f<=k&&(k--,f<=l&&l--),c.splice(f--,1);if(a.unique)break}}return this},has:function(a){if(c){var b=0,d=c.length;for(;b<d;b++)if(a===c[b])return!0}return!1},empty:function(){c=[];return this},disable:function(){c=d=e=b;return this},disabled:function(){return!c},lock:function(){d=b,(!e||e===!0)&&o.disable();return this},locked:function(){return!d},fireWith:function(b,c){d&&(i?a.once||d.push([b,c]):(!a.once||!e)&&n(b,c));return this},fire:function(){o.fireWith(this,arguments);return this},fired:function(){return!!e}};return o};var i=[].slice;f.extend({Deferred:function(a){var b=f.Callbacks("once memory"),c=f.Callbacks("once memory"),d=f.Callbacks("memory"),e="pending",g={resolve:b,reject:c,notify:d},h={done:b.add,fail:c.add,progress:d.add,state:function(){return e},isResolved:b.fired,isRejected:c.fired,then:function(a,b,c){i.done(a).fail(b).progress(c);return this},always:function(){i.done.apply(i,arguments).fail.apply(i,arguments);return this},pipe:function(a,b,c){return f.Deferred(function(d){f.each({done:[a,"resolve"],fail:[b,"reject"],progress:[c,"notify"]},function(a,b){var c=b[0],e=b[1],g;f.isFunction(c)?i[a](function(){g=c.apply(this,arguments),g&&f.isFunction(g.promise)?g.promise().then(d.resolve,d.reject,d.notify):d[e+"With"](this===i?d:this,[g])}):i[a](d[e])})}).promise()},promise:function(a){if(a==null)a=h;else for(var b in h)a[b]=h[b];return a}},i=h.promise({}),j;for(j in g)i[j]=g[j].fire,i[j+"With"]=g[j].fireWith;i.done(function(){e="resolved"},c.disable,d.lock).fail(function(){e="rejected"},b.disable,d.lock),a&&a.call(i,i);return i},when:function(a){function m(a){return function(b){e[a]=arguments.length>1?i.call(arguments,0):b,j.notifyWith(k,e)}}function l(a){return function(c){b[a]=arguments.length>1?i.call(arguments,0):c,--g||j.resolveWith(j,b)}}var b=i.call(arguments,0),c=0,d=b.length,e=Array(d),g=d,h=d,j=d<=1&&a&&f.isFunction(a.promise)?a:f.Deferred(),k=j.promise();if(d>1){for(;c<d;c++)b[c]&&b[c].promise&&f.isFunction(b[c].promise)?b[c].promise().then(l(c),j.reject,m(c)):--g;g||j.resolveWith(j,b)}else j!==a&&j.resolveWith(j,d?[a]:[]);return k}}),f.support=function(){var b,d,e,g,h,i,j,k,l,m,n,o,p,q=c.createElement("div"),r=c.documentElement;q.setAttribute("className","t"),q.innerHTML=" <link/><table></table><a href='/a' style='top:1px;float:left;opacity:.55;'>a</a><input type='checkbox'/>",d=q.getElementsByTagName("*"),e=q.getElementsByTagName("a")[0];if(!d||!d.length||!e)return{};g=c.createElement("select"),h=g.appendChild(c.createElement("option")),i=q.getElementsByTagName("input")[0],b={leadingWhitespace:q.firstChild.nodeType===3,tbody:!q.getElementsByTagName("tbody").length,htmlSerialize:!!q.getElementsByTagName("link").length,style:/top/.test(e.getAttribute("style")),hrefNormalized:e.getAttribute("href")==="/a",opacity:/^0.55/.test(e.style.opacity),cssFloat:!!e.style.cssFloat,checkOn:i.value==="on",optSelected:h.selected,getSetAttribute:q.className!=="t",enctype:!!c.createElement("form").enctype,html5Clone:c.createElement("nav").cloneNode(!0).outerHTML!=="<:nav></:nav>",submitBubbles:!0,changeBubbles:!0,focusinBubbles:!1,deleteExpando:!0,noCloneEvent:!0,inlineBlockNeedsLayout:!1,shrinkWrapBlocks:!1,reliableMarginRight:!0},i.checked=!0,b.noCloneChecked=i.cloneNode(!0).checked,g.disabled=!0,b.optDisabled=!h.disabled;try{delete q.test}catch(s){b.deleteExpando=!1}!q.addEventListener&&q.attachEvent&&q.fireEvent&&(q.attachEvent("onclick",function(){b.noCloneEvent=!1}),q.cloneNode(!0).fireEvent("onclick")),i=c.createElement("input"),i.value="t",i.setAttribute("type","radio"),b.radioValue=i.value==="t",i.setAttribute("checked","checked"),q.appendChild(i),k=c.createDocumentFragment(),k.appendChild(q.lastChild),b.checkClone=k.cloneNode(!0).cloneNode(!0).lastChild.checked,b.appendChecked=i.checked,k.removeChild(i),k.appendChild(q),q.innerHTML="",a.getComputedStyle&&(j=c.createElement("div"),j.style.width="0",j.style.marginRight="0",q.style.width="2px",q.appendChild(j),b.reliableMarginRight=(parseInt((a.getComputedStyle(j,null)||{marginRight:0}).marginRight,10)||0)===0);if(q.attachEvent)for(o in{submit:1,change:1,focusin:1})n="on"+o,p=n in q,p||(q.setAttribute(n,"return;"),p=typeof q[n]=="function"),b[o+"Bubbles"]=p;k.removeChild(q),k=g=h=j=q=i=null,f(function(){var a,d,e,g,h,i,j,k,m,n,o,r=c.getElementsByTagName("body")[0];!r||(j=1,k="position:absolute;top:0;left:0;width:1px;height:1px;margin:0;",m="visibility:hidden;border:0;",n="style='"+k+"border:5px solid #000;padding:0;'",o="<div "+n+"><div></div></div>"+"<table "+n+" cellpadding='0' cellspacing='0'>"+"<tr><td></td></tr></table>",a=c.createElement("div"),a.style.cssText=m+"width:0;height:0;position:static;top:0;margin-top:"+j+"px",r.insertBefore(a,r.firstChild),q=c.createElement("div"),a.appendChild(q),q.innerHTML="<table><tr><td style='padding:0;border:0;display:none'></td><td>t</td></tr></table>",l=q.getElementsByTagName("td"),p=l[0].offsetHeight===0,l[0].style.display="",l[1].style.display="none",b.reliableHiddenOffsets=p&&l[0].offsetHeight===0,q.innerHTML="",q.style.width=q.style.paddingLeft="1px",f.boxModel=b.boxModel=q.offsetWidth===2,typeof q.style.zoom!="undefined"&&(q.style.display="inline",q.style.zoom=1,b.inlineBlockNeedsLayout=q.offsetWidth===2,q.style.display="",q.innerHTML="<div style='width:4px;'></div>",b.shrinkWrapBlocks=q.offsetWidth!==2),q.style.cssText=k+m,q.innerHTML=o,d=q.firstChild,e=d.firstChild,h=d.nextSibling.firstChild.firstChild,i={doesNotAddBorder:e.offsetTop!==5,doesAddBorderForTableAndCells:h.offsetTop===5},e.style.position="fixed",e.style.top="20px",i.fixedPosition=e.offsetTop===20||e.offsetTop===15,e.style.position=e.style.top="",d.style.overflow="hidden",d.style.position="relative",i.subtractsBorderForOverflowNotVisible=e.offsetTop===-5,i.doesNotIncludeMarginInBodyOffset=r.offsetTop!==j,r.removeChild(a),q=a=null,f.extend(b,i))});return b}();var j=/^(?:\{.*\}|\[.*\])$/,k=/([A-Z])/g;f.extend({cache:{},uuid:0,expando:"jQuery"+(f.fn.jquery+Math.random()).replace(/\D/g,""),noData:{embed:!0,object:"clsid:D27CDB6E-AE6D-11cf-96B8-444553540000",applet:!0},hasData:function(a){a=a.nodeType?f.cache[a[f.expando]]:a[f.expando];return!!a&&!m(a)},data:function(a,c,d,e){if(!!f.acceptData(a)){var g,h,i,j=f.expando,k=typeof c=="string",l=a.nodeType,m=l?f.cache:a,n=l?a[j]:a[j]&&j,o=c==="events";if((!n||!m[n]||!o&&!e&&!m[n].data)&&k&&d===b)return;n||(l?a[j]=n=++f.uuid:n=j),m[n]||(m[n]={},l||(m[n].toJSON=f.noop));if(typeof c=="object"||typeof c=="function")e?m[n]=f.extend(m[n],c):m[n].data=f.extend(m[n].data,c);g=h=m[n],e||(h.data||(h.data={}),h=h.data),d!==b&&(h[f.camelCase(c)]=d);if(o&&!h[c])return g.events;k?(i=h[c],i==null&&(i=h[f.camelCase(c)])):i=h;return i}},removeData:function(a,b,c){if(!!f.acceptData(a)){var d,e,g,h=f.expando,i=a.nodeType,j=i?f.cache:a,k=i?a[h]:h;if(!j[k])return;if(b){d=c?j[k]:j[k].data;if(d){f.isArray(b)||(b in d?b=[b]:(b=f.camelCase(b),b in d?b=[b]:b=b.split(" ")));for(e=0,g=b.length;e<g;e++)delete d[b[e]];if(!(c?m:f.isEmptyObject)(d))return}}if(!c){delete j[k].data;if(!m(j[k]))return}f.support.deleteExpando||!j.setInterval?delete j[k]:j[k]=null,i&&(f.support.deleteExpando?delete a[h]:a.removeAttribute?a.removeAttribute(h):a[h]=null)}},_data:function(a,b,c){return f.data(a,b,c,!0)},acceptData:function(a){if(a.nodeName){var b=f.noData[a.nodeName.toLowerCase()];if(b)return b!==!0&&a.getAttribute("classid")===b}return!0}}),f.fn.extend({data:function(a,c){var d,e,g,h=null;if(typeof a=="undefined"){if(this.length){h=f.data(this[0]);if(this[0].nodeType===1&&!f._data(this[0],"parsedAttrs")){e=this[0].attributes;for(var i=0,j=e.length;i<j;i++)g=e[i].name,g.indexOf("data-")===0&&(g=f.camelCase(g.substring(5)),l(this[0],g,h[g]));f._data(this[0],"parsedAttrs",!0)}}return h}if(typeof a=="object")return this.each(function(){f.data(this,a)});d=a.split("."),d[1]=d[1]?"."+d[1]:"";if(c===b){h=this.triggerHandler("getData"+d[1]+"!",[d[0]]),h===b&&this.length&&(h=f.data(this[0],a),h=l(this[0],a,h));return h===b&&d[1]?this.data(d[0]):h}return this.each(function(){var b=f(this),e=[d[0],c];b.triggerHandler("setData"+d[1]+"!",e),f.data(this,a,c),b.triggerHandler("changeData"+d[1]+"!",e)})},removeData:function(a){return this.each(function(){f.removeData(this,a)})}}),f.extend({_mark:function(a,b){a&&(b=(b||"fx")+"mark",f._data(a,b,(f._data(a,b)||0)+1))},_unmark:function(a,b,c){a!==!0&&(c=b,b=a,a=!1);if(b){c=c||"fx";var d=c+"mark",e=a?0:(f._data(b,d)||1)-1;e?f._data(b,d,e):(f.removeData(b,d,!0),n(b,c,"mark"))}},queue:function(a,b,c){var d;if(a){b=(b||"fx")+"queue",d=f._data(a,b),c&&(!d||f.isArray(c)?d=f._data(a,b,f.makeArray(c)):d.push(c));return d||[]}},dequeue:function(a,b){b=b||"fx";var c=f.queue(a,b),d=c.shift(),e={};d==="inprogress"&&(d=c.shift()),d&&(b==="fx"&&c.unshift("inprogress"),f._data(a,b+".run",e),d.call(a,function(){f.dequeue(a,b)},e)),c.length||(f.removeData(a,b+"queue "+b+".run",!0),n(a,b,"queue"))}}),f.fn.extend({queue:function(a,c){typeof a!="string"&&(c=a,a="fx");if(c===b)return f.queue(this[0],a);return this.each(function(){var b=f.queue(this,a,c);a==="fx"&&b[0]!=="inprogress"&&f.dequeue(this,a)})},dequeue:function(a){return this.each(function(){f.dequeue(this,a)})},delay:function(a,b){a=f.fx?f.fx.speeds[a]||a:a,b=b||"fx";return this.queue(b,function(b,c){var d=setTimeout(b,a);c.stop=function(){clearTimeout(d)}})},clearQueue:function(a){return this.queue(a||"fx",[])},promise:function(a,c){function m(){--h||d.resolveWith(e,[e])}typeof a!="string"&&(c=a,a=b),a=a||"fx";var d=f.Deferred(),e=this,g=e.length,h=1,i=a+"defer",j=a+"queue",k=a+"mark",l;while(g--)if(l=f.data(e[g],i,b,!0)||(f.data(e[g],j,b,!0)||f.data(e[g],k,b,!0))&&f.data(e[g],i,f.Callbacks("once memory"),!0))h++,l.add(m);m();return d.promise()}});var o=/[\n\t\r]/g,p=/\s+/,q=/\r/g,r=/^(?:button|input)$/i,s=/^(?:button|input|object|select|textarea)$/i,t=/^a(?:rea)?$/i,u=/^(?:autofocus|autoplay|async|checked|controls|defer|disabled|hidden|loop|multiple|open|readonly|required|scoped|selected)$/i,v=f.support.getSetAttribute,w,x,y;f.fn.extend({attr:function(a,b){return f.access(this,a,b,!0,f.attr)},removeAttr:function(a){return this.each(function(){f.removeAttr(this,a)})},prop:function(a,b){return f.access(this,a,b,!0,f.prop)},removeProp:function(a){a=f.propFix[a]||a;return this.each(function(){try{this[a]=b,delete this[a]}catch(c){}})},addClass:function(a){var b,c,d,e,g,h,i;if(f.isFunction(a))return this.each(function(b){f(this).addClass(a.call(this,b,this.className))});if(a&&typeof a=="string"){b=a.split(p);for(c=0,d=this.length;c<d;c++){e=this[c];if(e.nodeType===1)if(!e.className&&b.length===1)e.className=a;else{g=" "+e.className+" ";for(h=0,i=b.length;h<i;h++)~g.indexOf(" "+b[h]+" ")||(g+=b[h]+" ");e.className=f.trim(g)}}}return this},removeClass:function(a){var c,d,e,g,h,i,j;if(f.isFunction(a))return this.each(function(b){f(this).removeClass(a.call(this,b,this.className))});if(a&&typeof a=="string"||a===b){c=(a||"").split(p);for(d=0,e=this.length;d<e;d++){g=this[d];if(g.nodeType===1&&g.className)if(a){h=(" "+g.className+" ").replace(o," ");for(i=0,j=c.length;i<j;i++)h=h.replace(" "+c[i]+" "," ");g.className=f.trim(h)}else g.className=""}}return this},toggleClass:function(a,b){var c=typeof a,d=typeof b=="boolean";if(f.isFunction(a))return this.each(function(c){f(this).toggleClass(a.call(this,c,this.className,b),b)});return this.each(function(){if(c==="string"){var e,g=0,h=f(this),i=b,j=a.split(p);while(e=j[g++])i=d?i:!h.hasClass(e),h[i?"addClass":"removeClass"](e)}else if(c==="undefined"||c==="boolean")this.className&&f._data(this,"__className__",this.className),this.className=this.className||a===!1?"":f._data(this,"__className__")||""})},hasClass:function(a){var b=" "+a+" ",c=0,d=this.length;for(;c<d;c++)if(this[c].nodeType===1&&(" "+this[c].className+" ").replace(o," ").indexOf(b)>-1)return!0;return!1},val:function(a){var c,d,e,g=this[0];{if(!!arguments.length){e=f.isFunction(a);return this.each(function(d){var g=f(this),h;if(this.nodeType===1){e?h=a.call(this,d,g.val()):h=a,h==null?h="":typeof h=="number"?h+="":f.isArray(h)&&(h=f.map(h,function(a){return a==null?"":a+""})),c=f.valHooks[this.nodeName.toLowerCase()]||f.valHooks[this.type];if(!c||!("set"in c)||c.set(this,h,"value")===b)this.value=h}})}if(g){c=f.valHooks[g.nodeName.toLowerCase()]||f.valHooks[g.type];if(c&&"get"in c&&(d=c.get(g,"value"))!==b)return d;d=g.value;return typeof d=="string"?d.replace(q,""):d==null?"":d}}}}),f.extend({valHooks:{option:{get:function(a){var b=a.attributes.value;return!b||b.specified?a.value:a.text}},select:{get:function(a){var b,c,d,e,g=a.selectedIndex,h=[],i=a.options,j=a.type==="select-one";if(g<0)return null;c=j?g:0,d=j?g+1:i.length;for(;c<d;c++){e=i[c];if(e.selected&&(f.support.optDisabled?!e.disabled:e.getAttribute("disabled")===null)&&(!e.parentNode.disabled||!f.nodeName(e.parentNode,"optgroup"))){b=f(e).val();if(j)return b;h.push(b)}}if(j&&!h.length&&i.length)return f(i[g]).val();return h},set:function(a,b){var c=f.makeArray(b);f(a).find("option").each(function(){this.selected=f.inArray(f(this).val(),c)>=0}),c.length||(a.selectedIndex=-1);return c}}},attrFn:{val:!0,css:!0,html:!0,text:!0,data:!0,width:!0,height:!0,offset:!0},attr:function(a,c,d,e){var g,h,i,j=a.nodeType;if(!!a&&j!==3&&j!==8&&j!==2){if(e&&c in f.attrFn)return f(a)[c](d);if(typeof a.getAttribute=="undefined")return f.prop(a,c,d);i=j!==1||!f.isXMLDoc(a),i&&(c=c.toLowerCase(),h=f.attrHooks[c]||(u.test(c)?x:w));if(d!==b){if(d===null){f.removeAttr(a,c);return}if(h&&"set"in h&&i&&(g=h.set(a,d,c))!==b)return g;a.setAttribute(c,""+d);return d}if(h&&"get"in h&&i&&(g=h.get(a,c))!==null)return g;g=a.getAttribute(c);return g===null?b:g}},removeAttr:function(a,b){var c,d,e,g,h=0;if(b&&a.nodeType===1){d=b.toLowerCase().split(p),g=d.length;for(;h<g;h++)e=d[h],e&&(c=f.propFix[e]||e,f.attr(a,e,""),a.removeAttribute(v?e:c),u.test(e)&&c in a&&(a[c]=!1))}},attrHooks:{type:{set:function(a,b){if(r.test(a.nodeName)&&a.parentNode)f.error("type property can't be changed");else if(!f.support.radioValue&&b==="radio"&&f.nodeName(a,"input")){var c=a.value;a.setAttribute("type",b),c&&(a.value=c);return b}}},value:{get:function(a,b){if(w&&f.nodeName(a,"button"))return w.get(a,b);return b in a?a.value:null},set:function(a,b,c){if(w&&f.nodeName(a,"button"))return w.set(a,b,c);a.value=b}}},propFix:{tabindex:"tabIndex",readonly:"readOnly","for":"htmlFor","class":"className",maxlength:"maxLength",cellspacing:"cellSpacing",cellpadding:"cellPadding",rowspan:"rowSpan",colspan:"colSpan",usemap:"useMap",frameborder:"frameBorder",contenteditable:"contentEditable"},prop:function(a,c,d){var e,g,h,i=a.nodeType;if(!!a&&i!==3&&i!==8&&i!==2){h=i!==1||!f.isXMLDoc(a),h&&(c=f.propFix[c]||c,g=f.propHooks[c]);return d!==b?g&&"set"in g&&(e=g.set(a,d,c))!==b?e:a[c]=d:g&&"get"in g&&(e=g.get(a,c))!==null?e:a[c]}},propHooks:{tabIndex:{get:function(a){var c=a.getAttributeNode("tabindex");return c&&c.specified?parseInt(c.value,10):s.test(a.nodeName)||t.test(a.nodeName)&&a.href?0:b}}}}),f.attrHooks.tabindex=f.propHooks.tabIndex,x={get:function(a,c){var d,e=f.prop(a,c);return e===!0||typeof e!="boolean"&&(d=a.getAttributeNode(c))&&d.nodeValue!==!1?c.toLowerCase():b},set:function(a,b,c){var d;b===!1?f.removeAttr(a,c):(d=f.propFix[c]||c,d in a&&(a[d]=!0),a.setAttribute(c,c.toLowerCase()));return c}},v||(y={name:!0,id:!0},w=f.valHooks.button={get:function(a,c){var d;d=a.getAttributeNode(c);return d&&(y[c]?d.nodeValue!=="":d.specified)?d.nodeValue:b},set:function(a,b,d){var e=a.getAttributeNode(d);e||(e=c.createAttribute(d),a.setAttributeNode(e));return e.nodeValue=b+""}},f.attrHooks.tabindex.set=w.set,f.each(["width","height"],function(a,b){f.attrHooks[b]=f.extend(f.attrHooks[b],{set:function(a,c){if(c===""){a.setAttribute(b,"auto");return c}}})}),f.attrHooks.contenteditable={get:w.get,set:function(a,b,c){b===""&&(b="false"),w.set(a,b,c)}}),f.support.hrefNormalized||f.each(["href","src","width","height"],function(a,c){f.attrHooks[c]=f.extend(f.attrHooks[c],{get:function(a){var d=a.getAttribute(c,2);return d===null?b:d}})}),f.support.style||(f.attrHooks.style={get:function(a){return a.style.cssText.toLowerCase()||b},set:function(a,b){return a.style.cssText=""+b}}),f.support.optSelected||(f.propHooks.selected=f.extend(f.propHooks.selected,{get:function(a){var b=a.parentNode;b&&(b.selectedIndex,b.parentNode&&b.parentNode.selectedIndex);return null}})),f.support.enctype||(f.propFix.enctype="encoding"),f.support.checkOn||f.each(["radio","checkbox"],function(){f.valHooks[this]={get:function(a){return a.getAttribute("value")===null?"on":a.value}}}),f.each(["radio","checkbox"],function(){f.valHooks[this]=f.extend(f.valHooks[this],{set:function(a,b){if(f.isArray(b))return a.checked=f.inArray(f(a).val(),b)>=0}})});var z=/^(?:textarea|input|select)$/i,A=/^([^\.]*)?(?:\.(.+))?$/,B=/\bhover(\.\S+)?\b/,C=/^key/,D=/^(?:mouse|contextmenu)|click/,E=/^(?:focusinfocus|focusoutblur)$/,F=/^(\w*)(?:#([\w\-]+))?(?:\.([\w\-]+))?$/,G=function(a){var b=F.exec(a);b&&(b[1]=(b[1]||"").toLowerCase(),b[3]=b[3]&&new RegExp("(?:^|\\s)"+b[3]+"(?:\\s|$)"));return b},H=function(a,b){var c=a.attributes||{};return(!b[1]||a.nodeName.toLowerCase()===b[1])&&(!b[2]||(c.id||{}).value===b[2])&&(!b[3]||b[3].test((c["class"]||{}).value))},I=function(a){return f.event.special.hover?a:a.replace(B,"mouseenter$1 mouseleave$1")}; - f.event={add:function(a,c,d,e,g){var h,i,j,k,l,m,n,o,p,q,r,s;if(!(a.nodeType===3||a.nodeType===8||!c||!d||!(h=f._data(a)))){d.handler&&(p=d,d=p.handler),d.guid||(d.guid=f.guid++),j=h.events,j||(h.events=j={}),i=h.handle,i||(h.handle=i=function(a){return typeof f!="undefined"&&(!a||f.event.triggered!==a.type)?f.event.dispatch.apply(i.elem,arguments):b},i.elem=a),c=f.trim(I(c)).split(" ");for(k=0;k<c.length;k++){l=A.exec(c[k])||[],m=l[1],n=(l[2]||"").split(".").sort(),s=f.event.special[m]||{},m=(g?s.delegateType:s.bindType)||m,s=f.event.special[m]||{},o=f.extend({type:m,origType:l[1],data:e,handler:d,guid:d.guid,selector:g,quick:G(g),namespace:n.join(".")},p),r=j[m];if(!r){r=j[m]=[],r.delegateCount=0;if(!s.setup||s.setup.call(a,e,n,i)===!1)a.addEventListener?a.addEventListener(m,i,!1):a.attachEvent&&a.attachEvent("on"+m,i)}s.add&&(s.add.call(a,o),o.handler.guid||(o.handler.guid=d.guid)),g?r.splice(r.delegateCount++,0,o):r.push(o),f.event.global[m]=!0}a=null}},global:{},remove:function(a,b,c,d,e){var g=f.hasData(a)&&f._data(a),h,i,j,k,l,m,n,o,p,q,r,s;if(!!g&&!!(o=g.events)){b=f.trim(I(b||"")).split(" ");for(h=0;h<b.length;h++){i=A.exec(b[h])||[],j=k=i[1],l=i[2];if(!j){for(j in o)f.event.remove(a,j+b[h],c,d,!0);continue}p=f.event.special[j]||{},j=(d?p.delegateType:p.bindType)||j,r=o[j]||[],m=r.length,l=l?new RegExp("(^|\\.)"+l.split(".").sort().join("\\.(?:.*\\.)?")+"(\\.|$)"):null;for(n=0;n<r.length;n++)s=r[n],(e||k===s.origType)&&(!c||c.guid===s.guid)&&(!l||l.test(s.namespace))&&(!d||d===s.selector||d==="**"&&s.selector)&&(r.splice(n--,1),s.selector&&r.delegateCount--,p.remove&&p.remove.call(a,s));r.length===0&&m!==r.length&&((!p.teardown||p.teardown.call(a,l)===!1)&&f.removeEvent(a,j,g.handle),delete o[j])}f.isEmptyObject(o)&&(q=g.handle,q&&(q.elem=null),f.removeData(a,["events","handle"],!0))}},customEvent:{getData:!0,setData:!0,changeData:!0},trigger:function(c,d,e,g){if(!e||e.nodeType!==3&&e.nodeType!==8){var h=c.type||c,i=[],j,k,l,m,n,o,p,q,r,s;if(E.test(h+f.event.triggered))return;h.indexOf("!")>=0&&(h=h.slice(0,-1),k=!0),h.indexOf(".")>=0&&(i=h.split("."),h=i.shift(),i.sort());if((!e||f.event.customEvent[h])&&!f.event.global[h])return;c=typeof c=="object"?c[f.expando]?c:new f.Event(h,c):new f.Event(h),c.type=h,c.isTrigger=!0,c.exclusive=k,c.namespace=i.join("."),c.namespace_re=c.namespace?new RegExp("(^|\\.)"+i.join("\\.(?:.*\\.)?")+"(\\.|$)"):null,o=h.indexOf(":")<0?"on"+h:"";if(!e){j=f.cache;for(l in j)j[l].events&&j[l].events[h]&&f.event.trigger(c,d,j[l].handle.elem,!0);return}c.result=b,c.target||(c.target=e),d=d!=null?f.makeArray(d):[],d.unshift(c),p=f.event.special[h]||{};if(p.trigger&&p.trigger.apply(e,d)===!1)return;r=[[e,p.bindType||h]];if(!g&&!p.noBubble&&!f.isWindow(e)){s=p.delegateType||h,m=E.test(s+h)?e:e.parentNode,n=null;for(;m;m=m.parentNode)r.push([m,s]),n=m;n&&n===e.ownerDocument&&r.push([n.defaultView||n.parentWindow||a,s])}for(l=0;l<r.length&&!c.isPropagationStopped();l++)m=r[l][0],c.type=r[l][1],q=(f._data(m,"events")||{})[c.type]&&f._data(m,"handle"),q&&q.apply(m,d),q=o&&m[o],q&&f.acceptData(m)&&q.apply(m,d)===!1&&c.preventDefault();c.type=h,!g&&!c.isDefaultPrevented()&&(!p._default||p._default.apply(e.ownerDocument,d)===!1)&&(h!=="click"||!f.nodeName(e,"a"))&&f.acceptData(e)&&o&&e[h]&&(h!=="focus"&&h!=="blur"||c.target.offsetWidth!==0)&&!f.isWindow(e)&&(n=e[o],n&&(e[o]=null),f.event.triggered=h,e[h](),f.event.triggered=b,n&&(e[o]=n));return c.result}},dispatch:function(c){c=f.event.fix(c||a.event);var d=(f._data(this,"events")||{})[c.type]||[],e=d.delegateCount,g=[].slice.call(arguments,0),h=!c.exclusive&&!c.namespace,i=[],j,k,l,m,n,o,p,q,r,s,t;g[0]=c,c.delegateTarget=this;if(e&&!c.target.disabled&&(!c.button||c.type!=="click")){m=f(this),m.context=this.ownerDocument||this;for(l=c.target;l!=this;l=l.parentNode||this){o={},q=[],m[0]=l;for(j=0;j<e;j++)r=d[j],s=r.selector,o[s]===b&&(o[s]=r.quick?H(l,r.quick):m.is(s)),o[s]&&q.push(r);q.length&&i.push({elem:l,matches:q})}}d.length>e&&i.push({elem:this,matches:d.slice(e)});for(j=0;j<i.length&&!c.isPropagationStopped();j++){p=i[j],c.currentTarget=p.elem;for(k=0;k<p.matches.length&&!c.isImmediatePropagationStopped();k++){r=p.matches[k];if(h||!c.namespace&&!r.namespace||c.namespace_re&&c.namespace_re.test(r.namespace))c.data=r.data,c.handleObj=r,n=((f.event.special[r.origType]||{}).handle||r.handler).apply(p.elem,g),n!==b&&(c.result=n,n===!1&&(c.preventDefault(),c.stopPropagation()))}}return c.result},props:"attrChange attrName relatedNode srcElement altKey bubbles cancelable ctrlKey currentTarget eventPhase metaKey relatedTarget shiftKey target timeStamp view which".split(" "),fixHooks:{},keyHooks:{props:"char charCode key keyCode".split(" "),filter:function(a,b){a.which==null&&(a.which=b.charCode!=null?b.charCode:b.keyCode);return a}},mouseHooks:{props:"button buttons clientX clientY fromElement offsetX offsetY pageX pageY screenX screenY toElement".split(" "),filter:function(a,d){var e,f,g,h=d.button,i=d.fromElement;a.pageX==null&&d.clientX!=null&&(e=a.target.ownerDocument||c,f=e.documentElement,g=e.body,a.pageX=d.clientX+(f&&f.scrollLeft||g&&g.scrollLeft||0)-(f&&f.clientLeft||g&&g.clientLeft||0),a.pageY=d.clientY+(f&&f.scrollTop||g&&g.scrollTop||0)-(f&&f.clientTop||g&&g.clientTop||0)),!a.relatedTarget&&i&&(a.relatedTarget=i===a.target?d.toElement:i),!a.which&&h!==b&&(a.which=h&1?1:h&2?3:h&4?2:0);return a}},fix:function(a){if(a[f.expando])return a;var d,e,g=a,h=f.event.fixHooks[a.type]||{},i=h.props?this.props.concat(h.props):this.props;a=f.Event(g);for(d=i.length;d;)e=i[--d],a[e]=g[e];a.target||(a.target=g.srcElement||c),a.target.nodeType===3&&(a.target=a.target.parentNode),a.metaKey===b&&(a.metaKey=a.ctrlKey);return h.filter?h.filter(a,g):a},special:{ready:{setup:f.bindReady},load:{noBubble:!0},focus:{delegateType:"focusin"},blur:{delegateType:"focusout"},beforeunload:{setup:function(a,b,c){f.isWindow(this)&&(this.onbeforeunload=c)},teardown:function(a,b){this.onbeforeunload===b&&(this.onbeforeunload=null)}}},simulate:function(a,b,c,d){var e=f.extend(new f.Event,c,{type:a,isSimulated:!0,originalEvent:{}});d?f.event.trigger(e,null,b):f.event.dispatch.call(b,e),e.isDefaultPrevented()&&c.preventDefault()}},f.event.handle=f.event.dispatch,f.removeEvent=c.removeEventListener?function(a,b,c){a.removeEventListener&&a.removeEventListener(b,c,!1)}:function(a,b,c){a.detachEvent&&a.detachEvent("on"+b,c)},f.Event=function(a,b){if(!(this instanceof f.Event))return new f.Event(a,b);a&&a.type?(this.originalEvent=a,this.type=a.type,this.isDefaultPrevented=a.defaultPrevented||a.returnValue===!1||a.getPreventDefault&&a.getPreventDefault()?K:J):this.type=a,b&&f.extend(this,b),this.timeStamp=a&&a.timeStamp||f.now(),this[f.expando]=!0},f.Event.prototype={preventDefault:function(){this.isDefaultPrevented=K;var a=this.originalEvent;!a||(a.preventDefault?a.preventDefault():a.returnValue=!1)},stopPropagation:function(){this.isPropagationStopped=K;var a=this.originalEvent;!a||(a.stopPropagation&&a.stopPropagation(),a.cancelBubble=!0)},stopImmediatePropagation:function(){this.isImmediatePropagationStopped=K,this.stopPropagation()},isDefaultPrevented:J,isPropagationStopped:J,isImmediatePropagationStopped:J},f.each({mouseenter:"mouseover",mouseleave:"mouseout"},function(a,b){f.event.special[a]={delegateType:b,bindType:b,handle:function(a){var c=this,d=a.relatedTarget,e=a.handleObj,g=e.selector,h;if(!d||d!==c&&!f.contains(c,d))a.type=e.origType,h=e.handler.apply(this,arguments),a.type=b;return h}}}),f.support.submitBubbles||(f.event.special.submit={setup:function(){if(f.nodeName(this,"form"))return!1;f.event.add(this,"click._submit keypress._submit",function(a){var c=a.target,d=f.nodeName(c,"input")||f.nodeName(c,"button")?c.form:b;d&&!d._submit_attached&&(f.event.add(d,"submit._submit",function(a){this.parentNode&&!a.isTrigger&&f.event.simulate("submit",this.parentNode,a,!0)}),d._submit_attached=!0)})},teardown:function(){if(f.nodeName(this,"form"))return!1;f.event.remove(this,"._submit")}}),f.support.changeBubbles||(f.event.special.change={setup:function(){if(z.test(this.nodeName)){if(this.type==="checkbox"||this.type==="radio")f.event.add(this,"propertychange._change",function(a){a.originalEvent.propertyName==="checked"&&(this._just_changed=!0)}),f.event.add(this,"click._change",function(a){this._just_changed&&!a.isTrigger&&(this._just_changed=!1,f.event.simulate("change",this,a,!0))});return!1}f.event.add(this,"beforeactivate._change",function(a){var b=a.target;z.test(b.nodeName)&&!b._change_attached&&(f.event.add(b,"change._change",function(a){this.parentNode&&!a.isSimulated&&!a.isTrigger&&f.event.simulate("change",this.parentNode,a,!0)}),b._change_attached=!0)})},handle:function(a){var b=a.target;if(this!==b||a.isSimulated||a.isTrigger||b.type!=="radio"&&b.type!=="checkbox")return a.handleObj.handler.apply(this,arguments)},teardown:function(){f.event.remove(this,"._change");return z.test(this.nodeName)}}),f.support.focusinBubbles||f.each({focus:"focusin",blur:"focusout"},function(a,b){var d=0,e=function(a){f.event.simulate(b,a.target,f.event.fix(a),!0)};f.event.special[b]={setup:function(){d++===0&&c.addEventListener(a,e,!0)},teardown:function(){--d===0&&c.removeEventListener(a,e,!0)}}}),f.fn.extend({on:function(a,c,d,e,g){var h,i;if(typeof a=="object"){typeof c!="string"&&(d=c,c=b);for(i in a)this.on(i,c,d,a[i],g);return this}d==null&&e==null?(e=c,d=c=b):e==null&&(typeof c=="string"?(e=d,d=b):(e=d,d=c,c=b));if(e===!1)e=J;else if(!e)return this;g===1&&(h=e,e=function(a){f().off(a);return h.apply(this,arguments)},e.guid=h.guid||(h.guid=f.guid++));return this.each(function(){f.event.add(this,a,e,d,c)})},one:function(a,b,c,d){return this.on.call(this,a,b,c,d,1)},off:function(a,c,d){if(a&&a.preventDefault&&a.handleObj){var e=a.handleObj;f(a.delegateTarget).off(e.namespace?e.type+"."+e.namespace:e.type,e.selector,e.handler);return this}if(typeof a=="object"){for(var g in a)this.off(g,c,a[g]);return this}if(c===!1||typeof c=="function")d=c,c=b;d===!1&&(d=J);return this.each(function(){f.event.remove(this,a,d,c)})},bind:function(a,b,c){return this.on(a,null,b,c)},unbind:function(a,b){return this.off(a,null,b)},live:function(a,b,c){f(this.context).on(a,this.selector,b,c);return this},die:function(a,b){f(this.context).off(a,this.selector||"**",b);return this},delegate:function(a,b,c,d){return this.on(b,a,c,d)},undelegate:function(a,b,c){return arguments.length==1?this.off(a,"**"):this.off(b,a,c)},trigger:function(a,b){return this.each(function(){f.event.trigger(a,b,this)})},triggerHandler:function(a,b){if(this[0])return f.event.trigger(a,b,this[0],!0)},toggle:function(a){var b=arguments,c=a.guid||f.guid++,d=0,e=function(c){var e=(f._data(this,"lastToggle"+a.guid)||0)%d;f._data(this,"lastToggle"+a.guid,e+1),c.preventDefault();return b[e].apply(this,arguments)||!1};e.guid=c;while(d<b.length)b[d++].guid=c;return this.click(e)},hover:function(a,b){return this.mouseenter(a).mouseleave(b||a)}}),f.each("blur focus focusin focusout load resize scroll unload click dblclick mousedown mouseup mousemove mouseover mouseout mouseenter mouseleave change select submit keydown keypress keyup error contextmenu".split(" "),function(a,b){f.fn[b]=function(a,c){c==null&&(c=a,a=null);return arguments.length>0?this.on(b,null,a,c):this.trigger(b)},f.attrFn&&(f.attrFn[b]=!0),C.test(b)&&(f.event.fixHooks[b]=f.event.keyHooks),D.test(b)&&(f.event.fixHooks[b]=f.event.mouseHooks)}),function(){function x(a,b,c,e,f,g){for(var h=0,i=e.length;h<i;h++){var j=e[h];if(j){var k=!1;j=j[a];while(j){if(j[d]===c){k=e[j.sizset];break}if(j.nodeType===1){g||(j[d]=c,j.sizset=h);if(typeof b!="string"){if(j===b){k=!0;break}}else if(m.filter(b,[j]).length>0){k=j;break}}j=j[a]}e[h]=k}}}function w(a,b,c,e,f,g){for(var h=0,i=e.length;h<i;h++){var j=e[h];if(j){var k=!1;j=j[a];while(j){if(j[d]===c){k=e[j.sizset];break}j.nodeType===1&&!g&&(j[d]=c,j.sizset=h);if(j.nodeName.toLowerCase()===b){k=j;break}j=j[a]}e[h]=k}}}var a=/((?:\((?:\([^()]+\)|[^()]+)+\)|\[(?:\[[^\[\]]*\]|['"][^'"]*['"]|[^\[\]'"]+)+\]|\\.|[^ >+~,(\[\\]+)+|[>+~])(\s*,\s*)?((?:.|\r|\n)*)/g,d="sizcache"+(Math.random()+"").replace(".",""),e=0,g=Object.prototype.toString,h=!1,i=!0,j=/\\/g,k=/\r\n/g,l=/\W/;[0,0].sort(function(){i=!1;return 0});var m=function(b,d,e,f){e=e||[],d=d||c;var h=d;if(d.nodeType!==1&&d.nodeType!==9)return[];if(!b||typeof b!="string")return e;var i,j,k,l,n,q,r,t,u=!0,v=m.isXML(d),w=[],x=b;do{a.exec(""),i=a.exec(x);if(i){x=i[3],w.push(i[1]);if(i[2]){l=i[3];break}}}while(i);if(w.length>1&&p.exec(b))if(w.length===2&&o.relative[w[0]])j=y(w[0]+w[1],d,f);else{j=o.relative[w[0]]?[d]:m(w.shift(),d);while(w.length)b=w.shift(),o.relative[b]&&(b+=w.shift()),j=y(b,j,f)}else{!f&&w.length>1&&d.nodeType===9&&!v&&o.match.ID.test(w[0])&&!o.match.ID.test(w[w.length-1])&&(n=m.find(w.shift(),d,v),d=n.expr?m.filter(n.expr,n.set)[0]:n.set[0]);if(d){n=f?{expr:w.pop(),set:s(f)}:m.find(w.pop(),w.length===1&&(w[0]==="~"||w[0]==="+")&&d.parentNode?d.parentNode:d,v),j=n.expr?m.filter(n.expr,n.set):n.set,w.length>0?k=s(j):u=!1;while(w.length)q=w.pop(),r=q,o.relative[q]?r=w.pop():q="",r==null&&(r=d),o.relative[q](k,r,v)}else k=w=[]}k||(k=j),k||m.error(q||b);if(g.call(k)==="[object Array]")if(!u)e.push.apply(e,k);else if(d&&d.nodeType===1)for(t=0;k[t]!=null;t++)k[t]&&(k[t]===!0||k[t].nodeType===1&&m.contains(d,k[t]))&&e.push(j[t]);else for(t=0;k[t]!=null;t++)k[t]&&k[t].nodeType===1&&e.push(j[t]);else s(k,e);l&&(m(l,h,e,f),m.uniqueSort(e));return e};m.uniqueSort=function(a){if(u){h=i,a.sort(u);if(h)for(var b=1;b<a.length;b++)a[b]===a[b-1]&&a.splice(b--,1)}return a},m.matches=function(a,b){return m(a,null,null,b)},m.matchesSelector=function(a,b){return m(b,null,null,[a]).length>0},m.find=function(a,b,c){var d,e,f,g,h,i;if(!a)return[];for(e=0,f=o.order.length;e<f;e++){h=o.order[e];if(g=o.leftMatch[h].exec(a)){i=g[1],g.splice(1,1);if(i.substr(i.length-1)!=="\\"){g[1]=(g[1]||"").replace(j,""),d=o.find[h](g,b,c);if(d!=null){a=a.replace(o.match[h],"");break}}}}d||(d=typeof b.getElementsByTagName!="undefined"?b.getElementsByTagName("*"):[]);return{set:d,expr:a}},m.filter=function(a,c,d,e){var f,g,h,i,j,k,l,n,p,q=a,r=[],s=c,t=c&&c[0]&&m.isXML(c[0]);while(a&&c.length){for(h in o.filter)if((f=o.leftMatch[h].exec(a))!=null&&f[2]){k=o.filter[h],l=f[1],g=!1,f.splice(1,1);if(l.substr(l.length-1)==="\\")continue;s===r&&(r=[]);if(o.preFilter[h]){f=o.preFilter[h](f,s,d,r,e,t);if(!f)g=i=!0;else if(f===!0)continue}if(f)for(n=0;(j=s[n])!=null;n++)j&&(i=k(j,f,n,s),p=e^i,d&&i!=null?p?g=!0:s[n]=!1:p&&(r.push(j),g=!0));if(i!==b){d||(s=r),a=a.replace(o.match[h],"");if(!g)return[];break}}if(a===q)if(g==null)m.error(a);else break;q=a}return s},m.error=function(a){throw new Error("Syntax error, unrecognized expression: "+a)};var n=m.getText=function(a){var b,c,d=a.nodeType,e="";if(d){if(d===1||d===9){if(typeof a.textContent=="string")return a.textContent;if(typeof a.innerText=="string")return a.innerText.replace(k,"");for(a=a.firstChild;a;a=a.nextSibling)e+=n(a)}else if(d===3||d===4)return a.nodeValue}else for(b=0;c=a[b];b++)c.nodeType!==8&&(e+=n(c));return e},o=m.selectors={order:["ID","NAME","TAG"],match:{ID:/#((?:[\w\u00c0-\uFFFF\-]|\\.)+)/,CLASS:/\.((?:[\w\u00c0-\uFFFF\-]|\\.)+)/,NAME:/\[name=['"]*((?:[\w\u00c0-\uFFFF\-]|\\.)+)['"]*\]/,ATTR:/\[\s*((?:[\w\u00c0-\uFFFF\-]|\\.)+)\s*(?:(\S?=)\s*(?:(['"])(.*?)\3|(#?(?:[\w\u00c0-\uFFFF\-]|\\.)*)|)|)\s*\]/,TAG:/^((?:[\w\u00c0-\uFFFF\*\-]|\\.)+)/,CHILD:/:(only|nth|last|first)-child(?:\(\s*(even|odd|(?:[+\-]?\d+|(?:[+\-]?\d*)?n\s*(?:[+\-]\s*\d+)?))\s*\))?/,POS:/:(nth|eq|gt|lt|first|last|even|odd)(?:\((\d*)\))?(?=[^\-]|$)/,PSEUDO:/:((?:[\w\u00c0-\uFFFF\-]|\\.)+)(?:\((['"]?)((?:\([^\)]+\)|[^\(\)]*)+)\2\))?/},leftMatch:{},attrMap:{"class":"className","for":"htmlFor"},attrHandle:{href:function(a){return a.getAttribute("href")},type:function(a){return a.getAttribute("type")}},relative:{"+":function(a,b){var c=typeof b=="string",d=c&&!l.test(b),e=c&&!d;d&&(b=b.toLowerCase());for(var f=0,g=a.length,h;f<g;f++)if(h=a[f]){while((h=h.previousSibling)&&h.nodeType!==1);a[f]=e||h&&h.nodeName.toLowerCase()===b?h||!1:h===b}e&&m.filter(b,a,!0)},">":function(a,b){var c,d=typeof b=="string",e=0,f=a.length;if(d&&!l.test(b)){b=b.toLowerCase();for(;e<f;e++){c=a[e];if(c){var g=c.parentNode;a[e]=g.nodeName.toLowerCase()===b?g:!1}}}else{for(;e<f;e++)c=a[e],c&&(a[e]=d?c.parentNode:c.parentNode===b);d&&m.filter(b,a,!0)}},"":function(a,b,c){var d,f=e++,g=x;typeof b=="string"&&!l.test(b)&&(b=b.toLowerCase(),d=b,g=w),g("parentNode",b,f,a,d,c)},"~":function(a,b,c){var d,f=e++,g=x;typeof b=="string"&&!l.test(b)&&(b=b.toLowerCase(),d=b,g=w),g("previousSibling",b,f,a,d,c)}},find:{ID:function(a,b,c){if(typeof b.getElementById!="undefined"&&!c){var d=b.getElementById(a[1]);return d&&d.parentNode?[d]:[]}},NAME:function(a,b){if(typeof b.getElementsByName!="undefined"){var c=[],d=b.getElementsByName(a[1]);for(var e=0,f=d.length;e<f;e++)d[e].getAttribute("name")===a[1]&&c.push(d[e]);return c.length===0?null:c}},TAG:function(a,b){if(typeof b.getElementsByTagName!="undefined")return b.getElementsByTagName(a[1])}},preFilter:{CLASS:function(a,b,c,d,e,f){a=" "+a[1].replace(j,"")+" ";if(f)return a;for(var g=0,h;(h=b[g])!=null;g++)h&&(e^(h.className&&(" "+h.className+" ").replace(/[\t\n\r]/g," ").indexOf(a)>=0)?c||d.push(h):c&&(b[g]=!1));return!1},ID:function(a){return a[1].replace(j,"")},TAG:function(a,b){return a[1].replace(j,"").toLowerCase()},CHILD:function(a){if(a[1]==="nth"){a[2]||m.error(a[0]),a[2]=a[2].replace(/^\+|\s*/g,"");var b=/(-?)(\d*)(?:n([+\-]?\d*))?/.exec(a[2]==="even"&&"2n"||a[2]==="odd"&&"2n+1"||!/\D/.test(a[2])&&"0n+"+a[2]||a[2]);a[2]=b[1]+(b[2]||1)-0,a[3]=b[3]-0}else a[2]&&m.error(a[0]);a[0]=e++;return a},ATTR:function(a,b,c,d,e,f){var g=a[1]=a[1].replace(j,"");!f&&o.attrMap[g]&&(a[1]=o.attrMap[g]),a[4]=(a[4]||a[5]||"").replace(j,""),a[2]==="~="&&(a[4]=" "+a[4]+" ");return a},PSEUDO:function(b,c,d,e,f){if(b[1]==="not")if((a.exec(b[3])||"").length>1||/^\w/.test(b[3]))b[3]=m(b[3],null,null,c);else{var g=m.filter(b[3],c,d,!0^f);d||e.push.apply(e,g);return!1}else if(o.match.POS.test(b[0])||o.match.CHILD.test(b[0]))return!0;return b},POS:function(a){a.unshift(!0);return a}},filters:{enabled:function(a){return a.disabled===!1&&a.type!=="hidden"},disabled:function(a){return a.disabled===!0},checked:function(a){return a.checked===!0},selected:function(a){a.parentNode&&a.parentNode.selectedIndex;return a.selected===!0},parent:function(a){return!!a.firstChild},empty:function(a){return!a.firstChild},has:function(a,b,c){return!!m(c[3],a).length},header:function(a){return/h\d/i.test(a.nodeName)},text:function(a){var b=a.getAttribute("type"),c=a.type;return a.nodeName.toLowerCase()==="input"&&"text"===c&&(b===c||b===null)},radio:function(a){return a.nodeName.toLowerCase()==="input"&&"radio"===a.type},checkbox:function(a){return a.nodeName.toLowerCase()==="input"&&"checkbox"===a.type},file:function(a){return a.nodeName.toLowerCase()==="input"&&"file"===a.type},password:function(a){return a.nodeName.toLowerCase()==="input"&&"password"===a.type},submit:function(a){var b=a.nodeName.toLowerCase();return(b==="input"||b==="button")&&"submit"===a.type},image:function(a){return a.nodeName.toLowerCase()==="input"&&"image"===a.type},reset:function(a){var b=a.nodeName.toLowerCase();return(b==="input"||b==="button")&&"reset"===a.type},button:function(a){var b=a.nodeName.toLowerCase();return b==="input"&&"button"===a.type||b==="button"},input:function(a){return/input|select|textarea|button/i.test(a.nodeName)},focus:function(a){return a===a.ownerDocument.activeElement}},setFilters:{first:function(a,b){return b===0},last:function(a,b,c,d){return b===d.length-1},even:function(a,b){return b%2===0},odd:function(a,b){return b%2===1},lt:function(a,b,c){return b<c[3]-0},gt:function(a,b,c){return b>c[3]-0},nth:function(a,b,c){return c[3]-0===b},eq:function(a,b,c){return c[3]-0===b}},filter:{PSEUDO:function(a,b,c,d){var e=b[1],f=o.filters[e];if(f)return f(a,c,b,d);if(e==="contains")return(a.textContent||a.innerText||n([a])||"").indexOf(b[3])>=0;if(e==="not"){var g=b[3];for(var h=0,i=g.length;h<i;h++)if(g[h]===a)return!1;return!0}m.error(e)},CHILD:function(a,b){var c,e,f,g,h,i,j,k=b[1],l=a;switch(k){case"only":case"first":while(l=l.previousSibling)if(l.nodeType===1)return!1;if(k==="first")return!0;l=a;case"last":while(l=l.nextSibling)if(l.nodeType===1)return!1;return!0;case"nth":c=b[2],e=b[3];if(c===1&&e===0)return!0;f=b[0],g=a.parentNode;if(g&&(g[d]!==f||!a.nodeIndex)){i=0;for(l=g.firstChild;l;l=l.nextSibling)l.nodeType===1&&(l.nodeIndex=++i);g[d]=f}j=a.nodeIndex-e;return c===0?j===0:j%c===0&&j/c>=0}},ID:function(a,b){return a.nodeType===1&&a.getAttribute("id")===b},TAG:function(a,b){return b==="*"&&a.nodeType===1||!!a.nodeName&&a.nodeName.toLowerCase()===b},CLASS:function(a,b){return(" "+(a.className||a.getAttribute("class"))+" ").indexOf(b)>-1},ATTR:function(a,b){var c=b[1],d=m.attr?m.attr(a,c):o.attrHandle[c]?o.attrHandle[c](a):a[c]!=null?a[c]:a.getAttribute(c),e=d+"",f=b[2],g=b[4];return d==null?f==="!=":!f&&m.attr?d!=null:f==="="?e===g:f==="*="?e.indexOf(g)>=0:f==="~="?(" "+e+" ").indexOf(g)>=0:g?f==="!="?e!==g:f==="^="?e.indexOf(g)===0:f==="$="?e.substr(e.length-g.length)===g:f==="|="?e===g||e.substr(0,g.length+1)===g+"-":!1:e&&d!==!1},POS:function(a,b,c,d){var e=b[2],f=o.setFilters[e];if(f)return f(a,c,b,d)}}},p=o.match.POS,q=function(a,b){return"\\"+(b-0+1)};for(var r in o.match)o.match[r]=new RegExp(o.match[r].source+/(?![^\[]*\])(?![^\(]*\))/.source),o.leftMatch[r]=new RegExp(/(^(?:.|\r|\n)*?)/.source+o.match[r].source.replace(/\\(\d+)/g,q));var s=function(a,b){a=Array.prototype.slice.call(a,0);if(b){b.push.apply(b,a);return b}return a};try{Array.prototype.slice.call(c.documentElement.childNodes,0)[0].nodeType}catch(t){s=function(a,b){var c=0,d=b||[];if(g.call(a)==="[object Array]")Array.prototype.push.apply(d,a);else if(typeof a.length=="number")for(var e=a.length;c<e;c++)d.push(a[c]);else for(;a[c];c++)d.push(a[c]);return d}}var u,v;c.documentElement.compareDocumentPosition?u=function(a,b){if(a===b){h=!0;return 0}if(!a.compareDocumentPosition||!b.compareDocumentPosition)return a.compareDocumentPosition?-1:1;return a.compareDocumentPosition(b)&4?-1:1}:(u=function(a,b){if(a===b){h=!0;return 0}if(a.sourceIndex&&b.sourceIndex)return a.sourceIndex-b.sourceIndex;var c,d,e=[],f=[],g=a.parentNode,i=b.parentNode,j=g;if(g===i)return v(a,b);if(!g)return-1;if(!i)return 1;while(j)e.unshift(j),j=j.parentNode;j=i;while(j)f.unshift(j),j=j.parentNode;c=e.length,d=f.length;for(var k=0;k<c&&k<d;k++)if(e[k]!==f[k])return v(e[k],f[k]);return k===c?v(a,f[k],-1):v(e[k],b,1)},v=function(a,b,c){if(a===b)return c;var d=a.nextSibling;while(d){if(d===b)return-1;d=d.nextSibling}return 1}),function(){var a=c.createElement("div"),d="script"+(new Date).getTime(),e=c.documentElement;a.innerHTML="<a name='"+d+"'/>",e.insertBefore(a,e.firstChild),c.getElementById(d)&&(o.find.ID=function(a,c,d){if(typeof c.getElementById!="undefined"&&!d){var e=c.getElementById(a[1]);return e?e.id===a[1]||typeof e.getAttributeNode!="undefined"&&e.getAttributeNode("id").nodeValue===a[1]?[e]:b:[]}},o.filter.ID=function(a,b){var c=typeof a.getAttributeNode!="undefined"&&a.getAttributeNode("id");return a.nodeType===1&&c&&c.nodeValue===b}),e.removeChild(a),e=a=null}(),function(){var a=c.createElement("div");a.appendChild(c.createComment("")),a.getElementsByTagName("*").length>0&&(o.find.TAG=function(a,b){var c=b.getElementsByTagName(a[1]);if(a[1]==="*"){var d=[];for(var e=0;c[e];e++)c[e].nodeType===1&&d.push(c[e]);c=d}return c}),a.innerHTML="<a href='#'></a>",a.firstChild&&typeof a.firstChild.getAttribute!="undefined"&&a.firstChild.getAttribute("href")!=="#"&&(o.attrHandle.href=function(a){return a.getAttribute("href",2)}),a=null}(),c.querySelectorAll&&function(){var a=m,b=c.createElement("div"),d="__sizzle__";b.innerHTML="<p class='TEST'></p>";if(!b.querySelectorAll||b.querySelectorAll(".TEST").length!==0){m=function(b,e,f,g){e=e||c;if(!g&&!m.isXML(e)){var h=/^(\w+$)|^\.([\w\-]+$)|^#([\w\-]+$)/.exec(b);if(h&&(e.nodeType===1||e.nodeType===9)){if(h[1])return s(e.getElementsByTagName(b),f);if(h[2]&&o.find.CLASS&&e.getElementsByClassName)return s(e.getElementsByClassName(h[2]),f)}if(e.nodeType===9){if(b==="body"&&e.body)return s([e.body],f);if(h&&h[3]){var i=e.getElementById(h[3]);if(!i||!i.parentNode)return s([],f);if(i.id===h[3])return s([i],f)}try{return s(e.querySelectorAll(b),f)}catch(j){}}else if(e.nodeType===1&&e.nodeName.toLowerCase()!=="object"){var k=e,l=e.getAttribute("id"),n=l||d,p=e.parentNode,q=/^\s*[+~]/.test(b);l?n=n.replace(/'/g,"\\$&"):e.setAttribute("id",n),q&&p&&(e=e.parentNode);try{if(!q||p)return s(e.querySelectorAll("[id='"+n+"'] "+b),f)}catch(r){}finally{l||k.removeAttribute("id")}}}return a(b,e,f,g)};for(var e in a)m[e]=a[e];b=null}}(),function(){var a=c.documentElement,b=a.matchesSelector||a.mozMatchesSelector||a.webkitMatchesSelector||a.msMatchesSelector;if(b){var d=!b.call(c.createElement("div"),"div"),e=!1;try{b.call(c.documentElement,"[test!='']:sizzle")}catch(f){e=!0}m.matchesSelector=function(a,c){c=c.replace(/\=\s*([^'"\]]*)\s*\]/g,"='$1']");if(!m.isXML(a))try{if(e||!o.match.PSEUDO.test(c)&&!/!=/.test(c)){var f=b.call(a,c);if(f||!d||a.document&&a.document.nodeType!==11)return f}}catch(g){}return m(c,null,null,[a]).length>0}}}(),function(){var a=c.createElement("div");a.innerHTML="<div class='test e'></div><div class='test'></div>";if(!!a.getElementsByClassName&&a.getElementsByClassName("e").length!==0){a.lastChild.className="e";if(a.getElementsByClassName("e").length===1)return;o.order.splice(1,0,"CLASS"),o.find.CLASS=function(a,b,c){if(typeof b.getElementsByClassName!="undefined"&&!c)return b.getElementsByClassName(a[1])},a=null}}(),c.documentElement.contains?m.contains=function(a,b){return a!==b&&(a.contains?a.contains(b):!0)}:c.documentElement.compareDocumentPosition?m.contains=function(a,b){return!!(a.compareDocumentPosition(b)&16)}:m.contains=function(){return!1},m.isXML=function(a){var b=(a?a.ownerDocument||a:0).documentElement;return b?b.nodeName!=="HTML":!1};var y=function(a,b,c){var d,e=[],f="",g=b.nodeType?[b]:b;while(d=o.match.PSEUDO.exec(a))f+=d[0],a=a.replace(o.match.PSEUDO,"");a=o.relative[a]?a+"*":a;for(var h=0,i=g.length;h<i;h++)m(a,g[h],e,c);return m.filter(f,e)};m.attr=f.attr,m.selectors.attrMap={},f.find=m,f.expr=m.selectors,f.expr[":"]=f.expr.filters,f.unique=m.uniqueSort,f.text=m.getText,f.isXMLDoc=m.isXML,f.contains=m.contains}();var L=/Until$/,M=/^(?:parents|prevUntil|prevAll)/,N=/,/,O=/^.[^:#\[\.,]*$/,P=Array.prototype.slice,Q=f.expr.match.POS,R={children:!0,contents:!0,next:!0,prev:!0};f.fn.extend({find:function(a){var b=this,c,d;if(typeof a!="string")return f(a).filter(function(){for(c=0,d=b.length;c<d;c++)if(f.contains(b[c],this))return!0});var e=this.pushStack("","find",a),g,h,i;for(c=0,d=this.length;c<d;c++){g=e.length,f.find(a,this[c],e);if(c>0)for(h=g;h<e.length;h++)for(i=0;i<g;i++)if(e[i]===e[h]){e.splice(h--,1);break}}return e},has:function(a){var b=f(a);return this.filter(function(){for(var a=0,c=b.length;a<c;a++)if(f.contains(this,b[a]))return!0})},not:function(a){return this.pushStack(T(this,a,!1),"not",a)},filter:function(a){return this.pushStack(T(this,a,!0),"filter",a)},is:function(a){return!!a&&(typeof a=="string"?Q.test(a)?f(a,this.context).index(this[0])>=0:f.filter(a,this).length>0:this.filter(a).length>0)},closest:function(a,b){var c=[],d,e,g=this[0];if(f.isArray(a)){var h=1;while(g&&g.ownerDocument&&g!==b){for(d=0;d<a.length;d++)f(g).is(a[d])&&c.push({selector:a[d],elem:g,level:h});g=g.parentNode,h++}return c}var i=Q.test(a)||typeof a!="string"?f(a,b||this.context):0;for(d=0,e=this.length;d<e;d++){g=this[d];while(g){if(i?i.index(g)>-1:f.find.matchesSelector(g,a)){c.push(g);break}g=g.parentNode;if(!g||!g.ownerDocument||g===b||g.nodeType===11)break}}c=c.length>1?f.unique(c):c;return this.pushStack(c,"closest",a)},index:function(a){if(!a)return this[0]&&this[0].parentNode?this.prevAll().length:-1;if(typeof a=="string")return f.inArray(this[0],f(a));return f.inArray(a.jquery?a[0]:a,this)},add:function(a,b){var c=typeof a=="string"?f(a,b):f.makeArray(a&&a.nodeType?[a]:a),d=f.merge(this.get(),c);return this.pushStack(S(c[0])||S(d[0])?d:f.unique(d))},andSelf:function(){return this.add(this.prevObject)}}),f.each({parent:function(a){var b=a.parentNode;return b&&b.nodeType!==11?b:null},parents:function(a){return f.dir(a,"parentNode")},parentsUntil:function(a,b,c){return f.dir(a,"parentNode",c)},next:function(a){return f.nth(a,2,"nextSibling")},prev:function(a){return f.nth(a,2,"previousSibling")},nextAll:function(a){return f.dir(a,"nextSibling")},prevAll:function(a){return f.dir(a,"previousSibling")},nextUntil:function(a,b,c){return f.dir(a,"nextSibling",c)},prevUntil:function(a,b,c){return f.dir(a,"previousSibling",c)},siblings:function(a){return f.sibling(a.parentNode.firstChild,a)},children:function(a){return f.sibling(a.firstChild)},contents:function(a){return f.nodeName(a,"iframe")?a.contentDocument||a.contentWindow.document:f.makeArray(a.childNodes)}},function(a,b){f.fn[a]=function(c,d){var e=f.map(this,b,c);L.test(a)||(d=c),d&&typeof d=="string"&&(e=f.filter(d,e)),e=this.length>1&&!R[a]?f.unique(e):e,(this.length>1||N.test(d))&&M.test(a)&&(e=e.reverse());return this.pushStack(e,a,P.call(arguments).join(","))}}),f.extend({filter:function(a,b,c){c&&(a=":not("+a+")");return b.length===1?f.find.matchesSelector(b[0],a)?[b[0]]:[]:f.find.matches(a,b)},dir:function(a,c,d){var e=[],g=a[c];while(g&&g.nodeType!==9&&(d===b||g.nodeType!==1||!f(g).is(d)))g.nodeType===1&&e.push(g),g=g[c];return e},nth:function(a,b,c,d){b=b||1;var e=0;for(;a;a=a[c])if(a.nodeType===1&&++e===b)break;return a},sibling:function(a,b){var c=[];for(;a;a=a.nextSibling)a.nodeType===1&&a!==b&&c.push(a);return c}});var V="abbr|article|aside|audio|canvas|datalist|details|figcaption|figure|footer|header|hgroup|mark|meter|nav|output|progress|section|summary|time|video",W=/ jQuery\d+="(?:\d+|null)"/g,X=/^\s+/,Y=/<(?!area|br|col|embed|hr|img|input|link|meta|param)(([\w:]+)[^>]*)\/>/ig,Z=/<([\w:]+)/,$=/<tbody/i,_=/<|&#?\w+;/,ba=/<(?:script|style)/i,bb=/<(?:script|object|embed|option|style)/i,bc=new RegExp("<(?:"+V+")","i"),bd=/checked\s*(?:[^=]|=\s*.checked.)/i,be=/\/(java|ecma)script/i,bf=/^\s*<!(?:\[CDATA\[|\-\-)/,bg={option:[1,"<select multiple='multiple'>","</select>"],legend:[1,"<fieldset>","</fieldset>"],thead:[1,"<table>","</table>"],tr:[2,"<table><tbody>","</tbody></table>"],td:[3,"<table><tbody><tr>","</tr></tbody></table>"],col:[2,"<table><tbody></tbody><colgroup>","</colgroup></table>"],area:[1,"<map>","</map>"],_default:[0,"",""]},bh=U(c);bg.optgroup=bg.option,bg.tbody=bg.tfoot=bg.colgroup=bg.caption=bg.thead,bg.th=bg.td,f.support.htmlSerialize||(bg._default=[1,"div<div>","</div>"]),f.fn.extend({text:function(a){if(f.isFunction(a))return this.each(function(b){var c=f(this);c.text(a.call(this,b,c.text()))});if(typeof a!="object"&&a!==b)return this.empty().append((this[0]&&this[0].ownerDocument||c).createTextNode(a));return f.text(this)},wrapAll:function(a){if(f.isFunction(a))return this.each(function(b){f(this).wrapAll(a.call(this,b))});if(this[0]){var b=f(a,this[0].ownerDocument).eq(0).clone(!0);this[0].parentNode&&b.insertBefore(this[0]),b.map(function(){var a=this;while(a.firstChild&&a.firstChild.nodeType===1)a=a.firstChild;return a}).append(this)}return this},wrapInner:function(a){if(f.isFunction(a))return this.each(function(b){f(this).wrapInner(a.call(this,b))});return this.each(function(){var b=f(this),c=b.contents();c.length?c.wrapAll(a):b.append(a)})},wrap:function(a){var b=f.isFunction(a);return this.each(function(c){f(this).wrapAll(b?a.call(this,c):a)})},unwrap:function(){return this.parent().each(function(){f.nodeName(this,"body")||f(this).replaceWith(this.childNodes)}).end()},append:function(){return this.domManip(arguments,!0,function(a){this.nodeType===1&&this.appendChild(a)})},prepend:function(){return this.domManip(arguments,!0,function(a){this.nodeType===1&&this.insertBefore(a,this.firstChild)})},before:function(){if(this[0]&&this[0].parentNode)return this.domManip(arguments,!1,function(a){this.parentNode.insertBefore(a,this)});if(arguments.length){var a=f.clean(arguments);a.push.apply(a,this.toArray());return this.pushStack(a,"before",arguments)}},after:function(){if(this[0]&&this[0].parentNode)return this.domManip(arguments,!1,function(a){this.parentNode.insertBefore(a,this.nextSibling)});if(arguments.length){var a=this.pushStack(this,"after",arguments);a.push.apply(a,f.clean(arguments));return a}},remove:function(a,b){for(var c=0,d;(d=this[c])!=null;c++)if(!a||f.filter(a,[d]).length)!b&&d.nodeType===1&&(f.cleanData(d.getElementsByTagName("*")),f.cleanData([d])),d.parentNode&&d.parentNode.removeChild(d);return this},empty:function() - {for(var a=0,b;(b=this[a])!=null;a++){b.nodeType===1&&f.cleanData(b.getElementsByTagName("*"));while(b.firstChild)b.removeChild(b.firstChild)}return this},clone:function(a,b){a=a==null?!1:a,b=b==null?a:b;return this.map(function(){return f.clone(this,a,b)})},html:function(a){if(a===b)return this[0]&&this[0].nodeType===1?this[0].innerHTML.replace(W,""):null;if(typeof a=="string"&&!ba.test(a)&&(f.support.leadingWhitespace||!X.test(a))&&!bg[(Z.exec(a)||["",""])[1].toLowerCase()]){a=a.replace(Y,"<$1></$2>");try{for(var c=0,d=this.length;c<d;c++)this[c].nodeType===1&&(f.cleanData(this[c].getElementsByTagName("*")),this[c].innerHTML=a)}catch(e){this.empty().append(a)}}else f.isFunction(a)?this.each(function(b){var c=f(this);c.html(a.call(this,b,c.html()))}):this.empty().append(a);return this},replaceWith:function(a){if(this[0]&&this[0].parentNode){if(f.isFunction(a))return this.each(function(b){var c=f(this),d=c.html();c.replaceWith(a.call(this,b,d))});typeof a!="string"&&(a=f(a).detach());return this.each(function(){var b=this.nextSibling,c=this.parentNode;f(this).remove(),b?f(b).before(a):f(c).append(a)})}return this.length?this.pushStack(f(f.isFunction(a)?a():a),"replaceWith",a):this},detach:function(a){return this.remove(a,!0)},domManip:function(a,c,d){var e,g,h,i,j=a[0],k=[];if(!f.support.checkClone&&arguments.length===3&&typeof j=="string"&&bd.test(j))return this.each(function(){f(this).domManip(a,c,d,!0)});if(f.isFunction(j))return this.each(function(e){var g=f(this);a[0]=j.call(this,e,c?g.html():b),g.domManip(a,c,d)});if(this[0]){i=j&&j.parentNode,f.support.parentNode&&i&&i.nodeType===11&&i.childNodes.length===this.length?e={fragment:i}:e=f.buildFragment(a,this,k),h=e.fragment,h.childNodes.length===1?g=h=h.firstChild:g=h.firstChild;if(g){c=c&&f.nodeName(g,"tr");for(var l=0,m=this.length,n=m-1;l<m;l++)d.call(c?bi(this[l],g):this[l],e.cacheable||m>1&&l<n?f.clone(h,!0,!0):h)}k.length&&f.each(k,bp)}return this}}),f.buildFragment=function(a,b,d){var e,g,h,i,j=a[0];b&&b[0]&&(i=b[0].ownerDocument||b[0]),i.createDocumentFragment||(i=c),a.length===1&&typeof j=="string"&&j.length<512&&i===c&&j.charAt(0)==="<"&&!bb.test(j)&&(f.support.checkClone||!bd.test(j))&&(f.support.html5Clone||!bc.test(j))&&(g=!0,h=f.fragments[j],h&&h!==1&&(e=h)),e||(e=i.createDocumentFragment(),f.clean(a,i,e,d)),g&&(f.fragments[j]=h?e:1);return{fragment:e,cacheable:g}},f.fragments={},f.each({appendTo:"append",prependTo:"prepend",insertBefore:"before",insertAfter:"after",replaceAll:"replaceWith"},function(a,b){f.fn[a]=function(c){var d=[],e=f(c),g=this.length===1&&this[0].parentNode;if(g&&g.nodeType===11&&g.childNodes.length===1&&e.length===1){e[b](this[0]);return this}for(var h=0,i=e.length;h<i;h++){var j=(h>0?this.clone(!0):this).get();f(e[h])[b](j),d=d.concat(j)}return this.pushStack(d,a,e.selector)}}),f.extend({clone:function(a,b,c){var d,e,g,h=f.support.html5Clone||!bc.test("<"+a.nodeName)?a.cloneNode(!0):bo(a);if((!f.support.noCloneEvent||!f.support.noCloneChecked)&&(a.nodeType===1||a.nodeType===11)&&!f.isXMLDoc(a)){bk(a,h),d=bl(a),e=bl(h);for(g=0;d[g];++g)e[g]&&bk(d[g],e[g])}if(b){bj(a,h);if(c){d=bl(a),e=bl(h);for(g=0;d[g];++g)bj(d[g],e[g])}}d=e=null;return h},clean:function(a,b,d,e){var g;b=b||c,typeof b.createElement=="undefined"&&(b=b.ownerDocument||b[0]&&b[0].ownerDocument||c);var h=[],i;for(var j=0,k;(k=a[j])!=null;j++){typeof k=="number"&&(k+="");if(!k)continue;if(typeof k=="string")if(!_.test(k))k=b.createTextNode(k);else{k=k.replace(Y,"<$1></$2>");var l=(Z.exec(k)||["",""])[1].toLowerCase(),m=bg[l]||bg._default,n=m[0],o=b.createElement("div");b===c?bh.appendChild(o):U(b).appendChild(o),o.innerHTML=m[1]+k+m[2];while(n--)o=o.lastChild;if(!f.support.tbody){var p=$.test(k),q=l==="table"&&!p?o.firstChild&&o.firstChild.childNodes:m[1]==="<table>"&&!p?o.childNodes:[];for(i=q.length-1;i>=0;--i)f.nodeName(q[i],"tbody")&&!q[i].childNodes.length&&q[i].parentNode.removeChild(q[i])}!f.support.leadingWhitespace&&X.test(k)&&o.insertBefore(b.createTextNode(X.exec(k)[0]),o.firstChild),k=o.childNodes}var r;if(!f.support.appendChecked)if(k[0]&&typeof (r=k.length)=="number")for(i=0;i<r;i++)bn(k[i]);else bn(k);k.nodeType?h.push(k):h=f.merge(h,k)}if(d){g=function(a){return!a.type||be.test(a.type)};for(j=0;h[j];j++)if(e&&f.nodeName(h[j],"script")&&(!h[j].type||h[j].type.toLowerCase()==="text/javascript"))e.push(h[j].parentNode?h[j].parentNode.removeChild(h[j]):h[j]);else{if(h[j].nodeType===1){var s=f.grep(h[j].getElementsByTagName("script"),g);h.splice.apply(h,[j+1,0].concat(s))}d.appendChild(h[j])}}return h},cleanData:function(a){var b,c,d=f.cache,e=f.event.special,g=f.support.deleteExpando;for(var h=0,i;(i=a[h])!=null;h++){if(i.nodeName&&f.noData[i.nodeName.toLowerCase()])continue;c=i[f.expando];if(c){b=d[c];if(b&&b.events){for(var j in b.events)e[j]?f.event.remove(i,j):f.removeEvent(i,j,b.handle);b.handle&&(b.handle.elem=null)}g?delete i[f.expando]:i.removeAttribute&&i.removeAttribute(f.expando),delete d[c]}}}});var bq=/alpha\([^)]*\)/i,br=/opacity=([^)]*)/,bs=/([A-Z]|^ms)/g,bt=/^-?\d+(?:px)?$/i,bu=/^-?\d/,bv=/^([\-+])=([\-+.\de]+)/,bw={position:"absolute",visibility:"hidden",display:"block"},bx=["Left","Right"],by=["Top","Bottom"],bz,bA,bB;f.fn.css=function(a,c){if(arguments.length===2&&c===b)return this;return f.access(this,a,c,!0,function(a,c,d){return d!==b?f.style(a,c,d):f.css(a,c)})},f.extend({cssHooks:{opacity:{get:function(a,b){if(b){var c=bz(a,"opacity","opacity");return c===""?"1":c}return a.style.opacity}}},cssNumber:{fillOpacity:!0,fontWeight:!0,lineHeight:!0,opacity:!0,orphans:!0,widows:!0,zIndex:!0,zoom:!0},cssProps:{"float":f.support.cssFloat?"cssFloat":"styleFloat"},style:function(a,c,d,e){if(!!a&&a.nodeType!==3&&a.nodeType!==8&&!!a.style){var g,h,i=f.camelCase(c),j=a.style,k=f.cssHooks[i];c=f.cssProps[i]||i;if(d===b){if(k&&"get"in k&&(g=k.get(a,!1,e))!==b)return g;return j[c]}h=typeof d,h==="string"&&(g=bv.exec(d))&&(d=+(g[1]+1)*+g[2]+parseFloat(f.css(a,c)),h="number");if(d==null||h==="number"&&isNaN(d))return;h==="number"&&!f.cssNumber[i]&&(d+="px");if(!k||!("set"in k)||(d=k.set(a,d))!==b)try{j[c]=d}catch(l){}}},css:function(a,c,d){var e,g;c=f.camelCase(c),g=f.cssHooks[c],c=f.cssProps[c]||c,c==="cssFloat"&&(c="float");if(g&&"get"in g&&(e=g.get(a,!0,d))!==b)return e;if(bz)return bz(a,c)},swap:function(a,b,c){var d={};for(var e in b)d[e]=a.style[e],a.style[e]=b[e];c.call(a);for(e in b)a.style[e]=d[e]}}),f.curCSS=f.css,f.each(["height","width"],function(a,b){f.cssHooks[b]={get:function(a,c,d){var e;if(c){if(a.offsetWidth!==0)return bC(a,b,d);f.swap(a,bw,function(){e=bC(a,b,d)});return e}},set:function(a,b){if(!bt.test(b))return b;b=parseFloat(b);if(b>=0)return b+"px"}}}),f.support.opacity||(f.cssHooks.opacity={get:function(a,b){return br.test((b&&a.currentStyle?a.currentStyle.filter:a.style.filter)||"")?parseFloat(RegExp.$1)/100+"":b?"1":""},set:function(a,b){var c=a.style,d=a.currentStyle,e=f.isNumeric(b)?"alpha(opacity="+b*100+")":"",g=d&&d.filter||c.filter||"";c.zoom=1;if(b>=1&&f.trim(g.replace(bq,""))===""){c.removeAttribute("filter");if(d&&!d.filter)return}c.filter=bq.test(g)?g.replace(bq,e):g+" "+e}}),f(function(){f.support.reliableMarginRight||(f.cssHooks.marginRight={get:function(a,b){var c;f.swap(a,{display:"inline-block"},function(){b?c=bz(a,"margin-right","marginRight"):c=a.style.marginRight});return c}})}),c.defaultView&&c.defaultView.getComputedStyle&&(bA=function(a,b){var c,d,e;b=b.replace(bs,"-$1").toLowerCase(),(d=a.ownerDocument.defaultView)&&(e=d.getComputedStyle(a,null))&&(c=e.getPropertyValue(b),c===""&&!f.contains(a.ownerDocument.documentElement,a)&&(c=f.style(a,b)));return c}),c.documentElement.currentStyle&&(bB=function(a,b){var c,d,e,f=a.currentStyle&&a.currentStyle[b],g=a.style;f===null&&g&&(e=g[b])&&(f=e),!bt.test(f)&&bu.test(f)&&(c=g.left,d=a.runtimeStyle&&a.runtimeStyle.left,d&&(a.runtimeStyle.left=a.currentStyle.left),g.left=b==="fontSize"?"1em":f||0,f=g.pixelLeft+"px",g.left=c,d&&(a.runtimeStyle.left=d));return f===""?"auto":f}),bz=bA||bB,f.expr&&f.expr.filters&&(f.expr.filters.hidden=function(a){var b=a.offsetWidth,c=a.offsetHeight;return b===0&&c===0||!f.support.reliableHiddenOffsets&&(a.style&&a.style.display||f.css(a,"display"))==="none"},f.expr.filters.visible=function(a){return!f.expr.filters.hidden(a)});var bD=/%20/g,bE=/\[\]$/,bF=/\r?\n/g,bG=/#.*$/,bH=/^(.*?):[ \t]*([^\r\n]*)\r?$/mg,bI=/^(?:color|date|datetime|datetime-local|email|hidden|month|number|password|range|search|tel|text|time|url|week)$/i,bJ=/^(?:about|app|app\-storage|.+\-extension|file|res|widget):$/,bK=/^(?:GET|HEAD)$/,bL=/^\/\//,bM=/\?/,bN=/<script\b[^<]*(?:(?!<\/script>)<[^<]*)*<\/script>/gi,bO=/^(?:select|textarea)/i,bP=/\s+/,bQ=/([?&])_=[^&]*/,bR=/^([\w\+\.\-]+:)(?:\/\/([^\/?#:]*)(?::(\d+))?)?/,bS=f.fn.load,bT={},bU={},bV,bW,bX=["*/"]+["*"];try{bV=e.href}catch(bY){bV=c.createElement("a"),bV.href="",bV=bV.href}bW=bR.exec(bV.toLowerCase())||[],f.fn.extend({load:function(a,c,d){if(typeof a!="string"&&bS)return bS.apply(this,arguments);if(!this.length)return this;var e=a.indexOf(" ");if(e>=0){var g=a.slice(e,a.length);a=a.slice(0,e)}var h="GET";c&&(f.isFunction(c)?(d=c,c=b):typeof c=="object"&&(c=f.param(c,f.ajaxSettings.traditional),h="POST"));var i=this;f.ajax({url:a,type:h,dataType:"html",data:c,complete:function(a,b,c){c=a.responseText,a.isResolved()&&(a.done(function(a){c=a}),i.html(g?f("<div>").append(c.replace(bN,"")).find(g):c)),d&&i.each(d,[c,b,a])}});return this},serialize:function(){return f.param(this.serializeArray())},serializeArray:function(){return this.map(function(){return this.elements?f.makeArray(this.elements):this}).filter(function(){return this.name&&!this.disabled&&(this.checked||bO.test(this.nodeName)||bI.test(this.type))}).map(function(a,b){var c=f(this).val();return c==null?null:f.isArray(c)?f.map(c,function(a,c){return{name:b.name,value:a.replace(bF,"\r\n")}}):{name:b.name,value:c.replace(bF,"\r\n")}}).get()}}),f.each("ajaxStart ajaxStop ajaxComplete ajaxError ajaxSuccess ajaxSend".split(" "),function(a,b){f.fn[b]=function(a){return this.on(b,a)}}),f.each(["get","post"],function(a,c){f[c]=function(a,d,e,g){f.isFunction(d)&&(g=g||e,e=d,d=b);return f.ajax({type:c,url:a,data:d,success:e,dataType:g})}}),f.extend({getScript:function(a,c){return f.get(a,b,c,"script")},getJSON:function(a,b,c){return f.get(a,b,c,"json")},ajaxSetup:function(a,b){b?b_(a,f.ajaxSettings):(b=a,a=f.ajaxSettings),b_(a,b);return a},ajaxSettings:{url:bV,isLocal:bJ.test(bW[1]),global:!0,type:"GET",contentType:"application/x-www-form-urlencoded",processData:!0,async:!0,accepts:{xml:"application/xml, text/xml",html:"text/html",text:"text/plain",json:"application/json, text/javascript","*":bX},contents:{xml:/xml/,html:/html/,json:/json/},responseFields:{xml:"responseXML",text:"responseText"},converters:{"* text":a.String,"text html":!0,"text json":f.parseJSON,"text xml":f.parseXML},flatOptions:{context:!0,url:!0}},ajaxPrefilter:bZ(bT),ajaxTransport:bZ(bU),ajax:function(a,c){function w(a,c,l,m){if(s!==2){s=2,q&&clearTimeout(q),p=b,n=m||"",v.readyState=a>0?4:0;var o,r,u,w=c,x=l?cb(d,v,l):b,y,z;if(a>=200&&a<300||a===304){if(d.ifModified){if(y=v.getResponseHeader("Last-Modified"))f.lastModified[k]=y;if(z=v.getResponseHeader("Etag"))f.etag[k]=z}if(a===304)w="notmodified",o=!0;else try{r=cc(d,x),w="success",o=!0}catch(A){w="parsererror",u=A}}else{u=w;if(!w||a)w="error",a<0&&(a=0)}v.status=a,v.statusText=""+(c||w),o?h.resolveWith(e,[r,w,v]):h.rejectWith(e,[v,w,u]),v.statusCode(j),j=b,t&&g.trigger("ajax"+(o?"Success":"Error"),[v,d,o?r:u]),i.fireWith(e,[v,w]),t&&(g.trigger("ajaxComplete",[v,d]),--f.active||f.event.trigger("ajaxStop"))}}typeof a=="object"&&(c=a,a=b),c=c||{};var d=f.ajaxSetup({},c),e=d.context||d,g=e!==d&&(e.nodeType||e instanceof f)?f(e):f.event,h=f.Deferred(),i=f.Callbacks("once memory"),j=d.statusCode||{},k,l={},m={},n,o,p,q,r,s=0,t,u,v={readyState:0,setRequestHeader:function(a,b){if(!s){var c=a.toLowerCase();a=m[c]=m[c]||a,l[a]=b}return this},getAllResponseHeaders:function(){return s===2?n:null},getResponseHeader:function(a){var c;if(s===2){if(!o){o={};while(c=bH.exec(n))o[c[1].toLowerCase()]=c[2]}c=o[a.toLowerCase()]}return c===b?null:c},overrideMimeType:function(a){s||(d.mimeType=a);return this},abort:function(a){a=a||"abort",p&&p.abort(a),w(0,a);return this}};h.promise(v),v.success=v.done,v.error=v.fail,v.complete=i.add,v.statusCode=function(a){if(a){var b;if(s<2)for(b in a)j[b]=[j[b],a[b]];else b=a[v.status],v.then(b,b)}return this},d.url=((a||d.url)+"").replace(bG,"").replace(bL,bW[1]+"//"),d.dataTypes=f.trim(d.dataType||"*").toLowerCase().split(bP),d.crossDomain==null&&(r=bR.exec(d.url.toLowerCase()),d.crossDomain=!(!r||r[1]==bW[1]&&r[2]==bW[2]&&(r[3]||(r[1]==="http:"?80:443))==(bW[3]||(bW[1]==="http:"?80:443)))),d.data&&d.processData&&typeof d.data!="string"&&(d.data=f.param(d.data,d.traditional)),b$(bT,d,c,v);if(s===2)return!1;t=d.global,d.type=d.type.toUpperCase(),d.hasContent=!bK.test(d.type),t&&f.active++===0&&f.event.trigger("ajaxStart");if(!d.hasContent){d.data&&(d.url+=(bM.test(d.url)?"&":"?")+d.data,delete d.data),k=d.url;if(d.cache===!1){var x=f.now(),y=d.url.replace(bQ,"$1_="+x);d.url=y+(y===d.url?(bM.test(d.url)?"&":"?")+"_="+x:"")}}(d.data&&d.hasContent&&d.contentType!==!1||c.contentType)&&v.setRequestHeader("Content-Type",d.contentType),d.ifModified&&(k=k||d.url,f.lastModified[k]&&v.setRequestHeader("If-Modified-Since",f.lastModified[k]),f.etag[k]&&v.setRequestHeader("If-None-Match",f.etag[k])),v.setRequestHeader("Accept",d.dataTypes[0]&&d.accepts[d.dataTypes[0]]?d.accepts[d.dataTypes[0]]+(d.dataTypes[0]!=="*"?", "+bX+"; q=0.01":""):d.accepts["*"]);for(u in d.headers)v.setRequestHeader(u,d.headers[u]);if(d.beforeSend&&(d.beforeSend.call(e,v,d)===!1||s===2)){v.abort();return!1}for(u in{success:1,error:1,complete:1})v[u](d[u]);p=b$(bU,d,c,v);if(!p)w(-1,"No Transport");else{v.readyState=1,t&&g.trigger("ajaxSend",[v,d]),d.async&&d.timeout>0&&(q=setTimeout(function(){v.abort("timeout")},d.timeout));try{s=1,p.send(l,w)}catch(z){if(s<2)w(-1,z);else throw z}}return v},param:function(a,c){var d=[],e=function(a,b){b=f.isFunction(b)?b():b,d[d.length]=encodeURIComponent(a)+"="+encodeURIComponent(b)};c===b&&(c=f.ajaxSettings.traditional);if(f.isArray(a)||a.jquery&&!f.isPlainObject(a))f.each(a,function(){e(this.name,this.value)});else for(var g in a)ca(g,a[g],c,e);return d.join("&").replace(bD,"+")}}),f.extend({active:0,lastModified:{},etag:{}});var cd=f.now(),ce=/(\=)\?(&|$)|\?\?/i;f.ajaxSetup({jsonp:"callback",jsonpCallback:function(){return f.expando+"_"+cd++}}),f.ajaxPrefilter("json jsonp",function(b,c,d){var e=b.contentType==="application/x-www-form-urlencoded"&&typeof b.data=="string";if(b.dataTypes[0]==="jsonp"||b.jsonp!==!1&&(ce.test(b.url)||e&&ce.test(b.data))){var g,h=b.jsonpCallback=f.isFunction(b.jsonpCallback)?b.jsonpCallback():b.jsonpCallback,i=a[h],j=b.url,k=b.data,l="$1"+h+"$2";b.jsonp!==!1&&(j=j.replace(ce,l),b.url===j&&(e&&(k=k.replace(ce,l)),b.data===k&&(j+=(/\?/.test(j)?"&":"?")+b.jsonp+"="+h))),b.url=j,b.data=k,a[h]=function(a){g=[a]},d.always(function(){a[h]=i,g&&f.isFunction(i)&&a[h](g[0])}),b.converters["script json"]=function(){g||f.error(h+" was not called");return g[0]},b.dataTypes[0]="json";return"script"}}),f.ajaxSetup({accepts:{script:"text/javascript, application/javascript, application/ecmascript, application/x-ecmascript"},contents:{script:/javascript|ecmascript/},converters:{"text script":function(a){f.globalEval(a);return a}}}),f.ajaxPrefilter("script",function(a){a.cache===b&&(a.cache=!1),a.crossDomain&&(a.type="GET",a.global=!1)}),f.ajaxTransport("script",function(a){if(a.crossDomain){var d,e=c.head||c.getElementsByTagName("head")[0]||c.documentElement;return{send:function(f,g){d=c.createElement("script"),d.async="async",a.scriptCharset&&(d.charset=a.scriptCharset),d.src=a.url,d.onload=d.onreadystatechange=function(a,c){if(c||!d.readyState||/loaded|complete/.test(d.readyState))d.onload=d.onreadystatechange=null,e&&d.parentNode&&e.removeChild(d),d=b,c||g(200,"success")},e.insertBefore(d,e.firstChild)},abort:function(){d&&d.onload(0,1)}}}});var cf=a.ActiveXObject?function(){for(var a in ch)ch[a](0,1)}:!1,cg=0,ch;f.ajaxSettings.xhr=a.ActiveXObject?function(){return!this.isLocal&&ci()||cj()}:ci,function(a){f.extend(f.support,{ajax:!!a,cors:!!a&&"withCredentials"in a})}(f.ajaxSettings.xhr()),f.support.ajax&&f.ajaxTransport(function(c){if(!c.crossDomain||f.support.cors){var d;return{send:function(e,g){var h=c.xhr(),i,j;c.username?h.open(c.type,c.url,c.async,c.username,c.password):h.open(c.type,c.url,c.async);if(c.xhrFields)for(j in c.xhrFields)h[j]=c.xhrFields[j];c.mimeType&&h.overrideMimeType&&h.overrideMimeType(c.mimeType),!c.crossDomain&&!e["X-Requested-With"]&&(e["X-Requested-With"]="XMLHttpRequest");try{for(j in e)h.setRequestHeader(j,e[j])}catch(k){}h.send(c.hasContent&&c.data||null),d=function(a,e){var j,k,l,m,n;try{if(d&&(e||h.readyState===4)){d=b,i&&(h.onreadystatechange=f.noop,cf&&delete ch[i]);if(e)h.readyState!==4&&h.abort();else{j=h.status,l=h.getAllResponseHeaders(),m={},n=h.responseXML,n&&n.documentElement&&(m.xml=n),m.text=h.responseText;try{k=h.statusText}catch(o){k=""}!j&&c.isLocal&&!c.crossDomain?j=m.text?200:404:j===1223&&(j=204)}}}catch(p){e||g(-1,p)}m&&g(j,k,m,l)},!c.async||h.readyState===4?d():(i=++cg,cf&&(ch||(ch={},f(a).unload(cf)),ch[i]=d),h.onreadystatechange=d)},abort:function(){d&&d(0,1)}}}});var ck={},cl,cm,cn=/^(?:toggle|show|hide)$/,co=/^([+\-]=)?([\d+.\-]+)([a-z%]*)$/i,cp,cq=[["height","marginTop","marginBottom","paddingTop","paddingBottom"],["width","marginLeft","marginRight","paddingLeft","paddingRight"],["opacity"]],cr;f.fn.extend({show:function(a,b,c){var d,e;if(a||a===0)return this.animate(cu("show",3),a,b,c);for(var g=0,h=this.length;g<h;g++)d=this[g],d.style&&(e=d.style.display,!f._data(d,"olddisplay")&&e==="none"&&(e=d.style.display=""),e===""&&f.css(d,"display")==="none"&&f._data(d,"olddisplay",cv(d.nodeName)));for(g=0;g<h;g++){d=this[g];if(d.style){e=d.style.display;if(e===""||e==="none")d.style.display=f._data(d,"olddisplay")||""}}return this},hide:function(a,b,c){if(a||a===0)return this.animate(cu("hide",3),a,b,c);var d,e,g=0,h=this.length;for(;g<h;g++)d=this[g],d.style&&(e=f.css(d,"display"),e!=="none"&&!f._data(d,"olddisplay")&&f._data(d,"olddisplay",e));for(g=0;g<h;g++)this[g].style&&(this[g].style.display="none");return this},_toggle:f.fn.toggle,toggle:function(a,b,c){var d=typeof a=="boolean";f.isFunction(a)&&f.isFunction(b)?this._toggle.apply(this,arguments):a==null||d?this.each(function(){var b=d?a:f(this).is(":hidden");f(this)[b?"show":"hide"]()}):this.animate(cu("toggle",3),a,b,c);return this},fadeTo:function(a,b,c,d){return this.filter(":hidden").css("opacity",0).show().end().animate({opacity:b},a,c,d)},animate:function(a,b,c,d){function g(){e.queue===!1&&f._mark(this);var b=f.extend({},e),c=this.nodeType===1,d=c&&f(this).is(":hidden"),g,h,i,j,k,l,m,n,o;b.animatedProperties={};for(i in a){g=f.camelCase(i),i!==g&&(a[g]=a[i],delete a[i]),h=a[g],f.isArray(h)?(b.animatedProperties[g]=h[1],h=a[g]=h[0]):b.animatedProperties[g]=b.specialEasing&&b.specialEasing[g]||b.easing||"swing";if(h==="hide"&&d||h==="show"&&!d)return b.complete.call(this);c&&(g==="height"||g==="width")&&(b.overflow=[this.style.overflow,this.style.overflowX,this.style.overflowY],f.css(this,"display")==="inline"&&f.css(this,"float")==="none"&&(!f.support.inlineBlockNeedsLayout||cv(this.nodeName)==="inline"?this.style.display="inline-block":this.style.zoom=1))}b.overflow!=null&&(this.style.overflow="hidden");for(i in a)j=new f.fx(this,b,i),h=a[i],cn.test(h)?(o=f._data(this,"toggle"+i)||(h==="toggle"?d?"show":"hide":0),o?(f._data(this,"toggle"+i,o==="show"?"hide":"show"),j[o]()):j[h]()):(k=co.exec(h),l=j.cur(),k?(m=parseFloat(k[2]),n=k[3]||(f.cssNumber[i]?"":"px"),n!=="px"&&(f.style(this,i,(m||1)+n),l=(m||1)/j.cur()*l,f.style(this,i,l+n)),k[1]&&(m=(k[1]==="-="?-1:1)*m+l),j.custom(l,m,n)):j.custom(l,h,""));return!0}var e=f.speed(b,c,d);if(f.isEmptyObject(a))return this.each(e.complete,[!1]);a=f.extend({},a);return e.queue===!1?this.each(g):this.queue(e.queue,g)},stop:function(a,c,d){typeof a!="string"&&(d=c,c=a,a=b),c&&a!==!1&&this.queue(a||"fx",[]);return this.each(function(){function h(a,b,c){var e=b[c];f.removeData(a,c,!0),e.stop(d)}var b,c=!1,e=f.timers,g=f._data(this);d||f._unmark(!0,this);if(a==null)for(b in g)g[b]&&g[b].stop&&b.indexOf(".run")===b.length-4&&h(this,g,b);else g[b=a+".run"]&&g[b].stop&&h(this,g,b);for(b=e.length;b--;)e[b].elem===this&&(a==null||e[b].queue===a)&&(d?e[b](!0):e[b].saveState(),c=!0,e.splice(b,1));(!d||!c)&&f.dequeue(this,a)})}}),f.each({slideDown:cu("show",1),slideUp:cu("hide",1),slideToggle:cu("toggle",1),fadeIn:{opacity:"show"},fadeOut:{opacity:"hide"},fadeToggle:{opacity:"toggle"}},function(a,b){f.fn[a]=function(a,c,d){return this.animate(b,a,c,d)}}),f.extend({speed:function(a,b,c){var d=a&&typeof a=="object"?f.extend({},a):{complete:c||!c&&b||f.isFunction(a)&&a,duration:a,easing:c&&b||b&&!f.isFunction(b)&&b};d.duration=f.fx.off?0:typeof d.duration=="number"?d.duration:d.duration in f.fx.speeds?f.fx.speeds[d.duration]:f.fx.speeds._default;if(d.queue==null||d.queue===!0)d.queue="fx";d.old=d.complete,d.complete=function(a){f.isFunction(d.old)&&d.old.call(this),d.queue?f.dequeue(this,d.queue):a!==!1&&f._unmark(this)};return d},easing:{linear:function(a,b,c,d){return c+d*a},swing:function(a,b,c,d){return(-Math.cos(a*Math.PI)/2+.5)*d+c}},timers:[],fx:function(a,b,c){this.options=b,this.elem=a,this.prop=c,b.orig=b.orig||{}}}),f.fx.prototype={update:function(){this.options.step&&this.options.step.call(this.elem,this.now,this),(f.fx.step[this.prop]||f.fx.step._default)(this)},cur:function(){if(this.elem[this.prop]!=null&&(!this.elem.style||this.elem.style[this.prop]==null))return this.elem[this.prop];var a,b=f.css(this.elem,this.prop);return isNaN(a=parseFloat(b))?!b||b==="auto"?0:b:a},custom:function(a,c,d){function h(a){return e.step(a)}var e=this,g=f.fx;this.startTime=cr||cs(),this.end=c,this.now=this.start=a,this.pos=this.state=0,this.unit=d||this.unit||(f.cssNumber[this.prop]?"":"px"),h.queue=this.options.queue,h.elem=this.elem,h.saveState=function(){e.options.hide&&f._data(e.elem,"fxshow"+e.prop)===b&&f._data(e.elem,"fxshow"+e.prop,e.start)},h()&&f.timers.push(h)&&!cp&&(cp=setInterval(g.tick,g.interval))},show:function(){var a=f._data(this.elem,"fxshow"+this.prop);this.options.orig[this.prop]=a||f.style(this.elem,this.prop),this.options.show=!0,a!==b?this.custom(this.cur(),a):this.custom(this.prop==="width"||this.prop==="height"?1:0,this.cur()),f(this.elem).show()},hide:function(){this.options.orig[this.prop]=f._data(this.elem,"fxshow"+this.prop)||f.style(this.elem,this.prop),this.options.hide=!0,this.custom(this.cur(),0)},step:function(a){var b,c,d,e=cr||cs(),g=!0,h=this.elem,i=this.options;if(a||e>=i.duration+this.startTime){this.now=this.end,this.pos=this.state=1,this.update(),i.animatedProperties[this.prop]=!0;for(b in i.animatedProperties)i.animatedProperties[b]!==!0&&(g=!1);if(g){i.overflow!=null&&!f.support.shrinkWrapBlocks&&f.each(["","X","Y"],function(a,b){h.style["overflow"+b]=i.overflow[a]}),i.hide&&f(h).hide();if(i.hide||i.show)for(b in i.animatedProperties)f.style(h,b,i.orig[b]),f.removeData(h,"fxshow"+b,!0),f.removeData(h,"toggle"+b,!0);d=i.complete,d&&(i.complete=!1,d.call(h))}return!1}i.duration==Infinity?this.now=e:(c=e-this.startTime,this.state=c/i.duration,this.pos=f.easing[i.animatedProperties[this.prop]](this.state,c,0,1,i.duration),this.now=this.start+(this.end-this.start)*this.pos),this.update();return!0}},f.extend(f.fx,{tick:function(){var a,b=f.timers,c=0;for(;c<b.length;c++)a=b[c],!a()&&b[c]===a&&b.splice(c--,1);b.length||f.fx.stop()},interval:13,stop:function(){clearInterval(cp),cp=null},speeds:{slow:600,fast:200,_default:400},step:{opacity:function(a){f.style(a.elem,"opacity",a.now)},_default:function(a){a.elem.style&&a.elem.style[a.prop]!=null?a.elem.style[a.prop]=a.now+a.unit:a.elem[a.prop]=a.now}}}),f.each(["width","height"],function(a,b){f.fx.step[b]=function(a){f.style(a.elem,b,Math.max(0,a.now)+a.unit)}}),f.expr&&f.expr.filters&&(f.expr.filters.animated=function(a){return f.grep(f.timers,function(b){return a===b.elem}).length});var cw=/^t(?:able|d|h)$/i,cx=/^(?:body|html)$/i;"getBoundingClientRect"in c.documentElement?f.fn.offset=function(a){var b=this[0],c;if(a)return this.each(function(b){f.offset.setOffset(this,a,b)});if(!b||!b.ownerDocument)return null;if(b===b.ownerDocument.body)return f.offset.bodyOffset(b);try{c=b.getBoundingClientRect()}catch(d){}var e=b.ownerDocument,g=e.documentElement;if(!c||!f.contains(g,b))return c?{top:c.top,left:c.left}:{top:0,left:0};var h=e.body,i=cy(e),j=g.clientTop||h.clientTop||0,k=g.clientLeft||h.clientLeft||0,l=i.pageYOffset||f.support.boxModel&&g.scrollTop||h.scrollTop,m=i.pageXOffset||f.support.boxModel&&g.scrollLeft||h.scrollLeft,n=c.top+l-j,o=c.left+m-k;return{top:n,left:o}}:f.fn.offset=function(a){var b=this[0];if(a)return this.each(function(b){f.offset.setOffset(this,a,b)});if(!b||!b.ownerDocument)return null;if(b===b.ownerDocument.body)return f.offset.bodyOffset(b);var c,d=b.offsetParent,e=b,g=b.ownerDocument,h=g.documentElement,i=g.body,j=g.defaultView,k=j?j.getComputedStyle(b,null):b.currentStyle,l=b.offsetTop,m=b.offsetLeft;while((b=b.parentNode)&&b!==i&&b!==h){if(f.support.fixedPosition&&k.position==="fixed")break;c=j?j.getComputedStyle(b,null):b.currentStyle,l-=b.scrollTop,m-=b.scrollLeft,b===d&&(l+=b.offsetTop,m+=b.offsetLeft,f.support.doesNotAddBorder&&(!f.support.doesAddBorderForTableAndCells||!cw.test(b.nodeName))&&(l+=parseFloat(c.borderTopWidth)||0,m+=parseFloat(c.borderLeftWidth)||0),e=d,d=b.offsetParent),f.support.subtractsBorderForOverflowNotVisible&&c.overflow!=="visible"&&(l+=parseFloat(c.borderTopWidth)||0,m+=parseFloat(c.borderLeftWidth)||0),k=c}if(k.position==="relative"||k.position==="static")l+=i.offsetTop,m+=i.offsetLeft;f.support.fixedPosition&&k.position==="fixed"&&(l+=Math.max(h.scrollTop,i.scrollTop),m+=Math.max(h.scrollLeft,i.scrollLeft));return{top:l,left:m}},f.offset={bodyOffset:function(a){var b=a.offsetTop,c=a.offsetLeft;f.support.doesNotIncludeMarginInBodyOffset&&(b+=parseFloat(f.css(a,"marginTop"))||0,c+=parseFloat(f.css(a,"marginLeft"))||0);return{top:b,left:c}},setOffset:function(a,b,c){var d=f.css(a,"position");d==="static"&&(a.style.position="relative");var e=f(a),g=e.offset(),h=f.css(a,"top"),i=f.css(a,"left"),j=(d==="absolute"||d==="fixed")&&f.inArray("auto",[h,i])>-1,k={},l={},m,n;j?(l=e.position(),m=l.top,n=l.left):(m=parseFloat(h)||0,n=parseFloat(i)||0),f.isFunction(b)&&(b=b.call(a,c,g)),b.top!=null&&(k.top=b.top-g.top+m),b.left!=null&&(k.left=b.left-g.left+n),"using"in b?b.using.call(a,k):e.css(k)}},f.fn.extend({position:function(){if(!this[0])return null;var a=this[0],b=this.offsetParent(),c=this.offset(),d=cx.test(b[0].nodeName)?{top:0,left:0}:b.offset();c.top-=parseFloat(f.css(a,"marginTop"))||0,c.left-=parseFloat(f.css(a,"marginLeft"))||0,d.top+=parseFloat(f.css(b[0],"borderTopWidth"))||0,d.left+=parseFloat(f.css(b[0],"borderLeftWidth"))||0;return{top:c.top-d.top,left:c.left-d.left}},offsetParent:function(){return this.map(function(){var a=this.offsetParent||c.body;while(a&&!cx.test(a.nodeName)&&f.css(a,"position")==="static")a=a.offsetParent;return a})}}),f.each(["Left","Top"],function(a,c){var d="scroll"+c;f.fn[d]=function(c){var e,g;if(c===b){e=this[0];if(!e)return null;g=cy(e);return g?"pageXOffset"in g?g[a?"pageYOffset":"pageXOffset"]:f.support.boxModel&&g.document.documentElement[d]||g.document.body[d]:e[d]}return this.each(function(){g=cy(this),g?g.scrollTo(a?f(g).scrollLeft():c,a?c:f(g).scrollTop()):this[d]=c})}}),f.each(["Height","Width"],function(a,c){var d=c.toLowerCase();f.fn["inner"+c]=function(){var a=this[0];return a?a.style?parseFloat(f.css(a,d,"padding")):this[d]():null},f.fn["outer"+c]=function(a){var b=this[0];return b?b.style?parseFloat(f.css(b,d,a?"margin":"border")):this[d]():null},f.fn[d]=function(a){var e=this[0];if(!e)return a==null?null:this;if(f.isFunction(a))return this.each(function(b){var c=f(this);c[d](a.call(this,b,c[d]()))});if(f.isWindow(e)){var g=e.document.documentElement["client"+c],h=e.document.body;return e.document.compatMode==="CSS1Compat"&&g||h&&h["client"+c]||g}if(e.nodeType===9)return Math.max(e.documentElement["client"+c],e.body["scroll"+c],e.documentElement["scroll"+c],e.body["offset"+c],e.documentElement["offset"+c]);if(a===b){var i=f.css(e,d),j=parseFloat(i);return f.isNumeric(j)?j:i}return this.css(d,typeof a=="string"?a:a+"px")}}),a.jQuery=a.$=f,typeof define=="function"&&define.amd&&define.amd.jQuery&&define("jquery",[],function(){return f})})(window); - diff --git a/lsp/main.css b/lsp/main.css index 8a5805eb..1a203bf7 100755 --- a/lsp/main.css +++ b/lsp/main.css @@ -3,7 +3,7 @@ body { padding: 0px; width: 100%; height: 100%; - font-family: sans-serif; + font-family: FuturaICGLight, sans-serif; } #shield { position: absolute; @@ -31,17 +31,16 @@ body { margin-left: 25px; padding: 3px 0px 0px 60px; height: 66px; - width: 190px; font-family: FuturaICGLight, sans-serif; font-weight: normal; - text-transform: uppercase; + font-size: 2em; background-image: url(''); background-repeat: no-repeat; } #logo > a { text-decoration: none; - font-size: 1.2em; color: #c7c7c7; + cursor: pointer; } #logo > a > span { color: #9cc1db; @@ -68,15 +67,38 @@ body { } #message { position: absolute; - bottom: 10px; + bottom: 0; + font-family: sans-serif; font-size: 0.8em; - color: #c7c7c7; + color: #777; } #message > a { text-decoration: underline; color: #505050; cursor: pointer; } +#ih-button, #ih-button.active:hover { + position: absolute; + top: 108px; + right: 1em; + width: 1.2em; + line-height: 1.2em; + height: 1.2em; + font-weight: bold; + border: 2px solid #bcbcbc; + border-radius: 2em; + text-align: center; + background-color: #ededed; + color: white; + box-shadow: 2px 2px 5px #ededed; + text-shadow: 1px 1px 2px #6696ce; + cursor: pointer; +} +#ih-button.active, #ih-button:hover { + border-color: #6696ce; + background-color: #9cc1db; + text-shadow: 1px 1px 2px #6696ce; +} #menu { position: fixed; top: 100px; @@ -88,25 +110,54 @@ body { background-color: #ededed; border-right: 1px solid #bcbcbc; } -#menu > .button { +.menu .button, +.menu .expandbutton { color: #333; margin: 5px auto 5px 13px; padding: 0px 15px 0px 15px; height: 31px; line-height: 30px; - width: 167px; cursor: pointer; display: block; text-decoration: none; + background-position: 100% 0; } -#menu > .button:hover { +#menu .button { + width: 167px; +} +.menu .button:hover { color: #000; background-image: url(); } -#menu > .button.current { +.menu .button.current { color: #fff; background-image: url(); } +.arrowdown { + width: 0; + height: 0; + display: inline-block; + border: 4px solid transparent; + border-top-color: #333; + border-bottom-width: 2px; +} +.menu .expandbutton .expandcontainer { + display: none; +} +.menu .expandbutton.active .expandcontainer, +.menu .expandbutton:hover .expandcontainer { + display: block; +} +.menu .expandbutton { + margin-bottom: 0; +} +.menu .expandcontainer .button { + margin-top: 0; + width: 149px; +} +.menu .expandcontainer { + margin-left: -10px; +} #page { position: fixed; top: 100px; @@ -114,17 +165,20 @@ body { left: 230px; right: 0px; overflow: auto; - padding: 10px 20px 10px 20px; + padding: 1em; } .description { font-size: 0.9em; color: #777; clear: both; text-align: justify; + width: 40em; + clear: none; } .input_container { margin-top: 25px; - width: 75%; + width: 40em; + overflow: hidden; } label, .pretend-label { width: 100%; @@ -137,13 +191,18 @@ label, .pretend-label { label > input, label > select, label > textarea, .pretend-label > input, .pretend-label > select, .pretend-label > textarea { float: right; + clear: right; width: 60%; - padding: 5px; - margin: 2px 0px 2px -8px; + padding: 0.3em; + margin: 0 0 0 -0.5em; color: #505050; border: 1px solid #b4b4b4; - height: 28px; + height: 2.3em; background-color: #fff; + font-size: 0.8em; +} +input[disabled], label > select[disabled], label > textarea[disabled] { + opacity: 0.5; } label > textarea, .pretend-label > textarea { height: 100px; @@ -166,24 +225,26 @@ label > span, .pretend-label > span { line-height: 16px; font-size: 0.8em; } -button { +button, .fakebutton { font-weight: bold; font-size: 1em; - height: 30px; + height: 1.8em; + line-height: 1.8em; background-color: #505050; color: #fff; border: none; - margin: 5px 0px 5px 5px; - padding: 0px 10px 0px 10px; + margin: 0.3em 0 0.3em 0.3em; + padding: 0 0.5em 0px 0.5em; cursor: pointer; float: right; + text-decoration: none; } -label > span > button, .pretend-label > span > button { - margin-top: -8px; - margin-bottom: -8px; +label > span > button, .pretend-label > span > button, +label > span > .fakebutton, .pretend-label > span > .fakebutton { + margin-top: -0.5em; + margin-bottom: -0.5em; } -table button { - height: 25px; +table button, table .fakebutton { float: none; } .red { @@ -221,6 +282,7 @@ td,th { padding: 5px; } td { + font-family: sans-serif; font-size: 0.8em; } tr:nth-child(even) { @@ -242,14 +304,83 @@ tr:nth-child(odd) { .sortable th.sortdesc { background-image: url(); } +/* negate the tables in the flot legend.. */ +.legend table { + width: auto; + margin: auto; +} +.legend tr { + background-color: rgba(255,255,255,0.5) +} + label > table, .pretend-label > table { float: right; width: 60%; } -label > .unit, .pretend-label > unit { +label > .unit, .pretend-label > .unit { width: auto; position: absolute; - left: 75%; + left: 41em; + padding-left: 0.2em; + font-size: 1em; +} +.ih-balloon { + position: absolute; + background-color: #ededed; + border-radius: 0.5em; + border: 0.1em solid #bcbcbc; + padding: 0 0.5em; + text-align: justify; + line-height: normal; +} +.ih-balloon.pageinfo { + margin: 2em 0 1em 1em; + float: right; + position: relative; + width: 20em; +} +.ih-balloon.inputinfo { + left: 45em; + margin-right: 1em; +} +.ih-balloon:before, .ih-balloon:after { + content: '.'; + color: transparent; + width: 0; + height: 0; + position: absolute; + border: 0.6em solid transparent; +} +.ih-balloon.pageinfo:before, .ih-balloon.pageinfo:after { + right: 0.5em; +} +.ih-balloon.inputinfo:before, .ih-balloon.inputinfo:after { + top: 0.5em; +} +.ih-balloon.pageinfo:before { + border-bottom-width: 0.7em; + border-bottom-color: #bcbcbc; + top: -1.25em; +} +.ih-balloon.pageinfo:after { + border-bottom-color: #ededed; + top: -1.1em; +} +.ih-balloon.inputinfo:before { + border-right-width: 0.7em; + border-right-color: #bcbcbc; + left: -1.25em; +} +.ih-balloon.inputinfo:after { + border-right-color: #ededed; + left: -1.1em; +} + +.ih-balloon p { + font-weight: normal; +} +.ih-balloon * { + font-size: 0.9em; } #input-validation-info { font-size: 0.8em; @@ -257,4 +388,106 @@ label > .unit, .pretend-label > unit { float: right; width: 60%; margin-right: -5px; +} +#tooltip { + display: none; + position: absolute; + z-index: 1337; + border: 1px solid #bcbcbc; + padding: 0.5em; + background-color: #ededed; +} +.checklist label { + width: auto; + height: auto; + min-height: 0; + padding: 0; +} +.checklist input[type=radio], +.checklist input[type=checkbox] { + width: 1em; + height: 1em; + float: none; + margin: 0; + margin-right: 0.5em; + vertical-align: baseline; +} +.LTS-only.LTSstuff_done, +#menu .LTS-only.LTSstuff_done, +.LTS-only.LTSstuff_done p, +.LTS-only.LTSstuff_done label { + color: #b4b4b4; +} +button[disabled=disabled] { + background-color: #a0a0a0; +} +.LTS-only.LTSstuff_done table { + opacity: 0.5; +} +.table { + display: table; +} +.row { + display: table-row; +} +.cell { + display: table-cell; +} +#graphcontainer { + width: 100%; + background-color: white; + display: -webkit-flex; + display: -moz-flex; + display: -ms-flex; + display: flex; + -webkit-flex-flow: row wrap; + -moz-flex-flow: row wrap; + -ms-flex-flow: row wrap; + flex-flow: row wrap; +} +#graphcontainer .graph-item { + -webkit-flex: 1 1 auto; + -moz-flex: 1 1 auto; + -ms-flex: 1 1 auto; + flex: 1 1 auto; + width: 400px; + border: 1px solid #ccc; + margin: 0.1em; + padding: 0 0.5em; +} +#graphcontainer .graph { + height: 250px; +} +.legend { + font-size: 0.8em; +} +.legend-list { + -webkit-column-width: 15em; + -moz-column-width: 15em; + column-width: 15em; +} +.series-color { + width: 1em; + height: 1em; + display: inline-block; + margin: 0 0.5em; +} + +@font-face { + font-family: FuturaICGLight; + src: url(data:font/ttf;base64,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) format('truetype'); +} +@font-face { + font-family: FuturaICGLight; + font-weight: bold; + src: url(data:font/ttf;base64,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BxEAfQciAGkCKgBuAtYAdwLPAHADewBiBNwAbgMeAFcECgBqAqoAZwUAAFsF7QB2Be0BxgXtAJQF7QCUBe0AbgXtANMF7QB7Be0AdAXtALcF7QCFAqEAYANfAF0EqgBuBOQAYQSqAGkFDgBzBpoAZwa3ADYFXQCUBWwAcQYDAJAEPQBlBEUAoQcDAG0GPQBtAv4AvgRbACgGUgCZBDcAjwf0ADIGzQBpB1AAbgVeAIoHiQBrBbsAjwVBAG8E+QByBhEAdgbFAD0JgAA5BuMAPAZwAEEGPQBuAsgAawUGAG4CzwBvBEUAZAUOAAkC1gB3BbMAZQWIAI0EGgB2BZAAOwVIAGUD2gBuBYgAQAUjAHMC3ACdAuEApAXYAKECswCyB7sAcwUrAG8FwgByBZgAogWQADsD6ACZBGMAawODAGYFIwBpBacAPQivAFcGWgBGBe8ARQUdAGcDCABpAfcAZgMIAGsD1QBOBTUAAAKdAGQEDABwBf0AdgZwAEEFXQBzBGkAagbAAG4DowB5BLwAaQa/AHcECgBqBr8AbgKTAFYEFgBfAsoAXwWBAG4DwgBtBMIAbQUPAG4GxAA2BrEANga8ADYGygA2BrwANgbYAD0IOQB4BWwAbgQtAGUEQwBlBCEAZAQhAEgC/wClAwEAuQMI/+gDBv/JBtgAaQddAG4HXQBuB10AbgdfAG4HbwBuB5UAbQYbAHYGEAB2BhsAdgYIAFkFpgBzBaMAZQWrAHMFowBlBZ0AZgWjAGUFngBlCNIAZQQMAGoFRgBlBUYAZQVMAGUFOABlAusAnALdAH4C6v/vAuj/wQUpAH0FsQBXBasAVwW5AGUFwQBlBbkAVwRqAHMFwgBzBSIAaQUaAGkFIgBpBSIAaQYAAGEJLgBiCKkAXAZ3AEEESAAACJAAAARIAAAIkAAAAtoAAAIkAAABbQAAAW0AAAESAAABtgAAAHkAAAQKAGoECgBqBAoAagYEAG4IWwBtAywAdgM7AHIDOwByBToAbQUzAHsFMwB7A9IAbwdyAGcBtgAAAiQAAAXrAGsEHgAAAAAAAAAAAAAAAABIAIABaAHgAowDHgM+A2oDmAPOBBAEMARKBIYElgTqBRAFYAXOBhoGfgbkBwQHbAfACBwIXghyCJQIqAkmCgQKNgqiCwALRguGC7wMIAxYDHYMsgzmDQwNZg2iDfoOTA7O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) format('truetype'); +} + +#header { + background-image: linear-gradient(to right, rgba(255,255,255,0.8) 40%, rgba(255,255,255,0)), + url(); + background-color: #9ec2da; + background-position: 0 0, 100% 0; + background-size: auto, 1060px; + background-repeat: no-repeat; } \ No newline at end of file diff --git a/lsp/main.js b/lsp/main.js index 8ebebe90..3f7b7630 100755 --- a/lsp/main.js +++ b/lsp/main.js @@ -14,6 +14,8 @@ function consolelog() { } } +var ih = false; + function confirmDelete(question){ return confirm(question); } @@ -56,6 +58,151 @@ function formatTime(date){ ('00' + d.getSeconds()).slice(-2) ].join(':'); } +/** + * Format a time duration to something like "2 days, 00:00:00.000" + * @param ms the duration to format in miliseconds + */ +function formatDuration(ms) { + var secs = Math.floor(ms / 1000), mins = 0; + ms = ms % 1000; + if (secs >= 60) { + mins = Math.floor(secs / 60); + secs = secs % 60; + } + if (mins >= 60) { + var hours = Math.floor(mins / 60); + mins = mins % 60; + } + var string = ('00'+mins).slice(-2)+':'+('00'+secs).slice(-2)+'.'+('000'+ms).slice(-3); + if (hours >= 24) { + var days = Math.floor(hours / 24); + hours = hours % 24; + } + if (hours > 0) { + string = ('00'+hours).slice(-2)+':'+string; + } + if (days > 0) { + string = days+' day'+(days > 1 ? 's' : '')+', '+string + } + return string; +} +/** + * Capitalize the first letter + * @param string the string + */ +function capFirstChar(string) { + if (string.length <= 0) { return ''; } + return string[0].toUpperCase() + string.slice(1); +} +/** + * Flot tick generator for bandwidth + * @param axis the axis + */ +function flotTicksBandwidthAxis(axis) { + var range = axis.max - axis.min; + var delta = range / 4; + var start = axis.min; + if (axis.max < 1024) { // unit: bytes/s + if (delta > 100) { delta = Math.floor(delta/100)*100; start = Math.floor(start/100)*100; } // to lowest 100 bytes/s + else if (delta > 10) { delta = Math.floor(delta/10)*10; start = Math.floor(start/10)*10; } // to lowest 10 bytes/s + } + else if (axis.max < 1048576) { //unit: kiB/s + if (delta > 102400) { delta = Math.floor(delta/102400)*102400; start = Math.floor(start/102400)*102400; } //to lowest 100 kiB/s + else if (delta > 10240) { delta = Math.floor(delta/10240)*10240; start = Math.floor(start/10240)*10240; } //to lowest 10 kiB/s + else if (delta > 1024) { delta = Math.floor(delta/1024)*1024; start = Math.floor(start/1024)*1024; } //to lowest 1 kiB/s + else { delta = Math.floor(delta/102.4)*102.4; start = Math.floor(start/102.4)*102.4; } //to lowest 0.1 kiB/s + } + else { //unit: miB/s + if (delta > 104857600) { delta = Math.floor(delta/104857600)*104857600; start = Math.floor(start/104857600)*104857600; } //to lowest 100 miB/s + else if (delta > 10485760) { delta = Math.floor(delta/10485760)*10485760; start = Math.floor(start/10485760)*10485760; } //to lowest 10 miB/s + else if (delta > 1048576) { delta = Math.floor(delta/1048576)*1048576; start = Math.floor(start/1048576)*1048576; } //to lowest 1 miB/s + else { delta = Math.floor(delta/104857.6)*104857.6; start = Math.floor(start/104857.6)*104857.6; } //to lowest 0.1 miB/s + } + var out = []; + for (var i = start; i <= axis.max; i += delta) { + out.push(i); + } + return out; +} +/** + * Flot axis formatter for bandwidth + * @param val the valuea + * @param axis the axis + */ +function flotFormatBandwidthAxis(val,axis) { + if (val < 0) { var sign = '-'; } + else { var sign = ''; } + val = Math.abs(val); + + if (val < 1024) { return sign+Math.round(val)+' bytes/s'; } // 0 bytes/s through 1023 bytes/s + if (val < 10235) { return sign+(val/1024).toFixed(2)+' kiB/s'; } // 1.00 kiB/s through 9.99 kiB/s + if (val < 102449) { return sign+(val/1024).toFixed(1)+' kiB/s'; } // 10.0 kiB/s through 99.9 kiB/s + if (val < 1048064) { return sign+Math.round(val/1024)+' kiB/s'; } // 100 kiB/s through 1023 kiB/s + if (val < 10480518) { return sign+(val/1048576).toFixed(2)+' miB/s'; } // 1.00 miB/s through 9.99 miB/s + if (val < 104805172) { return sign+(val/1048576).toFixed(1)+' miB/s'; } // 10.0 miB/s through 99.9 miB/s + return sign+Math.round(val/1048576)+' miB/s'; // 100 miB/s and up +} +/** + * Converts the statistics data into something flot understands + * @param stats the statistics.totals object + * @param cumulative cumulative mode if true + */ +function convertStatisticsToFlotFormat(stats,islive) { + var plotdata = [ + { label: 'Viewers', data: []}, + { label: 'Bandwidth (Up)', data: [], yaxis: 2}, + { label: 'Bandwidth (Down)', data: [], yaxis: 2} + ]; + + var oldtimestamp = 0; + var i = 0, up = 0, down = 0; + for (var timestamp in stats) { + if (islive) { + i++; + up += stats[timestamp].up; + down += stats[timestamp].down; + //average over 5 seconds to prevent super spiky unreadable graph + if ((i % 5) == 0) { + plotdata[0].data.push([Number(timestamp)*1000,stats[timestamp].count]); + plotdata[1].data.push([Number(timestamp)*1000,up/5]); + plotdata[2].data.push([Number(timestamp)*1000,down/5]); + up = 0; + down = 0; + } + } + else { + var dt = timestamp - oldtimestamp; + if (stats[oldtimestamp]) { + var up = (stats[timestamp].up - stats[oldtimestamp].up)/dt; + var down = (stats[timestamp].down - stats[oldtimestamp].down)/dt; + } + else { + var up = stats[timestamp].up; + var down = stats[timestamp].down; + } + plotdata[0].data.push([Number(timestamp)*1000,stats[timestamp].count]); + plotdata[1].data.push([Number(timestamp)*1000,up]); + plotdata[2].data.push([Number(timestamp)*1000,down]); + oldtimestamp = timestamp; + } + } + for (var timestamp in stats) { + var dt = timestamp - oldtimestamp; + plotdata[0].data.push([Number(timestamp)*1000,stats[timestamp].count]); + if (stats[oldtimestamp]) { + var up = (stats[timestamp].up - stats[oldtimestamp].up)/dt; + var down = (stats[timestamp].down - stats[oldtimestamp].down)/dt; + } + else { + var up = stats[timestamp].up; + var down = stats[timestamp].down; + } + plotdata[1].data.push([Number(timestamp)*1000,up]); + plotdata[2].data.push([Number(timestamp)*1000,down]); + oldtimestamp = timestamp; + } + return plotdata; +} /** * Check if an URL points to a live datastream or a recorded file * @param url the url in question @@ -136,9 +283,10 @@ function applyInput(){ //apply the inputs $('input.isSetting,select.isSetting').each(function(){ + var objpath = findObjPath($(this)); - if ($(this).val() == '') { + if (($(this).val() == '') || ($(this).val() == 0)) { eval('delete '+objpath+';'); } else { @@ -163,6 +311,37 @@ function findObjPath($element) { } } +function ihAddBalloons() { + var page = settings.ih.pages[settings.currentpage]; + if (!page) { return; } + + //something with pageinfo + if (page.pageinfo) { + $('#page').prepend( + $('<div>').addClass('ih-balloon').addClass('pageinfo').html(page.pageinfo) + ); + } + + for (inputid in page.inputs) { + $('#'+inputid).parent().prepend( + $('<div>').addClass('ih-balloon').addClass('inputinfo').attr('data-for',inputid).html(page.inputs[inputid]).hide() + ); + $('#'+inputid).focus(function(){ + $('.ih-balloon[data-for='+$(this).attr('id')+']').show(); + $('.ih-balloon.pageinfo').hide(); + }).blur(function(){ + $('.ih-balloon[data-for='+$(this).attr('id')+']').hide(); + $('.ih-balloon.pageinfo').show(); + }); + } + $('#page label').each(function(){ + $(this) + }); +} +function ihMakeBalloon(contents,forid) { + return $('<div>').addClass('ih-balloon').attr('data-for',forid).html(contents).hide(); +} + function getData(callBack,sendData,timeOut,doShield){ timeOut = timeOut | 30000; var data = {}; @@ -283,6 +462,81 @@ function getData(callBack,sendData,timeOut,doShield){ var jqxhr = $.ajax(obj); } + +function getWikiData(url,callBack) { + var wikiHost = 'http://rework.mistserver.org'; //must be changed when rework goes live + + $('#message').removeClass('red').text('Connecting to the MistServer wiki..').append( + $('<br>') + ).append( + $('<a>').text('Cancel request').click(function(){ + jqxhr.abort(); + }) + ); + + var obj = { + 'url': wikiHost+url, + 'type': 'GET', + 'crossDomain': true, + 'data': { + 'skin': 'plain' + }, + 'error':function(jqXHR,textStatus,errorThrown){ + switch (textStatus) { + case 'timeout': + textStatus = $('<i>').text('The connection timed out. '); + break; + case 'abort': + textStatus = $('<i>').text('The connection was aborted. '); + break; + default: + textStatus = $('<i>').text(textStatus+'. ').css('text-transform','capitalize'); + } + $('#message').addClass('red').text('An error occurred while attempting to communicate with the MistServer wiki:').append( + $('<br>') + ).append( + textStatus + ).append( + $('<a>').text('Send server request again').click(function(){ + getWikiData(url,callback); + }) + ); + }, + 'success': function(returnedData){ + $('#message').text('Wiki data received'); + + //convert to DOM elements + //returnedData = $.parseHTML(returnedData); + returnedData = $(returnedData); + + //fix broken slash-links in the imported data + returnedData.find('a[href]').each(function(){ + if ((this.hostname == '') || (this.hostname == undefined)) { + $(this).attr('href',wikiHost+$(this).attr('href')); + } + if (!$(this).attr('target')) { + $(this).attr('target','_blank'); + } + }).find('img[src]').each(function(){ + var a = $('<a>').attr('href',$(this).attr('src')); + if ((a.hostname == '') || (a.hostname == undefined)) { + $(this).attr('src',wikiHost+$(this).attr('src')); + } + }); + + consolelog('['+(new Date).toTimeString().split(' ')[0]+']','Received wiki data:',returnedData); + + if (callBack) { + callBack(returnedData); + } + $('#message').text('Last communication with the MistServer wiki at '+formatTime((new Date).getTime()/1000)); + + } + }; + + var jqxhr = $.ajax(obj); +} + function saveAndReload(tabName){ var sendData = $.extend(true,{},settings.settings); delete sendData.logs; @@ -345,12 +599,8 @@ function updateOverview() { var streams = 0; var streamsOnline = 0; - for (var index in data.statistics) { - if (data.statistics[index].curr) { - for (viewer in data.statistics[index].curr) { - viewers++; - } - } + if (data.clients && data.clients.data) { + viewers = data.clients.data.length; } for (var index in data.streams) { @@ -363,7 +613,9 @@ function updateOverview() { $('#cur_streams_online').text(streamsOnline+'/'+streams+' online'); $('#cur_num_viewers').text(seperateThousands(viewers,' ')); $('#settings-config-time').text(formatDateLong(data.config.time)); - }); + + settings.settings.statistics = data.statistics; + },{clients: {}}); } function updateProtocols() { getData(function(data){ @@ -380,6 +632,9 @@ function updateProtocols() { function displayProtocolSettings(theProtocol) { var capabilities = settings.settings.capabilities.connectors[theProtocol.connector]; + if (!capabilities) { + return ''; + } var settingsList = []; for (var index in capabilities.required) { if ((theProtocol[index]) && (theProtocol[index] != '')) { @@ -478,20 +733,31 @@ function buildProtocolParameterFields(data,required,objpath) { return $container.html(); } function updateStreams() { + var streamlist = []; + for (var stream in settings.settings.streams) { + streamlist.push(stream); + } getData(function(data){ + var datafields = {}; + for (var index in data.clients.fields) { + datafields[data.clients.fields[index]] = index; + } + var viewers = {}; + for (var index in data.clients.data) { + if (viewers[data.clients.data[index][datafields['stream']]]) { + viewers[data.clients.data[index][datafields['stream']]]++; + } + else { + viewers[data.clients.data[index][datafields['stream']]] = 1; + } + } for (var index in data.streams) { $('#status-of-'+index).html(formatStatus(data.streams[index])) + $('#viewers-of-'+index).text(seperateThousands(viewers[index],' ')); } - for (var index in data.statistics) { - var viewers = 0; - if (data.statistics[index].curr) { - for (var jndex in data.statistics[index].curr) { - viewers++; - } - } - $('#viewers-of-'+index).text(seperateThousands(viewers,' ')); - } - }); + + settings.settings.statistics = data.statistics; + },{clients:{}}); } function filterTable() { var displayRecorded = $('#stream-filter-recorded').is(':checked'); @@ -886,12 +1152,19 @@ function conversionSelectInput(theFiles) { applyInput(); + var extension = settings.settings.conversion.convert._new_.output.split('.'); + if (extension[extension.length-1] != 'dtsc') { + extension.push('dtsc'); + settings.settings.conversion.convert._new_.output = extension.join('.'); + } settings.settings.conversion.convert._new_.output = settings.settings.conversion.convert._new_.outputdir.replace(/\/$/,'')+'/'+settings.settings.conversion.convert._new_.output; delete settings.settings.conversion.convert._new_.outputdir; if ((settings.settings.conversion.convert._new_.video) && (settings.settings.conversion.convert._new_.video.fps)) { settings.settings.conversion.convert._new_.fpks = Math.floor(settings.settings.conversion.convert._new_.fps * 1000); } + + settings.settings.conversion.convert['c_'+(new Date).getTime()] = settings.settings.conversion.convert._new_; delete settings.settings.conversion.convert._new_; saveAndReload('conversion'); @@ -1019,10 +1292,85 @@ function updateServerstats() { },{capabilities:true}); } +function buildstreamembed(streamName,embedbase) { + $('#liststreams .button.current').removeClass('current') + $('#liststreams .button').filter(function(){ + return $(this).text() == streamName; + }).addClass('current'); + + $('#subpage').append( + $('<div>').addClass('input_container').html( + $('<label>').text('The info embed URL is:').append( + $('<input>').attr('type','text').attr('readonly','readonly').val(embedbase+'info_'+streamName+'.js') + ) + ).append( + $('<label>').text('The embed URL is:').append( + $('<input>').attr('type','text').attr('readonly','readonly').val(embedbase+'embed_'+streamName+'.js') + ) + ).append( + $('<label>').text('The embed code is:').css('overflow','hidden').append( + $('<textarea>').val('<div>\n <script src="'+embedbase+'embed_'+streamName+'.js"></' + 'script>\n</div>') + ) + ) + ).append( + $('<span>').attr('id','listprotocols').text('Loading..') + ).append( + $('<p>').text('Preview:') + ).append( + $('<div>').attr('id','preview-container') + ); + + // jQuery doesn't work -> use DOM magic + var script = document.createElement('script'); + script.src = embedbase+'embed_'+streamName+'.js'; + script.onload = function(){ + var priority = mistvideo[streamName].source; + if (priority.length > 0) { + priority.sort(function(a,b){ + return b.priority - a.priority; + }); + var $table = $('<table>').html( + $('<tr>').html( + $('<th>').text('URL') + ).append( + $('<th>').text('Type') + ).append( + $('<th>').text('Priority') + ) + ); + for (var i in priority) { + $table.append( + $('<tr>').html( + $('<td>').text(priority[i].url) + ).append( + $('<td>').text(priority[i].type) + ).append( + $('<td>').addClass('align-center').text(priority[i].priority) + ) + ); + } + $('#listprotocols').html($table); + } + else { + $('#listprotocols').html('No data in info embed file.'); + } + } + document.getElementById('preview-container').appendChild( script ); +} + $(function(){ - $('#menu div.button').click(function(){ - if ((settings.settings.LTS != 1) && ($(this).hasClass('LTS-only'))) { return; } + $('#logo > a').click(function(){ + if ($.isEmptyObject(settings.settings)) { + showTab('login') + } + else { + showTab('overview'); + } + }); + $('#menu div.button').click(function(e){ + //if ((settings.settings.LTS != 1) && ($(this).hasClass('LTS-only'))) { return; } showTab($(this).text().toLowerCase()); + e.stopPropagation(); }) $('body').on('keydown',function(e){ switch (e.which) { @@ -1101,6 +1449,41 @@ $(function(){ $(this).val(v); this.setSelectionRange(curpos,curpos); }); + + $('.expandbutton').click(function(){ + $(this).toggleClass('active'); + }); + + + $('#ih-button').click(function(){ + if (ih) { + $('.ih-balloon').remove(); + } + else { + getWikiData('/wiki/Integrated_Help',function(data){ + settings.ih = { + raw: data.find('#mw-content-text').contents(), + pages: {} + } + settings.ih.raw.filter('.page[data-pagename]').each(function(){ + var pagename = $(this).attr('data-pagename').replace(' ','_'); + settings.ih.pages[pagename] = { + raw: $(this).contents(), + pageinfo: $(this).find('.page-description').html(), + inputs: {} + } + $(this).children('.input-description[data-inputid]').each(function(){ + settings.ih.pages[pagename].inputs[$(this).attr('data-inputid')] = $(this).html(); + }); + }); + consolelog('New integrated help data:',settings.ih); + ihAddBalloons(); + }); + } + ih = !ih; + $(this).toggleClass('active'); + }); + }); $(window).on('hashchange', function(e) { diff --git a/lsp/pages.js b/lsp/pages.js index b43c4296..424a1469 100755 --- a/lsp/pages.js +++ b/lsp/pages.js @@ -6,6 +6,7 @@ var defaults = { }; function showTab(tabName,streamName) { + settings.currentpage = tabName.replace(' ','_'); ignoreHashChange = true; location.hash = location.hash.split('&')[0]+'&'+tabName+(streamName ? '@'+streamName : ''); @@ -14,9 +15,10 @@ function showTab(tabName,streamName) { $('#menu .button').removeClass('current').filter(function(i){ return $(this).text().toLowerCase() == tabName; - }).addClass('current'); + }).addClass('current').parents('.expandbutton').addClass('active'); $('#page').html(''); + $('#tooltip').remove(); clearInterval(theInterval); $('#menu').css('visibility', 'visible'); @@ -121,6 +123,10 @@ function showTab(tabName,streamName) { saveAndReload('overview'); }) + ).append( + $('<button>').text('Cancel').addClass('escape-to-cancel').click(function(){ + showTab('login'); + }) ) ); break; @@ -434,8 +440,12 @@ function showTab(tabName,streamName) { $('<td>').attr('id','status-of-'+index).html(formatStatus(theStream)) ).append( $('<td>').html( - $('<button>').text('Embed').click(function(){ - showTab('embed',$(this).parent().parent().data('stream')) + $('<button>').text('Preview').click(function(){ + showTab('preview',$(this).parent().parent().data('stream')) + }) + ).append( + $('<button>').text('Info').click(function(){ + showTab('streaminfo',$(this).parent().parent().data('stream')) }) ) ).append( @@ -497,7 +507,7 @@ function showTab(tabName,streamName) { $('<label>').text('Buffer time:').addClass('live-only').attr('for','settings-streams-'+streamName+'-DVR').append( $('<span>').addClass('unit').text('[ms]') ).append( - $('<input>').attr('type','text').attr('id','settings-streams-'+streamName+'-DVR').attr('placeholder','2 keyframes').addClass('isSetting').addClass('').addClass('validate-positive-integer') + $('<input>').attr('type','text').attr('id','settings-streams-'+streamName+'-DVR').attr('placeholder','30000').addClass('isSetting').addClass('').addClass('validate-positive-integer') ) ).append( $('<label>').text('Record to:').addClass('live-only').addClass('LTS-only').attr('for','settings-streams-'+streamName+'-record').attr('title','The path to the file to record to. Leave this field blank if you do not wish to record to file.').append( @@ -516,7 +526,7 @@ function showTab(tabName,streamName) { $('<p>').text('Encrypt this stream') ).append( $('<div>').addClass('description').text( - 'To enable encryption, the Licene Acquisition URL must be entered, as well as either the content key or the key ID and seed.' + 'To enable encryption, the Licence Acquisition URL must be entered, as well as either the content key or the key ID and seed.' ) ).append( $('<label>').text('Licence Acquisition URL:').attr('for','settings-streams-'+streamName+'-la_url').append( @@ -578,7 +588,131 @@ function showTab(tabName,streamName) { }) break; - case 'embed': + case 'streaminfo': + var meta = settings.settings.streams[streamName].meta; + if (!meta) { + $('#page').html('No info available for stream "'+streamName+'".'); + } + else { + $meta = $('<table>').css('width','auto'); + if (meta.live) { + $meta.html( + $('<tr>').html( + $('<td>').text('Type:') + ).append( + $('<td>').text('Live') + ) + ); + } + else { + $meta.html( + $('<tr>').html( + $('<td>').text('Type:') + ).append( + $('<td>').text('Pre-recorded (VoD)') + ) + ); + } + for (var index in meta.tracks) { + var track = meta.tracks[index]; + if (track.type == '') { continue; } + var $table = $('<table>').html( + $('<tr>').html( + $('<td>').text('Type:') + ).append( + $('<td>').text(capFirstChar(track.type)) + ) + ).append( + $('<tr>').html( + $('<td>').text('Codec:') + ).append( + $('<td>').text(track.codec) + ) + ).append( + $('<tr>').html( + $('<td>').text('Duration:') + ).append( + $('<td>').html( + formatDuration(track.lastms-track.firstms)+'<br>(from '+formatDuration(track.firstms)+' to '+formatDuration(track.lastms)+')' + ) + ) + ).append( + $('<tr>').html( + $('<td>').text('Average bitrate:') + ).append( + $('<td>').text(Math.round(track.bps/1024)+' KiB/s') + ) + ); + + if (track.height) { + $table.append( + $('<tr>').html( + $('<td>').text('Size:') + ).append( + $('<td>').text(track.width+'x'+track.height+' px') + ) + ); + } + if (track.fpks) { + $table.append( + $('<tr>').html( + $('<td>').text('Framerate:') + ).append( + $('<td>').text(track.fpks/1000+' fps') + ) + ); + } + if (track.channels) { + $table.append( + $('<tr>').html( + $('<td>').text('Channels:') + ).append( + $('<td>').text(track.channels) + ) + ); + } + if (track.rate) { + $table.append( + $('<tr>').html( + $('<td>').text('Samplerate:') + ).append( + $('<td>').text(seperateThousands(track.rate,' ')+' Hz') + ) + ); + } + + $meta.append( + $('<tr>').html( + $('<td>').text(capFirstChar(index)+':') + ).append( + $('<td>').html( + $table + ) + ) + ); + } + + $('#page').html( + $('<p>').text('Detailed information about stream "'+streamName+'"') + ).append( + $('<div>').css({'width':'100%','display':'table','table-layout':'fixed','min-height':'300px'}).html( + $('<div>').css('display','table-row').html( + $('<div>').attr('id','info-stream-meta').css({'display':'table-cell','max-width':'50%','overflow':'auto'}).html( + $meta + ) + ).append( + $('<div>').attr('id','info-stream-statistics').css({'display':'table-cell','text-align':'center','min-height':'200px'}) + ) + ) + ); + } + $('#page').append( + $('<button>').text('Back').addClass('escape-to-cancel').click(function(){ + showTab('streams'); + }) + ); + break; + case 'preview': var httpConnector = false; for (var index in settings.settings.config.protocols) { if ((settings.settings.config.protocols[index].connector == 'HTTP') || (settings.settings.config.protocols[index].connector == 'HTTP.exe')) { @@ -586,37 +720,29 @@ function showTab(tabName,streamName) { } } if (httpConnector) { - var embedbase = 'http://'+parseURL(settings.server).host+':'+(httpConnector.port ? httpConnector.port : 8080)+'/'; $('#page').html( - $('<div>').addClass('input_container').html( - $('<p>').text('Embed info for stream "'+streamName+'"') - ).append( - $('<label>').text('The info embed URL is:').append( - $('<input>').attr('type','text').attr('readonly','readonly').val(embedbase+'info_'+streamName+'.js') + $('<div>').addClass('table').html( + $('<div>').addClass('row').html( + $('<div>').addClass('cell').attr('id','liststreams').addClass('menu') + ).append( + $('<div>').addClass('cell').attr('id','subpage').css('padding-left','1em') ) - ).append( - $('<label>').text('The embed URL is:').append( - $('<input>').attr('type','text').attr('readonly','readonly').val(embedbase+'embed_'+streamName+'.js') - ) - ).append( - $('<label>').text('The embed code is:').css('overflow','hidden').append( - $('<textarea>').val('<div>\n <script src="'+embedbase+'embed_'+streamName+'.js"></' + 'script>\n</div>') - ) - ).append( - $('<button>').text('Back').addClass('escape-to-cancel').click(function(){ - showTab('streams'); - }) ) - ).append( - $('<p>').text('Preview:') - ).append( - $('<div>').attr('id','preview-container') ); + var embedbase = 'http://'+parseURL(settings.server).host+':'+(httpConnector.port ? httpConnector.port : 8080)+'/'; - // jQuery doesn't work -> use DOM magic - var script = document.createElement('script'); - script.src = embedbase+'embed_'+streamName+'.js'; - document.getElementById('preview-container').appendChild( script ); + for (var s in settings.settings.streams) { + if (!streamName) { + streamName = s; + } + $('#liststreams').append( + $('<div>').addClass('button').text(settings.settings.streams[s].name).click(function(){ + buildstreamembed($(this).text()); + }) + ); + } + + buildstreamembed(streamName,embedbase); } else { $('#page').html( @@ -627,29 +753,31 @@ function showTab(tabName,streamName) { case 'limits': var $tbody = $('<tbody>'); $('#page').html( - $('<div>').addClass('description').text('This is an overview of the limits that have been configured on MistServer.') - ).append( - $('<table>').html( - $('<thead>').html( - $('<tr>').html( - $('<th>').text('Applies to') - ).append( - $('<th>').text('Type') - ).append( - $('<th>').text('Name') - ).append( - $('<th>').text('Value') - ).append( - $('<th>') + $('<div>').addClass('LTS-only').html( + $('<div>').addClass('description').text('This is an overview of the limits that have been configured on MistServer.') + ).append( + $('<table>').html( + $('<thead>').html( + $('<tr>').html( + $('<th>').text('Applies to') + ).append( + $('<th>').text('Type') + ).append( + $('<th>').text('Name') + ).append( + $('<th>').text('Value') + ).append( + $('<th>') + ) ) + ).append( + $tbody ) ).append( - $tbody + $('<button>').text('New').click(function(){ + showTab('edit limit','_new_'); + }) ) - ).append( - $('<button>').text('New').click(function(){ - showTab('edit limit','_new_'); - }) ); for (var index in settings.settings.config.limits) { @@ -1041,6 +1169,551 @@ function showTab(tabName,streamName) { } $('#logs-refresh-every').val(defaults.logRefreshing[1]); break; + case 'statistics': + var graphs = {}; + var plot; + $('#page').html( + $('<div>').addClass('description').text('Here, you can select all kinds of data, and view them in a graph.') + ).append( + $('<div>').addClass('input_container').html( + $('<p>').text('Select the data to display') + ).append( + $('<label>').text('Add to graph:').append( + $('<select>').attr('id','graphid').html( + $('<option>').text('New graph').val('new') + ).change(function(){ + if ($(this).val() == 'new') { + $('#graphtype').removeAttr('disabled'); + } + else { + $('#graphtype').attr('disabled','disabled'); + //set to correct type + } + }) + ) + ).append( + $('<label>').text('Graph x-axis type:').append( + $('<select>').attr('id','graphtype').html( + $('<option>').text('Time line').val('time') + ).append( + $('<option>').text('Map').val('coords') + ).change(function(){ + $('#dataset option').hide(); + $('#dataset option.axis_'+$(this).val()).show(); + $('#dataset').val( $('#dataset option.axis_'+$(this).val()).first().val()); + }) + ) + ).append( + $('<label>').text('Select data set:').append( + $('<select>').attr('id','dataset').html( + $('<option>').text('Viewers').val('clients').addClass('axis_time') + ).append( + $('<option>').text('Bandwidth (up)').val('upbps').addClass('axis_time') + ).append( + $('<option>').text('Bandwidth (down)').val('downbps').addClass('axis_time') + ).append( + $('<option>').text('% CPU').val('cpuload').addClass('axis_time') + ).append( + $('<option>').text('Memory load').val('memload').addClass('axis_time') + ).append( + $('<option>').text('Viewer location').val('coords').addClass('axis_coords') + ).change(function(){ + switch ($(this).val()) { + case 'clients': + case 'upbps': + case 'downbps': + $('#dataset-details .replace-dataset').text('amount of viewers') + $('#dataset-details').show(); + break; + default: + $('#dataset-details').hide(); + } + }) + ) + ).append( + $('<div>').attr('id','dataset-details').addClass('checklist').css({ + 'padding':'0.5em 0 0 40%', + 'font-size':'0.9em' + }).html('Show <span class=replace-dataset></span> for:').append( + $('<label>').text('The total').prepend( + $('<input>').attr('type','radio').attr('name','cumutype').attr('checked','checked').val('all') + ) + ).append( + $('<label>').text('The stream ').append( + $('<select>').addClass('stream cumuval') + ).prepend( + $('<input>').attr('type','radio').attr('name','cumutype').val('stream') + ) + ).append( + $('<label>').text('The protocol ').append( + $('<select>').addClass('protocol cumuval') + ).prepend( + $('<input>').attr('type','radio').attr('name','cumutype').val('protocol') + ) + ) + ).append( + $('<button>').text('Add data set').click(function(){ + //the graph + if ($('#graphid').val() == 'new') { + var graph = {}; + graph.id = $('#graphid').val(); + graph.type = $('#graphtype').val(); + graph.id = 'graph_'+($('#graphcontainer .graph').length+1); + graph.datasets = []; + graphs[graph.id] = graph; + $('#graphcontainer').append( + $('<div>').attr('id',graph.id).addClass('graph-item').html( + $('<div>').addClass('legend') + ).append( + $('<div>').addClass('graph') + ) + ); + $('#graphid').append( + $('<option>').text(graph.id) + ).val(graph.id).trigger('change'); + } + else { + var graph = graphs[$('#graphid').val()]; + } + //the dataset itself + var d = { + display: true, + type: $('#dataset').val(), + label: '', + yaxistype: 'amount', + data: [], + lines: { show: true }, + points: { show: false } + }; + switch (d.type) { + case 'cpuload': + d.label = 'CPU load'; + d.yaxistype = 'percentage'; + break; + case 'memload': + d.label = 'Memory load'; + d.yaxistype = 'percentage'; + break; + case 'upbps': + case 'downbps': + case 'clients': + d.cumutype = $('#dataset-details input[name=cumutype]:checked').val(); + d.yaxistype = 'bytespersec'; + if (d.cumutype == 'all') { + switch (d.type) { + case 'clients': + d.label = 'Total viewers'; + d.yaxistype = 'amount'; + break; + case 'upbps': + d.label = 'Total bandwidth (up)'; + break; + case 'downbps': + d.label = 'Total bandwidth (down)'; + break; + } + } + else { + var which = $('#dataset-details.cumuval.'+d.cumutype).val(); + if (d.cumutype == 'stream') { + d.stream = which; + } + else if (d.cumutype == 'protocol') { + d.protocol = which; + } + switch (d.type) { + case 'clients': + d.label = 'Viewers ('+d.stream+')'; + d.yaxistype = 'amount'; + break; + case 'upbps': + d.label = 'Bandwidth (up) ('+d.stream+')'; + break; + case 'downbps': + d.label = 'Bandwidth (down) ('+d.stream+')'; + break; + } + } + break; + } + graph.datasets.push(d); + getPlotData(); + }) + )/*.append( + $('<p>').text('Switch data display type').css('clear','both') + ).append( + $('<label>').text('Show data in a:').append( + $('<select>').html( + $('<option>').text('graph') + ).append( + $('<option>').text('table') + ) + ) + )*/ + ).append( + $('<div>').attr('id','graphcontainer') + ); + for (var i in settings.settings.streams) { + $('#dataset-details .cumuval.stream').append( + $('<option>').text(settings.settings.streams[i].name).val(i) + ); + } + for (var i in settings.settings.config.protocols) { + $('#dataset-details .cumuval.protocol').append( + $('<option>').text(settings.settings.config.protocols[i].connector) + ); + } + $('#graphtype').trigger('change'); + + var lastitem = null; + var $tooltip = $('<div>').attr('id','tooltip'); + $('body').append($tooltip); + $('.graph').live('plothover',function(e,pos,item){ + if (item) { + var pos; + if (item.pageX > ($(window).width() / 2)) { + pos.left = 'auto'; + pos.right = $(window).width() - item.pageX + 8+'px'; + } + else { + pos.left = item.pageX + 8+'px'; + pos.right = 'auto'; + } + if (item.pageY > ($(window).height() / 2)) { + pos.top = 'auto'; + pos.bottom = $(window).height() - item.pageY + 8+'px'; + } + else { + pos.top = item.pageY + 8+'px'; + pos.bottom = 'auto'; + } + $tooltip.css({ + 'left': pos.left, + 'top': pos.top, + 'right': pos.right, + 'bottom': pos.bottom + }).html( + $('<p>').text(item.series.label).prepend( + $('<div>').css({ + 'background-color': item.series.color, + 'width': '20px', + 'height': '20px', + 'display': 'inline-block', + 'margin': '0 0.5em' + }) + ) + ).append( + $('<table>').html( + $('<tr>').html( + $('<td>').text('Time:') + ).append( + $('<td>').text(item.series.xaxis.tickFormatter(item.datapoint[0],item.series.xaxis)) + ) + ).append( + $('<tr>').html( + $('<td>').text(item.series.label+':') + ).append( + $('<td>').text(item.series.yaxis.tickFormatter(item.datapoint[1],item.series.yaxis)) + ) + ) + ).fadeIn(); + } + else { + $('#tooltip').hide(); + } + }); + + + theInterval = setInterval(function(){ + getPlotData(); + },10000); + + function getPlotData() { + getData(function(data){ + for (var j in graphs) { + for (var i in graphs[j].datasets) { + graphs[j].datasets[i] = findDataset(graphs[j].datasets[i],data); + } + drawGraph(graphs[j]); + } + },{capabilities:true,totals:{}}); + } + + function findDataset(dataobj,sourcedata) { + var now = sourcedata.config.time; + switch (dataobj.type) { + case 'cpuload': + //remove any data older than 10 minutes + var removebefore = false; + for (var i in dataobj.data) { + if (dataobj.data[i][0] < (now-600)*1000) { + removebefore = Number(i)+1; + } + } + if (removebefore !== false) { + dataobj.data.splice(0,removebefore); + } + dataobj.data.push([now*1000,sourcedata.capabilities.load.one]); + break; + case 'memload': + //remove any data older than 10 minutes + var removebefore = false; + for (var i in dataobj.data) { + if (dataobj.data[i][0] < (now-600)*1000) { + removebefore = Number(i)+1; + } + } + if (removebefore !== false) { + dataobj.data.splice(0,removebefore); + } + dataobj.data.push([now*1000,sourcedata.capabilities.load.memory]); + break; + case 'upbps': + case 'downbps': + case 'clients': + //todo: depending on the stream.. + if (!sourcedata.totals || !sourcedata.totals.data) { + dataobj.data.push([(now-600)*1000,0]); + dataobj.data.push([now*1000,0]); + } + else { + var fields = {}; + for (var index in sourcedata.totals.fields) { + fields[sourcedata.totals.fields[index]] = index; + } + var time = sourcedata.totals.start; + dataobj.data = []; + if (time > now-590) { + //prepend data with 0 + dataobj.data.push([(now-600)*1000,0]); + dataobj.data.push([time*1000-1,0]); + } + var index = 0; + dataobj.data.push([[time*1000,sourcedata.totals.data[index][fields[dataobj.type]]]]); + for (var i in sourcedata.totals.interval) { + if ((i % 2) == 1) { + //fill gaps with 0 + time += sourcedata.totals.interval[i][1]; + dataobj.data.push([time*1000,0]); + } + else { + for (var j = 0; j < sourcedata.totals.interval[i][0]; j++) { + time += sourcedata.totals.interval[i][1]; + index++; + dataobj.data.push([time*1000,sourcedata.totals.data[index][fields[dataobj.type]]]); + } + if (i < sourcedata.totals.interval.length-1) { + dataobj.data.push([time*1000+1,0]); + } + } + } + if (now > time + 10) { + //append data with 0 + dataobj.data.push([time*1000+1,0]); + dataobj.data.push([now*1000,0]); + } + } + break; + } + + + return dataobj; + } + + function drawGraph(graph){ + var datasets = graph.datasets; + if (datasets.length < 1) { + $('#'+graph.id).children('.graph,.legend').html(''); + return; + } + var yaxes = []; + var yaxesTemplates = { + percentage: { + name: 'percentage', + color: 'black', + display: false, + tickColor: 0, + tickDecimals: 0, + tickFormatter: function(val,axis){ + return val.toFixed(axis.tickDecimals) + '%'; + }, + tickLength: 0, + min: 0 + }, + amount: { + name: 'amount', + color: 'black', + display: false, + tickColor: 0, + tickDecimals: 0, + tickFormatter: function(val,axis){ + return seperateThousands(val.toFixed(axis.tickDecimals),' '); + }, + tickLength: 0, + min: 0 + }, + bytespersec: { + name: 'bytespersec', + color: 'black', + display: false, + tickColor: 0, + tickDecimals: 1, + tickFormatter: function(val,axis){ + var suffix = ['bytes','KiB','MiB','GiB','TiB','PiB']; + if (val == 0) { + val = val+' '+suffix[0]; + } + else { + var exponent = Math.floor(Math.log(Math.abs(val)) / Math.log(1024)); + if (exponent < 0) { + val = val.toFixed(axis.tickDecimals)+' '+suffix[0]; + } + else { + val = Math.round(val / Math.pow(1024,exponent) * Math.pow(10,axis.tickDecimals)) / Math.pow(10,axis.tickDecimals) +' '+suffix[exponent]; + } + } + return val + '/s'; + }, + tickLength: 0, + ticks: function(axis,a,b,c,d){ + //taken from flot source code (function setupTickGeneration), + //modified to think in multiples of 1024 by Carina van der Meer for DDVTECH + + // heuristic based on the model a*sqrt(x) fitted to + // some data points that seemed reasonable + var noTicks = 0.3 * Math.sqrt($('.graph').first().height()); + + var delta = (axis.max - axis.min) / noTicks, + exponent = Math.floor(Math.log(Math.abs(delta)) / Math.log(1024)), + correcteddelta = delta / Math.pow(1024,exponent), + dec = -Math.floor(Math.log(correcteddelta) / Math.LN10), + maxDec = axis.tickDecimals; + + if (maxDec != null && dec > maxDec) { + dec = maxDec; + } + + var magn = Math.pow(10, -dec), + norm = correcteddelta / magn, // norm is between 1.0 and 10.0 + size; + + if (norm < 1.5) { + size = 1; + } else if (norm < 3) { + size = 2; + // special case for 2.5, requires an extra decimal + if (norm > 2.25 && (maxDec == null || dec + 1 <= maxDec)) { + size = 2.5; + ++dec; + } + } else if (norm < 7.5) { + size = 5; + } else { + size = 10; + } + + size *= magn; + size = size * Math.pow(1024,exponent); + + if (axis.minTickSize != null && size < axis.minTickSize) { + size = axis.minTickSize; + } + + axis.delta = delta; + axis.tickDecimals = Math.max(0, maxDec != null ? maxDec : dec); + axis.tickSize = size; + + var ticks = [], + start = axis.tickSize * Math.floor(axis.min / axis.tickSize), + i = 0, + v = Number.NaN, + prev; + + do { + prev = v; + v = start + i * axis.tickSize; + ticks.push(v); + ++i; + } while (v < axis.max && v != prev); + return ticks; + }, + min: 0 + } + }; + var xaxistemplates = { + time: { + name: 'time', + mode: 'time', + timezone: 'browser', + ticks: 5 + } + } + var plotsets = []; + for (var i in datasets) { + if (datasets[i].display) { + if (yaxesTemplates[datasets[i].yaxistype].display === false) { + yaxes.push(yaxesTemplates[datasets[i].yaxistype]); + yaxesTemplates[datasets[i].yaxistype].display = yaxes.length; + } + datasets[i].yaxis = yaxesTemplates[datasets[i].yaxistype].display; + datasets[i].color = Number(i); + plotsets.push(datasets[i]); + } + } + if (yaxes[0]) { yaxes[0].color = 0; } + plot = $.plot( + $('#'+graph.id+' .graph'), + plotsets, + { + legend: {show: false}, + xaxis: xaxistemplates[graph.type], + yaxes: yaxes, + grid: { + hoverable: true, + borderWidth: {top: 0, right: 0, bottom: 1, left: 1}, + color: 'black', + backgroundColor: {colors: ['#fff','#ededed']} + } + } + ); + $('#'+graph.id+' .legend').html( + $('<div>').addClass('legend-list').addClass('checklist') + ); + var plotdata = plot.getOptions(); + for (var i in datasets) { + var $checkbox = $('<input>').attr('type','checkbox').data('dataset-index',i).click(function(){ + if ($(this).is(':checked')) { + datasets[$(this).data('dataset-index')].display = true; + } + else { + datasets[$(this).data('dataset-index')].display = false; + } + drawGraph($(this).parents('.graph-item')); + }); + if (datasets[i].display) { + $checkbox.attr('checked','checked'); + } + $('#'+graph.id+' .legend-list').append( + $('<label>').html( + $checkbox + ).append( + $('<div>').addClass('series-color').css('background-color',plotdata.colors[datasets[i].color % plotdata.colors.length]) + ).append( + datasets[i].label + ) + ); + } + if (datasets.length > 0) { + $('#'+graph.id+' .legend').append( + $('<button>').text('Clear all').click(function(){ + var graph = graphs[$(this).parents('.graph-item').attr('id')]; + graph.datasets = []; + drawGraph(graph); + }).css({'float':'none'}) + ); + } + } + break; case 'server stats': var $cont = $('<div>').addClass('input_container'); @@ -1094,6 +1767,55 @@ function showTab(tabName,streamName) { $('#page').html($cont); break; + case 'email for help': + var config = $.extend({},settings.settings); + delete config.statistics; + config = JSON.stringify(config); + $('#page').html( + $('<div>').addClass('description').html( + 'You can use this form to email MistServer support if you\'re having difficulties.<br>' + ).append( + 'A copy of your server config file will automatically be included.' + ) + ).append( + $('<div>').addClass('input_container').html( + $('<form>').html( + $('<label>').text('Your name:').append( + $('<input>').attr('type','text').attr('name','name') + ) + ).append( + $('<input>').attr('type','hidden').attr('name','company').val('-') + ).append( + $('<label>').text('Your email address:').append( + $('<input>').attr('type','email').attr('name','email') + ) + ).append( + $('<input>').attr('type','hidden').attr('name','subject').val('Integrated Help') + ).append( + $('<label>').text('Your message:').append( + $('<textarea>').attr('name','message').height('20em') + ) + ).append( + $('<label>').text('Your config file:').append( + $('<textarea>').attr('name','configfile').attr('readonly','readonly').css({'height':'20em','font-size':'0.7em'}).val(config) + ) + ).append( + $('<button>').text('Send').click(function(e){ + var data = $(this).parents('form').serialize(); + $.ajax({ + type: 'POST', + url: 'http://mistserver.org/contact_us?skin=plain', + data: data, + success: function(d) { + $('#page').html(d); + } + }); + e.preventDefault(); + }) + ) + ) + ); + break; case 'disconnect': showTab('login'); $('#connection').addClass('red').removeClass('green').text('Disconnected'); @@ -1110,17 +1832,32 @@ function showTab(tabName,streamName) { if ((settings.credentials.authstring) && (!settings.settings.LTS)) { $('.LTS-only input').add('.LTS-only select').add('.LTS-only button').attr('disabled','disabled'); - $('.LTS-only, .LTS-only p, .LTS-only label, .LTS-only button ').css('color','#b4b4b4'); + //$('.LTS-only, .LTS-only p, .LTS-only label, .LTS-only button').css('color','#b4b4b4'); $('.LTS-only, .LTS-only > *').filter(':not(.LTSstuff_done)').each(function(){ var t = []; if ($(this).attr('title')) { t.push($(this).attr('title')); } - t.push('This is feature is only available in the LTS version.'); + t.push('This feature is only available in the LTS version.'); $(this).attr('title',t.join(' ')).addClass('LTSstuff_done'); }); + $('#page .LTS-only').prepend( + $('<a>').text('Upgrade to LTS').attr('target','_blank').attr('href','http://mistserver.org/products/MistServer LTS').addClass('fakebutton') + ); + + $('.linktoReleaseNotes.notedited').each(function(){ + $(this).attr('href',$(this).attr('href')+'/'+settings.settings.config.version.split('-')[0]).removeClass('.notedited'); + }); } else if (settings.settings.LTS) { $('.LTS-only').removeClass('LTS-only'); + $('.linktoTnC.notLTSlink').attr('href','http://mistserver.org/wiki/MistServerLTS_license').removeClass('notLTSlink'); + $('.linktoReleaseNotes.notedited').each(function(){ + $(this).attr('href',$(this).attr('href')+'/'+settings.settings.config.version.split('-')[0]+'LTS').removeClass('.notedited'); + }); + } + + if (ih) { + ihAddBalloons(); } } diff --git a/lsp/plugins/jquery.flot.crosshair.min.js b/lsp/plugins/jquery.flot.crosshair.min.js new file mode 100644 index 00000000..f97ce65a --- /dev/null +++ b/lsp/plugins/jquery.flot.crosshair.min.js @@ -0,0 +1 @@ +(function($){var options={crosshair:{mode:null,color:"rgba(170, 0, 0, 0.80)",lineWidth:1}};function init(plot){var crosshair={x:-1,y:-1,locked:false};plot.setCrosshair=function setCrosshair(pos){if(!pos)crosshair.x=-1;else{var o=plot.p2c(pos);crosshair.x=Math.max(0,Math.min(o.left,plot.width()));crosshair.y=Math.max(0,Math.min(o.top,plot.height()))}plot.triggerRedrawOverlay()};plot.clearCrosshair=plot.setCrosshair;plot.lockCrosshair=function lockCrosshair(pos){if(pos)plot.setCrosshair(pos);crosshair.locked=true};plot.unlockCrosshair=function unlockCrosshair(){crosshair.locked=false};function onMouseOut(e){if(crosshair.locked)return;if(crosshair.x!=-1){crosshair.x=-1;plot.triggerRedrawOverlay()}}function onMouseMove(e){if(crosshair.locked)return;if(plot.getSelection&&plot.getSelection()){crosshair.x=-1;return}var offset=plot.offset();crosshair.x=Math.max(0,Math.min(e.pageX-offset.left,plot.width()));crosshair.y=Math.max(0,Math.min(e.pageY-offset.top,plot.height()));plot.triggerRedrawOverlay()}plot.hooks.bindEvents.push(function(plot,eventHolder){if(!plot.getOptions().crosshair.mode)return;eventHolder.mouseout(onMouseOut);eventHolder.mousemove(onMouseMove)});plot.hooks.drawOverlay.push(function(plot,ctx){var c=plot.getOptions().crosshair;if(!c.mode)return;var plotOffset=plot.getPlotOffset();ctx.save();ctx.translate(plotOffset.left,plotOffset.top);if(crosshair.x!=-1){var adj=plot.getOptions().crosshair.lineWidth%2===0?0:.5;ctx.strokeStyle=c.color;ctx.lineWidth=c.lineWidth;ctx.lineJoin="round";ctx.beginPath();if(c.mode.indexOf("x")!=-1){var drawX=Math.round(crosshair.x)+adj;ctx.moveTo(drawX,0);ctx.lineTo(drawX,plot.height())}if(c.mode.indexOf("y")!=-1){var drawY=Math.round(crosshair.y)+adj;ctx.moveTo(0,drawY);ctx.lineTo(plot.width(),drawY)}ctx.stroke()}ctx.restore()});plot.hooks.shutdown.push(function(plot,eventHolder){eventHolder.unbind("mouseout",onMouseOut);eventHolder.unbind("mousemove",onMouseMove)})}$.plot.plugins.push({init:init,options:options,name:"crosshair",version:"1.0"})})(jQuery); \ No newline at end of file diff --git a/lsp/plugins/jquery.flot.min.js b/lsp/plugins/jquery.flot.min.js new file mode 100644 index 00000000..9620fc00 --- /dev/null +++ b/lsp/plugins/jquery.flot.min.js @@ -0,0 +1,2 @@ +(function($){$.color={};$.color.make=function(r,g,b,a){var o={};o.r=r||0;o.g=g||0;o.b=b||0;o.a=a!=null?a:1;o.add=function(c,d){for(var i=0;i<c.length;++i)o[c.charAt(i)]+=d;return o.normalize()};o.scale=function(c,f){for(var i=0;i<c.length;++i)o[c.charAt(i)]*=f;return o.normalize()};o.toString=function(){if(o.a>=1){return"rgb("+[o.r,o.g,o.b].join(",")+")"}else{return"rgba("+[o.r,o.g,o.b,o.a].join(",")+")"}};o.normalize=function(){function clamp(min,value,max){return value<min?min:value>max?max:value}o.r=clamp(0,parseInt(o.r),255);o.g=clamp(0,parseInt(o.g),255);o.b=clamp(0,parseInt(o.b),255);o.a=clamp(0,o.a,1);return o};o.clone=function(){return $.color.make(o.r,o.b,o.g,o.a)};return o.normalize()};$.color.extract=function(elem,css){var c;do{c=elem.css(css).toLowerCase();if(c!=""&&c!="transparent")break;elem=elem.parent()}while(elem.length&&!$.nodeName(elem.get(0),"body"));if(c=="rgba(0, 0, 0, 0)")c="transparent";return $.color.parse(c)};$.color.parse=function(str){var res,m=$.color.make;if(res=/rgb\(\s*([0-9]{1,3})\s*,\s*([0-9]{1,3})\s*,\s*([0-9]{1,3})\s*\)/.exec(str))return m(parseInt(res[1],10),parseInt(res[2],10),parseInt(res[3],10));if(res=/rgba\(\s*([0-9]{1,3})\s*,\s*([0-9]{1,3})\s*,\s*([0-9]{1,3})\s*,\s*([0-9]+(?:\.[0-9]+)?)\s*\)/.exec(str))return m(parseInt(res[1],10),parseInt(res[2],10),parseInt(res[3],10),parseFloat(res[4]));if(res=/rgb\(\s*([0-9]+(?:\.[0-9]+)?)\%\s*,\s*([0-9]+(?:\.[0-9]+)?)\%\s*,\s*([0-9]+(?:\.[0-9]+)?)\%\s*\)/.exec(str))return m(parseFloat(res[1])*2.55,parseFloat(res[2])*2.55,parseFloat(res[3])*2.55);if(res=/rgba\(\s*([0-9]+(?:\.[0-9]+)?)\%\s*,\s*([0-9]+(?:\.[0-9]+)?)\%\s*,\s*([0-9]+(?:\.[0-9]+)?)\%\s*,\s*([0-9]+(?:\.[0-9]+)?)\s*\)/.exec(str))return m(parseFloat(res[1])*2.55,parseFloat(res[2])*2.55,parseFloat(res[3])*2.55,parseFloat(res[4]));if(res=/#([a-fA-F0-9]{2})([a-fA-F0-9]{2})([a-fA-F0-9]{2})/.exec(str))return m(parseInt(res[1],16),parseInt(res[2],16),parseInt(res[3],16));if(res=/#([a-fA-F0-9])([a-fA-F0-9])([a-fA-F0-9])/.exec(str))return m(parseInt(res[1]+res[1],16),parseInt(res[2]+res[2],16),parseInt(res[3]+res[3],16));var name=$.trim(str).toLowerCase();if(name=="transparent")return m(255,255,255,0);else{res=lookupColors[name]||[0,0,0];return m(res[0],res[1],res[2])}};var lookupColors={aqua:[0,255,255],azure:[240,255,255],beige:[245,245,220],black:[0,0,0],blue:[0,0,255],brown:[165,42,42],cyan:[0,255,255],darkblue:[0,0,139],darkcyan:[0,139,139],darkgrey:[169,169,169],darkgreen:[0,100,0],darkkhaki:[189,183,107],darkmagenta:[139,0,139],darkolivegreen:[85,107,47],darkorange:[255,140,0],darkorchid:[153,50,204],darkred:[139,0,0],darksalmon:[233,150,122],darkviolet:[148,0,211],fuchsia:[255,0,255],gold:[255,215,0],green:[0,128,0],indigo:[75,0,130],khaki:[240,230,140],lightblue:[173,216,230],lightcyan:[224,255,255],lightgreen:[144,238,144],lightgrey:[211,211,211],lightpink:[255,182,193],lightyellow:[255,255,224],lime:[0,255,0],magenta:[255,0,255],maroon:[128,0,0],navy:[0,0,128],olive:[128,128,0],orange:[255,165,0],pink:[255,192,203],purple:[128,0,128],violet:[128,0,128],red:[255,0,0],silver:[192,192,192],white:[255,255,255],yellow:[255,255,0]}})(jQuery);(function($){var hasOwnProperty=Object.prototype.hasOwnProperty;function Canvas(cls,container){var element=container.children("."+cls)[0];if(element==null){element=document.createElement("canvas");element.className=cls;$(element).css({direction:"ltr",position:"absolute",left:0,top:0}).appendTo(container);if(!element.getContext){if(window.G_vmlCanvasManager){element=window.G_vmlCanvasManager.initElement(element)}else{throw new Error("Canvas is not available. If you're using IE with a fall-back such as Excanvas, then there's either a mistake in your conditional include, or the page has no DOCTYPE and is rendering in Quirks Mode.")}}}this.element=element;var context=this.context=element.getContext("2d");var devicePixelRatio=window.devicePixelRatio||1,backingStoreRatio=context.webkitBackingStorePixelRatio||context.mozBackingStorePixelRatio||context.msBackingStorePixelRatio||context.oBackingStorePixelRatio||context.backingStorePixelRatio||1;this.pixelRatio=devicePixelRatio/backingStoreRatio;this.resize(container.width(),container.height());this.textContainer=null;this.text={};this._textCache={}}Canvas.prototype.resize=function(width,height){if(width<=0||height<=0){throw new Error("Invalid dimensions for plot, width = "+width+", height = "+height)}var element=this.element,context=this.context,pixelRatio=this.pixelRatio;if(this.width!=width){element.width=width*pixelRatio;element.style.width=width+"px";this.width=width}if(this.height!=height){element.height=height*pixelRatio;element.style.height=height+"px";this.height=height}context.restore();context.save();context.scale(pixelRatio,pixelRatio)};Canvas.prototype.clear=function(){this.context.clearRect(0,0,this.width,this.height)};Canvas.prototype.render=function(){var cache=this._textCache;for(var layerKey in cache){if(hasOwnProperty.call(cache,layerKey)){var layer=this.getTextLayer(layerKey),layerCache=cache[layerKey];layer.hide();for(var styleKey in layerCache){if(hasOwnProperty.call(layerCache,styleKey)){var styleCache=layerCache[styleKey];for(var key in styleCache){if(hasOwnProperty.call(styleCache,key)){var positions=styleCache[key].positions;for(var i=0,position;position=positions[i];i++){if(position.active){if(!position.rendered){layer.append(position.element);position.rendered=true}}else{positions.splice(i--,1);if(position.rendered){position.element.detach()}}}if(positions.length==0){delete styleCache[key]}}}}}layer.show()}}};Canvas.prototype.getTextLayer=function(classes){var layer=this.text[classes];if(layer==null){if(this.textContainer==null){this.textContainer=$("<div class='flot-text'></div>").css({position:"absolute",top:0,left:0,bottom:0,right:0,"font-size":"smaller",color:"#545454"}).insertAfter(this.element)}layer=this.text[classes]=$("<div></div>").addClass(classes).css({position:"absolute",top:0,left:0,bottom:0,right:0}).appendTo(this.textContainer)}return layer};Canvas.prototype.getTextInfo=function(layer,text,font,angle,width){var textStyle,layerCache,styleCache,info;text=""+text;if(typeof font==="object"){textStyle=font.style+" "+font.variant+" "+font.weight+" "+font.size+"px/"+font.lineHeight+"px "+font.family}else{textStyle=font}layerCache=this._textCache[layer];if(layerCache==null){layerCache=this._textCache[layer]={}}styleCache=layerCache[textStyle];if(styleCache==null){styleCache=layerCache[textStyle]={}}info=styleCache[text];if(info==null){var element=$("<div></div>").html(text).css({position:"absolute","max-width":width,top:-9999}).appendTo(this.getTextLayer(layer));if(typeof font==="object"){element.css({font:textStyle,color:font.color})}else if(typeof font==="string"){element.addClass(font)}info=styleCache[text]={width:element.outerWidth(true),height:element.outerHeight(true),element:element,positions:[]};element.detach()}return info};Canvas.prototype.addText=function(layer,x,y,text,font,angle,width,halign,valign){var info=this.getTextInfo(layer,text,font,angle,width),positions=info.positions;if(halign=="center"){x-=info.width/2}else if(halign=="right"){x-=info.width}if(valign=="middle"){y-=info.height/2}else if(valign=="bottom"){y-=info.height}for(var i=0,position;position=positions[i];i++){if(position.x==x&&position.y==y){position.active=true;return}}position={active:true,rendered:false,element:positions.length?info.element.clone():info.element,x:x,y:y};positions.push(position);position.element.css({top:Math.round(y),left:Math.round(x),"text-align":halign})};Canvas.prototype.removeText=function(layer,x,y,text,font,angle){if(text==null){var layerCache=this._textCache[layer];if(layerCache!=null){for(var styleKey in layerCache){if(hasOwnProperty.call(layerCache,styleKey)){var styleCache=layerCache[styleKey];for(var key in styleCache){if(hasOwnProperty.call(styleCache,key)){var positions=styleCache[key].positions;for(var i=0,position;position=positions[i];i++){position.active=false}}}}}}}else{var positions=this.getTextInfo(layer,text,font,angle).positions;for(var i=0,position;position=positions[i];i++){if(position.x==x&&position.y==y){position.active=false}}}};function Plot(placeholder,data_,options_,plugins){var series=[],options={colors:["#edc240","#afd8f8","#cb4b4b","#4da74d","#9440ed"],legend:{show:true,noColumns:1,labelFormatter:null,labelBoxBorderColor:"#ccc",container:null,position:"ne",margin:5,backgroundColor:null,backgroundOpacity:.85,sorted:null},xaxis:{show:null,position:"bottom",mode:null,font:null,color:null,tickColor:null,transform:null,inverseTransform:null,min:null,max:null,autoscaleMargin:null,ticks:null,tickFormatter:null,labelWidth:null,labelHeight:null,reserveSpace:null,tickLength:null,alignTicksWithAxis:null,tickDecimals:null,tickSize:null,minTickSize:null},yaxis:{autoscaleMargin:.02,position:"left"},xaxes:[],yaxes:[],series:{points:{show:false,radius:3,lineWidth:2,fill:true,fillColor:"#ffffff",symbol:"circle"},lines:{lineWidth:2,fill:false,fillColor:null,steps:false},bars:{show:false,lineWidth:2,barWidth:1,fill:true,fillColor:null,align:"left",horizontal:false,zero:true},shadowSize:3,highlightColor:null},grid:{show:true,aboveData:false,color:"#545454",backgroundColor:null,borderColor:null,tickColor:null,margin:0,labelMargin:5,axisMargin:8,borderWidth:2,minBorderMargin:null,markings:null,markingsColor:"#f4f4f4",markingsLineWidth:2,clickable:false,hoverable:false,autoHighlight:true,mouseActiveRadius:10},interaction:{redrawOverlayInterval:1e3/60},hooks:{}},surface=null,overlay=null,eventHolder=null,ctx=null,octx=null,xaxes=[],yaxes=[],plotOffset={left:0,right:0,top:0,bottom:0},plotWidth=0,plotHeight=0,hooks={processOptions:[],processRawData:[],processDatapoints:[],processOffset:[],drawBackground:[],drawSeries:[],draw:[],bindEvents:[],drawOverlay:[],shutdown:[]},plot=this;plot.setData=setData;plot.setupGrid=setupGrid;plot.draw=draw;plot.getPlaceholder=function(){return placeholder};plot.getCanvas=function(){return surface.element};plot.getPlotOffset=function(){return plotOffset};plot.width=function(){return plotWidth};plot.height=function(){return plotHeight};plot.offset=function(){var o=eventHolder.offset();o.left+=plotOffset.left;o.top+=plotOffset.top;return o};plot.getData=function(){return series};plot.getAxes=function(){var res={},i;$.each(xaxes.concat(yaxes),function(_,axis){if(axis)res[axis.direction+(axis.n!=1?axis.n:"")+"axis"]=axis});return res};plot.getXAxes=function(){return xaxes};plot.getYAxes=function(){return yaxes};plot.c2p=canvasToAxisCoords;plot.p2c=axisToCanvasCoords;plot.getOptions=function(){return options};plot.highlight=highlight;plot.unhighlight=unhighlight;plot.triggerRedrawOverlay=triggerRedrawOverlay;plot.pointOffset=function(point){return{left:parseInt(xaxes[axisNumber(point,"x")-1].p2c(+point.x)+plotOffset.left,10),top:parseInt(yaxes[axisNumber(point,"y")-1].p2c(+point.y)+plotOffset.top,10)}};plot.shutdown=shutdown;plot.destroy=function(){shutdown();placeholder.removeData("plot").empty();series=[];options=null;surface=null;overlay=null;eventHolder=null;ctx=null;octx=null;xaxes=[];yaxes=[];hooks=null;highlights=[];plot=null};plot.resize=function(){var width=placeholder.width(),height=placeholder.height();surface.resize(width,height);overlay.resize(width,height)};plot.hooks=hooks;initPlugins(plot);parseOptions(options_);setupCanvases();setData(data_);setupGrid();draw();bindEvents();function executeHooks(hook,args){args=[plot].concat(args);for(var i=0;i<hook.length;++i)hook[i].apply(this,args)}function initPlugins(){var classes={Canvas:Canvas};for(var i=0;i<plugins.length;++i){var p=plugins[i];p.init(plot,classes);if(p.options)$.extend(true,options,p.options)}}function parseOptions(opts){$.extend(true,options,opts);if(opts&&opts.colors){options.colors=opts.colors}if(options.xaxis.color==null)options.xaxis.color=$.color.parse(options.grid.color).scale("a",.22).toString();if(options.yaxis.color==null)options.yaxis.color=$.color.parse(options.grid.color).scale("a",.22).toString();if(options.xaxis.tickColor==null)options.xaxis.tickColor=options.grid.tickColor||options.xaxis.color;if(options.yaxis.tickColor==null)options.yaxis.tickColor=options.grid.tickColor||options.yaxis.color;if(options.grid.borderColor==null)options.grid.borderColor=options.grid.color;if(options.grid.tickColor==null)options.grid.tickColor=$.color.parse(options.grid.color).scale("a",.22).toString();var i,axisOptions,axisCount,fontSize=placeholder.css("font-size"),fontSizeDefault=fontSize?+fontSize.replace("px",""):13,fontDefaults={style:placeholder.css("font-style"),size:Math.round(.8*fontSizeDefault),variant:placeholder.css("font-variant"),weight:placeholder.css("font-weight"),family:placeholder.css("font-family")};axisCount=options.xaxes.length||1;for(i=0;i<axisCount;++i){axisOptions=options.xaxes[i];if(axisOptions&&!axisOptions.tickColor){axisOptions.tickColor=axisOptions.color}axisOptions=$.extend(true,{},options.xaxis,axisOptions);options.xaxes[i]=axisOptions;if(axisOptions.font){axisOptions.font=$.extend({},fontDefaults,axisOptions.font);if(!axisOptions.font.color){axisOptions.font.color=axisOptions.color}if(!axisOptions.font.lineHeight){axisOptions.font.lineHeight=Math.round(axisOptions.font.size*1.15)}}}axisCount=options.yaxes.length||1;for(i=0;i<axisCount;++i){axisOptions=options.yaxes[i];if(axisOptions&&!axisOptions.tickColor){axisOptions.tickColor=axisOptions.color}axisOptions=$.extend(true,{},options.yaxis,axisOptions);options.yaxes[i]=axisOptions;if(axisOptions.font){axisOptions.font=$.extend({},fontDefaults,axisOptions.font);if(!axisOptions.font.color){axisOptions.font.color=axisOptions.color}if(!axisOptions.font.lineHeight){axisOptions.font.lineHeight=Math.round(axisOptions.font.size*1.15)}}}if(options.xaxis.noTicks&&options.xaxis.ticks==null)options.xaxis.ticks=options.xaxis.noTicks;if(options.yaxis.noTicks&&options.yaxis.ticks==null)options.yaxis.ticks=options.yaxis.noTicks;if(options.x2axis){options.xaxes[1]=$.extend(true,{},options.xaxis,options.x2axis);options.xaxes[1].position="top"}if(options.y2axis){options.yaxes[1]=$.extend(true,{},options.yaxis,options.y2axis);options.yaxes[1].position="right"}if(options.grid.coloredAreas)options.grid.markings=options.grid.coloredAreas;if(options.grid.coloredAreasColor)options.grid.markingsColor=options.grid.coloredAreasColor;if(options.lines)$.extend(true,options.series.lines,options.lines);if(options.points)$.extend(true,options.series.points,options.points);if(options.bars)$.extend(true,options.series.bars,options.bars);if(options.shadowSize!=null)options.series.shadowSize=options.shadowSize;if(options.highlightColor!=null)options.series.highlightColor=options.highlightColor;for(i=0;i<options.xaxes.length;++i)getOrCreateAxis(xaxes,i+1).options=options.xaxes[i];for(i=0;i<options.yaxes.length;++i)getOrCreateAxis(yaxes,i+1).options=options.yaxes[i];for(var n in hooks)if(options.hooks[n]&&options.hooks[n].length)hooks[n]=hooks[n].concat(options.hooks[n]);executeHooks(hooks.processOptions,[options])}function setData(d){series=parseData(d);fillInSeriesOptions();processData()}function parseData(d){var res=[];for(var i=0;i<d.length;++i){var s=$.extend(true,{},options.series);if(d[i].data!=null){s.data=d[i].data;delete d[i].data;$.extend(true,s,d[i]);d[i].data=s.data}else s.data=d[i];res.push(s)}return res}function axisNumber(obj,coord){var a=obj[coord+"axis"];if(typeof a=="object")a=a.n;if(typeof a!="number")a=1;return a}function allAxes(){return $.grep(xaxes.concat(yaxes),function(a){return a})}function canvasToAxisCoords(pos){var res={},i,axis;for(i=0;i<xaxes.length;++i){axis=xaxes[i];if(axis&&axis.used)res["x"+axis.n]=axis.c2p(pos.left)}for(i=0;i<yaxes.length;++i){axis=yaxes[i];if(axis&&axis.used)res["y"+axis.n]=axis.c2p(pos.top)}if(res.x1!==undefined)res.x=res.x1;if(res.y1!==undefined)res.y=res.y1;return res}function axisToCanvasCoords(pos){var res={},i,axis,key;for(i=0;i<xaxes.length;++i){axis=xaxes[i];if(axis&&axis.used){key="x"+axis.n;if(pos[key]==null&&axis.n==1)key="x";if(pos[key]!=null){res.left=axis.p2c(pos[key]);break}}}for(i=0;i<yaxes.length;++i){axis=yaxes[i];if(axis&&axis.used){key="y"+axis.n;if(pos[key]==null&&axis.n==1)key="y";if(pos[key]!=null){res.top=axis.p2c(pos[key]);break}}}return res}function getOrCreateAxis(axes,number){if(!axes[number-1])axes[number-1]={n:number,direction:axes==xaxes?"x":"y",options:$.extend(true,{},axes==xaxes?options.xaxis:options.yaxis)};return axes[number-1]}function fillInSeriesOptions(){var neededColors=series.length,maxIndex=-1,i;for(i=0;i<series.length;++i){var sc=series[i].color;if(sc!=null){neededColors--;if(typeof sc=="number"&&sc>maxIndex){maxIndex=sc}}}if(neededColors<=maxIndex){neededColors=maxIndex+1}var c,colors=[],colorPool=options.colors,colorPoolSize=colorPool.length,variation=0;for(i=0;i<neededColors;i++){c=$.color.parse(colorPool[i%colorPoolSize]||"#666");if(i%colorPoolSize==0&&i){if(variation>=0){if(variation<.5){variation=-variation-.2}else variation=0}else variation=-variation}colors[i]=c.scale("rgb",1+variation)}var colori=0,s;for(i=0;i<series.length;++i){s=series[i];if(s.color==null){s.color=colors[colori].toString();++colori}else if(typeof s.color=="number")s.color=colors[s.color].toString();if(s.lines.show==null){var v,show=true;for(v in s)if(s[v]&&s[v].show){show=false;break}if(show)s.lines.show=true}if(s.lines.zero==null){s.lines.zero=!!s.lines.fill}s.xaxis=getOrCreateAxis(xaxes,axisNumber(s,"x"));s.yaxis=getOrCreateAxis(yaxes,axisNumber(s,"y"))}}function processData(){var topSentry=Number.POSITIVE_INFINITY,bottomSentry=Number.NEGATIVE_INFINITY,fakeInfinity=Number.MAX_VALUE,i,j,k,m,length,s,points,ps,x,y,axis,val,f,p,data,format;function updateAxis(axis,min,max){if(min<axis.datamin&&min!=-fakeInfinity)axis.datamin=min;if(max>axis.datamax&&max!=fakeInfinity)axis.datamax=max}$.each(allAxes(),function(_,axis){axis.datamin=topSentry;axis.datamax=bottomSentry;axis.used=false});for(i=0;i<series.length;++i){s=series[i];s.datapoints={points:[]};executeHooks(hooks.processRawData,[s,s.data,s.datapoints])}for(i=0;i<series.length;++i){s=series[i];data=s.data;format=s.datapoints.format;if(!format){format=[];format.push({x:true,number:true,required:true});format.push({y:true,number:true,required:true});if(s.bars.show||s.lines.show&&s.lines.fill){var autoscale=!!(s.bars.show&&s.bars.zero||s.lines.show&&s.lines.zero);format.push({y:true,number:true,required:false,defaultValue:0,autoscale:autoscale});if(s.bars.horizontal){delete format[format.length-1].y;format[format.length-1].x=true}}s.datapoints.format=format}if(s.datapoints.pointsize!=null)continue;s.datapoints.pointsize=format.length;ps=s.datapoints.pointsize;points=s.datapoints.points;var insertSteps=s.lines.show&&s.lines.steps;s.xaxis.used=s.yaxis.used=true;for(j=k=0;j<data.length;++j,k+=ps){p=data[j];var nullify=p==null;if(!nullify){for(m=0;m<ps;++m){val=p[m];f=format[m];if(f){if(f.number&&val!=null){val=+val;if(isNaN(val))val=null;else if(val==Infinity)val=fakeInfinity;else if(val==-Infinity)val=-fakeInfinity}if(val==null){if(f.required)nullify=true;if(f.defaultValue!=null)val=f.defaultValue}}points[k+m]=val}}if(nullify){for(m=0;m<ps;++m){val=points[k+m];if(val!=null){f=format[m];if(f.autoscale!==false){if(f.x){updateAxis(s.xaxis,val,val)}if(f.y){updateAxis(s.yaxis,val,val)}}}points[k+m]=null}}else{if(insertSteps&&k>0&&points[k-ps]!=null&&points[k-ps]!=points[k]&&points[k-ps+1]!=points[k+1]){for(m=0;m<ps;++m)points[k+ps+m]=points[k+m];points[k+1]=points[k-ps+1];k+=ps}}}}for(i=0;i<series.length;++i){s=series[i];executeHooks(hooks.processDatapoints,[s,s.datapoints])}for(i=0;i<series.length;++i){s=series[i];points=s.datapoints.points;ps=s.datapoints.pointsize;format=s.datapoints.format;var xmin=topSentry,ymin=topSentry,xmax=bottomSentry,ymax=bottomSentry;for(j=0;j<points.length;j+=ps){if(points[j]==null)continue;for(m=0;m<ps;++m){val=points[j+m];f=format[m];if(!f||f.autoscale===false||val==fakeInfinity||val==-fakeInfinity)continue;if(f.x){if(val<xmin)xmin=val;if(val>xmax)xmax=val}if(f.y){if(val<ymin)ymin=val;if(val>ymax)ymax=val}}}if(s.bars.show){var delta;switch(s.bars.align){case"left":delta=0;break;case"right":delta=-s.bars.barWidth;break;default:delta=-s.bars.barWidth/2}if(s.bars.horizontal){ymin+=delta;ymax+=delta+s.bars.barWidth}else{xmin+=delta;xmax+=delta+s.bars.barWidth}}updateAxis(s.xaxis,xmin,xmax);updateAxis(s.yaxis,ymin,ymax)}$.each(allAxes(),function(_,axis){if(axis.datamin==topSentry)axis.datamin=null;if(axis.datamax==bottomSentry)axis.datamax=null})}function setupCanvases(){placeholder.css("padding",0).children().filter(function(){return!$(this).hasClass("flot-overlay")&&!$(this).hasClass("flot-base")}).remove();if(placeholder.css("position")=="static")placeholder.css("position","relative");surface=new Canvas("flot-base",placeholder);overlay=new Canvas("flot-overlay",placeholder);ctx=surface.context;octx=overlay.context;eventHolder=$(overlay.element).unbind();var existing=placeholder.data("plot");if(existing){existing.shutdown();overlay.clear()}placeholder.data("plot",plot)}function bindEvents(){if(options.grid.hoverable){eventHolder.mousemove(onMouseMove);eventHolder.bind("mouseleave",onMouseLeave)}if(options.grid.clickable)eventHolder.click(onClick);executeHooks(hooks.bindEvents,[eventHolder])}function shutdown(){if(redrawTimeout)clearTimeout(redrawTimeout);eventHolder.unbind("mousemove",onMouseMove);eventHolder.unbind("mouseleave",onMouseLeave);eventHolder.unbind("click",onClick);executeHooks(hooks.shutdown,[eventHolder])}function setTransformationHelpers(axis){function identity(x){return x}var s,m,t=axis.options.transform||identity,it=axis.options.inverseTransform;if(axis.direction=="x"){s=axis.scale=plotWidth/Math.abs(t(axis.max)-t(axis.min));m=Math.min(t(axis.max),t(axis.min))}else{s=axis.scale=plotHeight/Math.abs(t(axis.max)-t(axis.min));s=-s;m=Math.max(t(axis.max),t(axis.min))}if(t==identity)axis.p2c=function(p){return(p-m)*s};else axis.p2c=function(p){return(t(p)-m)*s};if(!it)axis.c2p=function(c){return m+c/s};else axis.c2p=function(c){return it(m+c/s)}}function measureTickLabels(axis){var opts=axis.options,ticks=axis.ticks||[],labelWidth=opts.labelWidth||0,labelHeight=opts.labelHeight||0,maxWidth=labelWidth||(axis.direction=="x"?Math.floor(surface.width/(ticks.length||1)):null),legacyStyles=axis.direction+"Axis "+axis.direction+axis.n+"Axis",layer="flot-"+axis.direction+"-axis flot-"+axis.direction+axis.n+"-axis "+legacyStyles,font=opts.font||"flot-tick-label tickLabel";for(var i=0;i<ticks.length;++i){var t=ticks[i];if(!t.label)continue;var info=surface.getTextInfo(layer,t.label,font,null,maxWidth);labelWidth=Math.max(labelWidth,info.width);labelHeight=Math.max(labelHeight,info.height)}axis.labelWidth=opts.labelWidth||labelWidth;axis.labelHeight=opts.labelHeight||labelHeight}function allocateAxisBoxFirstPhase(axis){var lw=axis.labelWidth,lh=axis.labelHeight,pos=axis.options.position,isXAxis=axis.direction==="x",tickLength=axis.options.tickLength,axisMargin=options.grid.axisMargin,padding=options.grid.labelMargin,innermost=true,outermost=true,first=true,found=false;$.each(isXAxis?xaxes:yaxes,function(i,a){if(a&&a.reserveSpace){if(a===axis){found=true}else if(a.options.position===pos){if(found){outermost=false}else{innermost=false}}if(!found){first=false}}});if(outermost){axisMargin=0}if(tickLength==null){tickLength=first?"full":5}if(!isNaN(+tickLength))padding+=+tickLength;if(isXAxis){lh+=padding;if(pos=="bottom"){plotOffset.bottom+=lh+axisMargin;axis.box={top:surface.height-plotOffset.bottom,height:lh}}else{axis.box={top:plotOffset.top+axisMargin,height:lh};plotOffset.top+=lh+axisMargin}}else{lw+=padding;if(pos=="left"){axis.box={left:plotOffset.left+axisMargin,width:lw};plotOffset.left+=lw+axisMargin}else{plotOffset.right+=lw+axisMargin;axis.box={left:surface.width-plotOffset.right,width:lw}}}axis.position=pos;axis.tickLength=tickLength;axis.box.padding=padding;axis.innermost=innermost}function allocateAxisBoxSecondPhase(axis){if(axis.direction=="x"){axis.box.left=plotOffset.left-axis.labelWidth/2;axis.box.width=surface.width-plotOffset.left-plotOffset.right+axis.labelWidth}else{axis.box.top=plotOffset.top-axis.labelHeight/2;axis.box.height=surface.height-plotOffset.bottom-plotOffset.top+axis.labelHeight}}function adjustLayoutForThingsStickingOut(){var minMargin=options.grid.minBorderMargin,axis,i;if(minMargin==null){minMargin=0;for(i=0;i<series.length;++i)minMargin=Math.max(minMargin,2*(series[i].points.radius+series[i].points.lineWidth/2))}var margins={left:minMargin,right:minMargin,top:minMargin,bottom:minMargin};$.each(allAxes(),function(_,axis){if(axis.reserveSpace&&axis.ticks&&axis.ticks.length){var lastTick=axis.ticks[axis.ticks.length-1];if(axis.direction==="x"){margins.left=Math.max(margins.left,axis.labelWidth/2);if(lastTick.v<=axis.max){margins.right=Math.max(margins.right,axis.labelWidth/2)}}else{margins.bottom=Math.max(margins.bottom,axis.labelHeight/2);if(lastTick.v<=axis.max){margins.top=Math.max(margins.top,axis.labelHeight/2)}}}});plotOffset.left=Math.ceil(Math.max(margins.left,plotOffset.left));plotOffset.right=Math.ceil(Math.max(margins.right,plotOffset.right));plotOffset.top=Math.ceil(Math.max(margins.top,plotOffset.top));plotOffset.bottom=Math.ceil(Math.max(margins.bottom,plotOffset.bottom))}function setupGrid(){var i,axes=allAxes(),showGrid=options.grid.show;for(var a in plotOffset){var margin=options.grid.margin||0;plotOffset[a]=typeof margin=="number"?margin:margin[a]||0}executeHooks(hooks.processOffset,[plotOffset]);for(var a in plotOffset){if(typeof options.grid.borderWidth=="object"){plotOffset[a]+=showGrid?options.grid.borderWidth[a]:0}else{plotOffset[a]+=showGrid?options.grid.borderWidth:0}}$.each(axes,function(_,axis){axis.show=axis.options.show;if(axis.show==null)axis.show=axis.used;axis.reserveSpace=axis.show||axis.options.reserveSpace;setRange(axis)});if(showGrid){var allocatedAxes=$.grep(axes,function(axis){return axis.reserveSpace});$.each(allocatedAxes,function(_,axis){setupTickGeneration(axis);setTicks(axis);snapRangeToTicks(axis,axis.ticks);measureTickLabels(axis)});for(i=allocatedAxes.length-1;i>=0;--i)allocateAxisBoxFirstPhase(allocatedAxes[i]);adjustLayoutForThingsStickingOut();$.each(allocatedAxes,function(_,axis){allocateAxisBoxSecondPhase(axis)})}plotWidth=surface.width-plotOffset.left-plotOffset.right;plotHeight=surface.height-plotOffset.bottom-plotOffset.top;$.each(axes,function(_,axis){setTransformationHelpers(axis)});if(showGrid){drawAxisLabels()}insertLegend()}function setRange(axis){var opts=axis.options,min=+(opts.min!=null?opts.min:axis.datamin),max=+(opts.max!=null?opts.max:axis.datamax),delta=max-min;if(delta==0){var widen=max==0?1:.01;if(opts.min==null)min-=widen;if(opts.max==null||opts.min!=null)max+=widen}else{var margin=opts.autoscaleMargin;if(margin!=null){if(opts.min==null){min-=delta*margin;if(min<0&&axis.datamin!=null&&axis.datamin>=0)min=0}if(opts.max==null){max+=delta*margin;if(max>0&&axis.datamax!=null&&axis.datamax<=0)max=0}}}axis.min=min;axis.max=max}function setupTickGeneration(axis){var opts=axis.options;var noTicks;if(typeof opts.ticks=="number"&&opts.ticks>0)noTicks=opts.ticks;else noTicks=.3*Math.sqrt(axis.direction=="x"?surface.width:surface.height);var delta=(axis.max-axis.min)/noTicks,dec=-Math.floor(Math.log(delta)/Math.LN10),maxDec=opts.tickDecimals;if(maxDec!=null&&dec>maxDec){dec=maxDec}var magn=Math.pow(10,-dec),norm=delta/magn,size;if(norm<1.5){size=1}else if(norm<3){size=2;if(norm>2.25&&(maxDec==null||dec+1<=maxDec)){size=2.5;++dec}}else if(norm<7.5){size=5}else{size=10}size*=magn;if(opts.minTickSize!=null&&size<opts.minTickSize){size=opts.minTickSize}axis.delta=delta;axis.tickDecimals=Math.max(0,maxDec!=null?maxDec:dec);axis.tickSize=opts.tickSize||size;if(opts.mode=="time"&&!axis.tickGenerator){throw new Error("Time mode requires the flot.time plugin.")}if(!axis.tickGenerator){axis.tickGenerator=function(axis){var ticks=[],start=floorInBase(axis.min,axis.tickSize),i=0,v=Number.NaN,prev;do{prev=v;v=start+i*axis.tickSize;ticks.push(v);++i}while(v<axis.max&&v!=prev);return ticks};axis.tickFormatter=function(value,axis){var factor=axis.tickDecimals?Math.pow(10,axis.tickDecimals):1;var formatted=""+Math.round(value*factor)/factor;if(axis.tickDecimals!=null){var decimal=formatted.indexOf(".");var precision=decimal==-1?0:formatted.length-decimal-1;if(precision<axis.tickDecimals){return(precision?formatted:formatted+".")+(""+factor).substr(1,axis.tickDecimals-precision)}}return formatted}}if($.isFunction(opts.tickFormatter))axis.tickFormatter=function(v,axis){return""+opts.tickFormatter(v,axis)};if(opts.alignTicksWithAxis!=null){var otherAxis=(axis.direction=="x"?xaxes:yaxes)[opts.alignTicksWithAxis-1];if(otherAxis&&otherAxis.used&&otherAxis!=axis){var niceTicks=axis.tickGenerator(axis);if(niceTicks.length>0){if(opts.min==null)axis.min=Math.min(axis.min,niceTicks[0]);if(opts.max==null&&niceTicks.length>1)axis.max=Math.max(axis.max,niceTicks[niceTicks.length-1])}axis.tickGenerator=function(axis){var ticks=[],v,i;for(i=0;i<otherAxis.ticks.length;++i){v=(otherAxis.ticks[i].v-otherAxis.min)/(otherAxis.max-otherAxis.min);v=axis.min+v*(axis.max-axis.min);ticks.push(v)}return ticks};if(!axis.mode&&opts.tickDecimals==null){var extraDec=Math.max(0,-Math.floor(Math.log(axis.delta)/Math.LN10)+1),ts=axis.tickGenerator(axis);if(!(ts.length>1&&/\..*0$/.test((ts[1]-ts[0]).toFixed(extraDec))))axis.tickDecimals=extraDec}}}}function setTicks(axis){var oticks=axis.options.ticks,ticks=[];if(oticks==null||typeof oticks=="number"&&oticks>0)ticks=axis.tickGenerator(axis);else if(oticks){if($.isFunction(oticks))ticks=oticks(axis);else ticks=oticks}var i,v;axis.ticks=[];for(i=0;i<ticks.length;++i){var label=null;var t=ticks[i];if(typeof t=="object"){v=+t[0];if(t.length>1)label=t[1]}else v=+t;if(label==null)label=axis.tickFormatter(v,axis);if(!isNaN(v))axis.ticks.push({v:v,label:label})}}function snapRangeToTicks(axis,ticks){if(axis.options.autoscaleMargin&&ticks.length>0){if(axis.options.min==null)axis.min=Math.min(axis.min,ticks[0].v);if(axis.options.max==null&&ticks.length>1)axis.max=Math.max(axis.max,ticks[ticks.length-1].v)}}function draw(){surface.clear();executeHooks(hooks.drawBackground,[ctx]);var grid=options.grid;if(grid.show&&grid.backgroundColor)drawBackground();if(grid.show&&!grid.aboveData){drawGrid()}for(var i=0;i<series.length;++i){executeHooks(hooks.drawSeries,[ctx,series[i]]);drawSeries(series[i])}executeHooks(hooks.draw,[ctx]);if(grid.show&&grid.aboveData){drawGrid()}surface.render();triggerRedrawOverlay()}function extractRange(ranges,coord){var axis,from,to,key,axes=allAxes();for(var i=0;i<axes.length;++i){axis=axes[i];if(axis.direction==coord){key=coord+axis.n+"axis";if(!ranges[key]&&axis.n==1)key=coord+"axis";if(ranges[key]){from=ranges[key].from;to=ranges[key].to;break}}}if(!ranges[key]){axis=coord=="x"?xaxes[0]:yaxes[0];from=ranges[coord+"1"];to=ranges[coord+"2"]}if(from!=null&&to!=null&&from>to){var tmp=from;from=to;to=tmp}return{from:from,to:to,axis:axis}}function drawBackground(){ctx.save();ctx.translate(plotOffset.left,plotOffset.top);ctx.fillStyle=getColorOrGradient(options.grid.backgroundColor,plotHeight,0,"rgba(255, 255, 255, 0)");ctx.fillRect(0,0,plotWidth,plotHeight);ctx.restore()}function drawGrid(){var i,axes,bw,bc;ctx.save();ctx.translate(plotOffset.left,plotOffset.top);var markings=options.grid.markings;if(markings){if($.isFunction(markings)){axes=plot.getAxes();axes.xmin=axes.xaxis.min;axes.xmax=axes.xaxis.max;axes.ymin=axes.yaxis.min;axes.ymax=axes.yaxis.max;markings=markings(axes)}for(i=0;i<markings.length;++i){var m=markings[i],xrange=extractRange(m,"x"),yrange=extractRange(m,"y");if(xrange.from==null)xrange.from=xrange.axis.min;if(xrange.to==null)xrange.to=xrange.axis.max;if(yrange.from==null)yrange.from=yrange.axis.min;if(yrange.to==null)yrange.to=yrange.axis.max;if(xrange.to<xrange.axis.min||xrange.from>xrange.axis.max||yrange.to<yrange.axis.min||yrange.from>yrange.axis.max)continue;xrange.from=Math.max(xrange.from,xrange.axis.min);xrange.to=Math.min(xrange.to,xrange.axis.max); +yrange.from=Math.max(yrange.from,yrange.axis.min);yrange.to=Math.min(yrange.to,yrange.axis.max);if(xrange.from==xrange.to&&yrange.from==yrange.to)continue;xrange.from=xrange.axis.p2c(xrange.from);xrange.to=xrange.axis.p2c(xrange.to);yrange.from=yrange.axis.p2c(yrange.from);yrange.to=yrange.axis.p2c(yrange.to);if(xrange.from==xrange.to||yrange.from==yrange.to){ctx.beginPath();ctx.strokeStyle=m.color||options.grid.markingsColor;ctx.lineWidth=m.lineWidth||options.grid.markingsLineWidth;ctx.moveTo(xrange.from,yrange.from);ctx.lineTo(xrange.to,yrange.to);ctx.stroke()}else{ctx.fillStyle=m.color||options.grid.markingsColor;ctx.fillRect(xrange.from,yrange.to,xrange.to-xrange.from,yrange.from-yrange.to)}}}axes=allAxes();bw=options.grid.borderWidth;for(var j=0;j<axes.length;++j){var axis=axes[j],box=axis.box,t=axis.tickLength,x,y,xoff,yoff;if(!axis.show||axis.ticks.length==0)continue;ctx.lineWidth=1;if(axis.direction=="x"){x=0;if(t=="full")y=axis.position=="top"?0:plotHeight;else y=box.top-plotOffset.top+(axis.position=="top"?box.height:0)}else{y=0;if(t=="full")x=axis.position=="left"?0:plotWidth;else x=box.left-plotOffset.left+(axis.position=="left"?box.width:0)}if(!axis.innermost){ctx.strokeStyle=axis.options.color;ctx.beginPath();xoff=yoff=0;if(axis.direction=="x")xoff=plotWidth+1;else yoff=plotHeight+1;if(ctx.lineWidth==1){if(axis.direction=="x"){y=Math.floor(y)+.5}else{x=Math.floor(x)+.5}}ctx.moveTo(x,y);ctx.lineTo(x+xoff,y+yoff);ctx.stroke()}ctx.strokeStyle=axis.options.tickColor;ctx.beginPath();for(i=0;i<axis.ticks.length;++i){var v=axis.ticks[i].v;xoff=yoff=0;if(isNaN(v)||v<axis.min||v>axis.max||t=="full"&&(typeof bw=="object"&&bw[axis.position]>0||bw>0)&&(v==axis.min||v==axis.max))continue;if(axis.direction=="x"){x=axis.p2c(v);yoff=t=="full"?-plotHeight:t;if(axis.position=="top")yoff=-yoff}else{y=axis.p2c(v);xoff=t=="full"?-plotWidth:t;if(axis.position=="left")xoff=-xoff}if(ctx.lineWidth==1){if(axis.direction=="x")x=Math.floor(x)+.5;else y=Math.floor(y)+.5}ctx.moveTo(x,y);ctx.lineTo(x+xoff,y+yoff)}ctx.stroke()}if(bw){bc=options.grid.borderColor;if(typeof bw=="object"||typeof bc=="object"){if(typeof bw!=="object"){bw={top:bw,right:bw,bottom:bw,left:bw}}if(typeof bc!=="object"){bc={top:bc,right:bc,bottom:bc,left:bc}}if(bw.top>0){ctx.strokeStyle=bc.top;ctx.lineWidth=bw.top;ctx.beginPath();ctx.moveTo(0-bw.left,0-bw.top/2);ctx.lineTo(plotWidth,0-bw.top/2);ctx.stroke()}if(bw.right>0){ctx.strokeStyle=bc.right;ctx.lineWidth=bw.right;ctx.beginPath();ctx.moveTo(plotWidth+bw.right/2,0-bw.top);ctx.lineTo(plotWidth+bw.right/2,plotHeight);ctx.stroke()}if(bw.bottom>0){ctx.strokeStyle=bc.bottom;ctx.lineWidth=bw.bottom;ctx.beginPath();ctx.moveTo(plotWidth+bw.right,plotHeight+bw.bottom/2);ctx.lineTo(0,plotHeight+bw.bottom/2);ctx.stroke()}if(bw.left>0){ctx.strokeStyle=bc.left;ctx.lineWidth=bw.left;ctx.beginPath();ctx.moveTo(0-bw.left/2,plotHeight+bw.bottom);ctx.lineTo(0-bw.left/2,0);ctx.stroke()}}else{ctx.lineWidth=bw;ctx.strokeStyle=options.grid.borderColor;ctx.strokeRect(-bw/2,-bw/2,plotWidth+bw,plotHeight+bw)}}ctx.restore()}function drawAxisLabels(){$.each(allAxes(),function(_,axis){var box=axis.box,legacyStyles=axis.direction+"Axis "+axis.direction+axis.n+"Axis",layer="flot-"+axis.direction+"-axis flot-"+axis.direction+axis.n+"-axis "+legacyStyles,font=axis.options.font||"flot-tick-label tickLabel",tick,x,y,halign,valign;surface.removeText(layer);if(!axis.show||axis.ticks.length==0)return;for(var i=0;i<axis.ticks.length;++i){tick=axis.ticks[i];if(!tick.label||tick.v<axis.min||tick.v>axis.max)continue;if(axis.direction=="x"){halign="center";x=plotOffset.left+axis.p2c(tick.v);if(axis.position=="bottom"){y=box.top+box.padding}else{y=box.top+box.height-box.padding;valign="bottom"}}else{valign="middle";y=plotOffset.top+axis.p2c(tick.v);if(axis.position=="left"){x=box.left+box.width-box.padding;halign="right"}else{x=box.left+box.padding}}surface.addText(layer,x,y,tick.label,font,null,null,halign,valign)}})}function drawSeries(series){if(series.lines.show)drawSeriesLines(series);if(series.bars.show)drawSeriesBars(series);if(series.points.show)drawSeriesPoints(series)}function drawSeriesLines(series){function plotLine(datapoints,xoffset,yoffset,axisx,axisy){var points=datapoints.points,ps=datapoints.pointsize,prevx=null,prevy=null;ctx.beginPath();for(var i=ps;i<points.length;i+=ps){var x1=points[i-ps],y1=points[i-ps+1],x2=points[i],y2=points[i+1];if(x1==null||x2==null)continue;if(y1<=y2&&y1<axisy.min){if(y2<axisy.min)continue;x1=(axisy.min-y1)/(y2-y1)*(x2-x1)+x1;y1=axisy.min}else if(y2<=y1&&y2<axisy.min){if(y1<axisy.min)continue;x2=(axisy.min-y1)/(y2-y1)*(x2-x1)+x1;y2=axisy.min}if(y1>=y2&&y1>axisy.max){if(y2>axisy.max)continue;x1=(axisy.max-y1)/(y2-y1)*(x2-x1)+x1;y1=axisy.max}else if(y2>=y1&&y2>axisy.max){if(y1>axisy.max)continue;x2=(axisy.max-y1)/(y2-y1)*(x2-x1)+x1;y2=axisy.max}if(x1<=x2&&x1<axisx.min){if(x2<axisx.min)continue;y1=(axisx.min-x1)/(x2-x1)*(y2-y1)+y1;x1=axisx.min}else if(x2<=x1&&x2<axisx.min){if(x1<axisx.min)continue;y2=(axisx.min-x1)/(x2-x1)*(y2-y1)+y1;x2=axisx.min}if(x1>=x2&&x1>axisx.max){if(x2>axisx.max)continue;y1=(axisx.max-x1)/(x2-x1)*(y2-y1)+y1;x1=axisx.max}else if(x2>=x1&&x2>axisx.max){if(x1>axisx.max)continue;y2=(axisx.max-x1)/(x2-x1)*(y2-y1)+y1;x2=axisx.max}if(x1!=prevx||y1!=prevy)ctx.moveTo(axisx.p2c(x1)+xoffset,axisy.p2c(y1)+yoffset);prevx=x2;prevy=y2;ctx.lineTo(axisx.p2c(x2)+xoffset,axisy.p2c(y2)+yoffset)}ctx.stroke()}function plotLineArea(datapoints,axisx,axisy){var points=datapoints.points,ps=datapoints.pointsize,bottom=Math.min(Math.max(0,axisy.min),axisy.max),i=0,top,areaOpen=false,ypos=1,segmentStart=0,segmentEnd=0;while(true){if(ps>0&&i>points.length+ps)break;i+=ps;var x1=points[i-ps],y1=points[i-ps+ypos],x2=points[i],y2=points[i+ypos];if(areaOpen){if(ps>0&&x1!=null&&x2==null){segmentEnd=i;ps=-ps;ypos=2;continue}if(ps<0&&i==segmentStart+ps){ctx.fill();areaOpen=false;ps=-ps;ypos=1;i=segmentStart=segmentEnd+ps;continue}}if(x1==null||x2==null)continue;if(x1<=x2&&x1<axisx.min){if(x2<axisx.min)continue;y1=(axisx.min-x1)/(x2-x1)*(y2-y1)+y1;x1=axisx.min}else if(x2<=x1&&x2<axisx.min){if(x1<axisx.min)continue;y2=(axisx.min-x1)/(x2-x1)*(y2-y1)+y1;x2=axisx.min}if(x1>=x2&&x1>axisx.max){if(x2>axisx.max)continue;y1=(axisx.max-x1)/(x2-x1)*(y2-y1)+y1;x1=axisx.max}else if(x2>=x1&&x2>axisx.max){if(x1>axisx.max)continue;y2=(axisx.max-x1)/(x2-x1)*(y2-y1)+y1;x2=axisx.max}if(!areaOpen){ctx.beginPath();ctx.moveTo(axisx.p2c(x1),axisy.p2c(bottom));areaOpen=true}if(y1>=axisy.max&&y2>=axisy.max){ctx.lineTo(axisx.p2c(x1),axisy.p2c(axisy.max));ctx.lineTo(axisx.p2c(x2),axisy.p2c(axisy.max));continue}else if(y1<=axisy.min&&y2<=axisy.min){ctx.lineTo(axisx.p2c(x1),axisy.p2c(axisy.min));ctx.lineTo(axisx.p2c(x2),axisy.p2c(axisy.min));continue}var x1old=x1,x2old=x2;if(y1<=y2&&y1<axisy.min&&y2>=axisy.min){x1=(axisy.min-y1)/(y2-y1)*(x2-x1)+x1;y1=axisy.min}else if(y2<=y1&&y2<axisy.min&&y1>=axisy.min){x2=(axisy.min-y1)/(y2-y1)*(x2-x1)+x1;y2=axisy.min}if(y1>=y2&&y1>axisy.max&&y2<=axisy.max){x1=(axisy.max-y1)/(y2-y1)*(x2-x1)+x1;y1=axisy.max}else if(y2>=y1&&y2>axisy.max&&y1<=axisy.max){x2=(axisy.max-y1)/(y2-y1)*(x2-x1)+x1;y2=axisy.max}if(x1!=x1old){ctx.lineTo(axisx.p2c(x1old),axisy.p2c(y1))}ctx.lineTo(axisx.p2c(x1),axisy.p2c(y1));ctx.lineTo(axisx.p2c(x2),axisy.p2c(y2));if(x2!=x2old){ctx.lineTo(axisx.p2c(x2),axisy.p2c(y2));ctx.lineTo(axisx.p2c(x2old),axisy.p2c(y2))}}}ctx.save();ctx.translate(plotOffset.left,plotOffset.top);ctx.lineJoin="round";var lw=series.lines.lineWidth,sw=series.shadowSize;if(lw>0&&sw>0){ctx.lineWidth=sw;ctx.strokeStyle="rgba(0,0,0,0.1)";var angle=Math.PI/18;plotLine(series.datapoints,Math.sin(angle)*(lw/2+sw/2),Math.cos(angle)*(lw/2+sw/2),series.xaxis,series.yaxis);ctx.lineWidth=sw/2;plotLine(series.datapoints,Math.sin(angle)*(lw/2+sw/4),Math.cos(angle)*(lw/2+sw/4),series.xaxis,series.yaxis)}ctx.lineWidth=lw;ctx.strokeStyle=series.color;var fillStyle=getFillStyle(series.lines,series.color,0,plotHeight);if(fillStyle){ctx.fillStyle=fillStyle;plotLineArea(series.datapoints,series.xaxis,series.yaxis)}if(lw>0)plotLine(series.datapoints,0,0,series.xaxis,series.yaxis);ctx.restore()}function drawSeriesPoints(series){function plotPoints(datapoints,radius,fillStyle,offset,shadow,axisx,axisy,symbol){var points=datapoints.points,ps=datapoints.pointsize;for(var i=0;i<points.length;i+=ps){var x=points[i],y=points[i+1];if(x==null||x<axisx.min||x>axisx.max||y<axisy.min||y>axisy.max)continue;ctx.beginPath();x=axisx.p2c(x);y=axisy.p2c(y)+offset;if(symbol=="circle")ctx.arc(x,y,radius,0,shadow?Math.PI:Math.PI*2,false);else symbol(ctx,x,y,radius,shadow);ctx.closePath();if(fillStyle){ctx.fillStyle=fillStyle;ctx.fill()}ctx.stroke()}}ctx.save();ctx.translate(plotOffset.left,plotOffset.top);var lw=series.points.lineWidth,sw=series.shadowSize,radius=series.points.radius,symbol=series.points.symbol;if(lw==0)lw=1e-4;if(lw>0&&sw>0){var w=sw/2;ctx.lineWidth=w;ctx.strokeStyle="rgba(0,0,0,0.1)";plotPoints(series.datapoints,radius,null,w+w/2,true,series.xaxis,series.yaxis,symbol);ctx.strokeStyle="rgba(0,0,0,0.2)";plotPoints(series.datapoints,radius,null,w/2,true,series.xaxis,series.yaxis,symbol)}ctx.lineWidth=lw;ctx.strokeStyle=series.color;plotPoints(series.datapoints,radius,getFillStyle(series.points,series.color),0,false,series.xaxis,series.yaxis,symbol);ctx.restore()}function drawBar(x,y,b,barLeft,barRight,fillStyleCallback,axisx,axisy,c,horizontal,lineWidth){var left,right,bottom,top,drawLeft,drawRight,drawTop,drawBottom,tmp;if(horizontal){drawBottom=drawRight=drawTop=true;drawLeft=false;left=b;right=x;top=y+barLeft;bottom=y+barRight;if(right<left){tmp=right;right=left;left=tmp;drawLeft=true;drawRight=false}}else{drawLeft=drawRight=drawTop=true;drawBottom=false;left=x+barLeft;right=x+barRight;bottom=b;top=y;if(top<bottom){tmp=top;top=bottom;bottom=tmp;drawBottom=true;drawTop=false}}if(right<axisx.min||left>axisx.max||top<axisy.min||bottom>axisy.max)return;if(left<axisx.min){left=axisx.min;drawLeft=false}if(right>axisx.max){right=axisx.max;drawRight=false}if(bottom<axisy.min){bottom=axisy.min;drawBottom=false}if(top>axisy.max){top=axisy.max;drawTop=false}left=axisx.p2c(left);bottom=axisy.p2c(bottom);right=axisx.p2c(right);top=axisy.p2c(top);if(fillStyleCallback){c.fillStyle=fillStyleCallback(bottom,top);c.fillRect(left,top,right-left,bottom-top)}if(lineWidth>0&&(drawLeft||drawRight||drawTop||drawBottom)){c.beginPath();c.moveTo(left,bottom);if(drawLeft)c.lineTo(left,top);else c.moveTo(left,top);if(drawTop)c.lineTo(right,top);else c.moveTo(right,top);if(drawRight)c.lineTo(right,bottom);else c.moveTo(right,bottom);if(drawBottom)c.lineTo(left,bottom);else c.moveTo(left,bottom);c.stroke()}}function drawSeriesBars(series){function plotBars(datapoints,barLeft,barRight,fillStyleCallback,axisx,axisy){var points=datapoints.points,ps=datapoints.pointsize;for(var i=0;i<points.length;i+=ps){if(points[i]==null)continue;drawBar(points[i],points[i+1],points[i+2],barLeft,barRight,fillStyleCallback,axisx,axisy,ctx,series.bars.horizontal,series.bars.lineWidth)}}ctx.save();ctx.translate(plotOffset.left,plotOffset.top);ctx.lineWidth=series.bars.lineWidth;ctx.strokeStyle=series.color;var barLeft;switch(series.bars.align){case"left":barLeft=0;break;case"right":barLeft=-series.bars.barWidth;break;default:barLeft=-series.bars.barWidth/2}var fillStyleCallback=series.bars.fill?function(bottom,top){return getFillStyle(series.bars,series.color,bottom,top)}:null;plotBars(series.datapoints,barLeft,barLeft+series.bars.barWidth,fillStyleCallback,series.xaxis,series.yaxis);ctx.restore()}function getFillStyle(filloptions,seriesColor,bottom,top){var fill=filloptions.fill;if(!fill)return null;if(filloptions.fillColor)return getColorOrGradient(filloptions.fillColor,bottom,top,seriesColor);var c=$.color.parse(seriesColor);c.a=typeof fill=="number"?fill:.4;c.normalize();return c.toString()}function insertLegend(){if(options.legend.container!=null){$(options.legend.container).html("")}else{placeholder.find(".legend").remove()}if(!options.legend.show){return}var fragments=[],entries=[],rowStarted=false,lf=options.legend.labelFormatter,s,label;for(var i=0;i<series.length;++i){s=series[i];if(s.label){label=lf?lf(s.label,s):s.label;if(label){entries.push({label:label,color:s.color})}}}if(options.legend.sorted){if($.isFunction(options.legend.sorted)){entries.sort(options.legend.sorted)}else if(options.legend.sorted=="reverse"){entries.reverse()}else{var ascending=options.legend.sorted!="descending";entries.sort(function(a,b){return a.label==b.label?0:a.label<b.label!=ascending?1:-1})}}for(var i=0;i<entries.length;++i){var entry=entries[i];if(i%options.legend.noColumns==0){if(rowStarted)fragments.push("</tr>");fragments.push("<tr>");rowStarted=true}fragments.push('<td class="legendColorBox"><div style="border:1px solid '+options.legend.labelBoxBorderColor+';padding:1px"><div style="width:4px;height:0;border:5px solid '+entry.color+';overflow:hidden"></div></div></td>'+'<td class="legendLabel">'+entry.label+"</td>")}if(rowStarted)fragments.push("</tr>");if(fragments.length==0)return;var table='<table style="font-size:smaller;color:'+options.grid.color+'">'+fragments.join("")+"</table>";if(options.legend.container!=null)$(options.legend.container).html(table);else{var pos="",p=options.legend.position,m=options.legend.margin;if(m[0]==null)m=[m,m];if(p.charAt(0)=="n")pos+="top:"+(m[1]+plotOffset.top)+"px;";else if(p.charAt(0)=="s")pos+="bottom:"+(m[1]+plotOffset.bottom)+"px;";if(p.charAt(1)=="e")pos+="right:"+(m[0]+plotOffset.right)+"px;";else if(p.charAt(1)=="w")pos+="left:"+(m[0]+plotOffset.left)+"px;";var legend=$('<div class="legend">'+table.replace('style="','style="position:absolute;'+pos+";")+"</div>").appendTo(placeholder);if(options.legend.backgroundOpacity!=0){var c=options.legend.backgroundColor;if(c==null){c=options.grid.backgroundColor;if(c&&typeof c=="string")c=$.color.parse(c);else c=$.color.extract(legend,"background-color");c.a=1;c=c.toString()}var div=legend.children();$('<div style="position:absolute;width:'+div.width()+"px;height:"+div.height()+"px;"+pos+"background-color:"+c+';"> </div>').prependTo(legend).css("opacity",options.legend.backgroundOpacity)}}}var highlights=[],redrawTimeout=null;function findNearbyItem(mouseX,mouseY,seriesFilter){var maxDistance=options.grid.mouseActiveRadius,smallestDistance=maxDistance*maxDistance+1,item=null,foundPoint=false,i,j,ps;for(i=series.length-1;i>=0;--i){if(!seriesFilter(series[i]))continue;var s=series[i],axisx=s.xaxis,axisy=s.yaxis,points=s.datapoints.points,mx=axisx.c2p(mouseX),my=axisy.c2p(mouseY),maxx=maxDistance/axisx.scale,maxy=maxDistance/axisy.scale;ps=s.datapoints.pointsize;if(axisx.options.inverseTransform)maxx=Number.MAX_VALUE;if(axisy.options.inverseTransform)maxy=Number.MAX_VALUE;if(s.lines.show||s.points.show){for(j=0;j<points.length;j+=ps){var x=points[j],y=points[j+1];if(x==null)continue;if(x-mx>maxx||x-mx<-maxx||y-my>maxy||y-my<-maxy)continue;var dx=Math.abs(axisx.p2c(x)-mouseX),dy=Math.abs(axisy.p2c(y)-mouseY),dist=dx*dx+dy*dy;if(dist<smallestDistance){smallestDistance=dist;item=[i,j/ps]}}}if(s.bars.show&&!item){var barLeft,barRight;switch(s.bars.align){case"left":barLeft=0;break;case"right":barLeft=-s.bars.barWidth;break;default:barLeft=-s.bars.barWidth/2}barRight=barLeft+s.bars.barWidth;for(j=0;j<points.length;j+=ps){var x=points[j],y=points[j+1],b=points[j+2];if(x==null)continue;if(series[i].bars.horizontal?mx<=Math.max(b,x)&&mx>=Math.min(b,x)&&my>=y+barLeft&&my<=y+barRight:mx>=x+barLeft&&mx<=x+barRight&&my>=Math.min(b,y)&&my<=Math.max(b,y))item=[i,j/ps]}}}if(item){i=item[0];j=item[1];ps=series[i].datapoints.pointsize;return{datapoint:series[i].datapoints.points.slice(j*ps,(j+1)*ps),dataIndex:j,series:series[i],seriesIndex:i}}return null}function onMouseMove(e){if(options.grid.hoverable)triggerClickHoverEvent("plothover",e,function(s){return s["hoverable"]!=false})}function onMouseLeave(e){if(options.grid.hoverable)triggerClickHoverEvent("plothover",e,function(s){return false})}function onClick(e){triggerClickHoverEvent("plotclick",e,function(s){return s["clickable"]!=false})}function triggerClickHoverEvent(eventname,event,seriesFilter){var offset=eventHolder.offset(),canvasX=event.pageX-offset.left-plotOffset.left,canvasY=event.pageY-offset.top-plotOffset.top,pos=canvasToAxisCoords({left:canvasX,top:canvasY});pos.pageX=event.pageX;pos.pageY=event.pageY;var item=findNearbyItem(canvasX,canvasY,seriesFilter);if(item){item.pageX=parseInt(item.series.xaxis.p2c(item.datapoint[0])+offset.left+plotOffset.left,10);item.pageY=parseInt(item.series.yaxis.p2c(item.datapoint[1])+offset.top+plotOffset.top,10)}if(options.grid.autoHighlight){for(var i=0;i<highlights.length;++i){var h=highlights[i];if(h.auto==eventname&&!(item&&h.series==item.series&&h.point[0]==item.datapoint[0]&&h.point[1]==item.datapoint[1]))unhighlight(h.series,h.point)}if(item)highlight(item.series,item.datapoint,eventname)}placeholder.trigger(eventname,[pos,item])}function triggerRedrawOverlay(){var t=options.interaction.redrawOverlayInterval;if(t==-1){drawOverlay();return}if(!redrawTimeout)redrawTimeout=setTimeout(drawOverlay,t)}function drawOverlay(){redrawTimeout=null;octx.save();overlay.clear();octx.translate(plotOffset.left,plotOffset.top);var i,hi;for(i=0;i<highlights.length;++i){hi=highlights[i];if(hi.series.bars.show)drawBarHighlight(hi.series,hi.point);else drawPointHighlight(hi.series,hi.point)}octx.restore();executeHooks(hooks.drawOverlay,[octx])}function highlight(s,point,auto){if(typeof s=="number")s=series[s];if(typeof point=="number"){var ps=s.datapoints.pointsize;point=s.datapoints.points.slice(ps*point,ps*(point+1))}var i=indexOfHighlight(s,point);if(i==-1){highlights.push({series:s,point:point,auto:auto});triggerRedrawOverlay()}else if(!auto)highlights[i].auto=false}function unhighlight(s,point){if(s==null&&point==null){highlights=[];triggerRedrawOverlay();return}if(typeof s=="number")s=series[s];if(typeof point=="number"){var ps=s.datapoints.pointsize;point=s.datapoints.points.slice(ps*point,ps*(point+1))}var i=indexOfHighlight(s,point);if(i!=-1){highlights.splice(i,1);triggerRedrawOverlay()}}function indexOfHighlight(s,p){for(var i=0;i<highlights.length;++i){var h=highlights[i];if(h.series==s&&h.point[0]==p[0]&&h.point[1]==p[1])return i}return-1}function drawPointHighlight(series,point){var x=point[0],y=point[1],axisx=series.xaxis,axisy=series.yaxis,highlightColor=typeof series.highlightColor==="string"?series.highlightColor:$.color.parse(series.color).scale("a",.5).toString();if(x<axisx.min||x>axisx.max||y<axisy.min||y>axisy.max)return;var pointRadius=series.points.radius+series.points.lineWidth/2;octx.lineWidth=pointRadius;octx.strokeStyle=highlightColor;var radius=1.5*pointRadius;x=axisx.p2c(x);y=axisy.p2c(y);octx.beginPath();if(series.points.symbol=="circle")octx.arc(x,y,radius,0,2*Math.PI,false);else series.points.symbol(octx,x,y,radius,false);octx.closePath();octx.stroke()}function drawBarHighlight(series,point){var highlightColor=typeof series.highlightColor==="string"?series.highlightColor:$.color.parse(series.color).scale("a",.5).toString(),fillStyle=highlightColor,barLeft;switch(series.bars.align){case"left":barLeft=0;break;case"right":barLeft=-series.bars.barWidth;break;default:barLeft=-series.bars.barWidth/2}octx.lineWidth=series.bars.lineWidth;octx.strokeStyle=highlightColor;drawBar(point[0],point[1],point[2]||0,barLeft,barLeft+series.bars.barWidth,function(){return fillStyle},series.xaxis,series.yaxis,octx,series.bars.horizontal,series.bars.lineWidth)}function getColorOrGradient(spec,bottom,top,defaultColor){if(typeof spec=="string")return spec;else{var gradient=ctx.createLinearGradient(0,top,0,bottom);for(var i=0,l=spec.colors.length;i<l;++i){var c=spec.colors[i];if(typeof c!="string"){var co=$.color.parse(defaultColor);if(c.brightness!=null)co=co.scale("rgb",c.brightness);if(c.opacity!=null)co.a*=c.opacity;c=co.toString()}gradient.addColorStop(i/(l-1),c)}return gradient}}}$.plot=function(placeholder,data,options){var plot=new Plot($(placeholder),data,options,$.plot.plugins);return plot};$.plot.version="0.8.2";$.plot.plugins=[];$.fn.plot=function(data,options){return this.each(function(){$.plot(this,data,options)})};function floorInBase(n,base){return base*Math.floor(n/base)}})(jQuery); \ No newline at end of file diff --git a/lsp/plugins/jquery.flot.time.min.js b/lsp/plugins/jquery.flot.time.min.js new file mode 100644 index 00000000..aaf319c9 --- /dev/null +++ b/lsp/plugins/jquery.flot.time.min.js @@ -0,0 +1 @@ +(function($){var options={xaxis:{timezone:null,timeformat:null,twelveHourClock:false,monthNames:null}};function floorInBase(n,base){return base*Math.floor(n/base)}function formatDate(d,fmt,monthNames,dayNames){if(typeof d.strftime=="function"){return d.strftime(fmt)}var leftPad=function(n,pad){n=""+n;pad=""+(pad==null?"0":pad);return n.length==1?pad+n:n};var r=[];var escape=false;var hours=d.getHours();var isAM=hours<12;if(monthNames==null){monthNames=["Jan","Feb","Mar","Apr","May","Jun","Jul","Aug","Sep","Oct","Nov","Dec"]}if(dayNames==null){dayNames=["Sun","Mon","Tue","Wed","Thu","Fri","Sat"]}var hours12;if(hours>12){hours12=hours-12}else if(hours==0){hours12=12}else{hours12=hours}for(var i=0;i<fmt.length;++i){var c=fmt.charAt(i);if(escape){switch(c){case"a":c=""+dayNames[d.getDay()];break;case"b":c=""+monthNames[d.getMonth()];break;case"d":c=leftPad(d.getDate());break;case"e":c=leftPad(d.getDate()," ");break;case"h":case"H":c=leftPad(hours);break;case"I":c=leftPad(hours12);break;case"l":c=leftPad(hours12," ");break;case"m":c=leftPad(d.getMonth()+1);break;case"M":c=leftPad(d.getMinutes());break;case"q":c=""+(Math.floor(d.getMonth()/3)+1);break;case"S":c=leftPad(d.getSeconds());break;case"y":c=leftPad(d.getFullYear()%100);break;case"Y":c=""+d.getFullYear();break;case"p":c=isAM?""+"am":""+"pm";break;case"P":c=isAM?""+"AM":""+"PM";break;case"w":c=""+d.getDay();break}r.push(c);escape=false}else{if(c=="%"){escape=true}else{r.push(c)}}}return r.join("")}function makeUtcWrapper(d){function addProxyMethod(sourceObj,sourceMethod,targetObj,targetMethod){sourceObj[sourceMethod]=function(){return targetObj[targetMethod].apply(targetObj,arguments)}}var utc={date:d};if(d.strftime!=undefined){addProxyMethod(utc,"strftime",d,"strftime")}addProxyMethod(utc,"getTime",d,"getTime");addProxyMethod(utc,"setTime",d,"setTime");var props=["Date","Day","FullYear","Hours","Milliseconds","Minutes","Month","Seconds"];for(var p=0;p<props.length;p++){addProxyMethod(utc,"get"+props[p],d,"getUTC"+props[p]);addProxyMethod(utc,"set"+props[p],d,"setUTC"+props[p])}return utc}function dateGenerator(ts,opts){if(opts.timezone=="browser"){return new Date(ts)}else if(!opts.timezone||opts.timezone=="utc"){return makeUtcWrapper(new Date(ts))}else if(typeof timezoneJS!="undefined"&&typeof timezoneJS.Date!="undefined"){var d=new timezoneJS.Date;d.setTimezone(opts.timezone);d.setTime(ts);return d}else{return makeUtcWrapper(new Date(ts))}}var timeUnitSize={second:1e3,minute:60*1e3,hour:60*60*1e3,day:24*60*60*1e3,month:30*24*60*60*1e3,quarter:3*30*24*60*60*1e3,year:365.2425*24*60*60*1e3};var baseSpec=[[1,"second"],[2,"second"],[5,"second"],[10,"second"],[30,"second"],[1,"minute"],[2,"minute"],[5,"minute"],[10,"minute"],[30,"minute"],[1,"hour"],[2,"hour"],[4,"hour"],[8,"hour"],[12,"hour"],[1,"day"],[2,"day"],[3,"day"],[.25,"month"],[.5,"month"],[1,"month"],[2,"month"]];var specMonths=baseSpec.concat([[3,"month"],[6,"month"],[1,"year"]]);var specQuarters=baseSpec.concat([[1,"quarter"],[2,"quarter"],[1,"year"]]);function init(plot){plot.hooks.processOptions.push(function(plot,options){$.each(plot.getAxes(),function(axisName,axis){var opts=axis.options;if(opts.mode=="time"){axis.tickGenerator=function(axis){var ticks=[];var d=dateGenerator(axis.min,opts);var minSize=0;var spec=opts.tickSize&&opts.tickSize[1]==="quarter"||opts.minTickSize&&opts.minTickSize[1]==="quarter"?specQuarters:specMonths;if(opts.minTickSize!=null){if(typeof opts.tickSize=="number"){minSize=opts.tickSize}else{minSize=opts.minTickSize[0]*timeUnitSize[opts.minTickSize[1]]}}for(var i=0;i<spec.length-1;++i){if(axis.delta<(spec[i][0]*timeUnitSize[spec[i][1]]+spec[i+1][0]*timeUnitSize[spec[i+1][1]])/2&&spec[i][0]*timeUnitSize[spec[i][1]]>=minSize){break}}var size=spec[i][0];var unit=spec[i][1];if(unit=="year"){if(opts.minTickSize!=null&&opts.minTickSize[1]=="year"){size=Math.floor(opts.minTickSize[0])}else{var magn=Math.pow(10,Math.floor(Math.log(axis.delta/timeUnitSize.year)/Math.LN10));var norm=axis.delta/timeUnitSize.year/magn;if(norm<1.5){size=1}else if(norm<3){size=2}else if(norm<7.5){size=5}else{size=10}size*=magn}if(size<1){size=1}}axis.tickSize=opts.tickSize||[size,unit];var tickSize=axis.tickSize[0];unit=axis.tickSize[1];var step=tickSize*timeUnitSize[unit];if(unit=="second"){d.setSeconds(floorInBase(d.getSeconds(),tickSize))}else if(unit=="minute"){d.setMinutes(floorInBase(d.getMinutes(),tickSize))}else if(unit=="hour"){d.setHours(floorInBase(d.getHours(),tickSize))}else if(unit=="month"){d.setMonth(floorInBase(d.getMonth(),tickSize))}else if(unit=="quarter"){d.setMonth(3*floorInBase(d.getMonth()/3,tickSize))}else if(unit=="year"){d.setFullYear(floorInBase(d.getFullYear(),tickSize))}d.setMilliseconds(0);if(step>=timeUnitSize.minute){d.setSeconds(0)}if(step>=timeUnitSize.hour){d.setMinutes(0)}if(step>=timeUnitSize.day){d.setHours(0)}if(step>=timeUnitSize.day*4){d.setDate(1)}if(step>=timeUnitSize.month*2){d.setMonth(floorInBase(d.getMonth(),3))}if(step>=timeUnitSize.quarter*2){d.setMonth(floorInBase(d.getMonth(),6))}if(step>=timeUnitSize.year){d.setMonth(0)}var carry=0;var v=Number.NaN;var prev;do{prev=v;v=d.getTime();ticks.push(v);if(unit=="month"||unit=="quarter"){if(tickSize<1){d.setDate(1);var start=d.getTime();d.setMonth(d.getMonth()+(unit=="quarter"?3:1));var end=d.getTime();d.setTime(v+carry*timeUnitSize.hour+(end-start)*tickSize);carry=d.getHours();d.setHours(0)}else{d.setMonth(d.getMonth()+tickSize*(unit=="quarter"?3:1))}}else if(unit=="year"){d.setFullYear(d.getFullYear()+tickSize)}else{d.setTime(v+step)}}while(v<axis.max&&v!=prev);return ticks};axis.tickFormatter=function(v,axis){var d=dateGenerator(v,axis.options);if(opts.timeformat!=null){return formatDate(d,opts.timeformat,opts.monthNames,opts.dayNames)}var useQuarters=axis.options.tickSize&&axis.options.tickSize[1]=="quarter"||axis.options.minTickSize&&axis.options.minTickSize[1]=="quarter";var t=axis.tickSize[0]*timeUnitSize[axis.tickSize[1]];var span=axis.max-axis.min;var suffix=opts.twelveHourClock?" %p":"";var hourCode=opts.twelveHourClock?"%I":"%H";var fmt;if(t<timeUnitSize.minute){fmt=hourCode+":%M:%S"+suffix}else if(t<timeUnitSize.day){if(span<2*timeUnitSize.day){fmt=hourCode+":%M"+suffix}else{fmt="%b %d "+hourCode+":%M"+suffix}}else if(t<timeUnitSize.month){fmt="%b %d"}else if(useQuarters&&t<timeUnitSize.quarter||!useQuarters&&t<timeUnitSize.year){if(span<timeUnitSize.year){fmt="%b"}else{fmt="%b %Y"}}else if(useQuarters&&t<timeUnitSize.year){if(span<timeUnitSize.year){fmt="Q%q"}else{fmt="Q%q %Y"}}else{fmt="%Y"}var rt=formatDate(d,fmt,opts.monthNames,opts.dayNames);return rt}}})})}$.plot.plugins.push({init:init,options:options,name:"time",version:"1.0"});$.plot.formatDate=formatDate})(jQuery); \ No newline at end of file diff --git a/lsp/plugins/jquery.js b/lsp/plugins/jquery.js new file mode 100755 index 00000000..38837795 --- /dev/null +++ b/lsp/plugins/jquery.js @@ -0,0 +1,2 @@ +/*! jQuery v1.8.3 jquery.com | jquery.org/license */ +(function(e,t){function _(e){var t=M[e]={};return v.each(e.split(y),function(e,n){t[n]=!0}),t}function H(e,n,r){if(r===t&&e.nodeType===1){var i="data-"+n.replace(P,"-$1").toLowerCase();r=e.getAttribute(i);if(typeof r=="string"){try{r=r==="true"?!0:r==="false"?!1:r==="null"?null:+r+""===r?+r:D.test(r)?v.parseJSON(r):r}catch(s){}v.data(e,n,r)}else r=t}return r}function B(e){var t;for(t in e){if(t==="data"&&v.isEmptyObject(e[t]))continue;if(t!=="toJSON")return!1}return!0}function et(){return!1}function tt(){return!0}function ut(e){return!e||!e.parentNode||e.parentNode.nodeType===11}function at(e,t){do e=e[t];while(e&&e.nodeType!==1);return e}function ft(e,t,n){t=t||0;if(v.isFunction(t))return v.grep(e,function(e,r){var i=!!t.call(e,r,e);return i===n});if(t.nodeType)return v.grep(e,function(e,r){return e===t===n});if(typeof t=="string"){var r=v.grep(e,function(e){return e.nodeType===1});if(it.test(t))return v.filter(t,r,!n);t=v.filter(t,r)}return v.grep(e,function(e,r){return v.inArray(e,t)>=0===n})}function lt(e){var t=ct.split("|"),n=e.createDocumentFragment();if(n.createElement)while(t.length)n.createElement(t.pop());return n}function Lt(e,t){return e.getElementsByTagName(t)[0]||e.appendChild(e.ownerDocument.createElement(t))}function At(e,t){if(t.nodeType!==1||!v.hasData(e))return;var n,r,i,s=v._data(e),o=v._data(t,s),u=s.events;if(u){delete o.handle,o.events={};for(n in u)for(r=0,i=u[n].length;r<i;r++)v.event.add(t,n,u[n][r])}o.data&&(o.data=v.extend({},o.data))}function Ot(e,t){var n;if(t.nodeType!==1)return;t.clearAttributes&&t.clearAttributes(),t.mergeAttributes&&t.mergeAttributes(e),n=t.nodeName.toLowerCase(),n==="object"?(t.parentNode&&(t.outerHTML=e.outerHTML),v.support.html5Clone&&e.innerHTML&&!v.trim(t.innerHTML)&&(t.innerHTML=e.innerHTML)):n==="input"&&Et.test(e.type)?(t.defaultChecked=t.checked=e.checked,t.value!==e.value&&(t.value=e.value)):n==="option"?t.selected=e.defaultSelected:n==="input"||n==="textarea"?t.defaultValue=e.defaultValue:n==="script"&&t.text!==e.text&&(t.text=e.text),t.removeAttribute(v.expando)}function Mt(e){return typeof e.getElementsByTagName!="undefined"?e.getElementsByTagName("*"):typeof e.querySelectorAll!="undefined"?e.querySelectorAll("*"):[]}function _t(e){Et.test(e.type)&&(e.defaultChecked=e.checked)}function Qt(e,t){if(t in e)return t;var n=t.charAt(0).toUpperCase()+t.slice(1),r=t,i=Jt.length;while(i--){t=Jt[i]+n;if(t in e)return t}return r}function Gt(e,t){return e=t||e,v.css(e,"display")==="none"||!v.contains(e.ownerDocument,e)}function Yt(e,t){var n,r,i=[],s=0,o=e.length;for(;s<o;s++){n=e[s];if(!n.style)continue;i[s]=v._data(n,"olddisplay"),t?(!i[s]&&n.style.display==="none"&&(n.style.display=""),n.style.display===""&&Gt(n)&&(i[s]=v._data(n,"olddisplay",nn(n.nodeName)))):(r=Dt(n,"display"),!i[s]&&r!=="none"&&v._data(n,"olddisplay",r))}for(s=0;s<o;s++){n=e[s];if(!n.style)continue;if(!t||n.style.display==="none"||n.style.display==="")n.style.display=t?i[s]||"":"none"}return e}function Zt(e,t,n){var r=Rt.exec(t);return r?Math.max(0,r[1]-(n||0))+(r[2]||"px"):t}function en(e,t,n,r){var i=n===(r?"border":"content")?4:t==="width"?1:0,s=0;for(;i<4;i+=2)n==="margin"&&(s+=v.css(e,n+$t[i],!0)),r?(n==="content"&&(s-=parseFloat(Dt(e,"padding"+$t[i]))||0),n!=="margin"&&(s-=parseFloat(Dt(e,"border"+$t[i]+"Width"))||0)):(s+=parseFloat(Dt(e,"padding"+$t[i]))||0,n!=="padding"&&(s+=parseFloat(Dt(e,"border"+$t[i]+"Width"))||0));return s}function tn(e,t,n){var r=t==="width"?e.offsetWidth:e.offsetHeight,i=!0,s=v.support.boxSizing&&v.css(e,"boxSizing")==="border-box";if(r<=0||r==null){r=Dt(e,t);if(r<0||r==null)r=e.style[t];if(Ut.test(r))return r;i=s&&(v.support.boxSizingReliable||r===e.style[t]),r=parseFloat(r)||0}return r+en(e,t,n||(s?"border":"content"),i)+"px"}function nn(e){if(Wt[e])return Wt[e];var t=v("<"+e+">").appendTo(i.body),n=t.css("display");t.remove();if(n==="none"||n===""){Pt=i.body.appendChild(Pt||v.extend(i.createElement("iframe"),{frameBorder:0,width:0,height:0}));if(!Ht||!Pt.createElement)Ht=(Pt.contentWindow||Pt.contentDocument).document,Ht.write("<!doctype html><html><body>"),Ht.close();t=Ht.body.appendChild(Ht.createElement(e)),n=Dt(t,"display"),i.body.removeChild(Pt)}return Wt[e]=n,n}function fn(e,t,n,r){var i;if(v.isArray(t))v.each(t,function(t,i){n||sn.test(e)?r(e,i):fn(e+"["+(typeof i=="object"?t:"")+"]",i,n,r)});else if(!n&&v.type(t)==="object")for(i in t)fn(e+"["+i+"]",t[i],n,r);else r(e,t)}function Cn(e){return function(t,n){typeof t!="string"&&(n=t,t="*");var r,i,s,o=t.toLowerCase().split(y),u=0,a=o.length;if(v.isFunction(n))for(;u<a;u++)r=o[u],s=/^\+/.test(r),s&&(r=r.substr(1)||"*"),i=e[r]=e[r]||[],i[s?"unshift":"push"](n)}}function kn(e,n,r,i,s,o){s=s||n.dataTypes[0],o=o||{},o[s]=!0;var u,a=e[s],f=0,l=a?a.length:0,c=e===Sn;for(;f<l&&(c||!u);f++)u=a[f](n,r,i),typeof u=="string"&&(!c||o[u]?u=t:(n.dataTypes.unshift(u),u=kn(e,n,r,i,u,o)));return(c||!u)&&!o["*"]&&(u=kn(e,n,r,i,"*",o)),u}function Ln(e,n){var r,i,s=v.ajaxSettings.flatOptions||{};for(r in n)n[r]!==t&&((s[r]?e:i||(i={}))[r]=n[r]);i&&v.extend(!0,e,i)}function An(e,n,r){var i,s,o,u,a=e.contents,f=e.dataTypes,l=e.responseFields;for(s in l)s in r&&(n[l[s]]=r[s]);while(f[0]==="*")f.shift(),i===t&&(i=e.mimeType||n.getResponseHeader("content-type"));if(i)for(s in a)if(a[s]&&a[s].test(i)){f.unshift(s);break}if(f[0]in r)o=f[0];else{for(s in r){if(!f[0]||e.converters[s+" "+f[0]]){o=s;break}u||(u=s)}o=o||u}if(o)return o!==f[0]&&f.unshift(o),r[o]}function On(e,t){var n,r,i,s,o=e.dataTypes.slice(),u=o[0],a={},f=0;e.dataFilter&&(t=e.dataFilter(t,e.dataType));if(o[1])for(n in e.converters)a[n.toLowerCase()]=e.converters[n];for(;i=o[++f];)if(i!=="*"){if(u!=="*"&&u!==i){n=a[u+" "+i]||a["* "+i];if(!n)for(r in a){s=r.split(" ");if(s[1]===i){n=a[u+" "+s[0]]||a["* "+s[0]];if(n){n===!0?n=a[r]:a[r]!==!0&&(i=s[0],o.splice(f--,0,i));break}}}if(n!==!0)if(n&&e["throws"])t=n(t);else try{t=n(t)}catch(l){return{state:"parsererror",error:n?l:"No conversion from "+u+" to "+i}}}u=i}return{state:"success",data:t}}function Fn(){try{return new e.XMLHttpRequest}catch(t){}}function In(){try{return new e.ActiveXObject("Microsoft.XMLHTTP")}catch(t){}}function $n(){return setTimeout(function(){qn=t},0),qn=v.now()}function Jn(e,t){v.each(t,function(t,n){var r=(Vn[t]||[]).concat(Vn["*"]),i=0,s=r.length;for(;i<s;i++)if(r[i].call(e,t,n))return})}function Kn(e,t,n){var r,i=0,s=0,o=Xn.length,u=v.Deferred().always(function(){delete a.elem}),a=function(){var t=qn||$n(),n=Math.max(0,f.startTime+f.duration-t),r=n/f.duration||0,i=1-r,s=0,o=f.tweens.length;for(;s<o;s++)f.tweens[s].run(i);return u.notifyWith(e,[f,i,n]),i<1&&o?n:(u.resolveWith(e,[f]),!1)},f=u.promise({elem:e,props:v.extend({},t),opts:v.extend(!0,{specialEasing:{}},n),originalProperties:t,originalOptions:n,startTime:qn||$n(),duration:n.duration,tweens:[],createTween:function(t,n,r){var i=v.Tween(e,f.opts,t,n,f.opts.specialEasing[t]||f.opts.easing);return f.tweens.push(i),i},stop:function(t){var n=0,r=t?f.tweens.length:0;for(;n<r;n++)f.tweens[n].run(1);return t?u.resolveWith(e,[f,t]):u.rejectWith(e,[f,t]),this}}),l=f.props;Qn(l,f.opts.specialEasing);for(;i<o;i++){r=Xn[i].call(f,e,l,f.opts);if(r)return r}return Jn(f,l),v.isFunction(f.opts.start)&&f.opts.start.call(e,f),v.fx.timer(v.extend(a,{anim:f,queue:f.opts.queue,elem:e})),f.progress(f.opts.progress).done(f.opts.done,f.opts.complete).fail(f.opts.fail).always(f.opts.always)}function Qn(e,t){var n,r,i,s,o;for(n in e){r=v.camelCase(n),i=t[r],s=e[n],v.isArray(s)&&(i=s[1],s=e[n]=s[0]),n!==r&&(e[r]=s,delete e[n]),o=v.cssHooks[r];if(o&&"expand"in o){s=o.expand(s),delete e[r];for(n in s)n in e||(e[n]=s[n],t[n]=i)}else t[r]=i}}function Gn(e,t,n){var r,i,s,o,u,a,f,l,c,h=this,p=e.style,d={},m=[],g=e.nodeType&&Gt(e);n.queue||(l=v._queueHooks(e,"fx"),l.unqueued==null&&(l.unqueued=0,c=l.empty.fire,l.empty.fire=function(){l.unqueued||c()}),l.unqueued++,h.always(function(){h.always(function(){l.unqueued--,v.queue(e,"fx").length||l.empty.fire()})})),e.nodeType===1&&("height"in t||"width"in t)&&(n.overflow=[p.overflow,p.overflowX,p.overflowY],v.css(e,"display")==="inline"&&v.css(e,"float")==="none"&&(!v.support.inlineBlockNeedsLayout||nn(e.nodeName)==="inline"?p.display="inline-block":p.zoom=1)),n.overflow&&(p.overflow="hidden",v.support.shrinkWrapBlocks||h.done(function(){p.overflow=n.overflow[0],p.overflowX=n.overflow[1],p.overflowY=n.overflow[2]}));for(r in t){s=t[r];if(Un.exec(s)){delete t[r],a=a||s==="toggle";if(s===(g?"hide":"show"))continue;m.push(r)}}o=m.length;if(o){u=v._data(e,"fxshow")||v._data(e,"fxshow",{}),"hidden"in u&&(g=u.hidden),a&&(u.hidden=!g),g?v(e).show():h.done(function(){v(e).hide()}),h.done(function(){var t;v.removeData(e,"fxshow",!0);for(t in d)v.style(e,t,d[t])});for(r=0;r<o;r++)i=m[r],f=h.createTween(i,g?u[i]:0),d[i]=u[i]||v.style(e,i),i in u||(u[i]=f.start,g&&(f.end=f.start,f.start=i==="width"||i==="height"?1:0))}}function Yn(e,t,n,r,i){return new Yn.prototype.init(e,t,n,r,i)}function Zn(e,t){var n,r={height:e},i=0;t=t?1:0;for(;i<4;i+=2-t)n=$t[i],r["margin"+n]=r["padding"+n]=e;return t&&(r.opacity=r.width=e),r}function tr(e){return v.isWindow(e)?e:e.nodeType===9?e.defaultView||e.parentWindow:!1}var n,r,i=e.document,s=e.location,o=e.navigator,u=e.jQuery,a=e.$,f=Array.prototype.push,l=Array.prototype.slice,c=Array.prototype.indexOf,h=Object.prototype.toString,p=Object.prototype.hasOwnProperty,d=String.prototype.trim,v=function(e,t){return new v.fn.init(e,t,n)},m=/[\-+]?(?:\d*\.|)\d+(?:[eE][\-+]?\d+|)/.source,g=/\S/,y=/\s+/,b=/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g,w=/^(?:[^#<]*(<[\w\W]+>)[^>]*$|#([\w\-]*)$)/,E=/^<(\w+)\s*\/?>(?:<\/\1>|)$/,S=/^[\],:{}\s]*$/,x=/(?:^|:|,)(?:\s*\[)+/g,T=/\\(?:["\\\/bfnrt]|u[\da-fA-F]{4})/g,N=/"[^"\\\r\n]*"|true|false|null|-?(?:\d\d*\.|)\d+(?:[eE][\-+]?\d+|)/g,C=/^-ms-/,k=/-([\da-z])/gi,L=function(e,t){return(t+"").toUpperCase()},A=function(){i.addEventListener?(i.removeEventListener("DOMContentLoaded",A,!1),v.ready()):i.readyState==="complete"&&(i.detachEvent("onreadystatechange",A),v.ready())},O={};v.fn=v.prototype={constructor:v,init:function(e,n,r){var s,o,u,a;if(!e)return this;if(e.nodeType)return this.context=this[0]=e,this.length=1,this;if(typeof e=="string"){e.charAt(0)==="<"&&e.charAt(e.length-1)===">"&&e.length>=3?s=[null,e,null]:s=w.exec(e);if(s&&(s[1]||!n)){if(s[1])return n=n instanceof v?n[0]:n,a=n&&n.nodeType?n.ownerDocument||n:i,e=v.parseHTML(s[1],a,!0),E.test(s[1])&&v.isPlainObject(n)&&this.attr.call(e,n,!0),v.merge(this,e);o=i.getElementById(s[2]);if(o&&o.parentNode){if(o.id!==s[2])return r.find(e);this.length=1,this[0]=o}return this.context=i,this.selector=e,this}return!n||n.jquery?(n||r).find(e):this.constructor(n).find(e)}return v.isFunction(e)?r.ready(e):(e.selector!==t&&(this.selector=e.selector,this.context=e.context),v.makeArray(e,this))},selector:"",jquery:"1.8.3",length:0,size:function(){return this.length},toArray:function(){return l.call(this)},get:function(e){return e==null?this.toArray():e<0?this[this.length+e]:this[e]},pushStack:function(e,t,n){var r=v.merge(this.constructor(),e);return r.prevObject=this,r.context=this.context,t==="find"?r.selector=this.selector+(this.selector?" ":"")+n:t&&(r.selector=this.selector+"."+t+"("+n+")"),r},each:function(e,t){return v.each(this,e,t)},ready:function(e){return v.ready.promise().done(e),this},eq:function(e){return e=+e,e===-1?this.slice(e):this.slice(e,e+1)},first:function(){return this.eq(0)},last:function(){return this.eq(-1)},slice:function(){return this.pushStack(l.apply(this,arguments),"slice",l.call(arguments).join(","))},map:function(e){return this.pushStack(v.map(this,function(t,n){return e.call(t,n,t)}))},end:function(){return this.prevObject||this.constructor(null)},push:f,sort:[].sort,splice:[].splice},v.fn.init.prototype=v.fn,v.extend=v.fn.extend=function(){var e,n,r,i,s,o,u=arguments[0]||{},a=1,f=arguments.length,l=!1;typeof u=="boolean"&&(l=u,u=arguments[1]||{},a=2),typeof u!="object"&&!v.isFunction(u)&&(u={}),f===a&&(u=this,--a);for(;a<f;a++)if((e=arguments[a])!=null)for(n in e){r=u[n],i=e[n];if(u===i)continue;l&&i&&(v.isPlainObject(i)||(s=v.isArray(i)))?(s?(s=!1,o=r&&v.isArray(r)?r:[]):o=r&&v.isPlainObject(r)?r:{},u[n]=v.extend(l,o,i)):i!==t&&(u[n]=i)}return u},v.extend({noConflict:function(t){return e.$===v&&(e.$=a),t&&e.jQuery===v&&(e.jQuery=u),v},isReady:!1,readyWait:1,holdReady:function(e){e?v.readyWait++:v.ready(!0)},ready:function(e){if(e===!0?--v.readyWait:v.isReady)return;if(!i.body)return setTimeout(v.ready,1);v.isReady=!0;if(e!==!0&&--v.readyWait>0)return;r.resolveWith(i,[v]),v.fn.trigger&&v(i).trigger("ready").off("ready")},isFunction:function(e){return v.type(e)==="function"},isArray:Array.isArray||function(e){return v.type(e)==="array"},isWindow:function(e){return e!=null&&e==e.window},isNumeric:function(e){return!isNaN(parseFloat(e))&&isFinite(e)},type:function(e){return e==null?String(e):O[h.call(e)]||"object"},isPlainObject:function(e){if(!e||v.type(e)!=="object"||e.nodeType||v.isWindow(e))return!1;try{if(e.constructor&&!p.call(e,"constructor")&&!p.call(e.constructor.prototype,"isPrototypeOf"))return!1}catch(n){return!1}var r;for(r in e);return r===t||p.call(e,r)},isEmptyObject:function(e){var t;for(t in e)return!1;return!0},error:function(e){throw new Error(e)},parseHTML:function(e,t,n){var r;return!e||typeof e!="string"?null:(typeof t=="boolean"&&(n=t,t=0),t=t||i,(r=E.exec(e))?[t.createElement(r[1])]:(r=v.buildFragment([e],t,n?null:[]),v.merge([],(r.cacheable?v.clone(r.fragment):r.fragment).childNodes)))},parseJSON:function(t){if(!t||typeof t!="string")return null;t=v.trim(t);if(e.JSON&&e.JSON.parse)return e.JSON.parse(t);if(S.test(t.replace(T,"@").replace(N,"]").replace(x,"")))return(new Function("return "+t))();v.error("Invalid JSON: "+t)},parseXML:function(n){var r,i;if(!n||typeof n!="string")return null;try{e.DOMParser?(i=new DOMParser,r=i.parseFromString(n,"text/xml")):(r=new ActiveXObject("Microsoft.XMLDOM"),r.async="false",r.loadXML(n))}catch(s){r=t}return(!r||!r.documentElement||r.getElementsByTagName("parsererror").length)&&v.error("Invalid XML: "+n),r},noop:function(){},globalEval:function(t){t&&g.test(t)&&(e.execScript||function(t){e.eval.call(e,t)})(t)},camelCase:function(e){return e.replace(C,"ms-").replace(k,L)},nodeName:function(e,t){return e.nodeName&&e.nodeName.toLowerCase()===t.toLowerCase()},each:function(e,n,r){var i,s=0,o=e.length,u=o===t||v.isFunction(e);if(r){if(u){for(i in e)if(n.apply(e[i],r)===!1)break}else for(;s<o;)if(n.apply(e[s++],r)===!1)break}else if(u){for(i in e)if(n.call(e[i],i,e[i])===!1)break}else for(;s<o;)if(n.call(e[s],s,e[s++])===!1)break;return e},trim:d&&!d.call("\ufeff\u00a0")?function(e){return e==null?"":d.call(e)}:function(e){return e==null?"":(e+"").replace(b,"")},makeArray:function(e,t){var n,r=t||[];return e!=null&&(n=v.type(e),e.length==null||n==="string"||n==="function"||n==="regexp"||v.isWindow(e)?f.call(r,e):v.merge(r,e)),r},inArray:function(e,t,n){var r;if(t){if(c)return c.call(t,e,n);r=t.length,n=n?n<0?Math.max(0,r+n):n:0;for(;n<r;n++)if(n in t&&t[n]===e)return n}return-1},merge:function(e,n){var r=n.length,i=e.length,s=0;if(typeof r=="number")for(;s<r;s++)e[i++]=n[s];else while(n[s]!==t)e[i++]=n[s++];return e.length=i,e},grep:function(e,t,n){var r,i=[],s=0,o=e.length;n=!!n;for(;s<o;s++)r=!!t(e[s],s),n!==r&&i.push(e[s]);return i},map:function(e,n,r){var i,s,o=[],u=0,a=e.length,f=e instanceof v||a!==t&&typeof a=="number"&&(a>0&&e[0]&&e[a-1]||a===0||v.isArray(e));if(f)for(;u<a;u++)i=n(e[u],u,r),i!=null&&(o[o.length]=i);else for(s in e)i=n(e[s],s,r),i!=null&&(o[o.length]=i);return o.concat.apply([],o)},guid:1,proxy:function(e,n){var r,i,s;return typeof n=="string"&&(r=e[n],n=e,e=r),v.isFunction(e)?(i=l.call(arguments,2),s=function(){return e.apply(n,i.concat(l.call(arguments)))},s.guid=e.guid=e.guid||v.guid++,s):t},access:function(e,n,r,i,s,o,u){var a,f=r==null,l=0,c=e.length;if(r&&typeof r=="object"){for(l in r)v.access(e,n,l,r[l],1,o,i);s=1}else if(i!==t){a=u===t&&v.isFunction(i),f&&(a?(a=n,n=function(e,t,n){return a.call(v(e),n)}):(n.call(e,i),n=null));if(n)for(;l<c;l++)n(e[l],r,a?i.call(e[l],l,n(e[l],r)):i,u);s=1}return s?e:f?n.call(e):c?n(e[0],r):o},now:function(){return(new Date).getTime()}}),v.ready.promise=function(t){if(!r){r=v.Deferred();if(i.readyState==="complete")setTimeout(v.ready,1);else if(i.addEventListener)i.addEventListener("DOMContentLoaded",A,!1),e.addEventListener("load",v.ready,!1);else{i.attachEvent("onreadystatechange",A),e.attachEvent("onload",v.ready);var n=!1;try{n=e.frameElement==null&&i.documentElement}catch(s){}n&&n.doScroll&&function o(){if(!v.isReady){try{n.doScroll("left")}catch(e){return setTimeout(o,50)}v.ready()}}()}}return r.promise(t)},v.each("Boolean Number String Function Array Date RegExp Object".split(" "),function(e,t){O["[object "+t+"]"]=t.toLowerCase()}),n=v(i);var M={};v.Callbacks=function(e){e=typeof e=="string"?M[e]||_(e):v.extend({},e);var n,r,i,s,o,u,a=[],f=!e.once&&[],l=function(t){n=e.memory&&t,r=!0,u=s||0,s=0,o=a.length,i=!0;for(;a&&u<o;u++)if(a[u].apply(t[0],t[1])===!1&&e.stopOnFalse){n=!1;break}i=!1,a&&(f?f.length&&l(f.shift()):n?a=[]:c.disable())},c={add:function(){if(a){var t=a.length;(function r(t){v.each(t,function(t,n){var i=v.type(n);i==="function"?(!e.unique||!c.has(n))&&a.push(n):n&&n.length&&i!=="string"&&r(n)})})(arguments),i?o=a.length:n&&(s=t,l(n))}return this},remove:function(){return a&&v.each(arguments,function(e,t){var n;while((n=v.inArray(t,a,n))>-1)a.splice(n,1),i&&(n<=o&&o--,n<=u&&u--)}),this},has:function(e){return v.inArray(e,a)>-1},empty:function(){return a=[],this},disable:function(){return a=f=n=t,this},disabled:function(){return!a},lock:function(){return f=t,n||c.disable(),this},locked:function(){return!f},fireWith:function(e,t){return t=t||[],t=[e,t.slice?t.slice():t],a&&(!r||f)&&(i?f.push(t):l(t)),this},fire:function(){return c.fireWith(this,arguments),this},fired:function(){return!!r}};return c},v.extend({Deferred:function(e){var t=[["resolve","done",v.Callbacks("once memory"),"resolved"],["reject","fail",v.Callbacks("once memory"),"rejected"],["notify","progress",v.Callbacks("memory")]],n="pending",r={state:function(){return n},always:function(){return i.done(arguments).fail(arguments),this},then:function(){var e=arguments;return v.Deferred(function(n){v.each(t,function(t,r){var s=r[0],o=e[t];i[r[1]](v.isFunction(o)?function(){var e=o.apply(this,arguments);e&&v.isFunction(e.promise)?e.promise().done(n.resolve).fail(n.reject).progress(n.notify):n[s+"With"](this===i?n:this,[e])}:n[s])}),e=null}).promise()},promise:function(e){return e!=null?v.extend(e,r):r}},i={};return r.pipe=r.then,v.each(t,function(e,s){var o=s[2],u=s[3];r[s[1]]=o.add,u&&o.add(function(){n=u},t[e^1][2].disable,t[2][2].lock),i[s[0]]=o.fire,i[s[0]+"With"]=o.fireWith}),r.promise(i),e&&e.call(i,i),i},when:function(e){var t=0,n=l.call(arguments),r=n.length,i=r!==1||e&&v.isFunction(e.promise)?r:0,s=i===1?e:v.Deferred(),o=function(e,t,n){return function(r){t[e]=this,n[e]=arguments.length>1?l.call(arguments):r,n===u?s.notifyWith(t,n):--i||s.resolveWith(t,n)}},u,a,f;if(r>1){u=new Array(r),a=new Array(r),f=new Array(r);for(;t<r;t++)n[t]&&v.isFunction(n[t].promise)?n[t].promise().done(o(t,f,n)).fail(s.reject).progress(o(t,a,u)):--i}return i||s.resolveWith(f,n),s.promise()}}),v.support=function(){var t,n,r,s,o,u,a,f,l,c,h,p=i.createElement("div");p.setAttribute("className","t"),p.innerHTML=" <link/><table></table><a href='/a'>a</a><input type='checkbox'/>",n=p.getElementsByTagName("*"),r=p.getElementsByTagName("a")[0];if(!n||!r||!n.length)return{};s=i.createElement("select"),o=s.appendChild(i.createElement("option")),u=p.getElementsByTagName("input")[0],r.style.cssText="top:1px;float:left;opacity:.5",t={leadingWhitespace:p.firstChild.nodeType===3,tbody:!p.getElementsByTagName("tbody").length,htmlSerialize:!!p.getElementsByTagName("link").length,style:/top/.test(r.getAttribute("style")),hrefNormalized:r.getAttribute("href")==="/a",opacity:/^0.5/.test(r.style.opacity),cssFloat:!!r.style.cssFloat,checkOn:u.value==="on",optSelected:o.selected,getSetAttribute:p.className!=="t",enctype:!!i.createElement("form").enctype,html5Clone:i.createElement("nav").cloneNode(!0).outerHTML!=="<:nav></:nav>",boxModel:i.compatMode==="CSS1Compat",submitBubbles:!0,changeBubbles:!0,focusinBubbles:!1,deleteExpando:!0,noCloneEvent:!0,inlineBlockNeedsLayout:!1,shrinkWrapBlocks:!1,reliableMarginRight:!0,boxSizingReliable:!0,pixelPosition:!1},u.checked=!0,t.noCloneChecked=u.cloneNode(!0).checked,s.disabled=!0,t.optDisabled=!o.disabled;try{delete p.test}catch(d){t.deleteExpando=!1}!p.addEventListener&&p.attachEvent&&p.fireEvent&&(p.attachEvent("onclick",h=function(){t.noCloneEvent=!1}),p.cloneNode(!0).fireEvent("onclick"),p.detachEvent("onclick",h)),u=i.createElement("input"),u.value="t",u.setAttribute("type","radio"),t.radioValue=u.value==="t",u.setAttribute("checked","checked"),u.setAttribute("name","t"),p.appendChild(u),a=i.createDocumentFragment(),a.appendChild(p.lastChild),t.checkClone=a.cloneNode(!0).cloneNode(!0).lastChild.checked,t.appendChecked=u.checked,a.removeChild(u),a.appendChild(p);if(p.attachEvent)for(l in{submit:!0,change:!0,focusin:!0})f="on"+l,c=f in p,c||(p.setAttribute(f,"return;"),c=typeof p[f]=="function"),t[l+"Bubbles"]=c;return v(function(){var n,r,s,o,u="padding:0;margin:0;border:0;display:block;overflow:hidden;",a=i.getElementsByTagName("body")[0];if(!a)return;n=i.createElement("div"),n.style.cssText="visibility:hidden;border:0;width:0;height:0;position:static;top:0;margin-top:1px",a.insertBefore(n,a.firstChild),r=i.createElement("div"),n.appendChild(r),r.innerHTML="<table><tr><td></td><td>t</td></tr></table>",s=r.getElementsByTagName("td"),s[0].style.cssText="padding:0;margin:0;border:0;display:none",c=s[0].offsetHeight===0,s[0].style.display="",s[1].style.display="none",t.reliableHiddenOffsets=c&&s[0].offsetHeight===0,r.innerHTML="",r.style.cssText="box-sizing:border-box;-moz-box-sizing:border-box;-webkit-box-sizing:border-box;padding:1px;border:1px;display:block;width:4px;margin-top:1%;position:absolute;top:1%;",t.boxSizing=r.offsetWidth===4,t.doesNotIncludeMarginInBodyOffset=a.offsetTop!==1,e.getComputedStyle&&(t.pixelPosition=(e.getComputedStyle(r,null)||{}).top!=="1%",t.boxSizingReliable=(e.getComputedStyle(r,null)||{width:"4px"}).width==="4px",o=i.createElement("div"),o.style.cssText=r.style.cssText=u,o.style.marginRight=o.style.width="0",r.style.width="1px",r.appendChild(o),t.reliableMarginRight=!parseFloat((e.getComputedStyle(o,null)||{}).marginRight)),typeof r.style.zoom!="undefined"&&(r.innerHTML="",r.style.cssText=u+"width:1px;padding:1px;display:inline;zoom:1",t.inlineBlockNeedsLayout=r.offsetWidth===3,r.style.display="block",r.style.overflow="visible",r.innerHTML="<div></div>",r.firstChild.style.width="5px",t.shrinkWrapBlocks=r.offsetWidth!==3,n.style.zoom=1),a.removeChild(n),n=r=s=o=null}),a.removeChild(p),n=r=s=o=u=a=p=null,t}();var D=/(?:\{[\s\S]*\}|\[[\s\S]*\])$/,P=/([A-Z])/g;v.extend({cache:{},deletedIds:[],uuid:0,expando:"jQuery"+(v.fn.jquery+Math.random()).replace(/\D/g,""),noData:{embed:!0,object:"clsid:D27CDB6E-AE6D-11cf-96B8-444553540000",applet:!0},hasData:function(e){return e=e.nodeType?v.cache[e[v.expando]]:e[v.expando],!!e&&!B(e)},data:function(e,n,r,i){if(!v.acceptData(e))return;var s,o,u=v.expando,a=typeof n=="string",f=e.nodeType,l=f?v.cache:e,c=f?e[u]:e[u]&&u;if((!c||!l[c]||!i&&!l[c].data)&&a&&r===t)return;c||(f?e[u]=c=v.deletedIds.pop()||v.guid++:c=u),l[c]||(l[c]={},f||(l[c].toJSON=v.noop));if(typeof n=="object"||typeof n=="function")i?l[c]=v.extend(l[c],n):l[c].data=v.extend(l[c].data,n);return s=l[c],i||(s.data||(s.data={}),s=s.data),r!==t&&(s[v.camelCase(n)]=r),a?(o=s[n],o==null&&(o=s[v.camelCase(n)])):o=s,o},removeData:function(e,t,n){if(!v.acceptData(e))return;var r,i,s,o=e.nodeType,u=o?v.cache:e,a=o?e[v.expando]:v.expando;if(!u[a])return;if(t){r=n?u[a]:u[a].data;if(r){v.isArray(t)||(t in r?t=[t]:(t=v.camelCase(t),t in r?t=[t]:t=t.split(" ")));for(i=0,s=t.length;i<s;i++)delete r[t[i]];if(!(n?B:v.isEmptyObject)(r))return}}if(!n){delete u[a].data;if(!B(u[a]))return}o?v.cleanData([e],!0):v.support.deleteExpando||u!=u.window?delete u[a]:u[a]=null},_data:function(e,t,n){return v.data(e,t,n,!0)},acceptData:function(e){var t=e.nodeName&&v.noData[e.nodeName.toLowerCase()];return!t||t!==!0&&e.getAttribute("classid")===t}}),v.fn.extend({data:function(e,n){var r,i,s,o,u,a=this[0],f=0,l=null;if(e===t){if(this.length){l=v.data(a);if(a.nodeType===1&&!v._data(a,"parsedAttrs")){s=a.attributes;for(u=s.length;f<u;f++)o=s[f].name,o.indexOf("data-")||(o=v.camelCase(o.substring(5)),H(a,o,l[o]));v._data(a,"parsedAttrs",!0)}}return l}return typeof e=="object"?this.each(function(){v.data(this,e)}):(r=e.split(".",2),r[1]=r[1]?"."+r[1]:"",i=r[1]+"!",v.access(this,function(n){if(n===t)return l=this.triggerHandler("getData"+i,[r[0]]),l===t&&a&&(l=v.data(a,e),l=H(a,e,l)),l===t&&r[1]?this.data(r[0]):l;r[1]=n,this.each(function(){var t=v(this);t.triggerHandler("setData"+i,r),v.data(this,e,n),t.triggerHandler("changeData"+i,r)})},null,n,arguments.length>1,null,!1))},removeData:function(e){return this.each(function(){v.removeData(this,e)})}}),v.extend({queue:function(e,t,n){var r;if(e)return t=(t||"fx")+"queue",r=v._data(e,t),n&&(!r||v.isArray(n)?r=v._data(e,t,v.makeArray(n)):r.push(n)),r||[]},dequeue:function(e,t){t=t||"fx";var n=v.queue(e,t),r=n.length,i=n.shift(),s=v._queueHooks(e,t),o=function(){v.dequeue(e,t)};i==="inprogress"&&(i=n.shift(),r--),i&&(t==="fx"&&n.unshift("inprogress"),delete s.stop,i.call(e,o,s)),!r&&s&&s.empty.fire()},_queueHooks:function(e,t){var n=t+"queueHooks";return v._data(e,n)||v._data(e,n,{empty:v.Callbacks("once memory").add(function(){v.removeData(e,t+"queue",!0),v.removeData(e,n,!0)})})}}),v.fn.extend({queue:function(e,n){var r=2;return typeof e!="string"&&(n=e,e="fx",r--),arguments.length<r?v.queue(this[0],e):n===t?this:this.each(function(){var t=v.queue(this,e,n);v._queueHooks(this,e),e==="fx"&&t[0]!=="inprogress"&&v.dequeue(this,e)})},dequeue:function(e){return this.each(function(){v.dequeue(this,e)})},delay:function(e,t){return e=v.fx?v.fx.speeds[e]||e:e,t=t||"fx",this.queue(t,function(t,n){var r=setTimeout(t,e);n.stop=function(){clearTimeout(r)}})},clearQueue:function(e){return this.queue(e||"fx",[])},promise:function(e,n){var r,i=1,s=v.Deferred(),o=this,u=this.length,a=function(){--i||s.resolveWith(o,[o])};typeof e!="string"&&(n=e,e=t),e=e||"fx";while(u--)r=v._data(o[u],e+"queueHooks"),r&&r.empty&&(i++,r.empty.add(a));return a(),s.promise(n)}});var j,F,I,q=/[\t\r\n]/g,R=/\r/g,U=/^(?:button|input)$/i,z=/^(?:button|input|object|select|textarea)$/i,W=/^a(?:rea|)$/i,X=/^(?:autofocus|autoplay|async|checked|controls|defer|disabled|hidden|loop|multiple|open|readonly|required|scoped|selected)$/i,V=v.support.getSetAttribute;v.fn.extend({attr:function(e,t){return v.access(this,v.attr,e,t,arguments.length>1)},removeAttr:function(e){return this.each(function(){v.removeAttr(this,e)})},prop:function(e,t){return v.access(this,v.prop,e,t,arguments.length>1)},removeProp:function(e){return e=v.propFix[e]||e,this.each(function(){try{this[e]=t,delete this[e]}catch(n){}})},addClass:function(e){var t,n,r,i,s,o,u;if(v.isFunction(e))return this.each(function(t){v(this).addClass(e.call(this,t,this.className))});if(e&&typeof e=="string"){t=e.split(y);for(n=0,r=this.length;n<r;n++){i=this[n];if(i.nodeType===1)if(!i.className&&t.length===1)i.className=e;else{s=" "+i.className+" ";for(o=0,u=t.length;o<u;o++)s.indexOf(" "+t[o]+" ")<0&&(s+=t[o]+" ");i.className=v.trim(s)}}}return this},removeClass:function(e){var n,r,i,s,o,u,a;if(v.isFunction(e))return this.each(function(t){v(this).removeClass(e.call(this,t,this.className))});if(e&&typeof e=="string"||e===t){n=(e||"").split(y);for(u=0,a=this.length;u<a;u++){i=this[u];if(i.nodeType===1&&i.className){r=(" "+i.className+" ").replace(q," ");for(s=0,o=n.length;s<o;s++)while(r.indexOf(" "+n[s]+" ")>=0)r=r.replace(" "+n[s]+" "," ");i.className=e?v.trim(r):""}}}return this},toggleClass:function(e,t){var n=typeof e,r=typeof t=="boolean";return v.isFunction(e)?this.each(function(n){v(this).toggleClass(e.call(this,n,this.className,t),t)}):this.each(function(){if(n==="string"){var i,s=0,o=v(this),u=t,a=e.split(y);while(i=a[s++])u=r?u:!o.hasClass(i),o[u?"addClass":"removeClass"](i)}else if(n==="undefined"||n==="boolean")this.className&&v._data(this,"__className__",this.className),this.className=this.className||e===!1?"":v._data(this,"__className__")||""})},hasClass:function(e){var t=" "+e+" ",n=0,r=this.length;for(;n<r;n++)if(this[n].nodeType===1&&(" "+this[n].className+" ").replace(q," ").indexOf(t)>=0)return!0;return!1},val:function(e){var n,r,i,s=this[0];if(!arguments.length){if(s)return n=v.valHooks[s.type]||v.valHooks[s.nodeName.toLowerCase()],n&&"get"in n&&(r=n.get(s,"value"))!==t?r:(r=s.value,typeof r=="string"?r.replace(R,""):r==null?"":r);return}return i=v.isFunction(e),this.each(function(r){var s,o=v(this);if(this.nodeType!==1)return;i?s=e.call(this,r,o.val()):s=e,s==null?s="":typeof s=="number"?s+="":v.isArray(s)&&(s=v.map(s,function(e){return e==null?"":e+""})),n=v.valHooks[this.type]||v.valHooks[this.nodeName.toLowerCase()];if(!n||!("set"in n)||n.set(this,s,"value")===t)this.value=s})}}),v.extend({valHooks:{option:{get:function(e){var t=e.attributes.value;return!t||t.specified?e.value:e.text}},select:{get:function(e){var t,n,r=e.options,i=e.selectedIndex,s=e.type==="select-one"||i<0,o=s?null:[],u=s?i+1:r.length,a=i<0?u:s?i:0;for(;a<u;a++){n=r[a];if((n.selected||a===i)&&(v.support.optDisabled?!n.disabled:n.getAttribute("disabled")===null)&&(!n.parentNode.disabled||!v.nodeName(n.parentNode,"optgroup"))){t=v(n).val();if(s)return t;o.push(t)}}return o},set:function(e,t){var n=v.makeArray(t);return v(e).find("option").each(function(){this.selected=v.inArray(v(this).val(),n)>=0}),n.length||(e.selectedIndex=-1),n}}},attrFn:{},attr:function(e,n,r,i){var s,o,u,a=e.nodeType;if(!e||a===3||a===8||a===2)return;if(i&&v.isFunction(v.fn[n]))return v(e)[n](r);if(typeof e.getAttribute=="undefined")return v.prop(e,n,r);u=a!==1||!v.isXMLDoc(e),u&&(n=n.toLowerCase(),o=v.attrHooks[n]||(X.test(n)?F:j));if(r!==t){if(r===null){v.removeAttr(e,n);return}return o&&"set"in o&&u&&(s=o.set(e,r,n))!==t?s:(e.setAttribute(n,r+""),r)}return o&&"get"in o&&u&&(s=o.get(e,n))!==null?s:(s=e.getAttribute(n),s===null?t:s)},removeAttr:function(e,t){var n,r,i,s,o=0;if(t&&e.nodeType===1){r=t.split(y);for(;o<r.length;o++)i=r[o],i&&(n=v.propFix[i]||i,s=X.test(i),s||v.attr(e,i,""),e.removeAttribute(V?i:n),s&&n in e&&(e[n]=!1))}},attrHooks:{type:{set:function(e,t){if(U.test(e.nodeName)&&e.parentNode)v.error("type property can't be changed");else if(!v.support.radioValue&&t==="radio"&&v.nodeName(e,"input")){var n=e.value;return e.setAttribute("type",t),n&&(e.value=n),t}}},value:{get:function(e,t){return j&&v.nodeName(e,"button")?j.get(e,t):t in e?e.value:null},set:function(e,t,n){if(j&&v.nodeName(e,"button"))return j.set(e,t,n);e.value=t}}},propFix:{tabindex:"tabIndex",readonly:"readOnly","for":"htmlFor","class":"className",maxlength:"maxLength",cellspacing:"cellSpacing",cellpadding:"cellPadding",rowspan:"rowSpan",colspan:"colSpan",usemap:"useMap",frameborder:"frameBorder",contenteditable:"contentEditable"},prop:function(e,n,r){var i,s,o,u=e.nodeType;if(!e||u===3||u===8||u===2)return;return o=u!==1||!v.isXMLDoc(e),o&&(n=v.propFix[n]||n,s=v.propHooks[n]),r!==t?s&&"set"in s&&(i=s.set(e,r,n))!==t?i:e[n]=r:s&&"get"in s&&(i=s.get(e,n))!==null?i:e[n]},propHooks:{tabIndex:{get:function(e){var n=e.getAttributeNode("tabindex");return n&&n.specified?parseInt(n.value,10):z.test(e.nodeName)||W.test(e.nodeName)&&e.href?0:t}}}}),F={get:function(e,n){var r,i=v.prop(e,n);return i===!0||typeof i!="boolean"&&(r=e.getAttributeNode(n))&&r.nodeValue!==!1?n.toLowerCase():t},set:function(e,t,n){var r;return t===!1?v.removeAttr(e,n):(r=v.propFix[n]||n,r in e&&(e[r]=!0),e.setAttribute(n,n.toLowerCase())),n}},V||(I={name:!0,id:!0,coords:!0},j=v.valHooks.button={get:function(e,n){var r;return r=e.getAttributeNode(n),r&&(I[n]?r.value!=="":r.specified)?r.value:t},set:function(e,t,n){var r=e.getAttributeNode(n);return r||(r=i.createAttribute(n),e.setAttributeNode(r)),r.value=t+""}},v.each(["width","height"],function(e,t){v.attrHooks[t]=v.extend(v.attrHooks[t],{set:function(e,n){if(n==="")return e.setAttribute(t,"auto"),n}})}),v.attrHooks.contenteditable={get:j.get,set:function(e,t,n){t===""&&(t="false"),j.set(e,t,n)}}),v.support.hrefNormalized||v.each(["href","src","width","height"],function(e,n){v.attrHooks[n]=v.extend(v.attrHooks[n],{get:function(e){var r=e.getAttribute(n,2);return r===null?t:r}})}),v.support.style||(v.attrHooks.style={get:function(e){return e.style.cssText.toLowerCase()||t},set:function(e,t){return e.style.cssText=t+""}}),v.support.optSelected||(v.propHooks.selected=v.extend(v.propHooks.selected,{get:function(e){var t=e.parentNode;return t&&(t.selectedIndex,t.parentNode&&t.parentNode.selectedIndex),null}})),v.support.enctype||(v.propFix.enctype="encoding"),v.support.checkOn||v.each(["radio","checkbox"],function(){v.valHooks[this]={get:function(e){return e.getAttribute("value")===null?"on":e.value}}}),v.each(["radio","checkbox"],function(){v.valHooks[this]=v.extend(v.valHooks[this],{set:function(e,t){if(v.isArray(t))return e.checked=v.inArray(v(e).val(),t)>=0}})});var $=/^(?:textarea|input|select)$/i,J=/^([^\.]*|)(?:\.(.+)|)$/,K=/(?:^|\s)hover(\.\S+|)\b/,Q=/^key/,G=/^(?:mouse|contextmenu)|click/,Y=/^(?:focusinfocus|focusoutblur)$/,Z=function(e){return v.event.special.hover?e:e.replace(K,"mouseenter$1 mouseleave$1")};v.event={add:function(e,n,r,i,s){var o,u,a,f,l,c,h,p,d,m,g;if(e.nodeType===3||e.nodeType===8||!n||!r||!(o=v._data(e)))return;r.handler&&(d=r,r=d.handler,s=d.selector),r.guid||(r.guid=v.guid++),a=o.events,a||(o.events=a={}),u=o.handle,u||(o.handle=u=function(e){return typeof v=="undefined"||!!e&&v.event.triggered===e.type?t:v.event.dispatch.apply(u.elem,arguments)},u.elem=e),n=v.trim(Z(n)).split(" ");for(f=0;f<n.length;f++){l=J.exec(n[f])||[],c=l[1],h=(l[2]||"").split(".").sort(),g=v.event.special[c]||{},c=(s?g.delegateType:g.bindType)||c,g=v.event.special[c]||{},p=v.extend({type:c,origType:l[1],data:i,handler:r,guid:r.guid,selector:s,needsContext:s&&v.expr.match.needsContext.test(s),namespace:h.join(".")},d),m=a[c];if(!m){m=a[c]=[],m.delegateCount=0;if(!g.setup||g.setup.call(e,i,h,u)===!1)e.addEventListener?e.addEventListener(c,u,!1):e.attachEvent&&e.attachEvent("on"+c,u)}g.add&&(g.add.call(e,p),p.handler.guid||(p.handler.guid=r.guid)),s?m.splice(m.delegateCount++,0,p):m.push(p),v.event.global[c]=!0}e=null},global:{},remove:function(e,t,n,r,i){var s,o,u,a,f,l,c,h,p,d,m,g=v.hasData(e)&&v._data(e);if(!g||!(h=g.events))return;t=v.trim(Z(t||"")).split(" ");for(s=0;s<t.length;s++){o=J.exec(t[s])||[],u=a=o[1],f=o[2];if(!u){for(u in h)v.event.remove(e,u+t[s],n,r,!0);continue}p=v.event.special[u]||{},u=(r?p.delegateType:p.bindType)||u,d=h[u]||[],l=d.length,f=f?new RegExp("(^|\\.)"+f.split(".").sort().join("\\.(?:.*\\.|)")+"(\\.|$)"):null;for(c=0;c<d.length;c++)m=d[c],(i||a===m.origType)&&(!n||n.guid===m.guid)&&(!f||f.test(m.namespace))&&(!r||r===m.selector||r==="**"&&m.selector)&&(d.splice(c--,1),m.selector&&d.delegateCount--,p.remove&&p.remove.call(e,m));d.length===0&&l!==d.length&&((!p.teardown||p.teardown.call(e,f,g.handle)===!1)&&v.removeEvent(e,u,g.handle),delete h[u])}v.isEmptyObject(h)&&(delete g.handle,v.removeData(e,"events",!0))},customEvent:{getData:!0,setData:!0,changeData:!0},trigger:function(n,r,s,o){if(!s||s.nodeType!==3&&s.nodeType!==8){var u,a,f,l,c,h,p,d,m,g,y=n.type||n,b=[];if(Y.test(y+v.event.triggered))return;y.indexOf("!")>=0&&(y=y.slice(0,-1),a=!0),y.indexOf(".")>=0&&(b=y.split("."),y=b.shift(),b.sort());if((!s||v.event.customEvent[y])&&!v.event.global[y])return;n=typeof n=="object"?n[v.expando]?n:new v.Event(y,n):new v.Event(y),n.type=y,n.isTrigger=!0,n.exclusive=a,n.namespace=b.join("."),n.namespace_re=n.namespace?new RegExp("(^|\\.)"+b.join("\\.(?:.*\\.|)")+"(\\.|$)"):null,h=y.indexOf(":")<0?"on"+y:"";if(!s){u=v.cache;for(f in u)u[f].events&&u[f].events[y]&&v.event.trigger(n,r,u[f].handle.elem,!0);return}n.result=t,n.target||(n.target=s),r=r!=null?v.makeArray(r):[],r.unshift(n),p=v.event.special[y]||{};if(p.trigger&&p.trigger.apply(s,r)===!1)return;m=[[s,p.bindType||y]];if(!o&&!p.noBubble&&!v.isWindow(s)){g=p.delegateType||y,l=Y.test(g+y)?s:s.parentNode;for(c=s;l;l=l.parentNode)m.push([l,g]),c=l;c===(s.ownerDocument||i)&&m.push([c.defaultView||c.parentWindow||e,g])}for(f=0;f<m.length&&!n.isPropagationStopped();f++)l=m[f][0],n.type=m[f][1],d=(v._data(l,"events")||{})[n.type]&&v._data(l,"handle"),d&&d.apply(l,r),d=h&&l[h],d&&v.acceptData(l)&&d.apply&&d.apply(l,r)===!1&&n.preventDefault();return n.type=y,!o&&!n.isDefaultPrevented()&&(!p._default||p._default.apply(s.ownerDocument,r)===!1)&&(y!=="click"||!v.nodeName(s,"a"))&&v.acceptData(s)&&h&&s[y]&&(y!=="focus"&&y!=="blur"||n.target.offsetWidth!==0)&&!v.isWindow(s)&&(c=s[h],c&&(s[h]=null),v.event.triggered=y,s[y](),v.event.triggered=t,c&&(s[h]=c)),n.result}return},dispatch:function(n){n=v.event.fix(n||e.event);var r,i,s,o,u,a,f,c,h,p,d=(v._data(this,"events")||{})[n.type]||[],m=d.delegateCount,g=l.call(arguments),y=!n.exclusive&&!n.namespace,b=v.event.special[n.type]||{},w=[];g[0]=n,n.delegateTarget=this;if(b.preDispatch&&b.preDispatch.call(this,n)===!1)return;if(m&&(!n.button||n.type!=="click"))for(s=n.target;s!=this;s=s.parentNode||this)if(s.disabled!==!0||n.type!=="click"){u={},f=[];for(r=0;r<m;r++)c=d[r],h=c.selector,u[h]===t&&(u[h]=c.needsContext?v(h,this).index(s)>=0:v.find(h,this,null,[s]).length),u[h]&&f.push(c);f.length&&w.push({elem:s,matches:f})}d.length>m&&w.push({elem:this,matches:d.slice(m)});for(r=0;r<w.length&&!n.isPropagationStopped();r++){a=w[r],n.currentTarget=a.elem;for(i=0;i<a.matches.length&&!n.isImmediatePropagationStopped();i++){c=a.matches[i];if(y||!n.namespace&&!c.namespace||n.namespace_re&&n.namespace_re.test(c.namespace))n.data=c.data,n.handleObj=c,o=((v.event.special[c.origType]||{}).handle||c.handler).apply(a.elem,g),o!==t&&(n.result=o,o===!1&&(n.preventDefault(),n.stopPropagation()))}}return b.postDispatch&&b.postDispatch.call(this,n),n.result},props:"attrChange attrName relatedNode srcElement altKey bubbles cancelable ctrlKey currentTarget eventPhase metaKey relatedTarget shiftKey target timeStamp view which".split(" "),fixHooks:{},keyHooks:{props:"char charCode key keyCode".split(" "),filter:function(e,t){return e.which==null&&(e.which=t.charCode!=null?t.charCode:t.keyCode),e}},mouseHooks:{props:"button buttons clientX clientY fromElement offsetX offsetY pageX pageY screenX screenY toElement".split(" "),filter:function(e,n){var r,s,o,u=n.button,a=n.fromElement;return e.pageX==null&&n.clientX!=null&&(r=e.target.ownerDocument||i,s=r.documentElement,o=r.body,e.pageX=n.clientX+(s&&s.scrollLeft||o&&o.scrollLeft||0)-(s&&s.clientLeft||o&&o.clientLeft||0),e.pageY=n.clientY+(s&&s.scrollTop||o&&o.scrollTop||0)-(s&&s.clientTop||o&&o.clientTop||0)),!e.relatedTarget&&a&&(e.relatedTarget=a===e.target?n.toElement:a),!e.which&&u!==t&&(e.which=u&1?1:u&2?3:u&4?2:0),e}},fix:function(e){if(e[v.expando])return e;var t,n,r=e,s=v.event.fixHooks[e.type]||{},o=s.props?this.props.concat(s.props):this.props;e=v.Event(r);for(t=o.length;t;)n=o[--t],e[n]=r[n];return e.target||(e.target=r.srcElement||i),e.target.nodeType===3&&(e.target=e.target.parentNode),e.metaKey=!!e.metaKey,s.filter?s.filter(e,r):e},special:{load:{noBubble:!0},focus:{delegateType:"focusin"},blur:{delegateType:"focusout"},beforeunload:{setup:function(e,t,n){v.isWindow(this)&&(this.onbeforeunload=n)},teardown:function(e,t){this.onbeforeunload===t&&(this.onbeforeunload=null)}}},simulate:function(e,t,n,r){var i=v.extend(new v.Event,n,{type:e,isSimulated:!0,originalEvent:{}});r?v.event.trigger(i,null,t):v.event.dispatch.call(t,i),i.isDefaultPrevented()&&n.preventDefault()}},v.event.handle=v.event.dispatch,v.removeEvent=i.removeEventListener?function(e,t,n){e.removeEventListener&&e.removeEventListener(t,n,!1)}:function(e,t,n){var r="on"+t;e.detachEvent&&(typeof e[r]=="undefined"&&(e[r]=null),e.detachEvent(r,n))},v.Event=function(e,t){if(!(this instanceof v.Event))return new v.Event(e,t);e&&e.type?(this.originalEvent=e,this.type=e.type,this.isDefaultPrevented=e.defaultPrevented||e.returnValue===!1||e.getPreventDefault&&e.getPreventDefault()?tt:et):this.type=e,t&&v.extend(this,t),this.timeStamp=e&&e.timeStamp||v.now(),this[v.expando]=!0},v.Event.prototype={preventDefault:function(){this.isDefaultPrevented=tt;var e=this.originalEvent;if(!e)return;e.preventDefault?e.preventDefault():e.returnValue=!1},stopPropagation:function(){this.isPropagationStopped=tt;var e=this.originalEvent;if(!e)return;e.stopPropagation&&e.stopPropagation(),e.cancelBubble=!0},stopImmediatePropagation:function(){this.isImmediatePropagationStopped=tt,this.stopPropagation()},isDefaultPrevented:et,isPropagationStopped:et,isImmediatePropagationStopped:et},v.each({mouseenter:"mouseover",mouseleave:"mouseout"},function(e,t){v.event.special[e]={delegateType:t,bindType:t,handle:function(e){var n,r=this,i=e.relatedTarget,s=e.handleObj,o=s.selector;if(!i||i!==r&&!v.contains(r,i))e.type=s.origType,n=s.handler.apply(this,arguments),e.type=t;return n}}}),v.support.submitBubbles||(v.event.special.submit={setup:function(){if(v.nodeName(this,"form"))return!1;v.event.add(this,"click._submit keypress._submit",function(e){var n=e.target,r=v.nodeName(n,"input")||v.nodeName(n,"button")?n.form:t;r&&!v._data(r,"_submit_attached")&&(v.event.add(r,"submit._submit",function(e){e._submit_bubble=!0}),v._data(r,"_submit_attached",!0))})},postDispatch:function(e){e._submit_bubble&&(delete e._submit_bubble,this.parentNode&&!e.isTrigger&&v.event.simulate("submit",this.parentNode,e,!0))},teardown:function(){if(v.nodeName(this,"form"))return!1;v.event.remove(this,"._submit")}}),v.support.changeBubbles||(v.event.special.change={setup:function(){if($.test(this.nodeName)){if(this.type==="checkbox"||this.type==="radio")v.event.add(this,"propertychange._change",function(e){e.originalEvent.propertyName==="checked"&&(this._just_changed=!0)}),v.event.add(this,"click._change",function(e){this._just_changed&&!e.isTrigger&&(this._just_changed=!1),v.event.simulate("change",this,e,!0)});return!1}v.event.add(this,"beforeactivate._change",function(e){var t=e.target;$.test(t.nodeName)&&!v._data(t,"_change_attached")&&(v.event.add(t,"change._change",function(e){this.parentNode&&!e.isSimulated&&!e.isTrigger&&v.event.simulate("change",this.parentNode,e,!0)}),v._data(t,"_change_attached",!0))})},handle:function(e){var t=e.target;if(this!==t||e.isSimulated||e.isTrigger||t.type!=="radio"&&t.type!=="checkbox")return e.handleObj.handler.apply(this,arguments)},teardown:function(){return v.event.remove(this,"._change"),!$.test(this.nodeName)}}),v.support.focusinBubbles||v.each({focus:"focusin",blur:"focusout"},function(e,t){var n=0,r=function(e){v.event.simulate(t,e.target,v.event.fix(e),!0)};v.event.special[t]={setup:function(){n++===0&&i.addEventListener(e,r,!0)},teardown:function(){--n===0&&i.removeEventListener(e,r,!0)}}}),v.fn.extend({on:function(e,n,r,i,s){var o,u;if(typeof e=="object"){typeof n!="string"&&(r=r||n,n=t);for(u in e)this.on(u,n,r,e[u],s);return this}r==null&&i==null?(i=n,r=n=t):i==null&&(typeof n=="string"?(i=r,r=t):(i=r,r=n,n=t));if(i===!1)i=et;else if(!i)return this;return s===1&&(o=i,i=function(e){return v().off(e),o.apply(this,arguments)},i.guid=o.guid||(o.guid=v.guid++)),this.each(function(){v.event.add(this,e,i,r,n)})},one:function(e,t,n,r){return this.on(e,t,n,r,1)},off:function(e,n,r){var i,s;if(e&&e.preventDefault&&e.handleObj)return i=e.handleObj,v(e.delegateTarget).off(i.namespace?i.origType+"."+i.namespace:i.origType,i.selector,i.handler),this;if(typeof e=="object"){for(s in e)this.off(s,n,e[s]);return this}if(n===!1||typeof n=="function")r=n,n=t;return r===!1&&(r=et),this.each(function(){v.event.remove(this,e,r,n)})},bind:function(e,t,n){return this.on(e,null,t,n)},unbind:function(e,t){return this.off(e,null,t)},live:function(e,t,n){return v(this.context).on(e,this.selector,t,n),this},die:function(e,t){return v(this.context).off(e,this.selector||"**",t),this},delegate:function(e,t,n,r){return this.on(t,e,n,r)},undelegate:function(e,t,n){return arguments.length===1?this.off(e,"**"):this.off(t,e||"**",n)},trigger:function(e,t){return this.each(function(){v.event.trigger(e,t,this)})},triggerHandler:function(e,t){if(this[0])return v.event.trigger(e,t,this[0],!0)},toggle:function(e){var t=arguments,n=e.guid||v.guid++,r=0,i=function(n){var i=(v._data(this,"lastToggle"+e.guid)||0)%r;return v._data(this,"lastToggle"+e.guid,i+1),n.preventDefault(),t[i].apply(this,arguments)||!1};i.guid=n;while(r<t.length)t[r++].guid=n;return this.click(i)},hover:function(e,t){return this.mouseenter(e).mouseleave(t||e)}}),v.each("blur focus focusin focusout load resize scroll unload click dblclick mousedown mouseup mousemove mouseover mouseout mouseenter mouseleave change select submit keydown keypress keyup error contextmenu".split(" "),function(e,t){v.fn[t]=function(e,n){return n==null&&(n=e,e=null),arguments.length>0?this.on(t,null,e,n):this.trigger(t)},Q.test(t)&&(v.event.fixHooks[t]=v.event.keyHooks),G.test(t)&&(v.event.fixHooks[t]=v.event.mouseHooks)}),function(e,t){function nt(e,t,n,r){n=n||[],t=t||g;var i,s,a,f,l=t.nodeType;if(!e||typeof e!="string")return n;if(l!==1&&l!==9)return[];a=o(t);if(!a&&!r)if(i=R.exec(e))if(f=i[1]){if(l===9){s=t.getElementById(f);if(!s||!s.parentNode)return n;if(s.id===f)return n.push(s),n}else if(t.ownerDocument&&(s=t.ownerDocument.getElementById(f))&&u(t,s)&&s.id===f)return n.push(s),n}else{if(i[2])return S.apply(n,x.call(t.getElementsByTagName(e),0)),n;if((f=i[3])&&Z&&t.getElementsByClassName)return S.apply(n,x.call(t.getElementsByClassName(f),0)),n}return vt(e.replace(j,"$1"),t,n,r,a)}function rt(e){return function(t){var n=t.nodeName.toLowerCase();return n==="input"&&t.type===e}}function it(e){return function(t){var n=t.nodeName.toLowerCase();return(n==="input"||n==="button")&&t.type===e}}function st(e){return N(function(t){return t=+t,N(function(n,r){var i,s=e([],n.length,t),o=s.length;while(o--)n[i=s[o]]&&(n[i]=!(r[i]=n[i]))})})}function ot(e,t,n){if(e===t)return n;var r=e.nextSibling;while(r){if(r===t)return-1;r=r.nextSibling}return 1}function ut(e,t){var n,r,s,o,u,a,f,l=L[d][e+" "];if(l)return t?0:l.slice(0);u=e,a=[],f=i.preFilter;while(u){if(!n||(r=F.exec(u)))r&&(u=u.slice(r[0].length)||u),a.push(s=[]);n=!1;if(r=I.exec(u))s.push(n=new m(r.shift())),u=u.slice(n.length),n.type=r[0].replace(j," ");for(o in i.filter)(r=J[o].exec(u))&&(!f[o]||(r=f[o](r)))&&(s.push(n=new m(r.shift())),u=u.slice(n.length),n.type=o,n.matches=r);if(!n)break}return t?u.length:u?nt.error(e):L(e,a).slice(0)}function at(e,t,r){var i=t.dir,s=r&&t.dir==="parentNode",o=w++;return t.first?function(t,n,r){while(t=t[i])if(s||t.nodeType===1)return e(t,n,r)}:function(t,r,u){if(!u){var a,f=b+" "+o+" ",l=f+n;while(t=t[i])if(s||t.nodeType===1){if((a=t[d])===l)return t.sizset;if(typeof a=="string"&&a.indexOf(f)===0){if(t.sizset)return t}else{t[d]=l;if(e(t,r,u))return t.sizset=!0,t;t.sizset=!1}}}else while(t=t[i])if(s||t.nodeType===1)if(e(t,r,u))return t}}function ft(e){return e.length>1?function(t,n,r){var i=e.length;while(i--)if(!e[i](t,n,r))return!1;return!0}:e[0]}function lt(e,t,n,r,i){var s,o=[],u=0,a=e.length,f=t!=null;for(;u<a;u++)if(s=e[u])if(!n||n(s,r,i))o.push(s),f&&t.push(u);return o}function ct(e,t,n,r,i,s){return r&&!r[d]&&(r=ct(r)),i&&!i[d]&&(i=ct(i,s)),N(function(s,o,u,a){var f,l,c,h=[],p=[],d=o.length,v=s||dt(t||"*",u.nodeType?[u]:u,[]),m=e&&(s||!t)?lt(v,h,e,u,a):v,g=n?i||(s?e:d||r)?[]:o:m;n&&n(m,g,u,a);if(r){f=lt(g,p),r(f,[],u,a),l=f.length;while(l--)if(c=f[l])g[p[l]]=!(m[p[l]]=c)}if(s){if(i||e){if(i){f=[],l=g.length;while(l--)(c=g[l])&&f.push(m[l]=c);i(null,g=[],f,a)}l=g.length;while(l--)(c=g[l])&&(f=i?T.call(s,c):h[l])>-1&&(s[f]=!(o[f]=c))}}else g=lt(g===o?g.splice(d,g.length):g),i?i(null,o,g,a):S.apply(o,g)})}function ht(e){var t,n,r,s=e.length,o=i.relative[e[0].type],u=o||i.relative[" "],a=o?1:0,f=at(function(e){return e===t},u,!0),l=at(function(e){return T.call(t,e)>-1},u,!0),h=[function(e,n,r){return!o&&(r||n!==c)||((t=n).nodeType?f(e,n,r):l(e,n,r))}];for(;a<s;a++)if(n=i.relative[e[a].type])h=[at(ft(h),n)];else{n=i.filter[e[a].type].apply(null,e[a].matches);if(n[d]){r=++a;for(;r<s;r++)if(i.relative[e[r].type])break;return ct(a>1&&ft(h),a>1&&e.slice(0,a-1).join("").replace(j,"$1"),n,a<r&&ht(e.slice(a,r)),r<s&&ht(e=e.slice(r)),r<s&&e.join(""))}h.push(n)}return ft(h)}function pt(e,t){var r=t.length>0,s=e.length>0,o=function(u,a,f,l,h){var p,d,v,m=[],y=0,w="0",x=u&&[],T=h!=null,N=c,C=u||s&&i.find.TAG("*",h&&a.parentNode||a),k=b+=N==null?1:Math.E;T&&(c=a!==g&&a,n=o.el);for(;(p=C[w])!=null;w++){if(s&&p){for(d=0;v=e[d];d++)if(v(p,a,f)){l.push(p);break}T&&(b=k,n=++o.el)}r&&((p=!v&&p)&&y--,u&&x.push(p))}y+=w;if(r&&w!==y){for(d=0;v=t[d];d++)v(x,m,a,f);if(u){if(y>0)while(w--)!x[w]&&!m[w]&&(m[w]=E.call(l));m=lt(m)}S.apply(l,m),T&&!u&&m.length>0&&y+t.length>1&&nt.uniqueSort(l)}return T&&(b=k,c=N),x};return o.el=0,r?N(o):o}function dt(e,t,n){var r=0,i=t.length;for(;r<i;r++)nt(e,t[r],n);return n}function vt(e,t,n,r,s){var o,u,f,l,c,h=ut(e),p=h.length;if(!r&&h.length===1){u=h[0]=h[0].slice(0);if(u.length>2&&(f=u[0]).type==="ID"&&t.nodeType===9&&!s&&i.relative[u[1].type]){t=i.find.ID(f.matches[0].replace($,""),t,s)[0];if(!t)return n;e=e.slice(u.shift().length)}for(o=J.POS.test(e)?-1:u.length-1;o>=0;o--){f=u[o];if(i.relative[l=f.type])break;if(c=i.find[l])if(r=c(f.matches[0].replace($,""),z.test(u[0].type)&&t.parentNode||t,s)){u.splice(o,1),e=r.length&&u.join("");if(!e)return S.apply(n,x.call(r,0)),n;break}}}return a(e,h)(r,t,s,n,z.test(e)),n}function mt(){}var n,r,i,s,o,u,a,f,l,c,h=!0,p="undefined",d=("sizcache"+Math.random()).replace(".",""),m=String,g=e.document,y=g.documentElement,b=0,w=0,E=[].pop,S=[].push,x=[].slice,T=[].indexOf||function(e){var t=0,n=this.length;for(;t<n;t++)if(this[t]===e)return t;return-1},N=function(e,t){return e[d]=t==null||t,e},C=function(){var e={},t=[];return N(function(n,r){return t.push(n)>i.cacheLength&&delete e[t.shift()],e[n+" "]=r},e)},k=C(),L=C(),A=C(),O="[\\x20\\t\\r\\n\\f]",M="(?:\\\\.|[-\\w]|[^\\x00-\\xa0])+",_=M.replace("w","w#"),D="([*^$|!~]?=)",P="\\["+O+"*("+M+")"+O+"*(?:"+D+O+"*(?:(['\"])((?:\\\\.|[^\\\\])*?)\\3|("+_+")|)|)"+O+"*\\]",H=":("+M+")(?:\\((?:(['\"])((?:\\\\.|[^\\\\])*?)\\2|([^()[\\]]*|(?:(?:"+P+")|[^:]|\\\\.)*|.*))\\)|)",B=":(even|odd|eq|gt|lt|nth|first|last)(?:\\("+O+"*((?:-\\d)?\\d*)"+O+"*\\)|)(?=[^-]|$)",j=new RegExp("^"+O+"+|((?:^|[^\\\\])(?:\\\\.)*)"+O+"+$","g"),F=new RegExp("^"+O+"*,"+O+"*"),I=new RegExp("^"+O+"*([\\x20\\t\\r\\n\\f>+~])"+O+"*"),q=new RegExp(H),R=/^(?:#([\w\-]+)|(\w+)|\.([\w\-]+))$/,U=/^:not/,z=/[\x20\t\r\n\f]*[+~]/,W=/:not\($/,X=/h\d/i,V=/input|select|textarea|button/i,$=/\\(?!\\)/g,J={ID:new RegExp("^#("+M+")"),CLASS:new RegExp("^\\.("+M+")"),NAME:new RegExp("^\\[name=['\"]?("+M+")['\"]?\\]"),TAG:new RegExp("^("+M.replace("w","w*")+")"),ATTR:new RegExp("^"+P),PSEUDO:new RegExp("^"+H),POS:new RegExp(B,"i"),CHILD:new RegExp("^:(only|nth|first|last)-child(?:\\("+O+"*(even|odd|(([+-]|)(\\d*)n|)"+O+"*(?:([+-]|)"+O+"*(\\d+)|))"+O+"*\\)|)","i"),needsContext:new RegExp("^"+O+"*[>+~]|"+B,"i")},K=function(e){var t=g.createElement("div");try{return e(t)}catch(n){return!1}finally{t=null}},Q=K(function(e){return e.appendChild(g.createComment("")),!e.getElementsByTagName("*").length}),G=K(function(e){return e.innerHTML="<a href='#'></a>",e.firstChild&&typeof e.firstChild.getAttribute!==p&&e.firstChild.getAttribute("href")==="#"}),Y=K(function(e){e.innerHTML="<select></select>";var t=typeof e.lastChild.getAttribute("multiple");return t!=="boolean"&&t!=="string"}),Z=K(function(e){return e.innerHTML="<div class='hidden e'></div><div class='hidden'></div>",!e.getElementsByClassName||!e.getElementsByClassName("e").length?!1:(e.lastChild.className="e",e.getElementsByClassName("e").length===2)}),et=K(function(e){e.id=d+0,e.innerHTML="<a name='"+d+"'></a><div name='"+d+"'></div>",y.insertBefore(e,y.firstChild);var t=g.getElementsByName&&g.getElementsByName(d).length===2+g.getElementsByName(d+0).length;return r=!g.getElementById(d),y.removeChild(e),t});try{x.call(y.childNodes,0)[0].nodeType}catch(tt){x=function(e){var t,n=[];for(;t=this[e];e++)n.push(t);return n}}nt.matches=function(e,t){return nt(e,null,null,t)},nt.matchesSelector=function(e,t){return nt(t,null,null,[e]).length>0},s=nt.getText=function(e){var t,n="",r=0,i=e.nodeType;if(i){if(i===1||i===9||i===11){if(typeof e.textContent=="string")return e.textContent;for(e=e.firstChild;e;e=e.nextSibling)n+=s(e)}else if(i===3||i===4)return e.nodeValue}else for(;t=e[r];r++)n+=s(t);return n},o=nt.isXML=function(e){var t=e&&(e.ownerDocument||e).documentElement;return t?t.nodeName!=="HTML":!1},u=nt.contains=y.contains?function(e,t){var n=e.nodeType===9?e.documentElement:e,r=t&&t.parentNode;return e===r||!!(r&&r.nodeType===1&&n.contains&&n.contains(r))}:y.compareDocumentPosition?function(e,t){return t&&!!(e.compareDocumentPosition(t)&16)}:function(e,t){while(t=t.parentNode)if(t===e)return!0;return!1},nt.attr=function(e,t){var n,r=o(e);return r||(t=t.toLowerCase()),(n=i.attrHandle[t])?n(e):r||Y?e.getAttribute(t):(n=e.getAttributeNode(t),n?typeof e[t]=="boolean"?e[t]?t:null:n.specified?n.value:null:null)},i=nt.selectors={cacheLength:50,createPseudo:N,match:J,attrHandle:G?{}:{href:function(e){return e.getAttribute("href",2)},type:function(e){return e.getAttribute("type")}},find:{ID:r?function(e,t,n){if(typeof t.getElementById!==p&&!n){var r=t.getElementById(e);return r&&r.parentNode?[r]:[]}}:function(e,n,r){if(typeof n.getElementById!==p&&!r){var i=n.getElementById(e);return i?i.id===e||typeof i.getAttributeNode!==p&&i.getAttributeNode("id").value===e?[i]:t:[]}},TAG:Q?function(e,t){if(typeof t.getElementsByTagName!==p)return t.getElementsByTagName(e)}:function(e,t){var n=t.getElementsByTagName(e);if(e==="*"){var r,i=[],s=0;for(;r=n[s];s++)r.nodeType===1&&i.push(r);return i}return n},NAME:et&&function(e,t){if(typeof t.getElementsByName!==p)return t.getElementsByName(name)},CLASS:Z&&function(e,t,n){if(typeof t.getElementsByClassName!==p&&!n)return t.getElementsByClassName(e)}},relative:{">":{dir:"parentNode",first:!0}," ":{dir:"parentNode"},"+":{dir:"previousSibling",first:!0},"~":{dir:"previousSibling"}},preFilter:{ATTR:function(e){return e[1]=e[1].replace($,""),e[3]=(e[4]||e[5]||"").replace($,""),e[2]==="~="&&(e[3]=" "+e[3]+" "),e.slice(0,4)},CHILD:function(e){return e[1]=e[1].toLowerCase(),e[1]==="nth"?(e[2]||nt.error(e[0]),e[3]=+(e[3]?e[4]+(e[5]||1):2*(e[2]==="even"||e[2]==="odd")),e[4]=+(e[6]+e[7]||e[2]==="odd")):e[2]&&nt.error(e[0]),e},PSEUDO:function(e){var t,n;if(J.CHILD.test(e[0]))return null;if(e[3])e[2]=e[3];else if(t=e[4])q.test(t)&&(n=ut(t,!0))&&(n=t.indexOf(")",t.length-n)-t.length)&&(t=t.slice(0,n),e[0]=e[0].slice(0,n)),e[2]=t;return e.slice(0,3)}},filter:{ID:r?function(e){return e=e.replace($,""),function(t){return t.getAttribute("id")===e}}:function(e){return e=e.replace($,""),function(t){var n=typeof t.getAttributeNode!==p&&t.getAttributeNode("id");return n&&n.value===e}},TAG:function(e){return e==="*"?function(){return!0}:(e=e.replace($,"").toLowerCase(),function(t){return t.nodeName&&t.nodeName.toLowerCase()===e})},CLASS:function(e){var t=k[d][e+" "];return t||(t=new RegExp("(^|"+O+")"+e+"("+O+"|$)"))&&k(e,function(e){return t.test(e.className||typeof e.getAttribute!==p&&e.getAttribute("class")||"")})},ATTR:function(e,t,n){return function(r,i){var s=nt.attr(r,e);return s==null?t==="!=":t?(s+="",t==="="?s===n:t==="!="?s!==n:t==="^="?n&&s.indexOf(n)===0:t==="*="?n&&s.indexOf(n)>-1:t==="$="?n&&s.substr(s.length-n.length)===n:t==="~="?(" "+s+" ").indexOf(n)>-1:t==="|="?s===n||s.substr(0,n.length+1)===n+"-":!1):!0}},CHILD:function(e,t,n,r){return e==="nth"?function(e){var t,i,s=e.parentNode;if(n===1&&r===0)return!0;if(s){i=0;for(t=s.firstChild;t;t=t.nextSibling)if(t.nodeType===1){i++;if(e===t)break}}return i-=r,i===n||i%n===0&&i/n>=0}:function(t){var n=t;switch(e){case"only":case"first":while(n=n.previousSibling)if(n.nodeType===1)return!1;if(e==="first")return!0;n=t;case"last":while(n=n.nextSibling)if(n.nodeType===1)return!1;return!0}}},PSEUDO:function(e,t){var n,r=i.pseudos[e]||i.setFilters[e.toLowerCase()]||nt.error("unsupported pseudo: "+e);return r[d]?r(t):r.length>1?(n=[e,e,"",t],i.setFilters.hasOwnProperty(e.toLowerCase())?N(function(e,n){var i,s=r(e,t),o=s.length;while(o--)i=T.call(e,s[o]),e[i]=!(n[i]=s[o])}):function(e){return r(e,0,n)}):r}},pseudos:{not:N(function(e){var t=[],n=[],r=a(e.replace(j,"$1"));return r[d]?N(function(e,t,n,i){var s,o=r(e,null,i,[]),u=e.length;while(u--)if(s=o[u])e[u]=!(t[u]=s)}):function(e,i,s){return t[0]=e,r(t,null,s,n),!n.pop()}}),has:N(function(e){return function(t){return nt(e,t).length>0}}),contains:N(function(e){return function(t){return(t.textContent||t.innerText||s(t)).indexOf(e)>-1}}),enabled:function(e){return e.disabled===!1},disabled:function(e){return e.disabled===!0},checked:function(e){var t=e.nodeName.toLowerCase();return t==="input"&&!!e.checked||t==="option"&&!!e.selected},selected:function(e){return e.parentNode&&e.parentNode.selectedIndex,e.selected===!0},parent:function(e){return!i.pseudos.empty(e)},empty:function(e){var t;e=e.firstChild;while(e){if(e.nodeName>"@"||(t=e.nodeType)===3||t===4)return!1;e=e.nextSibling}return!0},header:function(e){return X.test(e.nodeName)},text:function(e){var t,n;return e.nodeName.toLowerCase()==="input"&&(t=e.type)==="text"&&((n=e.getAttribute("type"))==null||n.toLowerCase()===t)},radio:rt("radio"),checkbox:rt("checkbox"),file:rt("file"),password:rt("password"),image:rt("image"),submit:it("submit"),reset:it("reset"),button:function(e){var t=e.nodeName.toLowerCase();return t==="input"&&e.type==="button"||t==="button"},input:function(e){return V.test(e.nodeName)},focus:function(e){var t=e.ownerDocument;return e===t.activeElement&&(!t.hasFocus||t.hasFocus())&&!!(e.type||e.href||~e.tabIndex)},active:function(e){return e===e.ownerDocument.activeElement},first:st(function(){return[0]}),last:st(function(e,t){return[t-1]}),eq:st(function(e,t,n){return[n<0?n+t:n]}),even:st(function(e,t){for(var n=0;n<t;n+=2)e.push(n);return e}),odd:st(function(e,t){for(var n=1;n<t;n+=2)e.push(n);return e}),lt:st(function(e,t,n){for(var r=n<0?n+t:n;--r>=0;)e.push(r);return e}),gt:st(function(e,t,n){for(var r=n<0?n+t:n;++r<t;)e.push(r);return e})}},f=y.compareDocumentPosition?function(e,t){return e===t?(l=!0,0):(!e.compareDocumentPosition||!t.compareDocumentPosition?e.compareDocumentPosition:e.compareDocumentPosition(t)&4)?-1:1}:function(e,t){if(e===t)return l=!0,0;if(e.sourceIndex&&t.sourceIndex)return e.sourceIndex-t.sourceIndex;var n,r,i=[],s=[],o=e.parentNode,u=t.parentNode,a=o;if(o===u)return ot(e,t);if(!o)return-1;if(!u)return 1;while(a)i.unshift(a),a=a.parentNode;a=u;while(a)s.unshift(a),a=a.parentNode;n=i.length,r=s.length;for(var f=0;f<n&&f<r;f++)if(i[f]!==s[f])return ot(i[f],s[f]);return f===n?ot(e,s[f],-1):ot(i[f],t,1)},[0,0].sort(f),h=!l,nt.uniqueSort=function(e){var t,n=[],r=1,i=0;l=h,e.sort(f);if(l){for(;t=e[r];r++)t===e[r-1]&&(i=n.push(r));while(i--)e.splice(n[i],1)}return e},nt.error=function(e){throw new Error("Syntax error, unrecognized expression: "+e)},a=nt.compile=function(e,t){var n,r=[],i=[],s=A[d][e+" "];if(!s){t||(t=ut(e)),n=t.length;while(n--)s=ht(t[n]),s[d]?r.push(s):i.push(s);s=A(e,pt(i,r))}return s},g.querySelectorAll&&function(){var e,t=vt,n=/'|\\/g,r=/\=[\x20\t\r\n\f]*([^'"\]]*)[\x20\t\r\n\f]*\]/g,i=[":focus"],s=[":active"],u=y.matchesSelector||y.mozMatchesSelector||y.webkitMatchesSelector||y.oMatchesSelector||y.msMatchesSelector;K(function(e){e.innerHTML="<select><option selected=''></option></select>",e.querySelectorAll("[selected]").length||i.push("\\["+O+"*(?:checked|disabled|ismap|multiple|readonly|selected|value)"),e.querySelectorAll(":checked").length||i.push(":checked")}),K(function(e){e.innerHTML="<p test=''></p>",e.querySelectorAll("[test^='']").length&&i.push("[*^$]="+O+"*(?:\"\"|'')"),e.innerHTML="<input type='hidden'/>",e.querySelectorAll(":enabled").length||i.push(":enabled",":disabled")}),i=new RegExp(i.join("|")),vt=function(e,r,s,o,u){if(!o&&!u&&!i.test(e)){var a,f,l=!0,c=d,h=r,p=r.nodeType===9&&e;if(r.nodeType===1&&r.nodeName.toLowerCase()!=="object"){a=ut(e),(l=r.getAttribute("id"))?c=l.replace(n,"\\$&"):r.setAttribute("id",c),c="[id='"+c+"'] ",f=a.length;while(f--)a[f]=c+a[f].join("");h=z.test(e)&&r.parentNode||r,p=a.join(",")}if(p)try{return S.apply(s,x.call(h.querySelectorAll(p),0)),s}catch(v){}finally{l||r.removeAttribute("id")}}return t(e,r,s,o,u)},u&&(K(function(t){e=u.call(t,"div");try{u.call(t,"[test!='']:sizzle"),s.push("!=",H)}catch(n){}}),s=new RegExp(s.join("|")),nt.matchesSelector=function(t,n){n=n.replace(r,"='$1']");if(!o(t)&&!s.test(n)&&!i.test(n))try{var a=u.call(t,n);if(a||e||t.document&&t.document.nodeType!==11)return a}catch(f){}return nt(n,null,null,[t]).length>0})}(),i.pseudos.nth=i.pseudos.eq,i.filters=mt.prototype=i.pseudos,i.setFilters=new mt,nt.attr=v.attr,v.find=nt,v.expr=nt.selectors,v.expr[":"]=v.expr.pseudos,v.unique=nt.uniqueSort,v.text=nt.getText,v.isXMLDoc=nt.isXML,v.contains=nt.contains}(e);var nt=/Until$/,rt=/^(?:parents|prev(?:Until|All))/,it=/^.[^:#\[\.,]*$/,st=v.expr.match.needsContext,ot={children:!0,contents:!0,next:!0,prev:!0};v.fn.extend({find:function(e){var t,n,r,i,s,o,u=this;if(typeof e!="string")return v(e).filter(function(){for(t=0,n=u.length;t<n;t++)if(v.contains(u[t],this))return!0});o=this.pushStack("","find",e);for(t=0,n=this.length;t<n;t++){r=o.length,v.find(e,this[t],o);if(t>0)for(i=r;i<o.length;i++)for(s=0;s<r;s++)if(o[s]===o[i]){o.splice(i--,1);break}}return o},has:function(e){var t,n=v(e,this),r=n.length;return this.filter(function(){for(t=0;t<r;t++)if(v.contains(this,n[t]))return!0})},not:function(e){return this.pushStack(ft(this,e,!1),"not",e)},filter:function(e){return this.pushStack(ft(this,e,!0),"filter",e)},is:function(e){return!!e&&(typeof e=="string"?st.test(e)?v(e,this.context).index(this[0])>=0:v.filter(e,this).length>0:this.filter(e).length>0)},closest:function(e,t){var n,r=0,i=this.length,s=[],o=st.test(e)||typeof e!="string"?v(e,t||this.context):0;for(;r<i;r++){n=this[r];while(n&&n.ownerDocument&&n!==t&&n.nodeType!==11){if(o?o.index(n)>-1:v.find.matchesSelector(n,e)){s.push(n);break}n=n.parentNode}}return s=s.length>1?v.unique(s):s,this.pushStack(s,"closest",e)},index:function(e){return e?typeof e=="string"?v.inArray(this[0],v(e)):v.inArray(e.jquery?e[0]:e,this):this[0]&&this[0].parentNode?this.prevAll().length:-1},add:function(e,t){var n=typeof e=="string"?v(e,t):v.makeArray(e&&e.nodeType?[e]:e),r=v.merge(this.get(),n);return this.pushStack(ut(n[0])||ut(r[0])?r:v.unique(r))},addBack:function(e){return this.add(e==null?this.prevObject:this.prevObject.filter(e))}}),v.fn.andSelf=v.fn.addBack,v.each({parent:function(e){var t=e.parentNode;return t&&t.nodeType!==11?t:null},parents:function(e){return v.dir(e,"parentNode")},parentsUntil:function(e,t,n){return v.dir(e,"parentNode",n)},next:function(e){return at(e,"nextSibling")},prev:function(e){return at(e,"previousSibling")},nextAll:function(e){return v.dir(e,"nextSibling")},prevAll:function(e){return v.dir(e,"previousSibling")},nextUntil:function(e,t,n){return v.dir(e,"nextSibling",n)},prevUntil:function(e,t,n){return v.dir(e,"previousSibling",n)},siblings:function(e){return v.sibling((e.parentNode||{}).firstChild,e)},children:function(e){return v.sibling(e.firstChild)},contents:function(e){return v.nodeName(e,"iframe")?e.contentDocument||e.contentWindow.document:v.merge([],e.childNodes)}},function(e,t){v.fn[e]=function(n,r){var i=v.map(this,t,n);return nt.test(e)||(r=n),r&&typeof r=="string"&&(i=v.filter(r,i)),i=this.length>1&&!ot[e]?v.unique(i):i,this.length>1&&rt.test(e)&&(i=i.reverse()),this.pushStack(i,e,l.call(arguments).join(","))}}),v.extend({filter:function(e,t,n){return n&&(e=":not("+e+")"),t.length===1?v.find.matchesSelector(t[0],e)?[t[0]]:[]:v.find.matches(e,t)},dir:function(e,n,r){var i=[],s=e[n];while(s&&s.nodeType!==9&&(r===t||s.nodeType!==1||!v(s).is(r)))s.nodeType===1&&i.push(s),s=s[n];return i},sibling:function(e,t){var n=[];for(;e;e=e.nextSibling)e.nodeType===1&&e!==t&&n.push(e);return n}});var ct="abbr|article|aside|audio|bdi|canvas|data|datalist|details|figcaption|figure|footer|header|hgroup|mark|meter|nav|output|progress|section|summary|time|video",ht=/ jQuery\d+="(?:null|\d+)"/g,pt=/^\s+/,dt=/<(?!area|br|col|embed|hr|img|input|link|meta|param)(([\w:]+)[^>]*)\/>/gi,vt=/<([\w:]+)/,mt=/<tbody/i,gt=/<|&#?\w+;/,yt=/<(?:script|style|link)/i,bt=/<(?:script|object|embed|option|style)/i,wt=new RegExp("<(?:"+ct+")[\\s/>]","i"),Et=/^(?:checkbox|radio)$/,St=/checked\s*(?:[^=]|=\s*.checked.)/i,xt=/\/(java|ecma)script/i,Tt=/^\s*<!(?:\[CDATA\[|\-\-)|[\]\-]{2}>\s*$/g,Nt={option:[1,"<select multiple='multiple'>","</select>"],legend:[1,"<fieldset>","</fieldset>"],thead:[1,"<table>","</table>"],tr:[2,"<table><tbody>","</tbody></table>"],td:[3,"<table><tbody><tr>","</tr></tbody></table>"],col:[2,"<table><tbody></tbody><colgroup>","</colgroup></table>"],area:[1,"<map>","</map>"],_default:[0,"",""]},Ct=lt(i),kt=Ct.appendChild(i.createElement("div"));Nt.optgroup=Nt.option,Nt.tbody=Nt.tfoot=Nt.colgroup=Nt.caption=Nt.thead,Nt.th=Nt.td,v.support.htmlSerialize||(Nt._default=[1,"X<div>","</div>"]),v.fn.extend({text:function(e){return v.access(this,function(e){return e===t?v.text(this):this.empty().append((this[0]&&this[0].ownerDocument||i).createTextNode(e))},null,e,arguments.length)},wrapAll:function(e){if(v.isFunction(e))return this.each(function(t){v(this).wrapAll(e.call(this,t))});if(this[0]){var t=v(e,this[0].ownerDocument).eq(0).clone(!0);this[0].parentNode&&t.insertBefore(this[0]),t.map(function(){var e=this;while(e.firstChild&&e.firstChild.nodeType===1)e=e.firstChild;return e}).append(this)}return this},wrapInner:function(e){return v.isFunction(e)?this.each(function(t){v(this).wrapInner(e.call(this,t))}):this.each(function(){var t=v(this),n=t.contents();n.length?n.wrapAll(e):t.append(e)})},wrap:function(e){var t=v.isFunction(e);return this.each(function(n){v(this).wrapAll(t?e.call(this,n):e)})},unwrap:function(){return this.parent().each(function(){v.nodeName(this,"body")||v(this).replaceWith(this.childNodes)}).end()},append:function(){return this.domManip(arguments,!0,function(e){(this.nodeType===1||this.nodeType===11)&&this.appendChild(e)})},prepend:function(){return this.domManip(arguments,!0,function(e){(this.nodeType===1||this.nodeType===11)&&this.insertBefore(e,this.firstChild)})},before:function(){if(!ut(this[0]))return this.domManip(arguments,!1,function(e){this.parentNode.insertBefore(e,this)});if(arguments.length){var e=v.clean(arguments);return this.pushStack(v.merge(e,this),"before",this.selector)}},after:function(){if(!ut(this[0]))return this.domManip(arguments,!1,function(e){this.parentNode.insertBefore(e,this.nextSibling)});if(arguments.length){var e=v.clean(arguments);return this.pushStack(v.merge(this,e),"after",this.selector)}},remove:function(e,t){var n,r=0;for(;(n=this[r])!=null;r++)if(!e||v.filter(e,[n]).length)!t&&n.nodeType===1&&(v.cleanData(n.getElementsByTagName("*")),v.cleanData([n])),n.parentNode&&n.parentNode.removeChild(n);return this},empty:function(){var e,t=0;for(;(e=this[t])!=null;t++){e.nodeType===1&&v.cleanData(e.getElementsByTagName("*"));while(e.firstChild)e.removeChild(e.firstChild)}return this},clone:function(e,t){return e=e==null?!1:e,t=t==null?e:t,this.map(function(){return v.clone(this,e,t)})},html:function(e){return v.access(this,function(e){var n=this[0]||{},r=0,i=this.length;if(e===t)return n.nodeType===1?n.innerHTML.replace(ht,""):t;if(typeof e=="string"&&!yt.test(e)&&(v.support.htmlSerialize||!wt.test(e))&&(v.support.leadingWhitespace||!pt.test(e))&&!Nt[(vt.exec(e)||["",""])[1].toLowerCase()]){e=e.replace(dt,"<$1></$2>");try{for(;r<i;r++)n=this[r]||{},n.nodeType===1&&(v.cleanData(n.getElementsByTagName("*")),n.innerHTML=e);n=0}catch(s){}}n&&this.empty().append(e)},null,e,arguments.length)},replaceWith:function(e){return ut(this[0])?this.length?this.pushStack(v(v.isFunction(e)?e():e),"replaceWith",e):this:v.isFunction(e)?this.each(function(t){var n=v(this),r=n.html();n.replaceWith(e.call(this,t,r))}):(typeof e!="string"&&(e=v(e).detach()),this.each(function(){var t=this.nextSibling,n=this.parentNode;v(this).remove(),t?v(t).before(e):v(n).append(e)}))},detach:function(e){return this.remove(e,!0)},domManip:function(e,n,r){e=[].concat.apply([],e);var i,s,o,u,a=0,f=e[0],l=[],c=this.length;if(!v.support.checkClone&&c>1&&typeof f=="string"&&St.test(f))return this.each(function(){v(this).domManip(e,n,r)});if(v.isFunction(f))return this.each(function(i){var s=v(this);e[0]=f.call(this,i,n?s.html():t),s.domManip(e,n,r)});if(this[0]){i=v.buildFragment(e,this,l),o=i.fragment,s=o.firstChild,o.childNodes.length===1&&(o=s);if(s){n=n&&v.nodeName(s,"tr");for(u=i.cacheable||c-1;a<c;a++)r.call(n&&v.nodeName(this[a],"table")?Lt(this[a],"tbody"):this[a],a===u?o:v.clone(o,!0,!0))}o=s=null,l.length&&v.each(l,function(e,t){t.src?v.ajax?v.ajax({url:t.src,type:"GET",dataType:"script",async:!1,global:!1,"throws":!0}):v.error("no ajax"):v.globalEval((t.text||t.textContent||t.innerHTML||"").replace(Tt,"")),t.parentNode&&t.parentNode.removeChild(t)})}return this}}),v.buildFragment=function(e,n,r){var s,o,u,a=e[0];return n=n||i,n=!n.nodeType&&n[0]||n,n=n.ownerDocument||n,e.length===1&&typeof a=="string"&&a.length<512&&n===i&&a.charAt(0)==="<"&&!bt.test(a)&&(v.support.checkClone||!St.test(a))&&(v.support.html5Clone||!wt.test(a))&&(o=!0,s=v.fragments[a],u=s!==t),s||(s=n.createDocumentFragment(),v.clean(e,n,s,r),o&&(v.fragments[a]=u&&s)),{fragment:s,cacheable:o}},v.fragments={},v.each({appendTo:"append",prependTo:"prepend",insertBefore:"before",insertAfter:"after",replaceAll:"replaceWith"},function(e,t){v.fn[e]=function(n){var r,i=0,s=[],o=v(n),u=o.length,a=this.length===1&&this[0].parentNode;if((a==null||a&&a.nodeType===11&&a.childNodes.length===1)&&u===1)return o[t](this[0]),this;for(;i<u;i++)r=(i>0?this.clone(!0):this).get(),v(o[i])[t](r),s=s.concat(r);return this.pushStack(s,e,o.selector)}}),v.extend({clone:function(e,t,n){var r,i,s,o;v.support.html5Clone||v.isXMLDoc(e)||!wt.test("<"+e.nodeName+">")?o=e.cloneNode(!0):(kt.innerHTML=e.outerHTML,kt.removeChild(o=kt.firstChild));if((!v.support.noCloneEvent||!v.support.noCloneChecked)&&(e.nodeType===1||e.nodeType===11)&&!v.isXMLDoc(e)){Ot(e,o),r=Mt(e),i=Mt(o);for(s=0;r[s];++s)i[s]&&Ot(r[s],i[s])}if(t){At(e,o);if(n){r=Mt(e),i=Mt(o);for(s=0;r[s];++s)At(r[s],i[s])}}return r=i=null,o},clean:function(e,t,n,r){var s,o,u,a,f,l,c,h,p,d,m,g,y=t===i&&Ct,b=[];if(!t||typeof t.createDocumentFragment=="undefined")t=i;for(s=0;(u=e[s])!=null;s++){typeof u=="number"&&(u+="");if(!u)continue;if(typeof u=="string")if(!gt.test(u))u=t.createTextNode(u);else{y=y||lt(t),c=t.createElement("div"),y.appendChild(c),u=u.replace(dt,"<$1></$2>"),a=(vt.exec(u)||["",""])[1].toLowerCase(),f=Nt[a]||Nt._default,l=f[0],c.innerHTML=f[1]+u+f[2];while(l--)c=c.lastChild;if(!v.support.tbody){h=mt.test(u),p=a==="table"&&!h?c.firstChild&&c.firstChild.childNodes:f[1]==="<table>"&&!h?c.childNodes:[];for(o=p.length-1;o>=0;--o)v.nodeName(p[o],"tbody")&&!p[o].childNodes.length&&p[o].parentNode.removeChild(p[o])}!v.support.leadingWhitespace&&pt.test(u)&&c.insertBefore(t.createTextNode(pt.exec(u)[0]),c.firstChild),u=c.childNodes,c.parentNode.removeChild(c)}u.nodeType?b.push(u):v.merge(b,u)}c&&(u=c=y=null);if(!v.support.appendChecked)for(s=0;(u=b[s])!=null;s++)v.nodeName(u,"input")?_t(u):typeof u.getElementsByTagName!="undefined"&&v.grep(u.getElementsByTagName("input"),_t);if(n){m=function(e){if(!e.type||xt.test(e.type))return r?r.push(e.parentNode?e.parentNode.removeChild(e):e):n.appendChild(e)};for(s=0;(u=b[s])!=null;s++)if(!v.nodeName(u,"script")||!m(u))n.appendChild(u),typeof u.getElementsByTagName!="undefined"&&(g=v.grep(v.merge([],u.getElementsByTagName("script")),m),b.splice.apply(b,[s+1,0].concat(g)),s+=g.length)}return b},cleanData:function(e,t){var n,r,i,s,o=0,u=v.expando,a=v.cache,f=v.support.deleteExpando,l=v.event.special;for(;(i=e[o])!=null;o++)if(t||v.acceptData(i)){r=i[u],n=r&&a[r];if(n){if(n.events)for(s in n.events)l[s]?v.event.remove(i,s):v.removeEvent(i,s,n.handle);a[r]&&(delete a[r],f?delete i[u]:i.removeAttribute?i.removeAttribute(u):i[u]=null,v.deletedIds.push(r))}}}}),function(){var e,t;v.uaMatch=function(e){e=e.toLowerCase();var t=/(chrome)[ \/]([\w.]+)/.exec(e)||/(webkit)[ \/]([\w.]+)/.exec(e)||/(opera)(?:.*version|)[ \/]([\w.]+)/.exec(e)||/(msie) ([\w.]+)/.exec(e)||e.indexOf("compatible")<0&&/(mozilla)(?:.*? rv:([\w.]+)|)/.exec(e)||[];return{browser:t[1]||"",version:t[2]||"0"}},e=v.uaMatch(o.userAgent),t={},e.browser&&(t[e.browser]=!0,t.version=e.version),t.chrome?t.webkit=!0:t.webkit&&(t.safari=!0),v.browser=t,v.sub=function(){function e(t,n){return new e.fn.init(t,n)}v.extend(!0,e,this),e.superclass=this,e.fn=e.prototype=this(),e.fn.constructor=e,e.sub=this.sub,e.fn.init=function(r,i){return i&&i instanceof v&&!(i instanceof e)&&(i=e(i)),v.fn.init.call(this,r,i,t)},e.fn.init.prototype=e.fn;var t=e(i);return e}}();var Dt,Pt,Ht,Bt=/alpha\([^)]*\)/i,jt=/opacity=([^)]*)/,Ft=/^(top|right|bottom|left)$/,It=/^(none|table(?!-c[ea]).+)/,qt=/^margin/,Rt=new RegExp("^("+m+")(.*)$","i"),Ut=new RegExp("^("+m+")(?!px)[a-z%]+$","i"),zt=new RegExp("^([-+])=("+m+")","i"),Wt={BODY:"block"},Xt={position:"absolute",visibility:"hidden",display:"block"},Vt={letterSpacing:0,fontWeight:400},$t=["Top","Right","Bottom","Left"],Jt=["Webkit","O","Moz","ms"],Kt=v.fn.toggle;v.fn.extend({css:function(e,n){return v.access(this,function(e,n,r){return r!==t?v.style(e,n,r):v.css(e,n)},e,n,arguments.length>1)},show:function(){return Yt(this,!0)},hide:function(){return Yt(this)},toggle:function(e,t){var n=typeof e=="boolean";return v.isFunction(e)&&v.isFunction(t)?Kt.apply(this,arguments):this.each(function(){(n?e:Gt(this))?v(this).show():v(this).hide()})}}),v.extend({cssHooks:{opacity:{get:function(e,t){if(t){var n=Dt(e,"opacity");return n===""?"1":n}}}},cssNumber:{fillOpacity:!0,fontWeight:!0,lineHeight:!0,opacity:!0,orphans:!0,widows:!0,zIndex:!0,zoom:!0},cssProps:{"float":v.support.cssFloat?"cssFloat":"styleFloat"},style:function(e,n,r,i){if(!e||e.nodeType===3||e.nodeType===8||!e.style)return;var s,o,u,a=v.camelCase(n),f=e.style;n=v.cssProps[a]||(v.cssProps[a]=Qt(f,a)),u=v.cssHooks[n]||v.cssHooks[a];if(r===t)return u&&"get"in u&&(s=u.get(e,!1,i))!==t?s:f[n];o=typeof r,o==="string"&&(s=zt.exec(r))&&(r=(s[1]+1)*s[2]+parseFloat(v.css(e,n)),o="number");if(r==null||o==="number"&&isNaN(r))return;o==="number"&&!v.cssNumber[a]&&(r+="px");if(!u||!("set"in u)||(r=u.set(e,r,i))!==t)try{f[n]=r}catch(l){}},css:function(e,n,r,i){var s,o,u,a=v.camelCase(n);return n=v.cssProps[a]||(v.cssProps[a]=Qt(e.style,a)),u=v.cssHooks[n]||v.cssHooks[a],u&&"get"in u&&(s=u.get(e,!0,i)),s===t&&(s=Dt(e,n)),s==="normal"&&n in Vt&&(s=Vt[n]),r||i!==t?(o=parseFloat(s),r||v.isNumeric(o)?o||0:s):s},swap:function(e,t,n){var r,i,s={};for(i in t)s[i]=e.style[i],e.style[i]=t[i];r=n.call(e);for(i in t)e.style[i]=s[i];return r}}),e.getComputedStyle?Dt=function(t,n){var r,i,s,o,u=e.getComputedStyle(t,null),a=t.style;return u&&(r=u.getPropertyValue(n)||u[n],r===""&&!v.contains(t.ownerDocument,t)&&(r=v.style(t,n)),Ut.test(r)&&qt.test(n)&&(i=a.width,s=a.minWidth,o=a.maxWidth,a.minWidth=a.maxWidth=a.width=r,r=u.width,a.width=i,a.minWidth=s,a.maxWidth=o)),r}:i.documentElement.currentStyle&&(Dt=function(e,t){var n,r,i=e.currentStyle&&e.currentStyle[t],s=e.style;return i==null&&s&&s[t]&&(i=s[t]),Ut.test(i)&&!Ft.test(t)&&(n=s.left,r=e.runtimeStyle&&e.runtimeStyle.left,r&&(e.runtimeStyle.left=e.currentStyle.left),s.left=t==="fontSize"?"1em":i,i=s.pixelLeft+"px",s.left=n,r&&(e.runtimeStyle.left=r)),i===""?"auto":i}),v.each(["height","width"],function(e,t){v.cssHooks[t]={get:function(e,n,r){if(n)return e.offsetWidth===0&&It.test(Dt(e,"display"))?v.swap(e,Xt,function(){return tn(e,t,r)}):tn(e,t,r)},set:function(e,n,r){return Zt(e,n,r?en(e,t,r,v.support.boxSizing&&v.css(e,"boxSizing")==="border-box"):0)}}}),v.support.opacity||(v.cssHooks.opacity={get:function(e,t){return jt.test((t&&e.currentStyle?e.currentStyle.filter:e.style.filter)||"")?.01*parseFloat(RegExp.$1)+"":t?"1":""},set:function(e,t){var n=e.style,r=e.currentStyle,i=v.isNumeric(t)?"alpha(opacity="+t*100+")":"",s=r&&r.filter||n.filter||"";n.zoom=1;if(t>=1&&v.trim(s.replace(Bt,""))===""&&n.removeAttribute){n.removeAttribute("filter");if(r&&!r.filter)return}n.filter=Bt.test(s)?s.replace(Bt,i):s+" "+i}}),v(function(){v.support.reliableMarginRight||(v.cssHooks.marginRight={get:function(e,t){return v.swap(e,{display:"inline-block"},function(){if(t)return Dt(e,"marginRight")})}}),!v.support.pixelPosition&&v.fn.position&&v.each(["top","left"],function(e,t){v.cssHooks[t]={get:function(e,n){if(n){var r=Dt(e,t);return Ut.test(r)?v(e).position()[t]+"px":r}}}})}),v.expr&&v.expr.filters&&(v.expr.filters.hidden=function(e){return e.offsetWidth===0&&e.offsetHeight===0||!v.support.reliableHiddenOffsets&&(e.style&&e.style.display||Dt(e,"display"))==="none"},v.expr.filters.visible=function(e){return!v.expr.filters.hidden(e)}),v.each({margin:"",padding:"",border:"Width"},function(e,t){v.cssHooks[e+t]={expand:function(n){var r,i=typeof n=="string"?n.split(" "):[n],s={};for(r=0;r<4;r++)s[e+$t[r]+t]=i[r]||i[r-2]||i[0];return s}},qt.test(e)||(v.cssHooks[e+t].set=Zt)});var rn=/%20/g,sn=/\[\]$/,on=/\r?\n/g,un=/^(?:color|date|datetime|datetime-local|email|hidden|month|number|password|range|search|tel|text|time|url|week)$/i,an=/^(?:select|textarea)/i;v.fn.extend({serialize:function(){return v.param(this.serializeArray())},serializeArray:function(){return this.map(function(){return this.elements?v.makeArray(this.elements):this}).filter(function(){return this.name&&!this.disabled&&(this.checked||an.test(this.nodeName)||un.test(this.type))}).map(function(e,t){var n=v(this).val();return n==null?null:v.isArray(n)?v.map(n,function(e,n){return{name:t.name,value:e.replace(on,"\r\n")}}):{name:t.name,value:n.replace(on,"\r\n")}}).get()}}),v.param=function(e,n){var r,i=[],s=function(e,t){t=v.isFunction(t)?t():t==null?"":t,i[i.length]=encodeURIComponent(e)+"="+encodeURIComponent(t)};n===t&&(n=v.ajaxSettings&&v.ajaxSettings.traditional);if(v.isArray(e)||e.jquery&&!v.isPlainObject(e))v.each(e,function(){s(this.name,this.value)});else for(r in e)fn(r,e[r],n,s);return i.join("&").replace(rn,"+")};var ln,cn,hn=/#.*$/,pn=/^(.*?):[ \t]*([^\r\n]*)\r?$/mg,dn=/^(?:about|app|app\-storage|.+\-extension|file|res|widget):$/,vn=/^(?:GET|HEAD)$/,mn=/^\/\//,gn=/\?/,yn=/<script\b[^<]*(?:(?!<\/script>)<[^<]*)*<\/script>/gi,bn=/([?&])_=[^&]*/,wn=/^([\w\+\.\-]+:)(?:\/\/([^\/?#:]*)(?::(\d+)|)|)/,En=v.fn.load,Sn={},xn={},Tn=["*/"]+["*"];try{cn=s.href}catch(Nn){cn=i.createElement("a"),cn.href="",cn=cn.href}ln=wn.exec(cn.toLowerCase())||[],v.fn.load=function(e,n,r){if(typeof e!="string"&&En)return En.apply(this,arguments);if(!this.length)return this;var i,s,o,u=this,a=e.indexOf(" ");return a>=0&&(i=e.slice(a,e.length),e=e.slice(0,a)),v.isFunction(n)?(r=n,n=t):n&&typeof n=="object"&&(s="POST"),v.ajax({url:e,type:s,dataType:"html",data:n,complete:function(e,t){r&&u.each(r,o||[e.responseText,t,e])}}).done(function(e){o=arguments,u.html(i?v("<div>").append(e.replace(yn,"")).find(i):e)}),this},v.each("ajaxStart ajaxStop ajaxComplete ajaxError ajaxSuccess ajaxSend".split(" "),function(e,t){v.fn[t]=function(e){return this.on(t,e)}}),v.each(["get","post"],function(e,n){v[n]=function(e,r,i,s){return v.isFunction(r)&&(s=s||i,i=r,r=t),v.ajax({type:n,url:e,data:r,success:i,dataType:s})}}),v.extend({getScript:function(e,n){return v.get(e,t,n,"script")},getJSON:function(e,t,n){return v.get(e,t,n,"json")},ajaxSetup:function(e,t){return t?Ln(e,v.ajaxSettings):(t=e,e=v.ajaxSettings),Ln(e,t),e},ajaxSettings:{url:cn,isLocal:dn.test(ln[1]),global:!0,type:"GET",contentType:"application/x-www-form-urlencoded; charset=UTF-8",processData:!0,async:!0,accepts:{xml:"application/xml, text/xml",html:"text/html",text:"text/plain",json:"application/json, text/javascript","*":Tn},contents:{xml:/xml/,html:/html/,json:/json/},responseFields:{xml:"responseXML",text:"responseText"},converters:{"* text":e.String,"text html":!0,"text json":v.parseJSON,"text xml":v.parseXML},flatOptions:{context:!0,url:!0}},ajaxPrefilter:Cn(Sn),ajaxTransport:Cn(xn),ajax:function(e,n){function T(e,n,s,a){var l,y,b,w,S,T=n;if(E===2)return;E=2,u&&clearTimeout(u),o=t,i=a||"",x.readyState=e>0?4:0,s&&(w=An(c,x,s));if(e>=200&&e<300||e===304)c.ifModified&&(S=x.getResponseHeader("Last-Modified"),S&&(v.lastModified[r]=S),S=x.getResponseHeader("Etag"),S&&(v.etag[r]=S)),e===304?(T="notmodified",l=!0):(l=On(c,w),T=l.state,y=l.data,b=l.error,l=!b);else{b=T;if(!T||e)T="error",e<0&&(e=0)}x.status=e,x.statusText=(n||T)+"",l?d.resolveWith(h,[y,T,x]):d.rejectWith(h,[x,T,b]),x.statusCode(g),g=t,f&&p.trigger("ajax"+(l?"Success":"Error"),[x,c,l?y:b]),m.fireWith(h,[x,T]),f&&(p.trigger("ajaxComplete",[x,c]),--v.active||v.event.trigger("ajaxStop"))}typeof e=="object"&&(n=e,e=t),n=n||{};var r,i,s,o,u,a,f,l,c=v.ajaxSetup({},n),h=c.context||c,p=h!==c&&(h.nodeType||h instanceof v)?v(h):v.event,d=v.Deferred(),m=v.Callbacks("once memory"),g=c.statusCode||{},b={},w={},E=0,S="canceled",x={readyState:0,setRequestHeader:function(e,t){if(!E){var n=e.toLowerCase();e=w[n]=w[n]||e,b[e]=t}return this},getAllResponseHeaders:function(){return E===2?i:null},getResponseHeader:function(e){var n;if(E===2){if(!s){s={};while(n=pn.exec(i))s[n[1].toLowerCase()]=n[2]}n=s[e.toLowerCase()]}return n===t?null:n},overrideMimeType:function(e){return E||(c.mimeType=e),this},abort:function(e){return e=e||S,o&&o.abort(e),T(0,e),this}};d.promise(x),x.success=x.done,x.error=x.fail,x.complete=m.add,x.statusCode=function(e){if(e){var t;if(E<2)for(t in e)g[t]=[g[t],e[t]];else t=e[x.status],x.always(t)}return this},c.url=((e||c.url)+"").replace(hn,"").replace(mn,ln[1]+"//"),c.dataTypes=v.trim(c.dataType||"*").toLowerCase().split(y),c.crossDomain==null&&(a=wn.exec(c.url.toLowerCase()),c.crossDomain=!(!a||a[1]===ln[1]&&a[2]===ln[2]&&(a[3]||(a[1]==="http:"?80:443))==(ln[3]||(ln[1]==="http:"?80:443)))),c.data&&c.processData&&typeof c.data!="string"&&(c.data=v.param(c.data,c.traditional)),kn(Sn,c,n,x);if(E===2)return x;f=c.global,c.type=c.type.toUpperCase(),c.hasContent=!vn.test(c.type),f&&v.active++===0&&v.event.trigger("ajaxStart");if(!c.hasContent){c.data&&(c.url+=(gn.test(c.url)?"&":"?")+c.data,delete c.data),r=c.url;if(c.cache===!1){var N=v.now(),C=c.url.replace(bn,"$1_="+N);c.url=C+(C===c.url?(gn.test(c.url)?"&":"?")+"_="+N:"")}}(c.data&&c.hasContent&&c.contentType!==!1||n.contentType)&&x.setRequestHeader("Content-Type",c.contentType),c.ifModified&&(r=r||c.url,v.lastModified[r]&&x.setRequestHeader("If-Modified-Since",v.lastModified[r]),v.etag[r]&&x.setRequestHeader("If-None-Match",v.etag[r])),x.setRequestHeader("Accept",c.dataTypes[0]&&c.accepts[c.dataTypes[0]]?c.accepts[c.dataTypes[0]]+(c.dataTypes[0]!=="*"?", "+Tn+"; q=0.01":""):c.accepts["*"]);for(l in c.headers)x.setRequestHeader(l,c.headers[l]);if(!c.beforeSend||c.beforeSend.call(h,x,c)!==!1&&E!==2){S="abort";for(l in{success:1,error:1,complete:1})x[l](c[l]);o=kn(xn,c,n,x);if(!o)T(-1,"No Transport");else{x.readyState=1,f&&p.trigger("ajaxSend",[x,c]),c.async&&c.timeout>0&&(u=setTimeout(function(){x.abort("timeout")},c.timeout));try{E=1,o.send(b,T)}catch(k){if(!(E<2))throw k;T(-1,k)}}return x}return x.abort()},active:0,lastModified:{},etag:{}});var Mn=[],_n=/\?/,Dn=/(=)\?(?=&|$)|\?\?/,Pn=v.now();v.ajaxSetup({jsonp:"callback",jsonpCallback:function(){var e=Mn.pop()||v.expando+"_"+Pn++;return this[e]=!0,e}}),v.ajaxPrefilter("json jsonp",function(n,r,i){var s,o,u,a=n.data,f=n.url,l=n.jsonp!==!1,c=l&&Dn.test(f),h=l&&!c&&typeof a=="string"&&!(n.contentType||"").indexOf("application/x-www-form-urlencoded")&&Dn.test(a);if(n.dataTypes[0]==="jsonp"||c||h)return s=n.jsonpCallback=v.isFunction(n.jsonpCallback)?n.jsonpCallback():n.jsonpCallback,o=e[s],c?n.url=f.replace(Dn,"$1"+s):h?n.data=a.replace(Dn,"$1"+s):l&&(n.url+=(_n.test(f)?"&":"?")+n.jsonp+"="+s),n.converters["script json"]=function(){return u||v.error(s+" was not called"),u[0]},n.dataTypes[0]="json",e[s]=function(){u=arguments},i.always(function(){e[s]=o,n[s]&&(n.jsonpCallback=r.jsonpCallback,Mn.push(s)),u&&v.isFunction(o)&&o(u[0]),u=o=t}),"script"}),v.ajaxSetup({accepts:{script:"text/javascript, application/javascript, application/ecmascript, application/x-ecmascript"},contents:{script:/javascript|ecmascript/},converters:{"text script":function(e){return v.globalEval(e),e}}}),v.ajaxPrefilter("script",function(e){e.cache===t&&(e.cache=!1),e.crossDomain&&(e.type="GET",e.global=!1)}),v.ajaxTransport("script",function(e){if(e.crossDomain){var n,r=i.head||i.getElementsByTagName("head")[0]||i.documentElement;return{send:function(s,o){n=i.createElement("script"),n.async="async",e.scriptCharset&&(n.charset=e.scriptCharset),n.src=e.url,n.onload=n.onreadystatechange=function(e,i){if(i||!n.readyState||/loaded|complete/.test(n.readyState))n.onload=n.onreadystatechange=null,r&&n.parentNode&&r.removeChild(n),n=t,i||o(200,"success")},r.insertBefore(n,r.firstChild)},abort:function(){n&&n.onload(0,1)}}}});var Hn,Bn=e.ActiveXObject?function(){for(var e in Hn)Hn[e](0,1)}:!1,jn=0;v.ajaxSettings.xhr=e.ActiveXObject?function(){return!this.isLocal&&Fn()||In()}:Fn,function(e){v.extend(v.support,{ajax:!!e,cors:!!e&&"withCredentials"in e})}(v.ajaxSettings.xhr()),v.support.ajax&&v.ajaxTransport(function(n){if(!n.crossDomain||v.support.cors){var r;return{send:function(i,s){var o,u,a=n.xhr();n.username?a.open(n.type,n.url,n.async,n.username,n.password):a.open(n.type,n.url,n.async);if(n.xhrFields)for(u in n.xhrFields)a[u]=n.xhrFields[u];n.mimeType&&a.overrideMimeType&&a.overrideMimeType(n.mimeType),!n.crossDomain&&!i["X-Requested-With"]&&(i["X-Requested-With"]="XMLHttpRequest");try{for(u in i)a.setRequestHeader(u,i[u])}catch(f){}a.send(n.hasContent&&n.data||null),r=function(e,i){var u,f,l,c,h;try{if(r&&(i||a.readyState===4)){r=t,o&&(a.onreadystatechange=v.noop,Bn&&delete Hn[o]);if(i)a.readyState!==4&&a.abort();else{u=a.status,l=a.getAllResponseHeaders(),c={},h=a.responseXML,h&&h.documentElement&&(c.xml=h);try{c.text=a.responseText}catch(p){}try{f=a.statusText}catch(p){f=""}!u&&n.isLocal&&!n.crossDomain?u=c.text?200:404:u===1223&&(u=204)}}}catch(d){i||s(-1,d)}c&&s(u,f,c,l)},n.async?a.readyState===4?setTimeout(r,0):(o=++jn,Bn&&(Hn||(Hn={},v(e).unload(Bn)),Hn[o]=r),a.onreadystatechange=r):r()},abort:function(){r&&r(0,1)}}}});var qn,Rn,Un=/^(?:toggle|show|hide)$/,zn=new RegExp("^(?:([-+])=|)("+m+")([a-z%]*)$","i"),Wn=/queueHooks$/,Xn=[Gn],Vn={"*":[function(e,t){var n,r,i=this.createTween(e,t),s=zn.exec(t),o=i.cur(),u=+o||0,a=1,f=20;if(s){n=+s[2],r=s[3]||(v.cssNumber[e]?"":"px");if(r!=="px"&&u){u=v.css(i.elem,e,!0)||n||1;do a=a||".5",u/=a,v.style(i.elem,e,u+r);while(a!==(a=i.cur()/o)&&a!==1&&--f)}i.unit=r,i.start=u,i.end=s[1]?u+(s[1]+1)*n:n}return i}]};v.Animation=v.extend(Kn,{tweener:function(e,t){v.isFunction(e)?(t=e,e=["*"]):e=e.split(" ");var n,r=0,i=e.length;for(;r<i;r++)n=e[r],Vn[n]=Vn[n]||[],Vn[n].unshift(t)},prefilter:function(e,t){t?Xn.unshift(e):Xn.push(e)}}),v.Tween=Yn,Yn.prototype={constructor:Yn,init:function(e,t,n,r,i,s){this.elem=e,this.prop=n,this.easing=i||"swing",this.options=t,this.start=this.now=this.cur(),this.end=r,this.unit=s||(v.cssNumber[n]?"":"px")},cur:function(){var e=Yn.propHooks[this.prop];return e&&e.get?e.get(this):Yn.propHooks._default.get(this)},run:function(e){var t,n=Yn.propHooks[this.prop];return this.options.duration?this.pos=t=v.easing[this.easing](e,this.options.duration*e,0,1,this.options.duration):this.pos=t=e,this.now=(this.end-this.start)*t+this.start,this.options.step&&this.options.step.call(this.elem,this.now,this),n&&n.set?n.set(this):Yn.propHooks._default.set(this),this}},Yn.prototype.init.prototype=Yn.prototype,Yn.propHooks={_default:{get:function(e){var t;return e.elem[e.prop]==null||!!e.elem.style&&e.elem.style[e.prop]!=null?(t=v.css(e.elem,e.prop,!1,""),!t||t==="auto"?0:t):e.elem[e.prop]},set:function(e){v.fx.step[e.prop]?v.fx.step[e.prop](e):e.elem.style&&(e.elem.style[v.cssProps[e.prop]]!=null||v.cssHooks[e.prop])?v.style(e.elem,e.prop,e.now+e.unit):e.elem[e.prop]=e.now}}},Yn.propHooks.scrollTop=Yn.propHooks.scrollLeft={set:function(e){e.elem.nodeType&&e.elem.parentNode&&(e.elem[e.prop]=e.now)}},v.each(["toggle","show","hide"],function(e,t){var n=v.fn[t];v.fn[t]=function(r,i,s){return r==null||typeof r=="boolean"||!e&&v.isFunction(r)&&v.isFunction(i)?n.apply(this,arguments):this.animate(Zn(t,!0),r,i,s)}}),v.fn.extend({fadeTo:function(e,t,n,r){return this.filter(Gt).css("opacity",0).show().end().animate({opacity:t},e,n,r)},animate:function(e,t,n,r){var i=v.isEmptyObject(e),s=v.speed(t,n,r),o=function(){var t=Kn(this,v.extend({},e),s);i&&t.stop(!0)};return i||s.queue===!1?this.each(o):this.queue(s.queue,o)},stop:function(e,n,r){var i=function(e){var t=e.stop;delete e.stop,t(r)};return typeof e!="string"&&(r=n,n=e,e=t),n&&e!==!1&&this.queue(e||"fx",[]),this.each(function(){var t=!0,n=e!=null&&e+"queueHooks",s=v.timers,o=v._data(this);if(n)o[n]&&o[n].stop&&i(o[n]);else for(n in o)o[n]&&o[n].stop&&Wn.test(n)&&i(o[n]);for(n=s.length;n--;)s[n].elem===this&&(e==null||s[n].queue===e)&&(s[n].anim.stop(r),t=!1,s.splice(n,1));(t||!r)&&v.dequeue(this,e)})}}),v.each({slideDown:Zn("show"),slideUp:Zn("hide"),slideToggle:Zn("toggle"),fadeIn:{opacity:"show"},fadeOut:{opacity:"hide"},fadeToggle:{opacity:"toggle"}},function(e,t){v.fn[e]=function(e,n,r){return this.animate(t,e,n,r)}}),v.speed=function(e,t,n){var r=e&&typeof e=="object"?v.extend({},e):{complete:n||!n&&t||v.isFunction(e)&&e,duration:e,easing:n&&t||t&&!v.isFunction(t)&&t};r.duration=v.fx.off?0:typeof r.duration=="number"?r.duration:r.duration in v.fx.speeds?v.fx.speeds[r.duration]:v.fx.speeds._default;if(r.queue==null||r.queue===!0)r.queue="fx";return r.old=r.complete,r.complete=function(){v.isFunction(r.old)&&r.old.call(this),r.queue&&v.dequeue(this,r.queue)},r},v.easing={linear:function(e){return e},swing:function(e){return.5-Math.cos(e*Math.PI)/2}},v.timers=[],v.fx=Yn.prototype.init,v.fx.tick=function(){var e,n=v.timers,r=0;qn=v.now();for(;r<n.length;r++)e=n[r],!e()&&n[r]===e&&n.splice(r--,1);n.length||v.fx.stop(),qn=t},v.fx.timer=function(e){e()&&v.timers.push(e)&&!Rn&&(Rn=setInterval(v.fx.tick,v.fx.interval))},v.fx.interval=13,v.fx.stop=function(){clearInterval(Rn),Rn=null},v.fx.speeds={slow:600,fast:200,_default:400},v.fx.step={},v.expr&&v.expr.filters&&(v.expr.filters.animated=function(e){return v.grep(v.timers,function(t){return e===t.elem}).length});var er=/^(?:body|html)$/i;v.fn.offset=function(e){if(arguments.length)return e===t?this:this.each(function(t){v.offset.setOffset(this,e,t)});var n,r,i,s,o,u,a,f={top:0,left:0},l=this[0],c=l&&l.ownerDocument;if(!c)return;return(r=c.body)===l?v.offset.bodyOffset(l):(n=c.documentElement,v.contains(n,l)?(typeof l.getBoundingClientRect!="undefined"&&(f=l.getBoundingClientRect()),i=tr(c),s=n.clientTop||r.clientTop||0,o=n.clientLeft||r.clientLeft||0,u=i.pageYOffset||n.scrollTop,a=i.pageXOffset||n.scrollLeft,{top:f.top+u-s,left:f.left+a-o}):f)},v.offset={bodyOffset:function(e){var t=e.offsetTop,n=e.offsetLeft;return v.support.doesNotIncludeMarginInBodyOffset&&(t+=parseFloat(v.css(e,"marginTop"))||0,n+=parseFloat(v.css(e,"marginLeft"))||0),{top:t,left:n}},setOffset:function(e,t,n){var r=v.css(e,"position");r==="static"&&(e.style.position="relative");var i=v(e),s=i.offset(),o=v.css(e,"top"),u=v.css(e,"left"),a=(r==="absolute"||r==="fixed")&&v.inArray("auto",[o,u])>-1,f={},l={},c,h;a?(l=i.position(),c=l.top,h=l.left):(c=parseFloat(o)||0,h=parseFloat(u)||0),v.isFunction(t)&&(t=t.call(e,n,s)),t.top!=null&&(f.top=t.top-s.top+c),t.left!=null&&(f.left=t.left-s.left+h),"using"in t?t.using.call(e,f):i.css(f)}},v.fn.extend({position:function(){if(!this[0])return;var e=this[0],t=this.offsetParent(),n=this.offset(),r=er.test(t[0].nodeName)?{top:0,left:0}:t.offset();return n.top-=parseFloat(v.css(e,"marginTop"))||0,n.left-=parseFloat(v.css(e,"marginLeft"))||0,r.top+=parseFloat(v.css(t[0],"borderTopWidth"))||0,r.left+=parseFloat(v.css(t[0],"borderLeftWidth"))||0,{top:n.top-r.top,left:n.left-r.left}},offsetParent:function(){return this.map(function(){var e=this.offsetParent||i.body;while(e&&!er.test(e.nodeName)&&v.css(e,"position")==="static")e=e.offsetParent;return e||i.body})}}),v.each({scrollLeft:"pageXOffset",scrollTop:"pageYOffset"},function(e,n){var r=/Y/.test(n);v.fn[e]=function(i){return v.access(this,function(e,i,s){var o=tr(e);if(s===t)return o?n in o?o[n]:o.document.documentElement[i]:e[i];o?o.scrollTo(r?v(o).scrollLeft():s,r?s:v(o).scrollTop()):e[i]=s},e,i,arguments.length,null)}}),v.each({Height:"height",Width:"width"},function(e,n){v.each({padding:"inner"+e,content:n,"":"outer"+e},function(r,i){v.fn[i]=function(i,s){var o=arguments.length&&(r||typeof i!="boolean"),u=r||(i===!0||s===!0?"margin":"border");return v.access(this,function(n,r,i){var s;return v.isWindow(n)?n.document.documentElement["client"+e]:n.nodeType===9?(s=n.documentElement,Math.max(n.body["scroll"+e],s["scroll"+e],n.body["offset"+e],s["offset"+e],s["client"+e])):i===t?v.css(n,r,i,u):v.style(n,r,i,u)},n,o?i:t,o,null)}})}),e.jQuery=e.$=v,typeof define=="function"&&define.amd&&define.amd.jQuery&&define("jquery",[],function(){return v})})(window); \ No newline at end of file diff --git a/lsp/md5.js b/lsp/plugins/md5.js similarity index 100% rename from lsp/md5.js rename to lsp/plugins/md5.js diff --git a/lsp/placeholder.js b/lsp/plugins/placeholder.js similarity index 100% rename from lsp/placeholder.js rename to lsp/plugins/placeholder.js diff --git a/lsp/tablesort.js b/lsp/plugins/tablesort.js similarity index 100% rename from lsp/tablesort.js rename to lsp/plugins/tablesort.js diff --git a/lsp/server.html b/lsp/server.html index ea792182..e9780973 100755 --- a/lsp/server.html +++ b/lsp/server.html @@ -2,63 +2,67 @@ <html> <head> <meta http-equiv='content-type' content='text/html;charset=utf-8' /> - <title>MistServer Manager</title> - <script src='jquery.js'></script> - <script src='placeholder.js'></script> - <script src='md5.js'></script> + <title>MistServer MI</title> + <script src='plugins/jquery.js'></script> + <script src='plugins/placeholder.js'></script> + <script src='plugins/md5.js'></script> + <script src='plugins/tablesort.js'></script> + <script src='plugins/jquery.flot.min.js'></script> + <script src='plugins/jquery.flot.time.min.js'></script> + <script src='plugins/jquery.flot.crosshair.min.js'></script> <script src='main.js'></script> <script src='pages.js'></script> - <script src='tablesort.js'></script> <link rel='stylesheet' href='main.css' /> - <script> - //these are placed here because the compression compiler does not deal with the eval function properly. - //enter the values of the settings object into their input fields - function enterSettings(){ - $('.isSetting').each(function(){ - var objpath = findObjPath($(this)); - var val = ''; - try { - eval('val = '+objpath+';'); - } - catch(e) { } - if ($(this).is('input,select')) { - $(this).val(val); - } - else { - $(this).text(val); - } - }); - } - //this function is for moving limits (LTS only) - function moveLimit(destination,streamName,objpath) { - if ((streamName == '_new_') || (destination != streamName[0])) { - var target; - if (destination == 'server') { - if (!settings.settings.config.limits) { - settings.settings.config.limits = []; - } - target = settings.settings.config.limits; - } - else { - destination = destination.replace('stream-',''); - if (!settings.settings.streams[destination].limits) { - settings.settings.streams[destination].limits = []; - } - target = settings.settings.streams[destination].limits; - } - - eval('target.push(settings.'+objpath+');'); - eval('delete settings.'+objpath+';'); + <link rel='icon' href=''> + <script> + //these are placed here because the compression compiler does not deal with the eval function properly. + //enter the values of the settings object into their input fields + function enterSettings(){ + $('.isSetting').each(function(){ + var objpath = findObjPath($(this)); + var val = ''; + try { + eval('val = '+objpath+';'); } + catch(e) { } + if ($(this).is('input,select')) { + $(this).val(val); + } + else { + $(this).text(val); + } + }); + } + //this function is for moving limits (LTS only) + function moveLimit(destination,streamName,objpath) { + if ((streamName == '_new_') || (destination != streamName[0])) { + var target; + if (destination == 'server') { + if (!settings.settings.config.limits) { + settings.settings.config.limits = []; + } + target = settings.settings.config.limits; + } + else { + destination = destination.replace('stream-',''); + if (!settings.settings.streams[destination].limits) { + settings.settings.streams[destination].limits = []; + } + target = settings.settings.streams[destination].limits; + } + + eval('target.push(settings.'+objpath+');'); + eval('delete settings.'+objpath+';'); } - </script> + } + </script> </head> <body> <div id='header'> <div id='logo'> - <a href='http://mistserver.org' target='_blank'> - <span>Mist/</span> Server + <a> + <span>Mist</span>Server Management Interface </a> </div> <div id='status'> @@ -68,20 +72,33 @@ </div> </div> - <div id='menu'> + <div id='menu' class='menu'> <div class='button'>Overview</div> <div class='button'>Protocols</div> <div class='button'>Streams</div> + <div class='button'>Preview</div> <div class='button LTS-only'>Limits</div> <div class='button'>Conversion</div> <div class='button'>Logs</div> + <div class='button'>Statistics</div> <div class='button'>Server Stats</div> <br> <div class='button red'>Disconnect</div> - <a class='button' href='http://shop.mistserver.org' target='_blank'>Mist Shop</a> + <br> + <div class='expandbutton'> + Tools <span class=arrowdown></span> + <div class='expandcontainer'> + <a class='button linktoReleaseNotes notedited' href='http://mistserver.org/wiki/Release_notes/MistServer' target='_blank'>Release notes</a> + <a class='button' href='http://mistserver.org/products' target='_blank'>Mist Shop</a> + <a class='button' href='http://mistserver.org/streamtester' target='_blank'>Stream Tester</a> + <a class='button' href='http://mistserver.org/wiki/Category:Guides' target='_blank'>Guides</a> + <div class='button'>Email for Help</div> + <a class='button linktoTnC notLTSlink' href='http://mistserver.org/wiki/Mistserver_license' target='_blank'>Terms & Conditions</a> + </div> + </div> </div> <div id='page'></div> - + <div id='ih-button' title='Activate integrated help'>?</div> </body> </html> \ No newline at end of file diff --git a/src/analysers/dtsc_analyser.cpp b/src/analysers/dtsc_analyser.cpp index 9e6e70cd..c3c46667 100644 --- a/src/analysers/dtsc_analyser.cpp +++ b/src/analysers/dtsc_analyser.cpp @@ -3,10 +3,12 @@ #include <string> #include <iostream> +#include <sstream> #include <mist/dtsc.h> #include <mist/json.h> #include <mist/config.h> +#include <mist/defines.h> ///\brief Holds everything unique to the analysers. namespace Analysers { @@ -17,11 +19,34 @@ namespace Analysers { ///\return The return code of the analyser. int analyseDTSC(Util::Config conf){ DTSC::File F(conf.getString("filename")); - std::cout << F.getMeta().toJSON().toPrettyString() << std::endl; + std::stringstream meta; + F.getMeta().toPrettyString(meta,0, 0x03); + std::cout << meta.str() << std::endl; + int bPos = 0; + F.seek_bpos(0); F.parseNext(); - while (F.getJSON()){ - std::cout << F.getJSON().toPrettyString() << std::endl; + JSON::Value tmp; + std::string tmpStr; + while (F.getPacket()){ + tmpStr = std::string(F.getPacket().getData(), F.getPacket().getDataLen()); + switch (F.getPacket().getVersion()){ + case DTSC::DTSC_V1: { + unsigned int i = 8; + JSON::fromDTMI((const unsigned char*)tmpStr.data(), tmpStr.size(), i, tmp); + break; + } + case DTSC::DTSC_V2: { + unsigned int i = 8; + JSON::fromDTMI2((const unsigned char*)tmpStr.data(), tmpStr.size(), i, tmp); + break; + } + default: + DEBUG_MSG(DLVL_WARN,"Invalid dtsc packet @ bpos %d", bPos); + break; + } + std::cout << tmp.toPrettyString() << std::endl; + bPos = F.getBytePos(); F.parseNext(); } return 0; diff --git a/src/analysers/ogg_analyser.cpp b/src/analysers/ogg_analyser.cpp index ac519232..8fd881a2 100644 --- a/src/analysers/ogg_analyser.cpp +++ b/src/analysers/ogg_analyser.cpp @@ -10,7 +10,73 @@ #include <mist/theora.h> namespace Analysers{ - int analyseOGG(){ + std::string Opus_prettyPacket(char * part,int len){ + if (len < 1){ + return "Invalid packet (0 byte length)"; + } + std::stringstream r; + char config = part[0] >> 3; + char code = part[0] & 3; + if ((part[0] & 4) == 4){r << "Stereo, ";}else{r << "Mono, ";} + if (config < 14){ + r << "SILK, "; + if (config < 4){r << "NB, ";} + if (config < 8 && config > 3){r << "MB, ";} + if (config < 14 && config > 7){r << "WB, ";} + if (config % 4 == 0){r << "10ms";} + if (config % 4 == 1){r << "20ms";} + if (config % 4 == 2){r << "40ms";} + if (config % 4 == 3){r << "60ms";} + } + if (config < 16 && config > 13){ + r << "Hybrid, "; + if (config < 14){r << "SWB, ";}else{r << "FB, ";} + if (config % 2 == 0){r << "10ms";}else{r << "20ms";} + } + if (config > 15){ + r << "CELT, "; + if (config < 20){r << "NB, ";} + if (config < 24 && config > 19){r << "WB, ";} + if (config < 28 && config > 23){r << "SWB, ";} + if (config > 27){r << "FB, ";} + if (config % 4 == 0){r << "2.5ms";} + if (config % 4 == 1){r << "5ms";} + if (config % 4 == 2){r << "10ms";} + if (config % 4 == 3){r << "20ms";} + } + if (code == 0){ + r << ": 1 packet (" << (len-1) << "b)"; + return r.str(); + } + if (code == 1){ + r << ": 2 packets (" << ((len-1)/2) << "b / " << ((len-1)/2) << "b)"; + return r.str(); + } + if (code == 2){ + if (len < 2){ + return "Invalid packet (code 2 must be > 1 byte long)"; + } + if (part[1] < 252){ + r << ": 2 packets (" << (int)part[1] << "b / " << (int)(len-2-part[1]) << "b)"; + }else{ + int ilen = part[1] + part[2]*4; + r << ": 2 packets (" << ilen << "b / " << (int)(len-3-ilen) << "b)"; + } + return r.str(); + } + //code 3 + bool VBR = (part[1] & 128) == 128; + bool pad = (part[1] & 64) == 64; + bool packets = (part[1] & 63); + r << ": " << packets << " packets (VBR = " << VBR << ", padding = " << pad << ")"; + return r.str(); + } + + int analyseOGG(int argc, char ** argv){ + Util::Config conf = Util::Config(argv[0], PACKAGE_VERSION); + conf.addOption("pages", JSON::fromString("{\"long\":\"pages\", \"short\":\"p\", \"long_off\":\"nopages\", \"short_off\":\"P\", \"default\":0, \"help\":\"Enable/disable printing of Ogg pages\"}")); + conf.parseArgs(argc, argv); + std::map<int,std::string> sn2Codec; std::string oggBuffer; OGG::Page oggPage; @@ -22,17 +88,94 @@ namespace Analysers{ } //while OGG::page check function read while (oggPage.read(oggBuffer)){//reading ogg to string - oggPage.setInternalCodec(""); - if (oggPage.getHeaderType() & 0x02){ + //print the Ogg page details, if requested + if (conf.getBool("pages")){ + std::cout << oggPage.toPrettyString() << std::endl; + } + + //attempt to detect codec if this is the first page of a stream + if (oggPage.getHeaderType() & OGG::BeginOfStream){ if (memcmp("theora",oggPage.getFullPayload() + 1,6) == 0){ - sn2Codec[oggPage.getBitstreamSerialNumber()] = "theora"; + sn2Codec[oggPage.getBitstreamSerialNumber()] = "Theora"; } if (memcmp("vorbis",oggPage.getFullPayload() + 1,6) == 0){ - sn2Codec[oggPage.getBitstreamSerialNumber()] = "vorbis"; + sn2Codec[oggPage.getBitstreamSerialNumber()] = "Vorbis"; + } + if (memcmp("OpusHead",oggPage.getFullPayload(),8) == 0){ + sn2Codec[oggPage.getBitstreamSerialNumber()] = "Opus"; + } + if (sn2Codec[oggPage.getBitstreamSerialNumber()] != ""){ + std::cout << "Bitstream " << oggPage.getBitstreamSerialNumber() << " recognized as " << sn2Codec[oggPage.getBitstreamSerialNumber()] << std::endl; + }else{ + std::cout << "Bitstream " << oggPage.getBitstreamSerialNumber() << " could not be recognized as any known codec" << std::endl; + } + + } + + if (sn2Codec[oggPage.getBitstreamSerialNumber()] == "Theora"){ + std::cout << "Theora data" << std::endl; + int offset = 0; + for (unsigned int i = 0; i < oggPage.getSegmentTableDeque().size(); i++){ + theora::header tmpHeader; + int len = oggPage.getSegmentTableDeque()[i]; + if (tmpHeader.read(oggPage.getFullPayload()+offset,len)){ + std::cout << tmpHeader.toPrettyString(2); + } + theora::frame tmpFrame; + if (tmpFrame.read(oggPage.getFullPayload()+offset,len)){ + std::cout << tmpFrame.toPrettyString(2); + } + offset += len; + } + }else if(sn2Codec[oggPage.getBitstreamSerialNumber()] == "Vorbis"){ + std::cout << "Vorbis data" << std::endl; + int offset = 0; + for (unsigned int i = 0; i < oggPage.getSegmentTableDeque().size(); i++){ + vorbis::header tmpHeader; + int len = oggPage.getSegmentTableDeque()[i]; + if (tmpHeader.read(oggPage.getFullPayload()+offset,len)){ + std::cout << tmpHeader.toPrettyString(2); + } + offset += len; + } + }else if(sn2Codec[oggPage.getBitstreamSerialNumber()] == "Opus"){ + std::cout << "Opus data" << std::endl; + int offset = 0; + for (unsigned int i = 0; i < oggPage.getSegmentTableDeque().size(); i++){ + int len = oggPage.getSegmentTableDeque()[i]; + char * part = oggPage.getFullPayload() + offset; + if (len >= 8 && memcmp(part, "Opus", 4) == 0){ + if (memcmp(part, "OpusHead", 8) == 0){ + std::cout << " Version: " << (int)(part[8]) << std::endl; + std::cout << " Channels: " << (int)(part[9]) << std::endl; + std::cout << " Pre-skip: " << (int)(part[10] + (part[11] << 8)) << std::endl; + std::cout << " Orig. sample rate: " << (int)(part[12] + (part[13] << 8) + (part[14] << 16) + (part[15] << 24)) << std::endl; + std::cout << " Gain: " << (int)(part[16] + (part[17] << 8)) << std::endl; + std::cout << " Channel map: " << (int)(part[18]) << std::endl; + if (part[18] > 0){ + std::cout << " Channel map family " << (int)(part[18]) << " not implemented - output incomplete" << std::endl; + } + } + if (memcmp(part, "OpusTags", 8) == 0){ + unsigned int vendor_len = part[8] + (part[9]<<8) + (part[10]<<16) + (part[11]<<24); + std::cout << " Vendor: " << std::string(part+12, vendor_len) << std::endl; + char * str_data = part+12+vendor_len; + unsigned int strings = str_data[0] + (str_data[1]<<8) + (str_data[2]<<16) + (str_data[3]<<24); + std::cout << " Tags: (" << strings << ")" << std::endl; + str_data += 4; + for (unsigned int j = 0; j < strings; j++){ + unsigned int strlen = str_data[0] + (str_data[1]<<8) + (str_data[2]<<16) + (str_data[3]<<24); + str_data += 4; + std::cout << " [" << j << "] " << std::string(str_data, strlen) << std::endl; + str_data += strlen; + } + } + }else{ + std::cout << " " << Opus_prettyPacket(part,len) << std::endl; + } + offset += len; } } - oggPage.setInternalCodec(sn2Codec[oggPage.getBitstreamSerialNumber()]); - std::cout << oggPage.toPrettyString() << std::endl; } } return 0; @@ -40,8 +183,6 @@ namespace Analysers{ } int main(int argc, char ** argv){ - Util::Config conf = Util::Config(argv[0], PACKAGE_VERSION); - conf.parseArgs(argc, argv); - return Analysers::analyseOGG(); + return Analysers::analyseOGG(argc, argv); } diff --git a/src/buffer/buffer.cpp b/src/buffer/buffer.cpp deleted file mode 100644 index 413f6fdc..00000000 --- a/src/buffer/buffer.cpp +++ /dev/null @@ -1,345 +0,0 @@ -/// \file buffer.cpp -/// Contains the main code for the Buffer. - -#include <fcntl.h> -#include <iostream> -#include <string> -#include <vector> -#include <cstdlib> -#include <cstdio> -#include <string.h> -#include <unistd.h> -#include <signal.h> -#include <sstream> -#include <sys/time.h> -#include <mist/config.h> -#include <mist/timing.h> -#include "buffer_stream.h" -#include <mist/stream.h> -#include <mist/defines.h> - -/// Holds all code unique to the Buffer. -namespace Buffer { - - volatile bool buffer_running = true; ///< Set to false when shutting down. - Stream * thisStream = 0; - Socket::Server SS; ///< The server socket. - - ///\brief A function running in a thread to send all statistics. - ///\param empty A null pointer. - void handleStats(void * empty){ -#if defined(_TTHREAD_POSIX_) && defined(WITH_THREADNAMES) && !(defined(__FreeBSD__) || defined(__APPLE__) || defined(__MACH__) || defined(_WIN32) || defined(__CYGWIN__)) - pthread_setname_np(pthread_self(), "StatsHandler"); -#endif - if (empty != 0){ - return; - } - std::string double_newline = "\n\n"; - Socket::Connection StatsSocket = Socket::Connection(Util::getTmpFolder() + "statistics", true); - while (buffer_running){ - Util::sleep(1000); //sleep one second - if ( !StatsSocket.connected()){ - StatsSocket = Socket::Connection(Util::getTmpFolder() + "statistics", true); - } - if (StatsSocket.connected()){ - StatsSocket.SendNow(Stream::get()->getStats()); - StatsSocket.SendNow(double_newline); - if (StatsSocket.spool()){ - //Got a response. - buffer_running = false; - } - } - } - StatsSocket.close(); - } - - ///\brief A function to handle input data. - ///\param conn A socket reference. - void handlePushIn(Socket::Connection & conn){ - #if defined(_TTHREAD_POSIX_) && defined(WITH_THREADNAMES) && !(defined(__FreeBSD__) || defined(__APPLE__) || defined(__MACH__) || defined(_WIN32) || defined(__CYGWIN__)) - pthread_setname_np(pthread_self(), "Push Input"); - #endif - conn.setBlocking(true); - int sockNo = 0; - while (buffer_running && conn.connected()){ - while (thisStream->parsePacket(conn)){ - //do nothing while parsing - } - Util::sleep(10);//sleep to prevent high CPU usage - } - conn.close(); - if (buffer_running){ - thisStream->endStream(); - } - long long int wait_time = Util::getMS(); - while (Util::getMS() - wait_time < thisStream->metadata.bufferWindow){ - Util::sleep(thisStream->metadata.bufferWindow - (Util::getMS() - wait_time)); - } - thisStream->removeSocket(sockNo); - } - - ///\brief A function running a thread to handle input data through stdin. - ///Automatically slows down to realtime playback. - ///\param empty A null pointer. - void handleStdin(void * empty){ - if (empty != 0){ - return; - } - #if defined(_TTHREAD_POSIX_) && defined(WITH_THREADNAMES) && !(defined(__FreeBSD__) || defined(__APPLE__) || defined(__MACH__) || defined(_WIN32) || defined(__CYGWIN__)) - pthread_setname_np(pthread_self(), "Standard Input"); - #endif - long long int timeDiff = 0; //difference between local time and stream time - unsigned int lastPacket = 0; //last parsed packet timestamp - std::string inBuffer; - char charBuffer[1024 * 10]; - unsigned int charCount; - long long int now; - - while (std::cin.good() && buffer_running){ - //slow down packet receiving to real-time - now = Util::getMS(); - if (((now - timeDiff) >= lastPacket) || (lastPacket - (now - timeDiff) > 15000)){ - if (thisStream->parsePacket(inBuffer)){ - lastPacket = thisStream->getTime(); - if ((now - timeDiff - lastPacket) > 15000 || (now - timeDiff - lastPacket < -15000)){ - timeDiff = now - lastPacket; - } - }else{ - std::cin.read(charBuffer, 1024 * 10); - charCount = std::cin.gcount(); - inBuffer.append(charBuffer, charCount); - } - }else{ - Util::sleep(std::min(15LL, lastPacket - (now - timeDiff))); - } - } - buffer_running = false; - } - - ///\brief A function running in a thread to handle a new user connection. - ///\param v_usr The user that is connected. - void handleUser(void * v_usr){ - std::set<int> allowedTracks; - user * usr = (user*)v_usr; - thisStream->addUser(usr); -#if DEBUG >= 5 - std::cerr << "Thread launched for user " << usr->sID << ", socket number " << usr->S.getSocket() << std::endl; -#endif -#if defined(_TTHREAD_POSIX_) && defined(WITH_THREADNAMES) && !(defined(__FreeBSD__) || defined(__APPLE__) || defined(__MACH__) || defined(_WIN32) || defined(__CYGWIN__)) - pthread_setname_np(pthread_self(), usr->sID.c_str()); -#endif - usr->myRing = thisStream->getRing(); - thisStream->sendMeta(usr->S); - - while (usr->S.connected()){ - if (usr->myRing->playCount){ - if (usr->myRing->waiting){ - Stream::get()->waitForData(); - if ( !Stream::get()->isNewest(usr->myRing->b, allowedTracks)){ - usr->myRing->waiting = false; - usr->myRing->b = Stream::get()->getNext(usr->myRing->b, allowedTracks); - if ((Stream::get()->getPacket(usr->myRing->b).isMember("keyframe") && (usr->myRing->playCount > 0)) || (usr->playUntil && usr->playUntil <= Stream::get()->getPacket(usr->myRing->b)["time"].asInt())){ - usr->myRing->playCount--; - if (usr->myRing->playCount < 1 || usr->playUntil <= Stream::get()->getPacket(usr->myRing->b)["time"].asInt()){ - usr->myRing->playCount = 0; - JSON::Value pausemark; - pausemark["trackid"] = 0ll; - pausemark["mark"] = "pause"; - pausemark["time"] = Stream::get()->getPacket(usr->myRing->b)["time"].asInt(); - pausemark.sendTo(usr->S); - } - } - } - }else{ - //complete a send - Stream::get()->sendPacket(usr->myRing->b, usr->S); - if ( !usr->S.connected()){break;} - //switch to next buffer - if (Stream::get()->isNewest(usr->myRing->b, allowedTracks)){ - //no next buffer? go in waiting mode. - usr->myRing->waiting = true; - }else{ - usr->myRing->b = Stream::get()->getNext(usr->myRing->b, allowedTracks); - if ((Stream::get()->getPacket(usr->myRing->b).isMember("keyframe") && (usr->myRing->playCount > 0)) || (usr->playUntil && usr->playUntil <= Stream::get()->getPacket(usr->myRing->b)["time"].asInt())){ - usr->myRing->playCount--; - if (usr->myRing->playCount < 1 || usr->playUntil <= Stream::get()->getPacket(usr->myRing->b)["time"].asInt()){ - usr->myRing->playCount = 0; - JSON::Value pausemark; - pausemark["trackid"] = 0ll; - pausemark["mark"] = "pause"; - pausemark["time"] = Stream::get()->getPacket(usr->myRing->b)["time"].asInt(); - pausemark.sendTo(usr->S); - } - } - } - } - } - if (usr->S.spool()){ - while (usr->S.Received().size()){ - //delete anything that doesn't end with a newline - if ( !usr->S.Received().get().empty() && *(usr->S.Received().get().rbegin()) != '\n'){ - usr->S.Received().get().clear(); - continue; - } - usr->S.Received().get().resize(usr->S.Received().get().size() - 1); - if ( !usr->S.Received().get().empty()){ - switch (usr->S.Received().get()[0]){ - case 'P': { //Push - if (thisStream->checkWaitingIP(usr->S.Received().get().substr(2))){ - usr->S.Received().get().clear(); - Socket::Connection tmp = usr->S; - usr->S = Socket::Connection( -1); - thisStream->removeUser(usr); - thisStream->dropRing(usr->myRing); - delete usr; - return handlePushIn(tmp); - }else{ - usr->Disconnect("Push denied - invalid IP address!"); - } - break; - } - case 'S': { //Stats - usr->tmpStats = Stats(usr->S.Received().get().substr(2)); - unsigned int secs = usr->tmpStats.conntime - usr->lastStats.conntime; - if (secs < 1){ - secs = 1; - } - usr->curr_up = (usr->tmpStats.up - usr->lastStats.up) / secs; - usr->curr_down = (usr->tmpStats.down - usr->lastStats.down) / secs; - usr->lastStats = usr->tmpStats; - thisStream->saveStats(usr->sID, usr->tmpStats); - thisStream->sendMeta(usr->S); - break; - } - case 't': { - if (usr->S.Received().get().size() >= 3){ - allowedTracks.clear(); - std::string tmp = usr->S.Received().get().substr(2); - while (tmp != ""){ - allowedTracks.insert(atoi(tmp.substr(0,tmp.find(' ')).c_str())); - if (tmp.find(' ') != std::string::npos){ - tmp.erase(0,tmp.find(' ')+1); - }else{ - tmp = ""; - } - } - } - break; - } - case 's': { //second-seek - unsigned int ms = JSON::Value(usr->S.Received().get().substr(2)).asInt(); - usr->myRing->waiting = false; - usr->myRing->starved = false; - usr->myRing->b = thisStream->msSeek(ms, allowedTracks); - if (usr->myRing->playCount > 0){ - usr->myRing->playCount = 0; - } - break; - } - case 'p': { //play - usr->myRing->playCount = -1; - if (usr->S.Received().get().size() >= 2){ - usr->playUntil = atoi(usr->S.Received().get().substr(2).c_str()); - }else{ - usr->playUntil = 0; - } - break; - } - case 'o': { //once-play - if (usr->myRing->playCount >= 0){ - usr->myRing->playCount++; - } - break; - } - case 'q': { //quit-playing - usr->myRing->playCount = 0; - break; - } - } - usr->S.Received().get().clear(); - } - } - } - if (usr->myRing->waiting || !usr->myRing->playCount){ - Util::sleep(300); //sleep 300ms - } - } - usr->Disconnect("Socket closed."); - thisStream->dropRing(usr->myRing); - thisStream->removeUser(usr); - delete usr; - } - - ///\brief Starts a loop, waiting for connections to send data to. - ///\param argc The number of arguments to the program. - ///\param argv The arguments to the program. - ///\return The return code of the buffer. - int Start(int argc, char ** argv){ - Util::Config conf = Util::Config(argv[0], PACKAGE_VERSION); - conf.addOption("stream_name", - JSON::fromString("{\"arg_num\":1, \"arg\":\"string\", \"help\":\"Name of the stream this buffer will be providing.\"}")); - conf.addOption("awaiting_ip", - JSON::fromString( - "{\"arg_num\":2, \"arg\":\"string\", \"default\":\"\", \"help\":\"IP address to expect incoming data from. This will completely disable reading from standard input if used.\"}")); - conf.addOption("reportstats", - JSON::fromString("{\"default\":0, \"help\":\"Report stats to a controller process.\", \"short\":\"s\", \"long\":\"reportstats\"}")); - conf.addOption("time", - JSON::fromString( - "{\"default\":20000, \"arg\": \"integer\", \"help\":\"Buffer a specied amount of time in ms.\", \"short\":\"t\", \"long\":\"time\"}")); - conf.parseArgs(argc, argv); - - std::string name = conf.getString("stream_name"); - - SS = Util::Stream::makeLive(name); - if ( !SS.connected()){ - perror("Could not create stream socket"); - return 1; - } - SS.setBlocking(false); - conf.activate(); - #if defined(_TTHREAD_POSIX_) && defined(WITH_THREADNAMES) && !(defined(__FreeBSD__) || defined(__APPLE__) || defined(__MACH__) || defined(_WIN32) || defined(__CYGWIN__)) - pthread_setname_np(pthread_self(), "Main accepter"); - #endif - thisStream = Stream::get(); - thisStream->setName(name); - thisStream->setBufferTime(conf.getInteger("time")); - Socket::Connection incoming; - Socket::Connection std_input(fileno(stdin)); - - if (conf.getBool("reportstats")){ - tthread::thread StatsThread(handleStats, 0); - StatsThread.detach(); - } - std::string await_ip = conf.getString("awaiting_ip"); - if (await_ip == ""){ - tthread::thread StdinThread(handleStdin, 0); - StdinThread.detach(); - }else{ - thisStream->setWaitingIP(await_ip); - } - - unsigned int userId = 0; - SS.setBlocking(true); - while (buffer_running && SS.connected() && conf.is_active){ - //check for new connections, accept them if there are any - //starts a thread for every accepted connection - incoming = SS.accept(true); - if (incoming.connected()){ - tthread::thread thisUser(handleUser, (void *)new user(incoming, ++userId)); - thisUser.detach(); - } - } //main loop - - // disconnect listener - buffer_running = false; - SS.close(); - delete thisStream; - return 0; - } - -} //Buffer namespace - -///\brief Entry point for Buffer, simply calls Buffer::Start(). -int main(int argc, char ** argv){ - return Buffer::Start(argc, argv); -} //main diff --git a/src/buffer/buffer_stream.cpp b/src/buffer/buffer_stream.cpp deleted file mode 100644 index 0336ea0d..00000000 --- a/src/buffer/buffer_stream.cpp +++ /dev/null @@ -1,350 +0,0 @@ -/// \file buffer_stream.cpp -/// Contains definitions for buffer streams. - -#include "buffer_stream.h" -#include <mist/timing.h> -#include <mist/defines.h> -#include <stdlib.h> - -namespace Buffer { - - static JSON::Value ctrl_log; - - void Stream::Log(std::string type, std::string message){ - JSON::Value l; - l.append(type); - l.append(message); - ctrl_log.append(l); - } - - /// Stores the singleton reference. - Stream * Stream::ref = 0; - - /// Returns a reference to the singleton instance of this class. - /// \return A reference to the class. - Stream * Stream::get(){ - static tthread::mutex creator; - if (ref == 0){ - //prevent creating two at the same time - creator.lock(); - if (ref == 0){ - ref = new Stream(); - ref->metadata.live = true; - } - creator.unlock(); - } - return ref; - } - - /// Creates a new DTSC::Stream object, private function so only one instance can exist. - Stream::Stream() : DTSC::Stream(5){} - - /// Do cleanup on delete. - Stream::~Stream(){ - tthread::lock_guard<tthread::recursive_mutex> guard(stats_mutex); - if (users.size() > 0){ - for (usersIt = users.begin(); usersIt != users.end(); usersIt++){ - if (( * *usersIt).S.connected()){ - ( * *usersIt).S.close(); - } - } - } - moreData.notify_all(); - } - - /// Calculate and return the current statistics. - /// \return The current statistics in JSON format. - std::string & Stream::getStats(){ - static std::string ret; - long long int now = Util::epoch(); - unsigned int tot_up = 0, tot_down = 0, tot_count = 0; - tthread::lock_guard<tthread::recursive_mutex> guard(stats_mutex); - if (users.size() > 0){ - for (usersIt = users.begin(); usersIt != users.end(); usersIt++){ - tot_down += ( * *usersIt).curr_down; - tot_up += ( * *usersIt).curr_up; - tot_count++; - } - } - Storage["totals"]["down"] = tot_down; - Storage["totals"]["up"] = tot_up; - Storage["totals"]["count"] = tot_count; - Storage["totals"]["now"] = now; - Storage["buffer"] = name; - - rw_mutex.lock(); - Storage["meta"] = metadata.toJSON(); - rw_mutex.unlock(); - if (Storage["meta"].isMember("tracks")){ - for (JSON::ObjIter oIt = Storage["meta"]["tracks"].ObjBegin(); oIt != Storage["meta"]["tracks"].ObjEnd(); ++oIt){ - oIt->second.removeMember("fragments"); - oIt->second.removeMember("keys"); - oIt->second.removeMember("parts"); - oIt->second.removeMember("idheader"); - oIt->second.removeMember("commentheader"); - } - } - - Storage["ctrl_log"] = ctrl_log; - ctrl_log.null(); - - ret = Storage.toString(); - Storage["log"].null(); - return ret; - } - - /// Set the IP address to accept push data from. - /// \param ip The new IP to accept push data from. - void Stream::setWaitingIP(std::string ip){ - waiting_ip = ip; - } - - ///\brief Check if this is the IP address to accept push data from. - ///\param push_request The IP address to check, followed by a space and the password to check. - ///\return True if it is the correct address or password, false otherwise. - bool Stream::checkWaitingIP(std::string push_request){ - std::string ip = push_request.substr(0, push_request.find(' ')); - std::string pass = push_request.substr(push_request.find(' ') + 1); - if (waiting_ip.length() > 0 && waiting_ip[0] == '@'){ - if (pass == waiting_ip.substr(1)){ - return true; - }else{ - Log("BUFF", "Push to stream " + name + " denied, incorrect password: "+pass); - return false; - } - }else{ - if (ip == waiting_ip || ip == "::ffff:" + waiting_ip){ - return true; - }else{ - Log("BUFF", "Push to stream " + name + " denied, wrong IP: "+ip+" != (::ffff:)"+waiting_ip); - return false; - } - } - } - - /// Stores intermediate statistics. - /// \param username The name of the user. - /// \param stats The final statistics to store. - void Stream::saveStats(std::string username, Stats & stats){ - tthread::lock_guard<tthread::recursive_mutex> guard(stats_mutex); - Storage["curr"][username]["connector"] = stats.connector; - Storage["curr"][username]["up"] = stats.up; - Storage["curr"][username]["down"] = stats.down; - Storage["curr"][username]["conntime"] = stats.conntime; - Storage["curr"][username]["host"] = stats.host; - Storage["curr"][username]["start"] = Util::epoch() - stats.conntime; - } - - /// Stores final statistics. - /// \param username The name of the user. - /// \param stats The final statistics to store. - /// \param reason The reason for disconnecting. - void Stream::clearStats(std::string username, Stats & stats, std::string reason){ - tthread::lock_guard<tthread::recursive_mutex> guard(stats_mutex); - if (Storage["curr"].isMember(username)){ - Storage["curr"].removeMember(username); - #if DEBUG >= 4 - std::cout << "Disconnected user " << username << ": " << reason << ". " << stats.connector << " transferred " << stats.up << " up and " - << stats.down << " down in " << stats.conntime << " seconds to " << stats.host << std::endl; - #endif - } - Storage["log"][username]["connector"] = stats.connector; - Storage["log"][username]["up"] = stats.up; - Storage["log"][username]["down"] = stats.down; - Storage["log"][username]["conntime"] = stats.conntime; - Storage["log"][username]["host"] = stats.host; - Storage["log"][username]["start"] = Util::epoch() - stats.conntime; - } - - /// The deletion callback override that will disconnect users - /// whom are currently receiving a tag that is being deleted. - void Stream::deletionCallback(DTSC::livePos deleting){ - tthread::lock_guard<tthread::recursive_mutex> guard(stats_mutex); - for (usersIt = users.begin(); usersIt != users.end(); usersIt++){ - if ((*usersIt)->myRing->playCount && (*usersIt)->myRing->b == deleting){ - (*usersIt)->Disconnect("Buffer underrun"); - } - } - } - - /// Sets the buffer name. - /// \param n The new name of the buffer. - void Stream::setName(std::string n){ - name = n; - } - - void Stream::sendPacket(DTSC::livePos & num, Socket::Connection & S){ - rw_mutex.lock(); - if (!getPacket(num) && buffers.size()){ - DEBUG_MSG(DLVL_DEVEL, "Oh noes, ran out of packets! Resetting to beginning..."); - num = buffers.rbegin()->first; - } - getPacket(num).sendTo(S); - rw_mutex.unlock(); - } - - /// parsePacket override that will lock the rw_mutex during parsing. - bool Stream::parsePacket(std::string & buffer){ - rw_mutex.lock(); - bool ret = DTSC::Stream::parsePacket(buffer); - rw_mutex.unlock(); - if (ret){ - rw_change.notify_all(); - moreData.notify_all(); - } - return ret; - } - - /// getNext override that will lock the rw_mutex during checking. - DTSC::livePos Stream::getNext(DTSC::livePos & pos, std::set<int> & allowedTracks){ - tthread::lock_guard<tthread::mutex> guard(rw_mutex); - return DTSC::Stream::getNext(pos, allowedTracks); - } - - /// endStream override that will lock the rw_mutex - void Stream::endStream(){ - tthread::lock_guard<tthread::mutex> guard(rw_mutex); - return DTSC::Stream::endStream(); - } - - /// Removes a track and all related buffers from the stream. - void Stream::removeTrack(int trackId){ - rw_mutex.lock(); - metadata.tracks.erase(trackId); - rw_mutex.unlock(); - std::set<DTSC::livePos> toDelete; - for (std::map<DTSC::livePos, JSON::Value >::iterator it = buffers.begin(); it != buffers.end(); it++){ - if (it->first.trackID == (unsigned long long int)trackId){ - toDelete.insert(it->first); - } - } - while (toDelete.size()){ - deletionCallback(*toDelete.begin()); - buffers.erase(*toDelete.begin()); - keyframes[trackId].erase(*toDelete.begin()); - toDelete.erase(toDelete.begin()); - } - } - - /// Calls removeTrack on all tracks that were streaming from this socket number. - void Stream::removeSocket(int sockNo){ - std::set<int> toDelete; - std::map<int,DTSC::Track>::iterator it; - rw_mutex.lock(); - for (it = metadata.tracks.begin(); it != metadata.tracks.end(); ++it){ - if ((it->first & (sockNo << 16)) == (sockNo << 16)){ - toDelete.insert(it->first); - Log("BUFF", "Stream "+name+" lost input for track: "+ it->second.getIdentifier()); - } - } - rw_mutex.unlock(); - while (toDelete.size()){ - removeTrack(*toDelete.begin()); - toDelete.erase(toDelete.begin()); - } - } - - /// parsePacket override that will lock the rw_mutex during parsing. - bool Stream::parsePacket(Socket::Connection & c){ - bool ret = false; - if (!c.spool()){ - return ret; - } - rw_mutex.lock(); - while (DTSC::Stream::parsePacket(c.Received())){ - ret = true; - } - rw_mutex.unlock(); - if (ret){ - rw_change.notify_all(); - moreData.notify_all(); - } - return ret; - } - - /// Metadata sender that locks the rw_mutex during sending. - void Stream::sendMeta(Socket::Connection & s){ - if (metadata){ - rw_mutex.lock(); - DTSC::Meta tmpMeta = metadata; - rw_mutex.unlock(); - tmpMeta.send(s); - } - } - - /// Add a user to the userlist. - /// \param newUser The user to be added. - void Stream::addUser(user * newUser){ - tthread::lock_guard<tthread::recursive_mutex> guard(stats_mutex); - users.insert(newUser); - } - - /// Removes a user from the userlist. - /// \param oldUser The user to be removed. - void Stream::removeUser(user * oldUser){ - tthread::lock_guard<tthread::recursive_mutex> guard(stats_mutex); - users.erase(oldUser); - } - - /// Blocks the thread until new data is available. - void Stream::waitForData(){ - tthread::lock_guard<tthread::recursive_mutex> guard(stats_mutex); - moreData.wait(stats_mutex); - } - - ///Creates a new user from a newly connected socket. - ///Also prints "User connected" text to stdout. - ///\param fd A connection to the user. - ///\param ID Unique ID of the user. - user::user(Socket::Connection fd, long long ID){ - sID = JSON::Value(ID).asString(); - S = fd; - curr_up = 0; - curr_down = 0; - myRing = 0; - } //constructor - - ///Disconnects the current user. Doesn't do anything if already disconnected. - ///Prints "Disconnected user" to stdout if disconnect took place. - ///\param reason The reason for disconnecting the user. - void user::Disconnect(std::string reason){ - S.close(); - Stream::get()->clearStats(sID, lastStats, reason); - } //Disconnect - - ///Default stats constructor. - ///Should not be used. - Stats::Stats(){ - up = 0; - down = 0; - conntime = 0; - } - - ///Stats constructor reading a string. - ///Reads a stats string and parses it to the internal representation. - ///\param s The string of stats. - Stats::Stats(std::string s){ - size_t f = s.find(' '); - if (f != std::string::npos){ - host = s.substr(0, f); - s.erase(0, f + 1); - } - f = s.find(' '); - if (f != std::string::npos){ - connector = s.substr(0, f); - s.erase(0, f + 1); - } - f = s.find(' '); - if (f != std::string::npos){ - conntime = atoi(s.substr(0, f).c_str()); - s.erase(0, f + 1); - } - f = s.find(' '); - if (f != std::string::npos){ - up = atoi(s.substr(0, f).c_str()); - s.erase(0, f + 1); - down = atoi(s.c_str()); - } - } - -} diff --git a/src/buffer/buffer_stream.h b/src/buffer/buffer_stream.h deleted file mode 100644 index 441ab3f1..00000000 --- a/src/buffer/buffer_stream.h +++ /dev/null @@ -1,106 +0,0 @@ -/// \file buffer_stream.h -/// Contains definitions for buffer streams. - -#pragma once -#include <string> -#include <mist/dtsc.h> -#include <mist/json.h> -#include <mist/socket.h> -#include <mist/tinythread.h> - -namespace Buffer { - - /// Converts a stats line to up, down, host, connector and conntime values. - class Stats{ - public: - unsigned int up;///<The amount of bytes sent upstream. - unsigned int down;///<The amount of bytes received downstream. - std::string host;///<The connected host. - std::string connector;///<The connector the user is connected with. - unsigned int conntime;///<The amount of time the user is connected. - Stats(std::string s); - Stats(); - }; - - ///\brief Keeps track of connected users. - /// - ///Keeps track of which buffer the user currently uses, - ///and its connection status. - class user{ - public: - DTSC::Ring * myRing; ///< Ring of the buffer for this user. - unsigned int playUntil; ///< Time until where is being played or zero if undefined. - Stats lastStats; ///< Holds last known stats for this connection. - Stats tmpStats; ///< Holds temporary stats for this connection. - std::string sID; ///< Holds the connection ID. - unsigned int curr_up; ///< Holds the current estimated transfer speed up. - unsigned int curr_down; ///< Holds the current estimated transfer speed down. - Socket::Connection S; ///< Connection to user - /// Creates a new user from a newly connected socket. - user(Socket::Connection fd, long long int ID); - /// Disconnects the current user. Doesn't do anything if already disconnected. - void Disconnect(std::string reason); - }; - - /// Keeps track of a single streams inputs and outputs, taking care of thread safety and all other related issues. - class Stream : public DTSC::Stream{ - public: - /// Get a reference to this Stream object. - static Stream * get(); - /// Get the current statistics in JSON format. - std::string & getStats(); - /// Set the IP address to accept push data from. - void setWaitingIP(std::string ip); - /// Check if this is the IP address to accept push data from. - bool checkWaitingIP(std::string ip); - /// Sets the current socket for push data. - bool setInput(Socket::Connection S); - /// Gets the current socket for push data. - Socket::Connection & getIPInput(); - /// Send a packet while locking the mutex. - void sendPacket(DTSC::livePos & num, Socket::Connection & S); - /// Stores intermediate statistics. - void saveStats(std::string username, Stats & stats); - /// Stores final statistics. - void clearStats(std::string username, Stats & stats, std::string reason); - /// Sets the buffer name. - void setName(std::string n); - /// Add a user to the userlist. - void addUser(user * newUser); - /// Delete a user from the userlist. - void removeUser(user * oldUser); - /// Blocks the thread until new data is available. - void waitForData(); - /// Sends the metadata to a specific socket - void sendMeta(Socket::Connection & s); - /// Cleanup function - ~Stream(); - /// Removes a track and all related buffers from the stream. - void removeTrack(int trackId); - /// Calls removeTrack on all tracks that were streaming from this socket number. - void removeSocket(int sockNo); - /// Thread-safe parsePacket override. - bool parsePacket(std::string & buffer); - /// Thread-safe parsePacket override. - bool parsePacket(Socket::Connection & c); - /// Logs a message to the controller. - void Log(std::string type, std::string message); - DTSC::livePos getNext(DTSC::livePos & pos, std::set<int> & allowedTracks); - void endStream(); - private: - void deletionCallback(DTSC::livePos deleting); - tthread::mutex rw_mutex; ///< Mutex for read/write locking. - tthread::condition_variable rw_change; ///< Triggered when reader/writer count changes. - static Stream * ref; - Stream(); - JSON::Value Storage; ///< Global storage of data. - std::string waiting_ip; ///< IP address for media push. - Socket::Connection ip_input; ///< Connection used for media push. - tthread::recursive_mutex stats_mutex; ///< Mutex for stats/users modifications. - std::set<user*> users; ///< All connected users. - std::set<user*>::iterator usersIt; ///< Iterator for all connected users. - std::string name; ///< Name for this buffer. - tthread::condition_variable moreData; ///< Triggered when more data becomes available. - }; -} -; diff --git a/src/buffer/player.cpp b/src/buffer/player.cpp deleted file mode 100644 index b211e8fe..00000000 --- a/src/buffer/player.cpp +++ /dev/null @@ -1,276 +0,0 @@ -/// \file player.cpp -/// Holds all code for the MistPlayer application used for VoD streams. - -#include <iostream>//for std::cerr -#include <stdio.h> //for fileno -#include <stdlib.h> //for atoi -#include <sys/time.h> -#include <mist/dtsc.h> -#include <mist/json.h> -#include <mist/config.h> -#include <mist/socket.h> -#include <mist/timing.h> -#include <mist/procs.h> -#include <mist/stream.h> -#include <mist/defines.h> - -//under cygwin, recv blocks for ~15ms if no data is available. -//This is a hack to keep performance decent with that bug present. -#ifdef __CYGWIN__ -#define CYG_DEFI int cyg_count; -#define CYG_INCR cyg_count++; -#define CYG_LOOP (cyg_count % 20 == 0) && -#else -#define CYG_DEFI -#define CYG_INCR -#define CYG_LOOP -#endif - -///Converts a stats line to up, down, host, connector and conntime values. -class Stats{ - public: - unsigned int up;///<The amount of bytes sent upstream. - unsigned int down;///<The amount of bytes received downstream. - std::string host;///<The connected host. - std::string connector;///<The connector the user is connected with. - unsigned int conntime;///<The amount of time the user is connected. - ///\brief Default stats constructor. - /// - ///Should not be used. - Stats(){ - up = 0; - down = 0; - conntime = 0; - } - ; - ///\brief Stats constructor reading a string. - /// - ///Reads a stats string and parses it to the internal representation. - ///\param s The string of stats. - Stats(std::string s){ - size_t f = s.find(' '); - if (f != std::string::npos){ - host = s.substr(0, f); - s.erase(0, f + 1); - } - f = s.find(' '); - if (f != std::string::npos){ - connector = s.substr(0, f); - s.erase(0, f + 1); - } - f = s.find(' '); - if (f != std::string::npos){ - conntime = atoi(s.substr(0, f).c_str()); - s.erase(0, f + 1); - } - f = s.find(' '); - if (f != std::string::npos){ - up = atoi(s.substr(0, f).c_str()); - s.erase(0, f + 1); - down = atoi(s.c_str()); - } - } -}; - -std::string intToBin(long long int number){ - std::string result; - result.resize(8); - for (int i = 7; i >= 0; i--){ - result[i] = number & 0xFF; - number >>= 8; - } - return result; -} - -int main(int argc, char** argv){ - Util::Config conf(argv[0], PACKAGE_VERSION); - conf.addOption("filename", - JSON::fromString("{\"arg_num\":1, \"help\":\"Name of the file to write to stdout.\"}")); - conf.addOption("streamname", - JSON::fromString("{\"arg\":\"string\",\"short\":\"s\",\"long\":\"stream\",\"help\":\"The name of the stream that this connector will transmit.\"}")); - conf.parseArgs(argc, argv); - conf.activate(); - int playing = 0; - - Socket::Connection in_out = Socket::Connection(fileno(stdout), fileno(stdin)); - - DTSC::File source = DTSC::File(conf.getString("filename")); - - if ( !source.getMeta().isFixed()){ - std::cerr << "Encountered a non-fixed file." << std::endl; - return 1; - } - - std::string streamname = conf.getString("streamname"); - source.getMeta().send(in_out); - - JSON::Value pausemark; - pausemark["trackid"] = 0ll; - pausemark["mark"] = "pause"; - pausemark["time"] = 0ll; - - Socket::Connection StatsSocket = Socket::Connection(Util::getTmpFolder() + "statistics", true); - int lastSent = Util::epoch(); //time last packet was sent - - JSON::Value last_pack; - - bool meta_sent = false; - int playUntil = -1; - long long max_lead_time = 7500;//maximum time in ms that the player can be faster than real-time - long long now, prevTimestamp = 0; //for timing of sending packets - std::set<int> newSelect; - Stats sts; - CYG_DEFI - - while (in_out.connected() && (Util::epoch() - lastSent < 60) && conf.is_active){ - CYG_INCR - if (CYG_LOOP in_out.spool()){ - while (in_out.Received().size()){ - //delete anything that doesn't end with a newline - if ( *(in_out.Received().get().rbegin()) != '\n'){ - in_out.Received().get().clear(); - continue; - } - in_out.Received().get().resize(in_out.Received().get().size() - 1); - if ( !in_out.Received().get().empty()){ - DEBUG_MSG(DLVL_HIGH, "Player received: %s", in_out.Received().get().c_str()); - switch (in_out.Received().get()[0]){ - case 'P': { //Push -#if DEBUG >= 4 - std::cerr << "Received push - ignoring (" << in_out.Received().get() << ")" << std::endl; -#endif - in_out.close(); //pushing to VoD makes no sense - break; - } - case 'S': { //Stats - if ( !StatsSocket.connected()){ - StatsSocket = Socket::Connection(Util::getTmpFolder() + "statistics", true); - } - if (StatsSocket.connected()){ - sts = Stats(in_out.Received().get().substr(2)); - JSON::Value json_sts; - json_sts["vod"]["down"] = (long long int)sts.down; - json_sts["vod"]["up"] = (long long int)sts.up; - json_sts["vod"]["time"] = (long long int)sts.conntime; - json_sts["vod"]["host"] = sts.host; - json_sts["vod"]["connector"] = sts.connector; - json_sts["vod"]["filename"] = conf.getString("filename"); - json_sts["vod"]["now"] = Util::epoch(); - json_sts["vod"]["start"] = Util::epoch() - sts.conntime; - if ( !meta_sent){ - json_sts["vod"]["meta"] = source.getMeta().toJSON(); - json_sts["vod"]["meta"]["is_fixed"] = 1; - for (JSON::ObjIter oIt = json_sts["vod"]["meta"]["tracks"].ObjBegin(); oIt != json_sts["vod"]["meta"]["tracks"].ObjEnd(); oIt++){ - oIt->second.removeMember("keys"); - oIt->second.removeMember("fragments"); - oIt->second.removeMember("parts"); - } - meta_sent = true; - } - StatsSocket.SendNow(json_sts.toString()); - StatsSocket.SendNow("\n\n", 2); - StatsSocket.flush(); - } - break; - } - case 's': { //second-seek - int ms = JSON::Value(in_out.Received().get().substr(2)).asInt(); - source.seek_time(ms); - lastSent = Util::epoch(); - prevTimestamp = 0; - playUntil = 0; - break; - } - case 'p': { //play - playing = -1; - lastSent = Util::epoch(); - in_out.setBlocking(false); - prevTimestamp = 0; - if (in_out.Received().get().size() >= 2){ - playUntil = atoi(in_out.Received().get().substr(2).c_str()); - }else{ - playUntil = 0; - } - break; - } - case 'o': { //once-play - if (playing <= 0){ - playing = 1; - } - prevTimestamp = 0; - ++playing; - in_out.setBlocking(false); - break; - } - case 'q': { //quit-playing - if (playing != 0){ - DEBUG_MSG(DLVL_HIGH, "Pausemark sent"); - pausemark["time"] = source.getJSON()["time"]; - pausemark.sendTo(in_out); - } - playing = 0; - in_out.setBlocking(true); - break; - } - case 't': { - newSelect.clear(); - std::string tmp = in_out.Received().get().substr(2); - while (tmp != ""){ - newSelect.insert(atoi(tmp.substr(0,tmp.find(' ')).c_str())); - if (tmp.find(' ') != std::string::npos){ - tmp.erase(0,tmp.find(' ')+1); - }else{ - tmp = ""; - } - } - source.selectTracks(newSelect); - break; - } -#if DEBUG >= 4 - default: { - std::cerr << "MistPlayer received an unknown command: " << in_out.Received().get() << std::endl; - break; - } -#endif - } - in_out.Received().get().clear(); - } - } - } - if (playing != 0){ - now = Util::getMS(); - source.seekNext(); - if ( !source.getJSON()){ - DEBUG_MSG(DLVL_HIGH, "Seek failed (end of file?) - stopping playback"); - playing = 0; - } - if (playing > 0 && source.atKeyframe()){ - --playing; - } - if (prevTimestamp == 0){ - prevTimestamp = now - source.getJSON()["time"].asInt(); - } - if (playing == -1 && playUntil == 0 && source.getJSON()["time"].asInt() > now - prevTimestamp + max_lead_time){ - Util::sleep(source.getJSON()["time"].asInt() - (now - prevTimestamp + max_lead_time)); - } - if ( playUntil && playUntil <= source.getJSON()["time"].asInt()){ - playing = 0; - } - if (playing == 0){ - DEBUG_MSG(DLVL_HIGH, "Pausemark sent"); - pausemark["time"] = source.getJSON()["time"]; - pausemark.sendTo(in_out); - in_out.setBlocking(true); - }else{ - lastSent = Util::epoch(); - DEBUG_MSG(DLVL_HIGH, "Playing %lliT%lli", source.getJSON()["trackid"].asInt(), source.getJSON()["time"].asInt()); - source.getJSON().sendTo(in_out); - } - }else{ - Util::sleep(10); - } - } - StatsSocket.close(); - in_out.close(); - return 0; -} diff --git a/src/connectors/conn_http.cpp b/src/connectors/conn_http.cpp index 8b01c7f6..2fdd401f 100644 --- a/src/connectors/conn_http.cpp +++ b/src/connectors/conn_http.cpp @@ -25,6 +25,7 @@ #include "embed.js.h" + /// Holds everything unique to HTTP Connectors. namespace Connector_HTTP { @@ -115,14 +116,15 @@ namespace Connector_HTTP { ///Displays a friendly error message. ///\param H The request that was being handled upon timeout. ///\param conn The connection to the client that issued the request. + ///\param msg The message to print to the client. ///\return A timestamp indicating when the request was parsed. - long long int proxyHandleTimeout(HTTP::Parser & H, Socket::Connection & conn){ + long long int proxyHandleTimeout(HTTP::Parser & H, Socket::Connection & conn, std::string msg){ H.Clean(); H.SetHeader("Server", "mistserver/" PACKAGE_VERSION "/" + Util::Config::libver); H.SetBody( - "<!DOCTYPE html><html><head><title>Gateway timeout</title></head><body><h1>Gateway timeout</h1>Though the server understood your request and attempted to handle it, somehow handling it took longer than it should. Your request has been cancelled - please try again later.</body></html>"); + "<!DOCTYPE html><html><head><title>"+msg+"</title></head><body><h1>"+msg+"</h1>Though the server understood your request and attempted to handle it, somehow handling it took longer than it should. Your request has been cancelled - please try again later.</body></html>"); long long int ret = Util::getMS(); - conn.SendNow(H.BuildResponse("504", "Gateway Timeout")); + conn.SendNow(H.BuildResponse("504", msg)); return ret; } @@ -404,6 +406,7 @@ namespace Connector_HTTP { H.Clean(); ConnConn * myCConn = 0; + unsigned int counter = 0; //loop until a connection is available/created while (!myCConn){ //lock the connection mutex before trying anything @@ -412,6 +415,12 @@ namespace Connector_HTTP { if ( !connectorConnections.count(uid)){ connectorConnections[uid] = new ConnConn(new Socket::Connection(Util::getTmpFolder() + connector)); connectorConnections[uid]->conn->setBlocking(false); //do not block on spool() with no data + if (!connectorConnections[uid]->conn->spool() && !connectorConnections[uid]->conn){ + //unlock the connection mutex before exiting + connMutex.unlock(); + DEBUG_MSG(DLVL_FAIL, "Created new connection (%s) failed - aborting request!", uid.c_str()); + return Util::getMS(); + } DEBUG_MSG(DLVL_HIGH, "Created new connection %s", uid.c_str()); } @@ -420,11 +429,17 @@ namespace Connector_HTTP { myCConn = connectorConnections[uid]; //if the connection is dead, delete it and re-loop if (!myCConn->conn->spool() && !myCConn->conn->connected()){ + counter++; DEBUG_MSG(DLVL_HIGH, "Resetting existing connection %s", uid.c_str()); connectorConnections.erase(uid); myCConn->inUse.unlock(); delete myCConn; myCConn = 0; + if (counter++ > 2){ + connMutex.unlock(); + DEBUG_MSG(DLVL_FAIL, "Created new connection (%s) failed - aborting request!", uid.c_str()); + return Util::getMS(); + } }else{ DEBUG_MSG(DLVL_HIGH, "Using active connection %s", uid.c_str()); } @@ -477,7 +492,7 @@ namespace Connector_HTTP { myCConn->inUse.unlock(); //unset to only read headers H.headerOnly = false; - return proxyHandleTimeout(H, conn); + return proxyHandleTimeout(H, conn, "Timeout: fragment too new"); } myCConn->lastUse = 0; timeout = 0; @@ -495,9 +510,9 @@ namespace Connector_HTTP { myCConn->inUse.unlock(); //unset to only read headers H.headerOnly = false; - return proxyHandleTimeout(H, conn); + return proxyHandleTimeout(H, conn, "Gateway timeout while waiting for response"); }else{ - Util::sleep(5); + Util::sleep(100); } } //unset to only read headers @@ -506,7 +521,7 @@ namespace Connector_HTTP { //failure, disconnect and sent error to user myCConn->conn->close(); myCConn->inUse.unlock(); - return proxyHandleTimeout(H, conn); + return proxyHandleTimeout(H, conn, "Gateway connection dropped"); }else{ long long int ret = Util::getMS(); //success, check type of response @@ -699,6 +714,14 @@ int main(int argc, char ** argv){ Connector_HTTP::capabilities.removeMember((*it).substr(8)); } } + if ((*it).substr(0, 7) == "MistOut"){ + arg_one = Util::getMyPath() + (*it); + conn_args[0] = arg_one.c_str(); + Connector_HTTP::capabilities[(*it).substr(7)] = JSON::fromString(Util::Procs::getOutputOf((char**)conn_args)); + if (Connector_HTTP::capabilities[(*it).substr(7)].size() < 1){ + Connector_HTTP::capabilities.removeMember((*it).substr(7)); + } + } } return conf.serveThreadedSocket(Connector_HTTP::proxyHandleHTTPConnection); diff --git a/src/connectors/conn_http_dynamic.cpp b/src/connectors/conn_http_dynamic.cpp deleted file mode 100644 index cea7caa5..00000000 --- a/src/connectors/conn_http_dynamic.cpp +++ /dev/null @@ -1,333 +0,0 @@ -/// \file conn_http_dynamic.cpp -/// Contains the main code for the HTTP Dynamic Connector - -#include <iostream> -#include <sstream> -#include <queue> -#include <cstdlib> -#include <cstdio> -#include <cmath> -#include <unistd.h> -#include <sys/types.h> -#include <sys/wait.h> -#include <getopt.h> -#include <mist/socket.h> -#include <mist/http_parser.h> -#include <mist/json.h> -#include <mist/dtsc.h> -#include <mist/flv_tag.h> -#include <mist/base64.h> -#include <mist/amf.h> -#include <mist/mp4.h> -#include <mist/mp4_adobe.h> -#include <mist/config.h> -#include <sstream> -#include <mist/stream.h> -#include <mist/timing.h> - -/// Holds everything unique to HTTP Connectors. -namespace Connector_HTTP { - - std::set<int> videoTracks;///<< Holds valid video tracks for playback - long long int audioTrack = 0;///<< Holds audio track ID for playback - void getTracks(DTSC::Meta & metadata){ - videoTracks.clear(); - for (std::map<int,DTSC::Track>::iterator it = metadata.tracks.begin(); it != metadata.tracks.end(); it++){ - if (it->second.codec == "H264" || it->second.codec == "H263" || it->second.codec == "VP6"){ - videoTracks.insert(it->first); - } - if (it->second.codec == "AAC" || it->second.codec == "MP3"){ - audioTrack = it->first; - } - } - } - - - ///\brief Builds a bootstrap for use in HTTP Dynamic streaming. - ///\param streamName The name of the stream. - ///\param trackMeta The current metadata of this track, used to generate the index. - ///\param isLive Whether or not the stream is live. - ///\param fragnum The index of the current fragment. - ///\return The generated bootstrap. - std::string dynamicBootstrap(std::string & streamName, DTSC::Track & trackMeta, bool isLive = false, int fragnum = 0){ - std::string empty; - - MP4::ASRT asrt; - asrt.setUpdate(false); - asrt.setVersion(1); - //asrt.setQualityEntry(empty, 0); - if (isLive){ - asrt.setSegmentRun(1, 4294967295ul, 0); - }else{ - asrt.setSegmentRun(1, trackMeta.keys.size(), 0); - } - - MP4::AFRT afrt; - afrt.setUpdate(false); - afrt.setVersion(1); - afrt.setTimeScale(1000); - //afrt.setQualityEntry(empty, 0); - MP4::afrt_runtable afrtrun; - int i = 0; - for (std::deque<DTSC::Key>::iterator it = trackMeta.keys.begin(); it != trackMeta.keys.end(); it++){ - if (it->getLength()){ - afrtrun.firstFragment = it->getNumber(); - afrtrun.firstTimestamp = it->getTime(); - afrtrun.duration = it->getLength(); - afrt.setFragmentRun(afrtrun, i); - i++; - } - } - - MP4::ABST abst; - abst.setVersion(1); - abst.setBootstrapinfoVersion(1); - abst.setProfile(0); - abst.setUpdate(false); - abst.setTimeScale(1000); - abst.setLive(isLive); - abst.setCurrentMediaTime(trackMeta.lastms); - abst.setSmpteTimeCodeOffset(0); - abst.setMovieIdentifier(streamName); - abst.setSegmentRunTable(asrt, 0); - abst.setFragmentRunTable(afrt, 0); - - #if DEBUG >= 8 - std::cout << "Sending bootstrap:" << std::endl << abst.toPrettyString(0) << std::endl; - #endif - return std::string((char*)abst.asBox(), (int)abst.boxedSize()); - } - - ///\brief Builds an index file for HTTP Dynamic streaming. - ///\param streamName The name of the stream. - ///\param metadata The current metadata, used to generate the index. - ///\return The index file for HTTP Dynamic Streaming. - std::string dynamicIndex(std::string & streamName, DTSC::Meta & metadata){ - if ( !audioTrack){getTracks(metadata);} - std::stringstream Result; - Result << "<?xml version=\"1.0\" encoding=\"utf-8\"?>" << std::endl; - Result << " <manifest xmlns=\"http://ns.adobe.com/f4m/1.0\">" << std::endl; - Result << " <id>" << streamName << "</id>" << std::endl; - Result << " <mimeType>video/mp4</mimeType>" << std::endl; - Result << " <deliveryType>streaming</deliveryType>" << std::endl; - if (metadata.vod){ - Result << " <duration>" << metadata.tracks[*videoTracks.begin()].lastms / 1000 << ".000</duration>" << std::endl; - Result << " <streamType>recorded</streamType>" << std::endl; - }else{ - Result << " <duration>0.00</duration>" << std::endl; - Result << " <streamType>live</streamType>" << std::endl; - } - for (std::set<int>::iterator it = videoTracks.begin(); it != videoTracks.end(); it++){ - Result << " <bootstrapInfo " - "profile=\"named\" " - "id=\"boot" << (*it) << "\" " - "url=\"" << (*it) << ".abst\">" - "</bootstrapInfo>" << std::endl; - } - for (std::set<int>::iterator it = videoTracks.begin(); it != videoTracks.end(); it++){ - Result << " <media " - "url=\"" << (*it) << "-\" " - "bitrate=\"" << metadata.tracks[(*it)].bps * 8 << "\" " - "bootstrapInfoId=\"boot" << (*it) << "\" " - "width=\"" << metadata.tracks[(*it)].width << "\" " - "height=\"" << metadata.tracks[(*it)].height << "\">" << std::endl; - Result << " <metadata>AgAKb25NZXRhRGF0YQMAAAk=</metadata>" << std::endl; - Result << " </media>" << std::endl; - } - Result << "</manifest>" << std::endl; - #if DEBUG >= 8 - std::cerr << "Sending this manifest:" << std::endl << Result.str() << std::endl; - #endif - return Result.str(); - } //BuildManifest - - ///\brief Main function for the HTTP Dynamic Connector - ///\param conn A socket describing the connection the client. - ///\return The exit code of the connector. - int dynamicConnector(Socket::Connection & conn){ - FLV::Tag tmp; //temporary tag - - DTSC::Stream Strm; //Incoming stream buffer. - HTTP::Parser HTTP_R, HTTP_S; //HTTP Receiver en HTTP Sender. - - Socket::Connection ss( -1); - std::string streamname; - bool handlingRequest = false; - - int Quality = 0; - int ReqFragment = -1; - long long mstime = 0; - long long mslen = 0; - unsigned int lastStats = 0; - conn.setBlocking(false); //do not block on conn.spool() when no data is available - - while (conn.connected()){ - if ( !handlingRequest){ - if (conn.spool() && HTTP_R.Read(conn)){ - #if DEBUG >= 5 - std::cout << "Received request: " << HTTP_R.getUrl() << std::endl; - #endif - conn.setHost(HTTP_R.GetHeader("X-Origin")); - streamname = HTTP_R.GetHeader("X-Stream"); - if ( !ss){ - ss = Util::Stream::getStream(streamname); - if ( !ss.connected()){ - HTTP_S.Clean(); - HTTP_S.SetBody("No such stream is available on the system. Please try again.\n"); - HTTP_S.SendResponse("404", "Not found", conn); - continue; - } - Strm.waitForMeta(ss); - } - if (HTTP_R.url.find(".abst") != std::string::npos){ - std::string streamID = HTTP_R.url.substr(streamname.size() + 10); - streamID = streamID.substr(0, streamID.find(".abst")); - HTTP_S.Clean(); - HTTP_S.SetBody(dynamicBootstrap(streamname, Strm.metadata.tracks[atoll(streamID.c_str())], Strm.metadata.live)); - HTTP_S.SetHeader("Content-Type", "binary/octet"); - HTTP_S.SetHeader("Cache-Control", "no-cache"); - HTTP_S.SendResponse("200", "OK", conn); - HTTP_R.Clean(); //clean for any possible next requests - continue; - } - if (HTTP_R.url.find("f4m") == std::string::npos){ - std::string tmp_qual = HTTP_R.url.substr(HTTP_R.url.find("/", 10) + 1); - Quality = atoi(tmp_qual.substr(0, tmp_qual.find("Seg") - 1).c_str()); - int temp; - temp = HTTP_R.url.find("Seg") + 3; - temp = HTTP_R.url.find("Frag") + 4; - ReqFragment = atoi(HTTP_R.url.substr(temp).c_str()); - #if DEBUG >= 5 - printf("Video track %d, fragment %d\n", Quality, ReqFragment); - #endif - if (!audioTrack){getTracks(Strm.metadata);} - DTSC::Track & vidTrack = Strm.metadata.tracks[Quality]; - mstime = 0; - mslen = 0; - for (std::deque<DTSC::Key>::iterator it = vidTrack.keys.begin(); it != vidTrack.keys.end(); it++){ - if (it->getNumber() >= ReqFragment){ - mstime = it->getTime(); - mslen = it->getLength(); - if (Strm.metadata.live){ - if (it == vidTrack.keys.end() - 2){ - HTTP_S.Clean(); - HTTP_S.SetBody("Proxy, re-request this in a second or two.\n"); - HTTP_S.SendResponse("208", "Ask again later", conn); - HTTP_R.Clean(); //clean for any possible next requests - std::cout << "Fragment after fragment " << ReqFragment << " not available yet" << std::endl; - if (ss.spool()){ - while (Strm.parsePacket(ss.Received())){} - } - } - } - break; - } - } - if (HTTP_R.url == "/"){continue;}//Don't continue, but continue instead. - if (Strm.metadata.live){ - if (mstime == 0 && ReqFragment > 1){ - HTTP_S.Clean(); - HTTP_S.SetBody("The requested fragment is no longer kept in memory on the server and cannot be served.\n"); - HTTP_S.SendResponse("412", "Fragment out of range", conn); - HTTP_R.Clean(); //clean for any possible next requests - std::cout << "Fragment " << ReqFragment << " too old" << std::endl; - continue; - } - } - std::stringstream sstream; - sstream << "t " << Quality << " " << audioTrack << "\ns " << mstime << "\np " << (mstime + mslen) << "\n"; - ss.SendNow(sstream.str().c_str()); - - HTTP_S.Clean(); - HTTP_S.SetHeader("Content-Type", "video/mp4"); - HTTP_S.StartResponse(HTTP_R, conn); - //send the bootstrap - std::string bootstrap = dynamicBootstrap(streamname, Strm.metadata.tracks[Quality], Strm.metadata.live, ReqFragment); - HTTP_S.Chunkify(bootstrap, conn); - //send a zero-size mdat, meaning it stretches until end of file. - HTTP_S.Chunkify("\000\000\000\000mdat", 8, conn); - //send init data, if needed. - if (audioTrack > 0){ - tmp.DTSCAudioInit(Strm.metadata.tracks[audioTrack]); - tmp.tagTime(mstime); - HTTP_S.Chunkify(tmp.data, tmp.len, conn); - } - if (Quality > 0){ - tmp.DTSCVideoInit(Strm.metadata.tracks[Quality]); - tmp.tagTime(mstime); - HTTP_S.Chunkify(tmp.data, tmp.len, conn); - } - handlingRequest = true; - }else{ - HTTP_S.Clean(); - HTTP_S.SetHeader("Content-Type", "text/xml"); - HTTP_S.SetHeader("Cache-Control", "no-cache"); - HTTP_S.SetBody(dynamicIndex(streamname, Strm.metadata)); - HTTP_S.SendResponse("200", "OK", conn); - } - HTTP_R.Clean(); //clean for any possible next requests - }else{ - //sleep for 250ms before next attempt - Util::sleep(250); - } - } - if (ss.connected()){ - unsigned int now = Util::epoch(); - if (now != lastStats){ - lastStats = now; - ss.SendNow(conn.getStats("HTTP_Dynamic").c_str()); - } - if (handlingRequest && ss.spool()){ - while (Strm.parsePacket(ss.Received())){ - if (Strm.lastType() == DTSC::PAUSEMARK){ - //send an empty chunk to signify request is done - HTTP_S.Chunkify("", 0, conn); - handlingRequest = false; - } - if (Strm.lastType() == DTSC::VIDEO || Strm.lastType() == DTSC::AUDIO){ - //send a chunk with the new data - tmp.DTSCLoader(Strm); - HTTP_S.Chunkify(tmp.data, tmp.len, conn); - } - } - } - if ( !ss.connected()){ - break; - } - } - } - conn.close(); - ss.SendNow(conn.getStats("HTTP_Dynamic").c_str()); - ss.close(); - return 0; - } //Connector_HTTP_Dynamic main function - -} //Connector_HTTP_Dynamic namespace - -///\brief The standard process-spawning main function. -int main(int argc, char ** argv){ - Util::Config conf(argv[0], PACKAGE_VERSION); - JSON::Value capa; - capa["desc"] = "Enables HTTP protocol Adobe-specific dynamic streaming (also known as HDS)."; - capa["deps"] = "HTTP"; - capa["url_rel"] = "/dynamic/$/manifest.f4m"; - capa["url_prefix"] = "/dynamic/$/"; - capa["socket"] = "http_dynamic"; - capa["codecs"][0u][0u].append("H264"); - capa["codecs"][0u][0u].append("H263"); - capa["codecs"][0u][0u].append("VP6"); - capa["codecs"][0u][1u].append("AAC"); - capa["codecs"][0u][1u].append("MP3"); - capa["methods"][0u]["handler"] = "http"; - capa["methods"][0u]["type"] = "flash/11"; - capa["methods"][0u]["priority"] = 7ll; - conf.addBasicConnectorOptions(capa); - conf.parseArgs(argc, argv); - - if (conf.getBool("json")){ - std::cout << capa.toString() << std::endl; - return -1; - } - - return conf.serveForkedSocket(Connector_HTTP::dynamicConnector); -} //main diff --git a/src/connectors/conn_http_json.cpp b/src/connectors/conn_http_json.cpp deleted file mode 100644 index 5c36e107..00000000 --- a/src/connectors/conn_http_json.cpp +++ /dev/null @@ -1,200 +0,0 @@ -///\file conn_http_json.cpp -///\brief Contains the main code for the HTTP JSON Connector - -#include <iostream> -#include <queue> -#include <sstream> -#include <iomanip> - -#include <cstdlib> -#include <cstdio> -#include <cmath> -#include <unistd.h> -#include <sys/types.h> -#include <sys/wait.h> -#include <getopt.h> - -#include <mist/socket.h> -#include <mist/http_parser.h> -#include <mist/dtsc.h> -#include <mist/flv_tag.h> -#include <mist/amf.h> -#include <mist/config.h> -#include <mist/stream.h> -#include <mist/timing.h> - -///\brief Holds everything unique to HTTP Connectors. -namespace Connector_HTTP { - ///\brief Main function for the HTTP Progressive Connector - ///\param conn A socket describing the connection the client. - ///\return The exit code of the connector. - int JSONConnector(Socket::Connection & conn){ - DTSC::Stream Strm; //Incoming stream buffer. - HTTP::Parser HTTP_R, HTTP_S;//HTTP Receiver en HTTP Sender. - bool inited = false;//Whether the stream is initialized - Socket::Connection ss( -1);//The Stream Socket, used to connect to the desired stream. - std::string streamname;//Will contain the name of the stream. - - unsigned int lastStats = 0;//Indicates the last time that we have sent stats to the server socket. - unsigned int seek_sec = 0;//Seek position in ms - unsigned int seek_byte = 0;//Seek position in bytes - - std::stringstream jsondata; - - while (conn.connected()){ - //Only attempt to parse input when not yet init'ed. - if ( !inited){ - if (conn.spool() && HTTP_R.Read(conn)){ -#if DEBUG >= 5 - std::cout << "Received request: " << HTTP_R.getUrl() << std::endl; -#endif - conn.setHost(HTTP_R.GetHeader("X-Origin")); - streamname = HTTP_R.GetHeader("X-Stream"); - - int start = 0; - if ( !HTTP_R.GetVar("start").empty()){ - start = atoi(HTTP_R.GetVar("start").c_str()); - } - if ( !HTTP_R.GetVar("starttime").empty()){ - start = atoi(HTTP_R.GetVar("starttime").c_str()); - } - if ( !HTTP_R.GetVar("apstart").empty()){ - start = atoi(HTTP_R.GetVar("apstart").c_str()); - } - if ( !HTTP_R.GetVar("ec_seek").empty()){ - start = atoi(HTTP_R.GetVar("ec_seek").c_str()); - } - if ( !HTTP_R.GetVar("fs").empty()){ - start = atoi(HTTP_R.GetVar("fs").c_str()); - } - //under 3 hours we assume seconds, otherwise byte position - if (start < 10800){ - seek_byte = start * 1000; //ms, not s - }else{ - seek_byte = start * 1000; //divide by 1mbit, then *1000 for ms. - } - // ready4data = true; - HTTP_R.Clean(); //clean for any possible next requests - jsondata.clear(); - jsondata << "["; - - //we are ready, connect the socket! - if ( !ss.connected()){ - ss = Util::Stream::getStream(streamname); - } - if ( !ss.connected()){ - #if DEBUG >= 1 - fprintf(stderr, "Could not connect to server for %s!\n", streamname.c_str()); - #endif - ss.close(); - HTTP_S.Clean(); - HTTP_S.SetBody("No such stream is available on the system. Please try again.\n"); - conn.SendNow(HTTP_S.BuildResponse("404", "Not found")); - //ready4data = false; - inited = false; - continue; - } - - //wait until we have a header - while ( !Strm.metadata && ss.connected()){ - if (ss.spool()){ - Strm.parsePacket(ss.Received()); //read the metadata - }else{ - Util::sleep(5); - } - } - - seek_sec = seek_byte; - - std::stringstream cmd; - cmd << "t"; - - int tid = -1; - for (std::map<int,DTSC::Track>::iterator it = Strm.metadata.tracks.begin(); it != Strm.metadata.tracks.end(); it++){ - if (it->second.type == "meta" ){ - if (tid == -1){ - tid = it->second.trackID; - } - cmd << " " << it->second.trackID; - } - } - - if( cmd.str() == "t" ){ - cmd.str(""); - cmd.clear(); - } - - int maxTime = Strm.metadata.tracks[tid].lastms; - - cmd << "\ns " << seek_sec << "\np " << maxTime << "\n"; - ss.SendNow(cmd.str().c_str(), cmd.str().size()); - inited = true; - - } - } - if (inited){ - - unsigned int now = Util::epoch(); - if (now != lastStats){ - lastStats = now; - ss.SendNow(conn.getStats("HTTP_JSON").c_str()); - } - - if (ss.spool()){ - while (Strm.parsePacket(ss.Received())){ - if(Strm.lastType() == DTSC::PAUSEMARK){ - HTTP_S.Clean(); //make sure no parts of old requests are left in any buffers - HTTP_S.SetHeader("Content-Type", "application/json"); //Send the correct content-type for FLV files - jsondata << "]"; - HTTP_S.SetBody(jsondata.str()); - conn.SendNow(HTTP_S.BuildResponse("200", "OK")); //no SetBody = unknown length - this is intentional, we will stream the entire file - inited = false; - jsondata.str(""); // totally do this - jsondata.clear(); - break; - } - - if (jsondata.str().length() > 1){ - jsondata << ","; - } - - jsondata << Strm.getPacket().toString(); - } - }else{ - Util::sleep(1); - } - if ( !ss.connected()){ - break; - } - } - - } - conn.close(); - ss.SendNow(conn.getStats("HTTP_JSON").c_str()); - ss.close(); - return 0; - } //SRT main function - -} //Connector_HTTP namespace - -///\brief The standard process-spawning main function. -int main(int argc, char ** argv){ - Util::Config conf(argv[0], PACKAGE_VERSION); - JSON::Value capa; - capa["desc"] = "Enables HTTP protocol JSON streaming."; - capa["deps"] = "HTTP"; - capa["url_rel"] = "/$.json"; - capa["url_match"] = "/$.json"; - capa["url_handler"] = "http"; - capa["url_type"] = "json"; - capa["socket"] = "http_json"; - conf.addBasicConnectorOptions(capa); - conf.parseArgs(argc, argv); - - if (conf.getBool("json")){ - std::cout << capa.toString() << std::endl; - return -1; - } - - return conf.serveForkedSocket(Connector_HTTP::JSONConnector); -} //main diff --git a/src/connectors/conn_http_live.cpp b/src/connectors/conn_http_live.cpp deleted file mode 100644 index a0956c9f..00000000 --- a/src/connectors/conn_http_live.cpp +++ /dev/null @@ -1,354 +0,0 @@ -/// \file conn_http_dynamic.cpp -/// Contains the main code for the HTTP Dynamic Connector - -#include <iostream> -#include <iomanip> -#include <sstream> -#include <queue> -#include <cstdlib> -#include <cstdio> -#include <cmath> -#include <unistd.h> -#include <sys/types.h> -#include <sys/wait.h> -#include <getopt.h> -#include <mist/socket.h> -#include <mist/http_parser.h> -#include <mist/json.h> -#include <mist/dtsc.h> -#include <mist/mp4.h> -#include <mist/mp4_generic.h> -#include <mist/config.h> -#include <sstream> -#include <mist/stream.h> -#include <mist/timing.h> -#include <mist/ts_packet.h> - -/// Holds everything unique to HTTP Connectors. -namespace Connector_HTTP { - ///\brief Builds an index file for HTTP Live streaming. - ///\param metadata The current metadata, used to generate the index. - ///\param isLive Whether or not the stream is live. - ///\return The index file for HTTP Live Streaming. - std::string liveIndex(DTSC::Meta & metadata, bool isLive){ - std::stringstream result; - result << "#EXTM3U\r\n"; - int audioId = -1; - std::string audioName; - for (std::map<int,DTSC::Track>::iterator it = metadata.tracks.begin(); it != metadata.tracks.end(); it++){ - if (it->second.codec == "AAC"){ - audioId = it->first; - audioName = it->second.getIdentifier(); - break; - } - } - for (std::map<int,DTSC::Track>::iterator it = metadata.tracks.begin(); it != metadata.tracks.end(); it++){ - if (it->second.codec == "H264"){ - int bWidth = it->second.bps * 2; - if (audioId != -1){ - bWidth += metadata.tracks[audioId].bps * 2; - } - result << "#EXT-X-STREAM-INF:PROGRAM-ID=1,BANDWIDTH=" << bWidth * 10 << "\r\n"; - result << it->first; - if (audioId != -1){ - result << "_" << audioId; - } - result << "/index.m3u8\r\n"; - } - } -#if DEBUG >= 8 - std::cerr << "Sending this index:" << std::endl << result.str() << std::endl; -#endif - return result.str(); - } - - std::string liveIndex(DTSC::Track & metadata, bool isLive){ - std::stringstream result; - //parse single track - int longestFragment = 0; - for (std::deque<DTSC::Fragment>::iterator it = metadata.fragments.begin(); (it + 1) != metadata.fragments.end(); it++){ - if (it->getDuration() > longestFragment){ - longestFragment = it->getDuration(); - } - } - result << "#EXTM3U\r\n" - "#EXT-X-TARGETDURATION:" << (longestFragment / 1000) + 1 << "\r\n" - "#EXT-X-MEDIA-SEQUENCE:" << metadata.missedFrags << "\r\n"; - for (std::deque<DTSC::Fragment>::iterator it = metadata.fragments.begin(); it != metadata.fragments.end(); it++){ - long long int starttime = metadata.getKey(it->getNumber()).getTime(); - - if (it != (metadata.fragments.end() - 1)){ - result << "#EXTINF:" << ((it->getDuration() + 500) / 1000) << ", no desc\r\n" << starttime << "_" << it->getDuration() + starttime << ".ts\r\n"; - } - } - if ( !isLive){ - result << "#EXT-X-ENDLIST\r\n"; - } -#if DEBUG >= 8 - std::cerr << "Sending this index:" << std::endl << result.str() << std::endl; -#endif - return result.str(); - } //liveIndex - - ///\brief Main function for the HTTP Live Connector - ///\param conn A socket describing the connection the client. - ///\return The exit code of the connector. - int liveConnector(Socket::Connection & conn){ - DTSC::Stream Strm; //Incoming stream buffer. - HTTP::Parser HTTP_R, HTTP_S; //HTTP Receiver en HTTP Sender. - - bool ready4data = false; //Set to true when streaming is to begin. - bool AppleCompat = false; //Set to true when Apple device detected. - Socket::Connection ss( -1); - std::string streamname; - bool handlingRequest = false; - std::string recBuffer = ""; - - TS::Packet PackData; - int PacketNumber = 0; - long long unsigned int TimeStamp = 0; - unsigned int ThisNaluSize; - char VideoCounter = 0; - char AudioCounter = 0; - long long unsigned int lastVid = 0; - bool IsKeyFrame = false; - MP4::AVCC avccbox; - bool haveAvcc = false; - - std::vector<int> fragIndices; - - std::string manifestType; - - int Segment = -1; - int temp; - int trackID = 0; - int audioTrackID = 0; - unsigned int lastStats = 0; - conn.setBlocking(false); //do not block on conn.spool() when no data is available - - while (conn.connected()){ - if ( !handlingRequest){ - if (conn.spool() && HTTP_R.Read(conn)){ - #if DEBUG >= 5 - std::cout << "Received request: " << HTTP_R.getUrl() << std::endl; - #endif - conn.setHost(HTTP_R.GetHeader("X-Origin")); - AppleCompat = (HTTP_R.GetHeader("User-Agent").find("Apple") != std::string::npos); - streamname = HTTP_R.GetHeader("X-Stream"); - if ( !ss){ - ss = Util::Stream::getStream(streamname); - if ( !ss.connected()){ - #if DEBUG >= 1 - fprintf(stderr, "Could not connect to server!\n"); - #endif - HTTP_S.Clean(); - HTTP_S.SetBody("No such stream is available on the system. Please try again.\n"); - conn.SendNow(HTTP_S.BuildResponse("404", "Not found")); - ready4data = false; - continue; - } - ss.setBlocking(false); - Strm.waitForMeta(ss); - } - if (HTTP_R.url.find(".m3u") == std::string::npos){ - temp = HTTP_R.url.find("/", 5) + 1; - std::string allTracks = HTTP_R.url.substr(temp, HTTP_R.url.find("/", temp) - temp); - trackID = atoi(allTracks.c_str()); - audioTrackID = atoi(allTracks.substr(allTracks.find("_")+1).c_str()); - temp = HTTP_R.url.find("/", temp) + 1; - Segment = atoi(HTTP_R.url.substr(temp, HTTP_R.url.find("_", temp) - temp).c_str()); - lastVid = Segment * 90; - temp = HTTP_R.url.find("_", temp) + 1; - int frameCount = atoi(HTTP_R.url.substr(temp, HTTP_R.url.find(".ts", temp) - temp).c_str()); - if (Strm.metadata.live){ - int seekable = Strm.canSeekms(Segment); - if (seekable < 0){ - HTTP_S.Clean(); - HTTP_S.SetBody("The requested fragment is no longer kept in memory on the server and cannot be served.\n"); - conn.SendNow(HTTP_S.BuildResponse("412", "Fragment out of range")); - HTTP_R.Clean(); //clean for any possible next requests - std::cout << "Fragment @ " << Segment << " too old" << std::endl; - continue; - } - if (seekable > 0){ - HTTP_S.Clean(); - HTTP_S.SetBody("Proxy, re-request this in a second or two.\n"); - conn.SendNow(HTTP_S.BuildResponse("208", "Ask again later")); - HTTP_R.Clean(); //clean for any possible next requests - std::cout << "Fragment @ " << Segment << " not available yet" << std::endl; - continue; - } - } - for (unsigned int i = 0; i < allTracks.size(); i++){ - if (allTracks[i] == '_'){ - allTracks[i] = ' '; - } - } - std::stringstream sstream; - sstream << "t " << allTracks << "\n"; - sstream << "s " << Segment << "\n"; - sstream << "p " << frameCount << "\n"; - ss.SendNow(sstream.str().c_str()); - - HTTP_S.Clean(); - HTTP_S.SetHeader("Content-Type", "video/mp2t"); - HTTP_S.StartResponse(HTTP_R, conn); - handlingRequest = true; - }else{ - std::string request = HTTP_R.url.substr(HTTP_R.url.find("/", 5) + 1); - if (HTTP_R.url.find(".m3u8") != std::string::npos){ - manifestType = "audio/x-mpegurl"; - }else{ - manifestType = "audio/mpegurl"; - } - HTTP_S.Clean(); - HTTP_S.SetHeader("Content-Type", manifestType); - HTTP_S.SetHeader("Cache-Control", "no-cache"); - std::string manifest; - if (request.find("/") == std::string::npos){ - manifest = liveIndex(Strm.metadata, Strm.metadata.live); - }else{ - int selectId = atoi(request.substr(0,request.find("/")).c_str()); - manifest = liveIndex(Strm.metadata.tracks[selectId], Strm.metadata.live); - } - HTTP_S.SetBody(manifest); - conn.SendNow(HTTP_S.BuildResponse("200", "OK")); - } - ready4data = true; - HTTP_R.Clean(); //clean for any possible next requests - }else{ - Util::sleep(250); - } - } - if (ready4data){ - unsigned int now = Util::epoch(); - if (now != lastStats){ - lastStats = now; - ss.SendNow(conn.getStats("HTTP_Live").c_str()); - } - if (ss.spool()){ - while (Strm.parsePacket(ss.Received())){ - if (Strm.lastType() == DTSC::PAUSEMARK){ - HTTP_S.Chunkify("", 0, conn); - handlingRequest = false; - } - if ( !haveAvcc){ - avccbox.setPayload(Strm.metadata.tracks[trackID].init); - haveAvcc = true; - } - if (Strm.lastType() == DTSC::VIDEO || Strm.lastType() == DTSC::AUDIO){ - Socket::Buffer ToPack; - //write PAT and PMT TS packets - if (PacketNumber % 42 == 0){ - PackData.DefaultPAT(); - HTTP_S.Chunkify(PackData.ToString(), 188, conn); - PackData.DefaultPMT(); - HTTP_S.Chunkify(PackData.ToString(), 188, conn); - PacketNumber += 2; - } - - int PIDno = 0; - char * ContCounter = 0; - if (Strm.lastType() == DTSC::VIDEO){ - IsKeyFrame = Strm.getPacket().isMember("keyframe"); - if (IsKeyFrame){ - TimeStamp = (Strm.getPacket()["time"].asInt() * 27000); - } - ToPack.append(avccbox.asAnnexB()); - while (Strm.lastData().size() > 4){ - ThisNaluSize = (Strm.lastData()[0] << 24) + (Strm.lastData()[1] << 16) + (Strm.lastData()[2] << 8) + Strm.lastData()[3]; - Strm.lastData().replace(0, 4, "\000\000\000\001", 4); - if (ThisNaluSize + 4 == Strm.lastData().size()){ - ToPack.append(Strm.lastData()); - break; - }else{ - ToPack.append(Strm.lastData().c_str(), ThisNaluSize + 4); - Strm.lastData().erase(0, ThisNaluSize + 4); - } - } - ToPack.prepend(TS::Packet::getPESVideoLeadIn(0ul, Strm.getPacket()["time"].asInt() * 90)); - PIDno = 0x100 - 1 + Strm.getPacket()["trackid"].asInt(); - ContCounter = &VideoCounter; - }else if (Strm.lastType() == DTSC::AUDIO){ - ToPack.append(TS::GetAudioHeader(Strm.lastData().size(), Strm.metadata.tracks[audioTrackID].init)); - ToPack.append(Strm.lastData()); - if (AppleCompat){ - ToPack.prepend(TS::Packet::getPESAudioLeadIn(ToPack.bytes(1073741824ul), lastVid)); - }else{ - ToPack.prepend(TS::Packet::getPESAudioLeadIn(ToPack.bytes(1073741824ul), Strm.getPacket()["time"].asInt() * 90)); - } - PIDno = 0x100 - 1 + Strm.getPacket()["trackid"].asInt(); - ContCounter = &AudioCounter; - IsKeyFrame = false; - } - - //initial packet - PackData.Clear(); - PackData.PID(PIDno); - PackData.ContinuityCounter(( *ContCounter)++); - PackData.UnitStart(1); - if (IsKeyFrame){ - PackData.RandomAccess(1); - PackData.PCR(TimeStamp); - } - unsigned int toSend = PackData.AddStuffing(ToPack.bytes(184)); - std::string gonnaSend = ToPack.remove(toSend); - PackData.FillFree(gonnaSend); - HTTP_S.Chunkify(PackData.ToString(), 188, conn); - PacketNumber++; - - //rest of packets - while (ToPack.size()){ - PackData.Clear(); - PackData.PID(PIDno); - PackData.ContinuityCounter(( *ContCounter)++); - toSend = PackData.AddStuffing(ToPack.bytes(184)); - gonnaSend = ToPack.remove(toSend); - PackData.FillFree(gonnaSend); - HTTP_S.Chunkify(PackData.ToString(), 188, conn); - PacketNumber++; - } - - } - } - } - if ( !ss.connected()){ - break; - } - } - } - conn.close(); - ss.SendNow(conn.getStats("HTTP_Live").c_str()); - ss.close(); -#if DEBUG >= 5 - fprintf(stderr, "HLS: User %i disconnected.\n", conn.getSocket()); -#endif - return 0; - } //HLS_Connector main function - -} //Connector_HTTP namespace - -///\brief The standard process-spawning main function. -int main(int argc, char ** argv){ - Util::Config conf(argv[0], PACKAGE_VERSION); - JSON::Value capa; - capa["desc"] = "Enables HTTP protocol Apple-specific streaming (also known as HLS)."; - capa["deps"] = "HTTP"; - capa["url_rel"] = "/hls/$/index.m3u8"; - capa["url_prefix"] = "/hls/$/"; - capa["socket"] = "http_live"; - capa["codecs"][0u][0u].append("H264"); - capa["codecs"][0u][1u].append("AAC"); - capa["methods"][0u]["handler"] = "http"; - capa["methods"][0u]["type"] = "html5/application/vnd.apple.mpegurl"; - capa["methods"][0u]["priority"] = 9ll; - conf.addBasicConnectorOptions(capa); - conf.parseArgs(argc, argv); - - if (conf.getBool("json")){ - std::cout << capa.toString() << std::endl; - return -1; - } - - return conf.serveForkedSocket(Connector_HTTP::liveConnector); -} //main diff --git a/src/connectors/conn_http_progressive_flv.cpp b/src/connectors/conn_http_progressive_flv.cpp deleted file mode 100644 index 9ec8ff23..00000000 --- a/src/connectors/conn_http_progressive_flv.cpp +++ /dev/null @@ -1,217 +0,0 @@ -///\file conn_http_progressive_flv.cpp -///\brief Contains the main code for the HTTP Progressive FLV Connector - -#include <iostream> -#include <queue> -#include <sstream> - -#include <cstdlib> -#include <cstdio> -#include <cmath> -#include <unistd.h> -#include <sys/types.h> -#include <sys/wait.h> - -#include <mist/socket.h> -#include <mist/http_parser.h> -#include <mist/dtsc.h> -#include <mist/flv_tag.h> -#include <mist/amf.h> -#include <mist/config.h> -#include <mist/stream.h> -#include <mist/timing.h> - -///\brief Holds everything unique to HTTP Connectors. -namespace Connector_HTTP { - ///\brief Main function for the HTTP Progressive Connector - ///\param conn A socket describing the connection the client. - ///\return The exit code of the connector. - int progressiveConnector(Socket::Connection & conn){ - bool progressive_has_sent_header = false;//Indicates whether we have sent a header. - bool ready4data = false; //Set to true when streaming is to begin. - DTSC::Stream Strm; //Incoming stream buffer. - HTTP::Parser HTTP_R, HTTP_S;//HTTP Receiver en HTTP Sender. - bool inited = false;//Whether the stream is initialized - Socket::Connection ss( -1);//The Stream Socket, used to connect to the desired stream. - std::string streamname;//Will contain the name of the stream. - FLV::Tag tag;//Temporary tag buffer. - - unsigned int lastStats = 0;//Indicates the last time that we have sent stats to the server socket. - unsigned int seek_sec = 0;//Seek position in ms - unsigned int seek_byte = 0;//Seek position in bytes - - int videoID = -1; - int audioID = -1; - - while (conn.connected()){ - //Only attempt to parse input when not yet init'ed. - if ( !inited){ - if (conn.spool() && HTTP_R.Read(conn)){ -#if DEBUG >= 5 - std::cout << "Received request: " << HTTP_R.getUrl() << std::endl; -#endif - conn.setHost(HTTP_R.GetHeader("X-Origin")); - streamname = HTTP_R.GetHeader("X-Stream"); - int start = 0; - if ( !HTTP_R.GetVar("start").empty()){ - start = atoi(HTTP_R.GetVar("start").c_str()); - } - if ( !HTTP_R.GetVar("starttime").empty()){ - start = atoi(HTTP_R.GetVar("starttime").c_str()); - } - if ( !HTTP_R.GetVar("apstart").empty()){ - start = atoi(HTTP_R.GetVar("apstart").c_str()); - } - if ( !HTTP_R.GetVar("ec_seek").empty()){ - start = atoi(HTTP_R.GetVar("ec_seek").c_str()); - } - if ( !HTTP_R.GetVar("fs").empty()){ - start = atoi(HTTP_R.GetVar("fs").c_str()); - } - //under 3 hours we assume seconds, otherwise byte position - if (start < 10800){ - seek_sec = start * 1000; //ms, not s - seek_byte = 0; - }else{ - seek_byte = start; //divide by 1mbit, then *1000 for ms. - seek_sec = 0; - } - ready4data = true; - HTTP_R.Clean(); //clean for any possible next requests - } - } - if (ready4data){ - if ( !inited){ - //we are ready, connect the socket! - ss = Util::Stream::getStream(streamname); - if ( !ss.connected()){ -#if DEBUG >= 1 - fprintf(stderr, "Could not connect to server for %s!\n", streamname.c_str()); -#endif - ss.close(); - HTTP_S.Clean(); - HTTP_S.SetBody("No such stream is available on the system. Please try again.\n"); - conn.SendNow(HTTP_S.BuildResponse("404", "Not found")); - ready4data = false; - continue; - } - Strm.waitForMeta(ss); - int byterate = 0; - for (std::map<int,DTSC::Track>::iterator it = Strm.metadata.tracks.begin(); it != Strm.metadata.tracks.end(); it++){ - if (videoID == -1 && (it->second.codec == "H264" || it->second.codec == "H263" || it->second.codec == "VP6")){ - videoID = it->second.trackID; - } - if (audioID == -1 && (it->second.codec == "AAC" || it->second.codec == "MP3")){ - audioID = it->second.trackID; - } - } - if (videoID != -1){ - byterate += Strm.metadata.tracks[videoID].bps; - } - if (audioID != -1){ - byterate += Strm.metadata.tracks[audioID].bps; - } - if ( !byterate){byterate = 1;} - if (seek_byte){ - seek_sec = (seek_byte / byterate) * 1000; - } - std::stringstream cmd; - cmd << "t"; - if (videoID != -1){ - cmd << " " << videoID; - } - if (audioID != -1){ - cmd << " " << audioID; - } - cmd << "\ns " << seek_sec << "\np\n"; - ss.SendNow(cmd.str().c_str(), cmd.str().size()); - inited = true; - } - unsigned int now = Util::epoch(); - if (now != lastStats){ - lastStats = now; - ss.SendNow(conn.getStats("HTTP_Progressive_FLV")); - } - if (ss.spool()){ - while (Strm.parsePacket(ss.Received())){ - if ( !progressive_has_sent_header){ - HTTP_S.Clean(); //make sure no parts of old requests are left in any buffers - HTTP_S.SetHeader("Content-Type", "video/x-flv"); //Send the correct content-type for FLV files - //HTTP_S.SetHeader("Transfer-Encoding", "chunked"); - HTTP_S.protocol = "HTTP/1.0"; - conn.SendNow(HTTP_S.BuildResponse("200", "OK")); //no SetBody = unknown length - this is intentional, we will stream the entire file - conn.SendNow(FLV::Header, 13); //write FLV header - //write metadata - tag.DTSCMetaInit(Strm, Strm.metadata.tracks[videoID], Strm.metadata.tracks[audioID]); - conn.SendNow(tag.data, tag.len); - //write video init data, if needed - if (videoID != -1){ - tag.DTSCVideoInit(Strm.metadata.tracks[videoID]); - conn.SendNow(tag.data, tag.len); - } - //write audio init data, if needed - if (audioID != -1){ - tag.DTSCAudioInit(Strm.metadata.tracks[audioID]); - conn.SendNow(tag.data, tag.len); - } - progressive_has_sent_header = true; - } - if (Strm.lastType() == DTSC::PAUSEMARK){ - conn.close(); - } - if (Strm.lastType() == DTSC::INVALID){ - #if DEBUG >= 3 - fprintf(stderr, "Invalid packet received - closing connection.\n"); - #endif - conn.close(); - } - if (Strm.lastType() == DTSC::AUDIO || Strm.lastType() == DTSC::VIDEO){ - std::string codec = Strm.metadata.tracks[Strm.getPacket()["trackid"].asInt()].codec; - if (codec == "AAC" || codec == "MP3" || codec == "H264" || codec == "H263" || codec == "VP6"){ - tag.DTSCLoader(Strm); - conn.SendNow(tag.data, tag.len); //write the tag contents - } - } - } - }else{ - Util::sleep(1); - } - if ( !ss.connected()){ - break; - } - } - } - conn.close(); - ss.SendNow(conn.getStats("HTTP_Progressive_FLV").c_str()); - ss.close(); - return 0; - } //Progressive_Connector main function - -} //Connector_HTTP namespace - -///\brief The standard process-spawning main function. -int main(int argc, char ** argv){ - Util::Config conf(argv[0], PACKAGE_VERSION); - JSON::Value capa; - capa["desc"] = "Enables HTTP protocol progressive streaming."; - capa["deps"] = "HTTP"; - capa["url_rel"] = "/$.flv"; - capa["url_match"] = "/$.flv"; - capa["socket"] = "http_progressive_flv"; - capa["codecs"][0u][0u].append("H264"); - capa["codecs"][0u][0u].append("H263"); - capa["codecs"][0u][0u].append("VP6"); - capa["codecs"][0u][1u].append("AAC"); - capa["codecs"][0u][1u].append("MP3"); - capa["methods"][0u]["handler"] = "http"; - capa["methods"][0u]["type"] = "flash/7"; - capa["methods"][0u]["priority"] = 5ll; - conf.addBasicConnectorOptions(capa); - conf.parseArgs(argc, argv); - - if (conf.getBool("json")){ - std::cout << capa.toString() << std::endl; - return -1; - } - return conf.serveForkedSocket(Connector_HTTP::progressiveConnector); -} //main diff --git a/src/connectors/conn_http_progressive_mp3.cpp b/src/connectors/conn_http_progressive_mp3.cpp deleted file mode 100644 index 21a335f6..00000000 --- a/src/connectors/conn_http_progressive_mp3.cpp +++ /dev/null @@ -1,184 +0,0 @@ -///\file conn_http_progressive_mp3.cpp -///\brief Contains the main code for the HTTP Progressive MP3 Connector - -#include <iostream> -#include <queue> -#include <sstream> - -#include <cstdlib> -#include <cstdio> -#include <cmath> -#include <unistd.h> -#include <sys/types.h> -#include <sys/wait.h> - -#include <mist/socket.h> -#include <mist/http_parser.h> -#include <mist/dtsc.h> -#include <mist/flv_tag.h> -#include <mist/amf.h> -#include <mist/config.h> -#include <mist/stream.h> -#include <mist/timing.h> - -///\brief Holds everything unique to HTTP Connectors. -namespace Connector_HTTP { - ///\brief Main function for the HTTP Progressive Connector - ///\param conn A socket describing the connection the client. - ///\return The exit code of the connector. - int progressiveConnector(Socket::Connection & conn){ - bool progressive_has_sent_header = false;//Indicates whether we have sent a header. - bool ready4data = false; //Set to true when streaming is to begin. - DTSC::Stream Strm; //Incoming stream buffer. - HTTP::Parser HTTP_R, HTTP_S;//HTTP Receiver en HTTP Sender. - bool inited = false;//Whether the stream is initialized - Socket::Connection ss( -1);//The Stream Socket, used to connect to the desired stream. - std::string streamname;//Will contain the name of the stream. - FLV::Tag tag;//Temporary tag buffer. - - unsigned int lastStats = 0;//Indicates the last time that we have sent stats to the server socket. - unsigned int seek_sec = 0;//Seek position in ms - unsigned int seek_byte = 0;//Seek position in bytes - - int audioID = -1; - - while (conn.connected()){ - //Only attempt to parse input when not yet init'ed. - if ( !inited){ - if (conn.spool() && HTTP_R.Read(conn)){ -#if DEBUG >= 5 - std::cout << "Received request: " << HTTP_R.getUrl() << std::endl; -#endif - conn.setHost(HTTP_R.GetHeader("X-Origin")); - streamname = HTTP_R.GetHeader("X-Stream"); - int start = 0; - if ( !HTTP_R.GetVar("start").empty()){ - start = atoi(HTTP_R.GetVar("start").c_str()); - } - if ( !HTTP_R.GetVar("starttime").empty()){ - start = atoi(HTTP_R.GetVar("starttime").c_str()); - } - if ( !HTTP_R.GetVar("apstart").empty()){ - start = atoi(HTTP_R.GetVar("apstart").c_str()); - } - if ( !HTTP_R.GetVar("ec_seek").empty()){ - start = atoi(HTTP_R.GetVar("ec_seek").c_str()); - } - if ( !HTTP_R.GetVar("fs").empty()){ - start = atoi(HTTP_R.GetVar("fs").c_str()); - } - //under 3 hours we assume seconds, otherwise byte position - if (start < 10800){ - seek_sec = start * 1000; //ms, not s - }else{ - seek_byte = start; //divide by 1mbit, then *1000 for ms. - } - ready4data = true; - HTTP_R.Clean(); //clean for any possible next requests - } - } - if (ready4data){ - if ( !inited){ - //we are ready, connect the socket! - ss = Util::Stream::getStream(streamname); - if ( !ss.connected()){ -#if DEBUG >= 1 - fprintf(stderr, "Could not connect to server for %s!\n", streamname.c_str()); -#endif - ss.close(); - HTTP_S.Clean(); - HTTP_S.SetBody("No such stream is available on the system. Please try again.\n"); - conn.SendNow(HTTP_S.BuildResponse("404", "Not found")); - ready4data = false; - continue; - } - Strm.waitForMeta(ss); - int byterate = 0; - for (std::map<int,DTSC::Track>::iterator it = Strm.metadata.tracks.begin(); it != Strm.metadata.tracks.end(); it++){ - if (audioID == -1 && it->second.codec == "MP3"){ - audioID = it->second.trackID; - } - } - if (audioID != -1){ - byterate += Strm.metadata.tracks[audioID].bps; - } - if ( !byterate){byterate = 1;} - if (seek_byte){ - seek_sec = (seek_byte / byterate) * 1000; - } - std::stringstream cmd; - cmd << "t"; - if (audioID != -1){ - cmd << " " << audioID; - } - cmd << "\ns " << seek_sec << "\np\n"; - ss.SendNow(cmd.str().c_str(), cmd.str().size()); - inited = true; - } - unsigned int now = Util::epoch(); - if (now != lastStats){ - lastStats = now; - ss.SendNow(conn.getStats("HTTP_Progressive").c_str()); - } - if (ss.spool()){ - while (Strm.parsePacket(ss.Received())){ - if ( !progressive_has_sent_header){ - HTTP_S.Clean(); //make sure no parts of old requests are left in any buffers - HTTP_S.SetHeader("Content-Type", "audio/mpeg"); //Send the correct content-type for MP3 files - //HTTP_S.SetHeader("Transfer-Encoding", "chunked"); - HTTP_S.protocol = "HTTP/1.0"; - conn.SendNow(HTTP_S.BuildResponse("200", "OK")); //no SetBody = unknown length - this is intentional, we will stream the entire file - progressive_has_sent_header = true; - } - if (Strm.lastType() == DTSC::PAUSEMARK){ - conn.close(); - } - if (Strm.lastType() == DTSC::INVALID){ - #if DEBUG >= 3 - fprintf(stderr, "Invalid packet received - closing connection.\n"); - #endif - conn.close(); - } - if (Strm.lastType() == DTSC::AUDIO){ - conn.SendNow(Strm.lastData()); //write the MP3 contents - } - } - }else{ - Util::sleep(1); - } - if ( !ss.connected()){ - break; - } - } - } - conn.close(); - ss.SendNow(conn.getStats("HTTP_Dynamic").c_str()); - ss.close(); - return 0; - } //Progressive_Connector main function - -} //Connector_HTTP namespace - -///\brief The standard process-spawning main function. -int main(int argc, char ** argv){ - Util::Config conf(argv[0], PACKAGE_VERSION); - JSON::Value capa; - capa["desc"] = "Enables HTTP protocol progressive streaming."; - capa["deps"] = "HTTP"; - capa["codecs"][0u][0u].append("MP3"); - capa["url_rel"] = "/$.mp3"; - capa["url_match"] = "/$.mp3"; - capa["socket"] = "http_progressive_mp3"; - capa["methods"][0u]["handler"] = "http"; - capa["methods"][0u]["type"] = "mp3"; - capa["methods"][0u]["priority"] = 8ll; - conf.addBasicConnectorOptions(capa); - conf.parseArgs(argc, argv); - - if (conf.getBool("json")){ - std::cout << capa.toString() << std::endl; - return -1; - } - - return conf.serveForkedSocket(Connector_HTTP::progressiveConnector); -} //main diff --git a/src/connectors/conn_http_progressive_mp4.cpp b/src/connectors/conn_http_progressive_mp4.cpp deleted file mode 100644 index 5e8dbf79..00000000 --- a/src/connectors/conn_http_progressive_mp4.cpp +++ /dev/null @@ -1,656 +0,0 @@ -///\file conn_http_progressive_mp4.cpp -///\brief Contains the main code for the HTTP Progressive MP4 Connector - -#include <iostream> -#include <queue> -#include <sstream> - -#include <cstdlib> -#include <cstdio> -#include <cmath> -#include <unistd.h> -#include <sys/types.h> -#include <sys/wait.h> - -#include <mist/socket.h> -#include <mist/http_parser.h> -#include <mist/dtsc.h> -#include <mist/mp4.h> -#include <mist/mp4_generic.h> -#include <mist/amf.h> -#include <mist/config.h> -#include <mist/stream.h> -#include <mist/timing.h> -#include <mist/defines.h> - -///\brief Holds everything unique to HTTP Connectors. -namespace Connector_HTTP { - - struct keyPart{ - public: - bool operator < (const keyPart& rhs) const { - if (time < rhs.time){ - return true; - } - if (time == rhs.time){ - if (trackID < rhs.trackID){ - return true; - } - } - return false; - } - long unsigned int trackID; - long unsigned int size; - long long unsigned int time; - long long unsigned int endTime; - long unsigned int index; - }; - - std::string DTSCMeta2MP4Header(DTSC::Meta & metaData, std::set<int> & tracks, long long & size){ - std::stringstream header; - //ftyp box - MP4::FTYP ftypBox; - header << std::string(ftypBox.asBox(),ftypBox.boxedSize()); - - uint64_t mdatSize = 0; - //moov box - MP4::MOOV moovBox; - unsigned int moovOffset = 0; - { - //calculating longest duration - long long int firstms = -1; - long long int lastms = -1; - for (std::set<int>::iterator it = tracks.begin(); it != tracks.end(); it++) { - if (lastms == -1 || lastms < metaData.tracks[*it].lastms){ - lastms = metaData.tracks[*it].lastms; - } - if (firstms == -1 || firstms > metaData.tracks[*it].firstms){ - firstms = metaData.tracks[*it].firstms; - } - } - MP4::MVHD mvhdBox(lastms - firstms); - moovBox.setContent(mvhdBox, moovOffset++); - } - for (std::set<int>::iterator it = tracks.begin(); it != tracks.end(); it++) { - MP4::TRAK trakBox; - { - { - MP4::TKHD tkhdBox(*it, metaData.tracks[*it].lastms - metaData.tracks[*it].firstms, metaData.tracks[*it].width, metaData.tracks[*it].height); - trakBox.setContent(tkhdBox, 0); - }{ - MP4::MDIA mdiaBox; - unsigned int mdiaOffset = 0; - { - MP4::MDHD mdhdBox(metaData.tracks[*it].lastms - metaData.tracks[*it].firstms); - mdiaBox.setContent(mdhdBox, mdiaOffset++); - }//MDHD box - { - MP4::HDLR hdlrBox(metaData.tracks[*it].type, metaData.tracks[*it].getIdentifier()); - mdiaBox.setContent(hdlrBox, mdiaOffset++); - }//hdlr box - { - MP4::MINF minfBox; - unsigned int minfOffset = 0; - if (metaData.tracks[*it].type== "video"){ - MP4::VMHD vmhdBox; - vmhdBox.setFlags(1); - minfBox.setContent(vmhdBox,minfOffset++); - }else if (metaData.tracks[*it].type == "audio"){ - MP4::SMHD smhdBox; - minfBox.setContent(smhdBox,minfOffset++); - }//type box - { - MP4::DINF dinfBox; - MP4::DREF drefBox; - dinfBox.setContent(drefBox,0); - minfBox.setContent(dinfBox,minfOffset++); - }//dinf box - { - MP4::STBL stblBox; - unsigned int offset = 0; - { - MP4::STSD stsdBox; - stsdBox.setVersion(0); - if (metaData.tracks[*it].type == "video"){//boxname = codec - MP4::VisualSampleEntry vse; - if (metaData.tracks[*it].codec == "H264"){ - vse.setCodec("avc1"); - } - vse.setDataReferenceIndex(1); - vse.setWidth(metaData.tracks[*it].width); - vse.setHeight(metaData.tracks[*it].height); - MP4::AVCC avccBox; - avccBox.setPayload(metaData.tracks[*it].init); - vse.setCLAP(avccBox); - stsdBox.setEntry(vse,0); - }else if(metaData.tracks[*it].type == "audio"){//boxname = codec - MP4::AudioSampleEntry ase; - if (metaData.tracks[*it].codec == "AAC"){ - ase.setCodec("mp4a"); - ase.setDataReferenceIndex(1); - } - ase.setSampleRate(metaData.tracks[*it].rate); - ase.setChannelCount(metaData.tracks[*it].channels); - ase.setSampleSize(metaData.tracks[*it].size); - //MP4::ESDS esdsBox(metaData.tracks[*it].init, metaData.tracks[*it].bps); - MP4::ESDS esdsBox; - - //outputting these values first, so malloc isn't called as often. - esdsBox.setESHeaderStartCodes(metaData.tracks[*it].init); - esdsBox.setSLValue(2); - - esdsBox.setESDescriptorTypeLength(32+metaData.tracks[*it].init.size()); - esdsBox.setESID(2); - esdsBox.setStreamPriority(0); - esdsBox.setDecoderConfigDescriptorTypeLength(18 + metaData.tracks[*it].init.size()); - esdsBox.setByteObjectTypeID(0x40); - esdsBox.setStreamType(5); - esdsBox.setReservedFlag(1); - esdsBox.setBufferSize(1250000); - esdsBox.setMaximumBitRate(10000000); - esdsBox.setAverageBitRate(metaData.tracks[*it].bps * 8); - esdsBox.setConfigDescriptorTypeLength(5); - esdsBox.setSLConfigDescriptorTypeTag(0x6); - esdsBox.setSLConfigExtendedDescriptorTypeTag(0x808080); - esdsBox.setSLDescriptorTypeLength(1); - ase.setCodecBox(esdsBox); - stsdBox.setEntry(ase,0); - } - stblBox.setContent(stsdBox,offset++); - }//stsd box - { - MP4::STTS sttsBox; - sttsBox.setVersion(0); - if (metaData.tracks[*it].parts.size()){ - for (unsigned int part = 0; part < metaData.tracks[*it].parts.size(); part++){ - MP4::STTSEntry newEntry; - newEntry.sampleCount = 1; - newEntry.sampleDelta = metaData.tracks[*it].parts[part].getDuration(); - sttsBox.setSTTSEntry(newEntry, part); - } - } - stblBox.setContent(sttsBox,offset++); - }//stts box - if (metaData.tracks[*it].type == "video"){ - //STSS Box here - MP4::STSS stssBox; - stssBox.setVersion(0); - int tmpCount = 1; - int tmpItCount = 0; - for ( std::deque< DTSC::Key>::iterator tmpIt = metaData.tracks[*it].keys.begin(); tmpIt != metaData.tracks[*it].keys.end(); tmpIt ++) { - stssBox.setSampleNumber(tmpCount,tmpItCount); - tmpCount += tmpIt->getParts(); - tmpItCount ++; - } - stblBox.setContent(stssBox,offset++); - }//stss box - { - MP4::STSC stscBox; - stscBox.setVersion(0); - MP4::STSCEntry stscEntry; - stscEntry.firstChunk = 1; - stscEntry.samplesPerChunk = 1; - stscEntry.sampleDescriptionIndex = 1; - stscBox.setSTSCEntry(stscEntry, 0); - stblBox.setContent(stscBox,offset++); - }//stsc box - { - uint32_t total = 0; - MP4::STSZ stszBox; - stszBox.setVersion(0); - total = 0; - for (std::deque< DTSC::Part>::iterator partIt = metaData.tracks[*it].parts.begin(); partIt != metaData.tracks[*it].parts.end(); partIt ++) { - stszBox.setEntrySize(partIt->getSize(), total);//in bytes in file - size += partIt->getSize(); - total++; - } - stblBox.setContent(stszBox,offset++); - }//stsz box - //add STCO boxes here - { - MP4::STCO stcoBox; - stcoBox.setVersion(1); - //Inserting empty values on purpose here, will be fixed later. - if (metaData.tracks[*it].parts.size() != 0){ - stcoBox.setChunkOffset(0, metaData.tracks[*it].parts.size() - 1);//this inserts all empty entries at once - } - stblBox.setContent(stcoBox,offset++); - }//stco box - minfBox.setContent(stblBox,minfOffset++); - }//stbl box - mdiaBox.setContent(minfBox, mdiaOffset++); - }//minf box - trakBox.setContent(mdiaBox, 1); - } - }//trak Box - moovBox.setContent(trakBox, moovOffset++); - } - //initial offset length ftyp, length moov + 8 - unsigned long long int byteOffset = ftypBox.boxedSize() + moovBox.boxedSize() + 8; - //update all STCO from the following map; - std::map <int, MP4::STCO> checkStcoBoxes; - //for all tracks - for (unsigned int i = 1; i < moovBox.getContentCount(); i++){ - //10 lines to get the STCO box. - MP4::TRAK checkTrakBox; - MP4::Box checkMdiaBox; - MP4::Box checkTkhdBox; - MP4::MINF checkMinfBox; - MP4::STBL checkStblBox; - //MP4::STCO checkStcoBox; - checkTrakBox = ((MP4::TRAK&)moovBox.getContent(i)); - for (unsigned int j = 0; j < checkTrakBox.getContentCount(); j++){ - if (checkTrakBox.getContent(j).isType("mdia")){ - checkMdiaBox = checkTrakBox.getContent(j); - break; - } - if (checkTrakBox.getContent(j).isType("tkhd")){ - checkTkhdBox = checkTrakBox.getContent(j); - } - } - for (unsigned int j = 0; j < ((MP4::MDIA&)checkMdiaBox).getContentCount(); j++){ - if (((MP4::MDIA&)checkMdiaBox).getContent(j).isType("minf")){ - checkMinfBox = ((MP4::MINF&)((MP4::MDIA&)checkMdiaBox).getContent(j)); - break; - } - } - for (unsigned int j = 0; j < checkMinfBox.getContentCount(); j++){ - if (checkMinfBox.getContent(j).isType("stbl")){ - checkStblBox = ((MP4::STBL&)checkMinfBox.getContent(j)); - break; - } - } - for (unsigned int j = 0; j < checkStblBox.getContentCount(); j++){ - if (checkStblBox.getContent(j).isType("stco")){ - checkStcoBoxes.insert( std::pair<int, MP4::STCO>(((MP4::TKHD&)checkTkhdBox).getTrackID(), ((MP4::STCO&)checkStblBox.getContent(j)) )); - break; - } - } - } - //inserting right values in the STCO box header - //total = 0; - long long unsigned int totalByteOffset = 0; - //Current values are actual byte offset without header-sized offset - std::set <keyPart> sortSet;//filling sortset for interleaving parts - for (std::set<int>::iterator subIt = tracks.begin(); subIt != tracks.end(); subIt++) { - keyPart temp; - temp.trackID = *subIt; - temp.time = metaData.tracks[*subIt].firstms;//timeplace of frame - temp.endTime = metaData.tracks[*subIt].firstms + metaData.tracks[*subIt].parts[0].getDuration(); - temp.size = metaData.tracks[*subIt].parts[0].getSize();//bytesize of frame (alle parts all together) - temp.index = 0; - sortSet.insert(temp); - } - while (!sortSet.empty()){ - //setting the right STCO size in the STCO box - checkStcoBoxes[sortSet.begin()->trackID].setChunkOffset(totalByteOffset + byteOffset, sortSet.begin()->index); - totalByteOffset += sortSet.begin()->size; - //add keyPart to sortSet - keyPart temp; - temp.index = sortSet.begin()->index + 1; - temp.trackID = sortSet.begin()->trackID; - if(temp.index < metaData.tracks[temp.trackID].parts.size() ){//only insert when there are parts left - temp.time = sortSet.begin()->endTime;//timeplace of frame - temp.endTime = sortSet.begin()->endTime + metaData.tracks[temp.trackID].parts[temp.index].getDuration(); - temp.size = metaData.tracks[temp.trackID].parts[temp.index].getSize();//bytesize of frame - sortSet.insert(temp); - } - //remove highest keyPart - sortSet.erase(sortSet.begin()); - } - - mdatSize = totalByteOffset+8; - - header << std::string(moovBox.asBox(),moovBox.boxedSize()); - - header << (char)((mdatSize>>24) & 0xFF) << (char)((mdatSize>>16) & 0xFF) << (char)((mdatSize>>8) & 0xFF) << (char)(mdatSize & 0xFF) << "mdat"; - //end of header - - size += header.str().size(); - return header.str(); - } - - /// Calculate a seekPoint, based on byteStart, metadata, tracks and headerSize. - /// The seekPoint will be set to the timestamp of the first packet to send. - void findSeekPoint(long long byteStart, long long & seekPoint, DTSC::Meta & metadata, std::set<int> & tracks, unsigned int headerSize){ - seekPoint = 0; - //if we're starting in the header, seekPoint is always zero. - if (byteStart <= headerSize){return;} - //okay, we're past the header. Substract the headersize from the starting postion. - byteStart -= headerSize; - //initialize a list of sorted parts that this file contains - std::set <keyPart> sortSet; - for (std::set<int>::iterator subIt = tracks.begin(); subIt != tracks.end(); subIt++) { - keyPart temp; - temp.trackID = *subIt; - temp.time = metadata.tracks[*subIt].firstms;//timeplace of frame - temp.endTime = metadata.tracks[*subIt].firstms + metadata.tracks[*subIt].parts[0].getDuration(); - temp.size = metadata.tracks[*subIt].parts[0].getSize();//bytesize of frame (alle parts all together) - temp.index = 0; - sortSet.insert(temp); - } - //forward through the file by headers, until we reach the point where we need to be - while (!sortSet.empty()){ - //substract the size of this fragment from byteStart - byteStart -= sortSet.begin()->size; - //if that put us past the point where we wanted to be, return right now - if (byteStart < 0){return;} - //otherwise, set seekPoint to where we are now - seekPoint = sortSet.begin()->time; - //then find the next part - keyPart temp; - temp.index = sortSet.begin()->index + 1; - temp.trackID = sortSet.begin()->trackID; - if(temp.index < metadata.tracks[temp.trackID].parts.size() ){//only insert when there are parts left - temp.time = sortSet.begin()->endTime;//timeplace of frame - temp.endTime = sortSet.begin()->endTime + metadata.tracks[temp.trackID].parts[temp.index].getDuration(); - temp.size = metadata.tracks[temp.trackID].parts[temp.index].getSize();//bytesize of frame - sortSet.insert(temp); - } - //remove highest keyPart - sortSet.erase(sortSet.begin()); - } - //If we're here, we're in the last fragment. - //That's technically legal, of course. - } - - /// Parses a "Range: " header, setting byteStart, byteEnd and seekPoint using data from metadata and tracks to do - /// the calculations. - /// On error, byteEnd is set to zero. - void parseRange(std::string header, long long & byteStart, long long & byteEnd, long long & seekPoint, DTSC::Meta & metadata, std::set<int> & tracks, unsigned int headerSize){ - if (header.size() < 6 || header.substr(0, 6) != "bytes="){ - byteEnd = 0; - DEBUG_MSG(DLVL_WARN, "Invalid range header: %s", header.c_str()); - return; - } - header.erase(0, 6); - if (header.size() && header[0] == '-'){ - //negative range = count from end - byteStart = 0; - for (unsigned int i = 1; i < header.size(); ++i){ - if (header[i] >= '0' && header[i] <= '9'){ - byteStart *= 10; - byteStart += header[i] - '0'; - continue; - } - break; - } - if (byteStart > byteEnd){ - //entire file if starting before byte zero - byteStart = 0; - DEBUG_MSG(DLVL_DEVEL, "Full negative range: %lli-%lli", byteStart, byteEnd); - findSeekPoint(byteStart, seekPoint, metadata, tracks, headerSize); - return; - }else{ - //start byteStart bytes before byteEnd - byteStart = byteEnd - byteStart; - DEBUG_MSG(DLVL_DEVEL, "Partial negative range: %lli-%lli", byteStart, byteEnd); - findSeekPoint(byteStart, seekPoint, metadata, tracks, headerSize); - return; - } - }else{ - long long size = byteEnd; - byteEnd = 0; - byteStart = 0; - unsigned int i = 0; - for ( ; i < header.size(); ++i){ - if (header[i] >= '0' && header[i] <= '9'){ - byteStart *= 10; - byteStart += header[i] - '0'; - continue; - } - break; - } - if (header[i] != '-'){ - DEBUG_MSG(DLVL_WARN, "Invalid range header: %s", header.c_str()); - byteEnd = 0; - return; - } - ++i; - if (i < header.size()){ - for ( ; i < header.size(); ++i){ - if (header[i] >= '0' && header[i] <= '9'){ - byteEnd *= 10; - byteEnd += header[i] - '0'; - continue; - } - break; - } - if (byteEnd > size-1){byteEnd = size;} - }else{ - byteEnd = size; - } - DEBUG_MSG(DLVL_DEVEL, "Range request: %lli-%lli (%s)", byteStart, byteEnd, header.c_str()); - findSeekPoint(byteStart, seekPoint, metadata, tracks, headerSize); - return; - } - }//parseRange - - ///\brief Main function for the HTTP Progressive Connector - ///\param conn A socket describing the connection the client. - ///\return The exit code of the connector. - int progressiveConnector(Socket::Connection & conn){ - DTSC::Stream Strm; //Incoming stream buffer. - HTTP::Parser HTTP_R, HTTP_S;//HTTP Receiver en HTTP Sender. - long long byteStart = 0; - long long leftOver = 0; - long long currPos = 0; - bool inited = false;//Whether the stream is initialized - Socket::Connection ss( -1);//The Stream Socket, used to connect to the desired stream. - std::string streamname;//Will contain the name of the stream. - std::set <keyPart> sortSet;//filling sortset for interleaving parts - - unsigned int lastStats = 0;//Indicates the last time that we have sent stats to the server socket. - - while (conn.connected()){ - //Only attempt to parse input when not yet init'ed. - if ( !inited){ - if (conn.spool() && HTTP_R.Read(conn)){ - DEBUG_MSG(DLVL_DEVEL, "Received request: %s", HTTP_R.getUrl().c_str()); - conn.setHost(HTTP_R.GetHeader("X-Origin")); - streamname = HTTP_R.GetHeader("X-Stream"); - if (!ss){ - ss = Util::Stream::getStream(streamname); - if (ss){ - Strm.waitForMeta(ss); - } - if (!ss){ - DEBUG_MSG(DLVL_FAIL, "Could not connect to stream %s!", streamname.c_str()); - ss.close(); - HTTP_S.Clean(); - HTTP_R.Clean(); - HTTP_S.SetBody("No such stream is available on the system. Please try again.\n"); - HTTP_S.SendResponse("404", "Not found", conn); - continue; - } - } - int videoID = -1; - int audioID = -1; - if (HTTP_R.GetVar("audio") != ""){ - audioID = JSON::Value(HTTP_R.GetVar("audio")).asInt(); - } - if (HTTP_R.GetVar("video") != ""){ - videoID = JSON::Value(HTTP_R.GetVar("video")).asInt(); - } - for (std::map<int,DTSC::Track>::iterator it = Strm.metadata.tracks.begin(); it != Strm.metadata.tracks.end(); it++){ - if (videoID == -1 && it->second.type == "video" && it->second.codec == "H264"){ - videoID = it->first; - } - if (audioID == -1 && it->second.type == "audio" && it->second.codec == "AAC"){ - audioID = it->first; - } - } - - std::set<int> tracks; - if (videoID > 0){tracks.insert(videoID);} - if (audioID > 0){tracks.insert(audioID);} - - HTTP_S.Clean(); //make sure no parts of old requests are left in any buffers - HTTP_S.SetHeader("Content-Type", "video/MP4"); //Send the correct content-type for MP4 files - HTTP_S.SetHeader("Accept-Ranges", "bytes, parsec"); - long long size = 0; - std::string headerData = DTSCMeta2MP4Header(Strm.metadata, tracks, size); - byteStart = 0; - long long byteEnd = size-1; - long long seekPoint = 0; - if (HTTP_R.GetHeader("Range") != ""){ - parseRange(HTTP_R.GetHeader("Range"), byteStart, byteEnd, seekPoint, Strm.metadata, tracks, headerData.size()); - if (!byteEnd){ - if (HTTP_R.GetHeader("Range")[0] == 'p'){ - HTTP_S.SetBody("Starsystem not in communications range"); - HTTP_S.SendResponse("416", "Starsystem not in communications range", conn); - HTTP_R.Clean(); //clean for any possible next requests - continue; - }else{ - HTTP_S.SetBody("Requested Range Not Satisfiable"); - HTTP_S.SendResponse("416", "Requested Range Not Satisfiable", conn); - HTTP_R.Clean(); //clean for any possible next requests - continue; - } - }else{ - std::stringstream rangeReply; - rangeReply << "bytes " << byteStart << "-" << byteEnd << "/" << size; - HTTP_S.SetHeader("Content-Length", byteEnd - byteStart + 1); - //do not multiplex requests that are > 1MiB - if (byteEnd - byteStart + 1 > 1024*1024){ - HTTP_S.SetHeader("MistMultiplex", "No"); - } - HTTP_S.SetHeader("Content-Range", rangeReply.str()); - /// \todo Switch to chunked? - HTTP_S.SendResponse("206", "Partial content", conn); - //HTTP_S.StartResponse("206", "Partial content", HTTP_R, conn); - } - }else{ - HTTP_S.SetHeader("Content-Length", byteEnd - byteStart + 1); - //do not multiplex requests that aren't ranged - HTTP_S.SetHeader("MistMultiplex", "No"); - /// \todo Switch to chunked? - HTTP_S.SendResponse("200", "OK", conn); - //HTTP_S.StartResponse(HTTP_R, conn); - } - leftOver = byteEnd - byteStart + 1;//add one byte, because range "0-0" = 1 byte of data - currPos = 0; - if (byteStart < (long long)headerData.size()){ - /// \todo Switch to chunked? - //HTTP_S.Chunkify(headerData.data()+byteStart, std::min((long long)headerData.size(), byteEnd) - byteStart, conn);//send MP4 header - conn.SendNow(headerData.data()+byteStart, std::min((long long)headerData.size(), byteEnd) - byteStart);//send MP4 header - leftOver -= std::min((long long)headerData.size(), byteEnd) - byteStart; - } - currPos = headerData.size();//we're now guaranteed to be past the header point, no matter what - HTTP_R.Clean(); //clean for any possible next requests - {//using scope to have cmd not declared after action - std::stringstream cmd; - cmd << "t"; - for (std::set<int>::iterator it = tracks.begin(); it != tracks.end(); it++) { - cmd << " " << *it; - } - cmd << "\ns " << seekPoint << "\np\n"; - ss.SendNow(cmd.str()); - } - sortSet.clear(); - for (std::set<int>::iterator subIt = tracks.begin(); subIt != tracks.end(); subIt++) { - keyPart temp; - temp.trackID = *subIt; - temp.time = Strm.metadata.tracks[*subIt].firstms;//timeplace of frame - temp.endTime = Strm.metadata.tracks[*subIt].firstms + Strm.metadata.tracks[*subIt].parts[0].getDuration(); - temp.size = Strm.metadata.tracks[*subIt].parts[0].getSize();//bytesize of frame (alle parts all together) - temp.index = 0; - sortSet.insert(temp); - } - inited = true; - } - }else{ - unsigned int now = Util::epoch(); - if (now != lastStats){ - lastStats = now; - ss.SendNow(conn.getStats("HTTP_Progressive_MP4").c_str()); - } - if (ss.spool()){ - while (Strm.parsePacket(ss.Received())){ - if (Strm.lastType() == DTSC::PAUSEMARK){ - conn.close(); - }else if(Strm.lastType() == DTSC::AUDIO || Strm.lastType() == DTSC::VIDEO){ - //keep track of where we are - fast-forward until where we are now - while (!sortSet.empty() && ((long long)sortSet.begin()->trackID != Strm.getPacket()["trackid"].asInt() || (long long)sortSet.begin()->time != Strm.getPacket()["time"].asInt())){ - keyPart temp; - temp.index = sortSet.begin()->index + 1; - temp.trackID = sortSet.begin()->trackID; - if(temp.index < Strm.metadata.tracks[temp.trackID].parts.size() ){//only insert when there are parts left - temp.time = sortSet.begin()->endTime;//timeplace of frame - temp.endTime = sortSet.begin()->endTime + Strm.metadata.tracks[temp.trackID].parts[temp.index].getDuration(); - temp.size = Strm.metadata.tracks[temp.trackID].parts[temp.index].getSize();//bytesize of frame - sortSet.insert(temp); - } - currPos += sortSet.begin()->size; - //remove highest keyPart - sortSet.erase(sortSet.begin()); - } - if (currPos >= byteStart){ - sortSet.clear();//we don't need you anymore! - if (leftOver < (long long)Strm.lastData().size()){ - conn.SendNow(Strm.lastData().data(), leftOver); - }else{ - conn.SendNow(Strm.lastData()); - } - //HTTP_S.Chunkify(Strm.lastData().data(), Strm.lastData().size(), conn); - leftOver -= Strm.lastData().size(); - }else{ - if (currPos + (long long)Strm.lastData().size() > byteStart){ - conn.SendNow(Strm.lastData().data()+(byteStart-currPos), Strm.lastData().size()-(byteStart-currPos)); - leftOver -= Strm.lastData().size()-(byteStart-currPos); - currPos = byteStart; - sortSet.clear();//we don't need you anymore! - } - } - if (leftOver < 1){ - ss.SendNow("q\n");//stop playback - Strm.waitForPause(ss);//sync the stream - inited = false; - } - } - if (Strm.lastType() == DTSC::INVALID){ - DEBUG_MSG(DLVL_FAIL, "Invalid packet received - closing connection"); - conn.close(); - } - } - }else{ - Util::sleep(10); - } - if ( !ss.connected()){ - break; - } - } - } - conn.close(); - ss.SendNow(conn.getStats("HTTP_Progressive_MP4").c_str()); - ss.close(); - return 0; - } //Progressive_Connector main function - -} //Connector_HTTP namespace - -///\brief The standard process-spawning main function. -int main(int argc, char ** argv){ - Util::Config conf(argv[0], PACKAGE_VERSION); - JSON::Value capa; - capa["desc"] = "Enables HTTP protocol progressive streaming."; - capa["deps"] = "HTTP"; - capa["url_rel"] = "/$.mp4"; - capa["url_match"] = "/$.mp4"; - capa["codecs"][0u][0u].append("H264"); - capa["codecs"][0u][1u].append("AAC"); - capa["methods"][0u]["handler"] = "http"; - capa["methods"][0u]["type"] = "html5/video/mp4"; - capa["methods"][0u]["priority"] = 8ll; - capa["methods"][0u]["nolive"] = 1; - capa["socket"] = "http_progressive_mp4"; - conf.addBasicConnectorOptions(capa); - conf.parseArgs(argc, argv); - - if (conf.getBool("json")){ - std::cout << capa.toString() << std::endl; - return -1; - } - - return conf.serveForkedSocket(Connector_HTTP::progressiveConnector); -} //main diff --git a/src/connectors/conn_http_progressive_ogg.cpp b/src/connectors/conn_http_progressive_ogg.cpp deleted file mode 100644 index 6f12d2c8..00000000 --- a/src/connectors/conn_http_progressive_ogg.cpp +++ /dev/null @@ -1,186 +0,0 @@ -///\file conn_http_progressive_ogg.cpp -///\brief Contains the main code for the HTTP Progressive OGG Connector - -#include <iostream> -#include <queue> -#include <sstream> - -#include <cstdlib> -#include <cstdio> -#include <cmath> -#include <unistd.h> -#include <sys/types.h> -#include <sys/wait.h> - -#include <mist/socket.h> -#include <mist/http_parser.h> -#include <mist/dtsc.h> -#include <mist/ogg.h> -#include <mist/amf.h> -#include <mist/config.h> -#include <mist/stream.h> -#include <mist/timing.h> -#include "../converters/oggconv.h" - -///\brief Holds everything unique to HTTP Connectors. -namespace Connector_HTTP { - ///\brief Main function for the HTTP Progressive Connector - ///\param conn A socket describing the connection the client. - ///\return The exit code of the connector. - int progressiveConnector(Socket::Connection & conn){ - bool progressive_has_sent_header = false;//Indicates whether we have sent a header. - bool ready4data = false; //Set to true when streaming is to begin. - DTSC::Stream Strm; //Incoming stream buffer. - HTTP::Parser HTTP_R, HTTP_S;//HTTP Receiver en HTTP Sender. - bool inited = false;//Whether the stream is initialized - Socket::Connection ss( -1);//The Stream Socket, used to connect to the desired stream. - std::string streamname;//Will contain the name of the stream. - - //OGG specific variables - //OGG::headerPages oggMeta; - //OGG::Page curOggPage; - OGG::converter oggConv; - std::map <long long unsigned int, std::vector<JSON::Value> > DTSCBuffer; - //std::map <long long unsigned int, long long unsigned int> prevGran; - std::vector<unsigned int> curSegTable; - std::string sendBuffer; - - unsigned int lastStats = 0;//Indicates the last time that we have sent stats to the server socket. - - int videoID = -1; - int audioID = -1; - - while (conn.connected()){ - //Only attempt to parse input when not yet init'ed. - if ( !inited){ - if (conn.spool() && HTTP_R.Read(conn)){ -#if DEBUG >= 5 - std::cout << "Received request: " << HTTP_R.getUrl() << std::endl; -#endif - conn.setHost(HTTP_R.GetHeader("X-Origin")); - streamname = HTTP_R.GetHeader("X-Stream"); - ready4data = true; - HTTP_R.Clean(); //clean for any possible next requests - } - } - if (ready4data){ - if ( !inited){ - //we are ready, connect the socket! - ss = Util::Stream::getStream(streamname); - if ( !ss.connected()){ -#if DEBUG >= 1 - fprintf(stderr, "Could not connect to server for %s!\n", streamname.c_str()); -#endif - ss.close(); - HTTP_S.Clean(); - HTTP_S.SetBody("No such stream is available on the system. Please try again.\n"); - conn.SendNow(HTTP_S.BuildResponse("404", "Not found")); - ready4data = false; - continue; - } - Strm.waitForMeta(ss); - for (std::map<int,DTSC::Track>::iterator it = Strm.metadata.tracks.begin(); it != Strm.metadata.tracks.end(); it++){ - if (videoID == -1 && it->second.codec == "theora"){ - videoID = it->second.trackID; - } - if (audioID == -1 && it->second.codec == "vorbis"){ - audioID = it->second.trackID; - } - } - if (videoID == -1 && audioID == -1){ - HTTP_S.Clean(); //make sure no parts of old requests are left in any buffers - HTTP_S.SetBody("This stream contains no OGG compatible codecs"); - HTTP_S.SendResponse("406", "Not acceptable",conn); - HTTP_R.Clean(); - continue; - } - std::stringstream cmd; - cmd << "t"; - if (videoID != -1){ - cmd << " " << videoID; - } - if (audioID != -1){ - cmd << " " << audioID; - } - cmd << "\np\n"; - ss.SendNow(cmd.str().c_str(), cmd.str().size()); - inited = true; - } - unsigned int now = Util::epoch(); - if (now != lastStats){ - lastStats = now; - ss.SendNow(conn.getStats("HTTP_Progressive_Ogg").c_str()); - } - if (ss.spool()){ - while (Strm.parsePacket(ss.Received())){ - - if ( !progressive_has_sent_header){ - HTTP_S.Clean(); //make sure no parts of old requests are left in any buffers - HTTP_S.SetHeader("Content-Type", "video/ogg"); //Send the correct content-type for FLV files - HTTP_S.protocol = "HTTP/1.0"; - conn.SendNow(HTTP_S.BuildResponse("200", "OK")); //no SetBody = unknown length - this is intentional, we will stream the entire file - //Fill in ogg header here - oggConv.readDTSCHeader(Strm.metadata); - conn.SendNow((char*)oggConv.parsedPages.c_str(), oggConv.parsedPages.size()); - progressive_has_sent_header = true; - } - //parse DTSC to Ogg here - if (Strm.lastType() == DTSC::AUDIO || Strm.lastType() == DTSC::VIDEO){ - std::string tmpString; - oggConv.readDTSCVector(Strm.getPacket(), tmpString); - conn.SendNow(tmpString); - - } - if (Strm.lastType() == DTSC::PAUSEMARK){ - conn.close(); - ss.close(); - //last page output - } - if (Strm.lastType() == DTSC::INVALID){ - #if DEBUG >= 3 - fprintf(stderr, "Invalid packet received - closing connection.\n"); - #endif - conn.close(); - } - } - }else{ - Util::sleep(100); - } - if ( !ss.connected()){ - break; - } - } - } - conn.close(); - ss.SendNow(conn.getStats("HTTP_Dynamic").c_str()); - ss.close(); - return 0; - } //Progressive_Connector main function - -} //Connector_HTTP namespace - -///\brief The standard process-spawning main function. -int main(int argc, char ** argv){ - Util::Config conf(argv[0], PACKAGE_VERSION); - JSON::Value capa; - capa["desc"] = "Enables HTTP protocol progressive streaming."; - capa["deps"] = "HTTP"; - capa["url_rel"] = "/$.ogg"; - capa["url_match"] = "/$.ogg"; - capa["socket"] = "http_progressive_ogg"; - capa["codecs"][0u][0u].append("theora"); - capa["codecs"][0u][1u].append("vorbis"); - capa["methods"][0u]["handler"] = "http"; - capa["methods"][0u]["type"] = "html5/video/ogg"; - capa["methods"][0u]["priority"] = 8ll; - capa["methods"][0u]["nolive"] = 1; - conf.addBasicConnectorOptions(capa); - conf.parseArgs(argc, argv); - - if (conf.getBool("json")){ - std::cout << capa.toString() << std::endl; - return -1; - } - - return conf.serveForkedSocket(Connector_HTTP::progressiveConnector); -} //main diff --git a/src/connectors/conn_http_smooth.cpp b/src/connectors/conn_http_smooth.cpp deleted file mode 100644 index c269c7da..00000000 --- a/src/connectors/conn_http_smooth.cpp +++ /dev/null @@ -1,516 +0,0 @@ -///\file conn_http_smooth.cpp -///\brief Contains the main code for the HTTP Smooth Connector - -#include <iostream> -#include <iomanip> -#include <queue> -#include <sstream> - -#include <cstdlib> -#include <cstdio> -#include <cmath> -#include <unistd.h> -#include <sys/types.h> -#include <sys/wait.h> -#include <getopt.h> - -#include <mist/socket.h> -#include <mist/http_parser.h> -#include <mist/json.h> -#include <mist/dtsc.h> -#include <mist/base64.h> -#include <mist/amf.h> -#include <mist/mp4.h> -#include <mist/mp4_ms.h> -#include <mist/mp4_generic.h> -#include <mist/config.h> -#include <mist/stream.h> -#include <mist/timing.h> - -long long unsigned int binToInt(std::string & binary){ - long long int result = 0; - for ( int i = 0; i < 8; i++){ - result <<= 8; - result += binary[i]; - } - return result; -} - -std::string intToBin(long long unsigned int number){ - std::string result; - result.resize(8); - for( int i = 7; i >= 0; i--){ - result[i] = number & 0xFF; - number >>= 8; - } - return result; -} - -std::string toUTF16(std::string original){ - std::string result; - result += (char)0xFF; - result += (char)0xFE; - for (std::string::iterator it = original.begin(); it != original.end(); it++){ - result += (*it); - result += (char)0x00; - } - return result; -} - -///\brief Holds everything unique to HTTP Connectors. -namespace Connector_HTTP { - ///\brief Builds an index file for HTTP Smooth streaming. - ///\param metadata The current metadata, used to generate the index. - ///\return The index file for HTTP Smooth Streaming. - std::string smoothIndex(DTSC::Meta & metadata){ - std::stringstream Result; - Result << "<?xml version=\"1.0\" encoding=\"utf-16\"?>\n"; - Result << "<SmoothStreamingMedia " - "MajorVersion=\"2\" " - "MinorVersion=\"0\" " - "TimeScale=\"10000000\" "; - std::deque<std::map<int,DTSC::Track>::iterator> audioIters; - std::deque<std::map<int,DTSC::Track>::iterator> videoIters; - long long int maxWidth = 0; - long long int maxHeight = 0; - long long int minWidth = 99999999; - long long int minHeight = 99999999; - for (std::map<int,DTSC::Track>::iterator it = metadata.tracks.begin(); it != metadata.tracks.end(); it++){ - if (it->second.codec == "AAC"){ - audioIters.push_back(it); - } - if (it->second.type == "video" && it->second.codec == "H264"){ - videoIters.push_back(it); - if (it->second.width > maxWidth){maxWidth = it->second.width;} - if (it->second.width < minWidth){minWidth = it->second.width;} - if (it->second.height > maxHeight){maxHeight = it->second.height;} - if (it->second.height < minHeight){minHeight = it->second.height;} - } - } - if (metadata.vod){ - Result << "Duration=\"" << (*videoIters.begin())->second.lastms << "0000\""; - }else{ - Result << "Duration=\"0\" " - "IsLive=\"TRUE\" " - "LookAheadFragmentCount=\"2\" " - "DVRWindowLength=\"" << metadata.bufferWindow << "0000\" " - "CanSeek=\"TRUE\" " - "CanPause=\"TRUE\" "; - } - Result << ">\n"; - - //Add audio entries - if (audioIters.size()){ - Result << "<StreamIndex " - "Type=\"audio\" " - "QualityLevels=\"" << audioIters.size() << "\" " - "Name=\"audio\" " - "Chunks=\"" << (*audioIters.begin())->second.keys.size() << "\" " - "Url=\"Q({bitrate},{CustomAttributes})/A({start time})\">\n"; - int index = 0; - for (std::deque<std::map<int,DTSC::Track>::iterator>::iterator it = audioIters.begin(); it != audioIters.end(); it++){ - Result << "<QualityLevel " - "Index=\"" << index << "\" " - "Bitrate=\"" << (*it)->second.bps * 8 << "\" " - "CodecPrivateData=\"" << std::hex; - for (unsigned int i = 0; i < (*it)->second.init.size(); i++){ - Result << std::setfill('0') << std::setw(2) << std::right << (int)(*it)->second.init[i]; - } - Result << std::dec << "\" " - "SamplingRate=\"" << (*it)->second.rate << "\" " - "Channels=\"2\" " - "BitsPerSample=\"16\" " - "PacketSize=\"4\" " - "AudioTag=\"255\" " - "FourCC=\"AACL\" >\n"; - Result << "<CustomAttributes>\n" - "<Attribute Name = \"TrackID\" Value = \"" << (*it)->first << "\" />" - "</CustomAttributes>"; - Result << "</QualityLevel>\n"; - index++; - } - if ((*audioIters.begin())->second.keys.size()){ - for (std::deque<DTSC::Key>::iterator it = (*audioIters.begin())->second.keys.begin(); it != (((*audioIters.begin())->second.keys.end()) - 1); it++){ - Result << "<c "; - if (it == (*audioIters.begin())->second.keys.begin()){ - Result << "t=\"" << it->getTime() * 10000 << "\" "; - } - Result << "d=\"" << it->getLength() * 10000 << "\" />\n"; - } - } - Result << "</StreamIndex>\n"; - } - //Add video entries - if (videoIters.size()){ - Result << "<StreamIndex " - "Type=\"video\" " - "QualityLevels=\"" << videoIters.size() << "\" " - "Name=\"video\" " - "Chunks=\"" << (*videoIters.begin())->second.keys.size() << "\" " - "Url=\"Q({bitrate},{CustomAttributes})/V({start time})\" " - "MaxWidth=\"" << maxWidth << "\" " - "MaxHeight=\"" << maxHeight << "\" " - "DisplayWidth=\"" << maxWidth << "\" " - "DisplayHeight=\"" << maxHeight << "\">\n"; - int index = 0; - for (std::deque<std::map<int,DTSC::Track>::iterator>::iterator it = videoIters.begin(); it != videoIters.end(); it++){ - //Add video qualities - Result << "<QualityLevel " - "Index=\"" << index << "\" " - "Bitrate=\"" << (*it)->second.bps * 8 << "\" " - "CodecPrivateData=\"" << std::hex; - MP4::AVCC avccbox; - avccbox.setPayload((*it)->second.init); - std::string tmpString = avccbox.asAnnexB(); - for (unsigned int i = 0; i < tmpString.size(); i++){ - Result << std::setfill('0') << std::setw(2) << std::right << (int)tmpString[i]; - } - Result << std::dec << "\" " - "MaxWidth=\"" << (*it)->second.width << "\" " - "MaxHeight=\"" << (*it)->second.height << "\" " - "FourCC=\"AVC1\" >\n"; - Result << "<CustomAttributes>\n" - "<Attribute Name = \"TrackID\" Value = \"" << (*it)->first << "\" />" - "</CustomAttributes>"; - Result << "</QualityLevel>\n"; - index++; - } - if ((*videoIters.begin())->second.keys.size()){ - for (std::deque<DTSC::Key>::iterator it = (*videoIters.begin())->second.keys.begin(); it != (((*videoIters.begin())->second.keys.end()) - 1); it++){ - Result << "<c "; - if (it == (*videoIters.begin())->second.keys.begin()){ - Result << "t=\"" << it->getTime() * 10000 << "\" "; - } - Result << "d=\"" << it->getLength() * 10000 << "\" />\n"; - } - } - Result << "</StreamIndex>\n"; - } - Result << "</SmoothStreamingMedia>\n"; - -#if DEBUG >= 8 - std::cerr << "Sending this manifest:" << std::endl << Result << std::endl; -#endif - return toUTF16(Result.str()); - } //smoothIndex - - ///\brief Main function for the HTTP Smooth Connector - ///\param conn A socket describing the connection the client. - ///\return The exit code of the connector. - int smoothConnector(Socket::Connection & conn){ - std::deque<std::string> dataBuffer;//A buffer for the data that needs to be sent to the client. - - DTSC::Stream Strm;//Incoming stream buffer. - HTTP::Parser HTTP_R;//HTTP Receiver - HTTP::Parser HTTP_S;//HTTP Sender. - - bool ready4data = false;//Set to true when streaming is to begin. - Socket::Connection ss( -1);//The Stream Socket, used to connect to the desired stream. - std::string streamname;//Will contain the name of the stream. - bool handlingRequest = false; - - std::string Quality;//Indicates the request quality of the movie. - long long int requestedTime = -1;//Indicates the fragment requested. - std::string parseString;//A string used for parsing different aspects of the request. - unsigned int lastStats = 0;//Indicates the last time that we have sent stats to the server socket. - conn.setBlocking(false);//Set the client socket to non-blocking - - while (conn.connected()){ - if ( !handlingRequest){ - if (conn.spool() && HTTP_R.Read(conn)){ - #if DEBUG >= 5 - std::cout << "Received request: " << HTTP_R.getUrl() << std::endl; - #endif - //Get data set by the proxy. - conn.setHost(HTTP_R.GetHeader("X-Origin")); - streamname = HTTP_R.GetHeader("X-Stream"); - if ( !ss){ - //initiate Stream Socket - ss = Util::Stream::getStream(streamname); - if ( !ss.connected()){ - #if DEBUG >= 1 - fprintf(stderr, "Could not connect to server!\n"); - #endif - HTTP_S.Clean(); - HTTP_S.SetBody("No such stream is available on the system. Please try again.\n"); - conn.SendNow(HTTP_S.BuildResponse("404", "Not found")); - ready4data = false; - continue; - } - ss.setBlocking(false); - Strm.waitForMeta(ss); - } - - - if (HTTP_R.url.find(".xap") != std::string::npos){ -#include "xap.h" - - HTTP_S.Clean(); - HTTP_S.SetHeader("Content-Type", "application/siverlight"); - HTTP_S.SetHeader("Cache-Control", "cache"); - HTTP_S.SetBody(""); - HTTP_S.SetHeader("Content-Length", xap_len); - HTTP_S.SendResponse("200", "OK", conn); - conn.SendNow((const char *)xap_data, xap_len); - }else{ - if (HTTP_R.url.find("Manifest") == std::string::npos){ - //We have a non-manifest request, parse it. - - Quality = HTTP_R.url.substr(HTTP_R.url.find("TrackID=", 8) + 8); - Quality = Quality.substr(0, Quality.find(")")); - parseString = HTTP_R.url.substr(HTTP_R.url.find(")/") + 2); - parseString = parseString.substr(parseString.find("(") + 1); - requestedTime = atoll(parseString.substr(0, parseString.find(")")).c_str()); - long long int selectedQuality = atoll(Quality.c_str()); - DTSC::Track & myRef = Strm.metadata.tracks[selectedQuality]; - if (Strm.metadata.live){ - int seekable = Strm.canSeekms(requestedTime / 10000); - if (seekable == 0){ - // iff the fragment in question is available, check if the next is available too - for (std::deque<DTSC::Key>::iterator it = myRef.keys.begin(); it != myRef.keys.end(); it++){ - if (it->getTime() >= (requestedTime / 10000)){ - if ((it + 1) == myRef.keys.end()){ - seekable = 1; - } - break; - } - } - } - if (seekable < 0){ - HTTP_S.Clean(); - HTTP_S.SetBody("The requested fragment is no longer kept in memory on the server and cannot be served.\n"); - conn.SendNow(HTTP_S.BuildResponse("412", "Fragment out of range")); - HTTP_R.Clean(); //clean for any possible next requests - std::cout << "Fragment @ " << requestedTime / 10000 << "ms too old (" << myRef.keys.begin()->getTime() << " - " << myRef.keys.rbegin()->getTime() << " ms)" << std::endl; - continue; - } - if (seekable > 0){ - HTTP_S.Clean(); - HTTP_S.SetBody("Proxy, re-request this in a second or two.\n"); - conn.SendNow(HTTP_S.BuildResponse("208", "Ask again later")); - HTTP_R.Clean(); //clean for any possible next requests - std::cout << "Fragment @ " << requestedTime / 10000 << "ms not available yet (" << myRef.keys.begin()->getTime() << " - " << myRef.keys.rbegin()->getTime() << " ms)" << std::endl; - continue; - } - } - //Seek to the right place and send a play-once for a single fragment. - std::stringstream sstream; - - long long mstime = 0; - int partOffset = 0; - int keyDur = 0; - DTSC::Key keyObj; - for (std::deque<DTSC::Key>::iterator it = myRef.keys.begin(); it != myRef.keys.end(); it++){ - if (it->getTime() >= (requestedTime / 10000)){ - mstime = it->getTime(); - keyObj = (*it); - std::deque<DTSC::Key>::iterator nextIt = it; - nextIt++; - if (nextIt != myRef.keys.end()){ - keyDur = nextIt->getTime() - it->getTime(); - }else{ - keyDur = -1; - if (Strm.metadata.live){ - HTTP_S.Clean(); - HTTP_S.SetBody("Proxy, re-request this in a second or two.\n"); - conn.SendNow(HTTP_S.BuildResponse("208", "Ask again later")); - HTTP_R.Clean(); //clean for any possible next requests - std::cout << "Fragment after fragment @ " << (requestedTime / 10000) << " not available yet" << std::endl; - } - } - break; - } - partOffset += it->getParts(); - } - if (HTTP_R.url == "/"){continue;}//Don't continue, but continue instead. - if (Strm.metadata.live){ - if (mstime == 0 && (requestedTime / 10000) > 1){ - HTTP_S.Clean(); - HTTP_S.SetBody("The requested fragment is no longer kept in memory on the server and cannot be served.\n"); - conn.SendNow(HTTP_S.BuildResponse("412", "Fragment out of range")); - HTTP_R.Clean(); //clean for any possible next requests - std::cout << "Fragment @ " << (requestedTime / 10000) << " too old" << std::endl; - continue; - } - } - - sstream << "t " << myRef.trackID << "\n"; - sstream << "s " << keyObj.getTime() << "\n"; - if (keyDur != -1){ - sstream << "p " << keyObj.getTime() + keyDur << "\n"; - }else{ - sstream << "p\n"; - } - - ss.SendNow(sstream.str().c_str()); - - //Wrap everything in mp4 boxes - MP4::MFHD mfhd_box; - mfhd_box.setSequenceNumber(((keyObj.getNumber() - 1) * 2) + myRef.trackID); - - MP4::TFHD tfhd_box; - tfhd_box.setFlags(MP4::tfhdSampleFlag); - tfhd_box.setTrackID(myRef.trackID); - if (myRef.type == "video"){ - tfhd_box.setDefaultSampleFlags(0x00004001); - }else{ - tfhd_box.setDefaultSampleFlags(0x00008002); - } - - MP4::TRUN trun_box; - trun_box.setDataOffset(42); - unsigned int keySize = 0; - if (myRef.type == "video"){ - trun_box.setFlags(MP4::trundataOffset | MP4::trunfirstSampleFlags | MP4::trunsampleDuration | MP4::trunsampleSize | MP4::trunsampleOffsets); - }else{ - trun_box.setFlags(MP4::trundataOffset | MP4::trunsampleDuration | MP4::trunsampleSize); - } - trun_box.setFirstSampleFlags(0x00004002); - for (int i = 0; i < keyObj.getParts(); i++){ - MP4::trunSampleInformation trunSample; - trunSample.sampleSize = Strm.metadata.tracks[myRef.trackID].parts[i + partOffset].getSize(); - keySize += Strm.metadata.tracks[myRef.trackID].parts[i + partOffset].getSize(); - trunSample.sampleDuration = Strm.metadata.tracks[myRef.trackID].parts[i + partOffset].getDuration() * 10000; - if (myRef.type == "video"){ - trunSample.sampleOffset = Strm.metadata.tracks[myRef.trackID].parts[i + partOffset].getOffset() * 10000; - } - trun_box.setSampleInformation(trunSample, i); - } - - MP4::SDTP sdtp_box; - sdtp_box.setVersion(0); - if (myRef.type == "video"){ - sdtp_box.setValue(36, 4); - for (int i = 1; i < keyObj.getParts(); i++){ - sdtp_box.setValue(20, 4 + i); - } - }else{ - sdtp_box.setValue(40, 4); - for (int i = 1; i < keyObj.getParts(); i++){ - sdtp_box.setValue(40, 4 + i); - } - } - - MP4::TRAF traf_box; - traf_box.setContent(tfhd_box, 0); - traf_box.setContent(trun_box, 1); - traf_box.setContent(sdtp_box, 2); - - //If the stream is live, we want to have a fragref box if possible - if (Strm.metadata.live){ - MP4::UUID_TrackFragmentReference fragref_box; - fragref_box.setVersion(1); - fragref_box.setFragmentCount(0); - int fragCount = 0; - for (unsigned int i = 0; fragCount < 2 && i < myRef.keys.size() - 1; i++){ - if (myRef.keys[i].getTime() > (requestedTime / 10000)){ - fragref_box.setTime(fragCount, myRef.keys[i].getTime() * 10000); - fragref_box.setDuration(fragCount, myRef.keys[i].getLength() * 10000); - fragref_box.setFragmentCount(++fragCount); - } - } - traf_box.setContent(fragref_box, 3); - } - - MP4::MOOF moof_box; - moof_box.setContent(mfhd_box, 0); - //Setting the correct offsets. - moof_box.setContent(traf_box, 1); - trun_box.setDataOffset(moof_box.boxedSize() + 8); - traf_box.setContent(trun_box, 1); - moof_box.setContent(traf_box, 1); - - HTTP_S.Clean(); - HTTP_S.SetHeader("Content-Type", "video/mp4"); - HTTP_S.StartResponse(HTTP_R, conn); - HTTP_S.Chunkify(moof_box.asBox(), moof_box.boxedSize(), conn); - int size = htonl(keySize + 8); - HTTP_S.Chunkify((char*)&size, 4, conn); - HTTP_S.Chunkify("mdat", 4, conn); - handlingRequest = true; - }else{ - //We have a request for a Manifest, generate and send it. - - HTTP_S.Clean(); - HTTP_S.SetHeader("Content-Type", "text/xml"); - HTTP_S.SetHeader("Cache-Control", "no-cache"); - std::string manifest = smoothIndex(Strm.metadata); - HTTP_S.SetBody(manifest); - HTTP_S.SendResponse("200", "OK", conn); - } - } - ready4data = true; - //Clean for any possible next requests - HTTP_R.Clean(); - }else{ - //Wait 250ms before checking for new data. - Util::sleep(250); - } - }else{ - if (!ready4data){ - //Wait 250ms before checking for new data. - Util::sleep(250); - } - } - if (ready4data){ - unsigned int now = Util::epoch(); - if (now != lastStats){ - //Send new stats. - lastStats = now; - ss.SendNow(conn.getStats("HTTP_Smooth")); - } - if (ss.spool()){ - while (Strm.parsePacket(ss.Received())){ - if (Strm.lastType() == DTSC::AUDIO || Strm.lastType() == DTSC::VIDEO){ - HTTP_S.Chunkify(Strm.lastData(), conn); - } - if (Strm.lastType() == DTSC::PAUSEMARK){ - HTTP_S.Chunkify("", 0, conn); - handlingRequest = false; - } - } - }else{ - Util::sleep(10); - } - if ( !ss.connected()){ - break; - } - } - } - conn.close(); - ss.SendNow(conn.getStats("HTTP_Smooth").c_str()); - ss.close(); - return 0; - }//Smooth_Connector main function - -}//Connector_HTTP namespace - -///\brief The standard process-spawning main function. -int main(int argc, char ** argv){ - Util::Config conf(argv[0], PACKAGE_VERSION); - JSON::Value capa; - capa["desc"] = "Enables HTTP protocol Microsoft-specific smooth streaming through silverlight (also known as HSS)."; - capa["deps"] = "HTTP"; - capa["url_rel"] = "/smooth/$.ism/Manifest"; - capa["url_prefix"] = "/smooth/$.ism/"; - capa["socket"] = "http_smooth"; - capa["codecs"][0u][0u].append("H264"); - capa["codecs"][0u][1u].append("AAC"); - capa["methods"][0u]["handler"] = "http"; - capa["methods"][0u]["type"] = "html5/application/vnd.ms-ss"; - capa["methods"][0u]["priority"] = 9ll; - capa["methods"][0u]["nolive"] = 1; - capa["methods"][1u]["handler"] = "http"; - capa["methods"][1u]["type"] = "silverlight"; - capa["methods"][1u]["priority"] = 1ll; - capa["methods"][1u]["nolive"] = 1; - conf.addBasicConnectorOptions(capa); - conf.parseArgs(argc, argv); - - if (conf.getBool("json")){ - std::cout << capa.toString() << std::endl; - return -1; - } - - return conf.serveForkedSocket(Connector_HTTP::smoothConnector); -} //main diff --git a/src/connectors/conn_http_srt.cpp b/src/connectors/conn_http_srt.cpp deleted file mode 100644 index cbb8513e..00000000 --- a/src/connectors/conn_http_srt.cpp +++ /dev/null @@ -1,223 +0,0 @@ -///\file conn_http_srt.cpp -///\brief Contains the main code for the HTTP SRT Connector - -#include <iostream> -#include <queue> -#include <sstream> -#include <iomanip> - -#include <cstdlib> -#include <cstdio> -#include <cmath> -#include <unistd.h> -#include <sys/types.h> -#include <sys/wait.h> -#include <getopt.h> - -#include <mist/socket.h> -#include <mist/http_parser.h> -#include <mist/dtsc.h> -#include <mist/flv_tag.h> -#include <mist/amf.h> -#include <mist/config.h> -#include <mist/stream.h> -#include <mist/timing.h> - -///\brief Holds everything unique to HTTP Connectors. -namespace Connector_HTTP { - ///\brief Main function for the HTTP Progressive Connector - ///\param conn A socket describing the connection the client. - ///\return The exit code of the connector. - int SRTConnector(Socket::Connection & conn){ - DTSC::Stream Strm; //Incoming stream buffer. - HTTP::Parser HTTP_R, HTTP_S;//HTTP Receiver en HTTP Sender. - bool inited = false;//Whether the stream is initialized - Socket::Connection ss( -1);//The Stream Socket, used to connect to the desired stream. - std::string streamname;//Will contain the name of the stream. - - unsigned int lastStats = 0;//Indicates the last time that we have sent stats to the server socket. - unsigned int seek_time = 0;//Seek position in ms - int trackID = -1; // the track to be selected - int curIndex = 0; // SRT index - bool subtitleTrack = false; // check whether the requested track is a srt track - bool isWebVTT = false; - - std::stringstream srtdata; // ss output data - - while (conn.connected()){ - //Only attempt to parse input when not yet init'ed. - if ( !inited){ - if (conn.spool() && HTTP_R.Read(conn)){ -#if DEBUG >= 5 - std::cout << "Received request: " << HTTP_R.getUrl() << std::endl; -#endif - conn.setHost(HTTP_R.GetHeader("X-Origin")); - streamname = HTTP_R.GetHeader("X-Stream"); - - int start = 0; - if ( !HTTP_R.GetVar("start").empty()){ - start = atoi(HTTP_R.GetVar("start").c_str()); - } - if ( !HTTP_R.GetVar("starttime").empty()){ - start = atoi(HTTP_R.GetVar("starttime").c_str()); - } - if ( !HTTP_R.GetVar("apstart").empty()){ - start = atoi(HTTP_R.GetVar("apstart").c_str()); - } - if ( !HTTP_R.GetVar("ec_seek").empty()){ - start = atoi(HTTP_R.GetVar("ec_seek").c_str()); - } - if ( !HTTP_R.GetVar("fs").empty()){ - start = atoi(HTTP_R.GetVar("fs").c_str()); - } - if ( !HTTP_R.GetVar("trackid").empty()){ - trackID = atoi(HTTP_R.GetVar("trackid").c_str()); - } - if ( !HTTP_R.GetVar("webvtt").empty()){ - isWebVTT = true; - }else{ - isWebVTT = false; - } - //under 3 hours we assume seconds, otherwise byte position - if (start < 10800){ - seek_time = start * 1000; //ms, not s - }else{ - seek_time = start * 1000; //divide by 1mbit, then *1000 for ms. - } - - //we are ready, connect the socket! - if ( !ss.connected()){ - ss = Util::Stream::getStream(streamname); - } - if ( !ss.connected()){ - #if DEBUG >= 1 - fprintf(stderr, "Could not connect to server for %s!\n", streamname.c_str()); - #endif - ss.close(); - HTTP_S.Clean(); - HTTP_S.SetBody("No such stream is available on the system. Please try again.\n"); - conn.SendNow(HTTP_S.BuildResponse("404", "Not found")); - inited = false; - continue; - } - - Strm.waitForMeta(ss); - - if(trackID == -1){ - // no track was given. Fetch the first track that has SRT data - for (std::map<int,DTSC::Track>::iterator it = Strm.metadata.tracks.begin(); it != Strm.metadata.tracks.end(); it++){ - if (it->second.codec == "srt"){ - trackID = it->second.trackID; - subtitleTrack = true; - break; - } - } - }else{ - // track *was* given, but we have to check whether it's an actual srt track - subtitleTrack = Strm.metadata.tracks[trackID].codec == "srt"; - } - - if(!subtitleTrack){ - HTTP_S.Clean(); - HTTP_S.SetBody("# This track doesn't contain subtitle data.\n"); - conn.SendNow(HTTP_S.BuildResponse("404", "Not found")); - subtitleTrack = false; - HTTP_R.Clean(); - continue; - } - - std::stringstream cmd; - - cmd << "t " << trackID; - - int maxTime = Strm.metadata.tracks[trackID].lastms; - - cmd << "\ns " << seek_time << "\np " << maxTime << "\n"; - ss.SendNow(cmd.str().c_str(), cmd.str().size()); - - inited = true; - - HTTP_R.Clean(); //clean for any possible next requests - srtdata.clear(); - curIndex = 1; // set to 1, first srt 'track' - } - } - - unsigned int now = Util::epoch(); - if (now != lastStats){ - lastStats = now; - ss.SendNow(conn.getStats("HTTP_SRT").c_str()); - } - - if (inited){ - - if (ss.spool()){ - while (Strm.parsePacket(ss.Received())){ - - if(Strm.lastType() == DTSC::META){ - - if(!isWebVTT) - { - srtdata << curIndex++ << std::endl; - } - long long unsigned int time = Strm.getPacket()["time"].asInt(); - srtdata << std::setfill('0') << std::setw(2) << (time / 3600000) << ":"; - srtdata << std::setfill('0') << std::setw(2) << ((time % 3600000) / 60000) << ":"; - srtdata << std::setfill('0') << std::setw(2) << (((time % 3600000) % 60000) / 1000) << ","; - srtdata << std::setfill('0') << std::setw(3) << time % 1000 << " --> "; - time += Strm.getPacket()["duration"].asInt(); - srtdata << std::setfill('0') << std::setw(2) << (time / 3600000) << ":"; - srtdata << std::setfill('0') << std::setw(2) << ((time % 3600000) / 60000) << ":"; - srtdata << std::setfill('0') << std::setw(2) << (((time % 3600000) % 60000) / 1000) << ","; - srtdata << std::setfill('0') << std::setw(3) << time % 1000 << std::endl; - srtdata << Strm.lastData() << std::endl; - } - - if( Strm.lastType() == DTSC::PAUSEMARK){ - HTTP_S.Clean(); //make sure no parts of old requests are left in any buffers - HTTP_S.SetHeader("Content-Type", "text/plain"); //Send the correct content-type for FLV files - HTTP_S.SetBody( (isWebVTT ? "WEBVTT\n\n" : "") + srtdata.str()); - conn.SendNow(HTTP_S.BuildResponse("200", "OK")); //no SetBody = unknown length - this is intentional, we will stream the entire file - inited = false; - - srtdata.str(""); - srtdata.clear(); - } - } - }else{ - Util::sleep(200); - } - if ( !ss.connected()){ - break; - } - } - } - conn.close(); - ss.SendNow(conn.getStats("HTTP_SRT").c_str()); - ss.close(); - return 0; - } //SRT main function - -} //Connector_HTTP namespace - -///\brief The standard process-spawning main function. -int main(int argc, char ** argv){ - Util::Config conf(argv[0], PACKAGE_VERSION); - JSON::Value capa; - capa["desc"] = "Enables HTTP protocol subtitle streaming."; - capa["deps"] = "HTTP"; - capa["url_rel"] = "/$.srt"; - capa["url_match"] = "/$.srt"; - capa["url_handler"] = "http"; - capa["url_type"] = "subtitle"; - capa["socket"] = "http_srt"; - conf.addBasicConnectorOptions(capa); - conf.parseArgs(argc, argv); - - if (conf.getBool("json")){ - std::cout << capa.toString() << std::endl; - return -1; - } - - return conf.serveForkedSocket(Connector_HTTP::SRTConnector); -} //main diff --git a/src/connectors/conn_raw.cpp b/src/connectors/conn_raw.cpp deleted file mode 100644 index 646e9f51..00000000 --- a/src/connectors/conn_raw.cpp +++ /dev/null @@ -1,58 +0,0 @@ -/// \file conn_raw.cpp -/// Contains the main code for the RAW connector. - -#include <iostream> -#include <sstream> -#include <mist/config.h> -#include <mist/socket.h> -#include <mist/stream.h> -#include <mist/timing.h> - -///\brief Contains the main code for the RAW connector. -/// -///Expects a single commandline argument telling it which stream to connect to, -///then outputs the raw stream to stdout. -int main(int argc, char ** argv){ - Util::Config conf(argv[0], PACKAGE_VERSION); - JSON::Value capa; - conf.addBasicConnectorOptions(capa); - conf.addOption("stream_name", JSON::fromString("{\"arg_num\":1, \"help\":\"Name of the stream to write to stdout.\"}")); - conf.parseArgs(argc, argv); - - if (conf.getBool("json")){ - std::cout << "null" << std::endl; - return -1; - } - - //connect to the proper stream - Socket::Connection S = Util::Stream::getStream(conf.getString("stream_name")); - S.setBlocking(false); - if ( !S.connected()){ - std::cout << "Could not open stream " << conf.getString("stream_name") << std::endl; - return 1; - } - long long int lastStats = 0; - long long int started = Util::epoch(); - while (std::cout.good() && S.connected()){ - if (S.spool()){ - while (S.Received().size()){ - std::cout.write(S.Received().get().c_str(), S.Received().get().size()); - S.Received().get().clear(); - } - }else{ - Util::sleep(500); //sleep 500ms if no data - } - unsigned int now = Util::epoch(); - if (now != lastStats){ - lastStats = now; - std::stringstream st; - st << "S localhost RAW " << (Util::epoch() - started) << " " << S.dataDown() << " " << S.dataUp() << "\n"; - S.SendNow(st.str().c_str()); - } - } - std::stringstream st; - st << "S localhost RAW " << (Util::epoch() - started) << " " << S.dataDown() << " " << S.dataUp() << "\n"; - S.SendNow(st.str().c_str()); - S.close(); - return 0; -} diff --git a/src/connectors/conn_rtmp.cpp b/src/connectors/conn_rtmp.cpp deleted file mode 100644 index ce4f5ba4..00000000 --- a/src/connectors/conn_rtmp.cpp +++ /dev/null @@ -1,700 +0,0 @@ -/// \file conn_rtmp.cpp -/// Contains the main code for the RTMP Connector - -#include <iostream> -#include <sstream> - -#include <cstdlib> -#include <cstdio> -#include <cmath> -#include <unistd.h> -#include <signal.h> -#include <sys/types.h> -#include <sys/wait.h> -#include <getopt.h> - -#include <mist/socket.h> -#include <mist/config.h> -#include <mist/flv_tag.h> -#include <mist/amf.h> -#include <mist/rtmpchunks.h> -#include <mist/stream.h> -#include <mist/timing.h> - -///\brief Holds everything unique to the RTMP Connector -namespace Connector_RTMP { - - //for connection to server - bool ready4data = false; ///< Indicates whether streaming can start. - bool inited = false; ///< Indicates whether we are ready to connect to the Buffer. - bool noStats = false; ///< Indicates when no stats should be sent anymore. Used in push mode. - bool stopParsing = false; ///< Indicates when to stop all parsing. - bool streamReset = false; - - //for reply to play command - int playTransaction = -1;///<The transaction number of the reply. - int playStreamId = -1;///<The stream id of the reply. - int playMessageType = -1;///<The message type of the reply. - - //generic state keeping - bool streamInited = false;///<Indicates whether init data for audio/video was sent. - int videoID = -1; - int audioID = -1; - - Socket::Connection Socket; ///< A copy of the user socket to allow helper functions to directly send data. - Socket::Connection ss; ///< Socket connected to server. - std::string streamName; ///< Stream that will be opened. - std::string app_name; ///< Name of the application that was opened - - ///\brief Sends a RTMP command either in AMF or AMF3 mode. - ///\param amfReply The data to be sent over RTMP. - ///\param messageType The type of message. - ///\param streamId The ID of the AMF stream. - void sendCommand(AMF::Object & amfReply, int messageType, int streamId){ - #if DEBUG >= 8 - std::cerr << amfReply.Print() << std::endl; - #endif - if (messageType == 17){ - Socket.SendNow(RTMPStream::SendChunk(3, messageType, streamId, (char)0 + amfReply.Pack())); - }else{ - Socket.SendNow(RTMPStream::SendChunk(3, messageType, streamId, amfReply.Pack())); - } - } //sendCommand - - ///\brief Parses a single AMF command message, and sends a direct response through sendCommand(). - ///\param amfData The received request. - ///\param messageType The type of message. - ///\param streamId The ID of the AMF stream. - void parseAMFCommand(AMF::Object & amfData, int messageType, int streamId){ - #if DEBUG >= 5 - fprintf(stderr, "Received command: %s\n", amfData.Print().c_str()); - #endif - #if DEBUG >= 8 - fprintf(stderr, "AMF0 command: %s\n", amfData.getContentP(0)->StrValue().c_str()); - #endif - if (amfData.getContentP(0)->StrValue() == "connect"){ - double objencoding = 0; - if (amfData.getContentP(2)->getContentP("objectEncoding")){ - objencoding = amfData.getContentP(2)->getContentP("objectEncoding")->NumValue(); - } - #if DEBUG >= 6 - int tmpint; - if (amfData.getContentP(2)->getContentP("videoCodecs")){ - tmpint = (int)amfData.getContentP(2)->getContentP("videoCodecs")->NumValue(); - if (tmpint & 0x04){ - fprintf(stderr, "Sorensen video support detected\n"); - } - if (tmpint & 0x80){ - fprintf(stderr, "H264 video support detected\n"); - } - } - if (amfData.getContentP(2)->getContentP("audioCodecs")){ - tmpint = (int)amfData.getContentP(2)->getContentP("audioCodecs")->NumValue(); - if (tmpint & 0x04){ - fprintf(stderr, "MP3 audio support detected\n"); - } - if (tmpint & 0x400){ - fprintf(stderr, "AAC audio support detected\n"); - } - } - #endif - app_name = amfData.getContentP(2)->getContentP("tcUrl")->StrValue(); - app_name = app_name.substr(app_name.find('/', 7) + 1); - RTMPStream::chunk_snd_max = 4096; - Socket.Send(RTMPStream::SendCTL(1, RTMPStream::chunk_snd_max)); //send chunk size max (msg 1) - Socket.Send(RTMPStream::SendCTL(5, RTMPStream::snd_window_size)); //send window acknowledgement size (msg 5) - Socket.Send(RTMPStream::SendCTL(6, RTMPStream::rec_window_size)); //send rec window acknowledgement size (msg 6) - Socket.Send(RTMPStream::SendUSR(0, 1)); //send UCM StreamBegin (0), stream 1 - //send a _result reply - AMF::Object amfReply("container", AMF::AMF0_DDV_CONTAINER); - amfReply.addContent(AMF::Object("", "_result")); //result success - amfReply.addContent(amfData.getContent(1)); //same transaction ID - amfReply.addContent(AMF::Object("")); //server properties - amfReply.getContentP(2)->addContent(AMF::Object("fmsVer", "FMS/3,5,5,2004")); - amfReply.getContentP(2)->addContent(AMF::Object("capabilities", (double)31)); - amfReply.getContentP(2)->addContent(AMF::Object("mode", (double)1)); - amfReply.addContent(AMF::Object("")); //info - amfReply.getContentP(3)->addContent(AMF::Object("level", "status")); - amfReply.getContentP(3)->addContent(AMF::Object("code", "NetConnection.Connect.Success")); - amfReply.getContentP(3)->addContent(AMF::Object("description", "Connection succeeded.")); - amfReply.getContentP(3)->addContent(AMF::Object("clientid", 1337)); - amfReply.getContentP(3)->addContent(AMF::Object("objectEncoding", objencoding)); - //amfReply.getContentP(3)->addContent(AMF::Object("data", AMF::AMF0_ECMA_ARRAY)); - //amfReply.getContentP(3)->getContentP(4)->addContent(AMF::Object("version", "3,5,4,1004")); - sendCommand(amfReply, messageType, streamId); - //send onBWDone packet - no clue what it is, but real server sends it... - //amfReply = AMF::Object("container", AMF::AMF0_DDV_CONTAINER); - //amfReply.addContent(AMF::Object("", "onBWDone"));//result - //amfReply.addContent(amfData.getContent(1));//same transaction ID - //amfReply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL));//null - //sendCommand(amfReply, messageType, streamId); - return; - } //connect - if (amfData.getContentP(0)->StrValue() == "createStream"){ - //send a _result reply - AMF::Object amfReply("container", AMF::AMF0_DDV_CONTAINER); - amfReply.addContent(AMF::Object("", "_result")); //result success - amfReply.addContent(amfData.getContent(1)); //same transaction ID - amfReply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL)); //null - command info - amfReply.addContent(AMF::Object("", (double)1)); //stream ID - we use 1 - sendCommand(amfReply, messageType, streamId); - Socket.Send(RTMPStream::SendUSR(0, 1)); //send UCM StreamBegin (0), stream 1 - return; - } //createStream - if ((amfData.getContentP(0)->StrValue() == "closeStream") || (amfData.getContentP(0)->StrValue() == "deleteStream")){ - if (ss.connected()){ - ss.close(); - } - return; - } - if ((amfData.getContentP(0)->StrValue() == "FCUnpublish") || (amfData.getContentP(0)->StrValue() == "releaseStream")){ - // ignored - return; - } - if ((amfData.getContentP(0)->StrValue() == "FCPublish")){ - //send a FCPublic reply - AMF::Object amfReply("container", AMF::AMF0_DDV_CONTAINER); - amfReply.addContent(AMF::Object("", "onFCPublish")); //status reply - amfReply.addContent(AMF::Object("", 0, AMF::AMF0_NUMBER)); //same transaction ID - amfReply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL)); //null - command info - amfReply.addContent(AMF::Object("")); //info - amfReply.getContentP(3)->addContent(AMF::Object("code", "NetStream.Publish.Start")); - amfReply.getContentP(3)->addContent(AMF::Object("description", "Please followup with publish command...")); - sendCommand(amfReply, messageType, streamId); - return; - } //FCPublish - if (amfData.getContentP(0)->StrValue() == "releaseStream"){ - //send a _result reply - AMF::Object amfReply("container", AMF::AMF0_DDV_CONTAINER); - amfReply.addContent(AMF::Object("", "_result")); //result success - amfReply.addContent(amfData.getContent(1)); //same transaction ID - amfReply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL)); //null - command info - amfReply.addContent(AMF::Object("", AMF::AMF0_UNDEFINED)); //stream ID? - sendCommand(amfReply, messageType, streamId); - return; - }//releaseStream - if ((amfData.getContentP(0)->StrValue() == "getStreamLength") || (amfData.getContentP(0)->StrValue() == "getMovLen")){ - //send a _result reply - AMF::Object amfReply("container", AMF::AMF0_DDV_CONTAINER); - amfReply.addContent(AMF::Object("", "_result")); //result success - amfReply.addContent(amfData.getContent(1)); //same transaction ID - amfReply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL)); //null - command info - amfReply.addContent(AMF::Object("", (double)0)); //zero length - sendCommand(amfReply, messageType, streamId); - return; - } //getStreamLength - if ((amfData.getContentP(0)->StrValue() == "publish")){ - if (amfData.getContentP(3)){ - streamName = amfData.getContentP(3)->StrValue(); - /// \todo implement push for MistPlayer or restrict and change to getLive - ss = Util::Stream::getStream(streamName); - if ( !ss.connected()){ - #if DEBUG >= 1 - fprintf(stderr, "Could not connect to server!\n"); - #endif - Socket.close(); //disconnect user - return; - } - DTSC::Stream Strm; - Strm.waitForMeta(ss); - ss.Send("P "); - ss.Send(Socket.getHost().c_str()); - ss.Send(" "); - ss.Send(app_name); - ss.SendNow("\n"); - streamReset = true; - noStats = true; - } - //send a _result reply - AMF::Object amfReply("container", AMF::AMF0_DDV_CONTAINER); - amfReply.addContent(AMF::Object("", "_result")); //result success - amfReply.addContent(amfData.getContent(1)); //same transaction ID - amfReply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL)); //null - command info - amfReply.addContent(AMF::Object("", 1, AMF::AMF0_BOOL)); //publish success? - sendCommand(amfReply, messageType, streamId); - Socket.Send(RTMPStream::SendUSR(0, 1)); //send UCM StreamBegin (0), stream 1 - //send a status reply - amfReply = AMF::Object("container", AMF::AMF0_DDV_CONTAINER); - amfReply.addContent(AMF::Object("", "onStatus")); //status reply - amfReply.addContent(AMF::Object("", 0, AMF::AMF0_NUMBER)); //same transaction ID - amfReply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL)); //null - command info - amfReply.addContent(AMF::Object("")); //info - amfReply.getContentP(3)->addContent(AMF::Object("level", "status")); - amfReply.getContentP(3)->addContent(AMF::Object("code", "NetStream.Publish.Start")); - amfReply.getContentP(3)->addContent(AMF::Object("description", "Stream is now published!")); - amfReply.getContentP(3)->addContent(AMF::Object("clientid", (double)1337)); - sendCommand(amfReply, messageType, streamId); - return; - } //getStreamLength - if (amfData.getContentP(0)->StrValue() == "checkBandwidth"){ - //send a _result reply - AMF::Object amfReply("container", AMF::AMF0_DDV_CONTAINER); - amfReply.addContent(AMF::Object("", "_result")); //result success - amfReply.addContent(amfData.getContent(1)); //same transaction ID - amfReply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL)); //null - command info - amfReply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL)); //null - command info - sendCommand(amfReply, messageType, streamId); - return; - } //checkBandwidth - if ((amfData.getContentP(0)->StrValue() == "play") || (amfData.getContentP(0)->StrValue() == "play2")){ - //set reply number and stream name, actual reply is sent up in the ss.spool() handler - playTransaction = amfData.getContentP(1)->NumValue(); - playMessageType = messageType; - playStreamId = streamId; - streamName = amfData.getContentP(3)->StrValue(); - Connector_RTMP::ready4data = true; //start sending video data! - return; - } //play - if ((amfData.getContentP(0)->StrValue() == "seek")){ - //set reply number and stream name, actual reply is sent up in the ss.spool() handler - playTransaction = amfData.getContentP(1)->NumValue(); - playMessageType = messageType; - playStreamId = streamId; - streamInited = false; - - AMF::Object amfReply("container", AMF::AMF0_DDV_CONTAINER); - amfReply.addContent(AMF::Object("", "onStatus")); //status reply - amfReply.addContent(amfData.getContent(1)); //same transaction ID - amfReply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL)); //null - command info - amfReply.addContent(AMF::Object("")); //info - amfReply.getContentP(3)->addContent(AMF::Object("level", "status")); - amfReply.getContentP(3)->addContent(AMF::Object("code", "NetStream.Seek.Notify")); - amfReply.getContentP(3)->addContent(AMF::Object("description", "Seeking to the specified time")); - amfReply.getContentP(3)->addContent(AMF::Object("details", "DDV")); - amfReply.getContentP(3)->addContent(AMF::Object("clientid", (double)1337)); - sendCommand(amfReply, playMessageType, playStreamId); - ss.Send("s "); - ss.Send(JSON::Value((long long int)amfData.getContentP(3)->NumValue()).asString().c_str()); - ss.SendNow("\n"); - return; - } //seek - if ((amfData.getContentP(0)->StrValue() == "pauseRaw") || (amfData.getContentP(0)->StrValue() == "pause")){ - if (amfData.getContentP(3)->NumValue()){ - ss.Send("q\n"); //quit playing - //send a status reply - AMF::Object amfReply("container", AMF::AMF0_DDV_CONTAINER); - amfReply.addContent(AMF::Object("", "onStatus")); //status reply - amfReply.addContent(amfData.getContent(1)); //same transaction ID - amfReply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL)); //null - command info - amfReply.addContent(AMF::Object("")); //info - amfReply.getContentP(3)->addContent(AMF::Object("level", "status")); - amfReply.getContentP(3)->addContent(AMF::Object("code", "NetStream.Pause.Notify")); - amfReply.getContentP(3)->addContent(AMF::Object("description", "Pausing playback")); - amfReply.getContentP(3)->addContent(AMF::Object("details", "DDV")); - amfReply.getContentP(3)->addContent(AMF::Object("clientid", (double)1337)); - sendCommand(amfReply, playMessageType, playStreamId); - }else{ - ss.SendNow("p\n"); //start playing - //send a status reply - AMF::Object amfReply("container", AMF::AMF0_DDV_CONTAINER); - amfReply.addContent(AMF::Object("", "onStatus")); //status reply - amfReply.addContent(amfData.getContent(1)); //same transaction ID - amfReply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL)); //null - command info - amfReply.addContent(AMF::Object("")); //info - amfReply.getContentP(3)->addContent(AMF::Object("level", "status")); - amfReply.getContentP(3)->addContent(AMF::Object("code", "NetStream.Unpause.Notify")); - amfReply.getContentP(3)->addContent(AMF::Object("description", "Resuming playback")); - amfReply.getContentP(3)->addContent(AMF::Object("details", "DDV")); - amfReply.getContentP(3)->addContent(AMF::Object("clientid", (double)1337)); - sendCommand(amfReply, playMessageType, playStreamId); - } - return; - } //seek - - #if DEBUG >= 2 - fprintf(stderr, "AMF0 command not processed!\n%s\n", amfData.Print().c_str()); - #endif - } //parseAMFCommand - - ///\brief Gets and parses one RTMP chunk at a time. - ///\param inputBuffer A buffer filled with chunk data. - void parseChunk(Socket::Buffer & inputBuffer){ - //for DTSC conversion - static DTSC::Meta meta_out; - static std::stringstream prebuffer; // Temporary buffer before sending real data - static bool sending = false; - static unsigned int counter = 0; - //for chunk parsing - static RTMPStream::Chunk next; - static FLV::Tag F; - static AMF::Object amfdata("empty", AMF::AMF0_DDV_CONTAINER); - static AMF::Object amfelem("empty", AMF::AMF0_DDV_CONTAINER); - static AMF::Object3 amf3data("empty", AMF::AMF3_DDV_CONTAINER); - static AMF::Object3 amf3elem("empty", AMF::AMF3_DDV_CONTAINER); - - while (next.Parse(inputBuffer)){ - - //send ACK if we received a whole window - if ((RTMPStream::rec_cnt - RTMPStream::rec_window_at > RTMPStream::rec_window_size)){ - RTMPStream::rec_window_at = RTMPStream::rec_cnt; - Socket.Send(RTMPStream::SendCTL(3, RTMPStream::rec_cnt)); //send ack (msg 3) - } - - switch (next.msg_type_id){ - case 0: //does not exist - #if DEBUG >= 2 - fprintf(stderr, "UNKN: Received a zero-type message. Possible data corruption? Aborting!\n"); - #endif - while (inputBuffer.size()){ - inputBuffer.get().clear(); - } - ss.close(); - Socket.close(); - break; //happens when connection breaks unexpectedly - case 1: //set chunk size - RTMPStream::chunk_rec_max = ntohl(*(int*)next.data.c_str()); - #if DEBUG >= 5 - fprintf(stderr, "CTRL: Set chunk size: %i\n", RTMPStream::chunk_rec_max); - #endif - break; - case 2: //abort message - we ignore this one - #if DEBUG >= 5 - fprintf(stderr, "CTRL: Abort message\n"); - #endif - //4 bytes of stream id to drop - break; - case 3: //ack - #if DEBUG >= 8 - fprintf(stderr, "CTRL: Acknowledgement\n"); - #endif - RTMPStream::snd_window_at = ntohl(*(int*)next.data.c_str()); - RTMPStream::snd_window_at = RTMPStream::snd_cnt; - break; - case 4: { - //2 bytes event type, rest = event data - //types: - //0 = stream begin, 4 bytes ID - //1 = stream EOF, 4 bytes ID - //2 = stream dry, 4 bytes ID - //3 = setbufferlen, 4 bytes ID, 4 bytes length - //4 = streamisrecorded, 4 bytes ID - //6 = pingrequest, 4 bytes data - //7 = pingresponse, 4 bytes data - //we don't need to process this - #if DEBUG >= 5 - short int ucmtype = ntohs(*(short int*)next.data.c_str()); - switch (ucmtype){ - case 0: - fprintf(stderr, "CTRL: UCM StreamBegin %i\n", ntohl(*((int*)(next.data.c_str()+2)))); - break; - case 1: - fprintf(stderr, "CTRL: UCM StreamEOF %i\n", ntohl(*((int*)(next.data.c_str()+2)))); - break; - case 2: - fprintf(stderr, "CTRL: UCM StreamDry %i\n", ntohl(*((int*)(next.data.c_str()+2)))); - break; - case 3: - fprintf(stderr, "CTRL: UCM SetBufferLength %i %i\n", ntohl(*((int*)(next.data.c_str()+2))), ntohl(*((int*)(next.data.c_str()+6)))); - break; - case 4: - fprintf(stderr, "CTRL: UCM StreamIsRecorded %i\n", ntohl(*((int*)(next.data.c_str()+2)))); - break; - case 6: - fprintf(stderr, "CTRL: UCM PingRequest %i\n", ntohl(*((int*)(next.data.c_str()+2)))); - break; - case 7: - fprintf(stderr, "CTRL: UCM PingResponse %i\n", ntohl(*((int*)(next.data.c_str()+2)))); - break; - default: - fprintf(stderr, "CTRL: UCM Unknown (%hi)\n", ucmtype); - break; - } - #endif - } - break; - case 5: //window size of other end - #if DEBUG >= 5 - fprintf(stderr, "CTRL: Window size\n"); - #endif - RTMPStream::rec_window_size = ntohl(*(int*)next.data.c_str()); - RTMPStream::rec_window_at = RTMPStream::rec_cnt; - Socket.Send(RTMPStream::SendCTL(3, RTMPStream::rec_cnt)); //send ack (msg 3) - break; - case 6: - #if DEBUG >= 5 - fprintf(stderr, "CTRL: Set peer bandwidth\n"); - #endif - //4 bytes window size, 1 byte limit type (ignored) - RTMPStream::snd_window_size = ntohl(*(int*)next.data.c_str()); - Socket.Send(RTMPStream::SendCTL(5, RTMPStream::snd_window_size)); //send window acknowledgement size (msg 5) - break; - case 8: //audio data - case 9: //video data - case 18: //meta data - if (ss.connected()){ - if (streamReset){ - //reset push data to empty, in case stream properties change - meta_out.reset(); - prebuffer.str(""); - sending = false; - counter = 0; - streamReset = false; - } - F.ChunkLoader(next); - JSON::Value pack_out = F.toJSON(meta_out); - if ( !pack_out.isNull()){ - if ( !sending){ - counter++; - if (counter > 8){ - sending = true; - meta_out.send(ss); - ss.SendNow(prebuffer.str()); //write buffer - prebuffer.str(""); //clear buffer - pack_out.sendTo(ss); - }else{ - prebuffer << pack_out.toNetPacked(); - } - }else{ - pack_out.sendTo(ss); - } - } - }else{ - #if DEBUG >= 5 - fprintf(stderr, "Received useless media data\n"); - #endif - Socket.close(); - } - break; - case 15: - #if DEBUG >= 5 - fprintf(stderr, "Received AFM3 data message\n"); - #endif - break; - case 16: - #if DEBUG >= 5 - fprintf(stderr, "Received AFM3 shared object\n"); - #endif - break; - case 17: { - #if DEBUG >= 5 - fprintf(stderr, "Received AFM3 command message\n"); - #endif - if (next.data[0] != 0){ - next.data = next.data.substr(1); - amf3data = AMF::parse3(next.data); - #if DEBUG >= 5 - amf3data.Print(); - #endif - }else{ - #if DEBUG >= 5 - fprintf(stderr, "Received AFM3-0 command message\n"); - #endif - next.data = next.data.substr(1); - amfdata = AMF::parse(next.data); - parseAMFCommand(amfdata, 17, next.msg_stream_id); - } //parsing AMF0-style - } - break; - case 19: - #if DEBUG >= 5 - fprintf(stderr, "Received AFM0 shared object\n"); - #endif - break; - case 20: { //AMF0 command message - amfdata = AMF::parse(next.data); - parseAMFCommand(amfdata, 20, next.msg_stream_id); - } - break; - case 22: - #if DEBUG >= 5 - fprintf(stderr, "Received aggregate message\n"); - #endif - break; - default: - #if DEBUG >= 1 - fprintf(stderr, "Unknown chunk received! Probably protocol corruption, stopping parsing of incoming data.\n"); - #endif - stopParsing = true; - break; - } - } - } //parseChunk - - ///\brief Main function for the RTMP Connector - ///\param conn A socket describing the connection the client. - ///\return The exit code of the connector. - int rtmpConnector(Socket::Connection & conn){ - Socket = conn; - Socket.setBlocking(false); - FLV::Tag tag, init_tag; - DTSC::Stream Strm; - - while ( !Socket.Received().available(1537) && Socket.connected()){ - Socket.spool(); - Util::sleep(5); - } - RTMPStream::handshake_in = Socket.Received().remove(1537); - RTMPStream::rec_cnt += 1537; - - if (RTMPStream::doHandshake()){ - Socket.SendNow(RTMPStream::handshake_out); - while ( !Socket.Received().available(1536) && Socket.connected()){ - Socket.spool(); - Util::sleep(5); - } - Socket.Received().remove(1536); - RTMPStream::rec_cnt += 1536; - #if DEBUG >= 5 - fprintf(stderr, "Handshake succcess!\n"); - #endif - }else{ - fprintf(stderr, "RTMP: Handshake fail!\n"); - return 0; - } - - unsigned int lastStats = 0; - bool firsttime = true; - - while (Socket.connected()){ - if (Socket.spool() || firsttime){ - parseChunk(Socket.Received()); - firsttime = false; - }else{ - Util::sleep(1); //sleep 1ms to prevent high CPU usage - } - if (ready4data){ - if ( !inited){ - //we are ready, connect the socket! - ss = Util::Stream::getStream(streamName); - if ( !ss.connected()){ - #if DEBUG >= 1 - fprintf(stderr, "Could not connect to server!\n"); - #endif - Socket.close(); //disconnect user - break; - } - ss.setBlocking(false); - Strm.waitForMeta(ss); - //find first audio and video tracks - for (std::map<int,DTSC::Track>::iterator it = Strm.metadata.tracks.begin(); it != Strm.metadata.tracks.end(); it++){ - if (videoID == -1 && (it->second.codec == "H264" || it->second.codec == "H263" || it->second.codec == "VP6")){ - videoID = it->second.trackID; - } - if (audioID == -1 && (it->second.codec == "AAC" || it->second.codec == "MP3")){ - audioID = it->second.trackID; - } - } - //select the tracks and play - std::stringstream cmd; - cmd << "t"; - if (videoID != -1){ - cmd << " " << videoID; - } - if (audioID != -1){ - cmd << " " << audioID; - } - cmd << "\np\n"; - ss.SendNow(cmd.str().c_str()); - inited = true; - } - if (inited && !noStats){ - long long int now = Util::epoch(); - if (now != lastStats){ - lastStats = now; - ss.SendNow(Socket.getStats("RTMP")); - } - } - if (ss.spool()){ - while (Strm.parsePacket(ss.Received())){ - if (playTransaction != -1){ - //send a status reply - AMF::Object amfreply("container", AMF::AMF0_DDV_CONTAINER); - amfreply.addContent(AMF::Object("", "onStatus")); //status reply - amfreply.addContent(AMF::Object("", (double)playTransaction)); //same transaction ID - amfreply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL)); //null - command info - amfreply.addContent(AMF::Object("")); //info - amfreply.getContentP(3)->addContent(AMF::Object("level", "status")); - amfreply.getContentP(3)->addContent(AMF::Object("code", "NetStream.Play.Reset")); - amfreply.getContentP(3)->addContent(AMF::Object("description", "Playing and resetting...")); - amfreply.getContentP(3)->addContent(AMF::Object("details", "DDV")); - amfreply.getContentP(3)->addContent(AMF::Object("clientid", (double)1337)); - sendCommand(amfreply, playMessageType, playStreamId); - //send streamisrecorded if stream, well, is recorded. - if (Strm.metadata.vod){//isMember("length") && Strm.metadata["length"].asInt() > 0){ - Socket.Send(RTMPStream::SendUSR(4, 1)); //send UCM StreamIsRecorded (4), stream 1 - } - //send streambegin - Socket.Send(RTMPStream::SendUSR(0, 1)); //send UCM StreamBegin (0), stream 1 - //and more reply - amfreply = AMF::Object("container", AMF::AMF0_DDV_CONTAINER); - amfreply.addContent(AMF::Object("", "onStatus")); //status reply - amfreply.addContent(AMF::Object("", (double)playTransaction)); //same transaction ID - amfreply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL)); //null - command info - amfreply.addContent(AMF::Object("")); //info - amfreply.getContentP(3)->addContent(AMF::Object("level", "status")); - amfreply.getContentP(3)->addContent(AMF::Object("code", "NetStream.Play.Start")); - amfreply.getContentP(3)->addContent(AMF::Object("description", "Playing!")); - amfreply.getContentP(3)->addContent(AMF::Object("details", "DDV")); - amfreply.getContentP(3)->addContent(AMF::Object("clientid", (double)1337)); - sendCommand(amfreply, playMessageType, playStreamId); - RTMPStream::chunk_snd_max = 102400; //100KiB - Socket.Send(RTMPStream::SendCTL(1, RTMPStream::chunk_snd_max)); //send chunk size max (msg 1) - //send dunno? - Socket.Send(RTMPStream::SendUSR(32, 1)); //send UCM no clue?, stream 1 - playTransaction = -1; - } - - //sent init data if needed - if ( !streamInited){ - init_tag.DTSCMetaInit(Strm, Strm.metadata.tracks[videoID], Strm.metadata.tracks[audioID]); - if (init_tag.len){ - Socket.SendNow(RTMPStream::SendMedia(init_tag)); - } - if (audioID != -1){ - init_tag.DTSCAudioInit(Strm.metadata.tracks[audioID]); - if (init_tag.len){ - Socket.SendNow(RTMPStream::SendMedia(init_tag)); - } - } - if (videoID != -1){ - init_tag.DTSCVideoInit(Strm.metadata.tracks[videoID]); - if (init_tag.len){ - Socket.SendNow(RTMPStream::SendMedia(init_tag)); - } - } - streamInited = true; - } - //sent a tag - if (tag.DTSCLoader(Strm)){ - if (tag.len){ - Socket.SendNow(RTMPStream::SendMedia(tag)); - #if DEBUG >= 8 - fprintf(stderr, "Sent tag to %i: [%u] %s\n", Socket.getSocket(), tag.tagTime(), tag.tagType().c_str()); - #endif - } - } - } - } - } - } - Socket.close(); - ss.SendNow(Socket.getStats("RTMP").c_str()); - ss.close(); - return 0; - } //Connector_RTMP -} - -///\brief The standard process-spawning main function. -int main(int argc, char ** argv){ - Util::Config conf(argv[0], PACKAGE_VERSION); - JSON::Value capa; - capa["desc"] = "Enables the RTMP protocol which is used by Adobe Flash Player."; - capa["deps"] = ""; - capa["url_rel"] = "/play/$"; - capa["codecs"][0u][0u].append("H264"); - capa["codecs"][0u][0u].append("H263"); - capa["codecs"][0u][0u].append("VP6"); - capa["codecs"][0u][1u].append("AAC"); - capa["codecs"][0u][1u].append("MP3"); - capa["methods"][0u]["handler"] = "rtmp"; - capa["methods"][0u]["type"] = "flash/10"; - capa["methods"][0u]["priority"] = 6ll; - conf.addConnectorOptions(1935, capa); - conf.parseArgs(argc, argv); - if (conf.getBool("json")){ - std::cout << capa.toString() << std::endl; - return -1; - } - - return conf.serveForkedSocket(Connector_RTMP::rtmpConnector); -} //main diff --git a/src/connectors/conn_ts.cpp b/src/connectors/conn_ts.cpp deleted file mode 100644 index 0809257b..00000000 --- a/src/connectors/conn_ts.cpp +++ /dev/null @@ -1,215 +0,0 @@ -/// \file conn_ts.cpp -/// Contains the main code for the TS Connector - -#include <queue> -#include <string> -#include <iostream> - -#include <cmath> -#include <ctime> -#include <cstdio> -#include <cstdlib> -#include <cstring> -#include <unistd.h> -#include <getopt.h> -#include <sys/time.h> -#include <sys/wait.h> -#include <sys/types.h> - -#include <mist/socket.h> -#include <mist/config.h> -#include <mist/stream.h> -#include <mist/ts_packet.h> //TS support -#include <mist/dtsc.h> //DTSC support -#include <mist/mp4.h> //For initdata conversion -#include <mist/mp4_generic.h> - -///\brief Holds everything unique to the TS Connector -namespace Connector_TS { - std::string streamName; - std::string trackIDs; - - ///\brief Main function for the TS Connector - ///\param conn A socket describing the connection the client. - ///\return The exit code of the connector. - int tsConnector(Socket::Connection & conn){ - std::string ToPack; - TS::Packet PackData; - std::string DTMIData; - int PacketNumber = 0; - long long unsigned int TimeStamp = 0; - unsigned int ThisNaluSize; - char VideoCounter = 0; - char AudioCounter = 0; - bool IsKeyFrame = false; - MP4::AVCC avccbox; - bool haveAvcc = false; - - DTSC::Stream Strm; - bool inited = false; - Socket::Connection ss; - - while (conn.connected()){ - if ( !inited){ - ss = Util::Stream::getStream(streamName); - if ( !ss.connected()){ - #if DEBUG >= 1 - fprintf(stderr, "Could not connect to server!\n"); - #endif - conn.close(); - break; - } - - if(trackIDs == ""){ - std::stringstream tmpTracks; - // no track ids given? Find the first video and first audio track (if available) and use those! - int videoID = -1; - int audioID = -1; - - Strm.waitForMeta(ss); - for (std::map<int,DTSC::Track>::iterator it = Strm.metadata.tracks.begin(); it != Strm.metadata.tracks.end(); it++){ - if (audioID == -1 && it->second.codec == "AAC"){ - audioID = it->first; - tmpTracks << " " << it->first; - } - - if (videoID == -1 && it->second.codec == "H264"){ - videoID = it->first; - tmpTracks << " " << it->first; - } - - } // for iterator - trackIDs += tmpTracks.str(); - } // if trackIDs == "" - - std::string cmd = "t " + trackIDs + "\ns 0\np\n"; - ss.SendNow( cmd ); - inited = true; - } - if (ss.spool()){ - while (Strm.parsePacket(ss.Received())){ - - std::stringstream TSBuf; - Socket::Buffer ToPack; - //write PAT and PMT TS packets - if (PacketNumber == 0){ - PackData.DefaultPAT(); - TSBuf.write(PackData.ToString(), 188); - PackData.DefaultPMT(); - TSBuf.write(PackData.ToString(), 188); - PacketNumber += 2; - } - - int PIDno = 0; - char * ContCounter = 0; - if (Strm.lastType() == DTSC::VIDEO){ - if ( !haveAvcc){ - avccbox.setPayload(Strm.metadata.tracks[Strm.getPacket()["trackid"].asInt()].init); - haveAvcc = true; - } - - IsKeyFrame = Strm.getPacket().isMember("keyframe"); - if (IsKeyFrame){ - TimeStamp = (Strm.getPacket()["time"].asInt() * 27000); - } - ToPack.append(avccbox.asAnnexB()); - while (Strm.lastData().size() > 4){ - ThisNaluSize = (Strm.lastData()[0] << 24) + (Strm.lastData()[1] << 16) + (Strm.lastData()[2] << 8) + Strm.lastData()[3]; - Strm.lastData().replace(0, 4, "\000\000\000\001", 4); - if (ThisNaluSize + 4 == Strm.lastData().size()){ - ToPack.append(Strm.lastData()); - break; - }else{ - ToPack.append(Strm.lastData().c_str(), ThisNaluSize + 4); - Strm.lastData().erase(0, ThisNaluSize + 4); - } - } - ToPack.prepend(TS::Packet::getPESVideoLeadIn(0ul, Strm.getPacket()["time"].asInt() * 90)); - PIDno = 0x100 - 1 + Strm.getPacket()["trackid"].asInt(); - ContCounter = &VideoCounter; - }else if (Strm.lastType() == DTSC::AUDIO){ - ToPack.append(TS::GetAudioHeader(Strm.lastData().size(), Strm.metadata.tracks[Strm.getPacket()["trackid"].asInt()].init)); - ToPack.append(Strm.lastData()); - ToPack.prepend(TS::Packet::getPESAudioLeadIn(ToPack.bytes(1073741824ul), Strm.getPacket()["time"].asInt() * 90)); - PIDno = 0x100 - 1 + Strm.getPacket()["trackid"].asInt(); - ContCounter = &AudioCounter; - IsKeyFrame = false; - } - - //initial packet - PackData.Clear(); - PackData.PID(PIDno); - PackData.ContinuityCounter(( *ContCounter)++); - PackData.UnitStart(1); - if (IsKeyFrame){ - PackData.RandomAccess(1); - PackData.PCR(TimeStamp); - } - unsigned int toSend = PackData.AddStuffing(ToPack.bytes(184)); - std::string gonnaSend = ToPack.remove(toSend); - PackData.FillFree(gonnaSend); - TSBuf.write(PackData.ToString(), 188); - PacketNumber++; - - //rest of packets - while (ToPack.size()){ - PackData.Clear(); - PackData.PID(PIDno); - PackData.ContinuityCounter(( *ContCounter)++); - toSend = PackData.AddStuffing(ToPack.bytes(184)); - gonnaSend = ToPack.remove(toSend); - PackData.FillFree(gonnaSend); - TSBuf.write(PackData.ToString(), 188); - PacketNumber++; - } - - TSBuf.flush(); - if (TSBuf.str().size()){ - conn.SendNow(TSBuf.str().c_str(), TSBuf.str().size()); - TSBuf.str(""); - } - TSBuf.str(""); - PacketNumber = 0; - } - }else{ - Util::sleep(1000); - conn.spool(); - } - } - return 0; - } -} - -int main(int argc, char ** argv){ - Util::Config conf(argv[0], PACKAGE_VERSION); - JSON::Value capa; - capa["desc"] = "Enables the raw MPEG Transport Stream protocol over TCP."; - capa["deps"] = ""; - capa["required"]["streamname"]["name"] = "Stream"; - capa["required"]["streamname"]["help"] = "What streamname to serve. For multiple streams, add this protocol multiple times using different ports."; - capa["required"]["streamname"]["type"] = "str"; - capa["required"]["streamname"]["option"] = "--stream"; - capa["optional"]["tracks"]["name"] = "Tracks"; - capa["optional"]["tracks"]["help"] = "The track IDs of the stream that this connector will transmit separated by spaces"; - capa["optional"]["tracks"]["type"] = "str"; - capa["optional"]["tracks"]["option"] = "--tracks"; - conf.addOption("streamname", - JSON::fromString("{\"arg\":\"string\",\"short\":\"s\",\"long\":\"stream\",\"help\":\"The name of the stream that this connector will transmit.\"}")); - conf.addOption("tracks", - JSON::fromString("{\"arg\":\"string\",\"value\":[\"\"],\"short\": \"t\",\"long\":\"tracks\",\"help\":\"The track IDs of the stream that this connector will transmit separated by spaces.\"}")); - conf.addConnectorOptions(8888, capa); - bool ret = conf.parseArgs(argc, argv); - if (conf.getBool("json")){ - std::cout << capa.toString() << std::endl; - return -1; - } - if (!ret){ - std::cerr << "Usage error: missing argument(s)." << std::endl; - conf.printHelp(std::cout); - return 1; - } - - Connector_TS::streamName = conf.getString("streamname"); - Connector_TS::trackIDs = conf.getString("tracks"); - return conf.serveForkedSocket(Connector_TS::tsConnector); -} //main diff --git a/src/controller/controller.cpp b/src/controller/controller.cpp index bea98ceb..d36bf76b 100644 --- a/src/controller/controller.cpp +++ b/src/controller/controller.cpp @@ -20,8 +20,14 @@ #include "controller_connectors.h" #include "controller_streams.h" #include "controller_capabilities.h" +#include "controller_statistics.h" #include "server.html.h" + +#include <mist/tinythread.h> +#include <mist/shared_memory.h> + + #define UPLINK_INTERVAL 30 #ifndef COMPILED_USERNAME @@ -31,7 +37,8 @@ ///\brief Holds everything unique to the controller. namespace Controller { - + Util::Config conf; + Secure::Auth keychecker; ///< Checks key authorization. ///\brief A class storing information about a connected user. @@ -133,42 +140,23 @@ namespace Controller { out = in; } - ///\brief Parse received statistics. - ///\param stats The statistics to be parsed. - void CheckStats(JSON::Value & stats){ - long long int currTime = Util::epoch(); - for (JSON::ObjIter jit = stats.ObjBegin(); jit != stats.ObjEnd(); jit++){ - if (currTime - lastBuffer[jit->first] > 120){ - stats.removeMember(jit->first); - return; - }else{ - if (jit->second.isMember("curr") && jit->second["curr"].size() > 0){ - for (JSON::ObjIter u_it = jit->second["curr"].ObjBegin(); u_it != jit->second["curr"].ObjEnd(); ++u_it){ - if (u_it->second.isMember("now") && u_it->second["now"].asInt() < currTime - 3){ - jit->second["log"].append(u_it->second); - jit->second["curr"].removeMember(u_it->first); - if ( !jit->second["curr"].size()){ - break; - } - u_it = jit->second["curr"].ObjBegin(); - } - } - } - } - } - } } //Controller namespace /// the following function is a simple check if the user wants to proceed to fix (y), ignore (n) or abort on (a) a question -char yna(std::string user_input){ - if(user_input == "y" || user_input == "Y"){ - return 'y'; - }else if(user_input == "n" || user_input == "N"){ - return 'n'; - }else if(user_input == "a" || user_input == "A"){ - return 'a'; - }else{ - return 'x';//when no valid option is found, yna returns x +char yna(std::string & user_input){ + switch (user_input[0]){ + case 'y': case 'Y': + return 'y'; + break; + case 'n': case 'N': + return 'n'; + break; + case 'a': case 'A': + return 'a'; + break; + default: + return 'x'; + break; } } @@ -210,37 +198,37 @@ int main(int argc, char ** argv){ if ( !stored_user["default"]){ stored_user["default"] = "root"; } - Util::Config conf = Util::Config(argv[0], PACKAGE_VERSION " / " RELEASE); - conf.addOption("listen_port", stored_port); - conf.addOption("listen_interface", stored_interface); - conf.addOption("username", stored_user); - conf.addOption("daemonize", + Controller::conf = Util::Config(argv[0], PACKAGE_VERSION " / " RELEASE); + Controller::conf.addOption("listen_port", stored_port); + Controller::conf.addOption("listen_interface", stored_interface); + Controller::conf.addOption("username", stored_user); + Controller::conf.addOption("daemonize", JSON::fromString( "{\"long\":\"daemon\", \"short\":\"d\", \"default\":0, \"long_off\":\"nodaemon\", \"short_off\":\"n\", \"help\":\"Turns deamon mode on (-d) or off (-n). -d runs quietly in background, -n (default) enables verbose in foreground.\"}")); - conf.addOption("account", + Controller::conf.addOption("account", JSON::fromString( "{\"long\":\"account\", \"short\":\"a\", \"arg\":\"string\" \"default\":\"\", \"help\":\"A username:password string to create a new account with.\"}")); - conf.addOption("logfile", + Controller::conf.addOption("logfile", JSON::fromString( "{\"long\":\"logfile\", \"short\":\"L\", \"arg\":\"string\" \"default\":\"\",\"help\":\"Redirect all standard output to a log file, provided with an argument\"}")); - conf.addOption("configFile", + Controller::conf.addOption("configFile", JSON::fromString( "{\"long\":\"config\", \"short\":\"c\", \"arg\":\"string\" \"default\":\"config.json\", \"help\":\"Specify a config file other than default.\"}")); - conf.addOption("uplink", + Controller::conf.addOption("uplink", JSON::fromString( "{\"default\":\"\", \"arg\":\"string\", \"help\":\"MistSteward uplink host and port.\", \"short\":\"U\", \"long\":\"uplink\"}")); - conf.addOption("uplink-name", + Controller::conf.addOption("uplink-name", JSON::fromString( "{\"default\":\"" COMPILED_USERNAME "\", \"arg\":\"string\", \"help\":\"MistSteward uplink username.\", \"short\":\"N\", \"long\":\"uplink-name\"}")); - conf.addOption("uplink-pass", + Controller::conf.addOption("uplink-pass", JSON::fromString( "{\"default\":\"" COMPILED_PASSWORD "\", \"arg\":\"string\", \"help\":\"MistSteward uplink password.\", \"short\":\"P\", \"long\":\"uplink-pass\"}")); - conf.parseArgs(argc, argv); - if(conf.getString("logfile")!= ""){ + Controller::conf.parseArgs(argc, argv); + if(Controller::conf.getString("logfile")!= ""){ //open logfile, dup stdout to logfile - int output = open(conf.getString("logfile").c_str(),O_APPEND|O_CREAT|O_WRONLY,S_IRWXU); + int output = open(Controller::conf.getString("logfile").c_str(),O_APPEND|O_CREAT|O_WRONLY,S_IRWXU); if(output < 0){ - DEBUG_MSG(DLVL_ERROR, "Could not redirect output to %s: %s",conf.getString("logfile").c_str(),strerror(errno)); + DEBUG_MSG(DLVL_ERROR, "Could not redirect output to %s: %s",Controller::conf.getString("logfile").c_str(),strerror(errno)); return 7; }else{ dup2(output,STDOUT_FILENO); @@ -255,27 +243,25 @@ int main(int argc, char ** argv){ } } //Input custom config here - Controller::Storage = JSON::fromFile(conf.getString("configFile")); + Controller::Storage = JSON::fromFile(Controller::conf.getString("configFile")); //check for port, interface and username in arguments //if they are not there, take them from config file, if there - if (conf.getOption("listen_port", true).size() <= 1){ + if (Controller::conf.getOption("listen_port", true).size() <= 1){ if (Controller::Storage["config"]["controller"]["port"]){ - conf.getOption("listen_port") = Controller::Storage["config"]["controller"]["port"]; + Controller::conf.getOption("listen_port") = Controller::Storage["config"]["controller"]["port"]; } } - if (conf.getOption("listen_interface", true).size() <= 1){ + if (Controller::conf.getOption("listen_interface", true).size() <= 1){ if (Controller::Storage["config"]["controller"]["interface"]){ - conf.getOption("listen_interface") = Controller::Storage["config"]["controller"]["interface"]; + Controller::conf.getOption("listen_interface") = Controller::Storage["config"]["controller"]["interface"]; } } - if (conf.getOption("username", true).size() <= 1){ + if (Controller::conf.getOption("username", true).size() <= 1){ if (Controller::Storage["config"]["controller"]["username"]){ - conf.getOption("username") = Controller::Storage["config"]["controller"]["username"]; + Controller::conf.getOption("username") = Controller::Storage["config"]["controller"]["username"]; } } - - JSON::Value capabilities; //list available protocols and report about them std::deque<std::string> execs; @@ -284,6 +270,10 @@ int main(int argc, char ** argv){ char const * conn_args[] = {0, "-j", 0}; for (std::deque<std::string>::iterator it = execs.begin(); it != execs.end(); it++){ if ((*it).substr(0, 8) == "MistConn"){ + //skip if an MistOut already existed - MistOut takes precedence! + if (capabilities["connectors"].isMember((*it).substr(8))){ + continue; + } arg_one = Util::getMyPath() + (*it); conn_args[0] = arg_one.c_str(); capabilities["connectors"][(*it).substr(8)] = JSON::fromString(Util::Procs::getOutputOf((char**)conn_args)); @@ -291,9 +281,17 @@ int main(int argc, char ** argv){ capabilities["connectors"].removeMember((*it).substr(8)); } } + if ((*it).substr(0, 7) == "MistOut"){ + arg_one = Util::getMyPath() + (*it); + conn_args[0] = arg_one.c_str(); + capabilities["connectors"][(*it).substr(7)] = JSON::fromString(Util::Procs::getOutputOf((char**)conn_args)); + if (capabilities["connectors"][(*it).substr(7)].size() < 1){ + capabilities["connectors"].removeMember((*it).substr(7)); + } + } } - createAccount(conf.getString("account")); + createAccount(Controller::conf.getString("account")); /// User friendliness input added at this line if (isatty(fileno(stdin))){ @@ -340,11 +338,11 @@ int main(int argc, char ** argv){ } //check for streams if ( !Controller::Storage.isMember("streams") || Controller::Storage["streams"].size() < 1){ - std::cerr << "No streams configured, remember to set up streams through local settings page on port " << conf.getInteger("listen_port") << " or using the API." << std::endl; + std::cerr << "No streams configured, remember to set up streams through local settings page on port " << Controller::conf.getInteger("listen_port") << " or using the API." << std::endl; } } - std::string uplink_addr = conf.getString("uplink"); + std::string uplink_addr = Controller::conf.getString("uplink"); std::string uplink_host = ""; int uplink_port = 0; if (uplink_addr.size() > 0){ @@ -359,7 +357,7 @@ int main(int argc, char ** argv){ time_t lastuplink = 0; time_t processchecker = 0; - Socket::Server API_Socket = Socket::Server(conf.getInteger("listen_port"), conf.getString("listen_interface"), true); + Socket::Server API_Socket = Socket::Server(Controller::conf.getInteger("listen_port"), Controller::conf.getString("listen_interface"), true); Socket::Server Stats_Socket = Socket::Server(Util::getTmpFolder() + "statistics", true); Socket::Connection Incoming; std::vector<Controller::ConnectedUser> users; @@ -369,19 +367,21 @@ int main(int argc, char ** argv){ std::string jsonp; Controller::ConnectedUser * uplink = 0; Controller::Log("CONF", "Controller started"); - conf.activate(); - + Controller::conf.activate(); + //Create a converter class and automatically load in all encoders. Converter::Converter myConverter; - while (API_Socket.connected() && conf.is_active){ + tthread::thread statsThread(Controller::SharedMemStats, &Controller::conf); + + while (API_Socket.connected() && Controller::conf.is_active){ Util::sleep(10);//sleep for 10 ms - prevents 100% CPU time + - if (Util::epoch() - processchecker > 10){ + if (Util::epoch() - processchecker > 5){ processchecker = Util::epoch(); Controller::CheckProtocols(Controller::Storage["config"]["protocols"], capabilities); Controller::CheckAllStreams(Controller::Storage["streams"]); - Controller::CheckStats(Controller::Storage["statistics"]); myConverter.updateStatus(); } if (uplink_port && Util::epoch() - lastuplink > UPLINK_INTERVAL){ @@ -414,7 +414,8 @@ int main(int argc, char ** argv){ Response["config"] = Controller::Storage["config"]; Response["streams"] = Controller::Storage["streams"]; Response["log"] = Controller::Storage["log"]; - Response["statistics"] = Controller::Storage["statistics"]; + /// \todo Put this back in, someway, somehow... + //Response["statistics"] = Controller::Storage["statistics"]; Response["now"] = (unsigned int)lastuplink; uplink->H.Clean(); uplink->H.SetBody("command=" + HTTP::Parser::urlencode(Response.toString())); @@ -431,93 +432,6 @@ int main(int argc, char ** argv){ if (Incoming.connected()){ users.push_back((Controller::ConnectedUser)Incoming); } - Incoming = Stats_Socket.accept(true); - if (Incoming.connected()){ - buffers.push_back(Incoming); - } - if (buffers.size() > 0){ - for (std::vector<Socket::Connection>::iterator it = buffers.begin(); it != buffers.end(); it++){ - if ( !it->connected()){ - it->close(); - buffers.erase(it); - break; - } - if (it->spool()){ - while (it->Received().size()){ - it->Received().get().resize(it->Received().get().size() - 1); - Request = JSON::fromString(it->Received().get()); - it->Received().get().clear(); - if (Request.isMember("buffer")){ - std::string thisbuffer = Request["buffer"]; - Controller::lastBuffer[thisbuffer] = Util::epoch(); - //if metadata is available, store it - if (Request.isMember("meta")){ - Controller::Storage["streams"][thisbuffer]["meta"] = Request["meta"]; - } - if (Controller::Storage["streams"][thisbuffer].isMember("updated")){ - Controller::Storage["streams"][thisbuffer].removeMember("updated"); - if (Controller::Storage["streams"][thisbuffer].isMember("cut")){ - it->SendNow("c"+Controller::Storage["streams"][thisbuffer]["cut"].asString()+"\n"); - }else{ - it->SendNow("c0\n"); - } - if (Controller::Storage["streams"][thisbuffer].isMember("DVR")){ - it->SendNow("d"+Controller::Storage["streams"][thisbuffer]["DVR"].asString()+"\n"); - }else{ - it->SendNow("d20000\n"); - } - if (Controller::Storage["streams"][thisbuffer].isMember("source") && Controller::Storage["streams"][thisbuffer]["source"].asStringRef().substr(0, 7) == "push://"){ - it->SendNow("s"+Controller::Storage["streams"][thisbuffer]["source"].asStringRef().substr(7)+"\n"); - }else{ - it->SendNow("s127.0.01\n"); - } - } - if (Request.isMember("totals")){ - Controller::Storage["statistics"][thisbuffer]["curr"] = Request["curr"]; - std::string nowstr = Request["totals"]["now"].asString(); - Controller::Storage["statistics"][thisbuffer]["totals"][nowstr] = Request["totals"]; - Controller::Storage["statistics"][thisbuffer]["totals"][nowstr].removeMember("now"); - Controller::Storage["statistics"][thisbuffer]["totals"].shrink(600); //limit to 10 minutes of data - for (JSON::ObjIter jit = Request["log"].ObjBegin(); jit != Request["log"].ObjEnd(); jit++){ - Controller::Storage["statistics"][thisbuffer]["log"].append(jit->second); - Controller::Storage["statistics"][thisbuffer]["log"].shrink(1000); //limit to 1000 users per buffer - } - } - } - if (Request.isMember("vod")){ - std::string thisfile = Request["vod"]["filename"]; - for (JSON::ObjIter oit = Controller::Storage["streams"].ObjBegin(); oit != Controller::Storage["streams"].ObjEnd(); ++oit){ - if ((oit->second.isMember("source") && oit->second["source"].asString() == thisfile) - || (oit->second.isMember("channel") && oit->second["channel"]["URL"].asString() == thisfile)){ - Controller::lastBuffer[oit->first] = Util::epoch(); - if (Request["vod"].isMember("meta")){ - Controller::Storage["streams"][oit->first]["meta"] = Request["vod"]["meta"]; - } - JSON::Value sockit = (long long int)it->getSocket(); - std::string nowstr = Request["vod"]["now"].asString(); - Controller::Storage["statistics"][oit->first]["curr"][sockit.asString()] = Request["vod"]; - Controller::Storage["statistics"][oit->first]["curr"][sockit.asString()].removeMember("meta"); - JSON::Value nowtotal; - for (JSON::ObjIter u_it = Controller::Storage["statistics"][oit->first]["curr"].ObjBegin(); - u_it != Controller::Storage["statistics"][oit->first]["curr"].ObjEnd(); ++u_it){ - nowtotal["up"] = nowtotal["up"].asInt() + u_it->second["up"].asInt(); - nowtotal["down"] = nowtotal["down"].asInt() + u_it->second["down"].asInt(); - nowtotal["count"] = nowtotal["count"].asInt() + 1; - } - Controller::Storage["statistics"][oit->first]["totals"][nowstr] = nowtotal; - Controller::Storage["statistics"][oit->first]["totals"].shrink(600); - } - } - } - if (Request.isMember("ctrl_log") && Request["ctrl_log"].size() > 0){ - for (JSON::ArrIter it = Request["ctrl_log"].ArrBegin(); it != Request["ctrl_log"].ArrEnd(); it++){ - Controller::Log((*it)[0u], (*it)[1u]); - } - } - } - } - } - } if (users.size() > 0){ for (std::vector<Controller::ConnectedUser>::iterator it = users.begin(); it != users.end(); it++){ if ( !it->C.connected() || it->logins > 3){ @@ -543,12 +457,13 @@ int main(int argc, char ** argv){ Response["config"] = Controller::Storage["config"]; Response["streams"] = Controller::Storage["streams"]; Response["log"] = Controller::Storage["log"]; - Response["statistics"] = Controller::Storage["statistics"]; - Response["authorize"]["username"] = conf.getString("uplink-name"); + /// \todo Put this back in, someway, somehow... + //Response["statistics"] = Controller::Storage["statistics"]; + Response["authorize"]["username"] = Controller::conf.getString("uplink-name"); Controller::checkCapable(capabilities); Response["capabilities"] = capabilities; Controller::Log("UPLK", "Responding to login challenge: " + Request["authorize"]["challenge"].asString()); - Response["authorize"]["password"] = Secure::md5(conf.getString("uplink-pass") + Request["authorize"]["challenge"].asString()); + Response["authorize"]["password"] = Secure::md5(Controller::conf.getString("uplink-pass") + Request["authorize"]["challenge"].asString()); it->H.Clean(); it->H.SetBody("command=" + HTTP::Parser::urlencode(Response.toString())); it->H.BuildRequest(); @@ -568,7 +483,6 @@ int main(int argc, char ** argv){ } if (Request.isMember("clearstatlogs")){ Controller::Storage["log"].null(); - Controller::Storage["statistics"].null(); } } }else{ @@ -578,8 +492,9 @@ int main(int argc, char ** argv){ it->H.SetHeader("Content-Type", "text/html"); it->H.SetHeader("X-Info", "To force an API response, request the file /api"); it->H.SetHeader("Server", "mistserver/" PACKAGE_VERSION "/" + Util::Config::libver + "/" RELEASE); - it->H.SetBody(std::string((char*)server_html, (size_t)server_html_len)); - it->C.Send(it->H.BuildResponse("200", "OK")); + it->H.SetHeader("Content-Length", server_html_len); + it->H.SendResponse("200", "OK", it->C); + it->C.SendNow(server_html, server_html_len); it->H.Clean(); }else{ Authorize(Request, Response, ( *it)); @@ -622,10 +537,10 @@ int main(int argc, char ** argv){ } } if (Request.isMember("save")){ - if( Controller::WriteFile(conf.getString("configFile"), Controller::Storage.toString())){ + if( Controller::WriteFile(Controller::conf.getString("configFile"), Controller::Storage.toString())){ Controller::Log("CONF", "Config written to file on request through API"); }else{ - Controller::Log("ERROR", "Config " + conf.getString("configFile") + " could not be written"); + Controller::Log("ERROR", "Config " + Controller::conf.getString("configFile") + " could not be written"); } } //sent current configuration, no matter if it was changed or not @@ -640,11 +555,15 @@ int main(int argc, char ** argv){ } //sent any available logs and statistics Response["log"] = Controller::Storage["log"]; - Response["statistics"] = Controller::Storage["statistics"]; //clear log and statistics if requested if (Request.isMember("clearstatlogs")){ Controller::Storage["log"].null(); - Controller::Storage["statistics"].null(); + } + if (Request.isMember("clients")){ + Controller::fillClients(Request["clients"], Response["clients"]); + } + if (Request.isMember("totals")){ + Controller::fillTotals(Request["totals"], Response["totals"]); } } @@ -657,6 +576,11 @@ int main(int argc, char ** argv){ } it->H.Clean(); it->H.SetHeader("Content-Type", "text/javascript"); + it->H.SetHeader("Access-Control-Allow-Origin", "*"); + it->H.SetHeader("Access-Control-Allow-Methods", "GET, POST"); + it->H.SetHeader("Access-Control-Allow-Headers", "Content-Type, X-Requested-With"); + it->H.SetHeader("Access-Control-Allow-Credentials", "true"); + if (jsonp == ""){ it->H.SetBody(Response.toString() + "\n\n"); }else{ @@ -671,15 +595,17 @@ int main(int argc, char ** argv){ } } } - if (!conf.is_active){ + if (!Controller::conf.is_active){ Controller::Log("CONF", "Controller shutting down because of user request (received shutdown signal)"); } if (!API_Socket.connected()){ Controller::Log("CONF", "Controller shutting down because of socket problem (API port closed)"); } + Controller::conf.is_active = false; API_Socket.close(); - if ( !Controller::WriteFile(conf.getString("configFile"), Controller::Storage.toString())){ - std::cerr << "Error writing config " << conf.getString("configFile") << std::endl; + statsThread.join(); + if ( !Controller::WriteFile(Controller::conf.getString("configFile"), Controller::Storage.toString())){ + std::cerr << "Error writing config " << Controller::conf.getString("configFile") << std::endl; Controller::Storage.removeMember("log"); for (JSON::ObjIter it = Controller::Storage["streams"].ObjBegin(); it != Controller::Storage["streams"].ObjEnd(); it++){ it->second.removeMember("meta"); diff --git a/src/controller/controller_connectors.cpp b/src/controller/controller_connectors.cpp index e6ce4bff..f600f653 100644 --- a/src/controller/controller_connectors.cpp +++ b/src/controller/controller_connectors.cpp @@ -1,6 +1,7 @@ #include <stdio.h> // cout, cerr #include <string> #include <cstring> // strcpy +#include <sys/stat.h> //stat #include <mist/json.h> #include <mist/config.h> #include <mist/procs.h> @@ -55,7 +56,14 @@ namespace Controller { static inline void buildPipedArguments(JSON::Value & p, char * argarr[], JSON::Value & capabilities){ int argnum = 0; static std::string tmparg; - tmparg = Util::getMyPath() + std::string("MistConn") + p["connector"].asStringRef(); + tmparg = Util::getMyPath() + std::string("MistOut") + p["connector"].asStringRef(); + struct stat buf; + if (::stat(tmparg.c_str(), &buf) != 0){ + tmparg = Util::getMyPath() + std::string("MistConn") + p["connector"].asStringRef(); + } + if (::stat(tmparg.c_str(), &buf) != 0){ + return; + } argarr[argnum++] = (char*)tmparg.c_str(); argarr[argnum++] = (char*)"-n"; JSON::Value & pipedCapa = capabilities["connectors"][p["connector"].asStringRef()]; diff --git a/src/controller/controller_statistics.cpp b/src/controller/controller_statistics.cpp new file mode 100644 index 00000000..23848904 --- /dev/null +++ b/src/controller/controller_statistics.cpp @@ -0,0 +1,451 @@ +#include <cstdio> +#include <mist/config.h> +#include "controller_statistics.h" + +/// The STAT_CUTOFF define sets how many seconds of statistics history is kept. +#define STAT_CUTOFF 600 + +// These are used to store "clients" field requests in a bitfield for speedup. +#define STAT_CLI_HOST 1 +#define STAT_CLI_STREAM 2 +#define STAT_CLI_PROTO 4 +#define STAT_CLI_CONNTIME 8 +#define STAT_CLI_POSITION 16 +#define STAT_CLI_DOWN 32 +#define STAT_CLI_UP 64 +#define STAT_CLI_BPS_DOWN 128 +#define STAT_CLI_BPS_UP 256 +#define STAT_CLI_ALL 0xFFFF +// These are used to store "totals" field requests in a bitfield for speedup. +#define STAT_TOT_CLIENTS 1 +#define STAT_TOT_BPS_DOWN 2 +#define STAT_TOT_BPS_UP 4 +#define STAT_TOT_ALL 0xFF + + +std::multimap<unsigned long long int, Controller::statStorage> Controller::oldConns;///<Old connections, sorted on disconnect timestamp +std::map<unsigned long, Controller::statStorage> Controller::curConns;///<Connection storage, sorted on page location. + +/// This function runs as a thread and roughly once per second retrieves +/// statistics from all connected clients, as well as wipes +/// old statistics that have disconnected over 10 minutes ago. +void Controller::SharedMemStats(void * config){ + DEBUG_MSG(DLVL_HIGH, "Starting stats thread"); + IPC::sharedServer statServer("statistics", 88, true); + while(((Util::Config*)config)->is_active){ + //parse current users + statServer.parseEach(parseStatistics); + //wipe old statistics + while (oldConns.size() && oldConns.begin()->first < (unsigned long long)(Util::epoch() - STAT_CUTOFF)){ + oldConns.erase(oldConns.begin()); + } + Util::sleep(1000); + } + DEBUG_MSG(DLVL_HIGH, "Stopping stats thread"); +} + +/// This function is called by parseStatistics. +/// It updates the internally saved statistics data. +void Controller::statStorage::update(IPC::statExchange & data) { + if (streamName == ""){ + host = data.host(); + streamName = data.streamName(); + connector = data.connector(); + } + statLog tmp; + tmp.time = data.time(); + tmp.lastSecond = data.lastSecond(); + tmp.down = data.down(); + tmp.up = data.up(); + log[data.now()] = tmp; + //wipe data older than approx. STAT_CUTOFF seconds + if (log.size() > STAT_CUTOFF){ + log.erase(log.begin()); + } +} + +/// This function is called by the shared memory page that holds statistics. +/// It updates the internally saved statistics data, archiving if neccessary. +void Controller::parseStatistics(char * data, size_t len, unsigned int id){ + IPC::statExchange tmpEx(data); + curConns[id].update(tmpEx); + char counter = (*(data - 1)); + if (counter == 126 || counter == 127 || counter == 254 || counter == 255){ + oldConns.insert(std::pair<unsigned long long int, statStorage>(Util::epoch(), curConns[id])); + curConns.erase(id); + } +} + +/// Returns true if this stream has at least one connected client. +bool Controller::hasViewers(std::string streamName){ + if (curConns.size()){ + for (std::map<unsigned long, statStorage>::iterator it = curConns.begin(); it != curConns.end(); it++){ + if (it->second.streamName == streamName){ + return true; + } + } + } + return false; +} + +/// This takes a "clients" request, and fills in the response data. +/// +/// \api +/// `"client"` requests take the form of: +/// ~~~~~~~~~~~~~~~{.js} +/// { +/// //array of streamnames to accumulate. Empty means all. +/// "streams": ["streama", "streamb", "streamc"], +/// //array of protocols to accumulate. Empty means all. +/// "protocols": ["HLS", "HSS"], +/// //list of requested data fields. Empty means all. +/// "fields": ["host", "stream", "protocol", "conntime", "position", "down", "up", "downbps", "upbps"], +/// //unix timestamp of measuring moment. Negative means X seconds ago. Empty means now. +/// "time": 1234567 +/// } +/// ~~~~~~~~~~~~~~~ +/// and are responded to as: +/// ~~~~~~~~~~~~~~~{.js} +/// { +/// //unix timestamp of data. Always present, always absolute. +/// "time": 1234567, +/// //array of actually represented data fields. +/// "fields": [...] +/// //for all clients, the data in the order they appear in the "fields" field. +/// "data": [[x, y, z], [x, y, z], [x, y, z]] +/// } +/// ~~~~~~~~~~~~~~~ +void Controller::fillClients(JSON::Value & req, JSON::Value & rep){ + //first, figure out the timestamp wanted + long long int reqTime = 0; + if (req.isMember("time")){ + reqTime = req["time"].asInt(); + } + //to make sure no nasty timing business takes place, we store the case "now" as a bool. + bool now = (reqTime == 0); + //add the current time, if negative or zero. + if (reqTime <= 0){ + reqTime += Util::epoch(); + } + //at this point, reqTime is the absolute timestamp. + rep["time"] = reqTime; //fill the absolute timestamp + + unsigned int fields = 0; + //next, figure out the fields wanted + if (req.isMember("fields") && req["fields"].size()){ + for (JSON::ArrIter it = req["fields"].ArrBegin(); it != req["fields"].ArrEnd(); it++){ + if ((*it).asStringRef() == "host"){fields |= STAT_CLI_HOST;} + if ((*it).asStringRef() == "stream"){fields |= STAT_CLI_STREAM;} + if ((*it).asStringRef() == "protocol"){fields |= STAT_CLI_PROTO;} + if ((*it).asStringRef() == "conntime"){fields |= STAT_CLI_CONNTIME;} + if ((*it).asStringRef() == "position"){fields |= STAT_CLI_POSITION;} + if ((*it).asStringRef() == "down"){fields |= STAT_CLI_DOWN;} + if ((*it).asStringRef() == "up"){fields |= STAT_CLI_UP;} + if ((*it).asStringRef() == "downbps"){fields |= STAT_CLI_BPS_DOWN;} + if ((*it).asStringRef() == "upbps"){fields |= STAT_CLI_BPS_UP;} + } + } + //select all, if none selected + if (!fields){fields = STAT_CLI_ALL;} + //figure out what streams are wanted + std::set<std::string> streams; + if (req.isMember("streams") && req["streams"].size()){ + for (JSON::ArrIter it = req["streams"].ArrBegin(); it != req["streams"].ArrEnd(); it++){ + streams.insert((*it).asStringRef()); + } + } + //figure out what protocols are wanted + std::set<std::string> protos; + if (req.isMember("protocols") && req["protocols"].size()){ + for (JSON::ArrIter it = req["protocols"].ArrBegin(); it != req["protocols"].ArrEnd(); it++){ + protos.insert((*it).asStringRef()); + } + } + //output the selected fields + rep["fields"].null(); + if (fields & STAT_CLI_HOST){rep["fields"].append("host");} + if (fields & STAT_CLI_STREAM){rep["fields"].append("stream");} + if (fields & STAT_CLI_PROTO){rep["fields"].append("protocol");} + if (fields & STAT_CLI_CONNTIME){rep["fields"].append("conntime");} + if (fields & STAT_CLI_POSITION){rep["fields"].append("position");} + if (fields & STAT_CLI_DOWN){rep["fields"].append("down");} + if (fields & STAT_CLI_UP){rep["fields"].append("up");} + if (fields & STAT_CLI_BPS_DOWN){rep["fields"].append("downbps");} + if (fields & STAT_CLI_BPS_UP){rep["fields"].append("upbps");} + //output the data itself + rep["data"].null(); + //start with current connections + if (curConns.size()){ + for (std::map<unsigned long, statStorage>::iterator it = curConns.begin(); it != curConns.end(); it++){ + unsigned long long time = reqTime; + if (now){time = it->second.log.rbegin()->first;} + //data present and wanted? insert it! + if ((it->second.log.rbegin()->first >= time && it->second.log.begin()->first <= time) && (!streams.size() || streams.count(it->second.streamName)) && (!protos.size() || protos.count(it->second.connector))){ + JSON::Value d; + std::map<unsigned long long, statLog>::iterator statRef = it->second.log.lower_bound(time); + std::map<unsigned long long, statLog>::iterator prevRef = --(it->second.log.lower_bound(time)); + if (fields & STAT_CLI_HOST){d.append(it->second.host);} + if (fields & STAT_CLI_STREAM){d.append(it->second.streamName);} + if (fields & STAT_CLI_PROTO){d.append(it->second.connector);} + if (fields & STAT_CLI_CONNTIME){d.append((long long)statRef->second.time);} + if (fields & STAT_CLI_POSITION){d.append((long long)statRef->second.lastSecond);} + if (fields & STAT_CLI_DOWN){d.append(statRef->second.down);} + if (fields & STAT_CLI_UP){d.append(statRef->second.up);} + if (fields & STAT_CLI_BPS_DOWN){ + if (statRef != it->second.log.begin()){ + unsigned int diff = statRef->first - prevRef->first; + d.append((statRef->second.down - prevRef->second.down) / diff); + }else{ + d.append(statRef->second.down); + } + } + if (fields & STAT_CLI_BPS_UP){ + if (statRef != it->second.log.begin()){ + unsigned int diff = statRef->first - prevRef->first; + d.append((statRef->second.up - prevRef->second.up) / diff); + }else{ + d.append(statRef->second.up); + } + } + rep["data"].append(d); + } + } + } + //if we're only interested in current, don't even bother looking at history + if (now){ + return; + } + //look at history + if (oldConns.size()){ + for (std::map<unsigned long long int, statStorage>::iterator it = oldConns.begin(); it != oldConns.end(); it++){ + //data present and wanted? insert it! + if ((it->second.log.rbegin()->first >= (unsigned long long)reqTime && it->second.log.begin()->first <= (unsigned long long)reqTime) && (!streams.size() || streams.count(it->second.streamName)) && (!protos.size() || protos.count(it->second.connector))){ + JSON::Value d; + std::map<unsigned long long, statLog>::iterator statRef = it->second.log.lower_bound(reqTime); + std::map<unsigned long long, statLog>::iterator prevRef = --(it->second.log.lower_bound(reqTime)); + if (fields & STAT_CLI_HOST){d.append(it->second.host);} + if (fields & STAT_CLI_STREAM){d.append(it->second.streamName);} + if (fields & STAT_CLI_PROTO){d.append(it->second.connector);} + if (fields & STAT_CLI_CONNTIME){d.append((long long)statRef->second.time);} + if (fields & STAT_CLI_POSITION){d.append((long long)statRef->second.lastSecond);} + if (fields & STAT_CLI_DOWN){d.append(statRef->second.down);} + if (fields & STAT_CLI_UP){d.append(statRef->second.up);} + if (fields & STAT_CLI_BPS_DOWN){ + if (statRef != it->second.log.begin()){ + unsigned int diff = statRef->first - prevRef->first; + d.append((statRef->second.down - prevRef->second.down) / diff); + }else{ + d.append(statRef->second.down); + } + } + if (fields & STAT_CLI_BPS_UP){ + if (statRef != it->second.log.begin()){ + unsigned int diff = statRef->first - prevRef->first; + d.append((statRef->second.up - prevRef->second.up) / diff); + }else{ + d.append(statRef->second.up); + } + } + rep["data"].append(d); + } + } + } + //all done! return is by reference, so no need to return anything here. +} + +class totalsData { + public: + totalsData(){ + clients = 0; + downbps = 0; + upbps = 0; + } + void add(unsigned int down, unsigned int up){ + clients++; + downbps += down; + upbps += up; + } + long long clients; + long long downbps; + long long upbps; +}; + +/// This takes a "totals" request, and fills in the response data. +/// +/// \api +/// `"totals"` requests take the form of: +/// ~~~~~~~~~~~~~~~{.js} +/// { +/// //array of streamnames to accumulate. Empty means all. +/// "streams": ["streama", "streamb", "streamc"], +/// //array of protocols to accumulate. Empty means all. +/// "protocols": ["HLS", "HSS"], +/// //list of requested data fields. Empty means all. +/// "fields": ["clients", "downbps", "upbps"], +/// //unix timestamp of data start. Negative means X seconds ago. Empty means earliest available. +/// "start": 1234567 +/// //unix timestamp of data end. Negative means X seconds ago. Empty means latest available (usually 'now'). +/// "end": 1234567 +/// } +/// ~~~~~~~~~~~~~~~ +/// and are responded to as: +/// ~~~~~~~~~~~~~~~{.js} +/// { +/// //unix timestamp of start of data. Always present, always absolute. +/// "start": 1234567, +/// //unix timestamp of end of data. Always present, always absolute. +/// "end": 1234567, +/// //array of actually represented data fields. +/// "fields": [...] +/// // Time between datapoints. Here: 10 points with each 5 seconds afterwards, followed by 10 points with each 1 second afterwards. +/// "interval": [[10, 5], [10, 1]], +/// //the data for the times as mentioned in the "interval" field, in the order they appear in the "fields" field. +/// "data": [[x, y, z], [x, y, z], [x, y, z]] +/// } +/// ~~~~~~~~~~~~~~~ +void Controller::fillTotals(JSON::Value & req, JSON::Value & rep){ + //first, figure out the timestamps wanted + long long int reqStart = 0; + long long int reqEnd = 0; + if (req.isMember("start")){ + reqStart = req["start"].asInt(); + } + if (req.isMember("end")){ + reqEnd = req["end"].asInt(); + } + //add the current time, if negative or zero. + if (reqStart < 0){ + reqStart += Util::epoch(); + } + if (reqStart == 0){ + reqStart = Util::epoch() - STAT_CUTOFF; + } + if (reqEnd <= 0){ + reqEnd += Util::epoch(); + } + //at this point, reqStart and reqEnd are the absolute timestamp. + + unsigned int fields = 0; + //next, figure out the fields wanted + if (req.isMember("fields") && req["fields"].size()){ + for (JSON::ArrIter it = req["fields"].ArrBegin(); it != req["fields"].ArrEnd(); it++){ + if ((*it).asStringRef() == "clients"){fields |= STAT_TOT_CLIENTS;} + if ((*it).asStringRef() == "downbps"){fields |= STAT_TOT_BPS_DOWN;} + if ((*it).asStringRef() == "upbps"){fields |= STAT_TOT_BPS_UP;} + } + } + //select all, if none selected + if (!fields){fields = STAT_TOT_ALL;} + //figure out what streams are wanted + std::set<std::string> streams; + if (req.isMember("streams") && req["streams"].size()){ + for (JSON::ArrIter it = req["streams"].ArrBegin(); it != req["streams"].ArrEnd(); it++){ + streams.insert((*it).asStringRef()); + } + } + //figure out what protocols are wanted + std::set<std::string> protos; + if (req.isMember("protocols") && req["protocols"].size()){ + for (JSON::ArrIter it = req["protocols"].ArrBegin(); it != req["protocols"].ArrEnd(); it++){ + protos.insert((*it).asStringRef()); + } + } + //output the selected fields + rep["fields"].null(); + if (fields & STAT_TOT_CLIENTS){rep["fields"].append("clients");} + if (fields & STAT_TOT_BPS_DOWN){rep["fields"].append("downbps");} + if (fields & STAT_TOT_BPS_UP){rep["fields"].append("upbps");} + //start data collection + std::map<long long unsigned int, totalsData> totalsCount; + //start with current connections + if (curConns.size()){ + for (std::map<unsigned long, statStorage>::iterator it = curConns.begin(); it != curConns.end(); it++){ + //data present and wanted? insert it! + if (it->second.log.size() > 1 && (it->second.log.rbegin()->first >= (unsigned long long)reqStart || it->second.log.begin()->first <= (unsigned long long)reqEnd) && (!streams.size() || streams.count(it->second.streamName)) && (!protos.size() || protos.count(it->second.connector))){ + //keep track of the previous and current, starting at position 2 so there's always a delta down/up value. + std::map<unsigned long long, statLog>::iterator pi = it->second.log.begin(); + for (std::map<unsigned long long, statLog>::iterator li = ++(it->second.log.begin()); li != it->second.log.end(); li++){ + if (li->first < (unsigned long long)reqStart || pi->first > (unsigned long long)reqEnd){ + continue; + } + unsigned int diff = li->first - pi->first; + unsigned int ddown = (li->second.down - pi->second.down) / diff; + unsigned int dup = (li->second.up - pi->second.up) / diff; + for (long long unsigned int t = pi->first; t < li->first; t++){ + if (t >= (unsigned long long)reqStart && t <= (unsigned long long)reqEnd){ + totalsCount[t].add(ddown, dup); + } + } + pi = li;//set previous iterator to log iterator + } + } + } + } + //look at history + if (oldConns.size()){ + for (std::map<unsigned long long int, statStorage>::iterator it = oldConns.begin(); it != oldConns.end(); it++){ + //data present and wanted? insert it! + if (it->second.log.size() > 1 && (it->second.log.rbegin()->first >= (unsigned long long)reqStart || it->second.log.begin()->first <= (unsigned long long)reqEnd) && (!streams.size() || streams.count(it->second.streamName)) && (!protos.size() || protos.count(it->second.connector))){ + //keep track of the previous and current, starting at position 2 so there's always a delta down/up value. + std::map<unsigned long long, statLog>::iterator pi = it->second.log.begin(); + for (std::map<unsigned long long, statLog>::iterator li = ++(it->second.log.begin()); li != it->second.log.end(); li++){ + if (li->first < (unsigned long long)reqStart || pi->first > (unsigned long long)reqEnd){ + continue; + } + unsigned int diff = li->first - pi->first; + unsigned int ddown = (li->second.down - pi->second.down) / diff; + unsigned int dup = (li->second.up - pi->second.up) / diff; + for (long long unsigned int t = pi->first; t < li->first; t++){ + if (t >= (unsigned long long)reqStart && t <= (unsigned long long)reqEnd){ + totalsCount[t].add(ddown, dup); + } + } + pi = li;//set previous iterator to log iterator + } + } + } + } + //output the data itself + if (!totalsCount.size()){ + //Oh noes! No data. We'll just reply with a bunch of nulls. + rep["start"].null(); + rep["end"].null(); + rep["data"].null(); + rep["interval"].null(); + return; + } + //yay! We have data! + rep["start"] = (long long)totalsCount.begin()->first; + rep["end"] = (long long)totalsCount.rbegin()->first; + rep["data"].null(); + rep["interval"].null(); + long long prevT = 0; + JSON::Value i; + for (std::map<long long unsigned int, totalsData>::iterator it = totalsCount.begin(); it != totalsCount.end(); it++){ + JSON::Value d; + if (fields & STAT_TOT_CLIENTS){d.append(it->second.clients);} + if (fields & STAT_TOT_BPS_DOWN){d.append(it->second.downbps);} + if (fields & STAT_TOT_BPS_UP){d.append(it->second.upbps);} + rep["data"].append(d); + if (prevT){ + if (i.size() < 2){ + i.append(1ll); + i.append((long long)(it->first - prevT)); + }else{ + if (i[1u].asInt() != (long long)(it->first - prevT)){ + rep["interval"].append(i); + i[0u] = 1ll; + i[1u] = (long long)(it->first - prevT); + }else{ + i[0u] = i[0u].asInt() + 1; + } + } + } + prevT = it->first; + } + if (i.size() > 1){ + rep["interval"].append(i); + i.null(); + } + //all done! return is by reference, so no need to return anything here. +} diff --git a/src/controller/controller_statistics.h b/src/controller/controller_statistics.h new file mode 100644 index 00000000..5671fbb5 --- /dev/null +++ b/src/controller/controller_statistics.h @@ -0,0 +1,35 @@ +#include <mist/shared_memory.h> +#include <mist/timing.h> +#include <mist/defines.h> +#include <mist/json.h> +#include <string> +#include <map> + + +namespace Controller { + struct statLog { + long time; + long lastSecond; + long long down; + long long up; + }; + + class statStorage { + public: + void update(IPC::statExchange & data); + std::string host; + std::string streamName; + std::string connector; + std::map<unsigned long long, statLog> log; + }; + + + extern std::multimap<unsigned long long int, statStorage> oldConns; + extern std::map<unsigned long, statStorage> curConns; + void parseStatistics(char * data, size_t len, unsigned int id); + void fillClients(JSON::Value & req, JSON::Value & rep); + void fillTotals(JSON::Value & req, JSON::Value & rep); + void SharedMemStats(void * config); + bool hasViewers(std::string streamName); +} + diff --git a/src/controller/controller_streams.cpp b/src/controller/controller_streams.cpp index 1dd73f07..ccceb5ea 100644 --- a/src/controller/controller_streams.cpp +++ b/src/controller/controller_streams.cpp @@ -3,16 +3,17 @@ #include <mist/timing.h> #include <mist/stream.h> #include <mist/dtsc.h> +#include <mist/defines.h> +#include <mist/shared_memory.h> #include "controller_streams.h" #include "controller_storage.h" +#include "controller_statistics.h" #include <sys/stat.h> #include <map> ///\brief Holds everything unique to the controller. namespace Controller { - std::map<std::string, int> lastBuffer; ///< Last moment of contact with all buffers. - ///\brief Checks whether two streams are equal. ///\param one The first stream for the comparison. ///\param two The second stream for the comparison. @@ -43,22 +44,28 @@ namespace Controller { if (data.isMember("source")){ URL = data["source"].asString(); } - std::string buffcmd; if (URL == ""){ Log("STRM", "Error for stream " + name + "! Source parameter missing."); data["error"] = "Stream offline: Missing source parameter!"; return; } - buffcmd = "MistBuffer"; - if (data.isMember("DVR") && data["DVR"].asInt() > 0){ - data["DVR"] = data["DVR"].asInt(); - buffcmd += " -t " + data["DVR"].asString(); - } - buffcmd += " -s " + name; if (URL.substr(0, 4) == "push"){ - std::string pusher = URL.substr(7); - Util::Procs::Start(name, Util::getMyPath() + buffcmd + " " + pusher); - Log("BUFF", "(re)starting stream buffer " + name + " for push data from " + pusher); + if (hasViewers(name)){ + data["meta"].null(); + IPC::sharedPage streamIndex(name,0,false,false); + if (!streamIndex.mapped){ + return; + } + unsigned int i = 0; + JSON::fromDTMI((const unsigned char*)streamIndex.mapped + 8, streamIndex.len - 8, i, data["meta"]); + if (data["meta"].isMember("tracks") && data["meta"]["tracks"].size()){ + for(JSON::ObjIter trackIt = data["meta"]["tracks"].ObjBegin(); trackIt != data["meta"]["tracks"].ObjEnd(); trackIt++){ + trackIt->second.removeMember("fragments"); + trackIt->second.removeMember("keys"); + trackIt->second.removeMember("parts"); + } + } + } }else{ if (URL.substr(0, 1) == "/"){ data.removeMember("error"); @@ -74,6 +81,12 @@ namespace Controller { getMeta = true; data["l_meta"] = (long long)fileinfo.st_mtime; } + if (stat((URL+".dtsh").c_str(), &fileinfo) == 0 && !S_ISDIR(fileinfo.st_mode)){ + if ( !data.isMember("h_meta") || fileinfo.st_mtime != data["h_meta"].asInt()){ + getMeta = true; + data["h_meta"] = (long long)fileinfo.st_mtime; + } + } if ( !getMeta && data.isMember("meta") && data["meta"].isMember("tracks")){ for (JSON::ObjIter trIt = data["meta"]["tracks"].ObjBegin(); trIt != data["meta"]["tracks"].ObjEnd(); trIt++){ if (trIt->second["codec"] == "H264"){ @@ -107,6 +120,9 @@ namespace Controller { getMeta = true; } if (getMeta){ + if ((URL.substr(URL.size() - 5) != ".dtsc") && (stat((URL+".dtsh").c_str(), &fileinfo) != 0)){ + Util::Stream::getStream(name); + } char * tmp_cmd[3] = {0, 0, 0}; std::string mistinfo = Util::getMyPath() + "MistInfo"; tmp_cmd[0] = (char*)mistinfo.c_str(); @@ -127,7 +143,7 @@ namespace Controller { Util::Procs::getOutputOf(tmp_cmd); data.removeMember("meta"); } - if (Util::epoch() - lastBuffer[name] > 5){ + if (!hasViewers(name)){ if ( !data.isMember("error")){ data["error"] = "Available"; } @@ -136,9 +152,11 @@ namespace Controller { data["online"] = 1; } return; //MistPlayer handles VoD + }else{ + /// \todo Implement ffmpeg pulling again? + //Util::Procs::Start(name, "ffmpeg -re -async 2 -i " + URL + " -f flv -", Util::getMyPath() + "MistFLV2DTSC", Util::getMyPath() + buffcmd); + //Log("BUFF", "(re)starting stream buffer " + name + " for ffmpeg data: ffmpeg -re -async 2 -i " + URL + " -f flv -"); } - Util::Procs::Start(name, "ffmpeg -re -async 2 -i " + URL + " -f flv -", Util::getMyPath() + "MistFLV2DTSC", Util::getMyPath() + buffcmd); - Log("BUFF", "(re)starting stream buffer " + name + " for ffmpeg data: ffmpeg -re -async 2 -i " + URL + " -f flv -"); } } @@ -153,7 +171,7 @@ namespace Controller { if (!jit->second.isMember("name")){ jit->second["name"] = jit->first; } - if (currTime - lastBuffer[jit->first] > 5){ + if (!hasViewers(jit->first)){ if (jit->second.isMember("source") && jit->second["source"].asString().substr(0, 1) == "/" && jit->second.isMember("error") && jit->second["error"].asString().substr(0,15) != "Stream offline:"){ jit->second["online"] = 2; @@ -229,11 +247,8 @@ namespace Controller { WriteFile(Util::getTmpFolder() + "streamlist", strlist.toString()); } } - - ///\brief Parse a given stream configuration. - ///\param in The requested configuration. - ///\param out The new configuration after parsing. - void CheckStreams(JSON::Value & in, JSON::Value & out){ + + void AddStreams(JSON::Value & in, JSON::Value & out){ //check for new streams and updates for (JSON::ObjIter jit = in.ObjBegin(); jit != in.ObjEnd(); jit++){ if (out.isMember(jit->first)){ @@ -263,6 +278,14 @@ namespace Controller { startStream(jit->first, out[jit->first]); } } + } + + ///\brief Parse a given stream configuration. + ///\param in The requested configuration. + ///\param out The new configuration after parsing. + void CheckStreams(JSON::Value & in, JSON::Value & out){ + //check for new streams and updates + AddStreams(in, out); //check for deleted streams std::set<std::string> toDelete; diff --git a/src/controller/controller_streams.h b/src/controller/controller_streams.h index fe763960..48e90ef6 100644 --- a/src/controller/controller_streams.h +++ b/src/controller/controller_streams.h @@ -1,13 +1,12 @@ #include <mist/json.h> namespace Controller { - extern std::map<std::string, int> lastBuffer; ///< Last moment of contact with all buffers. - bool streamsEqual(JSON::Value & one, JSON::Value & two); void startStream(std::string name, JSON::Value & data); void CheckAllStreams(JSON::Value & data); void CheckStreams(JSON::Value & in, JSON::Value & out); - + void AddStreams(JSON::Value & in, JSON::Value & out); + struct liveCheck { long long int lastms; long long int last_active; diff --git a/src/converters/dtsc2srt.cpp b/src/converters/dtsc2srt.cpp index 82eb8a37..8a92f480 100644 --- a/src/converters/dtsc2srt.cpp +++ b/src/converters/dtsc2srt.cpp @@ -19,19 +19,21 @@ namespace Converters { int curIndex = 1; F.parseNext(); - while ( !F.getJSON().isNull()){ + std::string tmp; + while (F.getPacket()){ std::cout << curIndex++ << std::endl; - long long unsigned int time = F.getJSON()["time"].asInt(); + long long unsigned int time = F.getPacket().getTime(); std::cout << std::setfill('0') << std::setw(2) << (time / 3600000) << ":"; std::cout << std::setfill('0') << std::setw(2) << ((time % 3600000) / 60000) << ":"; std::cout << std::setfill('0') << std::setw(2) << (((time % 3600000) % 60000) / 1000) << ","; std::cout << std::setfill('0') << std::setw(3) << time % 1000 << " --> "; - time += F.getJSON()["duration"].asInt(); + time += F.getPacket().getInt("duration"); std::cout << std::setfill('0') << std::setw(2) << (time / 3600000) << ":"; std::cout << std::setfill('0') << std::setw(2) << ((time % 3600000) / 60000) << ":"; std::cout << std::setfill('0') << std::setw(2) << (((time % 3600000) % 60000) / 1000) << ","; std::cout << std::setfill('0') << std::setw(3) << time % 1000 << std::endl; - std::cout << F.getJSON()["data"].asString() << std::endl; + F.getPacket().getString("data", tmp); + std::cout << tmp << std::endl; F.parseNext(); } return 0; diff --git a/src/converters/dtscfix.cpp b/src/converters/dtscfix.cpp index 598fa5fc..c24cad66 100644 --- a/src/converters/dtscfix.cpp +++ b/src/converters/dtscfix.cpp @@ -15,7 +15,7 @@ namespace Converters { DTSC::File F(conf.getString("filename")); F.seek_bpos(0); F.parseNext(); - JSON::Value oriheader = F.getJSON(); + JSON::Value oriheader = F.getPacket().toJSON(); DTSC::Meta meta(F.getMeta()); if (meta.isFixed() && !conf.getBool("force")){ @@ -26,9 +26,11 @@ namespace Converters { meta.reset(); int bPos = F.getBytePos(); F.parseNext(); - while ( !F.getJSON().isNull()){ - F.getJSON()["bpos"] = bPos; - meta.update(F.getJSON()); + JSON::Value newPack; + while ( F.getPacket()){ + newPack = F.getPacket().toJSON(); + newPack["bpos"] = bPos; + meta.update(newPack); bPos = F.getBytePos(); F.parseNext(); } diff --git a/src/converters/dtscmerge.cpp b/src/converters/dtscmerge.cpp index b17b2338..41a088db 100644 --- a/src/converters/dtscmerge.cpp +++ b/src/converters/dtscmerge.cpp @@ -128,10 +128,11 @@ namespace Converters { inFiles[sortIt->second.fileName].selectTracks(trackSelector); inFiles[sortIt->second.fileName].seek_time(sortIt->second.keyTime); inFiles[sortIt->second.fileName].seekNext(); - while (inFiles[sortIt->second.fileName].getJSON() && inFiles[sortIt->second.fileName].getBytePos() <= sortIt->second.endBPos && !inFiles[sortIt->second.fileName].reachedEOF()){ - if (inFiles[sortIt->second.fileName].getJSON()["trackid"].asInt() == sortIt->second.trackID){ - inFiles[sortIt->second.fileName].getJSON()["trackid"] = trackMapping[sortIt->second.fileName][sortIt->second.trackID]; - outFile.writePacket(inFiles[sortIt->second.fileName].getJSON()); + while (inFiles[sortIt->second.fileName].getPacket() && inFiles[sortIt->second.fileName].getBytePos() <= sortIt->second.endBPos && !inFiles[sortIt->second.fileName].reachedEOF()){ + if (inFiles[sortIt->second.fileName].getPacket().getTrackId() == sortIt->second.trackID){ + JSON::Value tmp = inFiles[sortIt->second.fileName].getPacket().toJSON(); + tmp["trackid"] = trackMapping[sortIt->second.fileName][sortIt->second.trackID]; + outFile.writePacket(tmp); } inFiles[sortIt->second.fileName].seekNext(); } diff --git a/src/converters/oggconv.cpp b/src/converters/oggconv.cpp index 40d8daea..15fe331f 100644 --- a/src/converters/oggconv.cpp +++ b/src/converters/oggconv.cpp @@ -1,4 +1,4 @@ -#include"oggconv.h" +#include "oggconv.h" #include <stdlib.h> #include <mist/bitstream.h> @@ -11,32 +11,61 @@ namespace OGG{ srand (Util::getMS());//randomising with milliseconds from boot std::vector<unsigned int> curSegTable; //trackInf.clear(); - //Creating ID headers for theora and vorbis + /// \todo This is utter rubbish right now. + /// \todo We shouldn't assume all possible tracks are selected. + /// \todo We shouldn't be buffering, but sending. + /// \todo Especially not in a std::string. (Why, god, why?!) + //Creating headers for ( std::map<int,DTSC::Track>::iterator it = meta.tracks.begin(); it != meta.tracks.end(); it ++) { - curOggPage.clear(); - curOggPage.setVersion(); - curOggPage.setHeaderType(2);//headertype 2 = Begin of Stream - curOggPage.setGranulePosition(0); - trackInf[it->second.trackID].OGGSerial = rand() % 0xFFFFFFFE +1; //initialising on a random not 0 number - curOggPage.setBitstreamSerialNumber(trackInf[it->second.trackID].OGGSerial); - trackInf[it->second.trackID].seqNum = 0; - curOggPage.setPageSequenceNumber(trackInf[it->second.trackID].seqNum++); - curSegTable.clear(); - curSegTable.push_back(it->second.idHeader.size()); - curOggPage.setSegmentTable(curSegTable); - curOggPage.setPayload((char*)it->second.idHeader.c_str(), it->second.idHeader.size()); - curOggPage.setCRCChecksum(curOggPage.calcChecksum()); - //pages.push_back(curOggPage); - parsedPages += std::string(curOggPage.getPage(), curOggPage.getPageSize()); trackInf[it->second.trackID].codec = it->second.codec; + trackInf[it->second.trackID].OGGSerial = rand() % 0xFFFFFFFE +1; //initialising on a random not 0 number + trackInf[it->second.trackID].seqNum = 0; if (it->second.codec == "theora"){ + curOggPage.clear(); + curOggPage.setVersion(); + curOggPage.setHeaderType(2);//headertype 2 = Begin of Stream + curOggPage.setGranulePosition(0); + curOggPage.setBitstreamSerialNumber(trackInf[it->second.trackID].OGGSerial); + curOggPage.setPageSequenceNumber(trackInf[it->second.trackID].seqNum++); + curSegTable.clear(); + curSegTable.push_back(it->second.idHeader.size()); + curOggPage.setSegmentTable(curSegTable); + curOggPage.setPayload((char*)it->second.idHeader.c_str(), it->second.idHeader.size()); + curOggPage.setCRCChecksum(curOggPage.calcChecksum()); + parsedPages += std::string(curOggPage.getPage(), curOggPage.getPageSize()); trackInf[it->second.trackID].lastKeyFrame = 1; trackInf[it->second.trackID].sinceKeyFrame = 0; theora::header tempHead; std::string tempString = it->second.idHeader; tempHead.read((char*)tempString.c_str(),42); trackInf[it->second.trackID].significantValue = tempHead.getKFGShift(); + curOggPage.clear(); + curOggPage.setVersion(); + curOggPage.setHeaderType(0);//headertype 0 = normal + curOggPage.setGranulePosition(0); + curOggPage.setBitstreamSerialNumber(trackInf[it->second.trackID].OGGSerial); + curOggPage.setPageSequenceNumber(trackInf[it->second.trackID].seqNum++); + curSegTable.clear(); + curSegTable.push_back(it->second.commentHeader.size()); + curSegTable.push_back(it->second.init.size()); + curOggPage.setSegmentTable(curSegTable); + std::string fullHeader = it->second.commentHeader + it->second.init; + curOggPage.setPayload((char*)fullHeader.c_str(),fullHeader.size()); + curOggPage.setCRCChecksum(curOggPage.calcChecksum()); + parsedPages += std::string(curOggPage.getPage(), curOggPage.getPageSize()); }else if (it->second.codec == "vorbis"){ + curOggPage.clear(); + curOggPage.setVersion(); + curOggPage.setHeaderType(2);//headertype 2 = Begin of Stream + curOggPage.setGranulePosition(0); + curOggPage.setBitstreamSerialNumber(trackInf[it->second.trackID].OGGSerial); + curOggPage.setPageSequenceNumber(trackInf[it->second.trackID].seqNum++); + curSegTable.clear(); + curSegTable.push_back(it->second.idHeader.size()); + curOggPage.setSegmentTable(curSegTable); + curOggPage.setPayload((char*)it->second.idHeader.c_str(), it->second.idHeader.size()); + curOggPage.setCRCChecksum(curOggPage.calcChecksum()); + parsedPages += std::string(curOggPage.getPage(), curOggPage.getPageSize()); trackInf[it->second.trackID].lastKeyFrame = 0; trackInf[it->second.trackID].sinceKeyFrame = 0; trackInf[it->second.trackID].prevBlockFlag = -1; @@ -57,27 +86,53 @@ namespace OGG{ tempHead.read((char*)tempString.c_str(),tempString.size()); trackInf[it->second.trackID].vorbisModes = tempHead.readModeDeque(audioChannels); trackInf[it->second.trackID].hadFirst = false; + curOggPage.clear(); + curOggPage.setVersion(); + curOggPage.setHeaderType(0);//headertype 0 = normal + curOggPage.setGranulePosition(0); + curOggPage.setBitstreamSerialNumber(trackInf[it->second.trackID].OGGSerial); + curOggPage.setPageSequenceNumber(trackInf[it->second.trackID].seqNum++); + curSegTable.clear(); + curSegTable.push_back(it->second.commentHeader.size()); + curSegTable.push_back(it->second.init.size()); + curOggPage.setSegmentTable(curSegTable); + std::string fullHeader = it->second.commentHeader + it->second.init; + curOggPage.setPayload((char*)fullHeader.c_str(),fullHeader.size()); + curOggPage.setCRCChecksum(curOggPage.calcChecksum()); + parsedPages += std::string(curOggPage.getPage(), curOggPage.getPageSize()); + }else if (it->second.codec == "opus"){ + //OpusHead page + curOggPage.clear(); + curOggPage.setVersion(); + curOggPage.setHeaderType(2);//headertype 2 = Begin of Stream + curOggPage.setGranulePosition(0); + curOggPage.setBitstreamSerialNumber(trackInf[it->second.trackID].OGGSerial); + curOggPage.setPageSequenceNumber(trackInf[it->second.trackID].seqNum++); + curSegTable.clear(); + curSegTable.push_back(19); + curOggPage.setSegmentTable(curSegTable); + //version = 1, channels = 2, preskip=0x138, origRate=48k, gain=0, channelmap=0 + //we can safely hard-code these as everything is already overridden elsewhere anyway + // (except preskip - but this seems to be 0x138 for all files, and doesn't hurt much if it's wrong anyway) + curOggPage.setPayload((char*)"OpusHead\001\002\070\001\200\273\000\000\000\000\000", 19); + curOggPage.setCRCChecksum(curOggPage.calcChecksum()); + parsedPages += std::string(curOggPage.getPage(), curOggPage.getPageSize()); + //end of OpusHead, now moving on to OpusTags + curOggPage.clear(); + curOggPage.setVersion(); + curOggPage.setHeaderType(2);//headertype 2 = Begin of Stream + curOggPage.setGranulePosition(0); + curOggPage.setBitstreamSerialNumber(trackInf[it->second.trackID].OGGSerial); + curOggPage.setPageSequenceNumber(trackInf[it->second.trackID].seqNum++); + curSegTable.clear(); + curSegTable.push_back(26); + curOggPage.setSegmentTable(curSegTable); + //we send an encoder value of "MistServer" and no further tags + curOggPage.setPayload((char*)"OpusTags\012\000\000\000MistServer\000\000\000\000", 26); + curOggPage.setCRCChecksum(curOggPage.calcChecksum()); + parsedPages += std::string(curOggPage.getPage(), curOggPage.getPageSize()); } } - //Creating remaining headers for theora and vorbis - //for tracks in header - //create standard page with comment (empty) en setup header(init) - for ( std::map<int,DTSC::Track>::iterator it = meta.tracks.begin(); it != meta.tracks.end(); it ++) { - curOggPage.clear(); - curOggPage.setVersion(); - curOggPage.setHeaderType(0);//headertype 0 = normal - curOggPage.setGranulePosition(0); - curOggPage.setBitstreamSerialNumber(trackInf[it->second.trackID].OGGSerial); - curOggPage.setPageSequenceNumber(trackInf[it->second.trackID].seqNum++); - curSegTable.clear(); - curSegTable.push_back(it->second.commentHeader.size()); - curSegTable.push_back(it->second.init.size()); - curOggPage.setSegmentTable(curSegTable); - std::string fullHeader = it->second.commentHeader + it->second.init; - curOggPage.setPayload((char*)fullHeader.c_str(),fullHeader.size()); - curOggPage.setCRCChecksum(curOggPage.calcChecksum()); - parsedPages += std::string(curOggPage.getPage(), curOggPage.getPageSize()); - } } void converter::readDTSCVector(JSON::Value & DTSCPart, std::string & pageBuffer){ @@ -174,6 +229,8 @@ namespace OGG{ //add to granule position trackInf[DTSCID].lastKeyFrame += curPCMSamples; lastGran = trackInf[DTSCID].lastKeyFrame; + } else if (trackInf[DTSCID].codec == "opus"){ + lastGran = (int)((DTSCPart["time"].asInt() * 48.0) / 120.0 + 0.5) * 120; } //} //last parts of page put out diff --git a/src/input/input.cpp b/src/input/input.cpp new file mode 100644 index 00000000..240790ba --- /dev/null +++ b/src/input/input.cpp @@ -0,0 +1,338 @@ +#include <semaphore.h> +#include <fcntl.h> +#include <sys/stat.h> + +#include <mist/defines.h> +#include "input.h" +#include <sstream> +#include <fstream> +#include <iterator> + +namespace Mist { + Input * Input::singleton = NULL; + + void Input::userCallback(char * data, size_t len, unsigned int id){ + long tid = ((long)(data[0]) << 24) | ((long)(data[1]) << 16) | ((long)(data[2]) << 8) | ((long)(data[3])); + long keyNum = ((long)(data[4]) << 8) | ((long)(data[5])); + bufferFrame(tid, keyNum + 1);//Try buffer next frame + } + + void Input::doNothing(char * data, size_t len, unsigned int id){ + DEBUG_MSG(DLVL_DONTEVEN, "Doing 'nothing'"); + for (int i = 0; i < 5; i++){ + int tmp = ((long)(data[i*6]) << 24) | ((long)(data[i*6 + 1]) << 16) | ((long)(data[i*6 + 2]) << 8) | data[i*6 + 3]; + if (tmp){ + singleton->userCallback(data + (i*6), 6, id);//call the userCallback for this input + } + } + } + + Input::Input(Util::Config * cfg) { + config = cfg; + JSON::Value option; + option["long"] = "json"; + option["short"] = "j"; + option["help"] = "Output MistIn info in JSON format, then exit."; + option["value"].append(0ll); + config->addOption("json", option); + option.null(); + option["arg_num"] = 1ll; + option["arg"] = "string"; + option["help"] = "Name of the input file or - for stdin"; + option["value"].append("-"); + config->addOption("input", option); + option.null(); + option["arg_num"] = 2ll; + option["arg"] = "string"; + option["help"] = "Name of the output file or - for stdout"; + option["value"].append("-"); + config->addOption("output", option); + option.null(); + option["arg"] = "string"; + option["short"] = "s"; + option["long"] = "stream"; + option["help"] = "The name of the stream that this connector will transmit."; + config->addOption("streamname", option); + option.null(); + option["short"] = "p"; + option["long"] = "player"; + option["help"] = "Makes this connector into a player"; + config->addOption("player", option); + + packTime = 0; + lastActive = Util::epoch(); + playing = 0; + playUntil = 0; + + singleton = this; + isBuffer = false; + } + + int Input::run() { + if (config->getBool("json")) { + std::cerr << capa.toString() << std::endl; + return 0; + } + if (!setup()) { + std::cerr << config->getString("cmd") << " setup failed." << std::endl; + return 0; + } + if (!readHeader()) { + std::cerr << "Reading header for " << config->getString("input") << " failed." << std::endl; + return 0; + } + parseHeader(); + + if (!config->getBool("player")){ + //check filename for no - + if (config->getString("output") != "-"){ + //output to dtsc + DTSC::Meta newMeta = myMeta; + newMeta.reset(); + JSON::Value tempVal; + std::ofstream file(config->getString("output").c_str()); + long long int bpos = 0; + seek(0); + getNext(); + while (lastPack){ + tempVal = lastPack.toJSON(); + tempVal["bpos"] = bpos; + newMeta.update(tempVal); + file << std::string(lastPack.getData(), lastPack.getDataLen()); + bpos += lastPack.getDataLen(); + getNext(); + } + //close file + file.close(); + //create header + file.open((config->getString("output")+".dtsh").c_str()); + file << newMeta.toJSON().toNetPacked(); + file.close(); + }else{ + DEBUG_MSG(DLVL_FAIL,"No filename specified, exiting"); + } + }else{ + //after this player functionality + + metaPage.init(config->getString("streamname"), (isBuffer ? 8388608 : myMeta.getSendLen()), true); + myMeta.writeTo(metaPage.mapped); + userPage.init(config->getString("streamname") + "_users", 30, true); + + + if (!isBuffer){ + for (std::map<int,DTSC::Track>::iterator it = myMeta.tracks.begin(); it != myMeta.tracks.end(); it++){ + bufferFrame(it->first, 0); + } + } + + sem_t * waiting = sem_open(std::string("/wait_" + config->getString("streamname")).c_str(), O_CREAT | O_RDWR, ACCESSPERMS, 0); + if (waiting == SEM_FAILED){ + DEBUG_MSG(DLVL_FAIL, "Failed to open semaphore - cancelling"); + return -1; + } + sem_post(waiting); + sem_close(waiting); + + DEBUG_MSG(DLVL_HIGH,"Pre-While"); + + long long int activityCounter = Util::getMS(); + while ((Util::getMS() - activityCounter) < 10000){//1minute timeout + DEBUG_MSG(DLVL_HIGH, "Timer running"); + Util::sleep(1000); + removeUnused(); + userPage.parseEach(doNothing); + if (userPage.amount){ + activityCounter = Util::getMS(); + DEBUG_MSG(DLVL_HIGH, "Connected users: %d", userPage.amount); + } + } + DEBUG_MSG(DLVL_DEVEL,"Closing clean"); + //end player functionality + } + return 0; + } + + void Input::removeUnused(){ + for (std::map<unsigned int, std::map<unsigned int, unsigned int> >::iterator it = pageCounter.begin(); it != pageCounter.end(); it++){ + for (std::map<unsigned int, unsigned int>::iterator it2 = it->second.begin(); it2 != it->second.end(); it2++){ + it2->second--; + } + bool change = true; + while (change){ + change = false; + for (std::map<unsigned int, unsigned int>::iterator it2 = it->second.begin(); it2 != it->second.end(); it2++){ + if (!it2->second){ + DEBUG_MSG(DLVL_DEVEL, "Erasing page %u:%u", it->first, it2->first); + pagesByTrack[it->first].erase(it2->first); + pageCounter[it->first].erase(it2->first); + change = true; + break; + } + } + } + } + } + + void Input::parseHeader(){ + DEBUG_MSG(DLVL_DEVEL,"Parsing the header"); + //Select all tracks for parsing header + selectedTracks.clear(); + std::stringstream trackSpec; + for (std::map<int, DTSC::Track>::iterator it = myMeta.tracks.begin(); it != myMeta.tracks.end(); it++) { + DEBUG_MSG(DLVL_VERYHIGH, "Track %d encountered", it->first); + //selectedTracks.insert(it->first); + if (trackSpec.str() != ""){ + trackSpec << " "; + } + trackSpec << it->first; + DEBUG_MSG(DLVL_VERYHIGH, "Trackspec now %s", trackSpec.str().c_str()); + for (std::deque<DTSC::Key>::iterator it2 = it->second.keys.begin(); it2 != it->second.keys.end(); it2++){ + keyTimes[it->first].insert(it2->getTime()); + } + } + trackSelect(trackSpec.str()); + + std::map<int, DTSCPageData> curData; + std::map<int, booking> bookKeeping; + + seek(0); + getNext(); + + while(lastPack){//loop through all + int tid = lastPack.getTrackId(); + if (!bookKeeping.count(tid)){ + bookKeeping[tid].first = 0; + bookKeeping[tid].curPart = 0; + bookKeeping[tid].curKey = 0; + + curData[tid].lastKeyTime = 0xFFFFFFFF; + curData[tid].keyNum = 1; + curData[tid].partNum = 0; + curData[tid].dataSize = 0; + curData[tid].curOffset = 0; + curData[tid].firstTime = myMeta.tracks[tid].keys[0].getTime(); + + char tmpId[20]; + sprintf(tmpId, "%d", tid); + indexPages[tid].init(config->getString("streamname") + tmpId, 8192, true);//Pages of 8kb in size, room for 512 parts. + } + if (myMeta.tracks[tid].keys[bookKeeping[tid].curKey].getParts() == curData[tid].partNum){ + if (curData[tid].dataSize > 8388608) { + pagesByTrack[tid][bookKeeping[tid].first] = curData[tid]; + bookKeeping[tid].first += curData[tid].keyNum; + curData[tid].keyNum = 0; + curData[tid].dataSize = 0; + curData[tid].firstTime = myMeta.tracks[tid].keys[bookKeeping[tid].curKey].getTime(); + } + bookKeeping[tid].curKey++; + curData[tid].keyNum++; + curData[tid].partNum = 0; + } + curData[tid].dataSize += lastPack.getDataLen(); + curData[tid].partNum ++; + bookKeeping[tid].curPart ++; + getNext(false); + } + for (std::map<int, DTSC::Track>::iterator it = myMeta.tracks.begin(); it != myMeta.tracks.end(); it++) { + if (curData.count(it->first) && !pagesByTrack[it->first].count(bookKeeping[it->first].first)){ + pagesByTrack[it->first][bookKeeping[it->first].first] = curData[it->first]; + } + if (!pagesByTrack.count(it->first)){ + DEBUG_MSG(DLVL_WARN, "No pages for track %d found", it->first); + }else{ + DEBUG_MSG(DLVL_HIGH, "Track %d (%s) split into %lu pages", it->first, myMeta.tracks[it->first].codec.c_str(), pagesByTrack[it->first].size()); + for (std::map<int, DTSCPageData>::iterator it2 = pagesByTrack[it->first].begin(); it2 != pagesByTrack[it->first].end(); it2++){ + } + } + } + } + + + bool Input::bufferFrame(int track, int keyNum){ + DEBUG_MSG(DLVL_DONTEVEN, "Attempting to buffer %d:%d", track, keyNum); + if (!pagesByTrack.count(track)){ + return false; + } + std::map<int, DTSCPageData> ::iterator it = pagesByTrack[track].upper_bound(keyNum); + if (it == pagesByTrack[track].begin()){ + return false; + } + it --; + int pageNum = it->first; + pageCounter[track][pageNum] = 15;///Keep page 15seconds in memory after last use + + if (!dataPages[track].count(pageNum)){ + char pageId[100]; + int pageIdLen = sprintf(pageId, "%s%d_%d", config->getString("streamname").c_str(), track, pageNum); + std::string tmpString(pageId, pageIdLen); + dataPages[track][pageNum].init(tmpString, it->second.dataSize, true); + DEBUG_MSG(DLVL_HIGH, "Buffering page %d through %d / %lu", pageNum, pageNum + it->second.keyNum, myMeta.tracks[track].keys.size()); + + std::stringstream trackSpec; + trackSpec << track; + trackSelect(trackSpec.str()); + }else{ + return true; + } + seek(myMeta.tracks[track].keys[pageNum].getTime()); + long long unsigned int stopTime = myMeta.tracks[track].lastms + 1; + if ((int)myMeta.tracks[track].keys.size() > pageNum + it->second.keyNum){ + stopTime = myMeta.tracks[track].keys[pageNum + it->second.keyNum].getTime(); + } + DEBUG_MSG(DLVL_HIGH, "Playing from %ld to %llu", myMeta.tracks[track].keys[pageNum].getTime(), stopTime); + getNext(); + while (lastPack && lastPack.getTime() < stopTime){ + if (it->second.curOffset + lastPack.getDataLen() > pagesByTrack[track][pageNum].dataSize){ + DEBUG_MSG(DLVL_WARN, "Trying to write %u bytes past the end of page %u/%u", lastPack.getDataLen(), track, pageNum); + return true; + }else{ + memcpy(dataPages[track][pageNum].mapped + it->second.curOffset, lastPack.getData(), lastPack.getDataLen()); + it->second.curOffset += lastPack.getDataLen(); + } + getNext(); + } + for (int i = 0; i < indexPages[track].len / 8; i++){ + if (((long long int*)indexPages[track].mapped)[i] == 0){ + ((long long int*)indexPages[track].mapped)[i] = (((long long int)htonl(pageNum)) << 32) | htonl(it->second.keyNum); + break; + } + } + return true; + } + + bool Input::atKeyFrame(){ + static std::map<int, int> lastSeen; + //not in keyTimes? We're not at a keyframe. + unsigned int c = keyTimes[lastPack.getTrackId()].count(lastPack.getTime()); + if (!c){ + return false; + } + //skip double times + if (lastSeen.count(lastPack.getTrackId()) && lastSeen[lastPack.getTrackId()] == lastPack.getTime()){ + return false; + } + //set last seen, and return true + lastSeen[lastPack.getTrackId()] = lastPack.getTime(); + return true; + } + + void Input::play(int until) { + playing = -1; + playUntil = until; + initialTime = 0; + benchMark = Util::getMS(); + } + + void Input::playOnce() { + if (playing <= 0) { + playing = 1; + } + ++playing; + benchMark = Util::getMS(); + } + + void Input::quitPlay() { + playing = 0; + } +} + diff --git a/src/input/input.h b/src/input/input.h new file mode 100644 index 00000000..98425e4a --- /dev/null +++ b/src/input/input.h @@ -0,0 +1,84 @@ +#include <set> +#include <map> +#include <cstdlib> +#include <mist/config.h> +#include <mist/json.h> +#include <mist/timing.h> +#include <mist/dtsc.h> +#include <mist/shared_memory.h> + +namespace Mist { + struct DTSCPageData { + DTSCPageData() : keyNum(0), partNum(0), dataSize(0), curOffset(0), firstTime(0){} + int keyNum;///<The number of keyframes in this page. + int partNum;///<The number of parts in this page. + unsigned long long int dataSize;///<The full size this page should be. + unsigned long long int curOffset;///<The current write offset in the page. + unsigned long long int firstTime;///<The first timestamp of the page. + unsigned long lastKeyTime;///<The last key time encountered on this track. + }; + + struct booking { + int first; + int curKey; + int curPart; + }; + + class Input { + public: + Input(Util::Config * cfg); + int run(); + virtual ~Input() {}; + protected: + static void doNothing(char * data, size_t len, unsigned int id); + virtual bool setup() = 0; + virtual bool readHeader() = 0; + virtual bool atKeyFrame(); + virtual void getNext(bool smart = true) {}; + virtual void seek(int seekTime){}; + void play(int until = 0); + void playOnce(); + void quitPlay(); + virtual void removeUnused(); + virtual void trackSelect(std::string trackSpec){}; + virtual void userCallback(char * data, size_t len, unsigned int id); + + void parseHeader(); + bool bufferFrame(int track, int keyNum); + + unsigned int packTime;///Media-timestamp of the last packet. + int lastActive;///Timestamp of the last time we received or sent something. + int initialTime; + int playing; + unsigned int playUntil; + unsigned int benchMark; + std::set<int> selectedTracks; + + bool isBuffer; + + Util::Config * config; + JSON::Value capa; + Socket::Connection StatsSocket; + DTSC::Meta myMeta; + DTSC::Packet lastPack; + + std::map<int,std::set<int> > keyTimes; + IPC::sharedPage metaPage; + //Create server for user pages + IPC::sharedServer userPage; + + + //TrackIndex pages + std::map<int, IPC::sharedPage> indexPages; + std::map<int, std::map<int, IPC::sharedPage> > dataPages; + + //Page Overview + std::map<int, std::map<int, DTSCPageData> > pagesByTrack; + + std::map<unsigned int, std::map<unsigned int, unsigned int> > pageCounter; + + static Input * singleton; + }; + +} + diff --git a/src/input/input_buffer.cpp b/src/input/input_buffer.cpp new file mode 100644 index 00000000..a0f33115 --- /dev/null +++ b/src/input/input_buffer.cpp @@ -0,0 +1,274 @@ +#include <iostream> +#include <cstring> +#include <cerrno> +#include <cstdlib> +#include <cstdio> +#include <string> +#include <mist/stream.h> +#include <mist/defines.h> + +#include "input_buffer.h" + +namespace Mist { + inputBuffer::inputBuffer(Util::Config * cfg) : Input(cfg) { + JSON::Value option; + option["arg"] = "integer"; + option["long"] = "buffer"; + option["short"] = "b"; + option["help"] = "Buffertime for this stream."; + option["value"].append(30000LL); + config->addOption("bufferTime", option); + + capa["desc"] = "Enables buffered live input"; + capa["codecs"][0u][0u].append("*"); + capa["codecs"][0u][1u].append("*"); + capa["codecs"][0u][2u].append("*"); + capa["codecs"][0u][3u].append("*"); + capa["codecs"][0u][4u].append("*"); + capa["codecs"][0u][5u].append("*"); + capa["codecs"][0u][6u].append("*"); + capa["codecs"][0u][7u].append("*"); + capa["codecs"][0u][8u].append("*"); + capa["codecs"][0u][9u].append("*"); + DEBUG_MSG(DLVL_DEVEL, "Started MistInBuffer"); + isBuffer = true; + singleton = this; + bufferTime = 0; + cutTime = 0; + + } + + void inputBuffer::updateMeta(){ + long long unsigned int firstms = 0xFFFFFFFFFFFFFFFF; + long long unsigned int lastms = 0; + for (std::map<int,DTSC::Track>::iterator it = myMeta.tracks.begin(); it != myMeta.tracks.end(); it++){ + if (it->second.firstms < firstms){ + firstms = it->second.firstms; + } + if (it->second.firstms > lastms){ + lastms = it->second.lastms; + } + } + myMeta.bufferWindow = lastms - firstms; + myMeta.writeTo(metaPage.mapped); + } + + bool inputBuffer::removeKey(unsigned int tid){ + if (myMeta.tracks[tid].keys.size() < 2 || myMeta.tracks[tid].fragments.size() < 2){ + return false; + } + DEBUG_MSG(DLVL_HIGH, "Erasing key %d:%d", tid, myMeta.tracks[tid].keys[0].getNumber()); + //remove all parts of this key + for (int i = 0; i < myMeta.tracks[tid].keys[0].getParts(); i++){ + myMeta.tracks[tid].parts.pop_front(); + } + //remove the key itself + myMeta.tracks[tid].keys.pop_front(); + //re-calculate firstms + myMeta.tracks[tid].firstms = myMeta.tracks[tid].keys[0].getTime(); + //delete the fragment if it's no longer fully buffered + if (myMeta.tracks[tid].fragments[0].getNumber() < myMeta.tracks[tid].keys[0].getNumber()){ + myMeta.tracks[tid].fragments.pop_front(); + myMeta.tracks[tid].missedFrags ++; + } + //if there is more than one page buffered for this track... + if (inputLoc[tid].size() > 1){ + //Check if the first key starts on the second page or higher + if (myMeta.tracks[tid].keys[0].getNumber() >= (++(inputLoc[tid].begin()))->first){ + //Find page in indexpage and null it + for (int i = 0; i < 8192; i += 8){ + int thisKeyNum = ((((long long int *)(indexPages[tid].mapped + i))[0]) >> 32) & 0xFFFFFFFF; + if (thisKeyNum == htonl(pagesByTrack[tid].begin()->first) && ((((long long int *)(indexPages[tid].mapped + i))[0]) != 0)){ + (((long long int *)(indexPages[tid].mapped + i))[0]) = 0; + } + } + DEBUG_MSG(DLVL_DEVEL, "Erasing track %d, keys %lu-%lu from buffer", tid, inputLoc[tid].begin()->first, inputLoc[tid].begin()->first + inputLoc[tid].begin()->second.keyNum - 1); + inputLoc[tid].erase(inputLoc[tid].begin()); + dataPages[tid].erase(dataPages[tid].begin()); + }else{ + DEBUG_MSG(DLVL_HIGH, "%d still on first page (%lu - %lu)", myMeta.tracks[tid].keys[0].getNumber(), inputLoc[tid].begin()->first, inputLoc[tid].begin()->first + inputLoc[tid].begin()->second.keyNum - 1); + } + } + return true; + } + + void inputBuffer::removeUnused(){ + //find the earliest video keyframe stored + unsigned int firstVideo = 1; + for(std::map<int,DTSC::Track>::iterator it = myMeta.tracks.begin(); it != myMeta.tracks.end(); it++){ + if (it->second.type == "video"){ + if (it->second.firstms < firstVideo || firstVideo == 1){ + firstVideo = it->second.firstms; + } + } + } + for(std::map<int,DTSC::Track>::iterator it = myMeta.tracks.begin(); it != myMeta.tracks.end(); it++){ + //non-video tracks need to have a second keyframe that is <= firstVideo + if (it->second.type != "video"){ + if (it->second.keys.size() < 2 || it->second.keys[1].getTime() > firstVideo){ + continue; + } + } + //Buffer cutting + while(it->second.keys.size() > 1 && it->second.keys[0].getTime() < cutTime){ + if (!removeKey(it->first)){break;} + } + //Buffer size management + while(it->second.keys.size() > 1 && (it->second.lastms - it->second.keys[1].getTime()) > bufferTime){ + if (!removeKey(it->first)){break;} + } + } + updateMeta(); + } + + void inputBuffer::userCallback(char * data, size_t len, unsigned int id) { + unsigned long tmp = ((long)(data[0]) << 24) | ((long)(data[1]) << 16) | ((long)(data[2]) << 8) | ((long)(data[3])); + if (tmp & 0x80000000) { + //Track is set to "New track request", assign new track id and create shared memory page + unsigned long tNum = (givenTracks.size() ? (*givenTracks.rbegin()) : 0) + 1; + ///\todo Neatify this + data[0] = (tNum >> 24) & 0xFF; + data[1] = (tNum >> 16) & 0xFF; + data[2] = (tNum >> 8) & 0xFF; + data[3] = (tNum) & 0xFF; + givenTracks.insert(tNum); + char tmpChr[100]; + long tmpLen = sprintf(tmpChr, "liveStream_%s%lu", config->getString("streamname").c_str(), tNum); + metaPages[tNum].init(std::string(tmpChr, tmpLen), 8388608, true); + } else { + unsigned long tNum = ((long)(data[0]) << 24) | ((long)(data[1]) << 16) | ((long)(data[2]) << 8) | ((long)(data[3])); + if (!myMeta.tracks.count(tNum)) { + DEBUG_MSG(DLVL_DEVEL, "Tracknum not in meta: %lu, from user %u", tNum, id); + if (metaPages[tNum].mapped) { + if (metaPages[tNum].mapped[0] == 'D' && metaPages[tNum].mapped[1] == 'T') { + unsigned int len = ntohl(((int *)metaPages[tNum].mapped)[1]); + unsigned int i = 0; + JSON::Value tmpMeta; + JSON::fromDTMI((const unsigned char *)metaPages[tNum].mapped + 8, len, i, tmpMeta); + DTSC::Meta tmpTrack(tmpMeta); + int oldTNum = tmpTrack.tracks.begin()->first; + bool collision = false; + for (std::map<int, DTSC::Track>::iterator it = myMeta.tracks.begin(); it != myMeta.tracks.end(); it++) { + if (it->first == tNum) { + continue; + } + if (it->second.getIdentifier() == tmpTrack.tracks[oldTNum].getIdentifier()) { + collision = true; + break; + } + } + if (collision) { + /// \todo Erasing page for now, should do more here + DEBUG_MSG(DLVL_DEVEL, "Collision detected! Erasing page for now, should do more here"); + metaPages.erase(tNum); + data[0] = 0xFF; + data[1] = 0xFF; + data[2] = 0xFF; + data[3] = 0xFF; + } else { + if (!myMeta.tracks.count(tNum)) { + myMeta.tracks[tNum] = tmpTrack.tracks[oldTNum]; + data[4] = 0x00; + data[5] = 0x00; + updateMeta(); + char firstPage[100]; + sprintf(firstPage, "%s%lu", config->getString("streamname").c_str(), tNum); + indexPages[tNum].init(firstPage, 8192, true); + ((long long int *)indexPages[tNum].mapped)[0] = htonl(1000); + ///\todo Fix for non-first-key-pushing + sprintf(firstPage, "%s%lu_0", config->getString("streamname").c_str(), tNum); + ///\todo Make size dynamic / other solution. 25mb is too much. + dataPages[tNum][0].init(firstPage, 26214400, true); + } + } + } + } + } else { + //First check if the previous page has been finished: + if (!inputLoc[tNum].count(dataPages[tNum].rbegin()->first) || !inputLoc[tNum][dataPages[tNum].rbegin()->first].curOffset){ + if (dataPages[tNum].size() > 1){ + int prevPage = (++dataPages[tNum].rbegin())->first; + //update previous page. + updateMetaFromPage(tNum, prevPage); + } + } + //update current page + int curPage = dataPages[tNum].rbegin()->first; + updateMetaFromPage(tNum, curPage); + if (inputLoc[tNum][curPage].curOffset > 8388608) { + //create new page is > 8MB + int nxtPage = curPage + inputLoc[tNum][curPage].keyNum; + char nextPageName[100]; + sprintf(nextPageName, "%s%lu_%d", config->getString("streamname").c_str(), tNum, nxtPage); + dataPages[tNum][nxtPage].init(nextPageName, 20971520, true); + bool createdNew = false; + for (int i = 0; i < 8192; i += 8){ + int thisKeyNum = ((((long long int *)(indexPages[tNum].mapped + i))[0]) >> 32) & 0xFFFFFFFF; + if (thisKeyNum == htonl(curPage)){ + if((ntohl((((long long int*)(indexPages[tNum].mapped + i))[0]) & 0xFFFFFFFF) == 1000)){ + ((long long int *)(indexPages[tNum].mapped + i))[0] &= 0xFFFFFFFF00000000; + ((long long int *)(indexPages[tNum].mapped + i))[0] |= htonl(inputLoc[tNum][curPage].keyNum); + } + } + if (!createdNew && (((long long int*)(indexPages[tNum].mapped + i))[0]) == 0){ + createdNew = true; + ((long long int *)(indexPages[tNum].mapped + i))[0] = (((long long int)htonl(nxtPage)) << 32) | htonl(1000); + } + } + } + } + } + } + + void inputBuffer::updateMetaFromPage(int tNum, int pageNum){ + DTSC::Packet tmpPack; + tmpPack.reInit(dataPages[tNum][pageNum].mapped + inputLoc[tNum][pageNum].curOffset, 0); + while (tmpPack) { + myMeta.update(tmpPack); + if (inputLoc[tNum][pageNum].firstTime == 0){ + inputLoc[tNum][pageNum].firstTime = tmpPack.getTime(); + } + //Overloaded use of .firstTime to indicate last Keytime on non-video streams; + if (myMeta.tracks[tNum].type == "video"){ + inputLoc[tNum][pageNum].keyNum += tmpPack.getFlag("keyframe"); + }else{ + if ((tmpPack.getTime() > 5000) && ((tmpPack.getTime() - 5000) > inputLoc[tNum][pageNum].firstTime)){ + inputLoc[tNum][pageNum].keyNum ++; + } + } + inputLoc[tNum][pageNum].curOffset += tmpPack.getDataLen(); + tmpPack.reInit(dataPages[tNum][pageNum].mapped + inputLoc[tNum][pageNum].curOffset, 0); + } + updateMeta(); + } + + bool inputBuffer::setup() { + if (!bufferTime){ + bufferTime = config->getInteger("bufferTime"); + } + JSON::Value servConf = JSON::fromFile(Util::getTmpFolder() + "streamlist"); + if (servConf.isMember("streams") && servConf["streams"].isMember(config->getString("streamname"))){ + JSON::Value & streamConfig = servConf["streams"][config->getString("streamname")]; + if (streamConfig.isMember("DVR") && streamConfig["DVR"].asInt()){ + if (bufferTime != streamConfig["DVR"].asInt()){ + DEBUG_MSG(DLVL_DEVEL, "Setting bufferTime from %u to new value of %lli", bufferTime, streamConfig["DVR"].asInt()); + bufferTime = streamConfig["DVR"].asInt(); + } + } + } + return true; + } + + bool inputBuffer::readHeader() { + return true; + } + + void inputBuffer::getNext(bool smart) {} + + void inputBuffer::seek(int seekTime) {} + + void inputBuffer::trackSelect(std::string trackSpec) {} +} + + + diff --git a/src/input/input_buffer.h b/src/input/input_buffer.h new file mode 100644 index 00000000..20764f4d --- /dev/null +++ b/src/input/input_buffer.h @@ -0,0 +1,33 @@ +#include "input.h" +#include <mist/dtsc.h> +#include <mist/shared_memory.h> + +namespace Mist { + class inputBuffer : public Input { + public: + inputBuffer(Util::Config * cfg); + private: + unsigned int bufferTime; + unsigned int cutTime; + protected: + //Private Functions + bool setup(); + void updateMeta(); + bool readHeader(); + void getNext(bool smart = true); + void updateMetaFromPage(int tNum, int pageNum); + void seek(int seekTime); + void trackSelect(std::string trackSpec); + bool removeKey(unsigned int tid); + void removeUnused(); + void userCallback(char * data, size_t len, unsigned int id); + std::set<unsigned long> givenTracks; + std::map<unsigned long, IPC::sharedPage> metaPages; + std::map<unsigned long, std::map<unsigned long, DTSCPageData> > inputLoc; + inputBuffer * singleton; + }; +} + +typedef Mist::inputBuffer mistIn; + + diff --git a/src/input/input_dtsc.cpp b/src/input/input_dtsc.cpp new file mode 100644 index 00000000..0f6d9e47 --- /dev/null +++ b/src/input/input_dtsc.cpp @@ -0,0 +1,90 @@ +#include <iostream> +#include <cstring> +#include <cerrno> +#include <cstdlib> +#include <cstdio> +#include <string> +#include <mist/stream.h> +#include <mist/defines.h> + +#include "input_dtsc.h" + +namespace Mist { + inputDTSC::inputDTSC(Util::Config * cfg) : Input(cfg) { + capa["decs"] = "Enables DTSC Input"; + capa["codecs"][0u][0u].append("H264"); + capa["codecs"][0u][0u].append("H263"); + capa["codecs"][0u][0u].append("VP6"); + capa["codecs"][0u][0u].append("theora"); + capa["codecs"][0u][1u].append("AAC"); + capa["codecs"][0u][1u].append("MP3"); + capa["codecs"][0u][1u].append("vorbis"); + } + + bool inputDTSC::setup() { + if (config->getString("input") == "-") { + std::cerr << "Input from stream not yet supported" << std::endl; + return false; + } + if (config->getString("output") != "-") { + std::cerr << "Output to non-stdout not yet supported" << std::endl; + } + + //open File + inFile = DTSC::File(config->getString("input")); + if (!inFile) { + return false; + } + return true; + } + + bool inputDTSC::readHeader() { + if (!inFile) { + return false; + } + DTSC::File tmp(config->getString("input") + ".dtsh"); + if (tmp) { + myMeta = tmp.getMeta(); + DEBUG_MSG(DLVL_DEVEL,"Meta read in with %lu tracks", myMeta.tracks.size()); + return true; + } + if (inFile.getMeta().moreheader < 0 || inFile.getMeta().tracks.size() == 0) { + DEBUG_MSG(DLVL_FAIL,"Missing external header file"); + return false; + } + myMeta = DTSC::Meta(inFile.getMeta()); + DEBUG_MSG(DLVL_DEVEL,"Meta read in with %lu tracks", myMeta.tracks.size()); + return true; + } + + void inputDTSC::getNext(bool smart) { + if (smart){ + inFile.seekNext(); + }else{ + inFile.parseNext(); + } + lastPack = inFile.getPacket(); + } + + void inputDTSC::seek(int seekTime) { + inFile.seek_time(seekTime); + initialTime = 0; + playUntil = 0; + } + + void inputDTSC::trackSelect(std::string trackSpec) { + selectedTracks.clear(); + long long unsigned int index; + while (trackSpec != "") { + index = trackSpec.find(' '); + selectedTracks.insert(atoi(trackSpec.substr(0, index).c_str())); + if (index != std::string::npos) { + trackSpec.erase(0, index + 1); + } else { + trackSpec = ""; + } + } + inFile.selectTracks(selectedTracks); + } +} + diff --git a/src/input/input_dtsc.h b/src/input/input_dtsc.h new file mode 100644 index 00000000..9a9f12db --- /dev/null +++ b/src/input/input_dtsc.h @@ -0,0 +1,22 @@ +#include "input.h" +#include <mist/dtsc.h> + +namespace Mist { + class inputDTSC : public Input { + public: + inputDTSC(Util::Config * cfg); + protected: + //Private Functions + bool setup(); + bool readHeader(); + void getNext(bool smart = true); + void seek(int seekTime); + void trackSelect(std::string trackSpec); + + DTSC::File inFile; + }; +} + +typedef Mist::inputDTSC mistIn; + + diff --git a/src/input/input_flv.cpp b/src/input/input_flv.cpp new file mode 100644 index 00000000..5e5ef403 --- /dev/null +++ b/src/input/input_flv.cpp @@ -0,0 +1,130 @@ +#include <iostream> +#include <fstream> +#include <cstring> +#include <cerrno> +#include <cstdlib> +#include <cstdio> +#include <string> +#include <mist/stream.h> +#include <mist/flv_tag.h> +#include <mist/defines.h> + +#include "input_flv.h" + +namespace Mist { + inputFLV::inputFLV(Util::Config * cfg) : Input(cfg) { + capa["decs"] = "Enables FLV Input"; + capa["codecs"][0u][0u].append("H264"); + capa["codecs"][0u][0u].append("H263"); + capa["codecs"][0u][0u].append("VP6"); + capa["codecs"][0u][1u].append("AAC"); + capa["codecs"][0u][1u].append("MP3"); + } + + bool inputFLV::setup() { + if (config->getString("input") == "-") { + std::cerr << "Input from stream not yet supported" << std::endl; + return false; + } + if (config->getString("output") != "-") { + std::cerr << "Output to non-stdout not yet supported" << std::endl; + } + + //open File + inFile = fopen(config->getString("input").c_str(), "r"); + if (!inFile) { + return false; + } + return true; + } + + bool inputFLV::readHeader() { + JSON::Value lastPack; + if (!inFile) { + return false; + } + //See whether a separate header file exists. + DTSC::File tmp(config->getString("input") + ".dtsh"); + if (tmp){ + myMeta = tmp.getMeta(); + return true; + } + //Create header file from FLV data + fseek(inFile, 13, SEEK_SET); + FLV::Tag tmpTag; + long long int lastBytePos = 13; + while (!feof(inFile) && !FLV::Parse_Error){ + if (tmpTag.FileLoader(inFile)){ + lastPack.null(); + lastPack = tmpTag.toJSON(myMeta); + lastPack["bpos"] = lastBytePos; + myMeta.update(lastPack); + lastBytePos = ftell(inFile); + } + } + if (FLV::Parse_Error){ + std::cerr << FLV::Error_Str << std::endl; + return false; + } + std::ofstream oFile(std::string(config->getString("input") + ".dtsh").c_str()); + oFile << myMeta.toJSON().toNetPacked(); + oFile.close(); + return true; + } + + void inputFLV::getNext(bool smart) { + static JSON::Value thisPack; + thisPack.null(); + long long int lastBytePos = ftell(inFile); + FLV::Tag tmpTag; + while (!feof(inFile) && !FLV::Parse_Error){ + if (tmpTag.FileLoader(inFile)){ + thisPack = tmpTag.toJSON(myMeta); + thisPack["bpos"] = lastBytePos; + if ( !selectedTracks.count(thisPack["trackid"].asInt())){ + getNext(); + } + break; + } + } + if (FLV::Parse_Error){ + std::cerr << FLV::Error_Str << std::endl; + thisPack.null(); + lastPack.null(); + return; + } + std::string tmpStr = thisPack.toNetPacked(); + lastPack.reInit(tmpStr.data(), tmpStr.size()); + } + + void inputFLV::seek(int seekTime) { + //We will seek to the corresponding keyframe of the video track if selected, otherwise audio keyframe. + //Flv files are never multi-track, so track 1 is video, track 2 is audio. + int trackSeek = (selectedTracks.count(1) ? 1 : 2); + size_t seekPos = myMeta.tracks[trackSeek].keys[0].getBpos(); + for (int i = 0; i < myMeta.tracks[trackSeek].keys.size(); i++){ + if (myMeta.tracks[trackSeek].keys[i].getTime() > seekTime){ + DEBUG_MSG(DLVL_WARN, "Seeking to keyframe %d on track %d, timestamp %ld, bytepos %lu", i, trackSeek, myMeta.tracks[trackSeek].keys[i].getTime(), seekPos); + break; + } + seekPos = myMeta.tracks[trackSeek].keys[i].getBpos(); + } + fseek(inFile, seekPos, SEEK_SET); + } + + void inputFLV::trackSelect(std::string trackSpec) { + selectedTracks.clear(); + long long int index; + while (trackSpec != "") { + index = trackSpec.find(' '); + selectedTracks.insert(atoi(trackSpec.substr(0, index).c_str())); + DEBUG_MSG(DLVL_WARN, "Added track %d, index = %lld, (index == npos) = %d", atoi(trackSpec.substr(0, index).c_str()), index, index == std::string::npos); + if (index != std::string::npos) { + trackSpec.erase(0, index + 1); + } else { + trackSpec = ""; + } + } + } +} + diff --git a/src/input/input_flv.h b/src/input/input_flv.h new file mode 100644 index 00000000..6a8d5af9 --- /dev/null +++ b/src/input/input_flv.h @@ -0,0 +1,21 @@ +#include "input.h" +#include <mist/dtsc.h> + +namespace Mist { + class inputFLV : public Input { + public: + inputFLV(Util::Config * cfg); + protected: + //Private Functions + bool setup(); + bool readHeader(); + void getNext(bool smart = true); + void seek(int seekTime); + void trackSelect(std::string trackSpec); + + FILE * inFile; + }; +} + +typedef Mist::inputFLV mistIn; + diff --git a/src/input/input_ogg.cpp b/src/input/input_ogg.cpp new file mode 100644 index 00000000..df17b033 --- /dev/null +++ b/src/input/input_ogg.cpp @@ -0,0 +1,274 @@ +#include <iostream> +#include <fstream> +#include <cstring> +#include <cerrno> +#include <cstdlib> +#include <cstdio> +#include <string> +#include <mist/stream.h> +#include <mist/ogg.h> +#include <mist/defines.h> +#include <mist/bitstream.h> + +#include "input_ogg.h" + +namespace Mist { + inputOGG::inputOGG(Util::Config * cfg) : Input(cfg) { + capa["decs"] = "Enables OGG Input"; + capa["codecs"][0u][0u].append("theora"); + capa["codecs"][0u][1u].append("vorbis"); + } + + bool inputOGG::setup() { + if (config->getString("input") == "-") { + std::cerr << "Input from stream not yet supported" << std::endl; + return false; + } + if (config->getString("output") != "-") { + std::cerr << "Output to non-stdout not yet supported" << std::endl; + } + + //open File + inFile = fopen(config->getString("input").c_str(), "r"); + if (!inFile) { + return false; + } + return true; + } + + void inputOGG::parseBeginOfStream(OGG::Page & bosPage) { + long long int tid = snum2tid.size() + 1; + snum2tid[bosPage.getBitstreamSerialNumber()] = tid; + if (!memcmp(bosPage.getFullPayload() + 1, "theora", 6)) { + oggTracks[tid].codec = THEORA; + theora::header tmpHead(bosPage.getFullPayload(), bosPage.getPayloadSize()); + oggTracks[tid].msPerFrame = (double)(tmpHead.getFRD() * 1000) / tmpHead.getFRN(); + } + if (!memcmp(bosPage.getFullPayload() + 1, "vorbis", 6)) { + oggTracks[tid].codec = VORBIS; + vorbis::header tmpHead(bosPage.getFullPayload(), bosPage.getPayloadSize()); + oggTracks[tid].msPerFrame = (double)1000 / ntohl(tmpHead.getAudioSampleRate()); + } + } + + bool inputOGG::readHeader() { + JSON::Value lastPack; + if (!inFile) { + return false; + } + //See whether a separate header file exists. + DTSC::File tmp(config->getString("input") + ".dtsh"); + if (tmp) { + myMeta = tmp.getMeta(); + return true; + } + //Create header file from OGG data + fseek(inFile, 0, SEEK_SET); + OGG::Page tmpPage; + long long int lastBytePos = 0; + while (tmpPage.read(inFile)) { + DEBUG_MSG(DLVL_WARN,"Read a page"); + if (tmpPage.getHeaderType() & OGG::BeginOfStream){ + parseBeginOfStream(tmpPage); + DEBUG_MSG(DLVL_WARN,"Read BOS page for stream %lu, now track %lld", tmpPage.getBitstreamSerialNumber(), snum2tid[tmpPage.getBitstreamSerialNumber()]); + } + int offset = 0; + long long int tid = snum2tid[tmpPage.getBitstreamSerialNumber()]; + for (std::deque<unsigned int>::iterator it = tmpPage.getSegmentTableDeque().begin(); it != tmpPage.getSegmentTableDeque().end(); it++) { + if (oggTracks[tid].parsedHeaders) { + DEBUG_MSG(DLVL_WARN,"Parsing a page segment on track %lld", tid); + if ((it == (tmpPage.getSegmentTableDeque().end() - 1)) && (int)(tmpPage.getPageSegments()) == 255 && (int)(tmpPage.getSegmentTable()[254]) == 255) { + oggTracks[tid].contBuffer.append(tmpPage.getFullPayload() + offset, (*it)); + } else { + lastPack["trackid"] = tid; + lastPack["time"] = (long long)oggTracks[tid].lastTime; + if (oggTracks[tid].contBuffer.size()) { + lastPack["data"] = oggTracks[tid].contBuffer + std::string(tmpPage.getFullPayload() + offset, (*it)); + oggTracks[tid].contBuffer.clear(); + } else { + lastPack["data"] = std::string(tmpPage.getFullPayload() + offset, (*it)); + } + if (oggTracks[tid].codec == VORBIS) { + unsigned int blockSize = 0; + Utils::bitstreamLSBF packet; + packet.append(lastPack["data"].asString()); + if (!packet.get(1)) { + blockSize = oggTracks[tid].blockSize[oggTracks[tid].vModes[packet.get(vorbis::ilog(oggTracks[tid].vModes.size() - 1))].blockFlag]; + } else { + DEBUG_MSG(DLVL_WARN, "Packet type != 0"); + } + oggTracks[tid].lastTime += oggTracks[tid].msPerFrame * (blockSize / oggTracks[tid].channels); + } + if (oggTracks[tid].codec == THEORA) { + oggTracks[tid].lastTime += oggTracks[tid].msPerFrame; + if (it == (tmpPage.getSegmentTableDeque().end() - 1)) { + if (oggTracks[tid].idHeader.parseGranuleUpper(oggTracks[tid].lastGran) != oggTracks[tid].idHeader.parseGranuleUpper(tmpPage.getGranulePosition())) { + lastPack["keyframe"] = 1ll; + oggTracks[tid].lastGran = tmpPage.getGranulePosition(); + } else { + lastPack["interframe"] = 1ll; + } + } + } + lastPack["bpos"] = 0ll; + DEBUG_MSG(DLVL_WARN,"Parsed a packet of track %lld, new timestamp %f", tid, oggTracks[tid].lastTime); + myMeta.update(lastPack); + } + } else { + //Parsing headers + switch (oggTracks[tid].codec) { + case THEORA: { + theora::header tmpHead(tmpPage.getFullPayload() + offset, (*it)); + DEBUG_MSG(DLVL_WARN,"Theora header, type %d", tmpHead.getHeaderType()); + switch (tmpHead.getHeaderType()) { + case 0: { + oggTracks[tid].idHeader = tmpHead; + myMeta.tracks[tid].height = tmpHead.getPICH(); + myMeta.tracks[tid].width = tmpHead.getPICW(); + myMeta.tracks[tid].idHeader = std::string(tmpPage.getFullPayload() + offset, (*it)); + break; + } + case 1: { + myMeta.tracks[tid].commentHeader = std::string(tmpPage.getFullPayload() + offset, (*it)); + break; + } + case 2: { + myMeta.tracks[tid].codec = "theora"; + myMeta.tracks[tid].trackID = tid; + myMeta.tracks[tid].type = "video"; + myMeta.tracks[tid].init = std::string(tmpPage.getFullPayload() + offset, (*it)); + oggTracks[tid].parsedHeaders = true; + oggTracks[tid].lastGran = 0; + break; + } + } + break; + } + case VORBIS: { + vorbis::header tmpHead(tmpPage.getFullPayload() + offset, (*it)); + DEBUG_MSG(DLVL_WARN,"Vorbis header, type %d", tmpHead.getHeaderType()); + switch (tmpHead.getHeaderType()) { + case 1: { + myMeta.tracks[tid].channels = tmpHead.getAudioChannels(); + myMeta.tracks[tid].idHeader = std::string(tmpPage.getFullPayload() + offset, (*it)); + oggTracks[tid].channels = tmpHead.getAudioChannels(); + oggTracks[tid].blockSize[0] = 1 << tmpHead.getBlockSize0(); + oggTracks[tid].blockSize[1] = 1 << tmpHead.getBlockSize1(); + break; + } + case 3: { + myMeta.tracks[tid].commentHeader = std::string(tmpPage.getFullPayload() + offset, (*it)); + break; + } + case 5: { + myMeta.tracks[tid].codec = "vorbis"; + myMeta.tracks[tid].trackID = tid; + myMeta.tracks[tid].type = "audio"; + DEBUG_MSG(DLVL_WARN,"Set default values"); + myMeta.tracks[tid].init = std::string(tmpPage.getFullPayload() + offset, (*it)); + DEBUG_MSG(DLVL_WARN,"Set init values"); + oggTracks[tid].vModes = tmpHead.readModeDeque(oggTracks[tid].channels); + DEBUG_MSG(DLVL_WARN,"Set vmodevalues"); + oggTracks[tid].parsedHeaders = true; + break; + } + } + break; + } + } + offset += (*it); + } + } + lastBytePos = ftell(inFile); + DEBUG_MSG(DLVL_WARN,"End of Loop, @ filepos %lld", lastBytePos); + } + DEBUG_MSG(DLVL_WARN,"Exited while loop"); + std::ofstream oFile(std::string(config->getString("input") + ".dtsh").c_str()); + oFile << myMeta.toJSON().toNetPacked(); + oFile.close(); + return true; + } + + bool inputOGG::seekNextPage(int tid){ + fseek(inFile, oggTracks[tid].lastPageOffset, SEEK_SET); + bool res = true; + do { + res = oggTracks[tid].myPage.read(inFile); + } while(res && snum2tid[oggTracks[tid].myPage.getBitstreamSerialNumber()] != tid); + oggTracks[tid].lastPageOffset = ftell(inFile); + oggTracks[tid].nxtSegment = 0; + return res; + } + + void inputOGG::getNext(bool smart) { + if (!sortedSegments.size()){ + for (std::set<int>::iterator it = selectedTracks.begin(); it != selectedTracks.end(); it++){ + seekNextPage((*it)); + } + } + if (sortedSegments.size()){ + int tid = (*(sortedSegments.begin())).tid; + bool addedPacket = false; + while (!addedPacket){ + segPart tmpPart; + if (oggTracks[tid].myPage.getSegment(oggTracks[tid].nxtSegment, tmpPart.segData, tmpPart.len)){ + if (oggTracks[tid].nxtSegment == 0 && oggTracks[tid].myPage.getHeaderType() && OGG::Continued){ + segment tmpSeg = *(sortedSegments.begin()); + tmpSeg.parts.push_back(tmpPart); + sortedSegments.erase(sortedSegments.begin()); + sortedSegments.insert(tmpSeg); + }else{ + segment tmpSeg; + tmpSeg.parts.push_back(tmpPart); + tmpSeg.tid = tid; + tmpSeg.time = oggTracks[tid].lastTime; + if (oggTracks[tid].codec == VORBIS) { + std::string data; + data.append(tmpPart.segData, tmpPart.len); + unsigned int blockSize = 0; + Utils::bitstreamLSBF packet; + packet.append(data); + if (!packet.get(1)) { + blockSize = oggTracks[tid].blockSize[oggTracks[tid].vModes[packet.get(vorbis::ilog(oggTracks[tid].vModes.size() - 1))].blockFlag]; + } + oggTracks[tid].lastTime += oggTracks[tid].msPerFrame * (blockSize / oggTracks[tid].channels); + } + if (oggTracks[tid].codec == THEORA) { + oggTracks[tid].lastTime += oggTracks[tid].msPerFrame; + } + sortedSegments.insert(tmpSeg); + addedPacket = true; + } + oggTracks[tid].nxtSegment ++; + }else{ + if (!seekNextPage(tid)){ + break; + } + } + } + std::string data; + } + } + + void inputOGG::seek(int seekTime) { + DEBUG_MSG(DLVL_WARN,"Seeking is not yet supported for ogg files"); + //Do nothing, seeking is not yet implemented for ogg + } + + void inputOGG::trackSelect(std::string trackSpec) { + selectedTracks.clear(); + long long int index; + while (trackSpec != "") { + index = trackSpec.find(' '); + selectedTracks.insert(atoi(trackSpec.substr(0, index).c_str())); + DEBUG_MSG(DLVL_WARN, "Added track %d, index = %lld, (index == npos) = %d", atoi(trackSpec.substr(0, index).c_str()), index, index == std::string::npos); + if (index != std::string::npos) { + trackSpec.erase(0, index + 1); + } else { + trackSpec = ""; + } + } + } +} + + diff --git a/src/input/input_ogg.h b/src/input/input_ogg.h new file mode 100644 index 00000000..fbc33e1f --- /dev/null +++ b/src/input/input_ogg.h @@ -0,0 +1,65 @@ +#include "input.h" +#include <mist/dtsc.h> +#include <mist/ogg.h> + +namespace Mist { + enum codecType {THEORA, VORBIS}; + + struct segPart{ + char * segData; + unsigned int len; + }; + + struct segment{ + bool operator < (const segment & rhs) const { + return time < rhs.time || (time == rhs.time && tid < rhs.tid); + } + std::vector<segPart> parts; + unsigned int time; + unsigned int tid; + }; + + class oggTrack{ + public: + oggTrack() : lastTime(0), parsedHeaders(false), lastPageOffset(0), nxtSegment(0) { } + codecType codec; + std::string contBuffer;//buffer for continuing pages + double lastTime; + long long unsigned int lastGran; + bool parsedHeaders; + double msPerFrame; + long long unsigned int lastPageOffset; + OGG::Page myPage; + unsigned int nxtSegment; + //Codec specific elements + //theora + theora::header idHeader; + //vorbis + std::deque<vorbis::mode> vModes; + char channels; + long long unsigned int blockSize[2]; + }; + + class inputOGG : public Input { + public: + inputOGG(Util::Config * cfg); + protected: + //Private Functions + bool setup(); + bool readHeader(); + bool seekNextPage(int tid); + void getNext(bool smart = true); + void seek(int seekTime); + void trackSelect(std::string trackSpec); + + void parseBeginOfStream(OGG::Page & bosPage); + + FILE * inFile; + std::map<long long int, long long int> snum2tid; + std::map<long long int, oggTrack> oggTracks; + std::set<segment> sortedSegments; + }; +} + +typedef Mist::inputOGG mistIn; + diff --git a/src/input/mist_in.cpp b/src/input/mist_in.cpp new file mode 100644 index 00000000..6f087f36 --- /dev/null +++ b/src/input/mist_in.cpp @@ -0,0 +1,62 @@ +#include <errno.h> +#include <sys/types.h> +#include <sys/stat.h> +#include <fcntl.h> +#include <sys/wait.h> +#include <unistd.h> +#include <semaphore.h> + +#include INPUTTYPE +#include <mist/config.h> +#include <mist/defines.h> + +int main(int argc, char * argv[]) { + Util::Config conf(argv[0], PACKAGE_VERSION); + mistIn conv(&conf); + if (conf.parseArgs(argc, argv)) { + sem_t * playerLock = sem_open(std::string("/lock_" + conf.getString("streamname")).c_str(), O_CREAT | O_RDWR, ACCESSPERMS, 1); + if (sem_trywait(playerLock) == -1){ + DEBUG_MSG(DLVL_DEVEL, "A player for stream %s is already running", conf.getString("streamname").c_str()); + return 1; + } + conf.activate(); + while (conf.is_active){ + int pid = fork(); + if (pid == 0){ + sem_close(playerLock); + return conv.run(); + } + if (pid == -1){ + DEBUG_MSG(DLVL_FAIL, "Unable to spawn player process"); + sem_post(playerLock); + return 2; + } + //wait for the process to exit + int status; + while (waitpid(pid, &status, 0) != pid && errno == EINTR) continue; + //clean up the semaphore by waiting for it, if it's non-zero + sem_t * waiting = sem_open(std::string("/wait_" + conf.getString("streamname")).c_str(), O_CREAT | O_RDWR, ACCESSPERMS, 0); + if (waiting == SEM_FAILED){ + DEBUG_MSG(DLVL_FAIL, "Failed to open semaphore - cancelling"); + return -1; + } + int sem_val = 0; + sem_getvalue(waiting, &sem_val); + while (sem_val){ + while (sem_wait(waiting) == -1 && errno == EINTR) continue; + sem_getvalue(waiting, &sem_val); + } + sem_close(waiting); + //if the exit was clean, don't restart it + if (WIFEXITED(status) && (WEXITSTATUS(status) == 0)){ + DEBUG_MSG(DLVL_DEVEL, "Finished player succesfully"); + break; + } + } + sem_post(playerLock); + sem_close(playerLock); + } + return 0; +} + + diff --git a/src/output/mist_out.cpp b/src/output/mist_out.cpp new file mode 100644 index 00000000..b23ff21a --- /dev/null +++ b/src/output/mist_out.cpp @@ -0,0 +1,21 @@ +#include OUTPUTTYPE +#include <mist/config.h> +#include <mist/socket.h> + +int spawnForked(Socket::Connection & S){ + mistOut tmp(S); + return tmp.run(); +} + +int main(int argc, char * argv[]) { + Util::Config conf(argv[0], PACKAGE_VERSION); + mistOut::init(&conf); + if (conf.parseArgs(argc, argv)) { + if (conf.getBool("json")) { + std::cout << mistOut::capa.toString() << std::endl; + return -1; + } + conf.serveForkedSocket(spawnForked); + } + return 0; +} diff --git a/src/output/output.cpp b/src/output/output.cpp new file mode 100644 index 00000000..047133d5 --- /dev/null +++ b/src/output/output.cpp @@ -0,0 +1,493 @@ +#include <sys/types.h> +#include <sys/stat.h> +#include <fcntl.h> +#include <sys/wait.h> +#include <unistd.h> +#include <semaphore.h> +#include <iterator> //std::distance + +#include <mist/stream.h> +#include <mist/defines.h> +#include <mist/http_parser.h> +#include <mist/timing.h> +#include "output.h" + +namespace Mist { + Util::Config * Output::config = NULL; + JSON::Value Output::capa = JSON::Value(); + + int getDTSCLen(char * mapped, long long int offset){ + return ntohl(((int*)(mapped + offset))[1]); + } + + long long int getDTSCTime(char * mapped, long long int offset){ + char * timePoint = mapped + offset + 12; + return ((long long int)timePoint[0] << 56) | ((long long int)timePoint[1] << 48) | ((long long int)timePoint[2] << 40) | ((long long int)timePoint[3] << 32) | ((long long int)timePoint[4] << 24) | ((long long int)timePoint[5] << 16) | ((long long int)timePoint[6] << 8) | timePoint[7]; + } + + Output::Output(Socket::Connection & conn) : myConn(conn) { + firstTime = 0; + parseData = false; + wantRequest = true; + isInitialized = false; + isBlocking = false; + lastStats = 0; + maxSkipAhead = 7500; + minSkipAhead = 5000; + realTime = 1000; + if (myConn){ + setBlocking(true); + }else{ + DEBUG_MSG(DLVL_WARN, "Warning: MistOut created with closed socket!"); + } + sentHeader = false; + } + + void Output::setBlocking(bool blocking){ + isBlocking = blocking; + myConn.setBlocking(isBlocking); + } + + Output::~Output(){ + statsPage.finish(); + playerConn.finish(); + } + + void Output::updateMeta(){ + unsigned int i = 0; + //read metadata from page to myMeta variable + JSON::Value jsonMeta; + JSON::fromDTMI((const unsigned char*)streamIndex.mapped + 8, streamIndex.len - 8, i, jsonMeta); + myMeta = DTSC::Meta(jsonMeta); + } + + /// Called when stream initialization has failed. + /// The standard implementation will set isInitialized to false and close the client connection, + /// thus causing the process to exit cleanly. + void Output::onFail(){ + isInitialized = false; + myConn.close(); + } + + void Output::initialize(){ + if (isInitialized){ + return; + } + if (streamIndex.mapped){ + return; + } + isInitialized = true; + streamIndex.init(streamName,0,false,false); + if (!streamIndex.mapped){ + sem_t * waiting = sem_open(std::string("/wait_" + streamName).c_str(), O_CREAT | O_RDWR, ACCESSPERMS, 0); + Util::Stream::getStream(streamName); + if (waiting == SEM_FAILED){ + DEBUG_MSG(DLVL_FAIL, "Failed to open semaphore - cancelling"); + onFail(); + return; + } + #ifdef __APPLE__ + unsigned int timeout = 0; + while (++timeout < 300 && sem_trywait(waiting) == -1 && (errno == EINTR || errno == EAGAIN) ){ + Util::sleep(100); + } + #else + struct timespec ts; + ts.tv_sec = Util::epoch() + 30; + ts.tv_nsec = 0; + while (sem_timedwait(waiting, &ts) == -1 && errno == EINTR) continue; + #endif + sem_post(waiting); + sem_close(waiting); + streamIndex.init(streamName,0); + } + if (!streamIndex.mapped){ + DEBUG_MSG(DLVL_FAIL, "Could not connect to server for %s\n", streamName.c_str()); + onFail(); + return; + } + statsPage = IPC::sharedClient("statistics", 88, true); + playerConn = IPC::sharedClient(streamName + "_users", 30, true); + + updateMeta(); + + //check which tracks don't actually exist + std::set<long unsigned int> toRemove; + for (std::set<long unsigned int>::iterator it = selectedTracks.begin(); it != selectedTracks.end(); it++){ + if (!myMeta.tracks.count(*it)){ + toRemove.insert(*it); + } + } + //remove those from selectedtracks + for (std::set<long unsigned int>::iterator it = toRemove.begin(); it != toRemove.end(); it++){ + selectedTracks.erase(*it); + } + + //loop through all codec combinations, count max simultaneous active + unsigned int bestSoFar = 0; + unsigned int bestSoFarCount = 0; + unsigned int index = 0; + for (JSON::ArrIter it = capa["codecs"].ArrBegin(); it != capa["codecs"].ArrEnd(); it++){ + unsigned int genCounter = 0; + unsigned int selCounter = 0; + if ((*it).size() > 0){ + for (JSON::ArrIter itb = (*it).ArrBegin(); itb != (*it).ArrEnd(); itb++){ + if ((*itb).size() > 0){ + bool found = false; + for (JSON::ArrIter itc = (*itb).ArrBegin(); itc != (*itb).ArrEnd() && !found; itc++){ + for (std::set<long unsigned int>::iterator itd = selectedTracks.begin(); itd != selectedTracks.end(); itd++){ + if (myMeta.tracks[*itd].codec == (*itc).asStringRef()){ + selCounter++; + found = true; + break; + } + } + if (!found){ + for (std::map<int,DTSC::Track>::iterator trit = myMeta.tracks.begin(); trit != myMeta.tracks.end(); trit++){ + if (trit->second.codec == (*itc).asStringRef()){ + genCounter++; + found = true; + break; + } + } + } + } + } + } + if (selCounter == selectedTracks.size()){ + if (selCounter + genCounter > bestSoFarCount){ + bestSoFarCount = selCounter + genCounter; + bestSoFar = index; + DEBUG_MSG(DLVL_HIGH, "Match (%u/%u): %s", selCounter, selCounter+genCounter, (*it).toString().c_str()); + } + }else{ + DEBUG_MSG(DLVL_VERYHIGH, "Not a match for currently selected tracks: %s", (*it).toString().c_str()); + } + } + index++; + } + + DEBUG_MSG(DLVL_MEDIUM, "Trying to fill: %s", capa["codecs"][bestSoFar].toString().c_str()); + //try to fill as many codecs simultaneously as possible + if (capa["codecs"][bestSoFar].size() > 0){ + for (JSON::ArrIter itb = capa["codecs"][bestSoFar].ArrBegin(); itb != capa["codecs"][bestSoFar].ArrEnd(); itb++){ + if ((*itb).size() > 0){ + bool found = false; + for (JSON::ArrIter itc = (*itb).ArrBegin(); itc != (*itb).ArrEnd() && !found; itc++){ + for (std::set<long unsigned int>::iterator itd = selectedTracks.begin(); itd != selectedTracks.end(); itd++){ + if (myMeta.tracks[*itd].codec == (*itc).asStringRef()){ + found = true; + break; + } + } + if (!found){ + for (std::map<int,DTSC::Track>::iterator trit = myMeta.tracks.begin(); trit != myMeta.tracks.end(); trit++){ + if (trit->second.codec == (*itc).asStringRef()){ + selectedTracks.insert(trit->first); + found = true; + break; + } + } + } + } + } + } + } + + #if DEBUG >= DLVL_MEDIUM + //print the selected tracks + std::stringstream selected; + for (std::set<long unsigned int>::iterator it = selectedTracks.begin(); it != selectedTracks.end(); it++){ + if (it != selectedTracks.begin()){ + selected << ", "; + } + selected << (*it); + } + DEBUG_MSG(DLVL_MEDIUM, "Selected tracks: %s", selected.str().c_str()); + #endif + + unsigned int firstms = 0x0; + for (std::set<long unsigned int>::iterator it = selectedTracks.begin(); it != selectedTracks.end(); it++){ + lastKeyTime[*it] = 0xFFFFFFFF; + if (myMeta.tracks[*it].firstms > firstms){ + firstms = myMeta.tracks[*it].firstms; + } + } + if (myMeta.live){ + if (firstms < 5000){ + firstms = 0; + } + seek(firstms); + }else{ + seek(0); + } + } + + /// Clears the buffer, sets parseData to false, and generally makes not very much happen at all. + void Output::stop(){ + buffer.clear(); + parseData = false; + } + + unsigned int Output::getKeyForTime(long unsigned int trackId, long long timeStamp){ + unsigned int keyNo = 0; + for (std::deque<DTSC::Key>::iterator it = myMeta.tracks[trackId].keys.begin(); it != myMeta.tracks[trackId].keys.end(); it++){ + if (it->getTime() <= timeStamp){ + keyNo = it->getNumber(); + }else{ + break; + } + } + return keyNo; + } + + void Output::loadPageForKey(long unsigned int trackId, long long int keyNum){ + if (keyNum >= myMeta.tracks[trackId].keys.rbegin()->getNumber()){ + //curPages.erase(trackId); + return; + } + DEBUG_MSG(DLVL_MEDIUM, "Loading track %lu, containing key %lld", trackId, keyNum); + int pageNum = -1; + int keyAmount = -1; + unsigned int timeout = 0; + if (!indexPages.count(trackId)){ + char id[100]; + sprintf(id, "%s%lu", streamName.c_str(), trackId); + indexPages[trackId].init(id, 8192); + } + while (pageNum == -1 || keyAmount == -1){ + for (int i = 0; i < indexPages[trackId].len / 8; i++){ + long tmpKey = ntohl(((((long long int*)indexPages[trackId].mapped)[i]) >> 32) & 0xFFFFFFFF); + long amountKey = ntohl((((long long int*)indexPages[trackId].mapped)[i]) & 0xFFFFFFFF); + if (tmpKey <= keyNum && (tmpKey + amountKey) > keyNum){ + pageNum = tmpKey; + keyAmount = amountKey; + break; + } + } + if (pageNum == -1 || keyAmount == -1){ + if (!timeout){ + DEBUG_MSG(DLVL_DEVEL, "Requesting/waiting for page that has key %lu:%lld...", trackId, keyNum); + } + if (timeout++ > 100){ + DEBUG_MSG(DLVL_FAIL, "Timeout while waiting for requested page. Aborting."); + curPages.erase(trackId); + return; + } + nxtKeyNum[trackId] = keyNum-1; + stats(); + Util::sleep(100); + } + } + + nxtKeyNum[trackId] = pageNum; + + if (currKeyOpen.count(trackId) && currKeyOpen[trackId] == pageNum){ + return; + } + char id[100]; + sprintf(id, "%s%lu_%d", streamName.c_str(), trackId, pageNum); + curPages[trackId].init(std::string(id),0); + if (!(curPages[trackId].mapped)){ + DEBUG_MSG(DLVL_FAIL, "(%d) Initializing page %s failed", getpid(), curPages[trackId].name.c_str()); + return; + } + currKeyOpen[trackId] = pageNum; + } + + /// Prepares all tracks from selectedTracks for seeking to the specified ms position. + /// \todo Make this actually seek, instead of always loading position zero. + void Output::seek(long long pos){ + firstTime = Util::getMS() - pos; + if (!isInitialized){ + initialize(); + } + buffer.clear(); + currentPacket.null(); + updateMeta(); + for (std::set<long unsigned int>::iterator it = selectedTracks.begin(); it != selectedTracks.end(); it++){ + seek(*it, pos); + } + } + + bool Output::seek(int tid, long long pos, bool getNextKey){ + loadPageForKey(tid, getKeyForTime(tid, pos) + (getNextKey?1:0)); + if (!curPages.count(tid) || !curPages[tid].mapped){ + DEBUG_MSG(DLVL_DEVEL, "Aborting seek to %llims in track %d, not available.", pos, tid); + return false; + } + sortedPageInfo tmp; + tmp.tid = tid; + tmp.offset = 0; + DTSC::Packet tmpPack; + tmpPack.reInit(curPages[tid].mapped + tmp.offset, 0, true); + tmp.time = tmpPack.getTime(); + while ((long long)tmp.time < pos && tmpPack){ + tmp.offset += tmpPack.getDataLen(); + tmpPack.reInit(curPages[tid].mapped + tmp.offset, 0, true); + tmp.time = tmpPack.getTime(); + } + if (tmpPack){ + buffer.insert(tmp); + return true; + }else{ + //don't print anything for empty packets - not sign of corruption, just unfinished stream. + if (curPages[tid].mapped[tmp.offset] != 0){ + DEBUG_MSG(DLVL_FAIL, "Noes! Couldn't find packet on track %d because of some kind of corruption error or somesuch.", tid); + }else{ + DEBUG_MSG(DLVL_FAIL, "Track %d no data (key %u) - waiting...", tid, getKeyForTime(tid, pos) + (getNextKey?1:0)); + } + return false; + } + } + + int Output::run() { + bool firstData = true;//only the first time, we call OnRequest if there's data buffered already. + DEBUG_MSG(DLVL_MEDIUM, "MistOut client handler started"); + while (myConn.connected() && (wantRequest || parseData)){ + stats(); + if (wantRequest){ + if ((firstData && myConn.Received().size()) || myConn.spool()){ + firstData = false; + DEBUG_MSG(DLVL_VERYHIGH, "(%d) OnRequest", getpid()); + onRequest(); + }else{ + if (!isBlocking && !parseData){ + Util::sleep(500); + } + } + } + if (parseData){ + if (!isInitialized){ + initialize(); + } + if ( !sentHeader){ + DEBUG_MSG(DLVL_VERYHIGH, "(%d) SendHeader", getpid()); + sendHeader(); + } + prepareNext(); + if (currentPacket){ + sendNext(); + }else{ + if (!onFinish()){ + break; + } + } + } + } + DEBUG_MSG(DLVL_MEDIUM, "MistOut client handler shutting down: %s, %s, %s", myConn.connected() ? "conn_active" : "conn_closed", wantRequest ? "want_request" : "no_want_request", parseData ? "parsing_data" : "not_parsing_data"); + myConn.close(); + return 0; + } + + void Output::prepareNext(){ + static unsigned int emptyCount = 0; + if (!buffer.size()){ + currentPacket.null(); + DEBUG_MSG(DLVL_DEVEL, "Buffer completely played out"); + return; + } + sortedPageInfo nxt = *(buffer.begin()); + buffer.erase(buffer.begin()); + + DEBUG_MSG(DLVL_VERYHIGH, "Loading track %u (next=%lu), part @ %u/%lld", nxt.tid, nxtKeyNum[nxt.tid], nxt.offset, curPages[nxt.tid].len); + + if (nxt.offset >= curPages[nxt.tid].len){ + loadPageForKey(nxt.tid, ++nxtKeyNum[nxt.tid]); + nxt.offset = 0; + } + + if (!curPages.count(nxt.tid) || !curPages[nxt.tid].mapped){ + //mapping failure? Drop this track and go to next. + //not an error - usually means end of stream. + DEBUG_MSG(DLVL_DEVEL, "Track %u no page - dropping track.", nxt.tid); + prepareNext(); + return; + } + + if (!memcmp(curPages[nxt.tid].mapped + nxt.offset, "\000\000\000\000", 4)){ + if (!currentPacket.getTime()){ + DEBUG_MSG(DLVL_DEVEL, "Timeless empty packet on track %u - dropping track.", nxt.tid); + prepareNext(); + return; + } + Util::sleep(500); + updateMeta(); + if (myMeta && ++emptyCount < 20){ + if (!seek(nxt.tid, currentPacket.getTime(), true)){ + buffer.insert(nxt); + } + }else{ + DEBUG_MSG(DLVL_DEVEL, "Empty packet on track %u - could not reload, dropping track.", nxt.tid); + } + prepareNext(); + return; + } + currentPacket.reInit(curPages[nxt.tid].mapped + nxt.offset, 0, true); + if (currentPacket){ + nxtKeyNum[nxt.tid] = getKeyForTime(nxt.tid, currentPacket.getTime()); + emptyCount = 0; + } + nxt.offset += currentPacket.getDataLen(); + if (realTime && !myMeta.live){ + while (nxt.time > (Util::getMS() - firstTime + maxSkipAhead)*1000/realTime) { + Util::sleep(nxt.time - (Util::getMS() - firstTime + minSkipAhead)*1000/realTime); + } + } + if (curPages[nxt.tid]){ + if (nxt.offset < curPages[nxt.tid].len){ + nxt.time = getDTSCTime(curPages[nxt.tid].mapped, nxt.offset); + } + buffer.insert(nxt); + } + playerConn.keepAlive(); + } + + void Output::stats(){ + if (!statsPage.getData()){ + return; + } + unsigned long long int now = Util::epoch(); + if (now != lastStats){ + lastStats = now; + IPC::statExchange tmpEx(statsPage.getData()); + tmpEx.now(now); + tmpEx.host(myConn.getHost()); + tmpEx.streamName(streamName); + tmpEx.connector(capa["name"].asString()); + tmpEx.up(myConn.dataUp()); + tmpEx.down(myConn.dataDown()); + tmpEx.time(now - myConn.connTime()); + statsPage.keepAlive(); + } + int tNum = 0; + for (std::set<unsigned long>::iterator it = selectedTracks.begin(); it != selectedTracks.end() && tNum < 5; it++){ + char thisData[6]; + thisData[0] = ((*it >> 24) & 0xFF); + thisData[1] = ((*it >> 16) & 0xFF); + thisData[2] = ((*it >> 8) & 0xFF); + thisData[3] = ((*it) & 0xFF); + thisData[4] = ((nxtKeyNum[*it] >> 8) & 0xFF); + thisData[5] = ((nxtKeyNum[*it]) & 0xFF); + memcpy(playerConn.getData() + (6 * tNum), thisData, 6); + tNum ++; + playerConn.keepAlive(); + } + if (tNum >= 5){ + DEBUG_MSG(DLVL_WARN, "Too many tracks selected, using only first 5"); + } + } + + void Output::onRequest(){ + //simply clear the buffer, we don't support any kind of input by default + myConn.Received().clear(); + wantRequest = false; + } + + void Output::sendHeader(){ + //just set the sentHeader bool to true, by default + sentHeader = true; + } + +} + diff --git a/src/output/output.h b/src/output/output.h new file mode 100644 index 00000000..f8b5c02a --- /dev/null +++ b/src/output/output.h @@ -0,0 +1,98 @@ +#include <set> +#include <cstdlib> +#include <map> +#include <mist/config.h> +#include <mist/json.h> +#include <mist/flv_tag.h> +#include <mist/timing.h> +#include <mist/dtsc.h> +#include <mist/socket.h> +#include <mist/shared_memory.h> + +namespace Mist { + + /// This struct keeps packet information sorted in playback order, so the + /// Mist::Output class knows when to buffer which packet. + struct sortedPageInfo{ + bool operator < (const sortedPageInfo & rhs) const { + if (time < rhs.time){ + return true; + } + return (time == rhs.time && tid < rhs.tid); + } + int tid; + long long unsigned int time; + unsigned int offset; + }; + + /// The output class is intended to be inherited by MistOut process classes. + /// It contains all generic code and logic, while the child classes implement + /// anything specific to particular protocols or containers. + /// It contains several virtual functions, that may be overridden to "hook" into + /// the streaming process at those particular points, simplifying child class + /// logic and implementation details. + class Output { + public: + //constructor and destructor + Output(Socket::Connection & conn); + virtual ~Output(); + //static members for initialization and capabilities + static void init(Util::Config * cfg) {} + static JSON::Value capa; + //non-virtual generic functions + int run(); + void stats(); + void seek(long long pos); + bool seek(int tid, long long pos, bool getNextKey = false); + void stop(); + void setBlocking(bool blocking); + void updateMeta(); + //virtuals. The optional virtuals have default implementations that do as little as possible. + virtual void sendNext() {}//REQUIRED! Others are optional. + virtual void prepareNext(); + virtual void onRequest(); + virtual bool onFinish(){return false;} + virtual void initialize(); + virtual void sendHeader(); + virtual void onFail(); + private://these *should* not be messed with in child classes. + std::map<unsigned long, unsigned int> currKeyOpen; + void loadPageForKey(long unsigned int trackId, long long int keyNum); + bool isBlocking;///< If true, indicates that myConn is blocking. + unsigned int lastStats;///<Time of last sending of stats. + IPC::sharedClient statsPage;///< Shared memory used for statistics reporting. + long long unsigned int firstTime;///< Time of first packet after last seek. Used for real-time sending. + std::map<unsigned long, unsigned long> nxtKeyNum;///< Contains the number of the next key, for page seeking purposes. + std::set<sortedPageInfo> buffer;///< A sorted list of next-to-be-loaded packets. + std::map<unsigned long, unsigned long> lastKeyTime;///< Stores the time of the last keyframe, for preventing duplicates + protected://these are to be messed with by child classes + unsigned int getKeyForTime(long unsigned int trackId, long long timeStamp); + IPC::sharedPage streamIndex;///< Shared memory used for metadata + std::map<int,IPC::sharedPage> indexPages;///< Maintains index pages of each track, holding information about available pages with DTSC packets. + std::map<int,IPC::sharedPage> curPages;///< Holds the currently used pages with DTSC packets for each track. + /// \todo Privitize keyTimes + IPC::sharedClient playerConn;///< Shared memory used for connection to MistIn process. + std::map<int,std::set<int> > keyTimes;///< Per-track list of keyframe times, for keyframe detection. + //static member for initialization + static Util::Config * config;///< Static, global configuration for the MistOut process + + //stream delaying variables + unsigned int maxSkipAhead;///< Maximum ms that we will go ahead of the intended timestamps. + unsigned int minSkipAhead;///< Minimum ms that we will go ahead of the intended timestamps. + unsigned int realTime;///< Playback speed times 1000 (1000 == 1.0X). Zero is infinite. + + //Read/write status variables + Socket::Connection & myConn;///< Connection to the client. + std::string streamName;///< Name of the stream that will be opened by initialize() + std::set<unsigned long> selectedTracks; ///< Tracks that are selected for playback + bool wantRequest;///< If true, waits for a request. + bool parseData;///< If true, triggers initalization if not already done, sending of header, sending of packets. + bool isInitialized;///< If false, triggers initialization if parseData is true. + bool sentHeader;///< If false, triggers sendHeader if parseData is true. + + //Read-only stream data variables + DTSC::Packet currentPacket;///< The packet that is ready for sending now. + DTSC::Meta myMeta;///< Up to date stream metadata + }; + +} diff --git a/src/output/output_hds.cpp b/src/output/output_hds.cpp new file mode 100644 index 00000000..43d129d0 --- /dev/null +++ b/src/output/output_hds.cpp @@ -0,0 +1,266 @@ +#include "output_hds.h" +#include <mist/defines.h> +#include <mist/http_parser.h> +#include <mist/stream.h> +#include <unistd.h> + +#include <mist/amf.h> +#include <mist/mp4_adobe.h> + +namespace Mist { + + void OutHDS::getTracks(){ + /// \todo Why do we have only one audio track option? + videoTracks.clear(); + audioTrack = 0; + for (std::map<int,DTSC::Track>::iterator it = myMeta.tracks.begin(); it != myMeta.tracks.end(); it++){ + if (it->second.codec == "H264" || it->second.codec == "H263" || it->second.codec == "VP6"){ + videoTracks.insert(it->first); + } + if (it->second.codec == "AAC" || it->second.codec == "MP3"){ + audioTrack = it->first; + } + } + } + + ///\brief Builds a bootstrap for use in HTTP Dynamic streaming. + ///\param tid The track this bootstrap is generated for. + ///\return The generated bootstrap. + std::string OutHDS::dynamicBootstrap(int tid){ + updateMeta(); + std::string empty; + + MP4::ASRT asrt; + asrt.setUpdate(false); + asrt.setVersion(1); + //asrt.setQualityEntry(empty, 0); + if (myMeta.live){ + asrt.setSegmentRun(1, 4294967295ul, 0); + }else{ + asrt.setSegmentRun(1, myMeta.tracks[tid].keys.size(), 0); + } + + MP4::AFRT afrt; + afrt.setUpdate(false); + afrt.setVersion(1); + afrt.setTimeScale(1000); + //afrt.setQualityEntry(empty, 0); + MP4::afrt_runtable afrtrun; + int i = 0; + for (std::deque<DTSC::Key>::iterator it = myMeta.tracks[tid].keys.begin(); it != myMeta.tracks[tid].keys.end(); it++){ + if (it->getLength()){ + afrtrun.firstFragment = it->getNumber(); + afrtrun.firstTimestamp = it->getTime(); + afrtrun.duration = it->getLength(); + afrt.setFragmentRun(afrtrun, i); + i++; + } + } + + MP4::ABST abst; + abst.setVersion(1); + abst.setBootstrapinfoVersion(1); + abst.setProfile(0); + abst.setUpdate(false); + abst.setTimeScale(1000); + abst.setLive(myMeta.live); + abst.setCurrentMediaTime(myMeta.tracks[tid].lastms); + abst.setSmpteTimeCodeOffset(0); + abst.setMovieIdentifier(streamName); + abst.setSegmentRunTable(asrt, 0); + abst.setFragmentRunTable(afrt, 0); + + DEBUG_MSG(DLVL_VERYHIGH, "Sending bootstrap: %s", abst.toPrettyString(0).c_str()); + return std::string((char*)abst.asBox(), (int)abst.boxedSize()); + } + + ///\brief Builds an index file for HTTP Dynamic streaming. + ///\return The index file for HTTP Dynamic Streaming. + std::string OutHDS::dynamicIndex(){ + getTracks(); + std::stringstream Result; + Result << "<?xml version=\"1.0\" encoding=\"utf-8\"?>" << std::endl; + Result << " <manifest xmlns=\"http://ns.adobe.com/f4m/1.0\">" << std::endl; + Result << " <id>" << streamName << "</id>" << std::endl; + Result << " <mimeType>video/mp4</mimeType>" << std::endl; + Result << " <deliveryType>streaming</deliveryType>" << std::endl; + if (myMeta.vod){ + Result << " <duration>" << myMeta.tracks[*videoTracks.begin()].lastms / 1000 << ".000</duration>" << std::endl; + Result << " <streamType>recorded</streamType>" << std::endl; + }else{ + Result << " <duration>0.00</duration>" << std::endl; + Result << " <streamType>live</streamType>" << std::endl; + } + for (std::set<int>::iterator it = videoTracks.begin(); it != videoTracks.end(); it++){ + Result << " <bootstrapInfo " + "profile=\"named\" " + "id=\"boot" << (*it) << "\" " + "url=\"" << (*it) << ".abst\">" + "</bootstrapInfo>" << std::endl; + Result << " <media " + "url=\"" << (*it) << "-\" " + "bitrate=\"" << myMeta.tracks[(*it)].bps * 8 << "\" " + "bootstrapInfoId=\"boot" << (*it) << "\" " + "width=\"" << myMeta.tracks[(*it)].width << "\" " + "height=\"" << myMeta.tracks[(*it)].height << "\">" << std::endl; + Result << " <metadata>AgAKb25NZXRhRGF0YQMAAAk=</metadata>" << std::endl; + Result << " </media>" << std::endl; + } + Result << "</manifest>" << std::endl; + DEBUG_MSG(DLVL_HIGH, "Sending manifest: %s", Result.str().c_str()); + return Result.str(); + } //BuildManifest + + OutHDS::OutHDS(Socket::Connection & conn) : Output(conn) { + audioTrack = 0; + playUntil = 0; + } + + void OutHDS::onFail(){ + HTTP_S.Clean(); //make sure no parts of old requests are left in any buffers + HTTP_S.SetBody("Stream not found. Sorry, we tried."); + HTTP_S.SendResponse("404", "Stream not found", myConn); + Output::onFail(); + } + + OutHDS::~OutHDS() {} + + void OutHDS::init(Util::Config * cfg){ + capa["desc"] = "Enables HTTP protocol Adobe-specific dynamic streaming (also known as HDS)."; + capa["deps"] = "HTTP"; + capa["url_rel"] = "/dynamic/$/manifest.f4m"; + capa["url_prefix"] = "/dynamic/$/"; + capa["socket"] = "http_hds"; + capa["codecs"][0u][0u].append("H264"); + capa["codecs"][0u][0u].append("H263"); + capa["codecs"][0u][0u].append("VP6"); + capa["codecs"][0u][1u].append("AAC"); + capa["codecs"][0u][1u].append("MP3"); + capa["methods"][0u]["handler"] = "http"; + capa["methods"][0u]["type"] = "flash/11"; + capa["methods"][0u]["priority"] = 7ll; + cfg->addBasicConnectorOptions(capa); + config = cfg; + } + + void OutHDS::sendNext(){ + if (currentPacket.getTime() >= playUntil){ + DEBUG_MSG(DLVL_DEVEL, "(%d) Done sending fragment", getpid() ); + stop(); + wantRequest = true; + HTTP_S.Chunkify("", 0, myConn); + return; + } + tag.DTSCLoader(currentPacket, myMeta.tracks[currentPacket.getTrackId()]); + HTTP_S.Chunkify(tag.data, tag.len, myConn); + } + + void OutHDS::onRequest(){ + HTTP_R.Clean(); + while (HTTP_R.Read(myConn)){ + DEBUG_MSG(DLVL_DEVEL, "Received request: %s", HTTP_R.getUrl().c_str()); + if (HTTP_R.url.find(".abst") != std::string::npos){ + myConn.setHost(HTTP_R.GetHeader("X-Origin")); + streamName = HTTP_R.GetHeader("X-Stream"); + std::string streamID = HTTP_R.url.substr(streamName.size() + 10); + streamID = streamID.substr(0, streamID.find(".abst")); + HTTP_S.Clean(); + HTTP_S.SetBody(dynamicBootstrap(atoll(streamID.c_str()))); + HTTP_S.SetHeader("Content-Type", "binary/octet"); + HTTP_S.SetHeader("Cache-Control", "no-cache"); + HTTP_S.SendResponse("200", "OK", myConn); + HTTP_R.Clean(); //clean for any possible next requests + continue; + } + if (HTTP_R.url.find("f4m") == std::string::npos){ + myConn.setHost(HTTP_R.GetHeader("X-Origin")); + streamName = HTTP_R.GetHeader("X-Stream"); + initialize(); + std::string tmp_qual = HTTP_R.url.substr(HTTP_R.url.find("/", 10) + 1); + unsigned int tid; + unsigned int fragNum; + tid = atoi(tmp_qual.substr(0, tmp_qual.find("Seg") - 1).c_str()); + int temp; + temp = HTTP_R.url.find("Seg") + 3; + temp = HTTP_R.url.find("Frag") + 4; + fragNum = atoi(HTTP_R.url.substr(temp).c_str()); + DEBUG_MSG(DLVL_MEDIUM, "Video track %d, fragment %d\n", tid, fragNum); + if (!audioTrack){getTracks();} + unsigned int mstime = 0; + unsigned int mslen = 0; + for (std::deque<DTSC::Key>::iterator it = myMeta.tracks[tid].keys.begin(); it != myMeta.tracks[tid].keys.end(); it++){ + if (it->getNumber() >= fragNum){ + mstime = it->getTime(); + mslen = it->getLength(); + if (myMeta.live){ + if (it == myMeta.tracks[tid].keys.end() - 2){ + HTTP_S.Clean(); + HTTP_S.SetBody("Proxy, re-request this in a second or two.\n"); + HTTP_S.SendResponse("208", "Ask again later", myConn); + HTTP_R.Clean(); //clean for any possible next requests + std::cout << "Fragment after fragment " << fragNum << " not available yet" << std::endl; + /* + ///\todo patch this back in? + if (ss.spool()){ + while (Strm.parsePacket(ss.Received())){} + } + */ + } + } + break; + } + } + if (HTTP_R.url == "/"){continue;}//Don't continue, but continue instead. + if (myMeta.live){ + if (mstime == 0 && fragNum > 1){ + HTTP_S.Clean(); + HTTP_S.SetBody("The requested fragment is no longer kept in memory on the server and cannot be served.\n"); + HTTP_S.SendResponse("412", "Fragment out of range", myConn); + HTTP_R.Clean(); //clean for any possible next requests + std::cout << "Fragment " << fragNum << " too old" << std::endl; + continue; + } + } + selectedTracks.clear(); + selectedTracks.insert(tid); + selectedTracks.insert(audioTrack); + seek(mstime); + playUntil = mstime + mslen; + + HTTP_S.Clean(); + HTTP_S.SetHeader("Content-Type", "video/mp4"); + HTTP_S.StartResponse(HTTP_R, myConn); + //send the bootstrap + std::string bootstrap = dynamicBootstrap(tid); + HTTP_S.Chunkify(bootstrap, myConn); + //send a zero-size mdat, meaning it stretches until end of file. + HTTP_S.Chunkify("\000\000\000\000mdat", 8, myConn); + //send init data, if needed. + if (audioTrack > 0){ + tag.DTSCAudioInit(myMeta.tracks[audioTrack]); + tag.tagTime(mstime); + HTTP_S.Chunkify(tag.data, tag.len, myConn); + } + if (tid > 0){ + tag.DTSCVideoInit(myMeta.tracks[tid]); + tag.tagTime(mstime); + HTTP_S.Chunkify(tag.data, tag.len, myConn); + } + parseData = true; + wantRequest = false; + }else{ + myConn.setHost(HTTP_R.GetHeader("X-Origin")); + streamName = HTTP_R.GetHeader("X-Stream"); + initialize(); + std::stringstream tmpstr; + myMeta.toPrettyString(tmpstr); + HTTP_S.Clean(); + HTTP_S.SetHeader("Content-Type", "text/xml"); + HTTP_S.SetHeader("Cache-Control", "no-cache"); + HTTP_S.SetBody(dynamicIndex()); + HTTP_S.SendResponse("200", "OK", myConn); + } + HTTP_R.Clean(); //clean for any possible next requests + } + } +} diff --git a/src/output/output_hds.h b/src/output/output_hds.h new file mode 100644 index 00000000..89732e3b --- /dev/null +++ b/src/output/output_hds.h @@ -0,0 +1,30 @@ +#include "output.h" +#include <mist/http_parser.h> +#include <mist/ts_packet.h> +#include <mist/mp4.h> +#include <mist/mp4_generic.h> + +namespace Mist { + class OutHDS : public Output { + public: + OutHDS(Socket::Connection & conn); + ~OutHDS(); + static void init(Util::Config * cfg); + + void onRequest(); + void onFail(); + void sendNext(); + protected: + void getTracks(); + std::string dynamicBootstrap(int tid); + std::string dynamicIndex(); + HTTP::Parser HTTP_S; + HTTP::Parser HTTP_R; + std::set<int> videoTracks;///<< Holds valid video tracks for playback + long long int audioTrack;///<< Holds audio track ID for playback + long long unsigned int playUntil; + FLV::Tag tag; + }; +} + +typedef Mist::OutHDS mistOut; diff --git a/src/output/output_hls.cpp b/src/output/output_hls.cpp new file mode 100644 index 00000000..b1f3dd22 --- /dev/null +++ b/src/output/output_hls.cpp @@ -0,0 +1,282 @@ +#include "output_hls.h" +#include <mist/defines.h> +#include <mist/http_parser.h> +#include <mist/stream.h> +#include <unistd.h> + +namespace Mist { + ///\brief Builds an index file for HTTP Live streaming. + ///\return The index file for HTTP Live Streaming. + std::string OutHLS::liveIndex(){ + std::stringstream result; + result << "#EXTM3U\r\n"; + int audioId = -1; + std::string audioName; + for (std::map<int,DTSC::Track>::iterator it = myMeta.tracks.begin(); it != myMeta.tracks.end(); it++){ + if (it->second.codec == "AAC"){ + audioId = it->first; + audioName = it->second.getIdentifier(); + break; + } + } + for (std::map<int,DTSC::Track>::iterator it = myMeta.tracks.begin(); it != myMeta.tracks.end(); it++){ + if (it->second.codec == "H264"){ + int bWidth = it->second.bps * 2; + if (audioId != -1){ + bWidth += myMeta.tracks[audioId].bps * 2; + } + result << "#EXT-X-STREAM-INF:PROGRAM-ID=1,BANDWIDTH=" << bWidth * 10 << "\r\n"; + result << it->first; + if (audioId != -1){ + result << "_" << audioId; + } + result << "/index.m3u8\r\n"; + } + } +#if DEBUG >= 8 + std::cerr << "Sending this index:" << std::endl << result.str() << std::endl; +#endif + return result.str(); + } + + std::string OutHLS::liveIndex(int tid){ + updateMeta(); + std::stringstream result; + //parse single track + int longestFragment = 0; + if (!myMeta.tracks[tid].fragments.size()){ + DEBUG_MSG(DLVL_FAIL, "liveIndex called with track %d, which has no fragments!", tid); + return ""; + } + for (std::deque<DTSC::Fragment>::iterator it = myMeta.tracks[tid].fragments.begin(); (it + 1) != myMeta.tracks[tid].fragments.end(); it++){ + if (it->getDuration() > longestFragment){ + longestFragment = it->getDuration(); + } + } + result << "#EXTM3U\r\n" + "#EXT-X-TARGETDURATION:" << (longestFragment / 1000) + 1 << "\r\n" + "#EXT-X-MEDIA-SEQUENCE:" << myMeta.tracks[tid].missedFrags << "\r\n"; + for (std::deque<DTSC::Fragment>::iterator it = myMeta.tracks[tid].fragments.begin(); it != myMeta.tracks[tid].fragments.end(); it++){ + long long int starttime = myMeta.tracks[tid].getKey(it->getNumber()).getTime(); + + if (it != (myMeta.tracks[tid].fragments.end() - 1)){ + result << "#EXTINF:" << ((it->getDuration() + 500) / 1000) << ", no desc\r\n" << starttime << "_" << it->getDuration() + starttime << ".ts\r\n"; + } + } + if ( !myMeta.live){ + result << "#EXT-X-ENDLIST\r\n"; + } +#if DEBUG >= 8 + std::cerr << "Sending this index:" << std::endl << result.str() << std::endl; +#endif + return result.str(); + } //liveIndex + + + OutHLS::OutHLS(Socket::Connection & conn) : Output(conn) { + haveAvcc = false; + } + + OutHLS::~OutHLS() {} + + void OutHLS::onFail(){ + HTTP_S.Clean(); //make sure no parts of old requests are left in any buffers + HTTP_S.SetBody("Stream not found. Sorry, we tried."); + HTTP_S.SendResponse("404", "Stream not found", myConn); + Output::onFail(); + } + + void OutHLS::init(Util::Config * cfg){ + capa["name"] = "HTTP_Live"; + capa["desc"] = "Enables HTTP protocol Apple-specific streaming (also known as HLS)."; + capa["deps"] = "HTTP"; + capa["url_rel"] = "/hls/$/index.m3u8"; + capa["url_prefix"] = "/hls/$/"; + capa["socket"] = "http_hls"; + capa["codecs"][0u][0u].append("H264"); + capa["codecs"][0u][1u].append("AAC"); + capa["methods"][0u]["handler"] = "http"; + capa["methods"][0u]["type"] = "html5/application/vnd.apple.mpegurl"; + capa["methods"][0u]["priority"] = 9ll; + cfg->addBasicConnectorOptions(capa); + config = cfg; + } + + void OutHLS::sendNext(){ + Socket::Buffer ToPack; + char * ContCounter = 0; + bool IsKeyFrame = false; + + char * dataPointer = 0; + int dataLen = 0; + currentPacket.getString("data", dataPointer, dataLen); + + if (currentPacket.getTime() >= until){ + DEBUG_MSG(DLVL_DEVEL, "(%d) Done sending fragment", getpid() ); + stop(); + wantRequest = true; + HTTP_S.Chunkify("", 0, myConn); + HTTP_S.Clean(); + return; + } + + //detect packet type, and put converted data into ToPack. + if (myMeta.tracks[currentPacket.getTrackId()].type == "video"){ + ToPack.append(TS::Packet::getPESVideoLeadIn(0ul, currentPacket.getTime() * 90)); + + IsKeyFrame = currentPacket.getInt("keyframe"); + if (IsKeyFrame){ + if (!haveAvcc){ + avccbox.setPayload(myMeta.tracks[currentPacket.getTrackId()].init); + haveAvcc = true; + } + ToPack.append(avccbox.asAnnexB()); + } + unsigned int i = 0; + while (i + 4 < (unsigned int)dataLen){ + unsigned int ThisNaluSize = (dataPointer[i] << 24) + (dataPointer[i+1] << 16) + (dataPointer[i+2] << 8) + dataPointer[i+3]; + if (ThisNaluSize + i + 4 > (unsigned int)dataLen){ + DEBUG_MSG(DLVL_WARN, "Too big NALU detected (%u > %d) - skipping!", ThisNaluSize + i + 4, dataLen); + break; + } + ToPack.append("\000\000\000\001", 4); + i += 4; + ToPack.append(dataPointer + i, ThisNaluSize); + i += ThisNaluSize; + } + ContCounter = &VideoCounter; + }else if (myMeta.tracks[currentPacket.getTrackId()].type == "audio"){ + if (AppleCompat){ + ToPack.append(TS::Packet::getPESAudioLeadIn(7+dataLen, lastVid)); + }else{ + ToPack.append(TS::Packet::getPESAudioLeadIn(7+dataLen, currentPacket.getTime() * 90)); + } + ToPack.append(TS::GetAudioHeader(dataLen, myMeta.tracks[currentPacket.getTrackId()].init)); + ToPack.append(dataPointer, dataLen); + ContCounter = &AudioCounter; + } + + bool first = true; + //send TS packets + while (ToPack.size()){ + if (PacketNumber % 42 == 0){ + HTTP_S.Chunkify(TS::PAT, 188, myConn); + HTTP_S.Chunkify(TS::PMT, 188, myConn); + PacketNumber += 2; + } + PackData.Clear(); + /// \todo Update according to sendHeader()'s generated data. + //0x100 - 1 + currentPacket.getTrackId() + if (myMeta.tracks[currentPacket.getTrackId()].type == "video"){ + PackData.PID(0x100); + }else{ + PackData.PID(0x101); + } + PackData.ContinuityCounter((*ContCounter)++); + if (first){ + PackData.UnitStart(1); + if (IsKeyFrame){ + PackData.RandomAccess(1); + PackData.PCR(currentPacket.getTime() * 27000); + } + first = false; + } + unsigned int toSend = PackData.AddStuffing(ToPack.bytes(184)); + std::string gonnaSend = ToPack.remove(toSend); + PackData.FillFree(gonnaSend); + HTTP_S.Chunkify(PackData.ToString(), 188, myConn); + PacketNumber ++; + } + } + + int OutHLS::canSeekms(unsigned int ms){ + //no tracks? Frame too new by definition. + if ( !myMeta.tracks.size()){ + return 1; + } + //loop trough all the tracks + for (std::map<int,DTSC::Track>::iterator it = myMeta.tracks.begin(); it != myMeta.tracks.end(); it++){ + //return "too late" if one track is past this point + if (ms < it->second.firstms){ + return -1; + } + //return "too early" if one track is not yet at this point + if (ms > it->second.lastms){ + return 1; + } + } + return 0; + } + + void OutHLS::onRequest(){ + while (HTTP_R.Read(myConn)){ + DEBUG_MSG(DLVL_DEVEL, "Received request: %s", HTTP_R.getUrl().c_str()); + myConn.setHost(HTTP_R.GetHeader("X-Origin")); + AppleCompat = (HTTP_R.GetHeader("User-Agent").find("Apple") != std::string::npos); + streamName = HTTP_R.GetHeader("X-Stream"); + initialize(); + if (HTTP_R.url.find(".m3u") == std::string::npos){ + std::string tmpStr = HTTP_R.getUrl(); + std::string fmtStr = "/hls/" + streamName + "/%u_%u/%llu_%llu.ts"; + long long unsigned int from; + sscanf(tmpStr.c_str(), fmtStr.c_str(), &vidTrack, &audTrack, &from, &until); + DEBUG_MSG(DLVL_DEVEL, "Vid %u, Aud %u, From %llu, Until %llu", vidTrack, audTrack, from, until); + selectedTracks.clear(); + selectedTracks.insert(vidTrack); + selectedTracks.insert(audTrack); + + if (myMeta.live){ + /// \todo Detection of out-of-range parts. + int seekable = canSeekms(from); + if (seekable < 0){ + HTTP_S.Clean(); + HTTP_S.SetBody("The requested fragment is no longer kept in memory on the server and cannot be served.\n"); + myConn.SendNow(HTTP_S.BuildResponse("412", "Fragment out of range")); + HTTP_R.Clean(); //clean for any possible next requests + DEBUG_MSG(DLVL_WARN, "Fragment @ %llu too old", from); + continue; + } + if (seekable > 0){ + HTTP_S.Clean(); + HTTP_S.SetBody("Proxy, re-request this in a second or two.\n"); + myConn.SendNow(HTTP_S.BuildResponse("208", "Ask again later")); + HTTP_R.Clean(); //clean for any possible next requests + DEBUG_MSG(DLVL_WARN, "Fragment @ %llu not available yet", from); + continue; + } + } + + seek(from); + lastVid = from * 90; + + HTTP_S.Clean(); + HTTP_S.SetHeader("Content-Type", "video/mp2t"); + HTTP_S.StartResponse(HTTP_R, myConn); + PacketNumber = 0; + parseData = true; + wantRequest = false; + }else{ + streamName = HTTP_R.GetHeader("X-Stream"); + initialize(); + std::string request = HTTP_R.url.substr(HTTP_R.url.find("/", 5) + 1); + HTTP_S.Clean(); + if (HTTP_R.url.find(".m3u8") != std::string::npos){ + HTTP_S.SetHeader("Content-Type", "audio/x-mpegurl"); + }else{ + HTTP_S.SetHeader("Content-Type", "audio/mpegurl"); + } + HTTP_S.SetHeader("Cache-Control", "no-cache"); + std::string manifest; + if (request.find("/") == std::string::npos){ + manifest = liveIndex(); + }else{ + int selectId = atoi(request.substr(0,request.find("/")).c_str()); + manifest = liveIndex(selectId); + } + HTTP_S.SetBody(manifest); + HTTP_S.SendResponse("200", "OK", myConn); + } + HTTP_R.Clean(); //clean for any possible next requests + } + } +} diff --git a/src/output/output_hls.h b/src/output/output_hls.h new file mode 100644 index 00000000..70031026 --- /dev/null +++ b/src/output/output_hls.h @@ -0,0 +1,39 @@ +#include "output.h" +#include <mist/http_parser.h> +#include <mist/ts_packet.h> +#include <mist/mp4.h> +#include <mist/mp4_generic.h> + +namespace Mist { + class OutHLS : public Output { + public: + OutHLS(Socket::Connection & conn); + ~OutHLS(); + static void init(Util::Config * cfg); + + void onRequest(); + void onFail(); + void sendNext(); + protected: + HTTP::Parser HTTP_S; + HTTP::Parser HTTP_R; + std::string liveIndex(); + std::string liveIndex(int tid); + int canSeekms(unsigned int ms); + int keysToSend; + long long int playUntil; + TS::Packet PackData; + unsigned int PacketNumber; + bool haveAvcc; + char VideoCounter; + char AudioCounter; + MP4::AVCC avccbox; + bool AppleCompat; + long long unsigned int lastVid; + long long unsigned int until; + unsigned int vidTrack; + unsigned int audTrack; + }; +} + +typedef Mist::OutHLS mistOut; diff --git a/src/output/output_hss.cpp b/src/output/output_hss.cpp new file mode 100644 index 00000000..b5146ddf --- /dev/null +++ b/src/output/output_hss.cpp @@ -0,0 +1,484 @@ +#include "output_hss.h" +#include <mist/defines.h> +#include <mist/mp4.h> +#include <mist/mp4_ms.h> +#include <mist/mp4_generic.h> +#include <mist/mp4_encryption.h> +#include <mist/base64.h> +#include <mist/http_parser.h> +#include <mist/stream.h> +#include <unistd.h> + + + +///\todo Maybe move to util? +long long unsigned int binToInt(std::string & binary) { + long long int result = 0; + for (int i = 0; i < 8; i++) { + result <<= 8; + result += binary[i]; + } + return result; +} + +std::string intToBin(long long unsigned int number) { + std::string result; + result.resize(8); + for (int i = 7; i >= 0; i--) { + result[i] = number & 0xFF; + number >>= 8; + } + return result; +} + +std::string toUTF16(std::string original) { + std::string result; + result += (char)0xFF; + result += (char)0xFE; + for (std::string::iterator it = original.begin(); it != original.end(); it++) { + result += (*it); + result += (char)0x00; + } + return result; +} + + + +namespace Mist { + OutHSS::OutHSS(Socket::Connection & conn) : Output(conn) { } + + OutHSS::~OutHSS() {} + + void OutHSS::init(Util::Config * cfg) { + capa["name"] = "HTTP_Smooth"; + capa["desc"] = "Enables HTTP protocol Microsoft-specific smooth streaming through silverlight (also known as HSS)."; + capa["deps"] = "HTTP"; + capa["url_rel"] = "/smooth/$.ism/Manifest"; + capa["url_prefix"] = "/smooth/$.ism/"; + capa["socket"] = "http_hss"; + capa["codecs"][0u][0u].append("H264"); + capa["codecs"][0u][1u].append("AAC"); + capa["methods"][0u]["handler"] = "http"; + capa["methods"][0u]["type"] = "html5/application/vnd.ms-ss"; + capa["methods"][0u]["priority"] = 9ll; + capa["methods"][0u]["nolive"] = 1; + capa["methods"][1u]["handler"] = "http"; + capa["methods"][1u]["type"] = "silverlight"; + capa["methods"][1u]["priority"] = 1ll; + capa["methods"][1u]["nolive"] = 1; + cfg->addBasicConnectorOptions(capa); + config = cfg; + } + + void OutHSS::sendNext() { + if (currentPacket.getTime() >= playUntil) { + DEBUG_MSG(DLVL_DEVEL, "(%d) Done sending fragment %d:%d", getpid(), myTrackStor, myKeyStor); + stop(); + wantRequest = true; + HTTP_S.Chunkify("", 0, myConn); + HTTP_R.Clean(); + return; + } + char * dataPointer = 0; + int len = 0; + currentPacket.getString("data", dataPointer, len); + HTTP_S.Chunkify(dataPointer, len, myConn); + } + + void OutHSS::onFail(){ + HTTP_S.Clean(); //make sure no parts of old requests are left in any buffers + HTTP_S.SetBody("Stream not found. Sorry, we tried."); + HTTP_S.SendResponse("404", "Stream not found", myConn); + Output::onFail(); + } + + int OutHSS::canSeekms(unsigned int ms) { + //no tracks? Frame too new by definition. + if (!myMeta.tracks.size()) { + DEBUG_MSG(DLVL_DEVEL, "HSS Canseek to %d returns 1 because no tracks", ms); + return 1; + } + //loop trough all selected tracks + for (std::set<unsigned long>::iterator it = selectedTracks.begin(); it != selectedTracks.end(); it++) { + //return "too late" if one track is past this point + if (ms < myMeta.tracks[*it].firstms) { + DEBUG_MSG(DLVL_DEVEL, "HSS Canseek to %d returns -1 because track %lu firstms == %d", ms, *it, myMeta.tracks[*it].firstms); + return -1; + } + //return "too early" if one track is not yet at this point + if (ms > myMeta.tracks[*it].lastms) { + DEBUG_MSG(DLVL_DEVEL, "HSS Canseek to %d returns 1 because track %lu lastms == %d", ms, *it, myMeta.tracks[*it].lastms); + return 1; + } + } + return 0; + } + + + void OutHSS::sendHeader() { + //We have a non-manifest request, parse it. + std::string Quality = HTTP_R.url.substr(HTTP_R.url.find("TrackID=", 8) + 8); + Quality = Quality.substr(0, Quality.find(")")); + std::string parseString = HTTP_R.url.substr(HTTP_R.url.find(")/") + 2); + parseString = parseString.substr(parseString.find("(") + 1); + long long int seekTime = atoll(parseString.substr(0, parseString.find(")")).c_str()) / 10000; + unsigned int tid = atoll(Quality.c_str()); + selectedTracks.clear(); + selectedTracks.insert(tid); + if (myMeta.live) { + updateMeta(); + int seekable = canSeekms(seekTime / 10000); + if (seekable == 0){ + // iff the fragment in question is available, check if the next is available too + for (std::deque<DTSC::Key>::iterator it = myMeta.tracks[tid].keys.begin(); it != myMeta.tracks[tid].keys.end(); it++){ + if (it->getTime() >= (seekTime / 10000)){ + if ((it + 1) == myMeta.tracks[tid].keys.end()){ + seekable = 1; + } + break; + } + } + } + if (seekable < 0){ + HTTP_S.Clean(); + HTTP_S.SetBody("The requested fragment is no longer kept in memory on the server and cannot be served.\n"); + myConn.SendNow(HTTP_S.BuildResponse("412", "Fragment out of range")); + HTTP_R.Clean(); //clean for any possible next requests + std::cout << "Fragment @ " << seekTime / 10000 << "ms too old (" << myMeta.tracks[tid].firstms << " - " << myMeta.tracks[tid].lastms << " ms)" << std::endl; + stop(); + wantRequest = true; + return; + } + if (seekable > 0){ + HTTP_S.Clean(); + HTTP_S.SetBody("Proxy, re-request this in a second or two.\n"); + myConn.SendNow(HTTP_S.BuildResponse("208", "Ask again later")); + HTTP_R.Clean(); //clean for any possible next requests + std::cout << "Fragment @ " << seekTime / 10000 << "ms not available yet (" << myMeta.tracks[tid].firstms << " - " << myMeta.tracks[tid].lastms << " ms)" << std::endl; + stop(); + wantRequest = true; + return; + } + } + DEBUG_MSG(DLVL_DEVEL, "(%d) Seeking to time %lld on track %d", getpid(), seekTime, tid); + seek(seekTime); + playUntil = (*(keyTimes[tid].upper_bound(seekTime))); + DEBUG_MSG(DLVL_DEVEL, "Set playUntil to %lld", playUntil); + myTrackStor = tid; + myKeyStor = seekTime; + keysToSend = 1; + //Seek to the right place and send a play-once for a single fragment. + std::stringstream sstream; + + int partOffset = 0; + int keyDur = 0; + DTSC::Key keyObj; + for (std::deque<DTSC::Key>::iterator it = myMeta.tracks[tid].keys.begin(); it != myMeta.tracks[tid].keys.end(); it++) { + if (it->getTime() >= seekTime) { + keyObj = (*it); + keyDur = it->getLength(); + std::deque<DTSC::Key>::iterator nextIt = it; + nextIt++; + if (nextIt == myMeta.tracks[tid].keys.end()) { + if (myMeta.live) { + HTTP_S.Clean(); + HTTP_S.SetBody("Proxy, re-request this in a second or two.\n"); + myConn.SendNow(HTTP_S.BuildResponse("208", "Ask again later")); + HTTP_R.Clean(); //clean for any possible next requests + std::cout << "Fragment after fragment @ " << (seekTime / 10000) << " not available yet" << std::endl; + } + } + break; + } + partOffset += it->getParts(); + } + if (HTTP_R.url == "/") { + return; //Don't continue, but continue instead. + } + /* + if (myMeta.live) { + if (mstime == 0 && (seekTime / 10000) > 1){ + HTTP_S.Clean(); + HTTP_S.SetBody("The requested fragment is no longer kept in memory on the server and cannot be served.\n"); + myConn.SendNow(HTTP_S.BuildResponse("412", "Fragment out of range")); + HTTP_R.Clean(); //clean for any possible next requests + std::cout << "Fragment @ " << (seekTime / 10000) << " too old" << std::endl; + continue; + } + } + */ + + ///\todo Select correct track (tid); + + //Wrap everything in mp4 boxes + MP4::MFHD mfhd_box; + mfhd_box.setSequenceNumber(((keyObj.getNumber() - 1) * 2) + tid);///\todo Urgent: Check this for multitrack... :P wtf... :P + + MP4::TFHD tfhd_box; + tfhd_box.setFlags(MP4::tfhdSampleFlag); + tfhd_box.setTrackID(tid); + if (myMeta.tracks[tid].type == "video") { + tfhd_box.setDefaultSampleFlags(0x00004001); + } else { + tfhd_box.setDefaultSampleFlags(0x00008002); + } + + MP4::TRUN trun_box; + trun_box.setDataOffset(42);///\todo Check if this is a placeholder, or an actually correct number + unsigned int keySize = 0; + if (myMeta.tracks[tid].type == "video") { + trun_box.setFlags(MP4::trundataOffset | MP4::trunfirstSampleFlags | MP4::trunsampleDuration | MP4::trunsampleSize | MP4::trunsampleOffsets); + } else { + trun_box.setFlags(MP4::trundataOffset | MP4::trunsampleDuration | MP4::trunsampleSize); + } + trun_box.setFirstSampleFlags(0x00004002); + for (int i = 0; i < keyObj.getParts(); i++) { + MP4::trunSampleInformation trunSample; + trunSample.sampleSize = myMeta.tracks[tid].parts[i + partOffset].getSize(); + keySize += myMeta.tracks[tid].parts[i + partOffset].getSize(); + trunSample.sampleDuration = myMeta.tracks[tid].parts[i + partOffset].getDuration() * 10000; + if (myMeta.tracks[tid].type == "video") { + trunSample.sampleOffset = myMeta.tracks[tid].parts[i + partOffset].getOffset() * 10000; + } + trun_box.setSampleInformation(trunSample, i); + } + + MP4::SDTP sdtp_box; + sdtp_box.setVersion(0); + if (myMeta.tracks[tid].type == "video") { + sdtp_box.setValue(36, 4); + for (int i = 1; i < keyObj.getParts(); i++) { + sdtp_box.setValue(20, 4 + i); + } + } else { + sdtp_box.setValue(40, 4); + for (int i = 1; i < keyObj.getParts(); i++) { + sdtp_box.setValue(40, 4 + i); + } + } + + MP4::TRAF traf_box; + traf_box.setContent(tfhd_box, 0); + traf_box.setContent(trun_box, 1); + traf_box.setContent(sdtp_box, 2); + + //If the stream is live, we want to have a fragref box if possible + //////HEREHEREHERE + if (myMeta.live) { + MP4::UUID_TrackFragmentReference fragref_box; + fragref_box.setVersion(1); + fragref_box.setFragmentCount(0); + int fragCount = 0; + for (unsigned int i = 0; fragCount < 2 && i < myMeta.tracks[tid].keys.size() - 1; i++) { + if (myMeta.tracks[tid].keys[i].getTime() > seekTime) { + DEBUG_MSG(DLVL_DEVEL, "Key %d added to fragRef box, time %ld > %lld", i, myMeta.tracks[tid].keys[i].getTime(), seekTime); + fragref_box.setTime(fragCount, myMeta.tracks[tid].keys[i].getTime() * 10000); + fragref_box.setDuration(fragCount, myMeta.tracks[tid].keys[i].getLength() * 10000); + fragref_box.setFragmentCount(++fragCount); + } + } + traf_box.setContent(fragref_box, 3); + } + + MP4::MOOF moof_box; + moof_box.setContent(mfhd_box, 0); + moof_box.setContent(traf_box, 1); + //Setting the correct offsets. + moof_box.setContent(traf_box, 1); + trun_box.setDataOffset(moof_box.boxedSize() + 8); + traf_box.setContent(trun_box, 1); + moof_box.setContent(traf_box, 1); + + HTTP_S.Clean(); + HTTP_S.SetHeader("Content-Type", "video/mp4"); + HTTP_S.StartResponse(HTTP_R, myConn); + HTTP_S.Chunkify(moof_box.asBox(), moof_box.boxedSize(), myConn); + int size = htonl(keySize + 8); + HTTP_S.Chunkify((char *)&size, 4, myConn); + HTTP_S.Chunkify("mdat", 4, myConn); + sentHeader = true; + HTTP_R.Clean(); + DEBUG_MSG(DLVL_DEVEL, "(%d) Sent full header", getpid()); + } + + + ///\brief Builds an index file for HTTP Smooth streaming. + ///\return The index file for HTTP Smooth Streaming. + std::string OutHSS::smoothIndex(){ + updateMeta(); + std::stringstream Result; + Result << "<?xml version=\"1.0\" encoding=\"utf-16\"?>\n"; + Result << "<SmoothStreamingMedia " + "MajorVersion=\"2\" " + "MinorVersion=\"0\" " + "TimeScale=\"10000000\" "; + std::deque<std::map<int, DTSC::Track>::iterator> audioIters; + std::deque<std::map<int, DTSC::Track>::iterator> videoIters; + long long int maxWidth = 0; + long long int maxHeight = 0; + long long int minWidth = 99999999; + long long int minHeight = 99999999; + for (std::map<int, DTSC::Track>::iterator it = myMeta.tracks.begin(); it != myMeta.tracks.end(); it++) { + if (it->second.codec == "AAC") { + audioIters.push_back(it); + } + if (it->second.codec == "H264") { + videoIters.push_back(it); + if (it->second.width > maxWidth) { + maxWidth = it->second.width; + } + if (it->second.width < minWidth) { + minWidth = it->second.width; + } + if (it->second.height > maxHeight) { + maxHeight = it->second.height; + } + if (it->second.height < minHeight) { + minHeight = it->second.height; + } + } + } + DEBUG_MSG(DLVL_DEVEL, "Buffer window here %lld", myMeta.bufferWindow); + if (myMeta.vod) { + Result << "Duration=\"" << (*videoIters.begin())->second.lastms << "0000\""; + } else { + Result << "Duration=\"0\" " + "IsLive=\"TRUE\" " + "LookAheadFragmentCount=\"2\" " + "DVRWindowLength=\"" << myMeta.bufferWindow << "0000\" " + "CanSeek=\"TRUE\" " + "CanPause=\"TRUE\" "; + } + Result << ">\n"; + + //Add audio entries + if (audioIters.size()) { + Result << "<StreamIndex " + "Type=\"audio\" " + "QualityLevels=\"" << audioIters.size() << "\" " + "Name=\"audio\" " + "Chunks=\"" << (*audioIters.begin())->second.keys.size() << "\" " + "Url=\"Q({bitrate},{CustomAttributes})/A({start time})\">\n"; + int index = 0; + for (std::deque<std::map<int, DTSC::Track>::iterator>::iterator it = audioIters.begin(); it != audioIters.end(); it++) { + Result << "<QualityLevel " + "Index=\"" << index << "\" " + "Bitrate=\"" << (*it)->second.bps * 8 << "\" " + "CodecPrivateData=\"" << std::hex; + for (unsigned int i = 0; i < (*it)->second.init.size(); i++) { + Result << std::setfill('0') << std::setw(2) << std::right << (int)(*it)->second.init[i]; + } + Result << std::dec << "\" " + "SamplingRate=\"" << (*it)->second.rate << "\" " + "Channels=\"2\" " + "BitsPerSample=\"16\" " + "PacketSize=\"4\" " + "AudioTag=\"255\" " + "FourCC=\"AACL\" >\n"; + Result << "<CustomAttributes>\n" + "<Attribute Name = \"TrackID\" Value = \"" << (*it)->first << "\" />" + "</CustomAttributes>"; + Result << "</QualityLevel>\n"; + index++; + } + if ((*audioIters.begin())->second.keys.size()) { + for (std::deque<DTSC::Key>::iterator it = (*audioIters.begin())->second.keys.begin(); it != (((*audioIters.begin())->second.keys.end()) - 1); it++) { + Result << "<c "; + if (it == (*audioIters.begin())->second.keys.begin()) { + Result << "t=\"" << it->getTime() * 10000 << "\" "; + } + Result << "d=\"" << it->getLength() * 10000 << "\" />\n"; + } + } + Result << "</StreamIndex>\n"; + } + //Add video entries + if (videoIters.size()) { + Result << "<StreamIndex " + "Type=\"video\" " + "QualityLevels=\"" << videoIters.size() << "\" " + "Name=\"video\" " + "Chunks=\"" << (*videoIters.begin())->second.keys.size() << "\" " + "Url=\"Q({bitrate},{CustomAttributes})/V({start time})\" " + "MaxWidth=\"" << maxWidth << "\" " + "MaxHeight=\"" << maxHeight << "\" " + "DisplayWidth=\"" << maxWidth << "\" " + "DisplayHeight=\"" << maxHeight << "\">\n"; + int index = 0; + for (std::deque<std::map<int, DTSC::Track>::iterator>::iterator it = videoIters.begin(); it != videoIters.end(); it++) { + //Add video qualities + Result << "<QualityLevel " + "Index=\"" << index << "\" " + "Bitrate=\"" << (*it)->second.bps * 8 << "\" " + "CodecPrivateData=\"" << std::hex; + MP4::AVCC avccbox; + avccbox.setPayload((*it)->second.init); + std::string tmpString = avccbox.asAnnexB(); + for (unsigned int i = 0; i < tmpString.size(); i++) { + Result << std::setfill('0') << std::setw(2) << std::right << (int)tmpString[i]; + } + Result << std::dec << "\" " + "MaxWidth=\"" << (*it)->second.width << "\" " + "MaxHeight=\"" << (*it)->second.height << "\" " + "FourCC=\"AVC1\" >\n"; + Result << "<CustomAttributes>\n" + "<Attribute Name = \"TrackID\" Value = \"" << (*it)->first << "\" />" + "</CustomAttributes>"; + Result << "</QualityLevel>\n"; + index++; + } + if ((*videoIters.begin())->second.keys.size()) { + for (std::deque<DTSC::Key>::iterator it = (*videoIters.begin())->second.keys.begin(); it != (((*videoIters.begin())->second.keys.end()) - 1); it++) { + Result << "<c "; + if (it == (*videoIters.begin())->second.keys.begin()) { + Result << "t=\"" << it->getTime() * 10000 << "\" "; + } + Result << "d=\"" << it->getLength() * 10000 << "\" />\n"; + } + } + Result << "</StreamIndex>\n"; + } + Result << "</SmoothStreamingMedia>\n"; + +#if DEBUG >= 8 + std::cerr << "Sending this manifest:" << std::endl << Result << std::endl; +#endif + return toUTF16(Result.str()); + } //smoothIndex + + + void OutHSS::onRequest() { + sentHeader = false; + while (HTTP_R.Read(myConn)) { + DEBUG_MSG(DLVL_DEVEL, "(%d) Received request %s", getpid(), HTTP_R.getUrl().c_str()); + myConn.setHost(HTTP_R.GetHeader("X-Origin")); + streamName = HTTP_R.GetHeader("X-Stream"); + initialize(); + if (HTTP_R.url.find("Manifest") != std::string::npos) { + //Manifest, direct reply + HTTP_S.Clean(); + HTTP_S.SetHeader("Content-Type", "text/xml"); + HTTP_S.SetHeader("Cache-Control", "no-cache"); + std::string manifest = smoothIndex(); + HTTP_S.SetBody(manifest); + HTTP_S.SendResponse("200", "OK", myConn); + HTTP_R.Clean(); + } else { + parseData = true; + wantRequest = false; + } + } + } + + void OutHSS::initialize() { + Output::initialize(); + for (std::map<int, DTSC::Track>::iterator it = myMeta.tracks.begin(); it != myMeta.tracks.end(); it++) { + for (std::deque<DTSC::Key>::iterator it2 = it->second.keys.begin(); it2 != it->second.keys.end(); it2++) { + keyTimes[it->first].insert(it2->getTime()); + } + } + } + + +} + diff --git a/src/output/output_hss.h b/src/output/output_hss.h new file mode 100644 index 00000000..15f2d036 --- /dev/null +++ b/src/output/output_hss.h @@ -0,0 +1,29 @@ +#include "output.h" +#include <mist/http_parser.h> + +namespace Mist { + class OutHSS : public Output { + public: + OutHSS(Socket::Connection & conn); + ~OutHSS(); + static void init(Util::Config * cfg); + + void onRequest(); + void sendNext(); + void initialize(); + void onFail(); + void sendHeader(); + protected: + HTTP::Parser HTTP_S; + HTTP::Parser HTTP_R; + JSON::Value encryption; + std::string smoothIndex(); + int canSeekms(unsigned int ms); + int keysToSend; + int myTrackStor; + int myKeyStor; + long long int playUntil; + }; +} + +typedef Mist::OutHSS mistOut; diff --git a/src/output/output_json.cpp b/src/output/output_json.cpp new file mode 100644 index 00000000..94a83e5f --- /dev/null +++ b/src/output/output_json.cpp @@ -0,0 +1,84 @@ +#include "output_json.h" +#include <mist/http_parser.h> +#include <mist/defines.h> +#include <iomanip> + +namespace Mist { + OutJSON::OutJSON(Socket::Connection & conn) : Output(conn){ + realTime = 0; + } + + OutJSON::~OutJSON() {} + + void OutJSON::init(Util::Config * cfg){ + capa["desc"] = "Enables HTTP protocol JSON streaming."; + capa["deps"] = "HTTP"; + capa["url_rel"] = "/$.json"; + capa["url_match"] = "/$.json"; + capa["url_handler"] = "http"; + capa["url_type"] = "json"; + capa["socket"] = "http_json"; + cfg->addBasicConnectorOptions(capa); + config = cfg; + } + + void OutJSON::sendNext(){ + if(!first) { + myConn.SendNow(", ", 2); + }else{ + if (jsonp == ""){ + myConn.SendNow("[", 1); + }else{ + myConn.SendNow(jsonp + "(["); + } + first = false; + } + myConn.SendNow(currentPacket.toJSON().toString()); + } + + void OutJSON::sendHeader(){ + HTTP::Parser HTTP_S; + FLV::Tag tag; + HTTP_S.SetHeader("Content-Type", "text/javascript"); + HTTP_S.protocol = "HTTP/1.0"; + myConn.SendNow(HTTP_S.BuildResponse("200", "OK")); + sentHeader = true; + } + + bool OutJSON::onFinish(){ + if (jsonp == ""){ + myConn.SendNow("]\n\n", 3); + }else{ + myConn.SendNow("]);\n\n", 5); + } + return false; + } + + void OutJSON::onRequest(){ + HTTP::Parser HTTP_R; + while (HTTP_R.Read(myConn)){ + DEBUG_MSG(DLVL_DEVEL, "Received request %s", HTTP_R.getUrl().c_str()); + first = true; + myConn.setHost(HTTP_R.GetHeader("X-Origin")); + streamName = HTTP_R.GetHeader("X-Stream"); + jsonp = ""; + if (HTTP_R.GetVar("callback") != ""){ + jsonp = HTTP_R.GetVar("callback"); + } + if (HTTP_R.GetVar("jsonp") != ""){ + jsonp = HTTP_R.GetVar("jsonp"); + } + initialize(); + for (std::map<int,DTSC::Track>::iterator it = myMeta.tracks.begin(); it != myMeta.tracks.end(); it++){ + if (it->second.type == "meta" ){ + selectedTracks.insert(it->first); + } + } + seek(0); + parseData = true; + wantRequest = false; + HTTP_R.Clean(); + } + } + +} diff --git a/src/output/output_json.h b/src/output/output_json.h new file mode 100644 index 00000000..281a0bc9 --- /dev/null +++ b/src/output/output_json.h @@ -0,0 +1,20 @@ +#include "output.h" + + +namespace Mist { + class OutJSON : public Output { + public: + OutJSON(Socket::Connection & conn); + ~OutJSON(); + static void init(Util::Config * cfg); + void onRequest(); + bool onFinish(); + void sendNext(); + void sendHeader(); + protected: + std::string jsonp; + bool first; + }; +} + +typedef Mist::OutJSON mistOut; diff --git a/src/output/output_progressive_flv.cpp b/src/output/output_progressive_flv.cpp new file mode 100644 index 00000000..36c2ec51 --- /dev/null +++ b/src/output/output_progressive_flv.cpp @@ -0,0 +1,88 @@ +#include "output_progressive_flv.h" +#include <mist/http_parser.h> +#include <mist/defines.h> + +namespace Mist { + OutProgressiveFLV::OutProgressiveFLV(Socket::Connection & conn) : Output(conn) { } + + OutProgressiveFLV::~OutProgressiveFLV() {} + + void OutProgressiveFLV::init(Util::Config * cfg){ + capa["name"] = "HTTP_Progressive_FLV"; + capa["desc"] = "Enables HTTP protocol progressive streaming."; + capa["deps"] = "HTTP"; + capa["url_rel"] = "/$.flv"; + capa["url_match"] = "/$.flv"; + capa["socket"] = "http_progressive_flv"; + capa["codecs"][0u][0u].append("H264"); + capa["codecs"][0u][0u].append("H263"); + capa["codecs"][0u][0u].append("VP6"); + capa["codecs"][0u][1u].append("AAC"); + capa["codecs"][0u][1u].append("MP3"); + capa["methods"][0u]["handler"] = "http"; + capa["methods"][0u]["type"] = "flash/7"; + capa["methods"][0u]["priority"] = 5ll; + + cfg->addBasicConnectorOptions(capa); + config = cfg; + } + + void OutProgressiveFLV::sendNext(){ + FLV::Tag tag; + bool tmp = tag.DTSCLoader(currentPacket, myMeta.tracks[currentPacket.getTrackId()]); + if (!tmp){ + DEBUG_MSG(DLVL_DEVEL, "Invalid JSON"); + } + myConn.SendNow(tag.data, tag.len); + } + + void OutProgressiveFLV::sendHeader(){ + HTTP::Parser HTTP_S; + FLV::Tag tag; + HTTP_S.SetHeader("Content-Type", "video/x-flv"); + HTTP_S.protocol = "HTTP/1.0"; + myConn.SendNow(HTTP_S.BuildResponse("200", "OK")); + myConn.SendNow(FLV::Header, 13); + tag.DTSCMetaInit(myMeta, selectedTracks); + myConn.SendNow(tag.data, tag.len); + + for (std::set<long unsigned int>::iterator it = selectedTracks.begin(); it != selectedTracks.end(); it++){ + if (myMeta.tracks[*it].type == "video"){ + tag.DTSCVideoInit(myMeta.tracks[*it]); + myConn.SendNow(tag.data, tag.len); + } + if (myMeta.tracks[*it].type == "audio"){ + tag.DTSCAudioInit(myMeta.tracks[*it]); + myConn.SendNow(tag.data, tag.len); + } + } + sentHeader = true; + } + + void OutProgressiveFLV::onFail(){ + HTTP::Parser HTTP_S; + HTTP_S.Clean(); //make sure no parts of old requests are left in any buffers + HTTP_S.SetBody("Stream not found. Sorry, we tried."); + HTTP_S.SendResponse("404", "Stream not found", myConn); + Output::onFail(); + } + + void OutProgressiveFLV::onRequest(){ + HTTP::Parser HTTP_R; + while (HTTP_R.Read(myConn)){ + DEBUG_MSG(DLVL_DEVEL, "Received request %s", HTTP_R.getUrl().c_str()); + if (HTTP_R.GetVar("audio") != ""){ + selectedTracks.insert(JSON::Value(HTTP_R.GetVar("audio")).asInt()); + } + if (HTTP_R.GetVar("video") != ""){ + selectedTracks.insert(JSON::Value(HTTP_R.GetVar("video")).asInt()); + } + myConn.setHost(HTTP_R.GetHeader("X-Origin")); + streamName = HTTP_R.GetHeader("X-Stream"); + parseData = true; + wantRequest = false; + HTTP_R.Clean(); + } + } + +} diff --git a/src/output/output_progressive_flv.h b/src/output/output_progressive_flv.h new file mode 100644 index 00000000..aee4d066 --- /dev/null +++ b/src/output/output_progressive_flv.h @@ -0,0 +1,18 @@ +#include "output.h" + + +namespace Mist { + class OutProgressiveFLV : public Output { + public: + OutProgressiveFLV(Socket::Connection & conn); + ~OutProgressiveFLV(); + static void init(Util::Config * cfg); + void onRequest(); + void sendNext(); + void onFail(); + void sendHeader(); + protected: + }; +} + +typedef Mist::OutProgressiveFLV mistOut; diff --git a/src/output/output_progressive_mp3.cpp b/src/output/output_progressive_mp3.cpp new file mode 100644 index 00000000..80d37abf --- /dev/null +++ b/src/output/output_progressive_mp3.cpp @@ -0,0 +1,65 @@ +#include "output_progressive_mp3.h" +#include <mist/http_parser.h> +#include <mist/defines.h> + +namespace Mist { + OutProgressiveMP3::OutProgressiveMP3(Socket::Connection & conn) : Output(conn) { } + + OutProgressiveMP3::~OutProgressiveMP3() {} + + void OutProgressiveMP3::init(Util::Config * cfg){ + capa["name"] = "HTTP_Progressive_MP3"; + capa["desc"] = "Enables HTTP protocol progressive streaming."; + capa["deps"] = "HTTP"; + capa["url_rel"] = "/$.mp3"; + capa["url_match"] = "/$.mp3"; + capa["socket"] = "http_progressive_mp3"; + capa["codecs"][0u][0u].append("MP3"); + capa["methods"][0u]["handler"] = "http"; + capa["methods"][0u]["type"] = "mp3"; + capa["methods"][0u]["priority"] = 8ll; + + cfg->addBasicConnectorOptions(capa); + config = cfg; + } + + void OutProgressiveMP3::sendNext(){ + char * dataPointer = 0; + int len = 0; + currentPacket.getString("data", dataPointer, len); + myConn.SendNow(dataPointer, len); + } + + void OutProgressiveMP3::sendHeader(){ + HTTP::Parser HTTP_S; + FLV::Tag tag; + HTTP_S.SetHeader("Content-Type", "audio/mpeg"); + HTTP_S.protocol = "HTTP/1.0"; + myConn.SendNow(HTTP_S.BuildResponse("200", "OK")); + sentHeader = true; + } + + void OutProgressiveMP3::onFail(){ + HTTP::Parser HTTP_S; + HTTP_S.Clean(); //make sure no parts of old requests are left in any buffers + HTTP_S.SetBody("Stream not found. Sorry, we tried."); + HTTP_S.SendResponse("404", "Stream not found", myConn); + Output::onFail(); + } + + void OutProgressiveMP3::onRequest(){ + HTTP::Parser HTTP_R; + while (HTTP_R.Read(myConn)){ + DEBUG_MSG(DLVL_DEVEL, "Received request %s", HTTP_R.getUrl().c_str()); + if (HTTP_R.GetVar("audio") != ""){ + selectedTracks.insert(JSON::Value(HTTP_R.GetVar("audio")).asInt()); + } + myConn.setHost(HTTP_R.GetHeader("X-Origin")); + streamName = HTTP_R.GetHeader("X-Stream"); + parseData = true; + wantRequest = false; + HTTP_R.Clean(); + } + } + +} diff --git a/src/output/output_progressive_mp3.h b/src/output/output_progressive_mp3.h new file mode 100644 index 00000000..b7cb5a19 --- /dev/null +++ b/src/output/output_progressive_mp3.h @@ -0,0 +1,18 @@ +#include "output.h" + + +namespace Mist { + class OutProgressiveMP3 : public Output { + public: + OutProgressiveMP3(Socket::Connection & conn); + ~OutProgressiveMP3(); + static void init(Util::Config * cfg); + void onRequest(); + void sendNext(); + void onFail(); + void sendHeader(); + protected: + }; +} + +typedef Mist::OutProgressiveMP3 mistOut; diff --git a/src/output/output_progressive_mp4.cpp b/src/output/output_progressive_mp4.cpp new file mode 100644 index 00000000..4b49420f --- /dev/null +++ b/src/output/output_progressive_mp4.cpp @@ -0,0 +1,558 @@ +#include "output_progressive_mp4.h" +#include <mist/defines.h> +#include <mist/mp4.h> +#include <mist/mp4_generic.h> + +namespace Mist { + OutProgressiveMP4::OutProgressiveMP4(Socket::Connection & conn) : Output(conn) { } + + OutProgressiveMP4::~OutProgressiveMP4() {} + + void OutProgressiveMP4::init(Util::Config * cfg){ + capa["name"] = "HTTP_Progressive_MP4"; + capa["desc"] = "Enables HTTP protocol progressive streaming."; + capa["deps"] = "HTTP"; + capa["url_rel"] = "/$.mp4"; + capa["url_match"] = "/$.mp4"; + capa["socket"] = "http_progressive_mp4"; + capa["codecs"][0u][0u].append("H264"); + capa["codecs"][0u][1u].append("AAC"); + capa["methods"][0u]["handler"] = "http"; + capa["methods"][0u]["type"] = "html5/video/mp4"; + capa["methods"][0u]["priority"] = 8ll; + capa["methods"][0u]["nolive"] = 1; + + + cfg->addBasicConnectorOptions(capa); + config = cfg; + } + + std::string OutProgressiveMP4::DTSCMeta2MP4Header(long long & size){ + std::stringstream header; + //ftyp box + MP4::FTYP ftypBox; + header << std::string(ftypBox.asBox(),ftypBox.boxedSize()); + + uint64_t mdatSize = 0; + //moov box + MP4::MOOV moovBox; + unsigned int moovOffset = 0; + { + //calculating longest duration + long long int firstms = -1; + long long int lastms = -1; + for (std::set<long unsigned int>::iterator it = selectedTracks.begin(); it != selectedTracks.end(); it++) { + if (lastms == -1 || lastms < myMeta.tracks[*it].lastms){ + lastms = myMeta.tracks[*it].lastms; + } + if (firstms == -1 || firstms > myMeta.tracks[*it].firstms){ + firstms = myMeta.tracks[*it].firstms; + } + } + MP4::MVHD mvhdBox(lastms - firstms); + moovBox.setContent(mvhdBox, moovOffset++); + } + for (std::set<long unsigned int>::iterator it = selectedTracks.begin(); it != selectedTracks.end(); it++) { + MP4::TRAK trakBox; + { + { + MP4::TKHD tkhdBox(*it, myMeta.tracks[*it].lastms - myMeta.tracks[*it].firstms, myMeta.tracks[*it].width, myMeta.tracks[*it].height); + trakBox.setContent(tkhdBox, 0); + }{ + MP4::MDIA mdiaBox; + unsigned int mdiaOffset = 0; + { + MP4::MDHD mdhdBox(myMeta.tracks[*it].lastms - myMeta.tracks[*it].firstms); + mdiaBox.setContent(mdhdBox, mdiaOffset++); + }//MDHD box + { + MP4::HDLR hdlrBox(myMeta.tracks[*it].type, myMeta.tracks[*it].getIdentifier()); + mdiaBox.setContent(hdlrBox, mdiaOffset++); + }//hdlr box + { + MP4::MINF minfBox; + unsigned int minfOffset = 0; + if (myMeta.tracks[*it].type== "video"){ + MP4::VMHD vmhdBox; + vmhdBox.setFlags(1); + minfBox.setContent(vmhdBox,minfOffset++); + }else if (myMeta.tracks[*it].type == "audio"){ + MP4::SMHD smhdBox; + minfBox.setContent(smhdBox,minfOffset++); + }//type box + { + MP4::DINF dinfBox; + MP4::DREF drefBox; + dinfBox.setContent(drefBox,0); + minfBox.setContent(dinfBox,minfOffset++); + }//dinf box + { + MP4::STBL stblBox; + unsigned int offset = 0; + { + MP4::STSD stsdBox; + stsdBox.setVersion(0); + if (myMeta.tracks[*it].type == "video"){//boxname = codec + MP4::VisualSampleEntry vse; + if (myMeta.tracks[*it].codec == "H264"){ + vse.setCodec("avc1"); + } + vse.setDataReferenceIndex(1); + vse.setWidth(myMeta.tracks[*it].width); + vse.setHeight(myMeta.tracks[*it].height); + MP4::AVCC avccBox; + avccBox.setPayload(myMeta.tracks[*it].init); + vse.setCLAP(avccBox); + stsdBox.setEntry(vse,0); + }else if(myMeta.tracks[*it].type == "audio"){//boxname = codec + MP4::AudioSampleEntry ase; + if (myMeta.tracks[*it].codec == "AAC"){ + ase.setCodec("mp4a"); + ase.setDataReferenceIndex(1); + } + ase.setSampleRate(myMeta.tracks[*it].rate); + ase.setChannelCount(myMeta.tracks[*it].channels); + ase.setSampleSize(myMeta.tracks[*it].size); + //MP4::ESDS esdsBox(myMeta.tracks[*it].init, myMeta.tracks[*it].bps); + MP4::ESDS esdsBox; + + //outputting these values first, so malloc isn't called as often. + esdsBox.setESHeaderStartCodes(myMeta.tracks[*it].init); + esdsBox.setSLValue(2); + + esdsBox.setESDescriptorTypeLength(32+myMeta.tracks[*it].init.size()); + esdsBox.setESID(2); + esdsBox.setStreamPriority(0); + esdsBox.setDecoderConfigDescriptorTypeLength(18 + myMeta.tracks[*it].init.size()); + esdsBox.setByteObjectTypeID(0x40); + esdsBox.setStreamType(5); + esdsBox.setReservedFlag(1); + esdsBox.setBufferSize(1250000); + esdsBox.setMaximumBitRate(10000000); + esdsBox.setAverageBitRate(myMeta.tracks[*it].bps * 8); + esdsBox.setConfigDescriptorTypeLength(5); + esdsBox.setSLConfigDescriptorTypeTag(0x6); + esdsBox.setSLConfigExtendedDescriptorTypeTag(0x808080); + esdsBox.setSLDescriptorTypeLength(1); + ase.setCodecBox(esdsBox); + stsdBox.setEntry(ase,0); + } + stblBox.setContent(stsdBox,offset++); + }//stsd box + { + MP4::STTS sttsBox; + sttsBox.setVersion(0); + if (myMeta.tracks[*it].parts.size()){ + for (unsigned int part = 0; part < myMeta.tracks[*it].parts.size(); part++){ + MP4::STTSEntry newEntry; + newEntry.sampleCount = 1; + newEntry.sampleDelta = myMeta.tracks[*it].parts[part].getDuration(); + sttsBox.setSTTSEntry(newEntry, part); + } + } + stblBox.setContent(sttsBox,offset++); + }//stts box + if (myMeta.tracks[*it].type == "video"){ + //STSS Box here + MP4::STSS stssBox; + stssBox.setVersion(0); + int tmpCount = 0; + int tmpItCount = 0; + for ( std::deque< DTSC::Key>::iterator tmpIt = myMeta.tracks[*it].keys.begin(); tmpIt != myMeta.tracks[*it].keys.end(); tmpIt ++) { + stssBox.setSampleNumber(tmpCount,tmpItCount); + tmpCount += tmpIt->getParts(); + tmpItCount ++; + } + stblBox.setContent(stssBox,offset++); + }//stss box + { + MP4::STSC stscBox; + stscBox.setVersion(0); + MP4::STSCEntry stscEntry; + stscEntry.firstChunk = 1; + stscEntry.samplesPerChunk = 1; + stscEntry.sampleDescriptionIndex = 1; + stscBox.setSTSCEntry(stscEntry, 0); + stblBox.setContent(stscBox,offset++); + }//stsc box + { + uint32_t total = 0; + MP4::STSZ stszBox; + stszBox.setVersion(0); + total = 0; + for (std::deque< DTSC::Part>::iterator partIt = myMeta.tracks[*it].parts.begin(); partIt != myMeta.tracks[*it].parts.end(); partIt ++) { + stszBox.setEntrySize(partIt->getSize(), total);//in bytes in file + size += partIt->getSize(); + total++; + } + stblBox.setContent(stszBox,offset++); + }//stsz box + //add STCO boxes here + { + MP4::STCO stcoBox; + stcoBox.setVersion(1); + //Inserting empty values on purpose here, will be fixed later. + if (myMeta.tracks[*it].parts.size() != 0){ + stcoBox.setChunkOffset(0, myMeta.tracks[*it].parts.size() - 1);//this inserts all empty entries at once + } + stblBox.setContent(stcoBox,offset++); + }//stco box + minfBox.setContent(stblBox,minfOffset++); + }//stbl box + mdiaBox.setContent(minfBox, mdiaOffset++); + }//minf box + trakBox.setContent(mdiaBox, 1); + } + }//trak Box + moovBox.setContent(trakBox, moovOffset++); + } + //initial offset length ftyp, length moov + 8 + unsigned long long int byteOffset = ftypBox.boxedSize() + moovBox.boxedSize() + 8; + //update all STCO from the following map; + std::map <int, MP4::STCO> checkStcoBoxes; + //for all tracks + for (unsigned int i = 1; i < moovBox.getContentCount(); i++){ + //10 lines to get the STCO box. + MP4::TRAK checkTrakBox; + MP4::Box checkMdiaBox; + MP4::Box checkTkhdBox; + MP4::MINF checkMinfBox; + MP4::STBL checkStblBox; + //MP4::STCO checkStcoBox; + checkTrakBox = ((MP4::TRAK&)moovBox.getContent(i)); + for (unsigned int j = 0; j < checkTrakBox.getContentCount(); j++){ + if (checkTrakBox.getContent(j).isType("mdia")){ + checkMdiaBox = checkTrakBox.getContent(j); + break; + } + if (checkTrakBox.getContent(j).isType("tkhd")){ + checkTkhdBox = checkTrakBox.getContent(j); + } + } + for (unsigned int j = 0; j < ((MP4::MDIA&)checkMdiaBox).getContentCount(); j++){ + if (((MP4::MDIA&)checkMdiaBox).getContent(j).isType("minf")){ + checkMinfBox = ((MP4::MINF&)((MP4::MDIA&)checkMdiaBox).getContent(j)); + break; + } + } + for (unsigned int j = 0; j < checkMinfBox.getContentCount(); j++){ + if (checkMinfBox.getContent(j).isType("stbl")){ + checkStblBox = ((MP4::STBL&)checkMinfBox.getContent(j)); + break; + } + } + for (unsigned int j = 0; j < checkStblBox.getContentCount(); j++){ + if (checkStblBox.getContent(j).isType("stco")){ + checkStcoBoxes.insert( std::pair<int, MP4::STCO>(((MP4::TKHD&)checkTkhdBox).getTrackID(), ((MP4::STCO&)checkStblBox.getContent(j)) )); + break; + } + } + } + //inserting right values in the STCO box header + //total = 0; + long long unsigned int totalByteOffset = 0; + //Current values are actual byte offset without header-sized offset + std::set <keyPart> sortSet;//filling sortset for interleaving parts + for (std::set<long unsigned int>::iterator subIt = selectedTracks.begin(); subIt != selectedTracks.end(); subIt++) { + keyPart temp; + temp.trackID = *subIt; + temp.time = myMeta.tracks[*subIt].firstms;//timeplace of frame + temp.endTime = myMeta.tracks[*subIt].firstms + myMeta.tracks[*subIt].parts[0].getDuration(); + temp.size = myMeta.tracks[*subIt].parts[0].getSize();//bytesize of frame (alle parts all together) + temp.index = 0; + sortSet.insert(temp); + } + while (!sortSet.empty()){ + //setting the right STCO size in the STCO box + checkStcoBoxes[sortSet.begin()->trackID].setChunkOffset(totalByteOffset + byteOffset, sortSet.begin()->index); + totalByteOffset += sortSet.begin()->size; + //add keyPart to sortSet + keyPart temp; + temp.index = sortSet.begin()->index + 1; + temp.trackID = sortSet.begin()->trackID; + if(temp.index < myMeta.tracks[temp.trackID].parts.size() ){//only insert when there are parts left + temp.time = sortSet.begin()->endTime;//timeplace of frame + temp.endTime = sortSet.begin()->endTime + myMeta.tracks[temp.trackID].parts[temp.index].getDuration(); + temp.size = myMeta.tracks[temp.trackID].parts[temp.index].getSize();//bytesize of frame + sortSet.insert(temp); + } + //remove highest keyPart + sortSet.erase(sortSet.begin()); + } + + mdatSize = totalByteOffset+8; + + header << std::string(moovBox.asBox(),moovBox.boxedSize()); + + header << (char)((mdatSize>>24) & 0xFF) << (char)((mdatSize>>16) & 0xFF) << (char)((mdatSize>>8) & 0xFF) << (char)(mdatSize & 0xFF) << "mdat"; + //end of header + + size += header.str().size(); + return header.str(); + } + + /// Calculate a seekPoint, based on byteStart, metadata, tracks and headerSize. + /// The seekPoint will be set to the timestamp of the first packet to send. + void OutProgressiveMP4::findSeekPoint(long long byteStart, long long & seekPoint, unsigned int headerSize){ + seekPoint = 0; + //if we're starting in the header, seekPoint is always zero. + if (byteStart <= headerSize){return;} + //okay, we're past the header. Substract the headersize from the starting postion. + byteStart -= headerSize; + //initialize a list of sorted parts that this file contains + std::set <keyPart> sortSet; + for (std::set<long unsigned int>::iterator subIt = selectedTracks.begin(); subIt != selectedTracks.end(); subIt++) { + keyPart temp; + temp.trackID = *subIt; + temp.time = myMeta.tracks[*subIt].firstms;//timeplace of frame + temp.endTime = myMeta.tracks[*subIt].firstms + myMeta.tracks[*subIt].parts[0].getDuration(); + temp.size = myMeta.tracks[*subIt].parts[0].getSize();//bytesize of frame (alle parts all together) + temp.index = 0; + sortSet.insert(temp); + } + //forward through the file by headers, until we reach the point where we need to be + while (!sortSet.empty()){ + //substract the size of this fragment from byteStart + byteStart -= sortSet.begin()->size; + //if that put us past the point where we wanted to be, return right now + if (byteStart < 0){return;} + //otherwise, set seekPoint to where we are now + seekPoint = sortSet.begin()->time; + //then find the next part + keyPart temp; + temp.index = sortSet.begin()->index + 1; + temp.trackID = sortSet.begin()->trackID; + if(temp.index < myMeta.tracks[temp.trackID].parts.size() ){//only insert when there are parts left + temp.time = sortSet.begin()->endTime;//timeplace of frame + temp.endTime = sortSet.begin()->endTime + myMeta.tracks[temp.trackID].parts[temp.index].getDuration(); + temp.size = myMeta.tracks[temp.trackID].parts[temp.index].getSize();//bytesize of frame + sortSet.insert(temp); + } + //remove highest keyPart + sortSet.erase(sortSet.begin()); + } + //If we're here, we're in the last fragment. + //That's technically legal, of course. + } + + /// Parses a "Range: " header, setting byteStart, byteEnd and seekPoint using data from metadata and tracks to do + /// the calculations. + /// On error, byteEnd is set to zero. + void OutProgressiveMP4::parseRange(std::string header, long long & byteStart, long long & byteEnd, long long & seekPoint, unsigned int headerSize){ + if (header.size() < 6 || header.substr(0, 6) != "bytes="){ + byteEnd = 0; + DEBUG_MSG(DLVL_WARN, "Invalid range header: %s", header.c_str()); + return; + } + header.erase(0, 6); + if (header.size() && header[0] == '-'){ + //negative range = count from end + byteStart = 0; + for (unsigned int i = 1; i < header.size(); ++i){ + if (header[i] >= '0' && header[i] <= '9'){ + byteStart *= 10; + byteStart += header[i] - '0'; + continue; + } + break; + } + if (byteStart > byteEnd){ + //entire file if starting before byte zero + byteStart = 0; + DEBUG_MSG(DLVL_DEVEL, "Full negative range: %lli-%lli", byteStart, byteEnd); + findSeekPoint(byteStart, seekPoint, headerSize); + return; + }else{ + //start byteStart bytes before byteEnd + byteStart = byteEnd - byteStart; + DEBUG_MSG(DLVL_DEVEL, "Partial negative range: %lli-%lli", byteStart, byteEnd); + findSeekPoint(byteStart, seekPoint, headerSize); + return; + } + }else{ + long long size = byteEnd; + byteEnd = 0; + byteStart = 0; + unsigned int i = 0; + for ( ; i < header.size(); ++i){ + if (header[i] >= '0' && header[i] <= '9'){ + byteStart *= 10; + byteStart += header[i] - '0'; + continue; + } + break; + } + if (header[i] != '-'){ + DEBUG_MSG(DLVL_WARN, "Invalid range header: %s", header.c_str()); + byteEnd = 0; + return; + } + ++i; + if (i < header.size()){ + for ( ; i < header.size(); ++i){ + if (header[i] >= '0' && header[i] <= '9'){ + byteEnd *= 10; + byteEnd += header[i] - '0'; + continue; + } + break; + } + if (byteEnd > size-1){byteEnd = size;} + }else{ + byteEnd = size; + } + DEBUG_MSG(DLVL_DEVEL, "Range request: %lli-%lli (%s)", byteStart, byteEnd, header.c_str()); + findSeekPoint(byteStart, seekPoint, headerSize); + return; + } + } + + void OutProgressiveMP4::onRequest(){ + while (HTTP_R.Read(myConn)){ + DEBUG_MSG(DLVL_DEVEL, "Received request: %s", HTTP_R.getUrl().c_str()); + myConn.setHost(HTTP_R.GetHeader("X-Origin")); + streamName = HTTP_R.GetHeader("X-Stream"); + if (HTTP_R.GetVar("audio") != ""){ + DEBUG_MSG(DLVL_DEVEL, "GetVar Aud = %s", HTTP_R.GetVar("audio").c_str()); + selectedTracks.insert(JSON::Value(HTTP_R.GetVar("audio")).asInt()); + }else{ + DEBUG_MSG(DLVL_DEVEL, "No audio param given"); + } + if (HTTP_R.GetVar("video") != ""){ + DEBUG_MSG(DLVL_DEVEL, "GetVar Vid = %s", HTTP_R.GetVar("video").c_str()); + selectedTracks.insert(JSON::Value(HTTP_R.GetVar("video")).asInt()); + }else{ + DEBUG_MSG(DLVL_DEVEL, "No video param given"); + } + + parseData = true; + wantRequest = false; + } + } + + bool OutProgressiveMP4::onFinish(){ + HTTP_R.Clean(); + parseData = false; + wantRequest = true; + return true; + } + + void OutProgressiveMP4::onFail(){ + HTTP_S.Clean(); //make sure no parts of old requests are left in any buffers + HTTP_S.SetBody("Stream not found. Sorry, we tried."); + HTTP_S.SendResponse("404", "Stream not found", myConn); + Output::onFail(); + } + + void OutProgressiveMP4::sendNext(){ + char * dataPointer = 0; + int len = 0; + currentPacket.getString("data", dataPointer, len); + + //keep track of where we are - fast-forward until where we are now + while (!sortSet.empty() && ((long long)sortSet.begin()->trackID != currentPacket.getTrackId() || (long long)sortSet.begin()->time != currentPacket.getTime())){ + keyPart temp; + temp.index = sortSet.begin()->index + 1; + temp.trackID = sortSet.begin()->trackID; + if(temp.index < myMeta.tracks[temp.trackID].parts.size() ){//only insert when there are parts left + temp.time = sortSet.begin()->endTime;//timeplace of frame + temp.endTime = sortSet.begin()->endTime + myMeta.tracks[temp.trackID].parts[temp.index].getDuration(); + temp.size = myMeta.tracks[temp.trackID].parts[temp.index].getSize();//bytesize of frame + sortSet.insert(temp); + } + currPos += sortSet.begin()->size; + //remove highest keyPart + sortSet.erase(sortSet.begin()); + } + if (currPos >= byteStart){ + sortSet.clear();//we don't need you anymore! + myConn.SendNow(dataPointer, std::min(leftOver, (long long)len)); + //HTTP_S.Chunkify(Strm.lastData().data(), Strm.lastData().size(), conn); + leftOver -= len; + }else{ + if (currPos + (long long)len > byteStart){ + myConn.SendNow(dataPointer+(byteStart-currPos), len-(byteStart-currPos)); + leftOver -= len-(byteStart-currPos); + currPos = byteStart; + sortSet.clear();//we don't need you anymore! + } + } + if (leftOver < 1){ + //stop playback, wait for new request + stop(); + wantRequest = true; + } + } + + void OutProgressiveMP4::sendHeader(){ + fileSize = 0; + std::string headerData = DTSCMeta2MP4Header(fileSize); + byteStart = 0; + byteEnd = fileSize - 1; + long long seekPoint = 0; + char rangeType = ' '; + if (HTTP_R.GetHeader("Range") != ""){ + parseRange(HTTP_R.GetHeader("Range"), byteStart, byteEnd, seekPoint, headerData.size()); + rangeType = HTTP_R.GetHeader("Range")[0]; + } + sortSet.clear(); + for (std::set<long unsigned int>::iterator subIt = selectedTracks.begin(); subIt != selectedTracks.end(); subIt++) { + keyPart temp; + temp.trackID = *subIt; + temp.time = myMeta.tracks[*subIt].firstms;//timeplace of frame + temp.endTime = myMeta.tracks[*subIt].firstms + myMeta.tracks[*subIt].parts[0].getDuration(); + temp.size = myMeta.tracks[*subIt].parts[0].getSize();//bytesize of frame (alle parts all together) + temp.index = 0; + sortSet.insert(temp); + } + HTTP_S.Clean(); //make sure no parts of old requests are left in any buffers + HTTP_S.SetHeader("Content-Type", "video/MP4"); //Send the correct content-type for MP4 files + HTTP_S.SetHeader("Accept-Ranges", "bytes, parsec"); + if (rangeType != ' '){ + DEBUG_MSG(DLVL_DEVEL, "Ranged request"); + if (!byteEnd){ + if (rangeType == 'p'){ + HTTP_S.SetBody("Starsystem not in communications range"); + HTTP_S.SendResponse("416", "Starsystem not in communications range", myConn); + return; + }else{ + HTTP_S.SetBody("Requested Range Not Satisfiable"); + HTTP_S.SendResponse("416", "Requested Range Not Satisfiable", myConn); + return; + } + }else{ + std::stringstream rangeReply; + rangeReply << "bytes " << byteStart << "-" << byteEnd << "/" << fileSize; + HTTP_S.SetHeader("Content-Length", byteEnd - byteStart + 1); + //do not multiplex requests that are > 1MiB + if (byteEnd - byteStart + 1 > 1024*1024){ + HTTP_S.SetHeader("MistMultiplex", "No"); + } + HTTP_S.SetHeader("Content-Range", rangeReply.str()); + /// \todo Switch to chunked? + HTTP_S.SendResponse("206", "Partial content", myConn); + //HTTP_S.StartResponse("206", "Partial content", HTTP_R, conn); + } + }else{ + DEBUG_MSG(DLVL_DEVEL, "Non-Ranged request"); + HTTP_S.SetHeader("Content-Length", byteEnd - byteStart + 1); + //do not multiplex requests that aren't ranged + HTTP_S.SetHeader("MistMultiplex", "No"); + /// \todo Switch to chunked? + HTTP_S.SendResponse("200", "OK", myConn); + //HTTP_S.StartResponse(HTTP_R, conn); + } + leftOver = byteEnd - byteStart + 1;//add one byte, because range "0-0" = 1 byte of data + currPos = 0; + if (byteStart < (long long)headerData.size()){ + /// \todo Switch to chunked? + //HTTP_S.Chunkify(headerData.data()+byteStart, std::min((long long)headerData.size(), byteEnd) - byteStart, conn);//send MP4 header + myConn.SendNow(headerData.data()+byteStart, std::min((long long)headerData.size(), byteEnd) - byteStart);//send MP4 header + leftOver -= std::min((long long)headerData.size(), byteEnd) - byteStart; + } + currPos = headerData.size();//we're now guaranteed to be past the header point, no matter what + seek(seekPoint); + sentHeader = true; + } + +} diff --git a/src/output/output_progressive_mp4.h b/src/output/output_progressive_mp4.h new file mode 100644 index 00000000..f82f24a5 --- /dev/null +++ b/src/output/output_progressive_mp4.h @@ -0,0 +1,50 @@ +#include "output.h" +#include <mist/http_parser.h> + +namespace Mist { + struct keyPart{ + public: + bool operator < (const keyPart& rhs) const { + if (time < rhs.time){ + return true; + } + if (time == rhs.time){ + if (trackID < rhs.trackID){ + return true; + } + } + return false; + } + long unsigned int trackID; + long unsigned int size; + long long unsigned int time; + long long unsigned int endTime; + long unsigned int index; + }; + + class OutProgressiveMP4 : public Output { + public: + OutProgressiveMP4(Socket::Connection & conn); + ~OutProgressiveMP4(); + static void init(Util::Config * cfg); + void parseRange(std::string header, long long & byteStart, long long & byteEnd, long long & seekPoint, unsigned int headerSize); + std::string DTSCMeta2MP4Header(long long & size); + void findSeekPoint(long long byteStart, long long & seekPoint, unsigned int headerSize); + + void onRequest(); + void sendNext(); + bool onFinish(); + void sendHeader(); + void onFail(); + protected: + long long fileSize; + long long byteStart; + long long byteEnd; + long long leftOver; + long long currPos; + std::set <keyPart> sortSet;//filling sortset for interleaving parts + HTTP::Parser HTTP_R, HTTP_S; + }; +} + +typedef Mist::OutProgressiveMP4 mistOut; diff --git a/src/output/output_raw.cpp b/src/output/output_raw.cpp new file mode 100644 index 00000000..2bd71204 --- /dev/null +++ b/src/output/output_raw.cpp @@ -0,0 +1,69 @@ +#include "output_raw.h" + +namespace Mist { + OutRaw::OutRaw(Socket::Connection & conn) : Output(conn) { + streamName = config->getString("streamname"); + initialize(); + selectedTracks.clear(); + std::string tracks = config->getString("tracks"); + unsigned int currTrack = 0; + //loop over tracks, add any found track IDs to selectedTracks + if (tracks != ""){ + for (unsigned int i = 0; i < tracks.size(); ++i){ + if (tracks[i] >= '0' && tracks[i] <= '9'){ + currTrack = currTrack*10 + (tracks[i] - '0'); + }else{ + if (currTrack > 0){ + selectedTracks.insert(currTrack); + } + currTrack = 0; + } + } + if (currTrack > 0){ + selectedTracks.insert(currTrack); + } + } + parseData = true; + seek(config->getInteger("seek")); + } + + OutRaw::~OutRaw() {} + + void OutRaw::init(Util::Config * cfg){ + capa["name"] = "RAW"; + capa["desc"] = "Enables raw DTSC over TCP."; + capa["deps"] = ""; + capa["required"]["streamname"]["name"] = "Stream"; + capa["required"]["streamname"]["help"] = "What streamname to serve. For multiple streams, add this protocol multiple times using different ports."; + capa["required"]["streamname"]["type"] = "str"; + capa["required"]["streamname"]["option"] = "--stream"; + capa["optional"]["tracks"]["name"] = "Tracks"; + capa["optional"]["tracks"]["help"] = "The track IDs of the stream that this connector will transmit separated by spaces"; + capa["optional"]["tracks"]["type"] = "str"; + capa["optional"]["tracks"]["option"] = "--tracks"; + capa["optional"]["seek"]["name"] = "Seek point"; + capa["optional"]["seek"]["help"] = "The time in milliseconds to seek to, 0 by default."; + capa["optional"]["seek"]["type"] = "int"; + capa["optional"]["seek"]["option"] = "--seek"; + capa["codecs"][0u][0u].append("H264"); + capa["codecs"][0u][1u].append("AAC"); + cfg->addOption("streamname", + JSON::fromString("{\"arg\":\"string\",\"short\":\"s\",\"long\":\"stream\",\"help\":\"The name of the stream that this connector will transmit.\"}")); + cfg->addOption("tracks", + JSON::fromString("{\"arg\":\"string\",\"value\":[\"\"],\"short\": \"t\",\"long\":\"tracks\",\"help\":\"The track IDs of the stream that this connector will transmit separated by spaces.\"}")); + cfg->addOption("seek", + JSON::fromString("{\"arg\":\"integer\",\"value\":[0],\"short\": \"S\",\"long\":\"seek\",\"help\":\"The time in milliseconds to seek to, 0 by default.\"}")); + cfg->addConnectorOptions(666, capa); + config = cfg; + } + + void OutRaw::sendNext(){ + myConn.SendNow(currentPacket.getData(), currentPacket.getDataLen()); + } + + void OutRaw::sendHeader(){ + myMeta.send(myConn); + sentHeader = true; + } + +} diff --git a/src/output/output_raw.h b/src/output/output_raw.h new file mode 100644 index 00000000..82105f0b --- /dev/null +++ b/src/output/output_raw.h @@ -0,0 +1,15 @@ +#include "output.h" + + +namespace Mist { + class OutRaw : public Output { + public: + OutRaw(Socket::Connection & conn); + ~OutRaw(); + static void init(Util::Config * cfg); + void sendNext(); + void sendHeader(); + }; +} + +typedef Mist::OutRaw mistOut; diff --git a/src/output/output_rtmp.cpp b/src/output/output_rtmp.cpp new file mode 100644 index 00000000..e6e3eaff --- /dev/null +++ b/src/output/output_rtmp.cpp @@ -0,0 +1,754 @@ +#include "output_rtmp.h" +#include <mist/http_parser.h> +#include <mist/defines.h> +#include <mist/stream.h> +#include <cstring> +#include <cstdlib> + +namespace Mist { + OutRTMP::OutRTMP(Socket::Connection & conn) : Output(conn) { + playTransaction = -1; + playMessageType = -1; + playStreamId = -1; + setBlocking(false); + while (!conn.Received().available(1537) && conn.connected()) { + conn.spool(); + Util::sleep(5); + } + RTMPStream::handshake_in = conn.Received().remove(1537); + RTMPStream::rec_cnt += 1537; + + if (RTMPStream::doHandshake()) { + conn.SendNow(RTMPStream::handshake_out); + while (!conn.Received().available(1536) && conn.connected()) { + conn.spool(); + Util::sleep(5); + } + conn.Received().remove(1536); + RTMPStream::rec_cnt += 1536; + DEBUG_MSG(DLVL_HIGH, "Handshake success!"); + } else { + DEBUG_MSG(DLVL_DEVEL, "Handshake fail!"); + } + counter = 0; + sending = false; + streamReset = false; + } + + OutRTMP::~OutRTMP() {} + + void OutRTMP::init(Util::Config * cfg) { + capa["name"] = "RTMP"; + capa["desc"] = "Enables the RTMP protocol which is used by Adobe Flash Player."; + capa["deps"] = ""; + capa["url_rel"] = "/play/$"; + capa["codecs"][0u][0u].append("H264"); + capa["codecs"][0u][0u].append("H263"); + capa["codecs"][0u][0u].append("VP6"); + capa["codecs"][0u][1u].append("AAC"); + capa["codecs"][0u][1u].append("MP3"); + capa["methods"][0u]["handler"] = "rtmp"; + capa["methods"][0u]["type"] = "flash/10"; + capa["methods"][0u]["priority"] = 6ll; + cfg->addConnectorOptions(1935, capa); + config = cfg; + } + + void OutRTMP::sendNext() { + //sent a tag + FLV::Tag tag; + if (tag.DTSCLoader(currentPacket, myMeta.tracks[currentPacket.getTrackId()])) { + if (tag.len) { + myConn.SendNow(RTMPStream::SendMedia(tag)); +#if DEBUG >= 8 + fprintf(stderr, "Sent tag to %i: [%u] %s\n", myConn.getSocket(), tag.tagTime(), tag.tagType().c_str()); +#endif + } + } + } + + void OutRTMP::sendHeader() { + FLV::Tag tag; + tag.DTSCMetaInit(myMeta, selectedTracks); + if (tag.len) { + myConn.SendNow(RTMPStream::SendMedia(tag)); + } + + for (std::set<long unsigned int>::iterator it = selectedTracks.begin(); it != selectedTracks.end(); it++) { + if (myMeta.tracks[*it].type == "video") { + tag.DTSCVideoInit(myMeta.tracks[*it]); + if (tag.len) { + myConn.SendNow(RTMPStream::SendMedia(tag)); + } + } + if (myMeta.tracks[*it].type == "audio") { + tag.DTSCAudioInit(myMeta.tracks[*it]); + if (tag.len) { + myConn.SendNow(RTMPStream::SendMedia(tag)); + } + } + } + sentHeader = true; + } + + void OutRTMP::onRequest() { + parseChunk(myConn.Received()); + } + + ///\brief Sends a RTMP command either in AMF or AMF3 mode. + ///\param amfReply The data to be sent over RTMP. + ///\param messageType The type of message. + ///\param streamId The ID of the AMF stream. + void OutRTMP::sendCommand(AMF::Object & amfReply, int messageType, int streamId) { +#if DEBUG >= 8 + std::cerr << amfReply.Print() << std::endl; +#endif + if (messageType == 17) { + myConn.SendNow(RTMPStream::SendChunk(3, messageType, streamId, (char)0 + amfReply.Pack())); + } else { + myConn.SendNow(RTMPStream::SendChunk(3, messageType, streamId, amfReply.Pack())); + } + } //sendCommand + + ///\brief Parses a single AMF command message, and sends a direct response through sendCommand(). + ///\param amfData The received request. + ///\param messageType The type of message. + ///\param streamId The ID of the AMF stream. + void OutRTMP::parseAMFCommand(AMF::Object & amfData, int messageType, int streamId) { +#if DEBUG >= 5 + fprintf(stderr, "Received command: %s\n", amfData.Print().c_str()); +#endif +#if DEBUG >= 8 + fprintf(stderr, "AMF0 command: %s\n", amfData.getContentP(0)->StrValue().c_str()); +#endif + if (amfData.getContentP(0)->StrValue() == "connect") { + double objencoding = 0; + if (amfData.getContentP(2)->getContentP("objectEncoding")) { + objencoding = amfData.getContentP(2)->getContentP("objectEncoding")->NumValue(); + } +#if DEBUG >= 6 + int tmpint; + if (amfData.getContentP(2)->getContentP("videoCodecs")) { + tmpint = (int)amfData.getContentP(2)->getContentP("videoCodecs")->NumValue(); + if (tmpint & 0x04) { + fprintf(stderr, "Sorensen video support detected\n"); + } + if (tmpint & 0x80) { + fprintf(stderr, "H264 video support detected\n"); + } + } + if (amfData.getContentP(2)->getContentP("audioCodecs")) { + tmpint = (int)amfData.getContentP(2)->getContentP("audioCodecs")->NumValue(); + if (tmpint & 0x04) { + fprintf(stderr, "MP3 audio support detected\n"); + } + if (tmpint & 0x400) { + fprintf(stderr, "AAC audio support detected\n"); + } + } +#endif + app_name = amfData.getContentP(2)->getContentP("tcUrl")->StrValue(); + app_name = app_name.substr(app_name.find('/', 7) + 1); + RTMPStream::chunk_snd_max = 4096; + myConn.Send(RTMPStream::SendCTL(1, RTMPStream::chunk_snd_max)); //send chunk size max (msg 1) + myConn.Send(RTMPStream::SendCTL(5, RTMPStream::snd_window_size)); //send window acknowledgement size (msg 5) + myConn.Send(RTMPStream::SendCTL(6, RTMPStream::rec_window_size)); //send rec window acknowledgement size (msg 6) + myConn.Send(RTMPStream::SendUSR(0, 1)); //send UCM StreamBegin (0), stream 1 + //send a _result reply + AMF::Object amfReply("container", AMF::AMF0_DDV_CONTAINER); + amfReply.addContent(AMF::Object("", "_result")); //result success + amfReply.addContent(amfData.getContent(1)); //same transaction ID + amfReply.addContent(AMF::Object("")); //server properties + amfReply.getContentP(2)->addContent(AMF::Object("fmsVer", "FMS/3,5,5,2004")); + amfReply.getContentP(2)->addContent(AMF::Object("capabilities", (double)31)); + amfReply.getContentP(2)->addContent(AMF::Object("mode", (double)1)); + amfReply.addContent(AMF::Object("")); //info + amfReply.getContentP(3)->addContent(AMF::Object("level", "status")); + amfReply.getContentP(3)->addContent(AMF::Object("code", "NetConnection.Connect.Success")); + amfReply.getContentP(3)->addContent(AMF::Object("description", "Connection succeeded.")); + amfReply.getContentP(3)->addContent(AMF::Object("clientid", 1337)); + amfReply.getContentP(3)->addContent(AMF::Object("objectEncoding", objencoding)); + //amfReply.getContentP(3)->addContent(AMF::Object("data", AMF::AMF0_ECMA_ARRAY)); + //amfReply.getContentP(3)->getContentP(4)->addContent(AMF::Object("version", "3,5,4,1004")); + sendCommand(amfReply, messageType, streamId); + //send onBWDone packet - no clue what it is, but real server sends it... + //amfReply = AMF::Object("container", AMF::AMF0_DDV_CONTAINER); + //amfReply.addContent(AMF::Object("", "onBWDone"));//result + //amfReply.addContent(amfData.getContent(1));//same transaction ID + //amfReply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL));//null + //sendCommand(amfReply, messageType, streamId); + return; + } //connect + if (amfData.getContentP(0)->StrValue() == "createStream") { + //send a _result reply + AMF::Object amfReply("container", AMF::AMF0_DDV_CONTAINER); + amfReply.addContent(AMF::Object("", "_result")); //result success + amfReply.addContent(amfData.getContent(1)); //same transaction ID + amfReply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL)); //null - command info + amfReply.addContent(AMF::Object("", (double)1)); //stream ID - we use 1 + sendCommand(amfReply, messageType, streamId); + myConn.Send(RTMPStream::SendUSR(0, 1)); //send UCM StreamBegin (0), stream 1 + return; + } //createStream + if ((amfData.getContentP(0)->StrValue() == "closeStream") || (amfData.getContentP(0)->StrValue() == "deleteStream")) { + stop(); + return; + } + if ((amfData.getContentP(0)->StrValue() == "FCUnpublish") || (amfData.getContentP(0)->StrValue() == "releaseStream")) { + // ignored + return; + } + if ((amfData.getContentP(0)->StrValue() == "FCPublish")) { + //send a FCPublic reply + AMF::Object amfReply("container", AMF::AMF0_DDV_CONTAINER); + amfReply.addContent(AMF::Object("", "onFCPublish")); //status reply + amfReply.addContent(AMF::Object("", 0, AMF::AMF0_NUMBER)); //same transaction ID + amfReply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL)); //null - command info + amfReply.addContent(AMF::Object("")); //info + amfReply.getContentP(3)->addContent(AMF::Object("code", "NetStream.Publish.Start")); + amfReply.getContentP(3)->addContent(AMF::Object("description", "Please followup with publish command...")); + sendCommand(amfReply, messageType, streamId); + return; + } //FCPublish + if (amfData.getContentP(0)->StrValue() == "releaseStream") { + //send a _result reply + AMF::Object amfReply("container", AMF::AMF0_DDV_CONTAINER); + amfReply.addContent(AMF::Object("", "_result")); //result success + amfReply.addContent(amfData.getContent(1)); //same transaction ID + amfReply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL)); //null - command info + amfReply.addContent(AMF::Object("", AMF::AMF0_UNDEFINED)); //stream ID? + sendCommand(amfReply, messageType, streamId); + return; + }//releaseStream + if ((amfData.getContentP(0)->StrValue() == "getStreamLength") || (amfData.getContentP(0)->StrValue() == "getMovLen")) { + //send a _result reply + AMF::Object amfReply("container", AMF::AMF0_DDV_CONTAINER); + amfReply.addContent(AMF::Object("", "_result")); //result success + amfReply.addContent(amfData.getContent(1)); //same transaction ID + amfReply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL)); //null - command info + amfReply.addContent(AMF::Object("", (double)0)); //zero length + sendCommand(amfReply, messageType, streamId); + return; + } //getStreamLength + if ((amfData.getContentP(0)->StrValue() == "publish")) { + if (amfData.getContentP(3)) { + streamName = amfData.getContentP(3)->StrValue(); + //pull the server configuration + JSON::Value servConf = JSON::fromFile(Util::getTmpFolder() + "streamlist"); + if (servConf.isMember("streams") && servConf["streams"].isMember(streamName)){ + JSON::Value & streamConfig = servConf["streams"][streamName]; + if (!streamConfig.isMember("source") || streamConfig["source"].asStringRef().substr(0, 7) != "push://"){ + DEBUG_MSG(DLVL_FAIL, "Push rejected - stream not a push-able stream. (%s != push://*)", streamConfig["source"].asStringRef().c_str()); + myConn.close(); + return; + } + std::string source = streamConfig["source"].asStringRef().substr(7); + std::string IP = source.substr(0, source.find('@')); + if (IP != ""){ + if (!myConn.isAddress(IP)){ + DEBUG_MSG(DLVL_FAIL, "Push rejected - source host not whitelisted"); + myConn.close(); + return; + } + } + }else{ + DEBUG_MSG(DLVL_FAIL, "Push rejected - stream not configured."); + myConn.close(); + return; + } + initialize(); + } + //send a _result reply + AMF::Object amfReply("container", AMF::AMF0_DDV_CONTAINER); + amfReply.addContent(AMF::Object("", "_result")); //result success + amfReply.addContent(amfData.getContent(1)); //same transaction ID + amfReply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL)); //null - command info + amfReply.addContent(AMF::Object("", 1, AMF::AMF0_BOOL)); //publish success? + sendCommand(amfReply, messageType, streamId); + myConn.Send(RTMPStream::SendUSR(0, 1)); //send UCM StreamBegin (0), stream 1 + //send a status reply + amfReply = AMF::Object("container", AMF::AMF0_DDV_CONTAINER); + amfReply.addContent(AMF::Object("", "onStatus")); //status reply + amfReply.addContent(AMF::Object("", 0, AMF::AMF0_NUMBER)); //same transaction ID + amfReply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL)); //null - command info + amfReply.addContent(AMF::Object("")); //info + amfReply.getContentP(3)->addContent(AMF::Object("level", "status")); + amfReply.getContentP(3)->addContent(AMF::Object("code", "NetStream.Publish.Start")); + amfReply.getContentP(3)->addContent(AMF::Object("description", "Stream is now published!")); + amfReply.getContentP(3)->addContent(AMF::Object("clientid", (double)1337)); + sendCommand(amfReply, messageType, streamId); + return; + } //getStreamLength + if (amfData.getContentP(0)->StrValue() == "checkBandwidth") { + //send a _result reply + AMF::Object amfReply("container", AMF::AMF0_DDV_CONTAINER); + amfReply.addContent(AMF::Object("", "_result")); //result success + amfReply.addContent(amfData.getContent(1)); //same transaction ID + amfReply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL)); //null - command info + amfReply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL)); //null - command info + sendCommand(amfReply, messageType, streamId); + return; + } //checkBandwidth + if ((amfData.getContentP(0)->StrValue() == "play") || (amfData.getContentP(0)->StrValue() == "play2")) { + //set reply number and stream name, actual reply is sent up in the ss.spool() handler + int playTransaction = amfData.getContentP(1)->NumValue(); + int playMessageType = messageType; + int playStreamId = streamId; + streamName = amfData.getContentP(3)->StrValue(); + initialize(); + + //send a status reply + AMF::Object amfreply("container", AMF::AMF0_DDV_CONTAINER); + amfreply.addContent(AMF::Object("", "onStatus")); //status reply + amfreply.addContent(AMF::Object("", (double)playTransaction)); //same transaction ID + amfreply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL)); //null - command info + amfreply.addContent(AMF::Object("")); //info + amfreply.getContentP(3)->addContent(AMF::Object("level", "status")); + amfreply.getContentP(3)->addContent(AMF::Object("code", "NetStream.Play.Reset")); + amfreply.getContentP(3)->addContent(AMF::Object("description", "Playing and resetting...")); + amfreply.getContentP(3)->addContent(AMF::Object("details", "DDV")); + amfreply.getContentP(3)->addContent(AMF::Object("clientid", (double)1337)); + sendCommand(amfreply, playMessageType, playStreamId); + //send streamisrecorded if stream, well, is recorded. + if (myMeta.vod) { //isMember("length") && Strm.metadata["length"].asInt() > 0){ + myConn.Send(RTMPStream::SendUSR(4, 1)); //send UCM StreamIsRecorded (4), stream 1 + } + //send streambegin + myConn.Send(RTMPStream::SendUSR(0, 1)); //send UCM StreamBegin (0), stream 1 + //and more reply + amfreply = AMF::Object("container", AMF::AMF0_DDV_CONTAINER); + amfreply.addContent(AMF::Object("", "onStatus")); //status reply + amfreply.addContent(AMF::Object("", (double)playTransaction)); //same transaction ID + amfreply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL)); //null - command info + amfreply.addContent(AMF::Object("")); //info + amfreply.getContentP(3)->addContent(AMF::Object("level", "status")); + amfreply.getContentP(3)->addContent(AMF::Object("code", "NetStream.Play.Start")); + amfreply.getContentP(3)->addContent(AMF::Object("description", "Playing!")); + amfreply.getContentP(3)->addContent(AMF::Object("details", "DDV")); + amfreply.getContentP(3)->addContent(AMF::Object("clientid", (double)1337)); + sendCommand(amfreply, playMessageType, playStreamId); + RTMPStream::chunk_snd_max = 102400; //100KiB + myConn.Send(RTMPStream::SendCTL(1, RTMPStream::chunk_snd_max)); //send chunk size max (msg 1) + //send dunno? + myConn.Send(RTMPStream::SendUSR(32, 1)); //send UCM no clue?, stream 1 + + parseData = true; + return; + } //play + if ((amfData.getContentP(0)->StrValue() == "seek")) { + //set reply number and stream name, actual reply is sent up in the ss.spool() handler + int playTransaction = amfData.getContentP(1)->NumValue(); + int playMessageType = messageType; + int playStreamId = streamId; + + AMF::Object amfReply("container", AMF::AMF0_DDV_CONTAINER); + amfReply.addContent(AMF::Object("", "onStatus")); //status reply + amfReply.addContent(amfData.getContent(1)); //same transaction ID + amfReply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL)); //null - command info + amfReply.addContent(AMF::Object("")); //info + amfReply.getContentP(3)->addContent(AMF::Object("level", "status")); + amfReply.getContentP(3)->addContent(AMF::Object("code", "NetStream.Seek.Notify")); + amfReply.getContentP(3)->addContent(AMF::Object("description", "Seeking to the specified time")); + amfReply.getContentP(3)->addContent(AMF::Object("details", "DDV")); + amfReply.getContentP(3)->addContent(AMF::Object("clientid", (double)1337)); + sendCommand(amfReply, playMessageType, playStreamId); + seek((long long int)amfData.getContentP(3)->NumValue()); + + //send a status reply + AMF::Object amfreply("container", AMF::AMF0_DDV_CONTAINER); + amfreply.addContent(AMF::Object("", "onStatus")); //status reply + amfreply.addContent(AMF::Object("", (double)playTransaction)); //same transaction ID + amfreply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL)); //null - command info + amfreply.addContent(AMF::Object("")); //info + amfreply.getContentP(3)->addContent(AMF::Object("level", "status")); + amfreply.getContentP(3)->addContent(AMF::Object("code", "NetStream.Play.Reset")); + amfreply.getContentP(3)->addContent(AMF::Object("description", "Playing and resetting...")); + amfreply.getContentP(3)->addContent(AMF::Object("details", "DDV")); + amfreply.getContentP(3)->addContent(AMF::Object("clientid", (double)1337)); + sendCommand(amfreply, playMessageType, playStreamId); + //send streamisrecorded if stream, well, is recorded. + if (myMeta.vod) { //isMember("length") && Strm.metadata["length"].asInt() > 0){ + myConn.Send(RTMPStream::SendUSR(4, 1)); //send UCM StreamIsRecorded (4), stream 1 + } + //send streambegin + myConn.Send(RTMPStream::SendUSR(0, 1)); //send UCM StreamBegin (0), stream 1 + //and more reply + amfreply = AMF::Object("container", AMF::AMF0_DDV_CONTAINER); + amfreply.addContent(AMF::Object("", "onStatus")); //status reply + amfreply.addContent(AMF::Object("", (double)playTransaction)); //same transaction ID + amfreply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL)); //null - command info + amfreply.addContent(AMF::Object("")); //info + amfreply.getContentP(3)->addContent(AMF::Object("level", "status")); + amfreply.getContentP(3)->addContent(AMF::Object("code", "NetStream.Play.Start")); + amfreply.getContentP(3)->addContent(AMF::Object("description", "Playing!")); + amfreply.getContentP(3)->addContent(AMF::Object("details", "DDV")); + amfreply.getContentP(3)->addContent(AMF::Object("clientid", (double)1337)); + sendCommand(amfreply, playMessageType, playStreamId); + RTMPStream::chunk_snd_max = 102400; //100KiB + myConn.Send(RTMPStream::SendCTL(1, RTMPStream::chunk_snd_max)); //send chunk size max (msg 1) + //send dunno? + myConn.Send(RTMPStream::SendUSR(32, 1)); //send UCM no clue?, stream 1 + + return; + } //seek + if ((amfData.getContentP(0)->StrValue() == "pauseRaw") || (amfData.getContentP(0)->StrValue() == "pause")) { + if (amfData.getContentP(3)->NumValue()) { + parseData = false; + //send a status reply + AMF::Object amfReply("container", AMF::AMF0_DDV_CONTAINER); + amfReply.addContent(AMF::Object("", "onStatus")); //status reply + amfReply.addContent(amfData.getContent(1)); //same transaction ID + amfReply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL)); //null - command info + amfReply.addContent(AMF::Object("")); //info + amfReply.getContentP(3)->addContent(AMF::Object("level", "status")); + amfReply.getContentP(3)->addContent(AMF::Object("code", "NetStream.Pause.Notify")); + amfReply.getContentP(3)->addContent(AMF::Object("description", "Pausing playback")); + amfReply.getContentP(3)->addContent(AMF::Object("details", "DDV")); + amfReply.getContentP(3)->addContent(AMF::Object("clientid", (double)1337)); + sendCommand(amfReply, playMessageType, playStreamId); + } else { + parseData = true; + //send a status reply + AMF::Object amfReply("container", AMF::AMF0_DDV_CONTAINER); + amfReply.addContent(AMF::Object("", "onStatus")); //status reply + amfReply.addContent(amfData.getContent(1)); //same transaction ID + amfReply.addContent(AMF::Object("", (double)0, AMF::AMF0_NULL)); //null - command info + amfReply.addContent(AMF::Object("")); //info + amfReply.getContentP(3)->addContent(AMF::Object("level", "status")); + amfReply.getContentP(3)->addContent(AMF::Object("code", "NetStream.Unpause.Notify")); + amfReply.getContentP(3)->addContent(AMF::Object("description", "Resuming playback")); + amfReply.getContentP(3)->addContent(AMF::Object("details", "DDV")); + amfReply.getContentP(3)->addContent(AMF::Object("clientid", (double)1337)); + sendCommand(amfReply, playMessageType, playStreamId); + } + return; + } //seek + +#if DEBUG >= 2 + fprintf(stderr, "AMF0 command not processed!\n%s\n", amfData.Print().c_str()); +#endif + } //parseAMFCommand + + void OutRTMP::bufferPacket(JSON::Value & pack){ + if (!trackMap.count(pack["trackid"].asInt())){ + //declined track; + return; + } + pack["trackid"] = trackMap[pack["trackid"].asInt()]; + long long unsigned int tNum = pack["trackid"].asInt(); + if (!bookKeeping.count(tNum)){ + return; + } + int pageNum = bookKeeping[tNum].pageNum; + std::string tmp = pack.toNetPacked(); + if (bookKeeping[tNum].curOffset > 8388608 && pack.isMember("keyframe") && pack["keyframe"]){ + Util::sleep(500); + //open new page + char nextPage[100]; + sprintf(nextPage, "%s%llu_%d", streamName.c_str(), tNum, bookKeeping[tNum].pageNum + bookKeeping[tNum].keyNum); + curPages[tNum].init(nextPage, 0, false); + bookKeeping[tNum].pageNum += bookKeeping[tNum].keyNum; + bookKeeping[tNum].keyNum = 0; + bookKeeping[tNum].curOffset = 0; + } + if (bookKeeping[tNum].curOffset + tmp.size() < curPages[tNum].len){ + bookKeeping[tNum].keyNum += (pack.isMember("keyframe") && pack["keyframe"]); + memcpy(curPages[tNum].mapped + bookKeeping[tNum].curOffset, tmp.data(), tmp.size()); + bookKeeping[tNum].curOffset += tmp.size(); + }else{ + bookKeeping[tNum].curOffset += tmp.size(); + DEBUG_MSG(DLVL_WARN, "Can't buffer frame on page %d, track %llu, time %lld, keyNum %d, offset %llu", pageNum, tNum, pack["time"].asInt(), bookKeeping[tNum].pageNum + bookKeeping[tNum].keyNum, bookKeeping[tNum].curOffset); + ///\todo Open next page plx + } + playerConn.keepAlive(); + } + + + void OutRTMP::negotiatePushTracks() { + char * tmp = playerConn.getData(); + if (!tmp){ + DEBUG_MSG(DLVL_FAIL, "No userpage allocated"); + return; + } + memset(tmp, 0, 30); + unsigned int i = 0; + for (std::map<int, DTSC::Track>::iterator it = meta_out.tracks.begin(); it != meta_out.tracks.end() && i < 5; it++){ + DEBUG_MSG(DLVL_DEVEL, "Negotiating tracknum for id %d", it->first); + (tmp + 6 * i)[0] = 0x80; + (tmp + 6 * i)[1] = 0x00; + (tmp + 6 * i)[2] = 0x00; + (tmp + 6 * i)[3] = 0x00; + (tmp + 6 * i)[4] = (it->first >> 8) & 0xFF; + (tmp + 6 * i)[5] = (it->first) & 0xFF; + i++; + } + playerConn.keepAlive(); + bool gotAllNumbers = false; + while (!gotAllNumbers){ + Util::sleep(100); + gotAllNumbers = true; + i = 0; + for (std::map<int, DTSC::Track>::iterator it = meta_out.tracks.begin(); it != meta_out.tracks.end() && i < 5; it++){ + unsigned long tNum = (((long)(tmp + (6 * i))[0]) << 24) | (((long)(tmp + (6 * i))[1]) << 16) | (((long)(tmp + (6 * i))[2]) << 8) | (long)(tmp + (6 * i))[3]; + unsigned short oldNum = (((long)(tmp + (6 * i))[4]) << 8) | (long)(tmp + (6 * i))[5]; + if( tNum & 0x80000000){ + gotAllNumbers = false; + break; + }else{ + DEBUG_MSG(DLVL_DEVEL, "Mapped %d -> %lu", oldNum, tNum); + trackMap[oldNum] = tNum; + } + i++; + } + } + for (std::map<int, int>::iterator it = trackMap.begin(); it != trackMap.end(); it++){ + char tmp[100]; + sprintf( tmp, "liveStream_%s%d", streamName.c_str(), it->second); + metaPages[it->second].init(std::string(tmp), 0, false); + DTSC::Meta tmpMeta = meta_out; + tmpMeta.tracks.clear(); + tmpMeta.tracks[it->second] = meta_out.tracks[it->first]; + tmpMeta.tracks[it->second].trackID = it->second; + JSON::Value tmpVal = tmpMeta.toJSON(); + std::string tmpStr = tmpVal.toNetPacked(); + memcpy(metaPages[it->second].mapped, tmpStr.data(), tmpStr.size()); + DEBUG_MSG(DLVL_DEVEL, "Written meta for track %d", it->second); + } + gotAllNumbers = false; + while (!gotAllNumbers){ + Util::sleep(100); + gotAllNumbers = true; + i = 0; + unsigned int j = 0; + //update Metadata; + JSON::Value jsonMeta; + JSON::fromDTMI((const unsigned char*)streamIndex.mapped + 8, streamIndex.len - 8, j, jsonMeta); + myMeta = DTSC::Meta(jsonMeta); + tmp = playerConn.getData(); + for (std::map<int, DTSC::Track>::iterator it = meta_out.tracks.begin(); it != meta_out.tracks.end() && i < 5; it++){ + unsigned long tNum = (((long)(tmp + (6 * i))[0]) << 24) | (((long)(tmp + (6 * i))[1]) << 16) | (((long)(tmp + (6 * i))[2]) << 8) | (long)(tmp + (6 * i))[3]; + if( tNum == 0xFFFFFFFF){ + DEBUG_MSG(DLVL_DEVEL, "Skipping a declined track"); + i++; + continue; + } + if(!myMeta.tracks.count(tNum)){ + gotAllNumbers = false; + break; + } + i++; + } + } + i = 0; + tmp = playerConn.getData(); + for (std::map<int, DTSC::Track>::iterator it = meta_out.tracks.begin(); it != meta_out.tracks.end() && i < 5; it++){ + unsigned long tNum = ((long)(tmp[6*i]) << 24) | ((long)(tmp[6 * i + 1]) << 16) | ((long)(tmp[6 * i + 2]) << 8) | tmp[6 * i + 3]; + if( tNum == 0xFFFFFFFF){ + tNum = ((long)(tmp[6 * i + 4]) << 8) | (long)tmp[6 * i + 5]; + DEBUG_MSG(DLVL_WARN, "Buffer declined track %i", trackMap[tNum]); + trackMap.erase(tNum); + tmp[6*i] = 0; + tmp[6*i+1] = 0; + tmp[6*i+2] = 0; + tmp[6*i+3] = 0; + tmp[6*i+4] = 0; + tmp[6*i+5] = 0; + }else{ + char firstPage[100]; + sprintf(firstPage, "%s%lu_%d", streamName.c_str(), tNum, 0); + curPages[tNum].init(firstPage, 0, false); + bookKeeping[tNum] = DTSCPageData(); + DEBUG_MSG(DLVL_WARN, "Buffer accepted track %lu", tNum); + } + i++; + } + } + + ///\brief Gets and parses one RTMP chunk at a time. + ///\param inputBuffer A buffer filled with chunk data. + void OutRTMP::parseChunk(Socket::Buffer & inputBuffer) { + //for DTSC conversion + static std::stringstream prebuffer; // Temporary buffer before sending real data + //for chunk parsing + static RTMPStream::Chunk next; + static FLV::Tag F; + static AMF::Object amfdata("empty", AMF::AMF0_DDV_CONTAINER); + static AMF::Object amfelem("empty", AMF::AMF0_DDV_CONTAINER); + static AMF::Object3 amf3data("empty", AMF::AMF3_DDV_CONTAINER); + static AMF::Object3 amf3elem("empty", AMF::AMF3_DDV_CONTAINER); + + while (next.Parse(inputBuffer)) { + + //send ACK if we received a whole window + if ((RTMPStream::rec_cnt - RTMPStream::rec_window_at > RTMPStream::rec_window_size)) { + RTMPStream::rec_window_at = RTMPStream::rec_cnt; + myConn.Send(RTMPStream::SendCTL(3, RTMPStream::rec_cnt)); //send ack (msg 3) + } + + switch (next.msg_type_id) { + case 0: //does not exist +#if DEBUG >= 2 + fprintf(stderr, "UNKN: Received a zero-type message. Possible data corruption? Aborting!\n"); +#endif + while (inputBuffer.size()) { + inputBuffer.get().clear(); + } + stop(); + myConn.close(); + break; //happens when connection breaks unexpectedly + case 1: //set chunk size + RTMPStream::chunk_rec_max = ntohl(*(int *)next.data.c_str()); +#if DEBUG >= 5 + fprintf(stderr, "CTRL: Set chunk size: %i\n", RTMPStream::chunk_rec_max); +#endif + break; + case 2: //abort message - we ignore this one +#if DEBUG >= 5 + fprintf(stderr, "CTRL: Abort message\n"); +#endif + //4 bytes of stream id to drop + break; + case 3: //ack +#if DEBUG >= 8 + fprintf(stderr, "CTRL: Acknowledgement\n"); +#endif + RTMPStream::snd_window_at = ntohl(*(int *)next.data.c_str()); + RTMPStream::snd_window_at = RTMPStream::snd_cnt; + break; + case 4: { + //2 bytes event type, rest = event data + //types: + //0 = stream begin, 4 bytes ID + //1 = stream EOF, 4 bytes ID + //2 = stream dry, 4 bytes ID + //3 = setbufferlen, 4 bytes ID, 4 bytes length + //4 = streamisrecorded, 4 bytes ID + //6 = pingrequest, 4 bytes data + //7 = pingresponse, 4 bytes data + //we don't need to process this +#if DEBUG >= 5 + short int ucmtype = ntohs(*(short int *)next.data.c_str()); + switch (ucmtype) { + case 0: + fprintf(stderr, "CTRL: UCM StreamBegin %i\n", ntohl(*((int *)(next.data.c_str() + 2)))); + break; + case 1: + fprintf(stderr, "CTRL: UCM StreamEOF %i\n", ntohl(*((int *)(next.data.c_str() + 2)))); + break; + case 2: + fprintf(stderr, "CTRL: UCM StreamDry %i\n", ntohl(*((int *)(next.data.c_str() + 2)))); + break; + case 3: + fprintf(stderr, "CTRL: UCM SetBufferLength %i %i\n", ntohl(*((int *)(next.data.c_str() + 2))), ntohl(*((int *)(next.data.c_str() + 6)))); + break; + case 4: + fprintf(stderr, "CTRL: UCM StreamIsRecorded %i\n", ntohl(*((int *)(next.data.c_str() + 2)))); + break; + case 6: + fprintf(stderr, "CTRL: UCM PingRequest %i\n", ntohl(*((int *)(next.data.c_str() + 2)))); + break; + case 7: + fprintf(stderr, "CTRL: UCM PingResponse %i\n", ntohl(*((int *)(next.data.c_str() + 2)))); + break; + default: + fprintf(stderr, "CTRL: UCM Unknown (%hi)\n", ucmtype); + break; + } +#endif + } + break; + case 5: //window size of other end +#if DEBUG >= 5 + fprintf(stderr, "CTRL: Window size\n"); +#endif + RTMPStream::rec_window_size = ntohl(*(int *)next.data.c_str()); + RTMPStream::rec_window_at = RTMPStream::rec_cnt; + myConn.Send(RTMPStream::SendCTL(3, RTMPStream::rec_cnt)); //send ack (msg 3) + break; + case 6: +#if DEBUG >= 5 + fprintf(stderr, "CTRL: Set peer bandwidth\n"); +#endif + //4 bytes window size, 1 byte limit type (ignored) + RTMPStream::snd_window_size = ntohl(*(int *)next.data.c_str()); + myConn.Send(RTMPStream::SendCTL(5, RTMPStream::snd_window_size)); //send window acknowledgement size (msg 5) + break; + case 8: //audio data + case 9: //video data + case 18: {//meta data + if (!isInitialized) { + DEBUG_MSG(DLVL_MEDIUM, "Received useless media data\n"); + myConn.close(); + break; + } + if (streamReset) { + //reset push data to empty, in case stream properties change + meta_out.reset(); + preBuf.clear(); + sending = false; + counter = 0; + streamReset = false; + } + F.ChunkLoader(next); + JSON::Value pack_out = F.toJSON(meta_out); + if ( !pack_out.isNull()){ + if ( !sending){ + counter++; + if (counter > 8){ + sending = true; + negotiatePushTracks(); + for (std::deque<JSON::Value>::iterator it = preBuf.begin(); it != preBuf.end(); it++){ + bufferPacket((*it)); + } + preBuf.clear(); //clear buffer + bufferPacket(pack_out); + }else{ + preBuf.push_back(pack_out); + } + }else{ + bufferPacket(pack_out); + } + } + break; + } + case 15: + DEBUG_MSG(DLVL_MEDIUM, "Received AMF3 data message"); + break; + case 16: + DEBUG_MSG(DLVL_MEDIUM, "Received AMF3 shared object"); + break; + case 17: { + DEBUG_MSG(DLVL_MEDIUM, "Received AMF3 command message"); + if (next.data[0] != 0) { + next.data = next.data.substr(1); + amf3data = AMF::parse3(next.data); +#if DEBUG >= 5 + amf3data.Print(); +#endif + } else { + DEBUG_MSG(DLVL_MEDIUM, "Received AMF3-0 command message"); + next.data = next.data.substr(1); + amfdata = AMF::parse(next.data); + parseAMFCommand(amfdata, 17, next.msg_stream_id); + } //parsing AMF0-style + } + break; + case 19: + DEBUG_MSG(DLVL_MEDIUM, "Received AMF0 shared object"); + break; + case 20: { //AMF0 command message + amfdata = AMF::parse(next.data); + parseAMFCommand(amfdata, 20, next.msg_stream_id); + } + break; + case 22: + DEBUG_MSG(DLVL_MEDIUM, "Received aggregate message"); + break; + default: + DEBUG_MSG(DLVL_FAIL, "Unknown chunk received! Probably protocol corruption, stopping parsing of incoming data."); + break; + } + } + } +} + diff --git a/src/output/output_rtmp.h b/src/output/output_rtmp.h new file mode 100644 index 00000000..1c0493fa --- /dev/null +++ b/src/output/output_rtmp.h @@ -0,0 +1,48 @@ +#include "output.h" +#include <mist/flv_tag.h> +#include <mist/amf.h> +#include <mist/rtmpchunks.h> + + +namespace Mist { + struct DTSCPageData { + DTSCPageData() : pageNum(0), keyNum(0), partNum(0), dataSize(0), curOffset(0), firstTime(0){} + int pageNum;///<The current page number + int keyNum;///<The number of keyframes in this page. + int partNum;///<The number of parts in this page. + unsigned long long int dataSize;///<The full size this page should be. + unsigned long long int curOffset;///<The current write offset in the page. + unsigned long long int firstTime;///<The first timestamp of the page. + }; + + class OutRTMP : public Output { + public: + OutRTMP(Socket::Connection & conn); + ~OutRTMP(); + static void init(Util::Config * cfg); + + void onRequest(); + void sendNext(); + void sendHeader(); + void bufferPacket(JSON::Value & pack); + protected: + DTSC::Meta meta_out; + void negotiatePushTracks(); + std::string app_name; + bool sending; + int counter; + bool streamReset; + int playTransaction;///<The transaction number of the reply. + int playStreamId;///<The stream id of the reply. + int playMessageType;///<The message type of the reply. + void parseChunk(Socket::Buffer & inputBuffer); + void parseAMFCommand(AMF::Object & amfData, int messageType, int streamId); + void sendCommand(AMF::Object & amfReply, int messageType, int streamId); + std::deque<JSON::Value> preBuf; + std::map<int,int> trackMap; + std::map<int,IPC::sharedPage> metaPages; + std::map<int,DTSCPageData> bookKeeping; + }; +} + +typedef Mist::OutRTMP mistOut; diff --git a/src/output/output_srt.cpp b/src/output/output_srt.cpp new file mode 100644 index 00000000..c6dd23f4 --- /dev/null +++ b/src/output/output_srt.cpp @@ -0,0 +1,82 @@ +#include "output_srt.h" +#include <mist/http_parser.h> +#include <mist/defines.h> +#include <iomanip> + +namespace Mist { + OutProgressiveSRT::OutProgressiveSRT(Socket::Connection & conn) : Output(conn) { + realTime = 0; + } + + void OutProgressiveSRT::onFail(){ + HTTP::Parser HTTP_S; + HTTP_S.Clean(); //make sure no parts of old requests are left in any buffers + HTTP_S.SetBody("Stream not found. Sorry, we tried."); + HTTP_S.SendResponse("404", "Stream not found", myConn); + Output::onFail(); + } + + OutProgressiveSRT::~OutProgressiveSRT() {} + + void OutProgressiveSRT::init(Util::Config * cfg){ + capa["desc"] = "Enables HTTP protocol subtitle streaming."; + capa["deps"] = "HTTP"; + capa["url_rel"] = "/$.srt"; + capa["url_match"] = "/$.srt"; + capa["url_handler"] = "http"; + capa["url_type"] = "subtitle"; + capa["socket"] = "http_srt"; + + cfg->addBasicConnectorOptions(capa); + config = cfg; + } + + void OutProgressiveSRT::sendNext(){ + char * dataPointer = 0; + int len = 0; + currentPacket.getString("data", dataPointer, len); + std::stringstream tmp; + if(!webVTT) { + tmp << lastNum++ << std::endl; + } + long long unsigned int time = currentPacket.getTime(); + char tmpBuf[50]; + int tmpLen = sprintf(tmpBuf, "%0.2llu:%0.2llu:%0.2llu,%0.3llu", (time / 3600000), ((time % 3600000) / 60000), (((time % 3600000) % 60000) / 1000), time % 1000); + tmp.write(tmpBuf, tmpLen); + tmp << " --> "; + time += currentPacket.getInt("duration"); + tmpLen = sprintf(tmpBuf, "%0.2llu:%0.2llu:%0.2llu,%0.3llu", (time / 3600000), ((time % 3600000) / 60000), (((time % 3600000) % 60000) / 1000), time % 1000); + tmp.write(tmpBuf, tmpLen); + tmp << std::endl; + myConn.SendNow(tmp.str()); + myConn.SendNow(dataPointer, len); + myConn.SendNow("\n"); + } + + void OutProgressiveSRT::sendHeader(){ + HTTP::Parser HTTP_S; + FLV::Tag tag; + HTTP_S.SetHeader("Content-Type", "text/plain"); + HTTP_S.protocol = "HTTP/1.0"; + myConn.SendNow(HTTP_S.BuildResponse("200", "OK")); + sentHeader = true; + } + + void OutProgressiveSRT::onRequest(){ + HTTP::Parser HTTP_R; + while (HTTP_R.Read(myConn)){ + DEBUG_MSG(DLVL_DEVEL, "Received request %s", HTTP_R.getUrl().c_str()); + lastNum = 0; + webVTT = (HTTP_R.url.find(".webvtt") != std::string::npos); + if (HTTP_R.GetVar("track") != ""){ + selectedTracks.insert(JSON::Value(HTTP_R.GetVar("track")).asInt()); + } + myConn.setHost(HTTP_R.GetHeader("X-Origin")); + streamName = HTTP_R.GetHeader("X-Stream"); + parseData = true; + wantRequest = false; + HTTP_R.Clean(); + } + } + +} diff --git a/src/output/output_srt.h b/src/output/output_srt.h new file mode 100644 index 00000000..06fd8dc9 --- /dev/null +++ b/src/output/output_srt.h @@ -0,0 +1,20 @@ +#include "output.h" + + +namespace Mist { + class OutProgressiveSRT : public Output { + public: + OutProgressiveSRT(Socket::Connection & conn); + ~OutProgressiveSRT(); + static void init(Util::Config * cfg); + void onRequest(); + void sendNext(); + void onFail(); + void sendHeader(); + protected: + bool webVTT; + int lastNum; + }; +} + +typedef Mist::OutProgressiveSRT mistOut; diff --git a/src/output/output_ts.cpp b/src/output/output_ts.cpp new file mode 100644 index 00000000..8b97afd8 --- /dev/null +++ b/src/output/output_ts.cpp @@ -0,0 +1,134 @@ +#include "output_ts.h" +#include <mist/http_parser.h> +#include <mist/defines.h> + +namespace Mist { + OutTS::OutTS(Socket::Connection & conn) : Output(conn){ + haveAvcc = false; + AudioCounter = 0; + VideoCounter = 0; + std::string tracks = config->getString("tracks"); + unsigned int currTrack = 0; + //loop over tracks, add any found track IDs to selectedTracks + if (tracks != ""){ + for (unsigned int i = 0; i < tracks.size(); ++i){ + if (tracks[i] >= '0' && tracks[i] <= '9'){ + currTrack = currTrack*10 + (tracks[i] - '0'); + }else{ + if (currTrack > 0){ + selectedTracks.insert(currTrack); + } + currTrack = 0; + } + } + if (currTrack > 0){ + selectedTracks.insert(currTrack); + } + } + streamName = config->getString("streamname"); + parseData = true; + wantRequest = false; + initialize(); + } + + OutTS::~OutTS() {} + + void OutTS::init(Util::Config * cfg){ + capa["name"] = "TS"; + capa["desc"] = "Enables the raw MPEG Transport Stream protocol over TCP."; + capa["deps"] = ""; + capa["required"]["streamname"]["name"] = "Stream"; + capa["required"]["streamname"]["help"] = "What streamname to serve. For multiple streams, add this protocol multiple times using different ports."; + capa["required"]["streamname"]["type"] = "str"; + capa["required"]["streamname"]["option"] = "--stream"; + capa["optional"]["tracks"]["name"] = "Tracks"; + capa["optional"]["tracks"]["help"] = "The track IDs of the stream that this connector will transmit separated by spaces"; + capa["optional"]["tracks"]["type"] = "str"; + capa["optional"]["tracks"]["option"] = "--tracks"; + capa["codecs"][0u][0u].append("H264"); + capa["codecs"][0u][1u].append("AAC"); + cfg->addOption("streamname", + JSON::fromString("{\"arg\":\"string\",\"short\":\"s\",\"long\":\"stream\",\"help\":\"The name of the stream that this connector will transmit.\"}")); + cfg->addOption("tracks", + JSON::fromString("{\"arg\":\"string\",\"value\":[\"\"],\"short\": \"t\",\"long\":\"tracks\",\"help\":\"The track IDs of the stream that this connector will transmit separated by spaces.\"}")); + cfg->addConnectorOptions(8888, capa); + config = cfg; + } + + void OutTS::sendNext(){ + Socket::Buffer ToPack; + char * ContCounter = 0; + bool IsKeyFrame = false; + + char * dataPointer = 0; + int dataLen = 0; + currentPacket.getString("data", dataPointer, dataLen); + + //detect packet type, and put converted data into ToPack. + if (myMeta.tracks[currentPacket.getTrackId()].type == "video"){ + ToPack.append(TS::Packet::getPESVideoLeadIn(0ul, currentPacket.getTime() * 90)); + + IsKeyFrame = currentPacket.getInt("keyframe"); + if (IsKeyFrame){ + if (!haveAvcc){ + avccbox.setPayload(myMeta.tracks[currentPacket.getTrackId()].init); + haveAvcc = true; + } + ToPack.append(avccbox.asAnnexB()); + } + unsigned int i = 0; + while (i + 4 < (unsigned int)dataLen){ + unsigned int ThisNaluSize = (dataPointer[i] << 24) + (dataPointer[i+1] << 16) + (dataPointer[i+2] << 8) + dataPointer[i+3]; + if (ThisNaluSize + i + 4 > (unsigned int)dataLen){ + DEBUG_MSG(DLVL_WARN, "Too big NALU detected (%u > %d) - skipping!", ThisNaluSize + i + 4, dataLen); + break; + } + ToPack.append("\000\000\000\001", 4); + i += 4; + ToPack.append(dataPointer + i, ThisNaluSize); + i += ThisNaluSize; + } + ContCounter = &VideoCounter; + }else if (myMeta.tracks[currentPacket.getTrackId()].type == "audio"){ + ToPack.append(TS::Packet::getPESAudioLeadIn(7+dataLen, currentPacket.getTime() * 90)); + ToPack.append(TS::GetAudioHeader(dataLen, myMeta.tracks[currentPacket.getTrackId()].init)); + ToPack.append(dataPointer, dataLen); + ContCounter = &AudioCounter; + } + + bool first = true; + //send TS packets + while (ToPack.size()){ + PackData.Clear(); + /// \todo Update according to sendHeader()'s generated data. + //0x100 - 1 + currentPacket.getTrackId() + if (myMeta.tracks[currentPacket.getTrackId()].type == "video"){ + PackData.PID(0x100); + }else{ + PackData.PID(0x101); + } + PackData.ContinuityCounter((*ContCounter)++); + if (first){ + PackData.UnitStart(1); + if (IsKeyFrame){ + PackData.RandomAccess(1); + PackData.PCR(currentPacket.getTime() * 27000); + } + first = false; + } + unsigned int toSend = PackData.AddStuffing(ToPack.bytes(184)); + std::string gonnaSend = ToPack.remove(toSend); + PackData.FillFree(gonnaSend); + myConn.SendNow(PackData.ToString(), 188); + } + } + + void OutTS::sendHeader(){ + /// \todo Update this to actually generate these from the selected tracks. + /// \todo ts_packet.h contains all neccesary info for this + myConn.SendNow(TS::PAT, 188); + myConn.SendNow(TS::PMT, 188); + sentHeader = true; + } + +} diff --git a/src/output/output_ts.h b/src/output/output_ts.h new file mode 100644 index 00000000..78efee19 --- /dev/null +++ b/src/output/output_ts.h @@ -0,0 +1,23 @@ +#include "output.h" +#include <mist/mp4_generic.h> +#include <mist/ts_packet.h> + +namespace Mist { + class OutTS : public Output { + public: + OutTS(Socket::Connection & conn); + ~OutTS(); + static void init(Util::Config * cfg); + void sendNext(); + void sendHeader(); + protected: + TS::Packet PackData; + unsigned int PacketNumber; + bool haveAvcc; + char VideoCounter; + char AudioCounter; + MP4::AVCC avccbox; + }; +} + +typedef Mist::OutTS mistOut;