/gralej/src/gralej/server/SocketServer.java

http://gralej.googlecode.com/ · Java · 238 lines · 131 code · 44 blank · 63 comment · 11 complexity · 3c59a3123cd195734ad72033a53fa3aa MD5 · raw file

  1. package gralej.server;
  2. import gralej.controller.StreamInfo;
  3. import gralej.util.Log;
  4. import java.io.BufferedInputStream;
  5. import java.io.IOException;
  6. import java.net.InetAddress;
  7. import java.net.ServerSocket;
  8. import java.net.Socket;
  9. import java.net.UnknownHostException;
  10. import java.util.Vector;
  11. /**
  12. * A multi-client (=threaded) grale server binding to a TCP/IP socket.
  13. *
  14. * @author Niels
  15. * @version $Id:SocketServer.java 18 2007-11-13 16:26:47Z niels@drni.de $
  16. */
  17. public class SocketServer extends ServerBaseImpl {
  18. private int port;
  19. private InetAddress bindIP;
  20. private ServerSocket socket;
  21. private ConnectionWaiter waiter;
  22. private Vector<ConnectionHandler> handlerList;
  23. /**
  24. * A helper class that waits for incoming connections as a separate thread.
  25. * This class will invoke a {@link ConnectionHandler} for each incoming
  26. * connection.
  27. */
  28. private class ConnectionWaiter extends Thread {
  29. private boolean shutdown_state = false;
  30. public ConnectionWaiter() {
  31. super();
  32. // informative thread name for debugging
  33. this.setName("ConnectionWaiter (bind: "
  34. + socket.getInetAddress().getHostAddress() + ":"
  35. + socket.getLocalPort() + ")");
  36. }
  37. /**
  38. * Run this ConnectionWaiter: This waits for incoming connections and
  39. * invokes new handler threads if required. Must not be run elsehwere
  40. * but in {@link SocketServer#startListening()}
  41. */
  42. public void run() {
  43. try {
  44. // server main loop
  45. while (! shutdown_state) {
  46. Socket clientSocket = socket.accept();
  47. new ConnectionHandler(clientSocket).start();
  48. }
  49. } catch (IOException e) {
  50. //e.printStackTrace();
  51. if (! shutdown_state) {
  52. Log.error("An exception "
  53. + "occured while waiting for incoming connections. "
  54. + "Server thread terminates now, restart the server "
  55. + "to regain networking functionality. ");
  56. } else {
  57. Log.debug(this.getName() +
  58. ": Caught exception during server shutdown, " +
  59. "this may be normal.");
  60. }
  61. }
  62. // if this was a shutdown, then it's finished now
  63. shutdown_state = false;
  64. }
  65. synchronized void shutdownWaiter() throws IOException {
  66. shutdown_state = true;
  67. socket.close();
  68. }
  69. }
  70. /**
  71. * A class resp. thread that handles an incoming connection and informs the
  72. * listeners of this {@link SocketServer}.
  73. */
  74. private class ConnectionHandler extends Thread {
  75. private Socket clientSocket;
  76. private boolean shutdown_state = false;
  77. public ConnectionHandler(Socket clientSocket) {
  78. super();
  79. this.clientSocket = clientSocket;
  80. // informative thread name for debugging
  81. this.setName("ConnectionHandler (remote: "
  82. + clientSocket.getInetAddress().getHostAddress() + ":"
  83. + clientSocket.getPort() + ")");
  84. }
  85. /**
  86. * Runs this ConnectionHandler: This will detect the input protocol and
  87. * hand over to the listeners of the {@link SocketServer} afterwards.
  88. */
  89. public void run() {
  90. // System.err.println("- SocketServer accepted new connection!");
  91. registerConnHandler(this);
  92. try {
  93. BufferedInputStream s = new BufferedInputStream(clientSocket
  94. .getInputStream());
  95. StreamInfo info = new StreamInfo(StreamProtocolMagic
  96. .stream2type(s));
  97. notifyListeners(s, info);
  98. } catch (IOException e) {
  99. // the remote host closed the connection before something
  100. // useful has happened, we can ignore this.
  101. //e.printStackTrace();
  102. if ( ! shutdown_state) {
  103. Log.debug(this.getName()
  104. + ": Remote closed connection "
  105. + "before sending something useful. Closing handler.");
  106. } else {
  107. Log.debug(this.getName() + ": " +
  108. "Caught exception during connection shutdown, " +
  109. "this may be normal.");
  110. }
  111. }
  112. removeConnHandler(this);
  113. }
  114. synchronized private void killConnection() throws IOException {
  115. shutdown_state = true;
  116. clientSocket.close();
  117. }
  118. }
  119. /**
  120. * Instantiates a new socket server listening to 127.0.0.1 (aka localhost)
  121. * only.
  122. *
  123. * @param port
  124. * the port to bind to on localhost.
  125. */
  126. public SocketServer(int port) {
  127. try {
  128. bindIP = InetAddress.getByName("127.0.0.1");
  129. } catch (UnknownHostException e) {
  130. // this should never ever happen
  131. throw new RuntimeException(e);
  132. }
  133. this.port = port;
  134. handlerList = new Vector<ConnectionHandler>();
  135. }
  136. private void registerConnHandler(ConnectionHandler c) {
  137. // vectors are synchronized, this should be thread-safe
  138. handlerList.add(c);
  139. }
  140. private void removeConnHandler(ConnectionHandler c) {
  141. // vectors are synchronized, this should be thread-safe
  142. handlerList.removeElement(c);
  143. }
  144. /**
  145. * A new socket server listening to a given IP address, allowing binding to
  146. * public specific network interfaces.
  147. *
  148. * @param port
  149. * the port to bind to.
  150. * @param bind
  151. * address of the interface to bind to.
  152. * @throws NotImplementedInServerException
  153. */
  154. public SocketServer(int port, InetAddress bind)
  155. throws NotImplementedInServerException {
  156. this.port = port;
  157. this.bindIP = bind;
  158. // TODO: binding to public interfaces without access control is disabled
  159. throw new NotImplementedInServerException();
  160. }
  161. /**
  162. * @see IGraleServer#startListening()
  163. */
  164. public void startListening() throws IOException {
  165. if ( waiter != null ) {
  166. return;
  167. }
  168. // open the port, this may go wrong
  169. socket = new ServerSocket(port, 0, bindIP);
  170. // run the server main loop thread
  171. waiter = new ConnectionWaiter();
  172. waiter.start();
  173. }
  174. public boolean isListening() {
  175. return ( waiter != null && socket.isBound() );
  176. }
  177. public void stopListening() throws IOException {
  178. waiter.shutdownWaiter();
  179. // hopefully the garbage collector will do its job now...
  180. waiter = null;
  181. }
  182. public void killActiveConnections() throws IOException {
  183. // copy vector because it will be modified by terminating
  184. // connections
  185. // clone manually to ensure the right outcome
  186. Vector<ConnectionHandler> handlers = new Vector<ConnectionHandler>();
  187. for ( ConnectionHandler c : handlerList ) {
  188. handlers.add(c);
  189. }
  190. for ( ConnectionHandler c : handlers) {
  191. c.killConnection();
  192. }
  193. }
  194. }