您最多选择25个主题 主题必须以字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符

xmpp.bundle.js 476KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551455245534554455545564557455845594560456145624563456445654566456745684569457045714572457345744575457645774578457945804581458245834584458545864587458845894590459145924593459445954596459745984599460046014602460346044605460646074608460946104611461246134614461546164617461846194620462146224623462446254626462746284629463046314632463346344635463646374638463946404641464246434644464546464647464846494650465146524653465446554656465746584659466046614662466346644665466646674668466946704671467246734674467546764677467846794680468146824683468446854686468746884689469046914692469346944695469646974698469947004701470247034704470547064707470847094710471147124713471447154716471747184719472047214722472347244725472647274728472947304731473247334734473547364737473847394740474147424743474447454746474747484749475047514752475347544755475647574758475947604761476247634764476547664767476847694770477147724773477447754776477747784779478047814782478347844785478647874788478947904791479247934794479547964797479847994800480148024803480448054806480748084809481048114812481348144815481648174818481948204821482248234824482548264827482848294830483148324833483448354836483748384839484048414842484348444845484648474848484948504851485248534854485548564857485848594860486148624863486448654866486748684869487048714872487348744875487648774878487948804881488248834884488548864887488848894890489148924893489448954896489748984899490049014902490349044905490649074908490949104911491249134914491549164917491849194920492149224923492449254926492749284929493049314932493349344935493649374938493949404941494249434944494549464947494849494950495149524953495449554956495749584959496049614962496349644965496649674968496949704971497249734974497549764977497849794980498149824983498449854986498749884989499049914992499349944995499649974998499950005001500250035004500550065007500850095010501150125013501450155016501750185019502050215022502350245025502650275028502950305031503250335034503550365037503850395040504150425043504450455046504750485049505050515052505350545055505650575058505950605061506250635064506550665067506850695070507150725073507450755076507750785079508050815082
  1. !function(e){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=e();else if("function"==typeof define&&define.amd)define([],e);else{var f;"undefined"!=typeof window?f=window:"undefined"!=typeof global?f=global:"undefined"!=typeof self&&(f=self),f.xmpp=e()}}(function(){var define,module,exports;return (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){
  2. /* jshint -W117 */
  3. var TraceablePeerConnection = require("./TraceablePeerConnection");
  4. var SDPDiffer = require("./SDPDiffer");
  5. var SDPUtil = require("./SDPUtil");
  6. var SDP = require("./SDP");
  7. // Jingle stuff
  8. function JingleSession(me, sid, connection, service) {
  9. this.me = me;
  10. this.sid = sid;
  11. this.connection = connection;
  12. this.initiator = null;
  13. this.responder = null;
  14. this.isInitiator = null;
  15. this.peerjid = null;
  16. this.state = null;
  17. this.localSDP = null;
  18. this.remoteSDP = null;
  19. this.relayedStreams = [];
  20. this.startTime = null;
  21. this.stopTime = null;
  22. this.media_constraints = null;
  23. this.pc_constraints = null;
  24. this.ice_config = {};
  25. this.drip_container = [];
  26. this.service = service;
  27. this.usetrickle = true;
  28. this.usepranswer = false; // early transport warmup -- mind you, this might fail. depends on webrtc issue 1718
  29. this.usedrip = false; // dripping is sending trickle candidates not one-by-one
  30. this.hadstuncandidate = false;
  31. this.hadturncandidate = false;
  32. this.lasticecandidate = false;
  33. this.statsinterval = null;
  34. this.reason = null;
  35. this.addssrc = [];
  36. this.removessrc = [];
  37. this.pendingop = null;
  38. this.switchstreams = false;
  39. this.wait = true;
  40. this.localStreamsSSRC = null;
  41. /**
  42. * The indicator which determines whether the (local) video has been muted
  43. * in response to a user command in contrast to an automatic decision made
  44. * by the application logic.
  45. */
  46. this.videoMuteByUser = false;
  47. }
  48. JingleSession.prototype.initiate = function (peerjid, isInitiator) {
  49. var self = this;
  50. if (this.state !== null) {
  51. console.error('attempt to initiate on session ' + this.sid +
  52. 'in state ' + this.state);
  53. return;
  54. }
  55. this.isInitiator = isInitiator;
  56. this.state = 'pending';
  57. this.initiator = isInitiator ? this.me : peerjid;
  58. this.responder = !isInitiator ? this.me : peerjid;
  59. this.peerjid = peerjid;
  60. this.hadstuncandidate = false;
  61. this.hadturncandidate = false;
  62. this.lasticecandidate = false;
  63. this.peerconnection
  64. = new TraceablePeerConnection(
  65. this.connection.jingle.ice_config,
  66. this.connection.jingle.pc_constraints );
  67. this.peerconnection.onicecandidate = function (event) {
  68. self.sendIceCandidate(event.candidate);
  69. };
  70. this.peerconnection.onaddstream = function (event) {
  71. console.log("REMOTE STREAM ADDED: " + event.stream + " - " + event.stream.id);
  72. self.remoteStreamAdded(event);
  73. };
  74. this.peerconnection.onremovestream = function (event) {
  75. // Remove the stream from remoteStreams
  76. // FIXME: remotestreamremoved.jingle not defined anywhere(unused)
  77. $(document).trigger('remotestreamremoved.jingle', [event, self.sid]);
  78. };
  79. this.peerconnection.onsignalingstatechange = function (event) {
  80. if (!(self && self.peerconnection)) return;
  81. };
  82. this.peerconnection.oniceconnectionstatechange = function (event) {
  83. if (!(self && self.peerconnection)) return;
  84. switch (self.peerconnection.iceConnectionState) {
  85. case 'connected':
  86. this.startTime = new Date();
  87. break;
  88. case 'disconnected':
  89. this.stopTime = new Date();
  90. break;
  91. }
  92. onIceConnectionStateChange(self.sid, self);
  93. };
  94. // add any local and relayed stream
  95. RTC.localStreams.forEach(function(stream) {
  96. self.peerconnection.addStream(stream.getOriginalStream());
  97. });
  98. this.relayedStreams.forEach(function(stream) {
  99. self.peerconnection.addStream(stream);
  100. });
  101. };
  102. function onIceConnectionStateChange(sid, session) {
  103. switch (session.peerconnection.iceConnectionState) {
  104. case 'checking':
  105. session.timeChecking = (new Date()).getTime();
  106. session.firstconnect = true;
  107. break;
  108. case 'completed': // on caller side
  109. case 'connected':
  110. if (session.firstconnect) {
  111. session.firstconnect = false;
  112. var metadata = {};
  113. metadata.setupTime
  114. = (new Date()).getTime() - session.timeChecking;
  115. session.peerconnection.getStats(function (res) {
  116. if(res && res.result) {
  117. res.result().forEach(function (report) {
  118. if (report.type == 'googCandidatePair' &&
  119. report.stat('googActiveConnection') == 'true') {
  120. metadata.localCandidateType
  121. = report.stat('googLocalCandidateType');
  122. metadata.remoteCandidateType
  123. = report.stat('googRemoteCandidateType');
  124. // log pair as well so we can get nice pie
  125. // charts
  126. metadata.candidatePair
  127. = report.stat('googLocalCandidateType') +
  128. ';' +
  129. report.stat('googRemoteCandidateType');
  130. if (report.stat('googRemoteAddress').indexOf('[') === 0)
  131. {
  132. metadata.ipv6 = true;
  133. }
  134. }
  135. });
  136. }
  137. });
  138. }
  139. break;
  140. }
  141. }
  142. JingleSession.prototype.accept = function () {
  143. var self = this;
  144. this.state = 'active';
  145. var pranswer = this.peerconnection.localDescription;
  146. if (!pranswer || pranswer.type != 'pranswer') {
  147. return;
  148. }
  149. console.log('going from pranswer to answer');
  150. if (this.usetrickle) {
  151. // remove candidates already sent from session-accept
  152. var lines = SDPUtil.find_lines(pranswer.sdp, 'a=candidate:');
  153. for (var i = 0; i < lines.length; i++) {
  154. pranswer.sdp = pranswer.sdp.replace(lines[i] + '\r\n', '');
  155. }
  156. }
  157. while (SDPUtil.find_line(pranswer.sdp, 'a=inactive')) {
  158. // FIXME: change any inactive to sendrecv or whatever they were originally
  159. pranswer.sdp = pranswer.sdp.replace('a=inactive', 'a=sendrecv');
  160. }
  161. pranswer = simulcast.reverseTransformLocalDescription(pranswer);
  162. var prsdp = new SDP(pranswer.sdp);
  163. var accept = $iq({to: this.peerjid,
  164. type: 'set'})
  165. .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
  166. action: 'session-accept',
  167. initiator: this.initiator,
  168. responder: this.responder,
  169. sid: this.sid });
  170. prsdp.toJingle(accept, this.initiator == this.me ? 'initiator' : 'responder', this.localStreamsSSRC);
  171. var sdp = this.peerconnection.localDescription.sdp;
  172. while (SDPUtil.find_line(sdp, 'a=inactive')) {
  173. // FIXME: change any inactive to sendrecv or whatever they were originally
  174. sdp = sdp.replace('a=inactive', 'a=sendrecv');
  175. }
  176. var self = this;
  177. this.peerconnection.setLocalDescription(new RTCSessionDescription({type: 'answer', sdp: sdp}),
  178. function () {
  179. //console.log('setLocalDescription success');
  180. self.setLocalDescription();
  181. self.connection.sendIQ(accept,
  182. function () {
  183. var ack = {};
  184. ack.source = 'answer';
  185. $(document).trigger('ack.jingle', [self.sid, ack]);
  186. },
  187. function (stanza) {
  188. var error = ($(stanza).find('error').length) ? {
  189. code: $(stanza).find('error').attr('code'),
  190. reason: $(stanza).find('error :first')[0].tagName
  191. }:{};
  192. error.source = 'answer';
  193. JingleSession.onJingleError(self.sid, error);
  194. },
  195. 10000);
  196. },
  197. function (e) {
  198. console.error('setLocalDescription failed', e);
  199. }
  200. );
  201. };
  202. JingleSession.prototype.terminate = function (reason) {
  203. this.state = 'ended';
  204. this.reason = reason;
  205. this.peerconnection.close();
  206. if (this.statsinterval !== null) {
  207. window.clearInterval(this.statsinterval);
  208. this.statsinterval = null;
  209. }
  210. };
  211. JingleSession.prototype.active = function () {
  212. return this.state == 'active';
  213. };
  214. JingleSession.prototype.sendIceCandidate = function (candidate) {
  215. var self = this;
  216. if (candidate && !this.lasticecandidate) {
  217. var ice = SDPUtil.iceparams(this.localSDP.media[candidate.sdpMLineIndex], this.localSDP.session);
  218. var jcand = SDPUtil.candidateToJingle(candidate.candidate);
  219. if (!(ice && jcand)) {
  220. console.error('failed to get ice && jcand');
  221. return;
  222. }
  223. ice.xmlns = 'urn:xmpp:jingle:transports:ice-udp:1';
  224. if (jcand.type === 'srflx') {
  225. this.hadstuncandidate = true;
  226. } else if (jcand.type === 'relay') {
  227. this.hadturncandidate = true;
  228. }
  229. if (this.usetrickle) {
  230. if (this.usedrip) {
  231. if (this.drip_container.length === 0) {
  232. // start 20ms callout
  233. window.setTimeout(function () {
  234. if (self.drip_container.length === 0) return;
  235. self.sendIceCandidates(self.drip_container);
  236. self.drip_container = [];
  237. }, 20);
  238. }
  239. this.drip_container.push(candidate);
  240. return;
  241. } else {
  242. self.sendIceCandidate([candidate]);
  243. }
  244. }
  245. } else {
  246. //console.log('sendIceCandidate: last candidate.');
  247. if (!this.usetrickle) {
  248. //console.log('should send full offer now...');
  249. var init = $iq({to: this.peerjid,
  250. type: 'set'})
  251. .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
  252. action: this.peerconnection.localDescription.type == 'offer' ? 'session-initiate' : 'session-accept',
  253. initiator: this.initiator,
  254. sid: this.sid});
  255. this.localSDP = new SDP(this.peerconnection.localDescription.sdp);
  256. var self = this;
  257. var sendJingle = function (ssrc) {
  258. if(!ssrc)
  259. ssrc = {};
  260. self.localSDP.toJingle(init, self.initiator == self.me ? 'initiator' : 'responder', ssrc);
  261. self.connection.sendIQ(init,
  262. function () {
  263. //console.log('session initiate ack');
  264. var ack = {};
  265. ack.source = 'offer';
  266. $(document).trigger('ack.jingle', [self.sid, ack]);
  267. },
  268. function (stanza) {
  269. self.state = 'error';
  270. self.peerconnection.close();
  271. var error = ($(stanza).find('error').length) ? {
  272. code: $(stanza).find('error').attr('code'),
  273. reason: $(stanza).find('error :first')[0].tagName,
  274. }:{};
  275. error.source = 'offer';
  276. JingleSession.onJingleError(self.sid, error);
  277. },
  278. 10000);
  279. }
  280. sendJingle();
  281. }
  282. this.lasticecandidate = true;
  283. console.log('Have we encountered any srflx candidates? ' + this.hadstuncandidate);
  284. console.log('Have we encountered any relay candidates? ' + this.hadturncandidate);
  285. if (!(this.hadstuncandidate || this.hadturncandidate) && this.peerconnection.signalingState != 'closed') {
  286. $(document).trigger('nostuncandidates.jingle', [this.sid]);
  287. }
  288. }
  289. };
  290. JingleSession.prototype.sendIceCandidates = function (candidates) {
  291. console.log('sendIceCandidates', candidates);
  292. var cand = $iq({to: this.peerjid, type: 'set'})
  293. .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
  294. action: 'transport-info',
  295. initiator: this.initiator,
  296. sid: this.sid});
  297. for (var mid = 0; mid < this.localSDP.media.length; mid++) {
  298. var cands = candidates.filter(function (el) { return el.sdpMLineIndex == mid; });
  299. var mline = SDPUtil.parse_mline(this.localSDP.media[mid].split('\r\n')[0]);
  300. if (cands.length > 0) {
  301. var ice = SDPUtil.iceparams(this.localSDP.media[mid], this.localSDP.session);
  302. ice.xmlns = 'urn:xmpp:jingle:transports:ice-udp:1';
  303. cand.c('content', {creator: this.initiator == this.me ? 'initiator' : 'responder',
  304. name: (cands[0].sdpMid? cands[0].sdpMid : mline.media)
  305. }).c('transport', ice);
  306. for (var i = 0; i < cands.length; i++) {
  307. cand.c('candidate', SDPUtil.candidateToJingle(cands[i].candidate)).up();
  308. }
  309. // add fingerprint
  310. if (SDPUtil.find_line(this.localSDP.media[mid], 'a=fingerprint:', this.localSDP.session)) {
  311. var tmp = SDPUtil.parse_fingerprint(SDPUtil.find_line(this.localSDP.media[mid], 'a=fingerprint:', this.localSDP.session));
  312. tmp.required = true;
  313. cand.c(
  314. 'fingerprint',
  315. {xmlns: 'urn:xmpp:jingle:apps:dtls:0'})
  316. .t(tmp.fingerprint);
  317. delete tmp.fingerprint;
  318. cand.attrs(tmp);
  319. cand.up();
  320. }
  321. cand.up(); // transport
  322. cand.up(); // content
  323. }
  324. }
  325. // might merge last-candidate notification into this, but it is called alot later. See webrtc issue #2340
  326. //console.log('was this the last candidate', this.lasticecandidate);
  327. this.connection.sendIQ(cand,
  328. function () {
  329. var ack = {};
  330. ack.source = 'transportinfo';
  331. $(document).trigger('ack.jingle', [this.sid, ack]);
  332. },
  333. function (stanza) {
  334. var error = ($(stanza).find('error').length) ? {
  335. code: $(stanza).find('error').attr('code'),
  336. reason: $(stanza).find('error :first')[0].tagName,
  337. }:{};
  338. error.source = 'transportinfo';
  339. JingleSession.onJingleError(this.sid, error);
  340. },
  341. 10000);
  342. };
  343. JingleSession.prototype.sendOffer = function () {
  344. //console.log('sendOffer...');
  345. var self = this;
  346. this.peerconnection.createOffer(function (sdp) {
  347. self.createdOffer(sdp);
  348. },
  349. function (e) {
  350. console.error('createOffer failed', e);
  351. },
  352. this.media_constraints
  353. );
  354. };
  355. JingleSession.prototype.createdOffer = function (sdp) {
  356. //console.log('createdOffer', sdp);
  357. var self = this;
  358. this.localSDP = new SDP(sdp.sdp);
  359. //this.localSDP.mangle();
  360. var sendJingle = function () {
  361. var init = $iq({to: this.peerjid,
  362. type: 'set'})
  363. .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
  364. action: 'session-initiate',
  365. initiator: this.initiator,
  366. sid: this.sid});
  367. self.localSDP.toJingle(init, this.initiator == this.me ? 'initiator' : 'responder', this.localStreamsSSRC);
  368. self.connection.sendIQ(init,
  369. function () {
  370. var ack = {};
  371. ack.source = 'offer';
  372. $(document).trigger('ack.jingle', [self.sid, ack]);
  373. },
  374. function (stanza) {
  375. self.state = 'error';
  376. self.peerconnection.close();
  377. var error = ($(stanza).find('error').length) ? {
  378. code: $(stanza).find('error').attr('code'),
  379. reason: $(stanza).find('error :first')[0].tagName,
  380. }:{};
  381. error.source = 'offer';
  382. JingleSession.onJingleError(self.sid, error);
  383. },
  384. 10000);
  385. }
  386. sdp.sdp = this.localSDP.raw;
  387. this.peerconnection.setLocalDescription(sdp,
  388. function () {
  389. if(self.usetrickle)
  390. {
  391. sendJingle();
  392. }
  393. self.setLocalDescription();
  394. //console.log('setLocalDescription success');
  395. },
  396. function (e) {
  397. console.error('setLocalDescription failed', e);
  398. }
  399. );
  400. var cands = SDPUtil.find_lines(this.localSDP.raw, 'a=candidate:');
  401. for (var i = 0; i < cands.length; i++) {
  402. var cand = SDPUtil.parse_icecandidate(cands[i]);
  403. if (cand.type == 'srflx') {
  404. this.hadstuncandidate = true;
  405. } else if (cand.type == 'relay') {
  406. this.hadturncandidate = true;
  407. }
  408. }
  409. };
  410. JingleSession.prototype.setRemoteDescription = function (elem, desctype) {
  411. //console.log('setting remote description... ', desctype);
  412. this.remoteSDP = new SDP('');
  413. this.remoteSDP.fromJingle(elem);
  414. if (this.peerconnection.remoteDescription !== null) {
  415. console.log('setRemoteDescription when remote description is not null, should be pranswer', this.peerconnection.remoteDescription);
  416. if (this.peerconnection.remoteDescription.type == 'pranswer') {
  417. var pranswer = new SDP(this.peerconnection.remoteDescription.sdp);
  418. for (var i = 0; i < pranswer.media.length; i++) {
  419. // make sure we have ice ufrag and pwd
  420. if (!SDPUtil.find_line(this.remoteSDP.media[i], 'a=ice-ufrag:', this.remoteSDP.session)) {
  421. if (SDPUtil.find_line(pranswer.media[i], 'a=ice-ufrag:', pranswer.session)) {
  422. this.remoteSDP.media[i] += SDPUtil.find_line(pranswer.media[i], 'a=ice-ufrag:', pranswer.session) + '\r\n';
  423. } else {
  424. console.warn('no ice ufrag?');
  425. }
  426. if (SDPUtil.find_line(pranswer.media[i], 'a=ice-pwd:', pranswer.session)) {
  427. this.remoteSDP.media[i] += SDPUtil.find_line(pranswer.media[i], 'a=ice-pwd:', pranswer.session) + '\r\n';
  428. } else {
  429. console.warn('no ice pwd?');
  430. }
  431. }
  432. // copy over candidates
  433. var lines = SDPUtil.find_lines(pranswer.media[i], 'a=candidate:');
  434. for (var j = 0; j < lines.length; j++) {
  435. this.remoteSDP.media[i] += lines[j] + '\r\n';
  436. }
  437. }
  438. this.remoteSDP.raw = this.remoteSDP.session + this.remoteSDP.media.join('');
  439. }
  440. }
  441. var remotedesc = new RTCSessionDescription({type: desctype, sdp: this.remoteSDP.raw});
  442. this.peerconnection.setRemoteDescription(remotedesc,
  443. function () {
  444. //console.log('setRemoteDescription success');
  445. },
  446. function (e) {
  447. console.error('setRemoteDescription error', e);
  448. JingleSession.onJingleFatalError(self, e);
  449. }
  450. );
  451. };
  452. JingleSession.prototype.addIceCandidate = function (elem) {
  453. var self = this;
  454. if (this.peerconnection.signalingState == 'closed') {
  455. return;
  456. }
  457. if (!this.peerconnection.remoteDescription && this.peerconnection.signalingState == 'have-local-offer') {
  458. console.log('trickle ice candidate arriving before session accept...');
  459. // create a PRANSWER for setRemoteDescription
  460. if (!this.remoteSDP) {
  461. var cobbled = 'v=0\r\n' +
  462. 'o=- ' + '1923518516' + ' 2 IN IP4 0.0.0.0\r\n' +// FIXME
  463. 's=-\r\n' +
  464. 't=0 0\r\n';
  465. // first, take some things from the local description
  466. for (var i = 0; i < this.localSDP.media.length; i++) {
  467. cobbled += SDPUtil.find_line(this.localSDP.media[i], 'm=') + '\r\n';
  468. cobbled += SDPUtil.find_lines(this.localSDP.media[i], 'a=rtpmap:').join('\r\n') + '\r\n';
  469. if (SDPUtil.find_line(this.localSDP.media[i], 'a=mid:')) {
  470. cobbled += SDPUtil.find_line(this.localSDP.media[i], 'a=mid:') + '\r\n';
  471. }
  472. cobbled += 'a=inactive\r\n';
  473. }
  474. this.remoteSDP = new SDP(cobbled);
  475. }
  476. // then add things like ice and dtls from remote candidate
  477. elem.each(function () {
  478. for (var i = 0; i < self.remoteSDP.media.length; i++) {
  479. if (SDPUtil.find_line(self.remoteSDP.media[i], 'a=mid:' + $(this).attr('name')) ||
  480. self.remoteSDP.media[i].indexOf('m=' + $(this).attr('name')) === 0) {
  481. if (!SDPUtil.find_line(self.remoteSDP.media[i], 'a=ice-ufrag:')) {
  482. var tmp = $(this).find('transport');
  483. self.remoteSDP.media[i] += 'a=ice-ufrag:' + tmp.attr('ufrag') + '\r\n';
  484. self.remoteSDP.media[i] += 'a=ice-pwd:' + tmp.attr('pwd') + '\r\n';
  485. tmp = $(this).find('transport>fingerprint');
  486. if (tmp.length) {
  487. self.remoteSDP.media[i] += 'a=fingerprint:' + tmp.attr('hash') + ' ' + tmp.text() + '\r\n';
  488. } else {
  489. console.log('no dtls fingerprint (webrtc issue #1718?)');
  490. self.remoteSDP.media[i] += 'a=crypto:1 AES_CM_128_HMAC_SHA1_80 inline:BAADBAADBAADBAADBAADBAADBAADBAADBAADBAAD\r\n';
  491. }
  492. break;
  493. }
  494. }
  495. }
  496. });
  497. this.remoteSDP.raw = this.remoteSDP.session + this.remoteSDP.media.join('');
  498. // we need a complete SDP with ice-ufrag/ice-pwd in all parts
  499. // this makes the assumption that the PRANSWER is constructed such that the ice-ufrag is in all mediaparts
  500. // but it could be in the session part as well. since the code above constructs this sdp this can't happen however
  501. var iscomplete = this.remoteSDP.media.filter(function (mediapart) {
  502. return SDPUtil.find_line(mediapart, 'a=ice-ufrag:');
  503. }).length == this.remoteSDP.media.length;
  504. if (iscomplete) {
  505. console.log('setting pranswer');
  506. try {
  507. this.peerconnection.setRemoteDescription(new RTCSessionDescription({type: 'pranswer', sdp: this.remoteSDP.raw }),
  508. function() {
  509. },
  510. function(e) {
  511. console.log('setRemoteDescription pranswer failed', e.toString());
  512. });
  513. } catch (e) {
  514. console.error('setting pranswer failed', e);
  515. }
  516. } else {
  517. //console.log('not yet setting pranswer');
  518. }
  519. }
  520. // operate on each content element
  521. elem.each(function () {
  522. // would love to deactivate this, but firefox still requires it
  523. var idx = -1;
  524. var i;
  525. for (i = 0; i < self.remoteSDP.media.length; i++) {
  526. if (SDPUtil.find_line(self.remoteSDP.media[i], 'a=mid:' + $(this).attr('name')) ||
  527. self.remoteSDP.media[i].indexOf('m=' + $(this).attr('name')) === 0) {
  528. idx = i;
  529. break;
  530. }
  531. }
  532. if (idx == -1) { // fall back to localdescription
  533. for (i = 0; i < self.localSDP.media.length; i++) {
  534. if (SDPUtil.find_line(self.localSDP.media[i], 'a=mid:' + $(this).attr('name')) ||
  535. self.localSDP.media[i].indexOf('m=' + $(this).attr('name')) === 0) {
  536. idx = i;
  537. break;
  538. }
  539. }
  540. }
  541. var name = $(this).attr('name');
  542. // TODO: check ice-pwd and ice-ufrag?
  543. $(this).find('transport>candidate').each(function () {
  544. var line, candidate;
  545. line = SDPUtil.candidateFromJingle(this);
  546. candidate = new RTCIceCandidate({sdpMLineIndex: idx,
  547. sdpMid: name,
  548. candidate: line});
  549. try {
  550. self.peerconnection.addIceCandidate(candidate);
  551. } catch (e) {
  552. console.error('addIceCandidate failed', e.toString(), line);
  553. }
  554. });
  555. });
  556. };
  557. JingleSession.prototype.sendAnswer = function (provisional) {
  558. //console.log('createAnswer', provisional);
  559. var self = this;
  560. this.peerconnection.createAnswer(
  561. function (sdp) {
  562. self.createdAnswer(sdp, provisional);
  563. },
  564. function (e) {
  565. console.error('createAnswer failed', e);
  566. },
  567. this.media_constraints
  568. );
  569. };
  570. JingleSession.prototype.createdAnswer = function (sdp, provisional) {
  571. //console.log('createAnswer callback');
  572. var self = this;
  573. this.localSDP = new SDP(sdp.sdp);
  574. //this.localSDP.mangle();
  575. this.usepranswer = provisional === true;
  576. if (this.usetrickle) {
  577. if (this.usepranswer) {
  578. sdp.type = 'pranswer';
  579. for (var i = 0; i < this.localSDP.media.length; i++) {
  580. this.localSDP.media[i] = this.localSDP.media[i].replace('a=sendrecv\r\n', 'a=inactive\r\n');
  581. }
  582. this.localSDP.raw = this.localSDP.session + '\r\n' + this.localSDP.media.join('');
  583. }
  584. }
  585. var self = this;
  586. var sendJingle = function (ssrcs) {
  587. var accept = $iq({to: self.peerjid,
  588. type: 'set'})
  589. .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
  590. action: 'session-accept',
  591. initiator: self.initiator,
  592. responder: self.responder,
  593. sid: self.sid });
  594. var publicLocalDesc = simulcast.reverseTransformLocalDescription(sdp);
  595. var publicLocalSDP = new SDP(publicLocalDesc.sdp);
  596. publicLocalSDP.toJingle(accept, self.initiator == self.me ? 'initiator' : 'responder', ssrcs);
  597. self.connection.sendIQ(accept,
  598. function () {
  599. var ack = {};
  600. ack.source = 'answer';
  601. $(document).trigger('ack.jingle', [self.sid, ack]);
  602. },
  603. function (stanza) {
  604. var error = ($(stanza).find('error').length) ? {
  605. code: $(stanza).find('error').attr('code'),
  606. reason: $(stanza).find('error :first')[0].tagName,
  607. }:{};
  608. error.source = 'answer';
  609. JingleSession.onJingleError(self.sid, error);
  610. },
  611. 10000);
  612. }
  613. sdp.sdp = this.localSDP.raw;
  614. this.peerconnection.setLocalDescription(sdp,
  615. function () {
  616. //console.log('setLocalDescription success');
  617. if (self.usetrickle && !self.usepranswer) {
  618. sendJingle();
  619. }
  620. self.setLocalDescription();
  621. },
  622. function (e) {
  623. console.error('setLocalDescription failed', e);
  624. }
  625. );
  626. var cands = SDPUtil.find_lines(this.localSDP.raw, 'a=candidate:');
  627. for (var j = 0; j < cands.length; j++) {
  628. var cand = SDPUtil.parse_icecandidate(cands[j]);
  629. if (cand.type == 'srflx') {
  630. this.hadstuncandidate = true;
  631. } else if (cand.type == 'relay') {
  632. this.hadturncandidate = true;
  633. }
  634. }
  635. };
  636. JingleSession.prototype.sendTerminate = function (reason, text) {
  637. var self = this,
  638. term = $iq({to: this.peerjid,
  639. type: 'set'})
  640. .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
  641. action: 'session-terminate',
  642. initiator: this.initiator,
  643. sid: this.sid})
  644. .c('reason')
  645. .c(reason || 'success');
  646. if (text) {
  647. term.up().c('text').t(text);
  648. }
  649. this.connection.sendIQ(term,
  650. function () {
  651. self.peerconnection.close();
  652. self.peerconnection = null;
  653. self.terminate();
  654. var ack = {};
  655. ack.source = 'terminate';
  656. $(document).trigger('ack.jingle', [self.sid, ack]);
  657. },
  658. function (stanza) {
  659. var error = ($(stanza).find('error').length) ? {
  660. code: $(stanza).find('error').attr('code'),
  661. reason: $(stanza).find('error :first')[0].tagName,
  662. }:{};
  663. $(document).trigger('ack.jingle', [self.sid, error]);
  664. },
  665. 10000);
  666. if (this.statsinterval !== null) {
  667. window.clearInterval(this.statsinterval);
  668. this.statsinterval = null;
  669. }
  670. };
  671. JingleSession.prototype.addSource = function (elem, fromJid) {
  672. var self = this;
  673. // FIXME: dirty waiting
  674. if (!this.peerconnection.localDescription)
  675. {
  676. console.warn("addSource - localDescription not ready yet")
  677. setTimeout(function()
  678. {
  679. self.addSource(elem, fromJid);
  680. },
  681. 200
  682. );
  683. return;
  684. }
  685. console.log('addssrc', new Date().getTime());
  686. console.log('ice', this.peerconnection.iceConnectionState);
  687. var sdp = new SDP(this.peerconnection.remoteDescription.sdp);
  688. var mySdp = new SDP(this.peerconnection.localDescription.sdp);
  689. $(elem).each(function (idx, content) {
  690. var name = $(content).attr('name');
  691. var lines = '';
  692. tmp = $(content).find('ssrc-group[xmlns="urn:xmpp:jingle:apps:rtp:ssma:0"]').each(function() {
  693. var semantics = this.getAttribute('semantics');
  694. var ssrcs = $(this).find('>source').map(function () {
  695. return this.getAttribute('ssrc');
  696. }).get();
  697. if (ssrcs.length != 0) {
  698. lines += 'a=ssrc-group:' + semantics + ' ' + ssrcs.join(' ') + '\r\n';
  699. }
  700. });
  701. tmp = $(content).find('source[xmlns="urn:xmpp:jingle:apps:rtp:ssma:0"]'); // can handle both >source and >description>source
  702. tmp.each(function () {
  703. var ssrc = $(this).attr('ssrc');
  704. if(mySdp.containsSSRC(ssrc)){
  705. /**
  706. * This happens when multiple participants change their streams at the same time and
  707. * ColibriFocus.modifySources have to wait for stable state. In the meantime multiple
  708. * addssrc are scheduled for update IQ. See
  709. */
  710. console.warn("Got add stream request for my own ssrc: "+ssrc);
  711. return;
  712. }
  713. $(this).find('>parameter').each(function () {
  714. lines += 'a=ssrc:' + ssrc + ' ' + $(this).attr('name');
  715. if ($(this).attr('value') && $(this).attr('value').length)
  716. lines += ':' + $(this).attr('value');
  717. lines += '\r\n';
  718. });
  719. });
  720. sdp.media.forEach(function(media, idx) {
  721. if (!SDPUtil.find_line(media, 'a=mid:' + name))
  722. return;
  723. sdp.media[idx] += lines;
  724. if (!self.addssrc[idx]) self.addssrc[idx] = '';
  725. self.addssrc[idx] += lines;
  726. });
  727. sdp.raw = sdp.session + sdp.media.join('');
  728. });
  729. this.modifySources();
  730. };
  731. JingleSession.prototype.removeSource = function (elem, fromJid) {
  732. var self = this;
  733. // FIXME: dirty waiting
  734. if (!this.peerconnection.localDescription)
  735. {
  736. console.warn("removeSource - localDescription not ready yet")
  737. setTimeout(function()
  738. {
  739. self.removeSource(elem, fromJid);
  740. },
  741. 200
  742. );
  743. return;
  744. }
  745. console.log('removessrc', new Date().getTime());
  746. console.log('ice', this.peerconnection.iceConnectionState);
  747. var sdp = new SDP(this.peerconnection.remoteDescription.sdp);
  748. var mySdp = new SDP(this.peerconnection.localDescription.sdp);
  749. $(elem).each(function (idx, content) {
  750. var name = $(content).attr('name');
  751. var lines = '';
  752. tmp = $(content).find('ssrc-group[xmlns="urn:xmpp:jingle:apps:rtp:ssma:0"]').each(function() {
  753. var semantics = this.getAttribute('semantics');
  754. var ssrcs = $(this).find('>source').map(function () {
  755. return this.getAttribute('ssrc');
  756. }).get();
  757. if (ssrcs.length != 0) {
  758. lines += 'a=ssrc-group:' + semantics + ' ' + ssrcs.join(' ') + '\r\n';
  759. }
  760. });
  761. tmp = $(content).find('source[xmlns="urn:xmpp:jingle:apps:rtp:ssma:0"]'); // can handle both >source and >description>source
  762. tmp.each(function () {
  763. var ssrc = $(this).attr('ssrc');
  764. // This should never happen, but can be useful for bug detection
  765. if(mySdp.containsSSRC(ssrc)){
  766. console.error("Got remove stream request for my own ssrc: "+ssrc);
  767. return;
  768. }
  769. $(this).find('>parameter').each(function () {
  770. lines += 'a=ssrc:' + ssrc + ' ' + $(this).attr('name');
  771. if ($(this).attr('value') && $(this).attr('value').length)
  772. lines += ':' + $(this).attr('value');
  773. lines += '\r\n';
  774. });
  775. });
  776. sdp.media.forEach(function(media, idx) {
  777. if (!SDPUtil.find_line(media, 'a=mid:' + name))
  778. return;
  779. sdp.media[idx] += lines;
  780. if (!self.removessrc[idx]) self.removessrc[idx] = '';
  781. self.removessrc[idx] += lines;
  782. });
  783. sdp.raw = sdp.session + sdp.media.join('');
  784. });
  785. this.modifySources();
  786. };
  787. JingleSession.prototype.modifySources = function (successCallback) {
  788. var self = this;
  789. if (this.peerconnection.signalingState == 'closed') return;
  790. if (!(this.addssrc.length || this.removessrc.length || this.pendingop !== null || this.switchstreams)){
  791. // There is nothing to do since scheduled job might have been executed by another succeeding call
  792. this.setLocalDescription();
  793. if(successCallback){
  794. successCallback();
  795. }
  796. return;
  797. }
  798. // FIXME: this is a big hack
  799. // https://code.google.com/p/webrtc/issues/detail?id=2688
  800. // ^ has been fixed.
  801. if (!(this.peerconnection.signalingState == 'stable' && this.peerconnection.iceConnectionState == 'connected')) {
  802. console.warn('modifySources not yet', this.peerconnection.signalingState, this.peerconnection.iceConnectionState);
  803. this.wait = true;
  804. window.setTimeout(function() { self.modifySources(successCallback); }, 250);
  805. return;
  806. }
  807. if (this.wait) {
  808. window.setTimeout(function() { self.modifySources(successCallback); }, 2500);
  809. this.wait = false;
  810. return;
  811. }
  812. // Reset switch streams flag
  813. this.switchstreams = false;
  814. var sdp = new SDP(this.peerconnection.remoteDescription.sdp);
  815. // add sources
  816. this.addssrc.forEach(function(lines, idx) {
  817. sdp.media[idx] += lines;
  818. });
  819. this.addssrc = [];
  820. // remove sources
  821. this.removessrc.forEach(function(lines, idx) {
  822. lines = lines.split('\r\n');
  823. lines.pop(); // remove empty last element;
  824. lines.forEach(function(line) {
  825. sdp.media[idx] = sdp.media[idx].replace(line + '\r\n', '');
  826. });
  827. });
  828. this.removessrc = [];
  829. // FIXME:
  830. // this was a hack for the situation when only one peer exists
  831. // in the conference.
  832. // check if still required and remove
  833. if (sdp.media[0])
  834. sdp.media[0] = sdp.media[0].replace('a=recvonly', 'a=sendrecv');
  835. if (sdp.media[1])
  836. sdp.media[1] = sdp.media[1].replace('a=recvonly', 'a=sendrecv');
  837. sdp.raw = sdp.session + sdp.media.join('');
  838. this.peerconnection.setRemoteDescription(new RTCSessionDescription({type: 'offer', sdp: sdp.raw}),
  839. function() {
  840. if(self.signalingState == 'closed') {
  841. console.error("createAnswer attempt on closed state");
  842. return;
  843. }
  844. self.peerconnection.createAnswer(
  845. function(modifiedAnswer) {
  846. // change video direction, see https://github.com/jitsi/jitmeet/issues/41
  847. if (self.pendingop !== null) {
  848. var sdp = new SDP(modifiedAnswer.sdp);
  849. if (sdp.media.length > 1) {
  850. switch(self.pendingop) {
  851. case 'mute':
  852. sdp.media[1] = sdp.media[1].replace('a=sendrecv', 'a=recvonly');
  853. break;
  854. case 'unmute':
  855. sdp.media[1] = sdp.media[1].replace('a=recvonly', 'a=sendrecv');
  856. break;
  857. }
  858. sdp.raw = sdp.session + sdp.media.join('');
  859. modifiedAnswer.sdp = sdp.raw;
  860. }
  861. self.pendingop = null;
  862. }
  863. // FIXME: pushing down an answer while ice connection state
  864. // is still checking is bad...
  865. //console.log(self.peerconnection.iceConnectionState);
  866. // trying to work around another chrome bug
  867. //modifiedAnswer.sdp = modifiedAnswer.sdp.replace(/a=setup:active/g, 'a=setup:actpass');
  868. self.peerconnection.setLocalDescription(modifiedAnswer,
  869. function() {
  870. //console.log('modified setLocalDescription ok');
  871. self.setLocalDescription();
  872. if(successCallback){
  873. successCallback();
  874. }
  875. },
  876. function(error) {
  877. console.error('modified setLocalDescription failed', error);
  878. }
  879. );
  880. },
  881. function(error) {
  882. console.error('modified answer failed', error);
  883. }
  884. );
  885. },
  886. function(error) {
  887. console.error('modify failed', error);
  888. }
  889. );
  890. };
  891. /**
  892. * Switches video streams.
  893. * @param new_stream new stream that will be used as video of this session.
  894. * @param oldStream old video stream of this session.
  895. * @param success_callback callback executed after successful stream switch.
  896. */
  897. JingleSession.prototype.switchStreams = function (new_stream, oldStream, success_callback) {
  898. var self = this;
  899. // Remember SDP to figure out added/removed SSRCs
  900. var oldSdp = null;
  901. if(self.peerconnection) {
  902. if(self.peerconnection.localDescription) {
  903. oldSdp = new SDP(self.peerconnection.localDescription.sdp);
  904. }
  905. self.peerconnection.removeStream(oldStream, true);
  906. self.peerconnection.addStream(new_stream);
  907. }
  908. RTC.switchVideoStreams(new_stream, oldStream);
  909. // Conference is not active
  910. if(!oldSdp || !self.peerconnection) {
  911. success_callback();
  912. return;
  913. }
  914. self.switchstreams = true;
  915. self.modifySources(function() {
  916. console.log('modify sources done');
  917. success_callback();
  918. var newSdp = new SDP(self.peerconnection.localDescription.sdp);
  919. console.log("SDPs", oldSdp, newSdp);
  920. self.notifyMySSRCUpdate(oldSdp, newSdp);
  921. });
  922. };
  923. /**
  924. * Figures out added/removed ssrcs and send update IQs.
  925. * @param old_sdp SDP object for old description.
  926. * @param new_sdp SDP object for new description.
  927. */
  928. JingleSession.prototype.notifyMySSRCUpdate = function (old_sdp, new_sdp) {
  929. if (!(this.peerconnection.signalingState == 'stable' &&
  930. this.peerconnection.iceConnectionState == 'connected')){
  931. console.log("Too early to send updates");
  932. return;
  933. }
  934. // send source-remove IQ.
  935. sdpDiffer = new SDPDiffer(new_sdp, old_sdp);
  936. var remove = $iq({to: this.peerjid, type: 'set'})
  937. .c('jingle', {
  938. xmlns: 'urn:xmpp:jingle:1',
  939. action: 'source-remove',
  940. initiator: this.initiator,
  941. sid: this.sid
  942. }
  943. );
  944. var removed = sdpDiffer.toJingle(remove);
  945. if (removed) {
  946. this.connection.sendIQ(remove,
  947. function (res) {
  948. console.info('got remove result', res);
  949. },
  950. function (err) {
  951. console.error('got remove error', err);
  952. }
  953. );
  954. } else {
  955. console.log('removal not necessary');
  956. }
  957. // send source-add IQ.
  958. var sdpDiffer = new SDPDiffer(old_sdp, new_sdp);
  959. var add = $iq({to: this.peerjid, type: 'set'})
  960. .c('jingle', {
  961. xmlns: 'urn:xmpp:jingle:1',
  962. action: 'source-add',
  963. initiator: this.initiator,
  964. sid: this.sid
  965. }
  966. );
  967. var added = sdpDiffer.toJingle(add);
  968. if (added) {
  969. this.connection.sendIQ(add,
  970. function (res) {
  971. console.info('got add result', res);
  972. },
  973. function (err) {
  974. console.error('got add error', err);
  975. }
  976. );
  977. } else {
  978. console.log('addition not necessary');
  979. }
  980. };
  981. /**
  982. * Determines whether the (local) video is mute i.e. all video tracks are
  983. * disabled.
  984. *
  985. * @return <tt>true</tt> if the (local) video is mute i.e. all video tracks are
  986. * disabled; otherwise, <tt>false</tt>
  987. */
  988. JingleSession.prototype.isVideoMute = function () {
  989. var tracks = RTC.localVideo.getVideoTracks();
  990. var mute = true;
  991. for (var i = 0; i < tracks.length; ++i) {
  992. if (tracks[i].enabled) {
  993. mute = false;
  994. break;
  995. }
  996. }
  997. return mute;
  998. };
  999. /**
  1000. * Mutes/unmutes the (local) video i.e. enables/disables all video tracks.
  1001. *
  1002. * @param mute <tt>true</tt> to mute the (local) video i.e. to disable all video
  1003. * tracks; otherwise, <tt>false</tt>
  1004. * @param callback a function to be invoked with <tt>mute</tt> after all video
  1005. * tracks have been enabled/disabled. The function may, optionally, return
  1006. * another function which is to be invoked after the whole mute/unmute operation
  1007. * has completed successfully.
  1008. * @param options an object which specifies optional arguments such as the
  1009. * <tt>boolean</tt> key <tt>byUser</tt> with default value <tt>true</tt> which
  1010. * specifies whether the method was initiated in response to a user command (in
  1011. * contrast to an automatic decision made by the application logic)
  1012. */
  1013. JingleSession.prototype.setVideoMute = function (mute, callback, options) {
  1014. var byUser;
  1015. if (options) {
  1016. byUser = options.byUser;
  1017. if (typeof byUser === 'undefined') {
  1018. byUser = true;
  1019. }
  1020. } else {
  1021. byUser = true;
  1022. }
  1023. // The user's command to mute the (local) video takes precedence over any
  1024. // automatic decision made by the application logic.
  1025. if (byUser) {
  1026. this.videoMuteByUser = mute;
  1027. } else if (this.videoMuteByUser) {
  1028. return;
  1029. }
  1030. var self = this;
  1031. var localCallback = function (mute) {
  1032. self.connection.emuc.addVideoInfoToPresence(mute);
  1033. self.connection.emuc.sendPresence();
  1034. return callback(mute)
  1035. };
  1036. if (mute == RTC.localVideo.isMuted())
  1037. {
  1038. // Even if no change occurs, the specified callback is to be executed.
  1039. // The specified callback may, optionally, return a successCallback
  1040. // which is to be executed as well.
  1041. var successCallback = localCallback(mute);
  1042. if (successCallback) {
  1043. successCallback();
  1044. }
  1045. } else {
  1046. RTC.localVideo.setMute(!mute);
  1047. this.hardMuteVideo(mute);
  1048. this.modifySources(localCallback(mute));
  1049. }
  1050. };
  1051. // SDP-based mute by going recvonly/sendrecv
  1052. // FIXME: should probably black out the screen as well
  1053. JingleSession.prototype.toggleVideoMute = function (callback) {
  1054. this.service.setVideoMute(RTC.localVideo.isMuted(), callback);
  1055. };
  1056. JingleSession.prototype.hardMuteVideo = function (muted) {
  1057. this.pendingop = muted ? 'mute' : 'unmute';
  1058. };
  1059. JingleSession.prototype.sendMute = function (muted, content) {
  1060. var info = $iq({to: this.peerjid,
  1061. type: 'set'})
  1062. .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
  1063. action: 'session-info',
  1064. initiator: this.initiator,
  1065. sid: this.sid });
  1066. info.c(muted ? 'mute' : 'unmute', {xmlns: 'urn:xmpp:jingle:apps:rtp:info:1'});
  1067. info.attrs({'creator': this.me == this.initiator ? 'creator' : 'responder'});
  1068. if (content) {
  1069. info.attrs({'name': content});
  1070. }
  1071. this.connection.send(info);
  1072. };
  1073. JingleSession.prototype.sendRinging = function () {
  1074. var info = $iq({to: this.peerjid,
  1075. type: 'set'})
  1076. .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
  1077. action: 'session-info',
  1078. initiator: this.initiator,
  1079. sid: this.sid });
  1080. info.c('ringing', {xmlns: 'urn:xmpp:jingle:apps:rtp:info:1'});
  1081. this.connection.send(info);
  1082. };
  1083. JingleSession.prototype.getStats = function (interval) {
  1084. var self = this;
  1085. var recv = {audio: 0, video: 0};
  1086. var lost = {audio: 0, video: 0};
  1087. var lastrecv = {audio: 0, video: 0};
  1088. var lastlost = {audio: 0, video: 0};
  1089. var loss = {audio: 0, video: 0};
  1090. var delta = {audio: 0, video: 0};
  1091. this.statsinterval = window.setInterval(function () {
  1092. if (self && self.peerconnection && self.peerconnection.getStats) {
  1093. self.peerconnection.getStats(function (stats) {
  1094. var results = stats.result();
  1095. // TODO: there are so much statistics you can get from this..
  1096. for (var i = 0; i < results.length; ++i) {
  1097. if (results[i].type == 'ssrc') {
  1098. var packetsrecv = results[i].stat('packetsReceived');
  1099. var packetslost = results[i].stat('packetsLost');
  1100. if (packetsrecv && packetslost) {
  1101. packetsrecv = parseInt(packetsrecv, 10);
  1102. packetslost = parseInt(packetslost, 10);
  1103. if (results[i].stat('googFrameRateReceived')) {
  1104. lastlost.video = lost.video;
  1105. lastrecv.video = recv.video;
  1106. recv.video = packetsrecv;
  1107. lost.video = packetslost;
  1108. } else {
  1109. lastlost.audio = lost.audio;
  1110. lastrecv.audio = recv.audio;
  1111. recv.audio = packetsrecv;
  1112. lost.audio = packetslost;
  1113. }
  1114. }
  1115. }
  1116. }
  1117. delta.audio = recv.audio - lastrecv.audio;
  1118. delta.video = recv.video - lastrecv.video;
  1119. loss.audio = (delta.audio > 0) ? Math.ceil(100 * (lost.audio - lastlost.audio) / delta.audio) : 0;
  1120. loss.video = (delta.video > 0) ? Math.ceil(100 * (lost.video - lastlost.video) / delta.video) : 0;
  1121. $(document).trigger('packetloss.jingle', [self.sid, loss]);
  1122. });
  1123. }
  1124. }, interval || 3000);
  1125. return this.statsinterval;
  1126. };
  1127. JingleSession.onJingleError = function (session, error)
  1128. {
  1129. console.error("Jingle error", error);
  1130. }
  1131. JingleSession.onJingleFatalError = function (session, error)
  1132. {
  1133. this.service.sessionTerminated = true;
  1134. connection.emuc.doLeave();
  1135. UI.messageHandler.showError( "Sorry",
  1136. "Internal application error[setRemoteDescription]");
  1137. }
  1138. JingleSession.prototype.setLocalDescription = function () {
  1139. // put our ssrcs into presence so other clients can identify our stream
  1140. var newssrcs = [];
  1141. var media = simulcast.parseMedia(this.peerconnection.localDescription);
  1142. media.forEach(function (media) {
  1143. if(Object.keys(media.sources).length > 0) {
  1144. // TODO(gp) maybe exclude FID streams?
  1145. Object.keys(media.sources).forEach(function (ssrc) {
  1146. newssrcs.push({
  1147. 'ssrc': ssrc,
  1148. 'type': media.type,
  1149. 'direction': media.direction
  1150. });
  1151. });
  1152. }
  1153. else if(this.localStreamsSSRC && this.localStreamsSSRC[media.type])
  1154. {
  1155. newssrcs.push({
  1156. 'ssrc': this.localStreamsSSRC[media.type],
  1157. 'type': media.type,
  1158. 'direction': media.direction
  1159. });
  1160. }
  1161. });
  1162. console.log('new ssrcs', newssrcs);
  1163. // Have to clear presence map to get rid of removed streams
  1164. this.connection.emuc.clearPresenceMedia();
  1165. if (newssrcs.length > 0) {
  1166. for (var i = 1; i <= newssrcs.length; i ++) {
  1167. // Change video type to screen
  1168. if (newssrcs[i-1].type === 'video' && desktopsharing.isUsingScreenStream()) {
  1169. newssrcs[i-1].type = 'screen';
  1170. }
  1171. this.connection.emuc.addMediaToPresence(i,
  1172. newssrcs[i-1].type, newssrcs[i-1].ssrc, newssrcs[i-1].direction);
  1173. }
  1174. this.connection.emuc.sendPresence();
  1175. }
  1176. }
  1177. // an attempt to work around https://github.com/jitsi/jitmeet/issues/32
  1178. function sendKeyframe(pc) {
  1179. console.log('sendkeyframe', pc.iceConnectionState);
  1180. if (pc.iceConnectionState !== 'connected') return; // safe...
  1181. pc.setRemoteDescription(
  1182. pc.remoteDescription,
  1183. function () {
  1184. pc.createAnswer(
  1185. function (modifiedAnswer) {
  1186. pc.setLocalDescription(
  1187. modifiedAnswer,
  1188. function () {
  1189. // noop
  1190. },
  1191. function (error) {
  1192. console.log('triggerKeyframe setLocalDescription failed', error);
  1193. UI.messageHandler.showError();
  1194. }
  1195. );
  1196. },
  1197. function (error) {
  1198. console.log('triggerKeyframe createAnswer failed', error);
  1199. UI.messageHandler.showError();
  1200. }
  1201. );
  1202. },
  1203. function (error) {
  1204. console.log('triggerKeyframe setRemoteDescription failed', error);
  1205. UI.messageHandler.showError();
  1206. }
  1207. );
  1208. }
  1209. JingleSession.prototype.remoteStreamAdded = function (data) {
  1210. var self = this;
  1211. var thessrc;
  1212. // look up an associated JID for a stream id
  1213. if (data.stream.id && data.stream.id.indexOf('mixedmslabel') === -1) {
  1214. // look only at a=ssrc: and _not_ at a=ssrc-group: lines
  1215. var ssrclines
  1216. = SDPUtil.find_lines(this.peerconnection.remoteDescription.sdp, 'a=ssrc:');
  1217. ssrclines = ssrclines.filter(function (line) {
  1218. // NOTE(gp) previously we filtered on the mslabel, but that property
  1219. // is not always present.
  1220. // return line.indexOf('mslabel:' + data.stream.label) !== -1;
  1221. return ((line.indexOf('msid:' + data.stream.id) !== -1));
  1222. });
  1223. if (ssrclines.length) {
  1224. thessrc = ssrclines[0].substring(7).split(' ')[0];
  1225. // We signal our streams (through Jingle to the focus) before we set
  1226. // our presence (through which peers associate remote streams to
  1227. // jids). So, it might arrive that a remote stream is added but
  1228. // ssrc2jid is not yet updated and thus data.peerjid cannot be
  1229. // successfully set. Here we wait for up to a second for the
  1230. // presence to arrive.
  1231. if (!ssrc2jid[thessrc]) {
  1232. // TODO(gp) limit wait duration to 1 sec.
  1233. setTimeout(function(d) {
  1234. return function() {
  1235. self.remoteStreamAdded(d);
  1236. }
  1237. }(data), 250);
  1238. return;
  1239. }
  1240. // ok to overwrite the one from focus? might save work in colibri.js
  1241. console.log('associated jid', ssrc2jid[thessrc], data.peerjid);
  1242. if (ssrc2jid[thessrc]) {
  1243. data.peerjid = ssrc2jid[thessrc];
  1244. }
  1245. }
  1246. }
  1247. //TODO: this code should be removed when firefox implement multistream support
  1248. if(RTC.getBrowserType() == RTCBrowserType.RTC_BROWSER_FIREFOX)
  1249. {
  1250. if((notReceivedSSRCs.length == 0) ||
  1251. !ssrc2jid[notReceivedSSRCs[notReceivedSSRCs.length - 1]])
  1252. {
  1253. // TODO(gp) limit wait duration to 1 sec.
  1254. setTimeout(function(d) {
  1255. return function() {
  1256. self.remoteStreamAdded(d);
  1257. }
  1258. }(data), 250);
  1259. return;
  1260. }
  1261. thessrc = notReceivedSSRCs.pop();
  1262. if (ssrc2jid[thessrc]) {
  1263. data.peerjid = ssrc2jid[thessrc];
  1264. }
  1265. }
  1266. RTC.createRemoteStream(data, this.sid, thessrc);
  1267. var isVideo = data.stream.getVideoTracks().length > 0;
  1268. // an attempt to work around https://github.com/jitsi/jitmeet/issues/32
  1269. if (isVideo &&
  1270. data.peerjid && this.peerjid === data.peerjid &&
  1271. data.stream.getVideoTracks().length === 0 &&
  1272. RTC.localVideo.getTracks().length > 0) {
  1273. window.setTimeout(function () {
  1274. sendKeyframe(self.peerconnection);
  1275. }, 3000);
  1276. }
  1277. }
  1278. module.exports = JingleSession;
  1279. },{"./SDP":2,"./SDPDiffer":3,"./SDPUtil":4,"./TraceablePeerConnection":5}],2:[function(require,module,exports){
  1280. /* jshint -W117 */
  1281. var SDPUtil = require("./SDPUtil");
  1282. // SDP STUFF
  1283. function SDP(sdp) {
  1284. this.media = sdp.split('\r\nm=');
  1285. for (var i = 1; i < this.media.length; i++) {
  1286. this.media[i] = 'm=' + this.media[i];
  1287. if (i != this.media.length - 1) {
  1288. this.media[i] += '\r\n';
  1289. }
  1290. }
  1291. this.session = this.media.shift() + '\r\n';
  1292. this.raw = this.session + this.media.join('');
  1293. }
  1294. /**
  1295. * Returns map of MediaChannel mapped per channel idx.
  1296. */
  1297. SDP.prototype.getMediaSsrcMap = function() {
  1298. var self = this;
  1299. var media_ssrcs = {};
  1300. var tmp;
  1301. for (var mediaindex = 0; mediaindex < self.media.length; mediaindex++) {
  1302. tmp = SDPUtil.find_lines(self.media[mediaindex], 'a=ssrc:');
  1303. var mid = SDPUtil.parse_mid(SDPUtil.find_line(self.media[mediaindex], 'a=mid:'));
  1304. var media = {
  1305. mediaindex: mediaindex,
  1306. mid: mid,
  1307. ssrcs: {},
  1308. ssrcGroups: []
  1309. };
  1310. media_ssrcs[mediaindex] = media;
  1311. tmp.forEach(function (line) {
  1312. var linessrc = line.substring(7).split(' ')[0];
  1313. // allocate new ChannelSsrc
  1314. if(!media.ssrcs[linessrc]) {
  1315. media.ssrcs[linessrc] = {
  1316. ssrc: linessrc,
  1317. lines: []
  1318. };
  1319. }
  1320. media.ssrcs[linessrc].lines.push(line);
  1321. });
  1322. tmp = SDPUtil.find_lines(self.media[mediaindex], 'a=ssrc-group:');
  1323. tmp.forEach(function(line){
  1324. var semantics = line.substr(0, idx).substr(13);
  1325. var ssrcs = line.substr(14 + semantics.length).split(' ');
  1326. if (ssrcs.length != 0) {
  1327. media.ssrcGroups.push({
  1328. semantics: semantics,
  1329. ssrcs: ssrcs
  1330. });
  1331. }
  1332. });
  1333. }
  1334. return media_ssrcs;
  1335. };
  1336. /**
  1337. * Returns <tt>true</tt> if this SDP contains given SSRC.
  1338. * @param ssrc the ssrc to check.
  1339. * @returns {boolean} <tt>true</tt> if this SDP contains given SSRC.
  1340. */
  1341. SDP.prototype.containsSSRC = function(ssrc) {
  1342. var medias = this.getMediaSsrcMap();
  1343. var contains = false;
  1344. Object.keys(medias).forEach(function(mediaindex){
  1345. var media = medias[mediaindex];
  1346. //console.log("Check", channel, ssrc);
  1347. if(Object.keys(media.ssrcs).indexOf(ssrc) != -1){
  1348. contains = true;
  1349. }
  1350. });
  1351. return contains;
  1352. };
  1353. // remove iSAC and CN from SDP
  1354. SDP.prototype.mangle = function () {
  1355. var i, j, mline, lines, rtpmap, newdesc;
  1356. for (i = 0; i < this.media.length; i++) {
  1357. lines = this.media[i].split('\r\n');
  1358. lines.pop(); // remove empty last element
  1359. mline = SDPUtil.parse_mline(lines.shift());
  1360. if (mline.media != 'audio')
  1361. continue;
  1362. newdesc = '';
  1363. mline.fmt.length = 0;
  1364. for (j = 0; j < lines.length; j++) {
  1365. if (lines[j].substr(0, 9) == 'a=rtpmap:') {
  1366. rtpmap = SDPUtil.parse_rtpmap(lines[j]);
  1367. if (rtpmap.name == 'CN' || rtpmap.name == 'ISAC')
  1368. continue;
  1369. mline.fmt.push(rtpmap.id);
  1370. newdesc += lines[j] + '\r\n';
  1371. } else {
  1372. newdesc += lines[j] + '\r\n';
  1373. }
  1374. }
  1375. this.media[i] = SDPUtil.build_mline(mline) + '\r\n';
  1376. this.media[i] += newdesc;
  1377. }
  1378. this.raw = this.session + this.media.join('');
  1379. };
  1380. // remove lines matching prefix from session section
  1381. SDP.prototype.removeSessionLines = function(prefix) {
  1382. var self = this;
  1383. var lines = SDPUtil.find_lines(this.session, prefix);
  1384. lines.forEach(function(line) {
  1385. self.session = self.session.replace(line + '\r\n', '');
  1386. });
  1387. this.raw = this.session + this.media.join('');
  1388. return lines;
  1389. }
  1390. // remove lines matching prefix from a media section specified by mediaindex
  1391. // TODO: non-numeric mediaindex could match mid
  1392. SDP.prototype.removeMediaLines = function(mediaindex, prefix) {
  1393. var self = this;
  1394. var lines = SDPUtil.find_lines(this.media[mediaindex], prefix);
  1395. lines.forEach(function(line) {
  1396. self.media[mediaindex] = self.media[mediaindex].replace(line + '\r\n', '');
  1397. });
  1398. this.raw = this.session + this.media.join('');
  1399. return lines;
  1400. }
  1401. // add content's to a jingle element
  1402. SDP.prototype.toJingle = function (elem, thecreator, ssrcs) {
  1403. // console.log("SSRC" + ssrcs["audio"] + " - " + ssrcs["video"]);
  1404. var i, j, k, mline, ssrc, rtpmap, tmp, line, lines;
  1405. var self = this;
  1406. // new bundle plan
  1407. if (SDPUtil.find_line(this.session, 'a=group:')) {
  1408. lines = SDPUtil.find_lines(this.session, 'a=group:');
  1409. for (i = 0; i < lines.length; i++) {
  1410. tmp = lines[i].split(' ');
  1411. var semantics = tmp.shift().substr(8);
  1412. elem.c('group', {xmlns: 'urn:xmpp:jingle:apps:grouping:0', semantics:semantics});
  1413. for (j = 0; j < tmp.length; j++) {
  1414. elem.c('content', {name: tmp[j]}).up();
  1415. }
  1416. elem.up();
  1417. }
  1418. }
  1419. for (i = 0; i < this.media.length; i++) {
  1420. mline = SDPUtil.parse_mline(this.media[i].split('\r\n')[0]);
  1421. if (!(mline.media === 'audio' ||
  1422. mline.media === 'video' ||
  1423. mline.media === 'application'))
  1424. {
  1425. continue;
  1426. }
  1427. if (SDPUtil.find_line(this.media[i], 'a=ssrc:')) {
  1428. ssrc = SDPUtil.find_line(this.media[i], 'a=ssrc:').substring(7).split(' ')[0]; // take the first
  1429. } else {
  1430. if(ssrcs && ssrcs[mline.media])
  1431. {
  1432. ssrc = ssrcs[mline.media];
  1433. }
  1434. else
  1435. ssrc = false;
  1436. }
  1437. elem.c('content', {creator: thecreator, name: mline.media});
  1438. if (SDPUtil.find_line(this.media[i], 'a=mid:')) {
  1439. // prefer identifier from a=mid if present
  1440. var mid = SDPUtil.parse_mid(SDPUtil.find_line(this.media[i], 'a=mid:'));
  1441. elem.attrs({ name: mid });
  1442. }
  1443. if (SDPUtil.find_line(this.media[i], 'a=rtpmap:').length)
  1444. {
  1445. elem.c('description',
  1446. {xmlns: 'urn:xmpp:jingle:apps:rtp:1',
  1447. media: mline.media });
  1448. if (ssrc) {
  1449. elem.attrs({ssrc: ssrc});
  1450. }
  1451. for (j = 0; j < mline.fmt.length; j++) {
  1452. rtpmap = SDPUtil.find_line(this.media[i], 'a=rtpmap:' + mline.fmt[j]);
  1453. elem.c('payload-type', SDPUtil.parse_rtpmap(rtpmap));
  1454. // put any 'a=fmtp:' + mline.fmt[j] lines into <param name=foo value=bar/>
  1455. if (SDPUtil.find_line(this.media[i], 'a=fmtp:' + mline.fmt[j])) {
  1456. tmp = SDPUtil.parse_fmtp(SDPUtil.find_line(this.media[i], 'a=fmtp:' + mline.fmt[j]));
  1457. for (k = 0; k < tmp.length; k++) {
  1458. elem.c('parameter', tmp[k]).up();
  1459. }
  1460. }
  1461. this.RtcpFbToJingle(i, elem, mline.fmt[j]); // XEP-0293 -- map a=rtcp-fb
  1462. elem.up();
  1463. }
  1464. if (SDPUtil.find_line(this.media[i], 'a=crypto:', this.session)) {
  1465. elem.c('encryption', {required: 1});
  1466. var crypto = SDPUtil.find_lines(this.media[i], 'a=crypto:', this.session);
  1467. crypto.forEach(function(line) {
  1468. elem.c('crypto', SDPUtil.parse_crypto(line)).up();
  1469. });
  1470. elem.up(); // end of encryption
  1471. }
  1472. if (ssrc) {
  1473. // new style mapping
  1474. elem.c('source', { ssrc: ssrc, xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });
  1475. // FIXME: group by ssrc and support multiple different ssrcs
  1476. var ssrclines = SDPUtil.find_lines(this.media[i], 'a=ssrc:');
  1477. if(ssrclines.length > 0) {
  1478. ssrclines.forEach(function (line) {
  1479. idx = line.indexOf(' ');
  1480. var linessrc = line.substr(0, idx).substr(7);
  1481. if (linessrc != ssrc) {
  1482. elem.up();
  1483. ssrc = linessrc;
  1484. elem.c('source', { ssrc: ssrc, xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });
  1485. }
  1486. var kv = line.substr(idx + 1);
  1487. elem.c('parameter');
  1488. if (kv.indexOf(':') == -1) {
  1489. elem.attrs({ name: kv });
  1490. } else {
  1491. elem.attrs({ name: kv.split(':', 2)[0] });
  1492. elem.attrs({ value: kv.split(':', 2)[1] });
  1493. }
  1494. elem.up();
  1495. });
  1496. elem.up();
  1497. }
  1498. else
  1499. {
  1500. elem.up();
  1501. elem.c('source', { ssrc: ssrc, xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });
  1502. elem.c('parameter');
  1503. elem.attrs({name: "cname", value:Math.random().toString(36).substring(7)});
  1504. elem.up();
  1505. var msid = null;
  1506. if(mline.media == "audio")
  1507. {
  1508. msid = RTC.localAudio.getId();
  1509. }
  1510. else
  1511. {
  1512. msid = RTC.localVideo.getId();
  1513. }
  1514. if(msid != null)
  1515. {
  1516. msid = msid.replace(/[\{,\}]/g,"");
  1517. elem.c('parameter');
  1518. elem.attrs({name: "msid", value:msid});
  1519. elem.up();
  1520. elem.c('parameter');
  1521. elem.attrs({name: "mslabel", value:msid});
  1522. elem.up();
  1523. elem.c('parameter');
  1524. elem.attrs({name: "label", value:msid});
  1525. elem.up();
  1526. elem.up();
  1527. }
  1528. }
  1529. // XEP-0339 handle ssrc-group attributes
  1530. var ssrc_group_lines = SDPUtil.find_lines(this.media[i], 'a=ssrc-group:');
  1531. ssrc_group_lines.forEach(function(line) {
  1532. idx = line.indexOf(' ');
  1533. var semantics = line.substr(0, idx).substr(13);
  1534. var ssrcs = line.substr(14 + semantics.length).split(' ');
  1535. if (ssrcs.length != 0) {
  1536. elem.c('ssrc-group', { semantics: semantics, xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });
  1537. ssrcs.forEach(function(ssrc) {
  1538. elem.c('source', { ssrc: ssrc })
  1539. .up();
  1540. });
  1541. elem.up();
  1542. }
  1543. });
  1544. }
  1545. if (SDPUtil.find_line(this.media[i], 'a=rtcp-mux')) {
  1546. elem.c('rtcp-mux').up();
  1547. }
  1548. // XEP-0293 -- map a=rtcp-fb:*
  1549. this.RtcpFbToJingle(i, elem, '*');
  1550. // XEP-0294
  1551. if (SDPUtil.find_line(this.media[i], 'a=extmap:')) {
  1552. lines = SDPUtil.find_lines(this.media[i], 'a=extmap:');
  1553. for (j = 0; j < lines.length; j++) {
  1554. tmp = SDPUtil.parse_extmap(lines[j]);
  1555. elem.c('rtp-hdrext', { xmlns: 'urn:xmpp:jingle:apps:rtp:rtp-hdrext:0',
  1556. uri: tmp.uri,
  1557. id: tmp.value });
  1558. if (tmp.hasOwnProperty('direction')) {
  1559. switch (tmp.direction) {
  1560. case 'sendonly':
  1561. elem.attrs({senders: 'responder'});
  1562. break;
  1563. case 'recvonly':
  1564. elem.attrs({senders: 'initiator'});
  1565. break;
  1566. case 'sendrecv':
  1567. elem.attrs({senders: 'both'});
  1568. break;
  1569. case 'inactive':
  1570. elem.attrs({senders: 'none'});
  1571. break;
  1572. }
  1573. }
  1574. // TODO: handle params
  1575. elem.up();
  1576. }
  1577. }
  1578. elem.up(); // end of description
  1579. }
  1580. // map ice-ufrag/pwd, dtls fingerprint, candidates
  1581. this.TransportToJingle(i, elem);
  1582. if (SDPUtil.find_line(this.media[i], 'a=sendrecv', this.session)) {
  1583. elem.attrs({senders: 'both'});
  1584. } else if (SDPUtil.find_line(this.media[i], 'a=sendonly', this.session)) {
  1585. elem.attrs({senders: 'initiator'});
  1586. } else if (SDPUtil.find_line(this.media[i], 'a=recvonly', this.session)) {
  1587. elem.attrs({senders: 'responder'});
  1588. } else if (SDPUtil.find_line(this.media[i], 'a=inactive', this.session)) {
  1589. elem.attrs({senders: 'none'});
  1590. }
  1591. if (mline.port == '0') {
  1592. // estos hack to reject an m-line
  1593. elem.attrs({senders: 'rejected'});
  1594. }
  1595. elem.up(); // end of content
  1596. }
  1597. elem.up();
  1598. return elem;
  1599. };
  1600. SDP.prototype.TransportToJingle = function (mediaindex, elem) {
  1601. var i = mediaindex;
  1602. var tmp;
  1603. var self = this;
  1604. elem.c('transport');
  1605. // XEP-0343 DTLS/SCTP
  1606. if (SDPUtil.find_line(this.media[mediaindex], 'a=sctpmap:').length)
  1607. {
  1608. var sctpmap = SDPUtil.find_line(
  1609. this.media[i], 'a=sctpmap:', self.session);
  1610. if (sctpmap)
  1611. {
  1612. var sctpAttrs = SDPUtil.parse_sctpmap(sctpmap);
  1613. elem.c('sctpmap',
  1614. {
  1615. xmlns: 'urn:xmpp:jingle:transports:dtls-sctp:1',
  1616. number: sctpAttrs[0], /* SCTP port */
  1617. protocol: sctpAttrs[1], /* protocol */
  1618. });
  1619. // Optional stream count attribute
  1620. if (sctpAttrs.length > 2)
  1621. elem.attrs({ streams: sctpAttrs[2]});
  1622. elem.up();
  1623. }
  1624. }
  1625. // XEP-0320
  1626. var fingerprints = SDPUtil.find_lines(this.media[mediaindex], 'a=fingerprint:', this.session);
  1627. fingerprints.forEach(function(line) {
  1628. tmp = SDPUtil.parse_fingerprint(line);
  1629. tmp.xmlns = 'urn:xmpp:jingle:apps:dtls:0';
  1630. elem.c('fingerprint').t(tmp.fingerprint);
  1631. delete tmp.fingerprint;
  1632. line = SDPUtil.find_line(self.media[mediaindex], 'a=setup:', self.session);
  1633. if (line) {
  1634. tmp.setup = line.substr(8);
  1635. }
  1636. elem.attrs(tmp);
  1637. elem.up(); // end of fingerprint
  1638. });
  1639. tmp = SDPUtil.iceparams(this.media[mediaindex], this.session);
  1640. if (tmp) {
  1641. tmp.xmlns = 'urn:xmpp:jingle:transports:ice-udp:1';
  1642. elem.attrs(tmp);
  1643. // XEP-0176
  1644. if (SDPUtil.find_line(this.media[mediaindex], 'a=candidate:', this.session)) { // add any a=candidate lines
  1645. var lines = SDPUtil.find_lines(this.media[mediaindex], 'a=candidate:', this.session);
  1646. lines.forEach(function (line) {
  1647. elem.c('candidate', SDPUtil.candidateToJingle(line)).up();
  1648. });
  1649. }
  1650. }
  1651. elem.up(); // end of transport
  1652. }
  1653. SDP.prototype.RtcpFbToJingle = function (mediaindex, elem, payloadtype) { // XEP-0293
  1654. var lines = SDPUtil.find_lines(this.media[mediaindex], 'a=rtcp-fb:' + payloadtype);
  1655. lines.forEach(function (line) {
  1656. var tmp = SDPUtil.parse_rtcpfb(line);
  1657. if (tmp.type == 'trr-int') {
  1658. elem.c('rtcp-fb-trr-int', {xmlns: 'urn:xmpp:jingle:apps:rtp:rtcp-fb:0', value: tmp.params[0]});
  1659. elem.up();
  1660. } else {
  1661. elem.c('rtcp-fb', {xmlns: 'urn:xmpp:jingle:apps:rtp:rtcp-fb:0', type: tmp.type});
  1662. if (tmp.params.length > 0) {
  1663. elem.attrs({'subtype': tmp.params[0]});
  1664. }
  1665. elem.up();
  1666. }
  1667. });
  1668. };
  1669. SDP.prototype.RtcpFbFromJingle = function (elem, payloadtype) { // XEP-0293
  1670. var media = '';
  1671. var tmp = elem.find('>rtcp-fb-trr-int[xmlns="urn:xmpp:jingle:apps:rtp:rtcp-fb:0"]');
  1672. if (tmp.length) {
  1673. media += 'a=rtcp-fb:' + '*' + ' ' + 'trr-int' + ' ';
  1674. if (tmp.attr('value')) {
  1675. media += tmp.attr('value');
  1676. } else {
  1677. media += '0';
  1678. }
  1679. media += '\r\n';
  1680. }
  1681. tmp = elem.find('>rtcp-fb[xmlns="urn:xmpp:jingle:apps:rtp:rtcp-fb:0"]');
  1682. tmp.each(function () {
  1683. media += 'a=rtcp-fb:' + payloadtype + ' ' + $(this).attr('type');
  1684. if ($(this).attr('subtype')) {
  1685. media += ' ' + $(this).attr('subtype');
  1686. }
  1687. media += '\r\n';
  1688. });
  1689. return media;
  1690. };
  1691. // construct an SDP from a jingle stanza
  1692. SDP.prototype.fromJingle = function (jingle) {
  1693. var self = this;
  1694. this.raw = 'v=0\r\n' +
  1695. 'o=- ' + '1923518516' + ' 2 IN IP4 0.0.0.0\r\n' +// FIXME
  1696. 's=-\r\n' +
  1697. 't=0 0\r\n';
  1698. // http://tools.ietf.org/html/draft-ietf-mmusic-sdp-bundle-negotiation-04#section-8
  1699. if ($(jingle).find('>group[xmlns="urn:xmpp:jingle:apps:grouping:0"]').length) {
  1700. $(jingle).find('>group[xmlns="urn:xmpp:jingle:apps:grouping:0"]').each(function (idx, group) {
  1701. var contents = $(group).find('>content').map(function (idx, content) {
  1702. return content.getAttribute('name');
  1703. }).get();
  1704. if (contents.length > 0) {
  1705. self.raw += 'a=group:' + (group.getAttribute('semantics') || group.getAttribute('type')) + ' ' + contents.join(' ') + '\r\n';
  1706. }
  1707. });
  1708. }
  1709. this.session = this.raw;
  1710. jingle.find('>content').each(function () {
  1711. var m = self.jingle2media($(this));
  1712. self.media.push(m);
  1713. });
  1714. // reconstruct msid-semantic -- apparently not necessary
  1715. /*
  1716. var msid = SDPUtil.parse_ssrc(this.raw);
  1717. if (msid.hasOwnProperty('mslabel')) {
  1718. this.session += "a=msid-semantic: WMS " + msid.mslabel + "\r\n";
  1719. }
  1720. */
  1721. this.raw = this.session + this.media.join('');
  1722. };
  1723. // translate a jingle content element into an an SDP media part
  1724. SDP.prototype.jingle2media = function (content) {
  1725. var media = '',
  1726. desc = content.find('description'),
  1727. ssrc = desc.attr('ssrc'),
  1728. self = this,
  1729. tmp;
  1730. var sctp = content.find(
  1731. '>transport>sctpmap[xmlns="urn:xmpp:jingle:transports:dtls-sctp:1"]');
  1732. tmp = { media: desc.attr('media') };
  1733. tmp.port = '1';
  1734. if (content.attr('senders') == 'rejected') {
  1735. // estos hack to reject an m-line.
  1736. tmp.port = '0';
  1737. }
  1738. if (content.find('>transport>fingerprint').length || desc.find('encryption').length) {
  1739. if (sctp.length)
  1740. tmp.proto = 'DTLS/SCTP';
  1741. else
  1742. tmp.proto = 'RTP/SAVPF';
  1743. } else {
  1744. tmp.proto = 'RTP/AVPF';
  1745. }
  1746. if (!sctp.length)
  1747. {
  1748. tmp.fmt = desc.find('payload-type').map(
  1749. function () { return this.getAttribute('id'); }).get();
  1750. media += SDPUtil.build_mline(tmp) + '\r\n';
  1751. }
  1752. else
  1753. {
  1754. media += 'm=application 1 DTLS/SCTP ' + sctp.attr('number') + '\r\n';
  1755. media += 'a=sctpmap:' + sctp.attr('number') +
  1756. ' ' + sctp.attr('protocol');
  1757. var streamCount = sctp.attr('streams');
  1758. if (streamCount)
  1759. media += ' ' + streamCount + '\r\n';
  1760. else
  1761. media += '\r\n';
  1762. }
  1763. media += 'c=IN IP4 0.0.0.0\r\n';
  1764. if (!sctp.length)
  1765. media += 'a=rtcp:1 IN IP4 0.0.0.0\r\n';
  1766. tmp = content.find('>transport[xmlns="urn:xmpp:jingle:transports:ice-udp:1"]');
  1767. if (tmp.length) {
  1768. if (tmp.attr('ufrag')) {
  1769. media += SDPUtil.build_iceufrag(tmp.attr('ufrag')) + '\r\n';
  1770. }
  1771. if (tmp.attr('pwd')) {
  1772. media += SDPUtil.build_icepwd(tmp.attr('pwd')) + '\r\n';
  1773. }
  1774. tmp.find('>fingerprint').each(function () {
  1775. // FIXME: check namespace at some point
  1776. media += 'a=fingerprint:' + this.getAttribute('hash');
  1777. media += ' ' + $(this).text();
  1778. media += '\r\n';
  1779. if (this.getAttribute('setup')) {
  1780. media += 'a=setup:' + this.getAttribute('setup') + '\r\n';
  1781. }
  1782. });
  1783. }
  1784. switch (content.attr('senders')) {
  1785. case 'initiator':
  1786. media += 'a=sendonly\r\n';
  1787. break;
  1788. case 'responder':
  1789. media += 'a=recvonly\r\n';
  1790. break;
  1791. case 'none':
  1792. media += 'a=inactive\r\n';
  1793. break;
  1794. case 'both':
  1795. media += 'a=sendrecv\r\n';
  1796. break;
  1797. }
  1798. media += 'a=mid:' + content.attr('name') + '\r\n';
  1799. // <description><rtcp-mux/></description>
  1800. // see http://code.google.com/p/libjingle/issues/detail?id=309 -- no spec though
  1801. // and http://mail.jabber.org/pipermail/jingle/2011-December/001761.html
  1802. if (desc.find('rtcp-mux').length) {
  1803. media += 'a=rtcp-mux\r\n';
  1804. }
  1805. if (desc.find('encryption').length) {
  1806. desc.find('encryption>crypto').each(function () {
  1807. media += 'a=crypto:' + this.getAttribute('tag');
  1808. media += ' ' + this.getAttribute('crypto-suite');
  1809. media += ' ' + this.getAttribute('key-params');
  1810. if (this.getAttribute('session-params')) {
  1811. media += ' ' + this.getAttribute('session-params');
  1812. }
  1813. media += '\r\n';
  1814. });
  1815. }
  1816. desc.find('payload-type').each(function () {
  1817. media += SDPUtil.build_rtpmap(this) + '\r\n';
  1818. if ($(this).find('>parameter').length) {
  1819. media += 'a=fmtp:' + this.getAttribute('id') + ' ';
  1820. media += $(this).find('parameter').map(function () { return (this.getAttribute('name') ? (this.getAttribute('name') + '=') : '') + this.getAttribute('value'); }).get().join('; ');
  1821. media += '\r\n';
  1822. }
  1823. // xep-0293
  1824. media += self.RtcpFbFromJingle($(this), this.getAttribute('id'));
  1825. });
  1826. // xep-0293
  1827. media += self.RtcpFbFromJingle(desc, '*');
  1828. // xep-0294
  1829. tmp = desc.find('>rtp-hdrext[xmlns="urn:xmpp:jingle:apps:rtp:rtp-hdrext:0"]');
  1830. tmp.each(function () {
  1831. media += 'a=extmap:' + this.getAttribute('id') + ' ' + this.getAttribute('uri') + '\r\n';
  1832. });
  1833. content.find('>transport[xmlns="urn:xmpp:jingle:transports:ice-udp:1"]>candidate').each(function () {
  1834. media += SDPUtil.candidateFromJingle(this);
  1835. });
  1836. // XEP-0339 handle ssrc-group attributes
  1837. tmp = content.find('description>ssrc-group[xmlns="urn:xmpp:jingle:apps:rtp:ssma:0"]').each(function() {
  1838. var semantics = this.getAttribute('semantics');
  1839. var ssrcs = $(this).find('>source').map(function() {
  1840. return this.getAttribute('ssrc');
  1841. }).get();
  1842. if (ssrcs.length != 0) {
  1843. media += 'a=ssrc-group:' + semantics + ' ' + ssrcs.join(' ') + '\r\n';
  1844. }
  1845. });
  1846. tmp = content.find('description>source[xmlns="urn:xmpp:jingle:apps:rtp:ssma:0"]');
  1847. tmp.each(function () {
  1848. var ssrc = this.getAttribute('ssrc');
  1849. $(this).find('>parameter').each(function () {
  1850. media += 'a=ssrc:' + ssrc + ' ' + this.getAttribute('name');
  1851. if (this.getAttribute('value') && this.getAttribute('value').length)
  1852. media += ':' + this.getAttribute('value');
  1853. media += '\r\n';
  1854. });
  1855. });
  1856. return media;
  1857. };
  1858. module.exports = SDP;
  1859. },{"./SDPUtil":4}],3:[function(require,module,exports){
  1860. function SDPDiffer(mySDP, otherSDP) {
  1861. this.mySDP = mySDP;
  1862. this.otherSDP = otherSDP;
  1863. }
  1864. /**
  1865. * Returns map of MediaChannel that contains only media not contained in <tt>otherSdp</tt>. Mapped by channel idx.
  1866. * @param otherSdp the other SDP to check ssrc with.
  1867. */
  1868. SDPDiffer.prototype.getNewMedia = function() {
  1869. // this could be useful in Array.prototype.
  1870. function arrayEquals(array) {
  1871. // if the other array is a falsy value, return
  1872. if (!array)
  1873. return false;
  1874. // compare lengths - can save a lot of time
  1875. if (this.length != array.length)
  1876. return false;
  1877. for (var i = 0, l=this.length; i < l; i++) {
  1878. // Check if we have nested arrays
  1879. if (this[i] instanceof Array && array[i] instanceof Array) {
  1880. // recurse into the nested arrays
  1881. if (!this[i].equals(array[i]))
  1882. return false;
  1883. }
  1884. else if (this[i] != array[i]) {
  1885. // Warning - two different object instances will never be equal: {x:20} != {x:20}
  1886. return false;
  1887. }
  1888. }
  1889. return true;
  1890. }
  1891. var myMedias = this.mySDP.getMediaSsrcMap();
  1892. var othersMedias = this.otherSDP.getMediaSsrcMap();
  1893. var newMedia = {};
  1894. Object.keys(othersMedias).forEach(function(othersMediaIdx) {
  1895. var myMedia = myMedias[othersMediaIdx];
  1896. var othersMedia = othersMedias[othersMediaIdx];
  1897. if(!myMedia && othersMedia) {
  1898. // Add whole channel
  1899. newMedia[othersMediaIdx] = othersMedia;
  1900. return;
  1901. }
  1902. // Look for new ssrcs accross the channel
  1903. Object.keys(othersMedia.ssrcs).forEach(function(ssrc) {
  1904. if(Object.keys(myMedia.ssrcs).indexOf(ssrc) === -1) {
  1905. // Allocate channel if we've found ssrc that doesn't exist in our channel
  1906. if(!newMedia[othersMediaIdx]){
  1907. newMedia[othersMediaIdx] = {
  1908. mediaindex: othersMedia.mediaindex,
  1909. mid: othersMedia.mid,
  1910. ssrcs: {},
  1911. ssrcGroups: []
  1912. };
  1913. }
  1914. newMedia[othersMediaIdx].ssrcs[ssrc] = othersMedia.ssrcs[ssrc];
  1915. }
  1916. });
  1917. // Look for new ssrc groups across the channels
  1918. othersMedia.ssrcGroups.forEach(function(otherSsrcGroup){
  1919. // try to match the other ssrc-group with an ssrc-group of ours
  1920. var matched = false;
  1921. for (var i = 0; i < myMedia.ssrcGroups.length; i++) {
  1922. var mySsrcGroup = myMedia.ssrcGroups[i];
  1923. if (otherSsrcGroup.semantics == mySsrcGroup.semantics
  1924. && arrayEquals.apply(otherSsrcGroup.ssrcs, [mySsrcGroup.ssrcs])) {
  1925. matched = true;
  1926. break;
  1927. }
  1928. }
  1929. if (!matched) {
  1930. // Allocate channel if we've found an ssrc-group that doesn't
  1931. // exist in our channel
  1932. if(!newMedia[othersMediaIdx]){
  1933. newMedia[othersMediaIdx] = {
  1934. mediaindex: othersMedia.mediaindex,
  1935. mid: othersMedia.mid,
  1936. ssrcs: {},
  1937. ssrcGroups: []
  1938. };
  1939. }
  1940. newMedia[othersMediaIdx].ssrcGroups.push(otherSsrcGroup);
  1941. }
  1942. });
  1943. });
  1944. return newMedia;
  1945. };
  1946. /**
  1947. * Sends SSRC update IQ.
  1948. * @param sdpMediaSsrcs SSRCs map obtained from SDP.getNewMedia. Cntains SSRCs to add/remove.
  1949. * @param sid session identifier that will be put into the IQ.
  1950. * @param initiator initiator identifier.
  1951. * @param toJid destination Jid
  1952. * @param isAdd indicates if this is remove or add operation.
  1953. */
  1954. SDPDiffer.prototype.toJingle = function(modify) {
  1955. var sdpMediaSsrcs = this.getNewMedia();
  1956. var self = this;
  1957. // FIXME: only announce video ssrcs since we mix audio and dont need
  1958. // the audio ssrcs therefore
  1959. var modified = false;
  1960. Object.keys(sdpMediaSsrcs).forEach(function(mediaindex){
  1961. modified = true;
  1962. var media = sdpMediaSsrcs[mediaindex];
  1963. modify.c('content', {name: media.mid});
  1964. modify.c('description', {xmlns:'urn:xmpp:jingle:apps:rtp:1', media: media.mid});
  1965. // FIXME: not completly sure this operates on blocks and / or handles different ssrcs correctly
  1966. // generate sources from lines
  1967. Object.keys(media.ssrcs).forEach(function(ssrcNum) {
  1968. var mediaSsrc = media.ssrcs[ssrcNum];
  1969. modify.c('source', { xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });
  1970. modify.attrs({ssrc: mediaSsrc.ssrc});
  1971. // iterate over ssrc lines
  1972. mediaSsrc.lines.forEach(function (line) {
  1973. var idx = line.indexOf(' ');
  1974. var kv = line.substr(idx + 1);
  1975. modify.c('parameter');
  1976. if (kv.indexOf(':') == -1) {
  1977. modify.attrs({ name: kv });
  1978. } else {
  1979. modify.attrs({ name: kv.split(':', 2)[0] });
  1980. modify.attrs({ value: kv.split(':', 2)[1] });
  1981. }
  1982. modify.up(); // end of parameter
  1983. });
  1984. modify.up(); // end of source
  1985. });
  1986. // generate source groups from lines
  1987. media.ssrcGroups.forEach(function(ssrcGroup) {
  1988. if (ssrcGroup.ssrcs.length != 0) {
  1989. modify.c('ssrc-group', {
  1990. semantics: ssrcGroup.semantics,
  1991. xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0'
  1992. });
  1993. ssrcGroup.ssrcs.forEach(function (ssrc) {
  1994. modify.c('source', { ssrc: ssrc })
  1995. .up(); // end of source
  1996. });
  1997. modify.up(); // end of ssrc-group
  1998. }
  1999. });
  2000. modify.up(); // end of description
  2001. modify.up(); // end of content
  2002. });
  2003. return modified;
  2004. };
  2005. module.exports = SDPDiffer;
  2006. },{}],4:[function(require,module,exports){
  2007. SDPUtil = {
  2008. iceparams: function (mediadesc, sessiondesc) {
  2009. var data = null;
  2010. if (SDPUtil.find_line(mediadesc, 'a=ice-ufrag:', sessiondesc) &&
  2011. SDPUtil.find_line(mediadesc, 'a=ice-pwd:', sessiondesc)) {
  2012. data = {
  2013. ufrag: SDPUtil.parse_iceufrag(SDPUtil.find_line(mediadesc, 'a=ice-ufrag:', sessiondesc)),
  2014. pwd: SDPUtil.parse_icepwd(SDPUtil.find_line(mediadesc, 'a=ice-pwd:', sessiondesc))
  2015. };
  2016. }
  2017. return data;
  2018. },
  2019. parse_iceufrag: function (line) {
  2020. return line.substring(12);
  2021. },
  2022. build_iceufrag: function (frag) {
  2023. return 'a=ice-ufrag:' + frag;
  2024. },
  2025. parse_icepwd: function (line) {
  2026. return line.substring(10);
  2027. },
  2028. build_icepwd: function (pwd) {
  2029. return 'a=ice-pwd:' + pwd;
  2030. },
  2031. parse_mid: function (line) {
  2032. return line.substring(6);
  2033. },
  2034. parse_mline: function (line) {
  2035. var parts = line.substring(2).split(' '),
  2036. data = {};
  2037. data.media = parts.shift();
  2038. data.port = parts.shift();
  2039. data.proto = parts.shift();
  2040. if (parts[parts.length - 1] === '') { // trailing whitespace
  2041. parts.pop();
  2042. }
  2043. data.fmt = parts;
  2044. return data;
  2045. },
  2046. build_mline: function (mline) {
  2047. return 'm=' + mline.media + ' ' + mline.port + ' ' + mline.proto + ' ' + mline.fmt.join(' ');
  2048. },
  2049. parse_rtpmap: function (line) {
  2050. var parts = line.substring(9).split(' '),
  2051. data = {};
  2052. data.id = parts.shift();
  2053. parts = parts[0].split('/');
  2054. data.name = parts.shift();
  2055. data.clockrate = parts.shift();
  2056. data.channels = parts.length ? parts.shift() : '1';
  2057. return data;
  2058. },
  2059. /**
  2060. * Parses SDP line "a=sctpmap:..." and extracts SCTP port from it.
  2061. * @param line eg. "a=sctpmap:5000 webrtc-datachannel"
  2062. * @returns [SCTP port number, protocol, streams]
  2063. */
  2064. parse_sctpmap: function (line)
  2065. {
  2066. var parts = line.substring(10).split(' ');
  2067. var sctpPort = parts[0];
  2068. var protocol = parts[1];
  2069. // Stream count is optional
  2070. var streamCount = parts.length > 2 ? parts[2] : null;
  2071. return [sctpPort, protocol, streamCount];// SCTP port
  2072. },
  2073. build_rtpmap: function (el) {
  2074. var line = 'a=rtpmap:' + el.getAttribute('id') + ' ' + el.getAttribute('name') + '/' + el.getAttribute('clockrate');
  2075. if (el.getAttribute('channels') && el.getAttribute('channels') != '1') {
  2076. line += '/' + el.getAttribute('channels');
  2077. }
  2078. return line;
  2079. },
  2080. parse_crypto: function (line) {
  2081. var parts = line.substring(9).split(' '),
  2082. data = {};
  2083. data.tag = parts.shift();
  2084. data['crypto-suite'] = parts.shift();
  2085. data['key-params'] = parts.shift();
  2086. if (parts.length) {
  2087. data['session-params'] = parts.join(' ');
  2088. }
  2089. return data;
  2090. },
  2091. parse_fingerprint: function (line) { // RFC 4572
  2092. var parts = line.substring(14).split(' '),
  2093. data = {};
  2094. data.hash = parts.shift();
  2095. data.fingerprint = parts.shift();
  2096. // TODO assert that fingerprint satisfies 2UHEX *(":" 2UHEX) ?
  2097. return data;
  2098. },
  2099. parse_fmtp: function (line) {
  2100. var parts = line.split(' '),
  2101. i, key, value,
  2102. data = [];
  2103. parts.shift();
  2104. parts = parts.join(' ').split(';');
  2105. for (i = 0; i < parts.length; i++) {
  2106. key = parts[i].split('=')[0];
  2107. while (key.length && key[0] == ' ') {
  2108. key = key.substring(1);
  2109. }
  2110. value = parts[i].split('=')[1];
  2111. if (key && value) {
  2112. data.push({name: key, value: value});
  2113. } else if (key) {
  2114. // rfc 4733 (DTMF) style stuff
  2115. data.push({name: '', value: key});
  2116. }
  2117. }
  2118. return data;
  2119. },
  2120. parse_icecandidate: function (line) {
  2121. var candidate = {},
  2122. elems = line.split(' ');
  2123. candidate.foundation = elems[0].substring(12);
  2124. candidate.component = elems[1];
  2125. candidate.protocol = elems[2].toLowerCase();
  2126. candidate.priority = elems[3];
  2127. candidate.ip = elems[4];
  2128. candidate.port = elems[5];
  2129. // elems[6] => "typ"
  2130. candidate.type = elems[7];
  2131. candidate.generation = 0; // default value, may be overwritten below
  2132. for (var i = 8; i < elems.length; i += 2) {
  2133. switch (elems[i]) {
  2134. case 'raddr':
  2135. candidate['rel-addr'] = elems[i + 1];
  2136. break;
  2137. case 'rport':
  2138. candidate['rel-port'] = elems[i + 1];
  2139. break;
  2140. case 'generation':
  2141. candidate.generation = elems[i + 1];
  2142. break;
  2143. case 'tcptype':
  2144. candidate.tcptype = elems[i + 1];
  2145. break;
  2146. default: // TODO
  2147. console.log('parse_icecandidate not translating "' + elems[i] + '" = "' + elems[i + 1] + '"');
  2148. }
  2149. }
  2150. candidate.network = '1';
  2151. candidate.id = Math.random().toString(36).substr(2, 10); // not applicable to SDP -- FIXME: should be unique, not just random
  2152. return candidate;
  2153. },
  2154. build_icecandidate: function (cand) {
  2155. var line = ['a=candidate:' + cand.foundation, cand.component, cand.protocol, cand.priority, cand.ip, cand.port, 'typ', cand.type].join(' ');
  2156. line += ' ';
  2157. switch (cand.type) {
  2158. case 'srflx':
  2159. case 'prflx':
  2160. case 'relay':
  2161. if (cand.hasOwnAttribute('rel-addr') && cand.hasOwnAttribute('rel-port')) {
  2162. line += 'raddr';
  2163. line += ' ';
  2164. line += cand['rel-addr'];
  2165. line += ' ';
  2166. line += 'rport';
  2167. line += ' ';
  2168. line += cand['rel-port'];
  2169. line += ' ';
  2170. }
  2171. break;
  2172. }
  2173. if (cand.hasOwnAttribute('tcptype')) {
  2174. line += 'tcptype';
  2175. line += ' ';
  2176. line += cand.tcptype;
  2177. line += ' ';
  2178. }
  2179. line += 'generation';
  2180. line += ' ';
  2181. line += cand.hasOwnAttribute('generation') ? cand.generation : '0';
  2182. return line;
  2183. },
  2184. parse_ssrc: function (desc) {
  2185. // proprietary mapping of a=ssrc lines
  2186. // TODO: see "Jingle RTP Source Description" by Juberti and P. Thatcher on google docs
  2187. // and parse according to that
  2188. var lines = desc.split('\r\n'),
  2189. data = {};
  2190. for (var i = 0; i < lines.length; i++) {
  2191. if (lines[i].substring(0, 7) == 'a=ssrc:') {
  2192. var idx = lines[i].indexOf(' ');
  2193. data[lines[i].substr(idx + 1).split(':', 2)[0]] = lines[i].substr(idx + 1).split(':', 2)[1];
  2194. }
  2195. }
  2196. return data;
  2197. },
  2198. parse_rtcpfb: function (line) {
  2199. var parts = line.substr(10).split(' ');
  2200. var data = {};
  2201. data.pt = parts.shift();
  2202. data.type = parts.shift();
  2203. data.params = parts;
  2204. return data;
  2205. },
  2206. parse_extmap: function (line) {
  2207. var parts = line.substr(9).split(' ');
  2208. var data = {};
  2209. data.value = parts.shift();
  2210. if (data.value.indexOf('/') != -1) {
  2211. data.direction = data.value.substr(data.value.indexOf('/') + 1);
  2212. data.value = data.value.substr(0, data.value.indexOf('/'));
  2213. } else {
  2214. data.direction = 'both';
  2215. }
  2216. data.uri = parts.shift();
  2217. data.params = parts;
  2218. return data;
  2219. },
  2220. find_line: function (haystack, needle, sessionpart) {
  2221. var lines = haystack.split('\r\n');
  2222. for (var i = 0; i < lines.length; i++) {
  2223. if (lines[i].substring(0, needle.length) == needle) {
  2224. return lines[i];
  2225. }
  2226. }
  2227. if (!sessionpart) {
  2228. return false;
  2229. }
  2230. // search session part
  2231. lines = sessionpart.split('\r\n');
  2232. for (var j = 0; j < lines.length; j++) {
  2233. if (lines[j].substring(0, needle.length) == needle) {
  2234. return lines[j];
  2235. }
  2236. }
  2237. return false;
  2238. },
  2239. find_lines: function (haystack, needle, sessionpart) {
  2240. var lines = haystack.split('\r\n'),
  2241. needles = [];
  2242. for (var i = 0; i < lines.length; i++) {
  2243. if (lines[i].substring(0, needle.length) == needle)
  2244. needles.push(lines[i]);
  2245. }
  2246. if (needles.length || !sessionpart) {
  2247. return needles;
  2248. }
  2249. // search session part
  2250. lines = sessionpart.split('\r\n');
  2251. for (var j = 0; j < lines.length; j++) {
  2252. if (lines[j].substring(0, needle.length) == needle) {
  2253. needles.push(lines[j]);
  2254. }
  2255. }
  2256. return needles;
  2257. },
  2258. candidateToJingle: function (line) {
  2259. // a=candidate:2979166662 1 udp 2113937151 192.168.2.100 57698 typ host generation 0
  2260. // <candidate component=... foundation=... generation=... id=... ip=... network=... port=... priority=... protocol=... type=.../>
  2261. if (line.indexOf('candidate:') === 0) {
  2262. line = 'a=' + line;
  2263. } else if (line.substring(0, 12) != 'a=candidate:') {
  2264. console.log('parseCandidate called with a line that is not a candidate line');
  2265. console.log(line);
  2266. return null;
  2267. }
  2268. if (line.substring(line.length - 2) == '\r\n') // chomp it
  2269. line = line.substring(0, line.length - 2);
  2270. var candidate = {},
  2271. elems = line.split(' '),
  2272. i;
  2273. if (elems[6] != 'typ') {
  2274. console.log('did not find typ in the right place');
  2275. console.log(line);
  2276. return null;
  2277. }
  2278. candidate.foundation = elems[0].substring(12);
  2279. candidate.component = elems[1];
  2280. candidate.protocol = elems[2].toLowerCase();
  2281. candidate.priority = elems[3];
  2282. candidate.ip = elems[4];
  2283. candidate.port = elems[5];
  2284. // elems[6] => "typ"
  2285. candidate.type = elems[7];
  2286. candidate.generation = '0'; // default, may be overwritten below
  2287. for (i = 8; i < elems.length; i += 2) {
  2288. switch (elems[i]) {
  2289. case 'raddr':
  2290. candidate['rel-addr'] = elems[i + 1];
  2291. break;
  2292. case 'rport':
  2293. candidate['rel-port'] = elems[i + 1];
  2294. break;
  2295. case 'generation':
  2296. candidate.generation = elems[i + 1];
  2297. break;
  2298. case 'tcptype':
  2299. candidate.tcptype = elems[i + 1];
  2300. break;
  2301. default: // TODO
  2302. console.log('not translating "' + elems[i] + '" = "' + elems[i + 1] + '"');
  2303. }
  2304. }
  2305. candidate.network = '1';
  2306. candidate.id = Math.random().toString(36).substr(2, 10); // not applicable to SDP -- FIXME: should be unique, not just random
  2307. return candidate;
  2308. },
  2309. candidateFromJingle: function (cand) {
  2310. var line = 'a=candidate:';
  2311. line += cand.getAttribute('foundation');
  2312. line += ' ';
  2313. line += cand.getAttribute('component');
  2314. line += ' ';
  2315. line += cand.getAttribute('protocol'); //.toUpperCase(); // chrome M23 doesn't like this
  2316. line += ' ';
  2317. line += cand.getAttribute('priority');
  2318. line += ' ';
  2319. line += cand.getAttribute('ip');
  2320. line += ' ';
  2321. line += cand.getAttribute('port');
  2322. line += ' ';
  2323. line += 'typ';
  2324. line += ' ' + cand.getAttribute('type');
  2325. line += ' ';
  2326. switch (cand.getAttribute('type')) {
  2327. case 'srflx':
  2328. case 'prflx':
  2329. case 'relay':
  2330. if (cand.getAttribute('rel-addr') && cand.getAttribute('rel-port')) {
  2331. line += 'raddr';
  2332. line += ' ';
  2333. line += cand.getAttribute('rel-addr');
  2334. line += ' ';
  2335. line += 'rport';
  2336. line += ' ';
  2337. line += cand.getAttribute('rel-port');
  2338. line += ' ';
  2339. }
  2340. break;
  2341. }
  2342. if (cand.getAttribute('protocol').toLowerCase() == 'tcp') {
  2343. line += 'tcptype';
  2344. line += ' ';
  2345. line += cand.getAttribute('tcptype');
  2346. line += ' ';
  2347. }
  2348. line += 'generation';
  2349. line += ' ';
  2350. line += cand.getAttribute('generation') || '0';
  2351. return line + '\r\n';
  2352. }
  2353. };
  2354. module.exports = SDPUtil;
  2355. },{}],5:[function(require,module,exports){
  2356. function TraceablePeerConnection(ice_config, constraints) {
  2357. var self = this;
  2358. var RTCPeerconnection = navigator.mozGetUserMedia ? mozRTCPeerConnection : webkitRTCPeerConnection;
  2359. this.peerconnection = new RTCPeerconnection(ice_config, constraints);
  2360. this.updateLog = [];
  2361. this.stats = {};
  2362. this.statsinterval = null;
  2363. this.maxstats = 0; // limit to 300 values, i.e. 5 minutes; set to 0 to disable
  2364. // override as desired
  2365. this.trace = function (what, info) {
  2366. //console.warn('WTRACE', what, info);
  2367. self.updateLog.push({
  2368. time: new Date(),
  2369. type: what,
  2370. value: info || ""
  2371. });
  2372. };
  2373. this.onicecandidate = null;
  2374. this.peerconnection.onicecandidate = function (event) {
  2375. self.trace('onicecandidate', JSON.stringify(event.candidate, null, ' '));
  2376. if (self.onicecandidate !== null) {
  2377. self.onicecandidate(event);
  2378. }
  2379. };
  2380. this.onaddstream = null;
  2381. this.peerconnection.onaddstream = function (event) {
  2382. self.trace('onaddstream', event.stream.id);
  2383. if (self.onaddstream !== null) {
  2384. self.onaddstream(event);
  2385. }
  2386. };
  2387. this.onremovestream = null;
  2388. this.peerconnection.onremovestream = function (event) {
  2389. self.trace('onremovestream', event.stream.id);
  2390. if (self.onremovestream !== null) {
  2391. self.onremovestream(event);
  2392. }
  2393. };
  2394. this.onsignalingstatechange = null;
  2395. this.peerconnection.onsignalingstatechange = function (event) {
  2396. self.trace('onsignalingstatechange', self.signalingState);
  2397. if (self.onsignalingstatechange !== null) {
  2398. self.onsignalingstatechange(event);
  2399. }
  2400. };
  2401. this.oniceconnectionstatechange = null;
  2402. this.peerconnection.oniceconnectionstatechange = function (event) {
  2403. self.trace('oniceconnectionstatechange', self.iceConnectionState);
  2404. if (self.oniceconnectionstatechange !== null) {
  2405. self.oniceconnectionstatechange(event);
  2406. }
  2407. };
  2408. this.onnegotiationneeded = null;
  2409. this.peerconnection.onnegotiationneeded = function (event) {
  2410. self.trace('onnegotiationneeded');
  2411. if (self.onnegotiationneeded !== null) {
  2412. self.onnegotiationneeded(event);
  2413. }
  2414. };
  2415. self.ondatachannel = null;
  2416. this.peerconnection.ondatachannel = function (event) {
  2417. self.trace('ondatachannel', event);
  2418. if (self.ondatachannel !== null) {
  2419. self.ondatachannel(event);
  2420. }
  2421. };
  2422. if (!navigator.mozGetUserMedia && this.maxstats) {
  2423. this.statsinterval = window.setInterval(function() {
  2424. self.peerconnection.getStats(function(stats) {
  2425. var results = stats.result();
  2426. for (var i = 0; i < results.length; ++i) {
  2427. //console.log(results[i].type, results[i].id, results[i].names())
  2428. var now = new Date();
  2429. results[i].names().forEach(function (name) {
  2430. var id = results[i].id + '-' + name;
  2431. if (!self.stats[id]) {
  2432. self.stats[id] = {
  2433. startTime: now,
  2434. endTime: now,
  2435. values: [],
  2436. times: []
  2437. };
  2438. }
  2439. self.stats[id].values.push(results[i].stat(name));
  2440. self.stats[id].times.push(now.getTime());
  2441. if (self.stats[id].values.length > self.maxstats) {
  2442. self.stats[id].values.shift();
  2443. self.stats[id].times.shift();
  2444. }
  2445. self.stats[id].endTime = now;
  2446. });
  2447. }
  2448. });
  2449. }, 1000);
  2450. }
  2451. };
  2452. dumpSDP = function(description) {
  2453. return 'type: ' + description.type + '\r\n' + description.sdp;
  2454. }
  2455. if (TraceablePeerConnection.prototype.__defineGetter__ !== undefined) {
  2456. TraceablePeerConnection.prototype.__defineGetter__('signalingState', function() { return this.peerconnection.signalingState; });
  2457. TraceablePeerConnection.prototype.__defineGetter__('iceConnectionState', function() { return this.peerconnection.iceConnectionState; });
  2458. TraceablePeerConnection.prototype.__defineGetter__('localDescription', function() {
  2459. var publicLocalDescription = simulcast.reverseTransformLocalDescription(this.peerconnection.localDescription);
  2460. return publicLocalDescription;
  2461. });
  2462. TraceablePeerConnection.prototype.__defineGetter__('remoteDescription', function() {
  2463. var publicRemoteDescription = simulcast.reverseTransformRemoteDescription(this.peerconnection.remoteDescription);
  2464. return publicRemoteDescription;
  2465. });
  2466. }
  2467. TraceablePeerConnection.prototype.addStream = function (stream) {
  2468. this.trace('addStream', stream.id);
  2469. simulcast.resetSender();
  2470. try
  2471. {
  2472. this.peerconnection.addStream(stream);
  2473. }
  2474. catch (e)
  2475. {
  2476. console.error(e);
  2477. return;
  2478. }
  2479. };
  2480. TraceablePeerConnection.prototype.removeStream = function (stream, stopStreams) {
  2481. this.trace('removeStream', stream.id);
  2482. simulcast.resetSender();
  2483. if(stopStreams) {
  2484. stream.getAudioTracks().forEach(function (track) {
  2485. track.stop();
  2486. });
  2487. stream.getVideoTracks().forEach(function (track) {
  2488. track.stop();
  2489. });
  2490. }
  2491. this.peerconnection.removeStream(stream);
  2492. };
  2493. TraceablePeerConnection.prototype.createDataChannel = function (label, opts) {
  2494. this.trace('createDataChannel', label, opts);
  2495. return this.peerconnection.createDataChannel(label, opts);
  2496. };
  2497. TraceablePeerConnection.prototype.setLocalDescription = function (description, successCallback, failureCallback) {
  2498. var self = this;
  2499. description = simulcast.transformLocalDescription(description);
  2500. this.trace('setLocalDescription', dumpSDP(description));
  2501. this.peerconnection.setLocalDescription(description,
  2502. function () {
  2503. self.trace('setLocalDescriptionOnSuccess');
  2504. successCallback();
  2505. },
  2506. function (err) {
  2507. self.trace('setLocalDescriptionOnFailure', err);
  2508. failureCallback(err);
  2509. }
  2510. );
  2511. /*
  2512. if (this.statsinterval === null && this.maxstats > 0) {
  2513. // start gathering stats
  2514. }
  2515. */
  2516. };
  2517. TraceablePeerConnection.prototype.setRemoteDescription = function (description, successCallback, failureCallback) {
  2518. var self = this;
  2519. description = simulcast.transformRemoteDescription(description);
  2520. this.trace('setRemoteDescription', dumpSDP(description));
  2521. this.peerconnection.setRemoteDescription(description,
  2522. function () {
  2523. self.trace('setRemoteDescriptionOnSuccess');
  2524. successCallback();
  2525. },
  2526. function (err) {
  2527. self.trace('setRemoteDescriptionOnFailure', err);
  2528. failureCallback(err);
  2529. }
  2530. );
  2531. /*
  2532. if (this.statsinterval === null && this.maxstats > 0) {
  2533. // start gathering stats
  2534. }
  2535. */
  2536. };
  2537. TraceablePeerConnection.prototype.close = function () {
  2538. this.trace('stop');
  2539. if (this.statsinterval !== null) {
  2540. window.clearInterval(this.statsinterval);
  2541. this.statsinterval = null;
  2542. }
  2543. this.peerconnection.close();
  2544. };
  2545. TraceablePeerConnection.prototype.createOffer = function (successCallback, failureCallback, constraints) {
  2546. var self = this;
  2547. this.trace('createOffer', JSON.stringify(constraints, null, ' '));
  2548. this.peerconnection.createOffer(
  2549. function (offer) {
  2550. self.trace('createOfferOnSuccess', dumpSDP(offer));
  2551. successCallback(offer);
  2552. },
  2553. function(err) {
  2554. self.trace('createOfferOnFailure', err);
  2555. failureCallback(err);
  2556. },
  2557. constraints
  2558. );
  2559. };
  2560. TraceablePeerConnection.prototype.createAnswer = function (successCallback, failureCallback, constraints) {
  2561. var self = this;
  2562. this.trace('createAnswer', JSON.stringify(constraints, null, ' '));
  2563. this.peerconnection.createAnswer(
  2564. function (answer) {
  2565. answer = simulcast.transformAnswer(answer);
  2566. self.trace('createAnswerOnSuccess', dumpSDP(answer));
  2567. successCallback(answer);
  2568. },
  2569. function(err) {
  2570. self.trace('createAnswerOnFailure', err);
  2571. failureCallback(err);
  2572. },
  2573. constraints
  2574. );
  2575. };
  2576. TraceablePeerConnection.prototype.addIceCandidate = function (candidate, successCallback, failureCallback) {
  2577. var self = this;
  2578. this.trace('addIceCandidate', JSON.stringify(candidate, null, ' '));
  2579. this.peerconnection.addIceCandidate(candidate);
  2580. /* maybe later
  2581. this.peerconnection.addIceCandidate(candidate,
  2582. function () {
  2583. self.trace('addIceCandidateOnSuccess');
  2584. successCallback();
  2585. },
  2586. function (err) {
  2587. self.trace('addIceCandidateOnFailure', err);
  2588. failureCallback(err);
  2589. }
  2590. );
  2591. */
  2592. };
  2593. TraceablePeerConnection.prototype.getStats = function(callback, errback) {
  2594. if (navigator.mozGetUserMedia) {
  2595. // ignore for now...
  2596. if(!errback)
  2597. errback = function () {
  2598. }
  2599. this.peerconnection.getStats(null,callback,errback);
  2600. } else {
  2601. this.peerconnection.getStats(callback);
  2602. }
  2603. };
  2604. module.exports = TraceablePeerConnection;
  2605. },{}],6:[function(require,module,exports){
  2606. /* global $, $iq, config, connection, UI, messageHandler,
  2607. roomName, sessionTerminated, Strophe, Util */
  2608. /**
  2609. * Contains logic responsible for enabling/disabling functionality available
  2610. * only to moderator users.
  2611. */
  2612. var connection = null;
  2613. var focusUserJid;
  2614. var getNextTimeout = Util.createExpBackoffTimer(1000);
  2615. var getNextErrorTimeout = Util.createExpBackoffTimer(1000);
  2616. // External authentication stuff
  2617. var externalAuthEnabled = false;
  2618. // Sip gateway can be enabled by configuring Jigasi host in config.js or
  2619. // it will be enabled automatically if focus detects the component through
  2620. // service discovery.
  2621. var sipGatewayEnabled = config.hosts.call_control !== undefined;
  2622. var Moderator = {
  2623. isModerator: function () {
  2624. return connection && connection.emuc.isModerator();
  2625. },
  2626. isPeerModerator: function (peerJid) {
  2627. return connection &&
  2628. connection.emuc.getMemberRole(peerJid) === 'moderator';
  2629. },
  2630. isExternalAuthEnabled: function () {
  2631. return externalAuthEnabled;
  2632. },
  2633. isSipGatewayEnabled: function () {
  2634. return sipGatewayEnabled;
  2635. },
  2636. setConnection: function (con) {
  2637. connection = con;
  2638. },
  2639. init: function (xmpp) {
  2640. this.xmppService = xmpp;
  2641. this.onLocalRoleChange = function (from, member, pres) {
  2642. UI.onModeratorStatusChanged(Moderator.isModerator());
  2643. };
  2644. },
  2645. onMucLeft: function (jid) {
  2646. console.info("Someone left is it focus ? " + jid);
  2647. var resource = Strophe.getResourceFromJid(jid);
  2648. if (resource === 'focus' && !this.xmppService.sessionTerminated) {
  2649. console.info(
  2650. "Focus has left the room - leaving conference");
  2651. //hangUp();
  2652. // We'd rather reload to have everything re-initialized
  2653. // FIXME: show some message before reload
  2654. location.reload();
  2655. }
  2656. },
  2657. setFocusUserJid: function (focusJid) {
  2658. if (!focusUserJid) {
  2659. focusUserJid = focusJid;
  2660. console.info("Focus jid set to: " + focusUserJid);
  2661. }
  2662. },
  2663. getFocusUserJid: function () {
  2664. return focusUserJid;
  2665. },
  2666. getFocusComponent: function () {
  2667. // Get focus component address
  2668. var focusComponent = config.hosts.focus;
  2669. // If not specified use default: 'focus.domain'
  2670. if (!focusComponent) {
  2671. focusComponent = 'focus.' + config.hosts.domain;
  2672. }
  2673. return focusComponent;
  2674. },
  2675. createConferenceIq: function (roomName) {
  2676. // Generate create conference IQ
  2677. var elem = $iq({to: Moderator.getFocusComponent(), type: 'set'});
  2678. elem.c('conference', {
  2679. xmlns: 'http://jitsi.org/protocol/focus',
  2680. room: roomName
  2681. });
  2682. if (config.hosts.bridge !== undefined) {
  2683. elem.c(
  2684. 'property',
  2685. { name: 'bridge', value: config.hosts.bridge})
  2686. .up();
  2687. }
  2688. // Tell the focus we have Jigasi configured
  2689. if (config.hosts.call_control !== undefined) {
  2690. elem.c(
  2691. 'property',
  2692. { name: 'call_control', value: config.hosts.call_control})
  2693. .up();
  2694. }
  2695. if (config.channelLastN !== undefined) {
  2696. elem.c(
  2697. 'property',
  2698. { name: 'channelLastN', value: config.channelLastN})
  2699. .up();
  2700. }
  2701. if (config.adaptiveLastN !== undefined) {
  2702. elem.c(
  2703. 'property',
  2704. { name: 'adaptiveLastN', value: config.adaptiveLastN})
  2705. .up();
  2706. }
  2707. if (config.adaptiveSimulcast !== undefined) {
  2708. elem.c(
  2709. 'property',
  2710. { name: 'adaptiveSimulcast', value: config.adaptiveSimulcast})
  2711. .up();
  2712. }
  2713. if (config.openSctp !== undefined) {
  2714. elem.c(
  2715. 'property',
  2716. { name: 'openSctp', value: config.openSctp})
  2717. .up();
  2718. }
  2719. if (config.enableFirefoxSupport !== undefined) {
  2720. elem.c(
  2721. 'property',
  2722. { name: 'enableFirefoxHacks',
  2723. value: config.enableFirefoxSupport})
  2724. .up();
  2725. }
  2726. elem.up();
  2727. return elem;
  2728. },
  2729. parseConfigOptions: function (resultIq) {
  2730. Moderator.setFocusUserJid(
  2731. $(resultIq).find('conference').attr('focusjid'));
  2732. var extAuthParam
  2733. = $(resultIq).find('>conference>property[name=\'externalAuth\']');
  2734. if (extAuthParam.length) {
  2735. externalAuthEnabled = extAuthParam.attr('value') === 'true';
  2736. }
  2737. console.info("External authentication enabled: " + externalAuthEnabled);
  2738. // Check if focus has auto-detected Jigasi component(this will be also
  2739. // included if we have passed our host from the config)
  2740. if ($(resultIq).find(
  2741. '>conference>property[name=\'sipGatewayEnabled\']').length) {
  2742. sipGatewayEnabled = true;
  2743. }
  2744. console.info("Sip gateway enabled: " + sipGatewayEnabled);
  2745. },
  2746. // FIXME: we need to show the fact that we're waiting for the focus
  2747. // to the user(or that focus is not available)
  2748. allocateConferenceFocus: function (roomName, callback) {
  2749. // Try to use focus user JID from the config
  2750. Moderator.setFocusUserJid(config.focusUserJid);
  2751. // Send create conference IQ
  2752. var iq = Moderator.createConferenceIq(roomName);
  2753. connection.sendIQ(
  2754. iq,
  2755. function (result) {
  2756. if ('true' === $(result).find('conference').attr('ready')) {
  2757. // Reset both timers
  2758. getNextTimeout(true);
  2759. getNextErrorTimeout(true);
  2760. // Setup config options
  2761. Moderator.parseConfigOptions(result);
  2762. // Exec callback
  2763. callback();
  2764. } else {
  2765. var waitMs = getNextTimeout();
  2766. console.info("Waiting for the focus... " + waitMs);
  2767. // Reset error timeout
  2768. getNextErrorTimeout(true);
  2769. window.setTimeout(
  2770. function () {
  2771. Moderator.allocateConferenceFocus(
  2772. roomName, callback);
  2773. }, waitMs);
  2774. }
  2775. },
  2776. function (error) {
  2777. // Not authorized to create new room
  2778. if ($(error).find('>error>not-authorized').length) {
  2779. console.warn("Unauthorized to start the conference");
  2780. UI.onAuthenticationRequired(function () {
  2781. Moderator.allocateConferenceFocus(roomName, callback);
  2782. });
  2783. return;
  2784. }
  2785. var waitMs = getNextErrorTimeout();
  2786. console.error("Focus error, retry after " + waitMs, error);
  2787. // Show message
  2788. UI.messageHandler.notify(
  2789. 'Conference focus', 'disconnected',
  2790. Moderator.getFocusComponent() +
  2791. ' not available - retry in ' +
  2792. (waitMs / 1000) + ' sec');
  2793. // Reset response timeout
  2794. getNextTimeout(true);
  2795. window.setTimeout(
  2796. function () {
  2797. Moderator.allocateConferenceFocus(roomName, callback);
  2798. }, waitMs);
  2799. }
  2800. );
  2801. },
  2802. getAuthUrl: function (roomName, urlCallback) {
  2803. var iq = $iq({to: Moderator.getFocusComponent(), type: 'get'});
  2804. iq.c('auth-url', {
  2805. xmlns: 'http://jitsi.org/protocol/focus',
  2806. room: roomName
  2807. });
  2808. connection.sendIQ(
  2809. iq,
  2810. function (result) {
  2811. var url = $(result).find('auth-url').attr('url');
  2812. if (url) {
  2813. console.info("Got auth url: " + url);
  2814. urlCallback(url);
  2815. } else {
  2816. console.error(
  2817. "Failed to get auth url fro mthe focus", result);
  2818. }
  2819. },
  2820. function (error) {
  2821. console.error("Get auth url error", error);
  2822. }
  2823. );
  2824. }
  2825. };
  2826. module.exports = Moderator;
  2827. },{}],7:[function(require,module,exports){
  2828. /* global $, $iq, config, connection, focusMucJid, messageHandler, Moderator,
  2829. Toolbar, Util */
  2830. var Moderator = require("./moderator");
  2831. var recordingToken = null;
  2832. var recordingEnabled;
  2833. /**
  2834. * Whether to use a jirecon component for recording, or use the videobridge
  2835. * through COLIBRI.
  2836. */
  2837. var useJirecon = (typeof config.hosts.jirecon != "undefined");
  2838. /**
  2839. * The ID of the jirecon recording session. Jirecon generates it when we
  2840. * initially start recording, and it needs to be used in subsequent requests
  2841. * to jirecon.
  2842. */
  2843. var jireconRid = null;
  2844. function setRecordingToken(token) {
  2845. recordingToken = token;
  2846. }
  2847. function setRecording(state, token, callback) {
  2848. if (useJirecon){
  2849. this.setRecordingJirecon(state, token, callback);
  2850. } else {
  2851. this.setRecordingColibri(state, token, callback);
  2852. }
  2853. }
  2854. function setRecordingJirecon(state, token, callback) {
  2855. if (state == recordingEnabled){
  2856. return;
  2857. }
  2858. var iq = $iq({to: config.hosts.jirecon, type: 'set'})
  2859. .c('recording', {xmlns: 'http://jitsi.org/protocol/jirecon',
  2860. action: state ? 'start' : 'stop',
  2861. mucjid: connection.emuc.roomjid});
  2862. if (!state){
  2863. iq.attrs({rid: jireconRid});
  2864. }
  2865. console.log('Start recording');
  2866. connection.sendIQ(
  2867. iq,
  2868. function (result) {
  2869. // TODO wait for an IQ with the real status, since this is
  2870. // provisional?
  2871. jireconRid = $(result).find('recording').attr('rid');
  2872. console.log('Recording ' + (state ? 'started' : 'stopped') +
  2873. '(jirecon)' + result);
  2874. recordingEnabled = state;
  2875. if (!state){
  2876. jireconRid = null;
  2877. }
  2878. callback(state);
  2879. },
  2880. function (error) {
  2881. console.log('Failed to start recording, error: ', error);
  2882. callback(recordingEnabled);
  2883. });
  2884. }
  2885. // Sends a COLIBRI message which enables or disables (according to 'state')
  2886. // the recording on the bridge. Waits for the result IQ and calls 'callback'
  2887. // with the new recording state, according to the IQ.
  2888. function setRecordingColibri(state, token, callback) {
  2889. var elem = $iq({to: focusMucJid, type: 'set'});
  2890. elem.c('conference', {
  2891. xmlns: 'http://jitsi.org/protocol/colibri'
  2892. });
  2893. elem.c('recording', {state: state, token: token});
  2894. connection.sendIQ(elem,
  2895. function (result) {
  2896. console.log('Set recording "', state, '". Result:', result);
  2897. var recordingElem = $(result).find('>conference>recording');
  2898. var newState = ('true' === recordingElem.attr('state'));
  2899. recordingEnabled = newState;
  2900. callback(newState);
  2901. },
  2902. function (error) {
  2903. console.warn(error);
  2904. callback(recordingEnabled);
  2905. }
  2906. );
  2907. }
  2908. var Recording = {
  2909. toggleRecording: function (tokenEmptyCallback,
  2910. startingCallback, startedCallback) {
  2911. if (!Moderator.isModerator()) {
  2912. console.log(
  2913. 'non-focus, or conference not yet organized:' +
  2914. ' not enabling recording');
  2915. return;
  2916. }
  2917. // Jirecon does not (currently) support a token.
  2918. if (!recordingToken && !useJirecon) {
  2919. tokenEmptyCallback(function (value) {
  2920. setRecordingToken(value);
  2921. this.toggleRecording();
  2922. });
  2923. return;
  2924. }
  2925. var oldState = recordingEnabled;
  2926. startingCallback(!oldState);
  2927. setRecording(!oldState,
  2928. recordingToken,
  2929. function (state) {
  2930. console.log("New recording state: ", state);
  2931. if (state === oldState) {
  2932. // FIXME: new focus:
  2933. // this will not work when moderator changes
  2934. // during active session. Then it will assume that
  2935. // recording status has changed to true, but it might have
  2936. // been already true(and we only received actual status from
  2937. // the focus).
  2938. //
  2939. // SO we start with status null, so that it is initialized
  2940. // here and will fail only after second click, so if invalid
  2941. // token was used we have to press the button twice before
  2942. // current status will be fetched and token will be reset.
  2943. //
  2944. // Reliable way would be to return authentication error.
  2945. // Or status update when moderator connects.
  2946. // Or we have to stop recording session when current
  2947. // moderator leaves the room.
  2948. // Failed to change, reset the token because it might
  2949. // have been wrong
  2950. setRecordingToken(null);
  2951. }
  2952. startedCallback(state);
  2953. }
  2954. );
  2955. }
  2956. }
  2957. module.exports = Recording;
  2958. },{"./moderator":6}],8:[function(require,module,exports){
  2959. /* jshint -W117 */
  2960. /* a simple MUC connection plugin
  2961. * can only handle a single MUC room
  2962. */
  2963. var bridgeIsDown = false;
  2964. var Moderator = require("./moderator");
  2965. module.exports = function(XMPP, eventEmitter) {
  2966. Strophe.addConnectionPlugin('emuc', {
  2967. connection: null,
  2968. roomjid: null,
  2969. myroomjid: null,
  2970. members: {},
  2971. list_members: [], // so we can elect a new focus
  2972. presMap: {},
  2973. preziMap: {},
  2974. joined: false,
  2975. isOwner: false,
  2976. role: null,
  2977. init: function (conn) {
  2978. this.connection = conn;
  2979. },
  2980. initPresenceMap: function (myroomjid) {
  2981. this.presMap['to'] = myroomjid;
  2982. this.presMap['xns'] = 'http://jabber.org/protocol/muc';
  2983. },
  2984. doJoin: function (jid, password) {
  2985. this.myroomjid = jid;
  2986. console.info("Joined MUC as " + this.myroomjid);
  2987. this.initPresenceMap(this.myroomjid);
  2988. if (!this.roomjid) {
  2989. this.roomjid = Strophe.getBareJidFromJid(jid);
  2990. // add handlers (just once)
  2991. this.connection.addHandler(this.onPresence.bind(this), null, 'presence', null, null, this.roomjid, {matchBare: true});
  2992. this.connection.addHandler(this.onPresenceUnavailable.bind(this), null, 'presence', 'unavailable', null, this.roomjid, {matchBare: true});
  2993. this.connection.addHandler(this.onPresenceError.bind(this), null, 'presence', 'error', null, this.roomjid, {matchBare: true});
  2994. this.connection.addHandler(this.onMessage.bind(this), null, 'message', null, null, this.roomjid, {matchBare: true});
  2995. }
  2996. if (password !== undefined) {
  2997. this.presMap['password'] = password;
  2998. }
  2999. this.sendPresence();
  3000. },
  3001. doLeave: function () {
  3002. console.log("do leave", this.myroomjid);
  3003. var pres = $pres({to: this.myroomjid, type: 'unavailable' });
  3004. this.presMap.length = 0;
  3005. this.connection.send(pres);
  3006. },
  3007. createNonAnonymousRoom: function () {
  3008. // http://xmpp.org/extensions/xep-0045.html#createroom-reserved
  3009. var getForm = $iq({type: 'get', to: this.roomjid})
  3010. .c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'})
  3011. .c('x', {xmlns: 'jabber:x:data', type: 'submit'});
  3012. this.connection.sendIQ(getForm, function (form) {
  3013. if (!$(form).find(
  3014. '>query>x[xmlns="jabber:x:data"]' +
  3015. '>field[var="muc#roomconfig_whois"]').length) {
  3016. console.error('non-anonymous rooms not supported');
  3017. return;
  3018. }
  3019. var formSubmit = $iq({to: this.roomjid, type: 'set'})
  3020. .c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'});
  3021. formSubmit.c('x', {xmlns: 'jabber:x:data', type: 'submit'});
  3022. formSubmit.c('field', {'var': 'FORM_TYPE'})
  3023. .c('value')
  3024. .t('http://jabber.org/protocol/muc#roomconfig').up().up();
  3025. formSubmit.c('field', {'var': 'muc#roomconfig_whois'})
  3026. .c('value').t('anyone').up().up();
  3027. this.connection.sendIQ(formSubmit);
  3028. }, function (error) {
  3029. console.error("Error getting room configuration form");
  3030. });
  3031. },
  3032. onPresence: function (pres) {
  3033. var from = pres.getAttribute('from');
  3034. // What is this for? A workaround for something?
  3035. if (pres.getAttribute('type')) {
  3036. return true;
  3037. }
  3038. // Parse etherpad tag.
  3039. var etherpad = $(pres).find('>etherpad');
  3040. if (etherpad.length) {
  3041. if (config.etherpad_base && !Moderator.isModerator()) {
  3042. UI.initEtherpad(etherpad.text());
  3043. }
  3044. }
  3045. // Parse prezi tag.
  3046. var presentation = $(pres).find('>prezi');
  3047. if (presentation.length) {
  3048. var url = presentation.attr('url');
  3049. var current = presentation.find('>current').text();
  3050. console.log('presentation info received from', from, url);
  3051. if (this.preziMap[from] == null) {
  3052. this.preziMap[from] = url;
  3053. $(document).trigger('presentationadded.muc', [from, url, current]);
  3054. }
  3055. else {
  3056. $(document).trigger('gotoslide.muc', [from, url, current]);
  3057. }
  3058. }
  3059. else if (this.preziMap[from] != null) {
  3060. var url = this.preziMap[from];
  3061. delete this.preziMap[from];
  3062. $(document).trigger('presentationremoved.muc', [from, url]);
  3063. }
  3064. // Parse audio info tag.
  3065. var audioMuted = $(pres).find('>audiomuted');
  3066. if (audioMuted.length) {
  3067. $(document).trigger('audiomuted.muc', [from, audioMuted.text()]);
  3068. }
  3069. // Parse video info tag.
  3070. var videoMuted = $(pres).find('>videomuted');
  3071. if (videoMuted.length) {
  3072. $(document).trigger('videomuted.muc', [from, videoMuted.text()]);
  3073. }
  3074. var stats = $(pres).find('>stats');
  3075. if (stats.length) {
  3076. var statsObj = {};
  3077. Strophe.forEachChild(stats[0], "stat", function (el) {
  3078. statsObj[el.getAttribute("name")] = el.getAttribute("value");
  3079. });
  3080. connectionquality.updateRemoteStats(from, statsObj);
  3081. }
  3082. // Parse status.
  3083. if ($(pres).find('>x[xmlns="http://jabber.org/protocol/muc#user"]>status[code="201"]').length) {
  3084. this.isOwner = true;
  3085. this.createNonAnonymousRoom();
  3086. }
  3087. // Parse roles.
  3088. var member = {};
  3089. member.show = $(pres).find('>show').text();
  3090. member.status = $(pres).find('>status').text();
  3091. var tmp = $(pres).find('>x[xmlns="http://jabber.org/protocol/muc#user"]>item');
  3092. member.affiliation = tmp.attr('affiliation');
  3093. member.role = tmp.attr('role');
  3094. // Focus recognition
  3095. member.jid = tmp.attr('jid');
  3096. member.isFocus = false;
  3097. if (member.jid
  3098. && member.jid.indexOf(Moderator.getFocusUserJid() + "/") == 0) {
  3099. member.isFocus = true;
  3100. }
  3101. var nicktag = $(pres).find('>nick[xmlns="http://jabber.org/protocol/nick"]');
  3102. member.displayName = (nicktag.length > 0 ? nicktag.html() : null);
  3103. if (from == this.myroomjid) {
  3104. if (member.affiliation == 'owner') this.isOwner = true;
  3105. if (this.role !== member.role) {
  3106. this.role = member.role;
  3107. if (Moderator.onLocalRoleChange)
  3108. Moderator.onLocalRoleChange(from, member, pres);
  3109. UI.onLocalRoleChange(from, member, pres);
  3110. }
  3111. if (!this.joined) {
  3112. this.joined = true;
  3113. eventEmitter.emit(XMPPEvents.MUC_JOINED, from, member);
  3114. this.list_members.push(from);
  3115. }
  3116. } else if (this.members[from] === undefined) {
  3117. // new participant
  3118. this.members[from] = member;
  3119. this.list_members.push(from);
  3120. console.log('entered', from, member);
  3121. if (member.isFocus) {
  3122. focusMucJid = from;
  3123. console.info("Ignore focus: " + from + ", real JID: " + member.jid);
  3124. }
  3125. else {
  3126. var id = $(pres).find('>userID').text();
  3127. var email = $(pres).find('>email');
  3128. if (email.length > 0) {
  3129. id = email.text();
  3130. }
  3131. UI.onMucEntered(from, id, member.displayName);
  3132. API.triggerEvent("participantJoined", {jid: from});
  3133. }
  3134. } else {
  3135. // Presence update for existing participant
  3136. // Watch role change:
  3137. if (this.members[from].role != member.role) {
  3138. this.members[from].role = member.role;
  3139. UI.onMucRoleChanged(member.role, member.displayName);
  3140. }
  3141. }
  3142. // Always trigger presence to update bindings
  3143. $(document).trigger('presence.muc', [from, member, pres]);
  3144. this.parsePresence(from, member, pres);
  3145. // Trigger status message update
  3146. if (member.status) {
  3147. UI.onMucPresenceStatus(from, member);
  3148. }
  3149. return true;
  3150. },
  3151. onPresenceUnavailable: function (pres) {
  3152. var from = pres.getAttribute('from');
  3153. // Status code 110 indicates that this notification is "self-presence".
  3154. if (!$(pres).find('>x[xmlns="http://jabber.org/protocol/muc#user"]>status[code="110"]').length) {
  3155. delete this.members[from];
  3156. this.list_members.splice(this.list_members.indexOf(from), 1);
  3157. this.onParticipantLeft(from);
  3158. }
  3159. // If the status code is 110 this means we're leaving and we would like
  3160. // to remove everyone else from our view, so we trigger the event.
  3161. else if (this.list_members.length > 1) {
  3162. for (var i = 0; i < this.list_members.length; i++) {
  3163. var member = this.list_members[i];
  3164. delete this.members[i];
  3165. this.list_members.splice(i, 1);
  3166. this.onParticipantLeft(member);
  3167. }
  3168. }
  3169. if ($(pres).find('>x[xmlns="http://jabber.org/protocol/muc#user"]>status[code="307"]').length) {
  3170. $(document).trigger('kicked.muc', [from]);
  3171. if (this.myroomjid === from) {
  3172. XMPP.disposeConference(false);
  3173. eventEmitter.emit(XMPPEvents.KICKED);
  3174. }
  3175. }
  3176. return true;
  3177. },
  3178. onPresenceError: function (pres) {
  3179. var from = pres.getAttribute('from');
  3180. if ($(pres).find('>error[type="auth"]>not-authorized[xmlns="urn:ietf:params:xml:ns:xmpp-stanzas"]').length) {
  3181. console.log('on password required', from);
  3182. var self = this;
  3183. UI.onPasswordReqiured(function (value) {
  3184. self.doJoin(from, value);
  3185. });
  3186. } else if ($(pres).find(
  3187. '>error[type="cancel"]>not-allowed[xmlns="urn:ietf:params:xml:ns:xmpp-stanzas"]').length) {
  3188. var toDomain = Strophe.getDomainFromJid(pres.getAttribute('to'));
  3189. if (toDomain === config.hosts.anonymousdomain) {
  3190. // we are connected with anonymous domain and only non anonymous users can create rooms
  3191. // we must authorize the user
  3192. XMPP.promptLogin();
  3193. } else {
  3194. console.warn('onPresError ', pres);
  3195. UI.messageHandler.openReportDialog(null,
  3196. 'Oops! Something went wrong and we couldn`t connect to the conference.',
  3197. pres);
  3198. }
  3199. } else {
  3200. console.warn('onPresError ', pres);
  3201. UI.messageHandler.openReportDialog(null,
  3202. 'Oops! Something went wrong and we couldn`t connect to the conference.',
  3203. pres);
  3204. }
  3205. return true;
  3206. },
  3207. sendMessage: function (body, nickname) {
  3208. var msg = $msg({to: this.roomjid, type: 'groupchat'});
  3209. msg.c('body', body).up();
  3210. if (nickname) {
  3211. msg.c('nick', {xmlns: 'http://jabber.org/protocol/nick'}).t(nickname).up().up();
  3212. }
  3213. this.connection.send(msg);
  3214. API.triggerEvent("outgoingMessage", {"message": body});
  3215. },
  3216. setSubject: function (subject) {
  3217. var msg = $msg({to: this.roomjid, type: 'groupchat'});
  3218. msg.c('subject', subject);
  3219. this.connection.send(msg);
  3220. console.log("topic changed to " + subject);
  3221. },
  3222. onMessage: function (msg) {
  3223. // FIXME: this is a hack. but jingle on muc makes nickchanges hard
  3224. var from = msg.getAttribute('from');
  3225. var nick = $(msg).find('>nick[xmlns="http://jabber.org/protocol/nick"]').text() || Strophe.getResourceFromJid(from);
  3226. var txt = $(msg).find('>body').text();
  3227. var type = msg.getAttribute("type");
  3228. if (type == "error") {
  3229. UI.chatAddError($(msg).find('>text').text(), txt);
  3230. return true;
  3231. }
  3232. var subject = $(msg).find('>subject');
  3233. if (subject.length) {
  3234. var subjectText = subject.text();
  3235. if (subjectText || subjectText == "") {
  3236. UI.chatSetSubject(subjectText);
  3237. console.log("Subject is changed to " + subjectText);
  3238. }
  3239. }
  3240. if (txt) {
  3241. console.log('chat', nick, txt);
  3242. UI.updateChatConversation(from, nick, txt);
  3243. if (from != this.myroomjid)
  3244. API.triggerEvent("incomingMessage",
  3245. {"from": from, "nick": nick, "message": txt});
  3246. }
  3247. return true;
  3248. },
  3249. lockRoom: function (key, onSuccess, onError, onNotSupported) {
  3250. //http://xmpp.org/extensions/xep-0045.html#roomconfig
  3251. var ob = this;
  3252. this.connection.sendIQ($iq({to: this.roomjid, type: 'get'}).c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'}),
  3253. function (res) {
  3254. if ($(res).find('>query>x[xmlns="jabber:x:data"]>field[var="muc#roomconfig_roomsecret"]').length) {
  3255. var formsubmit = $iq({to: ob.roomjid, type: 'set'}).c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'});
  3256. formsubmit.c('x', {xmlns: 'jabber:x:data', type: 'submit'});
  3257. formsubmit.c('field', {'var': 'FORM_TYPE'}).c('value').t('http://jabber.org/protocol/muc#roomconfig').up().up();
  3258. formsubmit.c('field', {'var': 'muc#roomconfig_roomsecret'}).c('value').t(key).up().up();
  3259. // Fixes a bug in prosody 0.9.+ https://code.google.com/p/lxmppd/issues/detail?id=373
  3260. formsubmit.c('field', {'var': 'muc#roomconfig_whois'}).c('value').t('anyone').up().up();
  3261. // FIXME: is muc#roomconfig_passwordprotectedroom required?
  3262. this.connection.sendIQ(formsubmit,
  3263. onSuccess,
  3264. onError);
  3265. } else {
  3266. onNotSupported();
  3267. }
  3268. }, onError);
  3269. },
  3270. kick: function (jid) {
  3271. var kickIQ = $iq({to: this.roomjid, type: 'set'})
  3272. .c('query', {xmlns: 'http://jabber.org/protocol/muc#admin'})
  3273. .c('item', {nick: Strophe.getResourceFromJid(jid), role: 'none'})
  3274. .c('reason').t('You have been kicked.').up().up().up();
  3275. this.connection.sendIQ(
  3276. kickIQ,
  3277. function (result) {
  3278. console.log('Kick participant with jid: ', jid, result);
  3279. },
  3280. function (error) {
  3281. console.log('Kick participant error: ', error);
  3282. });
  3283. },
  3284. sendPresence: function () {
  3285. var pres = $pres({to: this.presMap['to'] });
  3286. pres.c('x', {xmlns: this.presMap['xns']});
  3287. if (this.presMap['password']) {
  3288. pres.c('password').t(this.presMap['password']).up();
  3289. }
  3290. pres.up();
  3291. // Send XEP-0115 'c' stanza that contains our capabilities info
  3292. if (this.connection.caps) {
  3293. this.connection.caps.node = config.clientNode;
  3294. pres.c('c', this.connection.caps.generateCapsAttrs()).up();
  3295. }
  3296. pres.c('user-agent', {xmlns: 'http://jitsi.org/jitmeet/user-agent'})
  3297. .t(navigator.userAgent).up();
  3298. if (this.presMap['bridgeIsDown']) {
  3299. pres.c('bridgeIsDown').up();
  3300. }
  3301. if (this.presMap['email']) {
  3302. pres.c('email').t(this.presMap['email']).up();
  3303. }
  3304. if (this.presMap['userId']) {
  3305. pres.c('userId').t(this.presMap['userId']).up();
  3306. }
  3307. if (this.presMap['displayName']) {
  3308. // XEP-0172
  3309. pres.c('nick', {xmlns: 'http://jabber.org/protocol/nick'})
  3310. .t(this.presMap['displayName']).up();
  3311. }
  3312. if (this.presMap['audions']) {
  3313. pres.c('audiomuted', {xmlns: this.presMap['audions']})
  3314. .t(this.presMap['audiomuted']).up();
  3315. }
  3316. if (this.presMap['videons']) {
  3317. pres.c('videomuted', {xmlns: this.presMap['videons']})
  3318. .t(this.presMap['videomuted']).up();
  3319. }
  3320. if (this.presMap['statsns']) {
  3321. var stats = pres.c('stats', {xmlns: this.presMap['statsns']});
  3322. for (var stat in this.presMap["stats"])
  3323. if (this.presMap["stats"][stat] != null)
  3324. stats.c("stat", {name: stat, value: this.presMap["stats"][stat]}).up();
  3325. pres.up();
  3326. }
  3327. if (this.presMap['prezins']) {
  3328. pres.c('prezi',
  3329. {xmlns: this.presMap['prezins'],
  3330. 'url': this.presMap['preziurl']})
  3331. .c('current').t(this.presMap['prezicurrent']).up().up();
  3332. }
  3333. if (this.presMap['etherpadns']) {
  3334. pres.c('etherpad', {xmlns: this.presMap['etherpadns']})
  3335. .t(this.presMap['etherpadname']).up();
  3336. }
  3337. if (this.presMap['medians']) {
  3338. pres.c('media', {xmlns: this.presMap['medians']});
  3339. var sourceNumber = 0;
  3340. Object.keys(this.presMap).forEach(function (key) {
  3341. if (key.indexOf('source') >= 0) {
  3342. sourceNumber++;
  3343. }
  3344. });
  3345. if (sourceNumber > 0)
  3346. for (var i = 1; i <= sourceNumber / 3; i++) {
  3347. pres.c('source',
  3348. {type: this.presMap['source' + i + '_type'],
  3349. ssrc: this.presMap['source' + i + '_ssrc'],
  3350. direction: this.presMap['source' + i + '_direction']
  3351. || 'sendrecv' }
  3352. ).up();
  3353. }
  3354. }
  3355. pres.up();
  3356. // console.debug(pres.toString());
  3357. this.connection.send(pres);
  3358. },
  3359. addDisplayNameToPresence: function (displayName) {
  3360. this.presMap['displayName'] = displayName;
  3361. },
  3362. addMediaToPresence: function (sourceNumber, mtype, ssrcs, direction) {
  3363. if (!this.presMap['medians'])
  3364. this.presMap['medians'] = 'http://estos.de/ns/mjs';
  3365. this.presMap['source' + sourceNumber + '_type'] = mtype;
  3366. this.presMap['source' + sourceNumber + '_ssrc'] = ssrcs;
  3367. this.presMap['source' + sourceNumber + '_direction'] = direction;
  3368. },
  3369. clearPresenceMedia: function () {
  3370. var self = this;
  3371. Object.keys(this.presMap).forEach(function (key) {
  3372. if (key.indexOf('source') != -1) {
  3373. delete self.presMap[key];
  3374. }
  3375. });
  3376. },
  3377. addPreziToPresence: function (url, currentSlide) {
  3378. this.presMap['prezins'] = 'http://jitsi.org/jitmeet/prezi';
  3379. this.presMap['preziurl'] = url;
  3380. this.presMap['prezicurrent'] = currentSlide;
  3381. },
  3382. removePreziFromPresence: function () {
  3383. delete this.presMap['prezins'];
  3384. delete this.presMap['preziurl'];
  3385. delete this.presMap['prezicurrent'];
  3386. },
  3387. addCurrentSlideToPresence: function (currentSlide) {
  3388. this.presMap['prezicurrent'] = currentSlide;
  3389. },
  3390. getPrezi: function (roomjid) {
  3391. return this.preziMap[roomjid];
  3392. },
  3393. addEtherpadToPresence: function (etherpadName) {
  3394. this.presMap['etherpadns'] = 'http://jitsi.org/jitmeet/etherpad';
  3395. this.presMap['etherpadname'] = etherpadName;
  3396. },
  3397. addAudioInfoToPresence: function (isMuted) {
  3398. this.presMap['audions'] = 'http://jitsi.org/jitmeet/audio';
  3399. this.presMap['audiomuted'] = isMuted.toString();
  3400. },
  3401. addVideoInfoToPresence: function (isMuted) {
  3402. this.presMap['videons'] = 'http://jitsi.org/jitmeet/video';
  3403. this.presMap['videomuted'] = isMuted.toString();
  3404. },
  3405. addConnectionInfoToPresence: function (stats) {
  3406. this.presMap['statsns'] = 'http://jitsi.org/jitmeet/stats';
  3407. this.presMap['stats'] = stats;
  3408. },
  3409. findJidFromResource: function (resourceJid) {
  3410. if (resourceJid &&
  3411. resourceJid === Strophe.getResourceFromJid(this.myroomjid)) {
  3412. return this.myroomjid;
  3413. }
  3414. var peerJid = null;
  3415. Object.keys(this.members).some(function (jid) {
  3416. peerJid = jid;
  3417. return Strophe.getResourceFromJid(jid) === resourceJid;
  3418. });
  3419. return peerJid;
  3420. },
  3421. addBridgeIsDownToPresence: function () {
  3422. this.presMap['bridgeIsDown'] = true;
  3423. },
  3424. addEmailToPresence: function (email) {
  3425. this.presMap['email'] = email;
  3426. },
  3427. addUserIdToPresence: function (userId) {
  3428. this.presMap['userId'] = userId;
  3429. },
  3430. isModerator: function () {
  3431. return this.role === 'moderator';
  3432. },
  3433. getMemberRole: function (peerJid) {
  3434. if (this.members[peerJid]) {
  3435. return this.members[peerJid].role;
  3436. }
  3437. return null;
  3438. },
  3439. onParticipantLeft: function (jid) {
  3440. UI.onMucLeft(jid);
  3441. API.triggerEvent("participantLeft", {jid: jid});
  3442. delete jid2Ssrc[jid];
  3443. this.connection.jingle.terminateByJid(jid);
  3444. if (this.getPrezi(jid)) {
  3445. $(document).trigger('presentationremoved.muc',
  3446. [jid, this.getPrezi(jid)]);
  3447. }
  3448. Moderator.onMucLeft(jid);
  3449. },
  3450. parsePresence: function (from, memeber, pres) {
  3451. if($(pres).find(">bridgeIsDown").length > 0 && !bridgeIsDown) {
  3452. bridgeIsDown = true;
  3453. eventEmitter.emit(XMPPEvents.BRIDGE_DOWN);
  3454. }
  3455. if(memeber.isFocus)
  3456. return;
  3457. // Remove old ssrcs coming from the jid
  3458. Object.keys(ssrc2jid).forEach(function (ssrc) {
  3459. if (ssrc2jid[ssrc] == jid) {
  3460. delete ssrc2jid[ssrc];
  3461. delete ssrc2videoType[ssrc];
  3462. }
  3463. });
  3464. var changedStreams = [];
  3465. $(pres).find('>media[xmlns="http://estos.de/ns/mjs"]>source').each(function (idx, ssrc) {
  3466. //console.log(jid, 'assoc ssrc', ssrc.getAttribute('type'), ssrc.getAttribute('ssrc'));
  3467. var ssrcV = ssrc.getAttribute('ssrc');
  3468. ssrc2jid[ssrcV] = from;
  3469. notReceivedSSRCs.push(ssrcV);
  3470. var type = ssrc.getAttribute('type');
  3471. ssrc2videoType[ssrcV] = type;
  3472. var direction = ssrc.getAttribute('direction');
  3473. changedStreams.push({type: type, direction: direction});
  3474. });
  3475. eventEmitter.emit(XMPPEvents.CHANGED_STREAMS, from, changedStreams);
  3476. var displayName = !config.displayJids
  3477. ? memeber.displayName : Strophe.getResourceFromJid(from);
  3478. if (displayName && displayName.length > 0)
  3479. {
  3480. // $(document).trigger('displaynamechanged',
  3481. // [jid, displayName]);
  3482. eventEmitter.emit(XMPPEvents.DISPLAY_NAME_CHANGED, from, displayName);
  3483. }
  3484. var id = $(pres).find('>userID').text();
  3485. var email = $(pres).find('>email');
  3486. if(email.length > 0) {
  3487. id = email.text();
  3488. }
  3489. eventEmitter.emit(XMPPEvents.USER_ID_CHANGED, from, id);
  3490. }
  3491. });
  3492. };
  3493. },{"./moderator":6}],9:[function(require,module,exports){
  3494. /* jshint -W117 */
  3495. var JingleSession = require("./JingleSession");
  3496. function CallIncomingJingle(sid, connection) {
  3497. var sess = connection.jingle.sessions[sid];
  3498. // TODO: do we check activecall == null?
  3499. activecall = sess;
  3500. statistics.onConferenceCreated(sess);
  3501. RTC.onConferenceCreated(sess);
  3502. // TODO: check affiliation and/or role
  3503. console.log('emuc data for', sess.peerjid, connection.emuc.members[sess.peerjid]);
  3504. sess.usedrip = true; // not-so-naive trickle ice
  3505. sess.sendAnswer();
  3506. sess.accept();
  3507. };
  3508. module.exports = function(XMPP)
  3509. {
  3510. Strophe.addConnectionPlugin('jingle', {
  3511. connection: null,
  3512. sessions: {},
  3513. jid2session: {},
  3514. ice_config: {iceServers: []},
  3515. pc_constraints: {},
  3516. media_constraints: {
  3517. mandatory: {
  3518. 'OfferToReceiveAudio': true,
  3519. 'OfferToReceiveVideo': true
  3520. }
  3521. // MozDontOfferDataChannel: true when this is firefox
  3522. },
  3523. init: function (conn) {
  3524. this.connection = conn;
  3525. if (this.connection.disco) {
  3526. // http://xmpp.org/extensions/xep-0167.html#support
  3527. // http://xmpp.org/extensions/xep-0176.html#support
  3528. this.connection.disco.addFeature('urn:xmpp:jingle:1');
  3529. this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:1');
  3530. this.connection.disco.addFeature('urn:xmpp:jingle:transports:ice-udp:1');
  3531. this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:audio');
  3532. this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:video');
  3533. // this is dealt with by SDP O/A so we don't need to annouce this
  3534. //this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:rtcp-fb:0'); // XEP-0293
  3535. //this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:rtp-hdrext:0'); // XEP-0294
  3536. if (config.useRtcpMux) {
  3537. this.connection.disco.addFeature('urn:ietf:rfc:5761'); // rtcp-mux
  3538. }
  3539. if (config.useBundle) {
  3540. this.connection.disco.addFeature('urn:ietf:rfc:5888'); // a=group, e.g. bundle
  3541. }
  3542. //this.connection.disco.addFeature('urn:ietf:rfc:5576'); // a=ssrc
  3543. }
  3544. this.connection.addHandler(this.onJingle.bind(this), 'urn:xmpp:jingle:1', 'iq', 'set', null, null);
  3545. },
  3546. onJingle: function (iq) {
  3547. var sid = $(iq).find('jingle').attr('sid');
  3548. var action = $(iq).find('jingle').attr('action');
  3549. var fromJid = iq.getAttribute('from');
  3550. // send ack first
  3551. var ack = $iq({type: 'result',
  3552. to: fromJid,
  3553. id: iq.getAttribute('id')
  3554. });
  3555. console.log('on jingle ' + action + ' from ' + fromJid, iq);
  3556. var sess = this.sessions[sid];
  3557. if ('session-initiate' != action) {
  3558. if (sess === null) {
  3559. ack.type = 'error';
  3560. ack.c('error', {type: 'cancel'})
  3561. .c('item-not-found', {xmlns: 'urn:ietf:params:xml:ns:xmpp-stanzas'}).up()
  3562. .c('unknown-session', {xmlns: 'urn:xmpp:jingle:errors:1'});
  3563. this.connection.send(ack);
  3564. return true;
  3565. }
  3566. // compare from to sess.peerjid (bare jid comparison for later compat with message-mode)
  3567. // local jid is not checked
  3568. if (Strophe.getBareJidFromJid(fromJid) != Strophe.getBareJidFromJid(sess.peerjid)) {
  3569. console.warn('jid mismatch for session id', sid, fromJid, sess.peerjid);
  3570. ack.type = 'error';
  3571. ack.c('error', {type: 'cancel'})
  3572. .c('item-not-found', {xmlns: 'urn:ietf:params:xml:ns:xmpp-stanzas'}).up()
  3573. .c('unknown-session', {xmlns: 'urn:xmpp:jingle:errors:1'});
  3574. this.connection.send(ack);
  3575. return true;
  3576. }
  3577. } else if (sess !== undefined) {
  3578. // existing session with same session id
  3579. // this might be out-of-order if the sess.peerjid is the same as from
  3580. ack.type = 'error';
  3581. ack.c('error', {type: 'cancel'})
  3582. .c('service-unavailable', {xmlns: 'urn:ietf:params:xml:ns:xmpp-stanzas'}).up();
  3583. console.warn('duplicate session id', sid);
  3584. this.connection.send(ack);
  3585. return true;
  3586. }
  3587. // FIXME: check for a defined action
  3588. this.connection.send(ack);
  3589. // see http://xmpp.org/extensions/xep-0166.html#concepts-session
  3590. switch (action) {
  3591. case 'session-initiate':
  3592. sess = new JingleSession(
  3593. $(iq).attr('to'), $(iq).find('jingle').attr('sid'),
  3594. this.connection, XMPP);
  3595. // configure session
  3596. sess.media_constraints = this.media_constraints;
  3597. sess.pc_constraints = this.pc_constraints;
  3598. sess.ice_config = this.ice_config;
  3599. sess.initiate(fromJid, false);
  3600. // FIXME: setRemoteDescription should only be done when this call is to be accepted
  3601. sess.setRemoteDescription($(iq).find('>jingle'), 'offer');
  3602. this.sessions[sess.sid] = sess;
  3603. this.jid2session[sess.peerjid] = sess;
  3604. // the callback should either
  3605. // .sendAnswer and .accept
  3606. // or .sendTerminate -- not necessarily synchronus
  3607. CallIncomingJingle(sess.sid, this.connection);
  3608. break;
  3609. case 'session-accept':
  3610. sess.setRemoteDescription($(iq).find('>jingle'), 'answer');
  3611. sess.accept();
  3612. $(document).trigger('callaccepted.jingle', [sess.sid]);
  3613. break;
  3614. case 'session-terminate':
  3615. // If this is not the focus sending the terminate, we have
  3616. // nothing more to do here.
  3617. if (Object.keys(this.sessions).length < 1
  3618. || !(this.sessions[Object.keys(this.sessions)[0]]
  3619. instanceof JingleSession))
  3620. {
  3621. break;
  3622. }
  3623. console.log('terminating...', sess.sid);
  3624. sess.terminate();
  3625. this.terminate(sess.sid);
  3626. if ($(iq).find('>jingle>reason').length) {
  3627. $(document).trigger('callterminated.jingle', [
  3628. sess.sid,
  3629. sess.peerjid,
  3630. $(iq).find('>jingle>reason>:first')[0].tagName,
  3631. $(iq).find('>jingle>reason>text').text()
  3632. ]);
  3633. } else {
  3634. $(document).trigger('callterminated.jingle',
  3635. [sess.sid, sess.peerjid]);
  3636. }
  3637. break;
  3638. case 'transport-info':
  3639. sess.addIceCandidate($(iq).find('>jingle>content'));
  3640. break;
  3641. case 'session-info':
  3642. var affected;
  3643. if ($(iq).find('>jingle>ringing[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').length) {
  3644. $(document).trigger('ringing.jingle', [sess.sid]);
  3645. } else if ($(iq).find('>jingle>mute[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').length) {
  3646. affected = $(iq).find('>jingle>mute[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').attr('name');
  3647. $(document).trigger('mute.jingle', [sess.sid, affected]);
  3648. } else if ($(iq).find('>jingle>unmute[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').length) {
  3649. affected = $(iq).find('>jingle>unmute[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').attr('name');
  3650. $(document).trigger('unmute.jingle', [sess.sid, affected]);
  3651. }
  3652. break;
  3653. case 'addsource': // FIXME: proprietary, un-jingleish
  3654. case 'source-add': // FIXME: proprietary
  3655. sess.addSource($(iq).find('>jingle>content'), fromJid);
  3656. break;
  3657. case 'removesource': // FIXME: proprietary, un-jingleish
  3658. case 'source-remove': // FIXME: proprietary
  3659. sess.removeSource($(iq).find('>jingle>content'), fromJid);
  3660. break;
  3661. default:
  3662. console.warn('jingle action not implemented', action);
  3663. break;
  3664. }
  3665. return true;
  3666. },
  3667. initiate: function (peerjid, myjid) { // initiate a new jinglesession to peerjid
  3668. var sess = new JingleSession(myjid || this.connection.jid,
  3669. Math.random().toString(36).substr(2, 12), // random string
  3670. this.connection, XMPP);
  3671. // configure session
  3672. sess.media_constraints = this.media_constraints;
  3673. sess.pc_constraints = this.pc_constraints;
  3674. sess.ice_config = this.ice_config;
  3675. sess.initiate(peerjid, true);
  3676. this.sessions[sess.sid] = sess;
  3677. this.jid2session[sess.peerjid] = sess;
  3678. sess.sendOffer();
  3679. return sess;
  3680. },
  3681. terminate: function (sid, reason, text) { // terminate by sessionid (or all sessions)
  3682. if (sid === null || sid === undefined) {
  3683. for (sid in this.sessions) {
  3684. if (this.sessions[sid].state != 'ended') {
  3685. this.sessions[sid].sendTerminate(reason || (!this.sessions[sid].active()) ? 'cancel' : null, text);
  3686. this.sessions[sid].terminate();
  3687. }
  3688. delete this.jid2session[this.sessions[sid].peerjid];
  3689. delete this.sessions[sid];
  3690. }
  3691. } else if (this.sessions.hasOwnProperty(sid)) {
  3692. if (this.sessions[sid].state != 'ended') {
  3693. this.sessions[sid].sendTerminate(reason || (!this.sessions[sid].active()) ? 'cancel' : null, text);
  3694. this.sessions[sid].terminate();
  3695. }
  3696. delete this.jid2session[this.sessions[sid].peerjid];
  3697. delete this.sessions[sid];
  3698. }
  3699. },
  3700. // Used to terminate a session when an unavailable presence is received.
  3701. terminateByJid: function (jid) {
  3702. if (this.jid2session.hasOwnProperty(jid)) {
  3703. var sess = this.jid2session[jid];
  3704. if (sess) {
  3705. sess.terminate();
  3706. console.log('peer went away silently', jid);
  3707. delete this.sessions[sess.sid];
  3708. delete this.jid2session[jid];
  3709. $(document).trigger('callterminated.jingle',
  3710. [sess.sid, jid], 'gone');
  3711. }
  3712. }
  3713. },
  3714. terminateRemoteByJid: function (jid, reason) {
  3715. if (this.jid2session.hasOwnProperty(jid)) {
  3716. var sess = this.jid2session[jid];
  3717. if (sess) {
  3718. sess.sendTerminate(reason || (!sess.active()) ? 'kick' : null);
  3719. sess.terminate();
  3720. console.log('terminate peer with jid', sess.sid, jid);
  3721. delete this.sessions[sess.sid];
  3722. delete this.jid2session[jid];
  3723. $(document).trigger('callterminated.jingle',
  3724. [sess.sid, jid, 'kicked']);
  3725. }
  3726. }
  3727. },
  3728. getStunAndTurnCredentials: function () {
  3729. // get stun and turn configuration from server via xep-0215
  3730. // uses time-limited credentials as described in
  3731. // http://tools.ietf.org/html/draft-uberti-behave-turn-rest-00
  3732. //
  3733. // see https://code.google.com/p/prosody-modules/source/browse/mod_turncredentials/mod_turncredentials.lua
  3734. // for a prosody module which implements this
  3735. //
  3736. // currently, this doesn't work with updateIce and therefore credentials with a long
  3737. // validity have to be fetched before creating the peerconnection
  3738. // TODO: implement refresh via updateIce as described in
  3739. // https://code.google.com/p/webrtc/issues/detail?id=1650
  3740. var self = this;
  3741. this.connection.sendIQ(
  3742. $iq({type: 'get', to: this.connection.domain})
  3743. .c('services', {xmlns: 'urn:xmpp:extdisco:1'}).c('service', {host: 'turn.' + this.connection.domain}),
  3744. function (res) {
  3745. var iceservers = [];
  3746. $(res).find('>services>service').each(function (idx, el) {
  3747. el = $(el);
  3748. var dict = {};
  3749. var type = el.attr('type');
  3750. switch (type) {
  3751. case 'stun':
  3752. dict.url = 'stun:' + el.attr('host');
  3753. if (el.attr('port')) {
  3754. dict.url += ':' + el.attr('port');
  3755. }
  3756. iceservers.push(dict);
  3757. break;
  3758. case 'turn':
  3759. case 'turns':
  3760. dict.url = type + ':';
  3761. if (el.attr('username')) { // https://code.google.com/p/webrtc/issues/detail?id=1508
  3762. if (navigator.userAgent.match(/Chrom(e|ium)\/([0-9]+)\./) && parseInt(navigator.userAgent.match(/Chrom(e|ium)\/([0-9]+)\./)[2], 10) < 28) {
  3763. dict.url += el.attr('username') + '@';
  3764. } else {
  3765. dict.username = el.attr('username'); // only works in M28
  3766. }
  3767. }
  3768. dict.url += el.attr('host');
  3769. if (el.attr('port') && el.attr('port') != '3478') {
  3770. dict.url += ':' + el.attr('port');
  3771. }
  3772. if (el.attr('transport') && el.attr('transport') != 'udp') {
  3773. dict.url += '?transport=' + el.attr('transport');
  3774. }
  3775. if (el.attr('password')) {
  3776. dict.credential = el.attr('password');
  3777. }
  3778. iceservers.push(dict);
  3779. break;
  3780. }
  3781. });
  3782. self.ice_config.iceServers = iceservers;
  3783. },
  3784. function (err) {
  3785. console.warn('getting turn credentials failed', err);
  3786. console.warn('is mod_turncredentials or similar installed?');
  3787. }
  3788. );
  3789. // implement push?
  3790. },
  3791. /**
  3792. * Populates the log data
  3793. */
  3794. populateData: function () {
  3795. var data = {};
  3796. Object.keys(this.sessions).forEach(function (sid) {
  3797. var session = this.sessions[sid];
  3798. if (session.peerconnection && session.peerconnection.updateLog) {
  3799. // FIXME: should probably be a .dump call
  3800. data["jingle_" + session.sid] = {
  3801. updateLog: session.peerconnection.updateLog,
  3802. stats: session.peerconnection.stats,
  3803. url: window.location.href
  3804. };
  3805. }
  3806. });
  3807. return data;
  3808. }
  3809. });
  3810. };
  3811. },{"./JingleSession":1}],10:[function(require,module,exports){
  3812. /* global Strophe */
  3813. module.exports = function () {
  3814. Strophe.addConnectionPlugin('logger', {
  3815. // logs raw stanzas and makes them available for download as JSON
  3816. connection: null,
  3817. log: [],
  3818. init: function (conn) {
  3819. this.connection = conn;
  3820. this.connection.rawInput = this.log_incoming.bind(this);
  3821. this.connection.rawOutput = this.log_outgoing.bind(this);
  3822. },
  3823. log_incoming: function (stanza) {
  3824. this.log.push([new Date().getTime(), 'incoming', stanza]);
  3825. },
  3826. log_outgoing: function (stanza) {
  3827. this.log.push([new Date().getTime(), 'outgoing', stanza]);
  3828. }
  3829. });
  3830. };
  3831. },{}],11:[function(require,module,exports){
  3832. /* global $, $iq, config, connection, focusMucJid, forceMuted,
  3833. setAudioMuted, Strophe */
  3834. /**
  3835. * Moderate connection plugin.
  3836. */
  3837. module.exports = function (XMPP) {
  3838. Strophe.addConnectionPlugin('moderate', {
  3839. connection: null,
  3840. init: function (conn) {
  3841. this.connection = conn;
  3842. this.connection.addHandler(this.onMute.bind(this),
  3843. 'http://jitsi.org/jitmeet/audio',
  3844. 'iq',
  3845. 'set',
  3846. null,
  3847. null);
  3848. },
  3849. setMute: function (jid, mute) {
  3850. console.info("set mute", mute);
  3851. var iqToFocus = $iq({to: focusMucJid, type: 'set'})
  3852. .c('mute', {
  3853. xmlns: 'http://jitsi.org/jitmeet/audio',
  3854. jid: jid
  3855. })
  3856. .t(mute.toString())
  3857. .up();
  3858. this.connection.sendIQ(
  3859. iqToFocus,
  3860. function (result) {
  3861. console.log('set mute', result);
  3862. },
  3863. function (error) {
  3864. console.log('set mute error', error);
  3865. });
  3866. },
  3867. onMute: function (iq) {
  3868. var from = iq.getAttribute('from');
  3869. if (from !== focusMucJid) {
  3870. console.warn("Ignored mute from non focus peer");
  3871. return false;
  3872. }
  3873. var mute = $(iq).find('mute');
  3874. if (mute.length) {
  3875. var doMuteAudio = mute.text() === "true";
  3876. UI.setAudioMuted(doMuteAudio);
  3877. XMPP.forceMuted = doMuteAudio;
  3878. }
  3879. return true;
  3880. },
  3881. eject: function (jid) {
  3882. // We're not the focus, so can't terminate
  3883. //connection.jingle.terminateRemoteByJid(jid, 'kick');
  3884. this.connection.emuc.kick(jid);
  3885. }
  3886. });
  3887. }
  3888. },{}],12:[function(require,module,exports){
  3889. /* jshint -W117 */
  3890. module.exports = function() {
  3891. Strophe.addConnectionPlugin('rayo',
  3892. {
  3893. RAYO_XMLNS: 'urn:xmpp:rayo:1',
  3894. connection: null,
  3895. init: function (conn) {
  3896. this.connection = conn;
  3897. if (this.connection.disco) {
  3898. this.connection.disco.addFeature('urn:xmpp:rayo:client:1');
  3899. }
  3900. this.connection.addHandler(
  3901. this.onRayo.bind(this), this.RAYO_XMLNS, 'iq', 'set', null, null);
  3902. },
  3903. onRayo: function (iq) {
  3904. console.info("Rayo IQ", iq);
  3905. },
  3906. dial: function (to, from, roomName, roomPass) {
  3907. var self = this;
  3908. var req = $iq(
  3909. {
  3910. type: 'set',
  3911. to: focusMucJid
  3912. }
  3913. );
  3914. req.c('dial',
  3915. {
  3916. xmlns: this.RAYO_XMLNS,
  3917. to: to,
  3918. from: from
  3919. });
  3920. req.c('header',
  3921. {
  3922. name: 'JvbRoomName',
  3923. value: roomName
  3924. }).up();
  3925. if (roomPass !== null && roomPass.length) {
  3926. req.c('header',
  3927. {
  3928. name: 'JvbRoomPassword',
  3929. value: roomPass
  3930. }).up();
  3931. }
  3932. this.connection.sendIQ(
  3933. req,
  3934. function (result) {
  3935. console.info('Dial result ', result);
  3936. var resource = $(result).find('ref').attr('uri');
  3937. this.call_resource = resource.substr('xmpp:'.length);
  3938. console.info(
  3939. "Received call resource: " + this.call_resource);
  3940. },
  3941. function (error) {
  3942. console.info('Dial error ', error);
  3943. }
  3944. );
  3945. },
  3946. hang_up: function () {
  3947. if (!this.call_resource) {
  3948. console.warn("No call in progress");
  3949. return;
  3950. }
  3951. var self = this;
  3952. var req = $iq(
  3953. {
  3954. type: 'set',
  3955. to: this.call_resource
  3956. }
  3957. );
  3958. req.c('hangup',
  3959. {
  3960. xmlns: this.RAYO_XMLNS
  3961. });
  3962. this.connection.sendIQ(
  3963. req,
  3964. function (result) {
  3965. console.info('Hangup result ', result);
  3966. self.call_resource = null;
  3967. },
  3968. function (error) {
  3969. console.info('Hangup error ', error);
  3970. self.call_resource = null;
  3971. }
  3972. );
  3973. }
  3974. }
  3975. );
  3976. };
  3977. },{}],13:[function(require,module,exports){
  3978. /**
  3979. * Strophe logger implementation. Logs from level WARN and above.
  3980. */
  3981. module.exports = function () {
  3982. Strophe.log = function (level, msg) {
  3983. switch (level) {
  3984. case Strophe.LogLevel.WARN:
  3985. console.warn("Strophe: " + msg);
  3986. break;
  3987. case Strophe.LogLevel.ERROR:
  3988. case Strophe.LogLevel.FATAL:
  3989. console.error("Strophe: " + msg);
  3990. break;
  3991. }
  3992. };
  3993. Strophe.getStatusString = function (status) {
  3994. switch (status) {
  3995. case Strophe.Status.ERROR:
  3996. return "ERROR";
  3997. case Strophe.Status.CONNECTING:
  3998. return "CONNECTING";
  3999. case Strophe.Status.CONNFAIL:
  4000. return "CONNFAIL";
  4001. case Strophe.Status.AUTHENTICATING:
  4002. return "AUTHENTICATING";
  4003. case Strophe.Status.AUTHFAIL:
  4004. return "AUTHFAIL";
  4005. case Strophe.Status.CONNECTED:
  4006. return "CONNECTED";
  4007. case Strophe.Status.DISCONNECTED:
  4008. return "DISCONNECTED";
  4009. case Strophe.Status.DISCONNECTING:
  4010. return "DISCONNECTING";
  4011. case Strophe.Status.ATTACHED:
  4012. return "ATTACHED";
  4013. default:
  4014. return "unknown";
  4015. }
  4016. };
  4017. };
  4018. },{}],14:[function(require,module,exports){
  4019. var Moderator = require("./moderator");
  4020. var EventEmitter = require("events");
  4021. var Recording = require("./recording");
  4022. var SDP = require("./SDP");
  4023. var eventEmitter = new EventEmitter();
  4024. var connection = null;
  4025. var authenticatedUser = false;
  4026. var activecall = null;
  4027. function connect(jid, password, uiCredentials) {
  4028. var bosh
  4029. = uiCredentials.bosh || config.bosh || '/http-bind';
  4030. connection = new Strophe.Connection(bosh);
  4031. Moderator.setConnection(connection);
  4032. var settings = UI.getSettings();
  4033. var email = settings.email;
  4034. var displayName = settings.displayName;
  4035. if(email) {
  4036. connection.emuc.addEmailToPresence(email);
  4037. } else {
  4038. connection.emuc.addUserIdToPresence(settings.uid);
  4039. }
  4040. if(displayName) {
  4041. connection.emuc.addDisplayNameToPresence(displayName);
  4042. }
  4043. if (connection.disco) {
  4044. // for chrome, add multistream cap
  4045. }
  4046. connection.jingle.pc_constraints = RTC.getPCConstraints();
  4047. if (config.useIPv6) {
  4048. // https://code.google.com/p/webrtc/issues/detail?id=2828
  4049. if (!connection.jingle.pc_constraints.optional)
  4050. connection.jingle.pc_constraints.optional = [];
  4051. connection.jingle.pc_constraints.optional.push({googIPv6: true});
  4052. }
  4053. if(!password)
  4054. password = uiCredentials.password;
  4055. var anonymousConnectionFailed = false;
  4056. connection.connect(jid, password, function (status, msg) {
  4057. console.log('Strophe status changed to',
  4058. Strophe.getStatusString(status));
  4059. if (status === Strophe.Status.CONNECTED) {
  4060. if (config.useStunTurn) {
  4061. connection.jingle.getStunAndTurnCredentials();
  4062. }
  4063. UI.disableConnect();
  4064. console.info("My Jabber ID: " + connection.jid);
  4065. if(password)
  4066. authenticatedUser = true;
  4067. maybeDoJoin();
  4068. } else if (status === Strophe.Status.CONNFAIL) {
  4069. if(msg === 'x-strophe-bad-non-anon-jid') {
  4070. anonymousConnectionFailed = true;
  4071. }
  4072. } else if (status === Strophe.Status.DISCONNECTED) {
  4073. if(anonymousConnectionFailed) {
  4074. // prompt user for username and password
  4075. XMPP.promptLogin();
  4076. }
  4077. } else if (status === Strophe.Status.AUTHFAIL) {
  4078. // wrong password or username, prompt user
  4079. XMPP.promptLogin();
  4080. }
  4081. });
  4082. }
  4083. function maybeDoJoin() {
  4084. if (connection && connection.connected &&
  4085. Strophe.getResourceFromJid(connection.jid)
  4086. && (RTC.localAudio || RTC.localVideo)) {
  4087. // .connected is true while connecting?
  4088. doJoin();
  4089. }
  4090. }
  4091. function doJoin() {
  4092. var roomName = UI.generateRoomName();
  4093. Moderator.allocateConferenceFocus(
  4094. roomName, UI.checkForNicknameAndJoin);
  4095. }
  4096. function initStrophePlugins()
  4097. {
  4098. require("./strophe.emuc")(XMPP, eventEmitter);
  4099. require("./strophe.jingle")();
  4100. require("./strophe.moderate")(XMPP);
  4101. require("./strophe.util")();
  4102. require("./strophe.rayo")();
  4103. require("./strophe.logger")();
  4104. }
  4105. function registerListeners() {
  4106. RTC.addStreamListener(maybeDoJoin,
  4107. StreamEventTypes.EVENT_TYPE_LOCAL_CREATED);
  4108. }
  4109. function setupEvents() {
  4110. $(window).bind('beforeunload', function () {
  4111. if (connection && connection.connected) {
  4112. // ensure signout
  4113. $.ajax({
  4114. type: 'POST',
  4115. url: config.bosh,
  4116. async: false,
  4117. cache: false,
  4118. contentType: 'application/xml',
  4119. data: "<body rid='" + (connection.rid || connection._proto.rid)
  4120. + "' xmlns='http://jabber.org/protocol/httpbind' sid='"
  4121. + (connection.sid || connection._proto.sid)
  4122. + "' type='terminate'>" +
  4123. "<presence xmlns='jabber:client' type='unavailable'/>" +
  4124. "</body>",
  4125. success: function (data) {
  4126. console.log('signed out');
  4127. console.log(data);
  4128. },
  4129. error: function (XMLHttpRequest, textStatus, errorThrown) {
  4130. console.log('signout error',
  4131. textStatus + ' (' + errorThrown + ')');
  4132. }
  4133. });
  4134. }
  4135. XMPP.disposeConference(true);
  4136. });
  4137. }
  4138. var XMPP = {
  4139. sessionTerminated: false,
  4140. /**
  4141. * Remembers if we were muted by the focus.
  4142. * @type {boolean}
  4143. */
  4144. forceMuted: false,
  4145. start: function (uiCredentials) {
  4146. setupEvents();
  4147. initStrophePlugins();
  4148. registerListeners();
  4149. Moderator.init();
  4150. var jid = uiCredentials.jid ||
  4151. config.hosts.anonymousdomain ||
  4152. config.hosts.domain ||
  4153. window.location.hostname;
  4154. connect(jid, null, uiCredentials);
  4155. },
  4156. promptLogin: function () {
  4157. UI.showLoginPopup(connect);
  4158. },
  4159. joinRooom: function(roomName, useNicks, nick)
  4160. {
  4161. var roomjid;
  4162. roomjid = roomName;
  4163. if (useNicks) {
  4164. if (nick) {
  4165. roomjid += '/' + nick;
  4166. } else {
  4167. roomjid += '/' + Strophe.getNodeFromJid(connection.jid);
  4168. }
  4169. } else {
  4170. var tmpJid = Strophe.getNodeFromJid(connection.jid);
  4171. if(!authenticatedUser)
  4172. tmpJid = tmpJid.substr(0, 8);
  4173. roomjid += '/' + tmpJid;
  4174. }
  4175. connection.emuc.doJoin(roomjid);
  4176. },
  4177. myJid: function () {
  4178. if(!connection)
  4179. return null;
  4180. return connection.emuc.myroomjid;
  4181. },
  4182. myResource: function () {
  4183. if(!connection || ! connection.emuc.myroomjid)
  4184. return null;
  4185. return Strophe.getResourceFromJid(connection.emuc.myroomjid);
  4186. },
  4187. disposeConference: function (onUnload) {
  4188. eventEmitter.emit(XMPPEvents.DISPOSE_CONFERENCE, onUnload);
  4189. var handler = activecall;
  4190. if (handler && handler.peerconnection) {
  4191. // FIXME: probably removing streams is not required and close() should
  4192. // be enough
  4193. if (RTC.localAudio) {
  4194. handler.peerconnection.removeStream(RTC.localAudio.getOriginalStream(), onUnload);
  4195. }
  4196. if (RTC.localVideo) {
  4197. handler.peerconnection.removeStream(RTC.localVideo.getOriginalStream(), onUnload);
  4198. }
  4199. handler.peerconnection.close();
  4200. }
  4201. activecall = null;
  4202. if(!onUnload)
  4203. {
  4204. this.sessionTerminated = true;
  4205. connection.emuc.doLeave();
  4206. }
  4207. },
  4208. addListener: function(type, listener)
  4209. {
  4210. eventEmitter.on(type, listener);
  4211. },
  4212. removeListener: function (type, listener) {
  4213. eventEmitter.removeListener(type, listener);
  4214. },
  4215. allocateConferenceFocus: function(roomName, callback) {
  4216. Moderator.allocateConferenceFocus(roomName, callback);
  4217. },
  4218. isModerator: function () {
  4219. return Moderator.isModerator();
  4220. },
  4221. isSipGatewayEnabled: function () {
  4222. return Moderator.isSipGatewayEnabled();
  4223. },
  4224. isExternalAuthEnabled: function () {
  4225. return Moderator.isExternalAuthEnabled();
  4226. },
  4227. switchStreams: function (stream, oldStream, callback) {
  4228. if (activecall) {
  4229. // FIXME: will block switchInProgress on true value in case of exception
  4230. activecall.switchStreams(stream, oldStream, callback);
  4231. } else {
  4232. // We are done immediately
  4233. console.error("No conference handler");
  4234. UI.messageHandler.showError('Error',
  4235. 'Unable to switch video stream.');
  4236. callback();
  4237. }
  4238. },
  4239. setVideoMute: function (mute, callback, options) {
  4240. if(activecall && connection && RTC.localVideo)
  4241. {
  4242. activecall.setVideoMute(mute, callback, options);
  4243. }
  4244. },
  4245. setAudioMute: function (mute, callback) {
  4246. if (!(connection && RTC.localAudio)) {
  4247. return false;
  4248. }
  4249. if (this.forceMuted && !mute) {
  4250. console.info("Asking focus for unmute");
  4251. connection.moderate.setMute(connection.emuc.myroomjid, mute);
  4252. // FIXME: wait for result before resetting muted status
  4253. this.forceMuted = false;
  4254. }
  4255. if (mute == RTC.localAudio.isMuted()) {
  4256. // Nothing to do
  4257. return true;
  4258. }
  4259. // It is not clear what is the right way to handle multiple tracks.
  4260. // So at least make sure that they are all muted or all unmuted and
  4261. // that we send presence just once.
  4262. RTC.localAudio.mute();
  4263. // isMuted is the opposite of audioEnabled
  4264. connection.emuc.addAudioInfoToPresence(mute);
  4265. connection.emuc.sendPresence();
  4266. callback();
  4267. return true;
  4268. },
  4269. // Really mute video, i.e. dont even send black frames
  4270. muteVideo: function (pc, unmute) {
  4271. // FIXME: this probably needs another of those lovely state safeguards...
  4272. // which checks for iceconn == connected and sigstate == stable
  4273. pc.setRemoteDescription(pc.remoteDescription,
  4274. function () {
  4275. pc.createAnswer(
  4276. function (answer) {
  4277. var sdp = new SDP(answer.sdp);
  4278. if (sdp.media.length > 1) {
  4279. if (unmute)
  4280. sdp.media[1] = sdp.media[1].replace('a=recvonly', 'a=sendrecv');
  4281. else
  4282. sdp.media[1] = sdp.media[1].replace('a=sendrecv', 'a=recvonly');
  4283. sdp.raw = sdp.session + sdp.media.join('');
  4284. answer.sdp = sdp.raw;
  4285. }
  4286. pc.setLocalDescription(answer,
  4287. function () {
  4288. console.log('mute SLD ok');
  4289. },
  4290. function (error) {
  4291. console.log('mute SLD error');
  4292. UI.messageHandler.showError('Error',
  4293. 'Oops! Something went wrong and we failed to ' +
  4294. 'mute! (SLD Failure)');
  4295. }
  4296. );
  4297. },
  4298. function (error) {
  4299. console.log(error);
  4300. UI.messageHandler.showError();
  4301. }
  4302. );
  4303. },
  4304. function (error) {
  4305. console.log('muteVideo SRD error');
  4306. UI.messageHandler.showError('Error',
  4307. 'Oops! Something went wrong and we failed to stop video!' +
  4308. '(SRD Failure)');
  4309. }
  4310. );
  4311. },
  4312. toggleRecording: function (tokenEmptyCallback,
  4313. startingCallback, startedCallback) {
  4314. Recording.toggleRecording(tokenEmptyCallback,
  4315. startingCallback, startedCallback);
  4316. },
  4317. addToPresence: function (name, value, dontSend) {
  4318. switch (name)
  4319. {
  4320. case "displayName":
  4321. connection.emuc.addDisplayNameToPresence(value);
  4322. break;
  4323. case "etherpad":
  4324. connection.emuc.addEtherpadToPresence(value);
  4325. break;
  4326. case "prezi":
  4327. connection.emuc.addPreziToPresence(value, 0);
  4328. break;
  4329. case "preziSlide":
  4330. connection.emuc.addCurrentSlideToPresence(value);
  4331. break;
  4332. case "connectionQuality":
  4333. connection.emuc.addConnectionInfoToPresence(value);
  4334. break;
  4335. case "email":
  4336. connection.emuc.addEmailToPresence(value);
  4337. default :
  4338. console.log("Unknown tag for presence.");
  4339. return;
  4340. }
  4341. if(!dontSend)
  4342. connection.emuc.sendPresence();
  4343. },
  4344. sendLogs: function (content) {
  4345. // XEP-0337-ish
  4346. var message = $msg({to: focusMucJid, type: 'normal'});
  4347. message.c('log', { xmlns: 'urn:xmpp:eventlog',
  4348. id: 'PeerConnectionStats'});
  4349. message.c('message').t(content).up();
  4350. if (deflate) {
  4351. message.c('tag', {name: "deflated", value: "true"}).up();
  4352. }
  4353. message.up();
  4354. connection.send(message);
  4355. },
  4356. populateData: function () {
  4357. var data = {};
  4358. if (connection.jingle) {
  4359. data = connection.jingle.populateData();
  4360. }
  4361. return data;
  4362. },
  4363. getLogger: function () {
  4364. if(connection.logger)
  4365. return connection.logger.log;
  4366. return null;
  4367. },
  4368. getPrezi: function () {
  4369. return connection.emuc.getPrezi(this.myJid());
  4370. },
  4371. removePreziFromPresence: function () {
  4372. connection.emuc.removePreziFromPresence();
  4373. connection.emuc.sendPresence();
  4374. },
  4375. sendChatMessage: function (message, nickname) {
  4376. connection.emuc.sendMessage(message, nickname);
  4377. },
  4378. setSubject: function (topic) {
  4379. connection.emuc.setSubject(topic);
  4380. },
  4381. lockRoom: function (key, onSuccess, onError, onNotSupported) {
  4382. connection.emuc.lockRoom(key, onSuccess, onError, onNotSupported);
  4383. },
  4384. dial: function (to, from, roomName,roomPass) {
  4385. connection.rayo.dial(to, from, roomName,roomPass);
  4386. },
  4387. setMute: function (jid, mute) {
  4388. connection.moderate.setMute(jid, mute);
  4389. },
  4390. eject: function (jid) {
  4391. connection.moderate.eject(jid);
  4392. },
  4393. findJidFromResource: function (resource) {
  4394. connection.emuc.findJidFromResource(resource);
  4395. },
  4396. getMembers: function () {
  4397. return connection.emuc.members;
  4398. }
  4399. };
  4400. module.exports = XMPP;
  4401. },{"./SDP":2,"./moderator":6,"./recording":7,"./strophe.emuc":8,"./strophe.jingle":9,"./strophe.logger":10,"./strophe.moderate":11,"./strophe.rayo":12,"./strophe.util":13,"events":15}],15:[function(require,module,exports){
  4402. // Copyright Joyent, Inc. and other Node contributors.
  4403. //
  4404. // Permission is hereby granted, free of charge, to any person obtaining a
  4405. // copy of this software and associated documentation files (the
  4406. // "Software"), to deal in the Software without restriction, including
  4407. // without limitation the rights to use, copy, modify, merge, publish,
  4408. // distribute, sublicense, and/or sell copies of the Software, and to permit
  4409. // persons to whom the Software is furnished to do so, subject to the
  4410. // following conditions:
  4411. //
  4412. // The above copyright notice and this permission notice shall be included
  4413. // in all copies or substantial portions of the Software.
  4414. //
  4415. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
  4416. // OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
  4417. // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
  4418. // NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
  4419. // DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
  4420. // OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
  4421. // USE OR OTHER DEALINGS IN THE SOFTWARE.
  4422. function EventEmitter() {
  4423. this._events = this._events || {};
  4424. this._maxListeners = this._maxListeners || undefined;
  4425. }
  4426. module.exports = EventEmitter;
  4427. // Backwards-compat with node 0.10.x
  4428. EventEmitter.EventEmitter = EventEmitter;
  4429. EventEmitter.prototype._events = undefined;
  4430. EventEmitter.prototype._maxListeners = undefined;
  4431. // By default EventEmitters will print a warning if more than 10 listeners are
  4432. // added to it. This is a useful default which helps finding memory leaks.
  4433. EventEmitter.defaultMaxListeners = 10;
  4434. // Obviously not all Emitters should be limited to 10. This function allows
  4435. // that to be increased. Set to zero for unlimited.
  4436. EventEmitter.prototype.setMaxListeners = function(n) {
  4437. if (!isNumber(n) || n < 0 || isNaN(n))
  4438. throw TypeError('n must be a positive number');
  4439. this._maxListeners = n;
  4440. return this;
  4441. };
  4442. EventEmitter.prototype.emit = function(type) {
  4443. var er, handler, len, args, i, listeners;
  4444. if (!this._events)
  4445. this._events = {};
  4446. // If there is no 'error' event listener then throw.
  4447. if (type === 'error') {
  4448. if (!this._events.error ||
  4449. (isObject(this._events.error) && !this._events.error.length)) {
  4450. er = arguments[1];
  4451. if (er instanceof Error) {
  4452. throw er; // Unhandled 'error' event
  4453. } else {
  4454. throw TypeError('Uncaught, unspecified "error" event.');
  4455. }
  4456. return false;
  4457. }
  4458. }
  4459. handler = this._events[type];
  4460. if (isUndefined(handler))
  4461. return false;
  4462. if (isFunction(handler)) {
  4463. switch (arguments.length) {
  4464. // fast cases
  4465. case 1:
  4466. handler.call(this);
  4467. break;
  4468. case 2:
  4469. handler.call(this, arguments[1]);
  4470. break;
  4471. case 3:
  4472. handler.call(this, arguments[1], arguments[2]);
  4473. break;
  4474. // slower
  4475. default:
  4476. len = arguments.length;
  4477. args = new Array(len - 1);
  4478. for (i = 1; i < len; i++)
  4479. args[i - 1] = arguments[i];
  4480. handler.apply(this, args);
  4481. }
  4482. } else if (isObject(handler)) {
  4483. len = arguments.length;
  4484. args = new Array(len - 1);
  4485. for (i = 1; i < len; i++)
  4486. args[i - 1] = arguments[i];
  4487. listeners = handler.slice();
  4488. len = listeners.length;
  4489. for (i = 0; i < len; i++)
  4490. listeners[i].apply(this, args);
  4491. }
  4492. return true;
  4493. };
  4494. EventEmitter.prototype.addListener = function(type, listener) {
  4495. var m;
  4496. if (!isFunction(listener))
  4497. throw TypeError('listener must be a function');
  4498. if (!this._events)
  4499. this._events = {};
  4500. // To avoid recursion in the case that type === "newListener"! Before
  4501. // adding it to the listeners, first emit "newListener".
  4502. if (this._events.newListener)
  4503. this.emit('newListener', type,
  4504. isFunction(listener.listener) ?
  4505. listener.listener : listener);
  4506. if (!this._events[type])
  4507. // Optimize the case of one listener. Don't need the extra array object.
  4508. this._events[type] = listener;
  4509. else if (isObject(this._events[type]))
  4510. // If we've already got an array, just append.
  4511. this._events[type].push(listener);
  4512. else
  4513. // Adding the second element, need to change to array.
  4514. this._events[type] = [this._events[type], listener];
  4515. // Check for listener leak
  4516. if (isObject(this._events[type]) && !this._events[type].warned) {
  4517. var m;
  4518. if (!isUndefined(this._maxListeners)) {
  4519. m = this._maxListeners;
  4520. } else {
  4521. m = EventEmitter.defaultMaxListeners;
  4522. }
  4523. if (m && m > 0 && this._events[type].length > m) {
  4524. this._events[type].warned = true;
  4525. console.error('(node) warning: possible EventEmitter memory ' +
  4526. 'leak detected. %d listeners added. ' +
  4527. 'Use emitter.setMaxListeners() to increase limit.',
  4528. this._events[type].length);
  4529. if (typeof console.trace === 'function') {
  4530. // not supported in IE 10
  4531. console.trace();
  4532. }
  4533. }
  4534. }
  4535. return this;
  4536. };
  4537. EventEmitter.prototype.on = EventEmitter.prototype.addListener;
  4538. EventEmitter.prototype.once = function(type, listener) {
  4539. if (!isFunction(listener))
  4540. throw TypeError('listener must be a function');
  4541. var fired = false;
  4542. function g() {
  4543. this.removeListener(type, g);
  4544. if (!fired) {
  4545. fired = true;
  4546. listener.apply(this, arguments);
  4547. }
  4548. }
  4549. g.listener = listener;
  4550. this.on(type, g);
  4551. return this;
  4552. };
  4553. // emits a 'removeListener' event iff the listener was removed
  4554. EventEmitter.prototype.removeListener = function(type, listener) {
  4555. var list, position, length, i;
  4556. if (!isFunction(listener))
  4557. throw TypeError('listener must be a function');
  4558. if (!this._events || !this._events[type])
  4559. return this;
  4560. list = this._events[type];
  4561. length = list.length;
  4562. position = -1;
  4563. if (list === listener ||
  4564. (isFunction(list.listener) && list.listener === listener)) {
  4565. delete this._events[type];
  4566. if (this._events.removeListener)
  4567. this.emit('removeListener', type, listener);
  4568. } else if (isObject(list)) {
  4569. for (i = length; i-- > 0;) {
  4570. if (list[i] === listener ||
  4571. (list[i].listener && list[i].listener === listener)) {
  4572. position = i;
  4573. break;
  4574. }
  4575. }
  4576. if (position < 0)
  4577. return this;
  4578. if (list.length === 1) {
  4579. list.length = 0;
  4580. delete this._events[type];
  4581. } else {
  4582. list.splice(position, 1);
  4583. }
  4584. if (this._events.removeListener)
  4585. this.emit('removeListener', type, listener);
  4586. }
  4587. return this;
  4588. };
  4589. EventEmitter.prototype.removeAllListeners = function(type) {
  4590. var key, listeners;
  4591. if (!this._events)
  4592. return this;
  4593. // not listening for removeListener, no need to emit
  4594. if (!this._events.removeListener) {
  4595. if (arguments.length === 0)
  4596. this._events = {};
  4597. else if (this._events[type])
  4598. delete this._events[type];
  4599. return this;
  4600. }
  4601. // emit removeListener for all listeners on all events
  4602. if (arguments.length === 0) {
  4603. for (key in this._events) {
  4604. if (key === 'removeListener') continue;
  4605. this.removeAllListeners(key);
  4606. }
  4607. this.removeAllListeners('removeListener');
  4608. this._events = {};
  4609. return this;
  4610. }
  4611. listeners = this._events[type];
  4612. if (isFunction(listeners)) {
  4613. this.removeListener(type, listeners);
  4614. } else {
  4615. // LIFO order
  4616. while (listeners.length)
  4617. this.removeListener(type, listeners[listeners.length - 1]);
  4618. }
  4619. delete this._events[type];
  4620. return this;
  4621. };
  4622. EventEmitter.prototype.listeners = function(type) {
  4623. var ret;
  4624. if (!this._events || !this._events[type])
  4625. ret = [];
  4626. else if (isFunction(this._events[type]))
  4627. ret = [this._events[type]];
  4628. else
  4629. ret = this._events[type].slice();
  4630. return ret;
  4631. };
  4632. EventEmitter.listenerCount = function(emitter, type) {
  4633. var ret;
  4634. if (!emitter._events || !emitter._events[type])
  4635. ret = 0;
  4636. else if (isFunction(emitter._events[type]))
  4637. ret = 1;
  4638. else
  4639. ret = emitter._events[type].length;
  4640. return ret;
  4641. };
  4642. function isFunction(arg) {
  4643. return typeof arg === 'function';
  4644. }
  4645. function isNumber(arg) {
  4646. return typeof arg === 'number';
  4647. }
  4648. function isObject(arg) {
  4649. return typeof arg === 'object' && arg !== null;
  4650. }
  4651. function isUndefined(arg) {
  4652. return arg === void 0;
  4653. }
  4654. },{}]},{},[14])(14)
  4655. });
  4656. //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["/usr/local/lib/node_modules/browserify/node_modules/browser-pack/_prelude.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/xmpp/JingleSession.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/xmpp/SDP.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/xmpp/SDPDiffer.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/xmpp/SDPUtil.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/xmpp/TraceablePeerConnection.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/xmpp/moderator.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/xmpp/recording.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/xmpp/strophe.emuc.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/xmpp/strophe.jingle.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/xmpp/strophe.logger.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/xmpp/strophe.moderate.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/xmpp/strophe.rayo.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/xmpp/strophe.util.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/xmpp/xmpp.js","/usr/local/lib/node_modules/browserify/node_modules/events/events.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC52CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5mBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5VA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1QA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/lBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9UA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3ZA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","/* jshint -W117 */\nvar TraceablePeerConnection = require(\"./TraceablePeerConnection\");\nvar SDPDiffer = require(\"./SDPDiffer\");\nvar SDPUtil = require(\"./SDPUtil\");\nvar SDP = require(\"./SDP\");\n\n// Jingle stuff\nfunction JingleSession(me, sid, connection, service) {\n    this.me = me;\n    this.sid = sid;\n    this.connection = connection;\n    this.initiator = null;\n    this.responder = null;\n    this.isInitiator = null;\n    this.peerjid = null;\n    this.state = null;\n    this.localSDP = null;\n    this.remoteSDP = null;\n    this.relayedStreams = [];\n    this.startTime = null;\n    this.stopTime = null;\n    this.media_constraints = null;\n    this.pc_constraints = null;\n    this.ice_config = {};\n    this.drip_container = [];\n    this.service = service;\n\n    this.usetrickle = true;\n    this.usepranswer = false; // early transport warmup -- mind you, this might fail. depends on webrtc issue 1718\n    this.usedrip = false; // dripping is sending trickle candidates not one-by-one\n\n    this.hadstuncandidate = false;\n    this.hadturncandidate = false;\n    this.lasticecandidate = false;\n\n    this.statsinterval = null;\n\n    this.reason = null;\n\n    this.addssrc = [];\n    this.removessrc = [];\n    this.pendingop = null;\n    this.switchstreams = false;\n\n    this.wait = true;\n    this.localStreamsSSRC = null;\n\n    /**\n     * The indicator which determines whether the (local) video has been muted\n     * in response to a user command in contrast to an automatic decision made\n     * by the application logic.\n     */\n    this.videoMuteByUser = false;\n}\n\nJingleSession.prototype.initiate = function (peerjid, isInitiator) {\n    var self = this;\n    if (this.state !== null) {\n        console.error('attempt to initiate on session ' + this.sid +\n            'in state ' + this.state);\n        return;\n    }\n    this.isInitiator = isInitiator;\n    this.state = 'pending';\n    this.initiator = isInitiator ? this.me : peerjid;\n    this.responder = !isInitiator ? this.me : peerjid;\n    this.peerjid = peerjid;\n    this.hadstuncandidate = false;\n    this.hadturncandidate = false;\n    this.lasticecandidate = false;\n\n    this.peerconnection\n        = new TraceablePeerConnection(\n            this.connection.jingle.ice_config,\n            this.connection.jingle.pc_constraints );\n\n    this.peerconnection.onicecandidate = function (event) {\n        self.sendIceCandidate(event.candidate);\n    };\n    this.peerconnection.onaddstream = function (event) {\n        console.log(\"REMOTE STREAM ADDED: \" + event.stream + \" - \" + event.stream.id);\n        self.remoteStreamAdded(event);\n    };\n    this.peerconnection.onremovestream = function (event) {\n        // Remove the stream from remoteStreams\n        // FIXME: remotestreamremoved.jingle not defined anywhere(unused)\n        $(document).trigger('remotestreamremoved.jingle', [event, self.sid]);\n    };\n    this.peerconnection.onsignalingstatechange = function (event) {\n        if (!(self && self.peerconnection)) return;\n    };\n    this.peerconnection.oniceconnectionstatechange = function (event) {\n        if (!(self && self.peerconnection)) return;\n        switch (self.peerconnection.iceConnectionState) {\n            case 'connected':\n                this.startTime = new Date();\n                break;\n            case 'disconnected':\n                this.stopTime = new Date();\n                break;\n        }\n        onIceConnectionStateChange(self.sid, self);\n    };\n    // add any local and relayed stream\n    RTC.localStreams.forEach(function(stream) {\n        self.peerconnection.addStream(stream.getOriginalStream());\n    });\n    this.relayedStreams.forEach(function(stream) {\n        self.peerconnection.addStream(stream);\n    });\n};\n\nfunction onIceConnectionStateChange(sid, session) {\n    switch (session.peerconnection.iceConnectionState) {\n        case 'checking':\n            session.timeChecking = (new Date()).getTime();\n            session.firstconnect = true;\n            break;\n        case 'completed': // on caller side\n        case 'connected':\n            if (session.firstconnect) {\n                session.firstconnect = false;\n                var metadata = {};\n                metadata.setupTime\n                    = (new Date()).getTime() - session.timeChecking;\n                session.peerconnection.getStats(function (res) {\n                    if(res && res.result) {\n                        res.result().forEach(function (report) {\n                            if (report.type == 'googCandidatePair' &&\n                                report.stat('googActiveConnection') == 'true') {\n                                metadata.localCandidateType\n                                    = report.stat('googLocalCandidateType');\n                                metadata.remoteCandidateType\n                                    = report.stat('googRemoteCandidateType');\n\n                                // log pair as well so we can get nice pie\n                                // charts\n                                metadata.candidatePair\n                                    = report.stat('googLocalCandidateType') +\n                                        ';' +\n                                        report.stat('googRemoteCandidateType');\n\n                                if (report.stat('googRemoteAddress').indexOf('[') === 0)\n                                {\n                                    metadata.ipv6 = true;\n                                }\n                            }\n                        });\n                    }\n                });\n            }\n            break;\n    }\n}\n\nJingleSession.prototype.accept = function () {\n    var self = this;\n    this.state = 'active';\n\n    var pranswer = this.peerconnection.localDescription;\n    if (!pranswer || pranswer.type != 'pranswer') {\n        return;\n    }\n    console.log('going from pranswer to answer');\n    if (this.usetrickle) {\n        // remove candidates already sent from session-accept\n        var lines = SDPUtil.find_lines(pranswer.sdp, 'a=candidate:');\n        for (var i = 0; i < lines.length; i++) {\n            pranswer.sdp = pranswer.sdp.replace(lines[i] + '\\r\\n', '');\n        }\n    }\n    while (SDPUtil.find_line(pranswer.sdp, 'a=inactive')) {\n        // FIXME: change any inactive to sendrecv or whatever they were originally\n        pranswer.sdp = pranswer.sdp.replace('a=inactive', 'a=sendrecv');\n    }\n    pranswer = simulcast.reverseTransformLocalDescription(pranswer);\n    var prsdp = new SDP(pranswer.sdp);\n    var accept = $iq({to: this.peerjid,\n        type: 'set'})\n        .c('jingle', {xmlns: 'urn:xmpp:jingle:1',\n            action: 'session-accept',\n            initiator: this.initiator,\n            responder: this.responder,\n            sid: this.sid });\n    prsdp.toJingle(accept, this.initiator == this.me ? 'initiator' : 'responder', this.localStreamsSSRC);\n    var sdp = this.peerconnection.localDescription.sdp;\n    while (SDPUtil.find_line(sdp, 'a=inactive')) {\n        // FIXME: change any inactive to sendrecv or whatever they were originally\n        sdp = sdp.replace('a=inactive', 'a=sendrecv');\n    }\n    var self = this;\n    this.peerconnection.setLocalDescription(new RTCSessionDescription({type: 'answer', sdp: sdp}),\n        function () {\n            //console.log('setLocalDescription success');\n            self.setLocalDescription();\n\n            self.connection.sendIQ(accept,\n                function () {\n                    var ack = {};\n                    ack.source = 'answer';\n                    $(document).trigger('ack.jingle', [self.sid, ack]);\n                },\n                function (stanza) {\n                    var error = ($(stanza).find('error').length) ? {\n                        code: $(stanza).find('error').attr('code'),\n                        reason: $(stanza).find('error :first')[0].tagName\n                    }:{};\n                    error.source = 'answer';\n                    JingleSession.onJingleError(self.sid, error);\n                },\n                10000);\n        },\n        function (e) {\n            console.error('setLocalDescription failed', e);\n        }\n    );\n};\n\nJingleSession.prototype.terminate = function (reason) {\n    this.state = 'ended';\n    this.reason = reason;\n    this.peerconnection.close();\n    if (this.statsinterval !== null) {\n        window.clearInterval(this.statsinterval);\n        this.statsinterval = null;\n    }\n};\n\nJingleSession.prototype.active = function () {\n    return this.state == 'active';\n};\n\nJingleSession.prototype.sendIceCandidate = function (candidate) {\n    var self = this;\n    if (candidate && !this.lasticecandidate) {\n        var ice = SDPUtil.iceparams(this.localSDP.media[candidate.sdpMLineIndex], this.localSDP.session);\n        var jcand = SDPUtil.candidateToJingle(candidate.candidate);\n        if (!(ice && jcand)) {\n            console.error('failed to get ice && jcand');\n            return;\n        }\n        ice.xmlns = 'urn:xmpp:jingle:transports:ice-udp:1';\n\n        if (jcand.type === 'srflx') {\n            this.hadstuncandidate = true;\n        } else if (jcand.type === 'relay') {\n            this.hadturncandidate = true;\n        }\n\n        if (this.usetrickle) {\n            if (this.usedrip) {\n                if (this.drip_container.length === 0) {\n                    // start 20ms callout\n                    window.setTimeout(function () {\n                        if (self.drip_container.length === 0) return;\n                        self.sendIceCandidates(self.drip_container);\n                        self.drip_container = [];\n                    }, 20);\n\n                }\n                this.drip_container.push(candidate);\n                return;\n            } else {\n                self.sendIceCandidate([candidate]);\n            }\n        }\n    } else {\n        //console.log('sendIceCandidate: last candidate.');\n        if (!this.usetrickle) {\n            //console.log('should send full offer now...');\n            var init = $iq({to: this.peerjid,\n                type: 'set'})\n                .c('jingle', {xmlns: 'urn:xmpp:jingle:1',\n                    action: this.peerconnection.localDescription.type == 'offer' ? 'session-initiate' : 'session-accept',\n                    initiator: this.initiator,\n                    sid: this.sid});\n            this.localSDP = new SDP(this.peerconnection.localDescription.sdp);\n            var self = this;\n            var sendJingle = function (ssrc) {\n                if(!ssrc)\n                    ssrc = {};\n                self.localSDP.toJingle(init, self.initiator == self.me ? 'initiator' : 'responder', ssrc);\n                self.connection.sendIQ(init,\n                    function () {\n                        //console.log('session initiate ack');\n                        var ack = {};\n                        ack.source = 'offer';\n                        $(document).trigger('ack.jingle', [self.sid, ack]);\n                    },\n                    function (stanza) {\n                        self.state = 'error';\n                        self.peerconnection.close();\n                        var error = ($(stanza).find('error').length) ? {\n                            code: $(stanza).find('error').attr('code'),\n                            reason: $(stanza).find('error :first')[0].tagName,\n                        }:{};\n                        error.source = 'offer';\n                        JingleSession.onJingleError(self.sid, error);\n                    },\n                    10000);\n            }\n            sendJingle();\n        }\n        this.lasticecandidate = true;\n        console.log('Have we encountered any srflx candidates? ' + this.hadstuncandidate);\n        console.log('Have we encountered any relay candidates? ' + this.hadturncandidate);\n\n        if (!(this.hadstuncandidate || this.hadturncandidate) && this.peerconnection.signalingState != 'closed') {\n            $(document).trigger('nostuncandidates.jingle', [this.sid]);\n        }\n    }\n};\n\nJingleSession.prototype.sendIceCandidates = function (candidates) {\n    console.log('sendIceCandidates', candidates);\n    var cand = $iq({to: this.peerjid, type: 'set'})\n        .c('jingle', {xmlns: 'urn:xmpp:jingle:1',\n            action: 'transport-info',\n            initiator: this.initiator,\n            sid: this.sid});\n    for (var mid = 0; mid < this.localSDP.media.length; mid++) {\n        var cands = candidates.filter(function (el) { return el.sdpMLineIndex == mid; });\n        var mline = SDPUtil.parse_mline(this.localSDP.media[mid].split('\\r\\n')[0]);\n        if (cands.length > 0) {\n            var ice = SDPUtil.iceparams(this.localSDP.media[mid], this.localSDP.session);\n            ice.xmlns = 'urn:xmpp:jingle:transports:ice-udp:1';\n            cand.c('content', {creator: this.initiator == this.me ? 'initiator' : 'responder',\n                name: (cands[0].sdpMid? cands[0].sdpMid : mline.media)\n            }).c('transport', ice);\n            for (var i = 0; i < cands.length; i++) {\n                cand.c('candidate', SDPUtil.candidateToJingle(cands[i].candidate)).up();\n            }\n            // add fingerprint\n            if (SDPUtil.find_line(this.localSDP.media[mid], 'a=fingerprint:', this.localSDP.session)) {\n                var tmp = SDPUtil.parse_fingerprint(SDPUtil.find_line(this.localSDP.media[mid], 'a=fingerprint:', this.localSDP.session));\n                tmp.required = true;\n                cand.c(\n                    'fingerprint',\n                    {xmlns: 'urn:xmpp:jingle:apps:dtls:0'})\n                    .t(tmp.fingerprint);\n                delete tmp.fingerprint;\n                cand.attrs(tmp);\n                cand.up();\n            }\n            cand.up(); // transport\n            cand.up(); // content\n        }\n    }\n    // might merge last-candidate notification into this, but it is called alot later. See webrtc issue #2340\n    //console.log('was this the last candidate', this.lasticecandidate);\n    this.connection.sendIQ(cand,\n        function () {\n            var ack = {};\n            ack.source = 'transportinfo';\n            $(document).trigger('ack.jingle', [this.sid, ack]);\n        },\n        function (stanza) {\n            var error = ($(stanza).find('error').length) ? {\n                code: $(stanza).find('error').attr('code'),\n                reason: $(stanza).find('error :first')[0].tagName,\n            }:{};\n            error.source = 'transportinfo';\n            JingleSession.onJingleError(this.sid, error);\n        },\n        10000);\n};\n\n\nJingleSession.prototype.sendOffer = function () {\n    //console.log('sendOffer...');\n    var self = this;\n    this.peerconnection.createOffer(function (sdp) {\n            self.createdOffer(sdp);\n        },\n        function (e) {\n            console.error('createOffer failed', e);\n        },\n        this.media_constraints\n    );\n};\n\nJingleSession.prototype.createdOffer = function (sdp) {\n    //console.log('createdOffer', sdp);\n    var self = this;\n    this.localSDP = new SDP(sdp.sdp);\n    //this.localSDP.mangle();\n    var sendJingle = function () {\n        var init = $iq({to: this.peerjid,\n            type: 'set'})\n            .c('jingle', {xmlns: 'urn:xmpp:jingle:1',\n                action: 'session-initiate',\n                initiator: this.initiator,\n                sid: this.sid});\n        self.localSDP.toJingle(init, this.initiator == this.me ? 'initiator' : 'responder', this.localStreamsSSRC);\n        self.connection.sendIQ(init,\n            function () {\n                var ack = {};\n                ack.source = 'offer';\n                $(document).trigger('ack.jingle', [self.sid, ack]);\n            },\n            function (stanza) {\n                self.state = 'error';\n                self.peerconnection.close();\n                var error = ($(stanza).find('error').length) ? {\n                    code: $(stanza).find('error').attr('code'),\n                    reason: $(stanza).find('error :first')[0].tagName,\n                }:{};\n                error.source = 'offer';\n                JingleSession.onJingleError(self.sid, error);\n            },\n            10000);\n    }\n    sdp.sdp = this.localSDP.raw;\n    this.peerconnection.setLocalDescription(sdp,\n        function () {\n            if(self.usetrickle)\n            {\n                sendJingle();\n            }\n            self.setLocalDescription();\n            //console.log('setLocalDescription success');\n        },\n        function (e) {\n            console.error('setLocalDescription failed', e);\n        }\n    );\n    var cands = SDPUtil.find_lines(this.localSDP.raw, 'a=candidate:');\n    for (var i = 0; i < cands.length; i++) {\n        var cand = SDPUtil.parse_icecandidate(cands[i]);\n        if (cand.type == 'srflx') {\n            this.hadstuncandidate = true;\n        } else if (cand.type == 'relay') {\n            this.hadturncandidate = true;\n        }\n    }\n};\n\nJingleSession.prototype.setRemoteDescription = function (elem, desctype) {\n    //console.log('setting remote description... ', desctype);\n    this.remoteSDP = new SDP('');\n    this.remoteSDP.fromJingle(elem);\n    if (this.peerconnection.remoteDescription !== null) {\n        console.log('setRemoteDescription when remote description is not null, should be pranswer', this.peerconnection.remoteDescription);\n        if (this.peerconnection.remoteDescription.type == 'pranswer') {\n            var pranswer = new SDP(this.peerconnection.remoteDescription.sdp);\n            for (var i = 0; i < pranswer.media.length; i++) {\n                // make sure we have ice ufrag and pwd\n                if (!SDPUtil.find_line(this.remoteSDP.media[i], 'a=ice-ufrag:', this.remoteSDP.session)) {\n                    if (SDPUtil.find_line(pranswer.media[i], 'a=ice-ufrag:', pranswer.session)) {\n                        this.remoteSDP.media[i] += SDPUtil.find_line(pranswer.media[i], 'a=ice-ufrag:', pranswer.session) + '\\r\\n';\n                    } else {\n                        console.warn('no ice ufrag?');\n                    }\n                    if (SDPUtil.find_line(pranswer.media[i], 'a=ice-pwd:', pranswer.session)) {\n                        this.remoteSDP.media[i] += SDPUtil.find_line(pranswer.media[i], 'a=ice-pwd:', pranswer.session) + '\\r\\n';\n                    } else {\n                        console.warn('no ice pwd?');\n                    }\n                }\n                // copy over candidates\n                var lines = SDPUtil.find_lines(pranswer.media[i], 'a=candidate:');\n                for (var j = 0; j < lines.length; j++) {\n                    this.remoteSDP.media[i] += lines[j] + '\\r\\n';\n                }\n            }\n            this.remoteSDP.raw = this.remoteSDP.session + this.remoteSDP.media.join('');\n        }\n    }\n    var remotedesc = new RTCSessionDescription({type: desctype, sdp: this.remoteSDP.raw});\n\n    this.peerconnection.setRemoteDescription(remotedesc,\n        function () {\n            //console.log('setRemoteDescription success');\n        },\n        function (e) {\n            console.error('setRemoteDescription error', e);\n            JingleSession.onJingleFatalError(self, e);\n        }\n    );\n};\n\nJingleSession.prototype.addIceCandidate = function (elem) {\n    var self = this;\n    if (this.peerconnection.signalingState == 'closed') {\n        return;\n    }\n    if (!this.peerconnection.remoteDescription && this.peerconnection.signalingState == 'have-local-offer') {\n        console.log('trickle ice candidate arriving before session accept...');\n        // create a PRANSWER for setRemoteDescription\n        if (!this.remoteSDP) {\n            var cobbled = 'v=0\\r\\n' +\n                'o=- ' + '1923518516' + ' 2 IN IP4 0.0.0.0\\r\\n' +// FIXME\n                's=-\\r\\n' +\n                't=0 0\\r\\n';\n            // first, take some things from the local description\n            for (var i = 0; i < this.localSDP.media.length; i++) {\n                cobbled += SDPUtil.find_line(this.localSDP.media[i], 'm=') + '\\r\\n';\n                cobbled += SDPUtil.find_lines(this.localSDP.media[i], 'a=rtpmap:').join('\\r\\n') + '\\r\\n';\n                if (SDPUtil.find_line(this.localSDP.media[i], 'a=mid:')) {\n                    cobbled += SDPUtil.find_line(this.localSDP.media[i], 'a=mid:') + '\\r\\n';\n                }\n                cobbled += 'a=inactive\\r\\n';\n            }\n            this.remoteSDP = new SDP(cobbled);\n        }\n        // then add things like ice and dtls from remote candidate\n        elem.each(function () {\n            for (var i = 0; i < self.remoteSDP.media.length; i++) {\n                if (SDPUtil.find_line(self.remoteSDP.media[i], 'a=mid:' + $(this).attr('name')) ||\n                    self.remoteSDP.media[i].indexOf('m=' + $(this).attr('name')) === 0) {\n                    if (!SDPUtil.find_line(self.remoteSDP.media[i], 'a=ice-ufrag:')) {\n                        var tmp = $(this).find('transport');\n                        self.remoteSDP.media[i] += 'a=ice-ufrag:' + tmp.attr('ufrag') + '\\r\\n';\n                        self.remoteSDP.media[i] += 'a=ice-pwd:' + tmp.attr('pwd') + '\\r\\n';\n                        tmp = $(this).find('transport>fingerprint');\n                        if (tmp.length) {\n                            self.remoteSDP.media[i] += 'a=fingerprint:' + tmp.attr('hash') + ' ' + tmp.text() + '\\r\\n';\n                        } else {\n                            console.log('no dtls fingerprint (webrtc issue #1718?)');\n                            self.remoteSDP.media[i] += 'a=crypto:1 AES_CM_128_HMAC_SHA1_80 inline:BAADBAADBAADBAADBAADBAADBAADBAADBAADBAAD\\r\\n';\n                        }\n                        break;\n                    }\n                }\n            }\n        });\n        this.remoteSDP.raw = this.remoteSDP.session + this.remoteSDP.media.join('');\n\n        // we need a complete SDP with ice-ufrag/ice-pwd in all parts\n        // this makes the assumption that the PRANSWER is constructed such that the ice-ufrag is in all mediaparts\n        // but it could be in the session part as well. since the code above constructs this sdp this can't happen however\n        var iscomplete = this.remoteSDP.media.filter(function (mediapart) {\n            return SDPUtil.find_line(mediapart, 'a=ice-ufrag:');\n        }).length == this.remoteSDP.media.length;\n\n        if (iscomplete) {\n            console.log('setting pranswer');\n            try {\n                this.peerconnection.setRemoteDescription(new RTCSessionDescription({type: 'pranswer', sdp: this.remoteSDP.raw }),\n                    function() {\n                    },\n                    function(e) {\n                        console.log('setRemoteDescription pranswer failed', e.toString());\n                    });\n            } catch (e) {\n                console.error('setting pranswer failed', e);\n            }\n        } else {\n            //console.log('not yet setting pranswer');\n        }\n    }\n    // operate on each content element\n    elem.each(function () {\n        // would love to deactivate this, but firefox still requires it\n        var idx = -1;\n        var i;\n        for (i = 0; i < self.remoteSDP.media.length; i++) {\n            if (SDPUtil.find_line(self.remoteSDP.media[i], 'a=mid:' + $(this).attr('name')) ||\n                self.remoteSDP.media[i].indexOf('m=' + $(this).attr('name')) === 0) {\n                idx = i;\n                break;\n            }\n        }\n        if (idx == -1) { // fall back to localdescription\n            for (i = 0; i < self.localSDP.media.length; i++) {\n                if (SDPUtil.find_line(self.localSDP.media[i], 'a=mid:' + $(this).attr('name')) ||\n                    self.localSDP.media[i].indexOf('m=' + $(this).attr('name')) === 0) {\n                    idx = i;\n                    break;\n                }\n            }\n        }\n        var name = $(this).attr('name');\n        // TODO: check ice-pwd and ice-ufrag?\n        $(this).find('transport>candidate').each(function () {\n            var line, candidate;\n            line = SDPUtil.candidateFromJingle(this);\n            candidate = new RTCIceCandidate({sdpMLineIndex: idx,\n                sdpMid: name,\n                candidate: line});\n            try {\n                self.peerconnection.addIceCandidate(candidate);\n            } catch (e) {\n                console.error('addIceCandidate failed', e.toString(), line);\n            }\n        });\n    });\n};\n\nJingleSession.prototype.sendAnswer = function (provisional) {\n    //console.log('createAnswer', provisional);\n    var self = this;\n    this.peerconnection.createAnswer(\n        function (sdp) {\n            self.createdAnswer(sdp, provisional);\n        },\n        function (e) {\n            console.error('createAnswer failed', e);\n        },\n        this.media_constraints\n    );\n};\n\nJingleSession.prototype.createdAnswer = function (sdp, provisional) {\n    //console.log('createAnswer callback');\n    var self = this;\n    this.localSDP = new SDP(sdp.sdp);\n    //this.localSDP.mangle();\n    this.usepranswer = provisional === true;\n    if (this.usetrickle) {\n        if (this.usepranswer) {\n            sdp.type = 'pranswer';\n            for (var i = 0; i < this.localSDP.media.length; i++) {\n                this.localSDP.media[i] = this.localSDP.media[i].replace('a=sendrecv\\r\\n', 'a=inactive\\r\\n');\n            }\n            this.localSDP.raw = this.localSDP.session + '\\r\\n' + this.localSDP.media.join('');\n        }\n    }\n    var self = this;\n    var sendJingle = function (ssrcs) {\n\n                var accept = $iq({to: self.peerjid,\n                    type: 'set'})\n                    .c('jingle', {xmlns: 'urn:xmpp:jingle:1',\n                        action: 'session-accept',\n                        initiator: self.initiator,\n                        responder: self.responder,\n                        sid: self.sid });\n                var publicLocalDesc = simulcast.reverseTransformLocalDescription(sdp);\n                var publicLocalSDP = new SDP(publicLocalDesc.sdp);\n                publicLocalSDP.toJingle(accept, self.initiator == self.me ? 'initiator' : 'responder', ssrcs);\n                self.connection.sendIQ(accept,\n                    function () {\n                        var ack = {};\n                        ack.source = 'answer';\n                        $(document).trigger('ack.jingle', [self.sid, ack]);\n                    },\n                    function (stanza) {\n                        var error = ($(stanza).find('error').length) ? {\n                            code: $(stanza).find('error').attr('code'),\n                            reason: $(stanza).find('error :first')[0].tagName,\n                        }:{};\n                        error.source = 'answer';\n                        JingleSession.onJingleError(self.sid, error);\n                    },\n                    10000);\n    }\n    sdp.sdp = this.localSDP.raw;\n    this.peerconnection.setLocalDescription(sdp,\n        function () {\n\n            //console.log('setLocalDescription success');\n            if (self.usetrickle && !self.usepranswer) {\n                sendJingle();\n            }\n            self.setLocalDescription();\n        },\n        function (e) {\n            console.error('setLocalDescription failed', e);\n        }\n    );\n    var cands = SDPUtil.find_lines(this.localSDP.raw, 'a=candidate:');\n    for (var j = 0; j < cands.length; j++) {\n        var cand = SDPUtil.parse_icecandidate(cands[j]);\n        if (cand.type == 'srflx') {\n            this.hadstuncandidate = true;\n        } else if (cand.type == 'relay') {\n            this.hadturncandidate = true;\n        }\n    }\n};\n\nJingleSession.prototype.sendTerminate = function (reason, text) {\n    var self = this,\n        term = $iq({to: this.peerjid,\n            type: 'set'})\n            .c('jingle', {xmlns: 'urn:xmpp:jingle:1',\n                action: 'session-terminate',\n                initiator: this.initiator,\n                sid: this.sid})\n            .c('reason')\n            .c(reason || 'success');\n\n    if (text) {\n        term.up().c('text').t(text);\n    }\n\n    this.connection.sendIQ(term,\n        function () {\n            self.peerconnection.close();\n            self.peerconnection = null;\n            self.terminate();\n            var ack = {};\n            ack.source = 'terminate';\n            $(document).trigger('ack.jingle', [self.sid, ack]);\n        },\n        function (stanza) {\n            var error = ($(stanza).find('error').length) ? {\n                code: $(stanza).find('error').attr('code'),\n                reason: $(stanza).find('error :first')[0].tagName,\n            }:{};\n            $(document).trigger('ack.jingle', [self.sid, error]);\n        },\n        10000);\n    if (this.statsinterval !== null) {\n        window.clearInterval(this.statsinterval);\n        this.statsinterval = null;\n    }\n};\n\nJingleSession.prototype.addSource = function (elem, fromJid) {\n\n    var self = this;\n    // FIXME: dirty waiting\n    if (!this.peerconnection.localDescription)\n    {\n        console.warn(\"addSource - localDescription not ready yet\")\n        setTimeout(function()\n            {\n                self.addSource(elem, fromJid);\n            },\n            200\n        );\n        return;\n    }\n\n    console.log('addssrc', new Date().getTime());\n    console.log('ice', this.peerconnection.iceConnectionState);\n    var sdp = new SDP(this.peerconnection.remoteDescription.sdp);\n    var mySdp = new SDP(this.peerconnection.localDescription.sdp);\n\n    $(elem).each(function (idx, content) {\n        var name = $(content).attr('name');\n        var lines = '';\n        tmp = $(content).find('ssrc-group[xmlns=\"urn:xmpp:jingle:apps:rtp:ssma:0\"]').each(function() {\n            var semantics = this.getAttribute('semantics');\n            var ssrcs = $(this).find('>source').map(function () {\n                return this.getAttribute('ssrc');\n            }).get();\n\n            if (ssrcs.length != 0) {\n                lines += 'a=ssrc-group:' + semantics + ' ' + ssrcs.join(' ') + '\\r\\n';\n            }\n        });\n        tmp = $(content).find('source[xmlns=\"urn:xmpp:jingle:apps:rtp:ssma:0\"]'); // can handle both >source and >description>source\n        tmp.each(function () {\n            var ssrc = $(this).attr('ssrc');\n            if(mySdp.containsSSRC(ssrc)){\n                /**\n                 * This happens when multiple participants change their streams at the same time and\n                 * ColibriFocus.modifySources have to wait for stable state. In the meantime multiple\n                 * addssrc are scheduled for update IQ. See\n                 */\n                console.warn(\"Got add stream request for my own ssrc: \"+ssrc);\n                return;\n            }\n            $(this).find('>parameter').each(function () {\n                lines += 'a=ssrc:' + ssrc + ' ' + $(this).attr('name');\n                if ($(this).attr('value') && $(this).attr('value').length)\n                    lines += ':' + $(this).attr('value');\n                lines += '\\r\\n';\n            });\n        });\n        sdp.media.forEach(function(media, idx) {\n            if (!SDPUtil.find_line(media, 'a=mid:' + name))\n                return;\n            sdp.media[idx] += lines;\n            if (!self.addssrc[idx]) self.addssrc[idx] = '';\n            self.addssrc[idx] += lines;\n        });\n        sdp.raw = sdp.session + sdp.media.join('');\n    });\n    this.modifySources();\n};\n\nJingleSession.prototype.removeSource = function (elem, fromJid) {\n\n    var self = this;\n    // FIXME: dirty waiting\n    if (!this.peerconnection.localDescription)\n    {\n        console.warn(\"removeSource - localDescription not ready yet\")\n        setTimeout(function()\n            {\n                self.removeSource(elem, fromJid);\n            },\n            200\n        );\n        return;\n    }\n\n    console.log('removessrc', new Date().getTime());\n    console.log('ice', this.peerconnection.iceConnectionState);\n    var sdp = new SDP(this.peerconnection.remoteDescription.sdp);\n    var mySdp = new SDP(this.peerconnection.localDescription.sdp);\n\n    $(elem).each(function (idx, content) {\n        var name = $(content).attr('name');\n        var lines = '';\n        tmp = $(content).find('ssrc-group[xmlns=\"urn:xmpp:jingle:apps:rtp:ssma:0\"]').each(function() {\n            var semantics = this.getAttribute('semantics');\n            var ssrcs = $(this).find('>source').map(function () {\n                return this.getAttribute('ssrc');\n            }).get();\n\n            if (ssrcs.length != 0) {\n                lines += 'a=ssrc-group:' + semantics + ' ' + ssrcs.join(' ') + '\\r\\n';\n            }\n        });\n        tmp = $(content).find('source[xmlns=\"urn:xmpp:jingle:apps:rtp:ssma:0\"]'); // can handle both >source and >description>source\n        tmp.each(function () {\n            var ssrc = $(this).attr('ssrc');\n            // This should never happen, but can be useful for bug detection\n            if(mySdp.containsSSRC(ssrc)){\n                console.error(\"Got remove stream request for my own ssrc: \"+ssrc);\n                return;\n            }\n            $(this).find('>parameter').each(function () {\n                lines += 'a=ssrc:' + ssrc + ' ' + $(this).attr('name');\n                if ($(this).attr('value') && $(this).attr('value').length)\n                    lines += ':' + $(this).attr('value');\n                lines += '\\r\\n';\n            });\n        });\n        sdp.media.forEach(function(media, idx) {\n            if (!SDPUtil.find_line(media, 'a=mid:' + name))\n                return;\n            sdp.media[idx] += lines;\n            if (!self.removessrc[idx]) self.removessrc[idx] = '';\n            self.removessrc[idx] += lines;\n        });\n        sdp.raw = sdp.session + sdp.media.join('');\n    });\n    this.modifySources();\n};\n\nJingleSession.prototype.modifySources = function (successCallback) {\n    var self = this;\n    if (this.peerconnection.signalingState == 'closed') return;\n    if (!(this.addssrc.length || this.removessrc.length || this.pendingop !== null || this.switchstreams)){\n        // There is nothing to do since scheduled job might have been executed by another succeeding call\n        this.setLocalDescription();\n        if(successCallback){\n            successCallback();\n        }\n        return;\n    }\n\n    // FIXME: this is a big hack\n    // https://code.google.com/p/webrtc/issues/detail?id=2688\n    // ^ has been fixed.\n    if (!(this.peerconnection.signalingState == 'stable' && this.peerconnection.iceConnectionState == 'connected')) {\n        console.warn('modifySources not yet', this.peerconnection.signalingState, this.peerconnection.iceConnectionState);\n        this.wait = true;\n        window.setTimeout(function() { self.modifySources(successCallback); }, 250);\n        return;\n    }\n    if (this.wait) {\n        window.setTimeout(function() { self.modifySources(successCallback); }, 2500);\n        this.wait = false;\n        return;\n    }\n\n    // Reset switch streams flag\n    this.switchstreams = false;\n\n    var sdp = new SDP(this.peerconnection.remoteDescription.sdp);\n\n    // add sources\n    this.addssrc.forEach(function(lines, idx) {\n        sdp.media[idx] += lines;\n    });\n    this.addssrc = [];\n\n    // remove sources\n    this.removessrc.forEach(function(lines, idx) {\n        lines = lines.split('\\r\\n');\n        lines.pop(); // remove empty last element;\n        lines.forEach(function(line) {\n            sdp.media[idx] = sdp.media[idx].replace(line + '\\r\\n', '');\n        });\n    });\n    this.removessrc = [];\n\n    // FIXME:\n    // this was a hack for the situation when only one peer exists\n    // in the conference.\n    // check if still required and remove\n    if (sdp.media[0])\n        sdp.media[0] = sdp.media[0].replace('a=recvonly', 'a=sendrecv');\n    if (sdp.media[1])\n        sdp.media[1] = sdp.media[1].replace('a=recvonly', 'a=sendrecv');\n\n    sdp.raw = sdp.session + sdp.media.join('');\n    this.peerconnection.setRemoteDescription(new RTCSessionDescription({type: 'offer', sdp: sdp.raw}),\n        function() {\n\n            if(self.signalingState == 'closed') {\n                console.error(\"createAnswer attempt on closed state\");\n                return;\n            }\n\n            self.peerconnection.createAnswer(\n                function(modifiedAnswer) {\n                    // change video direction, see https://github.com/jitsi/jitmeet/issues/41\n                    if (self.pendingop !== null) {\n                        var sdp = new SDP(modifiedAnswer.sdp);\n                        if (sdp.media.length > 1) {\n                            switch(self.pendingop) {\n                                case 'mute':\n                                    sdp.media[1] = sdp.media[1].replace('a=sendrecv', 'a=recvonly');\n                                    break;\n                                case 'unmute':\n                                    sdp.media[1] = sdp.media[1].replace('a=recvonly', 'a=sendrecv');\n                                    break;\n                            }\n                            sdp.raw = sdp.session + sdp.media.join('');\n                            modifiedAnswer.sdp = sdp.raw;\n                        }\n                        self.pendingop = null;\n                    }\n\n                    // FIXME: pushing down an answer while ice connection state\n                    // is still checking is bad...\n                    //console.log(self.peerconnection.iceConnectionState);\n\n                    // trying to work around another chrome bug\n                    //modifiedAnswer.sdp = modifiedAnswer.sdp.replace(/a=setup:active/g, 'a=setup:actpass');\n                    self.peerconnection.setLocalDescription(modifiedAnswer,\n                        function() {\n                            //console.log('modified setLocalDescription ok');\n                            self.setLocalDescription();\n                            if(successCallback){\n                                successCallback();\n                            }\n                        },\n                        function(error) {\n                            console.error('modified setLocalDescription failed', error);\n                        }\n                    );\n                },\n                function(error) {\n                    console.error('modified answer failed', error);\n                }\n            );\n        },\n        function(error) {\n            console.error('modify failed', error);\n        }\n    );\n};\n\n/**\n * Switches video streams.\n * @param new_stream new stream that will be used as video of this session.\n * @param oldStream old video stream of this session.\n * @param success_callback callback executed after successful stream switch.\n */\nJingleSession.prototype.switchStreams = function (new_stream, oldStream, success_callback) {\n\n    var self = this;\n\n    // Remember SDP to figure out added/removed SSRCs\n    var oldSdp = null;\n    if(self.peerconnection) {\n        if(self.peerconnection.localDescription) {\n            oldSdp = new SDP(self.peerconnection.localDescription.sdp);\n        }\n        self.peerconnection.removeStream(oldStream, true);\n        self.peerconnection.addStream(new_stream);\n    }\n\n    RTC.switchVideoStreams(new_stream, oldStream);\n\n    // Conference is not active\n    if(!oldSdp || !self.peerconnection) {\n        success_callback();\n        return;\n    }\n\n    self.switchstreams = true;\n    self.modifySources(function() {\n        console.log('modify sources done');\n\n        success_callback();\n\n        var newSdp = new SDP(self.peerconnection.localDescription.sdp);\n        console.log(\"SDPs\", oldSdp, newSdp);\n        self.notifyMySSRCUpdate(oldSdp, newSdp);\n    });\n};\n\n/**\n * Figures out added/removed ssrcs and send update IQs.\n * @param old_sdp SDP object for old description.\n * @param new_sdp SDP object for new description.\n */\nJingleSession.prototype.notifyMySSRCUpdate = function (old_sdp, new_sdp) {\n\n    if (!(this.peerconnection.signalingState == 'stable' &&\n        this.peerconnection.iceConnectionState == 'connected')){\n        console.log(\"Too early to send updates\");\n        return;\n    }\n\n    // send source-remove IQ.\n    sdpDiffer = new SDPDiffer(new_sdp, old_sdp);\n    var remove = $iq({to: this.peerjid, type: 'set'})\n        .c('jingle', {\n            xmlns: 'urn:xmpp:jingle:1',\n            action: 'source-remove',\n            initiator: this.initiator,\n            sid: this.sid\n        }\n    );\n    var removed = sdpDiffer.toJingle(remove);\n    if (removed) {\n        this.connection.sendIQ(remove,\n            function (res) {\n                console.info('got remove result', res);\n            },\n            function (err) {\n                console.error('got remove error', err);\n            }\n        );\n    } else {\n        console.log('removal not necessary');\n    }\n\n    // send source-add IQ.\n    var sdpDiffer = new SDPDiffer(old_sdp, new_sdp);\n    var add = $iq({to: this.peerjid, type: 'set'})\n        .c('jingle', {\n            xmlns: 'urn:xmpp:jingle:1',\n            action: 'source-add',\n            initiator: this.initiator,\n            sid: this.sid\n        }\n    );\n    var added = sdpDiffer.toJingle(add);\n    if (added) {\n        this.connection.sendIQ(add,\n            function (res) {\n                console.info('got add result', res);\n            },\n            function (err) {\n                console.error('got add error', err);\n            }\n        );\n    } else {\n        console.log('addition not necessary');\n    }\n};\n\n/**\n * Determines whether the (local) video is mute i.e. all video tracks are\n * disabled.\n *\n * @return <tt>true</tt> if the (local) video is mute i.e. all video tracks are\n * disabled; otherwise, <tt>false</tt>\n */\nJingleSession.prototype.isVideoMute = function () {\n    var tracks = RTC.localVideo.getVideoTracks();\n    var mute = true;\n\n    for (var i = 0; i < tracks.length; ++i) {\n        if (tracks[i].enabled) {\n            mute = false;\n            break;\n        }\n    }\n    return mute;\n};\n\n/**\n * Mutes/unmutes the (local) video i.e. enables/disables all video tracks.\n *\n * @param mute <tt>true</tt> to mute the (local) video i.e. to disable all video\n * tracks; otherwise, <tt>false</tt>\n * @param callback a function to be invoked with <tt>mute</tt> after all video\n * tracks have been enabled/disabled. The function may, optionally, return\n * another function which is to be invoked after the whole mute/unmute operation\n * has completed successfully.\n * @param options an object which specifies optional arguments such as the\n * <tt>boolean</tt> key <tt>byUser</tt> with default value <tt>true</tt> which\n * specifies whether the method was initiated in response to a user command (in\n * contrast to an automatic decision made by the application logic)\n */\nJingleSession.prototype.setVideoMute = function (mute, callback, options) {\n    var byUser;\n\n    if (options) {\n        byUser = options.byUser;\n        if (typeof byUser === 'undefined') {\n            byUser = true;\n        }\n    } else {\n        byUser = true;\n    }\n    // The user's command to mute the (local) video takes precedence over any\n    // automatic decision made by the application logic.\n    if (byUser) {\n        this.videoMuteByUser = mute;\n    } else if (this.videoMuteByUser) {\n        return;\n    }\n\n    var self = this;\n    var localCallback = function (mute) {\n        self.connection.emuc.addVideoInfoToPresence(mute);\n        self.connection.emuc.sendPresence();\n        return callback(mute)\n    };\n\n    if (mute == RTC.localVideo.isMuted())\n    {\n        // Even if no change occurs, the specified callback is to be executed.\n        // The specified callback may, optionally, return a successCallback\n        // which is to be executed as well.\n        var successCallback = localCallback(mute);\n\n        if (successCallback) {\n            successCallback();\n        }\n    } else {\n        RTC.localVideo.setMute(!mute);\n\n        this.hardMuteVideo(mute);\n\n        this.modifySources(localCallback(mute));\n    }\n};\n\n// SDP-based mute by going recvonly/sendrecv\n// FIXME: should probably black out the screen as well\nJingleSession.prototype.toggleVideoMute = function (callback) {\n    this.service.setVideoMute(RTC.localVideo.isMuted(), callback);\n};\n\nJingleSession.prototype.hardMuteVideo = function (muted) {\n    this.pendingop = muted ? 'mute' : 'unmute';\n};\n\nJingleSession.prototype.sendMute = function (muted, content) {\n    var info = $iq({to: this.peerjid,\n        type: 'set'})\n        .c('jingle', {xmlns: 'urn:xmpp:jingle:1',\n            action: 'session-info',\n            initiator: this.initiator,\n            sid: this.sid });\n    info.c(muted ? 'mute' : 'unmute', {xmlns: 'urn:xmpp:jingle:apps:rtp:info:1'});\n    info.attrs({'creator': this.me == this.initiator ? 'creator' : 'responder'});\n    if (content) {\n        info.attrs({'name': content});\n    }\n    this.connection.send(info);\n};\n\nJingleSession.prototype.sendRinging = function () {\n    var info = $iq({to: this.peerjid,\n        type: 'set'})\n        .c('jingle', {xmlns: 'urn:xmpp:jingle:1',\n            action: 'session-info',\n            initiator: this.initiator,\n            sid: this.sid });\n    info.c('ringing', {xmlns: 'urn:xmpp:jingle:apps:rtp:info:1'});\n    this.connection.send(info);\n};\n\nJingleSession.prototype.getStats = function (interval) {\n    var self = this;\n    var recv = {audio: 0, video: 0};\n    var lost = {audio: 0, video: 0};\n    var lastrecv = {audio: 0, video: 0};\n    var lastlost = {audio: 0, video: 0};\n    var loss = {audio: 0, video: 0};\n    var delta = {audio: 0, video: 0};\n    this.statsinterval = window.setInterval(function () {\n        if (self && self.peerconnection && self.peerconnection.getStats) {\n            self.peerconnection.getStats(function (stats) {\n                var results = stats.result();\n                // TODO: there are so much statistics you can get from this..\n                for (var i = 0; i < results.length; ++i) {\n                    if (results[i].type == 'ssrc') {\n                        var packetsrecv = results[i].stat('packetsReceived');\n                        var packetslost = results[i].stat('packetsLost');\n                        if (packetsrecv && packetslost) {\n                            packetsrecv = parseInt(packetsrecv, 10);\n                            packetslost = parseInt(packetslost, 10);\n\n                            if (results[i].stat('googFrameRateReceived')) {\n                                lastlost.video = lost.video;\n                                lastrecv.video = recv.video;\n                                recv.video = packetsrecv;\n                                lost.video = packetslost;\n                            } else {\n                                lastlost.audio = lost.audio;\n                                lastrecv.audio = recv.audio;\n                                recv.audio = packetsrecv;\n                                lost.audio = packetslost;\n                            }\n                        }\n                    }\n                }\n                delta.audio = recv.audio - lastrecv.audio;\n                delta.video = recv.video - lastrecv.video;\n                loss.audio = (delta.audio > 0) ? Math.ceil(100 * (lost.audio - lastlost.audio) / delta.audio) : 0;\n                loss.video = (delta.video > 0) ? Math.ceil(100 * (lost.video - lastlost.video) / delta.video) : 0;\n                $(document).trigger('packetloss.jingle', [self.sid, loss]);\n            });\n        }\n    }, interval || 3000);\n    return this.statsinterval;\n};\n\nJingleSession.onJingleError = function (session, error)\n{\n    console.error(\"Jingle error\", error);\n}\n\nJingleSession.onJingleFatalError = function (session, error)\n{\n    this.service.sessionTerminated = true;\n    connection.emuc.doLeave();\n    UI.messageHandler.showError(  \"Sorry\",\n        \"Internal application error[setRemoteDescription]\");\n}\n\nJingleSession.prototype.setLocalDescription = function () {\n    // put our ssrcs into presence so other clients can identify our stream\n    var newssrcs = [];\n    var media = simulcast.parseMedia(this.peerconnection.localDescription);\n    media.forEach(function (media) {\n\n        if(Object.keys(media.sources).length > 0) {\n            // TODO(gp) maybe exclude FID streams?\n            Object.keys(media.sources).forEach(function (ssrc) {\n                newssrcs.push({\n                    'ssrc': ssrc,\n                    'type': media.type,\n                    'direction': media.direction\n                });\n            });\n        }\n        else if(this.localStreamsSSRC && this.localStreamsSSRC[media.type])\n        {\n            newssrcs.push({\n                'ssrc': this.localStreamsSSRC[media.type],\n                'type': media.type,\n                'direction': media.direction\n            });\n        }\n\n    });\n\n    console.log('new ssrcs', newssrcs);\n\n    // Have to clear presence map to get rid of removed streams\n    this.connection.emuc.clearPresenceMedia();\n\n    if (newssrcs.length > 0) {\n        for (var i = 1; i <= newssrcs.length; i ++) {\n            // Change video type to screen\n            if (newssrcs[i-1].type === 'video' && desktopsharing.isUsingScreenStream()) {\n                newssrcs[i-1].type = 'screen';\n            }\n            this.connection.emuc.addMediaToPresence(i,\n                newssrcs[i-1].type, newssrcs[i-1].ssrc, newssrcs[i-1].direction);\n        }\n\n        this.connection.emuc.sendPresence();\n    }\n}\n\n// an attempt to work around https://github.com/jitsi/jitmeet/issues/32\nfunction sendKeyframe(pc) {\n    console.log('sendkeyframe', pc.iceConnectionState);\n    if (pc.iceConnectionState !== 'connected') return; // safe...\n    pc.setRemoteDescription(\n        pc.remoteDescription,\n        function () {\n            pc.createAnswer(\n                function (modifiedAnswer) {\n                    pc.setLocalDescription(\n                        modifiedAnswer,\n                        function () {\n                            // noop\n                        },\n                        function (error) {\n                            console.log('triggerKeyframe setLocalDescription failed', error);\n                            UI.messageHandler.showError();\n                        }\n                    );\n                },\n                function (error) {\n                    console.log('triggerKeyframe createAnswer failed', error);\n                    UI.messageHandler.showError();\n                }\n            );\n        },\n        function (error) {\n            console.log('triggerKeyframe setRemoteDescription failed', error);\n            UI.messageHandler.showError();\n        }\n    );\n}\n\n\nJingleSession.prototype.remoteStreamAdded = function (data) {\n    var self = this;\n    var thessrc;\n\n    // look up an associated JID for a stream id\n    if (data.stream.id && data.stream.id.indexOf('mixedmslabel') === -1) {\n        // look only at a=ssrc: and _not_ at a=ssrc-group: lines\n\n        var ssrclines\n            = SDPUtil.find_lines(this.peerconnection.remoteDescription.sdp, 'a=ssrc:');\n        ssrclines = ssrclines.filter(function (line) {\n            // NOTE(gp) previously we filtered on the mslabel, but that property\n            // is not always present.\n            // return line.indexOf('mslabel:' + data.stream.label) !== -1;\n\n            return ((line.indexOf('msid:' + data.stream.id) !== -1));\n        });\n        if (ssrclines.length) {\n            thessrc = ssrclines[0].substring(7).split(' ')[0];\n\n            // We signal our streams (through Jingle to the focus) before we set\n            // our presence (through which peers associate remote streams to\n            // jids). So, it might arrive that a remote stream is added but\n            // ssrc2jid is not yet updated and thus data.peerjid cannot be\n            // successfully set. Here we wait for up to a second for the\n            // presence to arrive.\n\n            if (!ssrc2jid[thessrc]) {\n                // TODO(gp) limit wait duration to 1 sec.\n                setTimeout(function(d) {\n                    return function() {\n                        self.remoteStreamAdded(d);\n                    }\n                }(data), 250);\n                return;\n            }\n\n            // ok to overwrite the one from focus? might save work in colibri.js\n            console.log('associated jid', ssrc2jid[thessrc], data.peerjid);\n            if (ssrc2jid[thessrc]) {\n                data.peerjid = ssrc2jid[thessrc];\n            }\n        }\n    }\n\n    //TODO: this code should be removed when firefox implement multistream support\n    if(RTC.getBrowserType() == RTCBrowserType.RTC_BROWSER_FIREFOX)\n    {\n        if((notReceivedSSRCs.length == 0) ||\n            !ssrc2jid[notReceivedSSRCs[notReceivedSSRCs.length - 1]])\n        {\n            // TODO(gp) limit wait duration to 1 sec.\n            setTimeout(function(d) {\n                return function() {\n                    self.remoteStreamAdded(d);\n                }\n            }(data), 250);\n            return;\n        }\n\n        thessrc = notReceivedSSRCs.pop();\n        if (ssrc2jid[thessrc]) {\n            data.peerjid = ssrc2jid[thessrc];\n        }\n    }\n\n    RTC.createRemoteStream(data, this.sid, thessrc);\n\n    var isVideo = data.stream.getVideoTracks().length > 0;\n    // an attempt to work around https://github.com/jitsi/jitmeet/issues/32\n    if (isVideo &&\n        data.peerjid && this.peerjid === data.peerjid &&\n        data.stream.getVideoTracks().length === 0 &&\n        RTC.localVideo.getTracks().length > 0) {\n        window.setTimeout(function () {\n            sendKeyframe(self.peerconnection);\n        }, 3000);\n    }\n}\n\nmodule.exports = JingleSession;","/* jshint -W117 */\nvar SDPUtil = require(\"./SDPUtil\");\n\n// SDP STUFF\nfunction SDP(sdp) {\n    this.media = sdp.split('\\r\\nm=');\n    for (var i = 1; i < this.media.length; i++) {\n        this.media[i] = 'm=' + this.media[i];\n        if (i != this.media.length - 1) {\n            this.media[i] += '\\r\\n';\n        }\n    }\n    this.session = this.media.shift() + '\\r\\n';\n    this.raw = this.session + this.media.join('');\n}\n/**\n * Returns map of MediaChannel mapped per channel idx.\n */\nSDP.prototype.getMediaSsrcMap = function() {\n    var self = this;\n    var media_ssrcs = {};\n    var tmp;\n    for (var mediaindex = 0; mediaindex < self.media.length; mediaindex++) {\n        tmp = SDPUtil.find_lines(self.media[mediaindex], 'a=ssrc:');\n        var mid = SDPUtil.parse_mid(SDPUtil.find_line(self.media[mediaindex], 'a=mid:'));\n        var media = {\n            mediaindex: mediaindex,\n            mid: mid,\n            ssrcs: {},\n            ssrcGroups: []\n        };\n        media_ssrcs[mediaindex] = media;\n        tmp.forEach(function (line) {\n            var linessrc = line.substring(7).split(' ')[0];\n            // allocate new ChannelSsrc\n            if(!media.ssrcs[linessrc]) {\n                media.ssrcs[linessrc] = {\n                    ssrc: linessrc,\n                    lines: []\n                };\n            }\n            media.ssrcs[linessrc].lines.push(line);\n        });\n        tmp = SDPUtil.find_lines(self.media[mediaindex], 'a=ssrc-group:');\n        tmp.forEach(function(line){\n            var semantics = line.substr(0, idx).substr(13);\n            var ssrcs = line.substr(14 + semantics.length).split(' ');\n            if (ssrcs.length != 0) {\n                media.ssrcGroups.push({\n                    semantics: semantics,\n                    ssrcs: ssrcs\n                });\n            }\n        });\n    }\n    return media_ssrcs;\n};\n/**\n * Returns <tt>true</tt> if this SDP contains given SSRC.\n * @param ssrc the ssrc to check.\n * @returns {boolean} <tt>true</tt> if this SDP contains given SSRC.\n */\nSDP.prototype.containsSSRC = function(ssrc) {\n    var medias = this.getMediaSsrcMap();\n    var contains = false;\n    Object.keys(medias).forEach(function(mediaindex){\n        var media = medias[mediaindex];\n        //console.log(\"Check\", channel, ssrc);\n        if(Object.keys(media.ssrcs).indexOf(ssrc) != -1){\n            contains = true;\n        }\n    });\n    return contains;\n};\n\n\n// remove iSAC and CN from SDP\nSDP.prototype.mangle = function () {\n    var i, j, mline, lines, rtpmap, newdesc;\n    for (i = 0; i < this.media.length; i++) {\n        lines = this.media[i].split('\\r\\n');\n        lines.pop(); // remove empty last element\n        mline = SDPUtil.parse_mline(lines.shift());\n        if (mline.media != 'audio')\n            continue;\n        newdesc = '';\n        mline.fmt.length = 0;\n        for (j = 0; j < lines.length; j++) {\n            if (lines[j].substr(0, 9) == 'a=rtpmap:') {\n                rtpmap = SDPUtil.parse_rtpmap(lines[j]);\n                if (rtpmap.name == 'CN' || rtpmap.name == 'ISAC')\n                    continue;\n                mline.fmt.push(rtpmap.id);\n                newdesc += lines[j] + '\\r\\n';\n            } else {\n                newdesc += lines[j] + '\\r\\n';\n            }\n        }\n        this.media[i] = SDPUtil.build_mline(mline) + '\\r\\n';\n        this.media[i] += newdesc;\n    }\n    this.raw = this.session + this.media.join('');\n};\n\n// remove lines matching prefix from session section\nSDP.prototype.removeSessionLines = function(prefix) {\n    var self = this;\n    var lines = SDPUtil.find_lines(this.session, prefix);\n    lines.forEach(function(line) {\n        self.session = self.session.replace(line + '\\r\\n', '');\n    });\n    this.raw = this.session + this.media.join('');\n    return lines;\n}\n// remove lines matching prefix from a media section specified by mediaindex\n// TODO: non-numeric mediaindex could match mid\nSDP.prototype.removeMediaLines = function(mediaindex, prefix) {\n    var self = this;\n    var lines = SDPUtil.find_lines(this.media[mediaindex], prefix);\n    lines.forEach(function(line) {\n        self.media[mediaindex] = self.media[mediaindex].replace(line + '\\r\\n', '');\n    });\n    this.raw = this.session + this.media.join('');\n    return lines;\n}\n\n// add content's to a jingle element\nSDP.prototype.toJingle = function (elem, thecreator, ssrcs) {\n//    console.log(\"SSRC\" + ssrcs[\"audio\"] + \" - \" + ssrcs[\"video\"]);\n    var i, j, k, mline, ssrc, rtpmap, tmp, line, lines;\n    var self = this;\n    // new bundle plan\n    if (SDPUtil.find_line(this.session, 'a=group:')) {\n        lines = SDPUtil.find_lines(this.session, 'a=group:');\n        for (i = 0; i < lines.length; i++) {\n            tmp = lines[i].split(' ');\n            var semantics = tmp.shift().substr(8);\n            elem.c('group', {xmlns: 'urn:xmpp:jingle:apps:grouping:0', semantics:semantics});\n            for (j = 0; j < tmp.length; j++) {\n                elem.c('content', {name: tmp[j]}).up();\n            }\n            elem.up();\n        }\n    }\n    for (i = 0; i < this.media.length; i++) {\n        mline = SDPUtil.parse_mline(this.media[i].split('\\r\\n')[0]);\n        if (!(mline.media === 'audio' ||\n              mline.media === 'video' ||\n              mline.media === 'application'))\n        {\n            continue;\n        }\n        if (SDPUtil.find_line(this.media[i], 'a=ssrc:')) {\n            ssrc = SDPUtil.find_line(this.media[i], 'a=ssrc:').substring(7).split(' ')[0]; // take the first\n        } else {\n            if(ssrcs && ssrcs[mline.media])\n            {\n                ssrc = ssrcs[mline.media];\n            }\n            else\n                ssrc = false;\n        }\n\n        elem.c('content', {creator: thecreator, name: mline.media});\n        if (SDPUtil.find_line(this.media[i], 'a=mid:')) {\n            // prefer identifier from a=mid if present\n            var mid = SDPUtil.parse_mid(SDPUtil.find_line(this.media[i], 'a=mid:'));\n            elem.attrs({ name: mid });\n        }\n\n        if (SDPUtil.find_line(this.media[i], 'a=rtpmap:').length)\n        {\n            elem.c('description',\n                {xmlns: 'urn:xmpp:jingle:apps:rtp:1',\n                    media: mline.media });\n            if (ssrc) {\n                elem.attrs({ssrc: ssrc});\n            }\n            for (j = 0; j < mline.fmt.length; j++) {\n                rtpmap = SDPUtil.find_line(this.media[i], 'a=rtpmap:' + mline.fmt[j]);\n                elem.c('payload-type', SDPUtil.parse_rtpmap(rtpmap));\n                // put any 'a=fmtp:' + mline.fmt[j] lines into <param name=foo value=bar/>\n                if (SDPUtil.find_line(this.media[i], 'a=fmtp:' + mline.fmt[j])) {\n                    tmp = SDPUtil.parse_fmtp(SDPUtil.find_line(this.media[i], 'a=fmtp:' + mline.fmt[j]));\n                    for (k = 0; k < tmp.length; k++) {\n                        elem.c('parameter', tmp[k]).up();\n                    }\n                }\n                this.RtcpFbToJingle(i, elem, mline.fmt[j]); // XEP-0293 -- map a=rtcp-fb\n\n                elem.up();\n            }\n            if (SDPUtil.find_line(this.media[i], 'a=crypto:', this.session)) {\n                elem.c('encryption', {required: 1});\n                var crypto = SDPUtil.find_lines(this.media[i], 'a=crypto:', this.session);\n                crypto.forEach(function(line) {\n                    elem.c('crypto', SDPUtil.parse_crypto(line)).up();\n                });\n                elem.up(); // end of encryption\n            }\n\n            if (ssrc) {\n                // new style mapping\n                elem.c('source', { ssrc: ssrc, xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });\n                // FIXME: group by ssrc and support multiple different ssrcs\n                var ssrclines = SDPUtil.find_lines(this.media[i], 'a=ssrc:');\n                if(ssrclines.length > 0) {\n                    ssrclines.forEach(function (line) {\n                        idx = line.indexOf(' ');\n                        var linessrc = line.substr(0, idx).substr(7);\n                        if (linessrc != ssrc) {\n                            elem.up();\n                            ssrc = linessrc;\n                            elem.c('source', { ssrc: ssrc, xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });\n                        }\n                        var kv = line.substr(idx + 1);\n                        elem.c('parameter');\n                        if (kv.indexOf(':') == -1) {\n                            elem.attrs({ name: kv });\n                        } else {\n                            elem.attrs({ name: kv.split(':', 2)[0] });\n                            elem.attrs({ value: kv.split(':', 2)[1] });\n                        }\n                        elem.up();\n                    });\n                    elem.up();\n                }\n                else\n                {\n                    elem.up();\n                    elem.c('source', { ssrc: ssrc, xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });\n                    elem.c('parameter');\n                    elem.attrs({name: \"cname\", value:Math.random().toString(36).substring(7)});\n                    elem.up();\n                    var msid = null;\n                    if(mline.media == \"audio\")\n                    {\n                        msid = RTC.localAudio.getId();\n                    }\n                    else\n                    {\n                        msid = RTC.localVideo.getId();\n                    }\n                    if(msid != null)\n                    {\n                        msid = msid.replace(/[\\{,\\}]/g,\"\");\n                        elem.c('parameter');\n                        elem.attrs({name: \"msid\", value:msid});\n                        elem.up();\n                        elem.c('parameter');\n                        elem.attrs({name: \"mslabel\", value:msid});\n                        elem.up();\n                        elem.c('parameter');\n                        elem.attrs({name: \"label\", value:msid});\n                        elem.up();\n                        elem.up();\n                    }\n\n\n                }\n\n                // XEP-0339 handle ssrc-group attributes\n                var ssrc_group_lines = SDPUtil.find_lines(this.media[i], 'a=ssrc-group:');\n                ssrc_group_lines.forEach(function(line) {\n                    idx = line.indexOf(' ');\n                    var semantics = line.substr(0, idx).substr(13);\n                    var ssrcs = line.substr(14 + semantics.length).split(' ');\n                    if (ssrcs.length != 0) {\n                        elem.c('ssrc-group', { semantics: semantics, xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });\n                        ssrcs.forEach(function(ssrc) {\n                            elem.c('source', { ssrc: ssrc })\n                                .up();\n                        });\n                        elem.up();\n                    }\n                });\n            }\n\n            if (SDPUtil.find_line(this.media[i], 'a=rtcp-mux')) {\n                elem.c('rtcp-mux').up();\n            }\n\n            // XEP-0293 -- map a=rtcp-fb:*\n            this.RtcpFbToJingle(i, elem, '*');\n\n            // XEP-0294\n            if (SDPUtil.find_line(this.media[i], 'a=extmap:')) {\n                lines = SDPUtil.find_lines(this.media[i], 'a=extmap:');\n                for (j = 0; j < lines.length; j++) {\n                    tmp = SDPUtil.parse_extmap(lines[j]);\n                    elem.c('rtp-hdrext', { xmlns: 'urn:xmpp:jingle:apps:rtp:rtp-hdrext:0',\n                        uri: tmp.uri,\n                        id: tmp.value });\n                    if (tmp.hasOwnProperty('direction')) {\n                        switch (tmp.direction) {\n                            case 'sendonly':\n                                elem.attrs({senders: 'responder'});\n                                break;\n                            case 'recvonly':\n                                elem.attrs({senders: 'initiator'});\n                                break;\n                            case 'sendrecv':\n                                elem.attrs({senders: 'both'});\n                                break;\n                            case 'inactive':\n                                elem.attrs({senders: 'none'});\n                                break;\n                        }\n                    }\n                    // TODO: handle params\n                    elem.up();\n                }\n            }\n            elem.up(); // end of description\n        }\n\n        // map ice-ufrag/pwd, dtls fingerprint, candidates\n        this.TransportToJingle(i, elem);\n\n        if (SDPUtil.find_line(this.media[i], 'a=sendrecv', this.session)) {\n            elem.attrs({senders: 'both'});\n        } else if (SDPUtil.find_line(this.media[i], 'a=sendonly', this.session)) {\n            elem.attrs({senders: 'initiator'});\n        } else if (SDPUtil.find_line(this.media[i], 'a=recvonly', this.session)) {\n            elem.attrs({senders: 'responder'});\n        } else if (SDPUtil.find_line(this.media[i], 'a=inactive', this.session)) {\n            elem.attrs({senders: 'none'});\n        }\n        if (mline.port == '0') {\n            // estos hack to reject an m-line\n            elem.attrs({senders: 'rejected'});\n        }\n        elem.up(); // end of content\n    }\n    elem.up();\n    return elem;\n};\n\nSDP.prototype.TransportToJingle = function (mediaindex, elem) {\n    var i = mediaindex;\n    var tmp;\n    var self = this;\n    elem.c('transport');\n\n    // XEP-0343 DTLS/SCTP\n    if (SDPUtil.find_line(this.media[mediaindex], 'a=sctpmap:').length)\n    {\n        var sctpmap = SDPUtil.find_line(\n            this.media[i], 'a=sctpmap:', self.session);\n        if (sctpmap)\n        {\n            var sctpAttrs = SDPUtil.parse_sctpmap(sctpmap);\n            elem.c('sctpmap',\n                {\n                    xmlns: 'urn:xmpp:jingle:transports:dtls-sctp:1',\n                    number: sctpAttrs[0], /* SCTP port */\n                    protocol: sctpAttrs[1], /* protocol */\n                });\n            // Optional stream count attribute\n            if (sctpAttrs.length > 2)\n                elem.attrs({ streams: sctpAttrs[2]});\n            elem.up();\n        }\n    }\n    // XEP-0320\n    var fingerprints = SDPUtil.find_lines(this.media[mediaindex], 'a=fingerprint:', this.session);\n    fingerprints.forEach(function(line) {\n        tmp = SDPUtil.parse_fingerprint(line);\n        tmp.xmlns = 'urn:xmpp:jingle:apps:dtls:0';\n        elem.c('fingerprint').t(tmp.fingerprint);\n        delete tmp.fingerprint;\n        line = SDPUtil.find_line(self.media[mediaindex], 'a=setup:', self.session);\n        if (line) {\n            tmp.setup = line.substr(8);\n        }\n        elem.attrs(tmp);\n        elem.up(); // end of fingerprint\n    });\n    tmp = SDPUtil.iceparams(this.media[mediaindex], this.session);\n    if (tmp) {\n        tmp.xmlns = 'urn:xmpp:jingle:transports:ice-udp:1';\n        elem.attrs(tmp);\n        // XEP-0176\n        if (SDPUtil.find_line(this.media[mediaindex], 'a=candidate:', this.session)) { // add any a=candidate lines\n            var lines = SDPUtil.find_lines(this.media[mediaindex], 'a=candidate:', this.session);\n            lines.forEach(function (line) {\n                elem.c('candidate', SDPUtil.candidateToJingle(line)).up();\n            });\n        }\n    }\n    elem.up(); // end of transport\n}\n\nSDP.prototype.RtcpFbToJingle = function (mediaindex, elem, payloadtype) { // XEP-0293\n    var lines = SDPUtil.find_lines(this.media[mediaindex], 'a=rtcp-fb:' + payloadtype);\n    lines.forEach(function (line) {\n        var tmp = SDPUtil.parse_rtcpfb(line);\n        if (tmp.type == 'trr-int') {\n            elem.c('rtcp-fb-trr-int', {xmlns: 'urn:xmpp:jingle:apps:rtp:rtcp-fb:0', value: tmp.params[0]});\n            elem.up();\n        } else {\n            elem.c('rtcp-fb', {xmlns: 'urn:xmpp:jingle:apps:rtp:rtcp-fb:0', type: tmp.type});\n            if (tmp.params.length > 0) {\n                elem.attrs({'subtype': tmp.params[0]});\n            }\n            elem.up();\n        }\n    });\n};\n\nSDP.prototype.RtcpFbFromJingle = function (elem, payloadtype) { // XEP-0293\n    var media = '';\n    var tmp = elem.find('>rtcp-fb-trr-int[xmlns=\"urn:xmpp:jingle:apps:rtp:rtcp-fb:0\"]');\n    if (tmp.length) {\n        media += 'a=rtcp-fb:' + '*' + ' ' + 'trr-int' + ' ';\n        if (tmp.attr('value')) {\n            media += tmp.attr('value');\n        } else {\n            media += '0';\n        }\n        media += '\\r\\n';\n    }\n    tmp = elem.find('>rtcp-fb[xmlns=\"urn:xmpp:jingle:apps:rtp:rtcp-fb:0\"]');\n    tmp.each(function () {\n        media += 'a=rtcp-fb:' + payloadtype + ' ' + $(this).attr('type');\n        if ($(this).attr('subtype')) {\n            media += ' ' + $(this).attr('subtype');\n        }\n        media += '\\r\\n';\n    });\n    return media;\n};\n\n// construct an SDP from a jingle stanza\nSDP.prototype.fromJingle = function (jingle) {\n    var self = this;\n    this.raw = 'v=0\\r\\n' +\n        'o=- ' + '1923518516' + ' 2 IN IP4 0.0.0.0\\r\\n' +// FIXME\n        's=-\\r\\n' +\n        't=0 0\\r\\n';\n    // http://tools.ietf.org/html/draft-ietf-mmusic-sdp-bundle-negotiation-04#section-8\n    if ($(jingle).find('>group[xmlns=\"urn:xmpp:jingle:apps:grouping:0\"]').length) {\n        $(jingle).find('>group[xmlns=\"urn:xmpp:jingle:apps:grouping:0\"]').each(function (idx, group) {\n            var contents = $(group).find('>content').map(function (idx, content) {\n                return content.getAttribute('name');\n            }).get();\n            if (contents.length > 0) {\n                self.raw += 'a=group:' + (group.getAttribute('semantics') || group.getAttribute('type')) + ' ' + contents.join(' ') + '\\r\\n';\n            }\n        });\n    }\n\n    this.session = this.raw;\n    jingle.find('>content').each(function () {\n        var m = self.jingle2media($(this));\n        self.media.push(m);\n    });\n\n    // reconstruct msid-semantic -- apparently not necessary\n    /*\n     var msid = SDPUtil.parse_ssrc(this.raw);\n     if (msid.hasOwnProperty('mslabel')) {\n     this.session += \"a=msid-semantic: WMS \" + msid.mslabel + \"\\r\\n\";\n     }\n     */\n\n    this.raw = this.session + this.media.join('');\n};\n\n// translate a jingle content element into an an SDP media part\nSDP.prototype.jingle2media = function (content) {\n    var media = '',\n        desc = content.find('description'),\n        ssrc = desc.attr('ssrc'),\n        self = this,\n        tmp;\n    var sctp = content.find(\n        '>transport>sctpmap[xmlns=\"urn:xmpp:jingle:transports:dtls-sctp:1\"]');\n\n    tmp = { media: desc.attr('media') };\n    tmp.port = '1';\n    if (content.attr('senders') == 'rejected') {\n        // estos hack to reject an m-line.\n        tmp.port = '0';\n    }\n    if (content.find('>transport>fingerprint').length || desc.find('encryption').length) {\n        if (sctp.length)\n            tmp.proto = 'DTLS/SCTP';\n        else\n            tmp.proto = 'RTP/SAVPF';\n    } else {\n        tmp.proto = 'RTP/AVPF';\n    }\n    if (!sctp.length)\n    {\n        tmp.fmt = desc.find('payload-type').map(\n            function () { return this.getAttribute('id'); }).get();\n        media += SDPUtil.build_mline(tmp) + '\\r\\n';\n    }\n    else\n    {\n        media += 'm=application 1 DTLS/SCTP ' + sctp.attr('number') + '\\r\\n';\n        media += 'a=sctpmap:' + sctp.attr('number') +\n            ' ' + sctp.attr('protocol');\n\n        var streamCount = sctp.attr('streams');\n        if (streamCount)\n            media += ' ' + streamCount + '\\r\\n';\n        else\n            media += '\\r\\n';\n    }\n\n    media += 'c=IN IP4 0.0.0.0\\r\\n';\n    if (!sctp.length)\n        media += 'a=rtcp:1 IN IP4 0.0.0.0\\r\\n';\n    tmp = content.find('>transport[xmlns=\"urn:xmpp:jingle:transports:ice-udp:1\"]');\n    if (tmp.length) {\n        if (tmp.attr('ufrag')) {\n            media += SDPUtil.build_iceufrag(tmp.attr('ufrag')) + '\\r\\n';\n        }\n        if (tmp.attr('pwd')) {\n            media += SDPUtil.build_icepwd(tmp.attr('pwd')) + '\\r\\n';\n        }\n        tmp.find('>fingerprint').each(function () {\n            // FIXME: check namespace at some point\n            media += 'a=fingerprint:' + this.getAttribute('hash');\n            media += ' ' + $(this).text();\n            media += '\\r\\n';\n            if (this.getAttribute('setup')) {\n                media += 'a=setup:' + this.getAttribute('setup') + '\\r\\n';\n            }\n        });\n    }\n    switch (content.attr('senders')) {\n        case 'initiator':\n            media += 'a=sendonly\\r\\n';\n            break;\n        case 'responder':\n            media += 'a=recvonly\\r\\n';\n            break;\n        case 'none':\n            media += 'a=inactive\\r\\n';\n            break;\n        case 'both':\n            media += 'a=sendrecv\\r\\n';\n            break;\n    }\n    media += 'a=mid:' + content.attr('name') + '\\r\\n';\n\n    // <description><rtcp-mux/></description>\n    // see http://code.google.com/p/libjingle/issues/detail?id=309 -- no spec though\n    // and http://mail.jabber.org/pipermail/jingle/2011-December/001761.html\n    if (desc.find('rtcp-mux').length) {\n        media += 'a=rtcp-mux\\r\\n';\n    }\n\n    if (desc.find('encryption').length) {\n        desc.find('encryption>crypto').each(function () {\n            media += 'a=crypto:' + this.getAttribute('tag');\n            media += ' ' + this.getAttribute('crypto-suite');\n            media += ' ' + this.getAttribute('key-params');\n            if (this.getAttribute('session-params')) {\n                media += ' ' + this.getAttribute('session-params');\n            }\n            media += '\\r\\n';\n        });\n    }\n    desc.find('payload-type').each(function () {\n        media += SDPUtil.build_rtpmap(this) + '\\r\\n';\n        if ($(this).find('>parameter').length) {\n            media += 'a=fmtp:' + this.getAttribute('id') + ' ';\n            media += $(this).find('parameter').map(function () { return (this.getAttribute('name') ? (this.getAttribute('name') + '=') : '') + this.getAttribute('value'); }).get().join('; ');\n            media += '\\r\\n';\n        }\n        // xep-0293\n        media += self.RtcpFbFromJingle($(this), this.getAttribute('id'));\n    });\n\n    // xep-0293\n    media += self.RtcpFbFromJingle(desc, '*');\n\n    // xep-0294\n    tmp = desc.find('>rtp-hdrext[xmlns=\"urn:xmpp:jingle:apps:rtp:rtp-hdrext:0\"]');\n    tmp.each(function () {\n        media += 'a=extmap:' + this.getAttribute('id') + ' ' + this.getAttribute('uri') + '\\r\\n';\n    });\n\n    content.find('>transport[xmlns=\"urn:xmpp:jingle:transports:ice-udp:1\"]>candidate').each(function () {\n        media += SDPUtil.candidateFromJingle(this);\n    });\n\n    // XEP-0339 handle ssrc-group attributes\n    tmp = content.find('description>ssrc-group[xmlns=\"urn:xmpp:jingle:apps:rtp:ssma:0\"]').each(function() {\n        var semantics = this.getAttribute('semantics');\n        var ssrcs = $(this).find('>source').map(function() {\n            return this.getAttribute('ssrc');\n        }).get();\n\n        if (ssrcs.length != 0) {\n            media += 'a=ssrc-group:' + semantics + ' ' + ssrcs.join(' ') + '\\r\\n';\n        }\n    });\n\n    tmp = content.find('description>source[xmlns=\"urn:xmpp:jingle:apps:rtp:ssma:0\"]');\n    tmp.each(function () {\n        var ssrc = this.getAttribute('ssrc');\n        $(this).find('>parameter').each(function () {\n            media += 'a=ssrc:' + ssrc + ' ' + this.getAttribute('name');\n            if (this.getAttribute('value') && this.getAttribute('value').length)\n                media += ':' + this.getAttribute('value');\n            media += '\\r\\n';\n        });\n    });\n\n    return media;\n};\n\n\nmodule.exports = SDP;\n\n","function SDPDiffer(mySDP, otherSDP) {\n    this.mySDP = mySDP;\n    this.otherSDP = otherSDP;\n}\n\n/**\n * Returns map of MediaChannel that contains only media not contained in <tt>otherSdp</tt>. Mapped by channel idx.\n * @param otherSdp the other SDP to check ssrc with.\n */\nSDPDiffer.prototype.getNewMedia = function() {\n\n    // this could be useful in Array.prototype.\n    function arrayEquals(array) {\n        // if the other array is a falsy value, return\n        if (!array)\n            return false;\n\n        // compare lengths - can save a lot of time\n        if (this.length != array.length)\n            return false;\n\n        for (var i = 0, l=this.length; i < l; i++) {\n            // Check if we have nested arrays\n            if (this[i] instanceof Array && array[i] instanceof Array) {\n                // recurse into the nested arrays\n                if (!this[i].equals(array[i]))\n                    return false;\n            }\n            else if (this[i] != array[i]) {\n                // Warning - two different object instances will never be equal: {x:20} != {x:20}\n                return false;\n            }\n        }\n        return true;\n    }\n\n    var myMedias = this.mySDP.getMediaSsrcMap();\n    var othersMedias = this.otherSDP.getMediaSsrcMap();\n    var newMedia = {};\n    Object.keys(othersMedias).forEach(function(othersMediaIdx) {\n        var myMedia = myMedias[othersMediaIdx];\n        var othersMedia = othersMedias[othersMediaIdx];\n        if(!myMedia && othersMedia) {\n            // Add whole channel\n            newMedia[othersMediaIdx] = othersMedia;\n            return;\n        }\n        // Look for new ssrcs accross the channel\n        Object.keys(othersMedia.ssrcs).forEach(function(ssrc) {\n            if(Object.keys(myMedia.ssrcs).indexOf(ssrc) === -1) {\n                // Allocate channel if we've found ssrc that doesn't exist in our channel\n                if(!newMedia[othersMediaIdx]){\n                    newMedia[othersMediaIdx] = {\n                        mediaindex: othersMedia.mediaindex,\n                        mid: othersMedia.mid,\n                        ssrcs: {},\n                        ssrcGroups: []\n                    };\n                }\n                newMedia[othersMediaIdx].ssrcs[ssrc] = othersMedia.ssrcs[ssrc];\n            }\n        });\n\n        // Look for new ssrc groups across the channels\n        othersMedia.ssrcGroups.forEach(function(otherSsrcGroup){\n\n            // try to match the other ssrc-group with an ssrc-group of ours\n            var matched = false;\n            for (var i = 0; i < myMedia.ssrcGroups.length; i++) {\n                var mySsrcGroup = myMedia.ssrcGroups[i];\n                if (otherSsrcGroup.semantics == mySsrcGroup.semantics\n                    && arrayEquals.apply(otherSsrcGroup.ssrcs, [mySsrcGroup.ssrcs])) {\n\n                    matched = true;\n                    break;\n                }\n            }\n\n            if (!matched) {\n                // Allocate channel if we've found an ssrc-group that doesn't\n                // exist in our channel\n\n                if(!newMedia[othersMediaIdx]){\n                    newMedia[othersMediaIdx] = {\n                        mediaindex: othersMedia.mediaindex,\n                        mid: othersMedia.mid,\n                        ssrcs: {},\n                        ssrcGroups: []\n                    };\n                }\n                newMedia[othersMediaIdx].ssrcGroups.push(otherSsrcGroup);\n            }\n        });\n    });\n    return newMedia;\n};\n\n/**\n * Sends SSRC update IQ.\n * @param sdpMediaSsrcs SSRCs map obtained from SDP.getNewMedia. Cntains SSRCs to add/remove.\n * @param sid session identifier that will be put into the IQ.\n * @param initiator initiator identifier.\n * @param toJid destination Jid\n * @param isAdd indicates if this is remove or add operation.\n */\nSDPDiffer.prototype.toJingle = function(modify) {\n    var sdpMediaSsrcs = this.getNewMedia();\n    var self = this;\n\n    // FIXME: only announce video ssrcs since we mix audio and dont need\n    //      the audio ssrcs therefore\n    var modified = false;\n    Object.keys(sdpMediaSsrcs).forEach(function(mediaindex){\n        modified = true;\n        var media = sdpMediaSsrcs[mediaindex];\n        modify.c('content', {name: media.mid});\n\n        modify.c('description', {xmlns:'urn:xmpp:jingle:apps:rtp:1', media: media.mid});\n        // FIXME: not completly sure this operates on blocks and / or handles different ssrcs correctly\n        // generate sources from lines\n        Object.keys(media.ssrcs).forEach(function(ssrcNum) {\n            var mediaSsrc = media.ssrcs[ssrcNum];\n            modify.c('source', { xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });\n            modify.attrs({ssrc: mediaSsrc.ssrc});\n            // iterate over ssrc lines\n            mediaSsrc.lines.forEach(function (line) {\n                var idx = line.indexOf(' ');\n                var kv = line.substr(idx + 1);\n                modify.c('parameter');\n                if (kv.indexOf(':') == -1) {\n                    modify.attrs({ name: kv });\n                } else {\n                    modify.attrs({ name: kv.split(':', 2)[0] });\n                    modify.attrs({ value: kv.split(':', 2)[1] });\n                }\n                modify.up(); // end of parameter\n            });\n            modify.up(); // end of source\n        });\n\n        // generate source groups from lines\n        media.ssrcGroups.forEach(function(ssrcGroup) {\n            if (ssrcGroup.ssrcs.length != 0) {\n\n                modify.c('ssrc-group', {\n                    semantics: ssrcGroup.semantics,\n                    xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0'\n                });\n\n                ssrcGroup.ssrcs.forEach(function (ssrc) {\n                    modify.c('source', { ssrc: ssrc })\n                        .up(); // end of source\n                });\n                modify.up(); // end of ssrc-group\n            }\n        });\n\n        modify.up(); // end of description\n        modify.up(); // end of content\n    });\n\n    return modified;\n};\n\nmodule.exports = SDPDiffer;","SDPUtil = {\n    iceparams: function (mediadesc, sessiondesc) {\n        var data = null;\n        if (SDPUtil.find_line(mediadesc, 'a=ice-ufrag:', sessiondesc) &&\n            SDPUtil.find_line(mediadesc, 'a=ice-pwd:', sessiondesc)) {\n            data = {\n                ufrag: SDPUtil.parse_iceufrag(SDPUtil.find_line(mediadesc, 'a=ice-ufrag:', sessiondesc)),\n                pwd: SDPUtil.parse_icepwd(SDPUtil.find_line(mediadesc, 'a=ice-pwd:', sessiondesc))\n            };\n        }\n        return data;\n    },\n    parse_iceufrag: function (line) {\n        return line.substring(12);\n    },\n    build_iceufrag: function (frag) {\n        return 'a=ice-ufrag:' + frag;\n    },\n    parse_icepwd: function (line) {\n        return line.substring(10);\n    },\n    build_icepwd: function (pwd) {\n        return 'a=ice-pwd:' + pwd;\n    },\n    parse_mid: function (line) {\n        return line.substring(6);\n    },\n    parse_mline: function (line) {\n        var parts = line.substring(2).split(' '),\n            data = {};\n        data.media = parts.shift();\n        data.port = parts.shift();\n        data.proto = parts.shift();\n        if (parts[parts.length - 1] === '') { // trailing whitespace\n            parts.pop();\n        }\n        data.fmt = parts;\n        return data;\n    },\n    build_mline: function (mline) {\n        return 'm=' + mline.media + ' ' + mline.port + ' ' + mline.proto + ' ' + mline.fmt.join(' ');\n    },\n    parse_rtpmap: function (line) {\n        var parts = line.substring(9).split(' '),\n            data = {};\n        data.id = parts.shift();\n        parts = parts[0].split('/');\n        data.name = parts.shift();\n        data.clockrate = parts.shift();\n        data.channels = parts.length ? parts.shift() : '1';\n        return data;\n    },\n    /**\n     * Parses SDP line \"a=sctpmap:...\" and extracts SCTP port from it.\n     * @param line eg. \"a=sctpmap:5000 webrtc-datachannel\"\n     * @returns [SCTP port number, protocol, streams]\n     */\n    parse_sctpmap: function (line)\n    {\n        var parts = line.substring(10).split(' ');\n        var sctpPort = parts[0];\n        var protocol = parts[1];\n        // Stream count is optional\n        var streamCount = parts.length > 2 ? parts[2] : null;\n        return [sctpPort, protocol, streamCount];// SCTP port\n    },\n    build_rtpmap: function (el) {\n        var line = 'a=rtpmap:' + el.getAttribute('id') + ' ' + el.getAttribute('name') + '/' + el.getAttribute('clockrate');\n        if (el.getAttribute('channels') && el.getAttribute('channels') != '1') {\n            line += '/' + el.getAttribute('channels');\n        }\n        return line;\n    },\n    parse_crypto: function (line) {\n        var parts = line.substring(9).split(' '),\n            data = {};\n        data.tag = parts.shift();\n        data['crypto-suite'] = parts.shift();\n        data['key-params'] = parts.shift();\n        if (parts.length) {\n            data['session-params'] = parts.join(' ');\n        }\n        return data;\n    },\n    parse_fingerprint: function (line) { // RFC 4572\n        var parts = line.substring(14).split(' '),\n            data = {};\n        data.hash = parts.shift();\n        data.fingerprint = parts.shift();\n        // TODO assert that fingerprint satisfies 2UHEX *(\":\" 2UHEX) ?\n        return data;\n    },\n    parse_fmtp: function (line) {\n        var parts = line.split(' '),\n            i, key, value,\n            data = [];\n        parts.shift();\n        parts = parts.join(' ').split(';');\n        for (i = 0; i < parts.length; i++) {\n            key = parts[i].split('=')[0];\n            while (key.length && key[0] == ' ') {\n                key = key.substring(1);\n            }\n            value = parts[i].split('=')[1];\n            if (key && value) {\n                data.push({name: key, value: value});\n            } else if (key) {\n                // rfc 4733 (DTMF) style stuff\n                data.push({name: '', value: key});\n            }\n        }\n        return data;\n    },\n    parse_icecandidate: function (line) {\n        var candidate = {},\n            elems = line.split(' ');\n        candidate.foundation = elems[0].substring(12);\n        candidate.component = elems[1];\n        candidate.protocol = elems[2].toLowerCase();\n        candidate.priority = elems[3];\n        candidate.ip = elems[4];\n        candidate.port = elems[5];\n        // elems[6] => \"typ\"\n        candidate.type = elems[7];\n        candidate.generation = 0; // default value, may be overwritten below\n        for (var i = 8; i < elems.length; i += 2) {\n            switch (elems[i]) {\n                case 'raddr':\n                    candidate['rel-addr'] = elems[i + 1];\n                    break;\n                case 'rport':\n                    candidate['rel-port'] = elems[i + 1];\n                    break;\n                case 'generation':\n                    candidate.generation = elems[i + 1];\n                    break;\n                case 'tcptype':\n                    candidate.tcptype = elems[i + 1];\n                    break;\n                default: // TODO\n                    console.log('parse_icecandidate not translating \"' + elems[i] + '\" = \"' + elems[i + 1] + '\"');\n            }\n        }\n        candidate.network = '1';\n        candidate.id = Math.random().toString(36).substr(2, 10); // not applicable to SDP -- FIXME: should be unique, not just random\n        return candidate;\n    },\n    build_icecandidate: function (cand) {\n        var line = ['a=candidate:' + cand.foundation, cand.component, cand.protocol, cand.priority, cand.ip, cand.port, 'typ', cand.type].join(' ');\n        line += ' ';\n        switch (cand.type) {\n            case 'srflx':\n            case 'prflx':\n            case 'relay':\n                if (cand.hasOwnAttribute('rel-addr') && cand.hasOwnAttribute('rel-port')) {\n                    line += 'raddr';\n                    line += ' ';\n                    line += cand['rel-addr'];\n                    line += ' ';\n                    line += 'rport';\n                    line += ' ';\n                    line += cand['rel-port'];\n                    line += ' ';\n                }\n                break;\n        }\n        if (cand.hasOwnAttribute('tcptype')) {\n            line += 'tcptype';\n            line += ' ';\n            line += cand.tcptype;\n            line += ' ';\n        }\n        line += 'generation';\n        line += ' ';\n        line += cand.hasOwnAttribute('generation') ? cand.generation : '0';\n        return line;\n    },\n    parse_ssrc: function (desc) {\n        // proprietary mapping of a=ssrc lines\n        // TODO: see \"Jingle RTP Source Description\" by Juberti and P. Thatcher on google docs\n        // and parse according to that\n        var lines = desc.split('\\r\\n'),\n            data = {};\n        for (var i = 0; i < lines.length; i++) {\n            if (lines[i].substring(0, 7) == 'a=ssrc:') {\n                var idx = lines[i].indexOf(' ');\n                data[lines[i].substr(idx + 1).split(':', 2)[0]] = lines[i].substr(idx + 1).split(':', 2)[1];\n            }\n        }\n        return data;\n    },\n    parse_rtcpfb: function (line) {\n        var parts = line.substr(10).split(' ');\n        var data = {};\n        data.pt = parts.shift();\n        data.type = parts.shift();\n        data.params = parts;\n        return data;\n    },\n    parse_extmap: function (line) {\n        var parts = line.substr(9).split(' ');\n        var data = {};\n        data.value = parts.shift();\n        if (data.value.indexOf('/') != -1) {\n            data.direction = data.value.substr(data.value.indexOf('/') + 1);\n            data.value = data.value.substr(0, data.value.indexOf('/'));\n        } else {\n            data.direction = 'both';\n        }\n        data.uri = parts.shift();\n        data.params = parts;\n        return data;\n    },\n    find_line: function (haystack, needle, sessionpart) {\n        var lines = haystack.split('\\r\\n');\n        for (var i = 0; i < lines.length; i++) {\n            if (lines[i].substring(0, needle.length) == needle) {\n                return lines[i];\n            }\n        }\n        if (!sessionpart) {\n            return false;\n        }\n        // search session part\n        lines = sessionpart.split('\\r\\n');\n        for (var j = 0; j < lines.length; j++) {\n            if (lines[j].substring(0, needle.length) == needle) {\n                return lines[j];\n            }\n        }\n        return false;\n    },\n    find_lines: function (haystack, needle, sessionpart) {\n        var lines = haystack.split('\\r\\n'),\n            needles = [];\n        for (var i = 0; i < lines.length; i++) {\n            if (lines[i].substring(0, needle.length) == needle)\n                needles.push(lines[i]);\n        }\n        if (needles.length || !sessionpart) {\n            return needles;\n        }\n        // search session part\n        lines = sessionpart.split('\\r\\n');\n        for (var j = 0; j < lines.length; j++) {\n            if (lines[j].substring(0, needle.length) == needle) {\n                needles.push(lines[j]);\n            }\n        }\n        return needles;\n    },\n    candidateToJingle: function (line) {\n        // a=candidate:2979166662 1 udp 2113937151 192.168.2.100 57698 typ host generation 0\n        //      <candidate component=... foundation=... generation=... id=... ip=... network=... port=... priority=... protocol=... type=.../>\n        if (line.indexOf('candidate:') === 0) {\n            line = 'a=' + line;\n        } else if (line.substring(0, 12) != 'a=candidate:') {\n            console.log('parseCandidate called with a line that is not a candidate line');\n            console.log(line);\n            return null;\n        }\n        if (line.substring(line.length - 2) == '\\r\\n') // chomp it\n            line = line.substring(0, line.length - 2);\n        var candidate = {},\n            elems = line.split(' '),\n            i;\n        if (elems[6] != 'typ') {\n            console.log('did not find typ in the right place');\n            console.log(line);\n            return null;\n        }\n        candidate.foundation = elems[0].substring(12);\n        candidate.component = elems[1];\n        candidate.protocol = elems[2].toLowerCase();\n        candidate.priority = elems[3];\n        candidate.ip = elems[4];\n        candidate.port = elems[5];\n        // elems[6] => \"typ\"\n        candidate.type = elems[7];\n\n        candidate.generation = '0'; // default, may be overwritten below\n        for (i = 8; i < elems.length; i += 2) {\n            switch (elems[i]) {\n                case 'raddr':\n                    candidate['rel-addr'] = elems[i + 1];\n                    break;\n                case 'rport':\n                    candidate['rel-port'] = elems[i + 1];\n                    break;\n                case 'generation':\n                    candidate.generation = elems[i + 1];\n                    break;\n                case 'tcptype':\n                    candidate.tcptype = elems[i + 1];\n                    break;\n                default: // TODO\n                    console.log('not translating \"' + elems[i] + '\" = \"' + elems[i + 1] + '\"');\n            }\n        }\n        candidate.network = '1';\n        candidate.id = Math.random().toString(36).substr(2, 10); // not applicable to SDP -- FIXME: should be unique, not just random\n        return candidate;\n    },\n    candidateFromJingle: function (cand) {\n        var line = 'a=candidate:';\n        line += cand.getAttribute('foundation');\n        line += ' ';\n        line += cand.getAttribute('component');\n        line += ' ';\n        line += cand.getAttribute('protocol'); //.toUpperCase(); // chrome M23 doesn't like this\n        line += ' ';\n        line += cand.getAttribute('priority');\n        line += ' ';\n        line += cand.getAttribute('ip');\n        line += ' ';\n        line += cand.getAttribute('port');\n        line += ' ';\n        line += 'typ';\n        line += ' ' + cand.getAttribute('type');\n        line += ' ';\n        switch (cand.getAttribute('type')) {\n            case 'srflx':\n            case 'prflx':\n            case 'relay':\n                if (cand.getAttribute('rel-addr') && cand.getAttribute('rel-port')) {\n                    line += 'raddr';\n                    line += ' ';\n                    line += cand.getAttribute('rel-addr');\n                    line += ' ';\n                    line += 'rport';\n                    line += ' ';\n                    line += cand.getAttribute('rel-port');\n                    line += ' ';\n                }\n                break;\n        }\n        if (cand.getAttribute('protocol').toLowerCase() == 'tcp') {\n            line += 'tcptype';\n            line += ' ';\n            line += cand.getAttribute('tcptype');\n            line += ' ';\n        }\n        line += 'generation';\n        line += ' ';\n        line += cand.getAttribute('generation') || '0';\n        return line + '\\r\\n';\n    }\n};\nmodule.exports = SDPUtil;","function TraceablePeerConnection(ice_config, constraints) {\n    var self = this;\n    var RTCPeerconnection = navigator.mozGetUserMedia ? mozRTCPeerConnection : webkitRTCPeerConnection;\n    this.peerconnection = new RTCPeerconnection(ice_config, constraints);\n    this.updateLog = [];\n    this.stats = {};\n    this.statsinterval = null;\n    this.maxstats = 0; // limit to 300 values, i.e. 5 minutes; set to 0 to disable\n\n    // override as desired\n    this.trace = function (what, info) {\n        //console.warn('WTRACE', what, info);\n        self.updateLog.push({\n            time: new Date(),\n            type: what,\n            value: info || \"\"\n        });\n    };\n    this.onicecandidate = null;\n    this.peerconnection.onicecandidate = function (event) {\n        self.trace('onicecandidate', JSON.stringify(event.candidate, null, ' '));\n        if (self.onicecandidate !== null) {\n            self.onicecandidate(event);\n        }\n    };\n    this.onaddstream = null;\n    this.peerconnection.onaddstream = function (event) {\n        self.trace('onaddstream', event.stream.id);\n        if (self.onaddstream !== null) {\n            self.onaddstream(event);\n        }\n    };\n    this.onremovestream = null;\n    this.peerconnection.onremovestream = function (event) {\n        self.trace('onremovestream', event.stream.id);\n        if (self.onremovestream !== null) {\n            self.onremovestream(event);\n        }\n    };\n    this.onsignalingstatechange = null;\n    this.peerconnection.onsignalingstatechange = function (event) {\n        self.trace('onsignalingstatechange', self.signalingState);\n        if (self.onsignalingstatechange !== null) {\n            self.onsignalingstatechange(event);\n        }\n    };\n    this.oniceconnectionstatechange = null;\n    this.peerconnection.oniceconnectionstatechange = function (event) {\n        self.trace('oniceconnectionstatechange', self.iceConnectionState);\n        if (self.oniceconnectionstatechange !== null) {\n            self.oniceconnectionstatechange(event);\n        }\n    };\n    this.onnegotiationneeded = null;\n    this.peerconnection.onnegotiationneeded = function (event) {\n        self.trace('onnegotiationneeded');\n        if (self.onnegotiationneeded !== null) {\n            self.onnegotiationneeded(event);\n        }\n    };\n    self.ondatachannel = null;\n    this.peerconnection.ondatachannel = function (event) {\n        self.trace('ondatachannel', event);\n        if (self.ondatachannel !== null) {\n            self.ondatachannel(event);\n        }\n    };\n    if (!navigator.mozGetUserMedia && this.maxstats) {\n        this.statsinterval = window.setInterval(function() {\n            self.peerconnection.getStats(function(stats) {\n                var results = stats.result();\n                for (var i = 0; i < results.length; ++i) {\n                    //console.log(results[i].type, results[i].id, results[i].names())\n                    var now = new Date();\n                    results[i].names().forEach(function (name) {\n                        var id = results[i].id + '-' + name;\n                        if (!self.stats[id]) {\n                            self.stats[id] = {\n                                startTime: now,\n                                endTime: now,\n                                values: [],\n                                times: []\n                            };\n                        }\n                        self.stats[id].values.push(results[i].stat(name));\n                        self.stats[id].times.push(now.getTime());\n                        if (self.stats[id].values.length > self.maxstats) {\n                            self.stats[id].values.shift();\n                            self.stats[id].times.shift();\n                        }\n                        self.stats[id].endTime = now;\n                    });\n                }\n            });\n\n        }, 1000);\n    }\n};\n\ndumpSDP = function(description) {\n    return 'type: ' + description.type + '\\r\\n' + description.sdp;\n}\n\nif (TraceablePeerConnection.prototype.__defineGetter__ !== undefined) {\n    TraceablePeerConnection.prototype.__defineGetter__('signalingState', function() { return this.peerconnection.signalingState; });\n    TraceablePeerConnection.prototype.__defineGetter__('iceConnectionState', function() { return this.peerconnection.iceConnectionState; });\n    TraceablePeerConnection.prototype.__defineGetter__('localDescription', function() {\n        var publicLocalDescription = simulcast.reverseTransformLocalDescription(this.peerconnection.localDescription);\n        return publicLocalDescription;\n    });\n    TraceablePeerConnection.prototype.__defineGetter__('remoteDescription', function() {\n        var publicRemoteDescription = simulcast.reverseTransformRemoteDescription(this.peerconnection.remoteDescription);\n        return publicRemoteDescription;\n    });\n}\n\nTraceablePeerConnection.prototype.addStream = function (stream) {\n    this.trace('addStream', stream.id);\n    simulcast.resetSender();\n    try\n    {\n        this.peerconnection.addStream(stream);\n    }\n    catch (e)\n    {\n        console.error(e);\n        return;\n    }\n};\n\nTraceablePeerConnection.prototype.removeStream = function (stream, stopStreams) {\n    this.trace('removeStream', stream.id);\n    simulcast.resetSender();\n    if(stopStreams) {\n        stream.getAudioTracks().forEach(function (track) {\n            track.stop();\n        });\n        stream.getVideoTracks().forEach(function (track) {\n            track.stop();\n        });\n    }\n    this.peerconnection.removeStream(stream);\n};\n\nTraceablePeerConnection.prototype.createDataChannel = function (label, opts) {\n    this.trace('createDataChannel', label, opts);\n    return this.peerconnection.createDataChannel(label, opts);\n};\n\nTraceablePeerConnection.prototype.setLocalDescription = function (description, successCallback, failureCallback) {\n    var self = this;\n    description = simulcast.transformLocalDescription(description);\n    this.trace('setLocalDescription', dumpSDP(description));\n    this.peerconnection.setLocalDescription(description,\n        function () {\n            self.trace('setLocalDescriptionOnSuccess');\n            successCallback();\n        },\n        function (err) {\n            self.trace('setLocalDescriptionOnFailure', err);\n            failureCallback(err);\n        }\n    );\n    /*\n     if (this.statsinterval === null && this.maxstats > 0) {\n     // start gathering stats\n     }\n     */\n};\n\nTraceablePeerConnection.prototype.setRemoteDescription = function (description, successCallback, failureCallback) {\n    var self = this;\n    description = simulcast.transformRemoteDescription(description);\n    this.trace('setRemoteDescription', dumpSDP(description));\n    this.peerconnection.setRemoteDescription(description,\n        function () {\n            self.trace('setRemoteDescriptionOnSuccess');\n            successCallback();\n        },\n        function (err) {\n            self.trace('setRemoteDescriptionOnFailure', err);\n            failureCallback(err);\n        }\n    );\n    /*\n     if (this.statsinterval === null && this.maxstats > 0) {\n     // start gathering stats\n     }\n     */\n};\n\nTraceablePeerConnection.prototype.close = function () {\n    this.trace('stop');\n    if (this.statsinterval !== null) {\n        window.clearInterval(this.statsinterval);\n        this.statsinterval = null;\n    }\n    this.peerconnection.close();\n};\n\nTraceablePeerConnection.prototype.createOffer = function (successCallback, failureCallback, constraints) {\n    var self = this;\n    this.trace('createOffer', JSON.stringify(constraints, null, ' '));\n    this.peerconnection.createOffer(\n        function (offer) {\n            self.trace('createOfferOnSuccess', dumpSDP(offer));\n            successCallback(offer);\n        },\n        function(err) {\n            self.trace('createOfferOnFailure', err);\n            failureCallback(err);\n        },\n        constraints\n    );\n};\n\nTraceablePeerConnection.prototype.createAnswer = function (successCallback, failureCallback, constraints) {\n    var self = this;\n    this.trace('createAnswer', JSON.stringify(constraints, null, ' '));\n    this.peerconnection.createAnswer(\n        function (answer) {\n            answer = simulcast.transformAnswer(answer);\n            self.trace('createAnswerOnSuccess', dumpSDP(answer));\n            successCallback(answer);\n        },\n        function(err) {\n            self.trace('createAnswerOnFailure', err);\n            failureCallback(err);\n        },\n        constraints\n    );\n};\n\nTraceablePeerConnection.prototype.addIceCandidate = function (candidate, successCallback, failureCallback) {\n    var self = this;\n    this.trace('addIceCandidate', JSON.stringify(candidate, null, ' '));\n    this.peerconnection.addIceCandidate(candidate);\n    /* maybe later\n     this.peerconnection.addIceCandidate(candidate,\n     function () {\n     self.trace('addIceCandidateOnSuccess');\n     successCallback();\n     },\n     function (err) {\n     self.trace('addIceCandidateOnFailure', err);\n     failureCallback(err);\n     }\n     );\n     */\n};\n\nTraceablePeerConnection.prototype.getStats = function(callback, errback) {\n    if (navigator.mozGetUserMedia) {\n        // ignore for now...\n        if(!errback)\n            errback = function () {\n\n            }\n        this.peerconnection.getStats(null,callback,errback);\n    } else {\n        this.peerconnection.getStats(callback);\n    }\n};\n\nmodule.exports = TraceablePeerConnection;\n\n","/* global $, $iq, config, connection, UI, messageHandler,\n roomName, sessionTerminated, Strophe, Util */\n/**\n * Contains logic responsible for enabling/disabling functionality available\n * only to moderator users.\n */\nvar connection = null;\nvar focusUserJid;\nvar getNextTimeout = Util.createExpBackoffTimer(1000);\nvar getNextErrorTimeout = Util.createExpBackoffTimer(1000);\n// External authentication stuff\nvar externalAuthEnabled = false;\n// Sip gateway can be enabled by configuring Jigasi host in config.js or\n// it will be enabled automatically if focus detects the component through\n// service discovery.\nvar sipGatewayEnabled = config.hosts.call_control !== undefined;\n\nvar Moderator = {\n    isModerator: function () {\n        return connection && connection.emuc.isModerator();\n    },\n\n    isPeerModerator: function (peerJid) {\n        return connection &&\n            connection.emuc.getMemberRole(peerJid) === 'moderator';\n    },\n\n    isExternalAuthEnabled: function () {\n        return externalAuthEnabled;\n    },\n\n    isSipGatewayEnabled: function () {\n        return sipGatewayEnabled;\n    },\n\n    setConnection: function (con) {\n        connection = con;\n    },\n\n    init: function (xmpp) {\n        this.xmppService = xmpp;\n        this.onLocalRoleChange = function (from, member, pres) {\n            UI.onModeratorStatusChanged(Moderator.isModerator());\n        };\n    },\n\n    onMucLeft: function (jid) {\n        console.info(\"Someone left is it focus ? \" + jid);\n        var resource = Strophe.getResourceFromJid(jid);\n        if (resource === 'focus' && !this.xmppService.sessionTerminated) {\n            console.info(\n                \"Focus has left the room - leaving conference\");\n            //hangUp();\n            // We'd rather reload to have everything re-initialized\n            // FIXME: show some message before reload\n            location.reload();\n        }\n    },\n    \n    setFocusUserJid: function (focusJid) {\n        if (!focusUserJid) {\n            focusUserJid = focusJid;\n            console.info(\"Focus jid set to: \" + focusUserJid);\n        }\n    },\n\n    getFocusUserJid: function () {\n        return focusUserJid;\n    },\n\n    getFocusComponent: function () {\n        // Get focus component address\n        var focusComponent = config.hosts.focus;\n        // If not specified use default: 'focus.domain'\n        if (!focusComponent) {\n            focusComponent = 'focus.' + config.hosts.domain;\n        }\n        return focusComponent;\n    },\n\n    createConferenceIq: function (roomName) {\n        // Generate create conference IQ\n        var elem = $iq({to: Moderator.getFocusComponent(), type: 'set'});\n        elem.c('conference', {\n            xmlns: 'http://jitsi.org/protocol/focus',\n            room: roomName\n        });\n        if (config.hosts.bridge !== undefined) {\n            elem.c(\n                'property',\n                { name: 'bridge', value: config.hosts.bridge})\n                .up();\n        }\n        // Tell the focus we have Jigasi configured\n        if (config.hosts.call_control !== undefined) {\n            elem.c(\n                'property',\n                { name: 'call_control', value: config.hosts.call_control})\n                .up();\n        }\n        if (config.channelLastN !== undefined) {\n            elem.c(\n                'property',\n                { name: 'channelLastN', value: config.channelLastN})\n                .up();\n        }\n        if (config.adaptiveLastN !== undefined) {\n            elem.c(\n                'property',\n                { name: 'adaptiveLastN', value: config.adaptiveLastN})\n                .up();\n        }\n        if (config.adaptiveSimulcast !== undefined) {\n            elem.c(\n                'property',\n                { name: 'adaptiveSimulcast', value: config.adaptiveSimulcast})\n                .up();\n        }\n        if (config.openSctp !== undefined) {\n            elem.c(\n                'property',\n                { name: 'openSctp', value: config.openSctp})\n                .up();\n        }\n        if (config.enableFirefoxSupport !== undefined) {\n            elem.c(\n                'property',\n                { name: 'enableFirefoxHacks',\n                    value: config.enableFirefoxSupport})\n                .up();\n        }\n        elem.up();\n        return elem;\n    },\n\n    parseConfigOptions: function (resultIq) {\n    \n        Moderator.setFocusUserJid(\n            $(resultIq).find('conference').attr('focusjid'));\n    \n        var extAuthParam\n            = $(resultIq).find('>conference>property[name=\\'externalAuth\\']');\n        if (extAuthParam.length) {\n            externalAuthEnabled = extAuthParam.attr('value') === 'true';\n        }\n    \n        console.info(\"External authentication enabled: \" + externalAuthEnabled);\n    \n        // Check if focus has auto-detected Jigasi component(this will be also\n        // included if we have passed our host from the config)\n        if ($(resultIq).find(\n            '>conference>property[name=\\'sipGatewayEnabled\\']').length) {\n            sipGatewayEnabled = true;\n        }\n    \n        console.info(\"Sip gateway enabled: \" + sipGatewayEnabled);\n    },\n\n    // FIXME: we need to show the fact that we're waiting for the focus\n    // to the user(or that focus is not available)\n    allocateConferenceFocus: function (roomName, callback) {\n        // Try to use focus user JID from the config\n        Moderator.setFocusUserJid(config.focusUserJid);\n        // Send create conference IQ\n        var iq = Moderator.createConferenceIq(roomName);\n        connection.sendIQ(\n            iq,\n            function (result) {\n                if ('true' === $(result).find('conference').attr('ready')) {\n                    // Reset both timers\n                    getNextTimeout(true);\n                    getNextErrorTimeout(true);\n                    // Setup config options\n                    Moderator.parseConfigOptions(result);\n                    // Exec callback\n                    callback();\n                } else {\n                    var waitMs = getNextTimeout();\n                    console.info(\"Waiting for the focus... \" + waitMs);\n                    // Reset error timeout\n                    getNextErrorTimeout(true);\n                    window.setTimeout(\n                        function () {\n                            Moderator.allocateConferenceFocus(\n                                roomName, callback);\n                        }, waitMs);\n                }\n            },\n            function (error) {\n                // Not authorized to create new room\n                if ($(error).find('>error>not-authorized').length) {\n                    console.warn(\"Unauthorized to start the conference\");\n                    UI.onAuthenticationRequired(function () {\n                        Moderator.allocateConferenceFocus(roomName, callback);\n                    });\n                    return;\n                }\n                var waitMs = getNextErrorTimeout();\n                console.error(\"Focus error, retry after \" + waitMs, error);\n                // Show message\n                UI.messageHandler.notify(\n                    'Conference focus', 'disconnected',\n                        Moderator.getFocusComponent() +\n                        ' not available - retry in ' +\n                        (waitMs / 1000) + ' sec');\n                // Reset response timeout\n                getNextTimeout(true);\n                window.setTimeout(\n                    function () {\n                        Moderator.allocateConferenceFocus(roomName, callback);\n                    }, waitMs);\n            }\n        );\n    },\n\n    getAuthUrl: function (roomName, urlCallback) {\n        var iq = $iq({to: Moderator.getFocusComponent(), type: 'get'});\n        iq.c('auth-url', {\n            xmlns: 'http://jitsi.org/protocol/focus',\n            room: roomName\n        });\n        connection.sendIQ(\n            iq,\n            function (result) {\n                var url = $(result).find('auth-url').attr('url');\n                if (url) {\n                    console.info(\"Got auth url: \" + url);\n                    urlCallback(url);\n                } else {\n                    console.error(\n                        \"Failed to get auth url fro mthe focus\", result);\n                }\n            },\n            function (error) {\n                console.error(\"Get auth url error\", error);\n            }\n        );\n    }\n};\n\nmodule.exports = Moderator;\n\n\n\n","/* global $, $iq, config, connection, focusMucJid, messageHandler, Moderator,\n   Toolbar, Util */\nvar Moderator = require(\"./moderator\");\n\n\nvar recordingToken = null;\nvar recordingEnabled;\n\n/**\n * Whether to use a jirecon component for recording, or use the videobridge\n * through COLIBRI.\n */\nvar useJirecon = (typeof config.hosts.jirecon != \"undefined\");\n\n/**\n * The ID of the jirecon recording session. Jirecon generates it when we\n * initially start recording, and it needs to be used in subsequent requests\n * to jirecon.\n */\nvar jireconRid = null;\n\nfunction setRecordingToken(token) {\n    recordingToken = token;\n}\n\nfunction setRecording(state, token, callback) {\n    if (useJirecon){\n        this.setRecordingJirecon(state, token, callback);\n    } else {\n        this.setRecordingColibri(state, token, callback);\n    }\n}\n\nfunction setRecordingJirecon(state, token, callback) {\n    if (state == recordingEnabled){\n        return;\n    }\n\n    var iq = $iq({to: config.hosts.jirecon, type: 'set'})\n        .c('recording', {xmlns: 'http://jitsi.org/protocol/jirecon',\n            action: state ? 'start' : 'stop',\n            mucjid: connection.emuc.roomjid});\n    if (!state){\n        iq.attrs({rid: jireconRid});\n    }\n\n    console.log('Start recording');\n\n    connection.sendIQ(\n        iq,\n        function (result) {\n            // TODO wait for an IQ with the real status, since this is\n            // provisional?\n            jireconRid = $(result).find('recording').attr('rid');\n            console.log('Recording ' + (state ? 'started' : 'stopped') +\n                '(jirecon)' + result);\n            recordingEnabled = state;\n            if (!state){\n                jireconRid = null;\n            }\n\n            callback(state);\n        },\n        function (error) {\n            console.log('Failed to start recording, error: ', error);\n            callback(recordingEnabled);\n        });\n}\n\n// Sends a COLIBRI message which enables or disables (according to 'state')\n// the recording on the bridge. Waits for the result IQ and calls 'callback'\n// with the new recording state, according to the IQ.\nfunction setRecordingColibri(state, token, callback) {\n    var elem = $iq({to: focusMucJid, type: 'set'});\n    elem.c('conference', {\n        xmlns: 'http://jitsi.org/protocol/colibri'\n    });\n    elem.c('recording', {state: state, token: token});\n\n    connection.sendIQ(elem,\n        function (result) {\n            console.log('Set recording \"', state, '\". Result:', result);\n            var recordingElem = $(result).find('>conference>recording');\n            var newState = ('true' === recordingElem.attr('state'));\n\n            recordingEnabled = newState;\n            callback(newState);\n        },\n        function (error) {\n            console.warn(error);\n            callback(recordingEnabled);\n        }\n    );\n}\n\nvar Recording = {\n    toggleRecording: function (tokenEmptyCallback,\n                               startingCallback, startedCallback) {\n        if (!Moderator.isModerator()) {\n            console.log(\n                    'non-focus, or conference not yet organized:' +\n                    ' not enabling recording');\n            return;\n        }\n\n        // Jirecon does not (currently) support a token.\n        if (!recordingToken && !useJirecon) {\n            tokenEmptyCallback(function (value) {\n                setRecordingToken(value);\n                this.toggleRecording();\n            });\n\n            return;\n        }\n\n        var oldState = recordingEnabled;\n        startingCallback(!oldState);\n        setRecording(!oldState,\n            recordingToken,\n            function (state) {\n                console.log(\"New recording state: \", state);\n                if (state === oldState) {\n                    // FIXME: new focus:\n                    // this will not work when moderator changes\n                    // during active session. Then it will assume that\n                    // recording status has changed to true, but it might have\n                    // been already true(and we only received actual status from\n                    // the focus).\n                    //\n                    // SO we start with status null, so that it is initialized\n                    // here and will fail only after second click, so if invalid\n                    // token was used we have to press the button twice before\n                    // current status will be fetched and token will be reset.\n                    //\n                    // Reliable way would be to return authentication error.\n                    // Or status update when moderator connects.\n                    // Or we have to stop recording session when current\n                    // moderator leaves the room.\n\n                    // Failed to change, reset the token because it might\n                    // have been wrong\n                    setRecordingToken(null);\n                }\n                startedCallback(state);\n\n            }\n        );\n    }\n\n}\n\nmodule.exports = Recording;","/* jshint -W117 */\n/* a simple MUC connection plugin\n * can only handle a single MUC room\n */\n\nvar bridgeIsDown = false;\n\nvar Moderator = require(\"./moderator\");\n\nmodule.exports = function(XMPP, eventEmitter) {\n    Strophe.addConnectionPlugin('emuc', {\n        connection: null,\n        roomjid: null,\n        myroomjid: null,\n        members: {},\n        list_members: [], // so we can elect a new focus\n        presMap: {},\n        preziMap: {},\n        joined: false,\n        isOwner: false,\n        role: null,\n        init: function (conn) {\n            this.connection = conn;\n        },\n        initPresenceMap: function (myroomjid) {\n            this.presMap['to'] = myroomjid;\n            this.presMap['xns'] = 'http://jabber.org/protocol/muc';\n        },\n        doJoin: function (jid, password) {\n            this.myroomjid = jid;\n\n            console.info(\"Joined MUC as \" + this.myroomjid);\n\n            this.initPresenceMap(this.myroomjid);\n\n            if (!this.roomjid) {\n                this.roomjid = Strophe.getBareJidFromJid(jid);\n                // add handlers (just once)\n                this.connection.addHandler(this.onPresence.bind(this), null, 'presence', null, null, this.roomjid, {matchBare: true});\n                this.connection.addHandler(this.onPresenceUnavailable.bind(this), null, 'presence', 'unavailable', null, this.roomjid, {matchBare: true});\n                this.connection.addHandler(this.onPresenceError.bind(this), null, 'presence', 'error', null, this.roomjid, {matchBare: true});\n                this.connection.addHandler(this.onMessage.bind(this), null, 'message', null, null, this.roomjid, {matchBare: true});\n            }\n            if (password !== undefined) {\n                this.presMap['password'] = password;\n            }\n            this.sendPresence();\n        },\n        doLeave: function () {\n            console.log(\"do leave\", this.myroomjid);\n            var pres = $pres({to: this.myroomjid, type: 'unavailable' });\n            this.presMap.length = 0;\n            this.connection.send(pres);\n        },\n        createNonAnonymousRoom: function () {\n            // http://xmpp.org/extensions/xep-0045.html#createroom-reserved\n\n            var getForm = $iq({type: 'get', to: this.roomjid})\n                .c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'})\n                .c('x', {xmlns: 'jabber:x:data', type: 'submit'});\n\n            this.connection.sendIQ(getForm, function (form) {\n\n                if (!$(form).find(\n                        '>query>x[xmlns=\"jabber:x:data\"]' +\n                        '>field[var=\"muc#roomconfig_whois\"]').length) {\n\n                    console.error('non-anonymous rooms not supported');\n                    return;\n                }\n\n                var formSubmit = $iq({to: this.roomjid, type: 'set'})\n                    .c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'});\n\n                formSubmit.c('x', {xmlns: 'jabber:x:data', type: 'submit'});\n\n                formSubmit.c('field', {'var': 'FORM_TYPE'})\n                    .c('value')\n                    .t('http://jabber.org/protocol/muc#roomconfig').up().up();\n\n                formSubmit.c('field', {'var': 'muc#roomconfig_whois'})\n                    .c('value').t('anyone').up().up();\n\n                this.connection.sendIQ(formSubmit);\n\n            }, function (error) {\n                console.error(\"Error getting room configuration form\");\n            });\n        },\n        onPresence: function (pres) {\n            var from = pres.getAttribute('from');\n\n            // What is this for? A workaround for something?\n            if (pres.getAttribute('type')) {\n                return true;\n            }\n\n            // Parse etherpad tag.\n            var etherpad = $(pres).find('>etherpad');\n            if (etherpad.length) {\n                if (config.etherpad_base && !Moderator.isModerator()) {\n                    UI.initEtherpad(etherpad.text());\n                }\n            }\n\n            // Parse prezi tag.\n            var presentation = $(pres).find('>prezi');\n            if (presentation.length) {\n                var url = presentation.attr('url');\n                var current = presentation.find('>current').text();\n\n                console.log('presentation info received from', from, url);\n\n                if (this.preziMap[from] == null) {\n                    this.preziMap[from] = url;\n\n                    $(document).trigger('presentationadded.muc', [from, url, current]);\n                }\n                else {\n                    $(document).trigger('gotoslide.muc', [from, url, current]);\n                }\n            }\n            else if (this.preziMap[from] != null) {\n                var url = this.preziMap[from];\n                delete this.preziMap[from];\n                $(document).trigger('presentationremoved.muc', [from, url]);\n            }\n\n            // Parse audio info tag.\n            var audioMuted = $(pres).find('>audiomuted');\n            if (audioMuted.length) {\n                $(document).trigger('audiomuted.muc', [from, audioMuted.text()]);\n            }\n\n            // Parse video info tag.\n            var videoMuted = $(pres).find('>videomuted');\n            if (videoMuted.length) {\n                $(document).trigger('videomuted.muc', [from, videoMuted.text()]);\n            }\n\n            var stats = $(pres).find('>stats');\n            if (stats.length) {\n                var statsObj = {};\n                Strophe.forEachChild(stats[0], \"stat\", function (el) {\n                    statsObj[el.getAttribute(\"name\")] = el.getAttribute(\"value\");\n                });\n                connectionquality.updateRemoteStats(from, statsObj);\n            }\n\n            // Parse status.\n            if ($(pres).find('>x[xmlns=\"http://jabber.org/protocol/muc#user\"]>status[code=\"201\"]').length) {\n                this.isOwner = true;\n                this.createNonAnonymousRoom();\n            }\n\n            // Parse roles.\n            var member = {};\n            member.show = $(pres).find('>show').text();\n            member.status = $(pres).find('>status').text();\n            var tmp = $(pres).find('>x[xmlns=\"http://jabber.org/protocol/muc#user\"]>item');\n            member.affiliation = tmp.attr('affiliation');\n            member.role = tmp.attr('role');\n\n            // Focus recognition\n            member.jid = tmp.attr('jid');\n            member.isFocus = false;\n            if (member.jid\n                && member.jid.indexOf(Moderator.getFocusUserJid() + \"/\") == 0) {\n                member.isFocus = true;\n            }\n\n            var nicktag = $(pres).find('>nick[xmlns=\"http://jabber.org/protocol/nick\"]');\n            member.displayName = (nicktag.length > 0 ? nicktag.html() : null);\n\n            if (from == this.myroomjid) {\n                if (member.affiliation == 'owner') this.isOwner = true;\n                if (this.role !== member.role) {\n                    this.role = member.role;\n                    if (Moderator.onLocalRoleChange)\n                        Moderator.onLocalRoleChange(from, member, pres);\n                    UI.onLocalRoleChange(from, member, pres);\n                }\n                if (!this.joined) {\n                    this.joined = true;\n                    eventEmitter.emit(XMPPEvents.MUC_JOINED, from, member);\n                    this.list_members.push(from);\n                }\n            } else if (this.members[from] === undefined) {\n                // new participant\n                this.members[from] = member;\n                this.list_members.push(from);\n                console.log('entered', from, member);\n                if (member.isFocus) {\n                    focusMucJid = from;\n                    console.info(\"Ignore focus: \" + from + \", real JID: \" + member.jid);\n                }\n                else {\n                    var id = $(pres).find('>userID').text();\n                    var email = $(pres).find('>email');\n                    if (email.length > 0) {\n                        id = email.text();\n                    }\n                    UI.onMucEntered(from, id, member.displayName);\n                    API.triggerEvent(\"participantJoined\", {jid: from});\n                }\n            } else {\n                // Presence update for existing participant\n                // Watch role change:\n                if (this.members[from].role != member.role) {\n                    this.members[from].role = member.role;\n                    UI.onMucRoleChanged(member.role, member.displayName);\n                }\n            }\n\n            // Always trigger presence to update bindings\n            $(document).trigger('presence.muc', [from, member, pres]);\n            this.parsePresence(from, member, pres);\n\n            // Trigger status message update\n            if (member.status) {\n                UI.onMucPresenceStatus(from, member);\n            }\n\n            return true;\n        },\n        onPresenceUnavailable: function (pres) {\n            var from = pres.getAttribute('from');\n            // Status code 110 indicates that this notification is \"self-presence\".\n            if (!$(pres).find('>x[xmlns=\"http://jabber.org/protocol/muc#user\"]>status[code=\"110\"]').length) {\n                delete this.members[from];\n                this.list_members.splice(this.list_members.indexOf(from), 1);\n                this.onParticipantLeft(from);\n            }\n            // If the status code is 110 this means we're leaving and we would like\n            // to remove everyone else from our view, so we trigger the event.\n            else if (this.list_members.length > 1) {\n                for (var i = 0; i < this.list_members.length; i++) {\n                    var member = this.list_members[i];\n                    delete this.members[i];\n                    this.list_members.splice(i, 1);\n                    this.onParticipantLeft(member);\n                }\n            }\n            if ($(pres).find('>x[xmlns=\"http://jabber.org/protocol/muc#user\"]>status[code=\"307\"]').length) {\n                $(document).trigger('kicked.muc', [from]);\n                if (this.myroomjid === from) {\n                    XMPP.disposeConference(false);\n                    eventEmitter.emit(XMPPEvents.KICKED);\n                }\n            }\n            return true;\n        },\n        onPresenceError: function (pres) {\n            var from = pres.getAttribute('from');\n            if ($(pres).find('>error[type=\"auth\"]>not-authorized[xmlns=\"urn:ietf:params:xml:ns:xmpp-stanzas\"]').length) {\n                console.log('on password required', from);\n                var self = this;\n                UI.onPasswordReqiured(function (value) {\n                    self.doJoin(from, value);\n                });\n            } else if ($(pres).find(\n                '>error[type=\"cancel\"]>not-allowed[xmlns=\"urn:ietf:params:xml:ns:xmpp-stanzas\"]').length) {\n                var toDomain = Strophe.getDomainFromJid(pres.getAttribute('to'));\n                if (toDomain === config.hosts.anonymousdomain) {\n                    // we are connected with anonymous domain and only non anonymous users can create rooms\n                    // we must authorize the user\n                    XMPP.promptLogin();\n                } else {\n                    console.warn('onPresError ', pres);\n                    UI.messageHandler.openReportDialog(null,\n                        'Oops! Something went wrong and we couldn`t connect to the conference.',\n                        pres);\n                }\n            } else {\n                console.warn('onPresError ', pres);\n                UI.messageHandler.openReportDialog(null,\n                    'Oops! Something went wrong and we couldn`t connect to the conference.',\n                    pres);\n            }\n            return true;\n        },\n        sendMessage: function (body, nickname) {\n            var msg = $msg({to: this.roomjid, type: 'groupchat'});\n            msg.c('body', body).up();\n            if (nickname) {\n                msg.c('nick', {xmlns: 'http://jabber.org/protocol/nick'}).t(nickname).up().up();\n            }\n            this.connection.send(msg);\n            API.triggerEvent(\"outgoingMessage\", {\"message\": body});\n        },\n        setSubject: function (subject) {\n            var msg = $msg({to: this.roomjid, type: 'groupchat'});\n            msg.c('subject', subject);\n            this.connection.send(msg);\n            console.log(\"topic changed to \" + subject);\n        },\n        onMessage: function (msg) {\n            // FIXME: this is a hack. but jingle on muc makes nickchanges hard\n            var from = msg.getAttribute('from');\n            var nick = $(msg).find('>nick[xmlns=\"http://jabber.org/protocol/nick\"]').text() || Strophe.getResourceFromJid(from);\n\n            var txt = $(msg).find('>body').text();\n            var type = msg.getAttribute(\"type\");\n            if (type == \"error\") {\n                UI.chatAddError($(msg).find('>text').text(), txt);\n                return true;\n            }\n\n            var subject = $(msg).find('>subject');\n            if (subject.length) {\n                var subjectText = subject.text();\n                if (subjectText || subjectText == \"\") {\n                    UI.chatSetSubject(subjectText);\n                    console.log(\"Subject is changed to \" + subjectText);\n                }\n            }\n\n\n            if (txt) {\n                console.log('chat', nick, txt);\n                UI.updateChatConversation(from, nick, txt);\n                if (from != this.myroomjid)\n                    API.triggerEvent(\"incomingMessage\",\n                        {\"from\": from, \"nick\": nick, \"message\": txt});\n            }\n            return true;\n        },\n        lockRoom: function (key, onSuccess, onError, onNotSupported) {\n            //http://xmpp.org/extensions/xep-0045.html#roomconfig\n            var ob = this;\n            this.connection.sendIQ($iq({to: this.roomjid, type: 'get'}).c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'}),\n                function (res) {\n                    if ($(res).find('>query>x[xmlns=\"jabber:x:data\"]>field[var=\"muc#roomconfig_roomsecret\"]').length) {\n                        var formsubmit = $iq({to: ob.roomjid, type: 'set'}).c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'});\n                        formsubmit.c('x', {xmlns: 'jabber:x:data', type: 'submit'});\n                        formsubmit.c('field', {'var': 'FORM_TYPE'}).c('value').t('http://jabber.org/protocol/muc#roomconfig').up().up();\n                        formsubmit.c('field', {'var': 'muc#roomconfig_roomsecret'}).c('value').t(key).up().up();\n                        // Fixes a bug in prosody 0.9.+ https://code.google.com/p/lxmppd/issues/detail?id=373\n                        formsubmit.c('field', {'var': 'muc#roomconfig_whois'}).c('value').t('anyone').up().up();\n                        // FIXME: is muc#roomconfig_passwordprotectedroom required?\n                        this.connection.sendIQ(formsubmit,\n                            onSuccess,\n                            onError);\n                    } else {\n                        onNotSupported();\n                    }\n                }, onError);\n        },\n        kick: function (jid) {\n            var kickIQ = $iq({to: this.roomjid, type: 'set'})\n                .c('query', {xmlns: 'http://jabber.org/protocol/muc#admin'})\n                .c('item', {nick: Strophe.getResourceFromJid(jid), role: 'none'})\n                .c('reason').t('You have been kicked.').up().up().up();\n\n            this.connection.sendIQ(\n                kickIQ,\n                function (result) {\n                    console.log('Kick participant with jid: ', jid, result);\n                },\n                function (error) {\n                    console.log('Kick participant error: ', error);\n                });\n        },\n        sendPresence: function () {\n            var pres = $pres({to: this.presMap['to'] });\n            pres.c('x', {xmlns: this.presMap['xns']});\n\n            if (this.presMap['password']) {\n                pres.c('password').t(this.presMap['password']).up();\n            }\n\n            pres.up();\n\n            // Send XEP-0115 'c' stanza that contains our capabilities info\n            if (this.connection.caps) {\n                this.connection.caps.node = config.clientNode;\n                pres.c('c', this.connection.caps.generateCapsAttrs()).up();\n            }\n\n            pres.c('user-agent', {xmlns: 'http://jitsi.org/jitmeet/user-agent'})\n                .t(navigator.userAgent).up();\n\n            if (this.presMap['bridgeIsDown']) {\n                pres.c('bridgeIsDown').up();\n            }\n\n            if (this.presMap['email']) {\n                pres.c('email').t(this.presMap['email']).up();\n            }\n\n            if (this.presMap['userId']) {\n                pres.c('userId').t(this.presMap['userId']).up();\n            }\n\n            if (this.presMap['displayName']) {\n                // XEP-0172\n                pres.c('nick', {xmlns: 'http://jabber.org/protocol/nick'})\n                    .t(this.presMap['displayName']).up();\n            }\n\n            if (this.presMap['audions']) {\n                pres.c('audiomuted', {xmlns: this.presMap['audions']})\n                    .t(this.presMap['audiomuted']).up();\n            }\n\n            if (this.presMap['videons']) {\n                pres.c('videomuted', {xmlns: this.presMap['videons']})\n                    .t(this.presMap['videomuted']).up();\n            }\n\n            if (this.presMap['statsns']) {\n                var stats = pres.c('stats', {xmlns: this.presMap['statsns']});\n                for (var stat in this.presMap[\"stats\"])\n                    if (this.presMap[\"stats\"][stat] != null)\n                        stats.c(\"stat\", {name: stat, value: this.presMap[\"stats\"][stat]}).up();\n                pres.up();\n            }\n\n            if (this.presMap['prezins']) {\n                pres.c('prezi',\n                    {xmlns: this.presMap['prezins'],\n                        'url': this.presMap['preziurl']})\n                    .c('current').t(this.presMap['prezicurrent']).up().up();\n            }\n\n            if (this.presMap['etherpadns']) {\n                pres.c('etherpad', {xmlns: this.presMap['etherpadns']})\n                    .t(this.presMap['etherpadname']).up();\n            }\n\n            if (this.presMap['medians']) {\n                pres.c('media', {xmlns: this.presMap['medians']});\n                var sourceNumber = 0;\n                Object.keys(this.presMap).forEach(function (key) {\n                    if (key.indexOf('source') >= 0) {\n                        sourceNumber++;\n                    }\n                });\n                if (sourceNumber > 0)\n                    for (var i = 1; i <= sourceNumber / 3; i++) {\n                        pres.c('source',\n                            {type: this.presMap['source' + i + '_type'],\n                                ssrc: this.presMap['source' + i + '_ssrc'],\n                                direction: this.presMap['source' + i + '_direction']\n                                    || 'sendrecv' }\n                        ).up();\n                    }\n            }\n\n            pres.up();\n//        console.debug(pres.toString());\n            this.connection.send(pres);\n        },\n        addDisplayNameToPresence: function (displayName) {\n            this.presMap['displayName'] = displayName;\n        },\n        addMediaToPresence: function (sourceNumber, mtype, ssrcs, direction) {\n            if (!this.presMap['medians'])\n                this.presMap['medians'] = 'http://estos.de/ns/mjs';\n\n            this.presMap['source' + sourceNumber + '_type'] = mtype;\n            this.presMap['source' + sourceNumber + '_ssrc'] = ssrcs;\n            this.presMap['source' + sourceNumber + '_direction'] = direction;\n        },\n        clearPresenceMedia: function () {\n            var self = this;\n            Object.keys(this.presMap).forEach(function (key) {\n                if (key.indexOf('source') != -1) {\n                    delete self.presMap[key];\n                }\n            });\n        },\n        addPreziToPresence: function (url, currentSlide) {\n            this.presMap['prezins'] = 'http://jitsi.org/jitmeet/prezi';\n            this.presMap['preziurl'] = url;\n            this.presMap['prezicurrent'] = currentSlide;\n        },\n        removePreziFromPresence: function () {\n            delete this.presMap['prezins'];\n            delete this.presMap['preziurl'];\n            delete this.presMap['prezicurrent'];\n        },\n        addCurrentSlideToPresence: function (currentSlide) {\n            this.presMap['prezicurrent'] = currentSlide;\n        },\n        getPrezi: function (roomjid) {\n            return this.preziMap[roomjid];\n        },\n        addEtherpadToPresence: function (etherpadName) {\n            this.presMap['etherpadns'] = 'http://jitsi.org/jitmeet/etherpad';\n            this.presMap['etherpadname'] = etherpadName;\n        },\n        addAudioInfoToPresence: function (isMuted) {\n            this.presMap['audions'] = 'http://jitsi.org/jitmeet/audio';\n            this.presMap['audiomuted'] = isMuted.toString();\n        },\n        addVideoInfoToPresence: function (isMuted) {\n            this.presMap['videons'] = 'http://jitsi.org/jitmeet/video';\n            this.presMap['videomuted'] = isMuted.toString();\n        },\n        addConnectionInfoToPresence: function (stats) {\n            this.presMap['statsns'] = 'http://jitsi.org/jitmeet/stats';\n            this.presMap['stats'] = stats;\n        },\n        findJidFromResource: function (resourceJid) {\n            if (resourceJid &&\n                resourceJid === Strophe.getResourceFromJid(this.myroomjid)) {\n                return this.myroomjid;\n            }\n            var peerJid = null;\n            Object.keys(this.members).some(function (jid) {\n                peerJid = jid;\n                return Strophe.getResourceFromJid(jid) === resourceJid;\n            });\n            return peerJid;\n        },\n        addBridgeIsDownToPresence: function () {\n            this.presMap['bridgeIsDown'] = true;\n        },\n        addEmailToPresence: function (email) {\n            this.presMap['email'] = email;\n        },\n        addUserIdToPresence: function (userId) {\n            this.presMap['userId'] = userId;\n        },\n        isModerator: function () {\n            return this.role === 'moderator';\n        },\n        getMemberRole: function (peerJid) {\n            if (this.members[peerJid]) {\n                return this.members[peerJid].role;\n            }\n            return null;\n        },\n        onParticipantLeft: function (jid) {\n            UI.onMucLeft(jid);\n\n            API.triggerEvent(\"participantLeft\", {jid: jid});\n\n            delete jid2Ssrc[jid];\n\n            this.connection.jingle.terminateByJid(jid);\n\n            if (this.getPrezi(jid)) {\n                $(document).trigger('presentationremoved.muc',\n                    [jid, this.getPrezi(jid)]);\n            }\n\n            Moderator.onMucLeft(jid);\n        },\n        parsePresence: function (from, memeber, pres) {\n            if($(pres).find(\">bridgeIsDown\").length > 0 && !bridgeIsDown) {\n                bridgeIsDown = true;\n                eventEmitter.emit(XMPPEvents.BRIDGE_DOWN);\n            }\n\n            if(memeber.isFocus)\n                return;\n\n            // Remove old ssrcs coming from the jid\n            Object.keys(ssrc2jid).forEach(function (ssrc) {\n                if (ssrc2jid[ssrc] == jid) {\n                    delete ssrc2jid[ssrc];\n                    delete ssrc2videoType[ssrc];\n                }\n            });\n\n            var changedStreams = [];\n            $(pres).find('>media[xmlns=\"http://estos.de/ns/mjs\"]>source').each(function (idx, ssrc) {\n                //console.log(jid, 'assoc ssrc', ssrc.getAttribute('type'), ssrc.getAttribute('ssrc'));\n                var ssrcV = ssrc.getAttribute('ssrc');\n                ssrc2jid[ssrcV] = from;\n                notReceivedSSRCs.push(ssrcV);\n\n                var type = ssrc.getAttribute('type');\n                ssrc2videoType[ssrcV] = type;\n\n                var direction = ssrc.getAttribute('direction');\n\n                changedStreams.push({type: type, direction: direction});\n\n            });\n\n            eventEmitter.emit(XMPPEvents.CHANGED_STREAMS, from, changedStreams);\n\n            var displayName = !config.displayJids\n                ? memeber.displayName : Strophe.getResourceFromJid(from);\n\n            if (displayName && displayName.length > 0)\n            {\n//                $(document).trigger('displaynamechanged',\n//                    [jid, displayName]);\n                eventEmitter.emit(XMPPEvents.DISPLAY_NAME_CHANGED, from, displayName);\n            }\n\n\n            var id = $(pres).find('>userID').text();\n            var email = $(pres).find('>email');\n            if(email.length > 0) {\n                id = email.text();\n            }\n\n            eventEmitter.emit(XMPPEvents.USER_ID_CHANGED, from, id);\n        }\n    });\n};\n\n","/* jshint -W117 */\n\nvar JingleSession = require(\"./JingleSession\");\n\nfunction CallIncomingJingle(sid, connection) {\n    var sess = connection.jingle.sessions[sid];\n\n    // TODO: do we check activecall == null?\n    activecall = sess;\n\n    statistics.onConferenceCreated(sess);\n    RTC.onConferenceCreated(sess);\n\n    // TODO: check affiliation and/or role\n    console.log('emuc data for', sess.peerjid, connection.emuc.members[sess.peerjid]);\n    sess.usedrip = true; // not-so-naive trickle ice\n    sess.sendAnswer();\n    sess.accept();\n\n};\n\nmodule.exports = function(XMPP)\n{\n    Strophe.addConnectionPlugin('jingle', {\n        connection: null,\n        sessions: {},\n        jid2session: {},\n        ice_config: {iceServers: []},\n        pc_constraints: {},\n        media_constraints: {\n            mandatory: {\n                'OfferToReceiveAudio': true,\n                'OfferToReceiveVideo': true\n            }\n            // MozDontOfferDataChannel: true when this is firefox\n        },\n        init: function (conn) {\n            this.connection = conn;\n            if (this.connection.disco) {\n                // http://xmpp.org/extensions/xep-0167.html#support\n                // http://xmpp.org/extensions/xep-0176.html#support\n                this.connection.disco.addFeature('urn:xmpp:jingle:1');\n                this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:1');\n                this.connection.disco.addFeature('urn:xmpp:jingle:transports:ice-udp:1');\n                this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:audio');\n                this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:video');\n\n\n                // this is dealt with by SDP O/A so we don't need to annouce this\n                //this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:rtcp-fb:0'); // XEP-0293\n                //this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:rtp-hdrext:0'); // XEP-0294\n                if (config.useRtcpMux) {\n                    this.connection.disco.addFeature('urn:ietf:rfc:5761'); // rtcp-mux\n                }\n                if (config.useBundle) {\n                    this.connection.disco.addFeature('urn:ietf:rfc:5888'); // a=group, e.g. bundle\n                }\n                //this.connection.disco.addFeature('urn:ietf:rfc:5576'); // a=ssrc\n            }\n            this.connection.addHandler(this.onJingle.bind(this), 'urn:xmpp:jingle:1', 'iq', 'set', null, null);\n        },\n        onJingle: function (iq) {\n            var sid = $(iq).find('jingle').attr('sid');\n            var action = $(iq).find('jingle').attr('action');\n            var fromJid = iq.getAttribute('from');\n            // send ack first\n            var ack = $iq({type: 'result',\n                to: fromJid,\n                id: iq.getAttribute('id')\n            });\n            console.log('on jingle ' + action + ' from ' + fromJid, iq);\n            var sess = this.sessions[sid];\n            if ('session-initiate' != action) {\n                if (sess === null) {\n                    ack.type = 'error';\n                    ack.c('error', {type: 'cancel'})\n                        .c('item-not-found', {xmlns: 'urn:ietf:params:xml:ns:xmpp-stanzas'}).up()\n                        .c('unknown-session', {xmlns: 'urn:xmpp:jingle:errors:1'});\n                    this.connection.send(ack);\n                    return true;\n                }\n                // compare from to sess.peerjid (bare jid comparison for later compat with message-mode)\n                // local jid is not checked\n                if (Strophe.getBareJidFromJid(fromJid) != Strophe.getBareJidFromJid(sess.peerjid)) {\n                    console.warn('jid mismatch for session id', sid, fromJid, sess.peerjid);\n                    ack.type = 'error';\n                    ack.c('error', {type: 'cancel'})\n                        .c('item-not-found', {xmlns: 'urn:ietf:params:xml:ns:xmpp-stanzas'}).up()\n                        .c('unknown-session', {xmlns: 'urn:xmpp:jingle:errors:1'});\n                    this.connection.send(ack);\n                    return true;\n                }\n            } else if (sess !== undefined) {\n                // existing session with same session id\n                // this might be out-of-order if the sess.peerjid is the same as from\n                ack.type = 'error';\n                ack.c('error', {type: 'cancel'})\n                    .c('service-unavailable', {xmlns: 'urn:ietf:params:xml:ns:xmpp-stanzas'}).up();\n                console.warn('duplicate session id', sid);\n                this.connection.send(ack);\n                return true;\n            }\n            // FIXME: check for a defined action\n            this.connection.send(ack);\n            // see http://xmpp.org/extensions/xep-0166.html#concepts-session\n            switch (action) {\n                case 'session-initiate':\n                    sess = new JingleSession(\n                        $(iq).attr('to'), $(iq).find('jingle').attr('sid'),\n                        this.connection, XMPP);\n                    // configure session\n\n                    sess.media_constraints = this.media_constraints;\n                    sess.pc_constraints = this.pc_constraints;\n                    sess.ice_config = this.ice_config;\n\n                    sess.initiate(fromJid, false);\n                    // FIXME: setRemoteDescription should only be done when this call is to be accepted\n                    sess.setRemoteDescription($(iq).find('>jingle'), 'offer');\n\n                    this.sessions[sess.sid] = sess;\n                    this.jid2session[sess.peerjid] = sess;\n\n                    // the callback should either\n                    // .sendAnswer and .accept\n                    // or .sendTerminate -- not necessarily synchronus\n                    CallIncomingJingle(sess.sid, this.connection);\n                    break;\n                case 'session-accept':\n                    sess.setRemoteDescription($(iq).find('>jingle'), 'answer');\n                    sess.accept();\n                    $(document).trigger('callaccepted.jingle', [sess.sid]);\n                    break;\n                case 'session-terminate':\n                    // If this is not the focus sending the terminate, we have\n                    // nothing more to do here.\n                    if (Object.keys(this.sessions).length < 1\n                        || !(this.sessions[Object.keys(this.sessions)[0]]\n                            instanceof JingleSession))\n                    {\n                        break;\n                    }\n                    console.log('terminating...', sess.sid);\n                    sess.terminate();\n                    this.terminate(sess.sid);\n                    if ($(iq).find('>jingle>reason').length) {\n                        $(document).trigger('callterminated.jingle', [\n                            sess.sid,\n                            sess.peerjid,\n                            $(iq).find('>jingle>reason>:first')[0].tagName,\n                            $(iq).find('>jingle>reason>text').text()\n                        ]);\n                    } else {\n                        $(document).trigger('callterminated.jingle',\n                            [sess.sid, sess.peerjid]);\n                    }\n                    break;\n                case 'transport-info':\n                    sess.addIceCandidate($(iq).find('>jingle>content'));\n                    break;\n                case 'session-info':\n                    var affected;\n                    if ($(iq).find('>jingle>ringing[xmlns=\"urn:xmpp:jingle:apps:rtp:info:1\"]').length) {\n                        $(document).trigger('ringing.jingle', [sess.sid]);\n                    } else if ($(iq).find('>jingle>mute[xmlns=\"urn:xmpp:jingle:apps:rtp:info:1\"]').length) {\n                        affected = $(iq).find('>jingle>mute[xmlns=\"urn:xmpp:jingle:apps:rtp:info:1\"]').attr('name');\n                        $(document).trigger('mute.jingle', [sess.sid, affected]);\n                    } else if ($(iq).find('>jingle>unmute[xmlns=\"urn:xmpp:jingle:apps:rtp:info:1\"]').length) {\n                        affected = $(iq).find('>jingle>unmute[xmlns=\"urn:xmpp:jingle:apps:rtp:info:1\"]').attr('name');\n                        $(document).trigger('unmute.jingle', [sess.sid, affected]);\n                    }\n                    break;\n                case 'addsource': // FIXME: proprietary, un-jingleish\n                case 'source-add': // FIXME: proprietary\n                    sess.addSource($(iq).find('>jingle>content'), fromJid);\n                    break;\n                case 'removesource': // FIXME: proprietary, un-jingleish\n                case 'source-remove': // FIXME: proprietary\n                    sess.removeSource($(iq).find('>jingle>content'), fromJid);\n                    break;\n                default:\n                    console.warn('jingle action not implemented', action);\n                    break;\n            }\n            return true;\n        },\n        initiate: function (peerjid, myjid) { // initiate a new jinglesession to peerjid\n            var sess = new JingleSession(myjid || this.connection.jid,\n                Math.random().toString(36).substr(2, 12), // random string\n                this.connection, XMPP);\n            // configure session\n\n            sess.media_constraints = this.media_constraints;\n            sess.pc_constraints = this.pc_constraints;\n            sess.ice_config = this.ice_config;\n\n            sess.initiate(peerjid, true);\n            this.sessions[sess.sid] = sess;\n            this.jid2session[sess.peerjid] = sess;\n            sess.sendOffer();\n            return sess;\n        },\n        terminate: function (sid, reason, text) { // terminate by sessionid (or all sessions)\n            if (sid === null || sid === undefined) {\n                for (sid in this.sessions) {\n                    if (this.sessions[sid].state != 'ended') {\n                        this.sessions[sid].sendTerminate(reason || (!this.sessions[sid].active()) ? 'cancel' : null, text);\n                        this.sessions[sid].terminate();\n                    }\n                    delete this.jid2session[this.sessions[sid].peerjid];\n                    delete this.sessions[sid];\n                }\n            } else if (this.sessions.hasOwnProperty(sid)) {\n                if (this.sessions[sid].state != 'ended') {\n                    this.sessions[sid].sendTerminate(reason || (!this.sessions[sid].active()) ? 'cancel' : null, text);\n                    this.sessions[sid].terminate();\n                }\n                delete this.jid2session[this.sessions[sid].peerjid];\n                delete this.sessions[sid];\n            }\n        },\n        // Used to terminate a session when an unavailable presence is received.\n        terminateByJid: function (jid) {\n            if (this.jid2session.hasOwnProperty(jid)) {\n                var sess = this.jid2session[jid];\n                if (sess) {\n                    sess.terminate();\n                    console.log('peer went away silently', jid);\n                    delete this.sessions[sess.sid];\n                    delete this.jid2session[jid];\n                    $(document).trigger('callterminated.jingle',\n                        [sess.sid, jid], 'gone');\n                }\n            }\n        },\n        terminateRemoteByJid: function (jid, reason) {\n            if (this.jid2session.hasOwnProperty(jid)) {\n                var sess = this.jid2session[jid];\n                if (sess) {\n                    sess.sendTerminate(reason || (!sess.active()) ? 'kick' : null);\n                    sess.terminate();\n                    console.log('terminate peer with jid', sess.sid, jid);\n                    delete this.sessions[sess.sid];\n                    delete this.jid2session[jid];\n                    $(document).trigger('callterminated.jingle',\n                        [sess.sid, jid, 'kicked']);\n                }\n            }\n        },\n        getStunAndTurnCredentials: function () {\n            // get stun and turn configuration from server via xep-0215\n            // uses time-limited credentials as described in\n            // http://tools.ietf.org/html/draft-uberti-behave-turn-rest-00\n            //\n            // see https://code.google.com/p/prosody-modules/source/browse/mod_turncredentials/mod_turncredentials.lua\n            // for a prosody module which implements this\n            //\n            // currently, this doesn't work with updateIce and therefore credentials with a long\n            // validity have to be fetched before creating the peerconnection\n            // TODO: implement refresh via updateIce as described in\n            //      https://code.google.com/p/webrtc/issues/detail?id=1650\n            var self = this;\n            this.connection.sendIQ(\n                $iq({type: 'get', to: this.connection.domain})\n                    .c('services', {xmlns: 'urn:xmpp:extdisco:1'}).c('service', {host: 'turn.' + this.connection.domain}),\n                function (res) {\n                    var iceservers = [];\n                    $(res).find('>services>service').each(function (idx, el) {\n                        el = $(el);\n                        var dict = {};\n                        var type = el.attr('type');\n                        switch (type) {\n                            case 'stun':\n                                dict.url = 'stun:' + el.attr('host');\n                                if (el.attr('port')) {\n                                    dict.url += ':' + el.attr('port');\n                                }\n                                iceservers.push(dict);\n                                break;\n                            case 'turn':\n                            case 'turns':\n                                dict.url = type + ':';\n                                if (el.attr('username')) { // https://code.google.com/p/webrtc/issues/detail?id=1508\n                                    if (navigator.userAgent.match(/Chrom(e|ium)\\/([0-9]+)\\./) && parseInt(navigator.userAgent.match(/Chrom(e|ium)\\/([0-9]+)\\./)[2], 10) < 28) {\n                                        dict.url += el.attr('username') + '@';\n                                    } else {\n                                        dict.username = el.attr('username'); // only works in M28\n                                    }\n                                }\n                                dict.url += el.attr('host');\n                                if (el.attr('port') && el.attr('port') != '3478') {\n                                    dict.url += ':' + el.attr('port');\n                                }\n                                if (el.attr('transport') && el.attr('transport') != 'udp') {\n                                    dict.url += '?transport=' + el.attr('transport');\n                                }\n                                if (el.attr('password')) {\n                                    dict.credential = el.attr('password');\n                                }\n                                iceservers.push(dict);\n                                break;\n                        }\n                    });\n                    self.ice_config.iceServers = iceservers;\n                },\n                function (err) {\n                    console.warn('getting turn credentials failed', err);\n                    console.warn('is mod_turncredentials or similar installed?');\n                }\n            );\n            // implement push?\n        },\n\n        /**\n         * Populates the log data\n         */\n        populateData: function () {\n            var data = {};\n            Object.keys(this.sessions).forEach(function (sid) {\n                var session = this.sessions[sid];\n                if (session.peerconnection && session.peerconnection.updateLog) {\n                    // FIXME: should probably be a .dump call\n                    data[\"jingle_\" + session.sid] = {\n                        updateLog: session.peerconnection.updateLog,\n                        stats: session.peerconnection.stats,\n                        url: window.location.href\n                    };\n                }\n            });\n            return data;\n        }\n    });\n};\n\n","/* global Strophe */\nmodule.exports = function () {\n\n    Strophe.addConnectionPlugin('logger', {\n        // logs raw stanzas and makes them available for download as JSON\n        connection: null,\n        log: [],\n        init: function (conn) {\n            this.connection = conn;\n            this.connection.rawInput = this.log_incoming.bind(this);\n            this.connection.rawOutput = this.log_outgoing.bind(this);\n        },\n        log_incoming: function (stanza) {\n            this.log.push([new Date().getTime(), 'incoming', stanza]);\n        },\n        log_outgoing: function (stanza) {\n            this.log.push([new Date().getTime(), 'outgoing', stanza]);\n        }\n    });\n};","/* global $, $iq, config, connection, focusMucJid, forceMuted,\n   setAudioMuted, Strophe */\n/**\n * Moderate connection plugin.\n */\nmodule.exports = function (XMPP) {\n    Strophe.addConnectionPlugin('moderate', {\n        connection: null,\n        init: function (conn) {\n            this.connection = conn;\n\n            this.connection.addHandler(this.onMute.bind(this),\n                'http://jitsi.org/jitmeet/audio',\n                'iq',\n                'set',\n                null,\n                null);\n        },\n        setMute: function (jid, mute) {\n            console.info(\"set mute\", mute);\n            var iqToFocus = $iq({to: focusMucJid, type: 'set'})\n                .c('mute', {\n                    xmlns: 'http://jitsi.org/jitmeet/audio',\n                    jid: jid\n                })\n                .t(mute.toString())\n                .up();\n\n            this.connection.sendIQ(\n                iqToFocus,\n                function (result) {\n                    console.log('set mute', result);\n                },\n                function (error) {\n                    console.log('set mute error', error);\n                });\n        },\n        onMute: function (iq) {\n            var from = iq.getAttribute('from');\n            if (from !== focusMucJid) {\n                console.warn(\"Ignored mute from non focus peer\");\n                return false;\n            }\n            var mute = $(iq).find('mute');\n            if (mute.length) {\n                var doMuteAudio = mute.text() === \"true\";\n                UI.setAudioMuted(doMuteAudio);\n                XMPP.forceMuted = doMuteAudio;\n            }\n            return true;\n        },\n        eject: function (jid) {\n            // We're not the focus, so can't terminate\n            //connection.jingle.terminateRemoteByJid(jid, 'kick');\n            this.connection.emuc.kick(jid);\n        }\n    });\n}","/* jshint -W117 */\nmodule.exports = function() {\n    Strophe.addConnectionPlugin('rayo',\n        {\n            RAYO_XMLNS: 'urn:xmpp:rayo:1',\n            connection: null,\n            init: function (conn) {\n                this.connection = conn;\n                if (this.connection.disco) {\n                    this.connection.disco.addFeature('urn:xmpp:rayo:client:1');\n                }\n\n                this.connection.addHandler(\n                    this.onRayo.bind(this), this.RAYO_XMLNS, 'iq', 'set', null, null);\n            },\n            onRayo: function (iq) {\n                console.info(\"Rayo IQ\", iq);\n            },\n            dial: function (to, from, roomName, roomPass) {\n                var self = this;\n                var req = $iq(\n                    {\n                        type: 'set',\n                        to: focusMucJid\n                    }\n                );\n                req.c('dial',\n                    {\n                        xmlns: this.RAYO_XMLNS,\n                        to: to,\n                        from: from\n                    });\n                req.c('header',\n                    {\n                        name: 'JvbRoomName',\n                        value: roomName\n                    }).up();\n\n                if (roomPass !== null && roomPass.length) {\n\n                    req.c('header',\n                        {\n                            name: 'JvbRoomPassword',\n                            value: roomPass\n                        }).up();\n                }\n\n                this.connection.sendIQ(\n                    req,\n                    function (result) {\n                        console.info('Dial result ', result);\n\n                        var resource = $(result).find('ref').attr('uri');\n                        this.call_resource = resource.substr('xmpp:'.length);\n                        console.info(\n                                \"Received call resource: \" + this.call_resource);\n                    },\n                    function (error) {\n                        console.info('Dial error ', error);\n                    }\n                );\n            },\n            hang_up: function () {\n                if (!this.call_resource) {\n                    console.warn(\"No call in progress\");\n                    return;\n                }\n\n                var self = this;\n                var req = $iq(\n                    {\n                        type: 'set',\n                        to: this.call_resource\n                    }\n                );\n                req.c('hangup',\n                    {\n                        xmlns: this.RAYO_XMLNS\n                    });\n\n                this.connection.sendIQ(\n                    req,\n                    function (result) {\n                        console.info('Hangup result ', result);\n                        self.call_resource = null;\n                    },\n                    function (error) {\n                        console.info('Hangup error ', error);\n                        self.call_resource = null;\n                    }\n                );\n            }\n        }\n    );\n};\n","/**\n * Strophe logger implementation. Logs from level WARN and above.\n */\nmodule.exports = function () {\n\n    Strophe.log = function (level, msg) {\n        switch (level) {\n            case Strophe.LogLevel.WARN:\n                console.warn(\"Strophe: \" + msg);\n                break;\n            case Strophe.LogLevel.ERROR:\n            case Strophe.LogLevel.FATAL:\n                console.error(\"Strophe: \" + msg);\n                break;\n        }\n    };\n\n    Strophe.getStatusString = function (status) {\n        switch (status) {\n            case Strophe.Status.ERROR:\n                return \"ERROR\";\n            case Strophe.Status.CONNECTING:\n                return \"CONNECTING\";\n            case Strophe.Status.CONNFAIL:\n                return \"CONNFAIL\";\n            case Strophe.Status.AUTHENTICATING:\n                return \"AUTHENTICATING\";\n            case Strophe.Status.AUTHFAIL:\n                return \"AUTHFAIL\";\n            case Strophe.Status.CONNECTED:\n                return \"CONNECTED\";\n            case Strophe.Status.DISCONNECTED:\n                return \"DISCONNECTED\";\n            case Strophe.Status.DISCONNECTING:\n                return \"DISCONNECTING\";\n            case Strophe.Status.ATTACHED:\n                return \"ATTACHED\";\n            default:\n                return \"unknown\";\n        }\n    };\n};\n","var Moderator = require(\"./moderator\");\nvar EventEmitter = require(\"events\");\nvar Recording = require(\"./recording\");\nvar SDP = require(\"./SDP\");\n\nvar eventEmitter = new EventEmitter();\nvar connection = null;\nvar authenticatedUser = false;\nvar activecall = null;\n\nfunction connect(jid, password, uiCredentials) {\n    var bosh\n        = uiCredentials.bosh || config.bosh || '/http-bind';\n    connection = new Strophe.Connection(bosh);\n    Moderator.setConnection(connection);\n\n    var settings = UI.getSettings();\n    var email = settings.email;\n    var displayName = settings.displayName;\n    if(email) {\n        connection.emuc.addEmailToPresence(email);\n    } else {\n        connection.emuc.addUserIdToPresence(settings.uid);\n    }\n    if(displayName) {\n        connection.emuc.addDisplayNameToPresence(displayName);\n    }\n\n    if (connection.disco) {\n        // for chrome, add multistream cap\n    }\n    connection.jingle.pc_constraints = RTC.getPCConstraints();\n    if (config.useIPv6) {\n        // https://code.google.com/p/webrtc/issues/detail?id=2828\n        if (!connection.jingle.pc_constraints.optional)\n            connection.jingle.pc_constraints.optional = [];\n        connection.jingle.pc_constraints.optional.push({googIPv6: true});\n    }\n\n    if(!password)\n        password = uiCredentials.password;\n\n    var anonymousConnectionFailed = false;\n    connection.connect(jid, password, function (status, msg) {\n        console.log('Strophe status changed to',\n            Strophe.getStatusString(status));\n        if (status === Strophe.Status.CONNECTED) {\n            if (config.useStunTurn) {\n                connection.jingle.getStunAndTurnCredentials();\n            }\n            UI.disableConnect();\n\n            console.info(\"My Jabber ID: \" + connection.jid);\n\n            if(password)\n                authenticatedUser = true;\n            maybeDoJoin();\n        } else if (status === Strophe.Status.CONNFAIL) {\n            if(msg === 'x-strophe-bad-non-anon-jid') {\n                anonymousConnectionFailed = true;\n            }\n        } else if (status === Strophe.Status.DISCONNECTED) {\n            if(anonymousConnectionFailed) {\n                // prompt user for username and password\n                XMPP.promptLogin();\n            }\n        } else if (status === Strophe.Status.AUTHFAIL) {\n            // wrong password or username, prompt user\n            XMPP.promptLogin();\n\n        }\n    });\n}\n\n\n\nfunction maybeDoJoin() {\n    if (connection && connection.connected &&\n        Strophe.getResourceFromJid(connection.jid)\n        && (RTC.localAudio || RTC.localVideo)) {\n        // .connected is true while connecting?\n        doJoin();\n    }\n}\n\nfunction doJoin() {\n    var roomName = UI.generateRoomName();\n\n    Moderator.allocateConferenceFocus(\n        roomName, UI.checkForNicknameAndJoin);\n}\n\nfunction initStrophePlugins()\n{\n    require(\"./strophe.emuc\")(XMPP, eventEmitter);\n    require(\"./strophe.jingle\")();\n    require(\"./strophe.moderate\")(XMPP);\n    require(\"./strophe.util\")();\n    require(\"./strophe.rayo\")();\n    require(\"./strophe.logger\")();\n}\n\nfunction registerListeners() {\n    RTC.addStreamListener(maybeDoJoin,\n        StreamEventTypes.EVENT_TYPE_LOCAL_CREATED);\n}\n\nfunction setupEvents() {\n    $(window).bind('beforeunload', function () {\n        if (connection && connection.connected) {\n            // ensure signout\n            $.ajax({\n                type: 'POST',\n                url: config.bosh,\n                async: false,\n                cache: false,\n                contentType: 'application/xml',\n                data: \"<body rid='\" + (connection.rid || connection._proto.rid)\n                    + \"' xmlns='http://jabber.org/protocol/httpbind' sid='\"\n                    + (connection.sid || connection._proto.sid)\n                    + \"' type='terminate'>\" +\n                    \"<presence xmlns='jabber:client' type='unavailable'/>\" +\n                    \"</body>\",\n                success: function (data) {\n                    console.log('signed out');\n                    console.log(data);\n                },\n                error: function (XMLHttpRequest, textStatus, errorThrown) {\n                    console.log('signout error',\n                            textStatus + ' (' + errorThrown + ')');\n                }\n            });\n        }\n        XMPP.disposeConference(true);\n    });\n}\n\nvar XMPP = {\n    sessionTerminated: false,\n    /**\n     * Remembers if we were muted by the focus.\n     * @type {boolean}\n     */\n    forceMuted: false,\n    start: function (uiCredentials) {\n        setupEvents();\n        initStrophePlugins();\n        registerListeners();\n        Moderator.init();\n        var jid = uiCredentials.jid ||\n            config.hosts.anonymousdomain ||\n            config.hosts.domain ||\n            window.location.hostname;\n        connect(jid, null, uiCredentials);\n    },\n    promptLogin: function () {\n        UI.showLoginPopup(connect);\n    },\n    joinRooom: function(roomName, useNicks, nick)\n    {\n        var roomjid;\n        roomjid = roomName;\n\n        if (useNicks) {\n            if (nick) {\n                roomjid += '/' + nick;\n            } else {\n                roomjid += '/' + Strophe.getNodeFromJid(connection.jid);\n            }\n        } else {\n\n            var tmpJid = Strophe.getNodeFromJid(connection.jid);\n\n            if(!authenticatedUser)\n                tmpJid = tmpJid.substr(0, 8);\n\n            roomjid += '/' + tmpJid;\n        }\n        connection.emuc.doJoin(roomjid);\n    },\n    myJid: function () {\n        if(!connection)\n            return null;\n        return connection.emuc.myroomjid;\n    },\n    myResource: function () {\n        if(!connection || ! connection.emuc.myroomjid)\n            return null;\n        return Strophe.getResourceFromJid(connection.emuc.myroomjid);\n    },\n    disposeConference: function (onUnload) {\n        eventEmitter.emit(XMPPEvents.DISPOSE_CONFERENCE, onUnload);\n        var handler = activecall;\n        if (handler && handler.peerconnection) {\n            // FIXME: probably removing streams is not required and close() should\n            // be enough\n            if (RTC.localAudio) {\n                handler.peerconnection.removeStream(RTC.localAudio.getOriginalStream(), onUnload);\n            }\n            if (RTC.localVideo) {\n                handler.peerconnection.removeStream(RTC.localVideo.getOriginalStream(), onUnload);\n            }\n            handler.peerconnection.close();\n        }\n        activecall = null;\n        if(!onUnload)\n        {\n            this.sessionTerminated = true;\n            connection.emuc.doLeave();\n        }\n    },\n    addListener: function(type, listener)\n    {\n        eventEmitter.on(type, listener);\n    },\n    removeListener: function (type, listener) {\n        eventEmitter.removeListener(type, listener);\n    },\n    allocateConferenceFocus: function(roomName, callback) {\n        Moderator.allocateConferenceFocus(roomName, callback);\n    },\n    isModerator: function () {\n        return Moderator.isModerator();\n    },\n    isSipGatewayEnabled: function () {\n        return Moderator.isSipGatewayEnabled();\n    },\n    isExternalAuthEnabled: function () {\n        return Moderator.isExternalAuthEnabled();\n    },\n    switchStreams: function (stream, oldStream, callback) {\n        if (activecall) {\n            // FIXME: will block switchInProgress on true value in case of exception\n            activecall.switchStreams(stream, oldStream, callback);\n        } else {\n            // We are done immediately\n            console.error(\"No conference handler\");\n            UI.messageHandler.showError('Error',\n                'Unable to switch video stream.');\n            callback();\n        }\n    },\n    setVideoMute: function (mute, callback, options) {\n       if(activecall && connection && RTC.localVideo)\n       {\n           activecall.setVideoMute(mute, callback, options);\n       }\n    },\n    setAudioMute: function (mute, callback) {\n        if (!(connection && RTC.localAudio)) {\n            return false;\n        }\n\n\n        if (this.forceMuted && !mute) {\n            console.info(\"Asking focus for unmute\");\n            connection.moderate.setMute(connection.emuc.myroomjid, mute);\n            // FIXME: wait for result before resetting muted status\n            this.forceMuted = false;\n        }\n\n        if (mute == RTC.localAudio.isMuted()) {\n            // Nothing to do\n            return true;\n        }\n\n        // It is not clear what is the right way to handle multiple tracks.\n        // So at least make sure that they are all muted or all unmuted and\n        // that we send presence just once.\n        RTC.localAudio.mute();\n        // isMuted is the opposite of audioEnabled\n        connection.emuc.addAudioInfoToPresence(mute);\n        connection.emuc.sendPresence();\n        callback();\n        return true;\n    },\n    // Really mute video, i.e. dont even send black frames\n    muteVideo: function (pc, unmute) {\n        // FIXME: this probably needs another of those lovely state safeguards...\n        // which checks for iceconn == connected and sigstate == stable\n        pc.setRemoteDescription(pc.remoteDescription,\n            function () {\n                pc.createAnswer(\n                    function (answer) {\n                        var sdp = new SDP(answer.sdp);\n                        if (sdp.media.length > 1) {\n                            if (unmute)\n                                sdp.media[1] = sdp.media[1].replace('a=recvonly', 'a=sendrecv');\n                            else\n                                sdp.media[1] = sdp.media[1].replace('a=sendrecv', 'a=recvonly');\n                            sdp.raw = sdp.session + sdp.media.join('');\n                            answer.sdp = sdp.raw;\n                        }\n                        pc.setLocalDescription(answer,\n                            function () {\n                                console.log('mute SLD ok');\n                            },\n                            function (error) {\n                                console.log('mute SLD error');\n                                UI.messageHandler.showError('Error',\n                                        'Oops! Something went wrong and we failed to ' +\n                                        'mute! (SLD Failure)');\n                            }\n                        );\n                    },\n                    function (error) {\n                        console.log(error);\n                        UI.messageHandler.showError();\n                    }\n                );\n            },\n            function (error) {\n                console.log('muteVideo SRD error');\n                UI.messageHandler.showError('Error',\n                        'Oops! Something went wrong and we failed to stop video!' +\n                        '(SRD Failure)');\n\n            }\n        );\n    },\n    toggleRecording: function (tokenEmptyCallback,\n                               startingCallback, startedCallback) {\n        Recording.toggleRecording(tokenEmptyCallback,\n            startingCallback, startedCallback);\n    },\n    addToPresence: function (name, value, dontSend) {\n        switch (name)\n        {\n            case \"displayName\":\n                connection.emuc.addDisplayNameToPresence(value);\n                break;\n            case \"etherpad\":\n                connection.emuc.addEtherpadToPresence(value);\n                break;\n            case \"prezi\":\n                connection.emuc.addPreziToPresence(value, 0);\n                break;\n            case \"preziSlide\":\n                connection.emuc.addCurrentSlideToPresence(value);\n                break;\n            case \"connectionQuality\":\n                connection.emuc.addConnectionInfoToPresence(value);\n                break;\n            case \"email\":\n                connection.emuc.addEmailToPresence(value);\n            default :\n                console.log(\"Unknown tag for presence.\");\n                return;\n        }\n        if(!dontSend)\n            connection.emuc.sendPresence();\n    },\n    sendLogs: function (content) {\n        // XEP-0337-ish\n        var message = $msg({to: focusMucJid, type: 'normal'});\n        message.c('log', { xmlns: 'urn:xmpp:eventlog',\n            id: 'PeerConnectionStats'});\n        message.c('message').t(content).up();\n        if (deflate) {\n            message.c('tag', {name: \"deflated\", value: \"true\"}).up();\n        }\n        message.up();\n\n        connection.send(message);\n    },\n    populateData: function () {\n        var data = {};\n        if (connection.jingle) {\n            data = connection.jingle.populateData();\n        }\n        return data;\n    },\n    getLogger: function () {\n        if(connection.logger)\n            return connection.logger.log;\n        return null;\n    },\n    getPrezi: function () {\n        return connection.emuc.getPrezi(this.myJid());\n    },\n    removePreziFromPresence: function () {\n        connection.emuc.removePreziFromPresence();\n        connection.emuc.sendPresence();\n    },\n    sendChatMessage: function (message, nickname) {\n        connection.emuc.sendMessage(message, nickname);\n    },\n    setSubject: function (topic) {\n        connection.emuc.setSubject(topic);\n    },\n    lockRoom: function (key, onSuccess, onError, onNotSupported) {\n        connection.emuc.lockRoom(key, onSuccess, onError, onNotSupported);\n    },\n    dial: function (to, from, roomName,roomPass) {\n        connection.rayo.dial(to, from, roomName,roomPass);\n    },\n    setMute: function (jid, mute) {\n        connection.moderate.setMute(jid, mute);\n    },\n    eject: function (jid) {\n        connection.moderate.eject(jid);\n    },\n    findJidFromResource: function (resource) {\n        connection.emuc.findJidFromResource(resource);\n    },\n    getMembers: function () {\n        return connection.emuc.members;\n    }\n\n};\n\nmodule.exports = XMPP;","// Copyright Joyent, Inc. and other Node contributors.\n//\n// Permission is hereby granted, free of charge, to any person obtaining a\n// copy of this software and associated documentation files (the\n// \"Software\"), to deal in the Software without restriction, including\n// without limitation the rights to use, copy, modify, merge, publish,\n// distribute, sublicense, and/or sell copies of the Software, and to permit\n// persons to whom the Software is furnished to do so, subject to the\n// following conditions:\n//\n// The above copyright notice and this permission notice shall be included\n// in all copies or substantial portions of the Software.\n//\n// THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS\n// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF\n// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN\n// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,\n// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR\n// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE\n// USE OR OTHER DEALINGS IN THE SOFTWARE.\n\nfunction EventEmitter() {\n  this._events = this._events || {};\n  this._maxListeners = this._maxListeners || undefined;\n}\nmodule.exports = EventEmitter;\n\n// Backwards-compat with node 0.10.x\nEventEmitter.EventEmitter = EventEmitter;\n\nEventEmitter.prototype._events = undefined;\nEventEmitter.prototype._maxListeners = undefined;\n\n// By default EventEmitters will print a warning if more than 10 listeners are\n// added to it. This is a useful default which helps finding memory leaks.\nEventEmitter.defaultMaxListeners = 10;\n\n// Obviously not all Emitters should be limited to 10. This function allows\n// that to be increased. Set to zero for unlimited.\nEventEmitter.prototype.setMaxListeners = function(n) {\n  if (!isNumber(n) || n < 0 || isNaN(n))\n    throw TypeError('n must be a positive number');\n  this._maxListeners = n;\n  return this;\n};\n\nEventEmitter.prototype.emit = function(type) {\n  var er, handler, len, args, i, listeners;\n\n  if (!this._events)\n    this._events = {};\n\n  // If there is no 'error' event listener then throw.\n  if (type === 'error') {\n    if (!this._events.error ||\n        (isObject(this._events.error) && !this._events.error.length)) {\n      er = arguments[1];\n      if (er instanceof Error) {\n        throw er; // Unhandled 'error' event\n      } else {\n        throw TypeError('Uncaught, unspecified \"error\" event.');\n      }\n      return false;\n    }\n  }\n\n  handler = this._events[type];\n\n  if (isUndefined(handler))\n    return false;\n\n  if (isFunction(handler)) {\n    switch (arguments.length) {\n      // fast cases\n      case 1:\n        handler.call(this);\n        break;\n      case 2:\n        handler.call(this, arguments[1]);\n        break;\n      case 3:\n        handler.call(this, arguments[1], arguments[2]);\n        break;\n      // slower\n      default:\n        len = arguments.length;\n        args = new Array(len - 1);\n        for (i = 1; i < len; i++)\n          args[i - 1] = arguments[i];\n        handler.apply(this, args);\n    }\n  } else if (isObject(handler)) {\n    len = arguments.length;\n    args = new Array(len - 1);\n    for (i = 1; i < len; i++)\n      args[i - 1] = arguments[i];\n\n    listeners = handler.slice();\n    len = listeners.length;\n    for (i = 0; i < len; i++)\n      listeners[i].apply(this, args);\n  }\n\n  return true;\n};\n\nEventEmitter.prototype.addListener = function(type, listener) {\n  var m;\n\n  if (!isFunction(listener))\n    throw TypeError('listener must be a function');\n\n  if (!this._events)\n    this._events = {};\n\n  // To avoid recursion in the case that type === \"newListener\"! Before\n  // adding it to the listeners, first emit \"newListener\".\n  if (this._events.newListener)\n    this.emit('newListener', type,\n              isFunction(listener.listener) ?\n              listener.listener : listener);\n\n  if (!this._events[type])\n    // Optimize the case of one listener. Don't need the extra array object.\n    this._events[type] = listener;\n  else if (isObject(this._events[type]))\n    // If we've already got an array, just append.\n    this._events[type].push(listener);\n  else\n    // Adding the second element, need to change to array.\n    this._events[type] = [this._events[type], listener];\n\n  // Check for listener leak\n  if (isObject(this._events[type]) && !this._events[type].warned) {\n    var m;\n    if (!isUndefined(this._maxListeners)) {\n      m = this._maxListeners;\n    } else {\n      m = EventEmitter.defaultMaxListeners;\n    }\n\n    if (m && m > 0 && this._events[type].length > m) {\n      this._events[type].warned = true;\n      console.error('(node) warning: possible EventEmitter memory ' +\n                    'leak detected. %d listeners added. ' +\n                    'Use emitter.setMaxListeners() to increase limit.',\n                    this._events[type].length);\n      if (typeof console.trace === 'function') {\n        // not supported in IE 10\n        console.trace();\n      }\n    }\n  }\n\n  return this;\n};\n\nEventEmitter.prototype.on = EventEmitter.prototype.addListener;\n\nEventEmitter.prototype.once = function(type, listener) {\n  if (!isFunction(listener))\n    throw TypeError('listener must be a function');\n\n  var fired = false;\n\n  function g() {\n    this.removeListener(type, g);\n\n    if (!fired) {\n      fired = true;\n      listener.apply(this, arguments);\n    }\n  }\n\n  g.listener = listener;\n  this.on(type, g);\n\n  return this;\n};\n\n// emits a 'removeListener' event iff the listener was removed\nEventEmitter.prototype.removeListener = function(type, listener) {\n  var list, position, length, i;\n\n  if (!isFunction(listener))\n    throw TypeError('listener must be a function');\n\n  if (!this._events || !this._events[type])\n    return this;\n\n  list = this._events[type];\n  length = list.length;\n  position = -1;\n\n  if (list === listener ||\n      (isFunction(list.listener) && list.listener === listener)) {\n    delete this._events[type];\n    if (this._events.removeListener)\n      this.emit('removeListener', type, listener);\n\n  } else if (isObject(list)) {\n    for (i = length; i-- > 0;) {\n      if (list[i] === listener ||\n          (list[i].listener && list[i].listener === listener)) {\n        position = i;\n        break;\n      }\n    }\n\n    if (position < 0)\n      return this;\n\n    if (list.length === 1) {\n      list.length = 0;\n      delete this._events[type];\n    } else {\n      list.splice(position, 1);\n    }\n\n    if (this._events.removeListener)\n      this.emit('removeListener', type, listener);\n  }\n\n  return this;\n};\n\nEventEmitter.prototype.removeAllListeners = function(type) {\n  var key, listeners;\n\n  if (!this._events)\n    return this;\n\n  // not listening for removeListener, no need to emit\n  if (!this._events.removeListener) {\n    if (arguments.length === 0)\n      this._events = {};\n    else if (this._events[type])\n      delete this._events[type];\n    return this;\n  }\n\n  // emit removeListener for all listeners on all events\n  if (arguments.length === 0) {\n    for (key in this._events) {\n      if (key === 'removeListener') continue;\n      this.removeAllListeners(key);\n    }\n    this.removeAllListeners('removeListener');\n    this._events = {};\n    return this;\n  }\n\n  listeners = this._events[type];\n\n  if (isFunction(listeners)) {\n    this.removeListener(type, listeners);\n  } else {\n    // LIFO order\n    while (listeners.length)\n      this.removeListener(type, listeners[listeners.length - 1]);\n  }\n  delete this._events[type];\n\n  return this;\n};\n\nEventEmitter.prototype.listeners = function(type) {\n  var ret;\n  if (!this._events || !this._events[type])\n    ret = [];\n  else if (isFunction(this._events[type]))\n    ret = [this._events[type]];\n  else\n    ret = this._events[type].slice();\n  return ret;\n};\n\nEventEmitter.listenerCount = function(emitter, type) {\n  var ret;\n  if (!emitter._events || !emitter._events[type])\n    ret = 0;\n  else if (isFunction(emitter._events[type]))\n    ret = 1;\n  else\n    ret = emitter._events[type].length;\n  return ret;\n};\n\nfunction isFunction(arg) {\n  return typeof arg === 'function';\n}\n\nfunction isNumber(arg) {\n  return typeof arg === 'number';\n}\n\nfunction isObject(arg) {\n  return typeof arg === 'object' && arg !== null;\n}\n\nfunction isUndefined(arg) {\n  return arg === void 0;\n}\n"]}