/src/main/java/org/dynmap/web/HttpServer.java

https://github.com/flames/dynmap · Java · 192 lines · 168 code · 19 blank · 5 comment · 27 complexity · 78e9049949383683d84650c540bf65e0 MD5 · raw file

  1. package org.dynmap.web;
  2. import java.io.BufferedReader;
  3. import java.io.File;
  4. import java.io.FileReader;
  5. import java.io.IOException;
  6. import java.net.InetAddress;
  7. import java.net.InetSocketAddress;
  8. import java.net.ServerSocket;
  9. import java.net.Socket;
  10. import java.net.SocketAddress;
  11. import java.util.Collections;
  12. import java.util.HashSet;
  13. import java.util.IdentityHashMap;
  14. import java.util.SortedMap;
  15. import java.util.TreeMap;
  16. import java.util.logging.Logger;
  17. import org.dynmap.Log;
  18. public class HttpServer extends Thread {
  19. protected static final Logger log = Logger.getLogger("Minecraft");
  20. private ServerSocket sock = null;
  21. private Thread listeningThread;
  22. private InetAddress bindAddress;
  23. private int port;
  24. private boolean check_banned_ips;
  25. private int max_sessions;
  26. public SortedMap<String, HttpHandler> handlers = new TreeMap<String, HttpHandler>(Collections.reverseOrder());
  27. private Object lock = new Object();
  28. private HashSet<HttpServerConnection> active_connections = new HashSet<HttpServerConnection>();
  29. private HashSet<HttpServerConnection> keepalive_connections = new HashSet<HttpServerConnection>();
  30. public HttpServer(InetAddress bindAddress, int port, boolean check_banned_ips, int max_sessions) {
  31. this.bindAddress = bindAddress;
  32. this.port = port;
  33. this.check_banned_ips = check_banned_ips;
  34. this.max_sessions = max_sessions;
  35. }
  36. public InetAddress getAddress() {
  37. return bindAddress;
  38. }
  39. public int getPort() {
  40. return port;
  41. }
  42. public void startServer() throws IOException {
  43. sock = new ServerSocket(port, 50, bindAddress); /* 5 too low - more than a couple users during render will get connect errors on some tile loads */
  44. listeningThread = this;
  45. start();
  46. Log.info("Dynmap WebServer started on " + bindAddress + ":" + port);
  47. }
  48. public void run() {
  49. try {
  50. ServerSocket s = sock;
  51. while (listeningThread == Thread.currentThread()) {
  52. try {
  53. Socket socket = s.accept();
  54. if(checkForBannedIp(socket.getRemoteSocketAddress())) {
  55. try { socket.close(); } catch (IOException iox) {}
  56. socket = null;
  57. }
  58. HttpServerConnection requestThread = new HttpServerConnection(socket, this);
  59. synchronized(lock) {
  60. active_connections.add(requestThread);
  61. requestThread.start();
  62. /* If we're at limit, wait here until we're free to accept another */
  63. while((listeningThread == Thread.currentThread()) &&
  64. (active_connections.size() >= max_sessions)) {
  65. lock.wait(500);
  66. }
  67. }
  68. } catch (IOException e) {
  69. if(listeningThread != null) /* Only report this if we didn't initiate the shutdown */
  70. Log.info("map WebServer.run() stops with IOException");
  71. break;
  72. }
  73. }
  74. Log.info("Webserver shut down.");
  75. } catch (Exception ex) {
  76. Log.severe("Exception on WebServer-thread", ex);
  77. }
  78. }
  79. public void shutdown() {
  80. Log.info("Shutting down webserver...");
  81. listeningThread = null;
  82. try {
  83. if (sock != null) {
  84. sock.close();
  85. sock = null;
  86. }
  87. /* And kill off the active connections */
  88. HashSet<HttpServerConnection> sc;
  89. synchronized(lock) {
  90. sc = new HashSet<HttpServerConnection>(active_connections);
  91. }
  92. for(HttpServerConnection c : sc) {
  93. c.shutdownConnection();
  94. }
  95. } catch (IOException e) {
  96. Log.warning("Exception while closing socket for webserver shutdown", e);
  97. }
  98. }
  99. public boolean canKeepAlive(HttpServerConnection c) {
  100. synchronized(lock) {
  101. /* If less than half of our limit are keep-alive, approve */
  102. if(keepalive_connections.size() < (max_sessions/2)) {
  103. keepalive_connections.add(c);
  104. return true;
  105. }
  106. }
  107. return false;
  108. }
  109. public void connectionEnded(HttpServerConnection c) {
  110. synchronized(lock) {
  111. active_connections.remove(c);
  112. keepalive_connections.remove(c);
  113. lock.notifyAll();
  114. }
  115. }
  116. private HashSet<String> banned_ips = new HashSet<String>();
  117. private HashSet<String> banned_ips_notified = new HashSet<String>();
  118. private long last_loaded = 0;
  119. private long lastmod = 0;
  120. private static final long BANNED_RELOAD_INTERVAL = 15000; /* Every 15 seconds */
  121. private void loadBannedIPs() {
  122. banned_ips.clear();
  123. banned_ips_notified.clear();
  124. File f = new File("banned-ips.txt");
  125. if(f.exists() == false)
  126. return;
  127. if(f.lastModified() == lastmod) {
  128. return;
  129. }
  130. lastmod = f.lastModified();
  131. BufferedReader rdr = null;
  132. try {
  133. rdr = new BufferedReader(new FileReader(f));
  134. String line;
  135. while((line = rdr.readLine()) != null) {
  136. line = line.trim().toLowerCase(); /* Trim it and case normalize it */
  137. if((line.length() == 0) || (line.charAt(0) == '#')) { /* Blank or comment? */
  138. continue;
  139. }
  140. banned_ips.add(line);
  141. }
  142. } catch (IOException iox) {
  143. Log.severe("Error reading banned-ips.txt!");
  144. } finally {
  145. if(rdr != null) {
  146. try { rdr.close(); } catch (IOException iox) {}
  147. rdr = null;
  148. }
  149. }
  150. }
  151. /* Return true if address is banned */
  152. public boolean checkForBannedIp(SocketAddress socketAddress) {
  153. if(!check_banned_ips)
  154. return false;
  155. long t = System.currentTimeMillis();
  156. if((t < last_loaded) || ((t-last_loaded) > BANNED_RELOAD_INTERVAL)) {
  157. loadBannedIPs();
  158. last_loaded = t;
  159. }
  160. /* Follow same technique as MC uses - toString the SocketAddress and clip out string between "/" and ":" */
  161. String ip = socketAddress.toString();
  162. ip = ip.substring(ip.indexOf("/") + 1);
  163. ip = ip.substring(0, ip.indexOf(":"));
  164. if(banned_ips.contains(ip)) {
  165. if(banned_ips_notified.contains(ip) == false) {
  166. Log.info("Rejected connection by banned IP address - " + socketAddress.toString());
  167. banned_ips_notified.add(ip);
  168. }
  169. return true;
  170. }
  171. return false;
  172. }
  173. }