retractions.js 67 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156
  1. (function (root, factory) {
  2. define([
  3. "jasmine",
  4. "mock",
  5. "test-utils"
  6. ], factory);
  7. } (this, function (jasmine, mock, test_utils) {
  8. "use strict";
  9. const { Strophe, $iq } = converse.env;
  10. const u = converse.env.utils;
  11. async function sendAndThenRetractMessage (_converse, view) {
  12. view.model.sendMessage('hello world');
  13. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg__text').length === 1);
  14. const msg_obj = view.model.messages.last();
  15. const reflection_stanza = u.toStanza(`
  16. <message xmlns="jabber:client"
  17. from="${msg_obj.get('from')}"
  18. to="${_converse.connection.jid}"
  19. type="groupchat">
  20. <msg_body>${msg_obj.get('message')}</msg_body>
  21. <stanza-id xmlns="urn:xmpp:sid:0"
  22. id="5f3dbc5e-e1d3-4077-a492-693f3769c7ad"
  23. by="lounge@montague.lit"/>
  24. <origin-id xmlns="urn:xmpp:sid:0" id="${msg_obj.get('origin_id')}"/>
  25. </message>`);
  26. await view.model.queueMessage(reflection_stanza);
  27. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg__body.chat-msg__body--received').length, 500);
  28. const retract_button = await u.waitUntil(() => view.el.querySelector('.chat-msg__content .chat-msg__action-retract'));
  29. retract_button.click();
  30. await u.waitUntil(() => u.isVisible(document.querySelector('#converse-modals .modal')));
  31. const submit_button = document.querySelector('#converse-modals .modal button[type="submit"]');
  32. submit_button.click();
  33. const sent_stanzas = _converse.connection.sent_stanzas;
  34. return u.waitUntil(() => sent_stanzas.filter(s => s.querySelector('message apply-to[xmlns="urn:xmpp:fasten:0"]')).pop());
  35. }
  36. describe("Message Retractions", function () {
  37. describe("A groupchat message retraction", function () {
  38. it("is not applied if it's not from the right author",
  39. mock.initConverse(
  40. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  41. async function (done, _converse) {
  42. const muc_jid = 'lounge@montague.lit';
  43. const features = [...mock.default_muc_features, Strophe.NS.MODERATE];
  44. await test_utils.openAndEnterChatRoom(_converse, muc_jid, 'romeo', features);
  45. const received_stanza = u.toStanza(`
  46. <message to='${_converse.jid}' from='${muc_jid}/eve' type='groupchat' id='${_converse.connection.getUniqueId()}'>
  47. <body>Hello world</body>
  48. <stanza-id xmlns='urn:xmpp:sid:0' id='stanza-id-1' by='${muc_jid}'/>
  49. </message>
  50. `);
  51. const view = _converse.api.chatviews.get(muc_jid);
  52. await view.model.queueMessage(received_stanza);
  53. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg').length === 1);
  54. expect(view.model.messages.at(0).get('retracted')).toBeFalsy();
  55. expect(view.model.messages.at(0).get('is_ephemeral')).toBeFalsy();
  56. const retraction_stanza = u.toStanza(`
  57. <message type="groupchat" id='retraction-id-1' from="${muc_jid}/mallory" to="${muc_jid}/romeo">
  58. <apply-to id="stanza-id-1" xmlns="urn:xmpp:fasten:0">
  59. <retract xmlns="urn:xmpp:message-retract:0" />
  60. </apply-to>
  61. </message>
  62. `);
  63. spyOn(view.model, 'handleRetraction').and.callThrough();
  64. _converse.connection._dataRecv(test_utils.createRequest(retraction_stanza));
  65. await u.waitUntil(() => view.model.handleRetraction.calls.count() === 1);
  66. expect(await view.model.handleRetraction.calls.first().returnValue).toBe(true);
  67. expect(view.el.querySelectorAll('.chat-msg').length).toBe(1);
  68. expect(view.model.messages.length).toBe(2);
  69. expect(view.model.messages.at(1).get('retracted')).toBeTruthy();
  70. expect(view.model.messages.at(1).get('is_ephemeral')).toBeFalsy();
  71. expect(view.model.messages.at(1).get('dangling_retraction')).toBe(true);
  72. expect(view.model.messages.at(0).get('retracted')).toBeFalsy();
  73. expect(view.model.messages.at(0).get('is_ephemeral')).toBeFalsy();
  74. done();
  75. }));
  76. it("can be received before the message it pertains to",
  77. mock.initConverse(
  78. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  79. async function (done, _converse) {
  80. const date = (new Date()).toISOString();
  81. const muc_jid = 'lounge@montague.lit';
  82. const features = [...mock.default_muc_features, Strophe.NS.MODERATE];
  83. await test_utils.openAndEnterChatRoom(_converse, muc_jid, 'romeo', features);
  84. const retraction_stanza = u.toStanza(`
  85. <message type="groupchat" id='retraction-id-1' from="${muc_jid}/eve" to="${muc_jid}/romeo">
  86. <apply-to id="origin-id-1" xmlns="urn:xmpp:fasten:0">
  87. <retract by="${muc_jid}/eve" xmlns="urn:xmpp:message-retract:0" />
  88. </apply-to>
  89. </message>
  90. `);
  91. const view = _converse.api.chatviews.get(muc_jid);
  92. spyOn(converse.env.log, 'warn');
  93. spyOn(view.model, 'handleRetraction').and.callThrough();
  94. _converse.connection._dataRecv(test_utils.createRequest(retraction_stanza));
  95. await u.waitUntil(() => view.model.handleRetraction.calls.count() === 1);
  96. await u.waitUntil(() => view.model.messages.length === 1);
  97. expect(await view.model.handleRetraction.calls.first().returnValue).toBe(true);
  98. expect(view.model.messages.length).toBe(1);
  99. expect(view.model.messages.at(0).get('retracted')).toBeTruthy();
  100. expect(view.model.messages.at(0).get('dangling_retraction')).toBe(true);
  101. const received_stanza = u.toStanza(`
  102. <message to='${_converse.jid}' from='${muc_jid}/eve' type='groupchat' id='${_converse.connection.getUniqueId()}'>
  103. <body>Hello world</body>
  104. <delay xmlns='urn:xmpp:delay' stamp='${date}'/>
  105. <stanza-id xmlns='urn:xmpp:sid:0' id='stanza-id-1' by='${muc_jid}'/>
  106. <origin-id xmlns="urn:xmpp:sid:0" id="origin-id-1"/>
  107. </message>
  108. `);
  109. _converse.connection._dataRecv(test_utils.createRequest(received_stanza));
  110. await u.waitUntil(() => view.model.handleRetraction.calls.count() === 2);
  111. expect(view.el.querySelectorAll('.chat-msg').length).toBe(0);
  112. expect(view.model.messages.length).toBe(1);
  113. const message = view.model.messages.at(0)
  114. expect(message.get('retracted')).toBeTruthy();
  115. expect(message.get('dangling_retraction')).toBe(false);
  116. expect(message.get('origin_id')).toBe('origin-id-1');
  117. expect(message.get(`stanza_id ${muc_jid}`)).toBe('stanza-id-1');
  118. expect(message.get('time')).toBe(date);
  119. expect(message.get('type')).toBe('groupchat');
  120. expect(await view.model.handleRetraction.calls.all().pop().returnValue).toBe(true);
  121. done();
  122. }));
  123. });
  124. describe("A groupchat message moderator retraction", function () {
  125. it("can be received before the message it pertains to",
  126. mock.initConverse(
  127. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  128. async function (done, _converse) {
  129. const date = (new Date()).toISOString();
  130. const muc_jid = 'lounge@montague.lit';
  131. const features = [...mock.default_muc_features, Strophe.NS.MODERATE];
  132. await test_utils.openAndEnterChatRoom(_converse, muc_jid, 'romeo', features);
  133. const retraction_stanza = u.toStanza(`
  134. <message xmlns="jabber:client" from="${muc_jid}" type="groupchat" id="retraction-id-1">
  135. <apply-to xmlns="urn:xmpp:fasten:0" id="stanza-id-1">
  136. <moderated xmlns="urn:xmpp:message-moderate:0" by="${muc_jid}/madison">
  137. <retract xmlns="urn:xmpp:message-retract:0"/>
  138. <reason>Insults</reason>
  139. </moderated>
  140. </apply-to>
  141. </message>
  142. `);
  143. const view = _converse.api.chatviews.get(muc_jid);
  144. spyOn(converse.env.log, 'warn');
  145. spyOn(view.model, 'handleModeration').and.callThrough();
  146. _converse.connection._dataRecv(test_utils.createRequest(retraction_stanza));
  147. await u.waitUntil(() => view.model.handleModeration.calls.count() === 1);
  148. await u.waitUntil(() => view.model.messages.length === 1);
  149. expect(await view.model.handleModeration.calls.first().returnValue).toBe(true);
  150. expect(view.model.messages.length).toBe(1);
  151. expect(view.model.messages.at(0).get('moderated')).toBe('retracted');
  152. expect(view.model.messages.at(0).get('dangling_moderation')).toBe(true);
  153. const received_stanza = u.toStanza(`
  154. <message to='${_converse.jid}' from='${muc_jid}/eve' type='groupchat' id='${_converse.connection.getUniqueId()}'>
  155. <body>Hello world</body>
  156. <delay xmlns='urn:xmpp:delay' stamp='${date}'/>
  157. <stanza-id xmlns='urn:xmpp:sid:0' id='stanza-id-1' by='${muc_jid}'/>
  158. </message>
  159. `);
  160. _converse.connection._dataRecv(test_utils.createRequest(received_stanza));
  161. await u.waitUntil(() => view.model.handleModeration.calls.count() === 2);
  162. expect(view.el.querySelectorAll('.chat-msg').length).toBe(1);
  163. expect(view.model.messages.length).toBe(1);
  164. const message = view.model.messages.at(0)
  165. expect(message.get('moderated')).toBe('retracted');
  166. expect(message.get('dangling_moderation')).toBe(false);
  167. expect(message.get(`stanza_id ${muc_jid}`)).toBe('stanza-id-1');
  168. expect(message.get('time')).toBe(date);
  169. expect(message.get('type')).toBe('groupchat');
  170. expect(await view.model.handleModeration.calls.all().pop().returnValue).toBe(true);
  171. done();
  172. }));
  173. });
  174. describe("A message retraction", function () {
  175. it("can be received before the message it pertains to",
  176. mock.initConverse(
  177. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  178. async function (done, _converse) {
  179. const date = (new Date()).toISOString();
  180. await test_utils.waitForRoster(_converse, 'current', 1);
  181. await test_utils.waitUntilDiscoConfirmed(_converse, _converse.bare_jid, [], [Strophe.NS.SID]);
  182. const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@montague.lit';
  183. const view = await test_utils.openChatBoxFor(_converse, contact_jid);
  184. spyOn(view.model, 'handleRetraction').and.callThrough();
  185. const retraction_stanza = u.toStanza(`
  186. <message id="${u.getUniqueId()}"
  187. to="${_converse.bare_jid}"
  188. from="${contact_jid}"
  189. type="chat"
  190. xmlns="jabber:client">
  191. <apply-to id="2e972ea0-0050-44b7-a830-f6638a2595b3" xmlns="urn:xmpp:fasten:0">
  192. <retract xmlns="urn:xmpp:message-retract:0"/>
  193. </apply-to>
  194. </message>
  195. `);
  196. const promise = new Promise(resolve => _converse.api.listen.on('messageAdded', resolve));
  197. _converse.connection._dataRecv(test_utils.createRequest(retraction_stanza));
  198. await u.waitUntil(() => view.model.messages.length === 1);
  199. await promise;
  200. const message = view.model.messages.at(0);
  201. expect(message.get('dangling_retraction')).toBe(true);
  202. expect(message.get('is_ephemeral')).toBe(false);
  203. expect(message.get('retracted')).toBeTruthy();
  204. expect(view.el.querySelectorAll('.chat-msg').length).toBe(0);
  205. const stanza = u.toStanza(`
  206. <message xmlns="jabber:client"
  207. to="${_converse.bare_jid}"
  208. type="chat"
  209. id="2e972ea0-0050-44b7-a830-f6638a2595b3"
  210. from="${contact_jid}">
  211. <body>Hello world</body>
  212. <delay xmlns='urn:xmpp:delay' stamp='${date}'/>
  213. <markable xmlns="urn:xmpp:chat-markers:0"/>
  214. <origin-id xmlns="urn:xmpp:sid:0" id="2e972ea0-0050-44b7-a830-f6638a2595b3"/>
  215. <stanza-id xmlns="urn:xmpp:sid:0" id="IxVDLJ0RYbWcWvqC" by="${_converse.bare_jid}"/>
  216. </message>`);
  217. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  218. await u.waitUntil(() => view.model.handleRetraction.calls.count() === 2);
  219. expect(view.model.messages.length).toBe(1);
  220. expect(message.get('retracted')).toBeTruthy();
  221. expect(message.get('dangling_retraction')).toBe(false);
  222. expect(message.get('origin_id')).toBe('2e972ea0-0050-44b7-a830-f6638a2595b3');
  223. expect(message.get('time')).toBe(date);
  224. expect(message.get('type')).toBe('chat');
  225. done();
  226. }));
  227. });
  228. describe("A Received Chat Message", function () {
  229. it("can be followed up by a retraction",
  230. mock.initConverse(
  231. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  232. async function (done, _converse) {
  233. await test_utils.waitForRoster(_converse, 'current', 1);
  234. await test_utils.waitUntilDiscoConfirmed(_converse, _converse.bare_jid, [], [Strophe.NS.SID]);
  235. const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@montague.lit';
  236. const view = await test_utils.openChatBoxFor(_converse, contact_jid);
  237. let stanza = u.toStanza(`
  238. <message xmlns="jabber:client"
  239. to="${_converse.bare_jid}"
  240. type="chat"
  241. id="29132ea0-0121-2897-b121-36638c259554"
  242. from="${contact_jid}">
  243. <body>😊</body>
  244. <markable xmlns="urn:xmpp:chat-markers:0"/>
  245. <origin-id xmlns="urn:xmpp:sid:0" id="29132ea0-0121-2897-b121-36638c259554"/>
  246. <stanza-id xmlns="urn:xmpp:sid:0" id="kxViLhgbnNMcWv10" by="${_converse.bare_jid}"/>
  247. </message>`);
  248. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  249. await u.waitUntil(() => view.model.messages.length === 1);
  250. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg').length === 1);
  251. stanza = u.toStanza(`
  252. <message xmlns="jabber:client"
  253. to="${_converse.bare_jid}"
  254. type="chat"
  255. id="2e972ea0-0050-44b7-a830-f6638a2595b3"
  256. from="${contact_jid}">
  257. <body>This message will be retracted</body>
  258. <markable xmlns="urn:xmpp:chat-markers:0"/>
  259. <origin-id xmlns="urn:xmpp:sid:0" id="2e972ea0-0050-44b7-a830-f6638a2595b3"/>
  260. <stanza-id xmlns="urn:xmpp:sid:0" id="IxVDLJ0RYbWcWvqC" by="${_converse.bare_jid}"/>
  261. </message>`);
  262. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  263. await u.waitUntil(() => view.model.messages.length === 2);
  264. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg').length === 2);
  265. const retraction_stanza = u.toStanza(`
  266. <message id="${u.getUniqueId()}"
  267. to="${_converse.bare_jid}"
  268. from="${contact_jid}"
  269. type="chat"
  270. xmlns="jabber:client">
  271. <apply-to id="2e972ea0-0050-44b7-a830-f6638a2595b3" xmlns="urn:xmpp:fasten:0">
  272. <retract xmlns="urn:xmpp:message-retract:0"/>
  273. </apply-to>
  274. </message>
  275. `);
  276. _converse.connection._dataRecv(test_utils.createRequest(retraction_stanza));
  277. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg--retracted').length === 1);
  278. expect(view.model.messages.length).toBe(2);
  279. const message = view.model.messages.at(1);
  280. expect(message.get('retracted')).toBeTruthy();
  281. expect(view.el.querySelectorAll('.chat-msg--retracted').length).toBe(1);
  282. const msg_el = view.el.querySelector('.chat-msg--retracted .chat-msg__message');
  283. expect(msg_el.textContent.trim()).toBe('Mercutio has removed this message');
  284. expect(u.hasClass('chat-msg--followup', view.el.querySelector('.chat-msg--retracted'))).toBe(true);
  285. done();
  286. }));
  287. });
  288. describe("A Sent Chat Message", function () {
  289. it("can be retracted by its author",
  290. mock.initConverse(
  291. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  292. async function (done, _converse) {
  293. await test_utils.waitForRoster(_converse, 'current', 1);
  294. const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@montague.lit';
  295. const view = await test_utils.openChatBoxFor(_converse, contact_jid);
  296. view.model.sendMessage('hello world');
  297. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg').length === 1);
  298. const message = view.model.messages.at(0);
  299. expect(view.model.messages.length).toBe(1);
  300. expect(message.get('retracted')).toBeFalsy();
  301. expect(message.get('editable')).toBeTruthy();
  302. const retract_button = await u.waitUntil(() => view.el.querySelector('.chat-msg__content .chat-msg__action-retract'));
  303. retract_button.click();
  304. await u.waitUntil(() => u.isVisible(document.querySelector('#converse-modals .modal')));
  305. const submit_button = document.querySelector('#converse-modals .modal button[type="submit"]');
  306. submit_button.click();
  307. const sent_stanzas = _converse.connection.sent_stanzas;
  308. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg--retracted').length === 1);
  309. const msg_obj = view.model.messages.at(0);
  310. const retraction_stanza = await u.waitUntil(() => sent_stanzas.filter(s => s.querySelector('message apply-to[xmlns="urn:xmpp:fasten:0"]')).pop());
  311. expect(Strophe.serialize(retraction_stanza)).toBe(
  312. `<message id="${retraction_stanza.getAttribute('id')}" to="${contact_jid}" type="chat" xmlns="jabber:client">`+
  313. `<store xmlns="urn:xmpp:hints"/>`+
  314. `<apply-to id="${msg_obj.get('origin_id')}" xmlns="urn:xmpp:fasten:0">`+
  315. `<retract xmlns="urn:xmpp:message-retract:0"/>`+
  316. `</apply-to>`+
  317. `</message>`);
  318. expect(view.model.messages.length).toBe(1);
  319. expect(message.get('retracted')).toBeTruthy();
  320. expect(message.get('editable')).toBeFalsy();
  321. expect(view.el.querySelectorAll('.chat-msg--retracted').length).toBe(1);
  322. const el = view.el.querySelector('.chat-msg--retracted .chat-msg__message');
  323. expect(el.textContent.trim()).toBe('Romeo Montague has removed this message');
  324. done();
  325. }));
  326. });
  327. describe("A Received Groupchat Message", function () {
  328. it("can be followed up by a retraction by the author",
  329. mock.initConverse(
  330. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  331. async function (done, _converse) {
  332. const muc_jid = 'lounge@montague.lit';
  333. const features = [...mock.default_muc_features, Strophe.NS.MODERATE];
  334. await test_utils.openAndEnterChatRoom(_converse, muc_jid, 'romeo', features);
  335. const received_stanza = u.toStanza(`
  336. <message to='${_converse.jid}' from='${muc_jid}/eve' type='groupchat' id='${_converse.connection.getUniqueId()}'>
  337. <body>Hello world</body>
  338. <stanza-id xmlns='urn:xmpp:sid:0' id='stanza-id-1' by='${muc_jid}'/>
  339. <origin-id xmlns='urn:xmpp:sid:0' id='origin-id-1' by='${muc_jid}'/>
  340. </message>
  341. `);
  342. const view = _converse.api.chatviews.get(muc_jid);
  343. await view.model.queueMessage(received_stanza);
  344. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg').length === 1);
  345. expect(view.model.messages.at(0).get('retracted')).toBeFalsy();
  346. expect(view.model.messages.at(0).get('is_ephemeral')).toBeFalsy();
  347. const retraction_stanza = u.toStanza(`
  348. <message type="groupchat" id='retraction-id-1' from="${muc_jid}/eve" to="${muc_jid}/romeo">
  349. <apply-to id="origin-id-1" xmlns="urn:xmpp:fasten:0">
  350. <retract by="${muc_jid}/eve" xmlns="urn:xmpp:message-retract:0" />
  351. </apply-to>
  352. </message>
  353. `);
  354. _converse.connection._dataRecv(test_utils.createRequest(retraction_stanza));
  355. // We opportunistically save the message as retracted, even before receiving the retraction message
  356. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg--retracted').length === 1);
  357. expect(view.model.messages.length).toBe(1);
  358. expect(view.model.messages.at(0).get('retracted')).toBeTruthy();
  359. expect(view.model.messages.at(0).get('editable')).toBe(false);
  360. expect(view.el.querySelectorAll('.chat-msg--retracted').length).toBe(1);
  361. const msg_el = view.el.querySelector('.chat-msg--retracted .chat-msg__message');
  362. expect(msg_el.textContent.trim()).toBe('eve has removed this message');
  363. expect(msg_el.querySelector('.chat-msg--retracted q')).toBe(null);
  364. done();
  365. }));
  366. it("can be retracted by a moderator, with the IQ response received before the retraction message",
  367. mock.initConverse(
  368. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  369. async function (done, _converse) {
  370. const muc_jid = 'lounge@montague.lit';
  371. const features = [...mock.default_muc_features, Strophe.NS.MODERATE];
  372. await test_utils.openAndEnterChatRoom(_converse, muc_jid, 'romeo', features);
  373. const view = _converse.api.chatviews.get(muc_jid);
  374. const occupant = view.model.getOwnOccupant();
  375. expect(occupant.get('role')).toBe('moderator');
  376. const received_stanza = u.toStanza(`
  377. <message to='${_converse.jid}' from='${muc_jid}/mallory' type='groupchat' id='${_converse.connection.getUniqueId()}'>
  378. <body>Visit this site to get free Bitcoin!</body>
  379. <stanza-id xmlns='urn:xmpp:sid:0' id='stanza-id-1' by='${muc_jid}'/>
  380. </message>
  381. `);
  382. await view.model.queueMessage(received_stanza);
  383. await u.waitUntil(() => view.model.messages.length === 1);
  384. expect(view.model.messages.at(0).get('retracted')).toBeFalsy();
  385. const reason = "This content is inappropriate for this forum!"
  386. const retract_button = await u.waitUntil(() => view.el.querySelector('.chat-msg__content .chat-msg__action-retract'));
  387. retract_button.click();
  388. await u.waitUntil(() => u.isVisible(document.querySelector('#converse-modals .modal')));
  389. const reason_input = document.querySelector('#converse-modals .modal input[name="reason"]');
  390. reason_input.value = 'This content is inappropriate for this forum!';
  391. const submit_button = document.querySelector('#converse-modals .modal button[type="submit"]');
  392. submit_button.click();
  393. const sent_IQs = _converse.connection.IQ_stanzas;
  394. const stanza = await u.waitUntil(() => sent_IQs.filter(iq => iq.querySelector('iq apply-to[xmlns="urn:xmpp:fasten:0"]')).pop());
  395. const message = view.model.messages.at(0);
  396. const stanza_id = message.get(`stanza_id ${view.model.get('jid')}`);
  397. expect(Strophe.serialize(stanza)).toBe(
  398. `<iq id="${stanza.getAttribute('id')}" to="${muc_jid}" type="set" xmlns="jabber:client">`+
  399. `<apply-to id="${stanza_id}" xmlns="urn:xmpp:fasten:0">`+
  400. `<moderate xmlns="urn:xmpp:message-moderate:0">`+
  401. `<retract xmlns="urn:xmpp:message-retract:0"/>`+
  402. `<reason>This content is inappropriate for this forum!</reason>`+
  403. `</moderate>`+
  404. `</apply-to>`+
  405. `</iq>`);
  406. const result_iq = $iq({'from': muc_jid, 'id': stanza.getAttribute('id'), 'to': _converse.bare_jid, 'type': 'result'});
  407. _converse.connection._dataRecv(test_utils.createRequest(result_iq));
  408. // We opportunistically save the message as retracted, even before receiving the retraction message
  409. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg--retracted').length === 1);
  410. expect(view.model.messages.length).toBe(1);
  411. expect(view.model.messages.at(0).get('moderated')).toBe('retracted');
  412. expect(view.model.messages.at(0).get('moderation_reason')).toBe(reason);
  413. expect(view.model.messages.at(0).get('is_ephemeral')).toBe(false);
  414. expect(view.model.messages.at(0).get('editable')).toBe(false);
  415. expect(view.el.querySelectorAll('.chat-msg--retracted').length).toBe(1);
  416. const msg_el = view.el.querySelector('.chat-msg--retracted .chat-msg__message');
  417. expect(msg_el.firstElementChild.textContent.trim()).toBe('romeo has removed this message');
  418. const qel = msg_el.querySelector('q');
  419. expect(qel.textContent.trim()).toBe('This content is inappropriate for this forum!');
  420. // The server responds with a retraction message
  421. const retraction = u.toStanza(`
  422. <message type="groupchat" id='retraction-id-1' from="${muc_jid}" to="${muc_jid}/romeo">
  423. <apply-to id="${stanza_id}" xmlns="urn:xmpp:fasten:0">
  424. <moderated by='${_converse.bare_jid}' xmlns='urn:xmpp:message-moderate:0'>
  425. <retract xmlns='urn:xmpp:message-retract:0' />
  426. <reason>${reason}</reason>
  427. </moderated>
  428. </apply-to>
  429. </message>`);
  430. await view.model.queueMessage(retraction);
  431. expect(view.model.messages.length).toBe(1);
  432. expect(view.model.messages.at(0).get('moderated')).toBe('retracted');
  433. expect(view.model.messages.at(0).get('moderation_reason')).toBe(reason);
  434. expect(view.model.messages.at(0).get('is_ephemeral')).toBe(false);
  435. expect(view.model.messages.at(0).get('editable')).toBe(false);
  436. done();
  437. }));
  438. it("can not be retracted if the MUC doesn't support message moderation",
  439. mock.initConverse(
  440. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  441. async function (done, _converse) {
  442. const muc_jid = 'lounge@montague.lit';
  443. await test_utils.openAndEnterChatRoom(_converse, muc_jid, 'romeo');
  444. const view = _converse.api.chatviews.get(muc_jid);
  445. const occupant = view.model.getOwnOccupant();
  446. expect(occupant.get('role')).toBe('moderator');
  447. const received_stanza = u.toStanza(`
  448. <message to='${_converse.jid}' from='${muc_jid}/mallory' type='groupchat' id='${_converse.connection.getUniqueId()}'>
  449. <body>Visit this site to get free Bitcoin!</body>
  450. <stanza-id xmlns='urn:xmpp:sid:0' id='stanza-id-1' by='${muc_jid}'/>
  451. </message>
  452. `);
  453. await view.model.queueMessage(received_stanza);
  454. await u.waitUntil(() => view.el.querySelector('.chat-msg__content'));
  455. expect(view.el.querySelector('.chat-msg__content .chat-msg__action-retract')).toBe(null);
  456. const result = await view.model.canModerateMessages();
  457. expect(result).toBe(false);
  458. done();
  459. }));
  460. it("can be retracted by a moderator, with the retraction message received before the IQ response",
  461. mock.initConverse(
  462. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  463. async function (done, _converse) {
  464. const muc_jid = 'lounge@montague.lit';
  465. const features = [...mock.default_muc_features, Strophe.NS.MODERATE];
  466. await test_utils.openAndEnterChatRoom(_converse, muc_jid, 'romeo', features);
  467. const view = _converse.api.chatviews.get(muc_jid);
  468. const occupant = view.model.getOwnOccupant();
  469. expect(occupant.get('role')).toBe('moderator');
  470. const received_stanza = u.toStanza(`
  471. <message to='${_converse.jid}' from='${muc_jid}/mallory' type='groupchat' id='${_converse.connection.getUniqueId()}'>
  472. <body>Visit this site to get free Bitcoin!</body>
  473. <stanza-id xmlns='urn:xmpp:sid:0' id='stanza-id-1' by='${muc_jid}'/>
  474. </message>
  475. `);
  476. await view.model.queueMessage(received_stanza);
  477. await u.waitUntil(() => view.model.messages.length === 1);
  478. expect(view.model.messages.length).toBe(1);
  479. const retract_button = await u.waitUntil(() => view.el.querySelector('.chat-msg__content .chat-msg__action-retract'));
  480. retract_button.click();
  481. await u.waitUntil(() => u.isVisible(document.querySelector('#converse-modals .modal')));
  482. const reason_input = document.querySelector('#converse-modals .modal input[name="reason"]');
  483. const reason = "This content is inappropriate for this forum!"
  484. reason_input.value = reason;
  485. const submit_button = document.querySelector('#converse-modals .modal button[type="submit"]');
  486. submit_button.click();
  487. const sent_IQs = _converse.connection.IQ_stanzas;
  488. const stanza = await u.waitUntil(() => sent_IQs.filter(iq => iq.querySelector('iq apply-to[xmlns="urn:xmpp:fasten:0"]')).pop());
  489. const message = view.model.messages.at(0);
  490. const stanza_id = message.get(`stanza_id ${view.model.get('jid')}`);
  491. // The server responds with a retraction message
  492. const retraction = u.toStanza(`
  493. <message type="groupchat" id='retraction-id-1' from="${muc_jid}" to="${muc_jid}/romeo">
  494. <apply-to id="${stanza_id}" xmlns="urn:xmpp:fasten:0">
  495. <moderated by='${_converse.bare_jid}' xmlns='urn:xmpp:message-moderate:0'>
  496. <retract xmlns='urn:xmpp:message-retract:0' />
  497. <reason>${reason}</reason>
  498. </moderated>
  499. </apply-to>
  500. </message>`);
  501. await view.model.queueMessage(retraction);
  502. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg--retracted').length === 1);
  503. expect(view.model.messages.length).toBe(1);
  504. expect(view.model.messages.at(0).get('moderated')).toBe('retracted');
  505. expect(view.el.querySelectorAll('.chat-msg--retracted').length).toBe(1);
  506. const msg_el = view.el.querySelector('.chat-msg--retracted .chat-msg__message div');
  507. expect(msg_el.textContent).toBe('romeo has removed this message');
  508. const qel = view.el.querySelector('.chat-msg--retracted .chat-msg__message q');
  509. expect(qel.textContent).toBe('This content is inappropriate for this forum!');
  510. const result_iq = $iq({'from': muc_jid, 'id': stanza.getAttribute('id'), 'to': _converse.bare_jid, 'type': 'result'});
  511. _converse.connection._dataRecv(test_utils.createRequest(result_iq));
  512. expect(view.model.messages.length).toBe(1);
  513. expect(view.model.messages.at(0).get('moderated')).toBe('retracted');
  514. expect(view.model.messages.at(0).get('moderated_by')).toBe(_converse.bare_jid);
  515. expect(view.model.messages.at(0).get('moderation_reason')).toBe(reason);
  516. expect(view.model.messages.at(0).get('editable')).toBe(false);
  517. done();
  518. }));
  519. });
  520. describe("A Sent Groupchat Message", function () {
  521. it("can be retracted by its author",
  522. mock.initConverse(
  523. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  524. async function (done, _converse) {
  525. const muc_jid = 'lounge@montague.lit';
  526. const features = [...mock.default_muc_features, Strophe.NS.MODERATE];
  527. await test_utils.openAndEnterChatRoom(_converse, muc_jid, 'romeo', features);
  528. const view = _converse.api.chatviews.get(muc_jid);
  529. const occupant = view.model.getOwnOccupant();
  530. expect(occupant.get('role')).toBe('moderator');
  531. occupant.save('role', 'member');
  532. const retraction_stanza = await sendAndThenRetractMessage(_converse, view);
  533. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg--retracted').length === 1);
  534. const msg_obj = view.model.messages.last();
  535. expect(Strophe.serialize(retraction_stanza)).toBe(
  536. `<message id="${retraction_stanza.getAttribute('id')}" to="${muc_jid}" type="groupchat" xmlns="jabber:client">`+
  537. `<store xmlns="urn:xmpp:hints"/>`+
  538. `<apply-to id="${msg_obj.get('origin_id')}" xmlns="urn:xmpp:fasten:0">`+
  539. `<retract xmlns="urn:xmpp:message-retract:0"/>`+
  540. `</apply-to>`+
  541. `</message>`);
  542. const message = view.model.messages.last();
  543. expect(message.get('retracted')).toBeTruthy();
  544. expect(message.get('is_ephemeral')).toBe(false);
  545. expect(message.get('editable')).toBeFalsy();
  546. const stanza_id = message.get(`stanza_id ${muc_jid}`);
  547. // The server responds with a retraction message
  548. const reflection = u.toStanza(`
  549. <message type="groupchat" id="${retraction_stanza.getAttribute('id')}" from="${muc_jid}" to="${muc_jid}/romeo">
  550. <apply-to id="${stanza_id}" xmlns="urn:xmpp:fasten:0">
  551. <moderated by='${_converse.bare_jid}' xmlns='urn:xmpp:message-moderate:0'>
  552. <retract xmlns='urn:xmpp:message-retract:0' />
  553. </moderated>
  554. </apply-to>
  555. </message>`);
  556. spyOn(view.model, 'handleRetraction').and.callThrough();
  557. _converse.connection._dataRecv(test_utils.createRequest(reflection));
  558. await u.waitUntil(() => view.model.handleRetraction.calls.count() === 1);
  559. expect(view.model.messages.length).toBe(2);
  560. expect(view.model.messages.last().get('retracted')).toBeTruthy();
  561. expect(view.model.messages.last().get('is_ephemeral')).toBe(false);
  562. expect(view.model.messages.last().get('editable')).toBe(false);
  563. expect(view.el.querySelectorAll('.chat-msg--retracted').length).toBe(1);
  564. const el = view.el.querySelector('.chat-msg--retracted .chat-msg__message div');
  565. expect(el.textContent).toBe('romeo has removed this message');
  566. done();
  567. }));
  568. it("can be retracted by its author, causing an error message in response",
  569. mock.initConverse(
  570. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  571. async function (done, _converse) {
  572. const muc_jid = 'lounge@montague.lit';
  573. const features = [...mock.default_muc_features, Strophe.NS.MODERATE];
  574. await test_utils.openAndEnterChatRoom(_converse, muc_jid, 'romeo', features);
  575. const view = _converse.api.chatviews.get(muc_jid);
  576. const occupant = view.model.getOwnOccupant();
  577. expect(occupant.get('role')).toBe('moderator');
  578. occupant.save('role', 'member');
  579. await u.waitUntil(() =>
  580. Array.from(view.el.querySelectorAll('.chat-info__message')).pop()?.textContent.trim() ===
  581. "romeo is no longer a moderator"
  582. );
  583. const retraction_stanza = await sendAndThenRetractMessage(_converse, view);
  584. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg--retracted').length === 1);
  585. expect(view.model.messages.length).toBe(2);
  586. expect(view.model.messages.last().get('retracted')).toBeTruthy();
  587. const el = view.el.querySelector('.chat-msg--retracted .chat-msg__message div');
  588. expect(el.textContent.trim()).toBe('romeo has removed this message');
  589. const message = view.model.messages.last();
  590. const stanza_id = message.get(`stanza_id ${view.model.get('jid')}`);
  591. // The server responds with an error message
  592. const error = u.toStanza(`
  593. <message type="error" id="${retraction_stanza.getAttribute('id')}" from="${muc_jid}" to="${view.model.get('jid')}/romeo">
  594. <error by='${muc_jid}' type='auth'>
  595. <forbidden xmlns='urn:ietf:params:xml:ns:xmpp-stanzas'/>
  596. </error>
  597. <apply-to id="${stanza_id}" xmlns="urn:xmpp:fasten:0">
  598. <moderated by='${_converse.bare_jid}' xmlns='urn:xmpp:message-moderate:0'>
  599. <retract xmlns='urn:xmpp:message-retract:0' />
  600. </moderated>
  601. </apply-to>
  602. </message>`);
  603. _converse.connection._dataRecv(test_utils.createRequest(error));
  604. await u.waitUntil(() => view.el.querySelectorAll('.chat-error').length === 1);
  605. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg--retracted').length === 0);
  606. expect(view.model.messages.length).toBe(2);
  607. expect(view.model.messages.last().get('retracted')).toBeFalsy();
  608. expect(view.model.messages.last().get('is_ephemeral')).toBeFalsy();
  609. expect(view.model.messages.last().get('editable')).toBeTruthy();
  610. expect(view.el.querySelectorAll('.chat-error').length).toBe(1);
  611. const errmsg = view.el.querySelector('.chat-error');
  612. expect(errmsg.textContent).toBe("Sorry, something went wrong while trying to retract your message.");
  613. done();
  614. }));
  615. it("can be retracted by its author, causing a timeout error in response",
  616. mock.initConverse(
  617. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  618. async function (done, _converse) {
  619. _converse.STANZA_TIMEOUT = 1;
  620. const muc_jid = 'lounge@montague.lit';
  621. const features = [...mock.default_muc_features, Strophe.NS.MODERATE];
  622. await test_utils.openAndEnterChatRoom(_converse, muc_jid, 'romeo', features);
  623. const view = _converse.api.chatviews.get(muc_jid);
  624. const occupant = view.model.getOwnOccupant();
  625. expect(occupant.get('role')).toBe('moderator');
  626. occupant.save('role', 'member');
  627. await u.waitUntil(() =>
  628. Array.from(view.el.querySelectorAll('.chat-info__message')).pop()?.textContent.trim() ===
  629. "romeo is no longer a moderator"
  630. );
  631. await sendAndThenRetractMessage(_converse, view);
  632. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg--retracted').length === 1);
  633. expect(view.model.messages.length).toBe(2);
  634. expect(view.model.messages.last().get('retracted')).toBeTruthy();
  635. const el = view.el.querySelector('.chat-msg--retracted .chat-msg__message div');
  636. expect(el.textContent.trim()).toBe('romeo has removed this message');
  637. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg').length === 1);
  638. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg--retracted').length === 0);
  639. expect(view.model.messages.length).toBe(2);
  640. expect(view.model.messages.last().get('retracted')).toBeFalsy();
  641. expect(view.model.messages.last().get('is_ephemeral')).toBeFalsy();
  642. expect(view.model.messages.last().get('editable')).toBeTruthy();
  643. const error_messages = view.el.querySelectorAll('.chat-error');
  644. expect(error_messages.length).toBe(2);
  645. expect(error_messages[0].textContent).toBe("Sorry, something went wrong while trying to retract your message.");
  646. expect(error_messages[1].textContent).toBe("Timeout Error: No response from server");
  647. done();
  648. }));
  649. it("can be retracted by a moderator",
  650. mock.initConverse(
  651. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  652. async function (done, _converse) {
  653. const muc_jid = 'lounge@montague.lit';
  654. const features = [...mock.default_muc_features, Strophe.NS.MODERATE];
  655. await test_utils.openAndEnterChatRoom(_converse, muc_jid, 'romeo', features);
  656. const view = _converse.api.chatviews.get(muc_jid);
  657. const occupant = view.model.getOwnOccupant();
  658. expect(occupant.get('role')).toBe('moderator');
  659. view.model.sendMessage('Visit this site to get free bitcoin');
  660. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg').length === 1);
  661. const stanza_id = 'retraction-id-1';
  662. const msg_obj = view.model.messages.at(0);
  663. const reflection_stanza = u.toStanza(`
  664. <message xmlns="jabber:client"
  665. from="${msg_obj.get('from')}"
  666. to="${_converse.connection.jid}"
  667. type="groupchat">
  668. <msg_body>${msg_obj.get('message')}</msg_body>
  669. <stanza-id xmlns="urn:xmpp:sid:0"
  670. id="${stanza_id}"
  671. by="lounge@montague.lit"/>
  672. <origin-id xmlns="urn:xmpp:sid:0" id="${msg_obj.get('origin_id')}"/>
  673. </message>`);
  674. await view.model.queueMessage(reflection_stanza);
  675. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg__body.chat-msg__body--received').length, 500);
  676. expect(view.model.messages.length).toBe(1);
  677. expect(view.model.messages.at(0).get('editable')).toBe(true);
  678. // The server responds with a retraction message
  679. const reason = "This content is inappropriate for this forum!"
  680. const retraction = u.toStanza(`
  681. <message type="groupchat" id='retraction-id-1' from="${muc_jid}" to="${muc_jid}/romeo">
  682. <apply-to id="${stanza_id}" xmlns="urn:xmpp:fasten:0">
  683. <moderated by='${_converse.bare_jid}' xmlns='urn:xmpp:message-moderate:0'>
  684. <retract xmlns='urn:xmpp:message-retract:0' />
  685. <reason>${reason}</reason>
  686. </moderated>
  687. </apply-to>
  688. </message>`);
  689. await view.model.queueMessage(retraction);
  690. expect(view.model.messages.length).toBe(1);
  691. expect(view.model.messages.at(0).get('moderated')).toBe('retracted');
  692. expect(view.model.messages.at(0).get('moderation_reason')).toBe(reason);
  693. expect(view.model.messages.at(0).get('is_ephemeral')).toBe(false);
  694. expect(view.model.messages.at(0).get('editable')).toBe(false);
  695. done();
  696. }));
  697. it("can be retracted by the sender if they're a moderator",
  698. mock.initConverse(
  699. ['rosterGroupsFetched', 'chatBoxesFetched'], {'allow_message_retraction': 'moderator'},
  700. async function (done, _converse) {
  701. const muc_jid = 'lounge@montague.lit';
  702. const features = [...mock.default_muc_features, Strophe.NS.MODERATE];
  703. await test_utils.openAndEnterChatRoom(_converse, muc_jid, 'romeo', features);
  704. const view = _converse.api.chatviews.get(muc_jid);
  705. const occupant = view.model.getOwnOccupant();
  706. expect(occupant.get('role')).toBe('moderator');
  707. view.model.sendMessage('Visit this site to get free bitcoin');
  708. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg').length === 1);
  709. const stanza_id = 'retraction-id-1';
  710. const msg_obj = view.model.messages.at(0);
  711. const reflection_stanza = u.toStanza(`
  712. <message xmlns="jabber:client"
  713. from="${msg_obj.get('from')}"
  714. to="${_converse.connection.jid}"
  715. type="groupchat">
  716. <msg_body>${msg_obj.get('message')}</msg_body>
  717. <stanza-id xmlns="urn:xmpp:sid:0"
  718. id="${stanza_id}"
  719. by="lounge@montague.lit"/>
  720. <origin-id xmlns="urn:xmpp:sid:0" id="${msg_obj.get('origin_id')}"/>
  721. </message>`);
  722. await view.model.queueMessage(reflection_stanza);
  723. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg__body.chat-msg__body--received').length, 500);
  724. expect(view.model.messages.length).toBe(1);
  725. expect(view.model.messages.at(0).get('editable')).toBe(true);
  726. const retract_button = await u.waitUntil(() => view.msgs_container.querySelector('.chat-msg__content .chat-msg__action-retract'));
  727. retract_button.click();
  728. await u.waitUntil(() => u.isVisible(document.querySelector('#converse-modals .modal')));
  729. const submit_button = document.querySelector('#converse-modals .modal button[type="submit"]');
  730. submit_button.click();
  731. const sent_IQs = _converse.connection.IQ_stanzas;
  732. const stanza = await u.waitUntil(() => sent_IQs.filter(iq => iq.querySelector('iq apply-to[xmlns="urn:xmpp:fasten:0"]')).pop());
  733. expect(Strophe.serialize(stanza)).toBe(
  734. `<iq id="${stanza.getAttribute('id')}" to="${muc_jid}" type="set" xmlns="jabber:client">`+
  735. `<apply-to id="${stanza_id}" xmlns="urn:xmpp:fasten:0">`+
  736. `<moderate xmlns="urn:xmpp:message-moderate:0">`+
  737. `<retract xmlns="urn:xmpp:message-retract:0"/>`+
  738. `<reason></reason>`+
  739. `</moderate>`+
  740. `</apply-to>`+
  741. `</iq>`);
  742. const result_iq = $iq({'from': muc_jid, 'id': stanza.getAttribute('id'), 'to': _converse.bare_jid, 'type': 'result'});
  743. _converse.connection._dataRecv(test_utils.createRequest(result_iq));
  744. // We opportunistically save the message as retracted, even before receiving the retraction message
  745. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg--retracted').length === 1);
  746. expect(view.model.messages.length).toBe(1);
  747. expect(view.model.messages.at(0).get('moderated')).toBe('retracted');
  748. expect(view.model.messages.at(0).get('moderation_reason')).toBe(undefined);
  749. expect(view.model.messages.at(0).get('is_ephemeral')).toBe(false);
  750. expect(view.model.messages.at(0).get('editable')).toBe(false);
  751. expect(view.el.querySelectorAll('.chat-msg--retracted').length).toBe(1);
  752. const msg_el = view.el.querySelector('.chat-msg--retracted .chat-msg__message');
  753. expect(msg_el.firstElementChild.textContent.trim()).toBe('romeo has removed this message');
  754. expect(msg_el.querySelector('q')).toBe(null);
  755. // The server responds with a retraction message
  756. const retraction = u.toStanza(`
  757. <message type="groupchat" id='retraction-id-1' from="${muc_jid}" to="${muc_jid}/romeo">
  758. <apply-to id="${stanza_id}" xmlns="urn:xmpp:fasten:0">
  759. <moderated by='${_converse.bare_jid}' xmlns='urn:xmpp:message-moderate:0'>
  760. <retract xmlns='urn:xmpp:message-retract:0' />
  761. </moderated>
  762. </apply-to>
  763. </message>`);
  764. await view.model.queueMessage(retraction);
  765. expect(view.model.messages.length).toBe(1);
  766. expect(view.model.messages.at(0).get('moderated')).toBe('retracted');
  767. expect(view.model.messages.at(0).get('moderation_reason')).toBe(undefined);
  768. expect(view.model.messages.at(0).get('is_ephemeral')).toBe(false);
  769. expect(view.model.messages.at(0).get('editable')).toBe(false);
  770. done();
  771. }));
  772. });
  773. describe("when archived", function () {
  774. it("may be returned as a tombstone message",
  775. mock.initConverse(
  776. ['discoInitialized'], {},
  777. async function (done, _converse) {
  778. await test_utils.waitForRoster(_converse, 'current', 1);
  779. const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@montague.lit';
  780. await test_utils.openChatBoxFor(_converse, contact_jid);
  781. await test_utils.waitUntilDiscoConfirmed(_converse, _converse.bare_jid, null, [Strophe.NS.MAM]);
  782. const sent_IQs = _converse.connection.IQ_stanzas;
  783. const stanza = await u.waitUntil(() => sent_IQs.filter(iq => iq.querySelector(`iq[type="set"] query[xmlns="${Strophe.NS.MAM}"]`)).pop());
  784. const queryid = stanza.querySelector('query').getAttribute('queryid');
  785. const view = _converse.chatboxviews.get(contact_jid);
  786. const first_id = u.getUniqueId();
  787. spyOn(view.model, 'handleRetraction').and.callThrough();
  788. const first_message = u.toStanza(`
  789. <message id='${u.getUniqueId()}' to='${_converse.jid}'>
  790. <result xmlns='urn:xmpp:mam:2' queryid='${queryid}' id="${first_id}">
  791. <forwarded xmlns='urn:xmpp:forward:0'>
  792. <delay xmlns='urn:xmpp:delay' stamp='2019-09-20T23:01:15Z'/>
  793. <message type="chat" from="${contact_jid}" to="${_converse.bare_jid}" id="message-id-0">
  794. <origin-id xmlns='urn:xmpp:sid:0' id="origin-id-0"/>
  795. <body>😊</body>
  796. </message>
  797. </forwarded>
  798. </result>
  799. </message>
  800. `);
  801. _converse.connection._dataRecv(test_utils.createRequest(first_message));
  802. const tombstone = u.toStanza(`
  803. <message id='${u.getUniqueId()}' to='${_converse.jid}'>
  804. <result xmlns='urn:xmpp:mam:2' queryid='${queryid}' id="${u.getUniqueId()}">
  805. <forwarded xmlns='urn:xmpp:forward:0'>
  806. <delay xmlns='urn:xmpp:delay' stamp='2019-09-20T23:08:25Z'/>
  807. <message type="chat" from="${contact_jid}" to="${_converse.bare_jid}" id="message-id-1">
  808. <origin-id xmlns='urn:xmpp:sid:0' id="origin-id-1"/>
  809. <retracted stamp='2019-09-20T23:09:32Z' xmlns='urn:xmpp:message-retract:0'/>
  810. </message>
  811. </forwarded>
  812. </result>
  813. </message>
  814. `);
  815. _converse.connection._dataRecv(test_utils.createRequest(tombstone));
  816. const last_id = u.getUniqueId();
  817. const retraction = u.toStanza(`
  818. <message id='${u.getUniqueId()}' to='${_converse.jid}'>
  819. <result xmlns='urn:xmpp:mam:2' queryid='${queryid}' id="${last_id}">
  820. <forwarded xmlns='urn:xmpp:forward:0'>
  821. <delay xmlns='urn:xmpp:delay' stamp='2019-09-20T23:08:25Z'/>
  822. <message from="${contact_jid}" to='${_converse.bare_jid}' id='retract-message-1'>
  823. <apply-to id="origin-id-1" xmlns="urn:xmpp:fasten:0">
  824. <retract xmlns='urn:xmpp:message-retract:0'/>
  825. </apply-to>
  826. </message>
  827. </forwarded>
  828. </result>
  829. </message>
  830. `);
  831. _converse.connection._dataRecv(test_utils.createRequest(retraction));
  832. const iq_result = $iq({'type': 'result', 'id': stanza.getAttribute('id')})
  833. .c('fin', {'xmlns': 'urn:xmpp:mam:2'})
  834. .c('set', {'xmlns': 'http://jabber.org/protocol/rsm'})
  835. .c('first', {'index': '0'}).t(first_id).up()
  836. .c('last').t(last_id).up()
  837. .c('count').t('2');
  838. _converse.connection._dataRecv(test_utils.createRequest(iq_result));
  839. await u.waitUntil(() => view.model.handleRetraction.calls.count() === 3);
  840. expect(view.model.messages.length).toBe(2);
  841. const message = view.model.messages.at(1);
  842. expect(message.get('retracted')).toBeTruthy();
  843. expect(message.get('is_tombstone')).toBe(true);
  844. expect(await view.model.handleRetraction.calls.first().returnValue).toBe(false);
  845. expect(await view.model.handleRetraction.calls.all()[1].returnValue).toBe(false);
  846. expect(await view.model.handleRetraction.calls.all()[2].returnValue).toBe(true);
  847. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg').length === 2);
  848. expect(view.el.querySelectorAll('.chat-msg--retracted').length).toBe(1);
  849. const el = view.el.querySelector('.chat-msg--retracted .chat-msg__message div');
  850. expect(el.textContent.trim()).toBe('Mercutio has removed this message');
  851. expect(u.hasClass('chat-msg--followup', el.parentElement)).toBe(false);
  852. done();
  853. }));
  854. it("may be returned as a tombstone groupchat message",
  855. mock.initConverse(
  856. ['discoInitialized'], {},
  857. async function (done, _converse) {
  858. const muc_jid = 'lounge@montague.lit';
  859. const features = [...mock.default_muc_features, Strophe.NS.MODERATE];
  860. await test_utils.openAndEnterChatRoom(_converse, muc_jid, 'romeo', features);
  861. const view = _converse.chatboxviews.get(muc_jid);
  862. const sent_IQs = _converse.connection.IQ_stanzas;
  863. const stanza = await u.waitUntil(() => sent_IQs.filter(iq => iq.querySelector(`iq[type="set"] query[xmlns="${Strophe.NS.MAM}"]`)).pop());
  864. const queryid = stanza.querySelector('query').getAttribute('queryid');
  865. const first_id = u.getUniqueId();
  866. const tombstone = u.toStanza(`
  867. <message id="${u.getUniqueId()}" to="${_converse.jid}" from="${muc_jid}">
  868. <result xmlns="urn:xmpp:mam:2" queryid="${queryid}" id="stanza-id">
  869. <forwarded xmlns="urn:xmpp:forward:0">
  870. <delay xmlns="urn:xmpp:delay" stamp="2019-09-20T23:08:25Z"/>
  871. <message type="groupchat" from="${muc_jid}/eve" to="${_converse.bare_jid}" id="message-id-1">
  872. <origin-id xmlns='urn:xmpp:sid:0' id="origin-id-1"/>
  873. <retracted stamp="2019-09-20T23:09:32Z" xmlns="urn:xmpp:message-retract:0"/>
  874. </message>
  875. </forwarded>
  876. </result>
  877. </message>
  878. `);
  879. spyOn(view.model, 'handleRetraction').and.callThrough();
  880. const promise = new Promise(resolve => _converse.api.listen.once('messageAdded', resolve));
  881. _converse.connection._dataRecv(test_utils.createRequest(tombstone));
  882. const last_id = u.getUniqueId();
  883. const retraction = u.toStanza(`
  884. <message id="${u.getUniqueId()}" to="${_converse.jid}" from="${muc_jid}">
  885. <result xmlns="urn:xmpp:mam:2" queryid="${queryid}" id="${last_id}">
  886. <forwarded xmlns="urn:xmpp:forward:0">
  887. <delay xmlns="urn:xmpp:delay" stamp="2019-09-20T23:08:25Z"/>
  888. <message type="groupchat" from="${muc_jid}/eve" to="${_converse.bare_jid}" id="retract-message-1">
  889. <apply-to id="origin-id-1" xmlns="urn:xmpp:fasten:0">
  890. <retract xmlns="urn:xmpp:message-retract:0"/>
  891. </apply-to>
  892. </message>
  893. </forwarded>
  894. </result>
  895. </message>
  896. `);
  897. _converse.connection._dataRecv(test_utils.createRequest(retraction));
  898. const iq_result = $iq({'type': 'result', 'id': stanza.getAttribute('id')})
  899. .c('fin', {'xmlns': 'urn:xmpp:mam:2'})
  900. .c('set', {'xmlns': 'http://jabber.org/protocol/rsm'})
  901. .c('first', {'index': '0'}).t(first_id).up()
  902. .c('last').t(last_id).up()
  903. .c('count').t('2');
  904. _converse.connection._dataRecv(test_utils.createRequest(iq_result));
  905. await promise;
  906. expect(view.model.messages.length).toBe(1);
  907. let message = view.model.messages.at(0);
  908. expect(message.get('retracted')).toBeTruthy();
  909. expect(message.get('is_tombstone')).toBe(true);
  910. await u.waitUntil(() => view.model.handleRetraction.calls.count() === 2);
  911. expect(await view.model.handleRetraction.calls.first().returnValue).toBe(false);
  912. expect(await view.model.handleRetraction.calls.all()[1].returnValue).toBe(true);
  913. expect(view.model.messages.length).toBe(1);
  914. message = view.model.messages.at(0);
  915. expect(message.get('retracted')).toBeTruthy();
  916. expect(message.get('is_tombstone')).toBe(true);
  917. expect(view.el.querySelectorAll('.chat-msg').length).toBe(1);
  918. expect(view.el.querySelectorAll('.chat-msg--retracted').length).toBe(1);
  919. const el = view.el.querySelector('.chat-msg--retracted .chat-msg__message div');
  920. expect(el.textContent.trim()).toBe('eve has removed this message');
  921. done();
  922. }));
  923. it("may be returned as a tombstone moderated groupchat message",
  924. mock.initConverse(
  925. ['discoInitialized', 'chatBoxesFetched'], {},
  926. async function (done, _converse) {
  927. const muc_jid = 'lounge@montague.lit';
  928. const features = [...mock.default_muc_features, Strophe.NS.MODERATE];
  929. await test_utils.openAndEnterChatRoom(_converse, muc_jid, 'romeo', features);
  930. const view = _converse.chatboxviews.get(muc_jid);
  931. const sent_IQs = _converse.connection.IQ_stanzas;
  932. const stanza = await u.waitUntil(() => sent_IQs.filter(iq => iq.querySelector(`iq[type="set"] query[xmlns="${Strophe.NS.MAM}"]`)).pop());
  933. const queryid = stanza.querySelector('query').getAttribute('queryid');
  934. const first_id = u.getUniqueId();
  935. const tombstone = u.toStanza(`
  936. <message id="${u.getUniqueId()}" to="${_converse.jid}" from="${muc_jid}">
  937. <result xmlns="urn:xmpp:mam:2" queryid="${queryid}" id="stanza-id">
  938. <forwarded xmlns="urn:xmpp:forward:0">
  939. <delay xmlns="urn:xmpp:delay" stamp="2019-09-20T23:08:25Z"/>
  940. <message type="groupchat" from="${muc_jid}/eve" to="${_converse.bare_jid}" id="message-id-1">
  941. <moderated by="${muc_jid}/bob" stamp="2019-09-20T23:09:32Z" xmlns='urn:xmpp:message-moderate:0'>
  942. <retracted xmlns="urn:xmpp:message-retract:0"/>
  943. <reason>This message contains inappropriate content</reason>
  944. </moderated>
  945. </message>
  946. </forwarded>
  947. </result>
  948. </message>
  949. `);
  950. spyOn(view.model, 'handleModeration').and.callThrough();
  951. const promise = new Promise(resolve => _converse.api.listen.once('messageAdded', resolve));
  952. _converse.connection._dataRecv(test_utils.createRequest(tombstone));
  953. const last_id = u.getUniqueId();
  954. const retraction = u.toStanza(`
  955. <message id="${u.getUniqueId()}" to="${_converse.jid}" from="${muc_jid}">
  956. <result xmlns="urn:xmpp:mam:2" queryid="${queryid}" id="${last_id}">
  957. <forwarded xmlns="urn:xmpp:forward:0">
  958. <delay xmlns="urn:xmpp:delay" stamp="2019-09-20T23:08:25Z"/>
  959. <message type="groupchat" from="${muc_jid}" to="${_converse.bare_jid}" id="retract-message-1">
  960. <apply-to id="stanza-id" xmlns="urn:xmpp:fasten:0">
  961. <moderated by="${muc_jid}/bob" xmlns='urn:xmpp:message-moderate:0'>
  962. <retract xmlns="urn:xmpp:message-retract:0"/>
  963. <reason>This message contains inappropriate content</reason>
  964. </moderated>
  965. </apply-to>
  966. </message>
  967. </forwarded>
  968. </result>
  969. </message>
  970. `);
  971. _converse.connection._dataRecv(test_utils.createRequest(retraction));
  972. const iq_result = $iq({'type': 'result', 'id': stanza.getAttribute('id')})
  973. .c('fin', {'xmlns': 'urn:xmpp:mam:2'})
  974. .c('set', {'xmlns': 'http://jabber.org/protocol/rsm'})
  975. .c('first', {'index': '0'}).t(first_id).up()
  976. .c('last').t(last_id).up()
  977. .c('count').t('2');
  978. _converse.connection._dataRecv(test_utils.createRequest(iq_result));
  979. await promise;
  980. expect(view.model.messages.length).toBe(1);
  981. let message = view.model.messages.at(0);
  982. expect(message.get('retracted')).toBeTruthy();
  983. expect(message.get('is_tombstone')).toBe(true);
  984. await u.waitUntil(() => view.model.handleModeration.calls.count() === 2);
  985. expect(await view.model.handleModeration.calls.first().returnValue).toBe(false);
  986. expect(await view.model.handleModeration.calls.all()[1].returnValue).toBe(true);
  987. expect(view.model.messages.length).toBe(1);
  988. message = view.model.messages.at(0);
  989. expect(message.get('retracted')).toBeTruthy();
  990. expect(message.get('is_tombstone')).toBe(true);
  991. expect(message.get('moderation_reason')).toBe("This message contains inappropriate content");
  992. expect(view.el.querySelectorAll('.chat-msg').length).toBe(1);
  993. expect(view.el.querySelectorAll('.chat-msg--retracted').length).toBe(1);
  994. const el = view.el.querySelector('.chat-msg--retracted .chat-msg__message div');
  995. expect(el.textContent.trim()).toBe('A moderator has removed this message');
  996. const qel = view.el.querySelector('.chat-msg--retracted .chat-msg__message q');
  997. expect(qel.textContent.trim()).toBe('This message contains inappropriate content');
  998. done();
  999. }));
  1000. });
  1001. })
  1002. }));