All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.ning.http.util.DefaultHostnameVerifier Maven / Gradle / Ivy

/*
 * To the extent possible under law, Kevin Locke has waived all copyright and
 * related or neighboring rights to this work.
 * 

* A legal description of this waiver is available in LICENSE.txt */ package com.ning.http.util; import javax.net.ssl.HostnameVerifier; import javax.net.ssl.SSLPeerUnverifiedException; import javax.net.ssl.SSLSession; import javax.security.auth.kerberos.KerberosPrincipal; import java.security.Principal; import java.security.cert.Certificate; import java.security.cert.CertificateException; import java.security.cert.X509Certificate; import java.util.logging.Level; import java.util.logging.Logger; /** * Uses the internal HostnameChecker to verify the server's hostname matches with the * certificate. This is a requirement for HTTPS, but the raw SSLEngine does not have * this functionality. As such, it has to be added in manually. For a more complete * description of hostname verification and why it's important, * please read * Fixing * Hostname Verification. *

* This code is based on Kevin Locke's guide . *

*/ public class DefaultHostnameVerifier implements HostnameVerifier { private HostnameChecker checker; private HostnameVerifier extraHostnameVerifier; // Logger to log exceptions. private static final Logger log = Logger.getLogger(DefaultHostnameVerifier.class.getName()); /** * A hostname verifier that uses the {{sun.security.util.HostnameChecker}} under the hood. */ public DefaultHostnameVerifier() { this.checker = new ProxyHostnameChecker(); } /** * A hostname verifier that takes an external hostname checker. Useful for testing. * * @param checker a hostnamechecker. */ public DefaultHostnameVerifier(HostnameChecker checker) { this.checker = checker; } /** * A hostname verifier that falls back to another hostname verifier if not found. * * @param extraHostnameVerifier another hostname verifier. */ public DefaultHostnameVerifier(HostnameVerifier extraHostnameVerifier) { this.checker = new ProxyHostnameChecker(); this.extraHostnameVerifier = extraHostnameVerifier; } /** * A hostname verifier with a hostname checker, that falls back to another hostname verifier if not found. * * @param checker a custom HostnameChecker. * @param extraHostnameVerifier another hostname verifier. */ public DefaultHostnameVerifier(HostnameChecker checker, HostnameVerifier extraHostnameVerifier) { this.checker = checker; this.extraHostnameVerifier = extraHostnameVerifier; } /** * Matches the hostname against the peer certificate in the session. * * @param hostname the IP address or hostname of the expected server. * @param session the SSL session containing the certificates with the ACTUAL hostname/ipaddress. * @return true if the hostname matches, false otherwise. */ private boolean hostnameMatches(String hostname, SSLSession session) { log.log(Level.FINE, "hostname = {0}, session = {1}", new Object[] { hostname, Base64.encode(session.getId()) }); try { final Certificate[] peerCertificates = session.getPeerCertificates(); if (peerCertificates.length == 0) { log.log(Level.FINE, "No peer certificates"); return false; } if (peerCertificates[0] instanceof X509Certificate) { X509Certificate peerCertificate = (X509Certificate) peerCertificates[0]; log.log(Level.FINE, "peerCertificate = {0}", peerCertificate); try { checker.match(hostname, peerCertificate); // Certificate matches hostname if no exception is thrown. return true; } catch (CertificateException ex) { log.log(Level.FINE, "Certificate does not match hostname", ex); } } else { log.log(Level.FINE, "Peer does not have any certificates or they aren't X.509"); } return false; } catch (SSLPeerUnverifiedException ex) { log.log(Level.FINE, "Not using certificates for peers, try verifying the principal"); try { Principal peerPrincipal = session.getPeerPrincipal(); log.log(Level.FINE, "peerPrincipal = {0}", peerPrincipal); if (peerPrincipal instanceof KerberosPrincipal) { return checker.match(hostname, (KerberosPrincipal) peerPrincipal); } else { log.log(Level.FINE, "Can't verify principal, not Kerberos"); } } catch (SSLPeerUnverifiedException ex2) { // Can't verify principal, no principal log.log(Level.FINE, "Can't verify principal, no principal", ex2); } return false; } } /** * Verifies the hostname against the peer certificates in a session. Falls back to extraHostnameVerifier if * there is no match. * * @param hostname the IP address or hostname of the expected server. * @param session the SSL session containing the certificates with the ACTUAL hostname/ipaddress. * @return true if the hostname matches, false otherwise. */ public boolean verify(String hostname, SSLSession session) { if (hostnameMatches(hostname, session)) { return true; } else { return extraHostnameVerifier != null && extraHostnameVerifier.verify(hostname, session); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy