com.paypal.base.SSLUtil Maven / Gradle / Ivy
package com.paypal.base;
import com.paypal.base.codec.binary.Base64;
import com.paypal.base.exception.SSLConfigurationException;
import com.paypal.base.rest.PayPalRESTException;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import javax.net.ssl.*;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.UnsupportedEncodingException;
import java.net.URL;
import java.security.*;
import java.security.cert.CertificateException;
import java.security.cert.CertificateFactory;
import java.security.cert.X509Certificate;
import java.util.*;
import java.util.zip.CRC32;
import java.util.zip.Checksum;
/**
* Class SSLUtil
*
*/
public abstract class SSLUtil {
private static final Logger log = LoggerFactory.getLogger(SSLUtil.class);
/**
* KeyManagerFactory used for {@link SSLContext} {@link KeyManager}
*/
private static final KeyManagerFactory KMF;
/**
* Private {@link Map} used for caching {@link KeyStore}s
*/
private static final Map STOREMAP;
/**
* Map used for dynamic configuration
*/
private static final Map CONFIG_MAP;
static {
try {
// Initialize KeyManagerFactory and local KeyStore cache
KMF = KeyManagerFactory.getInstance(KeyManagerFactory.getDefaultAlgorithm());
STOREMAP = new HashMap();
CONFIG_MAP = SDKUtil.combineDefaultMap(ConfigManager
.getInstance().getConfigurationMap());
} catch (NoSuchAlgorithmException e) {
throw new ExceptionInInitializerError(e);
}
}
/**
* Returns a SSLContext
*
* @param keymanagers
* KeyManager[] The key managers
* @return SSLContext with proper client certificate
* @throws SSLConfigurationException
*/
public static SSLContext getSSLContext(KeyManager[] keymanagers)
throws SSLConfigurationException {
try {
SSLContext ctx = null;
String protocol = CONFIG_MAP.get(Constants.SSLUTIL_PROTOCOL);
try {
ctx = SSLContext.getInstance("TLSv1.2");
} catch (NoSuchAlgorithmException e) {
log.warn("WARNING: Your system does not support TLSv1.2. Per PCI Security Council mandate (https://github.com/paypal/TLS-update), you MUST update to latest security library.");
ctx = SSLContext.getInstance(protocol);
}
ctx.init(keymanagers, null, null);
return ctx;
} catch (Exception e) {
throw new SSLConfigurationException(e.getMessage(), e);
}
}
/**
* Retrieves keyStore from the cached {@link Map}, if not present loads
* certificate into java keyStore and caches it for further references
*
* @param p12Path
* Path to the client certificate
* @param password
* {@link KeyStore} password
* @return keyStore {@link KeyStore} loaded with the certificate
* @throws NoSuchProviderException
* @throws KeyStoreException
* @throws CertificateException
* @throws NoSuchAlgorithmException
* @throws FileNotFoundException
* @throws IOException
*/
private static KeyStore p12ToKeyStore(String p12Path, String password)
throws NoSuchProviderException, KeyStoreException,
CertificateException, NoSuchAlgorithmException, IOException {
KeyStore keyStore = STOREMAP.get(p12Path);
if (keyStore == null) {
keyStore = KeyStore.getInstance("PKCS12", CONFIG_MAP.get(Constants.SSLUTIL_JRE));
FileInputStream in = null;
try {
in = new FileInputStream(p12Path);
keyStore.load(in, password.toCharArray());
STOREMAP.put(p12Path, keyStore);
} finally {
if (in != null) {
in.close();
}
}
}
return keyStore;
}
/**
* Create a SSLContext with provided client certificate
*
* @param certPath
* @param certPassword
* @return SSLContext
* @throws SSLConfigurationException
*/
public static SSLContext setupClientSSL(String certPath, String certPassword)
throws SSLConfigurationException {
SSLContext sslContext = null;
try {
KeyStore ks = p12ToKeyStore(certPath, certPassword);
KMF.init(ks, certPassword.toCharArray());
sslContext = getSSLContext(KMF.getKeyManagers());
} catch (NoSuchAlgorithmException e) {
throw new SSLConfigurationException(e.getMessage(), e);
} catch (KeyStoreException e) {
throw new SSLConfigurationException(e.getMessage(), e);
} catch (UnrecoverableKeyException e) {
throw new SSLConfigurationException(e.getMessage(), e);
} catch (CertificateException e) {
throw new SSLConfigurationException(e.getMessage(), e);
} catch (NoSuchProviderException e) {
throw new SSLConfigurationException(e.getMessage(), e);
} catch (IOException e) {
throw new SSLConfigurationException(e.getMessage(), e);
}
return sslContext;
}
/**
* Performs Certificate Chain Validation on provided certificates. The method verifies if the client certificates provided are generated from root certificates
* trusted by application.
*
* @param clientCerts Collection of X509Certificates provided in request
* @param trustCerts Collection of X509Certificates trusted by application
* @param authType Auth Type for Certificate
* @return true if client and server are chained together, false otherwise
* @throws PayPalRESTException
*/
public static boolean validateCertificateChain(Collection clientCerts, Collection trustCerts, String authType) throws PayPalRESTException {
TrustManager trustManagers[];
X509Certificate[] clientChain;
try {
clientChain = clientCerts.toArray(new X509Certificate[0]);
List list = Arrays.asList(clientChain);
clientChain = list.toArray(new X509Certificate[0]);
// Create a Keystore and load the Root CA Cert
KeyStore keyStore = KeyStore.getInstance(KeyStore.getDefaultType());
keyStore.load(null, "".toCharArray());
// Iterate through each certificate and add to keystore
int i = 0;
for (Iterator payPalCertificate = trustCerts.iterator(); payPalCertificate.hasNext();) {
X509Certificate x509Certificate = (X509Certificate) payPalCertificate.next();
keyStore.setCertificateEntry("paypalCert" + i, x509Certificate);
i++;
}
// Create TrustManager
TrustManagerFactory trustManagerFactory = TrustManagerFactory.getInstance(TrustManagerFactory.getDefaultAlgorithm());
trustManagerFactory.init(keyStore);
trustManagers = trustManagerFactory.getTrustManagers();
} catch (Exception ex) {
throw new PayPalRESTException(ex);
}
// For Each TrustManager of type X509
for(TrustManager trustManager : trustManagers) {
if(trustManager instanceof X509TrustManager) {
X509TrustManager pkixTrustManager = (X509TrustManager) trustManager;
// Check the trust manager if server is trusted
try {
pkixTrustManager.checkClientTrusted(clientChain, (authType == null || authType == "") ? "RSA" : authType);
// Checks that the certificate is currently valid. It is if the current date and time are within the validity period given in the certificate.
for (X509Certificate cert : clientChain) {
cert.checkValidity();
// Check for CN name matching
String dn = cert.getSubjectX500Principal().getName();
String[] tokens = dn.split(",");
boolean hasPaypalCn = false;
for (String token: tokens) {
if (token.startsWith("CN=messageverificationcerts") && token.endsWith(".paypal.com")) {
hasPaypalCn = true;
}
}
if (!hasPaypalCn) {
throw new PayPalRESTException("CN of client certificate does not match with trusted CN");
}
}
// If everything looks good, return true
return true;
} catch (CertificateException e) {
throw new PayPalRESTException(e);
}
}
}
return false;
}
/**
* Downloads Certificate from URL
* @deprecated Please use {@link #downloadCertificateFromPath(String, Map)} instead.
*
* @param urlPath
* @return InputStream containing certificate data
* @throws PayPalRESTException
*/
public static InputStream downloadCertificateFromPath(String urlPath) throws PayPalRESTException {
return downloadCertificateFromPath(urlPath, ConfigManager.getDefaultSDKMap());
}
/**
* Downloads Certificate from URL
*
* @param urlPath
* @param configurations Map of configurations.
* @return InputStream containing certificate data
* @throws PayPalRESTException
*/
public static InputStream downloadCertificateFromPath(String urlPath, Map configurations) throws PayPalRESTException {
if (urlPath == null || urlPath.trim() == "") {
throw new PayPalRESTException("Certificate Path cannot be empty");
}
try {
Map headerMap = new HashMap();
HttpConfiguration httpConfiguration = generateHttpConfiguration(urlPath, configurations);
HttpConnection connection = ConnectionManager.getInstance().getConnection();
connection.createAndconfigureHttpConnection(httpConfiguration);
URL url = new URL(urlPath);
headerMap.put("Host", url.getHost());
return connection.executeWithStream(url.toString(), "", headerMap);
} catch (Exception ex) {
throw new PayPalRESTException(ex);
}
}
private static HttpConfiguration generateHttpConfiguration(final String urlPath, final Map configurations) {
HttpConfiguration httpConfiguration = new HttpConfiguration();
httpConfiguration.setEndPointUrl(urlPath);
httpConfiguration.setConnectionTimeout(Integer.parseInt(configurations.get(Constants.HTTP_CONNECTION_TIMEOUT)));
httpConfiguration.setMaxRetry(Integer.parseInt(configurations.get(Constants.HTTP_CONNECTION_RETRY)));
httpConfiguration.setReadTimeout(Integer.parseInt(configurations.get(Constants.HTTP_CONNECTION_READ_TIMEOUT)));
httpConfiguration.setMaxHttpConnection(Integer.parseInt(configurations.get(Constants.HTTP_CONNECTION_MAX_CONNECTION)));
httpConfiguration.setHttpMethod("GET");
boolean useProxy = Boolean.parseBoolean(configurations.get(Constants.USE_HTTP_PROXY));
if(useProxy) {
httpConfiguration.setProxySet(useProxy);
httpConfiguration.setProxyHost(configurations.get(Constants.HTTP_PROXY_HOST));
httpConfiguration.setProxyPort(Integer.parseInt(configurations.get(Constants.HTTP_PROXY_PORT)));
String proxyUserName = configurations.get(Constants.HTTP_PROXY_USERNAME);
if(proxyUserName != null) {
httpConfiguration.setProxyUserName(proxyUserName);
httpConfiguration.setProxyPassword(configurations.get(Constants.HTTP_PROXY_PASSWORD));
}
}
return httpConfiguration;
}
/**
* Generate Collection of Certificate from Input Stream
*
* @param stream InputStream of Certificate data
* @return Collection
* @throws PayPalRESTException
*/
@SuppressWarnings("unchecked")
public static Collection getCertificateFromStream(InputStream stream) throws PayPalRESTException {
if (stream == null) { throw new PayPalRESTException("Certificate Not Found"); }
Collection certs = null;
try {
// Create a Certificate Factory
CertificateFactory cf = CertificateFactory.getInstance("X.509");
// Read the Trust Certs
certs = (Collection) cf.generateCertificates(stream);
} catch (CertificateException ex) {
throw new PayPalRESTException(ex);
}
return certs;
}
/**
* Generates a CRC 32 Value of String passed
*
* @param data
* @return long crc32 value of input. -1 if string is null
* @throws RuntimeException if UTF-8 is not a supported character set
*/
public static long crc32(String data) {
if (data == null) {
return -1;
}
try {
// get bytes from string
byte bytes[] = data.getBytes("UTF-8");
Checksum checksum = new CRC32();
// update the current checksum with the specified array of bytes
checksum.update(bytes, 0, bytes.length);
// get the current checksum value
return checksum.getValue();
} catch (UnsupportedEncodingException e) {
throw new RuntimeException(e);
}
}
/**
* Validates Webhook Signature validation based on https://developer.paypal.com/docs/integration/direct/rest-webhooks-overview/#event-signature
* Returns true if signature is valid
*
* @param clientCerts Client Certificates
* @param algo Algorithm used for signature creation by server
* @param actualSignatureEncoded Paypal-Transmission-Sig header value passed by server
* @param expectedSignature Signature generated by formatting data with CRC32 value of request body
* @param requestBody Request body from server
* @param webhookId Id for PayPal Webhook created for receiving the data
* @return true if signature is valid, false otherwise
*
* @throws NoSuchAlgorithmException
* @throws SignatureException
* @throws InvalidKeyException
*/
public static Boolean validateData(Collection clientCerts, String algo,
String actualSignatureEncoded, String expectedSignature, String requestBody, String webhookId) throws NoSuchAlgorithmException, SignatureException, InvalidKeyException {
// Get the signatureAlgorithm from the PAYPAL-AUTH-ALGO HTTP header
Signature signatureAlgorithm = Signature.getInstance(algo);
// Get the certData from the URL provided in the HTTP headers and cache it
X509Certificate[] clientChain = clientCerts.toArray(new X509Certificate[0]);
signatureAlgorithm.initVerify(clientChain[0].getPublicKey());
signatureAlgorithm.update(expectedSignature.getBytes());
// Actual signature is base 64 encoded and available in the HTTP headers
byte[] actualSignature = Base64.decodeBase64(actualSignatureEncoded.getBytes());
boolean isValid = signatureAlgorithm.verify(actualSignature);
return isValid;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy