This closes #2708
This commit is contained in:
commit
cac984ee6d
|
@ -81,7 +81,10 @@ public class MQTTConnectionManager {
|
||||||
session.getConnection().setClientID(clientId);
|
session.getConnection().setClientID(clientId);
|
||||||
ServerSessionImpl serverSession = createServerSession(username, password);
|
ServerSessionImpl serverSession = createServerSession(username, password);
|
||||||
serverSession.start();
|
serverSession.start();
|
||||||
session.setServerSession(serverSession);
|
ServerSessionImpl internalServerSession = createServerSession(username, password);
|
||||||
|
internalServerSession.disableSecurity();
|
||||||
|
internalServerSession.start();
|
||||||
|
session.setServerSession(serverSession, internalServerSession);
|
||||||
|
|
||||||
if (cleanSession) {
|
if (cleanSession) {
|
||||||
/* [MQTT-3.1.2-6] If CleanSession is set to 1, the Client and Server MUST discard any previous Session and
|
/* [MQTT-3.1.2-6] If CleanSession is set to 1, the Client and Server MUST discard any previous Session and
|
||||||
|
|
|
@ -90,7 +90,7 @@ public class MQTTPublishManager {
|
||||||
|
|
||||||
private void createManagementConsumer() throws Exception {
|
private void createManagementConsumer() throws Exception {
|
||||||
long consumerId = session.getServer().getStorageManager().generateID();
|
long consumerId = session.getServer().getStorageManager().generateID();
|
||||||
managementConsumer = session.getServerSession().createConsumer(consumerId, managementAddress, null, false, false, -1);
|
managementConsumer = session.getInternalServerSession().createConsumer(consumerId, managementAddress, null, false, false, -1);
|
||||||
managementConsumer.setStarted(true);
|
managementConsumer.setStarted(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -206,7 +206,8 @@ public class MQTTPublishManager {
|
||||||
Pair<Long, Long> ref = outboundStore.publishReceived(messageId);
|
Pair<Long, Long> ref = outboundStore.publishReceived(messageId);
|
||||||
if (ref != null) {
|
if (ref != null) {
|
||||||
Message m = MQTTUtil.createPubRelMessage(session, getManagementAddress(), messageId);
|
Message m = MQTTUtil.createPubRelMessage(session, getManagementAddress(), messageId);
|
||||||
session.getServerSession().send(m, true);
|
//send the management message via the internal server session to bypass security.
|
||||||
|
session.getInternalServerSession().send(m, true);
|
||||||
session.getServerSession().individualAcknowledge(ref.getB(), ref.getA());
|
session.getServerSession().individualAcknowledge(ref.getB(), ref.getA());
|
||||||
} else {
|
} else {
|
||||||
session.getProtocolHandler().sendPubRel(messageId);
|
session.getProtocolHandler().sendPubRel(messageId);
|
||||||
|
@ -219,7 +220,8 @@ public class MQTTPublishManager {
|
||||||
void handlePubComp(int messageId) throws Exception {
|
void handlePubComp(int messageId) throws Exception {
|
||||||
Pair<Long, Long> ref = session.getState().getOutboundStore().publishComplete(messageId);
|
Pair<Long, Long> ref = session.getState().getOutboundStore().publishComplete(messageId);
|
||||||
if (ref != null) {
|
if (ref != null) {
|
||||||
session.getServerSession().individualAcknowledge(managementConsumer.getID(), ref.getA());
|
//ack the message via the internal server session to bypass security.
|
||||||
|
session.getInternalServerSession().individualAcknowledge(managementConsumer.getID(), ref.getA());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -37,6 +37,8 @@ public class MQTTSession {
|
||||||
|
|
||||||
private ServerSessionImpl serverSession;
|
private ServerSessionImpl serverSession;
|
||||||
|
|
||||||
|
private ServerSessionImpl internalServerSession;
|
||||||
|
|
||||||
private MQTTPublishManager mqttPublishManager;
|
private MQTTPublishManager mqttPublishManager;
|
||||||
|
|
||||||
private MQTTConnectionManager mqttConnectionManager;
|
private MQTTConnectionManager mqttConnectionManager;
|
||||||
|
@ -97,6 +99,11 @@ public class MQTTSession {
|
||||||
serverSession.close(false);
|
serverSession.close(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (internalServerSession != null) {
|
||||||
|
internalServerSession.stop();
|
||||||
|
internalServerSession.close(false);
|
||||||
|
}
|
||||||
|
|
||||||
if (state != null) {
|
if (state != null) {
|
||||||
state.setAttached(false);
|
state.setAttached(false);
|
||||||
}
|
}
|
||||||
|
@ -141,6 +148,10 @@ public class MQTTSession {
|
||||||
return serverSession;
|
return serverSession;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ServerSessionImpl getInternalServerSession() {
|
||||||
|
return internalServerSession;
|
||||||
|
}
|
||||||
|
|
||||||
ActiveMQServer getServer() {
|
ActiveMQServer getServer() {
|
||||||
return protocolHandler.getServer();
|
return protocolHandler.getServer();
|
||||||
}
|
}
|
||||||
|
@ -157,8 +168,9 @@ public class MQTTSession {
|
||||||
return sessionCallback;
|
return sessionCallback;
|
||||||
}
|
}
|
||||||
|
|
||||||
void setServerSession(ServerSessionImpl serverSession) {
|
void setServerSession(ServerSessionImpl serverSession, ServerSessionImpl internalServerSession) {
|
||||||
this.serverSession = serverSession;
|
this.serverSession = serverSession;
|
||||||
|
this.internalServerSession = internalServerSession;
|
||||||
}
|
}
|
||||||
|
|
||||||
void setSessionState(MQTTSessionState state) {
|
void setSessionState(MQTTSessionState state) {
|
||||||
|
|
|
@ -0,0 +1,112 @@
|
||||||
|
/**
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright ownership.
|
||||||
|
* The ASF licenses this file to You under the Apache License, Version 2.0
|
||||||
|
* (the "License"); you may not use this file except in compliance with
|
||||||
|
* the License. You may obtain a copy of the License at
|
||||||
|
* <p>
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
* <p>
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.apache.activemq.artemis.tests.integration.mqtt.imported;
|
||||||
|
|
||||||
|
import org.apache.activemq.artemis.core.security.Role;
|
||||||
|
import org.apache.activemq.artemis.core.server.ActiveMQServer;
|
||||||
|
import org.apache.activemq.artemis.core.settings.HierarchicalRepository;
|
||||||
|
import org.apache.activemq.artemis.spi.core.security.ActiveMQJAASSecurityManager;
|
||||||
|
import org.eclipse.paho.client.mqttv3.IMqttDeliveryToken;
|
||||||
|
import org.eclipse.paho.client.mqttv3.MqttCallback;
|
||||||
|
import org.eclipse.paho.client.mqttv3.MqttClient;
|
||||||
|
import org.eclipse.paho.client.mqttv3.MqttConnectOptions;
|
||||||
|
import org.eclipse.paho.client.mqttv3.MqttException;
|
||||||
|
import org.eclipse.paho.client.mqttv3.MqttMessage;
|
||||||
|
import org.eclipse.paho.client.mqttv3.persist.MemoryPersistence;
|
||||||
|
import org.junit.Assert;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.Set;
|
||||||
|
import java.util.concurrent.CountDownLatch;
|
||||||
|
|
||||||
|
public class PahoMQTTQOS2SecurityTest extends MQTTTestSupport {
|
||||||
|
|
||||||
|
String user1 = "user1";
|
||||||
|
String password1 = "password1";
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void configureBrokerSecurity(ActiveMQServer server) {
|
||||||
|
super.configureBrokerSecurity(server);
|
||||||
|
ActiveMQJAASSecurityManager securityManager = (ActiveMQJAASSecurityManager) server.getSecurityManager();
|
||||||
|
|
||||||
|
// User additions
|
||||||
|
securityManager.getConfiguration().addUser(user1, password1);
|
||||||
|
securityManager.getConfiguration().addRole(user1, "addressOnly");
|
||||||
|
|
||||||
|
// Configure roles
|
||||||
|
HierarchicalRepository<Set<Role>> securityRepository = server.getSecurityRepository();
|
||||||
|
HashSet<Role> value = new HashSet<>();
|
||||||
|
value.add(new Role("addressOnly", true, true, true, true, false, false, false, false, true, true));
|
||||||
|
|
||||||
|
securityRepository.addMatch(getQueueName(), value);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isSecurityEnabled() {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(timeout = 300000)
|
||||||
|
public void testSendAndReceiveMQTT() throws Exception {
|
||||||
|
final CountDownLatch latch = new CountDownLatch(1);
|
||||||
|
|
||||||
|
MqttClient consumer = createPahoClient("consumerId");
|
||||||
|
MqttClient producer = createPahoClient("producerId");
|
||||||
|
MqttConnectOptions conOpt = new MqttConnectOptions();
|
||||||
|
conOpt.setCleanSession(true);
|
||||||
|
conOpt.setUserName(user1);
|
||||||
|
conOpt.setPassword(password1.toCharArray());
|
||||||
|
consumer.connect(conOpt);
|
||||||
|
consumer.subscribe(getQueueName(), 2);
|
||||||
|
final boolean[] failed = new boolean[1];
|
||||||
|
consumer.setCallback(new MqttCallback() {
|
||||||
|
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void connectionLost(Throwable cause) {
|
||||||
|
cause.printStackTrace();
|
||||||
|
failed[0] = true;
|
||||||
|
latch.countDown();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void messageArrived(String topic, MqttMessage message) throws Exception {
|
||||||
|
latch.countDown();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void deliveryComplete(IMqttDeliveryToken token) {
|
||||||
|
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
producer.connect(conOpt);
|
||||||
|
producer.publish(getQueueName(), "hello".getBytes(), 2, false);
|
||||||
|
|
||||||
|
waitForLatch(latch);
|
||||||
|
producer.disconnect();
|
||||||
|
producer.close();
|
||||||
|
Assert.assertFalse(failed[0]);
|
||||||
|
}
|
||||||
|
|
||||||
|
private MqttClient createPahoClient(String clientId) throws MqttException {
|
||||||
|
return new MqttClient("tcp://localhost:" + getPort(), clientId, new MemoryPersistence());
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
Loading…
Reference in New Issue