mirror of https://github.com/apache/activemq.git
https://issues.apache.org/jira/browse/AMQ-5884 https://issues.apache.org/jira/browse/AMQ-5885 Add additional validation of Topic names used in subscribe and unsubscriobe that test for spec compliance.
This commit is contained in:
parent
8ef44452a2
commit
5d6d42ce97
|
@ -367,6 +367,7 @@ public class MQTTProtocolConverter {
|
||||||
if (topics != null) {
|
if (topics != null) {
|
||||||
byte[] qos = new byte[topics.length];
|
byte[] qos = new byte[topics.length];
|
||||||
for (int i = 0; i < topics.length; i++) {
|
for (int i = 0; i < topics.length; i++) {
|
||||||
|
MQTTProtocolSupport.validate(topics[i].name().toString());
|
||||||
try {
|
try {
|
||||||
qos[i] = findSubscriptionStrategy().onSubscribe(topics[i]);
|
qos[i] = findSubscriptionStrategy().onSubscribe(topics[i]);
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
|
@ -383,6 +384,7 @@ public class MQTTProtocolConverter {
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
LOG.warn("No topics defined for Subscription " + command);
|
LOG.warn("No topics defined for Subscription " + command);
|
||||||
|
throw new MQTTProtocolException("SUBSCRIBE command received with no topic filter");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -394,16 +396,20 @@ public class MQTTProtocolConverter {
|
||||||
UTF8Buffer[] topics = command.topics();
|
UTF8Buffer[] topics = command.topics();
|
||||||
if (topics != null) {
|
if (topics != null) {
|
||||||
for (UTF8Buffer topic : topics) {
|
for (UTF8Buffer topic : topics) {
|
||||||
|
MQTTProtocolSupport.validate(topic.toString());
|
||||||
try {
|
try {
|
||||||
findSubscriptionStrategy().onUnSubscribe(topic.toString());
|
findSubscriptionStrategy().onUnSubscribe(topic.toString());
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
throw new MQTTProtocolException("Failed to process unsubscribe request", true, e);
|
throw new MQTTProtocolException("Failed to process unsubscribe request", true, e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
UNSUBACK ack = new UNSUBACK();
|
UNSUBACK ack = new UNSUBACK();
|
||||||
ack.messageId(command.messageId());
|
ack.messageId(command.messageId());
|
||||||
sendToMQTT(ack.encode());
|
sendToMQTT(ack.encode());
|
||||||
|
} else {
|
||||||
|
LOG.warn("No topics defined for Subscription " + command);
|
||||||
|
throw new MQTTProtocolException("UNSUBSCRIBE command received with no topic filter");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -16,6 +16,8 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.activemq.transport.mqtt;
|
package org.apache.activemq.transport.mqtt;
|
||||||
|
|
||||||
|
import java.io.UnsupportedEncodingException;
|
||||||
|
|
||||||
import org.fusesource.mqtt.codec.CONNECT;
|
import org.fusesource.mqtt.codec.CONNECT;
|
||||||
import org.fusesource.mqtt.codec.DISCONNECT;
|
import org.fusesource.mqtt.codec.DISCONNECT;
|
||||||
import org.fusesource.mqtt.codec.PINGREQ;
|
import org.fusesource.mqtt.codec.PINGREQ;
|
||||||
|
@ -32,6 +34,16 @@ import org.fusesource.mqtt.codec.UNSUBSCRIBE;
|
||||||
*/
|
*/
|
||||||
public class MQTTProtocolSupport {
|
public class MQTTProtocolSupport {
|
||||||
|
|
||||||
|
private static final int TOPIC_NAME_MIN_LENGTH = 1;
|
||||||
|
private static final int TOPIC_NAME_MAX_LENGTH = 65535;
|
||||||
|
|
||||||
|
private static final String MULTI_LEVEL_WILDCARD = "#";
|
||||||
|
private static final String SINGLE_LEVEL_WILDCARD = "+";
|
||||||
|
|
||||||
|
private static final char MULTI_LEVEL_WILDCARD_CHAR = '#';
|
||||||
|
private static final char SINGLE_LEVEL_WILDCARD_CHAR = '+';
|
||||||
|
private static final char TOPIC_LEVEL_SEPERATOR_CHAR = '/';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Converts an MQTT formatted Topic name into a suitable ActiveMQ Destination
|
* Converts an MQTT formatted Topic name into a suitable ActiveMQ Destination
|
||||||
* name string.
|
* name string.
|
||||||
|
@ -142,4 +154,79 @@ public class MQTTProtocolSupport {
|
||||||
return "UNKNOWN";
|
return "UNKNOWN";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Validate that the Topic names given by client commands are valid
|
||||||
|
* based on the MQTT protocol specification.
|
||||||
|
*
|
||||||
|
* @param topicName
|
||||||
|
* the given Topic name provided by the client.
|
||||||
|
*
|
||||||
|
* @throws MQTTProtocolException if the value given is invalid.
|
||||||
|
*/
|
||||||
|
public static void validate(String topicName) throws MQTTProtocolException {
|
||||||
|
int topicLen = 0;
|
||||||
|
try {
|
||||||
|
topicLen = topicName.getBytes("UTF-8").length;
|
||||||
|
} catch (UnsupportedEncodingException e) {
|
||||||
|
throw new MQTTProtocolException("Topic name contained invalid UTF-8 encoding.");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Spec: Unless stated otherwise all UTF-8 encoded strings can have any length in
|
||||||
|
// the range 0 to 65535 bytes.
|
||||||
|
if (topicLen < TOPIC_NAME_MIN_LENGTH || topicLen > TOPIC_NAME_MAX_LENGTH) {
|
||||||
|
throw new MQTTProtocolException("Topic name given had invliad length.");
|
||||||
|
}
|
||||||
|
|
||||||
|
// 4.7.1.2 and 4.7.1.3 these can stand alone
|
||||||
|
if (MULTI_LEVEL_WILDCARD.equals(topicName) || SINGLE_LEVEL_WILDCARD.equals(topicName)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Spec: 4.7.1.2
|
||||||
|
// The multi-level wildcard character MUST be specified either on its own or following a
|
||||||
|
// topic level separator. In either case it MUST be the last character specified in the
|
||||||
|
// Topic Filter [MQTT-4.7.1-2].
|
||||||
|
int numWildCards = 0;
|
||||||
|
for (int i = 0; i < topicName.length(); ++i) {
|
||||||
|
if (topicName.charAt(i) == MULTI_LEVEL_WILDCARD_CHAR) {
|
||||||
|
numWildCards++;
|
||||||
|
|
||||||
|
// If prev exists it must be a separator
|
||||||
|
if (i > 0 && topicName.charAt(i - 1) != TOPIC_LEVEL_SEPERATOR_CHAR) {
|
||||||
|
throw new MQTTProtocolException("The multi level wildcard must stand alone: " + topicName);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (numWildCards > 1) {
|
||||||
|
throw new MQTTProtocolException("Topic Filter can only have one multi-level filter: " + topicName);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (topicName.contains(MULTI_LEVEL_WILDCARD) && !topicName.endsWith(MULTI_LEVEL_WILDCARD)) {
|
||||||
|
throw new MQTTProtocolException("The multi-level filter must be at the end of the Topic name: " + topicName);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Spec: 4.7.1.3
|
||||||
|
// The single-level wildcard can be used at any level in the Topic Filter, including
|
||||||
|
// first and last levels. Where it is used it MUST occupy an entire level of the filter
|
||||||
|
//
|
||||||
|
// [MQTT-4.7.1-3]. It can be used at more than one level in the Topic Filter and can be
|
||||||
|
// used in conjunction with the multilevel wildcard.
|
||||||
|
for (int i = 0; i < topicName.length(); ++i) {
|
||||||
|
if (topicName.charAt(i) != SINGLE_LEVEL_WILDCARD_CHAR) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If prev exists it must be a separator
|
||||||
|
if (i > 0 && topicName.charAt(i - 1) != TOPIC_LEVEL_SEPERATOR_CHAR) {
|
||||||
|
throw new MQTTProtocolException("The single level wildcard must stand alone: " + topicName);
|
||||||
|
}
|
||||||
|
|
||||||
|
// If next exists it must be a separator
|
||||||
|
if (i < topicName.length() - 1 && topicName.charAt(i + 1) != TOPIC_LEVEL_SEPERATOR_CHAR) {
|
||||||
|
throw new MQTTProtocolException("The single level wildcard must stand alone: " + topicName);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1369,6 +1369,208 @@ public class MQTTTest extends MQTTTestSupport {
|
||||||
message.ack();
|
message.ack();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test(timeout = 30 * 10000)
|
||||||
|
public void testSubscribeWithZeroLengthTopic() throws Exception {
|
||||||
|
|
||||||
|
MQTT mqtt = createMQTTConnection();
|
||||||
|
mqtt.setClientId("MQTT-Client");
|
||||||
|
mqtt.setCleanSession(false);
|
||||||
|
|
||||||
|
Topic topic = new Topic("", QoS.EXACTLY_ONCE);
|
||||||
|
|
||||||
|
final BlockingConnection connection = mqtt.blockingConnection();
|
||||||
|
connection.connect();
|
||||||
|
|
||||||
|
LOG.info("Trying to subscrobe to topic: {}", topic.name());
|
||||||
|
|
||||||
|
try {
|
||||||
|
connection.subscribe(new Topic[] { topic });
|
||||||
|
fail("Should not be able to subscribe with invalid Topic");
|
||||||
|
} catch (Exception ex) {
|
||||||
|
LOG.info("Caught expected error on subscribe");
|
||||||
|
}
|
||||||
|
|
||||||
|
assertTrue("Should have lost connection", Wait.waitFor(new Wait.Condition() {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isSatisified() throws Exception {
|
||||||
|
return !connection.isConnected();
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(timeout = 30 * 10000)
|
||||||
|
public void testUnsubscribeWithZeroLengthTopic() throws Exception {
|
||||||
|
|
||||||
|
MQTT mqtt = createMQTTConnection();
|
||||||
|
mqtt.setClientId("MQTT-Client");
|
||||||
|
mqtt.setCleanSession(false);
|
||||||
|
|
||||||
|
Topic topic = new Topic("", QoS.EXACTLY_ONCE);
|
||||||
|
|
||||||
|
final BlockingConnection connection = mqtt.blockingConnection();
|
||||||
|
connection.connect();
|
||||||
|
|
||||||
|
LOG.info("Trying to subscrobe to topic: {}", topic.name());
|
||||||
|
|
||||||
|
try {
|
||||||
|
connection.unsubscribe(new String[] { topic.name().toString() });
|
||||||
|
fail("Should not be able to subscribe with invalid Topic");
|
||||||
|
} catch (Exception ex) {
|
||||||
|
LOG.info("Caught expected error on subscribe");
|
||||||
|
}
|
||||||
|
|
||||||
|
assertTrue("Should have lost connection", Wait.waitFor(new Wait.Condition() {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isSatisified() throws Exception {
|
||||||
|
return !connection.isConnected();
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(timeout = 30 * 10000)
|
||||||
|
public void testSubscribeWithInvalidMultiLevelWildcards() throws Exception {
|
||||||
|
|
||||||
|
MQTT mqtt = createMQTTConnection();
|
||||||
|
mqtt.setClientId("MQTT-Client");
|
||||||
|
mqtt.setCleanSession(false);
|
||||||
|
|
||||||
|
Topic[] topics = { new Topic("#/Foo", QoS.EXACTLY_ONCE),
|
||||||
|
new Topic("#/Foo/#", QoS.EXACTLY_ONCE),
|
||||||
|
new Topic("Foo/#/Level", QoS.EXACTLY_ONCE),
|
||||||
|
new Topic("Foo/X#", QoS.EXACTLY_ONCE) };
|
||||||
|
|
||||||
|
for (int i = 0; i < topics.length; ++i) {
|
||||||
|
final BlockingConnection connection = mqtt.blockingConnection();
|
||||||
|
connection.connect();
|
||||||
|
|
||||||
|
LOG.info("Trying to subscrobe to topic: {}", topics[i].name());
|
||||||
|
|
||||||
|
try {
|
||||||
|
connection.subscribe(new Topic[] { topics[i] });
|
||||||
|
fail("Should not be able to subscribe with invalid Topic");
|
||||||
|
} catch (Exception ex) {
|
||||||
|
LOG.info("Caught expected error on subscribe");
|
||||||
|
}
|
||||||
|
|
||||||
|
assertTrue("Should have lost connection", Wait.waitFor(new Wait.Condition() {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isSatisified() throws Exception {
|
||||||
|
return !connection.isConnected();
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(timeout = 30 * 10000)
|
||||||
|
public void testSubscribeWithInvalidSingleLevelWildcards() throws Exception {
|
||||||
|
|
||||||
|
MQTT mqtt = createMQTTConnection();
|
||||||
|
mqtt.setClientId("MQTT-Client");
|
||||||
|
mqtt.setCleanSession(false);
|
||||||
|
|
||||||
|
Topic[] topics = { new Topic("Foo+", QoS.EXACTLY_ONCE),
|
||||||
|
new Topic("+Foo/#", QoS.EXACTLY_ONCE),
|
||||||
|
new Topic("+#", QoS.EXACTLY_ONCE),
|
||||||
|
new Topic("Foo/+X/Level", QoS.EXACTLY_ONCE),
|
||||||
|
new Topic("Foo/+F", QoS.EXACTLY_ONCE) };
|
||||||
|
|
||||||
|
for (int i = 0; i < topics.length; ++i) {
|
||||||
|
final BlockingConnection connection = mqtt.blockingConnection();
|
||||||
|
connection.connect();
|
||||||
|
|
||||||
|
LOG.info("Trying to subscrobe to topic: {}", topics[i].name());
|
||||||
|
|
||||||
|
try {
|
||||||
|
connection.subscribe(new Topic[] { topics[i] });
|
||||||
|
fail("Should not be able to subscribe with invalid Topic");
|
||||||
|
} catch (Exception ex) {
|
||||||
|
LOG.info("Caught expected error on subscribe");
|
||||||
|
}
|
||||||
|
|
||||||
|
assertTrue("Should have lost connection", Wait.waitFor(new Wait.Condition() {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isSatisified() throws Exception {
|
||||||
|
return !connection.isConnected();
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(timeout = 30 * 10000)
|
||||||
|
public void testUnsubscribeWithInvalidMultiLevelWildcards() throws Exception {
|
||||||
|
|
||||||
|
MQTT mqtt = createMQTTConnection();
|
||||||
|
mqtt.setClientId("MQTT-Client");
|
||||||
|
mqtt.setCleanSession(false);
|
||||||
|
|
||||||
|
Topic[] topics = { new Topic("#/Foo", QoS.EXACTLY_ONCE),
|
||||||
|
new Topic("#/Foo/#", QoS.EXACTLY_ONCE),
|
||||||
|
new Topic("Foo/#/Level", QoS.EXACTLY_ONCE),
|
||||||
|
new Topic("Foo/X#", QoS.EXACTLY_ONCE) };
|
||||||
|
|
||||||
|
for (int i = 0; i < topics.length; ++i) {
|
||||||
|
final BlockingConnection connection = mqtt.blockingConnection();
|
||||||
|
connection.connect();
|
||||||
|
|
||||||
|
LOG.info("Trying to subscrobe to topic: {}", topics[i].name());
|
||||||
|
|
||||||
|
try {
|
||||||
|
connection.unsubscribe(new String[] { topics[i].name().toString() });
|
||||||
|
fail("Should not be able to unsubscribe with invalid Topic");
|
||||||
|
} catch (Exception ex) {
|
||||||
|
LOG.info("Caught expected error on subscribe");
|
||||||
|
}
|
||||||
|
|
||||||
|
assertTrue("Should have lost connection", Wait.waitFor(new Wait.Condition() {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isSatisified() throws Exception {
|
||||||
|
return !connection.isConnected();
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(timeout = 30 * 10000)
|
||||||
|
public void testUnsubscribeWithInvalidSingleLevelWildcards() throws Exception {
|
||||||
|
|
||||||
|
MQTT mqtt = createMQTTConnection();
|
||||||
|
mqtt.setClientId("MQTT-Client");
|
||||||
|
mqtt.setCleanSession(false);
|
||||||
|
|
||||||
|
Topic[] topics = { new Topic("Foo+", QoS.EXACTLY_ONCE),
|
||||||
|
new Topic("+Foo/#", QoS.EXACTLY_ONCE),
|
||||||
|
new Topic("+#", QoS.EXACTLY_ONCE),
|
||||||
|
new Topic("Foo/+X/Level", QoS.EXACTLY_ONCE),
|
||||||
|
new Topic("Foo/+F", QoS.EXACTLY_ONCE) };
|
||||||
|
|
||||||
|
for (int i = 0; i < topics.length; ++i) {
|
||||||
|
final BlockingConnection connection = mqtt.blockingConnection();
|
||||||
|
connection.connect();
|
||||||
|
|
||||||
|
LOG.info("Trying to subscrobe to topic: {}", topics[i].name());
|
||||||
|
|
||||||
|
try {
|
||||||
|
connection.unsubscribe(new String[] { topics[i].name().toString() });
|
||||||
|
fail("Should not be able to unsubscribe with invalid Topic");
|
||||||
|
} catch (Exception ex) {
|
||||||
|
LOG.info("Caught expected error on subscribe");
|
||||||
|
}
|
||||||
|
|
||||||
|
assertTrue("Should have lost connection", Wait.waitFor(new Wait.Condition() {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isSatisified() throws Exception {
|
||||||
|
return !connection.isConnected();
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Test(timeout = 30 * 10000)
|
@Test(timeout = 30 * 10000)
|
||||||
public void testSubscribeMultipleTopics() throws Exception {
|
public void testSubscribeMultipleTopics() throws Exception {
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue