From 40d7342dd9c9bd2a1d613edc848ed95a4d71aa18 Mon Sep 17 00:00:00 2001 From: Aleksander Machniak Date: Wed, 6 Jan 2016 14:10:05 +0100 Subject: [PATCH] Fix XSS issue in SVG images handling (#1490625) --- CHANGELOG | 1 + program/steps/mail/get.inc | 101 +++++++++++++++++++++++++++---------- 2 files changed, 76 insertions(+), 26 deletions(-) diff --git a/CHANGELOG b/CHANGELOG index 4a66c5138..dadf1f984 100644 --- a/CHANGELOG +++ b/CHANGELOG @@ -10,6 +10,7 @@ CHANGELOG Roundcube Webmail - Fix .htaccess rewrite rules to not block .well-known URIs (#1490615) - Fix mail view scaling on iOS (#1490551) - Fix PHP7 warning "session_start(): Session callback expects true/false return value" (#1490624) +- Fix XSS issue in SVG images handling (#1490625) RELEASE 1.2-beta ---------------- diff --git a/program/steps/mail/get.inc b/program/steps/mail/get.inc index 9806303f8..96cdd7717 100644 --- a/program/steps/mail/get.inc +++ b/program/steps/mail/get.inc @@ -93,6 +93,11 @@ else if ($_GET['_thumb']) { $mimetype = 'image/' . $imgtype; unlink($orig_name); } + else if (stripos($mimetype, 'image/svg') === 0) { + $content = rcmail_svg_filter(file_get_contents($orig_name)); + file_put_contents($cache_file, $content); + unlink($orig_name); + } else { rename($orig_name, $cache_file); } @@ -329,7 +334,7 @@ else if (strlen($part_id)) { } // convert image to jpeg and send it to the browser - if ($saved) { + if ($sent = $saved) { $image = new rcube_image($file_path); if ($image->convert(rcube_image::TYPE_JPG, $file_path)) { header("Content-Length: " . filesize($file_path)); @@ -338,32 +343,8 @@ else if (strlen($part_id)) { unlink($file_path); } } - // do content filtering to avoid XSS through fake images - else if (!empty($_REQUEST['_embed']) && $browser->ie && $browser->ver <= 8) { - if ($body) { - echo preg_match('/<(script|iframe|object)/i', $body) ? '' : $body; - $sent = true; - } - else if ($part->size) { - $stdout = fopen('php://output', 'w'); - stream_filter_register('rcube_content', 'rcube_content_filter') or die('Failed to register content filter'); - stream_filter_append($stdout, 'rcube_content'); - $sent = $MESSAGE->get_part_body($part->mime_id, true, 0, $stdout); - } - } - // send part as-it-is else { - if ($body && empty($plugin['download'])) { - header("Content-Length: " . strlen($body)); - echo $body; - $sent = true; - } - else if ($part->size) { - // Don't be tempted to set Content-Length to $part->d_parameters['size'] (#1490482) - // RFC2183 says "The size parameter indicates an approximate size" - - $sent = $MESSAGE->get_part_body($part->mime_id, false, 0, -1); - } + $sent = rcmail_message_part_output($body, $part, $mimetype, $plugin['download']); } // check connection status @@ -475,3 +456,71 @@ function rcmail_message_part_frame($attrib) return html::iframe($attrib); } + +/** + * Output attachment body with content filtering + */ +function rcmail_message_part_output($body, $part, $mimetype, $download) +{ + global $MESSAGE, $RCMAIL; + + if (!$part->size && !$body) { + return false; + } + + $browser = $RCMAIL->output->browser; + $secure = stripos($mimetype, 'image/') === false || $download; + + // Remove