ARTEMIS-857 Add JMX endpoints to view and reset groups
Expose method to return current mappings of groups to consumers Expose methods to reset (remove) specific group mapping from groupID to Consumer Expose methods to reset (remove) all group mappings
This commit is contained in:
parent
7156b61408
commit
af91d3ac82
|
@ -576,4 +576,28 @@ public interface QueueControl {
|
||||||
@Operation(desc = "Flush internal executors", impact = MBeanOperationInfo.ACTION)
|
@Operation(desc = "Flush internal executors", impact = MBeanOperationInfo.ACTION)
|
||||||
void flushExecutor();
|
void flushExecutor();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Will reset the all the groups.
|
||||||
|
* This is useful if you want a complete rebalance of the groups to consumers
|
||||||
|
*/
|
||||||
|
@Operation(desc = "Resets all groups", impact = MBeanOperationInfo.ACTION)
|
||||||
|
void resetAllGroups();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Will reset the group matching the given groupID.
|
||||||
|
* This is useful if you want the given group to be rebalanced to the consumers
|
||||||
|
*/
|
||||||
|
@Operation(desc = "Reset the specified group", impact = MBeanOperationInfo.ACTION)
|
||||||
|
void resetGroup(@Parameter(name = "groupID", desc = "ID of group to reset") String groupID);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Will return the current number of active groups.
|
||||||
|
*/
|
||||||
|
@Attribute(desc = "Get the current number of active groups")
|
||||||
|
int getGroupCount();
|
||||||
|
|
||||||
|
|
||||||
|
@Operation(desc = "List all the existent group to consumers mappings on the Queue")
|
||||||
|
String listGroupsAsJSON() throws Exception;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1189,6 +1189,70 @@ public class QueueControlImpl extends AbstractControl implements QueueControl {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void resetAllGroups() {
|
||||||
|
checkStarted();
|
||||||
|
|
||||||
|
clearIO();
|
||||||
|
try {
|
||||||
|
queue.resetAllGroups();
|
||||||
|
} finally {
|
||||||
|
blockOnIO();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void resetGroup(String groupID) {
|
||||||
|
checkStarted();
|
||||||
|
|
||||||
|
clearIO();
|
||||||
|
try {
|
||||||
|
queue.resetGroup(SimpleString.toSimpleString(groupID));
|
||||||
|
} finally {
|
||||||
|
blockOnIO();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getGroupCount() {
|
||||||
|
checkStarted();
|
||||||
|
|
||||||
|
clearIO();
|
||||||
|
try {
|
||||||
|
return queue.getGroupCount();
|
||||||
|
} finally {
|
||||||
|
blockOnIO();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String listGroupsAsJSON() throws Exception {
|
||||||
|
checkStarted();
|
||||||
|
|
||||||
|
clearIO();
|
||||||
|
try {
|
||||||
|
Map<SimpleString, Consumer> groups = queue.getGroups();
|
||||||
|
|
||||||
|
JsonArrayBuilder jsonArray = JsonLoader.createArrayBuilder();
|
||||||
|
|
||||||
|
for (Map.Entry<SimpleString, Consumer> group : groups.entrySet()) {
|
||||||
|
|
||||||
|
if (group.getValue() instanceof ServerConsumer) {
|
||||||
|
ServerConsumer serverConsumer = (ServerConsumer) group.getValue();
|
||||||
|
|
||||||
|
JsonObjectBuilder obj = JsonLoader.createObjectBuilder().add("groupID", group.getKey().toString()).add("consumerID", serverConsumer.getID()).add("connectionID", serverConsumer.getConnectionID().toString()).add("sessionID", serverConsumer.getSessionID()).add("browseOnly", serverConsumer.isBrowseOnly()).add("creationTime", serverConsumer.getCreationTime());
|
||||||
|
|
||||||
|
jsonArray.add(obj);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
return jsonArray.build().toString();
|
||||||
|
} finally {
|
||||||
|
blockOnIO();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String listConsumersAsJSON() throws Exception {
|
public String listConsumersAsJSON() throws Exception {
|
||||||
checkStarted();
|
checkStarted();
|
||||||
|
|
|
@ -255,6 +255,14 @@ public interface Queue extends Bindable,CriticalComponent {
|
||||||
|
|
||||||
Collection<Consumer> getConsumers();
|
Collection<Consumer> getConsumers();
|
||||||
|
|
||||||
|
Map<SimpleString, Consumer> getGroups();
|
||||||
|
|
||||||
|
void resetGroup(SimpleString groupID);
|
||||||
|
|
||||||
|
void resetAllGroups();
|
||||||
|
|
||||||
|
int getGroupCount();
|
||||||
|
|
||||||
boolean checkRedelivery(MessageReference ref, long timeBase, boolean ignoreRedeliveryDelay) throws Exception;
|
boolean checkRedelivery(MessageReference ref, long timeBase, boolean ignoreRedeliveryDelay) throws Exception;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -1038,6 +1038,26 @@ public class QueueImpl extends CriticalComponentImpl implements Queue {
|
||||||
return new HashSet<>(consumerSet);
|
return new HashSet<>(consumerSet);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public synchronized Map<SimpleString, Consumer> getGroups() {
|
||||||
|
return new HashMap<>(groups);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public synchronized void resetGroup(SimpleString groupId) {
|
||||||
|
groups.remove(groupId);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public synchronized void resetAllGroups() {
|
||||||
|
groups.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public synchronized int getGroupCount() {
|
||||||
|
return groups.size();
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean hasMatchingConsumer(final Message message) {
|
public boolean hasMatchingConsumer(final Message message) {
|
||||||
for (ConsumerHolder holder : consumerList) {
|
for (ConsumerHolder holder : consumerList) {
|
||||||
|
|
|
@ -1242,6 +1242,26 @@ public class ScheduledDeliveryHandlerTest extends Assert {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Map<SimpleString, Consumer> getGroups() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void resetGroup(SimpleString groupID) {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void resetAllGroups() {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getGroupCount() {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean checkRedelivery(MessageReference ref,
|
public boolean checkRedelivery(MessageReference ref,
|
||||||
long timeBase,
|
long timeBase,
|
||||||
|
|
|
@ -122,6 +122,12 @@ public abstract class ManagementTestBase extends ActiveMQTestBase {
|
||||||
return control.getMessageCount();
|
return control.getMessageCount();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected int getGroupCount(QueueControl control) throws Exception {
|
||||||
|
control.flushExecutor();
|
||||||
|
return control.getGroupCount();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
protected long getDurableMessageCount(QueueControl control) throws Exception {
|
protected long getDurableMessageCount(QueueControl control) throws Exception {
|
||||||
control.flushExecutor();
|
control.flushExecutor();
|
||||||
return control.getDurableMessageCount();
|
return control.getDurableMessageCount();
|
||||||
|
|
|
@ -2510,6 +2510,48 @@ public class QueueControlTest extends ManagementTestBase {
|
||||||
Assert.assertEquals(new String(body), "theBody");
|
Assert.assertEquals(new String(body), "theBody");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testResetGroups() throws Exception {
|
||||||
|
SimpleString address = RandomUtil.randomSimpleString();
|
||||||
|
SimpleString queue = RandomUtil.randomSimpleString();
|
||||||
|
|
||||||
|
session.createQueue(address, RoutingType.MULTICAST, queue, null, durable);
|
||||||
|
|
||||||
|
QueueControl queueControl = createManagementControl(address, queue);
|
||||||
|
|
||||||
|
ClientConsumer consumer = session.createConsumer(queue);
|
||||||
|
Assert.assertEquals(1, queueControl.getConsumerCount());
|
||||||
|
consumer.setMessageHandler(new MessageHandler() {
|
||||||
|
@Override
|
||||||
|
public void onMessage(ClientMessage message) {
|
||||||
|
System.out.println(message);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
session.start();
|
||||||
|
|
||||||
|
ClientProducer producer = session.createProducer(address);
|
||||||
|
producer.send(session.createMessage(durable).putStringProperty(Message.HDR_GROUP_ID, "group1"));
|
||||||
|
producer.send(session.createMessage(durable).putStringProperty(Message.HDR_GROUP_ID, "group2"));
|
||||||
|
producer.send(session.createMessage(durable).putStringProperty(Message.HDR_GROUP_ID, "group3"));
|
||||||
|
|
||||||
|
Wait.assertEquals(3, () -> getGroupCount(queueControl));
|
||||||
|
|
||||||
|
queueControl.resetGroup("group1");
|
||||||
|
|
||||||
|
Wait.assertEquals(2, () -> getGroupCount(queueControl));
|
||||||
|
|
||||||
|
producer.send(session.createMessage(durable).putStringProperty(Message.HDR_GROUP_ID, "group1"));
|
||||||
|
|
||||||
|
Wait.assertEquals(3, () -> getGroupCount(queueControl));
|
||||||
|
|
||||||
|
queueControl.resetAllGroups();
|
||||||
|
|
||||||
|
Wait.assertEquals(0, () -> getGroupCount(queueControl));
|
||||||
|
|
||||||
|
consumer.close();
|
||||||
|
session.deleteQueue(queue);
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testGetScheduledCountOnRemove() throws Exception {
|
public void testGetScheduledCountOnRemove() throws Exception {
|
||||||
long delay = Integer.MAX_VALUE;
|
long delay = Integer.MAX_VALUE;
|
||||||
|
|
|
@ -49,6 +49,38 @@ public class QueueControlUsingCoreTest extends QueueControlTest {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void resetAllGroups() {
|
||||||
|
try {
|
||||||
|
proxy.invokeOperation("resetAllGroups");
|
||||||
|
} catch (Exception e) {
|
||||||
|
throw new RuntimeException(e.getMessage(), e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void resetGroup(String groupID) {
|
||||||
|
try {
|
||||||
|
proxy.invokeOperation("resetGroup");
|
||||||
|
} catch (Exception e) {
|
||||||
|
throw new RuntimeException(e.getMessage(), e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getGroupCount() {
|
||||||
|
try {
|
||||||
|
return (Integer) proxy.invokeOperation("groupCount");
|
||||||
|
} catch (Exception e) {
|
||||||
|
throw new RuntimeException(e.getMessage(), e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String listGroupsAsJSON() throws Exception {
|
||||||
|
return (String) proxy.invokeOperation("listGroupsAsJSON");
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean changeMessagePriority(final long messageID, final int newPriority) throws Exception {
|
public boolean changeMessagePriority(final long messageID, final int newPriority) throws Exception {
|
||||||
return (Boolean) proxy.invokeOperation("changeMessagePriority", messageID, newPriority);
|
return (Boolean) proxy.invokeOperation("changeMessagePriority", messageID, newPriority);
|
||||||
|
|
|
@ -342,6 +342,7 @@ public class FakeQueue extends CriticalComponentImpl implements Queue {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
||||||
public int getConsumerCount() {
|
public int getConsumerCount() {
|
||||||
// no-op
|
// no-op
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -358,6 +359,26 @@ public class FakeQueue extends CriticalComponentImpl implements Queue {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Map<SimpleString, Consumer> getGroups() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void resetGroup(SimpleString groupID) {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void resetAllGroups() {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getGroupCount() {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public int getDeliveringCount() {
|
public int getDeliveringCount() {
|
||||||
// no-op
|
// no-op
|
||||||
|
|
Loading…
Reference in New Issue