/offsync/src/main/scala/org/offsync/forjava/NioServer.java

http://offsync.googlecode.com/ · Java · 209 lines · 136 code · 39 blank · 34 comment · 17 complexity · 06a5c98eac03ed91319a67547cc5481f MD5 · raw file

  1. package org.offsync.forjava;
  2. import java.io.IOException;
  3. import java.net.InetAddress;
  4. import java.net.InetSocketAddress;
  5. import java.nio.ByteBuffer;
  6. import java.nio.channels.SelectionKey;
  7. import java.nio.channels.Selector;
  8. import java.nio.channels.ServerSocketChannel;
  9. import java.nio.channels.SocketChannel;
  10. import java.nio.channels.spi.SelectorProvider;
  11. import java.util.*;
  12. public class NioServer implements Runnable {
  13. // The host:port combination to listen on
  14. private InetAddress hostAddress;
  15. private int port;
  16. // The channel on which we'll accept connections
  17. private ServerSocketChannel serverChannel;
  18. // The selector we'll be monitoring
  19. private Selector selector;
  20. // The buffer into which we'll read data when it's available
  21. private ByteBuffer readBuffer = ByteBuffer.allocate(8192);
  22. private EchoWorker worker;
  23. // A list of PendingChange instances
  24. private List<ChangeRequest> pendingChanges = new LinkedList<ChangeRequest>();
  25. // Maps a SocketChannel to a list of ByteBuffer instances
  26. private Map<SocketChannel, List<ByteBuffer>> pendingData = new HashMap<SocketChannel, List<ByteBuffer>>();
  27. public NioServer(InetAddress hostAddress, int port, EchoWorker worker) throws IOException {
  28. this.hostAddress = hostAddress;
  29. this.port = port;
  30. this.selector = this.initSelector();
  31. this.worker = worker;
  32. }
  33. public void send(SocketChannel socket, byte[] data) {
  34. synchronized (this.pendingChanges) {
  35. // Indicate we want the interest ops set changed
  36. this.pendingChanges.add(new ChangeRequest(socket, ChangeRequest.CHANGEOPS, SelectionKey.OP_WRITE));
  37. // And queue the data we want written
  38. synchronized (this.pendingData) {
  39. List<ByteBuffer> queue = this.pendingData.get(socket);
  40. if (queue == null) {
  41. queue = new ArrayList<ByteBuffer>();
  42. this.pendingData.put(socket, queue);
  43. }
  44. queue.add(ByteBuffer.wrap(data));
  45. }
  46. }
  47. // Finally, wake up our selecting thread so it can make the required changes
  48. this.selector.wakeup();
  49. }
  50. public void run() {
  51. while (true) {
  52. try {
  53. // Process any pending changes
  54. synchronized (this.pendingChanges) {
  55. Iterator<ChangeRequest> changes = this.pendingChanges.iterator();
  56. while (changes.hasNext()) {
  57. ChangeRequest change = (ChangeRequest) changes.next();
  58. switch (change.type) {
  59. case ChangeRequest.CHANGEOPS:
  60. SelectionKey key = change.socket.keyFor(this.selector);
  61. key.interestOps(change.ops);
  62. }
  63. }
  64. this.pendingChanges.clear();
  65. }
  66. // Wait for an event one of the registered channels
  67. this.selector.select();
  68. // Iterate over the set of keys for which events are available
  69. Iterator<?> selectedKeys = this.selector.selectedKeys().iterator();
  70. while (selectedKeys.hasNext()) {
  71. SelectionKey key = (SelectionKey) selectedKeys.next();
  72. selectedKeys.remove();
  73. if (!key.isValid()) {
  74. continue;
  75. }
  76. // Check what event is available and deal with it
  77. if (key.isAcceptable()) {
  78. this.accept(key);
  79. } else if (key.isReadable()) {
  80. this.read(key);
  81. } else if (key.isWritable()) {
  82. this.write(key);
  83. }
  84. }
  85. } catch (Exception e) {
  86. e.printStackTrace();
  87. }
  88. }
  89. }
  90. private void accept(SelectionKey key) throws IOException {
  91. // For an accept to be pending the channel must be a server socket channel.
  92. ServerSocketChannel serverSocketChannel = (ServerSocketChannel) key.channel();
  93. // Accept the connection and make it non-blocking
  94. SocketChannel socketChannel = serverSocketChannel.accept();
  95. socketChannel.configureBlocking(false);
  96. // Register the new SocketChannel with our Selector, indicating
  97. // we'd like to be notified when there's data waiting to be read
  98. socketChannel.register(this.selector, SelectionKey.OP_READ);
  99. }
  100. private void read(SelectionKey key) throws IOException {
  101. SocketChannel socketChannel = (SocketChannel) key.channel();
  102. // Clear out our read buffer so it's ready for new data
  103. this.readBuffer.clear();
  104. // Attempt to read off the channel
  105. int numRead;
  106. try {
  107. numRead = socketChannel.read(this.readBuffer);
  108. } catch (IOException e) {
  109. // The remote forcibly closed the connection, cancel
  110. // the selection key and close the channel.
  111. key.cancel();
  112. socketChannel.close();
  113. return;
  114. }
  115. if (numRead == -1) {
  116. // Remote entity shut the socket down cleanly. Do the
  117. // same from our end and cancel the channel.
  118. key.channel().close();
  119. key.cancel();
  120. return;
  121. }
  122. // Hand the data off to our worker thread
  123. this.worker.processData(this, socketChannel, this.readBuffer.array(), numRead);
  124. }
  125. private void write(SelectionKey key) throws IOException {
  126. SocketChannel socketChannel = (SocketChannel) key.channel();
  127. synchronized (this.pendingData) {
  128. List<ByteBuffer> queue = this.pendingData.get(socketChannel);
  129. // Write until there's not more data ...
  130. while (!queue.isEmpty()) {
  131. ByteBuffer buf = (ByteBuffer) queue.get(0);
  132. socketChannel.write(buf);
  133. if (buf.remaining() > 0) {
  134. // ... or the socket's buffer fills up
  135. break;
  136. }
  137. queue.remove(0);
  138. }
  139. if (queue.isEmpty()) {
  140. // We wrote away all data, so we're no longer interested
  141. // in writing on this socket. Switch back to waiting for
  142. // data.
  143. key.interestOps(SelectionKey.OP_READ);
  144. }
  145. }
  146. }
  147. private Selector initSelector() throws IOException {
  148. // Create a new selector
  149. Selector socketSelector = SelectorProvider.provider().openSelector();
  150. // Create a new non-blocking server socket channel
  151. this.serverChannel = ServerSocketChannel.open();
  152. serverChannel.configureBlocking(false);
  153. // Bind the server socket to the specified address and port
  154. InetSocketAddress isa = new InetSocketAddress(this.hostAddress, this.port);
  155. serverChannel.socket().bind(isa);
  156. // Register the server socket channel, indicating an interest in
  157. // accepting new connections
  158. serverChannel.register(socketSelector, SelectionKey.OP_ACCEPT);
  159. return socketSelector;
  160. }
  161. public static void main(String[] args) {
  162. try {
  163. EchoWorker worker = new EchoWorker();
  164. new Thread(worker).start();
  165. new Thread(new NioServer(null, 9876, worker)).start();
  166. } catch (IOException e) {
  167. e.printStackTrace();
  168. }
  169. }
  170. }