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

com.microsoft.azure.servicebus.primitives.Controller Maven / Gradle / Ivy

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

package com.microsoft.azure.servicebus.primitives;

import com.microsoft.azure.servicebus.ClientSettings;
import com.microsoft.azure.servicebus.TransactionContext;
import org.apache.qpid.proton.amqp.Binary;
import org.apache.qpid.proton.amqp.Symbol;
import org.apache.qpid.proton.amqp.UnsignedInteger;
import org.apache.qpid.proton.amqp.messaging.Accepted;
import org.apache.qpid.proton.amqp.messaging.AmqpValue;
import org.apache.qpid.proton.amqp.messaging.Source;
import org.apache.qpid.proton.amqp.transaction.Coordinator;
import org.apache.qpid.proton.amqp.transaction.Declare;
import org.apache.qpid.proton.amqp.transaction.Declared;
import org.apache.qpid.proton.amqp.transaction.Discharge;
import org.apache.qpid.proton.amqp.transport.SenderSettleMode;
import org.apache.qpid.proton.amqp.transport.Target;
import org.apache.qpid.proton.message.Message;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.net.URI;
import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.atomic.AtomicBoolean;

class Controller {
    private static final Logger TRACE_LOGGER = LoggerFactory.getLogger(Controller.class);
    private MessagingFactory messagingFactory;
    private CoreMessageSender internalSender;
    private AtomicBoolean isInitialized = new AtomicBoolean(false);
    private URI namespaceEndpointURI;
    private ClientSettings clientSettings;

    Controller(URI namespaceEndpointURI, MessagingFactory factory, ClientSettings clientSettings) {
        this.namespaceEndpointURI = namespaceEndpointURI;
        this.messagingFactory = factory;
        this.clientSettings = clientSettings;
    }

    synchronized CompletableFuture initializeAsync() {
        if (this.isInitialized.get()) {
            return CompletableFuture.completedFuture(null);
        } else {
            TRACE_LOGGER.info("Creating MessageSender to coordinator");
            CompletableFuture senderFuture = CoreMessageSender.create(
                    this.messagingFactory,
                    StringUtil.getShortRandomString(),
                    null,
                    Controller.getControllerLinkSettings(this.messagingFactory));
            CompletableFuture postSenderCreationFuture = new CompletableFuture();
            senderFuture.handleAsync((s, coreSenderCreationEx) -> {
                if (coreSenderCreationEx == null) {
                    this.internalSender = s;
                    this.isInitialized.set(true);
                    TRACE_LOGGER.info("Created MessageSender to coordinator");
                    postSenderCreationFuture.complete(null);
                } else {
                    Throwable cause = ExceptionUtil.extractAsyncCompletionCause(coreSenderCreationEx);
                    TRACE_LOGGER.info("Creating MessageSender to coordinator failed", cause);
                    this.messagingFactory.closeAsync();
                    postSenderCreationFuture.completeExceptionally(cause);
                }
                return null;
            }, MessagingFactory.INTERNAL_THREAD_POOL);
            return postSenderCreationFuture;
        }
    }

    public CompletableFuture declareAsync() {
        Message message = Message.Factory.create();
        Declare declare = new Declare();
        message.setBody(new AmqpValue(declare));

        return this.internalSender.sendAndReturnDeliveryStateAsync(
                message,
                TransactionContext.NULL_TXN)
                .thenApply(state -> {
                    Binary txnId = null;
                    if (state instanceof Declared) {
                        Declared declared = (Declared) state;
                        txnId = declared.getTxnId();
                        TRACE_LOGGER.debug("New TX started: {}", txnId);
                    } else {
                        CompletableFuture exceptionFuture = new CompletableFuture<>();
                        exceptionFuture.completeExceptionally(new UnsupportedOperationException("Received unknown state: " + state.toString()));
                    }

                    return txnId;
                });
    }

    public CompletableFuture dischargeAsync(Binary txnId, boolean isCommit) {
        Message message = Message.Factory.create();
        Discharge discharge = new Discharge();
        discharge.setFail(!isCommit);
        discharge.setTxnId(txnId);
        message.setBody(new AmqpValue(discharge));        

        return this.internalSender.sendAndReturnDeliveryStateAsync(
                message,
                TransactionContext.NULL_TXN)
                .thenCompose(state -> {
                    if (state instanceof Accepted) {
                        return CompletableFuture.completedFuture(null);
                    } else {
                        CompletableFuture returnTask = new CompletableFuture<>();
                        returnTask.completeExceptionally(new UnsupportedOperationException("Received unknown state: " + state.toString()));
                        return returnTask;
                    }
                });
    }

    protected CompletableFuture closeAsync() {
        return null;
    }

    private static SenderLinkSettings getControllerLinkSettings(MessagingFactory underlyingFactory) {
        SenderLinkSettings linkSettings = new SenderLinkSettings();
        linkSettings.linkPath = "coordinator";

        final Target target = new Coordinator();
        linkSettings.target = target;
        linkSettings.source = new Source();
        linkSettings.settleMode = SenderSettleMode.UNSETTLED;
        linkSettings.requiresAuthentication = false;

        Map linkProperties = new HashMap<>();
        // ServiceBus expects timeout to be of type unsignedint
        linkProperties.put(ClientConstants.LINK_TIMEOUT_PROPERTY, UnsignedInteger.valueOf(Util.adjustServerTimeout(underlyingFactory.getOperationTimeout()).toMillis()));
        linkSettings.linkProperties = linkProperties;

        return linkSettings;
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy