/rabbit4/src/rabbit/io/ConnectionHandler.java

# · Java · 314 lines · 243 code · 39 blank · 32 comment · 32 complexity · bf81bfc96ea055fd7eaf0820ad47359c MD5 · raw file

  1. package rabbit.io;
  2. import java.io.IOException;
  3. import java.net.InetAddress;
  4. import java.net.MalformedURLException;
  5. import java.net.URL;
  6. import java.util.ArrayList;
  7. import java.util.Collections;
  8. import java.util.List;
  9. import java.util.Map;
  10. import java.util.HashMap;
  11. import java.util.concurrent.ConcurrentHashMap;
  12. import java.util.logging.Level;
  13. import java.util.logging.Logger;
  14. import rabbit.http.HttpHeader;
  15. import rabbit.nio.NioHandler;
  16. import rabbit.nio.ReadHandler;
  17. import rabbit.util.Counter;
  18. import rabbit.util.SProperties;
  19. /** A class to handle the connections to the net.
  20. * Tries to reuse connections whenever possible.
  21. *
  22. * @author <a href="mailto:robo@khelekore.org">Robert Olofsson</a>
  23. */
  24. public class ConnectionHandler {
  25. // The logger to use
  26. private final Logger logger = Logger.getLogger (getClass ().getName ());
  27. // The counter to use.
  28. private final Counter counter;
  29. // The resolver to use
  30. private final Resolver resolver;
  31. // The available connections.
  32. private final Map<Address, List<WebConnection>> activeConnections;
  33. // The channels waiting for closing
  34. private final Map<WebConnection, CloseListener> wc2closer;
  35. // the keepalivetime.
  36. private long keepaliveTime = 1000;
  37. // should we use pipelining...
  38. private boolean usePipelining = true;
  39. // the nio handler
  40. private final NioHandler nioHandler;
  41. public ConnectionHandler (Counter counter, Resolver resolver,
  42. NioHandler nioHandler) {
  43. this.counter = counter;
  44. this.resolver = resolver;
  45. this.nioHandler = nioHandler;
  46. activeConnections = new HashMap<Address, List<WebConnection>> ();
  47. wc2closer = new ConcurrentHashMap<WebConnection, CloseListener> ();
  48. }
  49. /** Set the keep alive time for this handler.
  50. * @param milis the keep alive time in miliseconds.
  51. */
  52. public void setKeepaliveTime (long milis) {
  53. keepaliveTime = milis;
  54. }
  55. /** Get the current keep alive time.
  56. * @return the keep alive time in miliseconds.
  57. */
  58. public long getKeepaliveTime () {
  59. return keepaliveTime;
  60. }
  61. public Map<Address, List<WebConnection>> getActiveConnections () {
  62. return Collections.unmodifiableMap (activeConnections);
  63. }
  64. /** Get a WebConnection for the given header.
  65. * @param header the HttpHeader containing the URL to connect to.
  66. * @param wcl the Listener that wants the connection.
  67. */
  68. public void getConnection (final HttpHeader header,
  69. final WebConnectionListener wcl) {
  70. // TODO: should we use the Host: header if its available? probably...
  71. String requri = header.getRequestURI ();
  72. URL url = null;
  73. try {
  74. url = new URL (requri);
  75. } catch (MalformedURLException e) {
  76. wcl.failed (e);
  77. return;
  78. }
  79. int port = url.getPort () > 0 ? url.getPort () : 80;
  80. final int rport = resolver.getConnectPort (port);
  81. resolver.getInetAddress (url, new InetAddressListener () {
  82. public void lookupDone (InetAddress ia) {
  83. Address a = new Address (ia, rport);
  84. getConnection (header, wcl, a);
  85. }
  86. public void unknownHost (Exception e) {
  87. wcl.failed (e);
  88. }
  89. });
  90. }
  91. private void getConnection (HttpHeader header,
  92. WebConnectionListener wcl,
  93. Address a) {
  94. WebConnection wc = null;
  95. counter.inc ("WebConnections used");
  96. String method = header.getMethod ();
  97. if (method != null) {
  98. // since we should not retry POST (and other) we
  99. // have to get a fresh connection for them..
  100. method = method.trim ();
  101. if (!(method.equals ("GET") || method.equals ("HEAD"))) {
  102. wc = new WebConnection (a, counter);
  103. } else {
  104. wc = getPooledConnection (a, activeConnections);
  105. if (wc == null)
  106. wc = new WebConnection (a, counter);
  107. }
  108. try {
  109. wc.connect (nioHandler, wcl);
  110. } catch (IOException e) {
  111. wcl.failed (e);
  112. }
  113. } else {
  114. String err = "No method specified: " + header;
  115. wcl.failed (new IllegalArgumentException (err));
  116. }
  117. }
  118. private WebConnection
  119. getPooledConnection (Address a, Map<Address, List<WebConnection>> conns) {
  120. synchronized (conns) {
  121. List<WebConnection> pool = conns.get (a);
  122. if (pool != null) {
  123. if (pool.size () > 0) {
  124. WebConnection wc = pool.remove (pool.size () - 1);
  125. if (pool.isEmpty ())
  126. conns.remove (a);
  127. return unregister (wc);
  128. }
  129. }
  130. }
  131. return null;
  132. }
  133. private WebConnection unregister (WebConnection wc) {
  134. CloseListener closer = null;
  135. closer = wc2closer.remove (wc);
  136. if (closer != null)
  137. nioHandler.cancel (wc.getChannel (), closer);
  138. return wc;
  139. }
  140. private void removeFromPool (WebConnection wc,
  141. Map<Address, List<WebConnection>> conns) {
  142. synchronized (conns) {
  143. List<WebConnection> pool = conns.get (wc.getAddress ());
  144. if (pool != null) {
  145. pool.remove (wc);
  146. if (pool.isEmpty ())
  147. conns.remove (wc.getAddress ());
  148. }
  149. }
  150. }
  151. /** Return a WebConnection to the pool so that it may be reused.
  152. * @param wc the WebConnection to return.
  153. */
  154. public void releaseConnection (WebConnection wc) {
  155. counter.inc ("WebConnections released");
  156. if (!wc.getChannel ().isOpen ()) {
  157. return;
  158. }
  159. Address a = wc.getAddress ();
  160. if (!wc.getKeepalive ()) {
  161. closeWebConnection (wc);
  162. return;
  163. }
  164. synchronized (wc) {
  165. wc.setReleased ();
  166. }
  167. synchronized (activeConnections) {
  168. List<WebConnection> pool = activeConnections.get (a);
  169. if (pool == null) {
  170. pool = new ArrayList<WebConnection> ();
  171. activeConnections.put (a, pool);
  172. } else {
  173. if (pool.contains (wc)) {
  174. String err =
  175. "web connection already added to pool: " + wc;
  176. throw new IllegalStateException (err);
  177. }
  178. }
  179. try {
  180. pool.add (wc);
  181. CloseListener cl = new CloseListener (wc);
  182. wc2closer.put (wc, cl);
  183. cl.register ();
  184. } catch (IOException e) {
  185. logger.log (Level.WARNING,
  186. "Get IOException when setting up a CloseListener: ",
  187. e);
  188. closeWebConnection (wc);
  189. }
  190. }
  191. }
  192. private void closeWebConnection (WebConnection wc) {
  193. if (wc == null)
  194. return;
  195. if (!wc.getChannel ().isOpen ())
  196. return;
  197. try {
  198. wc.close ();
  199. } catch (IOException e) {
  200. logger.warning ("Failed to close WebConnection: " + wc);
  201. }
  202. }
  203. private class CloseListener implements ReadHandler {
  204. private WebConnection wc;
  205. private Long timeout;
  206. public CloseListener (WebConnection wc) throws IOException {
  207. this.wc = wc;
  208. }
  209. public void register () {
  210. timeout = nioHandler.getDefaultTimeout ();
  211. nioHandler.waitForRead (wc.getChannel (), this);
  212. }
  213. public void read () {
  214. closeChannel ();
  215. }
  216. public void closed () {
  217. closeChannel ();
  218. }
  219. public void timeout () {
  220. closeChannel ();
  221. }
  222. public Long getTimeout () {
  223. return timeout;
  224. }
  225. private void closeChannel () {
  226. try {
  227. wc2closer.remove (wc);
  228. removeFromPool (wc, activeConnections);
  229. wc.close ();
  230. } catch (IOException e) {
  231. String err =
  232. "CloseListener: Failed to close web connection: " + e;
  233. logger.warning (err);
  234. }
  235. }
  236. public boolean useSeparateThread () {
  237. return false;
  238. }
  239. public String getDescription () {
  240. return "ConnectionHandler$CloseListener: address: " +
  241. wc.getAddress ();
  242. }
  243. @Override public String toString () {
  244. return getClass ().getSimpleName () + "{wc: " + wc + "}@" +
  245. Integer.toString (hashCode (), 16);
  246. }
  247. }
  248. /** Mark a WebConnection ready for pipelining.
  249. * @param wc the WebConnection to mark ready for pipelining.
  250. */
  251. public void markForPipelining (WebConnection wc) {
  252. if (!usePipelining)
  253. return;
  254. synchronized (wc) {
  255. if (wc.getKeepalive ())
  256. wc.setMayPipeline (true);
  257. }
  258. }
  259. public void setup (SProperties config) {
  260. if (config == null)
  261. return;
  262. String kat = config.getProperty ("keepalivetime", "1000");
  263. try {
  264. setKeepaliveTime (Long.parseLong (kat));
  265. } catch (NumberFormatException e) {
  266. String err =
  267. "Bad number for ConnectionHandler keepalivetime: '" + kat + "'";
  268. logger.warning (err);
  269. }
  270. String up = config.get ("usepipelining");
  271. if (up == null)
  272. up = "true";
  273. usePipelining = up.equalsIgnoreCase ("true");
  274. }
  275. }