@@ -317,41 +317,166 @@ function test_wash_svg()
317317 }
318318
319319 /**
320- * Test SVG cleanup
320+ * Test cases for SVG tests
321321 */
322- function test_wash_svg2 ()
322+ function data_wash_svg_tests ()
323323 {
324- $ svg = '<head xmlns=""><script>alert(document.domain)</script>"><svg></svg></head> ' ;
325- $ exp = '<!-- html ignored --><!-- head ignored --><svg xmlns="http://www.w3.org/1999/xhtml"></svg> ' ;
326-
327- $ washer = new rcube_washtml ;
328- $ washed = $ washer ->wash ($ svg );
329-
330- $ this ->assertSame ($ washed , $ exp , "SVG content " );
331-
332- $ svg = '<head xmlns="" onload="alert(document.domain)">Hello victim!<svg></svg></head> ' ;
333- $ exp = '<!-- html ignored --><!-- head ignored -->Hello victim!<svg xmlns="http://www.w3.org/1999/xhtml"></svg> ' ;
334-
335- $ washer = new rcube_washtml ;
336- $ washed = $ washer ->wash ($ svg );
337-
338- $ this ->assertSame ($ washed , $ exp , "SVG content " );
339-
340- $ svg = '<p>Hello victim!<svg xmlns="" onload="alert(document.domain)"></svg></p> ' ;
341- $ exp = '<p>Hello victim!<svg /></p> ' ;
324+ $ svg1 = "<svg id='x' width='100' height='100'><a xlink:href='javascript:alert(1)'><rect x='0' y='0' width='100' height='100' /></a></svg> " ;
325+
326+ return [
327+ [
328+ '<head xmlns=""><script>alert(document.domain)</script>"><svg></svg></head> ' ,
329+ '<!-- html ignored --><!-- head ignored --><svg xmlns="http://www.w3.org/1999/xhtml"></svg> '
330+ ],
331+ [
332+ '<head xmlns="" onload="alert(document.domain)">Hello victim!<svg></svg></head> ' ,
333+ '<!-- html ignored --><!-- head ignored -->Hello victim!<svg xmlns="http://www.w3.org/1999/xhtml"></svg> '
334+ ],
335+ [
336+ '<p>Hello victim!<svg xmlns="" onload="alert(document.domain)"></svg></p> ' ,
337+ '<p>Hello victim!<svg /></p> '
338+ ],
339+ [
340+ '<html><p>Hello victim!<svg xmlns="" onload="alert(document.domain)"></svg></p> ' ,
341+ '<!-- html ignored --><!-- body ignored --><p>Hello victim!<svg xmlns="http://www.w3.org/1999/xhtml"></svg></p> '
342+ ],
343+ [
344+ '<svg xmlns="" onload="alert(document.domain)" /> ' ,
345+ '<svg xmlns="" onload="alert(document.domain)" /> '
346+ ],
347+ [
348+ '<html><svg xmlns="" onload="alert(document.domain)" /> ' ,
349+ '<!-- html ignored --><!-- body ignored --><svg xmlns="http://www.w3.org/1999/xhtml"></svg> '
350+ ],
351+ [
352+ '<svg><a xlink:href="javascript:alert(1)"><text x="20" y="20">XSS</text></a></svg> ' ,
353+ '<svg><a x-washed="xlink:href"><text x="20" y="20">XSS</text></a></svg> '
354+ ],
355+ [
356+ '<html><svg><a xlink:href="javascript:alert(1)"><text x="20" y="20">XSS</text></a></svg> ' ,
357+ '<!-- html ignored --><!-- body ignored --><svg xmlns="http://www.w3.org/1999/xhtml"><a x-washed="xlink:href"><text x="20" y="20">XSS</text></a></svg> '
358+ ],
359+ [
360+ '<svg><animate xlink:href="#xss" attributeName="href" values="javascript:alert(1)" /> '
361+ . '<a id="xss"><text x="20" y="20">XSS</text></a></svg> ' ,
362+ '<svg><!-- animate blocked --><a id="xss"><text x="20" y="20">XSS</text></a></svg> ' ,
363+ ],
364+ [
365+ '<html><svg><animate xlink:href="#xss" attributeName="href" values="javascript:alert(1)" /> '
366+ . '<a id="xss"><text x="20" y="20">XSS</text></a></svg> ' ,
367+ '<!-- html ignored --><!-- body ignored --><svg xmlns="http://www.w3.org/1999/xhtml"> '
368+ . '<!-- animate blocked --><a id="xss"><text x="20" y="20">XSS</text></a></svg> ' ,
369+ ],
370+ [
371+ '<svg><animate xlink:href="#xss" attributeName="href" from="javascript:alert(1)" to="1" /> '
372+ . '<a id="xss"><text x="20" y="20">XSS</text></a></svg> ' ,
373+ '<svg><!-- animate blocked --><a id="xss"><text x="20" y="20">XSS</text></a></svg> ' ,
374+ ],
375+ [
376+ '<svg><set xlink:href="#xss" attributeName="href" from="?" to="javascript:alert(1)" /> '
377+ . '<a id="xss"><text x="20" y="20">XSS</text></a></svg> ' ,
378+ '<svg><!-- set blocked --><a id="xss"><text x="20" y="20">XSS</text></a></svg> ' ,
379+ ],
380+ [
381+ '<svg><animate xlink:href="#xss" attributename="href" dur="5s" repeatCount="indefinite" keytimes="0;0;1" values="https://portswigger.net?;javascript:alert(1);0" /> '
382+ . '<a id="xss"><text x="20" y="20">XSS</text></a></svg> ' ,
383+ '<svg><!-- animate blocked --><a id="xss"><text x="20" y="20">XSS</text></a></svg> ' ,
384+ ],
385+ [
386+ "<svg><use href= \" "
390+ . "My5vcmcvMTk5OS94bGluayIgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIiBpZD0ie "
391+ . "CIgd2lkdGg9IjEwMCIgaGVpZ2h0PSIxMDAiPjxhIHgtd2FzaGVkPSJ4bGluazpocmVmIj48cmVjdC "
392+ . "B4PSIwIiB5PSIwIiB3aWR0aD0iMTAwIiBoZWlnaHQ9IjEwMCIgLz48L2E+PC9zdmc+ \" /></svg> "
393+ ],
394+ [
395+ "<svg><use href= \"data:image/svg+xml;base64, " . base64_encode ($ svg1 ) . "\"></use></svg> " ,
396+ "<svg><use href= \" "
397+ . "0PSIxMDAiPjxhIHgtd2FzaGVkPSJ4bGluazpocmVmIj48cmVjdCB4PSIwIiB5PSIwIiB3aWR0aD0 "
398+ . "iMTAwIiBoZWlnaHQ9IjEwMCIgLz48L2E+PC9zdmc+ \" /></svg> "
399+ ],
400+ [
401+ '<svg><script href="data:text/javascript,alert(1)" /><text x="20" y="20">XSS</text></svg> ' ,
402+ '<svg><!-- script not allowed --><text x="20" y="20">XSS</text></svg> '
403+ ],
404+ ];
405+ }
342406
407+ /**
408+ * Test SVG cleanup
409+ *
410+ * @dataProvider data_wash_svg_tests
411+ */
412+ function test_wash_svg_tests ($ input , $ expected )
413+ {
343414 $ washer = new rcube_washtml ;
344- $ washed = $ washer ->wash ($ svg );
415+ $ washed = $ washer ->wash ($ input );
345416
346- $ this ->assertSame ($ washed , $ exp , "SVG content " );
417+ $ this ->assertSame ($ expected , $ washed , "SVG content " );
418+ }
347419
348- $ svg = '<svg xmlns="" onload="alert(document.domain)" /> ' ;
349- $ exp = '<svg xmlns="" onload="alert(document.domain)" /> ' ;
420+ /**
421+ * Test cases for various XSS issues
422+ */
423+ function data_wash_xss_tests ()
424+ {
425+ return [
426+ [
427+ '<html><base href="javascript:/a/-alert(1)///////"><a href="../lol/safari.html">test</a> ' ,
428+ '<!-- html ignored --><body><!-- base ignored --><a x-washed="href">test</a></body> '
429+ ],
430+ [
431+ '<html><math><x href="javascript:alert(1)">blah</x> ' ,
432+ '<!-- html ignored --><body><math><!-- x ignored -->blah</math></body> '
433+ ],
434+ [
435+ '<html><a href="javascript:alert(1)">XSS</a> ' ,
436+ '<!-- html ignored --><body><a x-washed="href">XSS</a></body> '
437+ ],
438+ [
439+ '<html><a href="j avascript:alert(1)">XSS</a> ' ,
440+ '<!-- html ignored --><body><a x-washed="href">XSS</a></body> '
441+ ],
442+ [
443+ '<html><a href="j avascript:alert(1)">XSS</a> ' ,
444+ '<!-- html ignored --><body><a x-washed="href">XSS</a></body> '
445+ ],
446+ [
447+ '<html><body background="javascript:alert(1)"> ' ,
448+ '<!-- html ignored --><body x-washed="background"></body> '
449+ ],
450+ [
451+ '<html><math href="javascript:alert(location);"><mi>clickme</mi></math> ' ,
452+ '<!-- html ignored --><body><math x-washed="href"><mi>clickme</mi></math></body> ' ,
453+ ],
454+ [
455+ '<html><math><mstyle href="javascript:alert(location);"><mi>clickme</mi></mstyle></math> ' ,
456+ '<!-- html ignored --><body><math><mstyle x-washed="href"><mi>clickme</mi></mstyle></math></body> ' ,
457+ ],
458+ [
459+ '<html><math><msubsup href="javascript:alert(location);"><mi>clickme</mi></msubsup></math> ' ,
460+ '<!-- html ignored --><body><math><msubsup x-washed="href"><mi>clickme</mi></msubsup></math></body> ' ,
461+ ],
462+ [
463+ '<html><math><ms HREF="javascript:alert(location);">clickme</ms></math> ' ,
464+ '<!-- html ignored --><body><math><ms x-washed="href">clickme</ms></math></body> ' ,
465+ ],
466+ ];
467+ }
350468
351- $ washer = new rcube_washtml ;
352- $ washed = $ washer ->wash ($ svg );
469+ /**
470+ * Test various XSS issues
471+ *
472+ * @dataProvider data_wash_xss_tests
473+ */
474+ function test_wash_xss_tests ($ input , $ expected )
475+ {
476+ $ washer = new rcube_washtml (['allow_remote ' => true , 'html_elements ' => ['body ' ]]);
477+ $ washed = $ washer ->wash ($ input );
353478
354- $ this ->assertSame ($ washed , $ exp , "SVG content " );
479+ $ this ->assertSame ($ expected , $ washed , "XSS issues " );
355480 }
356481
357482 /**
0 commit comments