976db9269d
While implementing netty-handler-proxy, I realized various issues in our current socksx package. Here's the list of the modifications and their background: - Split message types into interfaces and default implementations - so that a user can implement an alternative message implementations - Use classes instead of enums when a user might want to define a new constant - so that a user can extend SOCKS5 protocol, such as: - defining a new error code - defining a new address type - Rename the message classes - to avoid abbreviated class names. e.g: - Cmd -> Command - Init -> Initial - so that the class names align better with the protocol specifications. e.g: - AuthRequest -> PasswordAuthRequest - AuthScheme -> AuthMethod - Rename the property names of the messages - so that the property names align better when the field names in the protocol specifications - Improve the decoder implementations - Give a user more control over when a decoder has to be removed - Use DecoderResult and DecoderResultProvider to handle decode failure gracefully. i.e. no more Unknown* message classes - Add SocksPortUnifinicationServerHandler since it's useful to the users who write a SOCKS server - Cleaned up and moved from the socksproxy example
146 lines
5.3 KiB
Java
146 lines
5.3 KiB
Java
/*
|
|
* Copyright 2014 The Netty Project
|
|
*
|
|
* The Netty Project licenses this file to you under the Apache License,
|
|
* version 2.0 (the "License"); you may not use this file except in compliance
|
|
* with the License. You may obtain a copy of the License at:
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
* License for the specific language governing permissions and limitations
|
|
* under the License.
|
|
*/
|
|
|
|
package io.netty.handler.proxy;
|
|
|
|
import io.netty.buffer.Unpooled;
|
|
import io.netty.channel.ChannelHandlerContext;
|
|
import io.netty.channel.ChannelPipeline;
|
|
import io.netty.channel.socket.SocketChannel;
|
|
import io.netty.handler.codec.LineBasedFrameDecoder;
|
|
import io.netty.handler.codec.socksx.v4.DefaultSocks4CommandResponse;
|
|
import io.netty.handler.codec.socksx.v4.Socks4CommandRequest;
|
|
import io.netty.handler.codec.socksx.v4.Socks4CommandResponse;
|
|
import io.netty.handler.codec.socksx.v4.Socks4CommandStatus;
|
|
import io.netty.handler.codec.socksx.v4.Socks4CommandType;
|
|
import io.netty.handler.codec.socksx.v4.Socks4ServerDecoder;
|
|
import io.netty.handler.codec.socksx.v4.Socks4ServerEncoder;
|
|
import io.netty.util.CharsetUtil;
|
|
|
|
import java.net.InetSocketAddress;
|
|
import java.net.SocketAddress;
|
|
|
|
import static org.hamcrest.Matchers.*;
|
|
import static org.junit.Assert.*;
|
|
|
|
final class Socks4ProxyServer extends ProxyServer {
|
|
|
|
Socks4ProxyServer(boolean useSsl, TestMode testMode, InetSocketAddress destination) {
|
|
super(useSsl, testMode, destination);
|
|
}
|
|
|
|
Socks4ProxyServer(boolean useSsl, TestMode testMode, InetSocketAddress destination, String username) {
|
|
super(useSsl, testMode, destination, username, null);
|
|
}
|
|
|
|
@Override
|
|
protected void configure(SocketChannel ch) throws Exception {
|
|
ChannelPipeline p = ch.pipeline();
|
|
switch (testMode) {
|
|
case INTERMEDIARY:
|
|
p.addLast(new Socks4ServerDecoder());
|
|
p.addLast(Socks4ServerEncoder.INSTANCE);
|
|
p.addLast(new Socks4IntermediaryHandler());
|
|
break;
|
|
case TERMINAL:
|
|
p.addLast(new Socks4ServerDecoder());
|
|
p.addLast(Socks4ServerEncoder.INSTANCE);
|
|
p.addLast(new Socks4TerminalHandler());
|
|
break;
|
|
case UNRESPONSIVE:
|
|
p.addLast(UnresponsiveHandler.INSTANCE);
|
|
break;
|
|
}
|
|
}
|
|
|
|
private boolean authenticate(ChannelHandlerContext ctx, Socks4CommandRequest req) {
|
|
assertThat(req.type(), is(Socks4CommandType.CONNECT));
|
|
|
|
if (testMode != TestMode.INTERMEDIARY) {
|
|
ctx.pipeline().addBefore(ctx.name(), "lineDecoder", new LineBasedFrameDecoder(64, false, true));
|
|
}
|
|
|
|
boolean authzSuccess;
|
|
if (username != null) {
|
|
authzSuccess = username.equals(req.userId());
|
|
} else {
|
|
authzSuccess = true;
|
|
}
|
|
return authzSuccess;
|
|
}
|
|
|
|
private final class Socks4IntermediaryHandler extends IntermediaryHandler {
|
|
|
|
private SocketAddress intermediaryDestination;
|
|
|
|
@Override
|
|
protected boolean handleProxyProtocol(ChannelHandlerContext ctx, Object msg) throws Exception {
|
|
Socks4CommandRequest req = (Socks4CommandRequest) msg;
|
|
Socks4CommandResponse res;
|
|
|
|
if (!authenticate(ctx, req)) {
|
|
res = new DefaultSocks4CommandResponse(Socks4CommandStatus.IDENTD_AUTH_FAILURE);
|
|
} else {
|
|
res = new DefaultSocks4CommandResponse(Socks4CommandStatus.SUCCESS);
|
|
intermediaryDestination = new InetSocketAddress(req.dstAddr(), req.dstPort());
|
|
}
|
|
|
|
ctx.write(res);
|
|
|
|
ctx.pipeline().remove(Socks4ServerDecoder.class);
|
|
ctx.pipeline().remove(Socks4ServerEncoder.class);
|
|
|
|
return true;
|
|
}
|
|
|
|
@Override
|
|
protected SocketAddress intermediaryDestination() {
|
|
return intermediaryDestination;
|
|
}
|
|
}
|
|
|
|
private final class Socks4TerminalHandler extends TerminalHandler {
|
|
@Override
|
|
protected boolean handleProxyProtocol(ChannelHandlerContext ctx, Object msg) throws Exception {
|
|
Socks4CommandRequest req = (Socks4CommandRequest) msg;
|
|
boolean authzSuccess = authenticate(ctx, req);
|
|
|
|
Socks4CommandResponse res;
|
|
boolean sendGreeting = false;
|
|
if (!authzSuccess) {
|
|
res = new DefaultSocks4CommandResponse(Socks4CommandStatus.IDENTD_AUTH_FAILURE);
|
|
} else if (!req.dstAddr().equals(destination.getHostString()) ||
|
|
req.dstPort() != destination.getPort()) {
|
|
res = new DefaultSocks4CommandResponse(Socks4CommandStatus.REJECTED_OR_FAILED);
|
|
} else {
|
|
res = new DefaultSocks4CommandResponse(Socks4CommandStatus.SUCCESS);
|
|
sendGreeting = true;
|
|
}
|
|
|
|
ctx.write(res);
|
|
|
|
ctx.pipeline().remove(Socks4ServerDecoder.class);
|
|
ctx.pipeline().remove(Socks4ServerEncoder.class);
|
|
|
|
if (sendGreeting) {
|
|
ctx.write(Unpooled.copiedBuffer("0\n", CharsetUtil.US_ASCII));
|
|
}
|
|
|
|
return true;
|
|
}
|
|
}
|
|
}
|