/** * 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 * * http://www.apache.org/licenses/LICENSE-2.0 * * 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.camel.component.rabbitmq; import java.io.ByteArrayInputStream; import java.io.IOException; import java.io.InputStream; import java.io.ObjectInputStream; import java.util.Date; import java.util.HashMap; import java.util.Map; import com.rabbitmq.client.AMQP; import com.rabbitmq.client.Envelope; import com.rabbitmq.client.LongString; import org.apache.camel.Exchange; import org.apache.camel.Message; import org.apache.camel.impl.DefaultMessage; import org.apache.camel.util.ObjectHelper; import org.slf4j.Logger; import org.slf4j.LoggerFactory; public class RabbitMQMessageConverter { protected static final Logger LOG = LoggerFactory.getLogger(RabbitMQMessageConverter.class); /** * Will take an {@link Exchange} and add header values back to the {@link Exchange#getIn()} */ public void mergeAmqpProperties(Exchange exchange, AMQP.BasicProperties properties) { if (properties.getType() != null) { exchange.getIn().setHeader(RabbitMQConstants.TYPE, properties.getType()); } if (properties.getAppId() != null) { exchange.getIn().setHeader(RabbitMQConstants.APP_ID, properties.getAppId()); } if (properties.getClusterId() != null) { exchange.getIn().setHeader(RabbitMQConstants.CLUSTERID, properties.getClusterId()); } if (properties.getContentEncoding() != null) { exchange.getIn().setHeader(RabbitMQConstants.CONTENT_ENCODING, properties.getContentEncoding()); } if (properties.getContentType() != null) { exchange.getIn().setHeader(RabbitMQConstants.CONTENT_TYPE, properties.getContentType()); } if (properties.getCorrelationId() != null) { exchange.getIn().setHeader(RabbitMQConstants.CORRELATIONID, properties.getCorrelationId()); } if (properties.getExpiration() != null) { exchange.getIn().setHeader(RabbitMQConstants.EXPIRATION, properties.getExpiration()); } if (properties.getMessageId() != null) { exchange.getIn().setHeader(RabbitMQConstants.MESSAGE_ID, properties.getMessageId()); } if (properties.getPriority() != null) { exchange.getIn().setHeader(RabbitMQConstants.PRIORITY, properties.getPriority()); } if (properties.getReplyTo() != null) { exchange.getIn().setHeader(RabbitMQConstants.REPLY_TO, properties.getReplyTo()); } if (properties.getTimestamp() != null) { exchange.getIn().setHeader(RabbitMQConstants.TIMESTAMP, properties.getTimestamp()); } if (properties.getUserId() != null) { exchange.getIn().setHeader(RabbitMQConstants.USERID, properties.getUserId()); } } public AMQP.BasicProperties.Builder buildProperties(Exchange exchange) { AMQP.BasicProperties.Builder properties = new AMQP.BasicProperties.Builder(); Message msg; if (exchange.hasOut()) { msg = exchange.getOut(); } else { msg = exchange.getIn(); } final Object contentType = msg.removeHeader(RabbitMQConstants.CONTENT_TYPE); if (contentType != null) { properties.contentType(contentType.toString()); } final Object priority = msg.removeHeader(RabbitMQConstants.PRIORITY); if (priority != null) { properties.priority(Integer.parseInt(priority.toString())); } final Object messageId = msg.removeHeader(RabbitMQConstants.MESSAGE_ID); if (messageId != null) { properties.messageId(messageId.toString()); } final Object clusterId = msg.removeHeader(RabbitMQConstants.CLUSTERID); if (clusterId != null) { properties.clusterId(clusterId.toString()); } final Object replyTo = msg.removeHeader(RabbitMQConstants.REPLY_TO); if (replyTo != null) { properties.replyTo(replyTo.toString()); } final Object correlationId = msg.removeHeader(RabbitMQConstants.CORRELATIONID); if (correlationId != null) { properties.correlationId(correlationId.toString()); } final Object deliveryMode = msg.removeHeader(RabbitMQConstants.DELIVERY_MODE); if (deliveryMode != null) { properties.deliveryMode(Integer.parseInt(deliveryMode.toString())); } final Object userId = msg.removeHeader(RabbitMQConstants.USERID); if (userId != null) { properties.userId(userId.toString()); } final Object type = msg.removeHeader(RabbitMQConstants.TYPE); if (type != null) { properties.type(type.toString()); } final Object contentEncoding = msg.removeHeader(RabbitMQConstants.CONTENT_ENCODING); if (contentEncoding != null) { properties.contentEncoding(contentEncoding.toString()); } final Object expiration = msg.removeHeader(RabbitMQConstants.EXPIRATION); if (expiration != null) { properties.expiration(expiration.toString()); } final Object appId = msg.removeHeader(RabbitMQConstants.APP_ID); if (appId != null) { properties.appId(appId.toString()); } final Object timestamp = msg.removeHeader(RabbitMQConstants.TIMESTAMP); if (timestamp != null) { properties.timestamp(convertTimestamp(timestamp)); } final Map<String, Object> headers = msg.getHeaders(); Map<String, Object> filteredHeaders = new HashMap<>(); // TODO: Add support for a HeaderFilterStrategy. See: org.apache.camel.component.jms.JmsBinding#shouldOutputHeader for (Map.Entry<String, Object> header : headers.entrySet()) { // filter header values. Object value = getValidRabbitMQHeaderValue(header.getValue()); if (value != null) { filteredHeaders.put(header.getKey(), header.getValue()); } else if (LOG.isDebugEnabled()) { if (header.getValue() == null) { LOG.debug("Ignoring header: {} with null value", header.getKey()); } else { LOG.debug("Ignoring header: {} of class: {} with value: {}", header.getKey(), ObjectHelper.classCanonicalName(header.getValue()), header.getValue()); } } } properties.headers(filteredHeaders); return properties; } private Date convertTimestamp(Object timestamp) { if (timestamp instanceof Date) { return (Date)timestamp; } return new Date(Long.parseLong(timestamp.toString())); } /** * Strategy to test if the given header is valid. Without this, the * com.rabbitmq.client.impl.Frame.java class will throw an * IllegalArgumentException (invalid value in table) and close the * connection. * * @param headerValue the header value * @return the value to use, <tt>null</tt> to ignore this header * @see com.rabbitmq.client.impl.Frame#fieldValueSize */ private Object getValidRabbitMQHeaderValue(Object headerValue) { if (headerValue instanceof String) { return headerValue; } else if (headerValue instanceof Number) { return headerValue; } else if (headerValue instanceof Boolean) { return headerValue; } else if (headerValue instanceof Date) { return headerValue; } else if (headerValue instanceof byte[]) { return headerValue; } else if (headerValue instanceof LongString) { return headerValue; } return null; } public void populateRabbitExchange(Exchange camelExchange, Envelope envelope, AMQP.BasicProperties properties, byte[] body, final boolean out) { Message message = resolveMessageFrom(camelExchange, out); populateMessageHeaders(message, envelope, properties); populateMessageBody(message, camelExchange, properties, body); } private Message resolveMessageFrom(final Exchange camelExchange, final boolean out) { Message message; if (out) { // use OUT message message = camelExchange.getOut(); } else { if (camelExchange.getIn() != null) { // Use the existing message so we keep the headers message = camelExchange.getIn(); } else { message = new DefaultMessage(); camelExchange.setIn(message); } } return message; } private void populateMessageHeaders(final Message message, final Envelope envelope, final AMQP.BasicProperties properties) { populateRoutingInfoHeaders(message, envelope); populateMessageHeadersFromRabbitMQHeaders(message, properties); } private void populateRoutingInfoHeaders(final Message message, final Envelope envelope) { if (envelope != null) { message.setHeader(RabbitMQConstants.ROUTING_KEY, envelope.getRoutingKey()); message.setHeader(RabbitMQConstants.EXCHANGE_NAME, envelope.getExchange()); message.setHeader(RabbitMQConstants.DELIVERY_TAG, envelope.getDeliveryTag()); } } private void populateMessageHeadersFromRabbitMQHeaders(final Message message, final AMQP.BasicProperties properties) { Map<String, Object> headers = properties.getHeaders(); if (headers != null) { for (Map.Entry<String, Object> entry : headers.entrySet()) { // Convert LongStrings to String. if (entry.getValue() instanceof LongString) { message.setHeader(entry.getKey(), entry.getValue().toString()); } else { message.setHeader(entry.getKey(), entry.getValue()); } } } } private void populateMessageBody(final Message message, final Exchange camelExchange, final AMQP.BasicProperties properties, final byte[] body) { if (hasSerializeHeader(properties)) { deserializeBody(camelExchange, message, body); } else { // Set the body as a byte[] and let the type converter deal with it message.setBody(body); } } private void deserializeBody(final Exchange camelExchange, final Message message, final byte[] body) { Object messageBody = null; try (InputStream b = new ByteArrayInputStream(body); ObjectInputStream o = new ObjectInputStream(b)) { messageBody = o.readObject(); } catch (IOException | ClassNotFoundException e) { LOG.warn("Could not deserialize the object"); camelExchange.setException(e); } if (messageBody instanceof Throwable) { LOG.debug("Reply was an Exception. Setting the Exception on the Exchange"); camelExchange.setException((Throwable) messageBody); } else { message.setBody(messageBody); } } private boolean hasSerializeHeader(AMQP.BasicProperties properties) { return hasHeaders(properties) && Boolean.TRUE.equals(isSerializeHeaderEnabled(properties)); } private boolean hasHeaders(final AMQP.BasicProperties properties) { return properties != null && properties.getHeaders() != null; } private Object isSerializeHeaderEnabled(final AMQP.BasicProperties properties) { return properties.getHeaders().get(RabbitMQEndpoint.SERIALIZE_HEADER); } }