原始碼連結
(修改前的原始碼,因為feihong師傅的倉庫已經不在了,這裡使用的是Jeromeyoung師傅的原始碼進行改造)https://github.com/Jeromeyoung/JNDIExploit-1
0x00 JNDI注入工具程式碼結構分析
controllers模組:負責LDAP請求的處理
data:image/s3,"s3://crabby-images/8d52c/8d52c9c01099b0a816480afe75a0abd66faa661a" alt=""
enum模組:負責儲存各種模板型別名稱,如:反序列化的Gadget、記憶體馬的型別
data:image/s3,"s3://crabby-images/4cc2b/4cc2b954884f3026e7787c46c50c6e2971075792" alt=""
異常模組:負責處理可能丟擲的異常
data:image/s3,"s3://crabby-images/12bf8/12bf8a7f6186024ccc21bfaf013b2137888f9d9b" alt=""
反序列化模組:各種反序列化鏈的處理
data:image/s3,"s3://crabby-images/96184/96184459a51cd99335c20f86a3b021494fdc8a8a" alt=""
模板模組:命令執行、回顯、DNS、記憶體馬模板
data:image/s3,"s3://crabby-images/b95ab/b95abe1e9db692714f114edf97e9e7b5ca34dac9" alt=""
工具類模組:主要是為了方便而編寫的一些工具類
data:image/s3,"s3://crabby-images/d50cd/d50cd594219959fe38beb8943a8a39e9d1bbc534" alt=""
協議服務及啟動模組:負責LDAP、HTTP協議的具體實現
data:image/s3,"s3://crabby-images/29b6d/29b6d6236aa65dce07f63f4c46c0de25913b2d68" alt=""
下面先對啟動及協議服務模組進行說明
data:image/s3,"s3://crabby-images/ceab8/ceab8fa6fc64eafc1124838f21d910b10623c15c" alt=""
com.feihong.ldap.Starter類為整個工具的啟動入口,從命令列接收引數傳參到com.feihong.ldap.utils.Config類
data:image/s3,"s3://crabby-images/e7199/e7199357a97ef4e78b496eff76f23f7f8f525074" alt=""
Config類中的@Parameter註解為引數的說明及代表的意思,類似於py中的argparse,命令接收到引數,分別賦值給Config類的幾個私有變數。
然後在之後的HTTPServer與LDAPServer中得到應用
data:image/s3,"s3://crabby-images/36252/362528105d5a58bf1e5cd30decf201ffadc51bb2" alt=""
關於LDAP服務端的編寫和Demo可以參考以下連結:https://www.freebuf.com/vuls/253545.html,HTTPServer則是正常的Java Demo編寫即可
data:image/s3,"s3://crabby-images/64023/64023bc122c1d7d26948fc17bab5a01b516706bd" alt=""
第一部分註解的應用
在JNDIEXP中,作者為了可以反射一種型別的類,Controller類,這裡的類起到了尋找特定類的作用,而且每個註解類都給定了uri屬性data:image/s3,"s3://crabby-images/3c8b2/3c8b28bae5f1693af9ec217b9c0281789d811a5a" alt=""
在LDAPServer.start()之前,會先透過new Reflections(this.getClass().getPackage().getName())的方法獲取到com.feihong.ldap包下面所有LdapMapping的類,之後將其以鍵值對(TreeMap<String, LdapController>)放入到Map中,以便於後續呼叫data:image/s3,"s3://crabby-images/0dd40/0dd405dfc72961969894b61344eabc8ddfb5a061" alt=""
比如com.feihong.ldap.controllers.BasicController類,在Map中的儲存格式就是(basic=>Object BasicController)data:image/s3,"s3://crabby-images/51c92/51c92969bec802cc502a8a3a08636f07e55e7d85" alt=""
根據傳送來的LDAP請求去決定呼叫哪個類,具體透過com.feihong.ldap.processSearchResultdata:image/s3,"s3://crabby-images/eac0e/eac0e465f0eba014446d3b04e077f70ccff7abb7" alt=""
根據工具執行的實際效果,如果我們的LDAP請求為ldap://192.168.85.1:1389/Basic/123data:image/s3,"s3://crabby-images/a486b/a486b2dd2f5335c168327b42bf8cbf907c152750" alt=""
那麼DN為Basic/123,首先根據DN中的開頭字串決定是哪個Controller來處理當前的LDAP的請求
第二部分服務端動態呼叫類
在LDAP呼叫LdapController介面實現類之後(以BasicController為例),會先呼叫process方法,以/為標誌分割,獲取到相應的模組名稱,比如ldap://0.0.0.0:1389/Basic/Command/whoami,那麼第一部分Basic用來指定是BasicController,Command指定執行BasicController下的命令執行模組
data:image/s3,"s3://crabby-images/cb777/cb777d5e67a84036313dac47605f851ebd564b8f" alt=""
data:image/s3,"s3://crabby-images/c3dfb/c3dfb7afb68d12e9dc4fe5de4ccaaca21ad83cc5" alt=""
如果是命令執行模組
data:image/s3,"s3://crabby-images/6c223/6c2235d18a489fb31e5642123b5a85fd2c01be03" alt=""
透過com.feihong.ldap.utils.Util的getCmdFromBase方法獲取到執行命令的內容,如果是base編碼的,進行base64解碼後返回內容data:image/s3,"s3://crabby-images/02792/027924a06266c66ed3b8378def6482cec1f4b96c" alt=""
將其賦值給params,之後再呼叫BasicController的sendResult方法,如果是command模組,初始化該模組,透過asm碼的方法,這裡為了避免出現類名重複的情況,使用隨機字元命令類名data:image/s3,"s3://crabby-images/21257/2125711c2447e19b211d33bc6d90c3eaf9a77032" alt=""
data:image/s3,"s3://crabby-images/17ca1/17ca1e744f4d94bab4f91d08d01f7a464a4d37e9" alt=""
data:image/s3,"s3://crabby-images/d45a7/d45a71872778e0825c4d52a29091893584da1a4b" alt=""
data:image/s3,"s3://crabby-images/ac390/ac390055e8be4232f2100231038e6422295affb1" alt=""
data:image/s3,"s3://crabby-images/c6ac6/c6ac684b356f1247133abc95cd8e7c856925ccf2" alt=""
之後命令執行模組呼叫cache方法
data:image/s3,"s3://crabby-images/b3882/b38825f8fd2a47fad4507201b51525499dcd03ff" alt=""
data:image/s3,"s3://crabby-images/91c4a/91c4a867791d17e71bdd55b4d7d60d90da82f33b" alt=""
data:image/s3,"s3://crabby-images/56de4/56de45db4177e1d4799f6a1b55d108b20ab67838" alt=""
將其儲存在map中,之後開始進行LDAP的步驟data:image/s3,"s3://crabby-images/58f0a/58f0a41f78a383082daa4d895ec0d35286db7a8d" alt=""
首先LDAP設定好
javaClassName:記錄序列化物件的類名,這樣應用程式就可以確定類資訊,而不必首先反序列化
javaClassNames:關於序列化物件的附加類資訊。
javaCodebase:例項化工廠所需的類定義的位置(HTTP地址)
javaFactory:用於儲存物件工廠的完全限定類的可選屬性(即類名)
一張圖說明LDAP請求的過程
data:image/s3,"s3://crabby-images/0d80e/0d80e92eef9a24ab7683c07a538869acce519aa9" alt="wKg0C2I4Qs6AAM4QAABoz1qOh2I020.png"
接下來看HTTPServer如何接收HTTP請求並返回data:image/s3,"s3://crabby-images/88be7/88be746e75f8aaabc2c56cc815ed468f53225600" alt=""
當HTTPServer接收到請求之後,將Cache類中map儲存的類寫入到響應中,這樣避免了class檔案落地的情況。
0x01 去除server console內容
在這個JNDI的注入工具中,會在注入記憶體馬的時候,使用系統輸出語句,在控制檯列印出東西,無關緊要的東西,直接去掉就好。
data:image/s3,"s3://crabby-images/730bc/730bcf63787c0c939a0f257c06f6ebf199864595" alt=""
0x02 冰蠍3.11記憶體馬注入
改造前的filter類,獲取session是透過ServletRequest的方式獲取的
data:image/s3,"s3://crabby-images/1b652/1b652fc81b5edbc11d303448e5627fd298245b99" alt=""
改造之後的filter類,不再使用自寫的classLoader而是直接使用URLClassLoader,同理對其他的元件也進行類似的改造,如:spring的Interceptor,weblogic和jboss的filter
data:image/s3,"s3://crabby-images/bd32c/bd32c3ef846e382671630bf6f17ea56a7cf87021" alt=""
所有回顯使用的是header頭的WWW-Authenticate欄位,而非cmd
data:image/s3,"s3://crabby-images/37b26/37b2637d47c21a4d3d8bcf1b72494f14f7ca3d85" alt=""
0x03 JDK版本匹配及tomcat版本相容解決
利用JDK向下相容的特性,這裡使用JDK6編譯filter class。
tomcat版本相容,根據先知不久之前提出的,tomcat6-9的通用StandardContext獲取方法,獲取到StandardContext,接下來的問題就是解決tomcat版本識別與不同版本間filter注入的問題。
tomcat8以下版本FilterDef、FilterMap都在org.apache.catalina.deploy包裡,tomcat8以上包括8都在org.apache.tomcat.util.descriptor.web裡。
2.識別版本之後,進行不同版本間的filter注入
實現邏輯大概如下:
對各個版本的filter記憶體馬全部轉為位元組碼Base64的編碼格式,透過loadClass的辦法,根據版本注入到記憶體中。(class一律使用JDK1.6編譯)
為了避免相關類載入不到的問題,這裡統一使用反射寫所有的filter注入步驟。
tomcat6789的StandardContext先知上已經給出瞭解決方案:
https://xz.aliyun.com/t/9914
對tomcat特定類進行篩選,來決定FilterDef和FilterMap類載入的使用,最後形成的Demo如下:
data:image/s3,"s3://crabby-images/fba2d/fba2dff5c828d88de86ebd085a84dfe2d5ff847d" alt=""
data:image/s3,"s3://crabby-images/490e5/490e5f0c97cdbf260496368b803c1f59ef787065" alt=""
data:image/s3,"s3://crabby-images/330db/330db9465c4a8f6eb7c6b3d80621f80cfa8f9aa5" alt=""
data:image/s3,"s3://crabby-images/32f1d/32f1dfb8149b772ee631aa222915f2b4e8d82077" alt=""
getStand的原始碼如下:
data:image/s3,"s3://crabby-images/eacfc/eacfc659689351ad743f7a66ec65c719dba5ccd8" alt=""
data:image/s3,"s3://crabby-images/afe3d/afe3dcce9c9aeb5348711eda67c5e70bc9c22be2" alt=""
data:image/s3,"s3://crabby-images/e7253/e725366b269bf25ade9f7c02d6322acea3b36ce9" alt=""
經測試,可用於常規的tomcat6、7(某凌)、8、9版本。
當所有的tomcat版本都測試完畢之後,發現這個通用方法對某遠並不適用,原因是致遠這裡的ServerPort會永遠等於-1。
解決方案是使用原版的feihong-jndi將TomcatMemShell2中的filter類替換為自定義的filter類即可。
0x04 spring memShell的改進
在一些版本的測試中,發現原版的JNDIExploit的spring並不適用於所有的spring框架場景,對此做出補充(借鑑自lz2y與microworld師傅的成果)
以ruoyi cms後臺的snake yaml為例,lz2y師傅這裡已經說了具體的原因(https://xz.aliyun.com/t/10651),直接使用師傅的Demo就好,我這裡使用的是Windows的環境,使用比較在Linux和Windows都通用的Demo獲取的WebApplicationContext
data:image/s3,"s3://crabby-images/662f6/662f6d5683cbc25312f345ab7ea3159421745e27" alt=""
使用com.ruoyi.common.utils.spring.SpringUtils在windows ruoyi4.6.0的環境中沒有成功(POC使用的是com.sun.rowset.JdbcRowSetImpl)
data:image/s3,"s3://crabby-images/d9dde/d9ddec485f91d55e60ceb8e31b07f8e656d414d2" alt=""
使用com.ruoyi.common.utils.spring.SpringUtils在windows ruoyi4.6.0的環境中成功(POC使用的是javax.script.ScriptEngineManager)
剩下的步驟直接注入記憶體馬即可,實現Demo如下:
data:image/s3,"s3://crabby-images/8d26d/8d26d00c518fd91b9d9a7d4f8c30a3ed6a1c1c34" alt=""
data:image/s3,"s3://crabby-images/7b8cb/7b8cb2a530455a253f3ecedf5ad09c35c284a015" alt=""
data:image/s3,"s3://crabby-images/03687/03687dee9e16a1d56bfcb5f3bb78cf571fafe645" alt=""
data:image/s3,"s3://crabby-images/cb9d5/cb9d5d08d77d4731857b754e6603ed753858313b" alt=""
data:image/s3,"s3://crabby-images/71064/710640946b07eecefa0fc723d6615e106a034589" alt=""
將其新增到類的引用處,對於spring的框架新增新增類的引用,需要修改的是四個類:
com.feihong.ldap.enumtypes.PayloadType
data:image/s3,"s3://crabby-images/f7856/f7856b61ffa9a3623d6cdbd78781a9f362fae838" alt=""
com.feihong.ldap.utils.Cache
data:image/s3,"s3://crabby-images/2a29f/2a29f0849e0560e4284b4df7f994b080ff4bb786" alt=""
com.feihong.ldap.controllers.BasicController
data:image/s3,"s3://crabby-images/0ae57/0ae57ede7f08b9c82915938ae421dc3d05c6f6bf" alt=""
com.feihong.ldap.controllers.TomcatBypassController
data:image/s3,"s3://crabby-images/5cc8a/5cc8aed578df1d9e5620de8d5ebff7a91d5619a1" alt=""
修改Cache類是為了能夠讓惡意類能夠在記憶體中載入,具體的原因已經在0.1中分析完畢,修改BasicController和TomcatBypassController是為了能在LDAP在JDK的高低版本中找到引用。
0x05 jetty memShell的改進
對於jetty記憶體馬的改進,主要是log4j對solr的影響,針對於實戰的場景,參考Qu3een師傅的文章(https://tttang.com/archive/1386/)
實現的Demo如下,直接整合到JNDI注入工具中
data:image/s3,"s3://crabby-images/3377a/3377a6a727617c3c7f9d2d0523d4e4a98f1e8386" alt=""
data:image/s3,"s3://crabby-images/3b507/3b507a8c8d1a3ffd2068f385902c5fd898c010c7" alt=""
修改com.feihong.ldap.enumtypes.PayloadType
data:image/s3,"s3://crabby-images/66ecd/66ecd79a7bdb702ce225efe7faa88b8cc381f5b4" alt=""
修改com.feihong.ldap.utils.Cache
data:image/s3,"s3://crabby-images/a770b/a770b69bfc20d19f141acc8741515a9bcf8880db" alt=""
修改com.feihong.ldap.controllers.BasicController
data:image/s3,"s3://crabby-images/6850e/6850ea7fc27301fddb6f937f12f88476dd897ff6" alt=""
0x06 webSphere memShell的改進
暫缺
0x07 增加resin中介軟體下的記憶體馬注入
基於pen4uin師傅的通用研究成果直接整合到JNDI注入工具中,原理不再贅述,參考pen4uin師傅的公眾號即可,直接給出Demo
data:image/s3,"s3://crabby-images/0513b/0513b0a3047944182f4aa28bb3322eefee005637" alt=""
data:image/s3,"s3://crabby-images/7ab50/7ab50003468ce9f119d19fb3a2c634b57c9e6496" alt=""
同樣的,修改com.feihong.ldap.enumtypes.PayloadType
data:image/s3,"s3://crabby-images/dd979/dd9790550873b15cc8574d5e9cda6ad21751263d" alt=""
修改com.feihong.ldap.utils.Cache
data:image/s3,"s3://crabby-images/68c5e/68c5e8a25d8635e13c6e06c5337d7db68b8fe4cb" alt=""
修改com.feihong.ldap.controllers.BasicController
data:image/s3,"s3://crabby-images/28646/286468e4ffc9db12fe5356d964813ca8a9ba5334" alt=""
data:image/s3,"s3://crabby-images/191ca/191ca7b357b3e394f5c7489dbb0a2ffbff31ffdf" alt=""
0x08 增加tomcatValue記憶體馬
在tomcat容器中,有些場景filter記憶體馬並不如valve記憶體馬,所以為了增加相容性,這裡增加valve記憶體馬,實現的Demo大致如下:
data:image/s3,"s3://crabby-images/7a3d0/7a3d0abe4b633e941355e2d1b4b73310ce23f17b" alt=""
data:image/s3,"s3://crabby-images/31ea7/31ea7664801a2b97abddb2d4c6d8aa19be10fa1b" alt=""
data:image/s3,"s3://crabby-images/36b3f/36b3f273ca78965178d526a96080affe27da5c5f" alt=""
data:image/s3,"s3://crabby-images/64add/64addb98d56e99e8a1a310466093d578bd161a2f" alt=""
data:image/s3,"s3://crabby-images/87316/8731620c6e381d5d879a43d6f5f70773f2facd73" alt=""
同樣的修改配置檔案,增加到BasicController、TomcatBypassController、Cache、Config中。
參考連結
https://landgrey.me/blog/19/
https://github.com/Mr-xn/JNDIExploit-1
https://tttang.com/archive/1386/
https://mp.weixin.qq.com/s/5s9eyRpaP7WhVd2estwtKg
https://github.com/lz2y/yaml-payload-for-ruoyi
https://xz.aliyun.com/t/10651