/src/net/sourceforge/jsocks/SocksSocket.java

https://code.google.com/p/sshtunnel/ · Java · 338 lines · 139 code · 29 blank · 170 comment · 8 complexity · 084c28c68f3069307c4647e4c76a30a9 MD5 · raw file

  1. package net.sourceforge.jsocks;
  2. import java.io.IOException;
  3. import java.io.InputStream;
  4. import java.io.OutputStream;
  5. import java.net.InetAddress;
  6. import java.net.Socket;
  7. import java.net.SocketException;
  8. import java.net.UnknownHostException;
  9. /**
  10. * SocksSocket tryies to look very similar to normal Socket, while allowing
  11. * connections through the SOCKS4 or 5 proxy. To use this class you will have to
  12. * identify proxy you need to use, Proxy class allows you to set default proxy,
  13. * which will be used by all Socks aware sockets. You can also create either
  14. * Socks4Proxy or Socks5Proxy, and use them by passing to the appropriate
  15. * constructors.
  16. * <P>
  17. * Using Socks package can be as easy as that:
  18. *
  19. * <pre>
  20. * <tt>
  21. *
  22. * import Socks.*;
  23. * ....
  24. *
  25. * try{
  26. * //Specify SOCKS5 proxy
  27. * Proxy.setDefaultProxy("socks-proxy",1080);
  28. *
  29. * //OR you still use SOCKS4
  30. * //Code below uses SOCKS4 proxy
  31. * //Proxy.setDefaultProxy("socks-proxy",1080,userName);
  32. *
  33. * Socket s = SocksSocket("some.host.of.mine",13);
  34. * readTimeFromSock(s);
  35. * }catch(SocksException sock_ex){
  36. * //Usually it will turn in more or less meaningfull message
  37. * System.err.println("SocksException:"+sock_ex);
  38. * }
  39. *
  40. * </tt>
  41. * </pre>
  42. * <P>
  43. * However if the need exist for more control, like resolving addresses
  44. * remotely, or using some non-trivial authentication schemes, it can be done.
  45. */
  46. public class SocksSocket extends Socket {
  47. // Data members
  48. protected Proxy proxy;
  49. protected String localHost, remoteHost;
  50. protected InetAddress localIP, remoteIP;
  51. protected int localPort, remotePort;
  52. private Socket directSock = null;
  53. /**
  54. * Connects to given ip and port using given Proxy server.
  55. *
  56. * @param p
  57. * Proxy to use.
  58. * @param ip
  59. * Machine to connect to.
  60. * @param port
  61. * Port to which to connect.
  62. */
  63. public SocksSocket(InetAddress ip, int port) throws SocksException {
  64. this.remoteIP = ip;
  65. this.remotePort = port;
  66. this.remoteHost = ip.getHostName();
  67. doDirect();
  68. }
  69. protected SocksSocket(InetAddress ip, int port, Proxy proxy) {
  70. remoteIP = ip;
  71. remotePort = port;
  72. this.proxy = proxy;
  73. this.localIP = proxy.proxySocket.getLocalAddress();
  74. this.localPort = proxy.proxySocket.getLocalPort();
  75. remoteHost = remoteIP.getHostName();
  76. }
  77. /**
  78. * Connects to host port using given proxy server.
  79. *
  80. * @param p
  81. * Proxy to use.
  82. * @param host
  83. * Machine to connect to.
  84. * @param port
  85. * Port to which to connect.
  86. * @throws UnknownHostException
  87. * If one of the following happens:
  88. * <ol>
  89. *
  90. * <li>Proxy settings say that address should be resolved
  91. * locally, but this fails.
  92. * <li>Proxy settings say that the host should be contacted
  93. * directly but host name can't be resolved.
  94. * </ol>
  95. * @throws SocksException
  96. * If one of the following happens:
  97. * <ul>
  98. * <li>Proxy is is null.
  99. * <li>Proxy settings say that the host should be contacted
  100. * directly but this fails.
  101. * <li>Socks Server can't be contacted.
  102. * <li>Authentication fails.
  103. * <li>Connection is not allowed by the SOCKS proxy.
  104. * <li>SOCKS proxy can't establish the connection.
  105. * <li>Any IO error occured.
  106. * <li>Any protocol error occured.
  107. * </ul>
  108. * @throws IOexception
  109. * if anything is wrong with I/O.
  110. * @see Socks5Proxy#resolveAddrLocally
  111. */
  112. public SocksSocket(Proxy p, String host, int port) throws SocksException,
  113. UnknownHostException {
  114. remoteHost = host;
  115. remotePort = port;
  116. remoteIP = InetAddress.getByName(host);
  117. doDirect();
  118. }
  119. /**
  120. * Tryies to connect to given host and port using default proxy. If no
  121. * default proxy speciefied it throws SocksException with error code
  122. * SOCKS_NO_PROXY.
  123. *
  124. * @param host
  125. * Machine to connect to.
  126. * @param port
  127. * Port to which to connect.
  128. * @see SocksSocket#SocksSocket(Proxy,String,int)
  129. * @see Socks5Proxy#resolveAddrLocally
  130. */
  131. public SocksSocket(String host, int port) throws SocksException,
  132. UnknownHostException {
  133. this(Proxy.defaultProxy, host, port);
  134. }
  135. /**
  136. * These 2 constructors are used by the SocksServerSocket. This socket
  137. * simply overrides remoteHost, remotePort
  138. */
  139. protected SocksSocket(String host, int port, Proxy proxy) {
  140. this.remotePort = port;
  141. this.proxy = proxy;
  142. this.localIP = proxy.proxySocket.getLocalAddress();
  143. this.localPort = proxy.proxySocket.getLocalPort();
  144. this.remoteHost = host;
  145. }
  146. /**
  147. * Same as Socket
  148. */
  149. @Override
  150. public void close() throws IOException {
  151. if (proxy != null)
  152. proxy.endSession();
  153. proxy = null;
  154. }
  155. private void doDirect() throws SocksException {
  156. try {
  157. // System.out.println("IP:"+remoteIP+":"+remotePort);
  158. directSock = new Socket(remoteIP, remotePort);
  159. proxy.out = directSock.getOutputStream();
  160. proxy.in = directSock.getInputStream();
  161. proxy.proxySocket = directSock;
  162. localIP = directSock.getLocalAddress();
  163. localPort = directSock.getLocalPort();
  164. } catch (IOException io_ex) {
  165. throw new SocksException(Proxy.SOCKS_DIRECT_FAILED,
  166. "Direct connect failed:" + io_ex);
  167. }
  168. }
  169. /**
  170. * Returns remote host name, it is usefull in cases when addresses are
  171. * resolved by proxy, and we can't create InetAddress object.
  172. *
  173. * @return The name of the host this socket is connected to.
  174. */
  175. public String getHost() {
  176. return remoteHost;
  177. }
  178. /**
  179. * Get remote host as InetAddress object, might return null if addresses are
  180. * resolved by proxy, and it is not possible to resolve it locally
  181. *
  182. * @return Ip address of the host this socket is connected to, or null if
  183. * address was returned by the proxy as DOMAINNAME and can't be
  184. * resolved locally.
  185. */
  186. @Override
  187. public InetAddress getInetAddress() {
  188. if (remoteIP == null) {
  189. try {
  190. remoteIP = InetAddress.getByName(remoteHost);
  191. } catch (UnknownHostException e) {
  192. return null;
  193. }
  194. }
  195. return remoteIP;
  196. }
  197. /**
  198. * Same as Socket
  199. */
  200. @Override
  201. public InputStream getInputStream() {
  202. return proxy.in;
  203. }
  204. /**
  205. * Get address assigned by proxy to make a remote connection, it might be
  206. * different from the host specified for the proxy. Can return null if socks
  207. * server returned this address as hostname and it can't be resolved
  208. * locally, use getLocalHost() then.
  209. *
  210. * @return Address proxy is using to make a connection.
  211. */
  212. @Override
  213. public InetAddress getLocalAddress() {
  214. if (localIP == null) {
  215. try {
  216. localIP = InetAddress.getByName(localHost);
  217. } catch (UnknownHostException e) {
  218. return null;
  219. }
  220. }
  221. return localIP;
  222. }
  223. /**
  224. * Get name of the host, proxy has assigned to make a remote connection for
  225. * this socket. This method is usefull when proxy have returned address as
  226. * hostname, and we can't resolve it on this machine.
  227. *
  228. * @return The name of the host proxy is using to make a connection.
  229. */
  230. public String getLocalHost() {
  231. return localHost;
  232. }
  233. /**
  234. * Get the port assigned by the proxy for the socket, not the port on locall
  235. * machine as in Socket.
  236. *
  237. * @return Port of the socket used on the proxy server.
  238. */
  239. @Override
  240. public int getLocalPort() {
  241. return localPort;
  242. }
  243. /**
  244. * Same as Socket
  245. */
  246. @Override
  247. public OutputStream getOutputStream() {
  248. return proxy.out;
  249. }
  250. /**
  251. * Same as Socket
  252. */
  253. @Override
  254. public int getPort() {
  255. return remotePort;
  256. }
  257. /**
  258. * Same as socket.
  259. */
  260. public int getSoLinger(int timeout) throws SocketException {
  261. return proxy.proxySocket.getSoLinger();
  262. }
  263. /**
  264. * Same as socket.
  265. */
  266. public int getSoTimeout(int timeout) throws SocketException {
  267. return proxy.proxySocket.getSoTimeout();
  268. }
  269. /**
  270. * Same as socket.
  271. */
  272. @Override
  273. public boolean getTcpNoDelay() throws SocketException {
  274. return proxy.proxySocket.getTcpNoDelay();
  275. }
  276. /**
  277. * Same as socket.
  278. */
  279. @Override
  280. public void setSoLinger(boolean on, int val) throws SocketException {
  281. proxy.proxySocket.setSoLinger(on, val);
  282. }
  283. /**
  284. * Same as socket.
  285. */
  286. @Override
  287. public void setSoTimeout(int timeout) throws SocketException {
  288. proxy.proxySocket.setSoTimeout(timeout);
  289. }
  290. /**
  291. * Same as socket.
  292. */
  293. @Override
  294. public void setTcpNoDelay(boolean on) throws SocketException {
  295. proxy.proxySocket.setTcpNoDelay(on);
  296. }
  297. // Private Methods
  298. // ////////////////
  299. /**
  300. * Get string representation of the socket.
  301. */
  302. @Override
  303. public String toString() {
  304. if (directSock != null)
  305. return "Direct connection:" + directSock;
  306. return ("Proxy:" + proxy + ";" + "addr:" + remoteHost + ",port:"
  307. + remotePort + ",localport:" + localPort);
  308. }
  309. }