• 【亲测可用】禁用AMQP配置中的明文身份验证机制漏洞解决方法(RabbitMQ开启SSL附SpringBoot连接测试代码)


    楔子

    近期公司程序被安全扫描出 远程主机允许明文身份验证 中风险漏洞,查了下修复方案发现网上的都是把 RabbitMQ 的认证机制改了,然后也没提供 Java 客户端连接测试结果,底下全是登录失败的回帖……

    想到 RabbitMQ 官方提供了SSL连接方式,而且 SpringBoot AMQP 也支持 SSL 连接,所以尝试以下将配置RabbitMQ开启SSL 并使用 SpringBoot Demo 测试连接。最终修复了这个漏洞,同时 Java 客户端连接正常。

    文章修订日志:

    • 2022-01-07 21:59写文章时此配置还未安全扫描复测,如果测试通过,本人将更新此文章状态为验证通过。
    • 2022-01-11 14:36 复测不通过,修正文章,调整认证机制为 EXTERNAL 与 插件认证方式,等待复测。
    • 2022-01-11 17:28 复测通过。

    配置 RabbitMQ 开启 SSL

    本文基于 CentOS 7 + Git + OpenSSL + yum 安装的 RabbitMQ,需要读者提前安装好,其他方式也可变通参考本文。

    生成证书

    #克隆生成证书的仓库到当前目录
    git clone --depth 1 https://github.com/Berico-Technologies/CMF-AMQP-Configuration.git
    cd CMF-AMQP-Configuration/ssl
    #生成ca证书,“MyRabbitMQCA”为自定义名称,名称任意。在当前目录下生成ca目录
    sh setup_ca.sh MyRabbitMQCA
    #生成服务端证书,第一个参数是服务端证书前缀,第二个参数是密码。密码任意,在当前目录下生成server目录
    sh make_server_cert.sh rabbitmq-server 123456
    #生成客户端证书,第一个参数是客户端证书前缀(同时也是rabbitmq用户名),第二个参数是密码。密码任意,在当前目录下生成client目录
    sh create_client_cert.sh rabbitmq-client 654321
    

    以上生成的客户端证书的CN为 rabbitmq-client,此名称会被 RabbitMQ服务端作为登录名使用,需要提前创建此用户以及给予权限。

    配置 RabbitMQ 服务端的证书如下:

    ca/cacert.pem #CA证书
    server/rabbitmq-server.cert.pem #服务端公钥
    server/rabbitmq-server.key.pem  #服务端私钥
    

    使用 RabbitMQ 服务端公钥证书生成 JKS 证书

    # -alias后为别称,-file后是服务端公钥位置,-keystore后是输出JSK证书位置,此处相对路径
    keytool -import -alias rabbitmq-server \
      -file server/rabbitmq-server.cert.pem \
      -keystore rabbitmqTrustStore -storepass changeit
    #输入y回车
    

    配置 RabbitMQ 客户端的证书如下:

    client/rabbitmq-client.keycert.p12 #PKCS12证书,包含客户端所需公私钥及中间证书
    rabbitmqTrustStore #服务端JKS格式公钥
    

    默认 RabbitMQ 配置目录在 /etc/rabbitmq,我们创建个证书目录存放服务端证书

    mkdir -p /etc/rabbitmq/ssl
    #复制服务端必要证书
    cp ca/cacert.pem \
    server/rabbitmq-server.cert.pem \
    server/rabbitmq-server.key.pem /etc/rabbitmq/ssl/
    

    修改 RabbitMQ 配置文件

    修改 RabbitMQ 配置文件 /etc/rabbitmq/rabbitmq.config,此文件默认不存在,需要手动创建

    [{rabbit, [
        {ssl_listeners, [5671]},
        {ssl_options, [
            {cacertfile, "/etc/rabbitmq/ssl/cacert.pem"},
            {certfile,   "/etc/rabbitmq/ssl/rabbitmq-server.cert.pem"},
            {keyfile,    "/etc/rabbitmq/ssl/rabbitmq-server.key.pem"},
            {verify, verify_peer},
            {fail_if_no_peer_cert, true},
            {ciphers, [
                "ECDHE-ECDSA-AES256-GCM-SHA384","ECDHE-RSA-AES256-GCM-SHA384",
                "ECDHE-ECDSA-AES256-SHA384","ECDHE-RSA-AES256-SHA384",
                "ECDHE-ECDSA-DES-CBC3-SHA","ECDH-ECDSA-AES256-GCM-SHA384",
                "ECDH-RSA-AES256-GCM-SHA384","ECDH-ECDSA-AES256-SHA384",
                "ECDH-RSA-AES256-SHA384","DHE-DSS-AES256-GCM-SHA384",
                "DHE-DSS-AES256-SHA256","AES256-GCM-SHA384",
                "AES256-SHA256","ECDHE-ECDSA-AES128-GCM-SHA256",
                "ECDHE-RSA-AES128-GCM-SHA256","ECDHE-ECDSA-AES128-SHA256",
                "ECDHE-RSA-AES128-SHA256","ECDH-ECDSA-AES128-GCM-SHA256",
                "ECDH-RSA-AES128-GCM-SHA256","ECDH-ECDSA-AES128-SHA256",
                "ECDH-RSA-AES128-SHA256","DHE-DSS-AES128-GCM-SHA256",
                "DHE-DSS-AES128-SHA256","AES128-GCM-SHA256",
                "AES128-SHA256","ECDHE-ECDSA-AES256-SHA",
                "ECDHE-RSA-AES256-SHA","DHE-DSS-AES256-SHA",
                "ECDH-ECDSA-AES256-SHA","ECDH-RSA-AES256-SHA",
                "AES256-SHA","ECDHE-ECDSA-AES128-SHA",
                "ECDHE-RSA-AES128-SHA","DHE-DSS-AES128-SHA",
                "ECDH-ECDSA-AES128-SHA","ECDH-RSA-AES128-SHA","AES128-SHA"
            ]}
        ]},
        {auth_mechanisms,['EXTERNAL']},
        {ssl_cert_login_from,common_name}
    ]}].
    

    主要配置项说明:

    • ssl_listeners 指定 SSL协议的端口号,官方文档 5671
    • ssl_options SSL 认证配置项
      • cacertfile CA 证书位置
      • certfile 公钥证书位置
      • keyfile 密钥证书位置
      • verify
        • verify_peer 客户端与服务端互相发送证书
        • verify_none 禁用证书交换与校验
      • fail_if_no_peer_cert
        • true 不接受没证书的客户端连接
        • false 接受没证书的客户端连接
      • ciphers 加密器(这个翻译不知道算不算对?)
    • auth_mechanisms 认证机制,此处使用 EXTERNAL 表示只使用插件提供认证功能
    • ssl_cert_login_from 使用证书中的哪些信息登录,如果不配置这项是走的DN,配置走CN
      • common_name CN名称

    启用插件

    #启用rabbitmq_auth_mechanism_ssl作为EXTERNAL认证机制的实现
    rabbitmq-plugins enable rabbitmq_auth_mechanism_ssl
    #查看启动结果
    rabbitmq-plugins list
    

    重启 RabbitMQ

    #关闭
    rabbitmqctl stop
    #启动
    rabbitmq-server -detached
    

    添加证书登录用户与授权

    #添加证书登录用户(用户名要与客户端证书名称前缀一致),密码任意
    rabbitmqctl add_user 'rabbitmq-client' '2a55f70a841f18b97c3a7db939b7adc9e34a0f1b'
    #给rabbitmq-client用户虚拟主机/的所有权限,如需其他虚拟主机替换/
    rabbitmqctl set_permissions -p "/" "rabbitmq-client" ".*" ".*" ".*"
    

    验证开启 SSL 是否成功

    使用 Rabbitmq 自带的诊断工具查看端口监听状态及使用协议

    #查看监听
    rabbitmq-diagnostics listeners
    #查看支持的TLS版本
    rabbitmq-diagnostics --silent tls_versions
    

    使用 OpenSSL CLI 工具验证证书是否有效

    cd 生成证书的ssl目录
    #使用客户端证书+CA证书连接RabbitMQ验证。本处MQ与生成证书是同一主机,其他情况请自行考虑。
    openssl s_client -connect localhost:5671 \
      -cert client/rabbitmq-client.cert.pem \
      -key client/rabbitmq-client.key.pem \
      -CAfile ca/cacert.pem
    

    除了命令行查看外,还可以通过管理界面查看,不过只能确定开启了 SSL 监听,无法确认证书是否通过验证。

    编写 SpringBoot 代码连接测试

    代码结构

    只是使用 start.spring.io 生成的 Maven 工程,依赖了 WEB 和 AMQP

    2022.02.10更新:由于读者反应测试代码行为不正常(原因是目录结构放得不对!),现已将测试代码已上传本人GitHub https://github.com/hellxz/rabbitmq-ssl-demo , 使用时注意替换证书文件!

    代码及配置

    pom.xml

    <?xml version="1.0" encoding="UTF-8"?>
    <project xmlns="http://maven.apache.org/POM/4.0.0"
    	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    	xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
    	<modelVersion>4.0.0</modelVersion>
    	<parent>
    		<groupId>org.springframework.boot</groupId>
    		<artifactId>spring-boot-starter-parent</artifactId>
    		<version>2.5.0</version>
    		<relativePath /> <!-- lookup parent from repository -->
    	</parent>
    	<groupId>com.example</groupId>
    	<artifactId>demo</artifactId>
    	<version>0.0.1-SNAPSHOT</version>
    	<name>demo</name>
    	<description>Demo project for Spring Boot</description>
    	<properties>
    		<java.version>1.8</java.version>
    	</properties>
    	<dependencies>
    		<dependency>
    			<groupId>org.springframework.boot</groupId>
    			<artifactId>spring-boot-starter-amqp</artifactId>
    		</dependency>
    		<dependency>
    			<groupId>org.springframework.boot</groupId>
    			<artifactId>spring-boot-starter-web</artifactId>
    		</dependency>
    		<!-- https://mvnrepository.com/artifact/org.apache.commons/commons-lang3 -->
    		<dependency>
    			<groupId>org.apache.commons</groupId>
    			<artifactId>commons-lang3</artifactId>
    		</dependency>
    
    		<dependency>
    			<groupId>org.springframework.boot</groupId>
    			<artifactId>spring-boot-starter-test</artifactId>
    			<scope>test</scope>
    		</dependency>
    		<dependency>
    			<groupId>org.springframework.amqp</groupId>
    			<artifactId>spring-rabbit-test</artifactId>
    			<scope>test</scope>
    		</dependency>
    	</dependencies>
    
    	<build>
    		<plugins>
    			<plugin>
    				<groupId>org.springframework.boot</groupId>
    				<artifactId>spring-boot-maven-plugin</artifactId>
    			</plugin>
    		</plugins>
    	</build>
    
    </project>
    

    启动类 DemoApplication.java

    package com.hellxz.rabbitmq.ssl;
    
    import org.springframework.boot.SpringApplication;
    import org.springframework.boot.autoconfigure.SpringBootApplication;
    
    @SpringBootApplication
    public class DemoApplication {
    
        public static void main(String[] args) {
            SpringApplication.run(DemoApplication.class, args);
        }
    
    }
    

    RabbitMQ客户端配置类 RabbitFanoutExchangeConfig.java

    package com.hellxz.rabbitmq.ssl;
    
    import javax.annotation.PostConstruct;
    
    import org.apache.commons.lang3.BooleanUtils;
    import org.springframework.amqp.core.Binding;
    import org.springframework.amqp.core.BindingBuilder;
    import org.springframework.amqp.core.FanoutExchange;
    import org.springframework.amqp.core.Queue;
    import org.springframework.amqp.rabbit.connection.CachingConnectionFactory;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.beans.factory.annotation.Qualifier;
    import org.springframework.boot.autoconfigure.amqp.RabbitProperties;
    import org.springframework.context.annotation.Bean;
    import org.springframework.context.annotation.Configuration;
    
    import com.rabbitmq.client.DefaultSaslConfig;
    
    
    @Configuration
    public class RabbitFanoutExchangeConfig {
        public static final String FANOUT_EXCHANGE = "fanout.exchange";
        public static final String FANOUT_QUEUE1 = "fanout.queue1";
    
        @Bean(name = FANOUT_EXCHANGE)
        public FanoutExchange fanoutExchange() {
            return new FanoutExchange(FANOUT_EXCHANGE, true, false);
        }
    
    
        @Bean(name = FANOUT_QUEUE1)
        public Queue fanoutQueue1() {
            return new Queue(FANOUT_QUEUE1, true, false, false);
        }
        
        @Bean
        public Binding bindingSimpleQueue1(@Qualifier(FANOUT_QUEUE1) Queue fanoutQueue1,
                                           @Qualifier(FANOUT_EXCHANGE) FanoutExchange fanoutExchange) {
            return BindingBuilder.bind(fanoutQueue1).to(fanoutExchange);
        }
        
        @Autowired
        RabbitProperties rabbitProperties;
        @Autowired
        CachingConnectionFactory cachingConnectionFactory;
        
        /**
         * 解决安全扫描 AMQP明文登录漏洞 仅当rabbitmq启用ssl时并且配置证书时,显式设置EXTERNAL认证机制<br/>
         * EXTERNAL认证机制使用X509认证方式,服务端读取客户端证书中的CN作为登录名称,同时忽略密码
         */
        @PostConstruct
        public void rabbitmqSslExternalPostConstruct() {
            boolean rabbitSslEnabled = BooleanUtils.toBoolean(rabbitProperties.getSsl().getEnabled());
            boolean rabbitSslKeyStoreExists = rabbitProperties.getSsl().getKeyStore() != null;
            if (rabbitSslEnabled && rabbitSslKeyStoreExists) {
                cachingConnectionFactory.getRabbitConnectionFactory().setSaslConfig(DefaultSaslConfig.EXTERNAL);
            }
        }
    }
    

    这里添加 @PostConstruct 作的处理是因为维护人员觉得把它做成可配置的收益不大,大部分人都不需要。

    发消息测试类 TestController.java

    package com.hellxz.rabbitmq.ssl;
    
    import org.springframework.amqp.core.Message;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.web.bind.annotation.GetMapping;
    import org.springframework.web.bind.annotation.RestController;
    
    @RestController
    public class TestController {
        @Autowired
        RabbitMQSenderService rabbitMQSenderService;
        
        @GetMapping("/test")
        public void sendMsg() {
            Message msg = new Message("hello world".getBytes());
            try {
                rabbitMQSenderService.send(RabbitFanoutExchangeConfig.FANOUT_EXCHANGE,
                        RabbitFanoutExchangeConfig.FANOUT_QUEUE1, msg);
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }
    

    发消息服务 RabbitMQSenderService.java

    package com.hellxz.rabbitmq.ssl;
    
    import java.util.UUID;
    
    import org.springframework.amqp.core.Message;
    import org.springframework.amqp.rabbit.connection.CorrelationData;
    import org.springframework.amqp.rabbit.core.RabbitTemplate;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.stereotype.Component;
    
    @Component
    public class RabbitMQSenderService {
        @Autowired
        private RabbitTemplate rabbitTemplate;
    
        public void send(String exchange, String routingkey, Message message) {
    
            CorrelationData correlationId = new CorrelationData(UUID.randomUUID().toString());
            System.out.println("start send msg : " + message);
            rabbitTemplate.convertAndSend(exchange, routingkey, message, correlationId);
            System.out.println("end send msg : " + message);
        }
    }
    

    消息接收者 RabbitMQReciver.java

    package com.hellxz.rabbitmq.ssl;
    
    import org.springframework.amqp.rabbit.annotation.RabbitListener;
    import org.springframework.stereotype.Component;
    
    @Component
    class RabbitMQReciver {
    
        @RabbitListener(queues = RabbitFanoutExchangeConfig.FANOUT_QUEUE1)
        public void reciveLogAll(String msg) throws Exception {
            System.out.println("received msg:" + msg);
        }
    }
    

    配置文件 application.properties

    server.port=8085
    
    #基础配置请根据实际配置,此种配置方式无需配置用户名与密码
    spring.rabbitmq.host=192.168.56.104
    #ssl协议端口
    spring.rabbitmq.port=5671
    spring.rabbitmq.virtual-host=/
    
    #启用rabbitmq客户端SSL连接
    spring.rabbitmq.ssl.enabled=true
    #客户端PKCS12证书及密码
    spring.rabbitmq.ssl.key-store=classpath:ssl/rabbitmq-client.keycert.p12
    spring.rabbitmq.ssl.key-store-password=654321
    #公钥证书及类型
    spring.rabbitmq.ssl.trust-store=classpath:ssl/rabbitmqTrustStore
    spring.rabbitmq.ssl.trust-store-type=JKS
    #不校验主机名,默认开启会导致连接失败
    spring.rabbitmq.ssl.verify-hostname=false
    

    src/main/resources 下创建 ssl 目录,将 客户端证书和服务端 JKS 公钥复制到 ssl 目录中。

    执行代码验证

    运行 DemoApplication.java,查看控制台是否有报错:

    如图,提示创建连接成功,说明已经连接成功了。

    启动成功的连接消息中 amqp://后的 guest 并非真实的登录名称,仅仅是一个占位符

    Created new connection: rabbitConnectionFactory#476ec9d0:0/SimpleConnection@474c9131 [delegate=amqp://guest@192.168.56.104:5671/, localPort= 7956]
    

    实际访问的用户可以在服务端日志处察看到,笔者已测试删除 guest 用户后重新连接测试,测试通过服务端打印真实客户端名称。

    我们再调用 TestController.java 中定义的 /test 接口

    消息发送与消费成功。

    参考

    本文同步于本人博客园(hellxz.cnblogs.com) 与 CSDN(https://blog.csdn.net/u012586326),禁止转载。

  • 相关阅读:
    JSP根据身份证号码计算生日
    JSP听课笔记(二)
    JSP听课笔记(一)
    JDBC连接数据库过程(转载)
    PL/SQL
    阅读笔记
    javaScript的一些奇妙动画
    原型与原型链之间一些奥秘
    CSS3 @keyframes 规则
    Java语言中的基本词汇
  • 原文地址:https://www.cnblogs.com/hellxz/p/15776987.html
Copyright © 2020-2023  润新知