Skip to content

yankwan/shiro-bug

Repository files navigation

shiro 1.2.4 反序列化漏洞

漏洞说明

1. 环境搭建

环境搭建参考: Shiro RememberMe 1.2.4 反序列化导致的命令执行漏洞

shiro 示例工程位于samples目录下的web目录, 需要对其pom文件进行如下修改

<!-- 修改为1.2版本 -->
<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>

2. 调试入口

部署运行成功后,在登录界面勾选Remember Me选项进行登录,在AbstractRememberMeManager#onSuccessfulLogin(...)方法打断点开始调试。

可以跟踪到如下代码:

protected byte[] convertPrincipalsToBytes(PrincipalCollection principals) {
    // 序列化
    byte[] bytes = serialize(principals);
    // AES加密
    if (getCipherService() != null) {
        bytes = encrypt(bytes);
    }
    return bytes;
}

其中encrypt加密方法中,getEncryptionCipherKey()用到了AbstractRememberMeManager中硬编码的key: Base64.decode("kPH+bIxk5D2deZiIxcaaaA==")

protected byte[] encrypt(byte[] serialized) {
    byte[] value = serialized;
    CipherService cipherService = getCipherService();
    if (cipherService != null) {
        ByteSource byteSource = cipherService.encrypt(serialized, getEncryptionCipherKey());
        value = byteSource.getBytes();
    }
    return value;
}

基于AES对称加密的特性,知道了AES的key就可以解密对应的密文了。

解密对应入口方法为: AbstractRememberMeManager#getRememberedPrincipals(...)

public PrincipalCollection getRememberedPrincipals(SubjectContext subjectContext) {
    PrincipalCollection principals = null;
    try {
        // 进行Base64解码
        byte[] bytes = getRememberedSerializedIdentity(subjectContext);
        //SHIRO-138 - only call convertBytesToPrincipals if bytes exist:
        // 进行AES解密及反序列化
        if (bytes != null && bytes.length > 0) {
            principals = convertBytesToPrincipals(bytes, subjectContext);
        }
    } catch (RuntimeException re) {
        principals = onRememberedPrincipalFailure(re, subjectContext);
    }

    return principals;
}

3. 利用ysoserial生成反序列化payload

通过java的反序列化漏洞,利用ysoserial工具对序列化后的二进制字节码注入其他操作,如打开当前机器的计算器程序。

这里需要构建出Base64编码和AES加密后的payload(即最后是序列化后的二进制文件),通过ysoserial修改这个payload注入执行打开计算器的command,见poc.py代码。

运行shiro web服务,并执行: python poc.py 即可弹出计算器。

4. 版本修复

1.2.4版本中在AbstractRememberMeManager类初始化时,设置了硬编码的key值:

public AbstractRememberMeManager() {
    this.serializer = new DefaultSerializer<PrincipalCollection>();
    this.cipherService = new AesCipherService();
    setCipherKey(DEFAULT_CIPHER_KEY_BYTES);
}

在后续版本中修复了这个问题,如shiro 1.4.1版本中,改为如下方式:

public AbstractRememberMeManager() {
    this.serializer = new DefaultSerializer<PrincipalCollection>();
    AesCipherService cipherService = new AesCipherService();
    this.cipherService = cipherService;
    setCipherKey(cipherService.generateNewKey().getEncoded());
}

这里通过调用cipherService.generateNewKey().getEncoded()每次服务启动的时候都会随机生成key值,并且直接返回encode后字节数组。

这样要伪造payload就变得困难了,因为不知道运行服务器那次生成的key值是多少。

About

No description, website, or topics provided.

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages