modtools.js 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368
  1. /*global mock */
  2. const _ = converse.env._;
  3. const $iq = converse.env.$iq;
  4. const $pres = converse.env.$pres;
  5. const sizzle = converse.env.sizzle;
  6. const Strophe = converse.env.Strophe;
  7. const u = converse.env.utils;
  8. describe("The groupchat moderator tool", function () {
  9. it("allows you to set affiliations and roles",
  10. mock.initConverse(
  11. ['rosterGroupsFetched'], {},
  12. async function (done, _converse) {
  13. spyOn(_converse.ChatRoomView.prototype, 'showModeratorToolsModal').and.callThrough();
  14. const muc_jid = 'lounge@montague.lit';
  15. let members = [
  16. {'jid': 'hag66@shakespeare.lit', 'nick': 'witch', 'affiliation': 'member'},
  17. {'jid': 'gower@shakespeare.lit', 'nick': 'gower', 'affiliation': 'member'},
  18. {'jid': 'wiccarocks@shakespeare.lit', 'nick': 'wiccan', 'affiliation': 'admin'},
  19. {'jid': 'crone1@shakespeare.lit', 'nick': 'thirdwitch', 'affiliation': 'owner'},
  20. {'jid': 'romeo@montague.lit', 'nick': 'romeo', 'affiliation': 'owner'},
  21. ];
  22. await mock.openAndEnterChatRoom(_converse, muc_jid, 'romeo', [], members);
  23. const view = _converse.chatboxviews.get(muc_jid);
  24. await u.waitUntil(() => (view.model.occupants.length === 5), 1000);
  25. const textarea = view.el.querySelector('.chat-textarea');
  26. textarea.value = '/modtools';
  27. const enter = { 'target': textarea, 'preventDefault': function preventDefault () {}, 'keyCode': 13 };
  28. view.onKeyDown(enter);
  29. await u.waitUntil(() => view.showModeratorToolsModal.calls.count());
  30. const modal = view.modtools_modal;
  31. await u.waitUntil(() => u.isVisible(modal.el), 1000);
  32. let tab = modal.el.querySelector('#affiliations-tab');
  33. // Clear so that we don't match older stanzas
  34. _converse.connection.IQ_stanzas = [];
  35. tab.click();
  36. let select = modal.el.querySelector('.select-affiliation');
  37. expect(select.value).toBe('owner');
  38. select.value = 'admin';
  39. let button = modal.el.querySelector('.btn-primary[name="users_with_affiliation"]');
  40. button.click();
  41. await u.waitUntil(() => !modal.loading_users_with_affiliation);
  42. let user_els = modal.el.querySelectorAll('.list-group--users > li');
  43. expect(user_els.length).toBe(1);
  44. expect(user_els[0].querySelector('.list-group-item.active').textContent.trim()).toBe('JID: wiccarocks@shakespeare.lit');
  45. expect(user_els[0].querySelector('.list-group-item:nth-child(2n)').textContent.trim()).toBe('Nickname: wiccan');
  46. expect(user_els[0].querySelector('.list-group-item:nth-child(3n) div').textContent.trim()).toBe('Affiliation: admin');
  47. _converse.connection.IQ_stanzas = [];
  48. select.value = 'owner';
  49. button.click();
  50. await u.waitUntil(() => !modal.loading_users_with_affiliation);
  51. user_els = modal.el.querySelectorAll('.list-group--users > li');
  52. expect(user_els.length).toBe(2);
  53. expect(user_els[0].querySelector('.list-group-item.active').textContent.trim()).toBe('JID: romeo@montague.lit');
  54. expect(user_els[0].querySelector('.list-group-item:nth-child(2n)').textContent.trim()).toBe('Nickname: romeo');
  55. expect(user_els[0].querySelector('.list-group-item:nth-child(3n) div').textContent.trim()).toBe('Affiliation: owner');
  56. expect(user_els[1].querySelector('.list-group-item.active').textContent.trim()).toBe('JID: crone1@shakespeare.lit');
  57. expect(user_els[1].querySelector('.list-group-item:nth-child(2n)').textContent.trim()).toBe('Nickname: thirdwitch');
  58. expect(user_els[1].querySelector('.list-group-item:nth-child(3n) div').textContent.trim()).toBe('Affiliation: owner');
  59. const toggle = user_els[1].querySelector('.list-group-item:nth-child(3n) .toggle-form');
  60. const form = user_els[1].querySelector('.list-group-item:nth-child(3n) .affiliation-form');
  61. expect(u.hasClass('hidden', form)).toBeTruthy();
  62. toggle.click();
  63. expect(u.hasClass('hidden', form)).toBeFalsy();
  64. select = form.querySelector('.select-affiliation');
  65. expect(select.value).toBe('owner');
  66. select.value = 'admin';
  67. const input = form.querySelector('input[name="reason"]');
  68. input.value = "You're an admin now";
  69. const submit = form.querySelector('.btn-primary');
  70. submit.click();
  71. spyOn(_converse.ChatRoomOccupants.prototype, 'fetchMembers').and.callThrough();
  72. const sent_IQ = _converse.connection.IQ_stanzas.pop();
  73. expect(Strophe.serialize(sent_IQ)).toBe(
  74. `<iq id="${sent_IQ.getAttribute('id')}" to="lounge@montague.lit" type="set" xmlns="jabber:client">`+
  75. `<query xmlns="http://jabber.org/protocol/muc#admin">`+
  76. `<item affiliation="admin" jid="crone1@shakespeare.lit">`+
  77. `<reason>You&apos;re an admin now</reason>`+
  78. `</item>`+
  79. `</query>`+
  80. `</iq>`);
  81. _converse.connection.IQ_stanzas = [];
  82. const stanza = $iq({
  83. 'type': 'result',
  84. 'id': sent_IQ.getAttribute('id'),
  85. 'from': view.model.get('jid'),
  86. 'to': _converse.connection.jid
  87. });
  88. _converse.connection._dataRecv(mock.createRequest(stanza));
  89. await u.waitUntil(() => view.model.occupants.fetchMembers.calls.count());
  90. members = [
  91. {'jid': 'hag66@shakespeare.lit', 'nick': 'witch', 'affiliation': 'member'},
  92. {'jid': 'gower@shakespeare.lit', 'nick': 'gower', 'affiliation': 'member'},
  93. {'jid': 'wiccarocks@shakespeare.lit', 'nick': 'wiccan', 'affiliation': 'admin'},
  94. {'jid': 'crone1@shakespeare.lit', 'nick': 'thirdwitch', 'affiliation': 'admin'},
  95. {'jid': 'romeo@montague.lit', 'nick': 'romeo', 'affiliation': 'owner'},
  96. ];
  97. await mock.returnMemberLists(_converse, muc_jid, members);
  98. await u.waitUntil(() => view.model.occupants.pluck('affiliation').filter(o => o === 'owner').length === 1);
  99. const alert = modal.el.querySelector('.alert-primary');
  100. expect(alert.textContent.trim()).toBe('Affiliation changed');
  101. user_els = modal.el.querySelectorAll('.list-group--users > li');
  102. expect(user_els.length).toBe(1);
  103. expect(user_els[0].querySelector('.list-group-item.active').textContent.trim()).toBe('JID: romeo@montague.lit');
  104. expect(user_els[0].querySelector('.list-group-item:nth-child(2n)').textContent.trim()).toBe('Nickname: romeo');
  105. expect(user_els[0].querySelector('.list-group-item:nth-child(3n) div').textContent.trim()).toBe('Affiliation: owner');
  106. tab = modal.el.querySelector('#roles-tab');
  107. tab.click();
  108. select = modal.el.querySelector('.select-role');
  109. expect(u.isVisible(select)).toBe(true);
  110. expect(select.value).toBe('moderator');
  111. button = modal.el.querySelector('.btn-primary[name="users_with_role"]');
  112. button.click();
  113. const roles_panel = modal.el.querySelector('#roles-tabpanel');
  114. await u.waitUntil(() => roles_panel.querySelectorAll('.list-group--users > li').length === 1);
  115. select.value = 'participant';
  116. button.click();
  117. await u.waitUntil(() => !modal.loading_users_with_affiliation);
  118. user_els = roles_panel.querySelectorAll('.list-group--users > li')
  119. expect(user_els.length).toBe(1);
  120. expect(user_els[0].textContent.trim()).toBe('No users with that role found.');
  121. done();
  122. }));
  123. it("allows you to filter affiliation search results",
  124. mock.initConverse(
  125. ['rosterGroupsFetched'], {},
  126. async function (done, _converse) {
  127. spyOn(_converse.ChatRoomView.prototype, 'showModeratorToolsModal').and.callThrough();
  128. const muc_jid = 'lounge@montague.lit';
  129. const members = [
  130. {'jid': 'hag66@shakespeare.lit', 'nick': 'witch', 'affiliation': 'member'},
  131. {'jid': 'gower@shakespeare.lit', 'nick': 'gower', 'affiliation': 'member'},
  132. {'jid': 'wiccarocks@shakespeare.lit', 'nick': 'wiccan', 'affiliation': 'member'},
  133. {'jid': 'crone1@shakespeare.lit', 'nick': 'thirdwitch', 'affiliation': 'member'},
  134. {'jid': 'romeo@montague.lit', 'nick': 'romeo', 'affiliation': 'member'},
  135. {'jid': 'juliet@capulet.lit', 'nick': 'juliet', 'affiliation': 'member'},
  136. ];
  137. await mock.openAndEnterChatRoom(_converse, muc_jid, 'romeo', [], members);
  138. const view = _converse.chatboxviews.get(muc_jid);
  139. await u.waitUntil(() => (view.model.occupants.length === 6), 1000);
  140. const textarea = view.el.querySelector('.chat-textarea');
  141. textarea.value = '/modtools';
  142. const enter = { 'target': textarea, 'preventDefault': function preventDefault () {}, 'keyCode': 13 };
  143. view.onKeyDown(enter);
  144. await u.waitUntil(() => view.showModeratorToolsModal.calls.count());
  145. const modal = view.modtools_modal;
  146. await u.waitUntil(() => u.isVisible(modal.el), 1000);
  147. // Clear so that we don't match older stanzas
  148. _converse.connection.IQ_stanzas = [];
  149. const select = modal.el.querySelector('.select-affiliation');
  150. expect(select.value).toBe('owner');
  151. select.value = 'member';
  152. const button = modal.el.querySelector('.btn-primary[name="users_with_affiliation"]');
  153. button.click();
  154. await u.waitUntil(() => !modal.loading_users_with_affiliation);
  155. const user_els = modal.el.querySelectorAll('.list-group--users > li');
  156. expect(user_els.length).toBe(6);
  157. const nicks = Array.from(modal.el.querySelectorAll('.list-group--users > li')).map(el => el.getAttribute('data-nick'));
  158. expect(nicks.join(' ')).toBe('gower juliet romeo thirdwitch wiccan witch');
  159. const filter = modal.el.querySelector('[name="filter"]');
  160. expect(filter).not.toBe(null);
  161. filter.value = 'romeo';
  162. u.triggerEvent(filter, "keyup", "KeyboardEvent");
  163. await u.waitUntil(() => ( modal.el.querySelectorAll('.list-group--users > li').length === 1));
  164. filter.value = 'r';
  165. u.triggerEvent(filter, "keyup", "KeyboardEvent");
  166. await u.waitUntil(() => ( modal.el.querySelectorAll('.list-group--users > li').length === 3));
  167. filter.value = 'gower';
  168. u.triggerEvent(filter, "keyup", "KeyboardEvent");
  169. await u.waitUntil(() => ( modal.el.querySelectorAll('.list-group--users > li').length === 1));
  170. done();
  171. }));
  172. it("allows you to filter role search results",
  173. mock.initConverse(
  174. ['rosterGroupsFetched'], {},
  175. async function (done, _converse) {
  176. spyOn(_converse.ChatRoomView.prototype, 'showModeratorToolsModal').and.callThrough();
  177. const muc_jid = 'lounge@montague.lit';
  178. await mock.openAndEnterChatRoom(_converse, muc_jid, 'romeo', []);
  179. const view = _converse.chatboxviews.get(muc_jid);
  180. _converse.connection._dataRecv(mock.createRequest(
  181. $pres({to: _converse.jid, from: `${muc_jid}/nomorenicks`})
  182. .c('x', {xmlns: Strophe.NS.MUC_USER})
  183. .c('item', {
  184. 'affiliation': 'none',
  185. 'jid': `nomorenicks@montague.lit`,
  186. 'role': 'participant'
  187. })
  188. ));
  189. _converse.connection._dataRecv(mock.createRequest(
  190. $pres({to: _converse.jid, from: `${muc_jid}/newb`})
  191. .c('x', {xmlns: Strophe.NS.MUC_USER})
  192. .c('item', {
  193. 'affiliation': 'none',
  194. 'jid': `newb@montague.lit`,
  195. 'role': 'participant'
  196. })
  197. ));
  198. _converse.connection._dataRecv(mock.createRequest(
  199. $pres({to: _converse.jid, from: `${muc_jid}/some1`})
  200. .c('x', {xmlns: Strophe.NS.MUC_USER})
  201. .c('item', {
  202. 'affiliation': 'none',
  203. 'jid': `some1@montague.lit`,
  204. 'role': 'participant'
  205. })
  206. ));
  207. _converse.connection._dataRecv(mock.createRequest(
  208. $pres({to: _converse.jid, from: `${muc_jid}/oldhag`})
  209. .c('x', {xmlns: Strophe.NS.MUC_USER})
  210. .c('item', {
  211. 'affiliation': 'none',
  212. 'jid': `oldhag@montague.lit`,
  213. 'role': 'participant'
  214. })
  215. ));
  216. _converse.connection._dataRecv(mock.createRequest(
  217. $pres({to: _converse.jid, from: `${muc_jid}/crone`})
  218. .c('x', {xmlns: Strophe.NS.MUC_USER})
  219. .c('item', {
  220. 'affiliation': 'none',
  221. 'jid': `crone@montague.lit`,
  222. 'role': 'participant'
  223. })
  224. ));
  225. _converse.connection._dataRecv(mock.createRequest(
  226. $pres({to: _converse.jid, from: `${muc_jid}/tux`})
  227. .c('x', {xmlns: Strophe.NS.MUC_USER})
  228. .c('item', {
  229. 'affiliation': 'none',
  230. 'jid': `tux@montague.lit`,
  231. 'role': 'participant'
  232. })
  233. ));
  234. await u.waitUntil(() => (view.model.occupants.length === 7), 1000);
  235. const textarea = view.el.querySelector('.chat-textarea');
  236. textarea.value = '/modtools';
  237. const enter = { 'target': textarea, 'preventDefault': function preventDefault () {}, 'keyCode': 13 };
  238. view.onKeyDown(enter);
  239. await u.waitUntil(() => view.showModeratorToolsModal.calls.count());
  240. const modal = view.modtools_modal;
  241. await u.waitUntil(() => u.isVisible(modal.el), 1000);
  242. const tab = modal.el.querySelector('#roles-tab');
  243. tab.click();
  244. // Clear so that we don't match older stanzas
  245. _converse.connection.IQ_stanzas = [];
  246. const select = modal.el.querySelector('.select-role');
  247. expect(select.value).toBe('moderator');
  248. select.value = 'participant';
  249. const button = modal.el.querySelector('.btn-primary[name="users_with_role"]');
  250. button.click();
  251. await u.waitUntil(() => !modal.loading_users_with_role);
  252. const user_els = modal.el.querySelectorAll('.list-group--users > li');
  253. expect(user_els.length).toBe(6);
  254. const nicks = Array.from(modal.el.querySelectorAll('.list-group--users > li')).map(el => el.getAttribute('data-nick'));
  255. expect(nicks.join(' ')).toBe('crone newb nomorenicks oldhag some1 tux');
  256. const filter = modal.el.querySelector('[name="filter"]');
  257. expect(filter).not.toBe(null);
  258. filter.value = 'tux';
  259. u.triggerEvent(filter, "keyup", "KeyboardEvent");
  260. await u.waitUntil(() => ( modal.el.querySelectorAll('.list-group--users > li').length === 1));
  261. filter.value = 'r';
  262. u.triggerEvent(filter, "keyup", "KeyboardEvent");
  263. await u.waitUntil(() => ( modal.el.querySelectorAll('.list-group--users > li').length === 2));
  264. filter.value = 'crone';
  265. u.triggerEvent(filter, "keyup", "KeyboardEvent");
  266. await u.waitUntil(() => ( modal.el.querySelectorAll('.list-group--users > li').length === 1));
  267. done();
  268. }));
  269. it("shows an error message if a particular affiliation list may not be retrieved",
  270. mock.initConverse(
  271. ['rosterGroupsFetched'], {},
  272. async function (done, _converse) {
  273. spyOn(_converse.ChatRoomView.prototype, 'showModeratorToolsModal').and.callThrough();
  274. const muc_jid = 'lounge@montague.lit';
  275. const members = [
  276. {'jid': 'hag66@shakespeare.lit', 'nick': 'witch', 'affiliation': 'member'},
  277. {'jid': 'gower@shakespeare.lit', 'nick': 'gower', 'affiliation': 'member'},
  278. {'jid': 'wiccarocks@shakespeare.lit', 'nick': 'wiccan', 'affiliation': 'admin'},
  279. {'jid': 'crone1@shakespeare.lit', 'nick': 'thirdwitch', 'affiliation': 'owner'},
  280. {'jid': 'romeo@montague.lit', 'nick': 'romeo', 'affiliation': 'owner'},
  281. ];
  282. await mock.openAndEnterChatRoom(_converse, muc_jid, 'romeo', [], members);
  283. const view = _converse.chatboxviews.get(muc_jid);
  284. await u.waitUntil(() => (view.model.occupants.length === 5));
  285. const textarea = view.el.querySelector('.chat-textarea');
  286. textarea.value = '/modtools';
  287. const enter = { 'target': textarea, 'preventDefault': function preventDefault () {}, 'keyCode': 13 };
  288. view.onKeyDown(enter);
  289. await u.waitUntil(() => view.showModeratorToolsModal.calls.count());
  290. const modal = view.modtools_modal;
  291. await u.waitUntil(() => u.isVisible(modal.el), 1000);
  292. const tab = modal.el.querySelector('#affiliations-tab');
  293. // Clear so that we don't match older stanzas
  294. _converse.connection.IQ_stanzas = [];
  295. const IQ_stanzas = _converse.connection.IQ_stanzas;
  296. tab.click();
  297. const select = modal.el.querySelector('.select-affiliation');
  298. select.value = 'outcast';
  299. const button = modal.el.querySelector('.btn-primary[name="users_with_affiliation"]');
  300. button.click();
  301. const iq_query = await u.waitUntil(() => _.filter(
  302. IQ_stanzas,
  303. s => sizzle(`iq[to="${muc_jid}"] query[xmlns="${Strophe.NS.MUC_ADMIN}"] item[affiliation="outcast"]`, s).length
  304. ).pop());
  305. const error = u.toStanza(
  306. `<iq from="${muc_jid}"
  307. id="${iq_query.getAttribute('id')}"
  308. type="error"
  309. to="${_converse.jid}">
  310. <error type="auth">
  311. <forbidden xmlns="${Strophe.NS.STANZAS}"/>
  312. </error>
  313. </iq>`);
  314. _converse.connection._dataRecv(mock.createRequest(error));
  315. await u.waitUntil(() => !modal.loading_users_with_affiliation);
  316. const user_els = modal.el.querySelectorAll('.list-group--users > li');
  317. expect(user_els.length).toBe(1);
  318. expect(user_els[0].textContent.trim()).toBe('Error: not allowed to fetch outcast list for MUC lounge@montague.lit');
  319. done();
  320. }));
  321. });