bosh.js 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249
  1. /*
  2. Copyright (c) 2004-2012, The Dojo Foundation All Rights Reserved.
  3. Available via Academic Free License >= 2.1 OR the modified BSD license.
  4. see: http://dojotoolkit.org/license for details
  5. */
  6. if(!dojo._hasResource["dojox.xmpp.bosh"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
  7. dojo._hasResource["dojox.xmpp.bosh"] = true;
  8. dojo.provide("dojox.xmpp.bosh");
  9. dojo.require("dojo.io.script");
  10. dojo.require("dojo.io.iframe");
  11. dojo.require("dojox.xml.parser");
  12. /*=====
  13. dojo.declare("dojox.xmpp.bosh.__initArgs", null, {
  14. constructor: function(){
  15. // summary:
  16. // The arguments passed to dojox.xmpp.bosh.initialize
  17. // iframes:
  18. // The number of iframes to use for transmission
  19. // load:
  20. // The function called when the first iframe is
  21. // loaded. Generally used to signal when to send
  22. // login information
  23. this.iframes = iframes;
  24. this.load = load;
  25. }
  26. });
  27. dojo.declare("dojox.xmpp.bosh.__ioArgs", dojo.__IoArgs, {
  28. constructor: function(){
  29. // summary:
  30. // All the properties described in the dojo.__ioArgs type, apply to this
  31. // type as well, EXCEPT "handleAs". It is not applicable to
  32. // dojox.xmpp.bosh.get() calls, since it is implied that the
  33. // return will be a string of XML.
  34. // rid:
  35. // The rid of the message being sent.
  36. this.rid = rid;
  37. }
  38. });
  39. =====*/
  40. dojox.xmpp.bosh = {
  41. transportIframes: [],
  42. initialize: function(/*dojox.xmpp.bosh.__initArgs*/ args){
  43. this.transportIframes = [];
  44. var scopedObj = dojox._scopeName + '.xmpp.bosh';
  45. var c = dojo.connect(dojo.getObject(scopedObj), '_iframeOnload', this, function(index){
  46. if(index==0){
  47. args.load();
  48. dojo.disconnect(c);
  49. }
  50. });
  51. for(var i = 0; i < args.iframes; i++){
  52. var fname = 'xmpp-transport-'+i;
  53. var iframe = dojo.byId('xmpp-transport-'+i);
  54. if(iframe){
  55. // we have to clean up the dojo.io.iframe references
  56. if(window[fname]){ window[fname] = null; }
  57. if(window.frames[fname]){ window.frames[fname] = null; }
  58. dojo.destroy(iframe);
  59. }
  60. iframe = dojo.io.iframe.create("xmpp-transport-" + i, scopedObj + "._iframeOnload("+i+");" );
  61. this.transportIframes.push(iframe);
  62. }
  63. },
  64. _iframeOnload: function(index){
  65. var doc = dojo.io.iframe.doc(dojo.byId("xmpp-transport-" + index));
  66. doc.write("<script>var isLoaded=true; var rid=0; var transmiting=false; function _BOSH_(msg) { transmiting=false; parent.dojox.xmpp.bosh.handle(msg, rid); } </script>");
  67. },
  68. findOpenIframe: function() {
  69. for(var i = 0; i < this.transportIframes.length; i++) {
  70. var iframe = this.transportIframes[i];
  71. var win = iframe.contentWindow;
  72. //console.log("Open transport?", win, win.isLoaded, win.transmiting);
  73. if(win.isLoaded && !win.transmiting) {
  74. return iframe;
  75. }
  76. }
  77. return false;
  78. },
  79. handle: function(msg, rid){
  80. var dfd = this['rid'+rid];
  81. var xmlMsg = dojox.xml.parser.parse(msg, 'text/xml');
  82. if(xmlMsg){
  83. dfd.ioArgs.xmppMessage = xmlMsg;
  84. }else{
  85. dfd.errback(new Error("Recieved bad document from server: " + msg));
  86. }
  87. },
  88. get: function(/*dojox.xmpp.bosh.__ioArgs*/args){
  89. // summary:
  90. // sends a get request using a dynamically created script tag.
  91. var iframe = this.findOpenIframe();
  92. var iframeDoc = dojo.io.iframe.doc(iframe);
  93. args.frameDoc = iframeDoc;
  94. var dfd = this._makeScriptDeferred(args);
  95. var ioArgs = dfd.ioArgs;
  96. iframe.contentWindow.rid=ioArgs.rid;
  97. iframe.contentWindow.transmiting=true;
  98. dojo._ioAddQueryToUrl(ioArgs);
  99. dojo._ioNotifyStart(dfd);
  100. dojo.io.script.attach(ioArgs.id, ioArgs.url, iframeDoc);
  101. dojo._ioWatch(dfd, this._validCheck, this._ioCheck, this._resHandle);
  102. return dfd;
  103. },
  104. remove: function(/*String*/id, /*Document?*/frameDocument){
  105. //summary: removes the script element with the given id, from the given frameDocument.
  106. //If no frameDocument is passed, the current document is used.
  107. dojo.destroy(dojo.byId(id, frameDocument));
  108. //Remove the BOSH callback on dojox.xmpp.bosh, if it exists.
  109. if(this[id]){
  110. delete this[id];
  111. }
  112. },
  113. _makeScriptDeferred: function(/*Object*/args){
  114. //summary:
  115. // sets up a Deferred object for an IO request.
  116. var dfd = dojo._ioSetArgs(args, this._deferredCancel, this._deferredOk, this._deferredError);
  117. var ioArgs = dfd.ioArgs;
  118. ioArgs.id = 'rid' + args.rid;
  119. ioArgs.rid = args.rid;
  120. ioArgs.canDelete = true;
  121. ioArgs.frameDoc = args.frameDoc;
  122. this[ioArgs.id] = dfd;
  123. return dfd; // dojo.Deferred
  124. },
  125. _deferredCancel: function(/*Deferred*/dfd){
  126. //summary: canceller function for dojo._ioSetArgs call.
  127. //DO NOT use "this" and expect it to be dojox.xmpp.bosh.
  128. dfd.canceled = true;
  129. if(dfd.ioArgs.canDelete){
  130. dojox.xmpp.bosh._addDeadScript(dfd.ioArgs);
  131. }
  132. },
  133. _deferredOk: function(/*Deferred*/dfd){
  134. //summary: okHandler function for dojo._ioSetArgs call.
  135. //DO NOT use "this" and expect it to be dojo.xmpp.bosh.
  136. var ioArgs = dfd.ioArgs;
  137. //Add script to list of things that can be removed.
  138. if(ioArgs.canDelete){
  139. dojox.xmpp.bosh._addDeadScript(ioArgs);
  140. }
  141. //Favor JSONP responses, script load events then lastly ioArgs.
  142. //The ioArgs are goofy, but cannot return the dfd since that stops
  143. //the callback chain in Deferred. The return value is not that important
  144. //in that case, probably a checkString case.
  145. return ioArgs.xmppMessage || ioArgs;
  146. },
  147. _deferredError: function(/*Error*/error, /*Deferred*/dfd){
  148. //summary: errHandler function for dojo._ioSetArgs call.
  149. if(dfd.ioArgs.canDelete){
  150. //DO NOT use "this" and expect it to be dojox.xmpp.bosh
  151. if(error.dojoType == "timeout"){
  152. //For timeouts, remove the script element immediately to
  153. //avoid a response from it coming back later and causing trouble.
  154. dojox.xmpp.bosh.remove(dfd.ioArgs.id, dfd.ioArgs.frameDoc);
  155. }else{
  156. dojox.xmpp.bosh._addDeadScript(dfd.ioArgs);
  157. }
  158. }
  159. return error;
  160. },
  161. _deadScripts: [],
  162. _addDeadScript: function(/*Object*/ioArgs){
  163. //summary: sets up an entry in the deadScripts array.
  164. dojox.xmpp.bosh._deadScripts.push({id: ioArgs.id, frameDoc: ioArgs.frameDoc});
  165. //Being extra paranoid about leaks:
  166. ioArgs.frameDoc = null;
  167. },
  168. _validCheck: function(/*Deferred*/dfd){
  169. //summary: inflight check function to see if dfd is still valid.
  170. //Do script cleanup here. We wait for one inflight pass
  171. //to make sure we don't get any weird things by trying to remove a script
  172. //tag that is part of the call chain (IE 6 has been known to
  173. //crash in that case).
  174. var _self = dojox.xmpp.bosh;
  175. var deadScripts = _self._deadScripts;
  176. if(deadScripts && deadScripts.length > 0){
  177. for(var i = 0; i < deadScripts.length; i++){
  178. //Remove the script tag
  179. _self.remove(deadScripts[i].id, deadScripts[i].frameDoc);
  180. deadScripts[i].frameDoc = null;
  181. }
  182. dojox.xmpp.bosh._deadScripts = [];
  183. }
  184. return true;
  185. },
  186. _ioCheck: function(/*Deferred*/dfd){
  187. //summary: inflight check function to see if IO finished.
  188. var ioArgs = dfd.ioArgs;
  189. //Check for returned message
  190. if(ioArgs.xmppMessage){
  191. return true;
  192. }
  193. return false;
  194. },
  195. _resHandle: function(/*Deferred*/dfd){
  196. //summary: inflight function to handle a completed response.
  197. if(dojox.xmpp.bosh._ioCheck(dfd)){
  198. dfd.callback(dfd);
  199. }else{
  200. //This path should never happen since the only way we can get
  201. //to _resHandle is if _ioCheck is true.
  202. dfd.errback(new Error("inconceivable dojox.xmpp.bosh._resHandle error"));
  203. }
  204. }
  205. };
  206. }