All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.microsoft.azure.servicebus.MessageConverter Maven / Gradle / Ivy

The newest version!
// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT License.

package com.microsoft.azure.servicebus;

import java.time.Duration;
import java.time.Instant;
import java.util.Date;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.UUID;

import org.apache.qpid.proton.Proton;
import org.apache.qpid.proton.amqp.Binary;
import org.apache.qpid.proton.amqp.Symbol;
import org.apache.qpid.proton.amqp.messaging.AmqpSequence;
import org.apache.qpid.proton.amqp.messaging.AmqpValue;
import org.apache.qpid.proton.amqp.messaging.ApplicationProperties;
import org.apache.qpid.proton.amqp.messaging.Data;
import org.apache.qpid.proton.amqp.messaging.MessageAnnotations;
import org.apache.qpid.proton.amqp.messaging.Properties;
import org.apache.qpid.proton.amqp.messaging.Section;

import com.microsoft.azure.servicebus.primitives.ClientConstants;
import com.microsoft.azure.servicebus.primitives.MessageWithDeliveryTag;
import com.microsoft.azure.servicebus.primitives.MessageWithLockToken;
import com.microsoft.azure.servicebus.primitives.StringUtil;
import com.microsoft.azure.servicebus.primitives.Util;

class MessageConverter {
    public static org.apache.qpid.proton.message.Message convertBrokeredMessageToAmqpMessage(Message brokeredMessage) {
        org.apache.qpid.proton.message.Message amqpMessage = Proton.message();
        MessageBody body = brokeredMessage.getMessageBody();
        if (body != null) {
            if (body.getBodyType() == MessageBodyType.VALUE) {
                amqpMessage.setBody(new AmqpValue(body.getValueData()));
            } else if (body.getBodyType() == MessageBodyType.SEQUENCE) {
                amqpMessage.setBody(new AmqpSequence(Utils.getSequenceFromMessageBody(body)));
            } else {
                amqpMessage.setBody(new Data(new Binary(Utils.getDataFromMessageBody(body))));
            }
        }

        if (brokeredMessage.getProperties() != null) {
            amqpMessage.setApplicationProperties(new ApplicationProperties(brokeredMessage.getProperties()));
        }

        if (brokeredMessage.getTimeToLive() != null) {
        	long ttlMillis = brokeredMessage.getTimeToLive().toMillis();
        	if (ttlMillis > ClientConstants.UNSIGNED_INT_MAX_VALUE) {
        		ttlMillis = ClientConstants.UNSIGNED_INT_MAX_VALUE;
        	}
            amqpMessage.setTtl(ttlMillis);
            Instant creationTime = Instant.now();
            Instant absoluteExpiryTime = creationTime.plus(brokeredMessage.getTimeToLive());
            amqpMessage.setCreationTime(creationTime.toEpochMilli());
            amqpMessage.setExpiryTime(absoluteExpiryTime.toEpochMilli());
        }

        amqpMessage.setMessageId(brokeredMessage.getMessageId());
        amqpMessage.setContentType(brokeredMessage.getContentType());
        amqpMessage.setCorrelationId(brokeredMessage.getCorrelationId());
        amqpMessage.setSubject(brokeredMessage.getLabel());
        amqpMessage.getProperties().setTo(brokeredMessage.getTo());
        amqpMessage.setReplyTo(brokeredMessage.getReplyTo());
        amqpMessage.setReplyToGroupId(brokeredMessage.getReplyToSessionId());
        amqpMessage.setGroupId(brokeredMessage.getSessionId());

        Map messageAnnotationsMap = new HashMap<>();
        if (brokeredMessage.getScheduledEnqueueTimeUtc() != null) {
            messageAnnotationsMap.put(Symbol.valueOf(ClientConstants.SCHEDULEDENQUEUETIMENAME), Date.from(brokeredMessage.getScheduledEnqueueTimeUtc()));
        }

        if (!StringUtil.isNullOrEmpty(brokeredMessage.getPartitionKey())) {
            messageAnnotationsMap.put(Symbol.valueOf(ClientConstants.PARTITIONKEYNAME), brokeredMessage.getPartitionKey());
        }

        if (!StringUtil.isNullOrEmpty(brokeredMessage.getViaPartitionKey())) {
            messageAnnotationsMap.put(Symbol.valueOf(ClientConstants.VIAPARTITIONKEYNAME), brokeredMessage.getViaPartitionKey());
        }

        amqpMessage.setMessageAnnotations(new MessageAnnotations(messageAnnotationsMap));

        return amqpMessage;
    }

    public static Message convertAmqpMessageToBrokeredMessage(org.apache.qpid.proton.message.Message amqpMessage) {
        return convertAmqpMessageToBrokeredMessage(amqpMessage, null);
    }

    public static Message convertAmqpMessageToBrokeredMessage(MessageWithDeliveryTag amqpMessageWithDeliveryTag) {
        org.apache.qpid.proton.message.Message amqpMessage = amqpMessageWithDeliveryTag.getMessage();
        byte[] deliveryTag = amqpMessageWithDeliveryTag.getDeliveryTag();
        return convertAmqpMessageToBrokeredMessage(amqpMessage, deliveryTag);
    }

    public static Message convertAmqpMessageToBrokeredMessage(MessageWithLockToken amqpMessageWithLockToken) {
        Message convertedMessage = convertAmqpMessageToBrokeredMessage(amqpMessageWithLockToken.getMessage(), null);
        convertedMessage.setLockToken(amqpMessageWithLockToken.getLockToken());
        return convertedMessage;
    }

    public static Message convertAmqpMessageToBrokeredMessage(org.apache.qpid.proton.message.Message amqpMessage, byte[] deliveryTag) {
        Message brokeredMessage;
        Section body = amqpMessage.getBody();
        if (body != null) {
            if (body instanceof Data) {
                Binary messageData = ((Data) body).getValue();
                brokeredMessage = new Message(Utils.fromBinay(messageData.getArray()));
            } else if (body instanceof AmqpValue) {
                Object messageData = ((AmqpValue) body).getValue();
                brokeredMessage = new Message(MessageBody.fromValueData(messageData));
            } else if (body instanceof AmqpSequence) {
                List messageData = ((AmqpSequence) body).getValue();
                brokeredMessage = new Message(Utils.fromSequence(messageData));
            } else {
                // Should never happen
                brokeredMessage = new Message();
            }
        } else {
            brokeredMessage = new Message();
        }

        // Application properties
        ApplicationProperties applicationProperties = amqpMessage.getApplicationProperties();
        if (applicationProperties != null) {
            brokeredMessage.setProperties(applicationProperties.getValue());
        }

        // Header
        // Delivery count for service bus starts from 1, for AMQP it starts from 0.
        brokeredMessage.setDeliveryCount(amqpMessage.getDeliveryCount() + 1);
        long ttlMillis = amqpMessage.getTtl();
        if (ttlMillis > 0l) {
        	brokeredMessage.setTimeToLive(Duration.ofMillis(ttlMillis));
        }

        // Properties
        // Override TimeToLive from CrationTime and ExpiryTime, as they support duration of any length, which ttl doesn't
        if (amqpMessage.getCreationTime() != 0l && amqpMessage.getExpiryTime() != 0l) {
        	ttlMillis = amqpMessage.getExpiryTime() - amqpMessage.getCreationTime();
        	if (ttlMillis > 0l)	{
        		brokeredMessage.setTimeToLive(Duration.ofMillis(ttlMillis));
        	}
        }
        
        Object messageId = amqpMessage.getMessageId();
        if (messageId != null) {
            brokeredMessage.setMessageId(messageId.toString());
        }

        brokeredMessage.setContentType(amqpMessage.getContentType());
        Object correlationId = amqpMessage.getCorrelationId();
        if (correlationId != null) {
            brokeredMessage.setCorrelationId(correlationId.toString());
        }

        Properties properties = amqpMessage.getProperties();
        if (properties != null) {
            brokeredMessage.setTo(properties.getTo());
        }

        brokeredMessage.setLabel(amqpMessage.getSubject());
        brokeredMessage.setReplyTo(amqpMessage.getReplyTo());
        brokeredMessage.setReplyToSessionId(amqpMessage.getReplyToGroupId());
        brokeredMessage.setSessionId(amqpMessage.getGroupId());

        // Message Annotations
        MessageAnnotations messageAnnotations = amqpMessage.getMessageAnnotations();
        if (messageAnnotations != null) {
            Map messageAnnotationsMap = messageAnnotations.getValue();
            if (messageAnnotationsMap != null) {
                for (Map.Entry entry : messageAnnotationsMap.entrySet()) {
                    String entryName = entry.getKey().toString();
                    switch (entryName) {
                        case ClientConstants.ENQUEUEDTIMEUTCNAME:
                            brokeredMessage.setEnqueuedTimeUtc(((Date) entry.getValue()).toInstant());
                            break;
                        case ClientConstants.SCHEDULEDENQUEUETIMENAME:
                            brokeredMessage.setScheduledEnqueueTimeUtc(((Date) entry.getValue()).toInstant());
                            break;
                        case ClientConstants.SEQUENCENUBMERNAME:
                            brokeredMessage.setSequenceNumber((long) entry.getValue());
                            break;
                        case ClientConstants.LOCKEDUNTILNAME:
                            brokeredMessage.setLockedUntilUtc(((Date) entry.getValue()).toInstant());
                            break;
                        case ClientConstants.PARTITIONKEYNAME:
                            brokeredMessage.setPartitionKey((String) entry.getValue());
                            break;
                        case ClientConstants.VIAPARTITIONKEYNAME:
                            brokeredMessage.setViaPartitionKey((String) entry.getValue());
                            break;
                        case ClientConstants.DEADLETTERSOURCENAME:
                            brokeredMessage.setDeadLetterSource((String) entry.getValue());
                            break;
                        default:
                            break;
                    }
                }
            }
        }

        if (deliveryTag != null && deliveryTag.length == ClientConstants.LOCKTOKENSIZE) {
            UUID lockToken = Util.convertDotNetBytesToUUID(deliveryTag);
            brokeredMessage.setLockToken(lockToken);
        } else {
            brokeredMessage.setLockToken(ClientConstants.ZEROLOCKTOKEN);
        }

        brokeredMessage.setDeliveryTag(deliveryTag);

        return brokeredMessage;
    }
}