2
0

retractions.js 61 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063
  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').length === 1);
  14. const msg_obj = view.model.messages.at(0);
  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.at(0);
  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.at(0);
  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(1);
  560. expect(view.model.messages.at(0).get('retracted')).toBeTruthy();
  561. expect(view.model.messages.at(0).get('is_ephemeral')).toBe(false);
  562. expect(view.model.messages.at(0).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. const retraction_stanza = await sendAndThenRetractMessage(_converse, view);
  580. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg--retracted').length === 1);
  581. expect(view.model.messages.length).toBe(1);
  582. expect(view.model.messages.at(0).get('retracted')).toBeTruthy();
  583. const el = view.el.querySelector('.chat-msg--retracted .chat-msg__message div');
  584. expect(el.textContent.trim()).toBe('romeo has removed this message');
  585. const message = view.model.messages.at(0);
  586. const stanza_id = message.get(`stanza_id ${view.model.get('jid')}`);
  587. // The server responds with an error message
  588. const error = u.toStanza(`
  589. <message type="error" id="${retraction_stanza.getAttribute('id')}" from="${muc_jid}" to="${view.model.get('jid')}/romeo">
  590. <error by='${muc_jid}' type='auth'>
  591. <forbidden xmlns='urn:ietf:params:xml:ns:xmpp-stanzas'/>
  592. </error>
  593. <apply-to id="${stanza_id}" xmlns="urn:xmpp:fasten:0">
  594. <moderated by='${_converse.bare_jid}' xmlns='urn:xmpp:message-moderate:0'>
  595. <retract xmlns='urn:xmpp:message-retract:0' />
  596. </moderated>
  597. </apply-to>
  598. </message>`);
  599. _converse.connection._dataRecv(test_utils.createRequest(error));
  600. await u.waitUntil(() => view.el.querySelectorAll('.chat-error').length === 1);
  601. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg--retracted').length === 0);
  602. expect(view.model.messages.length).toBe(1);
  603. expect(view.model.messages.at(0).get('retracted')).toBeFalsy();
  604. expect(view.model.messages.at(0).get('is_ephemeral')).toBeFalsy();
  605. expect(view.model.messages.at(0).get('editable')).toBeTruthy();
  606. expect(view.el.querySelectorAll('.chat-error').length).toBe(1);
  607. const errmsg = view.el.querySelector('.chat-error');
  608. expect(errmsg.textContent).toBe("Sorry, something went wrong while trying to retract your message.");
  609. done();
  610. }));
  611. it("can be retracted by its author, causing an timeout error in response",
  612. mock.initConverse(
  613. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  614. async function (done, _converse) {
  615. _converse.STANZA_TIMEOUT = 1;
  616. const muc_jid = 'lounge@montague.lit';
  617. const features = [...mock.default_muc_features, Strophe.NS.MODERATE];
  618. await test_utils.openAndEnterChatRoom(_converse, muc_jid, 'romeo', features);
  619. const view = _converse.api.chatviews.get(muc_jid);
  620. const occupant = view.model.getOwnOccupant();
  621. expect(occupant.get('role')).toBe('moderator');
  622. occupant.save('role', 'member');
  623. await sendAndThenRetractMessage(_converse, view);
  624. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg--retracted').length === 1);
  625. expect(view.model.messages.length).toBe(1);
  626. expect(view.model.messages.at(0).get('retracted')).toBeTruthy();
  627. const el = view.el.querySelector('.chat-msg--retracted .chat-msg__message div');
  628. expect(el.textContent.trim()).toBe('romeo has removed this message');
  629. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg').length === 1);
  630. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg--retracted').length === 0);
  631. expect(view.model.messages.length).toBe(1);
  632. expect(view.model.messages.at(0).get('retracted')).toBeFalsy();
  633. expect(view.model.messages.at(0).get('is_ephemeral')).toBeFalsy();
  634. expect(view.model.messages.at(0).get('editable')).toBeTruthy();
  635. const error_messages = view.el.querySelectorAll('.chat-error');
  636. expect(error_messages.length).toBe(2);
  637. expect(error_messages[0].textContent).toBe("Sorry, something went wrong while trying to retract your message.");
  638. expect(error_messages[1].textContent).toBe("Timeout Error: No response from server");
  639. done();
  640. }));
  641. it("can be retracted by a moderator",
  642. mock.initConverse(
  643. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  644. async function (done, _converse) {
  645. const muc_jid = 'lounge@montague.lit';
  646. const features = [...mock.default_muc_features, Strophe.NS.MODERATE];
  647. await test_utils.openAndEnterChatRoom(_converse, muc_jid, 'romeo', features);
  648. const view = _converse.api.chatviews.get(muc_jid);
  649. const occupant = view.model.getOwnOccupant();
  650. expect(occupant.get('role')).toBe('moderator');
  651. view.model.sendMessage('Visit this site to get free bitcoin');
  652. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg').length === 1);
  653. const stanza_id = 'retraction-id-1';
  654. const msg_obj = view.model.messages.at(0);
  655. const reflection_stanza = u.toStanza(`
  656. <message xmlns="jabber:client"
  657. from="${msg_obj.get('from')}"
  658. to="${_converse.connection.jid}"
  659. type="groupchat">
  660. <msg_body>${msg_obj.get('message')}</msg_body>
  661. <stanza-id xmlns="urn:xmpp:sid:0"
  662. id="${stanza_id}"
  663. by="lounge@montague.lit"/>
  664. <origin-id xmlns="urn:xmpp:sid:0" id="${msg_obj.get('origin_id')}"/>
  665. </message>`);
  666. await view.model.queueMessage(reflection_stanza);
  667. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg__body.chat-msg__body--received').length, 500);
  668. expect(view.model.messages.length).toBe(1);
  669. expect(view.model.messages.at(0).get('editable')).toBe(true);
  670. // The server responds with a retraction message
  671. const reason = "This content is inappropriate for this forum!"
  672. const retraction = u.toStanza(`
  673. <message type="groupchat" id='retraction-id-1' from="${muc_jid}" to="${muc_jid}/romeo">
  674. <apply-to id="${stanza_id}" xmlns="urn:xmpp:fasten:0">
  675. <moderated by='${_converse.bare_jid}' xmlns='urn:xmpp:message-moderate:0'>
  676. <retract xmlns='urn:xmpp:message-retract:0' />
  677. <reason>${reason}</reason>
  678. </moderated>
  679. </apply-to>
  680. </message>`);
  681. await view.model.queueMessage(retraction);
  682. expect(view.model.messages.length).toBe(1);
  683. expect(view.model.messages.at(0).get('moderated')).toBe('retracted');
  684. expect(view.model.messages.at(0).get('moderation_reason')).toBe(reason);
  685. expect(view.model.messages.at(0).get('is_ephemeral')).toBe(false);
  686. expect(view.model.messages.at(0).get('editable')).toBe(false);
  687. done();
  688. }));
  689. });
  690. describe("when archived", function () {
  691. it("may be returned as a tombstone message",
  692. mock.initConverse(
  693. ['discoInitialized'], {},
  694. async function (done, _converse) {
  695. await test_utils.waitForRoster(_converse, 'current', 1);
  696. const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@montague.lit';
  697. await test_utils.openChatBoxFor(_converse, contact_jid);
  698. await test_utils.waitUntilDiscoConfirmed(_converse, _converse.bare_jid, null, [Strophe.NS.MAM]);
  699. const sent_IQs = _converse.connection.IQ_stanzas;
  700. const stanza = await u.waitUntil(() => sent_IQs.filter(iq => iq.querySelector(`iq[type="set"] query[xmlns="${Strophe.NS.MAM}"]`)).pop());
  701. const queryid = stanza.querySelector('query').getAttribute('queryid');
  702. const view = _converse.chatboxviews.get(contact_jid);
  703. const first_id = u.getUniqueId();
  704. spyOn(view.model, 'handleRetraction').and.callThrough();
  705. const first_message = u.toStanza(`
  706. <message id='${u.getUniqueId()}' to='${_converse.jid}'>
  707. <result xmlns='urn:xmpp:mam:2' queryid='${queryid}' id="${first_id}">
  708. <forwarded xmlns='urn:xmpp:forward:0'>
  709. <delay xmlns='urn:xmpp:delay' stamp='2019-09-20T23:01:15Z'/>
  710. <message type="chat" from="${contact_jid}" to="${_converse.bare_jid}" id="message-id-0">
  711. <origin-id xmlns='urn:xmpp:sid:0' id="origin-id-0"/>
  712. <body>😊</body>
  713. </message>
  714. </forwarded>
  715. </result>
  716. </message>
  717. `);
  718. _converse.connection._dataRecv(test_utils.createRequest(first_message));
  719. const tombstone = u.toStanza(`
  720. <message id='${u.getUniqueId()}' to='${_converse.jid}'>
  721. <result xmlns='urn:xmpp:mam:2' queryid='${queryid}' id="${u.getUniqueId()}">
  722. <forwarded xmlns='urn:xmpp:forward:0'>
  723. <delay xmlns='urn:xmpp:delay' stamp='2019-09-20T23:08:25Z'/>
  724. <message type="chat" from="${contact_jid}" to="${_converse.bare_jid}" id="message-id-1">
  725. <origin-id xmlns='urn:xmpp:sid:0' id="origin-id-1"/>
  726. <retracted stamp='2019-09-20T23:09:32Z' xmlns='urn:xmpp:message-retract:0'/>
  727. </message>
  728. </forwarded>
  729. </result>
  730. </message>
  731. `);
  732. _converse.connection._dataRecv(test_utils.createRequest(tombstone));
  733. const last_id = u.getUniqueId();
  734. const retraction = u.toStanza(`
  735. <message id='${u.getUniqueId()}' to='${_converse.jid}'>
  736. <result xmlns='urn:xmpp:mam:2' queryid='${queryid}' id="${last_id}">
  737. <forwarded xmlns='urn:xmpp:forward:0'>
  738. <delay xmlns='urn:xmpp:delay' stamp='2019-09-20T23:08:25Z'/>
  739. <message from="${contact_jid}" to='${_converse.bare_jid}' id='retract-message-1'>
  740. <apply-to id="origin-id-1" xmlns="urn:xmpp:fasten:0">
  741. <retract xmlns='urn:xmpp:message-retract:0'/>
  742. </apply-to>
  743. </message>
  744. </forwarded>
  745. </result>
  746. </message>
  747. `);
  748. _converse.connection._dataRecv(test_utils.createRequest(retraction));
  749. const iq_result = $iq({'type': 'result', 'id': stanza.getAttribute('id')})
  750. .c('fin', {'xmlns': 'urn:xmpp:mam:2'})
  751. .c('set', {'xmlns': 'http://jabber.org/protocol/rsm'})
  752. .c('first', {'index': '0'}).t(first_id).up()
  753. .c('last').t(last_id).up()
  754. .c('count').t('2');
  755. _converse.connection._dataRecv(test_utils.createRequest(iq_result));
  756. await u.waitUntil(() => view.model.handleRetraction.calls.count() === 3);
  757. expect(view.model.messages.length).toBe(2);
  758. const message = view.model.messages.at(1);
  759. expect(message.get('retracted')).toBeTruthy();
  760. expect(message.get('is_tombstone')).toBe(true);
  761. expect(await view.model.handleRetraction.calls.first().returnValue).toBe(false);
  762. expect(await view.model.handleRetraction.calls.all()[1].returnValue).toBe(false);
  763. expect(await view.model.handleRetraction.calls.all()[2].returnValue).toBe(true);
  764. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg').length === 2);
  765. expect(view.el.querySelectorAll('.chat-msg--retracted').length).toBe(1);
  766. const el = view.el.querySelector('.chat-msg--retracted .chat-msg__message div');
  767. expect(el.textContent.trim()).toBe('Mercutio has removed this message');
  768. expect(u.hasClass('chat-msg--followup', el.parentElement)).toBe(false);
  769. done();
  770. }));
  771. it("may be returned as a tombstone groupchat message",
  772. mock.initConverse(
  773. ['discoInitialized'], {},
  774. async function (done, _converse) {
  775. const muc_jid = 'lounge@montague.lit';
  776. const features = [...mock.default_muc_features, Strophe.NS.MODERATE];
  777. await test_utils.openAndEnterChatRoom(_converse, muc_jid, 'romeo', features);
  778. const view = _converse.chatboxviews.get(muc_jid);
  779. const sent_IQs = _converse.connection.IQ_stanzas;
  780. const stanza = await u.waitUntil(() => sent_IQs.filter(iq => iq.querySelector(`iq[type="set"] query[xmlns="${Strophe.NS.MAM}"]`)).pop());
  781. const queryid = stanza.querySelector('query').getAttribute('queryid');
  782. const first_id = u.getUniqueId();
  783. const tombstone = u.toStanza(`
  784. <message id="${u.getUniqueId()}" to="${_converse.jid}" from="${muc_jid}">
  785. <result xmlns="urn:xmpp:mam:2" queryid="${queryid}" id="stanza-id">
  786. <forwarded xmlns="urn:xmpp:forward:0">
  787. <delay xmlns="urn:xmpp:delay" stamp="2019-09-20T23:08:25Z"/>
  788. <message type="groupchat" from="${muc_jid}/eve" to="${_converse.bare_jid}" id="message-id-1">
  789. <origin-id xmlns='urn:xmpp:sid:0' id="origin-id-1"/>
  790. <retracted stamp="2019-09-20T23:09:32Z" xmlns="urn:xmpp:message-retract:0"/>
  791. </message>
  792. </forwarded>
  793. </result>
  794. </message>
  795. `);
  796. spyOn(view.model, 'handleRetraction').and.callThrough();
  797. const promise = new Promise(resolve => _converse.api.listen.once('messageAdded', resolve));
  798. _converse.connection._dataRecv(test_utils.createRequest(tombstone));
  799. const last_id = u.getUniqueId();
  800. const retraction = u.toStanza(`
  801. <message id="${u.getUniqueId()}" to="${_converse.jid}" from="${muc_jid}">
  802. <result xmlns="urn:xmpp:mam:2" queryid="${queryid}" id="${last_id}">
  803. <forwarded xmlns="urn:xmpp:forward:0">
  804. <delay xmlns="urn:xmpp:delay" stamp="2019-09-20T23:08:25Z"/>
  805. <message type="groupchat" from="${muc_jid}/eve" to="${_converse.bare_jid}" id="retract-message-1">
  806. <apply-to id="origin-id-1" xmlns="urn:xmpp:fasten:0">
  807. <retract xmlns="urn:xmpp:message-retract:0"/>
  808. </apply-to>
  809. </message>
  810. </forwarded>
  811. </result>
  812. </message>
  813. `);
  814. _converse.connection._dataRecv(test_utils.createRequest(retraction));
  815. const iq_result = $iq({'type': 'result', 'id': stanza.getAttribute('id')})
  816. .c('fin', {'xmlns': 'urn:xmpp:mam:2'})
  817. .c('set', {'xmlns': 'http://jabber.org/protocol/rsm'})
  818. .c('first', {'index': '0'}).t(first_id).up()
  819. .c('last').t(last_id).up()
  820. .c('count').t('2');
  821. _converse.connection._dataRecv(test_utils.createRequest(iq_result));
  822. await promise;
  823. expect(view.model.messages.length).toBe(1);
  824. let message = view.model.messages.at(0);
  825. expect(message.get('retracted')).toBeTruthy();
  826. expect(message.get('is_tombstone')).toBe(true);
  827. await u.waitUntil(() => view.model.handleRetraction.calls.count() === 2);
  828. expect(await view.model.handleRetraction.calls.first().returnValue).toBe(false);
  829. expect(await view.model.handleRetraction.calls.all()[1].returnValue).toBe(true);
  830. expect(view.model.messages.length).toBe(1);
  831. message = view.model.messages.at(0);
  832. expect(message.get('retracted')).toBeTruthy();
  833. expect(message.get('is_tombstone')).toBe(true);
  834. expect(view.el.querySelectorAll('.chat-msg').length).toBe(1);
  835. expect(view.el.querySelectorAll('.chat-msg--retracted').length).toBe(1);
  836. const el = view.el.querySelector('.chat-msg--retracted .chat-msg__message div');
  837. expect(el.textContent.trim()).toBe('eve has removed this message');
  838. done();
  839. }));
  840. it("may be returned as a tombstone moderated groupchat message",
  841. mock.initConverse(
  842. ['discoInitialized', 'chatBoxesFetched'], {},
  843. async function (done, _converse) {
  844. const muc_jid = 'lounge@montague.lit';
  845. const features = [...mock.default_muc_features, Strophe.NS.MODERATE];
  846. await test_utils.openAndEnterChatRoom(_converse, muc_jid, 'romeo', features);
  847. const view = _converse.chatboxviews.get(muc_jid);
  848. const sent_IQs = _converse.connection.IQ_stanzas;
  849. const stanza = await u.waitUntil(() => sent_IQs.filter(iq => iq.querySelector(`iq[type="set"] query[xmlns="${Strophe.NS.MAM}"]`)).pop());
  850. const queryid = stanza.querySelector('query').getAttribute('queryid');
  851. const first_id = u.getUniqueId();
  852. const tombstone = u.toStanza(`
  853. <message id="${u.getUniqueId()}" to="${_converse.jid}" from="${muc_jid}">
  854. <result xmlns="urn:xmpp:mam:2" queryid="${queryid}" id="stanza-id">
  855. <forwarded xmlns="urn:xmpp:forward:0">
  856. <delay xmlns="urn:xmpp:delay" stamp="2019-09-20T23:08:25Z"/>
  857. <message type="groupchat" from="${muc_jid}/eve" to="${_converse.bare_jid}" id="message-id-1">
  858. <moderated by="${muc_jid}/bob" stamp="2019-09-20T23:09:32Z" xmlns='urn:xmpp:message-moderate:0'>
  859. <retracted xmlns="urn:xmpp:message-retract:0"/>
  860. <reason>This message contains inappropriate content</reason>
  861. </moderated>
  862. </message>
  863. </forwarded>
  864. </result>
  865. </message>
  866. `);
  867. spyOn(view.model, 'handleModeration').and.callThrough();
  868. const promise = new Promise(resolve => _converse.api.listen.once('messageAdded', resolve));
  869. _converse.connection._dataRecv(test_utils.createRequest(tombstone));
  870. const last_id = u.getUniqueId();
  871. const retraction = u.toStanza(`
  872. <message id="${u.getUniqueId()}" to="${_converse.jid}" from="${muc_jid}">
  873. <result xmlns="urn:xmpp:mam:2" queryid="${queryid}" id="${last_id}">
  874. <forwarded xmlns="urn:xmpp:forward:0">
  875. <delay xmlns="urn:xmpp:delay" stamp="2019-09-20T23:08:25Z"/>
  876. <message type="groupchat" from="${muc_jid}" to="${_converse.bare_jid}" id="retract-message-1">
  877. <apply-to id="stanza-id" xmlns="urn:xmpp:fasten:0">
  878. <moderated by="${muc_jid}/bob" xmlns='urn:xmpp:message-moderate:0'>
  879. <retract xmlns="urn:xmpp:message-retract:0"/>
  880. <reason>This message contains inappropriate content</reason>
  881. </moderated>
  882. </apply-to>
  883. </message>
  884. </forwarded>
  885. </result>
  886. </message>
  887. `);
  888. _converse.connection._dataRecv(test_utils.createRequest(retraction));
  889. const iq_result = $iq({'type': 'result', 'id': stanza.getAttribute('id')})
  890. .c('fin', {'xmlns': 'urn:xmpp:mam:2'})
  891. .c('set', {'xmlns': 'http://jabber.org/protocol/rsm'})
  892. .c('first', {'index': '0'}).t(first_id).up()
  893. .c('last').t(last_id).up()
  894. .c('count').t('2');
  895. _converse.connection._dataRecv(test_utils.createRequest(iq_result));
  896. await promise;
  897. expect(view.model.messages.length).toBe(1);
  898. let message = view.model.messages.at(0);
  899. expect(message.get('retracted')).toBeTruthy();
  900. expect(message.get('is_tombstone')).toBe(true);
  901. await u.waitUntil(() => view.model.handleModeration.calls.count() === 2);
  902. expect(await view.model.handleModeration.calls.first().returnValue).toBe(false);
  903. expect(await view.model.handleModeration.calls.all()[1].returnValue).toBe(true);
  904. expect(view.model.messages.length).toBe(1);
  905. message = view.model.messages.at(0);
  906. expect(message.get('retracted')).toBeTruthy();
  907. expect(message.get('is_tombstone')).toBe(true);
  908. expect(message.get('moderation_reason')).toBe("This message contains inappropriate content");
  909. expect(view.el.querySelectorAll('.chat-msg').length).toBe(1);
  910. expect(view.el.querySelectorAll('.chat-msg--retracted').length).toBe(1);
  911. const el = view.el.querySelector('.chat-msg--retracted .chat-msg__message div');
  912. expect(el.textContent.trim()).toBe('A moderator has removed this message');
  913. const qel = view.el.querySelector('.chat-msg--retracted .chat-msg__message q');
  914. expect(qel.textContent.trim()).toBe('This message contains inappropriate content');
  915. done();
  916. }));
  917. });
  918. })
  919. }));