gpt4 book ai didi

java - 请求自动或用户选择适当的客户端证书

转载 作者:塔克拉玛干 更新时间:2023-11-02 18:55:39 26 4
gpt4 key购买 nike

我正在开发一个可能连接到不同服务器的混合 cordova 应用程序。其中一些确实需要客户端证书。

在 Android 手机上安装相应的根证书 + 客户端证书。

在 Chrome 浏览器上,我会看到以下对话框,为 Web 连接选择相应的客户端证书。

Choose certificate on Chrome

使用 cordova 插件 cordova-client-cert-authentication为 WebView 中的 Http(s) 请求弹出相同的对话框。

我的问题是如何在native Android 平台上的Http(s) 请求上实现自动证书选择,而无需显式声明相应的客户端证书。或者是否有类似在 Chrome 上实现的类似于用户选择证书的东西

这是当前的实现,它抛出一个握手异常:

try {
URL url = new URL( versionUrl );
HttpsURLConnection urlConnection = ( HttpsURLConnection ) url.openConnection();

urlConnection.setConnectTimeout( 10000 );

InputStream in = urlConnection.getInputStream();
}
catch(Exception e)
{
//javax.net.ssl.SSLHandshakeException: Handshake failed
}

最佳答案

您可以使用先前安装在 Android KeyChain(系统 keystore )中的证书扩展 X509ExtendedKeyManager 来配置 URLConnection 使用的 SSLContext/p>

证书由您需要的别名引用。要使用类似于 chrome 使用的对话框提示用户进行选择:

KeyChain.choosePrivateKeyAlias(this, this, // Callback
new String[] {"RSA", "DSA"}, // Any key types.
null, // Any issuers.
null, // Any host
-1, // Any port
DEFAULT_ALIAS);

这是使用自定义 KeyManager 配置 SSL 连接的代码。它使用默认的 TrustManagerHostnameVerifier。如果服务器使用的是 Android 默认信任库中不存在的自签名证书,则需要配置它们(不建议信任所有证书)

//Configure trustManager if needed
TrustManager[] trustManagers = null;

//Configure keyManager to select the private key and the certificate chain from KeyChain
KeyManager keyManager = KeyChainKeyManager.fromAlias(
context, mClientCertAlias);

//Configure SSLContext
SSLContext sslContext = SSLContext.getInstance("TLS");
sslContext.init(new KeyManager[] {keyManager}, trustManagers, null);


//Perform the connection
URL url = new URL( versionUrl );
HttpsURLConnection urlConnection = ( HttpsURLConnection ) url.openConnection();
urlConnection.setSSLSocketFactory(sslContext.getSocketFactory());
//urlConnection.setHostnameVerifier(hostnameVerifier); //Configure hostnameVerifier if needed
urlConnection.setConnectTimeout( 10000 );
InputStream in = urlConnection.getInputStream();

最后,您拥有了从 here 中提取的自定义 X509ExtendedKeyManager 的完整实现。和 here负责选择客户端证书。我已经提取了所需的代码。

public static class KeyChainKeyManager extends X509ExtendedKeyManager {
private final String mClientAlias;
private final X509Certificate[] mCertificateChain;
private final PrivateKey mPrivateKey;

/**
* Builds an instance of a KeyChainKeyManager using the given certificate alias.
* If for any reason retrieval of the credentials from the system {@link android.security.KeyChain} fails,
* a {@code null} value will be returned.
*/
public static KeyChainKeyManager fromAlias(Context context, String alias)
throws CertificateException {
X509Certificate[] certificateChain;
try {
certificateChain = KeyChain.getCertificateChain(context, alias);
} catch (KeyChainException e) {
throw new CertificateException(e);
} catch (InterruptedException e) {
throw new CertificateException(e);
}

PrivateKey privateKey;
try {
privateKey = KeyChain.getPrivateKey(context, alias);
} catch (KeyChainException e) {
throw new CertificateException(e);
} catch (InterruptedException e) {
throw new CertificateException(e);
}

if (certificateChain == null || privateKey == null) {
throw new CertificateException("Can't access certificate from keystore");
}

return new KeyChainKeyManager(alias, certificateChain, privateKey);
}

private KeyChainKeyManager(
String clientAlias, X509Certificate[] certificateChain, PrivateKey privateKey) {
mClientAlias = clientAlias;
mCertificateChain = certificateChain;
mPrivateKey = privateKey;
}


@Override
public String chooseClientAlias(String[] keyTypes, Principal[] issuers, Socket socket) {
return mClientAlias;
}

@Override
public X509Certificate[] getCertificateChain(String alias) {
return mCertificateChain;
}

@Override
public PrivateKey getPrivateKey(String alias) {
return mPrivateKey;
}

@Override
public final String chooseServerAlias( String keyType, Principal[] issuers, Socket socket) {
// not a client SSLSocket callback
throw new UnsupportedOperationException();
}

@Override
public final String[] getClientAliases(String keyType, Principal[] issuers) {
// not a client SSLSocket callback
throw new UnsupportedOperationException();
}

@Override
public final String[] getServerAliases(String keyType, Principal[] issuers) {
// not a client SSLSocket callback
throw new UnsupportedOperationException();
}
}
}

我没有测试过。报告任何错误!

关于java - 请求自动或用户选择适当的客户端证书,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44929645/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com