/vt-ldap/branches/vt-ldap-3/src/main/java/edu/vt/middleware/ldap/ssl/AbstractTLSSocketFactory.java

http://vt-middleware.googlecode.com/ · Java · 371 lines · 164 code · 41 blank · 166 comment · 23 complexity · d180192e5ed5c14cee705bd5748957a7 MD5 · raw file

  1. /*
  2. $Id$
  3. Copyright (C) 2003-2010 Virginia Tech.
  4. All rights reserved.
  5. SEE LICENSE FOR MORE INFORMATION
  6. Author: Middleware Services
  7. Email: middleware@vt.edu
  8. Version: $Revision$
  9. Updated: $Date$
  10. */
  11. package edu.vt.middleware.ldap.ssl;
  12. import java.io.IOException;
  13. import java.net.InetAddress;
  14. import java.net.Socket;
  15. import java.security.GeneralSecurityException;
  16. import javax.net.ssl.HostnameVerifier;
  17. import javax.net.ssl.SSLPeerUnverifiedException;
  18. import javax.net.ssl.SSLSocket;
  19. import javax.net.ssl.SSLSocketFactory;
  20. /**
  21. * Provides common implementation for <code>TLSSocketFactory</code>.
  22. *
  23. * @author Middleware Services
  24. * @version $Revision: 1106 $ $Date: 2010-01-29 23:34:13 -0500 (Fri, 29 Jan 2010) $
  25. */
  26. public abstract class AbstractTLSSocketFactory extends SSLSocketFactory
  27. {
  28. /** Default SSL protocol, value is {@value}. */
  29. public static final String DEFAULT_PROTOCOL = "TLS";
  30. /** SSLSocketFactory used for creating SSL sockets. */
  31. protected SSLSocketFactory factory;
  32. /** Hostname verifier for this socket factory. */
  33. protected HostnameVerifier hostnameVerifier;
  34. /** Enabled cipher suites. */
  35. protected String[] cipherSuites;
  36. /** Enabled protocol versions. */
  37. protected String[] protocols;
  38. /**
  39. * Prepares this socket factory for use. Must be called before factory can be
  40. * used.
  41. *
  42. * @throws GeneralSecurityException if the factory cannot be initialized
  43. */
  44. public abstract void initialize()
  45. throws GeneralSecurityException;
  46. /**
  47. * This returns the underlying <code>SSLSocketFactory</code> that this class
  48. * uses for creating SSL Sockets.
  49. *
  50. * @return <code>SSLSocketFactory</code>
  51. */
  52. public SSLSocketFactory getFactory()
  53. {
  54. return this.factory;
  55. }
  56. /**
  57. * Returns the hostname verifier to invoke when sockets are created.
  58. *
  59. * @return hostname verifier
  60. */
  61. public HostnameVerifier getHostnameVerifier()
  62. {
  63. return hostnameVerifier;
  64. }
  65. /**
  66. * Sets the hostname verifier to invoke when sockets are created.
  67. *
  68. * @param verifier for SSL hostnames
  69. */
  70. public void setHostnameVerifier(final HostnameVerifier verifier)
  71. {
  72. hostnameVerifier = verifier;
  73. }
  74. /**
  75. * This returns the names of the SSL cipher suites which are currently enabled
  76. * for use on sockets created by this factory. A null value indicates that no
  77. * specific cipher suites have been enabled and that the default suites are in
  78. * use.
  79. *
  80. * @return <code>String[]</code> of cipher suites
  81. */
  82. public String[] getEnabledCipherSuites()
  83. {
  84. return this.cipherSuites;
  85. }
  86. /**
  87. * Sets the cipher suites enabled for use on sockets created by this factory.
  88. * See {@link javax.net.ssl.SSLSocket#setEnabledCipherSuites(String[])}.
  89. *
  90. * @param s <code>String[]</code> of cipher suites
  91. */
  92. public void setEnabledCipherSuites(final String[] s)
  93. {
  94. this.cipherSuites = s;
  95. }
  96. /**
  97. * This returns the names of the protocol versions which are currently enabled
  98. * for use on sockets created by this factory. A null value indicates that no
  99. * specific protocols have been enabled and that the default protocols are in
  100. * use.
  101. *
  102. * @return <code>String[]</code> of protocols
  103. */
  104. public String[] getEnabledProtocols()
  105. {
  106. return this.protocols;
  107. }
  108. /**
  109. * Sets the protocol versions enabled for use on sockets created by this
  110. * factory. See {@link javax.net.ssl.SSLSocket#setEnabledProtocols(String[])}.
  111. *
  112. * @param s <code>String[]</code> of cipher suites
  113. */
  114. public void setEnabledProtocols(final String[] s)
  115. {
  116. this.protocols = s;
  117. }
  118. /**
  119. * Initializes the supplied socket for use.
  120. *
  121. * @param s <code>SSLSocket</code> to initialize
  122. *
  123. * @return <code>SSLSocket</code>
  124. *
  125. * @throws IOException if an I/O error occurs when initializing the socket
  126. */
  127. protected SSLSocket initSSLSocket(final SSLSocket s)
  128. throws IOException
  129. {
  130. if (this.cipherSuites != null) {
  131. s.setEnabledCipherSuites(this.cipherSuites);
  132. }
  133. if (this.protocols != null) {
  134. s.setEnabledProtocols(this.protocols);
  135. }
  136. if (hostnameVerifier != null) {
  137. // calling getSession() will initiate the handshake if necessary
  138. final String hostname = s.getSession().getPeerHost();
  139. if (!hostnameVerifier.verify(hostname, s.getSession())) {
  140. s.close();
  141. s.getSession().invalidate();
  142. throw new SSLPeerUnverifiedException(
  143. String.format(
  144. "Hostname '%s' does not match the hostname in the server's " +
  145. "certificate", hostname));
  146. }
  147. }
  148. return s;
  149. }
  150. /**
  151. * This returns a socket layered over an existing socket connected to the
  152. * named host, at the given port.
  153. *
  154. * @param s <code>Socket</code> existing socket
  155. * @param host <code>String</code> server hostname
  156. * @param port <code>int</code> server port
  157. * @param autoClose <code>boolean</code> close the underlying socket when
  158. * this socket is closed
  159. *
  160. * @return <code>Socket</code> - connected to the specified host and port
  161. *
  162. * @throws IOException if an I/O error occurs when creating the socket
  163. */
  164. public Socket createSocket(
  165. final Socket s,
  166. final String host,
  167. final int port,
  168. final boolean autoClose)
  169. throws IOException
  170. {
  171. SSLSocket socket = null;
  172. if (this.factory != null) {
  173. socket = this.initSSLSocket(
  174. (SSLSocket) this.factory.createSocket(s, host, port, autoClose));
  175. }
  176. return socket;
  177. }
  178. /**
  179. * This creates an unconnected socket.
  180. *
  181. * @return <code>Socket</code> - unconnected socket
  182. *
  183. * @throws IOException if an I/O error occurs when creating the socket
  184. */
  185. public Socket createSocket()
  186. throws IOException
  187. {
  188. SSLSocket socket = null;
  189. if (this.factory != null) {
  190. socket = this.initSSLSocket((SSLSocket) this.factory.createSocket());
  191. }
  192. return socket;
  193. }
  194. /**
  195. * This creates a socket and connects it to the specified port number at the
  196. * specified address.
  197. *
  198. * @param host <code>InetAddress</code> server hostname
  199. * @param port <code>int</code> server port
  200. *
  201. * @return <code>Socket</code> - connected to the specified host and port
  202. *
  203. * @throws IOException if an I/O error occurs when creating the socket
  204. */
  205. public Socket createSocket(final InetAddress host, final int port)
  206. throws IOException
  207. {
  208. SSLSocket socket = null;
  209. if (this.factory != null) {
  210. socket = this.initSSLSocket(
  211. (SSLSocket) this.factory.createSocket(host, port));
  212. }
  213. return socket;
  214. }
  215. /**
  216. * This creates a socket and connect it to the specified port number at the
  217. * specified address. The socket will also be bound to the supplied local
  218. * address and port.
  219. *
  220. * @param address <code>InetAddress</code> server hostname
  221. * @param port <code>int</code> server port
  222. * @param localAddress <code>InetAddress</code> client hostname
  223. * @param localPort <code>int</code> client port
  224. *
  225. * @return <code>Socket</code> - connected to the specified host and port
  226. *
  227. * @throws IOException if an I/O error occurs when creating the socket
  228. */
  229. public Socket createSocket(
  230. final InetAddress address,
  231. final int port,
  232. final InetAddress localAddress,
  233. final int localPort)
  234. throws IOException
  235. {
  236. SSLSocket socket = null;
  237. if (this.factory != null) {
  238. socket = this.initSSLSocket(
  239. (SSLSocket) this.factory.createSocket(
  240. address,
  241. port,
  242. localAddress,
  243. localPort));
  244. }
  245. return socket;
  246. }
  247. /**
  248. * This creates a socket and connects it to the specified port number at the
  249. * specified address.
  250. *
  251. * @param host <code>String</code> server hostname
  252. * @param port <code>int</code> server port
  253. *
  254. * @return <code>Socket</code> - connected to the specified host and port
  255. *
  256. * @throws IOException if an I/O error occurs when creating the socket
  257. */
  258. public Socket createSocket(final String host, final int port)
  259. throws IOException
  260. {
  261. SSLSocket socket = null;
  262. if (this.factory != null) {
  263. socket = this.initSSLSocket(
  264. (SSLSocket) this.factory.createSocket(host, port));
  265. }
  266. return socket;
  267. }
  268. /**
  269. * This creates a socket and connect it to the specified port number at the
  270. * specified address. The socket will also be bound to the supplied local
  271. * address and port.
  272. *
  273. * @param host <code>String</code> server hostname
  274. * @param port <code>int</code> server port
  275. * @param localHost <code>InetAddress</code> client hostname
  276. * @param localPort <code>int</code> client port
  277. *
  278. * @return <code>Socket</code> - connected to the specified host and port
  279. *
  280. * @throws IOException if an I/O error occurs when creating the socket
  281. */
  282. public Socket createSocket(
  283. final String host,
  284. final int port,
  285. final InetAddress localHost,
  286. final int localPort)
  287. throws IOException
  288. {
  289. SSLSocket socket = null;
  290. if (this.factory != null) {
  291. socket = this.initSSLSocket(
  292. (SSLSocket) this.factory.createSocket(
  293. host,
  294. port,
  295. localHost,
  296. localPort));
  297. }
  298. return socket;
  299. }
  300. /**
  301. * This returns the list of cipher suites which are enabled by default.
  302. *
  303. * @return <code>String[]</code> - array of the cipher suites
  304. */
  305. public String[] getDefaultCipherSuites()
  306. {
  307. String[] ciphers = null;
  308. if (this.factory != null) {
  309. ciphers = this.factory.getDefaultCipherSuites();
  310. }
  311. return ciphers;
  312. }
  313. /**
  314. * This returns the names of the cipher suites which could be enabled for use
  315. * on an SSL connection.
  316. *
  317. * @return <code>String[]</code> - array of the cipher suites
  318. */
  319. public String[] getSupportedCipherSuites()
  320. {
  321. String[] ciphers = null;
  322. if (this.factory != null) {
  323. ciphers = this.factory.getSupportedCipherSuites();
  324. }
  325. return ciphers;
  326. }
  327. }