From 110a0a1b7a4c349587b93131e45d3b683fb043b2 Mon Sep 17 00:00:00 2001
From: zzz <zzz@mail.i2p>
Date: Mon, 23 Nov 2015 18:19:17 +0000
Subject: [PATCH] Remove singleton SAMv3DatagramServer; hang off of SAMBridge
 SAMv3DatagramSession whitespace fixes @since change to 0.9.24

---
 apps/sam/java/src/net/i2p/sam/ReadLine.java   |  2 +-
 apps/sam/java/src/net/i2p/sam/SAMBridge.java  | 41 +++++++++-
 .../src/net/i2p/sam/SAMMessageSession.java    |  2 +-
 .../src/net/i2p/sam/SAMv3DatagramServer.java  | 81 ++++++-------------
 .../src/net/i2p/sam/SAMv3DatagramSession.java | 51 ++++++------
 .../java/src/net/i2p/sam/SAMv3Handler.java    | 30 +++----
 .../java/src/net/i2p/sam/SAMv3RawSession.java |  6 +-
 .../net/i2p/sam/SSLServerSocketChannel.java   |  2 +-
 .../src/net/i2p/sam/SSLSocketChannel.java     |  2 +-
 apps/sam/java/src/net/i2p/sam/SSLUtil.java    |  2 +-
 10 files changed, 105 insertions(+), 114 deletions(-)

diff --git a/apps/sam/java/src/net/i2p/sam/ReadLine.java b/apps/sam/java/src/net/i2p/sam/ReadLine.java
index 889fc6c315..94cf708df3 100644
--- a/apps/sam/java/src/net/i2p/sam/ReadLine.java
+++ b/apps/sam/java/src/net/i2p/sam/ReadLine.java
@@ -11,7 +11,7 @@ import java.net.SocketTimeoutException;
 /**
  * Modified from I2PTunnelHTTPServer
  *
- * @since 0.9.22
+ * @since 0.9.24
  */
 class ReadLine {
 
diff --git a/apps/sam/java/src/net/i2p/sam/SAMBridge.java b/apps/sam/java/src/net/i2p/sam/SAMBridge.java
index 4b9a3293ca..905ae808d0 100644
--- a/apps/sam/java/src/net/i2p/sam/SAMBridge.java
+++ b/apps/sam/java/src/net/i2p/sam/SAMBridge.java
@@ -60,6 +60,8 @@ public class SAMBridge implements Runnable, ClientApp {
     private final boolean _useSSL;
     private final File _configFile;
     private volatile Thread _runner;
+    private final Object _v3DGServerLock = new Object();
+    private SAMv3DatagramServer _v3DGServer;
 
     /** 
      * filename in which the name to private key mapping should 
@@ -95,7 +97,8 @@ public class SAMBridge implements Runnable, ClientApp {
     public static final String PROP_DATAGRAM_HOST = "sam.udp.host";
     public static final String PROP_DATAGRAM_PORT = "sam.udp.port";
     protected static final String DEFAULT_DATAGRAM_HOST = "127.0.0.1";
-    protected static final String DEFAULT_DATAGRAM_PORT = "7655";
+    protected static final int DEFAULT_DATAGRAM_PORT_INT = 7655;
+    protected static final String DEFAULT_DATAGRAM_PORT = Integer.toString(DEFAULT_DATAGRAM_PORT_INT);
 
 
     /**
@@ -354,6 +357,40 @@ public class SAMBridge implements Runnable, ClientApp {
         }
     }
 
+    /**
+     * Was a static singleton, now a singleton for this bridge.
+     * Instantiate and start server if it doesn't exist.
+     * We only listen on one host and port, as specified in the
+     * sam.udp.host and sam.udp.port properties.
+     * TODO we could have multiple servers on different hosts/ports in the future.
+     *
+     * @param props non-null instantiate and start server if it doesn't exist
+     * @param return non-null
+     * @throws IOException if can't bind to host/port, or if different than existing
+     * @since 0.9.24
+     */
+    SAMv3DatagramServer getV3DatagramServer(Properties props) throws IOException {
+        String host = props.getProperty(PROP_DATAGRAM_HOST, DEFAULT_DATAGRAM_HOST);
+        int port;
+        String portStr = props.getProperty(PROP_DATAGRAM_PORT, DEFAULT_DATAGRAM_PORT);
+        try {
+            port = Integer.parseInt(portStr);
+        } catch (NumberFormatException e) {
+            port = DEFAULT_DATAGRAM_PORT_INT;
+        }
+        synchronized (_v3DGServerLock) {
+            if (_v3DGServer == null) {
+                _v3DGServer = new SAMv3DatagramServer(this, host, port, props);
+                _v3DGServer.start();
+            } else {
+                if (_v3DGServer.getPort() != port || !_v3DGServer.getHost().equals(host))
+                    throw new IOException("Already have V3 DatagramServer with host=" + host + " port=" + port);
+            }
+            return _v3DGServer;
+        }
+    }
+
+
     ////// begin ClientApp interface, use only if using correct construtor
 
     /**
@@ -750,7 +787,7 @@ public class SAMBridge implements Runnable, ClientApp {
         }
     }
 
-    /** @since 0.9.22 */
+    /** @since 0.9.24 */
     public void saveConfig() throws IOException {
         DataHelper.storeProps(i2cpProps, _configFile);
     }
diff --git a/apps/sam/java/src/net/i2p/sam/SAMMessageSession.java b/apps/sam/java/src/net/i2p/sam/SAMMessageSession.java
index 8613ed2d06..a2ab4a712d 100644
--- a/apps/sam/java/src/net/i2p/sam/SAMMessageSession.java
+++ b/apps/sam/java/src/net/i2p/sam/SAMMessageSession.java
@@ -249,7 +249,7 @@ abstract class SAMMessageSession {
                              I2PSession.PORT_UNSPECIFIED, I2PSession.PORT_UNSPECIFIED);
         }
 
-        /** @since 0.9.22 */
+        /** @since 0.9.24 */
         public void messageAvailable(I2PSession session, int msgId, long size,
                                      int proto, int fromPort, int toPort) {
 
diff --git a/apps/sam/java/src/net/i2p/sam/SAMv3DatagramServer.java b/apps/sam/java/src/net/i2p/sam/SAMv3DatagramServer.java
index 94268824c5..b410fb21b0 100644
--- a/apps/sam/java/src/net/i2p/sam/SAMv3DatagramServer.java
+++ b/apps/sam/java/src/net/i2p/sam/SAMv3DatagramServer.java
@@ -26,78 +26,42 @@ import net.i2p.util.I2PAppThread;
 import net.i2p.util.Log;
 
 /**
- *  This is a singleton listening on 127.0.0.1:7655 or as specified by
+ *  This is the thread listening on 127.0.0.1:7655 or as specified by
  *  sam.udp.host and sam.udp.port properties.
  *  This is used for both repliable and raw datagrams.
  *
- *  @since 0.9.22 moved from SAMv3Handler
+ *  @since 0.9.24 moved from SAMv3Handler
  */
 class SAMv3DatagramServer implements Handler {
 	
-	private static SAMv3DatagramServer _instance;
-	private static DatagramChannel server;
+	private final DatagramChannel _server;
 	private final Thread _listener;
 	private final SAMBridge _parent;
-	
-	/**
-	 *  Returns the singleton.
-	 *  If this is the first call, will be instantiated and will listen
-	 *  on the default host:port 127.0.0.1:7655.
-	 *  Don't make this the first call.
-	 */
-	public static SAMv3DatagramServer getInstance() throws IOException {
-		return getInstance(null, new Properties());
-	}
-		
-	/**
-	 *  Returns the singleton.
-	 *  If this is the first call, will be instantiated and will listen
-	 *  on the specified host:port, default 127.0.0.1:7655.
-	 *  Properties are sam.udp.host and sam.udp.port.
-	 *
-	 *  @param props ignored unless this is the first call
-	 */
-	public static SAMv3DatagramServer getInstance(SAMBridge parent, Properties props) throws IOException {
-		synchronized(SAMv3DatagramServer.class) {
-			if (_instance==null) {
-				_instance = new SAMv3DatagramServer(parent, props);
-				_instance.start();
-			}
-		}
-		return _instance;
-	}
+	private final String _host;
+	private final int _port;
 	
 	/**
 	 *  Does not start listener.
 	 *  Caller must call start().
 	 *
 	 *  @param parent may be null
+	 *  @param props ignored for now
 	 */
-	private SAMv3DatagramServer(SAMBridge parent, Properties props) throws IOException {
+	public SAMv3DatagramServer(SAMBridge parent, String host, int port, Properties props) throws IOException {
 		_parent = parent;
-		synchronized(SAMv3DatagramServer.class) {
-			if (server==null)
-				server = DatagramChannel.open();
-		}
+		_server = DatagramChannel.open();
 		
-		String host = props.getProperty(SAMBridge.PROP_DATAGRAM_HOST, SAMBridge.DEFAULT_DATAGRAM_HOST);
-		String portStr = props.getProperty(SAMBridge.PROP_DATAGRAM_PORT, SAMBridge.DEFAULT_DATAGRAM_PORT);
-		int port ;
-		try {
-			port = Integer.parseInt(portStr);
-		} catch (NumberFormatException e) {
-			port = Integer.parseInt(SAMBridge.DEFAULT_DATAGRAM_PORT);
-		}
-		
-		server.socket().bind(new InetSocketAddress(host, port));
-		_listener = new I2PAppThread(new Listener(server), "SAM DatagramListener " + port);
+		_server.socket().bind(new InetSocketAddress(host, port));
+		_listener = new I2PAppThread(new Listener(_server), "SAM DatagramListener " + port);
+		_host = host;
+		_port = port;
 	}
 	
 	/**
 	 *  Only call once.
 	 *  @since 0.9.22
 	 */
-	private synchronized void start() {
+	public synchronized void start() {
 		_listener.start();
 		if (_parent != null)
 			_parent.register(this);
@@ -108,23 +72,24 @@ class SAMv3DatagramServer implements Handler {
 	 *  @since 0.9.22
 	 */
 	public synchronized void stopHandling() {
-		synchronized(SAMv3DatagramServer.class) {
-			if (server != null) {
-				try {
-					server.close();
-				} catch (IOException ioe) {}
-				server = null;
-			}
-		}
+		try {
+			_server.close();
+		} catch (IOException ioe) {}
 		_listener.interrupt();
 		if (_parent != null)
 			_parent.unregister(this);
 	}
 	
 	public void send(SocketAddress addr, ByteBuffer msg) throws IOException {
-		server.send(msg, addr);
+		_server.send(msg, addr);
 	}
 
+	/** @since 0.9.24 */
+	public String getHost() { return _host; }
+
+	/** @since 0.9.24 */
+	public int getPort() { return _port; }
+
 	private static class Listener implements Runnable {
 		
 		private final DatagramChannel server;
diff --git a/apps/sam/java/src/net/i2p/sam/SAMv3DatagramSession.java b/apps/sam/java/src/net/i2p/sam/SAMv3DatagramSession.java
index e59ba367ca..ee27825592 100644
--- a/apps/sam/java/src/net/i2p/sam/SAMv3DatagramSession.java
+++ b/apps/sam/java/src/net/i2p/sam/SAMv3DatagramSession.java
@@ -30,45 +30,44 @@ class SAMv3DatagramSession extends SAMDatagramSession implements SAMv3Handler.Se
 	/**
 	 *   build a DatagramSession according to informations registered
 	 *   with the given nickname
+	 *
 	 * @param nick nickname of the session
 	 * @throws IOException
 	 * @throws DataFormatException
 	 * @throws I2PSessionException
 	 */
-	public SAMv3DatagramSession(String nick) 
-	throws IOException, DataFormatException, I2PSessionException, SAMException {
-		
+	public SAMv3DatagramSession(String nick, SAMv3DatagramServer dgServer) 
+			throws IOException, DataFormatException, I2PSessionException, SAMException {
 		super(SAMv3Handler.sSessionsHash.get(nick).getDest(),
 				SAMv3Handler.sSessionsHash.get(nick).getProps(),
 				null  // to be replaced by this
 				);
-		this.nick = nick ;
-		this.recv = this ;  // replacement
-		this.server = SAMv3DatagramServer.getInstance() ;
+		this.nick = nick;
+		this.recv = this;  // replacement
+		this.server = dgServer;
 
 		SAMv3Handler.SessionRecord rec = SAMv3Handler.sSessionsHash.get(nick);
-        if ( rec==null ) throw new SAMException("Record disappeared for nickname : \""+nick+"\"") ;
+		if (rec == null)
+			throw new SAMException("Record disappeared for nickname : \""+nick+"\"");
 
-        this.handler = rec.getHandler();
+		this.handler = rec.getHandler();
 		
-        Properties props = rec.getProps();
-    	String portStr = props.getProperty("PORT") ;
-    	if ( portStr==null ) {
-    		_log.debug("receiver port not specified. Current socket will be used.");
-    		this.clientAddress = null;
-    	}
-    	else {
-    		int port = Integer.parseInt(portStr);
-    	
-    		String host = props.getProperty("HOST");
-    		if ( host==null ) {    		
-    			host = rec.getHandler().getClientIP();
-    			_log.debug("no host specified. Taken from the client socket : " + host+':'+port);
-    		}
-
-    	
-    		this.clientAddress = new InetSocketAddress(host,port);
-    	}
+		Properties props = rec.getProps();
+		String portStr = props.getProperty("PORT");
+		if (portStr == null) {
+			if (_log.shouldDebug())
+				_log.debug("receiver port not specified. Current socket will be used.");
+			this.clientAddress = null;
+		} else {
+			int port = Integer.parseInt(portStr);
+			String host = props.getProperty("HOST");
+			if (host == null) {    		
+				host = rec.getHandler().getClientIP();
+				if (_log.shouldDebug())
+					_log.debug("no host specified. Taken from the client socket : " + host+':'+port);
+			}
+			this.clientAddress = new InetSocketAddress(host, port);
+		}
 	}
 
 	public void receiveDatagramBytes(Destination sender, byte[] data, int proto,
diff --git a/apps/sam/java/src/net/i2p/sam/SAMv3Handler.java b/apps/sam/java/src/net/i2p/sam/SAMv3Handler.java
index 7b073d2e29..246cd5fb98 100644
--- a/apps/sam/java/src/net/i2p/sam/SAMv3Handler.java
+++ b/apps/sam/java/src/net/i2p/sam/SAMv3Handler.java
@@ -250,7 +250,7 @@ class SAMv3Handler extends SAMv1Handler
 	/**
 	 *  For SAMv3DatagramServer
 	 *  @return may be null
-	 *  @since 0.9.22
+	 *  @since 0.9.24
 	 */
 	Session getSession() {
 		return session;
@@ -592,13 +592,13 @@ class SAMv3Handler extends SAMv1Handler
 				// Create the session
 
 				if (style.equals("RAW")) {
-					SAMv3DatagramServer.getInstance(bridge, i2cpProps);
-					SAMv3RawSession v3 = newSAMRawSession(nick);
+					SAMv3DatagramServer dgs = bridge.getV3DatagramServer(props);
+					SAMv3RawSession v3 = new SAMv3RawSession(nick, dgs);
                                         rawSession = v3;
 					this.session = v3;
 				} else if (style.equals("DATAGRAM")) {
-					SAMv3DatagramServer.getInstance(bridge, i2cpProps);
-					SAMv3DatagramSession v3 = newSAMDatagramSession(nick);
+					SAMv3DatagramServer dgs = bridge.getV3DatagramServer(props);
+					SAMv3DatagramSession v3 = new SAMv3DatagramSession(nick, dgs);
 					datagramSession = v3;
 					this.session = v3;
 				} else if (style.equals("STREAM")) {
@@ -652,18 +652,6 @@ class SAMv3Handler extends SAMv1Handler
 		return new SAMv3StreamSession( login ) ;
 	}
 
-	private static SAMv3RawSession newSAMRawSession(String login )
-			throws IOException, DataFormatException, SAMException, I2PSessionException
-	{
-		return new SAMv3RawSession( login ) ;
-	}
-
-	private static SAMv3DatagramSession newSAMDatagramSession(String login )
-			throws IOException, DataFormatException, SAMException, I2PSessionException
-	{
-		return new SAMv3DatagramSession( login ) ;
-	}
-
 	/* Parse and execute a STREAM message */
 	@Override
 	protected boolean execStreamMessage ( String opcode, Properties props )
@@ -863,7 +851,7 @@ class SAMv3Handler extends SAMv1Handler
 	    }
 	}
 	
-	/** @since 0.9.22 */
+	/** @since 0.9.24 */
 	public static void notifyStreamIncomingConnection(SocketChannel client, Destination d,
 	                                                  int fromPort, int toPort) throws IOException {
 	    if (!writeString(d.toBase64() + " FROM_PORT=" + fromPort + " TO_PORT=" + toPort + '\n', client)) {
@@ -871,7 +859,7 @@ class SAMv3Handler extends SAMv1Handler
 	    }
 	}
 
-	/** @since 0.9.22 */
+	/** @since 0.9.24 */
 	private boolean execAuthMessage(String opcode, Properties props) {
 		if (opcode.equals("ENABLE")) {
 			i2cpProps.setProperty(SAMBridge.PROP_AUTH, "true");
@@ -910,7 +898,7 @@ class SAMv3Handler extends SAMv1Handler
 	/**
 	 * Handle a PING.
 	 * Send a PONG.
-	 * @since 0.9.22
+	 * @since 0.9.24
 	 */
 	private void execPingMessage(StringTokenizer tok) {
 		StringBuilder buf = new StringBuilder();
@@ -924,7 +912,7 @@ class SAMv3Handler extends SAMv1Handler
 
 	/**
 	 * Handle a PONG.
-	 * @since 0.9.22
+	 * @since 0.9.24
 	 */
 	private void execPongMessage(StringTokenizer tok) {
 		String s;
diff --git a/apps/sam/java/src/net/i2p/sam/SAMv3RawSession.java b/apps/sam/java/src/net/i2p/sam/SAMv3RawSession.java
index aebcb482b5..658aaae0a4 100644
--- a/apps/sam/java/src/net/i2p/sam/SAMv3RawSession.java
+++ b/apps/sam/java/src/net/i2p/sam/SAMv3RawSession.java
@@ -36,14 +36,16 @@ class SAMv3RawSession extends SAMRawSession  implements SAMv3Handler.Session, SA
 	 * @throws DataFormatException
 	 * @throws I2PSessionException
 	 */
-	public SAMv3RawSession(String nick) throws IOException, DataFormatException, I2PSessionException {
+	public SAMv3RawSession(String nick, SAMv3DatagramServer dgServer) 
+			throws IOException, DataFormatException, I2PSessionException {
 		super(SAMv3Handler.sSessionsHash.get(nick).getDest(),
 		      SAMv3Handler.sSessionsHash.get(nick).getProps(),
 		      SAMv3Handler.sSessionsHash.get(nick).getHandler()  // to be replaced by this
 		);
 		this.nick = nick ;
 		this.recv = this ;  // replacement
-		this.server = SAMv3DatagramServer.getInstance() ;
+		this.server = dgServer;
+
 		SAMv3Handler.SessionRecord rec = SAMv3Handler.sSessionsHash.get(nick);
 		if (rec == null)
 			throw new InterruptedIOException() ;
diff --git a/apps/sam/java/src/net/i2p/sam/SSLServerSocketChannel.java b/apps/sam/java/src/net/i2p/sam/SSLServerSocketChannel.java
index f92f43e89e..546955ceef 100644
--- a/apps/sam/java/src/net/i2p/sam/SSLServerSocketChannel.java
+++ b/apps/sam/java/src/net/i2p/sam/SSLServerSocketChannel.java
@@ -19,7 +19,7 @@ import javax.net.ssl.SSLSocket;
  * Simple wrapper for a SSLServerSocket.
  * Cannot be used for asynch ops.
  *
- * @since 0.9.22
+ * @since 0.9.24
  */
 class SSLServerSocketChannel extends ServerSocketChannel {
 
diff --git a/apps/sam/java/src/net/i2p/sam/SSLSocketChannel.java b/apps/sam/java/src/net/i2p/sam/SSLSocketChannel.java
index 9530b396d3..7b956a16e3 100644
--- a/apps/sam/java/src/net/i2p/sam/SSLSocketChannel.java
+++ b/apps/sam/java/src/net/i2p/sam/SSLSocketChannel.java
@@ -18,7 +18,7 @@ import javax.net.ssl.SSLSocket;
  * Simple wrapper for a SSLSocket.
  * Cannot be used for asynch ops.
  *
- * @since 0.9.22
+ * @since 0.9.24
  */
 class SSLSocketChannel extends SocketChannel {
 
diff --git a/apps/sam/java/src/net/i2p/sam/SSLUtil.java b/apps/sam/java/src/net/i2p/sam/SSLUtil.java
index bc0331417e..bf4ebf1e42 100644
--- a/apps/sam/java/src/net/i2p/sam/SSLUtil.java
+++ b/apps/sam/java/src/net/i2p/sam/SSLUtil.java
@@ -20,7 +20,7 @@ import net.i2p.util.SecureDirectory;
 /**
  * Utilities for SAM SSL server sockets.
  *
- * @since 0.9.22 adopted from net.i2p.i2ptunnel.SSLClientUtil
+ * @since 0.9.24 adopted from net.i2p.i2ptunnel.SSLClientUtil
  */
 class SSLUtil {
 
-- 
GitLab