2
0

mam.js 36 KB

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