ensure protocols and ciphers are set on SSLSockets

Today, we simply return a SSLSocketFactory from a SSLContext, which provides
the default SSL configuration for sockets. This means that SSLv3 could still be
enabled on these sockets when running in an older JVM. This also means that
the ciphers and protocol settings are not honored for users of this socket factory,
which is currently the LDAP code.

This change returns a custom socket factory that delegates to the default socket
factory and sets the ciphers and protocols on the socket before returning the
socket.

Original commit: elastic/x-pack-elasticsearch@c4cfedfd51
This commit is contained in:
jaymode 2015-06-05 14:15:40 -04:00
parent 6f079dd2f2
commit b713d16803
3 changed files with 110 additions and 8 deletions

View File

@ -22,6 +22,9 @@ import org.elasticsearch.shield.ShieldSettingsException;
import javax.net.ssl.*; import javax.net.ssl.*;
import java.io.InputStream; import java.io.InputStream;
import java.nio.file.Files; import java.nio.file.Files;
import java.io.IOException;
import java.net.InetAddress;
import java.net.Socket;
import java.security.KeyStore; import java.security.KeyStore;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Arrays; import java.util.Arrays;
@ -56,7 +59,8 @@ public abstract class AbstractSSLService extends AbstractComponent {
* @return A SSLSocketFactory (for client-side SSL handshaking) * @return A SSLSocketFactory (for client-side SSL handshaking)
*/ */
public SSLSocketFactory sslSocketFactory() { public SSLSocketFactory sslSocketFactory() {
return sslContext(Settings.EMPTY).getSocketFactory(); SSLSocketFactory socketFactory = sslContext().getSocketFactory();
return new ShieldSSLSocketFactory(socketFactory, supportedProtocols(), supportedCiphers(socketFactory.getSupportedCipherSuites(), ciphers()));
} }
public String[] supportedProtocols() { public String[] supportedProtocols() {
@ -317,4 +321,77 @@ public abstract class AbstractSSLService extends AbstractComponent {
return result; return result;
} }
} }
/**
* This socket factory set the protocols and ciphers on each SSLSocket after it is created
*/
static class ShieldSSLSocketFactory extends SSLSocketFactory {
private final SSLSocketFactory delegate;
private final String[] supportedProtocols;
private final String[] ciphers;
ShieldSSLSocketFactory(SSLSocketFactory delegate, String[] supportedProtocols, String[] ciphers) {
this.delegate = delegate;
this.supportedProtocols = supportedProtocols;
this.ciphers = ciphers;
}
@Override
public String[] getDefaultCipherSuites() {
return ciphers;
}
@Override
public String[] getSupportedCipherSuites() {
return delegate.getSupportedCipherSuites();
}
@Override
public Socket createSocket() throws IOException {
SSLSocket sslSocket = (SSLSocket) delegate.createSocket();
configureSSLSocket(sslSocket);
return sslSocket;
}
@Override
public Socket createSocket(Socket socket, String host, int port, boolean autoClose) throws IOException {
SSLSocket sslSocket = (SSLSocket) delegate.createSocket(socket, host, port, autoClose);
configureSSLSocket(sslSocket);
return sslSocket;
}
@Override
public Socket createSocket(String host, int port) throws IOException {
SSLSocket sslSocket = (SSLSocket) delegate.createSocket(host, port);
configureSSLSocket(sslSocket);
return sslSocket;
}
@Override
public Socket createSocket(String host, int port, InetAddress localHost, int localPort) throws IOException {
SSLSocket sslSocket = (SSLSocket) delegate.createSocket(host, port, localHost, localPort);
configureSSLSocket(sslSocket);
return sslSocket;
}
@Override
public Socket createSocket(InetAddress host, int port) throws IOException {
SSLSocket sslSocket = (SSLSocket) delegate.createSocket(host, port);
configureSSLSocket(sslSocket);
return sslSocket;
}
@Override
public Socket createSocket(InetAddress address, int port, InetAddress localAddress, int localPort) throws IOException {
SSLSocket sslSocket = (SSLSocket) delegate.createSocket(address, port, localAddress, localPort);
configureSSLSocket(sslSocket);
return sslSocket;
}
private void configureSSLSocket(SSLSocket socket) {
socket.setEnabledProtocols(supportedProtocols);
socket.setEnabledCipherSuites(ciphers);
}
}
} }

View File

@ -17,10 +17,7 @@ import org.elasticsearch.test.junit.annotations.Network;
import org.junit.Before; import org.junit.Before;
import org.junit.Test; import org.junit.Test;
import javax.net.ssl.SSLContext; import javax.net.ssl.*;
import javax.net.ssl.SSLEngine;
import javax.net.ssl.SSLHandshakeException;
import javax.net.ssl.SSLSessionContext;
import java.nio.file.Path; import java.nio.file.Path;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Arrays; import java.util.Arrays;
@ -233,4 +230,19 @@ public class ClientSSLServiceTests extends ElasticsearchTestCase {
.build(), env); .build(), env);
sslService.createSSLEngine(); sslService.createSSLEngine();
} }
@Test
public void testThatSSLSocketFactoryHasProperCiphersAndProtocols() throws Exception {
ClientSSLService sslService = new ClientSSLService(settingsBuilder()
.put("shield.ssl.keystore.path", testclientStore)
.put("shield.ssl.keystore.password", "testclient")
.build(), env);
SSLSocketFactory factory = sslService.sslSocketFactory();
assertThat(factory.getDefaultCipherSuites(), is(sslService.ciphers()));
try (SSLSocket socket = (SSLSocket) factory.createSocket()) {
assertThat(socket.getEnabledCipherSuites(), is(sslService.ciphers()));
assertThat(socket.getEnabledProtocols(), is(sslService.supportedProtocols()));
}
}
} }

View File

@ -14,9 +14,7 @@ import org.elasticsearch.test.ElasticsearchTestCase;
import org.junit.Before; import org.junit.Before;
import org.junit.Test; import org.junit.Test;
import javax.net.ssl.SSLContext; import javax.net.ssl.*;
import javax.net.ssl.SSLEngine;
import javax.net.ssl.SSLSessionContext;
import java.nio.file.Path; import java.nio.file.Path;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Arrays; import java.util.Arrays;
@ -197,4 +195,19 @@ public class ServerSSLServiceTests extends ElasticsearchTestCase {
.build(), settingsFilter, env); .build(), settingsFilter, env);
sslService.createSSLEngine(); sslService.createSSLEngine();
} }
@Test
public void testThatSSLSocketFactoryHasProperCiphersAndProtocols() throws Exception {
ServerSSLService sslService = new ServerSSLService(settingsBuilder()
.put("shield.ssl.keystore.path", testnodeStore)
.put("shield.ssl.keystore.password", "testnode")
.build(), settingsFilter, env);
SSLSocketFactory factory = sslService.sslSocketFactory();
assertThat(factory.getDefaultCipherSuites(), is(sslService.ciphers()));
try (SSLSocket socket = (SSLSocket) factory.createSocket()) {
assertThat(socket.getEnabledCipherSuites(), is(sslService.ciphers()));
assertThat(socket.getEnabledProtocols(), is(sslService.supportedProtocols()));
}
}
} }