com.couchbase.client.vbucket.BucketUpdateResponseHandler Maven / Gradle / Ivy
/**
* Copyright (C) 2009-2013 Couchbase, Inc.
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALING
* IN THE SOFTWARE.
*/
package com.couchbase.client.vbucket;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;
import java.util.logging.Level;
import java.util.logging.Logger;
import org.jboss.netty.buffer.ChannelBuffer;
import org.jboss.netty.channel.ChannelEvent;
import org.jboss.netty.channel.ChannelFuture;
import org.jboss.netty.channel.ChannelHandlerContext;
import org.jboss.netty.channel.ChannelState;
import org.jboss.netty.channel.ChannelStateEvent;
import org.jboss.netty.channel.ExceptionEvent;
import org.jboss.netty.channel.MessageEvent;
import org.jboss.netty.channel.SimpleChannelUpstreamHandler;
import org.jboss.netty.handler.codec.http.HttpChunk;
import org.jboss.netty.handler.codec.http.HttpResponse;
import org.jboss.netty.util.CharsetUtil;
/**
* A BucketUpdateResponseHandler.
*/
public class BucketUpdateResponseHandler extends SimpleChannelUpstreamHandler {
private volatile boolean readingChunks;
private String lastResponse;
private ChannelFuture receivedFuture;
private CountDownLatch latch;
private StringBuilder partialResponse;
private BucketMonitor monitor;
private static final Logger LOGGER =
Logger.getLogger(BucketUpdateResponseHandler.class.getName());
@Override
public void messageReceived(final ChannelHandlerContext context,
final MessageEvent event) {
ChannelFuture channelFuture = event.getFuture();
setReceivedFuture(channelFuture);
if (this.partialResponse == null) {
this.partialResponse = new StringBuilder();
}
if (readingChunks) {
HttpChunk chunk = (HttpChunk) event.getMessage();
if (chunk.isLast()) {
readingChunks = false;
} else {
String curChunk = chunk.getContent().toString(CharsetUtil.UTF_8);
/*
* Server sends four new lines in a chunk as a sentinal between
* responses.
*/
if (curChunk.matches("\n\n\n\n")) {
setLastResponse(partialResponse.toString());
partialResponse = null;
getLatch().countDown();
if (monitor != null) {
monitor.replaceConfig();
}
} else {
finerLog(curChunk);
finerLog("Chunk length is: " + curChunk.length());
partialResponse.append(curChunk);
channelFuture.setSuccess();
}
}
} else {
HttpResponse response = (HttpResponse) event.getMessage();
logResponse(response);
}
}
private void logResponse(HttpResponse response) {
finerLog("STATUS: " + response.getStatus());
finerLog("VERSION: " + response.getProtocolVersion());
if (!response.getHeaderNames().isEmpty()) {
for (String name : response.getHeaderNames()) {
for (String value : response.getHeaders(name)) {
finerLog("HEADER: " + name + " = " + value);
}
}
finerLog(System.getProperty("line.separator"));
}
if (response.getStatus().getCode() == 200 && response.isChunked()) {
readingChunks = true;
finerLog("CHUNKED CONTENT {");
} else if(response.getStatus().getCode() == 200) {
ChannelBuffer content = response.getContent();
if (content.readable()) {
finerLog("CONTENT {");
finerLog(content.toString(CharsetUtil.UTF_8));
finerLog("} END OF CONTENT");
}
} else {
throw new ConnectionException("Could not retrieve configuration chunk. "
+ "Response Code is: " + response.getStatus());
}
}
/**
* @return the lastResponse
*/
protected String getLastResponse() {
ChannelFuture channelFuture = getReceivedFuture();
if (channelFuture.awaitUninterruptibly(30, TimeUnit.SECONDS)) {
return lastResponse;
} else { // TODO: make this work with multiple servers
throw new ConnectionException("Cannot contact any server in the pool");
}
}
/**
* @param newLastResponse the lastResponse to set
*/
private void setLastResponse(String newLastResponse) {
this.lastResponse = newLastResponse;
}
/**
* @return the receivedFuture
*/
private ChannelFuture getReceivedFuture() {
try {
getLatch().await();
} catch (InterruptedException ex) {
finerLog("Getting received future has been interrupted.");
}
return receivedFuture;
}
/**
* @param newReceivedFuture the receivedFuture to set
*/
private void setReceivedFuture(ChannelFuture newReceivedFuture) {
this.receivedFuture = newReceivedFuture;
}
/**
* @return the latch
*/
private CountDownLatch getLatch() {
if (this.latch == null) {
latch = new CountDownLatch(1);
}
return latch;
}
private void finerLog(String message) {
LOGGER.log(Level.FINER, message);
}
@Override
public void handleUpstream(ChannelHandlerContext context, ChannelEvent event)
throws Exception {
if (event instanceof ChannelStateEvent) {
ChannelStateEvent csEvent = (ChannelStateEvent)event;
LOGGER.log(Level.FINEST, "Channel state changed: {0}\n\n", csEvent);
if (csEvent.getValue() == null
&& csEvent.getState() == ChannelState.CONNECTED) { // a disconnect
LOGGER.log(Level.FINE, "Channel has been disconnected on us, "
+ "restarting the monitor.");
monitor.notifyDisconnected(); // connection has been dropped
} else if(csEvent.getState() == ChannelState.OPEN
&& !Boolean.valueOf(csEvent.getValue().toString())) {
LOGGER.log(Level.FINE, "Channel has been closed on us, "
+ "restarting the monitor.");
monitor.notifyDisconnected(); // connection has been closed
} else {
LOGGER.log(Level.FINER, "Channel state change is not a disconnect. "
+ "Event value is {0} and Channel State is {1}.",
new Object[]{csEvent.getValue().toString(),
csEvent.getState().toString()});
}
}
if (event.getChannel().isConnected()) {
super.handleUpstream(context, event);
}
}
protected void setBucketMonitor(BucketMonitor newMonitor) {
this.monitor = newMonitor;
}
/*
* @todo we need to investigate why the exception occurs, and if there is a
* better solution to the problem than just shutting down the connection. For
* now just invalidate the BucketMonitor, and we will recreate the connection.
*/
@Override
public void exceptionCaught(ChannelHandlerContext ctx, ExceptionEvent e)
throws Exception {
Throwable ex = e.getCause();
LOGGER.log(Level.WARNING, "Exception occurred: " + ex.getMessage() + "\n");
StringBuilder sb = new StringBuilder();
for (StackTraceElement one : ex.getStackTrace()) {
sb.append(one.toString());
sb.append("\n");
}
LOGGER.log(Level.WARNING, sb.toString());
if (monitor != null) {
monitor.replaceConfig();
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy