Commit cceb769c authored by csh's avatar csh

OF-706: Openfire does not close the stream with a stream error if the...

OF-706: Openfire does not close the stream with a stream error if the namespace is not 'http://etherx.jabber.org/streams'

git-svn-id: http://svn.igniterealtime.org/svn/repos/openfire/trunk@13946 b35dd754-fafc-0310-a699-88a17e54d16e
parent e3fdaae5
...@@ -19,9 +19,6 @@ ...@@ -19,9 +19,6 @@
package org.jivesoftware.openfire.net; package org.jivesoftware.openfire.net;
import java.io.IOException;
import java.io.StringReader;
import org.dom4j.Element; import org.dom4j.Element;
import org.dom4j.io.XMPPPacketReader; import org.dom4j.io.XMPPPacketReader;
import org.jivesoftware.openfire.Connection; import org.jivesoftware.openfire.Connection;
...@@ -38,13 +35,10 @@ import org.slf4j.Logger; ...@@ -38,13 +35,10 @@ import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
import org.xmlpull.v1.XmlPullParser; import org.xmlpull.v1.XmlPullParser;
import org.xmlpull.v1.XmlPullParserException; import org.xmlpull.v1.XmlPullParserException;
import org.xmpp.packet.IQ; import org.xmpp.packet.*;
import org.xmpp.packet.JID;
import org.xmpp.packet.Message; import java.io.IOException;
import org.xmpp.packet.PacketError; import java.io.StringReader;
import org.xmpp.packet.Presence;
import org.xmpp.packet.Roster;
import org.xmpp.packet.StreamError;
/** /**
* A StanzaHandler is the main responsible for handling incoming stanzas. Some stanzas like startTLS * A StanzaHandler is the main responsible for handling incoming stanzas. Some stanzas like startTLS
...@@ -53,7 +47,7 @@ import org.xmpp.packet.StreamError; ...@@ -53,7 +47,7 @@ import org.xmpp.packet.StreamError;
* @author Gaston Dombiak * @author Gaston Dombiak
*/ */
public abstract class StanzaHandler { public abstract class StanzaHandler {
private static final Logger Log = LoggerFactory.getLogger(StanzaHandler.class); private static final Logger Log = LoggerFactory.getLogger(StanzaHandler.class);
/** /**
...@@ -602,36 +596,36 @@ public abstract class StanzaHandler { ...@@ -602,36 +596,36 @@ public abstract class StanzaHandler {
// subdomain. If the value of the 'to' attribute is not valid then return a host-unknown // subdomain. If the value of the 'to' attribute is not valid then return a host-unknown
// error and close the underlying connection. // error and close the underlying connection.
String host = xpp.getAttributeValue("", "to"); String host = xpp.getAttributeValue("", "to");
StreamError streamError = null;
if (validateHost() && isHostUnknown(host)) { if (validateHost() && isHostUnknown(host)) {
StringBuilder sb = new StringBuilder(250); streamError = new StreamError(StreamError.Condition.host_unknown);
sb.append("<?xml version='1.0' encoding='");
sb.append(CHARSET);
sb.append("'?>");
// Append stream header
sb.append("<stream:stream ");
sb.append("from=\"").append(serverName).append("\" ");
sb.append("id=\"").append(StringUtils.randomString(5)).append("\" ");
sb.append("xmlns=\"").append(xpp.getNamespace(null)).append("\" ");
sb.append("xmlns:stream=\"").append(xpp.getNamespace("stream")).append("\" ");
sb.append("version=\"1.0\">");
// Set the host_unknown error
StreamError error = new StreamError(StreamError.Condition.host_unknown);
sb.append(error.toXML());
// Deliver stanza
connection.deliverRawText(sb.toString());
// Close the underlying connection
connection.close();
// Log a warning so that admins can track this cases from the server side // Log a warning so that admins can track this cases from the server side
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 +
". Connection: " + connection); ". Connection: " + connection);
} }
// Validate the stream namespace
else if (!"http://etherx.jabber.org/streams".equals(xpp.getNamespace())) {
// Include the invalid-namespace in the response
streamError = new StreamError(StreamError.Condition.invalid_namespace);
// Log a warning so that admins can track this cases from the server side
Log.warn("Closing session due to invalid_namespace in stream header. Namespace: " +
xpp.getNamespace() + ". Connection: " + connection);
}
// Create the correct session based on the sent namespace. At this point the server // Create the correct session based on the sent namespace. At this point the server
// may offer the client to secure the connection. If the client decides to secure // may offer the client to secure the connection. If the client decides to secure
// the connection then a <starttls> stanza should be received // the connection then a <starttls> stanza should be received
else if (!createSession(xpp.getNamespace(null), serverName, xpp, connection)) { else if (!createSession(xpp.getNamespace(null), serverName, xpp, connection)) {
// No session was created because of an invalid namespace prefix so answer a stream // No session was created because of an invalid namespace prefix so answer a stream
// error and close the underlying connection // error and close the underlying connection
// Include the bad-namespace-prefix in the response
streamError = new StreamError(StreamError.Condition.bad_namespace_prefix);
// Log a warning so that admins can track this cases from the server side
Log.warn("Closing session due to bad_namespace_prefix in stream header. Prefix: " +
xpp.getNamespace(null) + ". Connection: " + connection);
}
if (streamError != null) {
StringBuilder sb = new StringBuilder(250); StringBuilder sb = new StringBuilder(250);
sb.append("<?xml version='1.0' encoding='"); sb.append("<?xml version='1.0' encoding='");
sb.append(CHARSET); sb.append(CHARSET);
...@@ -641,18 +635,15 @@ public abstract class StanzaHandler { ...@@ -641,18 +635,15 @@ public abstract class StanzaHandler {
sb.append("from=\"").append(serverName).append("\" "); sb.append("from=\"").append(serverName).append("\" ");
sb.append("id=\"").append(StringUtils.randomString(5)).append("\" "); sb.append("id=\"").append(StringUtils.randomString(5)).append("\" ");
sb.append("xmlns=\"").append(xpp.getNamespace(null)).append("\" "); sb.append("xmlns=\"").append(xpp.getNamespace(null)).append("\" ");
sb.append("xmlns:stream=\"").append(xpp.getNamespace("stream")).append("\" "); sb.append("xmlns:stream=\"http://etherx.jabber.org/streams\" ");
sb.append("version=\"1.0\">"); sb.append("version=\"1.0\">");
// Include the bad-namespace-prefix in the response sb.append(streamError.toXML());
StreamError error = new StreamError(StreamError.Condition.bad_namespace_prefix); // Deliver stanza
sb.append(error.toXML());
connection.deliverRawText(sb.toString()); connection.deliverRawText(sb.toString());
// Close the underlying connection // Close the underlying connection
connection.close(); connection.close();
// Log a warning so that admins can track this cases from the server side
Log.warn("Closing session due to bad_namespace_prefix in stream header. Prefix: " +
xpp.getNamespace(null) + ". Connection: " + connection);
} }
} }
private boolean isHostUnknown(String host) { private boolean isHostUnknown(String host) {
...@@ -671,23 +662,23 @@ public abstract class StanzaHandler { ...@@ -671,23 +662,23 @@ public abstract class StanzaHandler {
/** /**
* Obtain the address of the XMPP entity for which this StanzaHandler * Obtain the address of the XMPP entity for which this StanzaHandler
* handles stanzas. * handles stanzas.
* *
* Note that the value that is returned for this method can * Note that the value that is returned for this method can
* change over time. For example, if no session has been established yet, * change over time. For example, if no session has been established yet,
* this method will return </tt>null</tt>, or, if resource binding occurs, * this method will return </tt>null</tt>, or, if resource binding occurs,
* the returned value might change. Values obtained from this method are * the returned value might change. Values obtained from this method are
* therefore best <em>not</em> cached. * therefore best <em>not</em> cached.
* *
* @return The address of the XMPP entity for. * @return The address of the XMPP entity for.
*/ */
public JID getAddress() { public JID getAddress() {
if (session == null) { if (session == null) {
return null; return null;
} }
return session.getAddress(); return session.getAddress();
} }
/** /**
* Returns the stream namespace. (E.g. jabber:client, jabber:server, etc.). * Returns the stream namespace. (E.g. jabber:client, jabber:server, etc.).
* *
......
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