/* Copyright (C) 2002-2007 MySQL AB This program is free software; you can redistribute it and/or modify it under the terms of version 2 of the GNU General Public License as published by the Free Software Foundation. There are special exceptions to the terms and conditions of the GPL as it is applied to this software. View the full text of the exception in file EXCEPTIONS-CONNECTOR-J in the directory of this software distribution. This program 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 for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ package com.mysql.management.util; import java.io.IOException; import java.lang.reflect.Constructor; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; import java.net.InetAddress; import java.net.Socket; import java.net.SocketException; import java.util.Properties; import com.mysql.jdbc.SocketFactory; /** * Socket factory for vanilla TCP/IP sockets (the standard) * * @author Mark Matthews */ @SuppressWarnings({ "rawtypes", "unchecked" }) public class PatchedStandardSocketFactory implements SocketFactory { // ~ Instance fields // -------------------------------------------------------- /** The hostname to connect to */ protected String host = null; /** The port number to connect to */ protected int port = 3306; /** The underlying TCP/IP socket to use */ protected Socket rawSocket = null; // ~ Methods // ---------------------------------------------------------------- /** * Called by the driver after issuing the MySQL protocol handshake and * reading the results of the handshake. * * @throws SocketException * if a socket error occurs * @throws IOException * if an I/O error occurs * * @return The socket to use after the handshake */ @Override public Socket afterHandshake() throws SocketException, IOException { return this.rawSocket; } /** * Called by the driver before issuing the MySQL protocol handshake. Should * return the socket instance that should be used during the handshake. * * @throws SocketException * if a socket error occurs * @throws IOException * if an I/O error occurs * * @return the socket to use before the handshake */ @Override public Socket beforeHandshake() throws SocketException, IOException { return this.rawSocket; } /** * @see com.mysql.jdbc.SocketFactory#createSocket(Properties) */ @Override public Socket connect(String hostname, int portNumber, Properties props) throws SocketException, IOException { if (props != null) { this.host = hostname; this.port = portNumber; Method connectWithTimeoutMethod = null; Method socketBindMethod = null; Class socketAddressClass = null; String localSocketHostname = props .getProperty("localSocketAddress"); String connectTimeoutStr = props.getProperty("connectTimeout"); int connectTimeout = 0; boolean wantsTimeout = (connectTimeoutStr != null && connectTimeoutStr.length() > 0 && !connectTimeoutStr .equals("0")); boolean wantsLocalBind = (localSocketHostname != null && localSocketHostname .length() > 0); if (wantsTimeout || wantsLocalBind) { if (connectTimeoutStr != null) { try { connectTimeout = Integer.parseInt(connectTimeoutStr); } catch (NumberFormatException nfe) { throw new SocketException("Illegal value '" + connectTimeoutStr + "' for connectTimeout"); } } try { // Have to do this with reflection, otherwise older JVMs // croak socketAddressClass = Class .forName("java.net.SocketAddress"); connectWithTimeoutMethod = Socket.class.getMethod( "connect", new Class[] { socketAddressClass, Integer.TYPE }); socketBindMethod = Socket.class.getMethod("bind", new Class[] { socketAddressClass }); } catch (NoClassDefFoundError noClassDefFound) { // ignore, we give a better error below if needed } catch (NoSuchMethodException noSuchMethodEx) { // ignore, we give a better error below if needed } catch (Throwable catchAll) { // ignore, we give a better error below if needed } if (wantsLocalBind && socketBindMethod == null) { throw new SocketException( "Can't specify \"localSocketAddress\" on JVMs older than 1.4"); } if (wantsTimeout && connectWithTimeoutMethod == null) { throw new SocketException( "Can't specify \"connectTimeout\" on JVMs older than 1.4"); } } if (this.host != null) { if (!(wantsLocalBind || wantsTimeout)) { InetAddress[] possibleAddresses = InetAddress .getAllByName(this.host); Throwable caughtWhileConnecting = null; // Need to loop through all possible addresses, in case // someone has IPV6 configured (SuSE, for example...) for (int i = 0; i < possibleAddresses.length; i++) { try { rawSocket = new Socket(possibleAddresses[i], port); break; } catch (Exception ex) { caughtWhileConnecting = ex; } } if (rawSocket == null) { unwrapExceptionToProperClassAndThrowIt(caughtWhileConnecting); } } else { // must explicitly state this due to classloader issues // when running on older JVMs :( try { InetAddress[] possibleAddresses = InetAddress .getAllByName(this.host); Throwable caughtWhileConnecting = null; Object localSockAddr = null; Class inetSocketAddressClass = null; Constructor addrConstructor = null; try { inetSocketAddressClass = Class .forName("java.net.InetSocketAddress"); addrConstructor = inetSocketAddressClass .getConstructor(new Class[] { InetAddress.class, Integer.TYPE }); if (wantsLocalBind) { localSockAddr = addrConstructor .newInstance(new Object[] { InetAddress .getByName(localSocketHostname), Integer.valueOf(0 /* * use ephemeral * port */) }); } } catch (Throwable ex) { unwrapExceptionToProperClassAndThrowIt(ex); } // Need to loop through all possible addresses, in case // someone has IPV6 configured (SuSE, for example...) for (int i = 0; i < possibleAddresses.length; i++) { try { rawSocket = new Socket(); Object sockAddr = addrConstructor .newInstance(new Object[] { possibleAddresses[i], Integer.valueOf(port) }); // bind to the local port, null is 'ok', it // means // use the ephemeral port socketBindMethod.invoke(rawSocket, new Object[] { localSockAddr }); connectWithTimeoutMethod.invoke(rawSocket, new Object[] { sockAddr, Integer.valueOf(connectTimeout) }); break; } catch (Exception ex) { rawSocket = null; caughtWhileConnecting = ex; } } if (rawSocket == null) { unwrapExceptionToProperClassAndThrowIt(caughtWhileConnecting); } } catch (Throwable t) { unwrapExceptionToProperClassAndThrowIt(t); } } try { this.rawSocket.setTcpNoDelay(true); } catch (Exception ex) { /* Ignore */ } return this.rawSocket; } } throw new SocketException("Unable to create socket"); } private void unwrapExceptionToProperClassAndThrowIt( Throwable caughtWhileConnecting) throws SocketException, IOException { if (caughtWhileConnecting instanceof InvocationTargetException) { // Replace it with the target, don't use 1.4 chaining as this still // needs to run on older VMs caughtWhileConnecting = ((InvocationTargetException) caughtWhileConnecting) .getTargetException(); } if (caughtWhileConnecting instanceof SocketException) { throw (SocketException) caughtWhileConnecting; } if (caughtWhileConnecting instanceof IOException) { throw (IOException) caughtWhileConnecting; } throw new SocketException(caughtWhileConnecting.toString()); } }