/*
* Copyright (C) 2014 The Android Open Source Project
* Copyright (c) 1997, 2010, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation. Oracle designates this
* particular file as subject to the "Classpath" exception as provided
* by Oracle in the LICENSE file that accompanied this code.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
package javax.net.ssl;
import java.net.*;
import javax.net.SocketFactory;
import java.io.IOException;
import java.security.*;
import java.util.Locale;
import sun.security.action.GetPropertyAction;
/**
* SSLSocketFactory
s create SSLSocket
s.
*
* @since 1.4
* @see SSLSocket
* @author David Brownell
*/
public abstract class SSLSocketFactory extends SocketFactory
{
private static SSLSocketFactory defaultSocketFactory;
private static int lastVersion = -1;
static final boolean DEBUG;
static {
String s = java.security.AccessController.doPrivileged(
new GetPropertyAction("javax.net.debug", "")).toLowerCase(
Locale.ENGLISH);
DEBUG = s.contains("all") || s.contains("ssl");
}
private static void log(String msg) {
if (DEBUG) {
System.out.println(msg);
}
}
/**
* Constructor is used only by subclasses.
*/
public SSLSocketFactory() {
}
/**
* Returns the default SSL socket factory.
*
*
The first time this method is called, the security property * "ssl.SocketFactory.provider" is examined. If it is non-null, a class by * that name is loaded and instantiated. If that is successful and the * object is an instance of SSLSocketFactory, it is made the default SSL * socket factory. * *
Otherwise, this method returns
* SSLContext.getDefault().getSocketFactory()
. If that
* call fails, an inoperative factory is returned.
*
* @return the default SocketFactory
* @see SSLContext#getDefault
*/
public static synchronized SocketFactory getDefault() {
// Android-changed: Use security version instead of propertyChecked.
if (defaultSocketFactory != null && lastVersion == Security.getVersion()) {
return defaultSocketFactory;
}
lastVersion = Security.getVersion();
SSLSocketFactory previousDefaultSocketFactory = defaultSocketFactory;
defaultSocketFactory = null;
String clsName = getSecurityProperty("ssl.SocketFactory.provider");
if (clsName != null) {
// The instance for the default socket factory is checked for updates quite
// often (for instance, every time a security provider is added). Which leads
// to unnecessary overload and excessive error messages in case of class-loading
// errors. Avoid creating a new object if the class name is the same as before.
if (previousDefaultSocketFactory != null
&& clsName.equals(previousDefaultSocketFactory.getClass().getName())) {
defaultSocketFactory = previousDefaultSocketFactory;
return defaultSocketFactory;
}
log("setting up default SSLSocketFactory");
try {
Class cls = null;
try {
cls = Class.forName(clsName);
} catch (ClassNotFoundException e) {
// Android-changed; Try the contextClassLoader first.
ClassLoader cl = Thread.currentThread().getContextClassLoader();
if (cl == null) {
cl = ClassLoader.getSystemClassLoader();
}
if (cl != null) {
cls = Class.forName(clsName, true, cl);
}
}
log("class " + clsName + " is loaded");
defaultSocketFactory = (SSLSocketFactory)cls.newInstance();
log("instantiated an instance of class " + clsName);
if (defaultSocketFactory != null) {
return defaultSocketFactory;
}
} catch (Exception e) {
log("SSLSocketFactory instantiation failed: " + e.toString());
}
}
// Android-changed: Allow for {@code null} SSLContext.getDefault.
try {
SSLContext context = SSLContext.getDefault();
if (context != null) {
defaultSocketFactory = context.getSocketFactory();
}
} catch (NoSuchAlgorithmException e) {
}
if (defaultSocketFactory == null) {
defaultSocketFactory = new DefaultSSLSocketFactory(new IllegalStateException("No factory found."));
}
return defaultSocketFactory;
}
static String getSecurityProperty(final String name) {
return AccessController.doPrivileged(new PrivilegedAction