MutableMessageHeaders serializes itself as regular MessageHeaders

Issue: SPR-15262
This commit is contained in:
Juergen Hoeller 2017-02-17 20:59:17 +01:00
parent 692e432f0b
commit 74bdcd8ed4
2 changed files with 48 additions and 54 deletions

View File

@ -1,5 +1,5 @@
/* /*
* Copyright 2002-2016 the original author or authors. * Copyright 2002-2017 the original author or authors.
* *
* Licensed under the Apache License, Version 2.0 (the "License"); * Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. * you may not use this file except in compliance with the License.
@ -136,19 +136,15 @@ public class MessageHeaderAccessor {
* A constructor to create new headers. * A constructor to create new headers.
*/ */
public MessageHeaderAccessor() { public MessageHeaderAccessor() {
this.headers = new MutableMessageHeaders(); this(null);
} }
/** /**
* A constructor accepting the headers of an existing message to copy. * A constructor accepting the headers of an existing message to copy.
* @param message a message to copy the headers from, or {@code null} if none
*/ */
public MessageHeaderAccessor(Message<?> message) { public MessageHeaderAccessor(Message<?> message) {
if (message != null) { this.headers = new MutableMessageHeaders(message != null ? message.getHeaders() : null);
this.headers = new MutableMessageHeaders(message.getHeaders());
}
else {
this.headers = new MutableMessageHeaders();
}
} }
@ -193,7 +189,6 @@ public class MessageHeaderAccessor {
* @since 4.1 * @since 4.1
*/ */
public void setImmutable() { public void setImmutable() {
this.headers.setIdAndTimestamp();
this.headers.setImmutable(); this.headers.setImmutable();
} }
@ -583,7 +578,7 @@ public class MessageHeaderAccessor {
if (messageHeaders instanceof MutableMessageHeaders) { if (messageHeaders instanceof MutableMessageHeaders) {
MutableMessageHeaders mutableHeaders = (MutableMessageHeaders) messageHeaders; MutableMessageHeaders mutableHeaders = (MutableMessageHeaders) messageHeaders;
MessageHeaderAccessor headerAccessor = mutableHeaders.getMessageHeaderAccessor(); MessageHeaderAccessor headerAccessor = mutableHeaders.getAccessor();
if (requiredType.isAssignableFrom(headerAccessor.getClass())) { if (requiredType.isAssignableFrom(headerAccessor.getClass())) {
return (T) headerAccessor; return (T) headerAccessor;
} }
@ -603,7 +598,7 @@ public class MessageHeaderAccessor {
public static MessageHeaderAccessor getMutableAccessor(Message<?> message) { public static MessageHeaderAccessor getMutableAccessor(Message<?> message) {
if (message.getHeaders() instanceof MutableMessageHeaders) { if (message.getHeaders() instanceof MutableMessageHeaders) {
MutableMessageHeaders mutableHeaders = (MutableMessageHeaders) message.getHeaders(); MutableMessageHeaders mutableHeaders = (MutableMessageHeaders) message.getHeaders();
MessageHeaderAccessor accessor = mutableHeaders.getMessageHeaderAccessor(); MessageHeaderAccessor accessor = mutableHeaders.getAccessor();
if (accessor != null) { if (accessor != null) {
return (accessor.isMutable() ? accessor : accessor.createAccessor(message)); return (accessor.isMutable() ? accessor : accessor.createAccessor(message));
} }
@ -615,38 +610,23 @@ public class MessageHeaderAccessor {
@SuppressWarnings("serial") @SuppressWarnings("serial")
private class MutableMessageHeaders extends MessageHeaders { private class MutableMessageHeaders extends MessageHeaders {
private boolean immutable; private boolean mutable = true;
public MutableMessageHeaders() {
this(null);
}
public MutableMessageHeaders(Map<String, Object> headers) { public MutableMessageHeaders(Map<String, Object> headers) {
super(headers, MessageHeaders.ID_VALUE_NONE, -1L); super(headers, MessageHeaders.ID_VALUE_NONE, -1L);
} }
public MessageHeaderAccessor getMessageHeaderAccessor() {
return MessageHeaderAccessor.this;
}
@Override @Override
public Map<String, Object> getRawHeaders() { public Map<String, Object> getRawHeaders() {
Assert.state(!this.immutable, "Already immutable"); Assert.state(this.mutable, "Already immutable");
return super.getRawHeaders(); return super.getRawHeaders();
} }
public void setImmutable() { public void setImmutable() {
this.immutable = true; if (!this.mutable) {
}
public boolean isMutable() {
return !this.immutable;
}
public void setIdAndTimestamp() {
if (!isMutable()) {
return; return;
} }
if (getId() == null) { if (getId() == null) {
IdGenerator idGenerator = (MessageHeaderAccessor.this.idGenerator != null ? IdGenerator idGenerator = (MessageHeaderAccessor.this.idGenerator != null ?
MessageHeaderAccessor.this.idGenerator : MessageHeaders.getIdGenerator()); MessageHeaderAccessor.this.idGenerator : MessageHeaders.getIdGenerator());
@ -655,11 +635,27 @@ public class MessageHeaderAccessor {
getRawHeaders().put(ID, id); getRawHeaders().put(ID, id);
} }
} }
if (getTimestamp() == null) { if (getTimestamp() == null) {
if (MessageHeaderAccessor.this.enableTimestamp) { if (MessageHeaderAccessor.this.enableTimestamp) {
getRawHeaders().put(TIMESTAMP, System.currentTimeMillis()); getRawHeaders().put(TIMESTAMP, System.currentTimeMillis());
} }
} }
this.mutable = false;
}
public boolean isMutable() {
return this.mutable;
}
public MessageHeaderAccessor getAccessor() {
return MessageHeaderAccessor.this;
}
protected Object writeReplace() {
// Serialize as regular MessageHeaders (without MessageHeaderAccessor reference)
return new MessageHeaders(this);
} }
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright 2002-2016 the original author or authors. * Copyright 2002-2017 the original author or authors.
* *
* Licensed under the Apache License, Version 2.0 (the "License"); * Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. * you may not use this file except in compliance with the License.
@ -17,10 +17,8 @@
package org.springframework.messaging.support; package org.springframework.messaging.support;
import java.nio.charset.StandardCharsets; import java.nio.charset.StandardCharsets;
import java.util.Arrays;
import java.util.Collections; import java.util.Collections;
import java.util.HashMap; import java.util.HashMap;
import java.util.HashSet;
import java.util.Map; import java.util.Map;
import java.util.UUID; import java.util.UUID;
@ -30,8 +28,8 @@ import org.junit.rules.ExpectedException;
import org.springframework.messaging.Message; import org.springframework.messaging.Message;
import org.springframework.messaging.MessageHeaders; import org.springframework.messaging.MessageHeaders;
import org.springframework.util.IdGenerator;
import org.springframework.util.MimeTypeUtils; import org.springframework.util.MimeTypeUtils;
import org.springframework.util.SerializationTestUtils;
import static org.hamcrest.CoreMatchers.*; import static org.hamcrest.CoreMatchers.*;
import static org.junit.Assert.*; import static org.junit.Assert.*;
@ -166,7 +164,7 @@ public class MessageHeaderAccessorTests {
headers.copyHeadersIfAbsent(null); headers.copyHeadersIfAbsent(null);
assertEquals(1, headers.getMessageHeaders().size()); assertEquals(1, headers.getMessageHeaders().size());
assertEquals(new HashSet<>(Arrays.asList("id")), headers.getMessageHeaders().keySet()); assertEquals(Collections.singleton("id"), headers.getMessageHeaders().keySet());
} }
@Test @Test
@ -280,12 +278,7 @@ public class MessageHeaderAccessorTests {
public void idGeneratorCustom() { public void idGeneratorCustom() {
final UUID id = new UUID(0L, 23L); final UUID id = new UUID(0L, 23L);
MessageHeaderAccessor accessor = new MessageHeaderAccessor(); MessageHeaderAccessor accessor = new MessageHeaderAccessor();
accessor.setIdGenerator(new IdGenerator() { accessor.setIdGenerator(() -> id);
@Override
public UUID generateId() {
return id;
}
});
assertSame(id, accessor.getMessageHeaders().getId()); assertSame(id, accessor.getMessageHeaders().getId());
} }
@ -299,12 +292,7 @@ public class MessageHeaderAccessorTests {
@Test @Test
public void idTimestampWithMutableHeaders() { public void idTimestampWithMutableHeaders() {
MessageHeaderAccessor accessor = new MessageHeaderAccessor(); MessageHeaderAccessor accessor = new MessageHeaderAccessor();
accessor.setIdGenerator(new IdGenerator() { accessor.setIdGenerator(() -> MessageHeaders.ID_VALUE_NONE);
@Override
public UUID generateId() {
return MessageHeaders.ID_VALUE_NONE;
}
});
accessor.setEnableTimestamp(false); accessor.setEnableTimestamp(false);
accessor.setLeaveMutable(true); accessor.setLeaveMutable(true);
MessageHeaders headers = accessor.getMessageHeaders(); MessageHeaders headers = accessor.getMessageHeaders();
@ -313,12 +301,7 @@ public class MessageHeaderAccessorTests {
assertNull(headers.getTimestamp()); assertNull(headers.getTimestamp());
final UUID id = new UUID(0L, 23L); final UUID id = new UUID(0L, 23L);
accessor.setIdGenerator(new IdGenerator() { accessor.setIdGenerator(() -> id);
@Override
public UUID generateId() {
return id;
}
});
accessor.setEnableTimestamp(true); accessor.setEnableTimestamp(true);
accessor.setImmutable(); accessor.setImmutable();
@ -396,10 +379,25 @@ public class MessageHeaderAccessorTests {
assertEquals("headers={contentType=text/plain} payload=" + sb + " > 80", actual); assertEquals("headers={contentType=text/plain} payload=" + sb + " > 80", actual);
} }
@Test
public void serializeMutableHeaders() throws Exception {
Map<String, Object> headers = new HashMap<>();
headers.put("foo", "bar");
Message<String> message = new GenericMessage<>("test", headers);
MessageHeaderAccessor mutableAccessor = MessageHeaderAccessor.getMutableAccessor(message);
mutableAccessor.setContentType(MimeTypeUtils.TEXT_PLAIN);
message = new GenericMessage<>(message.getPayload(), mutableAccessor.getMessageHeaders());
Message<?> output = (Message<?>) SerializationTestUtils.serializeAndDeserialize(message);
assertEquals("test", output.getPayload());
assertEquals("bar", output.getHeaders().get("foo"));
assertNotNull(output.getHeaders().get(MessageHeaders.CONTENT_TYPE));
}
public static class TestMessageHeaderAccessor extends MessageHeaderAccessor { public static class TestMessageHeaderAccessor extends MessageHeaderAccessor {
private TestMessageHeaderAccessor() { public TestMessageHeaderAccessor() {
} }
private TestMessageHeaderAccessor(Message<?> message) { private TestMessageHeaderAccessor(Message<?> message) {