百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT知识 > 正文

JAVA序列化和反序列化及漏洞补救

liuian 2025-03-04 13:06 24 浏览

上周,网络安全人员再一次在黑产面前遭到重挫,Joomla曝高危0Day漏洞,无需用户登陆就能触发。Joomla漏洞在官方发布升级版和补丁之前,已经在各种地下黑色产业链中流传了一段时间,恐怕并且已经有不少网站被黑客拿下。 这个恶意代码的进入点是用户代理字符串,这是每个浏览器都在广而告之的内容:让浏览器知道用户的技术结构从而为站点提供最佳或最合适的版本。很显然这个字符串存储于Joomla数据库中,但并没有被清洁处理以检测恶意代码。攻击者能够通过能够播报虚假用户代理字符串的方式利用特殊app和脚本拉力轻易编制一个定制化字符串并将恶意代码附着,这个安全隐患是在php中session是通过序列化存储的。

想要了解关于PHP相关的远程代码执行漏洞分析,可参考:vBulletin5远程代码执行漏洞分析

JAVA序列化和反序列化是啥?

在现有很多的应用当中,需要对某些对象进行序列化,让它们离开内存空间,入驻物理硬盘,以便可以长期保存,其中最常见的是Web服务器中的Session对象。对象的序列化一般有两种用途:把对象的字节序列永久地保存到硬盘上,通常存放在一个指定文件中;或者在网络上传送对象的字节序列。

而把字节序列恢复为对象的过程称为对象的反序列化。当两个进程在进行远程通信时,彼此可以发送各种类型的数据,而且无论是何种类型的数据,都会以二进制序列的形式在网络上传送。发送方需要把这个Java对象转换为字节序列,才能在网络上传送;接收方则需要把字节序列再恢复为Java对象。

其实,在不同的计算机语言中,数据结构、对象以及二进制串的表示方式并不相同。对于像Java这种完全面向对象的语言,程序员所操作的一切都是对象,来自于类的实例化。

JAVA序列化和反序列化实例

在Java语言中最接近数据结构的概念,就是 POJO(Plain Old Java Object)或者Javabean。小编更熟悉Java语言,还是以此为例说明一下序列化和反序列化的实现。

123456789101112131415161718192021222324252627282930313233343536373839404142434445publicstaticvoidmain(String[]args)throwsException{SerializeObject();//序列化Object对象Objecto=DeserializeObject();//反序列Object对象System.out.println(MessageFormat.format("name={0},age={1}, sex={2}",o.getName(),o.getSex(),o.getAge(),o.getHobby()));}/*** MethodName: SerializeObject* Description: 序列化Object对象* @author Haom* @throws FileNotFoundException* @throws IOException*/privatestaticvoidSerializeObject()throws FileNotFoundException,IOException{Objectobject=newObject();object.setName("haom");object.setSex("Female");object.setAge(18);object.setHobby("Taekwondo");// 对于ObjectOutputStream 对象输出流,将Object对象存储到M盘的object.txt文件中,完成对Object对象的序列化操作 ObjectOutputStream oo=newObjectOutputStream(newFileOutputStream(newFile("M:/object.txt")));oo.writeObject(object);System.out.println("Object Serialization success!");oo.close();}/*** MethodName: DeserializeObject* Description: 反序列Object对象* @author Haom* @throws Exception* @throws IOException*/privatestaticObjectDeserializeObject()throws Exception,IOException{ObjectInputStream ois=newObjectInputStream(newFileInputStream(newFile("M:/object.txt")));Objectobject=(Object)ois.readObject();System.out.println("Object deserialization success!");returnObject;}

以上代码说明:序列化Object成功后在M盘生成了一个object.txt文件,而反序列化Object是读取M盘的Object.txt后生成了一个Object对象。

当然,并不是一个实现了序列化接口的类的所有字段及属性,都是可以序列化的:

  • 如果该类有父类,则分两种情况来考虑:如果该父类已经实现了可序列化接口,则其父类的相应字段及属性的处理和该类相同;如果该类的父类没有实现可序列化接口,则该类的父类所有的字段属性将不会序列化,并且反序列化时会调用父类的默认构造函数来初始化父类的属性,而子类却不调用默认构造函数,而是直接从流中恢复属性的值。

  • 如果该类的某个属性标识为static类型的,则该属性不能序列化。

  • 如果该类的某个属性采用transient关键字标识,则该属性不能序列化。

那么,在什么情况下,需要自定义序列化的方式? 先举个简单的例子,如下:

1234567891011121314151617181920212223242526272829303132333435363738394041424344publicclassSeriDemo1implementsSerializable{privateStringname;transient privateStringpassword;// 瞬态,不可序列化状态,该字段的生命周期仅存于调用者的内存中publicSeriDemo1(){}publicSeriDemo1(Stringname,Stringpassword){this.name=name;this.password=password;}//模拟对密码进行加密privateStringchange(Stringpassword){returnpassword+"minna";}//写入privatevoidwriteObject(ObjectOutputStream outStream)throwsIOException{outStream.defaultWriteObject();outStream.writeObject(change(password));}//读取privatevoidreadObject(ObjectInputStream inStream)throws IOException,ClassNotFoundException{inStream.defaultReadObject();StringstrPassowrd=(String)inStream.readObject();//模拟对密码解密password=strPassowrd.substring(0,strPassowrd.length()-5);}//返回一个“以文本方式表示”此对象的字符串publicStringtoString(){return"SeriDemo1 [name="+name+", password="+password+"]";}//静态的mainpublicstaticvoidmain(String[]args)throwsException{SeriDemo1 demo=newSeriDemo1("haom","0123");ByteArrayOutputStream buf=newByteArrayOutputStream();ObjectOutputStream out=newObjectOutputStream(buf);out.writeObject(demo);ObjectInputStream in=newObjectInputStream(newByteArrayInputStream(buf.toByteArray()));demo=(SeriDemo1)in.readObject();System.out.println(demo);}}

如上代码说明,可以得知,以下情况需要自定义序列化的方式:

  • 为了确保序列化的安全性,可以对于一些敏感信息加密;

  • 确保对象的成员变量符合正确的约束条件;

  • 确保需要优化序列化的性能。

在序列化选型的过程中,安全性的考虑往往发生在跨局域网访问的场景。当通讯发生在公司之间或者跨机房的时候,出于安全的考虑,对于跨局域网的访问往往被限制为基于HTTP/HTTPS的80和443端口。如果使用的序列化协议没有兼容而成熟的HTTP传输层框架支持,可能会导致以下几种结果:

  • 因为访问限制而降低服务可用性。

  • 被迫重新实现安全协议而导致实施成本升高。

  • 开放更多的防火墙端口和协议访问,但是是以牺牲安全性为前提。

反序列化漏洞危害

当应用代码从用户接受序列化数据,并试图反序列化改数据进行下一步处理时,会产生反序列化漏洞,其中最有危害性的就是远程代码注入。

这种漏洞产生原因是,java类ObjectInputStream在执行反序列化时,并不会对自身的输入进行检查,这就说明恶意攻击者可能也可以构建特定的输入,在 ObjectInputStream类反序列化之后会产生非正常结果,利用这一方法就可以实现远程执行任意代码。

这个漏洞的严重风险在于,即使你的代码里没有使用到Apache Commons Collections里的类,只要Java应用的Classpath里有Apache Commons Collections的jar包,都可以远程代码执行。

漏洞的根本问题其实并不是Java序列化的问题,而是Apache Commons Collections允许链式的任意的类函数反射调用。攻击者通过允许Java序列化协议的端口,把攻击代码上传到服务器上,再由Apache Commons Collections里的TransformedMap来执行。

反序列化漏洞补救

现在,Apache Commons Collections在 3.2.2版本中做了一定的安全处理,对这些不安全的Java类的序列化支持增加了开关,默认为关闭状态。

涉及的类包括:CloneTransformer,ForClosure, InstantiateFactory, InstantiateTransformer, InvokerTransformer, PrototypeCloneFactory,
PrototypeSerializationFactory, WhileClosure。

RedHat发布JBoss相关产品的解决方案:
https://access.redhat.com/solutions/2045023。

严格意义说起来,Java相对来说安全性问题比较少,出现的一些问题大部分是利用反射,最终用Runtime.exec(String cmd)函数来执行外部命令的。如果可以禁止JVM执行外部命令,未知漏洞的危害性会大大降低,可以大大提高JVM的安全性。

比如:

123456789101112131415161718192021222324252627282930313233SecurityManager originalSecurityManager=System.getSecurityManager();if(originalSecurityManager==null){// 创建自己的SecurityManagerSecurityManager sm=newSecurityManager(){privatevoidcheck(Permission perm){// 禁止execif(perm instanceofjava.io.FilePermission){Stringactions=perm.getActions();if(actions!=null&&actions.contains("execute")){thrownewSecurityException("execute denied!");}}// 禁止设置新的SecurityManagerif(perm instanceofjava.lang.RuntimePermission){Stringname=perm.getName();if(name!=null&&name.contains("setSecurityManager")){thrownewSecurityException("System.setSecurityManager denied!");}}}@OverridepublicvoidcheckPermission(Permission perm){check(perm);}@OverridepublicvoidcheckPermission(Permission perm,Objectcontext){check(perm);}};System.setSecurityManager(sm);}

如上所示,只要在Java代码里简单加一段程序,就可以禁止执行外部程序了。

禁止JVM执行外部命令,是一个简单有效的提高JVM安全性的办法。可以考虑在代码安全扫描时,加强对Runtime.exec相关代码的检测。

小结

本文只是初步进行JAVA序列化和反序列化的科普,让大家对此问题及相关补救方式有个直观的印象和简单了解,下一步深入解还请继续关注绿盟技术博客。使用JAVA反序列化增多了数据的种类,但是还需要尽量避免使用反序列化的交互操作,减少风险的增加。目前,绿盟科技蜂巢社区启动应急机制,已经实现远程代码执行漏洞的在线检测。在社区中,大家可以进行网络安全扫描插件的开发及讨论。

相关推荐

搭建一个20人的办公网络(适用于20多人的小型办公网络环境)

楼主有5台机上网,则需要一个8口路由器,组网方法如下:设备:1、8口路由器一台,其中8口为LAN(局域网)端口,一个WAN(广域网)端口,价格100--400元2、网线N米,这个你自己会看了:)...

笔记本电脑各种参数介绍(笔记本电脑各项参数新手普及知识)

1、CPU:这个主要取决于频率和二级缓存,频率越高、二级缓存越大,速度越快,现在的CPU有三级缓存、四级缓存等,都影响相应速度。2、内存:内存的存取速度取决于接口、颗粒数量多少与储存大小,一般来说,内...

汉字上面带拼音输入法下载(字上面带拼音的输入法是哪个)

使用手机上的拼音输入法打成汉字的方法如下:1.打开手机上的拼音输入法,在输入框中输入汉字的拼音,例如“nihao”。2.根据输入法提示的候选词,选择正确的汉字。例如,如果输入“nihao”,输...

xpsp3安装版系统下载(windowsxpsp3安装教程)

xpsp3纯净版在采用微软封装部署技术的基础上,结合作者的实际工作经验,融合了许多实用的功能。它通过一键分区、一键装系统、自动装驱动、一键设定分辨率,一键填IP,一键Ghost备份(恢复)等一系列...

没有备份的手机数据怎么恢复

手机没有备份恢复数据方法如下1、使用数据线将手机与电脑连接好,在“我的电脑”中可以看到手机的盘符。  2、将手机开启USB调试模式。在手机设置中找到开发者选项,然后点击“开启USB调试模式”。  3、...

电脑怎么激活windows11专业版

win11专业版激活方法有多种,以下提供两种常用的激活方式:方法一:使用激活密钥激活。在win11桌面上右键点击“此电脑”,选择“属性”选项。进入属性页面后,点击“更改产品密钥或升级windows”。...

华为手机助手下载官网(华为手机助手app下载专区)

华为手机助手策略调整,已不支持从应用市场下载手机助手,目前华为手机助手是需要在电脑上下载或更新手机助手到最新版本,https://consumer.huawei.com/cn/support/his...

光纤线断了怎么接(宽带光纤线断了怎么接)

宽带光纤线断了可以重接,具体操作方法如下:1、光纤连接的时候要根据束管内,同色相连,同芯相连,按顺序进行连接,由大到小。一般有三种连接方法,分别是熔接、活动连接和机械连接。2、连接的时候要开剥光缆,抛...

深度操作系统安装教程(深度操作系统安装教程图解)
  • 深度操作系统安装教程(深度操作系统安装教程图解)
  • 深度操作系统安装教程(深度操作系统安装教程图解)
  • 深度操作系统安装教程(深度操作系统安装教程图解)
  • 深度操作系统安装教程(深度操作系统安装教程图解)
win7旗舰版和专业版区别(win7旗舰版跟专业版)

1、功能区别:Win7旗舰版比专业版多了三个功能,分别是Bitlocker、BitlockerToGo和多语言界面; 2、用途区别:旗舰版的功能是所有版本中最全最强大的,占用的系统资源,...

万能连接钥匙(万能wifi连接钥匙下载)

1、首先打开wifi万能钥匙软件,若手机没有开启WLAN,就根据软件提示打开WLAN开关;2、打开WLAN开关后,会显示附近的WiFi,如果知道密码,可点击相应WiFi后点击‘输入密码’连接;3、若不...

雨林木风音乐叫什么(雨林木风是啥)

雨林木风的创始人是陈年鑫先生。陈年鑫先生于1999年创立了雨林木风公司,其初衷是为满足中国市场对高品质、高性能电脑的需求。在陈年鑫先生的领导下,雨林木风以技术创新、产品质量和客户服务为核心价值,不断推...

aics6序列号永久序列号(aics6破解序列号)

关于AICS6这个版本,虽然是比较久远的版本,但是在功能上也是十分全面和强大的,作为一名平面设计师的话,AICS6的现有的功能已经能够应付几乎所有的设计工作了……到底AICC2019的功能是不是...

win7正在启动windows 卡住(win7正在启动windows卡住了 进入安全模式)
  • win7正在启动windows 卡住(win7正在启动windows卡住了 进入安全模式)
  • win7正在启动windows 卡住(win7正在启动windows卡住了 进入安全模式)
  • win7正在启动windows 卡住(win7正在启动windows卡住了 进入安全模式)
  • win7正在启动windows 卡住(win7正在启动windows卡住了 进入安全模式)
手机可以装电脑系统吗(手机可以装电脑系统吗怎么装)

答题公式1:手机可以通过数据线或无线连接的方式给电脑装系统。手机安装系统需要一定的技巧和软件支持,一般需要通过数据线或无线连接的方式与电脑连接,并下载相应的软件和系统文件进行安装。对于大部分手机用户来...