如何为 Java 服务器拥有多个 SSL 证书
我有一个用Java编写的内部HTTP服务器;完整的源代码可供我使用。HTTP 服务器可以配置任意数量的网站,每个网站都有一个单独的侦听套接字,创建:
skt=SSLServerSocketFactory.getDefault().createServerSocket(prt,bcklog,adr);
使用使用 Java keytool 创建的标准密钥存储区,我无法解决如何获取与不同侦听套接字关联的不同证书,以便每个配置的网站都有自己的证书。
我现在正处于时间紧要关头,因此非常感谢一些说明的代码示例。但是,我非常希望对JSSE在这方面如何结合在一起的任何很好的概述(我已经搜索了Sun的JSSE doco,直到我的大脑疼痛(从字面上看;尽管它可能同样多的咖啡因戒断))。
编辑
有没有简单的方法可以使用别名将密钥库中的服务器证书与侦听套接字相关联?因此:
- 客户有一个密钥存储区要管理所有证书,并且
- 无需摆弄多个密钥存储等。
我得到的印象是(今天下午早些时候),我可以写一个简单的KeyManager,只返回non-null,这是我想要的别名的名称 - 有人对这种推理有任何想法吗?chooseServerAlias(...)
溶液
我使用的解决方案是根据slyvarking的答案构建的,是创建一个临时密钥存储,并使用从单一外部密钥存储中提取的所需密钥/证书填充它。对于任何感兴趣的人,代码如下(svrctfals是我的“服务器证书别名”值):
SSLServerSocketFactory ssf; // server socket factory
SSLServerSocket skt; // server socket
// LOAD EXTERNAL KEY STORE
KeyStore mstkst;
try {
String kstfil=GlobalSettings.getString("javax.net.ssl.keyStore" ,System.getProperty("javax.net.ssl.keyStore" ,""));
String ksttyp=GlobalSettings.getString("javax.net.ssl.keyStoreType" ,System.getProperty("javax.net.ssl.keyStoreType" ,"jks"));
char[] kstpwd=GlobalSettings.getString("javax.net.ssl.keyStorePassword",System.getProperty("javax.net.ssl.keyStorePassword","")).toCharArray();
mstkst=KeyStore.getInstance(ksttyp);
mstkst.load(new FileInputStream(kstfil),kstpwd);
}
catch(java.security.GeneralSecurityException thr) {
throw new IOException("Cannot load keystore ("+thr+")");
}
// CREATE EPHEMERAL KEYSTORE FOR THIS SOCKET USING DESIRED CERTIFICATE
try {
SSLContext ctx=SSLContext.getInstance("TLS");
KeyManagerFactory kmf=KeyManagerFactory.getInstance(KeyManagerFactory.getDefaultAlgorithm());
KeyStore sktkst;
char[] blkpwd=new char[0];
sktkst=KeyStore.getInstance("jks");
sktkst.load(null,blkpwd);
sktkst.setKeyEntry(svrctfals,mstkst.getKey(svrctfals,blkpwd),blkpwd,mstkst.getCertificateChain(svrctfals));
kmf.init(sktkst,blkpwd);
ctx.init(kmf.getKeyManagers(),null,null);
ssf=ctx.getServerSocketFactory();
}
catch(java.security.GeneralSecurityException thr) {
throw new IOException("Cannot create secure socket ("+thr+")");
}
// CREATE AND INITIALIZE SERVER SOCKET
skt=(SSLServerSocket)ssf.createServerSocket(prt,bcklog,adr);
...
return skt;