HTTPS客户端java实现

本文介绍了HTTPS协议及其安全握手过程,详细阐述了在Java中创建HTTPS客户端的步骤,包括创建HttpsURLConnection对象、设置SSLSocketFactory,特别是实现自定义信任管理器进行证书校验的方法,涉及证书库的管理和证书导入。

https是http加入ssl安全协议后的数据传输协议,提供身份验证与加密通讯,默认端口为443,关于握手协议参考这里写链接内容

SSL是在安全套接层、传输层保障数据安全性的协议,安全认证过程:
1)客户端向服务器发送一个开始信息“Hello”以便开始一个新的会话连接;2)服务器根据客户的信息确定是否需要生成新的主密钥,如需要则服务器在响应客户的“Hello”信息时将包含生成主密钥所需的信息;3)客户根据收到的服务器响应信息,产生一个主密钥,并用服务器的公开密钥加密后传给服务器;4)服务器回复该主密钥,并返回给客户一个用主密钥认证的信息,以此让客户认证服务器。

实现https客户端过程:

1.创建HttpsURLConnection对象

URL url = new URL(null, path, new Handler());//handler是为了创建https协议的连接,否则下一步报错
HttpsURLConnection conn = (HttpsURLConnection)url.openConnection(); 
//实现自定义SSLSocketFactory,见下一步
SSLSocketFactory  sslFac = UMBX509TrustManager.getSSLFactory(); 

     if(StringUtils.isNotEmpty(method)){
         if(method.equalsIgnoreCase("GET")){
             conn.setRequestMethod("GET");
             //Get请求不需要DoOutPut
             conn.setDoOutput(false);
         }
         else if (method.equalsIgnoreCase("POST")){
             conn.setRequestMethod("POST");
             conn.setDoOutput(true);
             conn.setUseCaches(false);
         }
     }
     //设置SSLSocketFactory对象
     conn.setSSLSocketFactory(sslFac);
     conn.setDoInput(true);
     //设置连接超时时间和读取超时时间
     conn.setConnectTimeout(connectTimeout);
     conn.setReadTimeout(readTimeout);
     conn.setRequestProperty("Content-Type", "application/json; charset=utf-8");
     //connection初始化完成

2.创建SSLSocketFactory

//自定义信任管理器实现X509TrustManager接口方法
TrustManager[] trust = {new UMBX509TrustManager()};
SSLContext sslContext = SSLContext.getInstance("SSl", "SunJSSE");
//使用自定义信任管理器和随机数初始化
sslContext.init(null, trust, new SecureRandom());
return sslContext.getSocketFactory();

3.实现自定义信任管理器(实现接口X509TrustManager)
两种方式:
无需做证书校验:默认接口实现即可。

public interface X509TrustManager extends TrustManager {
    void checkClientTrusted(X509Certificate[] arg0, String arg1) throws CertificateException;

    void checkServerTrusted(X509Certificate[] arg0, String arg1) throws CertificateException;

    X509Certificate[] getAcceptedIssuers();
}

需要做证书校验:

  • 使用包含信任证书的信任库文件(.keystore、.jks)实例化x509TrustManager
//用于校验的x509管理器
X509TrustManager x509TrustManager;
//使用config文件作为受信任证书库
KeyStore keyStore = KeyStore.getInstance("JKS");
logger.info("use "+trustStorePath+" as trustStore");
//使用trustStorePath证书库文件,库密码trustStorePwd
keyStore.load(new FileInputStream(trustStorePath), 
trustStorePwd.toCharArray());
//获取证书库的trustManager
TrustManagerFactory factory = TrustManagerFactory
    .getInstance("SunX509", "SunJSSE"); 
factory.init(keyStore); 
TrustManager managers [] = factory.getTrustManagers(); 
/**
 * 遍历管理器,获取X509TrustManager的实现
*/
for (TrustManager manager : managers) { 
    if (manager instanceof X509TrustManager) { 
        x509TrustManager = (X509TrustManager) manager; 
     } 
} 
  • 重写校验方法
/**
 * 校验服务端证书
*/
    public void checkServerTrusted(X509Certificate[] chains, String authType) throws CertificateException {
    try {
    logger.info("checkServerTrusted:"+JSONArray.fromArray(chains).toString());
            x509TrustManager.checkServerTrusted(chains, authType); 
        } catch (CertificateException excep) { 
            //使用jdk默认证书校验,在初始化TrustManagerFactory时传入null
            initX509TrustManager(null).checkServerTrusted(chains, authType);
        } 
    }
    /**
     * 返回受信任的X509证书数组
     */
    public X509Certificate[] getAcceptedIssuers() {
         return x509TrustManager.getAcceptedIssuers(); 
    }
  • 信任证书来源不多说,得到信任证书后需要导入到证书库中或者新建一个库文件
    命令行进入到jdk bin目录下使用keytool工具:
    keytool -import -alias mycert -keystore myKeyStore.keystore -storepass password -file TruestedCert.cer
    -alias 在库中的别名
    -keystore 库文件,有则导入,没有新建
    -storepass 库密码,可以不设置
    -file 待导入的证书

4.建立连接

if("POST"){
    out = new BufferedWriter(new OutputStreamWriter(conn.getOutputStream(), "UTF-8"));
    out.write(data);
    out.flush();
}
if("GET"){
    conn.connect();
}   
try{
    return handleResponse(conn);
}catch (Exception e){
    if(!(e instanceof SocketException || e instanceof SocketTimeoutException)){
        logger.error(url,e);
    }
    throw e;
}
评论 3
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值