xss.js 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244
  1. (function (root, factory) {
  2. define(["jasmine", "mock", "test-utils" ], factory);
  3. } (this, function (jasmine, mock, test_utils) {
  4. const $pres = converse.env.$pres;
  5. const sizzle = converse.env.sizzle;
  6. const u = converse.env.utils;
  7. describe("XSS", function () {
  8. describe("A Chat Message", function () {
  9. it("will escape IMG payload XSS attempts",
  10. mock.initConverse(
  11. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  12. async function (done, _converse) {
  13. spyOn(window, 'alert').and.callThrough();
  14. await test_utils.waitForRoster(_converse, 'current');
  15. await test_utils.openControlBox(_converse);
  16. const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@montague.lit';
  17. await test_utils.openChatBoxFor(_converse, contact_jid)
  18. const view = _converse.api.chatviews.get(contact_jid);
  19. let message = "<img src=x onerror=alert('XSS');>";
  20. await test_utils.sendMessage(view, message);
  21. let msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  22. expect(msg.textContent).toEqual(message);
  23. expect(msg.innerHTML).toEqual("&lt;img src=x onerror=alert('XSS');&gt;");
  24. expect(window.alert).not.toHaveBeenCalled();
  25. message = "<img src=x onerror=alert('XSS')//";
  26. await test_utils.sendMessage(view, message);
  27. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  28. expect(msg.textContent).toEqual(message);
  29. expect(msg.innerHTML).toEqual("&lt;img src=x onerror=alert('XSS')//");
  30. message = "<img src=x onerror=alert(String.fromCharCode(88,83,83));>";
  31. await test_utils.sendMessage(view, message);
  32. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  33. expect(msg.textContent).toEqual(message);
  34. expect(msg.innerHTML).toEqual("&lt;img src=x onerror=alert(String.fromCharCode(88,83,83));&gt;");
  35. message = "<img src=x oneonerrorrror=alert(String.fromCharCode(88,83,83));>";
  36. await test_utils.sendMessage(view, message);
  37. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  38. expect(msg.textContent).toEqual(message);
  39. expect(msg.innerHTML).toEqual("&lt;img src=x oneonerrorrror=alert(String.fromCharCode(88,83,83));&gt;");
  40. message = "<img src=x:alert(alt) onerror=eval(src) alt=xss>";
  41. await test_utils.sendMessage(view, message);
  42. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  43. expect(msg.textContent).toEqual(message);
  44. expect(msg.innerHTML).toEqual("&lt;img src=x:alert(alt) onerror=eval(src) alt=xss&gt;");
  45. message = "><img src=x onerror=alert('XSS');>";
  46. await test_utils.sendMessage(view, message);
  47. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  48. expect(msg.textContent).toEqual(message);
  49. expect(msg.innerHTML).toEqual("&gt;&lt;img src=x onerror=alert('XSS');&gt;");
  50. message = "><img src=x onerror=alert(String.fromCharCode(88,83,83));>";
  51. await test_utils.sendMessage(view, message);
  52. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  53. expect(msg.textContent).toEqual(message);
  54. expect(msg.innerHTML).toEqual("&gt;&lt;img src=x onerror=alert(String.fromCharCode(88,83,83));&gt;");
  55. expect(window.alert).not.toHaveBeenCalled();
  56. done();
  57. }));
  58. it("will escape SVG payload XSS attempts",
  59. mock.initConverse(
  60. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  61. async function (done, _converse) {
  62. spyOn(window, 'alert').and.callThrough();
  63. await test_utils.waitForRoster(_converse, 'current');
  64. await test_utils.openControlBox(_converse);
  65. const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@montague.lit';
  66. await test_utils.openChatBoxFor(_converse, contact_jid)
  67. const view = _converse.api.chatviews.get(contact_jid);
  68. let message = "<svg onload=alert(1)>";
  69. await test_utils.sendMessage(view, message);
  70. let msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  71. expect(msg.textContent).toEqual(message);
  72. expect(msg.innerHTML).toEqual('&lt;svg onload=alert(1)&gt;');
  73. message = "<svg/onload=alert('XSS')>";
  74. await test_utils.sendMessage(view, message);
  75. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  76. expect(msg.textContent).toEqual(message);
  77. expect(msg.innerHTML).toEqual("&lt;svg/onload=alert('XSS')&gt;");
  78. message = "<svg onload=alert(1)//";
  79. await test_utils.sendMessage(view, message);
  80. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  81. expect(msg.textContent).toEqual(message);
  82. expect(msg.innerHTML).toEqual("&lt;svg onload=alert(1)//");
  83. message = "<svg/onload=alert(String.fromCharCode(88,83,83))>";
  84. await test_utils.sendMessage(view, message);
  85. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  86. expect(msg.textContent).toEqual(message);
  87. expect(msg.innerHTML).toEqual("&lt;svg/onload=alert(String.fromCharCode(88,83,83))&gt;");
  88. message = "<svg id=alert(1) onload=eval(id)>";
  89. await test_utils.sendMessage(view, message);
  90. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  91. expect(msg.textContent).toEqual(message);
  92. expect(msg.innerHTML).toEqual("&lt;svg id=alert(1) onload=eval(id)&gt;");
  93. message = '"><svg/onload=alert(String.fromCharCode(88,83,83))>';
  94. await test_utils.sendMessage(view, message);
  95. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  96. expect(msg.textContent).toEqual(message);
  97. expect(msg.innerHTML).toEqual('"&gt;&lt;svg/onload=alert(String.fromCharCode(88,83,83))&gt;');
  98. message = '"><svg/onload=alert(/XSS/)';
  99. await test_utils.sendMessage(view, message);
  100. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  101. expect(msg.textContent).toEqual(message);
  102. expect(msg.innerHTML).toEqual('"&gt;&lt;svg/onload=alert(/XSS/)');
  103. expect(window.alert).not.toHaveBeenCalled();
  104. done();
  105. }));
  106. it("will have properly escaped URLs",
  107. mock.initConverse(
  108. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  109. async function (done, _converse) {
  110. await test_utils.waitForRoster(_converse, 'current');
  111. await test_utils.openControlBox(_converse);
  112. const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@montague.lit';
  113. await test_utils.openChatBoxFor(_converse, contact_jid)
  114. const view = _converse.api.chatviews.get(contact_jid);
  115. let message = "http://www.opkode.com/'onmouseover='alert(1)'whatever";
  116. await test_utils.sendMessage(view, message);
  117. let msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  118. expect(msg.textContent).toEqual(message);
  119. expect(msg.innerHTML)
  120. .toEqual('<a target="_blank" rel="noopener" href="http://www.opkode.com/%27onmouseover=%27alert%281%29%27whatever">http://www.opkode.com/\'onmouseover=\'alert(1)\'whatever</a>');
  121. message = 'http://www.opkode.com/"onmouseover="alert(1)"whatever';
  122. await test_utils.sendMessage(view, message);
  123. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  124. expect(msg.textContent).toEqual(message);
  125. expect(msg.innerHTML).toEqual('<a target="_blank" rel="noopener" href="http://www.opkode.com/%22onmouseover=%22alert%281%29%22whatever">http://www.opkode.com/"onmouseover="alert(1)"whatever</a>');
  126. message = "https://en.wikipedia.org/wiki/Ender's_Game";
  127. await test_utils.sendMessage(view, message);
  128. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  129. expect(msg.textContent).toEqual(message);
  130. expect(msg.innerHTML).toEqual('<a target="_blank" rel="noopener" href="https://en.wikipedia.org/wiki/Ender%27s_Game">'+message+'</a>');
  131. message = "<https://bugs.documentfoundation.org/show_bug.cgi?id=123737>";
  132. await test_utils.sendMessage(view, message);
  133. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  134. expect(msg.textContent).toEqual(message);
  135. expect(msg.innerHTML).toEqual(
  136. `&lt;<a target="_blank" rel="noopener" href="https://bugs.documentfoundation.org/show_bug.cgi?id=123737">https://bugs.documentfoundation.org/show_bug.cgi?id=123737</a>&gt;`);
  137. message = '<http://www.opkode.com/"onmouseover="alert(1)"whatever>';
  138. await test_utils.sendMessage(view, message);
  139. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  140. expect(msg.textContent).toEqual(message);
  141. expect(msg.innerHTML).toEqual(
  142. '&lt;<a target="_blank" rel="noopener" href="http://www.opkode.com/%22onmouseover=%22alert%281%29%22whatever">http://www.opkode.com/"onmouseover="alert(1)"whatever</a>&gt;');
  143. message = `https://www.google.com/maps/place/Kochstraat+6,+2041+CE+Zandvoort/@52.3775999,4.548971,3a,15y,170.85h,88.39t/data=!3m6!1e1!3m4!1sQ7SdHo_bPLPlLlU8GSGWaQ!2e0!7i13312!8i6656!4m5!3m4!1s0x47c5ec1e56f845ad:0x1de0bc4a5771fb08!8m2!3d52.3773668!4d4.5489388!5m1!1e2`
  144. await test_utils.sendMessage(view, message);
  145. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  146. expect(msg.textContent).toEqual(message);
  147. expect(msg.innerHTML).toEqual(
  148. `<a target="_blank" rel="noopener" href="https://www.google.com/maps/place/Kochstraat+6,+2041+CE+Zandvoort/@52.3775999,4.548971,3a,15y,170.85h,88.39t/data=%213m6%211e1%213m4%211sQ7SdHo_bPLPlLlU8GSGWaQ%212e0%217i13312%218i6656%214m5%213m4%211s0x47c5ec1e56f845ad:0x1de0bc4a5771fb08%218m2%213d52.3773668%214d4.5489388%215m1%211e2">https://www.google.com/maps/place/Kochstraat+6,+2041+CE+Zandvoort/@52.3775999,4.548971,3a,15y,170.85h,88.39t/data=!3m6!1e1!3m4!1sQ7SdHo_bPLPlLlU8GSGWaQ!2e0!7i13312!8i6656!4m5!3m4!1s0x47c5ec1e56f845ad:0x1de0bc4a5771fb08!8m2!3d52.3773668!4d4.5489388!5m1!1e2</a>`);
  149. done();
  150. }));
  151. });
  152. describe("A Groupchat", function () {
  153. it("escapes occupant nicknames when rendering them, to avoid JS-injection attacks",
  154. mock.initConverse(['rosterGroupsFetched'], {},
  155. async function (done, _converse) {
  156. await test_utils.openAndEnterChatRoom(_converse, 'lounge@montague.lit', 'romeo');
  157. /* <presence xmlns="jabber:client" to="jc@chat.example.org/converse.js-17184538"
  158. * from="oo@conference.chat.example.org/&lt;img src=&quot;x&quot; onerror=&quot;alert(123)&quot;/&gt;">
  159. * <x xmlns="http://jabber.org/protocol/muc#user">
  160. * <item jid="jc@chat.example.org/converse.js-17184538" affiliation="owner" role="moderator"/>
  161. * <status code="110"/>
  162. * </x>
  163. * </presence>"
  164. */
  165. const presence = $pres({
  166. to:'romeo@montague.lit/pda',
  167. from:"lounge@montague.lit/&lt;img src=&quot;x&quot; onerror=&quot;alert(123)&quot;/&gt;"
  168. }).c('x').attrs({xmlns:'http://jabber.org/protocol/muc#user'})
  169. .c('item').attrs({
  170. jid: 'someone@montague.lit',
  171. role: 'moderator',
  172. }).up()
  173. .c('status').attrs({code:'110'}).nodeTree;
  174. _converse.connection._dataRecv(test_utils.createRequest(presence));
  175. const view = _converse.chatboxviews.get('lounge@montague.lit');
  176. await u.waitUntil(() => view.el.querySelectorAll('li .occupant-nick').length, 500);
  177. const occupants = view.el.querySelector('.occupant-list').querySelectorAll('li .occupant-nick');
  178. expect(occupants.length).toBe(2);
  179. expect(occupants[0].textContent.trim()).toBe("&lt;img src=&quot;x&quot; onerror=&quot;alert(123)&quot;/&gt;");
  180. done();
  181. }));
  182. it("escapes the subject before rendering it, to avoid JS-injection attacks",
  183. mock.initConverse(
  184. ['rosterGroupsFetched'], {},
  185. async function (done, _converse) {
  186. await test_utils.openAndEnterChatRoom(_converse, 'jdev@conference.jabber.org', 'jc');
  187. spyOn(window, 'alert');
  188. const subject = '<img src="x" onerror="alert(\'XSS\');"/>';
  189. const view = _converse.chatboxviews.get('jdev@conference.jabber.org');
  190. view.model.set({'subject': {
  191. 'text': subject,
  192. 'author': 'ralphm'
  193. }});
  194. expect(sizzle('.chat-event:last').pop().textContent.trim()).toBe('Topic set by ralphm');
  195. expect(view.el.querySelector('.chat-head__desc').textContent.trim()).toBe(subject);
  196. done();
  197. }));
  198. });
  199. });
  200. }));