com.microsoft.azure.eventhubs.amqp.SessionHandler Maven / Gradle / Ivy
/*
* Copyright (c) Microsoft. All rights reserved.
* Licensed under the MIT license. See LICENSE file in the project root for full license information.
*/
package com.microsoft.azure.eventhubs.amqp;
import java.io.IOException;
import java.util.Iterator;
import java.util.Locale;
import java.util.function.BiConsumer;
import java.util.function.Consumer;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.apache.qpid.proton.amqp.transport.ErrorCondition;
import org.apache.qpid.proton.engine.BaseHandler;
import org.apache.qpid.proton.engine.Connection;
import org.apache.qpid.proton.engine.EndpointState;
import org.apache.qpid.proton.engine.Event;
import org.apache.qpid.proton.engine.Handler;
import org.apache.qpid.proton.engine.Session;
import org.apache.qpid.proton.engine.Transport;
import org.apache.qpid.proton.reactor.Reactor;
import com.microsoft.azure.eventhubs.ClientConstants;
import com.microsoft.azure.eventhubs.EventHubException;
public class SessionHandler extends BaseHandler {
protected static final Logger TRACE_LOGGER = LoggerFactory.getLogger(SessionHandler.class);
private final String entityName;
private final Consumer onRemoteSessionOpen;
private final BiConsumer onRemoteSessionOpenError;
private boolean sessionCreated = false;
private boolean sessionOpenErrorDispatched = false;
public SessionHandler(final String entityName, final Consumer onRemoteSessionOpen, final BiConsumer onRemoteSessionOpenError) {
this.entityName = entityName;
this.onRemoteSessionOpenError = onRemoteSessionOpenError;
this.onRemoteSessionOpen = onRemoteSessionOpen;
}
@Override
public void onSessionLocalOpen(Event e) {
if (this.onRemoteSessionOpenError != null) {
ReactorHandler reactorHandler = null;
final Reactor reactor = e.getReactor();
final Iterator reactorEventHandlers = reactor.getHandler().children();
while (reactorEventHandlers.hasNext()) {
final Handler currentHandler = reactorEventHandlers.next();
if (currentHandler instanceof ReactorHandler) {
reactorHandler = (ReactorHandler) currentHandler;
break;
}
}
final ReactorDispatcher reactorDispatcher = reactorHandler.getReactorDispatcher();
final Session session = e.getSession();
try {
reactorDispatcher.invoke(ClientConstants.SESSION_OPEN_TIMEOUT_IN_MS, new SessionTimeoutHandler(session));
} catch (IOException ignore) {
if (TRACE_LOGGER.isWarnEnabled()) {
TRACE_LOGGER.warn(String.format(Locale.US, "entityName[%s], reactorDispatcherError[%s]", this.entityName, ignore.getMessage()));
}
session.close();
this.onRemoteSessionOpenError.accept(
null,
new EventHubException(
false,
String.format("underlying IO of reactorDispatcher faulted with error: %s", ignore.getMessage()),
ignore));
}
}
}
@Override
public void onSessionRemoteOpen(Event e) {
if (TRACE_LOGGER.isInfoEnabled()) {
TRACE_LOGGER.info(String.format(Locale.US, "entityName[%s], sessionIncCapacity[%s], sessionOutgoingWindow[%s]",
this.entityName, e.getSession().getIncomingCapacity(), e.getSession().getOutgoingWindow()));
}
final Session session = e.getSession();
if (session != null && session.getLocalState() == EndpointState.UNINITIALIZED) {
session.open();
}
sessionCreated = true;
if (this.onRemoteSessionOpen != null)
this.onRemoteSessionOpen.accept(session);
}
@Override
public void onSessionLocalClose(Event e) {
if (TRACE_LOGGER.isInfoEnabled()) {
TRACE_LOGGER.info(String.format(Locale.US, "entityName[%s], condition[%s]", this.entityName,
e.getSession().getCondition() == null ? "none" : e.getSession().getCondition().toString()));
}
}
@Override
public void onSessionRemoteClose(Event e) {
if (TRACE_LOGGER.isInfoEnabled()) {
TRACE_LOGGER.info(String.format(Locale.US, "entityName[%s], condition[%s]", this.entityName,
e.getSession().getRemoteCondition() == null ? "none" : e.getSession().getRemoteCondition().toString()));
}
final Session session = e.getSession();
if (session != null && session.getLocalState() != EndpointState.CLOSED) {
session.close();
}
this.sessionOpenErrorDispatched = true;
if (!sessionCreated && this.onRemoteSessionOpenError != null)
this.onRemoteSessionOpenError.accept(session.getRemoteCondition(), null);
}
@Override
public void onSessionFinal(Event e) {
if (TRACE_LOGGER.isInfoEnabled()) {
TRACE_LOGGER.info(String.format(Locale.US, "entityName[%s]", this.entityName));
}
}
private class SessionTimeoutHandler extends DispatchHandler {
private final Session session;
public SessionTimeoutHandler(final Session session) {
this.session = session;
}
@Override
public void onEvent() {
// notify - if connection or transport error'ed out before even session open completed
if (!sessionCreated && !sessionOpenErrorDispatched) {
final Connection connection = session.getConnection();
if (connection != null) {
if (connection.getRemoteCondition() != null && connection.getRemoteCondition().getCondition() != null) {
session.close();
onRemoteSessionOpenError.accept(connection.getRemoteCondition(), null);
return;
}
final Transport transport = connection.getTransport();
if (transport != null && transport.getCondition() != null && transport.getCondition().getCondition() != null) {
session.close();
onRemoteSessionOpenError.accept(transport.getCondition(), null);
return;
}
}
session.close();
onRemoteSessionOpenError.accept(null, new EventHubException(false, "session creation timedout."));
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy