This sample demonstrates how Citrus handles binary message content. The sample send some binary content to a JMS queue destination and receives that same content in a next step from the JMS destination. Read more about this in reference guide
We demonstrate the binary content handling by using binary JMS messages.
The Citrus project needs a JMS connection factory that is defined in the Spring application context as bean:
@Bean
public JmsEndpoint todoEndpoint() {
return CitrusEndpoints.jms()
.asynchronous()
.destination("jms.todo.inbound")
.connectionFactory(connectionFactory())
.build();
}
We use ActiveMQ as message broker so we use the respective connection factory implementation here. The message broker is automatically started with the Maven build lifecycle.
No we can send some content as binary message to the JMS queue destination.
$(send()
.endpoint(todoJmsEndpoint)
.type(MessageType.BINARY)
.message(new DefaultMessage("{ \"title\": \"${todoName}\", \"description\": \"${todoDescription}\", \"done\": ${done}}".getBytes()));
The sample uses the getBytes()
method of Java String class in order to get binary content as byte array. Citrus will automatically
take care on this binary content by creating a binary JMS message.
Now the next step is to receive the same binary message in Citrus in order to do some validation. We can receive the binary message content
by marking the message type as BINARY
. As binary content is not comparable we use a special message validator implementation that converts the
binary content to a String representation for comparison.
$(receive()
.endpoint(todoJmsEndpoint)
.validator(new BinaryMessageValidator())
.body("{ \"title\": \"${todoName}\", \"description\": \"${todoDescription}\", \"done\": ${done}}");
The binary message validator implementation is very simple and performs String equals for validation:
private class BinaryMessageValidator extends AbstractMessageValidator<DefaultValidationContext> {
@Override
public void validateMessage(Message receivedMessage, Message controlMessage,
TestContext context, DefaultValidationContext validationContext) {
Assert.isTrue(new String(receivedMessage.getPayload(byte[].class))
.equals(new String(controlMessage.getPayload(byte[].class))), "Binary message validation failed!");
}
@Override
public boolean supportsMessageType(String messageType, Message message) {
return messageType.equalsIgnoreCase(MessageType.BINARY.name());
}
@Override
protected Class getRequiredValidationContextType() {
return DefaultValidationContext.class;
}
}
This way you can implement your own validation as you know best how to handle the binary content.
We can also use base64 encoding for handling binary data in Citrus. The base64 encoding can be used to process the binary content
with basic comparison in BINARY_BASE64
message validator:
$(receive()
.endpoint(todoJmsEndpoint)
.body("citrus:encodeBase64('{ \"title\": \"${todoName}\", \"description\": \"${todoDescription}\" }')");
Just use the encodeBase64
function in Citrus to provide the expected payload content. Citrus will automatically convert the received
binary content to base64 encoded Strings then for you.
The sample application uses Maven as build tool. So you can compile, package and test the sample with Maven.
mvn clean verify -Dsystem.under.test.mode=embedded
This executes the Maven build lifecycle until phase verify
which includes the integration-test
and its pre-
and post-
phases. The embedded
option automatically starts a in-memory ActiveMQ message broker during the pre-integration-test
phase. This is everything we need for this sample as Citrus is both message producer and consumer at the same time.
During the build you will see Citrus performing some integration tests. After the tests are finished the ActiveMQ broker is automatically stopped.
For more information on Citrus see www.citrusframework.org, including a complete reference manual.