mam.js 65 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101
  1. (function (root, factory) {
  2. define(["jasmine", "mock", "test-utils"], factory);
  3. } (this, function (jasmine, mock, test_utils) {
  4. "use strict";
  5. const Model = converse.env.Model;
  6. const Strophe = converse.env.Strophe;
  7. const $iq = converse.env.$iq;
  8. const $msg = converse.env.$msg;
  9. const dayjs = converse.env.dayjs;
  10. const u = converse.env.utils;
  11. const sizzle = converse.env.sizzle;
  12. // See: https://xmpp.org/rfcs/rfc3921.html
  13. // Implements the protocol defined in https://xmpp.org/extensions/xep-0313.html#config
  14. describe("Message Archive Management", function () {
  15. describe("The XEP-0313 Archive", function () {
  16. it("is queried when the user enters a new MUC",
  17. mock.initConverse(['discoInitialized'], {'archived_messages_page_size': 2}, async function (done, _converse) {
  18. const sent_IQs = _converse.connection.IQ_stanzas;
  19. const muc_jid = 'orchard@chat.shakespeare.lit';
  20. await test_utils.openAndEnterChatRoom(_converse, muc_jid, 'romeo');
  21. let view = _converse.chatboxviews.get(muc_jid);
  22. let iq_get = await u.waitUntil(() => sent_IQs.filter(iq => iq.querySelector(`iq query[xmlns="${Strophe.NS.MAM}"]`)).pop());
  23. expect(Strophe.serialize(iq_get)).toBe(
  24. `<iq id="${iq_get.getAttribute('id')}" to="${muc_jid}" type="set" xmlns="jabber:client">`+
  25. `<query queryid="${iq_get.querySelector('query').getAttribute('queryid')}" xmlns="${Strophe.NS.MAM}">`+
  26. `<x type="submit" xmlns="jabber:x:data">`+
  27. `<field type="hidden" var="FORM_TYPE"><value>urn:xmpp:mam:2</value></field>`+
  28. `</x>`+
  29. `<set xmlns="http://jabber.org/protocol/rsm"><max>2</max><before></before></set>`+
  30. `</query>`+
  31. `</iq>`);
  32. let first_msg_id = _converse.connection.getUniqueId();
  33. let last_msg_id = _converse.connection.getUniqueId();
  34. let message = u.toStanza(
  35. `<message xmlns="jabber:client"
  36. to="romeo@montague.lit/orchard"
  37. from="${muc_jid}">
  38. <result xmlns="urn:xmpp:mam:2" queryid="${iq_get.querySelector('query').getAttribute('queryid')}" id="${first_msg_id}">
  39. <forwarded xmlns="urn:xmpp:forward:0">
  40. <delay xmlns="urn:xmpp:delay" stamp="2018-01-09T06:15:23Z"/>
  41. <message from="${muc_jid}/some1" type="groupchat">
  42. <body>2nd Message</body>
  43. </message>
  44. </forwarded>
  45. </result>
  46. </message>`);
  47. _converse.connection._dataRecv(test_utils.createRequest(message));
  48. message = u.toStanza(
  49. `<message xmlns="jabber:client"
  50. to="romeo@montague.lit/orchard"
  51. from="${muc_jid}">
  52. <result xmlns="urn:xmpp:mam:2" queryid="${iq_get.querySelector('query').getAttribute('queryid')}" id="${last_msg_id}">
  53. <forwarded xmlns="urn:xmpp:forward:0">
  54. <delay xmlns="urn:xmpp:delay" stamp="2018-01-09T06:16:23Z"/>
  55. <message from="${muc_jid}/some1" type="groupchat">
  56. <body>3rd Message</body>
  57. </message>
  58. </forwarded>
  59. </result>
  60. </message>`);
  61. _converse.connection._dataRecv(test_utils.createRequest(message));
  62. // Clear so that we don't match the older query
  63. while (sent_IQs.length) { sent_IQs.pop(); }
  64. // XXX: Even though the count is 3, when fetching messages for
  65. // the first time, we don't paginate, so that message
  66. // is not fetched. The user needs to manually load older
  67. // messages for it to be fetched.
  68. // TODO: we need to add a clickable link to load older messages
  69. let result = u.toStanza(
  70. `<iq type='result' id='${iq_get.getAttribute('id')}'>
  71. <fin xmlns='urn:xmpp:mam:2'>
  72. <set xmlns='http://jabber.org/protocol/rsm'>
  73. <first index='0'>${first_msg_id}</first>
  74. <last>${last_msg_id}</last>
  75. <count>3</count>
  76. </set>
  77. </fin>
  78. </iq>`);
  79. _converse.connection._dataRecv(test_utils.createRequest(result));
  80. await u.waitUntil(() => view.model.messages.length === 2);
  81. view.close();
  82. // Clear so that we don't match the older query
  83. while (sent_IQs.length) { sent_IQs.pop(); }
  84. await u.waitUntil(() => _converse.chatboxes.length === 1);
  85. await test_utils.openAndEnterChatRoom(_converse, muc_jid, 'romeo');
  86. view = _converse.chatboxviews.get(muc_jid);
  87. await u.waitUntil(() => view.model.messages.length);
  88. iq_get = await u.waitUntil(() => sent_IQs.filter(iq => iq.querySelector(`iq query[xmlns="${Strophe.NS.MAM}"]`)).pop());
  89. expect(Strophe.serialize(iq_get)).toBe(
  90. `<iq id="${iq_get.getAttribute('id')}" to="${muc_jid}" type="set" xmlns="jabber:client">`+
  91. `<query queryid="${iq_get.querySelector('query').getAttribute('queryid')}" xmlns="${Strophe.NS.MAM}">`+
  92. `<x type="submit" xmlns="jabber:x:data">`+
  93. `<field type="hidden" var="FORM_TYPE"><value>urn:xmpp:mam:2</value></field>`+
  94. `</x>`+
  95. `<set xmlns="http://jabber.org/protocol/rsm"><max>2</max><after>${message.querySelector('result').getAttribute('id')}</after></set>`+
  96. `</query>`+
  97. `</iq>`);
  98. first_msg_id = _converse.connection.getUniqueId();
  99. last_msg_id = _converse.connection.getUniqueId();
  100. message = u.toStanza(
  101. `<message xmlns="jabber:client"
  102. to="romeo@montague.lit/orchard"
  103. from="${muc_jid}">
  104. <result xmlns="urn:xmpp:mam:2" queryid="${iq_get.querySelector('query').getAttribute('queryid')}" id="${first_msg_id}">
  105. <forwarded xmlns="urn:xmpp:forward:0">
  106. <delay xmlns="urn:xmpp:delay" stamp="2018-01-09T06:17:23Z"/>
  107. <message from="${muc_jid}/some1" type="groupchat">
  108. <body>4th Message</body>
  109. </message>
  110. </forwarded>
  111. </result>
  112. </message>`);
  113. _converse.connection._dataRecv(test_utils.createRequest(message));
  114. message = u.toStanza(
  115. `<message xmlns="jabber:client"
  116. to="romeo@montague.lit/orchard"
  117. from="${muc_jid}">
  118. <result xmlns="urn:xmpp:mam:2" queryid="${iq_get.querySelector('query').getAttribute('queryid')}" id="${last_msg_id}">
  119. <forwarded xmlns="urn:xmpp:forward:0">
  120. <delay xmlns="urn:xmpp:delay" stamp="2018-01-09T06:18:23Z"/>
  121. <message from="${muc_jid}/some1" type="groupchat">
  122. <body>5th Message</body>
  123. </message>
  124. </forwarded>
  125. </result>
  126. </message>`);
  127. _converse.connection._dataRecv(test_utils.createRequest(message));
  128. // Clear so that we don't match the older query
  129. while (sent_IQs.length) { sent_IQs.pop(); }
  130. result = u.toStanza(
  131. `<iq type='result' id='${iq_get.getAttribute('id')}'>
  132. <fin xmlns='urn:xmpp:mam:2'>
  133. <set xmlns='http://jabber.org/protocol/rsm'>
  134. <first index='0'>${first_msg_id}</first>
  135. <last>${last_msg_id}</last>
  136. <count>5</count>
  137. </set>
  138. </fin>
  139. </iq>`);
  140. _converse.connection._dataRecv(test_utils.createRequest(result));
  141. await u.waitUntil(() => view.model.messages.length === 4);
  142. iq_get = await u.waitUntil(() => sent_IQs.filter(iq => iq.querySelector(`iq query[xmlns="${Strophe.NS.MAM}"]`)).pop());
  143. expect(Strophe.serialize(iq_get)).toBe(
  144. `<iq id="${iq_get.getAttribute('id')}" to="orchard@chat.shakespeare.lit" type="set" xmlns="jabber:client">`+
  145. `<query queryid="${iq_get.querySelector('query').getAttribute('queryid')}" xmlns="urn:xmpp:mam:2">`+
  146. `<x type="submit" xmlns="jabber:x:data">`+
  147. `<field type="hidden" var="FORM_TYPE"><value>urn:xmpp:mam:2</value></field>`+
  148. `</x>`+
  149. `<set xmlns="http://jabber.org/protocol/rsm">`+
  150. `<max>2</max><after>${last_msg_id}</after>`+
  151. `</set>`+
  152. `</query>`+
  153. `</iq>`);
  154. const msg_id = _converse.connection.getUniqueId();
  155. message = u.toStanza(
  156. `<message xmlns="jabber:client"
  157. to="romeo@montague.lit/orchard"
  158. from="${muc_jid}">
  159. <result xmlns="urn:xmpp:mam:2" queryid="${iq_get.querySelector('query').getAttribute('queryid')}" id="${msg_id}">
  160. <forwarded xmlns="urn:xmpp:forward:0">
  161. <delay xmlns="urn:xmpp:delay" stamp="2018-01-09T06:19:23Z"/>
  162. <message from="${muc_jid}/some1" type="groupchat">
  163. <body>6th Message</body>
  164. </message>
  165. </forwarded>
  166. </result>
  167. </message>`);
  168. _converse.connection._dataRecv(test_utils.createRequest(message));
  169. result = u.toStanza(
  170. `<iq type='result' id='${iq_get.getAttribute('id')}'>
  171. <fin xmlns="urn:xmpp:mam:2" complete="true">
  172. <set xmlns="http://jabber.org/protocol/rsm">
  173. <first index="0">${msg_id}</first>
  174. <last>${msg_id}</last>
  175. <count>6</count>
  176. </set>
  177. </fin>
  178. </iq>`);
  179. _converse.connection._dataRecv(test_utils.createRequest(result));
  180. await u.waitUntil(() => view.model.messages.length === 5);
  181. const msg_els = view.content.querySelectorAll('.chat-msg__text');
  182. expect(Array.from(msg_els).map(e => e.textContent).join(' ')).toBe("2nd Message 3rd Message 4th Message 5th Message 6th Message");
  183. done();
  184. }));
  185. });
  186. describe("An archived message", function () {
  187. describe("when received", function () {
  188. it("is discarded if it doesn't come from the right sender",
  189. mock.initConverse(
  190. ['discoInitialized'], {},
  191. async function (done, _converse) {
  192. await test_utils.waitForRoster(_converse, 'current', 1);
  193. const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@montague.lit';
  194. await test_utils.openChatBoxFor(_converse, contact_jid);
  195. await test_utils.waitUntilDiscoConfirmed(_converse, _converse.bare_jid, null, [Strophe.NS.MAM]);
  196. const sent_IQs = _converse.connection.IQ_stanzas;
  197. const stanza = await u.waitUntil(() => sent_IQs.filter(iq => iq.querySelector(`iq[type="set"] query[xmlns="${Strophe.NS.MAM}"]`)).pop());
  198. const queryid = stanza.querySelector('query').getAttribute('queryid');
  199. let msg = $msg({'id': _converse.connection.getUniqueId(), 'from': 'impersonator@capulet.lit', 'to': _converse.bare_jid})
  200. .c('result', {'xmlns': 'urn:xmpp:mam:2', 'queryid':queryid, 'id': _converse.connection.getUniqueId()})
  201. .c('forwarded', {'xmlns':'urn:xmpp:forward:0'})
  202. .c('delay', {'xmlns':'urn:xmpp:delay', 'stamp':'2010-07-10T23:08:25Z'}).up()
  203. .c('message', {
  204. 'xmlns':'jabber:client',
  205. 'to': _converse.bare_jid,
  206. 'id': _converse.connection.getUniqueId(),
  207. 'from': contact_jid,
  208. 'type':'chat'
  209. }).c('body').t("Meet me at the dance");
  210. spyOn(converse.env.log, 'warn');
  211. _converse.connection._dataRecv(test_utils.createRequest(msg));
  212. expect(converse.env.log.warn).toHaveBeenCalledWith(`Ignoring alleged MAM message from ${msg.nodeTree.getAttribute('from')}`);
  213. msg = $msg({'id': _converse.connection.getUniqueId(), 'to': _converse.bare_jid})
  214. .c('result', {'xmlns': 'urn:xmpp:mam:2', 'queryid':queryid, 'id': _converse.connection.getUniqueId()})
  215. .c('forwarded', {'xmlns':'urn:xmpp:forward:0'})
  216. .c('delay', {'xmlns':'urn:xmpp:delay', 'stamp':'2010-07-10T23:08:25Z'}).up()
  217. .c('message', {
  218. 'xmlns':'jabber:client',
  219. 'to': _converse.bare_jid,
  220. 'id': _converse.connection.getUniqueId(),
  221. 'from': contact_jid,
  222. 'type':'chat'
  223. }).c('body').t("Thrice the brinded cat hath mew'd.");
  224. _converse.connection._dataRecv(test_utils.createRequest(msg));
  225. const iq_result = $iq({'type': 'result', 'id': stanza.getAttribute('id')})
  226. .c('fin', {'xmlns': 'urn:xmpp:mam:2'})
  227. .c('set', {'xmlns': 'http://jabber.org/protocol/rsm'})
  228. .c('first', {'index': '0'}).t('23452-4534-1').up()
  229. .c('last').t('09af3-cc343-b409f').up()
  230. .c('count').t('16');
  231. _converse.connection._dataRecv(test_utils.createRequest(iq_result));
  232. const view = _converse.chatboxviews.get(contact_jid);
  233. await new Promise(resolve => view.once('messageInserted', resolve));
  234. expect(view.model.messages.length).toBe(1);
  235. expect(view.model.messages.at(0).get('message')).toBe("Thrice the brinded cat hath mew'd.");
  236. done();
  237. }));
  238. it("updates the is_archived value of an already cached version",
  239. mock.initConverse(
  240. ['discoInitialized'], {},
  241. async function (done, _converse) {
  242. await test_utils.openAndEnterChatRoom(_converse, 'trek-radio@conference.lightwitch.org', 'romeo');
  243. const view = _converse.chatboxviews.get('trek-radio@conference.lightwitch.org');
  244. let stanza = u.toStanza(
  245. `<message xmlns="jabber:client" to="romeo@montague.lit/orchard" type="groupchat" from="trek-radio@conference.lightwitch.org/some1">
  246. <body>Hello</body>
  247. <stanza-id xmlns="urn:xmpp:sid:0" id="45fbbf2a-1059-479d-9283-c8effaf05621" by="trek-radio@conference.lightwitch.org"/>
  248. </message>`);
  249. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  250. await u.waitUntil(() => view.content.querySelectorAll('.chat-msg').length);
  251. expect(view.model.messages.length).toBe(1);
  252. expect(view.model.messages.at(0).get('is_archived')).toBe(false);
  253. expect(view.model.messages.at(0).get('stanza_id trek-radio@conference.lightwitch.org')).toBe('45fbbf2a-1059-479d-9283-c8effaf05621');
  254. stanza = u.toStanza(
  255. `<message xmlns="jabber:client"
  256. to="romeo@montague.lit/orchard"
  257. from="trek-radio@conference.lightwitch.org">
  258. <result xmlns="urn:xmpp:mam:2" queryid="82d9db27-6cf8-4787-8c2c-5a560263d823" id="45fbbf2a-1059-479d-9283-c8effaf05621">
  259. <forwarded xmlns="urn:xmpp:forward:0">
  260. <delay xmlns="urn:xmpp:delay" stamp="2018-01-09T06:17:23Z"/>
  261. <message from="trek-radio@conference.lightwitch.org/some1" type="groupchat">
  262. <body>Hello</body>
  263. </message>
  264. </forwarded>
  265. </result>
  266. </message>`);
  267. spyOn(view.model, 'getDuplicateMessage').and.callThrough();
  268. spyOn(view.model, 'updateMessage').and.callThrough();
  269. view.model.queueMessage(stanza);
  270. await u.waitUntil(() => view.model.getDuplicateMessage.calls.count());
  271. expect(view.model.getDuplicateMessage.calls.count()).toBe(1);
  272. const result = view.model.getDuplicateMessage.calls.all()[0].returnValue
  273. expect(result instanceof _converse.Message).toBe(true);
  274. expect(view.content.querySelectorAll('.chat-msg').length).toBe(1);
  275. await u.waitUntil(() => view.model.updateMessage.calls.count());
  276. expect(view.model.messages.length).toBe(1);
  277. expect(view.model.messages.at(0).get('is_archived')).toBe(true);
  278. expect(view.model.messages.at(0).get('stanza_id trek-radio@conference.lightwitch.org')).toBe('45fbbf2a-1059-479d-9283-c8effaf05621');
  279. done();
  280. }));
  281. it("isn't shown as duplicate by comparing its stanza id or archive id",
  282. mock.initConverse(
  283. ['discoInitialized'], {},
  284. async function (done, _converse) {
  285. await test_utils.openAndEnterChatRoom(_converse, 'trek-radio@conference.lightwitch.org', 'jcbrand');
  286. const view = _converse.chatboxviews.get('trek-radio@conference.lightwitch.org');
  287. let stanza = u.toStanza(
  288. `<message xmlns="jabber:client" to="jcbrand@lightwitch.org/converse.js-73057452" type="groupchat" from="trek-radio@conference.lightwitch.org/comndrdukath#0805 (STO)">
  289. <body>negan</body>
  290. <stanza-id xmlns="urn:xmpp:sid:0" id="45fbbf2a-1059-479d-9283-c8effaf05621" by="trek-radio@conference.lightwitch.org"/>
  291. </message>`);
  292. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  293. await u.waitUntil(() => view.content.querySelectorAll('.chat-msg').length);
  294. // Not sure whether such a race-condition might pose a problem
  295. // in "real-world" situations.
  296. stanza = u.toStanza(
  297. `<message xmlns="jabber:client"
  298. to="jcbrand@lightwitch.org/converse.js-73057452"
  299. from="trek-radio@conference.lightwitch.org">
  300. <result xmlns="urn:xmpp:mam:2" queryid="82d9db27-6cf8-4787-8c2c-5a560263d823" id="45fbbf2a-1059-479d-9283-c8effaf05621">
  301. <forwarded xmlns="urn:xmpp:forward:0">
  302. <delay xmlns="urn:xmpp:delay" stamp="2018-01-09T06:17:23Z"/>
  303. <message from="trek-radio@conference.lightwitch.org/comndrdukath#0805 (STO)" type="groupchat">
  304. <body>negan</body>
  305. </message>
  306. </forwarded>
  307. </result>
  308. </message>`);
  309. spyOn(view.model, 'getDuplicateMessage').and.callThrough();
  310. view.model.queueMessage(stanza);
  311. await u.waitUntil(() => view.model.getDuplicateMessage.calls.count());
  312. expect(view.model.getDuplicateMessage.calls.count()).toBe(1);
  313. const result = await view.model.getDuplicateMessage.calls.all()[0].returnValue
  314. expect(result instanceof _converse.Message).toBe(true);
  315. expect(view.content.querySelectorAll('.chat-msg').length).toBe(1);
  316. done();
  317. }));
  318. it("isn't shown as duplicate by comparing only the archive id",
  319. mock.initConverse(
  320. ['discoInitialized'], {},
  321. async function (done, _converse) {
  322. await test_utils.openAndEnterChatRoom(_converse, 'discuss@conference.conversejs.org', 'romeo');
  323. const view = _converse.chatboxviews.get('discuss@conference.conversejs.org');
  324. let stanza = u.toStanza(
  325. `<message xmlns="jabber:client" to="romeo@montague.lit/orchard" from="discuss@conference.conversejs.org">
  326. <result xmlns="urn:xmpp:mam:2" queryid="06fea9ca-97c9-48c4-8583-009ff54ea2e8" id="7a9fde91-4387-4bf8-b5d3-978dab8f6bf3">
  327. <forwarded xmlns="urn:xmpp:forward:0">
  328. <delay xmlns="urn:xmpp:delay" stamp="2018-12-05T04:53:12Z"/>
  329. <message xmlns="jabber:client" to="discuss@conference.conversejs.org" type="groupchat" xml:lang="en" from="discuss@conference.conversejs.org/prezel">
  330. <body>looks like omemo fails completely with "bundle is undefined" when there is a device in the devicelist that has no keys published</body>
  331. <x xmlns="http://jabber.org/protocol/muc#user">
  332. <item affiliation="none" jid="prezel@blubber.im" role="participant"/>
  333. </x>
  334. </message>
  335. </forwarded>
  336. </result>
  337. </message>`);
  338. view.model.queueMessage(stanza);
  339. await u.waitUntil(() => view.content.querySelectorAll('.chat-msg').length);
  340. expect(view.content.querySelectorAll('.chat-msg').length).toBe(1);
  341. stanza = u.toStanza(
  342. `<message xmlns="jabber:client" to="romeo@montague.lit/orchard" from="discuss@conference.conversejs.org">
  343. <result xmlns="urn:xmpp:mam:2" queryid="06fea9ca-97c9-48c4-8583-009ff54ea2e8" id="7a9fde91-4387-4bf8-b5d3-978dab8f6bf3">
  344. <forwarded xmlns="urn:xmpp:forward:0">
  345. <delay xmlns="urn:xmpp:delay" stamp="2018-12-05T04:53:12Z"/>
  346. <message xmlns="jabber:client" to="discuss@conference.conversejs.org" type="groupchat" xml:lang="en" from="discuss@conference.conversejs.org/prezel">
  347. <body>looks like omemo fails completely with "bundle is undefined" when there is a device in the devicelist that has no keys published</body>
  348. <x xmlns="http://jabber.org/protocol/muc#user">
  349. <item affiliation="none" jid="prezel@blubber.im" role="participant"/>
  350. </x>
  351. </message>
  352. </forwarded>
  353. </result>
  354. </message>`);
  355. spyOn(view.model, 'getDuplicateMessage').and.callThrough();
  356. view.model.queueMessage(stanza);
  357. await u.waitUntil(() => view.model.getDuplicateMessage.calls.count());
  358. expect(view.model.getDuplicateMessage.calls.count()).toBe(1);
  359. const result = await view.model.getDuplicateMessage.calls.all()[0].returnValue
  360. expect(result instanceof _converse.Message).toBe(true);
  361. expect(view.content.querySelectorAll('.chat-msg').length).toBe(1);
  362. done();
  363. }))
  364. });
  365. });
  366. describe("The archive.query API", function () {
  367. it("can be used to query for all archived messages",
  368. mock.initConverse(['discoInitialized'], {}, async function (done, _converse) {
  369. const sendIQ = _converse.connection.sendIQ;
  370. await test_utils.waitUntilDiscoConfirmed(_converse, _converse.bare_jid, null, [Strophe.NS.MAM]);
  371. let sent_stanza, IQ_id;
  372. spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
  373. sent_stanza = iq;
  374. IQ_id = sendIQ.bind(this)(iq, callback, errback);
  375. });
  376. _converse.api.archive.query();
  377. await u.waitUntil(() => sent_stanza);
  378. const queryid = sent_stanza.nodeTree.querySelector('query').getAttribute('queryid');
  379. expect(sent_stanza.toString()).toBe(
  380. `<iq id="${IQ_id}" type="set" xmlns="jabber:client"><query queryid="${queryid}" xmlns="urn:xmpp:mam:2"/></iq>`);
  381. done();
  382. }));
  383. it("can be used to query for all messages to/from a particular JID",
  384. mock.initConverse([], {}, async function (done, _converse) {
  385. await test_utils.waitUntilDiscoConfirmed(_converse, _converse.bare_jid, null, [Strophe.NS.MAM]);
  386. let sent_stanza, IQ_id;
  387. const sendIQ = _converse.connection.sendIQ;
  388. spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
  389. sent_stanza = iq;
  390. IQ_id = sendIQ.bind(this)(iq, callback, errback);
  391. });
  392. _converse.api.archive.query({'with':'juliet@capulet.lit'});
  393. await u.waitUntil(() => sent_stanza);
  394. const queryid = sent_stanza.nodeTree.querySelector('query').getAttribute('queryid');
  395. expect(sent_stanza.toString()).toBe(
  396. `<iq id="${IQ_id}" type="set" xmlns="jabber:client">`+
  397. `<query queryid="${queryid}" xmlns="urn:xmpp:mam:2">`+
  398. `<x type="submit" xmlns="jabber:x:data">`+
  399. `<field type="hidden" var="FORM_TYPE">`+
  400. `<value>urn:xmpp:mam:2</value>`+
  401. `</field>`+
  402. `<field var="with">`+
  403. `<value>juliet@capulet.lit</value>`+
  404. `</field>`+
  405. `</x>`+
  406. `</query>`+
  407. `</iq>`);
  408. done();
  409. }));
  410. it("can be used to query for archived messages from a chat room",
  411. mock.initConverse([], {}, async function (done, _converse) {
  412. const room_jid = 'coven@chat.shakespeare.lit';
  413. _converse.api.archive.query({'with': room_jid, 'groupchat': true});
  414. await test_utils.waitUntilDiscoConfirmed(_converse, room_jid, null, [Strophe.NS.MAM]);
  415. const sent_stanzas = _converse.connection.sent_stanzas;
  416. const stanza = await u.waitUntil(
  417. () => sent_stanzas.filter(s => sizzle(`[xmlns="${Strophe.NS.MAM}"]`, s).length).pop());
  418. const queryid = stanza.querySelector('query').getAttribute('queryid');
  419. expect(Strophe.serialize(stanza)).toBe(
  420. `<iq id="${stanza.getAttribute('id')}" to="coven@chat.shakespeare.lit" type="set" xmlns="jabber:client">`+
  421. `<query queryid="${queryid}" xmlns="urn:xmpp:mam:2">`+
  422. `<x type="submit" xmlns="jabber:x:data">`+
  423. `<field type="hidden" var="FORM_TYPE">`+
  424. `<value>urn:xmpp:mam:2</value>`+
  425. `</field>`+
  426. `</x>`+
  427. `</query>`+
  428. `</iq>`);
  429. done();
  430. }));
  431. it("checks whether returned MAM messages from a MUC room are from the right JID",
  432. mock.initConverse([], {}, async function (done, _converse) {
  433. const room_jid = 'coven@chat.shakespeare.lit';
  434. const promise = _converse.api.archive.query({'with': room_jid, 'groupchat': true, 'max':'10'});
  435. await test_utils.waitUntilDiscoConfirmed(_converse, room_jid, null, [Strophe.NS.MAM]);
  436. const sent_stanzas = _converse.connection.sent_stanzas;
  437. const sent_stanza = await u.waitUntil(
  438. () => sent_stanzas.filter(s => sizzle(`[xmlns="${Strophe.NS.MAM}"]`, s).length).pop());
  439. const queryid = sent_stanza.querySelector('query').getAttribute('queryid');
  440. /* <message id='iasd207' from='coven@chat.shakespeare.lit' to='hag66@shakespeare.lit/pda'>
  441. * <result xmlns='urn:xmpp:mam:2' queryid='g27' id='34482-21985-73620'>
  442. * <forwarded xmlns='urn:xmpp:forward:0'>
  443. * <delay xmlns='urn:xmpp:delay' stamp='2002-10-13T23:58:37Z'/>
  444. * <message xmlns="jabber:client"
  445. * from='coven@chat.shakespeare.lit/firstwitch'
  446. * id='162BEBB1-F6DB-4D9A-9BD8-CFDCC801A0B2'
  447. * type='groupchat'>
  448. * <body>Thrice the brinded cat hath mew'd.</body>
  449. * <x xmlns='http://jabber.org/protocol/muc#user'>
  450. * <item affiliation='none'
  451. * jid='witch1@shakespeare.lit'
  452. * role='participant' />
  453. * </x>
  454. * </message>
  455. * </forwarded>
  456. * </result>
  457. * </message>
  458. */
  459. const msg1 = $msg({'id':'iasd207', 'from': 'other@chat.shakespear.lit', 'to': 'romeo@montague.lit'})
  460. .c('result', {'xmlns': 'urn:xmpp:mam:2', 'queryid':queryid, 'id':'34482-21985-73620'})
  461. .c('forwarded', {'xmlns':'urn:xmpp:forward:0'})
  462. .c('delay', {'xmlns':'urn:xmpp:delay', 'stamp':'2010-07-10T23:08:25Z'}).up()
  463. .c('message', {
  464. 'xmlns':'jabber:client',
  465. 'to':'romeo@montague.lit',
  466. 'id':'162BEBB1-F6DB-4D9A-9BD8-CFDCC801A0B2',
  467. 'from':'coven@chat.shakespeare.lit/firstwitch',
  468. 'type':'groupchat' })
  469. .c('body').t("Thrice the brinded cat hath mew'd.");
  470. _converse.connection._dataRecv(test_utils.createRequest(msg1));
  471. /* Send an <iq> stanza to indicate the end of the result set.
  472. *
  473. * <iq type='result' id='juliet1'>
  474. * <fin xmlns='urn:xmpp:mam:2'>
  475. * <set xmlns='http://jabber.org/protocol/rsm'>
  476. * <first index='0'>28482-98726-73623</first>
  477. * <last>09af3-cc343-b409f</last>
  478. * <count>20</count>
  479. * </set>
  480. * </iq>
  481. */
  482. const stanza = $iq({'type': 'result', 'id': sent_stanza.getAttribute('id')})
  483. .c('fin', {'xmlns': 'urn:xmpp:mam:2'})
  484. .c('set', {'xmlns': 'http://jabber.org/protocol/rsm'})
  485. .c('first', {'index': '0'}).t('23452-4534-1').up()
  486. .c('last').t('09af3-cc343-b409f').up()
  487. .c('count').t('16');
  488. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  489. const result = await promise;
  490. expect(result.messages.length).toBe(0);
  491. done();
  492. }));
  493. it("can be used to query for all messages in a certain timespan",
  494. mock.initConverse([], {}, async function (done, _converse) {
  495. await test_utils.waitUntilDiscoConfirmed(_converse, _converse.bare_jid, null, [Strophe.NS.MAM]);
  496. let sent_stanza, IQ_id;
  497. const sendIQ = _converse.connection.sendIQ;
  498. spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
  499. sent_stanza = iq;
  500. IQ_id = sendIQ.bind(this)(iq, callback, errback);
  501. });
  502. const start = '2010-06-07T00:00:00Z';
  503. const end = '2010-07-07T13:23:54Z';
  504. _converse.api.archive.query({
  505. 'start': start,
  506. 'end': end
  507. });
  508. await u.waitUntil(() => sent_stanza);
  509. const queryid = sent_stanza.nodeTree.querySelector('query').getAttribute('queryid');
  510. expect(sent_stanza.toString()).toBe(
  511. `<iq id="${IQ_id}" type="set" xmlns="jabber:client">`+
  512. `<query queryid="${queryid}" xmlns="urn:xmpp:mam:2">`+
  513. `<x type="submit" xmlns="jabber:x:data">`+
  514. `<field type="hidden" var="FORM_TYPE">`+
  515. `<value>urn:xmpp:mam:2</value>`+
  516. `</field>`+
  517. `<field var="start">`+
  518. `<value>${dayjs(start).toISOString()}</value>`+
  519. `</field>`+
  520. `<field var="end">`+
  521. `<value>${dayjs(end).toISOString()}</value>`+
  522. `</field>`+
  523. `</x>`+
  524. `</query>`+
  525. `</iq>`
  526. );
  527. done();
  528. }));
  529. it("throws a TypeError if an invalid date is provided",
  530. mock.initConverse([], {}, async function (done, _converse) {
  531. await test_utils.waitUntilDiscoConfirmed(_converse, _converse.bare_jid, null, [Strophe.NS.MAM]);
  532. try {
  533. await _converse.api.archive.query({'start': 'not a real date'});
  534. } catch (e) {
  535. expect(() => {throw e}).toThrow(new TypeError('archive.query: invalid date provided for: start'));
  536. }
  537. done();
  538. }));
  539. it("can be used to query for all messages after a certain time",
  540. mock.initConverse([], {}, async function (done, _converse) {
  541. await test_utils.waitUntilDiscoConfirmed(_converse, _converse.bare_jid, null, [Strophe.NS.MAM]);
  542. let sent_stanza, IQ_id;
  543. const sendIQ = _converse.connection.sendIQ;
  544. spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
  545. sent_stanza = iq;
  546. IQ_id = sendIQ.bind(this)(iq, callback, errback);
  547. });
  548. if (!_converse.disco_entities.get(_converse.domain).features.findWhere({'var': Strophe.NS.MAM})) {
  549. _converse.disco_entities.get(_converse.domain).features.create({'var': Strophe.NS.MAM});
  550. }
  551. const start = '2010-06-07T00:00:00Z';
  552. _converse.api.archive.query({'start': start});
  553. await u.waitUntil(() => sent_stanza);
  554. const queryid = sent_stanza.nodeTree.querySelector('query').getAttribute('queryid');
  555. expect(sent_stanza.toString()).toBe(
  556. `<iq id="${IQ_id}" type="set" xmlns="jabber:client">`+
  557. `<query queryid="${queryid}" xmlns="urn:xmpp:mam:2">`+
  558. `<x type="submit" xmlns="jabber:x:data">`+
  559. `<field type="hidden" var="FORM_TYPE">`+
  560. `<value>urn:xmpp:mam:2</value>`+
  561. `</field>`+
  562. `<field var="start">`+
  563. `<value>${dayjs(start).toISOString()}</value>`+
  564. `</field>`+
  565. `</x>`+
  566. `</query>`+
  567. `</iq>`
  568. );
  569. done();
  570. }));
  571. it("can be used to query for a limited set of results",
  572. mock.initConverse([], {}, async function (done, _converse) {
  573. await test_utils.waitUntilDiscoConfirmed(_converse, _converse.bare_jid, null, [Strophe.NS.MAM]);
  574. let sent_stanza, IQ_id;
  575. const sendIQ = _converse.connection.sendIQ;
  576. spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
  577. sent_stanza = iq;
  578. IQ_id = sendIQ.bind(this)(iq, callback, errback);
  579. });
  580. const start = '2010-06-07T00:00:00Z';
  581. _converse.api.archive.query({'start': start, 'max':10});
  582. await u.waitUntil(() => sent_stanza);
  583. const queryid = sent_stanza.nodeTree.querySelector('query').getAttribute('queryid');
  584. expect(sent_stanza.toString()).toBe(
  585. `<iq id="${IQ_id}" type="set" xmlns="jabber:client">`+
  586. `<query queryid="${queryid}" xmlns="urn:xmpp:mam:2">`+
  587. `<x type="submit" xmlns="jabber:x:data">`+
  588. `<field type="hidden" var="FORM_TYPE">`+
  589. `<value>urn:xmpp:mam:2</value>`+
  590. `</field>`+
  591. `<field var="start">`+
  592. `<value>${dayjs(start).toISOString()}</value>`+
  593. `</field>`+
  594. `</x>`+
  595. `<set xmlns="http://jabber.org/protocol/rsm">`+
  596. `<max>10</max>`+
  597. `</set>`+
  598. `</query>`+
  599. `</iq>`
  600. );
  601. done();
  602. }));
  603. it("can be used to page through results",
  604. mock.initConverse([], {}, async function (done, _converse) {
  605. await test_utils.waitUntilDiscoConfirmed(_converse, _converse.bare_jid, null, [Strophe.NS.MAM]);
  606. let sent_stanza, IQ_id;
  607. const sendIQ = _converse.connection.sendIQ;
  608. spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
  609. sent_stanza = iq;
  610. IQ_id = sendIQ.bind(this)(iq, callback, errback);
  611. });
  612. const start = '2010-06-07T00:00:00Z';
  613. _converse.api.archive.query({
  614. 'start': start,
  615. 'after': '09af3-cc343-b409f',
  616. 'max':10
  617. });
  618. await u.waitUntil(() => sent_stanza);
  619. const queryid = sent_stanza.nodeTree.querySelector('query').getAttribute('queryid');
  620. expect(sent_stanza.toString()).toBe(
  621. `<iq id="${IQ_id}" type="set" xmlns="jabber:client">`+
  622. `<query queryid="${queryid}" xmlns="urn:xmpp:mam:2">`+
  623. `<x type="submit" xmlns="jabber:x:data">`+
  624. `<field type="hidden" var="FORM_TYPE">`+
  625. `<value>urn:xmpp:mam:2</value>`+
  626. `</field>`+
  627. `<field var="start">`+
  628. `<value>${dayjs(start).toISOString()}</value>`+
  629. `</field>`+
  630. `</x>`+
  631. `<set xmlns="http://jabber.org/protocol/rsm">`+
  632. `<max>10</max>`+
  633. `<after>09af3-cc343-b409f</after>`+
  634. `</set>`+
  635. `</query>`+
  636. `</iq>`);
  637. done();
  638. }));
  639. it("accepts \"before\" with an empty string as value to reverse the order",
  640. mock.initConverse([], {}, async function (done, _converse) {
  641. await test_utils.waitUntilDiscoConfirmed(_converse, _converse.bare_jid, null, [Strophe.NS.MAM]);
  642. let sent_stanza, IQ_id;
  643. const sendIQ = _converse.connection.sendIQ;
  644. spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
  645. sent_stanza = iq;
  646. IQ_id = sendIQ.bind(this)(iq, callback, errback);
  647. });
  648. _converse.api.archive.query({'before': '', 'max':10});
  649. await u.waitUntil(() => sent_stanza);
  650. const queryid = sent_stanza.nodeTree.querySelector('query').getAttribute('queryid');
  651. expect(sent_stanza.toString()).toBe(
  652. `<iq id="${IQ_id}" type="set" xmlns="jabber:client">`+
  653. `<query queryid="${queryid}" xmlns="urn:xmpp:mam:2">`+
  654. `<x type="submit" xmlns="jabber:x:data">`+
  655. `<field type="hidden" var="FORM_TYPE">`+
  656. `<value>urn:xmpp:mam:2</value>`+
  657. `</field>`+
  658. `</x>`+
  659. `<set xmlns="http://jabber.org/protocol/rsm">`+
  660. `<max>10</max>`+
  661. `<before></before>`+
  662. `</set>`+
  663. `</query>`+
  664. `</iq>`);
  665. done();
  666. }));
  667. it("accepts a _converse.RSM object for the query options",
  668. mock.initConverse([], {}, async function (done, _converse) {
  669. await test_utils.waitUntilDiscoConfirmed(_converse, _converse.bare_jid, null, [Strophe.NS.MAM]);
  670. let sent_stanza, IQ_id;
  671. const sendIQ = _converse.connection.sendIQ;
  672. spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
  673. sent_stanza = iq;
  674. IQ_id = sendIQ.bind(this)(iq, callback, errback);
  675. });
  676. // Normally the user wouldn't manually make a _converse.RSM object
  677. // and pass it in. However, in the callback method an RSM object is
  678. // returned which can be reused for easy paging. This test is
  679. // more for that usecase.
  680. const rsm = new _converse.RSM({'max': '10'});
  681. rsm['with'] = 'romeo@montague.lit'; // eslint-disable-line dot-notation
  682. rsm.start = '2010-06-07T00:00:00Z';
  683. _converse.api.archive.query(rsm);
  684. await u.waitUntil(() => sent_stanza);
  685. const queryid = sent_stanza.nodeTree.querySelector('query').getAttribute('queryid');
  686. expect(sent_stanza.toString()).toBe(
  687. `<iq id="${IQ_id}" type="set" xmlns="jabber:client">`+
  688. `<query queryid="${queryid}" xmlns="urn:xmpp:mam:2">`+
  689. `<x type="submit" xmlns="jabber:x:data">`+
  690. `<field type="hidden" var="FORM_TYPE">`+
  691. `<value>urn:xmpp:mam:2</value>`+
  692. `</field>`+
  693. `<field var="with">`+
  694. `<value>romeo@montague.lit</value>`+
  695. `</field>`+
  696. `<field var="start">`+
  697. `<value>${dayjs(rsm.start).toISOString()}</value>`+
  698. `</field>`+
  699. `</x>`+
  700. `<set xmlns="http://jabber.org/protocol/rsm">`+
  701. `<max>10</max>`+
  702. `</set>`+
  703. `</query>`+
  704. `</iq>`);
  705. done();
  706. }));
  707. it("returns an object which includes the messages and a _converse.RSM object",
  708. mock.initConverse([], {}, async function (done, _converse) {
  709. await test_utils.waitUntilDiscoConfirmed(_converse, _converse.bare_jid, null, [Strophe.NS.MAM]);
  710. let sent_stanza, IQ_id;
  711. const sendIQ = _converse.connection.sendIQ;
  712. spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
  713. sent_stanza = iq;
  714. IQ_id = sendIQ.bind(this)(iq, callback, errback);
  715. });
  716. const promise = _converse.api.archive.query({'with': 'romeo@capulet.lit', 'max':'10'});
  717. await u.waitUntil(() => sent_stanza);
  718. const queryid = sent_stanza.nodeTree.querySelector('query').getAttribute('queryid');
  719. /* <message id='aeb213' to='juliet@capulet.lit/chamber'>
  720. * <result xmlns='urn:xmpp:mam:2' queryid='f27' id='28482-98726-73623'>
  721. * <forwarded xmlns='urn:xmpp:forward:0'>
  722. * <delay xmlns='urn:xmpp:delay' stamp='2010-07-10T23:08:25Z'/>
  723. * <message xmlns='jabber:client'
  724. * to='juliet@capulet.lit/balcony'
  725. * from='romeo@montague.lit/orchard'
  726. * type='chat'>
  727. * <body>Call me but love, and I'll be new baptized; Henceforth I never will be Romeo.</body>
  728. * </message>
  729. * </forwarded>
  730. * </result>
  731. * </message>
  732. */
  733. const msg1 = $msg({'id':'aeb212', 'to':'juliet@capulet.lit/chamber'})
  734. .c('result', {'xmlns': 'urn:xmpp:mam:2', 'queryid':queryid, 'id':'28482-98726-73623'})
  735. .c('forwarded', {'xmlns':'urn:xmpp:forward:0'})
  736. .c('delay', {'xmlns':'urn:xmpp:delay', 'stamp':'2010-07-10T23:08:25Z'}).up()
  737. .c('message', {
  738. 'xmlns':'jabber:client',
  739. 'to':'juliet@capulet.lit/balcony',
  740. 'from':'romeo@montague.lit/orchard',
  741. 'type':'chat' })
  742. .c('body').t("Call me but love, and I'll be new baptized;");
  743. _converse.connection._dataRecv(test_utils.createRequest(msg1));
  744. const msg2 = $msg({'id':'aeb213', 'to':'juliet@capulet.lit/chamber'})
  745. .c('result', {'xmlns': 'urn:xmpp:mam:2', 'queryid':queryid, 'id':'28482-98726-73624'})
  746. .c('forwarded', {'xmlns':'urn:xmpp:forward:0'})
  747. .c('delay', {'xmlns':'urn:xmpp:delay', 'stamp':'2010-07-10T23:08:25Z'}).up()
  748. .c('message', {
  749. 'xmlns':'jabber:client',
  750. 'to':'juliet@capulet.lit/balcony',
  751. 'from':'romeo@montague.lit/orchard',
  752. 'type':'chat' })
  753. .c('body').t("Henceforth I never will be Romeo.");
  754. _converse.connection._dataRecv(test_utils.createRequest(msg2));
  755. /* Send an <iq> stanza to indicate the end of the result set.
  756. *
  757. * <iq type='result' id='juliet1'>
  758. * <fin xmlns='urn:xmpp:mam:2'>
  759. * <set xmlns='http://jabber.org/protocol/rsm'>
  760. * <first index='0'>28482-98726-73623</first>
  761. * <last>09af3-cc343-b409f</last>
  762. * <count>20</count>
  763. * </set>
  764. * </iq>
  765. */
  766. const stanza = $iq({'type': 'result', 'id': IQ_id})
  767. .c('fin', {'xmlns': 'urn:xmpp:mam:2'})
  768. .c('set', {'xmlns': 'http://jabber.org/protocol/rsm'})
  769. .c('first', {'index': '0'}).t('23452-4534-1').up()
  770. .c('last').t('09af3-cc343-b409f').up()
  771. .c('count').t('16');
  772. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  773. const result = await promise;
  774. expect(result.messages.length).toBe(2);
  775. expect(result.messages[0].outerHTML).toBe(msg1.nodeTree.outerHTML);
  776. expect(result.messages[1].outerHTML).toBe(msg2.nodeTree.outerHTML);
  777. expect(result.rsm['with']).toBe('romeo@capulet.lit'); // eslint-disable-line dot-notation
  778. expect(result.rsm.max).toBe('10');
  779. expect(result.rsm.count).toBe('16');
  780. expect(result.rsm.first).toBe('23452-4534-1');
  781. expect(result.rsm.last).toBe('09af3-cc343-b409f');
  782. done()
  783. }));
  784. });
  785. describe("The default preference", function () {
  786. it("is set once server support for MAM has been confirmed",
  787. mock.initConverse([], {}, async function (done, _converse) {
  788. const entity = await _converse.api.disco.entities.get(_converse.domain);
  789. spyOn(_converse, 'onMAMPreferences').and.callThrough();
  790. _converse.message_archiving = 'never';
  791. const feature = new Model({
  792. 'var': Strophe.NS.MAM
  793. });
  794. spyOn(feature, 'save').and.callFake(feature.set); // Save will complain about a url not being set
  795. entity.onFeatureAdded(feature);
  796. const IQ_stanzas = _converse.connection.IQ_stanzas;
  797. let sent_stanza = await u.waitUntil(() => IQ_stanzas.filter(s => sizzle('iq[type="get"] prefs[xmlns="urn:xmpp:mam:2"]', s).length).pop());
  798. expect(Strophe.serialize(sent_stanza)).toBe(
  799. `<iq id="${sent_stanza.getAttribute('id')}" type="get" xmlns="jabber:client">`+
  800. `<prefs xmlns="urn:xmpp:mam:2"/>`+
  801. `</iq>`);
  802. /* Example 20. Server responds with current preferences
  803. *
  804. * <iq type='result' id='juliet2'>
  805. * <prefs xmlns='urn:xmpp:mam:0' default='roster'>
  806. * <always/>
  807. * <never/>
  808. * </prefs>
  809. * </iq>
  810. */
  811. let stanza = $iq({'type': 'result', 'id': sent_stanza.getAttribute('id')})
  812. .c('prefs', {'xmlns': Strophe.NS.MAM, 'default':'roster'})
  813. .c('always').c('jid').t('romeo@montague.lit').up().up()
  814. .c('never').c('jid').t('montague@montague.lit');
  815. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  816. await u.waitUntil(() => _converse.onMAMPreferences.calls.count());
  817. expect(_converse.onMAMPreferences).toHaveBeenCalled();
  818. sent_stanza = await u.waitUntil(() => IQ_stanzas.filter(s => sizzle('iq[type="set"] prefs[xmlns="urn:xmpp:mam:2"]', s).length).pop());
  819. expect(Strophe.serialize(sent_stanza)).toBe(
  820. `<iq id="${sent_stanza.getAttribute('id')}" type="set" xmlns="jabber:client">`+
  821. `<prefs default="never" xmlns="urn:xmpp:mam:2">`+
  822. `<always><jid>romeo@montague.lit</jid></always>`+
  823. `<never><jid>montague@montague.lit</jid></never>`+
  824. `</prefs>`+
  825. `</iq>`
  826. );
  827. expect(feature.get('preference')).toBe(undefined);
  828. /* <iq type='result' id='juliet3'>
  829. * <prefs xmlns='urn:xmpp:mam:0' default='always'>
  830. * <always>
  831. * <jid>romeo@montague.lit</jid>
  832. * </always>
  833. * <never>
  834. * <jid>montague@montague.lit</jid>
  835. * </never>
  836. * </prefs>
  837. * </iq>
  838. */
  839. stanza = $iq({'type': 'result', 'id': sent_stanza.getAttribute('id')})
  840. .c('prefs', {'xmlns': Strophe.NS.MAM, 'default':'always'})
  841. .c('always').up()
  842. .c('never');
  843. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  844. await u.waitUntil(() => feature.save.calls.count());
  845. expect(feature.save).toHaveBeenCalled();
  846. expect(feature.get('preferences')['default']).toBe('never'); // eslint-disable-line dot-notation
  847. done();
  848. }));
  849. });
  850. });
  851. describe("Chatboxes", function () {
  852. describe("A Chatbox", function () {
  853. it("will fetch archived messages once it's opened",
  854. mock.initConverse(['discoInitialized'], {}, async function (done, _converse) {
  855. await test_utils.waitForRoster(_converse, 'current', 1);
  856. const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@montague.lit';
  857. await test_utils.openChatBoxFor(_converse, contact_jid);
  858. await test_utils.waitUntilDiscoConfirmed(_converse, _converse.bare_jid, null, [Strophe.NS.MAM]);
  859. let sent_stanza, IQ_id;
  860. const sendIQ = _converse.connection.sendIQ;
  861. spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
  862. sent_stanza = iq;
  863. IQ_id = sendIQ.bind(this)(iq, callback, errback);
  864. });
  865. await u.waitUntil(() => sent_stanza);
  866. const stanza_el = sent_stanza.root().nodeTree;
  867. const queryid = stanza_el.querySelector('query').getAttribute('queryid');
  868. expect(sent_stanza.toString()).toBe(
  869. `<iq id="${stanza_el.getAttribute('id')}" type="set" xmlns="jabber:client">`+
  870. `<query queryid="${queryid}" xmlns="urn:xmpp:mam:2">`+
  871. `<x type="submit" xmlns="jabber:x:data">`+
  872. `<field type="hidden" var="FORM_TYPE"><value>urn:xmpp:mam:2</value></field>`+
  873. `<field var="with"><value>mercutio@montague.lit</value></field>`+
  874. `</x>`+
  875. `<set xmlns="http://jabber.org/protocol/rsm"><max>50</max><before></before></set>`+
  876. `</query>`+
  877. `</iq>`
  878. );
  879. const msg1 = $msg({'id':'aeb212', 'to': contact_jid})
  880. .c('result', {'xmlns': 'urn:xmpp:mam:2', 'queryid':queryid, 'id':'28482-98726-73623'})
  881. .c('forwarded', {'xmlns':'urn:xmpp:forward:0'})
  882. .c('delay', {'xmlns':'urn:xmpp:delay', 'stamp':'2010-07-10T23:08:25Z'}).up()
  883. .c('message', {
  884. 'xmlns':'jabber:client',
  885. 'to': contact_jid,
  886. 'from': _converse.bare_jid,
  887. 'type':'chat' })
  888. .c('body').t("Call me but love, and I'll be new baptized;");
  889. _converse.connection._dataRecv(test_utils.createRequest(msg1));
  890. const msg2 = $msg({'id':'aeb213', 'to': contact_jid})
  891. .c('result', {'xmlns': 'urn:xmpp:mam:2', 'queryid':queryid, 'id':'28482-98726-73624'})
  892. .c('forwarded', {'xmlns':'urn:xmpp:forward:0'})
  893. .c('delay', {'xmlns':'urn:xmpp:delay', 'stamp':'2010-07-10T23:08:25Z'}).up()
  894. .c('message', {
  895. 'xmlns':'jabber:client',
  896. 'to': contact_jid,
  897. 'from': _converse.bare_jid,
  898. 'type':'chat' })
  899. .c('body').t("Henceforth I never will be Romeo.");
  900. _converse.connection._dataRecv(test_utils.createRequest(msg2));
  901. const stanza = $iq({'type': 'result', 'id': IQ_id})
  902. .c('fin', {'xmlns': 'urn:xmpp:mam:2'})
  903. .c('set', {'xmlns': 'http://jabber.org/protocol/rsm'})
  904. .c('first', {'index': '0'}).t('23452-4534-1').up()
  905. .c('last').t('09af3-cc343-b409f').up()
  906. .c('count').t('16');
  907. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  908. done();
  909. }));
  910. it("will show an error message if the MAM query times out",
  911. mock.initConverse(['discoInitialized'], {}, async function (done, _converse) {
  912. const sendIQ = _converse.connection.sendIQ;
  913. let timeout_happened = false;
  914. spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
  915. sendIQ.bind(this)(iq, callback, errback);
  916. if (!timeout_happened) {
  917. if (typeof(iq.tree) === "function") {
  918. iq = iq.tree();
  919. }
  920. if (sizzle('query[xmlns="urn:xmpp:mam:2"]', iq).length) {
  921. // We emulate a timeout event
  922. callback(null);
  923. timeout_happened = true;
  924. }
  925. }
  926. });
  927. await test_utils.waitForRoster(_converse, 'current', 1);
  928. const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@montague.lit';
  929. await test_utils.openChatBoxFor(_converse, contact_jid);
  930. await test_utils.waitUntilDiscoConfirmed(_converse, _converse.bare_jid, null, [Strophe.NS.MAM]);
  931. const IQ_stanzas = _converse.connection.IQ_stanzas;
  932. let sent_stanza = await u.waitUntil(() => IQ_stanzas.filter(iq => sizzle('query[xmlns="urn:xmpp:mam:2"]', iq).length).pop());
  933. let queryid = sent_stanza.querySelector('query').getAttribute('queryid');
  934. expect(Strophe.serialize(sent_stanza)).toBe(
  935. `<iq id="${sent_stanza.getAttribute('id')}" type="set" xmlns="jabber:client">`+
  936. `<query queryid="${queryid}" xmlns="urn:xmpp:mam:2">`+
  937. `<x type="submit" xmlns="jabber:x:data">`+
  938. `<field type="hidden" var="FORM_TYPE"><value>urn:xmpp:mam:2</value></field>`+
  939. `<field var="with"><value>mercutio@montague.lit</value></field>`+
  940. `</x>`+
  941. `<set xmlns="http://jabber.org/protocol/rsm"><max>50</max><before></before></set>`+
  942. `</query>`+
  943. `</iq>`);
  944. const view = _converse.chatboxviews.get(contact_jid);
  945. expect(view.model.messages.length).toBe(1);
  946. expect(view.model.messages.at(0).get('is_ephemeral')).toBe(false);
  947. expect(view.model.messages.at(0).get('type')).toBe('error');
  948. expect(view.model.messages.at(0).get('message')).toBe('Timeout while trying to fetch archived messages.');
  949. let err_message = view.el.querySelector('.message.chat-error');
  950. err_message.querySelector('.retry').click();
  951. expect(err_message.querySelector('.spinner')).not.toBe(null);
  952. while (_converse.connection.IQ_stanzas.length) {
  953. _converse.connection.IQ_stanzas.pop();
  954. }
  955. sent_stanza = await u.waitUntil(() => IQ_stanzas.filter(iq => sizzle('query[xmlns="urn:xmpp:mam:2"]', iq).length).pop());
  956. queryid = sent_stanza.querySelector('query').getAttribute('queryid');
  957. expect(Strophe.serialize(sent_stanza)).toBe(
  958. `<iq id="${sent_stanza.getAttribute('id')}" type="set" xmlns="jabber:client">`+
  959. `<query queryid="${queryid}" xmlns="urn:xmpp:mam:2">`+
  960. `<x type="submit" xmlns="jabber:x:data">`+
  961. `<field type="hidden" var="FORM_TYPE"><value>urn:xmpp:mam:2</value></field>`+
  962. `<field var="with"><value>mercutio@montague.lit</value></field>`+
  963. `</x>`+
  964. `<set xmlns="http://jabber.org/protocol/rsm"><max>50</max><before></before></set>`+
  965. `</query>`+
  966. `</iq>`);
  967. const msg1 = $msg({'id':'aeb212', 'to': contact_jid})
  968. .c('result', {'xmlns': 'urn:xmpp:mam:2', 'queryid': queryid, 'id':'28482-98726-73623'})
  969. .c('forwarded', {'xmlns':'urn:xmpp:forward:0'})
  970. .c('delay', {'xmlns':'urn:xmpp:delay', 'stamp':'2010-07-10T23:08:25Z'}).up()
  971. .c('message', {
  972. 'xmlns':'jabber:client',
  973. 'to': contact_jid,
  974. 'from': _converse.bare_jid,
  975. 'type':'chat' })
  976. .c('body').t("Call me but love, and I'll be new baptized;");
  977. _converse.connection._dataRecv(test_utils.createRequest(msg1));
  978. const msg2 = $msg({'id':'aeb213', 'to': contact_jid})
  979. .c('result', {'xmlns': 'urn:xmpp:mam:2', 'queryid': queryid, 'id':'28482-98726-73624'})
  980. .c('forwarded', {'xmlns':'urn:xmpp:forward:0'})
  981. .c('delay', {'xmlns':'urn:xmpp:delay', 'stamp':'2010-07-10T23:18:25Z'}).up()
  982. .c('message', {
  983. 'xmlns':'jabber:client',
  984. 'to': contact_jid,
  985. 'from': _converse.bare_jid,
  986. 'type':'chat' })
  987. .c('body').t("Henceforth I never will be Romeo.");
  988. _converse.connection._dataRecv(test_utils.createRequest(msg2));
  989. const stanza = $iq({'type': 'result', 'id': sent_stanza.getAttribute('id')})
  990. .c('fin', {'xmlns': 'urn:xmpp:mam:2', 'complete': true})
  991. .c('set', {'xmlns': 'http://jabber.org/protocol/rsm'})
  992. .c('first', {'index': '0'}).t('28482-98726-73623').up()
  993. .c('last').t('28482-98726-73624').up()
  994. .c('count').t('2');
  995. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  996. await u.waitUntil(() => view.model.messages.length === 2, 500);
  997. err_message = view.el.querySelector('.message.chat-error');
  998. expect(err_message).toBe(null);
  999. done();
  1000. }));
  1001. });
  1002. });
  1003. }));