Commit 975e38f6 authored by Dave Cridland's avatar Dave Cridland

Merge pull request #468 from tevans/OF-1006

OF-1006: Improve handling for pooled resources
parents 8f26268b 99424f01
...@@ -44,6 +44,12 @@ ...@@ -44,6 +44,12 @@
Openfire WebSocket Plugin Changelog Openfire WebSocket Plugin Changelog
</h1> </h1>
<p><b>1.1.2</b> -- December 17, 2015</p>
<ul>
<li>OF-1006: Avoid connection blocking with improved handling for pooled resources.</li>
</ul>
<p><b>1.1</b> -- November 24, 2015</p> <p><b>1.1</b> -- November 24, 2015</p>
<ul> <ul>
......
...@@ -8,8 +8,8 @@ ...@@ -8,8 +8,8 @@
<name>Openfire WebSocket</name> <name>Openfire WebSocket</name>
<description>Provides WebSocket support for Openfire.</description> <description>Provides WebSocket support for Openfire.</description>
<author>Tom Evans</author> <author>Tom Evans</author>
<version>1.1.0</version> <version>1.1.2</version>
<date>11/24/2015</date> <date>12/17/2015</date>
<url>https://tools.ietf.org/html/rfc7395</url> <url>https://tools.ietf.org/html/rfc7395</url>
<minServerVersion>3.11.0</minServerVersion> <minServerVersion>3.11.0</minServerVersion>
</plugin> </plugin>
\ No newline at end of file
...@@ -103,22 +103,23 @@ public class XmppWebSocket { ...@@ -103,22 +103,23 @@ public class XmppWebSocket {
@OnWebSocketMessage @OnWebSocketMessage
public void onTextMethod(String stanza) public void onTextMethod(String stanza)
{ {
XMPPPacketReader reader = null;
try { try {
reader = readerPool.borrowObject();
XMPPPacketReader reader = readerPool.borrowObject();
Document doc = reader.read(new StringReader(stanza)); Document doc = reader.read(new StringReader(stanza));
readerPool.returnObject(reader);
if (xmppSession == null) { if (xmppSession == null) {
initiateSession(doc.getRootElement()); initiateSession(doc.getRootElement());
} else { } else {
processStanza(doc.getRootElement()); processStanza(doc.getRootElement());
} }
} catch (Exception ex) { } catch (Exception ex) {
Log.error("Failed to initiate XMPP session", ex); Log.error("Failed to process XMPP stanza", ex);
} finally {
if (reader != null) {
readerPool.returnObject(reader);
}
} }
} }
@OnWebSocketError @OnWebSocketError
...@@ -127,7 +128,9 @@ public class XmppWebSocket { ...@@ -127,7 +128,9 @@ public class XmppWebSocket {
Log.error("Error detected; session: " + wsSession, error); Log.error("Error detected; session: " + wsSession, error);
closeStream(new StreamError(StreamError.Condition.internal_server_error)); closeStream(new StreamError(StreamError.Condition.internal_server_error));
try { try {
if (wsSession != null) {
wsSession.disconnect(); wsSession.disconnect();
}
} catch ( Exception e ) { } catch ( Exception e ) {
Log.error("Error disconnecting websocket", e); Log.error("Error disconnecting websocket", e);
} }
...@@ -251,7 +254,7 @@ public class XmppWebSocket { ...@@ -251,7 +254,7 @@ public class XmppWebSocket {
Locale language = Locale.forLanguageTag(stanza.attributeValue(QName.get("lang", XMLConstants.XML_NS_URI), "en")); Locale language = Locale.forLanguageTag(stanza.attributeValue(QName.get("lang", XMLConstants.XML_NS_URI), "en"));
if (STREAM_FOOTER.equals(stanza.getName())) { if (STREAM_FOOTER.equals(stanza.getName())) {
// an error occurred while setting up the session // an error occurred while setting up the session
closeStream(null); Log.warn("Client closed stream before session was established");
} else if (!STREAM_HEADER.equals(stanza.getName())) { } else if (!STREAM_HEADER.equals(stanza.getName())) {
streamError = new StreamError(StreamError.Condition.unsupported_stanza_type); streamError = new StreamError(StreamError.Condition.unsupported_stanza_type);
Log.warn("Closing session due to incorrect stream header. Tag: " + stanza.getName()); Log.warn("Closing session due to incorrect stream header. Tag: " + stanza.getName());
...@@ -259,21 +262,20 @@ public class XmppWebSocket { ...@@ -259,21 +262,20 @@ public class XmppWebSocket {
// Validate the stream namespace (https://tools.ietf.org/html/rfc7395#section-3.3.2) // Validate the stream namespace (https://tools.ietf.org/html/rfc7395#section-3.3.2)
streamError = new StreamError(StreamError.Condition.invalid_namespace); streamError = new StreamError(StreamError.Condition.invalid_namespace);
Log.warn("Closing session due to invalid namespace in stream header. Namespace: " + stanza.getNamespace().getURI()); Log.warn("Closing session due to invalid namespace in stream header. Namespace: " + stanza.getNamespace().getURI());
} else if (STREAM_FOOTER.equals(stanza.getName())) {
closeStream(null);
} else if (!validateHost(host)) { } else if (!validateHost(host)) {
streamError = new StreamError(StreamError.Condition.host_unknown); streamError = new StreamError(StreamError.Condition.host_unknown);
Log.warn("Closing session due to incorrect hostname in stream header. Host: " + host); Log.warn("Closing session due to incorrect hostname in stream header. Host: " + host);
} else { } else {
// valid stream; initiate session
xmppSession = SessionManager.getInstance().createClientSession(wsConnection, language); xmppSession = SessionManager.getInstance().createClientSession(wsConnection, language);
xmppSession.setSessionData("ws", Boolean.TRUE); xmppSession.setSessionData("ws", Boolean.TRUE);
} }
if (streamError == null) { if (xmppSession == null) {
closeStream(streamError);
} else {
openStream(language.toLanguageTag(), stanza.attributeValue("from")); openStream(language.toLanguageTag(), stanza.attributeValue("from"));
configureStream(); configureStream();
} else {
closeStream(streamError);
} }
} }
...@@ -355,10 +357,10 @@ public class XmppWebSocket { ...@@ -355,10 +357,10 @@ public class XmppWebSocket {
private synchronized void initializePool() { private synchronized void initializePool() {
if (readerPool == null) { if (readerPool == null) {
readerPool = new GenericObjectPool<XMPPPacketReader>(new XMPPPPacketReaderFactory()); readerPool = new GenericObjectPool<XMPPPacketReader>(new XMPPPPacketReaderFactory());
readerPool.setMaxTotal(32); readerPool.setMaxTotal(-1);
readerPool.setBlockWhenExhausted(false);
readerPool.setTestOnReturn(true); readerPool.setTestOnReturn(true);
readerPool.setNumTestsPerEvictionRun(-2); // evict half of the idle instances readerPool.setTimeBetweenEvictionRunsMillis(JiveConstants.MINUTE);
readerPool.setTimeBetweenEvictionRunsMillis(JiveConstants.HOUR);
} }
} }
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment