menu arrow_back 湛蓝安全空间 |狂野湛蓝,暴躁每天 chevron_right ... chevron_right 166-Jizhicms chevron_right 004-Jizhicms 1.7.1 从sql注入到任意文件上传.md
  • home 首页
  • brightness_4 暗黑模式
  • cloud
    xLIYhHS7e34ez7Ma
    cloud
    湛蓝安全
    code
    Github
    004-Jizhicms 1.7.1 从sql注入到任意文件上传.md
    4.2 KB / 2021-07-17 00:01:28
        # Jizhicms 1.7.1 从sql注入到任意文件上传
    
    ## 一、漏洞简介
    
    ## 二、漏洞影响
    
    Jizhicms 1.7.1
    
    ## 三、复现过程
    
    从回显可以明确的看到这是一个报错注入,如果没有回显报错的话,为了查看是否进行了sql语句的拼接可以去查看mysql的log日志,可以通过Navicat的日志功能去查看
    
    ![1.png](images/2020_05_26/9dd049da95e04b0788566ecfcaa6bdcb.png)
    
    在对CMS不是很熟悉的情况下可以通过搜索关键字来定位大概的漏洞位置,customurl成功的引起了我的注意,这是表的名字,经过简单判断,定位到函数位置为
    /Home/c/HomeController.php中342-355行中,用户传入参数url然后进入到find函数中处理
    
    ![2.png](images/2020_05_26/40383b1fb1624ef18711685378756a13.png)
    
    跟进到find函数中,位于/FrPHP/lib/Model.php,find函数主要去调用findAll函数去拼接执行sql语句
    
    ```php
    public function find($where=null,$order=null,$fields=null,$limit=1)
        {
           if( $record = $this->findAll($where, $order, $fields, 1) ){
                return array_pop($record);
            }else{
                return FALSE;
            }
        }
    
    ```
    
    这里看代码看的头疼,为了直观展示代码的执行过程可以用phpstorm配合xdebug的方式去调试php代码,可以看到将我们传入的参数直接带入查询,然后调用getArray函数去执行
    
    ![3.png](images/2020_05_26/b625ff9b770b46d89ad870c31e3a71fc.png)
    
    ```php
    public function findAll($conditions=null,$order=null,$fields=null,$limit=null)
        {
            .....
            .....
            if(!empty($limit))$where .= " LIMIT {$limit}";
            $fields = empty($fields) ? "*" : $fields;
            $table = self::$table;
            $sql = "SELECT {$fields} FROM {$table} {$where}";
            return $this->db->getArray($sql);
        }
    
    ```
    
    在 /FrPHP/db/DBholder.php中, getArray函数调用query函数,如果有错误将输出错误信息
    
    ![4.png](images/2020_05_26/9cb63b1166d0430785094b6e6744d8ce.png)
    
    在接下来发现该CMS允许上传的文件类型是保存在数据库中的
    
    ![5.png](images/2020_05_26/7da1a914a9884cec9faf1c9ae2274407.png)
    
    通过数据库写入到缓存文件,在使用时从缓存文件中去看上传的类型是不是缓存文件中允许的,如果是则允许上传。那可以通过SQL注入漏洞更新下数据库,写入允许上传的后缀php,即可实现getshell
    
    ![6.png](images/2020_05_26/b52978fa16534b8b92bccb5882fc2ab5.png)
    
    然后登陆后台清空缓存让网站重新获得新的缓存,然后上传php文件。看到上传成功了
    
    ![7.png](images/2020_05_26/1490d7bb372543d0b4aeb9fb91ea408c.png)
    
    既然是代码审计,我们也来跟下网站获取上传类型的方式
    
    在/A/c/CommonController.php 中uploads函数中是从webconf中获得的fileType的值
    
    ```php
    $fileType = $this->webconf['fileType'];
                if(strpos($fileType,strtolower($pix))===false){
                    $data['error'] =  "Error: 文件类型不允许上传!";
                    $data['code'] = 1002;
                    JsonReturn($data);
                }
    
    ```
    
    webconf函数位于/Conf/Functions.php中,通过调用getCache函数来获取相关的值
    
    ```php
    function webConf($str=null){
        //v1.3 取消文件存储
        //$web_config = include(APP_PATH.'Conf/webconf.php');
        $webconfig = getCache('webconfig');
    }
    
    ```
    
    getCache函数位于/FrPHP/common/Functions.php
    
    ```php
    function getCache($str=false){
        if(!$str){
            return false;
        }
        //获取
        $s = md5($str).'frphp'.md5($str);
        $cache_file_data = APP_PATH.'cache/data/'.$s.'.php';
        if(!file_exists($cache_file_data)){
            return false;
        }
        $last_time = filemtime($cache_file_data);//创建文件时间
        $res = file_get_contents($cache_file_data);
        $res = substr($res,14);
        $data = json_decode($res,true);
    
        if(($data['frcache_time']+$last_time)<time() && $data['frcache_time']>=0){
            unlink($cache_file_data);
            return false;
        }else{
            return $data['frcache_data'];
        }
    }
    
    ```
    
    getCache从/cache/data/中获取相关的值 缓存文件的名字为md5(webconfig).'frphp'.md5(webconfig)
    可以看到已经成功缓存,php为允许上传的类型
    
    ![8.png](images/2020_05_26/1e93ff71d6df478596d3e11e2da1bba2.png)
    
    ## 参考链接
    
    > https://xz.aliyun.com/t/7775#toc-2
    
    
    
    links
    file_download