HttpsURLConnection 跳过ssl访问https

package com.gcno93.tls.nettyssl;

import javax.net.ssl.*;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.net.MalformedURLException;
import java.net.Socket;
import java.net.URL;
import java.nio.charset.StandardCharsets;
import java.security.KeyManagementException;
import java.security.NoSuchAlgorithmException;
import java.security.SecureRandom;
import java.security.cert.CertificateException;
import java.security.cert.X509Certificate;
import java.util.Map;

/**
 * @author lwh
 * @date 2022/1/12
 */
public class HttpsURLConnectionUtil {
    public static String sentRequest(String path, HttpMethod method, Map<String,String> properties,Map<String, String> params)  {
        HttpsURLConnection   connection = null;
        StringBuffer buffer = new StringBuffer();
        try{
            //信任策略,我自己实现的策略,无需验证服务器的ssl证书
            TrustManager[] tm = {new NonValidationVerX509TrustManager()};
            //getInstance的第一个参数,具体查看https://docs.oracle.com/javase/8/docs/technotes/guides/security/StandardNames.html#SSLContext
            SSLContext sslContext = SSLContext.getInstance("SSL");
            //第一个参数:密码仓库,我不需要验证证书,所以不需要,设置为空
            //第二个参数:信任策略,传我自己的
            //第三个参数:随机数
            sslContext.init(null, tm, new SecureRandom());
            //创建socket工厂
            SSLSocketFactory socketFactory = sslContext.getSocketFactory();
            //创建访问的url
            URL url = new URL(path);
            //获取HttpsURLConnection对象
            connection = (HttpsURLConnection) url.openConnection();
            //设置socket工厂
            connection.setSSLSocketFactory(socketFactory);
            //设置请求方式
            connection.setRequestMethod(method.toString());
            connection.setHostnameVerifier(new HostnameVerifier() {
                public boolean verify(String hostname, SSLSession session) {
                    return true;
                }
            });
            //设置Property
            HttpsURLConnection finalConnection = connection;
            if(properties!=null){
                for (Map.Entry<String, String> entry : properties.entrySet()) {
                    connection.setRequestProperty(entry.getKey(),entry.getValue());
                }
            }

            //设置请求参数
            if(params!=null){
                try(final OutputStreamWriter outputStream = new OutputStreamWriter(connection.getOutputStream());){
                        StringBuilder param = new StringBuilder();
                        for (Map.Entry<String, String> entry : params.entrySet()) {
                            if (param.length() > 0) {
                                param.append("&");
                            }
                            param.append(entry.getKey());
                            param.append("=");
                            param.append(entry.getValue());
                        }
                        System.out.println("param:" + param.toString());
                        outputStream.write(param.toString());
                }
            }


            //处理结果
            if(connection.getResponseCode()==200){
                try(InputStream  inputStream = connection.getInputStream();){
                    final byte[] buf = new byte[1024];
                    int len = 0;
                    while ((len =inputStream.read(buf) )!=-1){
                        buffer.append(new String(buf,0,len, StandardCharsets.UTF_8));
                    }
                }
            }
        } catch (MalformedURLException | NoSuchAlgorithmException | KeyManagementException e) {
            throw new RuntimeException("HttpsURLConnection 的SSL 配置异常"+e);
        } catch (IOException ioEx) {
            throw new RuntimeException("请求失败,原因:"+ ioEx);
        } finally {
            //关闭资源
            if(connection!=null){
                connection.disconnect();
            }
        }
        return buffer.toString();
    }
    public static void main(String[] args)throws Exception{
        final String s = HttpsURLConnectionUtil.sentRequest("https://blog.youkuaiyun.com/gcno93/article/details/122448520", HttpMethod.GET, null, null);
        System.out.println(s);
    }

    //请求类型 post/get
    public static enum HttpMethod{
        GET,POST;
    }


    public static class NonValidationVerX509TrustManager implements X509TrustManager {

        /**
         * 这里是服务器验证客户端证书的,这里不需要验证
         * @param chain 证书链,具体查看{@link X509TrustManager}
         * @param authType 基于客户端证书的认证类型 比如RSA,具体查看{@link X509TrustManager}
         * @throws CertificateException
         */
        @Override
        public void checkClientTrusted(X509Certificate[] chain, String authType) throws CertificateException {

        }

        /**
         * 这里是客户端验证服务器证书的,这里不需要验证
         * @param chain 证书链,具体查看{@link X509TrustManager}
         * @param authType 使用的密钥交换算法,具体查看{@link X509TrustManager}
         * @throws CertificateException 证书验证失败,会抛出此异常
         */
        @Override
        public void checkServerTrusted(X509Certificate[] chain, String authType) throws CertificateException {
            System.out.println("不需要验证");
        }

        /**
         * 获取受信任的证书列表
         * @return 受信任的证书列表
         */
        @Override
        public X509Certificate[] getAcceptedIssuers() {
            return new X509Certificate[0];
        }
    }
}

1.发现的问题
网上很多人多是 implements X509TrustManager,它最终会被包装成AbstractTrustManagerWrapper,我本地测试时,发现还是无法访问javax.net.ssl.SSLHandshakeException: java.security.cert.CertificateException: No subject alternative names present,查看源码发现AbstractTrustManagerWrapper走的是
在这里插入图片描述
多了一个验证,这个验证好像是验证主机名等一些额外的验证,所以无法通过
后来我发现
在这里插入图片描述
继承这个X509ExtendedTrustManager类就可以了
2.你也可以 implements X509TrustManager,上面的代码就是这种
只要添加这个就好

connection.setHostnameVerifier(new HostnameVerifier() {
                public boolean verify(String hostname, SSLSession session) {
                    return true;
                }
            });

3.为什么添加这个就可以呢?我看了一下源码总结出来的类图
请添加图片描述

<think>我们之前已经讨论过如何在Java中跳过SSL证书验证。用户现在遇到了一个具体的异常:`java.security.cert.CertificateException: No subject alternative names present`,这通常是因为证书中缺少主题备用名称(SAN)而导致的。用户希望跳过SSL证书验证来解决这个问题。 解决方案仍然是使用自定义的TrustManager来信任所有证书,并禁用主机名验证。下面我将提供一个完整的示例代码,并针对用户遇到的异常给出解释。 注意:跳过SSL验证仅用于测试环境,生产环境必须使用有效的证书。 步骤: 1. 创建一个自定义的TrustManager,它信任所有证书。 2. 初始化一个SSLContext,使用这个TrustManager。 3. 将上述SSLContext设置为默认的SSLSocketFactory。 4. 同时设置一个接受所有主机名的HostnameVerifier。 用户遇到的异常“No subject alternative names present”是因为证书中没有配置主题备用名称(SAN),而Java要求SAN或者Common Name(CN)与请求的主机名匹配。如果我们跳过证书验证,这个问题也会被忽略。 下面给出具体代码:</think>### Java HTTPS请求跳过SSL证书验证方法 在Java中跳过SSL证书验证可通过自定义`TrustManager`实现,以下是完整解决方案: ```java import javax.net.ssl.*; import java.security.cert.X509Certificate; import java.net.URL; import java.io.BufferedReader; import java.io.InputStreamReader; public class SSLValidationBypass { // 创建信任所有证书的TrustManager private static final TrustManager[] trustAllCerts = new TrustManager[] { new X509TrustManager() { @Override public java.security.cert.X509Certificate[] getAcceptedIssuers() { return new X509Certificate[0]; } @Override public void checkClientTrusted( java.security.cert.X509Certificate[] certs, String authType) { } @Override public void checkServerTrusted( java.security.cert.X509Certificate[] certs, String authType) { } } }; // 禁用SSL验证的主方法 public static void disableSSLVerification() { try { // 创建信任所有证书的SSL上下文 SSLContext sc = SSLContext.getInstance("SSL"); sc.init(null, trustAllCerts, new java.security.SecureRandom()); // 设置全局SSL参数 HttpsURLConnection.setDefaultSSLSocketFactory(sc.getSocketFactory()); // 禁用主机名验证 (解决No subject alternative names present错误) HttpsURLConnection.setDefaultHostnameVerifier( (hostname, session) -> true ); } catch (Exception e) { e.printStackTrace(); } } // 使用示例 public static void main(String[] args) throws Exception { // 禁用SSL验证 disableSSLVerification(); // 发起HTTPS请求 URL url = new URL("https://your-server.com/api"); HttpsURLConnection conn = (HttpsURLConnection) url.openConnection(); conn.setRequestMethod("GET"); // 读取响应 try (BufferedReader in = new BufferedReader( new InputStreamReader(conn.getInputStream()))) { String line; while ((line = in.readLine()) != null) { System.out.println(line); } } } } ``` ### 关键点说明: 1. **解决`No subject alternative names present`错误** 通过`HostnameVerifier`强制返回`true`,绕过主机名验证[^1] 2. **全局禁用验证** `setDefaultSSLSocketFactory`和`setDefaultHostnameVerifier`会影响当前JVM的所有HTTPS连接 3. **替代方案(仅影响当前连接)**: ```java // 仅针对单个连接禁用验证 HttpsURLConnection conn = (HttpsURLConnection) url.openConnection(); conn.setSSLSocketFactory(sc.getSocketFactory()); conn.setHostnameVerifier((hostname, session) -> true); ``` ### ⚠️ 重要安全提示 1. **仅限开发测试使用** 生产环境禁用证书验证会带来严重安全风险: - 中间人攻击风险 - 敏感数据泄露 - 服务器身份无法验证 2. **正确解决方案** ```mermaid graph LR A[证书问题] --> B{解决方案} B --> C[添加SAN到证书] B --> D[导入有效证书到信任库] B --> E[使用权威CA签发证书] ``` 3. **临时使用场景** - 测试自签名证书 - 开发环境快速验证 - 解决证书过期/配置错误的临时方案
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值