mam.js 32 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576
  1. (function (root, factory) {
  2. define(["jasmine", "mock", "converse-core", "test-utils"], factory);
  3. } (this, function (jasmine, mock, converse, test_utils) {
  4. "use strict";
  5. var _ = converse.env._;
  6. var Backbone = converse.env.Backbone;
  7. var Strophe = converse.env.Strophe;
  8. var $iq = converse.env.$iq;
  9. var $msg = converse.env.$msg;
  10. var moment = converse.env.moment;
  11. // See: https://xmpp.org/rfcs/rfc3921.html
  12. describe("Message Archive Management", function () {
  13. // Implement the protocol defined in https://xmpp.org/extensions/xep-0313.html#config
  14. describe("Archived Messages", function () {
  15. it("aren't shown as duplicates",
  16. mock.initConverseWithPromises(
  17. null, ['discoInitialized'], {},
  18. function (done, _converse) {
  19. test_utils.openAndEnterChatRoom(_converse, 'trek-radio', 'conference.lightwitch.org', 'jcbrand').then(function () {
  20. var chatroomview = _converse.chatboxviews.get('trek-radio@conference.lightwitch.org');
  21. var stanza = Strophe.xmlHtmlNode(
  22. `<message xmlns="jabber:client" to="jcbrand@lightwitch.org/converse.js-73057452" type="groupchat" from="trek-radio@conference.lightwitch.org/comndrdukath#0805 (STO)">
  23. <body>negan</body>
  24. <stanza-id xmlns="urn:xmpp:sid:0" id="45fbbf2a-1059-479d-9283-c8effaf05621" by="trek-radio@conference.lightwitch.org"/>
  25. </message>`).firstElementChild;
  26. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  27. stanza = Strophe.xmlHtmlNode(
  28. `<message xmlns="jabber:client" to="jcbrand@lightwitch.org/converse.js-73057452">
  29. <result xmlns="urn:xmpp:mam:2" queryid="82d9db27-6cf8-4787-8c2c-5a560263d823" id="45fbbf2a-1059-479d-9283-c8effaf05621">
  30. <forwarded xmlns="urn:xmpp:forward:0"><delay xmlns="urn:xmpp:delay" stamp="2018-01-09T06:17:23Z"/>
  31. <message from="trek-radio@conference.lightwitch.org/comndrdukath#0805 (STO)" type="groupchat">
  32. <body>negan</body>
  33. </message>
  34. </forwarded>
  35. </result>
  36. </message>`).firstElementChild;
  37. chatroomview.onChatRoomMessage(stanza);
  38. expect(chatroomview.content.querySelectorAll('.chat-message').length).toBe(1);
  39. done();
  40. });
  41. }))
  42. });
  43. describe("The archive.query API", function () {
  44. it("can be used to query for all archived messages",
  45. mock.initConverseWithPromises(
  46. null, ['discoInitialized'], {},
  47. function (done, _converse) {
  48. var sent_stanza, IQ_id;
  49. var sendIQ = _converse.connection.sendIQ;
  50. spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
  51. sent_stanza = iq;
  52. IQ_id = sendIQ.bind(this)(iq, callback, errback);
  53. });
  54. if (!_converse.disco_entities.get(_converse.domain).features.findWhere({'var': Strophe.NS.MAM})) {
  55. _converse.disco_entities.get(_converse.domain).features.create({'var': Strophe.NS.MAM});
  56. }
  57. _converse.api.archive.query();
  58. var queryid = sent_stanza.nodeTree.querySelector('query').getAttribute('queryid');
  59. expect(sent_stanza.toString()).toBe(
  60. "<iq type='set' xmlns='jabber:client' id='"+IQ_id+"'><query xmlns='urn:xmpp:mam:2' queryid='"+queryid+"'/></iq>");
  61. done();
  62. }));
  63. it("can be used to query for all messages to/from a particular JID", mock.initConverse(function (_converse) {
  64. var sent_stanza, IQ_id;
  65. var sendIQ = _converse.connection.sendIQ;
  66. spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
  67. sent_stanza = iq;
  68. IQ_id = sendIQ.bind(this)(iq, callback, errback);
  69. });
  70. if (!_converse.disco_entities.get(_converse.domain).features.findWhere({'var': Strophe.NS.MAM})) {
  71. _converse.disco_entities.get(_converse.domain).features.create({'var': Strophe.NS.MAM});
  72. }
  73. _converse.api.archive.query({'with':'juliet@capulet.lit'});
  74. var queryid = sent_stanza.nodeTree.querySelector('query').getAttribute('queryid');
  75. expect(sent_stanza.toString()).toBe(
  76. "<iq type='set' xmlns='jabber:client' id='"+IQ_id+"'>"+
  77. "<query xmlns='urn:xmpp:mam:2' queryid='"+queryid+"'>"+
  78. "<x xmlns='jabber:x:data' type='submit'>"+
  79. "<field var='FORM_TYPE' type='hidden'>"+
  80. "<value>urn:xmpp:mam:2</value>"+
  81. "</field>"+
  82. "<field var='with'>"+
  83. "<value>juliet@capulet.lit</value>"+
  84. "</field>"+
  85. "</x>"+
  86. "</query>"+
  87. "</iq>"
  88. );
  89. }));
  90. it("can be used to query for archived messages from a chat room", mock.initConverse(function (_converse) {
  91. if (!_converse.disco_entities.get(_converse.domain).features.findWhere({'var': Strophe.NS.MAM})) {
  92. _converse.disco_entities.get(_converse.domain).features.create({'var': Strophe.NS.MAM});
  93. }
  94. var sent_stanza, IQ_id;
  95. var sendIQ = _converse.connection.sendIQ;
  96. spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
  97. sent_stanza = iq;
  98. IQ_id = sendIQ.bind(this)(iq, callback, errback);
  99. });
  100. var callback = jasmine.createSpy('callback');
  101. _converse.api.archive.query({'with': 'coven@chat.shakespeare.lit', 'groupchat': true}, callback);
  102. var queryid = sent_stanza.nodeTree.querySelector('query').getAttribute('queryid');
  103. expect(sent_stanza.toString()).toBe(
  104. "<iq type='set' to='coven@chat.shakespeare.lit' xmlns='jabber:client' id='"+IQ_id+"'>"+
  105. "<query xmlns='urn:xmpp:mam:2' queryid='"+queryid+"'>"+
  106. "<x xmlns='jabber:x:data' type='submit'>"+
  107. "<field var='FORM_TYPE' type='hidden'>"+
  108. "<value>urn:xmpp:mam:2</value>"+
  109. "</field>"+
  110. "</x>"+
  111. "</query>"+
  112. "</iq>");
  113. }));
  114. it("checks whether returned MAM messages from a MUC room are from the right JID", mock.initConverse(function (_converse) {
  115. if (!_converse.disco_entities.get(_converse.domain).features.findWhere({'var': Strophe.NS.MAM})) {
  116. _converse.disco_entities.get(_converse.domain).features.create({'var': Strophe.NS.MAM});
  117. }
  118. var sent_stanza, IQ_id;
  119. var sendIQ = _converse.connection.sendIQ;
  120. spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
  121. sent_stanza = iq;
  122. IQ_id = sendIQ.bind(this)(iq, callback, errback);
  123. });
  124. var callback = jasmine.createSpy('callback');
  125. _converse.api.archive.query({'with': 'coven@chat.shakespear.lit', 'groupchat': true, 'max':'10'}, callback);
  126. var queryid = sent_stanza.nodeTree.querySelector('query').getAttribute('queryid');
  127. /* <message id='iasd207' from='coven@chat.shakespeare.lit' to='hag66@shakespeare.lit/pda'>
  128. * <result xmlns='urn:xmpp:mam:2' queryid='g27' id='34482-21985-73620'>
  129. * <forwarded xmlns='urn:xmpp:forward:0'>
  130. * <delay xmlns='urn:xmpp:delay' stamp='2002-10-13T23:58:37Z'/>
  131. * <message xmlns="jabber:client"
  132. * from='coven@chat.shakespeare.lit/firstwitch'
  133. * id='162BEBB1-F6DB-4D9A-9BD8-CFDCC801A0B2'
  134. * type='groupchat'>
  135. * <body>Thrice the brinded cat hath mew'd.</body>
  136. * <x xmlns='http://jabber.org/protocol/muc#user'>
  137. * <item affiliation='none'
  138. * jid='witch1@shakespeare.lit'
  139. * role='participant' />
  140. * </x>
  141. * </message>
  142. * </forwarded>
  143. * </result>
  144. * </message>
  145. */
  146. var msg1 = $msg({'id':'iasd207', 'from': 'other@chat.shakespear.lit', 'to': 'dummy@localhost'})
  147. .c('result', {'xmlns': 'urn:xmpp:mam:2', 'queryid':queryid, 'id':'34482-21985-73620'})
  148. .c('forwarded', {'xmlns':'urn:xmpp:forward:0'})
  149. .c('delay', {'xmlns':'urn:xmpp:delay', 'stamp':'2010-07-10T23:08:25Z'}).up()
  150. .c('message', {
  151. 'xmlns':'jabber:client',
  152. 'to':'dummy@localhost',
  153. 'id':'162BEBB1-F6DB-4D9A-9BD8-CFDCC801A0B2',
  154. 'from':'coven@chat.shakespeare.lit/firstwitch',
  155. 'type':'groupchat' })
  156. .c('body').t("Thrice the brinded cat hath mew'd.");
  157. _converse.connection._dataRecv(test_utils.createRequest(msg1));
  158. /* Send an <iq> stanza to indicate the end of the result set.
  159. *
  160. * <iq type='result' id='juliet1'>
  161. * <fin xmlns='urn:xmpp:mam:2'>
  162. * <set xmlns='http://jabber.org/protocol/rsm'>
  163. * <first index='0'>28482-98726-73623</first>
  164. * <last>09af3-cc343-b409f</last>
  165. * <count>20</count>
  166. * </set>
  167. * </iq>
  168. */
  169. var stanza = $iq({'type': 'result', 'id': IQ_id})
  170. .c('fin', {'xmlns': 'urn:xmpp:mam:2'})
  171. .c('set', {'xmlns': 'http://jabber.org/protocol/rsm'})
  172. .c('first', {'index': '0'}).t('23452-4534-1').up()
  173. .c('last').t('09af3-cc343-b409f').up()
  174. .c('count').t('16');
  175. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  176. expect(callback).toHaveBeenCalled();
  177. var args = callback.calls.argsFor(0);
  178. expect(args[0].length).toBe(0);
  179. }));
  180. it("can be used to query for all messages in a certain timespan", mock.initConverse(function (_converse) {
  181. var sent_stanza, IQ_id;
  182. var sendIQ = _converse.connection.sendIQ;
  183. spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
  184. sent_stanza = iq;
  185. IQ_id = sendIQ.bind(this)(iq, callback, errback);
  186. });
  187. if (!_converse.disco_entities.get(_converse.domain).features.findWhere({'var': Strophe.NS.MAM})) {
  188. _converse.disco_entities.get(_converse.domain).features.create({'var': Strophe.NS.MAM});
  189. }
  190. var start = '2010-06-07T00:00:00Z';
  191. var end = '2010-07-07T13:23:54Z';
  192. _converse.api.archive.query({
  193. 'start': start,
  194. 'end': end
  195. });
  196. var queryid = sent_stanza.nodeTree.querySelector('query').getAttribute('queryid');
  197. expect(sent_stanza.toString()).toBe(
  198. "<iq type='set' xmlns='jabber:client' id='"+IQ_id+"'>"+
  199. "<query xmlns='urn:xmpp:mam:2' queryid='"+queryid+"'>"+
  200. "<x xmlns='jabber:x:data' type='submit'>"+
  201. "<field var='FORM_TYPE' type='hidden'>"+
  202. "<value>urn:xmpp:mam:2</value>"+
  203. "</field>"+
  204. "<field var='start'>"+
  205. "<value>"+moment(start).format()+"</value>"+
  206. "</field>"+
  207. "<field var='end'>"+
  208. "<value>"+moment(end).format()+"</value>"+
  209. "</field>"+
  210. "</x>"+
  211. "</query>"+
  212. "</iq>"
  213. );
  214. }));
  215. it("throws a TypeError if an invalid date is provided", mock.initConverse(function (_converse) {
  216. if (!_converse.disco_entities.get(_converse.domain).features.findWhere({'var': Strophe.NS.MAM})) {
  217. _converse.disco_entities.get(_converse.domain).features.create({'var': Strophe.NS.MAM});
  218. }
  219. expect(_.partial(_converse.api.archive.query, {'start': 'not a real date'})).toThrow(
  220. new TypeError('archive.query: invalid date provided for: start')
  221. );
  222. }));
  223. it("can be used to query for all messages after a certain time", mock.initConverse(function (_converse) {
  224. var sent_stanza, IQ_id;
  225. var sendIQ = _converse.connection.sendIQ;
  226. spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
  227. sent_stanza = iq;
  228. IQ_id = sendIQ.bind(this)(iq, callback, errback);
  229. });
  230. if (!_converse.disco_entities.get(_converse.domain).features.findWhere({'var': Strophe.NS.MAM})) {
  231. _converse.disco_entities.get(_converse.domain).features.create({'var': Strophe.NS.MAM});
  232. }
  233. var start = '2010-06-07T00:00:00Z';
  234. _converse.api.archive.query({'start': start});
  235. var queryid = sent_stanza.nodeTree.querySelector('query').getAttribute('queryid');
  236. expect(sent_stanza.toString()).toBe(
  237. "<iq type='set' xmlns='jabber:client' id='"+IQ_id+"'>"+
  238. "<query xmlns='urn:xmpp:mam:2' queryid='"+queryid+"'>"+
  239. "<x xmlns='jabber:x:data' type='submit'>"+
  240. "<field var='FORM_TYPE' type='hidden'>"+
  241. "<value>urn:xmpp:mam:2</value>"+
  242. "</field>"+
  243. "<field var='start'>"+
  244. "<value>"+moment(start).format()+"</value>"+
  245. "</field>"+
  246. "</x>"+
  247. "</query>"+
  248. "</iq>"
  249. );
  250. }));
  251. it("can be used to query for a limited set of results", mock.initConverse(function (_converse) {
  252. var sent_stanza, IQ_id;
  253. var sendIQ = _converse.connection.sendIQ;
  254. spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
  255. sent_stanza = iq;
  256. IQ_id = sendIQ.bind(this)(iq, callback, errback);
  257. });
  258. if (!_converse.disco_entities.get(_converse.domain).features.findWhere({'var': Strophe.NS.MAM})) {
  259. _converse.disco_entities.get(_converse.domain).features.create({'var': Strophe.NS.MAM});
  260. }
  261. var start = '2010-06-07T00:00:00Z';
  262. _converse.api.archive.query({'start': start, 'max':10});
  263. var queryid = sent_stanza.nodeTree.querySelector('query').getAttribute('queryid');
  264. expect(sent_stanza.toString()).toBe(
  265. "<iq type='set' xmlns='jabber:client' id='"+IQ_id+"'>"+
  266. "<query xmlns='urn:xmpp:mam:2' queryid='"+queryid+"'>"+
  267. "<x xmlns='jabber:x:data' type='submit'>"+
  268. "<field var='FORM_TYPE' type='hidden'>"+
  269. "<value>urn:xmpp:mam:2</value>"+
  270. "</field>"+
  271. "<field var='start'>"+
  272. "<value>"+moment(start).format()+"</value>"+
  273. "</field>"+
  274. "</x>"+
  275. "<set xmlns='http://jabber.org/protocol/rsm'>"+
  276. "<max>10</max>"+
  277. "</set>"+
  278. "</query>"+
  279. "</iq>"
  280. );
  281. }));
  282. it("can be used to page through results", mock.initConverse(function (_converse) {
  283. var sent_stanza, IQ_id;
  284. var sendIQ = _converse.connection.sendIQ;
  285. spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
  286. sent_stanza = iq;
  287. IQ_id = sendIQ.bind(this)(iq, callback, errback);
  288. });
  289. if (!_converse.disco_entities.get(_converse.domain).features.findWhere({'var': Strophe.NS.MAM})) {
  290. _converse.disco_entities.get(_converse.domain).features.create({'var': Strophe.NS.MAM});
  291. }
  292. var start = '2010-06-07T00:00:00Z';
  293. _converse.api.archive.query({
  294. 'start': start,
  295. 'after': '09af3-cc343-b409f',
  296. 'max':10
  297. });
  298. var queryid = sent_stanza.nodeTree.querySelector('query').getAttribute('queryid');
  299. expect(sent_stanza.toString()).toBe(
  300. "<iq type='set' xmlns='jabber:client' id='"+IQ_id+"'>"+
  301. "<query xmlns='urn:xmpp:mam:2' queryid='"+queryid+"'>"+
  302. "<x xmlns='jabber:x:data' type='submit'>"+
  303. "<field var='FORM_TYPE' type='hidden'>"+
  304. "<value>urn:xmpp:mam:2</value>"+
  305. "</field>"+
  306. "<field var='start'>"+
  307. "<value>"+moment(start).format()+"</value>"+
  308. "</field>"+
  309. "</x>"+
  310. "<set xmlns='http://jabber.org/protocol/rsm'>"+
  311. "<max>10</max>"+
  312. "<after>09af3-cc343-b409f</after>"+
  313. "</set>"+
  314. "</query>"+
  315. "</iq>"
  316. );
  317. }));
  318. it("accepts \"before\" with an empty string as value to reverse the order", mock.initConverse(function (_converse) {
  319. var sent_stanza, IQ_id;
  320. var sendIQ = _converse.connection.sendIQ;
  321. spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
  322. sent_stanza = iq;
  323. IQ_id = sendIQ.bind(this)(iq, callback, errback);
  324. });
  325. if (!_converse.disco_entities.get(_converse.domain).features.findWhere({'var': Strophe.NS.MAM})) {
  326. _converse.disco_entities.get(_converse.domain).features.create({'var': Strophe.NS.MAM});
  327. }
  328. _converse.api.archive.query({'before': '', 'max':10});
  329. var queryid = sent_stanza.nodeTree.querySelector('query').getAttribute('queryid');
  330. expect(sent_stanza.toString()).toBe(
  331. "<iq type='set' xmlns='jabber:client' id='"+IQ_id+"'>"+
  332. "<query xmlns='urn:xmpp:mam:2' queryid='"+queryid+"'>"+
  333. "<x xmlns='jabber:x:data' type='submit'>"+
  334. "<field var='FORM_TYPE' type='hidden'>"+
  335. "<value>urn:xmpp:mam:2</value>"+
  336. "</field>"+
  337. "</x>"+
  338. "<set xmlns='http://jabber.org/protocol/rsm'>"+
  339. "<max>10</max>"+
  340. "<before></before>"+
  341. "</set>"+
  342. "</query>"+
  343. "</iq>"
  344. );
  345. }));
  346. it("accepts a Strophe.RSM object for the query options", mock.initConverse(function (_converse) {
  347. // Normally the user wouldn't manually make a Strophe.RSM object
  348. // and pass it in. However, in the callback method an RSM object is
  349. // returned which can be reused for easy paging. This test is
  350. // more for that usecase.
  351. if (!_converse.disco_entities.get(_converse.domain).features.findWhere({'var': Strophe.NS.MAM})) {
  352. _converse.disco_entities.get(_converse.domain).features.create({'var': Strophe.NS.MAM});
  353. }
  354. var sent_stanza, IQ_id;
  355. var sendIQ = _converse.connection.sendIQ;
  356. spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
  357. sent_stanza = iq;
  358. IQ_id = sendIQ.bind(this)(iq, callback, errback);
  359. });
  360. var rsm = new Strophe.RSM({'max': '10'});
  361. rsm['with'] = 'romeo@montague.lit'; // eslint-disable-line dot-notation
  362. rsm.start = '2010-06-07T00:00:00Z';
  363. _converse.api.archive.query(rsm);
  364. var queryid = sent_stanza.nodeTree.querySelector('query').getAttribute('queryid');
  365. expect(sent_stanza.toString()).toBe(
  366. "<iq type='set' xmlns='jabber:client' id='"+IQ_id+"'>"+
  367. "<query xmlns='urn:xmpp:mam:2' queryid='"+queryid+"'>"+
  368. "<x xmlns='jabber:x:data' type='submit'>"+
  369. "<field var='FORM_TYPE' type='hidden'>"+
  370. "<value>urn:xmpp:mam:2</value>"+
  371. "</field>"+
  372. "<field var='with'>"+
  373. "<value>romeo@montague.lit</value>"+
  374. "</field>"+
  375. "<field var='start'>"+
  376. "<value>"+moment(rsm.start).format()+"</value>"+
  377. "</field>"+
  378. "</x>"+
  379. "<set xmlns='http://jabber.org/protocol/rsm'>"+
  380. "<max>10</max>"+
  381. "</set>"+
  382. "</query>"+
  383. "</iq>"
  384. );
  385. }));
  386. it("accepts a callback function, which it passes the messages and a Strophe.RSM object", mock.initConverse(function (_converse) {
  387. if (!_converse.disco_entities.get(_converse.domain).features.findWhere({'var': Strophe.NS.MAM})) {
  388. _converse.disco_entities.get(_converse.domain).features.create({'var': Strophe.NS.MAM});
  389. }
  390. var sent_stanza, IQ_id;
  391. var sendIQ = _converse.connection.sendIQ;
  392. spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
  393. sent_stanza = iq;
  394. IQ_id = sendIQ.bind(this)(iq, callback, errback);
  395. });
  396. var callback = jasmine.createSpy('callback');
  397. _converse.api.archive.query({'with': 'romeo@capulet.lit', 'max':'10'}, callback);
  398. var queryid = sent_stanza.nodeTree.querySelector('query').getAttribute('queryid');
  399. /* <message id='aeb213' to='juliet@capulet.lit/chamber'>
  400. * <result xmlns='urn:xmpp:mam:2' queryid='f27' id='28482-98726-73623'>
  401. * <forwarded xmlns='urn:xmpp:forward:0'>
  402. * <delay xmlns='urn:xmpp:delay' stamp='2010-07-10T23:08:25Z'/>
  403. * <message xmlns='jabber:client'
  404. * to='juliet@capulet.lit/balcony'
  405. * from='romeo@montague.lit/orchard'
  406. * type='chat'>
  407. * <body>Call me but love, and I'll be new baptized; Henceforth I never will be Romeo.</body>
  408. * </message>
  409. * </forwarded>
  410. * </result>
  411. * </message>
  412. */
  413. var msg1 = $msg({'id':'aeb213', 'to':'juliet@capulet.lit/chamber'})
  414. .c('result', {'xmlns': 'urn:xmpp:mam:2', 'queryid':queryid, 'id':'28482-98726-73623'})
  415. .c('forwarded', {'xmlns':'urn:xmpp:forward:0'})
  416. .c('delay', {'xmlns':'urn:xmpp:delay', 'stamp':'2010-07-10T23:08:25Z'}).up()
  417. .c('message', {
  418. 'xmlns':'jabber:client',
  419. 'to':'juliet@capulet.lit/balcony',
  420. 'from':'romeo@montague.lit/orchard',
  421. 'type':'chat' })
  422. .c('body').t("Call me but love, and I'll be new baptized;");
  423. _converse.connection._dataRecv(test_utils.createRequest(msg1));
  424. var msg2 = $msg({'id':'aeb213', 'to':'juliet@capulet.lit/chamber'})
  425. .c('result', {'xmlns': 'urn:xmpp:mam:2', 'queryid':queryid, 'id':'28482-98726-73624'})
  426. .c('forwarded', {'xmlns':'urn:xmpp:forward:0'})
  427. .c('delay', {'xmlns':'urn:xmpp:delay', 'stamp':'2010-07-10T23:08:25Z'}).up()
  428. .c('message', {
  429. 'xmlns':'jabber:client',
  430. 'to':'juliet@capulet.lit/balcony',
  431. 'from':'romeo@montague.lit/orchard',
  432. 'type':'chat' })
  433. .c('body').t("Henceforth I never will be Romeo.");
  434. _converse.connection._dataRecv(test_utils.createRequest(msg2));
  435. /* Send an <iq> stanza to indicate the end of the result set.
  436. *
  437. * <iq type='result' id='juliet1'>
  438. * <fin xmlns='urn:xmpp:mam:2'>
  439. * <set xmlns='http://jabber.org/protocol/rsm'>
  440. * <first index='0'>28482-98726-73623</first>
  441. * <last>09af3-cc343-b409f</last>
  442. * <count>20</count>
  443. * </set>
  444. * </iq>
  445. */
  446. var stanza = $iq({'type': 'result', 'id': IQ_id})
  447. .c('fin', {'xmlns': 'urn:xmpp:mam:2'})
  448. .c('set', {'xmlns': 'http://jabber.org/protocol/rsm'})
  449. .c('first', {'index': '0'}).t('23452-4534-1').up()
  450. .c('last').t('09af3-cc343-b409f').up()
  451. .c('count').t('16');
  452. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  453. expect(callback).toHaveBeenCalled();
  454. var args = callback.calls.argsFor(0);
  455. expect(args[0].length).toBe(2);
  456. expect(args[0][0].outerHTML).toBe(msg1.nodeTree.outerHTML);
  457. expect(args[0][1].outerHTML).toBe(msg2.nodeTree.outerHTML);
  458. expect(args[1]['with']).toBe('romeo@capulet.lit'); // eslint-disable-line dot-notation
  459. expect(args[1].max).toBe('10');
  460. expect(args[1].count).toBe('16');
  461. expect(args[1].first).toBe('23452-4534-1');
  462. expect(args[1].last).toBe('09af3-cc343-b409f');
  463. }));
  464. });
  465. describe("The default preference", function () {
  466. it("is set once server support for MAM has been confirmed", mock.initConverse(function (_converse) {
  467. var sent_stanza, IQ_id;
  468. var sendIQ = _converse.connection.sendIQ;
  469. spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
  470. sent_stanza = iq;
  471. IQ_id = sendIQ.bind(this)(iq, callback, errback);
  472. });
  473. spyOn(_converse, 'onMAMPreferences').and.callThrough();
  474. _converse.message_archiving = 'never';
  475. var feature = new Backbone.Model({
  476. 'var': Strophe.NS.MAM
  477. });
  478. spyOn(feature, 'save').and.callFake(feature.set); // Save will complain about a url not being set
  479. _converse.disco_entities.get(_converse.domain).onFeatureAdded(feature);
  480. expect(_converse.connection.sendIQ).toHaveBeenCalled();
  481. expect(sent_stanza.toLocaleString()).toBe(
  482. "<iq type='get' xmlns='jabber:client' id='"+IQ_id+"'>"+
  483. "<prefs xmlns='urn:xmpp:mam:2'/>"+
  484. "</iq>"
  485. );
  486. /* Example 20. Server responds with current preferences
  487. *
  488. * <iq type='result' id='juliet2'>
  489. * <prefs xmlns='urn:xmpp:mam:0' default='roster'>
  490. * <always/>
  491. * <never/>
  492. * </prefs>
  493. * </iq>
  494. */
  495. var stanza = $iq({'type': 'result', 'id': IQ_id})
  496. .c('prefs', {'xmlns': Strophe.NS.MAM, 'default':'roster'})
  497. .c('always').c('jid').t('romeo@montague.lit').up().up()
  498. .c('never').c('jid').t('montague@montague.lit');
  499. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  500. expect(_converse.onMAMPreferences).toHaveBeenCalled();
  501. expect(_converse.connection.sendIQ.calls.count()).toBe(2);
  502. expect(sent_stanza.toString()).toBe(
  503. "<iq type='set' xmlns='jabber:client' id='"+IQ_id+"'>"+
  504. "<prefs xmlns='urn:xmpp:mam:2' default='never'>"+
  505. "<always><jid>romeo@montague.lit</jid></always>"+
  506. "<never><jid>montague@montague.lit</jid></never>"+
  507. "</prefs>"+
  508. "</iq>"
  509. );
  510. expect(feature.get('preference')).toBe(undefined);
  511. /* <iq type='result' id='juliet3'>
  512. * <prefs xmlns='urn:xmpp:mam:0' default='always'>
  513. * <always>
  514. * <jid>romeo@montague.lit</jid>
  515. * </always>
  516. * <never>
  517. * <jid>montague@montague.lit</jid>
  518. * </never>
  519. * </prefs>
  520. * </iq>
  521. */
  522. stanza = $iq({'type': 'result', 'id': IQ_id})
  523. .c('prefs', {'xmlns': Strophe.NS.MAM, 'default':'always'})
  524. .c('always').up()
  525. .c('never').up();
  526. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  527. expect(feature.save).toHaveBeenCalled();
  528. expect(feature.get('preferences')['default']).toBe('never'); // eslint-disable-line dot-notation
  529. // Restore
  530. _converse.message_archiving = 'never';
  531. }));
  532. });
  533. });
  534. }));