JAVA请求HTTPS认证/PostMethod发送post请求忽略HTTPS证书验证

本文介绍了一种在Java中创建自定义SSL上下文的方法,通过实现X509TrustManager接口来信任所有证书,适用于HTTPS服务调用。该方法允许在不验证SSL证书的情况下建立HTTPS连接,对于开发测试环境或特定需求场景尤为有用。
package com.wis.mes.core.mq.protocol;

import org.apache.commons.httpclient.ConnectTimeoutException;
import org.apache.commons.httpclient.params.HttpConnectionParams;
import org.apache.commons.httpclient.protocol.ProtocolSocketFactory;

import javax.net.SocketFactory;
import javax.net.ssl.SSLContext;
import javax.net.ssl.TrustManager;
import javax.net.ssl.X509TrustManager;
import java.io.IOException;
import java.net.*;
import java.security.KeyManagementException;
import java.security.NoSuchAlgorithmException;
import java.security.cert.CertificateException;
import java.security.cert.X509Certificate;
/**
 * @Author CHENEY/陈荣根
 * @Date 2019/12/18
 * @Version 1.0
 * @Last Modified By : CHENEY
 * @Last Modified Time : 2019/12/18
 * @Description : CXF -> SOAP remote service
 * @Type https basic protocol
 * @Copyright All rights reserved.2019 WIS Software Co.*
 */
public class ProtocolSSLFactory implements ProtocolSocketFactory {
    private SSLContext sslcontext = null;

    private SSLContext createSSLContext() {
        SSLContext sslcontext = null;
        try {
            sslcontext = SSLContext.getInstance("SSL");
            sslcontext.init(null, new TrustManager[]{new TrustAnyTrustManager()}, new java.security.SecureRandom());
        } catch (NoSuchAlgorithmException e) {
            e.printStackTrace();
        } catch (KeyManagementException e) {
            e.printStackTrace();
        }
        return sslcontext;
    }

    private SSLContext getSSLContext() {
        if (this.sslcontext == null) {
            this.sslcontext = createSSLContext();
        }
        return this.sslcontext;
    }

    public Socket createSocket(Socket socket, String host, int port, boolean autoClose)
            throws IOException, UnknownHostException {
        return getSSLContext().getSocketFactory().createSocket(
                socket,
                host,
                port,
                autoClose
        );
    }

    public Socket createSocket(String host, int port) throws IOException,
            UnknownHostException {
        return getSSLContext().getSocketFactory().createSocket(
                host,
                port
        );
    }


    public Socket createSocket(String host, int port, InetAddress clientHost, int clientPort)
            throws IOException, UnknownHostException {
        return getSSLContext().getSocketFactory().createSocket(host, port, clientHost, clientPort);
    }

    public Socket createSocket(String host, int port, InetAddress localAddress,
                               int localPort, HttpConnectionParams params) throws IOException,
            UnknownHostException, ConnectTimeoutException {
        if (params == null) {
            throw new IllegalArgumentException("Parameters may not be null");
        }
        int timeout = params.getConnectionTimeout();
        SocketFactory socketfactory = getSSLContext().getSocketFactory();
        if (timeout == 0) {
            return socketfactory.createSocket(host, port, localAddress, localPort);
        } else {
            Socket socket = socketfactory.createSocket();
            SocketAddress localaddr = new InetSocketAddress(localAddress, localPort);
            SocketAddress remoteaddr = new InetSocketAddress(host, port);
            socket.bind(localaddr);
            socket.connect(remoteaddr, timeout);
            return socket;
        }
    }

    /**
     * 自定义管理器实现
     */
    private static class TrustAnyTrustManager implements X509TrustManager {
        public void checkClientTrusted(X509Certificate[] chain, String authType) throws CertificateException {
        }

        public void checkServerTrusted(X509Certificate[] chain, String authType) throws CertificateException {
        }

        public X509Certificate[] getAcceptedIssuers() {
            return new X509Certificate[]{};
        }
    }

}

校验规则: 

HttpClient http = new HttpClient();
String url = "https://***";
Protocol myhttps = new Protocol("https", new MySSLProtocolSocketFactory(), 443); 
Protocol.registerProtocol("https", myhttps); 
PostMethod post = new PostMethod(url);

 应用:

private String static BaseCoreContext.HTTPS_BASIC = "https";    
public static String SOAPHeaderRespStream(String SOAP_URL, String RXML, String userName,String pwd,int timeout) throws Exception {
        if (SOAP_URL.startsWith(BaseCoreContext.HTTPS_BASIC)) {
            Protocol _protocol = new Protocol(BaseCoreContext.HTTPS_BASIC, new ProtocolSSLFactory(), 443);
            Protocol.registerProtocol(BaseCoreContext.HTTPS_BASIC, _protocol);
        }
        HttpClient client = new HttpClient();
        PostMethod method = new PostMethod(SOAP_URL);
        if (timeout != 0) {
            HttpConnectionManagerParams connectionManager= client.getHttpConnectionManager().getParams();
            // 设置 Http 连接超时为timeout 毫秒
            connectionManager.setConnectionTimeout(timeout);
            // 设置读数据超时时间(单位毫秒)
            connectionManager.setSoTimeout(120000);
            // 设置get方法请求超时为 5 秒
            method.getParams().setParameter(HttpMethodParams.SO_TIMEOUT,timeout );
        }
        InputStream is = new ByteArrayInputStream(RXML.getBytes("UTF-8"));
        RequestEntity re = new InputStreamRequestEntity(is, "application/soap+xml; charset=utf-8");
        method.setRequestHeader("Content-Type", "text/xml;charset=utf-8");
        method.setRequestHeader("UserName", userName);
        method.setRequestHeader("Password", pwd);
        method.setRequestEntity(re);
        client.executeMethod(method);
        return method.getResponseBodyAsString();
    }

package com.nitccs.jit.service; import com.fasterxml.jackson.core.JsonProcessingException; import com.mysql.cj.core.util.StringUtils; import com.nitccs.jit.entity.JITAuthRet; import com.nitccs.jit.entity.JITRet; import com.nitccs.jit.excption.CryptoException; import com.nitccs.jit.utils.Base64Util; import com.nitccs.jit.utils.ObjectUtil; import org.apache.commons.httpclient.*; import org.apache.commons.httpclient.cookie.CookiePolicy; import org.apache.commons.httpclient.methods.PostMethod; import org.apache.commons.httpclient.methods.StringRequestEntity; import org.apache.commons.httpclient.protocol.Protocol; import org.apache.commons.httpclient.protocol.SecureProtocolSocketFactory; import org.slf4j.Logger; import org.springframework.beans.factory.annotation.Value; import org.springframework.stereotype.Service; import javax.annotation.PostConstruct; import javax.net.ssl.SSLContext; import javax.net.ssl.SSLSocket; import javax.net.ssl.TrustManager; import javax.net.ssl.X509TrustManager; import java.io.IOException; import java.net.MalformedURLException; import java.net.Socket; import java.net.URL; import java.security.KeyManagementException; import java.security.NoSuchAlgorithmException; import java.security.cert.X509Certificate; import java.util.*; import java.util.concurrent.ConcurrentHashMap; /** * @author 华 * @date 2025/6/18 * @desc JIT 密管平台基础服务 */ @Service public class JITBaseService { private static final Logger logger = org.slf4j.LoggerFactory.getLogger(JITBaseService.class); @Value("${jit.url}") String url; @Value("${jit.appId}") String appId; @Value("${jit.appPwd}") String appPwd; @Value("${jit.storageKeyCode}") String storageKeyCode; // 存储加解密 @Value("${jit.cookieName:KMS_SESSIONID}") String cookieName; final String version = "@V1"; private final Map<String, String> DEFAULT_HEADERS = new ConcurrentHashMap<>(); private final List<NameValuePair> DEFAULT_PARAMS = new ArrayList<>(); HttpClient httpClient; public JITBaseService() { } // 初始化默认请求头和参数 @PostConstruct private void init() { if (StringUtils.isNullOrEmpty(url)) { throw new CryptoException("请配置平台地址!"); } if (StringUtils.isNullOrEmpty(appId) || StringUtils.isNullOrEmpty(appPwd)) { throw new CryptoException("请配置平台标识和密钥!"); } DEFAULT_HEADERS.put("Content-Type", "application/json"); DEFAULT_HEADERS.put("Authorization", "Basic " + Base64Util.encode(appId + ":" + appPwd)); DEFAULT_PARAMS.add(new NameValuePair("keyCode", storageKeyCode)); DEFAULT_PARAMS.add(new NameValuePair("algorithmParam", "SM4/CBC/PKCS7Padding")); try { MultiThreadedHttpConnectionManager manager = new MultiThreadedHttpConnectionManager(); httpClient = new HttpClient(manager); httpClient.getParams().setCookiePolicy(CookiePolicy.BROWSER_COMPATIBILITY); URL platformUrl = new URL(url); // 配置主机信任证书的SSL上下文 registerHttpsProtocol(platformUrl.getHost(), platformUrl.getPort()); // 配置baseURL httpClient.getHostConfiguration().setHost(platformUrl.getHost(), platformUrl.getPort(), platformUrl.getProtocol()); // 连接超时时间 5s,读取超时时间 30s httpClient.getHttpConnectionManager().getParams().setConnectionTimeout(5 * 1000); httpClient.getParams().setSoTimeout(30 * 1000); logger.info("httpClient 初始化成功,平台地址:{}", url); } catch (MalformedURLException e) { throw new CryptoException(String.format("【%s】平台地址无效!", url)); } catch (Exception e) { throw new CryptoException("初始化HTTP客户端失败", e); } } /** * 特定主机证书验证 */ private void registerHttpsProtocol(String mHost, int mPort) { try { // 创建默认的SSL上下文(用于正常验证) SSLContext defaultSSLContext = SSLContext.getInstance("TLS"); defaultSSLContext.init(null, null, null); // 创建信任所有证书的SSL上下文(用于特定主机) SSLContext trustAllSSLContext = SSLContext.getInstance("TLS"); trustAllSSLContext.init(null, new TrustManager[]{ new X509TrustManager() { public X509Certificate[] getAcceptedIssuers() { return null; } public void checkClientTrusted(X509Certificate[] certs, String authType) {} public void checkServerTrusted(X509Certificate[] certs, String authType) {} } }, null); SecureProtocolSocketFactory socketFactory = new SecureProtocolSocketFactory() { @Override public Socket createSocket(Socket socket, String host, int port, boolean autoClose) throws IOException { try { // 对特定主机使用信任所有证书的上下文 if (Objects.equals(host, mHost) && Objects.equals(port, mPort)) { SSLSocket sslSocket = (SSLSocket) trustAllSSLContext.getSocketFactory().createSocket(socket, host, port, autoClose); return sslSocket; } // 其他主机使用默认上下文 return defaultSSLContext.getSocketFactory().createSocket(socket, host, port, autoClose); } catch (Exception e) { throw new IOException("创建SSL连接失败", e); } } @Override public Socket createSocket(String host, int port, java.net.InetAddress clientHost, int clientPort) throws IOException { if (Objects.equals(host, mHost) && Objects.equals(port, mPort)) { return trustAllSSLContext.getSocketFactory().createSocket(host, port, clientHost, clientPort); } return defaultSSLContext.getSocketFactory().createSocket(host, port, clientHost, clientPort); } @Override public Socket createSocket(String host, int port) throws IOException { if (Objects.equals(host, mHost) && Objects.equals(port, mPort)) { return trustAllSSLContext.getSocketFactory().createSocket(host, port); } return defaultSSLContext.getSocketFactory().createSocket(host, port); } @Override public Socket createSocket(String host, int port, java.net.InetAddress localAddress, int localPort, org.apache.commons.httpclient.params.HttpConnectionParams params) throws IOException { if (Objects.equals(host, mHost) && Objects.equals(port, mPort)) { return trustAllSSLContext.getSocketFactory().createSocket(host, port, localAddress, localPort); } return defaultSSLContext.getSocketFactory().createSocket(host, port, localAddress, localPort); } }; // 注册自定义的HTTPS协议 Protocol https = new Protocol("https", socketFactory, 443); Protocol.registerProtocol("https", https); logger.info("已注册自定义HTTPS协议,只对 {}:{} 忽略证书验证", mHost, mPort); } catch (NoSuchAlgorithmException | KeyManagementException e) { logger.error("配置SSL信任管理器失败", e); throw new CryptoException("配置SSL信任管理器失败", e); } } /** * 进行身份认证,获取 cookie */ public void authentication() { final String url = "/app/auth/v1/authentication"; Map<String, Object> params = new HashMap<>(); params.put("code", appId); params.put("secret", appPwd); JITAuthRet ret = post(params, url, JITAuthRet.class); logger.info("JIT 口令认证结果:{}", ret); // 认证成功后,设置cookie Cookie cookie = new Cookie(httpClient.getHostConfiguration().getHost(), cookieName, ret.getToken(), "/api/", ret.getExpirationTime(), false); httpClient.getState().addCookie(cookie); logger.info("JIT 口令认证成功,设置cookie: {}", cookie); } /** * 检查cookie是否过期,过期则重新认证 */ public void validCookie() { boolean needAuth = Arrays.stream(httpClient.getState().getCookies()) .noneMatch(cookie -> cookieName.equals(cookie.getName()) && !cookie.isExpired() ); if (needAuth) authentication(); } /** * 发送HTTPS POST请求 需要认证 * @param params 请求参数 * @param url 请求地址 * @param clazz 响应数据类型 * @return 响应数据 */ public <T> T sendRequest(Map<String, Object> params, String url, Class<T> clazz) { // 检查cookie是否过期,过期则重新认证 validCookie(); // 合并统一参数和请求参数 Map<String, Object> allParams = new HashMap<>(); for (NameValuePair pair : DEFAULT_PARAMS) { allParams.put(pair.getName(), pair.getValue()); } allParams.putAll(params); return post(allParams, url, clazz); } /** * 发送HTTPS POST请求 * @param params 请求参数 * @param url 请求地址 * @param clazz 响应数据类型 * @return 响应数据 */ private <T> T post(Map<String, Object> params, String url, Class<T> clazz) { PostMethod postMethod = null; try { postMethod = new PostMethod(this.url + url); // 设置统一请求头 for (Map.Entry<String, String> header : DEFAULT_HEADERS.entrySet()) { postMethod.setRequestHeader(header.getKey(), header.getValue()); } // 合并统一参数和请求参数 String jsonBody = ObjectUtil.toJson(params); // 设置请求postMethod.setRequestEntity(new StringRequestEntity(jsonBody, "application/json", "UTF-8")); logger.info("请求URL: {}", url); logger.info("请求头: {}", Arrays.toString(postMethod.getRequestHeaders())); logger.info("请求体: {}", postMethod.getResponseBodyAsString()); logger.info("创建请求-URL={},headers={},params={}", url, postMethod.getRequestHeaders(), jsonBody); // 执行请求 int statusCode = httpClient.executeMethod(postMethod); if (statusCode == HttpStatus.SC_OK) { String responseBody = postMethod.getResponseBodyAsString(); logger.info("请求成功,结果: {}", responseBody); JITRet ret = ObjectUtil.toObject(responseBody, JITRet.class); if (ret.isSuccess()) { return ObjectUtil.convert(ret.getData(), clazz); } throw new CryptoException(ret.getMessage()); } else if (statusCode == HttpStatus.SC_INTERNAL_SERVER_ERROR) { // 处理服务器内部错误 logger.warn("密管平台服务异常,请求失败,地址:{},状态码: {}", postMethod.getPath(), statusCode); throw new CryptoException(String.format("服务器异常【%s】,请稍后再试!", statusCode)); } else { logger.warn("请求失败,地址:{},状态码: {}", postMethod.getPath(), statusCode); throw new CryptoException(String.format("请求失败,地址:%s,状态码: %s", this.url + postMethod.getPath(), statusCode)); } } catch (HttpException e) { logger.error("HTTP协议错误", e); throw new CryptoException(String.format("HTTP协议错误: %s", e.getMessage())); } catch (IOException e) { logger.error("读取错误响应体失败", e); throw new CryptoException(String.format("读取错误响应体失败: %s", e.getMessage())); } finally { if (postMethod != null) { postMethod.releaseConnection(); } } } /** * 合并请求参数 * @param params 请求参数 * @return JSON 字符串 * @throws JsonProcessingException */ private String combineParams(Map<String, Object> params) throws JsonProcessingException { Map<String, Object> allParams = new HashMap<>(); for (NameValuePair pair : DEFAULT_PARAMS) { allParams.put(pair.getName(), pair.getValue()); } allParams.putAll(params); return ObjectUtil.toJson(allParams); } } 发送请求返回500,但使用postman和浏览器发送请求正常
06-21
评论 2
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

兰舟轻帆

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值