webshell/jsp/WebSocket 内存马,一种新型内存马技术.md

123 lines
5.5 KiB
Markdown
Raw Normal View History

兼容性测试
-------
#### 1目前测试通过
Tomcat、Spring、Jetty、WebSphere、WebLogic
Nodejs (无法动态注入,需要修改代码后重启服务)
#### 2还未进行测试
Jboss(WildFly)
#### 3无法使用的场景
1.使用了Nginx等代理未配置Header转发 支持WebSocket
2.使用了CDNCDN供应商未支持WebSocket服务
1.前言
----
WebSocket是一种全双工通信协议即客户端可以向服务端发送请求服务端也可以主动向客户端推送数据。这样的特点使得它在一些实时性要求比较高的场景效果斐然比如微信朋友圈实时通知、在线协同编辑等。主流浏览器以及一些常见服务端通信框架Tomcat、Spring、Jetty、WebSphere、WebLogic等都对WebSocket进行了技术支持。
本文都以Tomcat进行介绍讨论其他框架也可实现WebSocket内存马
2.版本
----
2013年以前还没出JSR356标准Tomcat就对Websocket做了支持自定义API再后来有了JSR356Tomcat立马紧跟潮流废弃自定义的API实现JSR356那一套这就使得在Tomcat7.0.47之后的版本和之前的版本实现方式并不一样,接入方式也改变了。
JSR356 是java制定的websocket编程规范属于Java EE 7 的一部分所以要实现websocket功能并不需要任何第三方依赖。
3.服务端实现方式
---------
#### 1注解方式
```
@ServerEndpoint(value = "/ws/{userId}", encoders = {MessageEncoder.class}, decoders = {MessageDecoder.class}, configurator = MyServerConfigurator.class)
```
Tomcat在启动时会默认通过 WsSci 内的 ServletContainerInitializer 初始化 Listener 和 servlet。然后再扫描 `classpath`下带有 `@ServerEndpoint`注解的类进行 `addEndpoint`加入websocket服务
所以即使 Tomcat 没有扫描到 `@ServerEndpoint`注解的类也会进行Listener和 servlet注册这就是为什么所有Tomcat启动都能在memshell scanner内看到WsFilter
![](https://mmbiz.qpic.cn/sz_mmbiz_png/XB8gUH3cR11gW4QtAaQ9jibMic7dJpzlDyIPlQ1ibuyFSyyCyicibqfFHGibTgorHmo7StKqdRWdDULqZcj8MDegicyNg/640?wx_fmt=png)
#### 2继承抽象类Endpoint方式
继承抽象类 `Endpoint`方式比加注解 `@ServerEndpoint`方式更麻烦,主要是需要自己实现 `MessageHandler``ServerApplicationConfig`。`@ServerEndpoint`的话都是使用默认的,原理上差不多,只是注解更自动化,更简洁
可以用代码更方便的控制 ServerEndpointConfig 内的属性
```
ServerEndpointConfig serverEndpointConfig = ServerEndpointConfig.Builder.create(WebSocketServerEndpoint3.class, "/ws/{userId}").decoders(decoderList).encoders(encoderList).configurator(new MyServerConfigurator()).build();
```
3.websocket内存马实现方法
------------------
之前提到过 Tomcat 在启动时会默认通过 WsSci 内的 ServletContainerInitializer 初始化 Listener 和 servlet。然后再扫描 `classpath`下带有 `@ServerEndpoint`注解的类进行 `addEndpoint`加入websocket服务
那如果在服务启动后我们再 addEndpoint 加入websocket服务行不行呢答案是肯定的而且非常简单只需要三步。创建一个ServerEndpointConfig获取ws ServerContainer加入 ServerEndpointConfig即可
```
ServerEndpointConfig config = ServerEndpointConfig.Builder.create(EndpointInject.class, "/ws").build();
ServerContainer container = (ServerContainer) req.getServletContext().getAttribute(ServerContainer.class.getName());
container.addEndpoint(config);
```
4.效果
----
首先利用i.jsp注入一个websocket服务路径为/x注入后利用ws连接即可执行命令
![](https://mmbiz.qpic.cn/sz_mmbiz_png/XB8gUH3cR11gW4QtAaQ9jibMic7dJpzlDyUjXicQAW5TEAyHuOdJuS52vcXazKeO2iah5twL0KkTEoa0icJib3MVL9gw/640?wx_fmt=png)
且通过memshell scanner查询不到任何异常因为根本就没注册新的 Listener、servlet 或者 Filter
![](https://mmbiz.qpic.cn/sz_mmbiz_png/XB8gUH3cR11gW4QtAaQ9jibMic7dJpzlDyFz5ZbjxcicSa3ll4ic2uRyhaiccVIibC6IOy2fRDGMn5DuuSY8ibK0TK7CA/640?wx_fmt=png)
5.代理
----
WebSocket是一种全双工通信协议它可以用来做代理且速度和普通的TCP代理一样快这也是我研究websocket内存马的原因。
例如有一台不出网主机,有反序列化漏洞。
以前在这种场景下可能会考虑上reGeorg或者利用端口复用来搭建代理。
现在可以利用反序列化漏洞直接注入websocket代理内存马然后直接连上用上全双工通信协议的代理。
注入完内存马以后,使用 Gosthttps://github.com/go-gost/gost 连接代理
```
./gost -L "socks5://:1080" -F "ws://127.0.0.1:8080?path=/proxy"
```
然后连接本地1080端口socks5即可使用代理
6.多功能shell实现
------------
建议在了解 哥斯拉webshell工具 工作原理及代码,及 wsMemShell 原理及代码后,再阅读下面这篇 Freebuf 文章,获得更好的阅读体验。
Freebuf: WebSocket webshell 多功能shell实现\[1\]
![](https://mmbiz.qpic.cn/sz_mmbiz_png/XB8gUH3cR11gW4QtAaQ9jibMic7dJpzlDynQNvLb2icCJpgaglBCXfCHnos1KaibYqzHugibtfm6W1o2AicBKCic8qe6w/640?wx_fmt=png)
![](https://mmbiz.qpic.cn/sz_mmbiz_png/XB8gUH3cR11gW4QtAaQ9jibMic7dJpzlDyLUia3ZzEUysicKv3eraibWyoDyAECzvBGVCjUFm9onwvaYiaNnMt8aDdvw/640?wx_fmt=png)
版权声明
----
完整代码https://github.com/veo/wsMemShell
本文章著作权归作者所有。转载请注明出处https://github.com/veo
#### 引用链接
`[1]` WebSocket webshell 多功能shell实现: _https://www.freebuf.com/articles/web/339702.html_