1 package socks; 2 import socks.server.*; 3 import java.net.*; 4 import java.io.*; 5 6 9 class UDPRelayServer implements Runnable { 10 11 12 DatagramSocket client_sock; 13 DatagramSocket remote_sock; 14 15 Socket controlConnection; 16 17 int relayPort; 18 InetAddress relayIP; 19 20 Thread pipe_thread1,pipe_thread2; 21 Thread master_thread; 22 23 ServerAuthenticator auth; 24 25 long lastReadTime; 26 27 static PrintStream log = null; 28 static Proxy proxy = null; 29 static int datagramSize = 0xFFFF; static int iddleTimeout = 180000; 32 33 46 public UDPRelayServer(InetAddress clientIP,int clientPort, 47 Thread master_thread, 48 Socket controlConnection, 49 ServerAuthenticator auth) 50 throws IOException{ 51 this.master_thread = master_thread; 52 this.controlConnection = controlConnection; 53 this.auth = auth; 54 55 client_sock = new Socks5DatagramSocket(true,auth.getUdpEncapsulation(), 56 clientIP,clientPort); 57 relayPort = client_sock.getLocalPort(); 58 relayIP = client_sock.getLocalAddress(); 59 60 if(relayIP.getHostAddress().equals("0.0.0.0")) 61 relayIP = InetAddress.getLocalHost(); 62 63 if(proxy == null) 64 remote_sock = new DatagramSocket(); 65 else 66 remote_sock = new Socks5DatagramSocket(proxy,0,null); 67 } 68 69 70 73 74 79 80 static public void setTimeout(int timeout){ 81 iddleTimeout = timeout; 82 } 83 84 85 90 static public void setDatagramSize(int size){ 91 datagramSize = size; 92 } 93 94 97 public int getRelayPort(){ 98 return relayPort; 99 } 100 103 public InetAddress getRelayIP(){ 104 return relayIP; 105 } 106 107 111 public void start() throws IOException{ 112 remote_sock.setSoTimeout(iddleTimeout); 113 client_sock.setSoTimeout(iddleTimeout); 114 115 log("Starting UDP relay server on "+relayIP+":"+relayPort); 116 log("Remote socket "+remote_sock.getLocalAddress()+":"+ 117 remote_sock.getLocalPort()); 118 119 pipe_thread1 = new Thread (this,"pipe1"); 120 pipe_thread2 = new Thread (this,"pipe2"); 121 122 lastReadTime = System.currentTimeMillis(); 123 124 pipe_thread1.start(); 125 pipe_thread2.start(); 126 } 127 128 133 public synchronized void stop(){ 134 master_thread = null; 135 controlConnection = null; 136 abort(); 137 } 138 139 public void run(){ 142 try{ 143 if(Thread.currentThread().getName().equals("pipe1")) 144 pipe(remote_sock,client_sock,false); 145 else 146 pipe(client_sock,remote_sock,true); 147 }catch(IOException ioe){ 148 }finally{ 149 abort(); 150 log("UDP Pipe thread "+Thread.currentThread().getName()+" stopped."); 151 } 152 153 } 154 155 private synchronized void abort(){ 158 if(pipe_thread1 == null) return; 159 160 log("Aborting UDP Relay Server"); 161 162 remote_sock.close(); 163 client_sock.close(); 164 165 if(controlConnection != null) 166 try{ controlConnection.close();} catch(IOException ioe){} 167 168 if(master_thread!=null) master_thread.interrupt(); 169 170 pipe_thread1.interrupt(); 171 pipe_thread2.interrupt(); 172 173 pipe_thread1 = null; 174 } 175 176 177 static private void log(String s){ 178 if(log != null){ 179 log.println(s); 180 log.flush(); 181 } 182 } 183 184 private void pipe(DatagramSocket from,DatagramSocket to,boolean out) 185 throws IOException{ 186 byte[] data = new byte[datagramSize]; 187 DatagramPacket dp = new DatagramPacket(data,data.length); 188 189 while(true){ 190 try{ 191 from.receive(dp); 192 lastReadTime = System.currentTimeMillis(); 193 194 if(auth.checkRequest(dp,out)) 195 to.send(dp); 196 197 }catch(UnknownHostException uhe){ 198 log("Dropping datagram for unknown host"); 199 }catch(InterruptedIOException iioe){ 200 if(iddleTimeout == 0) return; 203 204 long timeSinceRead = System.currentTimeMillis() - lastReadTime; 206 if(timeSinceRead >= iddleTimeout -100) return; 208 } 209 dp.setLength(data.length); 210 } 211 } 212 } 213
| Popular Tags
|