1 /* 2 * Copyright (C) 2016 Google Inc. 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); you may not 5 * use this file except in compliance with the License. You may obtain a copy of 6 * the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT 12 * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the 13 * License for the specific language governing permissions and limitations under 14 * the License. 15 */ 16 17 package com.googlecode.android_scripting; 18 19 import com.google.common.collect.Lists; 20 21 import org.json.JSONException; 22 import org.json.JSONObject; 23 24 import java.io.BufferedReader; 25 import java.io.IOException; 26 import java.io.InputStreamReader; 27 import java.io.PrintWriter; 28 import java.net.BindException; 29 import java.net.Inet4Address; 30 import java.net.InetAddress; 31 import java.net.InetSocketAddress; 32 import java.net.NetworkInterface; 33 import java.net.ServerSocket; 34 import java.net.Socket; 35 import java.net.SocketException; 36 import java.net.UnknownHostException; 37 import java.util.Collections; 38 import java.util.Enumeration; 39 import java.util.List; 40 import java.util.concurrent.ConcurrentHashMap; 41 //import com.googlecode.android_scripting.jsonrpc.RpcReceiverManager; 42 43 /** 44 * A simple server. 45 */ 46 public abstract class SimpleServer { 47 private static int threadIndex = 0; 48 private final ConcurrentHashMap<Integer, ConnectionThread> mConnectionThreads = 49 new ConcurrentHashMap<Integer, ConnectionThread>(); 50 private final List<SimpleServerObserver> mObservers = Lists.newArrayList(); 51 private volatile boolean mStopServer = false; 52 private ServerSocket mServer; 53 private Thread mServerThread; 54 55 public interface SimpleServerObserver { onConnect()56 public void onConnect(); onDisconnect()57 public void onDisconnect(); 58 } 59 handleConnection(Socket socket)60 protected abstract void handleConnection(Socket socket) throws Exception; handleRPCConnection(Socket socket, Integer UID, BufferedReader reader, PrintWriter writer)61 protected abstract void handleRPCConnection(Socket socket, 62 Integer UID, 63 BufferedReader reader, 64 PrintWriter writer) throws Exception; 65 66 /** Adds an observer. */ addObserver(SimpleServerObserver observer)67 public void addObserver(SimpleServerObserver observer) { 68 mObservers.add(observer); 69 } 70 71 /** Removes an observer. */ removeObserver(SimpleServerObserver observer)72 public void removeObserver(SimpleServerObserver observer) { 73 mObservers.remove(observer); 74 } 75 notifyOnConnect()76 private void notifyOnConnect() { 77 for (SimpleServerObserver observer : mObservers) { 78 observer.onConnect(); 79 } 80 } 81 notifyOnDisconnect()82 private void notifyOnDisconnect() { 83 for (SimpleServerObserver observer : mObservers) { 84 observer.onDisconnect(); 85 } 86 } 87 88 private final class ConnectionThread extends Thread { 89 private final Socket mmSocket; 90 private final BufferedReader reader; 91 private final PrintWriter writer; 92 private final Integer UID; 93 private final boolean isRpc; 94 ConnectionThread(Socket socket, boolean rpc, Integer uid, BufferedReader reader, PrintWriter writer)95 private ConnectionThread(Socket socket, boolean rpc, Integer uid, BufferedReader reader, PrintWriter writer) { 96 setName("SimpleServer ConnectionThread " + getId()); 97 mmSocket = socket; 98 this.UID = uid; 99 this.reader = reader; 100 this.writer = writer; 101 this.isRpc = rpc; 102 } 103 104 @Override run()105 public void run() { 106 Log.v("Server thread " + getId() + " started."); 107 try { 108 if(isRpc) { 109 Log.d("Handling RPC connection in "+getId()); 110 handleRPCConnection(mmSocket, UID, reader, writer); 111 }else{ 112 Log.d("Handling Non-RPC connection in "+getId()); 113 handleConnection(mmSocket); 114 } 115 } catch (Exception e) { 116 if (!mStopServer) { 117 Log.e("Server error.", e); 118 } 119 } finally { 120 close(); 121 mConnectionThreads.remove(this.UID); 122 notifyOnDisconnect(); 123 Log.v("Server thread " + getId() + " stopped."); 124 } 125 } 126 close()127 private void close() { 128 if (mmSocket != null) { 129 try { 130 mmSocket.close(); 131 } catch (IOException e) { 132 Log.e(e.getMessage(), e); 133 } 134 } 135 } 136 } 137 138 /** Returns the number of active connections to this server. */ getNumberOfConnections()139 public int getNumberOfConnections() { 140 return mConnectionThreads.size(); 141 } 142 getPrivateInetAddress()143 public static InetAddress getPrivateInetAddress() throws UnknownHostException, SocketException { 144 145 InetAddress candidate = null; 146 Enumeration<NetworkInterface> nets = NetworkInterface.getNetworkInterfaces(); 147 for (NetworkInterface netint : Collections.list(nets)) { 148 if (!netint.isLoopback() || !netint.isUp()) { // Ignore if localhost or not active 149 continue; 150 } 151 Enumeration<InetAddress> addresses = netint.getInetAddresses(); 152 for (InetAddress address : Collections.list(addresses)) { 153 if (address instanceof Inet4Address) { 154 Log.d("local address " + address); 155 return address; // Prefer ipv4 156 } 157 candidate = address; // Probably an ipv6 158 } 159 } 160 if (candidate != null) { 161 return candidate; // return ipv6 address if no suitable ipv6 162 } 163 return InetAddress.getLocalHost(); // No damn matches. Give up, return local host. 164 } 165 getPublicInetAddress()166 public static InetAddress getPublicInetAddress() throws UnknownHostException, SocketException { 167 168 InetAddress candidate = null; 169 Enumeration<NetworkInterface> nets = NetworkInterface.getNetworkInterfaces(); 170 for (NetworkInterface netint : Collections.list(nets)) { 171 if (netint.isLoopback() || !netint.isUp()) { // Ignore if localhost or not active 172 continue; 173 } 174 Enumeration<InetAddress> addresses = netint.getInetAddresses(); 175 for (InetAddress address : Collections.list(addresses)) { 176 if (address instanceof Inet4Address) { 177 return address; // Prefer ipv4 178 } 179 candidate = address; // Probably an ipv6 180 } 181 } 182 if (candidate != null) { 183 return candidate; // return ipv6 address if no suitable ipv6 184 } 185 return InetAddress.getLocalHost(); // No damn matches. Give up, return local host. 186 } 187 188 /** 189 * Starts the RPC server bound to the localhost address. 190 * 191 * @param port 192 * the port to bind to or 0 to pick any unused port 193 * 194 * @return the port that the server is bound to 195 * @throws IOException 196 */ startLocal(int port)197 public InetSocketAddress startLocal(int port) { 198 InetAddress address; 199 try { 200 // address = InetAddress.getLocalHost(); 201 address = getPrivateInetAddress(); 202 mServer = new ServerSocket(port, 5, address); 203 } catch (BindException e) { 204 Log.e("Port " + port + " already in use."); 205 try { 206 address = getPrivateInetAddress(); 207 mServer = new ServerSocket(0, 5, address); 208 } catch (IOException e1) { 209 e1.printStackTrace(); 210 return null; 211 } 212 } catch (Exception e) { 213 Log.e("Failed to start server.", e); 214 return null; 215 } 216 int boundPort = start(); 217 return InetSocketAddress.createUnresolved(mServer.getInetAddress().getHostAddress(), boundPort); 218 } 219 220 /** 221 * data Starts the RPC server bound to the public facing address. 222 * 223 * @param port 224 * the port to bind to or 0 to pick any unused port 225 * 226 * @return the port that the server is bound to 227 */ startPublic(int port)228 public InetSocketAddress startPublic(int port) { 229 InetAddress address; 230 try { 231 // address = getPublicInetAddress(); 232 address = null; 233 mServer = new ServerSocket(port, 5 /* backlog */, address); 234 } catch (Exception e) { 235 Log.e("Failed to start server.", e); 236 return null; 237 } 238 int boundPort = start(); 239 return InetSocketAddress.createUnresolved(mServer.getInetAddress().getHostAddress(), boundPort); 240 } 241 242 /** 243 * data Starts the RPC server bound to all interfaces 244 * 245 * @param port 246 * the port to bind to or 0 to pick any unused port 247 * 248 * @return the port that the server is bound to 249 */ startAllInterfaces(int port)250 public InetSocketAddress startAllInterfaces(int port) { 251 try { 252 mServer = new ServerSocket(port, 5 /* backlog */); 253 } catch (Exception e) { 254 Log.e("Failed to start server.", e); 255 return null; 256 } 257 int boundPort = start(); 258 return InetSocketAddress.createUnresolved(mServer.getInetAddress().getHostAddress(), boundPort); 259 } 260 start()261 private int start() { 262 mServerThread = new Thread() { 263 @Override 264 public void run() { 265 while (!mStopServer) { 266 try { 267 Socket sock = mServer.accept(); 268 if (!mStopServer) { 269 startConnectionThread(sock); 270 } else { 271 sock.close(); 272 } 273 } catch (IOException e) { 274 if (!mStopServer) { 275 Log.e("Failed to accept connection.", e); 276 } 277 } catch (JSONException e) { 278 if (!mStopServer) { 279 Log.e("Failed to parse request.", e); 280 } 281 } 282 } 283 } 284 }; 285 mServerThread.start(); 286 Log.v("Bound to " + mServer.getInetAddress()); 287 return mServer.getLocalPort(); 288 } 289 startConnectionThread(final Socket sock)290 private void startConnectionThread(final Socket sock) throws IOException, JSONException { 291 BufferedReader reader = 292 new BufferedReader(new InputStreamReader(sock.getInputStream()), 8192); 293 PrintWriter writer = new PrintWriter(sock.getOutputStream(), true); 294 String data; 295 if((data = reader.readLine()) != null) { 296 Log.v("Received: " + data); 297 JSONObject request = new JSONObject(data); 298 if(request.has("cmd") && request.has("uid")) { 299 String cmd = request.getString("cmd"); 300 int uid = request.getInt("uid"); 301 JSONObject result = new JSONObject(); 302 if(cmd.equals("initiate")) { 303 Log.d("Initiate a new session"); 304 threadIndex += 1; 305 int mUID = threadIndex; 306 ConnectionThread networkThread = new ConnectionThread(sock,true,mUID,reader,writer); 307 mConnectionThreads.put(mUID, networkThread); 308 networkThread.start(); 309 notifyOnConnect(); 310 result.put("uid", mUID); 311 result.put("status",true); 312 result.put("error", null); 313 }else if(cmd.equals("continue")) { 314 Log.d("Continue an existing session"); 315 Log.d("keys: "+mConnectionThreads.keySet().toString()); 316 if(!mConnectionThreads.containsKey(uid)) { 317 result.put("uid", uid); 318 result.put("status",false); 319 result.put("error", "Session does not exist."); 320 }else{ 321 ConnectionThread networkThread = new ConnectionThread(sock,true,uid,reader,writer); 322 mConnectionThreads.put(uid, networkThread); 323 networkThread.start(); 324 notifyOnConnect(); 325 result.put("uid", uid); 326 result.put("status",true); 327 result.put("error", null); 328 } 329 }else { 330 result.put("uid", uid); 331 result.put("status",false); 332 result.put("error", "Unrecognized command."); 333 } 334 writer.write(result + "\n"); 335 writer.flush(); 336 Log.v("Sent: " + result); 337 }else{ 338 ConnectionThread networkThread = new ConnectionThread(sock,false,0,reader,writer); 339 mConnectionThreads.put(0, networkThread); 340 networkThread.start(); 341 notifyOnConnect(); 342 } 343 } 344 } 345 shutdown()346 public void shutdown() { 347 // Stop listening on the server socket to ensure that 348 // beyond this point there are no incoming requests. 349 mStopServer = true; 350 try { 351 mServer.close(); 352 } catch (IOException e) { 353 Log.e("Failed to close server socket.", e); 354 } 355 // Since the server is not running, the mNetworkThreads set can only 356 // shrink from this point onward. We can just stop all of the running helper 357 // threads. In the worst case, one of the running threads will already have 358 // shut down. Since this is a CopyOnWriteList, we don't have to worry about 359 // concurrency issues while iterating over the set of threads. 360 for (ConnectionThread connectionThread : mConnectionThreads.values()) { 361 connectionThread.close(); 362 } 363 for (SimpleServerObserver observer : mObservers) { 364 removeObserver(observer); 365 } 366 } 367 } 368