Reviewed-by: Tsz-Wo Nicholas Sze <szetszwo@apache.org> Co-authored-by: Wei-Chiu Chuang <weichiu@apache.org>
This commit is contained in:
parent
77cb778a44
commit
6847ec0647
@ -90,7 +90,7 @@
|
|||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>io.netty</groupId>
|
<groupId>io.netty</groupId>
|
||||||
<artifactId>netty</artifactId>
|
<artifactId>netty-all</artifactId>
|
||||||
<scope>compile</scope>
|
<scope>compile</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
|
@ -41,6 +41,8 @@ abstract public class MountdBase {
|
|||||||
private final RpcProgram rpcProgram;
|
private final RpcProgram rpcProgram;
|
||||||
private int udpBoundPort; // Will set after server starts
|
private int udpBoundPort; // Will set after server starts
|
||||||
private int tcpBoundPort; // Will set after server starts
|
private int tcpBoundPort; // Will set after server starts
|
||||||
|
private SimpleUdpServer udpServer = null;
|
||||||
|
private SimpleTcpServer tcpServer = null;
|
||||||
|
|
||||||
public RpcProgram getRpcProgram() {
|
public RpcProgram getRpcProgram() {
|
||||||
return rpcProgram;
|
return rpcProgram;
|
||||||
@ -57,7 +59,7 @@ public MountdBase(RpcProgram program) throws IOException {
|
|||||||
|
|
||||||
/* Start UDP server */
|
/* Start UDP server */
|
||||||
private void startUDPServer() {
|
private void startUDPServer() {
|
||||||
SimpleUdpServer udpServer = new SimpleUdpServer(rpcProgram.getPort(),
|
udpServer = new SimpleUdpServer(rpcProgram.getPort(),
|
||||||
rpcProgram, 1);
|
rpcProgram, 1);
|
||||||
rpcProgram.startDaemons();
|
rpcProgram.startDaemons();
|
||||||
try {
|
try {
|
||||||
@ -76,7 +78,7 @@ private void startUDPServer() {
|
|||||||
|
|
||||||
/* Start TCP server */
|
/* Start TCP server */
|
||||||
private void startTCPServer() {
|
private void startTCPServer() {
|
||||||
SimpleTcpServer tcpServer = new SimpleTcpServer(rpcProgram.getPort(),
|
tcpServer = new SimpleTcpServer(rpcProgram.getPort(),
|
||||||
rpcProgram, 1);
|
rpcProgram, 1);
|
||||||
rpcProgram.startDaemons();
|
rpcProgram.startDaemons();
|
||||||
try {
|
try {
|
||||||
@ -118,6 +120,14 @@ public void stop() {
|
|||||||
rpcProgram.unregister(PortmapMapping.TRANSPORT_TCP, tcpBoundPort);
|
rpcProgram.unregister(PortmapMapping.TRANSPORT_TCP, tcpBoundPort);
|
||||||
tcpBoundPort = 0;
|
tcpBoundPort = 0;
|
||||||
}
|
}
|
||||||
|
if (udpServer != null) {
|
||||||
|
udpServer.shutdown();
|
||||||
|
udpServer = null;
|
||||||
|
}
|
||||||
|
if (tcpServer != null) {
|
||||||
|
tcpServer.shutdown();
|
||||||
|
tcpServer = null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -35,6 +35,7 @@ public abstract class Nfs3Base {
|
|||||||
public static final Logger LOG = LoggerFactory.getLogger(Nfs3Base.class);
|
public static final Logger LOG = LoggerFactory.getLogger(Nfs3Base.class);
|
||||||
private final RpcProgram rpcProgram;
|
private final RpcProgram rpcProgram;
|
||||||
private int nfsBoundPort; // Will set after server starts
|
private int nfsBoundPort; // Will set after server starts
|
||||||
|
private SimpleTcpServer tcpServer = null;
|
||||||
|
|
||||||
public RpcProgram getRpcProgram() {
|
public RpcProgram getRpcProgram() {
|
||||||
return rpcProgram;
|
return rpcProgram;
|
||||||
@ -61,7 +62,7 @@ public void start(boolean register) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private void startTCPServer() {
|
private void startTCPServer() {
|
||||||
SimpleTcpServer tcpServer = new SimpleTcpServer(rpcProgram.getPort(),
|
tcpServer = new SimpleTcpServer(rpcProgram.getPort(),
|
||||||
rpcProgram, 0);
|
rpcProgram, 0);
|
||||||
rpcProgram.startDaemons();
|
rpcProgram.startDaemons();
|
||||||
try {
|
try {
|
||||||
@ -84,6 +85,10 @@ public void stop() {
|
|||||||
nfsBoundPort = 0;
|
nfsBoundPort = 0;
|
||||||
}
|
}
|
||||||
rpcProgram.stopDaemons();
|
rpcProgram.stopDaemons();
|
||||||
|
if (tcpServer != null) {
|
||||||
|
tcpServer.shutdown();
|
||||||
|
tcpServer = null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
/**
|
/**
|
||||||
* Priority of the nfsd shutdown hook.
|
* Priority of the nfsd shutdown hook.
|
||||||
|
@ -19,10 +19,9 @@
|
|||||||
|
|
||||||
import java.util.Arrays;
|
import java.util.Arrays;
|
||||||
|
|
||||||
|
import io.netty.buffer.ByteBuf;
|
||||||
|
import io.netty.channel.ChannelHandlerContext;
|
||||||
import org.apache.hadoop.oncrpc.RpcAcceptedReply.AcceptState;
|
import org.apache.hadoop.oncrpc.RpcAcceptedReply.AcceptState;
|
||||||
import org.jboss.netty.buffer.ChannelBuffer;
|
|
||||||
import org.jboss.netty.channel.ChannelHandlerContext;
|
|
||||||
import org.jboss.netty.channel.MessageEvent;
|
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
@ -58,10 +57,10 @@ private boolean validMessageLength(int len) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void messageReceived(ChannelHandlerContext ctx, MessageEvent e) {
|
public void channelRead(ChannelHandlerContext ctx, Object msg) {
|
||||||
ChannelBuffer buf = (ChannelBuffer) e.getMessage(); // Read reply
|
ByteBuf buf = (ByteBuf) msg; // Read reply
|
||||||
if (!validMessageLength(buf.readableBytes())) {
|
if (!validMessageLength(buf.readableBytes())) {
|
||||||
e.getChannel().close();
|
ctx.channel().close();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -83,7 +82,7 @@ public void messageReceived(ChannelHandlerContext ctx, MessageEvent e) {
|
|||||||
RpcDeniedReply deniedReply = (RpcDeniedReply) reply;
|
RpcDeniedReply deniedReply = (RpcDeniedReply) reply;
|
||||||
handle(deniedReply);
|
handle(deniedReply);
|
||||||
}
|
}
|
||||||
e.getChannel().close(); // shutdown now that request is complete
|
ctx.channel().close(); // shutdown now that request is complete
|
||||||
}
|
}
|
||||||
|
|
||||||
private void handle(RpcDeniedReply deniedReply) {
|
private void handle(RpcDeniedReply deniedReply) {
|
||||||
|
@ -19,9 +19,9 @@
|
|||||||
|
|
||||||
import java.net.SocketAddress;
|
import java.net.SocketAddress;
|
||||||
|
|
||||||
import org.jboss.netty.buffer.ChannelBuffer;
|
import io.netty.buffer.ByteBuf;
|
||||||
import org.jboss.netty.channel.Channel;
|
import io.netty.channel.Channel;
|
||||||
import org.jboss.netty.channel.ChannelHandlerContext;
|
import io.netty.channel.ChannelHandlerContext;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* RpcInfo records all contextual information of an RPC message. It contains
|
* RpcInfo records all contextual information of an RPC message. It contains
|
||||||
@ -29,11 +29,11 @@
|
|||||||
*/
|
*/
|
||||||
public final class RpcInfo {
|
public final class RpcInfo {
|
||||||
private final RpcMessage header;
|
private final RpcMessage header;
|
||||||
private final ChannelBuffer data;
|
private final ByteBuf data;
|
||||||
private final Channel channel;
|
private final Channel channel;
|
||||||
private final SocketAddress remoteAddress;
|
private final SocketAddress remoteAddress;
|
||||||
|
|
||||||
public RpcInfo(RpcMessage header, ChannelBuffer data,
|
public RpcInfo(RpcMessage header, ByteBuf data,
|
||||||
ChannelHandlerContext channelContext, Channel channel,
|
ChannelHandlerContext channelContext, Channel channel,
|
||||||
SocketAddress remoteAddress) {
|
SocketAddress remoteAddress) {
|
||||||
this.header = header;
|
this.header = header;
|
||||||
@ -46,7 +46,7 @@ public RpcMessage header() {
|
|||||||
return header;
|
return header;
|
||||||
}
|
}
|
||||||
|
|
||||||
public ChannelBuffer data() {
|
public ByteBuf data() {
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -22,16 +22,15 @@
|
|||||||
import java.net.InetSocketAddress;
|
import java.net.InetSocketAddress;
|
||||||
import java.net.SocketAddress;
|
import java.net.SocketAddress;
|
||||||
|
|
||||||
|
import io.netty.buffer.ByteBuf;
|
||||||
|
import io.netty.buffer.Unpooled;
|
||||||
|
import io.netty.channel.ChannelHandlerContext;
|
||||||
|
import io.netty.channel.ChannelInboundHandlerAdapter;
|
||||||
import org.apache.hadoop.thirdparty.com.google.common.annotations.VisibleForTesting;
|
import org.apache.hadoop.thirdparty.com.google.common.annotations.VisibleForTesting;
|
||||||
import org.apache.hadoop.oncrpc.RpcAcceptedReply.AcceptState;
|
import org.apache.hadoop.oncrpc.RpcAcceptedReply.AcceptState;
|
||||||
import org.apache.hadoop.oncrpc.security.VerifierNone;
|
import org.apache.hadoop.oncrpc.security.VerifierNone;
|
||||||
import org.apache.hadoop.portmap.PortmapMapping;
|
import org.apache.hadoop.portmap.PortmapMapping;
|
||||||
import org.apache.hadoop.portmap.PortmapRequest;
|
import org.apache.hadoop.portmap.PortmapRequest;
|
||||||
import org.jboss.netty.buffer.ChannelBuffer;
|
|
||||||
import org.jboss.netty.buffer.ChannelBuffers;
|
|
||||||
import org.jboss.netty.channel.ChannelHandlerContext;
|
|
||||||
import org.jboss.netty.channel.MessageEvent;
|
|
||||||
import org.jboss.netty.channel.SimpleChannelUpstreamHandler;
|
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
@ -39,7 +38,7 @@
|
|||||||
* Class for writing RPC server programs based on RFC 1050. Extend this class
|
* Class for writing RPC server programs based on RFC 1050. Extend this class
|
||||||
* and implement {@link #handleInternal} to handle the requests received.
|
* and implement {@link #handleInternal} to handle the requests received.
|
||||||
*/
|
*/
|
||||||
public abstract class RpcProgram extends SimpleChannelUpstreamHandler {
|
public abstract class RpcProgram extends ChannelInboundHandlerAdapter {
|
||||||
static final Logger LOG = LoggerFactory.getLogger(RpcProgram.class);
|
static final Logger LOG = LoggerFactory.getLogger(RpcProgram.class);
|
||||||
public static final int RPCB_PORT = 111;
|
public static final int RPCB_PORT = 111;
|
||||||
private final String program;
|
private final String program;
|
||||||
@ -161,9 +160,9 @@ public void startDaemons() {}
|
|||||||
public void stopDaemons() {}
|
public void stopDaemons() {}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void messageReceived(ChannelHandlerContext ctx, MessageEvent e)
|
public void channelRead(ChannelHandlerContext ctx, Object msg)
|
||||||
throws Exception {
|
throws Exception {
|
||||||
RpcInfo info = (RpcInfo) e.getMessage();
|
RpcInfo info = (RpcInfo) msg;
|
||||||
RpcCall call = (RpcCall) info.header();
|
RpcCall call = (RpcCall) info.header();
|
||||||
|
|
||||||
SocketAddress remoteAddress = info.remoteAddress();
|
SocketAddress remoteAddress = info.remoteAddress();
|
||||||
@ -221,7 +220,7 @@ private void sendAcceptedReply(RpcCall call, SocketAddress remoteAddress,
|
|||||||
out.writeInt(lowProgVersion);
|
out.writeInt(lowProgVersion);
|
||||||
out.writeInt(highProgVersion);
|
out.writeInt(highProgVersion);
|
||||||
}
|
}
|
||||||
ChannelBuffer b = ChannelBuffers.wrappedBuffer(out.asReadOnlyWrap()
|
ByteBuf b = Unpooled.wrappedBuffer(out.asReadOnlyWrap()
|
||||||
.buffer());
|
.buffer());
|
||||||
RpcResponse rsp = new RpcResponse(b, remoteAddress);
|
RpcResponse rsp = new RpcResponse(b, remoteAddress);
|
||||||
RpcUtil.sendRpcResponse(ctx, rsp);
|
RpcUtil.sendRpcResponse(ctx, rsp);
|
||||||
@ -234,7 +233,7 @@ protected static void sendRejectedReply(RpcCall call,
|
|||||||
RpcReply.ReplyState.MSG_DENIED,
|
RpcReply.ReplyState.MSG_DENIED,
|
||||||
RpcDeniedReply.RejectState.AUTH_ERROR, new VerifierNone());
|
RpcDeniedReply.RejectState.AUTH_ERROR, new VerifierNone());
|
||||||
reply.write(out);
|
reply.write(out);
|
||||||
ChannelBuffer buf = ChannelBuffers.wrappedBuffer(out.asReadOnlyWrap()
|
ByteBuf buf = Unpooled.wrappedBuffer(out.asReadOnlyWrap()
|
||||||
.buffer());
|
.buffer());
|
||||||
RpcResponse rsp = new RpcResponse(buf, remoteAddress);
|
RpcResponse rsp = new RpcResponse(buf, remoteAddress);
|
||||||
RpcUtil.sendRpcResponse(ctx, rsp);
|
RpcUtil.sendRpcResponse(ctx, rsp);
|
||||||
|
@ -19,27 +19,30 @@
|
|||||||
|
|
||||||
import java.net.SocketAddress;
|
import java.net.SocketAddress;
|
||||||
|
|
||||||
import org.jboss.netty.buffer.ChannelBuffer;
|
import io.netty.buffer.ByteBuf;
|
||||||
|
import io.netty.channel.DefaultAddressedEnvelope;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* RpcResponse encapsulates a response to a RPC request. It contains the data
|
* RpcResponse encapsulates a response to a RPC request. It contains the data
|
||||||
* that is going to cross the wire, as well as the information of the remote
|
* that is going to cross the wire, as well as the information of the remote
|
||||||
* peer.
|
* peer.
|
||||||
*/
|
*/
|
||||||
public class RpcResponse {
|
public class RpcResponse extends
|
||||||
private final ChannelBuffer data;
|
DefaultAddressedEnvelope<ByteBuf, SocketAddress> {
|
||||||
private final SocketAddress remoteAddress;
|
public RpcResponse(ByteBuf message, SocketAddress recipient) {
|
||||||
|
super(message, recipient, null);
|
||||||
public RpcResponse(ChannelBuffer data, SocketAddress remoteAddress) {
|
|
||||||
this.data = data;
|
|
||||||
this.remoteAddress = remoteAddress;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public ChannelBuffer data() {
|
public RpcResponse(ByteBuf message, SocketAddress recipient,
|
||||||
return data;
|
SocketAddress sender) {
|
||||||
|
super(message, recipient, sender);
|
||||||
|
}
|
||||||
|
|
||||||
|
public ByteBuf data() {
|
||||||
|
return this.content();
|
||||||
}
|
}
|
||||||
|
|
||||||
public SocketAddress remoteAddress() {
|
public SocketAddress remoteAddress() {
|
||||||
return remoteAddress;
|
return this.recipient();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -17,16 +17,18 @@
|
|||||||
*/
|
*/
|
||||||
package org.apache.hadoop.oncrpc;
|
package org.apache.hadoop.oncrpc;
|
||||||
|
|
||||||
|
import java.net.SocketAddress;
|
||||||
import java.nio.ByteBuffer;
|
import java.nio.ByteBuffer;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
import org.jboss.netty.buffer.ChannelBuffer;
|
import io.netty.buffer.ByteBuf;
|
||||||
import org.jboss.netty.buffer.ChannelBuffers;
|
import io.netty.buffer.Unpooled;
|
||||||
import org.jboss.netty.channel.Channel;
|
import io.netty.channel.ChannelHandler;
|
||||||
import org.jboss.netty.channel.ChannelHandlerContext;
|
import io.netty.channel.ChannelHandlerContext;
|
||||||
import org.jboss.netty.channel.Channels;
|
import io.netty.channel.ChannelInboundHandlerAdapter;
|
||||||
import org.jboss.netty.channel.MessageEvent;
|
import io.netty.channel.socket.DatagramPacket;
|
||||||
import org.jboss.netty.channel.SimpleChannelUpstreamHandler;
|
import io.netty.handler.codec.ByteToMessageDecoder;
|
||||||
import org.jboss.netty.handler.codec.frame.FrameDecoder;
|
import org.apache.hadoop.thirdparty.com.google.common.annotations.VisibleForTesting;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
@ -43,16 +45,16 @@ public static int getNewXid(String caller) {
|
|||||||
|
|
||||||
public static void sendRpcResponse(ChannelHandlerContext ctx,
|
public static void sendRpcResponse(ChannelHandlerContext ctx,
|
||||||
RpcResponse response) {
|
RpcResponse response) {
|
||||||
Channels.fireMessageReceived(ctx, response);
|
ctx.fireChannelRead(response);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static FrameDecoder constructRpcFrameDecoder() {
|
public static ByteToMessageDecoder constructRpcFrameDecoder() {
|
||||||
return new RpcFrameDecoder();
|
return new RpcFrameDecoder();
|
||||||
}
|
}
|
||||||
|
|
||||||
public static final SimpleChannelUpstreamHandler STAGE_RPC_MESSAGE_PARSER = new RpcMessageParserStage();
|
public static final ChannelInboundHandlerAdapter STAGE_RPC_MESSAGE_PARSER = new RpcMessageParserStage();
|
||||||
public static final SimpleChannelUpstreamHandler STAGE_RPC_TCP_RESPONSE = new RpcTcpResponseStage();
|
public static final ChannelInboundHandlerAdapter STAGE_RPC_TCP_RESPONSE = new RpcTcpResponseStage();
|
||||||
public static final SimpleChannelUpstreamHandler STAGE_RPC_UDP_RESPONSE = new RpcUdpResponseStage();
|
public static final ChannelInboundHandlerAdapter STAGE_RPC_UDP_RESPONSE = new RpcUdpResponseStage();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* An RPC client can separate a RPC message into several frames (i.e.,
|
* An RPC client can separate a RPC message into several frames (i.e.,
|
||||||
@ -62,44 +64,39 @@ public static FrameDecoder constructRpcFrameDecoder() {
|
|||||||
* RpcFrameDecoder is a stateful pipeline stage. It has to be constructed for
|
* RpcFrameDecoder is a stateful pipeline stage. It has to be constructed for
|
||||||
* each RPC client.
|
* each RPC client.
|
||||||
*/
|
*/
|
||||||
static class RpcFrameDecoder extends FrameDecoder {
|
static class RpcFrameDecoder extends ByteToMessageDecoder {
|
||||||
public static final Logger LOG =
|
public static final Logger LOG =
|
||||||
LoggerFactory.getLogger(RpcFrameDecoder.class);
|
LoggerFactory.getLogger(RpcFrameDecoder.class);
|
||||||
private ChannelBuffer currentFrame;
|
private volatile boolean isLast;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected Object decode(ChannelHandlerContext ctx, Channel channel,
|
protected void decode(ChannelHandlerContext ctx, ByteBuf buf,
|
||||||
ChannelBuffer buf) {
|
List<Object> out) {
|
||||||
|
|
||||||
if (buf.readableBytes() < 4)
|
if (buf.readableBytes() < 4) {
|
||||||
return null;
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
buf.markReaderIndex();
|
buf.markReaderIndex();
|
||||||
|
|
||||||
byte[] fragmentHeader = new byte[4];
|
byte[] fragmentHeader = new byte[4];
|
||||||
buf.readBytes(fragmentHeader);
|
buf.readBytes(fragmentHeader);
|
||||||
int length = XDR.fragmentSize(fragmentHeader);
|
int length = XDR.fragmentSize(fragmentHeader);
|
||||||
boolean isLast = XDR.isLastFragment(fragmentHeader);
|
isLast = XDR.isLastFragment(fragmentHeader);
|
||||||
|
|
||||||
if (buf.readableBytes() < length) {
|
if (buf.readableBytes() < length) {
|
||||||
buf.resetReaderIndex();
|
buf.resetReaderIndex();
|
||||||
return null;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
ChannelBuffer newFragment = buf.readSlice(length);
|
ByteBuf newFragment = buf.readSlice(length);
|
||||||
if (currentFrame == null) {
|
newFragment.retain();
|
||||||
currentFrame = newFragment;
|
out.add(newFragment);
|
||||||
} else {
|
|
||||||
currentFrame = ChannelBuffers.wrappedBuffer(currentFrame, newFragment);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isLast) {
|
@VisibleForTesting
|
||||||
ChannelBuffer completeFrame = currentFrame;
|
public boolean isLast() {
|
||||||
currentFrame = null;
|
return isLast;
|
||||||
return completeFrame;
|
|
||||||
} else {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -107,30 +104,44 @@ protected Object decode(ChannelHandlerContext ctx, Channel channel,
|
|||||||
* RpcMessageParserStage parses the network bytes and encapsulates the RPC
|
* RpcMessageParserStage parses the network bytes and encapsulates the RPC
|
||||||
* request into a RpcInfo instance.
|
* request into a RpcInfo instance.
|
||||||
*/
|
*/
|
||||||
static final class RpcMessageParserStage extends SimpleChannelUpstreamHandler {
|
@ChannelHandler.Sharable
|
||||||
|
static final class RpcMessageParserStage extends ChannelInboundHandlerAdapter {
|
||||||
private static final Logger LOG = LoggerFactory
|
private static final Logger LOG = LoggerFactory
|
||||||
.getLogger(RpcMessageParserStage.class);
|
.getLogger(RpcMessageParserStage.class);
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void messageReceived(ChannelHandlerContext ctx, MessageEvent e)
|
public void channelRead(ChannelHandlerContext ctx, Object msg)
|
||||||
throws Exception {
|
throws Exception {
|
||||||
ChannelBuffer buf = (ChannelBuffer) e.getMessage();
|
ByteBuf buf;
|
||||||
ByteBuffer b = buf.toByteBuffer().asReadOnlyBuffer();
|
SocketAddress remoteAddress;
|
||||||
|
if (msg instanceof DatagramPacket) {
|
||||||
|
DatagramPacket packet = (DatagramPacket)msg;
|
||||||
|
buf = packet.content();
|
||||||
|
remoteAddress = packet.sender();
|
||||||
|
} else {
|
||||||
|
buf = (ByteBuf) msg;
|
||||||
|
remoteAddress = ctx.channel().remoteAddress();
|
||||||
|
}
|
||||||
|
|
||||||
|
ByteBuffer b = buf.nioBuffer().asReadOnlyBuffer();
|
||||||
XDR in = new XDR(b, XDR.State.READING);
|
XDR in = new XDR(b, XDR.State.READING);
|
||||||
|
|
||||||
RpcInfo info = null;
|
RpcInfo info = null;
|
||||||
try {
|
try {
|
||||||
RpcCall callHeader = RpcCall.read(in);
|
RpcCall callHeader = RpcCall.read(in);
|
||||||
ChannelBuffer dataBuffer = ChannelBuffers.wrappedBuffer(in.buffer()
|
ByteBuf dataBuffer = Unpooled.wrappedBuffer(in.buffer()
|
||||||
.slice());
|
.slice());
|
||||||
info = new RpcInfo(callHeader, dataBuffer, ctx, e.getChannel(),
|
|
||||||
e.getRemoteAddress());
|
info = new RpcInfo(callHeader, dataBuffer, ctx, ctx.channel(),
|
||||||
|
remoteAddress);
|
||||||
} catch (Exception exc) {
|
} catch (Exception exc) {
|
||||||
LOG.info("Malformed RPC request from " + e.getRemoteAddress());
|
LOG.info("Malformed RPC request from " + remoteAddress);
|
||||||
|
} finally {
|
||||||
|
buf.release();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (info != null) {
|
if (info != null) {
|
||||||
Channels.fireMessageReceived(ctx, info);
|
ctx.fireChannelRead(info);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -139,16 +150,17 @@ public void messageReceived(ChannelHandlerContext ctx, MessageEvent e)
|
|||||||
* RpcTcpResponseStage sends an RpcResponse across the wire with the
|
* RpcTcpResponseStage sends an RpcResponse across the wire with the
|
||||||
* appropriate fragment header.
|
* appropriate fragment header.
|
||||||
*/
|
*/
|
||||||
private static class RpcTcpResponseStage extends SimpleChannelUpstreamHandler {
|
@ChannelHandler.Sharable
|
||||||
|
private static class RpcTcpResponseStage extends ChannelInboundHandlerAdapter {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void messageReceived(ChannelHandlerContext ctx, MessageEvent e)
|
public void channelRead(ChannelHandlerContext ctx, Object msg)
|
||||||
throws Exception {
|
throws Exception {
|
||||||
RpcResponse r = (RpcResponse) e.getMessage();
|
RpcResponse r = (RpcResponse) msg;
|
||||||
byte[] fragmentHeader = XDR.recordMark(r.data().readableBytes(), true);
|
byte[] fragmentHeader = XDR.recordMark(r.data().readableBytes(), true);
|
||||||
ChannelBuffer header = ChannelBuffers.wrappedBuffer(fragmentHeader);
|
ByteBuf header = Unpooled.wrappedBuffer(fragmentHeader);
|
||||||
ChannelBuffer d = ChannelBuffers.wrappedBuffer(header, r.data());
|
ByteBuf d = Unpooled.wrappedBuffer(header, r.data());
|
||||||
e.getChannel().write(d);
|
ctx.channel().writeAndFlush(d);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -156,14 +168,17 @@ public void messageReceived(ChannelHandlerContext ctx, MessageEvent e)
|
|||||||
* RpcUdpResponseStage sends an RpcResponse as a UDP packet, which does not
|
* RpcUdpResponseStage sends an RpcResponse as a UDP packet, which does not
|
||||||
* require a fragment header.
|
* require a fragment header.
|
||||||
*/
|
*/
|
||||||
|
@ChannelHandler.Sharable
|
||||||
private static final class RpcUdpResponseStage extends
|
private static final class RpcUdpResponseStage extends
|
||||||
SimpleChannelUpstreamHandler {
|
ChannelInboundHandlerAdapter {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void messageReceived(ChannelHandlerContext ctx, MessageEvent e)
|
public void channelRead(ChannelHandlerContext ctx, Object msg)
|
||||||
throws Exception {
|
throws Exception {
|
||||||
RpcResponse r = (RpcResponse) e.getMessage();
|
RpcResponse r = (RpcResponse) msg;
|
||||||
e.getChannel().write(r.data(), r.remoteAddress());
|
// TODO: check out https://github.com/netty/netty/issues/1282 for
|
||||||
|
// correct usage
|
||||||
|
ctx.channel().writeAndFlush(r.data());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -18,15 +18,16 @@
|
|||||||
package org.apache.hadoop.oncrpc;
|
package org.apache.hadoop.oncrpc;
|
||||||
|
|
||||||
import java.net.InetSocketAddress;
|
import java.net.InetSocketAddress;
|
||||||
import java.util.concurrent.Executors;
|
|
||||||
|
|
||||||
import org.jboss.netty.bootstrap.ClientBootstrap;
|
import io.netty.bootstrap.Bootstrap;
|
||||||
import org.jboss.netty.channel.ChannelFactory;
|
import io.netty.channel.ChannelFuture;
|
||||||
import org.jboss.netty.channel.ChannelFuture;
|
import io.netty.channel.ChannelInitializer;
|
||||||
import org.jboss.netty.channel.ChannelPipeline;
|
import io.netty.channel.ChannelOption;
|
||||||
import org.jboss.netty.channel.ChannelPipelineFactory;
|
import io.netty.channel.ChannelPipeline;
|
||||||
import org.jboss.netty.channel.Channels;
|
import io.netty.channel.nio.NioEventLoopGroup;
|
||||||
import org.jboss.netty.channel.socket.nio.NioClientSocketChannelFactory;
|
import io.netty.channel.socket.SocketChannel;
|
||||||
|
import io.netty.channel.socket.nio.NioSocketChannel;
|
||||||
|
import org.apache.hadoop.thirdparty.com.google.common.annotations.VisibleForTesting;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A simple TCP based RPC client which just sends a request to a server.
|
* A simple TCP based RPC client which just sends a request to a server.
|
||||||
@ -35,8 +36,9 @@ public class SimpleTcpClient {
|
|||||||
protected final String host;
|
protected final String host;
|
||||||
protected final int port;
|
protected final int port;
|
||||||
protected final XDR request;
|
protected final XDR request;
|
||||||
protected ChannelPipelineFactory pipelineFactory;
|
|
||||||
protected final boolean oneShot;
|
protected final boolean oneShot;
|
||||||
|
private NioEventLoopGroup workerGroup;
|
||||||
|
private ChannelFuture future;
|
||||||
|
|
||||||
public SimpleTcpClient(String host, int port, XDR request) {
|
public SimpleTcpClient(String host, int port, XDR request) {
|
||||||
this(host,port, request, true);
|
this(host,port, request, true);
|
||||||
@ -49,39 +51,53 @@ public SimpleTcpClient(String host, int port, XDR request, Boolean oneShot) {
|
|||||||
this.oneShot = oneShot;
|
this.oneShot = oneShot;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected ChannelPipelineFactory setPipelineFactory() {
|
protected ChannelInitializer<SocketChannel> setChannelHandler() {
|
||||||
this.pipelineFactory = new ChannelPipelineFactory() {
|
return new ChannelInitializer<SocketChannel>() {
|
||||||
@Override
|
@Override
|
||||||
public ChannelPipeline getPipeline() {
|
protected void initChannel(SocketChannel ch) throws Exception {
|
||||||
return Channels.pipeline(
|
ChannelPipeline p = ch.pipeline();
|
||||||
|
p.addLast(
|
||||||
RpcUtil.constructRpcFrameDecoder(),
|
RpcUtil.constructRpcFrameDecoder(),
|
||||||
new SimpleTcpClientHandler(request));
|
new SimpleTcpClientHandler(request)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
return this.pipelineFactory;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@VisibleForTesting
|
||||||
public void run() {
|
public void run() {
|
||||||
// Configure the client.
|
// Configure the client.
|
||||||
ChannelFactory factory = new NioClientSocketChannelFactory(
|
workerGroup = new NioEventLoopGroup();
|
||||||
Executors.newCachedThreadPool(), Executors.newCachedThreadPool(), 1, 1);
|
Bootstrap bootstrap = new Bootstrap()
|
||||||
ClientBootstrap bootstrap = new ClientBootstrap(factory);
|
.group(workerGroup)
|
||||||
|
.channel(NioSocketChannel.class);
|
||||||
// Set up the pipeline factory.
|
|
||||||
bootstrap.setPipelineFactory(setPipelineFactory());
|
|
||||||
|
|
||||||
bootstrap.setOption("tcpNoDelay", true);
|
|
||||||
bootstrap.setOption("keepAlive", true);
|
|
||||||
|
|
||||||
// Start the connection attempt.
|
|
||||||
ChannelFuture future = bootstrap.connect(new InetSocketAddress(host, port));
|
|
||||||
|
|
||||||
|
try {
|
||||||
|
future = bootstrap.handler(setChannelHandler())
|
||||||
|
.option(ChannelOption.TCP_NODELAY, true)
|
||||||
|
.option(ChannelOption.SO_KEEPALIVE, true)
|
||||||
|
.connect(new InetSocketAddress(host, port)).sync();
|
||||||
|
} catch (InterruptedException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
} finally {
|
||||||
if (oneShot) {
|
if (oneShot) {
|
||||||
// Wait until the connection is closed or the connection attempt fails.
|
stop();
|
||||||
future.getChannel().getCloseFuture().awaitUninterruptibly();
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void stop() {
|
||||||
|
try {
|
||||||
|
if (future != null) {
|
||||||
|
// Wait until the connection is closed or the connection attempt fails.
|
||||||
|
future.channel().closeFuture().sync();
|
||||||
|
}
|
||||||
|
|
||||||
|
} catch (InterruptedException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
} finally {
|
||||||
// Shut down thread pools to exit.
|
// Shut down thread pools to exit.
|
||||||
bootstrap.releaseExternalResources();
|
workerGroup.shutdownGracefully();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -17,19 +17,19 @@
|
|||||||
*/
|
*/
|
||||||
package org.apache.hadoop.oncrpc;
|
package org.apache.hadoop.oncrpc;
|
||||||
|
|
||||||
import org.jboss.netty.buffer.ChannelBuffer;
|
import io.netty.buffer.ByteBuf;
|
||||||
import org.jboss.netty.channel.ChannelHandlerContext;
|
import io.netty.channel.ChannelFuture;
|
||||||
import org.jboss.netty.channel.ChannelStateEvent;
|
import io.netty.channel.ChannelFutureListener;
|
||||||
import org.jboss.netty.channel.ExceptionEvent;
|
import io.netty.channel.ChannelHandlerContext;
|
||||||
import org.jboss.netty.channel.MessageEvent;
|
import io.netty.channel.ChannelInboundHandlerAdapter;
|
||||||
import org.jboss.netty.channel.SimpleChannelHandler;
|
import io.netty.util.ReferenceCountUtil;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A simple TCP based RPC client handler used by {@link SimpleTcpServer}.
|
* A simple TCP based RPC client handler used by {@link SimpleTcpServer}.
|
||||||
*/
|
*/
|
||||||
public class SimpleTcpClientHandler extends SimpleChannelHandler {
|
public class SimpleTcpClientHandler extends ChannelInboundHandlerAdapter {
|
||||||
public static final Logger LOG =
|
public static final Logger LOG =
|
||||||
LoggerFactory.getLogger(SimpleTcpClient.class);
|
LoggerFactory.getLogger(SimpleTcpClient.class);
|
||||||
protected final XDR request;
|
protected final XDR request;
|
||||||
@ -39,13 +39,13 @@ public SimpleTcpClientHandler(XDR request) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void channelConnected(ChannelHandlerContext ctx, ChannelStateEvent e) {
|
public void channelActive(ChannelHandlerContext ctx) throws Exception {
|
||||||
// Send the request
|
// Send the request
|
||||||
if (LOG.isDebugEnabled()) {
|
if (LOG.isDebugEnabled()) {
|
||||||
LOG.debug("sending PRC request");
|
LOG.debug("sending PRC request");
|
||||||
}
|
}
|
||||||
ChannelBuffer outBuf = XDR.writeMessageTcp(request, true);
|
ByteBuf outBuf = XDR.writeMessageTcp(request, true);
|
||||||
e.getChannel().write(outBuf);
|
ctx.channel().writeAndFlush(outBuf);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -53,13 +53,13 @@ public void channelConnected(ChannelHandlerContext ctx, ChannelStateEvent e) {
|
|||||||
* more interaction with the server.
|
* more interaction with the server.
|
||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
public void messageReceived(ChannelHandlerContext ctx, MessageEvent e) {
|
public void channelRead(ChannelHandlerContext ctx, Object msg) {
|
||||||
e.getChannel().close();
|
ctx.channel().close();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void exceptionCaught(ChannelHandlerContext ctx, ExceptionEvent e) {
|
public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) {
|
||||||
LOG.warn("Unexpected exception from downstream: ", e.getCause());
|
LOG.warn("Unexpected exception from downstream: ", cause.getCause());
|
||||||
e.getChannel().close();
|
ctx.channel().close();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -20,14 +20,17 @@
|
|||||||
import java.net.InetSocketAddress;
|
import java.net.InetSocketAddress;
|
||||||
import java.util.concurrent.Executors;
|
import java.util.concurrent.Executors;
|
||||||
|
|
||||||
import org.jboss.netty.bootstrap.ServerBootstrap;
|
import io.netty.bootstrap.ServerBootstrap;
|
||||||
import org.jboss.netty.channel.Channel;
|
import io.netty.channel.Channel;
|
||||||
import org.jboss.netty.channel.ChannelFactory;
|
import io.netty.channel.ChannelFuture;
|
||||||
import org.jboss.netty.channel.ChannelPipeline;
|
import io.netty.channel.ChannelInboundHandlerAdapter;
|
||||||
import org.jboss.netty.channel.ChannelPipelineFactory;
|
import io.netty.channel.ChannelInitializer;
|
||||||
import org.jboss.netty.channel.Channels;
|
import io.netty.channel.ChannelOption;
|
||||||
import org.jboss.netty.channel.SimpleChannelUpstreamHandler;
|
import io.netty.channel.ChannelPipeline;
|
||||||
import org.jboss.netty.channel.socket.nio.NioServerSocketChannelFactory;
|
import io.netty.channel.EventLoopGroup;
|
||||||
|
import io.netty.channel.nio.NioEventLoopGroup;
|
||||||
|
import io.netty.channel.socket.SocketChannel;
|
||||||
|
import io.netty.channel.socket.nio.NioServerSocketChannel;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
@ -39,9 +42,11 @@ public class SimpleTcpServer {
|
|||||||
LoggerFactory.getLogger(SimpleTcpServer.class);
|
LoggerFactory.getLogger(SimpleTcpServer.class);
|
||||||
protected final int port;
|
protected final int port;
|
||||||
protected int boundPort = -1; // Will be set after server starts
|
protected int boundPort = -1; // Will be set after server starts
|
||||||
protected final SimpleChannelUpstreamHandler rpcProgram;
|
protected final ChannelInboundHandlerAdapter rpcProgram;
|
||||||
private ServerBootstrap server;
|
private ServerBootstrap server;
|
||||||
private Channel ch;
|
private Channel ch;
|
||||||
|
private EventLoopGroup bossGroup;
|
||||||
|
private EventLoopGroup workerGroup;
|
||||||
|
|
||||||
/** The maximum number of I/O worker threads */
|
/** The maximum number of I/O worker threads */
|
||||||
protected final int workerCount;
|
protected final int workerCount;
|
||||||
@ -57,37 +62,32 @@ public SimpleTcpServer(int port, RpcProgram program, int workercount) {
|
|||||||
this.workerCount = workercount;
|
this.workerCount = workercount;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void run() {
|
public void run() throws InterruptedException {
|
||||||
// Configure the Server.
|
// Configure the Server.
|
||||||
ChannelFactory factory;
|
bossGroup = new NioEventLoopGroup();
|
||||||
if (workerCount == 0) {
|
workerGroup = new NioEventLoopGroup(workerCount, Executors.newCachedThreadPool());
|
||||||
// Use default workers: 2 * the number of available processors
|
|
||||||
factory = new NioServerSocketChannelFactory(
|
|
||||||
Executors.newCachedThreadPool(), Executors.newCachedThreadPool());
|
|
||||||
} else {
|
|
||||||
factory = new NioServerSocketChannelFactory(
|
|
||||||
Executors.newCachedThreadPool(), Executors.newCachedThreadPool(),
|
|
||||||
workerCount);
|
|
||||||
}
|
|
||||||
|
|
||||||
server = new ServerBootstrap(factory);
|
server = new ServerBootstrap();
|
||||||
server.setPipelineFactory(new ChannelPipelineFactory() {
|
|
||||||
|
|
||||||
|
server.group(bossGroup, workerGroup)
|
||||||
|
.channel(NioServerSocketChannel.class)
|
||||||
|
.childHandler(new ChannelInitializer<SocketChannel>() {
|
||||||
@Override
|
@Override
|
||||||
public ChannelPipeline getPipeline() throws Exception {
|
protected void initChannel(SocketChannel ch) throws Exception {
|
||||||
return Channels.pipeline(RpcUtil.constructRpcFrameDecoder(),
|
ChannelPipeline p = ch.pipeline();
|
||||||
|
p.addLast(RpcUtil.constructRpcFrameDecoder(),
|
||||||
RpcUtil.STAGE_RPC_MESSAGE_PARSER, rpcProgram,
|
RpcUtil.STAGE_RPC_MESSAGE_PARSER, rpcProgram,
|
||||||
RpcUtil.STAGE_RPC_TCP_RESPONSE);
|
RpcUtil.STAGE_RPC_TCP_RESPONSE);
|
||||||
}
|
}})
|
||||||
});
|
.childOption(ChannelOption.TCP_NODELAY, true)
|
||||||
server.setOption("child.tcpNoDelay", true);
|
.childOption(ChannelOption.SO_KEEPALIVE, true)
|
||||||
server.setOption("child.keepAlive", true);
|
.childOption(ChannelOption.SO_REUSEADDR, true)
|
||||||
server.setOption("child.reuseAddress", true);
|
.option(ChannelOption.SO_REUSEADDR, true);
|
||||||
server.setOption("reuseAddress", true);
|
|
||||||
|
|
||||||
// Listen to TCP port
|
// Listen to TCP port
|
||||||
ch = server.bind(new InetSocketAddress(port));
|
ChannelFuture f = server.bind(new InetSocketAddress(port)).sync();
|
||||||
InetSocketAddress socketAddr = (InetSocketAddress) ch.getLocalAddress();
|
ch = f.channel();
|
||||||
|
InetSocketAddress socketAddr = (InetSocketAddress) ch.localAddress();
|
||||||
boundPort = socketAddr.getPort();
|
boundPort = socketAddr.getPort();
|
||||||
|
|
||||||
LOG.info("Started listening to TCP requests at port " + boundPort + " for "
|
LOG.info("Started listening to TCP requests at port " + boundPort + " for "
|
||||||
@ -102,9 +102,17 @@ public int getBoundPort() {
|
|||||||
public void shutdown() {
|
public void shutdown() {
|
||||||
if (ch != null) {
|
if (ch != null) {
|
||||||
ch.close().awaitUninterruptibly();
|
ch.close().awaitUninterruptibly();
|
||||||
|
ch = null;
|
||||||
}
|
}
|
||||||
if (server != null) {
|
|
||||||
server.releaseExternalResources();
|
if (workerGroup != null) {
|
||||||
|
workerGroup.shutdownGracefully();
|
||||||
|
workerGroup = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (bossGroup != null) {
|
||||||
|
bossGroup.shutdownGracefully();
|
||||||
|
bossGroup = null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -20,12 +20,16 @@
|
|||||||
import java.net.InetSocketAddress;
|
import java.net.InetSocketAddress;
|
||||||
import java.util.concurrent.Executors;
|
import java.util.concurrent.Executors;
|
||||||
|
|
||||||
import org.jboss.netty.bootstrap.ConnectionlessBootstrap;
|
import io.netty.bootstrap.Bootstrap;
|
||||||
import org.jboss.netty.channel.Channel;
|
import io.netty.channel.Channel;
|
||||||
import org.jboss.netty.channel.Channels;
|
import io.netty.channel.ChannelFuture;
|
||||||
import org.jboss.netty.channel.SimpleChannelUpstreamHandler;
|
import io.netty.channel.ChannelInboundHandlerAdapter;
|
||||||
import org.jboss.netty.channel.socket.DatagramChannelFactory;
|
import io.netty.channel.ChannelInitializer;
|
||||||
import org.jboss.netty.channel.socket.nio.NioDatagramChannelFactory;
|
import io.netty.channel.ChannelOption;
|
||||||
|
import io.netty.channel.ChannelPipeline;
|
||||||
|
import io.netty.channel.EventLoopGroup;
|
||||||
|
import io.netty.channel.nio.NioEventLoopGroup;
|
||||||
|
import io.netty.channel.socket.nio.NioDatagramChannel;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
@ -39,36 +43,45 @@ public class SimpleUdpServer {
|
|||||||
private final int RECEIVE_BUFFER_SIZE = 65536;
|
private final int RECEIVE_BUFFER_SIZE = 65536;
|
||||||
|
|
||||||
protected final int port;
|
protected final int port;
|
||||||
protected final SimpleChannelUpstreamHandler rpcProgram;
|
protected final ChannelInboundHandlerAdapter rpcProgram;
|
||||||
protected final int workerCount;
|
protected final int workerCount;
|
||||||
protected int boundPort = -1; // Will be set after server starts
|
protected int boundPort = -1; // Will be set after server starts
|
||||||
private ConnectionlessBootstrap server;
|
private Bootstrap server;
|
||||||
private Channel ch;
|
private Channel ch;
|
||||||
|
private EventLoopGroup workerGroup;
|
||||||
|
|
||||||
public SimpleUdpServer(int port, SimpleChannelUpstreamHandler program,
|
public SimpleUdpServer(int port, ChannelInboundHandlerAdapter program,
|
||||||
int workerCount) {
|
int workerCount) {
|
||||||
this.port = port;
|
this.port = port;
|
||||||
this.rpcProgram = program;
|
this.rpcProgram = program;
|
||||||
this.workerCount = workerCount;
|
this.workerCount = workerCount;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void run() {
|
public void run() throws InterruptedException {
|
||||||
// Configure the client.
|
workerGroup = new NioEventLoopGroup(workerCount, Executors.newCachedThreadPool());
|
||||||
DatagramChannelFactory f = new NioDatagramChannelFactory(
|
|
||||||
Executors.newCachedThreadPool(), workerCount);
|
|
||||||
|
|
||||||
server = new ConnectionlessBootstrap(f);
|
server = new Bootstrap();
|
||||||
server.setPipeline(Channels.pipeline(RpcUtil.STAGE_RPC_MESSAGE_PARSER,
|
server.group(workerGroup)
|
||||||
rpcProgram, RpcUtil.STAGE_RPC_UDP_RESPONSE));
|
.channel(NioDatagramChannel.class)
|
||||||
|
.option(ChannelOption.SO_BROADCAST, true)
|
||||||
server.setOption("broadcast", "false");
|
.option(ChannelOption.SO_SNDBUF, SEND_BUFFER_SIZE)
|
||||||
server.setOption("sendBufferSize", SEND_BUFFER_SIZE);
|
.option(ChannelOption.SO_RCVBUF, RECEIVE_BUFFER_SIZE)
|
||||||
server.setOption("receiveBufferSize", RECEIVE_BUFFER_SIZE);
|
.option(ChannelOption.SO_REUSEADDR, true)
|
||||||
server.setOption("reuseAddress", true);
|
.handler(new ChannelInitializer<NioDatagramChannel>() {
|
||||||
|
@Override protected void initChannel(NioDatagramChannel ch)
|
||||||
|
throws Exception {
|
||||||
|
ChannelPipeline p = ch.pipeline();
|
||||||
|
p.addLast(
|
||||||
|
RpcUtil.STAGE_RPC_MESSAGE_PARSER,
|
||||||
|
rpcProgram,
|
||||||
|
RpcUtil.STAGE_RPC_UDP_RESPONSE);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
// Listen to the UDP port
|
// Listen to the UDP port
|
||||||
ch = server.bind(new InetSocketAddress(port));
|
ChannelFuture f = server.bind(new InetSocketAddress(port)).sync();
|
||||||
InetSocketAddress socketAddr = (InetSocketAddress) ch.getLocalAddress();
|
ch = f.channel();
|
||||||
|
InetSocketAddress socketAddr = (InetSocketAddress) ch.localAddress();
|
||||||
boundPort = socketAddr.getPort();
|
boundPort = socketAddr.getPort();
|
||||||
|
|
||||||
LOG.info("Started listening to UDP requests at port " + boundPort + " for "
|
LOG.info("Started listening to UDP requests at port " + boundPort + " for "
|
||||||
@ -83,9 +96,11 @@ public int getBoundPort() {
|
|||||||
public void shutdown() {
|
public void shutdown() {
|
||||||
if (ch != null) {
|
if (ch != null) {
|
||||||
ch.close().awaitUninterruptibly();
|
ch.close().awaitUninterruptibly();
|
||||||
|
ch = null;
|
||||||
}
|
}
|
||||||
if (server != null) {
|
if (workerGroup != null) {
|
||||||
server.releaseExternalResources();
|
workerGroup.shutdownGracefully();
|
||||||
|
workerGroup = null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -20,8 +20,8 @@
|
|||||||
import java.nio.ByteBuffer;
|
import java.nio.ByteBuffer;
|
||||||
import java.nio.charset.StandardCharsets;
|
import java.nio.charset.StandardCharsets;
|
||||||
|
|
||||||
import org.jboss.netty.buffer.ChannelBuffer;
|
import io.netty.buffer.ByteBuf;
|
||||||
import org.jboss.netty.buffer.ChannelBuffers;
|
import io.netty.buffer.Unpooled;
|
||||||
|
|
||||||
import org.apache.hadoop.thirdparty.com.google.common.annotations.VisibleForTesting;
|
import org.apache.hadoop.thirdparty.com.google.common.annotations.VisibleForTesting;
|
||||||
import org.apache.hadoop.thirdparty.com.google.common.base.Preconditions;
|
import org.apache.hadoop.thirdparty.com.google.common.base.Preconditions;
|
||||||
@ -242,7 +242,7 @@ static byte[] recordMark(int size, boolean last) {
|
|||||||
* @param last specifies last request or not
|
* @param last specifies last request or not
|
||||||
* @return TCP buffer
|
* @return TCP buffer
|
||||||
*/
|
*/
|
||||||
public static ChannelBuffer writeMessageTcp(XDR request, boolean last) {
|
public static ByteBuf writeMessageTcp(XDR request, boolean last) {
|
||||||
Preconditions.checkState(request.state == XDR.State.WRITING);
|
Preconditions.checkState(request.state == XDR.State.WRITING);
|
||||||
ByteBuffer b = request.buf.duplicate();
|
ByteBuffer b = request.buf.duplicate();
|
||||||
b.flip();
|
b.flip();
|
||||||
@ -250,7 +250,7 @@ public static ChannelBuffer writeMessageTcp(XDR request, boolean last) {
|
|||||||
ByteBuffer headerBuf = ByteBuffer.wrap(fragmentHeader);
|
ByteBuffer headerBuf = ByteBuffer.wrap(fragmentHeader);
|
||||||
|
|
||||||
// TODO: Investigate whether making a copy of the buffer is necessary.
|
// TODO: Investigate whether making a copy of the buffer is necessary.
|
||||||
return ChannelBuffers.copiedBuffer(headerBuf, b);
|
return Unpooled.wrappedBuffer(headerBuf, b);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -258,10 +258,10 @@ public static ChannelBuffer writeMessageTcp(XDR request, boolean last) {
|
|||||||
* @param response XDR response
|
* @param response XDR response
|
||||||
* @return UDP buffer
|
* @return UDP buffer
|
||||||
*/
|
*/
|
||||||
public static ChannelBuffer writeMessageUdp(XDR response) {
|
public static ByteBuf writeMessageUdp(XDR response) {
|
||||||
Preconditions.checkState(response.state == XDR.State.READING);
|
Preconditions.checkState(response.state == XDR.State.READING);
|
||||||
// TODO: Investigate whether making a copy of the buffer is necessary.
|
// TODO: Investigate whether making a copy of the buffer is necessary.
|
||||||
return ChannelBuffers.copiedBuffer(response.buf);
|
return Unpooled.copiedBuffer(response.buf);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static int fragmentSize(byte[] mark) {
|
public static int fragmentSize(byte[] mark) {
|
||||||
|
@ -22,21 +22,27 @@
|
|||||||
import java.util.concurrent.Executors;
|
import java.util.concurrent.Executors;
|
||||||
import java.util.concurrent.TimeUnit;
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
|
import io.netty.bootstrap.Bootstrap;
|
||||||
|
import io.netty.bootstrap.ServerBootstrap;
|
||||||
|
import io.netty.channel.Channel;
|
||||||
|
import io.netty.channel.ChannelFuture;
|
||||||
|
import io.netty.channel.ChannelInitializer;
|
||||||
|
import io.netty.channel.ChannelOption;
|
||||||
|
import io.netty.channel.ChannelPipeline;
|
||||||
|
import io.netty.channel.EventLoopGroup;
|
||||||
|
import io.netty.channel.group.ChannelGroup;
|
||||||
|
import io.netty.channel.group.DefaultChannelGroup;
|
||||||
|
import io.netty.channel.nio.NioEventLoopGroup;
|
||||||
|
import io.netty.channel.socket.SocketChannel;
|
||||||
|
import io.netty.channel.socket.nio.NioDatagramChannel;
|
||||||
|
import io.netty.channel.socket.nio.NioServerSocketChannel;
|
||||||
|
import io.netty.handler.logging.LogLevel;
|
||||||
|
import io.netty.handler.logging.LoggingHandler;
|
||||||
|
import io.netty.handler.timeout.IdleStateHandler;
|
||||||
|
import io.netty.util.concurrent.GlobalEventExecutor;
|
||||||
import org.apache.hadoop.oncrpc.RpcProgram;
|
import org.apache.hadoop.oncrpc.RpcProgram;
|
||||||
import org.apache.hadoop.oncrpc.RpcUtil;
|
import org.apache.hadoop.oncrpc.RpcUtil;
|
||||||
import org.apache.hadoop.util.StringUtils;
|
import org.apache.hadoop.util.StringUtils;
|
||||||
import org.jboss.netty.bootstrap.ConnectionlessBootstrap;
|
|
||||||
import org.jboss.netty.bootstrap.ServerBootstrap;
|
|
||||||
import org.jboss.netty.channel.Channel;
|
|
||||||
import org.jboss.netty.channel.ChannelPipeline;
|
|
||||||
import org.jboss.netty.channel.ChannelPipelineFactory;
|
|
||||||
import org.jboss.netty.channel.Channels;
|
|
||||||
import org.jboss.netty.channel.group.ChannelGroup;
|
|
||||||
import org.jboss.netty.channel.group.DefaultChannelGroup;
|
|
||||||
import org.jboss.netty.channel.socket.nio.NioDatagramChannelFactory;
|
|
||||||
import org.jboss.netty.channel.socket.nio.NioServerSocketChannelFactory;
|
|
||||||
import org.jboss.netty.handler.timeout.IdleStateHandler;
|
|
||||||
import org.jboss.netty.util.HashedWheelTimer;
|
|
||||||
|
|
||||||
import org.apache.hadoop.thirdparty.com.google.common.annotations.VisibleForTesting;
|
import org.apache.hadoop.thirdparty.com.google.common.annotations.VisibleForTesting;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
@ -49,11 +55,17 @@ final class Portmap {
|
|||||||
private static final Logger LOG = LoggerFactory.getLogger(Portmap.class);
|
private static final Logger LOG = LoggerFactory.getLogger(Portmap.class);
|
||||||
private static final int DEFAULT_IDLE_TIME_MILLISECONDS = 5000;
|
private static final int DEFAULT_IDLE_TIME_MILLISECONDS = 5000;
|
||||||
|
|
||||||
private ConnectionlessBootstrap udpServer;
|
private Bootstrap udpServer;
|
||||||
private ServerBootstrap tcpServer;
|
private ServerBootstrap tcpServer;
|
||||||
private ChannelGroup allChannels = new DefaultChannelGroup();
|
private ChannelGroup allChannels = new DefaultChannelGroup(
|
||||||
|
GlobalEventExecutor.INSTANCE);
|
||||||
private Channel udpChannel;
|
private Channel udpChannel;
|
||||||
private Channel tcpChannel;
|
private Channel tcpChannel;
|
||||||
|
|
||||||
|
EventLoopGroup bossGroup;
|
||||||
|
EventLoopGroup workerGroup;
|
||||||
|
EventLoopGroup udpGroup;
|
||||||
|
|
||||||
private final RpcProgramPortmap handler = new RpcProgramPortmap(allChannels);
|
private final RpcProgramPortmap handler = new RpcProgramPortmap(allChannels);
|
||||||
|
|
||||||
public static void main(String[] args) {
|
public static void main(String[] args) {
|
||||||
@ -73,18 +85,19 @@ public static void main(String[] args) {
|
|||||||
|
|
||||||
void shutdown() {
|
void shutdown() {
|
||||||
allChannels.close().awaitUninterruptibly();
|
allChannels.close().awaitUninterruptibly();
|
||||||
tcpServer.releaseExternalResources();
|
bossGroup.shutdownGracefully();
|
||||||
udpServer.releaseExternalResources();
|
workerGroup.shutdownGracefully();
|
||||||
|
udpGroup.shutdownGracefully();
|
||||||
}
|
}
|
||||||
|
|
||||||
@VisibleForTesting
|
@VisibleForTesting
|
||||||
SocketAddress getTcpServerLocalAddress() {
|
SocketAddress getTcpServerLocalAddress() {
|
||||||
return tcpChannel.getLocalAddress();
|
return tcpChannel.localAddress();
|
||||||
}
|
}
|
||||||
|
|
||||||
@VisibleForTesting
|
@VisibleForTesting
|
||||||
SocketAddress getUdpServerLoAddress() {
|
SocketAddress getUdpServerLoAddress() {
|
||||||
return udpChannel.getLocalAddress();
|
return udpChannel.localAddress();
|
||||||
}
|
}
|
||||||
|
|
||||||
@VisibleForTesting
|
@VisibleForTesting
|
||||||
@ -93,38 +106,55 @@ RpcProgramPortmap getHandler() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void start(final int idleTimeMilliSeconds, final SocketAddress tcpAddress,
|
void start(final int idleTimeMilliSeconds, final SocketAddress tcpAddress,
|
||||||
final SocketAddress udpAddress) {
|
final SocketAddress udpAddress) throws InterruptedException {
|
||||||
|
|
||||||
tcpServer = new ServerBootstrap(new NioServerSocketChannelFactory(
|
bossGroup = new NioEventLoopGroup();
|
||||||
Executors.newCachedThreadPool(), Executors.newCachedThreadPool()));
|
workerGroup = new NioEventLoopGroup(0, Executors.newCachedThreadPool());
|
||||||
tcpServer.setPipelineFactory(new ChannelPipelineFactory() {
|
|
||||||
private final HashedWheelTimer timer = new HashedWheelTimer();
|
tcpServer = new ServerBootstrap();
|
||||||
|
tcpServer.group(bossGroup, workerGroup)
|
||||||
|
.option(ChannelOption.SO_REUSEADDR, true)
|
||||||
|
.childOption(ChannelOption.SO_REUSEADDR, true)
|
||||||
|
.channel(NioServerSocketChannel.class)
|
||||||
|
.childHandler(new ChannelInitializer<SocketChannel>() {
|
||||||
private final IdleStateHandler idleStateHandler = new IdleStateHandler(
|
private final IdleStateHandler idleStateHandler = new IdleStateHandler(
|
||||||
timer, 0, 0, idleTimeMilliSeconds, TimeUnit.MILLISECONDS);
|
0, 0, idleTimeMilliSeconds, TimeUnit.MILLISECONDS);
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public ChannelPipeline getPipeline() throws Exception {
|
protected void initChannel(SocketChannel ch) throws Exception {
|
||||||
return Channels.pipeline(RpcUtil.constructRpcFrameDecoder(),
|
ChannelPipeline p = ch.pipeline();
|
||||||
|
|
||||||
|
p.addLast(RpcUtil.constructRpcFrameDecoder(),
|
||||||
RpcUtil.STAGE_RPC_MESSAGE_PARSER, idleStateHandler, handler,
|
RpcUtil.STAGE_RPC_MESSAGE_PARSER, idleStateHandler, handler,
|
||||||
RpcUtil.STAGE_RPC_TCP_RESPONSE);
|
RpcUtil.STAGE_RPC_TCP_RESPONSE);
|
||||||
|
}});
|
||||||
|
|
||||||
|
udpGroup = new NioEventLoopGroup(0, Executors.newCachedThreadPool());
|
||||||
|
|
||||||
|
udpServer = new Bootstrap();
|
||||||
|
udpServer.group(udpGroup)
|
||||||
|
.channel(NioDatagramChannel.class)
|
||||||
|
.handler(new ChannelInitializer<NioDatagramChannel>() {
|
||||||
|
@Override protected void initChannel(NioDatagramChannel ch)
|
||||||
|
throws Exception {
|
||||||
|
ChannelPipeline p = ch.pipeline();
|
||||||
|
p.addLast(
|
||||||
|
new LoggingHandler(LogLevel.DEBUG),
|
||||||
|
RpcUtil.STAGE_RPC_MESSAGE_PARSER, handler, RpcUtil.STAGE_RPC_UDP_RESPONSE);
|
||||||
}
|
}
|
||||||
});
|
})
|
||||||
tcpServer.setOption("reuseAddress", true);
|
.option(ChannelOption.SO_REUSEADDR, true);
|
||||||
tcpServer.setOption("child.reuseAddress", true);
|
|
||||||
|
|
||||||
udpServer = new ConnectionlessBootstrap(new NioDatagramChannelFactory(
|
ChannelFuture tcpChannelFuture = null;
|
||||||
Executors.newCachedThreadPool()));
|
tcpChannelFuture = tcpServer.bind(tcpAddress);
|
||||||
|
ChannelFuture udpChannelFuture = udpServer.bind(udpAddress);
|
||||||
|
tcpChannel = tcpChannelFuture.sync().channel();
|
||||||
|
udpChannel = udpChannelFuture.sync().channel();
|
||||||
|
|
||||||
udpServer.setPipeline(Channels.pipeline(RpcUtil.STAGE_RPC_MESSAGE_PARSER,
|
|
||||||
handler, RpcUtil.STAGE_RPC_UDP_RESPONSE));
|
|
||||||
udpServer.setOption("reuseAddress", true);
|
|
||||||
|
|
||||||
tcpChannel = tcpServer.bind(tcpAddress);
|
|
||||||
udpChannel = udpServer.bind(udpAddress);
|
|
||||||
allChannels.add(tcpChannel);
|
allChannels.add(tcpChannel);
|
||||||
allChannels.add(udpChannel);
|
allChannels.add(udpChannel);
|
||||||
|
|
||||||
LOG.info("Portmap server started at tcp://" + tcpChannel.getLocalAddress()
|
LOG.info("Portmap server started at tcp://" + tcpChannel.localAddress()
|
||||||
+ ", udp://" + udpChannel.getLocalAddress());
|
+ ", udp://" + udpChannel.localAddress());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -19,6 +19,14 @@
|
|||||||
|
|
||||||
import java.util.concurrent.ConcurrentHashMap;
|
import java.util.concurrent.ConcurrentHashMap;
|
||||||
|
|
||||||
|
import io.netty.buffer.ByteBuf;
|
||||||
|
import io.netty.buffer.Unpooled;
|
||||||
|
import io.netty.channel.ChannelHandler;
|
||||||
|
import io.netty.channel.ChannelHandlerContext;
|
||||||
|
import io.netty.channel.group.ChannelGroup;
|
||||||
|
import io.netty.handler.timeout.IdleState;
|
||||||
|
import io.netty.handler.timeout.IdleStateEvent;
|
||||||
|
import io.netty.handler.timeout.IdleStateHandler;
|
||||||
import org.apache.hadoop.oncrpc.RpcAcceptedReply;
|
import org.apache.hadoop.oncrpc.RpcAcceptedReply;
|
||||||
import org.apache.hadoop.oncrpc.RpcCall;
|
import org.apache.hadoop.oncrpc.RpcCall;
|
||||||
import org.apache.hadoop.oncrpc.RpcInfo;
|
import org.apache.hadoop.oncrpc.RpcInfo;
|
||||||
@ -27,20 +35,12 @@
|
|||||||
import org.apache.hadoop.oncrpc.RpcUtil;
|
import org.apache.hadoop.oncrpc.RpcUtil;
|
||||||
import org.apache.hadoop.oncrpc.XDR;
|
import org.apache.hadoop.oncrpc.XDR;
|
||||||
import org.apache.hadoop.oncrpc.security.VerifierNone;
|
import org.apache.hadoop.oncrpc.security.VerifierNone;
|
||||||
import org.jboss.netty.buffer.ChannelBuffer;
|
|
||||||
import org.jboss.netty.buffer.ChannelBuffers;
|
|
||||||
import org.jboss.netty.channel.ChannelHandlerContext;
|
|
||||||
import org.jboss.netty.channel.ChannelStateEvent;
|
|
||||||
import org.jboss.netty.channel.ExceptionEvent;
|
|
||||||
import org.jboss.netty.channel.MessageEvent;
|
|
||||||
import org.jboss.netty.channel.group.ChannelGroup;
|
|
||||||
import org.jboss.netty.handler.timeout.IdleState;
|
|
||||||
import org.jboss.netty.handler.timeout.IdleStateAwareChannelUpstreamHandler;
|
|
||||||
import org.jboss.netty.handler.timeout.IdleStateEvent;
|
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
final class RpcProgramPortmap extends IdleStateAwareChannelUpstreamHandler {
|
@ChannelHandler.Sharable
|
||||||
|
final class RpcProgramPortmap extends IdleStateHandler {
|
||||||
static final int PROGRAM = 100000;
|
static final int PROGRAM = 100000;
|
||||||
static final int VERSION = 2;
|
static final int VERSION = 2;
|
||||||
|
|
||||||
@ -60,6 +60,8 @@ final class RpcProgramPortmap extends IdleStateAwareChannelUpstreamHandler {
|
|||||||
private final ChannelGroup allChannels;
|
private final ChannelGroup allChannels;
|
||||||
|
|
||||||
RpcProgramPortmap(ChannelGroup allChannels) {
|
RpcProgramPortmap(ChannelGroup allChannels) {
|
||||||
|
super(1, 1, 1);
|
||||||
|
// FIXME: set default idle timeout 1 second.
|
||||||
this.allChannels = allChannels;
|
this.allChannels = allChannels;
|
||||||
PortmapMapping m = new PortmapMapping(PROGRAM, VERSION,
|
PortmapMapping m = new PortmapMapping(PROGRAM, VERSION,
|
||||||
PortmapMapping.TRANSPORT_TCP, RpcProgram.RPCB_PORT);
|
PortmapMapping.TRANSPORT_TCP, RpcProgram.RPCB_PORT);
|
||||||
@ -151,14 +153,14 @@ private XDR dump(int xid, XDR in, XDR out) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void messageReceived(ChannelHandlerContext ctx, MessageEvent e)
|
public void channelRead(ChannelHandlerContext ctx, Object msg)
|
||||||
throws Exception {
|
throws Exception {
|
||||||
|
|
||||||
RpcInfo info = (RpcInfo) e.getMessage();
|
RpcInfo info = (RpcInfo) msg;
|
||||||
RpcCall rpcCall = (RpcCall) info.header();
|
RpcCall rpcCall = (RpcCall) info.header();
|
||||||
final int portmapProc = rpcCall.getProcedure();
|
final int portmapProc = rpcCall.getProcedure();
|
||||||
int xid = rpcCall.getXid();
|
int xid = rpcCall.getXid();
|
||||||
XDR in = new XDR(info.data().toByteBuffer().asReadOnlyBuffer(),
|
XDR in = new XDR(info.data().nioBuffer().asReadOnlyBuffer(),
|
||||||
XDR.State.READING);
|
XDR.State.READING);
|
||||||
XDR out = new XDR();
|
XDR out = new XDR();
|
||||||
|
|
||||||
@ -181,29 +183,29 @@ public void messageReceived(ChannelHandlerContext ctx, MessageEvent e)
|
|||||||
reply.write(out);
|
reply.write(out);
|
||||||
}
|
}
|
||||||
|
|
||||||
ChannelBuffer buf = ChannelBuffers.wrappedBuffer(out.asReadOnlyWrap()
|
ByteBuf buf = Unpooled.wrappedBuffer(out.asReadOnlyWrap()
|
||||||
.buffer());
|
.buffer());
|
||||||
RpcResponse rsp = new RpcResponse(buf, info.remoteAddress());
|
RpcResponse rsp = new RpcResponse(buf, info.remoteAddress());
|
||||||
RpcUtil.sendRpcResponse(ctx, rsp);
|
RpcUtil.sendRpcResponse(ctx, rsp);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void channelOpen(ChannelHandlerContext ctx, ChannelStateEvent e)
|
public void channelActive(ChannelHandlerContext ctx)
|
||||||
throws Exception {
|
throws Exception {
|
||||||
allChannels.add(e.getChannel());
|
allChannels.add(ctx.channel());
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void channelIdle(ChannelHandlerContext ctx, IdleStateEvent e)
|
public void channelIdle(ChannelHandlerContext ctx, IdleStateEvent e)
|
||||||
throws Exception {
|
throws Exception {
|
||||||
if (e.getState() == IdleState.ALL_IDLE) {
|
if (e.state() == IdleState.ALL_IDLE) {
|
||||||
e.getChannel().close();
|
ctx.channel().close();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void exceptionCaught(ChannelHandlerContext ctx, ExceptionEvent e) {
|
public void exceptionCaught(ChannelHandlerContext ctx, Throwable t) {
|
||||||
LOG.warn("Encountered ", e.getCause());
|
LOG.warn("Encountered ", t);
|
||||||
e.getChannel().close();
|
ctx.channel().close();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -22,19 +22,19 @@
|
|||||||
import static org.junit.Assert.assertFalse;
|
import static org.junit.Assert.assertFalse;
|
||||||
import static org.junit.Assert.assertTrue;
|
import static org.junit.Assert.assertTrue;
|
||||||
|
|
||||||
import java.nio.ByteBuffer;
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
import java.util.Random;
|
import java.util.Random;
|
||||||
|
|
||||||
|
import io.netty.buffer.ByteBuf;
|
||||||
|
import io.netty.buffer.Unpooled;
|
||||||
|
import io.netty.channel.ChannelException;
|
||||||
|
import io.netty.channel.ChannelHandler;
|
||||||
|
import io.netty.channel.ChannelHandlerContext;
|
||||||
import org.apache.hadoop.oncrpc.RpcUtil.RpcFrameDecoder;
|
import org.apache.hadoop.oncrpc.RpcUtil.RpcFrameDecoder;
|
||||||
import org.apache.hadoop.oncrpc.security.CredentialsNone;
|
import org.apache.hadoop.oncrpc.security.CredentialsNone;
|
||||||
import org.apache.hadoop.oncrpc.security.VerifierNone;
|
import org.apache.hadoop.oncrpc.security.VerifierNone;
|
||||||
import org.apache.hadoop.test.GenericTestUtils;
|
import org.apache.hadoop.test.GenericTestUtils;
|
||||||
import org.jboss.netty.buffer.ByteBufferBackedChannelBuffer;
|
|
||||||
import org.jboss.netty.buffer.ChannelBuffer;
|
|
||||||
import org.jboss.netty.buffer.ChannelBuffers;
|
|
||||||
import org.jboss.netty.channel.Channel;
|
|
||||||
import org.jboss.netty.channel.ChannelException;
|
|
||||||
import org.jboss.netty.channel.ChannelHandlerContext;
|
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
import org.mockito.Mockito;
|
import org.mockito.Mockito;
|
||||||
import org.slf4j.event.Level;
|
import org.slf4j.event.Level;
|
||||||
@ -55,6 +55,7 @@ static void testRequest(XDR request, int serverPort) {
|
|||||||
tcpClient.run();
|
tcpClient.run();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ChannelHandler.Sharable
|
||||||
static class TestRpcProgram extends RpcProgram {
|
static class TestRpcProgram extends RpcProgram {
|
||||||
|
|
||||||
protected TestRpcProgram(String program, String host, int port,
|
protected TestRpcProgram(String program, String host, int port,
|
||||||
@ -83,7 +84,7 @@ protected void handleInternal(ChannelHandlerContext ctx, RpcInfo info) {
|
|||||||
new VerifierNone());
|
new VerifierNone());
|
||||||
XDR out = new XDR();
|
XDR out = new XDR();
|
||||||
reply.write(out);
|
reply.write(out);
|
||||||
ChannelBuffer b = ChannelBuffers.wrappedBuffer(out.asReadOnlyWrap().buffer());
|
ByteBuf b = Unpooled.wrappedBuffer(out.asReadOnlyWrap().buffer());
|
||||||
RpcResponse rsp = new RpcResponse(b, info.remoteAddress());
|
RpcResponse rsp = new RpcResponse(b, info.remoteAddress());
|
||||||
RpcUtil.sendRpcResponse(ctx, rsp);
|
RpcUtil.sendRpcResponse(ctx, rsp);
|
||||||
}
|
}
|
||||||
@ -99,13 +100,14 @@ public void testSingleFrame() {
|
|||||||
RpcFrameDecoder decoder = new RpcFrameDecoder();
|
RpcFrameDecoder decoder = new RpcFrameDecoder();
|
||||||
|
|
||||||
// Test "Length field is not received yet"
|
// Test "Length field is not received yet"
|
||||||
ByteBuffer buffer = ByteBuffer.allocate(1);
|
ByteBuf buf = Unpooled.directBuffer(1);
|
||||||
ChannelBuffer buf = new ByteBufferBackedChannelBuffer(buffer);
|
List<Object> outputBufs = new ArrayList<>();
|
||||||
ChannelBuffer channelBuffer = (ChannelBuffer) decoder.decode(
|
decoder.decode(
|
||||||
Mockito.mock(ChannelHandlerContext.class), Mockito.mock(Channel.class),
|
Mockito.mock(ChannelHandlerContext.class), buf,
|
||||||
buf);
|
outputBufs);
|
||||||
assertTrue(channelBuffer == null);
|
assertTrue(outputBufs.isEmpty());
|
||||||
|
|
||||||
|
decoder = new RpcFrameDecoder();
|
||||||
// Test all bytes are not received yet
|
// Test all bytes are not received yet
|
||||||
byte[] fragment = new byte[4 + 9];
|
byte[] fragment = new byte[4 + 9];
|
||||||
fragment[0] = (byte) (1 << 7); // final fragment
|
fragment[0] = (byte) (1 << 7); // final fragment
|
||||||
@ -114,15 +116,16 @@ public void testSingleFrame() {
|
|||||||
fragment[3] = (byte) 10; // fragment size = 10 bytes
|
fragment[3] = (byte) 10; // fragment size = 10 bytes
|
||||||
assertTrue(XDR.isLastFragment(fragment));
|
assertTrue(XDR.isLastFragment(fragment));
|
||||||
assertTrue(XDR.fragmentSize(fragment)==10);
|
assertTrue(XDR.fragmentSize(fragment)==10);
|
||||||
|
buf.release();
|
||||||
|
|
||||||
buffer = ByteBuffer.allocate(4 + 9);
|
buf = Unpooled.directBuffer(4 + 9);
|
||||||
buffer.put(fragment);
|
buf.writeBytes(fragment);
|
||||||
buffer.flip();
|
outputBufs = new ArrayList<>();
|
||||||
buf = new ByteBufferBackedChannelBuffer(buffer);
|
decoder.decode(
|
||||||
channelBuffer = (ChannelBuffer) decoder.decode(
|
Mockito.mock(ChannelHandlerContext.class), buf,
|
||||||
Mockito.mock(ChannelHandlerContext.class), Mockito.mock(Channel.class),
|
outputBufs);
|
||||||
buf);
|
assertTrue(decoder.isLast());
|
||||||
assertTrue(channelBuffer == null);
|
buf.release();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
@ -138,15 +141,14 @@ public void testMultipleFrames() {
|
|||||||
assertFalse(XDR.isLastFragment(fragment1));
|
assertFalse(XDR.isLastFragment(fragment1));
|
||||||
assertTrue(XDR.fragmentSize(fragment1)==10);
|
assertTrue(XDR.fragmentSize(fragment1)==10);
|
||||||
|
|
||||||
|
List<Object> outputBufs = new ArrayList<>();
|
||||||
|
|
||||||
// decoder should wait for the final fragment
|
// decoder should wait for the final fragment
|
||||||
ByteBuffer buffer = ByteBuffer.allocate(4 + 10);
|
ByteBuf buf = Unpooled.directBuffer(4 + 10, 4 + 10);
|
||||||
buffer.put(fragment1);
|
buf.writeBytes(fragment1);
|
||||||
buffer.flip();
|
decoder.decode(
|
||||||
ChannelBuffer buf = new ByteBufferBackedChannelBuffer(buffer);
|
Mockito.mock(ChannelHandlerContext.class), buf,
|
||||||
ChannelBuffer channelBuffer = (ChannelBuffer) decoder.decode(
|
outputBufs);
|
||||||
Mockito.mock(ChannelHandlerContext.class), Mockito.mock(Channel.class),
|
|
||||||
buf);
|
|
||||||
assertTrue(channelBuffer == null);
|
|
||||||
|
|
||||||
byte[] fragment2 = new byte[4 + 10];
|
byte[] fragment2 = new byte[4 + 10];
|
||||||
fragment2[0] = (byte) (1 << 7); // final fragment
|
fragment2[0] = (byte) (1 << 7); // final fragment
|
||||||
@ -155,21 +157,22 @@ public void testMultipleFrames() {
|
|||||||
fragment2[3] = (byte) 10; // fragment size = 10 bytes
|
fragment2[3] = (byte) 10; // fragment size = 10 bytes
|
||||||
assertTrue(XDR.isLastFragment(fragment2));
|
assertTrue(XDR.isLastFragment(fragment2));
|
||||||
assertTrue(XDR.fragmentSize(fragment2)==10);
|
assertTrue(XDR.fragmentSize(fragment2)==10);
|
||||||
|
buf.release();
|
||||||
|
|
||||||
buffer = ByteBuffer.allocate(4 + 10);
|
buf = Unpooled.directBuffer(4 + 10, 4 + 10);
|
||||||
buffer.put(fragment2);
|
buf.writeBytes(fragment2);
|
||||||
buffer.flip();
|
decoder.decode(
|
||||||
buf = new ByteBufferBackedChannelBuffer(buffer);
|
Mockito.mock(ChannelHandlerContext.class), buf,
|
||||||
channelBuffer = (ChannelBuffer) decoder.decode(
|
outputBufs);
|
||||||
Mockito.mock(ChannelHandlerContext.class), Mockito.mock(Channel.class),
|
// Expect two completed frames each 10 bytes
|
||||||
buf);
|
decoder.isLast();
|
||||||
assertTrue(channelBuffer != null);
|
assertEquals(2, outputBufs.size());
|
||||||
// Complete frame should have to total size 10+10=20
|
outputBufs.forEach(b -> assertEquals(((ByteBuf)b).readableBytes(), 10));
|
||||||
assertEquals(20, channelBuffer.readableBytes());
|
buf.release();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testFrames() {
|
public void testFrames() throws InterruptedException {
|
||||||
int serverPort = startRpcServer(true);
|
int serverPort = startRpcServer(true);
|
||||||
|
|
||||||
XDR xdrOut = createGetportMount();
|
XDR xdrOut = createGetportMount();
|
||||||
@ -187,7 +190,7 @@ public void testFrames() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testUnprivilegedPort() {
|
public void testUnprivilegedPort() throws InterruptedException {
|
||||||
// Don't allow connections from unprivileged ports. Given that this test is
|
// Don't allow connections from unprivileged ports. Given that this test is
|
||||||
// presumably not being run by root, this will be the case.
|
// presumably not being run by root, this will be the case.
|
||||||
int serverPort = startRpcServer(false);
|
int serverPort = startRpcServer(false);
|
||||||
@ -218,23 +221,28 @@ public void testUnprivilegedPort() {
|
|||||||
assertEquals(requestSize, resultSize);
|
assertEquals(requestSize, resultSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
private static int startRpcServer(boolean allowInsecurePorts) {
|
private static int startRpcServer(boolean allowInsecurePorts)
|
||||||
|
throws InterruptedException {
|
||||||
Random rand = new Random();
|
Random rand = new Random();
|
||||||
int serverPort = 30000 + rand.nextInt(10000);
|
int serverPort = 30000 + rand.nextInt(10000);
|
||||||
int retries = 10; // A few retries in case initial choice is in use.
|
int retries = 10; // A few retries in case initial choice is in use.
|
||||||
|
|
||||||
while (true) {
|
while (true) {
|
||||||
|
SimpleTcpServer tcpServer = null;
|
||||||
try {
|
try {
|
||||||
RpcProgram program = new TestFrameDecoder.TestRpcProgram("TestRpcProgram",
|
RpcProgram program = new TestFrameDecoder.TestRpcProgram("TestRpcProgram",
|
||||||
"localhost", serverPort, 100000, 1, 2, allowInsecurePorts);
|
"localhost", serverPort, 100000, 1, 2, allowInsecurePorts);
|
||||||
SimpleTcpServer tcpServer = new SimpleTcpServer(serverPort, program, 1);
|
tcpServer = new SimpleTcpServer(serverPort, program, 1);
|
||||||
tcpServer.run();
|
tcpServer.run();
|
||||||
break; // Successfully bound a port, break out.
|
break; // Successfully bound a port, break out.
|
||||||
} catch (ChannelException ce) {
|
} catch (InterruptedException | ChannelException e) {
|
||||||
|
if (tcpServer != null) {
|
||||||
|
tcpServer.shutdown();
|
||||||
|
}
|
||||||
if (retries-- > 0) {
|
if (retries-- > 0) {
|
||||||
serverPort += rand.nextInt(20); // Port in use? Try another.
|
serverPort += rand.nextInt(20); // Port in use? Try another.
|
||||||
} else {
|
} else {
|
||||||
throw ce; // Out of retries.
|
throw e; // Out of retries.
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -43,7 +43,7 @@ public class TestPortmap {
|
|||||||
private int xid;
|
private int xid;
|
||||||
|
|
||||||
@BeforeClass
|
@BeforeClass
|
||||||
public static void setup() {
|
public static void setup() throws InterruptedException {
|
||||||
pm.start(SHORT_TIMEOUT_MILLISECONDS, new InetSocketAddress("localhost", 0),
|
pm.start(SHORT_TIMEOUT_MILLISECONDS, new InetSocketAddress("localhost", 0),
|
||||||
new InetSocketAddress("localhost", 0));
|
new InetSocketAddress("localhost", 0));
|
||||||
}
|
}
|
||||||
|
@ -47,7 +47,7 @@ https://maven.apache.org/xsd/maven-4.0.0.xsd">
|
|||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>io.netty</groupId>
|
<groupId>io.netty</groupId>
|
||||||
<artifactId>netty</artifactId>
|
<artifactId>netty-all</artifactId>
|
||||||
<scope>compile</scope>
|
<scope>compile</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
|
@ -26,6 +26,10 @@
|
|||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
|
|
||||||
|
import io.netty.channel.ChannelHandler;
|
||||||
|
import io.netty.buffer.ByteBuf;
|
||||||
|
import io.netty.buffer.Unpooled;
|
||||||
|
import io.netty.channel.ChannelHandlerContext;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
import org.apache.hadoop.fs.FileSystem;
|
import org.apache.hadoop.fs.FileSystem;
|
||||||
@ -51,15 +55,13 @@
|
|||||||
import org.apache.hadoop.oncrpc.security.VerifierNone;
|
import org.apache.hadoop.oncrpc.security.VerifierNone;
|
||||||
import org.apache.hadoop.security.SecurityUtil;
|
import org.apache.hadoop.security.SecurityUtil;
|
||||||
import org.apache.hadoop.security.UserGroupInformation;
|
import org.apache.hadoop.security.UserGroupInformation;
|
||||||
import org.jboss.netty.buffer.ChannelBuffer;
|
|
||||||
import org.jboss.netty.buffer.ChannelBuffers;
|
|
||||||
import org.jboss.netty.channel.ChannelHandlerContext;
|
|
||||||
|
|
||||||
import org.apache.hadoop.thirdparty.com.google.common.annotations.VisibleForTesting;
|
import org.apache.hadoop.thirdparty.com.google.common.annotations.VisibleForTesting;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* RPC program corresponding to mountd daemon. See {@link Mountd}.
|
* RPC program corresponding to mountd daemon. See {@link Mountd}.
|
||||||
*/
|
*/
|
||||||
|
@ChannelHandler.Sharable
|
||||||
public class RpcProgramMountd extends RpcProgram implements MountInterface {
|
public class RpcProgramMountd extends RpcProgram implements MountInterface {
|
||||||
private static final Logger LOG =
|
private static final Logger LOG =
|
||||||
LoggerFactory.getLogger(RpcProgramMountd.class);
|
LoggerFactory.getLogger(RpcProgramMountd.class);
|
||||||
@ -262,8 +264,8 @@ RpcAcceptedReply.AcceptState.PROC_UNAVAIL, new VerifierNone())
|
|||||||
RpcAcceptedReply.AcceptState.PROC_UNAVAIL, new VerifierNone()).write(
|
RpcAcceptedReply.AcceptState.PROC_UNAVAIL, new VerifierNone()).write(
|
||||||
out);
|
out);
|
||||||
}
|
}
|
||||||
ChannelBuffer buf =
|
ByteBuf buf =
|
||||||
ChannelBuffers.wrappedBuffer(out.asReadOnlyWrap().buffer());
|
Unpooled.wrappedBuffer(out.asReadOnlyWrap().buffer());
|
||||||
RpcResponse rsp = new RpcResponse(buf, info.remoteAddress());
|
RpcResponse rsp = new RpcResponse(buf, info.remoteAddress());
|
||||||
RpcUtil.sendRpcResponse(ctx, rsp);
|
RpcUtil.sendRpcResponse(ctx, rsp);
|
||||||
}
|
}
|
||||||
|
@ -22,6 +22,8 @@
|
|||||||
import java.net.URI;
|
import java.net.URI;
|
||||||
import java.nio.file.FileSystemException;
|
import java.nio.file.FileSystemException;
|
||||||
|
|
||||||
|
import io.netty.buffer.ByteBuf;
|
||||||
|
import io.netty.channel.Channel;
|
||||||
import org.apache.hadoop.conf.Configuration;
|
import org.apache.hadoop.conf.Configuration;
|
||||||
import org.apache.hadoop.fs.FileSystem;
|
import org.apache.hadoop.fs.FileSystem;
|
||||||
import org.apache.hadoop.fs.FsConstants;
|
import org.apache.hadoop.fs.FsConstants;
|
||||||
@ -39,8 +41,6 @@
|
|||||||
import org.apache.hadoop.nfs.nfs3.response.WccData;
|
import org.apache.hadoop.nfs.nfs3.response.WccData;
|
||||||
import org.apache.hadoop.oncrpc.XDR;
|
import org.apache.hadoop.oncrpc.XDR;
|
||||||
import org.apache.hadoop.security.IdMappingServiceProvider;
|
import org.apache.hadoop.security.IdMappingServiceProvider;
|
||||||
import org.jboss.netty.buffer.ChannelBuffer;
|
|
||||||
import org.jboss.netty.channel.Channel;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Utility/helper methods related to NFS
|
* Utility/helper methods related to NFS
|
||||||
@ -147,16 +147,16 @@ public static void writeChannel(Channel channel, XDR out, int xid) {
|
|||||||
if (RpcProgramNfs3.LOG.isDebugEnabled()) {
|
if (RpcProgramNfs3.LOG.isDebugEnabled()) {
|
||||||
RpcProgramNfs3.LOG.debug(WRITE_RPC_END + xid);
|
RpcProgramNfs3.LOG.debug(WRITE_RPC_END + xid);
|
||||||
}
|
}
|
||||||
ChannelBuffer outBuf = XDR.writeMessageTcp(out, true);
|
ByteBuf outBuf = XDR.writeMessageTcp(out, true);
|
||||||
channel.write(outBuf);
|
channel.writeAndFlush(outBuf);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void writeChannelCommit(Channel channel, XDR out, int xid) {
|
public static void writeChannelCommit(Channel channel, XDR out, int xid) {
|
||||||
if (RpcProgramNfs3.LOG.isDebugEnabled()) {
|
if (RpcProgramNfs3.LOG.isDebugEnabled()) {
|
||||||
RpcProgramNfs3.LOG.debug("Commit done:" + xid);
|
RpcProgramNfs3.LOG.debug("Commit done:" + xid);
|
||||||
}
|
}
|
||||||
ChannelBuffer outBuf = XDR.writeMessageTcp(out, true);
|
ByteBuf outBuf = XDR.writeMessageTcp(out, true);
|
||||||
channel.write(outBuf);
|
channel.writeAndFlush(outBuf);
|
||||||
}
|
}
|
||||||
|
|
||||||
private static boolean isSet(int access, int bits) {
|
private static boolean isSet(int access, int bits) {
|
||||||
|
@ -31,6 +31,7 @@
|
|||||||
import java.util.concurrent.ConcurrentSkipListMap;
|
import java.util.concurrent.ConcurrentSkipListMap;
|
||||||
import java.util.concurrent.atomic.AtomicLong;
|
import java.util.concurrent.atomic.AtomicLong;
|
||||||
|
|
||||||
|
import io.netty.channel.Channel;
|
||||||
import org.apache.hadoop.fs.FSDataInputStream;
|
import org.apache.hadoop.fs.FSDataInputStream;
|
||||||
import org.apache.hadoop.hdfs.DFSClient;
|
import org.apache.hadoop.hdfs.DFSClient;
|
||||||
import org.apache.hadoop.hdfs.client.HdfsDataOutputStream;
|
import org.apache.hadoop.hdfs.client.HdfsDataOutputStream;
|
||||||
@ -55,7 +56,6 @@
|
|||||||
import org.apache.hadoop.security.IdMappingServiceProvider;
|
import org.apache.hadoop.security.IdMappingServiceProvider;
|
||||||
import org.apache.hadoop.util.Daemon;
|
import org.apache.hadoop.util.Daemon;
|
||||||
import org.apache.hadoop.util.Time;
|
import org.apache.hadoop.util.Time;
|
||||||
import org.jboss.netty.channel.Channel;
|
|
||||||
|
|
||||||
import org.apache.hadoop.thirdparty.com.google.common.annotations.VisibleForTesting;
|
import org.apache.hadoop.thirdparty.com.google.common.annotations.VisibleForTesting;
|
||||||
import org.apache.hadoop.thirdparty.com.google.common.base.Preconditions;
|
import org.apache.hadoop.thirdparty.com.google.common.base.Preconditions;
|
||||||
|
@ -28,6 +28,11 @@
|
|||||||
import java.nio.charset.Charset;
|
import java.nio.charset.Charset;
|
||||||
import java.util.EnumSet;
|
import java.util.EnumSet;
|
||||||
|
|
||||||
|
import io.netty.buffer.ByteBuf;
|
||||||
|
import io.netty.buffer.Unpooled;
|
||||||
|
import io.netty.channel.Channel;
|
||||||
|
import io.netty.channel.ChannelHandler;
|
||||||
|
import io.netty.channel.ChannelHandlerContext;
|
||||||
import org.apache.hadoop.fs.CommonConfigurationKeysPublic;
|
import org.apache.hadoop.fs.CommonConfigurationKeysPublic;
|
||||||
import org.apache.hadoop.fs.CreateFlag;
|
import org.apache.hadoop.fs.CreateFlag;
|
||||||
import org.apache.hadoop.fs.DirectoryListingStartAfterNotFoundException;
|
import org.apache.hadoop.fs.DirectoryListingStartAfterNotFoundException;
|
||||||
@ -129,10 +134,6 @@
|
|||||||
import org.apache.hadoop.security.UserGroupInformation;
|
import org.apache.hadoop.security.UserGroupInformation;
|
||||||
import org.apache.hadoop.security.authorize.AuthorizationException;
|
import org.apache.hadoop.security.authorize.AuthorizationException;
|
||||||
import org.apache.hadoop.util.JvmPauseMonitor;
|
import org.apache.hadoop.util.JvmPauseMonitor;
|
||||||
import org.jboss.netty.buffer.ChannelBuffer;
|
|
||||||
import org.jboss.netty.buffer.ChannelBuffers;
|
|
||||||
import org.jboss.netty.channel.Channel;
|
|
||||||
import org.jboss.netty.channel.ChannelHandlerContext;
|
|
||||||
|
|
||||||
import org.apache.hadoop.thirdparty.com.google.common.annotations.VisibleForTesting;
|
import org.apache.hadoop.thirdparty.com.google.common.annotations.VisibleForTesting;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
@ -141,6 +142,7 @@
|
|||||||
/**
|
/**
|
||||||
* RPC program corresponding to nfs daemon. See {@link Nfs3}.
|
* RPC program corresponding to nfs daemon. See {@link Nfs3}.
|
||||||
*/
|
*/
|
||||||
|
@ChannelHandler.Sharable
|
||||||
public class RpcProgramNfs3 extends RpcProgram implements Nfs3Interface {
|
public class RpcProgramNfs3 extends RpcProgram implements Nfs3Interface {
|
||||||
public static final int DEFAULT_UMASK = 0022;
|
public static final int DEFAULT_UMASK = 0022;
|
||||||
public static final FsPermission umask = new FsPermission(
|
public static final FsPermission umask = new FsPermission(
|
||||||
@ -2180,7 +2182,7 @@ public void handleInternal(ChannelHandlerContext ctx, RpcInfo info) {
|
|||||||
RpcDeniedReply.RejectState.AUTH_ERROR, new VerifierNone());
|
RpcDeniedReply.RejectState.AUTH_ERROR, new VerifierNone());
|
||||||
rdr.write(reply);
|
rdr.write(reply);
|
||||||
|
|
||||||
ChannelBuffer buf = ChannelBuffers.wrappedBuffer(reply.asReadOnlyWrap()
|
ByteBuf buf = Unpooled.wrappedBuffer(reply.asReadOnlyWrap()
|
||||||
.buffer());
|
.buffer());
|
||||||
RpcResponse rsp = new RpcResponse(buf, info.remoteAddress());
|
RpcResponse rsp = new RpcResponse(buf, info.remoteAddress());
|
||||||
RpcUtil.sendRpcResponse(ctx, rsp);
|
RpcUtil.sendRpcResponse(ctx, rsp);
|
||||||
@ -2291,7 +2293,7 @@ RpcAcceptedReply.AcceptState.PROC_UNAVAIL, new VerifierNone()).write(
|
|||||||
}
|
}
|
||||||
// TODO: currently we just return VerifierNone
|
// TODO: currently we just return VerifierNone
|
||||||
out = response.serialize(out, xid, new VerifierNone());
|
out = response.serialize(out, xid, new VerifierNone());
|
||||||
ChannelBuffer buf = ChannelBuffers.wrappedBuffer(out.asReadOnlyWrap()
|
ByteBuf buf = Unpooled.wrappedBuffer(out.asReadOnlyWrap()
|
||||||
.buffer());
|
.buffer());
|
||||||
RpcResponse rsp = new RpcResponse(buf, info.remoteAddress());
|
RpcResponse rsp = new RpcResponse(buf, info.remoteAddress());
|
||||||
|
|
||||||
|
@ -22,12 +22,12 @@
|
|||||||
import java.io.RandomAccessFile;
|
import java.io.RandomAccessFile;
|
||||||
import java.nio.ByteBuffer;
|
import java.nio.ByteBuffer;
|
||||||
|
|
||||||
|
import io.netty.channel.Channel;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
import org.apache.hadoop.hdfs.client.HdfsDataOutputStream;
|
import org.apache.hadoop.hdfs.client.HdfsDataOutputStream;
|
||||||
import org.apache.hadoop.nfs.nfs3.FileHandle;
|
import org.apache.hadoop.nfs.nfs3.FileHandle;
|
||||||
import org.apache.hadoop.nfs.nfs3.Nfs3Constant.WriteStableHow;
|
import org.apache.hadoop.nfs.nfs3.Nfs3Constant.WriteStableHow;
|
||||||
import org.jboss.netty.channel.Channel;
|
|
||||||
|
|
||||||
import org.apache.hadoop.thirdparty.com.google.common.annotations.VisibleForTesting;
|
import org.apache.hadoop.thirdparty.com.google.common.annotations.VisibleForTesting;
|
||||||
import org.apache.hadoop.thirdparty.com.google.common.base.Preconditions;
|
import org.apache.hadoop.thirdparty.com.google.common.base.Preconditions;
|
||||||
|
@ -20,6 +20,7 @@
|
|||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.util.EnumSet;
|
import java.util.EnumSet;
|
||||||
|
|
||||||
|
import io.netty.channel.Channel;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
import org.apache.hadoop.fs.CommonConfigurationKeysPublic;
|
import org.apache.hadoop.fs.CommonConfigurationKeysPublic;
|
||||||
@ -43,7 +44,6 @@
|
|||||||
import org.apache.hadoop.oncrpc.XDR;
|
import org.apache.hadoop.oncrpc.XDR;
|
||||||
import org.apache.hadoop.oncrpc.security.VerifierNone;
|
import org.apache.hadoop.oncrpc.security.VerifierNone;
|
||||||
import org.apache.hadoop.security.IdMappingServiceProvider;
|
import org.apache.hadoop.security.IdMappingServiceProvider;
|
||||||
import org.jboss.netty.channel.Channel;
|
|
||||||
|
|
||||||
import org.apache.hadoop.thirdparty.com.google.common.annotations.VisibleForTesting;
|
import org.apache.hadoop.thirdparty.com.google.common.annotations.VisibleForTesting;
|
||||||
|
|
||||||
|
@ -21,6 +21,12 @@
|
|||||||
import java.nio.ByteBuffer;
|
import java.nio.ByteBuffer;
|
||||||
import java.util.Arrays;
|
import java.util.Arrays;
|
||||||
|
|
||||||
|
import io.netty.buffer.ByteBuf;
|
||||||
|
import io.netty.channel.Channel;
|
||||||
|
import io.netty.channel.ChannelHandlerContext;
|
||||||
|
import io.netty.channel.ChannelInitializer;
|
||||||
|
import io.netty.channel.ChannelPipeline;
|
||||||
|
import io.netty.channel.socket.SocketChannel;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
import org.apache.hadoop.hdfs.nfs.conf.NfsConfigKeys;
|
import org.apache.hadoop.hdfs.nfs.conf.NfsConfigKeys;
|
||||||
@ -42,13 +48,6 @@
|
|||||||
import org.apache.hadoop.oncrpc.XDR;
|
import org.apache.hadoop.oncrpc.XDR;
|
||||||
import org.apache.hadoop.oncrpc.security.CredentialsNone;
|
import org.apache.hadoop.oncrpc.security.CredentialsNone;
|
||||||
import org.apache.hadoop.oncrpc.security.VerifierNone;
|
import org.apache.hadoop.oncrpc.security.VerifierNone;
|
||||||
import org.jboss.netty.buffer.ChannelBuffer;
|
|
||||||
import org.jboss.netty.channel.Channel;
|
|
||||||
import org.jboss.netty.channel.ChannelHandlerContext;
|
|
||||||
import org.jboss.netty.channel.ChannelPipeline;
|
|
||||||
import org.jboss.netty.channel.ChannelPipelineFactory;
|
|
||||||
import org.jboss.netty.channel.Channels;
|
|
||||||
import org.jboss.netty.channel.MessageEvent;
|
|
||||||
|
|
||||||
public class TestOutOfOrderWrite {
|
public class TestOutOfOrderWrite {
|
||||||
public final static Logger LOG =
|
public final static Logger LOG =
|
||||||
@ -100,9 +99,9 @@ public WriteHandler(XDR request) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void messageReceived(ChannelHandlerContext ctx, MessageEvent e) {
|
public void channelRead(ChannelHandlerContext ctx, Object msg) {
|
||||||
// Get handle from create response
|
// Get handle from create response
|
||||||
ChannelBuffer buf = (ChannelBuffer) e.getMessage();
|
ByteBuf buf = (ByteBuf) msg;
|
||||||
XDR rsp = new XDR(buf.array());
|
XDR rsp = new XDR(buf.array());
|
||||||
if (rsp.getBytes().length == 0) {
|
if (rsp.getBytes().length == 0) {
|
||||||
LOG.info("rsp length is zero, why?");
|
LOG.info("rsp length is zero, why?");
|
||||||
@ -125,7 +124,7 @@ public void messageReceived(ChannelHandlerContext ctx, MessageEvent e) {
|
|||||||
rsp.readBoolean(); // value follow
|
rsp.readBoolean(); // value follow
|
||||||
handle = new FileHandle();
|
handle = new FileHandle();
|
||||||
handle.deserialize(rsp);
|
handle.deserialize(rsp);
|
||||||
channel = e.getChannel();
|
channel = ctx.channel();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -136,16 +135,17 @@ public WriteClient(String host, int port, XDR request, Boolean oneShot) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected ChannelPipelineFactory setPipelineFactory() {
|
protected ChannelInitializer<SocketChannel> setChannelHandler() {
|
||||||
this.pipelineFactory = new ChannelPipelineFactory() {
|
return new ChannelInitializer<SocketChannel>() {
|
||||||
@Override
|
@Override
|
||||||
public ChannelPipeline getPipeline() {
|
protected void initChannel(SocketChannel ch) throws Exception {
|
||||||
return Channels.pipeline(
|
ChannelPipeline p = ch.pipeline();
|
||||||
|
p.addLast(
|
||||||
RpcUtil.constructRpcFrameDecoder(),
|
RpcUtil.constructRpcFrameDecoder(),
|
||||||
new WriteHandler(request));
|
new WriteHandler(request)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
return this.pipelineFactory;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -28,6 +28,7 @@
|
|||||||
import java.nio.ByteBuffer;
|
import java.nio.ByteBuffer;
|
||||||
import java.util.EnumSet;
|
import java.util.EnumSet;
|
||||||
|
|
||||||
|
import io.netty.channel.Channel;
|
||||||
import org.apache.hadoop.crypto.key.JavaKeyStoreProvider;
|
import org.apache.hadoop.crypto.key.JavaKeyStoreProvider;
|
||||||
import org.apache.hadoop.fs.CommonConfigurationKeys;
|
import org.apache.hadoop.fs.CommonConfigurationKeys;
|
||||||
import org.apache.hadoop.fs.CommonConfigurationKeysPublic;
|
import org.apache.hadoop.fs.CommonConfigurationKeysPublic;
|
||||||
@ -92,7 +93,6 @@
|
|||||||
import org.apache.hadoop.security.IdMappingConstant;
|
import org.apache.hadoop.security.IdMappingConstant;
|
||||||
import org.apache.hadoop.security.authorize.DefaultImpersonationProvider;
|
import org.apache.hadoop.security.authorize.DefaultImpersonationProvider;
|
||||||
import org.apache.hadoop.security.authorize.ProxyUsers;
|
import org.apache.hadoop.security.authorize.ProxyUsers;
|
||||||
import org.jboss.netty.channel.Channel;
|
|
||||||
import org.junit.AfterClass;
|
import org.junit.AfterClass;
|
||||||
import org.junit.Assert;
|
import org.junit.Assert;
|
||||||
import org.junit.Before;
|
import org.junit.Before;
|
||||||
|
@ -27,6 +27,7 @@
|
|||||||
import java.util.Arrays;
|
import java.util.Arrays;
|
||||||
import java.util.concurrent.ConcurrentNavigableMap;
|
import java.util.concurrent.ConcurrentNavigableMap;
|
||||||
|
|
||||||
|
import io.netty.channel.Channel;
|
||||||
import org.apache.hadoop.hdfs.DFSClient;
|
import org.apache.hadoop.hdfs.DFSClient;
|
||||||
import org.apache.hadoop.hdfs.DFSUtilClient;
|
import org.apache.hadoop.hdfs.DFSUtilClient;
|
||||||
import org.apache.hadoop.hdfs.MiniDFSCluster;
|
import org.apache.hadoop.hdfs.MiniDFSCluster;
|
||||||
@ -52,7 +53,6 @@
|
|||||||
import org.apache.hadoop.security.ShellBasedIdMapping;
|
import org.apache.hadoop.security.ShellBasedIdMapping;
|
||||||
import org.apache.hadoop.security.authorize.DefaultImpersonationProvider;
|
import org.apache.hadoop.security.authorize.DefaultImpersonationProvider;
|
||||||
import org.apache.hadoop.security.authorize.ProxyUsers;
|
import org.apache.hadoop.security.authorize.ProxyUsers;
|
||||||
import org.jboss.netty.channel.Channel;
|
|
||||||
import org.junit.Assert;
|
import org.junit.Assert;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
import org.mockito.Mockito;
|
import org.mockito.Mockito;
|
||||||
|
Loading…
Reference in New Issue
Block a user