mam.js 37 KB

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