xss.js 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328
  1. /*global mock */
  2. const $pres = converse.env.$pres;
  3. const sizzle = converse.env.sizzle;
  4. const u = converse.env.utils;
  5. describe("XSS", function () {
  6. describe("A Chat Message", function () {
  7. it("will escape IMG payload XSS attempts",
  8. mock.initConverse(
  9. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  10. async function (done, _converse) {
  11. spyOn(window, 'alert').and.callThrough();
  12. await mock.waitForRoster(_converse, 'current');
  13. await mock.openControlBox(_converse);
  14. const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@montague.lit';
  15. await mock.openChatBoxFor(_converse, contact_jid)
  16. const view = _converse.api.chatviews.get(contact_jid);
  17. let message = "<img src=x onerror=alert('XSS');>";
  18. await mock.sendMessage(view, message);
  19. let msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  20. expect(msg.textContent).toEqual(message);
  21. expect(msg.innerHTML.replace(/<!---->/g, '')).toEqual("&lt;img src=x onerror=alert('XSS');&gt;");
  22. expect(window.alert).not.toHaveBeenCalled();
  23. message = "<img src=x onerror=alert('XSS')//";
  24. await mock.sendMessage(view, message);
  25. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  26. expect(msg.textContent).toEqual(message);
  27. expect(msg.innerHTML.replace(/<!---->/g, '')).toEqual("&lt;img src=x onerror=alert('XSS')//");
  28. message = "<img src=x onerror=alert(String.fromCharCode(88,83,83));>";
  29. await mock.sendMessage(view, message);
  30. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  31. expect(msg.textContent).toEqual(message);
  32. expect(msg.innerHTML.replace(/<!---->/g, '')).toEqual("&lt;img src=x onerror=alert(String.fromCharCode(88,83,83));&gt;");
  33. message = "<img src=x oneonerrorrror=alert(String.fromCharCode(88,83,83));>";
  34. await mock.sendMessage(view, message);
  35. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  36. expect(msg.textContent).toEqual(message);
  37. expect(msg.innerHTML.replace(/<!---->/g, '')).toEqual("&lt;img src=x oneonerrorrror=alert(String.fromCharCode(88,83,83));&gt;");
  38. message = "<img src=x:alert(alt) onerror=eval(src) alt=xss>";
  39. await mock.sendMessage(view, message);
  40. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  41. expect(msg.textContent).toEqual(message);
  42. expect(msg.innerHTML.replace(/<!---->/g, '')).toEqual("&lt;img src=x:alert(alt) onerror=eval(src) alt=xss&gt;");
  43. message = "><img src=x onerror=alert('XSS');>";
  44. await mock.sendMessage(view, message);
  45. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  46. expect(msg.textContent).toEqual(message);
  47. expect(msg.innerHTML.replace(/<!---->/g, '')).toEqual("&gt;&lt;img src=x onerror=alert('XSS');&gt;");
  48. message = "><img src=x onerror=alert(String.fromCharCode(88,83,83));>";
  49. await mock.sendMessage(view, message);
  50. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  51. expect(msg.textContent).toEqual(message);
  52. expect(msg.innerHTML.replace(/<!---->/g, '')).toEqual("&gt;&lt;img src=x onerror=alert(String.fromCharCode(88,83,83));&gt;");
  53. expect(window.alert).not.toHaveBeenCalled();
  54. done();
  55. }));
  56. it("will escape SVG payload XSS attempts",
  57. mock.initConverse(
  58. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  59. async function (done, _converse) {
  60. spyOn(window, 'alert').and.callThrough();
  61. await mock.waitForRoster(_converse, 'current');
  62. await mock.openControlBox(_converse);
  63. const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@montague.lit';
  64. await mock.openChatBoxFor(_converse, contact_jid)
  65. const view = _converse.api.chatviews.get(contact_jid);
  66. let message = "<svg onload=alert(1)>";
  67. await mock.sendMessage(view, message);
  68. let msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  69. expect(msg.textContent).toEqual(message);
  70. expect(msg.innerHTML.replace(/<!---->/g, '')).toEqual('&lt;svg onload=alert(1)&gt;');
  71. message = "<svg/onload=alert('XSS')>";
  72. await mock.sendMessage(view, message);
  73. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  74. expect(msg.textContent).toEqual(message);
  75. expect(msg.innerHTML.replace(/<!---->/g, '')).toEqual("&lt;svg/onload=alert('XSS')&gt;");
  76. message = "<svg onload=alert(1)//";
  77. await mock.sendMessage(view, message);
  78. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  79. expect(msg.textContent).toEqual(message);
  80. expect(msg.innerHTML.replace(/<!---->/g, '')).toEqual("&lt;svg onload=alert(1)//");
  81. message = "<svg/onload=alert(String.fromCharCode(88,83,83))>";
  82. await mock.sendMessage(view, message);
  83. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  84. expect(msg.textContent).toEqual(message);
  85. expect(msg.innerHTML.replace(/<!---->/g, '')).toEqual("&lt;svg/onload=alert(String.fromCharCode(88,83,83))&gt;");
  86. message = "<svg id=alert(1) onload=eval(id)>";
  87. await mock.sendMessage(view, message);
  88. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  89. expect(msg.textContent).toEqual(message);
  90. expect(msg.innerHTML.replace(/<!---->/g, '')).toEqual("&lt;svg id=alert(1) onload=eval(id)&gt;");
  91. message = '"><svg/onload=alert(String.fromCharCode(88,83,83))>';
  92. await mock.sendMessage(view, message);
  93. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  94. expect(msg.textContent).toEqual(message);
  95. expect(msg.innerHTML.replace(/<!---->/g, '')).toEqual('"&gt;&lt;svg/onload=alert(String.fromCharCode(88,83,83))&gt;');
  96. message = '"><svg/onload=alert(/XSS/)';
  97. await mock.sendMessage(view, message);
  98. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  99. expect(msg.textContent).toEqual(message);
  100. expect(msg.innerHTML.replace(/<!---->/g, '')).toEqual('"&gt;&lt;svg/onload=alert(/XSS/)');
  101. expect(window.alert).not.toHaveBeenCalled();
  102. done();
  103. }));
  104. it("will have properly escaped URLs",
  105. mock.initConverse(
  106. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  107. async function (done, _converse) {
  108. await mock.waitForRoster(_converse, 'current');
  109. await mock.openControlBox(_converse);
  110. const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@montague.lit';
  111. await mock.openChatBoxFor(_converse, contact_jid)
  112. const view = _converse.api.chatviews.get(contact_jid);
  113. let message = "http://www.opkode.com/'onmouseover='alert(1)'whatever";
  114. await mock.sendMessage(view, message);
  115. let msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  116. expect(msg.textContent).toEqual(message);
  117. expect(msg.innerHTML.replace(/<!---->/g, ''))
  118. .toEqual('http://www.opkode.com/\'onmouseover=\'alert(1)\'whatever');
  119. await u.waitUntil(() => msg.innerHTML.replace(/<!---->/g, '') ===
  120. '<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 mock.sendMessage(view, message);
  123. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  124. expect(msg.textContent).toEqual(message);
  125. await u.waitUntil(() => msg.innerHTML.replace(/<!---->/g, '') ===
  126. '<a target="_blank" rel="noopener" href="http://www.opkode.com/%22onmouseover=%22alert%281%29%22whatever">http://www.opkode.com/"onmouseover="alert(1)"whatever</a>');
  127. message = "https://en.wikipedia.org/wiki/Ender's_Game";
  128. await mock.sendMessage(view, message);
  129. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  130. expect(msg.textContent).toEqual(message);
  131. await u.waitUntil(() => msg.innerHTML.replace(/<!---->/g, '') === '<a target="_blank" rel="noopener" href="https://en.wikipedia.org/wiki/Ender%27s_Game">'+message+'</a>');
  132. message = "<https://bugs.documentfoundation.org/show_bug.cgi?id=123737>";
  133. await mock.sendMessage(view, message);
  134. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  135. expect(msg.textContent).toEqual(message);
  136. await u.waitUntil(() => msg.innerHTML.replace(/<!---->/g, '') ===
  137. `&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;`);
  138. message = '<http://www.opkode.com/"onmouseover="alert(1)"whatever>';
  139. await mock.sendMessage(view, message);
  140. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  141. expect(msg.textContent).toEqual(message);
  142. await u.waitUntil(() => msg.innerHTML.replace(/<!---->/g, '') ===
  143. '&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;');
  144. 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`
  145. await mock.sendMessage(view, message);
  146. msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  147. expect(msg.textContent).toEqual(message);
  148. await u.waitUntil(() => msg.innerHTML.replace(/<!---->/g, '') ===
  149. `<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>`);
  150. done();
  151. }));
  152. it("will avoid malformed and unsafe urls urls from rendering as anchors",
  153. mock.initConverse(
  154. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  155. async function (done, _converse) {
  156. await mock.waitForRoster(_converse, 'current');
  157. await mock.openControlBox(_converse);
  158. const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@montague.lit';
  159. await mock.openChatBoxFor(_converse, contact_jid)
  160. const view = _converse.api.chatviews.get(contact_jid);
  161. const bad_urls =[
  162. 'http://^$^(*^#$%^_1*(',
  163. 'file://devili.sh'
  164. ];
  165. const good_urls =[{
  166. entered: 'http://www.google.com',
  167. href: 'http://www.google.com/'
  168. }, {
  169. entered: 'https://www.google.com/',
  170. href: 'https://www.google.com/'
  171. }, {
  172. entered: 'www.url.com/something?else=1',
  173. href: 'http://www.url.com/something?else=1',
  174. }, {
  175. entered: 'xmpp://anything/?join',
  176. href: 'xmpp://anything/?join',
  177. }, {
  178. entered: 'WWW.SOMETHING.COM/?x=dKasdDAsd4JAsd3OAJSD23osajAidj',
  179. href: 'http://WWW.SOMETHING.COM/?x=dKasdDAsd4JAsd3OAJSD23osajAidj',
  180. }, {
  181. entered: 'mailto:test@mail.org',
  182. href: 'mailto:test@mail.org',
  183. }];
  184. function checkNonParsedURL (url) {
  185. const msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  186. expect(msg.textContent).toEqual(url);
  187. expect(msg.innerHTML.replace(/<!---->/g, '')).toEqual(url);
  188. }
  189. async function checkParsedURL ({ entered, href }) {
  190. const msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  191. expect(msg.textContent).toEqual(entered);
  192. await u.waitUntil(() => msg.innerHTML.replace(/<!---->/g, '') === `<a target="_blank" rel="noopener" href="${href}">${entered}</a>`);
  193. }
  194. async function checkParsedXMPPURL ({ entered, href }) {
  195. const msg = sizzle('.chat-content .chat-msg:last .chat-msg__text', view.el).pop();
  196. expect(msg.textContent.trim()).toEqual(entered);
  197. await u.waitUntil(() => msg.innerHTML.replace(/<!---->/g, '').trim() === `<a target="_blank" rel="noopener" href="${href}">${entered}</a>`);
  198. }
  199. await mock.sendMessage(view, bad_urls[0]);
  200. checkNonParsedURL(bad_urls[0]);
  201. await mock.sendMessage(view, bad_urls[1]);
  202. checkNonParsedURL(bad_urls[1]);
  203. await mock.sendMessage(view, good_urls[0].entered);
  204. await checkParsedURL(good_urls[0]);
  205. await mock.sendMessage(view, good_urls[1].entered);
  206. await checkParsedURL(good_urls[1]);
  207. await mock.sendMessage(view, good_urls[2].entered);
  208. await checkParsedURL(good_urls[2]);
  209. await mock.sendMessage(view, good_urls[3].entered);
  210. await checkParsedXMPPURL(good_urls[3]);
  211. await mock.sendMessage(view, good_urls[4].entered);
  212. await checkParsedURL(good_urls[4]);
  213. await mock.sendMessage(view, good_urls[5].entered);
  214. await checkParsedURL(good_urls[5]);
  215. done();
  216. }));
  217. });
  218. describe("A Groupchat", function () {
  219. it("escapes occupant nicknames when rendering them, to avoid JS-injection attacks",
  220. mock.initConverse(['rosterGroupsFetched'], {},
  221. async function (done, _converse) {
  222. await mock.openAndEnterChatRoom(_converse, 'lounge@montague.lit', 'romeo');
  223. /* <presence xmlns="jabber:client" to="jc@chat.example.org/converse.js-17184538"
  224. * from="oo@conference.chat.example.org/&lt;img src=&quot;x&quot; onerror=&quot;alert(123)&quot;/&gt;">
  225. * <x xmlns="http://jabber.org/protocol/muc#user">
  226. * <item jid="jc@chat.example.org/converse.js-17184538" affiliation="owner" role="moderator"/>
  227. * <status code="110"/>
  228. * </x>
  229. * </presence>"
  230. */
  231. const presence = $pres({
  232. to:'romeo@montague.lit/pda',
  233. from:"lounge@montague.lit/&lt;img src=&quot;x&quot; onerror=&quot;alert(123)&quot;/&gt;"
  234. }).c('x').attrs({xmlns:'http://jabber.org/protocol/muc#user'})
  235. .c('item').attrs({
  236. jid: 'someone@montague.lit',
  237. role: 'moderator',
  238. }).up()
  239. .c('status').attrs({code:'110'}).nodeTree;
  240. _converse.connection._dataRecv(mock.createRequest(presence));
  241. const view = _converse.chatboxviews.get('lounge@montague.lit');
  242. await u.waitUntil(() => view.el.querySelectorAll('li .occupant-nick').length, 500);
  243. const occupants = view.el.querySelector('.occupant-list').querySelectorAll('li .occupant-nick');
  244. expect(occupants.length).toBe(2);
  245. expect(occupants[0].textContent.trim()).toBe("&lt;img src=&quot;x&quot; onerror=&quot;alert(123)&quot;/&gt;");
  246. done();
  247. }));
  248. it("escapes the subject before rendering it, to avoid JS-injection attacks",
  249. mock.initConverse(
  250. ['rosterGroupsFetched'], {},
  251. async function (done, _converse) {
  252. await mock.openAndEnterChatRoom(_converse, 'jdev@conference.jabber.org', 'jc');
  253. spyOn(window, 'alert');
  254. const subject = '<img src="x" onerror="alert(\'XSS\');"/>';
  255. const view = _converse.chatboxviews.get('jdev@conference.jabber.org');
  256. view.model.set({'subject': {
  257. 'text': subject,
  258. 'author': 'ralphm'
  259. }});
  260. const text = await u.waitUntil(() => view.el.querySelector('.chat-head__desc')?.textContent.trim());
  261. expect(text).toBe(subject);
  262. done();
  263. }));
  264. });
  265. });