Mark Slee | 83c52a8 | 2006-06-07 06:51:18 +0000 | [diff] [blame] | 1 | package com.facebook.thrift.transport; |
| 2 | |
| 3 | import java.io.IOException; |
Mark Slee | ffcddd6 | 2006-09-06 20:37:03 +0000 | [diff] [blame] | 4 | import java.net.InetSocketAddress; |
Mark Slee | 83c52a8 | 2006-06-07 06:51:18 +0000 | [diff] [blame] | 5 | import java.net.ServerSocket; |
| 6 | import java.net.Socket; |
Mark Slee | 5bcde6e | 2006-09-27 17:50:32 +0000 | [diff] [blame] | 7 | import java.net.SocketException; |
Mark Slee | 83c52a8 | 2006-06-07 06:51:18 +0000 | [diff] [blame] | 8 | |
| 9 | /** |
| 10 | * Wrapper around ServerSocket for Thrift. |
| 11 | * |
| 12 | * @author Mark Slee <mcslee@facebook.com> |
| 13 | */ |
| 14 | public class TServerSocket extends TServerTransport { |
| 15 | |
Mark Slee | 5bcde6e | 2006-09-27 17:50:32 +0000 | [diff] [blame] | 16 | /** |
| 17 | * Underlying serversocket object |
| 18 | */ |
Mark Slee | ffcddd6 | 2006-09-06 20:37:03 +0000 | [diff] [blame] | 19 | private ServerSocket serverSocket_ = null; |
Mark Slee | 5bcde6e | 2006-09-27 17:50:32 +0000 | [diff] [blame] | 20 | |
| 21 | /** |
| 22 | * Port to listen on |
| 23 | */ |
Mark Slee | ffcddd6 | 2006-09-06 20:37:03 +0000 | [diff] [blame] | 24 | private int port_ = 0; |
| 25 | |
Mark Slee | 5bcde6e | 2006-09-27 17:50:32 +0000 | [diff] [blame] | 26 | /** |
| 27 | * Timeout for client sockets from accept |
| 28 | */ |
| 29 | private int clientTimeout_ = 0; |
| 30 | |
| 31 | /** |
| 32 | * Creates a server socket from underlying socket object |
| 33 | */ |
Mark Slee | 83c52a8 | 2006-06-07 06:51:18 +0000 | [diff] [blame] | 34 | public TServerSocket(ServerSocket serverSocket) { |
Mark Slee | 5bcde6e | 2006-09-27 17:50:32 +0000 | [diff] [blame] | 35 | this(serverSocket, 0); |
Mark Slee | 83c52a8 | 2006-06-07 06:51:18 +0000 | [diff] [blame] | 36 | } |
| 37 | |
Mark Slee | 5bcde6e | 2006-09-27 17:50:32 +0000 | [diff] [blame] | 38 | /** |
| 39 | * Creates a server socket from underlying socket object |
| 40 | */ |
| 41 | public TServerSocket(ServerSocket serverSocket, int clientTimeout) { |
| 42 | serverSocket_ = serverSocket; |
| 43 | clientTimeout_ = clientTimeout; |
| 44 | } |
| 45 | |
| 46 | /** |
| 47 | * Creates just a port listening server socket |
| 48 | */ |
Mark Slee | ffcddd6 | 2006-09-06 20:37:03 +0000 | [diff] [blame] | 49 | public TServerSocket(int port) throws TTransportException { |
Mark Slee | 5bcde6e | 2006-09-27 17:50:32 +0000 | [diff] [blame] | 50 | this(port, 0); |
| 51 | } |
| 52 | |
| 53 | /** |
| 54 | * Creates just a port listening server socket |
| 55 | */ |
| 56 | public TServerSocket(int port, int clientTimeout) throws TTransportException { |
Mark Slee | ffcddd6 | 2006-09-06 20:37:03 +0000 | [diff] [blame] | 57 | port_ = port; |
Mark Slee | 5bcde6e | 2006-09-27 17:50:32 +0000 | [diff] [blame] | 58 | clientTimeout_ = clientTimeout; |
Mark Slee | ffcddd6 | 2006-09-06 20:37:03 +0000 | [diff] [blame] | 59 | try { |
Mark Slee | 5bcde6e | 2006-09-27 17:50:32 +0000 | [diff] [blame] | 60 | // Make server socket |
Mark Slee | ffcddd6 | 2006-09-06 20:37:03 +0000 | [diff] [blame] | 61 | serverSocket_ = new ServerSocket(); |
Mark Slee | 5bcde6e | 2006-09-27 17:50:32 +0000 | [diff] [blame] | 62 | // Prevent 2MSL delay problem on server restarts |
Mark Slee | ffcddd6 | 2006-09-06 20:37:03 +0000 | [diff] [blame] | 63 | serverSocket_.setReuseAddress(true); |
Mark Slee | 5bcde6e | 2006-09-27 17:50:32 +0000 | [diff] [blame] | 64 | // Bind to listening port |
Mark Slee | ffcddd6 | 2006-09-06 20:37:03 +0000 | [diff] [blame] | 65 | serverSocket_.bind(new InetSocketAddress(port_)); |
| 66 | } catch (IOException ioe) { |
| 67 | serverSocket_ = null; |
| 68 | throw new TTransportException("Could not create ServerSocket on port " + port + "."); |
| 69 | } |
| 70 | } |
| 71 | |
Mark Slee | 5bcde6e | 2006-09-27 17:50:32 +0000 | [diff] [blame] | 72 | public void listen() throws TTransportException { |
| 73 | // Make sure not to block on accept |
| 74 | if (serverSocket_ != null) { |
| 75 | try { |
| 76 | serverSocket_.setSoTimeout(0); |
| 77 | } catch (SocketException sx) { |
| 78 | sx.printStackTrace(); |
| 79 | } |
| 80 | } |
| 81 | } |
Mark Slee | 83c52a8 | 2006-06-07 06:51:18 +0000 | [diff] [blame] | 82 | |
| 83 | protected TSocket acceptImpl() throws TTransportException { |
| 84 | if (serverSocket_ == null) { |
| 85 | throw new TTransportException("No underlying server socket."); |
| 86 | } |
| 87 | try { |
| 88 | Socket result = serverSocket_.accept(); |
Mark Slee | 5bcde6e | 2006-09-27 17:50:32 +0000 | [diff] [blame] | 89 | TSocket result2 = new TSocket(result); |
| 90 | result2.setTimeout(clientTimeout_); |
| 91 | return result2; |
Mark Slee | 83c52a8 | 2006-06-07 06:51:18 +0000 | [diff] [blame] | 92 | } catch (IOException iox) { |
| 93 | throw new TTransportException(iox); |
| 94 | } |
| 95 | } |
| 96 | |
| 97 | public void close() { |
| 98 | if (serverSocket_ != null) { |
| 99 | try { |
| 100 | serverSocket_.close(); |
| 101 | } catch (IOException iox) { |
| 102 | System.err.println("WARNING: Could not close server socket: " + |
| 103 | iox.getMessage()); |
| 104 | } |
| 105 | serverSocket_ = null; |
| 106 | } |
| 107 | } |
| 108 | |
| 109 | } |