retractions.js 67 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148
  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. it("can be retracted by the sender if they're a moderator",
  690. mock.initConverse(
  691. ['rosterGroupsFetched', 'chatBoxesFetched'], {'allow_message_retraction': 'moderator'},
  692. async function (done, _converse) {
  693. const muc_jid = 'lounge@montague.lit';
  694. const features = [...mock.default_muc_features, Strophe.NS.MODERATE];
  695. await test_utils.openAndEnterChatRoom(_converse, muc_jid, 'romeo', features);
  696. const view = _converse.api.chatviews.get(muc_jid);
  697. const occupant = view.model.getOwnOccupant();
  698. expect(occupant.get('role')).toBe('moderator');
  699. view.model.sendMessage('Visit this site to get free bitcoin');
  700. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg').length === 1);
  701. const stanza_id = 'retraction-id-1';
  702. const msg_obj = view.model.messages.at(0);
  703. const reflection_stanza = u.toStanza(`
  704. <message xmlns="jabber:client"
  705. from="${msg_obj.get('from')}"
  706. to="${_converse.connection.jid}"
  707. type="groupchat">
  708. <msg_body>${msg_obj.get('message')}</msg_body>
  709. <stanza-id xmlns="urn:xmpp:sid:0"
  710. id="${stanza_id}"
  711. by="lounge@montague.lit"/>
  712. <origin-id xmlns="urn:xmpp:sid:0" id="${msg_obj.get('origin_id')}"/>
  713. </message>`);
  714. await view.model.queueMessage(reflection_stanza);
  715. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg__body.chat-msg__body--received').length, 500);
  716. expect(view.model.messages.length).toBe(1);
  717. expect(view.model.messages.at(0).get('editable')).toBe(true);
  718. const retract_button = await u.waitUntil(() => view.msgs_container.querySelector('.chat-msg__content .chat-msg__action-retract'));
  719. retract_button.click();
  720. await u.waitUntil(() => u.isVisible(document.querySelector('#converse-modals .modal')));
  721. const submit_button = document.querySelector('#converse-modals .modal button[type="submit"]');
  722. submit_button.click();
  723. const sent_IQs = _converse.connection.IQ_stanzas;
  724. const stanza = await u.waitUntil(() => sent_IQs.filter(iq => iq.querySelector('iq apply-to[xmlns="urn:xmpp:fasten:0"]')).pop());
  725. expect(Strophe.serialize(stanza)).toBe(
  726. `<iq id="${stanza.getAttribute('id')}" to="${muc_jid}" type="set" xmlns="jabber:client">`+
  727. `<apply-to id="${stanza_id}" xmlns="urn:xmpp:fasten:0">`+
  728. `<moderate xmlns="urn:xmpp:message-moderate:0">`+
  729. `<retract xmlns="urn:xmpp:message-retract:0"/>`+
  730. `<reason></reason>`+
  731. `</moderate>`+
  732. `</apply-to>`+
  733. `</iq>`);
  734. const result_iq = $iq({'from': muc_jid, 'id': stanza.getAttribute('id'), 'to': _converse.bare_jid, 'type': 'result'});
  735. _converse.connection._dataRecv(test_utils.createRequest(result_iq));
  736. // We opportunistically save the message as retracted, even before receiving the retraction message
  737. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg--retracted').length === 1);
  738. expect(view.model.messages.length).toBe(1);
  739. expect(view.model.messages.at(0).get('moderated')).toBe('retracted');
  740. expect(view.model.messages.at(0).get('moderation_reason')).toBe(undefined);
  741. expect(view.model.messages.at(0).get('is_ephemeral')).toBe(false);
  742. expect(view.model.messages.at(0).get('editable')).toBe(false);
  743. expect(view.el.querySelectorAll('.chat-msg--retracted').length).toBe(1);
  744. const msg_el = view.el.querySelector('.chat-msg--retracted .chat-msg__message');
  745. expect(msg_el.firstElementChild.textContent.trim()).toBe('romeo has removed this message');
  746. expect(msg_el.querySelector('q')).toBe(null);
  747. // The server responds with a retraction message
  748. const retraction = u.toStanza(`
  749. <message type="groupchat" id='retraction-id-1' from="${muc_jid}" to="${muc_jid}/romeo">
  750. <apply-to id="${stanza_id}" xmlns="urn:xmpp:fasten:0">
  751. <moderated by='${_converse.bare_jid}' xmlns='urn:xmpp:message-moderate:0'>
  752. <retract xmlns='urn:xmpp:message-retract:0' />
  753. </moderated>
  754. </apply-to>
  755. </message>`);
  756. await view.model.queueMessage(retraction);
  757. expect(view.model.messages.length).toBe(1);
  758. expect(view.model.messages.at(0).get('moderated')).toBe('retracted');
  759. expect(view.model.messages.at(0).get('moderation_reason')).toBe(undefined);
  760. expect(view.model.messages.at(0).get('is_ephemeral')).toBe(false);
  761. expect(view.model.messages.at(0).get('editable')).toBe(false);
  762. done();
  763. }));
  764. });
  765. describe("when archived", function () {
  766. it("may be returned as a tombstone message",
  767. mock.initConverse(
  768. ['discoInitialized'], {},
  769. async function (done, _converse) {
  770. await test_utils.waitForRoster(_converse, 'current', 1);
  771. const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@montague.lit';
  772. await test_utils.openChatBoxFor(_converse, contact_jid);
  773. await test_utils.waitUntilDiscoConfirmed(_converse, _converse.bare_jid, null, [Strophe.NS.MAM]);
  774. const sent_IQs = _converse.connection.IQ_stanzas;
  775. const stanza = await u.waitUntil(() => sent_IQs.filter(iq => iq.querySelector(`iq[type="set"] query[xmlns="${Strophe.NS.MAM}"]`)).pop());
  776. const queryid = stanza.querySelector('query').getAttribute('queryid');
  777. const view = _converse.chatboxviews.get(contact_jid);
  778. const first_id = u.getUniqueId();
  779. spyOn(view.model, 'handleRetraction').and.callThrough();
  780. const first_message = u.toStanza(`
  781. <message id='${u.getUniqueId()}' to='${_converse.jid}'>
  782. <result xmlns='urn:xmpp:mam:2' queryid='${queryid}' id="${first_id}">
  783. <forwarded xmlns='urn:xmpp:forward:0'>
  784. <delay xmlns='urn:xmpp:delay' stamp='2019-09-20T23:01:15Z'/>
  785. <message type="chat" from="${contact_jid}" to="${_converse.bare_jid}" id="message-id-0">
  786. <origin-id xmlns='urn:xmpp:sid:0' id="origin-id-0"/>
  787. <body>😊</body>
  788. </message>
  789. </forwarded>
  790. </result>
  791. </message>
  792. `);
  793. _converse.connection._dataRecv(test_utils.createRequest(first_message));
  794. const tombstone = u.toStanza(`
  795. <message id='${u.getUniqueId()}' to='${_converse.jid}'>
  796. <result xmlns='urn:xmpp:mam:2' queryid='${queryid}' id="${u.getUniqueId()}">
  797. <forwarded xmlns='urn:xmpp:forward:0'>
  798. <delay xmlns='urn:xmpp:delay' stamp='2019-09-20T23:08:25Z'/>
  799. <message type="chat" from="${contact_jid}" to="${_converse.bare_jid}" id="message-id-1">
  800. <origin-id xmlns='urn:xmpp:sid:0' id="origin-id-1"/>
  801. <retracted stamp='2019-09-20T23:09:32Z' xmlns='urn:xmpp:message-retract:0'/>
  802. </message>
  803. </forwarded>
  804. </result>
  805. </message>
  806. `);
  807. _converse.connection._dataRecv(test_utils.createRequest(tombstone));
  808. const last_id = u.getUniqueId();
  809. const retraction = u.toStanza(`
  810. <message id='${u.getUniqueId()}' to='${_converse.jid}'>
  811. <result xmlns='urn:xmpp:mam:2' queryid='${queryid}' id="${last_id}">
  812. <forwarded xmlns='urn:xmpp:forward:0'>
  813. <delay xmlns='urn:xmpp:delay' stamp='2019-09-20T23:08:25Z'/>
  814. <message from="${contact_jid}" to='${_converse.bare_jid}' id='retract-message-1'>
  815. <apply-to id="origin-id-1" xmlns="urn:xmpp:fasten:0">
  816. <retract xmlns='urn:xmpp:message-retract:0'/>
  817. </apply-to>
  818. </message>
  819. </forwarded>
  820. </result>
  821. </message>
  822. `);
  823. _converse.connection._dataRecv(test_utils.createRequest(retraction));
  824. const iq_result = $iq({'type': 'result', 'id': stanza.getAttribute('id')})
  825. .c('fin', {'xmlns': 'urn:xmpp:mam:2'})
  826. .c('set', {'xmlns': 'http://jabber.org/protocol/rsm'})
  827. .c('first', {'index': '0'}).t(first_id).up()
  828. .c('last').t(last_id).up()
  829. .c('count').t('2');
  830. _converse.connection._dataRecv(test_utils.createRequest(iq_result));
  831. await u.waitUntil(() => view.model.handleRetraction.calls.count() === 3);
  832. expect(view.model.messages.length).toBe(2);
  833. const message = view.model.messages.at(1);
  834. expect(message.get('retracted')).toBeTruthy();
  835. expect(message.get('is_tombstone')).toBe(true);
  836. expect(await view.model.handleRetraction.calls.first().returnValue).toBe(false);
  837. expect(await view.model.handleRetraction.calls.all()[1].returnValue).toBe(false);
  838. expect(await view.model.handleRetraction.calls.all()[2].returnValue).toBe(true);
  839. await u.waitUntil(() => view.el.querySelectorAll('.chat-msg').length === 2);
  840. expect(view.el.querySelectorAll('.chat-msg--retracted').length).toBe(1);
  841. const el = view.el.querySelector('.chat-msg--retracted .chat-msg__message div');
  842. expect(el.textContent.trim()).toBe('Mercutio has removed this message');
  843. expect(u.hasClass('chat-msg--followup', el.parentElement)).toBe(false);
  844. done();
  845. }));
  846. it("may be returned as a tombstone groupchat message",
  847. mock.initConverse(
  848. ['discoInitialized'], {},
  849. async function (done, _converse) {
  850. const muc_jid = 'lounge@montague.lit';
  851. const features = [...mock.default_muc_features, Strophe.NS.MODERATE];
  852. await test_utils.openAndEnterChatRoom(_converse, muc_jid, 'romeo', features);
  853. const view = _converse.chatboxviews.get(muc_jid);
  854. const sent_IQs = _converse.connection.IQ_stanzas;
  855. const stanza = await u.waitUntil(() => sent_IQs.filter(iq => iq.querySelector(`iq[type="set"] query[xmlns="${Strophe.NS.MAM}"]`)).pop());
  856. const queryid = stanza.querySelector('query').getAttribute('queryid');
  857. const first_id = u.getUniqueId();
  858. const tombstone = u.toStanza(`
  859. <message id="${u.getUniqueId()}" to="${_converse.jid}" from="${muc_jid}">
  860. <result xmlns="urn:xmpp:mam:2" queryid="${queryid}" id="stanza-id">
  861. <forwarded xmlns="urn:xmpp:forward:0">
  862. <delay xmlns="urn:xmpp:delay" stamp="2019-09-20T23:08:25Z"/>
  863. <message type="groupchat" from="${muc_jid}/eve" to="${_converse.bare_jid}" id="message-id-1">
  864. <origin-id xmlns='urn:xmpp:sid:0' id="origin-id-1"/>
  865. <retracted stamp="2019-09-20T23:09:32Z" xmlns="urn:xmpp:message-retract:0"/>
  866. </message>
  867. </forwarded>
  868. </result>
  869. </message>
  870. `);
  871. spyOn(view.model, 'handleRetraction').and.callThrough();
  872. const promise = new Promise(resolve => _converse.api.listen.once('messageAdded', resolve));
  873. _converse.connection._dataRecv(test_utils.createRequest(tombstone));
  874. const last_id = u.getUniqueId();
  875. const retraction = u.toStanza(`
  876. <message id="${u.getUniqueId()}" to="${_converse.jid}" from="${muc_jid}">
  877. <result xmlns="urn:xmpp:mam:2" queryid="${queryid}" id="${last_id}">
  878. <forwarded xmlns="urn:xmpp:forward:0">
  879. <delay xmlns="urn:xmpp:delay" stamp="2019-09-20T23:08:25Z"/>
  880. <message type="groupchat" from="${muc_jid}/eve" to="${_converse.bare_jid}" id="retract-message-1">
  881. <apply-to id="origin-id-1" xmlns="urn:xmpp:fasten:0">
  882. <retract xmlns="urn:xmpp:message-retract:0"/>
  883. </apply-to>
  884. </message>
  885. </forwarded>
  886. </result>
  887. </message>
  888. `);
  889. _converse.connection._dataRecv(test_utils.createRequest(retraction));
  890. const iq_result = $iq({'type': 'result', 'id': stanza.getAttribute('id')})
  891. .c('fin', {'xmlns': 'urn:xmpp:mam:2'})
  892. .c('set', {'xmlns': 'http://jabber.org/protocol/rsm'})
  893. .c('first', {'index': '0'}).t(first_id).up()
  894. .c('last').t(last_id).up()
  895. .c('count').t('2');
  896. _converse.connection._dataRecv(test_utils.createRequest(iq_result));
  897. await promise;
  898. expect(view.model.messages.length).toBe(1);
  899. let message = view.model.messages.at(0);
  900. expect(message.get('retracted')).toBeTruthy();
  901. expect(message.get('is_tombstone')).toBe(true);
  902. await u.waitUntil(() => view.model.handleRetraction.calls.count() === 2);
  903. expect(await view.model.handleRetraction.calls.first().returnValue).toBe(false);
  904. expect(await view.model.handleRetraction.calls.all()[1].returnValue).toBe(true);
  905. expect(view.model.messages.length).toBe(1);
  906. message = view.model.messages.at(0);
  907. expect(message.get('retracted')).toBeTruthy();
  908. expect(message.get('is_tombstone')).toBe(true);
  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('eve has removed this message');
  913. done();
  914. }));
  915. it("may be returned as a tombstone moderated groupchat message",
  916. mock.initConverse(
  917. ['discoInitialized', 'chatBoxesFetched'], {},
  918. async function (done, _converse) {
  919. const muc_jid = 'lounge@montague.lit';
  920. const features = [...mock.default_muc_features, Strophe.NS.MODERATE];
  921. await test_utils.openAndEnterChatRoom(_converse, muc_jid, 'romeo', features);
  922. const view = _converse.chatboxviews.get(muc_jid);
  923. const sent_IQs = _converse.connection.IQ_stanzas;
  924. const stanza = await u.waitUntil(() => sent_IQs.filter(iq => iq.querySelector(`iq[type="set"] query[xmlns="${Strophe.NS.MAM}"]`)).pop());
  925. const queryid = stanza.querySelector('query').getAttribute('queryid');
  926. const first_id = u.getUniqueId();
  927. const tombstone = u.toStanza(`
  928. <message id="${u.getUniqueId()}" to="${_converse.jid}" from="${muc_jid}">
  929. <result xmlns="urn:xmpp:mam:2" queryid="${queryid}" id="stanza-id">
  930. <forwarded xmlns="urn:xmpp:forward:0">
  931. <delay xmlns="urn:xmpp:delay" stamp="2019-09-20T23:08:25Z"/>
  932. <message type="groupchat" from="${muc_jid}/eve" to="${_converse.bare_jid}" id="message-id-1">
  933. <moderated by="${muc_jid}/bob" stamp="2019-09-20T23:09:32Z" xmlns='urn:xmpp:message-moderate:0'>
  934. <retracted xmlns="urn:xmpp:message-retract:0"/>
  935. <reason>This message contains inappropriate content</reason>
  936. </moderated>
  937. </message>
  938. </forwarded>
  939. </result>
  940. </message>
  941. `);
  942. spyOn(view.model, 'handleModeration').and.callThrough();
  943. const promise = new Promise(resolve => _converse.api.listen.once('messageAdded', resolve));
  944. _converse.connection._dataRecv(test_utils.createRequest(tombstone));
  945. const last_id = u.getUniqueId();
  946. const retraction = u.toStanza(`
  947. <message id="${u.getUniqueId()}" to="${_converse.jid}" from="${muc_jid}">
  948. <result xmlns="urn:xmpp:mam:2" queryid="${queryid}" id="${last_id}">
  949. <forwarded xmlns="urn:xmpp:forward:0">
  950. <delay xmlns="urn:xmpp:delay" stamp="2019-09-20T23:08:25Z"/>
  951. <message type="groupchat" from="${muc_jid}" to="${_converse.bare_jid}" id="retract-message-1">
  952. <apply-to id="stanza-id" xmlns="urn:xmpp:fasten:0">
  953. <moderated by="${muc_jid}/bob" xmlns='urn:xmpp:message-moderate:0'>
  954. <retract xmlns="urn:xmpp:message-retract:0"/>
  955. <reason>This message contains inappropriate content</reason>
  956. </moderated>
  957. </apply-to>
  958. </message>
  959. </forwarded>
  960. </result>
  961. </message>
  962. `);
  963. _converse.connection._dataRecv(test_utils.createRequest(retraction));
  964. const iq_result = $iq({'type': 'result', 'id': stanza.getAttribute('id')})
  965. .c('fin', {'xmlns': 'urn:xmpp:mam:2'})
  966. .c('set', {'xmlns': 'http://jabber.org/protocol/rsm'})
  967. .c('first', {'index': '0'}).t(first_id).up()
  968. .c('last').t(last_id).up()
  969. .c('count').t('2');
  970. _converse.connection._dataRecv(test_utils.createRequest(iq_result));
  971. await promise;
  972. expect(view.model.messages.length).toBe(1);
  973. let message = view.model.messages.at(0);
  974. expect(message.get('retracted')).toBeTruthy();
  975. expect(message.get('is_tombstone')).toBe(true);
  976. await u.waitUntil(() => view.model.handleModeration.calls.count() === 2);
  977. expect(await view.model.handleModeration.calls.first().returnValue).toBe(false);
  978. expect(await view.model.handleModeration.calls.all()[1].returnValue).toBe(true);
  979. expect(view.model.messages.length).toBe(1);
  980. message = view.model.messages.at(0);
  981. expect(message.get('retracted')).toBeTruthy();
  982. expect(message.get('is_tombstone')).toBe(true);
  983. expect(message.get('moderation_reason')).toBe("This message contains inappropriate content");
  984. expect(view.el.querySelectorAll('.chat-msg').length).toBe(1);
  985. expect(view.el.querySelectorAll('.chat-msg--retracted').length).toBe(1);
  986. const el = view.el.querySelector('.chat-msg--retracted .chat-msg__message div');
  987. expect(el.textContent.trim()).toBe('A moderator has removed this message');
  988. const qel = view.el.querySelector('.chat-msg--retracted .chat-msg__message q');
  989. expect(qel.textContent.trim()).toBe('This message contains inappropriate content');
  990. done();
  991. }));
  992. });
  993. })
  994. }));