org.infinispan.client.hotrod.impl.operations.FaultTolerantPingOperation Maven / Gradle / Ivy
package org.infinispan.client.hotrod.impl.operations;
import java.net.SocketAddress;
import java.util.concurrent.atomic.AtomicReference;
import org.infinispan.client.hotrod.ProtocolVersion;
import org.infinispan.client.hotrod.configuration.Configuration;
import org.infinispan.client.hotrod.exceptions.InvalidResponseException;
import org.infinispan.client.hotrod.impl.ClientTopology;
import org.infinispan.client.hotrod.impl.protocol.Codec;
import org.infinispan.client.hotrod.impl.protocol.HotRodConstants;
import org.infinispan.client.hotrod.impl.transport.netty.ChannelFactory;
import org.infinispan.client.hotrod.impl.transport.netty.HeaderDecoder;
import io.netty.buffer.ByteBuf;
import io.netty.channel.Channel;
import io.netty.handler.codec.DecoderException;
/**
* A fault tolerant ping operation that can survive to node failures.
*
* @author Galder Zamarreño
* @since 5.2
*/
public class FaultTolerantPingOperation extends RetryOnFailureOperation {
private final PingResponse.Decoder responseBuilder;
protected FaultTolerantPingOperation(Codec codec, ChannelFactory channelFactory,
byte[] cacheName, AtomicReference clientTopology, int flags,
Configuration cfg) {
super(PING_REQUEST, PING_RESPONSE, codec, channelFactory, cacheName, clientTopology, flags, cfg, null, null);
this.responseBuilder = new PingResponse.Decoder(cfg.version());
}
@Override
protected void executeOperation(Channel channel) {
sendHeaderAndRead(channel);
}
@Override
public void acceptResponse(ByteBuf buf, short status, HeaderDecoder decoder) {
responseBuilder.processResponse(codec, buf, decoder);
if (HotRodConstants.isSuccess(status)) {
PingResponse pingResponse = responseBuilder.build(status);
if (pingResponse.getVersion() != null && cfg.version() == ProtocolVersion.PROTOCOL_VERSION_AUTO) {
channelFactory.setNegotiatedCodec(pingResponse.getVersion().getCodec());
}
complete(pingResponse);
} else {
String hexStatus = Integer.toHexString(status);
if (log.isTraceEnabled())
log.tracef("Unknown response status: %s", hexStatus);
throw new InvalidResponseException("Unexpected response status: " + hexStatus);
}
}
@Override
protected Throwable handleException(Throwable cause, Channel channel, SocketAddress address) {
while (cause instanceof DecoderException && cause.getCause() != null) {
cause = cause.getCause();
}
PingResponse pingResponse = new PingResponse(cause);
if (pingResponse.isCacheNotFound()) {
complete(pingResponse);
return null;
}
return super.handleException(cause, channel, address);
}
@Override
protected void reset() {
super.reset();
responseBuilder.reset();
}
}