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

com.datastax.driver.core.Connection Maven / Gradle / Ivy

/*
 * Copyright DataStax, Inc.
 *
 * Licensed 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.
 */

/*
 * Copyright (C) 2019 ScyllaDB
 *
 * Modified by ScyllaDB
 */
package com.datastax.driver.core;

import static com.datastax.driver.core.Message.Response.Type.ERROR;
import static io.netty.handler.timeout.IdleState.READER_IDLE;

import com.datastax.driver.core.Responses.Result.SetKeyspace;
import com.datastax.driver.core.Responses.Supported;
import com.datastax.driver.core.exceptions.AuthenticationException;
import com.datastax.driver.core.exceptions.BusyConnectionException;
import com.datastax.driver.core.exceptions.ConnectionException;
import com.datastax.driver.core.exceptions.CrcMismatchException;
import com.datastax.driver.core.exceptions.DriverException;
import com.datastax.driver.core.exceptions.DriverInternalError;
import com.datastax.driver.core.exceptions.FrameTooLongException;
import com.datastax.driver.core.exceptions.OperationTimedOutException;
import com.datastax.driver.core.exceptions.TransportException;
import com.datastax.driver.core.exceptions.UnsupportedProtocolVersionException;
import com.datastax.driver.core.utils.MoreFutures;
import com.datastax.driver.core.utils.MoreObjects;
import com.google.common.annotations.VisibleForTesting;
import com.google.common.collect.Lists;
import com.google.common.collect.MapMaker;
import com.google.common.util.concurrent.AbstractFuture;
import com.google.common.util.concurrent.AsyncFunction;
import com.google.common.util.concurrent.FutureCallback;
import com.google.common.util.concurrent.Futures;
import com.google.common.util.concurrent.ListenableFuture;
import com.google.common.util.concurrent.ListeningExecutorService;
import com.google.common.util.concurrent.SettableFuture;
import com.google.common.util.concurrent.Uninterruptibles;
import io.netty.bootstrap.Bootstrap;
import io.netty.channel.Channel;
import io.netty.channel.ChannelFuture;
import io.netty.channel.ChannelFutureListener;
import io.netty.channel.ChannelHandler;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelInitializer;
import io.netty.channel.ChannelOption;
import io.netty.channel.ChannelPipeline;
import io.netty.channel.EventLoop;
import io.netty.channel.EventLoopGroup;
import io.netty.channel.SimpleChannelInboundHandler;
import io.netty.channel.group.ChannelGroup;
import io.netty.channel.group.DefaultChannelGroup;
import io.netty.channel.socket.SocketChannel;
import io.netty.handler.codec.DecoderException;
import io.netty.handler.ssl.SslHandler;
import io.netty.handler.timeout.IdleStateEvent;
import io.netty.handler.timeout.IdleStateHandler;
import io.netty.util.Timeout;
import io.netty.util.Timer;
import io.netty.util.TimerTask;
import io.netty.util.concurrent.GlobalEventExecutor;
import java.io.IOException;
import java.lang.ref.WeakReference;
import java.net.BindException;
import java.net.InetSocketAddress;
import java.net.ServerSocket;
import java.security.InvalidParameterException;
import java.text.MessageFormat;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Queue;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.concurrent.ConcurrentMap;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.Executor;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.atomic.AtomicReference;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

// For LoggingHandler
// import org.jboss.netty.handler.logging.LoggingHandler;
// import org.jboss.netty.logging.InternalLogLevel;

/** A connection to a Cassandra Node. */
class Connection {

  private static final Logger logger = LoggerFactory.getLogger(Connection.class);
  private static final byte[] EMPTY_BYTE_ARRAY = new byte[0];

  private static final boolean DISABLE_COALESCING =
      SystemProperties.getBoolean("com.datastax.driver.DISABLE_COALESCING", false);
  private static final int FLUSHER_SCHEDULE_PERIOD_NS =
      SystemProperties.getInt("com.datastax.driver.FLUSHER_SCHEDULE_PERIOD_NS", 10000);

  private static final long ADV_SHARD_AWARENESS_BLOCK_ON_NAT = 1000000L * 60L * 1000L;

  private static final long ADV_SHARD_AWARENESS_BLOCK_ON_ERROR = 5 * 60 * 1000;

  enum State {
    OPEN,
    TRASHED,
    RESURRECTING,
    GONE
  }

  final AtomicReference state = new AtomicReference(State.OPEN);

  volatile long maxIdleTime;

  EndPoint endPoint;
  private final String name;
  private volatile Integer shardId = null;

  private int requestedShardId = -1;

  @VisibleForTesting volatile Channel channel;
  private final Factory factory;

  @VisibleForTesting final Dispatcher dispatcher;

  // Used by connection pooling to count how many requests are "in flight" on that connection.
  final AtomicInteger inFlight = new AtomicInteger(0);

  private final AtomicInteger writer = new AtomicInteger(0);

  private final AtomicReference targetKeyspace;
  private final SetKeyspaceAttempt defaultKeyspaceAttempt;

  private volatile boolean isInitialized;
  private final AtomicBoolean isDefunct = new AtomicBoolean();
  private final AtomicBoolean signaled = new AtomicBoolean();

  private final AtomicReference closeFuture =
      new AtomicReference();

  private final AtomicReference ownerRef = new AtomicReference();

  /**
   * Create a new connection to a Cassandra node and associate it with the given pool.
   *
   * @param name the connection name
   * @param endPoint the information to connect to the node
   * @param factory the connection factory to use
   * @param owner the component owning this connection (may be null). Note that an existing
   *     connection can also be associated to an owner later with {@link #setOwner(Owner)}.
   */
  protected Connection(String name, EndPoint endPoint, Factory factory, Owner owner) {
    this.endPoint = endPoint;
    this.factory = factory;
    this.dispatcher = new Dispatcher();
    this.name = name;
    this.ownerRef.set(owner);
    ListenableFuture thisFuture = Futures.immediateFuture(this);
    this.defaultKeyspaceAttempt = new SetKeyspaceAttempt(null, thisFuture);
    this.targetKeyspace = new AtomicReference(defaultKeyspaceAttempt);
  }

  /** Create a new connection to a Cassandra node. */
  Connection(String name, EndPoint endPoint, Factory factory) {
    this(name, endPoint, factory, null);
  }

  ListenableFuture initAsync() {
    return initAsync(-1, 0);
  }

  ListenableFuture initAsync(final int shardId, int serverPort) {
    if (factory.isShutdown)
      return Futures.immediateFailedFuture(
          new ConnectionException(endPoint, "Connection factory is shut down"));

    this.requestedShardId = shardId;

    final ProtocolVersion protocolVersion =
        factory.protocolVersion == null ? ProtocolVersion.DEFAULT : factory.protocolVersion;
    final SettableFuture channelReadyFuture = SettableFuture.create();

    try {
      final ProtocolOptions protocolOptions = factory.configuration.getProtocolOptions();
      final Bootstrap bootstrap = factory.newBootstrap();
      prepareBootstrap(bootstrap, protocolVersion, protocolOptions);

      final InetSocketAddress serverAddress =
          (serverPort == 0)
              ? endPoint.resolve()
              : new InetSocketAddress(endPoint.resolve().getAddress(), serverPort);

      final Owner owner = ownerRef.get();
      final HostConnectionPool pool =
          owner instanceof HostConnectionPool ? (HostConnectionPool) owner : null;
      final ShardingInfo shardingInfo = pool == null ? null : pool.host.getShardingInfo();
      if ((shardingInfo == null) && shardId != -1) {
        throw new InvalidParameterException(
            MessageFormat.format(
                "Requested connection to shard {0} of host {1}:{2}, but sharding info or pool is absent",
                shardId, serverAddress.getAddress().getHostAddress(), serverPort));
      }

      ChannelFuture future;
      final int lowPort, highPort;
      if (pool != null) {
        lowPort = pool.manager.configuration().getProtocolOptions().getLowLocalPort();
        highPort = pool.manager.configuration().getProtocolOptions().getHighLocalPort();
      } else {
        lowPort = highPort = -1;
      }

      if (shardId == -1) {
        future = bootstrap.connect(serverAddress);
      } else {
        int localPort =
            PortAllocator.getNextAvailablePort(
                shardingInfo.getShardsCount(), shardId, lowPort, highPort);
        if (localPort == -1) {
          throw new RuntimeException("Can't find free local port to use");
        }

        future = bootstrap.connect(serverAddress, new InetSocketAddress(localPort));
        logger.debug(
            "Connecting to shard {} using local port {} (shardCount: {})\n",
            shardId,
            localPort,
            shardingInfo.getShardsCount());
      }

      final ChannelFutureListener channelListener =
          new ChannelFutureListener() {
            @Override
            public void operationComplete(ChannelFuture future) throws Exception {
              if (future.cause() != null) {
                // Local port busy, let's try another one
                if (shardId != -1 && future.cause().getCause() instanceof BindException) {
                  int localPort =
                      PortAllocator.getNextAvailablePort(
                          shardingInfo.getShardsCount(), shardId, lowPort, highPort);
                  if (localPort != -1) {
                    if (future.channel() != null) {
                      future
                          .channel()
                          .close()
                          .addListener(
                              new ChannelFutureListener() {
                                @Override
                                public void operationComplete(ChannelFuture future)
                                    throws Exception {
                                  if (future.cause() != null) {
                                    logger.warn("Error while closing old channel", future.cause());
                                  }
                                }
                              });
                    }
                    prepareBootstrap(bootstrap, protocolVersion, protocolOptions);
                    ChannelFuture newFuture =
                        bootstrap.connect(serverAddress, new InetSocketAddress(localPort));
                    newFuture.addListener(this);
                    logger.debug(
                        "Retrying connecting to shard {} using local port {} (shardCount: {})\n",
                        shardId,
                        localPort,
                        shardingInfo.getShardsCount());
                    return;
                  }
                }
                logger.warn("Error creating netty channel to " + endPoint, future.cause());
              }

              writer.decrementAndGet();

              // Note: future.channel() can be null in some error cases, so we need to guard against
              // it in the rest of the code below.
              channel = future.channel();
              if (isClosed() && channel != null) {
                channel
                    .close()
                    .addListener(
                        new ChannelFutureListener() {
                          @Override
                          public void operationComplete(ChannelFuture future) throws Exception {
                            channelReadyFuture.setException(
                                new TransportException(
                                    Connection.this.endPoint,
                                    "Connection closed during initialization."));
                          }
                        });
              } else {
                if (channel != null) {
                  Connection.this.factory.allChannels.add(channel);
                }
                if (!future.isSuccess()) {
                  if (logger.isDebugEnabled()) {
                    logger.debug(
                        String.format(
                            "%s Error connecting to %s%s",
                            Connection.this,
                            Connection.this.endPoint,
                            extractMessage(future.cause())));
                  }
                  channelReadyFuture.setException(
                      new TransportException(
                          Connection.this.endPoint, "Cannot connect", future.cause()));
                  if (shardId != -1) {
                    // We are using advanced shard awareness, so pool must be non-null.
                    pool.tempBlockAdvShardAwareness(ADV_SHARD_AWARENESS_BLOCK_ON_ERROR);
                  }

                } else {
                  assert channel != null;
                  logger.debug(
                      "{} Connection established, initializing transport", Connection.this);
                  channel.closeFuture().addListener(new ChannelCloseListener());
                  channelReadyFuture.set(null);
                }
              }
            }
          };

      writer.incrementAndGet();
      future.addListener(channelListener);
    } catch (RuntimeException e) {
      closeAsync().force();
      throw e;
    }

    Executor initExecutor =
        factory.manager.configuration.getPoolingOptions().getInitializationExecutor();

    ListenableFuture queryOptionsFuture =
        GuavaCompatibility.INSTANCE.transformAsync(
            channelReadyFuture, onChannelReady(protocolVersion, initExecutor), initExecutor);

    ListenableFuture initializeTransportFuture =
        GuavaCompatibility.INSTANCE.transformAsync(
            queryOptionsFuture, onOptionsReady(protocolVersion, initExecutor), initExecutor);

    // Fallback on initializeTransportFuture so we can properly propagate specific exceptions.
    ListenableFuture initFuture =
        GuavaCompatibility.INSTANCE.withFallback(
            initializeTransportFuture,
            new AsyncFunction() {
              @Override
              public ListenableFuture apply(Throwable t) throws Exception {
                SettableFuture future = SettableFuture.create();
                // Make sure the connection gets properly closed.
                if (t instanceof ClusterNameMismatchException
                    || t instanceof UnsupportedProtocolVersionException) {
                  // Just propagate
                  closeAsync().force();
                  future.setException(t);
                } else {
                  // Defunct to ensure that the error will be signaled (marking the host down)
                  Throwable e =
                      (t instanceof ConnectionException
                              || t instanceof DriverException
                              || t instanceof InterruptedException
                              || t instanceof Error)
                          ? t
                          : new ConnectionException(
                              Connection.this.endPoint,
                              String.format(
                                  "Unexpected error during transport initialization (%s)", t),
                              t);
                  future.setException(defunct(e));
                }
                return future;
              }
            },
            initExecutor);

    // Ensure the connection gets closed if the caller cancels the returned future.
    GuavaCompatibility.INSTANCE.addCallback(
        initFuture,
        new MoreFutures.FailureCallback() {
          @Override
          public void onFailure(Throwable t) {
            if (!isClosed()) {
              closeAsync().force();
            }
          }
        },
        initExecutor);

    return initFuture;
  }

  private Bootstrap prepareBootstrap(
      Bootstrap bootstrap, ProtocolVersion protocolVersion, ProtocolOptions protocolOptions) {
    bootstrap.handler(
        new Initializer(
            this,
            protocolVersion,
            protocolOptions.getCompression().compressor(),
            protocolOptions.getSSLOptions(),
            factory.configuration.getPoolingOptions().getHeartbeatIntervalSeconds(),
            factory.configuration.getNettyOptions(),
            factory.configuration.getCodecRegistry(),
            factory.configuration.getMetricsOptions().isEnabled()
                ? factory.manager.metrics
                : null));
    return bootstrap;
  }

  private static String extractMessage(Throwable t) {
    if (t == null) return "";
    String msg = t.getMessage() == null || t.getMessage().isEmpty() ? t.toString() : t.getMessage();
    return " (" + msg + ')';
  }

  public ListenableFuture optionsQuery() {
    Future startupOptionsFuture = write(new Requests.Options());

    return GuavaCompatibility.INSTANCE.transformAsync(startupOptionsFuture, onSupportedResponse());
  }

  private AsyncFunction onChannelReady(
      final ProtocolVersion protocolVersion, final Executor initExecutor) {
    return new AsyncFunction() {
      @Override
      public ListenableFuture apply(Void input) throws Exception {
        Future startupOptionsFuture = write(new Requests.Options());
        return GuavaCompatibility.INSTANCE.transformAsync(
            startupOptionsFuture, onOptionsResponse(protocolVersion, initExecutor), initExecutor);
      }
    };
  }

  private AsyncFunction onOptionsResponse(
      final ProtocolVersion protocolVersion, final Executor initExecutor) {
    return new AsyncFunction() {
      @Override
      public ListenableFuture apply(Message.Response response) throws Exception {
        switch (response.type) {
          case SUPPORTED:
            Responses.Supported msg = (Supported) response;
            ShardingInfo.ConnectionShardingInfo sharding =
                ShardingInfo.parseShardingInfo(msg.supported);
            if (sharding != null) {
              getHost().setShardingInfo(sharding.shardingInfo);
              Connection.this.shardId = sharding.shardId;
              if (Connection.this.requestedShardId != -1
                  && Connection.this.requestedShardId != sharding.shardId) {
                logger.warn(
                    "Advanced shard awareness: requested connection to shard {}, but connected to {}. Is there a NAT between client and server?",
                    Connection.this.requestedShardId,
                    sharding.shardId);
                // Owner is a HostConnectionPool if we are using adv. shard awareness
                ((HostConnectionPool) Connection.this.ownerRef.get())
                    .tempBlockAdvShardAwareness(ADV_SHARD_AWARENESS_BLOCK_ON_NAT);
              }
            } else {
              getHost().setShardingInfo(null);
              Connection.this.shardId = 0;
            }
            LwtInfo lwt = LwtInfo.parseLwtInfo(msg.supported);
            if (lwt != null) {
              getHost().setLwtInfo(lwt);
            }
            return MoreFutures.VOID_SUCCESS;
          case ERROR:
            Responses.Error error = (Responses.Error) response;
            if (isUnsupportedProtocolVersion(error))
              throw unsupportedProtocolVersionException(
                  protocolVersion, error.serverProtocolVersion);
            throw new TransportException(
                endPoint,
                String.format(
                    "Got ERROR response message from server to an OPTIONS message: %s",
                    error.message));
          default:
            throw new TransportException(
                endPoint,
                String.format(
                    "Unexpected %s response message from server to an OPTIONS message",
                    response.type));
        }
      }
    };
  }

  private AsyncFunction onOptionsReady(
      final ProtocolVersion protocolVersion, final Executor initExecutor) {
    return new AsyncFunction() {
      @Override
      public ListenableFuture apply(Void input) throws Exception {
        ProtocolOptions protocolOptions = factory.configuration.getProtocolOptions();
        Map extraOptions = new HashMap();
        LwtInfo lwtInfo = getHost().getLwtInfo();
        if (lwtInfo != null) {
          lwtInfo.addOption(extraOptions);
        }
        Future startupResponseFuture =
            write(
                new Requests.Startup(
                    protocolOptions.getCompression(), protocolOptions.isNoCompact(), extraOptions));
        return GuavaCompatibility.INSTANCE.transformAsync(
            startupResponseFuture, onStartupResponse(protocolVersion, initExecutor), initExecutor);
      }
    };
  }

  private AsyncFunction onSupportedResponse() {
    return new AsyncFunction() {
      @Override
      public ListenableFuture apply(Message.Response response) throws Exception {
        switch (response.type) {
          case SUPPORTED:
            return getProductType((Responses.Supported) response);
          case ERROR:
            Responses.Error error = (Responses.Error) response;
            throw new TransportException(
                endPoint, String.format("Error initializing connection: %s", error.message));
          default:
            throw new TransportException(
                endPoint,
                String.format(
                    "Unexpected %s response message from server to a STARTUP message",
                    response.type));
        }
      }
    };
  }

  private AsyncFunction onStartupResponse(
      final ProtocolVersion protocolVersion, final Executor initExecutor) {
    return new AsyncFunction() {
      @Override
      public ListenableFuture apply(Message.Response response) throws Exception {
        switch (response.type) {
          case READY:
            return checkClusterName(protocolVersion, initExecutor);
          case ERROR:
            Responses.Error error = (Responses.Error) response;
            if (isUnsupportedProtocolVersion(error))
              throw unsupportedProtocolVersionException(
                  protocolVersion, error.serverProtocolVersion);
            throw new TransportException(
                endPoint, String.format("Error initializing connection: %s", error.message));
          case AUTHENTICATE:
            Responses.Authenticate authenticate = (Responses.Authenticate) response;
            Authenticator authenticator;
            try {
              if (factory.authProvider instanceof ExtendedAuthProvider) {
                authenticator =
                    ((ExtendedAuthProvider) factory.authProvider)
                        .newAuthenticator(endPoint, authenticate.authenticator);
              } else {
                authenticator =
                    factory.authProvider.newAuthenticator(
                        endPoint.resolve(), authenticate.authenticator);
              }
            } catch (AuthenticationException e) {
              incrementAuthErrorMetric();
              throw e;
            }
            switch (protocolVersion) {
              case V1:
                if (authenticator instanceof ProtocolV1Authenticator)
                  return authenticateV1(authenticator, protocolVersion, initExecutor);
                else
                  // DSE 3.x always uses SASL authentication backported from protocol v2
                  return authenticateV2(authenticator, protocolVersion, initExecutor);
              case V2:
              case V3:
              case V4:
              case V5:
              case V6:
                return authenticateV2(authenticator, protocolVersion, initExecutor);
              default:
                throw defunct(protocolVersion.unsupported());
            }
          default:
            throw new TransportException(
                endPoint,
                String.format(
                    "Unexpected %s response message from server to a STARTUP message",
                    response.type));
        }
      }
    };
  }

  // Due to C* gossip bugs, system.peers may report nodes that are gone from the cluster.
  // If these nodes have been recommissionned to another cluster and are up, nothing prevents the
  // driver from connecting
  // to them. So we check that the cluster the node thinks it belongs to is our cluster (JAVA-397).
  private ListenableFuture checkClusterName(
      ProtocolVersion protocolVersion, final Executor executor) {
    final String expected = factory.manager.metadata.clusterName;

    // At initialization, the cluster is not known yet
    if (expected == null) {
      markInitialized();
      return MoreFutures.VOID_SUCCESS;
    }

    DefaultResultSetFuture clusterNameFuture =
        new DefaultResultSetFuture(
            null,
            protocolVersion,
            new Requests.Query("select cluster_name from system.local where key = 'local'"));
    try {
      write(clusterNameFuture);
      return GuavaCompatibility.INSTANCE.transformAsync(
          clusterNameFuture,
          new AsyncFunction() {
            @Override
            public ListenableFuture apply(ResultSet rs) throws Exception {
              Row row = rs.one();
              String actual = row.getString("cluster_name");
              if (!expected.equals(actual))
                throw new ClusterNameMismatchException(endPoint, actual, expected);
              markInitialized();
              return MoreFutures.VOID_SUCCESS;
            }
          },
          executor);
    } catch (Exception e) {
      return Futures.immediateFailedFuture(e);
    }
  }

  private ListenableFuture getProductType(Responses.Supported response) {
    if (response.supported.containsKey("PRODUCT_TYPE")
        && response.supported.get("PRODUCT_TYPE").size() > 0) {
      return Futures.immediateFuture(response.supported.get("PRODUCT_TYPE").get(0));
    } else {
      return Futures.immediateFuture("");
    }
  }

  private void markInitialized() {
    isInitialized = true;
    Host.statesLogger.debug("[{}] {} Transport initialized, connection ready", endPoint, this);
  }

  private ListenableFuture authenticateV1(
      Authenticator authenticator, final ProtocolVersion protocolVersion, final Executor executor) {
    Requests.Credentials creds =
        new Requests.Credentials(((ProtocolV1Authenticator) authenticator).getCredentials());
    try {
      Future authResponseFuture = write(creds);
      return GuavaCompatibility.INSTANCE.transformAsync(
          authResponseFuture,
          new AsyncFunction() {
            @Override
            public ListenableFuture apply(Message.Response authResponse) throws Exception {
              switch (authResponse.type) {
                case READY:
                  return checkClusterName(protocolVersion, executor);
                case ERROR:
                  incrementAuthErrorMetric();
                  throw new AuthenticationException(
                      endPoint, ((Responses.Error) authResponse).message);
                default:
                  throw new TransportException(
                      endPoint,
                      String.format(
                          "Unexpected %s response message from server to a CREDENTIALS message",
                          authResponse.type));
              }
            }
          },
          executor);
    } catch (Exception e) {
      return Futures.immediateFailedFuture(e);
    }
  }

  private ListenableFuture authenticateV2(
      final Authenticator authenticator,
      final ProtocolVersion protocolVersion,
      final Executor executor) {
    byte[] initialResponse = authenticator.initialResponse();
    if (null == initialResponse) initialResponse = EMPTY_BYTE_ARRAY;

    try {
      Future authResponseFuture = write(new Requests.AuthResponse(initialResponse));
      return GuavaCompatibility.INSTANCE.transformAsync(
          authResponseFuture, onV2AuthResponse(authenticator, protocolVersion, executor), executor);
    } catch (Exception e) {
      return Futures.immediateFailedFuture(e);
    }
  }

  private AsyncFunction onV2AuthResponse(
      final Authenticator authenticator,
      final ProtocolVersion protocolVersion,
      final Executor executor) {
    return new AsyncFunction() {
      @Override
      public ListenableFuture apply(Message.Response authResponse) throws Exception {
        switch (authResponse.type) {
          case AUTH_SUCCESS:
            logger.trace("{} Authentication complete", this);
            authenticator.onAuthenticationSuccess(((Responses.AuthSuccess) authResponse).token);
            return checkClusterName(protocolVersion, executor);
          case AUTH_CHALLENGE:
            byte[] responseToServer =
                authenticator.evaluateChallenge(((Responses.AuthChallenge) authResponse).token);
            if (responseToServer == null) {
              // If we generate a null response, then authentication has completed, proceed without
              // sending a further response back to the server.
              logger.trace("{} Authentication complete (No response to server)", this);
              return checkClusterName(protocolVersion, executor);
            } else {
              // Otherwise, send the challenge response back to the server
              logger.trace("{} Sending Auth response to challenge", this);
              Future nextResponseFuture = write(new Requests.AuthResponse(responseToServer));
              return GuavaCompatibility.INSTANCE.transformAsync(
                  nextResponseFuture,
                  onV2AuthResponse(authenticator, protocolVersion, executor),
                  executor);
            }
          case ERROR:
            // This is not very nice, but we're trying to identify if we
            // attempted v2 auth against a server which only supports v1
            // The AIOOBE indicates that the server didn't recognise the
            // initial AuthResponse message
            String message = ((Responses.Error) authResponse).message;
            if (message.startsWith("java.lang.ArrayIndexOutOfBoundsException: 15"))
              message =
                  String.format(
                      "Cannot use authenticator %s with protocol version 1, "
                          + "only plain text authentication is supported with this protocol version",
                      authenticator);
            incrementAuthErrorMetric();
            throw new AuthenticationException(endPoint, message);
          default:
            throw new TransportException(
                endPoint,
                String.format(
                    "Unexpected %s response message from server to authentication message",
                    authResponse.type));
        }
      }
    };
  }

  private void incrementAuthErrorMetric() {
    if (factory.manager.configuration.getMetricsOptions().isEnabled()) {
      factory.manager.metrics.getErrorMetrics().getAuthenticationErrors().inc();
    }
  }

  private boolean isUnsupportedProtocolVersion(Responses.Error error) {
    // Testing for a specific string is a tad fragile but well, we don't have much choice
    // C* 2.1 reports a server error instead of protocol error, see CASSANDRA-9451
    return (error.code == ExceptionCode.PROTOCOL_ERROR || error.code == ExceptionCode.SERVER_ERROR)
        && (error.message.contains("Invalid or unsupported protocol version")
            // JAVA-2924: server is behind driver and considers the proposed version as beta
            || error.message.contains("Beta version of the protocol used"));
  }

  private UnsupportedProtocolVersionException unsupportedProtocolVersionException(
      ProtocolVersion triedVersion, ProtocolVersion serverProtocolVersion) {
    UnsupportedProtocolVersionException e =
        new UnsupportedProtocolVersionException(endPoint, triedVersion, serverProtocolVersion);
    logger.debug(e.getMessage());
    return e;
  }

  boolean isDefunct() {
    return isDefunct.get();
  }

  int maxAvailableStreams() {
    return dispatcher.streamIdHandler.maxAvailableStreams();
  }

   E defunct(E e) {
    if (isDefunct.compareAndSet(false, true)) {

      if (Host.statesLogger.isTraceEnabled()) Host.statesLogger.trace("Defuncting " + this, e);
      else if (Host.statesLogger.isDebugEnabled())
        Host.statesLogger.debug("Defuncting {} because: {}", this, e.getMessage());

      Host host = getHost();
      if (host != null) {
        // Sometimes close() can be called before defunct(); avoid decrementing the connection count
        // twice, but
        // we still want to signal the error to the conviction policy.
        boolean decrement = signaled.compareAndSet(false, true);

        boolean hostDown = host.convictionPolicy.signalConnectionFailure(this, decrement);
        if (hostDown) {
          factory.manager.signalHostDown(host, host.wasJustAdded());
        } else {
          notifyOwnerWhenDefunct();
        }
      }

      // Force the connection to close to make sure the future completes. Otherwise force() might
      // never get called and
      // threads will wait on the future forever.
      // (this also errors out pending handlers)
      closeAsync().force();
    }
    return e;
  }

  private void notifyOwnerWhenDefunct() {
    // If an error happens during initialization, the owner will detect it and take appropriate
    // action
    if (!isInitialized) return;

    Owner owner = this.ownerRef.get();
    if (owner != null) owner.onConnectionDefunct(this);
  }

  String keyspace() {
    return targetKeyspace.get().keyspace;
  }

  void setKeyspace(String keyspace) throws ConnectionException {
    if (keyspace == null) return;

    if (MoreObjects.equal(keyspace(), keyspace)) return;

    try {
      Uninterruptibles.getUninterruptibly(setKeyspaceAsync(keyspace));
    } catch (ConnectionException e) {
      throw defunct(e);
    } catch (BusyConnectionException e) {
      logger.warn(
          "Tried to set the keyspace on busy {}. "
              + "This should not happen but is not critical (it will be retried)",
          this);
      throw new ConnectionException(endPoint, "Tried to set the keyspace on busy connection");
    } catch (ExecutionException e) {
      Throwable cause = e.getCause();
      if (cause instanceof OperationTimedOutException) {
        // Rethrow so that the caller doesn't try to use the connection, but do not defunct as we
        // don't want to mark down
        logger.warn(
            "Timeout while setting keyspace on {}. "
                + "This should not happen but is not critical (it will be retried)",
            this);
        throw new ConnectionException(endPoint, "Timeout while setting keyspace on connection");
      } else {
        throw defunct(new ConnectionException(endPoint, "Error while setting keyspace", cause));
      }
    }
  }

  ListenableFuture setKeyspaceAsync(final String keyspace)
      throws ConnectionException, BusyConnectionException {
    SetKeyspaceAttempt existingAttempt = targetKeyspace.get();
    if (MoreObjects.equal(existingAttempt.keyspace, keyspace)) return existingAttempt.future;

    final SettableFuture ksFuture = SettableFuture.create();
    final SetKeyspaceAttempt attempt = new SetKeyspaceAttempt(keyspace, ksFuture);

    // Check for an existing keyspace attempt.
    while (true) {
      existingAttempt = targetKeyspace.get();
      // if existing attempts' keyspace matches what we are trying to set, use it.
      if (attempt.equals(existingAttempt)) {
        return existingAttempt.future;
      } else if (!existingAttempt.future.isDone()) {
        // If the existing attempt is still in flight, fail this attempt immediately.
        ksFuture.setException(
            new DriverException(
                "Aborting attempt to set keyspace to '"
                    + keyspace
                    + "' since there is already an in flight attempt to set keyspace to '"
                    + existingAttempt.keyspace
                    + "'.  "
                    + "This can happen if you try to USE different keyspaces from the same session simultaneously."));
        return ksFuture;
      } else if (targetKeyspace.compareAndSet(existingAttempt, attempt)) {
        // Otherwise, if the existing attempt is done, start a new set keyspace attempt for the new
        // keyspace.
        logger.debug("{} Setting keyspace {}", this, keyspace);
        // Note: we quote the keyspace below, because the name is the one coming from Cassandra, so
        // it's in the right case already
        Future future = write(new Requests.Query("USE \"" + keyspace + '"'));
        GuavaCompatibility.INSTANCE.addCallback(
            future,
            new FutureCallback() {

              @Override
              public void onSuccess(Message.Response response) {
                if (response instanceof SetKeyspace) {
                  logger.debug("{} Keyspace set to {}", Connection.this, keyspace);
                  ksFuture.set(Connection.this);
                } else {
                  // Unset this attempt so new attempts may be made for the same keyspace.
                  targetKeyspace.compareAndSet(attempt, defaultKeyspaceAttempt);
                  if (response.type == ERROR) {
                    Responses.Error error = (Responses.Error) response;
                    ksFuture.setException(defunct(error.asException(endPoint)));
                  } else {
                    ksFuture.setException(
                        defunct(
                            new DriverInternalError(
                                "Unexpected response while setting keyspace: " + response)));
                  }
                }
              }

              @Override
              public void onFailure(Throwable t) {
                targetKeyspace.compareAndSet(attempt, defaultKeyspaceAttempt);
                ksFuture.setException(t);
              }
            },
            factory.manager.configuration.getPoolingOptions().getInitializationExecutor());

        return ksFuture;
      }
    }
  }

  /**
   * Write a request on this connection.
   *
   * @param request the request to send
   * @return a future on the server response
   * @throws ConnectionException if the connection is closed
   * @throws TransportException if an I/O error while sending the request
   */
  Future write(Message.Request request) throws ConnectionException, BusyConnectionException {
    Future future = new Future(request);
    write(future);
    return future;
  }

  ResponseHandler write(ResponseCallback callback)
      throws ConnectionException, BusyConnectionException {
    return write(callback, -1, true);
  }

  ResponseHandler write(
      ResponseCallback callback, long statementReadTimeoutMillis, boolean startTimeout)
      throws ConnectionException, BusyConnectionException {

    ResponseHandler handler = new ResponseHandler(this, statementReadTimeoutMillis, callback);
    dispatcher.add(handler);

    Message.Request request = callback.request().setStreamId(handler.streamId);

    /*
     * We check for close/defunct *after* having set the handler because closing/defuncting
     * will set their flag and then error out handler if need. So, by doing the check after
     * having set the handler, we guarantee that even if we race with defunct/close, we may
     * never leave a handler that won't get an answer or be errored out.
     */
    if (isDefunct.get()) {
      dispatcher.removeHandler(handler, true);
      throw new ConnectionException(endPoint, "Write attempt on defunct connection");
    }

    if (isClosed()) {
      dispatcher.removeHandler(handler, true);
      throw new ConnectionException(endPoint, "Connection has been closed");
    }

    logger.trace("{}, stream {}, writing request {}", this, request.getStreamId(), request);
    writer.incrementAndGet();

    if (DISABLE_COALESCING) {
      channel.writeAndFlush(request).addListener(writeHandler(request, handler));
    } else {
      flush(new FlushItem(channel, request, writeHandler(request, handler)));
    }
    if (startTimeout) handler.startTimeout();

    return handler;
  }

  private ChannelFutureListener writeHandler(
      final Message.Request request, final ResponseHandler handler) {
    return new ChannelFutureListener() {
      @Override
      public void operationComplete(ChannelFuture writeFuture) {

        writer.decrementAndGet();

        if (!writeFuture.isSuccess()) {
          logger.debug(
              "{}, stream {}, Error writing request {}",
              Connection.this,
              request.getStreamId(),
              request);
          // Remove this handler from the dispatcher so it don't get notified of the error
          // twice (we will fail that method already)
          dispatcher.removeHandler(handler, true);

          final ConnectionException ce;
          if (writeFuture.cause() instanceof java.nio.channels.ClosedChannelException) {
            ce = new TransportException(endPoint, "Error writing: Closed channel");
          } else {
            ce = new TransportException(endPoint, "Error writing", writeFuture.cause());
          }
          final long latency = System.nanoTime() - handler.startTime;
          // This handler is executed while holding the writeLock of the channel.
          // defunct might close the pool, which will close all of its connections; closing a
          // connection also
          // requires its writeLock.
          // Therefore if multiple connections in the same pool get a write error, they could
          // deadlock;
          // we run defunct on a separate thread to avoid that.
          ListeningExecutorService executor = factory.manager.executor;
          if (!executor.isShutdown())
            executor.execute(
                new Runnable() {
                  @Override
                  public void run() {
                    handler.callback.onException(
                        Connection.this, defunct(ce), latency, handler.retryCount);
                  }
                });
        } else {
          logger.trace(
              "{}, stream {}, request sent successfully", Connection.this, request.getStreamId());
        }
      }
    };
  }

  boolean hasOwner() {
    return this.ownerRef.get() != null;
  }

  /** @return whether the connection was already associated with an owner */
  boolean setOwner(Owner owner) {
    return ownerRef.compareAndSet(null, owner);
  }

  public int shardId() {
    return shardId == null ? 0 : shardId;
  }

  /**
   * If the connection is part of a pool, return it to the pool. The connection should generally not
   * be reused after that.
   */
  void release(boolean busy) {
    Owner owner = ownerRef.get();
    if (owner instanceof HostConnectionPool)
      ((HostConnectionPool) owner).returnConnection(this, busy);
  }

  void release() {
    release(false);
  }

  boolean isClosed() {
    return closeFuture.get() != null;
  }

  /**
   * Closes the connection: no new writes will be accepted after this method has returned.
   *
   * 

However, a closed connection might still have ongoing queries awaiting for their result. * When all these ongoing queries have completed, the underlying channel will be closed; we refer * to this final state as "terminated". * * @return a future that will complete once the connection has terminated. * @see #tryTerminate(boolean) */ CloseFuture closeAsync() { ConnectionCloseFuture future = new ConnectionCloseFuture(); if (!closeFuture.compareAndSet(null, future)) { // close had already been called, return the existing future return closeFuture.get(); } logger.debug("{} closing connection", this); // Only signal if defunct hasn't done it already if (signaled.compareAndSet(false, true)) { Host host = getHost(); if (host != null) { host.convictionPolicy.signalConnectionClosed(this); } } boolean terminated = tryTerminate(false); if (!terminated) { // The time by which all pending requests should have normally completed (use twice the read // timeout for a generous // estimate -- note that this does not cover the eventuality that read timeout is updated // dynamically, but we can live // with that). long terminateTime = System.currentTimeMillis() + 2 * factory.getReadTimeoutMillis(); factory.reaper.register(this, terminateTime); } return future; } private Host getHost() { Metadata metadata = factory.manager.metadata; Host host = metadata.getHost(endPoint); // During init the host might not be in metatada.hosts yet, try the contact points if (host == null) { host = metadata.getContactPoint(endPoint); } return host; } /** * Tries to terminate a closed connection, i.e. release system resources. * *

This is called both by "normal" code and by {@link Cluster.ConnectionReaper}. * * @param force whether to proceed if there are still outstanding requests. * @return whether the connection has actually terminated. * @see #closeAsync() */ boolean tryTerminate(boolean force) { assert isClosed(); ConnectionCloseFuture future = closeFuture.get(); if (future.isDone()) { logger.debug("{} has already terminated", this); return true; } else { if (force || dispatcher.pending.isEmpty()) { if (force) logger.warn( "Forcing termination of {}. This should not happen and is likely a bug, please report.", this); future.force(); return true; } else { logger.debug("Not terminating {}: there are still pending requests", this); return false; } } } @Override public String toString() { return String.format( "Connection[%s, inFlight=%d, closed=%b]", name, inFlight.get(), isClosed()); } static class PortAllocator { private static final AtomicInteger lastPort = new AtomicInteger(-1); public static int getNextAvailablePort(int shardCount, int shardId, int lowPort, int highPort) { int lastPortValue, foundPort = -1; do { lastPortValue = lastPort.get(); // We will scan from lastPortValue // (or lowPort is there was no lastPort or lastPort is too low) int scanStart = lastPortValue == -1 ? lowPort : lastPortValue; if (scanStart < lowPort) { scanStart = lowPort; } // Round it up to "% shardCount == shardId" scanStart += (shardCount - scanStart % shardCount) + shardId; // Scan from scanStart upwards to highPort. for (int port = scanStart; port <= highPort; port += shardCount) { if (isTcpPortAvailable(port)) { foundPort = port; break; } } // If we started scanning from a high scanStart port // there might have been not enough ports left that are // smaller than highPort. Scan from the beginning // from the lowPort. if (foundPort == -1) { scanStart = lowPort + (shardCount - lowPort % shardCount) + shardId; for (int port = scanStart; port <= highPort; port += shardCount) { if (isTcpPortAvailable(port)) { foundPort = port; break; } } } // No luck! All ports taken! if (foundPort == -1) { return -1; } } while (!lastPort.compareAndSet(lastPortValue, foundPort)); return foundPort; } public static boolean isTcpPortAvailable(int port) { try { ServerSocket serverSocket = new ServerSocket(); try { serverSocket.setReuseAddress(false); serverSocket.bind(new InetSocketAddress(port), 1); return true; } finally { serverSocket.close(); } } catch (IOException ex) { return false; } } } static class Factory { final Timer timer; final EventLoopGroup eventLoopGroup; private final Class channelClass; private final ChannelGroup allChannels = new DefaultChannelGroup(GlobalEventExecutor.INSTANCE); private final ConcurrentMap idGenerators = new ConcurrentHashMap(); final DefaultResponseHandler defaultHandler; final Cluster.Manager manager; final Cluster.ConnectionReaper reaper; final Configuration configuration; final AuthProvider authProvider; private volatile boolean isShutdown; volatile ProtocolVersion protocolVersion; private final NettyOptions nettyOptions; Factory(Cluster.Manager manager, Configuration configuration) { this.defaultHandler = manager; this.manager = manager; this.reaper = manager.reaper; this.configuration = configuration; this.authProvider = configuration.getProtocolOptions().getAuthProvider(); this.protocolVersion = configuration.getProtocolOptions().initialProtocolVersion; this.nettyOptions = configuration.getNettyOptions(); this.eventLoopGroup = nettyOptions.eventLoopGroup( manager .configuration .getThreadingOptions() .createThreadFactory(manager.clusterName, "nio-worker")); this.channelClass = nettyOptions.channelClass(); this.timer = nettyOptions.timer( manager .configuration .getThreadingOptions() .createThreadFactory(manager.clusterName, "timeouter")); } int getPort() { return configuration.getProtocolOptions().getPort(); } /** * Opens a new connection to the node this factory points to. * * @return the newly created (and initialized) connection. * @throws ConnectionException if connection attempt fails. */ Connection open(Host host) throws ConnectionException, InterruptedException, UnsupportedProtocolVersionException, ClusterNameMismatchException { EndPoint endPoint = host.getEndPoint(); if (isShutdown) throw new ConnectionException(endPoint, "Connection factory is shut down"); host.convictionPolicy.signalConnectionsOpening(1); Connection connection = new Connection(buildConnectionName(host), endPoint, this); // This method opens the connection synchronously, so wait until it's initialized try { connection.initAsync().get(); return connection; } catch (ExecutionException e) { throw launderAsyncInitException(e); } } /** Same as open, but associate the created connection to the provided connection pool. */ Connection open(HostConnectionPool pool) throws ConnectionException, InterruptedException, UnsupportedProtocolVersionException, ClusterNameMismatchException { return open(pool, -1, 0); } Connection open(HostConnectionPool pool, int shardId, int serverPort) throws ConnectionException, InterruptedException, UnsupportedProtocolVersionException, ClusterNameMismatchException { pool.host.convictionPolicy.signalConnectionsOpening(1); Connection connection = new Connection(buildConnectionName(pool.host), pool.host.getEndPoint(), this, pool); try { connection.initAsync(shardId, serverPort).get(); return connection; } catch (ExecutionException e) { throw launderAsyncInitException(e); } } /** * Creates new connections and associate them to the provided connection pool, but does not * start them. */ List newConnections(HostConnectionPool pool, int count) { pool.host.convictionPolicy.signalConnectionsOpening(count); List connections = Lists.newArrayListWithCapacity(count); for (int i = 0; i < count; i++) connections.add( new Connection(buildConnectionName(pool.host), pool.host.getEndPoint(), this, pool)); return connections; } private String buildConnectionName(Host host) { return host.getEndPoint().toString() + '-' + getIdGenerator(host).getAndIncrement(); } static RuntimeException launderAsyncInitException(ExecutionException e) throws ConnectionException, InterruptedException, UnsupportedProtocolVersionException, ClusterNameMismatchException { Throwable t = e.getCause(); if (t instanceof ConnectionException) throw (ConnectionException) t; if (t instanceof InterruptedException) throw (InterruptedException) t; if (t instanceof UnsupportedProtocolVersionException) throw (UnsupportedProtocolVersionException) t; if (t instanceof ClusterNameMismatchException) throw (ClusterNameMismatchException) t; if (t instanceof DriverException) throw (DriverException) t; if (t instanceof Error) throw (Error) t; return new RuntimeException("Unexpected exception during connection initialization", t); } private AtomicInteger getIdGenerator(Host host) { AtomicInteger g = idGenerators.get(host); if (g == null) { g = new AtomicInteger(1); AtomicInteger old = idGenerators.putIfAbsent(host, g); if (old != null) g = old; } return g; } long getReadTimeoutMillis() { return configuration.getSocketOptions().getReadTimeoutMillis(); } private Bootstrap newBootstrap() { Bootstrap b = new Bootstrap(); b.group(eventLoopGroup).channel(channelClass); SocketOptions options = configuration.getSocketOptions(); b.option(ChannelOption.CONNECT_TIMEOUT_MILLIS, options.getConnectTimeoutMillis()); Boolean keepAlive = options.getKeepAlive(); if (keepAlive != null) b.option(ChannelOption.SO_KEEPALIVE, keepAlive); Boolean reuseAddress = options.getReuseAddress(); if (reuseAddress != null) b.option(ChannelOption.SO_REUSEADDR, reuseAddress); Integer soLinger = options.getSoLinger(); if (soLinger != null) b.option(ChannelOption.SO_LINGER, soLinger); Boolean tcpNoDelay = options.getTcpNoDelay(); if (tcpNoDelay != null) b.option(ChannelOption.TCP_NODELAY, tcpNoDelay); Integer receiveBufferSize = options.getReceiveBufferSize(); if (receiveBufferSize != null) b.option(ChannelOption.SO_RCVBUF, receiveBufferSize); Integer sendBufferSize = options.getSendBufferSize(); if (sendBufferSize != null) b.option(ChannelOption.SO_SNDBUF, sendBufferSize); nettyOptions.afterBootstrapInitialized(b); return b; } void shutdown() { // Make sure we skip creating connection from now on. isShutdown = true; // All channels should be closed already, we call this just to be sure. And we know // we're not on an I/O thread or anything, so just call await. allChannels.close().awaitUninterruptibly(); nettyOptions.onClusterClose(eventLoopGroup); nettyOptions.onClusterClose(timer); } } private static final class Flusher implements Runnable { final WeakReference eventLoopRef; final Queue queued = new ConcurrentLinkedQueue(); final AtomicBoolean running = new AtomicBoolean(false); final HashSet channels = new HashSet(); private Flusher(EventLoop eventLoop) { this.eventLoopRef = new WeakReference(eventLoop); } void start() { if (!running.get() && running.compareAndSet(false, true)) { EventLoop eventLoop = eventLoopRef.get(); if (eventLoop != null) eventLoop.execute(this); } } @Override public void run() { FlushItem flush; while (null != (flush = queued.poll())) { Channel channel = flush.channel; if (channel.isActive()) { channels.add(channel); channel.write(flush.request).addListener(flush.listener); } } // Always flush what we have (don't artificially delay to try to coalesce more messages) for (Channel channel : channels) channel.flush(); channels.clear(); // either reschedule or cancel running.set(false); if (queued.isEmpty() || !running.compareAndSet(false, true)) return; EventLoop eventLoop = eventLoopRef.get(); if (eventLoop != null && !eventLoop.isShuttingDown()) { if (FLUSHER_SCHEDULE_PERIOD_NS > 0) { eventLoop.schedule(this, FLUSHER_SCHEDULE_PERIOD_NS, TimeUnit.NANOSECONDS); } else { eventLoop.execute(this); } } } } private static final ConcurrentMap flusherLookup = new MapMaker().concurrencyLevel(16).weakKeys().makeMap(); private static class FlushItem { final Channel channel; final Object request; final ChannelFutureListener listener; private FlushItem(Channel channel, Object request, ChannelFutureListener listener) { this.channel = channel; this.request = request; this.listener = listener; } } private void flush(FlushItem item) { EventLoop loop = item.channel.eventLoop(); Flusher flusher = flusherLookup.get(loop); if (flusher == null) { Flusher alt = flusherLookup.putIfAbsent(loop, flusher = new Flusher(loop)); if (alt != null) flusher = alt; } flusher.queued.add(item); flusher.start(); } @ChannelHandler.Sharable class Dispatcher extends SimpleChannelInboundHandler { final StreamIdGenerator streamIdHandler; private final ConcurrentMap pending = new ConcurrentHashMap(); Dispatcher() { ProtocolVersion protocolVersion = factory.protocolVersion; if (protocolVersion == null) { // This happens for the first control connection because the protocol version has not been // negotiated yet. protocolVersion = ProtocolVersion.V2; } streamIdHandler = StreamIdGenerator.newInstance(protocolVersion); } void add(ResponseHandler handler) { ResponseHandler old = pending.put(handler.streamId, handler); assert old == null; } void removeHandler(ResponseHandler handler, boolean releaseStreamId) { // If we don't release the ID, mark first so that we can rely later on the fact that if // we receive a response for an ID with no handler, it's that this ID has been marked. if (!releaseStreamId) streamIdHandler.mark(handler.streamId); // If a RequestHandler is cancelled right when the response arrives, this method (called with // releaseStreamId=false) will race with messageReceived. // messageReceived could have already released the streamId, which could have already been // reused by another request. We must not remove the handler // if it's not ours, because that would cause the other request to hang forever. boolean removed = pending.remove(handler.streamId, handler); if (!removed) { // We raced, so if we marked the streamId above, that was wrong. if (!releaseStreamId) streamIdHandler.unmark(handler.streamId); return; } handler.cancelTimeout(); if (releaseStreamId) streamIdHandler.release(handler.streamId); if (isClosed()) tryTerminate(false); } @Override protected void channelRead0(ChannelHandlerContext ctx, Message.Response response) throws Exception { int streamId = response.getStreamId(); if (logger.isTraceEnabled()) logger.trace( "{}, stream {}, received: {}", Connection.this, streamId, asDebugString(response)); if (streamId < 0) { factory.defaultHandler.handle(response); return; } ResponseHandler handler = pending.remove(streamId); streamIdHandler.release(streamId); if (handler == null) { /* * During normal operation, we should not receive responses for which we don't have a handler. There is * two cases however where this can happen: * 1) The connection has been defuncted due to some internal error and we've raced between removing the * handler and actually closing the connection; since the original error has been logged, we're fine * ignoring this completely. * 2) This request has timed out. In that case, we've already switched to another host (or errored out * to the user). So log it for debugging purpose, but it's fine ignoring otherwise. */ streamIdHandler.unmark(streamId); if (logger.isDebugEnabled()) logger.debug( "{} Response received on stream {} but no handler set anymore (either the request has " + "timed out or it was closed due to another error). Received message is {}", Connection.this, streamId, asDebugString(response)); return; } handler.cancelTimeout(); handler.callback.onSet( Connection.this, response, System.nanoTime() - handler.startTime, handler.retryCount); // If we happen to be closed and we're the last outstanding request, we need to terminate the // connection // (note: this is racy as the signaling can be called more than once, but that's not a // problem) if (isClosed()) tryTerminate(false); } @Override public void userEventTriggered(ChannelHandlerContext ctx, Object evt) throws Exception { if (isInitialized && !isClosed() && evt instanceof IdleStateEvent && ((IdleStateEvent) evt).state() == READER_IDLE) { logger.debug( "{} was inactive for {} seconds, sending heartbeat", Connection.this, factory.configuration.getPoolingOptions().getHeartbeatIntervalSeconds()); write(HEARTBEAT_CALLBACK); } } // Make sure we don't print huge responses in debug/error logs. private String asDebugString(Object obj) { if (obj == null) return "null"; String msg = obj.toString(); if (msg.length() < 500) return msg; return msg.substring(0, 500) + "... [message of size " + msg.length() + " truncated]"; } @Override public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception { if (logger.isDebugEnabled()) logger.debug(String.format("%s connection error", Connection.this), cause); // Ignore exception while writing, this will be handled by write() directly if (writer.get() > 0) return; if (cause instanceof DecoderException) { Throwable error = cause.getCause(); // Special case, if we encountered a FrameTooLongException, raise exception on handler and // don't defunct it since // the connection is in an ok state. if (error instanceof FrameTooLongException) { FrameTooLongException ftle = (FrameTooLongException) error; int streamId = ftle.getStreamId(); ResponseHandler handler = pending.remove(streamId); streamIdHandler.release(streamId); if (handler == null) { streamIdHandler.unmark(streamId); if (logger.isDebugEnabled()) logger.debug( "{} FrameTooLongException received on stream {} but no handler set anymore (either the request has " + "timed out or it was closed due to another error).", Connection.this, streamId); return; } handler.cancelTimeout(); handler.callback.onException( Connection.this, ftle, System.nanoTime() - handler.startTime, handler.retryCount); return; } else if (error instanceof CrcMismatchException) { // Fall back to the defunct call below, but we want a clear warning in the logs logger.warn("CRC mismatch while decoding a response, dropping the connection", error); } } defunct( new TransportException( endPoint, String.format("Unexpected exception triggered (%s)", cause), cause)); } void errorOutAllHandler(ConnectionException ce) { Iterator iter = pending.values().iterator(); while (iter.hasNext()) { ResponseHandler handler = iter.next(); handler.cancelTimeout(); handler.callback.onException( Connection.this, ce, System.nanoTime() - handler.startTime, handler.retryCount); iter.remove(); } } } private class ChannelCloseListener implements ChannelFutureListener { @Override public void operationComplete(ChannelFuture future) throws Exception { // If we've closed the channel client side then we don't really want to defunct the // connection, but // if there is remaining thread waiting on us, we still want to wake them up if (!isInitialized || isClosed()) { dispatcher.errorOutAllHandler(new TransportException(endPoint, "Channel has been closed")); // we still want to force so that the future completes Connection.this.closeAsync().force(); } else defunct(new TransportException(endPoint, "Channel has been closed")); } } private static final ResponseCallback HEARTBEAT_CALLBACK = new ResponseCallback() { @Override public Message.Request request() { return new Requests.Options(); } @Override public int retryCount() { return 0; // no retries here } @Override public void onSet( Connection connection, Message.Response response, long latency, int retryCount) { switch (response.type) { case SUPPORTED: logger.debug("{} heartbeat query succeeded", connection); break; case ERROR: Responses.Error error = (Responses.Error) response; fail( connection, new ConnectionException( connection.endPoint, String.format( "Got ERROR response message from server to a heartbeat query: %s", error.message))); default: fail( connection, new ConnectionException( connection.endPoint, "Unexpected heartbeat response: " + response)); } } @Override public void onException( Connection connection, Exception exception, long latency, int retryCount) { // Nothing to do: the connection is already defunct if we arrive here } @Override public boolean onTimeout(Connection connection, long latency, int retryCount) { fail( connection, new ConnectionException(connection.endPoint, "Heartbeat query timed out")); return true; } private void fail(Connection connection, Exception e) { connection.defunct(e); } }; private class ConnectionCloseFuture extends CloseFuture { @Override public ConnectionCloseFuture force() { // Note: we must not call releaseExternalResources on the bootstrap, because this shutdown the // executors, which are shared // This method can be thrown during initialization, at which point channel is not yet set. // This is ok. if (channel == null) { set(null); return this; } // We're going to close this channel. If anyone is waiting on that connection, we should // defunct it otherwise it'll wait // forever. In general this won't happen since we get there only when all ongoing query are // done, but this can happen // if the shutdown is forced. This is a no-op if there is no handler set anymore. dispatcher.errorOutAllHandler(new TransportException(endPoint, "Connection has been closed")); ChannelFuture future = channel.close(); future.addListener( new ChannelFutureListener() { @Override public void operationComplete(ChannelFuture future) { factory.allChannels.remove(channel); if (future.cause() != null) { logger.warn("Error closing channel", future.cause()); ConnectionCloseFuture.this.setException(future.cause()); } else ConnectionCloseFuture.this.set(null); } }); return this; } } private class SetKeyspaceAttempt { private final String keyspace; private final ListenableFuture future; SetKeyspaceAttempt(String keyspace, ListenableFuture future) { this.keyspace = keyspace; this.future = future; } @Override public boolean equals(Object o) { if (this == o) return true; if (!(o instanceof SetKeyspaceAttempt)) return false; SetKeyspaceAttempt that = (SetKeyspaceAttempt) o; return keyspace != null ? keyspace.equals(that.keyspace) : that.keyspace == null; } @Override public int hashCode() { return keyspace != null ? keyspace.hashCode() : 0; } } static class Future extends AbstractFuture implements RequestHandler.Callback { private final Message.Request request; private volatile EndPoint endPoint; private volatile Host host; Future(Message.Request request) { this.request = request; } public Host getHost() { return host; } @Override public void register(RequestHandler handler) { // noop, we don't care about the handler here so far } @Override public Message.Request request() { return request; } @Override public int retryCount() { // This is ignored, as there is no retry logic in this class return 0; } @Override public void onSet( Connection connection, Message.Response response, ExecutionInfo info, Statement statement, long latency) { onSet(connection, response, latency, 0); } @Override public void onSet( Connection connection, Message.Response response, long latency, int retryCount) { this.endPoint = connection.endPoint; this.host = connection.getHost(); super.set(response); } @Override public void onException( Connection connection, Exception exception, long latency, int retryCount) { // If all nodes are down, we will get a null connection here. This is fine, if we have // an exception, consumers shouldn't assume the address is not null. if (connection != null) { this.endPoint = connection.endPoint; this.host = connection.getHost(); } super.setException(exception); } @Override public boolean onTimeout(Connection connection, long latency, int retryCount) { assert connection != null; // We always timeout on a specific connection, so this shouldn't be null this.endPoint = connection.endPoint; this.host = connection.getHost(); return super.setException(new OperationTimedOutException(connection.endPoint)); } EndPoint getEndPoint() { return endPoint; } } interface ResponseCallback { Message.Request request(); int retryCount(); void onSet(Connection connection, Message.Response response, long latency, int retryCount); void onException(Connection connection, Exception exception, long latency, int retryCount); boolean onTimeout(Connection connection, long latency, int retryCount); } static class ResponseHandler { final Connection connection; final int streamId; final ResponseCallback callback; final int retryCount; private final long readTimeoutMillis; private final long startTime; private volatile Timeout timeout; private final AtomicBoolean isCancelled = new AtomicBoolean(); ResponseHandler( Connection connection, long statementReadTimeoutMillis, ResponseCallback callback) throws BusyConnectionException { this.connection = connection; this.readTimeoutMillis = (statementReadTimeoutMillis >= 0) ? statementReadTimeoutMillis : connection.factory.getReadTimeoutMillis(); this.streamId = connection.dispatcher.streamIdHandler.next(); if (streamId == -1) throw new BusyConnectionException(connection.endPoint); this.callback = callback; this.retryCount = callback.retryCount(); this.startTime = System.nanoTime(); } void startTimeout() { this.timeout = this.readTimeoutMillis <= 0 ? null : connection.factory.timer.newTimeout( onTimeoutTask(), this.readTimeoutMillis, TimeUnit.MILLISECONDS); } void cancelTimeout() { if (timeout != null) timeout.cancel(); } boolean cancelHandler() { if (!isCancelled.compareAndSet(false, true)) return false; // We haven't really received a response: we want to remove the handle because we gave up on // that // request and there is no point in holding the handler, but we don't release the streamId. If // we // were, a new request could reuse that ID but get the answer to the request we just gave up // on instead // of its own answer, and we would have no way to detect that. connection.dispatcher.removeHandler(this, false); return true; } private TimerTask onTimeoutTask() { return new TimerTask() { @Override public void run(Timeout timeout) { if (callback.onTimeout(connection, System.nanoTime() - startTime, retryCount)) cancelHandler(); } }; } } interface DefaultResponseHandler { void handle(Message.Response response); } private static class Initializer extends ChannelInitializer { // Stateless handlers private static final Message.ProtocolDecoder messageDecoder = new Message.ProtocolDecoder(); private static final Message.ProtocolEncoder messageEncoderV1 = new Message.ProtocolEncoder(ProtocolVersion.V1); private static final Message.ProtocolEncoder messageEncoderV2 = new Message.ProtocolEncoder(ProtocolVersion.V2); private static final Message.ProtocolEncoder messageEncoderV3 = new Message.ProtocolEncoder(ProtocolVersion.V3); private static final Message.ProtocolEncoder messageEncoderV4 = new Message.ProtocolEncoder(ProtocolVersion.V4); private static final Message.ProtocolEncoder messageEncoderV5 = new Message.ProtocolEncoder(ProtocolVersion.V5); private static final Message.ProtocolEncoder messageEncoderV6 = new Message.ProtocolEncoder(ProtocolVersion.V6); private static final Frame.Encoder frameEncoder = new Frame.Encoder(); private final ProtocolVersion protocolVersion; private final Connection connection; private final FrameCompressor compressor; private final SSLOptions sslOptions; private final NettyOptions nettyOptions; private final ChannelHandler idleStateHandler; private final CodecRegistry codecRegistry; private final Metrics metrics; Initializer( Connection connection, ProtocolVersion protocolVersion, FrameCompressor compressor, SSLOptions sslOptions, int heartBeatIntervalSeconds, NettyOptions nettyOptions, CodecRegistry codecRegistry, Metrics metrics) { this.connection = connection; this.protocolVersion = protocolVersion; this.compressor = compressor; this.sslOptions = sslOptions; this.nettyOptions = nettyOptions; this.codecRegistry = codecRegistry; this.idleStateHandler = new IdleStateHandler(heartBeatIntervalSeconds, 0, 0); this.metrics = metrics; } @Override protected void initChannel(SocketChannel channel) throws Exception { // set the codec registry so that it can be accessed by ProtocolDecoder channel.attr(Message.CODEC_REGISTRY_ATTRIBUTE_KEY).set(codecRegistry); ChannelPipeline pipeline = channel.pipeline(); if (sslOptions != null) { SslHandler handler; if (sslOptions instanceof ExtendedRemoteEndpointAwareSslOptions) { handler = ((ExtendedRemoteEndpointAwareSslOptions) sslOptions) .newSSLHandler(channel, connection.endPoint); } else if (sslOptions instanceof RemoteEndpointAwareSSLOptions) { handler = ((RemoteEndpointAwareSSLOptions) sslOptions) .newSSLHandler(channel, connection.endPoint.resolve()); } else { handler = sslOptions.newSSLHandler(channel); } pipeline.addLast("ssl", handler); } // pipeline.addLast("debug", new LoggingHandler(LogLevel.INFO)); if (metrics != null) { pipeline.addLast( "inboundTrafficMeter", new InboundTrafficMeter(metrics.getBytesReceived())); pipeline.addLast("outboundTrafficMeter", new OutboundTrafficMeter(metrics.getBytesSent())); } pipeline.addLast("frameDecoder", new Frame.Decoder()); pipeline.addLast("frameEncoder", frameEncoder); pipeline.addLast("framingFormatHandler", new FramingFormatHandler(connection.factory)); if (compressor != null // Frame-level compression is only done in legacy protocol versions. In V5 and above, it // happens at a higher level ("segment" that groups multiple frames), so never install // those handlers. && protocolVersion.compareTo(ProtocolVersion.V5) < 0) { pipeline.addLast("frameDecompressor", new Frame.Decompressor(compressor)); pipeline.addLast("frameCompressor", new Frame.Compressor(compressor)); } pipeline.addLast("messageDecoder", messageDecoder); pipeline.addLast("messageEncoder", messageEncoderFor(protocolVersion)); pipeline.addLast("idleStateHandler", idleStateHandler); pipeline.addLast("dispatcher", connection.dispatcher); nettyOptions.afterChannelInitialized(channel); } private Message.ProtocolEncoder messageEncoderFor(ProtocolVersion version) { switch (version) { case V1: return messageEncoderV1; case V2: return messageEncoderV2; case V3: return messageEncoderV3; case V4: return messageEncoderV4; case V5: return messageEncoderV5; case V6: return messageEncoderV6; default: throw new DriverInternalError("Unsupported protocol version " + protocolVersion); } } } /** A component that "owns" a connection, and should be notified when it dies. */ interface Owner { void onConnectionDefunct(Connection connection); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy