org.simplejavamail.internal.authenticatedsockssupport.socks5client.SocksCommandSender Maven / Gradle / Ivy
/*
* Copyright © 2009 Benny Bottema ([email protected])
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.simplejavamail.internal.authenticatedsockssupport.socks5client;
import org.jetbrains.annotations.NotNull;
import org.simplejavamail.internal.authenticatedsockssupport.common.SocksException;
import org.simplejavamail.internal.util.MiscUtil;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.InetAddress;
import java.net.InetSocketAddress;
import java.net.Socket;
import java.net.SocketAddress;
import static java.nio.charset.StandardCharsets.UTF_8;
final class SocksCommandSender {
private static final Logger LOGGER = LoggerFactory.getLogger(SocksCommandSender.class);
private static final byte SOCKS_VERSION = 0x05;
private static final int ADDRESS_TYPE_IPV4 = 0x01;
private static final int ADDRESS_TYPE_DOMAIN_NAME = 0x03;
private static final int ADDRESS_TYPE_IPV6 = 0x04;
private static final int COMMAND_CONNECT = 0x01;
private static final int LENGTH_OF_IPV4 = 4;
private static final int LENGTH_OF_IPV6 = 16;
private static final int RESERVED = 0x00;
private static final byte ATYPE_IPV4 = 0x01;
private static final byte ATYPE_DOMAINNAME = 0x03;
private static final byte ATYPE_IPV6 = 0x04;
private static final int REP_SUCCEEDED = 0x00;
public static void send(@NotNull final Socket socket, final InetAddress address, final int port)
throws IOException {
send(socket, new InetSocketAddress(address, port));
}
public static void send(@NotNull final Socket socket, final SocketAddress socketAddress)
throws IOException {
if (!(socketAddress instanceof InetSocketAddress)) {
throw new IllegalArgumentException("Unsupported address type");
}
final InputStream inputStream = socket.getInputStream();
final OutputStream outputStream = socket.getOutputStream();
final InetSocketAddress address = (InetSocketAddress) socketAddress;
final byte[] bytesOfAddress = address.getAddress().getAddress();
final int ADDRESS_LENGTH = bytesOfAddress.length;
final int port = address.getPort();
final byte addressType;
final byte[] bufferSent;
if (ADDRESS_LENGTH == LENGTH_OF_IPV4) {
addressType = ATYPE_IPV4;
bufferSent = new byte[6 + LENGTH_OF_IPV4];
} else if (ADDRESS_LENGTH == LENGTH_OF_IPV6) {
addressType = ATYPE_IPV6;
bufferSent = new byte[6 + LENGTH_OF_IPV6];
} else {
throw new SocksException("Address error");// TODO
}
bufferSent[0] = SOCKS_VERSION;
bufferSent[1] = (byte) COMMAND_CONNECT;
bufferSent[2] = RESERVED;
bufferSent[3] = addressType;
System.arraycopy(bytesOfAddress, 0, bufferSent, 4, ADDRESS_LENGTH);// copy address bytes
bufferSent[4 + ADDRESS_LENGTH] = (byte) ((port & 0xff00) >> 8);
bufferSent[5 + ADDRESS_LENGTH] = (byte) (port & 0xff);
outputStream.write(bufferSent);
outputStream.flush();
LOGGER.trace("{}", MiscUtil.buildLogStringForSOCKSCommunication(bufferSent, false));
checkServerReply(inputStream);
}
public static void send(@NotNull final Socket socket, final String host, final int port)
throws IOException, SocksException {
final InputStream inputStream = socket.getInputStream();
final OutputStream outputStream = socket.getOutputStream();
final int lengthOfHost = host.getBytes(UTF_8).length;
final byte[] bufferSent = new byte[7 + lengthOfHost];
bufferSent[0] = SOCKS_VERSION;
bufferSent[1] = (byte) COMMAND_CONNECT;
bufferSent[2] = RESERVED;
bufferSent[3] = ATYPE_DOMAINNAME;
bufferSent[4] = (byte) lengthOfHost;
final byte[] bytesOfHost = host.getBytes(UTF_8);
System.arraycopy(bytesOfHost, 0, bufferSent, 5, lengthOfHost);// copy host bytes.
bufferSent[5 + host.length()] = (byte) ((port & 0xff00) >> 8);
bufferSent[6 + host.length()] = (byte) (port & 0xff);
outputStream.write(bufferSent);
outputStream.flush();
LOGGER.trace("{}", MiscUtil.buildLogStringForSOCKSCommunication(bufferSent, false));
checkServerReply(inputStream);
}
private static void checkServerReply(final InputStream inputStream)
throws IOException, SocksException {
final ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
int temp = 0;
for (int i = 0; i < 4; i++) {
temp = inputStream.read();
byteArrayOutputStream.write(temp);
}
final byte addressType = (byte) temp;
switch (addressType) {
case ADDRESS_TYPE_IPV4:
for (int i = 0; i < 6; i++) {
byteArrayOutputStream.write(inputStream.read());
}
break;
case ADDRESS_TYPE_DOMAIN_NAME:
temp = inputStream.read();
byteArrayOutputStream.write(temp);
for (int i = 0; i < temp + 2; i++) {
byteArrayOutputStream.write(inputStream.read());
}
break;
case ADDRESS_TYPE_IPV6:
for (int i = 0; i < 18; i++) {
byteArrayOutputStream.write(inputStream.read());
}
break;
default:
throw new SocksException("Address type not support, type value: " + addressType);
}
final byte[] receivedData = byteArrayOutputStream.toByteArray();
LOGGER.trace("{}", MiscUtil.buildLogStringForSOCKSCommunication(receivedData, true));
final byte[] addressBytes;
final byte[] portBytes = new byte[2];
if (receivedData[3] == ADDRESS_TYPE_IPV4) {
addressBytes = new byte[4];
System.arraycopy(receivedData, 4, addressBytes, 0, addressBytes.length);
final int a = MiscUtil.toInt(addressBytes[0]);
final int b = MiscUtil.toInt(addressBytes[1]);
final int c = MiscUtil.toInt(addressBytes[2]);
final int d = MiscUtil.toInt(addressBytes[3]);
portBytes[0] = receivedData[8];
portBytes[1] = receivedData[9];
LOGGER.debug("Server replied:Address as IPv4:{}.{}.{}.{}, port:{}", a, b, c, d,
(MiscUtil.toInt(portBytes[0]) << 8) | (MiscUtil.toInt(portBytes[1])));
} else if (receivedData[3] == ADDRESS_TYPE_DOMAIN_NAME) {
int size = receivedData[4];
size = size & 0xFF;
addressBytes = new byte[size];
System.arraycopy(receivedData, 4, addressBytes, 0, size);
portBytes[0] = receivedData[4 + size];
portBytes[1] = receivedData[5 + size];
LOGGER.debug("Server replied:Address as host:{}, port:{}", new String(addressBytes, UTF_8),
(MiscUtil.toInt(portBytes[0]) << 8) | (MiscUtil.toInt(portBytes[1])));
} else if (receivedData[3] == ADDRESS_TYPE_IPV6) {
addressBytes = new byte[16];
System.arraycopy(receivedData, 4, addressBytes, 0, addressBytes.length);
LOGGER.debug("Server replied:Address as IPv6:{}", new String(addressBytes, UTF_8));
}
final byte serverReply = receivedData[1];
if (serverReply != REP_SUCCEEDED) {
throw SocksException.serverReplyException(serverReply);
}
LOGGER.debug("SOCKS server response success");
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy