Du kan inte välja fler än 25 ämnen Ämnen måste starta med en bokstav eller siffra, kan innehålla bindestreck ('-') och vara max 35 tecken långa.

strophejingle.bundle.js 145KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071
  1. var Base64=(function(){var keyStr="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=";var obj={encode:function(input){var output="";var chr1,chr2,chr3;var enc1,enc2,enc3,enc4;var i=0;do{chr1=input.charCodeAt(i++);chr2=input.charCodeAt(i++);chr3=input.charCodeAt(i++);enc1=chr1>>2;enc2=((chr1&3)<<4)|(chr2>>4);enc3=((chr2&15)<<2)|(chr3>>6);enc4=chr3&63;if(isNaN(chr2)){enc3=enc4=64}else{if(isNaN(chr3)){enc4=64}}output=output+keyStr.charAt(enc1)+keyStr.charAt(enc2)+keyStr.charAt(enc3)+keyStr.charAt(enc4)}while(i<input.length);return output},decode:function(input){var output="";var chr1,chr2,chr3;var enc1,enc2,enc3,enc4;var i=0;input=input.replace(/[^A-Za-z0-9\+\/\=]/g,"");do{enc1=keyStr.indexOf(input.charAt(i++));enc2=keyStr.indexOf(input.charAt(i++));enc3=keyStr.indexOf(input.charAt(i++));enc4=keyStr.indexOf(input.charAt(i++));chr1=(enc1<<2)|(enc2>>4);chr2=((enc2&15)<<4)|(enc3>>2);chr3=((enc3&3)<<6)|enc4;output=output+String.fromCharCode(chr1);if(enc3!=64){output=output+String.fromCharCode(chr2)}if(enc4!=64){output=output+String.fromCharCode(chr3)}}while(i<input.length);return output}};return obj})();var hexcase=0;var b64pad="=";var chrsz=8;function hex_sha1(s){return binb2hex(core_sha1(str2binb(s),s.length*chrsz))}function b64_sha1(s){return binb2b64(core_sha1(str2binb(s),s.length*chrsz))}function str_sha1(s){return binb2str(core_sha1(str2binb(s),s.length*chrsz))}function hex_hmac_sha1(key,data){return binb2hex(core_hmac_sha1(key,data))}function b64_hmac_sha1(key,data){return binb2b64(core_hmac_sha1(key,data))}function str_hmac_sha1(key,data){return binb2str(core_hmac_sha1(key,data))}function sha1_vm_test(){return hex_sha1("abc")=="a9993e364706816aba3e25717850c26c9cd0d89d"}function core_sha1(x,len){x[len>>5]|=128<<(24-len%32);x[((len+64>>9)<<4)+15]=len;var w=new Array(80);var a=1732584193;var b=-271733879;var c=-1732584194;var d=271733878;var e=-1009589776;var i,j,t,olda,oldb,oldc,oldd,olde;for(i=0;i<x.length;i+=16){olda=a;oldb=b;oldc=c;oldd=d;olde=e;for(j=0;j<80;j++){if(j<16){w[j]=x[i+j]}else{w[j]=rol(w[j-3]^w[j-8]^w[j-14]^w[j-16],1)}t=safe_add(safe_add(rol(a,5),sha1_ft(j,b,c,d)),safe_add(safe_add(e,w[j]),sha1_kt(j)));e=d;d=c;c=rol(b,30);b=a;a=t}a=safe_add(a,olda);b=safe_add(b,oldb);c=safe_add(c,oldc);d=safe_add(d,oldd);e=safe_add(e,olde)}return[a,b,c,d,e]}function sha1_ft(t,b,c,d){if(t<20){return(b&c)|((~b)&d)}if(t<40){return b^c^d}if(t<60){return(b&c)|(b&d)|(c&d)}return b^c^d}function sha1_kt(t){return(t<20)?1518500249:(t<40)?1859775393:(t<60)?-1894007588:-899497514}function core_hmac_sha1(key,data){var bkey=str2binb(key);if(bkey.length>16){bkey=core_sha1(bkey,key.length*chrsz)}var ipad=new Array(16),opad=new Array(16);for(var i=0;i<16;i++){ipad[i]=bkey[i]^909522486;opad[i]=bkey[i]^1549556828}var hash=core_sha1(ipad.concat(str2binb(data)),512+data.length*chrsz);return core_sha1(opad.concat(hash),512+160)}function safe_add(x,y){var lsw=(x&65535)+(y&65535);var msw=(x>>16)+(y>>16)+(lsw>>16);return(msw<<16)|(lsw&65535)}function rol(num,cnt){return(num<<cnt)|(num>>>(32-cnt))}function str2binb(str){var bin=[];var mask=(1<<chrsz)-1;for(var i=0;i<str.length*chrsz;i+=chrsz){bin[i>>5]|=(str.charCodeAt(i/chrsz)&mask)<<(32-chrsz-i%32)}return bin}function binb2str(bin){var str="";var mask=(1<<chrsz)-1;for(var i=0;i<bin.length*32;i+=chrsz){str+=String.fromCharCode((bin[i>>5]>>>(32-chrsz-i%32))&mask)}return str}function binb2hex(binarray){var hex_tab=hexcase?"0123456789ABCDEF":"0123456789abcdef";var str="";for(var i=0;i<binarray.length*4;i++){str+=hex_tab.charAt((binarray[i>>2]>>((3-i%4)*8+4))&15)+hex_tab.charAt((binarray[i>>2]>>((3-i%4)*8))&15)}return str}function binb2b64(binarray){var tab="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";var str="";var triplet,j;for(var i=0;i<binarray.length*4;i+=3){triplet=(((binarray[i>>2]>>8*(3-i%4))&255)<<16)|(((binarray[i+1>>2]>>8*(3-(i+1)%4))&255)<<8)|((binarray[i+2>>2]>>8*(3-(i+2)%4))&255);for(j=0;j<4;j++){if(i*8+j*6>binarray.length*32){str+=b64pad}else{str+=tab.charAt((triplet>>6*(3-j))&63)}}}return str}var MD5=(function(){var hexcase=0;var b64pad="";var chrsz=8;var safe_add=function(x,y){var lsw=(x&65535)+(y&65535);var msw=(x>>16)+(y>>16)+(lsw>>16);return(msw<<16)|(lsw&65535)};var bit_rol=function(num,cnt){return(num<<cnt)|(num>>>(32-cnt))};var str2binl=function(str){var bin=[];var mask=(1<<chrsz)-1;for(var i=0;i<str.length*chrsz;i+=chrsz){bin[i>>5]|=(str.charCodeAt(i/chrsz)&mask)<<(i%32)}return bin};var binl2str=function(bin){var str="";var mask=(1<<chrsz)-1;for(var i=0;i<bin.length*32;i+=chrsz){str+=String.fromCharCode((bin[i>>5]>>>(i%32))&mask)}return str};var binl2hex=function(binarray){var hex_tab=hexcase?"0123456789ABCDEF":"0123456789abcdef";var str="";for(var i=0;i<binarray.length*4;i++){str+=hex_tab.charAt((binarray[i>>2]>>((i%4)*8+4))&15)+hex_tab.charAt((binarray[i>>2]>>((i%4)*8))&15)}return str};var binl2b64=function(binarray){var tab="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";var str="";var triplet,j;for(var i=0;i<binarray.length*4;i+=3){triplet=(((binarray[i>>2]>>8*(i%4))&255)<<16)|(((binarray[i+1>>2]>>8*((i+1)%4))&255)<<8)|((binarray[i+2>>2]>>8*((i+2)%4))&255);for(j=0;j<4;j++){if(i*8+j*6>binarray.length*32){str+=b64pad}else{str+=tab.charAt((triplet>>6*(3-j))&63)}}}return str};var md5_cmn=function(q,a,b,x,s,t){return safe_add(bit_rol(safe_add(safe_add(a,q),safe_add(x,t)),s),b)};var md5_ff=function(a,b,c,d,x,s,t){return md5_cmn((b&c)|((~b)&d),a,b,x,s,t)};var md5_gg=function(a,b,c,d,x,s,t){return md5_cmn((b&d)|(c&(~d)),a,b,x,s,t)};var md5_hh=function(a,b,c,d,x,s,t){return md5_cmn(b^c^d,a,b,x,s,t)};var md5_ii=function(a,b,c,d,x,s,t){return md5_cmn(c^(b|(~d)),a,b,x,s,t)};var core_md5=function(x,len){x[len>>5]|=128<<((len)%32);x[(((len+64)>>>9)<<4)+14]=len;var a=1732584193;var b=-271733879;var c=-1732584194;var d=271733878;var olda,oldb,oldc,oldd;for(var i=0;i<x.length;i+=16){olda=a;oldb=b;oldc=c;oldd=d;a=md5_ff(a,b,c,d,x[i+0],7,-680876936);d=md5_ff(d,a,b,c,x[i+1],12,-389564586);c=md5_ff(c,d,a,b,x[i+2],17,606105819);b=md5_ff(b,c,d,a,x[i+3],22,-1044525330);a=md5_ff(a,b,c,d,x[i+4],7,-176418897);d=md5_ff(d,a,b,c,x[i+5],12,1200080426);c=md5_ff(c,d,a,b,x[i+6],17,-1473231341);b=md5_ff(b,c,d,a,x[i+7],22,-45705983);a=md5_ff(a,b,c,d,x[i+8],7,1770035416);d=md5_ff(d,a,b,c,x[i+9],12,-1958414417);c=md5_ff(c,d,a,b,x[i+10],17,-42063);b=md5_ff(b,c,d,a,x[i+11],22,-1990404162);a=md5_ff(a,b,c,d,x[i+12],7,1804603682);d=md5_ff(d,a,b,c,x[i+13],12,-40341101);c=md5_ff(c,d,a,b,x[i+14],17,-1502002290);b=md5_ff(b,c,d,a,x[i+15],22,1236535329);a=md5_gg(a,b,c,d,x[i+1],5,-165796510);d=md5_gg(d,a,b,c,x[i+6],9,-1069501632);c=md5_gg(c,d,a,b,x[i+11],14,643717713);b=md5_gg(b,c,d,a,x[i+0],20,-373897302);a=md5_gg(a,b,c,d,x[i+5],5,-701558691);d=md5_gg(d,a,b,c,x[i+10],9,38016083);c=md5_gg(c,d,a,b,x[i+15],14,-660478335);b=md5_gg(b,c,d,a,x[i+4],20,-405537848);a=md5_gg(a,b,c,d,x[i+9],5,568446438);d=md5_gg(d,a,b,c,x[i+14],9,-1019803690);c=md5_gg(c,d,a,b,x[i+3],14,-187363961);b=md5_gg(b,c,d,a,x[i+8],20,1163531501);a=md5_gg(a,b,c,d,x[i+13],5,-1444681467);d=md5_gg(d,a,b,c,x[i+2],9,-51403784);c=md5_gg(c,d,a,b,x[i+7],14,1735328473);b=md5_gg(b,c,d,a,x[i+12],20,-1926607734);a=md5_hh(a,b,c,d,x[i+5],4,-378558);d=md5_hh(d,a,b,c,x[i+8],11,-2022574463);c=md5_hh(c,d,a,b,x[i+11],16,1839030562);b=md5_hh(b,c,d,a,x[i+14],23,-35309556);a=md5_hh(a,b,c,d,x[i+1],4,-1530992060);d=md5_hh(d,a,b,c,x[i+4],11,1272893353);c=md5_hh(c,d,a,b,x[i+7],16,-155497632);b=md5_hh(b,c,d,a,x[i+10],23,-1094730640);a=md5_hh(a,b,c,d,x[i+13],4,681279174);d=md5_hh(d,a,b,c,x[i+0],11,-358537222);c=md5_hh(c,d,a,b,x[i+3],16,-722521979);b=md5_hh(b,c,d,a,x[i+6],23,76029189);a=md5_hh(a,b,c,d,x[i+9],4,-640364487);d=md5_hh(d,a,b,c,x[i+12],11,-421815835);c=md5_hh(c,d,a,b,x[i+15],16,530742520);b=md5_hh(b,c,d,a,x[i+2],23,-995338651);a=md5_ii(a,b,c,d,x[i+0],6,-198630844);d=md5_ii(d,a,b,c,x[i+7],10,1126891415);c=md5_ii(c,d,a,b,x[i+14],15,-1416354905);b=md5_ii(b,c,d,a,x[i+5],21,-57434055);a=md5_ii(a,b,c,d,x[i+12],6,1700485571);d=md5_ii(d,a,b,c,x[i+3],10,-1894986606);c=md5_ii(c,d,a,b,x[i+10],15,-1051523);b=md5_ii(b,c,d,a,x[i+1],21,-2054922799);a=md5_ii(a,b,c,d,x[i+8],6,1873313359);d=md5_ii(d,a,b,c,x[i+15],10,-30611744);c=md5_ii(c,d,a,b,x[i+6],15,-1560198380);b=md5_ii(b,c,d,a,x[i+13],21,1309151649);a=md5_ii(a,b,c,d,x[i+4],6,-145523070);d=md5_ii(d,a,b,c,x[i+11],10,-1120210379);c=md5_ii(c,d,a,b,x[i+2],15,718787259);b=md5_ii(b,c,d,a,x[i+9],21,-343485551);a=safe_add(a,olda);b=safe_add(b,oldb);c=safe_add(c,oldc);d=safe_add(d,oldd)}return[a,b,c,d]};var core_hmac_md5=function(key,data){var bkey=str2binl(key);if(bkey.length>16){bkey=core_md5(bkey,key.length*chrsz)}var ipad=new Array(16),opad=new Array(16);for(var i=0;i<16;i++){ipad[i]=bkey[i]^909522486;opad[i]=bkey[i]^1549556828}var hash=core_md5(ipad.concat(str2binl(data)),512+data.length*chrsz);return core_md5(opad.concat(hash),512+128)};var obj={hexdigest:function(s){return binl2hex(core_md5(str2binl(s),s.length*chrsz))},b64digest:function(s){return binl2b64(core_md5(str2binl(s),s.length*chrsz))},hash:function(s){return binl2str(core_md5(str2binl(s),s.length*chrsz))},hmac_hexdigest:function(key,data){return binl2hex(core_hmac_md5(key,data))},hmac_b64digest:function(key,data){return binl2b64(core_hmac_md5(key,data))},hmac_hash:function(key,data){return binl2str(core_hmac_md5(key,data))},test:function(){return MD5.hexdigest("abc")==="900150983cd24fb0d6963f7d28e17f72"}};return obj})();if(!Function.prototype.bind){Function.prototype.bind=function(obj){var func=this;var _slice=Array.prototype.slice;var _concat=Array.prototype.concat;var _args=_slice.call(arguments,1);return function(){return func.apply(obj?obj:this,_concat.call(_args,_slice.call(arguments,0)))}}}if(!Array.prototype.indexOf){Array.prototype.indexOf=function(elt){var len=this.length;var from=Number(arguments[1])||0;from=(from<0)?Math.ceil(from):Math.floor(from);if(from<0){from+=len}for(;from<len;from++){if(from in this&&this[from]===elt){return from}}return -1}}(function(callback){var Strophe;function $build(name,attrs){return new Strophe.Builder(name,attrs)}function $msg(attrs){return new Strophe.Builder("message",attrs)}function $iq(attrs){return new Strophe.Builder("iq",attrs)}function $pres(attrs){return new Strophe.Builder("presence",attrs)}Strophe={VERSION:"1.1.0",NS:{HTTPBIND:"http://jabber.org/protocol/httpbind",BOSH:"urn:xmpp:xbosh",CLIENT:"jabber:client",AUTH:"jabber:iq:auth",ROSTER:"jabber:iq:roster",PROFILE:"jabber:iq:profile",DISCO_INFO:"http://jabber.org/protocol/disco#info",DISCO_ITEMS:"http://jabber.org/protocol/disco#items",MUC:"http://jabber.org/protocol/muc",SASL:"urn:ietf:params:xml:ns:xmpp-sasl",STREAM:"http://etherx.jabber.org/streams",BIND:"urn:ietf:params:xml:ns:xmpp-bind",SESSION:"urn:ietf:params:xml:ns:xmpp-session",VERSION:"jabber:iq:version",STANZAS:"urn:ietf:params:xml:ns:xmpp-stanzas",XHTML_IM:"http://jabber.org/protocol/xhtml-im",XHTML:"http://www.w3.org/1999/xhtml"},XHTML:{tags:["a","blockquote","br","cite","em","img","li","ol","p","span","strong","ul","body"],attributes:{a:["href"],blockquote:["style"],br:[],cite:["style"],em:[],img:["src","alt","style","height","width"],li:["style"],ol:["style"],p:["style"],span:["style"],strong:[],ul:["style"],body:[]},css:["background-color","color","font-family","font-size","font-style","font-weight","margin-left","margin-right","text-align","text-decoration"],validTag:function(tag){for(var i=0;i<Strophe.XHTML.tags.length;i++){if(tag==Strophe.XHTML.tags[i]){return true}}return false},validAttribute:function(tag,attribute){if(typeof Strophe.XHTML.attributes[tag]!=="undefined"&&Strophe.XHTML.attributes[tag].length>0){for(var i=0;i<Strophe.XHTML.attributes[tag].length;i++){if(attribute==Strophe.XHTML.attributes[tag][i]){return true}}}return false},validCSS:function(style){for(var i=0;i<Strophe.XHTML.css.length;i++){if(style==Strophe.XHTML.css[i]){return true}}return false}},Status:{ERROR:0,CONNECTING:1,CONNFAIL:2,AUTHENTICATING:3,AUTHFAIL:4,CONNECTED:5,DISCONNECTED:6,DISCONNECTING:7,ATTACHED:8},LogLevel:{DEBUG:0,INFO:1,WARN:2,ERROR:3,FATAL:4},ElementType:{NORMAL:1,TEXT:3,CDATA:4,FRAGMENT:11},TIMEOUT:1.1,SECONDARY_TIMEOUT:0.1,addNamespace:function(name,value){Strophe.NS[name]=value},forEachChild:function(elem,elemName,func){var i,childNode;for(i=0;i<elem.childNodes.length;i++){childNode=elem.childNodes[i];if(childNode.nodeType==Strophe.ElementType.NORMAL&&(!elemName||this.isTagEqual(childNode,elemName))){func(childNode)}}},isTagEqual:function(el,name){return el.tagName.toLowerCase()==name.toLowerCase()},_xmlGenerator:null,_makeGenerator:function(){var doc;if(document.implementation.createDocument===undefined||document.implementation.createDocument&&document.documentMode&&document.documentMode<10){doc=this._getIEXmlDom();doc.appendChild(doc.createElement("strophe"))}else{doc=document.implementation.createDocument("jabber:client","strophe",null)}return doc},xmlGenerator:function(){if(!Strophe._xmlGenerator){Strophe._xmlGenerator=Strophe._makeGenerator()}return Strophe._xmlGenerator},_getIEXmlDom:function(){var doc=null;var docStrings=["Msxml2.DOMDocument.6.0","Msxml2.DOMDocument.5.0","Msxml2.DOMDocument.4.0","MSXML2.DOMDocument.3.0","MSXML2.DOMDocument","MSXML.DOMDocument","Microsoft.XMLDOM"];for(var d=0;d<docStrings.length;d++){if(doc===null){try{doc=new ActiveXObject(docStrings[d])}catch(e){doc=null}}else{break}}return doc},xmlElement:function(name){if(!name){return null}var node=Strophe.xmlGenerator().createElement(name);var a,i,k;for(a=1;a<arguments.length;a++){if(!arguments[a]){continue}if(typeof(arguments[a])=="string"||typeof(arguments[a])=="number"){node.appendChild(Strophe.xmlTextNode(arguments[a]))}else{if(typeof(arguments[a])=="object"&&typeof(arguments[a].sort)=="function"){for(i=0;i<arguments[a].length;i++){if(typeof(arguments[a][i])=="object"&&typeof(arguments[a][i].sort)=="function"){node.setAttribute(arguments[a][i][0],arguments[a][i][1])}}}else{if(typeof(arguments[a])=="object"){for(k in arguments[a]){if(arguments[a].hasOwnProperty(k)){node.setAttribute(k,arguments[a][k])}}}}}}return node},xmlescape:function(text){text=text.replace(/\&/g,"&amp;");text=text.replace(/</g,"&lt;");text=text.replace(/>/g,"&gt;");text=text.replace(/'/g,"&apos;");text=text.replace(/"/g,"&quot;");return text},xmlTextNode:function(text){return Strophe.xmlGenerator().createTextNode(text)},xmlHtmlNode:function(html){if(window.DOMParser){parser=new DOMParser();node=parser.parseFromString(html,"text/xml")}else{node=new ActiveXObject("Microsoft.XMLDOM");node.async="false";node.loadXML(html)}return node},getText:function(elem){if(!elem){return null}var str="";if(elem.childNodes.length===0&&elem.nodeType==Strophe.ElementType.TEXT){str+=elem.nodeValue}for(var i=0;i<elem.childNodes.length;i++){if(elem.childNodes[i].nodeType==Strophe.ElementType.TEXT){str+=elem.childNodes[i].nodeValue}}return Strophe.xmlescape(str)},copyElement:function(elem){var i,el;if(elem.nodeType==Strophe.ElementType.NORMAL){el=Strophe.xmlElement(elem.tagName);for(i=0;i<elem.attributes.length;i++){el.setAttribute(elem.attributes[i].nodeName.toLowerCase(),elem.attributes[i].value)}for(i=0;i<elem.childNodes.length;i++){el.appendChild(Strophe.copyElement(elem.childNodes[i]))}}else{if(elem.nodeType==Strophe.ElementType.TEXT){el=Strophe.xmlGenerator().createTextNode(elem.nodeValue)}}return el},createHtml:function(elem){var i,el,j,tag,attribute,value,css,cssAttrs,attr,cssName,cssValue,children,child;if(elem.nodeType==Strophe.ElementType.NORMAL){tag=elem.nodeName.toLowerCase();if(Strophe.XHTML.validTag(tag)){try{el=Strophe.xmlElement(tag);for(i=0;i<Strophe.XHTML.attributes[tag].length;i++){attribute=Strophe.XHTML.attributes[tag][i];value=elem.getAttribute(attribute);if(typeof value=="undefined"||value===null||value===""||value===false||value===0){continue}if(attribute=="style"&&typeof value=="object"){if(typeof value.cssText!="undefined"){value=value.cssText}}if(attribute=="style"){css=[];cssAttrs=value.split(";");for(j=0;j<cssAttrs.length;j++){attr=cssAttrs[j].split(":");cssName=attr[0].replace(/^\s*/,"").replace(/\s*$/,"").toLowerCase();if(Strophe.XHTML.validCSS(cssName)){cssValue=attr[1].replace(/^\s*/,"").replace(/\s*$/,"");css.push(cssName+": "+cssValue)}}if(css.length>0){value=css.join("; ");el.setAttribute(attribute,value)}}else{el.setAttribute(attribute,value)}}for(i=0;i<elem.childNodes.length;i++){el.appendChild(Strophe.createHtml(elem.childNodes[i]))}}catch(e){el=Strophe.xmlTextNode("")}}else{el=Strophe.xmlGenerator().createDocumentFragment();for(i=0;i<elem.childNodes.length;i++){el.appendChild(Strophe.createHtml(elem.childNodes[i]))}}}else{if(elem.nodeType==Strophe.ElementType.FRAGMENT){el=Strophe.xmlGenerator().createDocumentFragment();for(i=0;i<elem.childNodes.length;i++){el.appendChild(Strophe.createHtml(elem.childNodes[i]))}}else{if(elem.nodeType==Strophe.ElementType.TEXT){el=Strophe.xmlTextNode(elem.nodeValue)}}}return el},escapeNode:function(node){return node.replace(/^\s+|\s+$/g,"").replace(/\\/g,"\\5c").replace(/ /g,"\\20").replace(/\"/g,"\\22").replace(/\&/g,"\\26").replace(/\'/g,"\\27").replace(/\//g,"\\2f").replace(/:/g,"\\3a").replace(/</g,"\\3c").replace(/>/g,"\\3e").replace(/@/g,"\\40")},unescapeNode:function(node){return node.replace(/\\20/g," ").replace(/\\22/g,'"').replace(/\\26/g,"&").replace(/\\27/g,"'").replace(/\\2f/g,"/").replace(/\\3a/g,":").replace(/\\3c/g,"<").replace(/\\3e/g,">").replace(/\\40/g,"@").replace(/\\5c/g,"\\")},getNodeFromJid:function(jid){if(jid.indexOf("@")<0){return null}return jid.split("@")[0]},getDomainFromJid:function(jid){var bare=Strophe.getBareJidFromJid(jid);if(bare.indexOf("@")<0){return bare}else{var parts=bare.split("@");parts.splice(0,1);return parts.join("@")}},getResourceFromJid:function(jid){var s=jid.split("/");if(s.length<2){return null}s.splice(0,1);return s.join("/")},getBareJidFromJid:function(jid){return jid?jid.split("/")[0]:null},log:function(level,msg){return},debug:function(msg){this.log(this.LogLevel.DEBUG,msg)},info:function(msg){this.log(this.LogLevel.INFO,msg)},warn:function(msg){this.log(this.LogLevel.WARN,msg)},error:function(msg){this.log(this.LogLevel.ERROR,msg)},fatal:function(msg){this.log(this.LogLevel.FATAL,msg)},serialize:function(elem){var result;if(!elem){return null}if(typeof(elem.tree)==="function"){elem=elem.tree()}var nodeName=elem.nodeName;var i,child;if(elem.getAttribute("_realname")){nodeName=elem.getAttribute("_realname")}result="<"+nodeName;for(i=0;i<elem.attributes.length;i++){if(elem.attributes[i].nodeName!="_realname"){result+=" "+elem.attributes[i].nodeName.toLowerCase()+"='"+elem.attributes[i].value.replace(/&/g,"&amp;").replace(/\'/g,"&apos;").replace(/>/g,"&gt;").replace(/</g,"&lt;")+"'"}}if(elem.childNodes.length>0){result+=">";for(i=0;i<elem.childNodes.length;i++){child=elem.childNodes[i];switch(child.nodeType){case Strophe.ElementType.NORMAL:result+=Strophe.serialize(child);break;case Strophe.ElementType.TEXT:result+=Strophe.xmlescape(child.nodeValue);break;case Strophe.ElementType.CDATA:result+="<![CDATA["+child.nodeValue+"]]>"}}result+="</"+nodeName+">"}else{result+="/>"}return result},_requestId:0,_connectionPlugins:{},addConnectionPlugin:function(name,ptype){Strophe._connectionPlugins[name]=ptype}};Strophe.Builder=function(name,attrs){if(name=="presence"||name=="message"||name=="iq"){if(attrs&&!attrs.xmlns){attrs.xmlns=Strophe.NS.CLIENT}else{if(!attrs){attrs={xmlns:Strophe.NS.CLIENT}}}}this.nodeTree=Strophe.xmlElement(name,attrs);this.node=this.nodeTree};Strophe.Builder.prototype={tree:function(){return this.nodeTree},toString:function(){return Strophe.serialize(this.nodeTree)},up:function(){this.node=this.node.parentNode;return this},attrs:function(moreattrs){for(var k in moreattrs){if(moreattrs.hasOwnProperty(k)){this.node.setAttribute(k,moreattrs[k])}}return this},c:function(name,attrs,text){var child=Strophe.xmlElement(name,attrs,text);this.node.appendChild(child);if(!text){this.node=child}return this},cnode:function(elem){var impNode;var xmlGen=Strophe.xmlGenerator();try{impNode=(xmlGen.importNode!==undefined)}catch(e){impNode=false}var newElem=impNode?xmlGen.importNode(elem,true):Strophe.copyElement(elem);this.node.appendChild(newElem);this.node=newElem;return this},t:function(text){var child=Strophe.xmlTextNode(text);this.node.appendChild(child);return this},h:function(html){var fragment=document.createElement("body");fragment.innerHTML=html;var xhtml=Strophe.createHtml(fragment);while(xhtml.childNodes.length>0){this.node.appendChild(xhtml.childNodes[0])}return this}};Strophe.Handler=function(handler,ns,name,type,id,from,options){this.handler=handler;this.ns=ns;this.name=name;this.type=type;this.id=id;this.options=options||{matchBare:false};if(!this.options.matchBare){this.options.matchBare=false}if(this.options.matchBare){this.from=from?Strophe.getBareJidFromJid(from):null}else{this.from=from}this.user=true};Strophe.Handler.prototype={isMatch:function(elem){var nsMatch;var from=null;if(this.options.matchBare){from=Strophe.getBareJidFromJid(elem.getAttribute("from"))}else{from=elem.getAttribute("from")}nsMatch=false;if(!this.ns){nsMatch=true}else{var that=this;Strophe.forEachChild(elem,null,function(elem){if(elem.getAttribute("xmlns")==that.ns){nsMatch=true}});nsMatch=nsMatch||elem.getAttribute("xmlns")==this.ns}if(nsMatch&&(!this.name||Strophe.isTagEqual(elem,this.name))&&(!this.type||elem.getAttribute("type")==this.type)&&(!this.id||elem.getAttribute("id")==this.id)&&(!this.from||from==this.from)){return true}return false},run:function(elem){var result=null;try{result=this.handler(elem)}catch(e){if(e.sourceURL){Strophe.fatal("error: "+this.handler+" "+e.sourceURL+":"+e.line+" - "+e.name+": "+e.message)}else{if(e.fileName){if(typeof(console)!="undefined"){console.trace();console.error(this.handler," - error - ",e,e.message)}Strophe.fatal("error: "+this.handler+" "+e.fileName+":"+e.lineNumber+" - "+e.name+": "+e.message)}else{Strophe.fatal("error: "+e.message+"\n"+e.stack)}}throw e}return result},toString:function(){return"{Handler: "+this.handler+"("+this.name+","+this.id+","+this.ns+")}"}};Strophe.TimedHandler=function(period,handler){this.period=period;this.handler=handler;this.lastCalled=new Date().getTime();this.user=true};Strophe.TimedHandler.prototype={run:function(){this.lastCalled=new Date().getTime();return this.handler()},reset:function(){this.lastCalled=new Date().getTime()},toString:function(){return"{TimedHandler: "+this.handler+"("+this.period+")}"}};Strophe.Connection=function(service,options){this.service=service;this._options=options||{};var proto=this._options.protocol||"";if(service.indexOf("ws:")===0||service.indexOf("wss:")===0||proto.indexOf("ws")===0){this._proto=new Strophe.Websocket(this)}else{this._proto=new Strophe.Bosh(this)}this.jid="";this.domain=null;this.features=null;this._sasl_data={};this.do_session=false;this.do_bind=false;this.timedHandlers=[];this.handlers=[];this.removeTimeds=[];this.removeHandlers=[];this.addTimeds=[];this.addHandlers=[];this._authentication={};this._idleTimeout=null;this._disconnectTimeout=null;this.do_authentication=true;this.authenticated=false;this.disconnecting=false;this.connected=false;this.errors=0;this.paused=false;this._data=[];this._uniqueId=0;this._sasl_success_handler=null;this._sasl_failure_handler=null;this._sasl_challenge_handler=null;this.maxRetries=5;this._idleTimeout=setTimeout(this._onIdle.bind(this),100);for(var k in Strophe._connectionPlugins){if(Strophe._connectionPlugins.hasOwnProperty(k)){var ptype=Strophe._connectionPlugins[k];var F=function(){};F.prototype=ptype;this[k]=new F();this[k].init(this)}}};Strophe.Connection.prototype={reset:function(){this._proto._reset();this.do_session=false;this.do_bind=false;this.timedHandlers=[];this.handlers=[];this.removeTimeds=[];this.removeHandlers=[];this.addTimeds=[];this.addHandlers=[];this._authentication={};this.authenticated=false;this.disconnecting=false;this.connected=false;this.errors=0;this._requests=[];this._uniqueId=0},pause:function(){this.paused=true},resume:function(){this.paused=false},getUniqueId:function(suffix){if(typeof(suffix)=="string"||typeof(suffix)=="number"){return ++this._uniqueId+":"+suffix}else{return ++this._uniqueId+""}},connect:function(jid,pass,callback,wait,hold,route){this.jid=jid;this.authzid=Strophe.getBareJidFromJid(this.jid);this.authcid=Strophe.getNodeFromJid(this.jid);this.pass=pass;this.servtype="xmpp";this.connect_callback=callback;this.disconnecting=false;this.connected=false;this.authenticated=false;this.errors=0;this.domain=this.domain||Strophe.getDomainFromJid(this.jid);this._changeConnectStatus(Strophe.Status.CONNECTING,null);this._proto._connect(wait,hold,route)},attach:function(jid,sid,rid,callback,wait,hold,wind){this._proto.attach(jid,sid,rid,callback,wait,hold,wind)},xmlInput:function(elem){return},xmlOutput:function(elem){return},rawInput:function(data){return},rawOutput:function(data){return},send:function(elem){if(elem===null){return}if(typeof(elem.sort)==="function"){for(var i=0;i<elem.length;i++){this._queueData(elem[i])}}else{if(typeof(elem.tree)==="function"){this._queueData(elem.tree())}else{this._queueData(elem)}}this._proto._send()},flush:function(){clearTimeout(this._idleTimeout);this._onIdle()},sendIQ:function(elem,callback,errback,timeout){var timeoutHandler=null;var that=this;if(typeof(elem.tree)==="function"){elem=elem.tree()}var id=elem.getAttribute("id");if(!id){id=this.getUniqueId("sendIQ");elem.setAttribute("id",id)}var handler=this.addHandler(function(stanza){if(timeoutHandler){that.deleteTimedHandler(timeoutHandler)}var iqtype=stanza.getAttribute("type");if(iqtype=="result"){if(callback){callback(stanza)}}else{if(iqtype=="error"){if(errback){errback(stanza)}}else{throw {name:"StropheError",message:"Got bad IQ type of "+iqtype}}}},null,"iq",null,id);if(timeout){timeoutHandler=this.addTimedHandler(timeout,function(){that.deleteHandler(handler);if(errback){errback(null)}return false})}this.send(elem);return id},_queueData:function(element){if(element===null||!element.tagName||!element.childNodes){throw {name:"StropheError",message:"Cannot queue non-DOMElement."}}this._data.push(element)},_sendRestart:function(){this._data.push("restart");this._proto._sendRestart();this._idleTimeout=setTimeout(this._onIdle.bind(this),100)},addTimedHandler:function(period,handler){var thand=new Strophe.TimedHandler(period,handler);this.addTimeds.push(thand);return thand},deleteTimedHandler:function(handRef){this.removeTimeds.push(handRef)},addHandler:function(handler,ns,name,type,id,from,options){var hand=new Strophe.Handler(handler,ns,name,type,id,from,options);this.addHandlers.push(hand);return hand},deleteHandler:function(handRef){this.removeHandlers.push(handRef)},disconnect:function(reason){this._changeConnectStatus(Strophe.Status.DISCONNECTING,reason);Strophe.info("Disconnect was called because: "+reason);if(this.connected){var pres=false;this.disconnecting=true;if(this.authenticated){pres=$pres({xmlns:Strophe.NS.CLIENT,type:"unavailable"})}this._disconnectTimeout=this._addSysTimedHandler(3000,this._onDisconnectTimeout.bind(this));this._proto._disconnect(pres)}},_changeConnectStatus:function(status,condition){for(var k in Strophe._connectionPlugins){if(Strophe._connectionPlugins.hasOwnProperty(k)){var plugin=this[k];if(plugin.statusChanged){try{plugin.statusChanged(status,condition)}catch(err){Strophe.error(""+k+" plugin caused an exception changing status: "+err)}}}}if(this.connect_callback){try{this.connect_callback(status,condition)}catch(e){Strophe.error("User connection callback caused an exception: "+e)}}},_doDisconnect:function(){if(this._disconnectTimeout!==null){this.deleteTimedHandler(this._disconnectTimeout);this._disconnectTimeout=null}Strophe.info("_doDisconnect was called");this._proto._doDisconnect();this.authenticated=false;this.disconnecting=false;this.handlers=[];this.timedHandlers=[];this.removeTimeds=[];this.removeHandlers=[];this.addTimeds=[];this.addHandlers=[];this._changeConnectStatus(Strophe.Status.DISCONNECTED,null);this.connected=false},_dataRecv:function(req,raw){Strophe.info("_dataRecv called");var elem=this._proto._reqToData(req);if(elem===null){return}if(this.xmlInput!==Strophe.Connection.prototype.xmlInput){if(elem.nodeName===this._proto.strip&&elem.childNodes.length){this.xmlInput(elem.childNodes[0])}else{this.xmlInput(elem)}}if(this.rawInput!==Strophe.Connection.prototype.rawInput){if(raw){this.rawInput(raw)}else{this.rawInput(Strophe.serialize(elem))}}var i,hand;while(this.removeHandlers.length>0){hand=this.removeHandlers.pop();i=this.handlers.indexOf(hand);if(i>=0){this.handlers.splice(i,1)}}while(this.addHandlers.length>0){this.handlers.push(this.addHandlers.pop())}if(this.disconnecting&&this._proto._emptyQueue()){this._doDisconnect();return}var typ=elem.getAttribute("type");var cond,conflict;if(typ!==null&&typ=="terminate"){if(this.disconnecting){return}cond=elem.getAttribute("condition");conflict=elem.getElementsByTagName("conflict");if(cond!==null){if(cond=="remote-stream-error"&&conflict.length>0){cond="conflict"}this._changeConnectStatus(Strophe.Status.CONNFAIL,cond)}else{this._changeConnectStatus(Strophe.Status.CONNFAIL,"unknown")}this.disconnect("unknown stream-error");return}var that=this;Strophe.forEachChild(elem,null,function(child){var i,newList;newList=that.handlers;that.handlers=[];for(i=0;i<newList.length;i++){var hand=newList[i];try{if(hand.isMatch(child)&&(that.authenticated||!hand.user)){if(hand.run(child)){that.handlers.push(hand)}}else{that.handlers.push(hand)}}catch(e){Strophe.warn("Removing Strophe handlers due to uncaught exception: "+e.message)}}})},mechanisms:{},_connect_cb:function(req,_callback,raw){Strophe.info("_connect_cb was called");this.connected=true;var bodyWrap=this._proto._reqToData(req);if(!bodyWrap){return}if(this.xmlInput!==Strophe.Connection.prototype.xmlInput){if(bodyWrap.nodeName===this._proto.strip&&bodyWrap.childNodes.length){this.xmlInput(bodyWrap.childNodes[0])}else{this.xmlInput(bodyWrap)}}if(this.rawInput!==Strophe.Connection.prototype.rawInput){if(raw){this.rawInput(raw)}else{this.rawInput(Strophe.serialize(bodyWrap))}}var conncheck=this._proto._connect_cb(bodyWrap);if(conncheck===Strophe.Status.CONNFAIL){return}this._authentication.sasl_scram_sha1=false;this._authentication.sasl_plain=false;this._authentication.sasl_digest_md5=false;this._authentication.sasl_anonymous=false;this._authentication.legacy_auth=false;var hasFeatures=bodyWrap.getElementsByTagName("stream:features").length>0;if(!hasFeatures){hasFeatures=bodyWrap.getElementsByTagName("features").length>0}var mechanisms=bodyWrap.getElementsByTagName("mechanism");var matched=[];var i,mech,auth_str,hashed_auth_str,found_authentication=false;if(!hasFeatures){this._proto._no_auth_received(_callback);return}if(mechanisms.length>0){for(i=0;i<mechanisms.length;i++){mech=Strophe.getText(mechanisms[i]);if(this.mechanisms[mech]){matched.push(this.mechanisms[mech])}}}this._authentication.legacy_auth=bodyWrap.getElementsByTagName("auth").length>0;found_authentication=this._authentication.legacy_auth||matched.length>0;if(!found_authentication){this._proto._no_auth_received(_callback);return}if(this.do_authentication!==false){this.authenticate(matched)}},authenticate:function(matched){var i;for(i=0;i<matched.length-1;++i){var higher=i;for(var j=i+1;j<matched.length;++j){if(matched[j].prototype.priority>matched[higher].prototype.priority){higher=j}}if(higher!=i){var swap=matched[i];matched[i]=matched[higher];matched[higher]=swap}}var mechanism_found=false;for(i=0;i<matched.length;++i){if(!matched[i].test(this)){continue}this._sasl_success_handler=this._addSysHandler(this._sasl_success_cb.bind(this),null,"success",null,null);this._sasl_failure_handler=this._addSysHandler(this._sasl_failure_cb.bind(this),null,"failure",null,null);this._sasl_challenge_handler=this._addSysHandler(this._sasl_challenge_cb.bind(this),null,"challenge",null,null);this._sasl_mechanism=new matched[i]();this._sasl_mechanism.onStart(this);var request_auth_exchange=$build("auth",{xmlns:Strophe.NS.SASL,mechanism:this._sasl_mechanism.name});if(this._sasl_mechanism.isClientFirst){var response=this._sasl_mechanism.onChallenge(this,null);request_auth_exchange.t(Base64.encode(response))}this.send(request_auth_exchange.tree());mechanism_found=true;break}if(!mechanism_found){if(Strophe.getNodeFromJid(this.jid)===null){this._changeConnectStatus(Strophe.Status.CONNFAIL,"x-strophe-bad-non-anon-jid");this.disconnect("x-strophe-bad-non-anon-jid")}else{this._changeConnectStatus(Strophe.Status.AUTHENTICATING,null);this._addSysHandler(this._auth1_cb.bind(this),null,null,null,"_auth_1");this.send($iq({type:"get",to:this.domain,id:"_auth_1"}).c("query",{xmlns:Strophe.NS.AUTH}).c("username",{}).t(Strophe.getNodeFromJid(this.jid)).tree())}}},_sasl_challenge_cb:function(elem){var challenge=Base64.decode(Strophe.getText(elem));var response=this._sasl_mechanism.onChallenge(this,challenge);var stanza=$build("response",{xmlns:Strophe.NS.SASL});if(response!==""){stanza.t(Base64.encode(response))}this.send(stanza.tree());return true},_auth1_cb:function(elem){var iq=$iq({type:"set",id:"_auth_2"}).c("query",{xmlns:Strophe.NS.AUTH}).c("username",{}).t(Strophe.getNodeFromJid(this.jid)).up().c("password").t(this.pass);if(!Strophe.getResourceFromJid(this.jid)){this.jid=Strophe.getBareJidFromJid(this.jid)+"/strophe"}iq.up().c("resource",{}).t(Strophe.getResourceFromJid(this.jid));this._addSysHandler(this._auth2_cb.bind(this),null,null,null,"_auth_2");this.send(iq.tree());return false},_sasl_success_cb:function(elem){if(this._sasl_data["server-signature"]){var serverSignature;var success=Base64.decode(Strophe.getText(elem));var attribMatch=/([a-z]+)=([^,]+)(,|$)/;matches=success.match(attribMatch);if(matches[1]=="v"){serverSignature=matches[2]}if(serverSignature!=this._sasl_data["server-signature"]){this.deleteHandler(this._sasl_failure_handler);this._sasl_failure_handler=null;if(this._sasl_challenge_handler){this.deleteHandler(this._sasl_challenge_handler);this._sasl_challenge_handler=null}this._sasl_data={};return this._sasl_failure_cb(null)}}Strophe.info("SASL authentication succeeded.");if(this._sasl_mechanism){this._sasl_mechanism.onSuccess()}this.deleteHandler(this._sasl_failure_handler);this._sasl_failure_handler=null;if(this._sasl_challenge_handler){this.deleteHandler(this._sasl_challenge_handler);this._sasl_challenge_handler=null}this._addSysHandler(this._sasl_auth1_cb.bind(this),null,"stream:features",null,null);this._sendRestart();return false},_sasl_auth1_cb:function(elem){this.features=elem;var i,child;for(i=0;i<elem.childNodes.length;i++){child=elem.childNodes[i];if(child.nodeName=="bind"){this.do_bind=true}if(child.nodeName=="session"){this.do_session=true}}if(!this.do_bind){this._changeConnectStatus(Strophe.Status.AUTHFAIL,null);return false}else{this._addSysHandler(this._sasl_bind_cb.bind(this),null,null,null,"_bind_auth_2");var resource=Strophe.getResourceFromJid(this.jid);if(resource){this.send($iq({type:"set",id:"_bind_auth_2"}).c("bind",{xmlns:Strophe.NS.BIND}).c("resource",{}).t(resource).tree())}else{this.send($iq({type:"set",id:"_bind_auth_2"}).c("bind",{xmlns:Strophe.NS.BIND}).tree())}}return false},_sasl_bind_cb:function(elem){if(elem.getAttribute("type")=="error"){Strophe.info("SASL binding failed.");var conflict=elem.getElementsByTagName("conflict"),condition;if(conflict.length>0){condition="conflict"}this._changeConnectStatus(Strophe.Status.AUTHFAIL,condition);return false}var bind=elem.getElementsByTagName("bind");var jidNode;if(bind.length>0){jidNode=bind[0].getElementsByTagName("jid");if(jidNode.length>0){this.jid=Strophe.getText(jidNode[0]);if(this.do_session){this._addSysHandler(this._sasl_session_cb.bind(this),null,null,null,"_session_auth_2");this.send($iq({type:"set",id:"_session_auth_2"}).c("session",{xmlns:Strophe.NS.SESSION}).tree())}else{this.authenticated=true;this._changeConnectStatus(Strophe.Status.CONNECTED,null)}}}else{Strophe.info("SASL binding failed.");this._changeConnectStatus(Strophe.Status.AUTHFAIL,null);return false}},_sasl_session_cb:function(elem){if(elem.getAttribute("type")=="result"){this.authenticated=true;this._changeConnectStatus(Strophe.Status.CONNECTED,null)}else{if(elem.getAttribute("type")=="error"){Strophe.info("Session creation failed.");this._changeConnectStatus(Strophe.Status.AUTHFAIL,null);return false}}return false},_sasl_failure_cb:function(elem){if(this._sasl_success_handler){this.deleteHandler(this._sasl_success_handler);this._sasl_success_handler=null}if(this._sasl_challenge_handler){this.deleteHandler(this._sasl_challenge_handler);this._sasl_challenge_handler=null}if(this._sasl_mechanism){this._sasl_mechanism.onFailure()}this._changeConnectStatus(Strophe.Status.AUTHFAIL,null);return false},_auth2_cb:function(elem){if(elem.getAttribute("type")=="result"){this.authenticated=true;this._changeConnectStatus(Strophe.Status.CONNECTED,null)}else{if(elem.getAttribute("type")=="error"){this._changeConnectStatus(Strophe.Status.AUTHFAIL,null);this.disconnect("authentication failed")}}return false},_addSysTimedHandler:function(period,handler){var thand=new Strophe.TimedHandler(period,handler);thand.user=false;this.addTimeds.push(thand);return thand},_addSysHandler:function(handler,ns,name,type,id){var hand=new Strophe.Handler(handler,ns,name,type,id);hand.user=false;this.addHandlers.push(hand);return hand},_onDisconnectTimeout:function(){Strophe.info("_onDisconnectTimeout was called");this._proto._onDisconnectTimeout();this._doDisconnect();return false},_onIdle:function(){var i,thand,since,newList;while(this.addTimeds.length>0){this.timedHandlers.push(this.addTimeds.pop())}while(this.removeTimeds.length>0){thand=this.removeTimeds.pop();i=this.timedHandlers.indexOf(thand);if(i>=0){this.timedHandlers.splice(i,1)}}var now=new Date().getTime();newList=[];for(i=0;i<this.timedHandlers.length;i++){thand=this.timedHandlers[i];if(this.authenticated||!thand.user){since=thand.lastCalled+thand.period;if(since-now<=0){if(thand.run()){newList.push(thand)}}else{newList.push(thand)}}}this.timedHandlers=newList;var body,time_elapsed;clearTimeout(this._idleTimeout);this._proto._onIdle();if(this.connected){this._idleTimeout=setTimeout(this._onIdle.bind(this),100)}}};if(callback){callback(Strophe,$build,$msg,$iq,$pres)}Strophe.SASLMechanism=function(name,isClientFirst,priority){this.name=name;this.isClientFirst=isClientFirst;this.priority=priority};Strophe.SASLMechanism.prototype={_sasl_data:{},test:function(connection){return true},onStart:function(connection){this._connection=connection},onChallenge:function(connection,challenge){throw new Error("You should implement challenge handling!")},onFailure:function(){this._connection=null},onSuccess:function(){this._connection=null}};Strophe.SASLAnonymous=function(){};Strophe.SASLAnonymous.prototype=new Strophe.SASLMechanism("ANONYMOUS",false,10);Strophe.SASLAnonymous.test=function(connection){return connection.authcid===null};Strophe.Connection.prototype.mechanisms[Strophe.SASLAnonymous.prototype.name]=Strophe.SASLAnonymous;Strophe.SASLPlain=function(){};Strophe.SASLPlain.prototype=new Strophe.SASLMechanism("PLAIN",true,20);Strophe.SASLPlain.test=function(connection){return connection.authcid!==null};Strophe.SASLPlain.prototype.onChallenge=function(connection,challenge){var auth_str=connection.authzid;auth_str=auth_str+"\u0000";auth_str=auth_str+connection.authcid;auth_str=auth_str+"\u0000";auth_str=auth_str+connection.pass;return auth_str};Strophe.Connection.prototype.mechanisms[Strophe.SASLPlain.prototype.name]=Strophe.SASLPlain;Strophe.SASLSHA1=function(){};Strophe.SASLSHA1.prototype=new Strophe.SASLMechanism("SCRAM-SHA-1",true,40);Strophe.SASLSHA1.test=function(connection){return connection.authcid!==null};Strophe.SASLSHA1.prototype.onChallenge=function(connection,challenge,test_cnonce){var cnonce=test_cnonce||MD5.hexdigest(Math.random()*1234567890);var auth_str="n="+connection.authcid;auth_str+=",r=";auth_str+=cnonce;this._sasl_data.cnonce=cnonce;this._sasl_data["client-first-message-bare"]=auth_str;auth_str="n,,"+auth_str;this.onChallenge=function(connection,challenge){var nonce,salt,iter,Hi,U,U_old;var clientKey,serverKey,clientSignature;var responseText="c=biws,";var authMessage=this._sasl_data["client-first-message-bare"]+","+challenge+",";var cnonce=this._sasl_data.cnonce;var attribMatch=/([a-z]+)=([^,]+)(,|$)/;while(challenge.match(attribMatch)){matches=challenge.match(attribMatch);challenge=challenge.replace(matches[0],"");switch(matches[1]){case"r":nonce=matches[2];break;case"s":salt=matches[2];break;case"i":iter=matches[2];break}}if(nonce.substr(0,cnonce.length)!==cnonce){this._sasl_data={};return connection._sasl_failure_cb()}responseText+="r="+nonce;authMessage+=responseText;salt=Base64.decode(salt);salt+="\x00\x00\x00\x01";Hi=U_old=core_hmac_sha1(connection.pass,salt);for(i=1;i<iter;i++){U=core_hmac_sha1(connection.pass,binb2str(U_old));for(k=0;k<5;k++){Hi[k]^=U[k]}U_old=U}Hi=binb2str(Hi);clientKey=core_hmac_sha1(Hi,"Client Key");serverKey=str_hmac_sha1(Hi,"Server Key");clientSignature=core_hmac_sha1(str_sha1(binb2str(clientKey)),authMessage);connection._sasl_data["server-signature"]=b64_hmac_sha1(serverKey,authMessage);for(k=0;k<5;k++){clientKey[k]^=clientSignature[k]}responseText+=",p="+Base64.encode(binb2str(clientKey));return responseText}.bind(this);return auth_str};Strophe.Connection.prototype.mechanisms[Strophe.SASLSHA1.prototype.name]=Strophe.SASLSHA1;Strophe.SASLMD5=function(){};Strophe.SASLMD5.prototype=new Strophe.SASLMechanism("DIGEST-MD5",false,30);Strophe.SASLMD5.test=function(connection){return connection.authcid!==null};Strophe.SASLMD5.prototype._quote=function(str){return'"'+str.replace(/\\/g,"\\\\").replace(/"/g,'\\"')+'"'};Strophe.SASLMD5.prototype.onChallenge=function(connection,challenge,test_cnonce){var attribMatch=/([a-z]+)=("[^"]+"|[^,"]+)(?:,|$)/;var cnonce=test_cnonce||MD5.hexdigest(""+(Math.random()*1234567890));var realm="";var host=null;var nonce="";var qop="";var matches;while(challenge.match(attribMatch)){matches=challenge.match(attribMatch);challenge=challenge.replace(matches[0],"");matches[2]=matches[2].replace(/^"(.+)"$/,"$1");switch(matches[1]){case"realm":realm=matches[2];break;case"nonce":nonce=matches[2];break;case"qop":qop=matches[2];break;case"host":host=matches[2];break}}var digest_uri=connection.servtype+"/"+connection.domain;if(host!==null){digest_uri=digest_uri+"/"+host}var A1=MD5.hash(connection.authcid+":"+realm+":"+this._connection.pass)+":"+nonce+":"+cnonce;var A2="AUTHENTICATE:"+digest_uri;var responseText="";responseText+="charset=utf-8,";responseText+="username="+this._quote(connection.authcid)+",";responseText+="realm="+this._quote(realm)+",";responseText+="nonce="+this._quote(nonce)+",";responseText+="nc=00000001,";responseText+="cnonce="+this._quote(cnonce)+",";responseText+="digest-uri="+this._quote(digest_uri)+",";responseText+="response="+MD5.hexdigest(MD5.hexdigest(A1)+":"+nonce+":00000001:"+cnonce+":auth:"+MD5.hexdigest(A2))+",";responseText+="qop=auth";this.onChallenge=function(connection,challenge){return""}.bind(this);return responseText};Strophe.Connection.prototype.mechanisms[Strophe.SASLMD5.prototype.name]=Strophe.SASLMD5})(function(){window.Strophe=arguments[0];window.$build=arguments[1];window.$msg=arguments[2];window.$iq=arguments[3];window.$pres=arguments[4]});Strophe.Request=function(elem,func,rid,sends){this.id=++Strophe._requestId;this.xmlData=elem;this.data=Strophe.serialize(elem);this.origFunc=func;this.func=func;this.rid=rid;this.date=NaN;this.sends=sends||0;this.abort=false;this.dead=null;this.age=function(){if(!this.date){return 0}var now=new Date();return(now-this.date)/1000};this.timeDead=function(){if(!this.dead){return 0}var now=new Date();return(now-this.dead)/1000};this.xhr=this._newXHR()};Strophe.Request.prototype={getResponse:function(){var node=null;if(this.xhr.responseXML&&this.xhr.responseXML.documentElement){node=this.xhr.responseXML.documentElement;if(node.tagName=="parsererror"){Strophe.error("invalid response received");Strophe.error("responseText: "+this.xhr.responseText);Strophe.error("responseXML: "+Strophe.serialize(this.xhr.responseXML));throw"parsererror"}}else{if(this.xhr.responseText){Strophe.error("invalid response received");Strophe.error("responseText: "+this.xhr.responseText);Strophe.error("responseXML: "+Strophe.serialize(this.xhr.responseXML))}}return node},_newXHR:function(){var xhr=null;if(window.XMLHttpRequest){xhr=new XMLHttpRequest();if(xhr.overrideMimeType){xhr.overrideMimeType("text/xml")}}else{if(window.ActiveXObject){xhr=new ActiveXObject("Microsoft.XMLHTTP")}}xhr.onreadystatechange=this.func.bind(null,this);return xhr}};Strophe.Bosh=function(connection){this._conn=connection;this.rid=Math.floor(Math.random()*4294967295);this.sid=null;this.hold=1;this.wait=60;this.window=5;this._requests=[]};Strophe.Bosh.prototype={strip:null,_buildBody:function(){var bodyWrap=$build("body",{rid:this.rid++,xmlns:Strophe.NS.HTTPBIND});if(this.sid!==null){bodyWrap.attrs({sid:this.sid})}return bodyWrap},_reset:function(){this.rid=Math.floor(Math.random()*4294967295);this.sid=null},_connect:function(wait,hold,route){this.wait=wait||this.wait;this.hold=hold||this.hold;var body=this._buildBody().attrs({to:this._conn.domain,"xml:lang":"en",wait:this.wait,hold:this.hold,content:"text/xml; charset=utf-8",ver:"1.6","xmpp:version":"1.0","xmlns:xmpp":Strophe.NS.BOSH});if(route){body.attrs({route:route})}var _connect_cb=this._conn._connect_cb;this._requests.push(new Strophe.Request(body.tree(),this._onRequestStateChange.bind(this,_connect_cb.bind(this._conn)),body.tree().getAttribute("rid")));this._throttledRequestHandler()},_attach:function(jid,sid,rid,callback,wait,hold,wind){this._conn.jid=jid;this.sid=sid;this.rid=rid;this._conn.connect_callback=callback;this._conn.domain=Strophe.getDomainFromJid(this._conn.jid);this._conn.authenticated=true;this._conn.connected=true;this.wait=wait||this.wait;this.hold=hold||this.hold;this.window=wind||this.window;this._conn._changeConnectStatus(Strophe.Status.ATTACHED,null)},_connect_cb:function(bodyWrap){var typ=bodyWrap.getAttribute("type");var cond,conflict;if(typ!==null&&typ=="terminate"){Strophe.error("BOSH-Connection failed: "+cond);cond=bodyWrap.getAttribute("condition");conflict=bodyWrap.getElementsByTagName("conflict");if(cond!==null){if(cond=="remote-stream-error"&&conflict.length>0){cond="conflict"}this._conn._changeConnectStatus(Strophe.Status.CONNFAIL,cond)}else{this._conn._changeConnectStatus(Strophe.Status.CONNFAIL,"unknown")}this._conn._doDisconnect();return Strophe.Status.CONNFAIL}if(!this.sid){this.sid=bodyWrap.getAttribute("sid")}var wind=bodyWrap.getAttribute("requests");if(wind){this.window=parseInt(wind,10)}var hold=bodyWrap.getAttribute("hold");if(hold){this.hold=parseInt(hold,10)}var wait=bodyWrap.getAttribute("wait");if(wait){this.wait=parseInt(wait,10)}},_disconnect:function(pres){this._sendTerminate(pres)},_doDisconnect:function(){this.sid=null;this.rid=Math.floor(Math.random()*4294967295)},_emptyQueue:function(){return this._requests.length===0},_hitError:function(reqStatus){this.errors++;Strophe.warn("request errored, status: "+reqStatus+", number of errors: "+this.errors);if(this.errors>4){this._onDisconnectTimeout()}},_no_auth_received:function(_callback){if(_callback){_callback=_callback.bind(this._conn)}else{_callback=this._conn._connect_cb.bind(this._conn)}var body=this._buildBody();this._requests.push(new Strophe.Request(body.tree(),this._onRequestStateChange.bind(this,_callback.bind(this._conn)),body.tree().getAttribute("rid")));this._throttledRequestHandler()},_onDisconnectTimeout:function(){var req;while(this._requests.length>0){req=this._requests.pop();req.abort=true;req.xhr.abort();req.xhr.onreadystatechange=function(){}}},_onIdle:function(){var data=this._conn._data;if(this._conn.authenticated&&this._requests.length===0&&data.length===0&&!this._conn.disconnecting){Strophe.info("no requests during idle cycle, sending blank request");data.push(null)}if(this._requests.length<2&&data.length>0&&!this._conn.paused){body=this._buildBody();for(i=0;i<data.length;i++){if(data[i]!==null){if(data[i]==="restart"){body.attrs({to:this._conn.domain,"xml:lang":"en","xmpp:restart":"true","xmlns:xmpp":Strophe.NS.BOSH})}else{body.cnode(data[i]).up()}}}delete this._conn._data;this._conn._data=[];this._requests.push(new Strophe.Request(body.tree(),this._onRequestStateChange.bind(this,this._conn._dataRecv.bind(this._conn)),body.tree().getAttribute("rid")));this._processRequest(this._requests.length-1)}if(this._requests.length>0){time_elapsed=this._requests[0].age();if(this._requests[0].dead!==null){if(this._requests[0].timeDead()>Math.floor(Strophe.SECONDARY_TIMEOUT*this.wait)){this._throttledRequestHandler()}}if(time_elapsed>Math.floor(Strophe.TIMEOUT*this.wait)){Strophe.warn("Request "+this._requests[0].id+" timed out, over "+Math.floor(Strophe.TIMEOUT*this.wait)+" seconds since last activity");this._throttledRequestHandler()}}},_onRequestStateChange:function(func,req){Strophe.debug("request id "+req.id+"."+req.sends+" state changed to "+req.xhr.readyState);if(req.abort){req.abort=false;return}var reqStatus;if(req.xhr.readyState==4){reqStatus=0;try{reqStatus=req.xhr.status}catch(e){}if(typeof(reqStatus)=="undefined"){reqStatus=0}if(this.disconnecting){if(reqStatus>=400){this._hitError(reqStatus);return}}var reqIs0=(this._requests[0]==req);var reqIs1=(this._requests[1]==req);if((reqStatus>0&&reqStatus<500)||req.sends>5){this._removeRequest(req);Strophe.debug("request id "+req.id+" should now be removed")}if(reqStatus==200){if(reqIs1||(reqIs0&&this._requests.length>0&&this._requests[0].age()>Math.floor(Strophe.SECONDARY_TIMEOUT*this.wait))){this._restartRequest(0)}Strophe.debug("request id "+req.id+"."+req.sends+" got 200");func(req);this.errors=0}else{Strophe.error("request id "+req.id+"."+req.sends+" error "+reqStatus+" happened");if(reqStatus===0||(reqStatus>=400&&reqStatus<600)||reqStatus>=12000){this._hitError(reqStatus);if(reqStatus>=400&&reqStatus<500){this._conn._changeConnectStatus(Strophe.Status.DISCONNECTING,null);this._conn._doDisconnect()}}}if(!((reqStatus>0&&reqStatus<500)||req.sends>5)){this._throttledRequestHandler()}}},_processRequest:function(i){var self=this;var req=this._requests[i];var reqStatus=-1;try{if(req.xhr.readyState==4){reqStatus=req.xhr.status}}catch(e){Strophe.error("caught an error in _requests["+i+"], reqStatus: "+reqStatus)}if(typeof(reqStatus)=="undefined"){reqStatus=-1}if(req.sends>this.maxRetries){this._onDisconnectTimeout();return}var time_elapsed=req.age();var primaryTimeout=(!isNaN(time_elapsed)&&time_elapsed>Math.floor(Strophe.TIMEOUT*this.wait));var secondaryTimeout=(req.dead!==null&&req.timeDead()>Math.floor(Strophe.SECONDARY_TIMEOUT*this.wait));var requestCompletedWithServerError=(req.xhr.readyState==4&&(reqStatus<1||reqStatus>=500));if(primaryTimeout||secondaryTimeout||requestCompletedWithServerError){if(secondaryTimeout){Strophe.error("Request "+this._requests[i].id+" timed out (secondary), restarting")}req.abort=true;req.xhr.abort();req.xhr.onreadystatechange=function(){};this._requests[i]=new Strophe.Request(req.xmlData,req.origFunc,req.rid,req.sends);req=this._requests[i]}if(req.xhr.readyState===0){Strophe.debug("request id "+req.id+"."+req.sends+" posting");try{req.xhr.open("POST",this._conn.service,true)}catch(e2){Strophe.error("XHR open failed.");if(!this._conn.connected){this._conn._changeConnectStatus(Strophe.Status.CONNFAIL,"bad-service")}this._conn.disconnect();return}var sendFunc=function(){req.date=new Date();if(self._conn._options.customHeaders){var headers=self._conn._options.customHeaders;for(var header in headers){if(headers.hasOwnProperty(header)){req.xhr.setRequestHeader(header,headers[header])}}}req.xhr.send(req.data)};if(req.sends>1){var backoff=Math.min(Math.floor(Strophe.TIMEOUT*this.wait),Math.pow(req.sends,3))*1000;setTimeout(sendFunc,backoff)}else{sendFunc()}req.sends++;if(this._conn.xmlOutput!==Strophe.Connection.prototype.xmlOutput){if(req.xmlData.nodeName===this.strip&&req.xmlData.childNodes.length){this._conn.xmlOutput(req.xmlData.childNodes[0])}else{this._conn.xmlOutput(req.xmlData)}}if(this._conn.rawOutput!==Strophe.Connection.prototype.rawOutput){this._conn.rawOutput(req.data)}}else{Strophe.debug("_processRequest: "+(i===0?"first":"second")+" request has readyState of "+req.xhr.readyState)}},_removeRequest:function(req){Strophe.debug("removing request");var i;for(i=this._requests.length-1;i>=0;i--){if(req==this._requests[i]){this._requests.splice(i,1)}}req.xhr.onreadystatechange=function(){};this._throttledRequestHandler()},_restartRequest:function(i){var req=this._requests[i];if(req.dead===null){req.dead=new Date()}this._processRequest(i)},_reqToData:function(req){try{return req.getResponse()}catch(e){if(e!="parsererror"){throw e}this._conn.disconnect("strophe-parsererror")}},_sendTerminate:function(pres){Strophe.info("_sendTerminate was called");var body=this._buildBody().attrs({type:"terminate"});if(pres){body.cnode(pres.tree())}var req=new Strophe.Request(body.tree(),this._onRequestStateChange.bind(this,this._conn._dataRecv.bind(this._conn)),body.tree().getAttribute("rid"));this._requests.push(req);this._throttledRequestHandler()},_send:function(){clearTimeout(this._conn._idleTimeout);this._throttledRequestHandler();this._conn._idleTimeout=setTimeout(this._conn._onIdle.bind(this._conn),100)},_sendRestart:function(){this._throttledRequestHandler();clearTimeout(this._conn._idleTimeout)},_throttledRequestHandler:function(){if(!this._requests){Strophe.debug("_throttledRequestHandler called with undefined requests")}else{Strophe.debug("_throttledRequestHandler called with "+this._requests.length+" requests")}if(!this._requests||this._requests.length===0){return}if(this._requests.length>0){this._processRequest(0)}if(this._requests.length>1&&Math.abs(this._requests[0].rid-this._requests[1].rid)<this.window){this._processRequest(1)}}};Strophe.Websocket=function(connection){this._conn=connection;this.strip="stream:stream";var service=connection.service;if(service.indexOf("ws:")!==0&&service.indexOf("wss:")!==0){var new_service="";if(connection._options.protocol==="ws"&&window.location.protocol!=="https:"){new_service+="ws"}else{new_service+="wss"}new_service+="://"+window.location.host;if(service.indexOf("/")!==0){new_service+=window.location.pathname+service}else{new_service+=service}connection.service=new_service}};Strophe.Websocket.prototype={_buildStream:function(){return $build("stream:stream",{to:this._conn.domain,xmlns:Strophe.NS.CLIENT,"xmlns:stream":Strophe.NS.STREAM,version:"1.0"})},_check_streamerror:function(bodyWrap,connectstatus){var errors=bodyWrap.getElementsByTagName("stream:error");if(errors.length===0){return false}var error=errors[0];var condition="";var text="";ns="urn:ietf:params:xml:ns:xmpp-streams";for(i=0;i<error.childNodes.length;i++){e=error.childNodes[i];if(e.getAttribute("xmlns")!==ns){break}if(e.nodeName==="text"){text=e.textContent}else{condition=e.nodeName}}errorString="WebSocket stream error: ";if(condition){errorString+=condition}else{errorString+="unknown"}if(text){errorString+=" - "+condition}Strophe.error(errorString);this._conn._changeConnectStatus(connectstatus,condition);this._conn._doDisconnect();return true},_reset:function(){return},_connect:function(){this._closeSocket();this.socket=new WebSocket(this._conn.service,"xmpp");this.socket.onopen=this._onOpen.bind(this);this.socket.onerror=this._onError.bind(this);this.socket.onclose=this._onClose.bind(this);this.socket.onmessage=this._connect_cb_wrapper.bind(this)},_connect_cb:function(bodyWrap){var error=this._check_streamerror(bodyWrap,Strophe.Status.CONNFAIL);if(error){return Strophe.Status.CONNFAIL}},_handleStreamStart:function(message){var error=false;var ns=message.getAttribute("xmlns");if(typeof ns!=="string"){error="Missing xmlns in stream:stream"}else{if(ns!==Strophe.NS.CLIENT){error="Wrong xmlns in stream:stream: "+ns}}var ns_stream=message.namespaceURI;if(typeof ns_stream!=="string"){error="Missing xmlns:stream in stream:stream"}else{if(ns_stream!==Strophe.NS.STREAM){error="Wrong xmlns:stream in stream:stream: "+ns_stream}}var ver=message.getAttribute("version");if(typeof ver!=="string"){error="Missing version in stream:stream"}else{if(ver!=="1.0"){error="Wrong version in stream:stream: "+ver}}if(error){this._conn._changeConnectStatus(Strophe.Status.CONNFAIL,error);this._conn._doDisconnect();return false}return true},_connect_cb_wrapper:function(message){if(message.data.indexOf("<stream:stream ")===0||message.data.indexOf("<?xml")===0){var data=message.data.replace(/^(<\?.*?\?>\s*)*/,"");if(data===""){return}data=message.data.replace(/<stream:stream (.*[^\/])>/,"<stream:stream $1/>");var streamStart=new DOMParser().parseFromString(data,"text/xml").documentElement;this._conn.xmlInput(streamStart);this._conn.rawInput(message.data);if(this._handleStreamStart(streamStart)){this._connect_cb(streamStart);this.streamStart=message.data.replace(/^<stream:(.*)\/>$/,"<stream:$1>")}}else{if(message.data==="</stream:stream>"){this._conn.rawInput(message.data);this._conn.xmlInput(document.createElement("stream:stream"));this._conn._changeConnectStatus(Strophe.Status.CONNFAIL,error);this._conn._doDisconnect();return}else{var string=this._streamWrap(message.data);var elem=new DOMParser().parseFromString(string,"text/xml").documentElement;this.socket.onmessage=this._onMessage.bind(this);this._conn._connect_cb(elem,null,message.data)}}},_disconnect:function(pres){if(this.socket.readyState!==WebSocket.CLOSED){if(pres){this._conn.send(pres)}var close="</stream:stream>";this._conn.xmlOutput(document.createElement("stream:stream"));this._conn.rawOutput(close);try{this.socket.send(close)}catch(e){Strophe.info("Couldn't send closing stream tag.")}}this._conn._doDisconnect()},_doDisconnect:function(){Strophe.info("WebSockets _doDisconnect was called");this._closeSocket()},_streamWrap:function(stanza){return this.streamStart+stanza+"</stream:stream>"},_closeSocket:function(){if(this.socket){try{this.socket.close()}catch(e){}}this.socket=null},_emptyQueue:function(){return true},_onClose:function(event){if(this._conn.connected&&!this._conn.disconnecting){Strophe.error("Websocket closed unexcectedly");this._conn._doDisconnect()}else{Strophe.info("Websocket closed")}},_no_auth_received:function(_callback){Strophe.error("Server did not send any auth methods");this._conn._changeConnectStatus(Strophe.Status.CONNFAIL,"Server did not send any auth methods");if(_callback){_callback=_callback.bind(this._conn);_callback()}this._conn._doDisconnect()},_onDisconnectTimeout:function(){},_onError:function(error){Strophe.error("Websocket error "+error);this._conn._changeConnectStatus(Strophe.Status.CONNFAIL,"The WebSocket connection could not be established was disconnected.");this._disconnect()},_onIdle:function(){var data=this._conn._data;if(data.length>0&&!this._conn.paused){for(i=0;i<data.length;i++){if(data[i]!==null){var stanza,rawStanza;if(data[i]==="restart"){stanza=this._buildStream();rawStanza=this._removeClosingTag(stanza);stanza=stanza.tree()}else{stanza=data[i];rawStanza=Strophe.serialize(stanza)}this._conn.xmlOutput(stanza);this._conn.rawOutput(rawStanza);this.socket.send(rawStanza)}}this._conn._data=[]}},_onMessage:function(message){var elem;if(message.data==="</stream:stream>"){var close="</stream:stream>";this._conn.rawInput(close);this._conn.xmlInput(document.createElement("stream:stream"));if(!this._conn.disconnecting){this._conn._doDisconnect()}return}else{if(message.data.search("<stream:stream ")===0){data=message.data.replace(/<stream:stream (.*[^\/])>/,"<stream:stream $1/>");elem=new DOMParser().parseFromString(data,"text/xml").documentElement;if(!this._handleStreamStart(elem)){return}}else{data=this._streamWrap(message.data);elem=new DOMParser().parseFromString(data,"text/xml").documentElement}}if(this._check_streamerror(elem,Strophe.Status.ERROR)){return}if(this._conn.disconnecting&&elem.firstChild.nodeName==="presence"&&elem.firstChild.getAttribute("type")==="unavailable"){this._conn.xmlInput(elem);this._conn.rawInput(Strophe.serialize(elem));return}this._conn._dataRecv(elem,message.data)},_onOpen:function(){Strophe.info("Websocket open");var start=this._buildStream();this._conn.xmlOutput(start.tree());var startString=this._removeClosingTag(start);this._conn.rawOutput(startString);this.socket.send(startString)},_removeClosingTag:function(elem){var string=Strophe.serialize(elem);string=string.replace(/<(stream:stream .*[^\/])\/>$/,"<$1>");return string},_reqToData:function(stanza){return stanza},_send:function(){this._conn.flush()},_sendRestart:function(){clearTimeout(this._conn._idleTimeout);this._conn._onIdle.bind(this._conn)()}};Strophe.addConnectionPlugin("disco",{_connection:null,_identities:[],_features:[],_items:[],init:function(a){this._connection=a;this._identities=[];this._features=[];this._items=[];a.addHandler(this._onDiscoInfo.bind(this),Strophe.NS.DISCO_INFO,"iq","get",null,null);a.addHandler(this._onDiscoItems.bind(this),Strophe.NS.DISCO_ITEMS,"iq","get",null,null)},addIdentity:function(d,c,a,e){for(var b=0;b<this._identities.length;b++){if(this._identities[b].category==d&&this._identities[b].type==c&&this._identities[b].name==a&&this._identities[b].lang==e){return false}}this._identities.push({category:d,type:c,name:a,lang:e});return true},addFeature:function(b){for(var a=0;a<this._features.length;a++){if(this._features[a]==b){return false}}this._features.push(b);return true},removeFeature:function(b){for(var a=0;a<this._features.length;a++){if(this._features[a]===b){this._features.splice(a,1);return true}}return false},addItem:function(b,a,c,d){if(c&&!d){return false}this._items.push({jid:b,name:a,node:c,call_back:d});return true},info:function(c,d,g,b,e){var a={xmlns:Strophe.NS.DISCO_INFO};if(d){a.node=d}var f=$iq({from:this._connection.jid,to:c,type:"get"}).c("query",a);this._connection.sendIQ(f,g,b,e)},items:function(d,e,g,c,f){var b={xmlns:Strophe.NS.DISCO_ITEMS};if(e){b.node=e}var a=$iq({from:this._connection.jid,to:d,type:"get"}).c("query",b);this._connection.sendIQ(a,g,c,f)},_buildIQResult:function(c,b){var e=c.getAttribute("id");var d=c.getAttribute("from");var a=$iq({type:"result",id:e});if(d!==null){a.attrs({to:d})}return a.c("query",b)},_onDiscoInfo:function(e){var d=e.getElementsByTagName("query")[0].getAttribute("node");var a={xmlns:Strophe.NS.DISCO_INFO};if(d){a.node=d}var c=this._buildIQResult(e,a);for(var b=0;b<this._identities.length;b++){var a={category:this._identities[b].category,type:this._identities[b].type};if(this._identities[b].name){a.name=this._identities[b].name}if(this._identities[b].lang){a["xml:lang"]=this._identities[b].lang}c.c("identity",a).up()}for(var b=0;b<this._features.length;b++){c.c("feature",{"var":this._features[b]}).up()}this._connection.send(c.tree());return true},_onDiscoItems:function(g){var f={xmlns:Strophe.NS.DISCO_ITEMS};var e=g.getElementsByTagName("query")[0].getAttribute("node");if(e){f.node=e;var a=[];for(var c=0;c<this._items.length;c++){if(this._items[c].node==e){a=this._items[c].call_back(g);break}}}else{var a=this._items}var d=this._buildIQResult(g,f);for(var c=0;c<a.length;c++){var b={jid:a[c].jid};if(a[c].name){b.name=a[c].name}if(a[c].node){b.node=a[c].node}d.c("item",b).up()}this._connection.send(d.tree());return true}});/* jshint -W117 */
  2. // mozilla chrome compat layer -- very similar to adapter.js
  3. function setupRTC() {
  4. var RTC = null;
  5. if (navigator.mozGetUserMedia) {
  6. console.log('This appears to be Firefox');
  7. var version = parseInt(navigator.userAgent.match(/Firefox\/([0-9]+)\./)[1], 10);
  8. if (version >= 22) {
  9. RTC = {
  10. peerconnection: mozRTCPeerConnection,
  11. browser: 'firefox',
  12. getUserMedia: navigator.mozGetUserMedia.bind(navigator),
  13. attachMediaStream: function (element, stream) {
  14. element[0].mozSrcObject = stream;
  15. element[0].play();
  16. },
  17. pc_constraints: {}
  18. };
  19. if (!MediaStream.prototype.getVideoTracks)
  20. MediaStream.prototype.getVideoTracks = function () { return []; };
  21. if (!MediaStream.prototype.getAudioTracks)
  22. MediaStream.prototype.getAudioTracks = function () { return []; };
  23. RTCSessionDescription = mozRTCSessionDescription;
  24. RTCIceCandidate = mozRTCIceCandidate;
  25. }
  26. } else if (navigator.webkitGetUserMedia) {
  27. console.log('This appears to be Chrome');
  28. RTC = {
  29. peerconnection: webkitRTCPeerConnection,
  30. browser: 'chrome',
  31. getUserMedia: navigator.webkitGetUserMedia.bind(navigator),
  32. attachMediaStream: function (element, stream) {
  33. element.attr('src', webkitURL.createObjectURL(stream));
  34. },
  35. // pc_constraints: {} // FIVE-182
  36. pc_constraints: {'optional': [{'DtlsSrtpKeyAgreement': 'true'}]} // enable dtls support in canary
  37. };
  38. if (navigator.userAgent.indexOf('Android') != -1) {
  39. RTC.pc_constraints = {}; // disable DTLS on Android
  40. }
  41. if (!webkitMediaStream.prototype.getVideoTracks) {
  42. webkitMediaStream.prototype.getVideoTracks = function () {
  43. return this.videoTracks;
  44. };
  45. }
  46. if (!webkitMediaStream.prototype.getAudioTracks) {
  47. webkitMediaStream.prototype.getAudioTracks = function () {
  48. return this.audioTracks;
  49. };
  50. }
  51. }
  52. if (RTC === null) {
  53. try { console.log('Browser does not appear to be WebRTC-capable'); } catch (e) { }
  54. }
  55. return RTC;
  56. }
  57. function getUserMediaWithConstraints(um, resolution, bandwidth, fps) {
  58. var constraints = {audio: false, video: false};
  59. if (um.indexOf('video') >= 0) {
  60. constraints.video = {mandatory: {}};// same behaviour as true
  61. }
  62. if (um.indexOf('audio') >= 0) {
  63. constraints.audio = {};// same behaviour as true
  64. }
  65. if (um.indexOf('screen') >= 0) {
  66. constraints.video = {
  67. "mandatory": {
  68. "chromeMediaSource": "screen"
  69. }
  70. };
  71. }
  72. if (resolution && !constraints.video) {
  73. constraints.video = {mandatory: {}};// same behaviour as true
  74. }
  75. // see https://code.google.com/p/chromium/issues/detail?id=143631#c9 for list of supported resolutions
  76. switch (resolution) {
  77. // 16:9 first
  78. case '1080':
  79. case 'fullhd':
  80. constraints.video.mandatory.minWidth = 1920;
  81. constraints.video.mandatory.minHeight = 1080;
  82. constraints.video.mandatory.minAspectRatio = 1.77;
  83. break;
  84. case '720':
  85. case 'hd':
  86. constraints.video.mandatory.minWidth = 1280;
  87. constraints.video.mandatory.minHeight = 720;
  88. constraints.video.mandatory.minAspectRatio = 1.77;
  89. break;
  90. case '360':
  91. constraints.video.mandatory.minWidth = 640;
  92. constraints.video.mandatory.minHeight = 360;
  93. constraints.video.mandatory.minAspectRatio = 1.77;
  94. break;
  95. case '180':
  96. constraints.video.mandatory.minWidth = 320;
  97. constraints.video.mandatory.minHeight = 180;
  98. constraints.video.mandatory.minAspectRatio = 1.77;
  99. break;
  100. // 4:3
  101. case '960':
  102. constraints.video.mandatory.minWidth = 960;
  103. constraints.video.mandatory.minHeight = 720;
  104. break;
  105. case '640':
  106. case 'vga':
  107. constraints.video.mandatory.minWidth = 640;
  108. constraints.video.mandatory.minHeight = 480;
  109. break;
  110. case '320':
  111. constraints.video.mandatory.minWidth = 320;
  112. constraints.video.mandatory.minHeight = 240;
  113. break;
  114. default:
  115. if (navigator.userAgent.indexOf('Android') != -1) {
  116. constraints.video.mandatory.minWidth = 320;
  117. constraints.video.mandatory.minHeight = 240;
  118. constraints.video.mandatory.maxFrameRate = 15;
  119. }
  120. break;
  121. }
  122. if (bandwidth) { // doesn't work currently, see webrtc issue 1846
  123. if (!constraints.video) constraints.video = {mandatory: {}};//same behaviour as true
  124. constraints.video.optional = [{bandwidth: bandwidth}];
  125. }
  126. if (fps) { // for some cameras it might be necessary to request 30fps
  127. // so they choose 30fps mjpg over 10fps yuy2
  128. if (!constraints.video) constraints.video = {mandatory: {}};// same behaviour as tru;
  129. constraints.video.mandatory.minFrameRate = fps;
  130. }
  131. try {
  132. RTC.getUserMedia(constraints,
  133. function (stream) {
  134. console.log('onUserMediaSuccess');
  135. $(document).trigger('mediaready.jingle', [stream]);
  136. },
  137. function (error) {
  138. console.warn('Failed to get access to local media. Error ', error);
  139. $(document).trigger('mediafailure.jingle');
  140. });
  141. } catch (e) {
  142. console.error('GUM failed: ', e);
  143. $(document).trigger('mediafailure.jingle');
  144. }
  145. }
  146. /* jshint -W117 */
  147. Strophe.addConnectionPlugin('jingle', {
  148. connection: null,
  149. sessions: {},
  150. jid2session: {},
  151. ice_config: {iceServers: []},
  152. pc_constraints: {},
  153. media_constraints: {
  154. mandatory: {
  155. 'OfferToReceiveAudio': true,
  156. 'OfferToReceiveVideo': true
  157. }
  158. // MozDontOfferDataChannel: true when this is firefox
  159. },
  160. localStream: null,
  161. init: function (conn) {
  162. this.connection = conn;
  163. if (this.connection.disco) {
  164. // http://xmpp.org/extensions/xep-0167.html#support
  165. // http://xmpp.org/extensions/xep-0176.html#support
  166. this.connection.disco.addFeature('urn:xmpp:jingle:1');
  167. this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:1');
  168. this.connection.disco.addFeature('urn:xmpp:jingle:transports:ice-udp:1');
  169. this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:audio');
  170. this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:video');
  171. // this is dealt with by SDP O/A so we don't need to annouce this
  172. //this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:rtcp-fb:0'); // XEP-0293
  173. //this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:rtp-hdrext:0'); // XEP-0294
  174. this.connection.disco.addFeature('urn:ietf:rfc:5761'); // rtcp-mux
  175. //this.connection.disco.addFeature('urn:ietf:rfc:5888'); // a=group, e.g. bundle
  176. //this.connection.disco.addFeature('urn:ietf:rfc:5576'); // a=ssrc
  177. }
  178. this.connection.addHandler(this.onJingle.bind(this), 'urn:xmpp:jingle:1', 'iq', 'set', null, null);
  179. },
  180. onJingle: function (iq) {
  181. var sid = $(iq).find('jingle').attr('sid');
  182. var action = $(iq).find('jingle').attr('action');
  183. // send ack first
  184. var ack = $iq({type: 'result',
  185. to: iq.getAttribute('from'),
  186. id: iq.getAttribute('id')
  187. });
  188. console.log('on jingle ' + action);
  189. var sess = this.sessions[sid];
  190. if ('session-initiate' != action) {
  191. if (sess === null) {
  192. ack.type = 'error';
  193. ack.c('error', {type: 'cancel'})
  194. .c('item-not-found', {xmlns: 'urn:ietf:params:xml:ns:xmpp-stanzas'}).up()
  195. .c('unknown-session', {xmlns: 'urn:xmpp:jingle:errors:1'});
  196. this.connection.send(ack);
  197. return true;
  198. }
  199. // compare from to sess.peerjid (bare jid comparison for later compat with message-mode)
  200. // local jid is not checked
  201. if (Strophe.getBareJidFromJid(iq.getAttribute('from')) != Strophe.getBareJidFromJid(sess.peerjid)) {
  202. console.warn('jid mismatch for session id', sid, iq.getAttribute('from'), sess.peerjid);
  203. ack.type = 'error';
  204. ack.c('error', {type: 'cancel'})
  205. .c('item-not-found', {xmlns: 'urn:ietf:params:xml:ns:xmpp-stanzas'}).up()
  206. .c('unknown-session', {xmlns: 'urn:xmpp:jingle:errors:1'});
  207. this.connection.send(ack);
  208. return true;
  209. }
  210. } else if (sess !== undefined) {
  211. // existing session with same session id
  212. // this might be out-of-order if the sess.peerjid is the same as from
  213. ack.type = 'error';
  214. ack.c('error', {type: 'cancel'})
  215. .c('service-unavailable', {xmlns: 'urn:ietf:params:xml:ns:xmpp-stanzas'}).up();
  216. console.warn('duplicate session id', sid);
  217. this.connection.send(ack);
  218. return true;
  219. }
  220. // FIXME: check for a defined action
  221. this.connection.send(ack);
  222. // see http://xmpp.org/extensions/xep-0166.html#concepts-session
  223. switch (action) {
  224. case 'session-initiate':
  225. sess = new JingleSession($(iq).attr('to'), $(iq).find('jingle').attr('sid'), this.connection);
  226. // configure session
  227. if (this.localStream) {
  228. sess.localStreams.push(this.localStream);
  229. }
  230. sess.media_constraints = this.media_constraints;
  231. sess.pc_constraints = this.pc_constraints;
  232. sess.ice_config = this.ice_config;
  233. sess.initiate($(iq).attr('from'), false);
  234. // FIXME: setRemoteDescription should only be done when this call is to be accepted
  235. sess.setRemoteDescription($(iq).find('>jingle'), 'offer');
  236. this.sessions[sess.sid] = sess;
  237. this.jid2session[sess.peerjid] = sess;
  238. // the callback should either
  239. // .sendAnswer and .accept
  240. // or .sendTerminate -- not necessarily synchronus
  241. $(document).trigger('callincoming.jingle', [sess.sid]);
  242. break;
  243. case 'session-accept':
  244. sess.setRemoteDescription($(iq).find('>jingle'), 'answer');
  245. sess.accept();
  246. break;
  247. case 'session-terminate':
  248. console.log('terminating...');
  249. sess.terminate();
  250. this.terminate(sess.sid);
  251. if ($(iq).find('>jingle>reason').length) {
  252. $(document).trigger('callterminated.jingle', [
  253. sess.sid,
  254. $(iq).find('>jingle>reason>:first')[0].tagName,
  255. $(iq).find('>jingle>reason>text').text()
  256. ]);
  257. } else {
  258. $(document).trigger('callterminated.jingle', [sess.sid]);
  259. }
  260. break;
  261. case 'transport-info':
  262. sess.addIceCandidate($(iq).find('>jingle>content'));
  263. break;
  264. case 'session-info':
  265. var affected;
  266. if ($(iq).find('>jingle>ringing[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').length) {
  267. $(document).trigger('ringing.jingle', [sess.sid]);
  268. } else if ($(iq).find('>jingle>mute[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').length) {
  269. affected = $(iq).find('>jingle>mute[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').attr('name');
  270. $(document).trigger('mute.jingle', [sess.sid, affected]);
  271. } else if ($(iq).find('>jingle>unmute[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').length) {
  272. affected = $(iq).find('>jingle>unmute[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').attr('name');
  273. $(document).trigger('unmute.jingle', [sess.sid, affected]);
  274. }
  275. break;
  276. case 'addsource': // FIXME: proprietary
  277. sess.addSource($(iq).find('>jingle>content'));
  278. break;
  279. case 'removesource': // FIXME: proprietary
  280. sess.removeSource($(iq).find('>jingle>content'));
  281. break;
  282. default:
  283. console.warn('jingle action not implemented', action);
  284. break;
  285. }
  286. return true;
  287. },
  288. initiate: function (peerjid, myjid) { // initiate a new jinglesession to peerjid
  289. var sess = new JingleSession(myjid,
  290. Math.random().toString(36).substr(2, 12), // random string
  291. this.connection);
  292. // configure session
  293. if (this.localStream) {
  294. sess.localStreams.push(this.localStream);
  295. }
  296. sess.media_constraints = this.media_constraints;
  297. sess.pc_constraints = this.pc_constraints;
  298. sess.ice_config = this.ice_config;
  299. sess.initiate(peerjid, true);
  300. this.sessions[sess.sid] = sess;
  301. this.jid2session[sess.peerjid] = sess;
  302. sess.sendOffer();
  303. return sess;
  304. },
  305. terminate: function (sid, reason, text) { // terminate by sessionid (or all sessions)
  306. if (sid === null || sid === undefined) {
  307. for (sid in this.sessions) {
  308. if (this.sessions[sid].state != 'ended') {
  309. this.sessions[sid].sendTerminate(reason || (!this.sessions[sid].active()) ? 'cancel' : null, text);
  310. this.sessions[sid].terminate();
  311. }
  312. delete this.jid2session[this.sessions[sid].peerjid];
  313. delete this.sessions[sid];
  314. }
  315. } else if (this.sessions.hasOwnProperty(sid)) {
  316. if (this.sessions[sid].state != 'ended') {
  317. this.sessions[sid].sendTerminate(reason || (!this.sessions[sid].active()) ? 'cancel' : null, text);
  318. this.sessions[sid].terminate();
  319. }
  320. delete this.jid2session[this.sessions[sid].peerjid];
  321. delete this.sessions[sid];
  322. }
  323. },
  324. terminateByJid: function (jid) {
  325. if (this.jid2session.hasOwnProperty(jid)) {
  326. var sess = this.jid2session[jid];
  327. if (sess) {
  328. sess.terminate();
  329. console.log('peer went away silently', jid);
  330. delete this.sessions[sess.sid];
  331. delete this.jid2session[jid];
  332. $(document).trigger('callterminated.jingle', [sess.sid, 'gone']);
  333. }
  334. }
  335. },
  336. getStunAndTurnCredentials: function () {
  337. // get stun and turn configuration from server via xep-0215
  338. // uses time-limited credentials as described in
  339. // http://tools.ietf.org/html/draft-uberti-behave-turn-rest-00
  340. //
  341. // see https://code.google.com/p/prosody-modules/source/browse/mod_turncredentials/mod_turncredentials.lua
  342. // for a prosody module which implements this
  343. //
  344. // currently, this doesn't work with updateIce and therefore credentials with a long
  345. // validity have to be fetched before creating the peerconnection
  346. // TODO: implement refresh via updateIce as described in
  347. // https://code.google.com/p/webrtc/issues/detail?id=1650
  348. this.connection.sendIQ(
  349. $iq({type: 'get', to: this.connection.domain})
  350. .c('services', {xmlns: 'urn:xmpp:extdisco:1'}).c('service', {host: 'turn.' + this.connection.domain}),
  351. function (res) {
  352. var iceservers = [];
  353. $(res).find('>services>service').each(function (idx, el) {
  354. el = $(el);
  355. var dict = {};
  356. switch (el.attr('type')) {
  357. case 'stun':
  358. dict.url = 'stun:' + el.attr('host');
  359. if (el.attr('port')) {
  360. dict.url += ':' + el.attr('port');
  361. }
  362. iceservers.push(dict);
  363. break;
  364. case 'turn':
  365. dict.url = 'turn:';
  366. if (el.attr('username')) { // https://code.google.com/p/webrtc/issues/detail?id=1508
  367. if (navigator.userAgent.match(/Chrom(e|ium)\/([0-9]+)\./) && parseInt(navigator.userAgent.match(/Chrom(e|ium)\/([0-9]+)\./)[2], 10) < 28) {
  368. dict.url += el.attr('username') + '@';
  369. } else {
  370. dict.username = el.attr('username'); // only works in M28
  371. }
  372. }
  373. dict.url += el.attr('host');
  374. if (el.attr('port') && el.attr('port') != '3478') {
  375. dict.url += ':' + el.attr('port');
  376. }
  377. if (el.attr('transport') && el.attr('transport') != 'udp') {
  378. dict.url += '?transport=' + el.attr('transport');
  379. }
  380. if (el.attr('password')) {
  381. dict.credential = el.attr('password');
  382. }
  383. iceservers.push(dict);
  384. break;
  385. }
  386. });
  387. this.ice_config.iceServers = iceservers;
  388. },
  389. function (err) {
  390. console.warn('getting turn credentials failed', err);
  391. console.warn('is mod_turncredentials or similar installed?');
  392. }
  393. );
  394. // implement push?
  395. }
  396. });
  397. /* jshint -W117 */
  398. // SDP STUFF
  399. function SDP(sdp) {
  400. this.media = sdp.split('\r\nm=');
  401. for (var i = 1; i < this.media.length; i++) {
  402. this.media[i] = 'm=' + this.media[i];
  403. if (i != this.media.length - 1) {
  404. this.media[i] += '\r\n';
  405. }
  406. }
  407. this.session = this.media.shift() + '\r\n';
  408. this.raw = this.session + this.media.join('');
  409. }
  410. // remove iSAC and CN from SDP
  411. SDP.prototype.mangle = function () {
  412. var i, j, mline, lines, rtpmap, newdesc;
  413. for (i = 0; i < this.media.length; i++) {
  414. lines = this.media[i].split('\r\n');
  415. lines.pop(); // remove empty last element
  416. mline = SDPUtil.parse_mline(lines.shift());
  417. if (mline.media != 'audio')
  418. continue;
  419. newdesc = '';
  420. mline.fmt.length = 0;
  421. for (j = 0; j < lines.length; j++) {
  422. if (lines[j].substr(0, 9) == 'a=rtpmap:') {
  423. rtpmap = SDPUtil.parse_rtpmap(lines[j]);
  424. if (rtpmap.name == 'CN' || rtpmap.name == 'ISAC')
  425. continue;
  426. mline.fmt.push(rtpmap.id);
  427. newdesc += lines[j] + '\r\n';
  428. } else {
  429. newdesc += lines[j] + '\r\n';
  430. }
  431. }
  432. this.media[i] = SDPUtil.build_mline(mline) + '\r\n';
  433. this.media[i] += newdesc;
  434. }
  435. this.raw = this.session + this.media.join('');
  436. };
  437. // remove lines matching prefix from session section
  438. SDP.prototype.removeSessionLines = function(prefix) {
  439. var ob = this;
  440. var lines = SDPUtil.find_lines(this.session, prefix);
  441. lines.forEach(function(line) {
  442. ob.session = ob.session.replace(line + '\r\n', '');
  443. });
  444. this.raw = this.session + this.media.join('');
  445. return lines;
  446. }
  447. // remove lines matching prefix from a media section specified by mediaindex
  448. // TODO: non-numeric mediaindex could match mid
  449. SDP.prototype.removeMediaLines = function(mediaindex, prefix) {
  450. var ob = this;
  451. var lines = SDPUtil.find_lines(this.media[mediaindex], prefix);
  452. lines.forEach(function(line) {
  453. ob.media[mediaindex] = ob.media[mediaindex].replace(line + '\r\n', '');
  454. });
  455. this.raw = this.session + this.media.join('');
  456. return lines;
  457. }
  458. // add content's to a jingle element
  459. SDP.prototype.toJingle = function (elem, thecreator) {
  460. var i, j, k, mline, ssrc, rtpmap, tmp, line, lines;
  461. var ob = this;
  462. // new bundle plan
  463. if (SDPUtil.find_line(this.session, 'a=group:')) {
  464. lines = SDPUtil.find_lines(this.session, 'a=group:');
  465. for (i = 0; i < lines.length; i++) {
  466. tmp = lines[i].split(' ');
  467. var semantics = tmp.shift().substr(8);
  468. // new plan
  469. elem.c('group', {xmlns: 'urn:xmpp:jingle:apps:grouping:0', type: semantics, semantics:semantics});
  470. for (j = 0; j < tmp.length; j++) {
  471. elem.c('content', {name: tmp[j]}).up();
  472. }
  473. elem.up();
  474. // temporary plan, to be removed
  475. elem.c('group', {xmlns: 'urn:ietf:rfc:5888', type: semantics});
  476. for (j = 0; j < tmp.length; j++) {
  477. elem.c('content', {name: tmp[j]}).up();
  478. }
  479. elem.up();
  480. }
  481. }
  482. // old bundle plan, to be removed
  483. var bundle = [];
  484. if (SDPUtil.find_line(this.session, 'a=group:BUNDLE')) {
  485. bundle = SDPUtil.find_line(this.session, 'a=group:BUNDLE ').split(' ');
  486. bundle.shift();
  487. }
  488. for (i = 0; i < this.media.length; i++) {
  489. mline = SDPUtil.parse_mline(this.media[i].split('\r\n')[0]);
  490. if (!(mline.media == 'audio' || mline.media == 'video')) {
  491. continue;
  492. }
  493. if (SDPUtil.find_line(this.media[i], 'a=ssrc:')) {
  494. ssrc = SDPUtil.find_line(this.media[i], 'a=ssrc:').substring(7).split(' ')[0]; // take the first
  495. } else {
  496. ssrc = false;
  497. }
  498. elem.c('content', {creator: thecreator, name: mline.media});
  499. if (SDPUtil.find_line(this.media[i], 'a=mid:')) {
  500. // prefer identifier from a=mid if present
  501. var mid = SDPUtil.parse_mid(SDPUtil.find_line(this.media[i], 'a=mid:'));
  502. elem.attrs({ name: mid });
  503. // old BUNDLE plan, to be removed
  504. if (bundle.indexOf(mid) != -1) {
  505. elem.c('bundle', {xmlns: 'http://estos.de/ns/bundle'}).up();
  506. bundle.splice(bundle.indexOf(mid), 1);
  507. }
  508. }
  509. if (SDPUtil.find_line(this.media[i], 'a=rtpmap:').length) {
  510. elem.c('description',
  511. {xmlns: 'urn:xmpp:jingle:apps:rtp:1',
  512. media: mline.media });
  513. if (ssrc) {
  514. elem.attrs({ssrc: ssrc});
  515. }
  516. for (j = 0; j < mline.fmt.length; j++) {
  517. rtpmap = SDPUtil.find_line(this.media[i], 'a=rtpmap:' + mline.fmt[j]);
  518. elem.c('payload-type', SDPUtil.parse_rtpmap(rtpmap));
  519. // put any 'a=fmtp:' + mline.fmt[j] lines into <param name=foo value=bar/>
  520. if (SDPUtil.find_line(this.media[i], 'a=fmtp:' + mline.fmt[j])) {
  521. tmp = SDPUtil.parse_fmtp(SDPUtil.find_line(this.media[i], 'a=fmtp:' + mline.fmt[j]));
  522. for (k = 0; k < tmp.length; k++) {
  523. elem.c('parameter', tmp[k]).up();
  524. }
  525. }
  526. this.RtcpFbToJingle(this.media[i], elem, mline.fmt[j]); // XEP-0293 -- map a=rtcp-fb
  527. elem.up();
  528. }
  529. if (SDPUtil.find_line(this.media[i], 'a=crypto:', this.session)) {
  530. elem.c('encryption', {required: 1});
  531. var crypto = SDPUtil.find_lines(this.media[i], 'a=crypto:', this.session);
  532. crypto.forEach(function(line) {
  533. elem.c('crypto', SDPUtil.parse_crypto(line)).up();
  534. });
  535. elem.up(); // end of encryption
  536. }
  537. if (ssrc) {
  538. // new style mapping
  539. elem.c('source', { ssrc: ssrc, xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });
  540. // FIXME: group by ssrc and support multiple different ssrcs
  541. var ssrclines = SDPUtil.find_lines(this.media[i], 'a=ssrc:');
  542. ssrclines.forEach(function(line) {
  543. idx = line.indexOf(' ');
  544. var linessrc = line.substr(0, idx).substr(7);
  545. if (linessrc != ssrc) {
  546. elem.up();
  547. ssrc = linessrc;
  548. elem.c('source', { ssrc: ssrc, xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });
  549. }
  550. var kv = line.substr(idx + 1);
  551. elem.c('parameter');
  552. if (kv.indexOf(':') == -1) {
  553. elem.attrs({ name: kv });
  554. } else {
  555. elem.attrs({ name: kv.split(':', 2)[0] });
  556. elem.attrs({ value: kv.split(':', 2)[1] });
  557. }
  558. elem.up();
  559. });
  560. elem.up();
  561. // old proprietary mapping, to be removed at some point
  562. tmp = SDPUtil.parse_ssrc(this.media[i]);
  563. tmp.xmlns = 'http://estos.de/ns/ssrc';
  564. tmp.ssrc = ssrc;
  565. elem.c('ssrc', tmp).up(); // ssrc is part of description
  566. }
  567. if (SDPUtil.find_line(this.media[i], 'a=rtcp-mux')) {
  568. elem.c('rtcp-mux').up();
  569. }
  570. // XEP-0293 -- map a=rtcp-fb:*
  571. this.RtcpFbToJingle(this.media[i], elem, '*');
  572. // XEP-0294
  573. if (SDPUtil.find_line(this.media[i], 'a=extmap:')) {
  574. lines = SDPUtil.find_lines(this.media[i], 'a=extmap:');
  575. for (j = 0; j < lines.length; j++) {
  576. tmp = SDPUtil.parse_extmap(lines[j]);
  577. elem.c('rtp-hdrext', { xmlns: 'urn:xmpp:jingle:apps:rtp:rtp-hdrext:0',
  578. uri: tmp.uri,
  579. id: tmp.value });
  580. if (tmp.hasOwnProperty('direction')) {
  581. switch (tmp.direction) {
  582. case 'sendonly':
  583. elem.attrs({senders: 'responder'});
  584. break;
  585. case 'recvonly':
  586. elem.attrs({senders: 'initiator'});
  587. break;
  588. case 'sendrecv':
  589. elem.attrs({senders: 'both'});
  590. break;
  591. case 'inactive':
  592. elem.attrs({senders: 'none'});
  593. break;
  594. }
  595. }
  596. // TODO: handle params
  597. elem.up();
  598. }
  599. }
  600. elem.up(); // end of description
  601. }
  602. elem.c('transport', {xmlns: 'urn:xmpp:jingle:transports:ice-udp:1'});
  603. // XEP-0320
  604. var fingerprints = SDPUtil.find_lines(this.media[i], 'a=fingerprint:', this.session);
  605. fingerprints.forEach(function(line) {
  606. tmp = SDPUtil.parse_fingerprint(line);
  607. tmp.xmlns = 'urn:xmpp:tmp:jingle:apps:dtls:0';
  608. // tmp.xmlns = 'urn:xmpp:jingle:apps:dtls:0'; -- FIXME: update receivers first
  609. elem.c('fingerprint').t(tmp.fingerprint);
  610. delete tmp.fingerprint;
  611. line = SDPUtil.find_line(ob.media[i], 'a=setup:', ob.session);
  612. if (line) {
  613. tmp.setup = line.substr(8);
  614. }
  615. elem.attrs(tmp);
  616. elem.up();
  617. });
  618. tmp = SDPUtil.iceparams(this.media[i], this.session);
  619. if (tmp) {
  620. elem.attrs(tmp);
  621. // XEP-0176
  622. if (SDPUtil.find_line(this.media[i], 'a=candidate:', this.session)) { // add any a=candidate lines
  623. lines = SDPUtil.find_lines(this.media[i], 'a=candidate:', this.session);
  624. for (j = 0; j < lines.length; j++) {
  625. tmp = SDPUtil.candidateToJingle(lines[j]);
  626. elem.c('candidate', tmp).up();
  627. }
  628. }
  629. elem.up(); // end of transport
  630. }
  631. if (SDPUtil.find_line(this.media[i], 'a=sendrecv', this.session)) {
  632. elem.attrs({senders: 'both'});
  633. } else if (SDPUtil.find_line(this.media[i], 'a=sendonly', this.session)) {
  634. elem.attrs({senders: 'initiator'});
  635. } else if (SDPUtil.find_line(this.media[i], 'a=recvonly', this.session)) {
  636. elem.attrs({senders: 'responder'});
  637. } else if (SDPUtil.find_line(this.media[i], 'a=inactive', this.session)) {
  638. elem.attrs({senders: 'none'});
  639. }
  640. if (mline.port == '0') {
  641. // estos hack to reject an m-line
  642. elem.attrs({senders: 'rejected'});
  643. }
  644. elem.up(); // end of content
  645. }
  646. elem.up();
  647. return elem;
  648. };
  649. SDP.prototype.RtcpFbToJingle = function (sdp, elem, payloadtype) { // XEP-0293
  650. var lines = SDPUtil.find_lines(sdp, 'a=rtcp-fb:' + payloadtype);
  651. for (var i = 0; i < lines.length; i++) {
  652. var tmp = SDPUtil.parse_rtcpfb(lines[i]);
  653. if (tmp.type == 'trr-int') {
  654. elem.c('rtcp-fb-trr-int', {xmlns: 'urn:xmpp:jingle:apps:rtp:rtcp-fb:0', value: tmp.params[0]});
  655. elem.up();
  656. } else {
  657. elem.c('rtcp-fb', {xmlns: 'urn:xmpp:jingle:apps:rtp:rtcp-fb:0', type: tmp.type});
  658. if (tmp.params.length > 0) {
  659. elem.attrs({'subtype': tmp.params[0]});
  660. }
  661. elem.up();
  662. }
  663. }
  664. };
  665. SDP.prototype.RtcpFbFromJingle = function (elem, payloadtype) { // XEP-0293
  666. var media = '';
  667. var tmp = elem.find('>rtcp-fb-trr-int[xmlns="urn:xmpp:jingle:apps:rtp:rtcp-fb:0"]');
  668. if (tmp.length) {
  669. media += 'a=rtcp-fb:' + '*' + ' ' + 'trr-int' + ' ';
  670. if (tmp.attr('value')) {
  671. media += tmp.attr('value');
  672. } else {
  673. media += '0';
  674. }
  675. media += '\r\n';
  676. }
  677. tmp = elem.find('>rtcp-fb[xmlns="urn:xmpp:jingle:apps:rtp:rtcp-fb:0"]');
  678. tmp.each(function () {
  679. media += 'a=rtcp-fb:' + payloadtype + ' ' + $(this).attr('type');
  680. if ($(this).attr('subtype')) {
  681. media += ' ' + $(this).attr('subtype');
  682. }
  683. media += '\r\n';
  684. });
  685. return media;
  686. };
  687. // construct an SDP from a jingle stanza
  688. SDP.prototype.fromJingle = function (jingle) {
  689. var obj = this;
  690. this.raw = 'v=0\r\n' +
  691. 'o=- ' + '1923518516' + ' 2 IN IP4 0.0.0.0\r\n' +// FIXME
  692. 's=-\r\n' +
  693. 't=0 0\r\n';
  694. // http://tools.ietf.org/html/draft-ietf-mmusic-sdp-bundle-negotiation-04#section-8
  695. if ($(jingle).find('>group[xmlns="urn:xmpp:jingle:apps:grouping:0"]').length) {
  696. try {
  697. $(jingle).find('>group[xmlns="urn:xmpp:jingle:apps:grouping:0"]').each(function (idx, group) {
  698. var contents = $(group).find('>content').map(function (idx, content) {
  699. return $(content).attr('name');
  700. }).get();
  701. if (contents.length > 0) {
  702. obj.raw += 'a=group:' + ($(group).attr('semantics') || $(group).attr('type')) + ' ' + contents.join(' ') + '\r\n';
  703. }
  704. });
  705. } catch (e) { console.error(e.toString()); }
  706. } else if ($(jingle).find('>group[xmlns="urn:ietf:rfc:5888"]').length) {
  707. // temporary namespace, not to be used. to be removed soon.
  708. $(jingle).find('>group[xmlns="urn:ietf:rfc:5888"]').each(function (idx, group) {
  709. var contents = $(group).find('>content').map(function (idx, content) {
  710. return $(content).attr('name');
  711. }).get();
  712. if ($(group).attr('type') !== null && contents.length > 0) {
  713. obj.raw += 'a=group:' + $(group).attr('type') + ' ' + contents.join(' ') + '\r\n';
  714. }
  715. });
  716. } else {
  717. // for backward compability, to be removed soon
  718. // assume all contents are in the same bundle group, can be improved upon later
  719. var bundle = $(jingle).find('>content').filter(function (idx, content) {
  720. //elem.c('bundle', {xmlns:'http://estos.de/ns/bundle'});
  721. return $(content).find('>bundle').length > 0;
  722. }).map(function (idx, content) {
  723. return $(content).attr('name');
  724. }).get();
  725. if (bundle.length) {
  726. this.raw += 'a=group:BUNDLE ' + bundle.join(' ') + '\r\n';
  727. }
  728. }
  729. this.session = this.raw;
  730. jingle.find('>content').each(function () {
  731. var m = obj.jingle2media($(this));
  732. obj.media.push(m);
  733. });
  734. // reconstruct msid-semantic -- apparently not necessary
  735. /*
  736. var msid = SDPUtil.parse_ssrc(this.raw);
  737. if (msid.hasOwnProperty('mslabel')) {
  738. this.session += "a=msid-semantic: WMS " + msid.mslabel + "\r\n";
  739. }
  740. */
  741. this.raw = this.session + this.media.join('');
  742. };
  743. // translate a jingle content element into an an SDP media part
  744. SDP.prototype.jingle2media = function (content) {
  745. var media = '',
  746. desc = content.find('description'),
  747. ssrc = desc.attr('ssrc'),
  748. self = this,
  749. tmp;
  750. tmp = { media: desc.attr('media') };
  751. tmp.port = '1';
  752. if (content.attr('senders') == 'rejected') {
  753. // estos hack to reject an m-line.
  754. tmp.port = '0';
  755. }
  756. if (content.find('>transport>fingerprint').length || desc.find('encryption').length) {
  757. tmp.proto = 'RTP/SAVPF';
  758. } else {
  759. tmp.proto = 'RTP/AVPF';
  760. }
  761. tmp.fmt = desc.find('payload-type').map(function () { return $(this).attr('id'); }).get();
  762. media += SDPUtil.build_mline(tmp) + '\r\n';
  763. media += 'c=IN IP4 0.0.0.0\r\n';
  764. media += 'a=rtcp:1 IN IP4 0.0.0.0\r\n';
  765. tmp = content.find('>transport[xmlns="urn:xmpp:jingle:transports:ice-udp:1"]');
  766. if (tmp.length) {
  767. if (tmp.attr('ufrag')) {
  768. media += SDPUtil.build_iceufrag(tmp.attr('ufrag')) + '\r\n';
  769. }
  770. if (tmp.attr('pwd')) {
  771. media += SDPUtil.build_icepwd(tmp.attr('pwd')) + '\r\n';
  772. }
  773. tmp.find('>fingerprint').each(function () {
  774. // FIXME: check namespace at some point
  775. media += 'a=fingerprint:' + $(this).attr('hash');
  776. media += ' ' + $(this).text();
  777. media += '\r\n';
  778. if ($(this).attr('setup')) {
  779. media += 'a=setup:' + $(this).attr('setup') + '\r\n';
  780. }
  781. });
  782. }
  783. switch (content.attr('senders')) {
  784. case 'initiator':
  785. media += 'a=sendonly\r\n';
  786. break;
  787. case 'responder':
  788. media += 'a=recvonly\r\n';
  789. break;
  790. case 'none':
  791. media += 'a=inactive\r\n';
  792. break;
  793. case 'both':
  794. media += 'a=sendrecv\r\n';
  795. break;
  796. }
  797. media += 'a=mid:' + content.attr('name') + '\r\n';
  798. // <description><rtcp-mux/></description>
  799. // see http://code.google.com/p/libjingle/issues/detail?id=309 -- no spec though
  800. // and http://mail.jabber.org/pipermail/jingle/2011-December/001761.html
  801. if (desc.find('rtcp-mux').length) {
  802. media += 'a=rtcp-mux\r\n';
  803. }
  804. if (desc.find('encryption').length) {
  805. desc.find('encryption>crypto').each(function () {
  806. media += 'a=crypto:' + $(this).attr('tag');
  807. media += ' ' + $(this).attr('crypto-suite');
  808. media += ' ' + $(this).attr('key-params');
  809. if ($(this).attr('session-params')) {
  810. media += ' ' + $(this).attr('session-params');
  811. }
  812. media += '\r\n';
  813. });
  814. }
  815. desc.find('payload-type').each(function () {
  816. media += SDPUtil.build_rtpmap(this) + '\r\n';
  817. if ($(this).find('>parameter').length) {
  818. media += 'a=fmtp:' + $(this).attr('id') + ' ';
  819. media += $(this).find('parameter').map(function () { return ($(this).attr('name') ? ($(this).attr('name') + '=') : '') + $(this).attr('value'); }).get().join(';');
  820. media += '\r\n';
  821. }
  822. // xep-0293
  823. media += self.RtcpFbFromJingle($(this), $(this).attr('id'));
  824. });
  825. // xep-0293
  826. media += self.RtcpFbFromJingle(desc, '*');
  827. // xep-0294
  828. tmp = desc.find('>rtp-hdrext[xmlns="urn:xmpp:jingle:apps:rtp:rtp-hdrext:0"]');
  829. tmp.each(function () {
  830. media += 'a=extmap:' + $(this).attr('id') + ' ' + $(this).attr('uri') + '\r\n';
  831. });
  832. content.find('>transport[xmlns="urn:xmpp:jingle:transports:ice-udp:1"]>candidate').each(function () {
  833. media += SDPUtil.candidateFromJingle(this);
  834. });
  835. tmp = content.find('description>source[xmlns="urn:xmpp:jingle:apps:rtp:ssma:0"]');
  836. tmp.each(function () {
  837. var ssrc = $(this).attr('ssrc');
  838. $(this).find('>parameter').each(function () {
  839. media += 'a=ssrc:' + ssrc + ' ' + $(this).attr('name');
  840. if ($(this).attr('value') && $(this).attr('value').length)
  841. media += ':' + $(this).attr('value');
  842. media += '\r\n';
  843. });
  844. });
  845. if (tmp.length === 0) {
  846. // fallback to proprietary mapping of a=ssrc lines
  847. tmp = content.find('description>ssrc[xmlns="http://estos.de/ns/ssrc"]');
  848. if (tmp.length) {
  849. media += 'a=ssrc:' + ssrc + ' cname:' + tmp.attr('cname') + '\r\n';
  850. media += 'a=ssrc:' + ssrc + ' msid:' + tmp.attr('msid') + '\r\n';
  851. media += 'a=ssrc:' + ssrc + ' mslabel:' + tmp.attr('mslabel') + '\r\n';
  852. media += 'a=ssrc:' + ssrc + ' label:' + tmp.attr('label') + '\r\n';
  853. }
  854. }
  855. return media;
  856. };
  857. SDPUtil = {
  858. iceparams: function (mediadesc, sessiondesc) {
  859. var data = null;
  860. if (SDPUtil.find_line(mediadesc, 'a=ice-ufrag:', sessiondesc) &&
  861. SDPUtil.find_line(mediadesc, 'a=ice-pwd:', sessiondesc)) {
  862. data = {
  863. ufrag: SDPUtil.parse_iceufrag(SDPUtil.find_line(mediadesc, 'a=ice-ufrag:', sessiondesc)),
  864. pwd: SDPUtil.parse_icepwd(SDPUtil.find_line(mediadesc, 'a=ice-pwd:', sessiondesc))
  865. };
  866. }
  867. return data;
  868. },
  869. parse_iceufrag: function (line) {
  870. return line.substring(12);
  871. },
  872. build_iceufrag: function (frag) {
  873. return 'a=ice-ufrag:' + frag;
  874. },
  875. parse_icepwd: function (line) {
  876. return line.substring(10);
  877. },
  878. build_icepwd: function (pwd) {
  879. return 'a=ice-pwd:' + pwd;
  880. },
  881. parse_mid: function (line) {
  882. return line.substring(6);
  883. },
  884. parse_mline: function (line) {
  885. var parts = line.substring(2).split(' '),
  886. data = {};
  887. data.media = parts.shift();
  888. data.port = parts.shift();
  889. data.proto = parts.shift();
  890. if (parts[parts.length - 1] === '') { // trailing whitespace
  891. parts.pop();
  892. }
  893. data.fmt = parts;
  894. return data;
  895. },
  896. build_mline: function (mline) {
  897. return 'm=' + mline.media + ' ' + mline.port + ' ' + mline.proto + ' ' + mline.fmt.join(' ');
  898. },
  899. parse_rtpmap: function (line) {
  900. var parts = line.substring(9).split(' '),
  901. data = {};
  902. data.id = parts.shift();
  903. parts = parts[0].split('/');
  904. data.name = parts.shift();
  905. data.clockrate = parts.shift();
  906. data.channels = parts.length ? parts.shift() : '1';
  907. return data;
  908. },
  909. build_rtpmap: function (el) {
  910. var line = 'a=rtpmap:' + el.getAttribute('id') + ' ' + el.getAttribute('name') + '/' + el.getAttribute('clockrate');
  911. if (el.getAttribute('channels') && el.getAttribute('channels') != '1') {
  912. line += '/' + el.getAttribute('channels');
  913. }
  914. return line;
  915. },
  916. parse_crypto: function (line) {
  917. var parts = line.substring(9).split(' '),
  918. data = {};
  919. data.tag = parts.shift();
  920. data['crypto-suite'] = parts.shift();
  921. data['key-params'] = parts.shift();
  922. if (parts.length) {
  923. data['session-params'] = parts.join(' ');
  924. }
  925. return data;
  926. },
  927. parse_fingerprint: function (line) { // RFC 4572
  928. var parts = line.substring(14).split(' '),
  929. data = {};
  930. data.hash = parts.shift();
  931. data.fingerprint = parts.shift();
  932. // TODO assert that fingerprint satisfies 2UHEX *(":" 2UHEX) ?
  933. return data;
  934. },
  935. parse_fmtp: function (line) {
  936. var parts = line.split(' '),
  937. i, key, value,
  938. data = [];
  939. parts.shift();
  940. parts = parts.join(' ').split(';');
  941. for (i = 0; i < parts.length; i++) {
  942. key = parts[i].split('=')[0];
  943. while (key.length && key[0] == ' ') {
  944. key = key.substring(1);
  945. }
  946. value = parts[i].split('=')[1];
  947. if (key && value) {
  948. data.push({name: key, value: value});
  949. } else if (key) {
  950. // rfc 4733 (DTMF) style stuff
  951. data.push({name: '', value: key});
  952. }
  953. }
  954. return data;
  955. },
  956. parse_icecandidate: function (line) {
  957. var candidate = {},
  958. elems = line.split(' ');
  959. candidate.foundation = elems[0].substring(12);
  960. candidate.component = elems[1];
  961. candidate.protocol = elems[2].toLowerCase();
  962. candidate.priority = elems[3];
  963. candidate.ip = elems[4];
  964. candidate.port = elems[5];
  965. // elems[6] => "typ"
  966. candidate.type = elems[7];
  967. candidate.generation = 0; // default value, may be overwritten below
  968. for (var i = 8; i < elems.length; i += 2) {
  969. switch (elems[i]) {
  970. case 'raddr':
  971. candidate['rel-addr'] = elems[i + 1];
  972. break;
  973. case 'rport':
  974. candidate['rel-port'] = elems[i + 1];
  975. break;
  976. case 'generation':
  977. candidate.generation = elems[i + 1];
  978. break;
  979. default: // TODO
  980. console.log('parse_icecandidate not translating "' + elems[i] + '" = "' + elems[i + 1] + '"');
  981. }
  982. }
  983. candidate.network = '1';
  984. candidate.id = Math.random().toString(36).substr(2, 10); // not applicable to SDP -- FIXME: should be unique, not just random
  985. return candidate;
  986. },
  987. build_icecandidate: function (cand) {
  988. var line = ['a=candidate:' + cand.foundation, cand.component, cand.protocol, cand.priority, cand.ip, cand.port, 'typ', cand.type].join(' ');
  989. line += ' ';
  990. switch (cand.type) {
  991. case 'srflx':
  992. case 'prflx':
  993. case 'relay':
  994. if (cand.hasOwnAttribute('rel-addr') && cand.hasOwnAttribute('rel-port')) {
  995. line += 'raddr';
  996. line += ' ';
  997. line += cand['rel-addr'];
  998. line += ' ';
  999. line += 'rport';
  1000. line += ' ';
  1001. line += cand['rel-port'];
  1002. line += ' ';
  1003. }
  1004. break;
  1005. }
  1006. line += 'generation';
  1007. line += ' ';
  1008. line += cand.hasOwnAttribute('generation') ? cand.generation : '0';
  1009. return line;
  1010. },
  1011. parse_ssrc: function (desc) {
  1012. // proprietary mapping of a=ssrc lines
  1013. // TODO: see "Jingle RTP Source Description" by Juberti and P. Thatcher on google docs
  1014. // and parse according to that
  1015. var lines = desc.split('\r\n'),
  1016. data = {};
  1017. for (var i = 0; i < lines.length; i++) {
  1018. if (lines[i].substring(0, 7) == 'a=ssrc:') {
  1019. var idx = lines[i].indexOf(' ');
  1020. data[lines[i].substr(idx + 1).split(':', 2)[0]] = lines[i].substr(idx + 1).split(':', 2)[1];
  1021. }
  1022. }
  1023. return data;
  1024. },
  1025. parse_rtcpfb: function (line) {
  1026. var parts = line.substr(10).split(' ');
  1027. var data = {};
  1028. data.pt = parts.shift();
  1029. data.type = parts.shift();
  1030. data.params = parts;
  1031. return data;
  1032. },
  1033. parse_extmap: function (line) {
  1034. var parts = line.substr(9).split(' ');
  1035. var data = {};
  1036. data.value = parts.shift();
  1037. if (data.value.indexOf('/') != -1) {
  1038. data.direction = data.value.substr(data.value.indexOf('/') + 1);
  1039. data.value = data.value.substr(0, data.value.indexOf('/'));
  1040. } else {
  1041. data.direction = 'both';
  1042. }
  1043. data.uri = parts.shift();
  1044. data.params = parts;
  1045. return data;
  1046. },
  1047. find_line: function (haystack, needle, sessionpart) {
  1048. var lines = haystack.split('\r\n');
  1049. for (var i = 0; i < lines.length; i++) {
  1050. if (lines[i].substring(0, needle.length) == needle) {
  1051. return lines[i];
  1052. }
  1053. }
  1054. if (!sessionpart) {
  1055. return false;
  1056. }
  1057. // search session part
  1058. lines = sessionpart.split('\r\n');
  1059. for (var j = 0; j < lines.length; j++) {
  1060. if (lines[j].substring(0, needle.length) == needle) {
  1061. return lines[j];
  1062. }
  1063. }
  1064. return false;
  1065. },
  1066. find_lines: function (haystack, needle, sessionpart) {
  1067. var lines = haystack.split('\r\n'),
  1068. needles = [];
  1069. for (var i = 0; i < lines.length; i++) {
  1070. if (lines[i].substring(0, needle.length) == needle)
  1071. needles.push(lines[i]);
  1072. }
  1073. if (needles.length || !sessionpart) {
  1074. return needles;
  1075. }
  1076. // search session part
  1077. lines = sessionpart.split('\r\n');
  1078. for (var j = 0; j < lines.length; j++) {
  1079. if (lines[j].substring(0, needle.length) == needle) {
  1080. needles.push(lines[j]);
  1081. }
  1082. }
  1083. return needles;
  1084. },
  1085. candidateToJingle: function (line) {
  1086. // a=candidate:2979166662 1 udp 2113937151 192.168.2.100 57698 typ host generation 0
  1087. // <candidate component=... foundation=... generation=... id=... ip=... network=... port=... priority=... protocol=... type=.../>
  1088. if (line.substring(0, 12) != 'a=candidate:') {
  1089. console.log('parseCandidate called with a line that is not a candidate line');
  1090. console.log(line);
  1091. return null;
  1092. }
  1093. if (line.substring(line.length - 2) == '\r\n') // chomp it
  1094. line = line.substring(0, line.length - 2);
  1095. var candidate = {},
  1096. elems = line.split(' '),
  1097. i;
  1098. if (elems[6] != 'typ') {
  1099. console.log('did not find typ in the right place');
  1100. console.log(line);
  1101. return null;
  1102. }
  1103. candidate.foundation = elems[0].substring(12);
  1104. candidate.component = elems[1];
  1105. candidate.protocol = elems[2].toLowerCase();
  1106. candidate.priority = elems[3];
  1107. candidate.ip = elems[4];
  1108. candidate.port = elems[5];
  1109. // elems[6] => "typ"
  1110. candidate.type = elems[7];
  1111. for (i = 8; i < elems.length; i += 2) {
  1112. switch (elems[i]) {
  1113. case 'raddr':
  1114. candidate['rel-addr'] = elems[i + 1];
  1115. break;
  1116. case 'rport':
  1117. candidate['rel-port'] = elems[i + 1];
  1118. break;
  1119. case 'generation':
  1120. candidate.generation = elems[i + 1];
  1121. break;
  1122. default: // TODO
  1123. console.log('not translating "' + elems[i] + '" = "' + elems[i + 1] + '"');
  1124. }
  1125. }
  1126. candidate.network = '1';
  1127. candidate.id = Math.random().toString(36).substr(2, 10); // not applicable to SDP -- FIXME: should be unique, not just random
  1128. return candidate;
  1129. },
  1130. candidateFromJingle: function (cand) {
  1131. var line = 'a=candidate:';
  1132. line += cand.getAttribute('foundation');
  1133. line += ' ';
  1134. line += cand.getAttribute('component');
  1135. line += ' ';
  1136. line += cand.getAttribute('protocol'); //.toUpperCase(); // chrome M23 doesn't like this
  1137. line += ' ';
  1138. line += cand.getAttribute('priority');
  1139. line += ' ';
  1140. line += cand.getAttribute('ip');
  1141. line += ' ';
  1142. line += cand.getAttribute('port');
  1143. line += ' ';
  1144. line += 'typ';
  1145. line += ' ' + cand.getAttribute('type');
  1146. line += ' ';
  1147. switch (cand.getAttribute('type')) {
  1148. case 'srflx':
  1149. case 'prflx':
  1150. case 'relay':
  1151. if (cand.getAttribute('rel-addr') && cand.getAttribute('rel-port')) {
  1152. line += 'raddr';
  1153. line += ' ';
  1154. line += cand.getAttribute('rel-addr');
  1155. line += ' ';
  1156. line += 'rport';
  1157. line += ' ';
  1158. line += cand.getAttribute('rel-port');
  1159. line += ' ';
  1160. }
  1161. break;
  1162. }
  1163. line += 'generation';
  1164. line += ' ';
  1165. line += cand.getAttribute('generation') || '0';
  1166. return line + '\r\n';
  1167. }
  1168. };
  1169. /* jshint -W117 */
  1170. // Jingle stuff
  1171. function JingleSession(me, sid, connection) {
  1172. this.me = me;
  1173. this.sid = sid;
  1174. this.connection = connection;
  1175. this.initiator = null;
  1176. this.responder = null;
  1177. this.isInitiator = null;
  1178. this.peerjid = null;
  1179. this.state = null;
  1180. this.peerconnection = null;
  1181. this.remoteStream = null;
  1182. this.localSDP = null;
  1183. this.remoteSDP = null;
  1184. this.localStreams = [];
  1185. this.relayedStreams = [];
  1186. this.remoteStreams = [];
  1187. this.startTime = null;
  1188. this.stopTime = null;
  1189. this.media_constraints = null;
  1190. this.pc_constraints = null;
  1191. this.ice_config = {},
  1192. this.drip_container = [];
  1193. this.usetrickle = true;
  1194. this.usepranswer = false; // early transport warmup -- mind you, this might fail. depends on webrtc issue 1718
  1195. this.usedrip = false; // dripping is sending trickle candidates not one-by-one
  1196. this.hadstuncandidate = false;
  1197. this.hadturncandidate = false;
  1198. this.lasticecandidate = false;
  1199. this.statsinterval = null;
  1200. this.reason = null;
  1201. this.addssrc = [];
  1202. this.removessrc = [];
  1203. this.wait = true;
  1204. }
  1205. JingleSession.prototype.initiate = function (peerjid, isInitiator) {
  1206. var obj = this;
  1207. if (this.state !== null) {
  1208. console.error('attempt to initiate on session ' + this.sid +
  1209. 'in state ' + this.state);
  1210. return;
  1211. }
  1212. this.isInitiator = isInitiator;
  1213. this.state = 'pending';
  1214. this.initiator = isInitiator ? this.me : peerjid;
  1215. this.responder = !isInitiator ? this.me : peerjid;
  1216. this.peerjid = peerjid;
  1217. console.log('create PeerConnection ' + JSON.stringify(this.ice_config));
  1218. try {
  1219. this.peerconnection = new RTCPeerconnection(this.ice_config,
  1220. this.pc_constraints);
  1221. console.log('Created RTCPeerConnnection');
  1222. } catch (e) {
  1223. console.error('Failed to create PeerConnection, exception: ',
  1224. e.message);
  1225. console.error(e);
  1226. return;
  1227. }
  1228. this.hadstuncandidate = false;
  1229. this.hadturncandidate = false;
  1230. this.lasticecandidate = false;
  1231. this.peerconnection.onicecandidate = function (event) {
  1232. obj.sendIceCandidate(event.candidate);
  1233. };
  1234. this.peerconnection.onaddstream = function (event) {
  1235. obj.remoteStream = event.stream;
  1236. obj.remoteStreams.push(event.stream);
  1237. $(document).trigger('remotestreamadded.jingle', [event, obj.sid]);
  1238. };
  1239. this.peerconnection.onremovestream = function (event) {
  1240. obj.remoteStream = null;
  1241. // FIXME: remove from this.remoteStreams
  1242. $(document).trigger('remotestreamremoved.jingle', [event, obj.sid]);
  1243. };
  1244. this.peerconnection.onsignalingstatechange = function (event) {
  1245. if (!(obj && obj.peerconnection)) return;
  1246. console.log('signallingstate ', obj.peerconnection.signalingState, event);
  1247. };
  1248. this.peerconnection.oniceconnectionstatechange = function (event) {
  1249. if (!(obj && obj.peerconnection)) return;
  1250. console.log('iceconnectionstatechange', obj.peerconnection.iceConnectionState, event);
  1251. switch (obj.peerconnection.iceConnectionState) {
  1252. case 'connected':
  1253. this.startTime = new Date();
  1254. break;
  1255. case 'disconnected':
  1256. this.stopTime = new Date();
  1257. break;
  1258. }
  1259. $(document).trigger('iceconnectionstatechange.jingle', [obj.sid, obj]);
  1260. };
  1261. // add any local and relayed stream
  1262. this.localStreams.forEach(function(stream) {
  1263. obj.peerconnection.addStream(stream);
  1264. });
  1265. this.relayedStreams.forEach(function(stream) {
  1266. obj.peerconnection.addStream(stream);
  1267. });
  1268. };
  1269. JingleSession.prototype.accept = function () {
  1270. var ob = this;
  1271. this.state = 'active';
  1272. var pranswer = this.peerconnection.localDescription;
  1273. if (!pranswer || pranswer.type != 'pranswer') {
  1274. return;
  1275. }
  1276. console.log('going from pranswer to answer');
  1277. if (this.usetrickle) {
  1278. // remove candidates already sent from session-accept
  1279. var lines = SDPUtil.find_lines(pranswer.sdp, 'a=candidate:');
  1280. for (var i = 0; i < lines.length; i++) {
  1281. pranswer.sdp = pranswer.sdp.replace(lines[i] + '\r\n', '');
  1282. }
  1283. }
  1284. while (SDPUtil.find_line(pranswer.sdp, 'a=inactive')) {
  1285. // FIXME: change any inactive to sendrecv or whatever they were originally
  1286. pranswer.sdp = pranswer.sdp.replace('a=inactive', 'a=sendrecv');
  1287. }
  1288. var prsdp = new SDP(pranswer.sdp);
  1289. var accept = $iq({to: this.peerjid,
  1290. type: 'set'})
  1291. .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
  1292. action: 'session-accept',
  1293. initiator: this.initiator,
  1294. responder: this.responder,
  1295. sid: this.sid });
  1296. prsdp.toJingle(accept, this.initiator == this.me ? 'initiator' : 'responder');
  1297. this.connection.sendIQ(accept,
  1298. function () {
  1299. var ack = {};
  1300. ack.source = 'answer';
  1301. $(document).trigger('ack.jingle', [ob.sid, ack]);
  1302. },
  1303. function (stanza) {
  1304. var error = ($(stanza).find('error').length) ? {
  1305. code: $(stanza).find('error').attr('code'),
  1306. reason: $(stanza).find('error :first')[0].tagName,
  1307. }:{};
  1308. error.source = 'answer';
  1309. $(document).trigger('error.jingle', [ob.sid, error]);
  1310. },
  1311. 10000);
  1312. var sdp = this.peerconnection.localDescription.sdp;
  1313. while (SDPUtil.find_line(sdp, 'a=inactive')) {
  1314. // FIXME: change any inactive to sendrecv or whatever they were originally
  1315. sdp = sdp.replace('a=inactive', 'a=sendrecv');
  1316. }
  1317. this.peerconnection.setLocalDescription(new RTCSessionDescription({type: 'answer', sdp: sdp}),
  1318. function () {
  1319. console.log('setLocalDescription success');
  1320. },
  1321. function (e) {
  1322. console.error('setLocalDescription failed', e);
  1323. }
  1324. );
  1325. };
  1326. JingleSession.prototype.terminate = function (reason) {
  1327. this.state = 'ended';
  1328. this.reason = reason;
  1329. this.peerconnection.close();
  1330. if (this.statsinterval !== null) {
  1331. window.clearInterval(this.statsinterval);
  1332. this.statsinterval = null;
  1333. }
  1334. };
  1335. JingleSession.prototype.active = function () {
  1336. return this.state == 'active';
  1337. };
  1338. JingleSession.prototype.sendIceCandidate = function (candidate) {
  1339. var ob = this;
  1340. if (candidate && !this.lasticecandidate) {
  1341. var ice = SDPUtil.iceparams(this.localSDP.media[candidate.sdpMLineIndex], this.localSDP.session),
  1342. jcand = SDPUtil.candidateToJingle(candidate.candidate);
  1343. if (!(ice && jcand)) {
  1344. console.error('failed to get ice && jcand');
  1345. return;
  1346. }
  1347. ice.xmlns = 'urn:xmpp:jingle:transports:ice-udp:1';
  1348. if (jcand.type === 'srflx') {
  1349. this.hadstuncandidate = true;
  1350. } else if (jcand.type === 'relay') {
  1351. this.hadturncandidate = true;
  1352. }
  1353. console.log(event.candidate, jcand);
  1354. if (this.usetrickle) {
  1355. if (this.usedrip) {
  1356. if (this.drip_container.length === 0) {
  1357. // start 10ms callout
  1358. window.setTimeout(function () {
  1359. if (ob.drip_container.length === 0) return;
  1360. var allcands = ob.drip_container;
  1361. ob.drip_container = [];
  1362. var cand = $iq({to: ob.peerjid, type: 'set'})
  1363. .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
  1364. action: 'transport-info',
  1365. initiator: ob.initiator,
  1366. sid: ob.sid});
  1367. for (var mid = 0; mid < ob.localSDP.media.length; mid++) {
  1368. var cands = allcands.filter(function (el) { return el.sdpMLineIndex == mid; });
  1369. if (cands.length > 0) {
  1370. var ice = SDPUtil.iceparams(ob.localSDP.media[mid], ob.localSDP.session);
  1371. ice.xmlns = 'urn:xmpp:jingle:transports:ice-udp:1';
  1372. cand.c('content', {creator: ob.initiator == ob.me ? 'initiator' : 'responder',
  1373. name: cands[0].sdpMid
  1374. }).c('transport', ice);
  1375. for (var i = 0; i < cands.length; i++) {
  1376. cand.c('candidate', SDPUtil.candidateToJingle(cands[i].candidate)).up();
  1377. }
  1378. // add fingerprint
  1379. if (SDPUtil.find_line(ob.localSDP.media[mid], 'a=fingerprint:', ob.localSDP.session)) {
  1380. var tmp = SDPUtil.parse_fingerprint(SDPUtil.find_line(ob.localSDP.media[mid], 'a=fingerprint:', ob.localSDP.session));
  1381. tmp.required = true;
  1382. cand.c('fingerprint').t(tmp.fingerprint);
  1383. delete tmp.fingerprint;
  1384. cand.attrs(tmp);
  1385. cand.up();
  1386. }
  1387. cand.up(); // transport
  1388. cand.up(); // content
  1389. }
  1390. }
  1391. // might merge last-candidate notification into this, but it is called alot later. See webrtc issue #2340
  1392. //console.log('was this the last candidate', ob.lasticecandidate);
  1393. ob.connection.sendIQ(cand,
  1394. function () {
  1395. var ack = {};
  1396. ack.source = 'transportinfo';
  1397. $(document).trigger('ack.jingle', [ob.sid, ack]);
  1398. },
  1399. function (stanza) {
  1400. var error = ($(stanza).find('error').length) ? {
  1401. code: $(stanza).find('error').attr('code'),
  1402. reason: $(stanza).find('error :first')[0].tagName,
  1403. }:{};
  1404. error.source = 'transportinfo';
  1405. $(document).trigger('error.jingle', [ob.sid, error]);
  1406. },
  1407. 10000);
  1408. }, 10);
  1409. }
  1410. this.drip_container.push(event.candidate);
  1411. return;
  1412. }
  1413. // map to transport-info
  1414. var cand = $iq({to: this.peerjid, type: 'set'})
  1415. .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
  1416. action: 'transport-info',
  1417. initiator: this.initiator,
  1418. sid: this.sid})
  1419. .c('content', {creator: this.initiator == this.me ? 'initiator' : 'responder',
  1420. name: candidate.sdpMid
  1421. })
  1422. .c('transport', ice)
  1423. .c('candidate', jcand);
  1424. cand.up();
  1425. // add fingerprint
  1426. if (SDPUtil.find_line(this.localSDP.media[candidate.sdpMLineIndex], 'a=fingerprint:', this.localSDP.session)) {
  1427. var tmp = SDPUtil.parse_fingerprint(SDPUtil.find_line(this.localSDP.media[candidate.sdpMLineIndex], 'a=fingerprint:', this.localSDP.session));
  1428. tmp.required = true;
  1429. cand.c('fingerprint').t(tmp.fingerprint);
  1430. delete tmp.fingerprint;
  1431. cand.attrs(tmp);
  1432. cand.up();
  1433. }
  1434. this.connection.sendIQ(cand,
  1435. function () {
  1436. var ack = {};
  1437. ack.source = 'transportinfo';
  1438. $(document).trigger('ack.jingle', [ob.sid, ack]);
  1439. },
  1440. function (stanza) {
  1441. console.error('transport info error');
  1442. var error = ($(stanza).find('error').length) ? {
  1443. code: $(stanza).find('error').attr('code'),
  1444. reason: $(stanza).find('error :first')[0].tagName,
  1445. }:{};
  1446. error.source = 'transportinfo';
  1447. $(document).trigger('error.jingle', [ob.sid, error]);
  1448. },
  1449. 10000);
  1450. }
  1451. } else {
  1452. console.log('sendIceCandidate: last candidate.');
  1453. if (!this.usetrickle) {
  1454. console.log('should send full offer now...');
  1455. var init = $iq({to: this.peerjid,
  1456. type: 'set'})
  1457. .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
  1458. action: this.peerconnection.localDescription.type == 'offer' ? 'session-initiate' : 'session-accept',
  1459. initiator: this.initiator,
  1460. sid: this.sid});
  1461. this.localSDP = new SDP(this.peerconnection.localDescription.sdp);
  1462. this.localSDP.toJingle(init, this.initiator == this.me ? 'initiator' : 'responder');
  1463. this.connection.sendIQ(init,
  1464. function () {
  1465. console.log('session initiate ack');
  1466. var ack = {};
  1467. ack.source = 'offer';
  1468. $(document).trigger('ack.jingle', [ob.sid, ack]);
  1469. },
  1470. function (stanza) {
  1471. ob.state = 'error';
  1472. ob.peerconnection.close();
  1473. var error = ($(stanza).find('error').length) ? {
  1474. code: $(stanza).find('error').attr('code'),
  1475. reason: $(stanza).find('error :first')[0].tagName,
  1476. }:{};
  1477. error.source = 'offer';
  1478. $(document).trigger('error.jingle', [ob.sid, error]);
  1479. },
  1480. 10000);
  1481. }
  1482. this.lasticecandidate = true;
  1483. console.log('Have we encountered any srflx candidates? ' + this.hadstuncandidate);
  1484. console.log('Have we encountered any relay candidates? ' + this.hadturncandidate);
  1485. if (!(this.hadstuncandidate || this.hadturncandidate) && this.peerconnection.signalingState != 'closed') {
  1486. $(document).trigger('nostuncandidates.jingle', [this.sid]);
  1487. }
  1488. }
  1489. };
  1490. JingleSession.prototype.sendOffer = function () {
  1491. console.log('sendOffer...');
  1492. var ob = this;
  1493. this.peerconnection.createOffer(function (sdp) {
  1494. ob.createdOffer(sdp);
  1495. },
  1496. function (e) {
  1497. console.error('createOffer failed', e);
  1498. },
  1499. this.media_constraints
  1500. );
  1501. };
  1502. JingleSession.prototype.createdOffer = function (sdp) {
  1503. console.log('createdOffer', sdp);
  1504. var ob = this;
  1505. this.localSDP = new SDP(sdp.sdp);
  1506. //this.localSDP.mangle();
  1507. if (this.usetrickle) {
  1508. var init = $iq({to: this.peerjid,
  1509. type: 'set'})
  1510. .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
  1511. action: 'session-initiate',
  1512. initiator: this.initiator,
  1513. sid: this.sid});
  1514. this.localSDP.toJingle(init, this.initiator == this.me ? 'initiator' : 'responder');
  1515. this.connection.sendIQ(init,
  1516. function () {
  1517. var ack = {};
  1518. ack.source = 'offer';
  1519. $(document).trigger('ack.jingle', [ob.sid, ack]);
  1520. },
  1521. function (stanza) {
  1522. ob.state = 'error';
  1523. ob.peerconnection.close();
  1524. var error = ($(stanza).find('error').length) ? {
  1525. code: $(stanza).find('error').attr('code'),
  1526. reason: $(stanza).find('error :first')[0].tagName,
  1527. }:{};
  1528. error.source = 'offer';
  1529. $(document).trigger('error.jingle', [ob.sid, error]);
  1530. },
  1531. 10000);
  1532. }
  1533. sdp.sdp = this.localSDP.raw;
  1534. this.peerconnection.setLocalDescription(sdp, function () {
  1535. console.log('setLocalDescription success');
  1536. },
  1537. function (e) {
  1538. console.error('setLocalDescription failed', e);
  1539. }
  1540. );
  1541. var cands = SDPUtil.find_lines(this.localSDP.raw, 'a=candidate:');
  1542. for (var i = 0; i < cands.length; i++) {
  1543. var cand = SDPUtil.parse_icecandidate(cands[i]);
  1544. if (cand.type == 'srflx') {
  1545. this.hadstuncandidate = true;
  1546. } else if (cand.type == 'relay') {
  1547. this.hadturncandidate = true;
  1548. }
  1549. }
  1550. };
  1551. JingleSession.prototype.setRemoteDescription = function (elem, desctype) {
  1552. console.log('setting remote description... ', desctype);
  1553. this.remoteSDP = new SDP('');
  1554. this.remoteSDP.fromJingle(elem);
  1555. if (this.peerconnection.remoteDescription !== null) {
  1556. console.log('setRemoteDescription when remote description is not null, should be pranswer', this.peerconnection.remoteDescription);
  1557. if (this.peerconnection.remoteDescription.type == 'pranswer') {
  1558. var pranswer = new SDP(this.peerconnection.remoteDescription.sdp);
  1559. for (var i = 0; i < pranswer.media.length; i++) {
  1560. // make sure we have ice ufrag and pwd
  1561. if (!SDPUtil.find_line(this.remoteSDP.media[i], 'a=ice-ufrag:', this.remoteSDP.session)) {
  1562. if (SDPUtil.find_line(pranswer.media[i], 'a=ice-ufrag:', pranswer.session)) {
  1563. this.remoteSDP.media[i] += SDPUtil.find_line(pranswer.media[i], 'a=ice-ufrag:', pranswer.session) + '\r\n';
  1564. } else {
  1565. console.warn('no ice ufrag?');
  1566. }
  1567. if (SDPUtil.find_line(pranswer.media[i], 'a=ice-pwd:', pranswer.session)) {
  1568. this.remoteSDP.media[i] += SDPUtil.find_line(pranswer.media[i], 'a=ice-pwd:', pranswer.session) + '\r\n';
  1569. } else {
  1570. console.warn('no ice pwd?');
  1571. }
  1572. }
  1573. // copy over candidates
  1574. var lines = SDPUtil.find_lines(pranswer.media[i], 'a=candidate:');
  1575. for (var j = 0; j < lines.length; j++) {
  1576. this.remoteSDP.media[i] += lines[j] + '\r\n';
  1577. }
  1578. }
  1579. this.remoteSDP.raw = this.remoteSDP.session + this.remoteSDP.media.join('');
  1580. }
  1581. }
  1582. var remotedesc = new RTCSessionDescription({type: desctype, sdp: this.remoteSDP.raw});
  1583. this.peerconnection.setRemoteDescription(remotedesc,
  1584. function () {
  1585. console.log('setRemoteDescription success');
  1586. },
  1587. function (e) {
  1588. console.error('setRemoteDescription error', e);
  1589. }
  1590. );
  1591. };
  1592. JingleSession.prototype.addIceCandidate = function (elem) {
  1593. var obj = this;
  1594. if (this.peerconnection.signalingState == 'closed') {
  1595. return;
  1596. }
  1597. if (!this.peerconnection.remoteDescription && this.peerconnection.signalingState == 'have-local-offer') {
  1598. console.log('trickle ice candidate arriving before session accept...');
  1599. // create a PRANSWER for setRemoteDescription
  1600. if (!this.remoteSDP) {
  1601. var cobbled = 'v=0\r\n' +
  1602. 'o=- ' + '1923518516' + ' 2 IN IP4 0.0.0.0\r\n' +// FIXME
  1603. 's=-\r\n' +
  1604. 't=0 0\r\n';
  1605. // first, take some things from the local description
  1606. for (var i = 0; i < this.localSDP.media.length; i++) {
  1607. cobbled += SDPUtil.find_line(this.localSDP.media[i], 'm=') + '\r\n';
  1608. cobbled += SDPUtil.find_lines(this.localSDP.media[i], 'a=rtpmap:').join('\r\n') + '\r\n';
  1609. if (SDPUtil.find_line(this.localSDP.media[i], 'a=mid:')) {
  1610. cobbled += SDPUtil.find_line(this.localSDP.media[i], 'a=mid:') + '\r\n';
  1611. }
  1612. cobbled += 'a=inactive\r\n';
  1613. }
  1614. this.remoteSDP = new SDP(cobbled);
  1615. }
  1616. // then add things like ice and dtls from remote candidate
  1617. elem.each(function () {
  1618. for (var i = 0; i < obj.remoteSDP.media.length; i++) {
  1619. if (SDPUtil.find_line(obj.remoteSDP.media[i], 'a=mid:' + $(this).attr('name')) ||
  1620. obj.remoteSDP.media[i].indexOf('m=' + $(this).attr('name')) === 0) {
  1621. if (!SDPUtil.find_line(obj.remoteSDP.media[i], 'a=ice-ufrag:')) {
  1622. var tmp = $(this).find('transport');
  1623. obj.remoteSDP.media[i] += 'a=ice-ufrag:' + tmp.attr('ufrag') + '\r\n';
  1624. obj.remoteSDP.media[i] += 'a=ice-pwd:' + tmp.attr('pwd') + '\r\n';
  1625. tmp = $(this).find('transport>fingerprint');
  1626. if (tmp.length) {
  1627. obj.remoteSDP.media[i] += 'a=fingerprint:' + tmp.attr('hash') + ' ' + tmp.text() + '\r\n';
  1628. } else {
  1629. console.log('no dtls fingerprint (webrtc issue #1718?)');
  1630. obj.remoteSDP.media[i] += 'a=crypto:1 AES_CM_128_HMAC_SHA1_80 inline:BAADBAADBAADBAADBAADBAADBAADBAADBAADBAAD\r\n';
  1631. }
  1632. break;
  1633. }
  1634. }
  1635. }
  1636. });
  1637. this.remoteSDP.raw = this.remoteSDP.session + this.remoteSDP.media.join('');
  1638. // we need a complete SDP with ice-ufrag/ice-pwd in all parts
  1639. // this makes the assumption that the PRANSWER is constructed such that the ice-ufrag is in all mediaparts
  1640. // but it could be in the session part as well. since the code above constructs this sdp this can't happen however
  1641. var iscomplete = this.remoteSDP.media.filter(function (mediapart) {
  1642. return SDPUtil.find_line(mediapart, 'a=ice-ufrag:');
  1643. }).length == this.remoteSDP.media.length;
  1644. if (iscomplete) {
  1645. console.log('setting pranswer');
  1646. try {
  1647. this.peerconnection.setRemoteDescription(new RTCSessionDescription({type: 'pranswer', sdp: this.remoteSDP.raw }));
  1648. } catch (e) {
  1649. console.error('setting pranswer failed', e);
  1650. }
  1651. } else {
  1652. console.log('not yet setting pranswer');
  1653. }
  1654. }
  1655. // operate on each content element
  1656. elem.each(function () {
  1657. // would love to deactivate this, but firefox still requires it
  1658. var idx = -1;
  1659. var i;
  1660. for (i = 0; i < obj.remoteSDP.media.length; i++) {
  1661. if (SDPUtil.find_line(obj.remoteSDP.media[i], 'a=mid:' + $(this).attr('name')) ||
  1662. obj.remoteSDP.media[i].indexOf('m=' + $(this).attr('name')) === 0) {
  1663. idx = i;
  1664. break;
  1665. }
  1666. }
  1667. if (idx == -1) { // fall back to localdescription
  1668. for (i = 0; i < obj.localSDP.media.length; i++) {
  1669. if (SDPUtil.find_line(obj.localSDP.media[i], 'a=mid:' + $(this).attr('name')) ||
  1670. obj.localSDP.media[i].indexOf('m=' + $(this).attr('name')) === 0) {
  1671. idx = i;
  1672. break;
  1673. }
  1674. }
  1675. }
  1676. var name = $(this).attr('name');
  1677. // TODO: check ice-pwd and ice-ufrag?
  1678. $(this).find('transport>candidate').each(function () {
  1679. var line, candidate;
  1680. line = SDPUtil.candidateFromJingle(this);
  1681. candidate = new RTCIceCandidate({sdpMLineIndex: idx,
  1682. sdpMid: name,
  1683. candidate: line});
  1684. try {
  1685. obj.peerconnection.addIceCandidate(candidate);
  1686. } catch (e) {
  1687. console.error('addIceCandidate failed', e.toString(), line);
  1688. }
  1689. });
  1690. });
  1691. };
  1692. JingleSession.prototype.sendAnswer = function (provisional) {
  1693. console.log('createAnswer', provisional);
  1694. var ob = this;
  1695. this.peerconnection.createAnswer(
  1696. function (sdp) {
  1697. ob.createdAnswer(sdp, provisional);
  1698. },
  1699. function (e) {
  1700. console.error('createAnswer failed', e);
  1701. },
  1702. this.media_constraints
  1703. );
  1704. };
  1705. JingleSession.prototype.createdAnswer = function (sdp, provisional) {
  1706. console.log('createAnswer callback');
  1707. console.log(sdp);
  1708. var ob = this;
  1709. this.localSDP = new SDP(sdp.sdp);
  1710. //this.localSDP.mangle();
  1711. this.usepranswer = provisional === true;
  1712. if (this.usetrickle) {
  1713. if (!this.usepranswer) {
  1714. var accept = $iq({to: this.peerjid,
  1715. type: 'set'})
  1716. .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
  1717. action: 'session-accept',
  1718. initiator: this.initiator,
  1719. responder: this.responder,
  1720. sid: this.sid });
  1721. this.localSDP.toJingle(accept, this.initiator == this.me ? 'initiator' : 'responder');
  1722. this.connection.sendIQ(accept,
  1723. function () {
  1724. var ack = {};
  1725. ack.source = 'answer';
  1726. $(document).trigger('ack.jingle', [ob.sid, ack]);
  1727. },
  1728. function (stanza) {
  1729. var error = ($(stanza).find('error').length) ? {
  1730. code: $(stanza).find('error').attr('code'),
  1731. reason: $(stanza).find('error :first')[0].tagName,
  1732. }:{};
  1733. error.source = 'answer';
  1734. $(document).trigger('error.jingle', [ob.sid, error]);
  1735. },
  1736. 10000);
  1737. } else {
  1738. sdp.type = 'pranswer';
  1739. for (var i = 0; i < this.localSDP.media.length; i++) {
  1740. this.localSDP.media[i] = this.localSDP.media[i].replace('a=sendrecv\r\n', 'a=inactive\r\n');
  1741. }
  1742. this.localSDP.raw = this.localSDP.session + '\r\n' + this.localSDP.media.join('');
  1743. }
  1744. }
  1745. sdp.sdp = this.localSDP.raw;
  1746. this.peerconnection.setLocalDescription(sdp,
  1747. function () {
  1748. console.log('setLocalDescription success');
  1749. },
  1750. function (e) {
  1751. console.error('setLocalDescription failed', e);
  1752. }
  1753. );
  1754. var cands = SDPUtil.find_lines(this.localSDP.raw, 'a=candidate:');
  1755. for (var j = 0; j < cands.length; j++) {
  1756. var cand = SDPUtil.parse_icecandidate(cands[j]);
  1757. if (cand.type == 'srflx') {
  1758. this.hadstuncandidate = true;
  1759. } else if (cand.type == 'relay') {
  1760. this.hadturncandidate = true;
  1761. }
  1762. }
  1763. };
  1764. JingleSession.prototype.sendTerminate = function (reason, text) {
  1765. var obj = this,
  1766. term = $iq({to: this.peerjid,
  1767. type: 'set'})
  1768. .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
  1769. action: 'session-terminate',
  1770. initiator: this.initiator,
  1771. sid: this.sid})
  1772. .c('reason')
  1773. .c(reason || 'success');
  1774. if (text) {
  1775. term.up().c('text').t(text);
  1776. }
  1777. this.connection.sendIQ(term,
  1778. function () {
  1779. obj.peerconnection.close();
  1780. obj.peerconnection = null;
  1781. obj.terminate();
  1782. var ack = {};
  1783. ack.source = 'terminate';
  1784. $(document).trigger('ack.jingle', [obj.sid, ack]);
  1785. },
  1786. function (stanza) {
  1787. var error = ($(stanza).find('error').length) ? {
  1788. code: $(stanza).find('error').attr('code'),
  1789. reason: $(stanza).find('error :first')[0].tagName,
  1790. }:{};
  1791. $(document).trigger('ack.jingle', [obj.sid, error]);
  1792. },
  1793. 10000);
  1794. if (this.statsinterval !== null) {
  1795. window.clearInterval(this.statsinterval);
  1796. this.statsinterval = null;
  1797. }
  1798. };
  1799. JingleSession.prototype.addSource = function (elem) {
  1800. console.log('addssrc', new Date().getTime());
  1801. console.log('ice', this.peerconnection.iceConnectionState);
  1802. var sdp = new SDP(this.peerconnection.remoteDescription.sdp);
  1803. var ob = this;
  1804. $(elem).each(function (idx, content) {
  1805. var name = $(content).attr('name');
  1806. var lines = '';
  1807. tmp = $(content).find('>source[xmlns="urn:xmpp:jingle:apps:rtp:ssma:0"]');
  1808. tmp.each(function () {
  1809. var ssrc = $(this).attr('ssrc');
  1810. $(this).find('>parameter').each(function () {
  1811. lines += 'a=ssrc:' + ssrc + ' ' + $(this).attr('name');
  1812. if ($(this).attr('value') && $(this).attr('value').length)
  1813. lines += ':' + $(this).attr('value');
  1814. lines += '\r\n';
  1815. });
  1816. });
  1817. console.log(name, lines);
  1818. sdp.media.forEach(function(media, idx) {
  1819. if (!SDPUtil.find_line(media, 'a=mid:' + name))
  1820. return;
  1821. sdp.media[idx] += lines;
  1822. if (!ob.addssrc[idx]) ob.addssrc[idx] = '';
  1823. ob.addssrc[idx] += lines;
  1824. });
  1825. sdp.raw = sdp.session + sdp.media.join('');
  1826. });
  1827. this.modifySources();
  1828. };
  1829. JingleSession.prototype.removeSource = function (elem) {
  1830. console.log('removessrc', new Date().getTime());
  1831. console.log('ice', this.peerconnection.iceConnectionState);
  1832. var sdp = new SDP(this.peerconnection.remoteDescription.sdp);
  1833. var ob = this;
  1834. $(elem).each(function (idx, content) {
  1835. var name = $(content).attr('name');
  1836. var lines = '';
  1837. tmp = $(content).find('>source[xmlns="urn:xmpp:jingle:apps:rtp:ssma:0"]');
  1838. tmp.each(function () {
  1839. var ssrc = $(this).attr('ssrc');
  1840. $(this).find('>parameter').each(function () {
  1841. lines += 'a=ssrc:' + ssrc + ' ' + $(this).attr('name');
  1842. if ($(this).attr('value') && $(this).attr('value').length)
  1843. lines += ':' + $(this).attr('value');
  1844. lines += '\r\n';
  1845. });
  1846. });
  1847. console.log(name, lines);
  1848. sdp.media.forEach(function(media, idx) {
  1849. if (!SDPUtil.find_line(media, 'a=mid:' + name))
  1850. return;
  1851. sdp.media[idx] += lines;
  1852. if (!ob.addssrc[idx]) ob.removessrc[idx] = '';
  1853. ob.removessrc[idx] += lines;
  1854. });
  1855. sdp.raw = sdp.session + sdp.media.join('');
  1856. });
  1857. console.log(this.removessrc);
  1858. this.modifySources();
  1859. };
  1860. JingleSession.prototype.modifySources = function() {
  1861. var ob = this;
  1862. if (!(this.addssrc.length || this.removessrc.length)) return;
  1863. if (this.peerconnection.signalingState == 'closed') return;
  1864. if (!(this.peerconnection.signalingState == 'stable' && this.peerconnection.iceConnectionState == 'connected')) {
  1865. console.warn('modifySources not yet', this.peerconnection.signalingState, this.peerconnection.iceConnectionState);
  1866. this.wait = true;
  1867. window.setTimeout(function() { ob.modifySources(); }, 250);
  1868. return;
  1869. }
  1870. if (this.wait) {
  1871. window.setTimeout(function() { ob.modifySources(); }, 2500);
  1872. this.wait = false;
  1873. return;
  1874. }
  1875. console.log('ice', this.peerconnection.iceConnectionState);
  1876. var sdp = new SDP(this.peerconnection.remoteDescription.sdp);
  1877. // add sources
  1878. this.addssrc.forEach(function(lines, idx) {
  1879. sdp.media[idx] += lines;
  1880. });
  1881. this.addssrc = [];
  1882. // remove sources
  1883. this.removessrc.forEach(function(lines, idx) {
  1884. lines = lines.split('\r\n');
  1885. lines.pop(); // remove empty last element;
  1886. lines.forEach(function(line) {
  1887. sdp.media[idx] = sdp.media[idx].replace(line + '\r\n', '');
  1888. });
  1889. });
  1890. this.removessrc = [];
  1891. sdp.raw = sdp.session + sdp.media.join('');
  1892. this.peerconnection.setRemoteDescription(new RTCSessionDescription({type: 'offer', sdp: sdp.raw}),
  1893. function() {
  1894. console.log('modify ok');
  1895. ob.peerconnection.createAnswer(
  1896. function(modifiedAnswer) {
  1897. console.log('modified answer...');
  1898. ob.peerconnection.setLocalDescription(modifiedAnswer,
  1899. function() {
  1900. console.log('modified setLocalDescription ok');
  1901. },
  1902. function(error) {
  1903. console.log('modified setLocalDescription failed');
  1904. }
  1905. );
  1906. },
  1907. function(error) {
  1908. console.log('modified answer failed');
  1909. }
  1910. );
  1911. },
  1912. function(error) {
  1913. console.log('modify failed');
  1914. }
  1915. );
  1916. };
  1917. JingleSession.prototype.sendMute = function (muted, content) {
  1918. var info = $iq({to: this.peerjid,
  1919. type: 'set'})
  1920. .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
  1921. action: 'session-info',
  1922. initiator: this.initiator,
  1923. sid: this.sid });
  1924. info.c(muted ? 'mute' : 'unmute', {xmlns: 'urn:xmpp:jingle:apps:rtp:info:1'});
  1925. info.attrs({'creator': this.me == this.initiator ? 'creator' : 'responder'});
  1926. if (content) {
  1927. info.attrs({'name': content});
  1928. }
  1929. this.connection.send(info);
  1930. };
  1931. JingleSession.prototype.sendRinging = function () {
  1932. var info = $iq({to: this.peerjid,
  1933. type: 'set'})
  1934. .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
  1935. action: 'session-info',
  1936. initiator: this.initiator,
  1937. sid: this.sid });
  1938. info.c('ringing', {xmlns: 'urn:xmpp:jingle:apps:rtp:info:1'});
  1939. this.connection.send(info);
  1940. };
  1941. JingleSession.prototype.getStats = function (interval) {
  1942. var ob = this;
  1943. var recv = {audio: 0, video: 0};
  1944. var lost = {audio: 0, video: 0};
  1945. var lastrecv = {audio: 0, video: 0};
  1946. var lastlost = {audio: 0, video: 0};
  1947. var loss = {audio: 0, video: 0};
  1948. var delta = {audio: 0, video: 0};
  1949. this.statsinterval = window.setInterval(function () {
  1950. if (ob && ob.peerconnection && ob.peerconnection.getStats) {
  1951. ob.peerconnection.getStats(function (stats) {
  1952. var results = stats.result();
  1953. // TODO: there are so much statistics you can get from this..
  1954. for (var i = 0; i < results.length; ++i) {
  1955. if (results[i].type == 'ssrc') {
  1956. var packetsrecv = results[i].stat('packetsReceived');
  1957. var packetslost = results[i].stat('packetsLost');
  1958. if (packetsrecv && packetslost) {
  1959. packetsrecv = parseInt(packetsrecv, 10);
  1960. packetslost = parseInt(packetslost, 10);
  1961. if (results[i].stat('googFrameRateReceived')) {
  1962. lastlost.video = lost.video;
  1963. lastrecv.video = recv.video;
  1964. recv.video = packetsrecv;
  1965. lost.video = packetslost;
  1966. } else {
  1967. lastlost.audio = lost.audio;
  1968. lastrecv.audio = recv.audio;
  1969. recv.audio = packetsrecv;
  1970. lost.audio = packetslost;
  1971. }
  1972. }
  1973. }
  1974. }
  1975. delta.audio = recv.audio - lastrecv.audio;
  1976. delta.video = recv.video - lastrecv.video;
  1977. loss.audio = (delta.audio > 0) ? Math.ceil(100 * (lost.audio - lastlost.audio) / delta.audio) : 0;
  1978. loss.video = (delta.video > 0) ? Math.ceil(100 * (lost.video - lastlost.video) / delta.video) : 0;
  1979. $(document).trigger('packetloss.jingle', [ob.sid, loss]);
  1980. });
  1981. }
  1982. }, interval || 3000);
  1983. return this.statsinterval;
  1984. };