[Pkg-roundcube-maintainers] CVE-2024-42009/roundcube: bullseye-security upload
Guilhem Moulin
guilhem at debian.org
Mon Aug 12 12:05:09 BST 2024
Hi,
On Tue, 06 Aug 2024 at 18:31:37 +0200, Guilhem Moulin wrote:
> Unfortunately for bullseye the backport requires more work. I'll try to
> do that ASAP but I'm not yet sure that I will have time to finalize it
> before the suite is handed over to the LTS team.
Thanks for the bookworm DSA! Here comes a debdiff for bullseye-security.
The backports were quite intrusive but have now been thoroughly tested,
incl. on two production instances totaling ~200 users AFAIK none of whom
didn't report any misbehavior.
Note that the original patch for CVE-2024-42008 introduces a regression
(#1078456): it sets a too restrictive Content-Security-Policy on the
attachment preview page which breaks printing and other handling of
image attachments. I backported the fix for 1.4.15+dfsg.1-1+deb11u4,
but 1.6.5+dfsg-1+deb12u3 is affected. I assume this doesn't warrant a
follow-up DSA, right? Will go via s-pu in that case.
Cheers,
--
Guilhem.
-------------- next part --------------
diffstat for roundcube-1.4.15+dfsg.1 roundcube-1.4.15+dfsg.1
changelog | 13
patches/CVE-2024-42008.patch | 71 +
patches/CVE-2024-42009.patch | 616 ++++++++++
patches/CVE-2024-42010.patch | 611 +++++++++
patches/Fix-infinite-loop-when-parsing-malformed-Sieve-script.patch | 52
patches/Fix-regression-where-printing-scaling-rotating-image-atta.patch | 71 +
patches/series | 5
7 files changed, 1439 insertions(+)
diff -Nru roundcube-1.4.15+dfsg.1/debian/changelog roundcube-1.4.15+dfsg.1/debian/changelog
--- roundcube-1.4.15+dfsg.1/debian/changelog 2024-06-17 04:10:38.000000000 +0200
+++ roundcube-1.4.15+dfsg.1/debian/changelog 2024-08-08 23:48:56.000000000 +0200
@@ -1,3 +1,16 @@
+roundcube (1.4.15+dfsg.1-1+deb11u4) bullseye-security; urgency=high
+
+ * Fix CVE-2024-42008: Cross-site scripting (XSS) vulnerability in serving of
+ attachments other than HTML or SVG.
+ * Fix CVE-2024-42009: Cross-site scripting (XSS) vulnerability in
+ post-processing of sanitized HTML content. (Closes: #1077969)
+ * Fix CVE-2024-42010: Information leak (access to remote content) via
+ insufficient CSS filtering.
+ * Backport upstream fix for infinite loop when parsing malformed Sieve
+ script.
+
+ -- Guilhem Moulin <guilhem at debian.org> Thu, 08 Aug 2024 23:48:56 +0200
+
roundcube (1.4.15+dfsg.1-1+deb11u3) bullseye-security; urgency=high
* Fix CVE-2024-37384: Cross-site scripting (XSS) vulnerability in handling
diff -Nru roundcube-1.4.15+dfsg.1/debian/patches/CVE-2024-42008.patch roundcube-1.4.15+dfsg.1/debian/patches/CVE-2024-42008.patch
--- roundcube-1.4.15+dfsg.1/debian/patches/CVE-2024-42008.patch 1970-01-01 01:00:00.000000000 +0100
+++ roundcube-1.4.15+dfsg.1/debian/patches/CVE-2024-42008.patch 2024-08-08 23:48:56.000000000 +0200
@@ -0,0 +1,71 @@
+From: Aleksander Machniak <alec at alec.pl>
+Date: Sat, 3 Aug 2024 09:10:55 +0200
+Subject: Fix XSS vulnerability in serving of attachments other than HTML or
+ SVG
+
+Credits to Oskar Zeino-Mahmalat (Sonar) https://www.sonarsource.com
+
+Origin: https://github.com/roundcube/roundcubemail/commit/c222ea8b99448ead20ab3864fcc29c84ed17403a
+Bug-Debian: https://security-tracker.debian.org/tracker/CVE-2024-42008
+Bug-Debian: https://bugs.debian.org/1077969
+---
+ program/lib/Roundcube/rcube_output.php | 20 +++++++++++++++-----
+ program/steps/mail/get.inc | 5 +++++
+ 2 files changed, 20 insertions(+), 5 deletions(-)
+
+diff --git a/program/lib/Roundcube/rcube_output.php b/program/lib/Roundcube/rcube_output.php
+index f24c47f..a51b75b 100644
+--- a/program/lib/Roundcube/rcube_output.php
++++ b/program/lib/Roundcube/rcube_output.php
+@@ -242,13 +242,20 @@ abstract class rcube_output
+ $ctype = $params['type'];
+ }
+
+- if ($disposition == 'inline' && stripos($ctype, 'text') === 0) {
+- $charset = $this->charset;
+- if (!empty($params['type_charset']) && rcube_charset::is_valid($params['type_charset'])) {
+- $charset = $params['type_charset'];
++ // Send unsafe content as plain text
++ if ($disposition == 'inline') {
++ if (preg_match('~(javascript|jscript|ecmascript|xml|html|text/)~i', $ctype)) {
++ $ctype = 'text/plain';
+ }
+
+- $ctype .= "; charset={$charset}";
++ if (stripos($ctype, 'text') === 0) {
++ $charset = $this->charset;
++ if (!empty($params['type_charset']) && rcube_charset::is_valid($params['type_charset'])) {
++ $charset = $params['type_charset'];
++ }
++
++ $ctype .= "; charset={$charset}";
++ }
+ }
+
+ if (is_string($filename) && strlen($filename) > 0 && strlen($filename) <= 1024) {
+@@ -278,6 +285,9 @@ abstract class rcube_output
+ header("Content-Length: " . $params['length']);
+ }
+
++ // Use strict security policy to make sure no javascript content is executed
++ header("Content-Security-Policy: default-src 'none'");
++
+ // don't kill the connection if download takes more than 30 sec.
+ if (!array_key_exists('time_limit', $params)) {
+ $params['time_limit'] = 3600;
+diff --git a/program/steps/mail/get.inc b/program/steps/mail/get.inc
+index 8e3647f..db152cb 100644
+--- a/program/steps/mail/get.inc
++++ b/program/steps/mail/get.inc
+@@ -240,6 +240,11 @@ if (empty($_GET['_thumb']) && $attachment->is_valid()) {
+ $RCMAIL->gettext('allow'),
+ $RCMAIL->url(array_merge($_GET, array('_safe' => 1)))
+ );
++ } else {
++ // Use strict security policy to make sure no javascript is executed
++ // TODO: Make the above "blocked resources button" working with strict policy
++ // TODO: Move this to rcmail_html_page::write()?
++ header("Content-Security-Policy: script-src 'none'");
+ }
+ }
+
diff -Nru roundcube-1.4.15+dfsg.1/debian/patches/CVE-2024-42009.patch roundcube-1.4.15+dfsg.1/debian/patches/CVE-2024-42009.patch
--- roundcube-1.4.15+dfsg.1/debian/patches/CVE-2024-42009.patch 1970-01-01 01:00:00.000000000 +0100
+++ roundcube-1.4.15+dfsg.1/debian/patches/CVE-2024-42009.patch 2024-08-08 23:48:56.000000000 +0200
@@ -0,0 +1,616 @@
+From: Aleksander Machniak <alec at alec.pl>
+Date: Sat, 3 Aug 2024 09:09:09 +0200
+Subject: Fix XSS vulnerability in post-processing of sanitized HTML content
+
+Credits to Oskar Zeino-Mahmalat (https://www.sonarsource.com)
+
+Origin: https://github.com/roundcube/roundcubemail/commit/1b3bb11d4f3880ba2ac8d1f398d32bf99be47122
+Origin: https://github.com/roundcube/roundcubemail/commit/a25e48e2daec522432fea3c37f3917366e2948d1
+Bug-Debian: https://security-tracker.debian.org/tracker/CVE-2024-42009
+Bug-Debian: https://bugs.debian.org/1077969
+---
+ program/lib/Roundcube/rcube_washtml.php | 46 +++++---
+ program/steps/mail/compose.inc | 37 ++----
+ program/steps/mail/func.inc | 189 +++++++++++++------------------
+ program/steps/mail/show.inc | 4 -
+ program/steps/settings/save_identity.inc | 9 +-
+ tests/MailFunc.php | 69 +++++++----
+ 6 files changed, 170 insertions(+), 184 deletions(-)
+
+diff --git a/program/lib/Roundcube/rcube_washtml.php b/program/lib/Roundcube/rcube_washtml.php
+index 8cb0d24..8a3a5ab 100644
+--- a/program/lib/Roundcube/rcube_washtml.php
++++ b/program/lib/Roundcube/rcube_washtml.php
+@@ -164,7 +164,14 @@ class rcube_washtml
+ public $extlinks = false;
+
+ /** @var array Current settings */
+- private $config = array();
++ private $config = array(
++ 'add_comments' => true,
++ 'allow_remote' => false,
++ 'base_url' => '',
++ 'charset' => RCUBE_CHARSET,
++ 'cid_map' => array(),
++ 'show_washed' => true,
++ );
+
+ /** @var array Registered callback functions for tags */
+ private $handlers = array();
+@@ -217,7 +224,7 @@ class rcube_washtml
+
+ unset($p['html_elements'], $p['html_attribs'], $p['ignore_elements'], $p['void_elements'], $p['css_prefix']);
+
+- $this->config = $p + array('show_washed' => true, 'allow_remote' => false, 'cid_map' => array());
++ $this->config = array_merge($this->config, $p);
+ }
+
+ /**
+@@ -562,7 +569,7 @@ class rcube_washtml
+ true, false);
+ }
+
+- return '<!-- ignored -->';
++ return $this->config['add_comments'] ? '<!-- ignored -->' : '';
+ }
+
+ $node = $node->firstChild;
+@@ -576,7 +583,9 @@ class rcube_washtml
+ if ($tagName == 'link') {
+ $uri = $this->wash_uri($node->getAttribute('href'), false, false);
+ if (!$uri) {
+- $dump .= '<!-- link ignored -->';
++ if ($this->config['add_comments']) {
++ $dump .= '<!-- link ignored -->';
++ }
+ break;
+ }
+
+@@ -586,7 +595,9 @@ class rcube_washtml
+ && self::attribute_value($node, 'attributename', 'href')
+ ) {
+ // Insecure svg tags
+- $dump .= "<!-- $tagName blocked -->";
++ if ($this->config['add_comments']) {
++ $dump .= "<!-- {$tagName} blocked -->";
++ }
+ break;
+ }
+
+@@ -596,13 +607,13 @@ class rcube_washtml
+ }
+ else if (isset($this->_html_elements[$tagName])) {
+ $content = $this->dumpHtml($node, $level);
+- $dump .= '<' . $node->nodeName;
++ $tag = '<' . $node->nodeName;
+
+ if ($tagName == 'svg') {
+ $xpath = new DOMXPath($node->ownerDocument);
+ foreach ($xpath->query('namespace::*') as $ns) {
+ if ($ns->nodeName != 'xmlns:xml') {
+- $dump .= sprintf(' %s="%s"',
++ $tag .= sprintf(' %s="%s"',
+ $ns->nodeName,
+ htmlspecialchars($ns->nodeValue, ENT_QUOTES, $this->config['charset'])
+ );
+@@ -613,20 +624,25 @@ class rcube_washtml
+ $content = htmlspecialchars($content, ENT_QUOTES | ENT_SUBSTITUTE, $this->config['charset']);
+ }
+
+- $dump .= $this->wash_attribs($node);
++ $tag .= $this->wash_attribs($node);
+
+- if ($content === '' && ($this->is_xml || isset($this->_void_elements[$tagName]))) {
+- $dump .= ' />';
+- }
+- else {
+- $dump .= '>' . $content . '</' . $node->nodeName . '>';
++ if (isset($this->_ignore_elements[$tagName])) {
++ $dump .= $content;
++ } elseif ($content === '' && ($this->is_xml || isset($this->_void_elements[$tagName]))) {
++ $dump .= $tag . ' />';
++ } else {
++ $dump .= $tag . '>' . $content . '</' . $node->nodeName . '>';
+ }
+ }
+ else if (isset($this->_ignore_elements[$tagName])) {
+- $dump .= '<!-- ' . htmlspecialchars($node->nodeName, ENT_QUOTES, $this->config['charset']) . ' not allowed -->';
++ if ($this->config['add_comments']) {
++ $dump .= '<!-- ' . htmlspecialchars($node->nodeName, ENT_QUOTES, $this->config['charset']) . ' not allowed -->';
++ }
+ }
+ else {
+- $dump .= '<!-- ' . htmlspecialchars($node->nodeName, ENT_QUOTES, $this->config['charset']) . ' ignored -->';
++ if ($this->config['add_comments']) {
++ $dump .= '<!-- ' . htmlspecialchars($node->nodeName, ENT_QUOTES, $this->config['charset']) . ' ignored -->';
++ }
+ $dump .= $this->dumpHtml($node, $level); // ignore tags not its content
+ }
+ break;
+diff --git a/program/steps/mail/compose.inc b/program/steps/mail/compose.inc
+index 73be67b..06db2b4 100644
+--- a/program/steps/mail/compose.inc
++++ b/program/steps/mail/compose.inc
+@@ -335,7 +335,7 @@ function rcmail_process_compose_params(&$COMPOSE)
+ // clean HTML message body which can be submitted by URL
+ if (!empty($COMPOSE['param']['body'])) {
+ if ($COMPOSE['param']['html'] = strpos($COMPOSE['param']['body'], '<') !== false) {
+- $wash_params = array('safe' => false, 'inline_html' => true);
++ $wash_params = array('safe' => false);
+ $COMPOSE['param']['body'] = rcmail_prepare_html_body($COMPOSE['param']['body'], $wash_params);
+ }
+ }
+@@ -880,40 +880,25 @@ function rcmail_prepare_html_body($body, $wash_params = array())
+ static $part_no;
+
+ // Set attributes of the part container
+- $container_id = $COMPOSE['mode'] . 'body' . (++$part_no);
+- $container_attrib = array('id' => $container_id);
+-
+- $body_args = array(
+- 'safe' => $MESSAGE->is_safe,
+- 'plain' => false,
+- 'css_prefix' => 'v' . $part_no,
++ $container_id = $COMPOSE['mode'] . 'body' . (++$part_no);
++ $wash_params += array(
++ 'safe' => $MESSAGE->is_safe,
++ 'css_prefix' => 'v' . $part_no,
++ 'add_comments' => false,
+ );
+
+- // remove comments (produced by washtml)
+- $replace = array('/<!--[^>]+-->/' => '');
+-
+ if ($COMPOSE['mode'] == rcmail_sendmail::MODE_DRAFT) {
+ // convert TinyMCE's empty-line sequence (#1490463)
+- $replace['/<p>\xC2\xA0<\/p>/'] = '<p><br /></p>';
+- // remove <body> tags
+- $replace['/<body([^>]*)>/i'] = '';
+- $replace['/<\/body>/i'] = '';
++ $body = preg_replace('/<p>\xC2\xA0<\/p>/', '<p><br /></p>', $body);
++ // remove <body> tags (not their content)
++ $wash_params['ignore_elements'] = array('body');
+ }
+ else {
+- $body_args['container_id'] = $container_id;
+- $body_args['container_attrib'] = $container_attrib;
++ $wash_params['container_id'] = $container_id;
+ }
+
+ // Make the HTML content safe and clean
+- $body = rcmail_wash_html($body, $wash_params + $body_args, $CID_MAP);
+- $body = preg_replace(array_keys($replace), array_values($replace), $body);
+- $body = rcmail_html4inline($body, $body_args);
+-
+- if ($COMPOSE['mode'] != rcmail_sendmail::MODE_DRAFT) {
+- $body = html::div($container_attrib, $body);
+- }
+-
+- return $body;
++ return rcmail_wash_html($body, $wash_params, $CID_MAP);
+ }
+
+ // Removes signature from the message body
+diff --git a/program/steps/mail/func.inc b/program/steps/mail/func.inc
+index 4b96270..f8aed12 100644
+--- a/program/steps/mail/func.inc
++++ b/program/steps/mail/func.inc
+@@ -836,6 +836,7 @@ function rcmail_check_safe($message)
+ function rcmail_wash_html($html, $p, $cid_replaces = array())
+ {
+ global $REMOTE_OBJECTS, $RCMAIL;
++ global $wash_html_body_attrs;
+
+ $p += array('safe' => false, 'inline_html' => true);
+
+@@ -862,13 +863,17 @@ function rcmail_wash_html($html, $p, $cid_replaces = array())
+ // clean HTML with washhtml by Frederic Motte
+ $wash_opts = array(
+ 'show_washed' => false,
++ 'add_comments' => isset($p['add_comments']) ? $p['add_comments'] : true,
+ 'allow_remote' => $p['safe'],
+ 'blocked_src' => $RCMAIL->output->asset_url('program/resources/blocked.gif'),
+ 'charset' => RCUBE_CHARSET,
+ 'cid_map' => $cid_replaces,
+ 'html_elements' => array('body'),
+ 'css_prefix' => $p['css_prefix'],
++ 'ignore_elements' => isset($p['ignore_elements']) ? $p['ignore_elements'] : array(),
++ // internal configuration
+ 'container_id' => $p['container_id'],
++ 'body_class' => isset($p['body_class']) ? $p['body_class'] : '',
+ );
+
+ if (!$p['inline_html']) {
+@@ -889,6 +894,20 @@ function rcmail_wash_html($html, $p, $cid_replaces = array())
+ // initialize HTML washer
+ $washer = new rcube_washtml($wash_opts);
+
++ $wash_html_body_attrs = array();
++
++ if ($p['inline_html']) {
++ $washer->add_callback('body', 'rcmail_washtml_callback');
++
++ if ($wash_opts['body_class']) {
++ $wash_html_body_attrs['class'] = $wash_opts['body_class'];
++ }
++
++ if ($wash_opts['container_id']) {
++ $wash_html_body_attrs['id'] = $wash_opts['container_id'];
++ }
++ }
++
+ if (!$p['skip_washer_form_callback']) {
+ $washer->add_callback('form', 'rcmail_washtml_callback');
+ }
+@@ -911,6 +930,11 @@ function rcmail_wash_html($html, $p, $cid_replaces = array())
+ $html = $washer->wash($html);
+ $REMOTE_OBJECTS = $washer->extlinks;
+
++ // There was no <body>, but a wrapper element is required
++ if ($p['inline_html'] && !empty($wash_html_body_attrs)) {
++ $html = html::tag('div', $wash_html_body_attrs, $html);
++ }
++
+ return $html;
+ }
+
+@@ -999,6 +1023,7 @@ function rcmail_plain_body($body, $flowed = false, $delsp = false)
+ */
+ function rcmail_washtml_callback($tagname, $attrib, $content, $washtml)
+ {
++ global $wash_html_body_attrs;
+ switch ($tagname) {
+ case 'form':
+ $out = html::div('form', $content);
+@@ -1022,11 +1047,65 @@ function rcmail_washtml_callback($tagname, $attrib, $content, $washtml)
+ $washtml->extlinks = true;
+ }
+ else {
+- $out = html::tag('style', array('type' => 'text/css'), $decoded);
++ $out = $decoded;
+ }
+- break;
+ }
+
++ if (strlen($out)) {
++ $css_prefix = $washtml->get_config('css_prefix');
++ $is_safe = $washtml->get_config('allow_remote');
++ $body_class = $washtml->get_config('body_class') ?: '';
++ $cont_id = $washtml->get_config('container_id') ?: '';
++ $cont_id = trim($cont_id . ($body_class ? " div.{$body_class}" : ''));
++
++ $out = rcube_utils::mod_css_styles($out, $cont_id, $is_safe, $css_prefix);
++
++ $out = html::tag('style', array('type' => 'text/css'), $out);
++ }
++
++ break;
++
++ case 'body':
++ $style = array();
++ $attrs = $wash_html_body_attrs;
++
++ foreach (html::parse_attrib_string($attrib) as $attr_name => $value) {
++ switch (strtolower($attr_name)) {
++ case 'bgcolor':
++ // Get bgcolor, we'll set it as background-color of the message container
++ if (preg_match('/^([a-z0-9#]+)$/i', $value, $m)) {
++ $style['background-color'] = $value;
++ }
++ break;
++ case 'text':
++ // Get text color, we'll set it as font color of the message container
++ if (preg_match('/^([a-z0-9#]+)$/i', $value, $m)) {
++ $style['color'] = $value;
++ }
++ break;
++ case 'background':
++ // Get background, we'll set it as background-image of the message container
++ if (preg_match('/^([^\s]+)$/', $value, $m)) {
++ $style['background-image'] = "url({$value})";
++ }
++ break;
++ default:
++ $attrs[$attr_name] = $value;
++ }
++ }
++
++ if (!empty($style)) {
++ foreach ($style as $idx => $val) {
++ $style[$idx] = $idx . ': ' . $val;
++ }
++
++ $attrs['style'] = ($attrs['style'] ? trim($attrs['style'], ';') . '; ' : '') . implode('; ', $style);
++ }
++
++ $out = html::tag('div', $attrs, $content);
++ $wash_html_body_attrs = array();
++ break;
++
+ default:
+ $out = '';
+ }
+@@ -1083,112 +1162,6 @@ function rcmail_part_image_type($part)
+ }
+ }
+
+-/**
+- * Modify a HTML message that it can be displayed inside a HTML page
+- */
+-function rcmail_html4inline($body, &$args)
+-{
+- $last_pos = 0;
+- $cont_id = $args['container_id'] . ($args['body_class'] ? ' div.' . $args['body_class'] : '');
+-
+- // find STYLE tags
+- while (($pos = stripos($body, '<style', $last_pos)) !== false && ($pos2 = stripos($body, '</style>', $pos+1))) {
+- $pos = strpos($body, '>', $pos) + 1;
+- $len = $pos2 - $pos;
+-
+- // replace all css definitions with #container [def]
+- $styles = substr($body, $pos, $len);
+- $styles = rcube_utils::mod_css_styles($styles, $cont_id, $args['safe'], $args['css_prefix']);
+-
+- $body = substr_replace($body, $styles, $pos, $len);
+- $last_pos = $pos2 + strlen($styles) - $len;
+- }
+-
+- $replace = array(
+- // add comments around html and other tags
+- '/(<!DOCTYPE[^>]*>)/i' => '<!--\\1-->',
+- '/(<\?xml[^>]*>)/i' => '<!--\\1-->',
+- '/(<\/?html[^>]*>)/i' => '<!--\\1-->',
+- '/(<\/?head[^>]*>)/i' => '<!--\\1-->',
+- '/(<title[^>]*>.*<\/title>)/Ui' => '<!--\\1-->',
+- '/(<\/?meta[^>]*>)/i' => '<!--\\1-->',
+- // quote <? of php and xml files that are specified as text/html
+- '/<\?/' => '<?',
+- '/\?>/' => '?>',
+- );
+-
+- $regexp = '/<body([^>]*)/';
+-
+- // Handle body attributes that doesn't play nicely with div elements
+- if (preg_match($regexp, $body, $m)) {
+- $style = array();
+- $attrs = $m[0];
+-
+- // Get bgcolor, we'll set it as background-color of the message container
+- if ($m[1] && preg_match('/bgcolor=["\']*([a-z0-9#]+)["\']*/i', $attrs, $mb)) {
+- $style['background-color'] = $mb[1];
+- $attrs = preg_replace('/\s?bgcolor=["\']*[a-z0-9#]+["\']*/i', '', $attrs);
+- }
+-
+- // Get text color, we'll set it as font color of the message container
+- if ($m[1] && preg_match('/text=["\']*([a-z0-9#]+)["\']*/i', $attrs, $mb)) {
+- $style['color'] = $mb[1];
+- $attrs = preg_replace('/\s?text=["\']*[a-z0-9#]+["\']*/i', '', $attrs);
+- }
+-
+- // Get background, we'll set it as background-image of the message container
+- if ($m[1] && preg_match('/background=["\']*([^"\'>\s]+)["\']*/', $attrs, $mb)) {
+- $style['background-image'] = 'url('.$mb[1].')';
+- $attrs = preg_replace('/\s?background=["\']*([^"\'>\s]+)["\']*/', '', $attrs);
+- }
+-
+- if (!empty($style)) {
+- $body = preg_replace($regexp, rtrim($attrs), $body, 1);
+- }
+-
+- // handle body styles related to background image
+- if ($style['background-image']) {
+- // get body style
+- if (preg_match('/#'.preg_quote($cont_id, '/').'\s+\{([^}]+)}/i', $body, $m)) {
+- // get background related style
+- $regexp = '/(background-position|background-repeat)\s*:\s*([^;]+);/i';
+- if (preg_match_all($regexp, $m[1], $matches, PREG_SET_ORDER)) {
+- foreach ($matches as $m) {
+- $style[$m[1]] = $m[2];
+- }
+- }
+- }
+- }
+-
+- if (!empty($style)) {
+- foreach ($style as $idx => $val) {
+- $style[$idx] = $idx . ': ' . $val;
+- }
+-
+- $args['container_attrib']['style'] = implode('; ', $style);
+- }
+-
+- // replace <body> with <div>
+- if (!empty($args['body_class'])) {
+- $replace['/<body([^>]*)>/i'] = '<div class="' . $args['body_class'] . '"\\1>';
+- }
+- else {
+- $replace['/<body/i'] = '<div';
+- }
+-
+- $replace['/<\/body>/i'] = '</div>';
+- }
+- // make sure there's 'rcmBody' div, we need it for proper css modification
+- // its name is hardcoded in rcmail_message_body() also
+- else if (!empty($args['body_class'])) {
+- $body = '<div class="' . $args['body_class'] . '">' . $body . '</div>';
+- }
+-
+- // Clean up, and replace <body> with <div>
+- $body = preg_replace(array_keys($replace), array_values($replace), $body);
+-
+- return $body;
+-}
+
+ /**
+ * Parse link (a, link, area) attributes and set correct target
+diff --git a/program/steps/mail/show.inc b/program/steps/mail/show.inc
+index f7c6819..6ddc4bd 100644
+--- a/program/steps/mail/show.inc
++++ b/program/steps/mail/show.inc
+@@ -687,10 +687,6 @@ function rcmail_message_body($attrib)
+ $OUTPUT->set_env('is_pgp_content', '#' . $container_id);
+ }
+
+- if ($part->ctype_secondary == 'html') {
+- $body = rcmail_html4inline($body, $body_args);
+- }
+-
+ $out .= html::div($body_args['container_attrib'], $plugin['prefix'] . $body);
+ }
+ }
+diff --git a/program/steps/settings/save_identity.inc b/program/steps/settings/save_identity.inc
+index 052941e..ab44b77 100644
+--- a/program/steps/settings/save_identity.inc
++++ b/program/steps/settings/save_identity.inc
+@@ -244,6 +244,8 @@ function rcmail_wash_html($html)
+ 'charset' => RCUBE_CHARSET,
+ 'html_elements' => array('body', 'link'),
+ 'html_attribs' => array('rel', 'type'),
++ 'ignore_elements' => array('body'),
++ 'add_comments' => false,
+ );
+
+ // initialize HTML washer
+@@ -255,10 +257,5 @@ function rcmail_wash_html($html)
+ // Remove non-UTF8 characters (#1487813)
+ $html = rcube_charset::clean($html);
+
+- $html = $washer->wash($html);
+-
+- // remove unwanted comments and tags (produced by washtml)
+- $html = preg_replace(array('/<!--[^>]+-->/', '/<\/?body>/'), '', $html);
+-
+- return $html;
++ return $washer->wash($html);
+ }
+diff --git a/tests/MailFunc.php b/tests/MailFunc.php
+index a72b5c9..9c0ae05 100644
+--- a/tests/MailFunc.php
++++ b/tests/MailFunc.php
+@@ -41,11 +41,11 @@ class MailFunc extends \PHPUnit\Framework\TestCase
+ $part = $this->get_html_part('src/htmlbody.txt');
+ $part->replaces = array('ex1.jpg' => 'part_1.2.jpg', 'ex2.jpg' => 'part_1.2.jpg');
+
+- $params = array('container_id' => 'foo');
++ $params = array('container_id' => 'foo', 'safe' => false);
+
+ // render HTML in normal mode
+- $html = rcmail_html4inline(rcmail_print_body($part->body, $part, array('safe' => false)), $params);
+-
++ $html = rcmail_print_body($part->body, $part, $params);
++
+ $this->assertMatchesRegularExpression('/src="'.$part->replaces['ex1.jpg'].'"/', $html, "Replace reference to inline image");
+ $this->assertMatchesRegularExpression('#background="program/resources/blocked.gif"#', $html, "Replace external background image");
+ $this->assertDoesNotMatchRegularExpression('/ex3.jpg/', $html, "No references to external images");
+@@ -57,7 +57,8 @@ class MailFunc extends \PHPUnit\Framework\TestCase
+ $this->assertTrue($GLOBALS['REMOTE_OBJECTS'], "Remote object detected");
+
+ // render HTML in safe mode
+- $html2 = rcmail_html4inline(rcmail_print_body($part->body, $part, array('safe' => true)), $params);
++ $params['safe'] = true;
++ $html2 = rcmail_print_body($part->body, $part, $params);
+
+ $this->assertMatchesRegularExpression('/<style [^>]+>/', $html2, "Allow styles in safe mode");
+ $this->assertMatchesRegularExpression('#src="http://evilsite.net/mailings/ex3.jpg"#', $html2, "Allow external images in HTML (safe mode)");
+@@ -72,15 +73,12 @@ class MailFunc extends \PHPUnit\Framework\TestCase
+ function test_html_xss()
+ {
+ $part = $this->get_html_part('src/htmlxss.txt');
+- $washed = rcmail_print_body($part->body, $part, array('safe' => true));
+-
+- $this->assertDoesNotMatchRegularExpression('/src="skins/', $washed, "Remove local references");
+- $this->assertDoesNotMatchRegularExpression('/\son[a-z]+/', $washed, "Remove on* attributes");
+- $this->assertStringNotContainsString('onload', $washed, "Handle invalid style");
+-
+- $params = array('container_id' => 'foo');
+- $html = rcmail_html4inline($washed, $params);
++ $params = array('container_id' => 'foo', 'safe' => true);
++ $html = rcmail_print_body($part->body, $part, $params);
+
++ $this->assertDoesNotMatchRegularExpression('/src="skins/', $html, 'Remove local references');
++ $this->assertDoesNotMatchRegularExpression('/\son[a-z]+/', $html, 'Remove on* attributes');
++ $this->assertStringNotContainsString('onload', $html, 'Handle invalid style');
+ $this->assertDoesNotMatchRegularExpression('/onclick="return rcmail.command(\'compose\',\'xss at somehost.net\',this)"/', $html, "Clean mailto links");
+ $this->assertDoesNotMatchRegularExpression('/alert/', $html, "Remove alerts");
+ }
+@@ -93,7 +91,7 @@ class MailFunc extends \PHPUnit\Framework\TestCase
+ {
+ $part = $this->get_html_part('src/BID-26800.txt');
+ $params = array('container_id' => 'dabody', 'safe' => true);
+- $washed = rcmail_html4inline(rcmail_print_body($part->body, $part, array('safe' => true)), $params);
++ $washed = rcmail_print_body($part->body, $part, $params);
+
+ $this->assertDoesNotMatchRegularExpression('/alert|expression|javascript|xss/', $washed, "Remove evil style blocks");
+ $this->assertDoesNotMatchRegularExpression('/font-style:italic/', $washed, "Allow valid styles");
+@@ -113,20 +111,40 @@ class MailFunc extends \PHPUnit\Framework\TestCase
+ $this->assertDoesNotMatchRegularExpression('/vbscript:/', $washed, "Remove vbscript: links");
+ }
+
++ /**
++ * Test that HTML sanitization does not change attribute (evil) values
++ */
++ public function test_html_body_attributes()
++ {
++ $part = $this->get_html_part();
++ $part->body = '<body title="bgcolor=foo" name="bar style=animation-name:progress-bar-stripes onanimationstart=alert(origin) foo=bar">Foo</body>';
++
++ $params = array('safe' => true, 'add_comments' => false);
++ $washed = rcmail_print_body($part->body, $part, $params);
++
++ $this->assertSame(str_replace('body', 'div', $part->body) . "\n", $washed);
++
++ $params['inline_html'] = false;
++ $washed = rcmail_print_body($part->body, $part, $params);
++
++ $this->assertSame('<html><head></head>' . $part->body . '</html>', $washed);
++ }
++
+ /**
+ * Test handling of body style attributes
+ */
+- function test_html4inline_body_style()
++ public function test_wash_html_body_style()
+ {
+- $html = '<body background="test" bgcolor="#fff" style="font-size:11px" text="#000"><p>test</p></body>';
+- $params = array('container_id' => 'foo');
+- $html = rcmail_html4inline($html, $params);
+-
+- $this->assertMatchesRegularExpression('/<div style="font-size:11px">/', $html, "Body attributes");
+- $this->assertArrayHasKey('container_attrib', $params, "'container_attrib' param set");
+- $this->assertMatchesRegularExpression('/background-color: #fff;/', $params['container_attrib']['style'], "Body style (bgcolor)");
+- $this->assertMatchesRegularExpression('/background-image: url\(test\)/', $params['container_attrib']['style'], "Body style (background)");
+- $this->assertMatchesRegularExpression('/color: #000/', $params['container_attrib']['style'], "Body style (text)");
++ $html = '<body background="http://test.com/image" bgcolor="#fff" style="font-size: 11px" text="#000"><p>test</p></body>';
++ $params = array('container_id' => 'foo', 'add_comments' => false, 'safe' => false);
++ $washed = rcmail_wash_html($html, $params, array());
++
++ $this->assertSame('<div id="foo" style="font-size: 11px; background-image: url(program/resources/blocked.gif); background-color: #fff; color: #000"><p>test</p></div>' . "\n", $washed);
++
++ $params['safe'] = true;
++ $washed = rcmail_wash_html($html, $params, array());
++
++ $this->assertSame('<div id="foo" style="font-size: 11px; background-image: url(http://test.com/image); background-color: #fff; color: #000"><p>test</p></div>' . "\n", $washed);
+ }
+
+ /**
+@@ -149,6 +167,7 @@ class MailFunc extends \PHPUnit\Framework\TestCase
+ {
+ $meta = '<meta charset="'.RCUBE_CHARSET.'" />';
+ $args = array(
++ 'inline_html' => false,
+ 'html_elements' => array('html', 'body', 'meta', 'head'),
+ 'html_attribs' => array('charset'),
+ );
+@@ -201,10 +220,10 @@ class MailFunc extends \PHPUnit\Framework\TestCase
+ function test_mailto()
+ {
+ $part = $this->get_html_part('src/mailto.txt');
+- $params = array('container_id' => 'foo');
++ $params = array('container_id' => 'foo', 'safe' => false);
+
+ // render HTML in normal mode
+- $html = rcmail_html4inline(rcmail_print_body($part->body, $part, array('safe' => false)), $params);
++ $html = rcmail_print_body($part->body, $part, $params);
+
+ $mailto = '<a href="mailto:me at me.com"'
+ .' onclick="return rcmail.command(\'compose\',\'me at me.com?subject=this is the subject&body=this is the body\',this)" rel="noreferrer">e-mail</a>';
diff -Nru roundcube-1.4.15+dfsg.1/debian/patches/CVE-2024-42010.patch roundcube-1.4.15+dfsg.1/debian/patches/CVE-2024-42010.patch
--- roundcube-1.4.15+dfsg.1/debian/patches/CVE-2024-42010.patch 1970-01-01 01:00:00.000000000 +0100
+++ roundcube-1.4.15+dfsg.1/debian/patches/CVE-2024-42010.patch 2024-08-08 23:48:56.000000000 +0200
@@ -0,0 +1,611 @@
+From: Aleksander Machniak <alec at alec.pl>
+Date: Sat, 3 Aug 2024 09:24:19 +0200
+Subject: Fix information leak (access to remote content) via insufficient CSS
+ filtering
+
+Credits to Oskar Zeino-Mahmalat (Sonar) https://www.sonarsource.com
+
+Origin: https://github.com/roundcube/roundcubemail/commit/53da61f7fc8a5d8414c36c45e44901d2a568319a
+Origin: https://github.com/roundcube/roundcubemail/commit/9f19b931e3b89c2fa577e2bf719f7db84492eb66
+Bug-Debian: https://security-tracker.debian.org/tracker/CVE-2024-42010
+Bug-Debian: https://bugs.debian.org/1077969
+---
+ program/lib/Roundcube/rcube_utils.php | 236 ++++++++++++++++++++++++++------
+ program/lib/Roundcube/rcube_washtml.php | 95 +------------
+ tests/Framework/Utils.php | 146 +++++++++++++++++---
+ tests/Framework/Washtml.php | 2 +-
+ 4 files changed, 327 insertions(+), 152 deletions(-)
+
+diff --git a/program/lib/Roundcube/rcube_utils.php b/program/lib/Roundcube/rcube_utils.php
+index d427352..68f8684 100644
+--- a/program/lib/Roundcube/rcube_utils.php
++++ b/program/lib/Roundcube/rcube_utils.php
+@@ -373,19 +373,40 @@ class rcube_utils
+ */
+ public static function mod_css_styles($source, $container_id, $allow_remote = false, $prefix = '')
+ {
+- $last_pos = 0;
+- $replacements = new rcube_string_replacer;
+-
+- // ignore the whole block if evil styles are detected
+ $source = self::xss_entity_decode($source);
+- $stripped = preg_replace('/[^a-z\(:;]/i', '', $source);
+- $evilexpr = 'expression|behavior|javascript:|import[^a]' . (!$allow_remote ? '|url\((?!data:image)' : '');
+
+- if (preg_match("/$evilexpr/i", $stripped)) {
++ // No @import allowed
++ // TODO: We should just remove it, not invalidate the whole content
++ if (stripos($source, '@import') !== false) {
++ return '/* evil! */';
++ }
++
++ // Incomplete style expression
++ if (strpos($source, '{') === false) {
++ return '/* invalid! */';
++ }
++
++ // To prevent from a double-escaping tricks we consider a script with
++ // any escape sequences (after de-escaping them above) an evil script.
++ // This probably catches many valid scripts, but we\'re on the safe side.
++ if (preg_match('/\\\[0-9a-fA-F]{2}/', $source)) {
+ return '/* evil! */';
+ }
+
+- $strict_url_regexp = '!url\s*\(\s*["\']?(https?:)//[a-z0-9/._+-]+["\']?\s*\)!Uims';
++ // remove html comments
++ $source = preg_replace('/(^\s*<\!--)|(-->\s*$)/m', '', $source);
++
++ $url_callback = static function ($url) use ($allow_remote) {
++ if (strpos($url, 'data:image') === 0) {
++ return $url;
++ }
++ if ($allow_remote && preg_match('|^https?://[a-z0-9/._+-]+$|i', $url)) {
++ return $url;
++ }
++ };
++
++ $last_pos = 0;
++ $replacements = new rcube_string_replacer();
+
+ // cut out all contents between { and }
+ while (($pos = strpos($source, '{', $last_pos)) && ($pos2 = strpos($source, '}', $pos))) {
+@@ -394,45 +415,14 @@ class rcube_utils
+ $pos = $nested;
+ $length = $pos2 - $pos - 1;
+ $styles = substr($source, $pos+1, $length);
++ $styles = self::sanitize_css_block($styles, $url_callback);
+
+- // Convert position:fixed to position:absolute (#5264)
+- $styles = preg_replace('/position[^a-z]*:[\s\r\n]*fixed/i', 'position: absolute', $styles);
+-
+- // Remove 'page' attributes (#7604)
+- $styles = preg_replace('/((^|[\n\s;])page:)[^;]+;*/im', '\\1 unset;', $styles);
+-
+- // check every line of a style block...
+- if ($allow_remote) {
+- $a_styles = preg_split('/;[\r\n]*/', $styles, -1, PREG_SPLIT_NO_EMPTY);
+-
+- for ($i=0, $len=count($a_styles); $i < $len; $i++) {
+- $line = $a_styles[$i];
+- $stripped = preg_replace('/[^a-z\(:;]/i', '', $line);
+-
+- // allow data:image uri, join with continuation
+- if (stripos($stripped, 'url(data:image')) {
+- $a_styles[$i] .= ';' . $a_styles[$i+1];
+- unset($a_styles[$i+1]);
+- }
+- // allow strict url() values only
+- else if (stripos($stripped, 'url(') && !preg_match($strict_url_regexp, $line)) {
+- $a_styles = array('/* evil! */');
+- break;
+- }
+- }
+-
+- $styles = implode(";\n", $a_styles);
+- }
+-
+- $key = $replacements->add($styles);
++ $key = $replacements->add(strlen($styles) ? " {$styles} " : '');
+ $repl = $replacements->get_replacement($key);
+ $source = substr_replace($source, $repl, $pos+1, $length);
+ $last_pos = $pos2 - ($length - strlen($repl));
+ }
+
+- // remove html comments
+- $source = preg_replace('/(^\s*<\!--)|(-->\s*$)/m', '', $source);
+-
+ // add #container to each tag selector and prefix to id/class identifiers
+ if ($container_id || $prefix) {
+ // Exclude rcube_string_replacer pattern matches, this is needed
+@@ -475,6 +465,170 @@ class rcube_utils
+ return $source;
+ }
+
++ /**
++ * Parse and sanitize single CSS block
++ *
++ * @param string $styles CSS styles block
++ * @param ?callable $url_callback URL validator callback
++ *
++ * @return string
++ */
++ public static function sanitize_css_block($styles, $url_callback = null)
++ {
++ $output = array();
++
++ // check every css rule in the style block...
++ foreach (self::parse_css_block($styles) as $rule) {
++ $property = $rule[0];
++ $value = $rule[1];
++
++ if ($property == 'page') {
++ // Remove 'page' attributes (#7604)
++ continue;
++ } elseif ($property == 'position' && strcasecmp($value, 'fixed') === 0) {
++ // Convert position:fixed to position:absolute (#5264)
++ $value = 'absolute';
++ } elseif (preg_match('/expression|image-set/i', $value)) {
++ continue;
++ } else {
++ $value = '';
++ foreach (self::explode_css_property_block($rule[1]) as $val) {
++ if ($url_callback && preg_match('/^url\s*\(/i', $val)) {
++ if (preg_match('/^url\s*\(\s*[\'"]?([^\'"\)]*)[\'"]?\s*\)/iu', $val, $match)) {
++ if ($url = $url_callback($match[1])) {
++ $value .= ' url(' . $url . ')';
++ }
++ }
++ } else {
++ // whitelist ?
++ $value .= ' ' . $val;
++
++ // #1488535: Fix size units, so width:800 would be changed to width:800px
++ if ($val
++ && preg_match('/^(left|right|top|bottom|width|height)/i', $property)
++ && preg_match('/^[0-9]+$/', $val)
++ ) {
++ $value .= 'px';
++ }
++ }
++ }
++ }
++
++ if (strlen($value)) {
++ $output[] = $property . ': ' . trim($value);
++ }
++ }
++
++ return count($output) > 0 ? implode('; ', $output) . ';' : '';
++ }
++
++ /**
++ * Explode css style. Property names will be lower-cased and trimmed.
++ * Values will be trimmed. Invalid entries will be skipped.
++ *
++ * @param string $style CSS style
++ *
++ * @return array List of CSS rule pairs, e.g. [['color', 'red'], ['top', '0']]
++ */
++ public static function parse_css_block($style)
++ {
++ $pos = 0;
++
++ // first remove comments
++ while (($pos = strpos($style, '/*', $pos)) !== false) {
++ $end = strpos($style, '*/', $pos+2);
++
++ if ($end === false) {
++ $style = substr($style, 0, $pos);
++ }
++ else {
++ $style = substr_replace($style, '', $pos, $end - $pos + 2);
++ }
++ }
++
++ // Replace new lines with spaces
++ $style = preg_replace('/[\r\n]+/', ' ', $style);
++
++ $style = trim($style);
++ $length = strlen($style);
++ $result = array();
++ $pos = 0;
++
++ while ($pos < $length && ($colon_pos = strpos($style, ':', $pos))) {
++ // Property name
++ $name = strtolower(trim(substr($style, $pos, $colon_pos - $pos)));
++
++ // get the property value
++ $q = $s = false;
++ for ($i = $colon_pos + 1; $i < $length; $i++) {
++ if (($style[$i] == "\"" || $style[$i] == "'") && ($i == 0 || $style[$i-1] != "\\")) {
++ if ($q == $style[$i]) {
++ $q = false;
++ }
++ else if ($q === false) {
++ $q = $style[$i];
++ }
++ }
++ else if ($style[$i] == "(" && !$q && ($i == 0 || $style[$i-1] != "\\")) {
++ $q = "(";
++ }
++ else if ($style[$i] == ")" && $q == "(" && $style[$i-1] != "\\") {
++ $q = false;
++ }
++
++ if ($q === false && (($s = $style[$i] == ';') || $i == $length - 1)) {
++ break;
++ }
++ }
++
++ $value_length = $i - $colon_pos - ($s ? 1 : 0);
++ $value = trim(substr($style, $colon_pos + 1, $value_length));
++
++ if (strlen($name) && !preg_match('/[^a-z-]/', $name) && strlen($value) && $value !== ';') {
++ $result[] = array($name, $value);
++ }
++
++ $pos = $i + 1;
++ }
++
++ return $result;
++ }
++
++ /**
++ * Explode css style value
++ *
++ * @param string $style CSS style
++ *
++ * @return array List of CSS values
++ */
++ public static function explode_css_property_block($style)
++ {
++ $style = preg_replace('/\s+/', ' ', $style);
++ $result = array();
++ $strlen = strlen($style);
++ $q = false;
++
++ // explode value
++ for ($p = $i = 0; $i < $strlen; $i++) {
++ if (($style[$i] == '"' || $style[$i] == "'") && ($i == 0 || $style[$i - 1] != '\\')) {
++ if ($q == $style[$i]) {
++ $q = false;
++ } elseif (!$q) {
++ $q = $style[$i];
++ }
++ }
++
++ if (!$q && $style[$i] == ' ' && ($i == 0 || !preg_match('/[,\(]/', $style[$i - 1]))) {
++ $result[] = substr($style, $p, $i - $p);
++ $p = $i + 1;
++ }
++ }
++
++ $result[] = (string) substr($style, $p);
++
++ return $result;
++ }
++
+ /**
+ * Generate CSS classes from mimetype and filename extension
+ *
+diff --git a/program/lib/Roundcube/rcube_washtml.php b/program/lib/Roundcube/rcube_washtml.php
+index 8a3a5ab..43c9274 100644
+--- a/program/lib/Roundcube/rcube_washtml.php
++++ b/program/lib/Roundcube/rcube_washtml.php
+@@ -249,51 +249,19 @@ class rcube_washtml
+ {
+ $result = array();
+
+- // Remove unwanted white-space characters so regular expressions below work better
+- $style = preg_replace('/[\n\r\s\t]+/', ' ', $style);
+-
+ // Decode insecure character sequences
+ $style = rcube_utils::xss_entity_decode($style);
+
+- foreach (explode(';', $style) as $declaration) {
+- if (preg_match('/^\s*([a-z\\\-]+)\s*:\s*(.*)\s*$/i', $declaration, $match)) {
+- $cssid = $match[1];
+- $str = $match[2];
+- $value = '';
+-
+- foreach ($this->explode_style($str) as $val) {
+- if (preg_match('/^url\(/i', $val)) {
+- if (preg_match('/^url\(\s*[\'"]?([^\'"\)]*)[\'"]?\s*\)/iu', $val, $match)) {
+- if ($url = $this->wash_uri($match[1])) {
+- $value .= ' url(' . htmlspecialchars($url, ENT_QUOTES, $this->config['charset']) . ')';
+- }
+- }
+- }
+- else if (!preg_match('/^(behavior|expression)/i', $val)) {
+- // Set position:fixed to position:absolute for security (#5264)
+- if (!strcasecmp($cssid, 'position') && !strcasecmp($val, 'fixed')) {
+- $val = 'absolute';
+- }
+-
+- // whitelist ?
+- $value .= ' ' . $val;
+-
+- // #1488535: Fix size units, so width:800 would be changed to width:800px
+- if (preg_match('/^(left|right|top|bottom|width|height)/i', $cssid)
+- && preg_match('/^[0-9]+$/', $val)
+- ) {
+- $value .= 'px';
+- }
+- }
+- }
++ // Remove unwanted white-space characters
++ $style = preg_replace('/[\n\r\t]+/', ' ', $style);
+
+- if (isset($value[0])) {
+- $result[] = $cssid . ':' . $value;
+- }
++ $uri_callback = function ($uri) {
++ if ($uri = $this->wash_uri($uri)) {
++ return htmlspecialchars($uri, \ENT_QUOTES, $this->config['charset']);
+ }
+- }
++ };
+
+- return implode('; ', $result);
++ return rtrim(rcube_utils::sanitize_css_block($style, $uri_callback), ';');
+ }
+
+ /**
+@@ -970,53 +938,4 @@ class rcube_washtml
+
+ return $html;
+ }
+-
+- /**
+- * Explode css style value
+- *
+- * @param string $style CSS style
+- *
+- * @return array List of CSS rules
+- */
+- protected function explode_style($style)
+- {
+- $pos = 0;
+-
+- // first remove comments
+- while (($pos = strpos($style, '/*', $pos)) !== false) {
+- $end = strpos($style, '*/', $pos+2);
+-
+- if ($end === false) {
+- $style = substr($style, 0, $pos);
+- }
+- else {
+- $style = substr_replace($style, '', $pos, $end - $pos + 2);
+- }
+- }
+-
+- $style = trim($style);
+- $strlen = strlen($style);
+- $result = array();
+-
+- // explode value
+- for ($p=$i=0; $i < $strlen; $i++) {
+- if (($style[$i] == "\"" || $style[$i] == "'") && $style[$i-1] != "\\") {
+- if ($q == $style[$i]) {
+- $q = false;
+- }
+- else if (!$q) {
+- $q = $style[$i];
+- }
+- }
+-
+- if (!$q && $style[$i] == ' ' && !preg_match('/[,\(]/', $style[$i-1])) {
+- $result[] = substr($style, $p, $i - $p);
+- $p = $i + 1;
+- }
+- }
+-
+- $result[] = (string) substr($style, $p);
+-
+- return $result;
+- }
+ }
+diff --git a/tests/Framework/Utils.php b/tests/Framework/Utils.php
+index b7922e3..4bd9e52 100644
+--- a/tests/Framework/Utils.php
++++ b/tests/Framework/Utils.php
+@@ -182,8 +182,8 @@ class Framework_Utils extends \PHPUnit\Framework\TestCase
+ $mod = rcube_utils::mod_css_styles($css, 'rcmbody');
+
+ $this->assertStringContainsString('#rcmbody table[class=w600]', $mod, 'Replace styles nested in @media block');
+- $this->assertStringContainsString('#rcmbody {width:600px', $mod, 'Replace body selector nested in @media block');
+- $this->assertStringContainsString('#rcmbody {min-width:474px', $mod, 'Replace body selector nested in @media block (#5811)');
++ $this->assertStringContainsString('#rcmbody { width: 600px', $mod, 'Replace body selector nested in @media block');
++ $this->assertStringContainsString('#rcmbody { min-width: 474px', $mod, 'Replace body selector nested in @media block (#5811)');
+ }
+
+ /**
+@@ -191,26 +191,48 @@ class Framework_Utils extends \PHPUnit\Framework\TestCase
+ */
+ function test_mod_css_styles_xss()
+ {
+- $mod = rcube_utils::mod_css_styles("body.main2cols { background-image: url('../images/leftcol.png'); }", 'rcmbody');
+- $this->assertEquals("/* evil! */", $mod, "No url() values allowed");
++ $mod = \rcube_utils::mod_css_styles('font-size: 1em;', 'rcmbody');
++ $this->assertSame('/* invalid! */', $mod);
+
+ $mod = rcube_utils::mod_css_styles("@import url('http://localhost/somestuff/css/master.css');", 'rcmbody');
+- $this->assertEquals("/* evil! */", $mod, "No import statements");
++ $this->assertSame('/* evil! */', $mod);
+
+- $mod = rcube_utils::mod_css_styles("left:expression(document.body.offsetWidth-20)", 'rcmbody');
+- $this->assertEquals("/* evil! */", $mod, "No expression properties");
++ $mod = \rcube_utils::mod_css_styles("@\\69mport url('http://localhost/somestuff/css/master.css');", 'rcmbody');
++ $this->assertSame('/* evil! */', $mod);
+
+- $mod = rcube_utils::mod_css_styles("left:exp/* */ression( alert('xss3') )", 'rcmbody');
+- $this->assertEquals("/* evil! */", $mod, "Don't allow encoding quirks");
++ $mod = \rcube_utils::mod_css_styles("@\\5C 69mport url('http://localhost/somestuff/css/master.css'); a { color: red; }", '');
++ $this->assertSame('/* evil! */', $mod);
+
+- $mod = rcube_utils::mod_css_styles("background:\\0075\\0072\\00006c( javascript:alert('xss') )", 'rcmbody');
+- $this->assertEquals("/* evil! */", $mod, "Don't allow encoding quirks (2)");
++ $mod = \rcube_utils::mod_css_styles("body.main2cols { background-image: url('../images/leftcol.png'); }", 'rcmbody');
++ $this->assertSame('#rcmbody.main2cols {}', $mod);
+
+- $mod = rcube_utils::mod_css_styles("background: \\75 \\72 \\6C ('/images/img.png')", 'rcmbody');
+- $this->assertEquals("/* evil! */", $mod, "Don't allow encoding quirks (3)");
++ $mod = \rcube_utils::mod_css_styles('p { left:expression(document.body.offsetWidth-20); }', 'rcmbody');
++ $this->assertSame('#rcmbody p {}', $mod);
+
+- $mod = rcube_utils::mod_css_styles("background: u\\r\\l('/images/img.png')", 'rcmbody');
+- $this->assertEquals("/* evil! */", $mod, "Don't allow encoding quirks (4)");
++ $mod = \rcube_utils::mod_css_styles('p { left:exp/* */ression( alert('xss3') ); }', 'rcmbody');
++ $this->assertSame('#rcmbody p {}', $mod);
++
++ $mod = \rcube_utils::mod_css_styles("p { background:\\0075\\0072\\00006c('//evil.com/test'); }", 'rcmbody');
++ $this->assertSame('#rcmbody p {}', $mod);
++
++ $mod = \rcube_utils::mod_css_styles("p { background: \\75 \\72 \\6C ('/images/img.png'); }", 'rcmbody');
++ $this->assertSame('#rcmbody p {}', $mod);
++
++ $mod = \rcube_utils::mod_css_styles("p { background: u\\r\\l('/images/img2.png'); }", 'rcmbody');
++ $this->assertSame('#rcmbody p {}', $mod);
++
++ $mod = \rcube_utils::mod_css_styles("p { background: url('//żą.ść?data:image&leak') }", 'rcmbody');
++ $this->assertSame('#rcmbody p {}', $mod);
++
++ $mod = \rcube_utils::mod_css_styles("p { background: url('//data:image&leak'); }", 'rcmbody');
++ $this->assertSame('#rcmbody p {}', $mod);
++
++ // Note: This looks to me like a bug in browsers, for now we don't allow image-set at all
++ $mod = \rcube_utils::mod_css_styles("p { background: image-set('//evil.com/img.png' 1x); }", 'rcmbody');
++ $this->assertSame('#rcmbody p {}', $mod);
++
++ $mod = \rcube_utils::mod_css_styles('p { background: none !important; }', 'rcmbody');
++ $this->assertSame('#rcmbody p { background: none !important; }', $mod);
+
+ // position: fixed (#5264)
+ $mod = rcube_utils::mod_css_styles(".test { position: fixed; }", 'rcmbody');
+@@ -241,19 +263,20 @@ class Framework_Utils extends \PHPUnit\Framework\TestCase
+ // XSS issue, HTML in 'content' property
+ $style = "body { content: '</style><img src onerror=\"alert(\'hello\');\">'; color: red; }";
+ $mod = rcube_utils::mod_css_styles($style, 'rcmbody', true);
+- $this->assertSame("#rcmbody { content: '';\n color: red;\n }", $mod);
++ $this->assertSame("#rcmbody { content: ''; color: red; }", $mod);
+
+ $style = "body { content: '< page: ;/style>< page: ;img src onerror=\"alert(\'hello\');\">'; color: red; }";
+ $mod = rcube_utils::mod_css_styles($style, 'rcmbody', true);
+- $this->assertSame(
+- "#rcmbody { content: '< page: unset;/style>< page: unset;img src onerror=\"alert('hello');\">'; color: red; }",
+- str_replace("\n", '', $mod)
+- );
++ $this->assertSame("#rcmbody { content: '< page: ;/style>< page: ;img src onerror=\"alert('hello');\">'; color: red; }", $mod);
+
+ // Removing page: property
+- $style = "body { page: test; color: red; }";
++ $style = "body { page: test; color: red }";
+ $mod = rcube_utils::mod_css_styles($style, 'rcmbody', true);
+- $this->assertSame("#rcmbody { page: unset;\n color: red;\n }", $mod);
++ $this->assertSame("#rcmbody { color: red; }", $mod);
++
++ $style = "body { background:url(alert('URL!')); }";
++ $mod = rcube_utils::mod_css_styles($style, 'rcmbody', true);
++ $this->assertSame("#rcmbody {}", $mod);
+ }
+
+ /**
+@@ -308,6 +331,85 @@ class Framework_Utils extends \PHPUnit\Framework\TestCase
+ $this->assertStringContainsString('#foo', $mod, "Strip HTML comments from content, but not the content");
+ }
+
++ /**
++ * Test-Cases for parse_css_block() test
++ */
++ function data_parse_css_block()
++ {
++ return [
++ [
++ 'test:test2',
++ [['test', 'test2']],
++ ],
++ [
++ 'Test :teSt2 ;',
++ [['test', 'teSt2']],
++ ],
++ [
++ 'test : test2 test3;',
++ [['test', 'test2 test3']],
++ ],
++ [
++ '/* : */ test : val /* ; */ ;',
++ [['test', 'val']],
++ ],
++ [
++ '/* test : val */ ;',
++ [],
++ ],
++ [
++ 'test :"test1\\"test2" ;',
++ [['test', '"test1\\"test2"']],
++ ],
++ [
++ "test : 'test5 \\'test6';",
++ [['test', "'test5 \\'test6'"]],
++ ],
++ [
++ "test: test8\ntest6;",
++ [['test', 'test8 test6']],
++ ],
++ [
++ "PRop: val1; prop-two: 'val2: '",
++ [['prop', 'val1'], ['prop-two', "'val2: '"]],
++ ],
++ [
++ "prop: val1; prop-two :",
++ [['prop', 'val1']],
++ ],
++ [
++ "prop: val1; prop-two :;",
++ [['prop', 'val1']],
++ ],
++ [
++ "background: url()",
++ [['background', 'url()']],
++ ],
++ [
++ "background:url('')",
++ [['background', "url('')"]],
++ ],
++ [
++ "background: url(\"\")",
++ [['background', 'url("")']],
++ ],
++ [
++ 'font-family:"新細明體","serif";color:red',
++ [['font-family', '"新細明體","serif"'], ['color', 'red']]
++ ],
++ ];
++ }
++
++ /**
++ * Test parse_css_block()
++ *
++ * @dataProvider data_parse_css_block
++ */
++ function test_explode_style($input, $output)
++ {
++ $this->assertSame($output, rcube_utils::parse_css_block($input));
++ }
++
+ /**
+ * Check rcube_utils::explode_quoted_string()
+ */
+diff --git a/tests/Framework/Washtml.php b/tests/Framework/Washtml.php
+index 6848eb2..afddedf 100644
+--- a/tests/Framework/Washtml.php
++++ b/tests/Framework/Washtml.php
+@@ -252,7 +252,7 @@ class Framework_Washtml extends \PHPUnit\Framework\TestCase
+ $washer = new rcube_washtml;
+ $washed = $washer->wash($html);
+
+- $this->assertTrue(strpos($washed, $expected) !== false, "White-space and new-line characters handling");
++ $this->assertSame($this->cleanupResult($washed), $expected, 'White-space and new-line characters handling');
+ }
+
+ /**
diff -Nru roundcube-1.4.15+dfsg.1/debian/patches/Fix-infinite-loop-when-parsing-malformed-Sieve-script.patch roundcube-1.4.15+dfsg.1/debian/patches/Fix-infinite-loop-when-parsing-malformed-Sieve-script.patch
--- roundcube-1.4.15+dfsg.1/debian/patches/Fix-infinite-loop-when-parsing-malformed-Sieve-script.patch 1970-01-01 01:00:00.000000000 +0100
+++ roundcube-1.4.15+dfsg.1/debian/patches/Fix-infinite-loop-when-parsing-malformed-Sieve-script.patch 2024-08-08 23:48:56.000000000 +0200
@@ -0,0 +1,52 @@
+From: Aleksander Machniak <alec at alec.pl>
+Date: Wed, 31 Jul 2024 18:11:31 +0200
+Subject: Fix infinite loop when parsing malformed Sieve script
+
+Origin: https://github.com/roundcube/roundcubemail/commit/3567090a997e95aac6bb052bfb48bb301d0c03c3
+Bug: https://github.com/roundcube/roundcubemail/issues/9562
+---
+ plugins/managesieve/lib/Roundcube/rcube_sieve_script.php | 7 ++++---
+ plugins/managesieve/tests/Parser.php | 14 ++++++++++++++
+ 2 files changed, 18 insertions(+), 3 deletions(-)
+
+diff --git a/plugins/managesieve/lib/Roundcube/rcube_sieve_script.php b/plugins/managesieve/lib/Roundcube/rcube_sieve_script.php
+index 4b70b5e..46923d3 100644
+--- a/plugins/managesieve/lib/Roundcube/rcube_sieve_script.php
++++ b/plugins/managesieve/lib/Roundcube/rcube_sieve_script.php
+@@ -1305,9 +1305,10 @@ class rcube_sieve_script
+
+ // bracket-comment
+ case '/':
+- if ($str[$position + 1] == '*') {
+- if ($end_pos = strpos($str, '*/', $position + 2)) {
+- $position = $end_pos + 2;
++ $position++;
++ if ($str[$position] == '*') {
++ if ($end_pos = strpos($str, '*/', $position + 1)) {
++ $position = $end_pos + 1;
+ }
+ else {
+ // error
+diff --git a/plugins/managesieve/tests/Parser.php b/plugins/managesieve/tests/Parser.php
+index e7d511d..537bed0 100644
+--- a/plugins/managesieve/tests/Parser.php
++++ b/plugins/managesieve/tests/Parser.php
+@@ -59,4 +59,18 @@ class Parser extends \PHPUnit\Framework\TestCase
+
+ return $result;
+ }
++
++ /**
++ * Sieve script parsing
++ */
++ public function test_parser_bug9562()
++ {
++ // This is an obviously invalid script
++ $input = "vacation :subject \"a\" :from \"b\"\n<a href=\"https://test.org/\">test</a>";
++ $script = new rcube_sieve_script($input);
++ $result = $script->as_text();
++
++ // TODO: The output still is BS, but it at least does not cause an infinite loop
++ $this->assertSame("require [\"vacation\"];\r\nvacation :subject \"a\" :from \"b\" \"a\";\r\n", $result);
++ }
+ }
diff -Nru roundcube-1.4.15+dfsg.1/debian/patches/Fix-regression-where-printing-scaling-rotating-image-atta.patch roundcube-1.4.15+dfsg.1/debian/patches/Fix-regression-where-printing-scaling-rotating-image-atta.patch
--- roundcube-1.4.15+dfsg.1/debian/patches/Fix-regression-where-printing-scaling-rotating-image-atta.patch 1970-01-01 01:00:00.000000000 +0100
+++ roundcube-1.4.15+dfsg.1/debian/patches/Fix-regression-where-printing-scaling-rotating-image-atta.patch 2024-08-08 23:48:56.000000000 +0200
@@ -0,0 +1,71 @@
+From: Aleksander Machniak <alec at alec.pl>
+Date: Thu, 8 Aug 2024 13:54:32 +0200
+Subject: Fix regression where printing/scaling/rotating image attachments was
+ broken
+
+Origin: https://github.com/roundcube/roundcubemail/commit/44cec17e8f1b9a03af75f97a9cb6a77724586c47
+Bug: https://github.com/roundcube/roundcubemail/issues/9571
+Bug-Debian: https://bugs.debian.org/1078456
+---
+ program/js/app.js | 19 +++++++------------
+ program/lib/Roundcube/rcube_output.php | 3 ++-
+ 2 files changed, 9 insertions(+), 13 deletions(-)
+
+diff --git a/program/js/app.js b/program/js/app.js
+index d95bf3c..4aedf1d 100644
+--- a/program/js/app.js
++++ b/program/js/app.js
+@@ -422,13 +422,11 @@ function rcube_webmail()
+ var contents = $(this).contents();
+
+ // do not apply styles to an error page (with no image)
+- if (contents.find('img').length)
+- contents.find('head').append(
+- '<style type="text/css">'
+- + 'img { max-width:100%; max-height:100%; } ' // scale
+- + 'body { display:flex; align-items:center; justify-content:center; height:100%; margin:0; }' // align
+- + '</style>'
+- );
++ if (contents.find('img').length) {
++ contents.find('img').css({ maxWidth: '100%', maxHeight: '100%' });
++ contents.find('body').css({ display: 'flex', alignItems: 'center', justifyContent: 'center', height: '100%', margin: 0 });
++ contents.find('html').css({ height: '100%' });
++ }
+ });
+ }
+ // show printing dialog
+@@ -5725,9 +5723,7 @@ function rcube_webmail()
+ this.apply_image_style = function()
+ {
+ var style = [],
+- head = $(this.gui_objects.messagepartframe).contents().find('head');
+-
+- $('#image-style', head).remove();
++ img = $(this.gui_objects.messagepartframe).contents().find('img');
+
+ $.each({scale: '', rotate: 'deg'}, function(i, v) {
+ var val = ref.image_style[i];
+@@ -5735,8 +5731,7 @@ function rcube_webmail()
+ style.push(i + '(' + val + v + ')');
+ });
+
+- if (style)
+- head.append($('<style id="image-style">').text('img { transform: ' + style.join(' ') + '}'));
++ img.css('transform', style.join(' '));
+ };
+
+ // Update import dialog state
+diff --git a/program/lib/Roundcube/rcube_output.php b/program/lib/Roundcube/rcube_output.php
+index a51b75b..e070c5c 100644
+--- a/program/lib/Roundcube/rcube_output.php
++++ b/program/lib/Roundcube/rcube_output.php
+@@ -286,7 +286,8 @@ abstract class rcube_output
+ }
+
+ // Use strict security policy to make sure no javascript content is executed
+- header("Content-Security-Policy: default-src 'none'");
++ // img-src is needed to be able to print attachment preview page
++ header("Content-Security-Policy: default-src 'none'; img-src 'self'");
+
+ // don't kill the connection if download takes more than 30 sec.
+ if (!array_key_exists('time_limit', $params)) {
diff -Nru roundcube-1.4.15+dfsg.1/debian/patches/series roundcube-1.4.15+dfsg.1/debian/patches/series
--- roundcube-1.4.15+dfsg.1/debian/patches/series 2024-06-17 04:10:38.000000000 +0200
+++ roundcube-1.4.15+dfsg.1/debian/patches/series 2024-08-08 23:48:56.000000000 +0200
@@ -22,3 +22,8 @@
CVE-2023-47272.patch
CVE-2024-37384.patch
CVE-2024-37383.patch
+CVE-2024-42009.patch
+CVE-2024-42008.patch
+Fix-regression-where-printing-scaling-rotating-image-atta.patch
+CVE-2024-42010.patch
+Fix-infinite-loop-when-parsing-malformed-Sieve-script.patch
-------------- next part --------------
A non-text attachment was scrubbed...
Name: signature.asc
Type: application/pgp-signature
Size: 833 bytes
Desc: not available
URL: <http://alioth-lists.debian.net/pipermail/pkg-roundcube-maintainers/attachments/20240812/a14c8ab6/attachment-0001.sig>
More information about the Pkg-roundcube-maintainers
mailing list