0x01 背景
Webshell一般是指以服务端动态脚本形式存在的一种网页后门。在入侵检测的过程中,检测Webshell无疑是一大重点。比较常见的检测手法有:
1. 文件内容检测(静态检测)
2. 文件行为检测(动态检测)
3. 网络异常流量分析
4. ……
其中,静态检测是比较简单有效的检测Webshell的手段之一。根据Webshell的文件 特征建立异常模型,并使用大量的Webshell样本对模型进行训练,通过诸如异常函数、关键代码以及文件内容与普通业务代码的相似度等等关键点来进行分 析检测。 然而,如果Webshell脱离了服务端脚本页面形式的存在,基于文件特征的静态检测又将面临怎样的困境?我们不妨一起来看看。
0x02 JavaWeb应用
在Java Web应用中,Servlet是Java语言实现的一个接口,用于编写服务端程序[^1]。Servlet程序代码会预先编译成.class文件,部署在 Java容器中,响应用户各种协议的请求,大多数情况下基于HTTP协议,包括动态生成网页内容等等。但是Servlet由Java代码编写,不能有效地 区分页面的展示和处理逻辑,导致Servlet代码非常混乱,而用Java服务器页面(JSP)的出现,可以让程序员把展现层和数据层很好的区分管理起 来。
JSP作为HttpServlet的扩展,使用HTML的书写格式,在适当的地方加入Java代码片段,从而动态生成页面 内容。JSP在首次被访问时,JSP应用容器(应用服务器中用于管理Java组件的部分)将其转换为Java Servlet代码,并编译成.class字节码文件并执行。而下次该JSP文件被访问时,服务器将直接调用Servlet进行处理,除非JSP文件被修 改。
比如,在Apache Tomcat中,它提供了一个Jasper编译器用以将JSP编译成对应的Servlet。在JSP文件被访问后,在workDir生成对应的servlet源码与编译后的.class字节码文件。
JSP编译生成的.class文件默认存放在$CATALINA_BASE/work下,存放路径也可以通过Server.xml等配置文件中的Host标签的workDir属性进行配置 [^2]。
JSP文件再次被访问时,Tomcat会直接调用已编译好的字节码文件。当文件被修改,Tomcat会重新解析JSP文件,生成Servlet代码并编译执行。当文件被删除时,Tomcat返回404 Not Found。
而在在配置文件$CATALINA_BASE/conf/web.xml中,当Jasper运行在开发模式下时,我们可以配置modificationTestInterval参数,控制Tomcat在一定时间之内不检查JSP文件的修改状态[^3]。
设想,如果可以关闭Java容器对JSP文件修改状态的检查,是否可以将恶意代码存放在JSP编译后的.class字节码中,并通过JSP形式持久访问?
0x03 Resin的一个特性
我们注意到了另一款非常流行且性能优良的企业级应用服务器——Resin。Resin同样提供了Servlet和JSP运行引擎。以看到默认情况下,初次访问JSP后,Resin会在./WEB-INF/work/_jsp目录下生成Servlet源码和编译后的.class字节码文件。
与Apache Tomcat不同的是,Resin生成并编译Servlet之后,可以在JSP文件被删除的情况下,正常提供访问。 查看Resin生成的JSP对应的Servlet源码发现,生成的代码内包含了检查JSP文件修改状态相关方 法:_caucho_isModified()。
我们来看看这部分源码中的关键逻辑:
Servlet启动时,Resin会调用init()方法,结束时会调用destroy()方法[^4]。init()方法中实例化的Depend类用于检查文件修改, 这里调用的Depend构造函数中,第三个参数标志了在JSP文件被删除的情况下的处理逻辑。
public Depend(Path source, long digest, boolean requireSource)
requireSource为True时,如果JSP文件被删除则服务器返回404。默认为false,所以当已编译的JSP文件被删除时,Resin并不会判定该JSP页面被修改,依然会执行对应的字节码。
可以看到,Resin判断一个JSP文件是否修改的逻辑为
当web.xml中配置autoCompile属性为false时,Resin会关闭对JSP文件的自动编译,调用_caucho_setNeverModified()方法,从而不会检查JSP文件修改状态。
web.xml
0x04 Binary JSP Webshell
由于Resin这些特性,我们可以用JSP将Webshell字节码写入对应的JSP编译目标路径下,即可得到一个二进制形式存在的JSP Webshell。Resin自动编译存放的代码目录路径可以自定义配置[^5],默认为WEB-INF/work
目录,如:
如:默认配置下,利用JSP写入二进制字节码Webshell
利用脚本中Webshell的字节码内容可以在本地Resin服务器环境中编译获得,但是由于编译和运行的Resin版本不一致会被判定JSP文件已修 改,从而被重新编译,这不是我们想看到的。如0x03小节中所说,Resin中判断JSP是否修改的逻辑包含在JSP对应的Servlet代码中,于是我 们可以篡改这部分字节码中的逻辑,使得_caucho_isModified()函数永远返回false,JVM指令如下:
测 试效果如下:利用write_binary_shell.jsp文件,将字节码webshell写入对应的目录下,即可通过访问对应的JSP文件来访问 Webshell。 由于篡改了相关的判断逻辑,无论Web是否存在同名JSP文件,Resin依然会优先解析到该字节码Webshell。
0x05 References
1. https://zh.wikipedia.org/wiki/Java_Servlet
2. https://tomcat.apache.org/tomcat-8.0-doc/config/host.html
3. https://tomcat.apache.org/tomcat-8.0-doc/jasper-howto.html
4. http://www.caucho.com/resin-3.1/doc/servlet.xtp
5. http://www.caucho.com/resin-4.0/admin/config-el-ref.xtp#work-dir
VIA@【腾讯安全应急响应中心】
PS:近期项目多,每天都是在车上或者飞机上,后台存的一些好文章都发布完了,所以更新上慢了一些。过几天稳下来会继续发布的。