ARTEMIS-3175 - implement address setting management-message-attribute-size-limit to sensibly limit data returned by list/browse/filter management ops

This commit is contained in:
gtully 2021-03-12 12:18:00 +00:00 committed by Gary Tully
parent 7c5ef2796a
commit 02bb7031c2
14 changed files with 336 additions and 118 deletions

View File

@ -102,10 +102,11 @@ public interface ICoreMessage extends Message {
/** /**
* @return Returns the message in Map form, useful when encoding to JSON * @return Returns the message in Map form, useful when encoding to JSON
* @param valueSizeLimit
*/ */
@Override @Override
default Map<String, Object> toMap() { default Map<String, Object> toMap(int valueSizeLimit) {
Map map = toPropertyMap(); Map map = toPropertyMap(valueSizeLimit);
map.put("messageID", getMessageID()); map.put("messageID", getMessageID());
Object userID = getUserID(); Object userID = getUserID();
if (getUserID() != null) { if (getUserID() != null) {

View File

@ -28,6 +28,7 @@ import javax.management.openmbean.CompositeDataSupport;
import java.io.ByteArrayInputStream; import java.io.ByteArrayInputStream;
import java.io.StringReader; import java.io.StringReader;
import java.lang.reflect.Array; import java.lang.reflect.Array;
import java.util.Arrays;
import java.util.HashMap; import java.util.HashMap;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
@ -45,53 +46,8 @@ public final class JsonUtil {
JsonArrayBuilder jsonArray = JsonLoader.createArrayBuilder(); JsonArrayBuilder jsonArray = JsonLoader.createArrayBuilder();
for (Object parameter : array) { for (Object parameter : array) {
if (parameter instanceof Map) {
Map<String, Object> map = (Map<String, Object>) parameter;
JsonObjectBuilder jsonObject = JsonLoader.createObjectBuilder();
for (Map.Entry<String, Object> entry : map.entrySet()) {
String key = entry.getKey();
Object val = entry.getValue();
if (val != null) {
if (val.getClass().isArray()) {
JsonArray objectArray = toJSONArray((Object[]) val);
jsonObject.add(key, objectArray);
} else {
addToObject(key, val, jsonObject);
}
}
}
jsonArray.add(jsonObject);
} else {
if (parameter != null) {
Class<?> clz = parameter.getClass();
if (clz.isArray()) {
Object[] innerArray = (Object[]) parameter;
if (innerArray instanceof CompositeData[]) {
JsonArrayBuilder innerJsonArray = JsonLoader.createArrayBuilder();
for (Object data : innerArray) {
String s = Base64.encodeObject((CompositeDataSupport) data);
innerJsonArray.add(s);
}
JsonObjectBuilder jsonObject = JsonLoader.createObjectBuilder();
jsonObject.add(CompositeData.class.getName(), innerJsonArray);
jsonArray.add(jsonObject);
} else {
jsonArray.add(toJSONArray(innerArray));
}
} else {
addToArray(parameter, jsonArray); addToArray(parameter, jsonArray);
} }
} else {
jsonArray.addNull();
}
}
}
return jsonArray.build(); return jsonArray.build();
} }
@ -210,6 +166,12 @@ public final class JsonUtil {
} else if (param instanceof byte[]) { } else if (param instanceof byte[]) {
JsonArrayBuilder byteArrayObject = toJsonArrayBuilder((byte[]) param); JsonArrayBuilder byteArrayObject = toJsonArrayBuilder((byte[]) param);
jsonObjectBuilder.add(key, byteArrayObject); jsonObjectBuilder.add(key, byteArrayObject);
} else if (param instanceof Object[]) {
final JsonArrayBuilder objectArrayBuilder = JsonLoader.createArrayBuilder();
for (Object parameter : (Object[])param) {
addToArray(parameter, objectArrayBuilder);
}
jsonObjectBuilder.add(key, objectArrayBuilder);
} else { } else {
throw ActiveMQClientMessageBundle.BUNDLE.invalidManagementParam(param.getClass().getName()); throw ActiveMQClientMessageBundle.BUNDLE.invalidManagementParam(param.getClass().getName());
} }
@ -238,6 +200,21 @@ public final class JsonUtil {
} else if (param instanceof byte[]) { } else if (param instanceof byte[]) {
JsonArrayBuilder byteArrayObject = toJsonArrayBuilder((byte[]) param); JsonArrayBuilder byteArrayObject = toJsonArrayBuilder((byte[]) param);
jsonArrayBuilder.add(byteArrayObject); jsonArrayBuilder.add(byteArrayObject);
} else if (param instanceof CompositeData[]) {
JsonArrayBuilder innerJsonArray = JsonLoader.createArrayBuilder();
for (Object data : (CompositeData[])param) {
String s = Base64.encodeObject((CompositeDataSupport) data);
innerJsonArray.add(s);
}
JsonObjectBuilder jsonObject = JsonLoader.createObjectBuilder();
jsonObject.add(CompositeData.class.getName(), innerJsonArray);
jsonArrayBuilder.add(jsonObject);
} else if (param instanceof Object[]) {
JsonArrayBuilder objectArrayBuilder = JsonLoader.createArrayBuilder();
for (Object parameter : (Object[])param) {
addToArray(parameter, objectArrayBuilder);
}
jsonArrayBuilder.add(objectArrayBuilder);
} else { } else {
throw ActiveMQClientMessageBundle.BUNDLE.invalidManagementParam(param.getClass().getName()); throw ActiveMQClientMessageBundle.BUNDLE.invalidManagementParam(param.getClass().getName());
} }
@ -345,6 +322,29 @@ public final class JsonUtil {
private JsonUtil() { private JsonUtil() {
} }
public static Object truncate(final Object value, final int valueSizeLimit) {
Object result = value;
if (valueSizeLimit >= 0) {
if (String.class.equals(value.getClass())) {
String str = (String) value;
if (str.length() > valueSizeLimit) {
result = new StringBuilder(valueSizeLimit + 32).append(str.substring(0, valueSizeLimit)).append(", + ").append(str.length() - valueSizeLimit).append(" more").toString();
}
} else if (value.getClass().isArray()) {
if (byte[].class.equals(value.getClass())) {
if (((byte[]) value).length > valueSizeLimit) {
result = Arrays.copyOfRange((byte[]) value, 0, valueSizeLimit);
}
} else if (char[].class.equals(value.getClass())) {
if (((char[]) value).length > valueSizeLimit) {
result = Arrays.copyOfRange((char[]) value, 0, valueSizeLimit);
}
}
}
}
return result;
}
private static class NullableJsonString implements JsonValue, JsonString { private static class NullableJsonString implements JsonValue, JsonString {
private final String value; private final String value;

View File

@ -708,7 +708,15 @@ public interface Message {
* @return Returns the message in Map form, useful when encoding to JSON * @return Returns the message in Map form, useful when encoding to JSON
*/ */
default Map<String, Object> toMap() { default Map<String, Object> toMap() {
Map map = toPropertyMap(); return toMap(-1);
}
/**
* @return Returns the message in Map form, useful when encoding to JSON
* @param valueSizeLimit that limits [] map values
*/
default Map<String, Object> toMap(int valueSizeLimit) {
Map map = toPropertyMap(valueSizeLimit);
map.put("messageID", getMessageID()); map.put("messageID", getMessageID());
Object userID = getUserID(); Object userID = getUserID();
if (getUserID() != null) { if (getUserID() != null) {
@ -728,6 +736,14 @@ public interface Message {
* @return Returns the message properties in Map form, useful when encoding to JSON * @return Returns the message properties in Map form, useful when encoding to JSON
*/ */
default Map<String, Object> toPropertyMap() { default Map<String, Object> toPropertyMap() {
return toPropertyMap(-1);
}
/**
* @return Returns the message properties in Map form, useful when encoding to JSON
* @param valueSizeLimit that limits [] map values
*/
default Map<String, Object> toPropertyMap(int valueSizeLimit) {
Map map = new HashMap<>(); Map map = new HashMap<>();
for (SimpleString name : getPropertyNames()) { for (SimpleString name : getPropertyNames()) {
Object value = getObjectProperty(name.toString()); Object value = getObjectProperty(name.toString());
@ -735,12 +751,12 @@ public interface Message {
if (value instanceof SimpleString) { if (value instanceof SimpleString) {
value = value.toString(); value = value.toString();
} }
value = JsonUtil.truncate(value, valueSizeLimit);
map.put(name.toString(), value); map.put(name.toString(), value);
} }
return map; return map;
} }
/** This should make you convert your message into Core format. */ /** This should make you convert your message into Core format. */
ICoreMessage toCore(); ICoreMessage toCore();

View File

@ -17,6 +17,7 @@
package org.apache.activemq.artemis.message; package org.apache.activemq.artemis.message;
import java.nio.charset.StandardCharsets;
import java.util.LinkedList; import java.util.LinkedList;
import io.netty.buffer.ByteBuf; import io.netty.buffer.ByteBuf;
@ -211,6 +212,32 @@ public class CoreMessageTest {
} }
} }
@Test
public void testToMapLimit() throws Exception {
CoreMessage coreMessage = new CoreMessage().initBuffer(BIGGER_TEXT.length() * 2);
coreMessage.putStringProperty("prop", BIGGER_TEXT);
coreMessage.putBytesProperty("bytesProp", BIGGER_TEXT.getBytes(StandardCharsets.UTF_8));
Assert.assertEquals(BIGGER_TEXT.length(), ((String)coreMessage.toMap().get("prop")).length());
Assert.assertEquals(BIGGER_TEXT.getBytes(StandardCharsets.UTF_8).length, ((byte[])coreMessage.toMap().get("bytesProp")).length);
// limit the values
Assert.assertNotEquals(BIGGER_TEXT.getBytes(StandardCharsets.UTF_8).length, ((byte[])coreMessage.toMap(40).get("bytesProp")).length);
String mapVal = ((String)coreMessage.toMap(40).get("prop"));
Assert.assertNotEquals(BIGGER_TEXT.length(), mapVal.length());
Assert.assertTrue(mapVal.contains("more"));
mapVal = ((String)coreMessage.toMap(0).get("prop"));
Assert.assertNotEquals(BIGGER_TEXT.length(), mapVal.length());
Assert.assertTrue(mapVal.contains("more"));
Assert.assertEquals(BIGGER_TEXT.length(), Integer.parseInt(mapVal.substring(mapVal.lastIndexOf('+') + 1, mapVal.lastIndexOf('m')).trim()));
Assert.assertEquals(BIGGER_TEXT.getBytes(StandardCharsets.UTF_8).length, ((byte[])coreMessage.toPropertyMap().get("bytesProp")).length);
Assert.assertNotEquals(BIGGER_TEXT.getBytes(StandardCharsets.UTF_8).length, ((byte[])coreMessage.toPropertyMap(40).get("bytesProp")).length);
}
@Test @Test
public void testSaveReceiveLimitedBytes() { public void testSaveReceiveLimitedBytes() {
CoreMessage empty = new CoreMessage().initBuffer(100); CoreMessage empty = new CoreMessage().initBuffer(100);

View File

@ -31,6 +31,7 @@ import org.apache.activemq.artemis.api.core.ActiveMQBuffer;
import org.apache.activemq.artemis.api.core.ActiveMQException; import org.apache.activemq.artemis.api.core.ActiveMQException;
import org.apache.activemq.artemis.api.core.ActiveMQPropertyConversionException; import org.apache.activemq.artemis.api.core.ActiveMQPropertyConversionException;
import org.apache.activemq.artemis.api.core.ICoreMessage; import org.apache.activemq.artemis.api.core.ICoreMessage;
import org.apache.activemq.artemis.api.core.JsonUtil;
import org.apache.activemq.artemis.api.core.RefCountMessage; import org.apache.activemq.artemis.api.core.RefCountMessage;
import org.apache.activemq.artemis.api.core.RoutingType; import org.apache.activemq.artemis.api.core.RoutingType;
import org.apache.activemq.artemis.api.core.SimpleString; import org.apache.activemq.artemis.api.core.SimpleString;
@ -835,7 +836,7 @@ public abstract class AMQPMessage extends RefCountMessage implements org.apache.
public abstract int getMemoryEstimate(); public abstract int getMemoryEstimate();
@Override @Override
public Map<String, Object> toPropertyMap() { public Map<String, Object> toPropertyMap(int valueSizeLimit) {
Map map = new HashMap<>(); Map map = new HashMap<>();
for (SimpleString name : getPropertyNames()) { for (SimpleString name : getPropertyNames()) {
Object value = getObjectProperty(name.toString()); Object value = getObjectProperty(name.toString());
@ -843,6 +844,7 @@ public abstract class AMQPMessage extends RefCountMessage implements org.apache.
if (value instanceof Binary) { if (value instanceof Binary) {
value = ((Binary)value).getArray(); value = ((Binary)value).getArray();
} }
value = JsonUtil.truncate(value, valueSizeLimit);
map.put(name.toString(), value); map.put(name.toString(), value);
} }
return map; return map;

View File

@ -281,6 +281,8 @@ public final class FileConfigurationParser extends XMLConfigurationUtil {
private static final String MANAGEMENT_BROWSE_PAGE_SIZE = "management-browse-page-size"; private static final String MANAGEMENT_BROWSE_PAGE_SIZE = "management-browse-page-size";
private static final String MANAGEMENT_MESSAGE_ATTRIBUTE_SIZE_LIMIT = "management-message-attribute-size-limit";
private static final String MAX_CONNECTIONS_NODE_NAME = "max-connections"; private static final String MAX_CONNECTIONS_NODE_NAME = "max-connections";
private static final String MAX_QUEUES_NODE_NAME = "max-queues"; private static final String MAX_QUEUES_NODE_NAME = "max-queues";
@ -1250,6 +1252,8 @@ public final class FileConfigurationParser extends XMLConfigurationUtil {
addressSettings.setConfigDeleteAddresses(policy); addressSettings.setConfigDeleteAddresses(policy);
} else if (MANAGEMENT_BROWSE_PAGE_SIZE.equalsIgnoreCase(name)) { } else if (MANAGEMENT_BROWSE_PAGE_SIZE.equalsIgnoreCase(name)) {
addressSettings.setManagementBrowsePageSize(XMLUtil.parseInt(child)); addressSettings.setManagementBrowsePageSize(XMLUtil.parseInt(child));
} else if (MANAGEMENT_MESSAGE_ATTRIBUTE_SIZE_LIMIT.equalsIgnoreCase(name)) {
addressSettings.setManagementMessageAttributeSizeLimit(XMLUtil.parseInt(child));
} else if (DEFAULT_PURGE_ON_NO_CONSUMERS.equalsIgnoreCase(name)) { } else if (DEFAULT_PURGE_ON_NO_CONSUMERS.equalsIgnoreCase(name)) {
addressSettings.setDefaultPurgeOnNoConsumers(XMLUtil.parseBoolean(child)); addressSettings.setDefaultPurgeOnNoConsumers(XMLUtil.parseBoolean(child));
} else if (DEFAULT_MAX_CONSUMERS.equalsIgnoreCase(name)) { } else if (DEFAULT_MAX_CONSUMERS.equalsIgnoreCase(name)) {

View File

@ -788,11 +788,12 @@ public class QueueControlImpl extends AbstractControl implements QueueControl {
* @return * @return
*/ */
private Map<String, Object>[] convertMessagesToMaps(List<MessageReference> refs) throws ActiveMQException { private Map<String, Object>[] convertMessagesToMaps(List<MessageReference> refs) throws ActiveMQException {
final int attributeSizeLimit = addressSettingsRepository.getMatch(address).getManagementMessageAttributeSizeLimit();
Map<String, Object>[] messages = new Map[refs.size()]; Map<String, Object>[] messages = new Map[refs.size()];
int i = 0; int i = 0;
for (MessageReference ref : refs) { for (MessageReference ref : refs) {
Message message = ref.getMessage(); Message message = ref.getMessage();
messages[i++] = message.toMap(); messages[i++] = message.toMap(attributeSizeLimit);
} }
return messages; return messages;
} }
@ -847,7 +848,9 @@ public class QueueControlImpl extends AbstractControl implements QueueControl {
Filter filter = FilterImpl.createFilter(filterStr); Filter filter = FilterImpl.createFilter(filterStr);
List<Map<String, Object>> messages = new ArrayList<>(); List<Map<String, Object>> messages = new ArrayList<>();
queue.flushExecutor(); queue.flushExecutor();
final int limit = addressSettingsRepository.getMatch(queue.getAddress().toString()).getManagementBrowsePageSize(); final AddressSettings addressSettings = addressSettingsRepository.getMatch(address);
final int attributeSizeLimit = addressSettings.getManagementMessageAttributeSizeLimit();
final int limit = addressSettings.getManagementBrowsePageSize();
int count = 0; int count = 0;
try (LinkedListIterator<MessageReference> iterator = queue.browserIterator()) { try (LinkedListIterator<MessageReference> iterator = queue.browserIterator()) {
try { try {
@ -855,7 +858,7 @@ public class QueueControlImpl extends AbstractControl implements QueueControl {
MessageReference ref = iterator.next(); MessageReference ref = iterator.next();
if (filter == null || filter.match(ref.getMessage())) { if (filter == null || filter.match(ref.getMessage())) {
Message message = ref.getMessage(); Message message = ref.getMessage();
messages.add(message.toMap()); messages.add(message.toMap(attributeSizeLimit));
} }
} }
} catch (NoSuchElementException ignored) { } catch (NoSuchElementException ignored) {
@ -895,12 +898,13 @@ public class QueueControlImpl extends AbstractControl implements QueueControl {
try { try {
List<Map<String, Object>> messages = new ArrayList<>(); List<Map<String, Object>> messages = new ArrayList<>();
queue.flushExecutor(); queue.flushExecutor();
final int attributeSizeLimit = addressSettingsRepository.getMatch(address).getManagementMessageAttributeSizeLimit();
try (LinkedListIterator<MessageReference> iterator = queue.browserIterator()) { try (LinkedListIterator<MessageReference> iterator = queue.browserIterator()) {
// returns just the first, as it's the first only // returns just the first, as it's the first only
if (iterator.hasNext()) { if (iterator.hasNext()) {
MessageReference ref = iterator.next(); MessageReference ref = iterator.next();
Message message = ref.getMessage(); Message message = ref.getMessage();
messages.add(message.toMap()); messages.add(message.toMap(attributeSizeLimit));
} }
return messages.toArray(new Map[1]); return messages.toArray(new Map[1]);
} }
@ -985,7 +989,7 @@ public class QueueControlImpl extends AbstractControl implements QueueControl {
if (filter == null && groupByProperty == null) { if (filter == null && groupByProperty == null) {
result.put(null, getMessageCount()); result.put(null, getMessageCount());
} else { } else {
final int limit = addressSettingsRepository.getMatch(queue.getAddress().toString()).getManagementBrowsePageSize(); final int limit = addressSettingsRepository.getMatch(address).getManagementBrowsePageSize();
int count = 0; int count = 0;
try (LinkedListIterator<MessageReference> iterator = queue.browserIterator()) { try (LinkedListIterator<MessageReference> iterator = queue.browserIterator()) {
try { try {
@ -1552,13 +1556,14 @@ public class QueueControlImpl extends AbstractControl implements QueueControl {
Filter thefilter = FilterImpl.createFilter(filter); Filter thefilter = FilterImpl.createFilter(filter);
queue.flushExecutor(); queue.flushExecutor();
final int attributeSizeLimit = addressSettingsRepository.getMatch(address).getManagementMessageAttributeSizeLimit();
try (LinkedListIterator<MessageReference> iterator = queue.browserIterator()) { try (LinkedListIterator<MessageReference> iterator = queue.browserIterator()) {
try { try {
while (iterator.hasNext() && index < end) { while (iterator.hasNext() && index < end) {
MessageReference ref = iterator.next(); MessageReference ref = iterator.next();
if (thefilter == null || thefilter.match(ref.getMessage())) { if (thefilter == null || thefilter.match(ref.getMessage())) {
if (index >= start) { if (index >= start) {
c.add(OpenTypeSupport.convert(ref)); c.add(OpenTypeSupport.convert(ref, attributeSizeLimit));
} }
} }
index++; index++;
@ -1597,7 +1602,9 @@ public class QueueControlImpl extends AbstractControl implements QueueControl {
clearIO(); clearIO();
try { try {
int limit = addressSettingsRepository.getMatch(queue.getAddress().toString()).getManagementBrowsePageSize(); final AddressSettings addressSettings = addressSettingsRepository.getMatch(address);
final int attributeSizeLimit = addressSettings.getManagementMessageAttributeSizeLimit();
final int limit = addressSettings.getManagementBrowsePageSize();
int currentPageSize = 0; int currentPageSize = 0;
ArrayList<CompositeData> c = new ArrayList<>(); ArrayList<CompositeData> c = new ArrayList<>();
Filter thefilter = FilterImpl.createFilter(filter); Filter thefilter = FilterImpl.createFilter(filter);
@ -1607,7 +1614,7 @@ public class QueueControlImpl extends AbstractControl implements QueueControl {
while (iterator.hasNext() && currentPageSize++ < limit) { while (iterator.hasNext() && currentPageSize++ < limit) {
MessageReference ref = iterator.next(); MessageReference ref = iterator.next();
if (thefilter == null || thefilter.match(ref.getMessage())) { if (thefilter == null || thefilter.match(ref.getMessage())) {
c.add(OpenTypeSupport.convert(ref)); c.add(OpenTypeSupport.convert(ref, attributeSizeLimit));
} }
} }

View File

@ -34,6 +34,7 @@ import java.util.Map;
import org.apache.activemq.artemis.api.core.ActiveMQBuffer; import org.apache.activemq.artemis.api.core.ActiveMQBuffer;
import org.apache.activemq.artemis.api.core.ActiveMQException; import org.apache.activemq.artemis.api.core.ActiveMQException;
import org.apache.activemq.artemis.api.core.ICoreMessage; import org.apache.activemq.artemis.api.core.ICoreMessage;
import org.apache.activemq.artemis.api.core.JsonUtil;
import org.apache.activemq.artemis.api.core.Message; import org.apache.activemq.artemis.api.core.Message;
import org.apache.activemq.artemis.api.core.SimpleString; import org.apache.activemq.artemis.api.core.SimpleString;
import org.apache.activemq.artemis.core.server.MessageReference; import org.apache.activemq.artemis.core.server.MessageReference;
@ -46,7 +47,7 @@ public final class OpenTypeSupport {
private OpenTypeSupport() { private OpenTypeSupport() {
} }
public static CompositeData convert(MessageReference ref) throws OpenDataException { public static CompositeData convert(MessageReference ref, int valueSizeLimit) throws OpenDataException {
CompositeType ct; CompositeType ct;
ICoreMessage message = ref.getMessage().toCore(); ICoreMessage message = ref.getMessage().toCore();
@ -57,11 +58,11 @@ public final class OpenTypeSupport {
switch(type) { switch(type) {
case Message.TEXT_TYPE: case Message.TEXT_TYPE:
ct = TEXT_FACTORY.getCompositeType(); ct = TEXT_FACTORY.getCompositeType();
fields = TEXT_FACTORY.getFields(ref); fields = TEXT_FACTORY.getFields(ref, valueSizeLimit);
break; break;
default: default:
ct = BYTES_FACTORY.getCompositeType(); ct = BYTES_FACTORY.getCompositeType();
fields = BYTES_FACTORY.getFields(ref); fields = BYTES_FACTORY.getFields(ref, valueSizeLimit);
break; break;
} }
return new CompositeDataSupport(ct, fields); return new CompositeDataSupport(ct, fields);
@ -82,6 +83,7 @@ public final class OpenTypeSupport {
protected TabularType longPropertyTabularType; protected TabularType longPropertyTabularType;
protected TabularType floatPropertyTabularType; protected TabularType floatPropertyTabularType;
protected TabularType doublePropertyTabularType; protected TabularType doublePropertyTabularType;
protected Object[][] typedPropertyFields;
protected String getTypeName() { protected String getTypeName() {
return Message.class.getName(); return Message.class.getName();
@ -129,9 +131,21 @@ public final class OpenTypeSupport {
addItem(CompositeDataConstants.LONG_PROPERTIES, CompositeDataConstants.LONG_PROPERTIES_DESCRIPTION, longPropertyTabularType); addItem(CompositeDataConstants.LONG_PROPERTIES, CompositeDataConstants.LONG_PROPERTIES_DESCRIPTION, longPropertyTabularType);
addItem(CompositeDataConstants.FLOAT_PROPERTIES, CompositeDataConstants.FLOAT_PROPERTIES_DESCRIPTION, floatPropertyTabularType); addItem(CompositeDataConstants.FLOAT_PROPERTIES, CompositeDataConstants.FLOAT_PROPERTIES_DESCRIPTION, floatPropertyTabularType);
addItem(CompositeDataConstants.DOUBLE_PROPERTIES, CompositeDataConstants.DOUBLE_PROPERTIES_DESCRIPTION, doublePropertyTabularType); addItem(CompositeDataConstants.DOUBLE_PROPERTIES, CompositeDataConstants.DOUBLE_PROPERTIES_DESCRIPTION, doublePropertyTabularType);
typedPropertyFields = new Object[][] {
{CompositeDataConstants.STRING_PROPERTIES, stringPropertyTabularType, String.class},
{CompositeDataConstants.BOOLEAN_PROPERTIES, booleanPropertyTabularType, Boolean.class},
{CompositeDataConstants.BYTE_PROPERTIES, bytePropertyTabularType, Byte.class},
{CompositeDataConstants.SHORT_PROPERTIES, shortPropertyTabularType, Short.class},
{CompositeDataConstants.INT_PROPERTIES, intPropertyTabularType, Integer.class},
{CompositeDataConstants.LONG_PROPERTIES, longPropertyTabularType, Long.class},
{CompositeDataConstants.FLOAT_PROPERTIES, floatPropertyTabularType, Float.class},
{CompositeDataConstants.DOUBLE_PROPERTIES, doublePropertyTabularType, Double.class}
};
} }
public Map<String, Object> getFields(MessageReference ref) throws OpenDataException { public Map<String, Object> getFields(MessageReference ref, int valueSizeLimit) throws OpenDataException {
Map<String, Object> rc = new HashMap<>(); Map<String, Object> rc = new HashMap<>();
ICoreMessage m = ref.getMessage().toCore(); ICoreMessage m = ref.getMessage().toCore();
rc.put(CompositeDataConstants.MESSAGE_ID, "" + m.getMessageID()); rc.put(CompositeDataConstants.MESSAGE_ID, "" + m.getMessageID());
@ -154,49 +168,23 @@ public final class OpenTypeSupport {
rc.put(CompositeDataConstants.PERSISTENT_SIZE, -1); rc.put(CompositeDataConstants.PERSISTENT_SIZE, -1);
} }
Map<String, Object> propertyMap = m.toPropertyMap(); Map<String, Object> propertyMap = m.toPropertyMap(valueSizeLimit);
rc.put(CompositeDataConstants.PROPERTIES, "" + propertyMap); rc.put(CompositeDataConstants.PROPERTIES, JsonUtil.truncate("" + propertyMap, valueSizeLimit));
// only populate if there are some values
TabularDataSupport tabularData;
for (Object[] typedPropertyInfo : typedPropertyFields) {
tabularData = null;
try { try {
rc.put(CompositeDataConstants.STRING_PROPERTIES, createTabularData(propertyMap, stringPropertyTabularType, String.class)); tabularData = createTabularData(propertyMap, (TabularType) typedPropertyInfo[1], (Class) typedPropertyInfo[2]);
} catch (IOException e) { } catch (Exception ignored) {
rc.put(CompositeDataConstants.STRING_PROPERTIES, new TabularDataSupport(stringPropertyTabularType));
} }
try { if (tabularData != null && !tabularData.isEmpty()) {
rc.put(CompositeDataConstants.BOOLEAN_PROPERTIES, createTabularData(propertyMap, booleanPropertyTabularType, Boolean.class)); rc.put((String) typedPropertyInfo[0], tabularData);
} catch (IOException e) { } else {
rc.put(CompositeDataConstants.BOOLEAN_PROPERTIES, new TabularDataSupport(booleanPropertyTabularType)); rc.put((String) typedPropertyInfo[0], null);
} }
try {
rc.put(CompositeDataConstants.BYTE_PROPERTIES, createTabularData(propertyMap, bytePropertyTabularType, Byte.class));
} catch (IOException e) {
rc.put(CompositeDataConstants.BYTE_PROPERTIES, new TabularDataSupport(bytePropertyTabularType));
}
try {
rc.put(CompositeDataConstants.SHORT_PROPERTIES, createTabularData(propertyMap, shortPropertyTabularType, Short.class));
} catch (IOException e) {
rc.put(CompositeDataConstants.SHORT_PROPERTIES, new TabularDataSupport(shortPropertyTabularType));
}
try {
rc.put(CompositeDataConstants.INT_PROPERTIES, createTabularData(propertyMap, intPropertyTabularType, Integer.class));
} catch (IOException e) {
rc.put(CompositeDataConstants.INT_PROPERTIES, new TabularDataSupport(intPropertyTabularType));
}
try {
rc.put(CompositeDataConstants.LONG_PROPERTIES, createTabularData(propertyMap, longPropertyTabularType, Long.class));
} catch (IOException e) {
rc.put(CompositeDataConstants.LONG_PROPERTIES, new TabularDataSupport(longPropertyTabularType));
}
try {
rc.put(CompositeDataConstants.FLOAT_PROPERTIES, createTabularData(propertyMap, floatPropertyTabularType, Float.class));
} catch (IOException e) {
rc.put(CompositeDataConstants.FLOAT_PROPERTIES, new TabularDataSupport(floatPropertyTabularType));
}
try {
rc.put(CompositeDataConstants.DOUBLE_PROPERTIES, createTabularData(propertyMap, doublePropertyTabularType, Double.class));
} catch (IOException e) {
rc.put(CompositeDataConstants.DOUBLE_PROPERTIES, new TabularDataSupport(doublePropertyTabularType));
} }
return rc; return rc;
} }
@ -273,14 +261,14 @@ public final class OpenTypeSupport {
} }
@Override @Override
public Map<String, Object> getFields(MessageReference ref) throws OpenDataException { public Map<String, Object> getFields(MessageReference ref, int valueSizeLimit) throws OpenDataException {
Map<String, Object> rc = super.getFields(ref); Map<String, Object> rc = super.getFields(ref, valueSizeLimit);
ICoreMessage m = ref.getMessage().toCore(); ICoreMessage m = ref.getMessage().toCore();
if (!m.isLargeMessage()) { if (!m.isLargeMessage()) {
ActiveMQBuffer bodyCopy = m.getReadOnlyBodyBuffer(); ActiveMQBuffer bodyCopy = m.getReadOnlyBodyBuffer();
byte[] bytes = new byte[bodyCopy.readableBytes()]; byte[] bytes = new byte[bodyCopy.readableBytes() <= valueSizeLimit ? bodyCopy.readableBytes() : valueSizeLimit + 1];
bodyCopy.readBytes(bytes); bodyCopy.readBytes(bytes);
rc.put(CompositeDataConstants.BODY, bytes); rc.put(CompositeDataConstants.BODY, JsonUtil.truncate(bytes, valueSizeLimit));
} else { } else {
rc.put(CompositeDataConstants.BODY, new byte[0]); rc.put(CompositeDataConstants.BODY, new byte[0]);
} }
@ -298,15 +286,15 @@ public final class OpenTypeSupport {
} }
@Override @Override
public Map<String, Object> getFields(MessageReference ref) throws OpenDataException { public Map<String, Object> getFields(MessageReference ref, int valueSizeLimit) throws OpenDataException {
Map<String, Object> rc = super.getFields(ref); Map<String, Object> rc = super.getFields(ref, valueSizeLimit);
ICoreMessage m = ref.getMessage().toCore(); ICoreMessage m = ref.getMessage().toCore();
if (!m.isLargeMessage()) { if (!m.isLargeMessage()) {
if (m.containsProperty(Message.HDR_LARGE_COMPRESSED)) { if (m.containsProperty(Message.HDR_LARGE_COMPRESSED)) {
rc.put(CompositeDataConstants.TEXT_BODY, "[compressed]"); rc.put(CompositeDataConstants.TEXT_BODY, "[compressed]");
} else { } else {
SimpleString text = m.getReadOnlyBodyBuffer().readNullableSimpleString(); final String text = m.getReadOnlyBodyBuffer().readString();
rc.put(CompositeDataConstants.TEXT_BODY, text != null ? text.toString() : ""); rc.put(CompositeDataConstants.TEXT_BODY, text != null ? JsonUtil.truncate(text, valueSizeLimit) : "");
} }
} else { } else {
rc.put(CompositeDataConstants.TEXT_BODY, "[large message]"); rc.put(CompositeDataConstants.TEXT_BODY, "[large message]");

View File

@ -127,6 +127,8 @@ public class AddressSettings implements Mergeable<AddressSettings>, Serializable
public static final boolean DEFAULT_ENABLE_METRICS = true; public static final boolean DEFAULT_ENABLE_METRICS = true;
public static final int MANAGEMENT_MESSAGE_ATTRIBUTE_SIZE_LIMIT = 256;
private AddressFullMessagePolicy addressFullMessagePolicy = null; private AddressFullMessagePolicy addressFullMessagePolicy = null;
private Long maxSizeBytes = null; private Long maxSizeBytes = null;
@ -253,6 +255,8 @@ public class AddressSettings implements Mergeable<AddressSettings>, Serializable
private Boolean enableMetrics = null; private Boolean enableMetrics = null;
private Integer managementMessageAttributeSizeLimit = null;
//from amq5 //from amq5
//make it transient //make it transient
private transient Integer queuePrefetch = null; private transient Integer queuePrefetch = null;
@ -318,6 +322,7 @@ public class AddressSettings implements Mergeable<AddressSettings>, Serializable
this.defaultGroupFirstKey = other.defaultGroupFirstKey; this.defaultGroupFirstKey = other.defaultGroupFirstKey;
this.defaultRingSize = other.defaultRingSize; this.defaultRingSize = other.defaultRingSize;
this.enableMetrics = other.enableMetrics; this.enableMetrics = other.enableMetrics;
this.managementMessageAttributeSizeLimit = other.managementMessageAttributeSizeLimit;
} }
public AddressSettings() { public AddressSettings() {
@ -914,6 +919,15 @@ public class AddressSettings implements Mergeable<AddressSettings>, Serializable
return this; return this;
} }
public int getManagementMessageAttributeSizeLimit() {
return managementMessageAttributeSizeLimit != null ? managementMessageAttributeSizeLimit : AddressSettings.MANAGEMENT_MESSAGE_ATTRIBUTE_SIZE_LIMIT;
}
public AddressSettings setManagementMessageAttributeSizeLimit(int managementMessageAttributeSizeLimit) {
this.managementMessageAttributeSizeLimit = managementMessageAttributeSizeLimit;
return this;
}
/** /**
* merge 2 objects in to 1 * merge 2 objects in to 1
* *
@ -1029,6 +1043,9 @@ public class AddressSettings implements Mergeable<AddressSettings>, Serializable
if (managementBrowsePageSize == null) { if (managementBrowsePageSize == null) {
managementBrowsePageSize = merged.managementBrowsePageSize; managementBrowsePageSize = merged.managementBrowsePageSize;
} }
if (managementMessageAttributeSizeLimit == null) {
managementMessageAttributeSizeLimit = merged.managementMessageAttributeSizeLimit;
}
if (queuePrefetch == null) { if (queuePrefetch == null) {
queuePrefetch = merged.queuePrefetch; queuePrefetch = merged.queuePrefetch;
} }
@ -1320,6 +1337,10 @@ public class AddressSettings implements Mergeable<AddressSettings>, Serializable
defaultGroupRebalancePauseDispatch = BufferHelper.readNullableBoolean(buffer); defaultGroupRebalancePauseDispatch = BufferHelper.readNullableBoolean(buffer);
} }
if (buffer.readableBytes() > 0) {
managementMessageAttributeSizeLimit = BufferHelper.readNullableInteger(buffer);
}
} }
@Override @Override
@ -1383,7 +1404,8 @@ public class AddressSettings implements Mergeable<AddressSettings>, Serializable
SimpleString.sizeofNullableString(expiryQueuePrefix) + SimpleString.sizeofNullableString(expiryQueuePrefix) +
SimpleString.sizeofNullableString(expiryQueueSuffix) + SimpleString.sizeofNullableString(expiryQueueSuffix) +
BufferHelper.sizeOfNullableBoolean(enableMetrics) + BufferHelper.sizeOfNullableBoolean(enableMetrics) +
BufferHelper.sizeOfNullableBoolean(defaultGroupRebalancePauseDispatch); BufferHelper.sizeOfNullableBoolean(defaultGroupRebalancePauseDispatch) +
BufferHelper.sizeOfNullableInteger(managementMessageAttributeSizeLimit);
} }
@Override @Override
@ -1510,6 +1532,7 @@ public class AddressSettings implements Mergeable<AddressSettings>, Serializable
BufferHelper.writeNullableBoolean(buffer, defaultGroupRebalancePauseDispatch); BufferHelper.writeNullableBoolean(buffer, defaultGroupRebalancePauseDispatch);
BufferHelper.writeNullableInteger(buffer, managementMessageAttributeSizeLimit);
} }
/* (non-Javadoc) /* (non-Javadoc)
@ -1581,6 +1604,7 @@ public class AddressSettings implements Mergeable<AddressSettings>, Serializable
result = prime * result + ((expiryQueuePrefix == null) ? 0 : expiryQueuePrefix.hashCode()); result = prime * result + ((expiryQueuePrefix == null) ? 0 : expiryQueuePrefix.hashCode());
result = prime * result + ((expiryQueueSuffix == null) ? 0 : expiryQueueSuffix.hashCode()); result = prime * result + ((expiryQueueSuffix == null) ? 0 : expiryQueueSuffix.hashCode());
result = prime * result + ((enableMetrics == null) ? 0 : enableMetrics.hashCode()); result = prime * result + ((enableMetrics == null) ? 0 : enableMetrics.hashCode());
result = prime * result + ((managementMessageAttributeSizeLimit == null) ? 0 : managementMessageAttributeSizeLimit.hashCode());
return result; return result;
} }
@ -1796,6 +1820,11 @@ public class AddressSettings implements Mergeable<AddressSettings>, Serializable
return false; return false;
} else if (!managementBrowsePageSize.equals(other.managementBrowsePageSize)) } else if (!managementBrowsePageSize.equals(other.managementBrowsePageSize))
return false; return false;
if (managementMessageAttributeSizeLimit == null) {
if (other.managementMessageAttributeSizeLimit != null)
return false;
} else if (!managementMessageAttributeSizeLimit.equals(other.managementMessageAttributeSizeLimit))
return false;
if (queuePrefetch == null) { if (queuePrefetch == null) {
if (other.queuePrefetch != null) if (other.queuePrefetch != null)
return false; return false;
@ -2017,6 +2046,8 @@ public class AddressSettings implements Mergeable<AddressSettings>, Serializable
configDeleteAddresses + configDeleteAddresses +
", managementBrowsePageSize=" + ", managementBrowsePageSize=" +
managementBrowsePageSize + managementBrowsePageSize +
", managementMessageAttributeSizeLimit=" +
managementMessageAttributeSizeLimit +
", defaultMaxConsumers=" + ", defaultMaxConsumers=" +
defaultMaxConsumers + defaultMaxConsumers +
", defaultPurgeOnNoConsumers=" + ", defaultPurgeOnNoConsumers=" +

View File

@ -56,6 +56,6 @@ public class CoreTransactionDetail extends TransactionDetail {
@Override @Override
public Map<String, Object> decodeMessageProperties(Message msg) { public Map<String, Object> decodeMessageProperties(Message msg) {
return msg.toMap(); return msg.toMap(256);
} }
} }

View File

@ -3804,7 +3804,15 @@
<xsd:element name="management-browse-page-size" type="xsd:int" default="200" maxOccurs="1" minOccurs="0"> <xsd:element name="management-browse-page-size" type="xsd:int" default="200" maxOccurs="1" minOccurs="0">
<xsd:annotation> <xsd:annotation>
<xsd:documentation> <xsd:documentation>
how many message a management resource can browse how many message a management resource can browse, list or filter
</xsd:documentation>
</xsd:annotation>
</xsd:element>
<xsd:element name="management-message-attribute-size-limit" type="xsd:int" default="256" maxOccurs="1" minOccurs="0">
<xsd:annotation>
<xsd:documentation>
the size limit of any message attribute value returned from a browse ,list or filter. Attribute values that exceed with be truncated
</xsd:documentation> </xsd:documentation>
</xsd:annotation> </xsd:annotation>
</xsd:element> </xsd:element>

View File

@ -505,6 +505,8 @@
<default-consumer-window-size>10000</default-consumer-window-size> <default-consumer-window-size>10000</default-consumer-window-size>
<retroactive-message-count>10</retroactive-message-count> <retroactive-message-count>10</retroactive-message-count>
<enable-metrics>false</enable-metrics> <enable-metrics>false</enable-metrics>
<management-browse-page-size>400</management-browse-page-size>
<management-message-attribute-size-limit>265</management-message-attribute-size-limit>
</address-setting> </address-setting>
</address-settings> </address-settings>
<resource-limit-settings> <resource-limit-settings>

View File

@ -39,6 +39,7 @@ import javax.jms.Session;
import javax.jms.TextMessage; import javax.jms.TextMessage;
import javax.json.JsonArray; import javax.json.JsonArray;
import javax.json.JsonObject; import javax.json.JsonObject;
import java.nio.charset.StandardCharsets;
import java.util.Map; import java.util.Map;
public class JMXManagementTest extends JMSClientTestSupport { public class JMXManagementTest extends JMSClientTestSupport {
@ -107,8 +108,11 @@ public class JMXManagementTest extends JMSClientTestSupport {
session.begin(); session.begin();
AmqpMessage message = new AmqpMessage(); AmqpMessage message = new AmqpMessage();
message.setApplicationProperty("TEST_BINARY", new Binary("TEST".getBytes())); message.setApplicationProperty("TEST_BINARY", new Binary("TEST".getBytes()));
message.setApplicationProperty("TEST_STRING", "TEST");
message.setText("TEST"); final String oneK = new String(new char[1024]).replace("\0", "$");
message.setApplicationProperty("TEST_BIG_BINARY", new Binary(oneK.getBytes(StandardCharsets.UTF_8)));
message.setApplicationProperty("TEST_STRING", oneK);
message.setText("NOT_VISIBLE");
sender.send(message); sender.send(message);
session.commit(); session.commit();
@ -116,6 +120,18 @@ public class JMXManagementTest extends JMSClientTestSupport {
QueueControl queueControl = createManagementControl(queue, queue); QueueControl queueControl = createManagementControl(queue, queue);
String firstMessageAsJSON = queueControl.getFirstMessageAsJSON(); String firstMessageAsJSON = queueControl.getFirstMessageAsJSON();
Assert.assertNotNull(firstMessageAsJSON); Assert.assertNotNull(firstMessageAsJSON);
// Json is still bulky!
Assert.assertTrue(firstMessageAsJSON.length() < 1500);
Assert.assertFalse(firstMessageAsJSON.contains("NOT_VISIBLE"));
// composite data limits
Map<String, Object>[] result = queueControl.listMessages("");
assertEquals(1, result.length);
final Map<String, Object> msgMap = result[0];
Assert.assertTrue(msgMap.get("TEST_STRING").toString().length() < 512);
} finally { } finally {
connection.close(); connection.close();
} }

View File

@ -488,6 +488,122 @@ public class QueueControlTest extends ManagementTestBase {
session.deleteQueue(queue); session.deleteQueue(queue);
} }
@Test
public void testMessageAttributeLimits() throws Exception {
SimpleString address = RandomUtil.randomSimpleString();
SimpleString queue = RandomUtil.randomSimpleString();
AddressSettings addressSettings = new AddressSettings().setManagementMessageAttributeSizeLimit(100);
server.getAddressSettingsRepository().addMatch(address.toString(), addressSettings);
session.createQueue(new QueueConfiguration(queue).setAddress(address).setDurable(durable));
byte[] twoKBytes = new byte[2048];
for (int i = 0; i < 2048; i++) {
twoKBytes[i] = '*';
}
String twoKString = new String(twoKBytes);
ClientMessage clientMessage = session.createMessage(false);
clientMessage.putStringProperty("y", "valueY");
clientMessage.putStringProperty("bigString", twoKString);
clientMessage.putBytesProperty("bigBytes", twoKBytes);
clientMessage.putObjectProperty("bigObject", twoKString);
clientMessage.getBodyBuffer().writeBytes(twoKBytes);
QueueControl queueControl = createManagementControl(address, queue);
Assert.assertEquals(0, getMessageCount(queueControl));
ClientProducer producer = session.createProducer(address);
producer.send(clientMessage);
Wait.assertEquals(1, () -> getMessageCount(queueControl));
assertTrue(server.getPagingManager().getPageStore(address).getAddressSize() > 2048);
Map<String, Object>[] messages = queueControl.listMessages("");
assertEquals(1, messages.length);
for (String key : messages[0].keySet()) {
Object value = messages[0].get(key);
System.err.println( key + " " + value);
assertTrue(value.toString().length() <= 150);
if (value instanceof byte[]) {
assertTrue(((byte[])value).length <= 150);
}
}
String all = queueControl.listMessagesAsJSON("");
assertTrue(all.length() < 1024);
String first = queueControl.getFirstMessageAsJSON();
assertTrue(first.length() < 1024);
CompositeData[] browseResult = queueControl.browse(1, 100);
for (CompositeData compositeData : browseResult) {
for (String key : compositeData.getCompositeType().keySet()) {
Object value = compositeData.get(key);
System.err.println("" + key + ", " + value);
if (value != null) {
if (key.equals("StringProperties")) {
// these are very verbose composite data structures
assertTrue(value.toString().length() + " truncated? " + key, value.toString().length() <= 2048);
} else {
assertTrue(value.toString().length() + " truncated? " + key, value.toString().length() <= 512);
}
if (value instanceof byte[]) {
assertTrue("truncated? " + key, ((byte[]) value).length <= 150);
}
}
}
}
session.deleteQueue(queue);
}
@Test
public void testTextMessageAttributeLimits() throws Exception {
SimpleString address = RandomUtil.randomSimpleString();
SimpleString queue = RandomUtil.randomSimpleString();
AddressSettings addressSettings = new AddressSettings().setManagementMessageAttributeSizeLimit(10);
server.getAddressSettingsRepository().addMatch(address.toString(), addressSettings);
session.createQueue(new QueueConfiguration(queue).setAddress(address).setDurable(durable));
final String twentyBytes = new String(new char[20]).replace("\0", "#");
ClientMessage clientMessage = createTextMessage(session, twentyBytes, true);
clientMessage.putStringProperty("x", twentyBytes);
QueueControl queueControl = createManagementControl(address, queue);
Assert.assertEquals(0, getMessageCount(queueControl));
ClientProducer producer = session.createProducer(address);
producer.send(clientMessage);
Wait.assertEquals(1, () -> getMessageCount(queueControl));
Map<String, Object>[] messages = queueControl.listMessages("");
assertEquals(1, messages.length);
assertTrue("truncated? ", ((String)messages[0].get("x")).contains("more"));
CompositeData[] browseResult = queueControl.browse(1, 100);
for (CompositeData compositeData : browseResult) {
for (String key : new String[] {"text", "PropertiesText", "StringProperties"}) {
assertTrue("truncated? : " + key, compositeData.get(key).toString().contains("more"));
}
}
session.deleteQueue(queue);
}
@Test @Test
public void testGetMessagesAdded() throws Exception { public void testGetMessagesAdded() throws Exception {
SimpleString address = RandomUtil.randomSimpleString(); SimpleString address = RandomUtil.randomSimpleString();
@ -1082,7 +1198,7 @@ public class QueueControlTest extends ManagementTestBase {
QueueControl queueControl = createManagementControl(address, queue); QueueControl queueControl = createManagementControl(address, queue);
ClientProducer producer = session.createProducer(address); ClientProducer producer = session.createProducer(address);
producer.send(session.createMessage(durable)); producer.send(session.createMessage(durable).putBytesProperty("bytes", new byte[]{'%'}));
producer.send(session.createMessage(durable)); producer.send(session.createMessage(durable));
Wait.assertEquals(2, () -> queueControl.listMessages(null).length); Wait.assertEquals(2, () -> queueControl.listMessages(null).length);