某报表模板注入
某报表模板注入
这次公布的漏洞说是SQL注入,实际上是一个模板注入。
在产品中存在一个名为TemplateUtils
的类,这个类用来处理表达式相关的内容。
传进来的表达式会进入render
方法进行渲染:
data:image/s3,"s3://crabby-images/37b8c/37b8c09f189d7b85a3d3aa781be663a4e7fbf615" alt="image-20240723224905849"
进去看模板参数渲染使用的方法renderParameter4Tpl
,其中一直在调用同名方法render
,最开始传入了一个新的Calculator
:
data:image/s3,"s3://crabby-images/5ac40/5ac4085307f622a67035d79cbaa78f5e23ea4a13" alt="image-20240723225237830"
中间有一个evalRenderAction
指定了RenderAction
,其他的信息不太重要,往后看:
data:image/s3,"s3://crabby-images/0e145/0e1453df7559ee998721453153c803be5f810b8f" alt="image-20240723232033511"
进到最后一个renderTpl
后看到了解析表达式的位置,通过ParameterProvider.PARAMETERPATTERN
进行解析,然后就会获得表达式中的内容。
data:image/s3,"s3://crabby-images/b4f60/b4f6088bd1b9e34540ba361f099fad3857bd2c0b" alt="image-20240723225516954"
data:image/s3,"s3://crabby-images/510fb/510fb3214726fe3332b40b56c9b25c7bb73586ef" alt="image-20240723225420954"
经过处理后这里就会取出${}
中的内容,然后送到下面用前面传入的RenderAction
进行渲染:
data:image/s3,"s3://crabby-images/317a3/317a317569b76da83dffbfdae72efec0d068ae28" alt="image-20240723230757983"
data:image/s3,"s3://crabby-images/eb371/eb37123d799674614eae02e6e543800c999b95ee" alt="image-20240723230545758"
调用var2.render
自然也就是前面传进来的evalRenderAction
了:
data:image/s3,"s3://crabby-images/080e1/080e11f7cbd6de9e2c01fa8eb7c067cac084013e" alt="image-20240723231732934"
继续往下,会走到evalString
,在这里,会将传入的内容转换为Expression
data:image/s3,"s3://crabby-images/5fabb/5fabbe4e0685b9577a4c45ba2a955a94ed969db7" alt="image-20240723233046321"
然后跟入这个eval
,发现了下一个eval
,这里就是前面的Calculator
下的方法了:
data:image/s3,"s3://crabby-images/bec2c/bec2c66e027c28be435e8ffb33f02f56da778c1e" alt="image-20240723233646180"
接下来会去调用下一个eval
,这里就是前面表达式的eval
,实际调用的是RelationExpression
父类BinaryExpression
的eval
:
data:image/s3,"s3://crabby-images/6714b/6714baf4bb7c701ed748b6ddbe688e129471d249" alt="image-20240724002636625"
然后就是是调用下一个Node
的eval
方法,这里的eval
是使用索引进行检测的,实际是获取表达式=
右边的内容,如果是函数的话就是FunctionCall
:
data:image/s3,"s3://crabby-images/5e7ac/5e7ac06748c641a1faaf62d525e16617dcd4907b" alt="image-20240724002924552"
FunctionCall.eval
方法会去调用resolveMethod
,这也是这个漏洞的关键:
data:image/s3,"s3://crabby-images/23fde/23fdeadf88064de15480bb77f817885b3b8bc3cd" alt="image-20240724003555779"
resolveMethod
最终会获取一个命名空间然后调用getMethod
:
data:image/s3,"s3://crabby-images/56bc7/56bc75aac846e75ac6195a1445303d259172920f" alt="image-20240724003707923"
data:image/s3,"s3://crabby-images/9b96e/9b96e40d63290a54b27951032e280bba2a2bb77b" alt="image-20240724003842295"
data:image/s3,"s3://crabby-images/352c9/352c9408e3de12c12ff8ada9ec59eb15f633e463" alt="image-20240724003943915"
到了函数调用,也就是漏洞发生的地方,存在getMethod
相关的实现类中DefaultNameSpace
类的getMethod
这里通过传入的内容去加载类:
data:image/s3,"s3://crabby-images/9e06e/9e06ea99e114dec98c4ef6a3c1b88a881701b4c6" alt=""
实际会去调用com.fr.function.xxx
,这个包里面有很多函数可以调,返回后去调用evalExpression
:
data:image/s3,"s3://crabby-images/86bfc/86bfc2341f47c1d0fa8127884649fa7bb5b8d1ba" alt="image-20240724005229688"
data:image/s3,"s3://crabby-images/c9e99/c9e9920530edbfb3f8019bb6e01cbb8b8e12ee7b" alt="image-20240724005158851"
网传为SQL
,看下SQL
类,应该不用多说了吧:
data:image/s3,"s3://crabby-images/38881/38881eff1016734d09ddeffb8cb186ab2a3af527" alt="image-20240724005341582"
data:image/s3,"s3://crabby-images/9e627/9e6272fbe21999a516d60ba2392dfe603d53b79a" alt="image-20240724005009251"
至于入口:
其实可玩性还是挺高的,也不一定非要用SQL的形式去做POC来检测。
现已加入豪华检测套餐: