📄 socks5datagramsocket.java
字号:
package org.jivesoftware.smackx.filetransfer.jsocks;
import java.net.*;
import java.io.*;
/**
Datagram socket to interract through the firewall.
<BR>
Can be used same way as the normal DatagramSocket. One should
be carefull though with the datagram sizes used, as additional data
is present in both incomming and outgoing datagrams.
<p>
SOCKS5 protocol allows to send host address as either:
<ul>
<li> IPV4, normal 4 byte address. (10 bytes header size)
<li> IPV6, version 6 ip address (not supported by Java as for now).
22 bytes header size.
<li> Host name,(7+length of the host name bytes header size).
</ul>
As with other Socks equivalents, direct addresses are handled
transparently, that is data will be send directly when required
by the proxy settings.
<p>
<b>NOTE:</b><br>
Unlike other SOCKS Sockets, it <b>does not</b> support proxy chaining,
and will throw an exception if proxy has a chain proxy attached. The
reason for that is not my laziness, but rather the restrictions of
the SOCKSv5 protocol. Basicaly SOCKSv5 proxy server, needs to know from
which host:port datagrams will be send for association, and returns address
to which datagrams should be send by the client, but it does not
inform client from which host:port it is going to send datagrams, in fact
there is even no guarantee they will be send at all and from the same address
each time.
*/
public class Socks5DatagramSocket extends DatagramSocket{
InetAddress relayIP;
int relayPort;
Socks5Proxy proxy;
private boolean server_mode = false;
UDPEncapsulation encapsulation;
/**
Construct Datagram socket for communication over SOCKS5 proxy
server. This constructor uses default proxy, the one set with
Proxy.setDefaultProxy() method. If default proxy is not set or
it is set to version4 proxy, which does not support datagram
forwarding, throws SocksException.
*/
public Socks5DatagramSocket() throws SocksException,
IOException{
this(Proxy.defaultProxy,0,null);
}
/**
Construct Datagram socket for communication over SOCKS5 proxy
server. And binds it to the specified local port.
This constructor uses default proxy, the one set with
Proxy.setDefaultProxy() method. If default proxy is not set or
it is set to version4 proxy, which does not support datagram
forwarding, throws SocksException.
*/
public Socks5DatagramSocket(int port) throws SocksException,
IOException{
this(Proxy.defaultProxy,port,null);
}
/**
Construct Datagram socket for communication over SOCKS5 proxy
server. And binds it to the specified local port and address.
This constructor uses default proxy, the one set with
Proxy.setDefaultProxy() method. If default proxy is not set or
it is set to version4 proxy, which does not support datagram
forwarding, throws SocksException.
*/
public Socks5DatagramSocket(int port,InetAddress ip) throws SocksException,
IOException{
this(Proxy.defaultProxy,port,ip);
}
/**
Constructs datagram socket for communication over specified proxy.
And binds it to the given local address and port. Address of null
and port of 0, signify any availabale port/address.
Might throw SocksException, if:
<ol>
<li> Given version of proxy does not support UDP_ASSOCIATE.
<li> Proxy can't be reached.
<li> Authorization fails.
<li> Proxy does not want to perform udp forwarding, for any reason.
</ol>
Might throw IOException if binding dtagram socket to given address/port
fails.
See java.net.DatagramSocket for more details.
*/
public Socks5DatagramSocket(Proxy p,int port,InetAddress ip)
throws SocksException,
IOException{
super(port,ip);
if(p == null) throw new SocksException(Proxy.SOCKS_NO_PROXY);
if(!(p instanceof Socks5Proxy))
throw new SocksException(-1,"Datagram Socket needs Proxy version 5");
if(p.chainProxy != null)
throw new SocksException(Proxy.SOCKS_JUST_ERROR,
"Datagram Sockets do not support proxy chaining.");
proxy =(Socks5Proxy) p.copy();
ProxyMessage msg = proxy.udpAssociate(super.getLocalAddress(),
super.getLocalPort());
relayIP = msg.ip;
if(relayIP.getHostAddress().equals("0.0.0.0")) relayIP = proxy.proxyIP;
relayPort = msg.port;
encapsulation = proxy.udp_encapsulation;
//debug("Datagram Socket:"+getLocalAddress()+":"+getLocalPort()+"\n");
//debug("Socks5Datagram: "+relayIP+":"+relayPort+"\n");
}
/**
Used by UDPRelayServer.
*/
Socks5DatagramSocket(boolean server_mode,UDPEncapsulation encapsulation,
InetAddress relayIP,int relayPort)
throws IOException{
super();
this.server_mode = server_mode;
this.relayIP = relayIP;
this.relayPort = relayPort;
this.encapsulation = encapsulation;
this.proxy = null;
}
/**
Sends the Datagram either through the proxy or directly depending
on current proxy settings and destination address. <BR>
<B> NOTE: </B> DatagramPacket size should be at least 10 bytes less
than the systems limit.
<P>
See documentation on java.net.DatagramSocket
for full details on how to use this method.
@param dp Datagram to send.
@throws IOException If error happens with I/O.
*/
public void send(DatagramPacket dp) throws IOException{
//If the host should be accessed directly, send it as is.
if(!server_mode && proxy.isDirect(dp.getAddress())){
super.send(dp);
//debug("Sending directly:");
return;
}
byte[] head = formHeader(dp.getAddress(),dp.getPort());
byte[] buf = new byte[head.length + dp.getLength()];
byte[] data = dp.getData();
//Merge head and data
System.arraycopy(head,0,buf,0,head.length);
//System.arraycopy(data,dp.getOffset(),buf,head.length,dp.getLength());
System.arraycopy(data,0,buf,head.length,dp.getLength());
if(encapsulation != null)
buf = encapsulation.udpEncapsulate(buf,true);
super.send(new DatagramPacket(buf,buf.length,relayIP,relayPort));
}
/**
This method allows to send datagram packets with address type DOMAINNAME.
SOCKS5 allows to specify host as names rather than ip addresses.Using
this method one can send udp datagrams through the proxy, without having
to know the ip address of the destination host.
<p>
If proxy specified for that socket has an option resolveAddrLocally set
to true host will be resolved, and the datagram will be send with address
type IPV4, if resolve fails, UnknownHostException is thrown.
@param dp Datagram to send, it should contain valid port and data
@param host Host name to which datagram should be send.
@throws IOException If error happens with I/O, or the host can't be
resolved when proxy settings say that hosts should be resolved locally.
@see Socks5Proxy#resolveAddrLocally(boolean)
*/
public void send(DatagramPacket dp, String host) throws IOException{
if(proxy.isDirect(host)){
dp.setAddress(InetAddress.getByName(host));
super.send(dp);
return;
}
if(((Socks5Proxy)proxy).resolveAddrLocally){
dp.setAddress(InetAddress.getByName(host));
}
byte[] head = formHeader(host,dp.getPort());
byte[] buf = new byte[head.length + dp.getLength()];
byte[] data = dp.getData();
//Merge head and data
System.arraycopy(head,0,buf,0,head.length);
//System.arraycopy(data,dp.getOffset(),buf,head.length,dp.getLength());
System.arraycopy(data,0,buf,head.length,dp.getLength());
if(encapsulation != null)
buf = encapsulation.udpEncapsulate(buf,true);
super.send(new DatagramPacket(buf,buf.length,relayIP,relayPort));
}
/**
* Receives udp packet. If packet have arrived from the proxy relay server,
* it is processed and address and port of the packet are set to the
* address and port of sending host.<BR>
* If the packet arrived from anywhere else it is not changed.<br>
* <B> NOTE: </B> DatagramPacket size should be at least 10 bytes bigger
* than the largest packet you expect (this is for IPV4 addresses).
* For hostnames and IPV6 it is even more.
@param dp Datagram in which all relevent information will be copied.
*/
public void receive(DatagramPacket dp) throws IOException{
super.receive(dp);
if(server_mode){
//Drop all datagrams not from relayIP/relayPort
int init_length = dp.getLength();
int initTimeout = getSoTimeout();
long startTime = System.currentTimeMillis();
while(!relayIP.equals(dp.getAddress()) ||
relayPort != dp.getPort()){
//Restore datagram size
dp.setLength(init_length);
//If there is a non-infinit timeout on this socket
//Make sure that it happens no matter how often unexpected
//packets arrive.
if(initTimeout != 0){
int newTimeout = initTimeout - (int)(System.currentTimeMillis() -
startTime);
if(newTimeout <= 0) throw new InterruptedIOException(
"In Socks5DatagramSocket->receive()");
setSoTimeout(newTimeout);
}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -