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

com.microsoft.azure.eventhubs.impl.MessageSender Maven / Gradle / Ivy

There is a newer version: 3.3.0
Show newest version
// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT License.

package com.microsoft.azure.eventhubs.impl;

import com.microsoft.azure.eventhubs.ErrorContext;
import com.microsoft.azure.eventhubs.EventHubException;
import com.microsoft.azure.eventhubs.OperationCancelledException;
import com.microsoft.azure.eventhubs.PayloadSizeExceededException;
import com.microsoft.azure.eventhubs.RetryPolicy;
import com.microsoft.azure.eventhubs.ServerBusyException;
import com.microsoft.azure.eventhubs.TimeoutException;
import org.apache.qpid.proton.Proton;
import org.apache.qpid.proton.amqp.Binary;
import org.apache.qpid.proton.amqp.UnsignedLong;
import org.apache.qpid.proton.amqp.messaging.Accepted;
import org.apache.qpid.proton.amqp.messaging.Data;
import org.apache.qpid.proton.amqp.messaging.Rejected;
import org.apache.qpid.proton.amqp.messaging.Released;
import org.apache.qpid.proton.amqp.messaging.Source;
import org.apache.qpid.proton.amqp.messaging.Target;
import org.apache.qpid.proton.amqp.transport.DeliveryState;
import org.apache.qpid.proton.amqp.transport.ErrorCondition;
import org.apache.qpid.proton.amqp.transport.SenderSettleMode;
import org.apache.qpid.proton.engine.BaseHandler;
import org.apache.qpid.proton.engine.Delivery;
import org.apache.qpid.proton.engine.EndpointState;
import org.apache.qpid.proton.engine.Sender;
import org.apache.qpid.proton.engine.Session;
import org.apache.qpid.proton.engine.impl.DeliveryImpl;
import org.apache.qpid.proton.message.Message;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.IOException;
import java.io.Serializable;
import java.nio.BufferOverflowException;
import java.time.Duration;
import java.time.Instant;
import java.time.ZonedDateTime;
import java.util.Comparator;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.PriorityQueue;
import java.util.UUID;
import java.util.concurrent.CancellationException;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.RejectedExecutionException;
import java.util.function.BiConsumer;
import java.util.function.Consumer;

import static java.nio.charset.StandardCharsets.UTF_8;

/**
 * Abstracts all amqp related details
 * translates event-driven reactor model into async send Api
 */
public final class MessageSender extends ClientEntity implements AmqpSender, ErrorContextProvider {
    private static final Logger TRACE_LOGGER = LoggerFactory.getLogger(MessageSender.class);
    private static final String SEND_TIMED_OUT = "Send operation timed out";
    // TestHooks for code injection
    private static volatile Consumer onOpenRetry = null;
    private final MessagingFactory underlyingFactory;
    private final String sendPath;
    private final Duration operationTimeout;
    private final RetryPolicy retryPolicy;
    private final CompletableFuture linkClose;
    private final Object pendingSendLock;
    private final ConcurrentHashMap> pendingSendsData;
    private final PriorityQueue pendingSends;
    private final DispatchHandler sendWork;
    private final ActiveClientTokenManager activeClientTokenManager;
    private final String tokenAudience;
    private final Object errorConditionLock;
    private final Timer timer;
    private volatile int maxMessageSize;
    private volatile Sender sendLink;
    private volatile CompletableFuture linkFirstOpen;
    private volatile TimeoutTracker openLinkTracker;
    private volatile boolean creatingLink;
    private volatile CompletableFuture closeTimer;
    private volatile CompletableFuture openTimer;
    private Exception lastKnownLinkError;
    private Instant lastKnownErrorReportedAt;
    private String linkCreationTime;

    private MessageSender(final MessagingFactory factory, final String sendLinkName, final String senderPath) {
        super(sendLinkName, factory, factory.executor);

        this.sendPath = senderPath;
        this.underlyingFactory = factory;
        this.operationTimeout = factory.getOperationTimeout();
        this.timer = new Timer(factory);
        this.lastKnownLinkError = null;
        this.lastKnownErrorReportedAt = Instant.EPOCH;
        this.retryPolicy = factory.getRetryPolicy();
        this.maxMessageSize = ClientConstants.MAX_MESSAGE_LENGTH_BYTES;
        this.errorConditionLock = new Object();
        this.pendingSendLock = new Object();
        this.pendingSendsData = new ConcurrentHashMap<>();
        this.pendingSends = new PriorityQueue<>(1000, new DeliveryTagComparator());
        this.linkClose = new CompletableFuture<>();
        this.linkFirstOpen = new CompletableFuture<>();
        this.openLinkTracker = TimeoutTracker.create(factory.getOperationTimeout());
        this.sendWork = new DispatchHandler() {
            @Override
            public void onEvent() {
                MessageSender.this.processSendWork();
            }
        };
        this.tokenAudience = String.format(ClientConstants.TOKEN_AUDIENCE_FORMAT, underlyingFactory.getHostName(), sendPath);
        this.activeClientTokenManager = new ActiveClientTokenManager(
                this,
                new Runnable() {
                    @Override
                    public void run() {
                        underlyingFactory.getCBSChannel().sendToken(
                                underlyingFactory.getReactorDispatcher(),
                                underlyingFactory.getTokenProvider().getToken(tokenAudience, ClientConstants.TOKEN_VALIDITY),
                                tokenAudience,
                                new OperationResult() {
                                    @Override
                                    public void onComplete(Void result) {
                                        if (TRACE_LOGGER.isDebugEnabled()) {
                                            TRACE_LOGGER.debug(String.format(Locale.US,
                                                    "clientId[%s], path[%s], linkName[%s] - token renewed",
                                                    getClientId(), sendPath, getSendLinkName()));
                                        }
                                    }

                                    @Override
                                    public void onError(Exception error) {
                                        if (TRACE_LOGGER.isInfoEnabled()) {
                                            TRACE_LOGGER.info(String.format(Locale.US,
                                                    "clientId[%s], path[%s], linkName[%s] - tokenRenewalFailure[%s]",
                                                    getClientId(), sendPath, getSendLinkName(), error.getMessage()));
                                        }
                                    }
                                },
                            (exception) -> {
                                if (TRACE_LOGGER.isWarnEnabled()) {
                                    TRACE_LOGGER.warn(String.format(Locale.US,
                                                "clientId[%s], path[%s], linkName[%s] - tokenRenewalScheduleFailure[%s]",
                                                getClientId(), sendPath, getSendLinkName(), exception.getMessage()));
                                }
                            });
                    }
                },
                ClientConstants.TOKEN_REFRESH_INTERVAL,
                this.underlyingFactory);
    }

    public static CompletableFuture create(
            final MessagingFactory factory,
            final String sendLinkName,
            final String senderPath) {
        final MessageSender msgSender = new MessageSender(factory, sendLinkName, senderPath);

        try {
            msgSender.underlyingFactory.scheduleOnReactorThread(new DispatchHandler() {
                @Override
                public void onEvent() {
                    msgSender.createSendLink();
                }
            });
        } catch (IOException | RejectedExecutionException schedulerException) {
            msgSender.linkFirstOpen.completeExceptionally(schedulerException);
        }

        return msgSender.linkFirstOpen;
    }

    public String getSendPath() {
        return this.sendPath;
    }

    public int getMaxMessageSize() {
        return this.maxMessageSize;
    }

    private CompletableFuture send(byte[] bytes, int arrayOffset, int messageFormat) {
        return this.send(bytes, arrayOffset, messageFormat, null, null);
    }

    private CompletableFuture sendCore(
            final byte[] bytes,
            final int arrayOffset,
            final int messageFormat,
            final CompletableFuture onSend,
            final TimeoutTracker tracker,
            final Exception lastKnownError,
            final CompletableFuture timeoutTask) {
        this.throwIfClosed();

        final boolean isRetrySend = (onSend != null);

        final CompletableFuture onSendFuture = (onSend == null) ? new CompletableFuture<>() : onSend;

        final ReplayableWorkItem sendWaiterData = (tracker == null)
                ? new ReplayableWorkItem<>(bytes, arrayOffset, messageFormat, onSendFuture, this.operationTimeout)
                : new ReplayableWorkItem<>(bytes, arrayOffset, messageFormat, onSendFuture, tracker);

        final TimeoutTracker currentSendTracker = sendWaiterData.getTimeoutTracker();
        final String deliveryTag = UUID.randomUUID().toString().replace("-", StringUtil.EMPTY) + "_" + currentSendTracker.elapsed().getSeconds();

        if (lastKnownError != null) {
            sendWaiterData.setLastKnownException(lastKnownError);
        }

        if (timeoutTask != null) {
            timeoutTask.cancel(false);
        }

        final CompletableFuture timeoutTimerTask = this.timer.schedule(
                new SendTimeout(deliveryTag, sendWaiterData),
                currentSendTracker.remaining());

        // if the timeoutTask completed with scheduling error - notify sender
        if (timeoutTimerTask.isCompletedExceptionally()) {
            timeoutTimerTask.handleAsync(
                (unUsed, exception) -> {
                    if (exception != null && !(exception instanceof CancellationException)) {
                        onSendFuture.completeExceptionally(
                            new OperationCancelledException(String.format(Locale.US,
                                "Entity(%s): send failed while dispatching to Reactor, see cause for more details.",
                                this.sendPath), exception));
                    }
                    return null;
                }, this.executor);

            return onSendFuture;
        }

        sendWaiterData.setTimeoutTask(timeoutTimerTask);

        synchronized (this.pendingSendLock) {
            this.pendingSendsData.put(deliveryTag, sendWaiterData);
            this.pendingSends.offer(new WeightedDeliveryTag(deliveryTag, isRetrySend ? 1 : 0));
        }

        try {
            this.underlyingFactory.scheduleOnReactorThread(this.sendWork);
        } catch (IOException | RejectedExecutionException schedulerException) {
            onSendFuture.completeExceptionally(
                    new OperationCancelledException(String.format(Locale.US,
                            "Entity(%s): send failed while dispatching to Reactor, see cause for more details.",
                            this.sendPath), schedulerException));
        }

        return onSendFuture;
    }

    private CompletableFuture send(
            final byte[] bytes,
            final int arrayOffset,
            final int messageFormat,
            final CompletableFuture onSend,
            final TimeoutTracker tracker) {
        return this.sendCore(bytes, arrayOffset, messageFormat, onSend, tracker, null, null);
    }

    private String getSendLinkName() {
        return this.sendLink == null ? "null" : this.sendLink.getName();
    }

    public CompletableFuture send(final Iterable messages) {
        if (messages == null || IteratorUtil.sizeEquals(messages, 0)) {
            throw new IllegalArgumentException(String.format(Locale.US,
                    "Entity[%s}: sending Empty batch of messages is not allowed.", this.sendPath));
        }

        final Message firstMessage = messages.iterator().next();
        if (IteratorUtil.sizeEquals(messages, 1)) {
            return this.send(firstMessage);
        }

        // proton-j doesn't support multiple dataSections to be part of AmqpMessage
        // here's the alternate approach provided by them: https://github.com/apache/qpid-proton/pull/54
        final Message batchMessage = Proton.message();
        batchMessage.setMessageAnnotations(firstMessage.getMessageAnnotations());

        final int maxMessageSizeTemp = this.maxMessageSize;

        final byte[] bytes = new byte[maxMessageSizeTemp];
        int encodedSize = batchMessage.encode(bytes, 0, maxMessageSizeTemp);
        int byteArrayOffset = encodedSize;

        for (final Message amqpMessage : messages) {
            final Message messageWrappedByData = Proton.message();

            int payloadSize = 0;
            try {
                payloadSize = AmqpUtil.getDataSerializedSize(amqpMessage);
            } catch (EventHubException e) {
                final CompletableFuture payloadTask = new CompletableFuture();
                payloadTask.completeExceptionally(new EventHubException(false,
                    String.format(Locale.US, "Entity(%s): %s", this.sendPath, e.getMessage())));
                return payloadTask;
            }
            int allocationSize = Math.min(payloadSize + ClientConstants.MAX_EVENTHUB_AMQP_HEADER_SIZE_BYTES, maxMessageSizeTemp);

            byte[] messageBytes = new byte[allocationSize];
            int messageSizeBytes = amqpMessage.encode(messageBytes, 0, allocationSize);
            messageWrappedByData.setBody(new Data(new Binary(messageBytes, 0, messageSizeBytes)));

            try {
                encodedSize = messageWrappedByData.encode(bytes, byteArrayOffset, maxMessageSizeTemp - byteArrayOffset - 1);
            } catch (BufferOverflowException exception) {
                final CompletableFuture sendTask = new CompletableFuture<>();
                sendTask.completeExceptionally(new PayloadSizeExceededException(String.format(Locale.US,
                        "Entity(%s): size of the payload exceeded Maximum message size: %s kb",
                        this.sendPath, maxMessageSizeTemp / 1024), exception));
                return sendTask;
            }

            byteArrayOffset = byteArrayOffset + encodedSize;
        }

        return this.send(bytes, byteArrayOffset, AmqpConstants.AMQP_BATCH_MESSAGE_FORMAT);
    }

    public CompletableFuture send(Message msg) {
        int payloadSize = 0;
        try {
            payloadSize = AmqpUtil.getDataSerializedSize(msg);
        } catch (EventHubException e) {
            final CompletableFuture payloadTask = new CompletableFuture();
            payloadTask.completeExceptionally(new EventHubException(false,
                String.format(Locale.US, "Entity(%s): %s", this.sendPath, e.getMessage())));
            return payloadTask;
        }

        final int maxMessageSizeTemp = this.maxMessageSize;
        int allocationSize = Math.min(payloadSize + ClientConstants.MAX_EVENTHUB_AMQP_HEADER_SIZE_BYTES, maxMessageSizeTemp);

        final byte[] bytes = new byte[allocationSize];
        int encodedSize = 0;
        try {
            encodedSize = msg.encode(bytes, 0, allocationSize);
        } catch (BufferOverflowException exception) {
            final CompletableFuture sendTask = new CompletableFuture();
            sendTask.completeExceptionally(new PayloadSizeExceededException(String.format(Locale.US,
                    "Entity(%s): size of the payload exceeded Maximum message size: %s kb",
                    this.sendPath, maxMessageSizeTemp / 1024), exception));
            return sendTask;
        }

        return this.send(bytes, encodedSize, DeliveryImpl.DEFAULT_MESSAGE_FORMAT);
    }

    @Override
    public void onOpenComplete(Exception completionException) {
        this.creatingLink = false;

        if (completionException == null) {
            if (this.getIsClosingOrClosed()) {
                this.sendLink.close();
                return;
            }

            synchronized (this.errorConditionLock) {
                this.lastKnownLinkError = null;
            }

            this.retryPolicy.resetRetryCount(this.getClientId());

            final UnsignedLong remoteMaxMessageSize = this.sendLink.getRemoteMaxMessageSize();
            if (remoteMaxMessageSize != null) {
                this.maxMessageSize = remoteMaxMessageSize.intValue();
            }

            this.cancelOpenTimer();

            if (TRACE_LOGGER.isInfoEnabled()) {
                TRACE_LOGGER.info(String.format(Locale.US, "onOpenComplete - clientId[%s], sendPath[%s], linkName[%s]",
                        this.getClientId(), this.sendPath, this.getSendLinkName()));
            }

            if (!this.linkFirstOpen.isDone()) {
                this.linkFirstOpen.complete(this);
            } else {
                synchronized (this.pendingSendLock) {
                    if (!this.pendingSendsData.isEmpty()) {
                        final List unacknowledgedSends = new LinkedList<>();
                        unacknowledgedSends.addAll(this.pendingSendsData.keySet());

                        if (unacknowledgedSends.size() > 0) {
                            final Iterator reverseReader = unacknowledgedSends.iterator();
                            while (reverseReader.hasNext()) {
                                final String unacknowledgedSend = reverseReader.next();
                                if (this.pendingSendsData.get(unacknowledgedSend).isWaitingForAck()) {
                                    this.pendingSends.offer(new WeightedDeliveryTag(unacknowledgedSend, 1));
                                }
                            }
                        }

                        unacknowledgedSends.clear();
                    }
                }
            }
        } else {
            if (!this.linkFirstOpen.isDone()) {
                final Duration nextRetryInterval = this.retryPolicy.getNextRetryInterval(
                        this.getClientId(), completionException, this.openLinkTracker.remaining());

                if (nextRetryInterval != null) {
                    if (onOpenRetry != null) {
                        onOpenRetry.accept(this);
                    }

                    try {
                        this.underlyingFactory.scheduleOnReactorThread((int) nextRetryInterval.toMillis(), new DispatchHandler() {
                            @Override
                            public void onEvent() {
                                if (!MessageSender.this.getIsClosingOrClosed()
                                        && (sendLink == null || sendLink.getLocalState() == EndpointState.CLOSED || sendLink.getRemoteState() == EndpointState.CLOSED)) {
                                    recreateSendLink();
                                }
                            }
                        });
                    } catch (IOException | RejectedExecutionException schedulerException) {
                        if (TRACE_LOGGER.isWarnEnabled()) {
                            TRACE_LOGGER.warn(
                                    String.format(Locale.US, "clientId[%s], senderPath[%s], scheduling createLink encountered error: %s",
                                            this.getClientId(), this.sendPath, schedulerException.getLocalizedMessage()));
                        }

                        this.cancelOpen(schedulerException);
                    }
                } else if (completionException instanceof EventHubException) {
                    // If the error is not a transient exception, we want to cancel this open. Otherwise, we don't want
                    // log the transient exceptions and let it fall through.
                    if (!((EventHubException) completionException).getIsTransient()) {
                        this.cancelOpen(completionException);
                    }
                } else {
                    // We don't want this exception to fall into the abyss and we are out of retries, so log a message,
                    // and cancel this operation.
                    if (TRACE_LOGGER.isErrorEnabled()) {
                        TRACE_LOGGER.error("Could not open link.", completionException);
                    }

                    this.cancelOpen(completionException);
                }
            } else {
                this.cancelOpenTimer();
            }
        }
    }

    private void cancelOpen(final Exception completionException) {
        this.setClosed();
        ExceptionUtil.completeExceptionally(this.linkFirstOpen, completionException, this);
        this.cancelOpenTimer();
    }

    private void cancelOpenTimer() {
        if (this.openTimer != null && !this.openTimer.isCancelled()) {
            this.openTimer.cancel(false);
        }
    }

    @Override
    public void onClose(final ErrorCondition condition, final String errorContext) {
        if (this.sendLink != null) {
            this.underlyingFactory.deregisterForConnectionError(this.sendLink);
        }

        final Exception completionException = (condition != null && condition.getCondition() != null) ? ExceptionUtil.toException(condition) : null;
        this.onError(completionException, null);
    }

    @Override
    public void onError(final Exception completionException, final String failingLinkName) {
        if (this.getIsClosingOrClosed()) {
            if (this.closeTimer != null && !this.closeTimer.isDone()) {
                this.closeTimer.cancel(false);
            }

            synchronized (this.pendingSendLock) {
                for (Map.Entry> pendingSend : this.pendingSendsData.entrySet()) {
                    ExceptionUtil.completeExceptionally(pendingSend.getValue().getWork(),
                            completionException == null
                                    ? new OperationCancelledException(String.format(Locale.US,
                                    "Entity(%s): send cancelled as the Sender instance is Closed before the sendOperation completed.",
                                    this.sendPath))
                                    : completionException,
                            this);
                }

                this.pendingSendsData.clear();
                this.pendingSends.clear();
            }

            this.linkClose.complete(null);

            return;
        } else {
            synchronized (this.errorConditionLock) {
                this.lastKnownLinkError = completionException == null ? this.lastKnownLinkError : completionException;
                this.lastKnownErrorReportedAt = Instant.now();
            }

            final Exception finalCompletionException = completionException == null
                    ? new EventHubException(true, String.format(Locale.US,
                    "Entity(%s): client encountered transient error for unknown reasons, please retry the operation.",
                    this.sendPath)) : completionException;

            this.onOpenComplete(finalCompletionException);

            final Map.Entry> pendingSendEntry = IteratorUtil.getFirst(this.pendingSendsData.entrySet());
            if (pendingSendEntry != null && pendingSendEntry.getValue() != null) {
                final TimeoutTracker tracker = pendingSendEntry.getValue().getTimeoutTracker();
                if (tracker != null) {
                    final Duration nextRetryInterval = this.retryPolicy.getNextRetryInterval(this.getClientId(), finalCompletionException, tracker.remaining());
                    boolean scheduledRecreate = true;

                    if (nextRetryInterval != null) {
                        try {
                            this.underlyingFactory.scheduleOnReactorThread((int) nextRetryInterval.toMillis(), new DispatchHandler() {
                                @Override
                                public void onEvent() {
                                    if (!MessageSender.this.getIsClosingOrClosed()
                                            && (sendLink == null || sendLink.getLocalState() == EndpointState.CLOSED || sendLink.getRemoteState() == EndpointState.CLOSED)) {
                                        recreateSendLink();
                                    }
                                }
                            });
                        } catch (IOException | RejectedExecutionException ignore) {
                            scheduledRecreate = false;
                        }
                    }

                    if (nextRetryInterval == null || !scheduledRecreate) {
                        synchronized (this.pendingSendLock) {
                            for (Map.Entry> pendingSend : this.pendingSendsData.entrySet()) {
                                this.cleanupFailedSend(pendingSend.getValue(), finalCompletionException);
                            }

                            this.pendingSendsData.clear();
                            this.pendingSends.clear();
                        }
                    }
                }
            }
        }
    }

    @Override
    public void onSendComplete(final Delivery delivery) {
        final DeliveryState outcome = delivery.getRemoteState();
        final String deliveryTag = new String(delivery.getTag(), UTF_8);

        if (TRACE_LOGGER.isTraceEnabled()) {
            TRACE_LOGGER.trace(
                String.format(
                    Locale.US,
                    "clientId[%s], path[%s], linkName[%s], deliveryTag[%s]",
                    this.getClientId(), this.sendPath, this.getSendLinkName(), deliveryTag));
        }

        final ReplayableWorkItem pendingSendWorkItem = this.pendingSendsData.remove(deliveryTag);

        if (pendingSendWorkItem != null) {
            if (outcome instanceof Accepted) {
                synchronized (this.errorConditionLock) {
                    this.lastKnownLinkError = null;
                }

                this.retryPolicy.resetRetryCount(this.getClientId());

                pendingSendWorkItem.getTimeoutTask().cancel(false);
                pendingSendWorkItem.clearMessage();
                pendingSendWorkItem.getWork().complete(null);
            } else if (outcome instanceof Rejected) {
                final Rejected rejected = (Rejected) outcome;
                final ErrorCondition error = rejected.getError();

                final Exception exception = ExceptionUtil.toException(error);

                if (ExceptionUtil.isGeneralSendError(error.getCondition())) {
                    synchronized (this.errorConditionLock) {
                        this.lastKnownLinkError = exception;
                        this.lastKnownErrorReportedAt = Instant.now();
                    }
                }

                final Duration retryInterval = this.retryPolicy.getNextRetryInterval(
                        this.getClientId(), exception, pendingSendWorkItem.getTimeoutTracker().remaining());
                if (retryInterval == null) {
                    this.cleanupFailedSend(pendingSendWorkItem, exception);
                } else {
                    pendingSendWorkItem.setLastKnownException(exception);
                    try {
                        this.underlyingFactory.scheduleOnReactorThread((int) retryInterval.toMillis(),
                                new DispatchHandler() {
                                    @Override
                                    public void onEvent() {
                                        MessageSender.this.sendCore(
                                                pendingSendWorkItem.getMessage(),
                                                pendingSendWorkItem.getEncodedMessageSize(),
                                                pendingSendWorkItem.getMessageFormat(),
                                                pendingSendWorkItem.getWork(),
                                                pendingSendWorkItem.getTimeoutTracker(),
                                                pendingSendWorkItem.getLastKnownException(),
                                                pendingSendWorkItem.getTimeoutTask());
                                    }
                                });
                    } catch (IOException | RejectedExecutionException schedulerException) {
                        exception.initCause(schedulerException);
                        this.cleanupFailedSend(
                                pendingSendWorkItem,
                                new EventHubException(false, String.format(Locale.US,
                                        "Entity(%s): send operation failed while scheduling a retry on Reactor, see cause for more details.",
                                        this.sendPath),
                                        schedulerException));
                    }
                }
            } else if (outcome instanceof Released) {
                this.cleanupFailedSend(pendingSendWorkItem, new OperationCancelledException(outcome.toString()));
            } else {
                this.cleanupFailedSend(pendingSendWorkItem, new EventHubException(false, outcome.toString()));
            }
        } else {
            if (TRACE_LOGGER.isDebugEnabled()) {
                TRACE_LOGGER.debug(
                    String.format(Locale.US, "clientId[%s]. path[%s], linkName[%s], delivery[%s] - mismatch (or send timed out)",
                        this.getClientId(), this.sendPath, this.getSendLinkName(), deliveryTag));
            }
        }
    }

    private void cleanupFailedSend(final ReplayableWorkItem failedSend, final Exception exception) {
        if (failedSend.getTimeoutTask() != null) {
            failedSend.getTimeoutTask().cancel(false);
        }
        ExceptionUtil.completeExceptionally(failedSend.getWork(), exception, this);
    }

    private void createSendLink() {
        synchronized (this.errorConditionLock) {
            if (this.creatingLink) {
                if (TRACE_LOGGER.isInfoEnabled()) {
                    TRACE_LOGGER.info(
                            String.format(Locale.US,
                                    "clientId[%s], path[%s], operationTimeout[%s], creating a send link is already in progress",
                                    this.getClientId(), this.sendPath, this.operationTimeout));
                }

                return;
            }

            this.creatingLink = true;
        }

        if (TRACE_LOGGER.isInfoEnabled()) {
            TRACE_LOGGER.info(
                    String.format(Locale.US,
                            "clientId[%s], path[%s], operationTimeout[%s], creating a send link",
                            this.getClientId(), this.sendPath, this.operationTimeout));
        }

        this.linkCreationTime = Instant.now().toString();

        this.scheduleLinkOpenTimeout(TimeoutTracker.create(this.operationTimeout));

        final Consumer onSessionOpen = new Consumer() {
            @Override
            public void accept(Session session) {
                if (MessageSender.this.getIsClosingOrClosed()) {

                    session.close();
                    return;
                }

                final Sender sender = session.sender(TrackingUtil.getLinkName(session));
                final Target target = new Target();
                target.setAddress(sendPath);
                sender.setTarget(target);

                final Source source = new Source();
                sender.setSource(source);

                sender.setSenderSettleMode(SenderSettleMode.UNSETTLED);

                final SendLinkHandler handler = new SendLinkHandler(MessageSender.this, MessageSender.this.getClientId(),
                    MessageSender.this.underlyingFactory.executor);
                BaseHandler.setHandler(sender, handler);

                if (MessageSender.this.sendLink != null) {
                    MessageSender.this.underlyingFactory.deregisterForConnectionError(MessageSender.this.sendLink);
                }

                MessageSender.this.underlyingFactory.registerForConnectionError(sender);
                sender.open();

                synchronized (MessageSender.this.errorConditionLock) {
                    MessageSender.this.sendLink = sender;
                }
            }
        };

        final BiConsumer onSessionOpenError = new BiConsumer() {
            @Override
            public void accept(ErrorCondition t, Exception u) {
                if (t != null) {
                    MessageSender.this.onError(t.getCondition() != null ? ExceptionUtil.toException(t) : null, null);
                } else if (u != null) {
                    MessageSender.this.onError(u, null);
                }
            }
        };

        this.underlyingFactory.getCBSChannel().sendToken(
                this.underlyingFactory.getReactorDispatcher(),
                this.underlyingFactory.getTokenProvider().getToken(tokenAudience, ClientConstants.TOKEN_VALIDITY),
                tokenAudience,
                new OperationResult() {
                    @Override
                    public void onComplete(Void result) {
                        if (MessageSender.this.getIsClosingOrClosed()) {
                            return;
                        }
                        underlyingFactory.getSession(
                                sendPath,
                                onSessionOpen,
                                onSessionOpenError);
                    }

                    @Override
                    public void onError(Exception error) {
                        final Exception completionException;
                        if (error != null && error instanceof AmqpException) {
                            completionException = ExceptionUtil.toException(((AmqpException) error).getError());
                            if (completionException != error && completionException.getCause() == null) {
                                completionException.initCause(error);
                            }
                        } else {
                            completionException = error;
                        }

                        MessageSender.this.onError(completionException, null);
                    }
                },
            (exception) -> {
                MessageSender.this.onError(exception, null);
            });
    }

    private void scheduleLinkOpenTimeout(TimeoutTracker timeout) {
        // timer to signal a timeout if exceeds the operationTimeout on MessagingFactory
        this.openTimer = this.timer.schedule(
                new Runnable() {
                    public void run() {
                        creatingLink = false;

                        if (!MessageSender.this.linkFirstOpen.isDone()) {
                            final Exception lastReportedError;
                            synchronized (MessageSender.this.errorConditionLock) {
                                lastReportedError = MessageSender.this.lastKnownLinkError;
                            }

                            final Exception operationTimedout = new TimeoutException(
                                    String.format(Locale.US, "Open operation on entity(%s) timed out at %s.",
                                            MessageSender.this.getSendPath(), ZonedDateTime.now().toString()),
                                    lastReportedError);

                            if (TRACE_LOGGER.isWarnEnabled()) {
                                TRACE_LOGGER.warn(
                                        String.format(Locale.US, "clientId[%s], path[%s], open call timed out",
                                                MessageSender.this.getClientId(), MessageSender.this.sendPath),
                                        operationTimedout);
                            }

                            ExceptionUtil.completeExceptionally(MessageSender.this.linkFirstOpen, operationTimedout, MessageSender.this);
                            setClosed();
                        }
                    }
                }, timeout.remaining());

        this.openTimer.handleAsync(
            (unUsed, exception) -> {
                if (exception != null
                        && exception instanceof Exception
                        && !(exception instanceof CancellationException)) {
                    ExceptionUtil.completeExceptionally(linkFirstOpen, (Exception) exception, this);
                }

                return null;
            }, this.executor);
    }

    @Override
    public ErrorContext getContext() {
        final Sender link;
        synchronized (this.errorConditionLock) {
            link = this.sendLink;
        }

        final boolean isLinkOpened = this.linkFirstOpen != null && this.linkFirstOpen.isDone();
        final String referenceId = link != null && link.getRemoteProperties() != null && link.getRemoteProperties().containsKey(ClientConstants.TRACKING_ID_PROPERTY)
                ? link.getRemoteProperties().get(ClientConstants.TRACKING_ID_PROPERTY).toString()
                : ((link != null) ? link.getName() : null);

        final SenderContext errorContext = new SenderContext(
                this.underlyingFactory != null ? this.underlyingFactory.getHostName() : null,
                this.sendPath,
                referenceId,
                isLinkOpened && link != null ? link.getCredit() : null);
        return errorContext;
    }

    @Override
    public void onFlow(final int creditIssued) {
        synchronized (this.errorConditionLock) {
            this.lastKnownLinkError = null;
        }

        if (creditIssued <= 0) {
            return;
        }
        if (TRACE_LOGGER.isDebugEnabled()) {
            int numberOfSendsWaitingforCredit = this.pendingSends.size();
            TRACE_LOGGER.debug(String.format(Locale.US,
                    "clientId[%s], path[%s], linkName[%s], remoteLinkCredit[%s], pendingSendsWaitingForCredit[%s], pendingSendsWaitingDelivery[%s]",
                    this.getClientId(), this.sendPath, this.getSendLinkName(), creditIssued, numberOfSendsWaitingforCredit, this.pendingSendsData.size() - numberOfSendsWaitingforCredit));
        }

        this.sendWork.onEvent();
    }

    private void recreateSendLink() {
        this.createSendLink();
        this.retryPolicy.incrementRetryCount(this.getClientId());
    }

    // actual send on the SenderLink should happen only in this method & should run on Reactor Thread
    private void processSendWork() {
        if (this.sendLink == null || this.sendLink.getLocalState() == EndpointState.CLOSED || this.sendLink.getRemoteState() == EndpointState.CLOSED) {
            if (!this.getIsClosingOrClosed()) {
                this.recreateSendLink();
            }

            return;
        }

        while (this.sendLink.getLocalState() == EndpointState.ACTIVE && this.sendLink.getRemoteState() == EndpointState.ACTIVE
                && this.sendLink.getCredit() > 0) {
            final WeightedDeliveryTag weightedDelivery;
            final ReplayableWorkItem sendData;
            final String deliveryTag;
            synchronized (this.pendingSendLock) {
                weightedDelivery = this.pendingSends.poll();
                if (weightedDelivery != null) {
                    deliveryTag = weightedDelivery.getDeliveryTag();
                    sendData = this.pendingSendsData.get(deliveryTag);
                } else {
                    sendData = null;
                    deliveryTag = null;
                }
            }

            if (sendData != null) {
                if (sendData.getWork() != null && sendData.getWork().isDone()) {
                    // CoreSend could enque Sends into PendingSends Queue and can fail the SendCompletableFuture
                    // (when It fails to schedule the ProcessSendWork on reactor Thread)
                    this.pendingSendsData.remove(deliveryTag);
                    continue;
                }

                Delivery delivery = null;
                boolean linkAdvance = false;
                int sentMsgSize = 0;
                Exception sendException = null;

                try {
                    delivery = this.sendLink.delivery(deliveryTag.getBytes(UTF_8));
                    delivery.setMessageFormat(sendData.getMessageFormat());

                    sentMsgSize = this.sendLink.send(sendData.getMessage(), 0, sendData.getEncodedMessageSize());
                    assert sentMsgSize == sendData.getEncodedMessageSize() : "Contract of the ProtonJ library for Sender.Send API changed";

                    linkAdvance = this.sendLink.advance();
                } catch (Exception exception) {
                    sendException = exception;
                }

                if (linkAdvance) {
                    sendData.setWaitingForAck();
                } else {
                    if (TRACE_LOGGER.isDebugEnabled()) {
                        TRACE_LOGGER.debug(
                                String.format(Locale.US, "clientId[%s], path[%s], linkName[%s], deliveryTag[%s], sentMessageSize[%s], payloadActualSize[%s] - sendlink advance failed",
                                        this.getClientId(), this.sendPath, this.getSendLinkName(), deliveryTag, sentMsgSize, sendData.getEncodedMessageSize()));
                    }

                    if (delivery != null) {
                        delivery.free();
                    }

                    sendData.getWork().completeExceptionally(sendException != null
                            ? new OperationCancelledException(String.format(Locale.US,
                            "Entity(%s): send operation failed. Please see cause for more details", this.sendPath), sendException)
                            : new OperationCancelledException(
                            String.format(Locale.US, "Entity(%s): send operation failed while advancing delivery(tag: %s).", this.sendPath, deliveryTag)));
                }
            } else {
                if (deliveryTag != null) {
                    if (TRACE_LOGGER.isDebugEnabled()) {
                        TRACE_LOGGER.debug(
                                String.format(Locale.US, "clientId[%s], path[%s], linkName[%s], deliveryTag[%s] - sendData not found for this delivery.",
                                        this.getClientId(), this.sendPath, this.getSendLinkName(), deliveryTag));
                    }
                }

                break;
            }
        }
    }

    private void throwSenderTimeout(final CompletableFuture pendingSendWork, final Exception lastKnownException) {

        Exception cause = lastKnownException;
        if (lastKnownException == null) {
            final Exception lastReportedLinkLevelError;
            final Instant lastLinkErrorReportedAt;
            synchronized (this.errorConditionLock) {
                lastReportedLinkLevelError = this.lastKnownLinkError;
                lastLinkErrorReportedAt = this.lastKnownErrorReportedAt;
            }

            if (lastReportedLinkLevelError != null) {
                boolean isServerBusy = ((lastReportedLinkLevelError instanceof ServerBusyException)
                        && (lastLinkErrorReportedAt.isAfter(Instant.now().minusSeconds(ClientConstants.SERVER_BUSY_BASE_SLEEP_TIME_IN_SECS))));
                cause = isServerBusy || (lastLinkErrorReportedAt.isAfter(Instant.now().minusMillis(this.operationTimeout.toMillis())))
                        ? lastReportedLinkLevelError
                        : null;
            }
        }

        final boolean isClientSideTimeout = (cause == null || !(cause instanceof EventHubException));
        final EventHubException exception = isClientSideTimeout
                ? new TimeoutException(String.format(Locale.US, "Entity(%s): %s at %s.",
                this.sendPath, MessageSender.SEND_TIMED_OUT, ZonedDateTime.now()), cause)
                : (EventHubException) cause;

        ExceptionUtil.completeExceptionally(pendingSendWork, exception, this);
    }

    private void scheduleLinkCloseTimeout(final TimeoutTracker timeout) {
        // timer to signal a timeout if exceeds the operationTimeout on MessagingFactory
        this.closeTimer = this.timer.schedule(
                new Runnable() {
                    public void run() {
                        if (!linkClose.isDone()) {
                            final Sender link;
                            synchronized (MessageSender.this.errorConditionLock) {
                                link = MessageSender.this.sendLink;
                            }

                            final Exception operationTimedout = new TimeoutException(String.format(Locale.US,
                                    "Entity(%s): close operation timed out at %s", MessageSender.this.sendPath, ZonedDateTime.now()));
                            if (TRACE_LOGGER.isInfoEnabled()) {
                                TRACE_LOGGER.info(
                                        String.format(Locale.US, "clientId[%s], message sender(linkName: %s, path: %s) close call timed out",
                                                MessageSender.this.getClientId(), link.getName(), MessageSender.this.sendPath),
                                        operationTimedout);
                            }

                            ExceptionUtil.completeExceptionally(linkClose, operationTimedout, MessageSender.this);
                            MessageSender.this.onError((Exception) null, null);
                        }
                    }
                }, timeout.remaining());

        this.closeTimer.handleAsync(
            (unUsed, exception) -> {
                if (exception != null && exception instanceof Exception && !(exception instanceof CancellationException)) {
                    ExceptionUtil.completeExceptionally(linkClose, (Exception) exception, MessageSender.this);
                }

                return null;
            }, this.executor);
    }

    @Override
    protected CompletableFuture onClose() {
        if (!this.getIsClosed()) {
            try {
                this.activeClientTokenManager.cancel();
                scheduleLinkCloseTimeout(TimeoutTracker.create(operationTimeout));
                this.underlyingFactory.scheduleOnReactorThread(new DispatchHandler() {
                    @Override
                    public void onEvent() {
                        if (sendLink != null && sendLink.getLocalState() != EndpointState.CLOSED) {
                            sendLink.close();
                        } else if (sendLink == null || sendLink.getRemoteState() == EndpointState.CLOSED) {
                            if (closeTimer != null && !closeTimer.isCancelled()) {
                                closeTimer.cancel(false);
                            }

                            linkClose.complete(null);
                        }
                    }
                });

            } catch (IOException | RejectedExecutionException schedulerException) {
                this.linkClose.completeExceptionally(schedulerException);
            }
        }

        return this.linkClose;
    }

    @Override
    protected Exception getLastKnownError() {
        synchronized (this.errorConditionLock) {
            return this.lastKnownLinkError;
        }
    }

    private static class WeightedDeliveryTag {
        private final String deliveryTag;
        private final int priority;

        WeightedDeliveryTag(final String deliveryTag, final int priority) {
            this.deliveryTag = deliveryTag;
            this.priority = priority;
        }

        public String getDeliveryTag() {
            return this.deliveryTag;
        }

        public int getPriority() {
            return this.priority;
        }
    }

    private static class DeliveryTagComparator implements Comparator, Serializable {

        private static final long serialVersionUID = -7057500582037295635L;

        @Override
        public int compare(WeightedDeliveryTag deliveryTag0, WeightedDeliveryTag deliveryTag1) {
            return deliveryTag1.getPriority() - deliveryTag0.getPriority();
        }
    }

    private class SendTimeout implements Runnable {
        private final String deliveryTag;
        private final ReplayableWorkItem sendWaiterData;

        SendTimeout(
                final String deliveryTag,
                final ReplayableWorkItem sendWaiterData) {
            this.sendWaiterData = sendWaiterData;
            this.deliveryTag = deliveryTag;
        }

        @Override
        public void run() {
            if (!sendWaiterData.getWork().isDone()) {
                MessageSender.this.pendingSendsData.remove(deliveryTag);
                MessageSender.this.throwSenderTimeout(sendWaiterData.getWork(), sendWaiterData.getLastKnownException());
            }
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy