SpringCloud Function SpEL註入漏洞分析

来源:https://www.cnblogs.com/9eek/archive/2022/04/07/16113603.html
-Advertisement-
Play Games

SpringCloud Function SpEL註入 漏洞分析 ...


SpringCloud Function 介紹

SpringCloud 是一套分散式系統的解決方案,常見的還有阿裡巴巴的Dubbo,Fass(Function As A Service )的底層實現就是函數式編程,在視頻轉碼、音視頻轉換、數據倉庫ETL等與狀態相關度低的領域運用的比較多。開發者無需關註伺服器環境運維等問題上,專註於自身業務邏輯實現即可。

SpringCloud Function 就是Spring提供的分散式函數式編程組件。

image-20220401101253220

漏洞環境搭建

通過idea新建一個Spring項目,pom中引入spring-boot-starter-webspring-cloud-function-web,如下:

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.6.5</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>
    <groupId>com.example</groupId>
    <artifactId>SpringCloudDemo</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <name>SpringCloudDemo</name>
    <description>SpringCloudDemo</description>
    <properties>
        <java.version>1.8</java.version>
        <spring-cloud.version>2021.0.1</spring-cloud.version>
    </properties>
    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-function-web</artifactId>
            <version>3.2.2</version>
        </dependency>


        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
    </dependencies>
    <dependencyManagement>
        <dependencies>
            <dependency>
                <groupId>org.springframework.cloud</groupId>
                <artifactId>spring-cloud-dependencies</artifactId>
                <version>${spring-cloud.version}</version>
                <type>pom</type>
                <scope>import</scope>
            </dependency>
        </dependencies>
    </dependencyManagement>
</project>

image-20220401144838412

其中spring-cloud-function-web的依賴如上圖,核心實現為spring-cloud-function-core包。

先在main函數中新建兩個方法(uppercase將字元串變為大寫,reverse字元串反轉):

image-20220407095021125

當在pom中引入spring-cloud-function-web後,函數會自動添加為HTTP端點。

然後漏洞關鍵是在application.properties 或者yaml配置文件中新增一行:

spring.cloud.function.definition=functionRouter

這裡的屬性spring.cloud.function.definition 表示聲明式函數組合,這個功能允許在提供屬性時使用|(管道),;(過濾)分隔符以聲明的方式提供組合指令。例如

--spring.cloud.function.definition=uppercase|reverse

舉例:

當配置該屬性為uppercase時,訪問根路徑提交的參數會自動被uppercase函數接受轉化為大寫:

image-20220407100743588

image-20220407100648638

反之若配置為reverse則預設路徑函數功能為反轉字元串:

image-20220407100849083

image-20220407100912828

通俗來講這個屬性就是一個預設路由, 可以手動指定相關函數,也可以使用functionRouter ,指定的方式可以是配置文件、環境變數或者啟動參數等

functionRouter

如果設置為functionRouter則預設路由綁定的具體函數交由用戶進行控制,在 Spring Cloud Function Web裡面,可以通過設置http頭的方式來控制,使用spring.cloud.function.definitionspring.cloud.function.routing-expression 都可以,區別是後者允許使用Spring表達式語言(SpEL)

舉例:

image-20220407101221032

image-20220407101243914

image-20220407101308586

image-20220407101414750

因為spring.cloud.function.routing-expression 允許使用SpEL表達式,所以就可能存在SpEL註入

SpEL註入

這裡簡單介紹下SpEL,Spring Expression Language 是Spring提供的具有方法調用和基本的字元串模版功能的套件。類似OGNL、MVEL、JBoss EL。

SpEL可以字元串之間進行嵌套也可以單獨使用,嵌套時使用#{}(實現ParserContext介面)。

舉例:

image-20220407104131771

但因為Spel支持方法調用,所以如果使用的是StandardEvaluationContext 進行解析(預設),則可能會被濫用,如使用new ProcessBuilder('/System/Applications/Calculator.app/Contents/MacOS/Calculator').start()可觸發命令執行:

image-20220407104613567

漏洞復現

既然SpringCloud Function 中的functionRouter支持SpEL那是不是存在SpEL註入呢,我們在HTTP頭中插入上面調起計算器的SpEL表達式

Payload: spring.cloud.function.routing-expression: new ProcessBuilder('/System/Applications/Calculator.app/Contents/MacOS/Calculator').start()

非常簡單粗暴,漏洞復現成功:

image-20220407105317920

原理分析

在命令執行出下斷點,看下程式執行流程。

SpringCloud Function之所以能自動將函數建立http端點,是因為在包mvc.FunctionController中使用/** 監聽了get/post類型的所有端點。

  1. 當一個請求進入時,程式首先基於Springboot的自動配置,將配置文件註入到functionProperties,隨後將以“WebRequestConstants.handler”為key,function為值添加到request數組裡面。

image-20220407164023103

image-20220407164200379

  1. 請求正式進入Controller節點,Controller首先會將請求使用wrapper進行包裝,wrapper就是將request轉成FunctionInvocationWrapper 格式。

image-20220407164733082

  1. 隨後進入processRequest 對request進行處理,執行function的apply方法,跳轉到doApply()時會對function進行判斷,判斷是不是functionRouter方法,根據咱們的配置文件此時的function為RoutingFunction.FUNCTION_NAMEfunctionRouter所以會,一路跳轉到RoutingFunction.route

image-20220407170619009

image-20220407170151488

image-20220407170208535

  1. 隨後進入else if 分支, http頭spring.cloud.function.routing-expression 不為空,則傳入其值到functionFromExpression方法。
    image-20220407170804259

  2. 使用標準的StandardEvaluationContext 對header的值進行SpEL表達式解析:

image-20220407171027665

image-20220407171053490

後續就不用再跟下去了,至此可以發現,只要通過環境變數、配置文件或者參數等方式配置為spring.cloud.function.definition=functionRouter 即可觸發SpEL註入。

補丁分析

SpringCloud官方已經修複了此問題(https://github.com/spring-cloud/spring-cloud-function/commit/0e89ee27b2e76138c16bcba6f4bca906c4f3744f)

和其他SpEL註入修複方式一樣,使用了SimpleEvaluationContext替換StandardEvaluationContext,那這個漏洞基本就算修複完成了。但因為這個commit還沒有納入版本,所以目前springcloud Function3.0以上版本仍然暴露在風險之中。

image-20220407171533592

引用

公眾號

歡迎大家關註我的公眾號,這裡有乾貨滿滿的硬核安全知識,和我一起學起來吧!


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1. 求數值型數組中元素的最大值、最小值、平均值、總值等 2. 數組的複製、反轉、查找(線性查找、二分法查找) ...
  • 非同步任務-springboot 非同步:非同步與同步相對,當一個非同步過程調用發出後,調用者在沒有得到結果之前,就可以繼續執行後續操作。也就是說無論非同步方法執行代碼需要多長時間,跟主線程沒有任何影響,主線程可以繼續向下執行。 實例: 在service中寫一個hello方法,讓它延遲三秒 @Service ...
  • 由於``某些不可抗力原因,公司不允許使用itext系列的jar包,因此系統中使用的相關jar得替換成開源的。經比較和嘗試考慮使用org.apache.pdfbox來替換,同時修改系統中原有的方法,發現比itext系列稍顯簡潔一點,記錄如下: 加密文件 /** * 加密文件測試 * @date 202 ...
  • 背景 在項目使用了Spring Security之後,很多介面無法訪問了,從瀏覽器的網路調試窗看到的是CORS的報錯和403的報錯 分析 我們先來看一下CORS是什麼,和它很相似的CSRF是什麼,在SpringSecurity中如何配置以及起的什麼作用 CORS(Cross Origin Resou ...
  • 本文我們來看下 SpringSecurity + JWT 實現單點登錄操作,本文 2W 字,預計閱讀時間 30 min,文章提供了代碼骨架,建議收藏。 一、什麼是單點登陸 單點登錄(Single Sign On),簡稱為 SSO,是目前比較流行的企業業務整合的解決方案之一。SSO的定義是在多個應用系 ...
  • 一、字元流的由來 由於使用位元組流操控中文時不是很方便,Java就提供了字元流來進行操控中文 實現原理:位元組流+編碼表 為什麼用位元組流進行複製帶有中文的文本文件時沒有問題? 因為底層操作會自動進行位元組拼接成中文 怎樣識別該位元組是中文呢? 漢字在存儲時,無論是UTF-8還是GBK,第一個位元組都是負數用來 ...
  • 《零基礎學Java》 文件輸入/輸出流 程式運行期間,大部分數據都被存儲在記憶體中,當程式結束或被關閉時,存儲在記憶體中的數據將會消失。如果要永久保存數據,那麼最好的辦法就是把數據保存到磁碟的文件中。為此,Java提供了文件輸入/輸出流,即 FilelnputStream類 與 FilcOutputSr ...
  • 之前刷到一個視頻,老師上課點到用系統點到回答問題,然後就點名結束了。相信很多學校現在也會玩這招吧,今天就用Python給大家做一個點名系統。來吧,展示… 一.準備工作 1.Tkinter Tkinter 是 python 內置的 TK GUI 工具集。TK 是 Tcl 語言的原生 GUI 庫。作為 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...